From e6080ac8ad345f8cd7378c7cfa5f0f70f80e3926 Mon Sep 17 00:00:00 2001
From: yyb <995253665@qq.com>
Date: 星期一, 18 五月 2026 16:59:58 +0800
Subject: [PATCH] 增加保养部件和附件
---
src/pages/equipmentManagement/upkeep/index.vue | 6
src/pages/equipmentManagement/upkeep/add.vue | 304 ++++++++++++++++++++++++++++++++++++++++++++++++++
2 files changed, 305 insertions(+), 5 deletions(-)
diff --git a/src/pages/equipmentManagement/upkeep/add.vue b/src/pages/equipmentManagement/upkeep/add.vue
index bc8cae0..ce24cf3 100644
--- a/src/pages/equipmentManagement/upkeep/add.vue
+++ b/src/pages/equipmentManagement/upkeep/add.vue
@@ -54,12 +54,65 @@
</template>
</u-form-item>
- <u-form-item label="淇濆吇椤圭洰" prop="maintenanceItems" border-bottom>
+ <u-form-item label="淇濆吇閮ㄤ綅" prop="maintenanceLocation" required border-bottom>
+ <u-input
+ v-model="form.maintenanceLocation"
+ placeholder="璇疯緭鍏ヤ繚鍏婚儴浣�"
+ clearable
+ />
+ </u-form-item>
+
+ <u-form-item label="淇濆吇椤圭洰" prop="maintenanceItems" required border-bottom>
<u-input
v-model="form.maintenanceItems"
placeholder="璇疯緭鍏ヤ繚鍏婚」鐩�"
clearable
/>
+ </u-form-item>
+
+ <u-form-item label="闄勪欢" border-bottom>
+ <view class="attachment-upload">
+ <view class="upload-buttons">
+ <u-button
+ type="primary"
+ @click="chooseAttachment('camera')"
+ :loading="uploading"
+ :disabled="uploading"
+ :customStyle="{ marginRight: '10px', flex: 1 }"
+ >
+ <u-icon name="camera" size="18" color="#fff" style="margin-right: 5px;"></u-icon>
+ {{ uploading ? '涓婁紶涓�...' : '鎷嶇収' }}
+ </u-button>
+ <u-button
+ type="success"
+ @click="chooseAttachment('album')"
+ :loading="uploading"
+ :disabled="uploading"
+ :customStyle="{ flex: 1 }"
+ >
+ <u-icon name="photo" size="18" color="#fff" style="margin-right: 5px;"></u-icon>
+ {{ uploading ? '涓婁紶涓�...' : '鐩稿唽' }}
+ </u-button>
+ </view>
+ <view v-if="attachmentList.length" class="attachment-list">
+ <view
+ v-for="(file, index) in attachmentList"
+ :key="file.id || index"
+ class="attachment-item"
+ >
+ <image
+ :src="getFileAccessUrl(file)"
+ mode="aspectFill"
+ class="attachment-preview"
+ @click="previewAttachment(index)"
+ />
+ <view class="attachment-delete" @click="removeAttachment(file, index)">
+ <u-icon name="close" size="12" color="#fff" />
+ </view>
+ </view>
+ </view>
+ <view v-else class="attachment-empty">鏆傛棤闄勪欢锛屽彲鎷嶇収鎴栦粠鐩稿唽閫夋嫨</view>
+ </view>
</u-form-item>
<!-- 鎻愪氦鎸夐挳 -->
@@ -100,8 +153,17 @@
import { ref, computed, onMounted, onUnmounted } from 'vue';
import { onShow, onUnload } from '@dcloudio/uni-app';
import PageHeader from '@/components/PageHeader.vue';
+import config from '@/config';
+import { getToken } from '@/utils/auth';
import { getDeviceLedger } from '@/api/equipmentManagement/ledger';
-import { addUpkeep, editUpkeep, getUpkeepById } from '@/api/equipmentManagement/upkeep';
+import {
+ addUpkeep,
+ editUpkeep,
+ getUpkeepById,
+ listMaintenanceTaskFiles,
+ addMaintenanceTaskFile,
+ delMaintenanceTaskFile,
+} from '@/api/equipmentManagement/upkeep';
import { userListNoPageByTenantId } from '@/api/system/user';
import useUserStore from '@/store/modules/user';
import dayjs from "dayjs";
@@ -130,6 +192,8 @@
const formRef = ref(null);
const operationType = ref('add');
const loading = ref(false);
+const uploading = ref(false);
+const attachmentList = ref([]);
const showDevice = ref(false);
const showPerson = ref(false);
const showDate = ref(false);
@@ -165,6 +229,8 @@
deviceLedgerId: [{ required: true, trigger: "change", message: "璇烽�夋嫨璁惧鍚嶇О" }],
maintenancePlanTime: [{ required: true, trigger: "change", message: "璇烽�夋嫨璁″垝淇濆吇鏃ユ湡" }],
maintenancePerson: [{ required: true, trigger: "change", message: "璇烽�夋嫨淇濆吇浜�" }],
+ maintenanceLocation: [{ required: true, trigger: "blur", message: "璇疯緭鍏ヤ繚鍏婚儴浣�" }],
+ maintenanceItems: [{ required: true, trigger: "blur", message: "璇疯緭鍏ヤ繚鍏婚」鐩�" }],
};
// 浣跨敤 ref 澹版槑琛ㄥ崟鏁版嵁
@@ -173,6 +239,7 @@
deviceModel: undefined, // 瑙勬牸鍨嬪彿
maintenancePlanTime: dayjs().format("YYYY-MM-DD"), // 璁″垝淇濆吇鏃ユ湡
maintenancePerson: userStore.nickName || undefined, // 淇濆吇浜�
+ maintenanceLocation: undefined, // 淇濆吇閮ㄤ綅
maintenanceItems: undefined, // 淇濆吇椤圭洰
});
@@ -196,6 +263,169 @@
}
};
+// 闄勪欢鐩稿叧
+const getFileAccessUrl = (file = {}) => {
+ const url = file.url || file.tempFilePath || file.path || '';
+ if (!url) return '';
+ if (String(url).startsWith('http') || String(url).startsWith('blob:') || String(url).startsWith('file:') || String(url).startsWith('wxfile:')) {
+ return url;
+ }
+ const path = String(url).startsWith('/') ? url : `/${url}`;
+ return `${config.fileUrl}${path}`;
+};
+
+const fetchAttachmentList = async (id) => {
+ if (!id) {
+ attachmentList.value = [];
+ return;
+ }
+ try {
+ const { code, data } = await listMaintenanceTaskFiles({
+ current: 1,
+ size: 100,
+ deviceMaintenanceId: id,
+ });
+ if (code === 200) {
+ attachmentList.value = data?.records || [];
+ } else {
+ attachmentList.value = [];
+ }
+ } catch (e) {
+ attachmentList.value = [];
+ }
+};
+
+const chooseAttachment = (sourceType) => {
+ const source = sourceType === 'camera' ? ['camera'] : ['album'];
+ uni.chooseImage({
+ count: 9,
+ sizeType: ['original', 'compressed'],
+ sourceType: source,
+ success: (res) => {
+ const files = res.tempFiles || [];
+ if (!files.length) return;
+ const id = getPageId();
+ if (id) {
+ uploadAttachments(files, id);
+ return;
+ }
+ const tempItems = files.map((file, idx) => {
+ const filePath = file.path || res.tempFilePaths?.[idx];
+ return {
+ id: `temp_${Date.now()}_${idx}`,
+ url: filePath,
+ tempFilePath: filePath,
+ name: file.name || `闄勪欢_${Date.now()}_${idx}.jpg`,
+ isTemp: true,
+ };
+ });
+ attachmentList.value = [...attachmentList.value, ...tempItems];
+ showToast('宸叉坊鍔狅紝淇濆瓨璁″垝鍚庤嚜鍔ㄤ笂浼�');
+ },
+ fail: () => {
+ showToast('閫夋嫨鍥剧墖澶辫触');
+ },
+ });
+};
+
+const uploadAttachments = async (files, maintenanceId) => {
+ const commonId = normalizeId(maintenanceId);
+ if (!commonId) {
+ showToast('鏈幏鍙栧埌淇濆吇璁″垝ID锛屼笂浼犲け璐�');
+ return;
+ }
+ const token = getToken();
+ if (!token) {
+ showToast('鐧诲綍宸插け鏁堬紝璇烽噸鏂扮櫥褰�');
+ return;
+ }
+ uploading.value = true;
+ try {
+ for (const file of files) {
+ const filePath = file.path || file.tempFilePath;
+ if (!filePath) continue;
+ await new Promise((resolve, reject) => {
+ uni.uploadFile({
+ url: `${config.baseUrl}/file/upload`,
+ filePath,
+ name: 'file',
+ header: {
+ Authorization: `Bearer ${token}`,
+ },
+ success: (uploadRes) => {
+ try {
+ const parsed = JSON.parse(uploadRes.data || '{}');
+ if (uploadRes.statusCode === 200 && parsed.code === 200) {
+ const fileName = file.name || filePath.split('/').pop();
+ addMaintenanceTaskFile({
+ name: fileName,
+ deviceMaintenanceId: commonId,
+ url: parsed.data?.tempPath || parsed.data?.url || '',
+ })
+ .then((addRes) => {
+ if (addRes.code === 200) {
+ resolve(addRes);
+ } else {
+ reject(new Error(addRes.msg || '淇濆瓨闄勪欢淇℃伅澶辫触'));
+ }
+ })
+ .catch(reject);
+ } else {
+ reject(new Error(parsed.msg || '涓婁紶澶辫触'));
+ }
+ } catch (err) {
+ reject(new Error('涓婁紶鍝嶅簲瑙f瀽澶辫触'));
+ }
+ },
+ fail: () => reject(new Error('涓婁紶澶辫触')),
+ });
+ });
+ }
+ showToast('涓婁紶鎴愬姛');
+ await fetchAttachmentList(commonId);
+ } catch (e) {
+ showToast(e?.message || '涓婁紶澶辫触');
+ } finally {
+ uploading.value = false;
+ }
+};
+
+const previewAttachment = (index) => {
+ const urls = attachmentList.value
+ .map((item) => getFileAccessUrl(item))
+ .filter(Boolean);
+ if (!urls.length) return;
+ uni.previewImage({
+ urls,
+ current: urls[index] || urls[0],
+ });
+};
+
+const removeAttachment = (file, index) => {
+ if (!file?.id || file?.isTemp) {
+ attachmentList.value.splice(index, 1);
+ return;
+ }
+ uni.showModal({
+ title: '鎻愮ず',
+ content: '纭鍒犻櫎璇ラ檮浠跺悧锛�',
+ success: async (res) => {
+ if (!res.confirm) return;
+ try {
+ const { code } = await delMaintenanceTaskFile(file.id);
+ if (code === 200) {
+ attachmentList.value.splice(index, 1);
+ showToast('鍒犻櫎鎴愬姛');
+ } else {
+ showToast('鍒犻櫎澶辫触');
+ }
+ } catch (e) {
+ showToast('鍒犻櫎澶辫触');
+ }
+ },
+ });
+};
+
// 鍔犺浇琛ㄥ崟鏁版嵁锛堢紪杈戞ā寮忥級
const loadForm = async (id) => {
if (id) {
@@ -207,12 +437,14 @@
form.value.deviceModel = data.deviceModel;
form.value.maintenancePlanTime = dayjs(data.maintenancePlanTime).format("YYYY-MM-DD");
form.value.maintenancePerson = data.maintenancePerson;
- form.value.maintenanceItems = data.maintenanceItems || data.maintenanceLocation;
+ form.value.maintenanceLocation = data.maintenanceLocation;
+ form.value.maintenanceItems = data.maintenanceItems;
// 璁剧疆璁惧鍚嶇О鏄剧ず
const device = deviceOptions.value.find(item => item.id === data.deviceLedgerId);
if (device) {
form.value.deviceNameText = device.deviceName;
}
+ await fetchAttachmentList(id);
}
} catch (e) {
showToast('鑾峰彇璇︽儏澶辫触');
@@ -220,6 +452,7 @@
} else {
// 鏂板妯″紡
operationType.value = 'add';
+ attachmentList.value = [];
}
};
@@ -363,11 +596,27 @@
submitData.maintenancePlanTime = submitData.maintenancePlanTime + ' 00:00:00';
}
- const { code } = id
+ const result = id
? await editUpkeep({ id: id, ...submitData })
: await addUpkeep(submitData);
+ const { code, data } = result || {};
if (code == 200) {
+ if (!id) {
+ const newId = data?.id || data?.maintenanceId || data;
+ if (newId) {
+ const tempFiles = attachmentList.value
+ .filter((item) => item?.isTemp && (item.tempFilePath || item.url))
+ .map((item) => ({
+ path: item.tempFilePath || item.url,
+ tempFilePath: item.tempFilePath || item.url,
+ name: item.name,
+ }));
+ if (tempFiles.length) {
+ await uploadAttachments(tempFiles, newId);
+ }
+ }
+ }
showToast(`${id ? "缂栬緫" : "鏂板"}璁″垝鎴愬姛`);
setTimeout(() => {
uni.removeStorageSync('repairId');
@@ -401,6 +650,7 @@
loadForm(id);
} else {
operationType.value = 'add';
+ attachmentList.value = [];
loadForm();
}
};
@@ -471,4 +721,50 @@
margin-left: 8px;
cursor: pointer;
}
+
+.attachment-upload {
+ width: 100%;
+}
+
+.upload-buttons {
+ display: flex;
+ margin-bottom: 12px;
+}
+
+.attachment-list {
+ display: flex;
+ flex-wrap: wrap;
+ gap: 10px;
+}
+
+.attachment-item {
+ position: relative;
+ width: 80px;
+ height: 80px;
+}
+
+.attachment-preview {
+ width: 80px;
+ height: 80px;
+ border-radius: 6px;
+ background: #f5f5f5;
+}
+
+.attachment-delete {
+ position: absolute;
+ top: -6px;
+ right: -6px;
+ width: 18px;
+ height: 18px;
+ border-radius: 50%;
+ background: rgba(0, 0, 0, 0.65);
+ display: flex;
+ align-items: center;
+ justify-content: center;
+}
+
+.attachment-empty {
+ font-size: 12px;
+ color: #909399;
+}
</style>
\ No newline at end of file
diff --git a/src/pages/equipmentManagement/upkeep/index.vue b/src/pages/equipmentManagement/upkeep/index.vue
index 496f413..242786f 100644
--- a/src/pages/equipmentManagement/upkeep/index.vue
+++ b/src/pages/equipmentManagement/upkeep/index.vue
@@ -59,8 +59,12 @@
<text class="detail-value">{{ item.maintenancePerson || '-' }}</text>
</view>
<view class="detail-row">
+ <text class="detail-label">淇濆吇閮ㄤ綅</text>
+ <text class="detail-value">{{ item.maintenanceLocation || '-' }}</text>
+ </view>
+ <view class="detail-row">
<text class="detail-label">淇濆吇椤圭洰</text>
- <text class="detail-value">{{ item.maintenanceItems || item.maintenanceLocation || '-' }}</text>
+ <text class="detail-value">{{ item.maintenanceItems || '-' }}</text>
</view>
<view class="detail-row">
<text class="detail-label">褰曞叆浜�</text>
--
Gitblit v1.9.3