QS-SharePoster.js 45 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589
  1. import _app from './app.js';
  2. import QRCodeAlg from './QRCodeAlg.js';
  3. import {
  4. base64ToPath
  5. } from './image-tools.js';
  6. const ShreUserPosterBackgroundKey = 'ShrePosterBackground_'; // 背景图片缓存名称前缀
  7. const idKey = 'QSSHAREPOSTER_IDKEY'; //drawArray自动生成的idkey
  8. var nbgScale = 1;
  9. // export default
  10. function getSharePoster(obj) {
  11. let {
  12. type,
  13. formData,
  14. background,
  15. posterCanvasId,
  16. backgroundImage,
  17. reserve,
  18. textArray,
  19. drawArray,
  20. qrCodeArray,
  21. imagesArray,
  22. setCanvasWH,
  23. setCanvasToTempFilePath,
  24. canvas2image,
  25. setDraw,
  26. bgScale,
  27. Context,
  28. _this,
  29. delayTimeScale,
  30. drawDelayTime,
  31. draw
  32. } = obj;
  33. return new Promise(async (rs, rj) => {
  34. try {
  35. if (!Context) {
  36. _app.log('没有画布对象,创建画布对象');
  37. Context = uni.createCanvasContext(posterCanvasId, (_this || null));
  38. }
  39. let bgObj;
  40. if (background && background.width && background.height) {
  41. bgObj = background;
  42. } else {
  43. bgObj = await getShreUserPosterBackground({
  44. backgroundImage,
  45. type,
  46. formData
  47. });
  48. }
  49. bgScale = bgScale || nbgScale;
  50. bgObj.width = bgObj.width * bgScale;
  51. bgObj.height = bgObj.height * bgScale;
  52. _app.log('获取背景图信息对象成功:' + JSON.stringify(bgObj));
  53. const params = {
  54. bgObj,
  55. type,
  56. bgScale,
  57. getBgObj: function() {
  58. return params.bgObj;
  59. },
  60. setBgObj: function(newBgObj) {
  61. const n = {
  62. ...params.bgObj,
  63. ...newBgObj
  64. };
  65. params.bgObj = n;
  66. bgObj = n;
  67. }
  68. };
  69. if (imagesArray) {
  70. if (typeof(imagesArray) == 'function')
  71. imagesArray = imagesArray(params);
  72. _app.log('准备设置图片');
  73. imagesArray = await setImage(imagesArray);
  74. }
  75. if (textArray) {
  76. if (typeof(textArray) == 'function')
  77. textArray = textArray(params);
  78. textArray = setText(Context, textArray);
  79. }
  80. if (qrCodeArray) {
  81. if (typeof(qrCodeArray) == 'function')
  82. qrCodeArray = qrCodeArray(params);
  83. for (let i = 0; i < qrCodeArray.length; i++) {
  84. _app.log(i);
  85. if (qrCodeArray[i].image)
  86. qrCodeArray[i].image = await _app.downloadFile_PromiseFc(qrCodeArray[i].image);
  87. }
  88. }
  89. if (drawArray) {
  90. if (typeof(drawArray) == 'function') {
  91. drawArray = drawArray(params);
  92. }
  93. if (_app.isPromise(drawArray)) {
  94. drawArray = await drawArray;
  95. }
  96. if (_app.isArray(drawArray) && drawArray.length > 0) {
  97. // let hasAllInfoCallback = false;
  98. const addDrawArray = [];
  99. for (let i = 0; i < drawArray.length; i++) {
  100. const drawArrayItem = drawArray[i];
  101. // if (_app.isFn(drawArrayItem.allInfoCallback) && !hasAllInfoCallback)
  102. // hasAllInfoCallback = true;
  103. drawArrayItem[idKey] = i;
  104. let newData;
  105. let addDraw = false;
  106. switch (drawArrayItem.type) {
  107. case 'image':
  108. newData = await setImage(drawArrayItem);
  109. break;
  110. case 'text':
  111. newData = setText(Context, drawArrayItem, params.bgObj);
  112. // if (_app.isArray(setTextResult)) {
  113. // addDraw = true;
  114. // addDrawArray.push({
  115. // index: i,
  116. // items: setTextResult
  117. // });
  118. // } else {
  119. // newData = setTextResult;
  120. // }
  121. break;
  122. case 'qrcode':
  123. if (drawArrayItem.image)
  124. newData = {
  125. image: await _app.downloadFile_PromiseFc(drawArrayItem.image)
  126. };
  127. break;
  128. case 'custom':
  129. break;
  130. case 'fillrect':
  131. break;
  132. case 'strokeRect':
  133. break;
  134. case 'roundStrokeRect':
  135. break;
  136. case 'roundFillRect':
  137. break;
  138. default:
  139. _app.log('未识别的类型');
  140. break;
  141. }
  142. if (!addDraw && newData && _app.isObject(newData)) {
  143. drawArray[i] = {
  144. ...drawArrayItem,
  145. ...newData
  146. }
  147. };
  148. }
  149. // if (addDrawArray.length) {
  150. // for (let i = 0; i < addDrawArray.length; i++) {
  151. // const item = addDrawArray[i];
  152. // const index = drawArray.findIndex(ite => ite[idKey] == item.index);
  153. // if (-1 != index) {
  154. // item.items.forEach((ite, index) => {
  155. // ite[idKey] = drawArray.length + index;
  156. // ite.allInfoCallback = null;
  157. // });
  158. // drawArray.splice(index, 1, ...item.items)
  159. // }
  160. // }
  161. // }
  162. _app.log('AllInfoCallback之前', JSON.stringify(drawArray));
  163. // if (hasAllInfoCallback) {
  164. _app.log('----------------hasAllInfoCallback----------------');
  165. const drawArray_copy = [...drawArray];
  166. drawArray_copy.sort((a, b) => {
  167. const a_serialNum = !_app.isUndef(a.serialNum) && !_app.isNull(a
  168. .serialNum) ? Number(a.serialNum) : Number.NEGATIVE_INFINITY;
  169. const b_serialNum = !_app.isUndef(b.serialNum) && !_app.isNull(b
  170. .serialNum) ? Number(b.serialNum) : Number.NEGATIVE_INFINITY;
  171. return a_serialNum - b_serialNum;
  172. })
  173. _app.log('开始for循环');
  174. for (let i = 0; i < drawArray_copy.length; i++) {
  175. const item = {
  176. ...drawArray_copy[i]
  177. };
  178. const item_idKey = item[idKey];
  179. _app.log('item_idKey', item_idKey);
  180. const ind = drawArray.findIndex(it => it[idKey] == item_idKey);
  181. _app.log('ind', ind);
  182. if (-1 == ind) break;
  183. if (_app.isFn(item.allInfoCallback)) {
  184. let newData = item.allInfoCallback({
  185. drawArray
  186. });
  187. _app.log('newData', JSON.stringify(newData));
  188. if (_app.isPromise(newData)) newData = await newData;
  189. if (drawArray[ind].type === 'text' && newData.size) {
  190. const textLength = countTextLength(Context, {
  191. text: newData.text || drawArray[ind].text,
  192. size: newData.size
  193. });
  194. newData.textLength = textLength;
  195. }
  196. drawArray[ind] = {
  197. ...item,
  198. ...newData
  199. };
  200. }
  201. _app.log('drawArray[ind]', JSON.stringify(drawArray[ind]));
  202. if (drawArray[ind].type === 'text') {
  203. const setLineFeedResult = setLineFeed(Context, drawArray[ind], params.bgObj);
  204. if (_app.isArray(setLineFeedResult)) {
  205. setLineFeedResult.forEach((ite, index) => {
  206. ite[idKey] = drawArray.length + index;
  207. ite.allInfoCallback = null;
  208. })
  209. drawArray.splice(ind, 1, ...setLineFeedResult);
  210. } else {
  211. drawArray.splice(ind, 1, setLineFeedResult);
  212. }
  213. }
  214. }
  215. _app.log('for循环结束');
  216. _app.log('allInfocallback结束', JSON.stringify(drawArray));
  217. // }
  218. }
  219. }
  220. drawArray.sort((a, b) => {
  221. const a_zIndex = !_app.isUndef(a.zIndex) && !_app.isNull(a
  222. .zIndex) ? Number(a.zIndex) : Number.NEGATIVE_INFINITY;
  223. const b_zIndex = !_app.isUndef(b.zIndex) && !_app.isNull(b
  224. .zIndex) ? Number(b.zIndex) : Number.NEGATIVE_INFINITY;
  225. return a_zIndex - b_zIndex;
  226. });
  227. _app.log('params:' + JSON.stringify(params))
  228. if (setCanvasWH && typeof(setCanvasWH) == 'function') {
  229. await new Promise((resolve, reject) => {
  230. setCanvasWH(params);
  231. setTimeout(() => {
  232. resolve();
  233. }, 50)
  234. })
  235. }
  236. const poster = await drawShareImage({
  237. Context,
  238. type,
  239. posterCanvasId,
  240. reserve,
  241. drawArray,
  242. textArray,
  243. imagesArray,
  244. bgObj,
  245. qrCodeArray,
  246. setCanvasToTempFilePath,
  247. setDraw,
  248. bgScale,
  249. _this,
  250. delayTimeScale,
  251. drawDelayTime,
  252. canvas2image,
  253. draw
  254. });
  255. rs({
  256. bgObj,
  257. poster,
  258. type
  259. });
  260. } catch (e) {
  261. //TODO handle the exception
  262. _app.hideLoading();
  263. rj(e);
  264. }
  265. });
  266. }
  267. function drawShareImage(obj) { //绘制海报方法
  268. let {
  269. Context,
  270. type,
  271. posterCanvasId,
  272. reserve,
  273. bgObj,
  274. drawArray,
  275. textArray,
  276. qrCodeArray,
  277. imagesArray,
  278. setCanvasToTempFilePath,
  279. setDraw,
  280. bgScale,
  281. _this,
  282. delayTimeScale,
  283. drawDelayTime,
  284. canvas2image,
  285. draw
  286. } = obj;
  287. const params = {
  288. Context,
  289. bgObj,
  290. type,
  291. bgScale
  292. };
  293. delayTimeScale = delayTimeScale !== undefined ? delayTimeScale : 15;
  294. drawDelayTime = drawDelayTime !== undefined ? drawDelayTime : 100;
  295. return new Promise(async (rs, rj) => {
  296. try {
  297. _app.log('背景对象:' + JSON.stringify(bgObj));
  298. if (bgObj && bgObj.path) {
  299. _app.log('背景有图片路径');
  300. Context.drawImage(bgObj.path, 0, 0, bgObj.width, bgObj.height);
  301. } else {
  302. _app.log('背景没有图片路径');
  303. if (bgObj.backgroundColor) {
  304. _app.log('背景有背景颜色:' + bgObj.backgroundColor);
  305. Context.setFillStyle(bgObj.backgroundColor);
  306. Context.fillRect(0, 0, bgObj.width, bgObj.height);
  307. } else {
  308. _app.log('背景没有背景颜色');
  309. }
  310. }
  311. if (imagesArray && imagesArray.length > 0)
  312. drawImage(Context, imagesArray);
  313. if (setDraw && typeof(setDraw) == 'function') setDraw(params);
  314. if (textArray && textArray.length > 0)
  315. drawText(Context, textArray, bgObj);
  316. if (qrCodeArray && qrCodeArray.length > 0) {
  317. for (let i = 0; i < qrCodeArray.length; i++) {
  318. drawQrCode(Context, qrCodeArray[i]);
  319. }
  320. }
  321. if (drawArray && drawArray.length > 0) {
  322. for (let i = 0; i < drawArray.length; i++) {
  323. const drawArrayItem = drawArray[i];
  324. _app.log('绘制可控层级序列, drawArrayItem:' + JSON.stringify(drawArrayItem));
  325. switch (drawArrayItem.type) {
  326. case 'image':
  327. _app.log('绘制可控层级序列, 绘制图片');
  328. drawImage(Context, drawArrayItem);
  329. break;
  330. case 'text':
  331. _app.log('绘制可控层级序列, 绘制文本');
  332. drawText(Context, drawArrayItem, bgObj);
  333. break;
  334. case 'qrcode':
  335. _app.log('绘制可控层级序列, 绘制二维码');
  336. drawQrCode(Context, drawArrayItem);
  337. break;
  338. case 'custom':
  339. _app.log('绘制可控层级序列, 绘制自定义内容');
  340. if (drawArrayItem.setDraw && typeof drawArrayItem.setDraw === 'function')
  341. drawArrayItem.setDraw(Context);
  342. break;
  343. drawRoundStrokeRect, drawStrokeRect
  344. case 'fillRect':
  345. _app.log('绘制可控层级序列, 绘制填充直角矩形');
  346. drawFillRect(Context, drawArrayItem);
  347. break;
  348. case 'strokeRect':
  349. _app.log('绘制可控层级序列, 绘制线条直角矩形');
  350. drawStrokeRect(Context, drawArrayItem);
  351. break;
  352. case 'roundStrokeRect':
  353. _app.log('绘制可控层级序列, 绘制线条圆角矩形');
  354. drawRoundStrokeRect(Context, drawArrayItem);
  355. break;
  356. case 'roundFillRect':
  357. _app.log('绘制可控层级序列, 绘制填充圆角矩形');
  358. drawRoundFillRect(Context, drawArrayItem);
  359. break;
  360. default:
  361. _app.log('未识别的类型');
  362. break;
  363. }
  364. }
  365. }
  366. setTimeout(() => {
  367. _app.log('准备执行draw方法')
  368. _app.log('Context:' + Context);
  369. if (draw === false) {
  370. _app.log('draw属性为false,请自行调用canvas实例的draw方法');
  371. rs();
  372. return;
  373. }
  374. const fn = function() {
  375. let setObj = setCanvasToTempFilePath || {};
  376. if (setObj && typeof(setObj) == 'function')
  377. setObj = setCanvasToTempFilePath(bgObj, type);
  378. let canvasToTempFilePathFn;
  379. const dpr = uni.getSystemInfoSync().pixelRatio;
  380. const data = {
  381. // 注释的设置使用uni自己的默认值更为稳定
  382. // x: 0,
  383. // y: 0,
  384. // width: Number(bgObj.width),
  385. // height: Number(bgObj.height),
  386. // destWidth: Number(bgObj.width) * dpr,
  387. // destHeight: Number(bgObj.height) * dpr,
  388. quality: .8,
  389. fileType: 'jpg',
  390. ...setObj,
  391. canvasId: posterCanvasId,
  392. };
  393. if (canvas2image === false) {
  394. return rs({
  395. setCanvasToTempFilePath: data
  396. });
  397. }
  398. _app.log('canvasToTempFilePath的data对象:' + JSON.stringify(data));
  399. canvasToTempFilePathFn = function() {
  400. const toTempFilePathObj = { //输出为图片
  401. ...data,
  402. success(res) {
  403. rs({
  404. ...res,
  405. setCanvasToTempFilePath: data
  406. });
  407. },
  408. fail(err) {
  409. _app.hideLoading();
  410. _app.log('输出图片失败');
  411. _app.log('输出图片失败:' + JSON.stringify(err));
  412. rj('输出图片失败:' + JSON.stringify(err))
  413. }
  414. }
  415. uni.canvasToTempFilePath(toTempFilePathObj, _this || null);
  416. }
  417. let delayTime = 0;
  418. if (qrCodeArray) {
  419. qrCodeArray.forEach(item => {
  420. if (item.text) {
  421. delayTime += Number(item.text.length);
  422. }
  423. })
  424. }
  425. if (imagesArray) {
  426. imagesArray.forEach(() => {
  427. delayTime += delayTimeScale;
  428. })
  429. }
  430. if (textArray) {
  431. textArray.forEach(() => {
  432. delayTime += delayTimeScale;
  433. })
  434. }
  435. if (drawArray) {
  436. drawArray.forEach(item => {
  437. switch (item.type) {
  438. case 'text':
  439. if (item.text) {
  440. delayTime += item.text.length;
  441. }
  442. break;
  443. case 'qrcode':
  444. if (item.text) {
  445. delayTime += item.text.length * 2;
  446. }
  447. break;
  448. default:
  449. delayTime += delayTimeScale;
  450. break;
  451. }
  452. })
  453. }
  454. _app.log('延时系数:' + delayTimeScale);
  455. _app.log('总计延时:' + delayTime);
  456. setTimeout(canvasToTempFilePathFn, delayTime);
  457. }
  458. Context.draw((typeof(reserve) == 'boolean' ? reserve : false), fn);
  459. }, drawDelayTime);
  460. } catch (e) {
  461. //TODO handle the exception
  462. _app.hideLoading();
  463. rj(e);
  464. }
  465. });
  466. }
  467. // export
  468. function drawFillRect(Context, drawArrayItem = {}) { //填充矩形
  469. _app.log('进入绘制填充直角矩形方法, drawArrayItem:' + JSON.stringify(drawArrayItem));
  470. Context.setFillStyle(drawArrayItem.backgroundColor || 'black');
  471. Context.setGlobalAlpha(drawArrayItem.alpha || 1);
  472. Context.fillRect(drawArrayItem.dx || 0, drawArrayItem.dy || 0, drawArrayItem.width || 0, drawArrayItem.height || 0);
  473. Context.setGlobalAlpha(1);
  474. }
  475. // export
  476. function drawStrokeRect(Context, drawArrayItem = {}) { //线条矩形
  477. Context.setStrokeStyle(drawArrayItem.color || 'black');
  478. Context.setLineWidth(drawArrayItem.lineWidth || 1);
  479. Context.strokeRect(drawArrayItem.dx, drawArrayItem.dy, drawArrayItem.width, drawArrayItem.height);
  480. }
  481. // export
  482. function drawRoundStrokeRect(Context, drawArrayItem = {}) {
  483. let {
  484. dx,
  485. dy,
  486. width,
  487. height,
  488. r,
  489. lineWidth,
  490. color
  491. } = drawArrayItem;
  492. r = r || width * .1;
  493. if (width < 2 * r) {
  494. r = width / 2;
  495. }
  496. if (width < 2 * r) {
  497. r = width / 2;
  498. }
  499. Context.beginPath();
  500. Context.arc(dx + r, dy + r, r, 1 * Math.PI, 1.5 * Math.PI);
  501. Context.lineTo(dx + width - r, dy);
  502. Context.arc(dx + width - r, dy + r, r, 1.5 * Math.PI, 0);
  503. Context.lineTo(dx + width, dy + height - r);
  504. Context.arc(dx + width - r, dy + height - r, r, 0, .5 * Math.PI);
  505. Context.lineTo(dx + r, dy + height);
  506. Context.arc(dx + r, dy + height - r, r, .5 * Math.PI, 1 * Math.PI);
  507. Context.lineTo(dx, dy + r);
  508. Context.closePath();
  509. Context.setLineWidth(lineWidth || 1);
  510. Context.setStrokeStyle(color || 'black');
  511. Context.stroke();
  512. }
  513. // export
  514. function drawRoundFillRect(Context, drawArrayItem = {}) {
  515. let {
  516. dx,
  517. dy,
  518. width,
  519. height,
  520. r,
  521. backgroundColor
  522. } = drawArrayItem;
  523. r = r || width * .1;
  524. if (width < 2 * r) {
  525. r = width / 2;
  526. }
  527. if (width < 2 * r) {
  528. r = width / 2;
  529. }
  530. Context.beginPath();
  531. Context.arc(dx + r, dy + r, r, 1 * Math.PI, 1.5 * Math.PI);
  532. Context.lineTo(dx + width - r, dy);
  533. Context.arc(dx + width - r, dy + r, r, 1.5 * Math.PI, 0);
  534. Context.lineTo(dx + width, dy + height - r);
  535. Context.arc(dx + width - r, dy + height - r, r, 0, .5 * Math.PI);
  536. Context.lineTo(dx + r, dy + height);
  537. Context.arc(dx + r, dy + height - r, r, .5 * Math.PI, 1 * Math.PI);
  538. Context.lineTo(dx, dy + r);
  539. Context.closePath();
  540. Context.setFillStyle(backgroundColor);
  541. Context.fill();
  542. }
  543. // export
  544. function setText(Context, texts, bgObj) { // 设置文本数据
  545. _app.log('进入设置文字方法, texts:' + JSON.stringify(texts));
  546. if (texts && _app.isArray(texts)) {
  547. _app.log('texts是数组');
  548. if (texts.length > 0) {
  549. for (let i = 0; i < texts.length; i++) {
  550. _app.log('字符串信息-初始化之前:' + JSON.stringify(texts[i]));
  551. texts[i] = setTextFn(Context, texts[i], bgObj);
  552. }
  553. }
  554. } else {
  555. _app.log('texts是对象');
  556. texts = setTextFn(Context, texts, bgObj);
  557. _app.log('返回texts:' + JSON.stringify(texts));
  558. return texts;
  559. }
  560. }
  561. function setTextFn(Context, textItem, bgObj) {
  562. _app.log('进入设置文字方法, textItem:' + JSON.stringify(textItem));
  563. if (_app.isNotNull_string(textItem.text)) {
  564. textItem.text = String(textItem.text);
  565. textItem.alpha = textItem.alpha !== undefined ? Number(textItem.alpha) : 1;
  566. textItem.color = textItem.color || 'black';
  567. textItem.size = textItem.size !== undefined ? Number(textItem.size) : 10;
  568. textItem.textAlign = textItem.textAlign || 'left';
  569. textItem.textBaseline = textItem.textBaseline || 'middle';
  570. textItem.dx = Number(textItem.dx) || 0;
  571. textItem.dy = Number(textItem.dy) || 0;
  572. textItem.size = Math.ceil(Number(textItem.size));
  573. _app.log('字符串信息-初始化默认值后:' + JSON.stringify(textItem));
  574. let textLength = countTextLength(Context, {
  575. text: textItem.text,
  576. size: textItem.size
  577. });
  578. _app.log('字符串信息-初始化时的文本长度:' + textLength);
  579. let infoCallBackObj = {};
  580. if (textItem.infoCallBack && typeof(textItem.infoCallBack) === 'function') {
  581. infoCallBackObj = textItem.infoCallBack(textLength);
  582. }
  583. if (infoCallBackObj.size)
  584. textLength = countTextLength(Context, {
  585. text: textItem.text,
  586. size: textItem.size
  587. });
  588. textItem = {
  589. ...textItem,
  590. ...infoCallBackObj,
  591. textLength,
  592. }
  593. _app.log('字符串信息-infoCallBack后:' + JSON.stringify(textItem));
  594. }
  595. return textItem;
  596. }
  597. function setLineFeed(Context, textItem, bgObj) {
  598. if (textItem.text && textItem.lineFeed) {
  599. _app.log('设置换行')
  600. let lineNum = -1,
  601. maxWidth = bgObj.width,
  602. lineHeight = textItem.size,
  603. dx = textItem.dx;
  604. if (_app.isObject(textItem.lineFeed)) {
  605. const lineFeed = textItem.lineFeed;
  606. lineNum = (lineFeed.lineNum !== undefined && typeof(lineFeed.lineNum) === 'number') && lineFeed
  607. .lineNum >= 0 ?
  608. lineFeed.lineNum : lineNum;
  609. maxWidth = (lineFeed.maxWidth !== undefined && typeof(lineFeed.maxWidth) === 'number') ? lineFeed
  610. .maxWidth :
  611. maxWidth;
  612. lineHeight = (lineFeed.lineHeight !== undefined && typeof(lineFeed.lineHeight) === 'number') ? lineFeed
  613. .lineHeight :
  614. lineHeight;
  615. dx = (lineFeed.dx !== undefined && typeof(lineFeed.dx) === 'number') ? lineFeed.dx : dx;
  616. }
  617. _app.lineFeedTags.forEach(i => {
  618. textItem.text = textItem.text.split(i).join(_app.tagetLineFeedTag);
  619. })
  620. const chr = (textItem.text).split("");
  621. let temp = "";
  622. const row = [];
  623. //循环出几行文字组成数组
  624. for (let a = 0, len = chr.length; a < len; a++) {
  625. if (chr[a] === _app.tagetLineFeedTag) {
  626. row.push(temp);
  627. temp = chr[++a];
  628. continue;
  629. }
  630. if (countTextLength(Context, {
  631. text: temp,
  632. size: textItem.size
  633. }) <= maxWidth && countTextLength(Context, {
  634. text: (temp + chr[a]),
  635. size: textItem.size
  636. }) <= maxWidth) {
  637. temp += chr[a];
  638. if (a == (chr.length - 1)) {
  639. row.push(temp);
  640. }
  641. } else {
  642. row.push(temp);
  643. temp = chr[a];
  644. if (a == chr.length - 1) row.push(chr[a]);
  645. }
  646. }
  647. _app.log('循环出的文本数组:' + JSON.stringify(row));
  648. //只显示几行 变量间距lineHeight 变量行数lineNum
  649. let allNum = (lineNum >= 0 && lineNum < row.length) ? lineNum : row.length;
  650. const newArr = [];
  651. for (let i = 0; i < allNum; i++) {
  652. let str = row[i];
  653. if (i == (allNum - 1) && allNum < row.length && row.length > 1) {
  654. if (countTextLength(Context, {
  655. text: str,
  656. size: textItem.size
  657. }) > (maxWidth - textItem.size) * .9) {
  658. str = str.substring(0, str.length - 1) + '...';
  659. }
  660. }
  661. const obj = {
  662. ...textItem,
  663. text: str,
  664. dx: i === 0 ? textItem.dx : (dx >= 0 ? dx : textItem.dx),
  665. dy: textItem.dy + (i * lineHeight),
  666. textLength: countTextLength(Context, {
  667. text: str,
  668. size: textItem.size
  669. })
  670. };
  671. _app.log('重新组成的文本对象:' + JSON.stringify(obj));
  672. newArr.push(obj);
  673. }
  674. _app.log('newArr: -----', JSON.stringify(newArr))
  675. const result = newArr.length > 1 ? newArr : newArr[0];
  676. return result
  677. }
  678. return textItem;
  679. }
  680. function countTextLength(Context, obj) {
  681. _app.log('计算文字长度, obj:' + JSON.stringify(obj));
  682. const {
  683. text,
  684. size
  685. } = obj;
  686. Context.setFontSize(size);
  687. let textLength;
  688. try {
  689. textLength = Context.measureText(text); // 官方文档说 App端自定义组件编译模式暂时不可用measureText方法
  690. } catch (e) {
  691. //TODO handle the exception
  692. textLength = {};
  693. }
  694. _app.log('measureText计算文字长度, textLength:' + JSON.stringify(textLength));
  695. textLength = textLength && textLength.width ? textLength.width : 0;
  696. if (!textLength) {
  697. let l = 0;
  698. for (let j = 0; j < text.length; j++) {
  699. let t = text.substr(j, 1);
  700. const countL = countStrLength(t);
  701. _app.log('计算文字宽度系数:' + countL);
  702. l += countL;
  703. }
  704. _app.log('文字宽度总系数:' + l);
  705. textLength = l * size;
  706. }
  707. return textLength;
  708. }
  709. //计算字符长度系数
  710. function countStrLength(t) {
  711. let l;
  712. if (/a/.test(t)) {
  713. l = 0.552734375
  714. } else if (/b/.test(t)) {
  715. l = 0.638671875
  716. } else if (/c/.test(t)) {
  717. l = 0.50146484375
  718. } else if (/d/.test(t)) {
  719. l = 0.6396484375
  720. } else if (/e/.test(t)) {
  721. l = 0.5673828125
  722. } else if (/f/.test(t)) {
  723. l = 0.3466796875
  724. } else if (/g/.test(t)) {
  725. l = 0.6396484375
  726. } else if (/h/.test(t)) {
  727. l = 0.61572265625
  728. } else if (/i/.test(t)) {
  729. l = 0.26611328125
  730. } else if (/j/.test(t)) {
  731. l = 0.26708984375
  732. } else if (/k/.test(t)) {
  733. l = 0.54443359375
  734. } else if (/l/.test(t)) {
  735. l = 0.26611328125
  736. } else if (/m/.test(t)) {
  737. l = 0.93701171875
  738. } else if (/n/.test(t)) {
  739. l = 0.6162109375
  740. } else if (/o/.test(t)) {
  741. l = 0.6357421875
  742. } else if (/p/.test(t)) {
  743. l = 0.638671875
  744. } else if (/q/.test(t)) {
  745. l = 0.6396484375
  746. } else if (/r/.test(t)) {
  747. l = 0.3818359375
  748. } else if (/s/.test(t)) {
  749. l = 0.462890625
  750. } else if (/t/.test(t)) {
  751. l = 0.37255859375
  752. } else if (/u/.test(t)) {
  753. l = 0.6162109375
  754. } else if (/v/.test(t)) {
  755. l = 0.52490234375
  756. } else if (/w/.test(t)) {
  757. l = 0.78955078125
  758. } else if (/x/.test(t)) {
  759. l = 0.5068359375
  760. } else if (/y/.test(t)) {
  761. l = 0.529296875
  762. } else if (/z/.test(t)) {
  763. l = 0.49169921875
  764. } else if (/A/.test(t)) {
  765. l = 0.70361328125
  766. } else if (/B/.test(t)) {
  767. l = 0.62744140625
  768. } else if (/C/.test(t)) {
  769. l = 0.6689453125
  770. } else if (/D/.test(t)) {
  771. l = 0.76171875
  772. } else if (/E/.test(t)) {
  773. l = 0.5498046875
  774. } else if (/F/.test(t)) {
  775. l = 0.53125
  776. } else if (/G/.test(t)) {
  777. l = 0.74365234375
  778. } else if (/H/.test(t)) {
  779. l = 0.7734375
  780. } else if (/I/.test(t)) {
  781. l = 0.2939453125
  782. } else if (/J/.test(t)) {
  783. l = 0.39599609375
  784. } else if (/K/.test(t)) {
  785. l = 0.634765625
  786. } else if (/L/.test(t)) {
  787. l = 0.51318359375
  788. } else if (/M/.test(t)) {
  789. l = 0.97705078125
  790. } else if (/N/.test(t)) {
  791. l = 0.81298828125
  792. } else if (/O/.test(t)) {
  793. l = 0.81494140625
  794. } else if (/P/.test(t)) {
  795. l = 0.61181640625
  796. } else if (/Q/.test(t)) {
  797. l = 0.81494140625
  798. } else if (/R/.test(t)) {
  799. l = 0.65283203125
  800. } else if (/S/.test(t)) {
  801. l = 0.5771484375
  802. } else if (/T/.test(t)) {
  803. l = 0.5732421875
  804. } else if (/U/.test(t)) {
  805. l = 0.74658203125
  806. } else if (/V/.test(t)) {
  807. l = 0.67626953125
  808. } else if (/W/.test(t)) {
  809. l = 1.017578125
  810. } else if (/X/.test(t)) {
  811. l = 0.64501953125
  812. } else if (/Y/.test(t)) {
  813. l = 0.603515625
  814. } else if (/Z/.test(t)) {
  815. l = 0.6201171875
  816. } else if (/[0-9]/.test(t)) {
  817. l = 0.58642578125
  818. } else if (/[\u4e00-\u9fa5]/.test(t)) {
  819. l = 1
  820. } else if (/ /.test(t)) {
  821. l = 0.2958984375
  822. } else if (/\`/.test(t)) {
  823. l = 0.294921875
  824. } else if (/\~/.test(t)) {
  825. l = 0.74169921875
  826. } else if (/\!/.test(t)) {
  827. l = 0.3125
  828. } else if (/\@/.test(t)) {
  829. l = 1.03125
  830. } else if (/\#/.test(t)) {
  831. l = 0.63818359375
  832. } else if (/\$/.test(t)) {
  833. l = 0.58642578125
  834. } else if (/\%/.test(t)) {
  835. l = 0.8896484375
  836. } else if (/\^/.test(t)) {
  837. l = 0.74169921875
  838. } else if (/\&/.test(t)) {
  839. l = 0.8701171875
  840. } else if (/\*/.test(t)) {
  841. l = 0.455078125
  842. } else if (/\(/.test(t)) {
  843. l = 0.333984375
  844. } else if (/\)/.test(t)) {
  845. l = 0.333984375
  846. } else if (/\_/.test(t)) {
  847. l = 0.4482421875
  848. } else if (/\-/.test(t)) {
  849. l = 0.4326171875
  850. } else if (/\+/.test(t)) {
  851. l = 0.74169921875
  852. } else if (/\=/.test(t)) {
  853. l = 0.74169921875
  854. } else if (/\|/.test(t)) {
  855. l = 0.26904296875
  856. } else if (/\\/.test(t)) {
  857. l = 0.416015625
  858. } else if (/\[/.test(t)) {
  859. l = 0.333984375
  860. } else if (/\]/.test(t)) {
  861. l = 0.333984375
  862. } else if (/\;/.test(t)) {
  863. l = 0.24072265625
  864. } else if (/\'/.test(t)) {
  865. l = 0.25634765625
  866. } else if (/\,/.test(t)) {
  867. l = 0.24072265625
  868. } else if (/\./.test(t)) {
  869. l = 0.24072265625
  870. } else if (/\//.test(t)) {
  871. l = 0.42724609375
  872. } else if (/\{/.test(t)) {
  873. l = 0.333984375
  874. } else if (/\}/.test(t)) {
  875. l = 0.333984375
  876. } else if (/\:/.test(t)) {
  877. l = 0.24072265625
  878. } else if (/\"/.test(t)) {
  879. l = 0.435546875
  880. } else if (/\</.test(t)) {
  881. l = 0.74169921875
  882. } else if (/\>/.test(t)) {
  883. l = 0.74169921875
  884. } else if (/\?/.test(t)) {
  885. l = 0.48291015625
  886. } else {
  887. l = 1
  888. }
  889. return l;
  890. }
  891. // export
  892. function setImage(images) { // 设置图片数据
  893. _app.log('进入设置图片数据方法');
  894. return new Promise(async (resolve, reject) => {
  895. try {
  896. if (images && _app.isArray(images)) {
  897. _app.log('images是一个数组');
  898. for (let i = 0; i < images.length; i++) {
  899. _app.log('设置图片数据循环中:' + i);
  900. images[i] = await setImageFn(images[i]);
  901. }
  902. } else {
  903. _app.log('images是一个对象');
  904. images = await setImageFn(images);
  905. }
  906. resolve(images);
  907. } catch (e) {
  908. _app.hideLoading()
  909. //TODO handle the exception
  910. reject(e);
  911. }
  912. })
  913. }
  914. function base64ToPathFn(path) {
  915. var reg =
  916. /^\s*data:([a-z]+\/[a-z0-9-+.]+(;[a-z-]+=[a-z0-9-]+)?)?(;base64)?,([a-z0-9!$&',()*+;=\-._~:@\/?%\s]*?)\s*$/i;
  917. if (!reg.test(path)) {
  918. return Promise.resolve(path);
  919. }
  920. return base64ToPath(path);
  921. }
  922. function setImageFn(image) {
  923. return new Promise(async (resolve, reject) => {
  924. try {
  925. if (image.url) {
  926. image.url = (await base64ToPathFn(image.url));
  927. let imgUrl = image.url;
  928. const oldImgUrl = imgUrl;
  929. imgUrl = await _app.downloadFile_PromiseFc(imgUrl);
  930. image.url = imgUrl;
  931. const hasinfoCallBack = image.infoCallBack && typeof(image.infoCallBack) === 'function';
  932. let imageInfo = {};
  933. imageInfo = await _app.getImageInfo_PromiseFc(oldImgUrl);
  934. if (hasinfoCallBack) {
  935. image = {
  936. ...image,
  937. ...image.infoCallBack(imageInfo)
  938. };
  939. }
  940. image.dx = Number(image.dx) || 0;
  941. image.dy = Number(image.dy) || 0;
  942. image.dWidth = Number(image.dWidth || imageInfo.width);
  943. image.dHeight = Number(image.dHeight || imageInfo.height);
  944. image = {
  945. ...image,
  946. imageInfo
  947. }
  948. }
  949. resolve(image);
  950. } catch (e) {
  951. _app.hideLoading();
  952. reject(e)
  953. //TODO handle the exception
  954. }
  955. })
  956. }
  957. // export
  958. function drawText(Context, textArray, bgObj) { // 先遍历换行再绘制
  959. if (!_app.isArray(textArray)) {
  960. _app.log('遍历文本方法, 不是数组');
  961. textArray = [textArray];
  962. } else {
  963. _app.log('遍历文本方法, 是数组');
  964. }
  965. _app.log('遍历文本方法, textArray:' + JSON.stringify(textArray));
  966. const newArr = textArray;
  967. // if (textArray && textArray.length > 0) {
  968. // for (let j = 0; j < textArray.length; j++) {
  969. // const textItem = textArray[j];
  970. // if (textItem.text && textItem.lineFeed) {
  971. // let lineNum = -1,
  972. // maxWidth = bgObj.width,
  973. // lineHeight = textItem.size,
  974. // dx = textItem.dx;
  975. // if (_app.isObject(textItem.lineFeed)) {
  976. // const lineFeed = textItem.lineFeed;
  977. // lineNum = (lineFeed.lineNum !== undefined && typeof(lineFeed.lineNum) === 'number') && lineFeed
  978. // .lineNum >= 0 ?
  979. // lineFeed.lineNum : lineNum;
  980. // maxWidth = (lineFeed.maxWidth !== undefined && typeof(lineFeed.maxWidth) === 'number') ? lineFeed
  981. // .maxWidth :
  982. // maxWidth;
  983. // lineHeight = (lineFeed.lineHeight !== undefined && typeof(lineFeed.lineHeight) === 'number') ?
  984. // lineFeed.lineHeight :
  985. // lineHeight;
  986. // dx = (lineFeed.dx !== undefined && typeof(lineFeed.dx) === 'number') ? lineFeed.dx : dx;
  987. // }
  988. // const chr = (textItem.text).split("");
  989. // let temp = "";
  990. // const row = [];
  991. // //循环出几行文字组成数组
  992. // for (let a = 0, len = chr.length; a < len; a++) {
  993. // if (countTextLength(Context, {
  994. // text: temp,
  995. // size: textItem.size
  996. // }) <= maxWidth && countTextLength(Context, {
  997. // text: (temp + chr[a]),
  998. // size: textItem.size
  999. // }) <= maxWidth) {
  1000. // temp += chr[a];
  1001. // if (a == (chr.length - 1)) {
  1002. // row.push(temp);
  1003. // }
  1004. // } else {
  1005. // row.push(temp);
  1006. // temp = chr[a];
  1007. // if (a == chr.length - 1) row.push(chr[a]);
  1008. // }
  1009. // }
  1010. // _app.log('循环出的文本数组:' + JSON.stringify(row));
  1011. // //只显示几行 变量间距lineHeight 变量行数lineNum
  1012. // let allNum = (lineNum >= 0 && lineNum < row.length) ? lineNum : row.length;
  1013. // for (let i = 0; i < allNum; i++) {
  1014. // let str = row[i];
  1015. // if (i == (allNum - 1) && allNum < row.length) {
  1016. // str = str.substring(0, str.length - 1) + '...';
  1017. // }
  1018. // const obj = {
  1019. // ...textItem,
  1020. // text: str,
  1021. // dx: i === 0 ? textItem.dx : (dx >= 0 ? dx : textItem.dx),
  1022. // dy: textItem.dy + (i * lineHeight),
  1023. // textLength: countTextLength(Context, {
  1024. // text: str,
  1025. // size: textItem.size
  1026. // })
  1027. // };
  1028. // _app.log('重新组成的文本对象:' + JSON.stringify(obj));
  1029. // newArr.push(obj);
  1030. // }
  1031. // } else {
  1032. // newArr.push(textItem);
  1033. // }
  1034. // }
  1035. // }
  1036. _app.log('绘制文本新数组:' + JSON.stringify(newArr));
  1037. drawTexts(Context, newArr);
  1038. }
  1039. function setFont(textItem = {}) {
  1040. if (textItem.font && typeof(textItem.font) === 'string') {
  1041. _app.log(textItem.font)
  1042. return textItem.font;
  1043. } else {
  1044. let fontStyle = 'normal';
  1045. let fontVariant = 'normal';
  1046. let fontWeight = 'normal';
  1047. let fontSize = textItem.size || 10;
  1048. let fontFamily = 'sans-serif';
  1049. fontSize = Math.ceil(Number(fontSize));
  1050. if (textItem.fontStyle && typeof(textItem.fontStyle) === 'string')
  1051. fontStyle = textItem.fontStyle.trim();
  1052. if (textItem.fontVariant && typeof(textItem.fontVariant) === 'string')
  1053. fontVariant = textItem.fontVariant.trim();
  1054. if (textItem.fontWeight && (typeof(textItem.fontWeight) === 'string' || typeof(textItem.fontWeight) ===
  1055. 'number'))
  1056. fontWeight = textItem.fontWeight.trim();
  1057. if (textItem.fontFamily && typeof(textItem.fontFamily) === 'string')
  1058. fontFamily = textItem.fontFamily.trim();
  1059. return fontStyle + ' ' +
  1060. fontVariant + ' ' +
  1061. fontWeight + ' ' +
  1062. fontSize + 'px' + ' ' +
  1063. fontFamily;
  1064. }
  1065. }
  1066. function drawTexts(Context, texts) { // 绘制文本
  1067. _app.log('准备绘制文本方法, texts:' + JSON.stringify(texts));
  1068. if (texts && _app.isArray(texts)) {
  1069. _app.log('准备绘制文本方法, 是数组');
  1070. if (texts.length > 0) {
  1071. for (let i = 0; i < texts.length; i++) {
  1072. drawTextFn(Context, texts[i]);
  1073. }
  1074. }
  1075. } else {
  1076. _app.log('准备绘制文本方法, 不是数组');
  1077. drawTextFn(Context, texts);
  1078. }
  1079. }
  1080. function drawTextFn(Context, textItem) {
  1081. _app.log('进入绘制文本方法, textItem:' + JSON.stringify(textItem));
  1082. if (textItem && _app.isObject(textItem) && textItem.text) {
  1083. Context.font = setFont(textItem);
  1084. Context.setFillStyle(textItem.color);
  1085. Context.setGlobalAlpha(textItem.alpha);
  1086. Context.setTextAlign(textItem.textAlign);
  1087. Context.setTextBaseline(textItem.textBaseline);
  1088. Context.fillText(textItem.text, textItem.dx, textItem.dy);
  1089. if (textItem.lineThrough && _app.isObject(textItem.lineThrough)) {
  1090. _app.log('有删除线');
  1091. let lineThrough = textItem.lineThrough;
  1092. lineThrough.alpha = lineThrough.alpha !== undefined ? lineThrough.alpha : textItem.alpha;
  1093. lineThrough.style = lineThrough.style || textItem.color;
  1094. lineThrough.width = lineThrough.width !== undefined ? lineThrough.width : textItem.size / 10;
  1095. lineThrough.cap = lineThrough.cap !== undefined ? lineThrough.cap : 'butt';
  1096. _app.log('删除线对象:' + JSON.stringify(lineThrough));
  1097. Context.setGlobalAlpha(lineThrough.alpha);
  1098. Context.setStrokeStyle(lineThrough.style);
  1099. Context.setLineWidth(lineThrough.width);
  1100. Context.setLineCap(lineThrough.cap);
  1101. let mx, my;
  1102. switch (textItem.textAlign) {
  1103. case 'left':
  1104. mx = textItem.dx;
  1105. break;
  1106. case 'center':
  1107. mx = textItem.dx - (textItem.textLength) / 2;
  1108. break;
  1109. default:
  1110. mx = textItem.dx - (textItem.textLength);
  1111. break;
  1112. }
  1113. switch (textItem.textBaseline) {
  1114. case 'top':
  1115. my = textItem.dy + (textItem.size * .5);
  1116. break;
  1117. case 'middle':
  1118. my = textItem.dy;
  1119. break;
  1120. default:
  1121. my = textItem.dy - (textItem.size * .5);
  1122. break;
  1123. }
  1124. Context.beginPath();
  1125. Context.moveTo(mx, my);
  1126. Context.lineTo(mx + textItem.textLength, my);
  1127. Context.stroke();
  1128. Context.closePath();
  1129. _app.log('删除线完毕');
  1130. }
  1131. Context.setGlobalAlpha(1);
  1132. Context.font = '10px sans-serif';
  1133. }
  1134. }
  1135. // export
  1136. function drawImage(Context, images) { // 绘制图片
  1137. _app.log('判断图片数据类型:' + JSON.stringify(images))
  1138. if (images && _app.isArray(images)) {
  1139. if (images.length > 0) {
  1140. for (let i = 0; i < images.length; i++) {
  1141. readyDrawImageFn(Context, images[i]);
  1142. }
  1143. }
  1144. } else {
  1145. readyDrawImageFn(Context, images);
  1146. }
  1147. }
  1148. function readyDrawImageFn(Context, img) {
  1149. _app.log('判断绘制图片形状, img:' + JSON.stringify(img));
  1150. if (img.url) {
  1151. if (img.circleSet) {
  1152. drawCircleImage(Context, img);
  1153. } else if (img.roundRectSet) {
  1154. drawRoundRectImage(Context, img);
  1155. } else {
  1156. drawImageFn(Context, img);
  1157. }
  1158. }
  1159. }
  1160. const drawImageModes = {
  1161. scaleToFill(Context, img) {
  1162. _app.log('准备绘制mode为scaleToFill的图片')
  1163. Context.drawImage(img.url, Number(img.dx || 0), Number(img.dy || 0),
  1164. Number(img.dWidth) || false, Number(img.dHeight) || false);
  1165. _app.log('mode为scaleToFill的图片绘制完毕')
  1166. },
  1167. aspectFit(Context, img) {
  1168. _app.log('准备绘制mode为aspectFit的图片')
  1169. const {
  1170. imageInfo,
  1171. dWidth,
  1172. dHeight
  1173. } = img;
  1174. const {
  1175. height,
  1176. width
  1177. } = imageInfo;
  1178. let drawWidth = dWidth;
  1179. let drawHeight = height / width * drawWidth;
  1180. if (drawHeight < dHeight) {
  1181. const diffHeight = (Number(dHeight) - Number(drawHeight)) / Number(dHeight) * height;
  1182. img.dy = Number(img.dy) + diffHeight / 2;
  1183. } else {
  1184. drawHeight = dHeight;
  1185. drawWidth = width / height * drawHeight;
  1186. const diffWidth = (Number(dWidth) - Number(drawWidth)) / Number(dWidth) * width;
  1187. img.dx = Number(img.dx) + diffWidth / 2;
  1188. }
  1189. Context.drawImage(img.url, 0, 0, width, height, img.dx, img.dy, drawWidth, drawHeight);
  1190. _app.log('mode为aspectFit的图片绘制完毕')
  1191. },
  1192. aspectFill(Context, img) {
  1193. const dpr = uni.getSystemInfoSync().pixelRatio;
  1194. _app.log('准备绘制mode为aspectFill的图片')
  1195. const {
  1196. imageInfo,
  1197. dWidth,
  1198. dHeight
  1199. } = img;
  1200. const {
  1201. height,
  1202. width
  1203. } = imageInfo;
  1204. let sx = 0,
  1205. sy = 0,
  1206. sWidth = (width),
  1207. sHeight = (height);
  1208. let drawWidth = dWidth;
  1209. let drawHeight = height / width * drawWidth;
  1210. if (drawHeight < dHeight) {
  1211. _app.log('绘制高度 小于 预定高度')
  1212. drawHeight = dHeight;
  1213. drawWidth = width / height * drawHeight;
  1214. const diffWidth = ((Number(drawWidth) - Number(dWidth)) / Number(drawWidth)) * width;
  1215. sx = diffWidth / 2;
  1216. sWidth = width - diffWidth;
  1217. } else {
  1218. const diffHeight = ((Number(drawHeight) - Number(dHeight)) / Number(drawHeight)) * height;
  1219. sy = diffHeight / 2;
  1220. sHeight = (height - diffHeight);
  1221. }
  1222. _app.log(
  1223. `aspectFill 最终绘制: sx: ${sx}, sy: ${sy}, sWidth: ${sWidth}, sHeight: ${sHeight}, dx: ${img.dx}, dy: ${img.dy}, dWidth: ${dWidth}, dHeight: ${dHeight}`
  1224. )
  1225. Context.drawImage(img.url, sx, sy, sWidth, sHeight, img.dx, img.dy, dWidth, dHeight);
  1226. _app.log('mode为aspectFill的图片绘制完毕')
  1227. }
  1228. }
  1229. function drawImageFn(Context, img) {
  1230. _app.log('进入绘制默认图片方法, img:' + JSON.stringify(img));
  1231. if (img.url) {
  1232. const hasAlpha = !_app.isUndef(img.alpha);
  1233. img.alpha = Number(!_app.isUndef(img.alpha) ? img.alpha : 1);
  1234. Context.setGlobalAlpha(img.alpha);
  1235. _app.log('绘制默认图片方法, 有url');
  1236. if (img.dHeight === undefined) img.dHeight = img.imageInfo.height;
  1237. if (img.dWidth === undefined) img.dWidth = img.imageInfo.width;
  1238. const fn = drawImageModes[img.mode];
  1239. if (fn) {
  1240. fn(Context, img);
  1241. } else {
  1242. if (img.dWidth && img.dHeight && img.sx && img.sy && img.sWidth && img.sHeight) {
  1243. _app.log('绘制默认图片方法, 绘制第一种方案');
  1244. Context.drawImage(img.url,
  1245. Number(img.sx) || false, Number(img.sy) || false,
  1246. Number(img.sWidth) || false, Number(img.sHeight) || false,
  1247. Number(img.dx || 0), Number(img.dy || 0),
  1248. Number(img.dWidth) || false, Number(img.dHeight) || false, );
  1249. } else if (img.dWidth && img.dHeight) {
  1250. _app.log('绘制默认图片方法, 绘制第二种方案');
  1251. Context.drawImage(img.url, Number(img.dx || 0), Number(img.dy || 0),
  1252. Number(img.dWidth) || false, Number(img.dHeight) || false);
  1253. } else {
  1254. _app.log('绘制默认图片方法, 绘制第三种方案');
  1255. Context.drawImage(img.url, Number(img.dx || 0), Number(img.dy || 0));
  1256. }
  1257. }
  1258. if (hasAlpha) {
  1259. Context.setGlobalAlpha(1);
  1260. }
  1261. }
  1262. _app.log('绘制默认图片方法, 绘制完毕');
  1263. }
  1264. function drawCircleImage(Context, obj) {
  1265. _app.log('进入绘制圆形图片方法, obj:' + JSON.stringify(obj));
  1266. let {
  1267. dx,
  1268. dy,
  1269. dWidth,
  1270. dHeight,
  1271. circleSet,
  1272. imageInfo
  1273. } = obj;
  1274. let x, y, r;
  1275. if (typeof circleSet === 'object') {
  1276. x = circleSet.x;
  1277. y = circleSet.y;
  1278. r = circleSet.r;
  1279. }
  1280. if (!r) {
  1281. let d;
  1282. d = dWidth > dHeight ? dHeight : dWidth;
  1283. r = d / 2;
  1284. }
  1285. x = x ? dx + x : (dx || 0) + r;
  1286. y = y ? dy + y : (dy || 0) + r;
  1287. Context.save();
  1288. Context.beginPath();
  1289. Context.arc(x, y, r, 0, 2 * Math.PI, false);
  1290. Context.closePath();
  1291. Context.setGlobalAlpha(0);
  1292. Context.fillStyle = '#FFFFFF';
  1293. Context.fill();
  1294. Context.setGlobalAlpha(1);
  1295. Context.clip();
  1296. drawImageFn(Context, obj);
  1297. _app.log('默认图片绘制完毕');
  1298. Context.restore();
  1299. }
  1300. function drawRoundRectImage(Context, obj) { // 绘制矩形
  1301. _app.log('进入绘制矩形图片方法, obj:' + JSON.stringify(obj));
  1302. Context.save();
  1303. let {
  1304. dx,
  1305. dy,
  1306. dWidth,
  1307. dHeight,
  1308. roundRectSet,
  1309. imageInfo
  1310. } = obj;
  1311. let r;
  1312. if (typeof roundRectSet === 'object') {
  1313. r = roundRectSet.r;
  1314. }
  1315. r = r || dWidth * .1;
  1316. if (dWidth < 2 * r) {
  1317. r = dWidth / 2;
  1318. }
  1319. if (dHeight < 2 * r) {
  1320. r = dHeight / 2;
  1321. }
  1322. Context.beginPath();
  1323. // Context.moveTo(dx + r, dy);
  1324. Context.arc(dx + r, dy + r, r, 1 * Math.PI, 1.5 * Math.PI);
  1325. Context.lineTo(dx + dWidth - r, dy);
  1326. Context.arc(dx + dWidth - r, dy + r, r, 1.5 * Math.PI, 0);
  1327. Context.lineTo(dx + dWidth, dy + dHeight - r);
  1328. Context.arc(dx + dWidth - r, dy + dHeight - r, r, 0, .5 * Math.PI);
  1329. Context.lineTo(dx + r, dy + dHeight);
  1330. Context.arc(dx + r, dy + dHeight - r, r, .5 * Math.PI, 1 * Math.PI);
  1331. Context.lineTo(dx, dy + r);
  1332. // Context.arcTo(dx + dWidth, dy, dx + dWidth, dy + dHeight, r);
  1333. // Context.arcTo(dx + dWidth, dy + dHeight, dx, dy + dHeight, r);
  1334. // Context.arcTo(dx, dy + dHeight, dx, dy, r);
  1335. // Context.arcTo(dx, dy, dx + dWidth, dy, r);
  1336. Context.closePath();
  1337. Context.setGlobalAlpha(0);
  1338. Context.fillStyle = '#FFFFFF';
  1339. Context.fill();
  1340. Context.setGlobalAlpha(1);
  1341. Context.clip();
  1342. drawImageFn(Context, obj);
  1343. Context.restore();
  1344. _app.log('进入绘制矩形图片方法, 绘制完毕');
  1345. }
  1346. // export
  1347. function drawQrCode(Context, qrCodeObj) { //生成二维码方法, 参考了 诗小柒 的二维码生成器代码
  1348. _app.log('进入绘制二维码方法')
  1349. let qrcodeAlgObjCache = [];
  1350. let options = {
  1351. text: String(qrCodeObj.text || '') || '', // 生成内容
  1352. size: Number(qrCodeObj.size || 0) || 200, // 二维码大小
  1353. background: String(qrCodeObj.background || '') || '#ffffff', // 背景色
  1354. foreground: String(qrCodeObj.foreground || '') || '#000000', // 前景色
  1355. pdground: String(qrCodeObj.pdground || '') || '#000000', // 定位角点颜色
  1356. correctLevel: Number(qrCodeObj.correctLevel || 0) || 3, // 容错级别
  1357. image: String(qrCodeObj.image || '') || '', // 二维码图标
  1358. imageSize: Number(qrCodeObj.imageSize || 0) || 40, // 二维码图标大小
  1359. dx: Number(qrCodeObj.dx || 0) || 0, // x轴距离
  1360. dy: Number(qrCodeObj.dy || 0) || 0 // y轴距离
  1361. }
  1362. let qrCodeAlg = null;
  1363. let d = 0;
  1364. for (var i = 0, l = qrcodeAlgObjCache.length; i < l; i++) {
  1365. d = i;
  1366. if (qrcodeAlgObjCache[i].text == options.text && qrcodeAlgObjCache[i].text.correctLevel == options
  1367. .correctLevel) {
  1368. qrCodeAlg = qrcodeAlgObjCache[i].obj;
  1369. break;
  1370. }
  1371. }
  1372. if (d == l) {
  1373. qrCodeAlg = new QRCodeAlg(options.text, options.correctLevel);
  1374. qrcodeAlgObjCache.push({
  1375. text: options.text,
  1376. correctLevel: options.correctLevel,
  1377. obj: qrCodeAlg
  1378. });
  1379. }
  1380. let getForeGround = function(config) {
  1381. let options = config.options;
  1382. if (options.pdground && (
  1383. (config.row > 1 && config.row < 5 && config.col > 1 && config.col < 5) ||
  1384. (config.row > (config.count - 6) && config.row < (config.count - 2) && config.col > 1 && config
  1385. .col < 5) ||
  1386. (config.row > 1 && config.row < 5 && config.col > (config.count - 6) && config.col < (config.count -
  1387. 2))
  1388. )) {
  1389. return options.pdground;
  1390. }
  1391. return options.foreground;
  1392. }
  1393. let count = qrCodeAlg.getModuleCount();
  1394. let ratioSize = options.size;
  1395. let ratioImgSize = options.imageSize;
  1396. //计算每个点的长宽
  1397. let tileW = (ratioSize / count).toPrecision(4);
  1398. let tileH = (ratioSize / count).toPrecision(4);
  1399. //绘制
  1400. for (let row = 0; row < count; row++) {
  1401. for (let col = 0; col < count; col++) {
  1402. let w = (Math.ceil((col + 1) * tileW) - Math.floor(col * tileW));
  1403. let h = (Math.ceil((row + 1) * tileW) - Math.floor(row * tileW));
  1404. let foreground = getForeGround({
  1405. row: row,
  1406. col: col,
  1407. count: count,
  1408. options: options
  1409. });
  1410. Context.setFillStyle(qrCodeAlg.modules[row][col] ? foreground : options.background);
  1411. Context.fillRect(options.dx + Math.round(col * tileW), options.dy + Math.round(row * tileH), w, h);
  1412. }
  1413. }
  1414. if (options.image) {
  1415. let x = options.dx + Number(((ratioSize - ratioImgSize) / 2).toFixed(2));
  1416. let y = options.dy + Number(((ratioSize - ratioImgSize) / 2).toFixed(2));
  1417. drawRoundedRect(Context, x, y, ratioImgSize, ratioImgSize, 2, 6, true, true)
  1418. Context.drawImage(options.image, x, y, ratioImgSize, ratioImgSize);
  1419. // 画圆角矩形
  1420. function drawRoundedRect(ctxi, x, y, width, height, r, lineWidth, fill, stroke) {
  1421. ctxi.setLineWidth(lineWidth);
  1422. ctxi.setFillStyle(options.background);
  1423. ctxi.setStrokeStyle(options.background);
  1424. ctxi.beginPath(); // draw top and top right corner
  1425. ctxi.moveTo(x + r, y);
  1426. ctxi.arcTo(x + width, y, x + width, y + r, r); // draw right side and bottom right corner
  1427. ctxi.arcTo(x + width, y + height, x + width - r, y + height, r); // draw bottom and bottom left corner
  1428. ctxi.arcTo(x, y + height, x, y + height - r, r); // draw left and top left corner
  1429. ctxi.arcTo(x, y, x + r, y, r);
  1430. ctxi.closePath();
  1431. if (fill) {
  1432. ctxi.fill();
  1433. }
  1434. if (stroke) {
  1435. ctxi.stroke();
  1436. }
  1437. }
  1438. }
  1439. _app.log('进入绘制二维码方法完毕')
  1440. }
  1441. function getShreUserPosterBackground(objs) { //检查背景图是否存在于本地, 若存在直接返回, 否则调用getShreUserPosterBackgroundFc方法
  1442. let {
  1443. backgroundImage,
  1444. type
  1445. } = objs;
  1446. return new Promise(async (resolve, reject) => {
  1447. try {
  1448. const savedFilePath = await getShreUserPosterBackgroundFc(objs)
  1449. resolve(savedFilePath);
  1450. } catch (e) {
  1451. _app.hideLoading();
  1452. _app.showToast('获取分享用户背景图失败:' + JSON.stringify(e));
  1453. _app.log(JSON.stringify(e));
  1454. reject(e);
  1455. }
  1456. })
  1457. }
  1458. function getPosterStorage(type) {
  1459. return _app.getStorageSync(getStorageKey(type));
  1460. }
  1461. function removePosterStorage(type) {
  1462. const ShreUserPosterBackgroundKey = getStorageKey(type);
  1463. const pbg = _app.getStorageSync(ShreUserPosterBackgroundKey);
  1464. if (pbg && pbg.path) {
  1465. _app.removeStorageSync(ShreUserPosterBackgroundKey);
  1466. }
  1467. }
  1468. function setPosterStorage(type, data) {
  1469. _app.setStorage(getStorageKey(type), data);
  1470. }
  1471. function getStorageKey(type) {
  1472. return ShreUserPosterBackgroundKey + (type || 'default');
  1473. }
  1474. function getShreUserPosterBackgroundFc(objs, upimage) { //下载并保存背景图方法
  1475. let {
  1476. backgroundImage,
  1477. type
  1478. } = objs;
  1479. _app.log('获取分享背景图, 尝试清空本地数据');
  1480. return new Promise(async (resolve, reject) => {
  1481. try {
  1482. _app.log('没有从后端获取的背景图片路径, 尝试从后端获取背景图片路径');
  1483. let image = backgroundImage ? backgroundImage : (await _app.getPosterUrl(objs));
  1484. image = (await base64ToPathFn(image));
  1485. _app.log('尝试下载并保存背景图:' + image);
  1486. const savedFilePath = await _app.downLoadAndSaveFile_PromiseFc(image);
  1487. if (savedFilePath) {
  1488. _app.log('下载并保存背景图成功:' + savedFilePath);
  1489. const imageObj = await _app.getImageInfo_PromiseFc(image);
  1490. _app.log('获取图片信息成功');
  1491. const returnObj = {
  1492. path: savedFilePath,
  1493. width: imageObj.width,
  1494. height: imageObj.height,
  1495. name: _app.fileNameInPath(image)
  1496. }
  1497. _app.log('拼接背景图信息对象成功:' + JSON.stringify(returnObj));
  1498. // #ifndef H5
  1499. setPosterStorage(type, {
  1500. ...returnObj
  1501. });
  1502. // #endif
  1503. _app.log('返回背景图信息对象');
  1504. resolve({
  1505. ...returnObj
  1506. });
  1507. } else {
  1508. reject('not find savedFilePath');
  1509. }
  1510. } catch (e) {
  1511. //TODO handle the exception
  1512. _app.hideLoading();
  1513. reject(e);
  1514. }
  1515. });
  1516. }
  1517. module.exports = {
  1518. getShreUserPosterBackground,
  1519. getSharePoster,
  1520. setText,
  1521. setImage,
  1522. drawText,
  1523. drawImage,
  1524. drawQrCode,
  1525. drawFillRect,
  1526. drawStrokeRect,
  1527. drawRoundStrokeRect,
  1528. drawRoundFillRect
  1529. }