Vue实现图形化积木式编程(十二)

article/2025/10/7 12:48:19

执行Blockly生成代码

  • 路由
    • 下一篇
    • 历史回顾
      • Babylon.js部分
      • Blockly部分
  • 前言
  • 最终实现效果
  • 本文内容
    • 实现思路
    • 问题分析
      • 问题
      • 原因
      • 不优雅解决
      • 优雅解决
    • 完整代码
  • 后续计划
  • 开源项目GitHub链接
  • 资源下载链接
  • 你的点赞是我继续编写的动力

路由

下一篇

  • Vue实现图形化积木式编程(十三) ---- 步骤运行代码块高亮

历史回顾

Babylon.js部分

  • Vue实现图形化积木式编程(一) ---- Babylon.js基础场景搭建
  • Vue实现图形化积木式编程(二) ---- Babylon.js加载模型到场景中
  • Vue实现图形化积木式编程(三) ---- Babylon.js点击拖拽移动模型
  • Vue实现图形化积木式编程(四) ---- Babylon.js实现碰撞效果
  • Vue实现图形化积木式编程(五) ---- Babylon.js自定义启动界面
  • Vue实现图形化积木式编程(六) ---- Babylon.js相机控制与相机动画
  • Vue实现图形化积木式编程(七) ---- babylonjs-gui 按钮实现
  • Vue实现图形化积木式编程(八) ---- 将3d界面放入可拖动窗口中

Blockly部分

  • Vue实现图形化积木式编程(九) ---- Blockly代码块编辑区域基本场景搭建
  • Vue实现图形化积木式编程(十) ---- Blockly自定义块
  • Vue实现图形化积木式编程(十一) ---- Blockly插件使用

前言

TIPS:该案例设计主要参考iRobot Coding,只用做学习用途,侵删。

https://code.irobot.com/#/

最终实现效果

最终实现效果

本文内容

  • 在 Blockly代码块编辑区域基本场景搭建 这篇文章里笔者提到过在之后的文章中会对使用eval方法执行代码进行优化,对于通过Blockly代码块生成的代码字符串,本文介绍如何将代码字符串转化为执行语句来调用控制模型运动的方法。

本文主要内容是执行Blockly生成的代码字符串,会忽略通过babylonjs来控制模型对象运动的细节。实际上控制模型运动并不难,主要解决的是顺序执行当前生成的代码指令时,如move、arc等指令,需要设定在一定时间内模型的x、y、rotation等参数如何变化。

实现思路

如果想要将字符串转为可执行代码,可以使用eval()new Function([arg1,arg2,...], 代码字符串)

  • 现在存在一个robot的实例对象,包含一些控制运动的方法
//  robot.js
class Robot {constructor() {this.isRun = false}init() {console.log("robot模块化程序初始化")this.isRun = true}stop() {console.log("robot模块运行结束")this.isRun = false}checkStatus() {if (!this.isRun) {throw '程序需要初始化模块'}}async move(distance) {this.checkStatus()// 模拟小车运动return new Promise(resolve => {let moveDis = 0let interval = setInterval(() => {if (moveDis < distance) {console.log(`move ${moveDis++}`)} else {clearInterval(interval)interval = undefinedresolve()}}, 100)})}async arc(direction, degree, distance) {this.checkStatus()// 模拟小车运动return new Promise(resolve => {let moveDis = 0let interval = setInterval(() => {if (moveDis < distance) {console.log(`direction: ${direction}, move:${moveDis++}, degree: ${degree}`)} else {clearInterval(interval)interval = undefinedresolve()}}, 100)})}
}
export default Robot	
	import Robot from 'robot.js'const robot = new Robot()
  • 然后使用 Blockly自定义块 自定义了move、arc等指令的代码块,并通过这些代码块拼接并生成了如下代码字符串:
    Block生成代码
// 现在需要执行这一代码字符串
let code = `robot.init();robot.move(50);robot.arc(0, 90,50);robot.stop();
`
  1. 使用eval
  • 使用eval执行,需要将robot对象挂载到window对象上
	window.robot = roboteval(code)
  1. 使用 new Function ([arg1, arg2, ...], '代码字符串')
  • 使用new Function,最后一个参数是执行代码字符串,前面的参数都是提供给代码字符串的变量
	/** 这里相当于定义了一个函数function(robot) {//执行的code内容}*/let fn = new Function('robot', code)// 执行函数, 将robot作为变量传入fn(robot)

问题分析

问题

