SupplyCapabilityView.vue 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393
  1. <template>
  2. <div class="app-page">
  3. <!-- 筛选区 -->
  4. <section class="glass-card section-card">
  5. <el-form :model="filters" inline class="filter-form">
  6. <el-form-item label="供应商">
  7. <el-select v-model="filters.supplier" placeholder="全部" clearable filterable style="width:160px">
  8. <el-option v-for="s in supplierOptions" :key="s" :label="s" :value="s" />
  9. </el-select>
  10. </el-form-item>
  11. <el-form-item label="SKU">
  12. <el-input v-model="filters.sku" placeholder="SKU 编码" clearable style="width:160px" @keyup.enter="loadData" />
  13. </el-form-item>
  14. <el-form-item label="默认供应商">
  15. <el-select v-model="filters.isDefault" placeholder="全部" clearable style="width:120px">
  16. <el-option label="是" :value="true" />
  17. <el-option label="否" :value="false" />
  18. </el-select>
  19. </el-form-item>
  20. <el-form-item label="状态">
  21. <el-select v-model="filters.status" placeholder="全部" clearable style="width:120px">
  22. <el-option label="启用" value="启用" />
  23. <el-option label="停用" value="停用" />
  24. </el-select>
  25. </el-form-item>
  26. <el-form-item>
  27. <el-button type="primary" @click="loadData">查询</el-button>
  28. <el-button @click="resetFilters">重置</el-button>
  29. </el-form-item>
  30. </el-form>
  31. </section>
  32. <!-- 工具栏 -->
  33. <section class="glass-card section-card" style="padding:12px 24px">
  34. <div class="table-toolbar" style="margin-bottom:0">
  35. <div class="chip-list">
  36. <el-button type="primary" @click="openCreate">新增配置</el-button>
  37. </div>
  38. <el-button @click="loadData">刷新</el-button>
  39. </div>
  40. </section>
  41. <!-- 列表 -->
  42. <section class="glass-card section-card">
  43. <el-table :data="filteredItems" style="width:100%" v-loading="loading">
  44. <el-table-column prop="supplier" label="供应商" min-width="180" />
  45. <el-table-column prop="sku" label="SKU" width="140">
  46. <template #default="{ row }">
  47. <div>{{ row.sku }}</div>
  48. <div style="color:var(--cb-text-soft);font-size:12px">{{ row.productTitle }}</div>
  49. </template>
  50. </el-table-column>
  51. <el-table-column prop="leadTime" label="标准交期(天)" width="120" />
  52. <el-table-column prop="moq" label="MOQ" width="80" />
  53. <el-table-column prop="unit" label="采购单位" width="90" />
  54. <el-table-column label="阶梯价" min-width="200">
  55. <template #default="{ row }">
  56. <div v-if="row.tierPrices && row.tierPrices.length">
  57. <div v-for="(tier, idx) in row.tierPrices" :key="idx" class="tier-row">
  58. {{ tier.from }}~{{ tier.to }} : {{ tier.price }}
  59. </div>
  60. </div>
  61. <span v-else style="color:var(--cb-text-soft)">--</span>
  62. </template>
  63. </el-table-column>
  64. <el-table-column prop="isDefault" label="默认供应商" width="100">
  65. <template #default="{ row }">
  66. <el-tag v-if="row.isDefault" type="success" size="small">默认</el-tag>
  67. <span v-else style="color:var(--cb-text-soft)">--</span>
  68. </template>
  69. </el-table-column>
  70. <el-table-column prop="status" label="状态" width="80">
  71. <template #default="{ row }">
  72. <el-tag :type="getCapabilityStatus(row.status).type" size="small">{{ getCapabilityStatus(row.status).label }}</el-tag>
  73. </template>
  74. </el-table-column>
  75. <el-table-column label="操作" width="240" fixed="right">
  76. <template #default="{ row }">
  77. <el-button link type="primary" @click="openEdit(row)">编辑</el-button>
  78. <el-button link type="primary" @click="setDefault(row)" :disabled="row.isDefault">设为默认</el-button>
  79. <el-button link type="danger" @click="toggleStatus(row)">{{ row.status === '启用' ? '停用' : '启用' }}</el-button>
  80. </template>
  81. </el-table-column>
  82. <template #empty>
  83. <el-empty description="暂无数据" />
  84. </template>
  85. </el-table>
  86. <div style="display:flex;justify-content:flex-end;margin-top:16px" v-if="total > 0">
  87. <el-pagination v-model:current-page="page" v-model:page-size="pageSize" :total="total" :page-sizes="[10, 20, 50]" layout="total, sizes, prev, pager, next" @size-change="loadData" @current-change="loadData" />
  88. </div>
  89. </section>
  90. <!-- 新建/编辑弹窗 -->
  91. <el-dialog v-model="dialogVisible" :title="isEdit ? '编辑供货配置' : '新增供货配置'" width="700px" destroy-on-close>
  92. <el-form ref="formRef" :model="formData" :rules="formRules" label-width="110px" label-position="right">
  93. <el-row :gutter="20">
  94. <el-col :span="12">
  95. <el-form-item label="供应商" prop="supplier">
  96. <el-select v-model="formData.supplier" placeholder="请选择供应商" filterable style="width:100%">
  97. <el-option v-for="s in supplierOptions" :key="s" :label="s" :value="s" />
  98. </el-select>
  99. </el-form-item>
  100. </el-col>
  101. <el-col :span="12">
  102. <el-form-item label="SKU" prop="sku">
  103. <el-input v-model="formData.sku" placeholder="SKU 编码" />
  104. </el-form-item>
  105. </el-col>
  106. </el-row>
  107. <el-row :gutter="20">
  108. <el-col :span="8">
  109. <el-form-item label="标准交期" prop="leadTime">
  110. <el-input-number v-model="formData.leadTime" :min="0" :max="365" controls-position="right" style="width:100%" />
  111. </el-form-item>
  112. </el-col>
  113. <el-col :span="8">
  114. <el-form-item label="MOQ" prop="moq">
  115. <el-input-number v-model="formData.moq" :min="0" :max="999999" controls-position="right" style="width:100%" />
  116. </el-form-item>
  117. </el-col>
  118. <el-col :span="8">
  119. <el-form-item label="采购单位" prop="unit">
  120. <el-select v-model="formData.unit" placeholder="单位" style="width:100%">
  121. <el-option label="件" value="件" />
  122. <el-option label="箱" value="箱" />
  123. <el-option label="套" value="套" />
  124. <el-option label="个" value="个" />
  125. </el-select>
  126. </el-form-item>
  127. </el-col>
  128. </el-row>
  129. <el-row :gutter="20">
  130. <el-col :span="8">
  131. <el-form-item label="币种" prop="currency">
  132. <el-select v-model="formData.currency" placeholder="币种" style="width:100%">
  133. <el-option label="CNY" value="CNY" />
  134. <el-option label="USD" value="USD" />
  135. <el-option label="EUR" value="EUR" />
  136. </el-select>
  137. </el-form-item>
  138. </el-col>
  139. </el-row>
  140. <!-- 阶梯价 -->
  141. <el-divider content-position="left">阶梯价格</el-divider>
  142. <div v-for="(tier, idx) in formData.tierPrices" :key="idx" style="margin-bottom:8px">
  143. <el-row :gutter="8" align="middle">
  144. <el-col :span="6">
  145. <el-input-number v-model="tier.from" :min="0" controls-position="right" placeholder="起始量" style="width:100%" />
  146. </el-col>
  147. <el-col :span="6">
  148. <el-input-number v-model="tier.to" :min="0" controls-position="right" placeholder="结束量" style="width:100%" />
  149. </el-col>
  150. <el-col :span="6">
  151. <el-input v-model="tier.price" placeholder="单价" />
  152. </el-col>
  153. <el-col :span="6">
  154. <el-button link type="danger" @click="removeTier(idx)" :disabled="formData.tierPrices.length <= 1">删除</el-button>
  155. </el-col>
  156. </el-row>
  157. </div>
  158. <el-button type="primary" link @click="addTier" style="margin-bottom:12px">+ 添加阶梯</el-button>
  159. <el-form-item label="备注" prop="remark">
  160. <el-input v-model="formData.remark" type="textarea" :rows="3" placeholder="备注信息" />
  161. </el-form-item>
  162. </el-form>
  163. <template #footer>
  164. <el-button @click="dialogVisible = false">取消</el-button>
  165. <el-button type="primary" :loading="submitting" @click="handleSubmit">确认</el-button>
  166. </template>
  167. </el-dialog>
  168. </div>
  169. </template>
  170. <script setup lang="ts">
  171. import { computed, onMounted, reactive, ref } from 'vue';
  172. import { ElMessage, ElMessageBox } from 'element-plus';
  173. import type { FormInstance, FormRules } from 'element-plus';
  174. import { api } from '@/api/services';
  175. import type { SupplyCapabilityItem } from '@/types/page';
  176. import { getCapabilityStatus } from '@/utils/enumMappings';
  177. const items = ref<SupplyCapabilityItem[]>([]);
  178. const loading = ref(false);
  179. const page = ref(1);
  180. const pageSize = ref(10);
  181. const dialogVisible = ref(false);
  182. const isEdit = ref(false);
  183. const editId = ref('');
  184. const submitting = ref(false);
  185. const formRef = ref<FormInstance>();
  186. const supplierOptions = ref<string[]>([]);
  187. const filters = ref({
  188. supplier: '',
  189. sku: '',
  190. isDefault: null as boolean | null,
  191. status: ''
  192. });
  193. interface TierPrice {
  194. from: number;
  195. to: number;
  196. price: string;
  197. }
  198. interface FormState {
  199. supplier: string;
  200. sku: string;
  201. leadTime: number;
  202. moq: number;
  203. unit: string;
  204. currency: string;
  205. tierPrices: TierPrice[];
  206. remark: string;
  207. }
  208. const defaultFormData = (): FormState => ({
  209. supplier: '',
  210. sku: '',
  211. leadTime: 0,
  212. moq: 0,
  213. unit: '件',
  214. currency: 'CNY',
  215. tierPrices: [{ from: 1, to: 100, price: '' }],
  216. remark: ''
  217. });
  218. const formData = reactive<FormState>(defaultFormData());
  219. const formRules: FormRules = {
  220. supplier: [{ required: true, message: '请选择供应商', trigger: 'change' }],
  221. sku: [{ required: true, message: '请输入 SKU', trigger: 'blur' }],
  222. leadTime: [{ required: true, message: '请输入交期', trigger: 'blur' }],
  223. moq: [{ required: true, message: '请输入 MOQ', trigger: 'blur' }],
  224. unit: [{ required: true, message: '请选择单位', trigger: 'change' }],
  225. currency: [{ required: true, message: '请选择币种', trigger: 'change' }]
  226. };
  227. const filteredItems = computed(() => {
  228. return items.value.filter((item) => {
  229. if (filters.value.supplier && item.supplier !== filters.value.supplier) return false;
  230. if (filters.value.sku && !item.sku.includes(filters.value.sku)) return false;
  231. if (filters.value.isDefault !== null && item.isDefault !== filters.value.isDefault) return false;
  232. if (filters.value.status && item.status !== filters.value.status) return false;
  233. return true;
  234. });
  235. });
  236. const total = computed(() => filteredItems.value.length);
  237. const loadData = async () => {
  238. loading.value = true;
  239. try {
  240. const [capRes, supplierRes] = await Promise.all([
  241. api.getSupplyCapabilities(),
  242. api.getSuppliers()
  243. ] as const);
  244. items.value = capRes.items ?? [];
  245. supplierOptions.value = (supplierRes.items ?? []).map((s: any) => s.name);
  246. } finally {
  247. loading.value = false;
  248. }
  249. };
  250. const resetFilters = () => {
  251. filters.value = { supplier: '', sku: '', isDefault: null, status: '' };
  252. };
  253. const addTier = () => {
  254. const last = formData.tierPrices[formData.tierPrices.length - 1];
  255. formData.tierPrices.push({ from: last?.to ? last.to + 1 : 1, to: 0, price: '' });
  256. };
  257. const removeTier = (idx: number) => {
  258. formData.tierPrices.splice(idx, 1);
  259. };
  260. const openCreate = () => {
  261. isEdit.value = false;
  262. editId.value = '';
  263. Object.assign(formData, defaultFormData());
  264. dialogVisible.value = true;
  265. };
  266. const openEdit = (row: SupplyCapabilityItem) => {
  267. isEdit.value = true;
  268. editId.value = row.id;
  269. Object.assign(formData, {
  270. supplier: row.supplier,
  271. sku: row.sku,
  272. leadTime: row.leadTime,
  273. moq: row.moq,
  274. unit: row.unit,
  275. currency: 'CNY',
  276. tierPrices: row.tierPrices && row.tierPrices.length
  277. ? row.tierPrices.map((t) => ({ from: t.from, to: t.to, price: t.price }))
  278. : [{ from: 1, to: 100, price: '' }],
  279. remark: ''
  280. });
  281. dialogVisible.value = true;
  282. };
  283. const handleSubmit = async () => {
  284. if (!formRef.value) return;
  285. await formRef.value.validate();
  286. if (formData.leadTime < 0) {
  287. ElMessage.warning('标准交期不能为负数');
  288. return;
  289. }
  290. if (formData.moq < 0) {
  291. ElMessage.warning('MOQ 不能为负数');
  292. return;
  293. }
  294. submitting.value = true;
  295. try {
  296. const payload: Partial<SupplyCapabilityItem> = {
  297. supplier: formData.supplier,
  298. sku: formData.sku,
  299. leadTime: formData.leadTime,
  300. moq: formData.moq,
  301. unit: formData.unit,
  302. tierPrices: formData.tierPrices.map((t) => ({ from: t.from, to: t.to, price: t.price }))
  303. };
  304. if (isEdit.value) {
  305. await api.updateSupplyCapability(Number(editId.value), payload);
  306. ElMessage.success('供货配置更新成功');
  307. } else {
  308. await api.createSupplyCapability(payload);
  309. ElMessage.success('供货配置创建成功');
  310. }
  311. dialogVisible.value = false;
  312. loadData();
  313. } finally {
  314. submitting.value = false;
  315. }
  316. };
  317. const setDefault = async (row: SupplyCapabilityItem) => {
  318. try {
  319. await ElMessageBox.confirm(
  320. `确认将「${row.supplier}」设为 SKU「${row.sku}」的默认供应商?此操作将清除该 SKU 的其他默认供应商设置。`,
  321. '设为默认',
  322. { type: 'info' }
  323. );
  324. // Clear existing defaults for the same SKU
  325. const sameSkuDefaults = items.value.filter(
  326. (item) => item.sku === row.sku && item.isDefault && item.id !== row.id
  327. );
  328. for (const item of sameSkuDefaults) {
  329. await api.updateSupplyCapability(Number(item.id), { isDefault: false } as Partial<SupplyCapabilityItem>);
  330. }
  331. // Set the new default
  332. await api.updateSupplyCapability(Number(row.id), { isDefault: true } as Partial<SupplyCapabilityItem>);
  333. ElMessage.success('已设为默认供应商');
  334. loadData();
  335. } catch {
  336. // cancelled
  337. }
  338. };
  339. const toggleStatus = async (row: SupplyCapabilityItem) => {
  340. const target = row.status === '启用' ? '停用' : '启用';
  341. const action = target === '停用' ? '停用' : '启用';
  342. try {
  343. await ElMessageBox.confirm(
  344. `确认${action}「${row.supplier}」对 SKU「${row.sku}」的供货配置?`,
  345. `${action}确认`,
  346. { type: target === '停用' ? 'warning' : 'info' }
  347. );
  348. await api.updateSupplyCapability(Number(row.id), { status: target } as Partial<SupplyCapabilityItem>);
  349. ElMessage.success(`${action}成功`);
  350. loadData();
  351. } catch {
  352. // cancelled
  353. }
  354. };
  355. onMounted(loadData);
  356. </script>
  357. <style scoped>
  358. .filter-form :deep(.el-form-item) { margin-bottom: 0; }
  359. .tier-row { font-size: 13px; color: var(--cb-text-soft); line-height: 1.6; }
  360. </style>