admin-vben/apps/web-antd/src/views/mes/wm/stocktaking/task/data.ts

580 lines
14 KiB
TypeScript

import type { VbenFormApi, VbenFormSchema } from '#/adapter/form';
import type { VxeTableGridOptions } from '#/adapter/vxe-table';
import type { MesWmStockTakingPlanApi } from '#/api/mes/wm/stocktaking/plan';
import type { MesWmStockTakingTaskApi } from '#/api/mes/wm/stocktaking/task';
import type { MesWmStockTakingTaskLineApi } from '#/api/mes/wm/stocktaking/task/line';
import type { MesWmStockTakingResultApi } from '#/api/mes/wm/stocktaking/task/result';
import { h, markRaw } from 'vue';
import { DICT_TYPE, MesAutoCodeRuleCode, MesWmStockTakingTypeEnum } from '@vben/constants';
import { getDictOptions } from '@vben/hooks';
import { Button } from 'ant-design-vue';
import { z } from '#/adapter/form';
import { generateAutoCode } from '#/api/mes/md/autocode/record';
import { getSimpleUserList } from '#/api/system/user';
import { getRangePickerDefaultProps } from '#/utils';
import { MdItemSelect } from '#/views/mes/md/item/components';
import { StockTakingPlanSelect } from '#/views/mes/wm/stocktaking/plan/components';
import {
WmWarehouseAreaSelect,
WmWarehouseLocationSelect,
WmWarehouseSelect,
} from '#/views/mes/wm/warehouse/components';
/** 表单类型 */
export type FormType = 'create' | 'detail' | 'execute' | 'submit' | 'update';
/** 表单头部是否只读(提交、执行盘点、详情态) */
function isHeaderReadonly(formType: FormType): boolean {
return (
formType === 'detail' ||
formType === 'execute' ||
formType === 'submit'
);
}
/** 新增/修改的表单 */
export function useFormSchema(
formType: FormType,
formApi?: VbenFormApi,
): VbenFormSchema[] {
const headerReadonly = isHeaderReadonly(formType);
return [
{
fieldName: 'id',
component: 'Input',
dependencies: {
triggerFields: [''],
show: () => false,
},
},
{
fieldName: 'status',
component: 'Input',
dependencies: {
triggerFields: [''],
show: () => false,
},
},
{
fieldName: 'code',
label: '任务编码',
component: 'Input',
componentProps: {
placeholder: '请输入任务编码',
},
rules: 'required',
suffix: headerReadonly
? undefined
: () =>
h(
Button,
{
type: 'default',
onClick: async () => {
const code = await generateAutoCode(
MesAutoCodeRuleCode.WM_STOCK_TAKING_CODE,
);
await formApi?.setFieldValue('code', code);
},
},
{ default: () => '生成' },
),
},
{
fieldName: 'name',
label: '任务名称',
component: 'Input',
componentProps: {
placeholder: '请输入任务名称',
},
rules: 'required',
},
{
fieldName: 'planId',
label: '盘点方案',
component: markRaw(StockTakingPlanSelect),
componentProps: {
// 选择盘点方案后,自动带出名称、类型、起止时间和盲盘/冻结配置
onChange: async (plan?: MesWmStockTakingPlanApi.StockTakingPlan) => {
if (!plan) {
return;
}
await formApi?.setValues({
blindFlag: !!plan.blindFlag,
endTime: plan.endTime,
frozen: !!plan.frozen,
name: plan.name,
startTime: plan.startTime,
type: plan.type,
});
},
},
rules: 'selectRequired',
},
{
fieldName: 'type',
label: '盘点类型',
component: 'Select',
componentProps: {
options: getDictOptions(DICT_TYPE.MES_WM_STOCK_TAKING_TYPE, 'number'),
placeholder: '请选择盘点类型',
},
rules: 'selectRequired',
},
{
fieldName: 'startTime',
label: '开始时间',
component: 'DatePicker',
componentProps: {
placeholder: '请选择开始时间',
showTime: true,
valueFormat: 'x',
},
dependencies: {
triggerFields: ['type'],
show: (values) => values.type === MesWmStockTakingTypeEnum.DYNAMIC,
},
},
{
fieldName: 'endTime',
label: '结束时间',
component: 'DatePicker',
componentProps: {
placeholder: '请选择结束时间',
showTime: true,
valueFormat: 'x',
},
dependencies: {
triggerFields: ['type'],
show: (values) => values.type === MesWmStockTakingTypeEnum.DYNAMIC,
},
},
{
fieldName: 'takingDate',
label: '盘点日期',
component: 'DatePicker',
componentProps: {
format: 'YYYY-MM-DD',
placeholder: '请选择盘点日期',
valueFormat: 'YYYY-MM-DD',
},
rules: 'required',
},
{
fieldName: 'blindFlag',
label: '是否盲盘',
component: 'Switch',
componentProps: {
checkedChildren: '是',
unCheckedChildren: '否',
},
rules: z.boolean().default(false),
},
{
fieldName: 'frozen',
label: '是否冻结库存',
component: 'Switch',
componentProps: {
checkedChildren: '是',
unCheckedChildren: '否',
},
rules: z.boolean().default(false),
},
{
fieldName: 'userId',
label: '盘点人',
component: 'ApiSelect',
componentProps: {
allowClear: true,
api: getSimpleUserList,
labelField: 'nickname',
placeholder: '请选择盘点人',
valueField: 'id',
},
rules: 'selectRequired',
},
{
fieldName: 'remark',
label: '备注',
component: 'Textarea',
formItemClass: 'col-span-3',
componentProps: {
placeholder: '请输入备注',
rows: 3,
},
},
];
}
/** 列表的搜索表单 */
export function useGridFormSchema(): VbenFormSchema[] {
return [
{
fieldName: 'type',
label: '盘点类型',
component: 'Select',
componentProps: {
allowClear: true,
options: getDictOptions(DICT_TYPE.MES_WM_STOCK_TAKING_TYPE, 'number'),
placeholder: '请选择盘点类型',
},
},
{
fieldName: 'code',
label: '任务编码',
component: 'Input',
componentProps: {
allowClear: true,
placeholder: '请输入任务编码',
},
},
{
fieldName: 'name',
label: '任务名称',
component: 'Input',
componentProps: {
allowClear: true,
placeholder: '请输入任务名称',
},
},
{
fieldName: 'takingDate',
label: '盘点日期',
component: 'RangePicker',
componentProps: {
...getRangePickerDefaultProps(),
allowClear: true,
},
},
{
fieldName: 'status',
label: '单据状态',
component: 'Select',
componentProps: {
allowClear: true,
options: getDictOptions(
DICT_TYPE.MES_WM_STOCK_TAKING_TASK_STATUS,
'number',
),
placeholder: '请选择单据状态',
},
},
];
}
/** 列表的字段 */
export function useGridColumns(): VxeTableGridOptions<MesWmStockTakingTaskApi.StockTakingTask>['columns'] {
return [
{
field: 'code',
title: '任务编码',
minWidth: 160,
slots: { default: 'code' },
},
{
field: 'name',
title: '任务名称',
minWidth: 160,
},
{
field: 'type',
title: '盘点类型',
minWidth: 120,
cellRender: {
name: 'CellDict',
props: { type: DICT_TYPE.MES_WM_STOCK_TAKING_TYPE },
},
},
{
field: 'planName',
title: '盘点方案',
minWidth: 180,
},
{
field: 'takingDate',
title: '盘点日期',
minWidth: 180,
formatter: 'formatDate',
},
{
field: 'userNickname',
title: '盘点人',
minWidth: 120,
},
{
field: 'status',
title: '单据状态',
minWidth: 110,
cellRender: {
name: 'CellDict',
props: { type: DICT_TYPE.MES_WM_STOCK_TAKING_TASK_STATUS },
},
},
{
title: '操作',
width: 280,
fixed: 'right',
slots: { default: 'actions' },
},
];
}
/** 盘点任务行列表的字段 */
export function useLineGridColumns(
editable = true,
): VxeTableGridOptions<MesWmStockTakingTaskLineApi.StockTakingTaskLine>['columns'] {
return [
{
field: 'itemCode',
title: '物料编码',
minWidth: 140,
},
{
field: 'itemName',
title: '物料名称',
minWidth: 160,
},
{
field: 'specification',
title: '规格型号',
minWidth: 120,
},
{
field: 'unitMeasureName',
title: '单位',
width: 90,
},
{
field: 'batchCode',
title: '批次号',
minWidth: 120,
},
{
field: 'quantity',
title: '在库数量',
width: 120,
},
{
field: 'warehouseName',
title: '仓库',
minWidth: 120,
},
{
field: 'locationName',
title: '库区',
minWidth: 120,
},
{
field: 'areaName',
title: '库位',
minWidth: 120,
},
{
field: 'status',
title: '状态',
minWidth: 100,
cellRender: {
name: 'CellDict',
props: { type: DICT_TYPE.MES_WM_STOCK_TAKING_LINE_STATUS },
},
},
...(editable
? [
{
title: '操作',
width: 80,
fixed: 'right',
slots: { default: 'actions' },
} as const,
]
: []),
];
}
/** 盘点结果列表的字段 */
export function useResultGridColumns(
editable = true,
): VxeTableGridOptions<MesWmStockTakingResultApi.StockTakingResult>['columns'] {
return [
{
field: 'itemCode',
title: '产品物料编码',
minWidth: 140,
},
{
field: 'itemName',
title: '产品物料名称',
minWidth: 160,
},
{
field: 'specification',
title: '规格型号',
minWidth: 120,
},
{
field: 'unitMeasureName',
title: '单位名称',
width: 90,
},
{
field: 'warehouseName',
title: '仓库',
minWidth: 120,
},
{
field: 'locationName',
title: '库区',
minWidth: 120,
},
{
field: 'areaName',
title: '库位',
minWidth: 120,
},
{
field: 'quantity',
title: '数量',
minWidth: 120,
},
{
field: 'takingQuantity',
title: '盘点数量',
minWidth: 120,
},
...(editable
? [
{
title: '操作',
width: 160,
fixed: 'right',
slots: { default: 'actions' },
} as const,
]
: []),
];
}
/** 盘点结果新增/修改的表单 */
export function useResultFormSchema(
formApi?: VbenFormApi,
taskLines: MesWmStockTakingTaskLineApi.StockTakingTaskLine[] = [],
): VbenFormSchema[] {
return [
{
fieldName: 'lineId',
label: '盘点清单',
component: 'Select',
componentProps: {
allowClear: true,
// 选择盘点清单后,自动带出物料、批次和仓储位置信息
onChange: async (lineId?: number) => {
const line = taskLines.find((item) => item.id === lineId);
await formApi?.setValues({
areaId: line?.areaId,
batchCode: line?.batchCode,
batchId: line?.batchId,
itemId: line?.itemId,
locationId: line?.locationId,
materialStockId: line?.materialStockId,
warehouseId: line?.warehouseId,
});
},
options: taskLines.map((line) => ({
label: `${line.itemCode} - ${line.itemName} (${line.warehouseName}${
line.locationName ? ` / ${line.locationName}` : ''
}${line.areaName ? ` / ${line.areaName}` : ''})`,
value: line.id,
})),
placeholder: '请选择盘点清单(可选)',
},
formItemClass: 'col-span-3',
},
{
fieldName: 'itemId',
label: '物料',
component: markRaw(MdItemSelect),
componentProps: {
placeholder: '请选择物料',
},
rules: 'selectRequired',
},
{
fieldName: 'batchCode',
label: '批次编码',
component: 'Input',
componentProps: {
placeholder: '请输入批次编码',
},
},
{
fieldName: 'takingQuantity',
label: '盘点数量',
component: 'InputNumber',
componentProps: {
class: '!w-full',
placeholder: '请输入盘点数量',
precision: 2,
},
rules: 'required',
},
{
fieldName: 'warehouseId',
label: '仓库',
component: markRaw(WmWarehouseSelect),
componentProps: {
// 仓库变化时清空库区和库位
onChange: () =>
formApi?.setValues({
areaId: undefined,
locationId: undefined,
}),
placeholder: '请选择仓库',
},
rules: 'selectRequired',
},
{
fieldName: 'locationId',
label: '库区',
component: markRaw(WmWarehouseLocationSelect),
rules: 'selectRequired',
dependencies: {
triggerFields: ['warehouseId'],
show: (values) => !!values.warehouseId,
componentProps: (values) => ({
onChange: () => formApi?.setFieldValue('areaId', undefined),
placeholder: '请选择库区',
warehouseId: values.warehouseId,
}),
},
},
{
fieldName: 'areaId',
label: '库位',
component: markRaw(WmWarehouseAreaSelect),
rules: 'selectRequired',
dependencies: {
triggerFields: ['locationId'],
show: (values) => !!values.locationId,
componentProps: (values) => ({
locationId: values.locationId,
placeholder: '请选择库位',
}),
},
},
{
fieldName: 'remark',
label: '备注',
component: 'Textarea',
formItemClass: 'col-span-3',
componentProps: {
placeholder: '请输入备注',
rows: 3,
},
},
];
}