(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-devtools/userscript.css": /*!************************************************************************************!*\ !*** ./node_modules/css-loader!./src/addons/addons/editor-devtools/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, "#s3devToolBar {\n display: flex;\n white-space: nowrap;\n font-family: \"Helvetica Neue\", Helvetica, Arial, sans-serif;\n width: 100%;\n}\n\n.s3devLabel {\n display: flex;\n width: 100%;\n}\n\n.s3devLabel > label {\n /* padding instead of margin so clicking on the empty area will select the input */\n padding-left: 1.5em;\n padding-right: 1em;\n font-weight: bold;\n font-size: 0.625rem;\n user-select: none;\n cursor: default;\n white-space: nowrap;\n padding-top: 2px;\n line-height: 2;\n}\n\n[dir=\"rtl\"] .s3devLabel > label {\n padding-right: 1.5em;\n padding-left: 1em;\n}\n\n.s3devAction {\n transition: opacity 200ms ease-in;\n /*font-weight: bold;*/\n opacity: 0;\n}\n\n.s3devLabel:hover .s3devAction {\n opacity: 0.7;\n}\n\n/* Wrapper around find button and drop down */\n.s3devWrap {\n overflow: visible;\n /*margin-left:3em;*/\n position: relative;\n}\n\n#s3devFind {\n height: 2rem;\n width: 100%;\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/* Find Input Box */\ninput.s3devInp {\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\ninput.s3devInp:focus {\n /* Change Scratch default styles */\n box-shadow: none;\n}\n\n/* Drop down from find button */\n.s3devDDOut {\n display: block;\n top: -6px;\n z-index: 100;\n width: 100%;\n max-width: 16em;\n position: relative;\n /*padding: 2.2em 0 0;*/\n /*background-color: white;*/\n padding: 4px;\n}\n\n/* Drop down from find button */\nul.s3devDD {\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}\n\n.s3devDDOut.vis {\n position: absolute;\n width: 16em;\n /*box-shadow: 0 2px 3px rgba(0,0,0,0.3), 0 5px 8px rgba(0,0,0,0.2);*/\n box-shadow: 0px 0px 8px 1px rgba(0, 0, 0, 0.3);\n background-color: white;\n border: none;\n border-radius: 4px;\n}\n[theme=\"dark\"] .s3devDDOut.vis {\n background-color: #111;\n box-shadow: 0px 0px 0px 1px var(--ui-black-transparent);\n}\n\n.s3devDDOut.vis ul.s3devDD {\n display: block;\n border: none;\n}\n\n/* Drop down items */\n.s3devDD > 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.s3devDD > li > b {\n background-color: #aaffaa;\n color: black;\n}\n\n/* Drop down items */\n.s3devDD > li:hover,\n.s3devDD > li.sel {\n color: white;\n cursor: pointer;\n}\n\n#s3devDD > li::before {\n content: \"\\25CF \"; /* ● */\n}\n\n.define {\n color: rgb(255, 102, 128);\n}\n.define:hover,\n.define.sel {\n background-color: rgb(255, 102, 128);\n}\n\n.flag {\n color: #4cbf56;\n}\n.flag:hover,\n.flag.sel {\n background-color: #4cbf56;\n}\n\n.receive {\n color: rgb(255, 191, 0);\n}\n.receive:hover,\n.receive.sel {\n background-color: rgb(255, 191, 0);\n}\n\n.event {\n color: rgb(255, 191, 0);\n}\n.event:hover,\n.event.sel {\n background-color: rgb(255, 191, 0);\n}\n\n.var,\n.VAR {\n color: rgb(255, 120, 26);\n}\n.var:hover,\n.var.sel,\n.VAR:hover,\n.VAR.sel {\n background-color: rgb(255, 120, 26);\n}\n\n.list,\n.LIST {\n color: rgb(255, 102, 26);\n}\n.list:hover,\n.list.sel,\n.LIST:hover,\n.LIST.sel {\n background-color: rgb(255, 102, 26);\n}\n\n.costume {\n color: rgb(26, 66, 255);\n}\n.costume:hover,\n.costume.sel {\n background-color: rgb(26, 66, 255);\n}\n\n.operators {\n background-color: rgb(64, 191, 74);\n color: rgb(56, 148, 56);\n}\n.operators:hover,\n.operators.sel {\n background-color: rgb(56, 148, 56);\n}\n\n.data {\n background-color: rgb(255, 140, 26);\n color: rgb(219, 110, 0);\n}\n.data:hover,\n.data.sel {\n background-color: rgb(219, 110, 0);\n}\n\n.data-lists {\n background-color: rgb(255, 102, 26);\n color: rgb(255, 102, 26);\n}\n.data-lists:hover,\n.data-lists.sel {\n background-color: rgb(207, 81, 20);\n}\n\n.sensing {\n background-color: rgb(76, 191, 230);\n color: rgb(46, 142, 184);\n}\n.sensing:hover,\n.sensing.sel {\n background-color: rgb(46, 142, 184);\n}\n\n.looks {\n background-color: rgb(153, 102, 255);\n color: rgb(119, 77, 203);\n}\n.looks:hover,\n.looks.sel {\n background-color: rgb(119, 77, 203);\n}\n\n.sounds {\n background-color: rgb(214, 92, 214);\n color: rgb(189, 66, 189);\n}\n.sounds:hover,\n.sounds.sel {\n background-color: rgb(189, 66, 189);\n}\n\n.motion {\n background-color: rgb(76, 151, 255);\n color: rgb(51, 115, 204);\n}\n.motion:hover,\n.motion.sel {\n background-color: rgb(51, 115, 204);\n}\n\n.events {\n background-color: rgb(255, 213, 0);\n color: rgb(204, 153, 0);\n}\n.events:hover,\n.events.sel {\n background-color: rgb(204, 153, 0);\n}\n\n.control {\n background-color: rgb(255, 171, 25);\n color: rgb(207, 139, 23);\n}\n.control:hover,\n.control.sel {\n background-color: rgb(207, 139, 23);\n}\n\n.TurboWarp {\n background-color: #ff4c4c;\n color: rgb(207, 139, 23);\n}\n.TurboWarp:hover,\n.TurboWarp.sel {\n background-color: #e64444;\n}\n\n.null {\n background-color: rgb(255, 102, 128);\n color: rgb(255, 77, 106);\n}\n.null:hover,\n.null.sel {\n background-color: rgb(255, 77, 106);\n}\n\n.extension {\n background-color: rgb(15 189 140);\n}\n.extension:hover,\n.extension.sel {\n background-color: rgb(11 158 117);\n}\n\n/*.s3devDD>li.hat::before { content: '◠ ';}*/\n/*.s3devDD>li.hat::before { content: '▲ ';}*/\n/*.s3devDD>li.block::before { content: '■ ';}*/\n/*.s3devDD>li.reporter::before { content: '● ';}*/\n/*.s3devDD>li.boolean::before { content: '◆ ';}*/\n\n#s3devIDD > li {\n height: 19px;\n padding: 3px 8px;\n margin: 2px 0.3em;\n box-sizing: border-box;\n position: relative;\n color: white;\n font-weight: bold;\n width: min-content;\n}\n\n.s3devDD > li.hat {\n border-radius: 14px 14px 3px 3px;\n}\n.s3devDD > li.block {\n border-radius: 3px;\n}\n.s3devDD > li.reporter {\n border-radius: 10px;\n}\n\n.s3devDD > li.boolean {\n width: min-content;\n}\n.s3devDD > li.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 white;\n border-bottom: 10px solid white;\n}\n.s3devDD > li.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 white;\n border-bottom: 10px solid 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#s3devOverlay {\n position: fixed;\n top: 0;\n left: 0;\n right: 0;\n bottom: 0;\n background-color: rgba(80, 80, 80, 0.5);\n z-index: 9999;\n padding: 10em;\n color: white;\n font-weight: bold;\n}\n\n.s3devMulti {\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.s3devNav {\n padding: 0 6px;\n}\n\n.s3devNav:hover {\n color: #ffff80;\n}\n\n.s3devAction {\n margin-left: 2em;\n}\n\n@media screen and (max-width: 1180px) {\n .s3devAction {\n display: none;\n }\n}\n\n.s3devHide {\n display: none;\n}\n\n.s3dev-mi:hover {\n background-color: #d6e9f8;\n}\n[theme=\"dark\"] .s3dev-mi:hover {\n background-color: var(--ui-tertiary);\n}\n\n#s3devHelpPop > div {\n box-sizing: border-box;\n width: 800px;\n height: 60vh;\n max-height: 800px;\n max-width: 85%;\n margin: auto;\n margin-top: 20vh;\n}\n\n#s3devHelpPop [class*=\"modal_header-item-title\"] {\n margin: 0 -20rem 0 0;\n}\n#s3devHelpPop[dir=\"rtl\"] [class*=\"modal_header-item-title\"] {\n margin: 0 0 0 -20rem;\n}\n\n#s3devHelpContent {\n height: calc(100% - 3.125rem);\n overflow-y: auto;\n padding: 1em 2em 4em;\n background-color: white;\n user-select: text;\n}\n[theme=\"dark\"] #s3devHelpPop > div {\n background: #111;\n}\n\n#s3devFloatingBar {\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#s3devInsertLabel {\n padding: 0;\n align-items: center;\n}\n\n/* Drop down from find button */\nul#s3devIDD {\n max-height: 200px;\n margin-bottom: 0;\n}\n\n#s3devIDD > 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/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/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/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-controls-container {\n display: flex;\n flex-direction: row;\n height: calc(1.95rem + 2px);\n}\n\n.sa-onion-button {\n position: relative;\n}\n.sa-onion-button:focus-within {\n background-color: hsla(0, 100%, 65%, 0.2);\n}\n.sa-onion-button[data-enabled=\"true\"] {\n background-color: #ff4c4c;\n}\n\n.sa-onion-settings {\n position: absolute;\n bottom: 0;\n right: 16px;\n /* based on the styles for the color dropdown */\n padding: 4px;\n border-radius: 4px;\n border: 1px solid #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: white;\n color: black;\n}\n.sa-onion-settings[data-visible=\"true\"] {\n bottom: 16px;\n pointer-events: auto;\n opacity: 1;\n}\n[dir=\"rtl\"] .sa-onion-settings {\n right: initial;\n left: 24px;\n}\n.sa-onion-settings .sa-onion-button {\n border-color: #d9d9d9 !important; /* override dark mode */\n}\n\n.sa-onion-settings-line {\n display: flex;\n justify-content: flex-end;\n align-items: baseline;\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: 138px;\n}\n[dir=\"rtl\"] .sa-onion-settings-tip {\n right: 84px;\n}\n.sa-onion-settings-polygon {\n fill: white;\n stroke: #ddd;\n}\n\n[theme=\"dark\"] .sa-onion-image[data-image=\"settings\"] {\n filter: invert(1);\n}\n[theme=\"dark\"] .sa-onion-image[data-image=\"increment\"] {\n filter: none;\n}\n[theme=\"dark\"] .sa-onion-image[data-image=\"decrement\"] {\n filter: none;\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, global, 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; invertCherryPicking = 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 isCherryPicking = isDuplicating ? ctrlOrMetaPressed : enableCherryPicking && 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, global, 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/blockToDom.js": /*!*********************************************************!*\ !*** ./src/addons/addons/block-switching/blockToDom.js ***! \*********************************************************/ /*! exports provided: default */ /***/ (function(module, __webpack_exports__, __webpack_require__) { "use strict"; __webpack_require__.r(__webpack_exports__); /** * This file is imported from https://github.com/LLK/scratch-blocks/ * It has been cleaned up and unnecessary functions have been removed. * Original license is below. * * @license * Visual Blocks Editor * * Copyright 2012 Google Inc. * https://developers.google.com/blockly/ * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ /* eslint-disable */ const INPUT_VALUE = 1; const NEXT_STATEMENT = 3; const DUMMY_INPUT = 5; // Partially implements goog.dom.createDom. const createDom = function createDom(tagName, /* unused */ _params, children) { const element = document.createElement(tagName); if (children !== undefined) { if (!Array.isArray(children)) { children = [children]; } for (const child of children) { if (typeof child === "string") { element.appendChild(document.createTextNode(child)); } else { element.appendChild(child); } } } return element; }; // Partially implements goog.dom.removeNode const removeNode = node => { if (node.parentNode) { node.parentNode.removeChild(node); } }; const fieldToDomVariable_ = function fieldToDomVariable_(field) { // The field had not been initialized fully before being serialized. // This can happen if a block is created directly through a call to // workspace.newBlock instead of from XML. // The new block will be serialized for the first time when firing a block // creation event. if (field.getValue() == null) { field.initModel(); } // Get the variable directly from the field, instead of doing a lookup. This // will work even if the variable has already been deleted. This can happen // because the flyout defers deleting blocks until the next time the flyout is // opened. var variable = field.getVariable(); if (!variable) { throw Error("Tried to serialize a variable field with no variable."); } var container = createDom("field", null, variable.name); container.setAttribute("name", field.name); container.setAttribute("id", variable.getId()); container.setAttribute("variabletype", variable.type); return container; }; const fieldToDom_ = function fieldToDom_(field) { if (field.name && field.SERIALIZABLE) { if (field.referencesVariables()) { return fieldToDomVariable_(field); } else { var container = createDom("field", null, field.getValue()); container.setAttribute("name", field.name); return container; } } return null; }; const allFieldsToDom_ = function allFieldsToDom_(block, element) { for (var i = 0, input; input = block.inputList[i]; i++) { for (var j = 0, field; field = input.fieldRow[j]; j++) { var fieldDom = fieldToDom_(field); if (fieldDom) { element.appendChild(fieldDom); } } } }; const blockToDom = function blockToDom(block, opt_noId) { var element = createDom(block.isShadow() ? "shadow" : "block"); element.setAttribute("type", block.type); if (!opt_noId) { element.setAttribute("id", block.id); } if (block.mutationToDom) { // Custom data for an advanced block. var mutation = block.mutationToDom(); if (mutation && (mutation.hasChildNodes() || mutation.hasAttributes())) { element.appendChild(mutation); } } allFieldsToDom_(block, element); scratchCommentToDom_(block, element); if (block.data) { var dataElement = createDom("data", null, block.data); element.appendChild(dataElement); } for (var i = 0, input; input = block.inputList[i]; i++) { var container; var empty = true; if (input.type == DUMMY_INPUT) { continue; } else { var childBlock = input.connection.targetBlock(); if (input.type == INPUT_VALUE) { container = createDom("value"); } else if (input.type == NEXT_STATEMENT) { container = createDom("statement"); } var shadow = input.connection.getShadowDom(); if (shadow && (!childBlock || !childBlock.isShadow())) { var shadowClone = cloneShadow_(shadow); // Remove the ID from the shadow dom clone if opt_noId // is specified to true. if (opt_noId && shadowClone.getAttribute("id")) { shadowClone.removeAttribute("id"); } container.appendChild(shadowClone); } if (childBlock) { container.appendChild(blockToDom(childBlock, opt_noId)); empty = false; } } container.setAttribute("name", input.name); if (!empty) { element.appendChild(container); } } if (block.inputsInlineDefault != block.inputsInline) { element.setAttribute("inline", block.inputsInline); } if (block.isCollapsed()) { element.setAttribute("collapsed", true); } if (block.disabled) { element.setAttribute("disabled", true); } if (!block.isDeletable() && !block.isShadow()) { element.setAttribute("deletable", false); } if (!block.isMovable() && !block.isShadow()) { element.setAttribute("movable", false); } if (!block.isEditable()) { element.setAttribute("editable", false); } var nextBlock = block.getNextBlock(); if (nextBlock) { var container = createDom("next", null, blockToDom(nextBlock, opt_noId)); element.appendChild(container); } var shadow = block.nextConnection && block.nextConnection.getShadowDom(); if (shadow && (!nextBlock || !nextBlock.isShadow())) { container.appendChild(cloneShadow_(shadow)); } return element; }; const scratchCommentToDom_ = function scratchCommentToDom_(block, element) { var commentText = block.getCommentText(); if (commentText) { var commentElement = createDom("comment", null, commentText); if (typeof block.comment == "object") { commentElement.setAttribute("id", block.comment.id); commentElement.setAttribute("pinned", block.comment.isVisible()); var hw; // TODO: scratch-blocks uses `block.comment instanceof Blockly.ScratchBlockComment` if (block.comment.getHeightWidth) { hw = block.comment.getHeightWidth(); } else { hw = block.comment.getBubbleSize(); } commentElement.setAttribute("h", hw.height); commentElement.setAttribute("w", hw.width); var xy = block.comment.getXY(); commentElement.setAttribute("x", xy.x); commentElement.setAttribute("y", xy.y); commentElement.setAttribute("minimized", block.comment.isMinimized()); } element.appendChild(commentElement); } }; const cloneShadow_ = function cloneShadow_(shadow) { shadow = shadow.cloneNode(true); // Walk the tree looking for whitespace. Don't prune whitespace in a tag. var node = shadow; var textNode; while (node) { if (node.firstChild) { node = node.firstChild; } else { while (node && !node.nextSibling) { textNode = node; node = node.parentNode; if (textNode.nodeType == 3 && textNode.data.trim() == "" && node.firstChild != textNode) { // Prune whitespace after a tag. removeNode(textNode); } } if (node) { textNode = node; node = node.nextSibling; if (textNode.nodeType == 3 && textNode.data.trim() == "") { // Prune whitespace before a tag. removeNode(textNode); } } } } return shadow; }; const blockToDomWithXY = block => { const xml = blockToDom(block, false); const position = block.getRelativeToSurfaceXY(); xml.setAttribute("x", Math.round(block.workspace.RTL ? -position.x : position.x)); xml.setAttribute("y", Math.round(position.y)); return xml; }; /* harmony default export */ __webpack_exports__["default"] = (blockToDomWithXY); /***/ }), /***/ "./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 import */ var _blockToDom_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./blockToDom.js */ "./src/addons/addons/block-switching/blockToDom.js"); /* harmony default export */ __webpack_exports__["default"] = (async function ({ addon, global, console, msg }) { const ScratchBlocks = await addon.tab.traps.getBlockly(); const vm = addon.tab.traps.vm; let blockSwitches = {}; let procedureSwitches = {}; const noopSwitch = { isNoop: true }; 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", remap: { X: "DX" } }, { opcode: "motion_sety", remap: { X: "Y" } }, { opcode: "motion_changeyby", remap: { X: "DY" } }]; blockSwitches["motion_changexby"] = [{ opcode: "motion_setx", remap: { DX: "X" } }, noopSwitch, { opcode: "motion_sety", remap: { DX: "Y" } }, { opcode: "motion_changeyby", remap: { DX: "DY" } }]; blockSwitches["motion_sety"] = [{ opcode: "motion_setx", remap: { Y: "X" } }, { opcode: "motion_changexby", remap: { Y: "DX" } }, noopSwitch, { opcode: "motion_changeyby", remap: { Y: "DY" } }]; blockSwitches["motion_changeyby"] = [{ opcode: "motion_setx", remap: { DY: "X" } }, { opcode: "motion_changexby", remap: { DY: "DX" } }, { opcode: "motion_sety", remap: { 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", remap: { VALUE: "CHANGE" } }]; blockSwitches["looks_changeeffectby"] = [{ opcode: "looks_seteffectto", remap: { CHANGE: "VALUE" } }, noopSwitch]; blockSwitches["looks_setsizeto"] = [noopSwitch, { opcode: "looks_changesizeby", remap: { SIZE: "CHANGE" } }]; blockSwitches["looks_changesizeby"] = [{ opcode: "looks_setsizeto", remap: { 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_think"] = [noopSwitch, { opcode: "looks_say" }]; blockSwitches["looks_say"] = [{ opcode: "looks_think" }, noopSwitch]; blockSwitches["looks_thinkforsecs"] = [noopSwitch, { opcode: "looks_sayforsecs" }]; blockSwitches["looks_sayforsecs"] = [{ opcode: "looks_thinkforsecs" }, noopSwitch]; blockSwitches["looks_switchbackdropto"] = [noopSwitch, { opcode: "looks_switchbackdroptoandwait" }]; blockSwitches["looks_switchbackdroptoandwait"] = [{ opcode: "looks_switchbackdropto" }, 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", remap: { SUBSTACK2: "split" } }, noopSwitch]; blockSwitches["control_repeat_until"] = [noopSwitch, { opcode: "control_wait_until", remap: { SUBSTACK: "split" } }, { opcode: "control_forever", remap: { CONDITION: "split" } }]; 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]; } if (addon.settings.get("data")) { blockSwitches["data_setvariableto"] = [noopSwitch, { opcode: "data_changevariableby", remapValueType: { VALUE: "math_number" } }]; blockSwitches["data_changevariableby"] = [{ opcode: "data_setvariableto", remapValueType: { 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]; } 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); const menuCallbackFactory = (block, opcodeData) => () => { if (opcodeData.isNoop) { return; } if (opcodeData.fieldValue) { block.setFieldValue(opcodeData.fieldValue, "VALUE"); return; } const workspace = block.workspace; // 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 = Object(_blockToDom_js__WEBPACK_IMPORTED_MODULE_0__["default"])(block); if (opcodeData.opcode) { xml.setAttribute("type", opcodeData.opcode); } const id = block.id; const parent = block.getParent(); let parentConnection; let blockConnectionType; if (parent) { // If the block has a parent, find the parent -> child connection that will be reattached later. const parentConnections = parent.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_ === parent); blockConnectionType = blockToParentConnection.type; } const pasteSeparately = []; // Apply input remappings. if (opcodeData.remap) { for (const child of Array.from(xml.children)) { const oldName = child.getAttribute("name"); const newName = opcodeData.remap[oldName]; if (newName) { if (newName === "split") { // This input will be split off into its own script. const inputXml = child.firstChild; const inputId = inputXml.id; const inputBlock = workspace.getBlockById(inputId); const position = inputBlock.getRelativeToSurfaceXY(); inputXml.setAttribute("x", Math.round(workspace.RTL ? -position.x : position.x)); inputXml.setAttribute("y", Math.round(position.y)); pasteSeparately.push(inputXml); xml.removeChild(child); } else { child.setAttribute("name", newName); } } } } if (opcodeData.remapValueType) { for (const child of Array.from(xml.children)) { const name = child.getAttribute("name"); const newType = opcodeData.remapValueType[name]; if (newType) { const valueNode = child.firstChild; const fieldNode = valueNode.firstChild; valueNode.setAttribute("type", newType); fieldNode.setAttribute("name", newType === "text" ? "TEXT" : "NUM"); } } } if (opcodeData.mutate) { const mutation = xml.querySelector("mutation"); for (const [key, value] of Object.entries(opcodeData.mutate)) { mutation.setAttribute(key, value); } } try { ScratchBlocks.Events.setGroup(true); // Remove the old block and insert the new one. block.dispose(); workspace.paste(xml); for (const separateBlock of pasteSeparately) { workspace.paste(separateBlock); } // The new block will have the same ID as the old one. const newBlock = workspace.getBlockById(id); 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); } } 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: addon.settings.get("border") && i === 0 }); }); if (addon.settings.get("border") && (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, global, 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/BlockInstance.js": /*!************************************************************!*\ !*** ./src/addons/addons/editor-devtools/BlockInstance.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 BlockInstance; }); /** * Encapsulates a block (either in this sprite or another / Blockly, or native JSON block */ class BlockInstance { constructor(target, block) { this.targetId = target.id; this.id = block.id; } } /***/ }), /***/ "./src/addons/addons/editor-devtools/BlockItem.js": /*!********************************************************!*\ !*** ./src/addons/addons/editor-devtools/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.} */ 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/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 _blockly_Utils_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./blockly/Utils.js */ "./src/addons/addons/editor-devtools/blockly/Utils.js"); /* harmony import */ var _DomHelpers_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./DomHelpers.js */ "./src/addons/addons/editor-devtools/DomHelpers.js"); /* harmony import */ var _BlockInstance_js__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./BlockInstance.js */ "./src/addons/addons/editor-devtools/BlockInstance.js"); /* harmony import */ var _XML_js__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./XML.js */ "./src/addons/addons/editor-devtools/XML.js"); /* harmony import */ var _BlockItem_js__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ./BlockItem.js */ "./src/addons/addons/editor-devtools/BlockItem.js"); /* harmony import */ var _blockly_UndoGroup_js__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! ./blockly/UndoGroup.js */ "./src/addons/addons/editor-devtools/blockly/UndoGroup.js"); // import ShowBroadcast from "./show-broadcast.js"; class DevTools { constructor(addon, msg, m, helpHTML) { this.addon = addon; this.msg = msg; this.m = m; /** * @type {VirtualMachine} */ this.vm = addon.tab.traps.vm; this.utils = new _blockly_Utils_js__WEBPACK_IMPORTED_MODULE_0__["default"](addon); this.domHelpers = new _DomHelpers_js__WEBPACK_IMPORTED_MODULE_1__["default"](addon); this.multi = new Multi(this.utils); this._helpHTML = helpHTML; this.find = null; this.findInp = null; this.ddOut = null; this.dd = null; this.codeTab = null; this.costTab = null; this.costTabBody = null; this.selVarID = null; this.floatInp = null; this.blockCursor = 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: () => { let wksp = this.utils.getWorkspace(); let v = wksp.getVariableById(this.selVarID); let varName = 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 }); } isScriptEditor() { return this.codeTab.className.indexOf("gui_is-selected") >= 0; } isCostumeEditor() { return this.costTab.className.indexOf("gui_is-selected") >= 0; } eventClickHelp(e) { if (!document.getElementById("s3devHelpPop")) { document.body.insertAdjacentHTML("beforeend", this._helpHTML); document.getElementById("s3devHelpPop").addEventListener("mousedown", function (e) { if (e.target.id === "s3devHelpPop") { e.target.remove(); } }); document.querySelector("#s3devHelpPop .close-button").addEventListener("click", function (e) { document.getElementById("s3devHelpPop").remove(); }); } e.preventDefault(); } getScratchCostumes() { let costumes = this.costTabBody.querySelectorAll("div[class^='sprite-selector-item_sprite-name']"); // this.costTab[0].click(); let myBlocks = []; let myBlocksByProcCode = {}; /** * @param cls * @param txt * @param root * @returns BlockItem */ function addBlock(cls, txt, root) { let id = root.className; let items = new _BlockItem_js__WEBPACK_IMPORTED_MODULE_4__["default"](cls, txt, id, 0); myBlocks.push(items); myBlocksByProcCode[txt] = items; return items; } let i = 0; for (const costume of costumes) { addBlock("costume", costume.innerText, costume).y = i; i++; } return { procs: myBlocks }; } /** * Fetch the scratch 3 block list * @returns jsonFetch object */ getScratchBlocks() { // Access Blockly! let myBlocks = []; let myBlocksByProcCode = {}; // todo - get blockly from an svg??? let wksp = this.utils.getWorkspace(); let topBlocks = wksp.getTopBlocks(); // console.log(topBlocks); /** * @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 _BlockItem_js__WEBPACK_IMPORTED_MODULE_4__["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") { try { // let wksp2 = Blockly.getMainWorkspace().getTopBlocks()[2].inputList[0].fieldRow[1]; let fields = root.inputList[0]; // let typeDesc = fields.fieldRow[0].getText(); let eventName = fields.fieldRow[1].getText(); // addBlock('receive', typeDesc + ' ' + eventName, root).eventName = eventName; addBlock("receive", "event " + eventName, root).eventName = eventName; } catch (e) {// eat } 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 = wksp.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 runtime = vm.runtime; // Now let's also add event broadcasts (not just hat blocks) const target = runtime.getEditingTarget(); const blocks = target.blocks; if (blocks._blocks) { for (const id in blocks._blocks) { if (!blocks._blocks.hasOwnProperty(id)) { continue; } const block = blocks._blocks[id]; if (block.opcode === 'event_broadcast_menu') { debugger; // Now get the parent block that is the actual broadcast or broadcast and wait const broadcastBlock = blocks.getBlock(block.parent); let b = new BlockInstance(target, broadcastBlock); const fieldName = block.fields.BROADCAST_OPTION.value; addBlock('receive', 'event ' + fieldName, b).eventName = broadcastBlock.eventName; // // let fields = root.inputList[0]; // let typeDesc = fields.fieldRow[0].getText(); // let eventName = fields.fieldRow[1].getText(); // addBlock('receive', 'event ' + eventName, root).eventName = 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 { procs: myBlocks }; } showDropDown(e, focusID, instanceBlock) { clearTimeout(rhdd); rhdd = 0; if (!focusID && this.ddOut.classList.contains("vis")) { return; } // special '' vs null... - null forces a reevaluation prevVal = focusID ? "" : null; // Clear the previous value of the input search this.ddOut.classList.add("vis"); let scratchBlocks = this.isCostumeEditor() ? this.getScratchCostumes() : this.getScratchBlocks(); this.dom_removeChildren(this.dd); let foundLi = null; /** * @type {[BlockItem]} */ const procs = scratchBlocks.procs; for (const proc of procs) { let li = document.createElement("li"); li.innerText = proc.procCode; li.data = proc; li.className = proc.cls; if (focusID) { if (proc.matchesID(focusID)) { foundLi = li; li.classList.add("sel"); } else { li.style.display = "none"; } } this.dd.appendChild(li); } let label = document.getElementById("s3devFindLabel"); this.utils.offsetX = this.ddOut.getBoundingClientRect().right - label.getBoundingClientRect().left + 26; this.utils.offsetY = 32; if (foundLi) { this.clickDropDownRow(foundLi, this.utils.getWorkspace(), instanceBlock); } } hideDropDown() { clearTimeout(rhdd); rhdd = setTimeout(() => this.reallyHideDropDown(), 250); } reallyHideDropDown() { // Check focus of find box if (this.findInp === document.activeElement) { this.hideDropDown(); return; } // document.getElementById('s3devReplace').classList.add('s3devHide'); this.ddOut.classList.remove("vis"); rhdd = 0; } hideFloatDropDown() { clearTimeout(rhdd2); rhdd2 = setTimeout(() => this.reallyHideFloatDropDown(), 50); } reallyHideFloatDropDown(force) { // Check focus of find box if (!force && this.floatInp === document.activeElement) { this.hideFloatDropDown(); return; } let float = document.getElementById("s3devFloatingBar"); if (float) { float.remove(); } this.floatInp = null; rhdd2 = 0; } dom_removeChildren(myNode) { while (myNode.firstChild) { myNode.removeChild(myNode.firstChild); } } /** * 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.utils.getWorkspace(); let makeSpaceForBlock = block && block.getRootBlock(); _blockly_UndoGroup_js__WEBPACK_IMPORTED_MODULE_5__["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.utils.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()); } } } _blockly_UndoGroup_js__WEBPACK_IMPORTED_MODULE_5__["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.utils.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.} 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} */ let kids = topBlock.getDescendants(); for (const block of kids) { /** @type {!Array} */ let blockVariables = block.getVarModels(); if (blockVariables) { for (const blockVar of blockVariables) { if (blockVar.getId() === id) { uses.push(block); } } } } } return uses; } /** * Find all the uses of a named procedure. * @param {string} id ID of the variable to find. * @return {!Array.} Array of block usages. */ getCallsToProcedureById(id) { let w = this.utils.getWorkspace(); let procBlock = w.getBlockById(id); let label = procBlock.getChildren()[0]; let procCode = label.getProcCode(); let uses = [procBlock]; // Definition First, then calls to it let topBlocks = this.getTopBlocks(true); for (const topBlock of topBlocks) { /** @type {!Array} */ let kids = topBlock.getDescendants(); for (const block of kids) { if (block.type === "procedures_call") { if (block.getProcCode() === procCode) { uses.push(block); } } } } return uses; } /** * Find all the uses of a named procedure. * @param {string} name name of the variable to find. * @return {!Array.} Array of block usages. */ getCallsToEventsByName(name) { let uses = []; // Definition First, then calls to it const runtime = this.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]; // To find event broadcaster blocks, we look for the nested "event_broadcast_menu" blocks first that match the event name if (block.opcode === "event_broadcast_menu" && block.fields.BROADCAST_OPTION.value === name) { // Now get the parent block that is the actual broadcast or broadcast and wait const broadcastBlock = blocks.getBlock(block.parent); uses.push(new _BlockInstance_js__WEBPACK_IMPORTED_MODULE_2__["default"](target, broadcastBlock)); } else if (block.opcode === "event_whenbroadcastreceived" && block.fields.BROADCAST_OPTION.value === name) { uses.push(new _BlockInstance_js__WEBPACK_IMPORTED_MODULE_2__["default"](target, block)); } } } return uses; } /** * Find all the event broadcasters. * @return {[{eventName:string, block:Block}]} Array of event names and blocks. */ getCallsToEvents() { const uses = []; // Definition First, then calls to it const found = {}; let topBlocks = this.getTopBlocks(true); for (const topBlock of topBlocks) { /** @type {!Array} */ let kids = topBlock.getDescendants(); for (const block of kids) { if (block.type === "event_broadcast" || block.type === "event_broadcastandwait") { const eventName = block.getChildren()[0].inputList[0].fieldRow[0].getText(); if (!found[eventName]) { found[eventName] = block; uses.push({ eventName: eventName, block: block }); } } } } return uses; } buildNavigationCarousel(nav, li, blocks, instanceBlock) { if (nav && nav.parentNode === li) { // Same control... click again to go to next this.multi.navRight(); } else { if (nav) { nav.remove(); } li.insertAdjacentHTML("beforeend", "\n \n \u25C0\u25B6\n \n "); document.getElementById("s3devMultiLeft").addEventListener("mousedown", (...e) => this.multi.navLeft(...e)); document.getElementById("s3devMultiRight").addEventListener("mousedown", (...e) => this.multi.navRight(...e)); this.multi.idx = 0; if (instanceBlock) { for (let i = 0; i < blocks.length; i++) { const block = blocks[i]; if (block.id === instanceBlock.id) { this.multi.idx = i; break; } } // multi.idx = blocks.indexOf(instanceBlock); } this.multi.blocks = blocks; this.multi.update(); if (this.multi.idx < blocks.length) { this.centerTop(blocks[this.multi.idx]); } } } /** * Move a costume to the top or bottom of the list * @param top true for the top, false for the bottom * @param selected optional parameter to pass in the costume div to be moved */ moveCostumeTo(top, selected) { let isSelected = !selected || selected.className.indexOf("sprite-selector-item_is-selected") >= 0; if (!selected) { selected = this.costTabBody.querySelectorAll("div[class*='sprite-selector-item_is-selected']"); if (selected.length === 0) { return; } selected = selected[0].querySelectorAll("div[class^='sprite-selector-item_sprite-name']")[0]; } let costumes = this.costTabBody.querySelectorAll("div[class^='sprite-selector-item_sprite-name']"); // First scroll sprite view to reveal top or bottom otherwise this won't work. let scroller = selected.closest("div[class*=selector_list-area]"); let lastScroll = scroller.scrollTop; scroller.scrollTop = top ? 0 : scroller.scrollHeight; this.domHelpers.triggerDragAndDrop(selected, costumes[top ? 0 : costumes.length - 1], undefined); if (!isSelected) { // Restore Scroll position scroller.scrollTop = lastScroll; } } /** * * @param li * @param workspace * @param instanceBlock the instance to be highlighted (or null) */ clickDropDownRow(li, workspace, instanceBlock) { let nav = document.getElementById("s3devMulti"); let cls = li.data.cls; if (cls === "costume") { // Viewing costumes - jump to selected costume let costumes = this.costTabBody.querySelectorAll("div[class^='sprite-selector-item_sprite-name']"); let costume = costumes[li.data.y]; if (costume) { costume.click(); setTimeout(() => { let wrapper = costume.closest("div[class*=gui_flex-wrapper]"); costume.parentElement.parentElement.scrollIntoView({ behavior: "auto", block: "center", inline: "start" }); wrapper.scrollTop = 0; }, 10); } } else if (cls === "var" || cls === "VAR" || cls === "list" || cls === "LIST") { // Search now for all instances // let wksp = getWorkspace(); // let blocks = wksp.getVariableUsesById(li.data.labelID); let blocks = this.getVariableUsesById(li.data.labelID); this.buildNavigationCarousel(nav, li, blocks, instanceBlock); } else if (cls === "define") { let blocks = this.getCallsToProcedureById(li.data.labelID); this.buildNavigationCarousel(nav, li, blocks, instanceBlock); } else if (cls === "receive") { /* let blocks = [workspace.getBlockById(li.data.labelID)]; if (li.data.clones) { for (const cloneID of li.data.clones) { blocks.push(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(li.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.buildNavigationCarousel(nav, li, blocks, instanceBlock); } else if (li.data.clones) { let blocks = [workspace.getBlockById(li.data.labelID)]; for (const cloneID of li.data.clones) { blocks.push(workspace.getBlockById(cloneID)); } this.buildNavigationCarousel(nav, li, blocks, instanceBlock); } else { this.multi.blocks = null; this.centerTop(li.data.labelID); if (nav) { nav.remove(); } } } dropDownClick(e) { // console.log(e); let workspace = this.utils.getWorkspace(); if (prevVal === null) { prevVal = this.findInp.value; // Hack to stop filter change if not entered data into edt box, but clicked on row } let li = e.target; for (;;) { if (!li || li === this.dd) { return; } if (li.data) { break; } li = li.parentNode; } // If this was a mouse click, unselect the keyboard selection // e.navKey is set when this is called from the keyboard handler... if (!e.navKey) { let sel = this.dd.getElementsByClassName("sel"); sel = sel.length > 0 ? sel[0] : null; if (sel && sel !== li) { try { sel.classList.remove("sel"); } catch (e) { console.log(sel); console.error(e); } } if (li !== sel) { li.classList.add("sel"); } } this.clickDropDownRow(li, workspace); if (e && e.preventDefault) { e.preventDefault(); e.cancelBubble = true; } return false; } /** * Based on wksp.centerOnBlock(li.data.labelID); * @param e * @param force if true, the view always moves, otherwise only move if the selected element is not entirely visible */ centerTop(e, force) { this.utils.scrollBlockIntoView(e, force); } inputChange(e) { if (!this.ddOut.classList.contains("vis")) { this.showDropDown(); this.hideDropDown(); // Start timer to hide if not got focus } // Filter the list... let val = (this.findInp.value || "").toLowerCase(); if (val === prevVal) { // No change so don't re-filter return; } prevVal = val; this.multi.blocks = null; // // Hide items in list that do not contain filter text, and highlight the parts of words that match the filter text let listLI = this.dd.getElementsByTagName("li"); for (const li of listLI) { let procCode = li.data.procCode; let i = li.data.lower.indexOf(val); if (i >= 0) { li.style.display = "block"; this.dom_removeChildren(li); 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))); } // li.innerHTML = enc(procCode.substring(0, i)) + '' + enc(procCode.substr(i, val.length)) + "" + enc(procCode.substr(i + val.length)); } else { li.style.display = "none"; } } } /** * Select previous or next item in the drop down filter list * @param dir direction of navigation: -1=up, 1=down */ navigateFilter(dir) { let sel = this.dd.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.dd.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"); this.dropDownClick({ target: nxt, navKey: true }); // centerTop(nxt.data.labelID); } } 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; } // Left Arrow if (e.keyCode === 37) { let sel = this.dd.getElementsByClassName("sel"); if (sel && this.multi.blocks) { this.multi.navLeft(e); } } // Right Arrow if (e.keyCode === 39) { let sel = this.dd.getElementsByClassName("sel"); if (sel && this.multi.blocks) { this.multi.navRight(e); } } // Enter if (e.keyCode === 13) { // Any selected on enter? if not select now let sel = this.dd.getElementsByClassName("sel"); if (sel.length === 0) { this.navigateFilter(1); } // noinspection JSUnresolvedFunction document.activeElement.blur(); e.preventDefault(); return; } // Escape if (e.keyCode === 27) { if (this.findInp.value.length > 0) { this.findInp.value = ""; // Clear search first, then close on second press this.inputChange(e); } else { // noinspection JSUnresolvedFunction document.activeElement.blur(); } e.preventDefault(); return; } } /* function deepSearch(e) { document.body.insertAdjacentHTML( "beforeend", `
` ); let overlay = document.getElementById("s3devOverlay"); overlay.addEventListener("click", function (e) { overlay.remove(); }); // todo: use scratch runtime instead! const runtime = vm.runtime; const targets = runtime.targets; // The sprites / stage let dict = {}; for (const target of targets) { if (!target.isOriginal) { continue; // Skip clones } const name = target.getName(); const isStage = target.isStage; const blocks = target.blocks; const scripts = blocks.getScripts(); let divElement = document.createElement("div"); divElement.appendChild(document.createTextNode("Searching in " + name)); divElement.appendChild(document.createTextNode(", Top Scripts x" + scripts.length)); overlay.appendChild(divElement); const sprite = (dict[name] = {}); sprite.scripts = []; for (const script of scripts) { const block = blocks.getBlock(script); const top = { id: script, opcode: block.opcode, block: block }; sprite.scripts.push(top); } } e.preventDefault(); return true; } */ /** * 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.utils.getWorkspace(); let v = wksp.getVariable(newVarName, type); if (!v) { alert(this.msg("var-not-exist")); return; } let newVId = v.getId(); _blockly_UndoGroup_js__WEBPACK_IMPORTED_MODULE_5__["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 } } _blockly_UndoGroup_js__WEBPACK_IMPORTED_MODULE_5__["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} Set of top blocks */ getTopBlockIDs() { let wksp = this.utils.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.utils.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) { function 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.key === "f" && ctrlKey && !e.shiftKey) { // Ctrl + F (Override default Ctrl+F find) this.findInp.focus(); this.findInp.select(); e.cancelBubble = true; e.preventDefault(); return true; } if (e.key === " " && ctrlKey) { // Ctrl + Space (Inject Code) this.middleClickWorkspace(e); e.cancelBubble = true; e.preventDefault(); return true; } if (e.keyCode === 37 && ctrlKey) { // Ctrl + Left Arrow Key if (document.activeElement.tagName === "INPUT") { return; } // todo: if (!this.addon.settings.get("enableCtrlLeftRightNav")) { // return; // } if (this.isScriptEditor()) { this.utils.navigationHistory.goBack(); } else 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; } // todo: if (!this.addon.settings.get("enableCtrlLeftRightNav")) { // return; // } if (this.isScriptEditor()) { this.utils.navigationHistory.goForward(); } else 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.utils.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) { _blockly_UndoGroup_js__WEBPACK_IMPORTED_MODULE_5__["default"].startUndoGroup(wksp); block.dispose(true); _blockly_UndoGroup_js__WEBPACK_IMPORTED_MODULE_5__["default"].endUndoGroup(wksp); } }, 0); } } } eventMouseDown(e) { this.updateMousePosition(e); if (this.ddOut && this.ddOut.classList.contains("vis") && !e.target.closest("#s3devDDOut")) { // If we click outside the dropdown, then instigate the hide code... this.hideDropDown(); } if (this.floatInp && !e.target.closest("#s3devIDDOut")) { if (!e.shiftKey || !document.getElementsByClassName("injectionDiv")[0].contains(e.target) && // Focused inputs are not part of the injectionDiv, but to the user they are part of the code area so make multi-inject work there !e.target.classList.contains("blocklyHtmlInput") || // This selector targets workspace buttons (Make a Block etc.) and the extension (!) buttons, which most commonly trigger a popup window so always close the dropdown e.target.matches(".blocklyFlyoutButton, .blocklyFlyoutButton *, .blocklyTouchTargetBackground")) { // If we click outside the dropdown, then instigate the hide code... this.hideFloatDropDown(); } } if (e.button === 1 || e.shiftKey) { // Wheel button... try { this.middleClick(e); } catch (x) { console.error(x); } } else if (e.button === 2) {// Right click... /* let spriteSelector = e.target.closest("#react-tabs-3 div[class*='sprite-selector-item_sprite-selector-item']"); if (spriteSelector) { let contextMenu = spriteSelector.getElementsByTagName("nav")[0]; if (!contextMenu.querySelector("div.s3devSTT")) { contextMenu.insertAdjacentHTML( "beforeend", ` ` ); } } */ } else { let chk = e.target; if (chk && chk.tagName !== "BUTTON" && chk.getAttribute && !chk.getAttribute("role")) { chk = chk.parentNode; if (chk && chk.tagName !== "BUTTON" && chk.getAttribute && !chk.getAttribute("role")) { chk = chk.parentNode; } } if (chk && chk.className && chk.className.indexOf) { if (!this.canShare && chk.className.indexOf("share-button") >= 0) { // Commented for ScratchAddons /*e.cancelBubble = true; e.preventDefault(); if (confirm("Griffpatch: Are you sure you want to share?")) { // action the share! canShare = true; chk.click(); }*/ return; } } chk = e.target.tagName === "SPAN" ? e.target.parentNode : e.target; if (chk.classList.contains("s3devSTT")) { if (chk.textContent === this.m("top") || chk.textContent === this.m("bottom")) { let spriteSelector = e.target.closest("div[class*='sprite-selector-item_sprite-selector-item']"); this.moveCostumeTo(chk.textContent === this.m("top"), spriteSelector); e.cancelBubble = true; e.preventDefault(); } } } } eventMouseUp(e) { this.updateMousePosition(e); if (e.button === 1 && e.target.closest("svg.blocklySvg")) { // On Linux systems, middle click is often treated as a paste. // We do not want this as we assign our own functionality to middle mouse. e.preventDefault(); } } middleClickWorkspace(e) { if (!this.isScriptEditor()) { return; } // todo: if (!this.addon.settings.get("enableBlockInjector")) { // return; // } e.cancelBubble = true; e.preventDefault(); let floatBar = document.getElementById("s3devFloatingBar"); if (floatBar) { floatBar.remove(); } // Popup new input box for block injection document.body.insertAdjacentHTML("beforeend", "\n
\n \n
\n ")); floatBar = document.getElementById("s3devFloatingBar"); floatBar.style.left = this.mouseXY.x + 16 + "px"; floatBar.style.top = this.mouseXY.y - 8 + "px"; this.floatInp = document.getElementById("s3devIInp"); this.floatInp.focus(); // Build Filter List... this.buildFloatingFilterList(e, floatBar); const ddOut = document.getElementById("s3devIDDOut"); ddOut.addEventListener("mousedown", (...e) => this.dropDownFloatClick(...e)); this.floatInp.addEventListener("keyup", (...e) => this.floatInputChange(...e)); this.floatInp.addEventListener("focus", (...e) => this.floatInputChange(...e)); this.floatInp.addEventListener("keydown", (...e) => this.floatInputKeyDown(...e)); } middleClick(e) { // Intercept clicks to allow jump to...? let blockSvg = e.target.closest("[data-id]"); if (!blockSvg) { // Ok, so no selection... are we at least clicking on the workspace? if (e.target.closest("svg.blocklySvg")) { this.blockCursor = null; // Clear the cursor if using the mouse this.middleClickWorkspace(e); } return; } if (!this.addon.settings.get("enableMiddleClickFinder")) { return; } let w = this.utils.getWorkspace(); let dataId = blockSvg.getAttribute("data-id"); let block = w.getBlockById(dataId); if (!block) { return; } // Move outwards until we reach a block we can take action on for (; block; block = block.getSurroundParent()) { if (block.type === "procedures_call") { e.cancelBubble = true; e.preventDefault(); // todo: navigate to definition let findProcCode = block.getProcCode(); let wksp = this.utils.getWorkspace(); let topBlocks = wksp.getTopBlocks(); for (const root of topBlocks) { if (root.type === "procedures_definition") { let label = root.getChildren()[0]; let procCode = label.getProcCode(); if (procCode && procCode === findProcCode) { // Found... navigate to it! this.centerTop(root); return; } } } } if (block.type === "procedures_definition") { let id = block.id ? block.id : block.getId ? block.getId() : null; this.findInp.focus(); this.showDropDown(null, id); // findInp.select(); e.cancelBubble = true; e.preventDefault(); return; } if (block.type === "data_variable" || block.type === "data_changevariableby" || block.type === "data_setvariableto") { let id = block.getVars()[0]; this.findInp.focus(); this.showDropDown(null, id, block); // let button = document.getElementById('s3devReplace'); this.selVarID = id; // button.classList.remove('s3devHide'); e.cancelBubble = true; e.preventDefault(); return; } if (block.type === "event_whenbroadcastreceived" || block.type === "event_broadcastandwait" || block.type === "event_broadcast") { // todo: actually index the broadcasts...! let id = block.id; this.findInp.focus(); this.showDropDown(null, id, block); this.selVarID = id; e.cancelBubble = true; e.preventDefault(); return; } } e.cancelBubble = true; e.preventDefault(); } getEdgeTypeClass(block) { switch (block.edgeShape_) { case 1: return "boolean"; case 2: return "reporter"; default: return block.startHat_ ? "hat" : "block"; } } buildFloatingFilterList(e, floatBar) { // todo: Iterate through the toolbox? let options = []; let t = this.utils.getWorkspace().getToolbox(); let blocks = t.flyout_.getWorkspace().getTopBlocks(); // 107 blocks, not in order... but we can sort by y value or description right :) let fullDom = Blockly.Xml.workspaceToDom(t.flyout_.getWorkspace()); const doms = {}; for (const x of fullDom.children) { if (x.tagName === "BLOCK") { // let type = x.getAttribute('type'); let id = x.getAttribute("id"); doms[id] = x; } } for (const block of blocks) { this.getBlockText(block, options, doms); } // Griffpatch - on second thoughts - lets sort blocks by length so that shortest ones appear at the top. options.sort((a, b) => a.desc.length < b.desc.length ? -1 : a.desc.length > b.desc.length ? 1 : a.desc.localeCompare(b.desc)); // Previous sort was just alphabetical // options.sort((a, b) => a.desc.localeCompare(b.desc)); const dd = document.getElementById("s3devIDD"); let count = 0; //DROPDOWN_BLOCK_LIST_MAX_ROWS 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 }; li.className = "var " + (option.block.isScratchExtension ? "extension" : option.block.getCategory()) + " " + bType; // proc.cls; if (count > DROPDOWN_BLOCK_LIST_MAX_ROWS) { // Limit maximum number of rows to prevent lag when no filter is applied li.style.display = "none"; } dd.appendChild(li); } const ddOut = document.getElementById("s3devIDDOut"); ddOut.classList.add("vis"); // console.log(options); } /** * Flesh out a blocks description - duplicate up blocks with contained picklists (like list drop downs) * @param block * @param options * @param doms * @returns {string} */ 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") { 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; } floatInputKeyDown(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 dd = document.getElementById("s3devIDD"); let sel = dd.querySelector(".sel"); if (sel) { this.dropDownFloatClick(e); } e.cancelBubble = true; e.preventDefault(); return; } if (e.keyCode === 27) { // Escape let findInp = document.getElementById("s3devIInp"); if (findInp.value.length > 0) { findInp.value = ""; // Clear search first, then close on second press this.floatInputChange(e); } else { this.reallyHideFloatDropDown(true); } e.preventDefault(); return; } } navigateFloatFilter(dir) { let dd = document.getElementById("s3devIDD"); let sel = dd.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 = dd.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); } } /** * This is a feature in progress - can we have a virtual cursor that allows the next injected element position be automated * @param block a blockly block * @param typ type */ findNextHole(block, typ) { /* const inputs = block.inputList; if (inputs) { /!** Blockly.Input *!/ for (const input of inputs) { const fieldRow = input.fieldRow; if (fieldRow) { /!** Blockly.FieldNumber *!/ for (const field of fieldRow) { if (field.argType_ && field.argType_.includes(typ)) { } } } } } */ } /** * Inject the selected block into the script * @param e */ dropDownFloatClick(e) { e.cancelBubble = true; if (!e.target.closest("input")) { e.preventDefault(); } let wksp = this.utils.getWorkspace(); 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) { let dd = document.getElementById("s3devIDD"); sel = dd.querySelector(".sel"); } if (!sel) { return; } const xml = new _XML_js__WEBPACK_IMPORTED_MODULE_3__["default"](); let x = xml.xmlDoc.firstChild; 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]; // griffpatch - oops! option.option[1] not 0? } // Handle "stop other scripts in sprite" if (option.option[1] === "other scripts in sprite") { option.dom.querySelector("mutation").setAttribute("hasnext", "true"); } } x.appendChild(option.dom); let ids = Blockly.Xml.domToWorkspace(x, wksp); if (!e.shiftKey) { this.reallyHideFloatDropDown(true); } let block = wksp.getBlockById(ids[0]); if (this.blockCursor) { // What sort of block did we just inject? let typ = this.getEdgeTypeClass(option.block); if (typ === "boolean") { this.findNextHole(this.blockCursor, ""); } else if (typ === "reporter") { this.findNextHole(this.blockCursor, typ); } } this.domHelpers.triggerDragAndDrop(block.svgPath_, null, { x: this.mouseXY.x, y: this.mouseXY.y }, e.shiftKey); if (e.shiftKey) { document.getElementById("s3devIInp").focus(); } this.blockCursor = block; } floatInputChange(e) { let ddOut = document.getElementById("s3devIDDOut"); if (!ddOut.classList.contains("vis")) {// showDropDown(); // hideDropDown(); // Start timer to hide if not got focus } let findInp = document.getElementById("s3devIInp"); // Filter the list... let val = (findInp.value || "").toLowerCase(); if (val === prevVal) { return; } prevVal = val; this.multi.blocks = null; let dd = document.getElementById("s3devIDD"); let p = dd.parentNode; dd.remove(); let count = 0; let split = val.split(" "); let listLI = dd.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 < DROPDOWN_BLOCK_LIST_MAX_ROWS && match) { li.style.display = "block"; this.dom_removeChildren(li); 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(dd); } 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 + "]"); if (!document.getElementById("s3devFind")) { // noinspection JSUnresolvedVariable root.insertAdjacentHTML("beforeend", "\n
\n
\n \n \n \n \n ").concat(this.m("deep"), "\n
").concat(this.m("help"), "\n ").concat(this.m("tutorials"), "
\n
\n
\n ")); this.find = document.getElementById("s3devFind"); this.findInp = document.getElementById("s3devInp"); this.ddOut = document.getElementById("s3devDDOut"); this.domHelpers.bindOnce(this.ddOut, "mousedown", (...e) => this.dropDownClick(...e), undefined); this.dd = document.getElementById("s3devDD"); // bindOnce(find, 'mouseenter', showDropDown); // bindOnce(find, 'mouseleave', hideDropDown); this.domHelpers.bindOnce(this.findInp, "keyup", (...e) => this.inputChange(...e), undefined); this.domHelpers.bindOnce(this.findInp, "keydown", (...e) => this.inputKeyDown(...e), undefined); this.domHelpers.bindOnce(this.findInp, "focus", (...e) => this.inputChange(...e), undefined); this.domHelpers.bindOnce(document.getElementById("s3devHelp"), "click", (...e) => this.eventClickHelp(...e), undefined); 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 Multi { constructor(utils) { this.idx = 0; this.blocks = null; this.selID = null; /** * @type {Utils} */ this.utils = utils; } update() { const count = document.getElementById("s3devMultiCount"); count.innerText = this.blocks && this.blocks.length > 0 ? enc(this.idx + 1 + " / " + this.blocks.length) : "0"; this.selID = this.idx < this.blocks.length ? this.blocks[this.idx].id : null; } navLeft(e) { return this.navSideways(e, -1); } navRight(e) { return this.navSideways(e, 1); } navSideways(e, dir) { if (this.blocks && this.blocks.length > 0) { this.idx = (this.idx + dir + this.blocks.length) % this.blocks.length; // + length to fix negative modulo js issue. this.update(); this.utils.scrollBlockIntoView(this.blocks[this.idx]); } if (e) { e.cancelBubble = true; e.preventDefault(); } return false; } } 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; } } const DROPDOWN_BLOCK_LIST_MAX_ROWS = 25; let rhdd = 0; let rhdd2 = 0; let prevVal = ""; function enc(str) { return String(str).replace(/&/g, "&").replace(//g, ">").replace(/"/g, """); } /***/ }), /***/ "./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/XML.js": /*!**************************************************!*\ !*** ./src/addons/addons/editor-devtools/XML.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 XML; }); /** * A fake XML class to use in place of the real XML for faking things? */ class XML { constructor() { this.xmlDoc = document.implementation.createDocument(null, "xml"); } newXml(x, tagName, attrs) { let xAdd = this.xmlDoc.createElement(tagName); x.appendChild(xAdd); return this.setAttr(xAdd, attrs); } setAttr(x, attrs) { if (attrs) { for (const key of Object.keys(attrs)) { if (key === "text") { x.appendChild(this.xmlDoc.createTextNode(attrs[key])); } else { x.setAttribute(key, attrs[key]); } } } return x; } } /***/ }), /***/ "./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 _css_loader_userscript_css__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! css-loader!./userscript.css */ "./node_modules/css-loader/index.js!./src/addons/addons/editor-devtools/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-devtools/blockly/BlockFlasher.js": /*!*******************************************************************!*\ !*** ./src/addons/addons/editor-devtools/blockly/BlockFlasher.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 BlockFlasher; }); /** * Helper class to flash a Blockly scratch block in the users workspace */ class BlockFlasher { /** * FLash a block 3 times * @param block the block to flash */ static flash(block) { if (myFlash.timerID > 0) { clearTimeout(myFlash.timerID); if (myFlash.block.svgPath_) { myFlash.block.svgPath_.style.fill = ""; } } let count = 4; let flashOn = true; myFlash.block = block; /** * Internal method to switch the colour of a block between light yellow and it's original colour * @private */ function _flash() { if (myFlash.block.svgPath_) { myFlash.block.svgPath_.style.fill = flashOn ? "#ffff80" : ""; } flashOn = !flashOn; count--; if (count > 0) { myFlash.timerID = setTimeout(_flash, 200); } else { myFlash.timerID = 0; myFlash.block = null; } } _flash(); } } const myFlash = { block: null, timerID: null }; /***/ }), /***/ "./src/addons/addons/editor-devtools/blockly/UndoGroup.js": /*!****************************************************************!*\ !*** ./src/addons/addons/editor-devtools/blockly/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/blockly/Utils.js": /*!************************************************************!*\ !*** ./src/addons/addons/editor-devtools/blockly/Utils.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 Utils; }); /* harmony import */ var _BlockInstance_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../BlockInstance.js */ "./src/addons/addons/editor-devtools/BlockInstance.js"); /* harmony import */ var _BlockFlasher_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./BlockFlasher.js */ "./src/addons/addons/editor-devtools/blockly/BlockFlasher.js"); // A file to split Editor Devtools by features. class Utils { constructor(addon) { this.addon = addon; this.addon.tab.traps.getBlockly().then(blockly => { this.blockly = blockly; }); /** * Scratch Virtual Machine * @type {null|*} */ this.vm = this.addon.tab.traps.vm; // this._myFlash = { block: null, timerID: null, colour: null }; this.offsetX = 32; this.offsetY = 32; this.navigationHistory = new NavigationHistory(this); /** * The workspace */ this._workspace = null; } /** * Get the Scratch Editing Target * @returns {?Target} the scratch editing target */ getEditingTarget() { return this.vm.runtime.getEditingTarget(); } /** * Set the current workspace (switches sprites) * @param targetID {string} */ setEditingTarget(targetID) { if (this.getEditingTarget().id !== targetID) { this.vm.setEditingTarget(targetID); } } /** * Returns the main workspace * @returns !Blockly.Workspace */ getWorkspace() { const currentWorkspace = Blockly.getMainWorkspace(); if (currentWorkspace.getToolbox()) { // Sadly get get workspace does not always return the 'real' workspace... Not sure how to get that at the moment, // but we can work out whether it's the right one by whether it has a toolbox. this._workspace = currentWorkspace; } return this._workspace; } /** * Based on wksp.centerOnBlock(li.data.labelID); * @param blockOrId {Blockly.Block|{id}|BlockInstance} A Blockly Block, a block id, or a BlockInstance */ scrollBlockIntoView(blockOrId) { var _this$blockly; let workspace = this.getWorkspace(); /** @type {Blockly.Block} */ let block; // or is it really a Blockly.BlockSvg? if (blockOrId instanceof _BlockInstance_js__WEBPACK_IMPORTED_MODULE_0__["default"]) { // Switch to sprite this.setEditingTarget(blockOrId.targetId); // Highlight the block! block = workspace.getBlockById(blockOrId.id); } else { block = blockOrId && blockOrId.id ? blockOrId : workspace.getBlockById(blockOrId); } if (!block) { return; } /** * !Blockly.Block */ let root = block.getRootBlock(); let base = this.getTopOfStackFor(block); let ePos = base.getRelativeToSurfaceXY(), // Align with the top of the block rPos = root.getRelativeToSurfaceXY(), // Align with the left of the block 'stack' scale = workspace.scale, x = rPos.x * scale, y = ePos.y * scale, xx = block.width + x, // Turns out they have their x & y stored locally, and they are the actual size rather than scaled or including children... yy = block.height + y, s = workspace.getMetrics(); if (x < s.viewLeft + this.offsetX - 4 || xx > s.viewLeft + s.viewWidth || y < s.viewTop + this.offsetY - 4 || yy > s.viewTop + s.viewHeight) { // sx = s.contentLeft + s.viewWidth / 2 - x, let sx = x - s.contentLeft - this.offsetX, // sy = s.contentTop - y + Math.max(Math.min(32, 32 * scale), (s.viewHeight - yh) / 2); sy = y - s.contentTop - this.offsetY; this.navigationHistory.storeView(this.navigationHistory.peek(), 64); // workspace.hideChaff(), workspace.scrollbar.set(sx, sy); this.navigationHistory.storeView({ left: sx, top: sy }, 64); } (_this$blockly = this.blockly) === null || _this$blockly === void 0 ? void 0 : _this$blockly.hideChaff(); _BlockFlasher_js__WEBPACK_IMPORTED_MODULE_1__["default"].flash(block); } /** * Find the top stack block of a stack * @param block a block in a stack * @returns {*} a block that is the top of the stack of blocks */ getTopOfStackFor(block) { let base = block; while (base.getOutputShape() && base.getSurroundParent()) { base = base.getSurroundParent(); } return base; } } class NavigationHistory { constructor(utils) { this.utils = utils; this.views = []; this.forward = []; } /** * Keep a record of the scroll and zoom position */ storeView(next, dist) { this.forward = []; let workspace = this.utils.getWorkspace(), s = workspace.getMetrics(); let pos = { left: s.viewLeft, top: s.viewTop }; if (!next || distance(pos, next) > dist) { this.views.push(pos); } } peek() { return this.views.length > 0 ? this.views[this.views.length - 1] : null; } goBack() { const workspace = this.utils.getWorkspace(), s = workspace.getMetrics(); let pos = { left: s.viewLeft, top: s.viewTop }; let view = this.peek(); if (!view) { return; } if (distance(pos, view) < 64) { // Go back to current if we are already far away from it if (this.views.length > 1) { this.views.pop(); this.forward.push(view); } } view = this.peek(); if (!view) { return; } let sx = view.left - s.contentLeft, sy = view.top - s.contentTop; // transform.setTranslate(-600,0); workspace.scrollbar.set(sx, sy); /* let blocklySvg = document.getElementsByClassName('blocklySvg')[0]; let blocklyBlockCanvas = blocklySvg.getElementsByClassName('blocklyBlockCanvas')[0]; let transform = blocklyBlockCanvas.transform.baseVal.getItem(0); let scale = blocklyBlockCanvas.transform.baseVal.getItem(1); let transformMatrix = transform.matrix; let scaleMatrix = scale.matrix; console.log('Transform - getMetrics', s); console.log('sx, sy: ', sx, sy); console.log('left, top: ', view.left, view.top); console.log('contentLeft, right:', s.contentLeft, s.contentTop); console.log('transform, scale matrix: ', transformMatrix, scaleMatrix); */ } goForward() { let view = this.forward.pop(); if (!view) { return; } this.views.push(view); let workspace = this.utils.getWorkspace(), s = workspace.getMetrics(); let sx = view.left - s.contentLeft, sy = view.top - s.contentTop; workspace.scrollbar.set(sx, sy); } } function distance(pos, next) { return Math.sqrt(Math.pow(pos.left - next.left, 2) + Math.pow(pos.top - next.top, 2)); } /***/ }), /***/ "./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 _url_loader_icon_close_svg__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! url-loader!./icon--close.svg */ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/editor-devtools/icon--close.svg"); /* harmony import */ var _DevTools_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./DevTools.js */ "./src/addons/addons/editor-devtools/DevTools.js"); /* inserted by pull.js */ const _twGetAsset = path => { if (path === "/icon--close.svg") return _url_loader_icon_close_svg__WEBPACK_IMPORTED_MODULE_0__["default"]; throw new Error("Unknown asset: ".concat(path)); }; /* harmony default export */ __webpack_exports__["default"] = (async function ({ addon, global, console, msg, safeMsg: m }) { const guiDirection = addon.tab.direction; const helpHTML = "\n
\n
\n
\n
").concat(m("help-title"), "
\n
\n
\n\t \n\t
\n
\n
\n
\n

").concat(m("code-tab-features"), "

\n

").concat(m("interactive-find-bar"), " – ").concat(m("interactive-find-bar-desc"), "

\n

").concat(m("improved-tidy-up"), " – ").concat(m("improved-tidy-up-desc"), "

\n

").concat(m("help-make-space"), " ").concat(m("help-new"), " – ").concat(m("help-make-space-desc"), "

\n

").concat(m("copy-to-clipboard"), " – ").concat(m("copy-to-clipboard-desc"), "

\n

").concat(m("paste-from-clipboard"), " – ").concat(m("paste-from-clipboard-desc"), "

\n

").concat(m("swap-variable"), " – ").concat(m("swap-variable-desc"), "

\n

").concat(m("middleclick"), " – ").concat(m("middleclick-desc"), "

\n

").concat(m("ctrl-lr"), " – ").concat(m("ctrl-lr-desc"), "

\n

").concat(m("ctrl-space"), " – ").concat(m("ctrl-space-desc"), "

\n
\n

").concat(m("costume-tab-features"), "

\n

").concat(m("find-bar"), " – ").concat(m("find-bar-costume-desc"), "

\n

").concat(m("ctrl-lr"), " – ").concat(m("ctrl-lr-costume-desc"), "

\n
\n

").concat(m("youtube"), " - https://www.youtube.com/user/griffpatch

\n
\n
\n
\n"); const devTools = new _DevTools_js__WEBPACK_IMPORTED_MODULE_1__["default"](addon, msg, m, helpHTML); 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, global, 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 ADDON_ITEMS = ["createGlobalVariable", "createLocalVariable", "createGlobalList", "createLocalList", "createBroadcast"]; 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 id = menuItem.getValue(); switch (id) { case "createGlobalVariable": { Blockly.Events.setGroup(true); const variable = workspace.createVariable(searchBar.value); // Creating a variable can cause blocks in the flyout to be disposed and recreated // That could cause setValue to throw if (this.sourceBlock_) this.setValue(variable.getId()); Blockly.Events.setGroup(false); return; } case "createLocalVariable": { Blockly.Events.setGroup(true); const variable = workspace.createVariable(searchBar.value, "", null, true); if (this.sourceBlock_) this.setValue(variable.getId()); Blockly.Events.setGroup(false); return; } case "createGlobalList": { Blockly.Events.setGroup(true); const variable = workspace.createVariable(searchBar.value, "list"); if (this.sourceBlock_) this.setValue(variable.getId()); Blockly.Events.setGroup(false); return; } case "createLocalList": { Blockly.Events.setGroup(true); const variable = workspace.createVariable(searchBar.value, "list", null, true); if (this.sourceBlock_) this.setValue(variable.getId()); Blockly.Events.setGroup(false); return; } case "createBroadcast": { Blockly.Events.setGroup(true); const variable = workspace.createVariable(searchBar.value, "broadcast_msg"); 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 query = searchBar.value.toLowerCase().trim(); const rank = (item, index) => { // Negative number will hide // Higher numbers will appear first const option = currentDropdownOptions[index]; if (SCRATCH_ITEMS_TO_HIDE.includes(option[1])) { return query ? -1 : 0; } else if (ADDON_ITEMS.includes(option[1])) { item.element.lastChild.lastChild.textContent = getMenuItemMessage(option[1])[0]; return query ? 0 : -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) || "" }), message]; } }); /***/ }), /***/ "./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__); /* 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/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__); /* harmony import */ var _url_loader_folder_svg__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! url-loader!./folder.svg */ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/folders/folder.svg"); 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; } /* inserted by pull.js */ const _twGetAsset = path => { if (path === "/folder.svg") return _url_loader_folder_svg__WEBPACK_IMPORTED_MODULE_0__["default"]; throw new Error("Unknown asset: ".concat(path)); }; /* harmony default export */ __webpack_exports__["default"] = (async function ({ addon, global, 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: // // // // // // ... // // 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 _twGetAsset("/folder.svg"); } }; // https://github.com/LLK/scratch-gui/blob/develop/src/components/asset-panel/icon--sound.svg const imageIconSource = "\n\n \n \n \n"; 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;,"); 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 += ""); } } result += ""; 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); }; }; 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/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__); /* 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/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 import */ var _url_loader_decrement_svg__WEBPACK_IMPORTED_MODULE_0__ = __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_1__ = __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_2__ = __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_3__ = __webpack_require__(/*! url-loader!./toggle.svg */ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/onion-skinning/toggle.svg"); /* inserted by pull.js */ const _twGetAsset = path => { if (path === "/decrement.svg") return _url_loader_decrement_svg__WEBPACK_IMPORTED_MODULE_0__["default"]; if (path === "/increment.svg") return _url_loader_increment_svg__WEBPACK_IMPORTED_MODULE_1__["default"]; if (path === "/settings.svg") return _url_loader_settings_svg__WEBPACK_IMPORTED_MODULE_2__["default"]; if (path === "/toggle.svg") return _url_loader_toggle_svg__WEBPACK_IMPORTED_MODULE_3__["default"]; throw new Error("Unknown asset: ".concat(path)); }; /* harmony default export */ __webpack_exports__["default"] = (async function ({ addon, global, 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("]*>/); if (svgAttrs && svgAttrs[0].indexOf("xmlns=") === -1) { asset = asset.replace(" { 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 = _twGetAsset("/" + name + ".svg"); return el; }; const paintEditorControlsContainer = document.createElement("div"); paintEditorControlsContainer.className = "sa-onion-controls-container"; paintEditorControlsContainer.dir = ""; 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")); toggleControlsGroup.appendChild(toggleButton); const settingButton = createButton(); settingButton.addEventListener("click", () => setSettingsOpen(!areSettingsOpen())); settingButton.title = msg("settings"); settingButton.appendChild(createButtonImage("settings")); toggleControlsGroup.appendChild(settingButton); paintEditorControlsContainer.appendChild(toggleControlsGroup); // // Settings page // 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']"); const canvasContainer = document.querySelector("[class^='paint-editor_canvas-container']"); // TODO: when leaving the paint editor, references to the old zoom controls are kept around by our DOM // Need to investigate whether this leaks memory or other issues. const oldZoomControlsContainer = paintEditorControlsContainer.querySelector("[class^='paint-editor_zoom-controls']"); if (oldZoomControlsContainer) { oldZoomControlsContainer.parentNode.removeChild(oldZoomControlsContainer); } paintEditorControlsContainer.appendChild(zoomControlsContainer); canvasControls.appendChild(paintEditorControlsContainer); canvasContainer.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, global, console }) { const brand = Symbol(); const setIsPicking = picking => document.body.classList.toggle("sa-stage-color-picker-picking", picking); 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 && 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); }); } }); } 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: // // `rgbToRgb` // Handle bounds / percentage checking to conform to CSS color spec // // *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 // 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 // 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 // --------------------- // 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 // ------------------ // 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 // 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 () { // var CSS_INTEGER = "[-\\+]?\\d+%?"; // 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