nadesiko3 3.2.30 → 3.2.34

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 (39) hide show
  1. package/demo/index.html +0 -1
  2. package/package.json +13 -5
  3. package/release/_hash.txt +28 -28
  4. package/release/_script-tags.txt +13 -13
  5. package/release/command.json +1 -1
  6. package/release/command.json.js +1 -1
  7. package/release/command_cnako3.json +1 -1
  8. package/release/command_list.json +1 -1
  9. package/release/editor.js +1 -1
  10. package/release/josi.json +1 -0
  11. package/release/nako_gen_async.js +1 -1
  12. package/release/plugin_markup.js +1 -1
  13. package/release/stats.json +1 -1
  14. package/release/version.js +1 -1
  15. package/release/wnako3.js +1 -1
  16. package/release/wnako3webworker.js +1 -1
  17. package/release/wnako3webworker.js.LICENSE.txt +4 -0
  18. package/release/yoyakugo.json +3 -0
  19. package/src/nako3.js +6 -1
  20. package/src/nako_from_dncl.js +239 -0
  21. package/src/nako_gen.js +24 -9
  22. package/src/nako_gen_async.js +97 -10
  23. package/src/nako_josi_list.js +1 -1
  24. package/src/nako_parser3.js +147 -25
  25. package/src/nako_parser_base.js +3 -0
  26. package/src/nako_reserved_words.js +4 -1
  27. package/src/nako_version.js +2 -2
  28. package/src/plugin_browser.js +14 -0
  29. package/src/plugin_browser_ajax.js +4 -3
  30. package/src/plugin_browser_canvas.js +0 -1
  31. package/src/plugin_browser_chart.js +0 -2
  32. package/src/plugin_browser_dom_basic.js +0 -2
  33. package/src/plugin_browser_dom_parts.js +273 -79
  34. package/src/plugin_node.js +2 -2
  35. package/src/plugin_system.js +408 -20
  36. package/test/common/dncl_test.js +57 -0
  37. package/test/common/flow_test.js +14 -0
  38. package/test/common/plugin_system_test.js +117 -0
  39. package/test/common/variable_scope_test.js +3 -0
