From eac5aa21c4ff5beb6f2b432efb37914c199b163f Mon Sep 17 00:00:00 2001 From: github-actions-bot Date: Thu, 2 May 2024 17:19:10 +0000 Subject: [PATCH] Updates --- plugins/continuous-toolbox/README.html | 2 +- plugins/continuous-toolbox/build/test_bundle.js | 2 +- plugins/continuous-toolbox/test/index.html | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/plugins/continuous-toolbox/README.html b/plugins/continuous-toolbox/README.html index 3dcc24ba45..1ba1748793 100644 --- a/plugins/continuous-toolbox/README.html +++ b/plugins/continuous-toolbox/README.html @@ -24,7 +24,7 @@
@blockly/continuous-toolbox Demo
A Blockly plugin that adds a continous-scrolling style toolbox and flyout
- 5.0.16 + 5.0.17 View code View on npm diff --git a/plugins/continuous-toolbox/build/test_bundle.js b/plugins/continuous-toolbox/build/test_bundle.js index b5791f46f6..d6060f3af7 100644 --- a/plugins/continuous-toolbox/build/test_bundle.js +++ b/plugins/continuous-toolbox/build/test_bundle.js @@ -279,7 +279,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __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 */ \"ContinuousFlyout\": () => (/* binding */ ContinuousFlyout)\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/* harmony import */ var _ContinuousToolbox__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./ContinuousToolbox */ \"./src/ContinuousToolbox.js\");\n/* harmony import */ var _ContinuousMetricsFlyout__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./ContinuousMetricsFlyout */ \"./src/ContinuousMetricsFlyout.js\");\n/**\n * @license\n * Copyright 2020 Google LLC\n * SPDX-License-Identifier: Apache-2.0\n */\n\n/**\n * @fileoverview Flyout that supports always-open continuous scrolling.\n */\n\n\n\n\n\n/**\n * Class for continuous flyout.\n */\nclass ContinuousFlyout extends blockly_core__WEBPACK_IMPORTED_MODULE_0__.VerticalFlyout {\n /** @override */\n constructor(workspaceOptions) {\n super(workspaceOptions);\n\n /**\n * List of scroll positions for each category.\n * @type {!Array<{name: string, position: !Object}>}\n */\n this.scrollPositions = [];\n\n /**\n * Target scroll position, used to smoothly scroll to a given category\n * location when selected.\n * @type {?number}\n */\n this.scrollTarget = null;\n\n /**\n * The percentage of the distance to the scrollTarget that should be\n * scrolled at a time. Lower values will produce a smoother, slower scroll.\n * @type {number}\n */\n this.scrollAnimationFraction = 0.3;\n\n /**\n * Whether to recycle blocks when refreshing the flyout. When false, do not\n * allow anything to be recycled. The default is to recycle.\n * @type {boolean}\n * @private\n */\n this.recyclingEnabled_ = true;\n\n this.workspace_.setMetricsManager(\n new _ContinuousMetricsFlyout__WEBPACK_IMPORTED_MODULE_2__.ContinuousFlyoutMetrics(this.workspace_, this),\n );\n\n this.workspace_.addChangeListener((e) => {\n if (e.type === blockly_core__WEBPACK_IMPORTED_MODULE_0__.Events.VIEWPORT_CHANGE) {\n this.selectCategoryByScrollPosition_(-this.workspace_.scrollY);\n }\n });\n\n this.autoClose = false;\n }\n\n /**\n * Gets parent toolbox.\n * Since we registered the ContinuousToolbox, we know that's its type.\n * @returns {!ContinuousToolbox} Toolbox that owns this flyout.\n * @private\n */\n getParentToolbox_() {\n const toolbox = this.targetWorkspace.getToolbox();\n return /** @type {!ContinuousToolbox} */ (toolbox);\n }\n\n /**\n * Records scroll position for each category in the toolbox.\n * The scroll position is determined by the coordinates of each category's\n * label after the entire flyout has been rendered.\n * @package\n */\n recordScrollPositions() {\n this.scrollPositions = [];\n const categoryLabels = this.buttons_.filter(\n (button) =>\n button.isLabel() &&\n this.getParentToolbox_().getCategoryByName(button.getButtonText()),\n );\n for (const [index, button] of categoryLabels.entries()) {\n if (button.isLabel()) {\n const position = button.getPosition();\n const adjustedPosition = new blockly_core__WEBPACK_IMPORTED_MODULE_0__.utils.Coordinate(\n position.x,\n position.y - this.labelGaps[index],\n );\n this.scrollPositions.push({\n name: button.getButtonText(),\n position: adjustedPosition,\n });\n }\n }\n }\n\n /**\n * Returns the scroll position for the given category name.\n * @param {string} name Category name.\n * @returns {?Object} Scroll position for given category, or null if not\n * found.\n * @package\n */\n getCategoryScrollPosition(name) {\n for (const scrollInfo of this.scrollPositions) {\n if (scrollInfo.name === name) {\n return scrollInfo.position;\n }\n }\n console.warn(`Scroll position not recorded for category ${name}`);\n return null;\n }\n\n /**\n * Selects an item in the toolbox based on the scroll position of the flyout.\n * @param {number} position Current scroll position of the workspace.\n * @private\n */\n selectCategoryByScrollPosition_(position) {\n // If we are currently auto-scrolling, due to selecting a category by\n // clicking on it, do not update the category selection.\n if (this.scrollTarget !== null) {\n return;\n }\n const scaledPosition = Math.round(position / this.workspace_.scale);\n // Traverse the array of scroll positions in reverse, so we can select the\n // furthest category that the scroll position is beyond.\n for (let i = this.scrollPositions.length - 1; i >= 0; i--) {\n const category = this.scrollPositions[i];\n if (scaledPosition >= category.position.y) {\n this.getParentToolbox_().selectCategoryByName(category.name);\n return;\n }\n }\n }\n\n /**\n * Scrolls flyout to given position.\n * @param {number} position The Y coordinate to scroll to.\n */\n scrollTo(position) {\n // Set the scroll target to either the scaled position or the lowest\n // possible scroll point, whichever is smaller.\n const metrics = this.workspace_.getMetrics();\n this.scrollTarget = Math.min(\n position * this.workspace_.scale,\n metrics.scrollHeight - metrics.viewHeight,\n );\n\n this.stepScrollAnimation_();\n }\n\n /**\n * Step the scrolling animation by scrolling a fraction of the way to\n * a scroll target, and request the next frame if necessary.\n * @private\n */\n stepScrollAnimation_() {\n if (this.scrollTarget === null) {\n return;\n }\n\n const currentScrollPos = -this.workspace_.scrollY;\n const diff = this.scrollTarget - currentScrollPos;\n if (Math.abs(diff) < 1) {\n this.workspace_.scrollbar.setY(this.scrollTarget);\n this.scrollTarget = null;\n return;\n }\n this.workspace_.scrollbar.setY(\n currentScrollPos + diff * this.scrollAnimationFraction,\n );\n\n requestAnimationFrame(this.stepScrollAnimation_.bind(this));\n }\n\n /**\n * Add additional padding to the bottom of the flyout if needed,\n * in order to make it possible to scroll to the top of the last category.\n * @param {!Blockly.MetricsManager.ContainerRegion} contentMetrics Content\n * metrics for the flyout.\n * @param {!Blockly.MetricsManager.ContainerRegion} viewMetrics View metrics\n * for the flyout.\n * @returns {number} Additional bottom padding.\n */\n calculateBottomPadding(contentMetrics, viewMetrics) {\n if (this.scrollPositions.length > 0) {\n const lastCategory =\n this.scrollPositions[this.scrollPositions.length - 1];\n const lastPosition = lastCategory.position.y * this.workspace_.scale;\n const lastCategoryHeight = contentMetrics.height - lastPosition;\n if (lastCategoryHeight < viewMetrics.height) {\n return viewMetrics.height - lastCategoryHeight;\n }\n }\n return 0;\n }\n\n /** @override */\n getX() {\n if (\n this.isVisible() &&\n this.targetWorkspace.toolboxPosition === this.toolboxPosition_ &&\n this.targetWorkspace.getToolbox() &&\n this.toolboxPosition_ !== blockly_core__WEBPACK_IMPORTED_MODULE_0__.utils.toolbox.Position.LEFT\n ) {\n // This makes it so blocks cannot go under the flyout in RTL mode.\n return this.targetWorkspace.getMetricsManager().getViewMetrics().width;\n }\n\n return super.getX();\n }\n\n /**\n * @override\n */\n show(flyoutDef) {\n super.show(flyoutDef);\n this.recordScrollPositions();\n this.workspace_.resizeContents();\n this.selectCategoryByScrollPosition_(0);\n }\n\n /**\n * Determine if this block can be recycled in the flyout. Blocks that have no\n * variables and are not dynamic shadows can be recycled.\n * @param {!Blockly.BlockSvg} block The block to attempt to recycle.\n * @returns {boolean} True if the block can be recycled.\n * @protected\n */\n blockIsRecyclable_(block) {\n if (!this.recyclingEnabled_) {\n return false;\n }\n\n // If the block needs to parse mutations, never recycle.\n if (block.mutationToDom && block.domToMutation) {\n return false;\n }\n\n if (!block.isEnabled()) {\n return false;\n }\n\n for (const input of block.inputList) {\n for (const field of input.fieldRow) {\n // No variables.\n if (field.referencesVariables()) {\n return false;\n }\n if (field instanceof blockly_core__WEBPACK_IMPORTED_MODULE_0__.FieldDropdown) {\n if (field.isOptionListDynamic()) {\n return false;\n }\n }\n }\n // Check children.\n if (input.connection) {\n const targetBlock =\n /** @type {Blockly.BlockSvg} */\n (input.connection.targetBlock());\n if (targetBlock && !this.blockIsRecyclable_(targetBlock)) {\n return false;\n }\n }\n }\n return true;\n }\n\n /**\n * Sets the function used to determine whether a block is recyclable.\n * @param {function(!Blockly.BlockSvg):boolean} func The function used to\n * determine if a block is recyclable.\n * @public\n */\n setBlockIsRecyclable(func) {\n this.blockIsRecyclable_ = func;\n }\n\n /**\n * Set whether the flyout can recycle blocks.\n * @param {boolean} isEnabled True to allow blocks to be recycled, false\n * otherwise.\n * @public\n */\n setRecyclingEnabled(isEnabled) {\n this.recyclingEnabled_ = isEnabled;\n }\n\n /**\n * Lay out the blocks in the flyout.\n * @param {Array} contents The blocks and buttons to lay out.\n * @param {Array} gaps The visible gaps between blocks.\n */\n layout_(contents, gaps) {\n super.layout_(contents, gaps);\n this.labelGaps = [];\n for (const [index, item] of contents.entries()) {\n if (item.type === 'button' && item.button.isLabel()) {\n this.labelGaps.push(gaps[index - 1] ?? this.MARGIN);\n }\n }\n }\n}\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"./src/ContinuousFlyout.js.js","mappings":";;;;;;;;AAAA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEwC;AACc;AACY;;AAElE;AACA;AACA;AACO,+BAA+B,wDAAsB;AAC5D;AACA;AACA;;AAEA;AACA;AACA,cAAc,QAAQ,gCAAgC;AACtD;AACA;;AAEA;AACA;AACA;AACA,cAAc;AACd;AACA;;AAEA;AACA;AACA;AACA,cAAc;AACd;AACA;;AAEA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;;AAEA;AACA,UAAU,6EAAuB;AACjC;;AAEA;AACA,qBAAqB,gEAA8B;AACnD;AACA;AACA,KAAK;;AAEL;AACA;;AAEA;AACA;AACA;AACA,eAAe,oBAAoB;AACnC;AACA;AACA;AACA;AACA,sBAAsB,oBAAoB;AAC1C;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qCAAqC,0DAAwB;AAC7D;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;;AAEA;AACA;AACA,aAAa,QAAQ;AACrB,eAAe,SAAS;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,8DAA8D,KAAK;AACnE;AACA;;AAEA;AACA;AACA,aAAa,QAAQ;AACrB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kDAAkD,QAAQ;AAC1D;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA,aAAa,QAAQ;AACrB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA,aAAa,yCAAyC;AACtD;AACA,aAAa,yCAAyC;AACtD;AACA,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,gCAAgC,qEAAmC;AACnE;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,aAAa,mBAAmB;AAChC,eAAe,SAAS;AACxB;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,uDAAqB;AAClD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB,kBAAkB;AACvC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA,aAAa,qCAAqC;AAClD;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA,aAAa,SAAS;AACtB;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA,aAAa,kCAAkC;AAC/C,aAAa,eAAe;AAC5B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","sources":["webpack://@blockly/continuous-toolbox/./src/ContinuousFlyout.js?bcfc"],"sourcesContent":["/**\n * @license\n * Copyright 2020 Google LLC\n * SPDX-License-Identifier: Apache-2.0\n */\n\n/**\n * @fileoverview Flyout that supports always-open continuous scrolling.\n */\n\nimport * as Blockly from 'blockly/core';\nimport {ContinuousToolbox} from './ContinuousToolbox';\nimport {ContinuousFlyoutMetrics} from './ContinuousMetricsFlyout';\n\n/**\n * Class for continuous flyout.\n */\nexport class ContinuousFlyout extends Blockly.VerticalFlyout {\n  /** @override */\n  constructor(workspaceOptions) {\n    super(workspaceOptions);\n\n    /**\n     * List of scroll positions for each category.\n     * @type {!Array<{name: string, position: !Object}>}\n     */\n    this.scrollPositions = [];\n\n    /**\n     * Target scroll position, used to smoothly scroll to a given category\n     * location when selected.\n     * @type {?number}\n     */\n    this.scrollTarget = null;\n\n    /**\n     * The percentage of the distance to the scrollTarget that should be\n     * scrolled at a time. Lower values will produce a smoother, slower scroll.\n     * @type {number}\n     */\n    this.scrollAnimationFraction = 0.3;\n\n    /**\n     * Whether to recycle blocks when refreshing the flyout. When false, do not\n     * allow anything to be recycled. The default is to recycle.\n     * @type {boolean}\n     * @private\n     */\n    this.recyclingEnabled_ = true;\n\n    this.workspace_.setMetricsManager(\n      new ContinuousFlyoutMetrics(this.workspace_, this),\n    );\n\n    this.workspace_.addChangeListener((e) => {\n      if (e.type === Blockly.Events.VIEWPORT_CHANGE) {\n        this.selectCategoryByScrollPosition_(-this.workspace_.scrollY);\n      }\n    });\n\n    this.autoClose = false;\n  }\n\n  /**\n   * Gets parent toolbox.\n   * Since we registered the ContinuousToolbox, we know that's its type.\n   * @returns {!ContinuousToolbox} Toolbox that owns this flyout.\n   * @private\n   */\n  getParentToolbox_() {\n    const toolbox = this.targetWorkspace.getToolbox();\n    return /** @type {!ContinuousToolbox} */ (toolbox);\n  }\n\n  /**\n   * Records scroll position for each category in the toolbox.\n   * The scroll position is determined by the coordinates of each category's\n   * label after the entire flyout has been rendered.\n   * @package\n   */\n  recordScrollPositions() {\n    this.scrollPositions = [];\n    const categoryLabels = this.buttons_.filter(\n      (button) =>\n        button.isLabel() &&\n        this.getParentToolbox_().getCategoryByName(button.getButtonText()),\n    );\n    for (const [index, button] of categoryLabels.entries()) {\n      if (button.isLabel()) {\n        const position = button.getPosition();\n        const adjustedPosition = new Blockly.utils.Coordinate(\n          position.x,\n          position.y - this.labelGaps[index],\n        );\n        this.scrollPositions.push({\n          name: button.getButtonText(),\n          position: adjustedPosition,\n        });\n      }\n    }\n  }\n\n  /**\n   * Returns the scroll position for the given category name.\n   * @param {string} name Category name.\n   * @returns {?Object} Scroll position for given category, or null if not\n   *     found.\n   * @package\n   */\n  getCategoryScrollPosition(name) {\n    for (const scrollInfo of this.scrollPositions) {\n      if (scrollInfo.name === name) {\n        return scrollInfo.position;\n      }\n    }\n    console.warn(`Scroll position not recorded for category ${name}`);\n    return null;\n  }\n\n  /**\n   * Selects an item in the toolbox based on the scroll position of the flyout.\n   * @param {number} position Current scroll position of the workspace.\n   * @private\n   */\n  selectCategoryByScrollPosition_(position) {\n    // If we are currently auto-scrolling, due to selecting a category by\n    // clicking on it, do not update the category selection.\n    if (this.scrollTarget !== null) {\n      return;\n    }\n    const scaledPosition = Math.round(position / this.workspace_.scale);\n    // Traverse the array of scroll positions in reverse, so we can select the\n    // furthest category that the scroll position is beyond.\n    for (let i = this.scrollPositions.length - 1; i >= 0; i--) {\n      const category = this.scrollPositions[i];\n      if (scaledPosition >= category.position.y) {\n        this.getParentToolbox_().selectCategoryByName(category.name);\n        return;\n      }\n    }\n  }\n\n  /**\n   * Scrolls flyout to given position.\n   * @param {number} position The Y coordinate to scroll to.\n   */\n  scrollTo(position) {\n    // Set the scroll target to either the scaled position or the lowest\n    // possible scroll point, whichever is smaller.\n    const metrics = this.workspace_.getMetrics();\n    this.scrollTarget = Math.min(\n      position * this.workspace_.scale,\n      metrics.scrollHeight - metrics.viewHeight,\n    );\n\n    this.stepScrollAnimation_();\n  }\n\n  /**\n   * Step the scrolling animation by scrolling a fraction of the way to\n   * a scroll target, and request the next frame if necessary.\n   * @private\n   */\n  stepScrollAnimation_() {\n    if (this.scrollTarget === null) {\n      return;\n    }\n\n    const currentScrollPos = -this.workspace_.scrollY;\n    const diff = this.scrollTarget - currentScrollPos;\n    if (Math.abs(diff) < 1) {\n      this.workspace_.scrollbar.setY(this.scrollTarget);\n      this.scrollTarget = null;\n      return;\n    }\n    this.workspace_.scrollbar.setY(\n      currentScrollPos + diff * this.scrollAnimationFraction,\n    );\n\n    requestAnimationFrame(this.stepScrollAnimation_.bind(this));\n  }\n\n  /**\n   * Add additional padding to the bottom of the flyout if needed,\n   * in order to make it possible to scroll to the top of the last category.\n   * @param {!Blockly.MetricsManager.ContainerRegion} contentMetrics Content\n   *    metrics for the flyout.\n   * @param {!Blockly.MetricsManager.ContainerRegion} viewMetrics View metrics\n   *    for the flyout.\n   * @returns {number} Additional bottom padding.\n   */\n  calculateBottomPadding(contentMetrics, viewMetrics) {\n    if (this.scrollPositions.length > 0) {\n      const lastCategory =\n        this.scrollPositions[this.scrollPositions.length - 1];\n      const lastPosition = lastCategory.position.y * this.workspace_.scale;\n      const lastCategoryHeight = contentMetrics.height - lastPosition;\n      if (lastCategoryHeight < viewMetrics.height) {\n        return viewMetrics.height - lastCategoryHeight;\n      }\n    }\n    return 0;\n  }\n\n  /** @override */\n  getX() {\n    if (\n      this.isVisible() &&\n      this.targetWorkspace.toolboxPosition === this.toolboxPosition_ &&\n      this.targetWorkspace.getToolbox() &&\n      this.toolboxPosition_ !== Blockly.utils.toolbox.Position.LEFT\n    ) {\n      // This makes it so blocks cannot go under the flyout in RTL mode.\n      return this.targetWorkspace.getMetricsManager().getViewMetrics().width;\n    }\n\n    return super.getX();\n  }\n\n  /**\n   * @override\n   */\n  show(flyoutDef) {\n    super.show(flyoutDef);\n    this.recordScrollPositions();\n    this.workspace_.resizeContents();\n    this.selectCategoryByScrollPosition_(0);\n  }\n\n  /**\n   * Determine if this block can be recycled in the flyout.  Blocks that have no\n   * variables and are not dynamic shadows can be recycled.\n   * @param {!Blockly.BlockSvg} block The block to attempt to recycle.\n   * @returns {boolean} True if the block can be recycled.\n   * @protected\n   */\n  blockIsRecyclable_(block) {\n    if (!this.recyclingEnabled_) {\n      return false;\n    }\n\n    // If the block needs to parse mutations, never recycle.\n    if (block.mutationToDom && block.domToMutation) {\n      return false;\n    }\n\n    if (!block.isEnabled()) {\n      return false;\n    }\n\n    for (const input of block.inputList) {\n      for (const field of input.fieldRow) {\n        // No variables.\n        if (field.referencesVariables()) {\n          return false;\n        }\n        if (field instanceof Blockly.FieldDropdown) {\n          if (field.isOptionListDynamic()) {\n            return false;\n          }\n        }\n      }\n      // Check children.\n      if (input.connection) {\n        const targetBlock =\n          /** @type {Blockly.BlockSvg} */\n          (input.connection.targetBlock());\n        if (targetBlock && !this.blockIsRecyclable_(targetBlock)) {\n          return false;\n        }\n      }\n    }\n    return true;\n  }\n\n  /**\n   * Sets the function used to determine whether a block is recyclable.\n   * @param {function(!Blockly.BlockSvg):boolean} func The function used to\n   *     determine if a block is recyclable.\n   * @public\n   */\n  setBlockIsRecyclable(func) {\n    this.blockIsRecyclable_ = func;\n  }\n\n  /**\n   * Set whether the flyout can recycle blocks.\n   * @param {boolean} isEnabled True to allow blocks to be recycled, false\n   *     otherwise.\n   * @public\n   */\n  setRecyclingEnabled(isEnabled) {\n    this.recyclingEnabled_ = isEnabled;\n  }\n\n  /**\n   * Lay out the blocks in the flyout.\n   * @param {Array<Blockly.Flyout.FlyoutItem>} contents The blocks and buttons to lay out.\n   * @param {Array<number>} gaps The visible gaps between blocks.\n   */\n  layout_(contents, gaps) {\n    super.layout_(contents, gaps);\n    this.labelGaps = [];\n    for (const [index, item] of contents.entries()) {\n      if (item.type === 'button' && item.button.isLabel()) {\n        this.labelGaps.push(gaps[index - 1] ?? this.MARGIN);\n      }\n    }\n  }\n}\n"],"names":[],"sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/ContinuousFlyout.js\n"); +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"ContinuousFlyout\": () => (/* binding */ ContinuousFlyout)\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/* harmony import */ var _ContinuousToolbox__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./ContinuousToolbox */ \"./src/ContinuousToolbox.js\");\n/* harmony import */ var _ContinuousMetricsFlyout__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./ContinuousMetricsFlyout */ \"./src/ContinuousMetricsFlyout.js\");\n/**\n * @license\n * Copyright 2020 Google LLC\n * SPDX-License-Identifier: Apache-2.0\n */\n\n/**\n * @fileoverview Flyout that supports always-open continuous scrolling.\n */\n\n\n\n\n\n/**\n * Class for continuous flyout.\n */\nclass ContinuousFlyout extends blockly_core__WEBPACK_IMPORTED_MODULE_0__.VerticalFlyout {\n /** @override */\n constructor(workspaceOptions) {\n super(workspaceOptions);\n\n /**\n * List of scroll positions for each category.\n * @type {!Array<{name: string, position: !Object}>}\n */\n this.scrollPositions = [];\n\n /**\n * Target scroll position, used to smoothly scroll to a given category\n * location when selected.\n * @type {?number}\n */\n this.scrollTarget = null;\n\n /**\n * The percentage of the distance to the scrollTarget that should be\n * scrolled at a time. Lower values will produce a smoother, slower scroll.\n * @type {number}\n */\n this.scrollAnimationFraction = 0.3;\n\n /**\n * Whether to recycle blocks when refreshing the flyout. When false, do not\n * allow anything to be recycled. The default is to recycle.\n * @type {boolean}\n * @private\n */\n this.recyclingEnabled_ = true;\n\n this.workspace_.setMetricsManager(\n new _ContinuousMetricsFlyout__WEBPACK_IMPORTED_MODULE_2__.ContinuousFlyoutMetrics(this.workspace_, this),\n );\n\n this.workspace_.addChangeListener((e) => {\n if (e.type === blockly_core__WEBPACK_IMPORTED_MODULE_0__.Events.VIEWPORT_CHANGE) {\n this.selectCategoryByScrollPosition_(-this.workspace_.scrollY);\n }\n });\n\n this.autoClose = false;\n }\n\n /**\n * Gets parent toolbox.\n * Since we registered the ContinuousToolbox, we know that's its type.\n * @returns {!ContinuousToolbox} Toolbox that owns this flyout.\n * @private\n */\n getParentToolbox_() {\n const toolbox = this.targetWorkspace.getToolbox();\n return /** @type {!ContinuousToolbox} */ (toolbox);\n }\n\n /**\n * Records scroll position for each category in the toolbox.\n * The scroll position is determined by the coordinates of each category's\n * label after the entire flyout has been rendered.\n * @package\n */\n recordScrollPositions() {\n this.scrollPositions = [];\n const categoryLabels = this.buttons_.filter(\n (button) =>\n button.isLabel() &&\n this.getParentToolbox_().getCategoryByName(button.getButtonText()),\n );\n for (const [index, button] of categoryLabels.entries()) {\n if (button.isLabel()) {\n const position = button.getPosition();\n const adjustedPosition = new blockly_core__WEBPACK_IMPORTED_MODULE_0__.utils.Coordinate(\n position.x,\n position.y - this.labelGaps[index],\n );\n this.scrollPositions.push({\n name: button.getButtonText(),\n position: adjustedPosition,\n });\n }\n }\n }\n\n /**\n * Returns the scroll position for the given category name.\n * @param {string} name Category name.\n * @returns {?Object} Scroll position for given category, or null if not\n * found.\n * @package\n */\n getCategoryScrollPosition(name) {\n for (const scrollInfo of this.scrollPositions) {\n if (scrollInfo.name === name) {\n return scrollInfo.position;\n }\n }\n console.warn(`Scroll position not recorded for category ${name}`);\n return null;\n }\n\n /**\n * Selects an item in the toolbox based on the scroll position of the flyout.\n * @param {number} position Current scroll position of the workspace.\n * @private\n */\n selectCategoryByScrollPosition_(position) {\n // If we are currently auto-scrolling, due to selecting a category by\n // clicking on it, do not update the category selection.\n if (this.scrollTarget !== null) {\n return;\n }\n const scaledPosition = Math.round(position / this.workspace_.scale);\n // Traverse the array of scroll positions in reverse, so we can select the\n // furthest category that the scroll position is beyond.\n for (let i = this.scrollPositions.length - 1; i >= 0; i--) {\n const category = this.scrollPositions[i];\n if (scaledPosition >= category.position.y) {\n this.getParentToolbox_().selectCategoryByName(category.name);\n return;\n }\n }\n }\n\n /**\n * Scrolls flyout to given position.\n * @param {number} position The Y coordinate to scroll to.\n */\n scrollTo(position) {\n // Set the scroll target to either the scaled position or the lowest\n // possible scroll point, whichever is smaller.\n const metrics = this.workspace_.getMetrics();\n this.scrollTarget = Math.min(\n position * this.workspace_.scale,\n metrics.scrollHeight - metrics.viewHeight,\n );\n\n this.stepScrollAnimation_();\n }\n\n /**\n * Step the scrolling animation by scrolling a fraction of the way to\n * a scroll target, and request the next frame if necessary.\n * @private\n */\n stepScrollAnimation_() {\n if (this.scrollTarget === null) {\n return;\n }\n\n const currentScrollPos = -this.workspace_.scrollY;\n const diff = this.scrollTarget - currentScrollPos;\n if (Math.abs(diff) < 1) {\n this.workspace_.scrollbar.setY(this.scrollTarget);\n this.scrollTarget = null;\n return;\n }\n this.workspace_.scrollbar.setY(\n currentScrollPos + diff * this.scrollAnimationFraction,\n );\n\n requestAnimationFrame(this.stepScrollAnimation_.bind(this));\n }\n\n /**\n * Add additional padding to the bottom of the flyout if needed,\n * in order to make it possible to scroll to the top of the last category.\n * @param {!Blockly.MetricsManager.ContainerRegion} contentMetrics Content\n * metrics for the flyout.\n * @param {!Blockly.MetricsManager.ContainerRegion} viewMetrics View metrics\n * for the flyout.\n * @returns {number} Additional bottom padding.\n */\n calculateBottomPadding(contentMetrics, viewMetrics) {\n if (this.scrollPositions.length > 0) {\n const lastCategory =\n this.scrollPositions[this.scrollPositions.length - 1];\n const lastPosition = lastCategory.position.y * this.workspace_.scale;\n const lastCategoryHeight = contentMetrics.height - lastPosition;\n if (lastCategoryHeight < viewMetrics.height) {\n return viewMetrics.height - lastCategoryHeight;\n }\n }\n return 0;\n }\n\n /** @override */\n getX() {\n if (\n this.isVisible() &&\n this.targetWorkspace.toolboxPosition === this.toolboxPosition_ &&\n this.targetWorkspace.getToolbox() &&\n this.toolboxPosition_ !== blockly_core__WEBPACK_IMPORTED_MODULE_0__.utils.toolbox.Position.LEFT\n ) {\n // This makes it so blocks cannot go under the flyout in RTL mode.\n return this.targetWorkspace.getMetricsManager().getViewMetrics().width;\n }\n\n return super.getX();\n }\n\n /**\n * @override\n */\n show(flyoutDef) {\n super.show(flyoutDef);\n this.recordScrollPositions();\n this.workspace_.resizeContents();\n if (!this.getParentToolbox_().getSelectedItem()) {\n this.selectCategoryByScrollPosition_(0);\n }\n }\n\n /**\n * Determine if this block can be recycled in the flyout. Blocks that have no\n * variables and are not dynamic shadows can be recycled.\n * @param {!Blockly.BlockSvg} block The block to attempt to recycle.\n * @returns {boolean} True if the block can be recycled.\n * @protected\n */\n blockIsRecyclable_(block) {\n if (!this.recyclingEnabled_) {\n return false;\n }\n\n // If the block needs to parse mutations, never recycle.\n if (block.mutationToDom && block.domToMutation) {\n return false;\n }\n\n if (!block.isEnabled()) {\n return false;\n }\n\n for (const input of block.inputList) {\n for (const field of input.fieldRow) {\n // No variables.\n if (field.referencesVariables()) {\n return false;\n }\n if (field instanceof blockly_core__WEBPACK_IMPORTED_MODULE_0__.FieldDropdown) {\n if (field.isOptionListDynamic()) {\n return false;\n }\n }\n }\n // Check children.\n if (input.connection) {\n const targetBlock =\n /** @type {Blockly.BlockSvg} */\n (input.connection.targetBlock());\n if (targetBlock && !this.blockIsRecyclable_(targetBlock)) {\n return false;\n }\n }\n }\n return true;\n }\n\n /**\n * Sets the function used to determine whether a block is recyclable.\n * @param {function(!Blockly.BlockSvg):boolean} func The function used to\n * determine if a block is recyclable.\n * @public\n */\n setBlockIsRecyclable(func) {\n this.blockIsRecyclable_ = func;\n }\n\n /**\n * Set whether the flyout can recycle blocks.\n * @param {boolean} isEnabled True to allow blocks to be recycled, false\n * otherwise.\n * @public\n */\n setRecyclingEnabled(isEnabled) {\n this.recyclingEnabled_ = isEnabled;\n }\n\n /**\n * Lay out the blocks in the flyout.\n * @param {Array} contents The blocks and buttons to lay out.\n * @param {Array} gaps The visible gaps between blocks.\n */\n layout_(contents, gaps) {\n super.layout_(contents, gaps);\n this.labelGaps = [];\n for (const [index, item] of contents.entries()) {\n if (item.type === 'button' && item.button.isLabel()) {\n this.labelGaps.push(gaps[index - 1] ?? this.MARGIN);\n }\n }\n }\n}\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"./src/ContinuousFlyout.js.js","mappings":";;;;;;;;AAAA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEwC;AACc;AACY;;AAElE;AACA;AACA;AACO,+BAA+B,wDAAsB;AAC5D;AACA;AACA;;AAEA;AACA;AACA,cAAc,QAAQ,gCAAgC;AACtD;AACA;;AAEA;AACA;AACA;AACA,cAAc;AACd;AACA;;AAEA;AACA;AACA;AACA,cAAc;AACd;AACA;;AAEA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;;AAEA;AACA,UAAU,6EAAuB;AACjC;;AAEA;AACA,qBAAqB,gEAA8B;AACnD;AACA;AACA,KAAK;;AAEL;AACA;;AAEA;AACA;AACA;AACA,eAAe,oBAAoB;AACnC;AACA;AACA;AACA;AACA,sBAAsB,oBAAoB;AAC1C;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qCAAqC,0DAAwB;AAC7D;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;;AAEA;AACA;AACA,aAAa,QAAQ;AACrB,eAAe,SAAS;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,8DAA8D,KAAK;AACnE;AACA;;AAEA;AACA;AACA,aAAa,QAAQ;AACrB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kDAAkD,QAAQ;AAC1D;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA,aAAa,QAAQ;AACrB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA,aAAa,yCAAyC;AACtD;AACA,aAAa,yCAAyC;AACtD;AACA,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,gCAAgC,qEAAmC;AACnE;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,aAAa,mBAAmB;AAChC,eAAe,SAAS;AACxB;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,uDAAqB;AAClD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB,kBAAkB;AACvC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA,aAAa,qCAAqC;AAClD;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA,aAAa,SAAS;AACtB;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA,aAAa,kCAAkC;AAC/C,aAAa,eAAe;AAC5B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","sources":["webpack://@blockly/continuous-toolbox/./src/ContinuousFlyout.js?bcfc"],"sourcesContent":["/**\n * @license\n * Copyright 2020 Google LLC\n * SPDX-License-Identifier: Apache-2.0\n */\n\n/**\n * @fileoverview Flyout that supports always-open continuous scrolling.\n */\n\nimport * as Blockly from 'blockly/core';\nimport {ContinuousToolbox} from './ContinuousToolbox';\nimport {ContinuousFlyoutMetrics} from './ContinuousMetricsFlyout';\n\n/**\n * Class for continuous flyout.\n */\nexport class ContinuousFlyout extends Blockly.VerticalFlyout {\n  /** @override */\n  constructor(workspaceOptions) {\n    super(workspaceOptions);\n\n    /**\n     * List of scroll positions for each category.\n     * @type {!Array<{name: string, position: !Object}>}\n     */\n    this.scrollPositions = [];\n\n    /**\n     * Target scroll position, used to smoothly scroll to a given category\n     * location when selected.\n     * @type {?number}\n     */\n    this.scrollTarget = null;\n\n    /**\n     * The percentage of the distance to the scrollTarget that should be\n     * scrolled at a time. Lower values will produce a smoother, slower scroll.\n     * @type {number}\n     */\n    this.scrollAnimationFraction = 0.3;\n\n    /**\n     * Whether to recycle blocks when refreshing the flyout. When false, do not\n     * allow anything to be recycled. The default is to recycle.\n     * @type {boolean}\n     * @private\n     */\n    this.recyclingEnabled_ = true;\n\n    this.workspace_.setMetricsManager(\n      new ContinuousFlyoutMetrics(this.workspace_, this),\n    );\n\n    this.workspace_.addChangeListener((e) => {\n      if (e.type === Blockly.Events.VIEWPORT_CHANGE) {\n        this.selectCategoryByScrollPosition_(-this.workspace_.scrollY);\n      }\n    });\n\n    this.autoClose = false;\n  }\n\n  /**\n   * Gets parent toolbox.\n   * Since we registered the ContinuousToolbox, we know that's its type.\n   * @returns {!ContinuousToolbox} Toolbox that owns this flyout.\n   * @private\n   */\n  getParentToolbox_() {\n    const toolbox = this.targetWorkspace.getToolbox();\n    return /** @type {!ContinuousToolbox} */ (toolbox);\n  }\n\n  /**\n   * Records scroll position for each category in the toolbox.\n   * The scroll position is determined by the coordinates of each category's\n   * label after the entire flyout has been rendered.\n   * @package\n   */\n  recordScrollPositions() {\n    this.scrollPositions = [];\n    const categoryLabels = this.buttons_.filter(\n      (button) =>\n        button.isLabel() &&\n        this.getParentToolbox_().getCategoryByName(button.getButtonText()),\n    );\n    for (const [index, button] of categoryLabels.entries()) {\n      if (button.isLabel()) {\n        const position = button.getPosition();\n        const adjustedPosition = new Blockly.utils.Coordinate(\n          position.x,\n          position.y - this.labelGaps[index],\n        );\n        this.scrollPositions.push({\n          name: button.getButtonText(),\n          position: adjustedPosition,\n        });\n      }\n    }\n  }\n\n  /**\n   * Returns the scroll position for the given category name.\n   * @param {string} name Category name.\n   * @returns {?Object} Scroll position for given category, or null if not\n   *     found.\n   * @package\n   */\n  getCategoryScrollPosition(name) {\n    for (const scrollInfo of this.scrollPositions) {\n      if (scrollInfo.name === name) {\n        return scrollInfo.position;\n      }\n    }\n    console.warn(`Scroll position not recorded for category ${name}`);\n    return null;\n  }\n\n  /**\n   * Selects an item in the toolbox based on the scroll position of the flyout.\n   * @param {number} position Current scroll position of the workspace.\n   * @private\n   */\n  selectCategoryByScrollPosition_(position) {\n    // If we are currently auto-scrolling, due to selecting a category by\n    // clicking on it, do not update the category selection.\n    if (this.scrollTarget !== null) {\n      return;\n    }\n    const scaledPosition = Math.round(position / this.workspace_.scale);\n    // Traverse the array of scroll positions in reverse, so we can select the\n    // furthest category that the scroll position is beyond.\n    for (let i = this.scrollPositions.length - 1; i >= 0; i--) {\n      const category = this.scrollPositions[i];\n      if (scaledPosition >= category.position.y) {\n        this.getParentToolbox_().selectCategoryByName(category.name);\n        return;\n      }\n    }\n  }\n\n  /**\n   * Scrolls flyout to given position.\n   * @param {number} position The Y coordinate to scroll to.\n   */\n  scrollTo(position) {\n    // Set the scroll target to either the scaled position or the lowest\n    // possible scroll point, whichever is smaller.\n    const metrics = this.workspace_.getMetrics();\n    this.scrollTarget = Math.min(\n      position * this.workspace_.scale,\n      metrics.scrollHeight - metrics.viewHeight,\n    );\n\n    this.stepScrollAnimation_();\n  }\n\n  /**\n   * Step the scrolling animation by scrolling a fraction of the way to\n   * a scroll target, and request the next frame if necessary.\n   * @private\n   */\n  stepScrollAnimation_() {\n    if (this.scrollTarget === null) {\n      return;\n    }\n\n    const currentScrollPos = -this.workspace_.scrollY;\n    const diff = this.scrollTarget - currentScrollPos;\n    if (Math.abs(diff) < 1) {\n      this.workspace_.scrollbar.setY(this.scrollTarget);\n      this.scrollTarget = null;\n      return;\n    }\n    this.workspace_.scrollbar.setY(\n      currentScrollPos + diff * this.scrollAnimationFraction,\n    );\n\n    requestAnimationFrame(this.stepScrollAnimation_.bind(this));\n  }\n\n  /**\n   * Add additional padding to the bottom of the flyout if needed,\n   * in order to make it possible to scroll to the top of the last category.\n   * @param {!Blockly.MetricsManager.ContainerRegion} contentMetrics Content\n   *    metrics for the flyout.\n   * @param {!Blockly.MetricsManager.ContainerRegion} viewMetrics View metrics\n   *    for the flyout.\n   * @returns {number} Additional bottom padding.\n   */\n  calculateBottomPadding(contentMetrics, viewMetrics) {\n    if (this.scrollPositions.length > 0) {\n      const lastCategory =\n        this.scrollPositions[this.scrollPositions.length - 1];\n      const lastPosition = lastCategory.position.y * this.workspace_.scale;\n      const lastCategoryHeight = contentMetrics.height - lastPosition;\n      if (lastCategoryHeight < viewMetrics.height) {\n        return viewMetrics.height - lastCategoryHeight;\n      }\n    }\n    return 0;\n  }\n\n  /** @override */\n  getX() {\n    if (\n      this.isVisible() &&\n      this.targetWorkspace.toolboxPosition === this.toolboxPosition_ &&\n      this.targetWorkspace.getToolbox() &&\n      this.toolboxPosition_ !== Blockly.utils.toolbox.Position.LEFT\n    ) {\n      // This makes it so blocks cannot go under the flyout in RTL mode.\n      return this.targetWorkspace.getMetricsManager().getViewMetrics().width;\n    }\n\n    return super.getX();\n  }\n\n  /**\n   * @override\n   */\n  show(flyoutDef) {\n    super.show(flyoutDef);\n    this.recordScrollPositions();\n    this.workspace_.resizeContents();\n    if (!this.getParentToolbox_().getSelectedItem()) {\n      this.selectCategoryByScrollPosition_(0);\n    }\n  }\n\n  /**\n   * Determine if this block can be recycled in the flyout.  Blocks that have no\n   * variables and are not dynamic shadows can be recycled.\n   * @param {!Blockly.BlockSvg} block The block to attempt to recycle.\n   * @returns {boolean} True if the block can be recycled.\n   * @protected\n   */\n  blockIsRecyclable_(block) {\n    if (!this.recyclingEnabled_) {\n      return false;\n    }\n\n    // If the block needs to parse mutations, never recycle.\n    if (block.mutationToDom && block.domToMutation) {\n      return false;\n    }\n\n    if (!block.isEnabled()) {\n      return false;\n    }\n\n    for (const input of block.inputList) {\n      for (const field of input.fieldRow) {\n        // No variables.\n        if (field.referencesVariables()) {\n          return false;\n        }\n        if (field instanceof Blockly.FieldDropdown) {\n          if (field.isOptionListDynamic()) {\n            return false;\n          }\n        }\n      }\n      // Check children.\n      if (input.connection) {\n        const targetBlock =\n          /** @type {Blockly.BlockSvg} */\n          (input.connection.targetBlock());\n        if (targetBlock && !this.blockIsRecyclable_(targetBlock)) {\n          return false;\n        }\n      }\n    }\n    return true;\n  }\n\n  /**\n   * Sets the function used to determine whether a block is recyclable.\n   * @param {function(!Blockly.BlockSvg):boolean} func The function used to\n   *     determine if a block is recyclable.\n   * @public\n   */\n  setBlockIsRecyclable(func) {\n    this.blockIsRecyclable_ = func;\n  }\n\n  /**\n   * Set whether the flyout can recycle blocks.\n   * @param {boolean} isEnabled True to allow blocks to be recycled, false\n   *     otherwise.\n   * @public\n   */\n  setRecyclingEnabled(isEnabled) {\n    this.recyclingEnabled_ = isEnabled;\n  }\n\n  /**\n   * Lay out the blocks in the flyout.\n   * @param {Array<Blockly.Flyout.FlyoutItem>} contents The blocks and buttons to lay out.\n   * @param {Array<number>} gaps The visible gaps between blocks.\n   */\n  layout_(contents, gaps) {\n    super.layout_(contents, gaps);\n    this.labelGaps = [];\n    for (const [index, item] of contents.entries()) {\n      if (item.type === 'button' && item.button.isLabel()) {\n        this.labelGaps.push(gaps[index - 1] ?? this.MARGIN);\n      }\n    }\n  }\n}\n"],"names":[],"sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/ContinuousFlyout.js\n"); /***/ }), diff --git a/plugins/continuous-toolbox/test/index.html b/plugins/continuous-toolbox/test/index.html index be07dd5e8d..ac03afd43c 100644 --- a/plugins/continuous-toolbox/test/index.html +++ b/plugins/continuous-toolbox/test/index.html @@ -33,7 +33,7 @@
@blockly/continuous-toolbox Demo
A Blockly plugin that adds a continous-scrolling style toolbox and flyout
- 5.0.16 + 5.0.17 View code View on npm