From 600ea96a245bd5522c489fafc4993cafa0ce36db Mon Sep 17 00:00:00 2001
From: Administrator <15274802129@163.com>
Date: Fri, 01 Aug 2025 14:15:25 +0800
Subject: [PATCH] feat(ai): 新增产品知识点功能

---
 src/main/java/cc/mrbird/febs/ai/service/AiProductService.java                               |    3 
 src/main/java/cc/mrbird/febs/ai/controller/productPoint/AiProductPointController.java       |   65 +++
 src/main/resources/templates/febs/views/modules/ai/productPoint/info.html                   |  186 ++++++++++
 src/main/java/cc/mrbird/febs/ai/controller/memberRole/AiMemberRoleController.java           |    2 
 src/main/java/cc/mrbird/febs/ai/controller/memberRole/ViewController.java                   |    2 
 src/main/resources/templates/febs/views/modules/ai/productPoint/add.html                    |  159 ++++++++
 src/main/resources/templates/febs/views/modules/ai/product/list.html                        |   13 
 src/main/java/cc/mrbird/febs/ai/service/impl/AiProductPointLinkServiceImpl.java             |   35 +
 src/main/java/cc/mrbird/febs/ai/service/AiProductPointService.java                          |   14 
 src/main/java/cc/mrbird/febs/ai/service/impl/AiMemberRoleServiceImpl.java                   |   14 
 src/main/java/cc/mrbird/febs/ai/service/impl/AiProductPointServiceImpl.java                 |   72 +++
 src/main/resources/templates/febs/views/modules/ai/productPoint/list.html                   |  189 ++++++++++
 src/main/java/cc/mrbird/febs/ai/controller/product/ViewController.java                      |   62 +++
 src/main/resources/templates/febs/views/modules/ai/product/pointSet.html                    |  128 +++++++
 src/main/java/cc/mrbird/febs/ai/service/impl/AiProductServiceImpl.java                      |   38 +
 src/main/java/cc/mrbird/febs/ai/service/AiProductPointLinkService.java                      |   11 
 src/main/java/cc/mrbird/febs/ai/req/AdminMoveChooseInfoDto.java                             |    2 
 src/main/resources/templates/febs/views/modules/ai/memberRole/productSet.html               |   10 
 src/main/java/cc/mrbird/febs/ai/controller/product/AiProductController.java                 |   23 
 src/main/java/cc/mrbird/febs/ai/controller/productCategory/AiProductCategoryController.java |    2 
 src/main/java/cc/mrbird/febs/ai/controller/productPoint/ViewController.java                 |   47 ++
 21 files changed, 1,041 insertions(+), 36 deletions(-)

diff --git a/src/main/java/cc/mrbird/febs/ai/controller/memberRole/AiMemberRoleController.java b/src/main/java/cc/mrbird/febs/ai/controller/memberRole/AiMemberRoleController.java
index e03c29a..4300e07 100644
--- a/src/main/java/cc/mrbird/febs/ai/controller/memberRole/AiMemberRoleController.java
+++ b/src/main/java/cc/mrbird/febs/ai/controller/memberRole/AiMemberRoleController.java
@@ -54,7 +54,7 @@
     }
 
     @PostMapping("update")
-    @ControllerEndpoint(operation = "分类-更新", exceptionMessage = "操作失败")
+    @ControllerEndpoint(operation = "更新", exceptionMessage = "操作失败")
     public FebsResponse update(@RequestBody @Valid AiMemberRole dto) {
 
         return aiMemberRoleService.updateMemberRole(dto);
diff --git a/src/main/java/cc/mrbird/febs/ai/controller/memberRole/ViewController.java b/src/main/java/cc/mrbird/febs/ai/controller/memberRole/ViewController.java
index 9e673bf..24dd069 100644
--- a/src/main/java/cc/mrbird/febs/ai/controller/memberRole/ViewController.java
+++ b/src/main/java/cc/mrbird/febs/ai/controller/memberRole/ViewController.java
@@ -96,7 +96,7 @@
 
         model.addAttribute("productAll", vos);
         model.addAttribute("productSelected", productIds);
-        model.addAttribute("roleId", id);
+        model.addAttribute("chooseId", id);
         return FebsUtil.view("modules/ai/memberRole/productSet");
     }
 }
diff --git a/src/main/java/cc/mrbird/febs/ai/controller/product/AiProductController.java b/src/main/java/cc/mrbird/febs/ai/controller/product/AiProductController.java
index 692baed..c17f613 100644
--- a/src/main/java/cc/mrbird/febs/ai/controller/product/AiProductController.java
+++ b/src/main/java/cc/mrbird/febs/ai/controller/product/AiProductController.java
@@ -1,6 +1,7 @@
 package cc.mrbird.febs.ai.controller.product;
 
 import cc.mrbird.febs.ai.entity.AiProduct;
+import cc.mrbird.febs.ai.req.AdminMoveChooseInfoDto;
 import cc.mrbird.febs.ai.service.AiProductService;
 import cc.mrbird.febs.common.annotation.ControllerEndpoint;
 import cc.mrbird.febs.common.controller.BaseController;
@@ -25,12 +26,12 @@
 @RequestMapping(value = "/admin/product")
 public class AiProductController extends BaseController {
 
-    private final AiProductService service;
+    private final AiProductService aiProductService;
 
     @GetMapping("list")
     public FebsResponse list(AiProduct dto, QueryRequest request) {
 
-        Map<String, Object> data = getDataTable(service.listInPage(dto, request));
+        Map<String, Object> data = getDataTable(aiProductService.listInPage(dto, request));
         return new FebsResponse().success().data(data);
     }
 
@@ -42,21 +43,21 @@
             @NotNull(message = "{required}") @PathVariable Integer state
     ) {
 
-        return service.changeState(id,type,state);
+        return aiProductService.changeState(id,type,state);
     }
 
     @PostMapping("add")
     @ControllerEndpoint(operation = "新增", exceptionMessage = "操作失败")
     public FebsResponse add(@RequestBody @Valid AiProduct dto) {
 
-        return service.add(dto);
+        return aiProductService.add(dto);
     }
 
     @PostMapping("update")
-    @ControllerEndpoint(operation = "分类-更新", exceptionMessage = "操作失败")
+    @ControllerEndpoint(operation = "更新", exceptionMessage = "操作失败")
     public FebsResponse update(@RequestBody @Valid AiProduct dto) {
 
-        return service.update(dto);
+        return aiProductService.update(dto);
     }
 
     @GetMapping("delete/{id}")
@@ -65,6 +66,14 @@
             @NotNull(message = "{required}") @PathVariable String id
     ) {
 
-        return service.delete(id);
+        return aiProductService.delete(id);
+    }
+
+
+    @PostMapping("pointSet")
+    @ControllerEndpoint(operation = "知识点配置", exceptionMessage = "操作失败")
+    public FebsResponse pointSet(@RequestBody @Valid AdminMoveChooseInfoDto dto) {
+
+        return aiProductService.productSet(dto);
     }
 }
diff --git a/src/main/java/cc/mrbird/febs/ai/controller/product/ViewController.java b/src/main/java/cc/mrbird/febs/ai/controller/product/ViewController.java
index 65e03e8..8b34ca6 100644
--- a/src/main/java/cc/mrbird/febs/ai/controller/product/ViewController.java
+++ b/src/main/java/cc/mrbird/febs/ai/controller/product/ViewController.java
@@ -1,9 +1,19 @@
 package cc.mrbird.febs.ai.controller.product;
 
 import cc.mrbird.febs.ai.entity.AiProduct;
+import cc.mrbird.febs.ai.entity.AiProductPoint;
+import cc.mrbird.febs.ai.entity.AiProductPointLink;
+import cc.mrbird.febs.ai.res.AdminMoveChooseInfoVo;
+import cc.mrbird.febs.ai.service.AiProductPointLinkService;
+import cc.mrbird.febs.ai.service.AiProductPointService;
 import cc.mrbird.febs.ai.service.AiProductService;
 import cc.mrbird.febs.common.entity.FebsConstant;
 import cc.mrbird.febs.common.utils.FebsUtil;
