exportQuota2.vue 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486
  1. <template>
  2. <div class="pdf-wrap">
  3. <table
  4. id="pdfTarget"
  5. style="
  6. width: 850px;
  7. line-height: 20px;
  8. font-size: 12px;
  9. font-family:
  10. 'Source Han Sans', 'Open Sans', 'WebFont-Open Sans', Arial, sans-serif;
  11. "
  12. >
  13. <tbody>
  14. <tr>
  15. <td style="vertical-align: bottom">
  16. <img
  17. style="position: relative; left: -10pt"
  18. src="http://zohocrmapi.promocollection.com.au/static/uploads/logo/pdf_logo.png"
  19. height="50px"
  20. />
  21. </td>
  22. <td style="vertical-align: top; text-align: right">
  23. <h1>Quotation</h1>
  24. </td>
  25. </tr>
  26. <tr>
  27. <td style="vertical-align: top">
  28. <template v-if="city2 === 'US'">
  29. <p>Promocollection, LLC</p>
  30. <p>1309 COFFEEN AVE STE 1200</p>
  31. <p>SHERIDAN</p>
  32. <p>WY 82801</p>
  33. </template>
  34. <template v-else>
  35. <p>FAIR OCEAN TRADING AUSTRALIA</p>
  36. <p>PTY LTD</p>
  37. <p>15/10 Chilvers Road,</p>
  38. <p>THORNLEIGH NSW 2120,</p>
  39. <p>AUSTRALIA</p>
  40. </template>
  41. </td>
  42. <td style="vertical-align: top; text-align: right">
  43. <table style="width: 250px; float: right">
  44. <colgroup>
  45. <col width="100px" />
  46. <col width="150px" />
  47. </colgroup>
  48. <tr>
  49. <td><strong>Sales Person:</strong></td>
  50. <td>{{ computedCreator }}</td>
  51. </tr>
  52. <tr>
  53. <td><strong>Order Date:</strong></td>
  54. <td>{{ dayjs().format('DD MMM YYYY') }}</td>
  55. </tr>
  56. </table>
  57. </td>
  58. </tr>
  59. <tr>
  60. <td style="vertical-align: top">
  61. <template v-if="city2 !== 'US'">
  62. <p>
  63. Email:&nbsp;&nbsp;&nbsp;&nbsp;accounts@promocollection.com.au
  64. </p>
  65. <p>Phone Number:&nbsp;&nbsp;&nbsp;&nbsp;02 9008 1152</p>
  66. <p>Fax Number:&nbsp;&nbsp;&nbsp;&nbsp;02 9008 1157</p>
  67. </template>
  68. </td>
  69. <td
  70. rowspan="3"
  71. style="vertical-align: top; text-align: right"
  72. >
  73. <img
  74. :src="mainPicture"
  75. width="300px"
  76. />
  77. </td>
  78. </tr>
  79. <tr>
  80. <td style="vertical-align: top">
  81. <p><strong>Information</strong></p>
  82. <p>
  83. <strong>{{ productInfo.product_name }}</strong>
  84. </p>
  85. <p v-if="productInfo.package_size">
  86. <strong>Size:</strong>
  87. {{ productInfo.product_size }}
  88. </p>
  89. <p v-if="productInfo.product_hd">
  90. <strong>Thickness:</strong>
  91. {{ productInfo.product_hd }}
  92. </p>
  93. <p v-if="productInfo.product_capacity">
  94. <strong>Capacity:</strong>
  95. {{ productInfo.product_capacity }}
  96. </p>
  97. <p v-if="productInfo.product_material">
  98. <strong>Material:</strong>
  99. {{ productInfo.product_material }}
  100. </p>
  101. <p v-if="productInfo.product_require_print">
  102. <strong>Print:</strong>
  103. {{ productInfo.product_require_print }}
  104. </p>
  105. <p v-if="productInfo.product_require_color">
  106. <strong>Color requirements:</strong>
  107. {{ productInfo.product_require_color }}
  108. </p>
  109. <p v-if="productInfo.product_color">
  110. <strong>Product Colours Available:</strong>
  111. {{ productInfo.product_color }}
  112. </p>
  113. <p v-if="productInfo.product_battery">
  114. <strong>Whether with battery:</strong>
  115. {{ productInfo.product_battery }}
  116. </p>
  117. <p v-if="productInfo.package_info">
  118. <strong>Packaging:</strong>
  119. {{ productInfo.package_info }}
  120. </p>
  121. <p v-if="productInfo.product_other">
  122. <strong>More Details:</strong>
  123. {{ productInfo.product_other }}
  124. </p>
  125. </td>
  126. </tr>
  127. <tr style="vertical-align: top">
  128. <td><p></p></td>
  129. </tr>
  130. <tr style="vertical-align: top">
  131. <td colspan="2">
  132. <table style="width: 800px; border-spacing: 0">
  133. <tbody>
  134. <tr
  135. style="
  136. background-color: #eee;
  137. line-height: 45px;
  138. border-top: solid 1px #ccc;
  139. font-weight: bold;
  140. text-align: center;
  141. "
  142. >
  143. <td
  144. style="
  145. border-top: solid 1px #ccc;
  146. border-bottom: solid 1px #ccc;
  147. width: 20px;
  148. height: 45px;
  149. "
  150. >
  151. #
  152. </td>
  153. <td
  154. style="
  155. border-top: solid 1px #ccc;
  156. border-bottom: solid 1px #ccc;
  157. text-align: left;
  158. width: 230px;
  159. height: 45px;
  160. "
  161. >
  162. Items
  163. </td>
  164. <td
  165. style="
  166. border-top: solid 1px #ccc;
  167. border-bottom: solid 1px #ccc;
  168. width: 120px;
  169. height: 45px;
  170. "
  171. >
  172. Qty
  173. </td>
  174. <td
  175. style="
  176. border-top: solid 1px #ccc;
  177. border-bottom: solid 1px #ccc;
  178. width: 120px;
  179. height: 45px;
  180. "
  181. >
  182. Setup cost
  183. </td>
  184. <td
  185. style="
  186. border-top: solid 1px #ccc;
  187. border-bottom: solid 1px #ccc;
  188. width: 120px;
  189. height: 45px;
  190. "
  191. >
  192. Unit cost
  193. </td>
  194. <td
  195. style="
  196. border-top: solid 1px #ccc;
  197. border-bottom: solid 1px #ccc;
  198. width: 120px;
  199. height: 45px;
  200. "
  201. >
  202. Local Freight
  203. </td>
  204. <td
  205. style="
  206. border-top: solid 1px #ccc;
  207. border-bottom: solid 1px #ccc;
  208. width: 120px;
  209. height: 45px;
  210. "
  211. >
  212. {{ exportForm.gst_name ? 'Total(ex-GST)($)' : 'Total($)' }}
  213. </td>
  214. </tr>
  215. <tr
  216. v-for="(row, index) in step3FormList"
  217. :key="index"
  218. style="text-align: center; line-height: 45px"
  219. >
  220. <td style="border-bottom: solid 1px #ccc; height: 45px">
  221. {{ index + 1 }}
  222. </td>
  223. <td
  224. style="
  225. border-bottom: solid 1px #ccc;
  226. text-align: left;
  227. height: 45px;
  228. "
  229. >
  230. {{ productInfo.product_name }}
  231. <br />
  232. By {{ getFreightType(row) }}&nbsp;-&nbsp;{{
  233. exportForm[`zdy_date_${row.typeNumber}_${row.number}`]
  234. }}{{
  235. exportForm[`cycle_name_${row.typeNumber}_${row.number}`]
  236. }}
  237. </td>
  238. <td style="border-bottom: solid 1px #ccc; height: 45px">
  239. {{ row.number }} units
  240. </td>
  241. <td style="border-bottom: solid 1px #ccc; height: 45px">
  242. ${{ row.setup_cost }}
  243. {{ exportForm.gst_name ? '+GST' : '' }}
  244. </td>
  245. <td style="border-bottom: solid 1px #ccc; height: 45px">
  246. ${{ row.sold_unit }} {{ exportForm.gst_name ? '+GST' : '' }}
  247. </td>
  248. <td style="border-bottom: solid 1px #ccc; height: 45px">
  249. ${{ row.add_freight_cost }}
  250. {{ exportForm.gst_name ? '+GST' : '' }}
  251. </td>
  252. <td style="border-bottom: solid 1px #ccc; height: 45px">
  253. ${{ row.sold_price }}
  254. {{ exportForm.gst_name ? '+GST' : '' }}
  255. </td>
  256. </tr>
  257. <tr>
  258. <td
  259. colspan="7"
  260. style="text-align: left; line-height: 45px"
  261. >
  262. Freight to {{ city_short[city] || city }} is included.Price
  263. based on exchange rate of {{ exportForm.exchange }}.Price is
  264. only valid for {{ exportForm.days }} days.
  265. </td>
  266. </tr>
  267. </tbody>
  268. </table>
  269. </td>
  270. </tr>
  271. <!-- <tr>
  272. <td colspan="2">
  273. <p style="line-height: 45px">
  274. <strong style="border-bottom: dashed 2px #000; padding: 10px">
  275. Notes
  276. </strong>
  277. </p>
  278. </td>
  279. </tr>
  280. <tr>
  281. <td colspan="2">
  282. <p style="padding-top: 20px">{php}echo $notes;{/php}</p>
  283. </td>
  284. </tr> -->
  285. <tr>
  286. <td colspan="2">
  287. <p style="padding-top: 20px">
  288. <img
  289. v-for="(item, index) in otherPicture"
  290. :key="index"
  291. :src="item"
  292. width="250px"
  293. style="padding-right: 10px"
  294. />
  295. </p>
  296. </td>
  297. </tr>
  298. <tr>
  299. <td colspan="2">
  300. <p>
  301. <a :href="mainPicture">{{ mainPicture }}</a>
  302. </p>
  303. </td>
  304. </tr>
  305. <tr>
  306. <td colspan="2">
  307. <p
  308. v-for="p in otherPicture"
  309. :key="p"
  310. >
  311. <a
  312. :href="p"
  313. target="_blank"
  314. >
  315. {{ p }}
  316. </a>
  317. </p>
  318. </td>
  319. </tr>
  320. <tr>
  321. <td
  322. colspan="2"
  323. style="text-align: center"
  324. >
  325. <img
  326. style="margin-top: 10px"
  327. src="http://zohocrmapi.promocollection.com.au/static/uploads/logo/pdf_end.png"
  328. />
  329. </td>
  330. </tr>
  331. </tbody>
  332. </table>
  333. </div>
  334. </template>
  335. <script lang="ts" setup>
  336. import { defineComponent, computed } from 'vue'
  337. import jspdf from 'jspdf'
  338. import html2canvas from 'html2canvas'
  339. import dayjs from 'dayjs'
  340. import { downloadPDF } from '@/api/indent'
  341. defineComponent({
  342. name: 'ComponentExportQuota2',
  343. })
  344. const {
  345. city = '',
  346. city2 = '',
  347. productInfo = {} as any,
  348. exportForm = {} as any,
  349. step2FormList = [],
  350. step3FormList = [],
  351. creatorOptions = [],
  352. } = defineProps<{
  353. city: string
  354. city2: string
  355. productInfo: object
  356. exportForm: object
  357. step2FormList: any[]
  358. step3FormList: any[]
  359. creatorOptions: any[]
  360. }>()
  361. const city_short = {
  362. SYD: 'Sydney',
  363. Melb: 'Melbourne',
  364. Brisbane: 'Brisbane',
  365. SA: 'Adelaide',
  366. WA: 'Perth',
  367. US: 'US',
  368. '': '',
  369. } as any
  370. const getLogoPath = () => {
  371. return new URL('/assets/logo.png', import.meta.url).href
  372. }
  373. const getFreightType = (row: any) => {
  374. let result = 'Sea'
  375. if (row.typeNumber < 2) {
  376. result = 'Air'
  377. }
  378. if (row.typeNumber === 3) {
  379. const temp: any[] = step2FormList.filter(
  380. (i: any) => i.number === row.number,
  381. )
  382. if (
  383. temp.length &&
  384. temp[0].fclData?.method_fcl &&
  385. ['快递', '空运'].includes(temp[0].fclData.method_fcl)
  386. ) {
  387. result = 'Air'
  388. }
  389. }
  390. return result
  391. }
  392. const generatePDF = () => {
  393. downloadPDF({
  394. content: document.getElementById('pdfTarget')?.outerHTML,
  395. title: 'test_pdf',
  396. create_time: '',
  397. }).then((res: any) => {
  398. console.log(res, 'res pdf')
  399. if (res.result?.length) {
  400. window.open(import.meta.env.VITE_API_PREFIX + '/' + res.result)
  401. }
  402. })
  403. }
  404. // 把这个方法暴露给父组件, 否则无法在父组件被调用
  405. defineExpose({ generatePDF })
  406. const shouldSplit = (
  407. nodes: HTMLElement[],
  408. index: number,
  409. pageHeight: number,
  410. ) => {
  411. // 计算当前这块dom是否跨越了a4大小,以此分割
  412. // console.log(index, 'index start')
  413. if (
  414. nodes[index].offsetTop + nodes[index].clientHeight < pageHeight &&
  415. nodes[index + 1] &&
  416. nodes[index + 1].offsetTop + nodes[index + 1].clientHeight > pageHeight
  417. ) {
  418. return true
  419. }
  420. return false
  421. }
  422. const mainPicture = computed(() => {
  423. if (
  424. Array.isArray(productInfo.product_image) &&
  425. productInfo.product_image.length
  426. ) {
  427. return productInfo.product_image[0]
  428. }
  429. return ''
  430. })
  431. const otherPicture = computed(() => {
  432. if (
  433. Array.isArray(productInfo.product_image) &&
  434. productInfo.product_image.length > 1
  435. ) {
  436. return productInfo.product_image.slice(1)
  437. }
  438. return []
  439. })
  440. const computedCreator = computed(() => {
  441. let result = ''
  442. if (creatorOptions.length) {
  443. const a: any[] = creatorOptions.filter(
  444. (i: any) => i.value === exportForm.saleperson,
  445. )
  446. if (a.length) result = a[0].label
  447. }
  448. return result
  449. })
  450. </script>
  451. <style lang="scss" scoped>
  452. $subColor: #777;
  453. .pdf-wrap {
  454. position: fixed;
  455. // top: 0;
  456. // left: 0;
  457. // z-index: 9999;
  458. top: -9999px;
  459. right: -9999px;
  460. box-sizing: border-box;
  461. width: 21cm;
  462. margin: 0 auto 12px;
  463. box-shadow: 1px 1px 2pt 0px $subColor;
  464. }
  465. </style>