From 5041554e7bf313340dcb465a3149df4fba0e4119 Mon Sep 17 00:00:00 2001
From: peng <peng.com>
Date: 星期二, 30 九月 2025 14:29:10 +0800
Subject: [PATCH] 店铺扫码抽奖功能

---
 framework/src/main/java/cn/lili/modules/lmk/mapper/ScanPrizeMapper.java                    |   34 +
 framework/src/main/java/cn/lili/modules/lmk/domain/form/PrizeClaimRecordForm.java          |   66 ++
 framework/src/main/resources/mapper/lmk/StorePrizeProofMapper.xml                          |   41 +
 framework/src/main/java/cn/lili/modules/lmk/domain/form/ScanPrizeForm.java                 |   69 ++
 framework/src/main/java/cn/lili/modules/lmk/enums/general/MaterialStatusEnum.java          |   13 
 framework/src/main/java/cn/lili/modules/lmk/enums/general/GenerateStorePrizeStausEnum.java |   13 
 framework/src/main/java/cn/lili/modules/lmk/mapper/StorePrizeProofMapper.java              |   34 +
 framework/src/main/java/cn/lili/modules/lmk/service/PrizeClaimRecordService.java           |   65 ++
 framework/src/main/java/cn/lili/modules/lmk/service/ScanPrizeService.java                  |   69 ++
 framework/src/main/java/cn/lili/modules/lmk/domain/form/StorePrizeProofForm.java           |   42 +
 framework/src/main/java/cn/lili/modules/lmk/enums/general/StoreScanPrizeStausEnum.java     |   13 
 framework/src/main/java/cn/lili/modules/lmk/domain/entity/ScanPrize.java                   |   60 ++
 framework/src/main/java/cn/lili/modules/lmk/service/StorePrizeProofService.java            |   65 ++
 framework/src/main/java/cn/lili/modules/lmk/domain/vo/ScanPrizeVO.java                     |   63 ++
 framework/src/main/java/cn/lili/modules/lmk/domain/query/PrizeClaimRecordQuery.java        |   22 
 manager-api/src/main/java/cn/lili/controller/lmk/ScanPrizeController.java                  |   98 +++
 framework/src/main/resources/mapper/lmk/ScanPrizeMapper.xml                                |   59 ++
 framework/src/main/java/cn/lili/modules/lmk/domain/entity/PrizeClaimRecord.java            |   66 ++
 framework/src/main/java/cn/lili/modules/lmk/mapper/PrizeClaimRecordMapper.java             |   34 +
 framework/src/main/resources/mapper/lmk/PrizeClaimRecordMapper.xml                         |   59 ++
 framework/src/main/java/cn/lili/modules/lmk/domain/query/StorePrizeProofQuery.java         |   22 
 framework/src/main/java/cn/lili/modules/lmk/service/impl/ScanPrizeServiceImpl.java         |  218 ++++++++
 framework/src/main/java/cn/lili/modules/lmk/service/impl/PrizeClaimRecordServiceImpl.java  |  119 ++++
 framework/src/main/java/cn/lili/modules/lmk/domain/vo/StorePrizeProofVO.java               |   39 +
 framework/src/main/java/cn/lili/modules/lmk/service/impl/StorePrizeProofServiceImpl.java   |  119 ++++
 framework/src/main/java/cn/lili/modules/lmk/domain/vo/PrizeClaimRecordVO.java              |   63 ++
 framework/src/main/java/cn/lili/modules/lmk/domain/query/ScanPrizeQuery.java               |   22 
 framework/src/main/java/cn/lili/modules/lmk/domain/entity/StorePrizeProof.java             |   30 +
 28 files changed, 1,617 insertions(+), 0 deletions(-)

