From 75d8b0ad39a7eb04f72ef8654dbb895322f07cfd Mon Sep 17 00:00:00 2001
From: Administrator <15274802129@163.com>
Date: Tue, 21 Oct 2025 13:41:42 +0800
Subject: [PATCH] feat(ai): 新增会员答题分页及详情查询功能 - 在 AiMemberAnswerMapper 中新增 getAnswerPage 方法及对应 XML 查询语句 - 新增 ApiMemberAnswerPageDto 和 ApiMemberAnswerPageVo 用于分页查询参数和返回结果 - 在 AiMemberAnswerService 及其实现类中添加 getAnswerPage 方法 - 在 AiMemberService 及其实现类中新增 answerPage 和 answerInfo 接口实现 - 新增 ApiMemberAnswerInfoDto 和 ApiMemberAnswerInfoVo 用于答题详情接口参数和响应 - 在 ApiMemberController 中增加 /answerPage 和 /answerInfo两个 POST 接口 - 优化 AiMemberTeamPracticeVo,增加 memberUuid 字段 - 统一导入包路径,简化代码结构
---
src/main/java/cc/mrbird/febs/ai/service/impl/AiMemberServiceImpl.java | 84 +++++++++++++++-
src/main/java/cc/mrbird/febs/ai/req/member/ApiMemberAnswerInfoDto.java | 17 +++
src/main/java/cc/mrbird/febs/ai/service/AiMemberAnswerService.java | 5 +
src/main/java/cc/mrbird/febs/ai/service/AiMemberService.java | 8 +
src/main/java/cc/mrbird/febs/ai/service/impl/AiMemberAnswerServiceImpl.java | 8 +
src/main/java/cc/mrbird/febs/ai/res/member/ApiMemberTeamPracticeVo.java | 3
src/main/java/cc/mrbird/febs/ai/controller/member/ApiMemberController.java | 28 ++++-
src/main/java/cc/mrbird/febs/ai/res/member/ApiMemberAnswerPageVo.java | 19 +++
src/main/java/cc/mrbird/febs/ai/mapper/AiMemberAnswerMapper.java | 4
src/main/java/cc/mrbird/febs/ai/res/member/ApiMemberAnswerInfoVo.java | 37 +++++++
src/main/resources/mapper/modules/AiMemberAnswerMapper.xml | 15 +++
src/main/java/cc/mrbird/febs/ai/req/member/ApiMemberAnswerPageDto.java | 26 +++++
12 files changed, 238 insertions(+), 16 deletions(-)
diff --git a/src/main/java/cc/mrbird/febs/ai/controller/member/ApiMemberController.java b/src/main/java/cc/mrbird/febs/ai/controller/member/ApiMemberController.java
index 6e58da0..dfc495f 100644
--- a/src/main/java/cc/mrbird/febs/ai/controller/member/ApiMemberController.java
+++ b/src/main/java/cc/mrbird/febs/ai/controller/member/ApiMemberController.java
@@ -1,11 +1,7 @@
package cc.mrbird.febs.ai.controller.member;
-import cc.mrbird.febs.ai.req.member.ApiMemberTeamPageDto;
-import cc.mrbird.febs.ai.req.member.ApiMemberTeamPracticeDto;
-import cc.mrbird.febs.ai.req.member.ApiMemberTeamStudyDto;
-import cc.mrbird.febs.ai.res.member.ApiMemberTeamPageVo;
-import cc.mrbird.febs.ai.res.member.ApiMemberTeamPracticeVo;
-import cc.mrbird.febs.ai.res.member.ApiMemberTeamStudyVo;
+import cc.mrbird.febs.ai.req.member.*;
+import cc.mrbird.febs.ai.res.member.*;
import cc.mrbird.febs.ai.service.AiMemberService;
import cc.mrbird.febs.common.entity.FebsResponse;
import io.swagger.annotations.Api;
@@ -53,6 +49,26 @@
return aiMemberService.practice(dto);
}
+ @ApiOperation(value = "题目练习分页查询", notes = "题目练习分页查询")
+ @ApiResponses({
+ @ApiResponse(code = 200, message = "success", response = ApiMemberAnswerPageVo.class)
+ })
+ @PostMapping(value = "/answerPage")
+ public FebsResponse answerPage(@RequestBody @Validated ApiMemberAnswerPageDto dto) {
+
+ return aiMemberService.answerPage(dto);
+ }
+
+ @ApiOperation(value = "题目练习详情查询", notes = "题目练习详情查询")
+ @ApiResponses({
+ @ApiResponse(code = 200, message = "success", response = ApiMemberAnswerInfoVo.class)
+ })
+ @PostMapping(value = "/answerInfo")
+ public FebsResponse answerInfo(@RequestBody @Validated ApiMemberAnswerInfoDto dto) {
+
+ return aiMemberService.answerInfo(dto);
+ }
+
@ApiOperation(value = "学习时长", notes = "学习时长")
@ApiResponses({
@ApiResponse(code = 200, message = "success", response = ApiMemberTeamStudyVo.class)
diff --git a/src/main/java/cc/mrbird/febs/ai/mapper/AiMemberAnswerMapper.java b/src/main/java/cc/mrbird/febs/ai/mapper/AiMemberAnswerMapper.java
index 041afbd..625a112 100644
--- a/src/main/java/cc/mrbird/febs/ai/mapper/AiMemberAnswerMapper.java
+++ b/src/main/java/cc/mrbird/febs/ai/mapper/AiMemberAnswerMapper.java
@@ -1,7 +1,9 @@
package cc.mrbird.febs.ai.mapper;
import cc.mrbird.febs.ai.entity.AiMemberAnswer;
+import cc.mrbird.febs.ai.req.member.ApiMemberAnswerPageDto;
import cc.mrbird.febs.ai.req.memberAnswer.ApiMemberProductWorkPageDto;
+import cc.mrbird.febs.ai.res.member.ApiMemberAnswerPageVo;
import cc.mrbird.febs.ai.res.memberAnswer.ApiMemberProductWorkVo;
import com.baomidou.mybatisplus.core.mapper.BaseMapper;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
@@ -16,4 +18,6 @@
public interface AiMemberAnswerMapper extends BaseMapper<AiMemberAnswer> {
Page<ApiMemberProductWorkVo> getPageMyWorkListByQuery(Page<ApiMemberProductWorkVo> page, @Param("record")ApiMemberProductWorkPageDto dto);
+
+ Page<ApiMemberAnswerPageVo> getAnswerPage(Page<ApiMemberAnswerPageVo> page, @Param("record")ApiMemberAnswerPageDto dto);
}
diff --git a/src/main/java/cc/mrbird/febs/ai/req/member/ApiMemberAnswerInfoDto.java b/src/main/java/cc/mrbird/febs/ai/req/member/ApiMemberAnswerInfoDto.java
new file mode 100644
index 0000000..995eac7
--- /dev/null
+++ b/src/main/java/cc/mrbird/febs/ai/req/member/ApiMemberAnswerInfoDto.java
@@ -0,0 +1,17 @@
+package cc.mrbird.febs.ai.req.member;
+
+import io.swagger.annotations.ApiModel;
+import io.swagger.annotations.ApiModelProperty;
+import lombok.Data;
+
+import javax.validation.constraints.NotBlank;
+
+@Data
+@ApiModel(value = "ApiMemberAnswerInfoDto", description = "参数")
+public class ApiMemberAnswerInfoDto {
+
+ @NotBlank(message = "产品ID不能为空")
+ @ApiModelProperty(value = "产品ID", example = "10")
+ private String answerId;
+
+}
diff --git a/src/main/java/cc/mrbird/febs/ai/req/member/ApiMemberAnswerPageDto.java b/src/main/java/cc/mrbird/febs/ai/req/member/ApiMemberAnswerPageDto.java
new file mode 100644
index 0000000..e2e7232
--- /dev/null
+++ b/src/main/java/cc/mrbird/febs/ai/req/member/ApiMemberAnswerPageDto.java
@@ -0,0 +1,26 @@
+package cc.mrbird.febs.ai.req.member;
+
+import io.swagger.annotations.ApiModel;
+import io.swagger.annotations.ApiModelProperty;
+import lombok.Data;
+
+import javax.validation.constraints.NotBlank;
+import javax.validation.constraints.NotNull;
+
+@Data
+@ApiModel(value = "ApiMemberAnswerPageDto", description = "参数")
+public class ApiMemberAnswerPageDto {
+
+ @NotNull(message = "页码不能为空")
+ @ApiModelProperty(value = "页码", example = "1")
+ private Integer pageNow;
+
+ @NotNull(message = "每页数量不能为空")
+ @ApiModelProperty(value = "每页数量", example = "10")
+ private Integer pageSize;
+
+ @NotBlank(message = "会员ID不能为空")
+ @ApiModelProperty(value = "会员ID", example = "10")
+ private String memberUuid;
+
+}
diff --git a/src/main/java/cc/mrbird/febs/ai/res/member/ApiMemberAnswerInfoVo.java b/src/main/java/cc/mrbird/febs/ai/res/member/ApiMemberAnswerInfoVo.java
new file mode 100644
index 0000000..8b6f6f4
--- /dev/null
+++ b/src/main/java/cc/mrbird/febs/ai/res/member/ApiMemberAnswerInfoVo.java
@@ -0,0 +1,37 @@
+package cc.mrbird.febs.ai.res.member;
+
+import cc.mrbird.febs.ai.res.memberAnswer.ApiMemberAnswerPreviousItemVo;
+import com.fasterxml.jackson.annotation.JsonFormat;
+import io.swagger.annotations.ApiModel;
+import io.swagger.annotations.ApiModelProperty;
+import lombok.Data;
+
+import java.util.Date;
+import java.util.List;
+
+@Data
+@ApiModel(value = "ApiMemberAnswerInfoVo", description = "参数")
+public class ApiMemberAnswerInfoVo {
+
+ @ApiModelProperty(value = "题目")
+ private String title;
+
+ @ApiModelProperty(value = "难度:1-简单,2-中等,3-困难")
+ private Integer difficulty;
+
+ @ApiModelProperty(value = "用户答题答案ID")
+ private String memberAnswerQuestionId;
+
+ @ApiModelProperty(value = "答案解析")
+ private String answerAnalysis;
+
+ @ApiModelProperty(value = "题目ID")
+ private String productQuestionId;
+
+ @ApiModelProperty(value = "时间")
+ @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss", timezone = "GMT+8")
+ private Date createdTime;
+
+ @ApiModelProperty(value = "选项")
+ private List<ApiMemberAnswerPreviousItemVo> answerList;
+}
diff --git a/src/main/java/cc/mrbird/febs/ai/res/member/ApiMemberAnswerPageVo.java b/src/main/java/cc/mrbird/febs/ai/res/member/ApiMemberAnswerPageVo.java
new file mode 100644
index 0000000..7ccde27
--- /dev/null
+++ b/src/main/java/cc/mrbird/febs/ai/res/member/ApiMemberAnswerPageVo.java
@@ -0,0 +1,19 @@
+package cc.mrbird.febs.ai.res.member;
+
+import io.swagger.annotations.ApiModel;
+import io.swagger.annotations.ApiModelProperty;
+import lombok.Data;
+
+@Data
+@ApiModel(value = "ApiMemberAnswerPageVo", description = "参数")
+public class ApiMemberAnswerPageVo {
+
+ @ApiModelProperty(value = "答题ID")
+ private String id;
+
+ @ApiModelProperty(value = "名称")
+ private String name;
+
+ @ApiModelProperty(value = "目标")
+ private String target;
+}
diff --git a/src/main/java/cc/mrbird/febs/ai/res/member/ApiMemberTeamPracticeVo.java b/src/main/java/cc/mrbird/febs/ai/res/member/ApiMemberTeamPracticeVo.java
index e43b4a2..ddf9fe4 100644
--- a/src/main/java/cc/mrbird/febs/ai/res/member/ApiMemberTeamPracticeVo.java
+++ b/src/main/java/cc/mrbird/febs/ai/res/member/ApiMemberTeamPracticeVo.java
@@ -11,6 +11,9 @@
@ApiModel(value = "ApiMemberTeamPracticeVo", description = "参数")
public class ApiMemberTeamPracticeVo {
+ @ApiModelProperty(value = "会员ID")
+ private String memberUuid;
+
@ApiModelProperty(value = "名称")
private String name;
diff --git a/src/main/java/cc/mrbird/febs/ai/service/AiMemberAnswerService.java b/src/main/java/cc/mrbird/febs/ai/service/AiMemberAnswerService.java
index 5fe71c5..52659cf 100644
--- a/src/main/java/cc/mrbird/febs/ai/service/AiMemberAnswerService.java
+++ b/src/main/java/cc/mrbird/febs/ai/service/AiMemberAnswerService.java
@@ -1,9 +1,12 @@
package cc.mrbird.febs.ai.service;
import cc.mrbird.febs.ai.entity.AiMemberAnswer;
+import cc.mrbird.febs.ai.req.member.ApiMemberAnswerPageDto;
import cc.mrbird.febs.ai.req.memberAnswer.*;
+import cc.mrbird.febs.ai.res.member.ApiMemberAnswerPageVo;
import cc.mrbird.febs.common.entity.FebsResponse;
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
+import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.baomidou.mybatisplus.extension.service.IService;
import java.util.Date;
@@ -40,4 +43,6 @@
List<AiMemberAnswer> getListByCompanyId(String companyId);
List<AiMemberAnswer> getListByCompanyIdAndMemberUuid(String companyId, String memberUuid);
+
+ Page<ApiMemberAnswerPageVo> getAnswerPage(Page<ApiMemberAnswerPageVo> page, ApiMemberAnswerPageDto dto);
}
diff --git a/src/main/java/cc/mrbird/febs/ai/service/AiMemberService.java b/src/main/java/cc/mrbird/febs/ai/service/AiMemberService.java
index b090753..41bb61b 100644
--- a/src/main/java/cc/mrbird/febs/ai/service/AiMemberService.java
+++ b/src/main/java/cc/mrbird/febs/ai/service/AiMemberService.java
@@ -1,9 +1,7 @@
package cc.mrbird.febs.ai.service;
import cc.mrbird.febs.ai.entity.AiMember;
-import cc.mrbird.febs.ai.req.member.ApiMemberTeamPageDto;
-import cc.mrbird.febs.ai.req.member.ApiMemberTeamPracticeDto;
-import cc.mrbird.febs.ai.req.member.ApiMemberTeamStudyDto;
+import cc.mrbird.febs.ai.req.member.*;
import cc.mrbird.febs.common.entity.FebsResponse;
import com.baomidou.mybatisplus.extension.service.IService;
@@ -16,4 +14,8 @@
FebsResponse practice(ApiMemberTeamPracticeDto dto);
FebsResponse study(ApiMemberTeamStudyDto dto);
+
+ FebsResponse answerPage(ApiMemberAnswerPageDto dto);
+
+ FebsResponse answerInfo(ApiMemberAnswerInfoDto dto);
}
diff --git a/src/main/java/cc/mrbird/febs/ai/service/impl/AiMemberAnswerServiceImpl.java b/src/main/java/cc/mrbird/febs/ai/service/impl/AiMemberAnswerServiceImpl.java
index faac604..fa9b81f 100644
--- a/src/main/java/cc/mrbird/febs/ai/service/impl/AiMemberAnswerServiceImpl.java
+++ b/src/main/java/cc/mrbird/febs/ai/service/impl/AiMemberAnswerServiceImpl.java
@@ -3,7 +3,9 @@
import cc.mrbird.febs.ai.entity.*;
import cc.mrbird.febs.ai.enumerates.AiTypeEnum;
import cc.mrbird.febs.ai.mapper.AiMemberAnswerMapper;
+import cc.mrbird.febs.ai.req.member.ApiMemberAnswerPageDto;
import cc.mrbird.febs.ai.req.memberAnswer.*;
+import cc.mrbird.febs.ai.res.member.ApiMemberAnswerPageVo;
import cc.mrbird.febs.ai.res.memberAnswer.*;
import cc.mrbird.febs.ai.res.product.ApiProductVo;
import cc.mrbird.febs.ai.res.productQuestionItem.ApiMemberAnswerItemVo;
@@ -426,4 +428,10 @@
.eq(AiMemberAnswer::getState, AiTypeEnum.AI_MEMBER_ANSWER_STATE_DONE.getCode())
);
}
+
+ @Override
+ public Page<ApiMemberAnswerPageVo> getAnswerPage(Page<ApiMemberAnswerPageVo> page, ApiMemberAnswerPageDto dto) {
+
+ return aiMemberAnswerMapper.getAnswerPage(page,dto);
+ }
}
diff --git a/src/main/java/cc/mrbird/febs/ai/service/impl/AiMemberServiceImpl.java b/src/main/java/cc/mrbird/febs/ai/service/impl/AiMemberServiceImpl.java
index 5dc5c78..211e6bf 100644
--- a/src/main/java/cc/mrbird/febs/ai/service/impl/AiMemberServiceImpl.java
+++ b/src/main/java/cc/mrbird/febs/ai/service/impl/AiMemberServiceImpl.java
@@ -3,13 +3,9 @@
import cc.mrbird.febs.ai.entity.*;
import cc.mrbird.febs.ai.enumerates.AiTypeEnum;
import cc.mrbird.febs.ai.mapper.AiMemberMapper;
-import cc.mrbird.febs.ai.req.member.ApiMemberTeamPageDto;
-import cc.mrbird.febs.ai.req.member.ApiMemberTeamPracticeDto;
-import cc.mrbird.febs.ai.req.member.ApiMemberTeamStudyDto;
-import cc.mrbird.febs.ai.res.member.ApiMemberTeamPageVo;
-import cc.mrbird.febs.ai.res.member.ApiMemberTeamPracticeVo;
-import cc.mrbird.febs.ai.res.member.ApiMemberTeamStudyVo;
-import cc.mrbird.febs.ai.res.productPoint.ApiProductPointListVo;
+import cc.mrbird.febs.ai.req.member.*;
+import cc.mrbird.febs.ai.res.member.*;
+import cc.mrbird.febs.ai.res.memberAnswer.ApiMemberAnswerPreviousItemVo;
import cc.mrbird.febs.ai.service.*;
import cc.mrbird.febs.common.entity.FebsResponse;
import cc.mrbird.febs.common.exception.FebsException;
@@ -41,9 +37,11 @@
private final MallMemberMapper mallMemberMapper;
private final AiMemberTalkService aiMemberTalkService;
private final AiMemberAnswerService aiMemberAnswerService;
+ private final AiMemberAnswerItemService aiMemberAnswerItemService;
private final AiMemberPointService aiMemberPointService;
private final AiProductService aiProductService;
private final AiProductPointService aiProductPointService;
+ private final AiProductQuestionItemService aiProductQuestionItemService;
@Override
public AiMember getById(String id) {
return aiMemberMapper.selectById( id);
@@ -207,6 +205,7 @@
String productId = aiProduct.getId();
ApiMemberTeamPracticeVo vo = new ApiMemberTeamPracticeVo();
+ vo.setMemberUuid(memberUuid);
vo.setName(aiProduct.getName());
vo.setTarget(aiProduct.getTarget());
@@ -286,4 +285,75 @@
return new FebsResponse().success().data(vos);
}
+
+ @Override
+ public FebsResponse answerPage(ApiMemberAnswerPageDto dto) {
+
+ // 创建分页对象,传入当前页和每页大小
+ Page<ApiMemberAnswerPageVo> page = new Page<>(dto.getPageNow(), dto.getPageSize());
+ Page<ApiMemberAnswerPageVo> aiMemberAnswerPage = aiMemberAnswerService.getAnswerPage(page, dto);
+
+ return new FebsResponse().success().data(aiMemberAnswerPage);
+ }
+
+ @Override
+ public FebsResponse answerInfo(ApiMemberAnswerInfoDto dto) {
+ List<ApiMemberAnswerInfoVo> vos = new ArrayList<>();
+
+ String answerId = dto.getAnswerId();
+ AiMemberAnswer memberAnswer = aiMemberAnswerService.getById(answerId);
+ if (memberAnswer == null){
+ return new FebsResponse().success().data(vos);
+ }
+
+ List<AiMemberAnswerItem> records = aiMemberAnswerItemService.getBaseMapper().selectList(
+ Wrappers.lambdaQuery(AiMemberAnswerItem.class)
+ .eq(AiMemberAnswerItem::getAnswerId, memberAnswer.getId())
+ .orderByAsc(AiMemberAnswerItem::getCreatedTime)
+ );
+
+ if (CollUtil.isNotEmpty(records)){
+ Set<String> collect = records.stream().map(AiMemberAnswerItem::getProductQuestionId).collect(Collectors.toSet());
+
+ LambdaQueryWrapper<AiProductQuestionItem> questionItemQuery = Wrappers.lambdaQuery(AiProductQuestionItem.class);
+ questionItemQuery.in(AiProductQuestionItem::getProductQuestionId,collect);
+ List<AiProductQuestionItem> listByQuery = aiProductQuestionItemService.getListByQuery(questionItemQuery);
+
+ if (CollUtil.isNotEmpty(listByQuery)){
+ //利用stream流,操作集合listByQuery,返回一个map对象,key为productQuestionId,value为List<AiProductQuestionItem>对象
+ Map<String, List<AiProductQuestionItem>> questionItemMap =
+ listByQuery.stream().collect(Collectors.groupingBy(AiProductQuestionItem::getProductQuestionId));
+
+ for (AiMemberAnswerItem record : records){
+ String productQuestionId = record.getProductQuestionId();
+ ApiMemberAnswerInfoVo vo = new ApiMemberAnswerInfoVo();
+ vo.setTitle(record.getTitle());
+ vo.setDifficulty(record.getDifficulty());
+ vo.setMemberAnswerQuestionId(record.getMemberAnswerQuestionId());
+ vo.setAnswerAnalysis(record.getAnswerAnalysis());
+ vo.setProductQuestionId(record.getProductQuestionId());
+ vo.setCreatedTime(record.getCreatedTime());
+
+
+ List<AiProductQuestionItem> questionItemList = questionItemMap.get(productQuestionId);
+ List<ApiMemberAnswerPreviousItemVo> answerList = new ArrayList<>();
+ if (CollUtil.isNotEmpty(questionItemList)){
+ for (AiProductQuestionItem questionItem : questionItemList){
+ ApiMemberAnswerPreviousItemVo itemVo = new ApiMemberAnswerPreviousItemVo();
+ itemVo.setId(questionItem.getId());
+ itemVo.setAnswer(questionItem.getAnswer());
+ itemVo.setAnswerAnalysis(questionItem.getAnswerAnalysis());
+ itemVo.setCorrectAnswer(questionItem.getCorrectAnswer());
+ answerList.add(itemVo);
+ }
+ }
+ vo.setAnswerList(answerList);
+
+ vos.add( vo);
+ }
+ }
+ }
+
+ return new FebsResponse().success().data(vos);
+ }
}
diff --git a/src/main/resources/mapper/modules/AiMemberAnswerMapper.xml b/src/main/resources/mapper/modules/AiMemberAnswerMapper.xml
index 46ba6fb..22d1d20 100644
--- a/src/main/resources/mapper/modules/AiMemberAnswerMapper.xml
+++ b/src/main/resources/mapper/modules/AiMemberAnswerMapper.xml
@@ -24,4 +24,19 @@
group by a.product_id
order by a.CREATED_TIME desc
</select>
+
+
+
+ <select id="getAnswerPage" resultType="cc.mrbird.febs.ai.res.member.ApiMemberAnswerPageVo">
+ select
+ a.id as id,
+ b.name as name,
+ b.target as target
+ from ai_member_answer a
+ left join ai_product b on a.product_id = b.id and b.state = 1
+ where a.member_id = #{record.memberUuid}
+ and a.state = 1
+ group by a.product_id
+ order by a.CREATED_TIME desc
+ </select>
</mapper>
\ No newline at end of file
--
Gitblit v1.9.1