feat: 通知公告;兼容小程序;

property-only-app
wx-jincw 2 months ago
parent bd98401d23
commit 5a4a507604

@ -0,0 +1,42 @@
import request from '@/utils/request.js';
/**
* pubnotice 新增
* @param {Object} data
*/
export function pubnoticeCreateApi(data) {
return request.post('autogencode/pubnotice/save', data, { useAdminUrl: true });
}
/**
* pubnotice 更新
* @param {Object} data
*/
export function pubnoticeUpdateApi(data) {
return request.post('autogencode/pubnotice/update', data, { useAdminUrl: true });
}
/**
* pubnotice 详情
* @param {string} id noticeId
*/
export function pubnoticeDetailApi(id) {
return request.get(`autogencode/pubnotice/info/${id}`, {}, { useAdminUrl: true });
}
/**
* pubnotice 批量删除
* @param {string[]|string} ids
*/
export function pubnoticeDeleteApi(ids) {
return request.post('autogencode/pubnotice/delete', ids, { useAdminUrl: true });
}
/**
* pubnotice 列表
* @param {Object} params query params (page/limit + 各字段条件)
*/
export function pubnoticeListApi(params) {
return request.get('autogencode/pubnotice/list', params, { useAdminUrl: true });
}

@ -123,6 +123,7 @@
/* */
"mp-weixin" : {
"appid" : "",
"libVersion" : "latest",
"setting" : {
"urlCheck" : true,
"minified" : true,

@ -249,6 +249,22 @@
"navigationBarBackgroundColor": "#409EFF",
"navigationBarTextStyle": "white"
}
},
{
"path": "notice/index",
"style": {
"navigationBarTitleText": "通知公告列表",
"navigationBarBackgroundColor": "#409EFF",
"navigationBarTextStyle": "white"
}
},
{
"path": "notice_detail/index",
"style": {
"navigationBarTitleText": "通知详情",
"navigationBarBackgroundColor": "#409EFF",
"navigationBarTextStyle": "white"
}
}
]
},

