refactor: 通知公告重构

This commit is contained in:
Ray.Hao
2024-09-27 08:33:55 +08:00
parent ec180c0029
commit c25638f257
75 changed files with 750 additions and 805 deletions

View File

@@ -240,14 +240,14 @@ INSERT INTO `sys_menu` VALUES (123,120,'0,1,120','修改系统配置',4,NULL,'',
INSERT INTO `sys_menu` VALUES (124,120,'0,1,120','删除系统配置',4,NULL,'',NULL,'sys:config:delete',0,1,1,4,'',NULL,'2024-07-30 16:31:07','2024-07-30 16:31:07',NULL); INSERT INTO `sys_menu` VALUES (124,120,'0,1,120','删除系统配置',4,NULL,'',NULL,'sys:config:delete',0,1,1,4,'',NULL,'2024-07-30 16:31:07','2024-07-30 16:31:07',NULL);
INSERT INTO `sys_menu` VALUES (125,120,'0,1,120','刷新系统配置',4,NULL,'',NULL,'sys:config:refresh',0,1,1,5,'',NULL,'2024-07-30 16:31:25','2024-07-30 16:31:25',NULL); INSERT INTO `sys_menu` VALUES (125,120,'0,1,120','刷新系统配置',4,NULL,'',NULL,'sys:config:refresh',0,1,1,5,'',NULL,'2024-07-30 16:31:25','2024-07-30 16:31:25',NULL);
INSERT INTO `sys_menu` VALUES (126,1,'0,1','通知公告',1,'Notice','notice','system/notice/index',NULL,NULL,NULL,1,9,'',NULL,'2024-08-24 13:39:03','2024-08-24 13:39:03',NULL); INSERT INTO `sys_menu` VALUES (126,1,'0,1','通知公告',1,'Notice','notice','system/notice/index',NULL,NULL,NULL,1,9,'',NULL,'2024-08-24 13:39:03','2024-08-24 13:39:03',NULL);
INSERT INTO `sys_menu` VALUES (127,126,'0,1,132','查询',4,NULL,'',NULL,'system:notice:query',NULL,NULL,1,1,'',NULL,'2024-08-24 13:39:03','2024-08-24 13:39:03',NULL); INSERT INTO `sys_menu` VALUES (127,126,'0,1,132','查询',4,NULL,'',NULL,'sys:notice:query',NULL,NULL,1,1,'',NULL,'2024-08-24 13:39:03','2024-08-24 13:39:03',NULL);
INSERT INTO `sys_menu` VALUES (128,126,'0,1,133','新增',4,NULL,'',NULL,'system:notice:add',NULL,NULL,1,2,'',NULL,'2024-08-24 13:39:03','2024-08-24 13:39:03',NULL); INSERT INTO `sys_menu` VALUES (128,126,'0,1,133','新增',4,NULL,'',NULL,'sys:notice:add',NULL,NULL,1,2,'',NULL,'2024-08-24 13:39:03','2024-08-24 13:39:03',NULL);
INSERT INTO `sys_menu` VALUES (129,126,'0,1,134','编辑',4,NULL,'',NULL,'system:notice:edit',NULL,NULL,1,3,'',NULL,'2024-08-24 13:39:03','2024-08-24 13:39:03',NULL); INSERT INTO `sys_menu` VALUES (129,126,'0,1,134','编辑',4,NULL,'',NULL,'sys:notice:edit',NULL,NULL,1,3,'',NULL,'2024-08-24 13:39:03','2024-08-24 13:39:03',NULL);
INSERT INTO `sys_menu` VALUES (130,126,'0,1,135','删除',4,NULL,'',NULL,'system:notice:delete',NULL,NULL,1,4,'',NULL,'2024-08-24 13:39:03','2024-08-24 13:39:03',NULL); INSERT INTO `sys_menu` VALUES (130,126,'0,1,135','删除',4,NULL,'',NULL,'sys:notice:delete',NULL,NULL,1,4,'',NULL,'2024-08-24 13:39:03','2024-08-24 13:39:03',NULL);
INSERT INTO `sys_menu` VALUES (131,0,'0','消息中心',2,NULL,'/notice','Layout',NULL,1,1,1,13,'el-icon-Message',NULL,'2024-08-31 21:16:06','2024-08-31 21:16:41',NULL); INSERT INTO `sys_menu` VALUES (131,0,'0','消息中心',2,NULL,'/notice','Layout',NULL,1,1,1,13,'el-icon-Message',NULL,'2024-08-31 21:16:06','2024-08-31 21:16:41',NULL);
INSERT INTO `sys_menu` VALUES (132,131,'0,136','我的消息',1,'MyNotice','notice','notice/index',NULL,0,1,1,1,'',NULL,'2024-08-31 21:17:36','2024-09-12 11:28:42',NULL); INSERT INTO `sys_menu` VALUES (132,131,'0,136','我的消息',1,'MyNotice','notice','notice/index',NULL,0,1,1,1,'',NULL,'2024-08-31 21:17:36','2024-09-12 11:28:42',NULL);
INSERT INTO `sys_menu` VALUES (133,132,'0,1,131','发布',4,NULL,'',NULL,'system:notice:release',0,1,1,5,'',NULL,'2024-09-01 23:53:52','2024-09-01 23:53:52',NULL); INSERT INTO `sys_menu` VALUES (133,132,'0,1,131','发布',4,NULL,'',NULL,'sys:notice:publish',0,1,1,5,'',NULL,'2024-09-01 23:53:52','2024-09-01 23:53:52',NULL);
INSERT INTO `sys_menu` VALUES (134,132,'0,1,131','撤回',4,NULL,'',NULL,'system:notice:recall',0,1,1,6,'',NULL,'2024-09-01 23:54:16','2024-09-01 23:54:16',NULL); INSERT INTO `sys_menu` VALUES (134,132,'0,1,131','撤回',4,NULL,'',NULL,'sys:notice:revoke',0,1,1,6,'',NULL,'2024-09-01 23:54:16','2024-09-01 23:54:16',NULL);
-- ---------------------------- -- ----------------------------
-- Table structure for sys_message -- Table structure for sys_message
@@ -419,7 +419,7 @@ CREATE TABLE `sys_user` (
-- ---------------------------- -- ----------------------------
INSERT INTO `sys_user` VALUES (1, 'root', '有来技术', 0, '$2a$10$xVWsNOhHrCxh5UbpCE7/HuJ.PAOKcYAqRxD2CO2nVnJS.IAXkr5aq', NULL, 'https://oss.youlai.tech/youlai-boot/2023/05/16/811270ef31f548af9cffc026dfc3777b.gif', '17621590365', 1, 'youlaitech@163.com', NULL, NULL, NULL, NULL, 0); INSERT INTO `sys_user` VALUES (1, 'root', '有来技术', 0, '$2a$10$xVWsNOhHrCxh5UbpCE7/HuJ.PAOKcYAqRxD2CO2nVnJS.IAXkr5aq', NULL, 'https://oss.youlai.tech/youlai-boot/2023/05/16/811270ef31f548af9cffc026dfc3777b.gif', '17621590365', 1, 'youlaitech@163.com', NULL, NULL, NULL, NULL, 0);
INSERT INTO `sys_user` VALUES (2, 'admin', '系统管理员', 1, '$2a$10$xVWsNOhHrCxh5UbpCE7/HuJ.PAOKcYAqRxD2CO2nVnJS.IAXkr5aq', 1, 'https://oss.youlai.tech/youlai-boot/2023/05/16/811270ef31f548af9cffc026dfc3777b.gif', '17621210366', 1, '', '2019-10-10 13:41:22', NULL, '2022-07-31 12:39:30', NULL, 0); INSERT INTO `sys_user` VALUES (2, 'admin', '系统管理员', 1, '$2a$10$xVWsNOhHrCxh5UbpCE7/HuJ.PAOKcYAqRxD2CO2nVnJS.IAXkr5aq', 1, 'https://oss.youlai.tech/youlai-boot/2023/05/16/811270ef31f548af9cffc026dfc3777b.gif', '17621210366', 1, '', '2019-10-10 13:41:22', NULL, '2022-07-31 12:39:30', NULL, 0);
INSERT INTO `sys_user` VALUES (3, 'test', '测试小用户', 1, '$2a$10$xVWsNOhHrCxh5UbpCE7/HuJ.PAOKcYAqRxD2CO2nVnJS.IAXkr5aq', 3, 'https://oss.youlai.tech/youlai-boot/2023/05/16/811270ef31f548af9cffc026dfc3777b.gif', '17621210366', 1, 'youlaitech@163.com', '2021-06-05 01:31:29', NULL, '2021-06-05 01:31:29', NULL, 0); INSERT INTO `sys_user` VALUES (3, 'websocket', '测试小用户', 1, '$2a$10$xVWsNOhHrCxh5UbpCE7/HuJ.PAOKcYAqRxD2CO2nVnJS.IAXkr5aq', 3, 'https://oss.youlai.tech/youlai-boot/2023/05/16/811270ef31f548af9cffc026dfc3777b.gif', '17621210366', 1, 'youlaitech@163.com', '2021-06-05 01:31:29', NULL, '2021-06-05 01:31:29', NULL, 0);
-- ---------------------------- -- ----------------------------
-- Table structure for sys_user_role -- Table structure for sys_user_role

View File

@@ -228,14 +228,14 @@ INSERT INTO `sys_menu` VALUES (123,120,'0,1,120','修改系统配置',4,NULL,'',
INSERT INTO `sys_menu` VALUES (124,120,'0,1,120','删除系统配置',4,NULL,'',NULL,'sys:config:delete',0,1,1,4,'',NULL,'2024-07-30 16:31:07','2024-07-30 16:31:07',NULL); INSERT INTO `sys_menu` VALUES (124,120,'0,1,120','删除系统配置',4,NULL,'',NULL,'sys:config:delete',0,1,1,4,'',NULL,'2024-07-30 16:31:07','2024-07-30 16:31:07',NULL);
INSERT INTO `sys_menu` VALUES (125,120,'0,1,120','刷新系统配置',4,NULL,'',NULL,'sys:config:refresh',0,1,1,5,'',NULL,'2024-07-30 16:31:25','2024-07-30 16:31:25',NULL); INSERT INTO `sys_menu` VALUES (125,120,'0,1,120','刷新系统配置',4,NULL,'',NULL,'sys:config:refresh',0,1,1,5,'',NULL,'2024-07-30 16:31:25','2024-07-30 16:31:25',NULL);
INSERT INTO `sys_menu` VALUES (126,1,'0,1','通知公告',1,'Notice','notice','system/notice/index',NULL,NULL,NULL,1,9,'',NULL,'2024-08-24 13:39:03','2024-08-24 13:39:03',NULL); INSERT INTO `sys_menu` VALUES (126,1,'0,1','通知公告',1,'Notice','notice','system/notice/index',NULL,NULL,NULL,1,9,'',NULL,'2024-08-24 13:39:03','2024-08-24 13:39:03',NULL);
INSERT INTO `sys_menu` VALUES (127,126,'0,1,132','查询',4,NULL,'',NULL,'system:notice:query',NULL,NULL,1,1,'',NULL,'2024-08-24 13:39:03','2024-08-24 13:39:03',NULL); INSERT INTO `sys_menu` VALUES (127,126,'0,1,132','查询',4,NULL,'',NULL,'sys:notice:query',NULL,NULL,1,1,'',NULL,'2024-08-24 13:39:03','2024-08-24 13:39:03',NULL);
INSERT INTO `sys_menu` VALUES (128,126,'0,1,133','新增',4,NULL,'',NULL,'system:notice:add',NULL,NULL,1,2,'',NULL,'2024-08-24 13:39:03','2024-08-24 13:39:03',NULL); INSERT INTO `sys_menu` VALUES (128,126,'0,1,133','新增',4,NULL,'',NULL,'sys:notice:add',NULL,NULL,1,2,'',NULL,'2024-08-24 13:39:03','2024-08-24 13:39:03',NULL);
INSERT INTO `sys_menu` VALUES (129,126,'0,1,134','编辑',4,NULL,'',NULL,'system:notice:edit',NULL,NULL,1,3,'',NULL,'2024-08-24 13:39:03','2024-08-24 13:39:03',NULL); INSERT INTO `sys_menu` VALUES (129,126,'0,1,134','编辑',4,NULL,'',NULL,'sys:notice:edit',NULL,NULL,1,3,'',NULL,'2024-08-24 13:39:03','2024-08-24 13:39:03',NULL);
INSERT INTO `sys_menu` VALUES (130,126,'0,1,135','删除',4,NULL,'',NULL,'system:notice:delete',NULL,NULL,1,4,'',NULL,'2024-08-24 13:39:03','2024-08-24 13:39:03',NULL); INSERT INTO `sys_menu` VALUES (130,126,'0,1,135','删除',4,NULL,'',NULL,'sys:notice:delete',NULL,NULL,1,4,'',NULL,'2024-08-24 13:39:03','2024-08-24 13:39:03',NULL);
INSERT INTO `sys_menu` VALUES (131,0,'0','消息中心',2,NULL,'/notice','Layout',NULL,1,1,1,13,'el-icon-Message',NULL,'2024-08-31 21:16:06','2024-08-31 21:16:41',NULL); INSERT INTO `sys_menu` VALUES (131,0,'0','消息中心',2,NULL,'/notice','Layout',NULL,1,1,1,13,'el-icon-Message',NULL,'2024-08-31 21:16:06','2024-08-31 21:16:41',NULL);
INSERT INTO `sys_menu` VALUES (132,131,'0,136','我的消息',1,'MyNotice','notice','notice/index',NULL,0,1,1,1,'',NULL,'2024-08-31 21:17:36','2024-09-12 11:28:42',NULL); INSERT INTO `sys_menu` VALUES (132,131,'0,136','我的消息',1,'MyNotice','notice','notice/index',NULL,0,1,1,1,'',NULL,'2024-08-31 21:17:36','2024-09-12 11:28:42',NULL);
INSERT INTO `sys_menu` VALUES (133,132,'0,1,131','发布',4,NULL,'',NULL,'system:notice:release',0,1,1,5,'',NULL,'2024-09-01 23:53:52','2024-09-01 23:53:52',NULL); INSERT INTO `sys_menu` VALUES (133,132,'0,1,131','发布',4,NULL,'',NULL,'sys:notice:publish',0,1,1,5,'',NULL,'2024-09-01 23:53:52','2024-09-01 23:53:52',NULL);
INSERT INTO `sys_menu` VALUES (134,132,'0,1,131','撤回',4,NULL,'',NULL,'system:notice:recall',0,1,1,6,'',NULL,'2024-09-01 23:54:16','2024-09-01 23:54:16',NULL); INSERT INTO `sys_menu` VALUES (134,132,'0,1,131','撤回',4,NULL,'',NULL,'sys:notice:revoke',0,1,1,6,'',NULL,'2024-09-01 23:54:16','2024-09-01 23:54:16',NULL);
-- ---------------------------- -- ----------------------------
-- Table structure for sys_message -- Table structure for sys_message
@@ -407,7 +407,7 @@ CREATE TABLE `sys_user` (
-- ---------------------------- -- ----------------------------
INSERT INTO `sys_user` VALUES (1, 'root', '有来技术', 0, '$2a$10$xVWsNOhHrCxh5UbpCE7/HuJ.PAOKcYAqRxD2CO2nVnJS.IAXkr5aq', NULL, 'https://oss.youlai.tech/youlai-boot/2023/05/16/811270ef31f548af9cffc026dfc3777b.gif', '17621590365', 1, 'youlaitech@163.com', NULL, NULL, NULL, NULL, 0); INSERT INTO `sys_user` VALUES (1, 'root', '有来技术', 0, '$2a$10$xVWsNOhHrCxh5UbpCE7/HuJ.PAOKcYAqRxD2CO2nVnJS.IAXkr5aq', NULL, 'https://oss.youlai.tech/youlai-boot/2023/05/16/811270ef31f548af9cffc026dfc3777b.gif', '17621590365', 1, 'youlaitech@163.com', NULL, NULL, NULL, NULL, 0);
INSERT INTO `sys_user` VALUES (2, 'admin', '系统管理员', 1, '$2a$10$xVWsNOhHrCxh5UbpCE7/HuJ.PAOKcYAqRxD2CO2nVnJS.IAXkr5aq', 1, 'https://oss.youlai.tech/youlai-boot/2023/05/16/811270ef31f548af9cffc026dfc3777b.gif', '17621210366', 1, '', '2019-10-10 13:41:22', NULL, '2022-07-31 12:39:30', NULL, 0); INSERT INTO `sys_user` VALUES (2, 'admin', '系统管理员', 1, '$2a$10$xVWsNOhHrCxh5UbpCE7/HuJ.PAOKcYAqRxD2CO2nVnJS.IAXkr5aq', 1, 'https://oss.youlai.tech/youlai-boot/2023/05/16/811270ef31f548af9cffc026dfc3777b.gif', '17621210366', 1, '', '2019-10-10 13:41:22', NULL, '2022-07-31 12:39:30', NULL, 0);
INSERT INTO `sys_user` VALUES (3, 'test', '测试小用户', 1, '$2a$10$xVWsNOhHrCxh5UbpCE7/HuJ.PAOKcYAqRxD2CO2nVnJS.IAXkr5aq', 3, 'https://oss.youlai.tech/youlai-boot/2023/05/16/811270ef31f548af9cffc026dfc3777b.gif', '17621210366', 1, 'youlaitech@163.com', '2021-06-05 01:31:29', NULL, '2021-06-05 01:31:29', NULL, 0); INSERT INTO `sys_user` VALUES (3, 'websocket', '测试小用户', 1, '$2a$10$xVWsNOhHrCxh5UbpCE7/HuJ.PAOKcYAqRxD2CO2nVnJS.IAXkr5aq', 3, 'https://oss.youlai.tech/youlai-boot/2023/05/16/811270ef31f548af9cffc026dfc3777b.gif', '17621210366', 1, 'youlaitech@163.com', '2021-06-05 01:31:29', NULL, '2021-06-05 01:31:29', NULL, 0);
-- ---------------------------- -- ----------------------------
-- Table structure for sys_user_role -- Table structure for sys_user_role

View File

@@ -1,33 +0,0 @@
package com.youlai.boot.common.enums;
import com.youlai.boot.common.base.IBaseEnum;
import lombok.Getter;
import lombok.RequiredArgsConstructor;
/**
* 通知类型枚举
* 0-系统消息
*
* @since 2024-9-1 17:33:06
* @author Theo
*/
@Getter
@RequiredArgsConstructor
public enum NoticeTypeEnum implements IBaseEnum<Integer> {
/**
* 通知类型
*/
SYSTEM_MESSAGE(0, "系统消息");
@Getter
private Integer value;
@Getter
private String label;
NoticeTypeEnum(Integer value, String label) {
this.value = value;
this.label = label;
}
}

View File

@@ -1,30 +0,0 @@
package com.youlai.boot.common.enums;
import com.youlai.boot.common.base.IBaseEnum;
import lombok.Getter;
import lombok.RequiredArgsConstructor;
/**
* 通知方式枚举
* @author Theo
* @since 2024-9-2 14:32:58
*/
@Getter
@RequiredArgsConstructor
public enum NoticeWayEnum implements IBaseEnum<String> {
/**
* 通知方式
*/
WEBSOCKET("webSocket", "发送websocket消息");
@Getter
private String value;
@Getter
private String label;
NoticeWayEnum(String value, String label) {
this.value = value;
this.label = label;
}
}

View File

@@ -9,16 +9,16 @@ import lombok.Getter;
* @author haoxr * @author haoxr
* @since 2022/10/14 * @since 2022/10/14
*/ */
@Getter
public enum StatusEnum implements IBaseEnum<Integer> { public enum StatusEnum implements IBaseEnum<Integer> {
ENABLE(1, "启用"), ENABLE(1, "启用"),
DISABLE (0, "禁用"); DISABLE (0, "禁用");
@Getter private final Integer value;
private Integer value;
@Getter
private String label; private final String label;
StatusEnum(Integer value, String label) { StatusEnum(Integer value, String label) {
this.value = value; this.value = value;

View File

@@ -1,67 +0,0 @@
package com.youlai.boot.common.util;
import com.youlai.boot.common.constant.SymbolConstant;
import java.util.List;
import java.util.stream.Collectors;
/**
* 通用工具类
*
* @author Theo
* @since 2024-9-1 23:42:33
* @version 1.0.0
*/
public class CommonUtil {
private CommonUtil(){}
/**
* 将List转换为字符串
*
* @param list List
* @param separator 分隔符
* @return 字符串
*/
public static String listToStr(List<String> list, String separator) {
return list.stream().collect(Collectors.joining(separator));
}
/**
* 将字符串转换为List
*
* @param list List
* @return List
*/
public static String listToStr(List<String> list) {
return listToStr(list, SymbolConstant.COMMA);
}
/**
* 将字符串转换为List
*
* @param str 字符串
* @return List
*/
public static List<String> strToList(String str) {
return strToList(str, SymbolConstant.COMMA);
}
/**
* 将字符串转换为List
*
* @param str 字符串
* @param separator 分隔符
* @return List
*/
public static List<String> strToList(String str, String separator) {
return List.of(str.split(separator));
}
public static String delHtmlTags(String htmlStr) {
return htmlStr.replaceAll("<[^>]+>", "");
}
}

View File

@@ -1,13 +1,11 @@
package com.youlai.boot.module.auth.controller; package com.youlai.boot.module.auth.controller;
import com.youlai.boot.common.constant.RedisConstants;
import com.youlai.boot.common.enums.LogModuleEnum; import com.youlai.boot.common.enums.LogModuleEnum;
import com.youlai.boot.common.result.Result; import com.youlai.boot.common.result.Result;
import com.youlai.boot.module.auth.service.AuthService; import com.youlai.boot.module.auth.service.AuthService;
import com.youlai.boot.system.model.dto.CaptchaResult; import com.youlai.boot.system.model.dto.CaptchaResult;
import com.youlai.boot.system.model.dto.LoginResult; import com.youlai.boot.system.model.dto.LoginResult;
import com.youlai.boot.common.annotation.Log; import com.youlai.boot.common.annotation.Log;
import com.youlai.boot.system.service.ConfigService;
import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.Operation;
import io.swagger.v3.oas.annotations.Parameter; import io.swagger.v3.oas.annotations.Parameter;
import io.swagger.v3.oas.annotations.tags.Tag; import io.swagger.v3.oas.annotations.tags.Tag;

View File

@@ -1,4 +1,4 @@
package com.youlai.boot.common.enums; package com.youlai.boot.module.auth.enums;
/** /**
* EasyCaptcha 验证码类型枚举 * EasyCaptcha 验证码类型枚举

View File

@@ -9,7 +9,7 @@ import cn.hutool.json.JSONObject;
import cn.hutool.jwt.JWTPayload; import cn.hutool.jwt.JWTPayload;
import cn.hutool.jwt.JWTUtil; import cn.hutool.jwt.JWTUtil;
import com.youlai.boot.common.constant.SecurityConstants; import com.youlai.boot.common.constant.SecurityConstants;
import com.youlai.boot.common.enums.CaptchaTypeEnum; import com.youlai.boot.module.auth.enums.CaptchaTypeEnum;
import com.youlai.boot.module.auth.service.AuthService; import com.youlai.boot.module.auth.service.AuthService;
import com.youlai.boot.system.model.dto.CaptchaResult; import com.youlai.boot.system.model.dto.CaptchaResult;
import com.youlai.boot.system.model.dto.LoginResult; import com.youlai.boot.system.model.dto.LoginResult;

View File

@@ -1,4 +1,4 @@
package com.youlai.boot.common.enums; package com.youlai.boot.module.codegen.enums;
import com.baomidou.mybatisplus.annotation.EnumValue; import com.baomidou.mybatisplus.annotation.EnumValue;
import com.fasterxml.jackson.annotation.JsonCreator; import com.fasterxml.jackson.annotation.JsonCreator;

View File

@@ -1,4 +1,4 @@
package com.youlai.boot.common.enums; package com.youlai.boot.module.codegen.enums;
import lombok.Getter; import lombok.Getter;

View File

@@ -1,4 +1,4 @@
package com.youlai.boot.common.enums; package com.youlai.boot.module.codegen.enums;
import com.baomidou.mybatisplus.annotation.EnumValue; import com.baomidou.mybatisplus.annotation.EnumValue;
import com.fasterxml.jackson.annotation.JsonCreator; import com.fasterxml.jackson.annotation.JsonCreator;

View File

@@ -5,8 +5,8 @@ import com.baomidou.mybatisplus.annotation.TableName;
import com.fasterxml.jackson.annotation.JsonIgnore; import com.fasterxml.jackson.annotation.JsonIgnore;
import com.youlai.boot.common.base.BaseEntity; import com.youlai.boot.common.base.BaseEntity;
import com.youlai.boot.common.enums.FormTypeEnum; import com.youlai.boot.module.codegen.enums.FormTypeEnum;
import com.youlai.boot.common.enums.QueryTypeEnum; import com.youlai.boot.module.codegen.enums.QueryTypeEnum;
import lombok.Getter; import lombok.Getter;
import lombok.Setter; import lombok.Setter;

View File

@@ -1,7 +1,7 @@
package com.youlai.boot.module.codegen.model.form; package com.youlai.boot.module.codegen.model.form;
import com.youlai.boot.common.enums.FormTypeEnum; import com.youlai.boot.module.codegen.enums.FormTypeEnum;
import com.youlai.boot.common.enums.QueryTypeEnum; import com.youlai.boot.module.codegen.enums.QueryTypeEnum;
import io.swagger.v3.oas.annotations.media.Schema; import io.swagger.v3.oas.annotations.media.Schema;
import lombok.Data; import lombok.Data;

View File

@@ -10,12 +10,11 @@ import cn.hutool.extra.template.TemplateEngine;
import cn.hutool.extra.template.TemplateUtil; import cn.hutool.extra.template.TemplateUtil;
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page; import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.youlai.boot.common.enums.JavaTypeEnum; import com.youlai.boot.module.codegen.enums.JavaTypeEnum;
import com.youlai.boot.config.property.CodegenProperties; import com.youlai.boot.config.property.CodegenProperties;
import com.youlai.boot.module.codegen.service.GenConfigService; import com.youlai.boot.module.codegen.service.GenConfigService;
import com.youlai.boot.module.codegen.service.GenFieldConfigService; import com.youlai.boot.module.codegen.service.GenFieldConfigService;
import com.youlai.boot.module.codegen.service.CodegenService; import com.youlai.boot.module.codegen.service.CodegenService;
import com.youlai.boot.module.codegen.converter.CodegenConverter;
import com.youlai.boot.common.exception.BusinessException; import com.youlai.boot.common.exception.BusinessException;
import com.youlai.boot.module.codegen.mapper.DatabaseMapper; import com.youlai.boot.module.codegen.mapper.DatabaseMapper;
import com.youlai.boot.module.codegen.model.entity.GenConfig; import com.youlai.boot.module.codegen.model.entity.GenConfig;
@@ -23,10 +22,8 @@ import com.youlai.boot.module.codegen.model.entity.GenFieldConfig;
import com.youlai.boot.module.codegen.model.query.TablePageQuery; import com.youlai.boot.module.codegen.model.query.TablePageQuery;
import com.youlai.boot.module.codegen.model.vo.CodegenPreviewVO; import com.youlai.boot.module.codegen.model.vo.CodegenPreviewVO;
import com.youlai.boot.module.codegen.model.vo.TablePageVO; import com.youlai.boot.module.codegen.model.vo.TablePageVO;
import com.youlai.boot.system.service.MenuService;
import lombok.RequiredArgsConstructor; import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j; import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Service; import org.springframework.stereotype.Service;
import java.io.ByteArrayOutputStream; import java.io.ByteArrayOutputStream;
@@ -224,7 +221,7 @@ public class CodegenServiceImpl implements CodegenService {
bindMap.put("tableName", genConfig.getTableName()); bindMap.put("tableName", genConfig.getTableName());
bindMap.put("author", genConfig.getAuthor()); bindMap.put("author", genConfig.getAuthor());
bindMap.put("lowerFirstEntityName", StrUtil.lowerFirst(entityName)); // UserTest → userTest bindMap.put("lowerFirstEntityName", StrUtil.lowerFirst(entityName)); // UserTest → userTest
bindMap.put("kebabCaseEntityName", StrUtil.toSymbolCase(entityName, '-')); // UserTest → user-test bindMap.put("kebabCaseEntityName", StrUtil.toSymbolCase(entityName, '-')); // UserTest → user-websocket
bindMap.put("businessName", genConfig.getBusinessName()); bindMap.put("businessName", genConfig.getBusinessName());
bindMap.put("fieldConfigs", fieldConfigs); bindMap.put("fieldConfigs", fieldConfigs);

View File

@@ -7,9 +7,9 @@ import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
import com.youlai.boot.YouLaiApplication; import com.youlai.boot.YouLaiApplication;
import com.youlai.boot.common.enums.EnvEnum; import com.youlai.boot.common.enums.EnvEnum;
import com.youlai.boot.common.enums.FormTypeEnum; import com.youlai.boot.module.codegen.enums.FormTypeEnum;
import com.youlai.boot.common.enums.JavaTypeEnum; import com.youlai.boot.module.codegen.enums.JavaTypeEnum;
import com.youlai.boot.common.enums.QueryTypeEnum; import com.youlai.boot.module.codegen.enums.QueryTypeEnum;
import com.youlai.boot.common.exception.BusinessException; import com.youlai.boot.common.exception.BusinessException;
import com.youlai.boot.config.property.CodegenProperties; import com.youlai.boot.config.property.CodegenProperties;
import com.youlai.boot.module.codegen.converter.CodegenConverter; import com.youlai.boot.module.codegen.converter.CodegenConverter;

View File

@@ -138,10 +138,10 @@ public class MinioFileService implements FileService {
try { try {
String fileName; String fileName;
if (StrUtil.isNotBlank(customDomain)) { if (StrUtil.isNotBlank(customDomain)) {
// https://oss.youlai.tech/default/20221120/test.jpg → 20221120/test.jpg // https://oss.youlai.tech/default/20221120/test.jpg → 20221120/websocket.jpg
fileName = filePath.substring(customDomain.length() + 1 + bucketName.length() + 1); // 两个/占了2个字符长度 fileName = filePath.substring(customDomain.length() + 1 + bucketName.length() + 1); // 两个/占了2个字符长度
} else { } else {
// http://localhost:9000/default/20221120/test.jpg → 20221120/test.jpg // http://localhost:9000/default/20221120/test.jpg → 20221120/websocket.jpg
fileName = filePath.substring(endpoint.length() + 1 + bucketName.length() + 1); fileName = filePath.substring(endpoint.length() + 1 + bucketName.length() + 1);
} }
RemoveObjectArgs removeObjectArgs = RemoveObjectArgs.builder() RemoveObjectArgs removeObjectArgs = RemoveObjectArgs.builder()

View File

@@ -1,6 +1,5 @@
package com.youlai.boot.module.websocket.controller; package com.youlai.boot.module.websocket.controller;
import com.youlai.boot.common.enums.NoticeTypeEnum;
import com.youlai.boot.system.model.dto.ChatMessage; import com.youlai.boot.system.model.dto.ChatMessage;
import lombok.RequiredArgsConstructor; import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j; import lombok.extern.slf4j.Slf4j;
@@ -14,7 +13,9 @@ import org.springframework.web.bind.annotation.RestController;
import java.security.Principal; import java.security.Principal;
/** /**
* WebSocket 测试控制层 * WebSocket 测试用例控制层
* <p>
* 包含点对点/广播发送消息
* *
* @author Ray * @author Ray
* @since 2.3.0 * @since 2.3.0
@@ -28,7 +29,6 @@ public class WebsocketController {
private final SimpMessagingTemplate messagingTemplate; private final SimpMessagingTemplate messagingTemplate;
/** /**
* 广播发送消息 * 广播发送消息
* *
@@ -58,7 +58,7 @@ public class WebsocketController {
log.info("发送人:{}; 接收人:{}", sender, receiver); log.info("发送人:{}; 接收人:{}", sender, receiver);
// 发送消息给指定用户,拼接后路径 /user/{receiver}/queue/greeting // 发送消息给指定用户,拼接后路径 /user/{receiver}/queue/greeting
messagingTemplate.convertAndSendToUser(receiver, "/queue/greeting", new ChatMessage(sender, message, NoticeTypeEnum.SYSTEM_MESSAGE)); messagingTemplate.convertAndSendToUser(receiver, "/queue/greeting", new ChatMessage(sender, message));
} }
} }

View File

@@ -0,0 +1,44 @@
package com.youlai.boot.module.websocket.listener;
import com.youlai.boot.module.websocket.service.OnlineUserService;
import com.youlai.boot.system.event.UserConnectionEvent;
import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j;
import org.springframework.context.event.EventListener;
import org.springframework.messaging.simp.SimpMessagingTemplate;
import org.springframework.stereotype.Component;
/**
* 在线用户监听器
*
* @author haoxr
* @since 2024/9/25
*/
@Component
@RequiredArgsConstructor
@Slf4j
public class OnlineUserListener {
private final SimpMessagingTemplate messagingTemplate;
private final OnlineUserService onlineUserService;
/**
* 用户连接事件处理
*
* @param event 用户连接事件
*/
@EventListener
public void handleUserConnectionEvent(UserConnectionEvent event) {
String username = event.getUsername();
if (event.isConnected()) {
onlineUserService.addOnlineUser(username);
log.info("User connected: {}", username);
} else {
onlineUserService.removeOnlineUser(username);
log.info("User disconnected: {}", username);
}
// 推送在线用户人数
messagingTemplate.convertAndSend("/topic/onlineUserCount", onlineUserService.getOnlineUserCount());
}
}

View File

@@ -1,29 +0,0 @@
package com.youlai.boot.module.websocket.service;
import com.youlai.boot.common.enums.NoticeWayEnum;
import com.youlai.boot.system.model.dto.MessageDTO;
/**
* 消息服务接口
*
* @author Theo
* @since 2024-9-2 14:32:58
*/
public interface MessageService {
/**
* 检查消息类型
*
* @param messageType 消息类型
* @return 是否支持
*/
boolean check(NoticeWayEnum messageType);
/**
* 发送消息
*
* @param message 消息
*/
void sendMessage(MessageDTO message);
}

View File

@@ -0,0 +1,70 @@
package com.youlai.boot.module.websocket.service;
import org.springframework.stereotype.Service;
import java.util.Collections;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import java.util.stream.Collectors;
/**
* 在线用户服务
*
* @author haoxr
* @since 2024/9/26
*/
@Service
public class OnlineUserService {
private final Set<String> onlineUsers = ConcurrentHashMap.newKeySet();
/**
* 添加用户到在线用户集合
*
* @param username 用户名
*/
public void addOnlineUser(String username) {
onlineUsers.add(username);
}
/**
* 从在线用户集合移除用户
*
* @param username 用户名
*/
public void removeOnlineUser(String username) {
onlineUsers.remove(username);
}
/**
* 获取所有在线用户
*
* @return 在线用户集合
*/
public Set<String> getAllOnlineUsers() {
return Collections.unmodifiableSet(onlineUsers);
}
/**
* 获取在线的接收者
* 从所有接收者中过滤出在线的接收者
*
* @param receivers 接收者
* @return 在线的接收者集合
*/
public Set<String> getOnlineReceivers(Set<String> receivers) {
return receivers.stream().filter(onlineUsers::contains).collect(Collectors.toSet());
}
/**
* 获取在线用户数量
*
* @return 在线用户数量
*/
public int getOnlineUserCount() {
return onlineUsers.size();
}
}

View File

@@ -1,96 +0,0 @@
package com.youlai.boot.module.websocket.service.impl;
import com.youlai.boot.common.enums.NoticeWayEnum;
import com.youlai.boot.common.enums.NoticeTypeEnum;
import com.youlai.boot.module.websocket.service.MessageService;
import com.youlai.boot.system.event.UserConnectionEvent;
import com.youlai.boot.system.model.dto.ChatMessage;
import com.youlai.boot.system.model.dto.MessageDTO;
import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j;
import org.springframework.context.event.EventListener;
import org.springframework.messaging.simp.SimpMessagingTemplate;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Service;
import java.util.ArrayList;
import java.util.List;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import java.util.stream.Collectors;
/**
* WebSocket消息服务实现类
*
* @author ray
* @since 2024-9-2 14:32:58
*/
@Service
@Slf4j
@RequiredArgsConstructor
public class WebsocketServiceImpl implements MessageService {
private final SimpMessagingTemplate messagingTemplate;
private final Set<String> onlineUsers = ConcurrentHashMap.newKeySet();
/**
* 用户连接事件处理
*
* @param event 用户连接事件
*/
@EventListener
public void handleUserConnectionEvent(UserConnectionEvent event) {
String username = event.getUsername();
if (event.isConnected()) {
onlineUsers.add(username);
log.info("User connected: {}", username);
} else {
onlineUsers.remove(username);
log.info("User disconnected: {}", username);
}
// 推送在线用户人数
messagingTemplate.convertAndSend("/topic/onlineUserCount", onlineUsers.size());
}
/**
* 定时推送在线用户人数
*/
@Scheduled(fixedRate = 5000)
public void sendOnlineUserCount() {
messagingTemplate.convertAndSend("/topic/onlineUserCount", onlineUsers.size());
}
/**
* 策略模式检查
*
* @param noticeWayEnum 通知方式
* @return boolean 是否支持
*/
@Override
public boolean check(NoticeWayEnum noticeWayEnum) {
return noticeWayEnum.equals(NoticeWayEnum.WEBSOCKET);
}
/**
* 发送消息
*
* @param message 消息
*/
@Override
public void sendMessage(MessageDTO message) {
List<String> users = null;
if(message.getReceiver() == null || message.getReceiver().isEmpty()){
// 发送给所有在线用户 离线用户不发送,因为离线用户下次登录会直接查询未读消息
users = new ArrayList<>(onlineUsers);
}else{
users = message.getReceiver().stream().filter(onlineUsers::contains).collect(Collectors.toList());
}
//获取当前用户
ChatMessage chatMessage = new ChatMessage(message.getSender(), message.getContent(), NoticeTypeEnum.SYSTEM_MESSAGE);
users.forEach(receiver -> {
messagingTemplate.convertAndSendToUser(receiver, "/queue/message", chatMessage);
});
}
}

View File

@@ -4,11 +4,12 @@ import com.baomidou.mybatisplus.core.metadata.IPage;
import com.youlai.boot.common.result.PageResult; import com.youlai.boot.common.result.PageResult;
import com.youlai.boot.common.result.Result; import com.youlai.boot.common.result.Result;
import com.youlai.boot.system.model.form.NoticeForm; import com.youlai.boot.system.model.form.NoticeForm;
import com.youlai.boot.system.model.query.NoticeQuery; import com.youlai.boot.system.model.query.NoticePageQuery;
import com.youlai.boot.system.model.vo.NoticeStatusVO; import com.youlai.boot.system.model.vo.NoticeDetailVO;
import com.youlai.boot.system.model.vo.UserNoticePageVO;
import com.youlai.boot.system.model.vo.NoticeVO; import com.youlai.boot.system.model.vo.NoticeVO;
import com.youlai.boot.system.service.NoticeService; import com.youlai.boot.system.service.NoticeService;
import com.youlai.boot.system.service.NoticeStatusService; import com.youlai.boot.system.service.UserNoticeService;
import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.Operation;
import io.swagger.v3.oas.annotations.Parameter; import io.swagger.v3.oas.annotations.Parameter;
import io.swagger.v3.oas.annotations.tags.Tag; import io.swagger.v3.oas.annotations.tags.Tag;
@@ -18,6 +19,8 @@ import org.springframework.security.access.prepost.PreAuthorize;
import org.springframework.validation.annotation.Validated; import org.springframework.validation.annotation.Validated;
import org.springframework.web.bind.annotation.*; import org.springframework.web.bind.annotation.*;
import java.util.List;
/** /**
* 通知公告前端控制层 * 通知公告前端控制层
* *
@@ -32,19 +35,19 @@ public class NoticeController {
private final NoticeService noticeService; private final NoticeService noticeService;
private final NoticeStatusService noticeStatusService; private final UserNoticeService userNoticeService;
@Operation(summary = "通知公告分页列表") @Operation(summary = "通知公告分页列表")
@GetMapping("/page") @GetMapping("/page")
@PreAuthorize("@ss.hasPerm('system:notice:query')") @PreAuthorize("@ss.hasPerm('sys:notice:query')")
public PageResult<NoticeVO> getNoticePage(NoticeQuery queryParams ) { public PageResult<NoticeVO> getNoticePage(NoticePageQuery queryParams) {
IPage<NoticeVO> result = noticeService.getNoticePage(queryParams); IPage<NoticeVO> result = noticeService.getNoticePage(queryParams);
return PageResult.success(result); return PageResult.success(result);
} }
@Operation(summary = "新增通知公告") @Operation(summary = "新增通知公告")
@PostMapping @PostMapping
@PreAuthorize("@ss.hasPerm('system:notice:add')") @PreAuthorize("@ss.hasPerm('sys:notice:add')")
public Result<?> saveNotice(@RequestBody @Valid NoticeForm formData) { public Result<?> saveNotice(@RequestBody @Valid NoticeForm formData) {
boolean result = noticeService.saveNotice(formData); boolean result = noticeService.saveNotice(formData);
return Result.judge(result); return Result.judge(result);
@@ -52,24 +55,27 @@ public class NoticeController {
@Operation(summary = "获取通知公告表单数据") @Operation(summary = "获取通知公告表单数据")
@GetMapping("/{id}/form") @GetMapping("/{id}/form")
@PreAuthorize("@ss.hasPerm('system:notice:edit')") @PreAuthorize("@ss.hasPerm('sys:notice:edit')")
public Result<NoticeForm> getNoticeForm( public Result<NoticeForm> getNoticeForm(
@Parameter(description = "通知公告ID") @PathVariable Long id @Parameter(description = "通知公告ID") @PathVariable Long id
) { ) {
NoticeForm formData = noticeService.getNoticeFormData(id); NoticeForm formData = noticeService.getNoticeFormData(id);
return Result.success(formData); return Result.success(formData);
} }
@Operation(summary = "管理页面查看通知公告")
@GetMapping("/detail/{id}") @Operation(summary = "阅读获取通知公告详情")
public Result<?> getReadNoticeDetail( @GetMapping("/{id}/detail")
@Parameter(description = "通知公告ID")@PathVariable Long id) { public Result<NoticeDetailVO> getNoticeDetail(
return Result.success(noticeService.getReadNoticeDetail(id)); @Parameter(description = "通知公告ID") @PathVariable Long id
) {
NoticeDetailVO detailVO = noticeService.getNoticeDetail(id);
return Result.success(detailVO);
} }
@Operation(summary = "修改通知公告") @Operation(summary = "修改通知公告")
@PutMapping(value = "/{id}") @PutMapping(value = "/{id}")
@PreAuthorize("@ss.hasPerm('system:notice:edit')") @PreAuthorize("@ss.hasPerm('sys:notice:edit')")
public Result<?> updateNotice( public Result<Void> updateNotice(
@Parameter(description = "通知公告ID") @PathVariable Long id, @Parameter(description = "通知公告ID") @PathVariable Long id,
@RequestBody @Validated NoticeForm formData @RequestBody @Validated NoticeForm formData
) { ) {
@@ -78,25 +84,29 @@ public class NoticeController {
} }
@Operation(summary = "发布通知公告") @Operation(summary = "发布通知公告")
@PatchMapping(value = "/release/{id}") @PatchMapping(value = "/{id}/publish")
@PreAuthorize("@ss.hasPerm('system:notice:release')") @PreAuthorize("@ss.hasPerm('sys:notice:publish')")
public Result<?> releaseNotice(@Parameter(description = "通知公告ID") @PathVariable Long id) { public Result<Void> publishNotice(
boolean result = noticeService.releaseNotice(id); @Parameter(description = "通知公告ID") @PathVariable Long id
) {
boolean result = noticeService.publishNotice(id);
return Result.judge(result); return Result.judge(result);
} }
@Operation(summary = "撤回通知公告") @Operation(summary = "撤回通知公告")
@PatchMapping(value = "/recall/{id}") @PatchMapping(value = "/{id}/revoke")
@PreAuthorize("@ss.hasPerm('system:notice:recall')") @PreAuthorize("@ss.hasPerm('sys:notice:revoke')")
public Result<?> recallNotice(@Parameter(description = "通知公告ID") @PathVariable Long id) { public Result<Void> revokeNotice(
boolean result = noticeService.recallNotice(id); @Parameter(description = "通知公告ID") @PathVariable Long id
) {
boolean result = noticeService.revokeNotice(id);
return Result.judge(result); return Result.judge(result);
} }
@Operation(summary = "删除通知公告") @Operation(summary = "删除通知公告")
@DeleteMapping("/{ids}") @DeleteMapping("/{ids}")
@PreAuthorize("@ss.hasPerm('system:notice:delete')") @PreAuthorize("@ss.hasPerm('sys:notice:delete')")
public Result<?> deleteNotices( public Result<Void> deleteNotices(
@Parameter(description = "通知公告ID多个以英文逗号(,)分割") @PathVariable String ids @Parameter(description = "通知公告ID多个以英文逗号(,)分割") @PathVariable String ids
) { ) {
boolean result = noticeService.deleteNotices(ids); boolean result = noticeService.deleteNotices(ids);
@@ -105,27 +115,22 @@ public class NoticeController {
@Operation(summary = "获取未读的通知公告") @Operation(summary = "获取未读的通知公告")
@GetMapping("/unread") @GetMapping("/unread")
public Result<?> listUnreadNotices() { public Result<List<UserNoticePageVO>> listUnreadNotices() {
return Result.success(noticeStatusService.listUnreadNotices()); List<UserNoticePageVO> list = userNoticeService.listUnreadNotices();
} return Result.success(list);
@Operation(summary = "阅读通知公告")
@PatchMapping("/read/{id}")
public Result<?> readNotice(@PathVariable Long id) {
return Result.success(noticeService.readNotice(id));
} }
@Operation(summary = "全部已读") @Operation(summary = "全部已读")
@PatchMapping("/readAll") @PatchMapping("/read-all")
public Result<?> readAll() { public Result<Void> readAll() {
noticeStatusService.readAll(); userNoticeService.readAll();
return Result.success(); return Result.success();
} }
@Operation(summary = "获取我的通知公告") @Operation(summary = "获取我的通知公告分页列表")
@GetMapping("/my/page") @GetMapping("/my/page")
public PageResult<NoticeStatusVO> getMyNoticePage(NoticeQuery queryParams) { public PageResult<UserNoticePageVO> getMyNoticePage(NoticePageQuery queryParams) {
IPage<NoticeStatusVO> result = noticeService.getMyNoticePage(queryParams); IPage<UserNoticePageVO> result = noticeService.getMyNoticePage(queryParams);
return PageResult.success(result); return PageResult.success(result);
} }
} }

View File

@@ -6,7 +6,7 @@ import com.baomidou.mybatisplus.core.conditions.update.LambdaUpdateWrapper;
import com.baomidou.mybatisplus.core.metadata.IPage; import com.baomidou.mybatisplus.core.metadata.IPage;
import com.youlai.boot.common.annotation.Log; import com.youlai.boot.common.annotation.Log;
import com.youlai.boot.common.annotation.RepeatSubmit; import com.youlai.boot.common.annotation.RepeatSubmit;
import com.youlai.boot.common.enums.ContactType; import com.youlai.boot.system.enums.ContactType;
import com.youlai.boot.common.enums.LogModuleEnum; import com.youlai.boot.common.enums.LogModuleEnum;
import com.youlai.boot.common.model.Option; import com.youlai.boot.common.model.Option;
import com.youlai.boot.common.result.PageResult; import com.youlai.boot.common.result.PageResult;

View File

@@ -1,9 +1,11 @@
package com.youlai.boot.system.converter; package com.youlai.boot.system.converter;
import cn.hutool.core.util.StrUtil;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page; import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.youlai.boot.system.model.bo.NoticeBO; import com.youlai.boot.system.model.bo.NoticeBO;
import com.youlai.boot.system.model.entity.Notice; import com.youlai.boot.system.model.entity.Notice;
import com.youlai.boot.system.model.form.NoticeForm; import com.youlai.boot.system.model.form.NoticeForm;
import com.youlai.boot.system.model.vo.NoticeDetailVO;
import com.youlai.boot.system.model.vo.NoticeVO; import com.youlai.boot.system.model.vo.NoticeVO;
import org.mapstruct.Mapper; import org.mapstruct.Mapper;
import org.mapstruct.Mapping; import org.mapstruct.Mapping;
@@ -18,22 +20,20 @@ import org.mapstruct.Mappings;
@Mapper(componentModel = "spring") @Mapper(componentModel = "spring")
public interface NoticeConverter{ public interface NoticeConverter{
@Mappings({ @Mappings({
@Mapping(target = "tarIds", expression = "java(com.youlai.boot.common.util.CommonUtil.strToList(entity.getTarIds()))") @Mapping(target = "targetUserIds", expression = "java(cn.hutool.core.util.StrUtil.split(entity.getTargetUserIds(),\",\"))")
}) })
NoticeForm toForm(Notice entity); NoticeForm toForm(Notice entity);
@Mappings({ @Mappings({
@Mapping(target = "tarIds", expression = "java(com.youlai.boot.common.util.CommonUtil.listToStr(formData.getTarIds()))") @Mapping(target = "targetUserIds", expression = "java(cn.hutool.core.collection.CollUtil.join(formData.getTargetUserIds(),\",\"))")
}) })
Notice toEntity(NoticeForm formData); Notice toEntity(NoticeForm formData);
NoticeVO toVO(Notice notice); NoticeVO toPageVo(NoticeBO bo);
Page<NoticeVO> toPageVo(Page<NoticeBO> noticePage); Page<NoticeVO> toPageVo(Page<NoticeBO> noticePage);
@Mappings({ NoticeDetailVO toDetailVO(NoticeBO noticeBO);
})
NoticeVO toPageVo(NoticeBO bo);
} }

View File

@@ -24,7 +24,7 @@ import org.mapstruct.Mappings;
public interface UserConverter { public interface UserConverter {
@Mappings({ @Mappings({
@Mapping(target = "genderLabel", expression = "java(com.youlai.boot.common.base.IBaseEnum.getLabelByValue(bo.getGender(), com.youlai.boot.common.enums.GenderEnum.class))") @Mapping(target = "genderLabel", expression = "java(com.youlai.boot.common.base.IBaseEnum.getLabelByValue(bo.getGender(), com.youlai.boot.system.enums.GenderEnum.class))")
}) })
UserPageVO toPageVo(UserBO bo); UserPageVO toPageVo(UserBO bo);

View File

@@ -1,4 +1,4 @@
package com.youlai.boot.common.enums; package com.youlai.boot.system.enums;
/** /**
* 联系方式类型 * 联系方式类型

View File

@@ -1,4 +1,4 @@
package com.youlai.boot.common.enums; package com.youlai.boot.system.enums;
import com.youlai.boot.common.base.IBaseEnum; import com.youlai.boot.common.base.IBaseEnum;
import io.swagger.v3.oas.annotations.media.Schema; import io.swagger.v3.oas.annotations.media.Schema;

View File

@@ -1,4 +1,4 @@
package com.youlai.boot.common.enums; package com.youlai.boot.system.enums;
import com.baomidou.mybatisplus.annotation.EnumValue; import com.baomidou.mybatisplus.annotation.EnumValue;
import com.youlai.boot.common.base.IBaseEnum; import com.youlai.boot.common.base.IBaseEnum;

View File

@@ -0,0 +1,30 @@
package com.youlai.boot.system.enums;
import com.youlai.boot.common.base.IBaseEnum;
import io.swagger.v3.oas.annotations.media.Schema;
import lombok.Getter;
/**
* 通告发布状态枚举
*
* @author haoxr
* @since 2022/10/14
*/
@Getter
@Schema(enumAsRef = true)
public enum NoticePublishStatusEnum implements IBaseEnum<Integer> {
UNPUBLISHED(0, "未发布"),
PUBLISHED(1, "已发布"),
REVOKED(-1, "已撤回");
private final Integer value;
private final String label;
NoticePublishStatusEnum(Integer value, String label) {
this.value = value;
this.label = label;
}
}

View File

@@ -0,0 +1,29 @@
package com.youlai.boot.system.enums;
import com.youlai.boot.common.base.IBaseEnum;
import io.swagger.v3.oas.annotations.media.Schema;
import lombok.Getter;
/**
* 通知目标类型枚举
*
* @author haoxr
* @since 2022/10/14
*/
@Getter
@Schema(enumAsRef = true)
public enum NoticeTargetTypeEnum implements IBaseEnum<Integer> {
ALL(1, "全体"),
SPECIFIED(2, "指定");
private final Integer value;
private final String label;
NoticeTargetTypeEnum(Integer value, String label) {
this.value = value;
this.label = label;
}
}

View File

@@ -1,35 +0,0 @@
package com.youlai.boot.system.handler;
import com.youlai.boot.module.websocket.service.MessageService;
import com.youlai.boot.system.model.dto.MessageDTO;
import lombok.RequiredArgsConstructor;
import org.springframework.stereotype.Component;
import java.util.List;
/**
* 消息处理器
*
* @author Theo
* @since 2024-9-2 14:32:58
*/
@Component
@RequiredArgsConstructor
public class MessageHandler {
private final List<MessageService> messageServices;
/**
* 发送消息
* 如果后面有多种消息发送方式可以设置MessageDTO中的noticeWay调用不同的消息发送方式实现消息多种发送方式
* @param messageDTO 消息载体
*/
public void sendMessage(MessageDTO messageDTO) {
messageServices.forEach(messageService -> {
if (messageService.check(messageDTO.getNoticeWay())) {
messageService.sendMessage(messageDTO);
}
});
}
}

View File

@@ -8,13 +8,13 @@ import cn.hutool.json.JSONUtil;
import com.alibaba.excel.context.AnalysisContext; import com.alibaba.excel.context.AnalysisContext;
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
import com.youlai.boot.common.base.BaseAnalysisEventListener; import com.youlai.boot.common.base.BaseAnalysisEventListener;
import com.youlai.boot.system.enums.GenderEnum;
import com.youlai.boot.system.model.entity.Dept; import com.youlai.boot.system.model.entity.Dept;
import com.youlai.boot.system.model.entity.Role; import com.youlai.boot.system.model.entity.Role;
import com.youlai.boot.system.model.entity.User; import com.youlai.boot.system.model.entity.User;
import com.youlai.boot.system.model.entity.UserRole; import com.youlai.boot.system.model.entity.UserRole;
import com.youlai.boot.common.base.IBaseEnum; import com.youlai.boot.common.base.IBaseEnum;
import com.youlai.boot.common.constant.SystemConstants; import com.youlai.boot.common.constant.SystemConstants;
import com.youlai.boot.common.enums.GenderEnum;
import com.youlai.boot.common.enums.StatusEnum; import com.youlai.boot.common.enums.StatusEnum;
import com.youlai.boot.system.converter.UserConverter; import com.youlai.boot.system.converter.UserConverter;
import com.youlai.boot.system.model.dto.UserImportDTO; import com.youlai.boot.system.model.dto.UserImportDTO;
@@ -107,13 +107,12 @@ public class UserImportListener extends BaseAnalysisEventListener<UserImportDTO>
// 校验通过,持久化至数据库 // 校验通过,持久化至数据库
User entity = userConverter.toEntity(userImportDTO); User entity = userConverter.toEntity(userImportDTO);
entity.setPassword(passwordEncoder.encode(SystemConstants.DEFAULT_PASSWORD)); // 默认密码 entity.setPassword(passwordEncoder.encode(SystemConstants.DEFAULT_PASSWORD)); // 默认密码
// 性别翻译 // 性别逆向解析
String genderLabel = userImportDTO.getGenderLabel(); String genderLabel = userImportDTO.getGenderLabel();
if (StrUtil.isNotBlank(genderLabel)) { if (StrUtil.isNotBlank(genderLabel)) {
Integer genderValue = (Integer) IBaseEnum.getValueByLabel(genderLabel, GenderEnum.class); Integer genderValue = (Integer) IBaseEnum.getValueByLabel(genderLabel, GenderEnum.class);
entity.setGender(genderValue); entity.setGender(genderValue);
} }
// 角色解析 // 角色解析
String roleCodes = userImportDTO.getRoleCodes(); String roleCodes = userImportDTO.getRoleCodes();
List<Long> roleIds = null; List<Long> roleIds = null;
@@ -155,7 +154,7 @@ public class UserImportListener extends BaseAnalysisEventListener<UserImportDTO>
} }
} else { } else {
invalidCount++; invalidCount++;
msg.append("").append(validCount + invalidCount).append("行数据校验失败:").append(validationMsg + "<br/>"); msg.append("").append(validCount + invalidCount).append("行数据校验失败:").append(validationMsg).append("<br/>");
} }
} }
@@ -166,7 +165,6 @@ public class UserImportListener extends BaseAnalysisEventListener<UserImportDTO>
@Override @Override
public void doAfterAllAnalysed(AnalysisContext analysisContext) { public void doAfterAllAnalysed(AnalysisContext analysisContext) {
log.info("所有数据解析完成!"); log.info("所有数据解析完成!");
} }

View File

@@ -4,9 +4,8 @@ import com.baomidou.mybatisplus.core.mapper.BaseMapper;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page; import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.youlai.boot.system.model.bo.NoticeBO; import com.youlai.boot.system.model.bo.NoticeBO;
import com.youlai.boot.system.model.entity.Notice; import com.youlai.boot.system.model.entity.Notice;
import com.youlai.boot.system.model.query.NoticeQuery; import com.youlai.boot.system.model.query.NoticePageQuery;
import com.youlai.boot.system.model.vo.NoticeVO; import com.youlai.boot.system.model.vo.NoticeVO;
import com.youlai.boot.system.model.vo.NoticeDetailVO;
import org.apache.ibatis.annotations.Mapper; import org.apache.ibatis.annotations.Mapper;
import org.apache.ibatis.annotations.Param; import org.apache.ibatis.annotations.Param;
@@ -26,12 +25,13 @@ public interface NoticeMapper extends BaseMapper<Notice> {
* @param queryParams 查询参数 * @param queryParams 查询参数
* @return 通知公告分页数据 * @return 通知公告分页数据
*/ */
Page<NoticeBO> getNoticePage(Page<NoticeVO> page, @Param("queryParams") NoticeQuery queryParams); Page<NoticeBO> getNoticePage(Page<NoticeVO> page, @Param("queryParams") NoticePageQuery queryParams);
/** /**
* 获取阅读时通知公告详情 * 获取阅读时通知公告详情
*
* @param id 通知公告ID * @param id 通知公告ID
* @return 通知公告详情 * @return 通知公告详情
*/ */
NoticeDetailVO getReadNoticeVO(@Param("id") Long id); NoticeBO getNoticeDetail(@Param("id") Long id);
} }

View File

@@ -3,9 +3,9 @@ package com.youlai.boot.system.mapper;
import com.baomidou.mybatisplus.core.mapper.BaseMapper; import com.baomidou.mybatisplus.core.mapper.BaseMapper;
import com.baomidou.mybatisplus.core.metadata.IPage; import com.baomidou.mybatisplus.core.metadata.IPage;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page; import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.youlai.boot.system.model.entity.NoticeStatus; import com.youlai.boot.system.model.entity.UserNotice;
import com.youlai.boot.system.model.query.NoticeQuery; import com.youlai.boot.system.model.query.NoticePageQuery;
import com.youlai.boot.system.model.vo.NoticeStatusVO; import com.youlai.boot.system.model.vo.UserNoticePageVO;
import com.youlai.boot.system.model.vo.NoticeVO; import com.youlai.boot.system.model.vo.NoticeVO;
import org.apache.ibatis.annotations.Mapper; import org.apache.ibatis.annotations.Mapper;
import org.apache.ibatis.annotations.Param; import org.apache.ibatis.annotations.Param;
@@ -19,14 +19,14 @@ import java.util.List;
* @since 2024-08-28 16:56 * @since 2024-08-28 16:56
*/ */
@Mapper @Mapper
public interface NoticeStatusMapper extends BaseMapper<NoticeStatus> { public interface UserNoticeMapper extends BaseMapper<UserNotice> {
/** /**
* 获取未读的通知公告 * 获取未读的通知公告
* @param userId 用户ID * @param userId 用户ID
* @return 公告列表 * @return 公告列表
*/ */
List<NoticeStatusVO> listUnreadNotices(@Param("userId")Long userId); List<UserNoticePageVO> listUnreadNotices(@Param("userId")Long userId);
/** /**
* 分页获取我的通知公告 * 分页获取我的通知公告
@@ -34,5 +34,5 @@ public interface NoticeStatusMapper extends BaseMapper<NoticeStatus> {
* @param queryParams 查询参数 * @param queryParams 查询参数
* @return 通知公告分页列表 * @return 通知公告分页列表
*/ */
IPage<NoticeStatusVO> getMyNoticePage(Page<NoticeVO> page, @Param("queryParams") NoticeQuery queryParams); IPage<UserNoticePageVO> getMyNoticePage(Page<NoticeVO> page, @Param("queryParams") NoticePageQuery queryParams);
} }

View File

@@ -1,9 +1,7 @@
package com.youlai.boot.system.model.bo; package com.youlai.boot.system.model.bo;
import io.swagger.v3.oas.annotations.media.Schema;
import lombok.Data; import lombok.Data;
import java.io.Serial;
import java.time.LocalDateTime; import java.time.LocalDateTime;
/** /**
@@ -11,36 +9,57 @@ import java.time.LocalDateTime;
* *
* @author Theo * @author Theo
* @since 2024-09-01 10:31 * @since 2024-09-01 10:31
* @version 1.0.0
*/ */
@Data @Data
public class NoticeBO { public class NoticeBO {
@Serial /**
private static final long serialVersionUID = 1L; * 通知ID
*/
private Long id; private Long id;
@Schema(description = "通知标题")
/**
* 通知标题
*/
private String title; private String title;
@Schema(description = "通知类型") /**
private Integer noticeType; * 通知类型
*/
private Integer type;
@Schema(description = "发布人") /**
private String releaseBy; * 通知内容
*/
private String content;
@Schema(description = "优先级(0-低 1-中 2-高)") /**
private Integer priority; * 发布人姓名
*/
private String publisherName;
@Schema(description = "目标类型(0-全体 1-指定)") /**
private Integer tarType; * 通知等级L: 低, M: 中, H: 高)
*/
private Integer level;
@Schema(description = "发布状态(0-未发布 1已发布 2已撤回)") /**
private Integer releaseStatus; * 目标类型(1: 全体 2: 指定)
*/
private Integer targetType;
@Schema(description = "发布时间") /**
private LocalDateTime releaseTime; * 发布状态0: 未发布, 1: 已发布, -1: 已撤回)
*/
private Integer publishStatus;
@Schema(description = "撤回时间") /**
private LocalDateTime recallTime; * 发布时间
*/
private LocalDateTime publishTime;
/**
* 撤回时间
*/
private LocalDateTime revokeTime;
} }

View File

@@ -1,6 +1,6 @@
package com.youlai.boot.system.model.bo; package com.youlai.boot.system.model.bo;
import com.youlai.boot.common.enums.MenuTypeEnum; import com.youlai.boot.system.enums.MenuTypeEnum;
import lombok.Data; import lombok.Data;
/** /**

View File

@@ -1,6 +1,5 @@
package com.youlai.boot.system.model.dto; package com.youlai.boot.system.model.dto;
import com.youlai.boot.common.enums.NoticeTypeEnum;
import lombok.AllArgsConstructor; import lombok.AllArgsConstructor;
import lombok.Data; import lombok.Data;
import lombok.NoArgsConstructor; import lombok.NoArgsConstructor;
@@ -23,9 +22,4 @@ public class ChatMessage {
*/ */
private String content; private String content;
/**
* 消息类型
*/
private NoticeTypeEnum noticeType;
} }

