nadesiko3 3.3.49 → 3.3.52

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 (343) hide show
  1. package/LICENSE +21 -21
  2. package/README.md +143 -143
  3. package/batch/.DS_Store +0 -0
  4. package/batch/browsers.template.md +3 -3
  5. package/batch/build_browsers.nako3 +72 -72
  6. package/batch/build_command.nako3 +44 -44
  7. package/batch/build_nako_version.nako3 +42 -42
  8. package/batch/calc_hash.nako3 +29 -29
  9. package/batch/cmd_txt2json.nako3 +74 -74
  10. package/batch/command.txt +162 -83
  11. package/batch/command_nakopad.txt +69 -0
  12. package/batch/download-extlib.nako3 +43 -43
  13. package/batch/gen_command_nakopad.nako3 +57 -57
  14. package/batch/jsplugin2text.nako3 +285 -285
  15. package/batch/pickup_command.nako3 +113 -110
  16. package/batch/pickup_reserved_words.nako3 +11 -11
  17. package/batch/publish_version.nako3 +46 -46
  18. package/batch/show_agents.js +14 -14
  19. package/batch/turtle2js.nako3 +21 -21
  20. package/bin/cnako3 +10 -10
  21. package/core/LICENSE +21 -21
  22. package/core/README.md +66 -66
  23. package/core/batch/build_nako_version.nako3 +42 -42
  24. package/core/command/snako.mjs +105 -105
  25. package/core/command/snako.mts +116 -116
  26. package/core/index.mjs +21 -21
  27. package/core/package.json +47 -47
  28. package/core/sample/hello.nako3 +7 -7
  29. package/core/sample/hoge.mjs +4 -4
  30. package/core/sample/hoge.mts +6 -6
  31. package/core/src/nako3.mjs +870 -858
  32. package/core/src/nako3.mts +983 -967
  33. package/core/src/nako_colors.mjs +78 -78
  34. package/core/src/nako_colors.mts +86 -86
  35. package/core/src/nako_core_version.mjs +8 -8
  36. package/core/src/nako_core_version.mts +19 -19
  37. package/core/src/nako_csv.mjs +185 -185
  38. package/core/src/nako_csv.mts +188 -188
  39. package/core/src/nako_errors.mjs +173 -173
  40. package/core/src/nako_errors.mts +197 -197
  41. package/core/src/nako_from_dncl.mjs +255 -255
  42. package/core/src/nako_from_dncl.mts +250 -250
  43. package/core/src/nako_gen.mjs +1647 -1648
  44. package/core/src/nako_gen.mts +1718 -1719
  45. package/core/src/nako_gen_async.mjs +1659 -1659
  46. package/core/src/nako_gen_async.mts +1732 -1732
  47. package/core/src/nako_global.mjs +109 -107
  48. package/core/src/nako_global.mts +141 -138
  49. package/core/src/nako_indent.mjs +445 -445
  50. package/core/src/nako_indent.mts +492 -492
  51. package/core/src/nako_josi_list.mjs +38 -38
  52. package/core/src/nako_josi_list.mts +45 -45
  53. package/core/src/nako_lex_rules.mjs +253 -253
  54. package/core/src/nako_lex_rules.mts +260 -260
  55. package/core/src/nako_lexer.mjs +609 -609
  56. package/core/src/nako_lexer.mts +612 -612
  57. package/core/src/nako_logger.mjs +199 -199
  58. package/core/src/nako_logger.mts +232 -232
  59. package/core/src/nako_parser3.mjs +2441 -2439
  60. package/core/src/nako_parser3.mts +2197 -2195
  61. package/core/src/nako_parser_base.mjs +371 -370
  62. package/core/src/nako_parser_base.mts +372 -370
  63. package/core/src/nako_parser_const.mjs +37 -37
  64. package/core/src/nako_parser_const.mts +37 -37
  65. package/core/src/nako_prepare.mjs +304 -304
  66. package/core/src/nako_prepare.mts +315 -315
  67. package/core/src/nako_reserved_words.mjs +38 -38
  68. package/core/src/nako_reserved_words.mts +38 -38
  69. package/core/src/nako_source_mapping.mjs +207 -207
  70. package/core/src/nako_source_mapping.mts +262 -262
  71. package/core/src/nako_test.mjs +37 -37
  72. package/core/src/nako_types.mjs +13 -25
  73. package/core/src/nako_types.mts +144 -151
  74. package/core/src/plugin_csv.mjs +49 -49
  75. package/core/src/plugin_csv.mts +50 -50
  76. package/core/src/plugin_math.mjs +328 -328
  77. package/core/src/plugin_math.mts +326 -326
  78. package/core/src/plugin_promise.mjs +91 -91
  79. package/core/src/plugin_promise.mts +91 -91
  80. package/core/src/plugin_system.mjs +2832 -2832
  81. package/core/src/plugin_system.mts +2690 -2690
  82. package/core/src/plugin_test.mjs +34 -34
  83. package/core/src/plugin_test.mts +34 -34
  84. package/demo/.DS_Store +0 -0
  85. package/demo/ace_editor.html +89 -89
  86. package/demo/ace_editor_tabs.html +161 -161
  87. package/demo/basic.html +71 -71
  88. package/demo/browsers.html +10 -9
  89. package/demo/css/basic.css +3 -3
  90. package/demo/css/common.css +157 -157
  91. package/demo/css/editor.css +8 -8
  92. package/demo/css/flow.css +3 -3
  93. package/demo/css/index.css +3 -3
  94. package/demo/extlib/.DS_Store +0 -0
  95. package/demo/flow.html +98 -98
  96. package/demo/graph.html +53 -53
  97. package/demo/image/nakopad-icon256.png +0 -0
  98. package/demo/index.html +133 -133
  99. package/demo/js/common.js +17 -17
  100. package/demo/js/turtle3d_test.js +44 -44
  101. package/demo/js/turtle_test.js +45 -45
  102. package/demo/nako3/calc.nako3 +4 -4
  103. package/demo/runscript.html +47 -47
  104. package/demo/runscript2.html +33 -33
  105. package/demo/runscript3.html +35 -35
  106. package/demo/runscript4.html +33 -33
  107. package/demo/turtle.html +58 -58
  108. package/demo/turtle2.html +141 -141
  109. package/demo/turtle3.html +279 -279
  110. package/demo/turtle3d.html +58 -58
  111. package/demo/turtle3d2.html +107 -107
  112. package/demo/version.html +24 -24
  113. package/doc/SETUP.md +157 -157
  114. package/doc/about.md +17 -17
  115. package/doc/browsers.md +26 -26
  116. package/doc/docgen.md +21 -21
  117. package/doc/editor.md +44 -44
  118. package/doc/files.md +39 -39
  119. package/doc/plugins.md +234 -234
  120. package/doc/release.md +79 -79
  121. package/doc/textlint.md +43 -43
  122. package/doc/win32.md +57 -57
  123. package/package.json +195 -195
  124. package/release/_hash.txt +28 -28
  125. package/release/_script-tags.txt +14 -14
  126. package/release/command.json +1 -1
  127. package/release/command.json.js +1 -1
  128. package/release/command_cnako3.json +1 -1
  129. package/release/command_list.json +1 -1
  130. package/release/editor.js +1 -1
  131. package/release/nako_gen_async.js +2 -1
  132. package/release/nako_gen_async.js.LICENSE.txt +35 -0
  133. package/release/plugin_caniuse.js.LICENSE.txt +11 -0
  134. package/release/plugin_csv.js +1 -1
  135. package/release/plugin_csv.js.LICENSE.txt +15 -0
  136. package/release/plugin_datetime.js.LICENSE.txt +15 -0
  137. package/release/plugin_kansuji.js.LICENSE.txt +3 -0
  138. package/release/plugin_markup.js.LICENSE.txt +11 -0
  139. package/release/plugin_turtle.js.LICENSE.txt +15 -0
  140. package/release/plugin_webworker.js.LICENSE.txt +3 -0
  141. package/release/plugin_weykturtle3d.js.LICENSE.txt +3 -0
  142. package/release/stats.json +1 -1
  143. package/release/version.js +1 -1
  144. package/release/wnako3.js +1 -1
  145. package/release/wnako3.js.LICENSE.txt +203 -1
  146. package/release/wnako3webworker.js +2 -1
  147. package/release/wnako3webworker.js.LICENSE.txt +131 -0
  148. package/src/.DS_Store +0 -0
  149. package/src/browsers.txt +12 -11
  150. package/src/browsers_agents.json +2 -2
  151. package/src/browsers_agents.mjs +1 -1
  152. package/src/cnako3.mjs +17 -17
  153. package/src/cnako3.mts +18 -18
  154. package/src/cnako3mod.mjs +682 -707
  155. package/src/cnako3mod.mts +688 -712
  156. package/src/commander_ja.mjs +164 -164
  157. package/src/commander_ja.mts +161 -161
  158. package/src/enako3.mjs +68 -68
  159. package/src/era.mjs +22 -22
  160. package/src/image_turtle-elephant.mjs +2 -2
  161. package/src/image_turtle-panda.mjs +2 -2
  162. package/src/image_turtle64.mjs +2 -2
  163. package/src/index.mjs +9 -9
  164. package/src/index.mts +10 -10
  165. package/src/nako3editorfix.sfd +106 -106
  166. package/src/nako_version.mjs +8 -8
  167. package/src/nako_version.mts +2 -2
  168. package/src/plugin_browser.mjs +213 -213
  169. package/src/plugin_browser.mts +206 -206
  170. package/src/plugin_browser_ajax.mjs +399 -399
  171. package/src/plugin_browser_audio.mjs +109 -109
  172. package/src/plugin_browser_canvas.mjs +449 -449
  173. package/src/plugin_browser_chart.mjs +294 -294
  174. package/src/plugin_browser_color.mjs +49 -49
  175. package/src/plugin_browser_crypto.mjs +26 -26
  176. package/src/plugin_browser_dialog.mjs +53 -53
  177. package/src/plugin_browser_dom_basic.mjs +336 -336
  178. package/src/plugin_browser_dom_event.mjs +193 -193
  179. package/src/plugin_browser_dom_parts.mjs +396 -396
  180. package/src/plugin_browser_geolocation.mjs +51 -51
  181. package/src/plugin_browser_hotkey.mjs +25 -25
  182. package/src/plugin_browser_html.mjs +59 -59
  183. package/src/plugin_browser_in_worker.mjs +45 -45
  184. package/src/plugin_browser_location.mjs +21 -21
  185. package/src/plugin_browser_speech.mjs +111 -111
  186. package/src/plugin_browser_storage.mjs +121 -121
  187. package/src/plugin_browser_system.mjs +31 -31
  188. package/src/plugin_browser_websocket.mjs +73 -73
  189. package/src/plugin_caniuse.mjs +29 -29
  190. package/src/plugin_datetime.mjs +394 -394
  191. package/src/plugin_httpserver.mjs +277 -277
  192. package/src/plugin_httpserver.mts +286 -286
  193. package/src/plugin_kansuji.mjs +224 -224
  194. package/src/plugin_keigo.mjs +55 -55
  195. package/src/plugin_markup.mjs +32 -32
  196. package/src/plugin_node.mjs +1047 -1047
  197. package/src/plugin_node.mts +980 -980
  198. package/src/plugin_turtle.mjs +647 -647
  199. package/src/plugin_webworker.mjs +334 -334
  200. package/src/plugin_weykturtle3d.mjs +1214 -1214
  201. package/src/plugin_worker.mjs +95 -95
  202. package/src/repl.nako3 +63 -63
  203. package/src/wnako3.mjs +12 -12
  204. package/src/wnako3.mts +11 -11
  205. package/src/wnako3_editor.css +215 -215
  206. package/src/wnako3_editor.mjs +1542 -1542
  207. package/src/wnako3_editor.mts +1658 -1657
  208. package/src/wnako3mod.mjs +213 -213
  209. package/src/wnako3mod.mts +214 -214
  210. package/src/wnako3webworker.mjs +69 -69
  211. package/test/.DS_Store +0 -0
  212. package/test/ace_editor/karma.config.js +94 -94
  213. package/test/ace_editor/test/.babelrc.json +3 -3
  214. package/test/ace_editor/test/ace_editor_test.js +178 -178
  215. package/test/ace_editor/test/html/custom_context.html +139 -139
  216. package/test/async/async_basic_test.mjs +122 -122
  217. package/test/browser/.DS_Store +0 -0
  218. package/test/browser/karma.config.js +221 -221
  219. package/test/browser/test/.babelrc.json +3 -3
  220. package/test/browser/test/compare_util.js +50 -50
  221. package/test/browser/test/html/div_basic.html +2 -2
  222. package/test/browser/test/html/event_dom_form.html +4 -4
  223. package/test/browser/test/html/event_dom_scrolldiv.html +5 -5
  224. package/test/browser/test/import_plugin_checker.js +24 -24
  225. package/test/browser/test/plugin_browser_test.js +51 -51
  226. package/test/browser/test/plugin_browser_test_ajax.js +123 -123
  227. package/test/browser/test/plugin_browser_test_color.js +18 -18
  228. package/test/browser/test/plugin_browser_test_dialog.js +72 -72
  229. package/test/browser/test/plugin_browser_test_dom_event.js +598 -598
  230. package/test/browser/test/plugin_browser_test_dom_parts.js +125 -125
  231. package/test/browser/test/plugin_browser_test_system.js +9 -9
  232. package/test/browser/test/plugin_turtle_test.js +817 -817
  233. package/test/browser/test/plugin_webworker_test.js +86 -86
  234. package/test/browser/test/require_test.js +68 -68
  235. package/test/bundled/.DS_Store +0 -0
  236. package/test/bundled/karma.config.base.js +117 -117
  237. package/test/bundled/karma.config.js +86 -86
  238. package/test/bundled/test/.babelrc.json +3 -3
  239. package/test/bundled/test/bundled_test.js +69 -69
  240. package/test/bundled/test/html/custom_context.html +65 -65
  241. package/test/bundled/test/html/custom_debug.html +66 -66
  242. package/test/bundled/test4b.cmd +52 -52
  243. package/test/bundled/test_base/.DS_Store +0 -0
  244. package/test/bundled/test_base/.babelrc.json +3 -3
  245. package/test/bundled/test_base/_checktool_test.js +25 -25
  246. package/test/bundled/test_base/basic_ajax_test.js +56 -56
  247. package/test/bundled/test_base/basic_async_test.js +18 -18
  248. package/test/bundled/test_base/basic_test.js +153 -153
  249. package/test/bundled/test_base/calc_test.js +132 -132
  250. package/test/bundled/test_base/css/browsers_box.css +114 -114
  251. package/test/bundled/test_base/html/custom_context.html +69 -69
  252. package/test/bundled/test_base/html/custom_debug.html +71 -71
  253. package/test/bundled/test_base/js/browsers_box.js +72 -72
  254. package/test/bundled/test_base/plugin_csv_test.js +37 -37
  255. package/test/bundled/test_base/plugin_datetime_test.js +115 -115
  256. package/test/bundled/test_base/plugin_kansuji_test.js +49 -49
  257. package/test/bundled/test_base/plugin_system_test.js +410 -410
  258. package/test/bundled/test_base/plugin_webworker_test.js +53 -53
  259. package/test/bundled/test_base/test_utils.js +191 -191
  260. package/test/common/.DS_Store +0 -0
  261. package/test/common/plugin_browser_test.mjs +22 -22
  262. package/test/common/plugin_browser_ut_audio_test.mjs +108 -108
  263. package/test/common/plugin_browser_ut_color_test.mjs +21 -21
  264. package/test/common/plugin_browser_ut_dialog_test.mjs +100 -100
  265. package/test/common/plugin_browser_ut_html_test.mjs +13 -13
  266. package/test/common/plugin_browser_ut_system_test.mjs +10 -10
  267. package/test/common/plugin_markup_test.mjs +23 -23
  268. package/test/jsconfig.json +19 -19
  269. package/test/karma.config.js +91 -91
  270. package/test/node/.DS_Store +0 -0
  271. package/test/node/async_test.mjs +96 -96
  272. package/test/node/commander_ja_test.mjs +89 -89
  273. package/test/node/error_message_test.mjs +243 -243
  274. package/test/node/kai_test.nako3 +6 -6
  275. package/test/node/node_test.mjs +60 -60
  276. package/test/node/plugin_broken.js.txt +3 -3
  277. package/test/node/plugin_browser_ut_ajax_test.mjs.todo +357 -357
  278. package/test/node/plugin_browser_ut_location_test.mjs +42 -42
  279. package/test/node/plugin_markup_test.mjs +47 -47
  280. package/test/node/plugin_math_test.mjs +45 -45
  281. package/test/node/plugin_node_test.mjs +98 -98
  282. package/test/node/plugin_test.mjs +44 -44
  283. package/test/node/relative_import_test_1.nako3 +1 -1
  284. package/test/node/relative_import_test_2.nako3 +2 -2
  285. package/test/node/require_nako3_test.mjs +67 -67
  286. package/test/node/requiretest.nako3 +4 -4
  287. package/test/node/requiretest_indirect.nako3 +1 -1
  288. package/test/node/requiretest_name.nako3 +5 -5
  289. package/test/node/runtime_error.nako3 +2 -2
  290. package/test/node/scope1.nako3 +10 -10
  291. package/test/node/scope2.nako3 +12 -12
  292. package/test/node/side_effects_test.mjs +39 -39
  293. package/test/node/sjis.txt +5 -5
  294. package/test/node/syntax_error.nako3 +1 -1
  295. package/test/node/wnako3_editor_test.mjs +384 -384
  296. package/tools/.DS_Store +0 -0
  297. package/tools/README.md +12 -12
  298. package/tools/check_new_version.nako3 +25 -25
  299. package/tools/nako3edit/.DS_Store +0 -0
  300. package/tools/nako3edit/a.sqlite3 +0 -0
  301. package/tools/nako3edit/html/.DS_Store +0 -0
  302. package/tools/nako3edit/html/daisyui/LICENSE +22 -22
  303. package/tools/nako3edit/html/daisyui/version_2.14.1 +1 -1
  304. package/tools/nako3edit/html/edit.html +170 -170
  305. package/tools/nako3edit/html/edit_plugin.js +6 -6
  306. package/tools/nako3edit/html/files.html +125 -125
  307. package/tools/nako3edit/html/nako3edit.css +65 -65
  308. package/tools/nako3edit/index.mjs +248 -248
  309. package/tools/nako3server/index.html +10 -10
  310. package/tools/nako3server/index.mjs +116 -116
  311. package/tools/nako3server/index.nako3 +34 -34
  312. package/core/.editorconfig +0 -6
  313. package/core/.eslintrc.cjs +0 -33
  314. package/core/.github/dependabot.yml +0 -7
  315. package/core/.github/workflows/nodejs.yml +0 -37
  316. package/core/.github/workflows/super-linter.yml +0 -61
  317. package/core/.github/workflows/textlint.yml +0 -199
  318. package/core/index.mts +0 -21
  319. package/core/test/array_test.mjs +0 -34
  320. package/core/test/basic_test.mjs +0 -344
  321. package/core/test/calc_test.mjs +0 -140
  322. package/core/test/core_module_test.mjs +0 -23
  323. package/core/test/debug_test.mjs +0 -16
  324. package/core/test/dncl_test.mjs +0 -94
  325. package/core/test/error_message_test.mjs +0 -210
  326. package/core/test/error_test.mjs +0 -16
  327. package/core/test/flow_test.mjs +0 -373
  328. package/core/test/func_call.mjs +0 -160
  329. package/core/test/func_test.mjs +0 -149
  330. package/core/test/indent_test.mjs +0 -364
  331. package/core/test/lex_test.mjs +0 -168
  332. package/core/test/literal_test.mjs +0 -73
  333. package/core/test/nako_lexer_test.mjs +0 -35
  334. package/core/test/nako_logger_test.mjs +0 -76
  335. package/core/test/nako_logger_test.mts +0 -78
  336. package/core/test/plugin_csv_test.mjs +0 -38
  337. package/core/test/plugin_promise_test.mjs +0 -18
  338. package/core/test/plugin_system_test.mjs +0 -630
  339. package/core/test/prepare_test.mjs +0 -96
  340. package/core/test/re_test.mjs +0 -22
  341. package/core/test/side_effects_test.mjs +0 -92
  342. package/core/test/variable_scope_test.mjs +0 -149
  343. package/core/tsconfig.json +0 -101
@@ -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
+ })