soiz1 commited on
Commit
7f440b7
·
verified ·
1 Parent(s): f15cecf

Update local-scratch-vm/src/extensions/scratch3_new_handpose2scratch/index.js

Browse files
local-scratch-vm/src/extensions/scratch3_new_handpose2scratch/index.js CHANGED
@@ -2,42 +2,26 @@ 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': 'いり',
@@ -58,7 +42,7 @@ const Message = {
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': 'てくび',
@@ -169,272 +153,200 @@ const Message = {
169
  const AvailableLocales = ['en', 'ja', 'ja-Hira'];
170
 
171
  class Scratch3Handpose2ScratchBlocks {
172
- get LANDMARK_MENU () {
173
- const 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;
 
2
  const BlockType = require('../../extension-support/block-type');
3
  const Cast = require('../../util/cast');
4
  const formatMessage = require('format-message');
 
 
5
 
6
+ // eslint-disable-next-line max-len
7
  const blockIconURI = '';
8
 
9
  const Message = {
10
  getX: {
11
+ 'ja': '[HAND] つ目の [KEYPOINT] のx座標',
12
+ 'ja-Hira': '[HAND] つめの [KEYPOINT] のxざひょう',
13
+ 'en': 'x of [KEYPOINT] of hand no. [HAND]'
14
  },
15
  getY: {
16
+ 'ja': '[HAND] つ目の [KEYPOINT] のy座標',
17
+ 'ja-Hira': 'みぎ [KEYPOINT] のyざひょう',
18
+ 'en': 'y of [KEYPOINT] of hand no. [HAND NUMBER]'
 
 
 
 
 
19
  },
20
  videoToggle: {
21
  'ja': 'ビデオを [VIDEO_STATE] にする',
22
  'ja-Hira': 'ビデオを [VIDEO_STATE] にする',
23
  'en': 'turn video [VIDEO_STATE]'
24
  },
 
 
 
 
 
 
 
 
 
 
25
  on: {
26
  'ja': '入',
27
  'ja-Hira': 'いり',
 
42
  'ja-Hira': 'じゅんびにじかんがかかります。すこしのあいだ、そうさができなくなりますがおまちください。',
43
  'en': 'Setup takes a while. The browser will get stuck, but please wait.'
44
  },
45
+ keypoints: [
46
  {
47
  'ja': '手首',
48
  'ja-Hira': 'てくび',
 
153
  const AvailableLocales = ['en', 'ja', 'ja-Hira'];
154
 
155
  class Scratch3Handpose2ScratchBlocks {
156
+ get HANDS_MENU() {
157
+ return Array.from({ length: 10 }, (_, i) => ({ text: `${i + 1}`, value: `${i + 1}` }));
158
+ }
 
 
 
 
159
 
160
+ get KEYPOINTS_MENU() {
161
+ const keypoints = [];
162
+ for (let i = 1; i <= 21; i++) {
163
+ keypoints.push({ text: `${Message.keypoints[i - 1][this._locale]} (${i})`, value: String(i) })
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
164
  }
165
+ return keypoints;
166
+ }
167
 
168
+ get VIDEO_MENU() {
169
+ return [
170
+ {
171
+ text: Message.off[this._locale],
172
+ value: 'off'
173
+ },
174
+ {
175
+ text: Message.on[this._locale],
176
+ value: 'on'
177
+ },
178
+ {
179
+ text: Message.video_on_flipped[this._locale],
180
+ value: 'on-flipped'
181
+ }
182
+ ]
183
+ }
 
 
 
 
 
 
 
 
184
 
185
+ constructor(runtime) {
186
+ this.runtime = runtime;
187
+ this.keypoints = [];
188
 
189
+ const loadScriptSynchronously = (url) => {
190
+ const request = new XMLHttpRequest();
191
+ request.open('GET', url, false);
192
+ request.send(null);
193
+ if (request.status === 200) {
194
+ const script = document.createElement('script');
195
+ script.text = request.responseText;
196
+ document.head.appendChild(script);
197
+ }
198
+ };
199
 
200
+ loadScriptSynchronously('https://unpkg.com/ml5@1/dist/ml5.min.js');
 
201
 
202
+ ml5.handPose((handpose) => {
203
+ console.log("Model loaded!")
204
+ handpose.detectStart(this.video, (results) => {
205
+ this.hands = results;
206
+ });
207
+ });
208
 
209
+ this.runtime.ioDevices.video.enableVideo().then(() => {
210
+ this.video = this.runtime.ioDevices.video.provider.video
211
+ this.video.width = 480;
212
+ this.video.height = 360;
213
+ });
214
+ }
215
 
216
+ getInfo() {
217
+ this._locale = this.setLocale();
 
 
 
 
 
 
 
 
 
218
 
219
+ return {
220
+ id: 'handpose2scratch',
221
+ name: 'Handpose2Scratch',
222
+ blockIconURI: blockIconURI,
223
+ blocks: [
224
+ {
225
+ opcode: 'getX',
226
+ blockType: BlockType.REPORTER,
227
+ text: Message.getX[this._locale],
228
+ arguments: {
229
+ HAND: {
230
+ type: ArgumentType.STRING,
231
+ menu: 'handsMenu',
232
+ defaultValue: '1'
233
+ },
234
+ KEYPOINT: {
235
+ type: ArgumentType.STRING,
236
+ menu: 'keypointsMenu',
237
+ defaultValue: '1'
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
238
  }
239
+ }
240
+ },
241
+ {
242
+ opcode: 'getY',
243
+ blockType: BlockType.REPORTER,
244
+ text: Message.getY[this._locale],
245
+ arguments: {
246
+ HAND: {
247
+ type: ArgumentType.STRING,
248
+ menu: 'handsMenu',
249
+ defaultValue: '1'
250
+ },
251
+ KEYPOINT: {
252
+ type: ArgumentType.STRING,
253
+ menu: 'keypointsMenu',
254
+ defaultValue: '1'
255
+ }
256
+ }
257
+ },
258
+ {
259
+ opcode: 'videoToggle',
260
+ blockType: BlockType.COMMAND,
261
+ text: Message.videoToggle[this._locale],
262
+ arguments: {
263
+ VIDEO_STATE: {
264
+ type: ArgumentType.STRING,
265
+ menu: 'videoMenu',
266
+ defaultValue: 'off'
267
+ }
268
+ }
269
+ },
270
+ {
271
+ opcode: 'setVideoTransparency',
272
+ text: formatMessage({
273
+ id: 'videoSensing.setVideoTransparency',
274
+ default: 'set video transparency to [TRANSPARENCY]',
275
+ description: 'Controls transparency of the video preview layer'
276
+ }),
277
+ arguments: {
278
+ TRANSPARENCY: {
279
+ type: ArgumentType.NUMBER,
280
+ defaultValue: 50
281
+ }
282
+ }
283
  }
284
+ ],
285
+ menus: {
286
+ keypointsMenu: {
287
+ acceptReporters: true,
288
+ items: this.KEYPOINTS_MENU
289
+ },
290
+ videoMenu: {
291
+ acceptReporters: true,
292
+ items: this.VIDEO_MENU
293
+ },
294
+ handsMenu: {
295
+ acceptReporters: true,
296
+ items: this.HANDS_MENU
297
+ },
298
  }
299
+ };
300
+ }
301
 
302
+ getX(args) {
303
+ let keypoint = parseInt(args.KEYPOINT, 10) - 1;
304
+ let hand = parseInt(args.HAND, 10) - 1;
305
+ if (this.hands?.[hand]?.keypoints?.[keypoint]) {
306
+ if (this.runtime.ioDevices.video.mirror === false) {
307
+ return -1 * (240 - this.hands[hand].keypoints[keypoint].x);
308
  } else {
309
+ return 240 - this.hands[hand].keypoints[keypoint].x;
310
  }
311
+ } else {
312
+ return '';
313
  }
314
+ }
315
 
316
+ getY(args) {
317
+ let keypoint = parseInt(args.KEYPOINT, 10) - 1;
318
+ let hand = parseInt(args.HAND, 10) - 1;
319
+ if (this.hands?.[hand]?.keypoints?.[keypoint]) {
320
+ return 180 - this.hands[hand].keypoints[keypoint].y;
321
+ } else {
322
+ return '';
323
  }
324
+ }
325
 
326
+ videoToggle(args) {
327
+ let state = args.VIDEO_STATE;
328
+ if (state === 'off') {
329
+ this.runtime.ioDevices.video.disableVideo();
330
+ } else {
331
+ this.runtime.ioDevices.video.enableVideo().then(this.detectHand);
332
+ this.runtime.ioDevices.video.mirror = state === "on";
 
333
  }
334
+ }
335
 
336
+ setVideoTransparency(args) {
337
+ const transparency = Cast.toNumber(args.TRANSPARENCY);
338
+ this.globalVideoTransparency = transparency;
339
+ this.runtime.ioDevices.video.setPreviewGhost(transparency);
340
+ }
341
 
342
+ setLocale() {
343
+ let locale = formatMessage.setup().locale;
344
+ if (AvailableLocales.includes(locale)) {
345
+ return locale;
346
+ } else {
347
+ return 'en';
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
348
  }
349
+ }
350
  }
351
 
352
  module.exports = Scratch3Handpose2ScratchBlocks;