View File

@@ -1,10 +1,9 @@
package com.youlai.boot.system.model.dto; package com.youlai.boot.system.model.dto;
import com.youlai.boot.common.enums.NoticeWayEnum;
import io.swagger.v3.oas.annotations.media.Schema; import io.swagger.v3.oas.annotations.media.Schema;
import lombok.Data; import lombok.Data;
import java.util.List; import java.util.Set;
/** /**
* 消息载体 * 消息载体
@@ -23,8 +22,8 @@ public class MessageDTO {
private String sender; private String sender;
@Schema(description = "接收者") @Schema(description = "接收者")
private List<String> receiver; private Set<String> receivers;
@Schema(description = "通知方式")
private NoticeWayEnum noticeWay;
} }

View File

@@ -0,0 +1,27 @@
package com.youlai.boot.system.model.dto;
import io.swagger.v3.oas.annotations.media.Schema;
import lombok.Data;
import java.util.Set;
/**
* 通知传送对象
*
* @author Theo
* @since 2024-9-2 14:32:58
*/
@Data
public class NoticeDTO {
@Schema(description = "通知ID")
private Long id;
@Schema(description = "通知类型")
private Integer type;
@Schema(description = "通知标题")
private String title;
}

View File

@@ -2,7 +2,7 @@ package com.youlai.boot.system.model.entity;
import com.baomidou.mybatisplus.annotation.*; import com.baomidou.mybatisplus.annotation.*;
import com.youlai.boot.common.enums.MenuTypeEnum; import com.youlai.boot.system.enums.MenuTypeEnum;
import lombok.Getter; import lombok.Getter;
import lombok.Setter; import lombok.Setter;

