xinyu-cross-canvas.vue 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501
  1. <template>
  2. <view>
  3. <view
  4. :style="{width: styleWidth == -1 ? rpx(750) : rpx(styleWidth),height: styleHeight == -1 ? '100vh' : rpx(styleWidth)}">
  5. <!-- #ifdef MP-ALIPAY -->
  6. <canvas type="2d" :disable-scroll="true"
  7. :style="{width: styleWidth == -1 ? rpx(750) : rpx(styleWidth),height: styleHeight == -1 ? '100vh' : rpx(styleWidth)}"
  8. @touchstart="onTouch('start', $event)" @touchmove="onTouch('move', $event)"
  9. @touchend="onTouch('end', $event)" :id="id" @ready="onCanvasInitReadyAlipay"></canvas>
  10. <!-- #endif -->
  11. <!-- #ifdef MP-WEIXIN || MP-TOUTIAO -->
  12. <canvas type="2d" :disable-scroll="true"
  13. :style="{width: styleWidth == -1 ? rpx(750) : rpx(styleWidth),height: styleHeight == -1 ? '100vh' : rpx(styleWidth)}"
  14. @touchstart="onTouch('start', $event)" @touchmove="onTouch('move', $event)"
  15. @touchend="onTouch('end', $event)" :id="id"></canvas>
  16. <!-- #endif -->
  17. <!-- #ifndef MP -->
  18. <view :renderjs_data="renderjsContextProp" :change:renderjs_data="xinyucrosscanvas.setContextPropRenderjs">
  19. </view>
  20. <view :renderjs_data="renderjsContextMethodCall"
  21. :change:renderjs_data="xinyucrosscanvas.callContextMethodRenderjs">
  22. </view>
  23. <view :renderjs_data="renderjsData" :change:renderjs_data="xinyucrosscanvas.refreshDataRenderjs"></view>
  24. <view :renderjs_data="renderJSMountedEvent" :change:renderjs_data="xinyucrosscanvas.onMountedRenderjs">
  25. </view>
  26. <view :renderjs_data="renderJSLoadImage" :change:renderjs_data="xinyucrosscanvas.loadImageRenderjs">
  27. </view>
  28. <view class="xinyucrosscanvas"></view>
  29. <!-- #endif -->
  30. </view>
  31. </view>
  32. </template>
  33. <script>
  34. export default {
  35. props: {
  36. id: {
  37. type: String,
  38. default: "xinyu_cross_canvas"
  39. },
  40. styleWidth: {
  41. type: Number,
  42. default: -1
  43. },
  44. styleHeight: {
  45. type: Number,
  46. default: -1
  47. },
  48. width: {
  49. type: Number,
  50. default: 750
  51. },
  52. height: {
  53. type: Number,
  54. default: 1506
  55. }
  56. },
  57. data() {
  58. return {
  59. renderjsData: null,
  60. renderJSMountedEvent: false,
  61. renderjsContextProp: null,
  62. renderjsContextMethodCall: null,
  63. renderJSLoadImage: null,
  64. inited: false,
  65. recvMethod: {},
  66. currentGenerateIndex: 0,
  67. downXY: {
  68. x: 0,
  69. y: 0
  70. },
  71. };
  72. },
  73. async mounted() {
  74. this.imageHash = [];
  75. this.refreshRenderJS();
  76. this.renderJSMountedEvent = true;
  77. // #ifdef MP-WEIXIN || MP-TOUTIAO
  78. await this.onCanvasInitReady();
  79. this.inited = true;
  80. // #endif
  81. },
  82. methods: {
  83. async refreshWidthHeight(width, height) {
  84. this.$emit("input:width", width);
  85. this.$emit("input:height", height);
  86. // #ifdef MP
  87. if (this.canvas) {
  88. this.canvas.width = width;
  89. this.canvas.height = height;
  90. }
  91. // #endif
  92. // #ifndef MP
  93. await new Promise((recv) => {
  94. this.renderjsData = this.generateObject({
  95. widthRenderJS: width,
  96. heightRenderJS: height
  97. });
  98. this.recvMethod.refreshRenderJSCallback = recv;
  99. });
  100. delete this.recvMethod.refreshRenderJSCallback;
  101. // #endif
  102. },
  103. generateObject(obj) {
  104. obj.__ = Date.now() + "_" + this.currentGenerateIndex++;
  105. return obj;
  106. },
  107. rpx(rpx) {
  108. return uni.upx2px(rpx) + 'px';
  109. },
  110. onInitFinished() {
  111. this.inited = true;
  112. },
  113. onRenderCallback(callbackObject) {
  114. let {
  115. func,
  116. param
  117. } = callbackObject;
  118. if (this.recvMethod[func])
  119. this.recvMethod[func].apply(this.recvMethod[func], [param]);
  120. },
  121. onTouchREvent(e) {
  122. this.$emit("ctouch", e);
  123. },
  124. onTClick(e) {
  125. this.$emit("cclick", e);
  126. },
  127. onTouch(type, e) {
  128. const px = Math.round(e.changedTouches[0].x / uni.getSystemInfoSync().screenWidth * this.width);
  129. const py = Math.round(e.changedTouches[0].y / uni.getSystemInfoSync().screenHeight * this.height);
  130. this.$emit("touch", {
  131. type,
  132. px,
  133. py
  134. });
  135. if (type == "start") {
  136. this.downXY.x = px;
  137. this.downXY.y = py;
  138. } else if (type == "end") {
  139. if ((this.downXY.x - px) * (this.downXY.x - px) + (this.downXY.y - py) * (this.downXY.y - py) < 64)
  140. this.onTClick({
  141. px,
  142. py
  143. });
  144. this.downXY.x = 0;
  145. this.downXY.y = 0;
  146. }
  147. },
  148. getImageHash(src) {
  149. let ret = this.imageHash.filter((t) => t.key == src);
  150. if (ret.length > 0)
  151. return ret[0].val;
  152. return null;
  153. },
  154. setImageHash(src, image) {
  155. this.imageHash.push({
  156. key: src,
  157. val: image
  158. });
  159. },
  160. async loadImage(src) {
  161. // #ifdef MP
  162. await new Promise((recv) => {
  163. if (this.getImageHash(src))
  164. return recv(this.getImageHash(src));
  165. const img = this.canvas.createImage();
  166. this.setImageHash(src, img);
  167. img.onload = () => {
  168. recv(img);
  169. };
  170. img.src = src;
  171. });
  172. // #endif
  173. // #ifndef MP
  174. await new Promise((recv) => {
  175. if (src == this.renderJSLoadImage)
  176. return recv();
  177. this.renderJSLoadImage = src;
  178. this.recvMethod.loadImageRenderJSCallback = recv;
  179. });
  180. delete this.recvMethod.loadImageRenderJSCallback;
  181. // #endif
  182. return src;
  183. },
  184. async loadImageCacheMP(src) {
  185. return await new Promise((recv) => {
  186. if (this.getImageHash(src))
  187. return recv(this.getImageHash(src));
  188. const img = this.canvas.createImage();
  189. this.setImageHash(src, img);
  190. img.onload = () => {
  191. recv(img);
  192. };
  193. img.src = src;
  194. });
  195. },
  196. async drawImage(src, methodParams) {
  197. // #ifdef MP
  198. let image = await this.loadImageCacheMP(src);
  199. this.context["drawImage"].apply(this.context, [image].concat(methodParams));
  200. // #endif
  201. // #ifndef MP
  202. await new Promise((recv) => {
  203. this.renderjsContextMethodCall = this.generateObject({
  204. methodName: "drawImage",
  205. methodParams: [src].concat(methodParams)
  206. });
  207. this.recvMethod.callContextMethodCallback = recv;
  208. });
  209. delete this.recvMethod.callContextMethodCallback;
  210. // #endif
  211. },
  212. async init() {
  213. while (true) {
  214. if (this.inited)
  215. return;
  216. await this.refreshRenderJS();
  217. await new Promise((t) => setTimeout(t, 200));
  218. }
  219. },
  220. async onCanvasInitReady() {
  221. await new Promise((recv) => {
  222. uni.createSelectorQuery()
  223. .in(this)
  224. .select("#" + this.id)
  225. .fields({
  226. node: true,
  227. size: true
  228. })
  229. .exec((res) => {
  230. this.canvas = res[0].node;
  231. this.canvas.width = this.width;
  232. this.canvas.height = this.height;
  233. this.context = this.canvas.getContext("2d");
  234. recv();
  235. });
  236. });
  237. },
  238. async onCanvasInitReadyAlipay() {
  239. await new Promise((recv) => {
  240. my.createSelectorQuery()
  241. .in(this)
  242. .select("#" + this.id)
  243. .node()
  244. .exec((res) => {
  245. this.canvas = res[0].node;
  246. this.canvas.width = this.width;
  247. this.canvas.height = this.height;
  248. this.context = this.canvas.getContext("2d");
  249. this.inited = true;
  250. recv();
  251. });
  252. });
  253. },
  254. async refreshRenderJS() {
  255. // #ifdef MP
  256. if (this.canvas && (this.canvas.width != this.width || this.canvas.height != this.height)) {
  257. this.canvas.width = this.width;
  258. this.canvas.height = this.height;
  259. }
  260. // #endif
  261. // #ifndef MP
  262. await new Promise((recv) => {
  263. this.renderjsData = this.generateObject({
  264. styleWidthRenderJS: this.styleWidth == -1 ? uni.upx2px(750) : uni.upx2px(this
  265. .styleWidth),
  266. styleHeightRenderJS: this.styleHeight == -1 ? -1 : uni.upx2px(this
  267. .styleHeight),
  268. widthRenderJS: this.width,
  269. heightRenderJS: this.height
  270. });
  271. this.recvMethod.refreshRenderJSCallback = recv;
  272. });
  273. delete this.recvMethod.refreshRenderJSCallback;
  274. // #endif
  275. },
  276. async setContextProp(key, val) {
  277. // #ifdef MP
  278. this.context[key] = val;
  279. // #endif
  280. // #ifndef MP
  281. await new Promise((recv) => {
  282. this.renderjsContextProp = this.generateObject({
  283. key,
  284. val
  285. });
  286. this.recvMethod.setContextPropCallback = recv;
  287. });
  288. delete this.recvMethod.setContextPropCallback;
  289. // #endif
  290. },
  291. async callContextMethod(methodName, methodParams) {
  292. if (methodName == "drawImage")
  293. return await this.drawImage(methodParams[0], methodParams.slice(1));
  294. // #ifdef MP
  295. if (methodName == "toDataURL") {
  296. return this.canvas.toDataURL.apply(this.canvas, methodParams);
  297. }
  298. return this.context[methodName].apply(this.context, methodParams);
  299. // #endif
  300. // #ifndef MP
  301. let ret = await new Promise((recv) => {
  302. this.renderjsContextMethodCall = this.generateObject({
  303. methodName,
  304. methodParams
  305. });
  306. this.recvMethod.callContextMethodCallback = recv;
  307. });
  308. delete this.recvMethod.callContextMethodCallback;
  309. return ret;
  310. // #endif
  311. },
  312. async fillRectList(rectList) {
  313. // #ifdef MP
  314. for (let i in rectList) {
  315. this.context.fillStyle = rectList[i].fillStyle;
  316. this.context.fillRect(rectList[i].x, rectList[i].y, rectList[i].w, rectList[i].h);
  317. }
  318. // #endif
  319. // #ifndef MP
  320. let ret = await new Promise((recv) => {
  321. this.renderjsContextMethodCall = this.generateObject({
  322. methodName: "fillRectList",
  323. methodParams: [rectList]
  324. });
  325. this.recvMethod.callContextMethodCallback = recv;
  326. });
  327. delete this.recvMethod.callContextMethodCallback;
  328. return ret;
  329. // #endif
  330. }
  331. }
  332. }
  333. </script>
  334. <script module="xinyucrosscanvas" lang="renderjs">
  335. // #ifndef MP
  336. export default {
  337. data() {
  338. return {
  339. canvasRenderJS: null,
  340. contextRenderJS: null,
  341. styleWidthRenderJS: -1,
  342. styleHeightRenderJS: -1,
  343. widthRenderJS: 750,
  344. heightRenderJS: 1506,
  345. initedRenderJS: false,
  346. downXYRenderJS: {
  347. x: 0,
  348. y: 0
  349. },
  350. imageHashRenderJS: []
  351. };
  352. },
  353. methods: {
  354. getImageHashRenderjs(src) {
  355. let ret = this.imageHashRenderJS.filter((t) => t.key == src);
  356. if (ret.length > 0)
  357. return ret[0].val;
  358. return null;
  359. },
  360. setImageHashRenderjs(src, image) {
  361. this.imageHashRenderJS.push({
  362. key: src,
  363. val: image
  364. });
  365. },
  366. onMountedRenderjs(bool) {
  367. if (!bool || this.initedRenderJS)
  368. return;
  369. this.initedRenderJS = true;
  370. this.canvasRenderJS = document.createElement("canvas");
  371. this.canvasRenderJS.style.width = this.styleWidthRenderJS + 'px';
  372. this.canvasRenderJS.style.height = (this.styleHeightRenderJS == -1 ? window.innerHeight : this
  373. .styleHeightRenderJS) + 'px';
  374. this.canvasRenderJS.width = this.widthRenderJS;
  375. this.canvasRenderJS.height = this.heightRenderJS;
  376. this.contextRenderJS = this.canvasRenderJS.getContext('2d');
  377. this.canvasRenderJS.ontouchstart = (e) => {
  378. this.onRenderjsTouch('start', e);
  379. };
  380. this.canvasRenderJS.ontouchmove = (e) => {
  381. this.onRenderjsTouch('move', e);
  382. };
  383. this.canvasRenderJS.ontouchend = (e) => {
  384. this.onRenderjsTouch('end', e);
  385. };
  386. document.getElementsByClassName("xinyucrosscanvas")[0].appendChild(this.canvasRenderJS);
  387. this.$ownerInstance.callMethod("onInitFinished");
  388. },
  389. refreshDataRenderjs(data) {
  390. if (data == null)
  391. return;
  392. if (this.canvasRenderJS != null) {
  393. if (data.widthRenderJS != this.widthRenderJS || data.heightRenderJS != this.heightRenderJS) {
  394. this.canvasRenderJS.width = data.widthRenderJS;
  395. this.canvasRenderJS.height = data.heightRenderJS;
  396. }
  397. this.canvasRenderJS.style.width = this.styleWidthRenderJS + 'px';
  398. this.canvasRenderJS.style.height = (this.styleHeightRenderJS == -1 ? window.innerHeight : this
  399. .styleHeightRenderJS) + 'px';
  400. }
  401. Object.assign(this, data);
  402. this.$nextTick(() => {
  403. this.$ownerInstance.callMethod("onRenderCallback", {
  404. func: "refreshRenderJSCallback",
  405. param: null
  406. });
  407. });
  408. },
  409. async loadImageRenderjs(src) {
  410. if (src != null)
  411. await this.loadRenderJSImageCache(src);
  412. this.$ownerInstance.callMethod("onRenderCallback", {
  413. func: "loadImageRenderJSCallback",
  414. param: null
  415. });
  416. },
  417. async loadRenderJSImageCache(src) {
  418. if (src.startsWith("/"))
  419. src = "." + src;
  420. return await new Promise((recv) => {
  421. if (this.getImageHashRenderjs(src))
  422. return recv(this.getImageHashRenderjs(src));
  423. const img = new Image();
  424. img.setAttribute("crossOrigin", 'anonymous');
  425. this.setImageHashRenderjs(src, img);
  426. img.onload = () => {
  427. recv(img);
  428. };
  429. img.src = src;
  430. if (img.complete)
  431. recv(img);
  432. });
  433. },
  434. setContextPropRenderjs(prop) {
  435. if (prop == null)
  436. return;
  437. this.contextRenderJS[prop.key] = prop.val;
  438. this.$ownerInstance.callMethod("onRenderCallback", {
  439. func: "setContextPropCallback",
  440. param: null
  441. });
  442. },
  443. async callContextMethodRenderjs(prop) {
  444. if (prop == null)
  445. return;
  446. let ret = null;
  447. if (prop.methodName == "drawImage") {
  448. let param = prop.methodParams.slice(0);
  449. param[0] = await this.loadRenderJSImageCache(param[0]);
  450. ret = this.contextRenderJS.drawImage.apply(this.contextRenderJS, param);
  451. } else if (prop.methodName == "toDataURL") {
  452. ret = this.canvasRenderJS.toDataURL.apply(this.canvasRenderJS, prop.methodParams);
  453. } else if (prop.methodName == "fillRectList") {
  454. let rectList = prop.methodParams[0];
  455. for (let i in rectList) {
  456. this.contextRenderJS.fillStyle = rectList[i].fillStyle;
  457. this.contextRenderJS.fillRect(rectList[i].x, rectList[i].y, rectList[i].w, rectList[i].h);
  458. }
  459. } else
  460. ret = this.contextRenderJS[prop.methodName].apply(this.contextRenderJS, prop.methodParams);
  461. this.$ownerInstance.callMethod("onRenderCallback", {
  462. func: "callContextMethodCallback",
  463. param: ret
  464. });
  465. },
  466. onRenderjsTouch(type, e) {
  467. let canvasPosition = this.canvasRenderJS.getBoundingClientRect();
  468. const px = Math.round((e.changedTouches[0].clientX - canvasPosition.x) / uni.getSystemInfoSync()
  469. .screenWidth * this.width);
  470. const py = Math.round((e.changedTouches[0].clientY - canvasPosition.y) / uni.getSystemInfoSync()
  471. .screenHeight * this.height);
  472. this.$ownerInstance.callMethod("onTouchREvent", {
  473. type,
  474. px,
  475. py
  476. });
  477. if (type == "start") {
  478. this.downXYRenderJS.x = px;
  479. this.downXYRenderJS.y = py;
  480. } else if (type == "end") {
  481. if ((this.downXYRenderJS.x - px) * (this.downXYRenderJS.x - px) + (this.downXYRenderJS.y - py) * (this
  482. .downXYRenderJS.y - py) < 64)
  483. this.$ownerInstance.callMethod("onTClick", {
  484. px,
  485. py
  486. });
  487. this.downXYRenderJS.x = 0;
  488. this.downXYRenderJS.y = 0;
  489. }
  490. }
  491. }
  492. };
  493. // #endif
  494. </script>
  495. <style>
  496. </style>