diff --git a/plugins/field-bitmap/README.html b/plugins/field-bitmap/README.html
index a680ee5ccb..69ab698a15 100644
--- a/plugins/field-bitmap/README.html
+++ b/plugins/field-bitmap/README.html
@@ -24,7 +24,7 @@
@blockly/field-bitmap Demo
A field that lets users input a pixel grid with their mouse.
- 4.1.0
+ 4.1.1
View code
View on npm
diff --git a/plugins/field-bitmap/build/test_bundle.js b/plugins/field-bitmap/build/test_bundle.js
index e9fa9c6e8d..3ca2f11387 100644
--- a/plugins/field-bitmap/build/test_bundle.js
+++ b/plugins/field-bitmap/build/test_bundle.js
@@ -437,7 +437,7 @@ eval("var __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_ARRAY__, __WEBPAC
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
"use strict";
-eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"DEFAULT_HEIGHT\": () => (/* binding */ DEFAULT_HEIGHT),\n/* harmony export */ \"DEFAULT_WIDTH\": () => (/* binding */ DEFAULT_WIDTH),\n/* harmony export */ \"FieldBitmap\": () => (/* binding */ FieldBitmap)\n/* harmony export */ });\n/* harmony import */ var blockly_core__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! blockly/core */ \"./node_modules/blockly/core-browser.js\");\n/* harmony import */ var blockly_core__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(blockly_core__WEBPACK_IMPORTED_MODULE_0__);\n/**\n * @license\n * Copyright 2021 Google LLC\n * SPDX-License-Identifier: Apache-2.0\n */\n\nblockly_core__WEBPACK_IMPORTED_MODULE_0__.Msg.BUTTON_LABEL_RANDOMIZE = 'Randomize';\nblockly_core__WEBPACK_IMPORTED_MODULE_0__.Msg.BUTTON_LABEL_CLEAR = 'Clear';\nconst DEFAULT_HEIGHT = 5;\nconst DEFAULT_WIDTH = 5;\nconst DEFAULT_PIXEL_SIZE = 15;\nconst DEFAULT_PIXEL_COLOURS = {\n empty: '#fff',\n filled: '#363d80',\n};\nconst DEFAULT_BUTTONS = {\n randomize: true,\n clear: true,\n};\n/**\n * Field for inputting a small bitmap image.\n * Includes a grid of clickable pixels that's exported as a bitmap.\n */\nclass FieldBitmap extends blockly_core__WEBPACK_IMPORTED_MODULE_0__.Field {\n /**\n * Constructor for the bitmap field.\n *\n * @param value 2D rectangular array of 1s and 0s.\n * @param validator A function that is called to validate.\n * @param config Config A map of options used to configure the field.\n */\n constructor(value, validator, config) {\n var _a, _b;\n super(value, validator, config);\n this.initialValue = null;\n /**\n * Array holding info needed to unbind events.\n * Used for disposing.\n */\n this.boundEvents = [];\n /** References to UI elements */\n this.editorPixels = null;\n this.blockDisplayPixels = null;\n /** Stateful variables */\n this.mouseIsDown = false;\n this.SERIALIZABLE = true;\n this.CURSOR = 'default';\n this.buttonOptions = Object.assign(Object.assign({}, DEFAULT_BUTTONS), config === null || config === void 0 ? void 0 : config.buttons);\n this.pixelColours = Object.assign(Object.assign({}, DEFAULT_PIXEL_COLOURS), config === null || config === void 0 ? void 0 : config.colours);\n // Configure value, height, and width\n const currentValue = this.getValue();\n if (currentValue !== null) {\n this.imgHeight = currentValue.length;\n this.imgWidth = currentValue[0].length || 0;\n }\n else {\n this.imgHeight = (_a = config === null || config === void 0 ? void 0 : config.height) !== null && _a !== void 0 ? _a : DEFAULT_HEIGHT;\n this.imgWidth = (_b = config === null || config === void 0 ? void 0 : config.width) !== null && _b !== void 0 ? _b : DEFAULT_WIDTH;\n // Set a default empty value\n this.setValue(this.getEmptyArray());\n }\n if (config === null || config === void 0 ? void 0 : config.fieldHeight) {\n this.pixelSize = config.fieldHeight / this.imgHeight;\n }\n else {\n this.pixelSize = DEFAULT_PIXEL_SIZE;\n }\n }\n /**\n * Constructs a FieldBitmap from a JSON arg object.\n *\n * @param options A JSON object with options.\n * @returns The new field instance.\n */\n static fromJson(options) {\n var _a;\n // `this` might be a subclass of FieldBitmap if that class doesn't override the static fromJson method.\n return new this((_a = options.value) !== null && _a !== void 0 ? _a : blockly_core__WEBPACK_IMPORTED_MODULE_0__.Field.SKIP_SETUP, undefined, options);\n }\n /**\n * Returns the width of the image in pixels.\n *\n * @returns The width in pixels.\n */\n getImageWidth() {\n return this.imgWidth;\n }\n /**\n * Returns the height of the image in pixels.\n *\n * @returns The height in pixels.\n */\n getImageHeight() {\n return this.imgHeight;\n }\n /**\n * Validates that a new value meets the requirements for a valid bitmap array.\n *\n * @param newValue The new value to be tested.\n * @returns The new value if it's valid, or null.\n */\n // eslint-disable-next-line\n doClassValidation_(newValue = undefined) {\n if (!newValue) {\n return null;\n }\n // Check if the new value is an array\n if (!Array.isArray(newValue)) {\n return null;\n }\n const newHeight = newValue.length;\n // The empty list is not an acceptable bitmap\n if (newHeight == 0) {\n return null;\n }\n // Check that the width matches the existing width of the image if it\n // already has a value.\n const newWidth = newValue[0].length;\n for (const row of newValue) {\n if (!Array.isArray(row)) {\n return null;\n }\n if (row.length !== newWidth) {\n return null;\n }\n }\n // Check if all contents of the arrays are either 0 or 1\n for (const row of newValue) {\n for (const cell of row) {\n if (cell !== 0 && cell !== 1) {\n return null;\n }\n }\n }\n return newValue;\n }\n /**\n * Called when a new value has been validated and is about to be set.\n *\n * @param newValue The value that's about to be set.\n */\n // eslint-disable-next-line\n doValueUpdate_(newValue) {\n super.doValueUpdate_(newValue);\n if (newValue) {\n this.imgHeight = newValue.length;\n this.imgWidth = newValue[0] ? newValue[0].length : 0;\n }\n }\n /**\n * Show the bitmap editor dialog.\n *\n * @param e Optional mouse event that triggered the field to open, or\n * undefined if triggered programmatically.\n */\n // eslint-disable-next-line\n showEditor_(e) {\n const editor = this.dropdownCreate();\n blockly_core__WEBPACK_IMPORTED_MODULE_0__.DropDownDiv.getContentDiv().appendChild(editor);\n blockly_core__WEBPACK_IMPORTED_MODULE_0__.DropDownDiv.showPositionedByField(this, this.dropdownDispose.bind(this));\n }\n /**\n * Updates the block display and editor dropdown when the field re-renders.\n */\n // eslint-disable-next-line\n render_() {\n super.render_();\n if (!this.getValue()) {\n return;\n }\n if (this.blockDisplayPixels) {\n this.forAllCells((r, c) => {\n const pixel = this.getPixel(r, c);\n if (this.blockDisplayPixels) {\n this.blockDisplayPixels[r][c].style.fill = pixel\n ? this.pixelColours.filled\n : this.pixelColours.empty;\n }\n if (this.editorPixels) {\n this.editorPixels[r][c].style.background = pixel\n ? this.pixelColours.filled\n : this.pixelColours.empty;\n }\n });\n }\n }\n /**\n * Determines whether the field is editable.\n *\n * @returns True since it is always editable.\n */\n updateEditable() {\n const editable = super.updateEditable();\n // Blockly.Field's implementation sets these classes as appropriate, but\n // since this field has no text they just mess up the rendering of the grid\n // lines.\n const svgRoot = this.getSvgRoot();\n if (svgRoot) {\n blockly_core__WEBPACK_IMPORTED_MODULE_0__.utils.dom.removeClass(svgRoot, 'blocklyNonEditableText');\n blockly_core__WEBPACK_IMPORTED_MODULE_0__.utils.dom.removeClass(svgRoot, 'blocklyEditableText');\n }\n return editable;\n }\n /**\n * Gets the rectangle built out of dimensions matching SVG's element.\n *\n * @returns The newly created rectangle of same size as the SVG element.\n */\n getScaledBBox() {\n var _a;\n const boundingBox = (_a = this.getSvgRoot()) === null || _a === void 0 ? void 0 : _a.getBoundingClientRect();\n if (!boundingBox) {\n throw new Error('Tried to retrieve a bounding box without a rect');\n }\n return new blockly_core__WEBPACK_IMPORTED_MODULE_0__.utils.Rect(boundingBox.top, boundingBox.bottom, boundingBox.left, boundingBox.right);\n }\n /**\n * Creates the bitmap editor and add event listeners.\n *\n * @returns The newly created dropdown menu.\n */\n dropdownCreate() {\n const dropdownEditor = this.createElementWithClassname('div', 'dropdownEditor');\n const pixelContainer = this.createElementWithClassname('div', 'pixelContainer');\n dropdownEditor.appendChild(pixelContainer);\n this.bindEvent(dropdownEditor, 'mouseup', this.onMouseUp);\n this.bindEvent(dropdownEditor, 'mouseleave', this.onMouseUp);\n this.bindEvent(dropdownEditor, 'dragstart', (e) => {\n e.preventDefault();\n });\n this.editorPixels = [];\n for (let r = 0; r < this.imgHeight; r++) {\n this.editorPixels.push([]);\n const rowDiv = this.createElementWithClassname('div', 'pixelRow');\n for (let c = 0; c < this.imgWidth; c++) {\n // Add the button to the UI and save a reference to it\n const button = this.createElementWithClassname('div', 'pixelButton');\n this.editorPixels[r].push(button);\n rowDiv.appendChild(button);\n // Load the current pixel colour\n const isOn = this.getPixel(r, c);\n button.style.background = isOn\n ? this.pixelColours.filled\n : this.pixelColours.empty;\n // Handle clicking a pixel\n this.bindEvent(button, 'mousedown', () => {\n this.onMouseDownInPixel(r, c);\n return true;\n });\n // Handle dragging into a pixel when mouse is down\n this.bindEvent(button, 'mouseenter', () => {\n this.onMouseEnterPixel(r, c);\n });\n }\n pixelContainer.appendChild(rowDiv);\n }\n // Add control buttons below the pixel grid\n if (this.buttonOptions.randomize) {\n this.addControlButton(dropdownEditor, blockly_core__WEBPACK_IMPORTED_MODULE_0__.Msg.BUTTON_LABEL_RANDOMIZE, this.randomizePixels);\n }\n if (this.buttonOptions.clear) {\n this.addControlButton(dropdownEditor, blockly_core__WEBPACK_IMPORTED_MODULE_0__.Msg.BUTTON_LABEL_CLEAR, this.clearPixels);\n }\n if (this.blockDisplayPixels) {\n this.forAllCells((r, c) => {\n const pixel = this.getPixel(r, c);\n if (this.editorPixels) {\n this.editorPixels[r][c].style.background = pixel\n ? this.pixelColours.filled\n : this.pixelColours.empty;\n }\n });\n }\n // Store the initial value at the start of the edit.\n this.initialValue = this.getValue();\n return dropdownEditor;\n }\n /**\n * Initializes the on-block display.\n */\n initView() {\n this.blockDisplayPixels = [];\n for (let r = 0; r < this.imgHeight; r++) {\n const row = [];\n for (let c = 0; c < this.imgWidth; c++) {\n const square = blockly_core__WEBPACK_IMPORTED_MODULE_0__.utils.dom.createSvgElement('rect', {\n x: c * this.pixelSize,\n y: r * this.pixelSize,\n width: this.pixelSize,\n height: this.pixelSize,\n fill: this.pixelColours.empty,\n fill_opacity: 1, // eslint-disable-line\n }, this.getSvgRoot());\n row.push(square);\n }\n this.blockDisplayPixels.push(row);\n }\n }\n /**\n * Updates the size of the block based on the size of the underlying image.\n */\n // eslint-disable-next-line\n updateSize_() {\n {\n const newWidth = this.pixelSize * this.imgWidth;\n const newHeight = this.pixelSize * this.imgHeight;\n if (this.borderRect_) {\n this.borderRect_.setAttribute('width', String(newWidth));\n this.borderRect_.setAttribute('height', String(newHeight));\n }\n this.size_.width = newWidth;\n this.size_.height = newHeight;\n }\n }\n /**\n * Create control button.\n *\n * @param parent Parent HTML element to which control button will be added.\n * @param buttonText Text of the control button.\n * @param onClick Callback that will be attached to the control button.\n */\n addControlButton(parent, buttonText, onClick) {\n const button = this.createElementWithClassname('button', 'controlButton');\n button.innerText = buttonText;\n parent.appendChild(button);\n this.bindEvent(button, 'click', onClick);\n }\n /**\n * Disposes of events belonging to the bitmap editor.\n */\n dropdownDispose() {\n if (this.getSourceBlock() &&\n this.initialValue !== null &&\n this.initialValue !== this.getValue()) {\n blockly_core__WEBPACK_IMPORTED_MODULE_0__.Events.fire(new (blockly_core__WEBPACK_IMPORTED_MODULE_0__.Events.get(blockly_core__WEBPACK_IMPORTED_MODULE_0__.Events.BLOCK_CHANGE))(this.sourceBlock_, 'field', this.name || null, this.initialValue, this.getValue()));\n }\n for (const event of this.boundEvents) {\n blockly_core__WEBPACK_IMPORTED_MODULE_0__.browserEvents.unbind(event);\n }\n this.boundEvents.length = 0;\n this.editorPixels = null;\n // Set this.initialValue back to null.\n this.initialValue = null;\n }\n /**\n * Constructs an array of zeros with the specified width and height.\n *\n * @returns The new value.\n */\n getEmptyArray() {\n const newVal = [];\n for (let r = 0; r < this.imgHeight; r++) {\n newVal.push([]);\n for (let c = 0; c < this.imgWidth; c++) {\n newVal[r].push(0);\n }\n }\n return newVal;\n }\n /**\n * Called when a mousedown event occurs within the bounds of a pixel.\n *\n * @param r Row number of grid.\n * @param c Column number of grid.\n */\n onMouseDownInPixel(r, c) {\n // Toggle that pixel to the opposite of its value\n const newPixelValue = 1 - this.getPixel(r, c);\n this.setPixel(r, c, newPixelValue);\n this.mouseIsDown = true;\n this.valToPaintWith = newPixelValue;\n }\n /**\n * Called when the mouse drags over a pixel in the editor.\n *\n * @param r Row number of grid.\n * @param c Column number of grid.\n */\n onMouseEnterPixel(r, c) {\n if (!this.mouseIsDown) {\n return;\n }\n if (this.valToPaintWith !== undefined &&\n this.getPixel(r, c) !== this.valToPaintWith) {\n this.setPixel(r, c, this.valToPaintWith);\n }\n }\n /**\n * Resets mouse state (e.g. After either a mouseup event or if the mouse\n * leaves the editor area).\n */\n onMouseUp() {\n this.mouseIsDown = false;\n this.valToPaintWith = undefined;\n }\n /**\n * Sets all the pixels in the image to a random value.\n */\n randomizePixels() {\n const getRandBinary = () => Math.floor(Math.random() * 2);\n this.forAllCells((r, c) => {\n this.setPixel(r, c, getRandBinary());\n });\n }\n /**\n * Sets all the pixels to 0.\n */\n clearPixels() {\n const cleared = this.getEmptyArray();\n this.fireIntermediateChangeEvent(cleared);\n this.setValue(cleared, false);\n }\n /**\n * Sets the value of a particular pixel.\n *\n * @param r Row number of grid.\n * @param c Column number of grid.\n * @param newValue Value of the pixel.\n */\n setPixel(r, c, newValue) {\n const newGrid = JSON.parse(JSON.stringify(this.getValue()));\n newGrid[r][c] = newValue;\n this.fireIntermediateChangeEvent(newGrid);\n this.setValue(newGrid, false);\n }\n getPixel(row, column) {\n const value = this.getValue();\n if (!value) {\n throw new Error('Attempted to retrieve a pixel value when no value is set');\n }\n return value[row][column];\n }\n /**\n * Calls a given function for all cells in the image, with the cell\n * coordinates as the arguments.\n *\n * @param func A function to be applied.\n */\n forAllCells(func) {\n for (let r = 0; r < this.imgHeight; r++) {\n for (let c = 0; c < this.imgWidth; c++) {\n func(r, c);\n }\n }\n }\n /**\n * Creates a new element with the specified type and class.\n *\n * @param elementType Type of html element.\n * @param className ClassName of html element.\n * @returns The created element.\n */\n createElementWithClassname(elementType, className) {\n const newElt = document.createElement(elementType);\n newElt.className = className;\n return newElt;\n }\n /**\n * Binds an event listener to the specified element.\n *\n * @param element Specified element.\n * @param eventName Name of the event to bind.\n * @param callback Function to be called on specified event.\n */\n bindEvent(element, eventName, callback) {\n this.boundEvents.push(blockly_core__WEBPACK_IMPORTED_MODULE_0__.browserEvents.conditionalBind(element, eventName, this, callback));\n }\n fireIntermediateChangeEvent(newValue) {\n if (this.getSourceBlock()) {\n blockly_core__WEBPACK_IMPORTED_MODULE_0__.Events.fire(new (blockly_core__WEBPACK_IMPORTED_MODULE_0__.Events.get(blockly_core__WEBPACK_IMPORTED_MODULE_0__.Events.BLOCK_FIELD_INTERMEDIATE_CHANGE))(this.getSourceBlock(), this.name || null, this.getValue(), newValue));\n }\n }\n}\nblockly_core__WEBPACK_IMPORTED_MODULE_0__.fieldRegistry.register('field_bitmap', FieldBitmap);\n/**\n * CSS for bitmap field.\n */\nblockly_core__WEBPACK_IMPORTED_MODULE_0__.Css.register(`\n.dropdownEditor {\n align-items: center;\n flex-direction: column;\n display: flex;\n justify-content: center;\n margin-bottom: 20px;\n}\n.pixelContainer {\n margin: 20px;\n}\n.pixelRow {\n display: flex;\n flex-direction: row;\n padding: 0;\n margin: 0;\n height: ${DEFAULT_PIXEL_SIZE}\n}\n.pixelButton {\n width: ${DEFAULT_PIXEL_SIZE}px;\n height: ${DEFAULT_PIXEL_SIZE}px;\n border: 1px solid #000;\n}\n.pixelDisplay {\n white-space:pre-wrap;\n}\n.controlButton {\n margin: 5px 0;\n}\n`);\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"./src/field-bitmap.ts.js","mappings":"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","sources":["webpack://@blockly/field-bitmap/./src/field-bitmap.ts?ac63"],"sourcesContent":["/**\n * @license\n * Copyright 2021 Google LLC\n * SPDX-License-Identifier: Apache-2.0\n */\n\nimport * as Blockly from 'blockly/core';\n\nBlockly.Msg['BUTTON_LABEL_RANDOMIZE'] = 'Randomize';\nBlockly.Msg['BUTTON_LABEL_CLEAR'] = 'Clear';\n\nexport const DEFAULT_HEIGHT = 5;\nexport const DEFAULT_WIDTH = 5;\nconst DEFAULT_PIXEL_SIZE = 15;\nconst DEFAULT_PIXEL_COLOURS: PixelColours = {\n  empty: '#fff',\n  filled: '#363d80',\n};\nconst DEFAULT_BUTTONS: Buttons = {\n  randomize: true,\n  clear: true,\n};\n/**\n * Field for inputting a small bitmap image.\n * Includes a grid of clickable pixels that's exported as a bitmap.\n */\nexport class FieldBitmap extends Blockly.Field<number[][]> {\n  private initialValue: number[][] | null = null;\n  private imgHeight: number;\n  private imgWidth: number;\n  /**\n   * Array holding info needed to unbind events.\n   * Used for disposing.\n   */\n  private boundEvents: Blockly.browserEvents.Data[] = [];\n  /** References to UI elements */\n  private editorPixels: HTMLElement[][] | null = null;\n  private blockDisplayPixels: SVGElement[][] | null = null;\n  /** Stateful variables */\n  private mouseIsDown = false;\n  private valToPaintWith?: number;\n  buttonOptions: Buttons;\n  pixelSize: number;\n  pixelColours: {empty: string; filled: string};\n\n  /**\n   * Constructor for the bitmap field.\n   *\n   * @param value 2D rectangular array of 1s and 0s.\n   * @param validator A function that is called to validate.\n   * @param config Config A map of options used to configure the field.\n   */\n  constructor(\n    value: number[][] | typeof Blockly.Field.SKIP_SETUP,\n    validator?: Blockly.FieldValidator<number[][]>,\n    config?: FieldBitmapFromJsonConfig,\n  ) {\n    super(value, validator, config);\n\n    this.SERIALIZABLE = true;\n    this.CURSOR = 'default';\n    this.buttonOptions = {...DEFAULT_BUTTONS, ...config?.buttons};\n    this.pixelColours = {...DEFAULT_PIXEL_COLOURS, ...config?.colours};\n\n    // Configure value, height, and width\n    const currentValue = this.getValue();\n    if (currentValue !== null) {\n      this.imgHeight = currentValue.length;\n      this.imgWidth = currentValue[0].length || 0;\n    } else {\n      this.imgHeight = config?.height ?? DEFAULT_HEIGHT;\n      this.imgWidth = config?.width ?? DEFAULT_WIDTH;\n      // Set a default empty value\n      this.setValue(this.getEmptyArray());\n    }\n    if (config?.fieldHeight) {\n      this.pixelSize = config.fieldHeight / this.imgHeight;\n    } else {\n      this.pixelSize = DEFAULT_PIXEL_SIZE;\n    }\n  }\n\n  /**\n   * Constructs a FieldBitmap from a JSON arg object.\n   *\n   * @param options A JSON object with options.\n   * @returns The new field instance.\n   */\n  static fromJson(options: FieldBitmapFromJsonConfig) {\n    // `this` might be a subclass of FieldBitmap if that class doesn't override the static fromJson method.\n    return new this(\n      options.value ?? Blockly.Field.SKIP_SETUP,\n      undefined,\n      options,\n    );\n  }\n\n  /**\n   * Returns the width of the image in pixels.\n   *\n   * @returns The width in pixels.\n   */\n  getImageWidth() {\n    return this.imgWidth;\n  }\n\n  /**\n   * Returns the height of the image in pixels.\n   *\n   * @returns The height in pixels.\n   */\n  getImageHeight() {\n    return this.imgHeight;\n  }\n\n  /**\n   * Validates that a new value meets the requirements for a valid bitmap array.\n   *\n   * @param newValue The new value to be tested.\n   * @returns The new value if it's valid, or null.\n   */\n  // eslint-disable-next-line\n  protected override doClassValidation_(newValue: unknown = undefined) {\n    if (!newValue) {\n      return null;\n    }\n    // Check if the new value is an array\n    if (!Array.isArray(newValue)) {\n      return null;\n    }\n    const newHeight = newValue.length;\n    // The empty list is not an acceptable bitmap\n    if (newHeight == 0) {\n      return null;\n    }\n\n    // Check that the width matches the existing width of the image if it\n    // already has a value.\n    const newWidth = newValue[0].length;\n    for (const row of newValue) {\n      if (!Array.isArray(row)) {\n        return null;\n      }\n      if (row.length !== newWidth) {\n        return null;\n      }\n    }\n\n    // Check if all contents of the arrays are either 0 or 1\n    for (const row of newValue) {\n      for (const cell of row) {\n        if (cell !== 0 && cell !== 1) {\n          return null;\n        }\n      }\n    }\n    return newValue;\n  }\n\n  /**\n   * Called when a new value has been validated and is about to be set.\n   *\n   * @param newValue The value that's about to be set.\n   */\n  // eslint-disable-next-line\n  protected override doValueUpdate_(newValue: number[][]) {\n    super.doValueUpdate_(newValue);\n    if (newValue) {\n      this.imgHeight = newValue.length;\n      this.imgWidth = newValue[0] ? newValue[0].length : 0;\n    }\n  }\n\n  /**\n   * Show the bitmap editor dialog.\n   *\n   * @param e Optional mouse event that triggered the field to open, or\n   *    undefined if triggered programmatically.\n   */\n  // eslint-disable-next-line\n  protected override showEditor_(e?: Event) {\n    const editor = this.dropdownCreate();\n    Blockly.DropDownDiv.getContentDiv().appendChild(editor);\n    Blockly.DropDownDiv.showPositionedByField(\n      this,\n      this.dropdownDispose.bind(this),\n    );\n  }\n\n  /**\n   * Updates the block display and editor dropdown when the field re-renders.\n   */\n  // eslint-disable-next-line\n  protected override render_() {\n    super.render_();\n\n    if (!this.getValue()) {\n      return;\n    }\n\n    if (this.blockDisplayPixels) {\n      this.forAllCells((r, c) => {\n        const pixel = this.getPixel(r, c);\n\n        if (this.blockDisplayPixels) {\n          this.blockDisplayPixels[r][c].style.fill = pixel\n            ? this.pixelColours.filled\n            : this.pixelColours.empty;\n        }\n        if (this.editorPixels) {\n          this.editorPixels[r][c].style.background = pixel\n            ? this.pixelColours.filled\n            : this.pixelColours.empty;\n        }\n      });\n    }\n  }\n\n  /**\n   * Determines whether the field is editable.\n   *\n   * @returns True since it is always editable.\n   */\n  override updateEditable() {\n    const editable = super.updateEditable();\n    // Blockly.Field's implementation sets these classes as appropriate, but\n    // since this field has no text they just mess up the rendering of the grid\n    // lines.\n    const svgRoot = this.getSvgRoot();\n    if (svgRoot) {\n      Blockly.utils.dom.removeClass(svgRoot, 'blocklyNonEditableText');\n      Blockly.utils.dom.removeClass(svgRoot, 'blocklyEditableText');\n    }\n    return editable;\n  }\n\n  /**\n   * Gets the rectangle built out of dimensions matching SVG's <g> element.\n   *\n   * @returns The newly created rectangle of same size as the SVG element.\n   */\n  override getScaledBBox() {\n    const boundingBox = this.getSvgRoot()?.getBoundingClientRect();\n    if (!boundingBox) {\n      throw new Error('Tried to retrieve a bounding box without a rect');\n    }\n    return new Blockly.utils.Rect(\n      boundingBox.top,\n      boundingBox.bottom,\n      boundingBox.left,\n      boundingBox.right,\n    );\n  }\n\n  /**\n   * Creates the bitmap editor and add event listeners.\n   *\n   * @returns The newly created dropdown menu.\n   */\n  private dropdownCreate() {\n    const dropdownEditor = this.createElementWithClassname(\n      'div',\n      'dropdownEditor',\n    );\n    const pixelContainer = this.createElementWithClassname(\n      'div',\n      'pixelContainer',\n    );\n    dropdownEditor.appendChild(pixelContainer);\n\n    this.bindEvent(dropdownEditor, 'mouseup', this.onMouseUp);\n    this.bindEvent(dropdownEditor, 'mouseleave', this.onMouseUp);\n    this.bindEvent(dropdownEditor, 'dragstart', (e: Event) => {\n      e.preventDefault();\n    });\n\n    this.editorPixels = [];\n    for (let r = 0; r < this.imgHeight; r++) {\n      this.editorPixels.push([]);\n      const rowDiv = this.createElementWithClassname('div', 'pixelRow');\n      for (let c = 0; c < this.imgWidth; c++) {\n        // Add the button to the UI and save a reference to it\n        const button = this.createElementWithClassname('div', 'pixelButton');\n        this.editorPixels[r].push(button);\n        rowDiv.appendChild(button);\n\n        // Load the current pixel colour\n        const isOn = this.getPixel(r, c);\n        button.style.background = isOn\n          ? this.pixelColours.filled\n          : this.pixelColours.empty;\n\n        // Handle clicking a pixel\n        this.bindEvent(button, 'mousedown', () => {\n          this.onMouseDownInPixel(r, c);\n          return true;\n        });\n\n        // Handle dragging into a pixel when mouse is down\n        this.bindEvent(button, 'mouseenter', () => {\n          this.onMouseEnterPixel(r, c);\n        });\n      }\n      pixelContainer.appendChild(rowDiv);\n    }\n\n    // Add control buttons below the pixel grid\n    if (this.buttonOptions.randomize) {\n      this.addControlButton(\n        dropdownEditor,\n        Blockly.Msg['BUTTON_LABEL_RANDOMIZE'],\n        this.randomizePixels,\n      );\n    }\n    if (this.buttonOptions.clear) {\n      this.addControlButton(\n        dropdownEditor,\n        Blockly.Msg['BUTTON_LABEL_CLEAR'],\n        this.clearPixels,\n      );\n    }\n\n    if (this.blockDisplayPixels) {\n      this.forAllCells((r, c) => {\n        const pixel = this.getPixel(r, c);\n        if (this.editorPixels) {\n          this.editorPixels[r][c].style.background = pixel\n            ? this.pixelColours.filled\n            : this.pixelColours.empty;\n        }\n      });\n    }\n\n    // Store the initial value at the start of the edit.\n    this.initialValue = this.getValue();\n\n    return dropdownEditor;\n  }\n\n  /**\n   * Initializes the on-block display.\n   */\n  override initView() {\n    this.blockDisplayPixels = [];\n    for (let r = 0; r < this.imgHeight; r++) {\n      const row = [];\n      for (let c = 0; c < this.imgWidth; c++) {\n        const square = Blockly.utils.dom.createSvgElement(\n          'rect',\n          {\n            x: c * this.pixelSize,\n            y: r * this.pixelSize,\n            width: this.pixelSize,\n            height: this.pixelSize,\n            fill: this.pixelColours.empty,\n            fill_opacity: 1, // eslint-disable-line\n          },\n          this.getSvgRoot(),\n        );\n        row.push(square);\n      }\n      this.blockDisplayPixels.push(row);\n    }\n  }\n\n  /**\n   * Updates the size of the block based on the size of the underlying image.\n   */\n  // eslint-disable-next-line\n  protected override updateSize_() {\n    {\n      const newWidth = this.pixelSize * this.imgWidth;\n      const newHeight = this.pixelSize * this.imgHeight;\n      if (this.borderRect_) {\n        this.borderRect_.setAttribute('width', String(newWidth));\n        this.borderRect_.setAttribute('height', String(newHeight));\n      }\n\n      this.size_.width = newWidth;\n      this.size_.height = newHeight;\n    }\n  }\n\n  /**\n   * Create control button.\n   *\n   * @param parent Parent HTML element to which control button will be added.\n   * @param buttonText Text of the control button.\n   * @param onClick Callback that will be attached to the control button.\n   */\n  private addControlButton(\n    parent: HTMLElement,\n    buttonText: string,\n    onClick: () => void,\n  ) {\n    const button = this.createElementWithClassname('button', 'controlButton');\n    button.innerText = buttonText;\n    parent.appendChild(button);\n    this.bindEvent(button, 'click', onClick);\n  }\n\n  /**\n   * Disposes of events belonging to the bitmap editor.\n   */\n  private dropdownDispose() {\n    if (\n      this.getSourceBlock() &&\n      this.initialValue !== null &&\n      this.initialValue !== this.getValue()\n    ) {\n      Blockly.Events.fire(\n        new (Blockly.Events.get(Blockly.Events.BLOCK_CHANGE))(\n          this.sourceBlock_,\n          'field',\n          this.name || null,\n          this.initialValue,\n          this.getValue(),\n        ),\n      );\n    }\n\n    for (const event of this.boundEvents) {\n      Blockly.browserEvents.unbind(event);\n    }\n    this.boundEvents.length = 0;\n    this.editorPixels = null;\n    // Set this.initialValue back to null.\n    this.initialValue = null;\n  }\n\n  /**\n   * Constructs an array of zeros with the specified width and height.\n   *\n   * @returns The new value.\n   */\n  private getEmptyArray(): number[][] {\n    const newVal: number[][] = [];\n    for (let r = 0; r < this.imgHeight; r++) {\n      newVal.push([]);\n      for (let c = 0; c < this.imgWidth; c++) {\n        newVal[r].push(0);\n      }\n    }\n    return newVal;\n  }\n\n  /**\n   * Called when a mousedown event occurs within the bounds of a pixel.\n   *\n   * @param r Row number of grid.\n   * @param c Column number of grid.\n   */\n  private onMouseDownInPixel(r: number, c: number) {\n    // Toggle that pixel to the opposite of its value\n    const newPixelValue = 1 - this.getPixel(r, c);\n    this.setPixel(r, c, newPixelValue);\n    this.mouseIsDown = true;\n    this.valToPaintWith = newPixelValue;\n  }\n\n  /**\n   * Called when the mouse drags over a pixel in the editor.\n   *\n   * @param r Row number of grid.\n   * @param c Column number of grid.\n   */\n  private onMouseEnterPixel(r: number, c: number) {\n    if (!this.mouseIsDown) {\n      return;\n    }\n    if (\n      this.valToPaintWith !== undefined &&\n      this.getPixel(r, c) !== this.valToPaintWith\n    ) {\n      this.setPixel(r, c, this.valToPaintWith);\n    }\n  }\n\n  /**\n   * Resets mouse state (e.g. After either a mouseup event or if the mouse\n   * leaves the editor area).\n   */\n  private onMouseUp() {\n    this.mouseIsDown = false;\n    this.valToPaintWith = undefined;\n  }\n\n  /**\n   * Sets all the pixels in the image to a random value.\n   */\n  private randomizePixels() {\n    const getRandBinary = () => Math.floor(Math.random() * 2);\n    this.forAllCells((r, c) => {\n      this.setPixel(r, c, getRandBinary());\n    });\n  }\n\n  /**\n   * Sets all the pixels to 0.\n   */\n  private clearPixels() {\n    const cleared = this.getEmptyArray();\n    this.fireIntermediateChangeEvent(cleared);\n    this.setValue(cleared, false);\n  }\n\n  /**\n   * Sets the value of a particular pixel.\n   *\n   * @param r Row number of grid.\n   * @param c Column number of grid.\n   * @param newValue Value of the pixel.\n   */\n  private setPixel(r: number, c: number, newValue: number) {\n    const newGrid = JSON.parse(JSON.stringify(this.getValue()));\n    newGrid[r][c] = newValue;\n    this.fireIntermediateChangeEvent(newGrid);\n    this.setValue(newGrid, false);\n  }\n\n  private getPixel(row: number, column: number): number {\n    const value = this.getValue();\n    if (!value) {\n      throw new Error(\n        'Attempted to retrieve a pixel value when no value is set',\n      );\n    }\n\n    return value[row][column];\n  }\n\n  /**\n   * Calls a given function for all cells in the image, with the cell\n   * coordinates as the arguments.\n   *\n   * @param func A function to be applied.\n   */\n  private forAllCells(func: (row: number, col: number) => void) {\n    for (let r = 0; r < this.imgHeight; r++) {\n      for (let c = 0; c < this.imgWidth; c++) {\n        func(r, c);\n      }\n    }\n  }\n\n  /**\n   * Creates a new element with the specified type and class.\n   *\n   * @param elementType Type of html element.\n   * @param className ClassName of html element.\n   * @returns The created element.\n   */\n  private createElementWithClassname(elementType: string, className: string) {\n    const newElt = document.createElement(elementType);\n    newElt.className = className;\n    return newElt;\n  }\n\n  /**\n   * Binds an event listener to the specified element.\n   *\n   * @param element Specified element.\n   * @param eventName Name of the event to bind.\n   * @param callback Function to be called on specified event.\n   */\n  private bindEvent(\n    element: HTMLElement,\n    eventName: string,\n    callback: (e: Event) => void,\n  ) {\n    this.boundEvents.push(\n      Blockly.browserEvents.conditionalBind(element, eventName, this, callback),\n    );\n  }\n\n  private fireIntermediateChangeEvent(newValue: number[][]) {\n    if (this.getSourceBlock()) {\n      Blockly.Events.fire(\n        new (Blockly.Events.get(\n          Blockly.Events.BLOCK_FIELD_INTERMEDIATE_CHANGE,\n        ))(this.getSourceBlock(), this.name || null, this.getValue(), newValue),\n      );\n    }\n  }\n}\n\ninterface Buttons {\n  readonly randomize: boolean;\n  readonly clear: boolean;\n}\ninterface PixelColours {\n  readonly empty: string;\n  readonly filled: string;\n}\n\nexport interface FieldBitmapFromJsonConfig extends Blockly.FieldConfig {\n  value?: number[][];\n  width?: number;\n  height?: number;\n  buttons?: Buttons;\n  fieldHeight?: number;\n  colours?: PixelColours;\n}\n\nBlockly.fieldRegistry.register('field_bitmap', FieldBitmap);\n\n/**\n * CSS for bitmap field.\n */\nBlockly.Css.register(`\n.dropdownEditor {\n  align-items: center;\n  flex-direction: column;\n  display: flex;\n  justify-content: center;\n  margin-bottom: 20px;\n}\n.pixelContainer {\n  margin: 20px;\n}\n.pixelRow {\n  display: flex;\n  flex-direction: row;\n  padding: 0;\n  margin: 0;\n  height: ${DEFAULT_PIXEL_SIZE}\n}\n.pixelButton {\n  width: ${DEFAULT_PIXEL_SIZE}px;\n  height: ${DEFAULT_PIXEL_SIZE}px;\n  border: 1px solid #000;\n}\n.pixelDisplay {\n  white-space:pre-wrap;\n}\n.controlButton {\n  margin: 5px 0;\n}\n`);\n"],"names":[],"sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/field-bitmap.ts\n");
+eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"DEFAULT_HEIGHT\": () => (/* binding */ DEFAULT_HEIGHT),\n/* harmony export */ \"DEFAULT_WIDTH\": () => (/* binding */ DEFAULT_WIDTH),\n/* harmony export */ \"FieldBitmap\": () => (/* binding */ FieldBitmap)\n/* harmony export */ });\n/* harmony import */ var blockly_core__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! blockly/core */ \"./node_modules/blockly/core-browser.js\");\n/* harmony import */ var blockly_core__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(blockly_core__WEBPACK_IMPORTED_MODULE_0__);\n/**\n * @license\n * Copyright 2021 Google LLC\n * SPDX-License-Identifier: Apache-2.0\n */\n\nblockly_core__WEBPACK_IMPORTED_MODULE_0__.Msg.BUTTON_LABEL_RANDOMIZE = 'Randomize';\nblockly_core__WEBPACK_IMPORTED_MODULE_0__.Msg.BUTTON_LABEL_CLEAR = 'Clear';\nconst DEFAULT_HEIGHT = 5;\nconst DEFAULT_WIDTH = 5;\nconst DEFAULT_PIXEL_SIZE = 15;\nconst DEFAULT_PIXEL_COLOURS = {\n empty: '#fff',\n filled: '#363d80',\n};\nconst DEFAULT_BUTTONS = {\n randomize: true,\n clear: true,\n};\n/**\n * Field for inputting a small bitmap image.\n * Includes a grid of clickable pixels that's exported as a bitmap.\n */\nclass FieldBitmap extends blockly_core__WEBPACK_IMPORTED_MODULE_0__.Field {\n /**\n * Constructor for the bitmap field.\n *\n * @param value 2D rectangular array of 1s and 0s.\n * @param validator A function that is called to validate.\n * @param config Config A map of options used to configure the field.\n */\n constructor(value, validator, config) {\n var _a, _b;\n super(value, validator, config);\n this.initialValue = null;\n /**\n * Array holding info needed to unbind events.\n * Used for disposing.\n */\n this.boundEvents = [];\n /** References to UI elements */\n this.editorPixels = null;\n this.blockDisplayPixels = null;\n /** Stateful variables */\n this.mouseIsDown = false;\n this.SERIALIZABLE = true;\n this.CURSOR = 'default';\n this.buttonOptions = Object.assign(Object.assign({}, DEFAULT_BUTTONS), config === null || config === void 0 ? void 0 : config.buttons);\n this.pixelColours = Object.assign(Object.assign({}, DEFAULT_PIXEL_COLOURS), config === null || config === void 0 ? void 0 : config.colours);\n // Configure value, height, and width\n const currentValue = this.getValue();\n if (currentValue !== null) {\n this.imgHeight = currentValue.length;\n this.imgWidth = currentValue[0].length || 0;\n }\n else {\n this.imgHeight = (_a = config === null || config === void 0 ? void 0 : config.height) !== null && _a !== void 0 ? _a : DEFAULT_HEIGHT;\n this.imgWidth = (_b = config === null || config === void 0 ? void 0 : config.width) !== null && _b !== void 0 ? _b : DEFAULT_WIDTH;\n // Set a default empty value\n this.setValue(this.getEmptyArray());\n }\n if (config === null || config === void 0 ? void 0 : config.fieldHeight) {\n this.pixelSize = config.fieldHeight / this.imgHeight;\n }\n else {\n this.pixelSize = DEFAULT_PIXEL_SIZE;\n }\n }\n /**\n * Constructs a FieldBitmap from a JSON arg object.\n *\n * @param options A JSON object with options.\n * @returns The new field instance.\n */\n static fromJson(options) {\n var _a;\n // `this` might be a subclass of FieldBitmap if that class doesn't override the static fromJson method.\n return new this((_a = options.value) !== null && _a !== void 0 ? _a : blockly_core__WEBPACK_IMPORTED_MODULE_0__.Field.SKIP_SETUP, undefined, options);\n }\n /**\n * Returns the width of the image in pixels.\n *\n * @returns The width in pixels.\n */\n getImageWidth() {\n return this.imgWidth;\n }\n /**\n * Returns the height of the image in pixels.\n *\n * @returns The height in pixels.\n */\n getImageHeight() {\n return this.imgHeight;\n }\n /**\n * Validates that a new value meets the requirements for a valid bitmap array.\n *\n * @param newValue The new value to be tested.\n * @returns The new value if it's valid, or null.\n */\n // eslint-disable-next-line\n doClassValidation_(newValue = undefined) {\n if (!newValue) {\n return null;\n }\n // Check if the new value is an array\n if (!Array.isArray(newValue)) {\n return null;\n }\n const newHeight = newValue.length;\n // The empty list is not an acceptable bitmap\n if (newHeight == 0) {\n return null;\n }\n // Check that the width matches the existing width of the image if it\n // already has a value.\n const newWidth = newValue[0].length;\n for (const row of newValue) {\n if (!Array.isArray(row)) {\n return null;\n }\n if (row.length !== newWidth) {\n return null;\n }\n }\n // Check if all contents of the arrays are either 0 or 1\n for (const row of newValue) {\n for (const cell of row) {\n if (cell !== 0 && cell !== 1) {\n return null;\n }\n }\n }\n return newValue;\n }\n /**\n * Called when a new value has been validated and is about to be set.\n *\n * @param newValue The value that's about to be set.\n */\n // eslint-disable-next-line\n doValueUpdate_(newValue) {\n super.doValueUpdate_(newValue);\n if (newValue) {\n this.imgHeight = newValue.length;\n this.imgWidth = newValue[0] ? newValue[0].length : 0;\n }\n }\n /**\n * Show the bitmap editor dialog.\n *\n * @param e Optional mouse event that triggered the field to open, or\n * undefined if triggered programmatically.\n */\n // eslint-disable-next-line\n showEditor_(e) {\n const editor = this.dropdownCreate();\n blockly_core__WEBPACK_IMPORTED_MODULE_0__.DropDownDiv.getContentDiv().appendChild(editor);\n blockly_core__WEBPACK_IMPORTED_MODULE_0__.DropDownDiv.showPositionedByField(this, this.dropdownDispose.bind(this));\n }\n /**\n * Updates the block display and editor dropdown when the field re-renders.\n */\n // eslint-disable-next-line\n render_() {\n super.render_();\n if (!this.getValue()) {\n return;\n }\n if (this.blockDisplayPixels) {\n this.forAllCells((r, c) => {\n const pixel = this.getPixel(r, c);\n if (this.blockDisplayPixels) {\n this.blockDisplayPixels[r][c].style.fill = pixel\n ? this.pixelColours.filled\n : this.pixelColours.empty;\n }\n if (this.editorPixels) {\n this.editorPixels[r][c].style.background = pixel\n ? this.pixelColours.filled\n : this.pixelColours.empty;\n }\n });\n }\n }\n /**\n * Determines whether the field is editable.\n *\n * @returns True since it is always editable.\n */\n updateEditable() {\n const editable = super.updateEditable();\n // Blockly.Field's implementation sets these classes as appropriate, but\n // since this field has no text they just mess up the rendering of the grid\n // lines.\n const svgRoot = this.getSvgRoot();\n if (svgRoot) {\n blockly_core__WEBPACK_IMPORTED_MODULE_0__.utils.dom.removeClass(svgRoot, 'blocklyNonEditableText');\n blockly_core__WEBPACK_IMPORTED_MODULE_0__.utils.dom.removeClass(svgRoot, 'blocklyEditableText');\n }\n return editable;\n }\n /**\n * Gets the rectangle built out of dimensions matching SVG's element.\n *\n * @returns The newly created rectangle of same size as the SVG element.\n */\n getScaledBBox() {\n var _a;\n const boundingBox = (_a = this.getSvgRoot()) === null || _a === void 0 ? void 0 : _a.getBoundingClientRect();\n if (!boundingBox) {\n throw new Error('Tried to retrieve a bounding box without a rect');\n }\n return new blockly_core__WEBPACK_IMPORTED_MODULE_0__.utils.Rect(boundingBox.top, boundingBox.bottom, boundingBox.left, boundingBox.right);\n }\n /**\n * Creates the bitmap editor and add event listeners.\n *\n * @returns The newly created dropdown menu.\n */\n dropdownCreate() {\n const dropdownEditor = this.createElementWithClassname('div', 'dropdownEditor');\n if (this.buttonOptions.randomize || this.buttonOptions.clear) {\n dropdownEditor.classList.add('has-buttons');\n }\n const pixelContainer = this.createElementWithClassname('div', 'pixelContainer');\n dropdownEditor.appendChild(pixelContainer);\n // This prevents the normal max-height from adding a scroll bar for large images.\n blockly_core__WEBPACK_IMPORTED_MODULE_0__.DropDownDiv.getContentDiv().classList.add('contains-bitmap-editor');\n this.bindEvent(dropdownEditor, 'mouseup', this.onMouseUp);\n this.bindEvent(dropdownEditor, 'mouseleave', this.onMouseUp);\n this.bindEvent(dropdownEditor, 'dragstart', (e) => {\n e.preventDefault();\n });\n this.editorPixels = [];\n for (let r = 0; r < this.imgHeight; r++) {\n this.editorPixels.push([]);\n const rowDiv = this.createElementWithClassname('div', 'pixelRow');\n for (let c = 0; c < this.imgWidth; c++) {\n // Add the button to the UI and save a reference to it\n const button = this.createElementWithClassname('div', 'pixelButton');\n this.editorPixels[r].push(button);\n rowDiv.appendChild(button);\n // Load the current pixel colour\n const isOn = this.getPixel(r, c);\n button.style.background = isOn\n ? this.pixelColours.filled\n : this.pixelColours.empty;\n // Handle clicking a pixel\n this.bindEvent(button, 'mousedown', () => {\n this.onMouseDownInPixel(r, c);\n return true;\n });\n // Handle dragging into a pixel when mouse is down\n this.bindEvent(button, 'mouseenter', () => {\n this.onMouseEnterPixel(r, c);\n });\n }\n pixelContainer.appendChild(rowDiv);\n }\n // Add control buttons below the pixel grid\n if (this.buttonOptions.randomize) {\n this.addControlButton(dropdownEditor, blockly_core__WEBPACK_IMPORTED_MODULE_0__.Msg.BUTTON_LABEL_RANDOMIZE, this.randomizePixels);\n }\n if (this.buttonOptions.clear) {\n this.addControlButton(dropdownEditor, blockly_core__WEBPACK_IMPORTED_MODULE_0__.Msg.BUTTON_LABEL_CLEAR, this.clearPixels);\n }\n if (this.blockDisplayPixels) {\n this.forAllCells((r, c) => {\n const pixel = this.getPixel(r, c);\n if (this.editorPixels) {\n this.editorPixels[r][c].style.background = pixel\n ? this.pixelColours.filled\n : this.pixelColours.empty;\n }\n });\n }\n // Store the initial value at the start of the edit.\n this.initialValue = this.getValue();\n return dropdownEditor;\n }\n /**\n * Initializes the on-block display.\n */\n initView() {\n this.blockDisplayPixels = [];\n for (let r = 0; r < this.imgHeight; r++) {\n const row = [];\n for (let c = 0; c < this.imgWidth; c++) {\n const square = blockly_core__WEBPACK_IMPORTED_MODULE_0__.utils.dom.createSvgElement('rect', {\n x: c * this.pixelSize,\n y: r * this.pixelSize,\n width: this.pixelSize,\n height: this.pixelSize,\n fill: this.pixelColours.empty,\n fill_opacity: 1, // eslint-disable-line\n }, this.getSvgRoot());\n row.push(square);\n }\n this.blockDisplayPixels.push(row);\n }\n }\n /**\n * Updates the size of the block based on the size of the underlying image.\n */\n // eslint-disable-next-line\n updateSize_() {\n {\n const newWidth = this.pixelSize * this.imgWidth;\n const newHeight = this.pixelSize * this.imgHeight;\n if (this.borderRect_) {\n this.borderRect_.setAttribute('width', String(newWidth));\n this.borderRect_.setAttribute('height', String(newHeight));\n }\n this.size_.width = newWidth;\n this.size_.height = newHeight;\n }\n }\n /**\n * Create control button.\n *\n * @param parent Parent HTML element to which control button will be added.\n * @param buttonText Text of the control button.\n * @param onClick Callback that will be attached to the control button.\n */\n addControlButton(parent, buttonText, onClick) {\n const button = this.createElementWithClassname('button', 'controlButton');\n button.innerText = buttonText;\n parent.appendChild(button);\n this.bindEvent(button, 'click', onClick);\n }\n /**\n * Disposes of events belonging to the bitmap editor.\n */\n dropdownDispose() {\n if (this.getSourceBlock() &&\n this.initialValue !== null &&\n this.initialValue !== this.getValue()) {\n blockly_core__WEBPACK_IMPORTED_MODULE_0__.Events.fire(new (blockly_core__WEBPACK_IMPORTED_MODULE_0__.Events.get(blockly_core__WEBPACK_IMPORTED_MODULE_0__.Events.BLOCK_CHANGE))(this.sourceBlock_, 'field', this.name || null, this.initialValue, this.getValue()));\n }\n for (const event of this.boundEvents) {\n blockly_core__WEBPACK_IMPORTED_MODULE_0__.browserEvents.unbind(event);\n }\n this.boundEvents.length = 0;\n this.editorPixels = null;\n // Set this.initialValue back to null.\n this.initialValue = null;\n blockly_core__WEBPACK_IMPORTED_MODULE_0__.DropDownDiv.getContentDiv().classList.remove('contains-bitmap-editor');\n }\n /**\n * Constructs an array of zeros with the specified width and height.\n *\n * @returns The new value.\n */\n getEmptyArray() {\n const newVal = [];\n for (let r = 0; r < this.imgHeight; r++) {\n newVal.push([]);\n for (let c = 0; c < this.imgWidth; c++) {\n newVal[r].push(0);\n }\n }\n return newVal;\n }\n /**\n * Called when a mousedown event occurs within the bounds of a pixel.\n *\n * @param r Row number of grid.\n * @param c Column number of grid.\n */\n onMouseDownInPixel(r, c) {\n // Toggle that pixel to the opposite of its value\n const newPixelValue = 1 - this.getPixel(r, c);\n this.setPixel(r, c, newPixelValue);\n this.mouseIsDown = true;\n this.valToPaintWith = newPixelValue;\n }\n /**\n * Called when the mouse drags over a pixel in the editor.\n *\n * @param r Row number of grid.\n * @param c Column number of grid.\n */\n onMouseEnterPixel(r, c) {\n if (!this.mouseIsDown) {\n return;\n }\n if (this.valToPaintWith !== undefined &&\n this.getPixel(r, c) !== this.valToPaintWith) {\n this.setPixel(r, c, this.valToPaintWith);\n }\n }\n /**\n * Resets mouse state (e.g. After either a mouseup event or if the mouse\n * leaves the editor area).\n */\n onMouseUp() {\n this.mouseIsDown = false;\n this.valToPaintWith = undefined;\n }\n /**\n * Sets all the pixels in the image to a random value.\n */\n randomizePixels() {\n const getRandBinary = () => Math.floor(Math.random() * 2);\n this.forAllCells((r, c) => {\n this.setPixel(r, c, getRandBinary());\n });\n }\n /**\n * Sets all the pixels to 0.\n */\n clearPixels() {\n const cleared = this.getEmptyArray();\n this.fireIntermediateChangeEvent(cleared);\n this.setValue(cleared, false);\n }\n /**\n * Sets the value of a particular pixel.\n *\n * @param r Row number of grid.\n * @param c Column number of grid.\n * @param newValue Value of the pixel.\n */\n setPixel(r, c, newValue) {\n const newGrid = JSON.parse(JSON.stringify(this.getValue()));\n newGrid[r][c] = newValue;\n this.fireIntermediateChangeEvent(newGrid);\n this.setValue(newGrid, false);\n }\n getPixel(row, column) {\n const value = this.getValue();\n if (!value) {\n throw new Error('Attempted to retrieve a pixel value when no value is set');\n }\n return value[row][column];\n }\n /**\n * Calls a given function for all cells in the image, with the cell\n * coordinates as the arguments.\n *\n * @param func A function to be applied.\n */\n forAllCells(func) {\n for (let r = 0; r < this.imgHeight; r++) {\n for (let c = 0; c < this.imgWidth; c++) {\n func(r, c);\n }\n }\n }\n /**\n * Creates a new element with the specified type and class.\n *\n * @param elementType Type of html element.\n * @param className ClassName of html element.\n * @returns The created element.\n */\n createElementWithClassname(elementType, className) {\n const newElt = document.createElement(elementType);\n newElt.className = className;\n return newElt;\n }\n /**\n * Binds an event listener to the specified element.\n *\n * @param element Specified element.\n * @param eventName Name of the event to bind.\n * @param callback Function to be called on specified event.\n */\n bindEvent(element, eventName, callback) {\n this.boundEvents.push(blockly_core__WEBPACK_IMPORTED_MODULE_0__.browserEvents.conditionalBind(element, eventName, this, callback));\n }\n fireIntermediateChangeEvent(newValue) {\n if (this.getSourceBlock()) {\n blockly_core__WEBPACK_IMPORTED_MODULE_0__.Events.fire(new (blockly_core__WEBPACK_IMPORTED_MODULE_0__.Events.get(blockly_core__WEBPACK_IMPORTED_MODULE_0__.Events.BLOCK_FIELD_INTERMEDIATE_CHANGE))(this.getSourceBlock(), this.name || null, this.getValue(), newValue));\n }\n }\n}\nblockly_core__WEBPACK_IMPORTED_MODULE_0__.fieldRegistry.register('field_bitmap', FieldBitmap);\n/**\n * CSS for bitmap field.\n */\nblockly_core__WEBPACK_IMPORTED_MODULE_0__.Css.register(`\n.dropdownEditor {\n align-items: center;\n flex-direction: column;\n display: flex;\n justify-content: center;\n}\n.dropdownEditor.has-buttons {\n margin-bottom: 20px;\n}\n.pixelContainer {\n margin: 20px;\n}\n.pixelRow {\n display: flex;\n flex-direction: row;\n padding: 0;\n margin: 0;\n height: ${DEFAULT_PIXEL_SIZE}\n}\n.pixelButton {\n width: ${DEFAULT_PIXEL_SIZE}px;\n height: ${DEFAULT_PIXEL_SIZE}px;\n border: 1px solid #000;\n}\n.pixelDisplay {\n white-space:pre-wrap;\n}\n.controlButton {\n margin: 5px 0;\n}\n.blocklyDropDownContent.contains-bitmap-editor {\n max-height: none;\n}\n`);\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"./src/field-bitmap.ts.js","mappings":"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","sources":["webpack://@blockly/field-bitmap/./src/field-bitmap.ts?ac63"],"sourcesContent":["/**\n * @license\n * Copyright 2021 Google LLC\n * SPDX-License-Identifier: Apache-2.0\n */\n\nimport * as Blockly from 'blockly/core';\n\nBlockly.Msg['BUTTON_LABEL_RANDOMIZE'] = 'Randomize';\nBlockly.Msg['BUTTON_LABEL_CLEAR'] = 'Clear';\n\nexport const DEFAULT_HEIGHT = 5;\nexport const DEFAULT_WIDTH = 5;\nconst DEFAULT_PIXEL_SIZE = 15;\nconst DEFAULT_PIXEL_COLOURS: PixelColours = {\n  empty: '#fff',\n  filled: '#363d80',\n};\nconst DEFAULT_BUTTONS: Buttons = {\n  randomize: true,\n  clear: true,\n};\n/**\n * Field for inputting a small bitmap image.\n * Includes a grid of clickable pixels that's exported as a bitmap.\n */\nexport class FieldBitmap extends Blockly.Field<number[][]> {\n  private initialValue: number[][] | null = null;\n  private imgHeight: number;\n  private imgWidth: number;\n  /**\n   * Array holding info needed to unbind events.\n   * Used for disposing.\n   */\n  private boundEvents: Blockly.browserEvents.Data[] = [];\n  /** References to UI elements */\n  private editorPixels: HTMLElement[][] | null = null;\n  private blockDisplayPixels: SVGElement[][] | null = null;\n  /** Stateful variables */\n  private mouseIsDown = false;\n  private valToPaintWith?: number;\n  buttonOptions: Buttons;\n  pixelSize: number;\n  pixelColours: {empty: string; filled: string};\n\n  /**\n   * Constructor for the bitmap field.\n   *\n   * @param value 2D rectangular array of 1s and 0s.\n   * @param validator A function that is called to validate.\n   * @param config Config A map of options used to configure the field.\n   */\n  constructor(\n    value: number[][] | typeof Blockly.Field.SKIP_SETUP,\n    validator?: Blockly.FieldValidator<number[][]>,\n    config?: FieldBitmapFromJsonConfig,\n  ) {\n    super(value, validator, config);\n\n    this.SERIALIZABLE = true;\n    this.CURSOR = 'default';\n    this.buttonOptions = {...DEFAULT_BUTTONS, ...config?.buttons};\n    this.pixelColours = {...DEFAULT_PIXEL_COLOURS, ...config?.colours};\n\n    // Configure value, height, and width\n    const currentValue = this.getValue();\n    if (currentValue !== null) {\n      this.imgHeight = currentValue.length;\n      this.imgWidth = currentValue[0].length || 0;\n    } else {\n      this.imgHeight = config?.height ?? DEFAULT_HEIGHT;\n      this.imgWidth = config?.width ?? DEFAULT_WIDTH;\n      // Set a default empty value\n      this.setValue(this.getEmptyArray());\n    }\n    if (config?.fieldHeight) {\n      this.pixelSize = config.fieldHeight / this.imgHeight;\n    } else {\n      this.pixelSize = DEFAULT_PIXEL_SIZE;\n    }\n  }\n\n  /**\n   * Constructs a FieldBitmap from a JSON arg object.\n   *\n   * @param options A JSON object with options.\n   * @returns The new field instance.\n   */\n  static fromJson(options: FieldBitmapFromJsonConfig) {\n    // `this` might be a subclass of FieldBitmap if that class doesn't override the static fromJson method.\n    return new this(\n      options.value ?? Blockly.Field.SKIP_SETUP,\n      undefined,\n      options,\n    );\n  }\n\n  /**\n   * Returns the width of the image in pixels.\n   *\n   * @returns The width in pixels.\n   */\n  getImageWidth() {\n    return this.imgWidth;\n  }\n\n  /**\n   * Returns the height of the image in pixels.\n   *\n   * @returns The height in pixels.\n   */\n  getImageHeight() {\n    return this.imgHeight;\n  }\n\n  /**\n   * Validates that a new value meets the requirements for a valid bitmap array.\n   *\n   * @param newValue The new value to be tested.\n   * @returns The new value if it's valid, or null.\n   */\n  // eslint-disable-next-line\n  protected override doClassValidation_(newValue: unknown = undefined) {\n    if (!newValue) {\n      return null;\n    }\n    // Check if the new value is an array\n    if (!Array.isArray(newValue)) {\n      return null;\n    }\n    const newHeight = newValue.length;\n    // The empty list is not an acceptable bitmap\n    if (newHeight == 0) {\n      return null;\n    }\n\n    // Check that the width matches the existing width of the image if it\n    // already has a value.\n    const newWidth = newValue[0].length;\n    for (const row of newValue) {\n      if (!Array.isArray(row)) {\n        return null;\n      }\n      if (row.length !== newWidth) {\n        return null;\n      }\n    }\n\n    // Check if all contents of the arrays are either 0 or 1\n    for (const row of newValue) {\n      for (const cell of row) {\n        if (cell !== 0 && cell !== 1) {\n          return null;\n        }\n      }\n    }\n    return newValue;\n  }\n\n  /**\n   * Called when a new value has been validated and is about to be set.\n   *\n   * @param newValue The value that's about to be set.\n   */\n  // eslint-disable-next-line\n  protected override doValueUpdate_(newValue: number[][]) {\n    super.doValueUpdate_(newValue);\n    if (newValue) {\n      this.imgHeight = newValue.length;\n      this.imgWidth = newValue[0] ? newValue[0].length : 0;\n    }\n  }\n\n  /**\n   * Show the bitmap editor dialog.\n   *\n   * @param e Optional mouse event that triggered the field to open, or\n   *    undefined if triggered programmatically.\n   */\n  // eslint-disable-next-line\n  protected override showEditor_(e?: Event) {\n    const editor = this.dropdownCreate();\n    Blockly.DropDownDiv.getContentDiv().appendChild(editor);\n    Blockly.DropDownDiv.showPositionedByField(\n      this,\n      this.dropdownDispose.bind(this),\n    );\n  }\n\n  /**\n   * Updates the block display and editor dropdown when the field re-renders.\n   */\n  // eslint-disable-next-line\n  protected override render_() {\n    super.render_();\n\n    if (!this.getValue()) {\n      return;\n    }\n\n    if (this.blockDisplayPixels) {\n      this.forAllCells((r, c) => {\n        const pixel = this.getPixel(r, c);\n\n        if (this.blockDisplayPixels) {\n          this.blockDisplayPixels[r][c].style.fill = pixel\n            ? this.pixelColours.filled\n            : this.pixelColours.empty;\n        }\n        if (this.editorPixels) {\n          this.editorPixels[r][c].style.background = pixel\n            ? this.pixelColours.filled\n            : this.pixelColours.empty;\n        }\n      });\n    }\n  }\n\n  /**\n   * Determines whether the field is editable.\n   *\n   * @returns True since it is always editable.\n   */\n  override updateEditable() {\n    const editable = super.updateEditable();\n    // Blockly.Field's implementation sets these classes as appropriate, but\n    // since this field has no text they just mess up the rendering of the grid\n    // lines.\n    const svgRoot = this.getSvgRoot();\n    if (svgRoot) {\n      Blockly.utils.dom.removeClass(svgRoot, 'blocklyNonEditableText');\n      Blockly.utils.dom.removeClass(svgRoot, 'blocklyEditableText');\n    }\n    return editable;\n  }\n\n  /**\n   * Gets the rectangle built out of dimensions matching SVG's <g> element.\n   *\n   * @returns The newly created rectangle of same size as the SVG element.\n   */\n  override getScaledBBox() {\n    const boundingBox = this.getSvgRoot()?.getBoundingClientRect();\n    if (!boundingBox) {\n      throw new Error('Tried to retrieve a bounding box without a rect');\n    }\n    return new Blockly.utils.Rect(\n      boundingBox.top,\n      boundingBox.bottom,\n      boundingBox.left,\n      boundingBox.right,\n    );\n  }\n\n  /**\n   * Creates the bitmap editor and add event listeners.\n   *\n   * @returns The newly created dropdown menu.\n   */\n  private dropdownCreate() {\n    const dropdownEditor = this.createElementWithClassname(\n      'div',\n      'dropdownEditor',\n    );\n    if (this.buttonOptions.randomize || this.buttonOptions.clear) {\n      dropdownEditor.classList.add('has-buttons');\n    }\n    const pixelContainer = this.createElementWithClassname(\n      'div',\n      'pixelContainer',\n    );\n    dropdownEditor.appendChild(pixelContainer);\n\n    // This prevents the normal max-height from adding a scroll bar for large images.\n    Blockly.DropDownDiv.getContentDiv().classList.add('contains-bitmap-editor');\n\n    this.bindEvent(dropdownEditor, 'mouseup', this.onMouseUp);\n    this.bindEvent(dropdownEditor, 'mouseleave', this.onMouseUp);\n    this.bindEvent(dropdownEditor, 'dragstart', (e: Event) => {\n      e.preventDefault();\n    });\n\n    this.editorPixels = [];\n    for (let r = 0; r < this.imgHeight; r++) {\n      this.editorPixels.push([]);\n      const rowDiv = this.createElementWithClassname('div', 'pixelRow');\n      for (let c = 0; c < this.imgWidth; c++) {\n        // Add the button to the UI and save a reference to it\n        const button = this.createElementWithClassname('div', 'pixelButton');\n        this.editorPixels[r].push(button);\n        rowDiv.appendChild(button);\n\n        // Load the current pixel colour\n        const isOn = this.getPixel(r, c);\n        button.style.background = isOn\n          ? this.pixelColours.filled\n          : this.pixelColours.empty;\n\n        // Handle clicking a pixel\n        this.bindEvent(button, 'mousedown', () => {\n          this.onMouseDownInPixel(r, c);\n          return true;\n        });\n\n        // Handle dragging into a pixel when mouse is down\n        this.bindEvent(button, 'mouseenter', () => {\n          this.onMouseEnterPixel(r, c);\n        });\n      }\n      pixelContainer.appendChild(rowDiv);\n    }\n\n    // Add control buttons below the pixel grid\n    if (this.buttonOptions.randomize) {\n      this.addControlButton(\n        dropdownEditor,\n        Blockly.Msg['BUTTON_LABEL_RANDOMIZE'],\n        this.randomizePixels,\n      );\n    }\n    if (this.buttonOptions.clear) {\n      this.addControlButton(\n        dropdownEditor,\n        Blockly.Msg['BUTTON_LABEL_CLEAR'],\n        this.clearPixels,\n      );\n    }\n\n    if (this.blockDisplayPixels) {\n      this.forAllCells((r, c) => {\n        const pixel = this.getPixel(r, c);\n        if (this.editorPixels) {\n          this.editorPixels[r][c].style.background = pixel\n            ? this.pixelColours.filled\n            : this.pixelColours.empty;\n        }\n      });\n    }\n\n    // Store the initial value at the start of the edit.\n    this.initialValue = this.getValue();\n\n    return dropdownEditor;\n  }\n\n  /**\n   * Initializes the on-block display.\n   */\n  override initView() {\n    this.blockDisplayPixels = [];\n    for (let r = 0; r < this.imgHeight; r++) {\n      const row = [];\n      for (let c = 0; c < this.imgWidth; c++) {\n        const square = Blockly.utils.dom.createSvgElement(\n          'rect',\n          {\n            x: c * this.pixelSize,\n            y: r * this.pixelSize,\n            width: this.pixelSize,\n            height: this.pixelSize,\n            fill: this.pixelColours.empty,\n            fill_opacity: 1, // eslint-disable-line\n          },\n          this.getSvgRoot(),\n        );\n        row.push(square);\n      }\n      this.blockDisplayPixels.push(row);\n    }\n  }\n\n  /**\n   * Updates the size of the block based on the size of the underlying image.\n   */\n  // eslint-disable-next-line\n  protected override updateSize_() {\n    {\n      const newWidth = this.pixelSize * this.imgWidth;\n      const newHeight = this.pixelSize * this.imgHeight;\n      if (this.borderRect_) {\n        this.borderRect_.setAttribute('width', String(newWidth));\n        this.borderRect_.setAttribute('height', String(newHeight));\n      }\n\n      this.size_.width = newWidth;\n      this.size_.height = newHeight;\n    }\n  }\n\n  /**\n   * Create control button.\n   *\n   * @param parent Parent HTML element to which control button will be added.\n   * @param buttonText Text of the control button.\n   * @param onClick Callback that will be attached to the control button.\n   */\n  private addControlButton(\n    parent: HTMLElement,\n    buttonText: string,\n    onClick: () => void,\n  ) {\n    const button = this.createElementWithClassname('button', 'controlButton');\n    button.innerText = buttonText;\n    parent.appendChild(button);\n    this.bindEvent(button, 'click', onClick);\n  }\n\n  /**\n   * Disposes of events belonging to the bitmap editor.\n   */\n  private dropdownDispose() {\n    if (\n      this.getSourceBlock() &&\n      this.initialValue !== null &&\n      this.initialValue !== this.getValue()\n    ) {\n      Blockly.Events.fire(\n        new (Blockly.Events.get(Blockly.Events.BLOCK_CHANGE))(\n          this.sourceBlock_,\n          'field',\n          this.name || null,\n          this.initialValue,\n          this.getValue(),\n        ),\n      );\n    }\n\n    for (const event of this.boundEvents) {\n      Blockly.browserEvents.unbind(event);\n    }\n    this.boundEvents.length = 0;\n    this.editorPixels = null;\n    // Set this.initialValue back to null.\n    this.initialValue = null;\n\n    Blockly.DropDownDiv.getContentDiv().classList.remove(\n      'contains-bitmap-editor',\n    );\n  }\n\n  /**\n   * Constructs an array of zeros with the specified width and height.\n   *\n   * @returns The new value.\n   */\n  private getEmptyArray(): number[][] {\n    const newVal: number[][] = [];\n    for (let r = 0; r < this.imgHeight; r++) {\n      newVal.push([]);\n      for (let c = 0; c < this.imgWidth; c++) {\n        newVal[r].push(0);\n      }\n    }\n    return newVal;\n  }\n\n  /**\n   * Called when a mousedown event occurs within the bounds of a pixel.\n   *\n   * @param r Row number of grid.\n   * @param c Column number of grid.\n   */\n  private onMouseDownInPixel(r: number, c: number) {\n    // Toggle that pixel to the opposite of its value\n    const newPixelValue = 1 - this.getPixel(r, c);\n    this.setPixel(r, c, newPixelValue);\n    this.mouseIsDown = true;\n    this.valToPaintWith = newPixelValue;\n  }\n\n  /**\n   * Called when the mouse drags over a pixel in the editor.\n   *\n   * @param r Row number of grid.\n   * @param c Column number of grid.\n   */\n  private onMouseEnterPixel(r: number, c: number) {\n    if (!this.mouseIsDown) {\n      return;\n    }\n    if (\n      this.valToPaintWith !== undefined &&\n      this.getPixel(r, c) !== this.valToPaintWith\n    ) {\n      this.setPixel(r, c, this.valToPaintWith);\n    }\n  }\n\n  /**\n   * Resets mouse state (e.g. After either a mouseup event or if the mouse\n   * leaves the editor area).\n   */\n  private onMouseUp() {\n    this.mouseIsDown = false;\n    this.valToPaintWith = undefined;\n  }\n\n  /**\n   * Sets all the pixels in the image to a random value.\n   */\n  private randomizePixels() {\n    const getRandBinary = () => Math.floor(Math.random() * 2);\n    this.forAllCells((r, c) => {\n      this.setPixel(r, c, getRandBinary());\n    });\n  }\n\n  /**\n   * Sets all the pixels to 0.\n   */\n  private clearPixels() {\n    const cleared = this.getEmptyArray();\n    this.fireIntermediateChangeEvent(cleared);\n    this.setValue(cleared, false);\n  }\n\n  /**\n   * Sets the value of a particular pixel.\n   *\n   * @param r Row number of grid.\n   * @param c Column number of grid.\n   * @param newValue Value of the pixel.\n   */\n  private setPixel(r: number, c: number, newValue: number) {\n    const newGrid = JSON.parse(JSON.stringify(this.getValue()));\n    newGrid[r][c] = newValue;\n    this.fireIntermediateChangeEvent(newGrid);\n    this.setValue(newGrid, false);\n  }\n\n  private getPixel(row: number, column: number): number {\n    const value = this.getValue();\n    if (!value) {\n      throw new Error(\n        'Attempted to retrieve a pixel value when no value is set',\n      );\n    }\n\n    return value[row][column];\n  }\n\n  /**\n   * Calls a given function for all cells in the image, with the cell\n   * coordinates as the arguments.\n   *\n   * @param func A function to be applied.\n   */\n  private forAllCells(func: (row: number, col: number) => void) {\n    for (let r = 0; r < this.imgHeight; r++) {\n      for (let c = 0; c < this.imgWidth; c++) {\n        func(r, c);\n      }\n    }\n  }\n\n  /**\n   * Creates a new element with the specified type and class.\n   *\n   * @param elementType Type of html element.\n   * @param className ClassName of html element.\n   * @returns The created element.\n   */\n  private createElementWithClassname(elementType: string, className: string) {\n    const newElt = document.createElement(elementType);\n    newElt.className = className;\n    return newElt;\n  }\n\n  /**\n   * Binds an event listener to the specified element.\n   *\n   * @param element Specified element.\n   * @param eventName Name of the event to bind.\n   * @param callback Function to be called on specified event.\n   */\n  private bindEvent(\n    element: HTMLElement,\n    eventName: string,\n    callback: (e: Event) => void,\n  ) {\n    this.boundEvents.push(\n      Blockly.browserEvents.conditionalBind(element, eventName, this, callback),\n    );\n  }\n\n  private fireIntermediateChangeEvent(newValue: number[][]) {\n    if (this.getSourceBlock()) {\n      Blockly.Events.fire(\n        new (Blockly.Events.get(\n          Blockly.Events.BLOCK_FIELD_INTERMEDIATE_CHANGE,\n        ))(this.getSourceBlock(), this.name || null, this.getValue(), newValue),\n      );\n    }\n  }\n}\n\ninterface Buttons {\n  readonly randomize: boolean;\n  readonly clear: boolean;\n}\ninterface PixelColours {\n  readonly empty: string;\n  readonly filled: string;\n}\n\nexport interface FieldBitmapFromJsonConfig extends Blockly.FieldConfig {\n  value?: number[][];\n  width?: number;\n  height?: number;\n  buttons?: Buttons;\n  fieldHeight?: number;\n  colours?: PixelColours;\n}\n\nBlockly.fieldRegistry.register('field_bitmap', FieldBitmap);\n\n/**\n * CSS for bitmap field.\n */\nBlockly.Css.register(`\n.dropdownEditor {\n  align-items: center;\n  flex-direction: column;\n  display: flex;\n  justify-content: center;\n}\n.dropdownEditor.has-buttons {\n  margin-bottom: 20px;\n}\n.pixelContainer {\n  margin: 20px;\n}\n.pixelRow {\n  display: flex;\n  flex-direction: row;\n  padding: 0;\n  margin: 0;\n  height: ${DEFAULT_PIXEL_SIZE}\n}\n.pixelButton {\n  width: ${DEFAULT_PIXEL_SIZE}px;\n  height: ${DEFAULT_PIXEL_SIZE}px;\n  border: 1px solid #000;\n}\n.pixelDisplay {\n  white-space:pre-wrap;\n}\n.controlButton {\n  margin: 5px 0;\n}\n.blocklyDropDownContent.contains-bitmap-editor {\n  max-height: none;\n}\n`);\n"],"names":[],"sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/field-bitmap.ts\n");
/***/ }),
diff --git a/plugins/field-bitmap/test/index.html b/plugins/field-bitmap/test/index.html
index ac7f2245ac..3df35bae64 100644
--- a/plugins/field-bitmap/test/index.html
+++ b/plugins/field-bitmap/test/index.html
@@ -33,7 +33,7 @@
@blockly/field-bitmap Demo
A field that lets users input a pixel grid with their mouse.
- 4.1.0
+ 4.1.1
View code
View on npm