diff --git a/framework/src/main/java/cn/lili/modules/lmk/domain/entity/PrizeClaimRecord.java b/framework/src/main/java/cn/lili/modules/lmk/domain/entity/PrizeClaimRecord.java
new file mode 100644
index 0000000..83b2e1e
--- /dev/null
+++ b/framework/src/main/java/cn/lili/modules/lmk/domain/entity/PrizeClaimRecord.java
@@ -0,0 +1,66 @@
+package cn.lili.modules.lmk.domain.entity;
+
+import cn.lili.modules.lmk.enums.general.MaterialStatusEnum;
+import cn.lili.mybatis.BaseEntity;
+import com.baomidou.mybatisplus.annotation.TableField;
+import com.baomidou.mybatisplus.annotation.TableName;
+import java.io.Serializable;
+import lombok.Data;
+
+/**
+ * 搴楅摵鎶藉娲诲姩棰嗗彇琛�
+ *
+ * @author peng
+ * @since 2025-09-30
+ */
+@Data
+@TableName("lmk_prize_claim_record")
+public class PrizeClaimRecord extends BaseEntity {
+
+    private static final long serialVersionUID = 1L;
+
+    @TableField("no")
+    /** 缂栧彿 */
+    private String no;
+
+    @TableField("store_id")
+    /** 搴楅摵id */
+    private String storeId;
+
+    @TableField("store_prize_id")
+    /** 搴楅摵鎶藉娲诲姩id */
+    private String storePrizeId;
+
+    @TableField("store_name")
+    /** 搴楅摵鍚嶇О */
+    private String storeName;
+
+    @TableField("prize_activity_id")
+    /** 鎶藉娲诲姩id */
+    private String prizeActivityId;
+
+    @TableField("prize_activity_name")
+    /** 鎶藉娲诲姩鍚嶇О */
+    private String prizeActivityName;
+
+    @TableField("user_id")
+    /** 鐢ㄦ埛id */
+    private String userId;
+
+    @TableField("nick_name")
+    /** 鐢ㄦ埛鏄电О */
+    private String nickName;
+
+    @TableField("claim_status")
+    /** 棰嗗彇鐘舵�� */
+    private String claimStatus;
+
+    @TableField("material")
+    /** 鏉愭枡鐘舵��
+     *
+     * @see MaterialStatusEnum
+     * */
+    private String material;
+
+
+}
diff --git a/framework/src/main/java/cn/lili/modules/lmk/domain/entity/ScanPrize.java b/framework/src/main/java/cn/lili/modules/lmk/domain/entity/ScanPrize.java
new file mode 100644
index 0000000..2b88656
--- /dev/null
+++ b/framework/src/main/java/cn/lili/modules/lmk/domain/entity/ScanPrize.java
@@ -0,0 +1,60 @@
+package cn.lili.modules.lmk.domain.entity;
+
+import cn.lili.modules.lmk.enums.general.GenerateStorePrizeStausEnum;
+import cn.lili.modules.lmk.enums.general.StoreScanPrizeStausEnum;
+import cn.lili.mybatis.BaseEntity;
+import com.baomidou.mybatisplus.annotation.TableField;
+import com.baomidou.mybatisplus.annotation.TableName;
+import java.io.Serializable;
+import lombok.Data;
+
+/**
+ * 搴楅摵娲诲姩鍏宠仈
+ *
+ * @author peng
+ * @since 2025-09-30
+ */
+@Data
+@TableName("lmk_scan_prize")
+public class ScanPrize extends BaseEntity {
+
+    private static final long serialVersionUID = 1L;
+
+    @TableField("store_id")
+    /** 搴楅摵id */
+    private String storeId;
+
+    @TableField("store_name")
+    /** 搴楅摵鍚嶇О */
+    private String storeName;
+
+    @TableField("prize_activity_id")
+    /** 鎶藉娲诲姩id */
+    private String prizeActivityId;
+
+    @TableField("prize_activity_nmae")
+    /** 鎶藉娲诲姩鍚嶇О鍚嶇О */
+    private String prizeActivityNmae;
+
+    @TableField("generate_status")
+    /** 鐢熸垚鐘舵��
+     * @see GenerateStorePrizeStausEnum
+     * */
+    private String generateStatus;
+
+    @TableField("generate_num")
+    /** 鐢熸垚鏁伴噺 */
+    private Integer generateNum;
+
+    @TableField("claim_num")
+    /** 棰嗗彇鏁伴噺 */
+    private Integer claimNum;
+
+    @TableField("status")
+    /** 鏄惁鍚敤
+     * @see StoreScanPrizeStausEnum
+     * */
+    private String status;
+
+
+}
diff --git a/framework/src/main/java/cn/lili/modules/lmk/domain/entity/StorePrizeProof.java b/framework/src/main/java/cn/lili/modules/lmk/domain/entity/StorePrizeProof.java
new file mode 100644
index 0000000..891752b
--- /dev/null
+++ b/framework/src/main/java/cn/lili/modules/lmk/domain/entity/StorePrizeProof.java
@@ -0,0 +1,30 @@
+package cn.lili.modules.lmk.domain.entity;
+
+import cn.lili.mybatis.BaseEntity;
+import com.baomidou.mybatisplus.annotation.TableField;
+import com.baomidou.mybatisplus.annotation.TableName;
+import java.io.Serializable;
+import lombok.Data;
+
+/**
+ * 鎶藉娲诲姩琛ュ綍璇佹槑
+ *
+ * @author peng
+ * @since 2025-09-30
+ */
+@Data
+@TableName("lmk_store_prize_proof")
+public class StorePrizeProof extends BaseEntity {
+
+    private static final long serialVersionUID = 1L;
+
+    @TableField("store_prize_claim_id")
+    /** 鎶藉娲诲姩棰嗗彇琛╥d */
+    private String storePrizeClaimId;
+
+    @TableField("url_path")
+    /** 闄勪欢鍦板潃 */
+    private String urlPath;
+
+
+}
diff --git a/framework/src/main/java/cn/lili/modules/lmk/domain/form/PrizeClaimRecordForm.java b/framework/src/main/java/cn/lili/modules/lmk/domain/form/PrizeClaimRecordForm.java
new file mode 100644
index 0000000..720c822
--- /dev/null
+++ b/framework/src/main/java/cn/lili/modules/lmk/domain/form/PrizeClaimRecordForm.java
@@ -0,0 +1,66 @@
+package cn.lili.modules.lmk.domain.form;
+
+import cn.lili.group.Update;
+import cn.lili.group.Add;
+import cn.lili.base.AbsForm;
+import cn.lili.modules.lmk.domain.entity.PrizeClaimRecord;
+import org.springframework.beans.BeanUtils;
+import javax.validation.constraints.NotBlank;
+import javax.validation.constraints.NotNull;
+import org.springframework.lang.NonNull;
+import io.swagger.annotations.ApiModel;
+import io.swagger.annotations.ApiModelProperty;
+import lombok.Data;
+import java.util.Date;
+
+/**
+ * 搴楅摵鎶藉娲诲姩棰嗗彇琛ㄨ〃鍗�
+ *
+ * @author peng
+ * @since 2025-09-30
+ */
+@Data
+@ApiModel(value = "PrizeClaimRecord琛ㄥ崟", description = "搴楅摵鎶藉娲诲姩棰嗗彇琛ㄨ〃鍗�")
+public class PrizeClaimRecordForm extends AbsForm {
+
+    @NotBlank(message = "缂栧彿涓嶈兘涓虹┖", groups = {Add.class, Update.class})
+    @ApiModelProperty("缂栧彿")
+    private String no;
+
+    @NotNull(message = "搴楅摵id涓嶈兘涓虹┖", groups = {Add.class, Update.class})
+    @ApiModelProperty("搴楅摵id")
+    private Long storeId;
+
+    @NotBlank(message = "搴楅摵鍚嶇О涓嶈兘涓虹┖", groups = {Add.class, Update.class})
+    @ApiModelProperty("搴楅摵鍚嶇О")
+    private String storeName;
+
+    @NotNull(message = "鎶藉娲诲姩id涓嶈兘涓虹┖", groups = {Add.class, Update.class})
+    @ApiModelProperty("鎶藉娲诲姩id")
+    private Long prizeActivityId;
+
+    @NotBlank(message = "鎶藉娲诲姩鍚嶇О涓嶈兘涓虹┖", groups = {Add.class, Update.class})
+    @ApiModelProperty("鎶藉娲诲姩鍚嶇О")
+    private String prizeActivityName;
+
+    @NotNull(message = "鐢ㄦ埛id涓嶈兘涓虹┖", groups = {Add.class, Update.class})
+    @ApiModelProperty("鐢ㄦ埛id")
+    private Long userId;
+
+    @NotBlank(message = "鐢ㄦ埛鏄电О涓嶈兘涓虹┖", groups = {Add.class, Update.class})
+    @ApiModelProperty("鐢ㄦ埛鏄电О")
+    private String nickName;
+
+    @NotBlank(message = "棰嗗彇鐘舵�佷笉鑳戒负绌�", groups = {Add.class, Update.class})
+    @ApiModelProperty("棰嗗彇鐘舵��")
+    private String claimStatus;
+
+    public static PrizeClaimRecord getEntityByForm(@NonNull PrizeClaimRecordForm form, PrizeClaimRecord entity) {
+        if(entity == null) {
+          entity = new PrizeClaimRecord();
+        }
+        BeanUtils.copyProperties(form, entity);
+        return entity;
+    }
+
+}
diff --git a/framework/src/main/java/cn/lili/modules/lmk/domain/form/ScanPrizeForm.java b/framework/src/main/java/cn/lili/modules/lmk/domain/form/ScanPrizeForm.java
new file mode 100644
index 0000000..87e1a47
--- /dev/null
+++ b/framework/src/main/java/cn/lili/modules/lmk/domain/form/ScanPrizeForm.java
@@ -0,0 +1,69 @@
+package cn.lili.modules.lmk.domain.form;
+
+import cn.lili.group.Update;
+import cn.lili.group.Add;
+import cn.lili.base.AbsForm;
+import cn.lili.modules.lmk.domain.entity.ScanPrize;
+import cn.lili.modules.lmk.enums.general.GenerateStorePrizeStausEnum;
+import cn.lili.modules.lmk.enums.general.StoreScanPrizeStausEnum;
+import org.springframework.beans.BeanUtils;
+import javax.validation.constraints.NotBlank;
+import javax.validation.constraints.NotNull;
+import org.springframework.lang.NonNull;
+import io.swagger.annotations.ApiModel;
+import io.swagger.annotations.ApiModelProperty;
+import lombok.Data;
+import java.util.Date;
+
+/**
+ * 搴楅摵娲诲姩鍏宠仈琛ㄥ崟
+ *
+ * @author peng
+ * @since 2025-09-30
+ */
+@Data
+@ApiModel(value = "ScanPrize琛ㄥ崟", description = "搴楅摵娲诲姩鍏宠仈琛ㄥ崟")
+public class ScanPrizeForm extends AbsForm {
+
+    @NotNull(message = "搴楅摵id涓嶈兘涓虹┖", groups = {Add.class, Update.class})
+    @ApiModelProperty("搴楅摵id")
+    private Long storeId;
+
+    @NotNull(message = "搴楅摵鍚嶇О涓嶈兘涓虹┖", groups = {Add.class, Update.class})
+    @ApiModelProperty("搴楅摵鍚嶇О")
+    private Long storeName;
+
+    @NotNull(message = "鎶藉娲诲姩id涓嶈兘涓虹┖", groups = {Add.class, Update.class})
+    @ApiModelProperty("鎶藉娲诲姩id")
+    private Long prizeActivityId;
+
+    @NotBlank(message = "鎶藉娲诲姩鍚嶇О鍚嶇О涓嶈兘涓虹┖", groups = {Add.class, Update.class})
+    @ApiModelProperty("鎶藉娲诲姩鍚嶇О鍚嶇О")
+    private String prizeActivityNmae;
+    /**
+     * @see GenerateStorePrizeStausEnum
+     */
+    @ApiModelProperty("鐢熸垚鐘舵��")
+    private String generateStatus;
+
+    @NotNull(message = "鐢熸垚鏁伴噺涓嶈兘涓虹┖", groups = {Add.class, Update.class})
+    @ApiModelProperty("鐢熸垚鏁伴噺")
+    private Integer generateNum;
+
+    @ApiModelProperty("棰嗗彇鏁伴噺")
+    private Integer claimNum;
+    /**
+     * @see StoreScanPrizeStausEnum
+     */
+    @ApiModelProperty("鏄惁鍚敤")
+    private String status;
+
+    public static ScanPrize getEntityByForm(@NonNull ScanPrizeForm form, ScanPrize entity) {
+        if(entity == null) {
+          entity = new ScanPrize();
+        }
+        BeanUtils.copyProperties(form, entity);
+        return entity;
+    }
+
+}
diff --git a/framework/src/main/java/cn/lili/modules/lmk/domain/form/StorePrizeProofForm.java b/framework/src/main/java/cn/lili/modules/lmk/domain/form/StorePrizeProofForm.java
new file mode 100644
index 0000000..95aa13e
--- /dev/null
+++ b/framework/src/main/java/cn/lili/modules/lmk/domain/form/StorePrizeProofForm.java
@@ -0,0 +1,42 @@
+package cn.lili.modules.lmk.domain.form;
+
+import cn.lili.group.Update;
+import cn.lili.group.Add;
+import cn.lili.base.AbsForm;
+import cn.lili.modules.lmk.domain.entity.StorePrizeProof;
+import org.springframework.beans.BeanUtils;
+import javax.validation.constraints.NotBlank;
+import javax.validation.constraints.NotNull;
+import org.springframework.lang.NonNull;
+import io.swagger.annotations.ApiModel;
+import io.swagger.annotations.ApiModelProperty;
+import lombok.Data;
+import java.util.Date;
+
+/**
+ * 鎶藉娲诲姩琛ュ綍璇佹槑琛ㄥ崟
+ *
+ * @author peng
+ * @since 2025-09-30
+ */
+@Data
+@ApiModel(value = "StorePrizeProof琛ㄥ崟", description = "鎶藉娲诲姩琛ュ綍璇佹槑琛ㄥ崟")
+public class StorePrizeProofForm extends AbsForm {
+
+    @NotBlank(message = "鎶藉娲诲姩棰嗗彇琛╥d涓嶈兘涓虹┖", groups = {Add.class, Update.class})
+    @ApiModelProperty("鎶藉娲诲姩棰嗗彇琛╥d")
+    private String storePrizeClaimId;
+
+    @NotBlank(message = "闄勪欢鍦板潃涓嶈兘涓虹┖", groups = {Add.class, Update.class})
+    @ApiModelProperty("闄勪欢鍦板潃")
+    private String urlPath;
+
+    public static StorePrizeProof getEntityByForm(@NonNull StorePrizeProofForm form, StorePrizeProof entity) {
+        if(entity == null) {
+          entity = new StorePrizeProof();
+        }
+        BeanUtils.copyProperties(form, entity);
+        return entity;
+    }
+
+}
diff --git a/framework/src/main/java/cn/lili/modules/lmk/domain/query/PrizeClaimRecordQuery.java b/framework/src/main/java/cn/lili/modules/lmk/domain/query/PrizeClaimRecordQuery.java
new file mode 100644
index 0000000..330b3bc
--- /dev/null
+++ b/framework/src/main/java/cn/lili/modules/lmk/domain/query/PrizeClaimRecordQuery.java
@@ -0,0 +1,22 @@
+package cn.lili.modules.lmk.domain.query;
+
+import cn.lili.base.AbsQuery;
+import java.util.List;
+import org.springframework.lang.NonNull;
+import javax.validation.constraints.NotBlank;
+import javax.validation.constraints.NotNull;
+import io.swagger.annotations.ApiModel;
+import io.swagger.annotations.ApiModelProperty;
+import lombok.Data;
+
+/**
+ * 搴楅摵鎶藉娲诲姩棰嗗彇琛ㄦ煡璇�
+ *
+ * @author peng
+ * @since 2025-09-30
+ */
+@Data
+@ApiModel(value = "PrizeClaimRecord鏌ヨ鍙傛暟", description = "搴楅摵鎶藉娲诲姩棰嗗彇琛ㄦ煡璇㈠弬鏁�")
+public class PrizeClaimRecordQuery extends AbsQuery {
+}
+
diff --git a/framework/src/main/java/cn/lili/modules/lmk/domain/query/ScanPrizeQuery.java b/framework/src/main/java/cn/lili/modules/lmk/domain/query/ScanPrizeQuery.java
new file mode 100644
index 0000000..8daa571
--- /dev/null
+++ b/framework/src/main/java/cn/lili/modules/lmk/domain/query/ScanPrizeQuery.java
@@ -0,0 +1,22 @@
+package cn.lili.modules.lmk.domain.query;
+
+import cn.lili.base.AbsQuery;
+import java.util.List;
+import org.springframework.lang.NonNull;
+import javax.validation.constraints.NotBlank;
+import javax.validation.constraints.NotNull;
+import io.swagger.annotations.ApiModel;
+import io.swagger.annotations.ApiModelProperty;
+import lombok.Data;
+
+/**
+ * 搴楅摵娲诲姩鍏宠仈鏌ヨ
+ *
+ * @author peng
+ * @since 2025-09-30
+ */
+@Data
+@ApiModel(value = "ScanPrize鏌ヨ鍙傛暟", description = "搴楅摵娲诲姩鍏宠仈鏌ヨ鍙傛暟")
+public class ScanPrizeQuery extends AbsQuery {
+}
+
diff --git a/framework/src/main/java/cn/lili/modules/lmk/domain/query/StorePrizeProofQuery.java b/framework/src/main/java/cn/lili/modules/lmk/domain/query/StorePrizeProofQuery.java
new file mode 100644
index 0000000..e4911b6
--- /dev/null
+++ b/framework/src/main/java/cn/lili/modules/lmk/domain/query/StorePrizeProofQuery.java
@@ -0,0 +1,22 @@
+package cn.lili.modules.lmk.domain.query;
+
+import cn.lili.base.AbsQuery;
+import java.util.List;
+import org.springframework.lang.NonNull;
+import javax.validation.constraints.NotBlank;
+import javax.validation.constraints.NotNull;
+import io.swagger.annotations.ApiModel;
+import io.swagger.annotations.ApiModelProperty;
+import lombok.Data;
+
+/**
+ * 鎶藉娲诲姩琛ュ綍璇佹槑鏌ヨ
+ *
+ * @author peng
+ * @since 2025-09-30
+ */
+@Data
+@ApiModel(value = "StorePrizeProof鏌ヨ鍙傛暟", description = "鎶藉娲诲姩琛ュ綍璇佹槑鏌ヨ鍙傛暟")
+public class StorePrizeProofQuery extends AbsQuery {
+}
+
diff --git a/framework/src/main/java/cn/lili/modules/lmk/domain/vo/PrizeClaimRecordVO.java b/framework/src/main/java/cn/lili/modules/lmk/domain/vo/PrizeClaimRecordVO.java
new file mode 100644
index 0000000..0cb71c7
--- /dev/null
+++ b/framework/src/main/java/cn/lili/modules/lmk/domain/vo/PrizeClaimRecordVO.java
@@ -0,0 +1,63 @@
+package cn.lili.modules.lmk.domain.vo;
+
+import cn.lili.base.AbsVo;
+import cn.lili.modules.lmk.domain.entity.PrizeClaimRecord;
+import java.util.List;
+import org.springframework.lang.NonNull;
+import org.springframework.beans.BeanUtils;
+import io.swagger.annotations.ApiModel;
+import io.swagger.annotations.ApiModelProperty;
+import lombok.Data;
+import java.util.Date;
+
+/**
+ * 搴楅摵鎶藉娲诲姩棰嗗彇琛ㄥ睍绀�
+ *
+ * @author peng
+ * @since 2025-09-30
+ */
+@Data
+@ApiModel(value = "搴楅摵鎶藉娲诲姩棰嗗彇琛ㄥ搷搴旀暟鎹�", description = "搴楅摵鎶藉娲诲姩棰嗗彇琛ㄥ搷搴旀暟鎹�")
+public class PrizeClaimRecordVO extends AbsVo {
+
+    /** 缂栧彿 */
+    @ApiModelProperty("缂栧彿")
+    private String no;
+
+    /** 搴楅摵id */
+    @ApiModelProperty("搴楅摵id")
+    private Long storeId;
+
+    /** 搴楅摵鍚嶇О */
+    @ApiModelProperty("搴楅摵鍚嶇О")
+    private String storeName;
+
+    /** 鎶藉娲诲姩id */
+    @ApiModelProperty("鎶藉娲诲姩id")
+    private Long prizeActivityId;
+
+    /** 鎶藉娲诲姩鍚嶇О */
+    @ApiModelProperty("鎶藉娲诲姩鍚嶇О")
+    private String prizeActivityName;
+
+    /** 鐢ㄦ埛id */
+    @ApiModelProperty("鐢ㄦ埛id")
+    private Long userId;
+
+    /** 鐢ㄦ埛鏄电О */
+    @ApiModelProperty("鐢ㄦ埛鏄电О")
+    private String nickName;
+
+    /** 棰嗗彇鐘舵�� */
+    @ApiModelProperty("棰嗗彇鐘舵��")
+    private String claimStatus;
+
+    public static PrizeClaimRecordVO getVoByEntity(@NonNull PrizeClaimRecord entity, PrizeClaimRecordVO vo) {
+        if(vo == null) {
+            vo = new PrizeClaimRecordVO();
+        }
+        BeanUtils.copyProperties(entity, vo);
+        return vo;
+    }
+
+}
diff --git a/framework/src/main/java/cn/lili/modules/lmk/domain/vo/ScanPrizeVO.java b/framework/src/main/java/cn/lili/modules/lmk/domain/vo/ScanPrizeVO.java
new file mode 100644
index 0000000..d9570bf
--- /dev/null
+++ b/framework/src/main/java/cn/lili/modules/lmk/domain/vo/ScanPrizeVO.java
@@ -0,0 +1,63 @@
+package cn.lili.modules.lmk.domain.vo;
+
+import cn.lili.base.AbsVo;
+import cn.lili.modules.lmk.domain.entity.ScanPrize;
+import java.util.List;
+import org.springframework.lang.NonNull;
+import org.springframework.beans.BeanUtils;
+import io.swagger.annotations.ApiModel;
+import io.swagger.annotations.ApiModelProperty;
+import lombok.Data;
+import java.util.Date;
+
+/**
+ * 搴楅摵娲诲姩鍏宠仈灞曠ず
+ *
+ * @author peng
+ * @since 2025-09-30
+ */
+@Data
+@ApiModel(value = "搴楅摵娲诲姩鍏宠仈鍝嶅簲鏁版嵁", description = "搴楅摵娲诲姩鍏宠仈鍝嶅簲鏁版嵁")
+public class ScanPrizeVO extends AbsVo {
+
+    /** 搴楅摵id */
+    @ApiModelProperty("搴楅摵id")
+    private Long storeId;
+
+    /** 搴楅摵鍚嶇О */
+    @ApiModelProperty("搴楅摵鍚嶇О")
+    private Long storeName;
+
+    /** 鎶藉娲诲姩id */
+    @ApiModelProperty("鎶藉娲诲姩id")
+    private Long prizeActivityId;
+
+    /** 鎶藉娲诲姩鍚嶇О鍚嶇О */
+    @ApiModelProperty("鎶藉娲诲姩鍚嶇О鍚嶇О")
+    private String prizeActivityNmae;
+
+    /** 鐢熸垚鐘舵�� */
+    @ApiModelProperty("鐢熸垚鐘舵��")
+    private String generateStatus;
+
+    /** 鐢熸垚鏁伴噺 */
+    @ApiModelProperty("鐢熸垚鏁伴噺")
+    private Integer generateNum;
+
+    /** 棰嗗彇鏁伴噺 */
+    @ApiModelProperty("棰嗗彇鏁伴噺")
+    private Integer claimNum;
+
+    /** 鏄惁鍚敤 */
+    @ApiModelProperty("鏄惁鍚敤")
+    private String status;
+
+    public static ScanPrizeVO getVoByEntity(@NonNull ScanPrize entity, ScanPrizeVO vo) {
+        if(vo == null) {
+            vo = new ScanPrizeVO();
+        }
+        BeanUtils.copyProperties(entity, vo);
+        return vo;
+    }
+
+}
diff --git a/framework/src/main/java/cn/lili/modules/lmk/domain/vo/StorePrizeProofVO.java b/framework/src/main/java/cn/lili/modules/lmk/domain/vo/StorePrizeProofVO.java
new file mode 100644
index 0000000..b319fe7
--- /dev/null
+++ b/framework/src/main/java/cn/lili/modules/lmk/domain/vo/StorePrizeProofVO.java
@@ -0,0 +1,39 @@
+package cn.lili.modules.lmk.domain.vo;
+
+import cn.lili.base.AbsVo;
+import cn.lili.modules.lmk.domain.entity.StorePrizeProof;
+import java.util.List;
+import org.springframework.lang.NonNull;
+import org.springframework.beans.BeanUtils;
+import io.swagger.annotations.ApiModel;
+import io.swagger.annotations.ApiModelProperty;
+import lombok.Data;
+import java.util.Date;
+
+/**
+ * 鎶藉娲诲姩琛ュ綍璇佹槑灞曠ず
+ *
+ * @author peng
+ * @since 2025-09-30
+ */
+@Data
+@ApiModel(value = "鎶藉娲诲姩琛ュ綍璇佹槑鍝嶅簲鏁版嵁", description = "鎶藉娲诲姩琛ュ綍璇佹槑鍝嶅簲鏁版嵁")
+public class StorePrizeProofVO extends AbsVo {
+
+    /** 鎶藉娲诲姩棰嗗彇琛╥d */
+    @ApiModelProperty("鎶藉娲诲姩棰嗗彇琛╥d")
+    private String storePrizeClaimId;
+
+    /** 闄勪欢鍦板潃 */
+    @ApiModelProperty("闄勪欢鍦板潃")
+    private String urlPath;
+
+    public static StorePrizeProofVO getVoByEntity(@NonNull StorePrizeProof entity, StorePrizeProofVO vo) {
+        if(vo == null) {
+            vo = new StorePrizeProofVO();
+        }
+        BeanUtils.copyProperties(entity, vo);
+        return vo;
+    }
+
+}
diff --git a/framework/src/main/java/cn/lili/modules/lmk/enums/general/GenerateStorePrizeStausEnum.java b/framework/src/main/java/cn/lili/modules/lmk/enums/general/GenerateStorePrizeStausEnum.java
new file mode 100644
index 0000000..18d7756
--- /dev/null
+++ b/framework/src/main/java/cn/lili/modules/lmk/enums/general/GenerateStorePrizeStausEnum.java
@@ -0,0 +1,13 @@
+package cn.lili.modules.lmk.enums.general;
+
+import lombok.AllArgsConstructor;
+
+@AllArgsConstructor
+public enum GenerateStorePrizeStausEnum {
+    NOT_GENERATE("娌℃湁鐢熸垚"),
+    GENERATE("鐢熸垚");
+
+    private String des;
+
+
+}
diff --git a/framework/src/main/java/cn/lili/modules/lmk/enums/general/MaterialStatusEnum.java b/framework/src/main/java/cn/lili/modules/lmk/enums/general/MaterialStatusEnum.java
new file mode 100644
index 0000000..be48e56
--- /dev/null
+++ b/framework/src/main/java/cn/lili/modules/lmk/enums/general/MaterialStatusEnum.java
@@ -0,0 +1,13 @@
+package cn.lili.modules.lmk.enums.general;
+
+import lombok.AllArgsConstructor;
+
+@AllArgsConstructor
+public enum MaterialStatusEnum {
+    NOT_GENERATE("娌℃湁鐢熸垚"),
+    GENERATE("鐢熸垚");
+
+    private String des;
+
+
+}
diff --git a/framework/src/main/java/cn/lili/modules/lmk/enums/general/StoreScanPrizeStausEnum.java b/framework/src/main/java/cn/lili/modules/lmk/enums/general/StoreScanPrizeStausEnum.java
new file mode 100644
index 0000000..1ee2e0e
--- /dev/null
+++ b/framework/src/main/java/cn/lili/modules/lmk/enums/general/StoreScanPrizeStausEnum.java
@@ -0,0 +1,13 @@
+package cn.lili.modules.lmk.enums.general;
+
+import lombok.AllArgsConstructor;
+
+@AllArgsConstructor
+public enum StoreScanPrizeStausEnum {
+    ENABLE("鍚敤"),
+    DISABLE("绂佺敤");
+
+    private String des;
+
+
+}
diff --git a/framework/src/main/java/cn/lili/modules/lmk/mapper/PrizeClaimRecordMapper.java b/framework/src/main/java/cn/lili/modules/lmk/mapper/PrizeClaimRecordMapper.java
new file mode 100644
index 0000000..0b79874
--- /dev/null
+++ b/framework/src/main/java/cn/lili/modules/lmk/mapper/PrizeClaimRecordMapper.java
@@ -0,0 +1,34 @@
+package cn.lili.modules.lmk.mapper;
+
+import cn.lili.modules.lmk.domain.entity.PrizeClaimRecord;
+import com.baomidou.mybatisplus.core.metadata.IPage;
+import com.baomidou.mybatisplus.core.mapper.BaseMapper;
+import cn.lili.modules.lmk.domain.vo.PrizeClaimRecordVO;
+import cn.lili.modules.lmk.domain.form.PrizeClaimRecordForm;
+import cn.lili.modules.lmk.domain.query.PrizeClaimRecordQuery;
+import java.util.List;
+import org.apache.ibatis.annotations.Mapper;
+import org.apache.ibatis.annotations.Param;
+
+/**
+ * 搴楅摵鎶藉娲诲姩棰嗗彇琛� Mapper 鎺ュ彛
+ *
+ * @author peng
+ * @since 2025-09-30
+ */
+@Mapper
+public interface PrizeClaimRecordMapper extends BaseMapper<PrizeClaimRecord> {
+
+    /**
+     * id鏌ユ壘搴楅摵鎶藉娲诲姩棰嗗彇琛�
+     * @param id
+     * @return
+     */
+    PrizeClaimRecordVO getById(String id);
+
+    /**
+    *  鍒嗛〉
+    */
+    IPage getPage(IPage page, @Param("query") PrizeClaimRecordQuery query);
+
+}
diff --git a/framework/src/main/java/cn/lili/modules/lmk/mapper/ScanPrizeMapper.java b/framework/src/main/java/cn/lili/modules/lmk/mapper/ScanPrizeMapper.java
new file mode 100644
index 0000000..e4e4169
--- /dev/null
+++ b/framework/src/main/java/cn/lili/modules/lmk/mapper/ScanPrizeMapper.java
@@ -0,0 +1,34 @@
+package cn.lili.modules.lmk.mapper;
+
+import cn.lili.modules.lmk.domain.entity.ScanPrize;
+import com.baomidou.mybatisplus.core.metadata.IPage;
+import com.baomidou.mybatisplus.core.mapper.BaseMapper;
+import cn.lili.modules.lmk.domain.vo.ScanPrizeVO;
+import cn.lili.modules.lmk.domain.form.ScanPrizeForm;
+import cn.lili.modules.lmk.domain.query.ScanPrizeQuery;
+import java.util.List;
+import org.apache.ibatis.annotations.Mapper;
+import org.apache.ibatis.annotations.Param;
+
+/**
+ * 搴楅摵娲诲姩鍏宠仈 Mapper 鎺ュ彛
+ *
+ * @author peng
+ * @since 2025-09-30
+ */
+@Mapper
+public interface ScanPrizeMapper extends BaseMapper<ScanPrize> {
+
+    /**
+     * id鏌ユ壘搴楅摵娲诲姩鍏宠仈
+     * @param id
+     * @return
+     */
+    ScanPrizeVO getById(String id);
+
+    /**
+    *  鍒嗛〉
+    */
+    IPage getPage(IPage page, @Param("query") ScanPrizeQuery query);
+
+}
diff --git a/framework/src/main/java/cn/lili/modules/lmk/mapper/StorePrizeProofMapper.java b/framework/src/main/java/cn/lili/modules/lmk/mapper/StorePrizeProofMapper.java
new file mode 100644
index 0000000..413ed88
--- /dev/null
+++ b/framework/src/main/java/cn/lili/modules/lmk/mapper/StorePrizeProofMapper.java
@@ -0,0 +1,34 @@
+package cn.lili.modules.lmk.mapper;
+
+import cn.lili.modules.lmk.domain.entity.StorePrizeProof;
+import com.baomidou.mybatisplus.core.metadata.IPage;
+import com.baomidou.mybatisplus.core.mapper.BaseMapper;
+import cn.lili.modules.lmk.domain.vo.StorePrizeProofVO;
+import cn.lili.modules.lmk.domain.form.StorePrizeProofForm;
+import cn.lili.modules.lmk.domain.query.StorePrizeProofQuery;
+import java.util.List;
+import org.apache.ibatis.annotations.Mapper;
+import org.apache.ibatis.annotations.Param;
+
+/**
+ * 鎶藉娲诲姩琛ュ綍璇佹槑 Mapper 鎺ュ彛
+ *
+ * @author peng
+ * @since 2025-09-30
+ */
+@Mapper
+public interface StorePrizeProofMapper extends BaseMapper<StorePrizeProof> {
+
+    /**
+     * id鏌ユ壘鎶藉娲诲姩琛ュ綍璇佹槑
+     * @param id
+     * @return
+     */
+    StorePrizeProofVO getById(String id);
+
+    /**
+    *  鍒嗛〉
+    */
+    IPage getPage(IPage page, @Param("query") StorePrizeProofQuery query);
+
+}
diff --git a/framework/src/main/java/cn/lili/modules/lmk/service/PrizeClaimRecordService.java b/framework/src/main/java/cn/lili/modules/lmk/service/PrizeClaimRecordService.java
new file mode 100644
index 0000000..179988f
--- /dev/null
+++ b/framework/src/main/java/cn/lili/modules/lmk/service/PrizeClaimRecordService.java
@@ -0,0 +1,65 @@
+package cn.lili.modules.lmk.service;
+
+import cn.lili.modules.lmk.domain.entity.PrizeClaimRecord;
+import com.baomidou.mybatisplus.extension.service.IService;
+import cn.lili.base.Result;
+import cn.lili.modules.lmk.domain.form.PrizeClaimRecordForm;
+import cn.lili.modules.lmk.domain.query.PrizeClaimRecordQuery;
+import java.util.List;
+
+/**
+ * 搴楅摵鎶藉娲诲姩棰嗗彇琛� 鏈嶅姟绫�
+ *
+ * @author peng
+ * @since 2025-09-30
+ */
+public interface PrizeClaimRecordService extends IService<PrizeClaimRecord> {
+
+    /**
+     * 娣诲姞
+     * @param form
+     * @return
+     */
+    Result add(PrizeClaimRecordForm form);
+
+    /**
+     * 淇敼
+     * @param form
+     * @return
+     */
+    Result update(PrizeClaimRecordForm form);
+
+    /**
+     * 鎵归噺鍒犻櫎
+     * @param ids
+     * @return
+     */
+    Result remove(List<String> ids);
+
+    /**
+     * id鍒犻櫎
+     * @param id
+     * @return
+     */
+    Result removeById(String id);
+
+    /**
+     * 鍒嗛〉鏌ヨ
+     * @param query
+     * @return
+     */
+    Result page(PrizeClaimRecordQuery query);
+
+    /**
+     * 鏍规嵁id鏌ユ壘
+     * @param id
+     * @return
+     */
+    Result detail(String id);
+
+    /**
+     * 鍒楄〃
+     * @return
+     */
+    Result all();
+}
diff --git a/framework/src/main/java/cn/lili/modules/lmk/service/ScanPrizeService.java b/framework/src/main/java/cn/lili/modules/lmk/service/ScanPrizeService.java
new file mode 100644
index 0000000..925e543
--- /dev/null
+++ b/framework/src/main/java/cn/lili/modules/lmk/service/ScanPrizeService.java
@@ -0,0 +1,69 @@
+package cn.lili.modules.lmk.service;
+
+import cn.lili.modules.lmk.domain.entity.ScanPrize;
+import com.baomidou.mybatisplus.extension.service.IService;
+import cn.lili.base.Result;
+import cn.lili.modules.lmk.domain.form.ScanPrizeForm;
+import cn.lili.modules.lmk.domain.query.ScanPrizeQuery;
+import java.util.List;
+
+/**
+ * 搴楅摵娲诲姩鍏宠仈 鏈嶅姟绫�
+ *
+ * @author peng
+ * @since 2025-09-30
+ */
+public interface ScanPrizeService extends IService<ScanPrize> {
+
+    /**
+     * 娣诲姞
+     * @param form
+     * @return
+     */
+    Result add(ScanPrizeForm form);
+
+    /**
+     * 淇敼
+     * @param form
+     * @return
+     */
+    Result update(ScanPrizeForm form);
+
+    /**
+     * 鎵归噺鍒犻櫎
+     * @param ids
+     * @return
+     */
+    Result remove(List<String> ids);
+
+    /**
+     * id鍒犻櫎
+     * @param id
+     * @return
+     */
+    Result removeById(String id);
+
+    /**
+     * 鍒嗛〉鏌ヨ
+     * @param query
+     * @return
+     */
+    Result page(ScanPrizeQuery query);
+
+    /**
+     * 鏍规嵁id鏌ユ壘
+     * @param id
+     * @return
+     */
+    Result detail(String id);
+
+    /**
+     * 鍒楄〃
+     * @return
+     */
+    Result all();
+
+    Result changeStatus(String id);
+
+    Result generateStorePrize(String id);
+}
diff --git a/framework/src/main/java/cn/lili/modules/lmk/service/StorePrizeProofService.java b/framework/src/main/java/cn/lili/modules/lmk/service/StorePrizeProofService.java
new file mode 100644
index 0000000..a2b2f3b
--- /dev/null
+++ b/framework/src/main/java/cn/lili/modules/lmk/service/StorePrizeProofService.java
@@ -0,0 +1,65 @@
+package cn.lili.modules.lmk.service;
+
+import cn.lili.modules.lmk.domain.entity.StorePrizeProof;
+import com.baomidou.mybatisplus.extension.service.IService;
+import cn.lili.base.Result;
+import cn.lili.modules.lmk.domain.form.StorePrizeProofForm;
+import cn.lili.modules.lmk.domain.query.StorePrizeProofQuery;
+import java.util.List;
+
+/**
+ * 鎶藉娲诲姩琛ュ綍璇佹槑 鏈嶅姟绫�
+ *
+ * @author peng
+ * @since 2025-09-30
+ */
+public interface StorePrizeProofService extends IService<StorePrizeProof> {
+
+    /**
+     * 娣诲姞
+     * @param form
+     * @return
+     */
+    Result add(StorePrizeProofForm form);
+
+    /**
+     * 淇敼
+     * @param form
+     * @return
+     */
+    Result update(StorePrizeProofForm form);
+
+    /**
+     * 鎵归噺鍒犻櫎
+     * @param ids
+     * @return
+     */
+    Result remove(List<String> ids);
+
+    /**
+     * id鍒犻櫎
+     * @param id
+     * @return
+     */
+    Result removeById(String id);
+
+    /**
+     * 鍒嗛〉鏌ヨ
+     * @param query
+     * @return
+     */
+    Result page(StorePrizeProofQuery query);
+
+    /**
+     * 鏍规嵁id鏌ユ壘
+     * @param id
+     * @return
+     */
+    Result detail(String id);
+
+    /**
+     * 鍒楄〃
+     * @return
+     */
+    Result all();
+}
diff --git a/framework/src/main/java/cn/lili/modules/lmk/service/impl/PrizeClaimRecordServiceImpl.java b/framework/src/main/java/cn/lili/modules/lmk/service/impl/PrizeClaimRecordServiceImpl.java
new file mode 100644
index 0000000..eb80535
--- /dev/null
+++ b/framework/src/main/java/cn/lili/modules/lmk/service/impl/PrizeClaimRecordServiceImpl.java
@@ -0,0 +1,119 @@
+package cn.lili.modules.lmk.service.impl;
+
+import com.baomidou.mybatisplus.core.metadata.IPage;
+import cn.lili.modules.lmk.domain.entity.PrizeClaimRecord;
+import cn.lili.modules.lmk.mapper.PrizeClaimRecordMapper;
+import cn.lili.modules.lmk.service.PrizeClaimRecordService;
+import cn.lili.base.Result;
+import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
+import cn.lili.modules.lmk.domain.form.PrizeClaimRecordForm;
+import cn.lili.modules.lmk.domain.vo.PrizeClaimRecordVO;
+import cn.lili.modules.lmk.domain.query.PrizeClaimRecordQuery;
+import org.springframework.stereotype.Service;
+import lombok.RequiredArgsConstructor;
+import cn.lili.utils.PageUtil;
+import org.springframework.beans.BeanUtils;
+import org.springframework.util.Assert;
+
+import java.util.List;
+import java.util.stream.Collectors;
+
+/**
+ * 搴楅摵鎶藉娲诲姩棰嗗彇琛� 鏈嶅姟瀹炵幇绫�
+ *
+ * @author peng
+ * @since 2025-09-30
+ */
+@Service
+@RequiredArgsConstructor
+public class PrizeClaimRecordServiceImpl extends ServiceImpl<PrizeClaimRecordMapper, PrizeClaimRecord> implements PrizeClaimRecordService {
+
+    private final PrizeClaimRecordMapper prizeClaimRecordMapper;
+
+    /**
+     * 娣诲姞
+     * @param form
+     * @return
+     */
+    @Override
+    public Result add(PrizeClaimRecordForm form) {
+        PrizeClaimRecord entity = PrizeClaimRecordForm.getEntityByForm(form, null);
+        baseMapper.insert(entity);
+        return Result.ok("娣诲姞鎴愬姛");
+    }
+
+    /**
+     * 淇敼
+     * @param form
+     * @return
+     */
+    @Override
+    public Result update(PrizeClaimRecordForm form) {
+        PrizeClaimRecord entity = baseMapper.selectById(form.getId());
+
+        // 涓虹┖鎶汭llegalArgumentException锛屽仛鍏ㄥ眬寮傚父澶勭悊
+        Assert.notNull(entity, "璁板綍涓嶅瓨鍦�");
+        BeanUtils.copyProperties(form, entity);
+        baseMapper.updateById(entity);
+        return Result.ok("淇敼鎴愬姛");
+    }
+
+    /**
+     * 鎵归噺鍒犻櫎
+     * @param ids
+     * @return
+     */
+    @Override
+    public Result remove(List<String> ids) {
+        baseMapper.deleteBatchIds(ids);
+        return Result.ok("鍒犻櫎鎴愬姛");
+    }
+
+    /**
+     * id鍒犻櫎
+     * @param id
+     * @return
+     */
+    @Override
+    public Result removeById(String id) {
+        baseMapper.deleteById(id);
+        return Result.ok("鍒犻櫎鎴愬姛");
+    }
+
+    /**
+     * 鍒嗛〉鏌ヨ
+     * @param query
+     * @return
+     */
+    @Override
+    public Result page(PrizeClaimRecordQuery query) {
+        IPage<PrizeClaimRecordVO> page = PageUtil.getPage(query, PrizeClaimRecordVO.class);
+        baseMapper.getPage(page, query);
+        return Result.ok().data(page.getRecords()).total(page.getTotal());
+    }
+
+    /**
+     * 鏍规嵁id鏌ユ壘
+     * @param id
+     * @return
+     */
+    @Override
+    public Result detail(String id) {
+        PrizeClaimRecordVO vo = baseMapper.getById(id);
+        Assert.notNull(vo, "璁板綍涓嶅瓨鍦�");
+        return Result.ok().data(vo);
+    }
+
+    /**
+     * 鍒楄〃
+     * @return
+     */
+    @Override
+    public Result all() {
+        List<PrizeClaimRecord> entities = baseMapper.selectList(null);
+        List<PrizeClaimRecordVO> vos = entities.stream()
+                .map(entity -> PrizeClaimRecordVO.getVoByEntity(entity, null))
+                .collect(Collectors.toList());
+        return Result.ok().data(vos);
+    }
+}
diff --git a/framework/src/main/java/cn/lili/modules/lmk/service/impl/ScanPrizeServiceImpl.java b/framework/src/main/java/cn/lili/modules/lmk/service/impl/ScanPrizeServiceImpl.java
new file mode 100644
index 0000000..9e55bd0
--- /dev/null
+++ b/framework/src/main/java/cn/lili/modules/lmk/service/impl/ScanPrizeServiceImpl.java
@@ -0,0 +1,218 @@
+package cn.lili.modules.lmk.service.impl;
+
+import cn.lili.common.exception.ServiceException;
+import cn.lili.modules.lmk.domain.entity.PrizeClaimRecord;
+import cn.lili.modules.lmk.domain.entity.StoreCoupon;
+import cn.lili.modules.lmk.domain.entity.StoreCouponSingle;
+import cn.lili.modules.lmk.enums.general.*;
+import cn.lili.modules.lmk.service.PrizeClaimRecordService;
+import cn.lili.modules.order.order.entity.enums.ClaimStatusEnum;
+import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
+import com.baomidou.mybatisplus.core.metadata.IPage;
+import cn.lili.modules.lmk.domain.entity.ScanPrize;
+import cn.lili.modules.lmk.mapper.ScanPrizeMapper;
+import cn.lili.modules.lmk.service.ScanPrizeService;
+import cn.lili.base.Result;
+import com.baomidou.mybatisplus.core.toolkit.Wrappers;
+import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
+import cn.lili.modules.lmk.domain.form.ScanPrizeForm;
+import cn.lili.modules.lmk.domain.vo.ScanPrizeVO;
+import cn.lili.modules.lmk.domain.query.ScanPrizeQuery;
+import org.redisson.api.RLock;
+import org.redisson.api.RedissonClient;
+import org.springframework.stereotype.Service;
+import lombok.RequiredArgsConstructor;
+import cn.lili.utils.PageUtil;
+import org.springframework.beans.BeanUtils;
+import org.springframework.transaction.support.TransactionSynchronization;
+import org.springframework.transaction.support.TransactionSynchronizationManager;
+import org.springframework.util.Assert;
+
+import java.util.ArrayList;
+import java.util.List;
+import java.util.stream.Collectors;
+
+/**
+ * 搴楅摵娲诲姩鍏宠仈 鏈嶅姟瀹炵幇绫�
+ *
+ * @author peng
+ * @since 2025-09-30
+ */
+@Service
+@RequiredArgsConstructor
+public class ScanPrizeServiceImpl extends ServiceImpl<ScanPrizeMapper, ScanPrize> implements ScanPrizeService {
+
+    private final ScanPrizeMapper scanPrizeMapper;
+    private final RedissonClient redissonClient;
+    private static final String STORE_PRIZE_GENERATE = "store_prize_generate:";
+    private final PrizeClaimRecordService prizeClaimRecordService;
+
+    /**
+     * 娣诲姞
+     * @param form
+     * @return
+     */
+    @Override
+    public Result add(ScanPrizeForm form) {
+        ScanPrize entity = ScanPrizeForm.getEntityByForm(form, null);
+        entity.setStatus(StoreScanPrizeStausEnum.ENABLE.name());
+        entity.setGenerateStatus(GenerateStorePrizeStausEnum.NOT_GENERATE.name());
+        entity.setClaimNum(0);
+        baseMapper.insert(entity);
+        return Result.ok("娣诲姞鎴愬姛");
+    }
+
+    /**
+     * 淇敼
+     * @param form
+     * @return
+     */
+    @Override
+    public Result update(ScanPrizeForm form) {
+        ScanPrize entity = baseMapper.selectById(form.getId());
+
+        // 涓虹┖鎶汭llegalArgumentException锛屽仛鍏ㄥ眬寮傚父澶勭悊
+        Assert.notNull(entity, "璁板綍涓嶅瓨鍦�");
+        BeanUtils.copyProperties(form, entity);
+        baseMapper.updateById(entity);
+        return Result.ok("淇敼鎴愬姛");
+    }
+
+    /**
+     * 鎵归噺鍒犻櫎
+     * @param ids
+     * @return
+     */
+    @Override
+    public Result remove(List<String> ids) {
+        baseMapper.deleteBatchIds(ids);
+        return Result.ok("鍒犻櫎鎴愬姛");
+    }
+
+    /**
+     * id鍒犻櫎
+     * @param id
+     * @return
+     */
+    @Override
+    public Result removeById(String id) {
+        baseMapper.deleteById(id);
+        return Result.ok("鍒犻櫎鎴愬姛");
+    }
+
+    /**
+     * 鍒嗛〉鏌ヨ
+     * @param query
+     * @return
+     */
+    @Override
+    public Result page(ScanPrizeQuery query) {
+        IPage<ScanPrizeVO> page = PageUtil.getPage(query, ScanPrizeVO.class);
+        baseMapper.getPage(page, query);
+        return Result.ok().data(page.getRecords()).total(page.getTotal());
+    }
+
+    /**
+     * 鏍规嵁id鏌ユ壘
+     * @param id
+     * @return
+     */
+    @Override
+    public Result detail(String id) {
+        ScanPrizeVO vo = baseMapper.getById(id);
+        Assert.notNull(vo, "璁板綍涓嶅瓨鍦�");
+        return Result.ok().data(vo);
+    }
+
+    /**
+     * 鍒楄〃
+     * @return
+     */
+    @Override
+    public Result all() {
+        List<ScanPrize> entities = baseMapper.selectList(null);
+        List<ScanPrizeVO> vos = entities.stream()
+                .map(entity -> ScanPrizeVO.getVoByEntity(entity, null))
+                .collect(Collectors.toList());
+        return Result.ok().data(vos);
+    }
+
+    @Override
+    public Result changeStatus(String id) {
+        ScanPrize scanPrize = this.getById(id);
+        if (scanPrize == null) {
+            throw new ServiceException("褰撳墠搴楅摵鎶藉鏈轰細涓嶅瓨鍦�");
+        }
+        String status = scanPrize.getStatus();
+        if (StoreCouponStausEnum.ENABLE.name().equals(status)) {
+            scanPrize.setStatus(StoreCouponStausEnum.DISABLE.name());
+        }else if (StoreCouponStausEnum.DISABLE.name().equals(status)) {
+            scanPrize.setStatus(StoreCouponStausEnum.ENABLE.name());
+        }else {
+            throw new ServiceException("褰撳墠搴楅摵鎶藉鏈轰細鐘舵�佸紓甯告棤娉曚慨鏀�");
+        }
+        this.updateById(scanPrize);
+        return Result.ok();
+    }
+
+    @Override
+    public Result generateStorePrize(String id) {
+
+        RLock redissonLock = redissonClient.getLock(STORE_PRIZE_GENERATE + id);
+        try {
+            redissonLock.lock();
+            LambdaQueryWrapper<ScanPrize> forUpdate = Wrappers.<ScanPrize>lambdaQuery().eq(ScanPrize::getId, id).last("FOR UPDATE");
+            ScanPrize scanPrize = this.getOne(forUpdate);
+            if (scanPrize == null) {
+                throw new ServiceException("褰撳墠搴楅摵鎶藉鏈轰細涓嶅瓨鍦ㄦ棤娉曠敓鎴�");
+            }
+            if (!GenerateCouponStausEnum.NOT_GENERATE.name().equals( scanPrize.getGenerateStatus())) {
+                throw new ServiceException("褰撳墠搴楅摵鎶藉鏈轰細鐘舵�佸紓甯告棤娉曠敓鎴�");
+            }
+            //鐢熸垚浼樻儬鍗�
+            Integer generateNum = scanPrize.getGenerateNum();
+            if (generateNum == null) {
+                throw new ServiceException("鍙戣鏁伴噺涓虹┖涓嶈兘鐢熸垚");
+            }
+            List<PrizeClaimRecord> prizeClaimRecords = new ArrayList<>();
+            for (int i = 1; i <= generateNum; i++) {
+                PrizeClaimRecord prizeClaimRecord = getStoreCouponSingle(scanPrize, i);
+                prizeClaimRecords.add(prizeClaimRecord);
+            }
+            if (!prizeClaimRecords.isEmpty()) {
+                prizeClaimRecordService.saveBatch(prizeClaimRecords);
+            }
+            //鏇存柊鐘舵�佺敓鎴愮姸鎬�
+            scanPrize.setGenerateStatus(GenerateCouponStausEnum.GENERATE.name());
+            this.updateById(scanPrize);
+            return Result.ok();
+        }finally {
+            TransactionSynchronizationManager.registerSynchronization(new TransactionSynchronization() {
+                @Override
+                public void afterCommit() {
+                    if (redissonLock.isHeldByCurrentThread()) {
+                        redissonLock.unlock();
+                    }
+                }
+                @Override
+                public void afterCompletion(int status) {
+                    if (redissonLock.isHeldByCurrentThread()) {
+                        redissonLock.unlock();
+                    }
+                }
+            });
+        }
+    }
+    private static PrizeClaimRecord getStoreCouponSingle(ScanPrize scanPrize, int i) {
+        PrizeClaimRecord prizeClaimRecord = new PrizeClaimRecord();
+        prizeClaimRecord.setStorePrizeId(scanPrize.getId());
+        prizeClaimRecord.setStoreId(scanPrize.getStoreId());
+        prizeClaimRecord.setStoreName(scanPrize.getStoreName());
+        prizeClaimRecord.setPrizeActivityId(scanPrize.getPrizeActivityId());
+        prizeClaimRecord.setPrizeActivityName(scanPrize.getPrizeActivityNmae());
+        prizeClaimRecord.setClaimStatus(ClaimStatusEnum.NOT_CLAIM.name());
+        prizeClaimRecord.setNo(String.format("%08d", i));
+        prizeClaimRecord.setMaterial(MaterialStatusEnum.NOT_GENERATE.name());
+        return prizeClaimRecord;
+    }
+}
diff --git a/framework/src/main/java/cn/lili/modules/lmk/service/impl/StorePrizeProofServiceImpl.java b/framework/src/main/java/cn/lili/modules/lmk/service/impl/StorePrizeProofServiceImpl.java
new file mode 100644
index 0000000..4c55b59
--- /dev/null
+++ b/framework/src/main/java/cn/lili/modules/lmk/service/impl/StorePrizeProofServiceImpl.java
@@ -0,0 +1,119 @@
+package cn.lili.modules.lmk.service.impl;
+
+import com.baomidou.mybatisplus.core.metadata.IPage;
+import cn.lili.modules.lmk.domain.entity.StorePrizeProof;
+import cn.lili.modules.lmk.mapper.StorePrizeProofMapper;
+import cn.lili.modules.lmk.service.StorePrizeProofService;
+import cn.lili.base.Result;
+import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
+import cn.lili.modules.lmk.domain.form.StorePrizeProofForm;
+import cn.lili.modules.lmk.domain.vo.StorePrizeProofVO;
+import cn.lili.modules.lmk.domain.query.StorePrizeProofQuery;
+import org.springframework.stereotype.Service;
+import lombok.RequiredArgsConstructor;
+import cn.lili.utils.PageUtil;
+import org.springframework.beans.BeanUtils;
+import org.springframework.util.Assert;
+
+import java.util.List;
+import java.util.stream.Collectors;
+
+/**
+ * 鎶藉娲诲姩琛ュ綍璇佹槑 鏈嶅姟瀹炵幇绫�
+ *
+ * @author peng
+ * @since 2025-09-30
+ */
+@Service
+@RequiredArgsConstructor
+public class StorePrizeProofServiceImpl extends ServiceImpl<StorePrizeProofMapper, StorePrizeProof> implements StorePrizeProofService {
+
+    private final StorePrizeProofMapper storePrizeProofMapper;
+
+    /**
+     * 娣诲姞
+     * @param form
+     * @return
+     */
+    @Override
+    public Result add(StorePrizeProofForm form) {
+        StorePrizeProof entity = StorePrizeProofForm.getEntityByForm(form, null);
+        baseMapper.insert(entity);
+        return Result.ok("娣诲姞鎴愬姛");
+    }
+
+    /**
+     * 淇敼
+     * @param form
+     * @return
+     */
+    @Override
+    public Result update(StorePrizeProofForm form) {
+        StorePrizeProof entity = baseMapper.selectById(form.getId());
+
+        // 涓虹┖鎶汭llegalArgumentException锛屽仛鍏ㄥ眬寮傚父澶勭悊
+        Assert.notNull(entity, "璁板綍涓嶅瓨鍦�");
+        BeanUtils.copyProperties(form, entity);
+        baseMapper.updateById(entity);
+        return Result.ok("淇敼鎴愬姛");
+    }
+
+    /**
+     * 鎵归噺鍒犻櫎
+     * @param ids
+     * @return
+     */
+    @Override
+    public Result remove(List<String> ids) {
+        baseMapper.deleteBatchIds(ids);
+        return Result.ok("鍒犻櫎鎴愬姛");
+    }
+
+    /**
+     * id鍒犻櫎
+     * @param id
+     * @return
+     */
+    @Override
+    public Result removeById(String id) {
+        baseMapper.deleteById(id);
+        return Result.ok("鍒犻櫎鎴愬姛");
+    }
+
+    /**
+     * 鍒嗛〉鏌ヨ
+     * @param query
+     * @return
+     */
+    @Override
+    public Result page(StorePrizeProofQuery query) {
+        IPage<StorePrizeProofVO> page = PageUtil.getPage(query, StorePrizeProofVO.class);
+        baseMapper.getPage(page, query);
+        return Result.ok().data(page.getRecords()).total(page.getTotal());
+    }
+
+    /**
+     * 鏍规嵁id鏌ユ壘
+     * @param id
+     * @return
+     */
+    @Override
+    public Result detail(String id) {
+        StorePrizeProofVO vo = baseMapper.getById(id);
+        Assert.notNull(vo, "璁板綍涓嶅瓨鍦�");
+        return Result.ok().data(vo);
+    }
+
+    /**
+     * 鍒楄〃
+     * @return
+     */
+    @Override
+    public Result all() {
+        List<StorePrizeProof> entities = baseMapper.selectList(null);
+        List<StorePrizeProofVO> vos = entities.stream()
+                .map(entity -> StorePrizeProofVO.getVoByEntity(entity, null))
+                .collect(Collectors.toList());
+        return Result.ok().data(vos);
+    }
+}
diff --git a/framework/src/main/resources/mapper/lmk/PrizeClaimRecordMapper.xml b/framework/src/main/resources/mapper/lmk/PrizeClaimRecordMapper.xml
new file mode 100644
index 0000000..31743ee
--- /dev/null
+++ b/framework/src/main/resources/mapper/lmk/PrizeClaimRecordMapper.xml
@@ -0,0 +1,59 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
+<mapper namespace="cn.lili.modules.lmk.mapper.PrizeClaimRecordMapper">
+
+    <!-- 閫氱敤鏌ヨ鏄犲皠缁撴灉 -->
+    <resultMap id="BaseResultMap" type="cn.lili.modules.lmk.domain.vo.PrizeClaimRecordVO">
+        <id column="id" property="id"/>
+        <result column="no" property="no" />
+        <result column="store_id" property="storeId" />
+        <result column="store_name" property="storeName" />
+        <result column="prize_activity_id" property="prizeActivityId" />
+        <result column="prize_activity_name" property="prizeActivityName" />
+        <result column="user_id" property="userId" />
+        <result column="nick_name" property="nickName" />
+        <result column="claim_status" property="claimStatus" />
+    </resultMap>
+
+
+
+
+
+
+
+    <select id="getById" resultMap="BaseResultMap">
+        SELECT
+            LPCR.no,
+            LPCR.store_id,
+            LPCR.store_name,
+            LPCR.prize_activity_id,
+            LPCR.prize_activity_name,
+            LPCR.user_id,
+            LPCR.nick_name,
+            LPCR.claim_status,
+            LPCR.id
+        FROM
+            lmk_prize_claim_record LPCR
+        WHERE
+            LPCR.id = #{id} AND LPCR.delete_flag = 0
+    </select>
+
+
+    <select id="getPage" resultMap="BaseResultMap">
+        SELECT
+            LPCR.no,
+            LPCR.store_id,
+            LPCR.store_name,
+            LPCR.prize_activity_id,
+            LPCR.prize_activity_name,
+            LPCR.user_id,
+            LPCR.nick_name,
+            LPCR.claim_status,
+            LPCR.id
+        FROM
+            lmk_prize_claim_record LPCR
+        WHERE
+            LPCR.delete_flag = 0
+    </select>
+
+</mapper>
diff --git a/framework/src/main/resources/mapper/lmk/ScanPrizeMapper.xml b/framework/src/main/resources/mapper/lmk/ScanPrizeMapper.xml
new file mode 100644
index 0000000..3ddcba9
--- /dev/null
+++ b/framework/src/main/resources/mapper/lmk/ScanPrizeMapper.xml
@@ -0,0 +1,59 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
+<mapper namespace="cn.lili.modules.lmk.mapper.ScanPrizeMapper">
+
+    <!-- 閫氱敤鏌ヨ鏄犲皠缁撴灉 -->
+    <resultMap id="BaseResultMap" type="cn.lili.modules.lmk.domain.vo.ScanPrizeVO">
+        <id column="id" property="id"/>
+        <result column="store_id" property="storeId" />
+        <result column="store_name" property="storeName" />
+        <result column="prize_activity_id" property="prizeActivityId" />
+        <result column="prize_activity_nmae" property="prizeActivityNmae" />
+        <result column="generate_status" property="generateStatus" />
+        <result column="generate_num" property="generateNum" />
+        <result column="claim_num" property="claimNum" />
+        <result column="status" property="status" />
+    </resultMap>
+
+
+
+
+
+
+
+    <select id="getById" resultMap="BaseResultMap">
+        SELECT
+            LSP.store_id,
+            LSP.store_name,
+            LSP.prize_activity_id,
+            LSP.prize_activity_nmae,
+            LSP.generate_status,
+            LSP.generate_num,
+            LSP.claim_num,
+            LSP.status,
+            LSP.id
+        FROM
+            lmk_scan_prize LSP
+        WHERE
+            LSP.id = #{id} AND LSP.delete_flag = 0
+    </select>
+
+
+    <select id="getPage" resultMap="BaseResultMap">
+        SELECT
+            LSP.store_id,
+            LSP.store_name,
+            LSP.prize_activity_id,
+            LSP.prize_activity_nmae,
+            LSP.generate_status,
+            LSP.generate_num,
+            LSP.claim_num,
+            LSP.status,
+            LSP.id
+        FROM
+            lmk_scan_prize LSP
+        WHERE
+            LSP.delete_flag = 0
+    </select>
+
+</mapper>
diff --git a/framework/src/main/resources/mapper/lmk/StorePrizeProofMapper.xml b/framework/src/main/resources/mapper/lmk/StorePrizeProofMapper.xml
new file mode 100644
index 0000000..fcc03f2
--- /dev/null
+++ b/framework/src/main/resources/mapper/lmk/StorePrizeProofMapper.xml
@@ -0,0 +1,41 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
+<mapper namespace="cn.lili.modules.lmk.mapper.StorePrizeProofMapper">
+
+    <!-- 閫氱敤鏌ヨ鏄犲皠缁撴灉 -->
+    <resultMap id="BaseResultMap" type="cn.lili.modules.lmk.domain.vo.StorePrizeProofVO">
+        <id column="id" property="id"/>
+        <result column="store_prize_claim_id" property="storePrizeClaimId" />
+        <result column="url_path" property="urlPath" />
+    </resultMap>
+
+
+
+
+
+
+
+    <select id="getById" resultMap="BaseResultMap">
+        SELECT
+            LSPP.store_prize_claim_id,
+            LSPP.url_path,
+            LSPP.id
+        FROM
+            lmk_store_prize_proof LSPP
+        WHERE
+            LSPP.id = #{id} AND LSPP.delete_flag = 0
+    </select>
+
+
+    <select id="getPage" resultMap="BaseResultMap">
+        SELECT
+            LSPP.store_prize_claim_id,
+            LSPP.url_path,
+            LSPP.id
+        FROM
+            lmk_store_prize_proof LSPP
+        WHERE
+            LSPP.delete_flag = 0
+    </select>
+
+</mapper>
diff --git a/manager-api/src/main/java/cn/lili/controller/lmk/ScanPrizeController.java b/manager-api/src/main/java/cn/lili/controller/lmk/ScanPrizeController.java
new file mode 100644
index 0000000..1bd0e02
--- /dev/null
+++ b/manager-api/src/main/java/cn/lili/controller/lmk/ScanPrizeController.java
@@ -0,0 +1,98 @@
+package cn.lili.controller.lmk;
+
+import cn.lili.group.Update;
+import cn.lili.group.Add;
+import org.springframework.validation.annotation.Validated;
+import org.springframework.security.access.prepost.PreAuthorize;
+import lombok.RequiredArgsConstructor;
+import java.util.List;
+import org.springframework.validation.annotation.Validated;
+import javax.validation.constraints.NotEmpty;
+import io.swagger.annotations.Api;
+import io.swagger.annotations.ApiOperation;
+import cn.lili.modules.lmk.service.ScanPrizeService;
+import cn.lili.base.Result;
+import cn.lili.modules.lmk.domain.form.ScanPrizeForm;
+import cn.lili.modules.lmk.domain.query.ScanPrizeQuery;
+import lombok.RequiredArgsConstructor;
+import org.springframework.web.bind.annotation.*;
+
+/**
+ * 搴楅摵娲诲姩鍏宠仈 鍓嶇鎺у埗鍣�
+ *
+ * @author peng
+ * @since 2025-09-30
+ */
+@Validated
+@RequiredArgsConstructor
+@Api(value = "搴楅摵娲诲姩鍏宠仈", tags = "搴楅摵娲诲姩鍏宠仈绠$悊")
+@RestController
+@RequestMapping("/manager/lmk/scan-prize")
+public class ScanPrizeController {
+
+    private final ScanPrizeService scanPrizeService;
+
+
+    @PutMapping
+    @ApiOperation(value = "淇敼", notes = "淇敼")
+    public Result update(@RequestBody @Validated(Update.class) ScanPrizeForm form) {
+        return scanPrizeService.update(form);
+    }
+
+    @DeleteMapping("/{id}")
+    @ApiOperation(value = "ID鍒犻櫎", notes = "ID鍒犻櫎")
+    public Result removeById(@PathVariable("id") String id) {
+        return scanPrizeService.removeById(id);
+    }
+
+    @DeleteMapping("/batch")
+    @ApiOperation(value = "鎵归噺鍒犻櫎", notes = "鎵归噺鍒犻櫎")
+    public Result remove(@RequestBody @NotEmpty(message = "璇烽�夋嫨鏁版嵁") List<String> ids) {
+        return scanPrizeService.remove(ids);
+    }
+
+    @GetMapping("/page")
+    @ApiOperation(value = "鍒嗛〉", notes = "鍒嗛〉")
+    public Result page(ScanPrizeQuery query) {
+        return scanPrizeService.page(query);
+    }
+
+    @GetMapping("/{id}")
+    @ApiOperation(value = "璇︽儏", notes = "璇︽儏")
+    public Result detail(@PathVariable("id") String id) {
+        return scanPrizeService.detail(id);
+    }
+
+    @GetMapping("/list")
+    @ApiOperation(value = "鍒楄〃", notes = "鍒楄〃")
+    public Result list() {
+        return scanPrizeService.all();
+    }
+
+
+    @PostMapping
+    @ApiOperation(value = "娣诲姞", notes = "娣诲姞")
+    public Result add(@RequestBody @Validated(Add.class) ScanPrizeForm form) {
+        return scanPrizeService.add(form);
+    }
+
+    /**
+     * 淇敼璁㈠崟搴楅摵鎶藉鏈轰細鐘舵��
+     * @param id
+     * @return
+     */
+    @PostMapping("/changeStatus/{id}")
+    public Result changeStatus(@PathVariable String id){
+        return  scanPrizeService.changeStatus(id);
+    }
+
+    /**
+     * 鏍规嵁搴楅摵鍏宠仈鍏崇郴鐢熸垚鍗曞搧
+     * @param id
+     * @return
+     */
+    @PostMapping("/generateStorePrize/{id}")
+    public Result generateStorePrize(@PathVariable String id){
+        return  scanPrizeService.generateStorePrize(id);
+    }
+}

--
Gitblit v1.8.0