  • 执行函数后会发现,如果movearc都是一个异步函数,程序执行顺序有问题,执行顺序为init->stop->move和arc交替执行

原因

  • 上面生成的代码字符串是顺序执行的,然后move和arc是异步任务,会放进入宏任务列表,宏任务会在主线程的同步任务执行完之后再执行。

不优雅解决

  • 在有异步操作的代码块生成的代码中加入await关键字,异步代码需要在有async标记的函数中执行,所以需要在执行的代码块中包裹一个立即执行的异步匿名函数
Blockly.JavaScript['while_program_start'] = function () {var while_content = Blockly.JavaScript.statementToCode(block, 'while_content');const code = `
(async ()=>{
robot.init();\n${while_content}robot.stop();
})()
`return code;
};
Blockly.JavaScript['move'] = function (block) {var text_move_distance = block.getFieldValue('move_distance');var code = `await robot.move(${text_move_distance});\n`;return code;
};
Blockly.JavaScript['arc'] = function (block) {var dropdown_dirction = block.getFieldValue('dirction');var angle_degree = block.getFieldValue('degree');var radius = block.getFieldValue('radius');var code = `await robot.arc(${dropdown_dirction}, ${angle_degree}, ${radius});\n`;return code;
};
// 生成的代码字符串变成如下形式
(async ()=>{
robot.init();await robot.move(50);await robot.arc(0, 90, 50);
robot.stop();
})()

优雅解决

  • 优雅的方案是使用 JS-Interpreter - JavaScript语法解析器来将字符串代码可执行的代码。在解析器在创建时可自定义属性、自定函数和其对应执行的同步或异步方法,同时,可以通过step方法来步骤执行代码,这就可以配合Blockly的highlightBlock(blockId)方法来实现步骤执行代码块高亮的效果。
  • 接入js-interpreter,步骤运行block块将会再下一篇文章中详细描述哦

完整代码

