step-2.vue 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327
  1. <template>
  2. <el-card class="step-card">
  3. <div
  4. slot="header"
  5. class="flex between">
  6. <div class="step-card-title">Step2 Choose options</div>
  7. <div>&gt;&gt;</div>
  8. </div>
  9. <div v-loading="loading">
  10. <div v-if="computedPrintPriceData.length">
  11. <div class="card-sub-title">Decoration</div>
  12. <el-form
  13. :model="form"
  14. :rules="rules"
  15. label-width="0"
  16. ref="form">
  17. <div
  18. v-for="item in computedPrintPriceData"
  19. :key="item.id">
  20. <div class="divider"></div>
  21. <el-form-item>
  22. <div class="flex">
  23. <el-switch
  24. v-model="form[`${item.id}`].enable"
  25. active-color="#13ce66"
  26. inactive-color="#ff4949"></el-switch>
  27. <div
  28. class="form-item-label"
  29. style="margin-left: 12px">
  30. {{ item.point }}
  31. </div>
  32. </div>
  33. </el-form-item>
  34. <el-form-item>
  35. <div class="custom-checkbox-wrap">
  36. <div
  37. class="custom-checkbox"
  38. :class="{
  39. active: form[`${item.id}`].printService === decoration.id,
  40. disable: !form[`${item.id}`].enable,
  41. }"
  42. v-for="decoration in item.decorationList"
  43. @click="printServiceClick(decoration, item)"
  44. :key="decoration.id">
  45. {{ decoration.dec_code }}
  46. </div>
  47. </div>
  48. <el-radio-group
  49. style="display: none"
  50. v-model="form[`${item.id}`].printService"
  51. :disabled="!form[`${item.id}`].enable">
  52. <el-radio
  53. v-for="(decoration, index) in item.decorationList"
  54. :key="index"
  55. :label="decoration.id"
  56. >{{ decoration.dec_code }}
  57. </el-radio>
  58. </el-radio-group>
  59. </el-form-item>
  60. <div
  61. class="flex start"
  62. v-for="(decoration, index) in item.decorationList"
  63. v-show="decoration.id === form[`${item.id}`].printService"
  64. :key="index">
  65. <div
  66. class="flex-auto"
  67. v-show="decoration.max_color_count">
  68. <div class="form-item-label">Numbers of colours</div>
  69. <div
  70. style="height: 35px"
  71. class="flex">
  72. <el-radio-group
  73. :disabled="!form[`${item.id}`].enable"
  74. v-model="form[`${item.id}`].colorForm[index].colorNumber">
  75. <el-radio
  76. v-for="n in decoration.max_color_count"
  77. :key="n"
  78. :label="n"></el-radio>
  79. </el-radio-group>
  80. </div>
  81. </div>
  82. <div
  83. class="flex-auto"
  84. style="max-width: 50%">
  85. <div class="form-item-label">PMS Colours</div>
  86. <el-form-item
  87. prop="pmsColor"
  88. label-width="0">
  89. <el-input
  90. :disabled="!form[`${item.id}`].enable"
  91. v-model="form[`${item.id}`].colorForm[index].pmsColorText"
  92. size="small"
  93. placeholder="PMS"></el-input>
  94. </el-form-item>
  95. </div>
  96. </div>
  97. </div>
  98. </el-form>
  99. </div>
  100. <div v-if="hasAddition">
  101. <div class="divider"></div>
  102. <br />
  103. <div class="addition-title">Centre Back</div>
  104. <div
  105. class="flex"
  106. v-for="(addition, keyName, index) in priceData.additionList"
  107. :key="index">
  108. <div class="addition-title flex">{{ keyName }}</div>
  109. <div class="flex-auto flex wrap">
  110. <div
  111. @click="selectAddition(item, keyName)"
  112. class="addition-item"
  113. :class="[
  114. { selected: additionForm[`${keyName}`].includes(item.id) },
  115. ]"
  116. v-for="item in addition"
  117. :key="item.id">
  118. <div class="background"></div>
  119. <div class="name">{{ item.name }}</div>
  120. </div>
  121. </div>
  122. </div>
  123. </div>
  124. </div>
  125. </el-card>
  126. </template>
  127. <script>
  128. import stepMixin from './stepMixin'
  129. export default {
  130. name: 'Step2',
  131. mixins: [stepMixin],
  132. props: {
  133. // 步骤1中选中的周期. 不能直接拿form1来取form1.cycle, 否则步骤1变更时, 涉及到form1的computed会全部重新计算, 可能会导致其他异常
  134. cycle: {
  135. type: Number,
  136. default: function () {
  137. return 0
  138. },
  139. },
  140. },
  141. data() {
  142. return {
  143. form: {},
  144. additionForm: {},
  145. rules: {},
  146. // 标志符,用于控制监听是否启用
  147. loaded: false,
  148. }
  149. },
  150. computed: {
  151. // 商品 当前选中周期的 基础价格和打印价格数据
  152. computedPriceData() {
  153. return this.priceData.priceList.filter(
  154. item => item.cycle_id === this.cycle
  155. )
  156. },
  157. /**
  158. * 当前选中周期下, 商品对应的各型号打印价格.
  159. */
  160. computedPrintPriceData() {
  161. const model = this.computedPriceData.length
  162. ? this.computedPriceData[0]
  163. : {}
  164. // 属性‘1’里面是基础价格数据, 属性 ’2‘是打印价格数据
  165. if (model['2']) {
  166. return model['2'].slice()
  167. } else {
  168. return []
  169. }
  170. },
  171. // 计算是否存在附加项
  172. hasAddition() {
  173. if (!this.loaded) return false
  174. let result = false
  175. const target = this.priceData.additionList
  176. for (const key in target) {
  177. if (Array.isArray(target[key]) && target[key].length) {
  178. result = true
  179. }
  180. }
  181. return result
  182. },
  183. },
  184. watch: {
  185. computedPrintPriceData: {
  186. deep: true,
  187. handler: function () {
  188. if (!this.loaded) return
  189. const temp = {}
  190. for (const item of this.computedPrintPriceData) {
  191. let pId = 0
  192. // 从 product 页面来的用户选过的打印服务数据
  193. if (
  194. this.preData.printService &&
  195. this.preData.printService[`${item.id}`] &&
  196. this.preData.printService[`${item.id}`].id
  197. ) {
  198. pId = this.preData.printService[`${item.id}`].id
  199. }
  200. temp[`${item.id}`] = {
  201. // ...item直接把所有数据一起堆进去, 方便后续的价格计算, 不然计算需要各种遍历, 逻辑会非常难懂.
  202. ...item,
  203. enable: pId !== 0,
  204. // 打印服务. 如果选过的打印服务在这个周期的商品里面有, 则选上. 否则选当前打印服务项的第一个
  205. printService: item.decorationList.map(i => i.id).includes(pId)
  206. ? pId
  207. : item.decorationList[0].id,
  208. colorForm: item.decorationList.map(decoration => {
  209. return {
  210. id: decoration.id,
  211. // 默认是1, 后台数据允许留空成0, 但既然计算打印价格, 肯定会是1, 不然怎么打印、怎么计算价格
  212. colorNumber:
  213. pId === decoration.id
  214. ? Number(this.preData.printService[`${item.id}`].num)
  215. : 1,
  216. pmsColorText: '',
  217. }
  218. }),
  219. }
  220. }
  221. this.form = Object.assign(temp, this.additionForm)
  222. },
  223. },
  224. additionForm: {
  225. deep: true,
  226. handler: function () {
  227. if (!this.loaded) return
  228. Object.assign(this.form, this.additionForm)
  229. },
  230. },
  231. },
  232. mounted() {
  233. for (const key in this.priceData.additionList) {
  234. this.$set(
  235. this.additionForm,
  236. `${key}`,
  237. this.preData[`${key}`] ? this.preData[`${key}`].slice() : []
  238. )
  239. }
  240. this.loaded = true
  241. },
  242. methods: {
  243. checkForm() {
  244. return new Promise((resolve, reject) => {
  245. this.$refs.form.validate(valid => {
  246. if (valid) {
  247. resolve(JSON.parse(JSON.stringify(this.form)))
  248. } else {
  249. reject(new Error('validate step2 form error'))
  250. }
  251. })
  252. })
  253. },
  254. selectAddition(item, key) {
  255. if (this.additionForm[`${key}`].includes(item.id)) {
  256. // 选中清除
  257. this.$set(
  258. this.additionForm,
  259. `${key}`,
  260. this.additionForm[`${key}`].filter(i => i !== item.id)
  261. )
  262. } else {
  263. this.$set(
  264. this.additionForm,
  265. `${key}`,
  266. this.additionForm[`${key}`].concat([item.id])
  267. )
  268. }
  269. },
  270. printServiceClick(decoration, item) {
  271. if (!this.form[`${item.id}`].enable) return
  272. this.form[`${item.id}`].printService = decoration.id
  273. },
  274. },
  275. }
  276. </script>
  277. <style lang="scss" scoped>
  278. @import './step.scss';
  279. div {
  280. box-sizing: border-box;
  281. }
  282. .addition-title {
  283. font-size: 14px;
  284. font-family: Proxima Nova, sans-serif;
  285. color: #000;
  286. min-width: 104px;
  287. }
  288. .addition-item {
  289. border: 1px solid #dbdbdb;
  290. max-width: 108px;
  291. margin-bottom: 12px;
  292. margin-right: 12px;
  293. cursor: pointer;
  294. .name {
  295. word-break: break-all;
  296. height: 32px;
  297. line-height: 16px;
  298. padding: 0 4px;
  299. overflow: hidden;
  300. display: -webkit-box;
  301. -webkit-box-orient: vertical;
  302. -webkit-line-clamp: 2;
  303. }
  304. .background {
  305. border-bottom: 1px solid #dbdbdb;
  306. width: 108px;
  307. height: 108px;
  308. }
  309. &:hover,
  310. &.selected {
  311. border-color: #409eff;
  312. }
  313. }
  314. .custom-checkbox {
  315. &.disable {
  316. cursor: not-allowed;
  317. }
  318. }
  319. </style>