feat(mes):review 工序定义(pro_process)、工艺路线(pro_route)

pull/349/head
YunaiV 2026-05-25 09:25:01 +08:00
parent 01a1d3e001
commit 80a25ca767
11 changed files with 330 additions and 245 deletions

View File

@ -39,6 +39,7 @@ const selectValue = computed({
}, },
}); });
// TODO @AI
function handleFilter(input: string, option: any) { function handleFilter(input: string, option: any) {
const keyword = input.toLowerCase(); const keyword = input.toLowerCase();
const item = option?.item as MesProProcessApi.Process | undefined; const item = option?.item as MesProProcessApi.Process | undefined;

View File

@ -40,6 +40,7 @@ export function useFormSchema(formApi?: VbenFormApi): VbenFormSchema[] {
{ {
type: 'default', type: 'default',
onClick: async () => { onClick: async () => {
// TODO @AI需要 try catch 么?感觉直接去掉也可以?
try { try {
const code = await generateAutoCode( const code = await generateAutoCode(
MesAutoCodeRuleCode.PRO_PROCESS_CODE, MesAutoCodeRuleCode.PRO_PROCESS_CODE,

View File

@ -60,6 +60,7 @@ async function getList() {
/** 新增工序步骤 */ /** 新增工序步骤 */
function handleCreate() { function handleCreate() {
// TODO @AI max || 0
const maxSort = const maxSort =
list.value.length > 0 list.value.length > 0
? Math.max(...list.value.map((item) => item.sort || 0)) ? Math.max(...list.value.map((item) => item.sort || 0))

View File

@ -14,6 +14,10 @@ import { Button } from 'ant-design-vue';
import { z } from '#/adapter/form'; import { z } from '#/adapter/form';
import { generateAutoCode } from '#/api/mes/md/autocode/record'; import { generateAutoCode } from '#/api/mes/md/autocode/record';
import {
MdItemSelect,
MdProductBomSelect,
} from '#/views/mes/md/item/components';
import { MesAutoCodeRuleCode } from '#/views/mes/utils/constants'; import { MesAutoCodeRuleCode } from '#/views/mes/utils/constants';
/** 工艺路线表单 */ /** 工艺路线表单 */
@ -333,3 +337,119 @@ export function useRouteProductBomGridColumns(): VxeTableGridOptions<MesProRoute
}, },
]; ];
} }
/** 工艺路线产品表单 */
export function useRouteProductFormSchema(
onItemChange?: (item: any) => void,
): VbenFormSchema[] {
return [
{
fieldName: 'id',
component: 'Input',
dependencies: { triggerFields: [''], show: () => false },
},
{
fieldName: 'routeId',
component: 'Input',
dependencies: { triggerFields: [''], show: () => false },
},
// 产品物料使用业务自定义选择器change 回填编码/名称/规格/单位
{
fieldName: 'itemId',
label: '产品',
component: MdItemSelect as any,
componentProps: {
onChange: onItemChange,
},
formItemClass: 'col-span-2',
rules: 'selectRequired',
},
{
fieldName: 'quantity',
label: '生产数量',
component: 'InputNumber',
componentProps: { class: '!w-full', min: 1, precision: 0 },
rules: z.number().default(1),
},
{
fieldName: 'productionTime',
label: '生产用时',
component: 'InputNumber',
componentProps: { class: '!w-full', min: 0, precision: 2 },
rules: z.number().default(1),
},
{
fieldName: 'timeUnitType',
label: '时间单位',
component: 'Select',
componentProps: {
allowClear: true,
options: getDictOptions(DICT_TYPE.MES_TIME_UNIT_TYPE),
placeholder: '请选择',
},
rules: z.string().default('MINUTE'),
},
{
fieldName: 'remark',
label: '备注',
component: 'Textarea',
formItemClass: 'col-span-2',
componentProps: { maxLength: 250, placeholder: '请输入备注', rows: 2 },
},
];
}
/** 工艺路线产品 BOM 表单 */
export function useRouteProductBomFormSchema(
itemId: () => number,
onBomChange?: (bom: any) => void,
): VbenFormSchema[] {
return [
{
fieldName: 'id',
component: 'Input',
dependencies: { triggerFields: [''], show: () => false },
},
{
fieldName: 'routeId',
component: 'Input',
dependencies: { triggerFields: [''], show: () => false },
},
{
fieldName: 'processId',
component: 'Input',
dependencies: { triggerFields: [''], show: () => false },
},
{
fieldName: 'productId',
component: 'Input',
dependencies: { triggerFields: [''], show: () => false },
},
// BOM 物料:依赖产品物料,使用业务自定义选择器
{
fieldName: 'itemId',
label: 'BOM 物料',
component: MdProductBomSelect as any,
componentProps: () => ({
itemId: itemId(),
onChange: onBomChange,
placeholder: '请选择 BOM 物料',
}),
rules: 'selectRequired',
},
{
fieldName: 'quantity',
label: '用料比例',
component: 'InputNumber',
componentProps: { class: '!w-full', min: 0, precision: 2 },
rules: z.number().default(1),
},
{
fieldName: 'remark',
label: '备注',
component: 'Textarea',
componentProps: { maxLength: 250, placeholder: '请输入备注', rows: 2 },
},
];
}

View File

@ -0,0 +1,112 @@
<script lang="ts" setup>
import type { MesProRouteProductBomApi } from '#/api/mes/pro/route/productbom';
import { computed, ref } from 'vue';
import { useVbenModal } from '@vben/common-ui';
import { message } from 'ant-design-vue';
import { useVbenForm } from '#/adapter/form';
import {
createRouteProductBom,
updateRouteProductBom,
} from '#/api/mes/pro/route/productbom';
import { $t } from '#/locales';
import { useRouteProductBomFormSchema } from '../data';
const emit = defineEmits(['success']);
const formData = ref<MesProRouteProductBomApi.RouteProductBom>(); // BOM
const productId = ref<number>(0); // BOM BOM
const getTitle = computed(() =>
formData.value?.id
? $t('ui.actionTitle.edit', ['BOM 物料'])
: $t('ui.actionTitle.create', ['BOM 物料']),
);
/** MdProductBomSelect change 回调:把 BOM 物料编码/名称/规格/单位/默认用量回填到 form */
async function handleBomChange(bom?: any) {
if (!bom) {
return;
}
await formApi.setValues({
itemCode: bom.bomItemCode,
itemName: bom.bomItemName,
quantity: bom.quantity ?? 1,
specification: bom.specification,
unitName: bom.unitName,
});
}
const [Form, formApi] = useVbenForm({
commonConfig: {
componentProps: { class: 'w-full' },
formItemClass: 'col-span-1',
labelWidth: 110,
},
layout: 'horizontal',
schema: useRouteProductBomFormSchema(() => productId.value, handleBomChange),
showDefaultActions: false,
});
const [Modal, modalApi] = useVbenModal({
async onConfirm() {
const { valid } = await formApi.validate();
if (!valid) {
return;
}
modalApi.lock();
//
const data =
(await formApi.getValues()) as MesProRouteProductBomApi.RouteProductBom;
try {
await (formData.value?.id
? updateRouteProductBom(data)
: createRouteProductBom(data));
//
await modalApi.close();
emit('success');
message.success($t('ui.actionMessage.operationSuccess'));
} finally {
modalApi.unlock();
}
},
async onOpenChange(isOpen: boolean) {
if (!isOpen) {
formData.value = undefined;
return;
}
await formApi.resetForm();
//
const data = modalApi.getData<{
processId: number;
productId: number;
routeId: number;
row?: MesProRouteProductBomApi.RouteProductBom;
}>();
if (!data) {
return;
}
productId.value = data.productId;
if (data.row) {
formData.value = data.row;
// values
await formApi.setValues(data.row);
return;
}
await formApi.setValues({
processId: data.processId,
productId: data.productId,
quantity: 1,
routeId: data.routeId,
});
},
});
</script>
<template>
<Modal :title="getTitle" class="w-1/3">
<Form class="mx-4" />
</Modal>
</template>

View File

@ -48,8 +48,10 @@ const [Form, formApi] = useVbenForm({
wrapperClass: 'grid-cols-2', wrapperClass: 'grid-cols-2',
}); });
// TODO @AI
formApi.setState({ schema: useFormSchema(formApi) }); formApi.setState({ schema: useFormSchema(formApi) });
// TODO @AI
const [Modal, modalApi] = useVbenModal({ const [Modal, modalApi] = useVbenModal({
async onConfirm() { async onConfirm() {
if (isDetail.value) { if (isDetail.value) {

View File

@ -48,6 +48,7 @@ async function loadSchema(): Promise<VbenFormSchema[]> {
return useRouteProcessFormSchema(options); return useRouteProcessFormSchema(options);
} }
// TODO @AI
const [Modal, modalApi] = useVbenModal({ const [Modal, modalApi] = useVbenModal({
async onConfirm() { async onConfirm() {
const { valid } = await formApi.validate(); const { valid } = await formApi.validate();

View File

@ -45,6 +45,7 @@ const [Grid, gridApi] = useVbenVxeGrid({
} as VxeTableGridOptions<MesProRouteProcessApi.RouteProcess>, } as VxeTableGridOptions<MesProRouteProcessApi.RouteProcess>,
}); });
// TODO @AI
async function getList() { async function getList() {
gridApi.setLoading(true); gridApi.setLoading(true);
try { try {
@ -55,6 +56,7 @@ async function getList() {
} }
} }
// TODO @AI
function handleCreate() { function handleCreate() {
const maxSort = const maxSort =
list.value.length > 0 list.value.length > 0
@ -63,10 +65,12 @@ function handleCreate() {
processFormModalApi.setData({ maxSort, routeId: props.routeId }).open(); processFormModalApi.setData({ maxSort, routeId: props.routeId }).open();
} }
// TODO @AI
function handleEdit(row: MesProRouteProcessApi.RouteProcess) { function handleEdit(row: MesProRouteProcessApi.RouteProcess) {
processFormModalApi.setData({ id: row.id, routeId: props.routeId, row }).open(); processFormModalApi.setData({ id: row.id, routeId: props.routeId, row }).open();
} }
// TODO @AI
async function handleDelete(row: MesProRouteProcessApi.RouteProcess) { async function handleDelete(row: MesProRouteProcessApi.RouteProcess) {
await deleteRouteProcess(row.id!); await deleteRouteProcess(row.id!);
message.success($t('ui.actionMessage.deleteSuccess', ['工艺路线工序'])); message.success($t('ui.actionMessage.deleteSuccess', ['工艺路线工序']));

View File

@ -1,34 +1,23 @@
<script lang="ts" setup> <script lang="ts" setup>
import type { FormInstance } from 'ant-design-vue';
import type { VxeTableGridOptions } from '#/adapter/vxe-table'; import type { VxeTableGridOptions } from '#/adapter/vxe-table';
import type { MesProRouteProductBomApi } from '#/api/mes/pro/route/productbom'; import type { MesProRouteProductBomApi } from '#/api/mes/pro/route/productbom';
import { computed, reactive, ref, watch } from 'vue'; import { ref, watch } from 'vue';
import { import { useVbenModal } from '@vben/common-ui';
Form as AForm,
FormItem, import { message, TabPane, Tabs } from 'ant-design-vue';
InputNumber,
message,
Modal,
TabPane,
Tabs,
Textarea,
} from 'ant-design-vue';
import { TableAction, useVbenVxeGrid } from '#/adapter/vxe-table'; import { TableAction, useVbenVxeGrid } from '#/adapter/vxe-table';
import { getRouteProcessListByRoute } from '#/api/mes/pro/route/process'; import { getRouteProcessListByRoute } from '#/api/mes/pro/route/process';
import { import {
createRouteProductBom,
deleteRouteProductBom, deleteRouteProductBom,
getRouteProductBomList, getRouteProductBomList,
updateRouteProductBom,
} from '#/api/mes/pro/route/productbom'; } from '#/api/mes/pro/route/productbom';
import { $t } from '#/locales'; import { $t } from '#/locales';
import { MdProductBomSelect } from '#/views/mes/md/item/components';
import { useRouteProductBomGridColumns } from '../data'; import { useRouteProductBomGridColumns } from '../data';
import BomForm from './bom-form.vue';
const props = defineProps<{ const props = defineProps<{
productId: number; productId: number;
@ -36,27 +25,17 @@ const props = defineProps<{
routeId: number; routeId: number;
}>(); }>();
// TODO @AIconst vue agents md
const processOptions = ref< const processOptions = ref<
Array<{ processId: number; processName?: string }> Array<{ processId: number; processName?: string }>
>([]); >([]); // TODO @AIvue
const activeProcessId = ref<string>(''); const activeProcessId = ref<string>(''); // TODO @AI number vue3 + ep
const list = ref<MesProRouteProductBomApi.RouteProductBom[]>([]); const list = ref<MesProRouteProductBomApi.RouteProductBom[]>([]);
const formVisible = ref(false); const [BomFormModal, bomFormModalApi] = useVbenModal({
const formRef = ref<FormInstance>(); connectedComponent: BomForm,
const isUpdate = ref(false); destroyOnClose: true,
const formData = reactive<MesProRouteProductBomApi.RouteProductBom>({
quantity: 1,
}); });
const formRules = {
itemId: [{ message: 'BOM 物料不能为空', required: true }],
quantity: [{ message: '用料比例不能为空', required: true }],
};
const formTitle = computed(() =>
isUpdate.value
? $t('ui.actionTitle.edit', ['BOM 物料'])
: $t('ui.actionTitle.create', ['BOM 物料']),
);
const [Grid, gridApi] = useVbenVxeGrid({ const [Grid, gridApi] = useVbenVxeGrid({
gridOptions: { gridOptions: {
@ -72,6 +51,7 @@ const [Grid, gridApi] = useVbenVxeGrid({
} as VxeTableGridOptions<MesProRouteProductBomApi.RouteProductBom>, } as VxeTableGridOptions<MesProRouteProductBomApi.RouteProductBom>,
}); });
/** 加载路线下的工序列表,用于工序 Tab */
async function loadProcessList() { async function loadProcessList() {
const data = await getRouteProcessListByRoute(props.routeId); const data = await getRouteProcessListByRoute(props.routeId);
processOptions.value = (data || []).map((item) => ({ processOptions.value = (data || []).map((item) => ({
@ -88,6 +68,7 @@ async function loadProcessList() {
} }
} }
/** 加载当前工序下的 BOM 列表 */
async function getList() { async function getList() {
if (!activeProcessId.value) { if (!activeProcessId.value) {
return; return;
@ -105,70 +86,40 @@ async function getList() {
} }
} }
function resetForm() { /** 新增 BOM */
Object.assign(formData, {
id: undefined,
itemCode: undefined,
itemId: undefined,
itemName: undefined,
processId: Number(activeProcessId.value),
productId: props.productId,
quantity: 1,
remark: undefined,
routeId: props.routeId,
specification: undefined,
unitName: undefined,
});
formRef.value?.clearValidate();
}
function handleCreate() { function handleCreate() {
if (!activeProcessId.value) { if (!activeProcessId.value) {
message.warning('请先选择工序'); message.warning('请先选择工序');
return; return;
} }
resetForm(); bomFormModalApi
isUpdate.value = false; .setData({
formVisible.value = true; processId: Number(activeProcessId.value),
productId: props.productId,
routeId: props.routeId,
})
.open();
} }
/** 编辑 BOM */
function handleEdit(row: MesProRouteProductBomApi.RouteProductBom) { function handleEdit(row: MesProRouteProductBomApi.RouteProductBom) {
Object.assign(formData, row); bomFormModalApi
isUpdate.value = true; .setData({
formVisible.value = true; processId: Number(activeProcessId.value),
productId: props.productId,
routeId: props.routeId,
row,
})
.open();
} }
/** 删除 BOM */
async function handleDelete(row: MesProRouteProductBomApi.RouteProductBom) { async function handleDelete(row: MesProRouteProductBomApi.RouteProductBom) {
await deleteRouteProductBom(row.id!); await deleteRouteProductBom(row.id!);
message.success($t('ui.actionMessage.deleteSuccess', ['BOM 物料'])); message.success($t('ui.actionMessage.deleteSuccess', ['BOM 物料']));
await getList(); await getList();
} }
async function submitForm() {
try {
await formRef.value?.validate();
} catch {
return;
}
await (isUpdate.value
? updateRouteProductBom(formData)
: createRouteProductBom(formData));
message.success($t('ui.actionMessage.operationSuccess'));
formVisible.value = false;
await getList();
}
/** BOM 选中后回填用量比例 */
function handleBomChange(bom?: any) {
if (bom) {
formData.quantity = bom.quantity ?? 1;
formData.itemCode = bom.bomItemCode;
formData.itemName = bom.bomItemName;
formData.specification = bom.specification;
formData.unitName = bom.unitName;
}
}
watch( watch(
() => [props.routeId, props.productId], () => [props.routeId, props.productId],
() => { () => {
@ -181,6 +132,7 @@ watch(
</script> </script>
<template> <template>
<BomFormModal @success="getList" />
<Tabs v-model:active-key="activeProcessId" @change="getList"> <Tabs v-model:active-key="activeProcessId" @change="getList">
<TabPane <TabPane
v-for="item in processOptions" v-for="item in processOptions"
@ -222,44 +174,4 @@ watch(
/> />
</template> </template>
</Grid> </Grid>
<Modal
v-model:open="formVisible"
:title="formTitle"
width="500px"
@ok="submitForm"
>
<AForm
ref="formRef"
:label-col="{ span: 6 }"
:wrapper-col="{ span: 16 }"
:model="formData"
:rules="formRules"
>
<FormItem label="BOM 物料" name="itemId">
<MdProductBomSelect
v-model="formData.itemId"
:item-id="productId"
placeholder="请选择 BOM 物料"
@change="handleBomChange"
/>
</FormItem>
<FormItem label="用料比例" name="quantity">
<InputNumber
v-model:value="formData.quantity"
class="!w-full"
:min="0"
:precision="2"
/>
</FormItem>
<FormItem label="备注" name="remark">
<Textarea
v-model:value="formData.remark"
:max-length="250"
placeholder="请输入备注"
:rows="2"
/>
</FormItem>
</AForm>
</Modal>
</template> </template>

View File

@ -1,90 +1,73 @@
<script lang="ts" setup> <script lang="ts" setup>
import type { FormInstance } from 'ant-design-vue';
import type { MesProRouteProductApi } from '#/api/mes/pro/route/product'; import type { MesProRouteProductApi } from '#/api/mes/pro/route/product';
import { computed, reactive, ref } from 'vue'; import { computed, ref } from 'vue';
import { useVbenModal } from '@vben/common-ui'; import { useVbenModal } from '@vben/common-ui';
import { DICT_TYPE } from '@vben/constants';
import { getDictOptions } from '@vben/hooks';
import { import { Divider, message } from 'ant-design-vue';
Form as AForm,
Divider,
FormItem,
InputNumber,
message,
Select,
Textarea,
} from 'ant-design-vue';
import { useVbenForm } from '#/adapter/form';
import { import {
createRouteProduct, createRouteProduct,
updateRouteProduct, updateRouteProduct,
} from '#/api/mes/pro/route/product'; } from '#/api/mes/pro/route/product';
import { $t } from '#/locales'; import { $t } from '#/locales';
import { MdItemSelect } from '#/views/mes/md/item/components';
import { useRouteProductFormSchema } from '../data';
import ProductBomList from './product-bom-list.vue'; import ProductBomList from './product-bom-list.vue';
const emit = defineEmits(['success']); const emit = defineEmits(['success']);
const isUpdate = ref(false); const formData = ref<MesProRouteProductApi.RouteProduct>(); // /
const formRef = ref<FormInstance>();
const formData = reactive<MesProRouteProductApi.RouteProduct>({
productionTime: 1,
quantity: 1,
timeUnitType: 'MINUTE',
});
const formRules = {
itemId: [{ message: '产品不能为空', required: true }],
quantity: [{ message: '生产数量不能为空', required: true }],
};
const timeUnitOptions = computed(() =>
getDictOptions(DICT_TYPE.MES_TIME_UNIT_TYPE).map((item) => ({
label: item.label,
value: item.value as string,
})),
);
const getTitle = computed(() => const getTitle = computed(() =>
isUpdate.value formData.value?.id
? $t('ui.actionTitle.edit', ['工艺路线产品']) ? $t('ui.actionTitle.edit', ['工艺路线产品'])
: $t('ui.actionTitle.create', ['工艺路线产品']), : $t('ui.actionTitle.create', ['工艺路线产品']),
); );
/** 重置表单数据 */ /** MdItemSelect change 回调:把物料编码/名称/规格/单位回填到 form */
function resetForm(routeId?: number) { async function handleItemChange(item?: any) {
Object.assign(formData, { if (!item) {
id: undefined, return;
itemCode: undefined, }
itemId: undefined, await formApi.setValues({
itemName: undefined, itemCode: item.code,
productionTime: 1, itemName: item.name,
quantity: 1, specification: item.specification,
remark: undefined, unitName: item.unitMeasureName,
routeId: routeId ?? formData.routeId,
specification: undefined,
timeUnitType: 'MINUTE',
unitName: undefined,
}); });
formRef.value?.clearValidate();
} }
const [Form, formApi] = useVbenForm({
commonConfig: {
componentProps: { class: 'w-full' },
formItemClass: 'col-span-1',
labelWidth: 100,
},
layout: 'horizontal',
schema: useRouteProductFormSchema(handleItemChange),
showDefaultActions: false,
wrapperClass: 'grid-cols-2',
});
const [Modal, modalApi] = useVbenModal({ const [Modal, modalApi] = useVbenModal({
// TODO @AI
async onConfirm() { async onConfirm() {
try { const { valid } = await formApi.validate();
await formRef.value?.validate(); if (!valid) {
} catch {
return; return;
} }
modalApi.lock(); modalApi.lock();
//
const data =
(await formApi.getValues()) as MesProRouteProductApi.RouteProduct;
try { try {
if (isUpdate.value) { if (formData.value?.id) {
await updateRouteProduct(formData); await updateRouteProduct(data);
} else { } else {
const id = await createRouteProduct(formData); const id = await createRouteProduct(data);
formData.id = id; formData.value = { ...data, id };
isUpdate.value = true; await formApi.setFieldValue('id', id);
} }
emit('success'); emit('success');
message.success($t('ui.actionMessage.operationSuccess')); message.success($t('ui.actionMessage.operationSuccess'));
@ -92,12 +75,13 @@ const [Modal, modalApi] = useVbenModal({
modalApi.unlock(); modalApi.unlock();
} }
}, },
onOpenChange(isOpen: boolean) { async onOpenChange(isOpen: boolean) {
if (!isOpen) { if (!isOpen) {
resetForm(); formData.value = undefined;
isUpdate.value = false;
return; return;
} }
await formApi.resetForm();
//
const data = modalApi.getData<{ const data = modalApi.getData<{
id?: number; id?: number;
routeId: number; routeId: number;
@ -107,79 +91,21 @@ const [Modal, modalApi] = useVbenModal({
return; return;
} }
if (data.row) { if (data.row) {
Object.assign(formData, data.row); formData.value = data.row;
isUpdate.value = true; // values
} else { await formApi.setValues(data.row);
resetForm(data.routeId); return;
isUpdate.value = false;
} }
await formApi.setValues({ routeId: data.routeId });
}, },
}); });
/** 物料编号变化时回填名称等 */
function handleItemChange(item?: any) {
if (!item) {
return;
}
formData.itemCode = item.code;
formData.itemName = item.name;
formData.specification = item.specification;
formData.unitName = item.unitName;
}
</script> </script>
<template> <template>
<Modal :title="getTitle" class="w-3/5"> <Modal :title="getTitle" class="w-3/5">
<AForm <Form class="mx-4" />
ref="formRef" <!-- 编辑/详情模式下展示产品 BOM 子表新增模式下隐藏 -->
class="mx-4" <template v-if="formData?.id && formData?.itemId">
:label-col="{ flex: '0 0 110px' }"
:wrapper-col="{ flex: 'auto' }"
:model="formData"
:rules="formRules"
>
<div class="grid grid-cols-2 gap-x-4">
<FormItem class="col-span-2" label="产品" name="itemId">
<MdItemSelect
v-model="formData.itemId"
@change="handleItemChange"
/>
</FormItem>
<FormItem label="生产数量" name="quantity">
<InputNumber
v-model:value="formData.quantity"
class="!w-full"
:min="1"
:precision="0"
/>
</FormItem>
<FormItem label="生产用时" name="productionTime">
<InputNumber
v-model:value="formData.productionTime"
class="!w-full"
:min="0"
:precision="2"
/>
</FormItem>
<FormItem label="时间单位" name="timeUnitType">
<Select
v-model:value="formData.timeUnitType"
allow-clear
:options="timeUnitOptions"
placeholder="请选择"
/>
</FormItem>
<FormItem class="col-span-2" label="备注" name="remark">
<Textarea
v-model:value="formData.remark"
:max-length="250"
placeholder="请输入备注"
:rows="2"
/>
</FormItem>
</div>
</AForm>
<template v-if="isUpdate && formData.id && formData.itemId">
<Divider class="!my-3" orientation="left">产品 BOM 配置</Divider> <Divider class="!my-3" orientation="left">产品 BOM 配置</Divider>
<div class="mx-4"> <div class="mx-4">
<ProductBomList <ProductBomList

View File

@ -33,6 +33,7 @@ const [ProductFormModal, productFormModalApi] = useVbenModal({
destroyOnClose: true, destroyOnClose: true,
}); });
// TODO @AIps
const [Grid, gridApi] = useVbenVxeGrid({ const [Grid, gridApi] = useVbenVxeGrid({
gridOptions: { gridOptions: {
autoResize: true, autoResize: true,
@ -47,6 +48,7 @@ const [Grid, gridApi] = useVbenVxeGrid({
} as VxeTableGridOptions<MesProRouteProductApi.RouteProduct>, } as VxeTableGridOptions<MesProRouteProductApi.RouteProduct>,
}); });
// TODO @AI
async function getList() { async function getList() {
gridApi.setLoading(true); gridApi.setLoading(true);
try { try {
@ -57,14 +59,17 @@ async function getList() {
} }
} }
// TODO @AI
function handleCreate() { function handleCreate() {
productFormModalApi.setData({ routeId: props.routeId }).open(); productFormModalApi.setData({ routeId: props.routeId }).open();
} }
// TODO @AI
function handleEdit(row: MesProRouteProductApi.RouteProduct) { function handleEdit(row: MesProRouteProductApi.RouteProduct) {
productFormModalApi.setData({ id: row.id, routeId: props.routeId, row }).open(); productFormModalApi.setData({ id: row.id, routeId: props.routeId, row }).open();
} }
// TODO @AI
async function handleDelete(row: MesProRouteProductApi.RouteProduct) { async function handleDelete(row: MesProRouteProductApi.RouteProduct) {
await deleteRouteProduct(row.id!); await deleteRouteProduct(row.id!);
message.success($t('ui.actionMessage.deleteSuccess', ['工艺路线产品'])); message.success($t('ui.actionMessage.deleteSuccess', ['工艺路线产品']));