soiz1 commited on
Commit
93ff48c
·
verified ·
1 Parent(s): 4822619

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,26 +2,42 @@ 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
 
6
- // eslint-disable-next-line max-len
7
  const blockIconURI = 'data:image/png;base64,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';
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,7 +58,7 @@ const Message = {
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,200 +169,272 @@ const Message = {
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;
 
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 = 'data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAEgAAABICAYAAABV7bNHAAAABGdBTUEAALGPC/xhBQAAACBjSFJNAAB6JgAAgIQAAPoAAACA6AAAdTAAAOpgAAA6mAAAF3CculE8AAAAhGVYSWZNTQAqAAAACAAFARIAAwAAAAEAAQAAARoABQAAAAEAAABKARsABQAAAAEAAABSASgAAwAAAAEAAgAAh2kABAAAAAEAAABaAAAAAAAAAEgAAAABAAAASAAAAAEAA6ABAAMAAAABAAEAAKACAAQAAAABAAAASKADAAQAAAABAAAASAAAAABjCyvsAAAACXBIWXMAAAsTAAALEwEAmpwYAAACMmlUWHRYTUw6Y29tLmFkb2JlLnhtcAAAAAAAPHg6eG1wbWV0YSB4bWxuczp4PSJhZG9iZTpuczptZXRhLyIgeDp4bXB0az0iWE1QIENvcmUgNS40LjAiPgogICA8cmRmOlJERiB4bWxuczpyZGY9Imh0dHA6Ly93d3cudzMub3JnLzE5OTkvMDIvMjItcmRmLXN5bnRheC1ucyMiPgogICAgICA8cmRmOkRlc2NyaXB0aW9uIHJkZjphYm91dD0iIgogICAgICAgICAgICB4bWxuczp0aWZmPSJodHRwOi8vbnMuYWRvYmUuY29tL3RpZmYvMS4wLyIKICAgICAgICAgICAgeG1sbnM6ZXhpZj0iaHR0cDovL25zLmFkb2JlLmNvbS9leGlmLzEuMC8iPgogICAgICAgICA8dGlmZjpPcmllbnRhdGlvbj4xPC90aWZmOk9yaWVudGF0aW9uPgogICAgICAgICA8ZXhpZjpDb2xvclNwYWNlPjE8L2V4aWY6Q29sb3JTcGFjZT4KICAgICAgICAgPGV4aWY6UGl4ZWxYRGltZW5zaW9uPjE5MjwvZXhpZjpQaXhlbFhEaW1lbnNpb24+CiAgICAgICAgIDxleGlmOlBpeGVsWURpbWVuc2lvbj4xOTI8L2V4aWY6UGl4ZWxZRGltZW5zaW9uPgogICAgICA8L3JkZjpEZXNjcmlwdGlvbj4KICAgPC9yZGY6UkRGPgo8L3g6eG1wbWV0YT4KjxrQ6wAAFMJJREFUeAHtW3l0leWdfpLcLDf7vhFCSAgJkEBE3KDqICoW4YBWHT1HD7UepzMdph2dqR471Tp6nBanzj91ju2pB8pxOp0OuPRUq61bFSWisgiENYSQhITsC9nXeZ73u29yidnuTSw6h1+497vf973r8/729yUg/u1nhnCRxkUgcNw3F18YBC4CNAkjXAToIkCTIDDJ64scdBGgSRCY5PVFDvoqADTIQQ5MMtAL9fqCclCAB5iogAAkBASin/d69mUi14UcjACJJjiVA73A0CAyg8LQiiEEXchBjer7gnGQ4psI8ktlfy9ui0jGd+KyUTHQg8gvGQ9dUIBCtFpD/dhcsBaPF94MBLnQQU5yf4lAuiAASSHHU+eU93fhO/G5yI5KQlJoJB6OzUbLQDfCKHZflgj6Lw6QlHA/px9NgECRWh6bIT4ylBeVQo4aQDA5SABJR3Xzl666n0yB6721iDMF8F9cSduJDwzxV6ALCe5oTsuh3MgE/ghAODkoCYEI4+8g/u5k2TaKXidh0oqOtaoCp4+faNaJJfjNLN/O8prgdMCaEYA0AH3GGjgfm3d6H87BuznhFF6H9JTiFB0cpiKGYkIj+KwLe3tpx8hJsmzmExgMuEKRx6vaEWACS9yiPgWOOC0tIAilA32o6u9g+XCks3wHn0/HKk4LILtqMRxiJFftLCelZ5bswGM5jfjAQDQMDuBkXxflgGsdEoWbYnOQ7o61xZEdnYKjV20ybbT296CptxPNXW041F6HP52rxiddjQ5gnHwhJ99KiJoJ1iBBmBvgwqG+DlwTnoDHctZhe9U+/KKlHFkut+Emf0GaFkDSDTLLlYP8RX2SGRyBNg7WclIP38/hwBtpqQ52t3FV3XgwMQ+rk+djQdwspIbHIli6iCTOiHCFIC8u3dyP/nqorxulrbXYWXcCz9QdYXuNcAVHIodiOsQxHOrvxCq6Cy9ccjvSwmNYtgZoOIoojqmeCxdkYB/d6uT3fgMk9pYfU01uWOuOx9KoVDxRdwiZgaHG2YulKEmkDve2GnZ/ds7VWD2rADnRyWMOVdwmkIbIEUb8dMOHuncFBhlRXJo4B/rclX0F3qk+jLtOF+MYuQbkpvzQ6GFw+ghISUcD3YZQ4za4xuyR7U+B/AZI3KPwoJoDvC/3emyYsxTd+/rwdH0JrgqNRXFfp+Gqp9KX4e65VyIzMn54OIPSLRx0IEH0JnsXKK6yN56rgY1gBfAvOSwKdxKklWkL8HzpTvzw7AH8bsk3DOeovdrONvys9TRSXWFoYV/+ipfa8hsgqVYpRq1eTmSi2sLGrCvwdONxFHe34LLwRDybfyMuT8o27yxnBBAUA4B5ev6Xyuh9LcXx1Yr9aOntwvUEYUlCpgFG70QOwEAKLeDDBWtwT/ZVyIyIN8/V9js1h4HeDsRyoWqMeJ3fjy93fgEk7teq9AogDjoiyPjEyKf+2BidgcigYDxZuA5xIW6PyBBHltPfeGTBae/rwb17d+D15lKjs1C1C3uW3UvRyiIA1G9sxwKsexfvBc6AOIXg1FGpP3ZmL0Dd0z5N7tFY/QJI05Q3HKIJcxDtCjZJUrc/Wrgaqe4YuAmSHfT4sJhq5sujcnCguRKvt1VgeWS64aYPu5rxOjlCADnqfKSOFVEDnAf8lyr24nRnAy5xJ6CMxmM64qWeRvc50vskv+SUhXoAqqc5FgmQuRQ3gSMx0Ir6Sm7DjVT+9Gf6ySHyleI8vpJAHIsElH13dXKucSEqWV9GYiqLM1ab9pnvM2BNdSrxkpcr21PTTv+EpOE4Nkii4FvTVr8sjp+NJ6nYywnM7p425JKT1mcuNe1P9GVAIqCL6D68kbcaDX3tiOMYpCenA5Jvs/CM0K6WkU86f2H0X0QaiKNpfB+SakhUBPqi6FSuQCu2Zl+LPcvvwyz6SwLegqi+xiSzYMCNGYV4ILkAhwlSOr1rRwGMWWPSh34BJP2TTQdtT08rvp+2FOuzLnU68sJFk5XiNVdObjISAOKCPorm63XHKCYxhnOiGGJIdCcSFrWujwOyygLfomuhp+LpUP7yl/wCKIYTKWGiK58O4kMLbjDesHSO9yQ0Wa24uZphTgKS53U9rdAvW07in5gGifVYwfF0mQGGiyBA9LGWUGAsoEX9XkIejtNPSyYXaVH9IZ8A0iAUPpisH3XE1vybkBgWaVZ4tM4paT6DZ4+9ixdP70E74yqBpwmMR/bNidazQFcDrk3KcSY9TgWVN8BwEXoG+tFLUdeC6M8YCF7XpCyg3DrQMNz1i3wy81J4iq0UPshDvjI5R8s2bK2MueUg9zaU49I92zgDNs8YaVNjGX5adCtCGTIYXWKmdv54NVnR7sZT9GEisTB2lvNgnG9b/nen9+HZyk/gZtv/OHc5rqNjaakgLsNYtHKaeyXolAHwlabMQWpaVuGsAlM6YRtzlpu+FDRYMpaEN/9TtZ/fLlweGoNLw1PwbP0RJ3jk07GYyCrgc+S0ZxpO4GZmFtMi4kyzFghz4/nSQoh21ZZiQ8l2vEVd+PuOOqw68L840lLjsaBDSAqPxrej0tA52IsQLpw/NGWA1LgAauo7h22ZVzqWhQMVKN6ku1DKPIb6mLcZRJ0AJbn0bByyolfeVo+69iqsS8hBOH0pb53iXdX2WNxUTpc+HIsZlK4IUeItAKcUxZMGBgeNbiyIZJaSY5CoeC+mKTSFrykBpAEpcj9I52sWPdSvZyw2TY9mWLuyd2XSqoVE0tEb5K5FNx5NuwQ5MTTdpFF4mmeONgH2N1UYFiuKm22ei7PGIvv0EokhB+fiQu02wXEvOU9ZSTbjqThPcSLHEciCA8NPPS+ncJkUIIEjP8LkkGkRfpKxDEmMpqUIR3OP7vV8IS3IoUvuxNHeNjwYPw+PFa41MZPhCM3Ii/RM9XpZ723qKlAklThz6PyytpoFLorRumgvRaifOyK/zluLIqZDHHLqpjPs0apISSeRi5XGdXIJnmKTXCZV0loJbcOcYt4ng2b9JjphDo09eNvfHK0kRTKN+koBpVXg9r29yseR+DXQvG9rKcM/Jy3iAkSa16z2OVI7Mvud5OanS98z73cWbEBBfAZig93D5e3ihQTTCwoONwtQptwRxTGdPpzNbw9XGOfHpBwky5VKC9FF3fOTjMuQyO2Z8Sbr3YdMrwBq5+qKNGDL9racuEfJsC5O9s3qEmbdO3BDcp55PdqvsnXs9fXKz7Cj/jP8Z+YKfC2FPhPBUXuWu2y5CAFE5X+MC/xoSiHWhcUyyddv4rTR47F1vK8TAiRWFFseYDo1OSwBNzIjKBpjYc1z769A5qAFkAk4vV94fgtk+S2VHU24ungLvnlqJ/VWNKq7WkyJsQC1C1PZ0YzbSt/B/JgckzhTBeNtsz39eZOLY/heYj6OLLsbTxStx7pEBrMMQRK5MCZd4114jN8TAqSutI2ilX3G6J7IcS3LSNteA+TgpFtGaGTNbKkXyj7CnnNVWEZRDCX733v6Q5Sdq2cVOZbedR0uVFvbynbRmazFc7mrEB8Sbjh6tLdtRUwcv7loA/Jj000Ys7OtilrdjVa2rf23yWhcgFTVWC6yZkJYPG7wcM/IFCdu2llJB6DzpykL43CPntf2nOOAw41OWEKAZJKVSRxN/R6PeHd9GR6t2IkHZl2Fa9LzTTEjWORIcdhoEQsmp4RwoUQfnj2OF5rLkE/l3kSAxnc8THHzNSZAAkf5nig1TO756aylJr1pWXyk+gS/1AjFTBw0GiAbDqjzmxQOMC3aQ2A+7qjB6sg05FmXQFykP05cuqqDibkfHX+HgEbgb+ddTd/GCV/EPePFfVb06rrP4f7jb1FJu81emazaVBZ7TCumikpg1HLQoDds/R4+8oGIEAfeQ4A0wXEcIKc99pPvjsOmtMW4fc4ys/3j1JGgEQQ21cJtn5fKP8Ufm47g+bwNmO9xBfTu7TMlDFHK+SwZN88ucpxMTZ//BJ7itMcP/QGlzJUv5H7cGR/y1GMCJMs1j8n4A91N2JJzve/cw2nLRIsDezkYbexpqvbicGIgJ92F7xvlHIWfL7kFGcz7iKyYCJw2AvP04Tfx46aTlL4e/FVcHm6bu8yU09eLBO2Okhe5oqxb0YKnqL8eKVxjgFWXMhL/ceRNPMc9ssVhcSilypAvNFX6nIhpKomc2DHlmbkrsMbjNU+1QVtOk4uj/9orvSAOIjnftgTwzpnDKGk5gf+as9yAI9Mu8FTX0gsni/FU5QdQCCp90s0yvQNOhN7FMf6cO6gITcDSkAgURqTiX+pKcIZWztIxhh6PVO3GEs7lBDnV16M1nwNIDcdINGgKt9JyaWtFE7RWwXY86ZUspO3oHnKQ5QjVsW019nTggfIPERSVRefTCV3EZca8ewBV+fJOpnNpJNx07i6lEv+osx4VnhRvMJ9lh0ax1KDZQDioReWzUHrVlnqMwudC8YFyCb7SSEusqXUTwkfouIWQHe3ANUHvVZ1KJ6YtA5BHB3na1xD17i06hhWtJ7FjyT1I4KEFawD0Xv0pvyR/p0R+EUXrKP2xdsVbdPRmezYhj7RU4w8ddAkIzEcUHVnAl3LXmVBIwWoQjUR1TzsbdIRc8ZivNAyQqgrlFA7sDGOorZmruH0T7Rn4mIw2YV+yKmEcoFHSnpIappRmfXc77qTuKaI+uS59kXmr/i1IASyzr7ECKw++jNbuZnybmcHDTODH8GDC47krzc7qMSbWFu/7La1sJ15esBYJ9Ifi6PMUMGkvkg4UVetMAG/Ut2bhKw8NA6SKCv0quRJx5J6bZy/hnf8kjnNzou20IFYHmdFx4K8xTADTGk8UbRzeXLTgqcfXKg9g7eFXJI94ddGtWD27EC09nQhjCiSSGwRl7Q1YuX87t3Zb8MaSvzZ7/t4jNRzvQaiNiyGetZzrXW4qv4cBGrZcPGLy33lrPBG7H7rH06s4KJRr1sC8kNVBhuU7W3FvxS5cE5+Pa9Olep0wQX5OD8F8/vj72FT6Btzc7vmg8BYs5bazKFFnh0gVFLs79+3gVlMNXim8w4CjBRDAYhqT+iUa6r+bIndAOox6SVkpD1OpmSmTkR05cimKuZi7WRyRMhyx+9Og7Vl15cHqUOYwB/HZa8o2dlTjB1nLEc0diz5OQuA0UFc8vP9lbDr+e9ydVIjjl28cBse2eZbicv/+F/FJ6yn8duEt3PUo4iuh4fg7Ni9uF6SDCvrPXU0EKHjcmNC2Pd7VcJBQMol45o//dd71ZHsnvvHZcrEdC6rhIA9A5rgd35XT+vzN6V0MGAvwtTQnapclkj6598ArKG4pxVPZN+C7+auMKGnQ5RSnbTT1isuOchv6TwTnV/nrccfcy/RaUmi4xdyM+qphCqWSkcBsWj9FBv6QAUgn3Q9yl+K6yFlYNWuhP+2M1PEgJIsRSiVdx4k5GgB4mfvm6KrHdxetHz7w8Oeao1h5iPqGE/lNwR2c+OVGmapB7Yas2cs8cwd3OhjMgo7lj2evwMZ5y01/4kwtxHhU19liLFwUY706uhv+kEu6J01hG8XrEe4KRHEf3MkW+m65vAfgcFAQ+uW5Urme4WAfrPgAGyg+Kz26Z1vpLnzz2KsmnHlv6T24JnW+dxN4t/oIjtCKXeFOwgCB+JTiOCxGE4BjXZLj7fWmPc1E8IwP5XndnnfjSqXuKaFobWRq1CpNmdnpkjjIOfam3dIhvML9MfDY3GPLvmUG+wj1zb+ffhcrEwqwpegbyDInXBn/USy0bbS95iC2Np/krIJwnH5Org41kMsiqbcmImvBFH990nrG6B+BIz07leh9dNtOVper/A/ZK4Z3SO0qjS7sy70g1mcFcy+fMRm/qepjPJC5EhGh4bi7eBu21x/A5pw1+PsFqyiKLnzMNMbbZ4/iB7UHeIy1lj5Hotl7k456qGY/Pu5pwW30h27PGonDxhyPx543Mnrf0lGLWALbRnj8XXLX4d5z+GFqES7l+RvR9LnHYWSNM5KT+5Ci+7NTxWoZyg/f/9lLeJ97WM8tuAXXpszHHys+w79VfYI9VNCyNncnLsRtuTdyPHOR4dkb20CfTEHrAia9JtoO0vg9+OC0wpFeHl7gHr9OmfkjXmrPBfLQfdmO0psJ3WOHqBXTqVWJxRsdQ8hkP5t5OlUzuCUmEyeoH/6u7D2KXR3Co7OZHVxLHZRnDnmKo7wpd3iXQ1ZrYsVs6x3V/hgNRDB1Vw/r+BLB2zZ0dW2bswJZUYl+hxTejZnfniX8tOEUtjSXI4E7ClHUI10cbEFgCA7RWr4s3UJueTBpATak347C+ExzUMG2pZBDSFputqlXKd+JrJbqyzXRCZH3myuodHjilr/NSTjbuI/XAJ4JHNJKG+XmNyM6vdrVlaJN3fULasYBZJNzOjjhWplZ/ueVW6Nn4w5mEa9InjesmFVbdQ22lAVrhXycy3AsJ/HK2r0FGVwYbvRMa1aumQLHTNIzmLJzdcZtuIoxnbZYAriK14XFYHPe7SaYDPNKRyglKosnzvBXT4wA6bDvYZ4sURAbxRxQKxdG2VG98YcCZ4JzbMd2ghk8dQqGD8VMyCdy2mc7a3A/06nLuOspcMQt0nfq2+aTbRv+XgWAtb4fNZXzxmW2daTN/AVHYwmciXVTQyJxgSY9mwC9tWgD5oZFYw+9niezrsNaEzc5KQ2V02Rmsm+BLmpkTLejpZL6J4ThhRPAmhd+fgWw4ekAPGG3zQwWOxku6IyhaCa5dXTH1gLvqj2BFft/gzxuQuq/RCmKnw6db0+n05JXXYM4cddBcn1EWofJLJBXEz79VH9WvN6vpz/FBzoP1Murvw6iHcAXApDRRRI3y5z8/UWBYyaiftiHxOuXjQSI3vs527edqZ/X6QI8YbcCxXwmLDX9l+IgUUlTFcqY8FtM/aOd05mY3Ey04YzuAn7T0hh6l/+XTMGtDppa0KY7rC9USU93cFOpbxP9NUynpNM5TKZo0fOaStUplfl/wEEOrxTXMXyh35XAk7WyXDMF0VcaIBkBWS8l+186y/8jxtiri8I1XefQm7W+2gB5NM1Rhha/5n+hyhFAM2S9LEhfWYAkWNb3UaJNgXG4NipnTD07EP0f9vkI6bCxOM4AAAAASUVORK5CYII=';
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
  '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
  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;