+import cn.hutool.core.collection.CollUtil;
+import cn.hutool.core.util.ObjectUtil;
+import cn.hutool.core.util.StrUtil;
+import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
+import com.baomidou.mybatisplus.core.toolkit.Wrappers;
 import lombok.RequiredArgsConstructor;
 import org.apache.shiro.authz.annotation.RequiresPermissions;
 import org.springframework.stereotype.Controller;
@@ -11,6 +21,12 @@
 import org.springframework.web.bind.annotation.GetMapping;
 import org.springframework.web.bind.annotation.PathVariable;
 import org.springframework.web.bind.annotation.RequestMapping;
+
+import java.util.ArrayList;
+import java.util.HashSet;
+import java.util.List;
+import java.util.Set;
+import java.util.stream.Collectors;
 
 /**
  * @author Administrator
@@ -21,7 +37,9 @@
 public class ViewController {
 
 
-    private final AiProductService service;
+    private final AiProductService aiProductService;
+    private final AiProductPointLinkService aiProductPointLinkService;
+    private final AiProductPointService aiProductPointService;
 
     @GetMapping("list")
     @RequiresPermissions("productList:view")
@@ -40,8 +58,48 @@
     @GetMapping("info/{id}")
     @RequiresPermissions("productList:info")
     public String artInfo(@PathVariable String id, Model model) {
-        AiProduct entity = service.getById(id);
+        AiProduct entity = aiProductService.getById(id);
         model.addAttribute("aiProduct", entity);
         return FebsUtil.view("modules/ai/product/info");
     }
+
+
+
+    @GetMapping("pointSet/{id}")
+    @RequiresPermissions("productList:pointSet")
+    public String pointSet(@PathVariable String id, Model model) {
+        List<AdminMoveChooseInfoVo> vos = new ArrayList<>();
+        Set<String> productIds = new HashSet<>();
+
+        AiProduct entity = aiProductService.getById(id);
+        if(ObjectUtil.isNotNull(entity)){
+            //右侧数据
+            LambdaQueryWrapper<AiProductPointLink> query = Wrappers.lambdaQuery(AiProductPointLink.class);
+            if(StrUtil.isNotEmpty(id)){
+                query.eq(AiProductPointLink::getProductId, id);
+            }
+            List<AiProductPointLink> selectedList = aiProductPointLinkService.selectListByQuery(query);
+            if(CollUtil.isNotEmpty(selectedList)){
+                //stream流操作happyMemberLabelRecords,获取memberId的set集合
+                productIds = selectedList.stream().map(AiProductPointLink::getProductPointId).collect(Collectors.toSet());
+            }
+
+            //左侧数据
+            List<AiProductPoint> allList = aiProductPointService.pointTree();
+            if(CollUtil.isNotEmpty(allList)){
+                //stream流操作mallMembers,生成一个新的List<MallMemberVo>
+                vos = allList.stream().map(AiProductPoint -> {
+                    AdminMoveChooseInfoVo vo = new AdminMoveChooseInfoVo();
+                    vo.setId(AiProductPoint.getId());
+                    vo.setName(AiProductPoint.getTitle());
+                    return vo;
+                }).collect(Collectors.toList());
+            }
+        }
+
+        model.addAttribute("pointAll", vos);
+        model.addAttribute("pointSelected", productIds);
+        model.addAttribute("chooseId", id);
+        return FebsUtil.view("modules/ai/product/pointSet");
+    }
 }
diff --git a/src/main/java/cc/mrbird/febs/ai/controller/productCategory/AiProductCategoryController.java b/src/main/java/cc/mrbird/febs/ai/controller/productCategory/AiProductCategoryController.java
index 79e93bb..c50d176 100644
--- a/src/main/java/cc/mrbird/febs/ai/controller/productCategory/AiProductCategoryController.java
+++ b/src/main/java/cc/mrbird/febs/ai/controller/productCategory/AiProductCategoryController.java
@@ -53,7 +53,7 @@
     }
 
     @PostMapping("update")
-    @ControllerEndpoint(operation = "分类-更新", exceptionMessage = "操作失败")
+    @ControllerEndpoint(operation = "更新", exceptionMessage = "操作失败")
     public FebsResponse update(@RequestBody @Valid AiProductCategory dto) {
 
         return service.update(dto);
diff --git a/src/main/java/cc/mrbird/febs/ai/controller/productPoint/AiProductPointController.java b/src/main/java/cc/mrbird/febs/ai/controller/productPoint/AiProductPointController.java
new file mode 100644
index 0000000..0c53cf9
--- /dev/null
+++ b/src/main/java/cc/mrbird/febs/ai/controller/productPoint/AiProductPointController.java
@@ -0,0 +1,65 @@
+package cc.mrbird.febs.ai.controller.productPoint;
+
+import cc.mrbird.febs.ai.entity.AiProductPoint;
+import cc.mrbird.febs.ai.service.AiProductPointService;
+import cc.mrbird.febs.common.annotation.ControllerEndpoint;
+import cc.mrbird.febs.common.controller.BaseController;
+import cc.mrbird.febs.common.entity.FebsResponse;
+import cc.mrbird.febs.common.entity.QueryRequest;
+import lombok.RequiredArgsConstructor;
+import lombok.extern.slf4j.Slf4j;
+import org.springframework.validation.annotation.Validated;
+import org.springframework.web.bind.annotation.*;
+
+import javax.validation.Valid;
+import javax.validation.constraints.NotNull;
+import java.util.Map;
+
+/**
+ * @author Administrator
+ */
+@Slf4j
+@Validated
+@RestController
+@RequiredArgsConstructor
+@RequestMapping(value = "/admin/productPoint")
+public class AiProductPointController extends BaseController {
+
+    private final AiProductPointService service;
+
+    @GetMapping("list")
+    public FebsResponse list(AiProductPoint dto, QueryRequest request) {
+
+        Map<String, Object> data = getDataTable(service.listInPage(dto, request));
+        return new FebsResponse().success().data(data);
+    }
+
+    @PostMapping("add")
+    @ControllerEndpoint(operation = "新增", exceptionMessage = "操作失败")
+    public FebsResponse add(@RequestBody @Valid AiProductPoint dto) {
+
+        return service.add(dto);
+    }
+
+    @PostMapping("update")
+    @ControllerEndpoint(operation = "更新", exceptionMessage = "操作失败")
+    public FebsResponse update(@RequestBody @Valid AiProductPoint dto) {
+
+        return service.update(dto);
+    }
+
+    @GetMapping("delete/{id}")
+    @ControllerEndpoint(operation = "删除", exceptionMessage = "操作失败")
+    public FebsResponse delete(
+            @NotNull(message = "{required}") @PathVariable String id
+    ) {
+
+        return service.delete(id);
+    }
+
+    @GetMapping(value = "/pointTree")
+    public FebsResponse pointTree() {
+
+        return new FebsResponse().success().data(service.pointTree());
+    }
+}
diff --git a/src/main/java/cc/mrbird/febs/ai/controller/productPoint/ViewController.java b/src/main/java/cc/mrbird/febs/ai/controller/productPoint/ViewController.java
new file mode 100644
index 0000000..201d632
--- /dev/null
+++ b/src/main/java/cc/mrbird/febs/ai/controller/productPoint/ViewController.java
@@ -0,0 +1,47 @@
+package cc.mrbird.febs.ai.controller.productPoint;
+
+import cc.mrbird.febs.ai.entity.AiProductPoint;
+import cc.mrbird.febs.ai.service.AiProductPointService;
+import cc.mrbird.febs.common.entity.FebsConstant;
+import cc.mrbird.febs.common.utils.FebsUtil;
+import lombok.RequiredArgsConstructor;
+import org.apache.shiro.authz.annotation.RequiresPermissions;
+import org.springframework.stereotype.Controller;
+import org.springframework.ui.Model;
+import org.springframework.web.bind.annotation.GetMapping;
+import org.springframework.web.bind.annotation.PathVariable;
+import org.springframework.web.bind.annotation.RequestMapping;
+
+/**
+ * @author Administrator
+ */
+@Controller("AiProductPointView")
+@RequestMapping(FebsConstant.VIEW_PREFIX + "modules/ai/productPoint")
+@RequiredArgsConstructor
+public class ViewController {
+
+
+    private final AiProductPointService service;
+
+    @GetMapping("list")
+    @RequiresPermissions("pointList:view")
+    public String pointList() {
+
+        return FebsUtil.view("modules/ai/productPoint/list");
+    }
+
+    @GetMapping(value = "/add")
+    @RequiresPermissions("pointList:add")
+    public String artAdd() {
+
+        return FebsUtil.view("modules/ai/productPoint/add");
+    }
+
+    @GetMapping("info/{id}")
+    @RequiresPermissions("pointList:info")
+    public String artInfo(@PathVariable String id, Model model) {
+        AiProductPoint entity = service.getById(id);
+        model.addAttribute("aiProductPoint", entity);
+        return FebsUtil.view("modules/ai/productPoint/info");
+    }
+}
diff --git a/src/main/java/cc/mrbird/febs/ai/req/AdminMoveChooseInfoDto.java b/src/main/java/cc/mrbird/febs/ai/req/AdminMoveChooseInfoDto.java
index a1171f2..0c79a18 100644
--- a/src/main/java/cc/mrbird/febs/ai/req/AdminMoveChooseInfoDto.java
+++ b/src/main/java/cc/mrbird/febs/ai/req/AdminMoveChooseInfoDto.java
@@ -7,7 +7,7 @@
 @Data
 public class AdminMoveChooseInfoDto {
 
-    private String roleId;
+    private String chooseId;
 
     private List<String> chooseIds;
 }
