refactor: ♻️ 通知公告、字典重构

This commit is contained in:
ray
2024-10-03 17:42:51 +08:00
parent 476f257bc1
commit 3db4b2a6e5
25 changed files with 931 additions and 669 deletions

View File

@@ -16,18 +16,16 @@
</template>
<script setup lang="ts">
import DictAPI from "@/api/dict";
import DictDataAPI from "@/api/dict-data";
const props = defineProps({
/**
* 字典编码(eg: 性别-gender)
*/
code: {
type: String,
required: true,
},
modelValue: {
type: [String, Number],
required: true,
},
placeholder: {
type: String,
@@ -39,38 +37,42 @@ const props = defineProps({
},
});
const emits = defineEmits(["update:modelValue"]);
// 使用 defineEmits 声明 emits
const emit = defineEmits(["update:modelValue"]);
const options: Ref<OptionType[]> = ref([]);
// 下拉框选项
const options = ref<Array<{ label: string; value: string | number }>>([]);
const selectedValue = ref<string | number | undefined>(props.modelValue);
const selectedValue = ref<string | number | undefined>();
watch([options, () => props.modelValue], ([newOptions, newModelValue]) => {
if (newOptions.length === 0) {
// 下拉数据源加载未完成不回显
return;
// 监听 modelValue 变化
watch(
() => props.modelValue,
(newValue) => {
// 类型转换确保 selectedValue 和 option.value 类型一致
if (typeof options.value[0]?.value === "number") {
selectedValue.value = Number(newValue);
} else {
selectedValue.value = String(newValue);
}
}
if (newModelValue == undefined) {
selectedValue.value = undefined;
return;
}
if (typeof newOptions[0].value === "number") {
selectedValue.value = Number(newModelValue);
} else if (typeof newOptions[0].value === "string") {
selectedValue.value = String(newModelValue);
} else {
selectedValue.value = newModelValue;
}
});
);
function handleChange(val?: string | number | undefined) {
emits("update:modelValue", val);
// 监听 selectedValue 的变化并触发 update:modelValue
function handleChange(val?: string | number) {
emit("update:modelValue", val);
}
onBeforeMount(() => {
// 根据字典编码获取字典项
DictAPI.getOptions(props.code).then((data) => {
options.value = data;
});
// 获取字典数据
onBeforeMount(async () => {
const data = await DictDataAPI.getOptions(props.code);
options.value = data;
// 初次加载时处理类型一致性
if (props.modelValue !== undefined) {
if (typeof options.value[0]?.value === "number") {
selectedValue.value = Number(props.modelValue);
} else {
selectedValue.value = String(props.modelValue);
}
}
});
</script>

View File

@@ -1,43 +1,41 @@
<template>
<div>
<el-dropdown trigger="hover">
<el-badge :is-dot="messages.length > 0" :offset="offset">
<div class="flex-center h100% p10px">
<i-ep-bell />
</div>
<el-dropdown class="flex-center h-full align-middle">
<el-badge v-if="messages.length > 0" :value="messages.length" :max="99">
<div><i-ep-bell /></div>
</el-badge>
<el-badge v-else>
<i-ep-bell />
</el-badge>
<template #dropdown>
<div class="px-5 py-2">
<el-tabs v-model="activeTab">
<el-tab-pane
v-for="(label, key) in MessageTypeLabels"
:label="label"
:name="key"
:key="key"
<div class="p-5">
<template v-if="messages.length > 0">
<div
class="w400px flex-x-between py-2"
v-for="(item, index) in messages"
:key="index"
>
<template v-if="messages.length > 0">
<div
class="w-[380px] py-2"
v-for="(item, index) in messages"
:key="index"
<div>
<el-tag type="success" size="small">系统通知</el-tag>
<el-link
type="primary"
@click="readNotice(item.id)"
class="ml-1"
>
<el-link type="primary" @click="readNotice(item.id)">
<el-badge is-dot />
<el-text class="w-350px" size="default" truncated>
{{ item.title }}
</el-text>
</el-link>
</div>
</template>
<template v-else>
<div class="flex-center h-150px w-350px">
<el-text class="w-350px" size="default" truncated>
<el-empty :image-size="30" description="暂无消息" />
</el-text>
</div>
</template>
</el-tab-pane>
</el-tabs>
{{ item.title }}
</el-link>
</div>
<div>
{{ item.publishTime }}
</div>
</div>
</template>
<template v-else>
<div class="flex-center h150px w350px">
<el-empty :image-size="30" description="暂无消息" />
</div>
</template>
<el-divider />
<div class="flex-x-between">
<el-link type="primary" :underline="false" @click="viewMore">
@@ -59,101 +57,58 @@
</template>
</el-dropdown>
<!-- 弹窗部分 -->
<el-dialog
v-model="modalVisible"
:show-close="false"
append-to-body
:fullscreen="fullscreen"
style="z-index: revert"
>
<template #header="{ close }">
<div class="flex-x-between">
<h3>{{ currentMessage.title }}</h3>
<div class="flex-center">
<el-icon
class="ml10px"
@click="
() => {
fullscreen = !fullscreen;
}
"
>
<FullScreen />
</el-icon>
<el-icon @click="close" class="icon">
<Close />
</el-icon>
</div>
</div>
</template>
<div style="width: auto; text-align: left">
<span class="header-item">
<el-tag v-if="currentMessage.type === 2" type="warning">
系统通知
</el-tag>
<el-tag v-if="currentMessage.type === 1" type="success">
通知消息
</el-tag>
</span>
<div v-html="currentMessage.content"></div>
</div>
</el-dialog>
<NoticeDetail ref="noticeDetailRef" />
</div>
</template>
<script setup lang="ts">
import { MessageTypeEnum, MessageTypeLabels } from "@/enums/MessageTypeEnum";
import NoticeAPI from "@/api/notice";
import NoticeAPI, { NoticePageQuery, NoticePageVO } from "@/api/notice";
import WebSocketManager from "@/utils/socket";
import router from "@/router";
// 状态和引用
const activeTab = ref(MessageTypeEnum.MESSAGE);
const messages = ref<any[]>([]);
const offset = ref<[number, number]>([-15, 15]);
const currentMessage = ref<any>({});
const modalVisible = ref(false);
const fullscreen = ref(false);
const messages = ref<NoticePageVO[]>([]);
const noticeDetailRef = ref();
// 获取未读消息列表并连接 WebSocket
onMounted(() => {
NoticeAPI.getUnreadList().then((data) => {
messages.value = data;
});
NoticeAPI.getMyNoticePage({ pageNum: 1, pageSize: 5, isRead: 0 }).then(
(data) => {
messages.value = data.list;
}
);
WebSocketManager.getOrCreateClient("/user/queue/message", (message) => {
const parsedMessage = JSON.parse(message);
if (parsedMessage.noticeType === MessageTypeEnum.MESSAGE) {
const content = JSON.parse(parsedMessage.content);
if (content.type === "release") {
const id = content.id;
if (!messages.value.some((msg) => msg.id === id)) {
messages.value.unshift({
id,
title: content.title,
type: MessageTypeEnum.MESSAGE,
});
}
}
WebSocketManager.subscribeToTopic("/user/queue/message", (message) => {
console.log("收到消息:", message);
const data = JSON.parse(message);
const id = data.id;
if (!messages.value.some((msg) => msg.id === id)) {
messages.value.unshift({
id,
title: data.title,
});
ElNotification({
title: "您收到一条新的通知消息!",
message: data.title,
type: "success",
position: "bottom-right",
});
}
});
});
// 阅读通知公告
function readNotice(id: number) {
function readNotice(id: string) {
noticeDetailRef.value.openNotice(id);
const index = messages.value.findIndex((msg) => msg.id === id);
if (index >= 0) {
currentMessage.value = messages.value[index];
modalVisible.value = true;
messages.value.splice(index, 1); // 从消息列表中移除已读消息
}
}
// 查看更多
function viewMore() {
router.push({ path: "/notice/notice" });
router.push({ path: "/myNotice" });
}
// 全部已读
@@ -164,4 +119,4 @@ function markAllAsRead() {
}
</script>
<style scoped></style>
<style lang="scss" scoped></style>