This commit is contained in:
xyc 2025-09-30 09:37:26 +08:00
parent 4923dde629
commit da394abd70
15 changed files with 1377 additions and 983 deletions

View File

@ -88,6 +88,11 @@
console.error('获取用户信息失败', err) console.error('获取用户信息失败', err)
}) })
} }
// #ifdef APP-PLUS
plus.screen.lockOrientation('portrait-primary'); //
plus.navigator.setFullscreen(false);
// #endif
}, },
onShow: function() { onShow: function() {
console.log('App Show') console.log('App Show')

View File

@ -241,6 +241,7 @@
* 接单 * 接单
*/ */
openFile(orderId) { openFile(orderId) {
console.log('执行');
this.$emit('getOrder', orderId); this.$emit('getOrder', orderId);
}, },
/** /**

View File

@ -1,6 +1,12 @@
import App from './App' import App from './App'
import uView from "uview-ui"; import uView from "uview-ui";
import config from '@/config' import config from '@/config';
import {
checkPermi,
checkRole
} from './utils/permission'
Vue.prototype.checkPermi = checkPermi
Vue.prototype.checkRole = checkRole
const baseUrl = config.baseUrl const baseUrl = config.baseUrl
Vue.prototype.$baseUrl = baseUrl; Vue.prototype.$baseUrl = baseUrl;
const baseImageUrl = config.baseImageUrl const baseImageUrl = config.baseImageUrl

View File

@ -31,7 +31,7 @@
<image src="/static/images/j2.png" mode=""></image> <image src="/static/images/j2.png" mode=""></image>
<view class="">业务管理</view> <view class="">业务管理</view>
</view> </view>
<view class="jg_box" v-if="checkPermi(['repair_statistics_show'])"> <view class="jg_box" v-if="checkPermi(['repair_statistics_show'])" @click="goToPage(3)">
<image src="/static/images/j3.png" mode=""></image> <image src="/static/images/j3.png" mode=""></image>
<view class="">数据统计</view> <view class="">数据统计</view>
</view> </view>
@ -70,23 +70,24 @@
<view class="three1"> <view class="three1">
<view class=""> <view class="">
<view class="t_size">进厂数</view> <view class="t_size">进厂数</view>
<view class="t_num">{{ bossNum.inCompanyNum }}</view> <view class="t_num">{{ bossNum.newOrderNum }}</view>
</view> </view>
<image src="/static/images/t1.png" mode=""></image> <image src="/static/images/t1.png" mode=""></image>
</view> </view>
<view class="three2"> <view class="three2">
<view class="" @click="viewOrder('wxz')"> <view class="">
<view class="t_size">维修中</view> <view class="t_size">维修中</view>
<view class="t_num">{{ bossNum.workingNum }}</view> <view class="t_num">{{ bossNum.workingNum }}</view>
</view> </view>
<image src="/static/images/t2.png" mode=""></image> <image src="/static/images/t2.png" mode=""></image>
</view> </view>
<view class="three3"> <view class="three3">
<view class="" @click="viewOrder('yjg')"> <view class="">
<view class="t_size">已竣工</view> <view class="t_size">已竣工</view>
<view class="t_num">{{ bossNum.overNum }}</view> <view class="t_num">{{ bossNum.overNum }}</view>
</view> </view>
<image src="/static/images/t3.png" mode=""></image> <image src="/static/images/t3.png" mode="">
</image>
</view> </view>
</view> </view>
<view class="db_" style="margin: 30rpx auto; " v-if="checkRole(['repair_staff'])"> <view class="db_" style="margin: 30rpx auto; " v-if="checkRole(['repair_staff'])">
@ -423,6 +424,11 @@
url: '/pages-business/businessManage/businessManage' // 3. 使 url: '/pages-business/businessManage/businessManage' // 3. 使
}); });
break; break;
case 3: // 2.
uni.navigateTo({
url: '/pages-business/statistics/statistics' // 3. 使
});
break;
case 4: case 4:
uni.navigateTo({ uni.navigateTo({
url: '/pages-home/msg/message' // 3. 使 url: '/pages-home/msg/message' // 3. 使

View File

@ -57,28 +57,34 @@
配件申请表 配件申请表
</view> </view>
<view class="repairItem"> <view class="repairItem">
<image <image v-for="(url, index) in imageUrls" :key="index" :src="imgUrlPrex+url" class="image-item"
v-for="(url, index) in imageUrls" @click="prviewImage(imageUrls,index)" />
:key="index" </view>
:src="imgUrlPrex+url" </view>
class="image-item" <view class="repairInfo" v-if="info && info.remark">
@click="prviewImage(imageUrls,index)" <view class="header">
/> 配件备注
</view>
<view class="repairItem">
{{info.remark}}
</view> </view>
</view> </view>
<view class="repairInfo"> <view class="repairInfo">
<view class="header"> <view class="header">
配件信息 配件信息
<view v-if="canOperate" style="float: right; color: #0174F6" @click="addWares">添加配件</view> <view v-if="canOperate" style="float: right; color: #0174F6" @click="addWares">添加配件</view>
<view v-if="canOperate" style="float: right; color: #0174F6;margin-right: 1rem" @click="handleSetting">移交保险公司</view> <view v-if="canOperate" style="float: right; color: #0174F6;margin-right: 1rem"
@click="handleSetting">移交保险公司</view>
</view> </view>
<uni-collapse ref="collapse"> <uni-collapse ref="collapse">
<uni-collapse-item v-for="(groupItem, index) in repairList" :key="groupItem.groupId" <uni-collapse-item v-for="(groupItem, index) in repairList" :key="groupItem.groupId"
:title="groupItem.groupName+'(合计:'+groupItem.typeNums+'种'+groupItem.nums+'个配件'"> :title="groupItem.groupName+'(合计:'+groupItem.typeNums+'种'+groupItem.nums+'个配件'">
<view class="content"> <view class="content">
<view @click="changeChoose(item)" v-for="item in groupItem.twItemList" :key="item.id" class="repairItem" > <view @click="changeChoose(item)" v-for="item in groupItem.twItemList" :key="item.id"
class="repairItem">
<view class="repairName"> <view class="repairName">
<radio v-if="canOperate && item.waresStatus === ''" :value="item.id" :checked="item.selected"/> <radio v-if="canOperate && item.waresStatus === ''" :value="item.id"
:checked="item.selected" />
{{ item.waresName }}×{{ item.waresCount }}{{ item.unitText }} {{ item.waresName }}×{{ item.waresCount }}{{ item.unitText }}
</view> </view>
<view class="grid"> <view class="grid">
@ -92,7 +98,8 @@
</view> </view>
<view style="grid-area: c" class="girdItem"> <view style="grid-area: c" class="girdItem">
<text class="label">状态</text> <text class="label">状态</text>
<text :class="getWaresStatusClass(item.waresStatus)">{{ getWaresStatus(item.waresStatus) }}</text> <text
:class="getWaresStatusClass(item.waresStatus)">{{ getWaresStatus(item.waresStatus) }}</text>
</view> </view>
<view v-if="item.handleName" style="grid-area: d" class="girdItem"> <view v-if="item.handleName" style="grid-area: d" class="girdItem">
<text class="label">审核人</text> <text class="label">审核人</text>
@ -151,19 +158,22 @@
</template> </template>
<script> <script>
import VNavigationBar from "@/components/VNavigationBar.vue"; import VNavigationBar from "@/components/VNavigationBar.vue";
import request from '@/utils/request'; import request from '@/utils/request';
import {getDictTextByCodeAndValue, formatTimestampCustom} from "@/utils/utils"; import {
import config from "@/config"; getDictTextByCodeAndValue,
formatTimestampCustom
} from "@/utils/utils";
import config from "@/config";
export default { export default {
components: { components: {
VNavigationBar VNavigationBar
}, },
data() { data() {
return { return {
viewType: "", viewType: "",
imgUrlPrex:config.baseImageUrl, imgUrlPrex: config.baseImageUrl,
id: "", id: "",
title: "", title: "",
canOperate: false, canOperate: false,
@ -190,7 +200,7 @@ export default {
selectWares: [], selectWares: [],
info: {}, info: {},
imageUrls: [], imageUrls: [],
settingForm:{ settingForm: {
id: null, id: null,
toSafe: '0', toSafe: '0',
safeName: null, safeName: null,
@ -198,7 +208,7 @@ export default {
safeMobile: null, safeMobile: null,
}, },
// //
canClick:true, canClick: true,
subLoading: false subLoading: false
}; };
}, },
@ -206,15 +216,15 @@ export default {
this.canOperate = data.canOperate this.canOperate = data.canOperate
this.id = data.id this.id = data.id
}, },
onShow(){ onShow() {
this.getTicketWares() this.getTicketWares()
this.getDetail() this.getDetail()
}, },
methods: { methods: {
toggleToSafe(e){ toggleToSafe(e) {
this.settingForm.toSafe = e.detail.value ? '1' : '0'; this.settingForm.toSafe = e.detail.value ? '1' : '0';
}, },
doSetting(){ doSetting() {
request({ request({
url: '/admin-api/repair/tw/updateSafe', url: '/admin-api/repair/tw/updateSafe',
method: 'post', method: 'post',
@ -227,7 +237,7 @@ export default {
}) })
}) })
}, },
handleSetting(){ handleSetting() {
request({ request({
url: "/admin-api/repair/tw/getById?id=" + this.id, url: "/admin-api/repair/tw/getById?id=" + this.id,
method: 'get' method: 'get'
@ -241,7 +251,7 @@ export default {
this.$refs.settingPopup.open() this.$refs.settingPopup.open()
}) })
}, },
addWares(){ addWares() {
// //
uni.navigateTo({ uni.navigateTo({
url: '/pages-repair/apply/applyForm?twId=' + this.id url: '/pages-repair/apply/applyForm?twId=' + this.id
@ -253,7 +263,7 @@ export default {
prviewImage(imgList, index) { prviewImage(imgList, index) {
let urls = [] let urls = []
imgList.forEach(i => { imgList.forEach(i => {
urls.push(this.imgUrlPrex+i) urls.push(this.imgUrlPrex + i)
}) })
uni.previewImage({ uni.previewImage({
urls: urls, urls: urls,
@ -271,16 +281,16 @@ export default {
method: 'get' method: 'get'
}) })
this.info = res.data this.info = res.data
if (this.info.repairWork){ if (this.info.repairWork) {
this.getWorkToText() this.getWorkToText()
} }
await this.getTicketById(this.info.ticketId) await this.getTicketById(this.info.ticketId)
if (this.info.images && this.info.images.length > 0) { if (this.info.images && this.info.images.length > 0) {
this.getImageUrls(this.info.images) this.getImageUrls(this.info.images)
} }
}catch{} } catch {}
}, },
getWorkToText(){ getWorkToText() {
getDictTextByCodeAndValue('repair_work_type', this.info.repairWork).then(value => { getDictTextByCodeAndValue('repair_work_type', this.info.repairWork).then(value => {
this.info.repairWork = value this.info.repairWork = value
}).catch(() => { }).catch(() => {
@ -294,8 +304,12 @@ export default {
}, },
async getTicketById(id) { async getTicketById(id) {
const res = await request({ const res = await request({
url: '/admin-api/repair/tickets/get?id=' + id, url: '/admin-api/repair/tickets/get',
method: 'get', method: 'get',
params: {
id: id,
ifApp: true
}
}) })
this.info.carBrandName = res.data.carBrandName this.info.carBrandName = res.data.carBrandName
}, },
@ -357,7 +371,7 @@ export default {
}) })
}, },
confirmOpe(type) { confirmOpe(type) {
if (!this.subLoading){ if (!this.subLoading) {
this.subLoading = true this.subLoading = true
// if(this.selectWares.length==0){ // if(this.selectWares.length==0){
// uni.showToast({ // uni.showToast({
@ -386,7 +400,9 @@ export default {
dataObj.items = [] dataObj.items = []
this.repairList.map((groupItem) => { this.repairList.map((groupItem) => {
groupItem.twItemList.map((item) => { groupItem.twItemList.map((item) => {
dataObj.items.push({id: item.id}) dataObj.items.push({
id: item.id
})
}) })
}) })
} }
@ -406,27 +422,35 @@ export default {
}, 700) }, 700)
} }
this.subLoading = false this.subLoading = false
}).catch((e) => {
//
console.error(e)
}).finally(() => {
this.subLoading = false
this.canClick = true
}) })
} catch (e) { } catch (e) {
this.subLoading = false this.subLoading = false
this.canClick = true this.canClick = true
} finally {
this.subLoading = false
} }
} }
} }
}, },
} }
} }
</script> </script>
<style lang="less" scoped> <style lang="less" scoped>
.container { .container {
height: 100%; height: 100%;
background: #F3F5F7; background: #F3F5F7;
display: flex; display: flex;
flex-direction: column; flex-direction: column;
} }
.body { .body {
flex: 1; flex: 1;
height: 0; height: 0;
overflow: auto; overflow: auto;
@ -538,16 +562,17 @@ export default {
} }
} }
} }
} }
.footer { .footer {
background-color: #fff; background-color: #fff;
display: flex; display: flex;
justify-content: space-between; justify-content: space-between;
align-items: center; align-items: center;
padding: 30rpx 32rpx; padding: 30rpx 32rpx;
.yes, .no { .yes,
.no {
width: 310rpx; width: 310rpx;
height: 76rpx; height: 76rpx;
border-radius: 38rpx 38rpx 38rpx 38rpx; border-radius: 38rpx 38rpx 38rpx 38rpx;
@ -566,25 +591,26 @@ export default {
border: 2rpx solid #858BA0; border: 2rpx solid #858BA0;
color: #858BA0; color: #858BA0;
} }
} }
.pass { .pass {
color: #2979FF; color: #2979FF;
} }
.no_pass { .no_pass {
color: #E8A321; color: #E8A321;
} }
.image-item { .image-item {
width: 100rpx; width: 100rpx;
height: 100rpx; height: 100rpx;
border-radius: 8rpx; border-radius: 8rpx;
object-fit: cover; object-fit: cover;
margin-right: 10rpx; /* 增加右边距,使图片之间有间隔 */ margin-right: 10rpx;
} /* 增加右边距,使图片之间有间隔 */
}
.fullscreen-container { .fullscreen-container {
position: fixed; position: fixed;
top: 0; top: 0;
left: 0; left: 0;
@ -595,63 +621,77 @@ export default {
justify-content: center; justify-content: center;
align-items: center; align-items: center;
z-index: 9999; z-index: 9999;
} }
.fullscreen-image { .fullscreen-image {
max-width: 90vw; /* 最大宽度为视口宽度的90% */ max-width: 90vw;
max-height: 90vh; /* 最大高度为视口高度的90% */ /* 最大宽度为视口宽度的90% */
object-fit: contain; /* 保持宽高比 */ max-height: 90vh;
/* 最大高度为视口高度的90% */
object-fit: contain;
/* 保持宽高比 */
cursor: pointer; cursor: pointer;
margin: 0 20rpx; /* 增加左右边距,使图片距离屏幕边缘有一定距离 */ margin: 0 20rpx;
} /* 增加左右边距,使图片距离屏幕边缘有一定距离 */
.popup-content { }
.popup-content {
width: 80%; width: 80%;
max-width: 400px; max-width: 400px;
background-color: #fff; background-color: #fff;
padding: 20px; padding: 20px;
border-radius: 10px; border-radius: 10px;
margin: auto; margin: auto;
} }
.popup-title {
.popup-title {
font-size: 18px; font-size: 18px;
margin-bottom: 20px; margin-bottom: 20px;
text-align: center; text-align: center;
} }
.uni-form-item {
.uni-form-item {
display: flex; display: flex;
align-items: center; align-items: center;
margin-bottom: 15px; margin-bottom: 15px;
} }
.uni-label {
.uni-label {
width: 20rem; width: 20rem;
} }
.uni-input {
.uni-input {
padding: 2px; padding: 2px;
border: 1px solid #ccc; border: 1px solid #ccc;
border-radius: 5px; border-radius: 5px;
box-sizing: border-box; box-sizing: border-box;
height: 2rem; height: 2rem;
width: 26rem; width: 26rem;
} }
.popup-footer {
.popup-footer {
display: flex; display: flex;
justify-content: space-between; justify-content: space-between;
margin-top: 20px; margin-top: 20px;
} }
.confirm-btn, .cancel-btn {
.confirm-btn,
.cancel-btn {
flex: 1; flex: 1;
margin: 0 5px; margin: 0 5px;
height: 40px; height: 40px;
line-height: 40px; line-height: 40px;
border: none; border: none;
border-radius: 5px; border-radius: 5px;
} }
.confirm-btn {
.confirm-btn {
background-color: #0174F6; background-color: #0174F6;
color: #fff; color: #fff;
} }
.cancel-btn {
.cancel-btn {
background-color: #f5f5f5; background-color: #f5f5f5;
color: #666; color: #666;
} }
</style> </style>

View File

@ -25,8 +25,8 @@
汇报内容 汇报内容
</view> </view>
<!-- <u--textarea v-if="type != 'look'" v-model="report.reportContent" placeholder="请输入内容"></u--textarea> --> <!-- <u--textarea v-if="type != 'look'" v-model="report.reportContent" placeholder="请输入内容"></u--textarea> -->
<textarea auto-height v-model="report.reportContent" placeholder="请输入内容"></textarea> <textarea maxlength="-1" auto-height v-model="report.reportContent" placeholder="请输入内容"></textarea>
<view class="x_"></view> <!-- <view class="x_"></view> -->
<view class="d_b" @click="handleUpload" v-if="type != 'look'"> <view class="d_b" @click="handleUpload" v-if="type != 'look'">
<view class="">附件</view> <view class="">附件</view>
<view class="lan_">上传附件 <view class="lan_">上传附件

View File

@ -233,7 +233,7 @@
</picker> </picker>
</view> </view>
</view> </view>
<view style="display: flex;align-items: center"> <!-- <view style="display: flex;align-items: center">
<view class="infoItem" style="flex: 1" v-if="cusFromList && cusFromList.length>0"> <view class="infoItem" style="flex: 1" v-if="cusFromList && cusFromList.length>0">
<text class="label">客户来源</text> <text class="label">客户来源</text>
<picker @change="cusFromChange" :value="cusFromIndex" :range="cusFromList" <picker @change="cusFromChange" :value="cusFromIndex" :range="cusFromList"
@ -248,6 +248,12 @@
<view class="uni-input">{{ busiFromList[busiFromIndex].label}}</view> <view class="uni-input">{{ busiFromList[busiFromIndex].label}}</view>
</picker> </picker>
</view> </view>
</view> -->
<view style="display: flex;align-items: center">
<text class="label">渠道/来源</text>
<uni-data-picker :localdata="busiAndCusList" v-model='busIAndCusValue'
popup-title="请选择渠道和来源" :map="{text:'name',value:'name'}"
@change="onBusiAndCuschange"></uni-data-picker>
</view> </view>
<view style="display: flex;align-items: center"> <view style="display: flex;align-items: center">
<view class="infoItem" style="flex: 1"> <view class="infoItem" style="flex: 1">
@ -392,7 +398,8 @@
import config from "@/config"; import config from "@/config";
import { import {
formatTimestamp, formatTimestamp,
formatTimestampCustom formatTimestampCustom,
handleTree
} from "@/utils/utils"; } from "@/utils/utils";
import upload from "@/utils/upload"; import upload from "@/utils/upload";
import { import {
@ -424,6 +431,7 @@
cusFromIndex: 0, cusFromIndex: 0,
busiFromList: [], busiFromList: [],
busiFromIndex: 0, busiFromIndex: 0,
busiAndCusList: [],
partDisposals: [], partDisposals: [],
partDisposalIndex: 0, partDisposalIndex: 0,
formData: { formData: {
@ -479,6 +487,7 @@
ticketNo: '', ticketNo: '',
// //
pageData: {}, pageData: {},
busIAndCusValue: undefined
} }
}, },
watch: { watch: {
@ -543,6 +552,7 @@
this.initDict("repair_type") this.initDict("repair_type")
this.initDict("cus_data_from") this.initDict("cus_data_from")
this.initDict("repair_part_disposal") this.initDict("repair_part_disposal")
this.queryBusiAndCus()
}, },
onShow() {}, onShow() {},
methods: { methods: {
@ -588,8 +598,6 @@
this.partDisposalIndex = newIndex; this.partDisposalIndex = newIndex;
this.formData.partDisposal = this.partDisposals[newIndex].value; this.formData.partDisposal = this.partDisposals[newIndex].value;
}, },
async initDict(dictCode) { async initDict(dictCode) {
let dictArray = getStorageWithExpiry(dictCode); let dictArray = getStorageWithExpiry(dictCode);
console.log(dictArray, "partDisposals") console.log(dictArray, "partDisposals")
@ -652,6 +660,24 @@
}) })
} }
}, },
/**
* 获取业务来源和渠道
*/
queryBusiAndCus() {
request({
url: `/admin-api/business/list`,
method: 'GET',
params: {
systemCode: 'repair'
}
}).then(res => {
this.busiAndCusList = handleTree(res.data, 'id', 'pid')
})
},
onBusiAndCuschange(e) {
this.formData.busiFrom = e.detail.value[0].value
this.formData.cusFrom = e.detail.value[1].value
},
buildRepairType() { buildRepairType() {
if (this.pageData.repairType) { if (this.pageData.repairType) {
this.repairTypes.map((item, index) => { this.repairTypes.map((item, index) => {
@ -708,7 +734,8 @@
* 创建工单前上传图片 * 创建工单前上传图片
*/ */
submitUpload() { submitUpload() {
if (this.userInfo === null || this.carList.length === 0 || this.selectedProj.length === 0) { if (this.userInfo === null || this.carList.length === 0 || this.selectedProj.length === 0 || !this.formData
.busiFrom || !this.formData.cusFrom) {
uni.showToast({ uni.showToast({
title: '请完善信息', title: '请完善信息',
icon: 'none' icon: 'none'
@ -721,12 +748,12 @@
// //
submit() { submit() {
let fileStr = this.fileList.map(item => item.url.replace(config.baseImageUrl, "")).join(",") let fileStr = this.fileList.map(item => item.url.replace(config.baseImageUrl, "")).join(",")
if (!this.formData.busiFrom) { // if (!this.formData.busiFrom) {
this.formData.busiFrom = this.busiFromList[this.busiFromIndex].value // this.formData.busiFrom = this.busiFromList[this.busiFromIndex].value
} // }
if (!this.formData.cusFrom) { // if (!this.formData.cusFrom) {
this.formData.cusFrom = this.cusFromList[this.cusFromIndex].value // this.formData.cusFrom = this.cusFromList[this.cusFromIndex].value
} // }
const data = { const data = {
userId: this.userInfo.id, userId: this.userInfo.id,
ticketNo: this.ticketNo, ticketNo: this.ticketNo,
@ -798,9 +825,20 @@
method: 'GET', method: 'GET',
params: params params: params
}).then(res => { }).then(res => {
if (res.data.records.length > 0) { if (res.data.records.length == 1) {
this.userInfo = res.data.records[0] this.userInfo = res.data.records[0]
this.getCarList() this.getCarList()
} else if (res.data.records.length != 0) {
uni.showActionSheet({
title: '选择客户',
itemList: res.data.records.map(m => m.cusName + m.phoneNumber),
success: ({
tapIndex
}) => {
this.userInfo = res.data.records[tapIndex]
this.getCarList()
}
})
} }
}) })
} else { } else {

View File

@ -55,14 +55,15 @@
class="carImage" mode="aspectFit"></image> class="carImage" mode="aspectFit"></image>
<view class="carHeaderRight"> <view class="carHeaderRight">
<text class="carNumber">{{ ticketInfo.carNo }}</text> <text class="carNumber">{{ ticketInfo.carNo }}</text>
<text class="carType">{{ ticketInfo.carBrandName+" " }} <text <text class="carType">{{ ticketInfo.carBrandName+" " }} <span
v-if="ticketInfo.carInfo && ticketInfo.carInfo.carModel"> - v-if="ticketInfo.carInfo && ticketInfo.carInfo.carModel"> -
{{ticketInfo.carInfo.carModel}}</text></text> {{ticketInfo.carInfo.carModel}}</span></text>
</view> </view>
</view> </view>
<view class="line"></view> <view class="line"></view>
<view class="carBody"> <view class="carBody">
<view style="display: flex;flex-direction: column;"> <view style="display: flex;flex-direction: column;">
<view v-if="checkPermi(['repair:user:look'])">
<view class="infoItem"> <view class="infoItem">
<view class="label">车主</view> <view class="label">车主</view>
<view class="value">{{ ticketInfo.userName }}</view> <view class="value">{{ ticketInfo.userName }}</view>
@ -92,6 +93,8 @@
<text class="value">{{ ticketInfo.busiFrom }}</text> <text class="value">{{ ticketInfo.busiFrom }}</text>
</view> </view>
<view class="line"></view> <view class="line"></view>
</view>
<view class="infoItem"> <view class="infoItem">
<view class="label">车架号</view> <view class="label">车架号</view>
<view class="value">{{ ticketInfo.carVin }}</view> <view class="value">{{ ticketInfo.carVin }}</view>
@ -102,50 +105,64 @@
</view> </view>
<view class="infoItem"> <view class="infoItem">
<view class="label">车辆注册日期</view> <view class="label">车辆注册日期</view>
<view class="value">{{ ticketInfo.carRegisterDate }}</view> <view class="value">{{ formatDate(ticketInfo.carInfo.carRegisterDate) }}</view>
</view> </view>
<view class="infoItem"> <view class="infoItem">
<view class="label">车龄()</view> <view class="label">车龄()</view>
<view class="value">{{ ticketInfo.carInfo.carYear }}</view> <view class="value">{{ ticketInfo.carInfo.carYear }}</view>
</view> </view>
<view class="infoItem"> <view class="infoItem">
<view class="label">年检到期时间</view> <view class="label">当前表显里程</view>
<view class="value">{{ ticketInfo.nextInspectionDate}}</view> <view class="value">{{ ticketInfo.carInfo.mileageTraveled}}</view>
</view>
<view class="infoItem">
<view class="label">下次年检时间</view>
<view class="value">{{ formatDate(ticketInfo.carInfo.nextInspectionDate)}}</view>
</view> </view>
<view class="infoItem"> <view class="infoItem">
<view class="label">保险到期时间</view> <view class="label">保险到期时间</view>
<view class="value">{{ ticketInfo.insuranceExpiryDate}}</view> <view class="value">{{ formatDate(ticketInfo.carInfo.insuranceExpiryDate)}}</view>
</view> </view>
<view class="infoItem"> <!-- <view class="infoItem">
<view class="label">承保险种</view> <view class="label">承保险种</view>
<view class="value">{{ ticketInfo.insuranceType}}</view> <view class="value">{{ ticketInfo.insuranceType}}</view>
</view> </view>
<view class="infoItem"> <view class="infoItem">
<view class="label">上年保费</view> <view class="label">上年保费</view>
<view class="value">{{ ticketInfo.jiaoqiang}} {{ ticketInfo.shangye}}</view> <view class="value">{{ ticketInfo.jiaoqiang}} {{ ticketInfo.shangye}}</view>
</view> -->
<view class="infoItem">
<view class="label">下次保养里程</view>
<view class="value">{{ ticketInfo.carInfo.nextMaintenanceMileage}}</view>
</view> </view>
<view class="infoItem"> <view class="infoItem">
<view class="label">最近保养日期</view> <view class="label">最近保养日期</view>
<view class="value">{{ ticketInfo.maintenanceDate}}</view> <view class="value">{{ ticketInfo.maintenanceDate}}</view>
</view> </view>
<view class="infoItem">
<view class="label">表显里程</view>
<view class="value">{{ ticketInfo.mileageTraveled}}</view>
</view>
<view class="infoItem"> <view class="infoItem">
<view class="label">最近保养公里数</view> <view class="label">最近保养公里数</view>
<view class="value">{{ ticketInfo.maintenanceMileage}}</view> <view class="value">{{ ticketInfo.maintenanceMileage}}</view>
</view> </view>
</view> <view class="infoItem" style="display: block;">
<view class="label">档案资料</view>
<view class="value">
<view class="projList"> <view class="projList">
<view class="projImg" v-if="carImgList.length>0"> <view class="projImg" v-if="carImgList.length>0">
<image v-for="(img, imgIndex) in carImgList" <image v-for="(img, imgIndex) in carImgList"
@click="prviewImage(carImgList,imgIndex)" :key="imgIndex" @click="prviewImage(carImgList,imgIndex)" :key="imgIndex"
:src="imgUrlPrex + img.image" class="projImgItem"></image> :src="img.image" class="projImgItem"></image>
</view> </view>
</view> </view>
</view> </view>
</view> </view>
<view class="infoItem" v-if="checkPermi(['tickets:car:edit'])">
<view class="label" style="color: rgb(41, 121, 255);" @click="showCarPopup">点击编辑车辆信息
</view>
</view>
</view>
</view>
</view>
</view> </view>
<!-- 维修工时项目-编辑 --> <!-- 维修工时项目-编辑 -->
<view <view
@ -456,6 +473,32 @@
</view> </view>
</view> </view>
</uni-popup> </uni-popup>
<uni-popup ref="carPopup" type="center" border-radius="10px 10px 10px 10px" background-color="#fff">
<view class="car-popup-out">
<uni-forms ref="baseForm" :modelValue="carFormData">
<uni-forms-item label="下次年检日期" required>
<picker mode="date" :value="carFormData.nextInspectionDate" :start="startDate" :end="endDate"
@change="bindnextInspectionDateChange">
<view class="uni-input">
{{ carFormData.nextInspectionDate ? carFormData.nextInspectionDate : '选择日期' }}
</view>
</picker>
</uni-forms-item>
<uni-forms-item label="下次保养里程" required>
<uni-easyinput v-model="carFormData.nextMaintenanceMileage" placeholder="请输入" />
</uni-forms-item>
<uni-forms-item label="保险到期时间" required>
<picker mode="date" :value="carFormData.insuranceExpiryDate" :start="startDate" :end="endDate"
@change="bindInsuranceExpiryDateChange">
<view class="uni-input">
{{ carFormData.insuranceExpiryDate ? carFormData.insuranceExpiryDate : '选择日期' }}
</view>
</picker>
</uni-forms-item>
</uni-forms>
<button @click="submitCarForm">确定</button>
</view>
</uni-popup>
<view> <view>
<u-modal :show="show" :title="title" confirmText="电话" cancelText="短信" showCancelButton @cancel="message" <u-modal :show="show" :title="title" confirmText="电话" cancelText="短信" showCancelButton @cancel="message"
@confirm="phone " closeOnClickOverlay @close="show = false"> @confirm="phone " closeOnClickOverlay @close="show = false">
@ -487,7 +530,10 @@
setStorageWithExpiry, setStorageWithExpiry,
getStorageWithExpiry getStorageWithExpiry
} from '@/utils/auth' } from '@/utils/auth'
import config from '@/config' import config from '@/config';
import {
fixScreen
} from '@/utils/fixScreen.js';
export default { export default {
components: { components: {
VNavigationBar, VNavigationBar,
@ -526,10 +572,29 @@
repairItemId: "", repairItemId: "",
itemName: "", itemName: "",
}, },
//
carFormData: {
nextInspectionDate: undefined,
insuranceExpiryDate: undefined,
nextMaintenanceMileage: undefined,
mileageTraveled: undefined,
},
// //
canOpenCus: false, canOpenCus: false,
// //
canSeeMoney: false, canSeeMoney: false,
listStyles: {
//
border: false,
// 线
dividline: false,
// 线
borderStyle: {
width: 1,
color: 'blue',
radius: 2
}
},
//:working-|done_half-|done- //:working-|done_half-|done-
nowChooseOperate: "", nowChooseOperate: "",
carInfo: {}, carInfo: {},
@ -555,7 +620,8 @@
activeProjTabKey: 0, activeProjTabKey: 0,
activePartTabKey: 0, activePartTabKey: 0,
processList: [{}, {}], processList: [{}, {}],
carImgList: [] carImgList: [],
cararchivesPhotos: []
}; };
}, },
watch: { watch: {
@ -587,8 +653,20 @@
this.refreshData(false) this.refreshData(false)
} }
}, },
onShow() {
//
fixScreen();
},
computed: {
startDate() {
return this.getDate('start');
},
endDate() {
return this.getDate('end');
}
},
methods: { methods: {
formatDate,
chooseProjTab(index) { chooseProjTab(index) {
this.activeProjTabKey = index this.activeProjTabKey = index
}, },
@ -641,6 +719,67 @@
} }
}) })
}, },
/**
* 打开编辑车辆弹窗
*/
showCarPopup() {
this.carFormData = this.ticketInfo.carInfo
if (this.carFormData.insuranceExpiryDate) {
this.carFormData.insuranceExpiryDate = formatDate(this.carFormData.insuranceExpiryDate)
}
if (this.carFormData.nextInspectionDate) {
this.carFormData.nextInspectionDate = formatDate(this.carFormData.nextInspectionDate)
}
this.$refs.carPopup.open()
},
/**
* @param {Object} type获取时间
*/
getDate(type) {
const date = new Date();
let year = date.getFullYear();
let month = date.getMonth() + 1;
let day = date.getDate();
if (type === 'start') {
year = year - 10;
} else if (type === 'end') {
year = year + 10;
}
month = month > 9 ? month : '0' + month;
day = day > 9 ? day : '0' + day;
return `${year}-${month}-${day}`;
},
/**
* 修改下次年检时间
* @param {Object} e
*/
bindnextInspectionDateChange(e) {
console.log('e', e);
this.carFormData.nextInspectionDate = e.detail.value
},
bindInsuranceExpiryDateChange(e) {
this.carFormData.insuranceExpiryDate = e.detail.value
},
submitCarForm() {
// nextInspectionDate yyyy-MM-dd
if (this.carFormData.nextInspectionDate) {
this.carFormData.nextInspectionDate = new Date(this.carFormData.nextInspectionDate).getTime()
}
// nextInspectionDate yyyy-MM-dd
if (this.carFormData.insuranceExpiryDate) {
this.carFormData.insuranceExpiryDate = new Date(this.carFormData.insuranceExpiryDate).getTime()
}
request({
url: `/admin-api/base/carMain/update`,
method: 'PUT',
data: this.carFormData
})
this.$refs.carPopup.close()
},
/** /**
* 监听输入框 * 监听输入框
*/ */
@ -669,7 +808,11 @@
prviewImage(imgList, index) { prviewImage(imgList, index) {
let urls = [] let urls = []
imgList.forEach(i => { imgList.forEach(i => {
if (!i.image.includes("http")) {
urls.push(this.imgUrlPrex + i.image) urls.push(this.imgUrlPrex + i.image)
} else {
urls.push(i.image)
}
}) })
uni.previewImage({ uni.previewImage({
urls: urls, urls: urls,
@ -782,6 +925,7 @@
} }
}, },
afterRead(file) { afterRead(file) {
console.log('文件', file);
uni.showLoading({ uni.showLoading({
title: '正在上传中...', title: '正在上传中...',
mask: true mask: true
@ -1012,8 +1156,11 @@
} }
if (this.loginUser.roleCodes.includes("repair_staff")) { if (this.loginUser.roleCodes.includes("repair_staff")) {
// //
const has02 = this.ticketInfo.items.some(item => item.itemType === '02');
console.log('has02', has02);
this.content.push({ this.content.push({
text: '配件申请', // text: '',
text: !has02 ? '是否申请配件' : '是否增加配件',
active: false, active: false,
code: "apply" code: "apply"
}) })
@ -1275,6 +1422,7 @@
}).then((res) => { }).then((res) => {
let resultObj = res.data let resultObj = res.data
resultObj.statusStr = getOrderStatusText(res.data.ticketsStatus, res.data.isHandover) resultObj.statusStr = getOrderStatusText(res.data.ticketsStatus, res.data.isHandover)
// //
if (null != resultObj.carInfo.carRegisterDate) { if (null != resultObj.carInfo.carRegisterDate) {
resultObj.carRegisterDate = formatDate(resultObj.carInfo.carRegisterDate) resultObj.carRegisterDate = formatDate(resultObj.carInfo.carRegisterDate)
@ -1379,8 +1527,17 @@
} }
if (resultObj.carInfo.carLicenseImg) { if (resultObj.carInfo.carLicenseImg) {
resultObj.carInfo.carLicenseImg.split(",").map((item) => { resultObj.carInfo.carLicenseImg.split(",").map((item) => {
let url = item.includes("http") ? item : this.imgUrlPrex + item;
this.carImgList.push({ this.carImgList.push({
image: item image: url
});
});
}
if (resultObj.carInfo.archivesPhoto) {
resultObj.carInfo.archivesPhoto.split(",").map((item) => {
this.cararchivesPhotos.push({
url: this.$baseImageUrl + item
}) })
}) })
} }
@ -1457,7 +1614,7 @@
url: '/pages-order/reviewList/reviewList?formData=' + encodeURIComponent(JSON.stringify( url: '/pages-order/reviewList/reviewList?formData=' + encodeURIComponent(JSON.stringify(
formData)) formData))
}) })
} },
} }
} }
</script> </script>
@ -2309,4 +2466,8 @@
background-color: #efefef; background-color: #efefef;
} }
} }
.car-popup-out {
padding: 30rpx 40rpx;
}
</style> </style>

