mirror of
https://github.com/google/blockly.git
synced 2026-01-06 08:30:13 +01:00
...and rename Blockly.blocks.all (blocks/all.js) to Blockly.libraryBlocks (blocks/blocks.js BREAKING CHANGE: (only) because the exports object from the `blocks_compressed.js` chunk will be accessed as `Blockly.libraryBlocks` instead of `Blockly.blocks.all` when the chunk is loaded in a browser via a `<script>` tag. There will be no changes visible when the chunk is loaded via ES module `import` or CJS `require`.
593 lines
16 KiB
JavaScript
593 lines
16 KiB
JavaScript
/**
|
|
* @license
|
|
* Copyright 2012 Google LLC
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
/**
|
|
* @fileoverview Math blocks for Blockly.
|
|
* @suppress {checkTypes}
|
|
*/
|
|
'use strict';
|
|
|
|
goog.module('Blockly.libraryBlocks.math');
|
|
|
|
const Extensions = goog.require('Blockly.Extensions');
|
|
// N.B.: Blockly.FieldDropdown needed for type AND side-effects.
|
|
/* eslint-disable-next-line no-unused-vars */
|
|
const FieldDropdown = goog.require('Blockly.FieldDropdown');
|
|
const xmlUtils = goog.require('Blockly.utils.xml');
|
|
/* eslint-disable-next-line no-unused-vars */
|
|
const {Block} = goog.requireType('Blockly.Block');
|
|
/* eslint-disable-next-line no-unused-vars */
|
|
const {BlockDefinition} = goog.requireType('Blockly.blocks');
|
|
const {createBlockDefinitionsFromJsonArray, defineBlocks} = goog.require('Blockly.common');
|
|
/** @suppress {extraRequire} */
|
|
goog.require('Blockly.FieldLabel');
|
|
/** @suppress {extraRequire} */
|
|
goog.require('Blockly.FieldNumber');
|
|
/** @suppress {extraRequire} */
|
|
goog.require('Blockly.FieldVariable');
|
|
|
|
|
|
/**
|
|
* A dictionary of the block definitions provided by this module.
|
|
* @type {!Object<string, !BlockDefinition>}
|
|
*/
|
|
const blocks = createBlockDefinitionsFromJsonArray([
|
|
// Block for numeric value.
|
|
{
|
|
'type': 'math_number',
|
|
'message0': '%1',
|
|
'args0': [{
|
|
'type': 'field_number',
|
|
'name': 'NUM',
|
|
'value': 0,
|
|
}],
|
|
'output': 'Number',
|
|
'helpUrl': '%{BKY_MATH_NUMBER_HELPURL}',
|
|
'style': 'math_blocks',
|
|
'tooltip': '%{BKY_MATH_NUMBER_TOOLTIP}',
|
|
'extensions': ['parent_tooltip_when_inline'],
|
|
},
|
|
|
|
// Block for basic arithmetic operator.
|
|
{
|
|
'type': 'math_arithmetic',
|
|
'message0': '%1 %2 %3',
|
|
'args0': [
|
|
{
|
|
'type': 'input_value',
|
|
'name': 'A',
|
|
'check': 'Number',
|
|
},
|
|
{
|
|
'type': 'field_dropdown',
|
|
'name': 'OP',
|
|
'options': [
|
|
['%{BKY_MATH_ADDITION_SYMBOL}', 'ADD'],
|
|
['%{BKY_MATH_SUBTRACTION_SYMBOL}', 'MINUS'],
|
|
['%{BKY_MATH_MULTIPLICATION_SYMBOL}', 'MULTIPLY'],
|
|
['%{BKY_MATH_DIVISION_SYMBOL}', 'DIVIDE'],
|
|
['%{BKY_MATH_POWER_SYMBOL}', 'POWER'],
|
|
],
|
|
},
|
|
{
|
|
'type': 'input_value',
|
|
'name': 'B',
|
|
'check': 'Number',
|
|
},
|
|
],
|
|
'inputsInline': true,
|
|
'output': 'Number',
|
|
'style': 'math_blocks',
|
|
'helpUrl': '%{BKY_MATH_ARITHMETIC_HELPURL}',
|
|
'extensions': ['math_op_tooltip'],
|
|
},
|
|
|
|
// Block for advanced math operators with single operand.
|
|
{
|
|
'type': 'math_single',
|
|
'message0': '%1 %2',
|
|
'args0': [
|
|
{
|
|
'type': 'field_dropdown',
|
|
'name': 'OP',
|
|
'options': [
|
|
['%{BKY_MATH_SINGLE_OP_ROOT}', 'ROOT'],
|
|
['%{BKY_MATH_SINGLE_OP_ABSOLUTE}', 'ABS'],
|
|
['-', 'NEG'],
|
|
['ln', 'LN'],
|
|
['log10', 'LOG10'],
|
|
['e^', 'EXP'],
|
|
['10^', 'POW10'],
|
|
],
|
|
},
|
|
{
|
|
'type': 'input_value',
|
|
'name': 'NUM',
|
|
'check': 'Number',
|
|
},
|
|
],
|
|
'output': 'Number',
|
|
'style': 'math_blocks',
|
|
'helpUrl': '%{BKY_MATH_SINGLE_HELPURL}',
|
|
'extensions': ['math_op_tooltip'],
|
|
},
|
|
|
|
// Block for trigonometry operators.
|
|
{
|
|
'type': 'math_trig',
|
|
'message0': '%1 %2',
|
|
'args0': [
|
|
{
|
|
'type': 'field_dropdown',
|
|
'name': 'OP',
|
|
'options': [
|
|
['%{BKY_MATH_TRIG_SIN}', 'SIN'],
|
|
['%{BKY_MATH_TRIG_COS}', 'COS'],
|
|
['%{BKY_MATH_TRIG_TAN}', 'TAN'],
|
|
['%{BKY_MATH_TRIG_ASIN}', 'ASIN'],
|
|
['%{BKY_MATH_TRIG_ACOS}', 'ACOS'],
|
|
['%{BKY_MATH_TRIG_ATAN}', 'ATAN'],
|
|
],
|
|
},
|
|
{
|
|
'type': 'input_value',
|
|
'name': 'NUM',
|
|
'check': 'Number',
|
|
},
|
|
],
|
|
'output': 'Number',
|
|
'style': 'math_blocks',
|
|
'helpUrl': '%{BKY_MATH_TRIG_HELPURL}',
|
|
'extensions': ['math_op_tooltip'],
|
|
},
|
|
|
|
// Block for constants: PI, E, the Golden Ratio, sqrt(2), 1/sqrt(2), INFINITY.
|
|
{
|
|
'type': 'math_constant',
|
|
'message0': '%1',
|
|
'args0': [
|
|
{
|
|
'type': 'field_dropdown',
|
|
'name': 'CONSTANT',
|
|
'options': [
|
|
['\u03c0', 'PI'],
|
|
['e', 'E'],
|
|
['\u03c6', 'GOLDEN_RATIO'],
|
|
['sqrt(2)', 'SQRT2'],
|
|
['sqrt(\u00bd)', 'SQRT1_2'],
|
|
['\u221e', 'INFINITY'],
|
|
],
|
|
},
|
|
],
|
|
'output': 'Number',
|
|
'style': 'math_blocks',
|
|
'tooltip': '%{BKY_MATH_CONSTANT_TOOLTIP}',
|
|
'helpUrl': '%{BKY_MATH_CONSTANT_HELPURL}',
|
|
},
|
|
|
|
// Block for checking if a number is even, odd, prime, whole, positive,
|
|
// negative or if it is divisible by certain number.
|
|
{
|
|
'type': 'math_number_property',
|
|
'message0': '%1 %2',
|
|
'args0': [
|
|
{
|
|
'type': 'input_value',
|
|
'name': 'NUMBER_TO_CHECK',
|
|
'check': 'Number',
|
|
},
|
|
{
|
|
'type': 'field_dropdown',
|
|
'name': 'PROPERTY',
|
|
'options': [
|
|
['%{BKY_MATH_IS_EVEN}', 'EVEN'],
|
|
['%{BKY_MATH_IS_ODD}', 'ODD'],
|
|
['%{BKY_MATH_IS_PRIME}', 'PRIME'],
|
|
['%{BKY_MATH_IS_WHOLE}', 'WHOLE'],
|
|
['%{BKY_MATH_IS_POSITIVE}', 'POSITIVE'],
|
|
['%{BKY_MATH_IS_NEGATIVE}', 'NEGATIVE'],
|
|
['%{BKY_MATH_IS_DIVISIBLE_BY}', 'DIVISIBLE_BY'],
|
|
],
|
|
},
|
|
],
|
|
'inputsInline': true,
|
|
'output': 'Boolean',
|
|
'style': 'math_blocks',
|
|
'tooltip': '%{BKY_MATH_IS_TOOLTIP}',
|
|
'mutator': 'math_is_divisibleby_mutator',
|
|
},
|
|
|
|
// Block for adding to a variable in place.
|
|
{
|
|
'type': 'math_change',
|
|
'message0': '%{BKY_MATH_CHANGE_TITLE}',
|
|
'args0': [
|
|
{
|
|
'type': 'field_variable',
|
|
'name': 'VAR',
|
|
'variable': '%{BKY_MATH_CHANGE_TITLE_ITEM}',
|
|
},
|
|
{
|
|
'type': 'input_value',
|
|
'name': 'DELTA',
|
|
'check': 'Number',
|
|
},
|
|
],
|
|
'previousStatement': null,
|
|
'nextStatement': null,
|
|
'style': 'variable_blocks',
|
|
'helpUrl': '%{BKY_MATH_CHANGE_HELPURL}',
|
|
'extensions': ['math_change_tooltip'],
|
|
},
|
|
|
|
// Block for rounding functions.
|
|
{
|
|
'type': 'math_round',
|
|
'message0': '%1 %2',
|
|
'args0': [
|
|
{
|
|
'type': 'field_dropdown',
|
|
'name': 'OP',
|
|
'options': [
|
|
['%{BKY_MATH_ROUND_OPERATOR_ROUND}', 'ROUND'],
|
|
['%{BKY_MATH_ROUND_OPERATOR_ROUNDUP}', 'ROUNDUP'],
|
|
['%{BKY_MATH_ROUND_OPERATOR_ROUNDDOWN}', 'ROUNDDOWN'],
|
|
],
|
|
},
|
|
{
|
|
'type': 'input_value',
|
|
'name': 'NUM',
|
|
'check': 'Number',
|
|
},
|
|
],
|
|
'output': 'Number',
|
|
'style': 'math_blocks',
|
|
'helpUrl': '%{BKY_MATH_ROUND_HELPURL}',
|
|
'tooltip': '%{BKY_MATH_ROUND_TOOLTIP}',
|
|
},
|
|
|
|
// Block for evaluating a list of numbers to return sum, average, min, max,
|
|
// etc. Some functions also work on text (min, max, mode, median).
|
|
{
|
|
'type': 'math_on_list',
|
|
'message0': '%1 %2',
|
|
'args0': [
|
|
{
|
|
'type': 'field_dropdown',
|
|
'name': 'OP',
|
|
'options': [
|
|
['%{BKY_MATH_ONLIST_OPERATOR_SUM}', 'SUM'],
|
|
['%{BKY_MATH_ONLIST_OPERATOR_MIN}', 'MIN'],
|
|
['%{BKY_MATH_ONLIST_OPERATOR_MAX}', 'MAX'],
|
|
['%{BKY_MATH_ONLIST_OPERATOR_AVERAGE}', 'AVERAGE'],
|
|
['%{BKY_MATH_ONLIST_OPERATOR_MEDIAN}', 'MEDIAN'],
|
|
['%{BKY_MATH_ONLIST_OPERATOR_MODE}', 'MODE'],
|
|
['%{BKY_MATH_ONLIST_OPERATOR_STD_DEV}', 'STD_DEV'],
|
|
['%{BKY_MATH_ONLIST_OPERATOR_RANDOM}', 'RANDOM'],
|
|
],
|
|
},
|
|
{
|
|
'type': 'input_value',
|
|
'name': 'LIST',
|
|
'check': 'Array',
|
|
},
|
|
],
|
|
'output': 'Number',
|
|
'style': 'math_blocks',
|
|
'helpUrl': '%{BKY_MATH_ONLIST_HELPURL}',
|
|
'mutator': 'math_modes_of_list_mutator',
|
|
'extensions': ['math_op_tooltip'],
|
|
},
|
|
|
|
// Block for remainder of a division.
|
|
{
|
|
'type': 'math_modulo',
|
|
'message0': '%{BKY_MATH_MODULO_TITLE}',
|
|
'args0': [
|
|
{
|
|
'type': 'input_value',
|
|
'name': 'DIVIDEND',
|
|
'check': 'Number',
|
|
},
|
|
{
|
|
'type': 'input_value',
|
|
'name': 'DIVISOR',
|
|
'check': 'Number',
|
|
},
|
|
],
|
|
'inputsInline': true,
|
|
'output': 'Number',
|
|
'style': 'math_blocks',
|
|
'tooltip': '%{BKY_MATH_MODULO_TOOLTIP}',
|
|
'helpUrl': '%{BKY_MATH_MODULO_HELPURL}',
|
|
},
|
|
|
|
// Block for constraining a number between two limits.
|
|
{
|
|
'type': 'math_constrain',
|
|
'message0': '%{BKY_MATH_CONSTRAIN_TITLE}',
|
|
'args0': [
|
|
{
|
|
'type': 'input_value',
|
|
'name': 'VALUE',
|
|
'check': 'Number',
|
|
},
|
|
{
|
|
'type': 'input_value',
|
|
'name': 'LOW',
|
|
'check': 'Number',
|
|
},
|
|
{
|
|
'type': 'input_value',
|
|
'name': 'HIGH',
|
|
'check': 'Number',
|
|
},
|
|
],
|
|
'inputsInline': true,
|
|
'output': 'Number',
|
|
'style': 'math_blocks',
|
|
'tooltip': '%{BKY_MATH_CONSTRAIN_TOOLTIP}',
|
|
'helpUrl': '%{BKY_MATH_CONSTRAIN_HELPURL}',
|
|
},
|
|
|
|
// Block for random integer between [X] and [Y].
|
|
{
|
|
'type': 'math_random_int',
|
|
'message0': '%{BKY_MATH_RANDOM_INT_TITLE}',
|
|
'args0': [
|
|
{
|
|
'type': 'input_value',
|
|
'name': 'FROM',
|
|
'check': 'Number',
|
|
},
|
|
{
|
|
'type': 'input_value',
|
|
'name': 'TO',
|
|
'check': 'Number',
|
|
},
|
|
],
|
|
'inputsInline': true,
|
|
'output': 'Number',
|
|
'style': 'math_blocks',
|
|
'tooltip': '%{BKY_MATH_RANDOM_INT_TOOLTIP}',
|
|
'helpUrl': '%{BKY_MATH_RANDOM_INT_HELPURL}',
|
|
},
|
|
|
|
// Block for random integer between [X] and [Y].
|
|
{
|
|
'type': 'math_random_float',
|
|
'message0': '%{BKY_MATH_RANDOM_FLOAT_TITLE_RANDOM}',
|
|
'output': 'Number',
|
|
'style': 'math_blocks',
|
|
'tooltip': '%{BKY_MATH_RANDOM_FLOAT_TOOLTIP}',
|
|
'helpUrl': '%{BKY_MATH_RANDOM_FLOAT_HELPURL}',
|
|
},
|
|
|
|
// Block for calculating atan2 of [X] and [Y].
|
|
{
|
|
'type': 'math_atan2',
|
|
'message0': '%{BKY_MATH_ATAN2_TITLE}',
|
|
'args0': [
|
|
{
|
|
'type': 'input_value',
|
|
'name': 'X',
|
|
'check': 'Number',
|
|
},
|
|
{
|
|
'type': 'input_value',
|
|
'name': 'Y',
|
|
'check': 'Number',
|
|
},
|
|
],
|
|
'inputsInline': true,
|
|
'output': 'Number',
|
|
'style': 'math_blocks',
|
|
'tooltip': '%{BKY_MATH_ATAN2_TOOLTIP}',
|
|
'helpUrl': '%{BKY_MATH_ATAN2_HELPURL}',
|
|
},
|
|
]);
|
|
exports.blocks = blocks;
|
|
|
|
/**
|
|
* Mapping of math block OP value to tooltip message for blocks
|
|
* math_arithmetic, math_simple, math_trig, and math_on_lists.
|
|
* @see {Extensions#buildTooltipForDropdown}
|
|
* @package
|
|
* @readonly
|
|
*/
|
|
const TOOLTIPS_BY_OP = {
|
|
// math_arithmetic
|
|
'ADD': '%{BKY_MATH_ARITHMETIC_TOOLTIP_ADD}',
|
|
'MINUS': '%{BKY_MATH_ARITHMETIC_TOOLTIP_MINUS}',
|
|
'MULTIPLY': '%{BKY_MATH_ARITHMETIC_TOOLTIP_MULTIPLY}',
|
|
'DIVIDE': '%{BKY_MATH_ARITHMETIC_TOOLTIP_DIVIDE}',
|
|
'POWER': '%{BKY_MATH_ARITHMETIC_TOOLTIP_POWER}',
|
|
|
|
// math_simple
|
|
'ROOT': '%{BKY_MATH_SINGLE_TOOLTIP_ROOT}',
|
|
'ABS': '%{BKY_MATH_SINGLE_TOOLTIP_ABS}',
|
|
'NEG': '%{BKY_MATH_SINGLE_TOOLTIP_NEG}',
|
|
'LN': '%{BKY_MATH_SINGLE_TOOLTIP_LN}',
|
|
'LOG10': '%{BKY_MATH_SINGLE_TOOLTIP_LOG10}',
|
|
'EXP': '%{BKY_MATH_SINGLE_TOOLTIP_EXP}',
|
|
'POW10': '%{BKY_MATH_SINGLE_TOOLTIP_POW10}',
|
|
|
|
// math_trig
|
|
'SIN': '%{BKY_MATH_TRIG_TOOLTIP_SIN}',
|
|
'COS': '%{BKY_MATH_TRIG_TOOLTIP_COS}',
|
|
'TAN': '%{BKY_MATH_TRIG_TOOLTIP_TAN}',
|
|
'ASIN': '%{BKY_MATH_TRIG_TOOLTIP_ASIN}',
|
|
'ACOS': '%{BKY_MATH_TRIG_TOOLTIP_ACOS}',
|
|
'ATAN': '%{BKY_MATH_TRIG_TOOLTIP_ATAN}',
|
|
|
|
// math_on_lists
|
|
'SUM': '%{BKY_MATH_ONLIST_TOOLTIP_SUM}',
|
|
'MIN': '%{BKY_MATH_ONLIST_TOOLTIP_MIN}',
|
|
'MAX': '%{BKY_MATH_ONLIST_TOOLTIP_MAX}',
|
|
'AVERAGE': '%{BKY_MATH_ONLIST_TOOLTIP_AVERAGE}',
|
|
'MEDIAN': '%{BKY_MATH_ONLIST_TOOLTIP_MEDIAN}',
|
|
'MODE': '%{BKY_MATH_ONLIST_TOOLTIP_MODE}',
|
|
'STD_DEV': '%{BKY_MATH_ONLIST_TOOLTIP_STD_DEV}',
|
|
'RANDOM': '%{BKY_MATH_ONLIST_TOOLTIP_RANDOM}',
|
|
};
|
|
|
|
Extensions.register(
|
|
'math_op_tooltip',
|
|
Extensions.buildTooltipForDropdown('OP', TOOLTIPS_BY_OP));
|
|
|
|
|
|
/**
|
|
* Mixin for mutator functions in the 'math_is_divisibleby_mutator'
|
|
* extension.
|
|
* @mixin
|
|
* @augments Block
|
|
* @package
|
|
*/
|
|
const IS_DIVISIBLEBY_MUTATOR_MIXIN = {
|
|
/**
|
|
* Create XML to represent whether the 'divisorInput' should be present.
|
|
* Backwards compatible serialization implementation.
|
|
* @return {!Element} XML storage element.
|
|
* @this {Block}
|
|
*/
|
|
mutationToDom: function() {
|
|
const container = xmlUtils.createElement('mutation');
|
|
const divisorInput = (this.getFieldValue('PROPERTY') === 'DIVISIBLE_BY');
|
|
container.setAttribute('divisor_input', divisorInput);
|
|
return container;
|
|
},
|
|
/**
|
|
* Parse XML to restore the 'divisorInput'.
|
|
* Backwards compatible serialization implementation.
|
|
* @param {!Element} xmlElement XML storage element.
|
|
* @this {Block}
|
|
*/
|
|
domToMutation: function(xmlElement) {
|
|
const divisorInput = (xmlElement.getAttribute('divisor_input') === 'true');
|
|
this.updateShape_(divisorInput);
|
|
},
|
|
|
|
// This block does not need JSO serialization hooks (saveExtraState and
|
|
// loadExtraState) because the state of this object is already encoded in the
|
|
// dropdown values.
|
|
// XML hooks are kept for backwards compatibility.
|
|
|
|
/**
|
|
* Modify this block to have (or not have) an input for 'is divisible by'.
|
|
* @param {boolean} divisorInput True if this block has a divisor input.
|
|
* @private
|
|
* @this {Block}
|
|
*/
|
|
updateShape_: function(divisorInput) {
|
|
// Add or remove a Value Input.
|
|
const inputExists = this.getInput('DIVISOR');
|
|
if (divisorInput) {
|
|
if (!inputExists) {
|
|
this.appendValueInput('DIVISOR').setCheck('Number');
|
|
}
|
|
} else if (inputExists) {
|
|
this.removeInput('DIVISOR');
|
|
}
|
|
},
|
|
};
|
|
|
|
/**
|
|
* 'math_is_divisibleby_mutator' extension to the 'math_property' block that
|
|
* can update the block shape (add/remove divisor input) based on whether
|
|
* property is "divisible by".
|
|
* @this {Block}
|
|
* @package
|
|
*/
|
|
const IS_DIVISIBLE_MUTATOR_EXTENSION = function() {
|
|
this.getField('PROPERTY')
|
|
.setValidator(
|
|
/**
|
|
* @this {FieldDropdown}
|
|
* @param {*} option The selected dropdown option.
|
|
*/
|
|
function(option) {
|
|
const divisorInput = (option === 'DIVISIBLE_BY');
|
|
this.getSourceBlock().updateShape_(divisorInput);
|
|
});
|
|
};
|
|
|
|
Extensions.registerMutator(
|
|
'math_is_divisibleby_mutator', IS_DIVISIBLEBY_MUTATOR_MIXIN,
|
|
IS_DIVISIBLE_MUTATOR_EXTENSION);
|
|
|
|
// Update the tooltip of 'math_change' block to reference the variable.
|
|
Extensions.register(
|
|
'math_change_tooltip',
|
|
Extensions.buildTooltipWithFieldText('%{BKY_MATH_CHANGE_TOOLTIP}', 'VAR'));
|
|
|
|
/**
|
|
* Mixin with mutator methods to support alternate output based if the
|
|
* 'math_on_list' block uses the 'MODE' operation.
|
|
* @mixin
|
|
* @augments Block
|
|
* @package
|
|
* @readonly
|
|
*/
|
|
const LIST_MODES_MUTATOR_MIXIN = {
|
|
/**
|
|
* Modify this block to have the correct output type.
|
|
* @param {string} newOp Either 'MODE' or some op than returns a number.
|
|
* @private
|
|
* @this {Block}
|
|
*/
|
|
updateType_: function(newOp) {
|
|
if (newOp === 'MODE') {
|
|
this.outputConnection.setCheck('Array');
|
|
} else {
|
|
this.outputConnection.setCheck('Number');
|
|
}
|
|
},
|
|
/**
|
|
* Create XML to represent the output type.
|
|
* Backwards compatible serialization implementation.
|
|
* @return {!Element} XML storage element.
|
|
* @this {Block}
|
|
*/
|
|
mutationToDom: function() {
|
|
const container = xmlUtils.createElement('mutation');
|
|
container.setAttribute('op', this.getFieldValue('OP'));
|
|
return container;
|
|
},
|
|
/**
|
|
* Parse XML to restore the output type.
|
|
* Backwards compatible serialization implementation.
|
|
* @param {!Element} xmlElement XML storage element.
|
|
* @this {Block}
|
|
*/
|
|
domToMutation: function(xmlElement) {
|
|
this.updateType_(xmlElement.getAttribute('op'));
|
|
},
|
|
|
|
// This block does not need JSO serialization hooks (saveExtraState and
|
|
// loadExtraState) because the state of this object is already encoded in the
|
|
// dropdown values.
|
|
// XML hooks are kept for backwards compatibility.
|
|
};
|
|
|
|
/**
|
|
* Extension to 'math_on_list' blocks that allows support of
|
|
* modes operation (outputs a list of numbers).
|
|
* @this {Block}
|
|
* @package
|
|
*/
|
|
const LIST_MODES_MUTATOR_EXTENSION = function() {
|
|
this.getField('OP').setValidator(function(newOp) {
|
|
this.updateType_(newOp);
|
|
}.bind(this));
|
|
};
|
|
|
|
Extensions.registerMutator(
|
|
'math_modes_of_list_mutator', LIST_MODES_MUTATOR_MIXIN,
|
|
LIST_MODES_MUTATOR_EXTENSION);
|
|
|
|
// Register provided blocks.
|
|
defineBlocks(blocks);
|