_code.vue 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346
  1. <template>
  2. <div
  3. id="productBuilder"
  4. class="flex between stretch wrap">
  5. <template v-if="detail.id">
  6. <step1
  7. :loading="loading"
  8. :detail="detail"
  9. :priceData="priceData"
  10. :form2="form2"
  11. :preData="preData"
  12. @update-form="$event => updateForm($event, 1)"
  13. ref="step1"></step1>
  14. <step2
  15. :loading="loading"
  16. :detail="detail"
  17. :priceData="priceData"
  18. :cycle="form1.cycle"
  19. :preData="preData"
  20. @update-form="$event => updateForm($event, 2)"
  21. ref="step2"></step2>
  22. <step3
  23. :loading="loading"
  24. :detail="detail"
  25. :priceData="priceData"
  26. :weightInfo="weightInfo"
  27. :cycle="form1.cycle"
  28. :model="form1.model"
  29. :buyForm="form1.autoForm"
  30. :form2="form2"
  31. @check="checkForm"
  32. @update-form="$event => updateForm($event, 3)"
  33. ref="step3"></step3>
  34. </template>
  35. <div
  36. class="flex-auto flex center"
  37. v-else>
  38. <el-empty description="No Data"></el-empty>
  39. </div>
  40. </div>
  41. </template>
  42. <script>
  43. import step1 from '@/components/product-builder/step-1'
  44. import step2 from '@/components/product-builder/step-2'
  45. import step3 from '@/components/product-builder/step-3'
  46. export default {
  47. name: 'PageProductBuilder',
  48. components: {
  49. step1,
  50. step2,
  51. step3,
  52. },
  53. layout: 'product_builder_layout',
  54. async asyncData({ $axios, params }) {
  55. const detail = await $axios
  56. .get(`/api/au/goods/detail/${params.code}`)
  57. .catch(e => {
  58. console.log(e)
  59. return {}
  60. })
  61. let priceData = {}
  62. if (detail.result && detail.result.id) {
  63. priceData = await $axios
  64. .post('/api/quote/price', {
  65. id: detail.result.id,
  66. })
  67. .catch(e => {
  68. console.log(e)
  69. })
  70. }
  71. return {
  72. detail: detail.result || {},
  73. priceData: priceData.result || {},
  74. }
  75. },
  76. data() {
  77. return {
  78. // 商品详情数据
  79. detail: {},
  80. priceData: {},
  81. loading: false,
  82. // 用于3个step组件间互通数据
  83. form1: {},
  84. form2: {},
  85. form3: {},
  86. weightInfo: {},
  87. preData: {},
  88. }
  89. },
  90. head() {
  91. return {
  92. title: this.detail.product_name
  93. ? `Product Builder: ${this.detail.product_name}`
  94. : 'PromoCollection',
  95. meta: [
  96. { name: 'keywords', content: this.detail.main?.keywords },
  97. { name: 'description', content: this.detail.main?.seo_title },
  98. ],
  99. }
  100. },
  101. computed: {},
  102. beforeMount() {
  103. const preData = localStorage.getItem(
  104. `product-user-select-${this.$route.params.code}`
  105. )
  106. if (preData) {
  107. this.preData = JSON.parse(preData)
  108. }
  109. },
  110. mounted() {
  111. // 从页脚组件抄过来的. 运费计算需要里面的一些数据
  112. this.$axios.post('/api/home/indexConfigWebsite', { site: 1 }).then(res => {
  113. this.$store.commit('config/setConfigInfo', res.result)
  114. })
  115. if (!this.detail.id) return
  116. this.$axios.post('/api/quote/weight', { id: this.detail.id }).then(res => {
  117. this.weightInfo = res.result
  118. })
  119. },
  120. methods: {
  121. checkForm(type) {
  122. this.loading = true
  123. Promise.all([
  124. this.$refs.step1.checkForm(),
  125. this.$refs.step2.checkForm(),
  126. this.$refs.step3.checkForm(),
  127. ])
  128. .then(res => {
  129. if (type === 2) {
  130. this.sendEnquiry(res)
  131. } else {
  132. this.addToProject(res)
  133. }
  134. // 点击按钮后删除从product页面带过来的数据
  135. this.preData = {}
  136. localStorage.removeItem(
  137. `product-user-select-${this.$route.params.code}`
  138. )
  139. this.loading = false
  140. })
  141. .catch(e => {
  142. console.log(e)
  143. this.$message.closeAll()
  144. this.$message.error('Please make sure to fill in the required fields')
  145. this.loading = false
  146. })
  147. },
  148. addToProject(res) {
  149. const temp = {
  150. cycle_id: res[0].cycle,
  151. price_unit_id: res[0].model,
  152. label_type: res[2].job,
  153. goods_id: this.detail.id,
  154. // 这个字段从product页面抄过来的, 本来就是空的, 不知道用来干嘛.
  155. decoration_id: '',
  156. // 以下3字段在后续逻辑赋值
  157. addition_ids: '',
  158. packaging_addition_ids: '',
  159. decoration_methods: [],
  160. }
  161. if (Array.isArray(res[1].packaging)) {
  162. temp.packaging_addition_ids = res[1].packaging.join(',')
  163. }
  164. if (Array.isArray(res[1].addon)) {
  165. temp.addition_ids = res[1].addon.join(',')
  166. }
  167. for (const key in res[1]) {
  168. if (/\d+/.test(key)) {
  169. const t = res[1][key].colorForm.filter(
  170. i => i.id === res[1][key].printService
  171. )
  172. temp.decoration_methods.push({
  173. colours_number: t.length ? t[0].colorNumber : 1,
  174. decoration_id: res[1][key].printService,
  175. price_unit_id: res[1][key].id,
  176. })
  177. }
  178. }
  179. this.$axios
  180. .post('/api/goods_cart/cartAdd', temp)
  181. .then(res => {
  182. console.log(res)
  183. if (res.code === 1) {
  184. this.$notify({
  185. title: 'success',
  186. message: 'Submitted successfully',
  187. type: 'success',
  188. duration: 3000,
  189. })
  190. // 天坑. 本页面用了不同的layout, 如果直接用router.push跳转会导致页面空白...
  191. location.href = location.origin + '/home/projects'
  192. }
  193. })
  194. .catch(e => {
  195. console.log(e)
  196. })
  197. },
  198. sendEnquiry(res) {
  199. const result = {
  200. product_code: this.detail.product_code,
  201. cycle_id: res[0].cycle,
  202. price_unit_id: res[0].model,
  203. supply_chain: res[0].supplyChain,
  204. decorated_in: res[0].decorated,
  205. address_id: res[2].defaultAddr.id,
  206. freight_type: res[2].freight_type || 1,
  207. // 目前没这功能, 暂时写死固定值
  208. transit_time: '2 Days',
  209. job_name: res[2].job || 'default_job',
  210. note: res[2].note || '',
  211. item: [],
  212. // addon: [], // 在后续逻辑赋值
  213. // decoration: [],
  214. // packaging: [],
  215. }
  216. const priceData = this.$refs.step3.computedPriceTableData
  217. result.item = res[0].autoForm.map(i => {
  218. const temp = {
  219. qty: i.value,
  220. colour: i.color,
  221. colour_type: 1,
  222. model: res[0].model,
  223. unit: priceData[0][`${i.value}_value`],
  224. set_up: priceData[1][`${i.value}_value`],
  225. print_option: priceData[2][`${i.value}_value`],
  226. addon: priceData[3][`${i.value}_value`],
  227. packaging: priceData[4][`${i.value}_value`],
  228. fright: priceData[5][`${i.value}_value`],
  229. sub_total: priceData[6][`${i.value}_value`],
  230. }
  231. if (i.color === 'Unspecified') temp.colour_type = 3
  232. if (i.color === 'PMS') {
  233. temp.colour_type = 2
  234. temp.colour = res[0].colorPmsText
  235. }
  236. if (i.colorId >= 0) temp.colour_id = i.colorId
  237. return temp
  238. })
  239. if (this.priceData.additionList) {
  240. if (this.priceData.additionList.addon) {
  241. const tempAddon = this.priceData.additionList.addon
  242. .filter(item => res[1].addon.includes(item.id))
  243. .map(item => {
  244. return {
  245. addition_id: item.id,
  246. addition_config: item.addition_code,
  247. addition_config_id: item.addition_config_id,
  248. image: '',
  249. name: item.name,
  250. }
  251. })
  252. if (tempAddon.length) result.addon = tempAddon
  253. }
  254. if (this.priceData.additionList.packaging) {
  255. const tempPackaging = this.priceData.additionList.packaging
  256. .filter(item => res[1].packaging.includes(item.id))
  257. .map(item => {
  258. return {
  259. addition_id: item.id,
  260. addition_config: item.addition_code,
  261. addition_config_id: item.addition_config_id,
  262. image: '',
  263. name: item.name,
  264. }
  265. })
  266. if (tempPackaging.length) result.packaging = tempPackaging
  267. }
  268. }
  269. const tempPrint = []
  270. for (const key in res[1]) {
  271. const target = res[1][key]
  272. if (/\d+/.test(key) && target.enable) {
  273. const v = {
  274. point: target.point,
  275. decoration_id: target.printService,
  276. }
  277. const decoration = target.decorationList.filter(
  278. i => i.id === target.printService
  279. )
  280. if (decoration.length) {
  281. v.price_type_id = decoration[0].price_type_id
  282. v.price_type_name = decoration[0].pricetype.name || ''
  283. }
  284. const formItem = target.colorForm.filter(
  285. i => i.id === target.printService
  286. )
  287. if (formItem.length) {
  288. v.pms = formItem[0].pmsColorText || ''
  289. v.max_color_count = formItem[0].colorNumber || 0
  290. }
  291. tempPrint.push(v)
  292. }
  293. }
  294. if (tempPrint.length) result.decoration = tempPrint
  295. // console.log(result)
  296. this.$axios
  297. .post('/api/order/add', result)
  298. .then(res => {
  299. console.log(res)
  300. if (res.code === 1) {
  301. this.$notify({
  302. title: 'success',
  303. message: 'Submitted successfully',
  304. type: 'success',
  305. duration: 3000,
  306. })
  307. // 对应的跳转页面还没开发.
  308. location.href = location.origin + '/product/' + `${this.detail.product_code}`
  309. }
  310. })
  311. .catch(e => {
  312. console.log(e)
  313. })
  314. },
  315. updateForm(data, step) {
  316. switch (step) {
  317. case 1:
  318. this.form1 = data
  319. break
  320. case 2:
  321. this.form2 = data
  322. break
  323. case 3:
  324. this.form3 = data
  325. break
  326. }
  327. },
  328. },
  329. }
  330. </script>
  331. <style lang="scss" scoped>
  332. #productBuilder {
  333. position: relative;
  334. width: 100%;
  335. div {
  336. box-sizing: border-box;
  337. }
  338. }
  339. </style>