浏览代码

项目管理完成一小半

Huanyi 3 天之前
父节点
当前提交
8fd9492dc7
共有 33 个文件被更改,包括 1459 次插入7 次删除
  1. 14 1
      ruoyi-admin/src/main/java/com/yingpaipay/web/controller/AppletAuthController.java
  2. 21 4
      ruoyi-admin/src/main/java/com/yingpaipay/web/controller/AppletUserController.java
  3. 17 0
      ruoyi-admin/src/main/java/com/yingpaipay/web/domain/bo/AppletUserUpdateAvatarBo.java
  4. 19 0
      ruoyi-admin/src/main/java/com/yingpaipay/web/domain/bo/AppletUserUpdatePasswordBo.java
  5. 23 0
      ruoyi-admin/src/main/java/com/yingpaipay/web/domain/vo/AppletUserBasicInfoVo.java
  6. 9 0
      ruoyi-admin/src/main/java/com/yingpaipay/web/service/IAppletUserService.java
  7. 40 0
      ruoyi-admin/src/main/java/com/yingpaipay/web/service/impl/AppletUserServiceImpl.java
  8. 1 0
      ruoyi-admin/src/main/resources/application.yml
  9. 26 0
      ruoyi-modules/ruoyi-system/src/main/java/com/yingpaipay/system/domain/SysUserProject.java
  10. 14 0
      ruoyi-modules/ruoyi-system/src/main/java/com/yingpaipay/system/mapper/SysUserProjectMapper.java
  11. 2 0
      ruoyi-modules/ruoyi-system/src/main/java/org/dromara/system/domain/bo/SysUserBo.java
  12. 2 0
      ruoyi-modules/ruoyi-system/src/main/java/org/dromara/system/domain/vo/SysUserVo.java
  13. 22 0
      ruoyi-modules/ruoyi-system/src/main/java/org/dromara/system/service/impl/SysUserServiceImpl.java
  14. 6 0
      ruoyi-modules/yingpaipay-business/pom.xml
  15. 39 0
      ruoyi-modules/yingpaipay-business/src/main/java/com/yingpaipay/business/controller/FolderController.java
  16. 135 0
      ruoyi-modules/yingpaipay-business/src/main/java/com/yingpaipay/business/controller/ProjectController.java
  17. 67 0
      ruoyi-modules/yingpaipay-business/src/main/java/com/yingpaipay/business/domain/Folder.java
  18. 111 0
      ruoyi-modules/yingpaipay-business/src/main/java/com/yingpaipay/business/domain/Project.java
  19. 65 0
      ruoyi-modules/yingpaipay-business/src/main/java/com/yingpaipay/business/domain/bo/FolderBo.java
  20. 108 0
      ruoyi-modules/yingpaipay-business/src/main/java/com/yingpaipay/business/domain/bo/ProjectBo.java
  21. 19 0
      ruoyi-modules/yingpaipay-business/src/main/java/com/yingpaipay/business/domain/bo/ProjectUpdateStatusBo.java
  22. 80 0
      ruoyi-modules/yingpaipay-business/src/main/java/com/yingpaipay/business/domain/vo/FolderVo.java
  23. 19 0
      ruoyi-modules/yingpaipay-business/src/main/java/com/yingpaipay/business/domain/vo/ProjectListByNameVo.java
  24. 29 0
      ruoyi-modules/yingpaipay-business/src/main/java/com/yingpaipay/business/domain/vo/ProjectMemberListVo.java
  25. 130 0
      ruoyi-modules/yingpaipay-business/src/main/java/com/yingpaipay/business/domain/vo/ProjectVo.java
  26. 29 0
      ruoyi-modules/yingpaipay-business/src/main/java/com/yingpaipay/business/enumeration/ProjectStatusEnum.java
  27. 15 0
      ruoyi-modules/yingpaipay-business/src/main/java/com/yingpaipay/business/mapper/FolderMapper.java
  28. 15 0
      ruoyi-modules/yingpaipay-business/src/main/java/com/yingpaipay/business/mapper/ProjectMapper.java
  29. 11 0
      ruoyi-modules/yingpaipay-business/src/main/java/com/yingpaipay/business/service/IFolderService.java
  30. 79 0
      ruoyi-modules/yingpaipay-business/src/main/java/com/yingpaipay/business/service/IProjectService.java
  31. 23 0
      ruoyi-modules/yingpaipay-business/src/main/java/com/yingpaipay/business/service/impl/FolderServiceImpl.java
  32. 256 0
      ruoyi-modules/yingpaipay-business/src/main/java/com/yingpaipay/business/service/impl/ProjectServiceImpl.java
  33. 13 2
      script/sql/business/create.sql

+ 14 - 1
ruoyi-admin/src/main/java/com/yingpaipay/web/controller/AppletAuthController.java

@@ -1,12 +1,16 @@
 package com.yingpaipay.web.controller;
 
 import cn.dev33.satoken.annotation.SaIgnore;
+import cn.dev33.satoken.stp.StpUtil;
 import com.yingpaipay.web.domain.bo.AppletLoginBo;
 import com.yingpaipay.web.domain.vo.AppletLoginVo;
 import com.yingpaipay.web.service.IAppletAuthService;
 import com.yingpaipay.web.service.factory.AppletAuthStrategyFactory;
 import lombok.RequiredArgsConstructor;
+import lombok.extern.slf4j.Slf4j;
 import org.dromara.common.core.domain.R;
+import org.dromara.common.satoken.utils.LoginHelper;
+import org.dromara.common.web.core.BaseController;
 import org.dromara.web.domain.vo.LoginVo;
 import org.springframework.web.bind.annotation.*;
 
@@ -21,7 +25,8 @@ import org.springframework.web.bind.annotation.*;
 @RestController
 @RequiredArgsConstructor
 @RequestMapping("/applet/auth")
