nadesiko3 3.3.45 → 3.3.49

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (329) hide show
  1. package/LICENSE +21 -21
  2. package/README.md +143 -132
  3. package/batch/browsers.template.md +3 -3
  4. package/batch/build_browsers.nako3 +72 -72
  5. package/batch/build_command.nako3 +44 -44
  6. package/batch/build_nako_version.nako3 +42 -42
  7. package/batch/calc_hash.nako3 +29 -29
  8. package/batch/cmd_txt2json.nako3 +74 -74
  9. package/batch/command.txt +270 -338
  10. package/batch/command_nakopad.txt +9 -69
  11. package/batch/download-extlib.nako3 +43 -43
  12. package/batch/gen_command_nakopad.nako3 +57 -57
  13. package/batch/jsplugin2text.nako3 +285 -285
  14. package/batch/pickup_command.nako3 +110 -110
  15. package/batch/pickup_reserved_words.nako3 +11 -11
  16. package/batch/publish_version.nako3 +46 -46
  17. package/batch/show_agents.js +14 -14
  18. package/batch/turtle2js.nako3 +21 -21
  19. package/bin/cnako3 +10 -10
  20. package/core/.editorconfig +6 -0
  21. package/core/.eslintrc.cjs +33 -0
  22. package/core/.github/dependabot.yml +7 -0
  23. package/core/.github/workflows/nodejs.yml +37 -0
  24. package/core/.github/workflows/super-linter.yml +61 -0
  25. package/core/.github/workflows/textlint.yml +199 -0
  26. package/core/LICENSE +21 -0
  27. package/core/README.md +66 -0
  28. package/core/batch/build_nako_version.nako3 +42 -0
  29. package/core/command/snako.mjs +105 -0
  30. package/core/command/snako.mts +116 -0
  31. package/core/index.mjs +21 -0
  32. package/core/index.mts +21 -0
  33. package/core/package.json +47 -0
  34. package/core/sample/hello.nako3 +7 -0
  35. package/core/sample/hoge.mjs +4 -0
  36. package/core/sample/hoge.mts +6 -0
  37. package/core/src/nako3.mjs +858 -0
  38. package/core/src/nako3.mts +967 -0
  39. package/core/src/nako_colors.mjs +78 -0
  40. package/core/src/nako_colors.mts +86 -0
  41. package/core/src/nako_core_version.mjs +8 -0
  42. package/core/src/nako_core_version.mts +19 -0
  43. package/core/src/nako_csv.mjs +185 -0
  44. package/core/src/nako_csv.mts +188 -0
  45. package/core/src/nako_errors.mjs +173 -0
  46. package/core/src/nako_errors.mts +197 -0
  47. package/core/src/nako_from_dncl.mjs +255 -0
  48. package/core/src/nako_from_dncl.mts +250 -0
  49. package/core/src/nako_gen.mjs +1648 -0
  50. package/core/src/nako_gen.mts +1719 -0
  51. package/core/src/nako_gen_async.mjs +1659 -0
  52. package/core/src/nako_gen_async.mts +1732 -0
  53. package/core/src/nako_global.mjs +107 -0
  54. package/core/src/nako_global.mts +138 -0
  55. package/core/src/nako_indent.mjs +445 -0
  56. package/core/src/nako_indent.mts +492 -0
  57. package/core/src/nako_josi_list.mjs +38 -0
  58. package/core/src/nako_josi_list.mts +45 -0
  59. package/core/src/nako_lex_rules.mjs +253 -0
  60. package/core/src/nako_lex_rules.mts +260 -0
  61. package/core/src/nako_lexer.mjs +609 -0
  62. package/core/src/nako_lexer.mts +612 -0
  63. package/core/src/nako_logger.mjs +199 -0
  64. package/core/src/nako_logger.mts +232 -0
  65. package/core/src/nako_parser3.mjs +2439 -0
  66. package/core/src/nako_parser3.mts +2195 -0
  67. package/core/src/nako_parser_base.mjs +370 -0
  68. package/core/src/nako_parser_base.mts +370 -0
  69. package/core/src/nako_parser_const.mjs +37 -0
  70. package/core/src/nako_parser_const.mts +37 -0
  71. package/core/src/nako_prepare.mjs +304 -0
  72. package/core/src/nako_prepare.mts +315 -0
  73. package/core/src/nako_reserved_words.mjs +38 -0
  74. package/core/src/nako_reserved_words.mts +38 -0
  75. package/core/src/nako_source_mapping.mjs +207 -0
  76. package/core/src/nako_source_mapping.mts +262 -0
  77. package/core/src/nako_test.mjs +37 -0
  78. package/core/src/nako_types.mjs +25 -0
  79. package/core/src/nako_types.mts +151 -0
  80. package/core/src/plugin_csv.mjs +49 -0
  81. package/core/src/plugin_csv.mts +50 -0
  82. package/core/src/plugin_math.mjs +328 -0
  83. package/core/src/plugin_math.mts +326 -0
  84. package/core/src/plugin_promise.mjs +91 -0
  85. package/core/src/plugin_promise.mts +91 -0
  86. package/core/src/plugin_system.mjs +2832 -0
  87. package/core/src/plugin_system.mts +2690 -0
  88. package/core/src/plugin_test.mjs +34 -0
  89. package/core/src/plugin_test.mts +34 -0
  90. package/core/test/array_test.mjs +34 -0
  91. package/core/test/basic_test.mjs +344 -0
  92. package/core/test/calc_test.mjs +140 -0
  93. package/core/test/core_module_test.mjs +23 -0
  94. package/core/test/debug_test.mjs +16 -0
  95. package/core/test/dncl_test.mjs +94 -0
  96. package/core/test/error_message_test.mjs +210 -0
  97. package/core/test/error_test.mjs +16 -0
  98. package/core/test/flow_test.mjs +373 -0
  99. package/core/test/func_call.mjs +160 -0
  100. package/core/test/func_test.mjs +149 -0
  101. package/core/test/indent_test.mjs +364 -0
  102. package/core/test/lex_test.mjs +168 -0
  103. package/core/test/literal_test.mjs +73 -0
  104. package/core/test/nako_lexer_test.mjs +35 -0
  105. package/core/test/nako_logger_test.mjs +76 -0
  106. package/core/test/nako_logger_test.mts +78 -0
  107. package/core/test/plugin_csv_test.mjs +38 -0
  108. package/core/test/plugin_promise_test.mjs +18 -0
  109. package/core/test/plugin_system_test.mjs +630 -0
  110. package/core/test/prepare_test.mjs +96 -0
  111. package/core/test/re_test.mjs +22 -0
  112. package/core/test/side_effects_test.mjs +92 -0
  113. package/core/test/variable_scope_test.mjs +149 -0
  114. package/core/tsconfig.json +101 -0
  115. package/demo/ace_editor.html +89 -89
  116. package/demo/ace_editor_tabs.html +161 -161
  117. package/demo/basic.html +71 -71
  118. package/demo/browsers.html +9 -10
  119. package/demo/css/basic.css +3 -3
  120. package/demo/css/common.css +157 -157
  121. package/demo/css/editor.css +8 -8
  122. package/demo/css/flow.css +3 -3
  123. package/demo/css/index.css +3 -3
  124. package/demo/flow.html +98 -98
  125. package/demo/graph.html +53 -53
  126. package/demo/image/nakopad-icon256.png +0 -0
  127. package/demo/index.html +133 -133
  128. package/demo/js/common.js +17 -17
  129. package/demo/js/turtle3d_test.js +44 -44
  130. package/demo/js/turtle_test.js +45 -45
  131. package/demo/nako3/calc.nako3 +4 -4
  132. package/demo/runscript.html +47 -47
  133. package/demo/runscript2.html +33 -33
  134. package/demo/runscript3.html +35 -35
  135. package/demo/runscript4.html +33 -33
  136. package/demo/turtle.html +58 -58
  137. package/demo/turtle2.html +141 -141
  138. package/demo/turtle3.html +279 -279
  139. package/demo/turtle3d.html +58 -58
  140. package/demo/turtle3d2.html +107 -107
  141. package/demo/version.html +24 -24
  142. package/doc/SETUP.md +157 -157
  143. package/doc/about.md +17 -17
  144. package/doc/browsers.md +26 -26
  145. package/doc/docgen.md +21 -21
  146. package/doc/editor.md +44 -44
  147. package/doc/files.md +39 -39
  148. package/doc/plugins.md +234 -234
  149. package/doc/release.md +79 -79
  150. package/doc/textlint.md +43 -43
  151. package/doc/win32.md +57 -57
  152. package/package.json +195 -192
  153. package/release/_hash.txt +28 -28
  154. package/release/_script-tags.txt +14 -14
  155. package/release/command.json +1 -1
  156. package/release/command.json.js +1 -1
  157. package/release/command_cnako3.json +1 -1
  158. package/release/command_list.json +1 -1
  159. package/release/editor.js +1 -1
  160. package/release/nako_gen_async.js +1 -1
  161. package/release/plugin_csv.js +1 -1
  162. package/release/stats.json +1 -1
  163. package/release/version.js +1 -1
  164. package/release/wnako3.js +1 -1
  165. package/release/wnako3webworker.js +1 -1
  166. package/src/browsers.txt +11 -12
  167. package/src/browsers_agents.json +2 -2
  168. package/src/browsers_agents.mjs +1 -1
  169. package/src/cnako3.mjs +17 -10
  170. package/src/cnako3.mts +18 -12
  171. package/src/cnako3mod.mjs +707 -687
  172. package/src/cnako3mod.mts +712 -696
  173. package/src/commander_ja.mjs +164 -164
  174. package/src/commander_ja.mts +161 -161
  175. package/src/enako3.mjs +68 -69
  176. package/src/era.mjs +22 -22
  177. package/src/image_turtle-elephant.mjs +2 -5
  178. package/src/image_turtle-panda.mjs +2 -5
  179. package/src/image_turtle64.mjs +2 -5
  180. package/src/index.mjs +9 -9
  181. package/src/index.mts +10 -11
  182. package/src/nako3editorfix.sfd +106 -106
  183. package/src/nako_version.mjs +8 -8
  184. package/src/nako_version.mts +2 -2
  185. package/src/plugin_browser.mjs +213 -212
  186. package/src/plugin_browser.mts +206 -205
  187. package/src/plugin_browser_ajax.mjs +399 -399
  188. package/src/plugin_browser_audio.mjs +109 -109
  189. package/src/plugin_browser_canvas.mjs +449 -449
  190. package/src/plugin_browser_chart.mjs +294 -294
  191. package/src/plugin_browser_color.mjs +49 -49
  192. package/src/plugin_browser_crypto.mjs +26 -26
  193. package/src/plugin_browser_dialog.mjs +53 -53
  194. package/src/plugin_browser_dom_basic.mjs +336 -336
  195. package/src/plugin_browser_dom_event.mjs +193 -193
  196. package/src/plugin_browser_dom_parts.mjs +396 -396
  197. package/src/plugin_browser_geolocation.mjs +51 -51
  198. package/src/plugin_browser_hotkey.mjs +25 -25
  199. package/src/plugin_browser_html.mjs +59 -59
  200. package/src/plugin_browser_in_worker.mjs +45 -45
  201. package/src/plugin_browser_location.mjs +21 -21
  202. package/src/plugin_browser_speech.mjs +111 -111
  203. package/src/plugin_browser_storage.mjs +121 -121
  204. package/src/plugin_browser_system.mjs +31 -31
  205. package/src/plugin_browser_websocket.mjs +73 -73
  206. package/src/plugin_caniuse.mjs +29 -29
  207. package/src/plugin_datetime.mjs +394 -394
  208. package/src/plugin_httpserver.mjs +277 -0
  209. package/src/plugin_httpserver.mts +286 -0
  210. package/src/plugin_kansuji.mjs +224 -224
  211. package/src/plugin_keigo.mjs +55 -55
  212. package/src/plugin_markup.mjs +32 -32
  213. package/src/plugin_node.mjs +1047 -1047
  214. package/src/plugin_node.mts +980 -980
  215. package/src/plugin_turtle.mjs +647 -647
  216. package/src/plugin_webworker.mjs +334 -334
  217. package/src/plugin_weykturtle3d.mjs +1214 -1214
  218. package/src/plugin_worker.mjs +95 -95
  219. package/src/repl.nako3 +63 -63
  220. package/src/wnako3.mjs +12 -12
  221. package/src/wnako3.mts +11 -11
  222. package/src/wnako3_editor.css +215 -215
  223. package/src/wnako3_editor.mjs +1542 -1542
  224. package/src/wnako3_editor.mts +1657 -1656
  225. package/src/wnako3mod.mjs +213 -213
  226. package/src/wnako3mod.mts +214 -214
  227. package/src/wnako3webworker.mjs +69 -68
  228. package/test/ace_editor/karma.config.js +94 -94
  229. package/test/ace_editor/test/.babelrc.json +3 -3
  230. package/test/ace_editor/test/ace_editor_test.js +178 -178
  231. package/test/ace_editor/test/html/custom_context.html +139 -139
  232. package/test/async/async_basic_test.mjs +122 -122
  233. package/test/browser/karma.config.js +221 -221
  234. package/test/browser/test/.babelrc.json +3 -3
  235. package/test/browser/test/compare_util.js +50 -50
  236. package/test/browser/test/html/div_basic.html +2 -2
  237. package/test/browser/test/html/event_dom_form.html +4 -4
  238. package/test/browser/test/html/event_dom_scrolldiv.html +5 -5
  239. package/test/browser/test/import_plugin_checker.js +24 -24
  240. package/test/browser/test/plugin_browser_test.js +51 -51
  241. package/test/browser/test/plugin_browser_test_ajax.js +123 -123
  242. package/test/browser/test/plugin_browser_test_color.js +18 -18
  243. package/test/browser/test/plugin_browser_test_dialog.js +72 -72
  244. package/test/browser/test/plugin_browser_test_dom_event.js +598 -598
  245. package/test/browser/test/plugin_browser_test_dom_parts.js +125 -125
  246. package/test/browser/test/plugin_browser_test_system.js +9 -9
  247. package/test/browser/test/plugin_turtle_test.js +817 -817
  248. package/test/browser/test/plugin_webworker_test.js +86 -86
  249. package/test/browser/test/require_test.js +68 -68
  250. package/test/bundled/karma.config.base.js +117 -117
  251. package/test/bundled/karma.config.js +86 -86
  252. package/test/bundled/test/.babelrc.json +3 -3
  253. package/test/bundled/test/bundled_test.js +69 -69
  254. package/test/bundled/test/html/custom_context.html +65 -65
  255. package/test/bundled/test/html/custom_debug.html +66 -66
  256. package/test/bundled/test4b.cmd +52 -52
  257. package/test/bundled/test_base/.babelrc.json +3 -3
  258. package/test/bundled/test_base/_checktool_test.js +25 -25
  259. package/test/bundled/test_base/basic_ajax_test.js +56 -56
  260. package/test/bundled/test_base/basic_async_test.js +18 -18
  261. package/test/bundled/test_base/basic_test.js +153 -153
  262. package/test/bundled/test_base/calc_test.js +132 -132
  263. package/test/bundled/test_base/css/browsers_box.css +114 -114
  264. package/test/bundled/test_base/html/custom_context.html +69 -69
  265. package/test/bundled/test_base/html/custom_debug.html +71 -71
  266. package/test/bundled/test_base/js/browsers_box.js +72 -72
  267. package/test/bundled/test_base/plugin_csv_test.js +37 -37
  268. package/test/bundled/test_base/plugin_datetime_test.js +115 -115
  269. package/test/bundled/test_base/plugin_kansuji_test.js +49 -49
  270. package/test/bundled/test_base/plugin_system_test.js +410 -410
  271. package/test/bundled/test_base/plugin_webworker_test.js +53 -53
  272. package/test/bundled/test_base/test_utils.js +191 -191
  273. package/test/common/plugin_browser_test.mjs +22 -24
  274. package/test/common/plugin_browser_ut_audio_test.mjs +108 -108
  275. package/test/common/plugin_browser_ut_color_test.mjs +21 -21
  276. package/test/common/plugin_browser_ut_dialog_test.mjs +100 -100
  277. package/test/common/plugin_browser_ut_html_test.mjs +13 -13
  278. package/test/common/plugin_browser_ut_system_test.mjs +10 -10
  279. package/test/common/plugin_markup_test.mjs +23 -23
  280. package/test/jsconfig.json +19 -19
  281. package/test/karma.config.js +91 -91
  282. package/test/node/async_test.mjs +96 -82
  283. package/test/node/commander_ja_test.mjs +89 -89
  284. package/test/node/error_message_test.mjs +243 -253
  285. package/test/node/kai_test.nako3 +6 -6
  286. package/test/node/node_test.mjs +60 -60
  287. package/test/node/plugin_broken.js.txt +3 -3
  288. package/test/node/plugin_browser_ut_ajax_test.mjs.todo +357 -357
  289. package/test/node/plugin_browser_ut_location_test.mjs +42 -42
  290. package/test/node/plugin_markup_test.mjs +47 -46
  291. package/test/node/plugin_math_test.mjs +45 -44
  292. package/test/node/plugin_node_test.mjs +98 -97
  293. package/test/node/plugin_test.mjs +44 -43
  294. package/test/node/relative_import_test_1.nako3 +1 -1
  295. package/test/node/relative_import_test_2.nako3 +2 -2
  296. package/test/node/require_nako3_test.mjs +67 -66
  297. package/test/node/requiretest.nako3 +4 -4
  298. package/test/node/requiretest_indirect.nako3 +1 -1
  299. package/test/node/requiretest_name.nako3 +5 -5
  300. package/test/node/runtime_error.nako3 +2 -2
  301. package/test/node/scope1.nako3 +10 -10
  302. package/test/node/scope2.nako3 +12 -12
  303. package/test/node/side_effects_test.mjs +39 -119
  304. package/test/node/sjis.txt +5 -5
  305. package/test/node/syntax_error.nako3 +1 -1
  306. package/test/node/wnako3_editor_test.mjs +384 -384
  307. package/tools/README.md +12 -7
  308. package/tools/check_new_version.nako3 +25 -25
  309. package/tools/nako3edit/html/daisyui/LICENSE +22 -22
  310. package/tools/nako3edit/html/daisyui/version_2.14.1 +1 -1
  311. package/tools/nako3edit/html/edit.html +170 -170
  312. package/tools/nako3edit/html/edit_plugin.js +6 -6
  313. package/tools/nako3edit/html/files.html +125 -125
  314. package/tools/nako3edit/html/nako3edit.css +65 -65
  315. package/tools/nako3edit/index.mjs +248 -244
  316. package/tools/nako3server/index.html +10 -0
  317. package/tools/nako3server/index.mjs +116 -116
  318. package/tools/nako3server/index.nako3 +34 -0
  319. package/release/nako_gen_async.js.LICENSE.txt +0 -35
  320. package/release/plugin_caniuse.js.LICENSE.txt +0 -11
  321. package/release/plugin_csv.js.LICENSE.txt +0 -15
  322. package/release/plugin_datetime.js.LICENSE.txt +0 -15
  323. package/release/plugin_kansuji.js.LICENSE.txt +0 -3
  324. package/release/plugin_markup.js.LICENSE.txt +0 -11
  325. package/release/plugin_turtle.js.LICENSE.txt +0 -15
  326. package/release/plugin_webworker.js.LICENSE.txt +0 -3
  327. package/release/plugin_weykturtle3d.js.LICENSE.txt +0 -3
  328. package/release/wnako3webworker.js.LICENSE.txt +0 -131
  329. package/tools/nako3edit/a.sqlite3 +0 -0