View File

@ -34,7 +34,7 @@
</view> </view>
</view> </view>
<view style="text-align: center" v-if="repairList.length==0"> <view style="text-align: center" v-if="repairList.length==0">
<image class="" src="@/static/images/nothing.png" ></image> <image class="" src="@/static/images/nothing.png"></image>
</view> </view>
</scroll-view> </scroll-view>
</view> </view>
@ -65,7 +65,7 @@
</view> </view>
</view> </view>
<view style="text-align: center" v-if="selectedRepairList.length==0"> <view style="text-align: center" v-if="selectedRepairList.length==0">
<image class="" src="@/static/images/nothing.png" ></image> <image class="" src="@/static/images/nothing.png"></image>
</view> </view>
</view> </view>
<button type="primary" @click="closeWaresPopup">关闭</button> <button type="primary" @click="closeWaresPopup">关闭</button>
@ -74,6 +74,10 @@
<!-- 普通弹窗---拍照上传 --> <!-- 普通弹窗---拍照上传 -->
<uni-popup ref="popup" background-color="#fff"> <uni-popup ref="popup" background-color="#fff">
<view class="popup-content" style="padding: 15px;"> <view class="popup-content" style="padding: 15px;">
<view class="dl-avatar-box">
<text>备注</text>
<uni-easyinput v-model="remark" placeholder="请输入备注"></uni-easyinput>
</view>
<view class="dl-avatar-box"> <view class="dl-avatar-box">
<uni-file-picker :value="fileList" :sizeType="sizeType" @select="afterRead" @delete="deleteFile" <uni-file-picker :value="fileList" :sizeType="sizeType" @select="afterRead" @delete="deleteFile"
limit="9" title="请上传配件申请单照片最多选择9张图片"></uni-file-picker> limit="9" title="请上传配件申请单照片最多选择9张图片"></uni-file-picker>
@ -96,7 +100,9 @@
} from "@/utils/auth"; } from "@/utils/auth";
import upload from "@/utils/upload"; import upload from "@/utils/upload";
import config from "@/config"; import config from "@/config";
import {data} from "uview-ui/libs/mixin/mixin"; import {
data
} from "uview-ui/libs/mixin/mixin";
export default { export default {
components: { components: {
VNavigationBar VNavigationBar
@ -124,6 +130,8 @@
isTriggered: false, isTriggered: false,
userInfo: null, userInfo: null,
fileList: [], fileList: [],
//
remark: undefined,
sizeType: ['compressed'], sizeType: ['compressed'],
twId: null twId: null
}; };
@ -145,7 +153,7 @@
this.init() this.init()
}, },
methods: { methods: {
closeWaresPopup(){ closeWaresPopup() {
this.$refs.waresPopup.close() this.$refs.waresPopup.close()
// //
this.chooseTab("") this.chooseTab("")
@ -153,14 +161,14 @@
/** /**
* 打开已选择的配件弹出框 * 打开已选择的配件弹出框
*/ */
openChoosed(){ openChoosed() {
if(this.selectedRepairList.length==0){ if (this.selectedRepairList.length == 0) {
return return
} }
this.$refs.waresPopup.open("bottom") this.$refs.waresPopup.open("bottom")
}, },
addTwi(){ addTwi() {
if (!this.selectedRepairList || this.selectedRepairList.length === 0){ if (!this.selectedRepairList || this.selectedRepairList.length === 0) {
uni.showToast({ uni.showToast({
title: '请选择配件!', title: '请选择配件!',
icon: 'none' icon: 'none'
@ -177,10 +185,10 @@
name: item.name name: item.name
} }
})] })]
if(this.ifHouse){ if (this.ifHouse) {
// //
dataObj.ifHouseAdd = true dataObj.ifHouseAdd = true
}else{ } else {
// //
dataObj.ifHouseAdd = false dataObj.ifHouseAdd = false
} }
@ -204,7 +212,7 @@
afterRead(file) { afterRead(file) {
uni.showLoading({ uni.showLoading({
title: '正在上传中...', title: '正在上传中...',
mask:true mask: true
}); });
for (let i = 0; i < file.tempFilePaths.length; i++) { for (let i = 0; i < file.tempFilePaths.length; i++) {
upload({ upload({
@ -214,7 +222,7 @@
this.fileList.push({ this.fileList.push({
url: config.baseImageUrl + res.data url: config.baseImageUrl + res.data
}) })
if(i==file.tempFilePaths.length-1){ if (i == file.tempFilePaths.length - 1) {
uni.hideLoading() uni.hideLoading()
} }
console.log(this.fileList) console.log(this.fileList)
@ -324,7 +332,7 @@
} }
return m return m
}) })
console.log(thisDataList,"thisDataList") console.log(thisDataList, "thisDataList")
// concat n // concat n
if (that.pageNo != 1) { if (that.pageNo != 1) {
that.repairList = that.repairList.concat(thisDataList) that.repairList = that.repairList.concat(thisDataList)
@ -334,7 +342,7 @@
// //
that.total = res.data.total that.total = res.data.total
that.isTriggered = false that.isTriggered = false
}else{ } else {
that.isTriggered = false that.isTriggered = false
} }
}) })
@ -390,17 +398,20 @@
}) })
dataObj.items = itemList dataObj.items = itemList
} }
if (this.fileList.length > 0){ if (this.fileList.length > 0) {
dataObj.images = this.fileList.map(item => { dataObj.images = this.fileList.map(item => {
console.log(item) console.log(item)
return item.path.replace(config.baseImageUrl, '') return item.path.replace(config.baseImageUrl, '')
}).join(",") }).join(",")
} }
if (this.remark) {
dataObj.remark = this.remark
}
} }
request({ request({
url: '/admin-api/repair/tw/update', url: '/admin-api/repair/tw/update',
method: 'POST', method: 'POST',
data:dataObj data: dataObj
}).then((res) => { }).then((res) => {
console.log(res) console.log(res)
if (res.code == 200) { if (res.code == 200) {
@ -411,7 +422,7 @@
setTimeout(() => { setTimeout(() => {
uni.navigateBack() uni.navigateBack()
}, 700) }, 700)
}else{ } else {
uni.showToast({ uni.showToast({
title: '提交失败!', title: '提交失败!',
icon: 'none' icon: 'none'
@ -608,6 +619,7 @@
color: #FFFFFF; color: #FFFFFF;
} }
} }
.listItem { .listItem {
padding: 30rpx 0; padding: 30rpx 0;
border-bottom: 2rpx solid #DDDDDD; border-bottom: 2rpx solid #DDDDDD;

View File

@ -81,6 +81,13 @@
"style": { "style": {
"navigationBarTitleText": "" "navigationBarTitleText": ""
} }
},
{
"path" : "pages/white/white",
"style" :
{
"navigationBarTitleText" : ""
}
} }
], ],
"subPackages": [{ "subPackages": [{
@ -349,8 +356,7 @@
"style": { "style": {
"navigationBarTitleText": "详情页" "navigationBarTitleText": "详情页"
} }
} }]
]
}, },
{ {
"root": "pages-business", "root": "pages-business",
@ -368,7 +374,7 @@
} }
}, },
{ {
"path" : "white/white", "path" : "statistics/statistics",
"style" : "style" :
{ {
"navigationBarTitleText" : "" "navigationBarTitleText" : ""

View File

@ -1,5 +1,7 @@
## 2.0.102024-06-07 ## 2.0.122025-08-26
- 优化 uni-app x 中size 属性的类型 - 优化 uni-app x 下 size 类型问题
## 2.0.112025-08-18
- 修复 图标点击事件返回
## 2.0.92024-01-12 ## 2.0.92024-01-12
fix: 修复图标大小默认值错误的问题 fix: 修复图标大小默认值错误的问题
## 2.0.82023-12-14 ## 2.0.82023-12-14

View File

@ -11,7 +11,7 @@
* Icons 图标 * Icons 图标
* @description 用于展示 icon 图标 * @description 用于展示 icon 图标
* @tutorial https://ext.dcloud.net.cn/plugin?id=28 * @tutorial https://ext.dcloud.net.cn/plugin?id=28
* @property {Number,String} size 图标大小 * @property {Number} size 图标大小
* @property {String} type 图标图案,参考示例 * @property {String} type 图标图案,参考示例
* @property {String} color 图标颜色 * @property {String} color 图标颜色
* @property {String} customPrefix 自定义图标 * @property {String} customPrefix 自定义图标

View File

@ -85,8 +85,8 @@
} }
}, },
methods: { methods: {
_onClick() { _onClick(e) {
this.$emit('click') this.$emit('click', e)
} }
} }
} }

