├── README.md ├── arduino_server.py ├── extension_arduino_nano.py └── index.js /README.md: -------------------------------------------------------------------------------- 1 | 此插件使用[pymata-io](https://github.com/bilikyar/pymata-aio)库作为Arduino_nano的固件。 2 | 3 | 如果有什么问题,尽管开issue,欢迎来交流。 4 | 5 | ref: [创建你的第一个Scratch3.0 Extension](https://blog.just4fun.site/create-first-Scratch3-Extension.html) 6 | -------------------------------------------------------------------------------- /arduino_server.py: -------------------------------------------------------------------------------- 1 | ''' 2 | arduino nano 3 | requirement: 4 | pip3 install pymata-aio --user 5 | ''' 6 | import zmq 7 | from time import sleep 8 | 9 | from pymata_aio.pymata3 import PyMata3 10 | from pymata_aio.constants import Constants 11 | 12 | # zmq socket 13 | port = 38782 14 | context = zmq.Context() 15 | socket = context.socket(zmq.REP) 16 | socket.bind("tcp://*:%s" % port) 17 | 18 | 19 | def main(): 20 | ConnectedToArduino = False 21 | while True: 22 | if not ConnectedToArduino: 23 | try: 24 | board = PyMata3() 25 | except: 26 | pass 27 | else: 28 | ConnectedToArduino = True 29 | board.set_pin_mode(13, Constants.OUTPUT) 30 | board.digital_write(13, 1) 31 | 32 | arduino_code = socket.recv_json().get("arduino_code") 33 | 34 | if not arduino_code: 35 | socket.send_json({"result": { 36 | "pin_2_state": board.get_pin_state(2), 37 | "pin_3_state": board.get_pin_state(3), 38 | "pin_4_state": board.get_pin_state(4), 39 | "pin_5_state": board.get_pin_state(5), 40 | "pin_6_state": board.get_pin_state(6), 41 | "pin_7_state": board.get_pin_state(7), 42 | "pin_8_state": board.get_pin_state(8), 43 | "pin_9_state": board.get_pin_state(9), 44 | "pin_10_state": board.get_pin_state(10), 45 | "pin_11_state": board.get_pin_state(11), 46 | "pin_12_state": board.get_pin_state(12), 47 | "pin_13_state": board.get_pin_state(13), 48 | "digital_pin_2": board.digital_read(2), 49 | "digital_pin_3": board.digital_read(3), 50 | "digital_pin_4": board.digital_read(4), 51 | "digital_pin_5": board.digital_read(5), 52 | "digital_pin_6": board.digital_read(6), 53 | "digital_pin_7": board.digital_read(7), 54 | "digital_pin_8": board.digital_read(8), 55 | "digital_pin_9": board.digital_read(9), 56 | "digital_pin_10": board.digital_read(10), 57 | "digital_pin_11": board.digital_read(11), 58 | "digital_pin_12": board.digital_read(12), 59 | "digital_pin_13": board.digital_read(13), 60 | "analog_pin_0": board.analog_read(0), 61 | "analog_pin_1": board.analog_read(1), 62 | "analog_pin_2": board.analog_read(2), 63 | "analog_pin_3": board.analog_read(3), 64 | "analog_pin_4": board.analog_read(4), 65 | "analog_pin_5": board.analog_read(5), 66 | "analog_pin_6": board.analog_read(6), 67 | "analog_pin_7": board.analog_read(7), 68 | }}) 69 | sleep(0.05) 70 | continue 71 | 72 | if arduino_code == "quit!": 73 | output = eval("board.shutdown()", {}, { 74 | "board": board, "Constants": Constants}) 75 | socket.send_json({"result": "quit!"}) 76 | break 77 | else: 78 | try: 79 | output = eval(arduino_code, {}, { 80 | "board": board, "Constants": Constants}) 81 | # output = exec(arduino_code) # 安全性问题 82 | except Exception as e: 83 | output = e 84 | socket.send_json({ 85 | "result": { 86 | "output": str(output), 87 | "pin_2_state": board.get_pin_state(2), 88 | "pin_3_state": board.get_pin_state(3), 89 | "pin_4_state": board.get_pin_state(4), 90 | "pin_5_state": board.get_pin_state(5), 91 | "pin_6_state": board.get_pin_state(6), 92 | "pin_7_state": board.get_pin_state(7), 93 | "pin_8_state": board.get_pin_state(8), 94 | "pin_9_state": board.get_pin_state(9), 95 | "pin_10_state": board.get_pin_state(10), 96 | "pin_11_state": board.get_pin_state(11), 97 | "pin_12_state": board.get_pin_state(12), 98 | "pin_13_state": board.get_pin_state(13), 99 | "digital_pin_2": board.digital_read(2), 100 | "digital_pin_3": board.digital_read(3), 101 | "digital_pin_4": board.digital_read(4), 102 | "digital_pin_5": board.digital_read(5), 103 | "digital_pin_6": board.digital_read(6), 104 | "digital_pin_7": board.digital_read(7), 105 | "digital_pin_8": board.digital_read(8), 106 | "digital_pin_9": board.digital_read(9), 107 | "digital_pin_10": board.digital_read(10), 108 | "digital_pin_11": board.digital_read(11), 109 | "digital_pin_12": board.digital_read(12), 110 | "digital_pin_13": board.digital_read(13), 111 | "analog_pin_0": board.analog_read(0), 112 | "analog_pin_1": board.analog_read(1), 113 | "analog_pin_2": board.analog_read(2), 114 | "analog_pin_3": board.analog_read(3), 115 | "analog_pin_4": board.analog_read(4), 116 | "analog_pin_5": board.analog_read(5), 117 | "analog_pin_6": board.analog_read(6), 118 | "analog_pin_7": board.analog_read(7), 119 | }}) 120 | sleep(0.05) 121 | 122 | socket.close() 123 | context.term() 124 | 125 | 126 | if __name__ == '__main__': 127 | main() 128 | -------------------------------------------------------------------------------- /extension_arduino_nano.py: -------------------------------------------------------------------------------- 1 | 2 | ''' 3 | Arduino 4 | requirement: 5 | pip3 install pymata-aio --user 6 | ''' 7 | import zmq 8 | import subprocess 9 | import pathlib 10 | import platform 11 | import time 12 | import threading 13 | 14 | from codelab_adapter import settings 15 | from codelab_adapter.core_extension import Extension 16 | 17 | 18 | def get_python3_path(): 19 | # If it is not working, Please replace python3_path with your local python3 path. shell: which python3 20 | if (platform.system() == "Darwin"): 21 | # which python3 22 | # 不如用PATH python 23 | # 不确定 24 | path = "/usr/local/bin/python3" 25 | if platform.system() == "Windows": 26 | path = "python3" 27 | if platform.system() == "Linux": 28 | path = "/usr/bin/python3" 29 | return path 30 | 31 | 32 | python3_path = get_python3_path() 33 | 34 | 35 | class arduinoExtension(Extension): 36 | def __init__(self): 37 | name = type(self).__name__ # class name 38 | super().__init__(name) 39 | self.scratch3_message = {} 40 | self.TOPIC = "eim/arduino" 41 | self.first_start = 1 42 | 43 | 44 | def run(self): 45 | # 抽象掉这部分 Class 46 | port = 38782 # todo 随机分配 47 | context = zmq.Context.instance() 48 | socket = context.socket(zmq.REQ) 49 | socket.connect("tcp://localhost:%s" % port) 50 | 51 | codelab_adapter_server_dir = pathlib.Path.home( 52 | ) / "codelab_adapter" / "servers" 53 | script = "{}/arduino_server.py".format(codelab_adapter_server_dir) 54 | 55 | cmd = [python3_path, script] 56 | arduino_server = subprocess.Popen(cmd) 57 | settings.running_child_procs.append(arduino_server) 58 | 59 | lock = threading.Lock() 60 | 61 | def request(): 62 | while self._running: 63 | lock.acquire() 64 | self.scratch3_message = self.read() 65 | lock.release() 66 | 67 | bg_task = threading.Thread(target=request) 68 | self.logger.debug("thread start") 69 | bg_task.daemon = True 70 | bg_task.start() 71 | 72 | while self._running: 73 | scratch3_message = self.scratch3_message 74 | self.logger.debug("scratch3_message {}".format(scratch3_message)) 75 | self.scratch3_message = {} 76 | if scratch3_message == {}: 77 | scratch3_message = {"topic": self.TOPIC, "payload": ""} 78 | 79 | topic = scratch3_message.get('topic') 80 | arduino_code = scratch3_message.get("payload") 81 | 82 | if topic == self.TOPIC: 83 | socket.send_json({"arduino_code": arduino_code}) 84 | 85 | result = socket.recv_json().get("result") 86 | 87 | if self.first_start == 1: 88 | self.publish({"topic": "eim/arduino/init","payload": ""}) 89 | self.first_start = 0 90 | 91 | # 发往scratch3.0 92 | self.publish({"topic": self.TOPIC,"payload": result}) 93 | time.sleep(0.05) 94 | 95 | 96 | 97 | 98 | # release socket 99 | socket.send_json({"arduino_code": "quit!"}) 100 | result = socket.recv_json().get("result") 101 | arduino_server.terminate() 102 | arduino_server.wait() 103 | socket.close() 104 | context.term() 105 | 106 | 107 | export = arduinoExtension 108 | -------------------------------------------------------------------------------- /index.js: -------------------------------------------------------------------------------- 1 | const ArgumentType = require("../../extension-support/argument-type"); 2 | const BlockType = require("../../extension-support/block-type"); 3 | const formatMessage = require("format-message"); 4 | const io = require("socket.io-client"); // yarn add socket.io-client socket.io-client@2.2.0 5 | const Cast = require('../../util/cast'); 6 | 7 | /** 8 | * Icon svg to be displayed at the left edge of each extension block, encoded as a data URI. 9 | * @type {string} 10 | */ 11 | // eslint-disable-next-line max-len 12 | const blockIconURI = 13 | ""; 14 | const menuIconURI = blockIconURI; 15 | 16 | /** 17 | * Enum for icon parameter values. 18 | * @readonly 19 | * @enum {string} 20 | */ 21 | 22 | var board = { 23 | digital_pin_2: "None", 24 | digital_pin_3: "None", 25 | digital_pin_4: "None", 26 | digital_pin_5: "None", 27 | digital_pin_6: "None", 28 | digital_pin_7: "None", 29 | digital_pin_8: "None", 30 | digital_pin_9: "None", 31 | digital_pin_10: "None", 32 | digital_pin_11: "None", 33 | digital_pin_12: "None", 34 | digital_pin_13: "None", 35 | analog_pin_0: "None", 36 | analog_pin_1: "None", 37 | analog_pin_2: "None", 38 | analog_pin_3: "None", 39 | analog_pin_4: "None", 40 | analog_pin_5: "None", 41 | analog_pin_6: "None", 42 | analog_pin_7: "None" 43 | }; 44 | 45 | const USBSendInterval = 100; 46 | 47 | 48 | class arduinoBlocks { 49 | constructor(runtime) { 50 | /** 51 | * The runtime instantiating this block package. 52 | * @type {Runtime} 53 | */ 54 | this.runtime = runtime; 55 | 56 | const url = new URL(window.location.href); 57 | var adapterHost = url.searchParams.get("adapter_host"); // 支持树莓派(分布式使用) 58 | if (!adapterHost) { 59 | var adapterHost = "codelab-adapter.codelab.club"; 60 | } 61 | 62 | this.socket = io(`//${adapterHost}:12358` + "/test", { 63 | transports: ["websocket"] 64 | }); 65 | this.socket.on("sensor", msg => { 66 | this.message = msg.message; 67 | const topic = this.message.topic; 68 | const message = this.message.payload; 69 | const first_start = this.message.first_sart 70 | this.message = message; // 可能被清空 71 | this.topic = topic; 72 | this.origin_message = message; 73 | if (this.topic == "eim/arduino/init") { 74 | console.log("extention start"); 75 | board = { 76 | digital_pin_2: "None", 77 | digital_pin_3: "None", 78 | digital_pin_4: "None", 79 | digital_pin_5: "None", 80 | digital_pin_6: "None", 81 | digital_pin_7: "None", 82 | digital_pin_8: "None", 83 | digital_pin_9: "None", 84 | digital_pin_10: "None", 85 | digital_pin_11: "None", 86 | digital_pin_12: "None", 87 | digital_pin_13: "None", 88 | analog_pin_0: "None", 89 | analog_pin_1: "None", 90 | analog_pin_2: "None", 91 | analog_pin_3: "None", 92 | analog_pin_4: "None", 93 | analog_pin_5: "None", 94 | analog_pin_6: "None", 95 | analog_pin_7: "None" 96 | }; 97 | } 98 | }); 99 | } 100 | /** 101 | * The key to load & store a target's test-related state. 102 | * @type {string} 103 | */ 104 | static get STATE_KEY() { 105 | return "Scratch.arduino"; 106 | } 107 | 108 | /** 109 | * @returns {object} metadata for this extension and its blocks. 110 | */ 111 | getInfo() { 112 | return { 113 | id: "arduino", 114 | name: "arduino", 115 | menuIconURI: menuIconURI, 116 | blockIconURI: blockIconURI, 117 | blocks: [ 118 | { 119 | opcode: "read_analog_value", 120 | blockType: BlockType.REPORTER, // BOOLEAN, COMMAND 121 | text: formatMessage({ 122 | id: "arduino.read_analog_value", 123 | default: "read analog value from [analogPinNumber] ", 124 | description: "read_analog_value" 125 | }), 126 | arguments: { 127 | analogPinNumber: { 128 | type: ArgumentType.STRING, 129 | menu: "analogPinNumber", 130 | defaultValue: "0" 131 | } 132 | } 133 | }, 134 | { 135 | opcode: "changeLedState", 136 | blockType: BlockType.COMMAND, 137 | text: formatMessage({ 138 | id: "arduino.changeLedState", 139 | default: "led [digitalPinNumber] set [logicState]", 140 | description: "changeLedState" 141 | }), 142 | arguments: { 143 | digitalPinNumber: { 144 | type: ArgumentType.STRING, 145 | menu: "digitalPinNumber", 146 | defaultValue: "2" 147 | }, 148 | logicState: { 149 | type: ArgumentType.STRING, 150 | menu: "logicState", 151 | defaultValue: "1" 152 | } 153 | } 154 | }, 155 | { 156 | opcode: "changePwmLedValue", 157 | blockType: BlockType.COMMAND, 158 | text: formatMessage({ 159 | id: "arduino.changePwmLedValue", 160 | default: "Pwm_led [PwmPinNumber] set [pwmValue]", 161 | description: "changeLedState" 162 | }), 163 | arguments: { 164 | PwmPinNumber: { 165 | type: ArgumentType.STRING, 166 | menu: "PwmPinNumber", 167 | defaultValue: "9" 168 | }, 169 | pwmValue: { 170 | type: ArgumentType.STRING, 171 | defaultValue: "50" 172 | } 173 | } 174 | }, 175 | { 176 | opcode: "read_button_state", 177 | blockType: BlockType.REPORTER, // BOOLEAN, COMMAND 178 | text: formatMessage({ 179 | id: "arduino.read_button_state", 180 | default: "read button [digitalPinNumber] state", 181 | description: "read_button_state" 182 | }), 183 | arguments: { 184 | digitalPinNumber: { 185 | type: ArgumentType.STRING, 186 | menu: "digitalPinNumber", 187 | defaultValue: "12" 188 | } 189 | } 190 | }, 191 | { 192 | opcode: "changeServoDegree", 193 | blockType: BlockType.COMMAND, 194 | text: formatMessage({ 195 | id: "arduino.changeServoDegree", 196 | default: "servo [PwmPinNumber] set [degree]", 197 | description: "changeServoDegree" 198 | }), 199 | arguments: { 200 | PwmPinNumber: { 201 | type: ArgumentType.STRING, 202 | menu: "PwmPinNumber", 203 | defaultValue: "9" 204 | }, 205 | degree: { 206 | type: ArgumentType.STRING, 207 | defaultValue: "0" 208 | } 209 | } 210 | } 211 | ], 212 | menus: { 213 | digitalPinNumber: [ 214 | "2", 215 | "3", 216 | "4", 217 | "5", 218 | "6", 219 | "7", 220 | "8", 221 | "9", 222 | "10", 223 | "11", 224 | "12", 225 | "13" 226 | ], 227 | analogPinNumber: ["0", "1", "2", "3", "4", "5", "6", "7"], 228 | PwmPinNumber: ["3", "5", "6", "9", "10", "11"], 229 | logicState: ["1", "0"] 230 | } 231 | }; 232 | } 233 | 234 | /** 235 | * Retrieve the block primitives implemented by this package. 236 | * @return {object.} Mapping of opcode to Function. 237 | */ 238 | 239 | read_analog_value(args) { 240 | const topic = "eim/arduino"; 241 | let message = ""; 242 | if (board["analog_pin_" + args.analogPinNumber] == "ANALOG") { 243 | if (this.topic == topic) { 244 | return this.origin_message[ 245 | "analog_pin_" + args.analogPinNumber 246 | ]; 247 | } 248 | } else { 249 | message = 250 | "board.set_pin_mode(" + 251 | args.analogPinNumber + 252 | ", Constants.ANALOG)"; 253 | board["analog_pin_" + args.analogPinNumber] = "ANALOG"; 254 | console.log(message); 255 | this.socket.emit("actuator", { 256 | topic: topic, 257 | payload: message 258 | }); 259 | if (this.topic == topic) { 260 | return this.origin_message[ 261 | "analog_pin_" + args.analogPinNumber 262 | ]; 263 | } 264 | } 265 | } 266 | 267 | changeLedState(args, util) { 268 | const topic = "eim/arduino"; 269 | let message = ""; 270 | if (board["digital_pin_" + args.digitalPinNumber] == "OUTPUT") { 271 | message = 272 | "board.digital_write(" + 273 | args.digitalPinNumber + 274 | "," + 275 | args.logicState + 276 | ")"; 277 | console.log(message); 278 | } else { 279 | message = 280 | "board.set_pin_mode(" + 281 | args.digitalPinNumber + 282 | ", Constants.OUTPUT)"; 283 | board["digital_pin_" + args.digitalPinNumber] = "OUTPUT"; 284 | console.log(message); 285 | this.socket.emit("actuator", { 286 | topic: topic, 287 | payload: message 288 | }); 289 | message = 290 | "board.digital_write(" + 291 | args.digitalPinNumber + 292 | "," + 293 | args.logicState + 294 | ")"; 295 | console.log(message); 296 | } 297 | 298 | this.socket.emit("actuator", { 299 | topic: topic, 300 | payload: message 301 | }); 302 | 303 | 304 | return new Promise(resolve => { 305 | setTimeout(() => { 306 | resolve(); 307 | }, USBSendInterval); 308 | }); 309 | } 310 | 311 | read_button_state(args) { 312 | const topic = "eim/arduino"; 313 | let message = ""; 314 | if (board["digital_pin_" + args.digitalPinNumber] == "INPUT") { 315 | if (this.topic == topic) { 316 | return this.origin_message[ 317 | "digital_pin_" + args.digitalPinNumber 318 | ]; 319 | } 320 | } else { 321 | const message = 322 | "board.set_pin_mode(" + 323 | args.digitalPinNumber + 324 | ", Constants.INPUT)"; 325 | board["digital_pin_" + args.digitalPinNumber] = "INPUT"; 326 | console.log(message); 327 | this.socket.emit("actuator", { 328 | topic: topic, 329 | payload: message 330 | }); 331 | if (this.topic == topic) { 332 | return this.origin_message[ 333 | "digital_pin_" + args.digitalPinNumber 334 | ]; 335 | } 336 | } 337 | } 338 | 339 | changeServoDegree(args) { 340 | const topic = "eim/arduino"; 341 | let message = ""; 342 | if (board["digital_pin_" + args.digitalPinNumber] == "SERVO") { 343 | message = 344 | "board.analog_write(" + 345 | args.PwmPinNumber + 346 | "," + 347 | args.degree + 348 | ")"; 349 | console.log(message); 350 | } else { 351 | message = "board.servo_config(" + args.PwmPinNumber + ")"; 352 | 353 | board["digital_pin_" + args.digitalPinNumber] = "SERVO"; 354 | console.log(message); 355 | this.socket.emit("actuator", { 356 | topic: topic, 357 | payload: message 358 | }); 359 | message = 360 | "board.analog_write(" + 361 | args.PwmPinNumber + 362 | "," + 363 | args.degree + 364 | ")"; 365 | console.log(message); 366 | } 367 | 368 | this.socket.emit("actuator", { 369 | topic: topic, 370 | payload: message 371 | }); 372 | 373 | return new Promise(resolve => { 374 | setTimeout(() => { 375 | resolve(); 376 | }, USBSendInterval); 377 | }); 378 | } 379 | 380 | changePwmLedValue(args) { 381 | const topic = "eim/arduino"; 382 | let message = ""; 383 | if (board["digital_pin_" + args.PwmPinNumber] == "PWM") { 384 | message = 385 | "board.analog_write(" + 386 | args.PwmPinNumber + 387 | "," + 388 | args.pwmValue + 389 | ")"; 390 | console.log(message); 391 | } else { 392 | message = 393 | "board.set_pin_mode(" + args.PwmPinNumber + ", Constants.PWM)"; 394 | board["digital_pin_" + args.PwmPinNumber] = "PWM"; 395 | console.log(message); 396 | this.socket.emit("actuator", { 397 | topic: topic, 398 | payload: message 399 | }); 400 | message = 401 | "board.analog_write(" + 402 | args.PwmPinNumber + 403 | "," + 404 | args.pwmValue + 405 | ")"; 406 | console.log(message); 407 | } 408 | 409 | this.socket.emit("actuator", { 410 | topic: topic, 411 | payload: message 412 | }); 413 | return new Promise(resolve => { 414 | setTimeout(() => { 415 | resolve(); 416 | }, USBSendInterval); 417 | }); 418 | } 419 | } 420 | 421 | module.exports = arduinoBlocks; 422 | --------------------------------------------------------------------------------