@@ -1,817 +1,817 @@
1
- /* global __html__ */
2
- import { assert } from 'chai'
3
- import { NakoCompiler } from 'nadesiko3core/src/nako3.mjs'
4
- import { importStatus } from './import_plugin_checker.js'
5
- import PluginTurtle from 'nako3/plugin_turtle.mjs'
6
-
7
- const htmlPath = 'test/html/'
8
- const imagePath = '/test/image/'
9
-
10
- const waitTimer = (second) => {
11
- return new Promise((resolve) => {
12
- setTimeout(() => {
13
- resolve()
14
- }, second * 1000)
15
- })
16
- }
17
-
18
- const setImageSmoothingEnabledToCtx = (ctx, value) => {
19
- ctx.mozImageSmoothingEnabled = value
20
- ctx.webkitImageSmoothingEnabled = value
21
- ctx.msImageSmoothingEnabled = value
22
- ctx.imageSmoothingEnabled = value
23
- }
24
-
25
- const getImageDataFromUrl = (url) => {
26
- return new Promise((resolve, reject) => {
27
- const img = new Image()
28
- img.onload = () => {
29
- img.onload = null
30
- img.onerror = null
31
- const canvas = document.createElement('canvas')
32
- canvas.width = img.width
33
- canvas.height = img.height
34
- document.body.appendChild(canvas)
35
- const ctx = canvas.getContext('2d')
36
- ctx.clearRect(0, 0, canvas.width, canvas.height)
37
- ctx.globalCompositeOperation = 'copy'
38
- setImageSmoothingEnabledToCtx(ctx, false)
39
- ctx.drawImage(img, 0, 0)
40
-
41
- resolve(ctx.getImageData(0, 0, canvas.width, canvas.height))
42
- }
43
- img.onerror = () => {
44
- img.onload = null
45
- img.onerror = null
46
- reject(new Error('load error'))
47
- }
48
- img.src = url
49
- })
50
- }
51
-
52
- const getMergedCanvas = (canvasId, turtleIds) => {
53
- return new Promise((resolve, reject) => {
54
- const cv = document.getElementById(canvasId)
55
-
56
- if (cv == null) {
57
- reject(new Error('fail create canvas'))
58
- return
59
- }
60
-
61
- const rect = cv.getBoundingClientRect()
62
- const w = parseInt(cv.width, 10)
63
- const h = parseInt(cv.height, 10)
64
- const rx = rect.left + window.pageXOffset
65
- const ry = rect.top + window.pageYOffset
66
-
67
- const canvas = document.createElement('canvas')
68
- canvas.width = w
69
- canvas.height = h
70
- document.body.appendChild(canvas)
71
-
72
- const ctx = canvas.getContext('2d')
73
- ctx.clearRect(0, 0, w, h)
74
-
75
- const ctxm = cv.getContext('2d')
76
- const imgdtm = ctxm.getImageData(0, 0, w, h)
77
- ctx.globalCompositeOperation = 'copy'
78
- setImageSmoothingEnabledToCtx(ctx, false)
79
- ctx.putImageData(imgdtm, 0, 0)
80
- ctx.globalCompositeOperation = 'source-over'
81
-
82
- turtleIds.forEach(turtleId => {
83
- const cvt = document.getElementById(turtleId)
84
- if (cvt) {
85
- const dx = parseInt(cvt.style.left, 10) - rx
86
- const dy = parseInt(cvt.style.top, 10) - ry
87
- ctx.drawImage(cvt, dx, dy)
88
- }
89
- })
90
-
91
- resolve(canvas)
92
- })
93
- }
94
-
95
- const recreateCanvas = src => {
96
- return new Promise((resolve, reject) => {
97
- const img = new Image()
98
- img.onload = () => {
99
- img.onload = null
100
- img.onerror = null
101
- const canvas = document.createElement('canvas')
102
- canvas.width = img.width
103
- canvas.height = img.height
104
- const ctx = canvas.getContext('2d')
105
- setImageSmoothingEnabledToCtx(ctx, false)
106
- ctx.clearRect(0, 0, img.width, img.height)
107
- ctx.globalCompositeOperation = 'copy'
108
- ctx.drawImage(img, 0, 0)
109
- ctx.globalCompositeOperation = 'source-over'
110
-
111
- resolve(canvas)
112
- }
113
- img.onerror = () => {
114
- img.onload = null
115
- img.onerror = null
116
- reject(new Error('reload image error'))
117
- }
118
- img.src = src.toDataURL('image/png')
119
- })
120
- }
121
-
122
- const getBlobFromCanvas = (canvas, type) => {
123
- const dataurl = canvas.toDataURL(type)
124
- const bin = atob(dataurl.replace(/^.*,/, ''))
125
- const buffer = new Uint8Array(bin.length)
126
- for (let i = 0, l = bin.length; i < l; i++) {
127
- buffer[i] = bin.charCodeAt(i)
128
- }
129
- const blob = new Blob([buffer.buffer], {
130
- type: type
131
- })
132
- return blob
133
- }
134
-
135
- const uploadCanvasImage = async (canvas, filename) => {
136
- var formData = new FormData()
137
- const blob = getBlobFromCanvas(canvas, 'image/png')
138
- formData.append('file', blob, filename)
139
- const response = await fetch('/custom/uploadimage', {
140
- method: 'POST',
141
- body: formData
142
- })
143
- const result = await response.text()
144
- return result
145
- }
146
-
147
- describe('plugin_turtle_test', () => {
148
- let nako = null
149
- const matchmode = 'lazzy3'
150
- const cmpImageData = (imgData1, imgData2, invert) => {
151
- if (typeof invert === 'undefined') { invert = false }
152
- assert.strictEqual(imgData1.width, imgData2.width, 'images width unmatch')
153
- assert.strictEqual(imgData1.height, imgData2.height, 'images height unmatch')
154
-
155
- console.log(`brute match(${matchmode})`)
156
- let allmatch = true
157
- let i = 0
158
- for (let y = 0; y < imgData1.height; y++) {
159
- for (let x = 0; x < imgData1.width; x++, i += 4) {
160
- let nomatch = false
161
- if (matchmode === 'lazzy') {
162
- if ((imgData2.data[i + 3] !== 0 &&
163
- (Math.abs(imgData1.data[i + 0] - imgData2.data[i + 0]) > 10 ||
164
- Math.abs(imgData1.data[i + 1] - imgData2.data[i + 1]) > 10 ||
165
- Math.abs(imgData1.data[i + 2] - imgData2.data[i + 2]) > 10 ||
166
- Math.abs(imgData1.data[i + 3] - imgData2.data[i + 3]) > 10)) ||
167
- (imgData2.data[i + 3] === 0 && imgData1.data[i + 3] > 5)) {
168
- nomatch = true
169
- }
170
- } else
171
- if (matchmode === 'lazzy2') {
172
- if (imgData1.data[i + 3] === 255 &&
173
- imgData2.data[i + 3] === 255 &&
174
- (Math.abs(imgData1.data[i + 0] - imgData2.data[i + 0]) > 1 ||
175
- Math.abs(imgData1.data[i + 1] - imgData2.data[i + 1]) > 1 ||
176
- Math.abs(imgData1.data[i + 2] - imgData2.data[i + 2]) > 1)) {
177
- // 両画像が不透明なら、差異は各パレットで1まで
178
- nomatch = true
179
- } else
180
- if (imgData1.data[i + 3] === 0 &&
181
- imgData2.data[i + 3] === 255) {
182
- // 画像1が完全透明なら、画像2が不透明ではいけない。
183
- nomatch = true
184
- } else
185
- if (imgData1.data[i + 3] === 255 &&
186
- imgData2.data[i + 3] === 0) {
187
- // 画像1が不透明なら、画像2が完全透明ではいけない。
188
- nomatch = true
189
- }
190
- } else
191
- if (matchmode === 'lazzy3') {
192
- if (imgData1.data[i + 3] === 255 &&
193
- imgData2.data[i + 3] === 255 &&
194
- (Math.abs(imgData1.data[i + 0] - imgData2.data[i + 0]) > 48 ||
195
- Math.abs(imgData1.data[i + 1] - imgData2.data[i + 1]) > 48 ||
196
- Math.abs(imgData1.data[i + 2] - imgData2.data[i + 2]) > 48)) {
197
- // 両画像が不透明なら、差異は各パレットで48まで
198
- nomatch = true
199
- } else
200
- if (imgData1.data[i + 3] === 0 &&
201
- imgData2.data[i + 3] === 255) {
202
- // 画像1が完全透明なら、画像2が不透明ではいけない。
203
- nomatch = true
204
- } else
205
- if (imgData1.data[i + 3] === 255 &&
206
- imgData2.data[i + 3] === 0) {
207
- // 画像1が不透明なら、画像2が完全透明ではいけない。
208
- nomatch = true
209
- }
210
- } else {
211
- if (imgData1.data[i + 0] !== imgData2.data[i + 0] ||
212
- imgData1.data[i + 1] !== imgData2.data[i + 1] ||
213
- imgData1.data[i + 2] !== imgData2.data[i + 2] ||
214
- imgData1.data[i + 3] !== imgData2.data[i + 3]) {
215
- nomatch = true
216
- }
217
- }
218
- if (nomatch) {
219
- allmatch = false
220
- if (!invert) {
221
- console.log('unmacth(' + x + ',' + y + '):' +
222
- (imgData1.data[i + 0] < 16 ? '0' : '') + (0 + imgData1.data[i + 0]).toString(16) +
223
- (imgData1.data[i + 1] < 16 ? '0' : '') + (0 + imgData1.data[i + 1]).toString(16) +
224
- (imgData1.data[i + 2] < 16 ? '0' : '') + (0 + imgData1.data[i + 2]).toString(16) +
225
- (imgData1.data[i + 3] < 16 ? '0' : '') + (0 + imgData1.data[i + 3]).toString(16) +
226
- ':' +
227
- (imgData2.data[i + 0] < 16 ? '0' : '') + (0 + imgData2.data[i + 0]).toString(16) +
228
- (imgData2.data[i + 1] < 16 ? '0' : '') + (0 + imgData2.data[i + 1]).toString(16) +
229
- (imgData2.data[i + 2] < 16 ? '0' : '') + (0 + imgData2.data[i + 2]).toString(16) +
230
- (imgData2.data[i + 3] < 16 ? '0' : '') + (0 + imgData2.data[i + 3]).toString(16) +
231
- ',' + i)
232
- }
233
- }
234
- }
235
- }
236
- if (!invert) {
237
- assert.ok(allmatch, 'image nomatch')
238
- } else {
239
- assert.ok(!allmatch, 'image match')
240
- }
241
- }
242
- afterEach(() => {
243
- nako = null
244
- })
245
- beforeEach(() => {
246
- nako = new NakoCompiler()
247
- // const pluginClone = Object.assign({}, PluginTurtle)
248
- // nako.addPluginFile('PluginTurtle', 'plugin_turtle.js', pluginClone)
249
- nako.addPluginFile('PluginTurtle', 'plugin_turtle.js', PluginTurtle)
250
- nako.addFunc('getElementyID', [['の']], (id, sys) => {
251
- return document.getElementById(id)
252
- })
253
- nako.addFunc('smoothオフ', [['の']], (id, sys) => {
254
- let ctx = null
255
- if (id >= 0) {
256
- if (id < sys._turtle.list.length) {
257
- ctx = sys._turtle.list[id].ctx
258
- }
259
- } else {
260
- ctx = sys._turtle.ctx
261
- }
262
- if (ctx) {
263
- setImageSmoothingEnabledToCtx(ctx, false)
264
- }
265
- return id
266
- })
267
- })
268
-
269
- // --- test ---
270
- it('check env(canvas_basic.html)', () => {
271
- document.body.innerHTML = __html__[htmlPath+'canvas_basic.html']
272
- const canvasElement = document.getElementById('turtle_cv')
273
- assert.strictEqual(typeof (canvasElement), 'object')
274
- })
275
-
276
- it('auto import for browser', () => {
277
- const pluginName = 'PluginTurtle'
278
- const imported = importStatus.hasImport(pluginName)
279
- assert.ok(imported, 'was import')
280
- const autoImport = importStatus.getAutoImport(pluginName)
281
- assert.strictEqual(typeof (autoImport.obj), 'object')
282
- })
283
-
284
- it('set origin and direcion', async () => {
285
- document.body.innerHTML = __html__[htmlPath+'canvas_basic.html']
286
- const imgname = 'canvas_test1.png'
287
- const code = `カメ描画先=「turtle_cv」
288
- 0にカメ速度設定。
289
- カメ画像URLは、「turtle.png」
290
- カメ作成。それのsmoothオフ。-1のsmoothオフ。
291
- [30,30]にカメ起点移動。
292
- 180にカメ角度設定。
293
- `
294
- nako.run(code)
295
-
296
- await waitTimer(1)
297
-
298
- const dataPromise = getImageDataFromUrl(imagePath + imgname)
299
- const cvPromise = getMergedCanvas('turtle_cv', ['0'])
300
- const [data, cv] = await Promise.all([dataPromise, cvPromise])
301
-
302
- const cv2 = await recreateCanvas(cv)
303
- const w = parseInt(cv2.width, 10)
304
- const h = parseInt(cv2.height, 10)
305
- const ctx = cv2.getContext('2d')
306
-
307
- const actData = ctx.getImageData(0, 0, w, h)
308
-
309
- try {
310
- const rslt = await uploadCanvasImage(cv, imgname)
311
- if (rslt !== 'OK') {
312
- console.log('upload result:' + rslt)
313
- }
314
- } catch (ex) {
315
- console.log('upload error')
316
- console.log(ex)
317
- }
318
-
319
- cmpImageData(actData, data)
320
- }).timeout(5000)
321
-
322
- it('incorrect canvasid', () => {
323
- document.body.innerHTML = __html__[htmlPath+'canvas_basic.html']
324
- const code = `カメ描画先=「no_cv」
325
- 0にカメ速度設定。
326
- カメ作成。
327
- `
328
- assert.throws(() => { nako.run(code) }, Error)
329
- })
330
-
331
- it('incorrect command', () => {
332
- document.body.innerHTML = __html__[htmlPath+'canvas_basic.html']
333
- const code = `カメ描画先=「turtle_cv」
334
- 0にカメ速度設定。
335
- [25,25]にカメ起点移動。
336
- `
337
- assert.throws(() => { nako.run(code) }, Error)
338
- })
339
-
340
- it('set origin and direcion turtles', async () => {
341
- document.body.innerHTML = __html__[htmlPath+'canvas_basic.html']
342
- const imgname = 'canvas_test2.png'
343
- const code = `カメ描画先=「#turtle_cv」
344
- 0にカメ速度設定。
345
- カメ画像URLは、「turtle.png」
346
- カメ作成。それのsmoothオフ。-1のsmoothオフ。
347
- [25,25]にカメ起点移動。
348
- 180にカメ角度設定。
349
- カメ画像URLは、「turtle-elephant.png」
350
- カメ作成。それのsmoothオフ。それにカメ操作対象設定。
351
- [75,25]にカメ起点移動。
352
- 90にカメ角度設定。
353
- カメ画像URLは、「turtle-panda.png」
354
- カメ作成。それのsmoothオフ。それにカメ操作対象設定。
355
- [50,75]にカメ起点移動。
356
- 0にカメ角度設定。
357
- `
358
- nako.run(code)
359
-
360
- await waitTimer(1.8)
361
-
362
- const dataPromise = getImageDataFromUrl(imagePath + imgname)
363
- const cvPromise = getMergedCanvas('turtle_cv', ['0', '1', '2'])
364
- const [data, cv] = await Promise.all([dataPromise, cvPromise])
365
-
366
- const cv2 = await recreateCanvas(cv)
367
- const w = parseInt(cv2.width, 10)
368
- const h = parseInt(cv2.height, 10)
369
- const ctx = cv2.getContext('2d')
370
-
371
- const actData = ctx.getImageData(0, 0, w, h)
372
-
373
- try {
374
- const rslt = await uploadCanvasImage(cv, imgname)
375
- if (rslt !== 'OK') {
376
- console.log('upload result:' + rslt)
377
- }
378
- } catch (ex) {
379
- console.log('upload error')
380
- console.log(ex)
381
- }
382
-
383
- cmpImageData(actData, data)
384
- }).timeout(5000)
385
-
386
- it('move direct position turtles', async () => {
387
- document.body.innerHTML = __html__[htmlPath+'canvas_basic.html']
388
- const imgname = 'canvas_test3.png'
389
- const code = `カメ描画先=「turtle_cv」のgetElementyID
390
- 0にカメ速度設定。
391
- カメ画像URLは、「turtle.png」
392
- カメ作成。それのsmoothオフ。-1のsmoothオフ。
393
- [25,75]にカメ起点移動。
394
- [25,25]にカメ移動。
395
- カメ画像URLは、「turtle-elephant.png」
396
- カメ作成。それのsmoothオフ。それにカメ操作対象設定。
397
- [25,75]にカメ起点移動。
398
- [75,75]にカメ移動。
399
- カメ画像URLは、「turtle-panda.png」
400
- カメ作成。それのsmoothオフ。それにカメ操作対象設定。
401
- [25,75]にカメ起点移動。
402
- [75,25]にカメ移動。
403
- `
404
- nako.run(code)
405
-
406
- await waitTimer(2.0)
407
-
408
- const dataPromise = getImageDataFromUrl(imagePath + imgname)
409
- const cvPromise = getMergedCanvas('turtle_cv', ['0', '1', '2'])
410
- const [data, cv] = await Promise.all([dataPromise, cvPromise])
411
-
412
- const cv2 = await recreateCanvas(cv)
413
- const w = parseInt(cv2.width, 10)
414
- const h = parseInt(cv2.height, 10)
415
- const ctx = cv2.getContext('2d')
416
-
417
- const actData = ctx.getImageData(0, 0, w, h)
418
-
419
- try {
420
- const rslt = await uploadCanvasImage(cv, imgname)
421
- if (rslt !== 'OK') {
422
- console.log('upload result:' + rslt)
423
- }
424
- } catch (ex) {
425
- console.log('upload error')
426
- console.log(ex)
427
- }
428
-
429
- cmpImageData(actData, data)
430
- }).timeout(5000)
431
-
432
- it('check move,rotate,visible,color,width', async () => {
433
- document.body.innerHTML = __html__[htmlPath+'canvas_basic.html']
434
- const imgname = 'canvas_test4.png'
435
- const code = `カメ描画先=「turtle_cv」
436
- 10にカメ速度設定。
437
- カメ画像URLは、「turtle.png」
438
- カメ作成。それのsmoothオフ。-1のsmoothオフ。
439
- [10,10]にカメ起点移動。90にカメ角度設定。
440
- 3にカメペンサイズ設定。
441
- 「#000」にカメペン色設定。
442
- 10だけカメ進む。
443
- 30だけカメ右回転。
444
- 15だけカメ進む。
445
- 45だけカメ左回転。
446
- 0にカメペン設定。
447
- 15だけカメ戻る。
448
- 1にカメペン設定。
449
- 2にカメペンサイズ設定。
450
- 「#F00」にカメペン色設定。
451
- 15だけカメ右回転。
452
- 30だけカメ進む。
453
- カメ非表示。
454
- カメ画像URLは、「turtle-elephant.png」
455
- カメ作成。それのsmoothオフ。それにカメ操作対象設定。
456
- [75,75]にカメ起点移動。
457
- カメ非表示。
458
- カメ表示。
459
- `
460
- nako.run(code)
461
-
462
- await waitTimer(2.0)
463
-
464
- const dataPromise = getImageDataFromUrl(imagePath + imgname)
465
- const cvPromise = getMergedCanvas('turtle_cv', ['0', '1'])
466
- const [data, cv] = await Promise.all([dataPromise, cvPromise])
467
-
468
- const cv2 = await recreateCanvas(cv)
469
- const w = parseInt(cv2.width, 10)
470
- const h = parseInt(cv2.height, 10)
471
- const ctx = cv2.getContext('2d')
472
-
473
- const actData = ctx.getImageData(0, 0, w, h)
474
-
475
- try {
476
- const rslt = await uploadCanvasImage(cv, imgname)
477
- if (rslt !== 'OK') {
478
- console.log('upload result:' + rslt)
479
- }
480
- } catch (ex) {
481
- console.log('upload error')
482
- console.log(ex)
483
- }
484
-
485
- cmpImageData(actData, data)
486
- }).timeout(5000)
487
-
488
- it('check all clear', async () => {
489
- document.body.innerHTML = __html__[htmlPath+'canvas_basic.html']
490
- const imgname = 'canvas_test5_wasblank.png'
491
- const code = `カメ描画先=「turtle_cv」
492
- 0にカメ速度設定。
493
- カメ画像URLは、「turtle.png」
494
- カメ作成。それのsmoothオフ。-1のsmoothオフ。
495
- [25,75]にカメ起点移動。
496
- [75,25]にカメ移動。
497
- 135だけカメ左回転
498
- 50だけカメ進む
499
- 90だけカメ左回転
500
- 25だけカメ進む
501
- カメ表示。
502
- カメ全消去。
503
- `
504
- nako.run(code)
505
-
506
- await waitTimer(1.0)
507
-
508
- const dataPromise = getImageDataFromUrl(imagePath + 'canvas_test_blank.png')
509
- const cvPromise = getMergedCanvas('turtle_cv', ['0', '1', '2'])
510
- const [data, cv] = await Promise.all([dataPromise, cvPromise])
511
-
512
- const cv2 = await recreateCanvas(cv)
513
- const w = parseInt(cv2.width, 10)
514
- const h = parseInt(cv2.height, 10)
515
- const ctx = cv2.getContext('2d')
516
-
517
- const actData = ctx.getImageData(0, 0, w, h)
518
-
519
- try {
520
- const rslt = await uploadCanvasImage(cv, imgname)
521
- if (rslt !== 'OK') {
522
- console.log('upload result:' + rslt)
523
- }
524
- } catch (ex) {
525
- console.log('upload error')
526
- console.log(ex)
527
- }
528
-
529
- cmpImageData(actData, data)
530
- })
531
-
532
- it('check all clear before use', async () => {
533
- document.body.innerHTML = __html__[htmlPath+'canvas_basic.html']
534
- const imgname = 'canvas_test6_wasblank.png'
535
- const code = `カメ描画先=「turtle_cv」
536
- カメ画像URLは、「turtle.png」
537
- 0にカメ速度設定。
538
- カメ全消去。
539
- `
540
- nako.run(code)
541
-
542
- await waitTimer(1.0)
543
-
544
- const dataPromise = getImageDataFromUrl(imagePath + 'canvas_test_blank.png')
545
- const cvPromise = getMergedCanvas('turtle_cv', ['0', '1', '2'])
546
- const [data, cv] = await Promise.all([dataPromise, cvPromise])
547
-
548
- const cv2 = await recreateCanvas(cv)
549
- const w = parseInt(cv2.width, 10)
550
- const h = parseInt(cv2.height, 10)
551
- const ctx = cv2.getContext('2d')
552
-
553
- const actData = ctx.getImageData(0, 0, w, h)
554
-
555
- try {
556
- const rslt = await uploadCanvasImage(cv, imgname)
557
- if (rslt !== 'OK') {
558
- console.log('upload result:' + rslt)
559
- }
560
- } catch (ex) {
561
- console.log('upload error')
562
- console.log(ex)
563
- }
564
-
565
- cmpImageData(actData, data)
566
- }).timeout(5000)
567
-
568
- it('change turtle image delayed', async () => {
569
- document.body.innerHTML = __html__[htmlPath+'canvas_basic.html']
570
- const imgname = 'canvas_test7.png'
571
- const code = `カメ描画先=「turtle_cv」
572
- 1にカメ速度設定。
573
- カメ画像URLは、「turtle.png」
574
- カメ作成。それのsmoothオフ。-1のsmoothオフ。
575
- 「/custom/delayedimage/turtle-elephant.png」にカメ画像変更
576
- [50,50]にカメ起点移動。270にカメ角度設定。
577
- `
578
- nako.run(code)
579
-
580
- await waitTimer(2.0)
581
-
582
- const dataPromise = getImageDataFromUrl(imagePath + imgname)
583
- const cvPromise = getMergedCanvas('turtle_cv', ['0'])
584
- const [data, cv] = await Promise.all([dataPromise, cvPromise])
585
-
586
- const cv2 = await recreateCanvas(cv)
587
- const w = parseInt(cv2.width, 10)
588
- const h = parseInt(cv2.height, 10)
589
- const ctx = cv2.getContext('2d')
590
-
591
- const actData = ctx.getImageData(0, 0, w, h)
592
-
593
- try {
594
- const rslt = await uploadCanvasImage(cv, imgname)
595
- if (rslt !== 'OK') {
596
- console.log('upload result:' + rslt)
597
- }
598
- } catch (ex) {
599
- console.log('upload error')
600
- console.log(ex)
601
- }
602
-
603
- cmpImageData(actData, data)
604
- }).timeout(5000)
605
-
606
- it('change turtle image', async () => {
607
- document.body.innerHTML = __html__[htmlPath+'canvas_basic.html']
608
- const imgname = 'canvas_test8.png'
609
- const code = `カメ描画先=「turtle_cv」
610
- 0にカメ速度設定。
611
- カメ画像URLは、「turtle.png」
612
- カメ作成。それのsmoothオフ。-1のsmoothオフ。
613
- [50,50]にカメ起点移動。270にカメ角度設定。
614
- 「turtle-elephant.png」にカメ画像変更
615
- `
616
- nako.run(code)
617
-
618
- await waitTimer(2.0)
619
-
620
- const dataPromise = getImageDataFromUrl(imagePath + imgname)
621
- const cvPromise = getMergedCanvas('turtle_cv', ['0'])
622
- const [data, cv] = await Promise.all([dataPromise, cvPromise])
623
-
624
- const cv2 = await recreateCanvas(cv)
625
- const w = parseInt(cv2.width, 10)
626
- const h = parseInt(cv2.height, 10)
627
- const ctx = cv2.getContext('2d')
628
-
629
- const actData = ctx.getImageData(0, 0, w, h)
630
-
631
- try {
632
- const rslt = await uploadCanvasImage(cv, imgname)
633
- if (rslt !== 'OK') {
634
- console.log('upload result:' + rslt)
635
- }
636
- } catch (ex) {
637
- console.log('upload error')
638
- console.log(ex)
639
- }
640
-
641
- cmpImageData(actData, data)
642
- }).timeout(5000)
643
-
644
- it('load fail turtle image', async () => {
645
- document.body.innerHTML = __html__[htmlPath+'canvas_basic.html']
646
- const imgname = 'canvas_test9_wasblank.png'
647
- const code = `カメ描画先=「turtle_cv」
648
- 0にカメ速度設定。
649
- カメ画像URLは、「no_exists.png」
650
- カメ作成。
651
- [50,50]にカメ起点移動。270にカメ角度設定。
652
- `
653
- nako.run(code)
654
-
655
- await waitTimer(2.0)
656
-
657
- const dataPromise = getImageDataFromUrl(imagePath + 'canvas_test_blank.png')
658
- const cvPromise = getMergedCanvas('turtle_cv', ['0'])
659
- const [data, cv] = await Promise.all([dataPromise, cvPromise])
660
-
661
- const cv2 = await recreateCanvas(cv)
662
- const w = parseInt(cv2.width, 10)
663
- const h = parseInt(cv2.height, 10)
664
- const ctx = cv2.getContext('2d')
665
-
666
- const actData = ctx.getImageData(0, 0, w, h)
667
-
668
- try {
669
- const rslt = await uploadCanvasImage(cv, imgname)
670
- if (rslt !== 'OK') {
671
- console.log('upload result:' + rslt)
672
- }
673
- } catch (ex) {
674
- console.log('upload error')
675
- console.log(ex)
676
- }
677
-
678
- cmpImageData(actData, data)
679
- }).timeout(5000)
680
-
681
- it('click turtle', async () => {
682
- document.body.innerHTML = __html__[htmlPath+'canvas_basic.html']
683
- const code = `カメ描画先=「turtle_cv」
684
- 0にカメ速度設定。
685
- カメ作成。
686
- [50,50]にカメ起点移動。270にカメ角度設定。
687
- カメクリック時には、
688
- 対象を報告
689
- ここまで
690
- `
691
- let funcCalled = false
692
- let target = -1
693
- nako.addFunc('報告', [['を']], (ttcanvas) => {
694
- funcCalled = true
695
- target = ttcanvas
696
- })
697
- nako.run(code)
698
-
699
- await waitTimer(1.0)
700
- const e = document.getElementById('0')
701
- if (e) {
702
- e.click()
703
- }
704
- await waitTimer(1.0)
705
-
706
- assert.ok(funcCalled, 'イベント呼び出し')
707
- assert.strictEqual(target.id, '0', 'イベント対象のカメID')
708
- }).timeout(5000)
709
-
710
- it('turtle image basic(turtle)', async () => {
711
- document.body.innerHTML = __html__[htmlPath+'canvas_basic.html']
712
- const imgname = 'canvas_test10_wasnotblank.png'
713
- const code = `カメ描画先=「turtle_cv」
714
- 0にカメ速度設定。
715
- カメ作成。
716
- [50,50]にカメ起点移動。90にカメ角度設定。
717
- `
718
- nako.run(code)
719
-
720
- await waitTimer(1.0)
721
-
722
- const dataPromise = getImageDataFromUrl(imagePath + 'canvas_test_blank.png')
723
- const cvPromise = getMergedCanvas('turtle_cv', ['0'])
724
- const [data, cv] = await Promise.all([dataPromise, cvPromise])
725
-
726
- const cv2 = await recreateCanvas(cv)
727
- const w = parseInt(cv2.width, 10)
728
- const h = parseInt(cv2.height, 10)
729
- const ctx = cv2.getContext('2d')
730
-
731
- const actData = ctx.getImageData(0, 0, w, h)
732
-
733
- try {
734
- const rslt = await uploadCanvasImage(cv, imgname)
735
- if (rslt !== 'OK') {
736
- console.log('upload result:' + rslt)
737
- }
738
- } catch (ex) {
739
- console.log('upload error')
740
- console.log(ex)
741
- }
742
-
743
- cmpImageData(actData, data, true)
744
- }).timeout(5000)
745
-
746
- it('turtle image extend(elephant)', async () => {
747
- document.body.innerHTML = __html__[htmlPath+'canvas_basic.html']
748
- const imgname = 'canvas_test11_wasnotblank.png'
749
- const code = `カメ描画先=「turtle_cv」
750
- 0にカメ速度設定。
751
- ゾウ作成。
752
- [50,50]にカメ起点移動。270にカメ角度設定。
753
- `
754
- nako.run(code)
755
-
756
- await waitTimer(1.0)
757
-
758
- const dataPromise = getImageDataFromUrl(imagePath + 'canvas_test_blank.png')
759
- const cvPromise = getMergedCanvas('turtle_cv', ['0'])
760
- const [data, cv] = await Promise.all([dataPromise, cvPromise])
761
-
762
- const cv2 = await recreateCanvas(cv)
763
- const w = parseInt(cv2.width, 10)
764
- const h = parseInt(cv2.height, 10)
765
- const ctx = cv2.getContext('2d')
766
-
767
- const actData = ctx.getImageData(0, 0, w, h)
768
-
769
- try {
770
- const rslt = await uploadCanvasImage(cv, imgname)
771
- if (rslt !== 'OK') {
772
- console.log('upload result:' + rslt)
773
- }
774
- } catch (ex) {
775
- console.log('upload error')
776
- console.log(ex)
777
- }
778
-
779
- cmpImageData(actData, data, true)
780
- }).timeout(5000)
781
-
782
- it('turtle image extend(panda)', async () => {
783
- document.body.innerHTML = __html__[htmlPath+'canvas_basic.html']
784
- const imgname = 'canvas_test12_wasnotblank.png'
785
- const code = `カメ描画先=「turtle_cv」
786
- 0にカメ速度設定。
787
- パンダ作成。
788
- [50,50]にカメ起点移動。180にカメ角度設定。
789
- `
790
- nako.run(code)
791
-
792
- await waitTimer(1.0)
793
-
794
- const dataPromise = getImageDataFromUrl(imagePath + 'canvas_test_blank.png')
795
- const cvPromise = getMergedCanvas('turtle_cv', ['0'])
796
- const [data, cv] = await Promise.all([dataPromise, cvPromise])
797
-
798
- const cv2 = await recreateCanvas(cv)
799
- const w = parseInt(cv2.width, 10)
800
- const h = parseInt(cv2.height, 10)
801
- const ctx = cv2.getContext('2d')
802
-
803
- const actData = ctx.getImageData(0, 0, w, h)
804
-
805
- try {
806
- const rslt = await uploadCanvasImage(cv, imgname)
807
- if (rslt !== 'OK') {
808
- console.log('upload result:' + rslt)
809
- }
810
- } catch (ex) {
811
- console.log('upload error')
812
- console.log(ex)
813
- }
814
-
815
- cmpImageData(actData, data, true)
816
- }).timeout(5000)
817
- })
1
+ /* global __html__ */
2
+ import { assert } from 'chai'
3
+ import { NakoCompiler } from 'nadesiko3core/src/nako3.mjs'
4
+ import { importStatus } from './import_plugin_checker.js'
5
+ import PluginTurtle from 'nako3/plugin_turtle.mjs'
6
+
7
+ const htmlPath = 'test/html/'
8
+ const imagePath = '/test/image/'
9
+
10
+ const waitTimer = (second) => {
11
+ return new Promise((resolve) => {
12
+ setTimeout(() => {
13
+ resolve()
14
+ }, second * 1000)
15
+ })
16
+ }
17
+
18
+ const setImageSmoothingEnabledToCtx = (ctx, value) => {
19
+ ctx.mozImageSmoothingEnabled = value
20
+ ctx.webkitImageSmoothingEnabled = value
21
+ ctx.msImageSmoothingEnabled = value
22
+ ctx.imageSmoothingEnabled = value
23
+ }
24
+
25
+ const getImageDataFromUrl = (url) => {
26
+ return new Promise((resolve, reject) => {
27
+ const img = new Image()
28
+ img.onload = () => {
29
+ img.onload = null
30
+ img.onerror = null
31
+ const canvas = document.createElement('canvas')
32
+ canvas.width = img.width
33
+ canvas.height = img.height
34
+ document.body.appendChild(canvas)
35
+ const ctx = canvas.getContext('2d')
36
+ ctx.clearRect(0, 0, canvas.width, canvas.height)
37
+ ctx.globalCompositeOperation = 'copy'
38
+ setImageSmoothingEnabledToCtx(ctx, false)
39
+ ctx.drawImage(img, 0, 0)
40
+
41
+ resolve(ctx.getImageData(0, 0, canvas.width, canvas.height))
42
+ }
43
+ img.onerror = () => {
44
+ img.onload = null
45
+ img.onerror = null
46
+ reject(new Error('load error'))
47
+ }
48
+ img.src = url
49
+ })
50
+ }
51
+
52
+ const getMergedCanvas = (canvasId, turtleIds) => {
53
+ return new Promise((resolve, reject) => {
54
+ const cv = document.getElementById(canvasId)
55
+
56
+ if (cv == null) {
57
+ reject(new Error('fail create canvas'))
58
+ return
59
+ }
60
+
61
+ const rect = cv.getBoundingClientRect()
62
+ const w = parseInt(cv.width, 10)
63
+ const h = parseInt(cv.height, 10)
64
+ const rx = rect.left + window.pageXOffset
65
+ const ry = rect.top + window.pageYOffset
66
+
67
+ const canvas = document.createElement('canvas')
68
+ canvas.width = w
69
+ canvas.height = h
70
+ document.body.appendChild(canvas)
71
+
72
+ const ctx = canvas.getContext('2d')
73
+ ctx.clearRect(0, 0, w, h)
74
+
75
+ const ctxm = cv.getContext('2d')
76
+ const imgdtm = ctxm.getImageData(0, 0, w, h)
77
+ ctx.globalCompositeOperation = 'copy'
78
+ setImageSmoothingEnabledToCtx(ctx, false)
79
+ ctx.putImageData(imgdtm, 0, 0)
80
+ ctx.globalCompositeOperation = 'source-over'
81
+
82
+ turtleIds.forEach(turtleId => {
83
+ const cvt = document.getElementById(turtleId)
84
+ if (cvt) {
85
+ const dx = parseInt(cvt.style.left, 10) - rx
86
+ const dy = parseInt(cvt.style.top, 10) - ry
87
+ ctx.drawImage(cvt, dx, dy)
88
+ }
89
+ })
90
+
91
+ resolve(canvas)
92
+ })
93
+ }
94
+
95
+ const recreateCanvas = src => {
96
+ return new Promise((resolve, reject) => {
97
+ const img = new Image()
98
+ img.onload = () => {
99
+ img.onload = null
100
+ img.onerror = null
101
+ const canvas = document.createElement('canvas')
102
+ canvas.width = img.width
103
+ canvas.height = img.height
104
+ const ctx = canvas.getContext('2d')
105
+ setImageSmoothingEnabledToCtx(ctx, false)
106
+ ctx.clearRect(0, 0, img.width, img.height)
107
+ ctx.globalCompositeOperation = 'copy'
108
+ ctx.drawImage(img, 0, 0)
109
+ ctx.globalCompositeOperation = 'source-over'
110
+
111
+ resolve(canvas)
112
+ }
113
+ img.onerror = () => {
114
+ img.onload = null
115
+ img.onerror = null
116
+ reject(new Error('reload image error'))
117
+ }
118
+ img.src = src.toDataURL('image/png')
119
+ })
120
+ }
121
+
122
+ const getBlobFromCanvas = (canvas, type) => {
123
+ const dataurl = canvas.toDataURL(type)
124
+ const bin = atob(dataurl.replace(/^.*,/, ''))
125
+ const buffer = new Uint8Array(bin.length)
126
+ for (let i = 0, l = bin.length; i < l; i++) {
127
+ buffer[i] = bin.charCodeAt(i)
128
+ }
129
+ const blob = new Blob([buffer.buffer], {
130
+ type: type
131
+ })
132
+ return blob
133
+ }
134
+
135
+ const uploadCanvasImage = async (canvas, filename) => {
136
+ var formData = new FormData()
137
+ const blob = getBlobFromCanvas(canvas, 'image/png')
138
+ formData.append('file', blob, filename)
139
+ const response = await fetch('/custom/uploadimage', {
140
+ method: 'POST',
141
+ body: formData
142
+ })
143
+ const result = await response.text()
144
+ return result
145
+ }
146
+
147
+ describe('plugin_turtle_test', () => {
148
+ let nako = null
149
+ const matchmode = 'lazzy3'
150
+ const cmpImageData = (imgData1, imgData2, invert) => {
151
+ if (typeof invert === 'undefined') { invert = false }
152
+ assert.strictEqual(imgData1.width, imgData2.width, 'images width unmatch')
153
+ assert.strictEqual(imgData1.height, imgData2.height, 'images height unmatch')
154
+
155
+ console.log(`brute match(${matchmode})`)
156
+ let allmatch = true
157
+ let i = 0
158
+ for (let y = 0; y < imgData1.height; y++) {
159
+ for (let x = 0; x < imgData1.width; x++, i += 4) {
160
+ let nomatch = false
161
+ if (matchmode === 'lazzy') {
162
+ if ((imgData2.data[i + 3] !== 0 &&
163
+ (Math.abs(imgData1.data[i + 0] - imgData2.data[i + 0]) > 10 ||
164
+ Math.abs(imgData1.data[i + 1] - imgData2.data[i + 1]) > 10 ||
165
+ Math.abs(imgData1.data[i + 2] - imgData2.data[i + 2]) > 10 ||
166
+ Math.abs(imgData1.data[i + 3] - imgData2.data[i + 3]) > 10)) ||
167
+ (imgData2.data[i + 3] === 0 && imgData1.data[i + 3] > 5)) {
168
+ nomatch = true
169
+ }
170
+ } else
171
+ if (matchmode === 'lazzy2') {
172
+ if (imgData1.data[i + 3] === 255 &&
173
+ imgData2.data[i + 3] === 255 &&
174
+ (Math.abs(imgData1.data[i + 0] - imgData2.data[i + 0]) > 1 ||
175
+ Math.abs(imgData1.data[i + 1] - imgData2.data[i + 1]) > 1 ||
176
+ Math.abs(imgData1.data[i + 2] - imgData2.data[i + 2]) > 1)) {
177
+ // 両画像が不透明なら、差異は各パレットで1まで
178
+ nomatch = true
179
+ } else
180
+ if (imgData1.data[i + 3] === 0 &&
181
+ imgData2.data[i + 3] === 255) {
182
+ // 画像1が完全透明なら、画像2が不透明ではいけない。
183
+ nomatch = true
184
+ } else
185
+ if (imgData1.data[i + 3] === 255 &&
186
+ imgData2.data[i + 3] === 0) {
187
+ // 画像1が不透明なら、画像2が完全透明ではいけない。
188
+ nomatch = true
189
+ }
190
+ } else
191
+ if (matchmode === 'lazzy3') {
192
+ if (imgData1.data[i + 3] === 255 &&
193
+ imgData2.data[i + 3] === 255 &&
194
+ (Math.abs(imgData1.data[i + 0] - imgData2.data[i + 0]) > 48 ||
195
+ Math.abs(imgData1.data[i + 1] - imgData2.data[i + 1]) > 48 ||
196
+ Math.abs(imgData1.data[i + 2] - imgData2.data[i + 2]) > 48)) {
197
+ // 両画像が不透明なら、差異は各パレットで48まで
198
+ nomatch = true
199
+ } else
200
+ if (imgData1.data[i + 3] === 0 &&
201
+ imgData2.data[i + 3] === 255) {
202
+ // 画像1が完全透明なら、画像2が不透明ではいけない。
203
+ nomatch = true
204
+ } else
205
+ if (imgData1.data[i + 3] === 255 &&
206
+ imgData2.data[i + 3] === 0) {
207
+ // 画像1が不透明なら、画像2が完全透明ではいけない。
208
+ nomatch = true
209
+ }
210
+ } else {
211
+ if (imgData1.data[i + 0] !== imgData2.data[i + 0] ||
212
+ imgData1.data[i + 1] !== imgData2.data[i + 1] ||
213
+ imgData1.data[i + 2] !== imgData2.data[i + 2] ||
214
+ imgData1.data[i + 3] !== imgData2.data[i + 3]) {
215
+ nomatch = true
216
+ }
217
+ }
218
+ if (nomatch) {
219
+ allmatch = false
220
+ if (!invert) {
221
+ console.log('unmacth(' + x + ',' + y + '):' +
222
+ (imgData1.data[i + 0] < 16 ? '0' : '') + (0 + imgData1.data[i + 0]).toString(16) +
223
+ (imgData1.data[i + 1] < 16 ? '0' : '') + (0 + imgData1.data[i + 1]).toString(16) +
224
+ (imgData1.data[i + 2] < 16 ? '0' : '') + (0 + imgData1.data[i + 2]).toString(16) +
225
+ (imgData1.data[i + 3] < 16 ? '0' : '') + (0 + imgData1.data[i + 3]).toString(16) +
226
+ ':' +
227
+ (imgData2.data[i + 0] < 16 ? '0' : '') + (0 + imgData2.data[i + 0]).toString(16) +
228
+ (imgData2.data[i + 1] < 16 ? '0' : '') + (0 + imgData2.data[i + 1]).toString(16) +
229
+ (imgData2.data[i + 2] < 16 ? '0' : '') + (0 + imgData2.data[i + 2]).toString(16) +
230
+ (imgData2.data[i + 3] < 16 ? '0' : '') + (0 + imgData2.data[i + 3]).toString(16) +
231
+ ',' + i)
232
+ }
233
+ }
234
+ }
235
+ }
236
+ if (!invert) {
237
+ assert.ok(allmatch, 'image nomatch')
238
+ } else {
239
+ assert.ok(!allmatch, 'image match')
240
+ }
241
+ }
242
+ afterEach(() => {
243
+ nako = null
244
+ })
245
+ beforeEach(() => {
246
+ nako = new NakoCompiler()
247
+ // const pluginClone = Object.assign({}, PluginTurtle)
248
+ // nako.addPluginFile('PluginTurtle', 'plugin_turtle.js', pluginClone)
249
+ nako.addPluginFile('PluginTurtle', 'plugin_turtle.js', PluginTurtle)
250
+ nako.addFunc('getElementyID', [['の']], (id, sys) => {
251
+ return document.getElementById(id)
252
+ })
253
+ nako.addFunc('smoothオフ', [['の']], (id, sys) => {
254
+ let ctx = null
255
+ if (id >= 0) {
256
+ if (id < sys._turtle.list.length) {
257
+ ctx = sys._turtle.list[id].ctx
258
+ }
259
+ } else {
260
+ ctx = sys._turtle.ctx
261
+ }
262
+ if (ctx) {
263
+ setImageSmoothingEnabledToCtx(ctx, false)
264
+ }
265
+ return id
266
+ })
267
+ })
268
+
269
+ // --- test ---
270
+ it('check env(canvas_basic.html)', () => {
271
+ document.body.innerHTML = __html__[htmlPath+'canvas_basic.html']
272
+ const canvasElement = document.getElementById('turtle_cv')
273
+ assert.strictEqual(typeof (canvasElement), 'object')
274
+ })
275
+
276
+ it('auto import for browser', () => {
277
+ const pluginName = 'PluginTurtle'
278
+ const imported = importStatus.hasImport(pluginName)
279
+ assert.ok(imported, 'was import')
280
+ const autoImport = importStatus.getAutoImport(pluginName)
281
+ assert.strictEqual(typeof (autoImport.obj), 'object')
282
+ })
283
+
284
+ it('set origin and direcion', async () => {
285
+ document.body.innerHTML = __html__[htmlPath+'canvas_basic.html']
286
+ const imgname = 'canvas_test1.png'
287
+ const code = `カメ描画先=「turtle_cv」
288
+ 0にカメ速度設定。
289
+ カメ画像URLは、「turtle.png」
290
+ カメ作成。それのsmoothオフ。-1のsmoothオフ。
291
+ [30,30]にカメ起点移動。
292
+ 180にカメ角度設定。
293
+ `
294
+ nako.run(code)
295
+
296
+ await waitTimer(1)
297
+
298
+ const dataPromise = getImageDataFromUrl(imagePath + imgname)
299
+ const cvPromise = getMergedCanvas('turtle_cv', ['0'])
300
+ const [data, cv] = await Promise.all([dataPromise, cvPromise])
301
+
302
+ const cv2 = await recreateCanvas(cv)
303
+ const w = parseInt(cv2.width, 10)
304
+ const h = parseInt(cv2.height, 10)
305
+ const ctx = cv2.getContext('2d')
306
+
307
+ const actData = ctx.getImageData(0, 0, w, h)
308
+
309
+ try {
310
+ const rslt = await uploadCanvasImage(cv, imgname)
311
+ if (rslt !== 'OK') {
312
+ console.log('upload result:' + rslt)
313
+ }
314
+ } catch (ex) {
315
+ console.log('upload error')
316
+ console.log(ex)
317
+ }
318
+
319
+ cmpImageData(actData, data)
320
+ }).timeout(5000)
321
+
322
+ it('incorrect canvasid', () => {
323
+ document.body.innerHTML = __html__[htmlPath+'canvas_basic.html']
324
+ const code = `カメ描画先=「no_cv」
325
+ 0にカメ速度設定。
326
+ カメ作成。
327
+ `
328
+ assert.throws(() => { nako.run(code) }, Error)
329
+ })
330
+
331
+ it('incorrect command', () => {
332
+ document.body.innerHTML = __html__[htmlPath+'canvas_basic.html']
333
+ const code = `カメ描画先=「turtle_cv」
334
+ 0にカメ速度設定。
335
+ [25,25]にカメ起点移動。
336
+ `
337
+ assert.throws(() => { nako.run(code) }, Error)
338
+ })
339
+
340
+ it('set origin and direcion turtles', async () => {
341
+ document.body.innerHTML = __html__[htmlPath+'canvas_basic.html']
342
+ const imgname = 'canvas_test2.png'
343
+ const code = `カメ描画先=「#turtle_cv」
344
+ 0にカメ速度設定。
345
+ カメ画像URLは、「turtle.png」
346
+ カメ作成。それのsmoothオフ。-1のsmoothオフ。
347
+ [25,25]にカメ起点移動。
348
+ 180にカメ角度設定。
349
+ カメ画像URLは、「turtle-elephant.png」
350
+ カメ作成。それのsmoothオフ。それにカメ操作対象設定。
351
+ [75,25]にカメ起点移動。
352
+ 90にカメ角度設定。
353
+ カメ画像URLは、「turtle-panda.png」
354
+ カメ作成。それのsmoothオフ。それにカメ操作対象設定。
355
+ [50,75]にカメ起点移動。
356
+ 0にカメ角度設定。
357
+ `
358
+ nako.run(code)
359
+
360
+ await waitTimer(1.8)
361
+
362
+ const dataPromise = getImageDataFromUrl(imagePath + imgname)
363
+ const cvPromise = getMergedCanvas('turtle_cv', ['0', '1', '2'])
364
+ const [data, cv] = await Promise.all([dataPromise, cvPromise])
365
+
366
+ const cv2 = await recreateCanvas(cv)
367
+ const w = parseInt(cv2.width, 10)
368
+ const h = parseInt(cv2.height, 10)
369
+ const ctx = cv2.getContext('2d')
370
+
371
+ const actData = ctx.getImageData(0, 0, w, h)
372
+
373
+ try {
374
+ const rslt = await uploadCanvasImage(cv, imgname)
375
+ if (rslt !== 'OK') {
376
+ console.log('upload result:' + rslt)
377
+ }
378
+ } catch (ex) {
379
+ console.log('upload error')
380
+ console.log(ex)
381
+ }
382
+
383
+ cmpImageData(actData, data)
384
+ }).timeout(5000)
385
+
386
+ it('move direct position turtles', async () => {
387
+ document.body.innerHTML = __html__[htmlPath+'canvas_basic.html']
388
+ const imgname = 'canvas_test3.png'
389
+ const code = `カメ描画先=「turtle_cv」のgetElementyID
390
+ 0にカメ速度設定。
391
+ カメ画像URLは、「turtle.png」
392
+ カメ作成。それのsmoothオフ。-1のsmoothオフ。
393
+ [25,75]にカメ起点移動。
394
+ [25,25]にカメ移動。
395
+ カメ画像URLは、「turtle-elephant.png」
396
+ カメ作成。それのsmoothオフ。それにカメ操作対象設定。
397
+ [25,75]にカメ起点移動。
398
+ [75,75]にカメ移動。
399
+ カメ画像URLは、「turtle-panda.png」
400
+ カメ作成。それのsmoothオフ。それにカメ操作対象設定。
401
+ [25,75]にカメ起点移動。
402
+ [75,25]にカメ移動。
403
+ `
404
+ nako.run(code)
405
+
406
+ await waitTimer(2.0)
407
+
408
+ const dataPromise = getImageDataFromUrl(imagePath + imgname)
409
+ const cvPromise = getMergedCanvas('turtle_cv', ['0', '1', '2'])
410
+ const [data, cv] = await Promise.all([dataPromise, cvPromise])
411
+
412
+ const cv2 = await recreateCanvas(cv)
413
+ const w = parseInt(cv2.width, 10)
414
+ const h = parseInt(cv2.height, 10)
415
+ const ctx = cv2.getContext('2d')
416
+
417
+ const actData = ctx.getImageData(0, 0, w, h)
418
+
419
+ try {
420
+ const rslt = await uploadCanvasImage(cv, imgname)
421
+ if (rslt !== 'OK') {
422
+ console.log('upload result:' + rslt)
423
+ }
424
+ } catch (ex) {
425
+ console.log('upload error')
426
+ console.log(ex)
427
+ }
428
+
429
+ cmpImageData(actData, data)
430
+ }).timeout(5000)
431
+
432
+ it('check move,rotate,visible,color,width', async () => {
433
+ document.body.innerHTML = __html__[htmlPath+'canvas_basic.html']
434
+ const imgname = 'canvas_test4.png'
435
+ const code = `カメ描画先=「turtle_cv」
436
+ 10にカメ速度設定。
437
+ カメ画像URLは、「turtle.png」
438
+ カメ作成。それのsmoothオフ。-1のsmoothオフ。
439
+ [10,10]にカメ起点移動。90にカメ角度設定。
440
+ 3にカメペンサイズ設定。
441
+ 「#000」にカメペン色設定。
442
+ 10だけカメ進む。
443
+ 30だけカメ右回転。
444
+ 15だけカメ進む。
445
+ 45だけカメ左回転。
446
+ 0にカメペン設定。
447
+ 15だけカメ戻る。
448
+ 1にカメペン設定。
449
+ 2にカメペンサイズ設定。
450
+ 「#F00」にカメペン色設定。
451
+ 15だけカメ右回転。
452
+ 30だけカメ進む。
453
+ カメ非表示。
454
+ カメ画像URLは、「turtle-elephant.png」
455
+ カメ作成。それのsmoothオフ。それにカメ操作対象設定。
456
+ [75,75]にカメ起点移動。
457
+ カメ非表示。
458
+ カメ表示。
459
+ `
460
+ nako.run(code)
461
+
462
+ await waitTimer(2.0)
463
+
464
+ const dataPromise = getImageDataFromUrl(imagePath + imgname)
465
+ const cvPromise = getMergedCanvas('turtle_cv', ['0', '1'])
466
+ const [data, cv] = await Promise.all([dataPromise, cvPromise])
467
+
468
+ const cv2 = await recreateCanvas(cv)
469
+ const w = parseInt(cv2.width, 10)
470
+ const h = parseInt(cv2.height, 10)
471
+ const ctx = cv2.getContext('2d')
472
+
473
+ const actData = ctx.getImageData(0, 0, w, h)
474
+
475
+ try {
476
+ const rslt = await uploadCanvasImage(cv, imgname)
477
+ if (rslt !== 'OK') {
478
+ console.log('upload result:' + rslt)
479
+ }
480
+ } catch (ex) {
481
+ console.log('upload error')
482
+ console.log(ex)
483
+ }
484
+
485
+ cmpImageData(actData, data)
486
+ }).timeout(5000)
487
+
488
+ it('check all clear', async () => {
489
+ document.body.innerHTML = __html__[htmlPath+'canvas_basic.html']
490
+ const imgname = 'canvas_test5_wasblank.png'
491
+ const code = `カメ描画先=「turtle_cv」
492
+ 0にカメ速度設定。
493
+ カメ画像URLは、「turtle.png」
494
+ カメ作成。それのsmoothオフ。-1のsmoothオフ。
495
+ [25,75]にカメ起点移動。
496
+ [75,25]にカメ移動。
497
+ 135だけカメ左回転
498
+ 50だけカメ進む
499
+ 90だけカメ左回転
500
+ 25だけカメ進む
501
+ カメ表示。
502
+ カメ全消去。
503
+ `
504
+ nako.run(code)
505
+
506
+ await waitTimer(1.0)
507
+
508
+ const dataPromise = getImageDataFromUrl(imagePath + 'canvas_test_blank.png')
509
+ const cvPromise = getMergedCanvas('turtle_cv', ['0', '1', '2'])
510
+ const [data, cv] = await Promise.all([dataPromise, cvPromise])
511
+
512
+ const cv2 = await recreateCanvas(cv)
513
+ const w = parseInt(cv2.width, 10)
514
+ const h = parseInt(cv2.height, 10)
515
+ const ctx = cv2.getContext('2d')
516
+
517
+ const actData = ctx.getImageData(0, 0, w, h)
518
+
519
+ try {
520
+ const rslt = await uploadCanvasImage(cv, imgname)
521
+ if (rslt !== 'OK') {
522
+ console.log('upload result:' + rslt)
523
+ }
524
+ } catch (ex) {
525
+ console.log('upload error')
526
+ console.log(ex)
527
+ }
528
+
529
+ cmpImageData(actData, data)
530
+ })
531
+
532
+ it('check all clear before use', async () => {
533
+ document.body.innerHTML = __html__[htmlPath+'canvas_basic.html']
534
+ const imgname = 'canvas_test6_wasblank.png'
535
+ const code = `カメ描画先=「turtle_cv」
536
+ カメ画像URLは、「turtle.png」
537
+ 0にカメ速度設定。
538
+ カメ全消去。
539
+ `
540
+ nako.run(code)
541
+
542
+ await waitTimer(1.0)
543
+
544
+ const dataPromise = getImageDataFromUrl(imagePath + 'canvas_test_blank.png')
545
+ const cvPromise = getMergedCanvas('turtle_cv', ['0', '1', '2'])
546
+ const [data, cv] = await Promise.all([dataPromise, cvPromise])
547
+
548
+ const cv2 = await recreateCanvas(cv)
549
+ const w = parseInt(cv2.width, 10)
550
+ const h = parseInt(cv2.height, 10)
551
+ const ctx = cv2.getContext('2d')
552
+
553
+ const actData = ctx.getImageData(0, 0, w, h)
554
+
555
+ try {
556
+ const rslt = await uploadCanvasImage(cv, imgname)
557
+ if (rslt !== 'OK') {
558
+ console.log('upload result:' + rslt)
559
+ }
560
+ } catch (ex) {
561
+ console.log('upload error')
562
+ console.log(ex)
563
+ }
564
+
565
+ cmpImageData(actData, data)
566
+ }).timeout(5000)
567
+
568
+ it('change turtle image delayed', async () => {
569
+ document.body.innerHTML = __html__[htmlPath+'canvas_basic.html']
570
+ const imgname = 'canvas_test7.png'
571
+ const code = `カメ描画先=「turtle_cv」
572
+ 1にカメ速度設定。
573
+ カメ画像URLは、「turtle.png」
574
+ カメ作成。それのsmoothオフ。-1のsmoothオフ。
575
+ 「/custom/delayedimage/turtle-elephant.png」にカメ画像変更
576
+ [50,50]にカメ起点移動。270にカメ角度設定。
577
+ `
578
+ nako.run(code)
579
+
580
+ await waitTimer(2.0)
581
+
582
+ const dataPromise = getImageDataFromUrl(imagePath + imgname)
583
+ const cvPromise = getMergedCanvas('turtle_cv', ['0'])
584
+ const [data, cv] = await Promise.all([dataPromise, cvPromise])
585
+
586
+ const cv2 = await recreateCanvas(cv)
587
+ const w = parseInt(cv2.width, 10)
588
+ const h = parseInt(cv2.height, 10)
589
+ const ctx = cv2.getContext('2d')
590
+
591
+ const actData = ctx.getImageData(0, 0, w, h)
592
+
593
+ try {
594
+ const rslt = await uploadCanvasImage(cv, imgname)
595
+ if (rslt !== 'OK') {
596
+ console.log('upload result:' + rslt)
597
+ }
598
+ } catch (ex) {
599
+ console.log('upload error')
600
+ console.log(ex)
601
+ }
602
+
603
+ cmpImageData(actData, data)
604
+ }).timeout(5000)
605
+
606
+ it('change turtle image', async () => {
607
+ document.body.innerHTML = __html__[htmlPath+'canvas_basic.html']
608
+ const imgname = 'canvas_test8.png'
609
+ const code = `カメ描画先=「turtle_cv」
610
+ 0にカメ速度設定。
611
+ カメ画像URLは、「turtle.png」
612
+ カメ作成。それのsmoothオフ。-1のsmoothオフ。
613
+ [50,50]にカメ起点移動。270にカメ角度設定。
614
+ 「turtle-elephant.png」にカメ画像変更
615
+ `
616
+ nako.run(code)
617
+
618
+ await waitTimer(2.0)
619
+
620
+ const dataPromise = getImageDataFromUrl(imagePath + imgname)
621
+ const cvPromise = getMergedCanvas('turtle_cv', ['0'])
622
+ const [data, cv] = await Promise.all([dataPromise, cvPromise])
623
+
624
+ const cv2 = await recreateCanvas(cv)
625
+ const w = parseInt(cv2.width, 10)
626
+ const h = parseInt(cv2.height, 10)
627
+ const ctx = cv2.getContext('2d')
628
+
629
+ const actData = ctx.getImageData(0, 0, w, h)
630
+
631
+ try {
632
+ const rslt = await uploadCanvasImage(cv, imgname)
633
+ if (rslt !== 'OK') {
634
+ console.log('upload result:' + rslt)
635
+ }
636
+ } catch (ex) {
637
+ console.log('upload error')
638
+ console.log(ex)
639
+ }
640
+
641
+ cmpImageData(actData, data)
642
+ }).timeout(5000)
643
+
644
+ it('load fail turtle image', async () => {
645
+ document.body.innerHTML = __html__[htmlPath+'canvas_basic.html']
646
+ const imgname = 'canvas_test9_wasblank.png'
647
+ const code = `カメ描画先=「turtle_cv」
648
+ 0にカメ速度設定。
649
+ カメ画像URLは、「no_exists.png」
650
+ カメ作成。
651
+ [50,50]にカメ起点移動。270にカメ角度設定。
652
+ `
653
+ nako.run(code)
654
+
655
+ await waitTimer(2.0)
656
+
657
+ const dataPromise = getImageDataFromUrl(imagePath + 'canvas_test_blank.png')
658
+ const cvPromise = getMergedCanvas('turtle_cv', ['0'])
659
+ const [data, cv] = await Promise.all([dataPromise, cvPromise])
660
+
661
+ const cv2 = await recreateCanvas(cv)
662
+ const w = parseInt(cv2.width, 10)
663
+ const h = parseInt(cv2.height, 10)
664
+ const ctx = cv2.getContext('2d')
665
+
666
+ const actData = ctx.getImageData(0, 0, w, h)
667
+
668
+ try {
669
+ const rslt = await uploadCanvasImage(cv, imgname)
670
+ if (rslt !== 'OK') {
671
+ console.log('upload result:' + rslt)
672
+ }
673
+ } catch (ex) {
674
+ console.log('upload error')
675
+ console.log(ex)
676
+ }
677
+
678
+ cmpImageData(actData, data)
679
+ }).timeout(5000)
680
+
681
+ it('click turtle', async () => {
682
+ document.body.innerHTML = __html__[htmlPath+'canvas_basic.html']
683
+ const code = `カメ描画先=「turtle_cv」
684
+ 0にカメ速度設定。
685
+ カメ作成。
686
+ [50,50]にカメ起点移動。270にカメ角度設定。
687
+ カメクリック時には、
688
+ 対象を報告
689
+ ここまで
690
+ `
691
+ let funcCalled = false
692
+ let target = -1
693
+ nako.addFunc('報告', [['を']], (ttcanvas) => {
694
+ funcCalled = true
695
+ target = ttcanvas
696
+ })
697
+ nako.run(code)
698
+
699
+ await waitTimer(1.0)
700
+ const e = document.getElementById('0')
701
+ if (e) {
702
+ e.click()
703
+ }
704
+ await waitTimer(1.0)
705
+
706
+ assert.ok(funcCalled, 'イベント呼び出し')
707
+ assert.strictEqual(target.id, '0', 'イベント対象のカメID')
708
+ }).timeout(5000)
709
+
710
+ it('turtle image basic(turtle)', async () => {
711
+ document.body.innerHTML = __html__[htmlPath+'canvas_basic.html']
712
+ const imgname = 'canvas_test10_wasnotblank.png'
713
+ const code = `カメ描画先=「turtle_cv」
714
+ 0にカメ速度設定。
715
+ カメ作成。
716
+ [50,50]にカメ起点移動。90にカメ角度設定。
717
+ `
718
+ nako.run(code)
719
+
720
+ await waitTimer(1.0)
721
+
722
+ const dataPromise = getImageDataFromUrl(imagePath + 'canvas_test_blank.png')
723
+ const cvPromise = getMergedCanvas('turtle_cv', ['0'])
724
+ const [data, cv] = await Promise.all([dataPromise, cvPromise])
725
+
726
+ const cv2 = await recreateCanvas(cv)
727
+ const w = parseInt(cv2.width, 10)
728
+ const h = parseInt(cv2.height, 10)
729
+ const ctx = cv2.getContext('2d')
730
+
731
+ const actData = ctx.getImageData(0, 0, w, h)
732
+
733
+ try {
734
+ const rslt = await uploadCanvasImage(cv, imgname)
735
+ if (rslt !== 'OK') {
736
+ console.log('upload result:' + rslt)
737
+ }
738
+ } catch (ex) {
739
+ console.log('upload error')
740
+ console.log(ex)
741
+ }
742
+
743
+ cmpImageData(actData, data, true)
744
+ }).timeout(5000)
745
+
746
+ it('turtle image extend(elephant)', async () => {
747
+ document.body.innerHTML = __html__[htmlPath+'canvas_basic.html']
748
+ const imgname = 'canvas_test11_wasnotblank.png'
749
+ const code = `カメ描画先=「turtle_cv」
750
+ 0にカメ速度設定。
751
+ ゾウ作成。
752
+ [50,50]にカメ起点移動。270にカメ角度設定。
753
+ `
754
+ nako.run(code)
755
+
756
+ await waitTimer(1.0)
757
+
758
+ const dataPromise = getImageDataFromUrl(imagePath + 'canvas_test_blank.png')
759
+ const cvPromise = getMergedCanvas('turtle_cv', ['0'])
760
+ const [data, cv] = await Promise.all([dataPromise, cvPromise])
761
+
762
+ const cv2 = await recreateCanvas(cv)
763
+ const w = parseInt(cv2.width, 10)
764
+ const h = parseInt(cv2.height, 10)
765
+ const ctx = cv2.getContext('2d')
766
+
767
+ const actData = ctx.getImageData(0, 0, w, h)
768
+
769
+ try {
770
+ const rslt = await uploadCanvasImage(cv, imgname)
771
+ if (rslt !== 'OK') {
772
+ console.log('upload result:' + rslt)
773
+ }
774
+ } catch (ex) {
775
+ console.log('upload error')
776
+ console.log(ex)
777
+ }
778
+
779
+ cmpImageData(actData, data, true)
780
+ }).timeout(5000)
781
+
782
+ it('turtle image extend(panda)', async () => {
783
+ document.body.innerHTML = __html__[htmlPath+'canvas_basic.html']
784
+ const imgname = 'canvas_test12_wasnotblank.png'
785
+ const code = `カメ描画先=「turtle_cv」
786
+ 0にカメ速度設定。
787
+ パンダ作成。
788
+ [50,50]にカメ起点移動。180にカメ角度設定。
789
+ `
790
+ nako.run(code)
791
+
792
+ await waitTimer(1.0)
793
+
794
+ const dataPromise = getImageDataFromUrl(imagePath + 'canvas_test_blank.png')
795
+ const cvPromise = getMergedCanvas('turtle_cv', ['0'])
796
+ const [data, cv] = await Promise.all([dataPromise, cvPromise])
797
+
798
+ const cv2 = await recreateCanvas(cv)
799
+ const w = parseInt(cv2.width, 10)
800
+ const h = parseInt(cv2.height, 10)
801
+ const ctx = cv2.getContext('2d')
802
+
803
+ const actData = ctx.getImageData(0, 0, w, h)
804
+
805
+ try {
806
+ const rslt = await uploadCanvasImage(cv, imgname)
807
+ if (rslt !== 'OK') {
808
+ console.log('upload result:' + rslt)
809
+ }
810
+ } catch (ex) {
811
+ console.log('upload error')
812
+ console.log(ex)
813
+ }
814
+
815
+ cmpImageData(actData, data, true)
816
+ }).timeout(5000)
817
+ })