-public class AppletAuthController {
+@Slf4j
+public class AppletAuthController extends BaseController {
 
     private final AppletAuthStrategyFactory factory;
 
@@ -30,4 +35,12 @@ public class AppletAuthController {
         return R.ok(factory.getInstance(strategy).login(bo));
     }
 
+    @DeleteMapping("/logout")
+    public R<Void> logout() {
+        Object loginId = StpUtil.getLoginId();
+        StpUtil.logout();
+        log.warn("退出登录,ID: {}", loginId);
+        return R.ok();
+    }
+
 }

+ 21 - 4
ruoyi-admin/src/main/java/com/yingpaipay/web/controller/AppletUserController.java

@@ -1,12 +1,14 @@
 package com.yingpaipay.web.controller;
 
+import com.yingpaipay.web.domain.bo.AppletUserUpdateAvatarBo;
+import com.yingpaipay.web.domain.bo.AppletUserUpdatePasswordBo;
+import com.yingpaipay.web.domain.vo.AppletUserBasicInfoVo;
 import com.yingpaipay.web.domain.vo.AppletUserInfoVo;
 import com.yingpaipay.web.service.IAppletUserService;
 import lombok.RequiredArgsConstructor;
 import org.dromara.common.core.domain.R;
-import org.springframework.web.bind.annotation.GetMapping;
-import org.springframework.web.bind.annotation.RequestMapping;
-import org.springframework.web.bind.annotation.RestController;
+import org.dromara.common.web.core.BaseController;
+import org.springframework.web.bind.annotation.*;
 
 /**
  * @Author: Huanyi
@@ -18,7 +20,7 @@ import org.springframework.web.bind.annotation.RestController;
 @RestController
 @RequestMapping("/applet/user")
 @RequiredArgsConstructor
-public class AppletUserController {
+public class AppletUserController extends BaseController {
 
     private final IAppletUserService userService;
 
@@ -27,4 +29,19 @@ public class AppletUserController {
         return R.ok(userService.getInfo());
     }
 
+    @GetMapping("/getBasicInfo")
+    public R<AppletUserBasicInfoVo> getBasicInfo() {
+        return R.ok(userService.getBasicInfo());
+    }
+
+    @PutMapping("/updatePassword")
+    public R<Void> updatePassword(@RequestBody AppletUserUpdatePasswordBo bo) {
+        return toAjax(userService.updatePassword(bo));
+    }
+
+    @PutMapping("/updateAvatar")
+    public R<Void> updateAvatar(@RequestBody AppletUserUpdateAvatarBo bo) {
+        return toAjax(userService.updateAvatar(bo));
+    }
+
 }

+ 17 - 0
ruoyi-admin/src/main/java/com/yingpaipay/web/domain/bo/AppletUserUpdateAvatarBo.java

@@ -0,0 +1,17 @@
+package com.yingpaipay.web.domain.bo;
+
+import lombok.Data;
+
+/**
+ * @Author: Huanyi
+ * @CreateTime: 2025-12-08
+ * @Description:
+ * @Version: 1.0
+ */
+
+@Data
+public class AppletUserUpdateAvatarBo {
+
+    private Long id;
+
+}

+ 19 - 0
ruoyi-admin/src/main/java/com/yingpaipay/web/domain/bo/AppletUserUpdatePasswordBo.java

@@ -0,0 +1,19 @@
+package com.yingpaipay.web.domain.bo;
+
+import lombok.Data;
+
+/**
+ * @Author: Huanyi
+ * @CreateTime: 2025-12-08
+ * @Description:
+ * @Version: 1.0
+ */
+
+@Data
+public class AppletUserUpdatePasswordBo {
+
+    private String oldPassword;
+
+    private String newPassword;
+
+}

+ 23 - 0
ruoyi-admin/src/main/java/com/yingpaipay/web/domain/vo/AppletUserBasicInfoVo.java

@@ -0,0 +1,23 @@
+package com.yingpaipay.web.domain.vo;
+
+import lombok.Data;
+
+/**
+ * @Author: Huanyi
+ * @CreateTime: 2025-12-08
+ * @Description:
+ * @Version: 1.0
+ */
+
+@Data
+public class AppletUserBasicInfoVo {
+
+    private String nickname;
+
+    private String phoneNumber;
+
+    private String avatar;
+
+    private String gender;
+
+}

+ 9 - 0
ruoyi-admin/src/main/java/com/yingpaipay/web/service/IAppletUserService.java

@@ -1,5 +1,8 @@
 package com.yingpaipay.web.service;
 
+import com.yingpaipay.web.domain.bo.AppletUserUpdateAvatarBo;
+import com.yingpaipay.web.domain.bo.AppletUserUpdatePasswordBo;
+import com.yingpaipay.web.domain.vo.AppletUserBasicInfoVo;
 import com.yingpaipay.web.domain.vo.AppletUserInfoVo;
 
 /**
@@ -11,4 +14,10 @@ import com.yingpaipay.web.domain.vo.AppletUserInfoVo;
 
 public interface IAppletUserService {
     AppletUserInfoVo getInfo();
+
+    AppletUserBasicInfoVo getBasicInfo();
+
+    int updatePassword(AppletUserUpdatePasswordBo bo);
+
+    int updateAvatar(AppletUserUpdateAvatarBo bo);
 }

+ 40 - 0
ruoyi-admin/src/main/java/com/yingpaipay/web/service/impl/AppletUserServiceImpl.java

@@ -1,9 +1,16 @@
 package com.yingpaipay.web.service.impl;
 
+import cn.hutool.crypto.digest.BCrypt;
+import com.baomidou.mybatisplus.core.toolkit.Wrappers;
+import com.yingpaipay.web.domain.bo.AppletUserUpdateAvatarBo;
+import com.yingpaipay.web.domain.bo.AppletUserUpdatePasswordBo;
+import com.yingpaipay.web.domain.vo.AppletUserBasicInfoVo;
 import com.yingpaipay.web.domain.vo.AppletUserInfoVo;
 import com.yingpaipay.web.service.IAppletUserService;
 import lombok.RequiredArgsConstructor;
 import lombok.extern.slf4j.Slf4j;
+import org.dromara.common.core.exception.BusinessException;
+import org.dromara.common.core.utils.MessageUtils;
 import org.dromara.common.satoken.utils.LoginHelper;
 import org.dromara.system.domain.SysUser;
 import org.dromara.system.mapper.SysUserMapper;
@@ -36,4 +43,37 @@ public class AppletUserServiceImpl implements IAppletUserService {
         }
         return vo;
     }
+
+    @Override
+    public AppletUserBasicInfoVo getBasicInfo() {
+        SysUser user = userMapper.selectById(LoginHelper.getUserId());
+        AppletUserBasicInfoVo vo = new AppletUserBasicInfoVo();
+        vo.setNickname(user.getNickName());
+        vo.setPhoneNumber(user.getPhonenumber());
+        if (user.getAvatar() != null) {
+            vo.setAvatar(ossService.getById(user.getAvatar()).getUrl());
+        }
+        vo.setGender(user.getSex());
+
+        return vo;
+    }
+
+    @Override
+    public int updatePassword(AppletUserUpdatePasswordBo bo) {
+        SysUser user = userMapper.selectById(LoginHelper.getUserId());
+        if (!BCrypt.checkpw(bo.getOldPassword(), user.getPassword())) {
+            throw new BusinessException(MessageUtils.message("applet.auth.login.passworderror"));
+        }
+        user.setPassword(BCrypt.hashpw(bo.getNewPassword()));
+        return userMapper.updateById(user);
+    }
+
+    @Override
+    public int updateAvatar(AppletUserUpdateAvatarBo bo) {
+        return userMapper.update(
+            Wrappers.lambdaUpdate(SysUser.class)
+                .eq(SysUser::getUserId, LoginHelper.getUserId())
+                .set(bo.getId() != null, SysUser::getAvatar, bo.getId())
+        );
+    }
 }

+ 1 - 0
ruoyi-admin/src/main/resources/application.yml

@@ -127,6 +127,7 @@ tenant:
         - sys_dict_data
         - sys_dict_type
         - applet_setting
+        - sys_user_project
 
 # MyBatisPlus配置
 # https://baomidou.com/config/

+ 26 - 0
ruoyi-modules/ruoyi-system/src/main/java/com/yingpaipay/system/domain/SysUserProject.java

@@ -0,0 +1,26 @@
+package com.yingpaipay.system.domain;
+
+import com.baomidou.mybatisplus.annotation.IdType;
+import com.baomidou.mybatisplus.annotation.TableId;
+import com.baomidou.mybatisplus.annotation.TableName;
+import lombok.Data;
+import org.dromara.common.tenant.core.TenantEntity;
+
+
+/**
+ * @Author: Huanyi
+ * @CreateTime: 2025-12-08
+ * @Description:
+ * @Version: 1.0
+ */
+
+@Data
+@TableName("sys_user_project")
+public class SysUserProject {
+
+    @TableId(type = IdType.INPUT)
+    private Long userId;
+
+    private String projects;
+
+}

+ 14 - 0
ruoyi-modules/ruoyi-system/src/main/java/com/yingpaipay/system/mapper/SysUserProjectMapper.java

@@ -0,0 +1,14 @@
+package com.yingpaipay.system.mapper;
+
+import com.yingpaipay.system.domain.SysUserProject;
+import org.dromara.common.mybatis.core.mapper.BaseMapperPlus;
+
+/**
+ * @Author: Huanyi
+ * @CreateTime: 2025-12-08
+ * @Description:
+ * @Version: 1.0
+ */
+
+public interface SysUserProjectMapper extends BaseMapperPlus<SysUserProject, SysUserProject> {
+}

+ 2 - 0
ruoyi-modules/ruoyi-system/src/main/java/org/dromara/system/domain/bo/SysUserBo.java

@@ -114,6 +114,8 @@ public class SysUserBo extends BaseEntity {
      */
     private String excludeUserIds;
 
+    private String projects;
+
     public SysUserBo(Long userId) {
         this.userId = userId;
     }

+ 2 - 0
ruoyi-modules/ruoyi-system/src/main/java/org/dromara/system/domain/vo/SysUserVo.java

@@ -140,4 +140,6 @@ public class SysUserVo implements Serializable {
      */
     private Long roleId;
 
+    private String projects;
+
 }

+ 22 - 0
ruoyi-modules/ruoyi-system/src/main/java/org/dromara/system/service/impl/SysUserServiceImpl.java

@@ -11,6 +11,8 @@ import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper;
 import com.baomidou.mybatisplus.core.conditions.update.LambdaUpdateWrapper;
 import com.baomidou.mybatisplus.core.toolkit.Wrappers;
 import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
+import com.yingpaipay.system.domain.SysUserProject;
+import com.yingpaipay.system.mapper.SysUserProjectMapper;
 import lombok.RequiredArgsConstructor;
 import lombok.extern.slf4j.Slf4j;
 import org.dromara.common.core.constant.CacheNames;
@@ -22,6 +24,7 @@ import org.dromara.common.core.utils.*;
 import org.dromara.common.mybatis.core.page.PageQuery;
 import org.dromara.common.mybatis.core.page.TableDataInfo;
 import org.dromara.common.satoken.utils.LoginHelper;
+import org.dromara.common.tenant.helper.TenantHelper;
 import org.dromara.system.domain.SysUser;
 import org.dromara.system.domain.SysUserPost;
 import org.dromara.system.domain.SysUserRole;
@@ -55,6 +58,7 @@ public class SysUserServiceImpl implements ISysUserService, UserService {
     private final SysPostMapper postMapper;
     private final SysUserRoleMapper userRoleMapper;
     private final SysUserPostMapper userPostMapper;
+    private final SysUserProjectMapper userProjectMapper;
 
     @Override
     public TableDataInfo<SysUserVo> selectPageUserList(SysUserBo user, PageQuery pageQuery) {
@@ -182,6 +186,10 @@ public class SysUserServiceImpl implements ISysUserService, UserService {
             return user;
         }
         user.setRoles(roleMapper.selectRolesByUserId(user.getUserId()));
+        SysUserProject project = userProjectMapper.selectOne(Wrappers.lambdaQuery(SysUserProject.class).eq(SysUserProject::getUserId, userId));
+        if (project != null) {
+            user.setProjects(project.getProjects());
+        }
         return user;
     }
 
@@ -318,9 +326,21 @@ public class SysUserServiceImpl implements ISysUserService, UserService {
         insertUserPost(user, false);
         // 新增用户与角色管理
         insertUserRole(user, false);
+        // 新增用户与项目的关联
+        insertUserProjects(user, false);
         return rows;
     }
 
+    private void insertUserProjects(SysUserBo user, Boolean flag) {
+        if (flag) {
+            userProjectMapper.delete(Wrappers.lambdaQuery(SysUserProject.class).eq(SysUserProject::getUserId, user.getUserId()));
+        }
+        SysUserProject entity = new SysUserProject();
+        entity.setUserId(user.getUserId());
+        entity.setProjects(user.getProjects());
+        userProjectMapper.insert(entity);
+    }
+
     /**
      * 注册用户信息
      *
@@ -350,6 +370,8 @@ public class SysUserServiceImpl implements ISysUserService, UserService {
         insertUserRole(user, true);
         // 新增用户与岗位管理
         insertUserPost(user, true);
+        // 新增用户与项目的关联
+        insertUserProjects(user, true);
         SysUser sysUser = MapstructUtils.convert(user, SysUser.class);
         // 防止错误更新后导致的数据误删除
         int flag = baseMapper.updateById(sysUser);

+ 6 - 0
ruoyi-modules/yingpaipay-business/pom.xml

@@ -13,6 +13,12 @@
     <artifactId>yingpaipay-business</artifactId>
 
     <dependencies>
+
+        <dependency>
+            <groupId>org.dromara</groupId>
+            <artifactId>ruoyi-system</artifactId>
+        </dependency>
+
         <!-- 通用工具-->
         <dependency>
             <groupId>org.dromara</groupId>

+ 39 - 0
ruoyi-modules/yingpaipay-business/src/main/java/com/yingpaipay/business/controller/FolderController.java

@@ -0,0 +1,39 @@
+package com.yingpaipay.business.controller;
+
+import java.util.List;
+
+import lombok.RequiredArgsConstructor;
+import jakarta.servlet.http.HttpServletResponse;
+import jakarta.validation.constraints.*;
+import cn.dev33.satoken.annotation.SaCheckPermission;
+import org.springframework.web.bind.annotation.*;
+import org.springframework.validation.annotation.Validated;
+import org.dromara.common.idempotent.annotation.RepeatSubmit;
+import org.dromara.common.log.annotation.Log;
+import org.dromara.common.web.core.BaseController;
+import org.dromara.common.mybatis.core.page.PageQuery;
+import org.dromara.common.core.domain.R;
+import org.dromara.common.core.validate.AddGroup;
+import org.dromara.common.core.validate.EditGroup;
+import org.dromara.common.log.enums.BusinessType;
+import org.dromara.common.excel.utils.ExcelUtil;
+import com.yingpaipay.business.domain.vo.FolderVo;
+import com.yingpaipay.business.domain.bo.FolderBo;
+import com.yingpaipay.business.service.IFolderService;
+import org.dromara.common.mybatis.core.page.TableDataInfo;
+
+/**
+ * 文件夹管理
+ *
+ * @author Huanyi
+ * @date 2025-12-08
+ */
+@Validated
+@RequiredArgsConstructor
+@RestController
+@RequestMapping("/document/folder")
+public class FolderController extends BaseController {
+
+    private final IFolderService folderService;
+
+}

+ 135 - 0
ruoyi-modules/yingpaipay-business/src/main/java/com/yingpaipay/business/controller/ProjectController.java

@@ -0,0 +1,135 @@
+package com.yingpaipay.business.controller;
+
+import java.util.List;
+
+import com.yingpaipay.business.domain.bo.ProjectUpdateStatusBo;
+import com.yingpaipay.business.domain.vo.ProjectListByNameVo;
+import com.yingpaipay.business.domain.vo.ProjectMemberListVo;
+import lombok.RequiredArgsConstructor;
+import jakarta.servlet.http.HttpServletResponse;
+import jakarta.validation.constraints.*;
+import cn.dev33.satoken.annotation.SaCheckPermission;
+import org.springframework.web.bind.annotation.*;
+import org.springframework.validation.annotation.Validated;
+import org.dromara.common.idempotent.annotation.RepeatSubmit;
+import org.dromara.common.log.annotation.Log;
+import org.dromara.common.web.core.BaseController;
+import org.dromara.common.mybatis.core.page.PageQuery;
+import org.dromara.common.core.domain.R;
+import org.dromara.common.core.validate.AddGroup;
+import org.dromara.common.core.validate.EditGroup;
+import org.dromara.common.log.enums.BusinessType;
+import org.dromara.common.excel.utils.ExcelUtil;
+import com.yingpaipay.business.domain.vo.ProjectVo;
+import com.yingpaipay.business.domain.bo.ProjectBo;
+import com.yingpaipay.business.service.IProjectService;
+import org.dromara.common.mybatis.core.page.TableDataInfo;
+
+/**
+ * 项目管理
+ *
+ * @author Huanyi
+ * @date 2025-12-05
+ */
+@Validated
+@RequiredArgsConstructor
+@RestController
+@RequestMapping("/project/management")
+public class ProjectController extends BaseController {
+
+    private final IProjectService projectService;
+
+    /**
+     * 查询项目管理列表
+     */
+    @SaCheckPermission("project:management:list")
+    @GetMapping("/list")
+    public TableDataInfo<ProjectVo> list(ProjectBo bo, PageQuery pageQuery) {
+        return projectService.queryPageList(bo, pageQuery);
+    }
+
+    /**
+     * 导出项目管理列表
+     */
+    @SaCheckPermission("project:management:export")
+    @Log(title = "项目管理", businessType = BusinessType.EXPORT)
+    @PostMapping("/export")
+    public void export(ProjectBo bo, HttpServletResponse response) {
+        List<ProjectVo> list = projectService.queryList(bo);
+        ExcelUtil.exportExcel(list, "项目管理", ProjectVo.class, response);
+    }
+
+    /**
+     * 获取项目管理详细信息
+     *
+     * @param id 主键
+     */
+    @SaCheckPermission("project:management:query")
+    @GetMapping("/{id}")
+    public R<ProjectVo> getInfo(@NotNull(message = "主键不能为空")
+                                     @PathVariable Long id) {
+        return R.ok(projectService.queryById(id));
+    }
+
+    /**
+     * 新增项目管理
+     */
+    @SaCheckPermission("project:management:add")
+    @Log(title = "项目管理", businessType = BusinessType.INSERT)
+    @RepeatSubmit()
+    @PostMapping()
+    public R<Void> add(@Validated(AddGroup.class) @RequestBody ProjectBo bo) {
+        return toAjax(projectService.insertByBo(bo));
+    }
+
+    /**
+     * 修改项目管理
+     */
+    @SaCheckPermission("project:management:edit")
+    @Log(title = "项目管理", businessType = BusinessType.UPDATE)
+    @RepeatSubmit()
+    @PutMapping()
+    public R<Void> edit(@Validated(EditGroup.class) @RequestBody ProjectBo bo) {
+        return toAjax(projectService.updateByBo(bo));
+    }
+
+    /**
+     * 删除项目管理
+     *
+     * @param ids 主键串
+     */
+    @SaCheckPermission("project:management:remove")
+    @Log(title = "项目管理", businessType = BusinessType.DELETE)
+    @DeleteMapping("/{ids}")
+    public R<Void> remove(@NotEmpty(message = "主键不能为空")
+                          @PathVariable Long[] ids) {
+        return toAjax(projectService.deleteWithValidByIds(List.of(ids), true));
+    }
+
+    @GetMapping("/getByName")
+    public R<List<ProjectListByNameVo>> getListByName(@RequestParam("name") String name) {
+        return R.ok(projectService.getListByName(name));
+    }
+
+    @GetMapping("/listOnUser")
+    public R<List<ProjectListByNameVo>> getList() {
+        return R.ok(projectService.getList());
+    }
+
+    @SaCheckPermission("project:management:updateStatus")
+    @Log(title = "项目管理", businessType = BusinessType.UPDATE)
+    @RepeatSubmit()
+    @PutMapping("/updateStatus")
+    public R<Void> updateStatus(@RequestBody ProjectUpdateStatusBo bo) {
+        return toAjax(projectService.updateStatus(bo));
+    }
+
+    @SaCheckPermission("project:management:queryProjectMember")
+    @Log(title = "项目管理", businessType = BusinessType.UPDATE)
+    @RepeatSubmit()
+    @GetMapping("/queryProjectMember")
+    public TableDataInfo<ProjectMemberListVo> queryProjectMember(@RequestParam("id") Long id, PageQuery pageQuery) {
+        return projectService.selectMemberListById(id, pageQuery);
+    }
+
+}

+ 67 - 0
ruoyi-modules/yingpaipay-business/src/main/java/com/yingpaipay/business/domain/Folder.java

@@ -0,0 +1,67 @@
+package com.yingpaipay.business.domain;
+
+import org.dromara.common.tenant.core.TenantEntity;
+import com.baomidou.mybatisplus.annotation.*;
+import lombok.Data;
+import lombok.EqualsAndHashCode;
+
+import java.io.Serial;
+
+/**
+ * 文件夹管理对象 folder
+ *
+ * @author Huanyi
+ * @date 2025-12-08
+ */
+@Data
+@EqualsAndHashCode(callSuper = true)
+@TableName("folder")
+public class Folder extends TenantEntity {
+
+    @Serial
+    private static final long serialVersionUID = 1L;
+
+    /**
+     * 序号
+     */
+    @TableId(value = "id")
+    private Long id;
+
+    /**
+     * 所属项目
+     */
+    private Long projectId;
+
+    /**
+     * 父级
+     */
+    private Long parentId;
+
+    /**
+     * 类型
+     */
+    private Long type;
+
+    /**
+     * 名称
+     */
+    private String name;
+
+    /**
+     * 状态
+     */
+    private Long status;
+
+    /**
+     * 备注
+     */
+    private String note;
+
+    /**
+     * 删除标志(0代表存在 1代表删除)
+     */
+    @TableLogic
+    private String delFlag;
+
+
+}

+ 111 - 0
ruoyi-modules/yingpaipay-business/src/main/java/com/yingpaipay/business/domain/Project.java

@@ -0,0 +1,111 @@
+package com.yingpaipay.business.domain;
+
+import org.dromara.common.tenant.core.TenantEntity;
+import com.baomidou.mybatisplus.annotation.*;
+import lombok.Data;
+import lombok.EqualsAndHashCode;
+import org.dromara.common.translation.annotation.Translation;
+import java.util.Date;
+import com.fasterxml.jackson.annotation.JsonFormat;
+import org.dromara.common.translation.constant.TransConstant;
+
+import java.io.Serial;
+
+/**
+ * 项目管理对象 project
+ *
+ * @author Huanyi
+ * @date 2025-12-05
+ */
+@Data
+@EqualsAndHashCode(callSuper = true)
+@TableName("project")
+public class Project extends TenantEntity {
+
+    @Serial
+    private static final long serialVersionUID = 1L;
+
+    /**
+     * 序号
+     */
+    @TableId(value = "id")
+    private Long id;
+
+    /**
+     * 项目编号
+     */
+    private String code;
+
+    /**
+     * 名称
+     */
+    private String name;
+
+    /**
+     * 图标
+     */
+    private Long icon;
+
+    /**
+     * 项目语言
+     */
+    private String language;
+
+    /**
+     * 项目类型
+     */
+    private String type;
+
+    /**
+     * 状态
+     */
+    private Integer status;
+
+    /**
+     * PD/GPD
+     */
+    private String pdGpd;
+
+    /**
+     * PM/GPM
+     */
+    private String pmGpm;
+
+    /**
+     * CTA/GCTA
+     */
+    private String ctaGcta;
+
+    /**
+     * 申办方
+     */
+    private String sponsor;
+
+    /**
+     * CRO
+     */
+    private String cro;
+
+    /**
+     * 备注
+     */
+    private String note;
+
+    /**
+     * 开始时间
+     */
+    private Date startTime;
+
+    /**
+     * 结束时间
+     */
+    private Date endTime;
+
+    /**
+     * 删除标志(0代表存在 1代表删除)
+     */
+    @TableLogic
+    private String delFlag;
+
+
+}

+ 65 - 0
ruoyi-modules/yingpaipay-business/src/main/java/com/yingpaipay/business/domain/bo/FolderBo.java

@@ -0,0 +1,65 @@
+package com.yingpaipay.business.domain.bo;
+
+import com.yingpaipay.business.domain.Folder;
+import org.dromara.common.mybatis.core.domain.BaseEntity;
+import org.dromara.common.core.validate.AddGroup;
+import org.dromara.common.core.validate.EditGroup;
+import io.github.linpeilie.annotations.AutoMapper;
+import lombok.Data;
+import lombok.EqualsAndHashCode;
+import jakarta.validation.constraints.*;
+
+/**
+ * 文件夹管理业务对象 folder
+ *
+ * @author Huanyi
+ * @date 2025-12-08
+ */
+@Data
+@EqualsAndHashCode(callSuper = true)
+@AutoMapper(target = Folder.class, reverseConvertGenerate = false)
+public class FolderBo extends BaseEntity {
+
+    /**
+     * 序号
+     */
+    @NotNull(message = "序号不能为空", groups = { EditGroup.class })
+    private Long id;
+
+    /**
+     * 所属项目
+     */
+    @NotNull(message = "所属项目不能为空", groups = { AddGroup.class, EditGroup.class })
+    private Long projectId;
+
+    /**
+     * 父级
+     */
+//    @NotNull(message = "父级不能为空", groups = { AddGroup.class, EditGroup.class })
+    private Long parentId;
+
+    /**
+     * 类型
+     */
+    @NotNull(message = "类型不能为空", groups = { AddGroup.class, EditGroup.class })
+    private Long type;
+
+    /**
+     * 名称
+     */
+    @NotBlank(message = "名称不能为空", groups = { AddGroup.class, EditGroup.class })
+    private String name;
+
+    /**
+     * 状态
+     */
+    @NotNull(message = "状态不能为空", groups = { AddGroup.class, EditGroup.class })
+    private Long status;
+
+    /**
+     * 备注
+     */
+    private String note;
+
+
+}

+ 108 - 0
ruoyi-modules/yingpaipay-business/src/main/java/com/yingpaipay/business/domain/bo/ProjectBo.java

@@ -0,0 +1,108 @@
+package com.yingpaipay.business.domain.bo;
+
+import com.yingpaipay.business.domain.Project;
+import org.dromara.common.mybatis.core.domain.BaseEntity;
+import org.dromara.common.core.validate.AddGroup;
+import org.dromara.common.core.validate.EditGroup;
+import io.github.linpeilie.annotations.AutoMapper;
+import lombok.Data;
+import lombok.EqualsAndHashCode;
+import jakarta.validation.constraints.*;
+import org.dromara.common.translation.annotation.Translation;
+import java.util.Date;
+import com.fasterxml.jackson.annotation.JsonFormat;
+import org.dromara.common.translation.constant.TransConstant;
+
+/**
+ * 项目管理业务对象 project
+ *
+ * @author Huanyi
+ * @date 2025-12-05
+ */
+@Data
+@EqualsAndHashCode(callSuper = true)
+@AutoMapper(target = Project.class, reverseConvertGenerate = false)
+public class ProjectBo extends BaseEntity {
+
+    /**
+     * 序号
+     */
+    @NotNull(message = "序号不能为空", groups = { EditGroup.class })
+    private Long id;
+
+    /**
+     * 项目编号
+     */
+    @NotBlank(message = "项目编号不能为空", groups = { AddGroup.class, EditGroup.class })
+    private String code;
+
+    /**
+     * 名称
+     */
+    @NotBlank(message = "名称不能为空", groups = { AddGroup.class, EditGroup.class })
+    private String name;
+
+    /**
+     * 图标
+     */
+    private Long icon;
+
+    /**
+     * 项目语言
+     */
+    @NotBlank(message = "项目语言不能为空", groups = { AddGroup.class, EditGroup.class })
+    private String language;
+
+    /**
+     * 项目类型
+     */
+    @NotBlank(message = "项目类型不能为空", groups = { AddGroup.class, EditGroup.class })
+    private String type;
+
+    /**
+     * 状态
+     */
+    private Integer status;
+
+    /**
+     * PD/GPD
+     */
+    private String pdGpd;
+
+    /**
+     * PM/GPM
+     */
+    private String pmGpm;
+
+    /**
+     * CTA/GCTA
+     */
+    private String ctaGcta;
+
+    /**
+     * 申办方
+     */
+    private String sponsor;
+
+    /**
+     * CRO
+     */
+    private String cro;
+
+    /**
+     * 备注
+     */
+    private String note;
+
+    /**
+     * 开始时间
+     */
+    private Date startTime;
+
+    /**
+     * 结束时间
+     */
+    private Date endTime;
+
+
+}

+ 19 - 0
ruoyi-modules/yingpaipay-business/src/main/java/com/yingpaipay/business/domain/bo/ProjectUpdateStatusBo.java

@@ -0,0 +1,19 @@
+package com.yingpaipay.business.domain.bo;
+
+import lombok.Data;
+
+/**
+ * @Author: Huanyi
+ * @CreateTime: 2025-12-08
+ * @Description:
+ * @Version: 1.0
+ */
+
+@Data
+public class ProjectUpdateStatusBo {
+
+    private Long id;
+
+    private Integer status;
+
+}

+ 80 - 0
ruoyi-modules/yingpaipay-business/src/main/java/com/yingpaipay/business/domain/vo/FolderVo.java

@@ -0,0 +1,80 @@
+package com.yingpaipay.business.domain.vo;
+
+import com.yingpaipay.business.domain.Folder;
+import cn.idev.excel.annotation.ExcelIgnoreUnannotated;
+import cn.idev.excel.annotation.ExcelProperty;
+import org.dromara.common.excel.annotation.ExcelDictFormat;
+import org.dromara.common.excel.convert.ExcelDictConvert;
+import io.github.linpeilie.annotations.AutoMapper;
+import lombok.Data;
+
+import java.io.Serial;
+import java.io.Serializable;
+import java.util.Date;
+
+
+
+/**
+ * 文件夹管理视图对象 folder
+ *
+ * @author Huanyi
+ * @date 2025-12-08
+ */
+@Data
+@ExcelIgnoreUnannotated
+@AutoMapper(target = Folder.class)
+public class FolderVo implements Serializable {
+
+    @Serial
+    private static final long serialVersionUID = 1L;
+
+    /**
+     * 序号
+     */
+    @ExcelProperty(value = "序号")
+    private Long id;
+
+    /**
+     * 所属项目
+     */
+    @ExcelProperty(value = "所属项目")
+    private Long projectId;
+
+    /**
+     * 父级
+     */
+    @ExcelProperty(value = "父级")
+    private Long parentId;
+
+    /**
+     * 类型
+     */
+    @ExcelProperty(value = "类型")
+    private Long type;
+
+    /**
+     * 名称
+     */
+    @ExcelProperty(value = "名称")
+    private String name;
+
+    /**
+     * 状态
+     */
+    @ExcelProperty(value = "状态")
+    private Long status;
+
+    /**
+     * 创建时间
+     */
+    @ExcelProperty(value = "创建时间")
+    private Date createTime;
+
+    /**
+     * 更新时间
+     */
+    @ExcelProperty(value = "更新时间")
+    private Date updateTime;
+
+
+}

+ 19 - 0
ruoyi-modules/yingpaipay-business/src/main/java/com/yingpaipay/business/domain/vo/ProjectListByNameVo.java

@@ -0,0 +1,19 @@
+package com.yingpaipay.business.domain.vo;
+
+import lombok.Data;
+
+/**
+ * @Author: Huanyi
+ * @CreateTime: 2025-12-08
+ * @Description:
+ * @Version: 1.0
+ */
+
+@Data
+public class ProjectListByNameVo {
+
+    private Long id;
+
+    private String name;
+
+}

+ 29 - 0
ruoyi-modules/yingpaipay-business/src/main/java/com/yingpaipay/business/domain/vo/ProjectMemberListVo.java

@@ -0,0 +1,29 @@
+package com.yingpaipay.business.domain.vo;
+
+import lombok.Data;
+
+import java.util.Date;
+
+/**
+ * @Author: Huanyi
+ * @CreateTime: 2025-12-08
+ * @Description:
+ * @Version: 1.0
+ */
+
+@Data
+public class ProjectMemberListVo {
+
+    private Long id;
+
+    private String name;
+
+    private String phoneNumber;
+
+    private String dept;
+
+    private String role;
+
+    private Date time;
+
+}

+ 130 - 0
ruoyi-modules/yingpaipay-business/src/main/java/com/yingpaipay/business/domain/vo/ProjectVo.java

@@ -0,0 +1,130 @@
+package com.yingpaipay.business.domain.vo;
+
+import org.dromara.common.translation.annotation.Translation;
+import java.util.Date;
+import com.fasterxml.jackson.annotation.JsonFormat;
+import org.dromara.common.translation.constant.TransConstant;
+import com.yingpaipay.business.domain.Project;
+import cn.idev.excel.annotation.ExcelIgnoreUnannotated;
+import cn.idev.excel.annotation.ExcelProperty;
+import org.dromara.common.excel.annotation.ExcelDictFormat;
+import org.dromara.common.excel.convert.ExcelDictConvert;
+import io.github.linpeilie.annotations.AutoMapper;
+import lombok.Data;
+
+import java.io.Serial;
+import java.io.Serializable;
+import java.util.Date;
+
+
+
+/**
+ * 项目管理视图对象 project
+ *
+ * @author Huanyi
+ * @date 2025-12-05
+ */
+@Data
+@ExcelIgnoreUnannotated
+@AutoMapper(target = Project.class)
+public class ProjectVo implements Serializable {
+
+    @Serial
+    private static final long serialVersionUID = 1L;
+
+    /**
+     * 序号
+     */
+    @ExcelProperty(value = "序号")
+    private Long id;
+
+    /**
+     * 项目编号
+     */
+    @ExcelProperty(value = "项目编号")
+    private String code;
+
+    /**
+     * 名称
+     */
+    @ExcelProperty(value = "名称")
+    private String name;
+
+    /**
+     * 项目语言
+     */
+    @ExcelProperty(value = "项目语言", converter = ExcelDictConvert.class)
+    @ExcelDictFormat(dictType = "project_language")
+    private String language;
+
+    /**
+     * 项目类型
+     */
+    @ExcelProperty(value = "项目类型", converter = ExcelDictConvert.class)
+    @ExcelDictFormat(dictType = "project_type")
+    private String type;
+
+    /**
+     * 状态
+     */
+    @ExcelProperty(value = "状态")
+    private Integer status;
+
+    /**
+     * PD/GPD
+     */
+    @ExcelProperty(value = "PD/GPD")
+    private String pdGpd;
+
+    /**
+     * PM/GPM
+     */
+    @ExcelProperty(value = "PM/GPM")
+    private String pmGpm;
+
+    /**
+     * CTA/GCTA
+     */
+    @ExcelProperty(value = "CTA/GCTA")
+    private String ctaGcta;
+
+    /**
+     * 申办方
+     */
+    @ExcelProperty(value = "申办方")
+    private String sponsor;
+
+    /**
+     * CRO
+     */
+    @ExcelProperty(value = "CRO")
+    private String cro;
+
+    /**
+     * 开始时间
+     */
+    @ExcelProperty(value = "开始时间")
+    private Date startTime;
+
+    /**
+     * 结束时间
+     */
+    @ExcelProperty(value = "结束时间")
+    private Date endTime;
+
+    /**
+     * 创建时间
+     */
+    @ExcelProperty(value = "创建时间")
+    private Date createTime;
+
+    /**
+     * 更新时间
+     */
+    @ExcelProperty(value = "更新时间")
+    private Date updateTime;
+
+    private Long icon;
+    private String iconUrl;
+
+}

+ 29 - 0
ruoyi-modules/yingpaipay-business/src/main/java/com/yingpaipay/business/enumeration/ProjectStatusEnum.java

@@ -0,0 +1,29 @@
+package com.yingpaipay.business.enumeration;
+
+import lombok.Getter;
+
+/**
+ * @Author: Huanyi
+ * @CreateTime: 2025-12-08
+ * @Description:
+ * @Version: 1.0
+ */
+
+@Getter
+public enum ProjectStatusEnum {
+
+    UNPLAYED(0, "未开始"),
+    UNDERWAY(1, "进行中"),
+    PAUSED(2, "暂停中"),
+    FINISHED(3, "已完成"),
+    ;
+
+    private final Integer value;
+    private final String label;
+
+    ProjectStatusEnum(Integer value, String label) {
+        this.value = value;
+        this.label = label;
+    }
+
+}

+ 15 - 0
ruoyi-modules/yingpaipay-business/src/main/java/com/yingpaipay/business/mapper/FolderMapper.java

@@ -0,0 +1,15 @@
+package com.yingpaipay.business.mapper;
+
+import com.yingpaipay.business.domain.Folder;
+import com.yingpaipay.business.domain.vo.FolderVo;
+import org.dromara.common.mybatis.core.mapper.BaseMapperPlus;
+
+/**
+ * 文件夹管理Mapper接口
+ *
+ * @author Huanyi
+ * @date 2025-12-08
+ */
+public interface FolderMapper extends BaseMapperPlus<Folder, FolderVo> {
+
+}

+ 15 - 0
ruoyi-modules/yingpaipay-business/src/main/java/com/yingpaipay/business/mapper/ProjectMapper.java

@@ -0,0 +1,15 @@
+package com.yingpaipay.business.mapper;
+
+import com.yingpaipay.business.domain.Project;
+import com.yingpaipay.business.domain.vo.ProjectVo;
+import org.dromara.common.mybatis.core.mapper.BaseMapperPlus;
+
+/**
+ * 项目管理Mapper接口
+ *
+ * @author Huanyi
+ * @date 2025-12-05
+ */
+public interface ProjectMapper extends BaseMapperPlus<Project, ProjectVo> {
+
+}

+ 11 - 0
ruoyi-modules/yingpaipay-business/src/main/java/com/yingpaipay/business/service/IFolderService.java

@@ -0,0 +1,11 @@
+package com.yingpaipay.business.service;
+
+/**
+ * 文件夹管理Service接口
+ *
+ * @author Huanyi
+ * @date 2025-12-08
+ */
+public interface IFolderService {
+
+}

+ 79 - 0
ruoyi-modules/yingpaipay-business/src/main/java/com/yingpaipay/business/service/IProjectService.java

@@ -0,0 +1,79 @@
+package com.yingpaipay.business.service;
+
+import com.yingpaipay.business.domain.bo.ProjectUpdateStatusBo;
+import com.yingpaipay.business.domain.vo.ProjectListByNameVo;
+import com.yingpaipay.business.domain.vo.ProjectMemberListVo;
+import com.yingpaipay.business.domain.vo.ProjectVo;
+import com.yingpaipay.business.domain.bo.ProjectBo;
+import org.dromara.common.mybatis.core.page.TableDataInfo;
+import org.dromara.common.mybatis.core.page.PageQuery;
+
+import java.util.Collection;
+import java.util.List;
+
+/**
+ * 项目管理Service接口
+ *
+ * @author Huanyi
+ * @date 2025-12-05
+ */
+public interface IProjectService {
+
+    /**
+     * 查询项目管理
+     *
+     * @param id 主键
+     * @return 项目管理
+     */
+    ProjectVo queryById(Long id);
+
+    /**
+     * 分页查询项目管理列表
+     *
+     * @param bo        查询条件
+     * @param pageQuery 分页参数
+     * @return 项目管理分页列表
+     */
+    TableDataInfo<ProjectVo> queryPageList(ProjectBo bo, PageQuery pageQuery);
+
+    /**
+     * 查询符合条件的项目管理列表
+     *
+     * @param bo 查询条件
+     * @return 项目管理列表
+     */
+    List<ProjectVo> queryList(ProjectBo bo);
+
+    /**
+     * 新增项目管理
+     *
+     * @param bo 项目管理
+     * @return 是否新增成功
+     */
+    Boolean insertByBo(ProjectBo bo);
+
+    /**
+     * 修改项目管理
+     *
+     * @param bo 项目管理
+     * @return 是否修改成功
+     */
+    Boolean updateByBo(ProjectBo bo);
+
+    /**
+     * 校验并批量删除项目管理信息
+     *
+     * @param ids     待删除的主键集合
+     * @param isValid 是否进行有效性校验
+     * @return 是否删除成功
+     */
+    Boolean deleteWithValidByIds(Collection<Long> ids, Boolean isValid);
+
+    List<ProjectListByNameVo> getListByName(String name);
+
+    List<ProjectListByNameVo> getList();
+
+    int updateStatus(ProjectUpdateStatusBo bo);
+
+    TableDataInfo<ProjectMemberListVo> selectMemberListById(Long id, PageQuery pageQuery);
+}

+ 23 - 0
ruoyi-modules/yingpaipay-business/src/main/java/com/yingpaipay/business/service/impl/FolderServiceImpl.java

@@ -0,0 +1,23 @@
+package com.yingpaipay.business.service.impl;
+
+import lombok.RequiredArgsConstructor;
+import lombok.extern.slf4j.Slf4j;
+import org.springframework.stereotype.Service;
+import com.yingpaipay.business.mapper.FolderMapper;
+import com.yingpaipay.business.service.IFolderService;
+
+/**
+ * 文件夹管理Service业务层处理
+ *
+ * @author Huanyi
+ * @date 2025-12-08
+ */
+@Slf4j
+@RequiredArgsConstructor
+@Service
+public class FolderServiceImpl implements IFolderService {
+
+    private final FolderMapper baseMapper;
+
+
+}

+ 256 - 0
ruoyi-modules/yingpaipay-business/src/main/java/com/yingpaipay/business/service/impl/ProjectServiceImpl.java

@@ -0,0 +1,256 @@
+package com.yingpaipay.business.service.impl;
+
+import com.baomidou.mybatisplus.core.metadata.IPage;
+import com.yingpaipay.business.domain.bo.ProjectUpdateStatusBo;
+import com.yingpaipay.business.domain.vo.ProjectListByNameVo;
+import com.yingpaipay.business.domain.vo.ProjectMemberListVo;
+import com.yingpaipay.system.domain.SysUserProject;
+import com.yingpaipay.system.mapper.SysUserProjectMapper;
+import org.dromara.common.core.utils.MapstructUtils;
+import org.dromara.common.core.utils.StringUtils;
+import org.dromara.common.mybatis.core.page.TableDataInfo;
+import org.dromara.common.mybatis.core.page.PageQuery;
+import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
+import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
+import com.baomidou.mybatisplus.core.toolkit.Wrappers;
+import lombok.RequiredArgsConstructor;
+import lombok.extern.slf4j.Slf4j;
+import org.dromara.common.mybatis.helper.DataBaseHelper;
+import org.dromara.common.satoken.utils.LoginHelper;
+import org.dromara.system.domain.SysUser;
+import org.dromara.system.mapper.SysUserMapper;
+import org.dromara.system.service.ISysDeptService;
+import org.dromara.system.service.ISysOssService;
+import org.dromara.system.service.ISysRoleService;
+import org.springframework.stereotype.Service;
+import com.yingpaipay.business.domain.bo.ProjectBo;
+import com.yingpaipay.business.domain.vo.ProjectVo;
+import com.yingpaipay.business.domain.Project;
+import com.yingpaipay.business.mapper.ProjectMapper;
+import com.yingpaipay.business.service.IProjectService;
+
+import java.util.*;
+
+/**
+ * 项目管理Service业务层处理
+ *
+ * @author Huanyi
+ * @date 2025-12-05
+ */
+@Slf4j
+@RequiredArgsConstructor
+@Service
+public class ProjectServiceImpl implements IProjectService {
+
+    private final ProjectMapper baseMapper;
+    private final SysUserProjectMapper userProjectMapper;
+    private final SysUserMapper userMapper;
+
+    private final ISysOssService ossService;
+    private final ISysDeptService deptService;
+    private final ISysRoleService roleService;
+
+    /**
+     * 查询项目管理
+     *
+     * @param id 主键
+     * @return 项目管理
+     */
+    @Override
+    public ProjectVo queryById(Long id) {
+        ProjectVo vo = baseMapper.selectVoById(id);
+        if (vo.getIcon() != null) {
+            vo.setIconUrl(ossService.getById(vo.getIcon()).getUrl());
+        }
+        return vo;
+    }
+
+    /**
+     * 分页查询项目管理列表
+     *
+     * @param bo        查询条件
+     * @param pageQuery 分页参数
+     * @return 项目管理分页列表
+     */
+    @Override
+    public TableDataInfo<ProjectVo> queryPageList(ProjectBo bo, PageQuery pageQuery) {
+        LambdaQueryWrapper<Project> lqw = buildQueryWrapper(bo);
+        Page<ProjectVo> result = baseMapper.selectVoPage(pageQuery.build(), lqw);
+        List<Long> iconIds = new ArrayList<>();
+        result.getRecords().forEach(e -> iconIds.add(e.getIcon()));
+        if (!iconIds.isEmpty()) {
+            Map<Long, String> iconMap = new HashMap<>();
+            ossService.listByIds(iconIds).forEach(e -> iconMap.put(e.getOssId(), e.getUrl()));
+            result.getRecords().forEach(e -> e.setIconUrl(iconMap.get(e.getIcon())));
+        }
+        return TableDataInfo.build(result);
+    }
+
+    /**
+     * 查询符合条件的项目管理列表
+     *
+     * @param bo 查询条件
+     * @return 项目管理列表
+     */
+    @Override
+    public List<ProjectVo> queryList(ProjectBo bo) {
+        LambdaQueryWrapper<Project> lqw = buildQueryWrapper(bo);
+        return baseMapper.selectVoList(lqw);
+    }
+
+    private LambdaQueryWrapper<Project> buildQueryWrapper(ProjectBo bo) {
+        Map<String, Object> params = bo.getParams();
+        LambdaQueryWrapper<Project> lqw = Wrappers.lambdaQuery();
+        lqw.orderByDesc(Project::getId);
+        lqw.like(StringUtils.isNotBlank(bo.getCode()), Project::getCode, bo.getCode());
+        lqw.like(StringUtils.isNotBlank(bo.getName()), Project::getName, bo.getName());
+        lqw.eq(StringUtils.isNotBlank(bo.getLanguage()), Project::getLanguage, bo.getLanguage());
+        lqw.eq(StringUtils.isNotBlank(bo.getType()), Project::getType, bo.getType());
+        lqw.eq(bo.getStatus() != null, Project::getStatus, bo.getStatus());
+        lqw.like(StringUtils.isNotBlank(bo.getPdGpd()), Project::getPdGpd, bo.getPdGpd());
+        lqw.like(StringUtils.isNotBlank(bo.getPmGpm()), Project::getPmGpm, bo.getPmGpm());
+        lqw.like(StringUtils.isNotBlank(bo.getCtaGcta()), Project::getCtaGcta, bo.getCtaGcta());
+        lqw.like(StringUtils.isNotBlank(bo.getSponsor()), Project::getSponsor, bo.getSponsor());
+        lqw.like(StringUtils.isNotBlank(bo.getCro()), Project::getCro, bo.getCro());
+        lqw.between(params.get("beginStartTime") != null && params.get("endStartTime") != null,
+            Project::getStartTime, params.get("beginStartTime"), params.get("endStartTime"));
+        lqw.between(params.get("beginEndTime") != null && params.get("endEndTime") != null,
+            Project::getEndTime, params.get("beginEndTime"), params.get("endEndTime"));
+        lqw.between(params.get("beginCreateTime") != null && params.get("endCreateTime") != null,
+            Project::getCreateTime, params.get("beginCreateTime"), params.get("endCreateTime"));
+        lqw.between(params.get("beginUpdateTime") != null && params.get("endUpdateTime") != null,
+            Project::getUpdateTime, params.get("beginUpdateTime"), params.get("endUpdateTime"));
+
+        SysUserProject project = userProjectMapper.selectOne(Wrappers.lambdaQuery(SysUserProject.class).eq(SysUserProject::getUserId, LoginHelper.getUserId()));
+        if (project.getProjects().isEmpty()) {
+            lqw.in(Project::getId, List.of(-1L));
+        } else if (!project.getProjects().equals("*")) {
+            List<Long> projectIds = Arrays.stream(project.getProjects().split(",")).map(Long::valueOf).toList();
+            lqw.in(Project::getId, projectIds);
+        }
+
+        return lqw;
+    }
+
+    /**
+     * 新增项目管理
+     *
+     * @param bo 项目管理
+     * @return 是否新增成功
+     */
+    @Override
+    public Boolean insertByBo(ProjectBo bo) {
+        Project add = MapstructUtils.convert(bo, Project.class);
+        validEntityBeforeSave(add);
+        boolean flag = baseMapper.insert(add) > 0;
+        if (flag) {
+            bo.setId(add.getId());
+        }
+        return flag;
+    }
+
+    /**
+     * 修改项目管理
+     *
+     * @param bo 项目管理
+     * @return 是否修改成功
+     */
+    @Override
+    public Boolean updateByBo(ProjectBo bo) {
+        Project update = MapstructUtils.convert(bo, Project.class);
+        validEntityBeforeSave(update);
+        return baseMapper.updateById(update) > 0;
+    }
+
+    /**
+     * 保存前的数据校验
+     */
+    private void validEntityBeforeSave(Project entity) {
+        //TODO 做一些数据校验,如唯一约束
+    }
+
+    /**
+     * 校验并批量删除项目管理信息
+     *
+     * @param ids     待删除的主键集合
+     * @param isValid 是否进行有效性校验
+     * @return 是否删除成功
+     */
+    @Override
+    public Boolean deleteWithValidByIds(Collection<Long> ids, Boolean isValid) {
+        if (isValid) {
+            //TODO 做一些业务上的校验,判断是否需要校验
+        }
+        return baseMapper.deleteByIds(ids) > 0;
+    }
+
+    @Override
+    public List<ProjectListByNameVo> getListByName(String name) {
+        List<ProjectListByNameVo> vos = new ArrayList<>();
+        baseMapper.selectList(
+            Wrappers.lambdaQuery(Project.class)
+                .like(StringUtils.isNotBlank(name), Project::getName, name)
+        ).forEach(e -> {
+            ProjectListByNameVo vo = new ProjectListByNameVo();
+            vo.setId(e.getId());
+            vo.setName(e.getName());
+            vos.add(vo);
+        });
+        return vos;
+    }
+
+    @Override
+    public List<ProjectListByNameVo> getList() {
+        List<ProjectListByNameVo> vos = new ArrayList<>();
+        baseMapper.selectList().forEach(e -> {
+            ProjectListByNameVo vo = new ProjectListByNameVo();
+            vo.setId(e.getId());
+            vo.setName(e.getName());
+            vos.add(vo);
+        });
+        return vos;
+    }
+
+    @Override
+    public int updateStatus(ProjectUpdateStatusBo bo) {
+        return baseMapper.update(
+            Wrappers.lambdaUpdate(Project.class)
+                .eq(Project::getId, bo.getId())
+                .set(Project::getStatus, bo.getStatus())
+        );
+    }
+
+    @Override
+    public TableDataInfo<ProjectMemberListVo> selectMemberListById(Long id, PageQuery pageQuery) {
+        List<Long> ids = new ArrayList<>();
+        userProjectMapper.selectList(
+            Wrappers.lambdaQuery(SysUserProject.class)
+                .and(wrapper -> wrapper.eq(SysUserProject::getProjects, "*")
+                    .or()
+                    .apply(DataBaseHelper.findInSet(id, "projects")))
+        ).forEach(e -> ids.add(e.getUserId()));
+        if (ids.isEmpty()) {
+            return TableDataInfo.build();
+        }
+        IPage<SysUser> page = userMapper.selectPage(
+            pageQuery.build(),
+            Wrappers.lambdaQuery(SysUser.class).in(SysUser::getUserId, ids)
+        );
+        List<Long> deptIds = new ArrayList<>();
+        Map<Long, String> deptMap = new HashMap<>();
+        page.getRecords().forEach(e -> {
+            deptIds.add(e.getDeptId());
+        });
+        deptService.selectDeptByIds(deptIds).forEach(e -> deptMap.put(e.getDeptId(), e.getDeptName()));
+        return TableDataInfo.build(page.convert(e -> {
+            ProjectMemberListVo vo = new ProjectMemberListVo();
+            vo.setId(e.getUserId());
+            vo.setName(e.getNickName());
+            vo.setPhoneNumber(e.getPhonenumber());
+            vo.setDept(deptMap.get(e.getDeptId()));
+            vo.setRole(roleService.selectRolesByUserId(e.getUserId()).get(0).getRoleName());
+            vo.setTime(new Date());
+            return vo;
+        }));
+    }
+}

+ 13 - 2
script/sql/business/create.sql

@@ -7,9 +7,14 @@ CREATE TABLE `project`
     `language`    char(1)                     NOT NULL DEFAULT '0' COMMENT '项目语言',
     `type`        char(1)                     NOT NULL DEFAULT '0' COMMENT '项目类型',
     `status`      tinyint(1)                  NOT NULL DEFAULT 0 COMMENT '状态',
+    `pd_gpd`      varchar(128) COMMENT 'PD/GPD',
+    `pm_gpm`      varchar(128) COMMENT 'PM/GPM',
+    `cta_gcta`    varchar(128) COMMENT 'CTA/GCTA',
     `sponsor`     varchar(128) COMMENT '申办方',
     `cro`         varchar(128) COMMENT 'CRO',
     `note`        varchar(255) COMMENT '备注',
+    `start_time`  datetime COMMENT '开始时间',
+    `end_time`    datetime COMMENT '结束时间',
     `create_dept` bigint(20) COMMENT '创建部门',
     `create_by`   bigint(20) COMMENT '创建者',
     `create_time` datetime COMMENT '创建时间',
@@ -20,13 +25,19 @@ CREATE TABLE `project`
 ) ENGINE = InnoDB
   DEFAULT CHARSET = utf8mb4 COMMENT ='项目';
 
+CREATE TABLE `sys_user_project`
+(
+    `user_id`   bigint NOT NULL PRIMARY KEY COMMENT '用户ID',
+    `projects` varchar(255) NOT NULL COMMENT '项目ID'
+) ENGINE = InnoDB
+  DEFAULT CHARSET = utf8mb4 COMMENT ='用户项目关系表';
+
 CREATE TABLE `folder`
 (
     `id`          bigint unsigned PRIMARY KEY NOT NULL AUTO_INCREMENT COMMENT '序号',
     `project_id`  bigint unsigned             NOT NULL COMMENT '所属项目',
     `parent_id`   bigint unsigned             NOT NULL COMMENT '父级',
-    `parent_type` tinyint(1)                  NOT NULL COMMENT '父级类型',
-    `type`        tinyint(1)                  NOT NULL COMMENT '层级类型',
+    `type`        tinyint(1)                  NOT NULL COMMENT '类型',
     `name`        varchar(128)                NOT NULL COMMENT '名称',
     `status`      tinyint(1)                  NOT NULL DEFAULT 0 COMMENT '状态',
     `note`        varchar(255) COMMENT '备注',