@ -50,8 +50,9 @@
<view class="iconfont icon-xiangyou"></view>
</view>
<!-- 供应链管理功能入口 -->
<view class='nav acea-row'>
<navigator class='item' url='/pages/supply_chain/traceability/index' hover-class='none'>
<view class='nav acea-row' v-if="menus.length">
<view class="property-service-title">物业服务</view>
<!-- <navigator class='item' url='/pages/supply_chain/traceability/index' hover-class='none'>
<view class='pictrue'>
<image src="/static/images/wg/wg_source.png"></image>
</view>
@ -92,7 +93,7 @@
<image src="/static/images/wg/wg_get.png"></image>
</view>
<view class="menu-txt">物资领用</view>
</navigator>
</navigator> -->
<navigator class='item' url='/pages/supply_chain/complaint/index' hover-class='none'>
<view class='pictrue'>
<image src="/static/images/wg/wg_jy.png"></image>
@ -1426,6 +1427,16 @@
color: #454545;
}
.property-service-title {
width: 100%;
box-sizing: border-box;
padding: 20rpx 24rpx 8rpx;
font-size: 28rpx;
font-weight: 600;
color: #2f3a4a;
letter-spacing: 1rpx;
}
.mp-bg {
position: absolute;
left: 0;

@ -22,7 +22,7 @@
<view class="item-header">
<view class="header-left">
<text class="approval-type">{{ item.type }}</text>
<text class="approval-status" :class="getStatusClass(item.status)">{{ item.status }}</text>
<text class="approval-status" :class="statusClassMap[item.status]">{{ item.status }}</text>
</view>
<text class="approval-time">{{ item.time }}</text>
</view>
@ -64,6 +64,10 @@ export default {
data() {
return {
activeStatus: 'all',
statusClassMap: {
'待处理': 'status-pending',
'已处理': 'status-processed'
},
approvals: [
{ id: 1, type: '采购申请', status: '待处理', time: '2026-03-07 10:00', applicant: '张三', content: '采购有机蔬菜100斤', amount: 100 },
{ id: 2, type: '领用申请', status: '待处理', time: '2026-03-07 09:30', applicant: '李四', content: '领用维修耗材', amount: 50 },
@ -87,14 +91,6 @@ export default {
goBack() {
uni.navigateBack();
},
getStatusClass(status) {
if (status === '待处理') {
return 'status-pending';
} else if (status === '已处理') {
return 'status-processed';
}
return '';
},
approveApproval(id) {
//
uni.showToast({

@ -123,7 +123,7 @@
<text class="type">{{ item.submitChannel === '0' ? '投诉' : '建议' }}</text>
<text
class="status"
:class="statusClass(item.status)"
:class="statusClassMap[item.status] || statusClassMap['__default__']"
>
{{ statusText(item.status) }}
</text>
@ -204,7 +204,18 @@ export default {
page: 1,
limit: 10,
loading: false,
finished: false
finished: false,
statusClassMap: {
'0': 'status-pending',
0: 'status-pending',
'1': 'status-pending',
1: 'status-pending',
'2': 'status-processing',
2: 'status-processing',
'3': 'status-done',
3: 'status-done',
'__default__': 'status-pending'
}
};
},
computed: {
@ -327,14 +338,6 @@ export default {
if (status === '3') return '已处理';
return status;
},
statusClass(status) {
const text = this.statusText(status);
if (text === '已处理') return 'status-done';
if (text === '待处理') return 'status-pending';
if (text === '未处理') return 'status-pending';
if (text === '处理中') return 'status-processing';
return '';
}
}
};
</script>

@ -42,7 +42,7 @@
<view class="record-item" v-for="(item, index) in receiptRecords" :key="index">
<view class="item-header">
<text class="material-name">{{ item.materialName }}</text>
<text class="receipt-status" :class="getStatusClass(item.status)">{{ item.status }}</text>
<text class="receipt-status" :class="statusClassMap[item.status]">{{ item.status }}</text>
</view>
<view class="item-body">
<view class="info-item">
@ -82,6 +82,11 @@ export default {
purpose: '',
applicant: ''
},
statusClassMap: {
'待审批': 'status-pending',
'已审批': 'status-approved',
'已拒绝': 'status-rejected'
},
materials: [
{ id: 1, name: '维修耗材' },
{ id: 2, name: '办公用品' },
@ -133,16 +138,6 @@ export default {
showCancel: false
});
},
getStatusClass(status) {
if (status === '待审批') {
return 'status-pending';
} else if (status === '已审批') {
return 'status-approved';
} else if (status === '已拒绝') {
return 'status-rejected';
}
return '';
}
}
};
</script>

@ -0,0 +1,527 @@
<template>
<view class="notice-list-page">
<view class="header">
<view class="back" @click="goBack">
<text class="iconfont icon-xiangzuo"></text>
</view>
<view class="title">通知公告</view>
<view class="right"></view>
</view>
<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="tab-row">
<view class="tab" :class="typeTab === 'all' ? 'active' : ''" @click="switchType('all')"></view>
<view class="tab" :class="typeTab === '1' ? 'active' : ''" @click="switchType('1')"></view>
<view class="tab" :class="typeTab === '2' ? 'active' : ''" @click="switchType('2')"></view>
</view>
<view class="tab-row" style="margin-top: 20rpx;">
<view class="tab" :class="statusTab === 'all' ? 'active' : ''" @click="switchStatus('all')"></view>
<view class="tab" :class="statusTab === '0' ? 'active' : ''" @click="switchStatus('0')">稿</view>
<view class="tab" :class="statusTab === '1' ? 'active' : ''" @click="switchStatus('1')"></view>
</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 class="tag-row">
<text class="tag kind" :class="typeClassMap[item.type] || ''">
{{ typeText(item.type) }}
</text>
<text class="tag status" :class="statusClassMap[item.status] || statusClassMap['__default__']">
{{ statusText(item.status) }}
</text>
<text class="tag sms" :class="smsClassMap[item.smsStatus] || ''">
{{ smsText(item.smsStatus) }}
</text>
</view>
</view>
<text class="time">{{ formatDate(item.noticeTime) }}</text>
</view>
<view class="card-body">
<view class="row">
<text class="label">通知类型</text>
<text class="value">{{ 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.draftDept || '—' }}</text>
</view>
<view class="row" v-if="item.noticeSource">
<text class="label">通知来源</text>
<text class="value">{{ item.noticeSource }}</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 {
data() {
return {
query: {
noticeTitle: '',
},
typeTab: 'all', // all | 1 | 2
statusTab: 'all', // all | 0 | 1
records: [],
page: 1,
limit: 10,
loading: false,
finished: false,
typeClassMap: {
'1': 'tag-internal',
1: 'tag-internal',
'2': 'tag-external',
2: 'tag-external',
'__default__': ''
},
statusClassMap: {
'0': 'status-draft',
0: 'status-draft',
'1': 'status-submit',
1: 'status-submit',
'__default__': ''
},
smsClassMap: {
'0': 'sms-unsent',
0: 'sms-unsent',
'1': 'sms-sent',
1: 'sms-sent',
'__default__': ''
}
};
},
onLoad() {
uni.setNavigationBarTitle({ title: '通知公告' });
this.refreshList();
},
onPullDownRefresh() {
this.refreshList();
},
onShow() {
if (!this.records.length) this.refreshList();
},
methods: {
goBack() {
uni.navigateBack();
},
switchType(type) {
this.typeTab = type;
this.refreshList();
},
switchStatus(status) {
this.statusTab = status;
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.typeTab !== 'all') params.type = this.typeTab;
if (this.statusTab !== 'all') params.status = this.statusTab;
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(/\s+/g, ' ').trim();
if (!s) return '';
return s.length > 80 ? s.slice(0, 80) + '...' : s;
},
formatDate(val) {
if (!val) return '';
if (typeof val === 'string') {
return val.replace('T', ' ').slice(0, 19);
}
if (val instanceof Date) {
const pad = (n) => (n < 10 ? '0' + n : '' + n);
return `${val.getFullYear()}-${pad(val.getMonth() + 1)}-${pad(val.getDate())} ${pad(
val.getHours()
)}:${pad(val.getMinutes())}`;
}
return String(val);
},
typeText(type) {
if (type === 1 || type === '1') return '内部';
if (type === 2 || type === '2') return '外部';
return type || '—';
},
statusText(status) {
if (status === 0 || status === '0') return '草稿';
if (status === 1 || status === '1') return '提交';
return status ?? '—';
},
smsText(smsStatus) {
if (smsStatus === 0 || smsStatus === '0') return '未发';
if (smsStatus === 1 || smsStatus === '1') return '已发';
return smsStatus ?? '—';
},
},
};
</script>
<style lang="scss">
.notice-list-page {
.header {
display: flex;
align-items: center;
justify-content: space-between;
height: 88rpx;
padding: 0 30rpx;
background-color: #409eff;
color: #fff;
.back {
width: 60rpx;
height: 100%;
display: flex;
align-items: center;
.iconfont {
font-size: 32rpx;
color: #fff;
}
}
.title {
font-size: 32rpx;
font-weight: 600;
}
.right {
width: 60rpx;
}
}
.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;
}
}
.tab-row {
display: flex;
background-color: #fff;
border-radius: 12rpx;
box-shadow: 0 2rpx 10rpx rgba(0, 0, 0, 0.06);
.tab {
flex: 1;
height: 72rpx;
display: flex;
align-items: center;
justify-content: center;
font-size: 26rpx;
color: #666;
position: relative;
&.active {
color: #409eff;
font-weight: 600;
&::after {
content: '';
position: absolute;
bottom: 0;
left: 50%;
transform: translateX(-50%);
width: 60rpx;
height: 4rpx;
background-color: #409eff;
border-radius: 2rpx;
}
}
}
}
}
.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;
}
}
.tag-row {
display: flex;
flex-wrap: wrap;
gap: 10rpx;
}
.tag {
padding: 6rpx 16rpx;
border-radius: 20rpx;
font-size: 20rpx;
line-height: 1;
}
.tag-internal {
background-color: #e6f7ff;
color: #409eff;
}
.tag-external {
background-color: #fff7e6;
color: #fa8c16;
}
.status-draft {
background-color: #f5f5f5;
color: #666;
}
.status-submit {
background-color: #f6ffed;
color: #52c41a;
}
.sms-unsent {
background-color: #f5f5f5;
color: #999;
}
.sms-sent {
background-color: #e6fffb;
color: #13c2c2;
}
.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>

