PriceToImg.vue 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223
  1. <template>
  2. <div ref="printArea" style="font-family: ProximaNova-Regular;background-color: #fff;padding:20px;word-break: break-word;box-sizing: border-box;">
  3. <table>
  4. <tbody>
  5. <tr>
  6. <td style="margin-right: 14px;">
  7. <img :src="comInfo.image" width="320" height="320" v-if="isImgSrc">
  8. <img ref="productImg" width="250" height="250" v-else>
  9. </td>
  10. <td style="vertical-align: top;">
  11. <p style="font-weight: bold;font-size: 22px;padding-bottom: 24px;color:#00213B">
  12. <span style="margin-right: 5px;">{{pageData.product_code}}</span>
  13. <span>{{pageData.product_name}}</span>
  14. </p>
  15. <p style="line-height: 20px;color:#00213B;font-size: 14px;" v-html="comInfo.description"></p>
  16. </td>
  17. </tr>
  18. </tbody>
  19. </table>
  20. <div>
  21. <p style="font-weight: bold;font-size: 17px;margin: 12px 0 8px;color:#00213B">Specifications</p>
  22. <table style="width:100%;border-collapse: collapse;font-size: 14px;table-layout: fixed;text-align:center;">
  23. <tbody>
  24. <tr style="background-color: #F7F8FC;font-weight: bold;">
  25. <td v-for="(item,key) in headerColumns" :key="key" style="padding:10px 0;color:#00213B;width: 25%;font-size: 14px;border:1px solid #EFEFEF">{{item.label}}</td>
  26. </tr>
  27. <tr>
  28. <td v-for="(item,key) in headerColumns" :key="key" style="padding:10px 0 6px;color:#00213B;border:1px solid #EFEFEF">
  29. <div v-if="specificationsObj[item.prop]">
  30. <span v-if="item.prop != 'addonArr' ">{{specificationsObj[item.prop]}}</span>
  31. <div v-else v-for="(opt,idx) in specificationsObj.addonArr" :key="idx" style="padding-bottom:4px;" >{{opt}}</div>
  32. </div>
  33. <span v-else>-</span>
  34. </td>
  35. </tr>
  36. </tbody>
  37. </table>
  38. </div>
  39. <div>
  40. <p style="font-weight: bold;font-size: 17px;margin: 12px 0 8px;color:#00213B">Quote</p>
  41. <table style="width:100%;border-collapse: collapse;font-size: 14px;table-layout: fixed;text-align:center;">
  42. <tbody>
  43. <tr style="background-color: #F7F8FC;font-weight: bold;" v-if="comLocationNum !== 1">
  44. <td style="padding:10px 0;color:#00213B;border:1px solid #EFEFEF" :colspan="tableColumns.length-1">Quote</td>
  45. <td style="padding:10px 0;color:#00213B;border:1px solid #EFEFEF" :style="{width:comWidth}">Total</td>
  46. </tr>
  47. <tr style="background-color: #F7F8FC;font-weight: bold;">
  48. <td v-for="(item,key) in tableColumns" :key="key" style="padding:10px 0;color:#00213B;border:1px solid #EFEFEF" :style="{width:comWidth}">{{item.label}}</td>
  49. </tr>
  50. <tr style="padding:10px 0;" v-for="row in tableData" :key="row.project">
  51. <td v-for="column in tableColumns" :key="column.prop" style="padding:10px 0;color:#00213B;border:1px solid #EFEFEF;white-space: nowrap;" :style="getColumnHeaderStyle(column)">
  52. {{ row[column.prop].includes('.')?'$':'' }} {{ row[column.prop]?row[column.prop]:'0.00' }}
  53. </td>
  54. </tr>
  55. </tbody>
  56. </table>
  57. </div>
  58. <div v-if="emailForm.Comments" style="color:#00213B">
  59. <p style="margin: 20px 0;">{{emailForm.Comments}}</p>
  60. </div>
  61. </div>
  62. </template>
  63. <script>
  64. import html2Canvas from "html2canvas";
  65. export default {
  66. props: {
  67. pageData: {
  68. type: Object,
  69. default: {},
  70. },
  71. specificationsObj: {
  72. type: Object,
  73. default: {},
  74. },
  75. tableData: {
  76. type: Array,
  77. default: [],
  78. },
  79. tableColumns: {
  80. type: Array,
  81. default: [],
  82. },
  83. comLocationNum: {
  84. type: Number,
  85. default: 1,
  86. },
  87. emailForm:{
  88. type: Object,
  89. default: {},
  90. },
  91. loading: {
  92. type: Boolean,
  93. default: false
  94. },
  95. isImgSrc: {
  96. type: Boolean,
  97. default: true
  98. },
  99. width: {
  100. type: String,
  101. default: '100%'
  102. }
  103. },
  104. data(){
  105. return {
  106. headerColumns: [
  107. {
  108. label: "LeadTime",
  109. prop: "time",
  110. type: "pre" //去掉$符号
  111. },
  112. {
  113. label: "Model",
  114. prop: "model",
  115. type: "pre" //去掉$符号
  116. },
  117. {
  118. label: "Decoration",
  119. prop: "decoration",
  120. type: "pre" //去掉$符号
  121. },
  122. {
  123. label: "Add On",
  124. prop: "addonArr",
  125. type: "pre" //去掉$符号
  126. },
  127. ],
  128. }
  129. },
  130. computed:{
  131. comInfo(){
  132. return this.pageData?.main
  133. },
  134. comWidth(){
  135. return 100/this.tableColumns.length + '%'
  136. }
  137. },
  138. created(){
  139. if(!this.isImgSrc){
  140. this.getBase64Image(this.comInfo.image,'productImg')
  141. }
  142. },
  143. methods: {
  144. getColumnHeaderStyle(column) {
  145. if (column.prop === 'project') {
  146. return { 'font-weight': 'bold' };
  147. }
  148. return {};
  149. },
  150. //Base64生成产品图片,因为跨域
  151. getBase64Image (url, ref) {
  152. var that = this
  153. var image = new Image()
  154. image.src = url + '?v=' + Math.random() // 处理缓存
  155. image.crossOrigin = '*' // 支持跨域图片
  156. image.onload = function () {
  157. var base64 = that.drawBase64Image(image)
  158. that.$refs[ref].src = base64
  159. }
  160. },
  161. drawBase64Image (img) {
  162. var canvas = document.createElement('canvas')
  163. canvas.width = img.width
  164. canvas.height = img.height
  165. var ctx = canvas.getContext('2d')
  166. ctx.drawImage(img, 0, 0, img.width, img.height)
  167. var dataURL = canvas.toDataURL('image/png')
  168. return dataURL
  169. },
  170. htmlToPdf(title){
  171. const element = this.$refs.printArea;
  172. const clientWidth = element.offsetWidth;
  173. const clientHeight = element.offsetHeight;
  174. const kh = [clientWidth, clientHeight];
  175. const opts = {
  176. scale: 1.5, // 缩放比例,提高生成图片清晰度
  177. useCORS: true, // 允许加载跨域的图片
  178. // allowTaint: true, // 允许图片跨域,和 useCORS 二者不可共同使用
  179. tainttest: true, // 检测每张图片已经加载完成
  180. logging: true, // 日志开关,发布的时候记得改成 false
  181. };
  182. html2Canvas(element, opts).then((canvas) => {
  183. const dataURL = canvas.toDataURL('image/png');
  184. this.downloadImage(dataURL, kh, title);
  185. this.$emit('update:loading', false)
  186. })
  187. .catch(() => {
  188. this.$emit('update:loading', false)
  189. // Bus.$emit("finishDownload");
  190. });
  191. },
  192. getUrlBase64(url, kh) {
  193. return new Promise((resolve) => {
  194. let canvas = document.createElement('canvas');
  195. const ctx = canvas.getContext('2d');
  196. const img = new Image();
  197. img.crossOrigin = 'Anonymous'; // 允许跨域
  198. img.src = url;
  199. img.onload = () => {
  200. // eslint-disable-next-line prefer-destructuring
  201. canvas.height = kh[1];
  202. // eslint-disable-next-line prefer-destructuring
  203. canvas.width = kh[0];
  204. ctx.drawImage(img, 0, 0, kh[0], kh[1]);
  205. const dataURL = canvas.toDataURL('image/png');
  206. canvas = null;
  207. resolve(dataURL);
  208. };
  209. });
  210. },
  211. downloadImage (imgUrl, kh, title) {
  212. this.getUrlBase64(imgUrl, kh).then((base64) => {
  213. const link = document.createElement('a');
  214. link.href = base64;
  215. link.download = `${title}.png`;
  216. link.click();
  217. });
  218. },
  219. },
  220. };
  221. </script>