View File

@@ -2,11 +2,11 @@ package com.youlai.boot.system.model.entity;
import com.baomidou.mybatisplus.annotation.TableLogic; import com.baomidou.mybatisplus.annotation.TableLogic;
import com.baomidou.mybatisplus.annotation.TableName; import com.baomidou.mybatisplus.annotation.TableName;
import com.fasterxml.jackson.annotation.JsonFormat;
import com.youlai.boot.common.base.BaseEntity; import com.youlai.boot.common.base.BaseEntity;
import lombok.Getter; import lombok.Getter;
import lombok.Setter; import lombok.Setter;
import java.io.Serial;
import java.time.LocalDateTime; import java.time.LocalDateTime;
/** /**
* 通知公告实体对象 * 通知公告实体对象
@@ -19,6 +19,7 @@ import java.time.LocalDateTime;
@TableName("sys_notice") @TableName("sys_notice")
public class Notice extends BaseEntity { public class Notice extends BaseEntity {
@Serial
private static final long serialVersionUID = 1L; private static final long serialVersionUID = 1L;
/** /**
@@ -32,41 +33,48 @@ public class Notice extends BaseEntity {
/** /**
* 通知类型 * 通知类型
*/ */
private Integer noticeType; private Integer type;
/** /**
* 发布人 * 发布人
*/ */
private Long releaseBy; private Long publisherBy;
/** /**
* 优先级(0-低 1-中 2-高) * 通知等级L: 低, M: 中, H: 高)
*/ */
private Integer priority; private String level;
/** /**
* 目标类型(0-全体 1-指定) * 目标类型1: 全体, 2: 指定
*/ */
private Integer tarType; private Integer targetType;
/** /**
* 目标ID * 目标用户ID集合
*/ */
private String tarIds; private String targetUserIds;
/** /**
* 发布状态(0-未发布 1已发布 2已撤回) * 发布状态0: 未发布, 1: 已发布, -1: 已撤回
*/ */
private Integer releaseStatus; private Integer publishStatus;
/** /**
* 发布时间 * 发布时间
*/ */
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss") private LocalDateTime publishTime;
private LocalDateTime releaseTime;
/** /**
* 撤回时间 * 撤回时间
*/ */
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss") private LocalDateTime revokeTime;
private LocalDateTime recallTime;
/** /**
* 创建人ID * 创建人ID
*/ */
private Long createBy; private Long createBy;
/** /**
* 更新人ID * 更新人ID
*/ */

