You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
besure_web/src/views/erp/stock/out/index.vue

563 lines
19 KiB
Vue

<template>
<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.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-tabs>
<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.Out.${outTypeToInfoKey(queryParams.outType)}`)"
align="left"
sortable
prop="productNames"
min-width="200"
/>
<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-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-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"
/>
</ContentWrap>
</template>
<!-- 琛ㄥ崟闈㈡澘锛氭坊鍔?淇敼 -->
<StockOutForm v-else ref="formRef" @success="getList" @closed="formVisible = false" />
<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"
>
<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>
<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, 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 { UserVO } from '@/api/system/user'
import * as UserApi from '@/api/system/user'
import { erpCountTableColumnFormatter, erpPriceTableColumnFormatter } from '@/utils'
/** ERP ?*/
defineOptions({ name: 'ErpStockOut' })
const message = useMessage() //
const { t } = useI18n() // ?
const loading = ref(true) //
const list = ref<StockOutVO[]>([]) // 鍒楄〃鐨勬暟鎹?
const total = ref(0) // 鍒楄〃鐨勬€婚〉鏁?
const queryParams = reactive({
pageNo: 1,
pageSize: 10,
no: undefined,
productId: undefined,
outTime: [],
remark: undefined,
outType: undefined,
creator: undefined
})
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 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 {
const data = await StockOutApi.getStockOutPage(queryParams)
list.value = data.list
total.value = data.total
} finally {
loading.value = false
}
}
/** 鎼滅储鎸夐挳鎿嶄綔 */
const handleQuery = () => {
queryParams.pageNo = 1
getList()
}
/** 閲嶇疆鎸夐挳鎿嶄綔 */
const resetQuery = () => {
queryFormRef.value.resetFields()
queryParams.outType = activeName.value || stockOutTypeOptions.value[0]?.value
handleQuery()
}
/** 娣诲姞/淇敼鎿嶄綔 */
const formRef = ref()
const openForm = (type: string, id?: number) => {
formVisible.value = true
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 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 {
await StockOutApi.submitStockOut({
id: submitFormData.id,
auditUserId: submitFormData.auditUserId,
remark: submitFormData.remark
})
message.success(t('ErpStock.Out.submitSuccess'))
submitDialogVisible.value = false
await getList()
} 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 = {
...queryParams,
ids: ids.length ? ids.join(',') : undefined
}
const data = await StockOutApi.exportStockOut(params)
download.excel(data, t('ErpStock.Out.exportName'))
} catch {
} finally {
exportLoading.value = false
}
}
/** 閫変腑鎿嶄綔 */
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 () => {
syncDefaultOutType()
await getList()
// 鍔犺浇浜у搧銆佷粨搴撳垪琛ㄣ€佷緵搴斿晢
productList.value = await ProductApi.getProductSimpleList()
warehouseList.value = await WarehouseApi.getWarehouseSimpleList()
userList.value = await UserApi.getSimpleUserList()
})
const handleTabClick = (tab: TabsPaneContext) => {
queryParams.outType = String(tab.paneName || '')
activeName.value = queryParams.outType
handleQuery()
}
</script>