Quote.vue 47 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452
  1. <template>
  2. <div>
  3. <div
  4. v-if="quoteLoadState == 0"
  5. v-loading="true"
  6. class="com-loading"></div>
  7. <div v-else-if="quoteLoadState == 1">
  8. <tab-nav
  9. :borderType="'allBorder'" :liWidth="100" :liHeight="40" :marginRight="5"
  10. :currTab.sync="curWeek"
  11. :tabList="priceList" />
  12. <div
  13. v-for="(items, indexs) in priceList"
  14. :key="items.name"
  15. v-show="curWeek == indexs">
  16. <div>
  17. <tab-nav
  18. :borderType="'incompleteBorder'"
  19. :currTab.sync="items.curPrint"
  20. :tabList="items.basePrice"
  21. :allPrintTab="allPrintTab"
  22. :marginTop="0"
  23. :marginRight="5"
  24. :marginBottom="items.basePrice?.length>1?7:0"
  25. @handle="sortBy(items)" />
  26. <!-- <el-popover
  27. placement="bottom-start"
  28. trigger="hover">
  29. <div
  30. class="exclamation-icon"
  31. slot="reference">!</div>
  32. <div class="popover">
  33. <p class="title">Print Price Comparison</p>
  34. <div
  35. v-for="(item,index) in items.basePrice"
  36. :key="item.name">
  37. <total-table
  38. :tableColumns="[...undecorated_columns, ...items.part_columns]"
  39. :tableData="item.data"
  40. :tableType="'Comparison'"
  41. :title="item.name"
  42. :curDecoration="items.decorationOrign[items.curPrint]" :indexDecoration="items.decorationOrign[index]"></total-table>
  43. </div>
  44. </div>
  45. </el-popover> -->
  46. </div>
  47. <div
  48. v-for="(item, index) in items.basePrice"
  49. :key="item.name"
  50. v-show="items.curPrint == index">
  51. <new-price-table
  52. :tableColumns="[...undecorated_columns, ...item.part_columns]"
  53. :tableData="item.data"
  54. :tableType="'Undecorated'"
  55. :curPrint="item.name"
  56. :curModel="items.decorationOrign[index]?.model"
  57. :data="items"
  58. :selectRow.sync="item.selectBaseRow"></new-price-table>
  59. </div>
  60. <!-- decoration_multiple is_more_print 打印开关,ID1001092需求 ;additional_print 颜色数量开关
  61. 以上需求作废,改版为多色多面的需求-->
  62. <new-price-table
  63. v-if="items.decoration?.length || items.basePrice?.[items.curPrint]?.decoration_addition?.length"
  64. :tableData="items.decoration"
  65. :tableColumns="[...decoration_columns, ...comCurBaseColumns]"
  66. :tableType="'Decoration'"
  67. :marginBottom="0"
  68. :hasDecAdd="items.basePrice?.[items.curPrint]?.decoration_addition?.length"></new-price-table>
  69. <div
  70. v-for="(item, index) in items.basePrice"
  71. :key="item.name + index"
  72. v-show="items.curPrint == index && item.decoration_addition.length">
  73. <new-price-table
  74. :tableData="item.decoration_addition"
  75. :tableColumns="[...addon_columns, ...item.part_columns]"
  76. :isCheckBox="true"
  77. :tableType="'Addon'"
  78. :multipleSelection.sync="items.selectAdditionRow"></new-price-table>
  79. </div>
  80. </div>
  81. <section>
  82. <p class="title1">
  83. <span></span>
  84. <span>Currency AUD($)</span>
  85. </p>
  86. <total-table
  87. :comLocationNum="comLocationNum"
  88. :tableData="comBuyData"
  89. :tableColumns="[
  90. ...buy_columns,
  91. ...comCurBaseTotalColumns
  92. ]"
  93. :curOrder="`${curWeek}-${comCurPrint}`"
  94. :min="comCurBaseColumns[0]?.label"
  95. :title="'Buy Price'" @send-idx="getChildIdx"></total-table>
  96. <p class="text-red">Price is ex- gst</p>
  97. <ul>
  98. <li class="entry">
  99. <div class="entry-l">Delivery Locations</div>
  100. <el-input
  101. placeholder="Enter Num"
  102. v-model="comCurBasePrice.locationNum"
  103. size="small" max="7" min="1"
  104. style="width: 130px;"
  105. type="number" @input="setMaxNum"></el-input>
  106. </li>
  107. <li
  108. class="entry"
  109. v-if="comLocationNum > 1">
  110. <div class="entry-text">Qty / Location</div>
  111. </li>
  112. <li class="entry" v-for="(i,k) in comCurBaseTotalColumns" :key="i.prop" v-show="comLocationNum > k">
  113. <div class="entry-l">Delivery Postcode</div>
  114. <el-input
  115. clearable
  116. placeholder="Enter Postcode"
  117. v-model="
  118. i.postcode"
  119. size="small"
  120. style="width: 130px; margin-right: 20px;"
  121. @input="getDebFreight(k)"></el-input>
  122. <el-input
  123. v-if="comLocationNum > 1"
  124. placeholder="Enter Qty"
  125. type="number"
  126. :min="comCurBaseColumns[0]?.label"
  127. @blur="getNumber($event, comCurBaseTotalColumns[k],comCurBaseColumns[0]?.label)"
  128. v-model="i.label"
  129. size="small"
  130. style="width: 130px; margin-right: 20px;"></el-input>
  131. <div class="entry-r">
  132. <span>Freight Method</span>
  133. <el-radio-group
  134. v-model="i.freight_type"
  135. @change="getFreight(k)">
  136. <el-radio :label="1">Road Express</el-radio>
  137. <el-radio :label="2">Air Freight</el-radio>
  138. </el-radio-group>
  139. </div>
  140. </li>
  141. </ul>
  142. <div class="entry">
  143. <div class="entry-l">Setup</div>
  144. <div class="entry-l">include in unit price</div>
  145. <pc-switch v-model="setup_switch"></pc-switch>
  146. </div>
  147. <div class="entry">
  148. <div class="entry-l">Freight</div>
  149. <div class="entry-l">include in unit price</div>
  150. <div>
  151. <pc-switch v-model="freight_switch"></pc-switch>
  152. </div>
  153. </div>
  154. <el-button
  155. size="medium"
  156. class="button-black"
  157. @click="openMailDialog">
  158. Enquiry
  159. </el-button>
  160. <!-- 需求隐藏 加入car project弹框 -->
  161. <!-- <el-button class="button-white" @click="openProjectdDialog"><img src="@/assets/img/car.png"/><span> to my project</span></el-button> -->
  162. </section>
  163. <!-- <section>
  164. <unit-table
  165. :tableData="unitData"
  166. :tableColumns="[
  167. ...unit_columns,
  168. ...comCurBaseTotalColumns
  169. ]"></unit-table>
  170. 直接用comMergeColumns会出现summary数值问题,改用v-show="comLocationNum"控制
  171. <total-table
  172. v-show="comLocationNum == 1"
  173. :tableData="comSellData"
  174. :tableColumns="comNoSumCol"
  175. :comLocationNum="comLocationNum"
  176. :title="'Sell Price'"></total-table>
  177. <total-table
  178. v-show="comLocationNum != 1"
  179. :tableData="comSellData"
  180. :tableColumns="comHasSumCol"
  181. :comLocationNum="comLocationNum"
  182. :title="'Sell Price'"></total-table>
  183. <el-button
  184. class="button-black button-margin"
  185. size="medium"
  186. @click="openDownloadDialog(0)"
  187. >View Quote</el-button>
  188. <el-button
  189. class="button-black button-margin"
  190. size="medium"
  191. @click="openDownloadDialog(1)"
  192. >Send Order Enquiry</el-button>
  193. </section> -->
  194. </div>
  195. <!-- <div
  196. v-else
  197. class="quote-tips">
  198. Current item prices are missing, please contact<a
  199. href="mailto:Info@promocollection.com.au">
  200. info@promocollection.com.au</a>
  201. </div> -->
  202. <no-ssr>
  203. <div v-if="quoteLoadState == 1">
  204. <!-- Enquiry按钮 -->
  205. <form-dialog
  206. :emailForm="enquiryForm"
  207. :enquiryConfig="enquiryConfig"
  208. :visible.sync="enquiryFormVisible"
  209. @handleSend="sendPriceMail"
  210. :rules="rules"
  211. :labelWidth="
  212. enquiryConfig[3].selectlist?.length ? 140 : 82
  213. "></form-dialog>
  214. <mail-table
  215. ref="mailtable"
  216. :mailData="mailData"
  217. v-show="false">
  218. <price-to-img
  219. :pageData="pageData"
  220. :specificationsObj="specificationsObj"
  221. :tableData="comFilterSwitchSellData"
  222. :tableColumns="comMergeColumns"
  223. :comLocationNum="comLocationNum"
  224. :emailForm="commentObj"></price-to-img>
  225. </mail-table>
  226. <el-dialog
  227. ref="dialogRef"
  228. :lock-scroll="false"
  229. title="Customer Quote"
  230. :visible.sync="priceToImgVisible"
  231. custom-class="price-to-img-dialog">
  232. <price-to-img
  233. :isImgSrc="false"
  234. ref="priceToImgRef"
  235. :loading.sync="pdfLoading"
  236. :pageData="pageData"
  237. :specificationsObj="specificationsObj"
  238. :tableData="comFilterSwitchSellData"
  239. :tableColumns="comMergeColumns"
  240. :comLocationNum="comLocationNum"
  241. :emailForm="commentObj"></price-to-img>
  242. <p class="tips">Change default standard comment:</p>
  243. <el-input v-model="commentObj.Comments"></el-input>
  244. <div class="btn-wrap">
  245. <el-button
  246. class="button-black button-margin"
  247. size="medium"
  248. @click="handleDownloadPdf"
  249. :loading="pdfLoading"
  250. >Download</el-button>
  251. </div>
  252. </el-dialog>
  253. <form-dialog
  254. :emailForm="commentObj"
  255. :enquiryConfig="enquiryConfig1"
  256. :visible.sync="downloadDialogVisible"
  257. @handleSend="sendPdfMail"
  258. :rules="rules1"
  259. :labelWidth="120"
  260. :isSendPdf="true"></form-dialog>
  261. <price-to-img
  262. ref="pdf"
  263. v-show="false"
  264. :loading.sync="pdfLoading"
  265. :pageData="pageData"
  266. :specificationsObj="specificationsObj"
  267. :tableData="comFilterSwitchSellData"
  268. :tableColumns="comMergeColumns"
  269. :comLocationNum="comLocationNum"
  270. :emailForm="commentObj"></price-to-img>
  271. <!-- 报价图片弹框 -->
  272. <dialog-XX-success
  273. :visible.sync="xxContentVisible" :imgType="false"
  274. :content="'This quantity is below the MOQ for this product'"></dialog-XX-success>
  275. </div>
  276. </no-ssr>
  277. </div>
  278. </template>
  279. <script>
  280. import { mapState, mapMutations, mapActions } from 'vuex'
  281. import _ from 'lodash'
  282. import { times as npTimes,plus as npPlus, minus as npMinus, divide as npDivide } from 'number-precision';
  283. import NewPriceTable from '@/components/table/NewPriceTable'
  284. import TotalTable from '@/components/table/TotalTable'
  285. import UnitTable from '@/components/table/UnitTable'
  286. import MailTable from '@/components/table/MailTable'
  287. import dialogXXSuccess from '@/components/DIalogXXSuccess.vue'
  288. export default {
  289. components: {
  290. NewPriceTable,
  291. TotalTable,
  292. UnitTable,
  293. MailTable,
  294. 'dialog-XX-success': dialogXXSuccess
  295. },
  296. props: {
  297. pageData: {
  298. type: Object,
  299. default: () => {
  300. return {}
  301. },
  302. },
  303. id: {
  304. type: Number,
  305. default: null
  306. },
  307. },
  308. data() {
  309. return {
  310. quoteLoadState: 0,
  311. curWeek: 0,
  312. allPrintTab: [],
  313. priceList: [],
  314. initZeroObj: {},
  315. initSetupObj: {},
  316. initUnitObj: {},
  317. initChangeUnitObj: {},
  318. initFrightObj: {},
  319. initTotalObj: {},
  320. xxContentVisible: false,
  321. setup_switch: false,
  322. freight_switch: false,
  323. weight: {},
  324. undecorated_columns: [
  325. {
  326. label: 'MODEL',
  327. prop: 'model',
  328. type: 'text',
  329. align: 'left',
  330. width: 150,
  331. extraWidth: 50, // 配合组件 (820-tableColumns[0].width-tableColumns[0].extraWidth)/(tableColumns.length-1)
  332. isFirstColumn: true, // 去掉$符号
  333. },
  334. {
  335. label: 'SETUP',
  336. prop: 'website_setup',
  337. },
  338. ],
  339. decoration_columns: [
  340. {
  341. label: 'PRINT OPTION',
  342. prop: 'dec_code',
  343. type: 'radioInput',
  344. align: 'left',
  345. width: 150,
  346. extraWidth: 50,
  347. },
  348. {
  349. label: 'SETUP',
  350. prop: 'website_setup',
  351. },
  352. ],
  353. addon_columns: [
  354. {
  355. label: 'Addon',
  356. prop: 'name',
  357. type: 'text',
  358. width: 150,
  359. extraWidth: 50,
  360. isFirstColumn: true, // 去掉$符号
  361. },
  362. {
  363. label: 'Setup',
  364. prop: 'website_setup',
  365. },
  366. ],
  367. buy_columns: [
  368. {
  369. label: 'QTY',
  370. prop: 'project',
  371. isFirstColumn: true, // 去掉$符号
  372. },
  373. ],
  374. unit_columns: [
  375. {
  376. label: 'QTY',
  377. prop: 'total',
  378. align: 'left',
  379. isText: true,
  380. },
  381. ],
  382. buyData: [
  383. { project: 'Setup' },
  384. { project: 'Unit' },
  385. { project: 'Freight' },
  386. { project: 'Total' }
  387. ],
  388. unitData: [
  389. { total: 'Unit Rate %' },
  390. { total: 'Setup %' },
  391. { total: 'Freight %' },
  392. ],
  393. sellData: [
  394. { project: 'Postcode',summary: '-' },
  395. { project: 'Setup',summary: 0.00 },
  396. { project: 'Unit',summary: 0.00 },
  397. { project: 'Freight',summary: 0.00 },
  398. { project: 'Total',summary: 0.00 },
  399. ],
  400. POA_Config: ['111', '111.00', '999', '999.00'],
  401. enquiryFormVisible: false,
  402. enquiryForm: {
  403. Name: '',
  404. Email: '',
  405. Phone: '',
  406. // 'Customer manager': null,
  407. Comments: '',
  408. },
  409. enquiryConfig: [
  410. { prop: 'Name', type: 'input' },
  411. { prop: 'Email', type: 'input' },
  412. { prop: 'Phone', type: 'input' },
  413. {
  414. prop: 'Customer manager',
  415. type: 'select',
  416. selectlist: [],
  417. isShow: false,
  418. },
  419. { prop: 'Comments', type: 'textarea' },
  420. ],
  421. rules: {
  422. Name: [{ required: true, message: 'Please enter', trigger: 'blur' }],
  423. Email: [
  424. {
  425. required: true,
  426. message: 'Please enter the correct format',
  427. pattern:
  428. /^([a-zA-Z0-9_-])+@([a-zA-Z0-9_-])+((.[a-zA-Z0-9_-]{2,3}){1,2})$/,
  429. trigger: 'blur',
  430. },
  431. ],
  432. Phone: [
  433. {
  434. required: true,
  435. message: 'Please enter the correct format',
  436. // pattern: /^1[0-9]{10}$/,
  437. trigger: 'blur',
  438. },
  439. ],
  440. 'Customer manager': [
  441. { required: true, message: 'Please select', trigger: 'change' },
  442. ],
  443. },
  444. mailData: {
  445. Url: '',
  446. },
  447. priceToImgVisible: false,
  448. downloadDialogVisible: false,
  449. commentObj: {
  450. 'Email Address': '',
  451. Comments: '',
  452. },
  453. enquiryConfig1: [
  454. { prop: 'Email Address', type: 'input' },
  455. { prop: 'Comments', type: 'input' },
  456. ],
  457. rules1: {
  458. 'Email Address': [
  459. {
  460. required: true,
  461. message: 'Please enter the correct format',
  462. pattern:
  463. /^([a-zA-Z0-9_-])+@([a-zA-Z0-9_-])+((.[a-zA-Z0-9_-]{2,3}){1,2})$/,
  464. trigger: 'blur',
  465. },
  466. ],
  467. },
  468. specificationsObj: {
  469. time: null,
  470. model: null,
  471. decoration: null,
  472. addonArr: null,
  473. },
  474. pdfLoading: false,
  475. }
  476. },
  477. computed: {
  478. ...mapState('config', { configInfo: state => state.configInfo }),
  479. userInfo() {
  480. return this.$store.state.userInfo
  481. },
  482. comCurWeekPrice() {
  483. return this.priceList[this.curWeek] || {}
  484. },
  485. comCurPrint() {
  486. return this.comCurWeekPrice.curPrint
  487. },
  488. comCurBasePrice() {
  489. return this.comCurWeekPrice.basePrice[this.comCurPrint]
  490. },
  491. comLocationNum() {
  492. return +(this.comCurWeekPrice.basePrice[this.comCurPrint].locationNum)
  493. },
  494. comCurBaseColumns() {
  495. return this.comCurBasePrice.part_columns
  496. },
  497. comCurBaseTotalColumns() {
  498. const d = this.comCurWeekPrice.basePrice[this.comCurPrint].total_part_columns
  499. const n = this.comLocationNum
  500. if (n === 1) {
  501. return d
  502. }else{
  503. const concatAdd = [...d,...this.comCurWeekPrice.basePrice[this.comCurPrint].remainingAttr].slice(0, n);
  504. return concatAdd
  505. }
  506. },
  507. comTotalLabel(){
  508. // 计算所有元素label的和
  509. return this.comCurBaseTotalColumns.reduce((sum, item) => sum + Number(item.label), 0);
  510. },
  511. comSummaryColumns(){
  512. return [{ label: this.comTotalLabel.toString(), prop: 'summary' }];
  513. },
  514. comAttributeList() {
  515. const obj = {}
  516. this.comCurBaseTotalColumns?.forEach(item => {
  517. obj[item.prop] = item.label
  518. })
  519. obj.summary = this.comTotalLabel;
  520. return obj
  521. },
  522. // 当前已选基础价
  523. comBasePrice() {
  524. return this.comCurBasePrice.selectBaseRow
  525. },
  526. // 当前累加打印价
  527. comDecoPrice() {
  528. if (!this.comCurWeekPrice.decoration?.length) {
  529. return []
  530. }
  531. const arr = this.comCurWeekPrice.decoration
  532. .map((item, idx) => {
  533. const obj = {}
  534. if (+item.num > 0) {
  535. for (const i in this.initUnitObj) {
  536. if (idx === 0) {
  537. // 首个打印价要取附加价计算
  538. const keyArr = i.split('_')
  539. const supplier_val = item[`supplier_${keyArr[1]}`]
  540. if (this.POA_Config.includes(supplier_val)) {
  541. obj[`${i}`] = supplier_val
  542. } else {
  543. obj[`${i}`] = npTimes(+supplier_val, item.num)
  544. }
  545. } else {
  546. obj[`${i}`] = this.POA_Config.includes(item[i]) ? item[i] : npTimes(+item[i], item.num);
  547. }
  548. }
  549. }
  550. return obj
  551. })
  552. .filter(obj => Object.keys(obj).length !== 0)
  553. return arr
  554. },
  555. comBuyData() {
  556. this.calculateBuyData();
  557. return this.buyData
  558. },
  559. comNoSumCol(){
  560. return [
  561. ...this.buy_columns,
  562. ...this.comCurBaseTotalColumns
  563. ]
  564. },
  565. comHasSumCol(){
  566. return [
  567. ...this.buy_columns,
  568. ...this.comCurBaseTotalColumns,
  569. ...this.comSummaryColumns
  570. ];
  571. },
  572. comMergeColumns(){
  573. return this.comLocationNum == 1 ? this.comNoSumCol:this.comHasSumCol
  574. },
  575. comSellData() {
  576. if (Object.keys(this.comAttributeList).length !== 0) {
  577. this.calculateSellData();
  578. }
  579. return this.sellData
  580. },
  581. comFilterSwitchSellData() {
  582. return this.comSellData?.filter((item, i) => {
  583. if (i === 1 && this.setup_switch) {
  584. return false;
  585. } else if (i === 3 && this.freight_switch) {
  586. return false;
  587. } else {
  588. return item
  589. }
  590. })
  591. },
  592. },
  593. methods: {
  594. ...mapActions(['getUserInfo']),
  595. multiplyWithMargin(o, m) {
  596. return npDivide(Math.round(npTimes(o, (100 + parseFloat(m)))), 100).toFixed(2)
  597. },
  598. async getQuote() {
  599. let margin = 0
  600. let setupMargin = 0
  601. await this.$store
  602. .dispatch('getShopInfo')
  603. .then(res => {
  604. margin = parseFloat(res.margin) || 0
  605. setupMargin = parseFloat(res.setup_margin) || 0
  606. })
  607. this.$axios
  608. .post('/catalog/pricequote', { id: this.id })
  609. .then(res => {
  610. if (localStorage.getItem('unit')) {
  611. this.unitData = JSON.parse(localStorage.getItem('unit'))
  612. }
  613. const { attributeList, priceList, default: defaultID } = res.result
  614. if (
  615. !Object.keys(attributeList).length ||
  616. !Object.keys(priceList).length
  617. ) {
  618. this.quoteLoadState = 2
  619. return
  620. }
  621. for (const a in attributeList) {
  622. // 过滤掉website_qty标签无数量
  623. if (attributeList[a]) {
  624. this.initZeroObj[a] = 0
  625. }else{
  626. delete attributeList[a]
  627. }
  628. for (const unit of this.unitData) {
  629. // 如果对象中缺少 attributeList 的 key,则设置为 40
  630. if (!(a in unit) || unit[a] === '') {
  631. this.$set(unit,a,40)
  632. }
  633. }
  634. }
  635. // 把价格数据全部乘上margin比例.
  636. for(const key in priceList) {
  637. if (Object.hasOwn(priceList, key)) {
  638. for (const k in priceList[key]) {
  639. if (Object.hasOwn(priceList[key], k) && Array.isArray(priceList[key][k])) {
  640. priceList[key][k] = priceList[key][k].map(item => {
  641. const temp = {...item}
  642. for(let i = 1; i<=8; i++) {
  643. if (temp[`website_qty${i}`] && !this.POA_Config.includes(temp[`website_qty${i}`])) {
  644. temp[`website_qty${i}`] = this.multiplyWithMargin(parseFloat(temp[`website_qty${i}`]) || 0, margin).toString()
  645. }
  646. if (temp[`supplier_qty${i}`] && !this.POA_Config.includes(temp[`supplier_qty${i}`])) {
  647. temp[`supplier_qty${i}`] = this.multiplyWithMargin(parseFloat(temp[`supplier_qty${i}`]) || 0, margin).toString()
  648. }
  649. }
  650. temp.website_setup = this.multiplyWithMargin(parseFloat(temp.website_setup) || 0, setupMargin).toString()
  651. if (temp.supplier_setup) {
  652. temp.supplier_setup = this.multiplyWithMargin(parseFloat(temp.supplier_setup) || 0, setupMargin).toString()
  653. }
  654. if (k === 'decoration') {
  655. temp.decoration_addition = temp.decoration_addition.map(deco => {
  656. const d = {...deco}
  657. for(let i = 1; i<=8; i++) {
  658. if (d[`website_qty${i}`] && !this.POA_Config.includes(d[`website_qty${i}`])) {
  659. d[`website_qty${i}`] = this.multiplyWithMargin(parseFloat(d[`website_qty${i}`]) || 0, margin).toString()
  660. }
  661. if (d[`supplier_qty${i}`] && !this.POA_Config.includes(d[`supplier_qty${i}`])) {
  662. d[`supplier_qty${i}`] = this.multiplyWithMargin(parseFloat(d[`supplier_qty${i}`]) || 0, margin).toString()
  663. }
  664. }
  665. d.website_setup = this.multiplyWithMargin(parseFloat(d.website_setup) || 0, setupMargin).toString()
  666. if (d.supplier_setup) {
  667. d.supplier_setup = this.multiplyWithMargin(parseFloat(d.supplier_setup) || 0, setupMargin).toString()
  668. }
  669. return d
  670. })
  671. }
  672. return temp
  673. })
  674. }
  675. }
  676. }
  677. }
  678. // 格式化数据,对象改成数组
  679. for (const keys in priceList) {
  680. const vals = {}
  681. const { decoration, additionlist, printDescription, ...remaining } = priceList[keys]
  682. vals.basePrice = []
  683. vals.selectAdditionRow = []
  684. vals.curPrint = 0
  685. vals.name = this.formatDurationString(keys)
  686. vals.nameOrigin = keys
  687. vals.decorationOrign = [...decoration]
  688. vals.decorationID = decoration.map(i => i.id)
  689. vals.decoration = [...decoration].flatMap(item => {
  690. const result = [];
  691. if (item.max_color > 0) {
  692. result.push({ ...item, max_num: item.max_color,id:`${item.id}-1`,decName:item.max_color_name });
  693. }
  694. if (item.max_point > 0) {
  695. result.push({ ...item, max_num: item.max_point,id:`${item.id}-2`,decName:item.max_point_name });
  696. }
  697. return result;
  698. });
  699. vals.additionlist = (additionlist || []).map(item => {
  700. return {
  701. ...item,
  702. id: item.id + '-1' // 修改 id
  703. }
  704. })
  705. for (const k in remaining) {
  706. console.log(k, 'k')
  707. const o = {}
  708. o.name = k
  709. o.data = remaining[k]
  710. o.attributeList = this.copyData(attributeList)
  711. if (remaining[k]?.length>1) {
  712. // 检查数组中各元素的 website_qtyN 是否都等于 "-" or "POA",则删除key
  713. for (const a in attributeList) {
  714. const is111Poa = remaining[k].every(item => this.POA_Config.includes(item[a]));
  715. if (is111Poa) {
  716. delete o.attributeList[a]
  717. }
  718. }
  719. }else if (remaining[k]?.length === 1) {
  720. // 判断 remaining[k][0] 中是否包含 vals.attributeList 的键,并且对应的值都等于 "-" or "POA",则不删除key
  721. const obj = remaining[k][0];
  722. const attributeKeys = Object.keys(o.attributeList);
  723. const areAllKeysInPOAConfig = attributeKeys.every(key => {
  724. return Object.prototype.hasOwnProperty.call(obj, key) && this.POA_Config.includes(obj[key]);
  725. });
  726. if (!areAllKeysInPOAConfig) {
  727. attributeKeys.forEach(key => {
  728. if (Object.prototype.hasOwnProperty.call(obj, key) && this.POA_Config.includes(obj[key])) {
  729. delete o.attributeList[key];
  730. }
  731. });
  732. }
  733. }
  734. o.name = k
  735. o.data = remaining[k]
  736. o.locationNum = 1
  737. o.part_columns = Object.keys(o.attributeList).map(e => {
  738. return { label: o.attributeList[e].toString(), prop: e }
  739. })
  740. const freightParam = {
  741. postcode: '',
  742. freight_type: 1,
  743. freight: {}
  744. }
  745. o.total_part_columns = this.copyData(o.part_columns).map(obj => ({
  746. ...obj,
  747. ...freightParam
  748. }));
  749. o.remainingAttr = []
  750. const firstColumns = { ...o.part_columns[0],...freightParam}
  751. for(let i = 1; i < 9; i++){
  752. if(!Object.keys(o.attributeList).includes(`website_qty${i}`)){
  753. const newColumn = {...firstColumns}
  754. this.$set(newColumn,'prop',`website_qty${i}`)
  755. o.remainingAttr.push(newColumn)
  756. }
  757. };
  758. vals.basePrice.push(o)
  759. if (!this.allPrintTab.includes(k)) {
  760. this.allPrintTab.push(k)
  761. }
  762. }
  763. this.priceList.push(vals)
  764. }
  765. this.priceList.forEach((options, index) => {
  766. // 原始数据name没空格,需要映射到
  767. if (options.nameOrigin === defaultID?.name) {
  768. this.curWeek = index
  769. options.decorationOrign.forEach((opt, idx) => {
  770. if (opt.id === defaultID?.decoration?.id) {
  771. this.$set(options, 'curPrint', idx)
  772. }
  773. })
  774. }
  775. // 所有周期的所有打印,进来页面时默认勾选第一条价格
  776. options.basePrice.forEach((opt,idx) => {
  777. this.$set(opt, 'selectBaseRow', opt.data[0] ||{})
  778. // 打印原数据decorationOrign的name 匹配 basePrice(周期下的每个打印数据)的name
  779. const matchData = options.decorationOrign.find(d=>{
  780. return d.pricetype?.name === opt.name
  781. })
  782. opt.decoration_addition = [...(matchData?.decoration_addition || []),...(options.additionlist || [])]
  783. })
  784. options.decoration.forEach(opt => {
  785. this.$set(opt, 'num', 0)
  786. })
  787. })
  788. this.initSetupObj = Object.assign({}, this.initZeroObj)
  789. this.initUnitObj = Object.assign({}, this.initZeroObj, {
  790. website_setup: 0,
  791. })
  792. this.initFrightObj = Object.assign({}, this.initZeroObj)
  793. this.buyData = this.buyData.map(item => {
  794. return { ...item, ...this.initSetupObj }
  795. })
  796. this.sellData = this.sellData.map(item => {
  797. return { ...item, ...this.initSetupObj }
  798. })
  799. this.quoteLoadState = 1
  800. })
  801. .catch((e) => {
  802. console.log('this.quoteLoadState1: ', e);
  803. this.quoteLoadState = 2
  804. })
  805. },
  806. formatDurationString(input) {
  807. // 匹配数字和字符串的正则表达式
  808. const regex = /(\d+)([a-zA-Z]+)/g;
  809. const result = input.replace(regex, '$1 $2');
  810. return result;
  811. },
  812. getWeight() {
  813. this.$axios
  814. .post('/catalog/weight', { id: this.id })
  815. .then(res => {
  816. this.weight = res.result
  817. })
  818. .catch(() => {})
  819. },
  820. getFreight(k) {
  821. const postcode = this.comCurBaseTotalColumns[k].postcode
  822. const type = this.comCurBaseTotalColumns[k].freight_type
  823. const init = { basic:0,pickup:0,minimum:0 }
  824. if (postcode.length < 3) {
  825. this.comCurBaseTotalColumns[k].freight = init
  826. return
  827. }
  828. this.$axios
  829. .post('/catalog/freight', {
  830. postcode,
  831. type
  832. })
  833. .then(res => {
  834. this.comCurBaseTotalColumns[k].freight = Array.isArray(res.result)?init:res.result
  835. })
  836. },
  837. getDebFreight: _.debounce(function (k) {
  838. this.getFreight(k)
  839. }, 200),
  840. getNumber(e, row, min) {
  841. if (parseInt(e.target.value) < min) {
  842. this.xxContentVisible = true;
  843. row.label = min
  844. } else {
  845. this.xxContentVisible = false;
  846. }
  847. },
  848. sortBy(items) {
  849. // items.decoration.sort(this.customSort(items.decorationID))
  850. // const targetElement = items.decoration.splice(items.curPrint, 1)[0]
  851. // items.decoration.unshift(targetElement)
  852. },
  853. // 按原打印数组的id字段排序
  854. customSort(decorationID) {
  855. return (a, b) => {
  856. return decorationID.indexOf(a.id) - decorationID.indexOf(b.id)
  857. }
  858. },
  859. dividePrice(a) {
  860. return npDivide(+a, 100)
  861. },
  862. openMailDialog() {
  863. this.enquiryFormVisible = true
  864. // if (this.$utils.checkLogin()) {
  865. // this.getUserInfo()
  866. // const { contacts, email, phone, crm_users_id:crmUsersId } = this.userInfo
  867. // this.enquiryForm.Name = contacts
  868. // this.enquiryForm.Email = email
  869. // this.enquiryForm.Phone = phone
  870. // if (crmUsersId?.length) {
  871. // this.enquiryConfig[3].selectlist = crmUsersId
  872. // if (crmUsersId?.length === 1) {
  873. // this.enquiryForm['Customer manager'] = crmUsersId[0].name
  874. // }
  875. // } else {
  876. // this.enquiryConfig[3].isShow = false
  877. // this.$delete(this.enquiryForm, 'Customer manager')
  878. // }
  879. // this.enquiryFormVisible = true
  880. // } else {
  881. // this.openDialog()
  882. // }
  883. },
  884. getMailData() {
  885. this.mailData.Url = window.location.href
  886. const { Name, Email, Phone, Comments } = this.enquiryForm
  887. this.mailData['Customer Name'] = Name
  888. this.mailData['Customer Email'] = Email
  889. this.mailData['Customer Phone'] = Phone
  890. this.mailData.Comments = Comments
  891. // const { email, level, createTime } = this.userInfo
  892. // this.mailData['Member Account'] = email
  893. // this.mailData['Member Grade'] = level.name
  894. // this.mailData['Member Registration time'] = createTime
  895. this.mailData['Quote time'] = this.$utils.formatTime(new Date())
  896. },
  897. // 发送价格邮件 Enquiry按钮
  898. async sendPriceMail() {
  899. await this.getMailData()
  900. await this.getCustomerQuoteData()
  901. const shopInfo = await this.$store.dispatch('getShopInfo').then(res => res)
  902. const {
  903. Name,
  904. Email,
  905. Phone,
  906. Comments,
  907. } = this.enquiryForm
  908. this.$axios
  909. .post('/c-api/quote/sendenquiry', {
  910. site_id: shopInfo.id || '',
  911. content: this.$refs.mailtable.$el.innerHTML,
  912. name: Name,
  913. email: Email,
  914. phone: Phone,
  915. product_code: this.pageData.product_code,
  916. comments: Comments,
  917. })
  918. .then(res => {
  919. this.setLoading(false)
  920. this.enquiryFormVisible = false
  921. this.$confirm('Enquiry Sent', {
  922. confirmButtonText: 'OK',
  923. showCancelButton: false,
  924. type: 'success',
  925. center: true,
  926. showClose: false,
  927. confirmButtonClass: 'com-btnblack',
  928. }).then(() => {})
  929. })
  930. .catch(() => {
  931. this.setLoading(false)
  932. })
  933. },
  934. getCustomerQuoteData() {
  935. const { name, selectAdditionRow } = this.comCurWeekPrice
  936. this.specificationsObj.time = name
  937. this.specificationsObj.model = this.comBasePrice.model
  938. this.specificationsObj.decoration = this.comCurBasePrice.name
  939. if (selectAdditionRow.length) {
  940. this.specificationsObj.addonArr = selectAdditionRow.map(
  941. item => item.name
  942. )
  943. } else {
  944. this.specificationsObj.addonArr = null
  945. }
  946. },
  947. openDownloadDialog(type) {
  948. this.getCustomerQuoteData()
  949. type
  950. ? (this.downloadDialogVisible = true)
  951. : (this.priceToImgVisible = true)
  952. },
  953. handleDownloadPdf() {
  954. this.pdfLoading = true
  955. this.$nextTick(() => {
  956. this.$refs.priceToImgRef.htmlToPdf(this.pageData.product_code)
  957. })
  958. },
  959. // 发送PDF邮件
  960. sendPdfMail() {
  961. this.$axios
  962. .post('/quote/sendpdf', {
  963. content: this.$refs.pdf.$el.innerHTML,
  964. product_code: this.pageData.product_code,
  965. email: this.commentObj['Email Address'],
  966. })
  967. .then(res => {
  968. this.setLoading(false)
  969. this.downloadDialogVisible = false
  970. this.$confirm('Enquiry Sent', {
  971. confirmButtonText: 'OK',
  972. showCancelButton: false,
  973. type: 'success',
  974. center: true,
  975. showClose: false,
  976. confirmButtonClass: 'com-btnblack',
  977. })
  978. })
  979. .catch(() => {
  980. this.setLoading(false)
  981. })
  982. },
  983. copyData(data) {
  984. return JSON.parse(JSON.stringify(data))
  985. },
  986. calculateBuyData(){
  987. const selPriceArr = []
  988. if (JSON.stringify(this.comBasePrice) !== '{}') {
  989. selPriceArr.push(this.comBasePrice)
  990. }
  991. selPriceArr.push(...this.comDecoPrice, ...this.comCurWeekPrice.selectAdditionRow);
  992. if(selPriceArr.length === 0){
  993. this.buyData = this.buyData.map(item => ({ ...item, ...this.initZeroObj }));
  994. return
  995. }
  996. // 累加所选,初始数量对应的单价。赋值给this.initUnitObj
  997. for (const i in this.initUnitObj) {
  998. const columnSum = this.copyData(selPriceArr).reduce(function (prev, cur) {
  999. if (cur[i] === '111' || cur[i] === '111.00') {
  1000. return (cur[i] = '-')
  1001. } else if (cur[i] === '999' || cur[i] === '999.00') {
  1002. return (cur[i] = 'POA')
  1003. } else {
  1004. // 当基础价格是-POA,后续累加价格是正常数字价,依然返回-POA
  1005. if (prev === '-' || prev === 'POA') {
  1006. return prev
  1007. }
  1008. return npPlus(+cur[i], prev).toFixed(2)
  1009. }
  1010. }, 0)
  1011. this.$set(this.initUnitObj, i, columnSum)
  1012. }
  1013. for (const a in this.comAttributeList) {
  1014. // 每次更改选择价格,必须遍历整个comAttributeList
  1015. const tempIdx = this.comCurBaseColumns.findIndex(
  1016. items => +this.comAttributeList[a] < +items.label
  1017. )
  1018. if (tempIdx > 0) {
  1019. const key = this.comCurBaseColumns[tempIdx - 1].prop
  1020. this.$set(this.initChangeUnitObj, a, this.initUnitObj[key])
  1021. } else {
  1022. const length = this.comCurBaseColumns.length
  1023. const key = this.comCurBaseColumns[length - 1].prop
  1024. this.$set(this.initChangeUnitObj, a, this.initUnitObj[key])
  1025. }
  1026. if (
  1027. this.initChangeUnitObj[a] === 'POA' ||
  1028. this.initChangeUnitObj[a] === '-'
  1029. ) {
  1030. this.$set(this.initSetupObj, a, this.initChangeUnitObj[a])
  1031. this.$set(this.initFrightObj, a, this.initChangeUnitObj[a])
  1032. } else {
  1033. this.$set(this.initSetupObj, a, this.initUnitObj.website_setup)
  1034. // 计算运费
  1035. // 单独批次数量的总重
  1036. // weight定义是{},后端没数据传回[]
  1037. const unitWLocal = this.weight.unit_w_local ? +this.weight.unit_w_local : 0;
  1038. const totalWeight = Math.ceil(
  1039. npTimes(unitWLocal, this.comAttributeList[a])
  1040. )
  1041. // Road express 1 AAE:AAEFactor 2
  1042. const setupFuel = this.dividePrice(this.configInfo.fuel)
  1043. const setupBagFreight = this.dividePrice(this.configInfo.bag_freight)
  1044. const setupExpressFreight = this.dividePrice(
  1045. this.configInfo.express_freight
  1046. )
  1047. const AAEFactor = npPlus(1, setupBagFreight, setupFuel)
  1048. const expressFactor = npPlus(1, setupExpressFreight, setupFuel)
  1049. let frightCost = 0
  1050. let freightType
  1051. let postcode
  1052. let freight = {}
  1053. const matchIndex = this.comCurBaseTotalColumns.findIndex(item => item.prop === a)
  1054. // 1,所有按第一个运费统计;不为1,则只计算 comCurBaseTotalColumns的[地址数]长度
  1055. if (+this.comLocationNum === 1 || this.comLocationNum >= matchIndex + 1) {
  1056. ({ freight, freight_type: freightType, postcode } = this.comCurBaseTotalColumns[+this.comLocationNum === 1 ? 0 : matchIndex] || {});
  1057. }
  1058. if (freightType === 1 && postcode >= 3) {
  1059. if (totalWeight > 20) {
  1060. const a1 = npMinus(totalWeight, 20)
  1061. const a2 = npTimes(a1, +freight.basic)
  1062. const a3 = npPlus(+freight.pickup, a2)
  1063. frightCost = npTimes(a3, expressFactor)
  1064. } else {
  1065. frightCost = npTimes(+freight.pickup, expressFactor)
  1066. }
  1067. } else if (freightType === 2 && postcode >= 3) {
  1068. const a1 = npDivide(totalWeight, 5)
  1069. const a2 = npTimes(+freight.minimum, Math.ceil(a1))
  1070. frightCost = npTimes(a2, AAEFactor)
  1071. } else {
  1072. frightCost = 0
  1073. }
  1074. this.$set(this.initFrightObj, a, Math.ceil(frightCost).toFixed(2))
  1075. }
  1076. if (
  1077. this.initChangeUnitObj[a] === '-' ||
  1078. this.initChangeUnitObj[a] === 'POA'
  1079. ) {
  1080. this.$set(this.initTotalObj, a, this.initChangeUnitObj[a])
  1081. } else {
  1082. const unitQTY = npTimes(
  1083. this.initChangeUnitObj[a],
  1084. this.comAttributeList[a]
  1085. )
  1086. this.$set(
  1087. this.initTotalObj,
  1088. a,
  1089. npPlus(
  1090. this.initUnitObj.website_setup,
  1091. unitQTY,
  1092. this.initFrightObj[a]
  1093. ).toFixed(2)
  1094. )
  1095. }
  1096. if (this.setup_switch) {
  1097. // 业务:Setup价格/对应数量后向上取整
  1098. if (
  1099. this.initChangeUnitObj[a] === '-' ||
  1100. this.initChangeUnitObj[a] === 'POA'
  1101. ) {
  1102. this.$set(this.initChangeUnitObj, a, this.initChangeUnitObj[a])
  1103. } else {
  1104. const num =
  1105. Math.ceil(
  1106. (this.initSetupObj[a] / this.comAttributeList[a]) * 100
  1107. ) / 100
  1108. this.$set(
  1109. this.initChangeUnitObj,
  1110. a,
  1111. npPlus(this.initChangeUnitObj[a], num).toFixed(2)
  1112. )
  1113. }
  1114. }
  1115. if (this.freight_switch) {
  1116. if (
  1117. this.initChangeUnitObj[a] === '-' ||
  1118. this.initChangeUnitObj[a] === 'POA'
  1119. ) {
  1120. this.$set(this.initChangeUnitObj, a, this.initChangeUnitObj[a])
  1121. } else {
  1122. const num =
  1123. Math.ceil(
  1124. (this.initFrightObj[a] / this.comAttributeList[a]) * 100
  1125. ) / 100
  1126. this.$set(
  1127. this.initChangeUnitObj,
  1128. a,
  1129. npPlus(this.initChangeUnitObj[a], num).toFixed(2)
  1130. )
  1131. }
  1132. }
  1133. }
  1134. if (this.setup_switch) {
  1135. this.$set(this.buyData, 0, { ...this.buyData[0], ...this.initZeroObj, ...{'summary':0.00} })
  1136. } else {
  1137. this.$set(this.buyData, 0, { ...this.buyData[0], ...this.initSetupObj })
  1138. }
  1139. this.$set(this.buyData, 1, {
  1140. ...this.buyData[1],
  1141. ...this.initChangeUnitObj,
  1142. })
  1143. if (this.freight_switch) {
  1144. this.$set(this.buyData, 2, { ...this.buyData[2], ...this.initZeroObj })
  1145. } else {
  1146. this.$set(this.buyData, 2, {
  1147. ...this.buyData[2],
  1148. ...this.initFrightObj,
  1149. })
  1150. }
  1151. this.$set(this.buyData, 3, { ...this.buyData[3], ...this.initTotalObj })
  1152. },
  1153. calculateSellData() {
  1154. // 放置comSellData计算和副作用的代码
  1155. const buySetup = this.comBuyData[0]
  1156. const buyUnit = this.comBuyData[1]
  1157. const buyFright = this.comBuyData[2]
  1158. let frightSummary = 0
  1159. for (const i in this.comAttributeList) {
  1160. if (buyUnit[i] === '-' || buyUnit[i] === 'POA') {
  1161. this.sellData[0][i] = '-'
  1162. this.sellData[1][i] = buyUnit[i]
  1163. this.sellData[2][i] = buyUnit[i]
  1164. this.sellData[3][i] = buyUnit[i]
  1165. this.sellData[4][i] = buyUnit[i]
  1166. } else {
  1167. const item = this.comLocationNum === 1 ? this.comCurBaseTotalColumns[0] : this.comCurBaseTotalColumns.find(element => element.prop === i);
  1168. this.sellData[0][i] = item && item.postcode !== '' ? item.postcode : '-';
  1169. // 分别乘以Markup %表格
  1170. const key = this.comCurBaseTotalColumns[0]?.prop
  1171. if (this.unitData[1] && this.unitData[1][key]) {
  1172. const a = npPlus(this.unitData[1][key] / 100, 1)
  1173. this.sellData[1][i] = npTimes(buySetup[i], a).toFixed(2)
  1174. } else {
  1175. this.sellData[1][i] = buySetup[i]
  1176. }
  1177. if (this.unitData[0][i]) {
  1178. const a = npPlus(this.unitData[0][i] / 100, 1)
  1179. this.sellData[2][i] = npTimes(buyUnit[i], a).toFixed(2)
  1180. } else {
  1181. this.sellData[2][i] = buyUnit[i]
  1182. }
  1183. if (this.unitData[2] && this.unitData[2][key]) {
  1184. const a = npPlus(this.unitData[2][key] / 100, 1)
  1185. this.sellData[3][i] = npTimes(buyFright[i], a).toFixed(2)
  1186. // freight_switch会使qty的运费=0无法计算,故用了initFrightObj
  1187. const fa = npTimes(this.initFrightObj[i], a)
  1188. frightSummary = npPlus(frightSummary,fa).toFixed(2)
  1189. } else {
  1190. this.sellData[3][i] = buyFright[i]
  1191. frightSummary = npPlus(frightSummary,this.initFrightObj[i]).toFixed(2)
  1192. }
  1193. const unitQTY = npTimes(this.sellData[2][i], this.comAttributeList[i])
  1194. this.$set(
  1195. this.sellData[4],
  1196. i,
  1197. npPlus(this.sellData[1][i], unitQTY, this.sellData[3][i]).toFixed(2)
  1198. )
  1199. }
  1200. }
  1201. // 迭代加了Total列,需要统计qty的运费后才能合计运费
  1202. if (this.sellData[2].summary !== '-' && this.sellData[2].summary !== 'POA') {
  1203. if (this.freight_switch) {
  1204. const num = Math.ceil((frightSummary / this.comAttributeList.summary) * 100) / 100
  1205. this.sellData[2].summary = npPlus(this.sellData[2].summary, num).toFixed(2)
  1206. this.sellData[3].summary = 0.00
  1207. const a = npTimes(this.sellData[2].summary,this.comAttributeList.summary)
  1208. this.sellData[4].summary = npPlus(this.sellData[1].summary,a).toFixed(2)
  1209. } else {
  1210. this.sellData[3].summary = frightSummary
  1211. this.sellData[4].summary = npPlus(frightSummary,this.sellData[4].summary).toFixed(2)
  1212. }
  1213. }
  1214. },
  1215. // 低于最小起订量 把该栏的数值重置到最小起订量
  1216. getChildIdx(idx){
  1217. this.xxContentVisible = true;
  1218. this.$set(this.comCurBaseTotalColumns[idx-1],'label',this.comCurBaseColumns[0]?.label)
  1219. },
  1220. setMaxNum(val){
  1221. if(val > 7){
  1222. this.comCurBasePrice.locationNum = 7
  1223. }else if(val < 1){
  1224. this.comCurBasePrice.locationNum = 1
  1225. }
  1226. },
  1227. ...mapMutations({ openDialog: 'openDialog', setLoading: 'product/setLoading' })
  1228. },
  1229. }
  1230. </script>
  1231. <style lang='scss' scoped>
  1232. $deep-blue: #004a97;
  1233. .exclamation-icon {
  1234. width: 18px;
  1235. height: 18px;
  1236. line-height: 18px;
  1237. display: inline-block;
  1238. text-align: center;
  1239. color: #000;
  1240. border-radius: 100%;
  1241. border: 1px solid #000;
  1242. font-size: 18px;
  1243. font-weight: 550;
  1244. }
  1245. .popover {
  1246. border: 1px solid #eee;
  1247. padding: 20px;
  1248. .title {
  1249. text-align: center;
  1250. font-weight: bold;
  1251. font-size: 20px;
  1252. }
  1253. }
  1254. .title1 {
  1255. margin: 30px 0 0;
  1256. padding: 20px 0 0;
  1257. font-size: 16px;
  1258. color: #15263d;
  1259. display: flex;
  1260. font-family: ProximaNova-Regular;
  1261. font-weight: 600;
  1262. justify-content: space-between;
  1263. border-top: 1px solid #eee;
  1264. span {
  1265. &:nth-child(2) {
  1266. font-size: 14px;
  1267. color: #50596b;
  1268. }
  1269. }
  1270. }
  1271. .text-red {
  1272. text-align: right;
  1273. color: #e90000;
  1274. padding: 10px 10px 0 0;
  1275. font-size: 16px;
  1276. font-weight: bold;
  1277. }
  1278. .entry {
  1279. color: #50596b;
  1280. margin: 15px 10px;
  1281. display: flex;
  1282. flex-direction: row;
  1283. align-items: center;
  1284. .entry-l {
  1285. width: 120px;
  1286. margin-right: 10px;
  1287. }
  1288. .entry-text{
  1289. margin-left: 280px;
  1290. width: 130px;
  1291. text-align: center;
  1292. font-weight: bold;
  1293. }
  1294. :deep(.el-input__inner) {
  1295. text-align: center;
  1296. padding: 0;
  1297. }
  1298. .entry-r {
  1299. flex: 1;
  1300. text-align: right;
  1301. span {
  1302. margin-right: 10px;
  1303. }
  1304. //修改radio选中为✔样式
  1305. :deep(.el-radio__inner) {
  1306. border-radius: 0;
  1307. width: 18px;
  1308. height: 18px;
  1309. }
  1310. :deep(.el-radio__input.is-checked .el-radio__inner) {
  1311. background: $deep-blue;
  1312. border: 1px solid $deep-blue;
  1313. }
  1314. :deep(.el-radio__input.is-checked + .el-radio__label) {
  1315. color: $deep-blue;
  1316. }
  1317. :deep(.el-radio__input.is-checked .el-radio__inner::after) {
  1318. content: '';
  1319. width: 10px;
  1320. height: 5px;
  1321. border: 1px solid white;
  1322. border-top: transparent;
  1323. border-right: transparent;
  1324. text-align: center;
  1325. display: block;
  1326. position: absolute;
  1327. top: 3px;
  1328. left: 2px;
  1329. transform: rotate(-45deg);
  1330. border-radius: 0px;
  1331. background: none;
  1332. }
  1333. }
  1334. }
  1335. .button-margin {
  1336. margin: 20px 0 0;
  1337. }
  1338. .quote-tips {
  1339. font-size: 16px;
  1340. a {
  1341. color: #6495ed;
  1342. text-decoration: underline;
  1343. margin-left: 5px;
  1344. }
  1345. }
  1346. button {
  1347. font-size: 12px;
  1348. width: 200px;
  1349. height: 30px;
  1350. border-radius: 5px;
  1351. margin-right: 20px;
  1352. cursor: pointer;
  1353. border: none;
  1354. box-shadow: 0px 5px 10px rgba(0, 0, 0, 0.2);
  1355. }
  1356. .button-black {
  1357. background-color: $deep-blue;
  1358. color: #ffffff;
  1359. &:hover {
  1360. background-color: #013269;
  1361. }
  1362. }
  1363. .button-white {
  1364. background-color: #fff;
  1365. color: #50596b;
  1366. img,
  1367. span {
  1368. vertical-align: middle;
  1369. }
  1370. }
  1371. // 报价图片弹框样式
  1372. :deep(.price-to-img-dialog) {
  1373. min-width: 840px;
  1374. margin-top: 2vh !important;
  1375. .el-dialog__header {
  1376. background-color: #00213b;
  1377. padding: 10px 20px;
  1378. .el-dialog__title {
  1379. color: #fff;
  1380. }
  1381. .el-dialog__headerbtn {
  1382. top: 15px;
  1383. }
  1384. }
  1385. .el-dialog__body {
  1386. background-color: #f1f4f9;
  1387. padding: 13px 20px;
  1388. }
  1389. .tips {
  1390. font-size: 16px;
  1391. color: #00213b;
  1392. margin: 12px 0;
  1393. }
  1394. .btn-wrap {
  1395. text-align: left;
  1396. .button-black {
  1397. margin-top: 20px;
  1398. }
  1399. }
  1400. }
  1401. :deep(.el-switch) {
  1402. .el-switch__label {
  1403. position: absolute;
  1404. display: none;
  1405. color: #fff !important;
  1406. }
  1407. /*打开时文字位置设置*/
  1408. .el-switch__label--right {
  1409. z-index: 1;
  1410. right: -3px;
  1411. }
  1412. /*关闭时文字位置设置*/
  1413. .el-switch__label--left {
  1414. z-index: 1;
  1415. left: 20px;
  1416. }
  1417. /*显示文字*/
  1418. .el-switch__label.is-active {
  1419. display: block;
  1420. }
  1421. .el-switch .el-switch__core,
  1422. .el-switch .el-switch__label {
  1423. width: 50px !important;
  1424. }
  1425. .el-switch__label * {
  1426. font-size: 13px !important;
  1427. }
  1428. }
  1429. </style>