soiz1 commited on
Commit
1cfb0b0
·
verified ·
1 Parent(s): 99daed9

Create extensions/scratch3_facemesh2scratch/index.js

Browse files
local-scratch-vm/src/extensions/scratch3_facemesh2scratch/index.js ADDED
@@ -0,0 +1,354 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
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
+ const blockIconURI = '';
8
+
9
+ const Message = {
10
+ getX: {
11
+ 'ja': '[PERSON_NUMBER] 人目の [KEYPOINT] 番目の部位のx座標',
12
+ 'ja-Hira': '[PERSON_NUMBER] にんめの [KEYPOINT] ばんめのぶいのxざひょう',
13
+ 'en': 'x of person no: [PERSON_NUMBER] , keypoint no: [KEYPOINT]'
14
+ },
15
+ getY: {
16
+ 'ja': '[PERSON_NUMBER] 人目の [KEYPOINT] 番目の部位のy座標',
17
+ 'ja-Hira': '[PERSON_NUMBER] にんめの [KEYPOINT] ばんめのぶいのyざひょう',
18
+ 'en': 'y of person no: [PERSON_NUMBER] , keypoint no: [KEYPOINT]'
19
+ },
20
+ peopleCount: {
21
+ 'ja': '人数',
22
+ 'ja-Hira': 'にんずう',
23
+ 'en': 'people count'
24
+ },
25
+ videoToggle: {
26
+ 'ja': 'ビデオを [VIDEO_STATE] にする',
27
+ 'ja-Hira': 'ビデオを [VIDEO_STATE] にする',
28
+ 'en': 'turn video [VIDEO_STATE]'
29
+ },
30
+ setRatio: {
31
+ 'ja': '倍率を [RATIO] にする',
32
+ 'ja-Hira': 'ばいりつを [RATIO] にする',
33
+ 'en': 'set ratio to [RATIO]'
34
+ },
35
+ setInterval: {
36
+ 'ja': '認識を [INTERVAL] 秒ごとに行う',
37
+ 'ja-Hira': 'にんしきを [INTERVAL] びょうごとにおこなう',
38
+ 'en': 'Label once every [INTERVAL] seconds'
39
+ },
40
+ on: {
41
+ 'ja': '入',
42
+ 'ja-Hira': 'いり',
43
+ 'en': 'on'
44
+ },
45
+ off: {
46
+ 'ja': '切',
47
+ 'ja-Hira': 'きり',
48
+ 'en': 'off'
49
+ },
50
+ video_on_flipped: {
51
+ 'ja': '左右反転',
52
+ 'ja-Hira': 'さゆうはんてん',
53
+ 'en': 'on flipped',
54
+ },
55
+ please_wait: {
56
+ 'ja': '準備に時間がかかります。少しの間、操作ができなくなりますがお待ち下さい。',
57
+ 'ja-Hira': 'じゅんびにじかんがかかります。すこしのあいだ、そうさができなくなりますがおまちください。',
58
+ 'en': 'Setup takes a while. The browser will get stuck, but please wait.'
59
+ }
60
+ }
61
+ const AvailableLocales = ['en', 'ja', 'ja-Hira'];
62
+
63
+ class Scratch3Facemesh2ScratchBlocks {
64
+ get PERSON_NUMBER_MENU () {
65
+ let person_number_menu = []
66
+ for (let i = 1; i <= 10; i++) {
67
+ person_number_menu.push({text: String(i), value: String(i)})
68
+ }
69
+ return person_number_menu;
70
+ }
71
+
72
+ get KEYPOINT_MENU () {
73
+ let keypoint_menu = [];
74
+ for (let i = 1; i <= 468; i++) {
75
+ keypoint_menu.push({text: String(i), value: String(i)})
76
+ }
77
+ return keypoint_menu;
78
+ }
79
+
80
+ get VIDEO_MENU () {
81
+ return [
82
+ {
83
+ text: Message.off[this._locale],
84
+ value: 'off'
85
+ },
86
+ {
87
+ text: Message.on[this._locale],
88
+ value: 'on'
89
+ },
90
+ {
91
+ text: Message.video_on_flipped[this._locale],
92
+ value: 'on-flipped'
93
+ }
94
+ ]
95
+ }
96
+
97
+ get INTERVAL_MENU () {
98
+ return [
99
+ {
100
+ text: '0.1',
101
+ value: '0.1'
102
+ },
103
+ {
104
+ text: '0.2',
105
+ value: '0.2'
106
+ },
107
+ {
108
+ text: '0.5',
109
+ value: '0.5'
110
+ },
111
+ {
112
+ text: '1.0',
113
+ value: '1.0'
114
+ }
115
+ ]
116
+ }
117
+
118
+ get RATIO_MENU () {
119
+ return [
120
+ {
121
+ text: '0.5',
122
+ value: '0.5'
123
+ },
124
+ {
125
+ text: '0.75',
126
+ value: '0.75'
127
+ },
128
+ {
129
+ text: '1',
130
+ value: '1'
131
+ },
132
+ {
133
+ text: '1.5',
134
+ value: '1.5'
135
+ },
136
+ {
137
+ text: '2.0',
138
+ value: '2.0'
139
+ }
140
+ ]
141
+ }
142
+
143
+ constructor (runtime) {
144
+ this.runtime = runtime;
145
+
146
+ this.faces = [];
147
+ this.ratio = 0.75;
148
+
149
+ this.detectFace = () => {
150
+ // We should reuse the video element created by videoProvider instead of creating a new video element
151
+ // This is because iOS or iPad does not allow camera attached to two video elements
152
+ this.video = this.runtime.ioDevices.video.provider.video
153
+
154
+ alert(Message.please_wait[this._locale]);
155
+
156
+
157
+ this.facemesh = ml5.facemesh(this.video, function() {
158
+ console.log("Model loaded!")
159
+ });
160
+
161
+ this.facemesh.on('predict', faces => {
162
+ if (faces.length < this.faces.length) {
163
+ this.faces.splice(faces.length);
164
+ }
165
+ faces.forEach((face, index) => {
166
+ this.faces[index] = {keypoints: face.scaledMesh};
167
+ });
168
+ });
169
+ }
170
+
171
+ this.runtime.ioDevices.video.enableVideo().then(this.detectFace)
172
+ }
173
+
174
+ getInfo () {
175
+ this._locale = this.setLocale();
176
+
177
+ return {
178
+ id: 'facemesh2scratch',
179
+ name: 'Facemesh2Scratch',
180
+ blockIconURI: blockIconURI,
181
+ blocks: [
182
+ {
183
+ opcode: 'getX',
184
+ blockType: BlockType.REPORTER,
185
+ text: Message.getX[this._locale],
186
+ arguments: {
187
+ PERSON_NUMBER: {
188
+ type: ArgumentType.STRING,
189
+ menu: 'personNumberMenu',
190
+ defaultValue: '1'
191
+ },
192
+ KEYPOINT: {
193
+ type: ArgumentType.STRING,
194
+ menu: 'keypointMenu',
195
+ defaultValue: '1'
196
+ }
197
+ }
198
+ },
199
+ {
200
+ opcode: 'getY',
201
+ blockType: BlockType.REPORTER,
202
+ text: Message.getY[this._locale],
203
+ arguments: {
204
+ PERSON_NUMBER: {
205
+ type: ArgumentType.STRING,
206
+ menu: 'personNumberMenu',
207
+ defaultValue: '1'
208
+ },
209
+ KEYPOINT: {
210
+ type: ArgumentType.STRING,
211
+ menu: 'keypointMenu',
212
+ defaultValue: '1'
213
+ }
214
+ }
215
+ },
216
+ {
217
+ opcode: 'getPeopleCount',
218
+ blockType: BlockType.REPORTER,
219
+ text: Message.peopleCount[this._locale]
220
+ },
221
+ {
222
+ opcode: 'videoToggle',
223
+ blockType: BlockType.COMMAND,
224
+ text: Message.videoToggle[this._locale],
225
+ arguments: {
226
+ VIDEO_STATE: {
227
+ type: ArgumentType.STRING,
228
+ menu: 'videoMenu',
229
+ defaultValue: 'off'
230
+ }
231
+ }
232
+ },
233
+ {
234
+ opcode: 'setVideoTransparency',
235
+ text: formatMessage({
236
+ id: 'videoSensing.setVideoTransparency',
237
+ default: 'set video transparency to [TRANSPARENCY]',
238
+ description: 'Controls transparency of the video preview layer'
239
+ }),
240
+ arguments: {
241
+ TRANSPARENCY: {
242
+ type: ArgumentType.NUMBER,
243
+ defaultValue: 50
244
+ }
245
+ }
246
+ },
247
+ {
248
+ opcode: 'setRatio',
249
+ blockType: BlockType.COMMAND,
250
+ text: Message.setRatio[this._locale],
251
+ arguments: {
252
+ RATIO: {
253
+ type: ArgumentType.STRING,
254
+ menu: 'ratioMenu',
255
+ defaultValue: '0.75'
256
+ }
257
+ }
258
+ }
259
+ ],
260
+ menus: {
261
+ personNumberMenu: {
262
+ acceptReporters: true,
263
+ items: this.PERSON_NUMBER_MENU
264
+ },
265
+ keypointMenu: {
266
+ acceptReporters: true,
267
+ items: this.KEYPOINT_MENU
268
+ },
269
+ videoMenu: {
270
+ acceptReporters: true,
271
+ items: this.VIDEO_MENU
272
+ },
273
+ ratioMenu: {
274
+ acceptReporters: true,
275
+ items: this.RATIO_MENU
276
+ },
277
+ intervalMenu: {
278
+ acceptReporters: true,
279
+ items: this.INTERVAL_MENU
280
+ }
281
+ }
282
+ };
283
+ }
284
+
285
+ getX (args) {
286
+ let personNumber = parseInt(args.PERSON_NUMBER, 10) - 1;
287
+ let keypoint = parseInt(args.KEYPOINT, 10) - 1;
288
+
289
+ if (this.faces[personNumber].keypoints && this.faces[personNumber].keypoints[keypoint]) {
290
+ if (this.runtime.ioDevices.video.mirror === false) {
291
+ return -1 * (240 - this.faces[personNumber].keypoints[keypoint][0] * this.ratio);
292
+ } else {
293
+ return 240 - this.faces[personNumber].keypoints[keypoint][0] * this.ratio;
294
+ }
295
+ } else {
296
+ return "";
297
+ }
298
+ }
299
+
300
+ getY (args) {
301
+ let personNumber = parseInt(args.PERSON_NUMBER, 10) - 1;
302
+ let keypoint = parseInt(args.KEYPOINT, 10) - 1;
303
+
304
+ if (this.faces[personNumber].keypoints && this.faces[personNumber].keypoints[keypoint]) {
305
+ return 180 - this.faces[personNumber].keypoints[keypoint][1] * this.ratio;
306
+ } else {
307
+ return "";
308
+ }
309
+ }
310
+
311
+ getPeopleCount () {
312
+ return this.faces.length;
313
+ }
314
+
315
+ videoToggle (args) {
316
+ let state = args.VIDEO_STATE;
317
+ if (state === 'off') {
318
+ this.runtime.ioDevices.video.disableVideo();
319
+ this.facemesh.video = null; // Stop the model prediction if video is off
320
+ } else {
321
+ this.facemesh.removeAllListeners('predict');
322
+ this.runtime.ioDevices.video.enableVideo().then(this.detectFace);
323
+ this.runtime.ioDevices.video.mirror = state === "on";
324
+ }
325
+ }
326
+
327
+ /**
328
+ * A scratch command block handle that configures the video preview's
329
+ * transparency from passed arguments.
330
+ * @param {object} args - the block arguments
331
+ * @param {number} args.TRANSPARENCY - the transparency to set the video
332
+ * preview to
333
+ */
334
+ setVideoTransparency (args) {
335
+ const transparency = Cast.toNumber(args.TRANSPARENCY);
336
+ this.globalVideoTransparency = transparency;
337
+ this.runtime.ioDevices.video.setPreviewGhost(transparency);
338
+ }
339
+
340
+ setRatio (args) {
341
+ this.ratio = parseFloat(args.RATIO);
342
+ }
343
+
344
+ setLocale() {
345
+ let locale = formatMessage.setup().locale;
346
+ if (AvailableLocales.includes(locale)) {
347
+ return locale;
348
+ } else {
349
+ return 'en';
350
+ }
351
+ }
352
+ }
353
+
354
+ module.exports = Scratch3Facemesh2ScratchBlocks;