diff --git a/src/main/java/cc/mrbird/febs/ai/service/AiProductPointLinkService.java b/src/main/java/cc/mrbird/febs/ai/service/AiProductPointLinkService.java
index c312695..9ab09d4 100644
--- a/src/main/java/cc/mrbird/febs/ai/service/AiProductPointLinkService.java
+++ b/src/main/java/cc/mrbird/febs/ai/service/AiProductPointLinkService.java
@@ -1,10 +1,21 @@
 package cc.mrbird.febs.ai.service;
 
 import cc.mrbird.febs.ai.entity.AiProductPointLink;
+import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
 import com.baomidou.mybatisplus.extension.service.IService;
+
+import java.util.List;
 
 /**
  * @author Administrator
  */
 public interface AiProductPointLinkService extends IService<AiProductPointLink> {
+
+    void deleteByProductPointId(String id);
+
+    List<AiProductPointLink> selectListByPointId(String pointId);
+
+    List<AiProductPointLink> selectListByQuery(LambdaQueryWrapper<AiProductPointLink> query);
+
+    void deleteByQuery(LambdaQueryWrapper<AiProductPointLink> query);
 }
\ No newline at end of file
diff --git a/src/main/java/cc/mrbird/febs/ai/service/AiProductPointService.java b/src/main/java/cc/mrbird/febs/ai/service/AiProductPointService.java
index 54d38b1..6734e2c 100644
--- a/src/main/java/cc/mrbird/febs/ai/service/AiProductPointService.java
+++ b/src/main/java/cc/mrbird/febs/ai/service/AiProductPointService.java
@@ -1,6 +1,10 @@
 package cc.mrbird.febs.ai.service;
 
+import cc.mrbird.febs.ai.entity.AiProduct;
 import cc.mrbird.febs.ai.entity.AiProductPoint;
+import cc.mrbird.febs.common.entity.FebsResponse;
+import cc.mrbird.febs.common.entity.QueryRequest;
+import com.baomidou.mybatisplus.core.metadata.IPage;
 import com.baomidou.mybatisplus.extension.service.IService;
 import java.util.List;
 
@@ -20,4 +24,14 @@
     AiProductPoint getById(String id);
 
 
+    IPage<AiProductPoint> listInPage(AiProductPoint dto, QueryRequest request);
+
+    FebsResponse add(AiProductPoint dto);
+
+    FebsResponse update(AiProductPoint dto);
+
+    FebsResponse delete(String id);
+
+    List<AiProductPoint> pointTree();
+
 }
diff --git a/src/main/java/cc/mrbird/febs/ai/service/AiProductService.java b/src/main/java/cc/mrbird/febs/ai/service/AiProductService.java
index 45bf65b..7b20493 100644
--- a/src/main/java/cc/mrbird/febs/ai/service/AiProductService.java
+++ b/src/main/java/cc/mrbird/febs/ai/service/AiProductService.java
@@ -1,6 +1,7 @@
 package cc.mrbird.febs.ai.service;
 
 import cc.mrbird.febs.ai.entity.AiProduct;
+import cc.mrbird.febs.ai.req.AdminMoveChooseInfoDto;
 import cc.mrbird.febs.common.entity.FebsResponse;
 import cc.mrbird.febs.common.entity.QueryRequest;
 import cc.mrbird.febs.mall.entity.ClothesArt;
@@ -34,4 +35,6 @@
     FebsResponse delete(String id);
 
     List<AiProduct> selectList();
+
+    FebsResponse productSet(AdminMoveChooseInfoDto dto);
 }
diff --git a/src/main/java/cc/mrbird/febs/ai/service/impl/AiMemberRoleServiceImpl.java b/src/main/java/cc/mrbird/febs/ai/service/impl/AiMemberRoleServiceImpl.java
index 97ea8dc..688e684 100644
--- a/src/main/java/cc/mrbird/febs/ai/service/impl/AiMemberRoleServiceImpl.java
+++ b/src/main/java/cc/mrbird/febs/ai/service/impl/AiMemberRoleServiceImpl.java
@@ -10,7 +10,6 @@
 import cc.mrbird.febs.ai.util.UUID;
 import cc.mrbird.febs.common.entity.FebsResponse;
 import cc.mrbird.febs.common.entity.QueryRequest;
-import cc.mrbird.febs.mall.entity.ClothesTypeArt;
 import cn.hutool.core.collection.CollUtil;
 import cn.hutool.core.util.ObjectUtil;
 import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