View File

@@ -2,25 +2,24 @@ package com.youlai.boot.system.model.entity;
import com.baomidou.mybatisplus.annotation.IdType; import com.baomidou.mybatisplus.annotation.IdType;
import com.baomidou.mybatisplus.annotation.TableId; import com.baomidou.mybatisplus.annotation.TableId;
import com.baomidou.mybatisplus.annotation.TableLogic;
import com.baomidou.mybatisplus.annotation.TableName; import com.baomidou.mybatisplus.annotation.TableName;
import com.youlai.boot.common.base.BaseEntity;
import lombok.Getter; import lombok.Getter;
import lombok.Setter; import lombok.Setter;
import java.io.Serializable;
import java.time.LocalDateTime; import java.time.LocalDateTime;
/** /**
* 用户公告状态实体对象 * 用户通知公告实体对象
* *
* @author youlaitech * @author youlaitech
* @since 2024-08-28 16:56 * @since 2024-08-28 16:56
*/ */
@Getter @Getter
@Setter @Setter
@TableName("sys_notice_status") @TableName("sys_user_notice")
public class NoticeStatus implements Serializable { public class UserNotice extends BaseEntity {
private static final long serialVersionUID = 1L;
/** /**
* 主键ID * 主键ID
@@ -39,9 +38,15 @@ public class NoticeStatus implements Serializable {
/** /**
* 读取状态0未读1已读 * 读取状态0未读1已读
*/ */
private Integer readStatus; private Integer isRead;
/** /**
* 用户阅读时间 * 用户阅读时间
*/ */
private LocalDateTime readTime; private LocalDateTime readTime;
/**
* 逻辑删除标识(0-未删除 1-已删除)
*/
@TableLogic(value = "0", delval = "1")
private Integer isDeleted;
} }

