SupplyCapabilityView.vue 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392
  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" stripe 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="row.status === '启用' ? 'success' : 'info'" size="small">{{ row.status }}</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. const items = ref<SupplyCapabilityItem[]>([]);
  177. const loading = ref(false);
  178. const page = ref(1);
  179. const pageSize = ref(10);
  180. const dialogVisible = ref(false);
  181. const isEdit = ref(false);
  182. const editId = ref('');
  183. const submitting = ref(false);
  184. const formRef = ref<FormInstance>();
  185. const supplierOptions = ref<string[]>([]);
  186. const filters = ref({
  187. supplier: '',
  188. sku: '',
  189. isDefault: null as boolean | null,
  190. status: ''
  191. });
  192. interface TierPrice {
  193. from: number;
  194. to: number;
  195. price: string;
  196. }
  197. interface FormState {
  198. supplier: string;
  199. sku: string;
  200. leadTime: number;
  201. moq: number;
  202. unit: string;
  203. currency: string;
  204. tierPrices: TierPrice[];
  205. remark: string;
  206. }
  207. const defaultFormData = (): FormState => ({
  208. supplier: '',
  209. sku: '',
  210. leadTime: 0,
  211. moq: 0,
  212. unit: '件',
  213. currency: 'CNY',
  214. tierPrices: [{ from: 1, to: 100, price: '' }],
  215. remark: ''
  216. });
  217. const formData = reactive<FormState>(defaultFormData());
  218. const formRules: FormRules = {
  219. supplier: [{ required: true, message: '请选择供应商', trigger: 'change' }],
  220. sku: [{ required: true, message: '请输入 SKU', trigger: 'blur' }],
  221. leadTime: [{ required: true, message: '请输入交期', trigger: 'blur' }],
  222. moq: [{ required: true, message: '请输入 MOQ', trigger: 'blur' }],
  223. unit: [{ required: true, message: '请选择单位', trigger: 'change' }],
  224. currency: [{ required: true, message: '请选择币种', trigger: 'change' }]
  225. };
  226. const filteredItems = computed(() => {
  227. return items.value.filter((item) => {
  228. if (filters.value.supplier && item.supplier !== filters.value.supplier) return false;
  229. if (filters.value.sku && !item.sku.includes(filters.value.sku)) return false;
  230. if (filters.value.isDefault !== null && item.isDefault !== filters.value.isDefault) return false;
  231. if (filters.value.status && item.status !== filters.value.status) return false;
  232. return true;
  233. });
  234. });
  235. const total = computed(() => filteredItems.value.length);
  236. const loadData = async () => {
  237. loading.value = true;
  238. try {
  239. const [capRes, supplierRes] = await Promise.all([
  240. api.getSupplyCapabilities(),
  241. api.getSuppliers()
  242. ] as const);
  243. items.value = capRes.items;
  244. supplierOptions.value = supplierRes.items.map((s) => s.name);
  245. } finally {
  246. loading.value = false;
  247. }
  248. };
  249. const resetFilters = () => {
  250. filters.value = { supplier: '', sku: '', isDefault: null, status: '' };
  251. };
  252. const addTier = () => {
  253. const last = formData.tierPrices[formData.tierPrices.length - 1];
  254. formData.tierPrices.push({ from: last?.to ? last.to + 1 : 1, to: 0, price: '' });
  255. };
  256. const removeTier = (idx: number) => {
  257. formData.tierPrices.splice(idx, 1);
  258. };
  259. const openCreate = () => {
  260. isEdit.value = false;
  261. editId.value = '';
  262. Object.assign(formData, defaultFormData());
  263. dialogVisible.value = true;
  264. };
  265. const openEdit = (row: SupplyCapabilityItem) => {
  266. isEdit.value = true;
  267. editId.value = row.id;
  268. Object.assign(formData, {
  269. supplier: row.supplier,
  270. sku: row.sku,
  271. leadTime: row.leadTime,
  272. moq: row.moq,
  273. unit: row.unit,
  274. currency: 'CNY',
  275. tierPrices: row.tierPrices && row.tierPrices.length
  276. ? row.tierPrices.map((t) => ({ from: t.from, to: t.to, price: t.price }))
  277. : [{ from: 1, to: 100, price: '' }],
  278. remark: ''
  279. });
  280. dialogVisible.value = true;
  281. };
  282. const handleSubmit = async () => {
  283. if (!formRef.value) return;
  284. await formRef.value.validate();
  285. if (formData.leadTime < 0) {
  286. ElMessage.warning('标准交期不能为负数');
  287. return;
  288. }
  289. if (formData.moq < 0) {
  290. ElMessage.warning('MOQ 不能为负数');
  291. return;
  292. }
  293. submitting.value = true;
  294. try {
  295. const payload: Partial<SupplyCapabilityItem> = {
  296. supplier: formData.supplier,
  297. sku: formData.sku,
  298. leadTime: formData.leadTime,
  299. moq: formData.moq,
  300. unit: formData.unit,
  301. tierPrices: formData.tierPrices.map((t) => ({ from: t.from, to: t.to, price: t.price }))
  302. };
  303. if (isEdit.value) {
  304. await api.updateSupplyCapability(Number(editId.value), payload);
  305. ElMessage.success('供货配置更新成功');
  306. } else {
  307. await api.createSupplyCapability(payload);
  308. ElMessage.success('供货配置创建成功');
  309. }
  310. dialogVisible.value = false;
  311. loadData();
  312. } finally {
  313. submitting.value = false;
  314. }
  315. };
  316. const setDefault = async (row: SupplyCapabilityItem) => {
  317. try {
  318. await ElMessageBox.confirm(
  319. `确认将「${row.supplier}」设为 SKU「${row.sku}」的默认供应商?此操作将清除该 SKU 的其他默认供应商设置。`,
  320. '设为默认',
  321. { type: 'info' }
  322. );
  323. // Clear existing defaults for the same SKU
  324. const sameSkuDefaults = items.value.filter(
  325. (item) => item.sku === row.sku && item.isDefault && item.id !== row.id
  326. );
  327. for (const item of sameSkuDefaults) {
  328. await api.updateSupplyCapability(Number(item.id), { isDefault: false } as Partial<SupplyCapabilityItem>);
  329. }
  330. // Set the new default
  331. await api.updateSupplyCapability(Number(row.id), { isDefault: true } as Partial<SupplyCapabilityItem>);
  332. ElMessage.success('已设为默认供应商');
  333. loadData();
  334. } catch {
  335. // cancelled
  336. }
  337. };
  338. const toggleStatus = async (row: SupplyCapabilityItem) => {
  339. const target = row.status === '启用' ? '停用' : '启用';
  340. const action = target === '停用' ? '停用' : '启用';
  341. try {
  342. await ElMessageBox.confirm(
  343. `确认${action}「${row.supplier}」对 SKU「${row.sku}」的供货配置?`,
  344. `${action}确认`,
  345. { type: target === '停用' ? 'warning' : 'info' }
  346. );
  347. await api.updateSupplyCapability(Number(row.id), { status: target } as Partial<SupplyCapabilityItem>);
  348. ElMessage.success(`${action}成功`);
  349. loadData();
  350. } catch {
  351. // cancelled
  352. }
  353. };
  354. onMounted(loadData);
  355. </script>
  356. <style scoped>
  357. .filter-form :deep(.el-form-item) { margin-bottom: 0; }
  358. .tier-row { font-size: 13px; color: var(--cb-text-soft); line-height: 1.6; }
  359. </style>