设计使用 Blockly 的应用程序时,有几种范例可供选择。这些选择应尽早考虑,因为它们会影响用户所需的功能块。
许多 Blockly 应用程序用于描述配置,而不是可执行程序。配置应用程序通常从初始化工作空间上的一个根级别块开始。一个很好的例子是 Blockly 开发者工具 的 Block Factory 选项卡:
var xml =
'<xml><block type="factory_base" deletable="false" movable="false"></block></xml>';
Blockly.Xml.domToWorkspace(Blockly.Xml.textToDom(xml), workspace);
这将创建一个不可删除的,不可移动的块,其中包含所有用户的配置。可以随时将工作区导出到 XML,以确定当前配置。
此类应用程序可能希望自动禁用任何未连接到根块的块。只需一行即可在 Blockly 的 Web 版本上完成此操作:
workspace.addChangeListener(Blockly.Events.disableOrphans);
大多数 Blockly 应用程序旨在创建串行程序。用户将按顺序执行的块堆叠在一起。
工作区中的每个(非禁用)块都将构成程序的一部分。如果有多个堆栈块,则首先执行较高的堆栈。(如果两个堆栈的高度大致相同,则优先考虑左侧(在 RTL 模式下为右侧)的堆栈。)
工作区可以随时导出为可执行代码。此代码可以在 JavaScript 的客户端(使用 eval 或 JS Interpreter)执行,也可以在服务器端以任何语言执行。
var code = Blockly.JavaScript.workspaceToCode(workspace);
一些 Blockly 应用程序选择并行而非串行执行所有块堆栈。一个例子是音乐应用程序,其中鼓循环与旋律同时运行。
实现并行执行的一种方法是使用无头工作空间生成多个代码段:
var xml = Blockly.Xml.workspaceToDom(workspace);
// Find and remove all top blocks.
var topBlocks = [];
for (
var i = xml.childNodes.length - 1, node;
(block = xml.childNodes[i]);
i--
) {
if (block.tagName == "BLOCK") {
xml.removeChild(block);
topBlocks.unshift(block);
}
}
// Add each top block one by one and generate code.
var allCode = [];
for (var i = 0, block; (block = topBlocks[i]); i++) {
var headless = new Blockly.Workspace();
xml.appendChild(block);
Blockly.Xml.domToWorkspace(xml, headless);
allCode.push(Blockly.JavaScript.workspaceToCode(headless));
headless.dispose();
xml.removeChild(block);
}
如果目标语言是 JavaScript,则可以使用 allCode
数组创建多个 JS 解释器以同时执行。 如果目标语言是类似 Python 的语言,则可以将 allCode
数组组装为使用线程模块的单个程序。
与任何并行程序一样,必须对任何共享资源(例如变量和函数)做出谨慎的决定。
事件处理程序只是由系统而不是由程序调用的函数。这些块可以封装要执行的块堆栈,也可以是位于块堆栈顶部的标头。
一些开发人员喜欢在事件块的顶部添加一个“帽子”,以使它们看起来与其他块不同。这不是 Blockly 的默认外观,但可以通过设置 Blockly.BlockSvg.START_HAT = true;
()或添加主题并在 block style 上设置 hat 选项来添加。有关在主题上设置帽子的更多信息,请参见 此处。
一些开发人员喜欢在事件块的顶部添加一个“帽子”,以使它们看起来与其他块不同。 这不是 Blockly 的默认外观,但可以通过设置 Blockly.BlockSvg.START_HAT = true;
(自定义渲染器 Codelab-覆盖常量)来添加。 或通过添加主题并在方块样式上设置帽子选项。 有关在主题上设置帽子的更多信息,请参见 此处。
在事件驱动的模型中,也可以为程序启动创建处理程序。在此模型下,工作区上未连接到事件处理程序的任何块都将被忽略并且不会执行。
设计使用事件的系统时,请考虑是否可能或希望支持同一事件处理程序的多个实例。