@ -0,0 +1,347 @@
<template>
<view class="notice-detail-page">
<view class="header">
<view class="back" @click="goBack">
<text class="iconfont icon-xiangzuo"></text>
</view>
<view class="title">通知详情</view>
<view class="right"></view>
</view>
<view class="content" v-if="!loading">
<view class="detail-card" v-if="notice && (notice.noticeTitle || notice.noticeId)">
<view class="title-section">
<text class="detail-title">{{ notice.noticeTitle || '未命名通知' }}</text>
<view class="tag-row">
<text class="tag kind" :class="typeClassMap[notice.type] || ''">{{ typeText(notice.type) }}</text>
<text class="tag status" :class="statusClassMap[notice.status] || statusClassMap['__default__']">{{ statusText(notice.status) }}</text>
<text class="tag sms" :class="smsClassMap[notice.smsStatus] || ''">{{ smsText(notice.smsStatus) }}</text>
</view>
</view>
<view class="info-list">
<view class="row" v-if="notice.noticeType">
<text class="label">通知类型</text>
<text class="value">{{ notice.noticeType }}</text>
</view>
<view class="row" v-if="notice.noticeSource">
<text class="label">通知来源</text>
<text class="value">{{ notice.noticeSource }}</text>
</view>
<view class="row" v-if="notice.draftDept">
<text class="label">拟稿单位</text>
<text class="value">{{ notice.draftDept }}</text>
</view>
<view class="row" v-if="notice.noticeScope">
<text class="label">通知范围</text>
<text class="value">{{ notice.noticeScope }}</text>
</view>
<view class="row" v-if="notice.bizType || notice.bizId">
<text class="label">业务信息</text>
<text class="value">{{ bizText(notice.bizType, notice.bizId) }}</text>
</view>
<view class="row" v-if="notice.noticeTime">
<text class="label">通知时间</text>
<text class="value">{{ formatDate(notice.noticeTime) }}</text>
</view>
<view class="row" v-if="notice.remark">
<text class="label">备注</text>
<text class="value remark-value">{{ notice.remark }}</text>
</view>
</view>
<view class="content-box">
<jyf-parser
v-if="notice.noticeContent"
:html="notice.noticeContent"
ref="article"
:tag-style="tagStyle"
></jyf-parser>
<view class="empty" v-else></view>
</view>
</view>
<view class="empty-page" v-else>
<text class="iconfont icon-wushuju"></text>
<text class="text">暂无详情数据</text>
</view>
</view>
<view class="loading" v-else>
<text>加载中...</text>
</view>
</view>
</template>
<script>
import jyfParser from '@/components/jyf-parser/jyf-parser';
import { pubnoticeDetailApi } from '@/api/pubnotice.js';
export default {
components: {
'jyf-parser': jyfParser,
},
data() {
return {
loading: false,
noticeId: '',
notice: {},
tagStyle: {
img: 'max-width:100%;height:auto;display:block;',
table: 'width:100%;border-collapse:collapse;',
video: 'max-width:100%;',
},
typeClassMap: {
'1': 'tag-internal',
1: 'tag-internal',
'2': 'tag-external',
2: 'tag-external',
'__default__': ''
},
statusClassMap: {
'0': 'status-draft',
0: 'status-draft',
'1': 'status-submit',
1: 'status-submit',
'__default__': ''
},
smsClassMap: {
'0': 'sms-unsent',
0: 'sms-unsent',
'1': 'sms-sent',
1: 'sms-sent',
'__default__': ''
}
};
},
onLoad(options) {
uni.setNavigationBarTitle({ title: '通知详情' });
this.noticeId = options.noticeId || options.id || '';
if (this.noticeId) this.fetchDetail();
},
methods: {
goBack() {
uni.navigateBack();
},
async fetchDetail() {
this.loading = true;
try {
const res = await pubnoticeDetailApi(this.noticeId);
this.notice = res?.data || {};
} catch (e) {
uni.showToast({
title: typeof e === 'string' ? e : '获取通知详情失败',
icon: 'none',
});
} finally {
this.loading = false;
}
},
bizText(bizType, bizId) {
const t = bizType ? String(bizType) : '';
const id = bizId ? String(bizId) : '';
if (!t && !id) return '—';
if (t && id) return `${t} #${id}`;
return t || id;
},
formatDate(val) {
if (!val) return '';
if (typeof val === 'string') {
return val.replace('T', ' ').slice(0, 19);
}
if (val instanceof Date) {
const pad = (n) => (n < 10 ? '0' + n : '' + n);
return `${val.getFullYear()}-${pad(val.getMonth() + 1)}-${pad(val.getDate())} ${pad(
val.getHours()
)}:${pad(val.getMinutes())}`;
}
return String(val);
},
typeText(type) {
if (type === 1 || type === '1') return '内部';
if (type === 2 || type === '2') return '外部';
return type || '—';
},
statusText(status) {
if (status === 0 || status === '0') return '草稿';
if (status === 1 || status === '1') return '提交';
return status ?? '—';
},
smsText(smsStatus) {
if (smsStatus === 0 || smsStatus === '0') return '未发';
if (smsStatus === 1 || smsStatus === '1') return '已发';
return smsStatus ?? '—';
},
},
};
</script>
<style lang="scss">
.notice-detail-page {
.header {
display: flex;
align-items: center;
justify-content: space-between;
height: 88rpx;
padding: 0 30rpx;
background-color: #409eff;
color: #fff;
.back {
width: 60rpx;
height: 100%;
display: flex;
align-items: center;
.iconfont {
font-size: 32rpx;
color: #fff;
}
}
.title {
font-size: 32rpx;
font-weight: 600;
}
.right {
width: 60rpx;
}
}
.content {
padding: 30rpx;
}
.detail-card {
background-color: #fff;
border-radius: 12rpx;
box-shadow: 0 2rpx 10rpx rgba(0, 0, 0, 0.06);
padding: 24rpx 26rpx;
}
.title-section {
margin-bottom: 20rpx;
.detail-title {
font-size: 30rpx;
font-weight: 800;
color: #333;
display: block;
line-height: 1.4;
word-break: break-all;
}
.tag-row {
display: flex;
flex-wrap: wrap;
gap: 10rpx;
margin-top: 14rpx;
}
}
.tag {
padding: 6rpx 16rpx;
border-radius: 20rpx;
font-size: 20rpx;
line-height: 1;
}
.tag-internal {
background-color: #e6f7ff;
color: #409eff;
}
.tag-external {
background-color: #fff7e6;
color: #fa8c16;
}
.status-draft {
background-color: #f5f5f5;
color: #666;
}
.status-submit {
background-color: #f6ffed;
color: #52c41a;
}
.sms-unsent {
background-color: #f5f5f5;
color: #999;
}
.sms-sent {
background-color: #e6fffb;
color: #13c2c2;
}
.info-list {
.row {
display: flex;
margin-bottom: 14rpx;
.label {
width: 170rpx;
font-size: 24rpx;
color: #666;
flex-shrink: 0;
}
.value {
flex: 1;
font-size: 24rpx;
color: #333;
min-width: 0;
word-break: break-word;
}
}
.remark-value {
line-height: 1.5;
}
}
.content-box {
margin-top: 10rpx;
padding-top: 20rpx;
border-top: 1rpx solid #f5f5f5;
}
.empty-page {
margin-top: 120rpx;
display: flex;
flex-direction: column;
align-items: center;
color: #999;
.iconfont {
font-size: 80rpx;
margin-bottom: 16rpx;
}
.text {
font-size: 24rpx;
}
}
.empty {
font-size: 26rpx;
color: #999;
line-height: 2;
padding: 30rpx 0;
}
.loading {
height: 100%;
display: flex;
align-items: center;
justify-content: center;
color: #999;
font-size: 26rpx;
}
}
</style>

