step-1.vue 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627
  1. <template>
  2. <el-card class="step-card">
  3. <div
  4. slot="header"
  5. class="flex between">
  6. <div class="step-card-title">Step1 Choose model</div>
  7. <div>&gt;&gt;</div>
  8. </div>
  9. <div v-loading="loading">
  10. <div class="card-sub-title">{{ detail.product_name }}</div>
  11. <image-viewer
  12. v-if="isBigImageShow"
  13. :on-close="closeBigImage"
  14. :initial-index="0"
  15. :url-list="detail.main?.images" />
  16. <div
  17. class="img-wrap flex center"
  18. @click="openBigImage">
  19. <img :src="detail.main?.image" />
  20. </div>
  21. <div class="divider"></div>
  22. <el-form
  23. :model="form"
  24. :rules="rules"
  25. label-width="90px"
  26. ref="form">
  27. <el-form-item
  28. label="Time frame"
  29. prop="cycle">
  30. <el-radio-group v-model="form.cycle">
  31. <el-radio
  32. v-for="item in priceData.priceList"
  33. :key="item.cycle_id"
  34. :label="item.cycle_id"
  35. >{{ item.name }}
  36. </el-radio>
  37. </el-radio-group>
  38. </el-form-item>
  39. <div class="flex">
  40. <el-form-item
  41. label-width="120px"
  42. label="Decorated in"
  43. prop="decorated">
  44. <el-select
  45. v-model="form.decorated"
  46. size="small">
  47. <el-option
  48. :label="'Local'"
  49. :value="'Local'"></el-option>
  50. <el-option
  51. :label="'China'"
  52. :value="'China'"></el-option>
  53. </el-select>
  54. </el-form-item>
  55. <el-form-item
  56. label-width="120px"
  57. prop="supplyChain"
  58. label="Supply chain"
  59. size="small">
  60. <el-select v-model="form.supplyChain">
  61. <el-option
  62. :label="'AU Stock'"
  63. :value="'AU Stock'"></el-option>
  64. <el-option
  65. :label="'Air Freight'"
  66. :value="'Air Freight'"></el-option>
  67. <el-option
  68. :label="'Sea Freight'"
  69. :value="'Sea Freight'"></el-option>
  70. </el-select>
  71. </el-form-item>
  72. </div>
  73. <el-form-item
  74. label="Model"
  75. prop="model">
  76. <div class="custom-checkbox-wrap">
  77. <div
  78. class="custom-checkbox"
  79. :class="{
  80. active: form.model === item.id,
  81. disable: form.model && form.model !== item.id,
  82. }"
  83. v-for="item in computedBasePriceData"
  84. @click="modelClick(item)"
  85. :key="item.id">
  86. {{ item.point }}
  87. </div>
  88. </div>
  89. <el-radio-group
  90. v-model="form.model"
  91. style="display: none"
  92. size="small">
  93. <el-radio-button
  94. v-for="item in computedBasePriceData"
  95. :key="item.id"
  96. :label="item.id"
  97. >{{ item.point }}</el-radio-button
  98. >
  99. </el-radio-group>
  100. </el-form-item>
  101. <el-form-item
  102. label="Colour"
  103. prop="color">
  104. <div class="custom-checkbox-wrap">
  105. <div
  106. class="custom-checkbox"
  107. :class="{
  108. active: form.color.includes(0),
  109. }"
  110. @click="colorClick({ id: 0 })">
  111. Unspecified
  112. </div>
  113. <div
  114. class="custom-checkbox flex"
  115. :class="{
  116. active: form.color.includes(item.id),
  117. }"
  118. v-for="item in detail.color"
  119. @click="colorClick(item)"
  120. :key="item.id">
  121. <div class="color-selector-label flex center">
  122. <img :src="item.img" />
  123. </div>
  124. <div>
  125. {{ item.name }}
  126. </div>
  127. </div>
  128. <div
  129. class="custom-checkbox"
  130. :class="{
  131. active: form.color.includes(999),
  132. }"
  133. @click="colorClick({ id: 999 })">
  134. + PMS
  135. </div>
  136. </div>
  137. <el-checkbox-group
  138. v-model="form.color"
  139. style="display: none"
  140. size="small">
  141. <el-checkbox-button :label="0">Unspecified</el-checkbox-button>
  142. <el-checkbox-button
  143. v-for="item in detail.color"
  144. :label="item.id"
  145. :key="item.id">
  146. {{ item.name }}
  147. </el-checkbox-button>
  148. <el-checkbox-button :label="999">+ PMS</el-checkbox-button>
  149. </el-checkbox-group>
  150. </el-form-item>
  151. <el-form-item prop="colorPmsText">
  152. <el-input
  153. v-show="form.color.includes(999)"
  154. type="textarea"
  155. :rows="5"
  156. placeholder="Product color comments. Multi color instructions, specific color matchong (CMYK, RGB, PMS) etc.."
  157. v-model="form.colorPmsText">
  158. </el-input>
  159. </el-form-item>
  160. <div class="qty-title">
  161. QTY&nbsp;(&nbsp;MOQ:{{ computedMinBuyNumber }}&nbsp;)
  162. </div>
  163. <div
  164. class="divider"
  165. v-show="form.color.length > 0"></div>
  166. <div class="flex wrap">
  167. <el-form-item
  168. class="flex-auto color-form-item"
  169. label-width="140"
  170. v-for="(formItem, index) in form.autoForm"
  171. :prop="`autoForm.${index}.value`"
  172. :rules="{
  173. required: true,
  174. trigger: 'change',
  175. type: 'number',
  176. message: 'the number should not less then MOQ',
  177. min: computedMinBuyNumber,
  178. }"
  179. :key="index">
  180. <div class="flex end">
  181. <div class="flex-auto flex end">
  182. <img
  183. class="auto-form-color-image"
  184. v-if="formItem.img"
  185. :src="formItem.img"
  186. alt="" />
  187. <div class="auto-form-label form-item-label">
  188. {{ formItem.label }}
  189. </div>
  190. </div>
  191. <el-input
  192. @change="
  193. value => {
  194. if (value >= computedMinBuyNumber)
  195. $refs.form.clearValidate(`autoForm.${index}.value`)
  196. }
  197. "
  198. style="width: 80px"
  199. :min="computedMinBuyNumber"
  200. type="number"
  201. v-model.number="formItem.value"></el-input>
  202. </div>
  203. </el-form-item>
  204. </div>
  205. </el-form>
  206. </div>
  207. </el-card>
  208. </template>
  209. <script>
  210. import imageViewer from 'element-ui/packages/image/src/image-viewer'
  211. import stepMixin from './stepMixin'
  212. export default {
  213. name: 'Step1',
  214. components: {
  215. 'image-viewer': imageViewer,
  216. },
  217. mixins: [stepMixin],
  218. props: {
  219. // 步骤2的表单. 因其打印价格表单会反向影响本组件的最小起购数量, 故需从父组件获取. 只用来计算, 本组件不更改这个数据.
  220. form2: {
  221. type: Object,
  222. default: function () {
  223. return {}
  224. },
  225. },
  226. },
  227. data() {
  228. // this.detail 记录在mixin里;
  229. return {
  230. // 查看大图
  231. isBigImageShow: false,
  232. rules: {
  233. decorated: [{ required: true, message: '请选择', trigger: 'change' }],
  234. supplyChain: [{ required: true, message: '请选择', trigger: 'change' }],
  235. model: [
  236. {
  237. required: true,
  238. trigger: 'change',
  239. type: 'number',
  240. min: 1,
  241. message: 'Please select model',
  242. },
  243. ],
  244. color: [
  245. {
  246. required: true,
  247. validator: (rules, value, cb) => {
  248. value.length > 0
  249. ? cb()
  250. : cb(new Error('Please select at least one color'))
  251. },
  252. trigger: 'change',
  253. },
  254. ],
  255. colorPmsText: [{ trigger: 'blur', validator: this.checkColorPmsText }],
  256. cycle: [
  257. {
  258. required: true,
  259. message: 'Please select time frame',
  260. trigger: 'change',
  261. },
  262. ],
  263. },
  264. form: {
  265. // 必须要有的初始值, 否则必定会报错Cannot read properties of undefined
  266. color: [],
  267. model: 0,
  268. colorPmsText: '',
  269. },
  270. // 标志符,用于控制监听是否启用
  271. loaded: false,
  272. }
  273. },
  274. computed: {
  275. // 商品 当前选中周期的 基础价格和打印价格数据
  276. computedPriceData() {
  277. return this.priceData.priceList.filter(
  278. item => item.cycle_id === this.form.cycle
  279. )
  280. },
  281. // 当前选中周期下, 商品对应的各型号基础价格数据. 可以推断当前周期有几个型号
  282. computedBasePriceData() {
  283. const model = this.computedPriceData.length
  284. ? this.computedPriceData[0]
  285. : {}
  286. // 属性‘1’里面是基础价格数据, 属性 ’2‘是打印价格数据
  287. if (model['1']) {
  288. return model['1'].slice()
  289. } else {
  290. return []
  291. }
  292. },
  293. // 当前型号位于基础价格数据的index, 用于取出型号对应的基础数据
  294. computedBasePriceIndex() {
  295. return this.computedBasePriceData.findIndex(
  296. item => item.id === this.form.model
  297. )
  298. },
  299. // 该商品周期的最小起订量. attributeList的属性名(website_qty1, website_qty2)排序, 然后取属性值
  300. computedMinBuyNumber() {
  301. // 算基础价格的第一个有效值
  302. const target = this.computedBasePriceData.length
  303. ? this.computedBasePriceData[
  304. this.computedBasePriceIndex > 0 ? this.computedBasePriceIndex : 0
  305. ]
  306. : {}
  307. const candidate = [
  308. target.website_qty1,
  309. target.website_qty2,
  310. target.website_qty3,
  311. target.website_qty4,
  312. target.website_qty5,
  313. target.website_qty6,
  314. target.website_qty7,
  315. target.website_qty8,
  316. ]
  317. if (!candidate.some(item => item !== undefined)) return 1
  318. // 算出商品基础价格的第一个有效值
  319. let index = this.findEffectIndex(candidate)
  320. // 算出打印服务和附加服务的第一个有效价格所处位置.
  321. Object.entries(this.form2).forEach(current => {
  322. if (/\d+/.test(current[0])) {
  323. const decoration = current[1].decorationList.filter(
  324. i => i.id === current[1].printService
  325. )
  326. if (decoration.length && current[1].enable) {
  327. // 打印价格的基础阶梯价格
  328. const index2 = this.findEffectIndex([
  329. decoration[0].website_qty1,
  330. decoration[0].website_qty2,
  331. decoration[0].website_qty3,
  332. decoration[0].website_qty4,
  333. decoration[0].website_qty5,
  334. decoration[0].website_qty6,
  335. decoration[0].website_qty7,
  336. decoration[0].website_qty8,
  337. ])
  338. // 附加价格
  339. const index3 = this.findEffectIndex([
  340. decoration[0].supplier_qty1,
  341. decoration[0].supplier_qty2,
  342. decoration[0].supplier_qty3,
  343. decoration[0].supplier_qty4,
  344. decoration[0].supplier_qty5,
  345. decoration[0].supplier_qty6,
  346. decoration[0].supplier_qty7,
  347. decoration[0].supplier_qty8,
  348. ])
  349. // 让index记录最大值
  350. if (index2 > index) index = index2
  351. if (index3 > index) index = index3
  352. }
  353. } else {
  354. // 附加服务的价格
  355. const additions = this.priceData.additionList[current[0]].filter(
  356. item => current[1].includes(item.id)
  357. )
  358. if (additions.length) {
  359. additions.forEach(target => {
  360. const index4 = this.findEffectIndex([
  361. target.website_qty1,
  362. target.website_qty2,
  363. target.website_qty3,
  364. target.website_qty4,
  365. target.website_qty5,
  366. target.website_qty6,
  367. target.website_qty7,
  368. target.website_qty8,
  369. ])
  370. // 让index记录最大值
  371. if (index4 > index) index = index4
  372. })
  373. }
  374. }
  375. })
  376. const temp = Object.entries(this.priceData.attributeList)
  377. .sort((a, b) => a[0].localeCompare(b[0]))
  378. .filter((item, i) => index === i)
  379. if (temp.length) {
  380. return temp[0][1]
  381. } else {
  382. return 1
  383. }
  384. },
  385. },
  386. watch: {
  387. computedBasePriceData() {
  388. if (!this.loaded || !this.computedBasePriceData.length) return
  389. // fixme 如果有从商品详情页带过来的型号选择数据, 要选中该值, 而不是默认的第一个值
  390. if (
  391. this.computedBasePriceData.filter(i => i.id === this.preData.model)
  392. .length
  393. ) {
  394. this.form.model = this.preData.model
  395. } else {
  396. this.form.model = this.computedBasePriceData[0].id
  397. }
  398. },
  399. 'form.model': {
  400. immediate: true,
  401. handler: function () {
  402. if (!this.loaded) return
  403. this.generateColorForm()
  404. },
  405. },
  406. 'form.color': {
  407. immediate: true,
  408. handler: function () {
  409. if (!this.loaded) return
  410. this.generateColorForm()
  411. this.$refs.form.clearValidate('colorPmsText')
  412. },
  413. },
  414. form2: {
  415. immediate: true,
  416. handler: function () {
  417. if (!this.loaded) return
  418. this.generateColorForm()
  419. },
  420. },
  421. },
  422. mounted() {
  423. this.form = {
  424. supplyChain: 'AU Stock',
  425. decorated: 'Local',
  426. model: 0, // 型号的赋值在computedBasePriceData的监听里面, 因为其涉及到基础价格变更后的重新赋值.
  427. cycle: this.preData.cycle
  428. ? this.preData.cycle
  429. : this.priceData.priceList[0].cycle_id,
  430. color: [],
  431. colorPmsText: '',
  432. // 根据所选规格和颜色组合成的自动表单.需要在computedBasePriceData的watch里面生成
  433. autoForm: [],
  434. }
  435. this.loaded = true
  436. },
  437. methods: {
  438. checkForm() {
  439. return new Promise((resolve, reject) => {
  440. this.$refs.form.validate(valid => {
  441. if (valid) {
  442. resolve(JSON.parse(JSON.stringify(this.form)))
  443. } else {
  444. reject(new Error('validate step1 form error'))
  445. }
  446. })
  447. })
  448. },
  449. checkColorPmsText(rule, value, cb) {
  450. // fixme. 颜色的两个固定值的id还未确定
  451. if (!this.form.color.includes(999)) cb()
  452. if (this.form.colorPmsText.trim().length < 1) {
  453. cb(new Error('You must type color comment here if "+ PMS" is checked'))
  454. } else {
  455. cb()
  456. }
  457. },
  458. // 找出第一个有效价格的index. 非变异方法, 勿对参数进行赋值操作.
  459. // 价格留空和111不算到起购量, 为正常数值或者0或者999(POA)就可以算到起购量, 所以在此处将空手动重置为NaN
  460. findEffectIndex(array) {
  461. return array
  462. .map(item => (item === '' ? Number.NaN : Number(item)))
  463. .findIndex(item => !Number.isNaN(item) && item !== 111)
  464. },
  465. openBigImage() {
  466. this.isBigImageShow = true
  467. },
  468. closeBigImage() {
  469. this.isBigImageShow = false
  470. },
  471. generateColorForm() {
  472. if (!this.loaded || this.form.color.length < 1) {
  473. this.form.autoForm = []
  474. return
  475. }
  476. // 生成所选规格和颜色组合成的自动表单
  477. const targetModel = this.computedBasePriceData.filter(
  478. item => item.id === this.form.model
  479. )
  480. const oldForm = JSON.parse(JSON.stringify(this.form.autoForm))
  481. let color1 = []
  482. if (targetModel.length) {
  483. color1 = this.detail.color.reduce((total, item) => {
  484. if (this.form.color.includes(item.id)) {
  485. const temp = {
  486. img: item.img || '',
  487. label: `${item.name}, ${targetModel[0].point}`,
  488. color: item.name,
  489. colorId: item.id,
  490. value: this.computedMinBuyNumber,
  491. }
  492. const temp2 = oldForm.filter(item => item.label === temp.label)
  493. if (temp2.length && temp2[0].value >= this.computedMinBuyNumber) {
  494. // 如果原来有值且原值大于等于最小起购, 则沿用原值
  495. temp.value = temp2[0].value
  496. }
  497. total.push(temp)
  498. }
  499. return total
  500. }, [])
  501. if (this.form.color.includes(0)) {
  502. const temp = {
  503. img: '',
  504. label: `Unspecified, ${targetModel[0].point}`,
  505. color: 'Unspecified',
  506. colorId: -1,
  507. value: this.computedMinBuyNumber,
  508. }
  509. const temp2 = oldForm.filter(item => {
  510. return item.label === temp.label
  511. })
  512. if (temp2.length && temp2[0].value >= this.computedMinBuyNumber) {
  513. // 如果原来有值且原值大于等于最小起购, 则沿用原值
  514. temp.value = temp2[0].value
  515. }
  516. color1.unshift(temp)
  517. }
  518. if (this.form.color.includes(999)) {
  519. const temp = {
  520. img: '',
  521. label: `+ PMS, ${targetModel[0].point}`,
  522. color: 'PMS',
  523. colorId: -1,
  524. value: this.computedMinBuyNumber,
  525. }
  526. const temp2 = oldForm.filter(item => item.label === temp.label)
  527. if (temp2.length && temp2[0].value >= this.computedMinBuyNumber) {
  528. // 如果原来有值且原值大于等于最小起购, 则沿用原值
  529. temp.value = temp2[0].value
  530. }
  531. color1.push(temp)
  532. }
  533. }
  534. this.$set(this.form, 'autoForm', color1)
  535. },
  536. modelClick(item) {
  537. if (this.form.model === item.id) {
  538. this.form.model = 0
  539. } else {
  540. this.form.model = item.id
  541. }
  542. },
  543. colorClick(item) {
  544. const index = this.form.color.indexOf(item.id)
  545. if (index !== -1) {
  546. this.form.color.splice(index, 1)
  547. } else {
  548. this.form.color.push(item.id)
  549. }
  550. },
  551. },
  552. }
  553. </script>
  554. <style lang="scss" scoped>
  555. @import './step.scss';
  556. div {
  557. box-sizing: border-box;
  558. }
  559. .img-wrap {
  560. $width: 140px;
  561. width: $width;
  562. height: $width;
  563. position: relative;
  564. padding: 4px;
  565. border: 1px solid #eee;
  566. cursor: pointer;
  567. margin-bottom: 16px;
  568. img {
  569. width: 100%;
  570. }
  571. }
  572. .el-checkbox-button {
  573. min-width: 76px;
  574. }
  575. .qty-title {
  576. margin-bottom: 16px;
  577. font-size: 14px;
  578. font-family: Proxima Nova, sans-serif;
  579. color: #000;
  580. }
  581. // 选择颜色
  582. .color-selector-label {
  583. width: 16px;
  584. height: 16px;
  585. position: relative;
  586. margin-right: 4px;
  587. img {
  588. width: 100%;
  589. }
  590. }
  591. // 颜色型号对应购买数量输入项
  592. .color-form-item {
  593. box-sizing: border-box;
  594. max-width: 50%;
  595. padding-left: 4px;
  596. .auto-form-label {
  597. text-align: right;
  598. padding: 0 12px;
  599. min-width: 140px;
  600. }
  601. :deep(.el-form-item__error) {
  602. text-align: right;
  603. width: 100%;
  604. }
  605. .auto-form-color-image {
  606. width: 20px;
  607. height: 20px;
  608. margin-right: -4px;
  609. }
  610. }
  611. </style>