edit.vue 71 KB

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