View File

@@ -1,6 +1,6 @@
package com.youlai.boot.system.model.form; package com.youlai.boot.system.model.form;
import com.youlai.boot.common.enums.MenuTypeEnum; import com.youlai.boot.system.enums.MenuTypeEnum;
import com.youlai.boot.common.model.KeyValue; import com.youlai.boot.common.model.KeyValue;
import io.swagger.v3.oas.annotations.media.Schema; import io.swagger.v3.oas.annotations.media.Schema;
import lombok.Data; import lombok.Data;

View File

@@ -39,17 +39,16 @@ public class NoticeForm implements Serializable {
private String content; private String content;
@Schema(description = "通知类型") @Schema(description = "通知类型")
private Integer noticeType; private Integer type;
@Schema(description = "优先级(0-低 1-中 2-高)") @Schema(description = "优先级(L-低 M-中 H-高)")
@Range(min = 0, max = 2, message = "优先级取值范围[0,2]") private String level;
private Integer priority;
@Schema(description = "目标类型(0-全体 1-指定)") @Schema(description = "目标类型(1-全体 2-指定)")
@Range(min = 0, max = 1, message = "目标类型取值范围[0,1]") @Range(min = 1, max = 2, message = "目标类型取值范围[1,2]")
private Integer tarType; private Integer targetType;
@Schema(description = "接收人ID集合") @Schema(description = "接收人ID集合")
private List<String> tarIds; private List<String> targetUserIds;
} }

View File

