waves/public/assets/g/turbowarp/main/js/addon-default-entry.js
2025-04-09 17:11:14 -05:00

8789 lines
306 KiB
JavaScript

(window["webpackJsonpGUI"] = window["webpackJsonpGUI"] || []).push([["addon-default-entry"],{
/***/ "./node_modules/css-loader/index.js!./src/addons/addons/color-picker/style.css":
/*!****************************************************************************!*\
!*** ./node_modules/css-loader!./src/addons/addons/color-picker/style.css ***!
\****************************************************************************/
/*! no static exports found */
/***/ (function(module, exports, __webpack_require__) {
exports = module.exports = __webpack_require__(/*! ../../../../node_modules/css-loader/lib/css-base.js */ "./node_modules/css-loader/lib/css-base.js")(false);
// imports
// module
exports.push([module.i, ".sa-color-picker {\n display: flex;\n}\n\n.sa-color-picker-code {\n margin: 8px 0;\n}\n\n.sa-color-picker-paint {\n margin-top: 16px;\n margin-bottom: 4px;\n}\n\n.sa-color-picker > .sa-color-picker-color {\n border: none;\n border-top-left-radius: 1rem;\n border-bottom-left-radius: 1rem;\n padding: 0;\n padding-left: 0.6rem;\n padding-right: 0.4rem;\n margin-left: 0.5rem;\n outline: none;\n box-sizing: border-box;\n width: 3rem;\n height: 2rem;\n}\n[theme=\"dark\"] .sa-color-picker > .sa-color-picker-color {\n background: var(--ui-secondary);\n}\n\n.sa-color-picker > .sa-color-picker-text {\n box-sizing: border-box;\n width: calc(150px - 3rem);\n border-top-left-radius: 0;\n border-bottom-left-radius: 0;\n}\n\n[dir=\"rtl\"] .sa-color-picker > .sa-color-picker-color {\n border-top-left-radius: 0;\n border-bottom-left-radius: 0;\n border-top-right-radius: 1rem;\n border-bottom-right-radius: 1rem;\n margin-left: 0;\n margin-right: 0.5rem;\n}\n\n[dir=\"rtl\"] .sa-color-picker > .sa-color-picker-text {\n border-top-left-radius: 1rem;\n border-bottom-left-radius: 1rem;\n border-top-right-radius: 0;\n border-bottom-right-radius: 0;\n}\n\nbody.sa-hide-eye-dropper-background div[class*=\"stage_color-picker-background\"] {\n /* Do not show eye dropper background if the color picker is \"fake\" */\n display: none;\n}\n", ""]);
// exports
/***/ }),
/***/ "./node_modules/css-loader/index.js!./src/addons/addons/editor-comment-previews/userstyle.css":
/*!*******************************************************************************************!*\
!*** ./node_modules/css-loader!./src/addons/addons/editor-comment-previews/userstyle.css ***!
\*******************************************************************************************/
/*! no static exports found */
/***/ (function(module, exports, __webpack_require__) {
exports = module.exports = __webpack_require__(/*! ../../../../node_modules/css-loader/lib/css-base.js */ "./node_modules/css-loader/lib/css-base.js")(false);
// imports
// module
exports.push([module.i, ".sa-comment-preview-outer {\n position: fixed;\n top: 0;\n left: 0;\n z-index: 100000000;\n pointer-events: none;\n}\n\n.sa-comment-preview-inner {\n width: calc(200px - 16px);\n max-height: calc(132px - 8px);\n padding: 8px;\n overflow: hidden;\n\n font-size: 12px;\n white-space: pre-wrap;\n pointer-events: none;\n\n color: rgb(87, 94, 117);\n background-color: rgb(255 255 255 / 90%);\n border-style: none;\n border-radius: 8px;\n filter: drop-shadow(0px 5px 5px rgb(0 0 0 / 10%));\n\n transform: perspective(200px);\n}\n\n@supports (backdrop-filter: blur(16px)) {\n .sa-comment-preview-inner {\n background-color: rgb(255 255 255 / 75%);\n backdrop-filter: blur(16px);\n }\n}\n\n.sa-comment-preview-fade {\n transition: opacity 0.1s, filter 0.1s, transform 0.1s linear;\n}\n\n.sa-comment-preview-hidden {\n opacity: 0;\n filter: none;\n transform: perspective(200px) translateZ(-20px);\n}\n\n.sa-comment-preview-reduce-transparency {\n background-color: rgb(255 255 255);\n backdrop-filter: none;\n}\n", ""]);
// exports
/***/ }),
/***/ "./node_modules/css-loader/index.js!./src/addons/addons/editor-searchable-dropdowns/userscript.css":
/*!************************************************************************************************!*\
!*** ./node_modules/css-loader!./src/addons/addons/editor-searchable-dropdowns/userscript.css ***!
\************************************************************************************************/
/*! no static exports found */
/***/ (function(module, exports, __webpack_require__) {
exports = module.exports = __webpack_require__(/*! ../../../../node_modules/css-loader/lib/css-base.js */ "./node_modules/css-loader/lib/css-base.js")(false);
// imports
// module
exports.push([module.i, ".u-dropdown-searchbar {\n width: 100%;\n box-sizing: border-box;\n /* based on styles for the title input */\n color: white;\n background-color: hsla(0, 100%, 100%, 0.25);\n border: 1px solid hsla(0, 0%, 0%, 0.15);\n padding: 0.5rem;\n outline: none;\n transition: 0.25s ease-out;\n font-size: 13px;\n font-weight: bold;\n border-radius: 4px;\n}\n.u-dropdown-searchbar:hover {\n background-color: hsla(0, 100%, 100%, 0.5);\n}\n.u-dropdown-searchbar:focus {\n background-color: white;\n color: black;\n}\n.blocklyDropDownDiv .goog-menu {\n overflow-x: hidden;\n}\n", ""]);
// exports
/***/ }),
/***/ "./node_modules/css-loader/index.js!./src/addons/addons/editor-theme3/compatibility.css":
/*!*************************************************************************************!*\
!*** ./node_modules/css-loader!./src/addons/addons/editor-theme3/compatibility.css ***!
\*************************************************************************************/
/*! no static exports found */
/***/ (function(module, exports, __webpack_require__) {
exports = module.exports = __webpack_require__(/*! ../../../../node_modules/css-loader/lib/css-base.js */ "./node_modules/css-loader/lib/css-base.js")(false);
// imports
// module
exports.push([module.i, "/* Imported by other addons */\n\n.sa-block-color {\n --sa-block-colored-background: var(--sa-block-background-primary);\n --sa-block-colored-background-secondary: var(--sa-block-field-background);\n --sa-block-bright-background: var(--sa-block-background-primary);\n --sa-block-text: white;\n --sa-block-gray-text: white;\n --sa-block-colored-text: var(--sa-block-background-primary);\n --sa-block-text-on-bright-background: white;\n}\n\n.sa-block-color-motion {\n --sa-block-background-primary: var(--editorTheme3-motion-primary, #4c97ff);\n --sa-block-background-secondary: var(--editorTheme3-motion-secondary, #4280d7);\n --sa-block-background-tertiary: var(--editorTheme3-motion-tertiary, #3373cc);\n --sa-block-field-background: var(--editorTheme3-motion-field, #3373cc);\n}\n\n.sa-block-color-looks {\n --sa-block-background-primary: var(--editorTheme3-looks-primary, #9966ff);\n --sa-block-background-secondary: var(--editorTheme3-looks-secondary, #855cd6);\n --sa-block-background-tertiary: var(--editorTheme3-looks-tertiary, #774dcb);\n --sa-block-field-background: var(--editorTheme3-looks-field, #774dcb);\n}\n\n.sa-block-color-sounds {\n --sa-block-background-primary: var(--editorTheme3-sounds-primary, #cf63cf);\n --sa-block-background-secondary: var(--editorTheme3-sounds-secondary, #c94fc9);\n --sa-block-background-tertiary: var(--editorTheme3-sounds-tertiary, #bd42bd);\n --sa-block-field-background: var(--editorTheme3-sounds-field, #bd42bd);\n}\n\n.sa-block-color-events {\n --sa-block-background-primary: var(--editorTheme3-event-primary, #ffbf00);\n --sa-block-background-secondary: var(--editorTheme3-event-secondary, #e6ac00);\n --sa-block-background-tertiary: var(--editorTheme3-event-tertiary, #cc9900);\n --sa-block-field-background: var(--editorTheme3-event-field, #cc9900);\n}\n\n.sa-block-color-control {\n --sa-block-background-primary: var(--editorTheme3-control-primary, #ffab19);\n --sa-block-background-secondary: var(--editorTheme3-control-secondary, #ec9c13);\n --sa-block-background-tertiary: var(--editorTheme3-control-tertiary, #cf8b17);\n --sa-block-field-background: var(--editorTheme3-control-field, #cf8b17);\n}\n\n.sa-block-color-sensing {\n --sa-block-background-primary: var(--editorTheme3-sensing-primary, #5cb1d6);\n --sa-block-background-secondary: var(--editorTheme3-sensing-secondary, #47a8d1);\n --sa-block-background-tertiary: var(--editorTheme3-sensing-tertiary, #2e8eb8);\n --sa-block-field-background: var(--editorTheme3-sensing-field, #2e8eb8);\n}\n\n.sa-block-color-operators {\n --sa-block-background-primary: var(--editorTheme3-operators-primary, #59c059);\n --sa-block-background-secondary: var(--editorTheme3-operators-secondary, #46b946);\n --sa-block-background-tertiary: var(--editorTheme3-operators-tertiary, #389438);\n --sa-block-field-background: var(--editorTheme3-operators-field, #389438);\n}\n\n.sa-block-color-data {\n --sa-block-background-primary: var(--editorTheme3-data-primary, #ff8c1a);\n --sa-block-background-secondary: var(--editorTheme3-data-secondary, #ff8000);\n --sa-block-background-tertiary: var(--editorTheme3-data-tertiary, #db6e00);\n --sa-block-field-background: var(--editorTheme3-data-field, #db6e00);\n}\n\n.sa-block-color-data-lists,\n.sa-block-color-list {\n --sa-block-background-primary: var(--editorTheme3-data_lists-primary, #ff661a);\n --sa-block-background-secondary: var(--editorTheme3-data_lists-secondary, #ff5500);\n --sa-block-background-tertiary: var(--editorTheme3-data_lists-tertiary, #e64d00);\n --sa-block-field-background: var(--editorTheme3-data_lists-field, #e64d00);\n}\n\n.sa-block-color-more,\n.sa-block-color-null {\n --sa-block-background-primary: var(--editorTheme3-more-primary, #ff6680);\n --sa-block-background-secondary: var(--editorTheme3-more-secondary, #ff4d6a);\n --sa-block-background-tertiary: var(--editorTheme3-more-tertiary, #ff3355);\n --sa-block-field-background: var(--editorTheme3-more-field, #ff3355);\n}\n\n.sa-block-color-pen {\n --sa-block-background-primary: var(--editorTheme3-pen-primary, #0fbd8c);\n --sa-block-background-secondary: var(--editorTheme3-pen-secondary, #0da57a);\n --sa-block-background-tertiary: var(--editorTheme3-pen-tertiary, #0b8e69);\n --sa-block-field-background: var(--editorTheme3-pen-field, #0b8e69);\n}\n\n.sa-block-color-addon-custom-block {\n --sa-block-background-primary: var(--editorTheme3-sa-primary, #29beb8);\n --sa-block-background-secondary: var(--editorTheme3-sa-secondary, #3aa8a4);\n --sa-block-background-tertiary: var(--editorTheme3-sa-tertiary, #3aa8a4);\n --sa-block-field-background: var(--editorTheme3-sa-field, #3aa8a4);\n}\n\n.sa-block-color-TurboWarp {\n --sa-block-background-primary: var(--editorTheme3-tw-primary, #ff4c4c);\n --sa-block-background-secondary: var(--editorTheme3-tw-secondary, #e64444);\n --sa-block-background-tertiary: var(--editorTheme3-tw-tertiary, #e64444);\n --sa-block-field-background: var(--editorTheme3-tw-field, #e64444);\n}\n", ""]);
// exports
/***/ }),
/***/ "./node_modules/css-loader/index.js!./src/addons/addons/find-bar/userstyle.css":
/*!****************************************************************************!*\
!*** ./node_modules/css-loader!./src/addons/addons/find-bar/userstyle.css ***!
\****************************************************************************/
/*! no static exports found */
/***/ (function(module, exports, __webpack_require__) {
exports = module.exports = __webpack_require__(/*! ../../../../node_modules/css-loader/lib/css-base.js */ "./node_modules/css-loader/lib/css-base.js")(false);
// imports
exports.i(__webpack_require__(/*! -!../../../../node_modules/css-loader!../editor-theme3/compatibility.css */ "./node_modules/css-loader/index.js!./src/addons/addons/editor-theme3/compatibility.css"), "");
// module
exports.push([module.i, ".sa-find-bar {\n display: flex;\n align-items: center;\n white-space: nowrap;\n font-family: \"Helvetica Neue\", Helvetica, Arial, sans-serif;\n width: 100%;\n height: 100%;\n margin-left: 1em;\n}\n.sa-find-bar[hidden] {\n /* !important to override displayNoneWhileDisabled */\n display: none !important;\n}\n\n.sa-find-wrapper {\n overflow: visible;\n position: relative;\n height: 2rem;\n width: 100%;\n max-width: 16em;\n}\n\n.sa-find-dropdown-out {\n display: block;\n top: -6px;\n z-index: 100;\n width: 100%;\n max-width: 16em;\n position: relative;\n padding: 4px;\n border: none;\n border-radius: 4px;\n margin-top: 6px;\n}\n\n.sa-find-dropdown-out.visible {\n position: absolute;\n width: 16em;\n box-shadow: 0px 0px 8px 1px var(--ui-black-transparent, rgba(0, 0, 0, 0.3));\n background-color: var(--ui-primary, white);\n}\n\n/* We need to modify Scratch styles so that the place where the find bar is injected */\n/* has actually correct size information, which is used to make the find bar not cover up controls */\n[class*=\"gui_tab-list_\"] {\n width: 100%;\n}\n[class*=\"gui_tab_\"] {\n flex-grow: 0;\n}\n\n.sa-find-input {\n width: 100%;\n box-sizing: border-box !important;\n /* !important required for extension, because CSS injection method (and hence order) differs from addon */\n height: 1.5rem;\n\n /* Change Scratch default styles */\n border-radius: 0.25rem;\n font-size: 0.75rem;\n padding-left: 0.4em;\n}\n\n.sa-find-input:focus {\n /* Change Scratch default styles */\n box-shadow: none;\n}\n\n.sa-find-dropdown {\n display: none;\n position: relative;\n padding: 0.2em 0;\n font-size: 0.75rem;\n line-height: 1;\n overflow-y: auto;\n min-height: 128px;\n max-height: 65vh;\n user-select: none;\n max-width: 100%;\n margin-top: 6px;\n border: none;\n}\n\n.sa-find-dropdown-out.visible > .sa-find-dropdown {\n display: block;\n}\n\n.sa-find-dropdown > li {\n display: block;\n padding: 0.5em 0.3em;\n white-space: nowrap;\n margin: 0;\n font-weight: bold;\n text-overflow: ellipsis;\n overflow: hidden;\n}\n\n.sa-find-dropdown > li > b {\n background-color: #aaffaa;\n color: black;\n}\n\n/* Drop down items */\n.sa-find-dropdown > li:hover,\n.sa-find-dropdown > li.sel {\n color: var(--sa-block-text-on-bright-background);\n cursor: pointer;\n}\n\n.sa-find-dropdown > li::before {\n content: \"\\25CF \"; /* ● */\n}\n\n.sa-find-flag {\n color: #4cbf56;\n}\n/* .sa-find-dropdown added for specificity */\n.sa-find-dropdown > .sa-find-flag:hover,\n.sa-find-dropdown > .sa-find-flag.sel {\n background-color: #4cbf56;\n color: white;\n}\n\n.sa-find-dropdown .sa-block-color {\n color: var(--sa-block-colored-text);\n}\n.sa-find-dropdown .sa-block-color:hover,\n.sa-find-dropdown .sa-block-color.sel {\n background-color: var(--sa-block-bright-background);\n}\n\n.sa-find-carousel {\n font-weight: normal;\n position: absolute;\n right: 0;\n white-space: nowrap;\n background-color: inherit;\n z-index: 1;\n padding: 0;\n}\n\n.sa-find-carousel-control {\n padding: 0 6px;\n}\n\n.sa-find-carousel-control:hover {\n color: #ffff80;\n}\n", ""]);
// exports
/***/ }),
/***/ "./node_modules/css-loader/index.js!./src/addons/addons/folders/style.css":
/*!***********************************************************************!*\
!*** ./node_modules/css-loader!./src/addons/addons/folders/style.css ***!
\***********************************************************************/
/*! no static exports found */
/***/ (function(module, exports, __webpack_require__) {
exports = module.exports = __webpack_require__(/*! ../../../../node_modules/css-loader/lib/css-base.js */ "./node_modules/css-loader/lib/css-base.js")(false);
// imports
// module
exports.push([module.i, ".sa-folders-contextmenu-item {\n max-width: 250px;\n overflow: hidden;\n text-overflow: ellipsis;\n}\n\n[sa-folders-context-type=\"folder\"] .react-contextmenu > :not(.sa-ctx-menu) {\n display: none;\n}\n", ""]);
// exports
/***/ }),
/***/ "./node_modules/css-loader/index.js!./src/addons/addons/middle-click-popup/userstyle.css":
/*!**************************************************************************************!*\
!*** ./node_modules/css-loader!./src/addons/addons/middle-click-popup/userstyle.css ***!
\**************************************************************************************/
/*! no static exports found */
/***/ (function(module, exports, __webpack_require__) {
exports = module.exports = __webpack_require__(/*! ../../../../node_modules/css-loader/lib/css-base.js */ "./node_modules/css-loader/lib/css-base.js")(false);
// imports
exports.i(__webpack_require__(/*! -!../../../../node_modules/css-loader!../editor-theme3/compatibility.css */ "./node_modules/css-loader/index.js!./src/addons/addons/editor-theme3/compatibility.css"), "");
// module
exports.push([module.i, "/* Find Input Box */\n.sa-float-bar-input {\n width: 100%;\n box-sizing: border-box !important;\n /* !important required for extension, because CSS injection method (and hence order) differs from addon */\n height: 1.5rem;\n\n /* Change Scratch default styles */\n border-radius: 0.25rem;\n font-size: 0.75rem;\n padding-left: 0.4em;\n}\n[theme=\"dark\"] input.s3devInp {\n color: #eee;\n background: #3333;\n}\n[theme=\"dark\"] input.s3devInp:hover {\n background: #333;\n}\n\n.sa-float-bar-input:focus {\n /* Change Scratch default styles */\n box-shadow: none;\n}\n\n/* Drop down from find button */\n.sa-float-bar-dropdown-out {\n display: block;\n top: -6px;\n z-index: 100;\n max-width: 16em;\n padding: 4px;\n position: absolute;\n width: 16em;\n box-shadow: 0px 0px 8px 1px var(--ui-black-transparent, rgba(0, 0, 0, 0.3));\n background-color: var(--ui-primary, white);\n border: none;\n border-radius: 4px;\n}\n\n/* Drop down from find button */\n.sa-float-bar-dropdown {\n display: none;\n position: relative;\n padding: 0.2em 0;\n font-size: 0.75rem;\n line-height: 1;\n overflow-y: auto;\n min-height: 128px;\n user-select: none;\n max-width: 100%;\n max-height: 200px;\n margin-bottom: 0;\n}\n\n.sa-float-bar-dropdown-out.vis .sa-float-bar-dropdown {\n display: block;\n border: none;\n}\n\n/* Drop down items */\n.sa-float-bar-dropdown > li {\n display: block;\n padding: 0.5em 0.3em;\n white-space: nowrap;\n margin: 0;\n font-weight: bold;\n text-overflow: ellipsis;\n overflow: hidden;\n cursor: pointer;\n}\n\n.sa-float-bar-dropdown > li > b {\n background-color: #aaffaa;\n color: black;\n}\n\n.sa-float-bar-dropdown > li {\n height: 19px;\n padding: 3px 8px;\n margin: 2px 0.3em;\n box-sizing: border-box;\n position: relative;\n background-color: var(--sa-block-colored-background);\n color: var(--sa-block-text);\n font-weight: bold;\n width: min-content;\n}\n.sa-float-bar-dropdown > li:hover,\n.sa-float-bar-dropdown > li.sel {\n background-color: var(--sa-block-colored-background-secondary);\n}\n\n.sa-float-bar-dropdown > li.sa-hat {\n border-radius: 14px 14px 3px 3px;\n}\n.sa-float-bar-dropdown > li.sa-block {\n border-radius: 3px;\n}\n.sa-float-bar-dropdown > li.sa-reporter {\n border-radius: 10px;\n}\n\n.sa-float-bar-dropdown > li.sa-boolean {\n width: min-content;\n}\n.sa-float-bar-dropdown > li.sa-boolean::before {\n content: \"\";\n position: absolute;\n left: 0;\n top: 0;\n width: 0;\n height: 0;\n border-right: 9px solid transparent;\n border-top: 9px solid var(--ui-primary, white);\n border-bottom: 10px solid var(--ui-primary, white);\n}\n.sa-float-bar-dropdown > li.sa-boolean::after {\n content: \"\";\n position: absolute;\n right: 0;\n top: 0;\n width: 0;\n height: 0;\n border-left: 9px solid transparent;\n border-top: 9px solid var(--ui-primary, white);\n border-bottom: 10px solid var(--ui-primary, white);\n}\n[theme=\"dark\"] .s3devDD > li.boolean::before {\n border-top-color: #111;\n border-bottom-color: #111;\n}\n[theme=\"dark\"] .s3devDD > li.boolean::after {\n border-top-color: #111;\n border-bottom-color: #111;\n}\n\n.sa-float-bar {\n display: flex;\n white-space: nowrap;\n font-family: \"Helvetica Neue\", Helvetica, Arial, sans-serif;\n\n position: absolute;\n min-width: 128px;\n background-color: white;\n border-radius: 4px;\n box-shadow: rgba(0, 0, 0, 0.3) 0 0 3px, rgba(0, 0, 0, 0.2) 0 3px 10px;\n\n z-index: 999;\n}\n[theme=\"dark\"] #s3devFloatingBar {\n background-color: #111;\n}\n\n.sa-float-bar-dropdown > li > b {\n background-color: rgba(0, 0, 0, 0.6);\n color: white;\n}\n\n[data-highlighted=\"true\"] {\n background-color: hsla(30, 100%, 55%, 1) !important; /* orange */\n color: white !important;\n}\n", ""]);
// exports
/***/ }),
/***/ "./node_modules/css-loader/index.js!./src/addons/addons/onion-skinning/style.css":
/*!******************************************************************************!*\
!*** ./node_modules/css-loader!./src/addons/addons/onion-skinning/style.css ***!
\******************************************************************************/
/*! no static exports found */
/***/ (function(module, exports, __webpack_require__) {
exports = module.exports = __webpack_require__(/*! ../../../../node_modules/css-loader/lib/css-base.js */ "./node_modules/css-loader/lib/css-base.js")(false);
// imports
// module
exports.push([module.i, ".sa-onion-button {\n position: relative;\n}\n.sa-onion-button:focus-within {\n background-color: hsla(0, 100%, 65%, 0.2);\n}\n[theme=\"dark\"] .sa-onion-image {\n filter: brightness(0) invert(0.8);\n}\n.sa-onion-button[data-enabled=\"true\"] .sa-onion-image {\n filter: brightness(0) invert(1);\n}\n.sa-onion-button[data-enabled=\"true\"] {\n background-color: #ff4c4c;\n}\n\n.sa-onion-group {\n position: relative;\n flex-direction: row;\n}\n\n.sa-onion-settings-wrapper {\n position: absolute;\n justify-items: center;\n left: 50%;\n width: 1.95rem;\n height: 1.95rem;\n display: grid;\n}\n\n.sa-onion-settings {\n position: absolute;\n bottom: 100%;\n /* based on the styles for the color dropdown */\n padding: 4px;\n border-radius: 4px;\n border: 1px solid var(--paint-ui-pane-border, #ddd);\n box-shadow: 0px 0px 8px 1px rgba(0, 0, 0, 0.3);\n transition-property: bottom, opacity;\n transition-duration: 500ms;\n transition-timing-function: cubic-bezier(0.23, 1, 0.32, 1);\n opacity: 0;\n pointer-events: none;\n background: var(--ui-primary, white);\n min-height: 100%;\n min-width: 100%;\n display: flex;\n flex-direction: column;\n gap: 0.25em;\n}\n.sa-onion-settings[data-visible=\"true\"] {\n bottom: calc(100% + 22px);\n pointer-events: auto;\n opacity: 1;\n}\n\n.sa-onion-settings-line {\n display: flex;\n justify-content: flex-end;\n align-items: baseline;\n gap: 0.25em;\n}\n\n.sa-onion-settings-input {\n position: absolute;\n top: 0;\n left: 0;\n width: 100%;\n height: 100%;\n box-sizing: border-box;\n text-align: center;\n border: 0;\n background: transparent;\n -moz-appearance: textfield;\n border: 0;\n outline: 0;\n}\n\n.sa-onion-settings-input::-webkit-outer-spin-button,\n.sa-onion-settings-input::-webkit-inner-spin-button {\n -webkit-appearance: none;\n margin: 0;\n}\n\n.sa-onion-settings-tip {\n position: absolute;\n bottom: 0;\n transform: translateY(100%);\n right: calc(50% - 7px);\n}\n.sa-onion-settings-polygon {\n fill: var(--ui-primary, white);\n stroke: var(--paint-ui-pane-border, #ddd);\n}\n\n.sa-onion-settings-label {\n white-space: nowrap;\n}\n", ""]);
// exports
/***/ }),
/***/ "./node_modules/css-loader/index.js!./src/addons/addons/pick-colors-from-stage/style.css":
/*!**************************************************************************************!*\
!*** ./node_modules/css-loader!./src/addons/addons/pick-colors-from-stage/style.css ***!
\**************************************************************************************/
/*! no static exports found */
/***/ (function(module, exports, __webpack_require__) {
exports = module.exports = __webpack_require__(/*! ../../../../node_modules/css-loader/lib/css-base.js */ "./node_modules/css-loader/lib/css-base.js")(false);
// imports
// module
exports.push([module.i, ".sa-stage-color-picker-picking [class^=\"stage_color-picker-background_\"] {\n display: none;\n}\n", ""]);
// exports
/***/ }),
/***/ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/editor-devtools/icon--close.svg":
/*!*************************************************************************************************!*\
!*** ./node_modules/url-loader/dist/cjs.js!./src/addons/addons/editor-devtools/icon--close.svg ***!
\*************************************************************************************************/
/*! exports provided: default */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony default export */ __webpack_exports__["default"] = ("");
/***/ }),
/***/ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/folders/folder.svg":
/*!************************************************************************************!*\
!*** ./node_modules/url-loader/dist/cjs.js!./src/addons/addons/folders/folder.svg ***!
\************************************************************************************/
/*! exports provided: default */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony default export */ __webpack_exports__["default"] = ("");
/***/ }),
/***/ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/onion-skinning/decrement.svg":
/*!**********************************************************************************************!*\
!*** ./node_modules/url-loader/dist/cjs.js!./src/addons/addons/onion-skinning/decrement.svg ***!
\**********************************************************************************************/
/*! exports provided: default */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony default export */ __webpack_exports__["default"] = ("");
/***/ }),
/***/ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/onion-skinning/increment.svg":
/*!**********************************************************************************************!*\
!*** ./node_modules/url-loader/dist/cjs.js!./src/addons/addons/onion-skinning/increment.svg ***!
\**********************************************************************************************/
/*! exports provided: default */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony default export */ __webpack_exports__["default"] = ("");
/***/ }),
/***/ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/onion-skinning/settings.svg":
/*!*********************************************************************************************!*\
!*** ./node_modules/url-loader/dist/cjs.js!./src/addons/addons/onion-skinning/settings.svg ***!
\*********************************************************************************************/
/*! exports provided: default */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony default export */ __webpack_exports__["default"] = ("");
/***/ }),
/***/ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/onion-skinning/toggle.svg":
/*!*******************************************************************************************!*\
!*** ./node_modules/url-loader/dist/cjs.js!./src/addons/addons/onion-skinning/toggle.svg ***!
\*******************************************************************************************/
/*! exports provided: default */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony default export */ __webpack_exports__["default"] = ("");
/***/ }),
/***/ "./src/addons/addons/bitmap-copy/_runtime_entry.js":
/*!*********************************************************!*\
!*** ./src/addons/addons/bitmap-copy/_runtime_entry.js ***!
\*********************************************************/
/*! exports provided: resources */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "resources", function() { return resources; });
/* harmony import */ var _userscript_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./userscript.js */ "./src/addons/addons/bitmap-copy/userscript.js");
/* generated by pull.js */
const resources = {
"userscript.js": _userscript_js__WEBPACK_IMPORTED_MODULE_0__["default"]
};
/***/ }),
/***/ "./src/addons/addons/bitmap-copy/userscript.js":
/*!*****************************************************!*\
!*** ./src/addons/addons/bitmap-copy/userscript.js ***!
\*****************************************************/
/*! exports provided: default */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony default export */ __webpack_exports__["default"] = (async ({
addon,
console
}) => {
if (!addon.tab.redux.state) return console.warn("Redux is not available!");
addon.tab.redux.initialize();
addon.tab.redux.addEventListener("statechanged", ({
detail
}) => {
if (addon.self.disabled) return;
const e = detail;
if (!e.action || e.action.type !== "scratch-paint/clipboard/SET") return;
const items = e.next.scratchPaint.clipboard.items;
if (items.length !== 1) return;
const firstItem = items[0]; // TODO vector support
if (!Array.isArray(firstItem) || firstItem[0] !== "Raster") return console.log("copied element is vector");
const dataURL = firstItem[1].source;
addon.tab.copyImage(dataURL).then(() => console.log("Image successfully copied")).catch(e => console.error("Image could not be copied: ".concat(e)));
});
});
/***/ }),
/***/ "./src/addons/addons/block-cherry-picking/_runtime_entry.js":
/*!******************************************************************!*\
!*** ./src/addons/addons/block-cherry-picking/_runtime_entry.js ***!
\******************************************************************/
/*! exports provided: resources */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "resources", function() { return resources; });
/* harmony import */ var _userscript_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./userscript.js */ "./src/addons/addons/block-cherry-picking/userscript.js");
/* generated by pull.js */
const resources = {
"userscript.js": _userscript_js__WEBPACK_IMPORTED_MODULE_0__["default"]
};
/***/ }),
/***/ "./src/addons/addons/block-cherry-picking/userscript.js":
/*!**************************************************************!*\
!*** ./src/addons/addons/block-cherry-picking/userscript.js ***!
\**************************************************************/
/*! exports provided: default */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony import */ var _block_duplicate_module_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../block-duplicate/module.js */ "./src/addons/addons/block-duplicate/module.js");
/* harmony default export */ __webpack_exports__["default"] = (async function ({
addon,
console
}) {
const update = () => {
_block_duplicate_module_js__WEBPACK_IMPORTED_MODULE_0__["setCherryPicking"](!addon.self.disabled, addon.settings.get("invertDrag"));
};
addon.self.addEventListener("disabled", update);
addon.self.addEventListener("reenabled", update);
addon.settings.addEventListener("change", update);
update();
_block_duplicate_module_js__WEBPACK_IMPORTED_MODULE_0__["load"](addon);
});
/***/ }),
/***/ "./src/addons/addons/block-duplicate/_runtime_entry.js":
/*!*************************************************************!*\
!*** ./src/addons/addons/block-duplicate/_runtime_entry.js ***!
\*************************************************************/
/*! exports provided: resources */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "resources", function() { return resources; });
/* harmony import */ var _userscript_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./userscript.js */ "./src/addons/addons/block-duplicate/userscript.js");
/* generated by pull.js */
const resources = {
"userscript.js": _userscript_js__WEBPACK_IMPORTED_MODULE_0__["default"]
};
/***/ }),
/***/ "./src/addons/addons/block-duplicate/module.js":
/*!*****************************************************!*\
!*** ./src/addons/addons/block-duplicate/module.js ***!
\*****************************************************/
/*! exports provided: setCherryPicking, setDuplication, load */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "setCherryPicking", function() { return setCherryPicking; });
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "setDuplication", function() { return setDuplication; });
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "load", function() { return load; });
let enableCherryPicking = false;
let invertCherryPicking = false;
function setCherryPicking(newEnabled, newInverted) {
enableCherryPicking = newEnabled; // If cherry picking is disabled, also disable invert. Duplicating blocks can still cause
// this setting to be used.
invertCherryPicking = newEnabled && newInverted;
}
let enableDuplication = false;
function setDuplication(newEnabled) {
enableDuplication = newEnabled;
} // mostRecentEvent_ is sometimes a fake event, so we can't rely on reading its properties.
let ctrlOrMetaPressed = false;
let altPressed = false;
document.addEventListener("mousedown", function (e) {
ctrlOrMetaPressed = e.ctrlKey || e.metaKey;
altPressed = e.altKey;
}, {
capture: true
});
let loaded = false;
async function load(addon) {
if (loaded) {
return;
}
loaded = true;
const ScratchBlocks = await addon.tab.traps.getBlockly(); // https://github.com/LLK/scratch-blocks/blob/912b8cc728bea8fd91af85078c64fcdbfe21c87a/core/gesture.js#L454
const originalStartDraggingBlock = ScratchBlocks.Gesture.prototype.startDraggingBlock_;
ScratchBlocks.Gesture.prototype.startDraggingBlock_ = function (...args) {
let block = this.targetBlock_; // Scratch uses fake mouse events to implement right click > duplicate
const isRightClickDuplicate = !(this.mostRecentEvent_ instanceof MouseEvent);
const isDuplicating = enableDuplication && altPressed && !isRightClickDuplicate && !this.flyout_ && !this.shouldDuplicateOnDrag_ && this.targetBlock_.type !== "procedures_definition";
const isCherryPickingInverted = invertCherryPicking && !isRightClickDuplicate && block.getParent();
const canCherryPick = enableCherryPicking || isDuplicating;
const isCherryPicking = canCherryPick && ctrlOrMetaPressed === !isCherryPickingInverted && !block.isShadow();
if (isDuplicating || isCherryPicking) {
if (!ScratchBlocks.Events.getGroup()) {
// Scratch will disable grouping on its own later.
ScratchBlocks.Events.setGroup(true);
}
}
if (isDuplicating) {
// Based on https://github.com/LLK/scratch-blocks/blob/feda366947432b9d82a4f212f43ff6d4ab6f252f/core/scratch_blocks_utils.js#L171
// Setting this.shouldDuplicateOnDrag_ = true does NOT work because it doesn't call changeObscuredShadowIds
this.startWorkspace_.setResizesEnabled(false);
ScratchBlocks.Events.disable();
let newBlock;
try {
const xmlBlock = ScratchBlocks.Xml.blockToDom(block);
newBlock = ScratchBlocks.Xml.domToBlock(xmlBlock, this.startWorkspace_);
ScratchBlocks.scratchBlocksUtils.changeObscuredShadowIds(newBlock);
const xy = block.getRelativeToSurfaceXY();
newBlock.moveBy(xy.x, xy.y);
} catch (e) {
console.error(e);
}
ScratchBlocks.Events.enable();
if (newBlock) {
block = newBlock;
this.targetBlock_ = newBlock;
if (ScratchBlocks.Events.isEnabled()) {
ScratchBlocks.Events.fire(new ScratchBlocks.Events.BlockCreate(newBlock));
}
}
}
if (isCherryPicking) {
if (isRightClickDuplicate || isDuplicating) {
const nextBlock = block.getNextBlock();
if (nextBlock) {
nextBlock.dispose();
}
}
block.unplug(true);
}
return originalStartDraggingBlock.call(this, ...args);
};
}
/***/ }),
/***/ "./src/addons/addons/block-duplicate/userscript.js":
/*!*********************************************************!*\
!*** ./src/addons/addons/block-duplicate/userscript.js ***!
\*********************************************************/
/*! exports provided: default */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony import */ var _module_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./module.js */ "./src/addons/addons/block-duplicate/module.js");
/* harmony default export */ __webpack_exports__["default"] = (async function ({
addon,
console
}) {
const update = () => {
_module_js__WEBPACK_IMPORTED_MODULE_0__["setDuplication"](!addon.self.disabled);
};
addon.self.addEventListener("disabled", update);
addon.self.addEventListener("reenabled", update);
update();
_module_js__WEBPACK_IMPORTED_MODULE_0__["load"](addon);
});
/***/ }),
/***/ "./src/addons/addons/block-switching/_runtime_entry.js":
/*!*************************************************************!*\
!*** ./src/addons/addons/block-switching/_runtime_entry.js ***!
\*************************************************************/
/*! exports provided: resources */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "resources", function() { return resources; });
/* harmony import */ var _userscript_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./userscript.js */ "./src/addons/addons/block-switching/userscript.js");
/* generated by pull.js */
const resources = {
"userscript.js": _userscript_js__WEBPACK_IMPORTED_MODULE_0__["default"]
};
/***/ }),
/***/ "./src/addons/addons/block-switching/userscript.js":
/*!*********************************************************!*\
!*** ./src/addons/addons/block-switching/userscript.js ***!
\*********************************************************/
/*! exports provided: default */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony default export */ __webpack_exports__["default"] = (async function ({
addon,
console,
msg
}) {
const ScratchBlocks = await addon.tab.traps.getBlockly();
const vm = addon.tab.traps.vm;
let blockSwitches = {};
let procedureSwitches = {};
const noopSwitch = {
isNoop: true
};
const randomColor = () => {
const num = Math.floor(Math.random() * 256 * 256 * 256);
return "#".concat(num.toString(16).padStart(6, "0"));
};
const buildSwitches = () => {
blockSwitches = {};
procedureSwitches = {};
if (addon.settings.get("motion")) {
blockSwitches["motion_turnright"] = [noopSwitch, {
opcode: "motion_turnleft"
}];
blockSwitches["motion_turnleft"] = [{
opcode: "motion_turnright"
}, noopSwitch];
blockSwitches["motion_setx"] = [noopSwitch, {
opcode: "motion_changexby",
remapInputName: {
X: "DX"
}
}, {
opcode: "motion_sety",
remapInputName: {
X: "Y"
}
}, {
opcode: "motion_changeyby",
remapInputName: {
X: "DY"
}
}];
blockSwitches["motion_changexby"] = [{
opcode: "motion_setx",
remapInputName: {
DX: "X"
}
}, noopSwitch, {
opcode: "motion_sety",
remapInputName: {
DX: "Y"
}
}, {
opcode: "motion_changeyby",
remapInputName: {
DX: "DY"
}
}];
blockSwitches["motion_sety"] = [{
opcode: "motion_setx",
remapInputName: {
Y: "X"
}
}, {
opcode: "motion_changexby",
remapInputName: {
Y: "DX"
}
}, noopSwitch, {
opcode: "motion_changeyby",
remapInputName: {
Y: "DY"
}
}];
blockSwitches["motion_changeyby"] = [{
opcode: "motion_setx",
remapInputName: {
DY: "X"
}
}, {
opcode: "motion_changexby",
remapInputName: {
DY: "DX"
}
}, {
opcode: "motion_sety",
remapInputName: {
DY: "Y"
}
}, noopSwitch];
blockSwitches["motion_xposition"] = [noopSwitch, {
opcode: "motion_yposition"
}];
blockSwitches["motion_yposition"] = [{
opcode: "motion_xposition"
}, noopSwitch];
}
if (addon.settings.get("looks")) {
blockSwitches["looks_seteffectto"] = [noopSwitch, {
opcode: "looks_changeeffectby",
remapInputName: {
VALUE: "CHANGE"
}
}];
blockSwitches["looks_changeeffectby"] = [{
opcode: "looks_seteffectto",
remapInputName: {
CHANGE: "VALUE"
}
}, noopSwitch];
blockSwitches["looks_setsizeto"] = [noopSwitch, {
opcode: "looks_changesizeby",
remapInputName: {
SIZE: "CHANGE"
}
}];
blockSwitches["looks_changesizeby"] = [{
opcode: "looks_setsizeto",
remapInputName: {
CHANGE: "SIZE"
}
}, noopSwitch];
blockSwitches["looks_costumenumbername"] = [noopSwitch, {
opcode: "looks_backdropnumbername"
}];
blockSwitches["looks_backdropnumbername"] = [{
opcode: "looks_costumenumbername"
}, noopSwitch];
blockSwitches["looks_show"] = [noopSwitch, {
opcode: "looks_hide"
}];
blockSwitches["looks_hide"] = [{
opcode: "looks_show"
}, noopSwitch];
blockSwitches["looks_nextcostume"] = [noopSwitch, {
opcode: "looks_nextbackdrop"
}];
blockSwitches["looks_nextbackdrop"] = [{
opcode: "looks_nextcostume"
}, noopSwitch];
blockSwitches["looks_say"] = [noopSwitch, {
opcode: "looks_sayforsecs",
createInputs: {
SECS: {
shadowType: "math_number",
value: "2"
}
}
}, {
opcode: "looks_think"
}, {
opcode: "looks_thinkforsecs",
createInputs: {
SECS: {
shadowType: "math_number",
value: "2"
}
}
}];
blockSwitches["looks_think"] = [{
opcode: "looks_say"
}, {
opcode: "looks_sayforsecs",
createInputs: {
SECS: {
shadowType: "math_number",
value: "2"
}
}
}, noopSwitch, {
opcode: "looks_thinkforsecs",
createInputs: {
SECS: {
shadowType: "math_number",
value: "2"
}
}
}];
blockSwitches["looks_sayforsecs"] = [{
opcode: "looks_say",
splitInputs: ["SECS"]
}, {
opcode: "looks_think",
splitInputs: ["SECS"]
}, noopSwitch, {
opcode: "looks_thinkforsecs"
}];
blockSwitches["looks_thinkforsecs"] = [{
opcode: "looks_say",
splitInputs: ["SECS"]
}, {
opcode: "looks_think",
splitInputs: ["SECS"]
}, {
opcode: "looks_sayforsecs"
}, noopSwitch];
blockSwitches["looks_switchbackdropto"] = [noopSwitch, {
opcode: "looks_switchbackdroptoandwait"
}];
blockSwitches["looks_switchbackdroptoandwait"] = [{
opcode: "looks_switchbackdropto"
}, noopSwitch];
blockSwitches["looks_gotofrontback"] = [noopSwitch, {
opcode: "looks_goforwardbackwardlayers",
remapInputName: {
FRONT_BACK: "FORWARD_BACKWARD"
},
mapFieldValues: {
FRONT_BACK: {
front: "forward",
back: "backward"
}
},
createInputs: {
NUM: {
shadowType: "math_integer",
value: "1"
}
}
}];
blockSwitches["looks_goforwardbackwardlayers"] = [{
opcode: "looks_gotofrontback",
splitInputs: ["NUM"],
remapInputName: {
FORWARD_BACKWARD: "FRONT_BACK"
},
mapFieldValues: {
FORWARD_BACKWARD: {
forward: "front",
backward: "back"
}
}
}, noopSwitch];
}
if (addon.settings.get("sound")) {
blockSwitches["sound_play"] = [noopSwitch, {
opcode: "sound_playuntildone"
}];
blockSwitches["sound_playuntildone"] = [{
opcode: "sound_play"
}, noopSwitch];
blockSwitches["sound_seteffectto"] = [noopSwitch, {
opcode: "sound_changeeffectby"
}];
blockSwitches["sound_changeeffectby"] = [{
opcode: "sound_seteffectto"
}, noopSwitch];
blockSwitches["sound_setvolumeto"] = [noopSwitch, {
opcode: "sound_changevolumeby"
}];
blockSwitches["sound_changevolumeby"] = [{
opcode: "sound_setvolumeto"
}, noopSwitch];
}
if (addon.settings.get("event")) {
blockSwitches["event_broadcast"] = [noopSwitch, {
opcode: "event_broadcastandwait"
}];
blockSwitches["event_broadcastandwait"] = [{
opcode: "event_broadcast"
}, noopSwitch];
}
if (addon.settings.get("control")) {
blockSwitches["control_if"] = [noopSwitch, {
opcode: "control_if_else"
}];
blockSwitches["control_if_else"] = [{
opcode: "control_if",
splitInputs: ["SUBSTACK2"]
}, noopSwitch];
blockSwitches["control_repeat_until"] = [noopSwitch, {
opcode: "control_wait_until",
splitInputs: ["SUBSTACK"]
}, {
opcode: "control_forever",
splitInputs: ["CONDITION"]
}];
blockSwitches["control_forever"] = [{
opcode: "control_repeat_until"
}, noopSwitch];
blockSwitches["control_wait_until"] = [{
opcode: "control_repeat_until"
}, noopSwitch];
}
if (addon.settings.get("operator")) {
blockSwitches["operator_equals"] = [{
opcode: "operator_gt"
}, noopSwitch, {
opcode: "operator_lt"
}];
blockSwitches["operator_gt"] = [noopSwitch, {
opcode: "operator_equals"
}, {
opcode: "operator_lt"
}];
blockSwitches["operator_lt"] = [{
opcode: "operator_gt"
}, {
opcode: "operator_equals"
}, noopSwitch];
blockSwitches["operator_add"] = [noopSwitch, {
opcode: "operator_subtract"
}, {
opcode: "operator_multiply"
}, {
opcode: "operator_divide"
}, {
opcode: "operator_mod"
}];
blockSwitches["operator_subtract"] = [{
opcode: "operator_add"
}, noopSwitch, {
opcode: "operator_multiply"
}, {
opcode: "operator_divide"
}, {
opcode: "operator_mod"
}];
blockSwitches["operator_multiply"] = [{
opcode: "operator_add"
}, {
opcode: "operator_subtract"
}, noopSwitch, {
opcode: "operator_divide"
}, {
opcode: "operator_mod"
}];
blockSwitches["operator_divide"] = [{
opcode: "operator_add"
}, {
opcode: "operator_subtract"
}, {
opcode: "operator_multiply"
}, noopSwitch, {
opcode: "operator_mod"
}];
blockSwitches["operator_mod"] = [{
opcode: "operator_add"
}, {
opcode: "operator_subtract"
}, {
opcode: "operator_multiply"
}, {
opcode: "operator_divide"
}, noopSwitch];
blockSwitches["operator_and"] = [noopSwitch, {
opcode: "operator_or"
}];
blockSwitches["operator_or"] = [{
opcode: "operator_and"
}, noopSwitch];
}
if (addon.settings.get("sensing")) {
blockSwitches["sensing_mousex"] = [noopSwitch, {
opcode: "sensing_mousey"
}];
blockSwitches["sensing_mousey"] = [{
opcode: "sensing_mousex"
}, noopSwitch];
blockSwitches["sensing_touchingcolor"] = [noopSwitch, {
opcode: "sensing_coloristouchingcolor",
createInputs: {
COLOR2: {
shadowType: "colour_picker",
value: randomColor
}
}
}];
blockSwitches["sensing_coloristouchingcolor"] = [{
opcode: "sensing_touchingcolor",
splitInputs: ["COLOR2"]
}, noopSwitch];
}
if (addon.settings.get("data")) {
blockSwitches["data_setvariableto"] = [noopSwitch, {
opcode: "data_changevariableby",
remapShadowType: {
VALUE: "math_number"
}
}];
blockSwitches["data_changevariableby"] = [{
opcode: "data_setvariableto",
remapShadowType: {
VALUE: "text"
}
}, noopSwitch];
blockSwitches["data_showvariable"] = [noopSwitch, {
opcode: "data_hidevariable"
}];
blockSwitches["data_hidevariable"] = [{
opcode: "data_showvariable"
}, noopSwitch];
blockSwitches["data_showlist"] = [noopSwitch, {
opcode: "data_hidelist"
}];
blockSwitches["data_hidelist"] = [{
opcode: "data_showlist"
}, noopSwitch];
blockSwitches["data_replaceitemoflist"] = [noopSwitch, {
opcode: "data_insertatlist"
}];
blockSwitches["data_insertatlist"] = [{
opcode: "data_replaceitemoflist"
}, noopSwitch];
blockSwitches["data_deleteoflist"] = [noopSwitch, {
opcode: "data_deletealloflist",
splitInputs: ["INDEX"]
}];
blockSwitches["data_deletealloflist"] = [{
opcode: "data_deleteoflist",
createInputs: {
INDEX: {
shadowType: "math_integer",
value: "1"
}
}
}, noopSwitch];
}
if (addon.settings.get("extension")) {
blockSwitches["pen_penDown"] = [noopSwitch, {
opcode: "pen_penUp"
}];
blockSwitches["pen_penUp"] = [{
opcode: "pen_penDown"
}, noopSwitch];
blockSwitches["pen_setPenColorParamTo"] = [noopSwitch, {
opcode: "pen_changePenColorParamBy"
}];
blockSwitches["pen_changePenColorParamBy"] = [{
opcode: "pen_setPenColorParamTo"
}, noopSwitch];
blockSwitches["pen_setPenHueToNumber"] = [noopSwitch, {
opcode: "pen_changePenHueBy"
}];
blockSwitches["pen_changePenHueBy"] = [{
opcode: "pen_setPenHueToNumber"
}, noopSwitch];
blockSwitches["pen_setPenShadeToNumber"] = [noopSwitch, {
opcode: "pen_changePenShadeBy"
}];
blockSwitches["pen_changePenShadeBy"] = [{
opcode: "pen_setPenShadeToNumber"
}, noopSwitch];
blockSwitches["pen_setPenSizeTo"] = [noopSwitch, {
opcode: "pen_changePenSizeBy"
}];
blockSwitches["pen_changePenSizeBy"] = [{
opcode: "pen_setPenSizeTo"
}, noopSwitch];
blockSwitches["music_setTempo"] = [noopSwitch, {
opcode: "music_changeTempo"
}];
blockSwitches["music_changeTempo"] = [{
opcode: "music_setTempo"
}, noopSwitch];
}
if (addon.settings.get("sa")) {
const logProc = "\u200B\u200Blog\u200B\u200B %s";
const warnProc = "\u200B\u200Bwarn\u200B\u200B %s";
const errorProc = "\u200B\u200Berror\u200B\u200B %s";
const logMessage = msg("debugger_log");
const warnMessage = msg("debugger_warn");
const errorMessage = msg("debugger_error");
const logSwitch = {
mutate: {
proccode: logProc
},
msg: logMessage
};
const warnSwitch = {
mutate: {
proccode: warnProc
},
msg: warnMessage
};
const errorSwitch = {
mutate: {
proccode: errorProc
},
msg: errorMessage
};
procedureSwitches[logProc] = [{
msg: logMessage,
isNoop: true
}, warnSwitch, errorSwitch];
procedureSwitches[warnProc] = [logSwitch, {
msg: warnMessage,
isNoop: true
}, errorSwitch];
procedureSwitches[errorProc] = [logSwitch, warnSwitch, {
msg: errorMessage,
isNoop: true
}];
} // Switching for these is implemented by Scratch. We only define them here to optionally add a border.
// Because we don't implement the switching ourselves, this is not controlled by the data category option.
blockSwitches["data_variable"] = [];
blockSwitches["data_listcontents"] = [];
};
buildSwitches();
addon.settings.addEventListener("change", buildSwitches);
/**
* @param {*} workspace
* @param {Element} xmlBlock
*/
const pasteBlockXML = (workspace, xmlBlock) => {
// Similar to https://github.com/LLK/scratch-blocks/blob/7575c9a0f2c267676569c4b102b76d77f35d9fd6/core/workspace_svg.js#L1020
// but without the collision checking.
const block = ScratchBlocks.Xml.domToBlock(xmlBlock, workspace);
const x = +xmlBlock.getAttribute("x");
const y = +xmlBlock.getAttribute("y"); // Don't need to handle RTL here
block.moveBy(x, y);
return block;
};
/**
* @param {string} shadowType The type of shadow eg. "math_number"
* @returns {string} The name of the shadow's inner field that contains the user-visible value
*/
const getShadowFieldName = shadowType => {
// This is non-comprehensive.
if (shadowType === "text") {
return "TEXT";
}
if (shadowType === "colour_picker") {
return "COLOUR";
}
return "NUM";
};
/**
* @template T
* @param {T|()=>T} value
* @returns {T}
*/
const callIfFunction = value => {
if (typeof value === "function") {
return value();
}
return value;
};
const menuCallbackFactory = (block, opcodeData) => () => {
if (opcodeData.isNoop) {
return;
}
if (opcodeData.fieldValue) {
block.setFieldValue(opcodeData.fieldValue, "VALUE");
return;
}
try {
ScratchBlocks.Events.setGroup(true);
const workspace = block.workspace;
const blocksToBringToForeground = []; // Split inputs before we clone the block.
if (opcodeData.splitInputs) {
for (const inputName of opcodeData.splitInputs) {
const input = block.getInput(inputName);
if (!input) {
continue;
}
const connection = input.connection;
if (!connection) {
continue;
}
if (connection.isConnected()) {
const targetBlock = connection.targetBlock();
if (targetBlock.isShadow()) {// Deleting shadows is handled later.
} else {
connection.disconnect();
blocksToBringToForeground.push(targetBlock);
}
}
}
} // Make a copy of the block with the proper type set.
// It doesn't seem to be possible to change a Block's type after it's created, so we'll just make a new block instead.
const xml = ScratchBlocks.Xml.blockToDom(block); // blockToDomWithXY's handling of RTL is strange, so we encode the position ourselves.
const position = block.getRelativeToSurfaceXY();
xml.setAttribute("x", position.x);
xml.setAttribute("y", position.y);
if (opcodeData.opcode) {
xml.setAttribute("type", opcodeData.opcode);
}
const parentBlock = block.getParent();
let parentConnection;
let blockConnectionType;
if (parentBlock) {
// If the block has a parent, find the parent -> child connection that will be reattached later.
const parentConnections = parentBlock.getConnections_();
parentConnection = parentConnections.find(c => c.targetConnection && c.targetConnection.sourceBlock_ === block); // There's two types of connections from child -> parent. We need to figure out which one is used.
const blockConnections = block.getConnections_();
const blockToParentConnection = blockConnections.find(c => c.targetConnection && c.targetConnection.sourceBlock_ === parentBlock);
blockConnectionType = blockToParentConnection.type;
} // Array.from creates a clone of the children list. This is important as we may remove
// children as we iterate.
for (const child of Array.from(xml.children)) {
const oldName = child.getAttribute("name"); // Any inputs that were supposed to be split that were not should be removed.
// (eg. shadow inputs)
if (opcodeData.splitInputs && opcodeData.splitInputs.includes(oldName)) {
xml.removeChild(child);
continue;
}
const newName = opcodeData.remapInputName && opcodeData.remapInputName[oldName];
if (newName) {
child.setAttribute("name", newName);
}
const newShadowType = opcodeData.remapShadowType && opcodeData.remapShadowType[oldName];
if (newShadowType) {
const valueNode = child.firstChild;
const fieldNode = valueNode.firstChild;
valueNode.setAttribute("type", newShadowType);
fieldNode.setAttribute("name", getShadowFieldName(newShadowType));
}
const fieldValueMap = opcodeData.mapFieldValues && opcodeData.mapFieldValues[oldName];
if (fieldValueMap && child.tagName === "FIELD") {
const oldValue = child.innerText;
const newValue = fieldValueMap[oldValue];
if (typeof newValue === "string") {
child.innerText = newValue;
}
}
}
if (opcodeData.mutate) {
const mutation = xml.querySelector("mutation");
for (const [key, value] of Object.entries(opcodeData.mutate)) {
mutation.setAttribute(key, value);
}
}
if (opcodeData.createInputs) {
for (const [inputName, inputData] of Object.entries(opcodeData.createInputs)) {
const valueElement = document.createElement("value");
valueElement.setAttribute("name", inputName);
const shadowElement = document.createElement("shadow");
shadowElement.setAttribute("type", inputData.shadowType);
const shadowFieldElement = document.createElement("field");
shadowFieldElement.setAttribute("name", getShadowFieldName(inputData.shadowType));
shadowFieldElement.innerText = callIfFunction(inputData.value);
shadowElement.appendChild(shadowFieldElement);
valueElement.appendChild(shadowElement);
xml.appendChild(valueElement);
}
} // Remove the old block and insert the new one.
block.dispose();
const newBlock = pasteBlockXML(workspace, xml);
if (parentConnection) {
// Search for the same type of connection on the new block as on the old block.
const newBlockConnections = newBlock.getConnections_();
const newBlockConnection = newBlockConnections.find(c => c.type === blockConnectionType);
newBlockConnection.connect(parentConnection);
}
for (const otherBlock of blocksToBringToForeground) {
// By re-appending the element, we move it to the end, which will make it display
// on top.
const svgRoot = otherBlock.getSvgRoot();
svgRoot.parentNode.appendChild(svgRoot);
}
} finally {
ScratchBlocks.Events.setGroup(false);
}
};
const uniques = array => [...new Set(array)];
addon.tab.createBlockContextMenu((items, block) => {
if (!addon.self.disabled) {
const type = block.type;
let switches = blockSwitches[block.type] || [];
const customArgsMode = addon.settings.get("customargs") ? addon.settings.get("customargsmode") : "off";
if (customArgsMode !== "off" && ["argument_reporter_boolean", "argument_reporter_string_number"].includes(type) && // if the arg is a shadow, it's in a procedures_prototype so we don't want it to be switchable
!block.isShadow()) {
const customBlocks = getCustomBlocks();
if (customArgsMode === "all") {
switch (type) {
case "argument_reporter_string_number":
switches = Object.values(customBlocks).map(cb => cb.stringArgs).flat(1);
break;
case "argument_reporter_boolean":
switches = Object.values(customBlocks).map(cb => cb.boolArgs).flat(1);
break;
}
} else if (customArgsMode === "defOnly") {
const root = block.getRootBlock();
if (root.type !== "procedures_definition") return items;
const customBlockObj = customBlocks[root.getChildren(true)[0].getProcCode()];
switch (type) {
case "argument_reporter_string_number":
switches = customBlockObj.stringArgs;
break;
case "argument_reporter_boolean":
switches = customBlockObj.boolArgs;
break;
}
}
const currentValue = block.getFieldValue("VALUE");
switches = uniques(switches).map(i => ({
isNoop: i === currentValue,
fieldValue: i,
msg: i
}));
}
if (block.type === "procedures_call") {
const proccode = block.getProcCode();
if (procedureSwitches[proccode]) {
switches = procedureSwitches[proccode];
}
}
if (!addon.settings.get("noop")) {
switches = switches.filter(i => !i.isNoop);
}
switches.forEach((opcodeData, i) => {
const makeSpaceItemIndex = items.findIndex(obj => obj._isDevtoolsFirstItem);
const insertBeforeIndex = makeSpaceItemIndex !== -1 ? // If "make space" button exists, add own items before it
makeSpaceItemIndex : // If there's no such button, insert at end
items.length;
const text = opcodeData.msg ? opcodeData.msg : opcodeData.opcode ? msg(opcodeData.opcode) : msg(block.type);
items.splice(insertBeforeIndex, 0, {
enabled: true,
text,
callback: menuCallbackFactory(block, opcodeData),
separator: i === 0
});
});
if (block.type === "data_variable" || block.type === "data_listcontents") {
// Add top border to first variable (if it exists)
const delBlockIndex = items.findIndex(item => item.text === ScratchBlocks.Msg.DELETE_BLOCK); // firstVariableItem might be undefined, a variable to switch to,
// or an item added by editor-devtools (or any addon before this one)
const firstVariableItem = items[delBlockIndex + 1];
if (firstVariableItem) firstVariableItem.separator = true;
}
}
return items;
}, {
blocks: true
}); // https://github.com/LLK/scratch-blocks/blob/abbfe93136fef57fdfb9a077198b0bc64726f012/blocks_vertical/procedures.js#L207-L215
// Returns a list like ["%s", "%d"]
const parseArguments = code => code.split(/(?=[^\\]%[nbs])/g).map(i => i.trim()).filter(i => i.charAt(0) === "%").map(i => i.substring(0, 2));
const getCustomBlocks = () => {
const customBlocks = {};
const target = vm.editingTarget;
Object.values(target.blocks._blocks).filter(block => block.opcode === "procedures_prototype").forEach(block => {
const procCode = block.mutation.proccode;
const argumentNames = JSON.parse(block.mutation.argumentnames); // argumentdefaults is unreliable, so we have to parse the procedure code to determine argument types
const parsedArguments = parseArguments(procCode);
const stringArgs = [];
const boolArgs = [];
for (let i = 0; i < argumentNames.length; i++) {
if (parsedArguments[i] === "%b") {
boolArgs.push(argumentNames[i]);
} else {
stringArgs.push(argumentNames[i]);
}
}
customBlocks[procCode] = {
stringArgs,
boolArgs
};
});
return customBlocks;
};
});
/***/ }),
/***/ "./src/addons/addons/color-picker/_runtime_entry.js":
/*!**********************************************************!*\
!*** ./src/addons/addons/color-picker/_runtime_entry.js ***!
\**********************************************************/
/*! exports provided: resources */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "resources", function() { return resources; });
/* harmony import */ var _userscript_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./userscript.js */ "./src/addons/addons/color-picker/userscript.js");
/* harmony import */ var _css_loader_style_css__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! css-loader!./style.css */ "./node_modules/css-loader/index.js!./src/addons/addons/color-picker/style.css");
/* harmony import */ var _css_loader_style_css__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_css_loader_style_css__WEBPACK_IMPORTED_MODULE_1__);
/* generated by pull.js */
const resources = {
"userscript.js": _userscript_js__WEBPACK_IMPORTED_MODULE_0__["default"],
"style.css": _css_loader_style_css__WEBPACK_IMPORTED_MODULE_1___default.a
};
/***/ }),
/***/ "./src/addons/addons/color-picker/code-editor.js":
/*!*******************************************************!*\
!*** ./src/addons/addons/color-picker/code-editor.js ***!
\*******************************************************/
/*! exports provided: default */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony import */ var _libraries_common_cs_normalize_color_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../../libraries/common/cs/normalize-color.js */ "./src/addons/libraries/common/cs/normalize-color.js");
/* harmony import */ var _libraries_common_cs_rate_limiter_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../../libraries/common/cs/rate-limiter.js */ "./src/addons/libraries/common/cs/rate-limiter.js");
/* harmony import */ var _libraries_thirdparty_cs_tinycolor_min_js__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ../../libraries/thirdparty/cs/tinycolor-min.js */ "./src/addons/libraries/thirdparty/cs/tinycolor-min.js");
/* harmony default export */ __webpack_exports__["default"] = (async ({
addon,
console,
msg
}) => {
// 250-ms rate limit
const rateLimiter = new _libraries_common_cs_rate_limiter_js__WEBPACK_IMPORTED_MODULE_1__["default"](250);
const getColor = element => {
const {
children
} = element.parentElement; // h: 0 - 360
const h = children[1].getAttribute("aria-valuenow"); // s: 0 - 1
const s = children[3].getAttribute("aria-valuenow"); // v: 0 - 255, divide by 255
const vMultipliedBy255 = children[5].getAttribute("aria-valuenow");
const v = Number(vMultipliedBy255) / 255;
return Object(_libraries_thirdparty_cs_tinycolor_min_js__WEBPACK_IMPORTED_MODULE_2__["default"])("hsv(".concat(h, ", ").concat(s, ", ").concat(v || 0, ")")).toHexString();
};
const setColor = (hex, element) => {
hex = Object(_libraries_common_cs_normalize_color_js__WEBPACK_IMPORTED_MODULE_0__["normalizeHex"])(hex);
if (!addon.tab.redux.state || !addon.tab.redux.state.scratchGui) return; // The only way to reliably set color is to invoke eye dropper via click()
// then faking that the eye dropper reported the value.
const onEyeDropperClosed = ({
detail
}) => {
if (detail.action.type !== "scratch-gui/color-picker/DEACTIVATE_COLOR_PICKER") return;
addon.tab.redux.removeEventListener("statechanged", onEyeDropperClosed);
setTimeout(() => {
document.body.classList.remove("sa-hide-eye-dropper-background");
}, 50);
};
const onEyeDropperOpened = ({
detail
}) => {
if (detail.action.type !== "scratch-gui/color-picker/ACTIVATE_COLOR_PICKER") return;
addon.tab.redux.removeEventListener("statechanged", onEyeDropperOpened);
addon.tab.redux.addEventListener("statechanged", onEyeDropperClosed);
setTimeout(() => {
addon.tab.redux.dispatch({
type: "scratch-gui/color-picker/DEACTIVATE_COLOR_PICKER",
color: hex
});
}, 50);
};
addon.tab.redux.addEventListener("statechanged", onEyeDropperOpened);
document.body.classList.add("sa-hide-eye-dropper-background");
element.click();
};
const addColorPicker = () => {
const element = document.querySelector("button.scratchEyedropper");
rateLimiter.abort(false);
addon.tab.redux.initialize();
const defaultColor = getColor(element);
const saColorPicker = Object.assign(document.createElement("div"), {
className: "sa-color-picker sa-color-picker-code"
});
addon.tab.displayNoneWhileDisabled(saColorPicker, {
display: "flex"
});
const saColorPickerColor = Object.assign(document.createElement("input"), {
className: "sa-color-picker-color sa-color-picker-code-color",
type: "color",
value: defaultColor || "#000000"
});
const saColorPickerText = Object.assign(document.createElement("input"), {
className: addon.tab.scratchClass("input_input-form", {
others: "sa-color-picker-text sa-color-picker-code-text"
}),
type: "text",
pattern: "^#?([0-9a-fA-F]{3}){1,2}$",
placeholder: msg("hex"),
value: defaultColor || ""
});
saColorPickerColor.addEventListener("input", () => rateLimiter.limit(() => setColor(saColorPickerText.value = saColorPickerColor.value, element)));
saColorPickerText.addEventListener("change", () => {
const {
value
} = saColorPickerText;
if (!Object(_libraries_common_cs_normalize_color_js__WEBPACK_IMPORTED_MODULE_0__["getHexRegex"])().test(value)) return;
setColor(saColorPickerColor.value = Object(_libraries_common_cs_normalize_color_js__WEBPACK_IMPORTED_MODULE_0__["normalizeHex"])(value), element);
});
saColorPicker.appendChild(saColorPickerColor);
saColorPicker.appendChild(saColorPickerText);
element.parentElement.insertBefore(saColorPicker, element);
};
const ScratchBlocks = await addon.tab.traps.getBlockly();
const originalShowEditor = ScratchBlocks.FieldColourSlider.prototype.showEditor_;
ScratchBlocks.FieldColourSlider.prototype.showEditor_ = function (...args) {
const r = originalShowEditor.call(this, ...args);
addColorPicker();
return r;
};
});
/***/ }),
/***/ "./src/addons/addons/color-picker/userscript.js":
/*!******************************************************!*\
!*** ./src/addons/addons/color-picker/userscript.js ***!
\******************************************************/
/*! exports provided: default */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony import */ var _code_editor_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./code-editor.js */ "./src/addons/addons/color-picker/code-editor.js");
/* harmony default export */ __webpack_exports__["default"] = (async api => {
Object(_code_editor_js__WEBPACK_IMPORTED_MODULE_0__["default"])(api);
});
/***/ }),
/***/ "./src/addons/addons/editor-comment-previews/_runtime_entry.js":
/*!*********************************************************************!*\
!*** ./src/addons/addons/editor-comment-previews/_runtime_entry.js ***!
\*********************************************************************/
/*! exports provided: resources */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "resources", function() { return resources; });
/* harmony import */ var _userscript_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./userscript.js */ "./src/addons/addons/editor-comment-previews/userscript.js");
/* harmony import */ var _css_loader_userstyle_css__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! css-loader!./userstyle.css */ "./node_modules/css-loader/index.js!./src/addons/addons/editor-comment-previews/userstyle.css");
/* harmony import */ var _css_loader_userstyle_css__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_css_loader_userstyle_css__WEBPACK_IMPORTED_MODULE_1__);
/* generated by pull.js */
const resources = {
"userscript.js": _userscript_js__WEBPACK_IMPORTED_MODULE_0__["default"],
"userstyle.css": _css_loader_userstyle_css__WEBPACK_IMPORTED_MODULE_1___default.a
};
/***/ }),
/***/ "./src/addons/addons/editor-comment-previews/userscript.js":
/*!*****************************************************************!*\
!*** ./src/addons/addons/editor-comment-previews/userscript.js ***!
\*****************************************************************/
/*! exports provided: default */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony default export */ __webpack_exports__["default"] = (async function ({
addon,
console
}) {
const vm = addon.tab.traps.vm;
const updateStyles = () => {
previewInner.classList.toggle("sa-comment-preview-delay", addon.settings.get("delay") !== "none");
previewInner.classList.toggle("sa-comment-preview-reduce-transparency", addon.settings.get("reduce-transparency"));
previewInner.classList.toggle("sa-comment-preview-fade", !addon.settings.get("reduce-animation"));
};
const afterDelay = cb => {
if (!previewInner.classList.contains("sa-comment-preview-hidden")) {
// If not hidden, updating immediately is preferred
cb();
return;
}
const delay = addon.settings.get("delay");
if (delay === "long") return setTimeout(cb, 500);
if (delay === "short") return setTimeout(cb, 300);
cb();
};
let hoveredElement = null;
let showTimeout = null;
let mouseX = 0;
let mouseY = 0;
let doNotShowUntilMoveMouse = false;
const previewOuter = document.createElement("div");
previewOuter.classList.add("sa-comment-preview-outer");
const previewInner = document.createElement("div");
previewInner.classList.add("sa-comment-preview-inner");
previewInner.classList.add("sa-comment-preview-hidden");
updateStyles();
addon.settings.addEventListener("change", updateStyles);
previewOuter.appendChild(previewInner);
document.body.appendChild(previewOuter);
const getBlock = id => vm.editingTarget.blocks.getBlock(id) || vm.runtime.flyoutBlocks.getBlock(id);
const getComment = block => block && block.comment && vm.editingTarget.comments[block.comment];
const getProcedureDefinitionBlock = procCode => {
const procedurePrototype = Object.values(vm.editingTarget.blocks._blocks).find(i => i.opcode === "procedures_prototype" && i.mutation.proccode === procCode);
if (procedurePrototype) {
// Usually `parent` will exist but sometimes it doesn't
if (procedurePrototype.parent) {
return getBlock(procedurePrototype.parent);
}
const id = procedurePrototype.id;
return Object.values(vm.editingTarget.blocks._blocks).find(i => i.opcode === "procedures_definition" && i.inputs.custom_block && i.inputs.custom_block.block === id);
}
return null;
};
const setText = text => {
previewInner.innerText = text;
previewInner.classList.remove("sa-comment-preview-hidden");
updateMousePosition();
};
const updateMousePosition = () => {
previewOuter.style.transform = "translate(".concat(mouseX + 8, "px, ").concat(mouseY + 8, "px)");
};
const hidePreview = () => {
if (hoveredElement) {
hoveredElement = null;
previewInner.classList.add("sa-comment-preview-hidden");
}
};
document.addEventListener("mouseover", e => {
if (addon.self.disabled) {
return;
}
clearTimeout(showTimeout);
if (doNotShowUntilMoveMouse) {
return;
}
const el = e.target.closest(".blocklyBubbleCanvas > g, .blocklyBlockCanvas .blocklyDraggable[data-id]");
if (el === hoveredElement) {
// Nothing to do.
return;
}
if (!el) {
hidePreview();
return;
}
let text = null;
if (addon.settings.get("hover-view") && e.target.closest(".blocklyBubbleCanvas > g") && // Hovering over the thin line that connects comments to blocks should never show a preview
!e.target.closest("line")) {
const collapsedText = el.querySelector("text.scratchCommentText");
if (collapsedText.getAttribute("display") !== "none") {
const textarea = el.querySelector("textarea");
text = textarea.value;
}
} else if (e.target.closest(".blocklyBlockCanvas .blocklyDraggable[data-id]")) {
const id = el.dataset.id;
const block = getBlock(id);
const comment = getComment(block);
if (addon.settings.get("hover-view-block") && comment) {
text = comment.text;
} else if (block && block.opcode === "procedures_call" && addon.settings.get("hover-view-procedure")) {
const procCode = block.mutation.proccode;
const procedureDefinitionBlock = getProcedureDefinitionBlock(procCode);
const procedureComment = getComment(procedureDefinitionBlock);
if (procedureComment) {
text = procedureComment.text;
}
}
}
if (text !== null && text.trim() !== "") {
showTimeout = afterDelay(() => {
hoveredElement = el;
setText(text);
});
} else {
hidePreview();
}
});
document.addEventListener("mousemove", e => {
mouseX = e.clientX;
mouseY = e.clientY;
doNotShowUntilMoveMouse = false;
if (addon.settings.get("follow-mouse") && !previewInner.classList.contains("sa-comment-preview-hidden")) {
updateMousePosition();
}
});
document.addEventListener("mousedown", () => {
hidePreview();
doNotShowUntilMoveMouse = true;
}, {
capture: true
});
});
/***/ }),
/***/ "./src/addons/addons/editor-devtools/DevTools.js":
/*!*******************************************************!*\
!*** ./src/addons/addons/editor-devtools/DevTools.js ***!
\*******************************************************/
/*! exports provided: default */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "default", function() { return DevTools; });
/* harmony import */ var _DomHelpers_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./DomHelpers.js */ "./src/addons/addons/editor-devtools/DomHelpers.js");
/* harmony import */ var _UndoGroup_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./UndoGroup.js */ "./src/addons/addons/editor-devtools/UndoGroup.js");
// import ShowBroadcast from "./show-broadcast.js";
class DevTools {
constructor(addon, msg, m) {
this.addon = addon;
this.msg = msg;
this.m = m;
/**
* @type {VirtualMachine}
*/
this.domHelpers = new _DomHelpers_js__WEBPACK_IMPORTED_MODULE_0__["default"](addon);
this.codeTab = null;
this.costTab = null;
this.costTabBody = null;
this.selVarID = null;
this.canShare = false;
this.mouseXY = {
x: 0,
y: 0
};
}
async init() {
this.addContextMenus();
while (true) {
const root = await this.addon.tab.waitForElement("ul[class*=gui_tab-list_]", {
markAsSeen: true,
reduxEvents: ["scratch-gui/mode/SET_PLAYER", "fontsLoaded/SET_FONTS_LOADED", "scratch-gui/locales/SELECT_LOCALE"],
reduxCondition: state => !state.scratchGui.mode.isPlayerOnly
});
this.initInner(root);
}
}
async addContextMenus() {
const blockly = await this.addon.tab.traps.getBlockly();
const oldCleanUpFunc = blockly.WorkspaceSvg.prototype.cleanUp;
const self = this;
blockly.WorkspaceSvg.prototype.cleanUp = function () {
if (self.addon.settings.get("enableCleanUpPlus")) {
self.doCleanUp();
} else {
oldCleanUpFunc.call(this);
}
};
let originalMsg = blockly.Msg.CLEAN_UP;
if (this.addon.settings.get("enableCleanUpPlus")) blockly.Msg.CLEAN_UP = this.m("clean-plus");
this.addon.settings.addEventListener("change", () => {
if (this.addon.settings.get("enableCleanUpPlus")) blockly.Msg.CLEAN_UP = this.m("clean-plus");else blockly.Msg.CLEAN_UP = originalMsg;
});
this.addon.tab.createBlockContextMenu((items, block) => {
items.push({
enabled: blockly.clipboardXml_,
text: this.m("paste"),
separator: true,
_isDevtoolsFirstItem: true,
callback: () => {
let ids = this.getTopBlockIDs();
document.dispatchEvent(new KeyboardEvent("keydown", {
keyCode: 86,
ctrlKey: true,
griff: true
}));
setTimeout(() => {
this.beginDragOfNewBlocksNotInIDs(ids);
}, 10);
}
});
return items;
}, {
workspace: true
});
this.addon.tab.createBlockContextMenu((items, block) => {
items.push({
enabled: true,
text: this.m("make-space"),
_isDevtoolsFirstItem: true,
callback: () => {
this.doCleanUp(block);
},
separator: true
}, {
enabled: true,
text: this.m("copy-all"),
callback: () => {
this.eventCopyClick(block);
},
separator: true
}, {
enabled: true,
text: this.m("copy-block"),
callback: () => {
this.eventCopyClick(block, 1);
}
}, {
enabled: true,
text: this.m("cut-block"),
callback: () => {
this.eventCopyClick(block, 2);
}
}); // const BROADCAST_BLOCKS = ["event_whenbroadcastreceived", "event_broadcast", "event_broadcastandwait"];
// if (BROADCAST_BLOCKS.includes(block.type)) {
// // Show Broadcast
// const broadcastId = this.showBroadcastSingleton.getAssociatedBroadcastId(block.id);
// if (broadcastId) {
// ["Senders", "Receivers"].forEach((showKey, i) => {
// items.push({
// enabled: true,
// text: this.msg(`show-${showKey}`.toLowerCase()),
// callback: () => {
// this.showBroadcastSingleton[`show${showKey}`](broadcastId);
// },
// separator: i == 0,
// });
// });
// }
// }
return items;
}, {
blocks: true
});
this.addon.tab.createBlockContextMenu((items, block) => {
if (block.getCategory() === "data" || block.getCategory() === "data-lists") {
this.selVarID = block.getVars()[0];
items.push({
enabled: true,
text: this.m("swap", {
var: block.getCategory() === "data" ? this.m("variables") : this.m("lists")
}),
callback: async () => {
let wksp = this.getWorkspace();
let v = wksp.getVariableById(this.selVarID); // prompt() returns Promise in desktop app
let varName = await window.prompt(this.msg("replace", {
name: v.name
}));
if (varName) {
this.doReplaceVariable(this.selVarID, varName, v.type);
}
},
separator: true
});
}
return items;
}, {
blocks: true,
flyout: true
});
}
getWorkspace() {
return Blockly.getMainWorkspace();
}
isCostumeEditor() {
return this.costTab.className.indexOf("gui_is-selected") >= 0;
}
/**
* A nicely ordered version of the top blocks
* @returns {[Blockly.Block]}
*/
getTopBlocks() {
let result = this.getOrderedTopBlockColumns();
let columns = result.cols;
/**
* @type {[[Blockly.Block]]}
*/
let topBlocks = [];
for (const col of columns) {
topBlocks = topBlocks.concat(col.blocks);
}
return topBlocks;
}
/**
* A much nicer way of laying out the blocks into columns
*/
doCleanUp(block) {
let workspace = this.getWorkspace();
let makeSpaceForBlock = block && block.getRootBlock();
_UndoGroup_js__WEBPACK_IMPORTED_MODULE_1__["default"].startUndoGroup(workspace);
let result = this.getOrderedTopBlockColumns(true);
let columns = result.cols;
let orphanCount = result.orphans.blocks.length;
if (orphanCount > 0 && !block) {
let message = this.msg("orphaned", {
count: orphanCount
});
if (confirm(message)) {
for (const block of result.orphans.blocks) {
block.dispose();
}
} else {
columns.unshift(result.orphans);
}
}
let cursorX = 48;
let maxWidths = result.maxWidths;
for (const column of columns) {
let cursorY = 64;
let maxWidth = 0;
for (const block of column.blocks) {
let extraWidth = block === makeSpaceForBlock ? 380 : 0;
let extraHeight = block === makeSpaceForBlock ? 480 : 72;
let xy = block.getRelativeToSurfaceXY();
if (cursorX - xy.x !== 0 || cursorY - xy.y !== 0) {
block.moveBy(cursorX - xy.x, cursorY - xy.y);
}
let heightWidth = block.getHeightWidth();
cursorY += heightWidth.height + extraHeight;
let maxWidthWithComments = maxWidths[block.id] || 0;
maxWidth = Math.max(maxWidth, Math.max(heightWidth.width + extraWidth, maxWidthWithComments));
}
cursorX += maxWidth + 96;
}
let topComments = workspace.getTopComments();
for (const comment of topComments) {
if (comment.setVisible) {
comment.setVisible(false);
comment.needsAutoPositioning_ = true;
comment.setVisible(true);
}
}
setTimeout(() => {
// Locate unused local variables...
let workspace = this.getWorkspace();
let map = workspace.getVariableMap();
let vars = map.getVariablesOfType("");
let unusedLocals = [];
for (const row of vars) {
if (row.isLocal) {
let usages = map.getVariableUsesById(row.getId());
if (!usages || usages.length === 0) {
unusedLocals.push(row);
}
}
}
if (unusedLocals.length > 0) {
const unusedCount = unusedLocals.length;
let message = this.msg("unused-var", {
count: unusedCount
});
for (let i = 0; i < unusedLocals.length; i++) {
let orphan = unusedLocals[i];
if (i > 0) {
message += ", ";
}
message += orphan.name;
}
if (confirm(message)) {
for (const orphan of unusedLocals) {
workspace.deleteVariableById(orphan.getId());
}
}
} // Locate unused local lists...
let lists = map.getVariablesOfType("list");
let unusedLists = [];
for (const row of lists) {
if (row.isLocal) {
let usages = map.getVariableUsesById(row.getId());
if (!usages || usages.length === 0) {
unusedLists.push(row);
}
}
}
if (unusedLists.length > 0) {
const unusedCount = unusedLists.length;
let message = this.msg("unused-list", {
count: unusedCount
});
for (let i = 0; i < unusedLists.length; i++) {
let orphan = unusedLists[i];
if (i > 0) {
message += ", ";
}
message += orphan.name;
}
if (confirm(message)) {
for (const orphan of unusedLists) {
workspace.deleteVariableById(orphan.getId());
}
}
}
_UndoGroup_js__WEBPACK_IMPORTED_MODULE_1__["default"].endUndoGroup(workspace);
}, 100);
}
/**
* Badly Orphaned - might want to delete these!
* @param topBlock
* @returns {boolean}
*/
isBlockAnOrphan(topBlock) {
return !!topBlock.outputConnection;
}
/**
* Split the top blocks into ordered columns
* @param separateOrphans true to keep all orphans separate
* @returns {{orphans: {blocks: [Block], x: number, count: number}, cols: [Col]}}
*/
getOrderedTopBlockColumns(separateOrphans) {
let w = this.getWorkspace();
let topBlocks = w.getTopBlocks();
let maxWidths = {};
if (separateOrphans) {
let topComments = w.getTopComments(); // todo: tie comments to blocks... find widths and width of block stack row...
for (const comment of topComments) {
// comment.autoPosition_();
// Hiding and showing repositions the comment right next to it's block - nice!
if (comment.setVisible) {
comment.setVisible(false);
comment.needsAutoPositioning_ = true;
comment.setVisible(true); // let bb = comment.block_.svgPath_.getBBox();
let right = comment.getBoundingRectangle().bottomRight.x; // Get top block for stack...
let root = comment.block_.getRootBlock();
let left = root.getBoundingRectangle().topLeft.x;
maxWidths[root.id] = Math.max(right - left, maxWidths[root.id] || 0);
}
}
} // Default scratch ordering is horrid... Lets try something more clever.
/**
* @type {Col[]}
*/
let cols = [];
const TOLERANCE = 256;
let orphans = {
x: -999999,
count: 0,
blocks: []
};
for (const topBlock of topBlocks) {
// let r = b.getBoundingRectangle();
let position = topBlock.getRelativeToSurfaceXY();
/**
* @type {Col}
*/
let bestCol = null;
let bestError = TOLERANCE;
if (separateOrphans && this.isBlockAnOrphan(topBlock)) {
orphans.blocks.push(topBlock);
continue;
} // Find best columns
for (const col of cols) {
let err = Math.abs(position.x - col.x);
if (err < bestError) {
bestError = err;
bestCol = col;
}
}
if (bestCol) {
// We found a column that we fitted into
bestCol.x = (bestCol.x * bestCol.count + position.x) / ++bestCol.count; // re-average the columns as more items get added...
bestCol.blocks.push(topBlock);
} else {
// Create a new column
cols.push(new Col(position.x, 1, [topBlock]));
}
} // if (orphans.blocks.length > 0) {
// cols.push(orphans);
// }
// Sort columns, then blocks inside the columns
cols.sort((a, b) => a.x - b.x);
for (const col of cols) {
col.blocks.sort((a, b) => a.getRelativeToSurfaceXY().y - b.getRelativeToSurfaceXY().y);
}
return {
cols: cols,
orphans: orphans,
maxWidths: maxWidths
};
}
/**
* Find all the uses of a named variable.
* @param {string} id ID of the variable to find.
* @return {!Array.<!Blockly.Block>} Array of block usages.
*/
getVariableUsesById(id) {
let uses = [];
let topBlocks = this.getTopBlocks(true); // todo: Confirm this was the right getTopBlocks?
for (const topBlock of topBlocks) {
/** @type {!Array<!Blockly.Block>} */
let kids = topBlock.getDescendants();
for (const block of kids) {
/** @type {!Array<!Blockly.VariableModel>} */
let blockVariables = block.getVarModels();
if (blockVariables) {
for (const blockVar of blockVariables) {
if (blockVar.getId() === id) {
uses.push(block);
}
}
}
}
}
return uses;
}
/**
* Quick and dirty replace all instances of one variable / list with another variable / list
* @param varId original variable name
* @param newVarName new variable name
* @param type type of variable ("" = variable, anything else is a list?
*/
doReplaceVariable(varId, newVarName, type) {
let wksp = this.getWorkspace();
let v = wksp.getVariable(newVarName, type);
if (!v) {
alert(this.msg("var-not-exist"));
return;
}
let newVId = v.getId();
_UndoGroup_js__WEBPACK_IMPORTED_MODULE_1__["default"].startUndoGroup(wksp);
let blocks = this.getVariableUsesById(varId);
for (const block of blocks) {
try {
if (type === "") {
block.getField("VARIABLE").setValue(newVId);
} else {
block.getField("LIST").setValue(newVId);
}
} catch (e) {// ignore
}
}
_UndoGroup_js__WEBPACK_IMPORTED_MODULE_1__["default"].endUndoGroup(wksp);
}
/*
function doInjectScripts(codeString) {
let w = getWorkspace();
let xml = new XML(); // document.implementation.createDocument(null, "xml");
let x = xml.xmlDoc.firstChild;
let tree = math.parse(codeString);
console.log(tree);
const binaryOperatorTypes = {
add: "operator_add",
subtract: "operator_subtract",
this.multiply: "operator_multiply",
divide: "operator_divide",
};
const BLOCK_TYPE = {
number: "math_number",
text: "text",
};
function translateMathToXml(x, tree, shadowType) {
let xShadowField = null;
if (shadowType) {
let xShadow = xml.newXml(x, "shadow", { type: shadowType });
if (shadowType === BLOCK_TYPE.number) {
xShadowField = xml.newXml(xShadow, "field", { name: "NUM" });
} else if (shadowType === BLOCK_TYPE.text) {
xShadowField = xml.newXml(xShadow, "field", { name: "TEXT" });
}
}
if (!tree || !tree.type) {
return;
}
if (tree.type === "OperatorNode") {
let operatorType = binaryOperatorTypes[tree.fn];
if (operatorType) {
let xOp = newXml(x, "block", { type: operatorType });
translateMathToXml(xml.newXml(xOp, "value", { name: "NUM1" }), tree.args[0], BLOCK_TYPE.number);
translateMathToXml(xml.newXml(xOp, "value", { name: "NUM2" }), tree.args[1], BLOCK_TYPE.number);
return;
}
return;
}
if (tree.type === "ConstantNode") {
// number or text in quotes
if (xShadowField) {
xml.setAttr(xShadowField, { text: tree.value });
}
return;
}
if (tree.type === "SymbolNode") {
// variable
let xVar = xml.newXml(x, "block", { type: "data_variable" });
xml.newXml(xVar, "field", { name: "VARIABLE", text: tree.name });
return;
}
if (tree.type === "FunctionNode") {
// Method Call
if (tree.fn.name === "join") {
let xOp = newXml(x, "block", { type: "operator_join" });
translateMathToXml(xml.newXml(xOp, "value", { name: "STRING1" }), tree.args[0], BLOCK_TYPE.text);
translateMathToXml(xml.newXml(xOp, "value", { name: "STRING2" }), tree.args[1], BLOCK_TYPE.text);
return;
}
}
}
translateMathToXml(x, tree);
console.log(x);
let ids = Blockly.Xml.domToWorkspace(x, w);
console.log(ids);
}
*/
/*
function clickInject(e) {
let codeString = window.prompt("Griffpatch: Enter an expression (i.e. a+2*3)");
if (codeString) {
doInjectScripts(codeString);
}
e.preventDefault();
return false;
}
*/
/**
* Returns a Set of the top blocks in this workspace / sprite
* @returns {Set<any>} Set of top blocks
*/
getTopBlockIDs() {
let wksp = this.getWorkspace();
let topBlocks = wksp.getTopBlocks();
let ids = new Set();
for (const block of topBlocks) {
ids.add(block.id);
}
return ids;
}
/**
* Initiates a drag event for all block stacks except those in the set of ids.
* But why? - Because we know all the ids of the existing stacks before we paste / duplicate - so we can find the
* new stack by excluding all the known ones.
* @param ids Set of previously known ids
*/
beginDragOfNewBlocksNotInIDs(ids) {
if (!this.addon.settings.get("enablePasteBlocksAtMouse")) {
return;
}
let wksp = this.getWorkspace();
let topBlocks = wksp.getTopBlocks();
for (const block of topBlocks) {
if (!ids.has(block.id)) {
// console.log("I found a new block!!! - " + block.id);
// todo: move the block to the mouse pointer?
let mouseXYClone = {
x: this.mouseXY.x,
y: this.mouseXY.y
};
block.setIntersects(true); // fixes offscreen block pasting in TurboWarp
this.domHelpers.triggerDragAndDrop(block.svgPath_, null, mouseXYClone);
}
}
}
updateMousePosition(e) {
this.mouseXY.x = e.clientX;
this.mouseXY.y = e.clientY;
}
eventMouseMove(e) {
this.updateMousePosition(e);
}
eventKeyDown(e) {
const switchCostume = up => {
// todo: select previous costume
let selected = this.costTabBody.querySelector("div[class*='sprite-selector-item_is-selected']");
let node = up ? selected.parentNode.previousSibling : selected.parentNode.nextSibling;
if (node) {
let wrapper = node.closest("div[class*=gui_flex-wrapper]");
node.querySelector("div[class^='sprite-selector-item_sprite-name']").click();
node.scrollIntoView({
behavior: "auto",
block: "center",
inline: "start"
});
wrapper.scrollTop = 0;
}
};
if (document.URL.indexOf("editor") <= 0) {
return;
}
let ctrlKey = e.ctrlKey || e.metaKey;
if (e.keyCode === 37 && ctrlKey) {
// Ctrl + Left Arrow Key
if (document.activeElement.tagName === "INPUT") {
return;
}
if (this.isCostumeEditor()) {
switchCostume(true);
e.cancelBubble = true;
e.preventDefault();
return true;
}
}
if (e.keyCode === 39 && ctrlKey) {
// Ctrl + Right Arrow Key
if (document.activeElement.tagName === "INPUT") {
return;
}
if (this.isCostumeEditor()) {
switchCostume(false);
e.cancelBubble = true;
e.preventDefault();
return true;
}
}
if (e.keyCode === 86 && ctrlKey && !e.griff) {
// Ctrl + V
// Set a timeout so we can take control of the paste after the event
let ids = this.getTopBlockIDs();
setTimeout(() => {
this.beginDragOfNewBlocksNotInIDs(ids);
}, 10);
} // if (e.keyCode === 220 && (!document.activeElement || document.activeElement.tagName === 'INPUT')) {
//
// }
}
eventCopyClick(block, blockOnly) {
let wksp = this.getWorkspace();
if (block) {
block.select();
let next = blockOnly ? block.getNextBlock() : null;
if (next) {
next.unplug(false); // setParent(null);
} // separate child temporarily
document.dispatchEvent(new KeyboardEvent("keydown", {
keyCode: 67,
ctrlKey: true
}));
if (next || blockOnly === 2) {
setTimeout(() => {
if (next) {
wksp.undo(); // undo the unplug above...
}
if (blockOnly === 2) {
_UndoGroup_js__WEBPACK_IMPORTED_MODULE_1__["default"].startUndoGroup(wksp);
block.dispose(true);
_UndoGroup_js__WEBPACK_IMPORTED_MODULE_1__["default"].endUndoGroup(wksp);
}
}, 0);
}
}
}
eventMouseDown(e) {
this.updateMousePosition(e);
}
eventMouseUp(e) {
this.updateMousePosition(e);
}
initInner(root) {
let guiTabs = root.childNodes;
if (this.codeTab && guiTabs[0] !== this.codeTab) {
// We have been CHANGED!!! - Happens when going to project page, and then back inside again!!!
this.domHelpers.unbindAllEvents();
}
this.codeTab = guiTabs[0];
this.costTab = guiTabs[1];
this.costTabBody = document.querySelector("div[aria-labelledby=" + this.costTab.id + "]");
this.domHelpers.bindOnce(document, "keydown", (...e) => this.eventKeyDown(...e), true);
this.domHelpers.bindOnce(document, "mousemove", (...e) => this.eventMouseMove(...e), true);
this.domHelpers.bindOnce(document, "mousedown", (...e) => this.eventMouseDown(...e), true); // true to capture all mouse downs 'before' the dom events handle them
this.domHelpers.bindOnce(document, "mouseup", (...e) => this.eventMouseUp(...e), true);
}
}
class Col {
/**
* @param x {Number} x position (for ordering)
* @param count {Number}
* @param blocks {[Block]}
*/
constructor(x, count, blocks) {
/**
* x position (for ordering)
* @type {Number}
*/
this.x = x;
/**
* @type {Number}
*/
this.count = count;
/**
* @type {[Blockly.Block]}
*/
this.blocks = blocks;
}
}
/***/ }),
/***/ "./src/addons/addons/editor-devtools/DomHelpers.js":
/*!*********************************************************!*\
!*** ./src/addons/addons/editor-devtools/DomHelpers.js ***!
\*********************************************************/
/*! exports provided: default */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "default", function() { return DomHelpers; });
class DomHelpers {
constructor(addon) {
this.addon = addon;
this.vm = addon.tab.traps.vm;
/**
* @type {eventDetails[]}
*/
this.events = [];
}
/**
* Simulate a drag and drop programmatically through javascript
* @param selectorDrag
* @param selectorDrop
* @param mouseXY
* @param [shiftKey=false]
* @returns {boolean}
*/
triggerDragAndDrop(selectorDrag, selectorDrop, mouseXY, shiftKey) {
// function for triggering mouse events
shiftKey = shiftKey || false;
let fireMouseEvent = function fireMouseEvent(type, elem, centerX, centerY) {
let evt = document.createEvent("MouseEvents");
evt.initMouseEvent(type, true, true, window, 1, 1, 1, centerX, centerY, shiftKey, false, false, false, 0, elem);
elem.dispatchEvent(evt);
}; // fetch target elements
let elemDrag = selectorDrag; // document.querySelector(selectorDrag);
let elemDrop = selectorDrop; // document.querySelector(selectorDrop);
if (!elemDrag
/* || !elemDrop*/
) {
return false;
} // calculate positions
let pos = elemDrag.getBoundingClientRect();
let center1X = Math.floor((pos.left + pos.right) / 2);
let center1Y = Math.floor((pos.top + pos.bottom) / 2); // mouse over dragged element and mousedown
fireMouseEvent("mouseover", elemDrag, center1X, center1Y);
fireMouseEvent("mousedown", elemDrag, center1X, center1Y); // start dragging process over to drop target
fireMouseEvent("dragstart", elemDrag, center1X, center1Y);
fireMouseEvent("drag", elemDrag, center1X, center1Y);
fireMouseEvent("mousemove", elemDrag, center1X, center1Y);
if (!elemDrop) {
if (mouseXY) {
// console.log(mouseXY);
let center2X = mouseXY.x;
let center2Y = mouseXY.y;
fireMouseEvent("drag", elemDrag, center2X, center2Y);
fireMouseEvent("mousemove", elemDrag, center2X, center2Y);
}
return false;
}
pos = elemDrop.getBoundingClientRect();
let center2X = Math.floor((pos.left + pos.right) / 2);
let center2Y = Math.floor((pos.top + pos.bottom) / 2);
fireMouseEvent("drag", elemDrag, center2X, center2Y);
fireMouseEvent("mousemove", elemDrop, center2X, center2Y); // trigger dragging process on top of drop target
fireMouseEvent("mouseenter", elemDrop, center2X, center2Y);
fireMouseEvent("dragenter", elemDrop, center2X, center2Y);
fireMouseEvent("mouseover", elemDrop, center2X, center2Y);
fireMouseEvent("dragover", elemDrop, center2X, center2Y); // release dragged element on top of drop target
fireMouseEvent("drop", elemDrop, center2X, center2Y);
fireMouseEvent("dragend", elemDrag, center2X, center2Y);
fireMouseEvent("mouseup", elemDrag, center2X, center2Y);
return true;
}
bindOnce(dom, event, func, capture) {
capture = !!capture;
dom.removeEventListener(event, func, capture);
dom.addEventListener(event, func, capture);
this.events.push(new eventDetails(dom, event, func, capture));
}
unbindAllEvents() {
for (const event of this.events) {
event.dom.removeEventListener(event.event, event.func, event.capture);
}
this.events = [];
}
}
/**
* A record of an event
*/
class eventDetails {
constructor(dom, event, func, capture) {
this.dom = dom;
this.event = event;
this.func = func;
this.capture = capture;
}
}
/***/ }),
/***/ "./src/addons/addons/editor-devtools/UndoGroup.js":
/*!********************************************************!*\
!*** ./src/addons/addons/editor-devtools/UndoGroup.js ***!
\********************************************************/
/*! exports provided: default */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "default", function() { return UndoGroup; });
/**
* This class is dedicated to maintaining the Undo stack of Blockly
* It allows us to initiate an undo group such that all subsequent operations are recorded as a single
* undoable transaction.
*/
class UndoGroup {
/**
* Start an Undo group - begin recording
* @param workspace the workspace
*/
static startUndoGroup(workspace) {
const undoStack = workspace.undoStack_;
if (undoStack.length) {
undoStack[undoStack.length - 1]._devtoolsLastUndo = true;
}
}
/**
* End an Undo group - stops recording
* @param workspace the workspace
*/
static endUndoGroup(workspace) {
const undoStack = workspace.undoStack_; // Events (responsible for undoStack updates) are delayed with a setTimeout(f, 0)
// https://github.com/LLK/scratch-blocks/blob/f159a1779e5391b502d374fb2fdd0cb5ca43d6a2/core/events.js#L182
setTimeout(() => {
const group = generateUID();
for (let i = undoStack.length - 1; i >= 0 && !undoStack[i]._devtoolsLastUndo; i--) {
undoStack[i].group = group;
}
}, 0);
}
}
/**
* https://github.com/LLK/scratch-blocks/blob/f159a1779e5391b502d374fb2fdd0cb5ca43d6a2/core/events.js#L182
* @returns {string}
* @private
*/
function generateUID() {
const CHARACTERS = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789!#$%()*+,-./:;=?@[]^_`{|}~";
let result = "";
for (let i = 0; i < 20; i++) {
result += CHARACTERS[Math.floor(Math.random() * CHARACTERS.length)];
}
return result;
}
/***/ }),
/***/ "./src/addons/addons/editor-devtools/_runtime_entry.js":
/*!*************************************************************!*\
!*** ./src/addons/addons/editor-devtools/_runtime_entry.js ***!
\*************************************************************/
/*! exports provided: resources */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "resources", function() { return resources; });
/* harmony import */ var _userscript_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./userscript.js */ "./src/addons/addons/editor-devtools/userscript.js");
/* harmony import */ var _url_loader_icon_close_svg__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! url-loader!./icon--close.svg */ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/editor-devtools/icon--close.svg");
/* generated by pull.js */
const resources = {
"userscript.js": _userscript_js__WEBPACK_IMPORTED_MODULE_0__["default"],
"icon--close.svg": _url_loader_icon_close_svg__WEBPACK_IMPORTED_MODULE_1__["default"]
};
/***/ }),
/***/ "./src/addons/addons/editor-devtools/userscript.js":
/*!*********************************************************!*\
!*** ./src/addons/addons/editor-devtools/userscript.js ***!
\*********************************************************/
/*! exports provided: default */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony import */ var _DevTools_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./DevTools.js */ "./src/addons/addons/editor-devtools/DevTools.js");
/* harmony default export */ __webpack_exports__["default"] = (async function ({
addon,
console,
msg,
safeMsg: m
}) {
const devTools = new _DevTools_js__WEBPACK_IMPORTED_MODULE_0__["default"](addon, msg, m);
devTools.init();
});
/***/ }),
/***/ "./src/addons/addons/editor-searchable-dropdowns/_runtime_entry.js":
/*!*************************************************************************!*\
!*** ./src/addons/addons/editor-searchable-dropdowns/_runtime_entry.js ***!
\*************************************************************************/
/*! exports provided: resources */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "resources", function() { return resources; });
/* harmony import */ var _userscript_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./userscript.js */ "./src/addons/addons/editor-searchable-dropdowns/userscript.js");
/* harmony import */ var _css_loader_userscript_css__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! css-loader!./userscript.css */ "./node_modules/css-loader/index.js!./src/addons/addons/editor-searchable-dropdowns/userscript.css");
/* harmony import */ var _css_loader_userscript_css__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_css_loader_userscript_css__WEBPACK_IMPORTED_MODULE_1__);
/* generated by pull.js */
const resources = {
"userscript.js": _userscript_js__WEBPACK_IMPORTED_MODULE_0__["default"],
"userscript.css": _css_loader_userscript_css__WEBPACK_IMPORTED_MODULE_1___default.a
};
/***/ }),
/***/ "./src/addons/addons/editor-searchable-dropdowns/userscript.js":
/*!*********************************************************************!*\
!*** ./src/addons/addons/editor-searchable-dropdowns/userscript.js ***!
\*********************************************************************/
/*! exports provided: default */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony default export */ __webpack_exports__["default"] = (async function ({
addon,
console,
msg
}) {
const Blockly = await addon.tab.traps.getBlockly();
const vm = addon.tab.traps.vm;
const SCRATCH_ITEMS_TO_HIDE = ["RENAME_VARIABLE_ID", "DELETE_VARIABLE_ID", "NEW_BROADCAST_MESSAGE_ID", // From rename-broadcasts addon
"RENAME_BROADCAST_MESSAGE_ID"];
const canUseAsGlobalVariableName = (name, type) => {
return !vm.runtime.getAllVarNamesOfType(type).includes(name);
};
const canUseAsLocalVariableName = (name, type) => {
return !vm.editingTarget.lookupVariableByNameAndType(name, type);
};
const ADDON_ITEMS = {
createGlobalVariable: {
enabled: name => canUseAsGlobalVariableName(name, ""),
createVariable: (workspace, name) => workspace.createVariable(name)
},
createLocalVariable: {
enabled: name => canUseAsLocalVariableName(name, ""),
createVariable: (workspace, name) => workspace.createVariable(name, "", null, true)
},
createGlobalList: {
enabled: name => canUseAsGlobalVariableName(name, "list"),
createVariable: (workspace, name) => workspace.createVariable(name, "list")
},
createLocalList: {
enabled: name => canUseAsLocalVariableName(name, "list"),
createVariable: (workspace, name) => workspace.createVariable(name, "list", null, true)
},
createBroadcast: {
enabled: name => canUseAsGlobalVariableName(name, "broadcast_msg"),
createVariable: (workspace, name) => workspace.createVariable(name, "broadcast_msg")
}
};
let blocklyDropDownContent = null;
let blocklyDropdownMenu = null;
let searchBar = null; // Contains DOM and addon state
let items = [];
let searchedItems = []; // Tracks internal Scratch state
let currentDropdownOptions = [];
let resultOfLastGetOptions = [];
const oldDropDownDivShow = Blockly.DropDownDiv.show;
Blockly.DropDownDiv.show = function (...args) {
blocklyDropdownMenu = document.querySelector(".blocklyDropdownMenu");
if (!blocklyDropdownMenu) {
return oldDropDownDivShow.call(this, ...args);
}
blocklyDropdownMenu.focus = () => {}; // no-op focus() so it can't steal it from the search bar
searchBar = document.createElement("input");
addon.tab.displayNoneWhileDisabled(searchBar, {
display: "flex"
});
searchBar.type = "text";
searchBar.addEventListener("input", updateSearch);
searchBar.addEventListener("keydown", handleKeyDownEvent);
searchBar.classList.add("u-dropdown-searchbar");
blocklyDropdownMenu.insertBefore(searchBar, blocklyDropdownMenu.firstChild);
items = Array.from(blocklyDropdownMenu.children).filter(child => child.tagName !== "INPUT").map(element => ({
element,
text: element.textContent
}));
currentDropdownOptions = resultOfLastGetOptions;
updateSearch(); // Call the original show method after adding everything so that it can perform the correct size calculations
const ret = oldDropDownDivShow.call(this, ...args); // Lock the size of the dropdown
blocklyDropDownContent = Blockly.DropDownDiv.getContentDiv();
blocklyDropDownContent.style.width = getComputedStyle(blocklyDropDownContent).width;
blocklyDropDownContent.style.height = getComputedStyle(blocklyDropDownContent).height; // This is really strange, but if you don't reinsert the search bar into the DOM then focus() doesn't work
blocklyDropdownMenu.insertBefore(searchBar, blocklyDropdownMenu.firstChild);
searchBar.focus();
return ret;
};
const oldDropDownDivClearContent = Blockly.DropDownDiv.clearContent;
Blockly.DropDownDiv.clearContent = function () {
oldDropDownDivClearContent.call(this);
items = [];
searchedItems = [];
Blockly.DropDownDiv.content_.style.height = "";
};
const oldFieldDropdownGetOptions = Blockly.FieldDropdown.prototype.getOptions;
Blockly.FieldDropdown.prototype.getOptions = function () {
const options = oldFieldDropdownGetOptions.call(this);
const block = this.sourceBlock_;
const isStage = vm.editingTarget && vm.editingTarget.isStage;
if (block) {
if (block.category_ === "data") {
options.push(getMenuItemMessage("createGlobalVariable"));
if (!isStage) {
options.push(getMenuItemMessage("createLocalVariable"));
}
} else if (block.category_ === "data-lists") {
options.push(getMenuItemMessage("createGlobalList"));
if (!isStage) {
options.push(getMenuItemMessage("createLocalList"));
}
} else if (block.type === "event_broadcast_menu" || block.type === "event_whenbroadcastreceived") {
options.push(getMenuItemMessage("createBroadcast"));
}
} // Options aren't normally stored anywhere, so we'll store them ourselves.
resultOfLastGetOptions = options;
return options;
};
const oldFieldVariableOnItemSelected = Blockly.FieldVariable.prototype.onItemSelected;
Blockly.FieldVariable.prototype.onItemSelected = function (menu, menuItem) {
const sourceBlock = this.sourceBlock_;
if (sourceBlock && sourceBlock.workspace && searchBar.value.length !== 0) {
const workspace = sourceBlock.workspace;
const optionId = menuItem.getValue();
if (Object.prototype.hasOwnProperty.call(ADDON_ITEMS, optionId)) {
const addonItem = ADDON_ITEMS[optionId];
Blockly.Events.setGroup(true);
const variable = addonItem.createVariable(workspace, searchBar.value.trim());
if (this.sourceBlock_) this.setValue(variable.getId());
Blockly.Events.setGroup(false);
return;
}
}
return oldFieldVariableOnItemSelected.call(this, menu, menuItem);
};
function selectItem(item, click) {
// You can't just use click() or focus() because Blockly uses mousedown and mouseup handlers, not click handlers.
item.dispatchEvent(new MouseEvent("mousedown", {
relatedTarget: item,
bubbles: true
}));
if (click) item.dispatchEvent(new MouseEvent("mouseup", {
relatedTarget: item,
bubbles: true
})); // Scroll the item into view if it is offscreen.
const itemTop = item.offsetTop;
const itemEnd = itemTop + item.offsetHeight;
const scrollTop = blocklyDropDownContent.scrollTop;
const scrollHeight = blocklyDropDownContent.offsetHeight;
const scrollEnd = scrollTop + scrollHeight;
if (scrollTop > itemTop) {
blocklyDropDownContent.scrollTop = itemTop;
} else if (itemEnd > scrollEnd) {
blocklyDropDownContent.scrollTop = itemEnd - scrollHeight;
}
}
function performSearch() {
const rawQuery = searchBar.value.trim();
const query = rawQuery.trim().toLowerCase();
const rank = (item, index) => {
// Negative number will hide
// Higher numbers will appear first
const option = currentDropdownOptions[index];
const optionId = option[1];
if (SCRATCH_ITEMS_TO_HIDE.includes(optionId)) {
return query ? -1 : 0;
} else if (Object.prototype.hasOwnProperty.call(ADDON_ITEMS, optionId)) {
if (!query) {
return -1;
}
const addonInfo = ADDON_ITEMS[optionId];
if (addonInfo.enabled(rawQuery)) {
item.element.lastChild.lastChild.textContent = getMenuItemMessage(optionId)[0];
return 0;
}
return -1;
}
const itemText = item.text.toLowerCase();
if (query === itemText) {
return 2;
}
if (itemText.startsWith(query)) {
return 1;
}
if (itemText.includes(query)) {
return 0;
}
return -1;
};
return items.map((item, index) => ({
item,
score: rank(item, index)
})).sort(({
score: scoreA
}, {
score: scoreB
}) => Math.max(0, scoreB) - Math.max(0, scoreA));
}
function updateSearch() {
const previousSearchedItems = searchedItems;
searchedItems = performSearch();
let needToUpdateDOM = previousSearchedItems.length !== searchedItems.length;
if (!needToUpdateDOM) {
for (let i = 0; i < searchedItems.length; i++) {
if (searchedItems[i].item !== previousSearchedItems[i].item) {
needToUpdateDOM = true;
break;
}
}
}
if (needToUpdateDOM && previousSearchedItems.length > 0) {
for (const {
item
} of previousSearchedItems) {
item.element.remove();
}
for (const {
item
} of searchedItems) {
blocklyDropdownMenu.appendChild(item.element);
}
}
for (const {
item,
score
} of searchedItems) {
item.element.hidden = score < 0;
}
}
function handleKeyDownEvent(event) {
if (event.key === "Enter") {
// Reimplement enter to select item to account for hidden items and default to the top item.
event.stopPropagation();
event.preventDefault();
const selectedItem = document.querySelector(".goog-menuitem-highlight");
if (selectedItem && !selectedItem.hidden) {
selectItem(selectedItem, true);
return;
}
const selectedBlock = Blockly.selected;
if (searchBar.value === "" && selectedBlock) {
if (selectedBlock.type === "event_broadcast" || selectedBlock.type === "event_broadcastandwait" || selectedBlock.type === "event_whenbroadcastreceived") {
// The top item of these dropdowns is always "New message"
// When pressing enter on an empty search bar, we close the dropdown instead of making a new broadcast.
Blockly.DropDownDiv.hide();
return;
}
}
for (const {
item
} of searchedItems) {
if (!item.element.hidden) {
selectItem(item.element, true);
break;
}
} // If there is no top value, do nothing and leave the dropdown open
} else if (event.key === "Escape") {
Blockly.DropDownDiv.hide();
} else if (event.key === "ArrowDown" || event.key === "ArrowUp") {
// Reimplement keyboard navigation to account for hidden items.
event.preventDefault();
event.stopPropagation();
const items = searchedItems.filter(i => i.score >= 0).map(i => i.item);
if (items.length === 0) {
return;
}
let selectedIndex = -1;
for (let i = 0; i < items.length; i++) {
if (items[i].element.classList.contains("goog-menuitem-highlight")) {
selectedIndex = i;
break;
}
}
const lastIndex = items.length - 1;
let newIndex = 0;
if (event.key === "ArrowDown") {
if (selectedIndex === -1 || selectedIndex === lastIndex) {
newIndex = 0;
} else {
newIndex = selectedIndex + 1;
}
} else {
if (selectedIndex === -1 || selectedIndex === 0) {
newIndex = lastIndex;
} else {
newIndex = selectedIndex - 1;
}
}
selectItem(items[newIndex].element, false);
}
}
function getMenuItemMessage(message) {
var _searchBar;
// Format used internally by Scratch:
// [human readable name, internal name]
return [msg(message, {
name: ((_searchBar = searchBar) === null || _searchBar === void 0 ? void 0 : _searchBar.value.trim()) || ""
}), message];
}
});
/***/ }),
/***/ "./src/addons/addons/find-bar/_runtime_entry.js":
/*!******************************************************!*\
!*** ./src/addons/addons/find-bar/_runtime_entry.js ***!
\******************************************************/
/*! exports provided: resources */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "resources", function() { return resources; });
/* harmony import */ var _userscript_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./userscript.js */ "./src/addons/addons/find-bar/userscript.js");
/* harmony import */ var _css_loader_userstyle_css__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! css-loader!./userstyle.css */ "./node_modules/css-loader/index.js!./src/addons/addons/find-bar/userstyle.css");
/* harmony import */ var _css_loader_userstyle_css__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_css_loader_userstyle_css__WEBPACK_IMPORTED_MODULE_1__);
/* generated by pull.js */
const resources = {
"userscript.js": _userscript_js__WEBPACK_IMPORTED_MODULE_0__["default"],
"userstyle.css": _css_loader_userstyle_css__WEBPACK_IMPORTED_MODULE_1___default.a
};
/***/ }),
/***/ "./src/addons/addons/find-bar/blockly/BlockItem.js":
/*!*********************************************************!*\
!*** ./src/addons/addons/find-bar/blockly/BlockItem.js ***!
\*********************************************************/
/*! exports provided: default */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "default", function() { return BlockItem; });
class BlockItem {
constructor(cls, procCode, labelID, y) {
this.cls = cls;
this.procCode = procCode;
this.labelID = labelID;
this.y = y;
this.lower = procCode.toLowerCase();
/**
* An Array of block ids
* @type {Array.<string>}
*/
this.clones = null;
this.eventName = null;
}
/**
* True if the blockID matches a black represented by this BlockItem
* @param id
* @returns {boolean}
*/
matchesID(id) {
if (this.labelID === id) {
return true;
}
if (this.clones) {
for (const cloneID of this.clones) {
if (cloneID === id) {
return true;
}
}
}
return false;
}
}
/***/ }),
/***/ "./src/addons/addons/find-bar/userscript.js":
/*!**************************************************!*\
!*** ./src/addons/addons/find-bar/userscript.js ***!
\**************************************************/
/*! exports provided: default */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony import */ var _blockly_BlockItem_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./blockly/BlockItem.js */ "./src/addons/addons/find-bar/blockly/BlockItem.js");
/* harmony import */ var _blockly_BlockInstance_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./blockly/BlockInstance.js */ "./src/addons/addons/find-bar/blockly/BlockInstance.js");
/* harmony import */ var _blockly_Utils_js__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./blockly/Utils.js */ "./src/addons/addons/find-bar/blockly/Utils.js");
/* harmony default export */ __webpack_exports__["default"] = (async function ({
addon,
msg,
console
}) {
const Blockly = await addon.tab.traps.getBlockly();
class FindBar {
constructor() {
this.utils = new _blockly_Utils_js__WEBPACK_IMPORTED_MODULE_2__["default"](addon);
this.prevValue = "";
this.findBarOuter = null;
this.findWrapper = null;
this.findInput = null;
this.dropdownOut = null;
this.dropdown = new Dropdown(this.utils);
document.addEventListener("keydown", e => this.eventKeyDown(e), true);
}
get workspace() {
return Blockly.getMainWorkspace();
}
createDom(root) {
this.findBarOuter = document.createElement("div");
this.findBarOuter.className = "sa-find-bar";
addon.tab.displayNoneWhileDisabled(this.findBarOuter, {
display: "flex"
});
root.appendChild(this.findBarOuter);
this.findWrapper = this.findBarOuter.appendChild(document.createElement("span"));
this.findWrapper.className = "sa-find-wrapper";
this.dropdownOut = this.findWrapper.appendChild(document.createElement("label"));
this.dropdownOut.className = "sa-find-dropdown-out";
this.findInput = this.dropdownOut.appendChild(document.createElement("input"));
this.findInput.className = addon.tab.scratchClass("input_input-form", {
others: "sa-find-input"
}); // for <label>
this.findInput.id = "sa-find-input";
this.findInput.type = "search";
this.findInput.placeholder = msg("find-placeholder");
this.findInput.autocomplete = "off";
this.dropdownOut.appendChild(this.dropdown.createDom());
this.bindEvents();
this.tabChanged();
}
bindEvents() {
this.findInput.addEventListener("focus", () => this.inputChange());
this.findInput.addEventListener("keydown", e => this.inputKeyDown(e));
this.findInput.addEventListener("keyup", () => this.inputChange());
this.findInput.addEventListener("focusout", () => this.hideDropDown());
}
tabChanged() {
if (!this.findBarOuter) {
return;
}
const tab = addon.tab.redux.state.scratchGui.editorTab.activeTabIndex;
const visible = tab === 0 || tab === 1 || tab === 2;
this.findBarOuter.hidden = !visible;
}
inputChange() {
this.showDropDown(); // Filter the list...
let val = (this.findInput.value || "").toLowerCase();
if (val === this.prevValue) {
// No change so don't re-filter
return;
}
this.prevValue = val;
this.dropdown.blocks = null; // Hide items in list that do not contain filter text
let listLI = this.dropdown.items;
for (const li of listLI) {
let procCode = li.data.procCode;
let i = li.data.lower.indexOf(val);
if (i >= 0) {
li.style.display = "block";
while (li.firstChild) {
li.removeChild(li.firstChild);
}
if (i > 0) {
li.appendChild(document.createTextNode(procCode.substring(0, i)));
}
let bText = document.createElement("b");
bText.appendChild(document.createTextNode(procCode.substr(i, val.length)));
li.appendChild(bText);
if (i + val.length < procCode.length) {
li.appendChild(document.createTextNode(procCode.substr(i + val.length)));
}
} else {
li.style.display = "none";
}
}
}
inputKeyDown(e) {
this.dropdown.inputKeyDown(e); // Enter
if (e.keyCode === 13) {
this.findInput.blur();
return;
} // Escape
if (e.keyCode === 27) {
if (this.findInput.value.length > 0) {
this.findInput.value = ""; // Clear search first, then close on second press
this.inputChange();
} else {
this.findInput.blur();
}
e.preventDefault();
return;
}
}
eventKeyDown(e) {
if (addon.self.disabled || !this.findBarOuter) return;
let ctrlKey = e.ctrlKey || e.metaKey;
if (e.key === "f" && ctrlKey && !e.shiftKey) {
// Ctrl + F (Override default Ctrl+F find)
this.findInput.focus();
this.findInput.select();
e.cancelBubble = true;
e.preventDefault();
return true;
}
if (e.keyCode === 37 && ctrlKey) {
// Ctrl + Left Arrow Key
if (document.activeElement.tagName === "INPUT") {
return;
}
if (this.selectedTab === 0) {
this.utils.navigationHistory.goBack();
e.cancelBubble = true;
e.preventDefault();
return true;
}
}
if (e.keyCode === 39 && ctrlKey) {
// Ctrl + Right Arrow Key
if (document.activeElement.tagName === "INPUT") {
return;
}
if (this.selectedTab === 0) {
this.utils.navigationHistory.goForward();
e.cancelBubble = true;
e.preventDefault();
return true;
}
}
}
showDropDown(focusID, instanceBlock) {
if (!focusID && this.dropdownOut.classList.contains("visible")) {
return;
} // special '' vs null... - null forces a reevaluation
this.prevValue = focusID ? "" : null; // Clear the previous value of the input search
this.dropdownOut.classList.add("visible");
let scratchBlocks = this.selectedTab === 0 ? this.getScratchBlocks() : this.selectedTab === 1 ? this.getScratchCostumes() : this.selectedTab === 2 ? this.getScratchSounds() : [];
this.dropdown.empty();
for (const proc of scratchBlocks) {
let item = this.dropdown.addItem(proc);
if (focusID) {
if (proc.matchesID(focusID)) {
this.dropdown.onItemClick(item, instanceBlock);
} else {
item.style.display = "none";
}
}
}
this.utils.offsetX = this.dropdownOut.getBoundingClientRect().width + 32;
this.utils.offsetY = 32;
}
hideDropDown() {
this.dropdownOut.classList.remove("visible");
}
get selectedTab() {
return addon.tab.redux.state.scratchGui.editorTab.activeTabIndex;
}
getScratchBlocks() {
let myBlocks = [];
let myBlocksByProcCode = {};
let topBlocks = this.workspace.getTopBlocks();
/**
* @param cls
* @param txt
* @param root
* @returns BlockItem
*/
function addBlock(cls, txt, root) {
let id = root.id ? root.id : root.getId ? root.getId() : null;
let clone = myBlocksByProcCode[txt];
if (clone) {
if (!clone.clones) {
clone.clones = [];
}
clone.clones.push(id);
return clone;
}
let items = new _blockly_BlockItem_js__WEBPACK_IMPORTED_MODULE_0__["default"](cls, txt, id, 0);
items.y = root.getRelativeToSurfaceXY ? root.getRelativeToSurfaceXY().y : null;
myBlocks.push(items);
myBlocksByProcCode[txt] = items;
return items;
}
function getDescFromField(root) {
let fields = root.inputList[0];
let desc;
for (const fieldRow of fields.fieldRow) {
desc = (desc ? desc + " " : "") + fieldRow.getText();
}
return desc;
}
for (const root of topBlocks) {
if (root.type === "procedures_definition") {
const label = root.getChildren()[0];
const procCode = label.getProcCode();
if (!procCode) {
continue;
}
const indexOfLabel = root.inputList.findIndex(i => i.fieldRow.length > 0);
if (indexOfLabel === -1) {
continue;
}
const translatedDefine = root.inputList[indexOfLabel].fieldRow[0].getText();
const message = indexOfLabel === 0 ? "".concat(translatedDefine, " ").concat(procCode) : "".concat(procCode, " ").concat(translatedDefine);
addBlock("define", message, root);
continue;
}
if (root.type === "event_whenflagclicked") {
addBlock("flag", getDescFromField(root), root); // "When Flag Clicked"
continue;
}
if (root.type === "event_whenbroadcastreceived") {
const fieldRow = root.inputList[0].fieldRow;
let eventName = fieldRow.find(input => input.name === "BROADCAST_OPTION").getText();
addBlock("receive", "event " + eventName, root).eventName = eventName;
continue;
}
if (root.type.substr(0, 10) === "event_when") {
addBlock("event", getDescFromField(root), root); // "When Flag Clicked"
continue;
}
if (root.type === "control_start_as_clone") {
addBlock("event", getDescFromField(root), root); // "when I start as a clone"
continue;
}
}
let map = this.workspace.getVariableMap();
let vars = map.getVariablesOfType("");
for (const row of vars) {
addBlock(row.isLocal ? "var" : "VAR", (row.isLocal ? "var " : "VAR ") + row.name, row);
}
let lists = map.getVariablesOfType("list");
for (const row of lists) {
addBlock(row.isLocal ? "list" : "LIST", (row.isLocal ? "list " : "LIST ") + row.name, row);
}
const events = this.getCallsToEvents();
for (const event of events) {
addBlock("receive", "event " + event.eventName, event.block).eventName = event.eventName;
}
const clsOrder = {
flag: 0,
receive: 1,
event: 2,
define: 3,
var: 4,
VAR: 5,
list: 6,
LIST: 7
};
myBlocks.sort((a, b) => {
let t = clsOrder[a.cls] - clsOrder[b.cls];
if (t !== 0) {
return t;
}
if (a.lower < b.lower) {
return -1;
}
if (a.lower > b.lower) {
return 1;
}
return a.y - b.y;
});
return myBlocks;
}
getScratchCostumes() {
let costumes = this.utils.getEditingTarget().getCostumes();
let items = [];
let i = 0;
for (const costume of costumes) {
let item = new _blockly_BlockItem_js__WEBPACK_IMPORTED_MODULE_0__["default"]("costume", costume.name, costume.assetId, i);
items.push(item);
i++;
}
return items;
}
getScratchSounds() {
let sounds = this.utils.getEditingTarget().getSounds();
let items = [];
let i = 0;
for (const sound of sounds) {
let item = new _blockly_BlockItem_js__WEBPACK_IMPORTED_MODULE_0__["default"]("sound", sound.name, sound.assetId, i);
items.push(item);
i++;
}
return items;
}
getCallsToEvents() {
const uses = [];
const alreadyFound = new Set();
for (const block of this.workspace.getAllBlocks()) {
if (block.type !== "event_broadcast" && block.type !== "event_broadcastandwait") {
continue;
}
const broadcastInput = block.getChildren()[0];
if (!broadcastInput) {
continue;
}
let eventName = "";
if (broadcastInput.type === "event_broadcast_menu") {
eventName = broadcastInput.inputList[0].fieldRow[0].getText();
} else {
eventName = msg("complex-broadcast");
}
if (!alreadyFound.has(eventName)) {
alreadyFound.add(eventName);
uses.push({
eventName: eventName,
block: block
});
}
}
return uses;
}
}
class Dropdown {
constructor(utils) {
this.utils = utils;
this.el = null;
this.items = [];
this.selected = null;
this.carousel = new Carousel(this.utils);
}
get workspace() {
return Blockly.getMainWorkspace();
}
createDom() {
this.el = document.createElement("ul");
this.el.className = "sa-find-dropdown";
return this.el;
}
inputKeyDown(e) {
// Up Arrow
if (e.keyCode === 38) {
this.navigateFilter(-1);
e.preventDefault();
return;
} // Down Arrow
if (e.keyCode === 40) {
this.navigateFilter(1);
e.preventDefault();
return;
} // Enter
if (e.keyCode === 13) {
// Any selected on enter? if not select now
if (this.selected) {
this.navigateFilter(1);
}
e.preventDefault();
return;
}
this.carousel.inputKeyDown(e);
}
navigateFilter(dir) {
let nxt;
if (this.selected && this.selected.style.display !== "none") {
nxt = dir === -1 ? this.selected.previousSibling : this.selected.nextSibling;
} else {
nxt = this.items[0];
dir = 1;
}
while (nxt && nxt.style.display === "none") {
nxt = dir === -1 ? nxt.previousSibling : nxt.nextSibling;
}
if (nxt) {
nxt.scrollIntoView({
block: "nearest"
});
this.onItemClick(nxt);
}
}
addItem(proc) {
const item = document.createElement("li");
item.innerText = proc.procCode;
item.data = proc;
const colorIds = {
receive: "events",
event: "events",
define: "more",
var: "data",
VAR: "data",
list: "data-lists",
LIST: "data-lists",
costume: "looks",
sound: "sounds"
};
if (proc.cls === "flag") {
item.className = "sa-find-flag";
} else {
const colorId = colorIds[proc.cls];
item.className = "sa-block-color sa-block-color-".concat(colorId);
}
item.addEventListener("mousedown", e => {
this.onItemClick(item);
e.preventDefault();
e.cancelBubble = true;
return false;
});
this.items.push(item);
this.el.appendChild(item);
return item;
}
onItemClick(item, instanceBlock) {
if (this.selected && this.selected !== item) {
this.selected.classList.remove("sel");
this.selected = null;
}
if (this.selected !== item) {
item.classList.add("sel");
this.selected = item;
}
let cls = item.data.cls;
if (cls === "costume" || cls === "sound") {
// Viewing costumes/sounds - jump to selected costume/sound
const assetPanel = document.querySelector("[class^=asset-panel_wrapper]");
if (assetPanel) {
const reactInstance = assetPanel[addon.tab.traps.getInternalKey(assetPanel)];
const reactProps = reactInstance.child.stateNode.props;
reactProps.onItemClick(item.data.y);
const selectorList = assetPanel.firstChild.firstChild;
selectorList.children[item.data.y].scrollIntoView({
behavior: "auto",
block: "center",
inline: "start"
}); // The wrapper seems to scroll when we use the function above.
let wrapper = assetPanel.closest("div[class*=gui_flex-wrapper]");
wrapper.scrollTop = 0;
}
} else if (cls === "var" || cls === "VAR" || cls === "list" || cls === "LIST") {
// Search now for all instances
let blocks = this.getVariableUsesById(item.data.labelID);
this.carousel.build(item, blocks, instanceBlock);
} else if (cls === "define") {
let blocks = this.getCallsToProcedureById(item.data.labelID);
this.carousel.build(item, blocks, instanceBlock);
} else if (cls === "receive") {
/*
let blocks = [this.workspace.getBlockById(li.data.labelID)];
if (li.data.clones) {
for (const cloneID of li.data.clones) {
blocks.push(this.workspace.getBlockById(cloneID))
}
}
blocks = blocks.concat(getCallsToEventsByName(li.data.eventName));
*/
// Now, fetch the events from the scratch runtime instead of blockly
let blocks = this.getCallsToEventsByName(item.data.eventName);
if (!instanceBlock) {
// Can we start by selecting the first block on 'this' sprite
const currentTargetID = this.utils.getEditingTarget().id;
for (const block of blocks) {
if (block.targetId === currentTargetID) {
instanceBlock = block;
break;
}
}
}
this.carousel.build(item, blocks, instanceBlock);
} else if (item.data.clones) {
let blocks = [this.workspace.getBlockById(item.data.labelID)];
for (const cloneID of item.data.clones) {
blocks.push(this.workspace.getBlockById(cloneID));
}
this.carousel.build(item, blocks, instanceBlock);
} else {
this.utils.scrollBlockIntoView(item.data.labelID);
this.carousel.remove();
}
}
getVariableUsesById(id) {
let uses = [];
let topBlocks = this.workspace.getTopBlocks();
for (const topBlock of topBlocks) {
/** @type {!Array<!Blockly.Block>} */
let kids = topBlock.getDescendants();
for (const block of kids) {
/** @type {!Array<!Blockly.VariableModel>} */
let blockVariables = block.getVarModels();
if (blockVariables) {
for (const blockVar of blockVariables) {
if (blockVar.getId() === id) {
uses.push(block);
}
}
}
}
}
return uses;
}
getCallsToProcedureById(id) {
let procBlock = this.workspace.getBlockById(id);
let label = procBlock.getChildren()[0];
let procCode = label.getProcCode();
let uses = [procBlock]; // Definition First, then calls to it
let topBlocks = this.workspace.getTopBlocks();
for (const topBlock of topBlocks) {
/** @type {!Array<!Blockly.Block>} */
let kids = topBlock.getDescendants();
for (const block of kids) {
if (block.type === "procedures_call") {
if (block.getProcCode() === procCode) {
uses.push(block);
}
}
}
}
return uses;
}
getCallsToEventsByName(name) {
let uses = []; // Definition First, then calls to it
const runtime = addon.tab.traps.vm.runtime;
const targets = runtime.targets; // The sprites / stage
for (const target of targets) {
if (!target.isOriginal) {
continue; // Skip clones
}
const blocks = target.blocks;
if (!blocks._blocks) {
continue;
}
for (const id of Object.keys(blocks._blocks)) {
const block = blocks._blocks[id];
if (block.opcode === "event_whenbroadcastreceived" && block.fields.BROADCAST_OPTION.value === name) {
uses.push(new _blockly_BlockInstance_js__WEBPACK_IMPORTED_MODULE_1__["default"](target, block));
} else if (block.opcode === "event_broadcast" || block.opcode === "event_broadcastandwait") {
const broadcastInputBlockId = block.inputs.BROADCAST_INPUT.block;
const broadcastInputBlock = blocks._blocks[broadcastInputBlockId];
if (broadcastInputBlock) {
let eventName;
if (broadcastInputBlock.opcode === "event_broadcast_menu") {
eventName = broadcastInputBlock.fields.BROADCAST_OPTION.value;
} else {
eventName = msg("complex-broadcast");
}
if (eventName === name) {
uses.push(new _blockly_BlockInstance_js__WEBPACK_IMPORTED_MODULE_1__["default"](target, block));
}
}
}
}
}
return uses;
}
empty() {
for (const item of this.items) {
if (this.el.contains(item)) {
this.el.removeChild(item);
}
}
this.items = [];
this.selected = null;
}
}
class Carousel {
constructor(utils) {
this.utils = utils;
this.el = null;
this.count = null;
this.blocks = [];
this.idx = 0;
}
build(item, blocks, instanceBlock) {
if (this.el && this.el.parentNode === item) {
// Same control... click again to go to next
this.navRight();
} else {
this.remove();
this.blocks = blocks;
item.appendChild(this.createDom());
this.idx = 0;
if (instanceBlock) {
for (const idx of Object.keys(this.blocks)) {
const block = this.blocks[idx];
if (block.id === instanceBlock.id) {
this.idx = Number(idx);
break;
}
}
}
if (this.idx < this.blocks.length) {
this.utils.scrollBlockIntoView(this.blocks[this.idx]);
}
}
}
createDom() {
this.el = document.createElement("span");
this.el.className = "sa-find-carousel";
const leftControl = this.el.appendChild(document.createElement("span"));
leftControl.className = "sa-find-carousel-control";
leftControl.textContent = "◀";
leftControl.addEventListener("mousedown", e => this.navLeft(e));
this.count = this.el.appendChild(document.createElement("span"));
this.count.innerText = this.blocks.length > 0 ? this.idx + 1 + " / " + this.blocks.length : "0";
const rightControl = this.el.appendChild(document.createElement("span"));
rightControl.className = "sa-find-carousel-control";
rightControl.textContent = "▶";
rightControl.addEventListener("mousedown", e => this.navRight(e));
return this.el;
}
inputKeyDown(e) {
// Left Arrow
if (e.keyCode === 37) {
if (this.el && this.blocks) {
this.navLeft(e);
}
} // Right Arrow
if (e.keyCode === 39) {
if (this.el && this.blocks) {
this.navRight(e);
}
}
}
navLeft(e) {
return this.navSideways(e, -1);
}
navRight(e) {
return this.navSideways(e, 1);
}
navSideways(e, dir) {
if (this.blocks.length > 0) {
this.idx = (this.idx + dir + this.blocks.length) % this.blocks.length; // + length to fix negative modulo js issue.
this.count.innerText = this.idx + 1 + " / " + this.blocks.length;
this.utils.scrollBlockIntoView(this.blocks[this.idx]);
}
if (e) {
e.cancelBubble = true;
e.preventDefault();
}
}
remove() {
if (this.el) {
this.el.remove();
this.blocks = [];
this.idx = 0;
}
}
}
const findBar = new FindBar();
const _doBlockClick_ = Blockly.Gesture.prototype.doBlockClick_;
Blockly.Gesture.prototype.doBlockClick_ = function () {
if (!addon.self.disabled && (this.mostRecentEvent_.button === 1 || this.mostRecentEvent_.shiftKey)) {
// Wheel button...
// Intercept clicks to allow jump to...?
let block = this.startBlock_;
for (; block; block = block.getSurroundParent()) {
if (block.type === "procedures_definition" || !this.jumpToDef && block.type === "procedures_call") {
let id = block.id ? block.id : block.getId ? block.getId() : null;
findBar.findInput.focus();
findBar.showDropDown(id);
return;
}
if (block.type === "data_variable" || block.type === "data_changevariableby" || block.type === "data_setvariableto") {
let id = block.getVars()[0];
findBar.findInput.focus();
findBar.showDropDown(id, block);
findBar.selVarID = id;
return;
}
if (block.type === "event_whenbroadcastreceived" || block.type === "event_broadcastandwait" || block.type === "event_broadcast") {
// todo: actually index the broadcasts...!
let id = block.id;
findBar.findInput.focus();
findBar.showDropDown(id, block);
findBar.selVarID = id;
return;
}
}
}
_doBlockClick_.call(this);
};
addon.tab.redux.initialize();
addon.tab.redux.addEventListener("statechanged", e => {
if (e.detail.action.type === "scratch-gui/navigation/ACTIVATE_TAB") {
findBar.tabChanged();
}
});
while (true) {
const root = await addon.tab.waitForElement("ul[class*=gui_tab-list_]", {
markAsSeen: true,
reduxEvents: ["scratch-gui/mode/SET_PLAYER", "fontsLoaded/SET_FONTS_LOADED", "scratch-gui/locales/SELECT_LOCALE"],
reduxCondition: state => !state.scratchGui.mode.isPlayerOnly
});
findBar.createDom(root);
}
});
/***/ }),
/***/ "./src/addons/addons/folders/_runtime_entry.js":
/*!*****************************************************!*\
!*** ./src/addons/addons/folders/_runtime_entry.js ***!
\*****************************************************/
/*! exports provided: resources */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "resources", function() { return resources; });
/* harmony import */ var _userscript_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./userscript.js */ "./src/addons/addons/folders/userscript.js");
/* harmony import */ var _css_loader_style_css__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! css-loader!./style.css */ "./node_modules/css-loader/index.js!./src/addons/addons/folders/style.css");
/* harmony import */ var _css_loader_style_css__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_css_loader_style_css__WEBPACK_IMPORTED_MODULE_1__);
/* harmony import */ var _url_loader_folder_svg__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! url-loader!./folder.svg */ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/folders/folder.svg");
/* generated by pull.js */
const resources = {
"userscript.js": _userscript_js__WEBPACK_IMPORTED_MODULE_0__["default"],
"style.css": _css_loader_style_css__WEBPACK_IMPORTED_MODULE_1___default.a,
"folder.svg": _url_loader_folder_svg__WEBPACK_IMPORTED_MODULE_2__["default"]
};
/***/ }),
/***/ "./src/addons/addons/folders/userscript.js":
/*!*************************************************!*\
!*** ./src/addons/addons/folders/userscript.js ***!
\*************************************************/
/*! exports provided: default */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); if (enumerableOnly) { symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; }); } keys.push.apply(keys, symbols); } return keys; }
function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i] != null ? arguments[i] : {}; if (i % 2) { ownKeys(Object(source), true).forEach(function (key) { _defineProperty(target, key, source[key]); }); } else if (Object.getOwnPropertyDescriptors) { Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)); } else { ownKeys(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } } return target; }
function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
/* harmony default export */ __webpack_exports__["default"] = (async function ({
addon,
console,
msg
}) {
// The basic premise of how this addon works is relative simple.
// scratch-gui renders the sprite selectors and asset selectors in a hierarchy like this:
// <SelectorHOC>
// <SpriteSelectorItem />
// <SpriteSelectorItem />
// <SpriteSelectorItem />
// <SpriteSelectorItem />
// ...
// </SelectorHOC>
// It's obviously more complicated than that, but there are two important parts:
// SelectorHOC - We override this to change which items are displayed
// SpriteSelectorItem - We override this to change how items are displayed.
// Folders are just items rendered differently
// These two components communicate through the `name` property of the items.
// We touch some things on the VM to make dragging items work properly.
const REACT_INTERNAL_PREFIX = "__reactInternalInstance$";
const TYPE_SPRITES = 1;
const TYPE_ASSETS = 2; // We run too early, will be set later
let vm;
let reactInternalKey;
let currentSpriteFolder;
let currentAssetFolder;
let currentSpriteItems;
let currentAssetItems;
const DIVIDER = "//";
/**
* getFolderFromName("B") === null
* getFolderFromName("A//b") === "A"
*/
const getFolderFromName = name => {
const idx = name.indexOf(DIVIDER);
if (idx === -1 || idx === 0) {
return null;
}
return name.substr(0, idx);
};
/**
* getNameWithoutFolder("B") === "B"
* getNameWithoutFolder("A//b") === "b"
*/
const getNameWithoutFolder = name => {
const idx = name.indexOf(DIVIDER);
if (idx === -1 || idx === 0) {
return name;
}
return name.substr(idx + DIVIDER.length);
};
/**
* setFolderOfName("B", "y") === "y//B"
* setFolderOfName("c//B", "y") === "y//B"
* setFolderOfName("B", null) === "B"
* setFolderOfName("c//B", null) === "B"
*/
const setFolderOfName = (name, folder) => {
const basename = getNameWithoutFolder(name);
if (folder) {
return "".concat(folder).concat(DIVIDER).concat(basename);
}
return basename;
};
const isValidFolderName = name => {
return !name.includes(DIVIDER) && !name.endsWith("/");
};
const RESERVED_NAMES = ["_mouse_", "_stage_", "_edge_", "_myself_", "_random_"];
const ensureNotReserved = name => {
if (name === "") return "2";
if (RESERVED_NAMES.includes(name)) return "".concat(name, "2");
return name;
};
const getSortableHOCFromElement = el => {
const nearestSpriteSelector = el.closest("[class*='sprite-selector_sprite-selector']");
if (nearestSpriteSelector) {
return nearestSpriteSelector[reactInternalKey].child.sibling.child.stateNode;
}
const nearestAssetPanelWrapper = el.closest('[class*="asset-panel_wrapper"]');
if (nearestAssetPanelWrapper) {
return nearestAssetPanelWrapper[reactInternalKey].child.child.stateNode;
}
throw new Error("cannot find SortableHOC");
};
const getBackpackFromElement = el => {
const gui = el.closest('[class*="gui_editor-wrapper"]');
if (!gui) throw new Error("cannot find Backpack");
return gui[reactInternalKey].child.sibling.child.child.stateNode;
};
const clamp = (n, min, max) => {
return Math.min(Math.max(n, min), max);
};
/**
* @typedef {Object} ItemData
* @property {string} realName
* @property {number} realIndex
* @property {string} inFolder
* @property {string} folder
* @property {boolean} folderOpen
*/
/**
* @returns {ItemData|null}
*/
const getItemData = item => {
if (item && item.name && typeof item.name === "object") {
return item.name;
}
return null;
};
const openFolderAsset = {
assetId: "&__sa_folders_folder",
encodeDataURI() {
// Doesn't actually need to be a data: URI
return addon.self.getResource("/folder.svg")
/* rewritten by pull.js */
;
}
}; // https://github.com/LLK/scratch-gui/blob/develop/src/components/asset-panel/icon--sound.svg
const imageIconSource = "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n<svg width=\"100px\" height=\"100px\" viewBox=\"0 0 20 20\" version=\"1.1\" xmlns=\"http://www.w3.org/2000/svg\" xmlns:xlink=\"http://www.w3.org/1999/xlink\">\n <g id=\"Sound\" stroke=\"none\" stroke-width=\"1\" fill=\"none\" fill-rule=\"evenodd\">\n <path d=\"M12.4785058,12.6666667 C12.3144947,12.6666667 12.1458852,12.6272044 11.9926038,12.5440517 C11.537358,12.2960031 11.3856094,11.7562156 11.6553847,11.3376335 C12.1688774,10.5371131 12.1688774,9.54491867 11.6553847,8.74580756 C11.3856094,8.32581618 11.537358,7.78602861 11.9926038,7.53798001 C12.452448,7.29275014 13.0379829,7.43086811 13.3046926,7.84804076 C14.1737981,9.20103311 14.1737981,10.8809986 13.3046926,12.233991 C13.1268862,12.5130457 12.806528,12.6666667 12.4785058,12.6666667 Z M15.3806784,13.8333333 C15.2408902,13.8333333 15.0958763,13.796281 14.9665396,13.7182064 C14.5785295,13.485306 14.4491928,12.9784829 14.6791247,12.5854634 C15.5949331,11.0160321 15.5949331,9.065491 14.6791247,7.49738299 C14.4491928,7.10436352 14.5785295,6.59621712 14.9665396,6.36331669 C15.3558562,6.13438616 15.8549129,6.26274605 16.0848448,6.65444223 C17.3050517,8.74260632 17.3050517,11.3389168 16.0848448,13.4270809 C15.9319924,13.6890939 15.6602547,13.8333333 15.3806784,13.8333333 Z M10.3043478,5.62501557 L10.3043478,13.873675 C10.3043478,14.850934 9.10969849,15.3625101 8.36478311,14.7038052 L6.7566013,13.2797607 C6.18712394,12.7762834 5.44499329,12.4968737 4.67362297,12.4968737 L4.3923652,12.4968737 C3.62377961,12.4968737 3,11.8935108 3,11.1470686 L3,8.36646989 C3,7.62137743 3.62377961,7.01666471 4.3923652,7.01666471 L4.65830695,7.01666471 C5.42967727,7.01666471 6.17180792,6.73725504 6.74128529,6.23377771 L8.36478311,4.79623519 C9.10969849,4.13753026 10.3043478,4.64910643 10.3043478,5.62501557 Z\" id=\"Combined-Shape\" fill=\"#575E75\"></path>\n </g>\n</svg>";
const soundIconHref = "data:image/svg+xml;base64,".concat(btoa(imageIconSource));
let folderColorStylesheet = null;
const folderColors = Object.create(null);
const getFolderColorClass = folderName => {
const mulberry32 = a => {
// https://stackoverflow.com/a/47593316
return function () {
var t = a += 0x6d2b79f5;
t = Math.imul(t ^ t >>> 15, t | 1);
t ^= t + Math.imul(t ^ t >>> 7, t | 61);
return ((t ^ t >>> 14) >>> 0) / 4294967296;
};
};
const hashCode = str => {
// Based on Java's String.hashCode
// https://hg.openjdk.java.net/jdk8/jdk8/jdk/file/687fd7c7986d/src/share/classes/java/lang/String.java#l1452
let hash = 0;
for (let i = 0; i < str.length; i++) {
hash = 31 * hash + str.charCodeAt(i);
hash = hash | 0;
}
return hash;
};
const random = str => {
const seed = hashCode(str);
const rng = mulberry32(seed); // Run RNG a few times to get more random numbers, otherwise similar seeds tend to give somewhat similar results
rng();
rng();
rng();
rng();
return rng();
};
if (!folderColors[folderName]) {
if (!folderColorStylesheet) {
folderColorStylesheet = document.createElement("style");
document.head.appendChild(folderColorStylesheet);
}
const hue = random(folderName) * 360;
const color = "hsla(".concat(hue, "deg, 100%, 85%, 0.5)");
const id = Object.keys(folderColors).length;
const className = "sa-folders-color-".concat(id);
folderColors[folderName] = className;
folderColorStylesheet.textContent += ".".concat(className, "{background-color:").concat(color, " !important;}");
folderColorStylesheet.textContent += ".".concat(className, "[class*=\"sprite-selector_raised\"]:not([class*=\"sa-folders-folder\"]){background-color:hsla(").concat(hue, "deg, 100%, 77%, 1) !important;}");
}
return folderColors[folderName];
};
const fixOrderOfItemsInFolders = items => {
const folders = Object.create(null);
const result = [];
for (const item of items) {
const name = item.getName ? item.getName() : item.name;
const folder = getFolderFromName(name);
if (typeof folder === "string") {
if (!folders[folder]) {
folders[folder] = [];
result.push(folders[folder]);
}
folders[folder].push(item);
} else {
result.push(item);
}
}
const flatResult = result.flat();
for (let i = 0; i < items.length; i++) {
if (result[i] !== items[i]) {
return {
items: flatResult,
changed: true
};
}
}
return {
items: flatResult,
changed: false
};
};
const fixTargetOrder = () => {
const {
items,
changed
} = fixOrderOfItemsInFolders(vm.runtime.targets);
if (changed) {
vm.runtime.targets = items;
vm.emitTargetsUpdate();
}
};
const fixCostumeOrder = (target = vm.editingTarget) => {
const {
items,
changed
} = fixOrderOfItemsInFolders(target.sprite.costumes);
if (changed) {
target.sprite.costumes = items;
vm.emitTargetsUpdate();
}
};
const fixSoundOrder = (target = vm.editingTarget) => {
const {
items,
changed
} = fixOrderOfItemsInFolders(target.sprite.sounds);
if (changed) {
target.sprite.sounds = items;
vm.emitTargetsUpdate();
}
};
const verifySortableHOC = sortableHOCInstance => {
const SortableHOC = sortableHOCInstance.constructor;
if (Array.isArray(sortableHOCInstance.props.items) && (typeof sortableHOCInstance.props.selectedId === "string" || typeof sortableHOCInstance.props.selectedItemIndex === "number") && typeof sortableHOCInstance.containerBox !== "undefined" && typeof SortableHOC.prototype.componentDidMount === "undefined" && typeof SortableHOC.prototype.componentDidUpdate === "undefined" && typeof SortableHOC.prototype.handleAddSortable === "function" && typeof SortableHOC.prototype.handleRemoveSortable === "function" && typeof SortableHOC.prototype.setRef === "function") return;
throw new Error("Can not comprehend SortableHOC");
};
const verifySpriteSelectorItem = spriteSelectorItemInstance => {
const SpriteSelectorItem = spriteSelectorItemInstance.constructor;
if (typeof spriteSelectorItemInstance.props.asset === "object" && typeof spriteSelectorItemInstance.props.name === "string" && typeof spriteSelectorItemInstance.props.dragType === "string" && typeof SpriteSelectorItem.prototype.handleClick === "function" && typeof SpriteSelectorItem.prototype.setRef === "function" && typeof SpriteSelectorItem.prototype.handleDrag === "function" && typeof SpriteSelectorItem.prototype.handleDragEnd === "function" && typeof SpriteSelectorItem.prototype.handleDelete === "function" && typeof SpriteSelectorItem.prototype.handleDuplicate === "function" && typeof SpriteSelectorItem.prototype.handleExport === "function") return;
throw new Error("Can not comprehend SpriteSelectorItem");
};
const verifyVM = vm => {
const target = vm.runtime.targets[0];
if (typeof vm.installTargets === "function" && typeof vm.reorderTarget === "function" && typeof target.reorderCostume === "function" && typeof target.reorderSound === "function" && typeof target.addCostume === "function" && typeof target.addSound === "function") return;
throw new Error("Can not comprehend VM");
};
const verifyBackpack = backpackInstance => {
const Backpack = backpackInstance.constructor;
if (typeof Backpack.prototype.handleDrop === "function" && typeof Backpack.prototype.componentDidUpdate === "undefined") {
return;
}
throw new Error("Can not comprehend Backpack");
};
class Cache {
constructor() {
this.cache = new Map();
this.usedThisTick = new Set();
}
has(id) {
return this.cache.has(id);
}
get(id) {
this.usedThisTick.add(id);
return this.cache.get(id);
}
set(id, value) {
this.usedThisTick.add(id);
this.cache.set(id, value);
}
startTick() {
this.usedThisTick.clear();
}
endTick() {
for (const id of Array.from(this.cache.keys())) {
if (!this.usedThisTick.has(id)) {
this.cache.delete(id);
}
}
}
clear() {
this.usedThisTick.clear();
this.cache.clear();
}
}
const patchSortableHOC = (SortableHOC, type) => {
// SortableHOC should be: https://github.com/LLK/scratch-gui/blob/29d9851778febe4e69fa5111bf7559160611e366/src/lib/sortable-hoc.jsx#L8
const itemCache = new Cache();
const folderItemCache = new Cache();
const folderAssetCache = new Cache();
const PREVIEW_SIZE = 80;
const PREVIEW_POSITIONS = [// x, y
[0, 0], [PREVIEW_SIZE / 2, 0], [0, PREVIEW_SIZE / 2], [PREVIEW_SIZE / 2, PREVIEW_SIZE / 2]];
const createFolderPreview = items => {
// Directly generate a string instead of using DOM API for performance as we deal with very large inlined images
// Because the result is only used as an img src, XSS shouldn't be a concern
let result = "data:image/svg+xml;,<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"".concat(PREVIEW_SIZE, "\" height=\"").concat(PREVIEW_SIZE, "\">");
for (let i = 0; i < Math.min(PREVIEW_POSITIONS.length, items.length); i++) {
const item = items[i];
const width = PREVIEW_SIZE / 2;
const height = PREVIEW_SIZE / 2;
const [x, y] = PREVIEW_POSITIONS[i];
let src;
if (item.asset) {
// TW: We can be 100% certain that escaping here is unnecessary
src = item.asset.encodeDataURI();
} else if (item.costume && item.costume.asset) {
src = item.costume.asset.encodeDataURI();
} else if (item.url) {
src = soundIconHref;
}
if (src) {
result += "<image width=\"".concat(width, "\" height=\"").concat(height, "\" x=\"").concat(x, "\" y=\"").concat(y, "\" href=\"").concat(src, "\"/>");
}
}
result += "</svg>";
return result;
};
const getUniqueIdOfFolderItems = items => {
let id = "sa_folder&&";
for (let i = 0; i < Math.min(PREVIEW_POSITIONS.length, items.length); i++) {
const item = items[i];
if (item.asset) {
id += item.asset.assetId;
} else if (item.costume && item.costume.asset) {
id += item.costume.asset.assetId;
} else if (item.url) {
id += item.url;
}
id += "&&";
}
return id;
};
const processItems = (openFolders, props) => {
const processItem = item => {
const itemId = item.name;
let newItem;
let itemData;
if (itemCache.has(itemId)) {
newItem = itemCache.get(itemId);
itemData = newItem.name;
} else {
itemData = {
toString() {
return "_".concat(item.name);
}
};
newItem = {};
itemCache.set(itemId, newItem);
}
const itemFolderName = getFolderFromName(item.name);
Object.assign(newItem, item);
itemData.realName = item.name;
itemData.realIndex = i;
itemData.inFolder = itemFolderName;
newItem.name = itemData;
return {
newItem,
itemData
};
};
itemCache.startTick();
folderItemCache.startTick();
folderAssetCache.startTick();
const folderOccurrences = new Map();
const items = [];
const result = {
items
};
let i = 0;
while (i < props.items.length) {
const item = props.items[i];
const folderName = getFolderFromName(item.name);
if (folderName === null) {
items.push(processItem(item).newItem);
if (type === TYPE_ASSETS) {
const isSelected = props.selectedItemIndex === i;
if (isSelected) {
result.selectedItemIndex = items.length - 1;
}
}
} else {
const isOpen = openFolders.indexOf(folderName) !== -1;
const folderItems = [];
while (i < props.items.length) {
const childItem = props.items[i];
const processedItem = processItem(childItem);
if (getFolderFromName(childItem.name) !== folderName) {
break;
}
folderItems.push(processedItem.newItem);
if (type === TYPE_ASSETS) {
const isSelected = props.selectedItemIndex === i;
if (isSelected) {
if (isOpen) {
result.selectedItemIndex = items.length + folderItems.length;
} else {
result.selectedItemIndex = -1;
}
}
}
i++;
}
i--;
const occurrence = folderOccurrences.get(folderName) || 0;
folderOccurrences.set(folderName, occurrence + 1);
const baseUniqueId = getUniqueIdOfFolderItems(folderItems);
const itemUniqueId = "".concat(isOpen, "&").concat(occurrence, "&").concat(folderName, "&").concat(baseUniqueId, "&");
const reactKey = "&__".concat(occurrence, "_").concat(folderName);
const assetUniqueId = baseUniqueId;
let folderItem;
let folderData;
if (folderItemCache.has(itemUniqueId)) {
folderItem = folderItemCache.get(itemUniqueId);
folderData = folderItem.name;
} else {
folderItem = {
// Can be used as a react key
id: {
toString() {
return reactKey;
}
}
};
folderData = {
// Can be used as a react key
toString() {
return reactKey;
}
};
folderItemCache.set(itemUniqueId, folderItem);
}
folderData.folder = folderName;
folderData.folderOpen = isOpen;
folderItem.items = folderItems;
folderItem.name = folderData;
let folderAsset;
if (isOpen) {
folderAsset = openFolderAsset;
} else {
if (folderAssetCache.has(assetUniqueId)) {
folderAsset = folderAssetCache.get(assetUniqueId);
} else {
folderAsset = {
assetId: assetUniqueId,
encodeDataURI() {
return createFolderPreview(folderItems);
}
};
folderAssetCache.set(assetUniqueId, folderAsset);
}
}
if (type === TYPE_SPRITES) {
if (!folderItem.costume) folderItem.costume = {};
folderItem.costume.asset = folderAsset; // For sprite items, `id` is used as the drag payload and toString is used as a React key
if (!folderItem.id) folderItem.id = {};
folderItem.id.sa_folder_items = folderItems;
folderItem.id.toString = () => reactKey;
} else {
folderItem.asset = folderAsset;
if (!folderItem.dragPayload) folderItem.dragPayload = {};
folderItem.dragPayload.sa_folder_items = folderItems;
}
items.push(folderItem);
if (isOpen) {
for (const item of folderItems) {
items.push(item);
}
}
}
i++;
}
itemCache.endTick();
folderItemCache.endTick();
folderAssetCache.endTick();
return result;
};
const getSelectedItem = sortable => {
if (type === TYPE_SPRITES) {
const selectedItem = sortable.props.items.find(i => i.id === sortable.props.selectedId);
return selectedItem;
} else if (type === TYPE_ASSETS) {
const selectedItem = sortable.props.items[sortable.props.selectedItemIndex];
return selectedItem;
}
return null;
};
SortableHOC.prototype.saInitialSetup = function () {
itemCache.clear();
folderItemCache.clear();
folderAssetCache.clear();
const folders = [];
const selectedItem = getSelectedItem(this);
if (selectedItem && !selectedItem.isStage) {
const folder = getFolderFromName(selectedItem.name);
folders.push(folder);
if (type === TYPE_SPRITES) {
currentSpriteFolder = folder;
} else if (type === TYPE_ASSETS) {
currentAssetFolder = folder;
}
}
this.setState({
folders
});
};
SortableHOC.prototype.componentDidMount = function () {
// Do part of componentDidUpdate on mount as well
const selectedItem = getSelectedItem(this);
if (selectedItem) {
const folder = getFolderFromName(selectedItem.name);
if (type === TYPE_SPRITES) {
currentSpriteFolder = folder;
} else if (type === TYPE_ASSETS) {
currentAssetFolder = folder;
}
}
this.saInitialSetup();
};
SortableHOC.prototype.componentDidUpdate = function (prevProps, prevState) {
const selectedItem = getSelectedItem(this);
if (selectedItem) {
const folder = getFolderFromName(selectedItem.name);
const currentFolder = this.state.folders.includes(folder) ? folder : null;
if (type === TYPE_SPRITES) {
currentSpriteFolder = currentFolder;
} else if (type === TYPE_ASSETS) {
currentAssetFolder = currentFolder;
}
let selectedItemChanged;
if (this.props.selectedId) {
selectedItemChanged = this.props.selectedId !== prevProps.selectedId;
} else {
selectedItemChanged = this.props.items[this.props.selectedItemIndex] && prevProps.items[prevProps.selectedItemIndex] && this.props.items[this.props.selectedItemIndex].name !== prevProps.items[prevProps.selectedItemIndex].name;
}
if (selectedItemChanged) {
if (!selectedItem.isStage) {
if (typeof folder === "string" && !this.state.folders.includes(folder)) {
this.setState(prevState => ({
folders: [...prevState.folders, folder]
}));
}
}
}
}
};
const originalSortableHOCRender = SortableHOC.prototype.render;
SortableHOC.prototype.render = function () {
const originalProps = this.props;
this.props = _objectSpread(_objectSpread({}, this.props), processItems(this.state && this.state.folders || [], this.props));
if (type === TYPE_SPRITES) {
currentSpriteItems = this.props.items;
} else if (type === TYPE_ASSETS) {
currentAssetItems = this.props.items;
}
const result = originalSortableHOCRender.call(this);
this.props = originalProps;
return result;
};
};
const getAllFolders = component => {
const result = new Set();
let items;
if (component.props.dragType === "SPRITE") {
items = currentSpriteItems;
} else {
items = currentAssetItems;
}
for (const item of items) {
const data = getItemData(item);
if (typeof data.folder === "string") {
result.add(data.folder);
}
}
return Array.from(result);
};
const isFolderOpen = (component, folder) => {
const sortableHOCInstance = getSortableHOCFromElement(component.ref);
const folders = sortableHOCInstance.state && sortableHOCInstance.state.folders || [];
return folders.includes(folder);
};
const setFolderOpen = (component, folder, open) => {
const sortableHOCInstance = getSortableHOCFromElement(component.ref);
sortableHOCInstance.setState(prevState => {
let folders = prevState && prevState.folders || [];
folders = folders.filter(i => i !== folder);
if (open) {
return {
folders: [...folders, folder]
};
}
return {
folders
};
});
};
addon.tab.createEditorContextMenu((ctxType, ctx) => {
if (ctxType !== "sprite" && ctxType !== "costume" && ctxType !== "sound") return;
const component = ctx.target[addon.tab.traps.getInternalKey(ctx.target)].return.return.return.stateNode;
const data = getItemData(component.props);
if (!data) return;
if (typeof data.folder === "string") {
ctx.target.setAttribute("sa-folders-context-type", "folder");
const renameItems = newName => {
const isOpen = isFolderOpen(component, data.folder);
setFolderOpen(component, data.folder, false);
if (isOpen && typeof newName === "string") {
setFolderOpen(component, newName, true);
}
if (component.props.dragType === "SPRITE") {
for (const target of vm.runtime.targets) {
if (target.isOriginal) {
if (getFolderFromName(target.getName()) === data.folder) {
vm.renameSprite(target.id, ensureNotReserved(setFolderOfName(target.getName(), newName)));
}
}
}
vm.emitWorkspaceUpdate();
fixTargetOrder();
} else if (component.props.dragType === "COSTUME") {
for (let i = 0; i < vm.editingTarget.sprite.costumes.length; i++) {
const costume = vm.editingTarget.sprite.costumes[i];
if (getFolderFromName(costume.name) === data.folder) {
vm.renameCostume(i, setFolderOfName(costume.name, newName));
}
}
fixCostumeOrder();
} else if (component.props.dragType === "SOUND") {
for (let i = 0; i < vm.editingTarget.sprite.sounds.length; i++) {
const sound = vm.editingTarget.sprite.sounds[i];
if (getFolderFromName(sound.name) === data.folder) {
vm.renameSound(i, setFolderOfName(sound.name, newName));
}
}
fixSoundOrder();
}
};
const renameFolder = async () => {
let newName = await addon.tab.prompt(msg("rename-folder-prompt-title"), msg("rename-folder-prompt"), data.folder, {
useEditorClasses: true
}); // Prompt cancelled, do not rename
if (newName === null) {
return;
}
if (!isValidFolderName(newName)) {
alert(msg("name-not-allowed"));
return;
} // Empty name will remove the folder
if (!newName) {
newName = null;
}
renameItems(newName);
};
const removeFolder = () => {
renameItems(null);
};
return [{
className: "sa-folders-rename-folder",
label: msg("rename-folder"),
callback: renameFolder,
position: "assetContextMenuAfterDelete",
order: 10
}, {
className: "sa-folders-remove-folder",
label: msg("remove-folder"),
callback: removeFolder,
position: "assetContextMenuAfterDelete",
order: 11
}];
} else {
ctx.target.setAttribute("sa-folders-context-type", "asset");
const setFolder = folder => {
if (component.props.dragType === "SPRITE") {
const target = vm.runtime.getTargetById(component.props.id);
vm.renameSprite(component.props.id, ensureNotReserved(setFolderOfName(target.getName(), folder)));
fixTargetOrder();
vm.emitWorkspaceUpdate();
} else if (component.props.dragType === "COSTUME") {
const data = getItemData(component.props);
const index = data.realIndex;
const asset = vm.editingTarget.sprite.costumes[index];
vm.renameCostume(vm.editingTarget.sprite.costumes.indexOf(asset), setFolderOfName(asset.name, folder));
fixCostumeOrder();
} else if (component.props.dragType === "SOUND") {
const data = getItemData(component.props);
const index = data.realIndex;
const asset = vm.editingTarget.sprite.sounds[index];
vm.renameSound(vm.editingTarget.sprite.sounds.indexOf(asset), setFolderOfName(asset.name, folder));
fixSoundOrder();
}
};
const createFolder = async () => {
const name = await addon.tab.prompt(msg("name-prompt-title"), msg("name-prompt"), getNameWithoutFolder(data.realName), {
useEditorClasses: true
});
if (name === null) {
return;
}
if (!isValidFolderName(name)) {
alert(msg("name-not-allowed"));
return;
}
setFolder(name);
};
const base = [{
border: true,
className: "sa-folders-create-folder",
label: msg("create-folder"),
callback: createFolder,
position: "assetContextMenuAfterDelete",
order: 13
}];
const currentFolder = data.inFolder;
if (typeof currentFolder === "string") {
base.push({
className: "sa-folders-remove-from-folder",
label: msg("remove-from-folder"),
callback: () => setFolder(null),
position: "assetContextMenuAfterDelete",
order: 14
});
}
return base.concat(getAllFolders(component).filter(folder => folder !== currentFolder).map((folder, i) => {
return {
className: "sa-folders-add-to-folder",
label: msg("add-to-folder", {
folder
}),
callback: () => setFolder(folder),
position: "assetContextMenuAfterDelete",
order: 20 + i
};
}));
}
});
const patchSpriteSelectorItem = SpriteSelectorItem => {
for (const method of ["handleDelete", "handleDuplicate", "handleExport"]) {
const original = SpriteSelectorItem.prototype[method];
SpriteSelectorItem.prototype[method] = function (...args) {
if (typeof this.props.id === "number") {
const itemData = getItemData(this.props);
if (itemData) {
const originalProps = this.props;
this.props = _objectSpread(_objectSpread({}, originalProps), {}, {
id: itemData.realIndex
});
const ret = original.call(this, ...args);
this.props = originalProps;
return ret;
}
}
return original.call(this, ...args);
};
}
const originalHandleDragEnd = SpriteSelectorItem.prototype.handleDragEnd;
SpriteSelectorItem.prototype.handleDragEnd = function (...args) {
const itemData = getItemData(this.props);
if (itemData) {
if (typeof itemData.realIndex === "number" && this.props.dragging) {
// If the item is being dragged onto another group (eg. costume list -> sprite list)
// then we fake a drag event to make the `index` be the real index
const originalIndex = this.props.index;
const realIndex = itemData.realIndex;
if (originalIndex !== realIndex) {
const currentOffset = addon.tab.redux.state.scratchGui.assetDrag.currentOffset;
const sortableHOCInstance = getSortableHOCFromElement(this.ref);
if (currentOffset && sortableHOCInstance && sortableHOCInstance.getMouseOverIndex() === null) {
this.props.index = realIndex;
this.handleDrag(currentOffset);
this.props.index = originalIndex;
}
}
}
}
return originalHandleDragEnd.call(this, ...args);
};
const originalHandleClick = SpriteSelectorItem.prototype.handleClick;
SpriteSelectorItem.prototype.handleClick = function (...args) {
const e = args[0];
if (e && !this.noClick) {
const itemData = getItemData(this.props);
if (itemData) {
if (typeof itemData.folder === "string") {
e.preventDefault();
setFolderOpen(this, itemData.folder, !isFolderOpen(this, itemData.folder));
return;
}
if (typeof this.props.number === "number" && typeof itemData.realIndex === "number") {
e.preventDefault();
if (this.props.onClick) {
this.props.onClick(itemData.realIndex);
}
return;
}
}
}
return originalHandleClick.call(this, ...args);
};
const originalRender = SpriteSelectorItem.prototype.render;
SpriteSelectorItem.prototype.render = function () {
const itemData = getItemData(this.props);
if (itemData) {
const originalProps = this.props;
this.props = _objectSpread({}, this.props);
if (typeof itemData.realName === "string") {
this.props.name = getNameWithoutFolder(itemData.realName);
}
if (typeof this.props.number === "number" && typeof itemData.realIndex === "number") {
// Convert 0-indexed to 1-indexed
this.props.number = itemData.realIndex + 1;
}
if (typeof itemData.folder === "string") {
this.props.name = itemData.folder;
if (itemData.folderOpen) {
this.props.details = msg("open-folder");
} else {
this.props.details = msg("closed-folder");
}
this.props.selected = false;
this.props.number = null;
this.props.className += " ".concat(getFolderColorClass(itemData.folder), " sa-folders-folder");
}
if (typeof itemData.inFolder === "string") {
this.props.className += " ".concat(getFolderColorClass(itemData.inFolder));
}
const result = originalRender.call(this);
this.props = originalProps;
return result;
}
return originalRender.call(this);
};
};
const patchVM = () => {
const RenderedTarget = vm.runtime.targets[0].constructor;
const originalInstallTargets = vm.installTargets;
vm.installTargets = function (...args) {
if (currentSpriteFolder !== null) {
const targets = args[0];
const wholeProject = args[2];
if (Array.isArray(targets) && !wholeProject) {
for (const target of targets) {
if (target.sprite) {
target.sprite.name = setFolderOfName(target.sprite.name, currentSpriteFolder);
}
}
}
}
return originalInstallTargets.call(this, ...args).then(r => {
fixTargetOrder();
return r;
});
};
const originalAddCostume = RenderedTarget.prototype.addCostume;
RenderedTarget.prototype.addCostume = function (...args) {
if (currentAssetFolder !== null) {
const costume = args[0];
if (costume && typeof getFolderFromName(costume.name) !== "string") {
costume.name = setFolderOfName(costume.name, currentAssetFolder);
}
}
const r = originalAddCostume.call(this, ...args);
fixCostumeOrder(this);
return r;
};
const originalAddSound = RenderedTarget.prototype.addSound;
RenderedTarget.prototype.addSound = function (...args) {
if (currentAssetFolder !== null) {
const sound = args[0];
if (sound && typeof getFolderFromName(sound.name) !== "string") {
sound.name = setFolderOfName(sound.name, currentAssetFolder);
}
}
const r = originalAddSound.call(this, ...args);
fixSoundOrder(this);
return r;
};
const abstractReorder = ({
guiItems,
getAll,
set,
rename,
getVMItemFromGUIItem,
zeroIndexed,
onFolderChanged
}, itemIndex, newIndex) => {
// First index depends on zeroIndexed
itemIndex = clamp(itemIndex, zeroIndexed ? 0 : 1, zeroIndexed ? guiItems.length - 1 : guiItems.length);
newIndex = clamp(newIndex, zeroIndexed ? 0 : 1, zeroIndexed ? guiItems.length - 1 : guiItems.length);
if (itemIndex === newIndex) {
return false;
}
let assets = getAll();
const originalAssets = getAll();
const targetItem = guiItems[itemIndex - (zeroIndexed ? 0 : 1)];
const itemAtNewIndex = guiItems[newIndex - (zeroIndexed ? 0 : 1)];
const targetItemData = getItemData(targetItem);
const itemAtNewIndexData = getItemData(itemAtNewIndex);
if (!targetItemData || !itemAtNewIndexData) {
console.warn("should never happen");
return false;
}
const reorderingItems = typeof targetItemData.folder === "string" ? targetItem.items : [targetItem];
const reorderingAssets = reorderingItems.map(i => getVMItemFromGUIItem(i, assets)).filter(i => i);
if (typeof itemAtNewIndexData.realIndex === "number") {
const newTarget = getVMItemFromGUIItem(itemAtNewIndex, assets);
if (!newTarget || reorderingAssets.includes(newTarget)) {
// Dragging folder into itself or target doesn't exist. Ignore.
return false;
}
}
let newFolder = null;
assets = assets.filter(i => !reorderingAssets.includes(i));
let realNewIndex;
if (newIndex === (zeroIndexed ? 0 : 1)) {
realNewIndex = zeroIndexed ? 0 : 1;
} else if (newIndex === guiItems.length - (zeroIndexed ? 1 : 0)) {
realNewIndex = assets.length;
} else if (typeof itemAtNewIndexData.realIndex === "number") {
newFolder = typeof itemAtNewIndexData.inFolder === "string" ? itemAtNewIndexData.inFolder : null;
let newAsset = getVMItemFromGUIItem(itemAtNewIndex, assets);
if (!newAsset) {
console.warn("should never happen");
return false;
}
realNewIndex = assets.indexOf(newAsset);
if (newIndex > itemIndex) {
realNewIndex++;
}
} else if (typeof itemAtNewIndexData.folder === "string") {
let item;
let offset = 0;
if (newIndex < itemIndex) {
// A B [C D E] F G
// ^----------*
// A B C [D] E F G
// ^--------*
item = itemAtNewIndex.items[0];
} else if (itemAtNewIndexData.folderOpen) {
// A B [C D E] F G
// *---^
item = itemAtNewIndex.items[0];
newFolder = itemAtNewIndexData.folder;
} else {
// A B [C] D E F G
// *----^
item = itemAtNewIndex.items[itemAtNewIndex.items.length - 1];
offset = 1;
}
let newAsset = getVMItemFromGUIItem(item, assets);
if (newAsset) {
realNewIndex = assets.indexOf(newAsset) + offset;
} else {
// Edge case: Dragging the first item of a list on top of the folder item
// A B [C D E] F G
// ^---*
newAsset = getVMItemFromGUIItem(item, originalAssets);
if (!newAsset) {
console.warn("should never happen");
return false;
}
realNewIndex = originalAssets.indexOf(newAsset) + offset;
}
} else {
console.warn("should never happen");
return false;
}
if (typeof targetItemData.folder === "string" && newFolder !== null) {
// Cannot drag a folder into another folder
return;
}
if (realNewIndex < (zeroIndexed ? 0 : 1) || realNewIndex > assets.length) {
console.warn("should never happen");
return false;
}
assets.splice(realNewIndex, 0, ...reorderingAssets);
set(assets); // If the folder has changed, update item names to match.
if (typeof targetItemData.folder !== "string" && targetItemData.inFolder !== newFolder) {
for (const asset of reorderingAssets) {
const name = asset.getName ? asset.getName() : asset.name;
rename(asset, setFolderOfName(name, newFolder));
}
if (onFolderChanged) {
onFolderChanged();
}
}
return true;
};
vm.constructor.prototype.reorderTarget = function (targetIndex, newIndex) {
return abstractReorder({
getAll: () => {
return this.runtime.targets;
},
set: targets => {
this.runtime.targets = targets;
this.emitTargetsUpdate();
},
rename: (item, name) => {
this.renameSprite(item.id, ensureNotReserved(name));
},
getVMItemFromGUIItem: (item, targets) => {
return targets.find(i => i.id === item.id);
},
onFolderChanged: () => {
this.emitWorkspaceUpdate();
},
guiItems: currentSpriteItems,
zeroIndexed: false
}, targetIndex, newIndex);
};
RenderedTarget.prototype.reorderCostume = function (costumeIndex, newIndex) {
return abstractReorder({
getAll: () => {
return this.sprite.costumes;
},
set: assets => {
this.sprite.costumes = assets;
},
rename: (item, name) => {
this.renameCostume(this.sprite.costumes.indexOf(item), name);
},
getVMItemFromGUIItem: (item, costumes) => {
const itemData = getItemData(item);
return costumes.find(c => c.name === itemData.realName);
},
guiItems: currentAssetItems,
zeroIndexed: true
}, costumeIndex, newIndex);
};
RenderedTarget.prototype.reorderSound = function (soundIndex, newIndex) {
return abstractReorder({
getAll: () => {
return this.sprite.sounds;
},
set: assets => {
this.sprite.sounds = assets;
},
rename: (item, name) => {
this.renameSound(this.sprite.sounds.indexOf(item), name);
},
getVMItemFromGUIItem: (item, sounds) => {
const itemData = getItemData(item);
return sounds.find(c => c.name === itemData.realName);
},
guiItems: currentAssetItems,
zeroIndexed: true
}, soundIndex, newIndex);
}; // Temporal bug fix for #5762
const originalShareSoundToTarget = vm.shareSoundToTarget;
vm.shareSoundToTarget = function (...args) {
const target = this.runtime.getTargetById(args[1]);
if (!target) {
// Avoid reading property from null
return Promise.reject(new Error("Dropping sound into folder is not supported")); // This would also work no matter what we returned, probably
// Original method returns a promise, so here too
}
return originalShareSoundToTarget.call(this, ...args);
};
};
const patchBackpack = backpackInstance => {
const Backpack = backpackInstance.constructor;
Backpack.prototype.sa_loadNextItem = function () {
if (!this.sa_queuedItems) return;
const item = this.sa_queuedItems.pop();
if (item) {
let payload;
let type;
if (item.dragPayload) {
if (item.url) {
type = "SOUND";
} else {
type = "COSTUME";
}
payload = item.dragPayload;
} else if (item.id) {
type = "SPRITE";
payload = item.id;
}
if (type && payload) {
originalHandleDrop.call(this, {
dragType: type,
payload: payload
});
}
}
};
Backpack.prototype.componentDidUpdate = function (prevProps, prevState) {
if (!this.state.loading && prevState.loading && !this.state.error) {
this.sa_loadNextItem();
}
};
const originalHandleDrop = Backpack.prototype.handleDrop;
Backpack.prototype.handleDrop = function (...args) {
// When a folder is dropped into the backpack, upload all the items in the folder.
const dragInfo = args[0];
const folderItems = dragInfo && dragInfo.payload && dragInfo.payload.sa_folder_items;
if (Array.isArray(folderItems)) {
addon.tab.confirm("", msg("confirm-backpack-folder"), {
useEditorClasses: true
}).then(result => {
if (!result) return;
this.sa_queuedItems = folderItems;
this.sa_loadNextItem();
});
return;
}
return originalHandleDrop.call(this, ...args);
};
backpackInstance.handleDrop = Backpack.prototype.handleDrop.bind(backpackInstance);
}; // Backpack
{
const clickListener = e => {
if (!e.target.closest('[class*="backpack_backpack-header_"]')) {
return;
}
setTimeout(() => {
const backpackContainer = document.querySelector("[class^='backpack_backpack-list_']");
if (!backpackContainer) {
return;
}
document.removeEventListener("click", clickListener);
const backpackInstance = getBackpackFromElement(backpackContainer);
verifyBackpack(backpackInstance);
patchBackpack(backpackInstance);
});
};
document.addEventListener("click", clickListener, true);
} // Sprite list
{
const spriteSelectorItemElement = await addon.tab.waitForElement("[class^='sprite-selector_sprite-wrapper']", {
reduxCondition: state => !state.scratchGui.mode.isPlayerOnly
});
vm = addon.tab.traps.vm;
reactInternalKey = Object.keys(spriteSelectorItemElement).find(i => i.startsWith(REACT_INTERNAL_PREFIX));
const sortableHOCInstance = getSortableHOCFromElement(spriteSelectorItemElement);
const spriteSelectorItemInstance = spriteSelectorItemElement[reactInternalKey].child.child.child.stateNode;
verifySortableHOC(sortableHOCInstance);
verifySpriteSelectorItem(spriteSelectorItemInstance);
verifyVM(vm);
patchSortableHOC(sortableHOCInstance.constructor, TYPE_SPRITES);
patchSpriteSelectorItem(spriteSelectorItemInstance.constructor);
sortableHOCInstance.saInitialSetup();
patchVM();
} // Costume and sound list
{
const selectorListItem = await addon.tab.waitForElement("[class*='selector_list-item']", {
reduxCondition: state => state.scratchGui.editorTab.activeTabIndex !== 0 && !state.scratchGui.mode.isPlayerOnly
});
const sortableHOCInstance = getSortableHOCFromElement(selectorListItem);
verifySortableHOC(sortableHOCInstance);
patchSortableHOC(sortableHOCInstance.constructor, TYPE_ASSETS);
sortableHOCInstance.saInitialSetup();
}
});
/***/ }),
/***/ "./src/addons/addons/middle-click-popup/_runtime_entry.js":
/*!****************************************************************!*\
!*** ./src/addons/addons/middle-click-popup/_runtime_entry.js ***!
\****************************************************************/
/*! exports provided: resources */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "resources", function() { return resources; });
/* harmony import */ var _userscript_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./userscript.js */ "./src/addons/addons/middle-click-popup/userscript.js");
/* harmony import */ var _css_loader_userstyle_css__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! css-loader!./userstyle.css */ "./node_modules/css-loader/index.js!./src/addons/addons/middle-click-popup/userstyle.css");
/* harmony import */ var _css_loader_userstyle_css__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_css_loader_userstyle_css__WEBPACK_IMPORTED_MODULE_1__);
/* generated by pull.js */
const resources = {
"userscript.js": _userscript_js__WEBPACK_IMPORTED_MODULE_0__["default"],
"userstyle.css": _css_loader_userstyle_css__WEBPACK_IMPORTED_MODULE_1___default.a
};
/***/ }),
/***/ "./src/addons/addons/middle-click-popup/userscript.js":
/*!************************************************************!*\
!*** ./src/addons/addons/middle-click-popup/userscript.js ***!
\************************************************************/
/*! exports provided: default */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony default export */ __webpack_exports__["default"] = (async function ({
addon,
msg,
console
}) {
const Blockly = await addon.tab.traps.getBlockly();
let mouse = {
x: 0,
y: 0
};
class FloatingInput {
constructor() {
this.floatBar = null;
this.floatInput = null;
this.dropdownOut = null;
this.dropdown = null;
this.prevVal = "";
this.DROPDOWN_BLOCK_LIST_MAX_ROWS = 25;
this.createDom();
}
get workspace() {
return Blockly.getMainWorkspace();
}
get selectedTab() {
return addon.tab.redux.state.scratchGui.editorTab.activeTabIndex;
}
createDom() {
// Popup new input box for block injection
this.floatBar = document.body.appendChild(document.createElement("div"));
this.floatBar.className = "sa-float-bar";
this.floatBar.dir = addon.tab.direction;
this.floatBar.style.display = "none";
this.dropdownOut = this.floatBar.appendChild(document.createElement("div"));
this.dropdownOut.className = "sa-float-bar-dropdown-out";
this.floatInput = this.dropdownOut.appendChild(document.createElement("input"));
this.floatInput.placeholder = msg("start-typing");
this.floatInput.className = "sa-float-bar-input";
this.floatInput.className = addon.tab.scratchClass("input_input-form", {
others: "sa-float-bar-input"
});
this.dropdown = this.dropdownOut.appendChild(document.createElement("ul"));
this.dropdown.className = "sa-float-bar-dropdown";
this.floatInput.addEventListener("keyup", () => this.inputChange());
this.floatInput.addEventListener("focus", () => this.inputChange());
this.floatInput.addEventListener("keydown", (...e) => this.inputKeyDown(...e));
this.floatInput.addEventListener("focusout", () => this.hide());
this.dropdownOut.addEventListener("mousedown", (...e) => this.onClick(...e));
document.addEventListener("keydown", e => {
if (addon.tab.editorMode !== "editor") {
return;
}
let ctrlKey = e.ctrlKey || e.metaKey;
if (e.key === " " && ctrlKey) {
// Ctrl + Space (Inject Code)
this.show(e);
e.cancelBubble = true;
e.preventDefault();
return true;
}
});
}
show(e) {
var _e$clientX, _e$clientY;
if (this.selectedTab !== 0) {
return;
}
e.cancelBubble = true;
e.preventDefault();
this.buildFilterList();
this.floatBar.style.left = ((_e$clientX = e.clientX) !== null && _e$clientX !== void 0 ? _e$clientX : mouse.x) + 16 + "px";
this.floatBar.style.top = ((_e$clientY = e.clientY) !== null && _e$clientY !== void 0 ? _e$clientY : mouse.y) - 8 + "px";
this.floatBar.style.display = "";
this.floatInput.value = "";
this.floatInput.focus();
}
onClick(e) {
e.cancelBubble = true;
if (!e.target.closest("input")) {
e.preventDefault();
}
let sel = e && e.target;
if (sel.tagName === "B") {
sel = sel.parentNode;
}
if (e instanceof MouseEvent && sel.tagName !== "LI") {
// Mouse clicks need to be on a block...
return;
}
if (!sel || !sel.data) {
sel = this.dropdown.querySelector(".sel");
}
if (!sel) {
return;
}
this.createDraggingBlock(sel, e);
if (e.shiftKey) {
this.floatBar.style.display = "";
this.floatInput.focus();
}
}
createDraggingBlock(sel, e) {
let option = sel.data.option; // block:option.block, dom:option.dom, option:option.option
if (option.option) {
// We need to tweak the dropdown in this xml...
let field = option.dom.querySelector("field[name=" + option.pickField + "]");
if (field.getAttribute("id")) {
field.innerText = option.option[0];
field.setAttribute("id", option.option[1] + "-" + option.option[0]);
} else {
field.innerText = option.option[1];
} // Handle "stop other scripts in sprite"
if (option.option[1] === "other scripts in sprite") {
option.dom.querySelector("mutation").setAttribute("hasnext", "true");
}
} // This is mostly copied from https://github.com/LLK/scratch-blocks/blob/893c7e7ad5bfb416eaed75d9a1c93bdce84e36ab/core/scratch_blocks_utils.js#L171
// Some bits were removed or changed to fit our needs.
this.workspace.setResizesEnabled(false);
Blockly.Events.disable();
try {
var newBlock = Blockly.Xml.domToBlock(option.dom, this.workspace);
Blockly.scratchBlocksUtils.changeObscuredShadowIds(newBlock);
var svgRootNew = newBlock.getSvgRoot();
if (!svgRootNew) {
throw new Error("newBlock is not rendered.");
}
let blockBounds = newBlock.svgPath_.getBoundingClientRect();
let newBlockX = Math.floor((mouse.x - (blockBounds.left + blockBounds.right) / 2) / this.workspace.scale);
let newBlockY = Math.floor((mouse.y - (blockBounds.top + blockBounds.bottom) / 2) / this.workspace.scale);
newBlock.moveBy(newBlockX, newBlockY);
} finally {
Blockly.Events.enable();
}
if (Blockly.Events.isEnabled()) {
Blockly.Events.fire(new Blockly.Events.BlockCreate(newBlock));
}
var fakeEvent = {
clientX: mouse.x,
clientY: mouse.y,
type: "mousedown",
preventDefault: function preventDefault() {
e.preventDefault();
},
stopPropagation: function stopPropagation() {
e.stopPropagation();
},
target: sel
};
this.workspace.startDragWithFakeEvent(fakeEvent, newBlock);
}
inputChange() {
// Filter the list...
let val = (this.floatInput.value || "").toLowerCase();
if (val === this.prevVal) {
return;
}
this.prevVal = val;
let p = this.dropdown.parentNode;
this.dropdown.remove();
let count = 0;
let split = val.split(" ");
let listLI = this.dropdown.getElementsByTagName("li");
for (const li of listLI) {
const procCode = li.data.text;
const lower = li.data.lower; // let i = li.data.lower.indexOf(val);
// let array = regExp.exec(li.data.lower);
let im = 0;
let match = [];
for (let si = 0; si < split.length; si++) {
let find = " " + split[si];
let idx = lower.indexOf(find, im);
if (idx === -1) {
match = null;
break;
}
match.push(idx);
im = idx + find.length;
}
if (count < this.DROPDOWN_BLOCK_LIST_MAX_ROWS && match) {
li.style.display = "block";
while (li.firstChild) {
li.removeChild(li.firstChild);
}
let i = 0;
for (let iM = 0; iM < match.length; iM++) {
let im = match[iM];
if (im > i) {
li.appendChild(document.createTextNode(procCode.substring(i, im)));
i = im;
}
let bText = document.createElement("b");
let len = split[iM].length;
bText.appendChild(document.createTextNode(procCode.substr(i, len)));
li.appendChild(bText);
i += len;
}
if (i < procCode.length) {
li.appendChild(document.createTextNode(procCode.substr(i)));
}
if (count === 0) {
li.classList.add("sel");
} else {
li.classList.remove("sel");
}
count++;
} else {
li.style.display = "none";
li.classList.remove("sel");
}
}
p.append(this.dropdown);
}
inputKeyDown(e) {
if (e.keyCode === 38) {
this.navigateFloatFilter(-1);
e.preventDefault();
return;
}
if (e.keyCode === 40) {
this.navigateFloatFilter(1);
e.preventDefault();
return;
}
if (e.keyCode === 13) {
// Enter
let sel = this.dropdown.querySelector(".sel");
if (sel) {
this.onClick(e);
this.hide();
}
e.cancelBubble = true;
e.preventDefault();
return;
}
if (e.keyCode === 27) {
// Escape
if (this.floatInput.value.length > 0) {
this.floatInput.value = ""; // Clear search first, then close on second press
this.inputChange(e);
} else {
this.hide();
}
e.preventDefault();
return;
}
}
buildFilterList() {
let options = [];
let toolbox = this.workspace.getToolbox(); // This can happen during custom block creation, for example
if (!toolbox) return;
let blocks = toolbox.flyout_.getWorkspace().getTopBlocks(); // 107 blocks, not in order... but we can sort by y value or description right :)
let fullDom = Blockly.Xml.workspaceToDom(toolbox.flyout_.getWorkspace());
const doms = {};
for (const x of fullDom.children) {
if (x.tagName === "BLOCK") {
let id = x.getAttribute("id");
doms[id] = x;
}
}
for (const block of blocks) {
this.getBlockText(block, options, doms);
}
options.sort((a, b) => a.desc.length < b.desc.length ? -1 : a.desc.length > b.desc.length ? 1 : a.desc.localeCompare(b.desc));
let count = 0;
while (this.dropdown.firstChild) {
this.dropdown.removeChild(this.dropdown.firstChild);
}
for (const option of options) {
const li = document.createElement("li");
const desc = option.desc; // bType = hat block reporter boolean
let bType = this.getEdgeTypeClass(option.block);
count++;
li.innerText = desc;
li.data = {
text: desc,
lower: " " + desc.toLowerCase(),
option: option
};
const blockTypes = {
// Some of these blocks in the flyout have a category of `null` for some reason, the
// same as procedures. Without making bigger changes to the custom block color system
// hardcoding these is the best solution for now.
sensing_of: "sensing",
event_whenbackdropswitchesto: "events"
};
let ending = option.block.getCategory() || blockTypes[option.block.type] || "null";
if (option.block.isScratchExtension) {
ending = "pen";
} else if (addon.tab.getCustomBlock(option.block.procCode_)) {
ending = "addon-custom-block";
}
li.className = "sa-block-color sa-block-color-" + ending + " sa-" + bType;
if (count > this.DROPDOWN_BLOCK_LIST_MAX_ROWS) {
// Limit maximum number of rows to prevent lag when no filter is applied
li.style.display = "none";
}
this.dropdown.appendChild(li);
}
this.dropdownOut.classList.add("vis");
}
navigateFloatFilter(dir) {
let sel = this.dropdown.getElementsByClassName("sel");
let nxt;
if (sel.length > 0 && sel[0].style.display !== "none") {
nxt = dir === -1 ? sel[0].previousSibling : sel[sel.length - 1].nextSibling;
} else {
nxt = this.dropdown.children[0];
dir = 1;
}
while (nxt && nxt.style.display === "none") {
nxt = dir === -1 ? nxt.previousSibling : nxt.nextSibling;
}
if (nxt) {
for (const i of sel) {
i.classList.remove("sel");
}
nxt.classList.add("sel"); // centerTop(nxt.data.labelID);
}
}
getBlockText(block, options, doms) {
// block.type; "looks_nextbackdrop"
let desc;
let picklist, pickField;
let dom = doms[block.id]; // dom = doms[block.type];
const process = block => {
for (const input of block.inputList) {
// input.name = "", input.type = 5
let fields = input.fieldRow;
for (const field of fields) {
// field --- Blockly.FieldLabel .className = "blocklyText"
// Blockly.FieldDropdown --- .className = "blocklyText blocklyDropdownText"
let text;
if (!picklist && field.className_ === "blocklyText blocklyDropdownText") {
picklist = field.getOptions();
pickField = field.name;
if (picklist && picklist.length > 0) {
text = "^^";
} else {
text = field.getText();
}
} else {
text = field.getText();
}
desc = (desc ? desc + " " : "") + text;
}
if (input.connection) {
let innerBlock = input.connection.targetBlock();
if (innerBlock) {
process(innerBlock); // Recursive process connected child blocks...
}
}
}
};
process(block);
if (picklist) {
for (const item of picklist) {
let code = item[1];
if (typeof code !== "string" || // Audio Record is a function!
code === "DELETE_VARIABLE_ID" || code === "RENAME_VARIABLE_ID" || code === "NEW_BROADCAST_MESSAGE_ID" || code === "NEW_BROADCAST_MESSAGE_ID" || // editor-searchable-dropdowns compatibility
code === "createGlobalVariable" || code === "createLocalVariable" || code === "createGlobalList" || code === "createLocalList" || code === "createBroadcast" || // rename-broadcasts compatibility
code === "RENAME_BROADCAST_MESSAGE_ID") {
continue; // Skip these
}
options.push({
desc: desc.replace("^^", item[0]),
block: block,
dom: dom,
option: item,
pickField: pickField
});
}
} else {
options.push({
desc: desc,
block: block,
dom: dom
});
}
return desc;
}
getEdgeTypeClass(block) {
switch (block.edgeShape_) {
case 1:
return "boolean";
case 2:
return "reporter";
default:
return block.startHat_ ? "hat" : "block";
}
}
hide() {
this.floatBar.style.display = "none";
}
}
const floatingInput = new FloatingInput();
const _doWorkspaceClick_ = Blockly.Gesture.prototype.doWorkspaceClick_;
Blockly.Gesture.prototype.doWorkspaceClick_ = function () {
if (!addon.self.disabled && (this.mostRecentEvent_.button === 1 || this.mostRecentEvent_.shiftKey)) {
// Wheel button...
floatingInput.show(this.mostRecentEvent_);
}
_doWorkspaceClick_.call(this);
};
document.addEventListener("mousemove", e => {
mouse = {
x: e.clientX,
y: e.clientY
};
});
});
/***/ }),
/***/ "./src/addons/addons/move-to-top-bottom/_runtime_entry.js":
/*!****************************************************************!*\
!*** ./src/addons/addons/move-to-top-bottom/_runtime_entry.js ***!
\****************************************************************/
/*! exports provided: resources */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "resources", function() { return resources; });
/* harmony import */ var _userscript_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./userscript.js */ "./src/addons/addons/move-to-top-bottom/userscript.js");
/* generated by pull.js */
const resources = {
"userscript.js": _userscript_js__WEBPACK_IMPORTED_MODULE_0__["default"]
};
/***/ }),
/***/ "./src/addons/addons/move-to-top-bottom/userscript.js":
/*!************************************************************!*\
!*** ./src/addons/addons/move-to-top-bottom/userscript.js ***!
\************************************************************/
/*! exports provided: default */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony default export */ __webpack_exports__["default"] = (async ({
addon,
console,
msg
}) => {
const types = ["sound", "costume"];
addon.tab.createEditorContextMenu(ctx => {
const target = addon.tab.traps.vm.editingTarget;
if (ctx.type === "sound") {
target.reorderSound(ctx.index, 0);
} else {
target.reorderCostume(ctx.index, 0);
}
queueMicrotask(() => {
addon.tab.traps.vm.emitTargetsUpdate();
addon.tab.traps.vm.runtime.emitProjectChanged();
ctx.target.click();
});
}, {
types,
position: "assetContextMenuAfterExport",
order: 1,
label: msg("top"),
condition: ctx => ctx.index !== 0
});
addon.tab.createEditorContextMenu(ctx => {
const target = addon.tab.traps.vm.editingTarget;
if (ctx.type === "sound") {
target.reorderSound(ctx.index, Infinity);
} else {
target.reorderCostume(ctx.index, Infinity);
}
queueMicrotask(() => {
addon.tab.traps.vm.emitTargetsUpdate();
addon.tab.traps.vm.runtime.emitProjectChanged();
ctx.target.click();
});
}, {
types,
position: "assetContextMenuAfterExport",
order: 2,
label: msg("bottom"),
condition: ctx => ctx.index !== ctx.target.parentNode.parentNode.childElementCount - 1
});
});
/***/ }),
/***/ "./src/addons/addons/onion-skinning/_runtime_entry.js":
/*!************************************************************!*\
!*** ./src/addons/addons/onion-skinning/_runtime_entry.js ***!
\************************************************************/
/*! exports provided: resources */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "resources", function() { return resources; });
/* harmony import */ var _userscript_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./userscript.js */ "./src/addons/addons/onion-skinning/userscript.js");
/* harmony import */ var _css_loader_style_css__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! css-loader!./style.css */ "./node_modules/css-loader/index.js!./src/addons/addons/onion-skinning/style.css");
/* harmony import */ var _css_loader_style_css__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_css_loader_style_css__WEBPACK_IMPORTED_MODULE_1__);
/* harmony import */ var _url_loader_decrement_svg__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! url-loader!./decrement.svg */ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/onion-skinning/decrement.svg");
/* harmony import */ var _url_loader_increment_svg__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! url-loader!./increment.svg */ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/onion-skinning/increment.svg");
/* harmony import */ var _url_loader_settings_svg__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! url-loader!./settings.svg */ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/onion-skinning/settings.svg");
/* harmony import */ var _url_loader_toggle_svg__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! url-loader!./toggle.svg */ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/onion-skinning/toggle.svg");
/* generated by pull.js */
const resources = {
"userscript.js": _userscript_js__WEBPACK_IMPORTED_MODULE_0__["default"],
"style.css": _css_loader_style_css__WEBPACK_IMPORTED_MODULE_1___default.a,
"decrement.svg": _url_loader_decrement_svg__WEBPACK_IMPORTED_MODULE_2__["default"],
"increment.svg": _url_loader_increment_svg__WEBPACK_IMPORTED_MODULE_3__["default"],
"settings.svg": _url_loader_settings_svg__WEBPACK_IMPORTED_MODULE_4__["default"],
"toggle.svg": _url_loader_toggle_svg__WEBPACK_IMPORTED_MODULE_5__["default"]
};
/***/ }),
/***/ "./src/addons/addons/onion-skinning/userscript.js":
/*!********************************************************!*\
!*** ./src/addons/addons/onion-skinning/userscript.js ***!
\********************************************************/
/*! exports provided: default */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony default export */ __webpack_exports__["default"] = (async function ({
addon,
console,
msg
}) {
const paper = await addon.tab.traps.getPaper();
const paintEditorCanvasContainer = await addon.tab.waitForElement("[class^='paint-editor_canvas-container']");
try {
if (!("colorIndex" in addon.tab.redux.state.scratchPaint.fillMode)) {
console.error("Detected new paint editor; this will be supported in future versions.");
return;
}
} catch (_) {// The check can technically fail when Redux isn't supported (rare cases)
// Just ignore in this case
}
const paperCanvas = paintEditorCanvasContainer[addon.tab.traps.getInternalKey(paintEditorCanvasContainer)].child.child.child.stateNode;
const storedOnionLayers = [];
const parseHexColor = color => {
const hexString = color.substr(1);
const hexNumber = parseInt(hexString, 16);
return [hexNumber >> 16 & 0xff, // R
hexNumber >> 8 & 0xff, // G
hexNumber & 0xff // B
];
};
const settings = {
enabled: addon.settings.get("default") && !addon.self.disabled,
previous: +addon.settings.get("previous"),
next: +addon.settings.get("next"),
opacity: +addon.settings.get("opacity"),
opacityStep: +addon.settings.get("opacityStep"),
layering: addon.settings.get("layering"),
mode: addon.settings.get("mode"),
beforeTint: parseHexColor(addon.settings.get("beforeTint")),
afterTint: parseHexColor(addon.settings.get("afterTint"))
};
const getPaperCenter = () => {
const backgroundGuideLayer = paper.project.layers.find(i => i.data.isBackgroundGuideLayer);
return backgroundGuideLayer.children[0].position;
};
const injectPaper = () => {
// When background guide layer is added, show onion layers.
// https://github.com/LLK/scratch-paint/blob/cdf0afc217633e6cfb8ba90ea4ae38b79882cf6c/src/helper/layer.js#L145
const originalAddLayer = paper.Project.prototype.addLayer;
paper.Project.prototype.addLayer = function (layer) {
const result = originalAddLayer.call(this, layer);
if (layer.data.isBackgroundGuideLayer) {
let onion;
while (onion = storedOnionLayers.shift()) {
originalAddLayer.call(this, onion);
}
relayerOnionLayers();
}
return result;
}; // Scratch uses importJSON to undo or redo
// https://github.com/LLK/scratch-paint/blob/cdf0afc217633e6cfb8ba90ea4ae38b79882cf6c/src/helper/undo.js#L37
// The code prior to this will remove our onion layers, so we have to manually add them back.
const originalImportJSON = paper.Project.prototype.importJSON;
paper.Project.prototype.importJSON = function (json) {
const result = originalImportJSON.call(this, json);
if (settings.enabled) {
updateOnionLayers();
}
return result;
}; // https://github.com/LLK/scratch-paint/blob/cdf0afc217633e6cfb8ba90ea4ae38b79882cf6c/src/helper/layer.js#L114
// When background guide layer is removed, hide onion layers.
const originalRemoveLayer = paper.Layer.prototype.remove;
paper.Layer.prototype.remove = function () {
if (this.data.isBackgroundGuideLayer) {
for (const layer of paper.project.layers) {
if (layer.data.sa_isOnionLayer) {
storedOnionLayers.push(layer);
}
}
for (const layer of storedOnionLayers) {
layer.remove();
}
}
return originalRemoveLayer.call(this);
};
};
const injectPaperCanvas = () => {
let expectingImport = false;
const PaperCanvas = paperCanvas.constructor; // importImage is called to start loading an image.
// https://github.com/LLK/scratch-paint/blob/cdf0afc217633e6cfb8ba90ea4ae38b79882cf6c/src/containers/paper-canvas.jsx#L124
const originalImportImage = PaperCanvas.prototype.importImage;
PaperCanvas.prototype.importImage = function (...args) {
expectingImport = true;
removeOnionLayers();
return originalImportImage.call(this, ...args);
}; // recalibrateSize is called when the canvas finishes loading an image.
// all paths of importImage will result in a call to this method.
// https://github.com/LLK/scratch-paint/blob/cdf0afc217633e6cfb8ba90ea4ae38b79882cf6c/src/containers/paper-canvas.jsx#L310-L327
// We use this to know when to add layers.
const originalRecalibrateSize = PaperCanvas.prototype.recalibrateSize;
PaperCanvas.prototype.recalibrateSize = function (callback) {
return originalRecalibrateSize.call(this, () => {
if (callback) callback();
if (expectingImport) {
expectingImport = false;
if (settings.enabled) {
updateOnionLayers();
}
}
});
}; // Prototype overrides will work for all future instances, but Scratch manually binds some methods to `this`
// so we have to manually copy them for the current instance (but not future instances)
paperCanvas.recalibrateSize = PaperCanvas.prototype.recalibrateSize.bind(paperCanvas);
paperCanvas.importImage = PaperCanvas.prototype.importImage.bind(paperCanvas);
};
const createOnionLayer = () => {
const layer = new paper.Layer();
layer.locked = true;
layer.guide = true;
layer.data.sa_isOnionLayer = true;
return layer;
}; // Each onion layer update is given an ID
// Because updating layers is async, we need this to cancel all but the most recent update
let globalUpdateId = 0;
const cancelOngoingUpdatesAndGetNewId = () => ++globalUpdateId;
const removeOnionLayers = () => {
cancelOngoingUpdatesAndGetNewId();
const project = paper.project;
if (!project) {
return;
}
storedOnionLayers.length = 0;
const layers = project.layers; // Iterate downward because we remove items mid-iteration
for (let i = layers.length - 1; i >= 0; i--) {
const layer = layers[i];
if (layer.data.sa_isOnionLayer) {
layer.remove();
}
}
};
const relayerOnionLayers = () => {
const project = paper.project;
if (!project) {
return;
}
const onionLayer = project.layers.find(i => i.data.sa_isOnionLayer);
if (!onionLayer) {
return;
}
if (settings.layering === "front") {
project.addLayer(onionLayer);
} else {
const rasterLayer = project.layers.find(i => i.data.isRasterLayer);
if (rasterLayer.index === 0) {
project.insertLayer(0, onionLayer);
} else {
project.insertLayer(1, onionLayer);
}
}
};
const recursePaperItem = (item, callback) => {
if (item.children) {
for (const child of item.children) {
recursePaperItem(child, callback);
}
}
callback(item);
};
const getTint = (red, green, blue, isBefore) => {
const referenceColor = isBefore ? settings.beforeTint : settings.afterTint;
const colorAverage = (red + green + blue) / 3 / 255;
const WEIGHT = 1.5;
const weighted = colorAverage / WEIGHT + (1 - 1 / WEIGHT);
return [referenceColor[0] * weighted, referenceColor[1] * weighted, referenceColor[2] * weighted];
};
const toHexColor = ([red, green, blue]) => {
const r = Math.round(red).toString(16).padStart(2, "0");
const g = Math.round(green).toString(16).padStart(2, "0");
const b = Math.round(blue).toString(16).padStart(2, "0");
return "#".concat(r).concat(g).concat(b);
};
const getPaperColorTint = (color, isBefore) => toHexColor(getTint(color.red * 255, color.green * 255, color.blue * 255, isBefore));
const tintRaster = (raster, isBefore) => {
const {
width,
height
} = raster.canvas;
const context = raster.context; // TODO: check to see if this is a performance issue
const imageData = context.getImageData(0, 0, width, height);
const data = imageData.data;
for (let i = 0; i < data.length; i += 4
/* RGBA */
) {
const red = data[i + 0];
const green = data[i + 1];
const blue = data[i + 2];
const alpha = data[i + 3];
if (alpha === 0) {
continue;
}
const newTint = getTint(red, green, blue, isBefore);
data[i + 0] = newTint[0];
data[i + 1] = newTint[1];
data[i + 2] = newTint[2];
}
context.putImageData(imageData, 0, 0);
};
const waitForAllRastersToLoad = root => {
const promises = [];
recursePaperItem(root, item => {
if (item instanceof paper.Raster) {
promises.push(new Promise((resolve, reject) => {
item.on("load", () => resolve());
item.on("error", () => reject(new Error("Raster inside SVG failed to load")));
}));
}
});
return Promise.all(promises);
};
const rasterizeVector = root => {
const bounds = root.strokeBounds;
const {
width,
height
} = bounds; // Some browsers experience extremely poor performance when this value exceeds 3840.
const MAX_SIZE = 3000;
const maxScale = Math.min(MAX_SIZE / width, MAX_SIZE / height);
const raster = new paper.Raster(new paper.Size(width, height));
raster.remove();
raster.smoothing = true;
raster.guide = true;
raster.locked = true;
let renderedAtScale = 0;
const originalDraw = raster.draw;
raster.draw = function (...args) {
const displayedSize = this.getView().getZoom() * window.devicePixelRatio;
const newScale = Math.max(1, Math.min(maxScale, 2 ** Math.ceil(Math.log2(displayedSize))));
if (newScale > renderedAtScale) {
renderedAtScale = newScale;
const canvas = this.canvas;
const ctx = this.context; // Based on https://github.com/LLK/paper.js/blob/16d5ff0267e3a0ef647c25e58182a27300afad20/src/item/Item.js#L1761
const scaledWidth = width * newScale;
const scaledHeight = height * newScale;
canvas.width = scaledWidth;
canvas.height = scaledHeight;
this._size = new paper.Size(scaledWidth, scaledHeight);
const topLeft = bounds.getTopLeft();
const bottomRight = bounds.getBottomRight();
const size = new paper.Size(bottomRight.subtract(topLeft));
const matrix = new paper.Matrix().scale(newScale).translate(topLeft.negate());
ctx.save();
matrix.applyToContext(ctx);
root.draw(ctx, new paper.Base({
matrices: [matrix]
}));
ctx.restore();
this.matrix.reset();
this.transform(new paper.Matrix().translate(topLeft.add(size.divide(2))).scale(1 / newScale));
}
return originalDraw.call(this, ...args);
};
return raster;
};
const makeVectorOnion = (opacity, costume, asset, isBefore) => new Promise((resolve, reject) => {
const {
rotationCenterX,
rotationCenterY
} = costume; // https://github.com/LLK/scratch-paint/blob/cdf0afc217633e6cfb8ba90ea4ae38b79882cf6c/src/containers/paper-canvas.jsx#L196-L218
asset = asset.split(/<\s*svg:/).join("<");
asset = asset.split(/<\/\s*svg:/).join("</");
const svgAttrs = asset.match(/<svg [^>]*>/);
if (svgAttrs && svgAttrs[0].indexOf("xmlns=") === -1) {
asset = asset.replace("<svg ", '<svg xmlns="http://www.w3.org/2000/svg" ');
}
const parser = new DOMParser();
const svgDom = parser.parseFromString(asset, "text/xml");
const viewBox = svgDom.documentElement.attributes.viewBox ? svgDom.documentElement.attributes.viewBox.value.match(/\S+/g) : null;
if (viewBox) {
for (let i = 0; i < viewBox.length; i++) {
viewBox[i] = parseFloat(viewBox[i]);
}
}
const handleLoad = root => {
root.opacity = opacity; // https://github.com/LLK/scratch-paint/blob/cdf0afc217633e6cfb8ba90ea4ae38b79882cf6c/src/containers/paper-canvas.jsx#L274-L275
recursePaperItem(root, i => {
if (i.className === "PathItem") {
i.clockwise = true;
}
if (i.className !== "PointText" && !i.children) {
if (i.strokeWidth) {
i.strokeWidth = i.strokeWidth * 2;
}
}
i.locked = true;
i.guide = true;
});
root.scale(2, new paper.Point(0, 0));
if (settings.mode === "tint") {
const gradients = new Set();
recursePaperItem(root, i => {
if (i.strokeColor) {
i.strokeColor = getPaperColorTint(i.strokeColor, isBefore);
}
if (i.fillColor) {
const gradient = i.fillColor.gradient;
if (gradient) {
if (gradients.has(gradient)) return;
gradients.add(gradient);
for (const stop of gradient.stops) {
stop.color = getPaperColorTint(stop.color, isBefore);
}
} else {
i.fillColor = getPaperColorTint(i.fillColor, isBefore);
}
}
if (i.canvas) {
tintRaster(i, isBefore);
}
});
}
const paperCenter = getPaperCenter(); // https://github.com/LLK/scratch-paint/blob/cdf0afc217633e6cfb8ba90ea4ae38b79882cf6c/src/containers/paper-canvas.jsx#L277-L287
if (typeof rotationCenterX !== "undefined" && typeof rotationCenterY !== "undefined") {
let rotationPoint = new paper.Point(rotationCenterX, rotationCenterY);
if (viewBox && viewBox.length >= 2 && !isNaN(viewBox[0]) && !isNaN(viewBox[1])) {
rotationPoint = rotationPoint.subtract(viewBox[0], viewBox[1]);
}
root.translate(paperCenter.subtract(rotationPoint.multiply(2)));
} else {
root.translate(paperCenter.subtract(root.bounds.width, root.bounds.height));
}
return rasterizeVector(root);
};
paper.project.importSVG(asset, {
expandShapes: true,
insert: false,
onLoad: root => {
if (!root) {
reject(new Error("could not load onion skin"));
return;
}
resolve(waitForAllRastersToLoad(root).then(() => handleLoad(root)));
}
});
});
const makeRasterOnion = (opacity, costume, asset, isBefore) => new Promise((resolve, reject) => {
let {
rotationCenterX,
rotationCenterY
} = costume;
const image = new Image();
image.onload = () => {
const paperCenter = getPaperCenter();
const width = Math.min(paperCenter.x * 2, image.width);
const height = Math.min(paperCenter.y * 2, image.height); // https://github.com/LLK/scratch-paint/blob/cdf0afc217633e6cfb8ba90ea4ae38b79882cf6c/src/containers/paper-canvas.jsx#L151-L156
if (typeof rotationCenterX === "undefined") {
rotationCenterX = width / 2;
}
if (typeof rotationCenterY === "undefined") {
rotationCenterY = height / 2;
}
const raster = new paper.Raster(image);
raster.opacity = opacity;
raster.guide = true;
raster.locked = true;
const x = width / 2 + (paperCenter.x - rotationCenterX);
const y = height / 2 + (paperCenter.y - rotationCenterY);
raster.position = new paper.Point(x, y);
raster.remove();
if (settings.mode === "tint") {
tintRaster(raster, isBefore);
}
resolve(raster);
};
image.onerror = () => {
reject(new Error("could not load image"));
};
image.src = asset;
});
const getSelectedCostumeIndex = () => {
const item = document.querySelector("[class*='selector_list-item'][class*='sprite-selector-item_is-selected']");
if (!item) return -1;
const numberEl = item.querySelector("[class*='sprite-selector-item_number']");
if (!numberEl) return -1;
return +numberEl.textContent - 1;
};
const updateOnionLayers = async () => {
const project = paper.project;
if (!project) {
return;
}
const selectedCostumeIndex = getSelectedCostumeIndex();
if (selectedCostumeIndex === -1) {
return;
}
removeOnionLayers();
const localUpdateId = cancelOngoingUpdatesAndGetNewId();
const vm = addon.tab.traps.vm;
if (!vm) {
return;
}
const originalActiveLayer = project.activeLayer;
const costumes = vm.editingTarget.sprite.costumes;
const startIndex = Math.max(0, selectedCostumeIndex - settings.previous);
const endIndex = Math.min(costumes.length - 1, selectedCostumeIndex + settings.next);
try {
const layersToCreate = [];
for (let i = startIndex; i <= endIndex; i++) {
if (i === selectedCostumeIndex) {
continue;
}
const isBefore = i < selectedCostumeIndex;
const distance = Math.abs(i - selectedCostumeIndex) - 1;
const opacity = (settings.opacity - settings.opacityStep * distance) / 100;
if (opacity <= 0) {
continue;
}
layersToCreate.push({
index: i,
isBefore,
opacity
});
}
const onions = await Promise.all(layersToCreate.map(({
index,
isBefore,
opacity
}) => {
const onionCostume = costumes[index];
const onionAsset = vm.getCostume(index);
if (onionCostume.dataFormat === "svg") {
return makeVectorOnion(opacity, onionCostume, onionAsset, isBefore);
} else if (onionCostume.dataFormat === "png" || onionCostume.dataFormat === "jpg") {
return makeRasterOnion(opacity, onionCostume, onionAsset, isBefore);
} else {
throw new Error("Unknown data format: ".concat(onionCostume.dataFormat));
}
})); // Make sure we haven't been cancelled
if (globalUpdateId === localUpdateId) {
const layer = createOnionLayer();
for (const item of onions) {
layer.addChild(item);
}
relayerOnionLayers();
}
} catch (e) {
console.error(e);
} // We must make sure to always reset the active layer to avoid corruption.
originalActiveLayer.activate();
};
const setEnabled = _enabled => {
if (settings.enabled === _enabled) {
return;
}
settings.enabled = _enabled;
if (settings.enabled) {
if (settings.next === 0 && settings.previous === 0) {
settings.previous = 1;
layerInputs.previous.value = settings.previous;
}
if (settings.opacity === 0) {
settings.opacity = 25;
layerInputs.opacity.value = settings.opacity;
}
updateOnionLayers();
} else {
removeOnionLayers();
}
toggleButton.dataset.enabled = settings.enabled;
}; //
// Controls below editor
//
const settingsChanged = onlyRelayerNeeded => {
if (settings.previous === 0 && settings.next === 0 || settings.opacity === 0) {
setEnabled(false);
return;
}
if (settings.enabled) {
if (onlyRelayerNeeded) {
relayerOnionLayers();
} else {
updateOnionLayers();
}
} else if (settings.previous > 0 || settings.next > 0) {
setEnabled(true);
}
};
const createGroup = () => {
const el = document.createElement("div");
el.className = "sa-onion-group";
return el;
};
const createButton = ({
useButtonTag
} = {}) => {
const el = document.createElement(useButtonTag ? "button" : "span");
el.className = "sa-onion-button";
el.setAttribute("role", "button");
return el;
};
const createButtonImage = name => {
const el = document.createElement("img");
el.className = "sa-onion-image";
el.draggable = false;
el.dataset.image = name;
el.loading = "lazy";
el.src = addon.self.getResource("/" + name + ".svg")
/* rewritten by pull.js */
;
return el;
};
const toggleControlsGroup = createGroup();
addon.tab.displayNoneWhileDisabled(toggleControlsGroup, {
display: "flex"
});
const toggleButton = createButton();
toggleButton.dataset.enabled = settings.enabled;
toggleButton.addEventListener("click", () => setEnabled(!settings.enabled));
toggleButton.title = msg("toggle");
toggleButton.appendChild(createButtonImage("toggle"));
const settingButton = createButton();
settingButton.addEventListener("click", () => setSettingsOpen(!areSettingsOpen()));
settingButton.title = msg("settings");
settingButton.appendChild(createButtonImage("settings")); //
// Settings page
//
const settingPageWrapper = document.createElement("div");
settingPageWrapper.className = "sa-onion-settings-wrapper";
toggleControlsGroup.append(settingPageWrapper, toggleButton, settingButton);
const settingsPage = document.createElement("div");
settingsPage.className = "sa-onion-settings";
const setSettingsOpen = open => {
settingButton.dataset.enabled = open;
settingsPage.dataset.visible = open;
};
const areSettingsOpen = () => settingsPage.dataset.visible === "true";
const layerInputs = {};
for (const type of ["previous", "next", "opacity", "opacityStep"]) {
const container = document.createElement("label");
container.className = "sa-onion-settings-line";
const label = document.createElement("div");
label.className = "sa-onion-settings-label";
label.textContent = msg(type);
container.appendChild(label);
const group = createGroup();
const currentButton = createButton();
const filler = document.createElement("div");
filler.style.width = "20px";
currentButton.appendChild(filler);
const currentInput = document.createElement("input");
layerInputs[type] = currentInput;
currentInput.className = "sa-onion-settings-input";
currentInput.type = "number";
currentInput.step = "1";
currentInput.min = "0";
currentInput.max = "100";
currentInput.value = settings[type];
currentInput.addEventListener("input", e => {
if (currentInput.value.length === 0) {
settings[type] = 0;
settingsChanged();
return;
}
let value = +currentInput.value;
if (value > +currentInput.max) {
value = +currentInput.max;
} else if (value < 0) {
value = 0;
}
currentInput.value = value;
settings[type] = value;
settingsChanged();
});
currentInput.addEventListener("blur", () => {
if (!currentInput.value) {
currentInput.value = "0";
}
});
currentButton.appendChild(currentInput);
const decrementButton = createButton();
decrementButton.appendChild(createButtonImage("decrement"));
decrementButton.addEventListener("click", () => {
if (settings[type] > 0) {
settings[type]--;
currentInput.value = settings[type];
settingsChanged();
}
});
const incrementButton = createButton();
incrementButton.appendChild(createButtonImage("increment"));
incrementButton.addEventListener("click", () => {
if (settings[type] < +currentInput.max) {
settings[type]++;
currentInput.value = settings[type];
settingsChanged();
}
});
group.appendChild(decrementButton);
group.appendChild(currentButton);
group.appendChild(incrementButton);
container.appendChild(group);
settingsPage.appendChild(container);
}
const modeContainer = document.createElement("div");
modeContainer.className = "sa-onion-settings-line";
const modeLabel = document.createElement("div");
modeLabel.className = "sa-onion-settings-label";
modeLabel.textContent = msg("mode");
const modeGroup = createGroup();
modeContainer.appendChild(modeLabel);
const modeMergeButton = createButton({
useButtonTag: true
});
modeMergeButton.appendChild(document.createTextNode(msg("merge")));
modeGroup.appendChild(modeMergeButton);
modeMergeButton.addEventListener("click", e => {
settings.mode = "merge";
modeTintButton.dataset.enabled = false;
modeMergeButton.dataset.enabled = true;
settingsChanged();
});
modeMergeButton.dataset.enabled = settings.mode === "merge";
const modeTintButton = createButton({
useButtonTag: true
});
modeTintButton.appendChild(document.createTextNode(msg("tint")));
modeGroup.appendChild(modeTintButton);
modeTintButton.addEventListener("click", e => {
settings.mode = "tint";
modeTintButton.dataset.enabled = true;
modeMergeButton.dataset.enabled = false;
settingsChanged();
});
modeTintButton.dataset.enabled = settings.mode === "tint";
modeContainer.appendChild(modeGroup);
settingsPage.appendChild(modeContainer);
const layeringContainer = document.createElement("div");
layeringContainer.className = "sa-onion-settings-line";
const layeringLabel = document.createElement("div");
layeringLabel.className = "sa-onion-settings-label";
layeringLabel.textContent = msg("layering");
const layeringGroup = createGroup();
layeringContainer.appendChild(layeringLabel);
const layeringFrontButton = createButton({
useButtonTag: true
});
layeringFrontButton.appendChild(document.createTextNode(msg("front")));
layeringGroup.appendChild(layeringFrontButton);
layeringFrontButton.addEventListener("click", e => {
settings.layering = "front";
layeringBehindButton.dataset.enabled = false;
layeringFrontButton.dataset.enabled = true;
settingsChanged(true);
});
layeringFrontButton.dataset.enabled = settings.layering === "front";
const layeringBehindButton = createButton({
useButtonTag: true
});
layeringBehindButton.appendChild(document.createTextNode(msg("behind")));
layeringGroup.appendChild(layeringBehindButton);
layeringBehindButton.addEventListener("click", e => {
settings.layering = "behind";
layeringBehindButton.dataset.enabled = true;
layeringFrontButton.dataset.enabled = false;
settingsChanged(true);
});
layeringBehindButton.dataset.enabled = settings.layering === "behind";
layeringContainer.appendChild(layeringGroup);
settingsPage.appendChild(layeringContainer);
const SVG_NS = "http://www.w3.org/2000/svg";
const settingsTip = document.createElementNS(SVG_NS, "svg");
settingsTip.setAttribute("class", "sa-onion-settings-tip");
settingsTip.setAttribute("width", "14");
settingsTip.setAttribute("height", "7");
const settingsTipShape = document.createElementNS(SVG_NS, "polygon");
settingsTipShape.setAttribute("class", "sa-onion-settings-polygon");
settingsTipShape.setAttribute("points", "0,0 7,7, 14,0");
settingsTip.appendChild(settingsTipShape);
settingsPage.appendChild(settingsTip);
let oldEnabled = null;
addon.self.addEventListener("disabled", () => {
setSettingsOpen(false);
oldEnabled = settings.enabled;
setEnabled(false);
});
addon.self.addEventListener("reenabled", () => {
setEnabled(oldEnabled);
});
const controlsLoop = async () => {
let hasRunOnce = false;
while (true) {
const canvasControls = await addon.tab.waitForElement("[class^='paint-editor_canvas-controls']", {
markAsSeen: true,
reduxEvents: ["scratch-gui/navigation/ACTIVATE_TAB", "scratch-gui/mode/SET_PLAYER", "fontsLoaded/SET_FONTS_LOADED", "scratch-gui/locales/SELECT_LOCALE", "scratch-gui/targets/UPDATE_TARGET_LIST"],
reduxCondition: state => state.scratchGui.editorTab.activeTabIndex === 1 && !state.scratchGui.mode.isPlayerOnly
});
const zoomControlsContainer = canvasControls.querySelector("[class^='paint-editor_zoom-controls']");
addon.tab.appendToSharedSpace({
space: "paintEditorZoomControls",
element: toggleControlsGroup,
order: 1
});
settingPageWrapper.appendChild(settingsPage);
if (!hasRunOnce) {
hasRunOnce = true;
const groupClass = zoomControlsContainer.firstChild.className;
const buttonClass = zoomControlsContainer.firstChild.firstChild.className;
const imageClass = zoomControlsContainer.firstChild.firstChild.firstChild.className;
for (const el of document.querySelectorAll(".sa-onion-group")) {
el.className += " " + groupClass;
}
for (const el of document.querySelectorAll(".sa-onion-button")) {
el.className += " " + buttonClass;
}
for (const el of document.querySelectorAll(".sa-onion-image")) {
el.className += " " + imageClass;
}
}
if (settings.enabled) {
updateOnionLayers();
}
}
};
injectPaper();
injectPaperCanvas();
controlsLoop();
});
/***/ }),
/***/ "./src/addons/addons/pick-colors-from-stage/_runtime_entry.js":
/*!********************************************************************!*\
!*** ./src/addons/addons/pick-colors-from-stage/_runtime_entry.js ***!
\********************************************************************/
/*! exports provided: resources */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "resources", function() { return resources; });
/* harmony import */ var _userscript_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./userscript.js */ "./src/addons/addons/pick-colors-from-stage/userscript.js");
/* harmony import */ var _css_loader_style_css__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! css-loader!./style.css */ "./node_modules/css-loader/index.js!./src/addons/addons/pick-colors-from-stage/style.css");
/* harmony import */ var _css_loader_style_css__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_css_loader_style_css__WEBPACK_IMPORTED_MODULE_1__);
/* generated by pull.js */
const resources = {
"userscript.js": _userscript_js__WEBPACK_IMPORTED_MODULE_0__["default"],
"style.css": _css_loader_style_css__WEBPACK_IMPORTED_MODULE_1___default.a
};
/***/ }),
/***/ "./src/addons/addons/pick-colors-from-stage/userscript.js":
/*!****************************************************************!*\
!*** ./src/addons/addons/pick-colors-from-stage/userscript.js ***!
\****************************************************************/
/*! exports provided: default */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); if (enumerableOnly) { symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; }); } keys.push.apply(keys, symbols); } return keys; }
function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i] != null ? arguments[i] : {}; if (i % 2) { ownKeys(Object(source), true).forEach(function (key) { _defineProperty(target, key, source[key]); }); } else if (Object.getOwnPropertyDescriptors) { Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)); } else { ownKeys(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } } return target; }
function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
/* harmony default export */ __webpack_exports__["default"] = (async function ({
addon,
msg,
console
}) {
const brand = Symbol();
const setIsPicking = picking => document.body.classList.toggle("sa-stage-color-picker-picking", picking); // We only want to handle color picker events from the user clicking on the button, not from
// addons or other scripts pressing it with click().
let isMostRecentClickUserInitiated = false;
document.addEventListener("click", e => {
isMostRecentClickUserInitiated = e.isTrusted;
}, {
capture: true
});
addon.tab.redux.initialize();
addon.tab.redux.addEventListener("statechanged", e => {
const action = e.detail.action; // Do not process events emitted by ourselves.
if (action[brand]) {
return;
}
if (!addon.self.disabled && isMostRecentClickUserInitiated && action.type === "scratch-paint/eye-dropper/ACTIVATE_COLOR_PICKER") {
setIsPicking(true); // When scratch-paint's color picker is activated, also activate scratch-gui's color picker.
addon.tab.redux.dispatch({
type: "scratch-gui/color-picker/ACTIVATE_COLOR_PICKER",
callback: color => {
// callback is called from reducer; do not dispatch events in reducer
queueMicrotask(() => {
// By the time we get here, scratch-paint will have already deactivated its eye dropper.
// If we were to just call the callback, the color would indeed update, but the sliders
// in the color selector would not update.
// https://github.com/LLK/scratch-paint/blob/970b72c3e75d0ad44ab54e403a44786ca5f45512/src/containers/color-picker.jsx#L64
// To work around this, we will re-enable the color picker before running the callback.
addon.tab.redux.dispatch(_objectSpread(_objectSpread({}, action), {}, {
[brand]: true
}));
action.callback(color);
if (action.previousMode) {
action.previousMode.activate();
}
addon.tab.redux.dispatch({
type: "scratch-paint/eye-dropper/DEACTIVATE_COLOR_PICKER",
[brand]: true
});
setIsPicking(false);
});
}
});
} // Don't check for addon being disabled here in case we were dynamically disabled while color
// picking. This code won't do anything anyways when the previous code doesn't run.
if (action.type === "scratch-paint/eye-dropper/DEACTIVATE_COLOR_PICKER") {
setIsPicking(false); // When someone selects a color in the scratch-paint picker, cancel the scratch-gui picker
if (addon.tab.redux.state.scratchGui.colorPicker.active) {
addon.tab.redux.dispatch({
type: "scratch-gui/color-picker/DEACTIVATE_COLOR_PICKER",
[brand]: true
});
}
}
});
});
/***/ }),
/***/ "./src/addons/addons/rename-broadcasts/_runtime_entry.js":
/*!***************************************************************!*\
!*** ./src/addons/addons/rename-broadcasts/_runtime_entry.js ***!
\***************************************************************/
/*! exports provided: resources */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "resources", function() { return resources; });
/* harmony import */ var _userscript_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./userscript.js */ "./src/addons/addons/rename-broadcasts/userscript.js");
/* generated by pull.js */
const resources = {
"userscript.js": _userscript_js__WEBPACK_IMPORTED_MODULE_0__["default"]
};
/***/ }),
/***/ "./src/addons/addons/rename-broadcasts/userscript.js":
/*!***********************************************************!*\
!*** ./src/addons/addons/rename-broadcasts/userscript.js ***!
\***********************************************************/
/*! exports provided: default */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony default export */ __webpack_exports__["default"] = (async function ({
addon,
msg,
console
}) {
const vm = addon.tab.traps.vm;
const Blockly = await addon.tab.traps.getBlockly(); // editor-searchable-dropdowns relies on this value
const RENAME_BROADCAST_MESSAGE_ID = "RENAME_BROADCAST_MESSAGE_ID";
const BROADCAST_MESSAGE_TYPE = Blockly.BROADCAST_MESSAGE_VARIABLE_TYPE;
const _dropdownCreate = Blockly.FieldVariable.dropdownCreate;
Blockly.FieldVariable.dropdownCreate = function () {
const options = _dropdownCreate.call(this);
if (!addon.self.disabled && this.defaultType_ === BROADCAST_MESSAGE_TYPE && // Disable when workspace has no actual broadcast to rename
this.sourceBlock_.workspace.getVariableTypes().includes("broadcast_msg")) {
options.push([msg("RENAME_BROADCAST"), RENAME_BROADCAST_MESSAGE_ID]);
}
return options;
};
const _onItemSelected = Blockly.FieldVariable.prototype.onItemSelected;
Blockly.FieldVariable.prototype.onItemSelected = function (menu, menuItem) {
const workspace = this.sourceBlock_.workspace;
if (this.sourceBlock_ && workspace) {
if (menuItem.getValue() === RENAME_BROADCAST_MESSAGE_ID) {
promptRenameBroadcast(workspace, this.variable_);
return;
}
}
return _onItemSelected.call(this, menu, menuItem);
};
const resetVMCaches = () => {
const blockContainers = new Set(vm.runtime.targets.map(i => i.blocks));
for (const blocks of blockContainers) {
blocks.resetCache();
}
};
const addUndoRedoHook = callback => {
const eventQueue = Blockly.Events.FIRE_QUEUE_; // After a rename is emitted, some unrelated garbage events also get emitted
// So we should trap the first event
const undoItem = eventQueue[0];
const originalRun = undoItem.run;
undoItem.run = function (isRedo) {
originalRun.call(this, isRedo);
callback(isRedo);
};
};
const renameBroadcastInVM = (id, newName) => {
// Editor's rename won't completely rename the variable.
const vmVariable = vm.runtime.getTargetForStage().variables[id];
vmVariable.name = newName;
vmVariable.value = newName; // Update all references to the broadcast. Broadcasts won't work if these
// don't match.
const blockContainers = new Set(vm.runtime.targets.map(i => i.blocks));
for (const blockContainer of blockContainers) {
for (const block of Object.values(blockContainer._blocks)) {
const broadcastOption = block.fields && block.fields.BROADCAST_OPTION;
if (broadcastOption && broadcastOption.id === id) {
broadcastOption.value = newName;
}
}
}
resetVMCaches();
};
const renameBroadcast = (workspace, id, oldName, newName) => {
// Rename in editor. Undo/redo will work automatically.
workspace.renameVariableById(id, newName); // Rename in VM. Need to manually implement undo/redo.
renameBroadcastInVM(id, newName);
addUndoRedoHook(isRedo => {
if (isRedo) {
renameBroadcastInVM(id, newName);
} else {
renameBroadcastInVM(id, oldName);
}
});
};
const mergeBroadcast = (workspace, oldId, oldName, newName) => {
const newVmVariable = vm.runtime.getTargetForStage().lookupBroadcastByInputValue(newName);
const newId = newVmVariable.id; // Merge in editor. Undo/redo will work automatically for this.
// Use group so that everything here is undone/redone at the same time.
Blockly.Events.setGroup(true);
for (const block of workspace.getAllBlocks()) {
for (const input of block.inputList) {
for (const field of input.fieldRow) {
if (field.name === "BROADCAST_OPTION" && field.getValue() === oldId) {
field.setValue(newId);
}
}
}
} // Remove the broadcast from the editor so it doesn't appear in dropdowns.
// Undo/redo will work automatically for this.
workspace.deleteVariableById(oldId);
Blockly.Events.setGroup(false); // Merge in VM to update sprites that aren't open. Need to manually implement undo/redo.
// To figure out how to undo this operation, we first figure out which blocks we're
// going to touch and keep hold of that list.
const vmBlocksToUpdate = [];
const blockContainers = new Set(vm.runtime.targets.map(i => i.blocks));
for (const blockContainer of blockContainers) {
for (const block of Object.values(blockContainer._blocks)) {
const broadcastOption = block.fields && block.fields.BROADCAST_OPTION;
if (broadcastOption && broadcastOption.id === oldId) {
vmBlocksToUpdate.push(block);
}
}
}
const applyVmEdits = isRedo => {
const idToReplaceWith = isRedo ? newId : oldId;
const nameToReplaceWith = isRedo ? newName : oldName;
for (const block of vmBlocksToUpdate) {
const broadcastOption = block.fields.BROADCAST_OPTION;
broadcastOption.id = idToReplaceWith;
broadcastOption.value = nameToReplaceWith;
}
resetVMCaches();
};
applyVmEdits(true); // Earlier editor updates are guaranteed to generate at least 1 event that we can hook as the
// broadcast block must exist in the editor for the user to rename it.
addUndoRedoHook(isRedo => {
applyVmEdits(isRedo);
});
};
const promptRenameBroadcast = (workspace, variable) => {
const modalTitle = msg("RENAME_BROADCAST_MODAL_TITLE");
const oldName = variable.name;
const id = variable.getId();
const promptText = msg("RENAME_BROADCAST_TITLE", {
name: oldName
});
const promptDefaultText = oldName;
Blockly.prompt(promptText, promptDefaultText, function (newName) {
newName = Blockly.Variables.trimName_(newName);
const nameIsEmpty = !newName;
if (nameIsEmpty) {
return;
}
const variableAlreadyExists = !!workspace.getVariable(newName, BROADCAST_MESSAGE_TYPE);
if (variableAlreadyExists) {
mergeBroadcast(workspace, id, oldName, newName);
} else {
renameBroadcast(workspace, id, oldName, newName);
}
}, modalTitle, BROADCAST_MESSAGE_TYPE);
};
const updateExistingMenuGenerators = () => {
const workspace = Blockly.getMainWorkspace();
const flyout = workspace && workspace.getFlyout();
if (workspace && flyout) {
const allBlocks = [...workspace.getAllBlocks(), ...flyout.getWorkspace().getAllBlocks()];
for (const block of allBlocks) {
for (const input of block.inputList) {
for (const field of input.fieldRow) {
if (field instanceof Blockly.FieldVariable) {
field.menuGenerator_ = Blockly.FieldVariable.dropdownCreate;
}
}
}
}
}
};
updateExistingMenuGenerators();
});
/***/ }),
/***/ "./src/addons/libraries/common/cs/normalize-color.js":
/*!***********************************************************!*\
!*** ./src/addons/libraries/common/cs/normalize-color.js ***!
\***********************************************************/
/*! exports provided: getHexRegex, normalizeHex */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "getHexRegex", function() { return getHexRegex; });
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "normalizeHex", function() { return normalizeHex; });
const getHexRegex = () => /^#?[0-9a-fA-F]{3,8}$/;
const normalizeHex = input => {
let hex = String(input);
if (!getHexRegex().test(hex)) return "#000000";
if (!hex.startsWith("#")) hex = "#".concat(hex);
if (hex.length === 4) {
const [_, r, g, b] = hex;
hex = "#".concat(r).concat(r).concat(g).concat(g).concat(b).concat(b);
}
return hex.toLowerCase();
};
/***/ }),
/***/ "./src/addons/libraries/common/cs/rate-limiter.js":
/*!********************************************************!*\
!*** ./src/addons/libraries/common/cs/rate-limiter.js ***!
\********************************************************/
/*! exports provided: default */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "default", function() { return RateLimiter; });
class RateLimiter {
constructor(wait) {
this.timeout = null;
this.callback = null;
this.wait = wait;
}
abort(call = true) {
if (this.timeout) {
clearTimeout(this.timeout);
if (call) this.callback();
this.timeout = this.callback = null;
}
}
limit(callback) {
this.abort(false);
this.callback = callback;
this.timeout = setTimeout(() => {
this.timeout = this.callback = null;
callback();
}, this.wait);
}
}
/***/ }),
/***/ "./src/addons/libraries/thirdparty/cs/tinycolor-min.js":
/*!*************************************************************!*\
!*** ./src/addons/libraries/thirdparty/cs/tinycolor-min.js ***!
\*************************************************************/
/*! exports provided: default */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
// TinyColor v1.4.2
// https://github.com/bgrins/TinyColor
// Brian Grinstead, MIT License
// Modified to use ES6 export
const tinycolor = function (Math) {
var trimLeft = /^\s+/,
trimRight = /\s+$/,
tinyCounter = 0,
mathRound = Math.round,
mathMin = Math.min,
mathMax = Math.max,
mathRandom = Math.random;
function tinycolor(color, opts) {
color = color ? color : '';
opts = opts || {}; // If input is already a tinycolor, return itself
if (color instanceof tinycolor) {
return color;
} // If we are called as a function, call using new instead
if (!(this instanceof tinycolor)) {
return new tinycolor(color, opts);
}
var rgb = inputToRGB(color);
this._originalInput = color, this._r = rgb.r, this._g = rgb.g, this._b = rgb.b, this._a = rgb.a, this._roundA = mathRound(100 * this._a) / 100, this._format = opts.format || rgb.format;
this._gradientType = opts.gradientType; // Don't let the range of [0,255] come back in [0,1].
// Potentially lose a little bit of precision here, but will fix issues where
// .5 gets interpreted as half of the total, instead of half of 1
// If it was supposed to be 128, this was already taken care of by `inputToRgb`
if (this._r < 1) {
this._r = mathRound(this._r);
}
if (this._g < 1) {
this._g = mathRound(this._g);
}
if (this._b < 1) {
this._b = mathRound(this._b);
}
this._ok = rgb.ok;
this._tc_id = tinyCounter++;
}
tinycolor.prototype = {
isDark: function isDark() {
return this.getBrightness() < 128;
},
isLight: function isLight() {
return !this.isDark();
},
isValid: function isValid() {
return this._ok;
},
getOriginalInput: function getOriginalInput() {
return this._originalInput;
},
getFormat: function getFormat() {
return this._format;
},
getAlpha: function getAlpha() {
return this._a;
},
getBrightness: function getBrightness() {
//http://www.w3.org/TR/AERT#color-contrast
var rgb = this.toRgb();
return (rgb.r * 299 + rgb.g * 587 + rgb.b * 114) / 1000;
},
getLuminance: function getLuminance() {
//http://www.w3.org/TR/2008/REC-WCAG20-20081211/#relativeluminancedef
var rgb = this.toRgb();
var RsRGB, GsRGB, BsRGB, R, G, B;
RsRGB = rgb.r / 255;
GsRGB = rgb.g / 255;
BsRGB = rgb.b / 255;
if (RsRGB <= 0.03928) {
R = RsRGB / 12.92;
} else {
R = Math.pow((RsRGB + 0.055) / 1.055, 2.4);
}
if (GsRGB <= 0.03928) {
G = GsRGB / 12.92;
} else {
G = Math.pow((GsRGB + 0.055) / 1.055, 2.4);
}
if (BsRGB <= 0.03928) {
B = BsRGB / 12.92;
} else {
B = Math.pow((BsRGB + 0.055) / 1.055, 2.4);
}
return 0.2126 * R + 0.7152 * G + 0.0722 * B;
},
setAlpha: function setAlpha(value) {
this._a = boundAlpha(value);
this._roundA = mathRound(100 * this._a) / 100;
return this;
},
toHsv: function toHsv() {
var hsv = rgbToHsv(this._r, this._g, this._b);
return {
h: hsv.h * 360,
s: hsv.s,
v: hsv.v,
a: this._a
};
},
toHsvString: function toHsvString() {
var hsv = rgbToHsv(this._r, this._g, this._b);
var h = mathRound(hsv.h * 360),
s = mathRound(hsv.s * 100),
v = mathRound(hsv.v * 100);
return this._a == 1 ? "hsv(" + h + ", " + s + "%, " + v + "%)" : "hsva(" + h + ", " + s + "%, " + v + "%, " + this._roundA + ")";
},
toHsl: function toHsl() {
var hsl = rgbToHsl(this._r, this._g, this._b);
return {
h: hsl.h * 360,
s: hsl.s,
l: hsl.l,
a: this._a
};
},
toHslString: function toHslString() {
var hsl = rgbToHsl(this._r, this._g, this._b);
var h = mathRound(hsl.h * 360),
s = mathRound(hsl.s * 100),
l = mathRound(hsl.l * 100);
return this._a == 1 ? "hsl(" + h + ", " + s + "%, " + l + "%)" : "hsla(" + h + ", " + s + "%, " + l + "%, " + this._roundA + ")";
},
toHex: function toHex(allow3Char) {
return rgbToHex(this._r, this._g, this._b, allow3Char);
},
toHexString: function toHexString(allow3Char) {
return '#' + this.toHex(allow3Char);
},
toHex8: function toHex8(allow4Char) {
return rgbaToHex(this._r, this._g, this._b, this._a, allow4Char);
},
toHex8String: function toHex8String(allow4Char) {
return '#' + this.toHex8(allow4Char);
},
toRgb: function toRgb() {
return {
r: mathRound(this._r),
g: mathRound(this._g),
b: mathRound(this._b),
a: this._a
};
},
toRgbString: function toRgbString() {
return this._a == 1 ? "rgb(" + mathRound(this._r) + ", " + mathRound(this._g) + ", " + mathRound(this._b) + ")" : "rgba(" + mathRound(this._r) + ", " + mathRound(this._g) + ", " + mathRound(this._b) + ", " + this._roundA + ")";
},
toPercentageRgb: function toPercentageRgb() {
return {
r: mathRound(bound01(this._r, 255) * 100) + "%",
g: mathRound(bound01(this._g, 255) * 100) + "%",
b: mathRound(bound01(this._b, 255) * 100) + "%",
a: this._a
};
},
toPercentageRgbString: function toPercentageRgbString() {
return this._a == 1 ? "rgb(" + mathRound(bound01(this._r, 255) * 100) + "%, " + mathRound(bound01(this._g, 255) * 100) + "%, " + mathRound(bound01(this._b, 255) * 100) + "%)" : "rgba(" + mathRound(bound01(this._r, 255) * 100) + "%, " + mathRound(bound01(this._g, 255) * 100) + "%, " + mathRound(bound01(this._b, 255) * 100) + "%, " + this._roundA + ")";
},
toName: function toName() {
if (this._a === 0) {
return "transparent";
}
if (this._a < 1) {
return false;
}
return hexNames[rgbToHex(this._r, this._g, this._b, true)] || false;
},
toFilter: function toFilter(secondColor) {
var hex8String = '#' + rgbaToArgbHex(this._r, this._g, this._b, this._a);
var secondHex8String = hex8String;
var gradientType = this._gradientType ? "GradientType = 1, " : "";
if (secondColor) {
var s = tinycolor(secondColor);
secondHex8String = '#' + rgbaToArgbHex(s._r, s._g, s._b, s._a);
}
return "progid:DXImageTransform.Microsoft.gradient(" + gradientType + "startColorstr=" + hex8String + ",endColorstr=" + secondHex8String + ")";
},
toString: function toString(format) {
var formatSet = !!format;
format = format || this._format;
var formattedString = false;
var hasAlpha = this._a < 1 && this._a >= 0;
var needsAlphaFormat = !formatSet && hasAlpha && (format === "hex" || format === "hex6" || format === "hex3" || format === "hex4" || format === "hex8" || format === "name");
if (needsAlphaFormat) {
// Special case for "transparent", all other non-alpha formats
// will return rgba when there is transparency.
if (format === "name" && this._a === 0) {
return this.toName();
}
return this.toRgbString();
}
if (format === "rgb") {
formattedString = this.toRgbString();
}
if (format === "prgb") {
formattedString = this.toPercentageRgbString();
}
if (format === "hex" || format === "hex6") {
formattedString = this.toHexString();
}
if (format === "hex3") {
formattedString = this.toHexString(true);
}
if (format === "hex4") {
formattedString = this.toHex8String(true);
}
if (format === "hex8") {
formattedString = this.toHex8String();
}
if (format === "name") {
formattedString = this.toName();
}
if (format === "hsl") {
formattedString = this.toHslString();
}
if (format === "hsv") {
formattedString = this.toHsvString();
}
return formattedString || this.toHexString();
},
clone: function clone() {
return tinycolor(this.toString());
},
_applyModification: function _applyModification(fn, args) {
var color = fn.apply(null, [this].concat([].slice.call(args)));
this._r = color._r;
this._g = color._g;
this._b = color._b;
this.setAlpha(color._a);
return this;
},
lighten: function lighten() {
return this._applyModification(_lighten, arguments);
},
brighten: function brighten() {
return this._applyModification(_brighten, arguments);
},
darken: function darken() {
return this._applyModification(_darken, arguments);
},
desaturate: function desaturate() {
return this._applyModification(_desaturate, arguments);
},
saturate: function saturate() {
return this._applyModification(_saturate, arguments);
},
greyscale: function greyscale() {
return this._applyModification(_greyscale, arguments);
},
spin: function spin() {
return this._applyModification(_spin, arguments);
},
_applyCombination: function _applyCombination(fn, args) {
return fn.apply(null, [this].concat([].slice.call(args)));
},
analogous: function analogous() {
return this._applyCombination(_analogous, arguments);
},
complement: function complement() {
return this._applyCombination(_complement, arguments);
},
monochromatic: function monochromatic() {
return this._applyCombination(_monochromatic, arguments);
},
splitcomplement: function splitcomplement() {
return this._applyCombination(_splitcomplement, arguments);
},
triad: function triad() {
return this._applyCombination(_triad, arguments);
},
tetrad: function tetrad() {
return this._applyCombination(_tetrad, arguments);
}
}; // If input is an object, force 1 into "1.0" to handle ratios properly
// String input requires "1.0" as input, so 1 will be treated as 1
tinycolor.fromRatio = function (color, opts) {
if (typeof color == "object") {
var newColor = {};
for (var i in color) {
if (color.hasOwnProperty(i)) {
if (i === "a") {
newColor[i] = color[i];
} else {
newColor[i] = convertToPercentage(color[i]);
}
}
}
color = newColor;
}
return tinycolor(color, opts);
}; // Given a string or object, convert that input to RGB
// Possible string inputs:
//
// "red"
// "#f00" or "f00"
// "#ff0000" or "ff0000"
// "#ff000000" or "ff000000"
// "rgb 255 0 0" or "rgb (255, 0, 0)"
// "rgb 1.0 0 0" or "rgb (1, 0, 0)"
// "rgba (255, 0, 0, 1)" or "rgba 255, 0, 0, 1"
// "rgba (1.0, 0, 0, 1)" or "rgba 1.0, 0, 0, 1"
// "hsl(0, 100%, 50%)" or "hsl 0 100% 50%"
// "hsla(0, 100%, 50%, 1)" or "hsla 0 100% 50%, 1"
// "hsv(0, 100%, 100%)" or "hsv 0 100% 100%"
//
function inputToRGB(color) {
var rgb = {
r: 0,
g: 0,
b: 0
};
var a = 1;
var s = null;
var v = null;
var l = null;
var ok = false;
var format = false;
if (typeof color == "string") {
color = stringInputToObject(color);
}
if (typeof color == "object") {
if (isValidCSSUnit(color.r) && isValidCSSUnit(color.g) && isValidCSSUnit(color.b)) {
rgb = rgbToRgb(color.r, color.g, color.b);
ok = true;
format = String(color.r).substr(-1) === "%" ? "prgb" : "rgb";
} else if (isValidCSSUnit(color.h) && isValidCSSUnit(color.s) && isValidCSSUnit(color.v)) {
s = convertToPercentage(color.s);
v = convertToPercentage(color.v);
rgb = hsvToRgb(color.h, s, v);
ok = true;
format = "hsv";
} else if (isValidCSSUnit(color.h) && isValidCSSUnit(color.s) && isValidCSSUnit(color.l)) {
s = convertToPercentage(color.s);
l = convertToPercentage(color.l);
rgb = hslToRgb(color.h, s, l);
ok = true;
format = "hsl";
}
if (color.hasOwnProperty("a")) {
a = color.a;
}
}
a = boundAlpha(a);
return {
ok: ok,
format: color.format || format,
r: mathMin(255, mathMax(rgb.r, 0)),
g: mathMin(255, mathMax(rgb.g, 0)),
b: mathMin(255, mathMax(rgb.b, 0)),
a: a
};
} // Conversion Functions
// --------------------
// `rgbToHsl`, `rgbToHsv`, `hslToRgb`, `hsvToRgb` modified from:
// <http://mjijackson.com/2008/02/rgb-to-hsl-and-rgb-to-hsv-color-model-conversion-algorithms-in-javascript>
// `rgbToRgb`
// Handle bounds / percentage checking to conform to CSS color spec
// <http://www.w3.org/TR/css3-color/>
// *Assumes:* r, g, b in [0, 255] or [0, 1]
// *Returns:* { r, g, b } in [0, 255]
function rgbToRgb(r, g, b) {
return {
r: bound01(r, 255) * 255,
g: bound01(g, 255) * 255,
b: bound01(b, 255) * 255
};
} // `rgbToHsl`
// Converts an RGB color value to HSL.
// *Assumes:* r, g, and b are contained in [0, 255] or [0, 1]
// *Returns:* { h, s, l } in [0,1]
function rgbToHsl(r, g, b) {
r = bound01(r, 255);
g = bound01(g, 255);
b = bound01(b, 255);
var max = mathMax(r, g, b),
min = mathMin(r, g, b);
var h,
s,
l = (max + min) / 2;
if (max == min) {
h = s = 0; // achromatic
} else {
var d = max - min;
s = l > 0.5 ? d / (2 - max - min) : d / (max + min);
switch (max) {
case r:
h = (g - b) / d + (g < b ? 6 : 0);
break;
case g:
h = (b - r) / d + 2;
break;
case b:
h = (r - g) / d + 4;
break;
}
h /= 6;
}
return {
h: h,
s: s,
l: l
};
} // `hslToRgb`
// Converts an HSL color value to RGB.
// *Assumes:* h is contained in [0, 1] or [0, 360] and s and l are contained [0, 1] or [0, 100]
// *Returns:* { r, g, b } in the set [0, 255]
function hslToRgb(h, s, l) {
var r, g, b;
h = bound01(h, 360);
s = bound01(s, 100);
l = bound01(l, 100);
function hue2rgb(p, q, t) {
if (t < 0) t += 1;
if (t > 1) t -= 1;
if (t < 1 / 6) return p + (q - p) * 6 * t;
if (t < 1 / 2) return q;
if (t < 2 / 3) return p + (q - p) * (2 / 3 - t) * 6;
return p;
}
if (s === 0) {
r = g = b = l; // achromatic
} else {
var q = l < 0.5 ? l * (1 + s) : l + s - l * s;
var p = 2 * l - q;
r = hue2rgb(p, q, h + 1 / 3);
g = hue2rgb(p, q, h);
b = hue2rgb(p, q, h - 1 / 3);
}
return {
r: r * 255,
g: g * 255,
b: b * 255
};
} // `rgbToHsv`
// Converts an RGB color value to HSV
// *Assumes:* r, g, and b are contained in the set [0, 255] or [0, 1]
// *Returns:* { h, s, v } in [0,1]
function rgbToHsv(r, g, b) {
r = bound01(r, 255);
g = bound01(g, 255);
b = bound01(b, 255);
var max = mathMax(r, g, b),
min = mathMin(r, g, b);
var h,
s,
v = max;
var d = max - min;
s = max === 0 ? 0 : d / max;
if (max == min) {
h = 0; // achromatic
} else {
switch (max) {
case r:
h = (g - b) / d + (g < b ? 6 : 0);
break;
case g:
h = (b - r) / d + 2;
break;
case b:
h = (r - g) / d + 4;
break;
}
h /= 6;
}
return {
h: h,
s: s,
v: v
};
} // `hsvToRgb`
// Converts an HSV color value to RGB.
// *Assumes:* h is contained in [0, 1] or [0, 360] and s and v are contained in [0, 1] or [0, 100]
// *Returns:* { r, g, b } in the set [0, 255]
function hsvToRgb(h, s, v) {
h = bound01(h, 360) * 6;
s = bound01(s, 100);
v = bound01(v, 100);
var i = Math.floor(h),
f = h - i,
p = v * (1 - s),
q = v * (1 - f * s),
t = v * (1 - (1 - f) * s),
mod = i % 6,
r = [v, q, p, p, t, v][mod],
g = [t, v, v, q, p, p][mod],
b = [p, p, t, v, v, q][mod];
return {
r: r * 255,
g: g * 255,
b: b * 255
};
} // `rgbToHex`
// Converts an RGB color to hex
// Assumes r, g, and b are contained in the set [0, 255]
// Returns a 3 or 6 character hex
function rgbToHex(r, g, b, allow3Char) {
var hex = [pad2(mathRound(r).toString(16)), pad2(mathRound(g).toString(16)), pad2(mathRound(b).toString(16))]; // Return a 3 character hex if possible
if (allow3Char && hex[0].charAt(0) == hex[0].charAt(1) && hex[1].charAt(0) == hex[1].charAt(1) && hex[2].charAt(0) == hex[2].charAt(1)) {
return hex[0].charAt(0) + hex[1].charAt(0) + hex[2].charAt(0);
}
return hex.join("");
} // `rgbaToHex`
// Converts an RGBA color plus alpha transparency to hex
// Assumes r, g, b are contained in the set [0, 255] and
// a in [0, 1]. Returns a 4 or 8 character rgba hex
function rgbaToHex(r, g, b, a, allow4Char) {
var hex = [pad2(mathRound(r).toString(16)), pad2(mathRound(g).toString(16)), pad2(mathRound(b).toString(16)), pad2(convertDecimalToHex(a))]; // Return a 4 character hex if possible
if (allow4Char && hex[0].charAt(0) == hex[0].charAt(1) && hex[1].charAt(0) == hex[1].charAt(1) && hex[2].charAt(0) == hex[2].charAt(1) && hex[3].charAt(0) == hex[3].charAt(1)) {
return hex[0].charAt(0) + hex[1].charAt(0) + hex[2].charAt(0) + hex[3].charAt(0);
}
return hex.join("");
} // `rgbaToArgbHex`
// Converts an RGBA color to an ARGB Hex8 string
// Rarely used, but required for "toFilter()"
function rgbaToArgbHex(r, g, b, a) {
var hex = [pad2(convertDecimalToHex(a)), pad2(mathRound(r).toString(16)), pad2(mathRound(g).toString(16)), pad2(mathRound(b).toString(16))];
return hex.join("");
} // `equals`
// Can be called with any tinycolor input
tinycolor.equals = function (color1, color2) {
if (!color1 || !color2) {
return false;
}
return tinycolor(color1).toRgbString() == tinycolor(color2).toRgbString();
};
tinycolor.random = function () {
return tinycolor.fromRatio({
r: mathRandom(),
g: mathRandom(),
b: mathRandom()
});
}; // Modification Functions
// ----------------------
// Thanks to less.js for some of the basics here
// <https://github.com/cloudhead/less.js/blob/master/lib/less/functions.js>
function _desaturate(color, amount) {
amount = amount === 0 ? 0 : amount || 10;
var hsl = tinycolor(color).toHsl();
hsl.s -= amount / 100;
hsl.s = clamp01(hsl.s);
return tinycolor(hsl);
}
function _saturate(color, amount) {
amount = amount === 0 ? 0 : amount || 10;
var hsl = tinycolor(color).toHsl();
hsl.s += amount / 100;
hsl.s = clamp01(hsl.s);
return tinycolor(hsl);
}
function _greyscale(color) {
return tinycolor(color).desaturate(100);
}
function _lighten(color, amount) {
amount = amount === 0 ? 0 : amount || 10;
var hsl = tinycolor(color).toHsl();
hsl.l += amount / 100;
hsl.l = clamp01(hsl.l);
return tinycolor(hsl);
}
function _brighten(color, amount) {
amount = amount === 0 ? 0 : amount || 10;
var rgb = tinycolor(color).toRgb();
rgb.r = mathMax(0, mathMin(255, rgb.r - mathRound(255 * -(amount / 100))));
rgb.g = mathMax(0, mathMin(255, rgb.g - mathRound(255 * -(amount / 100))));
rgb.b = mathMax(0, mathMin(255, rgb.b - mathRound(255 * -(amount / 100))));
return tinycolor(rgb);
}
function _darken(color, amount) {
amount = amount === 0 ? 0 : amount || 10;
var hsl = tinycolor(color).toHsl();
hsl.l -= amount / 100;
hsl.l = clamp01(hsl.l);
return tinycolor(hsl);
} // Spin takes a positive or negative amount within [-360, 360] indicating the change of hue.
// Values outside of this range will be wrapped into this range.
function _spin(color, amount) {
var hsl = tinycolor(color).toHsl();
var hue = (hsl.h + amount) % 360;
hsl.h = hue < 0 ? 360 + hue : hue;
return tinycolor(hsl);
} // Combination Functions
// ---------------------
// Thanks to jQuery xColor for some of the ideas behind these
// <https://github.com/infusion/jQuery-xcolor/blob/master/jquery.xcolor.js>
function _complement(color) {
var hsl = tinycolor(color).toHsl();
hsl.h = (hsl.h + 180) % 360;
return tinycolor(hsl);
}
function _triad(color) {
var hsl = tinycolor(color).toHsl();
var h = hsl.h;
return [tinycolor(color), tinycolor({
h: (h + 120) % 360,
s: hsl.s,
l: hsl.l
}), tinycolor({
h: (h + 240) % 360,
s: hsl.s,
l: hsl.l
})];
}
function _tetrad(color) {
var hsl = tinycolor(color).toHsl();
var h = hsl.h;
return [tinycolor(color), tinycolor({
h: (h + 90) % 360,
s: hsl.s,
l: hsl.l
}), tinycolor({
h: (h + 180) % 360,
s: hsl.s,
l: hsl.l
}), tinycolor({
h: (h + 270) % 360,
s: hsl.s,
l: hsl.l
})];
}
function _splitcomplement(color) {
var hsl = tinycolor(color).toHsl();
var h = hsl.h;
return [tinycolor(color), tinycolor({
h: (h + 72) % 360,
s: hsl.s,
l: hsl.l
}), tinycolor({
h: (h + 216) % 360,
s: hsl.s,
l: hsl.l
})];
}
function _analogous(color, results, slices) {
results = results || 6;
slices = slices || 30;
var hsl = tinycolor(color).toHsl();
var part = 360 / slices;
var ret = [tinycolor(color)];
for (hsl.h = (hsl.h - (part * results >> 1) + 720) % 360; --results;) {
hsl.h = (hsl.h + part) % 360;
ret.push(tinycolor(hsl));
}
return ret;
}
function _monochromatic(color, results) {
results = results || 6;
var hsv = tinycolor(color).toHsv();
var h = hsv.h,
s = hsv.s,
v = hsv.v;
var ret = [];
var modification = 1 / results;
while (results--) {
ret.push(tinycolor({
h: h,
s: s,
v: v
}));
v = (v + modification) % 1;
}
return ret;
} // Utility Functions
// ---------------------
tinycolor.mix = function (color1, color2, amount) {
amount = amount === 0 ? 0 : amount || 50;
var rgb1 = tinycolor(color1).toRgb();
var rgb2 = tinycolor(color2).toRgb();
var p = amount / 100;
var rgba = {
r: (rgb2.r - rgb1.r) * p + rgb1.r,
g: (rgb2.g - rgb1.g) * p + rgb1.g,
b: (rgb2.b - rgb1.b) * p + rgb1.b,
a: (rgb2.a - rgb1.a) * p + rgb1.a
};
return tinycolor(rgba);
}; // Readability Functions
// ---------------------
// <http://www.w3.org/TR/2008/REC-WCAG20-20081211/#contrast-ratiodef (WCAG Version 2)
// `contrast`
// Analyze the 2 colors and returns the color contrast defined by (WCAG Version 2)
tinycolor.readability = function (color1, color2) {
var c1 = tinycolor(color1);
var c2 = tinycolor(color2);
return (Math.max(c1.getLuminance(), c2.getLuminance()) + 0.05) / (Math.min(c1.getLuminance(), c2.getLuminance()) + 0.05);
}; // `isReadable`
// Ensure that foreground and background color combinations meet WCAG2 guidelines.
// The third argument is an optional Object.
// the 'level' property states 'AA' or 'AAA' - if missing or invalid, it defaults to 'AA';
// the 'size' property states 'large' or 'small' - if missing or invalid, it defaults to 'small'.
// If the entire object is absent, isReadable defaults to {level:"AA",size:"small"}.
// *Example*
// tinycolor.isReadable("#000", "#111") => false
// tinycolor.isReadable("#000", "#111",{level:"AA",size:"large"}) => false
tinycolor.isReadable = function (color1, color2, wcag2) {
var readability = tinycolor.readability(color1, color2);
var wcag2Parms, out;
out = false;
wcag2Parms = validateWCAG2Parms(wcag2);
switch (wcag2Parms.level + wcag2Parms.size) {
case "AAsmall":
case "AAAlarge":
out = readability >= 4.5;
break;
case "AAlarge":
out = readability >= 3;
break;
case "AAAsmall":
out = readability >= 7;
break;
}
return out;
}; // `mostReadable`
// Given a base color and a list of possible foreground or background
// colors for that base, returns the most readable color.
// Optionally returns Black or White if the most readable color is unreadable.
// *Example*
// tinycolor.mostReadable(tinycolor.mostReadable("#123", ["#124", "#125"],{includeFallbackColors:false}).toHexString(); // "#112255"
// tinycolor.mostReadable(tinycolor.mostReadable("#123", ["#124", "#125"],{includeFallbackColors:true}).toHexString(); // "#ffffff"
// tinycolor.mostReadable("#a8015a", ["#faf3f3"],{includeFallbackColors:true,level:"AAA",size:"large"}).toHexString(); // "#faf3f3"
// tinycolor.mostReadable("#a8015a", ["#faf3f3"],{includeFallbackColors:true,level:"AAA",size:"small"}).toHexString(); // "#ffffff"
tinycolor.mostReadable = function (baseColor, colorList, args) {
var bestColor = null;
var bestScore = 0;
var readability;
var includeFallbackColors, level, size;
args = args || {};
includeFallbackColors = args.includeFallbackColors;
level = args.level;
size = args.size;
for (var i = 0; i < colorList.length; i++) {
readability = tinycolor.readability(baseColor, colorList[i]);
if (readability > bestScore) {
bestScore = readability;
bestColor = tinycolor(colorList[i]);
}
}
if (tinycolor.isReadable(baseColor, bestColor, {
"level": level,
"size": size
}) || !includeFallbackColors) {
return bestColor;
} else {
args.includeFallbackColors = false;
return tinycolor.mostReadable(baseColor, ["#fff", "#000"], args);
}
}; // Big List of Colors
// ------------------
// <http://www.w3.org/TR/css3-color/#svg-color>
var names = tinycolor.names = {
aliceblue: "f0f8ff",
antiquewhite: "faebd7",
aqua: "0ff",
aquamarine: "7fffd4",
azure: "f0ffff",
beige: "f5f5dc",
bisque: "ffe4c4",
black: "000",
blanchedalmond: "ffebcd",
blue: "00f",
blueviolet: "8a2be2",
brown: "a52a2a",
burlywood: "deb887",
burntsienna: "ea7e5d",
cadetblue: "5f9ea0",
chartreuse: "7fff00",
chocolate: "d2691e",
coral: "ff7f50",
cornflowerblue: "6495ed",
cornsilk: "fff8dc",
crimson: "dc143c",
cyan: "0ff",
darkblue: "00008b",
darkcyan: "008b8b",
darkgoldenrod: "b8860b",
darkgray: "a9a9a9",
darkgreen: "006400",
darkgrey: "a9a9a9",
darkkhaki: "bdb76b",
darkmagenta: "8b008b",
darkolivegreen: "556b2f",
darkorange: "ff8c00",
darkorchid: "9932cc",
darkred: "8b0000",
darksalmon: "e9967a",
darkseagreen: "8fbc8f",
darkslateblue: "483d8b",
darkslategray: "2f4f4f",
darkslategrey: "2f4f4f",
darkturquoise: "00ced1",
darkviolet: "9400d3",
deeppink: "ff1493",
deepskyblue: "00bfff",
dimgray: "696969",
dimgrey: "696969",
dodgerblue: "1e90ff",
firebrick: "b22222",
floralwhite: "fffaf0",
forestgreen: "228b22",
fuchsia: "f0f",
gainsboro: "dcdcdc",
ghostwhite: "f8f8ff",
gold: "ffd700",
goldenrod: "daa520",
gray: "808080",
green: "008000",
greenyellow: "adff2f",
grey: "808080",
honeydew: "f0fff0",
hotpink: "ff69b4",
indianred: "cd5c5c",
indigo: "4b0082",
ivory: "fffff0",
khaki: "f0e68c",
lavender: "e6e6fa",
lavenderblush: "fff0f5",
lawngreen: "7cfc00",
lemonchiffon: "fffacd",
lightblue: "add8e6",
lightcoral: "f08080",
lightcyan: "e0ffff",
lightgoldenrodyellow: "fafad2",
lightgray: "d3d3d3",
lightgreen: "90ee90",
lightgrey: "d3d3d3",
lightpink: "ffb6c1",
lightsalmon: "ffa07a",
lightseagreen: "20b2aa",
lightskyblue: "87cefa",
lightslategray: "789",
lightslategrey: "789",
lightsteelblue: "b0c4de",
lightyellow: "ffffe0",
lime: "0f0",
limegreen: "32cd32",
linen: "faf0e6",
magenta: "f0f",
maroon: "800000",
mediumaquamarine: "66cdaa",
mediumblue: "0000cd",
mediumorchid: "ba55d3",
mediumpurple: "9370db",
mediumseagreen: "3cb371",
mediumslateblue: "7b68ee",
mediumspringgreen: "00fa9a",
mediumturquoise: "48d1cc",
mediumvioletred: "c71585",
midnightblue: "191970",
mintcream: "f5fffa",
mistyrose: "ffe4e1",
moccasin: "ffe4b5",
navajowhite: "ffdead",
navy: "000080",
oldlace: "fdf5e6",
olive: "808000",
olivedrab: "6b8e23",
orange: "ffa500",
orangered: "ff4500",
orchid: "da70d6",
palegoldenrod: "eee8aa",
palegreen: "98fb98",
paleturquoise: "afeeee",
palevioletred: "db7093",
papayawhip: "ffefd5",
peachpuff: "ffdab9",
peru: "cd853f",
pink: "ffc0cb",
plum: "dda0dd",
powderblue: "b0e0e6",
purple: "800080",
rebeccapurple: "663399",
red: "f00",
rosybrown: "bc8f8f",
royalblue: "4169e1",
saddlebrown: "8b4513",
salmon: "fa8072",
sandybrown: "f4a460",
seagreen: "2e8b57",
seashell: "fff5ee",
sienna: "a0522d",
silver: "c0c0c0",
skyblue: "87ceeb",
slateblue: "6a5acd",
slategray: "708090",
slategrey: "708090",
snow: "fffafa",
springgreen: "00ff7f",
steelblue: "4682b4",
tan: "d2b48c",
teal: "008080",
thistle: "d8bfd8",
tomato: "ff6347",
turquoise: "40e0d0",
violet: "ee82ee",
wheat: "f5deb3",
white: "fff",
whitesmoke: "f5f5f5",
yellow: "ff0",
yellowgreen: "9acd32"
}; // Make it easy to access colors via `hexNames[hex]`
var hexNames = tinycolor.hexNames = flip(names); // Utilities
// ---------
// `{ 'name1': 'val1' }` becomes `{ 'val1': 'name1' }`
function flip(o) {
var flipped = {};
for (var i in o) {
if (o.hasOwnProperty(i)) {
flipped[o[i]] = i;
}
}
return flipped;
} // Return a valid alpha value [0,1] with all invalid values being set to 1
function boundAlpha(a) {
a = parseFloat(a);
if (isNaN(a) || a < 0 || a > 1) {
a = 1;
}
return a;
} // Take input from [0, n] and return it as [0, 1]
function bound01(n, max) {
if (isOnePointZero(n)) {
n = "100%";
}
var processPercent = isPercentage(n);
n = mathMin(max, mathMax(0, parseFloat(n))); // Automatically convert percentage into number
if (processPercent) {
n = parseInt(n * max, 10) / 100;
} // Handle floating point rounding errors
if (Math.abs(n - max) < 0.000001) {
return 1;
} // Convert into [0, 1] range if it isn't already
return n % max / parseFloat(max);
} // Force a number between 0 and 1
function clamp01(val) {
return mathMin(1, mathMax(0, val));
} // Parse a base-16 hex value into a base-10 integer
function parseIntFromHex(val) {
return parseInt(val, 16);
} // Need to handle 1.0 as 100%, since once it is a number, there is no difference between it and 1
// <http://stackoverflow.com/questions/7422072/javascript-how-to-detect-number-as-a-decimal-including-1-0>
function isOnePointZero(n) {
return typeof n == "string" && n.indexOf('.') != -1 && parseFloat(n) === 1;
} // Check to see if string passed in is a percentage
function isPercentage(n) {
return typeof n === "string" && n.indexOf('%') != -1;
} // Force a hex value to have 2 characters
function pad2(c) {
return c.length == 1 ? '0' + c : '' + c;
} // Replace a decimal with it's percentage value
function convertToPercentage(n) {
if (n <= 1) {
n = n * 100 + "%";
}
return n;
} // Converts a decimal to a hex value
function convertDecimalToHex(d) {
return Math.round(parseFloat(d) * 255).toString(16);
} // Converts a hex value to a decimal
function convertHexToDecimal(h) {
return parseIntFromHex(h) / 255;
}
var matchers = function () {
// <http://www.w3.org/TR/css3-values/#integers>
var CSS_INTEGER = "[-\\+]?\\d+%?"; // <http://www.w3.org/TR/css3-values/#number-value>
var CSS_NUMBER = "[-\\+]?\\d*\\.\\d+%?"; // Allow positive/negative integer/number. Don't capture the either/or, just the entire outcome.
var CSS_UNIT = "(?:" + CSS_NUMBER + ")|(?:" + CSS_INTEGER + ")"; // Actual matching.
// Parentheses and commas are optional, but not required.
// Whitespace can take the place of commas or opening paren
var PERMISSIVE_MATCH3 = "[\\s|\\(]+(" + CSS_UNIT + ")[,|\\s]+(" + CSS_UNIT + ")[,|\\s]+(" + CSS_UNIT + ")\\s*\\)?";
var PERMISSIVE_MATCH4 = "[\\s|\\(]+(" + CSS_UNIT + ")[,|\\s]+(" + CSS_UNIT + ")[,|\\s]+(" + CSS_UNIT + ")[,|\\s]+(" + CSS_UNIT + ")\\s*\\)?";
return {
CSS_UNIT: new RegExp(CSS_UNIT),
rgb: new RegExp("rgb" + PERMISSIVE_MATCH3),
rgba: new RegExp("rgba" + PERMISSIVE_MATCH4),
hsl: new RegExp("hsl" + PERMISSIVE_MATCH3),
hsla: new RegExp("hsla" + PERMISSIVE_MATCH4),
hsv: new RegExp("hsv" + PERMISSIVE_MATCH3),
hsva: new RegExp("hsva" + PERMISSIVE_MATCH4),
hex3: /^#?([0-9a-fA-F]{1})([0-9a-fA-F]{1})([0-9a-fA-F]{1})$/,
hex6: /^#?([0-9a-fA-F]{2})([0-9a-fA-F]{2})([0-9a-fA-F]{2})$/,
hex4: /^#?([0-9a-fA-F]{1})([0-9a-fA-F]{1})([0-9a-fA-F]{1})([0-9a-fA-F]{1})$/,
hex8: /^#?([0-9a-fA-F]{2})([0-9a-fA-F]{2})([0-9a-fA-F]{2})([0-9a-fA-F]{2})$/
};
}(); // `isValidCSSUnit`
// Take in a single string / number and check to see if it looks like a CSS unit
// (see `matchers` above for definition).
function isValidCSSUnit(color) {
return !!matchers.CSS_UNIT.exec(color);
} // `stringInputToObject`
// Permissive string parsing. Take in a number of formats, and output an object
// based on detected format. Returns `{ r, g, b }` or `{ h, s, l }` or `{ h, s, v}`
function stringInputToObject(color) {
color = color.replace(trimLeft, '').replace(trimRight, '').toLowerCase();
var named = false;
if (names[color]) {
color = names[color];
named = true;
} else if (color == 'transparent') {
return {
r: 0,
g: 0,
b: 0,
a: 0,
format: "name"
};
} // Try to match string input using regular expressions.
// Keep most of the number bounding out of this function - don't worry about [0,1] or [0,100] or [0,360]
// Just return an object and let the conversion functions handle that.
// This way the result will be the same whether the tinycolor is initialized with string or object.
var match;
if (match = matchers.rgb.exec(color)) {
return {
r: match[1],
g: match[2],
b: match[3]
};
}
if (match = matchers.rgba.exec(color)) {
return {
r: match[1],
g: match[2],
b: match[3],
a: match[4]
};
}
if (match = matchers.hsl.exec(color)) {
return {
h: match[1],
s: match[2],
l: match[3]
};
}
if (match = matchers.hsla.exec(color)) {
return {
h: match[1],
s: match[2],
l: match[3],
a: match[4]
};
}
if (match = matchers.hsv.exec(color)) {
return {
h: match[1],
s: match[2],
v: match[3]
};
}
if (match = matchers.hsva.exec(color)) {
return {
h: match[1],
s: match[2],
v: match[3],
a: match[4]
};
}
if (match = matchers.hex8.exec(color)) {
return {
r: parseIntFromHex(match[1]),
g: parseIntFromHex(match[2]),
b: parseIntFromHex(match[3]),
a: convertHexToDecimal(match[4]),
format: named ? "name" : "hex8"
};
}
if (match = matchers.hex6.exec(color)) {
return {
r: parseIntFromHex(match[1]),
g: parseIntFromHex(match[2]),
b: parseIntFromHex(match[3]),
format: named ? "name" : "hex"
};
}
if (match = matchers.hex4.exec(color)) {
return {
r: parseIntFromHex(match[1] + '' + match[1]),
g: parseIntFromHex(match[2] + '' + match[2]),
b: parseIntFromHex(match[3] + '' + match[3]),
a: convertHexToDecimal(match[4] + '' + match[4]),
format: named ? "name" : "hex8"
};
}
if (match = matchers.hex3.exec(color)) {
return {
r: parseIntFromHex(match[1] + '' + match[1]),
g: parseIntFromHex(match[2] + '' + match[2]),
b: parseIntFromHex(match[3] + '' + match[3]),
format: named ? "name" : "hex"
};
}
return false;
}
function validateWCAG2Parms(parms) {
// return valid WCAG2 parms for isReadable.
// If input parms are invalid, return {"level":"AA", "size":"small"}
var level, size;
parms = parms || {
"level": "AA",
"size": "small"
};
level = (parms.level || "AA").toUpperCase();
size = (parms.size || "small").toLowerCase();
if (level !== "AA" && level !== "AAA") {
level = "AA";
}
if (size !== "small" && size !== "large") {
size = "small";
}
return {
"level": level,
"size": size
};
}
/*// Node: Export function
if (typeof module !== "undefined" && module.exports) {
module.exports = tinycolor;
}
// AMD/requirejs: Define the module
else if (typeof define === 'function' && define.amd) {
define(function () {return tinycolor;});
}
// Browser: Expose to window
else {
window.tinycolor = tinycolor;
}*/
return tinycolor;
}(Math);
/* harmony default export */ __webpack_exports__["default"] = (tinycolor);
/***/ })
}]);
//# sourceMappingURL=addon-default-entry.js.map