|
|
|
|
@ -1,343 +1,410 @@
|
|
|
|
|
<template>
|
|
|
|
|
<div class="stock-out-page">
|
|
|
|
|
<div class="dv-repair-page">
|
|
|
|
|
<template v-if="!formVisible">
|
|
|
|
|
<ContentWrap>
|
|
|
|
|
<!-- 搜索工作栏 -->
|
|
|
|
|
<el-form
|
|
|
|
|
class="-mb-15px"
|
|
|
|
|
:model="queryParams"
|
|
|
|
|
ref="queryFormRef"
|
|
|
|
|
:inline="true"
|
|
|
|
|
label-width="auto"
|
|
|
|
|
>
|
|
|
|
|
<el-form-item :label="t('ErpStock.Out.no')" prop="no">
|
|
|
|
|
<el-input
|
|
|
|
|
v-model="queryParams.no"
|
|
|
|
|
:placeholder="t('ErpStock.Out.placeholderNo')"
|
|
|
|
|
clearable
|
|
|
|
|
@keyup.enter="handleQuery"
|
|
|
|
|
class="!w-240px"
|
|
|
|
|
/>
|
|
|
|
|
</el-form-item>
|
|
|
|
|
<el-form-item :label="t('ErpStock.In.product')" prop="productId">
|
|
|
|
|
<el-select
|
|
|
|
|
v-model="queryParams.productId"
|
|
|
|
|
clearable
|
|
|
|
|
filterable
|
|
|
|
|
:placeholder="t('ErpStock.In.placeholderProduct')"
|
|
|
|
|
class="!w-240px"
|
|
|
|
|
>
|
|
|
|
|
<el-option
|
|
|
|
|
v-for="item in productList"
|
|
|
|
|
:key="item.id"
|
|
|
|
|
:label="item.name"
|
|
|
|
|
:value="item.id"
|
|
|
|
|
/>
|
|
|
|
|
</el-select>
|
|
|
|
|
</el-form-item>
|
|
|
|
|
<el-form-item :label="t('ErpStock.Out.outTime')" prop="outTime">
|
|
|
|
|
<el-date-picker
|
|
|
|
|
v-model="queryParams.outTime"
|
|
|
|
|
value-format="YYYY-MM-DD HH:mm:ss"
|
|
|
|
|
type="daterange"
|
|
|
|
|
:start-placeholder="t('common.startTimeText')"
|
|
|
|
|
:end-placeholder="t('common.endTimeText')"
|
|
|
|
|
:default-time="[new Date('1 00:00:00'), new Date('1 23:59:59')]"
|
|
|
|
|
class="!w-240px"
|
|
|
|
|
/>
|
|
|
|
|
</el-form-item>
|
|
|
|
|
<!-- <el-form-item label="客户" prop="customerId">
|
|
|
|
|
<el-select
|
|
|
|
|
v-model="queryParams.customerId"
|
|
|
|
|
clearable
|
|
|
|
|
filterable
|
|
|
|
|
placeholder="请选择客户"
|
|
|
|
|
class="!w-240px"
|
|
|
|
|
<!-- 鎼滅储宸ヤ綔鏍?-->
|
|
|
|
|
<el-form
|
|
|
|
|
class="-mb-15px"
|
|
|
|
|
:model="queryParams"
|
|
|
|
|
ref="queryFormRef"
|
|
|
|
|
:inline="true"
|
|
|
|
|
label-width="auto"
|
|
|
|
|
>
|
|
|
|
|
<el-option
|
|
|
|
|
v-for="item in customerList"
|
|
|
|
|
:key="item.id"
|
|
|
|
|
:label="item.name"
|
|
|
|
|
:value="item.id"
|
|
|
|
|
<el-form-item :label="t('ErpStock.Out.no')" prop="no">
|
|
|
|
|
<el-input
|
|
|
|
|
v-model="queryParams.no"
|
|
|
|
|
:placeholder="t('ErpStock.Out.placeholderNo')"
|
|
|
|
|
clearable
|
|
|
|
|
@keyup.enter="handleQuery"
|
|
|
|
|
class="!w-240px"
|
|
|
|
|
/>
|
|
|
|
|
</el-form-item>
|
|
|
|
|
<el-form-item :label="t('ErpStock.Out.product')" prop="productId">
|
|
|
|
|
<el-select
|
|
|
|
|
v-model="queryParams.productId"
|
|
|
|
|
clearable
|
|
|
|
|
filterable
|
|
|
|
|
:placeholder="t('ErpStock.Out.placeholderProduct')"
|
|
|
|
|
class="!w-240px"
|
|
|
|
|
>
|
|
|
|
|
<el-option
|
|
|
|
|
v-for="item in productList"
|
|
|
|
|
:key="item.id"
|
|
|
|
|
:label="item.name"
|
|
|
|
|
:value="item.id"
|
|
|
|
|
/>
|
|
|
|
|
</el-select>
|
|
|
|
|
</el-form-item>
|
|
|
|
|
<el-form-item :label="t('ErpStock.Out.outTime')" prop="outTime">
|
|
|
|
|
<el-date-picker
|
|
|
|
|
v-model="queryParams.outTime"
|
|
|
|
|
value-format="YYYY-MM-DD HH:mm:ss"
|
|
|
|
|
type="daterange"
|
|
|
|
|
:start-placeholder="t('common.startTimeText')"
|
|
|
|
|
:end-placeholder="t('common.endTimeText')"
|
|
|
|
|
:default-time="[new Date('1 00:00:00'), new Date('1 23:59:59')]"
|
|
|
|
|
class="!w-220px"
|
|
|
|
|
/>
|
|
|
|
|
</el-form-item>
|
|
|
|
|
<el-form-item :label="t('ErpStock.Out.warehouse')" prop="warehouseId" v-show="showAllFilters">
|
|
|
|
|
<el-select
|
|
|
|
|
v-model="queryParams.warehouseId"
|
|
|
|
|
clearable
|
|
|
|
|
filterable
|
|
|
|
|
:placeholder="t('ErpStock.Out.placeholderWarehouse')"
|
|
|
|
|
class="!w-240px"
|
|
|
|
|
>
|
|
|
|
|
<el-option
|
|
|
|
|
v-for="item in warehouseList"
|
|
|
|
|
:key="item.id"
|
|
|
|
|
:label="item.name"
|
|
|
|
|
:value="item.id"
|
|
|
|
|
/>
|
|
|
|
|
</el-select>
|
|
|
|
|
</el-form-item>
|
|
|
|
|
<el-form-item :label="t('ErpStock.Out.creator')" prop="creator" v-show="showAllFilters">
|
|
|
|
|
<el-select
|
|
|
|
|
v-model="queryParams.creator"
|
|
|
|
|
clearable
|
|
|
|
|
filterable
|
|
|
|
|
:placeholder="t('ErpStock.Out.placeholderCreator')"
|
|
|
|
|
class="!w-240px"
|
|
|
|
|
>
|
|
|
|
|
<el-option
|
|
|
|
|
v-for="item in userList"
|
|
|
|
|
:key="item.id"
|
|
|
|
|
:label="item.nickname"
|
|
|
|
|
:value="item.id"
|
|
|
|
|
/>
|
|
|
|
|
</el-select>
|
|
|
|
|
</el-form-item>
|
|
|
|
|
<el-form-item :label="t('ErpStock.Out.status')" prop="status" v-show="showAllFilters">
|
|
|
|
|
<el-select
|
|
|
|
|
v-model="queryParams.status"
|
|
|
|
|
:placeholder="t('ErpStock.Out.placeholderStatus')"
|
|
|
|
|
clearable
|
|
|
|
|
class="!w-240px"
|
|
|
|
|
>
|
|
|
|
|
<el-option
|
|
|
|
|
v-for="dict in getIntDictOptions(DICT_TYPE.ERP_AUDIT_STATUS)"
|
|
|
|
|
:key="dict.value"
|
|
|
|
|
:label="dict.label"
|
|
|
|
|
:value="dict.value"
|
|
|
|
|
/>
|
|
|
|
|
</el-select>
|
|
|
|
|
</el-form-item>
|
|
|
|
|
<el-form-item :label="t('ErpStock.Out.remark')" prop="remark" v-show="showAllFilters">
|
|
|
|
|
<el-input
|
|
|
|
|
v-model="queryParams.remark"
|
|
|
|
|
:placeholder="t('ErpStock.Out.placeholderRemark')"
|
|
|
|
|
clearable
|
|
|
|
|
@keyup.enter="handleQuery"
|
|
|
|
|
class="!w-240px"
|
|
|
|
|
/>
|
|
|
|
|
</el-form-item>
|
|
|
|
|
<el-form-item v-if="filterCount > 3">
|
|
|
|
|
<el-button type="text" class="text-primary" @click="toggleFilters">
|
|
|
|
|
<Icon :icon="showAllFilters ? 'ep:arrow-up' : 'ep:arrow-down'" class="mr-5px" />
|
|
|
|
|
{{ showAllFilters ? t('FactoryModeling.FactoryStructure.collapseText') :
|
|
|
|
|
t('FactoryModeling.FactoryStructure.expandText') }}
|
|
|
|
|
</el-button>
|
|
|
|
|
</el-form-item>
|
|
|
|
|
<el-form-item>
|
|
|
|
|
<el-button @click="handleQuery"><Icon icon="ep:search" class="mr-5px" /> {{ t('common.query') }}</el-button>
|
|
|
|
|
<el-button @click="resetQuery"><Icon icon="ep:refresh" class="mr-5px" /> {{ t('common.reset') }}</el-button>
|
|
|
|
|
<el-button
|
|
|
|
|
type="primary"
|
|
|
|
|
plain
|
|
|
|
|
@click="openForm('create')"
|
|
|
|
|
v-hasPermi="['erp:stock-out:create']"
|
|
|
|
|
>
|
|
|
|
|
<Icon icon="ep:plus" class="mr-5px" /> {{ t('action.add') }}
|
|
|
|
|
</el-button>
|
|
|
|
|
<el-button
|
|
|
|
|
type="success"
|
|
|
|
|
plain
|
|
|
|
|
@click="handleExport"
|
|
|
|
|
:loading="exportLoading"
|
|
|
|
|
v-hasPermi="['erp:stock-out:export']"
|
|
|
|
|
>
|
|
|
|
|
<Icon icon="ep:download" class="mr-5px" /> {{ t('action.export') }}
|
|
|
|
|
</el-button>
|
|
|
|
|
<el-button
|
|
|
|
|
type="danger"
|
|
|
|
|
plain
|
|
|
|
|
@click="handleDelete(selectionList.map((item) => item.id))"
|
|
|
|
|
v-hasPermi="['erp:stock-out:delete']"
|
|
|
|
|
:disabled="selectionList.length === 0"
|
|
|
|
|
>
|
|
|
|
|
<Icon icon="ep:delete" class="mr-5px" /> {{ t('action.del') }}
|
|
|
|
|
</el-button>
|
|
|
|
|
</el-form-item>
|
|
|
|
|
</el-form>
|
|
|
|
|
</ContentWrap>
|
|
|
|
|
|
|
|
|
|
<!-- 鍒楄〃 -->
|
|
|
|
|
<ContentWrap>
|
|
|
|
|
<el-tabs v-model="activeName" @tab-click="handleTabClick">
|
|
|
|
|
<el-tab-pane
|
|
|
|
|
v-for="dict in stockOutTypeOptions"
|
|
|
|
|
:key="dict.value"
|
|
|
|
|
:label="dict.label"
|
|
|
|
|
:name="dict.value"
|
|
|
|
|
/>
|
|
|
|
|
</el-select>
|
|
|
|
|
</el-form-item> -->
|
|
|
|
|
<el-form-item :label="t('ErpStock.Out.warehouse')" prop="warehouseId" v-show="showAllFilters">
|
|
|
|
|
<el-select
|
|
|
|
|
v-model="queryParams.warehouseId"
|
|
|
|
|
clearable
|
|
|
|
|
filterable
|
|
|
|
|
:placeholder="t('ErpStock.In.placeholderWarehouse')"
|
|
|
|
|
class="!w-240px"
|
|
|
|
|
</el-tabs>
|
|
|
|
|
|
|
|
|
|
<el-table
|
|
|
|
|
v-loading="loading"
|
|
|
|
|
:data="list"
|
|
|
|
|
:stripe="true"
|
|
|
|
|
:show-overflow-tooltip="true"
|
|
|
|
|
@selection-change="handleSelectionChange"
|
|
|
|
|
>
|
|
|
|
|
<el-option
|
|
|
|
|
v-for="item in warehouseList"
|
|
|
|
|
:key="item.id"
|
|
|
|
|
:label="item.name"
|
|
|
|
|
:value="item.id"
|
|
|
|
|
<el-table-column width="30" :label="t('common.select')" type="selection" />
|
|
|
|
|
<el-table-column
|
|
|
|
|
min-width="180"
|
|
|
|
|
:label="t('ErpStock.Out.no')"
|
|
|
|
|
align="center"
|
|
|
|
|
prop="no"
|
|
|
|
|
sortable />
|
|
|
|
|
<el-table-column
|
|
|
|
|
:label="t(`ErpStock.Out.${outTypeToInfoKey(queryParams.outType)}`)"
|
|
|
|
|
align="left"
|
|
|
|
|
sortable
|
|
|
|
|
prop="productNames"
|
|
|
|
|
min-width="200"
|
|
|
|
|
/>
|
|
|
|
|
</el-select>
|
|
|
|
|
</el-form-item>
|
|
|
|
|
<el-form-item :label="t('ErpStock.In.creator')" prop="creator" v-show="showAllFilters">
|
|
|
|
|
<el-select
|
|
|
|
|
v-model="queryParams.creator"
|
|
|
|
|
clearable
|
|
|
|
|
filterable
|
|
|
|
|
:placeholder="t('ErpStock.In.placeholderCreator')"
|
|
|
|
|
class="!w-240px"
|
|
|
|
|
>
|
|
|
|
|
<el-option
|
|
|
|
|
v-for="item in userList"
|
|
|
|
|
:key="item.id"
|
|
|
|
|
:label="item.nickname"
|
|
|
|
|
:value="item.id"
|
|
|
|
|
<el-table-column
|
|
|
|
|
:label="t('ErpStock.Out.outTime')"
|
|
|
|
|
align="center"
|
|
|
|
|
prop="outTime"
|
|
|
|
|
:formatter="dateFormatter2"
|
|
|
|
|
width="120px"
|
|
|
|
|
sortable />
|
|
|
|
|
<el-table-column :label="t('ErpStock.Out.creator')" align="center" prop="creatorName" sortable />
|
|
|
|
|
<el-table-column
|
|
|
|
|
:label="t('ErpStock.Out.count')"
|
|
|
|
|
align="right"
|
|
|
|
|
sortable
|
|
|
|
|
prop="totalCount"
|
|
|
|
|
:formatter="erpCountTableColumnFormatter"
|
|
|
|
|
/>
|
|
|
|
|
</el-select>
|
|
|
|
|
</el-form-item>
|
|
|
|
|
<el-form-item :label="t('ErpStock.In.status')" prop="status" v-show="showAllFilters">
|
|
|
|
|
<el-select
|
|
|
|
|
v-model="queryParams.status"
|
|
|
|
|
:placeholder="t('ErpStock.In.placeholderStatus')"
|
|
|
|
|
clearable
|
|
|
|
|
class="!w-240px"
|
|
|
|
|
>
|
|
|
|
|
<el-option
|
|
|
|
|
v-for="dict in getIntDictOptions(DICT_TYPE.ERP_AUDIT_STATUS)"
|
|
|
|
|
:key="dict.value"
|
|
|
|
|
:label="dict.label"
|
|
|
|
|
:value="dict.value"
|
|
|
|
|
<el-table-column
|
|
|
|
|
:label="t('ErpStock.Out.status')"
|
|
|
|
|
align="center"
|
|
|
|
|
fixed="right"
|
|
|
|
|
width="100"
|
|
|
|
|
prop="status"
|
|
|
|
|
sortable>
|
|
|
|
|
<template #default="scope">
|
|
|
|
|
<dict-tag :type="DICT_TYPE.ERP_AUDIT_STATUS" :value="scope.row.status" />
|
|
|
|
|
</template>
|
|
|
|
|
</el-table-column>
|
|
|
|
|
<el-table-column
|
|
|
|
|
:label="t('ErpStock.Out.auditUser')"
|
|
|
|
|
align="right"
|
|
|
|
|
sortable
|
|
|
|
|
prop="auditUserName"
|
|
|
|
|
/>
|
|
|
|
|
</el-select>
|
|
|
|
|
</el-form-item>
|
|
|
|
|
<el-form-item :label="t('ErpStock.Out.remark')" prop="remark" v-show="showAllFilters">
|
|
|
|
|
<el-input
|
|
|
|
|
v-model="queryParams.remark"
|
|
|
|
|
:placeholder="t('ErpStock.Out.placeholderRemark')"
|
|
|
|
|
clearable
|
|
|
|
|
@keyup.enter="handleQuery"
|
|
|
|
|
class="!w-240px"
|
|
|
|
|
<el-table-column :label="t('common.operate')" align="center" fixed="right" width="300">
|
|
|
|
|
<template #default="scope">
|
|
|
|
|
<el-button
|
|
|
|
|
link
|
|
|
|
|
@click="openForm('detail', scope.row.id)"
|
|
|
|
|
v-hasPermi="['erp:stock-out:query']"
|
|
|
|
|
>
|
|
|
|
|
{{ t('action.detail') }}
|
|
|
|
|
</el-button>
|
|
|
|
|
<el-button
|
|
|
|
|
link
|
|
|
|
|
type="primary"
|
|
|
|
|
@click="openForm('update', scope.row.id)"
|
|
|
|
|
v-hasPermi="['erp:stock-out:update']"
|
|
|
|
|
:disabled="scope.row.status === 20"
|
|
|
|
|
>
|
|
|
|
|
{{ t('action.edit') }}
|
|
|
|
|
</el-button>
|
|
|
|
|
<el-button
|
|
|
|
|
v-if="[0, 1].includes(Number(scope.row.status))"
|
|
|
|
|
link
|
|
|
|
|
type="primary"
|
|
|
|
|
@click="openSubmitDialog(scope.row)"
|
|
|
|
|
v-hasPermi="['erp:stock-out:update-status']"
|
|
|
|
|
>
|
|
|
|
|
{{ t('ErpStock.Out.submitAudit') }}
|
|
|
|
|
</el-button>
|
|
|
|
|
<el-button
|
|
|
|
|
v-if="Number(scope.row.status) === 10"
|
|
|
|
|
link
|
|
|
|
|
type="primary"
|
|
|
|
|
@click="openAuditDialog(scope.row, 20)"
|
|
|
|
|
v-hasPermi="['erp:stock-out:update-status']"
|
|
|
|
|
>
|
|
|
|
|
{{ t('ErpStock.Out.auditApprove') }}
|
|
|
|
|
</el-button>
|
|
|
|
|
<el-button
|
|
|
|
|
v-if="Number(scope.row.status) === 10"
|
|
|
|
|
link
|
|
|
|
|
type="danger"
|
|
|
|
|
@click="openAuditDialog(scope.row, 1)"
|
|
|
|
|
v-hasPermi="['erp:stock-out:update-status']"
|
|
|
|
|
>
|
|
|
|
|
{{ t('ErpStock.Out.auditReject') }}
|
|
|
|
|
</el-button>
|
|
|
|
|
<el-button
|
|
|
|
|
link
|
|
|
|
|
type="danger"
|
|
|
|
|
@click="handleDelete([scope.row.id])"
|
|
|
|
|
v-hasPermi="['erp:stock-out:delete']"
|
|
|
|
|
>
|
|
|
|
|
{{ t('action.del') }}
|
|
|
|
|
</el-button>
|
|
|
|
|
</template>
|
|
|
|
|
</el-table-column>
|
|
|
|
|
</el-table>
|
|
|
|
|
<!-- 鍒嗛〉 -->
|
|
|
|
|
<Pagination
|
|
|
|
|
:total="total"
|
|
|
|
|
v-model:page="queryParams.pageNo"
|
|
|
|
|
v-model:limit="queryParams.pageSize"
|
|
|
|
|
@pagination="getList"
|
|
|
|
|
/>
|
|
|
|
|
</el-form-item>
|
|
|
|
|
<el-form-item v-if="filterCount > 3">
|
|
|
|
|
<el-button type="text" class="text-primary" @click="toggleFilters">
|
|
|
|
|
<Icon :icon="showAllFilters ? 'ep:arrow-up' : 'ep:arrow-down'" class="mr-5px" />
|
|
|
|
|
{{ showAllFilters ? t('FactoryModeling.FactoryStructure.collapseText') :
|
|
|
|
|
t('FactoryModeling.FactoryStructure.expandText') }}
|
|
|
|
|
</el-button>
|
|
|
|
|
</el-form-item>
|
|
|
|
|
<el-form-item>
|
|
|
|
|
<el-button @click="handleQuery"><Icon icon="ep:search" class="mr-5px" /> {{ t('common.query') }}</el-button>
|
|
|
|
|
<el-button @click="resetQuery"><Icon icon="ep:refresh" class="mr-5px" /> {{ t('common.reset') }}</el-button>
|
|
|
|
|
<el-button
|
|
|
|
|
type="primary"
|
|
|
|
|
plain
|
|
|
|
|
@click="openForm('create')"
|
|
|
|
|
v-hasPermi="['erp:stock-out:create']"
|
|
|
|
|
>
|
|
|
|
|
<Icon icon="ep:plus" class="mr-5px" /> {{ t('action.add') }}
|
|
|
|
|
</el-button>
|
|
|
|
|
<el-button
|
|
|
|
|
type="success"
|
|
|
|
|
plain
|
|
|
|
|
@click="handleExport"
|
|
|
|
|
:loading="exportLoading"
|
|
|
|
|
v-hasPermi="['erp:stock-out:export']"
|
|
|
|
|
>
|
|
|
|
|
<Icon icon="ep:download" class="mr-5px" /> {{ t('action.export') }}
|
|
|
|
|
</el-button>
|
|
|
|
|
<el-button
|
|
|
|
|
type="danger"
|
|
|
|
|
plain
|
|
|
|
|
@click="handleDelete(selectionList.map((item) => item.id))"
|
|
|
|
|
v-hasPermi="['erp:stock-out:delete']"
|
|
|
|
|
:disabled="selectionList.length === 0"
|
|
|
|
|
>
|
|
|
|
|
<Icon icon="ep:delete" class="mr-5px" /> {{ t('action.del') }}
|
|
|
|
|
</el-button>
|
|
|
|
|
</el-form-item>
|
|
|
|
|
</el-form>
|
|
|
|
|
</ContentWrap>
|
|
|
|
|
</ContentWrap>
|
|
|
|
|
</template>
|
|
|
|
|
|
|
|
|
|
<!-- 列表 -->
|
|
|
|
|
<ContentWrap>
|
|
|
|
|
<el-tabs v-model="activeName" @tab-click="handleTabClick">
|
|
|
|
|
<el-tab-pane :label="t('ErpStock.Out.tabPick')" name="领料出库" />
|
|
|
|
|
<!-- <el-tab-pane :label="t('ErpStock.Out.tabProduct')" name="产品出库" /> -->
|
|
|
|
|
<!-- <el-tab-pane :label="t('ErpStock.Out.tabMaterial')" name="原料出库" /> -->
|
|
|
|
|
<el-tab-pane :label="t('ErpStock.Out.tabPart')" name="备件出库" />
|
|
|
|
|
<el-tab-pane :label="t('ErpStock.Out.tabOther')" name="其他出库" />
|
|
|
|
|
</el-tabs>
|
|
|
|
|
<!-- 琛ㄥ崟闈㈡澘锛氭坊鍔?淇敼 -->
|
|
|
|
|
<StockOutForm v-else ref="formRef" @success="getList" @closed="formVisible = false" />
|
|
|
|
|
|
|
|
|
|
<el-table
|
|
|
|
|
v-loading="loading"
|
|
|
|
|
:data="list"
|
|
|
|
|
:stripe="true"
|
|
|
|
|
:show-overflow-tooltip="true"
|
|
|
|
|
@selection-change="handleSelectionChange"
|
|
|
|
|
>
|
|
|
|
|
<el-table-column width="30" :label="t('common.select')" type="selection" />
|
|
|
|
|
<el-table-column
|
|
|
|
|
min-width="180"
|
|
|
|
|
:label="t('ErpStock.Out.no')"
|
|
|
|
|
align="center"
|
|
|
|
|
prop="no"
|
|
|
|
|
sortable />
|
|
|
|
|
<el-table-column
|
|
|
|
|
:label="t('ErpStock.In.productInfo')"
|
|
|
|
|
align="left"
|
|
|
|
|
sortable
|
|
|
|
|
prop="productNames"
|
|
|
|
|
min-width="180"
|
|
|
|
|
/>
|
|
|
|
|
<el-table-column label="生产计划编码" align="center" prop="planCode" min-width="160" sortable>
|
|
|
|
|
<template #default="scope">
|
|
|
|
|
<span>{{ scope.row.planCode ? scope.row.planCode : '-' }}</span>
|
|
|
|
|
</template>
|
|
|
|
|
</el-table-column>
|
|
|
|
|
<!-- <el-table-column label="客户" align="center" prop="customerName" sortable /> -->
|
|
|
|
|
<el-table-column
|
|
|
|
|
:label="t('ErpStock.Out.outTime')"
|
|
|
|
|
align="center"
|
|
|
|
|
sortable
|
|
|
|
|
prop="outTime"
|
|
|
|
|
:formatter="dateFormatter2"
|
|
|
|
|
width="120px"
|
|
|
|
|
/>
|
|
|
|
|
<el-table-column :label="t('ErpStock.In.creator')" align="center" prop="creatorName" sortable />
|
|
|
|
|
<el-table-column
|
|
|
|
|
:label="t('ErpStock.In.count')"
|
|
|
|
|
align="right"
|
|
|
|
|
sortable
|
|
|
|
|
prop="totalCount"
|
|
|
|
|
:formatter="erpCountTableColumnFormatter"
|
|
|
|
|
/>
|
|
|
|
|
<!-- <el-table-column
|
|
|
|
|
label="金额"
|
|
|
|
|
align="right"
|
|
|
|
|
sortable
|
|
|
|
|
prop="totalPrice"
|
|
|
|
|
:formatter="erpPriceTableColumnFormatter"
|
|
|
|
|
/> -->
|
|
|
|
|
<el-table-column
|
|
|
|
|
:label="t('ErpStock.In.status')"
|
|
|
|
|
align="center"
|
|
|
|
|
fixed="right"
|
|
|
|
|
width="90"
|
|
|
|
|
prop="status"
|
|
|
|
|
sortable>
|
|
|
|
|
<template #default="scope">
|
|
|
|
|
<dict-tag :type="DICT_TYPE.ERP_AUDIT_STATUS" :value="scope.row.status" />
|
|
|
|
|
</template>
|
|
|
|
|
</el-table-column>
|
|
|
|
|
<el-table-column :label="t('common.operate')" align="center" fixed="right" width="220">
|
|
|
|
|
<template #default="scope">
|
|
|
|
|
<el-button
|
|
|
|
|
link
|
|
|
|
|
@click="openForm('detail', scope.row.id)"
|
|
|
|
|
v-hasPermi="['erp:stock-out:query']"
|
|
|
|
|
>
|
|
|
|
|
{{ t('action.detail') }}
|
|
|
|
|
</el-button>
|
|
|
|
|
<el-button
|
|
|
|
|
link
|
|
|
|
|
type="primary"
|
|
|
|
|
@click="openForm('update', scope.row.id)"
|
|
|
|
|
v-hasPermi="['erp:stock-out:update']"
|
|
|
|
|
:disabled="scope.row.status === 20"
|
|
|
|
|
>
|
|
|
|
|
{{ t('action.edit') }}
|
|
|
|
|
</el-button>
|
|
|
|
|
<el-button
|
|
|
|
|
link
|
|
|
|
|
type="primary"
|
|
|
|
|
@click="handleUpdateStatus(scope.row.id, 20)"
|
|
|
|
|
v-hasPermi="['erp:stock-out:update-status']"
|
|
|
|
|
v-if="scope.row.status === 10"
|
|
|
|
|
>
|
|
|
|
|
{{ t('action.approve') }}
|
|
|
|
|
</el-button>
|
|
|
|
|
<el-button
|
|
|
|
|
link
|
|
|
|
|
type="danger"
|
|
|
|
|
@click="handleUpdateStatus(scope.row.id, 10)"
|
|
|
|
|
v-hasPermi="['erp:stock-out:update-status']"
|
|
|
|
|
v-else
|
|
|
|
|
<Dialog :title="t('ErpStock.Out.submitAudit')" v-model="submitDialogVisible" width="520px">
|
|
|
|
|
<el-form ref="submitFormRef" :model="submitFormData" :rules="submitFormRules" label-width="90px">
|
|
|
|
|
<el-form-item :label="t('ErpStock.Out.auditUser')" prop="auditUserId">
|
|
|
|
|
<el-select
|
|
|
|
|
v-model="submitFormData.auditUserId"
|
|
|
|
|
clearable
|
|
|
|
|
filterable
|
|
|
|
|
:placeholder="t('ErpStock.Out.placeholderAuditUser')"
|
|
|
|
|
class="!w-1/1"
|
|
|
|
|
>
|
|
|
|
|
{{ t('action.unapprove') }}
|
|
|
|
|
</el-button>
|
|
|
|
|
<el-button
|
|
|
|
|
link
|
|
|
|
|
type="danger"
|
|
|
|
|
@click="handleDelete([scope.row.id])"
|
|
|
|
|
v-hasPermi="['erp:stock-out:delete']"
|
|
|
|
|
>
|
|
|
|
|
{{ t('action.del') }}
|
|
|
|
|
</el-button>
|
|
|
|
|
</template>
|
|
|
|
|
</el-table-column>
|
|
|
|
|
</el-table>
|
|
|
|
|
<!-- 分页 -->
|
|
|
|
|
<Pagination
|
|
|
|
|
:total="total"
|
|
|
|
|
v-model:page="queryParams.pageNo"
|
|
|
|
|
v-model:limit="queryParams.pageSize"
|
|
|
|
|
@pagination="getList"
|
|
|
|
|
/>
|
|
|
|
|
</ContentWrap>
|
|
|
|
|
</template>
|
|
|
|
|
<el-option
|
|
|
|
|
v-for="item in userList"
|
|
|
|
|
:key="String(item.id)"
|
|
|
|
|
:label="item.nickname"
|
|
|
|
|
:value="String(item.id)"
|
|
|
|
|
/>
|
|
|
|
|
</el-select>
|
|
|
|
|
</el-form-item>
|
|
|
|
|
<el-form-item :label="t('ErpStock.Out.remark')" prop="remark">
|
|
|
|
|
<el-input
|
|
|
|
|
v-model="submitFormData.remark"
|
|
|
|
|
type="textarea"
|
|
|
|
|
:rows="3"
|
|
|
|
|
:placeholder="t('ErpStock.Out.placeholderSubmitRemark')"
|
|
|
|
|
/>
|
|
|
|
|
</el-form-item>
|
|
|
|
|
</el-form>
|
|
|
|
|
<template #footer>
|
|
|
|
|
<el-button @click="submitDialogVisible = false">{{ t('common.cancel') }}</el-button>
|
|
|
|
|
<el-button type="primary" :loading="actionLoading" @click="handleSubmitStockOut">{{ t('common.ok') }}</el-button>
|
|
|
|
|
</template>
|
|
|
|
|
</Dialog>
|
|
|
|
|
|
|
|
|
|
<StockOutForm v-else ref="formRef" @success="getList" @closed="formVisible = false" />
|
|
|
|
|
<Dialog :title="auditDialogTitle" v-model="auditDialogVisible" width="520px">
|
|
|
|
|
<el-form ref="auditFormRef" :model="auditFormData" label-width="90px">
|
|
|
|
|
<el-form-item :label="t('ErpStock.Out.remark')" prop="remark">
|
|
|
|
|
<el-input
|
|
|
|
|
v-model="auditFormData.remark"
|
|
|
|
|
type="textarea"
|
|
|
|
|
:rows="3"
|
|
|
|
|
:placeholder="auditRemarkPlaceholder"
|
|
|
|
|
/>
|
|
|
|
|
</el-form-item>
|
|
|
|
|
</el-form>
|
|
|
|
|
<template #footer>
|
|
|
|
|
<el-button @click="auditDialogVisible = false">{{ t('common.cancel') }}</el-button>
|
|
|
|
|
<el-button type="primary" :loading="actionLoading" @click="handleAuditStockOut">{{ t('common.ok') }}</el-button>
|
|
|
|
|
</template>
|
|
|
|
|
</Dialog>
|
|
|
|
|
</div>
|
|
|
|
|
</template>
|
|
|
|
|
|
|
|
|
|
<script setup lang="ts">
|
|
|
|
|
import { getIntDictOptions, DICT_TYPE } from '@/utils/dict'
|
|
|
|
|
import { getIntDictOptions, getStrDictOptions, DICT_TYPE } from '@/utils/dict'
|
|
|
|
|
import { dateFormatter2 } from '@/utils/formatTime'
|
|
|
|
|
import download from '@/utils/download'
|
|
|
|
|
import { StockOutApi, StockOutVO } from '@/api/erp/stock/out'
|
|
|
|
|
import StockOutForm from './StockOutForm.vue'
|
|
|
|
|
import { ProductApi, ProductVO } from '@/api/erp/product/product'
|
|
|
|
|
import { WarehouseApi, WarehouseVO } from '@/api/erp/stock/warehouse'
|
|
|
|
|
import { SupplierApi, SupplierVO } from '@/api/erp/purchase/supplier'
|
|
|
|
|
import { UserVO } from '@/api/system/user'
|
|
|
|
|
import * as UserApi from '@/api/system/user'
|
|
|
|
|
import { erpCountTableColumnFormatter, erpPriceTableColumnFormatter } from '@/utils'
|
|
|
|
|
import { CustomerApi, CustomerVO } from '@/api/erp/sale/customer'
|
|
|
|
|
|
|
|
|
|
/** ERP 其它出库单列表 */
|
|
|
|
|
/** ERP 鍏跺畠鍏ュ簱鍗曞垪琛?*/
|
|
|
|
|
defineOptions({ name: 'ErpStockOut' })
|
|
|
|
|
|
|
|
|
|
const message = useMessage() // 消息弹窗
|
|
|
|
|
const { t } = useI18n() // 国际化
|
|
|
|
|
const message = useMessage() // 娑堟伅寮圭獥
|
|
|
|
|
const { t } = useI18n() // 鍥介檯鍖?
|
|
|
|
|
|
|
|
|
|
const loading = ref(true) // 列表的加载中
|
|
|
|
|
const list = ref<StockOutVO[]>([]) // 列表的数据
|
|
|
|
|
const total = ref(0) // 列表的总页数
|
|
|
|
|
const loading = ref(true) // 鍒楄〃鐨勫姞杞戒腑
|
|
|
|
|
const list = ref<StockOutVO[]>([]) // 鍒楄〃鐨勬暟鎹?
|
|
|
|
|
const total = ref(0) // 鍒楄〃鐨勬€婚〉鏁?
|
|
|
|
|
const queryParams = reactive({
|
|
|
|
|
pageNo: 1,
|
|
|
|
|
pageSize: 10,
|
|
|
|
|
no: undefined,
|
|
|
|
|
productId: undefined,
|
|
|
|
|
customerId: undefined,
|
|
|
|
|
outType: undefined,
|
|
|
|
|
warehouseId: undefined,
|
|
|
|
|
outTime: [],
|
|
|
|
|
remark: undefined,
|
|
|
|
|
outType: undefined,
|
|
|
|
|
creator: undefined
|
|
|
|
|
})
|
|
|
|
|
const queryFormRef = ref() // 搜索的表单
|
|
|
|
|
const exportLoading = ref(false) // 导出的加载中
|
|
|
|
|
const WAREHOUSE_DOCUMENT_TYPES = 'warehouse_document_types'
|
|
|
|
|
const stockOutTypeOptions = computed(() => getStrDictOptions(WAREHOUSE_DOCUMENT_TYPES))
|
|
|
|
|
const outTypeToInfoKey = (outType: string) => {
|
|
|
|
|
const map: Record<string, string> = {
|
|
|
|
|
'产品出库': 'productInfo',
|
|
|
|
|
'物料出库': 'materialInfo',
|
|
|
|
|
'原料出库': 'materialInfo',
|
|
|
|
|
'领料出库': 'materialInfo',
|
|
|
|
|
'备件出库': 'sparePartInfo'
|
|
|
|
|
}
|
|
|
|
|
return map[outType] || 'productInfo'
|
|
|
|
|
}
|
|
|
|
|
const activeName = ref('')
|
|
|
|
|
const queryFormRef = ref() // 鎼滅储鐨勮〃鍗?
|
|
|
|
|
const exportLoading = ref(false) // 瀵煎嚭鐨勫姞杞戒腑
|
|
|
|
|
const showAllFilters = ref(false)
|
|
|
|
|
const filterCount = 7
|
|
|
|
|
const toggleFilters = () => {
|
|
|
|
|
showAllFilters.value = !showAllFilters.value
|
|
|
|
|
}
|
|
|
|
|
const productList = ref<ProductVO[]>([]) // 产品列表
|
|
|
|
|
const warehouseList = ref<WarehouseVO[]>([]) // 仓库列表
|
|
|
|
|
const customerList = ref<CustomerVO[]>([]) // 客户列表
|
|
|
|
|
const userList = ref<UserVO[]>([]) // 用户列表
|
|
|
|
|
const productList = ref<ProductVO[]>([]) // 浜у搧鍒楄〃
|
|
|
|
|
const warehouseList = ref<WarehouseVO[]>([]) // 浠撳簱鍒楄〃
|
|
|
|
|
const userList = ref<UserVO[]>([]) // 鐢ㄦ埛鍒楄〃
|
|
|
|
|
|
|
|
|
|
/** 查询列表 */
|
|
|
|
|
const formVisible = ref(false) // 琛ㄥ崟鏄惁鍙
|
|
|
|
|
const actionLoading = ref(false)
|
|
|
|
|
const submitDialogVisible = ref(false)
|
|
|
|
|
const submitFormRef = ref()
|
|
|
|
|
const submitFormData = reactive({
|
|
|
|
|
id: '',
|
|
|
|
|
auditUserId: '',
|
|
|
|
|
remark: ''
|
|
|
|
|
})
|
|
|
|
|
const submitFormRules = reactive({
|
|
|
|
|
auditUserId: [{ required: true, message: t('ErpStock.Out.validatorAuditUserRequired'), trigger: 'change' }]
|
|
|
|
|
})
|
|
|
|
|
const auditDialogVisible = ref(false)
|
|
|
|
|
const auditFormRef = ref()
|
|
|
|
|
const auditFormData = reactive({
|
|
|
|
|
id: '',
|
|
|
|
|
status: '',
|
|
|
|
|
remark: ''
|
|
|
|
|
})
|
|
|
|
|
const auditDialogTitle = computed(() => auditFormData.status === '20' ? t('ErpStock.Out.auditApprove') : t('ErpStock.Out.auditReject'))
|
|
|
|
|
const auditRemarkPlaceholder = computed(() => auditFormData.status === '20' ? t('ErpStock.Out.placeholderAuditApproveRemark') : t('ErpStock.Out.placeholderAuditRejectRemark'))
|
|
|
|
|
|
|
|
|
|
/** 鏌ヨ鍒楄〃 */
|
|
|
|
|
const getList = async () => {
|
|
|
|
|
loading.value = true
|
|
|
|
|
try {
|
|
|
|
|
@ -349,63 +416,96 @@ const getList = async () => {
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** 搜索按钮操作 */
|
|
|
|
|
/** 鎼滅储鎸夐挳鎿嶄綔 */
|
|
|
|
|
const handleQuery = () => {
|
|
|
|
|
queryParams.pageNo = 1
|
|
|
|
|
getList()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** 重置按钮操作 */
|
|
|
|
|
/** 閲嶇疆鎸夐挳鎿嶄綔 */
|
|
|
|
|
const resetQuery = () => {
|
|
|
|
|
queryFormRef.value.resetFields()
|
|
|
|
|
queryParams.outType = activeName.value || stockOutTypeOptions.value[0]?.value
|
|
|
|
|
handleQuery()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** 添加/修改操作 */
|
|
|
|
|
/** 娣诲姞/淇敼鎿嶄綔 */
|
|
|
|
|
const formRef = ref()
|
|
|
|
|
const formVisible = ref(false)
|
|
|
|
|
|
|
|
|
|
const openForm = async (type: string, id?: number) => {
|
|
|
|
|
const openForm = (type: string, id?: number) => {
|
|
|
|
|
formVisible.value = true
|
|
|
|
|
await nextTick()
|
|
|
|
|
formRef.value?.open(type, id)
|
|
|
|
|
nextTick(() => {
|
|
|
|
|
formRef.value.open(type, id)
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** 删除按钮操作 */
|
|
|
|
|
/** 鍒犻櫎鎸夐挳鎿嶄綔 */
|
|
|
|
|
const handleDelete = async (ids: number[]) => {
|
|
|
|
|
try {
|
|
|
|
|
// 删除的二次确认
|
|
|
|
|
// 鍒犻櫎鐨勪簩娆$‘璁?
|
|
|
|
|
await message.delConfirm()
|
|
|
|
|
// 发起删除
|
|
|
|
|
// 鍙戣捣鍒犻櫎
|
|
|
|
|
await StockOutApi.deleteStockOut(ids)
|
|
|
|
|
message.success(t('common.delSuccess'))
|
|
|
|
|
// 刷新列表
|
|
|
|
|
// 鍒锋柊鍒楄〃
|
|
|
|
|
await getList()
|
|
|
|
|
selectionList.value = selectionList.value.filter((item) => !ids.includes(item.id))
|
|
|
|
|
} catch {}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** 审批/反审批操作 */
|
|
|
|
|
const handleUpdateStatus = async (id: number, status: number) => {
|
|
|
|
|
const openSubmitDialog = (row: StockOutVO) => {
|
|
|
|
|
submitFormData.id = String(row.id)
|
|
|
|
|
submitFormData.auditUserId = ''
|
|
|
|
|
submitFormData.remark = ''
|
|
|
|
|
submitDialogVisible.value = true
|
|
|
|
|
nextTick(() => submitFormRef.value?.clearValidate?.())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const handleSubmitStockOut = async () => {
|
|
|
|
|
await submitFormRef.value.validate()
|
|
|
|
|
actionLoading.value = true
|
|
|
|
|
try {
|
|
|
|
|
// 审批的二次确认
|
|
|
|
|
const confirmText = status === 20 ? t('ErpStock.Out.confirmApprove') : t('ErpStock.Out.confirmReverseApprove')
|
|
|
|
|
await message.confirm(confirmText)
|
|
|
|
|
// 发起审批
|
|
|
|
|
await StockOutApi.updateStockOutStatus(id, status)
|
|
|
|
|
const successText = status === 20 ? t('ErpStock.Out.approveSuccess') : t('ErpStock.Out.reverseApproveSuccess')
|
|
|
|
|
message.success(successText)
|
|
|
|
|
// 刷新列表
|
|
|
|
|
await StockOutApi.submitStockOut({
|
|
|
|
|
id: submitFormData.id,
|
|
|
|
|
auditUserId: submitFormData.auditUserId,
|
|
|
|
|
remark: submitFormData.remark
|
|
|
|
|
})
|
|
|
|
|
message.success(t('ErpStock.Out.submitSuccess'))
|
|
|
|
|
submitDialogVisible.value = false
|
|
|
|
|
await getList()
|
|
|
|
|
} catch {}
|
|
|
|
|
} finally {
|
|
|
|
|
actionLoading.value = false
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const openAuditDialog = (row: StockOutVO, status: 20 | 1) => {
|
|
|
|
|
auditFormData.id = String(row.id)
|
|
|
|
|
auditFormData.status = String(status)
|
|
|
|
|
auditFormData.remark = status === 20 ? t('ErpStock.Out.auditApprove') : ''
|
|
|
|
|
auditDialogVisible.value = true
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** 导出按钮操作 */
|
|
|
|
|
const handleAuditStockOut = async () => {
|
|
|
|
|
actionLoading.value = true
|
|
|
|
|
try {
|
|
|
|
|
await StockOutApi.auditStockOut({
|
|
|
|
|
id: auditFormData.id,
|
|
|
|
|
status: auditFormData.status,
|
|
|
|
|
remark: auditFormData.remark
|
|
|
|
|
})
|
|
|
|
|
message.success(auditFormData.status === '20' ? t('ErpStock.Out.auditApproveSuccess') : t('ErpStock.Out.auditRejectSuccess'))
|
|
|
|
|
auditDialogVisible.value = false
|
|
|
|
|
await getList()
|
|
|
|
|
} finally {
|
|
|
|
|
actionLoading.value = false
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** 瀵煎嚭鎸夐挳鎿嶄綔 */
|
|
|
|
|
const handleExport = async () => {
|
|
|
|
|
try {
|
|
|
|
|
// 导出的二次确认
|
|
|
|
|
// 瀵煎嚭鐨勪簩娆$‘璁?
|
|
|
|
|
await message.exportConfirm()
|
|
|
|
|
// 发起导出
|
|
|
|
|
// 鍙戣捣瀵煎嚭
|
|
|
|
|
exportLoading.value = true
|
|
|
|
|
const ids = selectionList.value.map((item) => item.id).filter((v) => v != null)
|
|
|
|
|
const params = {
|
|
|
|
|
@ -420,28 +520,43 @@ const handleExport = async () => {
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** 选中操作 */
|
|
|
|
|
/** 閫変腑鎿嶄綔 */
|
|
|
|
|
const selectionList = ref<StockOutVO[]>([])
|
|
|
|
|
const handleSelectionChange = (rows: StockOutVO[]) => {
|
|
|
|
|
selectionList.value = rows
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** 初始化 **/
|
|
|
|
|
const syncDefaultOutType = () => {
|
|
|
|
|
if (queryParams.outType) {
|
|
|
|
|
activeName.value = queryParams.outType
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
const defaultOutType = stockOutTypeOptions.value[0]?.value
|
|
|
|
|
if (defaultOutType) {
|
|
|
|
|
queryParams.outType = defaultOutType
|
|
|
|
|
activeName.value = defaultOutType
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
watch(stockOutTypeOptions, (options) => {
|
|
|
|
|
if (!queryParams.outType && options.length) {
|
|
|
|
|
syncDefaultOutType()
|
|
|
|
|
handleQuery()
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
/** 鍒濆鍖?**/
|
|
|
|
|
onMounted(async () => {
|
|
|
|
|
queryParams.outType ='产品出库'
|
|
|
|
|
syncDefaultOutType()
|
|
|
|
|
await getList()
|
|
|
|
|
// 加载产品、仓库列表、客户
|
|
|
|
|
// 鍔犺浇浜у搧銆佷粨搴撳垪琛ㄣ€佷緵搴斿晢
|
|
|
|
|
productList.value = await ProductApi.getProductSimpleList()
|
|
|
|
|
warehouseList.value = await WarehouseApi.getWarehouseSimpleList()
|
|
|
|
|
customerList.value = await CustomerApi.getCustomerSimpleList()
|
|
|
|
|
userList.value = await UserApi.getSimpleUserList()
|
|
|
|
|
})
|
|
|
|
|
// TODO 芋艿:可优化功能:列表界面,支持导入
|
|
|
|
|
// TODO 芋艿:可优化功能:详情界面,支持打印
|
|
|
|
|
/** tab 切换 */
|
|
|
|
|
let activeName = '领料出库'
|
|
|
|
|
const handleTabClick = (tab: TabsPaneContext) => {
|
|
|
|
|
queryParams.outType = tab.paneName
|
|
|
|
|
queryParams.outType = String(tab.paneName || '')
|
|
|
|
|
activeName.value = queryParams.outType
|
|
|
|
|
handleQuery()
|
|
|
|
|
}
|
|
|
|
|
</script>
|
|
|
|
|
|