View File

@ -1,7 +1,7 @@
{ {
"id": "uni-icons", "id": "uni-icons",
"displayName": "uni-icons 图标", "displayName": "uni-icons 图标",
"version": "2.0.10", "version": "2.0.12",
"description": "图标组件,用于展示移动端常见的图标,可自定义颜色、大小。", "description": "图标组件,用于展示移动端常见的图标,可自定义颜色、大小。",
"keywords": [ "keywords": [
"uni-ui", "uni-ui",
@ -11,12 +11,14 @@
], ],
"repository": "https://github.com/dcloudio/uni-ui", "repository": "https://github.com/dcloudio/uni-ui",
"engines": { "engines": {
"HBuilderX": "^3.2.14" "HBuilderX": "^3.2.14",
"uni-app": "^4.08",
"uni-app-x": "^4.61"
}, },
"directories": { "directories": {
"example": "../../temps/example_temps" "example": "../../temps/example_temps"
}, },
"dcloudext": { "dcloudext": {
"sale": { "sale": {
"regular": { "regular": {
"price": "0.00" "price": "0.00"
@ -34,54 +36,74 @@
"permissions": "无" "permissions": "无"
}, },
"npmurl": "https://www.npmjs.com/package/@dcloudio/uni-ui", "npmurl": "https://www.npmjs.com/package/@dcloudio/uni-ui",
"type": "component-vue" "type": "component-vue",
"darkmode": "x",
"i18n": "x",
"widescreen": "x"
}, },
"uni_modules": { "uni_modules": {
"dependencies": ["uni-scss"], "dependencies": [
"uni-scss"
],
"encrypt": [], "encrypt": [],
"platforms": { "platforms": {
"cloud": { "cloud": {
"tcb": "y", "tcb": "x",
"aliyun": "y", "aliyun": "x",
"alipay": "n" "alipay": "x"
}, },
"client": { "client": {
"App": { "uni-app": {
"app-vue": "y", "vue": {
"app-nvue": "y", "vue2": "√",
"app-uvue": "y" "vue3": "√"
}, },
"H5-mobile": { "web": {
"Safari": "y", "safari": "√",
"Android Browser": "y", "chrome": "√"
"微信浏览器(Android)": "y",
"QQ浏览器(Android)": "y"
}, },
"H5-pc": { "app": {
"Chrome": "y", "vue": "√",
"IE": "y", "nvue": "-",
"Edge": "y", "android": {
"Firefox": "y", "extVersion": "",
"Safari": "y" "minVersion": "29"
}, },
"小程序": { "ios": "√",
"微信": "y", "harmony": "√"
"阿里": "y",
"百度": "y",
"字节跳动": "y",
"QQ": "y",
"钉钉": "y",
"快手": "y",
"飞书": "y",
"京东": "y"
}, },
"快应用": { "mp": {
"华为": "y", "weixin": "√",
"联盟": "y" "alipay": "√",
"toutiao": "√",
"baidu": "√",
"kuaishou": "-",
"jd": "-",
"harmony": "-",
"qq": "√",
"lark": "-"
}, },
"Vue": { "quickapp": {
"vue2": "y", "huawei": "√",
"vue3": "y" "union": "√"
}
},
"uni-app-x": {
"web": {
"safari": "√",
"chrome": "√"
},
"app": {
"android": {
"extVersion": "",
"minVersion": "29"
},
"ios": "√",
"harmony": "√"
},
"mp": {
"weixin": "√"
}
} }
} }
} }

View File

@ -333,6 +333,9 @@ export function builderOrder(order) {
} }
export function formatDate(timestamp) { export function formatDate(timestamp) {
if (!timestamp) {
return ''
}
// 将时间戳转换为Date对象 // 将时间戳转换为Date对象
const date = new Date(timestamp); const date = new Date(timestamp);
// 获取年月日时分秒 // 获取年月日时分秒
@ -504,3 +507,95 @@ export function formatDateCus(date) {
const d = date.getDate().toString().padStart(2, '0') const d = date.getDate().toString().padStart(2, '0')
return `${y}-${m}-${d}` return `${y}-${m}-${d}`
} }
/**
* 将扁平数组转换为树形结构并支持按 sort 字段排序
* @param {Array} data 源数据
* @param {String} id 节点ID字段名默认为 'id'
* @param {String} parentId 父节点ID字段名默认为 'parentId'
* @param {String} children 子节点字段名默认为 'children'
* @param {Number|String} rootId 根节点ID值默认为最小的parentId或0
* @param {String} sortKey 排序字段名默认为 'sort'
* @returns {Array} 树形结构数据
*/
export function handleTree(data, id, parentId, children, rootId, sortKey) {
// 参数默认值处理
id = id || 'id';
parentId = parentId || 'parentId';
children = children || 'children';
sortKey = sortKey || 'sort';
// 自动计算根节点ID取最小的parentId若无则用0
rootId = rootId || Math.min(...data.map(item => item[parentId])) || 0;
// 深拷贝源数据以避免污染原数组
const cloneData = JSON.parse(JSON.stringify(data));
// 先对所有数据进行排序(确保父节点在前)
cloneData.sort((a, b) => {
const aSort = a[sortKey] ?? 0; // 使用空值合并运算符处理undefined
const bSort = b[sortKey] ?? 0;
return aSort - bSort; // 升序排序
});
// 构建哈希表加速查找
const nodeMap = {};
cloneData.forEach(item => {
nodeMap[item[id]] = item;
item[children] = []; // 初始化children数组
});
// 构建树形结构
const tree = [];
cloneData.forEach(item => {
if (item[parentId] === rootId) {
// 根节点直接加入结果
tree.push(item);
} else {
// 非根节点找到父节点并插入
const parent = nodeMap[item[parentId]];
parent?.[children]?.push(item);
}
});
// 递归排序所有子节点
const sortChildren = (nodes) => {
nodes.forEach(node => {
if (node[children]?.length) {
node[children].sort((a, b) => {
const aSort = a[sortKey] ?? 0;
const bSort = b[sortKey] ?? 0;
return aSort - bSort;
});
sortChildren(node[children]);
}
});
};
sortChildren(tree);
return tree.length ? tree : data; // 空树时返回原数据
}
/**
* 将树形结构数据转成 uniapp multiSelector picker 可用的二维数组
* @param {Array} tree 树形结构数据
* @param {String} labelKey 节点显示字段默认为 'name'
* @param {String} childrenKey 子节点字段默认为 'children'
* @returns {Array} picker二维数组例如 [[一级], [二级], [三级]...]
*/
export function toPickerData(tree, labelKey = 'name', childrenKey = 'children') {
const result = []
function buildColumns(nodes, level = 0) {
if (!nodes || !nodes.length) return
// 当前层级的所有 name
result[level] = nodes.map(n => n[labelKey])
// 默认取第一个子节点继续递归
if (nodes[0][childrenKey] && nodes[0][childrenKey].length) {
buildColumns(nodes[0][childrenKey], level + 1)
}
}
buildColumns(tree, 0)
return result
}