draw.js 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941
  1. import { toPx, CHAR_WIDTH_SCALE_MAP, isNumber, getImageInfo } from './utils'
  2. import { GD } from './gradient'
  3. import QR from './qrcode'
  4. let id = 0
  5. export class Draw {
  6. constructor(context, canvas, use2dCanvas = false, isH5PathToBase64 = false, boundary) {
  7. this.ctx = context
  8. this.canvas = canvas || null
  9. this.root = boundary
  10. this.use2dCanvas = use2dCanvas
  11. this.isH5PathToBase64 = isH5PathToBase64
  12. }
  13. roundRect(x, y, w, h, r, fill = false, stroke = false, ) {
  14. if (r < 0) return
  15. const {ctx} = this
  16. ctx.beginPath()
  17. if(!r) {
  18. ctx.rect(x, y, w, h)
  19. } else if(typeof r === 'number' && [0,1,-1].includes(w - r * 2) && [0, 1, -1].includes(h - r * 2)) {
  20. ctx.arc(x + w - r, y + h - r, r, 0, Math.PI * 2)
  21. } else {
  22. let {
  23. borderTopLeftRadius: tl = r || 0,
  24. borderTopRightRadius: tr = r || 0,
  25. borderBottomRightRadius: br = r || 0,
  26. borderBottomLeftRadius: bl = r || 0
  27. } = r || {r,r,r,r}
  28. ctx.beginPath()
  29. // 右下角
  30. ctx.arc(x + w - br, y + h - br, br, 0, Math.PI * 0.5)
  31. ctx.lineTo(x + bl, y + h)
  32. // 左下角
  33. ctx.arc(x + bl, y + h - bl, bl, Math.PI * 0.5, Math.PI)
  34. ctx.lineTo(x, y + tl)
  35. // 左上角
  36. ctx.arc(x + tl, y + tl, tl, Math.PI, Math.PI * 1.5)
  37. ctx.lineTo(x + w - tr, y)
  38. // 右上角
  39. ctx.arc(x + w - tr, y + tr, tr, Math.PI * 1.5, Math.PI * 2)
  40. ctx.lineTo(x + w, y + h - br)
  41. }
  42. ctx.closePath()
  43. if (stroke) ctx.stroke()
  44. if (fill) ctx.fill()
  45. }
  46. measureText(text, fontSize) {
  47. const { ctx } = this
  48. // #ifndef APP-PLUS
  49. return ctx.measureText(text).width
  50. // #endif
  51. // #ifdef APP-PLUS
  52. // app measureText为0需要累加计算0
  53. return text.split("").reduce((widthScaleSum, char) => {
  54. let code = char.charCodeAt(0);
  55. let widthScale = CHAR_WIDTH_SCALE_MAP[code - 0x20] || 1;
  56. return widthScaleSum + widthScale;
  57. }, 0) * fontSize;
  58. // #endif
  59. }
  60. setFont({fontFamily: ff = 'sans-serif', fontSize: fs = 14, fontWeight: fw = 'normal' , textStyle: ts = 'normal'}) {
  61. let ctx = this.ctx
  62. // 设置属性
  63. // #ifndef MP-TOUTIAO
  64. // fw = fw == 'bold' ? 'bold' : 'normal'
  65. // ts = ts == 'italic' ? 'italic' : 'normal'
  66. // #endif
  67. // #ifdef MP-TOUTIAO
  68. fw = fw == 'bold' ? 'bold' : ''
  69. ts = ts == 'italic' ? 'italic' : ''
  70. // #endif
  71. // fs = toPx(fs)
  72. ctx.font = `${ts} ${fw} ${fs}px ${ff}`;
  73. }
  74. setTransform(box, {transform}) {
  75. const {ctx} = this
  76. const {
  77. scaleX = 1,
  78. scaleY = 1,
  79. translateX = 0,
  80. translateY = 0,
  81. rotate = 0,
  82. skewX = 0,
  83. skewY = 0
  84. } = transform || {}
  85. let {
  86. left: x,
  87. top: y,
  88. width: w,
  89. height: h
  90. } = box
  91. ctx.scale(scaleX, scaleY)
  92. ctx.translate(
  93. w * (scaleX > 0 ? 1 : -1) / 2 + (x + translateX) / scaleX,
  94. h * (scaleY > 0 ? 1 : -1) / 2 + (y + translateY) / scaleY)
  95. if(rotate) {
  96. ctx.rotate(rotate * Math.PI / 180)
  97. }
  98. if(skewX || skewY) {
  99. ctx.transform(1, Math.tan(skewY * Math.PI/180), Math.tan(skewX * Math.PI/180), 1 , 0, 0)
  100. }
  101. }
  102. setBackground(bd, w, h) {
  103. const {ctx} = this
  104. if (!bd) {
  105. // #ifndef MP-TOUTIAO || MP-BAIDU
  106. ctx.setFillStyle('transparent')
  107. // #endif
  108. // #ifdef MP-TOUTIAO || MP-BAIDU
  109. ctx.setFillStyle('rgba(0,0,0,0)')
  110. // #endif
  111. } else if(GD.isGradient(bd)) {
  112. GD.doGradient(bd, w, h, ctx);
  113. } else {
  114. ctx.setFillStyle(bd)
  115. }
  116. }
  117. setShadow({boxShadow: bs = []}) {
  118. // #ifndef APP-NVUE
  119. const {ctx} = this
  120. if (bs.length) {
  121. const [x, y, b, c] = bs
  122. ctx.setShadow(x, y, b, c)
  123. }
  124. // #endif
  125. }
  126. setBorder(box, style) {
  127. const {ctx} = this
  128. let {
  129. left: x,
  130. top: y,
  131. width: w,
  132. height: h
  133. } = box
  134. const {border, borderBottom, borderTop, borderRight, borderLeft, borderRadius: r, opacity = 1} = style;
  135. const {
  136. borderWidth : bw = 0,
  137. borderStyle : bs,
  138. borderColor : bc,
  139. } = border || {}
  140. const {
  141. borderBottomWidth : bbw = bw,
  142. borderBottomStyle : bbs = bs,
  143. borderBottomColor : bbc= bc,
  144. } = borderBottom || {}
  145. const {
  146. borderTopWidth: btw = bw,
  147. borderTopStyle: bts = bs,
  148. borderTopColor: btc = bc,
  149. } = borderTop || {}
  150. const {
  151. borderRightWidth: brw = bw,
  152. borderRightStyle: brs = bs,
  153. borderRightColor: brc = bc,
  154. } = borderRight || {}
  155. const {
  156. borderLeftWidth: blw = bw,
  157. borderLeftStyle: bls = bs,
  158. borderLeftColor: blc = bc,
  159. } = borderLeft || {}
  160. let {
  161. borderTopLeftRadius: tl = r || 0,
  162. borderTopRightRadius: tr = r || 0,
  163. borderBottomRightRadius: br = r || 0,
  164. borderBottomLeftRadius: bl = r || 0
  165. } = r || {r,r,r,r}
  166. if(!borderBottom && !borderLeft && !borderTop && !borderRight && !border) return;
  167. const _borderType = (w, s, c) => {
  168. // #ifndef APP-NVUE
  169. if (s == 'dashed') {
  170. // #ifdef MP
  171. ctx.setLineDash([Math.ceil(w * 4 / 3), Math.ceil(w * 4 / 3)])
  172. // #endif
  173. // #ifndef MP
  174. ctx.setLineDash([Math.ceil(w * 6), Math.ceil(w * 6)])
  175. // #endif
  176. } else if (s == 'dotted') {
  177. ctx.setLineDash([w, w])
  178. }
  179. // #endif
  180. ctx.setStrokeStyle(c)
  181. }
  182. const _setBorder = (x1, y1, x2, y2, x3, y3, r1, r2, p1, p2, p3, bw, bs, bc) => {
  183. ctx.save()
  184. this.setOpacity(style)
  185. this.setTransform(box, style)
  186. ctx.setLineWidth(bw)
  187. _borderType(bw, bs, bc)
  188. ctx.beginPath()
  189. ctx.arc(x1, y1, r1, Math.PI * p1, Math.PI * p2)
  190. ctx.lineTo(x2, y2)
  191. ctx.arc(x3, y3, r2, Math.PI * p2, Math.PI * p3)
  192. ctx.stroke()
  193. ctx.restore()
  194. }
  195. if(border) {
  196. ctx.save()
  197. this.setOpacity(style)
  198. this.setTransform(box, style)
  199. ctx.setLineWidth(bw)
  200. _borderType(bw, bs, bc)
  201. this.roundRect(-w/2, -h/2, w, h, r, false, bc ? true : false)
  202. ctx.restore()
  203. }
  204. x = -w/2
  205. y = -h/2
  206. if(borderBottom) {
  207. _setBorder(x + w - br, y + h - br, x + bl, y + h, x + bl, y + h - bl, br, bl, 0.25, 0.5, 0.75, bbw, bbs, bbc)
  208. }
  209. if(borderLeft) {
  210. // 左下角
  211. _setBorder(x + bl, y + h - bl, x, y + tl, x + tl, y + tl, bl, tl, 0.75, 1, 1.25, blw, bls, blc)
  212. }
  213. if(borderTop) {
  214. // 左上角
  215. _setBorder(x + tl, y + tl, x + w - tr, y, x + w - tr, y + tr, tl, tr, 1.25, 1.5, 1.75, btw, bts, btc)
  216. }
  217. if(borderRight) {
  218. // 右上角
  219. _setBorder(x + w - tr, y + tr, x + w, y + h - br, x + w - br, y + h - br, tr, br, 1.75, 2, 0.25, btw, bts, btc)
  220. }
  221. }
  222. setOpacity({opacity = 1}) {
  223. this.ctx.setGlobalAlpha(opacity)
  224. }
  225. drawView(box, style) {
  226. const {ctx} = this
  227. const {
  228. left: x,
  229. top: y,
  230. width: w,
  231. height: h
  232. } = box
  233. let {
  234. borderRadius = 0,
  235. border,
  236. borderTop,
  237. borderBottom,
  238. borderLeft,
  239. borderRight,
  240. color = '#000000',
  241. backgroundColor: bg,
  242. rotate,
  243. shadow
  244. } = style || {}
  245. ctx.save()
  246. this.setOpacity(style)
  247. this.setTransform(box, style)
  248. this.setShadow(style)
  249. this.setBackground(bg, w, h)
  250. this.roundRect(-w/2, -h/2, w, h, borderRadius, true, false)
  251. ctx.restore()
  252. this.setBorder(box, style)
  253. }
  254. async drawImage(img, box = {}, style = {}, custom = true) {
  255. await new Promise(async (resolve, reject) => {
  256. const {ctx} = this
  257. const canvas = this.canvas
  258. const {
  259. borderRadius = 0,
  260. mode,
  261. backgroundColor: bg,
  262. } = style
  263. let {
  264. left: x,
  265. top: y,
  266. width: w,
  267. height: h
  268. } = box
  269. ctx.save()
  270. if(!custom) {
  271. this.setOpacity(style)
  272. this.setTransform(box, style)
  273. this.setBackground(bg, w, h)
  274. this.setShadow(style)
  275. x = -w/2
  276. y = -h/2
  277. this.roundRect(x, y, w, h, borderRadius, true, false)
  278. }
  279. ctx.clip()
  280. const _modeImage = (img) => {
  281. // 获得图片原始大小
  282. let rWidth = img.width
  283. let rHeight = img.height
  284. let startX = 0
  285. let startY = 0
  286. // 绘画区域比例
  287. const cp = w / h
  288. // 原图比例
  289. const op = rWidth / rHeight
  290. if (cp >= op) {
  291. rHeight = rWidth / cp;
  292. // startY = Math.round((h - rHeight) / 2)
  293. } else {
  294. rWidth = rHeight * cp;
  295. startX = Math.round(((img.width || w) - rWidth) / 2)
  296. }
  297. if (mode === 'scaleToFill' || !img.width) {
  298. ctx.drawImage(img.src, x, y, w, h);
  299. } else {
  300. // 百度小程序 开发工具 顺序有问题 暂不知晓真机
  301. // #ifdef MP-BAIDU
  302. ctx.drawImage(img.src, x, y, w, h, startX, startY, rWidth, rHeight)
  303. // #endif
  304. // #ifndef MP-BAIDU
  305. ctx.drawImage(img.src, startX, startY, rWidth, rHeight, x, y, w, h)
  306. // #endif
  307. }
  308. }
  309. const _drawImage = (img) => {
  310. if (this.use2dCanvas) {
  311. const Image = canvas.createImage()
  312. Image.onload = () => {
  313. img.src = Image
  314. _modeImage(img)
  315. _restore()
  316. }
  317. Image.onerror = () => {
  318. console.error(`createImage fail: ${img}`)
  319. reject(new Error(`createImage fail: ${img}`))
  320. }
  321. Image.src = img.src
  322. } else {
  323. _modeImage(img)
  324. _restore()
  325. }
  326. }
  327. const _restore = () => {
  328. ctx.restore()
  329. this.setBorder(box, style)
  330. setTimeout(() => {
  331. resolve(true)
  332. }, this.root.sleep)
  333. }
  334. if(typeof img === 'string') {
  335. const {path: src, width, height} = await getImageInfo(img, this.isH5PathToBase64)
  336. _drawImage({src, width, height})
  337. } else {
  338. _drawImage(img)
  339. }
  340. })
  341. }
  342. drawText(text, box, style, rules) {
  343. const {ctx} = this
  344. let {
  345. left: x,
  346. top: y,
  347. width: w,
  348. height: h,
  349. offsetLeft: ol = 0
  350. } = box
  351. let {
  352. color = '#000000',
  353. lineHeight = '1.4em',
  354. fontSize = 14,
  355. fontWeight,
  356. fontFamily,
  357. textStyle,
  358. textAlign = 'left',
  359. verticalAlign: va = 'top',
  360. backgroundColor: bg,
  361. maxLines,
  362. textDecoration: td
  363. } = style
  364. lineHeight = toPx(lineHeight, fontSize)
  365. if (!text) return
  366. ctx.save()
  367. this.setOpacity(style)
  368. this.setTransform(box, style)
  369. x = -w/2
  370. y = -h/2
  371. ctx.setTextBaseline(va)
  372. this.setFont({fontFamily, fontSize, fontWeight, textStyle})
  373. ctx.setTextAlign(textAlign)
  374. if(bg) {
  375. this.setBackground(bg, w, h)
  376. this.roundRect(x, y, w, h, 1, bg)
  377. }
  378. this.setShadow(style)
  379. ctx.setFillStyle(color)
  380. let rulesObj = {};
  381. if(rules) {
  382. if (rules.word.length > 0) {
  383. for (let i = 0; i < rules.word.length; i++) {
  384. let startIndex = 0,
  385. index;
  386. while ((index = text.indexOf(rules.word[i], startIndex)) > -1) {
  387. rulesObj[index] = {
  388. reset: true
  389. };
  390. for (let j = 0; j < rules.word[i].length; j++) {
  391. rulesObj[index + j] = {
  392. reset: true
  393. };
  394. }
  395. startIndex = index + 1;
  396. }
  397. }
  398. }
  399. }
  400. // 水平布局
  401. switch (textAlign) {
  402. case 'left':
  403. break
  404. case 'center':
  405. x += 0.5 * w
  406. break
  407. case 'right':
  408. x += w
  409. break
  410. default:
  411. break
  412. }
  413. const textWidth = this.measureText(text, fontSize)
  414. const actualHeight = Math.ceil(textWidth / w) * lineHeight
  415. let paddingTop = Math.ceil((h - actualHeight) / 2)
  416. if (paddingTop < 0) paddingTop = 0
  417. // 垂直布局
  418. switch (va) {
  419. case 'top':
  420. break
  421. case 'middle':
  422. y += fontSize / 2
  423. break
  424. case 'bottom':
  425. y += fontSize
  426. break
  427. default:
  428. break
  429. }
  430. // 绘线
  431. const _drawLine = (x, y, textWidth) => {
  432. const { system } = uni.getSystemInfoSync()
  433. if(/win|mac/.test(system)){
  434. y += (fontSize / 3)
  435. }
  436. // 垂直布局
  437. switch (va) {
  438. case 'top':
  439. break
  440. case 'middle':
  441. y -= fontSize / 2
  442. break
  443. case 'bottom':
  444. y -= fontSize
  445. break
  446. default:
  447. break
  448. }
  449. let to = x
  450. switch (textAlign) {
  451. case 'left':
  452. x = x
  453. to+= textWidth
  454. break
  455. case 'center':
  456. x = x - textWidth / 2
  457. to = x + textWidth
  458. break
  459. case 'right':
  460. to = x
  461. x = x - textWidth
  462. break
  463. default:
  464. break
  465. }
  466. if(td) {
  467. ctx.setLineWidth(fontSize / 13);
  468. ctx.beginPath();
  469. if (/\bunderline\b/.test(td)) {
  470. y -= inlinePaddingTop * 0.8
  471. ctx.moveTo(x, y);
  472. ctx.lineTo(to, y);
  473. }
  474. if (/\boverline\b/.test(td)) {
  475. y += inlinePaddingTop
  476. ctx.moveTo(x, y - lineHeight);
  477. ctx.lineTo(to, y - lineHeight);
  478. }
  479. if (/\bline-through\b/.test(td)) {
  480. ctx.moveTo(x , y - lineHeight / 2 );
  481. ctx.lineTo(to, y - lineHeight /2 );
  482. }
  483. ctx.closePath();
  484. ctx.setStrokeStyle(color);
  485. ctx.stroke();
  486. }
  487. }
  488. const _reset = (text, x, y) => {
  489. const rs = Object.keys(rulesObj)
  490. for (let i = 0; i < rs.length; i++) {
  491. const item = rulesObj[rs[i]]
  492. // ctx.globalCompositeOperation = "destination-out";
  493. ctx.save();
  494. ctx.setFillStyle(rules.color);
  495. if(item.char) {
  496. ctx.fillText(item.char, item.x , item.y)
  497. }
  498. ctx.restore();
  499. }
  500. }
  501. const _setText = (isReset, char) => {
  502. if(isReset) {
  503. const t1 = Math.round(this.measureText('\u0020', fontSize))
  504. const t2 = Math.round(this.measureText('\u3000', fontSize))
  505. const t3 = Math.round(this.measureText(char, fontSize))
  506. let _char = ''
  507. let _num = 1
  508. if(t3 == t2){
  509. _char ='\u3000'
  510. _num = 1
  511. } else {
  512. _char = '\u0020'
  513. _num = Math.ceil(t3 / t1)
  514. }
  515. return new Array(_num).fill(_char).join('')
  516. } else {
  517. return char
  518. }
  519. }
  520. const _setRulesObj = (text, index, x, y) => {
  521. rulesObj[index].x = x
  522. rulesObj[index].y = y
  523. rulesObj[index].char = text
  524. }
  525. const inlinePaddingTop = Math.ceil((lineHeight - fontSize) / 2)
  526. // 不超过一行
  527. if (textWidth + ol <= w && !text.includes('\n')) {
  528. x = x + ol
  529. const rs = Object.keys(rulesObj)
  530. let _text = text.split('')
  531. if(rs) {
  532. for (let i = 0; i < rs.length; i++) {
  533. const index = rs[i]
  534. const t = _text[index]
  535. let char = _setText(rulesObj[index], t)
  536. _text[index] = char
  537. _setRulesObj(t, index, x + this.measureText(text.substring(0, index), fontSize), y + inlinePaddingTop)
  538. }
  539. _reset()
  540. }
  541. ctx.fillText(_text.join(''), x, y + inlinePaddingTop)
  542. y += lineHeight
  543. _drawLine(x, y, textWidth)
  544. ctx.restore()
  545. this.setBorder(box, style)
  546. return
  547. }
  548. // 多行文本
  549. const chars = text.split('')
  550. const _y = y
  551. let _x = x
  552. // 逐行绘制
  553. let line = ''
  554. let lineIndex = 0
  555. for(let index = 0 ; index <= chars.length; index++){
  556. let ch = chars[index] || ''
  557. const isLine = ch === '\n'
  558. const isRight = ch == ''// index == chars.length
  559. ch = isLine ? '' : ch;
  560. let textline = line + _setText(rulesObj[index], ch)
  561. let textWidth = this.measureText(textline, fontSize)
  562. // 绘制行数大于最大行数,则直接跳出循环
  563. if (lineIndex >= maxLines) {
  564. break;
  565. }
  566. if(lineIndex == 0) {
  567. textWidth = textWidth + ol
  568. _x += ol
  569. }
  570. if(rulesObj[index]) {
  571. _setRulesObj(ch, index, _x + this.measureText(line, fontSize), y + inlinePaddingTop)
  572. }
  573. if (textWidth > w || isLine || isRight) {
  574. lineIndex++
  575. line = isRight && textWidth <= w ? textline : line
  576. if(lineIndex === maxLines && textWidth > w) {
  577. while( this.measureText(`${line}...`, fontSize) > w) {
  578. if (line.length <= 1) {
  579. // 如果只有一个字符时,直接跳出循环
  580. break;
  581. }
  582. line = line.substring(0, line.length - 1);
  583. if(rulesObj[index - 1]) {
  584. rulesObj[index - 1].char = ''
  585. }
  586. }
  587. line += '...'
  588. }
  589. ctx.fillText(line, _x, y + inlinePaddingTop)
  590. y += lineHeight
  591. _drawLine(_x, y, textWidth)
  592. line = ch
  593. if ((y + lineHeight) > (_y + h)) break
  594. } else {
  595. line = textline
  596. }
  597. }
  598. const rs = Object.keys(rulesObj)
  599. if(rs) {
  600. _reset()
  601. }
  602. ctx.restore()
  603. this.setBorder(box, style)
  604. }
  605. async findNode(element, parent = {}, index = 0, siblings = [], source) {
  606. let computedStyle = Object.assign({}, this.getComputedStyle(element, parent, index));
  607. let attributes = await this.getAttributes(element)
  608. let node = {
  609. id: id++,
  610. parent,
  611. computedStyle,
  612. rules: element.rules,
  613. attributes: Object.assign({}, attributes),
  614. name: element?.type || 'view',
  615. }
  616. if(JSON.stringify(parent) === '{}' && !element.type) {
  617. const {left = 0, top = 0, width = 0, height = 0 } = computedStyle
  618. node.layoutBox = {left, top, width, height }
  619. } else {
  620. node.layoutBox = Object.assign({left: 0, top: 0}, this.getLayoutBox(node, parent, index, siblings, source))
  621. }
  622. if (element?.views) {
  623. let childrens = []
  624. node.children = []
  625. for (let i = 0; i < element.views.length; i++) {
  626. let v = element.views[i]
  627. childrens.push(await this.findNode(v, node, i, childrens, element))
  628. }
  629. node.children = childrens
  630. }
  631. return node
  632. }
  633. getComputedStyle(element, parent = {}, index = 0) {
  634. const style = {}
  635. const node = JSON.stringify(parent) == '{}' && !element.type ? element : element.css;
  636. if(parent.computedStyle) {
  637. for (let value of Object.keys(parent.computedStyle)){
  638. const item = parent.computedStyle[value]
  639. if(['color', 'fontSize', 'lineHeight', 'verticalAlign', 'fontWeight', 'textAlign'].includes(value)) {
  640. style[value] = /em|px$/.test(item) ? toPx(item, node?.fontSize) : item
  641. }
  642. }
  643. }
  644. if(!node) return style
  645. for (let value of Object.keys(node)) {
  646. const item = node[value]
  647. if(value == 'views') {
  648. continue
  649. }
  650. if (['boxShadow', 'shadow'].includes(value)) {
  651. let shadows = item.split(' ').map(v => /^\d/.test(v) ? toPx(v) : v)
  652. style.boxShadow = shadows
  653. continue
  654. }
  655. if (value.includes('border') && !value.includes('adius')) {
  656. const prefix = value.match(/^border([BTRLa-z]+)?/)[0]
  657. const type = value.match(/[W|S|C][a-z]+/)
  658. let v = item.split(' ').map(v => /^\d/.test(v) ? toPx(v) : v)
  659. if(v.length > 1) {
  660. style[prefix] = {
  661. [prefix + 'Width'] : v[0] || 1,
  662. [prefix + 'Style'] : v[1] || 'solid',
  663. [prefix + 'Color'] : v[2] || 'black'
  664. }
  665. } else {
  666. style[prefix] = {
  667. [prefix + 'Width'] : 1,
  668. [prefix + 'Style'] : 'solid',
  669. [prefix + 'Color'] : 'black'
  670. }
  671. style[prefix][prefix + type[0]] = v[0]
  672. }
  673. continue
  674. }
  675. if (['background', 'backgroundColor'].includes(value)) {
  676. style['backgroundColor'] = item
  677. continue
  678. }
  679. if(value.includes('padding') || value.includes('margin') || value.includes('adius')) {
  680. let isRadius = value.includes('adius')
  681. let prefix = isRadius ? 'borderRadius' : value.match(/[a-z]+/)[0]
  682. let pre = [0,0,0,0].map((item, i) => isRadius ? ['borderTopLeftRadius', 'borderTopRightRadius', 'borderBottomRightRadius', 'borderBottomLeftRadius'][i] : [prefix + 'Top', prefix + 'Right', prefix + 'Bottom', prefix + 'Left'][i] )
  683. if(value === 'padding' || value === 'margin' || value === 'radius' || value === 'borderRadius') {
  684. let v = item?.split(' ').map((item) => /^\d/.test(item) && toPx(item, node['width']), []) ||[0];
  685. let type = isRadius ? 'borderRadius' : value;
  686. if(v.length == 1) {
  687. style[type] = v[0]
  688. } else {
  689. let [t, r, b, l] = v
  690. style[type] = {
  691. [pre[0]]: t,
  692. [pre[1]]: isNumber(r) ? r : t,
  693. [pre[2]]: isNumber(b) ? b : t,
  694. [pre[3]]: isNumber(l) ? l : r
  695. }
  696. }
  697. } else {
  698. if(typeof style[prefix] === 'object') {
  699. style[prefix][value] = toPx(item, node['width'])
  700. } else {
  701. style[prefix] = {
  702. [pre[0]]: style[prefix] || 0,
  703. [pre[1]]: style[prefix] || 0,
  704. [pre[2]]: style[prefix] || 0,
  705. [pre[3]]: style[prefix] || 0
  706. }
  707. style[prefix][value] = toPx(item, node['width'])
  708. }
  709. }
  710. continue
  711. }
  712. if(value.includes('width') || value.includes('height')) {
  713. if(/%$/.test(item)) {
  714. style[value] = toPx(item, parent.layoutBox[value])
  715. } else {
  716. style[value] = /px|rpx$/.test(item) ? toPx(item) : item
  717. }
  718. continue
  719. }
  720. if(value.includes('transform')) {
  721. style[value]= {}
  722. item.replace(/([a-zA-Z]+)\(([0-9,-\.%rpxdeg\s]+)\)/g, (g1, g2, g3) => {
  723. const v = g3.split(',').map(k => k.replace(/(^\s*)|(\s*$)/g,''))
  724. const transform = (v, r) => {
  725. return v.includes('deg') ? v * 1 : toPx(v, r)
  726. }
  727. if(g2.includes('matrix')) {
  728. style[value][g2] = v.map(v => v * 1)
  729. } else if(g2.includes('rotate')) {
  730. style[value][g2] = g3.match(/\d+/)[0] * 1
  731. }else if(/[X, Y]/.test(g2)) {
  732. style[value][g2] = /[X]/.test(g2) ? transform(v[0], node['width']) : transform(v[0], node['height'])
  733. } else {
  734. style[value][g2+'X'] = transform(v[0], node['width'])
  735. style[value][g2+'Y'] = transform(v[1] || v[0], node['height'])
  736. }
  737. })
  738. continue
  739. }
  740. if(/em$/.test(item) && !value.includes('lineHeight')) {
  741. style[value] = Math.ceil(parseFloat(item.replace('em')) * toPx(node['fontSize'] || 14))
  742. } else {
  743. style[value] = /%|px|rpx$/.test(item) ? toPx(item, node['width']) : item
  744. }
  745. }
  746. if((element.name == 'image' || element.type == 'image') && !style.mode) {
  747. style.mode = 'aspectFill'
  748. if((!node.width || node.width == 'auto') && (!node.height || node.width == 'auto') ) {
  749. style.mode = ''
  750. }
  751. }
  752. return style
  753. }
  754. getLayoutBox(element, parent = {}, index = 0, siblings = [], source = {}) {
  755. let box = {}
  756. let {name, computedStyle: cstyle, layoutBox, attributes} = element || {}
  757. if(!name) return box
  758. const {ctx} = this
  759. const pbox = parent.layoutBox || this.root
  760. const pstyle = parent.computedStyle
  761. let {
  762. verticalAlign: v,
  763. left: x,
  764. top: y,
  765. width: w,
  766. height: h,
  767. fontSize = 14,
  768. lineHeight = '1.4em',
  769. maxLines,
  770. fontWeight,
  771. fontFamily,
  772. textStyle,
  773. position,
  774. display
  775. } = cstyle;
  776. const { paddingTop: pt = 0, paddingRight: pr = 0, paddingBottom: pb = 0, paddingLeft: pl = 0, } = cstyle.padding || {}
  777. const { marginTop: mt = 0, marginRight: mr = 0, marginBottom: mb = 0, marginLeft: ml = 0, } = cstyle.margin || {}
  778. if(position == 'relative') {
  779. x += pbox.left
  780. y += pbox.top
  781. }
  782. if(name === 'text') {
  783. const text = attributes.text ||''
  784. lineHeight = toPx(lineHeight, fontSize)
  785. ctx.save()
  786. this.setFont({fontFamily, fontSize, fontWeight, textStyle})
  787. const {layoutBox: lbox, computedStyle: ls} = siblings[index - 1] || {}
  788. const {layoutBox: rbox, computedStyle: rs} = siblings[index + 1] || {}
  789. const isLeft = index == 0
  790. const isblock = display === 'block' || ls?.display === 'block'
  791. const isOnly = isLeft && !rbox || !parent?.id
  792. const lboxR = isLeft || isblock ? 0 : lbox.offsetRight || 0
  793. let texts = text.split('\n')
  794. let lineIndex = 1
  795. let line = ''
  796. const textIndent = cstyle.textIndent || 0
  797. if(!isOnly) {
  798. texts.map((t, i) => {
  799. lineIndex += i
  800. const chars = t.split('')
  801. for (let j = 0; j < chars.length; j++) {
  802. let ch = chars[j]
  803. let textline = line + ch
  804. let textWidth = this.measureText(textline, fontSize)
  805. if(lineIndex == 1) {
  806. textWidth = textWidth + (isblock ? 0 : lboxR) + textIndent
  807. }
  808. if(textWidth > pbox.width) {
  809. lineIndex++
  810. line = ch
  811. } else {
  812. line = textline
  813. }
  814. }
  815. })
  816. } else {
  817. line = text
  818. lineIndex = Math.max(texts.length, Math.ceil(this.measureText(text, fontSize) / ((w || pbox.width) - this.measureText('0', fontSize))))
  819. }
  820. box.offsetLeft = (isNumber(x) || isblock || isOnly ? textIndent : lboxR) + pl + ml;
  821. // 剩下的字宽度
  822. const remain = (this.measureText(line, fontSize))
  823. let width = lineIndex > 1 ? pbox.width : remain + box.offsetLeft;
  824. box.offsetRight = box.offsetLeft + (w ? w : (isblock ? pbox.width : remain)) + pr + mr;
  825. const _getLeft = () => {
  826. return (x || pbox.left)
  827. }
  828. const _getWidth = () => {
  829. return w || (isOnly ? pbox.width : (width > pbox.width - box.left || lineIndex > 1 ? pbox.width - box.left : width))
  830. }
  831. const _getHeight = () => {
  832. if(h) {
  833. return h
  834. } else if(lineIndex > 1 ) {
  835. return (maxLines || lineIndex) * lineHeight + pt + pb + mt + mb
  836. } else {
  837. return lineHeight + pt + pb + mt + mb
  838. }
  839. }
  840. const _getTop = () => {
  841. let _y = y
  842. if(_y) {
  843. return _y + pt + mt
  844. }
  845. if(isLeft) {
  846. _y = pbox.top
  847. } else if(lbox.width < pbox.width) {
  848. _y = lbox.top
  849. } else {
  850. _y = lbox.top + lbox.height - (ls?.lineHeight || 0)
  851. }
  852. return _y + pt + mt + (isblock && ls?.lineHeight || 0 )
  853. }
  854. box.left = _getLeft()
  855. box.width = _getWidth()
  856. box.height = _getHeight()
  857. box.top = _getTop()
  858. ctx.restore()
  859. } else if(['view', 'qrcode'].includes(name)) {
  860. box.left = x || pbox.left
  861. box.width = (w || pbox?.width) - pl - pr
  862. box.height = h || 0
  863. box.top = y || pbox.top
  864. } else if(name === 'image') {
  865. box.left = x || pbox.left
  866. box.width = (w || pbox?.width) - pl - pr
  867. const {
  868. width: rWidth,
  869. height: rHeight
  870. } = attributes
  871. box.height = h || box.width * rHeight / rWidth
  872. box.top = y || pbox.top
  873. }
  874. return box
  875. }
  876. async getAttributes(element) {
  877. let arr = { }
  878. if(element?.url || element?.src) {
  879. arr.src = element.url || element?.src;
  880. const {width = 0, height = 0, path: src} = await getImageInfo(arr.src, this.isH5PathToBase64) || {}
  881. arr = Object.assign({}, arr, {width, height, src})
  882. }
  883. if(element?.text) {
  884. arr.text = element.text
  885. }
  886. return arr
  887. }
  888. async drawBoard(element) {
  889. const node = await this.findNode(element)
  890. return this.drawNode(node)
  891. }
  892. async drawNode(element) {
  893. const {
  894. layoutBox,
  895. computedStyle,
  896. name,
  897. rules
  898. } = element
  899. const {
  900. src,
  901. text
  902. } = element.attributes
  903. if (name === 'view') {
  904. this.drawView(layoutBox, computedStyle)
  905. } else if (name === 'image' && src) {
  906. await this.drawImage(element.attributes, layoutBox, computedStyle, false)
  907. } else if (name === 'text') {
  908. this.drawText(text, layoutBox, computedStyle, rules)
  909. } else if (name === 'qrcode') {
  910. QR.api.draw(text, this, layoutBox, computedStyle)
  911. }
  912. if (!element.children) return
  913. const childs = Object.values ? Object.values(element.children) : Object.keys(element.children).map((key) => element.children[key]);
  914. for (const child of childs) {
  915. await this.drawNode(child)
  916. }
  917. }
  918. }