mirror of
https://gitee.com/y_project/RuoYi-Cloud.git
synced 2026-04-26 17:27:51 +08:00
230 lines
7.5 KiB
Vue
230 lines
7.5 KiB
Vue
<template>
|
|
<div>
|
|
<el-popover ref="noticePopover" placement="bottom-end" width="320" trigger="manual" :value="noticeVisible" popper-class="notice-popover">
|
|
<div class="notice-header">
|
|
<span class="notice-title">通知公告</span>
|
|
<span class="notice-mark-all" @click="markAllRead">全部已读</span>
|
|
</div>
|
|
<div v-if="noticeLoading" class="notice-loading"><i class="el-icon-loading"></i> 加载中...</div>
|
|
<div v-else-if="noticeList.length === 0" class="notice-empty"><i class="el-icon-inbox"></i><br>暂无公告</div>
|
|
<div v-else>
|
|
<div v-for="item in noticeList" :key="item.noticeId" class="notice-item" :class="{ 'is-read': item.isRead }" @click="previewNotice(item)">
|
|
<el-tag size="mini" :type="item.noticeType === '1' ? 'warning' : 'success'" class="notice-tag">
|
|
{{ item.noticeType === '1' ? '通知' : '公告' }}
|
|
</el-tag>
|
|
<span class="notice-item-title">{{ item.noticeTitle }}</span>
|
|
<span class="notice-item-date">{{ item.createTime }}</span>
|
|
</div>
|
|
</div>
|
|
</el-popover>
|
|
|
|
<div v-popover:noticePopover class="right-menu-item hover-effect notice-trigger" @mouseenter="onNoticeEnter" @mouseleave="onNoticeLeave">
|
|
<svg-icon icon-class="bell" />
|
|
<span v-if="unreadCount > 0" class="notice-badge">{{ unreadCount }}</span>
|
|
</div>
|
|
|
|
<el-dialog :title="previewTitle" :visible.sync="previewVisible" width="680px" append-to-body custom-class="notice-preview-dialog">
|
|
<div class="notice-preview-meta">
|
|
<el-tag size="small" :type="previewNoticeType === '1' ? 'warning' : 'success'">
|
|
{{ previewNoticeType === '1' ? '通知' : '公告' }}
|
|
</el-tag>
|
|
<span class="notice-preview-info"><i class="el-icon-user"></i> {{ previewCreateBy }}</span>
|
|
<span class="notice-preview-info"><i class="el-icon-time"></i> {{ previewCreateTime }}</span>
|
|
</div>
|
|
<div class="notice-preview-divider"></div>
|
|
<div class="notice-preview-content" v-html="previewContent"></div>
|
|
</el-dialog>
|
|
</div>
|
|
</template>
|
|
|
|
<script>
|
|
import { listNoticeTop, markNoticeRead, markNoticeReadAll, getNotice } from '@/api/system/notice'
|
|
|
|
export default {
|
|
name: 'HeaderNotice',
|
|
data() {
|
|
return {
|
|
noticeList: [], // 通知列表
|
|
unreadCount: 0, // 未读数量
|
|
noticeLoading: false, // 加载状态
|
|
noticeVisible: false, // 弹出层显示状态
|
|
noticeLeaveTimer: null, // 鼠标离开计时器
|
|
previewVisible: false, // 预览弹窗显示状态
|
|
previewTitle: '', // 预览弹窗标题
|
|
previewContent: '', // 预览弹窗内容
|
|
previewNoticeType: '', // 预览弹窗类型
|
|
previewCreateBy: '', // 预览弹窗创建人
|
|
previewCreateTime: '' // 预览弹窗创建时间
|
|
}
|
|
},
|
|
mounted() {
|
|
this.loadNoticeTop()
|
|
},
|
|
methods: {
|
|
// 鼠标移入铃铛区域
|
|
onNoticeEnter() {
|
|
clearTimeout(this.noticeLeaveTimer)
|
|
this.noticeVisible = true
|
|
this.$nextTick(() => {
|
|
const popper = this.$refs.noticePopover.$refs.popper
|
|
if (popper && !popper._noticeBound) {
|
|
popper._noticeBound = true
|
|
popper.addEventListener('mouseenter', () => clearTimeout(this.noticeLeaveTimer))
|
|
popper.addEventListener('mouseleave', () => {
|
|
this.noticeLeaveTimer = setTimeout(() => { this.noticeVisible = false }, 100)
|
|
})
|
|
}
|
|
})
|
|
},
|
|
// 鼠标离开铃铛区域
|
|
onNoticeLeave() {
|
|
this.noticeLeaveTimer = setTimeout(() => { this.noticeVisible = false }, 150)
|
|
},
|
|
// 加载顶部公告列表
|
|
loadNoticeTop() {
|
|
this.noticeLoading = true
|
|
listNoticeTop().then(res => {
|
|
this.noticeList = res.data || []
|
|
this.unreadCount = res.unreadCount !== undefined ? res.unreadCount : this.noticeList.filter(n => !n.isRead).length
|
|
}).finally(() => {
|
|
this.noticeLoading = false
|
|
})
|
|
},
|
|
// 预览公告详情
|
|
previewNotice(item) {
|
|
if (!item.isRead) {
|
|
markNoticeRead(item.noticeId).catch(() => {})
|
|
item.isRead = true
|
|
const idx = this.noticeList.indexOf(item)
|
|
if (idx !== -1) this.$set(this.noticeList, idx, { ...item, isRead: true })
|
|
this.unreadCount = Math.max(0, this.unreadCount - 1)
|
|
}
|
|
getNotice(item.noticeId).then(res => {
|
|
const notice = res.data
|
|
this.previewTitle = notice.noticeTitle
|
|
this.previewContent = notice.noticeContent
|
|
this.previewNoticeType = notice.noticeType
|
|
this.previewCreateBy = notice.createBy
|
|
this.previewCreateTime = notice.createTime
|
|
this.previewVisible = true
|
|
})
|
|
},
|
|
// 全部已读
|
|
markAllRead() {
|
|
const ids = this.noticeList.map(n => n.noticeId).join(',')
|
|
if (!ids) return
|
|
markNoticeReadAll(ids).catch(() => {})
|
|
this.noticeList = this.noticeList.map(n => ({ ...n, isRead: true }))
|
|
this.unreadCount = 0
|
|
}
|
|
}
|
|
}
|
|
</script>
|
|
|
|
<style lang="scss" scoped>
|
|
.notice-trigger {
|
|
position: relative;
|
|
transform: translateX(-6px);
|
|
.svg-icon { width: 1.2em; height: 1.2em; vertical-align: -0.2em; }
|
|
.notice-badge {
|
|
position: absolute;
|
|
top: 7px;
|
|
right: -3px;
|
|
background: #f56c6c;
|
|
color: #fff;
|
|
border-radius: 10px;
|
|
font-size: 10px;
|
|
height: 16px;
|
|
line-height: 16px;
|
|
padding: 0 4px;
|
|
min-width: 16px;
|
|
text-align: center;
|
|
white-space: nowrap;
|
|
pointer-events: none;
|
|
}
|
|
}
|
|
.notice-popover {
|
|
padding: 0 !important;
|
|
}
|
|
.notice-popover .notice-header {
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: space-between;
|
|
padding: 10px 14px;
|
|
background: #f7f9fb;
|
|
border-bottom: 1px solid #eee;
|
|
font-size: 13px;
|
|
font-weight: 600;
|
|
color: #333;
|
|
}
|
|
.notice-popover .notice-mark-all {
|
|
font-size: 12px;
|
|
color: #409EFF;
|
|
font-weight: normal;
|
|
cursor: pointer;
|
|
}
|
|
.notice-popover .notice-mark-all:hover { color: #2b7cc1; }
|
|
.notice-popover .notice-loading,
|
|
.notice-popover .notice-empty {
|
|
padding: 24px;
|
|
text-align: center;
|
|
color: #bbb;
|
|
font-size: 12px;
|
|
line-height: 1.8;
|
|
}
|
|
.notice-popover .notice-item {
|
|
display: flex;
|
|
align-items: center;
|
|
gap: 8px;
|
|
padding: 10px 14px;
|
|
border-bottom: 1px solid #f5f5f5;
|
|
cursor: pointer;
|
|
transition: background 0.15s;
|
|
}
|
|
.notice-popover .notice-item:last-child { border-bottom: none; }
|
|
.notice-popover .notice-item:hover { background: #f7f9fb; }
|
|
.notice-popover .notice-item.is-read .notice-tag,
|
|
.notice-popover .notice-item.is-read .notice-item-title,
|
|
.notice-popover .notice-item.is-read .notice-item-date { opacity: 0.45; filter: grayscale(1); color: #999; }
|
|
.notice-popover .notice-tag { flex-shrink: 0; }
|
|
.notice-popover .notice-item-title {
|
|
flex: 1;
|
|
font-size: 12px;
|
|
color: #333;
|
|
overflow: hidden;
|
|
white-space: nowrap;
|
|
text-overflow: ellipsis;
|
|
}
|
|
.notice-popover .notice-item-date {
|
|
flex-shrink: 0;
|
|
font-size: 11px;
|
|
color: #bbb;
|
|
}
|
|
::v-deep .notice-preview-dialog {
|
|
.el-dialog__body { padding: 0 20px 20px; }
|
|
.notice-preview-meta {
|
|
display: flex;
|
|
align-items: center;
|
|
gap: 14px;
|
|
padding: 12px 0;
|
|
font-size: 12px;
|
|
color: #888;
|
|
.notice-preview-info { display: flex; align-items: center; gap: 4px; }
|
|
}
|
|
.notice-preview-divider {
|
|
height: 1px;
|
|
background: linear-gradient(to right, transparent, #e2e8f0, transparent);
|
|
margin-bottom: 16px;
|
|
}
|
|
.notice-preview-content {
|
|
font-size: 14px;
|
|
line-height: 1.85;
|
|
color: #2d3748;
|
|
word-break: break-word;
|
|
img { max-width: 100%; border-radius: 4px; }
|
|
p { margin: 0 0 1em; }
|
|
a { color: #409EFF; text-decoration: underline; }
|
|
}
|
|
}
|
|
</style>
|