Spaces:
Running
Running
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 = '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': 'いり',
|
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;
|