@@ -2,8 +2,9 @@
2
2
  module.exports = {
3
3
  // @DOM部品操作
4
4
  'DOM親要素': { type: 'const', value: '' }, // @DOMおやようそ
5
- 'DOM生成個数': { type: 'const', value: 0 }, // @DOMせいせいこすう
6
- 'DOM親要素設定': { // @「ボタン作成」「エディタ作成」などのDOM要素を追加する対象を指定(デフォルトはdocument)して親要素のDOMオブジェクトを返す // @DOMおやようそせってい
5
+ 'DOM部品個数': { type: 'const', value: 0 }, // @DOMせいせいこすう
6
+ 'DOM部品オプション': { type: 'const', value: {'自動改行': false, 'テーブル背景色': ['#AA4040', '#ffffff','#fff0f0']} }, // @DOMぶひんおぷしょん
7
+ 'DOM親要素設定': { // @「ボタン作成」「エディタ作成」など『DOM部品作成』で追加する要素の親要素を指定(デフォルトはdocument)して要素を返す。 // @DOMおやようそせってい
7
8
  type: 'func',
8
9
  josi: [['に', 'へ']],
9
10
  pure: false,
@@ -13,23 +14,54 @@ module.exports = {
13
14
  return el
14
15
  }
15
16
  },
17
+ 'DOM親部品設定': { // @ DOM部品作成でDOM要素を追加する親の対象を指定。『DOM親要素設定』と同じ。// @DOMおやぶひんせってい
18
+ type: 'func',
19
+ josi: [['に', 'へ']],
20
+ pure: false,
21
+ fn: function (el, sys) {
22
+ return sys.__exec('DOM親要素設定', [el, sys])
23
+ }
24
+ },
16
25
  'DOMスキン': { type: 'const', value: '' }, // @DOMすきん
17
26
  'DOMスキン辞書': { type: 'const', value: {} }, // @DOMすきんじしょ
18
- 'ボタン作成': { // @ラベルlabelを持つbutton要素を追加しDOMオブジェクトを返す // @ぼたんさくせい
27
+ 'DOMスキン設定': { // @「ボタン作成」「エディタ作成」などで適用するスキンを指定する(#1033) // @DOMすきんせってい
28
+ type: 'func',
29
+ josi: [['を', 'に', 'の']],
30
+ pure: false,
31
+ fn: function (skin, sys) {
32
+ sys.__v0['DOMスキン'] = skin
33
+ },
34
+ return_none: true
35
+ },
36
+ 'DOM部品作成': { // @elmの要素を作成して『DOM親要素設定』で指定した要素に追加して、DOMオブジェクトを返す。(elmがDOM要素なら追加する) // @DOMぶひんさくせい
19
37
  type: 'func',
20
38
  josi: [['の']],
21
39
  pure: false,
22
- fn: function (label, sys) {
40
+ fn: function (elm, sys) {
23
41
  const parent = sys.__v0['DOM親要素']
24
- const btn = document.createElement('button')
25
- btn.innerHTML = label
26
- btn.id = 'nadesi-dom-' + sys.__v0['DOM生成個数']
42
+ const btn = (typeof(elm) === 'string') ? document.createElement(elm) : elm
43
+ btn.id = 'nadesi-dom-' + sys.__v0['DOM部品個数']
27
44
  // スキン適用
28
45
  const func = sys.__v0['DOMスキン辞書'][sys.__v0['DOMスキン']]
29
- if (typeof (func) === 'function') { func('ボタン作成', btn, sys) }
46
+ if (typeof (func) === 'function') { func(elm, btn, sys) }
30
47
  // DOM追加
31
48
  parent.appendChild(btn)
32
- sys.__v0['DOM生成個数']++
49
+ sys.__v0['DOM部品個数']++
50
+ // オプションを適用
51
+ const opt = sys.__v0['DOM部品オプション']
52
+ if (opt['自動改行']) {
53
+ parent.appendChild(document.createElement('br'));
54
+ }
55
+ return btn
56
+ }
57
+ },
58
+ 'ボタン作成': { // @ラベルlabelを持つbutton要素を追加しDOMオブジェクトを返す // @ぼたんさくせい
59
+ type: 'func',
60
+ josi: [['の']],
61
+ pure: false,
62
+ fn: function (label, sys) {
63
+ const btn = sys.__exec('DOM部品作成', ['button', sys])
64
+ btn.innerHTML = label
33
65
  return btn
34
66
  }
35
67
  },
@@ -38,17 +70,9 @@ module.exports = {
38
70
  josi: [['の']],
39
71
  pure: false,
40
72
  fn: function (text, sys) {
41
- const parent = sys.__v0['DOM親要素']
42
- const inp = document.createElement('input')
73
+ const inp = sys.__exec('DOM部品作成', ['input', sys])
43
74
  inp.type = 'text'
44
75
  inp.value = text
45
- inp.id = 'nadesi-dom-' + sys.__v0['DOM生成個数']
46
- // スキン適用
47
- const func = sys.__v0['DOMスキン辞書'][sys.__v0['DOMスキン']]
48
- if (typeof (func) === 'function') { func('エディタ作成', inp, sys) }
49
- // DOM追加
50
- parent.appendChild(inp)
51
- sys.__v0['DOM生成個数']++
52
76
  return inp
53
77
  }
54
78
  },
@@ -57,16 +81,8 @@ module.exports = {
57
81
  josi: [['の']],
58
82
  pure: false,
59
83
  fn: function (text, sys) {
60
- const parent = sys.__v0['DOM親要素']
61
- const te = document.createElement('textarea')
84
+ const te = sys.__exec('DOM部品作成', ['textarea', sys])
62
85
  te.value = text
63
- te.id = 'nadesi-dom-' + sys.__v0['DOM生成個数']
64
- // スキン適用
65
- const func = sys.__v0['DOMスキン辞書'][sys.__v0['DOMスキン']]
66
- if (typeof (func) === 'function') { func('テキストエリア作成', te, sys) }
67
- // DOM追加
68
- parent.appendChild(te)
69
- sys.__v0['DOM生成個数']++
70
86
  return te
71
87
  }
72
88
  },
@@ -75,17 +91,9 @@ module.exports = {
75
91
  josi: [['の']],
76
92
  pure: false,
77
93
  fn: function (text, sys) {
78
- const parent = sys.__v0['DOM親要素']
79
- const te = document.createElement('span')
80
- te.innerHTML = text
81
- te.id = 'nadesi-dom-' + sys.__v0['DOM生成個数']
82
- // スキン適用
83
- const func = sys.__v0['DOMスキン辞書'][sys.__v0['DOMスキン']]
84
- if (typeof (func) === 'function') { func('ラベル作成', te, sys) }
85
- // DOM追加
86
- parent.appendChild(te)
87
- sys.__v0['DOM生成個数']++
88
- return te
94
+ const lbl = sys.__exec('DOM部品作成', ['span', sys])
95
+ lbl.innerHTML = text
96
+ return lbl
89
97
  }
90
98
  },
91
99
  'キャンバス作成': { // @大きさ[幅, 高]のcanvas要素を追加しDOMオブジェクトを返す // @きゃんばすさくせい
@@ -93,40 +101,33 @@ module.exports = {
93
101
  josi: [['の']],
94
102
  pure: false,
95
103
  fn: function (size, sys) {
96
- const parent = sys.__v0['DOM親要素']
97
- const cv = document.createElement('canvas')
104
+ const cv = sys.__exec('DOM部品作成', ['canvas', sys])
98
105
  cv.width = size[0]
99
106
  cv.height = size[1]
100
107
  cv.style.width = size[0]
101
108
  cv.style.height = size[1]
102
- cv.id = 'nadesi-dom-' + sys.__v0['DOM生成個数']
103
- // スキン適用
104
- const func = sys.__v0['DOMスキン辞書'][sys.__v0['DOMスキン']]
105
- if (typeof (func) === 'function') { func('キャンバス作成', cv, sys) }
106
- // DOM追加
107
- parent.appendChild(cv)
108
- sys.__v0['DOM生成個数']++
109
109
  // 描画中キャンバスを移動する
110
110
  sys.__exec('描画開始', [cv, sys])
111
- console.log(cv)
112
111
  return cv
113
112
  }
114
113
  },
114
+ '画像作成': { // @URLを指定してimg要素を追加しDOMオブジェクトを返す // @がぞうさくせい
115
+ type: 'func',
116
+ josi: [['の', 'から']],
117
+ pure: false,
118
+ fn: function (url, sys) {
119
+ const img = sys.__exec('DOM部品作成', ['img', sys])
120
+ img.src = url
121
+ return img
122
+ }
123
+ },
115
124
  '改行作成': { // @改行(br要素)を追加しDOMオブジェクトを返す // @かいぎょうさくせい
116
125
  type: 'func',
117
126
  josi: [],
118
127
  pure: false,
119
128
  fn: function (sys) {
120
- const parent = sys.__v0['DOM親要素']
121
- const te = document.createElement('br')
122
- te.id = 'nadesi-dom-' + sys.__v0['DOM生成個数']
123
- // スキン適用
124
- const func = sys.__v0['DOMスキン辞書'][sys.__v0['DOMスキン']]
125
- if (typeof (func) === 'function') { func('改行作成', te, sys) }
126
- // DOM追加
127
- parent.appendChild(te)
128
- sys.__v0['DOM生成個数']++
129
- return te
129
+ const br = sys.__exec('DOM部品作成', ['br', sys])
130
+ return br
130
131
  }
131
132
  },
132
133
  'チェックボックス作成': { // @textのラベルを持つチェックボックス(input[type='checkbox'])要素を追加しDOMオブジェクトを返す // @ちぇっくぼっくすさくせい
@@ -134,22 +135,19 @@ module.exports = {
134
135
  josi: [['の']],
135
136
  pure: false,
136
137
  fn: function (text, sys) {
137
- const parent = sys.__v0['DOM親要素']
138
+ // チェックボックスは、<span><input><label></span>で成り立つように構築
138
139
  const span = document.createElement('span')
139
140
  const inp = document.createElement('input')
140
141
  inp.type = 'checkbox'
141
- inp.id = 'nadesi-dom-' + sys.__v0['DOM生成個数']
142
+ inp.id = 'nadesi-dom-' + sys.__v0['DOM部品個数']
143
+ sys.__v0['DOM部品個数']++
142
144
  const label = document.createElement('label')
143
145
  label.innerHTML = text
144
146
  label.htmlFor = inp.id
145
147
  span.appendChild(inp)
146
148
  span.appendChild(label)
147
- // スキン適用
148
- const func = sys.__v0['DOMスキン辞書'][sys.__v0['DOMスキン']]
149
- if (typeof (func) === 'function') { func('チェックボックス作成', span, sys) }
150
- // DOM追加
151
- parent.appendChild(span)
152
- sys.__v0['DOM生成個数']++
149
+ // 親部品に追加
150
+ sys.__exec('DOM部品作成', [span, sys])
153
151
  return inp
154
152
  }
155
153
  },
@@ -158,30 +156,226 @@ module.exports = {
158
156
  josi: [['の']],
159
157
  pure: false,
160
158
  fn: function (options, sys) {
161
- const parent = sys.__v0['DOM親要素']
162
159
  const dom = document.createElement('select')
163
- dom.id = 'nadesi-dom-' + (sys.__v0['DOM生成個数']++)
164
160
  for (let i = 0; i < options.length; i++) {
165
161
  const item = document.createElement('option')
166
162
  item.value = options[i]
167
163
  item.appendChild(document.createTextNode(options[i]))
168
164
  dom.appendChild(item)
169
165
  }
170
- // スキン適用
171
- const func = sys.__v0['DOMスキン辞書'][sys.__v0['DOMスキン']]
172
- if (typeof (func) === 'function') { func('セレクトボックス作成', dom, sys) }
173
- // DOM追加
174
- parent.appendChild(dom)
166
+ // 親部品に追加
167
+ const obj = sys.__exec('DOM部品作成', [dom, sys])
175
168
  return dom
176
169
  }
177
170
  },
178
- 'DOMスキン設定': { // @「ボタン作成」「エディタ作成」などで適用するスキンを指定する(#1033) // @DOMすきんせってい
171
+ '色選択ボックス作成': { // @色選択ボックス(input[type='color'])を作成しDOMオブジェクトを返す // @いろせんたくぼっくすさくせい
179
172
  type: 'func',
180
- josi: [['を', 'に', 'の']],
173
+ josi: [],
181
174
  pure: false,
182
- fn: function (skin, sys) {
183
- sys.__v0['DOMスキン'] = skin
184
- },
185
- return_none: true
175
+ fn: function (sys) {
176
+ const inp = sys.__exec('DOM部品作成', ['input', sys])
177
+ inp.type = 'color'
178
+ return inp
179
+ }
180
+ },
181
+ '日付選択ボックス作成': { // @日付選択ボックス(input[type='date'])を作成しDOMオブジェクトを返す // @ひづけせんたくぼっくすさくせい
182
+ type: 'func',
183
+ josi: [],
184
+ pure: false,
185
+ fn: function (sys) {
186
+ const inp = sys.__exec('DOM部品作成', ['input', sys])
187
+ inp.type = 'date'
188
+ return inp
189
+ }
190
+ },
191
+ 'パスワード入力エディタ作成': { // @パスワード入力エディタ(input[type='password'])を作成し初期値Sを設定し、DOMオブジェクトを返す // @ぱすわーどにゅうりょくさくせい
192
+ type: 'func',
193
+ josi: [['の','で']],
194
+ pure: false,
195
+ fn: function (s, sys) {
196
+ const inp = sys.__exec('DOM部品作成', ['input', sys])
197
+ inp.type = 'password'
198
+ inp.value = s
199
+ return inp
200
+ }
201
+ },
202
+ '値指定バー作成': { // @範囲RANGE(配列で[最小,最大[,値]])を指定するバー(input[type='range'])を作成しDOMオブジェクトを返す // @ぱすわーどにゅうりょくさくせい
203
+ type: 'func',
204
+ josi: [['の','で']],
205
+ pure: false,
206
+ fn: function (range, sys) {
207
+ if (!(range instanceof Array) || range.length < 2) {
208
+ range = [0, 100, 50]
209
+ }
210
+ if (range.length <= 2) { // 3つ目を省略したとき
211
+ range.push(Math.floor((range[1] - range[0]) / 2))
212
+ }
213
+ const inp = sys.__exec('DOM部品作成', ['input', sys])
214
+ inp.type = 'range'
215
+ inp.min = range[0]
216
+ inp.max = range[1]
217
+ inp.value = range[2]
218
+ return inp
219
+ }
220
+ },
221
+ '送信ボタン作成': { // @ラベルSの送信ボタン(input[type='submit'])を作成しDOMオブジェクトを返す // @そうしんぼたんさくせい
222
+ type: 'func',
223
+ josi: [['の']],
224
+ pure: false,
225
+ fn: function (label, sys) {
226
+ const inp = sys.__exec('DOM部品作成', ['input', sys])
227
+ inp.type = 'submit'
228
+ inp.value = label
229
+ return inp
230
+ }
231
+ },
232
+ 'フォーム作成': { // @属性OBJ{method:"GET",action:"..."}で項目一覧S「a=初期値{改行}b=初期値{改行}色=?c#fff0f0{改行}=?送信」を送信フォームを作成しDOMオブジェクトを返す // @ふぉーむさくせい
233
+ type: 'func',
234
+ josi: [['で','の'],['を']],
235
+ pure: false,
236
+ fn: function (obj, s, sys) {
237
+ const frm = sys.__exec('DOM部品作成', ['form', sys])
238
+ // 可能ならformにobjの値を移し替える
239
+ if (obj instanceof Object) {
240
+ for (let key in obj) {
241
+ if (frm[key]) { frm[key] = obj[key] }
242
+ }
243
+ }
244
+ // 入力項目をtableで作る
245
+ const rows = s.split('\n')
246
+ const table = document.createElement('table')
247
+ for (let rowIndex in rows) {
248
+ let row = '' + (rows[rowIndex])
249
+ if (row === '') {continue}
250
+ if (row.indexOf('=') < 0) { row += '=' }
251
+ const cols = row.split('=')
252
+ const key = cols[0]
253
+ const val = cols[1]
254
+ // key
255
+ const th = document.createElement('th')
256
+ th.innerHTML = sys.__tohtmlQ(key)
257
+ // val
258
+ const td = document.createElement('td')
259
+ if (val.substring(0, 2) === '?(') {
260
+ // select box
261
+ const it = val.substring(2) + ')'
262
+ const ita = it.split(')')
263
+ const its = ita[0]
264
+ const def = ita[1]
265
+ const items = its.split('|')
266
+ const select = document.createElement('select')
267
+ select.name = key
268
+ for (let it of items) {
269
+ const option = document.createElement('option')
270
+ option.value = it
271
+ option.text = it
272
+ select.appendChild(option)
273
+ }
274
+ const idx = items.indexOf(def)
275
+ if (idx >= 0) { select.selectedIndex = idx }
276
+ td.appendChild(select)
277
+ } else {
278
+ // input element
279
+ const inp = document.createElement('input')
280
+ td.appendChild(inp)
281
+ inp.id = 'nako3form_' + key
282
+ if (val === '?送信' || val === '?submit') {
283
+ inp.type = 'submit'
284
+ inp.value = val.substring(1)
285
+ if (key != '') { inp.name = key }
286
+ } else if (val.substring(0, 2) === '?c') {
287
+ inp.type = 'color'
288
+ inp.value = val.substring(2)
289
+ inp.name = key
290
+ } else {
291
+ inp.type = 'text'
292
+ inp.value = val
293
+ inp.name = key
294
+ }
295
+ }
296
+ const tr = document.createElement('tr')
297
+ tr.appendChild(th)
298
+ tr.appendChild(td)
299
+ table.appendChild(tr)
300
+ }
301
+ frm.appendChild(table)
302
+ return frm
303
+ }
304
+ },
305
+ 'フォーム入力一括取得': { // @DOMのフォームを取得し、そのフォーム以下にある入力項目のnameとvalueを辞書形式で返す // @ふぉーむにゅうりょくいっかつしゅとく
306
+ type: 'func',
307
+ josi: [['の','から']],
308
+ pure: true,
309
+ fn: function (dom) {
310
+ if (typeof(dom) === 'string') { dom = document.querySelector(dom) }
311
+ const res = {}
312
+ const getChildren = (pa) => {
313
+ if (!pa || !pa.childNodes) {return}
314
+ for (let i = 0; i < pa.childNodes.length; i++) {
315
+ const el = pa.childNodes[i]
316
+ if (!el.tagName) {return}
317
+ const tag = el.tagName.toLowerCase()
318
+ if (tag === 'input') {
319
+ if (el.type === 'checkbox') {
320
+ res[el.name] = el.checked ? el.value : ''
321
+ continue
322
+ }
323
+ res[el.name] = el.value
324
+ continue
325
+ }
326
+ else if (tag === 'textarea') {
327
+ res[el.name] = el.value
328
+ }
329
+ else if (tag === 'select') {
330
+ if (el.selectedIndex >= 0) {
331
+ res[el.name] = el.options[el.selectedIndex].value
332
+ } else {
333
+ res[el.name] = ''
334
+ }
335
+ }
336
+ getChildren(el)
337
+ }
338
+ }
339
+ getChildren(dom)
340
+ return res
341
+ }
342
+ },
343
+ 'テーブル作成': { // @二次元配列AA(あるいは文字列の簡易CSVデータ)からTABLE要素を作成し、DOMオブジェクトを返す // @てーぶるさくせい
344
+ type: 'func',
345
+ josi: [['の', 'から']],
346
+ pure: false,
347
+ fn: function (aa, sys) {
348
+ if (typeof(aa) === 'string') {
349
+ const rr = []
350
+ const rows = aa.split('\n')
351
+ for (let row of rows) { rr.push(row.split(',')) }
352
+ aa = rr
353
+ }
354
+ const bgColor = JSON.parse(JSON.stringify(sys.__v0['DOM部品オプション']['テーブル背景色']))
355
+ for (let i = 0; i < 3; i++) {bgColor.push('')}
356
+ const bgHead = bgColor.shift()
357
+ const table = sys.__exec('DOM部品作成', ['table', sys])
358
+ for (let rowNo in aa) {
359
+ const row = aa[rowNo]
360
+ const tr = document.createElement('tr')
361
+ for (let col of row) {
362
+ col = '' + col
363
+ const td = document.createElement((rowNo == 0) ? 'th' : 'td')
364
+ td.innerHTML = sys.__tohtml(col)
365
+ // 色指定
366
+ if (bgHead != '') {
367
+ td.style.backgroundColor = (rowNo == 0) ? bgHead : bgColor[rowNo % 2]
368
+ td.style.color = (rowNo == 0) ? 'white' : 'black'
369
+ }
370
+ if (col.match(/^\d+$/)) { // number?
371
+ td.style.textAlign = 'right'
372
+ }
373
+ tr.appendChild(td)
374
+ }
375
+ table.appendChild(tr)
376
+ }
377
+ sys.__exec('DOM親要素設定', [table, sys])
378
+ return table
379
+ }
186
380
  }
187
381
  }
@@ -882,7 +882,7 @@ const PluginNode = {
882
882
  if (sys.__genMode !== '非同期モード') {
883
883
  throw new Error('『AJAX受信』を使うには、プログラムの冒頭で「!非同期モード」と宣言してください。')
884
884
  }
885
- sys.async = true
885
+ const sysenv = sys.setAsync(sys)
886
886
  let options = sys.__v0['AJAXオプション']
887
887
  if (options === '') { options = { method: 'GET' } }
888
888
  // fetch 実行
@@ -894,7 +894,7 @@ const PluginNode = {
894
894
  }
895
895
  }).then(text => {
896
896
  sys.__v0['対象'] = text
897
- sys.nextAsync(sys)
897
+ sys.compAsync(sys, sysenv)
898
898
  }).catch(err => {
899
899
  console.error('[AJAX受信のエラー]', err)
900
900
  sys.__errorAsync(err, sys)