addEdit.vue 36 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180
  1. <template>
  2. <el-dialog
  3. :title="title"
  4. :center="true"
  5. align="left"
  6. top="8vh"
  7. width="1040px"
  8. @close="closeModel"
  9. :close-on-click-modal="false"
  10. :visible.sync="showModelThis"
  11. v-loading="loading"
  12. element-loading-text="拼命加载中"
  13. element-loading-spinner="el-icon-loading"
  14. element-loading-background="rgba(0, 0, 0, 0.8)"
  15. >
  16. <el-card>
  17. <el-row :gutter="10">
  18. <el-col :span="24">
  19. <el-col :span="10" :offset="7">
  20. <div style="margin: -5px 0 10px 0">
  21. <el-steps
  22. :active="step"
  23. simple
  24. process-status="finish"
  25. finish-status="success"
  26. >
  27. <el-step
  28. v-for="(item, index) in stepList"
  29. :key="index"
  30. :title="item"
  31. />
  32. </el-steps>
  33. </div>
  34. </el-col>
  35. <el-row :gutter="10" v-if="step === 0">
  36. <el-form
  37. :model="baseForm"
  38. status-icon
  39. :rules="baseRules"
  40. ref="baseForm"
  41. label-width="110px"
  42. class="demo-ruleForm"
  43. >
  44. <el-col :span="24">
  45. <el-form-item label="商品名称" prop="good_name">
  46. <el-input
  47. placeholder="请输入商品名称"
  48. v-model="baseForm.good_name"
  49. ></el-input>
  50. </el-form-item>
  51. </el-col>
  52. <el-col :span="12">
  53. <el-form-item label="供应商联系人" prop="supplier">
  54. <el-select
  55. v-model="baseForm.supplier"
  56. filterable
  57. clearable
  58. style="width: 100%"
  59. placeholder="请选择供应商联系人"
  60. >
  61. <el-option
  62. v-for="group in supplierOptions"
  63. :key="group.id"
  64. :label="group.name"
  65. :value="group.id"
  66. >
  67. <span>{{ group.name }}</span
  68. ><span class="fr">{{ group.company }}</span>
  69. </el-option>
  70. </el-select>
  71. </el-form-item>
  72. <el-form-item label="商品单位" prop="unit">
  73. <el-input
  74. placeholder="请输入单位"
  75. v-model="baseForm.unit"
  76. :disabled="isDetail"
  77. ></el-input>
  78. </el-form-item>
  79. </el-col>
  80. <el-col :span="12">
  81. <el-form-item label="商品分类" prop="catid">
  82. <el-cascader
  83. v-model="baseForm.catid"
  84. style="width: 100%"
  85. filterable
  86. placeholder="请选择商品分类"
  87. clearable
  88. :options="catOptions"
  89. :props="{ expandTrigger: 'hover', checkStrictly: true }"
  90. ></el-cascader>
  91. </el-form-item>
  92. <el-form-item label="商品品牌" prop="brandid">
  93. <el-select
  94. style="width: 100%"
  95. v-model="baseForm.brandid"
  96. clearable
  97. placeholder="请选择商品品牌"
  98. >
  99. <el-option
  100. v-for="item in brandidOptions"
  101. :key="item.id"
  102. :label="item.brand_name"
  103. :value="item.id"
  104. >
  105. </el-option>
  106. </el-select>
  107. </el-form-item>
  108. </el-col>
  109. <el-col :span="24">
  110. <el-form-item
  111. label="商品主图"
  112. prop="image"
  113. class="activity-upload"
  114. >
  115. <div class="img-list-main clearfix">
  116. <div
  117. class="img-item"
  118. v-for="(item, index) in baseForm.image"
  119. :key="item + index"
  120. >
  121. <img :src="item" class="avatar" />
  122. <i class="el-icon-close" @click="closeImg(index)"></i>
  123. </div>
  124. <div
  125. class="btnupload"
  126. v-if="baseForm.image.length < 5"
  127. style="position: relative"
  128. >
  129. <i class="el-icon-plus avatar-uploader-icon"></i>
  130. <file-upload
  131. class="Upload"
  132. :accept="'.jpg,.png,.jpeg'"
  133. :multiple="true"
  134. :uploadcondition="beforeAvatarUpload"
  135. @UploadErrorEvent="UploadErrorEvent"
  136. @UploadSuccessEvent="UploadSuccessEvent"
  137. ></file-upload>
  138. </div>
  139. </div>
  140. <div class="txt-tips fl">
  141. <p>
  142. <span sty>大小:小于1M;</span><span>尺寸:100*100;</span
  143. ><span>类型:jpg.png.jpeg</span>
  144. </p>
  145. </div>
  146. </el-form-item>
  147. </el-col>
  148. </el-form>
  149. <el-col :span="24" style="text-align: right">
  150. <el-tag type="warning" class="fl" style="margin-left: 100px"
  151. >更换商品分类,会影响规格与价格,请慎重修改</el-tag
  152. >
  153. <el-button type="primary" @click="baseSubmitForm"
  154. >下一步
  155. </el-button>
  156. <el-button @click="showModelThis = false">{{
  157. isDetail ? "关 闭" : "取 消"
  158. }}</el-button>
  159. </el-col>
  160. </el-row>
  161. <el-row :gutter="10" v-else>
  162. <el-col :span="24" style="max-height: 560px; overflow-y: scroll">
  163. <el-form
  164. :model="ruleForm"
  165. status-icon
  166. ref="ruleForm"
  167. label-width="60px"
  168. class="demo-dealGoodsPool-ruleForm"
  169. >
  170. <el-form-item label="规格" required>
  171. <div class="mack-mian clearfix">
  172. <div
  173. class="mock-item-div clearfix"
  174. v-for="(item, index) in mock"
  175. :key="'specItem' + index"
  176. >
  177. <div class="mock-title clearfix">
  178. <el-input
  179. :size="size"
  180. class="fr"
  181. style="width: 250px"
  182. maxlength="40"
  183. placeholder="新增规格值,如红色"
  184. v-model="item.value"
  185. >
  186. <el-button
  187. slot="append"
  188. @click="specinfoadd(item.spec_id, item.value)"
  189. >提交</el-button
  190. >
  191. </el-input>
  192. <span class="fl">{{ item.spec_name }}</span>
  193. </div>
  194. <div v-if="item.child && item.child.length > 0">
  195. <el-checkbox-group
  196. v-model="item.change"
  197. @change="groupChange($event, index, item)"
  198. >
  199. <el-checkbox
  200. v-for="(sub, i) in item.child"
  201. :key="'sub' + i"
  202. :label="sub.id"
  203. >{{ sub.spec_value }}</el-checkbox
  204. >
  205. </el-checkbox-group>
  206. </div>
  207. <div v-else style="padding: 0 0 12px 0">
  208. <el-tag type="info">暂无规格值</el-tag>
  209. </div>
  210. </div>
  211. </div>
  212. </el-form-item>
  213. <div class="tips-error">{{ ruleForm.spec }}</div>
  214. <el-form-item label="价格" required style="padding: 0 0 12px 0">
  215. <div
  216. class="ladder-table clearfix"
  217. v-if="left && left.header && left && left.header.length > 0"
  218. >
  219. <div class="left" :style="{ width: leftS }">
  220. <div class="ladder-header" v-if="left && left.header">
  221. <ul class="header-spec clearfix">
  222. <li
  223. v-for="(h, hi) in left.header"
  224. :key="'leftHeader' + hi"
  225. >
  226. <span>{{ h.spec_name }}</span>
  227. </li>
  228. </ul>
  229. </div>
  230. <div class="ladder-body" v-if="left.specList">
  231. <ul
  232. v-for="(item, index) in left.specList"
  233. :key="'item' + index"
  234. >
  235. <li
  236. class="clearfix"
  237. v-for="(sub, subi) in item"
  238. :key="'sub' + subi"
  239. >
  240. <span>
  241. {{ sub.spec_value }}
  242. </span>
  243. </li>
  244. </ul>
  245. </div>
  246. </div>
  247. <div class="right" :style="{ paddingLeft: leftS }">
  248. <div class="ladder-header" v-if="right && right.header">
  249. <ul class="header-spec clearfix">
  250. <li
  251. v-for="(h, hi) in right.header"
  252. :key="'rightHeader' + hi"
  253. >
  254. <div class="all">
  255. <div class="title">
  256. <span> 阶梯{{ hi + 1 }}</span>
  257. <i
  258. class="el-icon-close"
  259. @click="closeImg(index)"
  260. ></i>
  261. </div>
  262. <div class="main">
  263. <span>{{ h.num }}</span>
  264. <span>{{ h.price }}</span>
  265. </div>
  266. </div>
  267. </li>
  268. </ul>
  269. </div>
  270. <div class="ladder-body" v-if="right && right.specList">
  271. <ul
  272. v-for="(h, hi) in right.specList"
  273. :key="'rightspec' + hi"
  274. >
  275. <li v-for="(s, si) in h" :key="'rightspecsi' + si">
  276. <div
  277. class="input-main"
  278. v-for="(n, ni) in s.limit"
  279. :key="'rightni' + ni"
  280. >
  281. <el-input-number
  282. style="width: 50%"
  283. v-model="n.num"
  284. :step="1"
  285. :min="0"
  286. :max="99999999"
  287. :precision="0"
  288. placeholder="起订量"
  289. step-strictly
  290. :controls="false"
  291. ></el-input-number>
  292. <el-input-number
  293. style="width: 50%"
  294. v-model="n.price"
  295. :step="0.01"
  296. :min="0"
  297. :max="99999999"
  298. :precision="2"
  299. step-strictly
  300. placeholder="成本价"
  301. :controls="false"
  302. ></el-input-number>
  303. </div>
  304. </li>
  305. </ul>
  306. </div>
  307. </div>
  308. </div>
  309. <div v-else>
  310. <el-tag type="warning">请选择规格值</el-tag>
  311. </div>
  312. </el-form-item>
  313. <div class="tips-error">{{ ruleForm.table }}</div>
  314. </el-form>
  315. </el-col>
  316. <el-col :span="24" style="text-align: right">
  317. <el-button type="primary" @click="step--" v-if="!isDetail"
  318. >上一步
  319. </el-button>
  320. <el-button type="primary" @click="submitForm" v-if="!isDetail"
  321. >保 存
  322. </el-button>
  323. <el-button @click="showModelThis = false">{{
  324. isDetail ? "关 闭" : "取 消"
  325. }}</el-button>
  326. </el-col>
  327. </el-row>
  328. </el-col>
  329. </el-row>
  330. </el-card>
  331. </el-dialog>
  332. </template>
  333. <script>
  334. import asyncRequest from "@/apis/service/goodStore/dealGoodsPool";
  335. export default {
  336. name: "dealGoodsPool",
  337. props: ["showModel", "id", "isDetail"],
  338. data() {
  339. return {
  340. leftS: "0px",
  341. size: "small",
  342. mock: [],
  343. ladderNum: 5, //阶梯数
  344. left: {
  345. header: [],
  346. middle: [],
  347. specList: [],
  348. },
  349. right: {
  350. header: [],
  351. specList: [],
  352. },
  353. headLadder: {
  354. num: "起订量≥",
  355. price: "成本价",
  356. },
  357. catOptions: [],
  358. brandidOptions: [],
  359. supplierOptions: [],
  360. oldcatid: [],
  361. stepList: ["基础信息", "规格与价格"],
  362. step: 0,
  363. loading: false,
  364. title: "添加商品",
  365. showModelThis: this.showModel,
  366. baseForm: {
  367. id: this.id,
  368. catid: [],
  369. good_name: "",
  370. unit: "",
  371. supplier: "",
  372. brandid: "",
  373. image: [],
  374. },
  375. baseRules: {
  376. supplier: [
  377. {
  378. required: true,
  379. message: "请选择供应商联系人",
  380. trigger: "change",
  381. },
  382. ],
  383. image: [
  384. {
  385. required: true,
  386. message: "请选择商品主图",
  387. trigger: "change",
  388. },
  389. ],
  390. catid: [
  391. {
  392. required: true,
  393. message: "请选择商品分类",
  394. trigger: "change",
  395. },
  396. ],
  397. brandid: [
  398. {
  399. required: true,
  400. message: "请选择商品品牌",
  401. trigger: "change",
  402. },
  403. ],
  404. good_name: [
  405. {
  406. required: true,
  407. message: "请输入商品名称",
  408. trigger: "blur",
  409. },
  410. {
  411. min: 2,
  412. max: 50,
  413. message: "长度在 2 到 50 个字符",
  414. trigger: "blur",
  415. },
  416. ],
  417. unit: [
  418. {
  419. required: true,
  420. message: "请输入商品单位",
  421. trigger: "blur",
  422. },
  423. {
  424. min: 1,
  425. max: 10,
  426. message: "长度在 1 到 10 个字符",
  427. trigger: "blur",
  428. },
  429. ],
  430. },
  431. ruleForm: {
  432. spec: "",
  433. table: "",
  434. },
  435. };
  436. },
  437. methods: {
  438. closeModel() {
  439. this.showModelThis = false;
  440. },
  441. async initForm() {
  442. this.mock = [];
  443. this.left = {
  444. header: [],
  445. middle: [],
  446. specList: [],
  447. };
  448. this.right = {
  449. header: [],
  450. specList: [],
  451. };
  452. this.step = 0;
  453. this.loading = true;
  454. // brandidOptions
  455. await this.getAllbrandid();
  456. await this.getAllCat();
  457. await this.getAllSupplier();
  458. if (this.id === "add") {
  459. this.title = "添加商品";
  460. // this.ruleForm.isAdmin = 0;
  461. await this.resetForm();
  462. } else {
  463. if (this.isDetail) {
  464. this.title = "商品详情";
  465. } else {
  466. this.title = "修改商品";
  467. }
  468. await this.resetForm();
  469. await this.initData();
  470. }
  471. this.loading = false;
  472. },
  473. async initData() {
  474. this.loading = true;
  475. let res = await asyncRequest.detail({ id: this.id });
  476. this.loading = false;
  477. if (res.code === 0) {
  478. await this.resetForm(res.data);
  479. }
  480. },
  481. async resetForm(data) {
  482. // 重置
  483. await this.$nextTick(async () => {
  484. if (this.$refs.baseForm) {
  485. this.$refs.baseForm.resetFields();
  486. this.$refs.baseForm.clearValidate();
  487. if (data) {
  488. this.baseForm = {
  489. id: this.id,
  490. catid: data.good_cat.split(","),
  491. good_name: data.good_name,
  492. unit: data.good_unit,
  493. supplier: data.supplier,
  494. brandid: data.brand_id,
  495. image: data.galley.split(","),
  496. };
  497. this.oldcatid = this.baseForm.catid;
  498. await this.specByCat(false);
  499. this.dealWithEditMock(data.attr);
  500. } else {
  501. this.baseForm = {
  502. id: this.id,
  503. catid: ["1", "2", "3"],
  504. good_name:
  505. "2018春秋新款V领修身丝绒西装女 双排扣复古极简中长款工装外套",
  506. unit: "件",
  507. supplier: "8",
  508. brandid: "3",
  509. image: [
  510. "http://cum.sit.wanyuhengtong.com/20210425/1edfa0da4eee9b24d1f35d52b6a0ec0a.jpg",
  511. "http://cum.sit.wanyuhengtong.com/20210425/cc70a4c06c83d9d8ac27ae35b118a23f.jpg",
  512. "http://cum.sit.wanyuhengtong.com/20210425/a5e05dc1c5c75030bcc95f1da1d2d9cb.jpg",
  513. "http://cum.sit.wanyuhengtong.com/20210425/9d505e7632891459a29f2fafe28d6705.jpg",
  514. "http://cum.sit.wanyuhengtong.com/20210425/714914d7545184e99e240b3affeada07.jpg",
  515. ],
  516. };
  517. }
  518. this.ruleForm = {
  519. spec: "",
  520. table: "",
  521. };
  522. }
  523. });
  524. },
  525. dealWithEditMock(attr) {
  526. console.log(this.mock);
  527. attr.forEach((v1) => {
  528. if (v1 && v1.spec && v1.spec.length > 0) {
  529. v1.spec.forEach((v2) => {
  530. this.mock.forEach((v3, i) => {
  531. if (v2.spec_id === v3.spec_id) {
  532. console.log(v2.spec_info_id);
  533. console.log(v3);
  534. let index = v3.change.findIndex((v4) => {
  535. v4 === v2.spec_info_id;
  536. });
  537. if (index === -1) {
  538. v3.change.push(v2.spec_info_id);
  539. }
  540. }
  541. this.$set(this.mock, i, v3);
  542. });
  543. });
  544. }
  545. });
  546. // console.log(attr);
  547. console.log(this.mock);
  548. this.groupChange();
  549. this.resume(attr);
  550. },
  551. resume(attr) {
  552. this.right.specList.forEach((v1) => {
  553. v1.forEach((v2) => {
  554. attr.forEach((v3) => {
  555. let is = true;
  556. v3.spec.forEach((v4) => {
  557. let index = v2.specid.findIndex((v5) => v5 === v4.spec_info_id);
  558. if (index === -1) {
  559. is = false;
  560. }
  561. });
  562. if (is) {
  563. v2.limit.forEach((v6, i) => {
  564. if (v3.limit[i]) {
  565. v6.num = v3.limit[i].begin_num;
  566. v6.price = v3.limit[i].shop_price;
  567. }
  568. });
  569. }
  570. });
  571. });
  572. });
  573. console.log(attr);
  574. console.log(this.right.specList);
  575. },
  576. async submitForm() {
  577. await this.$refs.ruleForm.validate(async (valid) => {
  578. if (valid) {
  579. let change = false;
  580. this.mock.forEach((v1) => {
  581. if (v1.change && v1.change.length > 0) {
  582. change = true;
  583. }
  584. });
  585. if (!change) {
  586. this.ruleForm.spec = "请选择规格!";
  587. return;
  588. } else {
  589. this.ruleForm.spec = "";
  590. let table = false;
  591. let hasnot = false;
  592. let stock = [];
  593. this.right.specList.forEach((v1) => {
  594. if (v1) {
  595. v1.forEach((v2) => {
  596. if (v2 && v2.limit && v2.limit.length > 0) {
  597. let list = [];
  598. v2.limit.forEach((v3) => {
  599. if (v3.num != 0 && v3.price != 0) {
  600. table = true;
  601. list.push(v3);
  602. }
  603. if (
  604. (v3.num != 0 && v3.price == 0) ||
  605. (v3.num == 0 && v3.price !== 0)
  606. ) {
  607. hasnot = true;
  608. }
  609. });
  610. if (list.length > 0) {
  611. let model = Object.assign({}, v2);
  612. model.limit = list;
  613. stock.push(model);
  614. }
  615. }
  616. });
  617. }
  618. });
  619. if (hasnot) {
  620. this.ruleForm.table =
  621. "请选择起订量与成本价需成对填写,未成对请设为零!";
  622. } else {
  623. if (!table) {
  624. this.ruleForm.table = "至少填写一堆起订量与成本价!";
  625. } else {
  626. this.ruleForm.table = "";
  627. this.loading = true;
  628. let obj = JSON.parse(JSON.stringify(this.baseForm));
  629. obj.stock = stock;
  630. obj.catid = obj.catid[obj.catid.length - 1];
  631. obj.image = obj.image.join();
  632. //split(",");
  633. console.log(obj);
  634. let res = {};
  635. if (this.id === "add") {
  636. delete obj["id"];
  637. res = await asyncRequest.add(obj);
  638. } else {
  639. res = await asyncRequest.update(obj);
  640. }
  641. this.loading = false;
  642. if (res.code === 0) {
  643. let title = this.id === "add" ? "添加成功" : "修改成功";
  644. this.$notify.success({
  645. title,
  646. message: "",
  647. });
  648. this.showModelThis = false;
  649. // 刷新
  650. this.$emit("refresh");
  651. }
  652. }
  653. }
  654. }
  655. } else {
  656. console.log("error submit!!");
  657. return false;
  658. }
  659. });
  660. },
  661. async baseSubmitForm() {
  662. await this.$refs.baseForm.validate(async (valid) => {
  663. if (valid) {
  664. await this.handleChange();
  665. }
  666. });
  667. },
  668. groupChange($event, index, item) {
  669. this.$set(this.mock, index, item);
  670. this.actionSpecList();
  671. },
  672. actionSpecList() {
  673. // this.leftS = "0px";
  674. this.left.header = [];
  675. this.left.middle = [];
  676. this.left.specList = [];
  677. console.log(this.mock);
  678. console.log(this.right.specList);
  679. this.mock.forEach((v1) => {
  680. if (
  681. v1.change &&
  682. v1.change.length > 0 &&
  683. v1.child &&
  684. v1.child.length > 0
  685. ) {
  686. this.left.header.push({
  687. spec_id: v1.spec_id,
  688. spec_name: v1.spec_name,
  689. });
  690. let arr = [];
  691. let changeList = v1.change;
  692. changeList.forEach((v2) => {
  693. let index = v1.child.findIndex((v3) => v2 === v3.id);
  694. if (index !== -1) {
  695. let x = {
  696. spec_id: v1.spec_id,
  697. spec_name: v1.spec_name,
  698. id: v1.child[index].id,
  699. spec_value: v1.child[index].spec_value,
  700. };
  701. arr.push(x);
  702. }
  703. });
  704. this.left.middle.push(arr);
  705. }
  706. });
  707. let lang = this.left.header.length;
  708. this.leftS = lang * 70 + "px";
  709. if (this.left.middle.length > 1) {
  710. this.left.specList = this.doExchange(this.left.middle);
  711. } else if (this.left.middle.length === 0) {
  712. this.left.specList = [];
  713. } else {
  714. let list1 = [];
  715. this.left.middle.forEach((v1) => {
  716. if (v1) {
  717. v1.forEach((v2) => {
  718. let list2 = [];
  719. list2.push(v2);
  720. list1.push(list2);
  721. });
  722. }
  723. });
  724. this.left.specList = list1;
  725. }
  726. this.right.header = [];
  727. for (let i = 0; i < this.ladderNum; i++) {
  728. this.right.header.push(this.headLadder);
  729. }
  730. this.right.specList = [];
  731. this.left.specList.forEach((v1) => {
  732. let arr = [];
  733. let model = {
  734. specid: [],
  735. limit: [],
  736. };
  737. v1.forEach((v2) => {
  738. model.specid.push(v2.id);
  739. });
  740. for (let i = 0; i < this.ladderNum; i++) {
  741. let bodyLadder = {
  742. num: 0,
  743. price: 0,
  744. };
  745. model.limit.push(bodyLadder);
  746. }
  747. arr.push(model);
  748. this.right.specList.push(arr);
  749. });
  750. console.log(this.right.specList);
  751. },
  752. closeLadder(index) {
  753. console.log(index);
  754. },
  755. async handleChange(type) {
  756. if (this.baseForm.catid !== this.oldcatid && this.oldcatid.length === 0) {
  757. await this.specByCat();
  758. if (this.step === 0 && !type) {
  759. this.step = 1;
  760. }
  761. } else if (
  762. this.baseForm.catid !== this.oldcatid &&
  763. this.oldcatid.length > 0
  764. ) {
  765. await this.$confirm(
  766. "分类修改会影响该商品规格与价格!",
  767. "确定要修改?",
  768. {
  769. confirmButtonText: "确定",
  770. cancelButtonText: "取消",
  771. type: "warning",
  772. }
  773. )
  774. .then(async () => {
  775. await this.specByCat();
  776. if (this.step === 0) {
  777. this.step = 1;
  778. }
  779. })
  780. .catch(() => {
  781. this.baseForm.catid = this.oldcatid;
  782. });
  783. } else {
  784. if (this.step === 0 && !type) {
  785. this.step = 1;
  786. }
  787. }
  788. },
  789. // 商品种类ID换取规格值
  790. async specByCat(type) {
  791. console.log("准备拉取!");
  792. this.mock = [];
  793. this.$set(this.mock);
  794. let list = this.baseForm.catid;
  795. let res = await asyncRequest.specByCat({ catid: list[list.length - 1] });
  796. if (res.code === 0) {
  797. this.dealWitHInitMock(res.data, type);
  798. this.oldcatid = list;
  799. } else if (res.code !== 101 && res.code !== 102) {
  800. this.baseForm.catid = this.oldcatid;
  801. }
  802. },
  803. // 规格属性值新建
  804. async specinfoadd(id, value) {
  805. console.log(id, value);
  806. let list = this.baseForm.catid;
  807. let model = {
  808. catid: list[list.length - 1],
  809. specid: id,
  810. spec_value: value,
  811. };
  812. let res = await asyncRequest.specinfoadd(model);
  813. if (res.code === 0) {
  814. await this.specByCat(true);
  815. }
  816. },
  817. dealWitHInitMock(list, type) {
  818. console.log(list, type);
  819. if (!type) {
  820. this.mock = [];
  821. console.log("12345678");
  822. console.log(this.mock);
  823. this.left.header = [];
  824. this.left.specList = [];
  825. this.right.header = [];
  826. this.right.specList = [];
  827. list.map((v1) => {
  828. v1.value = "";
  829. v1.change = [];
  830. return v1;
  831. });
  832. this.mock = list;
  833. console.log(this.mock);
  834. this.mock.forEach((v3, i) => {
  835. this.$set(this.mock, i, v3);
  836. });
  837. } else {
  838. let arr = JSON.parse(JSON.stringify(this.mock));
  839. list.map((v1) => {
  840. v1.value = "";
  841. let index = arr.findIndex((v2) => v2.spec_id === v1.spec_id);
  842. if (index !== -1) {
  843. v1.change = arr[index].change;
  844. }
  845. return v1;
  846. });
  847. this.mock = list;
  848. this.mock.forEach((v3, i) => {
  849. this.$set(this.mock, i, v3);
  850. });
  851. }
  852. },
  853. async getAllbrandid() {
  854. const res = await asyncRequest.brandAll({});
  855. if (res.code === 0 && res.data) {
  856. this.brandidOptions = res.data;
  857. }
  858. },
  859. async getAllCat() {
  860. const res = await asyncRequest.catAll({});
  861. if (res.code === 0 && res.data) {
  862. let list = res.data;
  863. list.map((v1) => {
  864. v1.value = v1.id;
  865. v1.label = v1.cat_name;
  866. if (v1.child && v1.child.length > 0) {
  867. v1.child.map((v2) => {
  868. v2.value = v2.id;
  869. v2.label = v2.cat_name;
  870. if (v2.child && v2.child.length > 0) {
  871. v2.child.map((v3) => {
  872. v3.value = v3.id;
  873. v3.label = v3.cat_name;
  874. return v3;
  875. });
  876. v2.children = v2.child;
  877. }
  878. return v2;
  879. });
  880. v1.children = v1.child;
  881. }
  882. return v1;
  883. });
  884. this.catOptions = list;
  885. }
  886. },
  887. async getAllSupplier() {
  888. const res = await asyncRequest.supplierAll({});
  889. if (res.code === 0 && res.data) {
  890. let list = res.data;
  891. this.supplierOptions = list;
  892. }
  893. },
  894. //图片上传失败
  895. UploadErrorEvent(res) {
  896. if (res !== "break") {
  897. this.$message.error("图片上传失败!");
  898. this.$refs.ruleForm.validateField("image");
  899. }
  900. },
  901. closeImg(index) {
  902. this.baseForm.image.splice(index, 1);
  903. this.$refs.baseForm.validateField("image");
  904. },
  905. //图片上传成功
  906. async UploadSuccessEvent(data) {
  907. const { url } = data;
  908. if (url === "noToken") {
  909. await this.logout();
  910. } else {
  911. this.baseForm.image.push(url);
  912. this.$message.success("图片上传成功!");
  913. this.$refs.baseForm.validateField("image");
  914. }
  915. },
  916. //判断图片规格
  917. beforeAvatarUpload(file) {
  918. console.log(file);
  919. let isJPG = false;
  920. if (
  921. file.type === "image/jpg" ||
  922. file.type === "image/png" ||
  923. file.type === "image/jpeg"
  924. ) {
  925. isJPG = true;
  926. }
  927. const isLt2M = file.size / 1024 / 1024 < 1;
  928. if (!isJPG) {
  929. this.$message.error("图片格式不正确!");
  930. }
  931. if (!isLt2M) {
  932. this.$message.error("图片大小不能超过 1MB!");
  933. }
  934. return isJPG && isLt2M;
  935. },
  936. doExchange(arr) {
  937. let len = arr.length;
  938. // 当数组大于等于2个的时候
  939. if (len >= 2) {
  940. // 第一个数组的长度
  941. let len1 = arr[0].length;
  942. // 第二个数组的长度
  943. let len2 = arr[1].length;
  944. // 2个数组产生的组合数
  945. let lenBoth = len1 * len2;
  946. // 申明一个新数组
  947. let items = new Array(lenBoth);
  948. // 申明新数组的索引
  949. let index = 0;
  950. for (let i = 0; i < len1; i++) {
  951. for (let j = 0; j < len2; j++) {
  952. if (arr[0][i] instanceof Array) {
  953. items[index] = arr[0][i].concat(arr[1][j]);
  954. } else {
  955. items[index] = [arr[0][i]].concat(arr[1][j]);
  956. }
  957. index++;
  958. }
  959. }
  960. let newArr = new Array(len - 1);
  961. for (let i = 2; i < arr.length; i++) {
  962. newArr[i - 1] = arr[i];
  963. }
  964. newArr[0] = items;
  965. return this.doExchange(newArr);
  966. } else {
  967. return arr[0];
  968. }
  969. },
  970. },
  971. watch: {
  972. showModel: function (val) {
  973. this.showModelThis = val;
  974. if (val) {
  975. this.initForm();
  976. }
  977. },
  978. showModelThis(val) {
  979. if (!val) {
  980. this.$emit("cancel");
  981. }
  982. },
  983. },
  984. };
  985. </script>
  986. <style lang="scss" scoped>
  987. .dealGoodsPool {
  988. $dealGoodsPoolBoderColor: #dfe6ec;
  989. .tips-error {
  990. color: #ff8888;
  991. font-size: 12px;
  992. height: 22px;
  993. line-height: 20px;
  994. padding-left: 60px;
  995. }
  996. .ladder-table {
  997. position: relative;
  998. width: 100%;
  999. border: 1px solid $dealGoodsPoolBoderColor;
  1000. .left {
  1001. position: absolute;
  1002. top: 0;
  1003. left: 0;
  1004. z-index: 3;
  1005. background: #fff;
  1006. // width: 140px;
  1007. .ladder-header {
  1008. position: relative;
  1009. width: 100%;
  1010. .header-spec {
  1011. position: relative;
  1012. // border-bottom: 1px solid $dealGoodsPoolBoderColor;
  1013. li {
  1014. float: left;
  1015. width: 70px;
  1016. height: 50px;
  1017. line-height: 50px;
  1018. overflow: hidden;
  1019. padding: 0 10px;
  1020. font-size: 12px;
  1021. color: #909399;
  1022. border-right: 1px solid $dealGoodsPoolBoderColor;
  1023. span {
  1024. height: 25px;
  1025. line-height: 25px;
  1026. min-height: 50px;
  1027. overflow: hidden;
  1028. }
  1029. }
  1030. }
  1031. }
  1032. .ladder-body {
  1033. position: relative;
  1034. width: 100%;
  1035. ul {
  1036. width: 100%;
  1037. display: flex;
  1038. li {
  1039. flex: 1;
  1040. height: 40px;
  1041. line-height: 40px;
  1042. padding: 0 10px;
  1043. // border-top: 1px solid $dealGoodsPoolBoderColor;
  1044. // border-right: 1px solid $goodsBoderColor;
  1045. span {
  1046. display: block;
  1047. height: 40px;
  1048. width: 49px;
  1049. overflow: hidden;
  1050. line-height: 40px;
  1051. color: #606266;
  1052. font-size: 12px;
  1053. text-overflow: ellipsis;
  1054. white-space: normal;
  1055. word-break: break-all;
  1056. // overflow: hidden;
  1057. // text-overflow:ellipsis;
  1058. // white-space: nowrap;
  1059. }
  1060. }
  1061. }
  1062. }
  1063. }
  1064. .right {
  1065. position: relative;
  1066. // padding-left: 140px;
  1067. width: 100%;
  1068. overflow-x: scroll;
  1069. background: #fff;
  1070. .ladder-header {
  1071. position: relative;
  1072. width: 100%;
  1073. .header-spec {
  1074. display: flex;
  1075. li {
  1076. min-width: 240px;
  1077. flex: 1;
  1078. .all {
  1079. width: 100%;
  1080. .title {
  1081. width: 100%;
  1082. height: 25px;
  1083. padding: 0 10px;
  1084. font-size: 12px;
  1085. color: #909399;
  1086. line-height: 25px;
  1087. text-align: center;
  1088. position: relative;
  1089. // border-right: 1px solid $goodsBoderColor;
  1090. // border-bottom: 1px solid $goodsBoderColor;
  1091. i {
  1092. position: absolute;
  1093. top: 2px;
  1094. right: 2px;
  1095. z-index: 2;
  1096. &:hover {
  1097. cursor: pointer;
  1098. }
  1099. }
  1100. }
  1101. .main {
  1102. width: 100%;
  1103. height: 25px;
  1104. span {
  1105. vertical-align: top;
  1106. width: 50%;
  1107. height: 25px;
  1108. line-height: 25px;
  1109. text-align: center;
  1110. padding: 0 10px;
  1111. font-size: 12px;
  1112. color: #909399;
  1113. line-height: 25px;
  1114. display: inline-block;
  1115. // border-right: 1px solid $goodsBoderColor;
  1116. // border-bottom: 1px solid $goodsBoderColor;
  1117. }
  1118. }
  1119. }
  1120. }
  1121. }
  1122. }
  1123. .ladder-body {
  1124. position: relative;
  1125. width: 100%;
  1126. ul {
  1127. width: 100%;
  1128. position: relative;
  1129. li {
  1130. width: 100%;
  1131. display: flex;
  1132. .input-main {
  1133. flex: 1;
  1134. min-width: 240px;
  1135. .el-input-number {
  1136. .el-input {
  1137. input.el-input_inner {
  1138. border-radius: 0;
  1139. }
  1140. }
  1141. }
  1142. }
  1143. }
  1144. }
  1145. }
  1146. }
  1147. }
  1148. .mack-mian {
  1149. position: relative;
  1150. width: 100%;
  1151. .mock-item-div {
  1152. width: 100%;
  1153. background: rgba(246, 247, 248, 1);
  1154. padding: 12px 12px 0 12px;
  1155. margin: 0 0 10px 0;
  1156. .mock-title {
  1157. width: 100%;
  1158. padding: 0 0 5px 0;
  1159. border-bottom: 1px solid rgba(226, 226, 226, 1);
  1160. }
  1161. }
  1162. }
  1163. }
  1164. </style>