@@ -16,9 +16,7 @@ import java.util.List;
@Data @Data
@EqualsAndHashCode(callSuper = true) @EqualsAndHashCode(callSuper = true)
@Schema(description ="通知公告查询对象") @Schema(description ="通知公告查询对象")
public class NoticeQuery extends BasePageQuery { public class NoticePageQuery extends BasePageQuery {
private static final long serialVersionUID = 1L;
@Schema(description = "通知标题") @Schema(description = "通知标题")
private String title; private String title;

View File

@@ -1,7 +1,7 @@
package com.youlai.boot.system.model.vo; package com.youlai.boot.system.model.vo;
import com.fasterxml.jackson.annotation.JsonInclude; import com.fasterxml.jackson.annotation.JsonInclude;
import com.youlai.boot.common.enums.MenuTypeEnum; import com.youlai.boot.system.enums.MenuTypeEnum;
import io.swagger.v3.oas.annotations.media.Schema; import io.swagger.v3.oas.annotations.media.Schema;
import lombok.Data; import lombok.Data;

View File

@@ -25,18 +25,18 @@ public class NoticeDetailVO {
private String content; private String content;
@Schema(description = "通知类型") @Schema(description = "通知类型")
private String noticeType; private Integer type;
@Schema(description = "发布人") @Schema(description = "发布人")
private String releaseBy; private String publisherName;
@Schema(description = "优先级(0-低 1-中 2-高)") @Schema(description = "优先级(L-低 M-中 H-高)")
private Integer priority; private String level;
@Schema(description = "发布状态(0-未发布 1已发布 2已撤回) 冗余字段,方便判断是否已经发布") @Schema(description = "发布状态(0-未发布 1已发布 2已撤回) 冗余字段,方便判断是否已经发布")
private Integer releaseStatus; private Integer publishStatus;
@Schema(description = "发布时间") @Schema(description = "发布时间")
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss") @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss")
private LocalDateTime releaseTime; private LocalDateTime publishTime;
} }

View File

@@ -7,14 +7,14 @@ import lombok.Data;
import java.time.LocalDateTime; import java.time.LocalDateTime;
/** /**
* 用户公告状态VO * 用户公告VO
* *
* @author Theo * @author Theo
* @since 2024-08-28 16:56 * @since 2024-08-28 16:56
*/ */
@Data @Data
@Schema(description = "用户公告状态VO") @Schema(description = "用户公告VO")
public class NoticeStatusVO { public class UserNoticePageVO {
@Schema(description = "通知ID") @Schema(description = "通知ID")
private Long id; private Long id;
@@ -23,19 +23,19 @@ public class NoticeStatusVO {
private String title; private String title;
@Schema(description = "通知类型") @Schema(description = "通知类型")
private String noticeType; private String typeLabel;
@Schema(description = "发布人") @Schema(description = "发布人姓名")
private String releaseBy; private String publisherName;
@Schema(description = "优先级(0-低 1-中 2-高)") @Schema(description = "通知级别")
private Integer priority; private String levelLabel;
@Schema(description = "发布时间") @Schema(description = "发布时间")
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss") @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss")
private LocalDateTime releaseTime; private LocalDateTime publishTime;
@Schema(description = "是否已读") @Schema(description = "是否已读")
private Integer readStatus; private String isReadLabel;
} }

View File

@@ -4,8 +4,8 @@ import com.baomidou.mybatisplus.core.metadata.IPage;
import com.baomidou.mybatisplus.extension.service.IService; import com.baomidou.mybatisplus.extension.service.IService;
import com.youlai.boot.system.model.entity.Notice; import com.youlai.boot.system.model.entity.Notice;
import com.youlai.boot.system.model.form.NoticeForm; import com.youlai.boot.system.model.form.NoticeForm;
import com.youlai.boot.system.model.query.NoticeQuery; import com.youlai.boot.system.model.query.NoticePageQuery;
import com.youlai.boot.system.model.vo.NoticeStatusVO; import com.youlai.boot.system.model.vo.UserNoticePageVO;
import com.youlai.boot.system.model.vo.NoticeVO; import com.youlai.boot.system.model.vo.NoticeVO;
import com.youlai.boot.system.model.vo.NoticeDetailVO; import com.youlai.boot.system.model.vo.NoticeDetailVO;
@@ -22,7 +22,7 @@ public interface NoticeService extends IService<Notice> {
* *
* @return 通知公告分页列表 * @return 通知公告分页列表
*/ */
IPage<NoticeVO> getNoticePage(NoticeQuery queryParams); IPage<NoticeVO> getNoticePage(NoticePageQuery queryParams);
/** /**
* 获取通知公告表单数据 * 获取通知公告表单数据
@@ -63,7 +63,7 @@ public interface NoticeService extends IService<Notice> {
* @param id 通知公告ID * @param id 通知公告ID
* @return 是否发布成功 * @return 是否发布成功
*/ */
boolean releaseNotice(Long id); boolean publishNotice(Long id);
/** /**
* 撤回通知公告 * 撤回通知公告
@@ -71,27 +71,21 @@ public interface NoticeService extends IService<Notice> {
* @param id 通知公告ID * @param id 通知公告ID
* @return 是否撤回成功 * @return 是否撤回成功
*/ */
boolean recallNotice(Long id); boolean revokeNotice(Long id);
/** /**
* 阅读通知公告 * 阅读获取通知公告详情
* *
* @param id 通知公告ID * @param id 通知公告ID
* @return 通知公告对象
*/
NoticeDetailVO readNotice(Long id);
/**
* 获取阅读时通知公告详情
* @param id 通知公告ID
* @return 通知公告详情 * @return 通知公告详情
*/ */
NoticeDetailVO getReadNoticeDetail(Long id); NoticeDetailVO getNoticeDetail(Long id);
/** /**
* 获取我的通知公告分页列表 * 获取我的通知公告分页列表
*
* @param queryParams 查询参数 * @param queryParams 查询参数
* @return 通知公告分页列表 * @return 通知公告分页列表
*/ */
IPage<NoticeStatusVO> getMyNoticePage(NoticeQuery queryParams); IPage<UserNoticePageVO> getMyNoticePage(NoticePageQuery queryParams);
} }

View File