@@ -118,20 +117,21 @@
 
     @Override
     public FebsResponse productSet(AdminMoveChooseInfoDto dto) {
-        String roleId = dto.getRoleId();
+        String chooseId = dto.getChooseId();
         List<String> chooseIds = dto.getChooseIds();
-        AiMemberRole aiMemberRole = this.getById(roleId);
+        AiMemberRole aiMemberRole = this.getById(chooseId);
         if (ObjectUtil.isNotNull(aiMemberRole)) {
             aiMemberRoleProductService.deleteByQuery(
                     Wrappers.lambdaQuery(AiMemberRoleProduct.class)
-                            .eq(AiMemberRoleProduct::getRoleId,roleId)
+                            .eq(AiMemberRoleProduct::getRoleId,chooseId)
             );
             if(CollUtil.isNotEmpty(chooseIds)){
                 Date createdTime = new Date();
-                for (String chooseId : chooseIds){
+                for (String item : chooseIds){
                     AiMemberRoleProduct entity = new AiMemberRoleProduct();
-                    entity.setRoleId(roleId);
-                    entity.setProductId(chooseId);
+                    entity.setId(UUID.getSimpleUUIDString());
+                    entity.setRoleId(chooseId);
+                    entity.setProductId(item);
                     entity.setCreatedTime(createdTime);
                     aiMemberRoleProductService.getBaseMapper().insert(entity);
                 }
diff --git a/src/main/java/cc/mrbird/febs/ai/service/impl/AiProductPointLinkServiceImpl.java b/src/main/java/cc/mrbird/febs/ai/service/impl/AiProductPointLinkServiceImpl.java
index 9818302..682fcec 100644
--- a/src/main/java/cc/mrbird/febs/ai/service/impl/AiProductPointLinkServiceImpl.java
+++ b/src/main/java/cc/mrbird/febs/ai/service/impl/AiProductPointLinkServiceImpl.java
@@ -3,10 +3,18 @@
 import cc.mrbird.febs.ai.entity.AiProductPointLink;
 import cc.mrbird.febs.ai.mapper.AiProductPointLinkMapper;
 import cc.mrbird.febs.ai.service.AiProductPointLinkService;
+import cc.mrbird.febs.ai.service.AiProductPointService;
+import cn.hutool.core.collection.CollUtil;
+import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
+import com.baomidou.mybatisplus.core.toolkit.Wrappers;
 import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
 import lombok.RequiredArgsConstructor;
 import org.springframework.stereotype.Service;
 import org.springframework.transaction.annotation.Transactional;
+
+import java.util.List;
+import java.util.Set;
+import java.util.stream.Collectors;
 
 /**
  * @author Administrator
@@ -16,4 +24,31 @@
 @Transactional
 public class AiProductPointLinkServiceImpl extends ServiceImpl<AiProductPointLinkMapper, AiProductPointLink> implements AiProductPointLinkService {
 
+    private final AiProductPointLinkMapper aiProductPointLinkMapper;
+    @Override
+    public void deleteByProductPointId(String id) {
+        List<AiProductPointLink> aiProductPointLinks = this.selectListByPointId(id);
+        if(CollUtil.isNotEmpty(aiProductPointLinks)){
+            Set<String> collect = aiProductPointLinks.stream().map(AiProductPointLink::getId).collect(Collectors.toSet());
+            aiProductPointLinkMapper.deleteBatchIds(collect);
+        }
+    }
+
+    @Override
+    public List<AiProductPointLink> selectListByPointId(String pointId) {
+        LambdaQueryWrapper<AiProductPointLink> query = Wrappers.lambdaQuery(AiProductPointLink.class);
+        query.eq(AiProductPointLink::getProductPointId, pointId);
+        List<AiProductPointLink> aiProductPointLinks = aiProductPointLinkMapper.selectList(query);
+        return aiProductPointLinks;
+    }
+
+    @Override
+    public List<AiProductPointLink> selectListByQuery(LambdaQueryWrapper<AiProductPointLink> query) {
+        return aiProductPointLinkMapper.selectList( query);
+    }
+
+    @Override
+    public void deleteByQuery(LambdaQueryWrapper<AiProductPointLink> query) {
+        aiProductPointLinkMapper.delete(query);
+    }
 }
diff --git a/src/main/java/cc/mrbird/febs/ai/service/impl/AiProductPointServiceImpl.java b/src/main/java/cc/mrbird/febs/ai/service/impl/AiProductPointServiceImpl.java
index f76b1d3..8bead4e 100644
--- a/src/main/java/cc/mrbird/febs/ai/service/impl/AiProductPointServiceImpl.java
+++ b/src/main/java/cc/mrbird/febs/ai/service/impl/AiProductPointServiceImpl.java
@@ -1,15 +1,24 @@
 package cc.mrbird.febs.ai.service.impl;
 
+import cc.mrbird.febs.ai.entity.AiProduct;
 import cc.mrbird.febs.ai.entity.AiProductPoint;
 import cc.mrbird.febs.ai.mapper.AiProductPointMapper;
+import cc.mrbird.febs.ai.service.AiProductPointLinkService;
 import cc.mrbird.febs.ai.service.AiProductPointService;
+import cc.mrbird.febs.ai.util.UUID;
+import cc.mrbird.febs.common.entity.FebsResponse;
+import cc.mrbird.febs.common.entity.QueryRequest;
+import cn.hutool.core.util.ObjectUtil;
 import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
+import com.baomidou.mybatisplus.core.metadata.IPage;
+import com.baomidou.mybatisplus.core.toolkit.Wrappers;
+import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
 import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
 import lombok.RequiredArgsConstructor;
 import lombok.extern.slf4j.Slf4j;
 import org.springframework.stereotype.Service;
-import org.springframework.transaction.annotation.Transactional;
 
+import java.util.Date;
 import java.util.List;
 
 /**
@@ -24,11 +33,72 @@
 public class AiProductPointServiceImpl extends ServiceImpl<AiProductPointMapper, AiProductPoint> implements AiProductPointService {
 
     private final AiProductPointMapper aiProductPointMapper;
+    private final AiProductPointLinkService aiProductPointLinkService;
 
     @Override
     public AiProductPoint getById(String id) {
         return aiProductPointMapper.selectById(id);
     }
 
+    @Override
+    public IPage<AiProductPoint> listInPage(AiProductPoint dto, QueryRequest request) {
+        Page<AiProductPoint> page = new Page<>(request.getPageNum(), request.getPageSize());
+        LambdaQueryWrapper<AiProductPoint> query = Wrappers.lambdaQuery(AiProductPoint.class);
+        Page<AiProductPoint> pages = aiProductPointMapper.selectPage(page, query);
+        return pages;
+    }
+
+    @Override
+    public FebsResponse add(AiProductPoint dto) {
+        AiProductPoint entity = new AiProductPoint();
+        entity.setId(UUID.getSimpleUUIDString());
+        entity.setCompanyId(dto.getCompanyId());
+        entity.setIsNormal(dto.getIsNormal() );
+        entity.setFinderUserName(dto.getFinderUserName());
+        entity.setFeedId(dto.getFeedId());
+        entity.setTitle(dto.getTitle());
+        entity.setDescription(dto.getDescription());
+        entity.setCreatedTime(new Date());
+        this.save(entity);
+        return new FebsResponse().success().message("操作成功");
+    }
+
+    @Override
+    public FebsResponse update(AiProductPoint dto) {
+        String id = dto.getId();
+        AiProductPoint entity = this.getById(id);
+        if (ObjectUtil.isNotNull( entity)){
+            this.update(null,
+                    Wrappers.lambdaUpdate(AiProductPoint.class)
+                            .set(AiProductPoint::getIsNormal, dto.getIsNormal())
+                            .set(AiProductPoint::getFinderUserName, dto.getFinderUserName())
+                            .set(AiProductPoint::getFeedId, dto.getFeedId())
+                            .set(AiProductPoint::getTitle, dto.getTitle())
+                            .set(AiProductPoint::getDescription, dto.getDescription())
+                            .set(AiProductPoint::getUpdatedTime, new Date())
+                            .eq(AiProductPoint::getId, id)
+            );
+        }
+        return new FebsResponse().success().message("操作成功");
+    }
+
+    @Override
+    public FebsResponse delete(String id) {
+        AiProductPoint entity = this.getById(id);
+        if(ObjectUtil.isNotNull( entity)){
+            // 删除
+            aiProductPointMapper.deleteById( id);
+            aiProductPointLinkService.deleteByProductPointId(id);
+        }
+        return new FebsResponse().success().message("操作成功");
+    }
+
+    @Override
+    public List<AiProductPoint> pointTree() {
+
+        return aiProductPointMapper.selectList(null);
+    }
+
+
 
 }
diff --git a/src/main/java/cc/mrbird/febs/ai/service/impl/AiProductServiceImpl.java b/src/main/java/cc/mrbird/febs/ai/service/impl/AiProductServiceImpl.java
index a1f0f5f..a67e99d 100644
--- a/src/main/java/cc/mrbird/febs/ai/service/impl/AiProductServiceImpl.java
+++ b/src/main/java/cc/mrbird/febs/ai/service/impl/AiProductServiceImpl.java
@@ -1,14 +1,14 @@
 package cc.mrbird.febs.ai.service.impl;
 
-import cc.mrbird.febs.ai.entity.AiProduct;
-import cc.mrbird.febs.ai.entity.AiProductCategory;
+import cc.mrbird.febs.ai.entity.*;
 import cc.mrbird.febs.ai.mapper.AiProductMapper;
+import cc.mrbird.febs.ai.req.AdminMoveChooseInfoDto;
 import cc.mrbird.febs.ai.service.AiProductCategoryService;
+import cc.mrbird.febs.ai.service.AiProductPointLinkService;
 import cc.mrbird.febs.ai.service.AiProductService;
 import cc.mrbird.febs.ai.util.UUID;
 import cc.mrbird.febs.common.entity.FebsResponse;
 import cc.mrbird.febs.common.entity.QueryRequest;
-import cc.mrbird.febs.mall.entity.ClothesArt;
 import cn.hutool.core.collection.CollUtil;
 import cn.hutool.core.util.ObjectUtil;
 import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
@@ -39,6 +39,7 @@
 
     private final AiProductMapper aiProductMapper;
     private final AiProductCategoryService aiProductCategoryService;
+    private final AiProductPointLinkService aiProductPointLinkService;
 
     @Override
     public AiProduct getById(String id) {
@@ -157,6 +158,35 @@
 
     @Override
     public List<AiProduct> selectList() {
-        return aiProductMapper.selectList( null);
+        return aiProductMapper.selectList(
+                Wrappers.lambdaQuery(AiProduct.class)
+                .ne(AiProduct::getState, 2)
+        );
+    }
+
+    @Override
+    public FebsResponse productSet(AdminMoveChooseInfoDto dto) {
+
+        String chooseId = dto.getChooseId();
+        List<String> chooseIds = dto.getChooseIds();
+        AiProduct aiProduct = this.getById(chooseId);
+        if (ObjectUtil.isNotNull(aiProduct)) {
+            aiProductPointLinkService.deleteByQuery(
+                    Wrappers.lambdaQuery(AiProductPointLink.class)
+                            .eq(AiProductPointLink::getProductId,chooseId)
+            );
+            if(CollUtil.isNotEmpty(chooseIds)){
+                Date createdTime = new Date();
+                for (String item : chooseIds){
+                    AiProductPointLink entity = new AiProductPointLink();
+                    entity.setId(UUID.getSimpleUUIDString());
+                    entity.setProductId(chooseId);
+                    entity.setProductPointId(item);
+                    entity.setCreatedTime(createdTime);
+                    aiProductPointLinkService.getBaseMapper().insert(entity);
+                }
+            }
+        }
+        return new FebsResponse().success().message("操作成功");
     }
 }
diff --git a/src/main/resources/templates/febs/views/modules/ai/memberRole/productSet.html b/src/main/resources/templates/febs/views/modules/ai/memberRole/productSet.html
index f9b3857..7e412a3 100644
--- a/src/main/resources/templates/febs/views/modules/ai/memberRole/productSet.html
+++ b/src/main/resources/templates/febs/views/modules/ai/memberRole/productSet.html
@@ -8,7 +8,7 @@
                             <li class="layui-this">产品配置</li>
                         </ul>
                         <div class="layui-tab-content">
-                            <input type="text" name="roleId"
+                            <input type="text" name="chooseId"
                                    placeholder="" autoComplete="off" class="layui-input febs-hide">
                             <div class="layui-tab-item layui-show">
                                 <div class="layui-form-item">
@@ -35,7 +35,7 @@
             transfer = layui.transfer,
             productTypeAll = [[${productAll}]],
             productTypeChoose = [[${productSelected}]],
-            roleId = [[${roleId}]],
+            chooseId = [[${chooseId}]],
             $view = $('#product-set'),
 
             $productSetMoveQuery = $view.find('#productSetMoveQuery')
@@ -62,9 +62,9 @@
         function initProductTypeSet() {
             console.log("productTypeAll:", productTypeAll); // 调试信息
             console.log("productTypeChoose:", productTypeChoose); // 调试信息
-            console.log("roleId:", roleId); // 调试信息
+            console.log("chooseId:", chooseId); // 调试信息
             form.val("product-type-set-form", {
-                "roleId": roleId,
+                "chooseId": chooseId,
             });
             // 转换数据格式(假设接口返回的数据结构需要处理)
             var dataLeft = productTypeAll.map(function(item){
@@ -99,7 +99,7 @@
                 return item.value;
             });
             data.field.chooseIds = productIdList;
-            data.field.roleId = roleId;
+            data.field.chooseId = chooseId;
             $.ajax({
                 'url':ctx + 'admin/memberRole/productSet',
                 'type':'post',
diff --git a/src/main/resources/templates/febs/views/modules/ai/product/list.html b/src/main/resources/templates/febs/views/modules/ai/product/list.html
index 581d9df..f9a57e5 100644
--- a/src/main/resources/templates/febs/views/modules/ai/product/list.html
+++ b/src/main/resources/templates/febs/views/modules/ai/product/list.html
@@ -51,13 +51,14 @@
 
 <script type="text/html" id="productToolbar">
     <div class="layui-btn-container">
-        <button class="layui-btn layui-btn-sm layui-btn-primary febs-button-green-plain" type="button" shiro:hasPermission="categoryList:add" lay-event="productAdd">新增</button>
+        <button class="layui-btn layui-btn-sm layui-btn-primary febs-button-green-plain" shiro:hasPermission="productList:add" lay-event="productAdd">新增</button>
+        <button class="layui-btn layui-btn-sm layui-btn-primary febs-button-green-plain" shiro:hasPermission="productList:pointSet" lay-event="pointSet">知识点配置</button>
     </div>
 </script>
 
 <script type="text/html" id="productOption">
-    <button class="layui-btn layui-btn-normal layui-btn-sm" type="button" shiro:hasPermission="categoryList:info" lay-event="productInfoEvent">编辑</button>
-    <button class="layui-btn layui-btn-danger layui-btn-sm" type="button" shiro:hasPermission="categoryList:info" lay-event="productDeleteEvent">删除</button>
+    <button class="layui-btn layui-btn-normal layui-btn-sm" type="button" shiro:hasPermission="productList:info" lay-event="productInfoEvent">编辑</button>
+    <button class="layui-btn layui-btn-danger layui-btn-sm" type="button" shiro:hasPermission="productList:info" lay-event="productDeleteEvent">删除</button>
 </script>
 
 
@@ -245,17 +246,17 @@
                 });
             }
 
-            if (layEvent === 'productSet') {
+            if (layEvent === 'pointSet') {
                 var checkData = table.checkStatus('productTable').data;
                 if (checkData.length > 1 || checkData.length === 0) {
                     febs.alert.warn('每次操作只能操作一行数据');
                     return;
                 }
-                febs.modal.open('工艺配置', 'modules/clothesType/productSet/' + checkData[0].id, {
+                febs.modal.open('知识点配置', 'modules/ai/product/pointSet/' + checkData[0].id, {
                     btn: ['提交', '取消'],
                     area:['100%','100%'],
                     yes: function (index, layero) {
-                        $('#art-set').find('#submit').trigger('click');
+                        $('#point-set').find('#submit').trigger('click');
                     },
                     btn2: function () {
                         layer.closeAll();
diff --git a/src/main/resources/templates/febs/views/modules/ai/product/pointSet.html b/src/main/resources/templates/febs/views/modules/ai/product/pointSet.html
new file mode 100644
index 0000000..798e7f6
--- /dev/null
+++ b/src/main/resources/templates/febs/views/modules/ai/product/pointSet.html
@@ -0,0 +1,128 @@
+<div class="layui-fluid layui-anim febs-anim" id="point-set" lay-title="知识点配置">
+    <div class="layui-row febs-container">
+        <div class="layui-col-md12">
+            <div class="layui-fluid" id="point-type-set">
+                <form class="layui-form" action="" lay-filter="point-type-set-form">
+                    <div class="layui-tab layui-tab-brief" lay-filter="docDemoTabBrief">
+                        <ul class="layui-tab-title">
+                            <li class="layui-this">知识点配置</li>
+                        </ul>
+                        <div class="layui-tab-content">
+                            <input type="text" name="chooseId"
+                                   placeholder="" autoComplete="off" class="layui-input febs-hide">
+                            <div class="layui-tab-item layui-show">
+                                <div class="layui-form-item">
+                                    <div id="pointSetMove"></div>
+                                </div>
+                            </div>
+                        </div>
+                    </div>
+                    <div class="layui-form-item febs-hide">
+                        <button class="layui-btn" lay-submit="" lay-filter="point-type-set-form-submit" id="submit">保存</button>
+                    </div>
+                </form>
+            </div>
+        </div>
+    </div>
+</div>
+
+<script data-th-inline="javascript">
+    layui.use(['febs','form', 'transfer'], function () {
+        var $ = layui.jquery,
+            febs = layui.febs,
+            layer = layui.layer,
+            form = layui.form,
+            transfer = layui.transfer,
+            pointTypeAll = [[${pointAll}]],
+            pointTypeChoose = [[${pointSelected}]],
+            chooseId = [[${chooseId}]],
+            $view = $('#point-set'),
+
+            $pointSetMoveQuery = $view.find('#pointSetMoveQuery')
+        ;
+
+        // 查询按钮
+        $pointSetMoveQuery.on('click', function () {
+            console.log(transfer.getData('pointSetMove-set'))
+
+            let data1 = transfer.getData('pointSetMove-set');
+            //获取data1中的value,返回一个数组
+            let pointIdList = data1.map(function(item){
+                return item.value;
+            });
+
+            console.log(pointIdList)
+        });
+
+
+        form.render();
+
+        initpointTypeSet();
+
+        function initpointTypeSet() {
+            console.log("pointTypeAll:", pointTypeAll); // 调试信息
+            console.log("pointTypeChoose:", pointTypeChoose); // 调试信息
+            console.log("chooseId:", chooseId); // 调试信息
+            form.val("point-type-set-form", {
+                "chooseId": chooseId,
+            });
+            // 转换数据格式(假设接口返回的数据结构需要处理)
+            var dataLeft = pointTypeAll.map(function(item){
+                return {
+                    value: item.id,  // 值字段
+                    title: item.name // 显示文本
+                }
+            });
+            var dataRight = pointTypeChoose.map(function(item){
+                return {
+                    value: item,  // 值字段
+                }
+            });
+
+            // 渲染穿梭框
+            transfer.render({
+                elem: '#pointSetMove',
+                data: dataLeft,
+                id: 'pointSetMove-set', // 唯一标识
+                title: ['待选择列表', '已选择列表'],
+                width: 300,
+                height: 400,
+                showSearch: true,
+                value: pointTypeChoose,
+            });
+        }
+
+        form.on('submit(point-type-set-form-submit)', function (data) {
+            let data1 = transfer.getData('pointSetMove-set');
+            //获取data1中的value,返回一个数组
+            let pointIdList = data1.map(function(item){
+                return item.value;
+            });
+            data.field.chooseIds = pointIdList;
+            data.field.chooseId = chooseId;
+            $.ajax({
+                'url':ctx + 'admin/product/pointSet',
+                'type':'post',
+                'dataType':'json',
+                'headers' : {'Content-Type' : 'application/json;charset=utf-8'}, //接口json格式
+                'traditional': true,//ajax传递数组必须添加属性
+                'data':JSON.stringify(data.field),
+                'success':function (data) {
+                    if(data.code==200){
+                        layer.closeAll();
+                        febs.alert.success(data.message);
+                        $('#febs-type').find('#query').click();
+                    }else{
+                        febs.alert.warn(data.message);
+                    }
+                },
+                'error':function () {
+                    febs.alert.warn('服务器繁忙');
+                }
+            })
+            return false;
+        });
+
+
+    });
+</script>
\ No newline at end of file
diff --git a/src/main/resources/templates/febs/views/modules/ai/productPoint/add.html b/src/main/resources/templates/febs/views/modules/ai/productPoint/add.html
new file mode 100644
index 0000000..e27e06d
--- /dev/null
+++ b/src/main/resources/templates/febs/views/modules/ai/productPoint/add.html
@@ -0,0 +1,159 @@
+<div class="layui-fluid layui-anim febs-anim" id="febs-productPoint-add" lay-title="新增">
+    <div class="layui-row febs-container">
+        <div class="layui-col-md12">
+            <div class="layui-fluid" id="productPoint-add">
+                <form class="layui-form" action="" lay-filter="productPoint-add-form">
+                    <div class="layui-tab layui-tab-brief" lay-filter="docDemoTabBrief">
+                        <ul class="layui-tab-title">
+                            <li class="layui-this">基础信息</li>
+                        </ul>
+                        <div class="layui-tab-content">
+                            <div class="layui-tab-item layui-show">
+                                <div class="layui-row layui-col-space10 layui-form-item">
+                                    <div class="layui-col-lg6">
+                                        <label class="layui-form-label febs-form-item-require">类型:</label>
+                                        <div class="layui-input-block">
+                                            <select name="isNormal" class="point-type" lay-filter="point-type-select">
+                                                <option value="1">普通内容</option>
+                                                <option value="2">视频号内容</option>
+                                            </select>
+                                        </div>
+                                    </div>
+                                </div>
+
+                                <blockquote class="layui-elem-quote blue-border febs-hide tc-set">视频号信息</blockquote>
+                                <div class="layui-form-item febs-hide tc-set">
+                                    <div class="layui-col-lg6">
+                                        <label class="layui-form-label">视频号 id:</label>
+                                        <div class="layui-input-block">
+                                            <input type="text" name="finderUserName" placeholder="请输入" autocomplete="off" class="layui-input">
+                                            <div class="layui-form-mid layui-word-aux">视频号 id,以“sph”开头的id,可在视频号助手获取。</div>
+                                        </div>
+                                    </div>
+                                    <div class="layui-col-lg6">
+                                        <label class="layui-form-label">视频 feedId:</label>
+                                        <div class="layui-input-block">
+                                            <input type="text" name="feedId" placeholder="请输入" autocomplete="off" class="layui-input">
+                                        </div>
+                                    </div>
+                                </div>
+                                <div class="layui-row layui-col-space10 layui-form-item">
+                                    <div class="layui-col-lg6">
+                                        <label class="layui-form-label febs-form-item-require">标题:</label>
+                                        <div class="layui-input-block">
+                                            <input type="text" name="title" lay-verify="required"
+                                                   placeholder="" autocomplete="off" class="layui-input">
+                                        </div>
+                                    </div>
+                                </div>
+
+
+                                <div class="layui-form-item">
+                                    <label class="layui-form-label febs-form-item-require">详情:</label>
+                                    <div class="layui-input-block">
+                                        <div style="border: 1px solid #ccc;">
+                                            <div id="product-point-toolbar-container" class="toolbar"></div>
+                                            <div id="product-point-text-container" class="text" style="height: 450px;"></div>
+                                        </div>
+                                    </div>
+                                </div>
+
+
+
+                            </div>
+                        </div>
+                    </div>
+
+                    <div class="layui-form-item febs-hide">
+                        <button class="layui-btn" lay-submit="" lay-filter="productPoint-add-form-submit" id="submit">保存</button>
+                    </div>
+                </form>
+            </div>
+        </div>
+    </div>
+</div>
+
+<!-- 表格操作栏 end -->
+<script data-th-inline="javascript">
+    layui.use(['febs', 'form', 'formSelects', 'validate', 'treeSelect', 'eleTree','dropdown', 'laydate', 'layedit', 'upload', 'element', 'table', 'xmSelect','jquery'], function () {
+        var $ = layui.jquery,
+            febs = layui.febs,
+            layer = layui.layer,
+            table = layui.table,
+            formSelects = layui.formSelects,
+            treeSelect = layui.treeSelect,
+            form = layui.form,
+            laydate = layui.laydate,
+            eleTree = layui.eleTree,
+            $view = $('#productPoint-add'),
+            layedit = layui.layedit,
+            upload = layui.upload,
+            validate = layui.validate,
+            element = layui.element;
+
+        form.render();
+
+        const E = window.wangEditor;
+        const editor = new E('#product-point-toolbar-container', '#product-point-text-container'); // 传入两个元素
+        editor.config.showLinkImg = false;
+        editor.config.uploadFileName = 'file';
+        editor.config.customUploadImg = function (files, insertImgFn) {
+            // files 是 input 中选中的文件列表
+            // insertImgFn 是获取图片 url 后,插入到编辑器的方法
+            // 上传图片,返回结果,将图片插入到编辑器中
+            for (let i = 0; i < files.length; i++){
+                var form = new FormData();
+                form.append("file", files[0]);
+                $.ajax({
+                    url:'/admin/goods/uploadFileBase64',
+                    type: "post",
+                    processData: false,
+                    contentType: false,
+                    data: form,
+                    dataType: 'json',
+                    success(res) {
+                        // 上传代码返回结果之后,将图片插入到编辑器中
+                        insertImgFn(res.data.src, res.data.title, '')
+                    }
+                })
+            }
+        };
+        editor.create();
+
+        form.on('select(point-type-select)', function(data){
+            $('.tc-set').each(function() {
+                if (data.value == 2) {
+                    $(this).show();
+                } else {
+                    $(this).hide();
+                }
+            })
+        });
+
+        form.on('submit(productPoint-add-form-submit)', function (data) {
+            data.field.description = editor.txt.html();
+            $.ajax({
+                'url':ctx + 'admin/productPoint/add',
+                'type':'post',
+                'dataType':'json',
+                'headers' : {'Content-Type' : 'application/json;charset=utf-8'}, //接口json格式
+                'traditional': true,//ajax传递数组必须添加属性
+                'data':JSON.stringify(data.field),
+                'success':function (data) {
+                    if(data.code==200){
+                        layer.closeAll();
+                        febs.alert.success(data.message);
+                        $('#febs-productPoint').find('#query').click();
+                    }else{
+                        febs.alert.warn(data.message);
+                    }
+                },
+                'error':function () {
+                    febs.alert.warn('服务器繁忙');
+                }
+            })
+            return false;
+        });
+
+    });
+</script>
diff --git a/src/main/resources/templates/febs/views/modules/ai/productPoint/info.html b/src/main/resources/templates/febs/views/modules/ai/productPoint/info.html
new file mode 100644
index 0000000..0d86d8b
--- /dev/null
+++ b/src/main/resources/templates/febs/views/modules/ai/productPoint/info.html
@@ -0,0 +1,186 @@
+<div class="layui-fluid layui-anim febs-anim" id="febs-productPoint-Info" lay-title="编辑">
+    <div class="layui-row febs-container">
+        <div class="layui-col-md12">
+            <div class="layui-fluid" id="productPoint-info">
+                <form class="layui-form" action="" lay-filter="productPoint-info-form">
+                    <div class="layui-tab layui-tab-brief" lay-filter="docDemoTabBrief">
+                        <ul class="layui-tab-title">
+                            <li class="layui-this">基础信息</li>
+                        </ul>
+                        <div class="layui-tab-content">
+                            <input type="text" name="id"
+                                   placeholder="" autoComplete="off" class="layui-input febs-hide">
+                            <div class="layui-tab-item layui-show">
+
+                                <div class="layui-row layui-col-space10 layui-form-item">
+                                    <div class="layui-col-lg6">
+                                        <label class="layui-form-label febs-form-item-require">类型:</label>
+                                        <div class="layui-input-block">
+                                            <select name="isNormal" class="point-type" lay-filter="point-type-select">
+                                                <option value="1">普通内容</option>
+                                                <option value="2">视频号内容</option>
+                                            </select>
+                                        </div>
+                                    </div>
+                                </div>
+
+                                <blockquote class="layui-elem-quote blue-border febs-hide tc-set">视频号信息</blockquote>
+                                <div class="layui-form-item febs-hide tc-set">
+                                    <div class="layui-col-lg6">
+                                        <label class="layui-form-label">视频号 id:</label>
+                                        <div class="layui-input-block">
+                                            <input type="text" name="finderUserName" placeholder="请输入" autocomplete="off" class="layui-input">
+                                            <div class="layui-form-mid layui-word-aux">视频号 id,以“sph”开头的id,可在视频号助手获取。</div>
+                                        </div>
+                                    </div>
+                                    <div class="layui-col-lg6">
+                                        <label class="layui-form-label">视频 feedId:</label>
+                                        <div class="layui-input-block">
+                                            <input type="text" name="feedId" placeholder="请输入" autocomplete="off" class="layui-input">
+                                        </div>
+                                    </div>
+                                </div>
+                                <div class="layui-row layui-col-space10 layui-form-item">
+                                    <div class="layui-col-lg6">
+                                        <label class="layui-form-label febs-form-item-require">标题:</label>
+                                        <div class="layui-input-block">
+                                            <input type="text" name="title" lay-verify="required"
+                                                   placeholder="" autocomplete="off" class="layui-input">
+                                        </div>
+                                    </div>
+                                </div>
+
+                                <div class="layui-form-item">
+                                    <label class="layui-form-label febs-form-item-require">详情:</label>
+                                    <div class="layui-input-block">
+                                        <div style="border: 1px solid #ccc;">
+                                            <div id="product-point-toolbar-container" class="toolbar"></div>
+                                            <div id="product-point-text-container" class="text" style="height: 450px;"></div>
+                                        </div>
+                                    </div>
+                                </div>
+
+                            </div>
+                        </div>
+                    </div>
+                    <div class="layui-form-item febs-hide">
+                        <button class="layui-btn" lay-submit="" lay-filter="productPoint-info-form-submit" id="submit">保存</button>
+                    </div>
+                </form>
+            </div>
+        </div>
+    </div>
+</div>
+<style>
+    .blue-border {
+        border-left-color: #2db7f5;
+        font-size: 18px;
+    }
+    .layui-table-cell {
+        height:auto;
+    }
+    .layui-upload-list {
+        margin: 0 !important;
+    }
+    .multi-images {
+        margin: 0 5px !important;
+    }
+</style>
+<!-- 表格操作栏 end -->
+<script data-th-inline="javascript">
+    layui.use(['febs', 'form', 'formSelects', 'validate', 'treeSelect', 'eleTree','dropdown', 'laydate', 'layedit', 'upload', 'element', 'table', 'xmSelect','jquery'], function () {
+        var $ = layui.jquery,
+            febs = layui.febs,
+            layer = layui.layer,
+            table = layui.table,
+            form = layui.form,
+            $view = $('#productPoint-info'),
+            aiProductPoint = [[${aiProductPoint}]],
+            upload = layui.upload,
+            validate = layui.validate;
+
+        form.render();
+
+        const E = window.wangEditor;
+        const editor = new E('#product-point-toolbar-container', '#product-point-text-container'); // 传入两个元素
+        editor.config.showLinkImg = false;
+        editor.config.uploadFileName = 'file';
+        editor.config.customUploadImg = function (files, insertImgFn) {
+            // files 是 input 中选中的文件列表
+            // insertImgFn 是获取图片 url 后,插入到编辑器的方法
+            // 上传图片,返回结果,将图片插入到编辑器中
+            for (let i = 0; i < files.length; i++){
+                var form = new FormData();
+                form.append("file", files[0]);
+                $.ajax({
+                    url:'/admin/goods/uploadFileBase64',
+                    type: "post",
+                    processData: false,
+                    contentType: false,
+                    data: form,
+                    dataType: 'json',
+                    success(res) {
+                        // 上传代码返回结果之后,将图片插入到编辑器中
+                        insertImgFn(res.data.src, res.data.title, '')
+                    }
+                })
+            }
+        };
+        editor.create();
+
+        form.on('select(point-type-select)', function(data){
+            $('.tc-set').each(function() {
+                if (data.value == 2) {
+                    $(this).show();
+                } else {
+                    $(this).hide();
+                }
+            })
+        });
+
+        setTimeout(() => {
+            initProductPointInfo();
+        }, 500);
+        function initProductPointInfo() {
+            form.val("productPoint-info-form", {
+                "id": aiProductPoint.id,
+                "title": aiProductPoint.title,
+                "isNormal": aiProductPoint.isNormal,
+                "finderUserName": aiProductPoint.finderUserName,
+                "feedId": aiProductPoint.feedId,
+            });
+
+            editor.txt.html(aiProductPoint.description);
+
+            if (aiProductPoint.isNormal == 2) {
+                $(".tc-set").show();
+            }
+        }
+
+        form.on('submit(productPoint-info-form-submit)', function (data) {
+            data.field.description = editor.txt.html();
+            $.ajax({
+                'url':ctx + 'admin/productPoint/update',
+                'type':'post',
+                'dataType':'json',
+                'headers' : {'Content-Type' : 'application/json;charset=utf-8'}, //接口json格式
+                'traditional': true,//ajax传递数组必须添加属性
+                'data':JSON.stringify(data.field),
+                'success':function (data) {
+                    if(data.code==200){
+                        layer.closeAll();
+                        febs.alert.success(data.message);
+                        $('#febs-productPoint').find('#query').click();
+                    }else{
+                        febs.alert.warn(data.message);
+                    }
+                },
+                'error':function () {
+                    febs.alert.warn('服务器繁忙');
+                }
+            })
+            return false;
+        });
+
+    });
+</script>
\ No newline at end of file
diff --git a/src/main/resources/templates/febs/views/modules/ai/productPoint/list.html b/src/main/resources/templates/febs/views/modules/ai/productPoint/list.html
new file mode 100644
index 0000000..b6c4bfb
--- /dev/null
+++ b/src/main/resources/templates/febs/views/modules/ai/productPoint/list.html
@@ -0,0 +1,189 @@
+<div class="layui-fluid layui-anim febs-anim" id="febs-productPoint" lay-title="产品知识点">
+    <div class="layui-row febs-container">
+        <div class="layui-col-md12">
+            <div class="layui-card">
+                <div class="layui-card-body febs-table-full">
+                    <form class="layui-form layui-table-form" lay-filter="productPoint-table-form">
+                        <div class="layui-row">
+                            <div class="layui-col-md10">
+                                <div class="layui-form-item">
+                                </div>
+                            </div>
+                            <div class="layui-col-md2 layui-col-sm12 layui-col-xs12 table-action-area">
+                                <div class="layui-btn layui-btn-sm layui-btn-primary febs-button-blue-plain table-action" id="query">
+                                    <i class="layui-icon">&#xe848;</i>
+                                </div>
+                                <div class="layui-btn layui-btn-sm layui-btn-primary febs-button-green-plain table-action" id="reset">
+                                    <i class="layui-icon">&#xe79b;</i>
+                                </div>
+                            </div>
+                        </div>
+                    </form>
+                    <table lay-filter="productPointTable" lay-data="{id: 'productPointTable'}"></table>
+
+                    <style type="text/css">
+                        .layui-table-cell{
+                            text-align:center;
+                            height: auto;
+                            white-space: nowrap; /*文本不会换行,在同一行显示*/
+                            overflow: hidden; /*超出隐藏*/
+                            text-overflow: ellipsis; /*省略号显示*/
+                        }
+                        .layui-table img{
+                            max-width:100px
+                        }
+                        ::-webkit-scrollbar {
+                            height: 20px !important;
+                            background-color: #f4f4f4;
+                        }
+                    </style>
+                </div>
+            </div>
+        </div>
+    </div>
+</div>
+
+<script type="text/html" id="productPointToolbar">
+    <div class="layui-btn-container">
+        <button class="layui-btn layui-btn-sm layui-btn-primary febs-button-green-plain" shiro:hasPermission="pointList:add" lay-event="productPointAdd">新增</button>
+    </div>
+</script>
+
+<script type="text/html" id="productPointOption">
+    <button class="layui-btn layui-btn-normal layui-btn-sm" type="button" shiro:hasPermission="pointList:info" lay-event="productPointInfoEvent">编辑</button>
+    <button class="layui-btn layui-btn-danger layui-btn-sm" type="button" shiro:hasPermission="pointList:info" lay-event="productPointDeleteEvent">删除</button>
+</script>
+
+<script type="text/html" id="pointTypeFormat">
+    {{# if(d.isNormal == 1) { }}
+    <span>普通内容</span>
+    {{# }else if(d.isNormal == 2) { }}
+    <span>视频号内容</span>
+    {{# } else { }}
+    <span>-</span>
+    {{# } }}
+</script>
+
+<style>
+    .layui-form-onswitch {
+        background-color: #5FB878 !important;
+    }
+</style>
+<!-- 表格操作栏 end -->
+<script data-th-inline="none" type="text/javascript">
+    // 引入组件并初始化
+    layui.use([ 'jquery', 'form', 'table', 'febs'], function () {
+        var $ = layui.jquery,
+            febs = layui.febs,
+            form = layui.form,
+            table = layui.table,
+            $view = $('#febs-productPoint'),
+            $query = $view.find('#query'),
+            $reset = $view.find('#reset'),
+            $searchForm = $view.find('form'),
+            sortObject = {field: 'orderNum', type: null},
+            tableIns;
+
+        form.render();
+
+        // 表格初始化
+        initProductPointTable();
+
+        // 初始化表格操作栏各个按钮功能
+        table.on('tool(productPointTable)', function (obj) {
+            console.log("触发事件:", obj.event); // 调试信息
+            var data = obj.data,
+                layEvent = obj.event;
+
+            if (layEvent === 'productPointInfoEvent') {
+                if (data.state == 1){
+                    febs.alert.warn('请先禁用这行数据');
+                    return;
+                }
+                febs.modal.open('编辑','modules/ai/productPoint/info/' + data.id, {
+                    btn: ['提交', '取消'],
+                    area: ['100%', '100%'],
+                    yes: function (index, layero) {
+                        $('#febs-productPoint-Info').find('#submit').trigger('click');
+                    },
+                    btn2: function () {
+                        layer.closeAll();
+                    }
+                });
+            }
+
+            if (layEvent === 'productPointDeleteEvent') {
+                if (data.state == 1){
+                    febs.alert.warn('请先禁用这行数据');
+                    return;
+                }
+                febs.modal.confirm('删除', '确认删除?', function () {
+                    productPointDeleteEvent(data.id);
+                });
+            }
+        });
+
+        function productPointDeleteEvent(id) {
+            febs.get(ctx + 'admin/productPoint/delete/' + id, null, function (data) {
+                febs.alert.success(data.message);
+                $query.click();
+            });
+        }
+
+        // 初始化表格操作栏各个按钮功能
+        table.on('toolbar(productPointTable)', function (obj) {
+            let data = obj.data,
+                layEvent = obj.event;
+            if(layEvent === 'productPointAdd'){
+                febs.modal.open('新增', 'modules/ai/productPoint/add/', {
+                    btn: ['提交', '取消'],
+                    area:['100%','100%'],
+                    yes: function (index, layero) {
+                        $('#febs-productPoint-add').find('#submit').trigger('click');
+                    },
+                    btn2: function () {
+                        layer.closeAll();
+                    }
+                });
+            }
+        });
+
+        function initProductPointTable() {
+            tableIns = febs.table.init({
+                elem: $view.find('table'),
+                id: 'productPointTable',
+                url: ctx + 'admin/productPoint/list',
+                toolbar:"#productPointToolbar",
+                defaultToolbar:[],
+                cols: [[
+                    {type: 'checkbox'},
+                    {type: 'numbers', title: '', width: 80},
+                    {title: '操作', toolbar: '#productPointOption', minWidth: 200, align: 'center'},
+                    {field: 'title', title: '标题', minWidth: 100,align:'center'},
+                    {templet:"#pointTypeFormat",  title: '类型', minWidth: 140,align:'center'},
+                    {field: 'companyId', title: '公司编码', minWidth: 150,align:'center'},
+                ]]
+            });
+        }
+
+
+        // 查询按钮
+        $query.on('click', function () {
+            var params = $.extend(getQueryParams(), {field: sortObject.field, order: sortObject.type});
+            tableIns.reload({where: params, page: {curr: 1}});
+        });
+
+        // 刷新按钮
+        $reset.on('click', function () {
+            $searchForm[0].reset();
+            sortObject.type = 'null';
+            tableIns.reload({where: getQueryParams(), page: {curr: 1}, initSort: sortObject});
+        });
+        // 获取查询参数
+        function getQueryParams() {
+            return {
+            };
+        }
+
+    })
+</script>

--
Gitblit v1.9.1