You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
crmeb/app/pages/supply_chain/notice/index.vue

405 lines
9.7 KiB

<template>
<view class="notice-list-page">
<view class="content">
<!-- 搜索与筛选 -->
<view class="search-section">
<view class="search-box">
<input v-model="query.noticeTitle" type="text" placeholder="请输入通知标题" />
<view class="search-btn" @click="refreshList">
搜索
</view>
</view>
<view class="filter-row">
<text class="filter-label">通知类型</text>
<picker
mode="selector"
:range="noticeTypeOptions"
range-key="label"
:value="selectedNoticeTypeIndex"
@change="onNoticeTypeChange"
>
<view class="filter-picker">
<text class="picker-text">{{ selectedNoticeTypeLabel }}</text>
<text class="picker-arrow">▼</text>
</view>
</picker>
</view>
</view>
<!-- 列表 -->
<scroll-view scroll-y class="list-scroll" @scrolltolower="loadMore">
<view class="empty" v-if="!loading && records.length === 0">
<text class="iconfont icon-wushuju"></text>
<text class="text">暂无通知公告</text>
</view>
<view
v-for="item in records"
:key="item.noticeId"
class="record-card"
@click="openDetail(item.noticeId)"
>
<view class="card-header">
<view class="left">
<view class="title-row">
<text class="type line1">{{ item.noticeTitle || '未命名通知' }}</text>
</view>
</view>
<text class="time">{{ item.noticeTime }}</text>
</view>
<view class="card-body">
<view class="row">
<text class="label">通知类型:</text>
<text class="value">{{ dict.getLabel('sys_notice_type', item.noticeType) || item.noticeType || '—' }}</text>
</view>
<view class="row">
<text class="label">通知范围:</text>
<text class="value">{{ item.noticeScope || '—' }}</text>
</view>
<view class="row">
<text class="label">通知单位:</text>
<text class="value">{{ item.draftDeptName || '—' }}</text>
</view>
<view class="preview" v-if="item.noticeContent">
{{ contentPreview(item.noticeContent) }}
</view>
</view>
<view class="card-footer">
<view class="detail-btn" @click.stop="openDetail(item.noticeId)">查看详情</view>
</view>
</view>
<view class="load-more" v-if="loading">加载中...</view>
<view class="load-more" v-else-if="finished && records.length > 0">已加载全部</view>
</scroll-view>
</view>
</view>
</template>
<script>
import { pubnoticeListApi } from '@/api/pubnotice.js';
export default {
dicts: ['sys_notice_type'],
data() {
return {
query: {
noticeTitle: '',
noticeType: '',
},
records: [],
page: 1,
limit: 10,
loading: false,
finished: false,
};
},
onLoad() {
this.refreshList();
},
onPullDownRefresh() {
this.refreshList();
},
computed: {
noticeTypeOptions() {
const dictList = this.dict.get('sys_notice_type') || [];
const options = dictList.map(item => ({
label: item.dictLabel,
value: item.dictValue,
}));
return [{ label: '全部', value: '' }].concat(options);
},
selectedNoticeTypeIndex() {
const index = this.noticeTypeOptions.findIndex(item => item.value === this.query.noticeType);
return index >= 0 ? index : 0;
},
selectedNoticeTypeLabel() {
const current = this.noticeTypeOptions[this.selectedNoticeTypeIndex];
return current ? current.label : '全部';
},
},
methods: {
onNoticeTypeChange(e) {
const index = Number(e?.detail?.value);
const current = this.noticeTypeOptions[index];
this.query.noticeType = current ? current.value : '';
this.refreshList();
},
async refreshList() {
this.page = 1;
this.records = [];
this.finished = false;
await this.fetchList();
uni.stopPullDownRefresh && uni.stopPullDownRefresh();
},
async loadMore() {
if (this.loading || this.finished) return;
this.page += 1;
await this.fetchList();
},
buildParams() {
const params = {
page: this.page,
limit: this.limit,
delFlag: '1',
};
const noticeTitle = (this.query.noticeTitle || '').trim();
if (noticeTitle) params.noticeTitle = noticeTitle;
if (this.query.noticeType) {
params.noticeType = this.query.noticeType;
// 兼容后端仍使用 type 字段筛选的场景
params.type = this.query.noticeType;
}
return params;
},
async fetchList() {
this.loading = true;
try {
const params = this.buildParams();
const res = await pubnoticeListApi(params);
const data = res?.data || {};
const list = data?.list || [];
if (this.page === 1) {
this.records = list;
} else {
this.records = this.records.concat(list);
}
const total = data?.total ?? data?.count ?? res?.total ?? res?.count;
if (!list.length) {
this.finished = true;
} else if (typeof total === 'number' && total <= this.records.length) {
this.finished = true;
}
} catch (e) {
uni.showToast({
title: typeof e === 'string' ? e : '获取通知公告失败',
icon: 'none',
});
} finally {
this.loading = false;
}
},
openDetail(noticeId) {
if (!noticeId) return;
uni.navigateTo({
url: `/pages/supply_chain/notice_detail/index?noticeId=${noticeId}`,
});
},
contentPreview(content) {
if (content === undefined || content === null) return '';
const s = String(content)
.replace(/<[^>]+>/g, ' ')
.replace(/\s+/g, ' ')
.trim();
if (!s) return '';
return s.length > 80 ? s.slice(0, 80) + '...' : s;
},
},
};
</script>
<style lang="scss">
.notice-list-page {
.content {
padding: 30rpx;
}
.search-section {
margin-bottom: 20rpx;
.search-box {
display: flex;
align-items: center;
background-color: #fff;
border-radius: 12rpx;
box-shadow: 0 2rpx 10rpx rgba(0, 0, 0, 0.06);
padding: 14rpx 16rpx;
margin-bottom: 20rpx;
input {
flex: 1;
height: 64rpx;
font-size: 26rpx;
color: #333;
}
.search-btn {
height: 64rpx;
padding: 0 18rpx;
background-color: #409eff;
color: #fff;
border-radius: 32rpx;
display: flex;
align-items: center;
justify-content: center;
font-size: 24rpx;
font-weight: 600;
}
}
.filter-row {
display: flex;
align-items: center;
justify-content: space-between;
background-color: #fff;
border-radius: 12rpx;
box-shadow: 0 2rpx 10rpx rgba(0, 0, 0, 0.06);
padding: 0 20rpx;
height: 72rpx;
.filter-label {
font-size: 26rpx;
color: #666;
}
.filter-picker {
display: flex;
align-items: center;
max-width: 420rpx;
.picker-text {
font-size: 26rpx;
color: #409eff;
max-width: 360rpx;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
.picker-arrow {
margin-left: 10rpx;
font-size: 18rpx;
color: #999;
}
}
}
}
.list-scroll {
max-height: calc(100vh - 310rpx);
}
.record-card {
background-color: #fff;
border-radius: 10rpx;
padding: 24rpx 26rpx;
margin-bottom: 20rpx;
box-shadow: 0 2rpx 10rpx rgba(0, 0, 0, 0.06);
.card-header {
display: flex;
align-items: flex-start;
justify-content: space-between;
margin-bottom: 16rpx;
.left {
flex: 1;
min-width: 0;
}
.title-row {
margin-bottom: 10rpx;
}
.type {
font-size: 28rpx;
font-weight: 700;
color: #333;
max-width: 520rpx;
}
.time {
font-size: 22rpx;
color: #999;
white-space: nowrap;
}
}
.card-body {
.row {
display: flex;
margin-bottom: 10rpx;
.label {
width: 170rpx;
font-size: 24rpx;
color: #666;
}
.value {
flex: 1;
font-size: 24rpx;
color: #333;
min-width: 0;
}
}
.preview {
margin-top: 16rpx;
font-size: 24rpx;
color: #999;
display: -webkit-box;
-webkit-line-clamp: 2;
-webkit-box-orient: vertical;
overflow: hidden;
}
}
.card-footer {
margin-top: 20rpx;
display: flex;
justify-content: flex-end;
.detail-btn {
padding: 8rpx 16rpx;
background-color: #409eff;
color: #fff;
border-radius: 16rpx;
font-size: 20rpx;
}
}
}
.empty {
margin-top: 80rpx;
display: flex;
flex-direction: column;
align-items: center;
color: #999;
.iconfont {
font-size: 80rpx;
margin-bottom: 16rpx;
}
.text {
font-size: 24rpx;
}
}
.load-more {
text-align: center;
padding: 20rpx 0 10rpx;
font-size: 22rpx;
color: #999;
}
.line1 {
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
}
}
</style>