wxml2canvas.js 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545
  1. /* eslint-disable */
  2. const PROPERTIES = ['hover-class', 'hover-start-time', 'space', 'src']
  3. const COMPUTED_STYLE = [
  4. 'color',
  5. 'font-size',
  6. 'font-weight',
  7. 'font-family',
  8. 'backgroundColor',
  9. 'border',
  10. 'border-radius',
  11. 'box-sizing',
  12. 'line-height',
  13. ]
  14. const DEFAULT_BORDER = '0px none rgb(0, 0, 0)'
  15. const DEFAULT_BORDER_RADIUS = '0px'
  16. // default z-index??
  17. const DEFAULT_RANK = {
  18. view: 0,
  19. image: 1,
  20. text: 2,
  21. }
  22. const drawWrapper = (context, data) => {
  23. const { backgroundColor, width, height } = data
  24. context.setFillStyle(backgroundColor)
  25. context.fillRect(0, 0, width, height)
  26. }
  27. // todo: do more for different language
  28. const strLen = str => {
  29. let count = 0
  30. for (let i = 0, len = str.length; i < len; i++) {
  31. count += str.charCodeAt(i) < 256 ? 1 : 2
  32. }
  33. return count / 2
  34. }
  35. const isMuitlpleLine = (data, text) => {
  36. const { 'font-size': letterWidth, width } = data
  37. const length = strLen(text)
  38. const rowlineLength = length * parseInt(letterWidth, 10)
  39. return rowlineLength > width
  40. }
  41. const drawMutipleLine = (context, data, text) => {
  42. const {
  43. 'font-size': letterWidth,
  44. width,
  45. left,
  46. top,
  47. 'line-height': lineHeightAttr,
  48. } = data
  49. const lineHieght = lineHeightAttr === 'normal' ? Math.round(1.2 * letterWidth) : lineHeightAttr
  50. const rowLetterCount = Math.floor(width / parseInt(letterWidth, 10))
  51. const length = strLen(text)
  52. for (let i = 0; i < length; i += rowLetterCount) {
  53. const lineText = text.substring(i, i + rowLetterCount)
  54. const rowNumber = Math.floor(i / rowLetterCount)
  55. const rowTop = top + rowNumber * parseInt(lineHieght, 10)
  56. context.fillText(lineText, left, rowTop)
  57. }
  58. }
  59. // enable color, font, for now only support chinese
  60. const drawText = (context, data) => {
  61. const {
  62. dataset: { text },
  63. left,
  64. top,
  65. color,
  66. 'font-weight': fontWeight,
  67. 'font-size': fontSize,
  68. 'font-family': fontFamily,
  69. } = data
  70. const canvasText = Array.isArray(text) ? text[0] : text
  71. context.font = `${fontWeight} ${Math.round(
  72. parseFloat(fontSize),
  73. )}px ${fontFamily}`
  74. context.setFillStyle(color)
  75. if (isMuitlpleLine(data, canvasText)) {
  76. drawMutipleLine(context, data, canvasText)
  77. } else {
  78. context.fillText(canvasText, left, top)
  79. }
  80. context.restore()
  81. }
  82. const getImgInfo = src =>
  83. new Promise((resolve, reject) => {
  84. wx.getImageInfo({
  85. src,
  86. success(res) {
  87. resolve(res)
  88. },
  89. })
  90. })
  91. const hasBorder = border => border !== DEFAULT_BORDER
  92. const hasBorderRadius = borderRadius => borderRadius !== DEFAULT_BORDER_RADIUS
  93. const getBorderAttributes = border => {
  94. let borderColor, borderStyle
  95. let borderWidth = 0
  96. if (hasBorder) {
  97. borderWidth = parseInt(border.split(/\s/)[0], 10)
  98. borderStyle = border.split(/\s/)[1]
  99. borderColor = border.match(/(rgb).*/gi)[0]
  100. }
  101. return {
  102. borderWidth,
  103. borderStyle,
  104. borderColor,
  105. }
  106. }
  107. const getImgRect = (imgData, borderWidth) => {
  108. const { width, height, left, top } = imgData
  109. const imgWidth = width - 2 * borderWidth
  110. const imgHeight = height - 2 * borderWidth
  111. const imgLeft = left + borderWidth
  112. const imgTop = top + borderWidth
  113. return {
  114. imgWidth,
  115. imgHeight,
  116. imgLeft,
  117. imgTop,
  118. }
  119. }
  120. const getArcCenterPosition = imgData => {
  121. const { width, height, left, top } = imgData
  122. const coordX = width / 2 + left
  123. const coordY = height / 2 + top
  124. return {
  125. coordX,
  126. coordY,
  127. }
  128. }
  129. const getArcRadius = (imgData, borderWidth = 0) => {
  130. const { width } = imgData
  131. return width / 2 - borderWidth / 2
  132. }
  133. const getCalculatedImagePosition = (imgData, naturalWidth, naturalHeight) => {
  134. const { border } = imgData
  135. const { borderWidth } = getBorderAttributes(border)
  136. const { imgWidth, imgHeight, imgLeft, imgTop } = getImgRect(
  137. imgData,
  138. borderWidth,
  139. )
  140. const ratio = naturalWidth / naturalHeight
  141. // tweak for real width and position => center center
  142. const realWidth = ratio > 0 ? imgWidth : imgHeight * ratio
  143. const realHeight = ratio > 0 ? imgWidth * (1 / ratio) : imgHeight
  144. const offsetLeft = ratio > 0 ? 0 : (imgWidth - realWidth) / 2
  145. const offsetTop = ratio > 0 ? (imgHeight - realHeight) / 2 : 0
  146. return {
  147. realWidth,
  148. realHeight,
  149. left: imgLeft + offsetLeft,
  150. top: imgTop + offsetTop,
  151. }
  152. }
  153. const drawArcImage = (context, imgData) => {
  154. const { src } = imgData
  155. const { coordX, coordY } = getArcCenterPosition(imgData)
  156. return getImgInfo(src).then(res => {
  157. const { width: naturalWidth, height: naturalHeight } = res
  158. const arcRadius = getArcRadius(imgData)
  159. context.save()
  160. context.beginPath()
  161. context.arc(coordX, coordY, arcRadius, 0, 2 * Math.PI)
  162. context.closePath()
  163. context.clip()
  164. const { left, top, realWidth, realHeight } = getCalculatedImagePosition(
  165. imgData,
  166. naturalWidth,
  167. naturalHeight,
  168. )
  169. context.drawImage(
  170. src,
  171. 0,
  172. 0,
  173. naturalWidth,
  174. naturalHeight,
  175. left,
  176. top,
  177. realWidth,
  178. realHeight,
  179. )
  180. context.restore()
  181. })
  182. }
  183. const drawRectImage = (context, imgData) => {
  184. const { src, width, height, left, top } = imgData
  185. return getImgInfo(src).then(res => {
  186. const { width: naturalWidth, height: naturalHeight } = res
  187. context.save()
  188. context.beginPath()
  189. context.rect(left, top, width, height)
  190. context.closePath()
  191. context.clip()
  192. const {
  193. left: realLeft,
  194. top: realTop,
  195. realWidth,
  196. realHeight,
  197. } = getCalculatedImagePosition(imgData, naturalWidth, naturalHeight)
  198. context.drawImage(
  199. src,
  200. 0,
  201. 0,
  202. naturalWidth,
  203. naturalHeight,
  204. realLeft,
  205. realTop,
  206. realWidth,
  207. realHeight,
  208. )
  209. context.restore()
  210. })
  211. }
  212. const drawArcBorder = (context, imgData) => {
  213. const { border } = imgData
  214. const { coordX, coordY } = getArcCenterPosition(imgData)
  215. const { borderWidth, borderColor } = getBorderAttributes(border)
  216. const arcRadius = getArcRadius(imgData, borderWidth)
  217. context.save()
  218. context.beginPath()
  219. context.setLineWidth(borderWidth)
  220. context.setStrokeStyle(borderColor)
  221. context.arc(coordX, coordY, arcRadius, 0, 2 * Math.PI)
  222. context.stroke()
  223. context.restore()
  224. }
  225. const drawRectBorder = (context, imgData) => {
  226. const { border } = imgData
  227. const { left, top, width, height } = imgData
  228. const { borderWidth, borderColor } = getBorderAttributes(border)
  229. const correctedBorderWidth = borderWidth + 1 // draw may cause empty 0.5 space
  230. context.save()
  231. context.beginPath()
  232. context.setLineWidth(correctedBorderWidth)
  233. context.setStrokeStyle(borderColor)
  234. context.rect(
  235. left + borderWidth / 2,
  236. top + borderWidth / 2,
  237. width - borderWidth,
  238. height - borderWidth,
  239. )
  240. context.stroke()
  241. context.restore()
  242. }
  243. // image, enable border-radius: 50%, border, bgColor
  244. const drawImage = (context, imgData) => {
  245. const { border, 'border-radius': borderRadius } = imgData
  246. let drawImagePromise
  247. if (hasBorderRadius(borderRadius)) {
  248. drawImagePromise = drawArcImage(context, imgData)
  249. } else {
  250. drawImagePromise = drawRectImage(context, imgData)
  251. }
  252. return drawImagePromise.then(() => {
  253. if (hasBorder(border)) {
  254. if (hasBorderRadius(borderRadius)) {
  255. return drawArcBorder(context, imgData)
  256. } else {
  257. return drawRectBorder(context, imgData)
  258. }
  259. }
  260. return Promise.resolve()
  261. })
  262. }
  263. // e.g. 10%, 4px
  264. const getBorderRadius = imgData => {
  265. const { width, height, 'border-radius': borderRadiusAttr } = imgData
  266. const borderRadius = parseInt(borderRadiusAttr, 10)
  267. if (borderRadiusAttr.indexOf('%') !== -1) {
  268. const borderRadiusX = parseInt(borderRadius / 100 * width, 10)
  269. const borderRadiusY = parseInt(borderRadius / 100 * height, 10)
  270. return {
  271. isCircle: borderRadiusX === borderRadiusY,
  272. borderRadius: borderRadiusX,
  273. borderRadiusX,
  274. borderRadiusY,
  275. }
  276. } else {
  277. return {
  278. isCircle: true,
  279. borderRadius,
  280. }
  281. }
  282. }
  283. const drawViewArcBorder = (context, imgData) => {
  284. const { width, height, left, top, backgroundColor, border } = imgData
  285. const { borderRadius } = getBorderRadius(imgData)
  286. const { borderWidth, borderColor } = getBorderAttributes(border)
  287. // console.log('🐞-imgData', imgData)
  288. context.beginPath()
  289. context.moveTo(left + borderRadius, top)
  290. context.lineTo(left + width - borderRadius, top)
  291. context.arcTo(
  292. left + width,
  293. top,
  294. left + width,
  295. top + borderRadius,
  296. borderRadius,
  297. )
  298. context.lineTo(left + width, top + height - borderRadius)
  299. context.arcTo(
  300. left + width,
  301. top + height,
  302. left + width - borderRadius,
  303. top + height,
  304. borderRadius,
  305. )
  306. context.lineTo(left + borderRadius, top + height)
  307. context.arcTo(
  308. left,
  309. top + height,
  310. left,
  311. top + height - borderRadius,
  312. borderRadius,
  313. )
  314. context.lineTo(left, top + borderRadius)
  315. context.arcTo(left, top, left + borderRadius, top, borderRadius)
  316. context.closePath()
  317. if (backgroundColor) {
  318. context.setFillStyle(backgroundColor)
  319. context.fill()
  320. }
  321. if (borderColor && borderWidth) {
  322. context.setLineWidth(borderWidth)
  323. context.setStrokeStyle(borderColor)
  324. context.stroke()
  325. }
  326. }
  327. const drawViewBezierBorder = (context, imgData) => {
  328. const { width, height, left, top, backgroundColor, border } = imgData
  329. const { borderWidth, borderColor } = getBorderAttributes(border)
  330. const { borderRadiusX, borderRadiusY } = getBorderRadius(imgData)
  331. context.beginPath()
  332. context.moveTo(left + borderRadiusX, top)
  333. context.lineTo(left + width - borderRadiusX, top)
  334. context.quadraticCurveTo(left + width, top, left + width, top + borderRadiusY)
  335. context.lineTo(left + width, top + height - borderRadiusY)
  336. context.quadraticCurveTo(
  337. left + width,
  338. top + height,
  339. left + width - borderRadiusX,
  340. top + height,
  341. )
  342. context.lineTo(left + borderRadiusX, top + height)
  343. context.quadraticCurveTo(
  344. left,
  345. top + height,
  346. left,
  347. top + height - borderRadiusY,
  348. )
  349. context.lineTo(left, top + borderRadiusY)
  350. context.quadraticCurveTo(left, top, left + borderRadiusX, top)
  351. context.closePath()
  352. if (backgroundColor) {
  353. context.setFillStyle(backgroundColor)
  354. context.fill()
  355. }
  356. if (borderColor && borderWidth) {
  357. context.setLineWidth(borderWidth)
  358. context.setStrokeStyle(borderColor)
  359. context.stroke()
  360. }
  361. }
  362. // enable border, border-radius, bgColor, position
  363. const drawView = (context, imgData) => {
  364. const { isCircle } = getBorderRadius(imgData)
  365. if (isCircle) {
  366. drawViewArcBorder(context, imgData)
  367. } else {
  368. drawViewBezierBorder(context, imgData)
  369. }
  370. }
  371. const isTextElement = item => {
  372. const { dataset: { text }, type } = item
  373. return Boolean(text) || type === 'text'
  374. }
  375. const isImageElement = item => {
  376. const { src, type } = item
  377. return Boolean(src) || type === 'image'
  378. }
  379. const isViewElement = item => {
  380. const { type } = item
  381. return type === 'view'
  382. }
  383. const formatElementData = elements =>
  384. elements.map(element => {
  385. if (isTextElement(element)) {
  386. element.type = 'text'
  387. element.rank = DEFAULT_RANK.text
  388. } else if (isImageElement(element)) {
  389. element.type = 'image'
  390. element.rank = DEFAULT_RANK.image
  391. } else {
  392. element.type = 'view'
  393. element.rank = DEFAULT_RANK.view
  394. }
  395. return element
  396. })
  397. // todo: use z-index as order to draw??
  398. const getSortedElementsData = elements =>
  399. elements.sort((a, b) => {
  400. if (a.rank < b.rank) {
  401. return -1
  402. } else if (a.rank > b.rank) {
  403. return 1
  404. }
  405. return 0
  406. })
  407. const drawElements = (context, storeItems) => {
  408. const itemPromise = []
  409. storeItems.forEach(item => {
  410. if (isTextElement(item)) {
  411. const text = drawText(context, item)
  412. itemPromise.push(text)
  413. } else if (isImageElement(item)) {
  414. const image = drawImage(context, item)
  415. itemPromise.push(image)
  416. } else {
  417. const view = drawView(context, item)
  418. itemPromise.push(view)
  419. }
  420. })
  421. return itemPromise
  422. }
  423. // storeObject: { 0: [...], 1: [...] }
  424. // chain call promise based on Object key
  425. const drawElementBaseOnIndex = (context, storeObject, key = 0, drawPromise) => {
  426. if (typeof drawPromise === 'undefined') {
  427. drawPromise = Promise.resolve()
  428. }
  429. const objectKey = key // note: key is changing when execute promise then
  430. const chainPromise = drawPromise.then(() => {
  431. const nextPromise = storeObject[objectKey]
  432. ? Promise.all(drawElements(context, storeObject[objectKey]))
  433. : Promise.resolve()
  434. return nextPromise
  435. })
  436. if (key >= Object.keys(storeObject).length) {
  437. return chainPromise
  438. } else {
  439. return drawElementBaseOnIndex(context, storeObject, key + 1, chainPromise)
  440. }
  441. }
  442. const drawCanvas = (canvasId, wrapperData, innerData) => {
  443. const context = wx.createCanvasContext(canvasId)
  444. context.setTextBaseline('top')
  445. // todo: use this after weixin fix stupid clip can't work bug in fillRect
  446. // for now, just set canvas background as a compromise
  447. drawWrapper(context, wrapperData[0])
  448. const storeObject = {}
  449. const sortedElementData = getSortedElementsData(formatElementData(innerData)) // fake z-index
  450. sortedElementData.forEach(item => {
  451. if (!storeObject[item.rank]) {
  452. // initialize
  453. storeObject[item.rank] = []
  454. }
  455. if (isTextElement(item) || isImageElement(item) || isViewElement(item)) {
  456. storeObject[item.rank].push(item)
  457. }
  458. })
  459. // note: draw is async
  460. return drawElementBaseOnIndex(context, storeObject).then(
  461. () =>
  462. new Promise((resolve, reject) => {
  463. context.draw(true, () => {
  464. resolve()
  465. })
  466. }),
  467. )
  468. }
  469. const wxSelectorQuery = element =>
  470. new Promise((resolve, reject) => {
  471. try {
  472. wx
  473. .createSelectorQuery()
  474. .selectAll(element)
  475. .fields(
  476. {
  477. dataset: true,
  478. size: true,
  479. rect: true,
  480. properties: PROPERTIES,
  481. computedStyle: COMPUTED_STYLE,
  482. },
  483. res => {
  484. resolve(res)
  485. },
  486. )
  487. .exec()
  488. } catch (error) {
  489. reject(error)
  490. }
  491. })
  492. const wxml2canvas = (wrapperId, elementsClass, canvasId) => {
  493. const getWrapperElement = wxSelectorQuery(wrapperId)
  494. const getInnerElements = wxSelectorQuery(elementsClass)
  495. return Promise.all([getWrapperElement, getInnerElements]).then(data => {
  496. return drawCanvas(canvasId, data[0], data[1])
  497. })
  498. }
  499. // export default wxml2canvas
  500. module.exports = wxml2canvas