From 9581c0ae9f0d9a2e92744f3dca78960780b9a2df Mon Sep 17 00:00:00 2001
From: yyb <995253665@qq.com>
Date: 星期一, 18 五月 2026 17:42:21 +0800
Subject: [PATCH] 附件
---
src/pages/equipmentManagement/upkeep/add.vue | 459 +++++++++++++++++++++++++++++++++++++++++++++++++++++----
1 files changed, 425 insertions(+), 34 deletions(-)
diff --git a/src/pages/equipmentManagement/upkeep/add.vue b/src/pages/equipmentManagement/upkeep/add.vue
index c0e7e34..abddae6 100644
--- a/src/pages/equipmentManagement/upkeep/add.vue
+++ b/src/pages/equipmentManagement/upkeep/add.vue
@@ -40,6 +40,80 @@
<u-icon name="arrow-right" @click="showDatePicker" />
</template>
</u-form-item>
+
+ <u-form-item label="淇濆吇浜�" prop="maintenancePerson" required border-bottom>
+ <u-input
+ v-model="form.maintenancePerson"
+ placeholder="璇烽�夋嫨淇濆吇浜�"
+ readonly
+ @click="showPersonPicker"
+ clearable
+ />
+ <template #right>
+ <u-icon name="arrow-right" @click="showPersonPicker" />
+ </template>
+ </u-form-item>
+
+ <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>
<!-- 鎻愪氦鎸夐挳 -->
<view class="footer-btns">
@@ -56,6 +130,14 @@
@select="onDeviceConfirm"
@close="showDevice = false"
/>
+ <!-- 淇濆吇浜洪�夋嫨鍣� -->
+ <up-action-sheet
+ :show="showPerson"
+ :actions="personActions"
+ title="閫夋嫨淇濆吇浜�"
+ @select="onPersonConfirm"
+ @close="showPerson = false"
+ />
<up-datetime-picker
:show="showDate"
v-model="pickerDateValue"
@@ -69,12 +151,24 @@
<script setup>
import { ref, computed, onMounted, onUnmounted } from 'vue';
-import { onShow } from '@dcloudio/uni-app';
+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,
+ delMaintenanceTaskFile,
+} from '@/api/equipmentManagement/upkeep';
+import { userListNoPageByTenantId } from '@/api/system/user';
+import useUserStore from '@/store/modules/user';
import dayjs from "dayjs";
import { formatDateToYMD } from '@/utils/ruoyi';
+
+const userStore = useUserStore();
defineOptions({
name: "璁惧淇濆吇璁″垝琛ㄥ崟",
@@ -86,11 +180,21 @@
})
}
+const normalizeId = (raw) => {
+ if (raw === null || raw === undefined) return undefined;
+ const val = String(raw).trim();
+ if (!val || val === 'undefined' || val === 'null') return undefined;
+ return val;
+};
+
// 琛ㄥ崟寮曠敤
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);
const pickerDateValue = ref(Date.now());
const currentDate = ref([new Date().getFullYear(), new Date().getMonth() + 1, new Date().getDate()]);
@@ -98,12 +202,20 @@
// 璁惧閫夐」
const deviceOptions = ref([]);
const deviceNameText = ref('');
+// 淇濆吇浜洪�夐」
+const userOptions = ref([]);
// 杞崲涓� action-sheet 闇�瑕佺殑鏍煎紡
const deviceActions = computed(() => {
return deviceOptions.value.map(item => ({
text: item.deviceName,
value: item.id,
data: item
+ }));
+});
+const personActions = computed(() => {
+ return userOptions.value.map(item => ({
+ name: item.nickName,
+ value: item.userId,
}));
});
@@ -115,6 +227,9 @@
const formRules = {
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 澹版槑琛ㄥ崟鏁版嵁
@@ -122,6 +237,10 @@
deviceLedgerId: undefined, // 璁惧ID
deviceModel: undefined, // 瑙勬牸鍨嬪彿
maintenancePlanTime: dayjs().format("YYYY-MM-DD"), // 璁″垝淇濆吇鏃ユ湡
+ maintenancePerson: userStore.nickName || undefined, // 淇濆吇浜�
+ maintenanceLocation: undefined, // 淇濆吇閮ㄤ綅
+ maintenanceItems: undefined, // 淇濆吇鍐呭
+ tempFileIds: [], // 鏈涓婁紶闄勪欢鐨勪复鏃舵枃浠禝D锛屼繚瀛樻椂鎻愪氦
});
// 鍔犺浇璁惧鍒楄〃
@@ -134,6 +253,204 @@
}
};
+// 鍔犺浇淇濆吇浜哄垪琛�
+const loadUserOptions = async () => {
+ try {
+ const { data } = await userListNoPageByTenantId();
+ userOptions.value = data || [];
+ } catch (e) {
+ showToast('鑾峰彇淇濆吇浜哄垪琛ㄥけ璐�');
+ }
+};
+
+// 闄勪欢鐩稿叧
+const extractTempFileId = (uploadedFile) => {
+ if (!uploadedFile) return undefined;
+ const data = Array.isArray(uploadedFile) ? uploadedFile[0] : uploadedFile;
+ return data?.tempId ?? data?.tempFileId ?? data?.id;
+};
+
+const syncTempFileIds = () => {
+ form.value.tempFileIds = attachmentList.value
+ .filter((item) => item.isTempFile)
+ .map((item) => item.tempId ?? item.tempFileId)
+ .filter((v) => v !== undefined && v !== null && v !== '');
+};
+
+const getFileAccessUrl = (file = {}) => {
+ if (file?._localPreviewUrl) return file._localPreviewUrl;
+ const url = file.url || file.tempPath || 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 = [];
+ form.value.tempFileIds = [];
+ return;
+ }
+ try {
+ const { code, data } = await listMaintenanceTaskFiles({
+ current: 1,
+ size: 100,
+ deviceMaintenanceId: id,
+ });
+ if (code === 200) {
+ const records = data?.records || [];
+ attachmentList.value = records.map((file) => ({
+ ...file,
+ isTempFile: false,
+ }));
+ } else {
+ attachmentList.value = [];
+ }
+ } catch (e) {
+ attachmentList.value = [];
+ }
+ syncTempFileIds();
+};
+
+const chooseAttachment = (sourceType) => {
+ const source = sourceType === 'camera' ? ['camera'] : ['album'];
+ const remaining = 9 - attachmentList.value.length;
+ if (remaining <= 0) {
+ showToast('鏈�澶氫笂浼�9寮犻檮浠�');
+ return;
+ }
+ uni.chooseImage({
+ count: Math.min(remaining, 9),
+ sizeType: ['original', 'compressed'],
+ sourceType: source,
+ success: (res) => {
+ const files = res.tempFiles || [];
+ if (!files.length) return;
+ uploadAttachments(files, res.tempFilePaths);
+ },
+ fail: () => {
+ showToast('閫夋嫨鍥剧墖澶辫触');
+ },
+ });
+};
+
+const handleUploadSuccess = (response, file) => {
+ let uploadedFile = response?.data;
+ if (Array.isArray(uploadedFile)) {
+ uploadedFile = uploadedFile[0];
+ }
+ const tempId = extractTempFileId(uploadedFile);
+ if (!tempId) {
+ showToast('鏈幏鍙栧埌鏂囦欢ID');
+ return;
+ }
+ attachmentList.value.push({
+ tempId,
+ tempFileId: tempId,
+ isTempFile: true,
+ url: uploadedFile?.tempPath || uploadedFile?.url || uploadedFile?.downloadUrl || file.tempFilePath || file.path,
+ tempPath: uploadedFile?.tempPath || '',
+ name: uploadedFile?.originalName || uploadedFile?.originalFilename || file.name,
+ _localPreviewUrl: file.tempFilePath || file.path || '',
+ });
+ syncTempFileIds();
+};
+
+const uploadAttachments = async (files, tempFilePaths = []) => {
+ const token = getToken();
+ if (!token) {
+ showToast('鐧诲綍宸插け鏁堬紝璇烽噸鏂扮櫥褰�');
+ return;
+ }
+ uploading.value = true;
+ try {
+ for (let i = 0; i < files.length; i++) {
+ const file = files[i];
+ const filePath = file.path || file.tempFilePath || tempFilePaths[i];
+ 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) {
+ handleUploadSuccess(parsed, {
+ ...file,
+ tempFilePath: filePath,
+ path: filePath,
+ name: file.name || `闄勪欢_${Date.now()}_${i}.jpg`,
+ });
+ resolve(parsed);
+ } else {
+ reject(new Error(parsed.msg || '涓婁紶澶辫触'));
+ }
+ } catch (err) {
+ reject(new Error('涓婁紶鍝嶅簲瑙f瀽澶辫触'));
+ }
+ },
+ fail: () => reject(new Error('涓婁紶澶辫触')),
+ });
+ });
+ }
+ showToast('涓婁紶鎴愬姛');
+ } 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?.isTempFile) {
+ attachmentList.value.splice(index, 1);
+ syncTempFileIds();
+ return;
+ }
+ if (!file?.id) {
+ attachmentList.value.splice(index, 1);
+ syncTempFileIds();
+ 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) {
@@ -144,11 +461,15 @@
form.value.deviceLedgerId = data.deviceLedgerId;
form.value.deviceModel = data.deviceModel;
form.value.maintenancePlanTime = dayjs(data.maintenancePlanTime).format("YYYY-MM-DD");
+ form.value.maintenancePerson = data.maintenancePerson;
+ 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('鑾峰彇璇︽儏澶辫触');
@@ -156,6 +477,8 @@
} else {
// 鏂板妯″紡
operationType.value = 'add';
+ attachmentList.value = [];
+ form.value.tempFileIds = [];
}
};
@@ -187,24 +510,25 @@
}
isScanning.value = true;
- showToast('鎵爜鎴愬姛锛�3绉掑悗鑷姩濉厖璁惧淇℃伅');
+ showToast('鎵爜鎴愬姛');
// 3绉掑悗澶勭悊鎵爜缁撴灉
scanTimer.value = setTimeout(() => {
processScanResult(scanResult);
isScanning.value = false;
- }, 3000);
+ }, 1000);
};
-
+function getDeviceIdByRegExp(url) {
+ // 鍖归厤deviceId=鍚庨潰鐨勬暟瀛�
+ const reg = /deviceId=(\d+)/;
+ const match = url.match(reg);
+ // 濡傛灉鍖归厤鍒扮粨鏋滐紝杩斿洖鏁板瓧绫诲瀷锛屽惁鍒欒繑鍥瀗ull
+ return match ? Number(match[1]) : null;
+}
// 澶勭悊鎵爜缁撴灉骞跺尮閰嶈澶�
const processScanResult = (scanResult) => {
- // 鍦ㄨ澶囧垪琛ㄤ腑鏌ユ壘鍖归厤鐨勮澶�
- // 鍋囪浜岀淮鐮佸唴瀹规槸璁惧鍚嶇О鎴栬澶囩紪鍙�
- const matchedDevice = deviceOptions.value.find(device =>
- device.deviceName === scanResult ||
- device.deviceCode === scanResult ||
- device.id.toString() === scanResult
- );
+ const deviceId = getDeviceIdByRegExp(scanResult);
+ const matchedDevice = deviceOptions.value.find(item => item.id == deviceId);
if (matchedDevice) {
// 鎵惧埌鍖归厤鐨勮澶囷紝鑷姩濉厖
@@ -221,6 +545,22 @@
// 鏄剧ず璁惧閫夋嫨鍣�
const showDevicePicker = () => {
showDevice.value = true;
+};
+
+// 鏄剧ず淇濆吇浜洪�夋嫨鍣�
+const showPersonPicker = () => {
+ if (!userOptions.value.length) {
+ showToast('鏆傛棤鍙�変繚鍏讳汉');
+ return;
+ }
+ showPerson.value = true;
+};
+
+// 纭淇濆吇浜洪�夋嫨
+const onPersonConfirm = (selected) => {
+ const user = userOptions.value.find(item => item.userId === selected.value);
+ form.value.maintenancePerson = user?.nickName || selected.name || '';
+ showPerson.value = false;
};
// 纭璁惧閫夋嫨
@@ -252,8 +592,9 @@
});
onMounted(() => {
- // 椤甸潰鍔犺浇鏃惰幏鍙栬澶囧垪琛ㄥ拰鍙傛暟
+ // 椤甸潰鍔犺浇鏃惰幏鍙栬澶囧垪琛ㄣ�佷繚鍏讳汉鍒楄〃鍜屽弬鏁�
loadDeviceName();
+ loadUserOptions();
getPageParams();
});
@@ -274,20 +615,24 @@
loading.value = true;
const id = getPageId();
+ syncTempFileIds();
// 鍑嗗鎻愪氦鏁版嵁
const submitData = { ...form.value };
+ submitData.tempFileIds = form.value.tempFileIds || [];
// 纭繚鏃ユ湡鏍煎紡姝g‘
if (submitData.maintenancePlanTime && !submitData.maintenancePlanTime.includes(':')) {
submitData.maintenancePlanTime = submitData.maintenancePlanTime + ' 00:00:00';
}
- const { code } = id
+ const result = id
? await editUpkeep({ id: id, ...submitData })
: await addUpkeep(submitData);
+ const { code } = result || {};
if (code == 200) {
showToast(`${id ? "缂栬緫" : "鏂板"}璁″垝鎴愬姛`);
setTimeout(() => {
+ uni.removeStorageSync('repairId');
uni.navigateBack();
}, 1500);
} else {
@@ -301,32 +646,32 @@
// 杩斿洖涓婁竴椤�
const goBack = () => {
+ // 娓呴櫎瀛樺偍鐨刬d
+ uni.removeStorageSync('repairId');
uni.navigateBack();
-};
-
-// 鑾峰彇椤甸潰鍙傛暟
-const getPageParams = () => {
- const pages = getCurrentPages();
- const currentPage = pages[pages.length - 1];
- const options = currentPage.options;
-
- // 鏍规嵁鏄惁鏈塱d鍙傛暟鏉ュ垽鏂槸鏂板杩樻槸缂栬緫
- if (options.id) {
- // 缂栬緫妯″紡锛岃幏鍙栬鎯�
- loadForm(options.id);
- } else {
- // 鏂板妯″紡
- loadForm();
- }
};
// 鑾峰彇椤甸潰ID
const getPageId = () => {
- const pages = getCurrentPages();
- const currentPage = pages[pages.length - 1];
- const options = currentPage.options;
- return options.id;
+ return normalizeId(uni.getStorageSync('repairId'));
};
+
+// 鑾峰彇椤甸潰鍙傛暟
+const getPageParams = () => {
+ const id = getPageId();
+ if (id) {
+ loadForm(id);
+ } else {
+ operationType.value = 'add';
+ attachmentList.value = [];
+ form.value.tempFileIds = [];
+ loadForm();
+ }
+};
+
+onUnload(() => {
+ uni.removeStorageSync('repairId');
+});
</script>
<style scoped lang="scss">
@@ -390,4 +735,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
--
Gitblit v1.9.3