poster.vue 9.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263
  1. <template>
  2. <view class="paddingY30 poster" style="width: 0px; height: 0px; overflow: hidden">
  3. <canvas
  4. type="2d"
  5. width="315"
  6. height="485"
  7. style="width: 315px: height: 485px margin:0 auto"
  8. id="myCanvas"
  9. ></canvas>
  10. </view>
  11. </template>
  12. <script>
  13. import resource from '@/utils/resource'
  14. import ossurl from '@/utils/ossurl'
  15. import { WX_APP_ID } from '@/utils/config'
  16. import canvasUtil from '../../utils/canvas'
  17. export default {
  18. props: {
  19. imgs: Array,
  20. page: String,
  21. scene: String,
  22. tip: String,
  23. userId: String
  24. },
  25. data() {
  26. return {
  27. filePath: ''
  28. }
  29. },
  30. methods: {
  31. share() {
  32. if (this.filePath) {
  33. this.beginDraw(this.filePath)
  34. return
  35. }
  36. this.getQrcode((buffer) => {
  37. const qrCodeFile = wx.env.USER_DATA_PATH + '/temp_image.jpeg'
  38. const fsm = uni.getFileSystemManager()
  39. fsm.writeFile({
  40. filePath: qrCodeFile,
  41. data: buffer,
  42. encoding: 'binary',
  43. success: (res) => {
  44. this.filePath = qrCodeFile
  45. this.beginDraw(this.filePath)
  46. },
  47. complete: () => {
  48. uni.hideLoading()
  49. }
  50. })
  51. })
  52. },
  53. getQrcode(callback) {
  54. uni.showLoading({
  55. title: '加载中...'
  56. })
  57. var scene = this.scene + ':' + this.userId
  58. let url = `${this.$service.base.apis.MiNI_CODE}?clientId=${WX_APP_ID}&wxaCodeUnlimitBytesRequest.scene=${scene}&envVersion=trial`
  59. url += `&wxaCodeUnlimitBytesRequest.page=${encodeURIComponent(this.page)}`
  60. wx.request({
  61. url,
  62. method: 'get',
  63. responseType: 'arraybuffer',
  64. header: {
  65. track: JSON.stringify({
  66. os: 4,
  67. bundleId: 'com.supermart.mart',
  68. clientId: WX_APP_ID
  69. })
  70. },
  71. success: (res) => {
  72. if (res.statusCode == 200 && res.data && res.data.byteLength > 0) {
  73. callback(res.data)
  74. } else {
  75. uni.hideLoading()
  76. }
  77. },
  78. fail: (err) => {
  79. uni.hideLoading()
  80. }
  81. })
  82. },
  83. beginDraw(qrCodeFile) {
  84. const query = wx.createSelectorQuery().in(this)
  85. query
  86. .select('#myCanvas')
  87. .fields({ node: true, size: true })
  88. .exec((res) => {
  89. const canvas = res[0].node
  90. canvas.width = 945
  91. canvas.height = 1455
  92. this.draw(canvas, canvas.width, canvas.height, qrCodeFile)
  93. })
  94. },
  95. draw(canvas, width, height, qrCodeFile) {
  96. const ctx = canvas.getContext('2d')
  97. ctx.clearRect(0, 0, canvas.width, canvas.height)
  98. canvasUtil.circleImg(ctx, null, 0, 0, width, height, 24)
  99. let bgImg = canvas.createImage()
  100. bgImg.src = ossurl.common.posterBg
  101. bgImg.onload = (e) => {
  102. canvasUtil.circleImg(ctx, bgImg, 0, 0, width, height, 24)
  103. // canvasUtil.drawOneText(
  104. // ctx,
  105. // '长摁识别小程序码 去燚火漫域领手办吧',
  106. // 'normal bold 33px Arial',
  107. // '#fff',
  108. // width,
  109. // 1400
  110. // )
  111. let qrCode = canvas.createImage()
  112. qrCode.src = qrCodeFile
  113. qrCode.onload = (e) => {
  114. canvasUtil.circleImg(ctx, qrCode, (width + 260) / 2, 1155, 185, 185, 76)
  115. }
  116. // canvasUtil.drawOneText(ctx, this.tip, 'normal bold 48px Arial', '#fff', width, 1020)
  117. // this.drawImgs(canvas, ctx)
  118. uni.showLoading({
  119. title: '绘制中...'
  120. })
  121. let time = 1300 + this.imgs.length * 100
  122. time = time > 2000 ? 2000 : time
  123. setTimeout(() => {
  124. wx.canvasToTempFilePath({
  125. canvas,
  126. success: (res) => {
  127. // 生成的图片临时文件路径
  128. const tempFilePath = res.tempFilePath
  129. wx.showShareImageMenu({
  130. path: tempFilePath,
  131. success: (res) => {
  132. this.$emit('success')
  133. },
  134. fail: (err) => {
  135. console.log(err)
  136. }
  137. })
  138. },
  139. complete: () => {
  140. uni.hideLoading()
  141. }
  142. })
  143. }, time)
  144. }
  145. },
  146. drawImgs(canvas, ctx) {
  147. let w = 255
  148. let h = 300
  149. let imageW = 0
  150. let imageH = 0
  151. if (this.imgs.length === 1) {
  152. w = 382
  153. h = 450
  154. imageW = w - 44
  155. imageH = h - 112
  156. this.drawImgItem(canvas, ctx, this.imgs[0].cover, w, h, 282, 386, imageW, imageH)
  157. } else if (this.imgs.length === 2) {
  158. w = 331
  159. h = 390
  160. imageW = w - 40
  161. imageH = h - 96
  162. this.drawImgItem(canvas, ctx, this.imgs[0].cover, w, h, 111, 416, imageW, imageH)
  163. this.drawImgItem(canvas, ctx, this.imgs[1].cover, w, h, 503, 416, imageW, imageH)
  164. } else {
  165. imageW = w - 30
  166. imageH = h - 74
  167. this.drawNormals(canvas, ctx, w, h, imageW, imageH)
  168. }
  169. },
  170. drawNormals(canvas, ctx, w, h, imageW, imageH) {
  171. let datas = []
  172. switch (this.imgs.length) {
  173. case 3:
  174. datas = [this.imgs]
  175. break
  176. case 4:
  177. datas = [this.imgs.slice(0, 2), this.imgs.slice(2)]
  178. break
  179. case 5:
  180. datas = [this.imgs.slice(0, 3), this.imgs.slice(3)]
  181. break
  182. case 6:
  183. datas = [this.imgs.slice(0, 3), this.imgs.slice(3)]
  184. break
  185. }
  186. for (let row = 0; row < datas.length; row++) {
  187. let top = 0
  188. if (datas.length === 1) {
  189. top = 461
  190. } else {
  191. top = 296 + row * (h + 30)
  192. }
  193. this.drawRow(canvas, ctx, datas[row], w, h, imageW, imageH, top)
  194. }
  195. },
  196. drawRow(canvas, ctx, data, w, h, imageW, imageH, top) {
  197. let beginLeft = data.length === 2 ? 203 : 60
  198. for (let i = 0; i < data.length; i++) {
  199. let left = beginLeft + i * (w + 30)
  200. this.drawImgItem(canvas, ctx, data[i].cover, w, h, left, top, imageW, imageH)
  201. }
  202. },
  203. drawImgItem(canvas, ctx, src, w, h, left, top, imageW, imageH) {
  204. let wrapper = canvas.createImage()
  205. wrapper.src = resource.poster_imgW
  206. wrapper.onload = (e) => {
  207. canvasUtil.circleImg(ctx, wrapper, left, top, w, h, 10)
  208. wx.getImageInfo({
  209. src: src,
  210. success: (res) => {
  211. let wRatio = res.width / imageW
  212. let ratioHeight = res.height / wRatio
  213. //如果按比例后高度 大于 最大高度,那么安高的比例来
  214. if (ratioHeight > imageH) {
  215. imageH = imageH
  216. imageW = res.width / (res.height / imageH)
  217. } else {
  218. imageW = imageW
  219. imageH = res.height / (res.width / imageW)
  220. }
  221. let img = canvas.createImage()
  222. img.src = res.path
  223. img.onload = (e) => {
  224. canvasUtil.circleImg(
  225. ctx,
  226. img,
  227. left + (w - imageW) / 2,
  228. top + (h - imageH) / 2,
  229. imageW,
  230. imageH,
  231. 0
  232. )
  233. }
  234. }
  235. })
  236. }
  237. }
  238. }
  239. }
  240. </script>
  241. <style lang="scss">
  242. .poster {
  243. canvas {
  244. height: 380px !important;
  245. }
  246. }
  247. </style>