step-3.vue 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603
  1. <template>
  2. <el-card class="step-card">
  3. <div
  4. slot="header"
  5. class="flex between">
  6. <div class="step-card-title">Step3 Confirm</div>
  7. </div>
  8. <div v-loading="loading">
  9. <div>
  10. <div v-if="addressList.length">
  11. <div class="card-sub-title">Ship to</div>
  12. <div class="form-item-label">Shipping address</div>
  13. <div class="flex between">
  14. <div>
  15. <div>
  16. <div class="flex">
  17. <div class="address-text name">{{ defaultAddr.name }}</div>
  18. <div class="address-text">&nbsp;{{ defaultAddr.phone }}</div>
  19. </div>
  20. <div class="address-text">{{ defaultAddr.address }}</div>
  21. <div class="flex">
  22. <div class="address-text">{{ defaultAddr.city }}</div>
  23. <div class="address-text">
  24. &nbsp;{{ defaultAddr.post_code }}
  25. </div>
  26. </div>
  27. </div>
  28. </div>
  29. <el-button
  30. type="text"
  31. @click="visibleOfSelectAddressDialog = true"
  32. >edit address</el-button
  33. >
  34. </div>
  35. </div>
  36. <el-button
  37. @click="addAddress"
  38. v-else
  39. size="mini"
  40. plain
  41. >+&nbsp;Add delivery address</el-button
  42. >
  43. </div>
  44. <el-form
  45. :model="form"
  46. :rules="rules"
  47. label-width="0"
  48. label-position="left"
  49. ref="form">
  50. <el-form-item
  51. label-width="120px"
  52. label="Freight type">
  53. <el-radio-group
  54. v-loading="freightLoading"
  55. v-model="form.freight_type">
  56. <el-radio :label="2">AAE</el-radio>
  57. <el-radio :label="1">Road express</el-radio>
  58. </el-radio-group>
  59. </el-form-item>
  60. <div class="price-title flex">
  61. <div>Pricing Optiopns</div>
  62. </div>
  63. <div class="divide-solid"></div>
  64. <div class="price-table">
  65. <el-table
  66. class=""
  67. v-loading="freightLoading"
  68. :data="computedPriceTableData">
  69. <el-table-column
  70. v-for="(column, index) in computedPriceTableColumns"
  71. :fixed="column.fixed || false"
  72. :width="column.width || '100px'"
  73. :key="`${column.prop}-${index}`"
  74. :label="column.label || column.prop"
  75. :prop="column.prop"></el-table-column>
  76. </el-table>
  77. </div>
  78. <el-form-item
  79. label-width="0"
  80. prop="job">
  81. <div class="form-item-label">
  82. <span style="color: red">*</span>&nbsp;Reference or Job name
  83. </div>
  84. <el-select
  85. v-model="form.job"
  86. filterable
  87. allow-create
  88. default-first-option
  89. style="width: 100%"
  90. placeholder="Ref, Choose or create new one">
  91. <el-option
  92. v-for="item in jobOptions"
  93. :key="item.label_type"
  94. :label="item.label_type"
  95. :value="item.label_type">
  96. </el-option>
  97. </el-select>
  98. </el-form-item>
  99. <div class="form-item-label">Note</div>
  100. <el-form-item
  101. prop="note"
  102. label-width="0">
  103. <el-input
  104. v-model="form.note"
  105. placeholder="Any note"></el-input>
  106. </el-form-item>
  107. </el-form>
  108. <div class="flex center">
  109. <el-button
  110. type="primary"
  111. @click="addToProject">
  112. <div class="flex center">
  113. <span class="btn-icon el-icon-shopping-cart-2"></span>&nbsp; Add to
  114. project
  115. </div>
  116. </el-button>
  117. <el-button
  118. type="info"
  119. @click="sendEnquiry">
  120. <div class="flex">
  121. <span class="btn-icon el-icon-shopping-bag-1"></span>&nbsp; Send
  122. enquiry
  123. </div>
  124. </el-button>
  125. </div>
  126. <add-address-dialog
  127. :dialogVisible.sync="visibleOfEditAddressDialog"
  128. :data="addressDetail"
  129. :componentVisible="patternOfEditAddressDialog"
  130. @close="closeAddressDialog"
  131. @update="getAddressData" />
  132. <select-address-dialog
  133. :addressDetail="form.defaultAddr"
  134. :addressList="addressList"
  135. :visible.sync="visibleOfSelectAddressDialog"
  136. @add-new-addr="addAddress"
  137. @select-addr="selectAddress"
  138. @close="closeSelectAddressDialog"></select-address-dialog>
  139. </div>
  140. </el-card>
  141. </template>
  142. <script>
  143. import { plus } from 'number-precision'
  144. import _ from 'lodash'
  145. import { mapState } from 'vuex'
  146. import stepMixin from './stepMixin'
  147. import selectAddressDialog from './DialogSelectAddr.vue'
  148. import addAddressDialog from '@/components/addAddressDialog.vue'
  149. import {
  150. formatPrice,
  151. getUnit,
  152. getSetup,
  153. getPrint,
  154. getAddon,
  155. getPackaging,
  156. getFright,
  157. } from '@/utils/price'
  158. export default {
  159. name: 'Step3',
  160. components: {
  161. addAddressDialog,
  162. selectAddressDialog,
  163. },
  164. mixins: [stepMixin],
  165. props: {
  166. // 步骤1中选中的周期. 不能直接拿form1来取form1.cycle, 否则步骤1变更时, 涉及到form1的computed会全部重新计算, 可能会导致其他异常
  167. cycle: {
  168. type: Number,
  169. default: function () {
  170. return 0
  171. },
  172. },
  173. // 同cycle
  174. model: {
  175. type: Number,
  176. default: function () {
  177. return 0
  178. },
  179. },
  180. // 同cycle. form1的autoForm, 记录对应颜色的 商品购买数量表单.
  181. buyForm: {
  182. type: Array,
  183. default: function () {
  184. return []
  185. },
  186. },
  187. // 步骤2的表单. 因其设计选中规格, 为当前步骤的前置数据, 故需从父组件获取. 只用来计算, 本组件不更改这个数据.
  188. form2: {
  189. type: Object,
  190. default: function () {
  191. return {}
  192. },
  193. },
  194. weightInfo: {
  195. type: Object,
  196. default: function () {
  197. return {}
  198. },
  199. },
  200. },
  201. data() {
  202. return {
  203. rules: {
  204. job: [
  205. {
  206. required: true,
  207. message: 'Please select or create a job name',
  208. trigger: 'change',
  209. },
  210. ],
  211. },
  212. form: {
  213. // 默认的配送地址
  214. defaultAddr: {},
  215. freight_type: 1,
  216. job: '',
  217. note: '',
  218. },
  219. freightLoading: false,
  220. // 运费信息
  221. freightInfo: {},
  222. // 运费浮动比. 从product页面来的数据, 记录在local storage里面
  223. sellFreight: 0,
  224. addressList: [],
  225. addressDetail: {},
  226. visibleOfSelectAddressDialog: false,
  227. visibleOfEditAddressDialog: false,
  228. patternOfEditAddressDialog: 1,
  229. jobOptions: [],
  230. priceTableColumns: [
  231. {
  232. prop: 'qty',
  233. label: 'Qty',
  234. fixed: 'left',
  235. },
  236. ],
  237. dynamicPriceTableColumns: [],
  238. dynamicPriceTableData: [],
  239. priceTableData: [
  240. {
  241. qty: 'Unit',
  242. },
  243. {
  244. qty: 'Set up',
  245. },
  246. {
  247. qty: 'Print Option',
  248. },
  249. {
  250. qty: 'Addon',
  251. },
  252. {
  253. qty: 'Packing',
  254. },
  255. {
  256. qty: 'Fright',
  257. },
  258. {
  259. qty: 'SubTotal',
  260. },
  261. ],
  262. }
  263. },
  264. computed: {
  265. ...mapState('config', { configInfo: state => state.configInfo }),
  266. // 商品 当前选中周期的 基础价格和打印价格数据
  267. computedPriceData() {
  268. return this.priceData.priceList.filter(
  269. item => item.cycle_id === this.cycle
  270. )
  271. },
  272. /**
  273. * 当前选中周期下, 商品对应的各型号基础价格数据. 可以推断当前周期有几个型号
  274. */
  275. computedBasePriceData() {
  276. const model = this.computedPriceData.length
  277. ? this.computedPriceData[0]
  278. : {}
  279. // 属性‘1’里面是基础价格数据, 属性 ’2‘是打印价格数据
  280. if (model['1']) {
  281. return model['1'].slice()
  282. } else {
  283. return []
  284. }
  285. },
  286. // 当前型号位于基础价格数据的index, 用于取出型号对应的基础数据
  287. computedBasePriceIndex() {
  288. return this.computedBasePriceData.findIndex(
  289. item => item.id === this.model
  290. )
  291. },
  292. /**
  293. * 当前选中周期下, 商品对应的各型号打印价格.
  294. */
  295. computedPrintPriceData() {
  296. const model = this.computedPriceData.length
  297. ? this.computedPriceData[0]
  298. : {}
  299. // 属性‘1’里面是基础价格数据, 属性 ’2‘是打印价格数据
  300. if (model['2']) {
  301. return model['2'].slice()
  302. } else {
  303. return []
  304. }
  305. },
  306. // 其实没必要的, 只是方便html上不用读多一层属性 from.defaultAddr.****这样
  307. defaultAddr() {
  308. return this.form.defaultAddr
  309. },
  310. computedPriceTableColumns() {
  311. return this.priceTableColumns.concat(this.dynamicPriceTableColumns)
  312. },
  313. computedPriceTableData() {
  314. return this.priceTableData.map((item, index) =>
  315. Object.assign({}, item, this.dynamicPriceTableData[index] || {})
  316. )
  317. },
  318. },
  319. watch: {
  320. addressList() {
  321. const temp = this.addressList.filter(item => item.is_default === 1)
  322. if (temp.length) {
  323. this.form.defaultAddr = temp[0]
  324. } else if (this.addressList.length) {
  325. this.form.defaultAddr = this.addressList[0]
  326. } else {
  327. this.form.defaultAddr = {}
  328. }
  329. if (this.form.defaultAddr.post_code) {
  330. this.getFreight()
  331. }
  332. },
  333. 'form.freight_type'() {
  334. this.getFreight()
  335. },
  336. 'form.defaultAddr'() {
  337. this.getFreight()
  338. },
  339. buyForm: {
  340. deep: true,
  341. immediate: true,
  342. handler: function () {
  343. const temp = []
  344. if (!this.loaded) return temp
  345. this.dynamicPriceTableColumns = this.buyForm.map(item => {
  346. return {
  347. prop: `${item.value}`,
  348. buyNum: item.value,
  349. }
  350. })
  351. this.getDynamicPriceTableData()
  352. },
  353. },
  354. form2: {
  355. deep: true,
  356. immediate: true,
  357. handler: function () {
  358. this.getDynamicPriceTableData()
  359. },
  360. },
  361. },
  362. mounted() {
  363. if (localStorage.getItem('sellFreight')) {
  364. this.sellFreight = localStorage.getItem('sellFreight')
  365. }
  366. this.getAddressData().then(() => {
  367. this.loaded = true
  368. })
  369. this.getJobList()
  370. },
  371. methods: {
  372. checkForm() {
  373. return new Promise((resolve, reject) => {
  374. this.$refs.form.validate(valid => {
  375. if (valid) {
  376. resolve(JSON.parse(JSON.stringify(this.form)))
  377. } else {
  378. reject(new Error('validate step3 form error'))
  379. }
  380. })
  381. })
  382. },
  383. addToProject() {
  384. this.$emit('check', 1)
  385. },
  386. sendEnquiry() {
  387. this.$emit('check', 2)
  388. },
  389. // Job 下拉/输入框候选数据
  390. getJobList() {
  391. this.$axios
  392. .post('/api/goods_cart/cartLabelLists', { keyword: [] })
  393. .then(res => {
  394. this.jobOptions = res.result
  395. })
  396. },
  397. // 获取配送地址数据
  398. async getAddressData() {
  399. return await this.$axios
  400. .get('/api/address/list', {
  401. params: {
  402. keyword: '',
  403. page: 1,
  404. limit: 100,
  405. },
  406. })
  407. .then(res => {
  408. if (res.code === 1) {
  409. this.addressList = res.result.data
  410. this.addressTotal = res.result.total
  411. }
  412. })
  413. },
  414. addAddress() {
  415. this.addressDetail = {}
  416. this.patternOfEditAddressDialog = 2
  417. this.visibleOfEditAddressDialog = true
  418. },
  419. closeAddressDialog() {
  420. this.visibleOfEditAddressDialog = false
  421. },
  422. selectAddress(data) {
  423. const temp = this.addressList.filter(item => item.id === data)
  424. if (temp.length) this.form.defaultAddr = temp[0]
  425. this.closeSelectAddressDialog()
  426. },
  427. closeSelectAddressDialog() {
  428. this.visibleOfSelectAddressDialog = false
  429. },
  430. getDynamicPriceTableData() {
  431. const temp = [
  432. {}, // unit
  433. {}, // setup 打印和附加价格之和
  434. {}, // print option
  435. {}, // addon 附加服务除了packing之外的总价
  436. {}, // packing / packaging
  437. {}, // fright 运费
  438. {}, // 汇总
  439. ]
  440. const ratio = plus(this.sellFreight / 100, 1)
  441. // { buyNum数量, prop标签label }
  442. this.dynamicPriceTableColumns.forEach(v => {
  443. const unitPrice = getUnit(
  444. v.buyNum,
  445. this.computedBasePriceIndex,
  446. this.priceData.attributeList,
  447. this.computedBasePriceData
  448. )
  449. temp[0][`${v.prop}`] = formatPrice(unitPrice)
  450. temp[0][`${v.prop}_value`] = unitPrice
  451. const setupPrice = getSetup(
  452. v.buyNum,
  453. this.form2,
  454. this.priceData.additionList
  455. )
  456. temp[1][`${v.prop}`] = formatPrice(setupPrice)
  457. temp[1][`${v.prop}_value`] = setupPrice
  458. const printPrice = getPrint(
  459. v.buyNum,
  460. this.form2,
  461. this.priceData.attributeList
  462. )
  463. temp[2][`${v.prop}`] = formatPrice(printPrice)
  464. temp[2][`${v.prop}_value`] = printPrice
  465. const addonPrice = getAddon(
  466. v.buyNum,
  467. this.form2,
  468. this.priceData.attributeList,
  469. this.priceData.additionList
  470. )
  471. temp[3][`${v.prop}`] = formatPrice(addonPrice)
  472. temp[3][`${v.prop}_value`] = addonPrice
  473. const packagingPrice = getPackaging(
  474. v.buyNum,
  475. this.form2,
  476. this.priceData.attributeList,
  477. this.priceData.additionList
  478. )
  479. temp[4][`${v.prop}`] = formatPrice(packagingPrice)
  480. temp[4][`${v.prop}_value`] = packagingPrice
  481. const frightPrice = getFright(
  482. v.buyNum,
  483. this.configInfo,
  484. this.freightInfo,
  485. this.weightInfo,
  486. ratio
  487. )
  488. temp[5][`${v.prop}`] = formatPrice(frightPrice)
  489. temp[5][`${v.prop}_value`] = frightPrice
  490. // 最后一行汇总行的数据
  491. const total = [
  492. temp[0][`${v.prop}_value`],
  493. temp[1][`${v.prop}_value`],
  494. temp[2][`${v.prop}_value`],
  495. temp[3][`${v.prop}_value`],
  496. temp[4][`${v.prop}_value`],
  497. temp[5][`${v.prop}_value`],
  498. ].reduce((total, curr) => {
  499. const value = Number(curr)
  500. if (total === 'POA' || curr === 'POA') {
  501. return 'POA'
  502. } else if (typeof value === 'number') {
  503. return plus(total, value)
  504. } else {
  505. return total
  506. }
  507. }, 0)
  508. temp[6][`${v.prop}`] =
  509. typeof total === 'number' ? formatPrice(total) : total
  510. temp[6][`${v.prop}_value`] = total
  511. })
  512. // console.log(temp, '计算动态表格数据')
  513. this.dynamicPriceTableData = temp
  514. },
  515. getFreight: _.debounce(function () {
  516. if (!this.form.defaultAddr.post_code) return
  517. this.freightLoading = true
  518. this.$axios
  519. .post('/api/quote/freight', {
  520. postcode: this.form.defaultAddr.post_code,
  521. type: this.form.freight_type,
  522. })
  523. .then(res => {
  524. this.freightInfo = res.result
  525. this.getDynamicPriceTableData()
  526. })
  527. .finally(() => {
  528. setTimeout(() => {
  529. this.freightLoading = false
  530. }, 300)
  531. })
  532. }, 200),
  533. },
  534. }
  535. </script>
  536. <style lang="scss" scoped>
  537. @import './step.scss';
  538. div {
  539. box-sizing: border-box;
  540. }
  541. .btn-icon {
  542. font-size: 20px;
  543. }
  544. .address-text {
  545. font-size: 14px;
  546. line-height: 18px;
  547. color: #000;
  548. font-family: Proxima Nova, sans-serif;
  549. &.name {
  550. font-weight: bold;
  551. }
  552. }
  553. .price-title {
  554. height: 32px;
  555. padding: 0 8px;
  556. background: #f8f9fd;
  557. width: 100%;
  558. & > div {
  559. font-size: 16px;
  560. color: #000;
  561. font-family: Proxima Nova, sans-serif;
  562. }
  563. }
  564. .divide-solid {
  565. height: 4px;
  566. width: 100%;
  567. background-color: #a9aeba;
  568. margin: 10px auto 12px;
  569. }
  570. .price-table {
  571. :deep(.el-table) {
  572. font-family: Proxima Nova;
  573. &::before {
  574. background-color: transparent;
  575. }
  576. td.el-table__cell {
  577. font-size: 15px;
  578. border-bottom: 2px solid #dddcdc;
  579. }
  580. th.el-table__cell {
  581. font-weight: 600;
  582. font-size: 20px;
  583. color: #fff;
  584. background-color: #626469;
  585. border-bottom: none;
  586. }
  587. }
  588. }
  589. </style>