Merge branch 'master' of http://47.109.37.87:3000/by2025/admin-vben5
Some checks failed
/ Explore-Gitea-Actions (push) Has been cancelled

This commit is contained in:
FLL
2025-08-25 15:06:21 +08:00
11 changed files with 127 additions and 46 deletions

View File

@@ -59,3 +59,19 @@ export function workOrdersUpdate(data: WorkOrdersForm) {
export function workOrdersRemove(id: ID | IDS) { export function workOrdersRemove(id: ID | IDS) {
return requestClient.deleteWithMsg<void>(`/property/workOrders/${id}`); return requestClient.deleteWithMsg<void>(`/property/workOrders/${id}`);
} }
/**
* 派单
* @param data
*/
export function workOrdersDispatch(data: WorkOrdersForm) {
return requestClient.putWithMsg<void>('/property/workOrders/dispatch', data);
}
/**
* 抢单
* @param data
*/
export function workOrdersGrab(data: WorkOrdersForm) {
return requestClient.putWithMsg<void>('/property/workOrders/grab', data);
}

View File

@@ -83,7 +83,7 @@ export interface WorkOrdersVO {
*/ */
imgUrl: string; imgUrl: string;
evaluateImgPath: string | undefined; evaluateImgPath: string[];
/** /**
* 评价内容 * 评价内容
*/ */
@@ -101,7 +101,7 @@ export interface WorkOrdersVO {
*/ */
orderImgUrl: string; orderImgUrl: string;
orderImgPath: string|undefined; orderImgPath: string[];
reportingType: string; reportingType: string;

View File

@@ -37,6 +37,11 @@ export interface WorkOrdersTypeVO {
isTransfers: string; isTransfers: string;
excludeId: string; excludeId: string;
/**
* 处理优先级
*/
processingWeight: string;
} }
export interface WorkOrdersTypeForm extends BaseEntity { export interface WorkOrdersTypeForm extends BaseEntity {
@@ -79,6 +84,11 @@ export interface WorkOrdersTypeForm extends BaseEntity {
* 上级类型id * 上级类型id
*/ */
parentId?: string; parentId?: string;
/**
* 处理优先级
*/
processingWeight: string;
} }
export interface WorkOrdersTypeQuery extends PageQuery { export interface WorkOrdersTypeQuery extends PageQuery {

View File

@@ -8,7 +8,7 @@ import {
import { import {
workOrdersList, workOrdersList,
} from '#/api/property/businessManagement/workOrders'; } from '#/api/property/businessManagement/workOrders';
import workOrdersDetail from './work-orders-detail.vue'; import workOrdersDetail from '../workOrders/work-orders-detail.vue';
import {columns, querySchema} from './data'; import {columns, querySchema} from './data';
const formOptions: VbenFormProps = { const formOptions: VbenFormProps = {

View File

@@ -56,7 +56,7 @@ export const columns: VxeGridProps['columns'] = [
field: 'reportingType', field: 'reportingType',
slots: { slots: {
default: ({row}) => { default: ({row}) => {
return renderDict(row.reportingType, 'wy_gdsblx'); return row.reportingType!=null?renderDict(row.reportingType, 'wy_gdsblx'):'';
}, },
}, },
width: 100, width: 100,
@@ -86,7 +86,7 @@ export const columns: VxeGridProps['columns'] = [
field: 'processingWeight', field: 'processingWeight',
slots: { slots: {
default: ({row}) => { default: ({row}) => {
return renderDict(row.processingWeight, 'pro_processing_weight'); return row.processingWeight!=null?renderDict(row.processingWeight, 'pro_processing_weight'):'';
}, },
}, },
width: 100, width: 100,

View File

@@ -184,14 +184,16 @@ onMounted(async () => {
<template #action="{ row }"> <template #action="{ row }">
<Space> <Space>
<ghost-button <ghost-button
v-access:code="['property:workOrders:dispatch']"
@click.stop="handleOrders(row,'group')" @click.stop="handleOrders(row,'group')"
v-if="row.status!=='1' && row.status!=='2'" v-if="row.status!=='1' && row.status!=='2'&&row.operationMode!=='2'"
> >
{{ '派单' }} {{ '派单' }}
</ghost-button> </ghost-button>
<ghost-button <ghost-button
v-access:code="['property:workOrders:grab']"
@click.stop="handleOrders(row,'rob')" @click.stop="handleOrders(row,'rob')"
v-if="row.status!=='1' && row.status!=='2'" v-if="row.status!=='1' && row.status!=='2'&&row.operationMode!=='2'"
> >
{{ '抢单' }} {{ '抢单' }}
</ghost-button> </ghost-button>

View File

@@ -3,6 +3,7 @@ import {useVbenModal} from '@vben/common-ui';
import {cloneDeep} from '@vben/utils'; import {cloneDeep} from '@vben/utils';
import {useVbenForm} from '#/adapter/form'; import {useVbenForm} from '#/adapter/form';
import { import {
workOrdersDispatch, workOrdersGrab,
workOrdersInfo, workOrdersInfo,
workOrdersUpdate workOrdersUpdate
} from '#/api/property/businessManagement/workOrders'; } from '#/api/property/businessManagement/workOrders';
@@ -84,11 +85,15 @@ async function handleConfirm() {
} }
const data = cloneDeep(await formApi.getValues()); const data = cloneDeep(await formApi.getValues());
if(title.value === '派单'){ if(title.value === '派单'){
//派单
record.value.handler = data.handler record.value.handler = data.handler
await workOrdersDispatch(record.value)
}else{ }else{
//抢单
record.value.handler = userStore.userInfo.userId record.value.handler = userStore.userInfo.userId
await workOrdersGrab(record.value)
} }
await workOrdersUpdate(record.value)
resetInitialized(); resetInitialized();
emit('reload'); emit('reload');
modalApi.close(); modalApi.close();

View File

@@ -1,7 +1,14 @@
<script setup lang="ts"> <script setup lang="ts">
import { ref, shallowRef} from 'vue'; import {ref, shallowRef} from 'vue';
import {useVbenModal} from '@vben/common-ui'; import {useVbenModal} from '@vben/common-ui';
import {Descriptions, DescriptionsItem, Timeline, TimelineItem, Rate,Divider} from 'ant-design-vue'; import {
Descriptions,
DescriptionsItem,
Timeline,
TimelineItem,
Rate,
Divider
} from 'ant-design-vue';
import dayjs from 'dayjs'; import dayjs from 'dayjs';
import duration from 'dayjs/plugin/duration'; import duration from 'dayjs/plugin/duration';
import relativeTime from 'dayjs/plugin/relativeTime'; import relativeTime from 'dayjs/plugin/relativeTime';
@@ -21,7 +28,8 @@ const [BasicModal, modalApi] = useVbenModal({
}); });
const orderDetail = shallowRef<null | WorkOrdersVO>(null); const orderDetail = shallowRef<null | WorkOrdersVO>(null);
const handleRecords=ref<HandleRecords[]>([]) const handleRecords = ref<HandleRecords[]>([])
async function handleOpenChange(open: boolean) { async function handleOpenChange(open: boolean) {
if (!open) { if (!open) {
return null; return null;
@@ -29,8 +37,8 @@ async function handleOpenChange(open: boolean) {
modalApi.modalLoading(true); modalApi.modalLoading(true);
const {id} = modalApi.getData() as { id: number | string }; const {id} = modalApi.getData() as { id: number | string };
orderDetail.value = await workOrdersInfo(id); orderDetail.value = await workOrdersInfo(id);
if(orderDetail.value.workOrdersRecordVoList){ if (orderDetail.value.workOrdersRecordVoList) {
handleRecords.value = orderDetail.value.workOrdersRecordVoList.map( (item, index) => ({ handleRecords.value = orderDetail.value.workOrdersRecordVoList.map((item, index) => ({
status: item.status, status: item.status,
createTime: item.createTime, createTime: item.createTime,
handlerName: index === 0 ? item.initiatorPeople : item.handlerName handlerName: index === 0 ? item.initiatorPeople : item.handlerName
@@ -39,16 +47,23 @@ async function handleOpenChange(open: boolean) {
try { try {
if (orderDetail.value.orderImgUrl) { if (orderDetail.value.orderImgUrl) {
const res = await ossInfo([orderDetail.value.orderImgUrl]); const res = await ossInfo([orderDetail.value.orderImgUrl]);
orderDetail.value.orderImgPath = res?.[0]?.url let imgUrls = [];
res.forEach(item => {
imgUrls.push(item.url)
})
orderDetail.value.orderImgPath = imgUrls;
} }
if (orderDetail.value.imgUrl) { if (orderDetail.value.imgUrl) {
const res = await ossInfo([orderDetail.value.imgUrl]); const res = await ossInfo([orderDetail.value.imgUrl]);
orderDetail.value.evaluateImgPath = res?.[0]?.url let imgUrls = [];
res.forEach(item => {
imgUrls.push(item.url)
})
orderDetail.value.evaluateImgPath = imgUrls
} }
} catch (e) { } catch (e) {
} }
modalApi.modalLoading(false); modalApi.modalLoading(false);
} }
</script> </script>
@@ -63,16 +78,16 @@ async function handleOpenChange(open: boolean) {
{{ orderDetail.orderName }} {{ orderDetail.orderName }}
</DescriptionsItem> </DescriptionsItem>
<DescriptionsItem label="工单类型"> <DescriptionsItem label="工单类型">
{{orderDetail.typeName}} {{ orderDetail.typeName }}
</DescriptionsItem> </DescriptionsItem>
<DescriptionsItem label="上报类型"> <DescriptionsItem label="上报类型">
<component <component v-if="orderDetail.reportingType!=null"
:is="renderDict(orderDetail.reportingType,'wy_gdsblx')" :is="renderDict(orderDetail.reportingType,'wy_gdsblx')"
/> />
</DescriptionsItem> </DescriptionsItem>
<DescriptionsItem label="处理优先级"> <DescriptionsItem label="处理优先级">
<component <component v-if="orderDetail.processingWeight!=null"
:is="renderDict(orderDetail.processingWeight,'pro_processing_weight')" :is="renderDict(orderDetail.processingWeight,'pro_processing_weight')"
/> />
</DescriptionsItem> </DescriptionsItem>
<DescriptionsItem label="发起人"> <DescriptionsItem label="发起人">
@@ -87,9 +102,9 @@ async function handleOpenChange(open: boolean) {
<DescriptionsItem label="具体位置" :span="2"> <DescriptionsItem label="具体位置" :span="2">
{{ orderDetail.location }} {{ orderDetail.location }}
</DescriptionsItem> </DescriptionsItem>
<!-- <DescriptionsItem label="计划完成时间">--> <!-- <DescriptionsItem label="计划完成时间">-->
<!-- {{ orderDetail.planCompleTime }}--> <!-- {{ orderDetail.planCompleTime }}-->
<!-- </DescriptionsItem>--> <!-- </DescriptionsItem>-->
<DescriptionsItem label="备注" :span="2"> <DescriptionsItem label="备注" :span="2">
@@ -101,12 +116,15 @@ async function handleOpenChange(open: boolean) {
<DescriptionsItem label="完成时间"> <DescriptionsItem label="完成时间">
{{ orderDetail.compleTime }} {{ orderDetail.compleTime }}
</DescriptionsItem> </DescriptionsItem>
<DescriptionsItem label="是否超时" :span="2">
<DescriptionsItem label="是否超时" v-if="orderDetail.isTimeOut!=null"> <component v-if="orderDetail.isTimeOut!=null"
<component :is="renderDict(orderDetail.isTimeOut,'wy_sf')"
:is="renderDict(orderDetail.isTimeOut,'wy_sf')"
/> />
</DescriptionsItem> </DescriptionsItem>
<DescriptionsItem label="工单图片" v-if="orderDetail.orderImgPath" :span="2">
<img v-for="item in orderDetail.orderImgPath" :src="item" alt="图片加载失败"
class="orders-img"/>
</DescriptionsItem>
<DescriptionsItem label="服务评价" v-if="orderDetail.serviceEvalua!=null" :span="2"> <DescriptionsItem label="服务评价" v-if="orderDetail.serviceEvalua!=null" :span="2">
<Rate :value="orderDetail.serviceEvalua" disabled/> <Rate :value="orderDetail.serviceEvalua" disabled/>
</DescriptionsItem> </DescriptionsItem>
@@ -114,10 +132,8 @@ async function handleOpenChange(open: boolean) {
{{ orderDetail.serviceEvaluaText }} {{ orderDetail.serviceEvaluaText }}
</DescriptionsItem> </DescriptionsItem>
<DescriptionsItem label="评价图片" v-if="orderDetail.evaluateImgPath" :span="2"> <DescriptionsItem label="评价图片" v-if="orderDetail.evaluateImgPath" :span="2">
<img :src="orderDetail.evaluateImgPath" alt="图片加载失败" class="w-[100px] h-[100px]"/> <img v-for="item in orderDetail.evaluateImgPath" :src="item" alt="图片加载失败"
</DescriptionsItem> class="orders-img"/>
<DescriptionsItem label="工单图片" v-if="orderDetail.orderImgPath" :span="2">
<img :src="orderDetail.orderImgPath" alt="图片加载失败" class="w-[100px] h-[100px]"/>
</DescriptionsItem> </DescriptionsItem>
</Descriptions> </Descriptions>
<Divider orientation="left" orientation-margin="0px"> <Divider orientation="left" orientation-margin="0px">
@@ -125,14 +141,22 @@ async function handleOpenChange(open: boolean) {
</Divider> </Divider>
<Timeline> <Timeline>
<TimelineItem v-for="(item,index) in handleRecords" :key="index"> <TimelineItem v-for="(item,index) in handleRecords" :key="index">
<p style="display: flex;">类型 <p style="display: flex;">类型
<component <component v-if="item.status!=null"
:is="renderDict(item.status,'wy_gdclzt')" :is="renderDict(item.status,'wy_gdclzt')"
/></p> />
<p>时间{{item.createTime}}</p> </p>
<p>处理人{{item.handlerName}}</p> <p>时间{{ item.createTime }}</p>
<p>处理人{{ item.handlerName }}</p>
</TimelineItem> </TimelineItem>
</Timeline> </Timeline>
</div> </div>
</BasicModal> </BasicModal>
</template> </template>
<style scoped lang="scss">
.orders-img {
width: 100px;
margin: 5px 10px;
display: inline-block;
}
</style>

View File

@@ -15,6 +15,7 @@ import {defaultFormValueGetter, useBeforeCloseDiff} from '#/utils/popup';
import {modalSchema} from './data'; import {modalSchema} from './data';
import {workOrdersTypeTree} from "#/api/property/businessManagement/workOrdersType"; import {workOrdersTypeTree} from "#/api/property/businessManagement/workOrdersType";
import type {WorkOrdersTypeVO} from "#/api/property/businessManagement/workOrdersType/model";
const emit = defineEmits<{ reload: [] }>(); const emit = defineEmits<{ reload: [] }>();
@@ -108,7 +109,7 @@ async function queryWorkOrdersType() {
const options = await workOrdersTypeTree() const options = await workOrdersTypeTree()
formApi.updateSchema([ formApi.updateSchema([
{ {
componentProps: () => ({ componentProps: (formModel) => ({
class: 'w-full', class: 'w-full',
fieldNames: { fieldNames: {
key: 'id', key: 'id',
@@ -123,16 +124,15 @@ async function queryWorkOrdersType() {
treeLine: { showLeafIcon: false }, treeLine: { showLeafIcon: false },
treeNodeFilterProp: 'orderTypeName', treeNodeFilterProp: 'orderTypeName',
treeNodeLabelProp: 'orderTypeName', treeNodeLabelProp: 'orderTypeName',
async onSelect(type:string,node:WorkOrdersTypeVO) {
formModel.processingWeight = node.processingWeight;
},
}), }),
fieldName: 'type', fieldName: 'type',
}, },
]); ]);
} }
const filterOption = (input: string, option: any) => {
return option.label.toLowerCase().indexOf(input.toLowerCase()) >= 0;
};
</script> </script>
<template> <template>

View File

@@ -46,6 +46,16 @@ export const columns: VxeGridProps['columns'] = [
}, },
width:180, width:180,
}, },
{
title: '处理优先级',
field: 'processingWeight',
slots: {
default: ({row}) => {
return row.processingWeight!=null?renderDict(row.processingWeight, 'pro_processing_weight'):'';
},
},
width: 100,
},
{ {
title: '排序值', title: '排序值',
field: 'sort', field: 'sort',
@@ -105,6 +115,15 @@ export const modalSchema: FormSchemaGetter = () => [
}, },
rules: 'selectRequired', rules: 'selectRequired',
}, },
{
label: '处理优先级',
fieldName: 'processingWeight',
component: 'Select',
componentProps: {
options: getDictOptions('pro_processing_weight'),
},
rules: 'selectRequired',
},
{ {
label: '完成时效(小时)', label: '完成时效(小时)',
fieldName: 'completionNumber', fieldName: 'completionNumber',

View File

@@ -37,11 +37,16 @@ async function handleOpenChange(open: boolean) {
<DescriptionsItem label="类型名称"> <DescriptionsItem label="类型名称">
{{ workOrdersTypeInfoDetail.orderTypeName }} {{ workOrdersTypeInfoDetail.orderTypeName }}
</DescriptionsItem> </DescriptionsItem>
<DescriptionsItem label="运作模式" v-if="workOrdersTypeInfoDetail.operationMode!=null"> <DescriptionsItem label="运作模式" >
<component <component v-if="workOrdersTypeInfoDetail.operationMode!=null"
:is="renderDict(workOrdersTypeInfoDetail.operationMode,'pro_operation_pattern')" :is="renderDict(workOrdersTypeInfoDetail.operationMode,'pro_operation_pattern')"
/> />
</DescriptionsItem> </DescriptionsItem>
<DescriptionsItem label="处理优先级">
<component v-if="workOrdersTypeInfoDetail.processingWeight!=null"
:is="renderDict(workOrdersTypeInfoDetail.processingWeight,'pro_processing_weight')"
/>
</DescriptionsItem>
<DescriptionsItem label="排序值"> <DescriptionsItem label="排序值">
{{ workOrdersTypeInfoDetail.sort }} {{ workOrdersTypeInfoDetail.sort }}
</DescriptionsItem> </DescriptionsItem>