@ -72,7 +72,7 @@
<view class="order-item" v-for="(item, index) in orders" :key="index">
<view class="item-header">
<text class="order-no">订单号{{ item.orderNo }}</text>
<text class="order-status" :class="getStatusClass(item.status)">{{ item.status }}</text>
<text class="order-status" :class="statusClassMap[item.status]">{{ item.status }}</text>
</view>
<view class="item-body">
<view class="info-item">
@ -150,6 +150,11 @@ export default {
description: '',
images: []
},
statusClassMap: {
'已下单': 'status-ordered',
'配送中': 'status-delivering',
'已完成': 'status-completed'
},
orders: [
{ id: 1, orderNo: 'CG20260307001', name: '有机蔬菜', quantity: 100, status: '已下单', orderTime: '2026-03-07', expectedTime: '2026-03-09' },
{ id: 2, orderNo: 'CG20260307002', name: '新鲜水果', quantity: 50, status: '配送中', orderTime: '2026-03-06', expectedTime: '2026-03-08' },
@ -262,18 +267,6 @@ export default {
};
}, 1500);
},
getStatusClass(status) {
switch (status) {
case '已下单':
return 'status-ordered';
case '配送中':
return 'status-delivering';
case '已完成':
return 'status-completed';
default:
return '';
}
}
}
};
</script>
@ -395,7 +388,7 @@ export default {
}
.upload-box {
.display: flex;
display: flex;
flex-wrap: wrap;
.upload-btn {
@ -562,7 +555,7 @@ export default {
}
.item-footer {
.display: flex;
display: flex;
justify-content: flex-end;
.detail-btn {

@ -59,7 +59,7 @@
</picker>
</view>
<view class="form-item">
<!-- <view class="form-item">
<text class="label">联系方式</text>
<input
class="input"
@ -68,18 +68,18 @@
placeholder="请输入手机号"
maxlength="20"
/>
</view>
</view> -->
<view class="form-item">
<text class="label">报修内容</text>
<textarea
class="textarea"
v-model="form.handleContent"
v-model="form.faultDesc"
placeholder="请描述故障位置、现象等,便于快速处理"
maxlength="300"
:auto-height="true"
/>
<text class="count">{{ form.handleContent.length }}/300</text>
<text class="count">{{ form.faultDesc.length }}/300</text>
</view>
<view class="form-item">
@ -130,13 +130,13 @@
</text>
<text
class="status"
:class="statusClass(item.status)"
:class="statusClassMap[item.status] || statusClassMap['__default__']"
>
{{ statusText(item.status) }}
</text>
</view>
<text class="time">
{{ item.assignTime || item.createTime || '' }}
{{ item.reportTime || item.createTime || '' }}
</text>
</view>
@ -147,41 +147,28 @@
{{ getFaultTypeLabel(item.faultType) }}
</text>
</view>
<view class="row" v-if="item.assigneContext">
<view class="row" v-if="item.faultDesc">
<text class="label">报修内容</text>
<text class="value">
{{ item.assigneContext }}
</text>
</view>
<view class="row" v-if="item.executorName">
<text class="label">执行人</text>
<text class="value">
{{ item.executorName }}
{{ item.faultDesc }}
</text>
</view>
<view class="row" v-if="item.expectedCompleteTime">
<text class="label">预计完成</text>
<text class="value">
{{ item.expectedCompleteTime }}
</text>
</view>
<view class="row" v-if="item.completeTime">
<text class="label">实际完成</text>
<text class="value">
{{ item.completeTime }}
</text>
</view>
<view class="row" v-if="item.handleContent">
<text class="label">处理情况</text>
<text class="value">
{{ item.handleContent }}
</text>
</view>
<view class="row" v-if="item.costAmount">
<text class="label">维修费用</text>
<text class="value">
{{ item.costAmount }}
</text>
<view
class="image-grid"
v-if="item.files && item.files.length"
>
<view
class="image-item"
v-for="(file, imgIndex) in item.files"
:key="imgIndex"
@click="previewRecordImages(item.files, imgIndex)"
>
<image
:src="HTTP_ADMIN_URL + '/' + (file.url || file.attDir)"
mode="aspectFill"
></image>
</view>
</view>
</view>
</view>
@ -221,9 +208,9 @@
<text class="name">
{{ (item.unitNo ? item.unitNo + '单元' : '') + (item.floorNo ? ' ' + item.floorNo + '层' : '') + (item.houseNo ? ' ' + item.houseNo + '室' : '') || item.houseName || '房屋' }}
</text>
<text class="desc">
<!-- <text class="desc">
{{ item.buildingId ? '楼栋 ' + item.buildingId : '' }}
</text>
</text> -->
</view>
<view
class="empty"
@ -257,11 +244,10 @@ export default {
activeTab: 'form',
form: {
phone: '',
handleContent: '',
expectedTime: '',
faultDesc: '',
faultType: ''
},
faultTypeIndex: 0,
faultTypeIndex: -1,
images: [],
selectedHouseId: null,
selectedHouseName: '',
@ -272,11 +258,25 @@ export default {
finished: false,
housePopupVisible: false,
houseList: [],
houseLoading: false
houseLoading: false,
statusClassMap: {
'0': 'status-pending',
0: 'status-pending',
'1': 'status-doing',
1: 'status-doing',
'2': 'status-done',
2: 'status-done',
'99': 'status-done',
99: 'status-done',
'__default__': 'status-pending'
}
};
},
computed: {
selectedFaultType() {
if (this.faultTypeIndex == -1) {
return '';
}
const types = this.dict.get('fault_type');
if (types && types.length > this.faultTypeIndex) {
return types[this.faultTypeIndex].dictLabel;
@ -333,7 +333,7 @@ export default {
const unitPart = item.unitNo ? item.unitNo + '单元' : '';
const floorPart = item.floorNo ? item.floorNo + '层' : '';
const housePart = item.houseNo ? item.houseNo + '室' : '';
this.selectedHouseName = [unitPart, floorPart, housePart].filter(Boolean).join(' ');
this.selectedHouseName = [unitPart, floorPart, housePart].filter(Boolean).join('');
this.housePopupVisible = false;
},
chooseImage() {
@ -410,7 +410,7 @@ export default {
});
return;
}
if (!this.form.handleContent.trim()) {
if (!this.form.faultDesc.trim()) {
uni.showToast({
title: '请填写报修内容',
icon: 'none'
@ -421,12 +421,7 @@ export default {
//
const imagefiles = this.images.map(file => {
let attDir = file.url;
// /file/public/
if (attDir.startsWith('file/public/')) {
attDir = attDir.replace('file/public/', '')
} else if (attDir.startsWith('file/')) {
attDir = attDir.replace('file/', '')
}
return {
attId: file.id + '',
name: file.fileName,
@ -441,10 +436,10 @@ export default {
});
const payload = {
phone: this.form.phone.trim(),
assigneContext: this.form.handleContent.trim(),
remark: this.form.expectedTime.trim(),
// phone: this.form.phone.trim(),
faultDesc: this.form.faultDesc.trim(),
status: this.form.status || '0',
houseId: this.selectedHouseId,
houseName: this.selectedHouseName,
files: imagefiles,
faultType: this.form.faultType
@ -460,10 +455,9 @@ export default {
});
this.form.phone = '';
this.form.handleContent = '';
this.form.expectedTime = '';
this.form.faultDesc = '';
this.form.faultType = '';
this.faultTypeIndex = 0;
this.faultTypeIndex = -1;
this.images = [];
this.page = 1;
@ -517,6 +511,17 @@ export default {
this.loading = false;
}
},
previewRecordImages(files, index) {
if (!files || !files.length) return;
const urls = files.map(file => {
const path = file.url || file.attDir || file.filePath;
return this.HTTP_ADMIN_URL + '/' + path;
});
uni.previewImage({
current: index,
urls
});
},
statusText(status) {
if (status === 0 || status === '0') return '待处理';
if (status === 1 || status === '1') return '处理中';
@ -524,13 +529,6 @@ export default {
if (status === 99 || status === '99') return '已办结';
return status;
},
statusClass(status) {
const text = this.statusText(status);
if (text === '已处理' || text === '已办结') return 'status-done';
if (text === '处理中') return 'status-doing';
if (text === '待处理') return 'status-pending';
return '';
},
getFaultTypeLabel(value) {
const types = this.dict.get('fault_type');
if (types) {
@ -799,6 +797,7 @@ export default {
font-weight: 600;
color: #333;
margin-right: 16rpx;
max-width: 300rpx;
}
.status {
@ -846,6 +845,28 @@ export default {
color: #333;
}
}
.image-grid {
margin-top: 12rpx;
margin-left: 0;
display: flex;
flex-wrap: wrap;
gap: 12rpx;
.image-item {
width: calc((100% - 24rpx) / 3);
aspect-ratio: 1 / 1;
border-radius: 8rpx;
overflow: hidden;
background-color: #f5f5f5;
image {
width: 100%;
height: 100%;
display: block;
}
}
}
}
}

@ -1,7 +1,7 @@
{
"appid": "wxd29e5ba4ccbcca2f",
"compileType": "miniprogram",
"libVersion": "3.1.4",
"libVersion": "latest",
"packOptions": {
"ignore": [],
"include": []

@ -4,5 +4,5 @@
"setting": {
"compileHotReLoad": true
},
"libVersion": "3.1.4"
"libVersion": "latest"
}
Loading…
Cancel
Save