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"></i>
+ </div>
+ <div class="layui-btn layui-btn-sm layui-btn-primary febs-button-green-plain table-action" id="reset">
+ <i class="layui-icon"></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