soiz1 commited on
Commit
ddeba6f
·
verified ·
1 Parent(s): a0baee5

Create scratch3_new_handpose2scratch/index.js

Browse files
local-scratch-vm/src/extensions/scratch3_new_handpose2scratch/index.js ADDED
@@ -0,0 +1,352 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
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
+
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': 'いり',
28
+ 'en': 'on'
29
+ },
30
+ off: {
31
+ 'ja': '切',
32
+ 'ja-Hira': 'きり',
33
+ 'en': 'off'
34
+ },
35
+ video_on_flipped: {
36
+ 'ja': '左右反転',
37
+ 'ja-Hira': 'さゆうはんてん',
38
+ 'en': 'on flipped'
39
+ },
40
+ please_wait: {
41
+ 'ja': '準備に時間がかかります。少しの間、操作ができなくなりますがお待ち下さい。',
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': 'てくび',
49
+ 'en': 'wrist'
50
+ },
51
+ {
52
+ 'ja': '親指の根元',
53
+ 'ja-Hira': 'おやゆびのねもと',
54
+ 'en': 'the base of thumb'
55
+ },
56
+ {
57
+ 'ja': '親指の第2関節',
58
+ 'ja-Hira': 'おやゆびのだい2かんせつ',
59
+ 'en': 'the 2nd joint of thumb'
60
+ },
61
+ {
62
+ 'ja': '親指の第1関節',
63
+ 'ja-Hira': 'おやゆびのだい1かんせつ',
64
+ 'en': 'the 1st joint of thumb'
65
+ },
66
+ {
67
+ 'ja': '親指の先端',
68
+ 'ja-Hira': 'おやゆびのさき',
69
+ 'en': 'thumb'
70
+ },
71
+ {
72
+ 'ja': '人差し指の第3関節',
73
+ 'ja-Hira': 'ひとさしゆびのだい3かんせつ',
74
+ 'en': 'the 3rd joint of index finger'
75
+ },
76
+ {
77
+ 'ja': '人差し指の第2関節',
78
+ 'ja-Hira': 'ひとさしゆびのだい2かんせつ',
79
+ 'en': 'the 2nd joint of index finger'
80
+ },
81
+ {
82
+ 'ja': '人差し指の第1関節',
83
+ 'ja-Hira': 'ひとさしゆびのだい1かんせつ',
84
+ 'en': 'the 1st joint of index finger'
85
+ },
86
+ {
87
+ 'ja': '人差し指の先端',
88
+ 'ja-Hira': 'ひとさしゆびのせんたん',
89
+ 'en': 'index finger'
90
+ },
91
+ {
92
+ 'ja': '中指の第3関節',
93
+ 'ja-Hira': 'なかゆびのだい3かんせつ',
94
+ 'en': 'the 3rd joint of middle finger'
95
+ },
96
+ {
97
+ 'ja': '中指の第2関節',
98
+ 'ja-Hira': 'なかゆびのだい2かんせつ',
99
+ 'en': 'the 2nd joint of middle finger'
100
+ },
101
+ {
102
+ 'ja': '中指の第1関節',
103
+ 'ja-Hira': 'なかゆびのだい1かんせつ',
104
+ 'en': 'the 1st joint of middle finger'
105
+ },
106
+ {
107
+ 'ja': '中指の先端',
108
+ 'ja-Hira': 'なかゆびのせんたん',
109
+ 'en': 'middle finger'
110
+ },
111
+ {
112
+ 'ja': '薬指の第3関節',
113
+ 'ja-Hira': 'くすりゆびのだい3かんせつ',
114
+ 'en': 'the 3rd joint of ring finger'
115
+ },
116
+ {
117
+ 'ja': '薬指の第2関節',
118
+ 'ja-Hira': 'くすりゆびのだい2かんせつ',
119
+ 'en': 'the 2nd joint of ring finger'
120
+ },
121
+ {
122
+ 'ja': '薬指の第1関節',
123
+ 'ja-Hira': 'くすりゆびのだい1かんせつ',
124
+ 'en': 'the 1st joint of ring finger'
125
+ },
126
+ {
127
+ 'ja': '薬指の先端',
128
+ 'ja-Hira': 'くすりゆびのせんたん',
129
+ 'en': 'ring finger'
130
+ },
131
+ {
132
+ 'ja': '小指の第3関節',
133
+ 'ja-Hira': 'こゆびのだい3かんせつ',
134
+ 'en': 'the 3rd joint of little finger'
135
+ },
136
+ {
137
+ 'ja': '小指の第2関節',
138
+ 'ja-Hira': 'こゆびのだい2かんせつ',
139
+ 'en': 'the 2nd joint of little finger'
140
+ },
141
+ {
142
+ 'ja': '小指の第1関節',
143
+ 'ja-Hira': 'こゆびのだい1かんせつ',
144
+ 'en': 'the 1st joint of little finger'
145
+ },
146
+ {
147
+ 'ja': '小指の先端',
148
+ 'ja-Hira': 'こゆびのせんたん',
149
+ 'en': 'little finger'
150
+ }
151
+ ]
152
+ }
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;