edit.vue 70 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457
  1. <template>
  2. <div>
  3. <div
  4. v-if="loading"
  5. v-loading="true"
  6. class="view-window"
  7. ></div>
  8. <div class="flex fixed-button-area">
  9. <el-button
  10. type="info"
  11. @click="formVisible = !formVisible"
  12. >
  13. <el-icon
  14. size="18px"
  15. color="#fff"
  16. >
  17. <Switch />
  18. </el-icon>
  19. &nbsp;{{ formVisible ? 'Preview PDF' : 'Show Form' }}
  20. </el-button>
  21. <el-button
  22. type="primary"
  23. :disabled="!formVisible"
  24. @click="checkForm(mainForm)"
  25. >
  26. <el-icon
  27. size="20px"
  28. color="#fff"
  29. >
  30. <FolderAdd />
  31. </el-icon>
  32. &nbsp;Save & Attach
  33. </el-button>
  34. </div>
  35. <div
  36. v-show="formVisible"
  37. class="screen"
  38. >
  39. <div class="flex between page-title-wrap">
  40. <div class="flex">
  41. <el-icon
  42. size="36px"
  43. :color="variables.mainColor"
  44. >
  45. <ShoppingCart />
  46. </el-icon>
  47. <div class="page-title">New Purchase Order</div>
  48. </div>
  49. </div>
  50. <el-form
  51. ref="mainForm"
  52. :model="form"
  53. :rules="formRule"
  54. label-width="120px"
  55. >
  56. <div class="flex start">
  57. <div class="layout-left">
  58. <el-form-item label="Order Type">
  59. <!-- <el-select
  60. v-model="form.Order_Type"
  61. style="width: 100%"
  62. placeholder="please select order type"
  63. >
  64. <el-option
  65. v-for="(item, index) in orderTypeList"
  66. :key="index"
  67. :label="item.label"
  68. :value="item.label"
  69. />
  70. </el-select> -->
  71. <el-radio-group v-model="form.Order_Type">
  72. <el-radio
  73. v-for="(item, index) in orderTypeList"
  74. :key="index"
  75. :label="item.label"
  76. >
  77. {{ item.label }}
  78. </el-radio>
  79. </el-radio-group>
  80. </el-form-item>
  81. <el-form-item
  82. label="Artwork Link"
  83. prop="Artwork_Link"
  84. >
  85. <el-input
  86. v-model="form.Artwork_Link"
  87. placeholder="Please input"
  88. type="textarea"
  89. :rows="3"
  90. ></el-input>
  91. </el-form-item>
  92. <el-form-item label="Currency">
  93. <el-select
  94. v-model="form.Currency"
  95. style="width: 100%"
  96. placeholder="please select currency"
  97. >
  98. <el-option
  99. v-for="(item, index) in currencyList"
  100. :key="index"
  101. :label="item.label"
  102. :value="item.label"
  103. />
  104. </el-select>
  105. </el-form-item>
  106. </div>
  107. <div class="layout-right flex-auto">
  108. <div style="position: absolute; right: 24px; top: 0">
  109. <el-form-item label="Template:">
  110. <el-select
  111. v-model="currentCompany"
  112. style="width: 150px"
  113. @change="onCompanyTemplateChange"
  114. >
  115. <el-option
  116. v-for="company in companyList"
  117. :key="company.id"
  118. :value="company.id"
  119. :label="company.label"
  120. ></el-option>
  121. </el-select>
  122. </el-form-item>
  123. </div>
  124. <div class="company-info">
  125. <div class="company-name">{{ computedCompany.name }}</div>
  126. <div class="flex center">
  127. <div class="company-addr">
  128. 地址:&nbsp;{{ computedCompany.addr }}
  129. </div>
  130. <div class="company-phone">
  131. &nbsp;|&nbsp;电话:&nbsp;{{ computedCompany.phone }}
  132. </div>
  133. <div
  134. v-show="computedCompany.fax"
  135. class="company-fax"
  136. >
  137. &nbsp;|&nbsp;传真:&nbsp;{{ computedCompany.phone }}
  138. </div>
  139. </div>
  140. </div>
  141. <div class="pdf-title">采购合同</div>
  142. <div class="pdf-title-bg">
  143. <div class="left"></div>
  144. <div class="right"></div>
  145. </div>
  146. <div class="flex start between form-area">
  147. <div class="form-area-left">
  148. <div class="">
  149. <span style="color: #f56c6c">*</span>
  150. 供应商(乙方) :
  151. </div>
  152. <div>{{ computedVendor.Primary_Contact_name }}</div>
  153. <el-form-item
  154. label-width="0"
  155. prop="currentVendor"
  156. >
  157. <el-select-v2
  158. v-model="form.currentVendor"
  159. :remote-method="getSupplierLists"
  160. remote
  161. style="width: 100%"
  162. :loading="vendorLoading"
  163. :options="vendorList"
  164. filterable
  165. clearable
  166. ></el-select-v2>
  167. </el-form-item>
  168. <div style="white-space: pre-wrap">
  169. {{ computedVendor.PDF_display }}
  170. </div>
  171. </div>
  172. <div class="form-area-right">
  173. <el-form-item label="采购订单 #:">
  174. Generate once PO submitted
  175. </el-form-item>
  176. <el-form-item label="订单号 # :">
  177. <el-input
  178. v-model="form.Reference"
  179. disabled
  180. ></el-input>
  181. </el-form-item>
  182. <el-form-item
  183. label="日期:"
  184. prop="PO_Date"
  185. >
  186. <el-date-picker
  187. v-model="form.PO_Date"
  188. format="YYYY-MM-DD"
  189. value-format="YYYY-MM-DD"
  190. style="width: 100%"
  191. type="date"
  192. placeholder=""
  193. ></el-date-picker>
  194. </el-form-item>
  195. <el-form-item label="付款方式 :">
  196. <el-select
  197. v-model="form.Payment_Terms"
  198. style="width: 100%"
  199. >
  200. <el-option
  201. v-for="(item, index) in supplierPaymentTermsLists"
  202. :key="index"
  203. :value="item.label"
  204. :label="item.label"
  205. ></el-option>
  206. </el-select>
  207. </el-form-item>
  208. <el-form-item label="参考 :">
  209. <el-input v-model="form.Title"></el-input>
  210. </el-form-item>
  211. <!-- 退税版没有这几项 -->
  212. <template
  213. v-if="typeof computedCompany.taxReimbursement === 'undefined'"
  214. >
  215. <el-form-item label="收货详情 :">
  216. <el-select
  217. v-model="form.field9"
  218. style="width: 100%"
  219. >
  220. <el-option
  221. v-for="(item, index) in addressList"
  222. :key="index"
  223. :value="item.label"
  224. :label="item.label"
  225. ></el-option>
  226. </el-select>
  227. </el-form-item>
  228. <el-form-item
  229. label="收件人 :"
  230. prop="field7"
  231. >
  232. <el-input v-model="form.field7"></el-input>
  233. </el-form-item>
  234. <el-form-item
  235. label="联系电话 :"
  236. prop="field8"
  237. >
  238. <el-input v-model="form.field8"></el-input>
  239. </el-form-item>
  240. </template>
  241. <el-form-item
  242. label="工厂交货日期 :"
  243. prop="field6"
  244. >
  245. <el-date-picker
  246. v-model="form.field6"
  247. style="width: 100%"
  248. format="YYYY-MM-DD"
  249. value-format="YYYY-MM-DD"
  250. ></el-date-picker>
  251. </el-form-item>
  252. </div>
  253. </div>
  254. <div class="product-table-separator"></div>
  255. <div class="product-table">
  256. <table
  257. border="0"
  258. cellspacing="0"
  259. >
  260. <tr>
  261. <th class="product">Product</th>
  262. <th class="quantity">Quantity</th>
  263. <th class="rate">Rate</th>
  264. <th class="requirement">Requirement</th>
  265. <th class="amount">Amount</th>
  266. <th class="discount">Discount</th>
  267. <th class="action">Action</th>
  268. </tr>
  269. <tr
  270. v-for="(product, index) in form.productList"
  271. :key="index"
  272. >
  273. <td class="product">
  274. <el-form-item
  275. label=""
  276. label-width="0"
  277. :prop="'productList.' + index + '.id'"
  278. :rules="{
  279. required: true,
  280. message: '必填项',
  281. trigger: ['blur', 'change'],
  282. }"
  283. >
  284. <el-select-v2
  285. v-model="product.id"
  286. style="width: 100%; margin-bottom: 4pt"
  287. :options="product.candidate"
  288. :loading="productLoading"
  289. :remote-method="(e:string) => getProductList(e, product)"
  290. remote
  291. filterable
  292. clearable
  293. @change="(e) => onProductSelect(e, product)"
  294. ></el-select-v2>
  295. </el-form-item>
  296. <el-input
  297. v-model="product.desc"
  298. type="textarea"
  299. placeholder="Free text"
  300. ></el-input>
  301. </td>
  302. <td class="quantity">
  303. <el-form-item
  304. label=""
  305. label-width="0"
  306. :prop="'productList.' + index + '.quantity'"
  307. :rules="{
  308. required: true,
  309. message: '必填项',
  310. trigger: ['blur', 'change'],
  311. }"
  312. >
  313. <el-input
  314. v-model="product.quantity"
  315. type="number"
  316. min="1"
  317. @change="(e) => onInputChange(e, product, 'quantity')"
  318. ></el-input>
  319. </el-form-item>
  320. </td>
  321. <td class="rate">
  322. <el-form-item
  323. label=""
  324. label-width="0"
  325. :prop="'productList.' + index + '.rate'"
  326. :rules="{
  327. required: true,
  328. message: '必填项',
  329. trigger: ['blur', 'change'],
  330. }"
  331. >
  332. <el-input
  333. v-model="product.rate"
  334. type="number"
  335. @change="(e) => onInputChange(e, product, 'rate')"
  336. ></el-input>
  337. </el-form-item>
  338. </td>
  339. <td class="requirement">
  340. <el-input
  341. v-model="product.requirement"
  342. rows="2"
  343. type="textarea"
  344. placeholder="Free text"
  345. ></el-input>
  346. </td>
  347. <td class="amount">
  348. <el-input
  349. v-model="product.amount"
  350. type="number"
  351. disabled
  352. ></el-input>
  353. </td>
  354. <td class="discount">
  355. <el-input
  356. v-model="product.discount"
  357. type="number"
  358. @change="(e) => onInputChange(e, product, 'discount')"
  359. ></el-input>
  360. </td>
  361. <td class="action">
  362. <el-button
  363. size="small"
  364. type="danger"
  365. plain
  366. @click="form.productList.splice(index, 1)"
  367. >
  368. Delete
  369. </el-button>
  370. </td>
  371. </tr>
  372. </table>
  373. </div>
  374. <div>
  375. <el-button
  376. type="primary"
  377. size="small"
  378. plain
  379. @click="addRow"
  380. >
  381. Add Row
  382. </el-button>
  383. </div>
  384. <div class="flex end">
  385. <div class="product-total-table">
  386. <div class="total-item">
  387. <div class="label">Sub Total</div>
  388. <div class="value">{{ subTotal.toFixed(2) }}</div>
  389. </div>
  390. <div class="total-item">
  391. <div class="label">Total Discount</div>
  392. <div class="value">
  393. {{ totalDiscount.toFixed(2) }}
  394. </div>
  395. </div>
  396. <div class="total-item">
  397. <div class="label">Adjustment</div>
  398. <div class="value">
  399. <el-input
  400. v-model="adjustment"
  401. style="width: 100%"
  402. type="number"
  403. ></el-input>
  404. </div>
  405. </div>
  406. <div class="total-item">
  407. <div class="label">Grand Total</div>
  408. <div class="value">
  409. {{ grandTotal.toFixed(2) }}
  410. </div>
  411. </div>
  412. </div>
  413. </div>
  414. <div class="note-form-area">
  415. <div class="sub-form-title">注意事项:</div>
  416. <el-form-item label="印刷质量:">
  417. <el-input
  418. v-model="form.field12"
  419. type="textarea"
  420. rows="3"
  421. ></el-input>
  422. </el-form-item>
  423. <el-form-item label="产品质量:">
  424. <el-input
  425. v-model="form.field13"
  426. type="textarea"
  427. rows="3"
  428. ></el-input>
  429. </el-form-item>
  430. <el-form-item label="质量承诺:">
  431. <el-input
  432. v-model="form.field10"
  433. type="textarea"
  434. rows="3"
  435. ></el-input>
  436. </el-form-item>
  437. <el-form-item label="箱子箱唛:">
  438. <el-input
  439. v-model="form.field11"
  440. type="textarea"
  441. rows="3"
  442. ></el-input>
  443. </el-form-item>
  444. </div>
  445. <div class="sub-form-title">服务条款</div>
  446. <template v-if="currentCompany === 'PangeaTaxReimbursement'">
  447. <div class="rule-item flex">
  448. 一、合同价为含税价,税率为:
  449. <span style="color: #f56c6c">*</span>
  450. <el-form-item
  451. label-width="10"
  452. prop="field4"
  453. :rules="{
  454. required: true,
  455. message: '必填项',
  456. trigger: ['blur', 'change'],
  457. }"
  458. >
  459. <el-input
  460. v-model="form.field4"
  461. size="small"
  462. style="width: 120px"
  463. >
  464. <template #append>%</template>
  465. </el-input>
  466. </el-form-item>
  467. </div>
  468. <div class="rule-item flex">
  469. 二、运费条款:
  470. <span style="color: #f56c6c">*</span>
  471. <el-form-item label-width="10">
  472. <el-select
  473. v-model="form.field5"
  474. size="small"
  475. style="width: 110px"
  476. >
  477. <el-option
  478. v-for="(item, index) in field5_lists"
  479. :key="index"
  480. :label="item.label"
  481. :value="item.label"
  482. ></el-option>
  483. </el-select>
  484. </el-form-item>
  485. </div>
  486. </template>
  487. <div
  488. v-for="(item, index) in currentServiceRule"
  489. :key="index"
  490. class="rule-item"
  491. >
  492. {{ item }}
  493. </div>
  494. </div>
  495. </div>
  496. </el-form>
  497. <br />
  498. </div>
  499. <div
  500. class="print"
  501. :class="{ hidden: formVisible && !loading }"
  502. >
  503. <div class="pdf-wrap">
  504. <div
  505. id="pdfElement"
  506. ref="pdfElement"
  507. class="preview-area"
  508. >
  509. <div class="company-info">
  510. <div class="company-name">{{ computedCompany.name }}</div>
  511. <div class="flex center">
  512. <div class="company-addr">
  513. 地址:&nbsp;{{ computedCompany.addr }}
  514. </div>
  515. <div class="company-phone">
  516. &nbsp;|&nbsp;电话:&nbsp;{{ computedCompany.phone }}
  517. </div>
  518. <div
  519. v-show="computedCompany.fax"
  520. class="company-fax"
  521. >
  522. &nbsp;|&nbsp;传真:&nbsp;{{ computedCompany.phone }}
  523. </div>
  524. </div>
  525. </div>
  526. <div>
  527. <table class="pdf-title-bg">
  528. <tr>
  529. <td class="left">
  530. <div>&nbsp;</div>
  531. </td>
  532. <td class="center">
  533. <div class="pdf-title">采购合同</div>
  534. </td>
  535. <td class="right">
  536. <div>&nbsp;</div>
  537. </td>
  538. </tr>
  539. </table>
  540. </div>
  541. <div class="form-area flex start">
  542. <div class="flex-auto">
  543. <div>供应商(乙方) :</div>
  544. <div
  545. v-if="computedVendor.Primary_Contact_name"
  546. class="column-vendor"
  547. >
  548. {{ computedVendor.Primary_Contact_name }}
  549. </div>
  550. <div class="column-vendor">
  551. {{ computedVendor.label }}
  552. </div>
  553. <div class="column-vendor">
  554. {{ computedVendor.PDF_display }}
  555. </div>
  556. </div>
  557. <div class="flex-auto">
  558. <div class="flex">
  559. <div class="column-form-label">采购订单 #:</div>
  560. <div class="column-form-value">
  561. {{ longPONumber || '' }}
  562. </div>
  563. </div>
  564. <div class="flex">
  565. <div class="column-form-label">订单号 # :</div>
  566. <div class="column-form-value">
  567. {{ form.Reference }}
  568. </div>
  569. </div>
  570. <div class="flex">
  571. <div class="column-form-label">日期 :</div>
  572. <div class="column-form-value">
  573. {{ form.PO_Date }}
  574. </div>
  575. </div>
  576. <div class="flex">
  577. <div class="column-form-label">付款方式 :</div>
  578. <div class="column-form-value">
  579. {{
  580. form.Payment_Terms !== '-None-' ? form.Payment_Terms : ''
  581. }}
  582. </div>
  583. </div>
  584. <div class="flex start">
  585. <div class="column-form-label">参考 :</div>
  586. <div class="column-form-value">
  587. {{ form.Title }}
  588. </div>
  589. </div>
  590. <template
  591. v-if="typeof computedCompany.taxReimbursement === 'undefined'"
  592. >
  593. <div class="flex start">
  594. <div class="column-form-label">收货详情 :</div>
  595. <div class="column-form-value">
  596. {{ form.field9 }}
  597. </div>
  598. </div>
  599. <div class="flex">
  600. <div class="column-form-label">收件人 :</div>
  601. <div class="column-form-value">
  602. {{ form.field7 }}
  603. </div>
  604. </div>
  605. <div class="flex">
  606. <div class="column-form-label">联系电话 :</div>
  607. <div class="column-form-value">
  608. {{ form.field8 }}
  609. </div>
  610. </div>
  611. </template>
  612. <div class="flex">
  613. <div class="column-form-label">工厂交货日期 :</div>
  614. <div class="column-form-value">
  615. {{ form.field6 }}
  616. </div>
  617. </div>
  618. </div>
  619. </div>
  620. <div class="product-table-separator"></div>
  621. <div class="product-table">
  622. <table
  623. border="0"
  624. cellspacing="0"
  625. >
  626. <tr>
  627. <th class="row-index">#</th>
  628. <th>品目</th>
  629. <th>要求</th>
  630. <th>数量</th>
  631. <th>价钱</th>
  632. <th>金额</th>
  633. </tr>
  634. <tr
  635. v-for="(product, index) in form.productList"
  636. :key="index"
  637. >
  638. <td class="row-index">{{ index + 1 }}.</td>
  639. <td>
  640. <div>{{ product.name }}</div>
  641. <div class="desc">{{ product.desc }}</div>
  642. </td>
  643. <td>
  644. {{ product.requirement }}
  645. </td>
  646. <td>
  647. {{ product.quantity }}
  648. </td>
  649. <td>
  650. {{ currentCurrency.country
  651. }}{{ currentCurrency.symbol }}&nbsp;{{
  652. Number(product.rate).toFixed(computedDeci)
  653. }}
  654. </td>
  655. <td>
  656. {{ currentCurrency.country
  657. }}{{ currentCurrency.symbol }}&nbsp;{{
  658. Number(product.amount).toFixed(computedDeci)
  659. }}
  660. </td>
  661. </tr>
  662. </table>
  663. </div>
  664. <!-- <br /> -->
  665. <div class="flex between start">
  666. <div class="note-form-area">
  667. <div class="sub-form-title">注意事项:</div>
  668. <div class="">
  669. <div class="label">印刷质量:</div>
  670. <div class="value">{{ form.field12 }}</div>
  671. </div>
  672. <div class="">
  673. <div class="label">产品质量:</div>
  674. <div class="value">{{ form.field13 }}</div>
  675. </div>
  676. <div class="">
  677. <div class="label">质量承诺:</div>
  678. <div class="value">{{ form.field10 }}</div>
  679. </div>
  680. <div class="">
  681. <div class="label">箱子箱唛:</div>
  682. <div class="value">{{ form.field11 }}</div>
  683. </div>
  684. </div>
  685. <div class="product-total-table">
  686. <div class="total-item flex">
  687. <div class="label">小计&nbsp;</div>
  688. <div class="value">
  689. {{ currentCurrency.country
  690. }}{{ currentCurrency.symbol }}&nbsp;{{
  691. subTotal.toFixed(computedDeci)
  692. }}
  693. </div>
  694. </div>
  695. <div class="total-item flex">
  696. <div class="label">合计&nbsp;</div>
  697. <div class="value">
  698. {{ currentCurrency.country
  699. }}{{ currentCurrency.symbol }}&nbsp;{{
  700. grandTotal.toFixed(computedDeci)
  701. }}
  702. </div>
  703. </div>
  704. </div>
  705. </div>
  706. <br />
  707. <div class="sub-form-title">服务条款</div>
  708. <template v-if="currentCompany === 'PangeaTaxReimbursement'">
  709. <div class="rule-item">
  710. 一、合同价为含税价,税率为: {{ form.field4 }} %
  711. </div>
  712. <div class="rule-item">二、运费条款: {{ form.field5 }}</div>
  713. </template>
  714. <div
  715. v-for="(item, index) in currentServiceRule"
  716. :key="index"
  717. class="rule-item"
  718. >
  719. {{ item }}
  720. </div>
  721. <br />
  722. <div class="signature-area">
  723. <div class="company-seal"></div>
  724. <div class="first-party">
  725. <div
  726. class="flex"
  727. style="align-items: flex-end"
  728. >
  729. <div v-show="!computedCompany.label.includes('Pangea')">
  730. 甲方(盖章):
  731. </div>
  732. <div
  733. class="sign-wrap"
  734. :class="{ pangea: computedCompany.label.includes('Pangea') }"
  735. >
  736. <img
  737. v-if="computedCompany.signPath"
  738. :src="computedCompany.signPath"
  739. alt=""
  740. />
  741. </div>
  742. </div>
  743. <div class="">代表人: {{ userInfo.full_name }}</div>
  744. <div class="">日期: {{ form.PO_Date }}</div>
  745. </div>
  746. <div class="second-party">
  747. <div class="">乙方(盖章):</div>
  748. </div>
  749. </div>
  750. </div>
  751. </div>
  752. <div
  753. v-if="computedCompany.taxReimbursement"
  754. class="pdf-wrap"
  755. >
  756. <div
  757. id=""
  758. ref="pdfElement2"
  759. class="preview-area2"
  760. >
  761. <div class="supplier-name">
  762. {{ computedVendor.Suppliers_Name || computedVendor.Vendor_Name }}
  763. </div>
  764. <div class="billing-addr">
  765. {{ computedVendor.Billing_Address || '' }}
  766. </div>
  767. <div class="flex around contact-info">
  768. <div class="flex">
  769. <div>Tel No.:</div>
  770. <div>&nbsp;{{ computedVendor.Phone }}</div>
  771. </div>
  772. <div class="flex center">
  773. <div>E-mail:</div>
  774. <div class="">&nbsp;{{ computedVendor.Email }}</div>
  775. </div>
  776. </div>
  777. <div class="base-info-area">
  778. <div class="flex">
  779. <div class="flex start left">
  780. <div class="base-info-label">TO:</div>
  781. <div
  782. v-if="currentCompany === 'AZYTaxReimbursement'"
  783. class="base-info-value"
  784. >
  785. Azy Trading Pty Ltd
  786. </div>
  787. <div
  788. v-else-if="currentCompany === 'FOTTaxReimbursement'"
  789. class="base-info-value"
  790. >
  791. Fair Ocean Trading Australia Pty Ltd
  792. </div>
  793. </div>
  794. <div class="right flex">
  795. <div class="base-info-label">Date:</div>
  796. <div class="base-info-value">{{ form.PO_Date }}</div>
  797. </div>
  798. </div>
  799. <div class="flex start">
  800. <div class="left flex start">
  801. <div class="base-info-label">ADD:</div>
  802. <div>
  803. <div
  804. v-if="currentCompany === 'AZYTaxReimbursement'"
  805. class="base-info-value"
  806. >
  807. UNIT 12,21/F WAYSON COMM BLDG NO 28 CONNAUGHT RD WEST SHEUNG
  808. WAN, HK
  809. </div>
  810. <div
  811. v-else-if="currentCompany === 'FOTTaxReimbursement'"
  812. class="base-info-value"
  813. >
  814. 15/10 Chilvers Road, Thornleigh, NSW 2120
  815. </div>
  816. </div>
  817. </div>
  818. <div class="right flex">
  819. <div class="base-info-label">P/I. NO.</div>
  820. <div class="base-info-value">{{ longPONumber }}</div>
  821. </div>
  822. </div>
  823. <div class="flex start">
  824. <div class="left flex">
  825. <div class="base-info-label">ATTN:</div>
  826. <div class="base-info-value">Accounts</div>
  827. </div>
  828. <div class="right flex">
  829. <div class="base-info-label">Reference:</div>
  830. <div class="base-info-value">{{ PONumber }}</div>
  831. </div>
  832. </div>
  833. <div class="flex start">
  834. <div class="flex left">
  835. <template v-if="currentCompany === 'FOTTaxReimbursement'">
  836. <div class="base-info-label">TEL NO:</div>
  837. <div class="base-info-value">02 9008 1322</div>
  838. </template>
  839. <div v-else></div>
  840. </div>
  841. <div class="right flex">
  842. <div class="base-info-label">Payment Term:</div>
  843. <div class="base-info-value">
  844. {{ computedVendor.Payment_Terms }}
  845. </div>
  846. </div>
  847. </div>
  848. </div>
  849. <div class="table-title">PROFORMA INVOICE</div>
  850. <div class="product-table">
  851. <table cellspacing="0">
  852. <tr>
  853. <th>ITEM</th>
  854. <th>DESCRIPTION</th>
  855. <th>QTY(pcs)</th>
  856. <th>UNIT PRICE</th>
  857. <th>TOPTAL VALUE</th>
  858. </tr>
  859. <tr
  860. v-for="(product, index) in form.productList"
  861. :key="index"
  862. >
  863. <td>
  864. <div>{{ product.name }}</div>
  865. </td>
  866. <td>
  867. <div class="desc">{{ product.desc }}</div>
  868. </td>
  869. <td>
  870. {{ product.quantity }}
  871. </td>
  872. <td>
  873. {{ currentCurrency.country
  874. }}{{ currentCurrency.symbol }}&nbsp;{{
  875. Number(product.rate).toFixed(computedDeci)
  876. }}
  877. </td>
  878. <td>
  879. {{ currentCurrency.country
  880. }}{{ currentCurrency.symbol }}&nbsp;{{
  881. Number(product.amount).toFixed(computedDeci)
  882. }}
  883. </td>
  884. </tr>
  885. </table>
  886. <div class="flex end">
  887. <div
  888. class="flex"
  889. style="margin-top: 10pt"
  890. >
  891. <div style="margin: 0 100pt 0 0">TOTAL</div>
  892. <div>
  893. {{ currentCurrency.country
  894. }}{{ currentCurrency.symbol }}&nbsp;{{ subTotal }}
  895. </div>
  896. </div>
  897. </div>
  898. </div>
  899. <br />
  900. <div class="bank-info-area">
  901. <div class="flex">
  902. <div>Bank Information</div>
  903. <div></div>
  904. </div>
  905. <div class="flex">
  906. <div class="bank-info-label">Beneficiary:</div>
  907. <div class="bank-info-value">
  908. {{ computedVendor.Suppliers_Name }}dd
  909. </div>
  910. </div>
  911. <div class="flex">
  912. <div class="bank-info-label">Bank:</div>
  913. <div class="bank-info-value">
  914. {{ computedVendor.Bank_Branch_Name }}
  915. </div>
  916. </div>
  917. <div class="flex">
  918. <div class="bank-info-label">Bank Add:</div>
  919. <div class="bank-info-value">{{}}</div>
  920. </div>
  921. <div class="flex">
  922. <div class="bank-info-label">A/C No.:</div>
  923. <div class="bank-info-value">
  924. {{ computedVendor.Bank_Account }}
  925. </div>
  926. </div>
  927. <div class="flex">
  928. <div class="bank-info-label">Bank Code:</div>
  929. <div class="bank-info-value">{{}}</div>
  930. </div>
  931. <div class="flex">
  932. <div class="bank-info-label">SWIFT CODE:</div>
  933. <div class="bank-info-value">
  934. {{ computedVendor.Swift_Code_IBAN }}
  935. </div>
  936. </div>
  937. <div class="flex">
  938. <div class="bank-info-label">Add:</div>
  939. <div class="bank-info-value">{{}}</div>
  940. </div>
  941. </div>
  942. <br />
  943. <div class="flex between start">
  944. <div class="">
  945. <div class="">BUYERS' SIGNATURE</div>
  946. <div class="sign-wrap">
  947. <img
  948. v-if="computedCompany.signPath"
  949. :src="computedCompany.signPath"
  950. alt=""
  951. />
  952. </div>
  953. </div>
  954. <div class="">
  955. <div class="">SELLER'S SIGNATURE</div>
  956. </div>
  957. </div>
  958. </div>
  959. </div>
  960. </div>
  961. </div>
  962. </template>
  963. <script lang="ts">
  964. export default defineComponent({
  965. name: 'PurchaseOrderEdit',
  966. })
  967. </script>
  968. <script lang="ts" setup>
  969. import { computed, defineComponent, ref, unref } from 'vue'
  970. import { useRoute } from 'vue-router'
  971. import {
  972. ElButton,
  973. ElSelectV2,
  974. ElSelect,
  975. ElOption,
  976. ElIcon,
  977. ElInput,
  978. ElForm,
  979. ElFormItem,
  980. ElDatePicker,
  981. ElMessage,
  982. ElNotification,
  983. ElRadioGroup,
  984. ElRadio,
  985. } from 'element-plus'
  986. import type { FormInstance, FormRules } from 'element-plus'
  987. import { ShoppingCart, FolderAdd, Switch } from '@element-plus/icons-vue'
  988. import axios from 'axios'
  989. import dayjs from 'dayjs'
  990. import jspdf from 'jspdf'
  991. import html2canvas from 'html2canvas'
  992. import utils from '@/utils/index'
  993. import variables from '@/assets/css/var.module.scss'
  994. import {
  995. ServiceTypeKeyEnum,
  996. TypeService,
  997. ISelectItem,
  998. IProductItem,
  999. IUser,
  1000. ICompanyItem,
  1001. IVendorItem,
  1002. IForm,
  1003. } from '@/interface'
  1004. computed(() => {
  1005. return variables
  1006. })
  1007. const formVisible = ref(true)
  1008. const loading = ref(false)
  1009. // 订单ID. 例如 5757019000000683001, 不是串号.
  1010. const POID = ref('')
  1011. // po 串号. 例如 PO2025
  1012. const PONumber = ref('')
  1013. // po 长串号. 例如 ZCPO2025
  1014. const longPONumber = ref('')
  1015. const mainForm = ref<FormInstance>()
  1016. const checkForm = function (formEl: FormInstance | undefined) {
  1017. if (!formEl) return
  1018. formEl.validate((valid, fields) => {
  1019. if (valid) {
  1020. submit()
  1021. } else {
  1022. console.log('check form has not pass!', fields)
  1023. ElMessage.error('请检查表单必填项')
  1024. }
  1025. })
  1026. }
  1027. const submit = () => {
  1028. loading.value = true
  1029. createPurchaseOrders()
  1030. .then((res) => {
  1031. console.log(res)
  1032. getPurchaseOrdersData()
  1033. .then(() => {
  1034. generatePDF()
  1035. .then(() => {
  1036. ElNotification({
  1037. duration: 0,
  1038. type: 'success',
  1039. title: '任务已成功处理',
  1040. message: '本页面将在数秒后自动跳转',
  1041. })
  1042. const url =
  1043. import.meta.env.VITE_PO_PATH +
  1044. POID.value +
  1045. import.meta.env.VITE_PO_APPEND
  1046. setTimeout(() => {
  1047. window.location.replace(url)
  1048. }, 5000)
  1049. })
  1050. .finally(() => {
  1051. loading.value = false
  1052. })
  1053. })
  1054. .catch(() => {
  1055. // loading.value = false
  1056. const msg = '获取PO详情失败, 未能正确生成PDF'
  1057. console.log(msg)
  1058. ElNotification({ duration: 0, title: msg, type: 'error' })
  1059. })
  1060. })
  1061. .catch(() => {
  1062. // loading.value = false
  1063. const msg = '创建PO失败'
  1064. console.log(msg)
  1065. ElNotification({ duration: 0, title: msg, type: 'error' })
  1066. })
  1067. }
  1068. const pdfElement = ref()
  1069. const pdfElement2 = ref()
  1070. const generatePDF = (ele = pdfElement, isFirstTime = true) => {
  1071. const A4_WIDTH = 592.28
  1072. const A4_HEIGHT = 841.89
  1073. let imageWrapper = unref(ele) // 获取DOM
  1074. // let pageHeight = (imageWrapper.scrollWidth / A4_WIDTH) * A4_HEIGHT
  1075. let pageHeight = (imageWrapper.clientWidth / A4_WIDTH) * A4_HEIGHT
  1076. let lableListID = imageWrapper.querySelectorAll('div')
  1077. // 进行分割操作,当dom内容已超出a4的高度,则将该dom前插入一个空dom,把他挤下去,分割
  1078. for (let i = 0; i < lableListID.length; i++) {
  1079. let multiple = Math.ceil(
  1080. (lableListID[i].offsetTop + lableListID[i].offsetHeight) / pageHeight,
  1081. )
  1082. if (isSplit(lableListID, i, multiple * pageHeight)) {
  1083. let divParent = lableListID[i].parentNode // 获取该div的父节点
  1084. let newNode = document.createElement('div')
  1085. newNode.className = 'empty-div'
  1086. newNode.style.background = '#fff'
  1087. let _H =
  1088. multiple * pageHeight -
  1089. (lableListID[i].offsetTop + lableListID[i].offsetHeight)
  1090. //留白
  1091. newNode.style.height = _H + 30 + 'px'
  1092. newNode.style.width = '100%'
  1093. let next = lableListID[i].nextSibling // 获取div的下一个兄弟节点
  1094. // 判断兄弟节点是否存在
  1095. if (next) {
  1096. // 存在则将新节点插入到div的下一个兄弟节点之前,即div之后
  1097. divParent.insertBefore(newNode, next)
  1098. } else {
  1099. // 不存在则直接添加到最后,appendChild默认添加到divParent的最后
  1100. divParent.appendChild(newNode)
  1101. }
  1102. }
  1103. }
  1104. return new Promise((resolve, reject) => {
  1105. html2canvas(imageWrapper, {
  1106. allowTaint: true,
  1107. useCORS: true,
  1108. backgroundColor: '#fff', //一定要设背景颜色,否则有的浏览器就会变花~,比如Edge
  1109. scale: 3, // 缩放倍率调整清晰度
  1110. }).then((canvas) => {
  1111. let pdf = new jspdf('p', 'mm', 'a4') //A4纸,纵向
  1112. let ctx = canvas.getContext('2d'),
  1113. a4ContentWidth = 190,
  1114. a4ContentHeight = 277, //A4大小,210mm x 297mm,四边各保留10mm的边距,显示区域190x277
  1115. imgHeight = Math.floor(
  1116. (a4ContentHeight / a4ContentWidth) * canvas.width,
  1117. ), //按A4显示比例换算一页图像的像素高度
  1118. renderedHeight = 0
  1119. while (renderedHeight < canvas.height) {
  1120. let page = document.createElement('canvas')
  1121. page.width = canvas.width
  1122. page.height = Math.min(imgHeight, canvas.height - renderedHeight) //可能内容不足一页
  1123. //用getImageData剪裁指定区域,并画到前面创建的canvas对象中
  1124. // @ts-ignore: Object is possibly 'null'.
  1125. page
  1126. .getContext('2d')
  1127. .putImageData(
  1128. ctx!.getImageData(
  1129. 0,
  1130. renderedHeight,
  1131. canvas.width,
  1132. Math.min(imgHeight, canvas.height - renderedHeight),
  1133. ),
  1134. 0,
  1135. 0,
  1136. )
  1137. // document.body.appendChild(page)
  1138. pdf.addImage(
  1139. page.toDataURL('image/jpeg', 0.2),
  1140. 'JPEG',
  1141. 10,
  1142. 10,
  1143. a4ContentWidth,
  1144. Math.min(
  1145. a4ContentHeight,
  1146. a4ContentWidth * (page.height / page.width),
  1147. ),
  1148. ) //添加图像到页面,保留10mm边距
  1149. renderedHeight += imgHeight
  1150. if (renderedHeight < canvas.height) pdf.addPage() //如果后面还有内容,添加一个空页
  1151. }
  1152. const prefix =
  1153. `${PONumber.value}-${computedVendor.value.label}` +
  1154. (form.value.Title?.length ? `-${form.value.Title}` : '')
  1155. const fileName = isFirstTime ? prefix : prefix + '_2'
  1156. sendPDF(pdf.output('datauristring'), fileName)
  1157. .then(() => {
  1158. // 有两个退税版模版, 需要生成两个PDF.
  1159. if (computedCompany.value.taxReimbursement === true && isFirstTime) {
  1160. ElNotification({
  1161. duration: 0,
  1162. title: '正在生成附件2 PDF',
  1163. type: 'info',
  1164. })
  1165. generatePDF(pdfElement2, false).then(() => {
  1166. resolve(true)
  1167. })
  1168. } else {
  1169. resolve(true)
  1170. }
  1171. })
  1172. .catch((e) => {
  1173. reject(e)
  1174. })
  1175. // 本地备份一下
  1176. pdf.save(fileName + '.pdf')
  1177. })
  1178. })
  1179. }
  1180. const isSplit = function (nodes: any, index: number, pageHeight: number) {
  1181. // 计算当前这块dom是否跨越了a4大小,以此分割
  1182. if (
  1183. nodes[index].offsetTop + nodes[index].offsetHeight < pageHeight &&
  1184. nodes[index + 1] &&
  1185. nodes[index + 1].offsetTop + nodes[index + 1].offsetHeight > pageHeight
  1186. ) {
  1187. return true
  1188. }
  1189. return false
  1190. }
  1191. const sendPDF = function (file: string, filename = `attachment_file`) {
  1192. const temp = file.split(',')
  1193. // @ts-ignore: Object is possibly 'null'.
  1194. const mimeType = temp[0].match(/:(.*?);/)[1]
  1195. let extName = mimeType.split('/')[1]
  1196. let bstr = window.atob(temp[1])
  1197. let n = bstr.length
  1198. let result = new Uint8Array(n)
  1199. while (n--) {
  1200. result[n] = bstr.charCodeAt(n)
  1201. }
  1202. const data = {
  1203. id: POID.value,
  1204. file: new File([result], `${filename}.${extName}`, { type: mimeType }),
  1205. }
  1206. return new Promise((resolve, reject) => {
  1207. axios
  1208. .post('/api/Purchase_orders/uploadAttachmentFile', data, {
  1209. headers: {
  1210. 'Content-Type': 'multipart/form-data',
  1211. },
  1212. })
  1213. .then((response) => {
  1214. if (response.data.code !== 1) {
  1215. const msg = '自动创建附件失败'
  1216. ElNotification({ duration: 0, title: msg, type: 'error' })
  1217. reject(msg)
  1218. return
  1219. }
  1220. ElNotification({
  1221. duration: 0,
  1222. title: '自动创建 PDF 成功',
  1223. type: 'success',
  1224. })
  1225. resolve(true)
  1226. })
  1227. .catch((e) => {
  1228. reject(e)
  1229. })
  1230. })
  1231. }
  1232. const createPurchaseOrders = function () {
  1233. const data = JSON.parse(JSON.stringify(unref(form)))
  1234. data.Status = 'Created'
  1235. data.Sub_Total = subTotal.value
  1236. data.Total_Taxes = 0
  1237. data.Total_Discount = totalDiscount.value
  1238. data.Adjustment = Number(adjustment.value)
  1239. data.Grand_Total = grandTotal.value
  1240. data.Subject = '这里不用填'
  1241. data.Created_By = {
  1242. id: userInfo.value.users_id || route.query.user,
  1243. name: userInfo.value.full_name || '',
  1244. }
  1245. data.Vendor_Name = {
  1246. id: computedVendor.value.value,
  1247. name: computedVendor.value.label,
  1248. }
  1249. // console.log(data.Related_Sales_Order, 'ddd')
  1250. data.Related_Sales_Order = {
  1251. id: form.value.saleOrderId,
  1252. name: form.value.Title,
  1253. }
  1254. data.Product_Details = data.productList.map((item: any) => {
  1255. return {
  1256. quantity: item.quantity,
  1257. Discount: item.discount || 0,
  1258. list_price: item.rate,
  1259. Tax: 0,
  1260. total: item.amount,
  1261. total_after_discount: Number(item.amount) - Number(item.discount),
  1262. net_total: Number(item.amount) - Number(item.discount),
  1263. description: item.desc,
  1264. requirement: item.requirement,
  1265. // line_tax: [],
  1266. // book: '',
  1267. // id: item.id,
  1268. product: {
  1269. // name: item.name,
  1270. id: item.id,
  1271. // Product_Code: item.Product_Code,
  1272. },
  1273. // 这两个是crm有的, 但是文档没有
  1274. List_Price_Non_Currency: item.rate,
  1275. Price_Book_Name: '',
  1276. }
  1277. })
  1278. if (data.productList) delete data.productList
  1279. console.log(data, 'create po params')
  1280. return new Promise((resolve, reject) => {
  1281. axios
  1282. .post('/api/Purchase_orders/createPurchaseOrders', data, {})
  1283. .then((response) => {
  1284. if (response.data.code !== 1) return
  1285. const res = response.data.result
  1286. console.log(res, 'create po')
  1287. if (res.data && res.data.length && res.data[0].code === 'SUCCESS') {
  1288. POID.value = res.data[0].details?.id || ''
  1289. resolve(res.data[0].details?.id)
  1290. ElNotification({
  1291. duration: 0,
  1292. title: '创建 Purchase Order 成功',
  1293. type: 'success',
  1294. })
  1295. } else {
  1296. ElNotification({
  1297. duration: 0,
  1298. title: '未能成功创建PO, 请稍后重试或者联系管理员',
  1299. type: 'error',
  1300. })
  1301. }
  1302. reject('未能成功创建PO')
  1303. })
  1304. .catch((e) => {
  1305. reject(e)
  1306. })
  1307. })
  1308. }
  1309. const getPurchaseOrdersData = function () {
  1310. return new Promise((resolve, reject) => {
  1311. axios
  1312. .post('/api/purchase_orders/getPurchaseOrdersData', {
  1313. id: POID.value,
  1314. })
  1315. .then((response) => {
  1316. if (response.data.code !== 1) {
  1317. reject(false)
  1318. return
  1319. }
  1320. const res = Array.isArray(response.data.result)
  1321. ? response.data.result[0]
  1322. : response.data.result
  1323. longPONumber.value = res.PO_Number || ''
  1324. PONumber.value = res.Purchase_Order_Number || ''
  1325. resolve(true)
  1326. })
  1327. .catch((e) => {
  1328. reject(e)
  1329. })
  1330. })
  1331. }
  1332. // getPurchaseOrdersData()
  1333. const currencyList = ref([
  1334. {
  1335. label: 'AUD',
  1336. symbol: '',
  1337. country: '',
  1338. },
  1339. {
  1340. label: 'NZD',
  1341. symbol: '',
  1342. country: '',
  1343. },
  1344. {
  1345. label: 'CNY',
  1346. symbol: '¥',
  1347. country: 'CN',
  1348. },
  1349. {
  1350. label: 'GBP',
  1351. symbol: '',
  1352. country: '',
  1353. },
  1354. {
  1355. label: 'EUR',
  1356. symbol: '',
  1357. country: '',
  1358. },
  1359. {
  1360. label: 'USD',
  1361. symbol: '',
  1362. country: '',
  1363. },
  1364. ])
  1365. const currentCurrency = computed(() => {
  1366. const temp = currencyList.value.filter((i) => i.label === form.value.Currency)
  1367. // 2是人民币
  1368. return temp.length ? temp[0] : temp[2]
  1369. })
  1370. let orderTypeList = ref([
  1371. {
  1372. label: '-None-',
  1373. },
  1374. {
  1375. label: '打样',
  1376. },
  1377. {
  1378. label: '大货',
  1379. },
  1380. ])
  1381. const form = ref<IForm>({
  1382. Order_Type: '-None-',
  1383. Artwork_Link: '',
  1384. Currency: 'CNY',
  1385. // 订单号
  1386. saleOrderId: '',
  1387. Reference: '',
  1388. // 日期
  1389. PO_Date: '',
  1390. // 付款方式
  1391. Payment_Terms: '-None-',
  1392. // 参考, so的job name
  1393. Title: '',
  1394. // 收货地址
  1395. field9: '广州市越秀区八旗二马路广东航运大厦1904室 邮编510110',
  1396. // 收件人
  1397. field7: '',
  1398. // 联系电话
  1399. field8: '18925020659',
  1400. // 工厂交货日期
  1401. field6: '',
  1402. productList: [
  1403. {
  1404. name: '',
  1405. id: '',
  1406. quantity: '',
  1407. rate: '',
  1408. requirement: '',
  1409. desc: '',
  1410. amount: 0,
  1411. discount: '',
  1412. candidate: [],
  1413. },
  1414. ] as IProductItem[],
  1415. // 印刷质量
  1416. field12: '',
  1417. // 产品质量
  1418. field13: '',
  1419. // 质量承诺
  1420. field10: '',
  1421. // 箱子箱唛
  1422. field11: '',
  1423. // 税率
  1424. field4: '0',
  1425. // 运费条款
  1426. field5: '供应商承担',
  1427. currentVendor: '',
  1428. })
  1429. const formRule = ref<FormRules>({
  1430. Artwork_Link: [
  1431. {
  1432. required: true,
  1433. message: '必填项',
  1434. trigger: 'blur',
  1435. },
  1436. ],
  1437. PO_Date: [
  1438. {
  1439. required: true,
  1440. message: '必填项',
  1441. trigger: 'blur',
  1442. },
  1443. ],
  1444. // saleOrderId: [
  1445. // {
  1446. // required: true,
  1447. // message: '必填项',
  1448. // trigger: 'blur',
  1449. // },
  1450. // ],
  1451. field7: [
  1452. {
  1453. required: true,
  1454. message: '必填项',
  1455. trigger: 'blur',
  1456. },
  1457. ],
  1458. field8: [
  1459. {
  1460. required: true,
  1461. message: '必填项',
  1462. trigger: 'blur',
  1463. },
  1464. ],
  1465. field6: [
  1466. {
  1467. required: true,
  1468. message: '必填项',
  1469. trigger: 'blur',
  1470. },
  1471. ],
  1472. currentVendor: [
  1473. {
  1474. required: true,
  1475. message: '必填项',
  1476. trigger: 'change',
  1477. },
  1478. ],
  1479. })
  1480. const emptyProductItem = {
  1481. label: '',
  1482. value: '',
  1483. candidate: [],
  1484. name: '',
  1485. desc: '',
  1486. id: '',
  1487. quantity: '',
  1488. rate: '',
  1489. requirement: '',
  1490. // 未计算优惠(discount)之前的总额
  1491. amount: 0,
  1492. // 优惠额度
  1493. discount: '',
  1494. }
  1495. const addRow = function () {
  1496. form.value.productList.push(JSON.parse(JSON.stringify(emptyProductItem)))
  1497. }
  1498. const keyNeedCompute = ['quantity', 'rate']
  1499. const onInputChange = function (e: any, obj: IProductItem, key: string) {
  1500. // console.log(e, 'input value before enfore change')
  1501. if (typeof e === 'string') {
  1502. if (e.length) {
  1503. // 强制转换为数字类型
  1504. // obj[key] = Math.round(utils.multiply(Number(e))) / 100
  1505. obj[key] = Math.round(utils.multiply(Number(e), 1000)) / 1000
  1506. // 计算每行的总额
  1507. if (keyNeedCompute.includes(key)) {
  1508. obj.amount = utils.multiply(Number(obj.quantity), Number(obj.rate))
  1509. }
  1510. } else {
  1511. // obj[key] = minValue
  1512. }
  1513. }
  1514. }
  1515. const onProductSelect = function (e: any, product: IProductItem) {
  1516. if (e) {
  1517. const temp = product.candidate.filter((i) => i.value === e)
  1518. if (temp.length) {
  1519. product.name = temp[0].label
  1520. product.Product_Code = temp[0].Product_Code
  1521. }
  1522. } else {
  1523. product.name = ''
  1524. }
  1525. }
  1526. // 小计
  1527. const subTotal = computed(() => {
  1528. return form.value.productList.reduce((total, current) => {
  1529. total = total + Number(current.amount)
  1530. return total
  1531. }, 0)
  1532. })
  1533. // 总优惠额度
  1534. const totalDiscount = computed(() => {
  1535. return form.value.productList.reduce((total, current) => {
  1536. total = total + Number(current.discount)
  1537. return total
  1538. }, 0)
  1539. })
  1540. // 价格统计修正值
  1541. const adjustment = ref<number | string>('')
  1542. // 总计
  1543. const grandTotal = computed(() => {
  1544. return subTotal.value - totalDiscount.value + Number(adjustment.value)
  1545. })
  1546. const productLoading = ref(false)
  1547. const getProductList = utils.debounce(
  1548. (keyword: string, target: IProductItem) => {
  1549. productLoading.value = true
  1550. const data = {
  1551. value: keyword.trim(),
  1552. name: 'Products',
  1553. api_name: 'Product_Name',
  1554. contains: 'contains',
  1555. page: 1,
  1556. limit: 200,
  1557. }
  1558. getSearchData(data)
  1559. .then((response) => {
  1560. if (response.data.code !== 1) return
  1561. const res = response.data.result
  1562. target.candidate = res.data.map((item: any) => {
  1563. return {
  1564. ...item,
  1565. label: item.Product_Name,
  1566. value: item.products_id || item.id,
  1567. }
  1568. })
  1569. })
  1570. .finally(() => {
  1571. productLoading.value = false
  1572. })
  1573. },
  1574. 1000,
  1575. )
  1576. const vendorList = ref<IVendorItem[]>([])
  1577. const computedVendor = computed(() => {
  1578. const temp = vendorList.value.filter(
  1579. (i) => i.value === form.value.currentVendor,
  1580. )
  1581. return temp.length
  1582. ? temp[0]
  1583. : {
  1584. Primary_Contact_name: '',
  1585. PDF_display: '',
  1586. label: '',
  1587. value: '',
  1588. }
  1589. })
  1590. const vendorLoading = ref(false)
  1591. const getSupplierLists = utils.debounce(function (string: string) {
  1592. const data = {
  1593. value: string.trim(),
  1594. name: 'Vendors',
  1595. api_name: 'Vendor_Name',
  1596. contains: 'contains',
  1597. page: 1,
  1598. limit: 200,
  1599. }
  1600. vendorLoading.value = true
  1601. getSearchData(data)
  1602. .then((response) => {
  1603. if (response.data.code !== 1) return
  1604. const res = response.data.result
  1605. vendorList.value = res.data.map((i: any) => {
  1606. return {
  1607. ...i,
  1608. label: i.Suppliers_Name || i.Vendor_Name,
  1609. value: i.supplier_id || i.id,
  1610. Primary_Contact_name: i.Primary_Contact_name || '',
  1611. PDF_display: i.PDF_display || '',
  1612. }
  1613. })
  1614. })
  1615. .finally(() => {
  1616. vendorLoading.value = false
  1617. })
  1618. }, 1000)
  1619. getSupplierLists('')
  1620. const getSearchData = async function (p: any) {
  1621. return await axios
  1622. .post('/api/common/getPublicLists', p, {
  1623. headers: { 'Content-Type': 'multipart/form-data' },
  1624. })
  1625. .then((response) => {
  1626. return response
  1627. })
  1628. }
  1629. const getPath = function (path: string) {
  1630. return new URL(path, import.meta.url).href
  1631. }
  1632. // 用于切换合同模版
  1633. const companyList = ref<ICompanyItem[]>([
  1634. {
  1635. id: 'PC',
  1636. label: 'PC',
  1637. name: '澳之原贸易有限公司',
  1638. addr: 'UNIT 12,21/F WAYSON COMM BLDG NO 28 CONNAUGHT RD WEST SHEUNG WAN,HK',
  1639. phone: '020-81609790',
  1640. signPath: getPath('/assets/sign/AZY.png'),
  1641. },
  1642. {
  1643. id: 'Pangea',
  1644. label: 'Pangea',
  1645. name: '庞吉亚国际商务服务(广州)有限公司',
  1646. addr: '广州市越秀区八旗二马路48号内自编1号主楼自编1605、1606房',
  1647. phone: '020-81609790',
  1648. fax: '020-81519492',
  1649. signPath: getPath('/assets/sign/Pangea.png'),
  1650. },
  1651. {
  1652. id: 'PangeaTaxReimbursement',
  1653. label: 'Pangea_退税版',
  1654. name: '庞吉亚国际商务服务(广州)有限公司',
  1655. addr: '广州市越秀区八旗二马路48号内自编1号主楼自编1605、1606房',
  1656. phone: '020-81609790',
  1657. fax: '020-81519492',
  1658. // 名字叫退税版, 实际上合同没有第二份pdf. 判断是否退税版的依据是键名存在与否, 而不是这个值的真假
  1659. taxReimbursement: false,
  1660. signPath: getPath('/assets/sign/Pangea.png'),
  1661. },
  1662. {
  1663. id: 'AZYTaxReimbursement',
  1664. label: 'AZY_退税版',
  1665. name: '澳之原贸易有限公司',
  1666. addr: 'UNIT 12,21/F WAYSON COMM BLDG NO 28 CONNAUGHT RD WEST SHEUNG WAN,HK',
  1667. phone: '020-81609790',
  1668. taxReimbursement: true,
  1669. signPath: getPath('/assets/sign/AZY.png'),
  1670. },
  1671. {
  1672. id: 'FOTTaxReimbursement',
  1673. label: 'FOT_退税版',
  1674. name: 'FAIR OCEAN TRADING AUSTRALIA PTY.LTD',
  1675. addr: '15/10 Chilvers Road, Thornleigh, NSW 2120',
  1676. phone: '020-81609790',
  1677. taxReimbursement: true,
  1678. signPath: getPath('/assets/sign/FOT.png'),
  1679. },
  1680. ])
  1681. // 切换模版后的处理
  1682. const onCompanyTemplateChange = function () {
  1683. // 重置这两个字段
  1684. form.value.field5 = '供应商承担'
  1685. form.value.field4 = ''
  1686. if (typeof computedCompany.value.taxReimbursement === 'undefined') {
  1687. form.value.field7 = soOwner.value
  1688. form.value.field8 = '18925020659'
  1689. form.value.field9 = '广州市越秀区八旗二马路广东航运大厦1904室 邮编510110'
  1690. } else {
  1691. // 退税版没有这三个表单项, 直接重置值. 相应的, 非退税版要重新赋值.
  1692. form.value.field7 = ''
  1693. form.value.field8 = ''
  1694. form.value.field9 = ''
  1695. }
  1696. }
  1697. // 合同的服务条款
  1698. const serviceRule = ref<TypeService>({
  1699. PC: [
  1700. '一、本采购订单签署原件一式两份,双方各持一份。',
  1701. '二、验收标准:按乙方所寄样板及国家出口标准作为验收标准。',
  1702. '三、交货期限:交货期限必须严格执行。',
  1703. '四、交货地点:甲方广州公司地址,如上收货地址。',
  1704. '五、付款方式:甲方通过银行转帐的方式付款,乙方账户信息如上。',
  1705. '六、如因未能按上述标准及期限交货而产生经济损失,将由乙方承担。',
  1706. '七、凡因执行本合同所发生的或与本合同有关的一切争议,如经友好协商不能解决时,交由广州当地法院进行判决。',
  1707. ],
  1708. Pangea: [
  1709. '一、本采购订单签署原件一式两份,双方各持一份。',
  1710. '二、验收标准:按乙方所寄样板及国家出口标准作为验收标准。',
  1711. '三、交货期限:交货期限必须严格执行。',
  1712. '四、交货地点:甲方广州公司地址,如上收货地址。',
  1713. '五、付款方式:甲方通过银行转帐的方式付款,乙方账户信息如上。',
  1714. '六、如因未能按上述标准及期限交货而产生经济损失,将由乙方承担。',
  1715. '七、凡因执行本合同所发生的或与本合同有关的一切争议,如经友好协商不能解决时,交由广州当地法院进行判决。',
  1716. ],
  1717. PangeaTaxReimbursement: [
  1718. '三、本采购订单签署原件一式两份,双方各持一份。',
  1719. '四、验收标准:按乙方所寄样板及国家出口标准作为验收标准。',
  1720. '五、交货期限:交货期限必须严格执行。',
  1721. '六、交货地点: 广州市越秀区八旗二马路48号航运大厦1605、1606房',
  1722. '七、付款方式:甲方通过银行转帐的方式付款,乙方账户信息如上。',
  1723. '八、如因未能按上述标准及期限交货而产生经济损失,将由乙方承担。',
  1724. '九、凡因执行本合同所发生的或与本合同有关的一切争议,如经友好协商不能解决时,交由广州当地法院进行判决。',
  1725. '十、我司开票资料:',
  1726. '账户名称:庞吉亚国际商务服务(广州)有限公司',
  1727. '统一社会代码:91440101MA9XUQNB4B',
  1728. '公司地址:广州市越秀区八旗二马路48号内自编1号主楼16楼自编1605、1606房 ',
  1729. '公司电话:81609790',
  1730. '基本账户:中国光大银行股份有限公司广州分行越秀支行',
  1731. '账户号码:77910180803936888',
  1732. '开户银行地址:广州市越秀区文德南路69号',
  1733. ],
  1734. AZYTaxReimbursement: [
  1735. '一、本采购订单签署原件一式两份,双方各持一份。',
  1736. '二、验收标准:按乙方所寄样板及国家出口标准作为验收标准。',
  1737. '三、交货期限:交货期限必须严格执行。',
  1738. '四、收货地址:澳竣元代收,广州市越秀区八旗二马路48号航运大厦1902房。',
  1739. '五、付款方式:甲方通过银行转帐的方式付款,乙方账户信息如上。',
  1740. '六、如因未能按上述标准及期限交货而产生经济损失,将由乙方承担。',
  1741. '七、凡因执行本合同所发生的或与本合同有关的一切争议,如经友好协商不能解决时,交由广州当地法院进行判决。',
  1742. ],
  1743. FOTTaxReimbursement: [
  1744. '一、本采购订单签署原件一式两份,双方各持一份。',
  1745. '二、验收标准:按乙方所寄样板及国家出口标准作为验收标准。',
  1746. '三、交货期限:交货期限必须严格执行。',
  1747. '四、收货地址:澳竣元代收,广州市越秀区八旗二马路48号航运大厦1902房。',
  1748. '五、付款方式:甲方通过银行转帐的方式付款,乙方账户信息如上。',
  1749. '六、如因未能按上述标准及期限交货而产生经济损失,将由乙方承担。',
  1750. '七、凡因执行本合同所发生的或与本合同有关的一切争议,如经友好协商不能解决时,交由广州当地法院进行判决。',
  1751. ],
  1752. })
  1753. const currentServiceRule = computed(() => {
  1754. return serviceRule.value[currentCompany.value as ServiceTypeKeyEnum]
  1755. })
  1756. const currentCompany = ref('PC')
  1757. const computedCompany = computed(() => {
  1758. const result = companyList.value.filter((i) => i.id === currentCompany.value)
  1759. return result.length ? result[0] : companyList.value[0]
  1760. })
  1761. // 格式化输出价格的小数位数. 退税版需要显示三位小数, 其他只需要两位小数
  1762. const computedDeci = computed(() => {
  1763. return typeof computedCompany.value.taxReimbursement !== 'undefined' ? 3 : 2
  1764. })
  1765. // 候选 收货地址
  1766. let addressList = ref<ISelectItem[]>([])
  1767. // 候选 运费条款. 目前只有庞吉亚退税版用到, 其他模版默认供应商承担
  1768. let field5_lists = ref<ISelectItem[]>([])
  1769. let supplierPaymentTermsLists = ref<ISelectItem[]>([])
  1770. loading.value = true
  1771. // 获取下拉框非动态候选数据
  1772. const p1 = axios
  1773. .post('/api/common/getfieldsData')
  1774. .then((response: any) => {
  1775. // console.log(response, '/common/getfieldsData')
  1776. const res = response.data.result
  1777. orderTypeList.value = res.Order_Type_lists.map((i: any) => {
  1778. return {
  1779. label: i,
  1780. }
  1781. })
  1782. addressList.value = res.field9_lists.map((i: any) => {
  1783. return {
  1784. label: i,
  1785. }
  1786. })
  1787. field5_lists.value = res.field5_lists
  1788. .filter((i: string) => !/^-?[Nn]one-?$/.test(i))
  1789. .map((i: any) => {
  1790. return {
  1791. label: i,
  1792. }
  1793. })
  1794. supplierPaymentTermsLists.value = res.Supplier_Payment_Terms_lists.map(
  1795. (i: any) => {
  1796. return {
  1797. label: i,
  1798. }
  1799. },
  1800. )
  1801. })
  1802. .catch((e) => {
  1803. console.log(e, '下拉框')
  1804. ElMessage.error('获取下拉框数据出错, 请联系管理员.')
  1805. })
  1806. const route = useRoute()
  1807. const productBlackList = ['4791186000046982872', '4791186000046982896']
  1808. const soOwner = ref('')
  1809. // 获取销售订单详情
  1810. const p2 = axios
  1811. .post('/api/common/getSalesOrdersData', { id: route.params.id })
  1812. .then((response) => {
  1813. if (response.data.code !== 1) return
  1814. const res = response.data.result
  1815. form.value.field7 = res.Owner_name || res.Owner.name || ''
  1816. soOwner.value = res.Owner_name || res.Owner.name || ''
  1817. form.value.Title = res.Contract_Title || res.Sales_Order_Title_Job_Name || ''
  1818. form.value.saleOrderId = res.sales_orders_id || res.id || ''
  1819. form.value.PO_Date = dayjs(new Date()).format('YYYY-MM-DD')
  1820. form.value.Reference = res.Reference || ''
  1821. let temp = []
  1822. if (res.details) {
  1823. temp = res.details.filter(
  1824. (item: any) => !productBlackList.includes(item.product_id),
  1825. )
  1826. } else if (res.Product_Details) {
  1827. temp = res.Product_Details.filter(
  1828. (item: any) => !productBlackList.includes(item.product.id),
  1829. )
  1830. }
  1831. if (!temp.length) return
  1832. form.value.productList = []
  1833. temp.forEach((item: any) => {
  1834. form.value.productList.push(
  1835. Object.assign({}, emptyProductItem, {
  1836. candidate: [
  1837. {
  1838. Product_Code: item.product_Product_Code,
  1839. label: item.product.name,
  1840. value: item.product.id,
  1841. },
  1842. ],
  1843. Product_Code: item.product.Product_Code,
  1844. id: item.product.id,
  1845. name: item.product.name,
  1846. label: item.product.name,
  1847. value: item.product.id,
  1848. desc: item.product_description || '',
  1849. quantity: Number(item.quantity),
  1850. }),
  1851. )
  1852. })
  1853. })
  1854. // 根据url传递过来的用户ID获取的用户身份信息
  1855. const userInfo = ref({} as IUser)
  1856. const p3 = axios
  1857. .post('/api/common/getUsersData', { id: route.query.user })
  1858. .then((response) => {
  1859. const res = response.data
  1860. if (res.code !== 1) return
  1861. if (res.result.users && res.result.users.length) {
  1862. userInfo.value = res.result.users[0]
  1863. } else if (res.result.id) {
  1864. userInfo.value = res.result || {}
  1865. } else if (Array.isArray(res.result) && res.result.length) {
  1866. userInfo.value = res.result[0] || {}
  1867. } else {
  1868. ElMessage.error('获取当前用户身份异常, 请联系管理员')
  1869. }
  1870. })
  1871. Promise.all([p1, p2, p3]).finally(() => {
  1872. loading.value = false
  1873. })
  1874. </script>
  1875. <style lang="scss">
  1876. input[type='number'] {
  1877. -moz-appearance: textfield;
  1878. appearance: textfield;
  1879. &:hover {
  1880. -moz-appearance: textfield;
  1881. appearance: textfield;
  1882. &::-webkit-inner-spin-button,
  1883. &::-webkit-outer-spin-button {
  1884. -webkit-appearance: none;
  1885. margin: 0;
  1886. }
  1887. }
  1888. &::-webkit-inner-spin-button,
  1889. &::-webkit-outer-spin-button {
  1890. -webkit-appearance: none;
  1891. margin: 0;
  1892. }
  1893. }
  1894. </style>
  1895. <style lang="scss" scoped>
  1896. .view-window {
  1897. height: 100vh;
  1898. width: 100vw;
  1899. position: fixed;
  1900. z-index: 999;
  1901. background-color: rgba(#fff, 0.3);
  1902. left: 0;
  1903. top: 0;
  1904. }
  1905. .fixed-button-area {
  1906. position: fixed;
  1907. top: 20px;
  1908. right: 24px;
  1909. z-index: 2;
  1910. @media screen and (min-width: 1600px) {
  1911. right: calc((100% - 1600px) / 2 + 24px);
  1912. }
  1913. }
  1914. .screen {
  1915. margin: 0 auto;
  1916. font-size: 12pt;
  1917. line-height: 22pt;
  1918. }
  1919. .print {
  1920. padding: 20px 0;
  1921. }
  1922. .page-title-wrap {
  1923. padding: 12px 36px;
  1924. border-bottom: 1px solid #eee;
  1925. margin-bottom: 24px;
  1926. }
  1927. .page-title {
  1928. font-size: 36px;
  1929. margin-left: 8px;
  1930. }
  1931. .layout-left {
  1932. width: 435px;
  1933. min-width: 435px;
  1934. padding-right: 24px;
  1935. }
  1936. .layout-right {
  1937. position: relative;
  1938. border-left: 1px solid #eee;
  1939. padding: 0 24px;
  1940. }
  1941. .rule-item {
  1942. .el-form-item {
  1943. margin-bottom: 0;
  1944. }
  1945. }
  1946. .sign-wrap {
  1947. width: 100pt;
  1948. &.pangea {
  1949. width: 180pt;
  1950. }
  1951. img {
  1952. width: 100%;
  1953. }
  1954. }
  1955. .screen {
  1956. background-color: rgba(#fff, 1);
  1957. box-shadow: 0 0 0 1px rgba(255, 255, 255, 0.4) inset,
  1958. 0 0.5em 1em rgba(0, 0, 0, 0.6);
  1959. max-width: 1600px;
  1960. min-width: 1600px;
  1961. margin: 0 auto;
  1962. .company-info {
  1963. text-align: center;
  1964. font-size: 10pt;
  1965. color: $subColor;
  1966. margin-bottom: 16pt;
  1967. }
  1968. .company-name {
  1969. margin-bottom: 4pt;
  1970. color: $mainColor;
  1971. font-size: 16pt;
  1972. line-height: 22pt;
  1973. }
  1974. .company-addr {
  1975. line-height: 12pt;
  1976. height: 12pt;
  1977. }
  1978. .company-phone {
  1979. line-height: 12pt;
  1980. height: 12pt;
  1981. }
  1982. .company-fax {
  1983. line-height: 12pt;
  1984. height: 12pt;
  1985. }
  1986. .form-area {
  1987. position: relative;
  1988. }
  1989. .form-area-left {
  1990. width: 30%;
  1991. }
  1992. .form-area-right {
  1993. width: 45%;
  1994. position: relative;
  1995. text-align: right;
  1996. vertical-align: top;
  1997. .label {
  1998. display: inline-block;
  1999. width: 40%;
  2000. vertical-align: top;
  2001. }
  2002. .value {
  2003. width: 60%;
  2004. word-break: break-word;
  2005. display: inline-block;
  2006. }
  2007. }
  2008. .product-table-separator {
  2009. width: 95%;
  2010. }
  2011. .product-total-table {
  2012. display: inline-block;
  2013. border: 1px solid #eee;
  2014. min-width: 220pt;
  2015. border-radius: 8pt;
  2016. position: relative;
  2017. right: 0;
  2018. top: 0;
  2019. .total-item {
  2020. & > div {
  2021. display: inline-block;
  2022. height: 26pt;
  2023. line-height: 26pt;
  2024. padding: 0 8pt;
  2025. }
  2026. &:nth-of-type(n + 1) {
  2027. border-top: 1px solid #eee;
  2028. }
  2029. }
  2030. .label {
  2031. width: 50%;
  2032. text-align: right;
  2033. }
  2034. .value {
  2035. width: 50%;
  2036. text-align: left;
  2037. }
  2038. }
  2039. .product-table {
  2040. table {
  2041. border: 1pt solid #eee;
  2042. width: 100%;
  2043. }
  2044. tr:nth-of-type(n + 3) td {
  2045. border-top: 1pt solid #eee;
  2046. }
  2047. th {
  2048. text-align: center;
  2049. background-color: $tableHeaderBgColor;
  2050. }
  2051. td {
  2052. padding: 8pt 8pt 4pt;
  2053. min-width: 30pt;
  2054. max-width: 100pt;
  2055. }
  2056. th + th,
  2057. td + td {
  2058. border-left: 1pt solid #eee;
  2059. }
  2060. .action {
  2061. text-align: center;
  2062. width: 50pt;
  2063. }
  2064. .quantity,
  2065. .rate,
  2066. .amount,
  2067. .discount {
  2068. width: 80pt;
  2069. }
  2070. .requirement {
  2071. width: 180pt;
  2072. }
  2073. .product {
  2074. width: 180pt;
  2075. }
  2076. }
  2077. .note-form-area {
  2078. width: 40%;
  2079. }
  2080. .rule-item {
  2081. color: $subColor;
  2082. line-height: 26pt;
  2083. }
  2084. .sub-form-title {
  2085. padding-left: 4pt;
  2086. border-left: 2pt solid #efefef;
  2087. margin-bottom: 8pt;
  2088. }
  2089. }
  2090. .preview-area {
  2091. background-color: #fff;
  2092. font-size: 10pt;
  2093. line-height: 22pt;
  2094. color: $mainColor;
  2095. box-sizing: border-box;
  2096. * {
  2097. padding: 0;
  2098. margin: 0;
  2099. }
  2100. .company-info {
  2101. text-align: center;
  2102. font-size: 9pt;
  2103. color: $subColor;
  2104. margin-bottom: 16pt;
  2105. }
  2106. .company-name {
  2107. margin-bottom: 4pt;
  2108. color: $mainColor;
  2109. font-size: 16pt;
  2110. line-height: 22pt;
  2111. }
  2112. .company-addr {
  2113. line-height: 12pt;
  2114. height: 12pt;
  2115. }
  2116. .company-phone {
  2117. line-height: 12pt;
  2118. height: 12pt;
  2119. }
  2120. .company-fax {
  2121. line-height: 12pt;
  2122. height: 12pt;
  2123. }
  2124. $titleWidth: 100%;
  2125. .pdf-title {
  2126. text-align: center;
  2127. font-size: 28pt;
  2128. line-height: 36pt;
  2129. height: 36pt;
  2130. width: $titleWidth;
  2131. margin: 0 auto;
  2132. background-color: #fff;
  2133. }
  2134. .pdf-title-bg {
  2135. width: 100%;
  2136. height: 2pt;
  2137. .center {
  2138. width: 33%;
  2139. text-align: center;
  2140. }
  2141. .right,
  2142. .left {
  2143. width: 33%;
  2144. height: 3pt;
  2145. line-height: 3pt;
  2146. & > div {
  2147. width: 100%;
  2148. height: 2pt;
  2149. line-height: 2pt;
  2150. background-color: $bgColor;
  2151. }
  2152. }
  2153. }
  2154. .form-area {
  2155. padding-top: 24pt;
  2156. line-height: 14pt;
  2157. $formAreaFontSize: 10pt;
  2158. white-space: pre-wrap;
  2159. font-size: $formAreaFontSize;
  2160. & > .flex-auto {
  2161. width: 50%;
  2162. & > div {
  2163. padding-bottom: 8pt;
  2164. }
  2165. }
  2166. .column-vendor {
  2167. vertical-align: top;
  2168. }
  2169. .column-form-label {
  2170. width: 40%;
  2171. }
  2172. .column-form-value {
  2173. width: 60%;
  2174. }
  2175. }
  2176. .product-table-separator {
  2177. width: 100%;
  2178. margin: 15pt auto 10pt;
  2179. height: 2pt;
  2180. background-color: $bgColor;
  2181. }
  2182. .product-table {
  2183. table {
  2184. margin: 0 auto 6pt;
  2185. width: 100%;
  2186. border-radius: 6pt;
  2187. text-align: left;
  2188. }
  2189. tr:nth-of-type(n + 1) td {
  2190. border-top: 1pt solid #eee;
  2191. }
  2192. th {
  2193. background-color: $bgColor;
  2194. min-width: 20pt;
  2195. padding: 0 2pt;
  2196. }
  2197. td {
  2198. min-width: 20pt;
  2199. max-width: 100pt;
  2200. padding: 0 2pt;
  2201. .desc {
  2202. white-space: pre-wrap;
  2203. line-height: 16pt;
  2204. color: $subColor;
  2205. }
  2206. }
  2207. .row-index {
  2208. width: 20pt;
  2209. }
  2210. }
  2211. .note-form-area {
  2212. width: 40%;
  2213. .label {
  2214. color: $subColor;
  2215. display: inline-block;
  2216. width: 40%;
  2217. vertical-align: top;
  2218. }
  2219. .value {
  2220. color: $subColor;
  2221. width: 60%;
  2222. word-break: break-word;
  2223. white-space: pre-wrap;
  2224. display: inline-block;
  2225. line-height: 16pt;
  2226. }
  2227. }
  2228. .sub-form-title {
  2229. color: $subColor;
  2230. font-weight: bold;
  2231. }
  2232. .rule-item {
  2233. color: $subColor;
  2234. line-height: 26pt;
  2235. }
  2236. .signature-area {
  2237. color: $subColor;
  2238. .first-party {
  2239. padding: 30pt 0 0;
  2240. }
  2241. .second-party {
  2242. padding: 20pt 0 40pt;
  2243. }
  2244. }
  2245. }
  2246. .preview-area2 {
  2247. background-color: #fff;
  2248. .supplier-name {
  2249. font-size: 22pt;
  2250. text-align: center;
  2251. }
  2252. .billing-addr {
  2253. line-height: 18pt;
  2254. min-height: 18pt;
  2255. font-size: 12pt;
  2256. text-align: center;
  2257. }
  2258. .contact-info {
  2259. margin-bottom: 12pt;
  2260. min-height: 18pt;
  2261. line-height: 18pt;
  2262. font-size: 12pt;
  2263. }
  2264. .product-table {
  2265. table {
  2266. border: 1px solid $bgColor;
  2267. border-radius: 3pt;
  2268. margin: 0 auto 6pt;
  2269. width: 100%;
  2270. font-size: 10pt;
  2271. }
  2272. th {
  2273. text-align: left;
  2274. }
  2275. td {
  2276. white-space: pre-wrap;
  2277. }
  2278. th,
  2279. td {
  2280. padding: 4pt 6pt;
  2281. &:nth-of-type(n + 2) {
  2282. border-left: 1px solid $bgColor;
  2283. }
  2284. }
  2285. tr:nth-of-type(n + 2) {
  2286. td {
  2287. border-top: 1px solid $bgColor;
  2288. }
  2289. }
  2290. }
  2291. .table-title {
  2292. text-align: center;
  2293. font-weight: bold;
  2294. font-size: 16pt;
  2295. margin: 12pt auto 8pt;
  2296. }
  2297. .base-info-area {
  2298. margin-top: 12pt;
  2299. font-size: 11pt;
  2300. .base-info-label {
  2301. min-width: 90pt;
  2302. width: 90pt;
  2303. font-family: serif;
  2304. }
  2305. .base-info-value {
  2306. white-space: pre-wrap;
  2307. vertical-align: top;
  2308. flex: auto;
  2309. text-align: left;
  2310. font-family: sans-serif;
  2311. font-weight: 400;
  2312. }
  2313. .left {
  2314. padding-right: 8pt;
  2315. .base-info-label {
  2316. min-width: 60pt;
  2317. width: 60pt;
  2318. }
  2319. }
  2320. .right {
  2321. padding-left: 8pt;
  2322. }
  2323. .left,
  2324. .right {
  2325. width: 50%;
  2326. margin-bottom: 12pt;
  2327. }
  2328. }
  2329. .bank-info-area {
  2330. font-family: sans-serif;
  2331. font-weight: bold;
  2332. font-size: 11pt;
  2333. margin-bottom: 10pt;
  2334. .bank-info-label {
  2335. font-weight: bold;
  2336. font-family: serif;
  2337. width: 160pt;
  2338. max-width: 160pt;
  2339. }
  2340. .bank-info-value {
  2341. font-family: serif;
  2342. font-weight: normal;
  2343. }
  2344. }
  2345. }
  2346. @media print {
  2347. .screen {
  2348. display: none;
  2349. }
  2350. .preview-area,
  2351. .preview-area2 {
  2352. border: none;
  2353. border-color: transparent;
  2354. }
  2355. }
  2356. @media screen {
  2357. .print.hidden {
  2358. position: fixed;
  2359. right: -10000px;
  2360. bottom: -10000px;
  2361. }
  2362. .pdf-wrap {
  2363. position: relative;
  2364. width: 21cm;
  2365. margin: 0 auto;
  2366. box-shadow: 1px 1px 2pt 0px $subColor;
  2367. & + .pdf-wrap {
  2368. margin-top: 20px;
  2369. }
  2370. }
  2371. .preview-area,
  2372. .preview-area2 {
  2373. padding: 1cm;
  2374. min-height: 29.69cm;
  2375. }
  2376. }
  2377. </style>