前言
前段时间想要做一个web端的图形化积木式编程(类似少儿编程)的案例,网上冲浪了一圈又一圈,终于技术选型好,然后代码一顿敲,终于出来了一个雏形。
TIPS:该案例设计主要参考iRobot Coding,只用做学习用途,侵删。
最终实现效果
本文实现效果
完整代码
- Blockly自定义块
<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">执行代码</button>
</div>
</template>
<script>
import Blockly from 'blockly'
import BlocklyJS from 'blockly/javascript';
import './customBlock'
export default {
name: "blocklyClass1",
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": "turn",
},
{
"kind": "block",
"type": "arc"
},
{
"kind": "block",
"type": "draw"
},
{
"kind": "block",
"type": "pencilcolor"
},
{
"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);
},
methods:{
/**
* block代码块转为代码
*/
block2code(){
this.code = BlocklyJS.workspaceToCode(this.$refs.blocklyDiv.workspace)
},
/**
* 执行生成代码
*/
runCode(){
if(!this.code){alert('请先点击生成代码');return}
eval(this.code)
},
},
}
</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>
- 封装的自定义块方法 - customBlock.js
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": "turn",
"message0": "向 %1 %2",
"args0": [
{
"type": "field_dropdown",
"name": "dirction",
"options": [
[
"左转",
"0"
],
[
"右转",
"1"
]
]
},
{
"type": "field_angle",
"name": "degree",
"angle": 90
}
],
"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": ""
},
{
"type": "draw",
"message0": "设置 %1",
"args0": [
{
"type": "field_dropdown",
"name": "pencilState",
"options": [
[
{
"src": "",
"width": 50,
"height": 50,
"alt": "pencil down"
},
"1"
],
[
{
"src": "",
"width": 50,
"height": 50,
"alt": "pencil up"
},
"0"
]
]
}
],
"previousStatement": null,
"nextStatement": null,
"colour": "#81C679",
"tooltip": "",
"helpUrl": ""
},
{
"type": "pencilcolor",
"message0": "设置笔颜色: 红 %1 绿 %2 蓝 %3",
"args0": [
{
"type": "field_number",
"name": "red",
"value": 100,
"min": 0,
"max": 255
},
{
"type": "field_number",
"name": "green",
"value": 100,
"min": 0,
"max": 255
},
{
"type": "field_number",
"name": "blue",
"value": 100,
"min": 0,
"max": 255
}
],
"previousStatement": null,
"nextStatement": null,
"colour": "#81C679",
"tooltip": "",
"helpUrl": ""
}
]
);
/**
* 自定义组件生成代码
* @param block
* @returns {string}
*/
Blockly.JavaScript['while_program_start'] = function (block) {
block
var while_content = Blockly.JavaScript.statementToCode(block, 'while_content');
var code =
'robot.init();\n'
+ while_content +
'robot.stop();\n';
return code;
};
Blockly.JavaScript['move'] = function (block) {
var text_move_distance = block.getFieldValue('move_distance');
var code = 'robot.move(' + text_move_distance + ');\n';
return code;
};
Blockly.JavaScript['turn'] = function (block) {
var dropdown_dirction = block.getFieldValue('dirction');
var angle_degree = block.getFieldValue('degree');
var code = 'robot.turn(' + dropdown_dirction + ', ' + angle_degree + ');\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 = 'robot.arc(' + dropdown_dirction + ', ' + angle_degree + ',' + radius + ');\n';
return code;
};
Blockly.JavaScript['draw'] = function (block) {
var dropdown_pencilstate = block.getFieldValue('pencilState');
var code = 'robot.drawable(' + dropdown_pencilstate + ');\n';
return code;
};
Blockly.JavaScript['pencilcolor'] = function (block) {
var number_red = block.getFieldValue('red') / 255.0;
var number_green = block.getFieldValue('green') / 255.0;
var number_blue = block.getFieldValue('blue') / 255.0;
var code = 'await robot.pencilcolor(' + number_red + ',' + number_green + ',' + number_blue + ');\n';
return code;
};
代码分解
自定义块主要分3部分:
1、定义块行为
2、注册块
3、定义块生成代码
4、引入块
0.代码块前置知识
0.1 汉化
import * as hans from 'blockly/msg/zh-hans'
Blockly.setLocale(hans);//汉化
0.2 预置块
Blockly原本已经预置了很多代码块了
-
逻辑块(logic_compare、logic_operation、logic_negate、logic_boolean)
-
循环控制块(controls_repeat_ext、controls_whileUntil、controls_if)
-
数学块(math_number、math_arithmetic、math_single)
-
文本块(text、text_length、text_print)
-
变量块(variables_get、variables_set)
0.3 不同样式块含义
0.3.1 连接方式
-
上下连接
以javascript为例,在一个事件循环中,代码块会顺序执行(不包含异步代码块),代码块之间就是上下连接方式
let loop = 10//上连接为空,下连接为print
print('hello world')//上连接为赋值语句,下连接为for循环代码块
for(let i = 0; i < 10; i++){...}//上连接为print,下连接为while循环代码块
while(loop){...}//上连接为for循环代码块,下连接为空
-
左连接
左连接可以理解为其为输出
任何值的输出都使用左连接方式,如(数字模块输出数字、文本模块输出文本、逻辑模块输出真假)
0.3.2 输入方式
输入方式,即函数传入的参数
输入的连接方式不一样,日常使用中感觉不出区别
输入可以现在输入的类型,如if、when模块的外接类型输入限制了类型只能为逻辑模块
-
内联
-
外接
0.3.2 输入类型
输入的模块的类型分为值输入和块输入
值输入(数字、文本、逻辑)
块输入(代码块)
0.4 代码块构建代码块
- Blockly预置的很多代码块基本符合日常编程的各种使用场景,而其更出色的地方在于其能自定义代码块,而且是通过代码块来生成代码块
官方的自定义代码块链接:自定义代码块构建工具
工具的食用方法可参考大佬文章:blockly构建自定义块及其工具
-
而在本文中的案例中,控制小车的移动等操作,小车的动作是依次执行的,其实用得最多是上下连接的代码块,该代码块中包含内联的一个或多个值输入:
-
对应的构造如下:
-
还有就是小车启动的函数,用的是上下连接的代码块,该代码块中包含外接的块输入:
-
对应的构造如下,其中空输入只是为了在其中填入文本让其能单独一行显示(好看一点):
1. 自定义块
-
熟悉Block Definition的生成规则当然是最好的了,但是官方提供的代码块构建工具,进行了可视化预览、定义配置、块的生成代码,极大方便了我们的开发。
- 所以自定义块的流程变得很简单啦
1、通过toolbox提供的各种工具块,拖动到编辑区,根据自己的需求区构建自定义块。
2、通过预览区实时查看自定义块是否达到预期效果
3、达到预期效果后,将定义配置区中的JSON配置信息在项目中使用Blockly.defineBlocksWithJsonArray的方法注册自定义块
4、如果是需要生成JavaScript代码,生成代码区选择JavaScript,然后将代码复制到项目中
5、最后就是将代码块引入到toolbox工具箱中
1.1 定义块行为
- 定义块行为就是上面所说的编辑区编辑块规则,或者熟悉Block Definition的同学通过块定义的规则编写JSON格式文本,从而形成块的形状(其形状基本是决定了请行为和功能)
1.2 注册块
- 将定义配置区中的JSON配置信息在项目中使用Blockly.defineBlocksWithJsonArray的方法注册自定义块
/**
* 自定义组件注册
*/
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": "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": ""
}
]);
1.2 定义块生成代码
- 生成代码区选择JavaScript,然后将代码复制到项目中,其中代码已经写好了变量的获取方式,返回值就是生成的代码,我们需要编写的就是返回值的内容。
- 而对于控制小车对象而言,笔者这里统一使用robot对象,然后调用对象中的各种方法,如下面代码中拼接出来的代码为
robot.arc('方向','弧的角度','弧的半径')
/**
* 弧度运动自定义块
* @param block
* @returns {string}
*/
Blockly.JavaScript['arc'] = function(block) {
var dropdown_dirction = block.getFieldValue('dirction');
var angle_degree = block.getFieldValue('degree');
var radius = block.getFieldValue('radius');
// TODO: Assemble JavaScript into code variable.
var code = 'robot.arc(' + dropdown_dirction + ', ' + angle_degree + ',' + radius + ');\n';
return code;
};
- 如下代码是控制程序生命周期的,拼接出来的代码为
robot.init();
...(控制小车运动的代码块)
robot.stop();
/**
*程序启动自定义块
* @param block
* @returns {string}
*/
Blockly.JavaScript['while_program_start'] = function (block) {
block
var while_content = Blockly.JavaScript.statementToCode(block, 'while_content');
var code = 'robot.init();\n' +
while_content +
'robot.stop();\n';
return code;
};
1.3 引入块
- 将自定义块引入到toolbox工具箱中
data() {
options: {
toolbox: {
"kind": "flyoutToolbox",
"contents": [
{
"kind": "block",
"type": "while_program_start",
},
{
"kind": "block",
"type": "arc"
},
...
]
}
}
},
mounted() {
//注入选项
Blockly.inject(this.$refs.blocklyDiv, this.options);
},
后续计划
Blockly
- blockly第三方组件使用
- 小车控制方法的封装
- 接入js-interpreter,步骤运行block块
- ......(想到啥写啥)