mirror of
https://github.com/google/blockly.git
synced 2026-05-13 07:30:10 +02:00
b0475b0c68
* fix: Remove spurious blank lines Remove extraneous blank lines introduced by deletion of 'use strict'; pragmas. Also fix the location of the goog.declareModuleId call in core/utils/array.ts. * fix: Add missing double-blank-line before body of modules Our convention is to have two blank lines between the imports (or module ID, if there are no imports) and the beginning of the body of the module. Enforce this. * fix: one addition format error for PR #6243
420 lines
13 KiB
TypeScript
420 lines
13 KiB
TypeScript
/**
|
|
* @license
|
|
* Copyright 2020 Google LLC
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
/**
|
|
* @fileoverview Utility functions for the toolbox and flyout.
|
|
*/
|
|
|
|
/**
|
|
* Utility functions for the toolbox and flyout.
|
|
* @namespace Blockly.utils.toolbox
|
|
*/
|
|
import * as goog from '../../closure/goog/goog.js';
|
|
goog.declareModuleId('Blockly.utils.toolbox');
|
|
|
|
/* eslint-disable-next-line no-unused-vars */
|
|
// Unused import preserved for side-effects. Remove if unneeded.
|
|
import '../toolbox/category';
|
|
/* eslint-disable-next-line no-unused-vars */
|
|
// Unused import preserved for side-effects. Remove if unneeded.
|
|
import '../toolbox/separator';
|
|
|
|
/* eslint-disable-next-line no-unused-vars */
|
|
import {ConnectionState} from '../serialization/blocks.js';
|
|
import {CssConfig as CategoryCssConfig} from '../toolbox/category.js';
|
|
import {CssConfig as SeparatorCssConfig} from '../toolbox/separator.js';
|
|
import * as Xml from '../xml.js';
|
|
import * as userAgent from './useragent.js';
|
|
|
|
|
|
/**
|
|
* The information needed to create a block in the toolbox.
|
|
* Note that disabled has a different type for backwards compatibility.
|
|
* @alias Blockly.utils.toolbox.BlockInfo
|
|
*/
|
|
export interface BlockInfo {
|
|
kind: string;
|
|
blockxml: string|Node|undefined;
|
|
type: string|undefined;
|
|
gap: string|number|undefined;
|
|
disabled: string|boolean|undefined;
|
|
enabled: boolean|undefined;
|
|
id: string|undefined;
|
|
x: number|undefined;
|
|
y: number|undefined;
|
|
collapsed: boolean|undefined;
|
|
inline: boolean|undefined;
|
|
data: string|undefined;
|
|
extraState: AnyDuringMigration|undefined;
|
|
icons: {[key: string]: AnyDuringMigration}|undefined;
|
|
fields: {[key: string]: AnyDuringMigration}|undefined;
|
|
inputs: {[key: string]: ConnectionState}|undefined;
|
|
next: ConnectionState|undefined;
|
|
}
|
|
|
|
/**
|
|
* The information needed to create a separator in the toolbox.
|
|
* @alias Blockly.utils.toolbox.SeparatorInfo
|
|
*/
|
|
export interface SeparatorInfo {
|
|
kind: string;
|
|
id: string|undefined;
|
|
gap: number|undefined;
|
|
cssconfig: SeparatorCssConfig|undefined;
|
|
}
|
|
|
|
/**
|
|
* The information needed to create a button in the toolbox.
|
|
* @alias Blockly.utils.toolbox.ButtonInfo
|
|
*/
|
|
export interface ButtonInfo {
|
|
kind: string;
|
|
text: string;
|
|
callbackkey: string;
|
|
}
|
|
|
|
/**
|
|
* The information needed to create a label in the toolbox.
|
|
* @alias Blockly.utils.toolbox.LabelInfo
|
|
*/
|
|
export interface LabelInfo {
|
|
kind: string;
|
|
text: string;
|
|
id: string|undefined;
|
|
}
|
|
|
|
/**
|
|
* The information needed to create either a button or a label in the flyout.
|
|
* @alias Blockly.utils.toolbox.ButtonOrLabelInfo
|
|
*/
|
|
export type ButtonOrLabelInfo = ButtonInfo|LabelInfo;
|
|
|
|
/**
|
|
* The information needed to create a category in the toolbox.
|
|
* @alias Blockly.utils.toolbox.StaticCategoryInfo
|
|
*/
|
|
export interface StaticCategoryInfo {
|
|
kind: string;
|
|
name: string;
|
|
contents: ToolboxItemInfo[];
|
|
id: string|undefined;
|
|
categorystyle: string|undefined;
|
|
colour: string|undefined;
|
|
cssconfig: CategoryCssConfig|undefined;
|
|
hidden: string|undefined;
|
|
}
|
|
|
|
/**
|
|
* The information needed to create a custom category.
|
|
* @alias Blockly.utils.toolbox.DynamicCategoryInfo
|
|
*/
|
|
export interface DynamicCategoryInfo {
|
|
kind: string;
|
|
custom: string;
|
|
id: string|undefined;
|
|
categorystyle: string|undefined;
|
|
colour: string|undefined;
|
|
cssconfig: CategoryCssConfig|undefined;
|
|
hidden: string|undefined;
|
|
}
|
|
|
|
/**
|
|
* The information needed to create either a dynamic or static category.
|
|
* @alias Blockly.utils.toolbox.CategoryInfo
|
|
*/
|
|
export type CategoryInfo = StaticCategoryInfo|DynamicCategoryInfo;
|
|
|
|
/**
|
|
* Any information that can be used to create an item in the toolbox.
|
|
* @alias Blockly.utils.toolbox.ToolboxItemInfo
|
|
*/
|
|
export type ToolboxItemInfo = FlyoutItemInfo|StaticCategoryInfo;
|
|
|
|
/**
|
|
* All the different types that can be displayed in a flyout.
|
|
* @alias Blockly.utils.toolbox.FlyoutItemInfo
|
|
*/
|
|
export type FlyoutItemInfo =
|
|
BlockInfo|SeparatorInfo|ButtonInfo|LabelInfo|DynamicCategoryInfo;
|
|
|
|
/**
|
|
* The JSON definition of a toolbox.
|
|
* @alias Blockly.utils.toolbox.ToolboxInfo
|
|
*/
|
|
export interface ToolboxInfo {
|
|
kind?: string;
|
|
contents: ToolboxItemInfo[];
|
|
}
|
|
|
|
/**
|
|
* An array holding flyout items.
|
|
* @alias Blockly.utils.toolbox.FlyoutItemInfoArray
|
|
*/
|
|
export type FlyoutItemInfoArray = FlyoutItemInfo[];
|
|
|
|
/**
|
|
* All of the different types that can create a toolbox.
|
|
* @alias Blockly.utils.toolbox.ToolboxDefinition
|
|
*/
|
|
export type ToolboxDefinition = Node|ToolboxInfo|string;
|
|
|
|
/**
|
|
* All of the different types that can be used to show items in a flyout.
|
|
* @alias Blockly.utils.toolbox.FlyoutDefinition
|
|
*/
|
|
export type FlyoutDefinition = FlyoutItemInfoArray|NodeList|ToolboxInfo|Node[];
|
|
|
|
/**
|
|
* The name used to identify a toolbox that has category like items.
|
|
* This only needs to be used if a toolbox wants to be treated like a category
|
|
* toolbox but does not actually contain any toolbox items with the kind
|
|
* 'category'.
|
|
*/
|
|
const CATEGORY_TOOLBOX_KIND = 'categoryToolbox';
|
|
|
|
/**
|
|
* The name used to identify a toolbox that has no categories and is displayed
|
|
* as a simple flyout displaying blocks, buttons, or labels.
|
|
*/
|
|
const FLYOUT_TOOLBOX_KIND = 'flyoutToolbox';
|
|
|
|
/**
|
|
* Position of the toolbox and/or flyout relative to the workspace.
|
|
* @alias Blockly.utils.toolbox.Position
|
|
*/
|
|
export enum Position {
|
|
TOP,
|
|
BOTTOM,
|
|
LEFT,
|
|
RIGHT
|
|
}
|
|
|
|
/**
|
|
* Converts the toolbox definition into toolbox JSON.
|
|
* @param toolboxDef The definition of the toolbox in one of its many forms.
|
|
* @return Object holding information for creating a toolbox.
|
|
* @alias Blockly.utils.toolbox.convertToolboxDefToJson
|
|
*/
|
|
export function convertToolboxDefToJson(toolboxDef: ToolboxDefinition|
|
|
null): ToolboxInfo|null {
|
|
if (!toolboxDef) {
|
|
return null;
|
|
}
|
|
|
|
if (toolboxDef instanceof Element || typeof toolboxDef === 'string') {
|
|
toolboxDef = parseToolboxTree(toolboxDef);
|
|
// AnyDuringMigration because: Argument of type 'Node | null' is not
|
|
// assignable to parameter of type 'Node'.
|
|
toolboxDef = convertToToolboxJson(toolboxDef as AnyDuringMigration);
|
|
}
|
|
|
|
const toolboxJson = toolboxDef as ToolboxInfo;
|
|
validateToolbox(toolboxJson);
|
|
return toolboxJson;
|
|
}
|
|
|
|
/**
|
|
* Validates the toolbox JSON fields have been set correctly.
|
|
* @param toolboxJson Object holding information for creating a toolbox.
|
|
* @throws {Error} if the toolbox is not the correct format.
|
|
*/
|
|
function validateToolbox(toolboxJson: ToolboxInfo) {
|
|
const toolboxKind = toolboxJson['kind'];
|
|
const toolboxContents = toolboxJson['contents'];
|
|
|
|
if (toolboxKind) {
|
|
if (toolboxKind !== FLYOUT_TOOLBOX_KIND &&
|
|
toolboxKind !== CATEGORY_TOOLBOX_KIND) {
|
|
throw Error(
|
|
'Invalid toolbox kind ' + toolboxKind + '.' +
|
|
' Please supply either ' + FLYOUT_TOOLBOX_KIND + ' or ' +
|
|
CATEGORY_TOOLBOX_KIND);
|
|
}
|
|
}
|
|
if (!toolboxContents) {
|
|
throw Error('Toolbox must have a contents attribute.');
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Converts the flyout definition into a list of flyout items.
|
|
* @param flyoutDef The definition of the flyout in one of its many forms.
|
|
* @return A list of flyout items.
|
|
* @alias Blockly.utils.toolbox.convertFlyoutDefToJsonArray
|
|
*/
|
|
export function convertFlyoutDefToJsonArray(flyoutDef: FlyoutDefinition|
|
|
null): FlyoutItemInfoArray {
|
|
if (!flyoutDef) {
|
|
return [];
|
|
}
|
|
|
|
if ((flyoutDef as AnyDuringMigration)['contents']) {
|
|
return (flyoutDef as AnyDuringMigration)['contents'];
|
|
}
|
|
// If it is already in the correct format return the flyoutDef.
|
|
// AnyDuringMigration because: Property 'nodeType' does not exist on type
|
|
// 'Node | FlyoutItemInfo'.
|
|
if (Array.isArray(flyoutDef) && flyoutDef.length > 0 &&
|
|
!((flyoutDef[0]) as AnyDuringMigration).nodeType) {
|
|
// AnyDuringMigration because: Type 'FlyoutItemInfoArray | Node[]' is not
|
|
// assignable to type 'FlyoutItemInfoArray'.
|
|
return flyoutDef as AnyDuringMigration;
|
|
}
|
|
|
|
// AnyDuringMigration because: Type 'ToolboxItemInfo[] | FlyoutItemInfoArray'
|
|
// is not assignable to type 'FlyoutItemInfoArray'.
|
|
return xmlToJsonArray(flyoutDef as Node[] | NodeList) as AnyDuringMigration;
|
|
}
|
|
|
|
/**
|
|
* Whether or not the toolbox definition has categories.
|
|
* @param toolboxJson Object holding information for creating a toolbox.
|
|
* @return True if the toolbox has categories.
|
|
* @alias Blockly.utils.toolbox.hasCategories
|
|
*/
|
|
export function hasCategories(toolboxJson: ToolboxInfo|null): boolean {
|
|
if (!toolboxJson) {
|
|
return false;
|
|
}
|
|
|
|
const toolboxKind = toolboxJson['kind'];
|
|
if (toolboxKind) {
|
|
return toolboxKind === CATEGORY_TOOLBOX_KIND;
|
|
}
|
|
|
|
const categories = toolboxJson['contents'].filter(function(item) {
|
|
return item['kind'].toUpperCase() === 'CATEGORY';
|
|
});
|
|
return !!categories.length;
|
|
}
|
|
|
|
/**
|
|
* Whether or not the category is collapsible.
|
|
* @param categoryInfo Object holing information for creating a category.
|
|
* @return True if the category has subcategories.
|
|
* @alias Blockly.utils.toolbox.isCategoryCollapsible
|
|
*/
|
|
export function isCategoryCollapsible(categoryInfo: CategoryInfo): boolean {
|
|
if (!categoryInfo || !(categoryInfo as AnyDuringMigration)['contents']) {
|
|
return false;
|
|
}
|
|
|
|
const categories =
|
|
(categoryInfo as AnyDuringMigration)['contents'].filter(function(
|
|
item: AnyDuringMigration) {
|
|
return item['kind'].toUpperCase() === 'CATEGORY';
|
|
});
|
|
return !!categories.length;
|
|
}
|
|
|
|
/**
|
|
* Parses the provided toolbox definition into a consistent format.
|
|
* @param toolboxDef The definition of the toolbox in one of its many forms.
|
|
* @return Object holding information for creating a toolbox.
|
|
*/
|
|
function convertToToolboxJson(toolboxDef: Node): ToolboxInfo {
|
|
const contents = xmlToJsonArray(toolboxDef as Node | Node[]);
|
|
const toolboxJson = {'contents': contents};
|
|
if (toolboxDef instanceof Node) {
|
|
addAttributes(toolboxDef, toolboxJson);
|
|
}
|
|
return toolboxJson;
|
|
}
|
|
|
|
/**
|
|
* Converts the xml for a toolbox to JSON.
|
|
* @param toolboxDef The definition of the toolbox in one of its many forms.
|
|
* @return A list of objects in the toolbox.
|
|
*/
|
|
function xmlToJsonArray(toolboxDef: Node|Node[]|NodeList): FlyoutItemInfoArray|
|
|
ToolboxItemInfo[] {
|
|
const arr = [];
|
|
// If it is a node it will have children.
|
|
// AnyDuringMigration because: Property 'childNodes' does not exist on type
|
|
// 'Node | NodeList | Node[]'.
|
|
let childNodes = (toolboxDef as AnyDuringMigration).childNodes;
|
|
if (!childNodes) {
|
|
// Otherwise the toolboxDef is an array or collection.
|
|
childNodes = toolboxDef;
|
|
}
|
|
for (let i = 0, child; child = childNodes[i]; i++) {
|
|
if (!child.tagName) {
|
|
continue;
|
|
}
|
|
const obj = {};
|
|
const tagName = child.tagName.toUpperCase();
|
|
(obj as AnyDuringMigration)['kind'] = tagName;
|
|
|
|
// Store the XML for a block.
|
|
if (tagName === 'BLOCK') {
|
|
(obj as AnyDuringMigration)['blockxml'] = child;
|
|
} else if (child.childNodes && child.childNodes.length > 0) {
|
|
// Get the contents of a category
|
|
(obj as AnyDuringMigration)['contents'] = xmlToJsonArray(child);
|
|
}
|
|
|
|
// Add XML attributes to object
|
|
addAttributes(child, obj);
|
|
arr.push(obj);
|
|
}
|
|
// AnyDuringMigration because: Type '{}[]' is not assignable to type
|
|
// 'ToolboxItemInfo[] | FlyoutItemInfoArray'.
|
|
return arr as AnyDuringMigration;
|
|
}
|
|
|
|
/**
|
|
* Adds the attributes on the node to the given object.
|
|
* @param node The node to copy the attributes from.
|
|
* @param obj The object to copy the attributes to.
|
|
*/
|
|
function addAttributes(node: Node, obj: AnyDuringMigration) {
|
|
// AnyDuringMigration because: Property 'attributes' does not exist on type
|
|
// 'Node'.
|
|
for (let j = 0; j < (node as AnyDuringMigration).attributes.length; j++) {
|
|
// AnyDuringMigration because: Property 'attributes' does not exist on type
|
|
// 'Node'.
|
|
const attr = (node as AnyDuringMigration).attributes[j];
|
|
if (attr.nodeName.indexOf('css-') > -1) {
|
|
obj['cssconfig'] = obj['cssconfig'] || {};
|
|
obj['cssconfig'][attr.nodeName.replace('css-', '')] = attr.value;
|
|
} else {
|
|
obj[attr.nodeName] = attr.value;
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Parse the provided toolbox tree into a consistent DOM format.
|
|
* @param toolboxDef DOM tree of blocks, or text representation of same.
|
|
* @return DOM tree of blocks, or null.
|
|
* @alias Blockly.utils.toolbox.parseToolboxTree
|
|
*/
|
|
export function parseToolboxTree(toolboxDef: Element|null|string): Element|
|
|
null {
|
|
if (toolboxDef) {
|
|
if (typeof toolboxDef !== 'string') {
|
|
if (userAgent.IE && toolboxDef.outerHTML) {
|
|
// In this case the tree will not have been properly built by the
|
|
// browser. The HTML will be contained in the element, but it will
|
|
// not have the proper DOM structure since the browser doesn't support
|
|
// XSLTProcessor (XML -> HTML).
|
|
toolboxDef = toolboxDef.outerHTML;
|
|
} else if (!(toolboxDef instanceof Element)) {
|
|
toolboxDef = null;
|
|
}
|
|
}
|
|
if (typeof toolboxDef === 'string') {
|
|
toolboxDef = Xml.textToDom(toolboxDef);
|
|
if (toolboxDef.nodeName.toLowerCase() !== 'xml') {
|
|
throw TypeError('Toolbox should be an <xml> document.');
|
|
}
|
|
}
|
|
} else {
|
|
toolboxDef = null;
|
|
}
|
|
return toolboxDef;
|
|
}
|