blockly.js 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708
  1. import * as Blockly from "blockly";
  2. import 'blockly/msg/zh-hans';
  3. import {javascriptGenerator} from "blockly/javascript";
  4. import {pythonGenerator} from "blockly/python";
  5. // 导入图片资源
  6. import about_turnImage from '@/assets/images/blockly/component/about_turn.png';
  7. import move_backwardImage from '@/assets/images/blockly/component/move_backward.png';
  8. import move_forwardImage from '@/assets/images/blockly/component/move_forward.png';
  9. import pick_upImage from '@/assets/images/blockly/component/pick_up.png';
  10. import turn_leftImage from '@/assets/images/blockly/component/turn_left.png';
  11. import turn_rightImage from '@/assets/images/blockly/component/turn_right.png';
  12. import useImage from '@/assets/images/blockly/component/use.png';
  13. import pauseImage from '@/assets/images/blockly/component/pause.png';
  14. import play_soundImage from '@/assets/images/blockly/component/sound.png';
  15. import constructImage from '@/assets/images/blockly/component/construct.png';
  16. import {DICT_TYPE} from '@/utils/dictUtils.js';
  17. // 安全地获取和解析字典数据
  18. const getDictDataFromStorage = (dictType) => {
  19. try {
  20. const data = localStorage.getItem(dictType);
  21. return data ? JSON.parse(data) : [];
  22. } catch (error) {
  23. console.error(`解析${dictType}字典数据失败:`, error);
  24. return [];
  25. }
  26. };
  27. // 地图元素类型字典
  28. const savedMapTypeDict = getDictDataFromStorage(DICT_TYPE.AI_BLOCKLY_MAP_TYPE);
  29. const BLOCKLY_MAP_TYPE_DICT = savedMapTypeDict.reduce((acc, item) => {
  30. acc[item.value.toUpperCase()] = item.value;
  31. return acc;
  32. }, {});
  33. // 特殊积木字典
  34. const blocklyMapSpecialDict = getDictDataFromStorage(DICT_TYPE.BLOCKLY_MAP_SPECIAL);
  35. const BLOCKLY_MAP_SPECIAL_DICT = blocklyMapSpecialDict.reduce((acc, item) => {
  36. acc[item.value.toUpperCase()] = item.value;
  37. return acc;
  38. }, {});
  39. // 地图标记字典
  40. const blocklyMapMarkDict = getDictDataFromStorage(DICT_TYPE.BLOCKLY_MAP_MARK);
  41. const BLOCKLY_MAP_MARK_DICT = blocklyMapMarkDict.reduce((acc, item) => {
  42. acc[item.value.toUpperCase()] = {
  43. value: item.value,
  44. label: item.label,
  45. img: item.remark
  46. };
  47. return acc;
  48. }, {});
  49. /**
  50. * 定义所有可用的自定义积木配置
  51. */
  52. const availableBlocks = {
  53. // 通用积木 - 始终可用(移动控制)
  54. move_forward: {
  55. jsonConfig: {
  56. "type": "move_forward",
  57. "message0": "%1 向前移动",
  58. "args0": [
  59. {
  60. "type": "field_image",
  61. "src": move_forwardImage,
  62. "width": 20,
  63. "height": 20,
  64. "alt": "向前移动"
  65. }
  66. ],
  67. "previousStatement": null,
  68. "nextStatement": null,
  69. "colour": 230,
  70. "tooltip": "控制角色向前移动一格",
  71. "helpUrl": ""
  72. },
  73. isGeneral: true // 标记为通用积木
  74. },
  75. move_forward_param: {
  76. jsonConfig: {
  77. "type": "move_forward_param",
  78. "message0": "%1 向前移动 %2 次",
  79. "args0": [
  80. {
  81. "type": "field_image",
  82. "src": move_forwardImage,
  83. "width": 20,
  84. "height": 20,
  85. "alt": "向前移动多次"
  86. },
  87. {
  88. "type": "field_number",
  89. "name": "PARAM",
  90. "value": 1,
  91. "min": 1,
  92. "max": 10,
  93. "precision": 1
  94. }
  95. ],
  96. "previousStatement": null,
  97. "nextStatement": null,
  98. "colour": 230,
  99. "tooltip": "控制角色向前移动指定次数",
  100. "helpUrl": ""
  101. },
  102. isGeneral: true // 标记为通用积木
  103. },
  104. move_backward: {
  105. jsonConfig: {
  106. "type": "move_backward",
  107. "message0": "%1 向后移动",
  108. "args0": [
  109. {
  110. "type": "field_image",
  111. "src": move_backwardImage,
  112. "width": 20,
  113. "height": 20,
  114. "alt": "向后移动"
  115. }
  116. ],
  117. "previousStatement": null,
  118. "nextStatement": null,
  119. "colour": 230,
  120. "tooltip": "控制角色向后移动一格",
  121. "helpUrl": ""
  122. },
  123. isGeneral: true
  124. },
  125. move_backward_param: {
  126. jsonConfig: {
  127. "type": "move_backward_param",
  128. "message0": "%1 向后移动 %2 次",
  129. "args0": [
  130. {
  131. "type": "field_image",
  132. "src": move_backwardImage,
  133. "width": 20,
  134. "height": 20,
  135. "alt": "向后移动多次"
  136. },
  137. {
  138. "type": "field_number",
  139. "name": "PARAM",
  140. "value": 1,
  141. "min": 1,
  142. "max": 10,
  143. "precision": 1
  144. }
  145. ],
  146. "previousStatement": null,
  147. "nextStatement": null,
  148. "colour": 230,
  149. "tooltip": "控制角色向后移动指定次数",
  150. "helpUrl": ""
  151. },
  152. isGeneral: true
  153. },
  154. turn_left: {
  155. jsonConfig: {
  156. "type": "turn_left",
  157. "message0": "%1 向左转",
  158. "args0": [
  159. {
  160. "type": "field_image",
  161. "src": turn_leftImage,
  162. "width": 20,
  163. "height": 20,
  164. "alt": "向左转"
  165. }
  166. ],
  167. "previousStatement": null,
  168. "nextStatement": null,
  169. "colour": 230,
  170. "tooltip": "控制角色向左转",
  171. "helpUrl": ""
  172. },
  173. isGeneral: true
  174. },
  175. turn_right: {
  176. jsonConfig: {
  177. "type": "turn_right",
  178. "message0": "%1 向右转",
  179. "args0": [
  180. {
  181. "type": "field_image",
  182. "src": turn_rightImage,
  183. "width": 20,
  184. "height": 20,
  185. "alt": "向右转"
  186. }
  187. ],
  188. "previousStatement": null,
  189. "nextStatement": null,
  190. "colour": 230,
  191. "tooltip": "控制角色向右转",
  192. "helpUrl": ""
  193. },
  194. isGeneral: true
  195. },
  196. turn_around: {
  197. jsonConfig: {
  198. "type": "turn_around",
  199. "message0": "%1 向后转",
  200. "args0": [
  201. {
  202. "type": "field_image",
  203. "src": about_turnImage,
  204. "width": 20,
  205. "height": 20,
  206. "alt": "向后转"
  207. }
  208. ],
  209. "previousStatement": null,
  210. "nextStatement": null,
  211. "colour": 230,
  212. "tooltip": "控制角色向后转",
  213. "helpUrl": ""
  214. },
  215. isGeneral: true
  216. },
  217. // 通用积木 - 始终可用(功能积木)
  218. pickup_item: {
  219. jsonConfig: {
  220. "type": "pickup_item",
  221. "message0": "%1 拾取物品",
  222. "args0": [
  223. {
  224. "type": "field_image",
  225. "src": pick_upImage,
  226. "width": 20,
  227. "height": 20,
  228. "alt": "拾取物品"
  229. }
  230. ],
  231. "previousStatement": null,
  232. "nextStatement": null,
  233. "colour": 30,
  234. "tooltip": "尝试拾取当前位置的物品",
  235. "helpUrl": ""
  236. },
  237. isGeneral: true
  238. },
  239. use_item: {
  240. jsonConfig: {
  241. "type": "use_item",
  242. "message0": "%1 使用物品",
  243. "args0": [
  244. {
  245. "type": "field_image",
  246. "src": useImage,
  247. "width": 20,
  248. "height": 20,
  249. "alt": "使用物品"
  250. }
  251. ],
  252. "previousStatement": null,
  253. "nextStatement": null,
  254. "colour": 30,
  255. "tooltip": "在当前位置使用物品",
  256. "helpUrl": ""
  257. },
  258. isGeneral: true
  259. },
  260. // 当经过某个形状的积木(可作为参数使用的value类型)
  261. when_passed: {
  262. jsonConfig: {
  263. "type": "when_passed",
  264. "message0": "当经过 : %1",
  265. "args0": [
  266. {
  267. "type": "field_dropdown",
  268. "name": "SHAPE",
  269. "options": Object.values(BLOCKLY_MAP_MARK_DICT).map(shape => [
  270. {
  271. "src": shape.img,
  272. "width": 30,
  273. "height": 30,
  274. "alt": shape.label
  275. },
  276. shape.value
  277. ])
  278. }
  279. ],
  280. "output": "Boolean",
  281. "colour": 30,
  282. "tooltip": "检查角色是否经过指定形状,可作为参数使用",
  283. "helpUrl": ""
  284. },
  285. isGeneral: true
  286. },
  287. // 特殊积木 - 可根据配置动态加载
  288. pause: {
  289. jsonConfig: {
  290. "type": "pause",
  291. "message0": "%1 暂停 %2 秒",
  292. "args0": [
  293. {
  294. "type": "field_image",
  295. "src": pauseImage,
  296. "width": 20,
  297. "height": 20,
  298. "alt": "暂停"
  299. },
  300. {
  301. "type": "field_number",
  302. "name": "SECONDS",
  303. "value": 1,
  304. "min": 1,
  305. "max": 10,
  306. "precision": 1
  307. }
  308. ],
  309. "previousStatement": null,
  310. "nextStatement": null,
  311. "colour": 0,
  312. "tooltip": "让角色在原地停留指定的秒数",
  313. "helpUrl": ""
  314. },
  315. isGeneral: false
  316. },
  317. play_sound: {
  318. jsonConfig: {
  319. "type": "play_sound",
  320. "message0": "%1 声音",
  321. "args0": [
  322. {
  323. "type": "field_image",
  324. "src": play_soundImage,
  325. "width": 20,
  326. "height": 20,
  327. "alt": "声音"
  328. }
  329. ],
  330. "previousStatement": null,
  331. "nextStatement": null,
  332. "colour": 160,
  333. "tooltip": "自动触发特效提示音,并根据当前方格类型执行相应操作",
  334. "helpUrl": ""
  335. },
  336. isGeneral: false
  337. },
  338. construct: {
  339. jsonConfig: {
  340. "type": "construct",
  341. "message0": "%1 修建",
  342. "args0": [
  343. {
  344. "type": "field_image",
  345. "src": constructImage,
  346. "width": 20,
  347. "height": 20,
  348. "alt": "修建"
  349. }
  350. ],
  351. "previousStatement": null,
  352. "nextStatement": null,
  353. "colour": 160,
  354. "tooltip": "修建处理",
  355. "helpUrl": ""
  356. },
  357. isGeneral: false
  358. }
  359. };
  360. /**
  361. * 定义所有可用的JavaScript生成器配置
  362. */
  363. const availableGenerators = {
  364. // 通用生成器 - 始终可用
  365. move_forward: function(block) {
  366. return 'await moveForward();\n';
  367. },
  368. move_forward_param: function(block) {
  369. const stepCount = block.getFieldValue('PARAM');
  370. return `await moveForward(${stepCount});\n`;
  371. },
  372. move_backward: function(block) {
  373. return 'await moveBackward();\n';
  374. },
  375. move_backward_param: function(block) {
  376. const stepCount = block.getFieldValue('PARAM');
  377. return `await moveBackward(${stepCount});\n`;
  378. },
  379. turn_left: function(block) {
  380. return 'await turnLeft();\n';
  381. },
  382. turn_right: function(block) {
  383. return 'await turnRight();\n';
  384. },
  385. turn_around: function(block) {
  386. return 'await turnAround();\n';
  387. },
  388. pickup_item: function(block) {
  389. return 'await pickupItem();\n';
  390. },
  391. use_item: function(block) {
  392. return 'await useItem();\n';
  393. },
  394. when_passed: function(block) {
  395. const shape = block.getFieldValue('SHAPE');
  396. return [`whenPassed('${shape}')`, javascriptGenerator.ORDER_FUNCTION_CALL];
  397. },
  398. // 特殊生成器 - 可根据配置动态加载
  399. pause: function(block) {
  400. const seconds = block.getFieldValue('SECONDS');
  401. return `await pause(${seconds});\n`;
  402. },
  403. play_sound: function(block) {
  404. return 'await playSound();\n';
  405. },
  406. construct: function(block) {
  407. return 'await construct();\n';
  408. }
  409. };
  410. /**
  411. * 定义所有可用的Python生成器配置
  412. */
  413. const availablePythonGenerators = {
  414. // 通用生成器 - 始终可用
  415. move_forward: function(block) {
  416. return 'move_forward()\n';
  417. },
  418. move_forward_param: function(block) {
  419. const stepCount = block.getFieldValue('PARAM');
  420. return `move_forward(${stepCount})\n`;
  421. },
  422. move_backward: function(block) {
  423. return 'move_backward()\n';
  424. },
  425. move_backward_param: function(block) {
  426. const stepCount = block.getFieldValue('PARAM');
  427. return `move_backward(${stepCount})\n`;
  428. },
  429. turn_left: function(block) {
  430. return 'turn_left()\n';
  431. },
  432. turn_right: function(block) {
  433. return 'turn_right()\n';
  434. },
  435. turn_around: function(block) {
  436. return 'turn_around()\n';
  437. },
  438. pickup_item: function(block) {
  439. return 'pickup_item()\n';
  440. },
  441. use_item: function(block) {
  442. return 'use_item()\n';
  443. },
  444. when_passed: function(block) {
  445. const shape = block.getFieldValue('SHAPE');
  446. return [`when_passed('${shape}')`, pythonGenerator.ORDER_FUNCTION_CALL];
  447. },
  448. // 特殊生成器 - 可根据配置动态加载
  449. pause: function(block) {
  450. const seconds = block.getFieldValue('SECONDS');
  451. return `pause(${seconds})\n`;
  452. },
  453. play_sound: function(block) {
  454. return 'play_sound()\n';
  455. },
  456. construct: function(block) {
  457. return 'construct()\n';
  458. }
  459. };
  460. /**
  461. * 注册自定义Blockly积木
  462. * @param {Array<string>} blocklySpecialBlocks - 特殊积木方法标识集合
  463. */
  464. export function registerCustomBlocks(blocklySpecialBlocks = []) {
  465. // 确保blocklySpecialBlocks始终是数组
  466. blocklySpecialBlocks = Array.isArray(blocklySpecialBlocks) ? blocklySpecialBlocks : [];
  467. // 遍历所有可用积木
  468. for (const [blockName, blockConfig] of Object.entries(availableBlocks)) {
  469. // 如果是通用积木,或者是允许的特殊积木,则注册
  470. if (blockConfig.isGeneral || blocklySpecialBlocks.includes(blockName)) {
  471. Blockly.Blocks[blockName] = {
  472. init: function() {
  473. this.jsonInit(blockConfig.jsonConfig);
  474. }
  475. };
  476. }
  477. }
  478. }
  479. /**
  480. * 设置Blockly中文本地化
  481. */
  482. export function setupBlocklyChineseLocale() {
  483. // 使用扩展方式覆盖默认的英文文本为中文
  484. const locale = {
  485. // 逻辑积木中文配置
  486. CONTROLS_IF_MSG_IF: "如果",
  487. CONTROLS_IF_MSG_ELSEIF: "否则如果",
  488. CONTROLS_IF_MSG_ELSE: "否则",
  489. CONTROLS_IF_MSG_THEN: "执行",
  490. CONTROLS_IF_MSG_DO: "执行",
  491. CONTROLS_IF_TOOLTIP_1: "如果条件为真,则执行相应的代码。",
  492. CONTROLS_IF_TOOLTIP_2: "如果条件为真,则执行相应的代码;否则执行其他代码。",
  493. CONTROLS_IF_TOOLTIP_3: "如果条件为真,则执行相应的代码;否则检查其他条件。",
  494. CONTROLS_IF_TOOLTIP_4: "如果条件为真,则执行相应的代码;否则检查其他条件,如果都不满足则执行最后代码。",
  495. // 控制块的弹出菜单文本
  496. CONTROLS_IF_IF: "如果",
  497. CONTROLS_IF_ELSEIF: "否则如果",
  498. CONTROLS_IF_ELSE: "否则",
  499. CONTROLS_IF_IF_TITLE: "如果",
  500. CONTROLS_IF_IF_TITLE_IF: "如果",
  501. CONTROLS_IF_ELSEIF_TITLE_ELSEIF: "否则如果",
  502. CONTROLS_IF_ELSE_TITLE: "否则",
  503. CONTROLS_IF_ELSE_TITLE_ELSE: "否则",
  504. // 条件块的提示文本和帮助URL
  505. CONTROLS_IF_HELPURL: "条件语句帮助",
  506. CONTROLS_IF_TOOLTIP_IF: "添加或删除条件。",
  507. CONTROLS_IF_TOOLTIP_ELSE: "添加或删除否则块。",
  508. // 添加设置按钮相关的中文配置
  509. CONTROLS_IF_ELSEIF_TOOLTIP: "否则如果条件为真,则执行相应的代码。",
  510. CONTROLS_IF_ELSE_TOOLTIP: "否则执行相应的代码。",
  511. // 比较运算符中文配置
  512. LOGIC_COMPARE_EQ: "等于",
  513. LOGIC_COMPARE_NEQ: "不等于",
  514. LOGIC_COMPARE_LT: "小于",
  515. LOGIC_COMPARE_LTE: "小于等于",
  516. LOGIC_COMPARE_GT: "大于",
  517. LOGIC_COMPARE_GTE: "大于等于",
  518. LOGIC_COMPARE_TOOLTIP_EQ: "比较两个值是否相等。",
  519. LOGIC_COMPARE_TOOLTIP_NEQ: "比较两个值是否不相等。",
  520. LOGIC_COMPARE_TOOLTIP_LT: "比较第一个值是否小于第二个值。",
  521. LOGIC_COMPARE_TOOLTIP_LTE: "比较第一个值是否小于或等于第二个值。",
  522. LOGIC_COMPARE_TOOLTIP_GT: "比较第一个值是否大于第二个值。",
  523. LOGIC_COMPARE_TOOLTIP_GTE: "比较第一个值是否大于或等于第二个值。",
  524. // 逻辑运算中文配置
  525. LOGIC_OPERATION_AND: "且",
  526. LOGIC_OPERATION_OR: "或",
  527. LOGIC_OPERATION_TOOLTIP_AND: "如果两个条件都为真,则结果为真。",
  528. LOGIC_OPERATION_TOOLTIP_OR: "如果任一条件为真,则结果为真。",
  529. // 布尔值配置
  530. LOGIC_BOOLEAN_TRUE: "真",
  531. LOGIC_BOOLEAN_FALSE: "假",
  532. LOGIC_BOOLEAN_TOOLTIP: "返回一个布尔值:真或假。",
  533. // 循环积木中文配置
  534. CONTROLS_REPEAT_TITLE: "重复%1次",
  535. CONTROLS_REPEAT_TOOLTIP: "重复执行内部代码指定次数。",
  536. CONTROLS_REPEAT_MSG_DO: "执行",
  537. CONTROLS_REPEAT_INPUT_DO: "执行",
  538. CONTROLS_WHILEUNTIL_MSG_DO: "执行",
  539. CONTROLS_WHILEUNTIL_INPUT_DO: "执行",
  540. CONTROLS_FOR_MSG_DO: "执行",
  541. CONTROLS_FOR_INPUT_DO: "执行",
  542. CONTROLS_FOR_EACH_MSG_DO: "执行",
  543. CONTROLS_FOR_EACH_INPUT_DO: "执行",
  544. // 数学积木中文配置
  545. MATH_NUMBER_TOOLTIP: "一个数字。在编辑器中双击以更改。",
  546. MATH_ADDITION_SYMBOL: "加",
  547. MATH_SUBTRACTION_SYMBOL: "减",
  548. MATH_MULTIPLICATION_SYMBOL: "乘",
  549. MATH_DIVISION_SYMBOL: "除",
  550. MATH_ARITHMETIC_TOOLTIP_ADD: "返回两个数的和。",
  551. MATH_ARITHMETIC_TOOLTIP_SUBTRACT: "返回第一个数减去第二个数的差。",
  552. MATH_ARITHMETIC_TOOLTIP_MULTIPLY: "返回两个数的积。",
  553. MATH_ARITHMETIC_TOOLTIP_DIVIDE: "返回第一个数除以第二个数的商。"
  554. };
  555. // 使用Object.assign来合并配置,避免直接修改导入对象
  556. Object.assign(Blockly.Msg, locale);
  557. }
  558. /**
  559. * 注册JavaScript生成器
  560. * @param {Array<string>} blocklySpecialBlocks - 特殊积木方法标识集合
  561. */
  562. export function registerJavaScriptGenerators(blocklySpecialBlocks = []) {
  563. // 确保blocklySpecialBlocks始终是数组
  564. blocklySpecialBlocks = Array.isArray(blocklySpecialBlocks) ? blocklySpecialBlocks : [];
  565. // 遍历所有可用生成器
  566. for (const [blockName, generatorFunc] of Object.entries(availableGenerators)) {
  567. // 如果是通用生成器,或者是允许的特殊生成器对应的生成器,则注册
  568. if (availableBlocks[blockName].isGeneral || blocklySpecialBlocks.includes(blockName)) {
  569. javascriptGenerator.forBlock[blockName] = generatorFunc;
  570. }
  571. }
  572. // 为text_print块添加生成器,用于调试(始终可用)
  573. javascriptGenerator.forBlock['text_print'] = function(block) {
  574. const msg = javascriptGenerator.valueToCode(block, 'TEXT', javascriptGenerator.ORDER_NONE) || '';
  575. return msg;
  576. };
  577. }
  578. /**
  579. * 注册Python生成器
  580. * @param {Array<string>} blocklySpecialBlocks - 特殊积木方法标识集合
  581. */
  582. export function registerPythonGenerators(blocklySpecialBlocks = []) {
  583. // 确保blocklySpecialBlocks始终是数组
  584. blocklySpecialBlocks = Array.isArray(blocklySpecialBlocks) ? blocklySpecialBlocks : [];
  585. // 遍历所有可用生成器
  586. for (const [blockName, generatorFunc] of Object.entries(availablePythonGenerators)) {
  587. // 如果是通用生成器,或者是允许的特殊生成器对应的生成器,则注册
  588. if (availableBlocks[blockName].isGeneral || blocklySpecialBlocks.includes(blockName)) {
  589. pythonGenerator.forBlock[blockName] = generatorFunc;
  590. }
  591. }
  592. // 为text_print块添加生成器,用于调试(始终可用)
  593. pythonGenerator.forBlock['text_print'] = function(block) {
  594. const msg = pythonGenerator.valueToCode(block, 'TEXT', pythonGenerator.ORDER_NONE) || '';
  595. return msg;
  596. };
  597. }
  598. /**
  599. * 初始化Blockly工作区
  600. * @param {HTMLElement} blocklyDiv - Blockly容器元素
  601. * @param {HTMLElement} toolbox - 工具箱元素
  602. * @returns {Blockly.WorkspaceSvg} - Blockly工作区实例
  603. */
  604. export function initBlockly(options = {}) {
  605. // 应用中文配置
  606. setupBlocklyChineseLocale();
  607. const defaultOptions = {
  608. toolbox: document.getElementById('toolbox'),
  609. collapse: false,
  610. comments: true,
  611. disable: false, // 设为false以允许编辑
  612. maxBlocks: Infinity,
  613. trashcan: true,
  614. horizontalLayout: false,
  615. toolboxPosition: 'start',
  616. toolboxCollapse: false, // 设置工具箱默认展开
  617. toolboxAlwaysExpanded: true, // 确保点击类别后保持展开状态
  618. css: true,
  619. media: 'https://unpkg.com/blockly/media/',
  620. rtl: false,
  621. scrollbars: true,
  622. sounds: false, // 禁用声音以提高性能
  623. oneBasedIndex: true,
  624. grid: {
  625. spacing: 20,
  626. length: 3,
  627. colour: "#ccc",
  628. snap: true
  629. },
  630. zoom: {
  631. controls: true,
  632. wheel: true,
  633. startScale: 1.0,
  634. maxScale: 3,
  635. minScale: 0.3,
  636. scaleSpeed: 1.2
  637. }
  638. };
  639. // 合并默认选项和用户提供的选项
  640. const finalOptions = { ...defaultOptions, ...options };
  641. return Blockly.inject('blocklyDiv', finalOptions);
  642. }
  643. /**
  644. * 获取所有可用的积木配置
  645. * @returns {Object} 所有可用积木的配置对象
  646. */
  647. export function getAllBlocksConfig() {
  648. return {...availableBlocks};
  649. }
  650. /**
  651. * 获取所有可用的JavaScript生成器配置
  652. * @returns {Object} 所有可用JavaScript生成器的配置对象
  653. */
  654. export function getAllGeneratorsConfig() {
  655. return {...availableGenerators};
  656. }
  657. /**
  658. * 获取所有可用的Python生成器配置
  659. * @returns {Object} 所有可用Python生成器的配置对象
  660. */
  661. export function getAllPythonGeneratorsConfig() {
  662. return {...availablePythonGenerators};
  663. }
  664. // 导出字典常量
  665. export { BLOCKLY_MAP_TYPE_DICT, BLOCKLY_MAP_MARK_DICT, BLOCKLY_MAP_SPECIAL_DICT };
  666. export default {
  667. registerCustomBlocks,
  668. registerJavaScriptGenerators,
  669. registerPythonGenerators,
  670. setupBlocklyChineseLocale,
  671. initBlockly,
  672. getAllBlocksConfig,
  673. getAllGeneratorsConfig,
  674. BLOCKLY_MAP_TYPE_DICT,
  675. BLOCKLY_MAP_MARK_DICT,
  676. BLOCKLY_MAP_SPECIAL_DICT
  677. };