soiz1 commited on
Commit
782523f
·
verified ·
1 Parent(s): b34210f

Create scratch3_handpose2scratch/index.js

Browse files
local-scratch-vm/src/extensions/scratch3_handpose2scratch/index.js ADDED
@@ -0,0 +1,440 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ const ArgumentType = require('../../extension-support/argument-type');
2
+ const BlockType = require('../../extension-support/block-type');
3
+ const Cast = require('../../util/cast');
4
+ const formatMessage = require('format-message');
5
+ const ml5 = require('ml5');
6
+
7
+
8
+ const blockIconURI = '';
9
+
10
+ const Message = {
11
+ getX: {
12
+ 'ja': '[LANDMARK] のx座標',
13
+ 'ja-Hira': '[LANDMARK] のxざひょう',
14
+ 'en': 'x of [LANDMARK]'
15
+ },
16
+ getY: {
17
+ 'ja': '[LANDMARK] のy座標',
18
+ 'ja-Hira': '[LANDMARK] のyざひょう',
19
+ 'en': 'y of [LANDMARK]'
20
+ },
21
+ getZ: {
22
+ 'ja': '[LANDMARK] のz座標',
23
+ 'ja-Hira': '[LANDMARK] のzざひょう',
24
+ 'en': 'z of [LANDMARK]'
25
+ },
26
+ videoToggle: {
27
+ 'ja': 'ビデオを [VIDEO_STATE] にする',
28
+ 'ja-Hira': 'ビデオを [VIDEO_STATE] にする',
29
+ 'en': 'turn video [VIDEO_STATE]'
30
+ },
31
+ setRatio: {
32
+ 'ja': '倍率を [RATIO] にする',
33
+ 'ja-Hira': 'ばいりつを [RATIO] にする',
34
+ 'en': 'set ratio to [RATIO]'
35
+ },
36
+ setInterval: {
37
+ 'ja': '認識を [INTERVAL] 秒ごとに行う',
38
+ 'ja-Hira': 'にんしきを [INTERVAL] びょうごとにおこなう',
39
+ 'en': 'Label once every [INTERVAL] seconds'
40
+ },
41
+ on: {
42
+ 'ja': '入',
43
+ 'ja-Hira': 'いり',
44
+ 'en': 'on'
45
+ },
46
+ off: {
47
+ 'ja': '切',
48
+ 'ja-Hira': 'きり',
49
+ 'en': 'off'
50
+ },
51
+ video_on_flipped: {
52
+ 'ja': '左右反転',
53
+ 'ja-Hira': 'さゆうはんてん',
54
+ 'en': 'on flipped'
55
+ },
56
+ please_wait: {
57
+ 'ja': '準備に時間がかかります。少しの間、操作ができなくなりますがお待ち下さい。',
58
+ 'ja-Hira': 'じゅんびにじかんがかかります。すこしのあいだ、そうさができなくなりますがおまちください。',
59
+ 'en': 'Setup takes a while. The browser will get stuck, but please wait.'
60
+ },
61
+ landmarks: [
62
+ {
63
+ 'ja': '手首',
64
+ 'ja-Hira': 'てくび',
65
+ 'en': 'wrist'
66
+ },
67
+ {
68
+ 'ja': '親指の根元',
69
+ 'ja-Hira': 'おやゆびのねもと',
70
+ 'en': 'the base of thumb'
71
+ },
72
+ {
73
+ 'ja': '親指の第2関節',
74
+ 'ja-Hira': 'おやゆびのだい2かんせつ',
75
+ 'en': 'the 2nd joint of thumb'
76
+ },
77
+ {
78
+ 'ja': '親指の第1関節',
79
+ 'ja-Hira': 'おやゆびのだい1かんせつ',
80
+ 'en': 'the 1st joint of thumb'
81
+ },
82
+ {
83
+ 'ja': '親指の先端',
84
+ 'ja-Hira': 'おやゆびのさき',
85
+ 'en': 'thumb'
86
+ },
87
+ {
88
+ 'ja': '人差し指の第3関節',
89
+ 'ja-Hira': 'ひとさしゆびのだい3かんせつ',
90
+ 'en': 'the 3rd joint of index finger'
91
+ },
92
+ {
93
+ 'ja': '人差し指の第2関節',
94
+ 'ja-Hira': 'ひとさしゆびのだい2かんせつ',
95
+ 'en': 'the 2nd joint of index finger'
96
+ },
97
+ {
98
+ 'ja': '人差し指の第1関節',
99
+ 'ja-Hira': 'ひとさしゆびのだい1かんせつ',
100
+ 'en': 'the 1st joint of index finger'
101
+ },
102
+ {
103
+ 'ja': '人差し指の先端',
104
+ 'ja-Hira': 'ひとさしゆびのせんたん',
105
+ 'en': 'index finger'
106
+ },
107
+ {
108
+ 'ja': '中指の第3関節',
109
+ 'ja-Hira': 'なかゆびのだい3かんせつ',
110
+ 'en': 'the 3rd joint of middle finger'
111
+ },
112
+ {
113
+ 'ja': '中指の第2関節',
114
+ 'ja-Hira': 'なかゆびのだい2かんせつ',
115
+ 'en': 'the 2nd joint of middle finger'
116
+ },
117
+ {
118
+ 'ja': '中指の第1関節',
119
+ 'ja-Hira': 'なかゆびのだい1かんせつ',
120
+ 'en': 'the 1st joint of middle finger'
121
+ },
122
+ {
123
+ 'ja': '中指の先端',
124
+ 'ja-Hira': 'なかゆびのせんたん',
125
+ 'en': 'middle finger'
126
+ },
127
+ {
128
+ 'ja': '薬指の第3関節',
129
+ 'ja-Hira': 'くすりゆびのだい3かんせつ',
130
+ 'en': 'the 3rd joint of ring finger'
131
+ },
132
+ {
133
+ 'ja': '薬指の第2関節',
134
+ 'ja-Hira': 'くすりゆびのだい2かんせつ',
135
+ 'en': 'the 2nd joint of ring finger'
136
+ },
137
+ {
138
+ 'ja': '薬指の第1関節',
139
+ 'ja-Hira': 'くすりゆびのだい1かんせつ',
140
+ 'en': 'the 1st joint of ring finger'
141
+ },
142
+ {
143
+ 'ja': '薬指の先端',
144
+ 'ja-Hira': 'くすりゆびのせんたん',
145
+ 'en': 'ring finger'
146
+ },
147
+ {
148
+ 'ja': '小指の第3関節',
149
+ 'ja-Hira': 'こゆびのだい3かんせつ',
150
+ 'en': 'the 3rd joint of little finger'
151
+ },
152
+ {
153
+ 'ja': '小指の第2関節',
154
+ 'ja-Hira': 'こゆびのだい2かんせつ',
155
+ 'en': 'the 2nd joint of little finger'
156
+ },
157
+ {
158
+ 'ja': '小指の第1関節',
159
+ 'ja-Hira': 'こゆびのだい1かんせつ',
160
+ 'en': 'the 1st joint of little finger'
161
+ },
162
+ {
163
+ 'ja': '小指の先端',
164
+ 'ja-Hira': 'こゆびのせんたん',
165
+ 'en': 'little finger'
166
+ }
167
+ ]
168
+ }
169
+ const AvailableLocales = ['en', 'ja', 'ja-Hira'];
170
+
171
+ class Scratch3Handpose2ScratchBlocks {
172
+ get LANDMARK_MENU () {
173
+ landmark_menu = [];
174
+ for (let i = 1; i <= 21; i++) {
175
+ landmark_menu.push({text: `${Message.landmarks[i - 1][this._locale]} (${i})`, value: String(i)})
176
+ }
177
+ return landmark_menu;
178
+ }
179
+
180
+ get VIDEO_MENU () {
181
+ return [
182
+ {
183
+ text: Message.off[this._locale],
184
+ value: 'off'
185
+ },
186
+ {
187
+ text: Message.on[this._locale],
188
+ value: 'on'
189
+ },
190
+ {
191
+ text: Message.video_on_flipped[this._locale],
192
+ value: 'on-flipped'
193
+ }
194
+ ]
195
+ }
196
+
197
+ get INTERVAL_MENU () {
198
+ return [
199
+ {
200
+ text: '0.1',
201
+ value: '0.1'
202
+ },
203
+ {
204
+ text: '0.2',
205
+ value: '0.2'
206
+ },
207
+ {
208
+ text: '0.5',
209
+ value: '0.5'
210
+ },
211
+ {
212
+ text: '1.0',
213
+ value: '1.0'
214
+ }
215
+ ]
216
+ }
217
+
218
+ get RATIO_MENU () {
219
+ return [
220
+ {
221
+ text: '0.5',
222
+ value: '0.5'
223
+ },
224
+ {
225
+ text: '0.75',
226
+ value: '0.75'
227
+ },
228
+ {
229
+ text: '1',
230
+ value: '1'
231
+ },
232
+ {
233
+ text: '1.5',
234
+ value: '1.5'
235
+ },
236
+ {
237
+ text: '2.0',
238
+ value: '2.0'
239
+ }
240
+ ]
241
+ }
242
+
243
+ constructor (runtime) {
244
+ this.runtime = runtime;
245
+
246
+ this.landmarks = [];
247
+ this.ratio = 0.75;
248
+
249
+ this.detectHand = () => {
250
+ this.video = this.runtime.ioDevices.video.provider.video;
251
+
252
+ alert(Message.please_wait[this._locale]);
253
+
254
+ const handpose = ml5.handpose(this.video, function() {
255
+ console.log("Model loaded!")
256
+ });
257
+
258
+ handpose.on('predict', hands => {
259
+ hands.forEach(hand => {
260
+ this.landmarks = hand.landmarks;
261
+ });
262
+ });
263
+ }
264
+ this.runtime.ioDevices.video.enableVideo().then(this.detectHand)
265
+ }
266
+
267
+ getInfo () {
268
+ this._locale = this.setLocale();
269
+
270
+ return {
271
+ id: 'handpose2scratch',
272
+ name: 'Handpose2Scratch',
273
+ blockIconURI: blockIconURI,
274
+ blocks: [
275
+ {
276
+ opcode: 'getX',
277
+ blockType: BlockType.REPORTER,
278
+ text: Message.getX[this._locale],
279
+ arguments: {
280
+ LANDMARK: {
281
+ type: ArgumentType.STRING,
282
+ menu: 'landmark',
283
+ defaultValue: '1'
284
+ }
285
+ }
286
+ },
287
+ {
288
+ opcode: 'getY',
289
+ blockType: BlockType.REPORTER,
290
+ text: Message.getY[this._locale],
291
+ arguments: {
292
+ LANDMARK: {
293
+ type: ArgumentType.STRING,
294
+ menu: 'landmark',
295
+ defaultValue: '1'
296
+ }
297
+ }
298
+ },
299
+ {
300
+ opcode: 'getZ',
301
+ blockType: BlockType.REPORTER,
302
+ text: Message.getZ[this._locale],
303
+ arguments: {
304
+ LANDMARK: {
305
+ type: ArgumentType.STRING,
306
+ menu: 'landmark',
307
+ defaultValue: '1'
308
+ }
309
+ }
310
+ },
311
+ {
312
+ opcode: 'videoToggle',
313
+ blockType: BlockType.COMMAND,
314
+ text: Message.videoToggle[this._locale],
315
+ arguments: {
316
+ VIDEO_STATE: {
317
+ type: ArgumentType.STRING,
318
+ menu: 'videoMenu',
319
+ defaultValue: 'off'
320
+ }
321
+ }
322
+ },
323
+ {
324
+ opcode: 'setVideoTransparency',
325
+ text: formatMessage({
326
+ id: 'videoSensing.setVideoTransparency',
327
+ default: 'set video transparency to [TRANSPARENCY]',
328
+ description: 'Controls transparency of the video preview layer'
329
+ }),
330
+ arguments: {
331
+ TRANSPARENCY: {
332
+ type: ArgumentType.NUMBER,
333
+ defaultValue: 50
334
+ }
335
+ }
336
+ },
337
+ {
338
+ opcode: 'setRatio',
339
+ blockType: BlockType.COMMAND,
340
+ text: Message.setRatio[this._locale],
341
+ arguments: {
342
+ RATIO: {
343
+ type: ArgumentType.STRING,
344
+ menu: 'ratioMenu',
345
+ defaultValue: '0.75'
346
+ }
347
+ }
348
+ }
349
+ ],
350
+ menus: {
351
+ landmark: {
352
+ acceptReporters: true,
353
+ items: this.LANDMARK_MENU
354
+ },
355
+ videoMenu: {
356
+ acceptReporters: true,
357
+ items: this.VIDEO_MENU
358
+ },
359
+ ratioMenu: {
360
+ acceptReporters: true,
361
+ items: this.RATIO_MENU
362
+ },
363
+ intervalMenu: {
364
+ acceptReporters: true,
365
+ items: this.INTERVAL_MENU
366
+ }
367
+ }
368
+ };
369
+ }
370
+
371
+ getX (args) {
372
+ let landmark = parseInt(args.LANDMARK, 10) - 1;
373
+ if (this.landmarks[landmark]) {
374
+ if (this.runtime.ioDevices.video.mirror === false) {
375
+ return -1 * (240 - this.landmarks[landmark][0] * this.ratio);
376
+ } else {
377
+ return 240 - this.landmarks[landmark][0] * this.ratio;
378
+ }
379
+ } else {
380
+ return "";
381
+ }
382
+ }
383
+
384
+ getY (args) {
385
+ let landmark = parseInt(args.LANDMARK, 10) - 1;
386
+ if (this.landmarks[landmark]) {
387
+ return 180 - this.landmarks[landmark][1] * this.ratio;
388
+ } else {
389
+ return "";
390
+ }
391
+ }
392
+
393
+ getZ (args) {
394
+ let landmark = parseInt(args.LANDMARK, 10) - 1;
395
+ if (this.landmarks[landmark]) {
396
+ return this.landmarks[landmark][2];
397
+ } else {
398
+ return "";
399
+ }
400
+ }
401
+
402
+ videoToggle (args) {
403
+ let state = args.VIDEO_STATE;
404
+ if (state === 'off') {
405
+ this.runtime.ioDevices.video.disableVideo();
406
+ } else {
407
+ this.runtime.ioDevices.video.enableVideo().then(this.detectHand);
408
+ this.runtime.ioDevices.video.mirror = state === "on";
409
+ }
410
+ }
411
+
412
+
413
+ /**
414
+ * A scratch command block handle that configures the video preview's
415
+ * transparency from passed arguments.
416
+ * @param {object} args - the block arguments
417
+ * @param {number} args.TRANSPARENCY - the transparency to set the video
418
+ * preview to
419
+ */
420
+ setVideoTransparency (args) {
421
+ const transparency = Cast.toNumber(args.TRANSPARENCY);
422
+ this.globalVideoTransparency = transparency;
423
+ this.runtime.ioDevices.video.setPreviewGhost(transparency);
424
+ }
425
+
426
+ setRatio (args) {
427
+ this.ratio = parseFloat(args.RATIO);
428
+ }
429
+
430
+ setLocale() {
431
+ let locale = formatMessage.setup().locale;
432
+ if (AvailableLocales.includes(locale)) {
433
+ return locale;
434
+ } else {
435
+ return 'en';
436
+ }
437
+ }
438
+ }
439
+
440
+ module.exports = Scratch3Handpose2ScratchBlocks;