@@ -3,9 +3,9 @@ package com.youlai.boot.system.service;
import com.baomidou.mybatisplus.core.metadata.IPage; import com.baomidou.mybatisplus.core.metadata.IPage;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page; import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.baomidou.mybatisplus.extension.service.IService; import com.baomidou.mybatisplus.extension.service.IService;
import com.youlai.boot.system.model.entity.NoticeStatus; import com.youlai.boot.system.model.entity.UserNotice;
import com.youlai.boot.system.model.query.NoticeQuery; import com.youlai.boot.system.model.query.NoticePageQuery;
import com.youlai.boot.system.model.vo.NoticeStatusVO; import com.youlai.boot.system.model.vo.UserNoticePageVO;
import com.youlai.boot.system.model.vo.NoticeVO; import com.youlai.boot.system.model.vo.NoticeVO;
import java.util.List; import java.util.List;
@@ -16,16 +16,17 @@ import java.util.List;
* @author youlaitech * @author youlaitech
* @since 2024-08-28 16:56 * @since 2024-08-28 16:56
*/ */
public interface NoticeStatusService extends IService<NoticeStatus> { public interface UserNoticeService extends IService<UserNotice> {
/** /**
* 获取未读的通知公告 * 获取未读的通知公告
* @return 公告列表 * @return 公告列表
*/ */
List<NoticeStatusVO> listUnreadNotices(); List<UserNoticePageVO> listUnreadNotices();
/** /**
* 全部标记为已读 * 全部标记为已读
*
* @return 是否成功 * @return 是否成功
*/ */
boolean readAll(); boolean readAll();
@@ -36,5 +37,5 @@ public interface NoticeStatusService extends IService<NoticeStatus> {
* @param queryParams 查询参数 * @param queryParams 查询参数
* @return 我的通知公告分页列表 * @return 我的通知公告分页列表
*/ */
IPage<NoticeStatusVO> getMyNoticePage(Page<NoticeVO> page, NoticeQuery queryParams); IPage<UserNoticePageVO> getMyNoticePage(Page<NoticeVO> page, NoticePageQuery queryParams);
} }

View File

@@ -3,7 +3,7 @@ package com.youlai.boot.system.service;
import com.baomidou.mybatisplus.core.metadata.IPage; import com.baomidou.mybatisplus.core.metadata.IPage;
import com.baomidou.mybatisplus.extension.service.IService; import com.baomidou.mybatisplus.extension.service.IService;
import com.youlai.boot.common.enums.ContactType; import com.youlai.boot.system.enums.ContactType;
import com.youlai.boot.common.model.Option; import com.youlai.boot.common.model.Option;
import com.youlai.boot.system.model.dto.UserAuthInfo; import com.youlai.boot.system.model.dto.UserAuthInfo;
import com.youlai.boot.system.model.dto.UserExportDTO; import com.youlai.boot.system.model.dto.UserExportDTO;

View File

@@ -20,7 +20,7 @@ import com.youlai.boot.system.model.query.MenuQuery;
import com.youlai.boot.system.model.vo.MenuVO; import com.youlai.boot.system.model.vo.MenuVO;
import com.youlai.boot.system.model.vo.RouteVO; import com.youlai.boot.system.model.vo.RouteVO;
import com.youlai.boot.common.constant.SystemConstants; import com.youlai.boot.common.constant.SystemConstants;
import com.youlai.boot.common.enums.MenuTypeEnum; import com.youlai.boot.system.enums.MenuTypeEnum;
import com.youlai.boot.common.enums.StatusEnum; import com.youlai.boot.common.enums.StatusEnum;
import com.youlai.boot.common.model.KeyValue; import com.youlai.boot.common.model.KeyValue;
import com.youlai.boot.common.model.Option; import com.youlai.boot.common.model.Option;

View File

@@ -1,45 +1,48 @@
package com.youlai.boot.system.service.impl; package com.youlai.boot.system.service.impl;
import cn.hutool.core.lang.Assert; import cn.hutool.core.collection.CollectionUtil;
import cn.hutool.core.util.StrUtil; import cn.hutool.core.util.StrUtil;
import cn.hutool.json.JSONObject;
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
import com.baomidou.mybatisplus.core.conditions.update.LambdaUpdateWrapper;
import com.baomidou.mybatisplus.core.metadata.IPage; import com.baomidou.mybatisplus.core.metadata.IPage;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page; import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
import com.youlai.boot.common.constant.SymbolConstant; import com.youlai.boot.common.exception.BusinessException;
import com.youlai.boot.common.enums.NoticeWayEnum;
import com.youlai.boot.core.security.util.SecurityUtils; import com.youlai.boot.core.security.util.SecurityUtils;
import com.youlai.boot.module.websocket.service.OnlineUserService;
import com.youlai.boot.system.converter.NoticeConverter; import com.youlai.boot.system.converter.NoticeConverter;
import com.youlai.boot.system.handler.MessageHandler; import com.youlai.boot.system.enums.NoticePublishStatusEnum;
import com.youlai.boot.system.enums.NoticeTargetTypeEnum;
import com.youlai.boot.system.mapper.NoticeMapper; import com.youlai.boot.system.mapper.NoticeMapper;
import com.youlai.boot.system.model.bo.NoticeBO; import com.youlai.boot.system.model.bo.NoticeBO;
import com.youlai.boot.system.model.dto.MessageDTO; import com.youlai.boot.system.model.dto.NoticeDTO;
import com.youlai.boot.system.model.entity.Notice; import com.youlai.boot.system.model.entity.Notice;
import com.youlai.boot.system.model.entity.NoticeStatus; import com.youlai.boot.system.model.entity.UserNotice;
import com.youlai.boot.system.model.entity.User; import com.youlai.boot.system.model.entity.User;
import com.youlai.boot.system.model.form.NoticeForm; import com.youlai.boot.system.model.form.NoticeForm;
import com.youlai.boot.system.model.query.NoticeQuery; import com.youlai.boot.system.model.query.NoticePageQuery;
import com.youlai.boot.system.model.vo.NoticeStatusVO; import com.youlai.boot.system.model.vo.UserNoticePageVO;
import com.youlai.boot.system.model.vo.NoticeVO; import com.youlai.boot.system.model.vo.NoticeVO;
import com.youlai.boot.system.model.vo.NoticeDetailVO; import com.youlai.boot.system.model.vo.NoticeDetailVO;
import com.youlai.boot.system.service.NoticeService; import com.youlai.boot.system.service.NoticeService;
import com.youlai.boot.system.service.NoticeStatusService; import com.youlai.boot.system.service.UserNoticeService;
import com.youlai.boot.system.service.UserService; import com.youlai.boot.system.service.UserService;
import lombok.RequiredArgsConstructor; import lombok.RequiredArgsConstructor;
import org.springframework.messaging.simp.SimpMessagingTemplate;
import org.springframework.stereotype.Service; import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional; import org.springframework.transaction.annotation.Transactional;
import java.time.LocalDateTime; import java.time.LocalDateTime;
import java.time.format.DateTimeFormatter;
import java.util.Arrays; import java.util.Arrays;
import java.util.HashSet;
import java.util.List; import java.util.List;
import java.util.Set;
import java.util.stream.Collectors; import java.util.stream.Collectors;
/** /**
* 通知公告服务实现类 * 通知公告服务实现类
* *
* @author youlaitech * @author Theo
* @since 2024-08-27 10:31 * @since 2024-08-27 10:31
*/ */
@Service @Service
@@ -48,12 +51,13 @@ public class NoticeServiceImpl extends ServiceImpl<NoticeMapper, Notice> impleme
private final NoticeConverter noticeConverter; private final NoticeConverter noticeConverter;
private final MessageHandler messageHandler; private final UserNoticeService userNoticeService;
private final NoticeStatusService noticeStatusService;
private final UserService userService; private final UserService userService;
private final SimpMessagingTemplate messagingTemplate;
private final OnlineUserService onlineUserService;
/** /**
* 获取通知公告分页列表 * 获取通知公告分页列表
* *
@@ -61,7 +65,7 @@ public class NoticeServiceImpl extends ServiceImpl<NoticeMapper, Notice> impleme
* @return {@link IPage<NoticeVO>} 通知公告分页列表 * @return {@link IPage<NoticeVO>} 通知公告分页列表
*/ */
@Override @Override
public IPage<NoticeVO> getNoticePage(NoticeQuery queryParams) { public IPage<NoticeVO> getNoticePage(NoticePageQuery queryParams) {
Page<NoticeBO> noticePage = this.baseMapper.getNoticePage( Page<NoticeBO> noticePage = this.baseMapper.getNoticePage(
new Page<>(queryParams.getPageNum(), queryParams.getPageSize()), new Page<>(queryParams.getPageNum(), queryParams.getPageSize()),
queryParams queryParams
@@ -89,12 +93,15 @@ public class NoticeServiceImpl extends ServiceImpl<NoticeMapper, Notice> impleme
*/ */
@Override @Override
public boolean saveNotice(NoticeForm formData) { public boolean saveNotice(NoticeForm formData) {
Notice entity = noticeConverter.toEntity(formData);
entity.setReleaseStatus(0); if (NoticeTargetTypeEnum.SPECIFIED.getValue().equals(formData.getTargetType())) {
entity.setCreateBy(SecurityUtils.getUserId()); List<String> targetUserIdList = formData.getTargetUserIds();
if (entity.getTarType() == 1) { if (CollectionUtil.isEmpty(targetUserIdList)) {
Assert.notBlank(entity.getTarIds(), "指定用户不能为空"); throw new BusinessException("推送指定用户不能为空");
} }
}
Notice entity = noticeConverter.toEntity(formData);
entity.setCreateBy(SecurityUtils.getUserId());
return this.save(entity); return this.save(entity);
} }
@@ -107,11 +114,14 @@ public class NoticeServiceImpl extends ServiceImpl<NoticeMapper, Notice> impleme
*/ */
@Override @Override
public boolean updateNotice(Long id, NoticeForm formData) { public boolean updateNotice(Long id, NoticeForm formData) {
Notice entity = noticeConverter.toEntity(formData); if (NoticeTargetTypeEnum.SPECIFIED.getValue().equals(formData.getTargetType())) {
entity.setUpdateBy(SecurityUtils.getUserId()); List<String> targetUserIdList = formData.getTargetUserIds();
if (entity.getTarType() == 1) { if (CollectionUtil.isEmpty(targetUserIdList)) {
Assert.notBlank(entity.getTarIds(), "指定用户不能为空"); throw new BusinessException("推送指定用户不能为空");
} }
}
Notice entity = noticeConverter.toEntity(formData);
return this.updateById(entity); return this.updateById(entity);
} }
@@ -122,87 +132,97 @@ public class NoticeServiceImpl extends ServiceImpl<NoticeMapper, Notice> impleme
* @return {@link Boolean} 是否删除成功 * @return {@link Boolean} 是否删除成功
*/ */
@Override @Override
@Transactional(rollbackFor = Exception.class) @Transactional
public boolean deleteNotices(String ids) { public boolean deleteNotices(String ids) {
Assert.isTrue(StrUtil.isNotBlank(ids), "删除的通知公告数据为空"); if (StrUtil.isBlank(ids)) {
// 逻辑删除 throw new BusinessException("删除的通知公告数据为空");
List<Long> idList = Arrays.stream(ids.split(SymbolConstant.COMMA))
.map(Long::parseLong)
.toList();
boolean b = this.removeByIds(idList);
if (b) {
//删除通知公告的同时,需要删除通知公告对应的用户通知状态
noticeStatusService.remove(new LambdaQueryWrapper<NoticeStatus>().in(NoticeStatus::getNoticeId, idList));
} }
return true; // 逻辑删除
List<Long> idList = Arrays.stream(ids.split(","))
.map(Long::parseLong)
.toList();
boolean isRemoved = this.removeByIds(idList);
if (isRemoved) {
// 删除通知公告的同时,需要删除通知公告对应的用户通知状态
userNoticeService.remove(new LambdaQueryWrapper<UserNotice>().in(UserNotice::getNoticeId, idList));
}
return isRemoved;
} }
/** /**
* 发布通知公告 * 发布通知公告
*
* @param id 通知公告ID * @param id 通知公告ID
* @return 是否发布成功 * @return 是否发布成功
*/ */
@Override @Override
@Transactional(rollbackFor = Exception.class) @Transactional
public boolean releaseNotice(Long id) { public boolean publishNotice(Long id) {
Notice notice = this.getById(id); Notice notice = this.getById(id);
Assert.notNull(notice, "通知公告不存在"); if (notice == null) {
Assert.isTrue(notice.getReleaseStatus() != 1, "通知公告已发布"); throw new BusinessException("通知公告不存在");
notice.setReleaseStatus(1);
notice.setReleaseBy(SecurityUtils.getUserId());
notice.setReleaseTime(LocalDateTime.now());
this.updateById(notice);
//发布通知公告的同时,需要将通知公告发送给目标用户
//先删除掉该通知公告之前对应的用户信息
noticeStatusService.remove(new LambdaQueryWrapper<NoticeStatus>().eq(NoticeStatus::getNoticeId, id));
LambdaQueryWrapper<User> queryWrapper = new LambdaQueryWrapper<>();
if (notice.getTarType() == 1) {
Assert.notBlank(notice.getTarIds(), "指定用户不能为空");
queryWrapper.in(User::getId, Arrays.asList(notice.getTarIds().split(SymbolConstant.COMMA)));
}
//查询出目标用户,增加用户通知状态
List<User> list = userService.list(queryWrapper);
List<NoticeStatus> needSaveList = list.stream().map(user -> {
NoticeStatus noticeStatus = new NoticeStatus();
noticeStatus.setNoticeId(id);
noticeStatus.setUserId(user.getId());
noticeStatus.setReadStatus(0);
return noticeStatus;
}).toList();
if(needSaveList.size() > 0){
noticeStatusService.saveBatch(needSaveList);
}
//最后给当前在线的用户发送websocket消息
List<String> usernameList = null;
if(notice.getTarType() == 1){
List<Long> collect = needSaveList.stream().map(NoticeStatus::getUserId).collect(Collectors.toList());
List<User> userList = userService.list(new LambdaQueryWrapper<User>().in(User::getId, collect).select(User::getUsername));
usernameList = userList.stream().map(User::getUsername).collect(Collectors.toList());
}
MessageDTO message = new MessageDTO();
message.setNoticeWay(NoticeWayEnum.WEBSOCKET);
message.setReceiver(usernameList);
message.setContent(getNoticeContent(notice));
message.setSender(SecurityUtils.getUsername());
messageHandler.sendMessage(message);
return this.updateById(notice);
} }
/** if (NoticePublishStatusEnum.PUBLISHED.getValue().equals(notice.getPublishStatus())) {
* 自定义组合公告内容 throw new BusinessException("通知公告已发布");
* }
* @param notice 通知公告
* @return 自定义组合通知公告内容 Integer targetType = notice.getTargetType();
*/ String targetUserIds = notice.getTargetUserIds();
private String getNoticeContent(Notice notice) { if (NoticeTargetTypeEnum.SPECIFIED.getValue().equals(targetType)
JSONObject jsonObject = new JSONObject(); && StrUtil.isBlank(targetUserIds)) {
jsonObject.set("id", notice.getId()); throw new BusinessException("推送指定用户不能为空");
jsonObject.set("title", notice.getTitle()); }
jsonObject.set("messageType", notice.getNoticeType());
jsonObject.set("releaseTime", notice.getReleaseTime().format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss"))); notice.setPublishStatus(NoticePublishStatusEnum.PUBLISHED.getValue());
jsonObject.set("type", "release"); notice.setPublisherBy(SecurityUtils.getUserId());
return jsonObject.toString(); notice.setPublishTime(LocalDateTime.now());
boolean publishResult = this.updateById(notice);
if (publishResult) {
// 发布通知公告的同时,删除该通告之前的用户通知数据,因为可能是重新发布
userNoticeService.remove(
new LambdaQueryWrapper<UserNotice>().eq(UserNotice::getNoticeId, id)
);
// 添加新的用户通知数据
List<String> targetUserIdList = Arrays.asList(targetUserIds.split(","));
List<User> targetUserList = userService.list(
new LambdaQueryWrapper<User>()
// 如果是指定用户,则筛选出指定用户
.in(NoticeTargetTypeEnum.SPECIFIED.getValue().equals(targetType), User::getId, targetUserIdList)
);
List<UserNotice> userNoticeList = targetUserList.stream().map(user -> {
UserNotice userNotice = new UserNotice();
userNotice.setNoticeId(id);
userNotice.setUserId(user.getId());
userNotice.setIsRead(0);
return userNotice;
}).toList();
if (CollectionUtil.isNotEmpty(userNoticeList)) {
userNoticeService.saveBatch(userNoticeList);
}
Set<String> receivers = targetUserList.stream().map(User::getUsername).collect(Collectors.toSet());
Set<String> allOnlineUsers = onlineUserService.getAllOnlineUsers();
// 找出在线用户的通知接收者
Set<String> onlineReceivers = new HashSet<>(CollectionUtil.intersection(receivers, allOnlineUsers));
NoticeDTO noticeDTO = new NoticeDTO();
noticeDTO.setId(id);
noticeDTO.setTitle(notice.getTitle());
noticeDTO.setType(notice.getType());
onlineReceivers.forEach(receiver -> messagingTemplate.convertAndSendToUser(receiver, "/queue/message", noticeDTO));
}
return publishResult;
} }
/** /**
@@ -212,67 +232,65 @@ public class NoticeServiceImpl extends ServiceImpl<NoticeMapper, Notice> impleme
* @return 是否撤回成功 * @return 是否撤回成功
*/ */
@Override @Override
@Transactional(rollbackFor = Exception.class) @Transactional
public boolean recallNotice(Long id) { public boolean revokeNotice(Long id) {
Notice notice = this.getById(id); Notice notice = this.getById(id);
Assert.notNull(notice, "通知公告不存在"); if (notice == null) {
Assert.isTrue(notice.getReleaseStatus() == 1, "通知公告未发布"); throw new BusinessException("通知公告不存在");
notice.setReleaseStatus(2);
notice.setRecallTime(LocalDateTime.now());
if (!this.updateById(notice)) {
return false;
} }
//先删除掉该通知公告之前对应的用户信息
noticeStatusService.remove(new LambdaQueryWrapper<NoticeStatus>().eq(NoticeStatus::getNoticeId, id)); if (!NoticePublishStatusEnum.PUBLISHED.getValue().equals(notice.getPublishStatus())) {
return true; throw new BusinessException("通知公告未发布或已撤回");
}
notice.setPublishStatus(NoticePublishStatusEnum.REVOKED.getValue());
notice.setRevokeTime(LocalDateTime.now());
notice.setUpdateBy(SecurityUtils.getUserId());
boolean revokeResult = this.updateById(notice);
if (revokeResult) {
// 撤回通知公告的同时,需要删除通知公告对应的用户通知状态
userNoticeService.remove(new LambdaQueryWrapper<UserNotice>()
.eq(UserNotice::getNoticeId, id)
);
}
return revokeResult;
} }
/** /**
* 阅读通知公告 * 阅读获取通知公告详情
* @param id 通知公告ID *
* @return 通知公告表单对象
*/
@Override
public NoticeDetailVO readNotice(Long id) {
NoticeDetailVO noticeDetailVO = this.getReadNoticeDetail(id);
Assert.isTrue(noticeDetailVO != null && noticeDetailVO.getReleaseStatus() == 1, "公告不存在或未发布");
//获取当前登录用户
Long userId = SecurityUtils.getUserId();
LambdaQueryWrapper<NoticeStatus> queryWrapper = new LambdaQueryWrapper<NoticeStatus>()
.eq(NoticeStatus::getUserId, userId)
.eq(NoticeStatus::getNoticeId, id)
.eq(NoticeStatus::getReadStatus, 0);
NoticeStatus noticeStatus = noticeStatusService.getOne(queryWrapper);
if (noticeStatus != null) {
noticeStatus.setReadStatus(1);
noticeStatusService.updateById(noticeStatus);
}
return noticeDetailVO;
}
/**
* 获取阅读时通知公告详情
* @param id 通知公告ID * @param id 通知公告ID
* @return * @return
*/ */
@Override @Override
public NoticeDetailVO getReadNoticeDetail(Long id) { public NoticeDetailVO getNoticeDetail(Long id) {
Assert.notNull(id, "公告ID不能为空"); NoticeBO noticeBO = this.baseMapper.getNoticeDetail(id);
NoticeDetailVO noticeDetailVO = this.baseMapper.getReadNoticeVO(id); // 更新用户通知公告的阅读状态
Assert.isTrue(noticeDetailVO != null, "公告不存在"); Long userId = SecurityUtils.getUserId();
return noticeDetailVO; userNoticeService.update(new LambdaUpdateWrapper<UserNotice>()
.eq(UserNotice::getNoticeId, id)
.eq(UserNotice::getUserId, userId)
.eq(UserNotice::getIsRead, 0)
.set(UserNotice::getIsRead, 1)
);
return noticeConverter.toDetailVO(noticeBO);
} }
/** /**
* 获取当前登录用户的通知公告列表 * 获取当前登录用户的通知公告列表
*
* @param queryParams 查询参数 * @param queryParams 查询参数
* @return 通知公告分页列表 * @return 通知公告分页列表
*/ */
@Override @Override
public IPage<NoticeStatusVO> getMyNoticePage(NoticeQuery queryParams) { public IPage<UserNoticePageVO> getMyNoticePage(NoticePageQuery queryParams) {
Long userId = SecurityUtils.getUserId(); queryParams.setUserId(SecurityUtils.getUserId());
queryParams.setUserId(userId); return userNoticeService.getMyNoticePage(
return noticeStatusService.getMyNoticePage(new Page<>(queryParams.getPageNum(), queryParams.getPageSize()),queryParams); new Page<>(queryParams.getPageNum(), queryParams.getPageSize()),
queryParams
);
} }
} }

View File

@@ -1,67 +0,0 @@
package com.youlai.boot.system.service.impl;
import com.baomidou.mybatisplus.core.conditions.update.LambdaUpdateWrapper;
import com.baomidou.mybatisplus.core.metadata.IPage;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
import com.youlai.boot.core.security.util.SecurityUtils;
import com.youlai.boot.system.mapper.NoticeStatusMapper;
import com.youlai.boot.system.model.entity.NoticeStatus;
import com.youlai.boot.system.model.query.NoticeQuery;
import com.youlai.boot.system.model.vo.NoticeStatusVO;
import com.youlai.boot.system.model.vo.NoticeVO;
import com.youlai.boot.system.service.NoticeStatusService;
import lombok.RequiredArgsConstructor;
import org.springframework.stereotype.Service;
import java.util.List;
/**
* 用户公告状态服务实现类
*
* @author youlaitech
* @since 2024-08-28 16:56
*/
@Service
@RequiredArgsConstructor
public class NoticeStatusServiceImpl extends ServiceImpl<NoticeStatusMapper, NoticeStatus> implements NoticeStatusService {
private final NoticeStatusMapper noticeStatusMapper;
/**
* 获取未读的通知公告
* @return 公告列表
*/
@Override
public List<NoticeStatusVO> listUnreadNotices() {
//获取当前登录用户
Long userId = SecurityUtils.getUserId();
return noticeStatusMapper.listUnreadNotices(userId);
}
/**
* 全部标记为已读
* @return 是否成功
*/
@Override
public boolean readAll() {
Long userId = SecurityUtils.getUserId();
LambdaUpdateWrapper<NoticeStatus> updateWrapper = new LambdaUpdateWrapper<>();
updateWrapper.eq(NoticeStatus::getUserId, userId);
updateWrapper.set(NoticeStatus::getReadStatus, 1);
return this.update(updateWrapper);
}
/**
* 分页获取我的通知公告
* @param page 分页对象
* @param queryParams 查询参数
* @return 通知公告分页列表
*/
@Override
public IPage<NoticeStatusVO> getMyNoticePage(Page<NoticeVO> page, NoticeQuery queryParams) {
return this.getBaseMapper().getMyNoticePage(new Page<>(queryParams.getPageNum(), queryParams.getPageSize()),queryParams);
}
}

View File

@@ -0,0 +1,74 @@
package com.youlai.boot.system.service.impl;
import com.baomidou.mybatisplus.core.conditions.update.LambdaUpdateWrapper;
import com.baomidou.mybatisplus.core.metadata.IPage;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
import com.youlai.boot.core.security.util.SecurityUtils;
import com.youlai.boot.system.mapper.UserNoticeMapper;
import com.youlai.boot.system.model.entity.UserNotice;
import com.youlai.boot.system.model.query.NoticePageQuery;
import com.youlai.boot.system.model.vo.UserNoticePageVO;
import com.youlai.boot.system.model.vo.NoticeVO;
import com.youlai.boot.system.service.UserNoticeService;
import lombok.RequiredArgsConstructor;
import org.springframework.stereotype.Service;
import java.util.List;
/**
* 用户公告状态服务实现类
*
* @author youlaitech
* @since 2024-08-28 16:56
*/
@Service
@RequiredArgsConstructor
public class UserNoticeServiceImpl extends ServiceImpl<UserNoticeMapper, UserNotice> implements UserNoticeService {
private final UserNoticeMapper userNoticeMapper;
/**
* 获取未读的通知公告
*
* @return 公告列表
*/
@Override
public List<UserNoticePageVO> listUnreadNotices() {
//获取当前登录用户
Long userId = SecurityUtils.getUserId();
return userNoticeMapper.listUnreadNotices(userId);
}
/**
* 全部标记为已读
*
* @return 是否成功
*/
@Override
public boolean readAll() {
Long userId = SecurityUtils.getUserId();
return this.update(new LambdaUpdateWrapper<UserNotice>()
.eq(UserNotice::getUserId, userId)
.eq(UserNotice::getIsRead, 0)
.set(UserNotice::getIsRead, 1)
);
}
/**
* 我的通知公告分页列表
*
* @param page 分页对象
* @param queryParams 查询参数
* @return 通知公告分页列表
*/
@Override
public IPage<UserNoticePageVO> getMyNoticePage(Page<NoticeVO> page, NoticePageQuery queryParams) {
return this.getBaseMapper().getMyNoticePage(
new Page<>(queryParams.getPageNum(), queryParams.getPageSize()),
queryParams
);
}
}

View File

@@ -10,7 +10,7 @@ import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
import com.youlai.boot.common.constant.RedisConstants; import com.youlai.boot.common.constant.RedisConstants;
import com.youlai.boot.common.constant.SystemConstants; import com.youlai.boot.common.constant.SystemConstants;
import com.youlai.boot.common.enums.ContactType; import com.youlai.boot.system.enums.ContactType;
import com.youlai.boot.common.model.Option; import com.youlai.boot.common.model.Option;
import com.youlai.boot.module.mail.service.MailService; import com.youlai.boot.module.mail.service.MailService;
import com.youlai.boot.module.sms.service.SmsService; import com.youlai.boot.module.sms.service.SmsService;

View File

@@ -56,6 +56,7 @@ spring:
from: youlaitech@163.com from: youlaitech@163.com
mybatis-plus: mybatis-plus:
mapper-locations: classpath*:/mapper/**/*.xml
global-config: global-config:
db-config: db-config:
# 主键ID类型 # 主键ID类型
@@ -73,6 +74,7 @@ mybatis-plus:
log-impl: org.apache.ibatis.logging.stdout.StdOutImpl log-impl: org.apache.ibatis.logging.stdout.StdOutImpl
# 安全配置 # 安全配置
security: security:
jwt: jwt:

View File

@@ -40,6 +40,7 @@ spring:
# 缓存null值防止缓存穿透 # 缓存null值防止缓存穿透
cache-null-values: true cache-null-values: true
mybatis-plus: mybatis-plus:
mapper-locations: classpath*:/mapper/**/*.xml
global-config: global-config:
db-config: db-config:
# 主键ID类型 # 主键ID类型

View File

@@ -1,49 +0,0 @@
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.youlai.boot.system.mapper.NoticeStatusMapper">
<select id="listUnreadNotices" resultType="com.youlai.boot.system.model.vo.NoticeStatusVO">
SELECT
sn.id,
sn.title,
sns.read_status
FROM
sys_notice_status sns
LEFT JOIN
sys_notice sn
ON
sn.id = sns.notice_id
WHERE
user_id = #{userId}
AND sns.read_status = 0
ORDER BY
sn.release_time DESC
</select>
<select id="getMyNoticePage" resultType="com.youlai.boot.system.model.vo.NoticeStatusVO">
SELECT
sn.id,
sn.title,
sn.notice_type,
sn.release_by,
sn.priority,
sn.release_time,
sns.read_status
FROM
sys_notice sn
LEFT JOIN
sys_notice_status sns
ON
sn.id = sns.notice_id
<where>
<if test="queryParams.userId != null">
sns.user_id = #{queryParams.userId}
</if>
<if test="queryParams.title != null and queryParams.title != ''">
AND sn.title LIKE CONCAT('%',#{queryParams.title},'%')
</if>
</where>
ORDER BY
sn.release_time DESC
</select>
</mapper>

View File

@@ -44,7 +44,7 @@
INNER JOIN sys_role_menu t2 ON t1.id = t2.menu_id INNER JOIN sys_role_menu t2 ON t1.id = t2.menu_id
INNER JOIN sys_role t3 ON t2.role_id = t3.id AND t3.status = 1 AND t3.is_deleted = 0 INNER JOIN sys_role t3 ON t2.role_id = t3.id AND t3.status = 1 AND t3.is_deleted = 0
WHERE WHERE
t1.type != '${@com.youlai.boot.common.enums.MenuTypeEnum@BUTTON.getValue()}' t1.type != '${@com.youlai.boot.system.enums.MenuTypeEnum@BUTTON.getValue()}'
<if test="roles != null and roles.size() > 0"> <if test="roles != null and roles.size() > 0">
<!-- ROOT 可查看所有菜单 --> <!-- ROOT 可查看所有菜单 -->
<if test="!roles.contains('ROOT')"> <if test="!roles.contains('ROOT')">

View File

@@ -5,59 +5,56 @@
<!-- 获取通知公告分页列表 --> <!-- 获取通知公告分页列表 -->
<select id="getNoticePage" resultType="com.youlai.boot.system.model.bo.NoticeBO"> <select id="getNoticePage" resultType="com.youlai.boot.system.model.bo.NoticeBO">
SELECT SELECT
sn.id, t1.id,
sn.title, t1.title,
sn.notice_type, t1.notice_type,
su.nickname AS release_by, t2.nickname AS release_by,
sn.priority, t1.priority,
sn.tar_type, t1.tar_type,
sn.release_status, t1.release_status,
sn.release_time, t1.release_time,
sn.recall_time t1.recall_time
FROM FROM
sys_notice sn sys_notice t1
LEFT JOIN LEFT JOIN sys_user t2 ON t2.id = t1.release_by
sys_user su ON su.id = sn.release_by
where where
sn.is_deleted = 0 t1.is_deleted = 0
<if test="queryParams.title != null and queryParams.title != ''"> <if test="queryParams.title != null and queryParams.title != ''">
AND sn.title LIKE CONCAT('%',#{queryParams.title},'%') AND t1.title LIKE CONCAT('%',#{queryParams.title},'%')
</if> </if>
<if test="queryParams.releaseStatus != null"> <if test="queryParams.releaseStatus != null">
AND sn.release_status = #{queryParams.releaseStatus} AND t1.release_status = #{queryParams.releaseStatus}
</if> </if>
<if test="queryParams.releaseTime != null"> <if test="queryParams.releaseTime != null">
<if test="queryParams.releaseTime[0] != null and queryParams.releaseTime[0] != ''"> <if test="queryParams.releaseTime[0] != null and queryParams.releaseTime[0] != ''">
<bind name="startDate" value="queryParams.releaseTime[0].length() == 10 ? queryParams.releaseTime[0] + ' 00:00:00' : queryParams.releaseTime[0]"/> <bind name="startDate" value="queryParams.releaseTime[0].length() == 10 ? queryParams.releaseTime[0] + ' 00:00:00' : queryParams.releaseTime[0]"/>
AND sn.release_time &gt;= #{startDate} AND t1.release_time &gt;= #{startDate}
</if> </if>
<if test="queryParams.releaseTime[1] != null and queryParams.releaseTime[1] != ''"> <if test="queryParams.releaseTime[1] != null and queryParams.releaseTime[1] != ''">
<bind name="endDate" value="queryParams.releaseTime[1].length() == 10 ? queryParams.releaseTime[1] + ' 23:59:59' : queryParams.releaseTime[1]"/> <bind name="endDate" value="queryParams.releaseTime[1].length() == 10 ? queryParams.releaseTime[1] + ' 23:59:59' : queryParams.releaseTime[1]"/>
AND sn.release_time &lt;= #{endDate} AND t1.release_time &lt;= #{endDate}
</if> </if>
</if> </if>
ORDER BY ORDER BY
id t1.create_time DESC
DESC
</select> </select>
<select id="getReadNoticeVO" resultType="com.youlai.boot.system.model.vo.NoticeDetailVO"> <!-- 获取通知公告详情 -->
<select id="getNoticeDetail" resultType="com.youlai.boot.system.model.bo.NoticeBO">
SELECT SELECT
sn.id, t1.id,
sn.title, t1.title,
sn.content, t1.content,
sn.notice_type, t1.type,
su.nickname AS release_by, t2.nickname AS release_by,
sn.priority, t1.level,
sn.release_status, t1.release_status,
sn.release_time t1.release_time
FROM FROM
sys_notice sn sys_notice t1
LEFT JOIN LEFT JOIN sys_user t2 ON t2.id = t1.release_by
sys_user su ON su.id = sn.release_by WHERE
where t1.id = #{id} AND t1.is_deleted = 0
sn.is_deleted = 0
AND sn.id = #{id}
</select> </select>
</mapper> </mapper>

View File

@@ -33,7 +33,7 @@
INNER JOIN sys_role t2 ON t1.role_id = t2.id AND t2.is_deleted = 0 AND t2.`status` = 1 INNER JOIN sys_role t2 ON t1.role_id = t2.id AND t2.is_deleted = 0 AND t2.`status` = 1
INNER JOIN sys_menu t3 ON t1.menu_id = t3.id INNER JOIN sys_menu t3 ON t1.menu_id = t3.id
WHERE WHERE
type = '${@com.youlai.boot.common.enums.MenuTypeEnum@BUTTON.getValue()}' type = '${@com.youlai.boot.system.enums.MenuTypeEnum@BUTTON.getValue()}'
<if test="roleCode!=null and roleCode.trim() neq ''"> <if test="roleCode!=null and roleCode.trim() neq ''">
AND t2.`code` = #{roleCode} AND t2.`code` = #{roleCode}
</if> </if>
@@ -48,7 +48,7 @@
INNER JOIN sys_menu t2 ON t2.id = t1.menu_id INNER JOIN sys_menu t2 ON t2.id = t1.menu_id
INNER JOIN sys_role t3 ON t3.id = t1.role_id INNER JOIN sys_role t3 ON t3.id = t1.role_id
WHERE WHERE
t2.type = '${@com.youlai.boot.common.enums.MenuTypeEnum@BUTTON.getValue()}' t2.type = '${@com.youlai.boot.system.enums.MenuTypeEnum@BUTTON.getValue()}'
AND t2.perm IS NOT NULL AND t2.perm IS NOT NULL
AND t3.CODE IN AND t3.CODE IN
<foreach collection="roles" item="role" separator="," open="(" close=")"> <foreach collection="roles" item="role" separator="," open="(" close=")">

View File

@@ -0,0 +1,44 @@
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.youlai.boot.system.mapper.UserNoticeMapper">
<!-- 获取用户未读通知列表 -->
<select id="listUnreadNotices" resultType="com.youlai.boot.system.model.vo.UserNoticePageVO">
SELECT
t1.id,
t1.title,
t2.is_read
FROM
sys_user_notice t1
INNER JOIN sys_notice t2 ON t2.id = t1.notice_id AND t2.publish_status = 1
WHERE
user_id = #{userId} AND t1.is_read = 0
ORDER BY
t2.publish_time DESC
</select>
<!-- 获取我的通知分页列表 -->
<select id="getMyNoticePage" resultType="com.youlai.boot.system.model.vo.UserNoticePageVO">
SELECT
t1.id,
t1.title,
t4.name typeLabel,
t3.nickname publisherName,
t5.name levelLabel,
t1.publish_time,
t2.read_status
FROM
sys_user_notice t1
INNER JOIN sys_notice ON t2.notice_id = t1.id AND t2.publish_status = 1
LEFT JOIN sys_user t3 ON t2.publisher_id = t3.id
LEFT JOIN sys_dict_item t4 ON t2.type = t4.value AND t4.dict_code = 'notice_type'
LEFT JOIN sys_dict_item t5 ON t2.level = t5.value AND t4.dict_code = 'notice_level'
WHERE
t1.user_id = #{queryParams.userId}
<if test="queryParams.title != null and queryParams.title != ''">
AND t2.title LIKE CONCAT('%',#{queryParams.title},'%')
</if>
ORDER BY
t2.release_time DESC
</select>
</mapper>