  • 测试用例
<template><div id="blockly"><!-- 工作区 --><div id="blocklyDiv" ref="blocklyDiv" style="height: 500px; width: 800px;"></div><button style="position: fixed;left: 50px;top: 10px;" @click="block2code">生成代码</button><!-- 代码显示区 --><div style="background-color: lightgrey;width: 800px;text-align: left"><pre v-html="code?code:'请点击生成代码按钮'"></pre></div><button style="position: fixed;left: 150px;top: 10px;" @click="runCode">eval执行代码</button><button style="position: fixed;left: 300px;top: 10px;" @click="runCode2">new Function执行代码</button></div>
</template><script>
import Blockly from 'blockly'
import BlocklyJS from 'blockly/javascript';
import './customBlock'
import Robot from './robot'
export default {name: "blocklyClass3",data() {return {code:'',options: {horizontalLayout: true,//工具箱水平toolboxPosition: "end",//工具箱在底部toolbox: {"kind": "flyoutToolbox","contents": [{"kind": "block","type": "while_program_start",},{"kind": "block","type": "move",},{"kind": "block","type": "arc"},{"kind": "block","type": "controls_repeat_ext"},{"kind": "block","type": "controls_whileUntil"},{"kind": "block","type": "controls_for"},{"kind": "block","type": "controls_if"},{"kind": "block","type": "logic_compare"},{"kind": "block","type": "logic_operation"},{"kind": "block","type": "logic_negate"},{"kind": "block","type": "logic_boolean"},{"kind": "sep","gap": "32"},{"kind": "block","blockxml": "<block type='math_number'><field name='NUM'>10</field></block>"},{"kind": "block","type": "math_arithmetic"},{"kind": "block","type": "math_single"},{"kind": "block","type": "text"},{"kind": "block","type": "text_length"},{"kind": "block","type": "text_print"},{"kind": "block","type": "variables_get"},{"kind": "block","type": "variables_set"},]}}}},mounted() {Blockly.inject(this.$refs.blocklyDiv, this.options);this.robot = new Robot()},methods:{/*** block代码块转为代码*/block2code(){this.code = BlocklyJS.workspaceToCode(this.$refs.blocklyDiv.workspace)},/*** 执行生成代码*/runCode(){if(!this.code){alert('请先点击生成代码');return}window.robot = this.roboteval(this.code)},runCode2(){if(!this.code){alert('请先点击生成代码');return}let fn = new Function('robot', this.code)fn(this.robot)}},
}
</script><style scoped>
#blockly {position: absolute;left: 50px;top: 50px;bottom: 0;width: calc(100vw - 50px);height: calc(100vh - 50px);display: flex;flex-direction: column;
}
</style>
  • Blockly自定义组件
import * as Blockly from 'blockly/core'import * as hans from 'blockly/msg/zh-hans'Blockly.setLocale(hans);//汉化/*** 自定义组件注册*/
Blockly.defineBlocksWithJsonArray([//事件{"type": "while_program_start","message0": "当程序运行 %1 %2","args0": [{"type": "input_dummy"},{"type": "input_statement","name": "while_content"}],"previousStatement": null,"nextStatement": null,"colour": "#609FD6","strokeColour": "#4088C8","tooltip": "123","helpUrl": "1"},//指令{"type": "move","message0": "移动 %1 CM","args0": [{"type": "field_input","name": "move_distance","text": "50"}],"previousStatement": null,"nextStatement": null,"colour": "#F7D233","strokeColour": "#CCAD2B","tooltip": "","helpUrl": ""},{"type": "arc","message0": "弧形 %1 %2 ,半径 %3 CM","args0": [{"type": "field_dropdown","name": "dirction","options": [["向左","0"],["向右","1"]]},{"type": "field_angle","name": "degree","angle": 90},{"type": "field_number","name": "radius","value": 50,"min": 1,"max": 100}],"previousStatement": null,"nextStatement": null,"colour": "#F7D233","strokeColour": "#CCAD2B","tooltip": "","helpUrl": ""}],"previousStatement": null,"nextStatement": null,"colour": "#81C679","tooltip": "","helpUrl": ""},]
);/*** 自定义组件生成代码* @param block* @returns {string}*/
Blockly.JavaScript['while_program_start'] = function (block) {var while_content = Blockly.JavaScript.statementToCode(block, 'while_content');const code = `
(async ()=>{
robot.init();\n${while_content}robot.stop();
})()
`return code;
};Blockly.JavaScript['move'] = function (block) {var text_move_distance = block.getFieldValue('move_distance');var code = `await robot.move(${text_move_distance});\n`;return code;
};Blockly.JavaScript['arc'] = function (block) {var dropdown_dirction = block.getFieldValue('dirction');var angle_degree = block.getFieldValue('degree');var radius = block.getFieldValue('radius');var code = `await robot.arc(${dropdown_dirction}, ${angle_degree}, ${radius});\n`;return code;
};
  • Robot控制类
// eslint-disable-next-line no-unused-vars
class Robot {constructor() {this.isRun = false}init() {console.log("robot模块化程序初始化")this.isRun = true}stop() {console.log("robot模块运行结束")this.isRun = false}checkStatus() {if (!this.isRun) {throw '程序需要初始化模块'}}async move(distance) {this.checkStatus()// 模拟小车运动return new Promise(resolve => {let moveDis = 0let interval = setInterval(() => {if (moveDis < distance) {console.log(`move ${moveDis++}`)} else {clearInterval(interval)interval = undefinedresolve()}}, 100)})}async arc(direction, degree, distance) {this.checkStatus()// 模拟小车运动return new Promise(resolve => {let moveDis = 0let interval = setInterval(() => {if (moveDis < distance) {console.log(`direction: ${direction}, move:${moveDis++}, degree: ${degree}`)} else {clearInterval(interval)interval = undefinedresolve()}}, 100)})}
}
export default Robot

后续计划

  • 接入js-interpreter,步骤运行block块

开源项目GitHub链接

https://github.com/Wenbile/Child-Programming-Web

资源下载链接

  • Vue前端源码
  • ThinkJS后端源码

你的点赞是我继续编写的动力


http://chatgpt.dhexx.cn/article/y4USwljf.shtml

相关文章

Vue实现图形化积木式编程(二)

Babylon.js加载模型到场景中 路由下一篇历史回顾 前言最终实现效果本文实现效果完整代码操作分解&#xff08;Babylon.js模型格式转换与导入&#xff09;0.在开源模型网上下载一个模型/自己制作一个1.转换为.babylon文件2.将模型文件放在服务器上方案一&#xff08;最新发现的方…

python积木式编程_TurnipBit—MicroPython开发板:从积木式编程语言开始学作小小创客...

编程、建模、制做动画和游戏……这些当初咱们默认只有成年人玩得转的事情,如今早已经被无数小孩子给颠覆甚至玩出新境界了。热爱科技和动手的“创客”(Maker)如今在全世界都煊赫一时。今年以来,对青少年的创客教育在中国不管是庙堂仍是大众而言亦是热门话题。编程 从TurnipBi…

c++手机编程软件_积木编程软件手机版下载-积木编程软件下载v1.0.1 安卓版

积木编程软件是可以在线可以学习编程的软件,平台给用户提供了大量的编程学习教程&#xff0c;软件还支持在线编程&#xff0c;学习内容丰富&#xff0c;让用户学习起来不会枯燥乏味&#xff0c;就算你是零基础的小白&#xff0c;也可以体验到编程的乐趣。 积木编程官方简介 积木…

Vue实现图形化积木式编程(十三)

步骤运行代码块高亮 路由历史回顾Babylon.js部分Blockly部分 前言最终实现效果本文内容实现1. 安装依赖2. 简化语法3. 引入js解析器4. 运行代码5. 加入高亮 完整代码本文章的用例代码已经同步到github上&#xff0c;运行程序后访问链接即可查看效果: [http://localhost:3000/#/…

搭积木php编程,Primo Toys,一款可以同时搭积木与编程的玩具

原标题&#xff1a;Primo Toys,一款可以同时搭积木与编程的玩具 时下社会&#xff0c;“编程”一词早已不是成人世界的一个职业&#xff0c;如今&#xff0c;编程能力就像蜗牛日常的读写一样&#xff0c;渗透到孩子生活的方方面面&#xff0c;不论是APP&#xff0c;游戏&#x…

python的积木式编程

edu.codemao.cn https://wood.codemao.cn/ https://static.codemao.cn/wood_docs/web/code/Control.html 编程猫和scratch大同小异&#xff0c;和其他Python平台相比较&#xff1a; 优势&#xff1a; 支持硬件编程丰富的Python库积木/代码一键转换搭积木 学编程使用方便&a…

c语言的积木编程,c语言入门第3节,掌握它就能随心所欲的编程了,自己造积木...

前面两节介绍了如何快速搭建 C 语言开发学习环境&#xff0c;并且给出并解释了 hello world 的代码。上一节介绍了C语言的函数&#xff0c;应该注意到&#xff0c;到目前为止&#xff0c;我们使用的函数都是现有的系统函数(例如 printf&#xff0c;sin 等函数)。事实上&#xf…

等边三角形的积木编程

用Scratch里的画笔&#xff0c;可以很方便的画出等边三角形。步骤如下&#xff1a; 1、定义"等边三角形"积木 点击Scratch里的"自制积木" --》制作新的积木 --》直线 --》添加4个参数&#xff0c;如图(1)、(2)所示&#xff1a; 图(1) 定义"等边三角…

积木编程软件

当然这种观点是很荒谬的&#xff0c;不会编程不等于文盲。虽然大多数父母都有文化和读写能力&#xff0c;但是大多数父母都不是程序员&#xff0c;也不知道程序员需要什么样的技能。格物斯坦认为&#xff1a;针对孩子们的编程书籍给出的都是一些问题的“标准”答案。如果你的孩…

金字塔的积木编程

金字塔是一个上尖下方的四面体&#xff0c;它的截面是一个三角形&#xff0c;可以用长方形的砖块去堆叠成一个大的三角形来表示金字塔。     下面介绍使用Scratch来画金字塔。 1、定义"砖块"积木 图(1) 定义"砖块"积木 图(2) "砖块"积木代码 …

OpenBlock Desktop v2.3.1积木编程

最新版 2.3.1 Release OpenBlock Desktop v2.3.1 openblockcc/openblock-desktop GitHubhttps://github.com/openblockcc/openblock-desktop/releases/tag/v2.3.1 新功能 用户可以在串口终端中使用回车键发送数据。提高 esp32/8266 的默认上传波特率以提高上传速度。删除 es…

python积木编程软件_积木编程软件手机版下载

积木编程软件是可以在线可以学习编程的软件,平台给用户提供了大量的编程学习教程&#xff0c;软件还支持在线编程&#xff0c;学习内容丰富&#xff0c;让用户学习起来不会枯燥乏味&#xff0c;就算你是零基础的小白&#xff0c;也可以体验到编程的乐趣。 积木编程官方简介 积木…

零代码积木编程案例分享

和大家分享一个通过积木编程技术实现的复杂业务系统的案例&#xff0c;主要比较传统有代码和零代码积木编程在项目各个阶段的人力投入和成本比较。 客户是一家工业丝线工厂&#xff0c;期望实现生产、仓库、设备的管理&#xff0c;同时连线近百台设备&#xff0c;还要集成AGV、…

时钟的积木编程

家里壁挂的时钟滴答滴答响着&#xff0c;里面有时针、分钟和秒针。在时钟刻度盘里&#xff0c;秒针每60秒走完一周&#xff0c;即秒针的最小角度为360/60 6&#xff1b;分针每60分钟走完一周&#xff0c;即分针的最小角度为360/60 6&#xff1b;时针每12小时&#xff0c;走完一…

长方形的积木编程

用Scratch画长方形有多种方法&#xff0c;可以用一个“点长宽”来画&#xff0c;也可以使用"长方形的2个对角顶点来画"&#xff0c;还可以使用"长翻转90宽翻转90来画"。 1、"点长宽"方式 1.1 定义"直线"积木 点击Scratch左侧的&quo…

Netsparker介绍

Netsparker是一款综合型的web应用安全漏洞扫描工具&#xff0c;它分为专业版和免费版&#xff0c;免费版的功能也比较强大。Netsparker与其他综合性的web应用安全扫描工具相比的一个特点是它能够更好的检测SQL Injection和 Cross-site Scripting类型的安全漏洞。‍‍ 转载于:ht…

使用Netsparker扫描及**某站点

1.1使用Netsparker扫描及某站点 目前市面上漏洞扫描软件有多款,前面介绍了一些扫描软件例如awvs对目标站点进行扫描,其实国外还有一款扫描软件Netsparker也挺厉害,扫描误报率低,扫描效果还不错。在本文中,对一个目标站点进行扫描,并未发现高危漏洞,但通过列目录漏洞…

开源的网络监控工具:Sniffnet,简单而有趣!

在当今数字化的世界中&#xff0c;网络监控工具对于管理和保护网络的安全至关重要。网络管理员和安全专业人员需要一种高效且易于使用的工具&#xff0c;以监视和分析网络流量&#xff0c;识别潜在的安全威胁和性能问题。Sniffnet 是一个开源的网络监控工具&#xff0c;它提供了…

Netspark自动批量扫描powershell脚本

脚本需要创建2个文件&#xff0c;1个文件夹&#xff0c;都放到Netsparker目录下&#xff0c;将脚本中涉及到netspark目录的修改成自己的目录。 注意&#xff1a;Netsparker目录不能含有中文名。 文件1&#xff1a;脚本文件&#xff0c;文件名auto.ps1&#xff0c;内容如下&am…

WebSocket in ASP.NET Core

回到目录 一、WebSocket WebSocket是HTML5出的东西&#xff08;协议&#xff09;&#xff0c;也就是说HTTP协议没有变化&#xff0c;或者说没关系&#xff0c;但HTTP是不支持持久连接的&#xff08;长连接&#xff0c;循环连接的不算&#xff09; 首先HTTP有1.1和1.0之说&#…