mirror of
https://gitlab.com/skysthelimit.dev/selenite.git
synced 2025-06-16 10:32:08 -05:00
9383 lines
323 KiB
JavaScript
9383 lines
323 KiB
JavaScript
(window["webpackJsonpGUI"] = window["webpackJsonpGUI"] || []).push([["addon-default-entry"],{
|
|
|
|
/***/ "./node_modules/css-loader/index.js!./src/addons/addons/color-picker/style.css":
|
|
/*!****************************************************************************!*\
|
|
!*** ./node_modules/css-loader!./src/addons/addons/color-picker/style.css ***!
|
|
\****************************************************************************/
|
|
/*! no static exports found */
|
|
/***/ (function(module, exports, __webpack_require__) {
|
|
|
|
exports = module.exports = __webpack_require__(/*! ../../../../node_modules/css-loader/lib/css-base.js */ "./node_modules/css-loader/lib/css-base.js")(false);
|
|
// imports
|
|
|
|
|
|
// module
|
|
exports.push([module.i, ".sa-color-picker {\n display: flex;\n}\n\n.sa-color-picker-code {\n margin: 8px 0;\n}\n\n.sa-color-picker-paint {\n margin-top: 16px;\n margin-bottom: 4px;\n}\n\n.sa-color-picker > .sa-color-picker-color {\n border: none;\n border-top-left-radius: 1rem;\n border-bottom-left-radius: 1rem;\n padding: 0;\n padding-left: 0.6rem;\n padding-right: 0.4rem;\n margin-left: 0.5rem;\n outline: none;\n box-sizing: border-box;\n width: 3rem;\n height: 2rem;\n}\n[theme=\"dark\"] .sa-color-picker > .sa-color-picker-color {\n background: var(--ui-secondary);\n}\n\n.sa-color-picker > .sa-color-picker-text {\n box-sizing: border-box;\n width: calc(150px - 3rem);\n border-top-left-radius: 0;\n border-bottom-left-radius: 0;\n}\n\n[dir=\"rtl\"] .sa-color-picker > .sa-color-picker-color {\n border-top-left-radius: 0;\n border-bottom-left-radius: 0;\n border-top-right-radius: 1rem;\n border-bottom-right-radius: 1rem;\n margin-left: 0;\n margin-right: 0.5rem;\n}\n\n[dir=\"rtl\"] .sa-color-picker > .sa-color-picker-text {\n border-top-left-radius: 1rem;\n border-bottom-left-radius: 1rem;\n border-top-right-radius: 0;\n border-bottom-right-radius: 0;\n}\n\nbody.sa-hide-eye-dropper-background div[class*=\"stage_color-picker-background\"] {\n /* Do not show eye dropper background if the color picker is \"fake\" */\n display: none;\n}\n", ""]);
|
|
|
|
// exports
|
|
|
|
|
|
/***/ }),
|
|
|
|
/***/ "./node_modules/css-loader/index.js!./src/addons/addons/editor-comment-previews/userstyle.css":
|
|
/*!*******************************************************************************************!*\
|
|
!*** ./node_modules/css-loader!./src/addons/addons/editor-comment-previews/userstyle.css ***!
|
|
\*******************************************************************************************/
|
|
/*! no static exports found */
|
|
/***/ (function(module, exports, __webpack_require__) {
|
|
|
|
exports = module.exports = __webpack_require__(/*! ../../../../node_modules/css-loader/lib/css-base.js */ "./node_modules/css-loader/lib/css-base.js")(false);
|
|
// imports
|
|
|
|
|
|
// module
|
|
exports.push([module.i, ".sa-comment-preview-outer {\n position: fixed;\n top: 0;\n left: 0;\n z-index: 100000000;\n pointer-events: none;\n}\n\n.sa-comment-preview-inner {\n width: calc(200px - 16px);\n max-height: calc(132px - 8px);\n padding: 8px;\n overflow: hidden;\n\n font-size: 12px;\n white-space: pre-wrap;\n pointer-events: none;\n\n color: rgb(87, 94, 117);\n background-color: rgb(255 255 255 / 90%);\n border-style: none;\n border-radius: 8px;\n filter: drop-shadow(0px 5px 5px rgb(0 0 0 / 10%));\n\n transform: perspective(200px);\n}\n\n@supports (backdrop-filter: blur(16px)) {\n .sa-comment-preview-inner {\n background-color: rgb(255 255 255 / 75%);\n backdrop-filter: blur(16px);\n }\n}\n\n.sa-comment-preview-fade {\n transition: opacity 0.1s, filter 0.1s, transform 0.1s linear;\n}\n\n.sa-comment-preview-hidden {\n opacity: 0;\n filter: none;\n transform: perspective(200px) translateZ(-20px);\n}\n\n.sa-comment-preview-reduce-transparency {\n background-color: rgb(255 255 255);\n backdrop-filter: none;\n}\n", ""]);
|
|
|
|
// exports
|
|
|
|
|
|
/***/ }),
|
|
|
|
/***/ "./node_modules/css-loader/index.js!./src/addons/addons/editor-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.<string>}
|
|
*/
|
|
|
|
this.clones = null;
|
|
this.eventName = null;
|
|
}
|
|
/**
|
|
* True if the blockID matches a black represented by this BlockItem
|
|
* @param id
|
|
* @returns {boolean}
|
|
*/
|
|
|
|
|
|
matchesID(id) {
|
|
if (this.labelID === id) {
|
|
return true;
|
|
}
|
|
|
|
if (this.clones) {
|
|
for (const cloneID of this.clones) {
|
|
if (cloneID === id) {
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
}
|
|
|
|
/***/ }),
|
|
|
|
/***/ "./src/addons/addons/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.<!Blockly.Block>} Array of block usages.
|
|
*/
|
|
|
|
|
|
getVariableUsesById(id) {
|
|
let uses = [];
|
|
let topBlocks = this.getTopBlocks(true); // todo: Confirm this was the right getTopBlocks?
|
|
|
|
for (const topBlock of topBlocks) {
|
|
/** @type {!Array<!Blockly.Block>} */
|
|
let kids = topBlock.getDescendants();
|
|
|
|
for (const block of kids) {
|
|
/** @type {!Array<!Blockly.VariableModel>} */
|
|
let blockVariables = block.getVarModels();
|
|
|
|
if (blockVariables) {
|
|
for (const blockVar of blockVariables) {
|
|
if (blockVar.getId() === id) {
|
|
uses.push(block);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return uses;
|
|
}
|
|
/**
|
|
* Find all the uses of a named procedure.
|
|
* @param {string} id ID of the variable to find.
|
|
* @return {!Array.<!Blockly.Block>} 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<!Blockly.Block>} */
|
|
let kids = topBlock.getDescendants();
|
|
|
|
for (const block of kids) {
|
|
if (block.type === "procedures_call") {
|
|
if (block.getProcCode() === procCode) {
|
|
uses.push(block);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return uses;
|
|
}
|
|
/**
|
|
* Find all the uses of a named procedure.
|
|
* @param {string} name name of the variable to find.
|
|
* @return {!Array.<!Blockly.Block>} 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<!Blockly.Block>} */
|
|
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 <span id=\"s3devMulti\" class=\"s3devMulti\">\n <span id=\"s3devMultiLeft\" class=\"s3devNav\">\u25C0</span><span id=\"s3devMultiCount\"></span><span id=\"s3devMultiRight\" class=\"s3devNav\">\u25B6</span>\n </span>\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)) + '<b>' + enc(procCode.substr(i, val.length)) + "</b>" + 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",
|
|
`
|
|
<div id="s3devOverlay">
|
|
</div>
|
|
`
|
|
);
|
|
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<any>} 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",
|
|
`
|
|
<div class="${this.addon.tab.scratchClass(
|
|
"context-menu_menu-item",
|
|
"context-menu_menu-item-bordered",
|
|
{
|
|
others: ["react-contextmenu-item", "s3devSTT"],
|
|
}
|
|
)}" role="menuitem"
|
|
tabindex="-1" aria-disabled="false"><span>${this.m("top")}</span></div>
|
|
<div class="${this.addon.tab.scratchClass("context-menu_menu-item", {
|
|
others: ["react-contextmenu-item", "s3devSTT"],
|
|
})}" role="menuitem"
|
|
tabindex="-1" aria-disabled="false"><span>${this.m("bottom")}</span></div>
|
|
`
|
|
);
|
|
}
|
|
}
|
|
*/
|
|
} 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 <div id=\"s3devFloatingBar\" dir=\"".concat(this.addon.tab.direction, "\">\n <label class='title s3devLabel' id=s3devInsertLabel>\n <span style=\"display:none;\">").concat(this.m("insert"), " </span>\n <span id=s3devInsert class=\"s3devWrap\">\n <div id='s3devIDDOut' class=\"s3devDDOut\">\n <input id='s3devIInp' class=\"").concat(this.addon.tab.scratchClass("input_input-form", {
|
|
others: "s3devInp"
|
|
}), "\" type='search' placeholder='").concat(this.m("start-typing"), "' autocomplete='off'>\n <ul id='s3devIDD' class=\"s3devDD\"></ul>\n </div>\n </span>\n </label>\n </div>\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 <div id=\"s3devToolBar\">\n <div class='title s3devLabel' id=s3devFindLabel>\n <label for=\"s3devInp\">".concat(this.m("find"), " ").concat(this.addon.self._isDevtoolsExtension ? "" : '<a href="#" class="s3devAction" id="s3devHelp" style="/*s-a*/ margin-left: 0; font-size: 10px; /*s-a*/">(?)</a>', " </label>\n <span id=s3devFind class=\"s3devWrap\">\n <label id='s3devDDOut' class=\"s3devDDOut\">\n <input id='s3devInp' class=\"").concat(this.addon.tab.scratchClass("input_input-form", {
|
|
others: "s3devInp"
|
|
}), "\" type='search' placeholder='").concat(this.m("find-placeholder"), "' autocomplete='off'>\n <ul id='s3devDD' class=\"s3devDD\"></ul>\n </label>\n </span>\n <a id=\"s3devDeep\" class=\"s3devAction s3devHide\" href=\"#\">").concat(this.m("deep"), "</a>\n <div ").concat(this.addon.self._isDevtoolsExtension ? "" : 'style="display: none;"', "><a href=\"#\" class=\"s3devAction\" id=\"s3devHelp\"><b>").concat(this.m("help"), "</b></a>\n <a href=\"https://www.youtube.com/griffpatch\" class=\"s3devAction\" target=\"_blank\" id=\"s3devHelp\" rel=\"noreferrer noopener\">").concat(this.m("tutorials"), "</a></div>\n </div>\n </div>\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, ">").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<div id=\"s3devHelpPop\" class=\"".concat(addon.tab.scratchClass("modal_modal-overlay"), "\" dir=\"").concat(guiDirection, "\">\n<div class=\"").concat(addon.tab.scratchClass("modal_modal-content"), "\">\n<div class=\"").concat(addon.tab.scratchClass("modal_header"), "\">\n <div class=\"").concat(addon.tab.scratchClass("modal_header-item", "modal_header-item-title"), "\">").concat(m("help-title"), "</div>\n <div class=\"").concat(addon.tab.scratchClass("modal_header-item", "modal_header-item-close"), "\">\n <div class=\"").concat(addon.tab.scratchClass("close-button_close-button", "close-button_large", {
|
|
others: "close-button"
|
|
}), "\">\n\t <img class=\"").concat(addon.tab.scratchClass("close-button_close-icon"), "\" src=\"").concat(_twGetAsset("/icon--close.svg"), "\">\n\t</div>\n </div>\n</div>\n<div id=\"s3devHelpContent\">\n<h2><strong>").concat(m("code-tab-features"), "</strong></h2>\n<p dir=\"auto\"><strong>").concat(m("interactive-find-bar"), "</strong> – ").concat(m("interactive-find-bar-desc"), "</p>\n<p dir=\"auto\"><strong>").concat(m("improved-tidy-up"), "</strong> – ").concat(m("improved-tidy-up-desc"), "</p>\n<p dir=\"auto\"><strong>").concat(m("help-make-space"), " ").concat(m("help-new"), "</strong> – ").concat(m("help-make-space-desc"), "</p>\n<p dir=\"auto\"><strong>").concat(m("copy-to-clipboard"), "</strong> – ").concat(m("copy-to-clipboard-desc"), "</p>\n<p dir=\"auto\"><strong>").concat(m("paste-from-clipboard"), "</strong> – ").concat(m("paste-from-clipboard-desc"), "</p>\n<p dir=\"auto\"><strong>").concat(m("swap-variable"), "</strong> – ").concat(m("swap-variable-desc"), "</p>\n<p dir=\"auto\"><strong>").concat(m("middleclick"), "</strong> – ").concat(m("middleclick-desc"), "</p>\n<p dir=\"auto\"><strong>").concat(m("ctrl-lr"), "</strong> – ").concat(m("ctrl-lr-desc"), "</p>\n<p dir=\"auto\"><strong>").concat(m("ctrl-space"), "</strong> – ").concat(m("ctrl-space-desc"), "</p>\n<hr />\n<h2><strong>").concat(m("costume-tab-features"), "</strong></h2>\n<p dir=\"auto\"><strong>").concat(m("find-bar"), "</strong> – ").concat(m("find-bar-costume-desc"), "</p>\n<p dir=\"auto\"><strong>").concat(m("ctrl-lr"), "</strong> – ").concat(m("ctrl-lr-costume-desc"), "</p>\n<hr />\n<p>").concat(m("youtube"), " - <a target=\"_blank\" href=\"https://www.youtube.com/griffpatch\" rel=\"noopener noreferrer\">https://www.youtube.com/user/griffpatch</a></p>\n</div>\n</div>\n</div>\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:
|
|
// <SelectorHOC>
|
|
// <SpriteSelectorItem />
|
|
// <SpriteSelectorItem />
|
|
// <SpriteSelectorItem />
|
|
// <SpriteSelectorItem />
|
|
// ...
|
|
// </SelectorHOC>
|
|
// It's obviously more complicated than that, but there are two important parts:
|
|
// SelectorHOC - We override this to change which items are displayed
|
|
// SpriteSelectorItem - We override this to change how items are displayed.
|
|
// Folders are just items rendered differently
|
|
// These two components communicate through the `name` property of the items.
|
|
// We touch some things on the VM to make dragging items work properly.
|
|
const REACT_INTERNAL_PREFIX = "__reactInternalInstance$";
|
|
const TYPE_SPRITES = 1;
|
|
const TYPE_ASSETS = 2; // We run too early, will be set later
|
|
|
|
let vm;
|
|
let reactInternalKey;
|
|
let currentSpriteFolder;
|
|
let currentAssetFolder;
|
|
let currentSpriteItems;
|
|
let currentAssetItems;
|
|
const DIVIDER = "//";
|
|
/**
|
|
* getFolderFromName("B") === null
|
|
* getFolderFromName("A//b") === "A"
|
|
*/
|
|
|
|
const getFolderFromName = name => {
|
|
const idx = name.indexOf(DIVIDER);
|
|
|
|
if (idx === -1 || idx === 0) {
|
|
return null;
|
|
}
|
|
|
|
return name.substr(0, idx);
|
|
};
|
|
/**
|
|
* getNameWithoutFolder("B") === "B"
|
|
* getNameWithoutFolder("A//b") === "b"
|
|
*/
|
|
|
|
|
|
const getNameWithoutFolder = name => {
|
|
const idx = name.indexOf(DIVIDER);
|
|
|
|
if (idx === -1 || idx === 0) {
|
|
return name;
|
|
}
|
|
|
|
return name.substr(idx + DIVIDER.length);
|
|
};
|
|
/**
|
|
* setFolderOfName("B", "y") === "y//B"
|
|
* setFolderOfName("c//B", "y") === "y//B"
|
|
* setFolderOfName("B", null) === "B"
|
|
* setFolderOfName("c//B", null) === "B"
|
|
*/
|
|
|
|
|
|
const setFolderOfName = (name, folder) => {
|
|
const basename = getNameWithoutFolder(name);
|
|
|
|
if (folder) {
|
|
return "".concat(folder).concat(DIVIDER).concat(basename);
|
|
}
|
|
|
|
return basename;
|
|
};
|
|
|
|
const isValidFolderName = name => {
|
|
return !name.includes(DIVIDER) && !name.endsWith("/");
|
|
};
|
|
|
|
const RESERVED_NAMES = ["_mouse_", "_stage_", "_edge_", "_myself_", "_random_"];
|
|
|
|
const ensureNotReserved = name => {
|
|
if (name === "") return "2";
|
|
if (RESERVED_NAMES.includes(name)) return "".concat(name, "2");
|
|
return name;
|
|
};
|
|
|
|
const getSortableHOCFromElement = el => {
|
|
const nearestSpriteSelector = el.closest("[class*='sprite-selector_sprite-selector']");
|
|
|
|
if (nearestSpriteSelector) {
|
|
return nearestSpriteSelector[reactInternalKey].child.sibling.child.stateNode;
|
|
}
|
|
|
|
const nearestAssetPanelWrapper = el.closest('[class*="asset-panel_wrapper"]');
|
|
|
|
if (nearestAssetPanelWrapper) {
|
|
return nearestAssetPanelWrapper[reactInternalKey].child.child.stateNode;
|
|
}
|
|
|
|
throw new Error("cannot find SortableHOC");
|
|
};
|
|
|
|
const getBackpackFromElement = el => {
|
|
const gui = el.closest('[class*="gui_editor-wrapper"]');
|
|
if (!gui) throw new Error("cannot find Backpack");
|
|
return gui[reactInternalKey].child.sibling.child.child.stateNode;
|
|
};
|
|
|
|
const clamp = (n, min, max) => {
|
|
return Math.min(Math.max(n, min), max);
|
|
};
|
|
/**
|
|
* @typedef {Object} ItemData
|
|
* @property {string} realName
|
|
* @property {number} realIndex
|
|
* @property {string} inFolder
|
|
* @property {string} folder
|
|
* @property {boolean} folderOpen
|
|
*/
|
|
|
|
/**
|
|
* @returns {ItemData|null}
|
|
*/
|
|
|
|
|
|
const getItemData = item => {
|
|
if (item && item.name && typeof item.name === "object") {
|
|
return item.name;
|
|
}
|
|
|
|
return null;
|
|
};
|
|
|
|
const openFolderAsset = {
|
|
assetId: "&__sa_folders_folder",
|
|
|
|
encodeDataURI() {
|
|
// Doesn't actually need to be a data: URI
|
|
return _twGetAsset("/folder.svg");
|
|
}
|
|
|
|
}; // https://github.com/LLK/scratch-gui/blob/develop/src/components/asset-panel/icon--sound.svg
|
|
|
|
const imageIconSource = "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n<svg width=\"100px\" height=\"100px\" viewBox=\"0 0 20 20\" version=\"1.1\" xmlns=\"http://www.w3.org/2000/svg\" xmlns:xlink=\"http://www.w3.org/1999/xlink\">\n <g id=\"Sound\" stroke=\"none\" stroke-width=\"1\" fill=\"none\" fill-rule=\"evenodd\">\n <path d=\"M12.4785058,12.6666667 C12.3144947,12.6666667 12.1458852,12.6272044 11.9926038,12.5440517 C11.537358,12.2960031 11.3856094,11.7562156 11.6553847,11.3376335 C12.1688774,10.5371131 12.1688774,9.54491867 11.6553847,8.74580756 C11.3856094,8.32581618 11.537358,7.78602861 11.9926038,7.53798001 C12.452448,7.29275014 13.0379829,7.43086811 13.3046926,7.84804076 C14.1737981,9.20103311 14.1737981,10.8809986 13.3046926,12.233991 C13.1268862,12.5130457 12.806528,12.6666667 12.4785058,12.6666667 Z M15.3806784,13.8333333 C15.2408902,13.8333333 15.0958763,13.796281 14.9665396,13.7182064 C14.5785295,13.485306 14.4491928,12.9784829 14.6791247,12.5854634 C15.5949331,11.0160321 15.5949331,9.065491 14.6791247,7.49738299 C14.4491928,7.10436352 14.5785295,6.59621712 14.9665396,6.36331669 C15.3558562,6.13438616 15.8549129,6.26274605 16.0848448,6.65444223 C17.3050517,8.74260632 17.3050517,11.3389168 16.0848448,13.4270809 C15.9319924,13.6890939 15.6602547,13.8333333 15.3806784,13.8333333 Z M10.3043478,5.62501557 L10.3043478,13.873675 C10.3043478,14.850934 9.10969849,15.3625101 8.36478311,14.7038052 L6.7566013,13.2797607 C6.18712394,12.7762834 5.44499329,12.4968737 4.67362297,12.4968737 L4.3923652,12.4968737 C3.62377961,12.4968737 3,11.8935108 3,11.1470686 L3,8.36646989 C3,7.62137743 3.62377961,7.01666471 4.3923652,7.01666471 L4.65830695,7.01666471 C5.42967727,7.01666471 6.17180792,6.73725504 6.74128529,6.23377771 L8.36478311,4.79623519 C9.10969849,4.13753026 10.3043478,4.64910643 10.3043478,5.62501557 Z\" id=\"Combined-Shape\" fill=\"#575E75\"></path>\n </g>\n</svg>";
|
|
const soundIconHref = "data:image/svg+xml;base64,".concat(btoa(imageIconSource));
|
|
let folderColorStylesheet = null;
|
|
const folderColors = Object.create(null);
|
|
|
|
const getFolderColorClass = folderName => {
|
|
const mulberry32 = a => {
|
|
// https://stackoverflow.com/a/47593316
|
|
return function () {
|
|
var t = a += 0x6d2b79f5;
|
|
t = Math.imul(t ^ t >>> 15, t | 1);
|
|
t ^= t + Math.imul(t ^ t >>> 7, t | 61);
|
|
return ((t ^ t >>> 14) >>> 0) / 4294967296;
|
|
};
|
|
};
|
|
|
|
const hashCode = str => {
|
|
// Based on Java's String.hashCode
|
|
// https://hg.openjdk.java.net/jdk8/jdk8/jdk/file/687fd7c7986d/src/share/classes/java/lang/String.java#l1452
|
|
let hash = 0;
|
|
|
|
for (let i = 0; i < str.length; i++) {
|
|
hash = 31 * hash + str.charCodeAt(i);
|
|
hash = hash | 0;
|
|
}
|
|
|
|
return hash;
|
|
};
|
|
|
|
const random = str => {
|
|
const seed = hashCode(str);
|
|
const rng = mulberry32(seed); // Run RNG a few times to get more random numbers, otherwise similar seeds tend to give somewhat similar results
|
|
|
|
rng();
|
|
rng();
|
|
rng();
|
|
rng();
|
|
return rng();
|
|
};
|
|
|
|
if (!folderColors[folderName]) {
|
|
if (!folderColorStylesheet) {
|
|
folderColorStylesheet = document.createElement("style");
|
|
document.head.appendChild(folderColorStylesheet);
|
|
}
|
|
|
|
const hue = random(folderName) * 360;
|
|
const color = "hsla(".concat(hue, "deg, 100%, 85%, 0.5)");
|
|
const id = Object.keys(folderColors).length;
|
|
const className = "sa-folders-color-".concat(id);
|
|
folderColors[folderName] = className;
|
|
folderColorStylesheet.textContent += ".".concat(className, "{background-color:").concat(color, " !important;}");
|
|
folderColorStylesheet.textContent += ".".concat(className, "[class*=\"sprite-selector_raised\"]:not([class*=\"sa-folders-folder\"]){background-color:hsla(").concat(hue, "deg, 100%, 77%, 1) !important;}");
|
|
}
|
|
|
|
return folderColors[folderName];
|
|
};
|
|
|
|
const fixOrderOfItemsInFolders = items => {
|
|
const folders = Object.create(null);
|
|
const result = [];
|
|
|
|
for (const item of items) {
|
|
const name = item.getName ? item.getName() : item.name;
|
|
const folder = getFolderFromName(name);
|
|
|
|
if (typeof folder === "string") {
|
|
if (!folders[folder]) {
|
|
folders[folder] = [];
|
|
result.push(folders[folder]);
|
|
}
|
|
|
|
folders[folder].push(item);
|
|
} else {
|
|
result.push(item);
|
|
}
|
|
}
|
|
|
|
const flatResult = result.flat();
|
|
|
|
for (let i = 0; i < items.length; i++) {
|
|
if (result[i] !== items[i]) {
|
|
return {
|
|
items: flatResult,
|
|
changed: true
|
|
};
|
|
}
|
|
}
|
|
|
|
return {
|
|
items: flatResult,
|
|
changed: false
|
|
};
|
|
};
|
|
|
|
const fixTargetOrder = () => {
|
|
const {
|
|
items,
|
|
changed
|
|
} = fixOrderOfItemsInFolders(vm.runtime.targets);
|
|
|
|
if (changed) {
|
|
vm.runtime.targets = items;
|
|
vm.emitTargetsUpdate();
|
|
}
|
|
};
|
|
|
|
const fixCostumeOrder = (target = vm.editingTarget) => {
|
|
const {
|
|
items,
|
|
changed
|
|
} = fixOrderOfItemsInFolders(target.sprite.costumes);
|
|
|
|
if (changed) {
|
|
target.sprite.costumes = items;
|
|
vm.emitTargetsUpdate();
|
|
}
|
|
};
|
|
|
|
const fixSoundOrder = (target = vm.editingTarget) => {
|
|
const {
|
|
items,
|
|
changed
|
|
} = fixOrderOfItemsInFolders(target.sprite.sounds);
|
|
|
|
if (changed) {
|
|
target.sprite.sounds = items;
|
|
vm.emitTargetsUpdate();
|
|
}
|
|
};
|
|
|
|
const verifySortableHOC = sortableHOCInstance => {
|
|
const SortableHOC = sortableHOCInstance.constructor;
|
|
if (Array.isArray(sortableHOCInstance.props.items) && (typeof sortableHOCInstance.props.selectedId === "string" || typeof sortableHOCInstance.props.selectedItemIndex === "number") && typeof sortableHOCInstance.containerBox !== "undefined" && typeof SortableHOC.prototype.componentDidMount === "undefined" && typeof SortableHOC.prototype.componentDidUpdate === "undefined" && typeof SortableHOC.prototype.handleAddSortable === "function" && typeof SortableHOC.prototype.handleRemoveSortable === "function" && typeof SortableHOC.prototype.setRef === "function") return;
|
|
throw new Error("Can not comprehend SortableHOC");
|
|
};
|
|
|
|
const verifySpriteSelectorItem = spriteSelectorItemInstance => {
|
|
const SpriteSelectorItem = spriteSelectorItemInstance.constructor;
|
|
if (typeof spriteSelectorItemInstance.props.asset === "object" && typeof spriteSelectorItemInstance.props.name === "string" && typeof spriteSelectorItemInstance.props.dragType === "string" && typeof SpriteSelectorItem.prototype.handleClick === "function" && typeof SpriteSelectorItem.prototype.setRef === "function" && typeof SpriteSelectorItem.prototype.handleDrag === "function" && typeof SpriteSelectorItem.prototype.handleDragEnd === "function" && typeof SpriteSelectorItem.prototype.handleDelete === "function" && typeof SpriteSelectorItem.prototype.handleDuplicate === "function" && typeof SpriteSelectorItem.prototype.handleExport === "function") return;
|
|
throw new Error("Can not comprehend SpriteSelectorItem");
|
|
};
|
|
|
|
const verifyVM = vm => {
|
|
const target = vm.runtime.targets[0];
|
|
if (typeof vm.installTargets === "function" && typeof vm.reorderTarget === "function" && typeof target.reorderCostume === "function" && typeof target.reorderSound === "function" && typeof target.addCostume === "function" && typeof target.addSound === "function") return;
|
|
throw new Error("Can not comprehend VM");
|
|
};
|
|
|
|
const verifyBackpack = backpackInstance => {
|
|
const Backpack = backpackInstance.constructor;
|
|
|
|
if (typeof Backpack.prototype.handleDrop === "function" && typeof Backpack.prototype.componentDidUpdate === "undefined") {
|
|
return;
|
|
}
|
|
|
|
throw new Error("Can not comprehend Backpack");
|
|
};
|
|
|
|
class Cache {
|
|
constructor() {
|
|
this.cache = new Map();
|
|
this.usedThisTick = new Set();
|
|
}
|
|
|
|
has(id) {
|
|
return this.cache.has(id);
|
|
}
|
|
|
|
get(id) {
|
|
this.usedThisTick.add(id);
|
|
return this.cache.get(id);
|
|
}
|
|
|
|
set(id, value) {
|
|
this.usedThisTick.add(id);
|
|
this.cache.set(id, value);
|
|
}
|
|
|
|
startTick() {
|
|
this.usedThisTick.clear();
|
|
}
|
|
|
|
endTick() {
|
|
for (const id of Array.from(this.cache.keys())) {
|
|
if (!this.usedThisTick.has(id)) {
|
|
this.cache.delete(id);
|
|
}
|
|
}
|
|
}
|
|
|
|
clear() {
|
|
this.usedThisTick.clear();
|
|
this.cache.clear();
|
|
}
|
|
|
|
}
|
|
|
|
const patchSortableHOC = (SortableHOC, type) => {
|
|
// SortableHOC should be: https://github.com/LLK/scratch-gui/blob/29d9851778febe4e69fa5111bf7559160611e366/src/lib/sortable-hoc.jsx#L8
|
|
const itemCache = new Cache();
|
|
const folderItemCache = new Cache();
|
|
const folderAssetCache = new Cache();
|
|
const PREVIEW_SIZE = 80;
|
|
const PREVIEW_POSITIONS = [// x, y
|
|
[0, 0], [PREVIEW_SIZE / 2, 0], [0, PREVIEW_SIZE / 2], [PREVIEW_SIZE / 2, PREVIEW_SIZE / 2]];
|
|
|
|
const createFolderPreview = items => {
|
|
// Directly generate a string instead of using DOM API for performance as we deal with very large inlined images
|
|
// Because the result is only used as an img src, XSS shouldn't be a concern
|
|
let result = "data:image/svg+xml;,<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"".concat(PREVIEW_SIZE, "\" height=\"").concat(PREVIEW_SIZE, "\">");
|
|
|
|
for (let i = 0; i < Math.min(PREVIEW_POSITIONS.length, items.length); i++) {
|
|
const item = items[i];
|
|
const width = PREVIEW_SIZE / 2;
|
|
const height = PREVIEW_SIZE / 2;
|
|
const [x, y] = PREVIEW_POSITIONS[i];
|
|
let src;
|
|
|
|
if (item.asset) {
|
|
// TW: We can be 100% certain that escaping here is unnecessary
|
|
src = item.asset.encodeDataURI();
|
|
} else if (item.costume && item.costume.asset) {
|
|
src = item.costume.asset.encodeDataURI();
|
|
} else if (item.url) {
|
|
src = soundIconHref;
|
|
}
|
|
|
|
if (src) {
|
|
result += "<image width=\"".concat(width, "\" height=\"").concat(height, "\" x=\"").concat(x, "\" y=\"").concat(y, "\" href=\"").concat(src, "\"/>");
|
|
}
|
|
}
|
|
|
|
result += "</svg>";
|
|
return result;
|
|
};
|
|
|
|
const getUniqueIdOfFolderItems = items => {
|
|
let id = "sa_folder&&";
|
|
|
|
for (let i = 0; i < Math.min(PREVIEW_POSITIONS.length, items.length); i++) {
|
|
const item = items[i];
|
|
|
|
if (item.asset) {
|
|
id += item.asset.assetId;
|
|
} else if (item.costume && item.costume.asset) {
|
|
id += item.costume.asset.assetId;
|
|
} else if (item.url) {
|
|
id += item.url;
|
|
}
|
|
|
|
id += "&&";
|
|
}
|
|
|
|
return id;
|
|
};
|
|
|
|
const processItems = (openFolders, props) => {
|
|
const processItem = item => {
|
|
const itemId = item.name;
|
|
let newItem;
|
|
let itemData;
|
|
|
|
if (itemCache.has(itemId)) {
|
|
newItem = itemCache.get(itemId);
|
|
itemData = newItem.name;
|
|
} else {
|
|
itemData = {
|
|
toString() {
|
|
return "_".concat(item.name);
|
|
}
|
|
|
|
};
|
|
newItem = {};
|
|
itemCache.set(itemId, newItem);
|
|
}
|
|
|
|
const itemFolderName = getFolderFromName(item.name);
|
|
Object.assign(newItem, item);
|
|
itemData.realName = item.name;
|
|
itemData.realIndex = i;
|
|
itemData.inFolder = itemFolderName;
|
|
newItem.name = itemData;
|
|
return {
|
|
newItem,
|
|
itemData
|
|
};
|
|
};
|
|
|
|
itemCache.startTick();
|
|
folderItemCache.startTick();
|
|
folderAssetCache.startTick();
|
|
const folderOccurrences = new Map();
|
|
const items = [];
|
|
const result = {
|
|
items
|
|
};
|
|
let i = 0;
|
|
|
|
while (i < props.items.length) {
|
|
const item = props.items[i];
|
|
const folderName = getFolderFromName(item.name);
|
|
|
|
if (folderName === null) {
|
|
items.push(processItem(item).newItem);
|
|
|
|
if (type === TYPE_ASSETS) {
|
|
const isSelected = props.selectedItemIndex === i;
|
|
|
|
if (isSelected) {
|
|
result.selectedItemIndex = items.length - 1;
|
|
}
|
|
}
|
|
} else {
|
|
const isOpen = openFolders.indexOf(folderName) !== -1;
|
|
const folderItems = [];
|
|
|
|
while (i < props.items.length) {
|
|
const childItem = props.items[i];
|
|
const processedItem = processItem(childItem);
|
|
|
|
if (getFolderFromName(childItem.name) !== folderName) {
|
|
break;
|
|
}
|
|
|
|
folderItems.push(processedItem.newItem);
|
|
|
|
if (type === TYPE_ASSETS) {
|
|
const isSelected = props.selectedItemIndex === i;
|
|
|
|
if (isSelected) {
|
|
if (isOpen) {
|
|
result.selectedItemIndex = items.length + folderItems.length;
|
|
} else {
|
|
result.selectedItemIndex = -1;
|
|
}
|
|
}
|
|
}
|
|
|
|
i++;
|
|
}
|
|
|
|
i--;
|
|
const occurrence = folderOccurrences.get(folderName) || 0;
|
|
folderOccurrences.set(folderName, occurrence + 1);
|
|
const baseUniqueId = getUniqueIdOfFolderItems(folderItems);
|
|
const itemUniqueId = "".concat(isOpen, "&").concat(occurrence, "&").concat(folderName, "&").concat(baseUniqueId, "&");
|
|
const reactKey = "&__".concat(occurrence, "_").concat(folderName);
|
|
const assetUniqueId = baseUniqueId;
|
|
let folderItem;
|
|
let folderData;
|
|
|
|
if (folderItemCache.has(itemUniqueId)) {
|
|
folderItem = folderItemCache.get(itemUniqueId);
|
|
folderData = folderItem.name;
|
|
} else {
|
|
folderItem = {
|
|
// Can be used as a react key
|
|
id: {
|
|
toString() {
|
|
return reactKey;
|
|
}
|
|
|
|
}
|
|
};
|
|
folderData = {
|
|
// Can be used as a react key
|
|
toString() {
|
|
return reactKey;
|
|
}
|
|
|
|
};
|
|
folderItemCache.set(itemUniqueId, folderItem);
|
|
}
|
|
|
|
folderData.folder = folderName;
|
|
folderData.folderOpen = isOpen;
|
|
folderItem.items = folderItems;
|
|
folderItem.name = folderData;
|
|
let folderAsset;
|
|
|
|
if (isOpen) {
|
|
folderAsset = openFolderAsset;
|
|
} else {
|
|
if (folderAssetCache.has(assetUniqueId)) {
|
|
folderAsset = folderAssetCache.get(assetUniqueId);
|
|
} else {
|
|
folderAsset = {
|
|
assetId: assetUniqueId,
|
|
|
|
encodeDataURI() {
|
|
return createFolderPreview(folderItems);
|
|
}
|
|
|
|
};
|
|
folderAssetCache.set(assetUniqueId, folderAsset);
|
|
}
|
|
}
|
|
|
|
if (type === TYPE_SPRITES) {
|
|
if (!folderItem.costume) folderItem.costume = {};
|
|
folderItem.costume.asset = folderAsset; // For sprite items, `id` is used as the drag payload and toString is used as a React key
|
|
|
|
if (!folderItem.id) folderItem.id = {};
|
|
folderItem.id.sa_folder_items = folderItems;
|
|
|
|
folderItem.id.toString = () => reactKey;
|
|
} else {
|
|
folderItem.asset = folderAsset;
|
|
if (!folderItem.dragPayload) folderItem.dragPayload = {};
|
|
folderItem.dragPayload.sa_folder_items = folderItems;
|
|
}
|
|
|
|
items.push(folderItem);
|
|
|
|
if (isOpen) {
|
|
for (const item of folderItems) {
|
|
items.push(item);
|
|
}
|
|
}
|
|
}
|
|
|
|
i++;
|
|
}
|
|
|
|
itemCache.endTick();
|
|
folderItemCache.endTick();
|
|
folderAssetCache.endTick();
|
|
return result;
|
|
};
|
|
|
|
const getSelectedItem = sortable => {
|
|
if (type === TYPE_SPRITES) {
|
|
const selectedItem = sortable.props.items.find(i => i.id === sortable.props.selectedId);
|
|
return selectedItem;
|
|
} else if (type === TYPE_ASSETS) {
|
|
const selectedItem = sortable.props.items[sortable.props.selectedItemIndex];
|
|
return selectedItem;
|
|
}
|
|
|
|
return null;
|
|
};
|
|
|
|
SortableHOC.prototype.saInitialSetup = function () {
|
|
itemCache.clear();
|
|
folderItemCache.clear();
|
|
folderAssetCache.clear();
|
|
const folders = [];
|
|
const selectedItem = getSelectedItem(this);
|
|
|
|
if (selectedItem && !selectedItem.isStage) {
|
|
const folder = getFolderFromName(selectedItem.name);
|
|
folders.push(folder);
|
|
|
|
if (type === TYPE_SPRITES) {
|
|
currentSpriteFolder = folder;
|
|
} else if (type === TYPE_ASSETS) {
|
|
currentAssetFolder = folder;
|
|
}
|
|
}
|
|
|
|
this.setState({
|
|
folders
|
|
});
|
|
};
|
|
|
|
SortableHOC.prototype.componentDidMount = function () {
|
|
// Do part of componentDidUpdate on mount as well
|
|
const selectedItem = getSelectedItem(this);
|
|
|
|
if (selectedItem) {
|
|
const folder = getFolderFromName(selectedItem.name);
|
|
|
|
if (type === TYPE_SPRITES) {
|
|
currentSpriteFolder = folder;
|
|
} else if (type === TYPE_ASSETS) {
|
|
currentAssetFolder = folder;
|
|
}
|
|
}
|
|
|
|
this.saInitialSetup();
|
|
};
|
|
|
|
SortableHOC.prototype.componentDidUpdate = function (prevProps, prevState) {
|
|
const selectedItem = getSelectedItem(this);
|
|
|
|
if (selectedItem) {
|
|
const folder = getFolderFromName(selectedItem.name);
|
|
const currentFolder = this.state.folders.includes(folder) ? folder : null;
|
|
|
|
if (type === TYPE_SPRITES) {
|
|
currentSpriteFolder = currentFolder;
|
|
} else if (type === TYPE_ASSETS) {
|
|
currentAssetFolder = currentFolder;
|
|
}
|
|
|
|
let selectedItemChanged;
|
|
|
|
if (this.props.selectedId) {
|
|
selectedItemChanged = this.props.selectedId !== prevProps.selectedId;
|
|
} else {
|
|
selectedItemChanged = this.props.items[this.props.selectedItemIndex] && prevProps.items[prevProps.selectedItemIndex] && this.props.items[this.props.selectedItemIndex].name !== prevProps.items[prevProps.selectedItemIndex].name;
|
|
}
|
|
|
|
if (selectedItemChanged) {
|
|
if (!selectedItem.isStage) {
|
|
if (typeof folder === "string" && !this.state.folders.includes(folder)) {
|
|
this.setState(prevState => ({
|
|
folders: [...prevState.folders, folder]
|
|
}));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
const originalSortableHOCRender = SortableHOC.prototype.render;
|
|
|
|
SortableHOC.prototype.render = function () {
|
|
const originalProps = this.props;
|
|
this.props = _objectSpread(_objectSpread({}, this.props), processItems(this.state && this.state.folders || [], this.props));
|
|
|
|
if (type === TYPE_SPRITES) {
|
|
currentSpriteItems = this.props.items;
|
|
} else if (type === TYPE_ASSETS) {
|
|
currentAssetItems = this.props.items;
|
|
}
|
|
|
|
const result = originalSortableHOCRender.call(this);
|
|
this.props = originalProps;
|
|
return result;
|
|
};
|
|
};
|
|
|
|
const getAllFolders = component => {
|
|
const result = new Set();
|
|
let items;
|
|
|
|
if (component.props.dragType === "SPRITE") {
|
|
items = currentSpriteItems;
|
|
} else {
|
|
items = currentAssetItems;
|
|
}
|
|
|
|
for (const item of items) {
|
|
const data = getItemData(item);
|
|
|
|
if (typeof data.folder === "string") {
|
|
result.add(data.folder);
|
|
}
|
|
}
|
|
|
|
return Array.from(result);
|
|
};
|
|
|
|
const isFolderOpen = (component, folder) => {
|
|
const sortableHOCInstance = getSortableHOCFromElement(component.ref);
|
|
const folders = sortableHOCInstance.state && sortableHOCInstance.state.folders || [];
|
|
return folders.includes(folder);
|
|
};
|
|
|
|
const setFolderOpen = (component, folder, open) => {
|
|
const sortableHOCInstance = getSortableHOCFromElement(component.ref);
|
|
sortableHOCInstance.setState(prevState => {
|
|
let folders = prevState && prevState.folders || [];
|
|
folders = folders.filter(i => i !== folder);
|
|
|
|
if (open) {
|
|
return {
|
|
folders: [...folders, folder]
|
|
};
|
|
}
|
|
|
|
return {
|
|
folders
|
|
};
|
|
});
|
|
};
|
|
|
|
addon.tab.createEditorContextMenu((ctxType, ctx) => {
|
|
if (ctxType !== "sprite" && ctxType !== "costume" && ctxType !== "sound") return;
|
|
const component = ctx.target[addon.tab.traps.getInternalKey(ctx.target)].return.return.return.stateNode;
|
|
const data = getItemData(component.props);
|
|
if (!data) return;
|
|
|
|
if (typeof data.folder === "string") {
|
|
ctx.target.setAttribute("sa-folders-context-type", "folder");
|
|
|
|
const renameItems = newName => {
|
|
const isOpen = isFolderOpen(component, data.folder);
|
|
setFolderOpen(component, data.folder, false);
|
|
|
|
if (isOpen && typeof newName === "string") {
|
|
setFolderOpen(component, newName, true);
|
|
}
|
|
|
|
if (component.props.dragType === "SPRITE") {
|
|
for (const target of vm.runtime.targets) {
|
|
if (target.isOriginal) {
|
|
if (getFolderFromName(target.getName()) === data.folder) {
|
|
vm.renameSprite(target.id, ensureNotReserved(setFolderOfName(target.getName(), newName)));
|
|
}
|
|
}
|
|
}
|
|
|
|
vm.emitWorkspaceUpdate();
|
|
fixTargetOrder();
|
|
} else if (component.props.dragType === "COSTUME") {
|
|
for (let i = 0; i < vm.editingTarget.sprite.costumes.length; i++) {
|
|
const costume = vm.editingTarget.sprite.costumes[i];
|
|
|
|
if (getFolderFromName(costume.name) === data.folder) {
|
|
vm.renameCostume(i, setFolderOfName(costume.name, newName));
|
|
}
|
|
}
|
|
|
|
fixCostumeOrder();
|
|
} else if (component.props.dragType === "SOUND") {
|
|
for (let i = 0; i < vm.editingTarget.sprite.sounds.length; i++) {
|
|
const sound = vm.editingTarget.sprite.sounds[i];
|
|
|
|
if (getFolderFromName(sound.name) === data.folder) {
|
|
vm.renameSound(i, setFolderOfName(sound.name, newName));
|
|
}
|
|
}
|
|
|
|
fixSoundOrder();
|
|
}
|
|
};
|
|
|
|
const renameFolder = async () => {
|
|
let newName = await addon.tab.prompt(msg("rename-folder-prompt-title"), msg("rename-folder-prompt"), data.folder, {
|
|
useEditorClasses: true
|
|
}); // Prompt cancelled, do not rename
|
|
|
|
if (newName === null) {
|
|
return;
|
|
}
|
|
|
|
if (!isValidFolderName(newName)) {
|
|
alert(msg("name-not-allowed"));
|
|
return;
|
|
} // Empty name will remove the folder
|
|
|
|
|
|
if (!newName) {
|
|
newName = null;
|
|
}
|
|
|
|
renameItems(newName);
|
|
};
|
|
|
|
const removeFolder = () => {
|
|
renameItems(null);
|
|
};
|
|
|
|
return [{
|
|
className: "sa-folders-rename-folder",
|
|
label: msg("rename-folder"),
|
|
callback: renameFolder,
|
|
position: "assetContextMenuAfterDelete",
|
|
order: 10
|
|
}, {
|
|
className: "sa-folders-remove-folder",
|
|
label: msg("remove-folder"),
|
|
callback: removeFolder,
|
|
position: "assetContextMenuAfterDelete",
|
|
order: 11
|
|
}];
|
|
} else {
|
|
ctx.target.setAttribute("sa-folders-context-type", "asset");
|
|
|
|
const setFolder = folder => {
|
|
if (component.props.dragType === "SPRITE") {
|
|
const target = vm.runtime.getTargetById(component.props.id);
|
|
vm.renameSprite(component.props.id, ensureNotReserved(setFolderOfName(target.getName(), folder)));
|
|
fixTargetOrder();
|
|
vm.emitWorkspaceUpdate();
|
|
} else if (component.props.dragType === "COSTUME") {
|
|
const data = getItemData(component.props);
|
|
const index = data.realIndex;
|
|
const asset = vm.editingTarget.sprite.costumes[index];
|
|
vm.renameCostume(vm.editingTarget.sprite.costumes.indexOf(asset), setFolderOfName(asset.name, folder));
|
|
fixCostumeOrder();
|
|
} else if (component.props.dragType === "SOUND") {
|
|
const data = getItemData(component.props);
|
|
const index = data.realIndex;
|
|
const asset = vm.editingTarget.sprite.sounds[index];
|
|
vm.renameSound(vm.editingTarget.sprite.sounds.indexOf(asset), setFolderOfName(asset.name, folder));
|
|
fixSoundOrder();
|
|
}
|
|
};
|
|
|
|
const createFolder = async () => {
|
|
const name = await addon.tab.prompt(msg("name-prompt-title"), msg("name-prompt"), getNameWithoutFolder(data.realName), {
|
|
useEditorClasses: true
|
|
});
|
|
|
|
if (name === null) {
|
|
return;
|
|
}
|
|
|
|
if (!isValidFolderName(name)) {
|
|
alert(msg("name-not-allowed"));
|
|
return;
|
|
}
|
|
|
|
setFolder(name);
|
|
};
|
|
|
|
const base = [{
|
|
border: true,
|
|
className: "sa-folders-create-folder",
|
|
label: msg("create-folder"),
|
|
callback: createFolder,
|
|
position: "assetContextMenuAfterDelete",
|
|
order: 13
|
|
}];
|
|
const currentFolder = data.inFolder;
|
|
|
|
if (typeof currentFolder === "string") {
|
|
base.push({
|
|
className: "sa-folders-remove-from-folder",
|
|
label: msg("remove-from-folder"),
|
|
callback: () => setFolder(null),
|
|
position: "assetContextMenuAfterDelete",
|
|
order: 14
|
|
});
|
|
}
|
|
|
|
return base.concat(getAllFolders(component).filter(folder => folder !== currentFolder).map((folder, i) => {
|
|
return {
|
|
className: "sa-folders-add-to-folder",
|
|
label: msg("add-to-folder", {
|
|
folder
|
|
}),
|
|
callback: () => setFolder(folder),
|
|
position: "assetContextMenuAfterDelete",
|
|
order: 20 + i
|
|
};
|
|
}));
|
|
}
|
|
});
|
|
|
|
const patchSpriteSelectorItem = SpriteSelectorItem => {
|
|
for (const method of ["handleDelete", "handleDuplicate", "handleExport"]) {
|
|
const original = SpriteSelectorItem.prototype[method];
|
|
|
|
SpriteSelectorItem.prototype[method] = function (...args) {
|
|
if (typeof this.props.id === "number") {
|
|
const itemData = getItemData(this.props);
|
|
|
|
if (itemData) {
|
|
const originalProps = this.props;
|
|
this.props = _objectSpread(_objectSpread({}, originalProps), {}, {
|
|
id: itemData.realIndex
|
|
});
|
|
const ret = original.call(this, ...args);
|
|
this.props = originalProps;
|
|
return ret;
|
|
}
|
|
}
|
|
|
|
return original.call(this, ...args);
|
|
};
|
|
}
|
|
|
|
const originalHandleDragEnd = SpriteSelectorItem.prototype.handleDragEnd;
|
|
|
|
SpriteSelectorItem.prototype.handleDragEnd = function (...args) {
|
|
const itemData = getItemData(this.props);
|
|
|
|
if (itemData) {
|
|
if (typeof itemData.realIndex === "number" && this.props.dragging) {
|
|
// If the item is being dragged onto another group (eg. costume list -> sprite list)
|
|
// then we fake a drag event to make the `index` be the real index
|
|
const originalIndex = this.props.index;
|
|
const realIndex = itemData.realIndex;
|
|
|
|
if (originalIndex !== realIndex) {
|
|
const currentOffset = addon.tab.redux.state.scratchGui.assetDrag.currentOffset;
|
|
const sortableHOCInstance = getSortableHOCFromElement(this.ref);
|
|
|
|
if (currentOffset && sortableHOCInstance && sortableHOCInstance.getMouseOverIndex() === null) {
|
|
this.props.index = realIndex;
|
|
this.handleDrag(currentOffset);
|
|
this.props.index = originalIndex;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return originalHandleDragEnd.call(this, ...args);
|
|
};
|
|
|
|
const originalHandleClick = SpriteSelectorItem.prototype.handleClick;
|
|
|
|
SpriteSelectorItem.prototype.handleClick = function (...args) {
|
|
const e = args[0];
|
|
|
|
if (e && !this.noClick) {
|
|
const itemData = getItemData(this.props);
|
|
|
|
if (itemData) {
|
|
if (typeof itemData.folder === "string") {
|
|
e.preventDefault();
|
|
setFolderOpen(this, itemData.folder, !isFolderOpen(this, itemData.folder));
|
|
return;
|
|
}
|
|
|
|
if (typeof this.props.number === "number" && typeof itemData.realIndex === "number") {
|
|
e.preventDefault();
|
|
|
|
if (this.props.onClick) {
|
|
this.props.onClick(itemData.realIndex);
|
|
}
|
|
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
return originalHandleClick.call(this, ...args);
|
|
};
|
|
|
|
const originalRender = SpriteSelectorItem.prototype.render;
|
|
|
|
SpriteSelectorItem.prototype.render = function () {
|
|
const itemData = getItemData(this.props);
|
|
|
|
if (itemData) {
|
|
const originalProps = this.props;
|
|
this.props = _objectSpread({}, this.props);
|
|
|
|
if (typeof itemData.realName === "string") {
|
|
this.props.name = getNameWithoutFolder(itemData.realName);
|
|
}
|
|
|
|
if (typeof this.props.number === "number" && typeof itemData.realIndex === "number") {
|
|
// Convert 0-indexed to 1-indexed
|
|
this.props.number = itemData.realIndex + 1;
|
|
}
|
|
|
|
if (typeof itemData.folder === "string") {
|
|
this.props.name = itemData.folder;
|
|
|
|
if (itemData.folderOpen) {
|
|
this.props.details = msg("open-folder");
|
|
} else {
|
|
this.props.details = msg("closed-folder");
|
|
}
|
|
|
|
this.props.selected = false;
|
|
this.props.number = null;
|
|
this.props.className += " ".concat(getFolderColorClass(itemData.folder), " sa-folders-folder");
|
|
}
|
|
|
|
if (typeof itemData.inFolder === "string") {
|
|
this.props.className += " ".concat(getFolderColorClass(itemData.inFolder));
|
|
}
|
|
|
|
const result = originalRender.call(this);
|
|
this.props = originalProps;
|
|
return result;
|
|
}
|
|
|
|
return originalRender.call(this);
|
|
};
|
|
};
|
|
|
|
const patchVM = () => {
|
|
const RenderedTarget = vm.runtime.targets[0].constructor;
|
|
const originalInstallTargets = vm.installTargets;
|
|
|
|
vm.installTargets = function (...args) {
|
|
if (currentSpriteFolder !== null) {
|
|
const targets = args[0];
|
|
const wholeProject = args[2];
|
|
|
|
if (Array.isArray(targets) && !wholeProject) {
|
|
for (const target of targets) {
|
|
if (target.sprite) {
|
|
target.sprite.name = setFolderOfName(target.sprite.name, currentSpriteFolder);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return originalInstallTargets.call(this, ...args).then(r => {
|
|
fixTargetOrder();
|
|
return r;
|
|
});
|
|
};
|
|
|
|
const originalAddCostume = RenderedTarget.prototype.addCostume;
|
|
|
|
RenderedTarget.prototype.addCostume = function (...args) {
|
|
if (currentAssetFolder !== null) {
|
|
const costume = args[0];
|
|
|
|
if (costume && typeof getFolderFromName(costume.name) !== "string") {
|
|
costume.name = setFolderOfName(costume.name, currentAssetFolder);
|
|
}
|
|
}
|
|
|
|
const r = originalAddCostume.call(this, ...args);
|
|
fixCostumeOrder(this);
|
|
return r;
|
|
};
|
|
|
|
const originalAddSound = RenderedTarget.prototype.addSound;
|
|
|
|
RenderedTarget.prototype.addSound = function (...args) {
|
|
if (currentAssetFolder !== null) {
|
|
const sound = args[0];
|
|
|
|
if (sound && typeof getFolderFromName(sound.name) !== "string") {
|
|
sound.name = setFolderOfName(sound.name, currentAssetFolder);
|
|
}
|
|
}
|
|
|
|
const r = originalAddSound.call(this, ...args);
|
|
fixSoundOrder(this);
|
|
return r;
|
|
};
|
|
|
|
const abstractReorder = ({
|
|
guiItems,
|
|
getAll,
|
|
set,
|
|
rename,
|
|
getVMItemFromGUIItem,
|
|
zeroIndexed,
|
|
onFolderChanged
|
|
}, itemIndex, newIndex) => {
|
|
// First index depends on zeroIndexed
|
|
itemIndex = clamp(itemIndex, zeroIndexed ? 0 : 1, zeroIndexed ? guiItems.length - 1 : guiItems.length);
|
|
newIndex = clamp(newIndex, zeroIndexed ? 0 : 1, zeroIndexed ? guiItems.length - 1 : guiItems.length);
|
|
|
|
if (itemIndex === newIndex) {
|
|
return false;
|
|
}
|
|
|
|
let assets = getAll();
|
|
const originalAssets = getAll();
|
|
const targetItem = guiItems[itemIndex - (zeroIndexed ? 0 : 1)];
|
|
const itemAtNewIndex = guiItems[newIndex - (zeroIndexed ? 0 : 1)];
|
|
const targetItemData = getItemData(targetItem);
|
|
const itemAtNewIndexData = getItemData(itemAtNewIndex);
|
|
|
|
if (!targetItemData || !itemAtNewIndexData) {
|
|
console.warn("should never happen");
|
|
return false;
|
|
}
|
|
|
|
const reorderingItems = typeof targetItemData.folder === "string" ? targetItem.items : [targetItem];
|
|
const reorderingAssets = reorderingItems.map(i => getVMItemFromGUIItem(i, assets)).filter(i => i);
|
|
|
|
if (typeof itemAtNewIndexData.realIndex === "number") {
|
|
const newTarget = getVMItemFromGUIItem(itemAtNewIndex, assets);
|
|
|
|
if (!newTarget || reorderingAssets.includes(newTarget)) {
|
|
// Dragging folder into itself or target doesn't exist. Ignore.
|
|
return false;
|
|
}
|
|
}
|
|
|
|
let newFolder = null;
|
|
assets = assets.filter(i => !reorderingAssets.includes(i));
|
|
let realNewIndex;
|
|
|
|
if (newIndex === (zeroIndexed ? 0 : 1)) {
|
|
realNewIndex = zeroIndexed ? 0 : 1;
|
|
} else if (newIndex === guiItems.length - (zeroIndexed ? 1 : 0)) {
|
|
realNewIndex = assets.length;
|
|
} else if (typeof itemAtNewIndexData.realIndex === "number") {
|
|
newFolder = typeof itemAtNewIndexData.inFolder === "string" ? itemAtNewIndexData.inFolder : null;
|
|
let newAsset = getVMItemFromGUIItem(itemAtNewIndex, assets);
|
|
|
|
if (!newAsset) {
|
|
console.warn("should never happen");
|
|
return false;
|
|
}
|
|
|
|
realNewIndex = assets.indexOf(newAsset);
|
|
|
|
if (newIndex > itemIndex) {
|
|
realNewIndex++;
|
|
}
|
|
} else if (typeof itemAtNewIndexData.folder === "string") {
|
|
let item;
|
|
let offset = 0;
|
|
|
|
if (newIndex < itemIndex) {
|
|
// A B [C D E] F G
|
|
// ^----------*
|
|
// A B C [D] E F G
|
|
// ^--------*
|
|
item = itemAtNewIndex.items[0];
|
|
} else if (itemAtNewIndexData.folderOpen) {
|
|
// A B [C D E] F G
|
|
// *---^
|
|
item = itemAtNewIndex.items[0];
|
|
newFolder = itemAtNewIndexData.folder;
|
|
} else {
|
|
// A B [C] D E F G
|
|
// *----^
|
|
item = itemAtNewIndex.items[itemAtNewIndex.items.length - 1];
|
|
offset = 1;
|
|
}
|
|
|
|
let newAsset = getVMItemFromGUIItem(item, assets);
|
|
|
|
if (newAsset) {
|
|
realNewIndex = assets.indexOf(newAsset) + offset;
|
|
} else {
|
|
// Edge case: Dragging the first item of a list on top of the folder item
|
|
// A B [C D E] F G
|
|
// ^---*
|
|
newAsset = getVMItemFromGUIItem(item, originalAssets);
|
|
|
|
if (!newAsset) {
|
|
console.warn("should never happen");
|
|
return false;
|
|
}
|
|
|
|
realNewIndex = originalAssets.indexOf(newAsset) + offset;
|
|
}
|
|
} else {
|
|
console.warn("should never happen");
|
|
return false;
|
|
}
|
|
|
|
if (typeof targetItemData.folder === "string" && newFolder !== null) {
|
|
// Cannot drag a folder into another folder
|
|
return;
|
|
}
|
|
|
|
if (realNewIndex < (zeroIndexed ? 0 : 1) || realNewIndex > assets.length) {
|
|
console.warn("should never happen");
|
|
return false;
|
|
}
|
|
|
|
assets.splice(realNewIndex, 0, ...reorderingAssets);
|
|
set(assets); // If the folder has changed, update item names to match.
|
|
|
|
if (typeof targetItemData.folder !== "string" && targetItemData.inFolder !== newFolder) {
|
|
for (const asset of reorderingAssets) {
|
|
const name = asset.getName ? asset.getName() : asset.name;
|
|
rename(asset, setFolderOfName(name, newFolder));
|
|
}
|
|
|
|
if (onFolderChanged) {
|
|
onFolderChanged();
|
|
}
|
|
}
|
|
|
|
return true;
|
|
};
|
|
|
|
vm.constructor.prototype.reorderTarget = function (targetIndex, newIndex) {
|
|
return abstractReorder({
|
|
getAll: () => {
|
|
return this.runtime.targets;
|
|
},
|
|
set: targets => {
|
|
this.runtime.targets = targets;
|
|
this.emitTargetsUpdate();
|
|
},
|
|
rename: (item, name) => {
|
|
this.renameSprite(item.id, ensureNotReserved(name));
|
|
},
|
|
getVMItemFromGUIItem: (item, targets) => {
|
|
return targets.find(i => i.id === item.id);
|
|
},
|
|
onFolderChanged: () => {
|
|
this.emitWorkspaceUpdate();
|
|
},
|
|
guiItems: currentSpriteItems,
|
|
zeroIndexed: false
|
|
}, targetIndex, newIndex);
|
|
};
|
|
|
|
RenderedTarget.prototype.reorderCostume = function (costumeIndex, newIndex) {
|
|
return abstractReorder({
|
|
getAll: () => {
|
|
return this.sprite.costumes;
|
|
},
|
|
set: assets => {
|
|
this.sprite.costumes = assets;
|
|
},
|
|
rename: (item, name) => {
|
|
this.renameCostume(this.sprite.costumes.indexOf(item), name);
|
|
},
|
|
getVMItemFromGUIItem: (item, costumes) => {
|
|
const itemData = getItemData(item);
|
|
return costumes.find(c => c.name === itemData.realName);
|
|
},
|
|
guiItems: currentAssetItems,
|
|
zeroIndexed: true
|
|
}, costumeIndex, newIndex);
|
|
};
|
|
|
|
RenderedTarget.prototype.reorderSound = function (soundIndex, newIndex) {
|
|
return abstractReorder({
|
|
getAll: () => {
|
|
return this.sprite.sounds;
|
|
},
|
|
set: assets => {
|
|
this.sprite.sounds = assets;
|
|
},
|
|
rename: (item, name) => {
|
|
this.renameSound(this.sprite.sounds.indexOf(item), name);
|
|
},
|
|
getVMItemFromGUIItem: (item, sounds) => {
|
|
const itemData = getItemData(item);
|
|
return sounds.find(c => c.name === itemData.realName);
|
|
},
|
|
guiItems: currentAssetItems,
|
|
zeroIndexed: true
|
|
}, soundIndex, newIndex);
|
|
};
|
|
};
|
|
|
|
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("</");
|
|
const svgAttrs = asset.match(/<svg [^>]*>/);
|
|
|
|
if (svgAttrs && svgAttrs[0].indexOf("xmlns=") === -1) {
|
|
asset = asset.replace("<svg ", '<svg xmlns="http://www.w3.org/2000/svg" ');
|
|
}
|
|
|
|
const parser = new DOMParser();
|
|
const svgDom = parser.parseFromString(asset, "text/xml");
|
|
const viewBox = svgDom.documentElement.attributes.viewBox ? svgDom.documentElement.attributes.viewBox.value.match(/\S+/g) : null;
|
|
|
|
if (viewBox) {
|
|
for (let i = 0; i < viewBox.length; i++) {
|
|
viewBox[i] = parseFloat(viewBox[i]);
|
|
}
|
|
}
|
|
|
|
const handleLoad = root => {
|
|
root.opacity = opacity; // https://github.com/LLK/scratch-paint/blob/cdf0afc217633e6cfb8ba90ea4ae38b79882cf6c/src/containers/paper-canvas.jsx#L274-L275
|
|
|
|
recursePaperItem(root, i => {
|
|
if (i.className === "PathItem") {
|
|
i.clockwise = true;
|
|
}
|
|
|
|
if (i.className !== "PointText" && !i.children) {
|
|
if (i.strokeWidth) {
|
|
i.strokeWidth = i.strokeWidth * 2;
|
|
}
|
|
}
|
|
|
|
i.locked = true;
|
|
i.guide = true;
|
|
});
|
|
root.scale(2, new paper.Point(0, 0));
|
|
|
|
if (settings.mode === "tint") {
|
|
const gradients = new Set();
|
|
recursePaperItem(root, i => {
|
|
if (i.strokeColor) {
|
|
i.strokeColor = getPaperColorTint(i.strokeColor, isBefore);
|
|
}
|
|
|
|
if (i.fillColor) {
|
|
const gradient = i.fillColor.gradient;
|
|
|
|
if (gradient) {
|
|
if (gradients.has(gradient)) return;
|
|
gradients.add(gradient);
|
|
|
|
for (const stop of gradient.stops) {
|
|
stop.color = getPaperColorTint(stop.color, isBefore);
|
|
}
|
|
} else {
|
|
i.fillColor = getPaperColorTint(i.fillColor, isBefore);
|
|
}
|
|
}
|
|
|
|
if (i.canvas) {
|
|
tintRaster(i, isBefore);
|
|
}
|
|
});
|
|
}
|
|
|
|
const paperCenter = getPaperCenter(); // https://github.com/LLK/scratch-paint/blob/cdf0afc217633e6cfb8ba90ea4ae38b79882cf6c/src/containers/paper-canvas.jsx#L277-L287
|
|
|
|
if (typeof rotationCenterX !== "undefined" && typeof rotationCenterY !== "undefined") {
|
|
let rotationPoint = new paper.Point(rotationCenterX, rotationCenterY);
|
|
|
|
if (viewBox && viewBox.length >= 2 && !isNaN(viewBox[0]) && !isNaN(viewBox[1])) {
|
|
rotationPoint = rotationPoint.subtract(viewBox[0], viewBox[1]);
|
|
}
|
|
|
|
root.translate(paperCenter.subtract(rotationPoint.multiply(2)));
|
|
} else {
|
|
root.translate(paperCenter.subtract(root.bounds.width, root.bounds.height));
|
|
}
|
|
|
|
return rasterizeVector(root);
|
|
};
|
|
|
|
paper.project.importSVG(asset, {
|
|
expandShapes: true,
|
|
insert: false,
|
|
onLoad: root => {
|
|
if (!root) {
|
|
reject(new Error("could not load onion skin"));
|
|
return;
|
|
}
|
|
|
|
resolve(waitForAllRastersToLoad(root).then(() => handleLoad(root)));
|
|
}
|
|
});
|
|
});
|
|
|
|
const makeRasterOnion = (opacity, costume, asset, isBefore) => new Promise((resolve, reject) => {
|
|
let {
|
|
rotationCenterX,
|
|
rotationCenterY
|
|
} = costume;
|
|
const image = new Image();
|
|
|
|
image.onload = () => {
|
|
const paperCenter = getPaperCenter();
|
|
const width = Math.min(paperCenter.x * 2, image.width);
|
|
const height = Math.min(paperCenter.y * 2, image.height); // https://github.com/LLK/scratch-paint/blob/cdf0afc217633e6cfb8ba90ea4ae38b79882cf6c/src/containers/paper-canvas.jsx#L151-L156
|
|
|
|
if (typeof rotationCenterX === "undefined") {
|
|
rotationCenterX = width / 2;
|
|
}
|
|
|
|
if (typeof rotationCenterY === "undefined") {
|
|
rotationCenterY = height / 2;
|
|
}
|
|
|
|
const raster = new paper.Raster(image);
|
|
raster.opacity = opacity;
|
|
raster.guide = true;
|
|
raster.locked = true;
|
|
const x = width / 2 + (paperCenter.x - rotationCenterX);
|
|
const y = height / 2 + (paperCenter.y - rotationCenterY);
|
|
raster.position = new paper.Point(x, y);
|
|
raster.remove();
|
|
|
|
if (settings.mode === "tint") {
|
|
tintRaster(raster, isBefore);
|
|
}
|
|
|
|
resolve(raster);
|
|
};
|
|
|
|
image.onerror = () => {
|
|
reject(new Error("could not load image"));
|
|
};
|
|
|
|
image.src = asset;
|
|
});
|
|
|
|
const getSelectedCostumeIndex = () => {
|
|
const item = document.querySelector("[class*='selector_list-item'][class*='sprite-selector-item_is-selected']");
|
|
if (!item) return -1;
|
|
const numberEl = item.querySelector("[class*='sprite-selector-item_number']");
|
|
if (!numberEl) return -1;
|
|
return +numberEl.textContent - 1;
|
|
};
|
|
|
|
const updateOnionLayers = async () => {
|
|
const project = paper.project;
|
|
|
|
if (!project) {
|
|
return;
|
|
}
|
|
|
|
const selectedCostumeIndex = getSelectedCostumeIndex();
|
|
|
|
if (selectedCostumeIndex === -1) {
|
|
return;
|
|
}
|
|
|
|
removeOnionLayers();
|
|
const localUpdateId = cancelOngoingUpdatesAndGetNewId();
|
|
const vm = addon.tab.traps.vm;
|
|
|
|
if (!vm) {
|
|
return;
|
|
}
|
|
|
|
const originalActiveLayer = project.activeLayer;
|
|
const costumes = vm.editingTarget.sprite.costumes;
|
|
const startIndex = Math.max(0, selectedCostumeIndex - settings.previous);
|
|
const endIndex = Math.min(costumes.length - 1, selectedCostumeIndex + settings.next);
|
|
|
|
try {
|
|
const layersToCreate = [];
|
|
|
|
for (let i = startIndex; i <= endIndex; i++) {
|
|
if (i === selectedCostumeIndex) {
|
|
continue;
|
|
}
|
|
|
|
const isBefore = i < selectedCostumeIndex;
|
|
const distance = Math.abs(i - selectedCostumeIndex) - 1;
|
|
const opacity = (settings.opacity - settings.opacityStep * distance) / 100;
|
|
|
|
if (opacity <= 0) {
|
|
continue;
|
|
}
|
|
|
|
layersToCreate.push({
|
|
index: i,
|
|
isBefore,
|
|
opacity
|
|
});
|
|
}
|
|
|
|
const onions = await Promise.all(layersToCreate.map(({
|
|
index,
|
|
isBefore,
|
|
opacity
|
|
}) => {
|
|
const onionCostume = costumes[index];
|
|
const onionAsset = vm.getCostume(index);
|
|
|
|
if (onionCostume.dataFormat === "svg") {
|
|
return makeVectorOnion(opacity, onionCostume, onionAsset, isBefore);
|
|
} else if (onionCostume.dataFormat === "png" || onionCostume.dataFormat === "jpg") {
|
|
return makeRasterOnion(opacity, onionCostume, onionAsset, isBefore);
|
|
} else {
|
|
throw new Error("Unknown data format: ".concat(onionCostume.dataFormat));
|
|
}
|
|
})); // Make sure we haven't been cancelled
|
|
|
|
if (globalUpdateId === localUpdateId) {
|
|
const layer = createOnionLayer();
|
|
|
|
for (const item of onions) {
|
|
layer.addChild(item);
|
|
}
|
|
|
|
relayerOnionLayers();
|
|
}
|
|
} catch (e) {
|
|
console.error(e);
|
|
} // We must make sure to always reset the active layer to avoid corruption.
|
|
|
|
|
|
originalActiveLayer.activate();
|
|
};
|
|
|
|
const setEnabled = _enabled => {
|
|
if (settings.enabled === _enabled) {
|
|
return;
|
|
}
|
|
|
|
settings.enabled = _enabled;
|
|
|
|
if (settings.enabled) {
|
|
if (settings.next === 0 && settings.previous === 0) {
|
|
settings.previous = 1;
|
|
layerInputs.previous.value = settings.previous;
|
|
}
|
|
|
|
if (settings.opacity === 0) {
|
|
settings.opacity = 25;
|
|
layerInputs.opacity.value = settings.opacity;
|
|
}
|
|
|
|
updateOnionLayers();
|
|
} else {
|
|
removeOnionLayers();
|
|
}
|
|
|
|
toggleButton.dataset.enabled = settings.enabled;
|
|
}; //
|
|
// Controls below editor
|
|
//
|
|
|
|
|
|
const settingsChanged = onlyRelayerNeeded => {
|
|
if (settings.previous === 0 && settings.next === 0 || settings.opacity === 0) {
|
|
setEnabled(false);
|
|
return;
|
|
}
|
|
|
|
if (settings.enabled) {
|
|
if (onlyRelayerNeeded) {
|
|
relayerOnionLayers();
|
|
} else {
|
|
updateOnionLayers();
|
|
}
|
|
} else if (settings.previous > 0 || settings.next > 0) {
|
|
setEnabled(true);
|
|
}
|
|
};
|
|
|
|
const createGroup = () => {
|
|
const el = document.createElement("div");
|
|
el.className = "sa-onion-group";
|
|
return el;
|
|
};
|
|
|
|
const createButton = ({
|
|
useButtonTag
|
|
} = {}) => {
|
|
const el = document.createElement(useButtonTag ? "button" : "span");
|
|
el.className = "sa-onion-button";
|
|
el.setAttribute("role", "button");
|
|
return el;
|
|
};
|
|
|
|
const createButtonImage = name => {
|
|
const el = document.createElement("img");
|
|
el.className = "sa-onion-image";
|
|
el.draggable = false;
|
|
el.dataset.image = name;
|
|
el.loading = "lazy";
|
|
el.src = _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:
|
|
// <http://mjijackson.com/2008/02/rgb-to-hsl-and-rgb-to-hsv-color-model-conversion-algorithms-in-javascript>
|
|
// `rgbToRgb`
|
|
// Handle bounds / percentage checking to conform to CSS color spec
|
|
// <http://www.w3.org/TR/css3-color/>
|
|
// *Assumes:* r, g, b in [0, 255] or [0, 1]
|
|
// *Returns:* { r, g, b } in [0, 255]
|
|
|
|
|
|
function rgbToRgb(r, g, b) {
|
|
return {
|
|
r: bound01(r, 255) * 255,
|
|
g: bound01(g, 255) * 255,
|
|
b: bound01(b, 255) * 255
|
|
};
|
|
} // `rgbToHsl`
|
|
// Converts an RGB color value to HSL.
|
|
// *Assumes:* r, g, and b are contained in [0, 255] or [0, 1]
|
|
// *Returns:* { h, s, l } in [0,1]
|
|
|
|
|
|
function rgbToHsl(r, g, b) {
|
|
r = bound01(r, 255);
|
|
g = bound01(g, 255);
|
|
b = bound01(b, 255);
|
|
var max = mathMax(r, g, b),
|
|
min = mathMin(r, g, b);
|
|
var h,
|
|
s,
|
|
l = (max + min) / 2;
|
|
|
|
if (max == min) {
|
|
h = s = 0; // achromatic
|
|
} else {
|
|
var d = max - min;
|
|
s = l > 0.5 ? d / (2 - max - min) : d / (max + min);
|
|
|
|
switch (max) {
|
|
case r:
|
|
h = (g - b) / d + (g < b ? 6 : 0);
|
|
break;
|
|
|
|
case g:
|
|
h = (b - r) / d + 2;
|
|
break;
|
|
|
|
case b:
|
|
h = (r - g) / d + 4;
|
|
break;
|
|
}
|
|
|
|
h /= 6;
|
|
}
|
|
|
|
return {
|
|
h: h,
|
|
s: s,
|
|
l: l
|
|
};
|
|
} // `hslToRgb`
|
|
// Converts an HSL color value to RGB.
|
|
// *Assumes:* h is contained in [0, 1] or [0, 360] and s and l are contained [0, 1] or [0, 100]
|
|
// *Returns:* { r, g, b } in the set [0, 255]
|
|
|
|
|
|
function hslToRgb(h, s, l) {
|
|
var r, g, b;
|
|
h = bound01(h, 360);
|
|
s = bound01(s, 100);
|
|
l = bound01(l, 100);
|
|
|
|
function hue2rgb(p, q, t) {
|
|
if (t < 0) t += 1;
|
|
if (t > 1) t -= 1;
|
|
if (t < 1 / 6) return p + (q - p) * 6 * t;
|
|
if (t < 1 / 2) return q;
|
|
if (t < 2 / 3) return p + (q - p) * (2 / 3 - t) * 6;
|
|
return p;
|
|
}
|
|
|
|
if (s === 0) {
|
|
r = g = b = l; // achromatic
|
|
} else {
|
|
var q = l < 0.5 ? l * (1 + s) : l + s - l * s;
|
|
var p = 2 * l - q;
|
|
r = hue2rgb(p, q, h + 1 / 3);
|
|
g = hue2rgb(p, q, h);
|
|
b = hue2rgb(p, q, h - 1 / 3);
|
|
}
|
|
|
|
return {
|
|
r: r * 255,
|
|
g: g * 255,
|
|
b: b * 255
|
|
};
|
|
} // `rgbToHsv`
|
|
// Converts an RGB color value to HSV
|
|
// *Assumes:* r, g, and b are contained in the set [0, 255] or [0, 1]
|
|
// *Returns:* { h, s, v } in [0,1]
|
|
|
|
|
|
function rgbToHsv(r, g, b) {
|
|
r = bound01(r, 255);
|
|
g = bound01(g, 255);
|
|
b = bound01(b, 255);
|
|
var max = mathMax(r, g, b),
|
|
min = mathMin(r, g, b);
|
|
var h,
|
|
s,
|
|
v = max;
|
|
var d = max - min;
|
|
s = max === 0 ? 0 : d / max;
|
|
|
|
if (max == min) {
|
|
h = 0; // achromatic
|
|
} else {
|
|
switch (max) {
|
|
case r:
|
|
h = (g - b) / d + (g < b ? 6 : 0);
|
|
break;
|
|
|
|
case g:
|
|
h = (b - r) / d + 2;
|
|
break;
|
|
|
|
case b:
|
|
h = (r - g) / d + 4;
|
|
break;
|
|
}
|
|
|
|
h /= 6;
|
|
}
|
|
|
|
return {
|
|
h: h,
|
|
s: s,
|
|
v: v
|
|
};
|
|
} // `hsvToRgb`
|
|
// Converts an HSV color value to RGB.
|
|
// *Assumes:* h is contained in [0, 1] or [0, 360] and s and v are contained in [0, 1] or [0, 100]
|
|
// *Returns:* { r, g, b } in the set [0, 255]
|
|
|
|
|
|
function hsvToRgb(h, s, v) {
|
|
h = bound01(h, 360) * 6;
|
|
s = bound01(s, 100);
|
|
v = bound01(v, 100);
|
|
var i = Math.floor(h),
|
|
f = h - i,
|
|
p = v * (1 - s),
|
|
q = v * (1 - f * s),
|
|
t = v * (1 - (1 - f) * s),
|
|
mod = i % 6,
|
|
r = [v, q, p, p, t, v][mod],
|
|
g = [t, v, v, q, p, p][mod],
|
|
b = [p, p, t, v, v, q][mod];
|
|
return {
|
|
r: r * 255,
|
|
g: g * 255,
|
|
b: b * 255
|
|
};
|
|
} // `rgbToHex`
|
|
// Converts an RGB color to hex
|
|
// Assumes r, g, and b are contained in the set [0, 255]
|
|
// Returns a 3 or 6 character hex
|
|
|
|
|
|
function rgbToHex(r, g, b, allow3Char) {
|
|
var hex = [pad2(mathRound(r).toString(16)), pad2(mathRound(g).toString(16)), pad2(mathRound(b).toString(16))]; // Return a 3 character hex if possible
|
|
|
|
if (allow3Char && hex[0].charAt(0) == hex[0].charAt(1) && hex[1].charAt(0) == hex[1].charAt(1) && hex[2].charAt(0) == hex[2].charAt(1)) {
|
|
return hex[0].charAt(0) + hex[1].charAt(0) + hex[2].charAt(0);
|
|
}
|
|
|
|
return hex.join("");
|
|
} // `rgbaToHex`
|
|
// Converts an RGBA color plus alpha transparency to hex
|
|
// Assumes r, g, b are contained in the set [0, 255] and
|
|
// a in [0, 1]. Returns a 4 or 8 character rgba hex
|
|
|
|
|
|
function rgbaToHex(r, g, b, a, allow4Char) {
|
|
var hex = [pad2(mathRound(r).toString(16)), pad2(mathRound(g).toString(16)), pad2(mathRound(b).toString(16)), pad2(convertDecimalToHex(a))]; // Return a 4 character hex if possible
|
|
|
|
if (allow4Char && hex[0].charAt(0) == hex[0].charAt(1) && hex[1].charAt(0) == hex[1].charAt(1) && hex[2].charAt(0) == hex[2].charAt(1) && hex[3].charAt(0) == hex[3].charAt(1)) {
|
|
return hex[0].charAt(0) + hex[1].charAt(0) + hex[2].charAt(0) + hex[3].charAt(0);
|
|
}
|
|
|
|
return hex.join("");
|
|
} // `rgbaToArgbHex`
|
|
// Converts an RGBA color to an ARGB Hex8 string
|
|
// Rarely used, but required for "toFilter()"
|
|
|
|
|
|
function rgbaToArgbHex(r, g, b, a) {
|
|
var hex = [pad2(convertDecimalToHex(a)), pad2(mathRound(r).toString(16)), pad2(mathRound(g).toString(16)), pad2(mathRound(b).toString(16))];
|
|
return hex.join("");
|
|
} // `equals`
|
|
// Can be called with any tinycolor input
|
|
|
|
|
|
tinycolor.equals = function (color1, color2) {
|
|
if (!color1 || !color2) {
|
|
return false;
|
|
}
|
|
|
|
return tinycolor(color1).toRgbString() == tinycolor(color2).toRgbString();
|
|
};
|
|
|
|
tinycolor.random = function () {
|
|
return tinycolor.fromRatio({
|
|
r: mathRandom(),
|
|
g: mathRandom(),
|
|
b: mathRandom()
|
|
});
|
|
}; // Modification Functions
|
|
// ----------------------
|
|
// Thanks to less.js for some of the basics here
|
|
// <https://github.com/cloudhead/less.js/blob/master/lib/less/functions.js>
|
|
|
|
|
|
function _desaturate(color, amount) {
|
|
amount = amount === 0 ? 0 : amount || 10;
|
|
var hsl = tinycolor(color).toHsl();
|
|
hsl.s -= amount / 100;
|
|
hsl.s = clamp01(hsl.s);
|
|
return tinycolor(hsl);
|
|
}
|
|
|
|
function _saturate(color, amount) {
|
|
amount = amount === 0 ? 0 : amount || 10;
|
|
var hsl = tinycolor(color).toHsl();
|
|
hsl.s += amount / 100;
|
|
hsl.s = clamp01(hsl.s);
|
|
return tinycolor(hsl);
|
|
}
|
|
|
|
function _greyscale(color) {
|
|
return tinycolor(color).desaturate(100);
|
|
}
|
|
|
|
function _lighten(color, amount) {
|
|
amount = amount === 0 ? 0 : amount || 10;
|
|
var hsl = tinycolor(color).toHsl();
|
|
hsl.l += amount / 100;
|
|
hsl.l = clamp01(hsl.l);
|
|
return tinycolor(hsl);
|
|
}
|
|
|
|
function _brighten(color, amount) {
|
|
amount = amount === 0 ? 0 : amount || 10;
|
|
var rgb = tinycolor(color).toRgb();
|
|
rgb.r = mathMax(0, mathMin(255, rgb.r - mathRound(255 * -(amount / 100))));
|
|
rgb.g = mathMax(0, mathMin(255, rgb.g - mathRound(255 * -(amount / 100))));
|
|
rgb.b = mathMax(0, mathMin(255, rgb.b - mathRound(255 * -(amount / 100))));
|
|
return tinycolor(rgb);
|
|
}
|
|
|
|
function _darken(color, amount) {
|
|
amount = amount === 0 ? 0 : amount || 10;
|
|
var hsl = tinycolor(color).toHsl();
|
|
hsl.l -= amount / 100;
|
|
hsl.l = clamp01(hsl.l);
|
|
return tinycolor(hsl);
|
|
} // Spin takes a positive or negative amount within [-360, 360] indicating the change of hue.
|
|
// Values outside of this range will be wrapped into this range.
|
|
|
|
|
|
function _spin(color, amount) {
|
|
var hsl = tinycolor(color).toHsl();
|
|
var hue = (hsl.h + amount) % 360;
|
|
hsl.h = hue < 0 ? 360 + hue : hue;
|
|
return tinycolor(hsl);
|
|
} // Combination Functions
|
|
// ---------------------
|
|
// Thanks to jQuery xColor for some of the ideas behind these
|
|
// <https://github.com/infusion/jQuery-xcolor/blob/master/jquery.xcolor.js>
|
|
|
|
|
|
function _complement(color) {
|
|
var hsl = tinycolor(color).toHsl();
|
|
hsl.h = (hsl.h + 180) % 360;
|
|
return tinycolor(hsl);
|
|
}
|
|
|
|
function _triad(color) {
|
|
var hsl = tinycolor(color).toHsl();
|
|
var h = hsl.h;
|
|
return [tinycolor(color), tinycolor({
|
|
h: (h + 120) % 360,
|
|
s: hsl.s,
|
|
l: hsl.l
|
|
}), tinycolor({
|
|
h: (h + 240) % 360,
|
|
s: hsl.s,
|
|
l: hsl.l
|
|
})];
|
|
}
|
|
|
|
function _tetrad(color) {
|
|
var hsl = tinycolor(color).toHsl();
|
|
var h = hsl.h;
|
|
return [tinycolor(color), tinycolor({
|
|
h: (h + 90) % 360,
|
|
s: hsl.s,
|
|
l: hsl.l
|
|
}), tinycolor({
|
|
h: (h + 180) % 360,
|
|
s: hsl.s,
|
|
l: hsl.l
|
|
}), tinycolor({
|
|
h: (h + 270) % 360,
|
|
s: hsl.s,
|
|
l: hsl.l
|
|
})];
|
|
}
|
|
|
|
function _splitcomplement(color) {
|
|
var hsl = tinycolor(color).toHsl();
|
|
var h = hsl.h;
|
|
return [tinycolor(color), tinycolor({
|
|
h: (h + 72) % 360,
|
|
s: hsl.s,
|
|
l: hsl.l
|
|
}), tinycolor({
|
|
h: (h + 216) % 360,
|
|
s: hsl.s,
|
|
l: hsl.l
|
|
})];
|
|
}
|
|
|
|
function _analogous(color, results, slices) {
|
|
results = results || 6;
|
|
slices = slices || 30;
|
|
var hsl = tinycolor(color).toHsl();
|
|
var part = 360 / slices;
|
|
var ret = [tinycolor(color)];
|
|
|
|
for (hsl.h = (hsl.h - (part * results >> 1) + 720) % 360; --results;) {
|
|
hsl.h = (hsl.h + part) % 360;
|
|
ret.push(tinycolor(hsl));
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
function _monochromatic(color, results) {
|
|
results = results || 6;
|
|
var hsv = tinycolor(color).toHsv();
|
|
var h = hsv.h,
|
|
s = hsv.s,
|
|
v = hsv.v;
|
|
var ret = [];
|
|
var modification = 1 / results;
|
|
|
|
while (results--) {
|
|
ret.push(tinycolor({
|
|
h: h,
|
|
s: s,
|
|
v: v
|
|
}));
|
|
v = (v + modification) % 1;
|
|
}
|
|
|
|
return ret;
|
|
} // Utility Functions
|
|
// ---------------------
|
|
|
|
|
|
tinycolor.mix = function (color1, color2, amount) {
|
|
amount = amount === 0 ? 0 : amount || 50;
|
|
var rgb1 = tinycolor(color1).toRgb();
|
|
var rgb2 = tinycolor(color2).toRgb();
|
|
var p = amount / 100;
|
|
var rgba = {
|
|
r: (rgb2.r - rgb1.r) * p + rgb1.r,
|
|
g: (rgb2.g - rgb1.g) * p + rgb1.g,
|
|
b: (rgb2.b - rgb1.b) * p + rgb1.b,
|
|
a: (rgb2.a - rgb1.a) * p + rgb1.a
|
|
};
|
|
return tinycolor(rgba);
|
|
}; // Readability Functions
|
|
// ---------------------
|
|
// <http://www.w3.org/TR/2008/REC-WCAG20-20081211/#contrast-ratiodef (WCAG Version 2)
|
|
// `contrast`
|
|
// Analyze the 2 colors and returns the color contrast defined by (WCAG Version 2)
|
|
|
|
|
|
tinycolor.readability = function (color1, color2) {
|
|
var c1 = tinycolor(color1);
|
|
var c2 = tinycolor(color2);
|
|
return (Math.max(c1.getLuminance(), c2.getLuminance()) + 0.05) / (Math.min(c1.getLuminance(), c2.getLuminance()) + 0.05);
|
|
}; // `isReadable`
|
|
// Ensure that foreground and background color combinations meet WCAG2 guidelines.
|
|
// The third argument is an optional Object.
|
|
// the 'level' property states 'AA' or 'AAA' - if missing or invalid, it defaults to 'AA';
|
|
// the 'size' property states 'large' or 'small' - if missing or invalid, it defaults to 'small'.
|
|
// If the entire object is absent, isReadable defaults to {level:"AA",size:"small"}.
|
|
// *Example*
|
|
// tinycolor.isReadable("#000", "#111") => false
|
|
// tinycolor.isReadable("#000", "#111",{level:"AA",size:"large"}) => false
|
|
|
|
|
|
tinycolor.isReadable = function (color1, color2, wcag2) {
|
|
var readability = tinycolor.readability(color1, color2);
|
|
var wcag2Parms, out;
|
|
out = false;
|
|
wcag2Parms = validateWCAG2Parms(wcag2);
|
|
|
|
switch (wcag2Parms.level + wcag2Parms.size) {
|
|
case "AAsmall":
|
|
case "AAAlarge":
|
|
out = readability >= 4.5;
|
|
break;
|
|
|
|
case "AAlarge":
|
|
out = readability >= 3;
|
|
break;
|
|
|
|
case "AAAsmall":
|
|
out = readability >= 7;
|
|
break;
|
|
}
|
|
|
|
return out;
|
|
}; // `mostReadable`
|
|
// Given a base color and a list of possible foreground or background
|
|
// colors for that base, returns the most readable color.
|
|
// Optionally returns Black or White if the most readable color is unreadable.
|
|
// *Example*
|
|
// tinycolor.mostReadable(tinycolor.mostReadable("#123", ["#124", "#125"],{includeFallbackColors:false}).toHexString(); // "#112255"
|
|
// tinycolor.mostReadable(tinycolor.mostReadable("#123", ["#124", "#125"],{includeFallbackColors:true}).toHexString(); // "#ffffff"
|
|
// tinycolor.mostReadable("#a8015a", ["#faf3f3"],{includeFallbackColors:true,level:"AAA",size:"large"}).toHexString(); // "#faf3f3"
|
|
// tinycolor.mostReadable("#a8015a", ["#faf3f3"],{includeFallbackColors:true,level:"AAA",size:"small"}).toHexString(); // "#ffffff"
|
|
|
|
|
|
tinycolor.mostReadable = function (baseColor, colorList, args) {
|
|
var bestColor = null;
|
|
var bestScore = 0;
|
|
var readability;
|
|
var includeFallbackColors, level, size;
|
|
args = args || {};
|
|
includeFallbackColors = args.includeFallbackColors;
|
|
level = args.level;
|
|
size = args.size;
|
|
|
|
for (var i = 0; i < colorList.length; i++) {
|
|
readability = tinycolor.readability(baseColor, colorList[i]);
|
|
|
|
if (readability > bestScore) {
|
|
bestScore = readability;
|
|
bestColor = tinycolor(colorList[i]);
|
|
}
|
|
}
|
|
|
|
if (tinycolor.isReadable(baseColor, bestColor, {
|
|
"level": level,
|
|
"size": size
|
|
}) || !includeFallbackColors) {
|
|
return bestColor;
|
|
} else {
|
|
args.includeFallbackColors = false;
|
|
return tinycolor.mostReadable(baseColor, ["#fff", "#000"], args);
|
|
}
|
|
}; // Big List of Colors
|
|
// ------------------
|
|
// <http://www.w3.org/TR/css3-color/#svg-color>
|
|
|
|
|
|
var names = tinycolor.names = {
|
|
aliceblue: "f0f8ff",
|
|
antiquewhite: "faebd7",
|
|
aqua: "0ff",
|
|
aquamarine: "7fffd4",
|
|
azure: "f0ffff",
|
|
beige: "f5f5dc",
|
|
bisque: "ffe4c4",
|
|
black: "000",
|
|
blanchedalmond: "ffebcd",
|
|
blue: "00f",
|
|
blueviolet: "8a2be2",
|
|
brown: "a52a2a",
|
|
burlywood: "deb887",
|
|
burntsienna: "ea7e5d",
|
|
cadetblue: "5f9ea0",
|
|
chartreuse: "7fff00",
|
|
chocolate: "d2691e",
|
|
coral: "ff7f50",
|
|
cornflowerblue: "6495ed",
|
|
cornsilk: "fff8dc",
|
|
crimson: "dc143c",
|
|
cyan: "0ff",
|
|
darkblue: "00008b",
|
|
darkcyan: "008b8b",
|
|
darkgoldenrod: "b8860b",
|
|
darkgray: "a9a9a9",
|
|
darkgreen: "006400",
|
|
darkgrey: "a9a9a9",
|
|
darkkhaki: "bdb76b",
|
|
darkmagenta: "8b008b",
|
|
darkolivegreen: "556b2f",
|
|
darkorange: "ff8c00",
|
|
darkorchid: "9932cc",
|
|
darkred: "8b0000",
|
|
darksalmon: "e9967a",
|
|
darkseagreen: "8fbc8f",
|
|
darkslateblue: "483d8b",
|
|
darkslategray: "2f4f4f",
|
|
darkslategrey: "2f4f4f",
|
|
darkturquoise: "00ced1",
|
|
darkviolet: "9400d3",
|
|
deeppink: "ff1493",
|
|
deepskyblue: "00bfff",
|
|
dimgray: "696969",
|
|
dimgrey: "696969",
|
|
dodgerblue: "1e90ff",
|
|
firebrick: "b22222",
|
|
floralwhite: "fffaf0",
|
|
forestgreen: "228b22",
|
|
fuchsia: "f0f",
|
|
gainsboro: "dcdcdc",
|
|
ghostwhite: "f8f8ff",
|
|
gold: "ffd700",
|
|
goldenrod: "daa520",
|
|
gray: "808080",
|
|
green: "008000",
|
|
greenyellow: "adff2f",
|
|
grey: "808080",
|
|
honeydew: "f0fff0",
|
|
hotpink: "ff69b4",
|
|
indianred: "cd5c5c",
|
|
indigo: "4b0082",
|
|
ivory: "fffff0",
|
|
khaki: "f0e68c",
|
|
lavender: "e6e6fa",
|
|
lavenderblush: "fff0f5",
|
|
lawngreen: "7cfc00",
|
|
lemonchiffon: "fffacd",
|
|
lightblue: "add8e6",
|
|
lightcoral: "f08080",
|
|
lightcyan: "e0ffff",
|
|
lightgoldenrodyellow: "fafad2",
|
|
lightgray: "d3d3d3",
|
|
lightgreen: "90ee90",
|
|
lightgrey: "d3d3d3",
|
|
lightpink: "ffb6c1",
|
|
lightsalmon: "ffa07a",
|
|
lightseagreen: "20b2aa",
|
|
lightskyblue: "87cefa",
|
|
lightslategray: "789",
|
|
lightslategrey: "789",
|
|
lightsteelblue: "b0c4de",
|
|
lightyellow: "ffffe0",
|
|
lime: "0f0",
|
|
limegreen: "32cd32",
|
|
linen: "faf0e6",
|
|
magenta: "f0f",
|
|
maroon: "800000",
|
|
mediumaquamarine: "66cdaa",
|
|
mediumblue: "0000cd",
|
|
mediumorchid: "ba55d3",
|
|
mediumpurple: "9370db",
|
|
mediumseagreen: "3cb371",
|
|
mediumslateblue: "7b68ee",
|
|
mediumspringgreen: "00fa9a",
|
|
mediumturquoise: "48d1cc",
|
|
mediumvioletred: "c71585",
|
|
midnightblue: "191970",
|
|
mintcream: "f5fffa",
|
|
mistyrose: "ffe4e1",
|
|
moccasin: "ffe4b5",
|
|
navajowhite: "ffdead",
|
|
navy: "000080",
|
|
oldlace: "fdf5e6",
|
|
olive: "808000",
|
|
olivedrab: "6b8e23",
|
|
orange: "ffa500",
|
|
orangered: "ff4500",
|
|
orchid: "da70d6",
|
|
palegoldenrod: "eee8aa",
|
|
palegreen: "98fb98",
|
|
paleturquoise: "afeeee",
|
|
palevioletred: "db7093",
|
|
papayawhip: "ffefd5",
|
|
peachpuff: "ffdab9",
|
|
peru: "cd853f",
|
|
pink: "ffc0cb",
|
|
plum: "dda0dd",
|
|
powderblue: "b0e0e6",
|
|
purple: "800080",
|
|
rebeccapurple: "663399",
|
|
red: "f00",
|
|
rosybrown: "bc8f8f",
|
|
royalblue: "4169e1",
|
|
saddlebrown: "8b4513",
|
|
salmon: "fa8072",
|
|
sandybrown: "f4a460",
|
|
seagreen: "2e8b57",
|
|
seashell: "fff5ee",
|
|
sienna: "a0522d",
|
|
silver: "c0c0c0",
|
|
skyblue: "87ceeb",
|
|
slateblue: "6a5acd",
|
|
slategray: "708090",
|
|
slategrey: "708090",
|
|
snow: "fffafa",
|
|
springgreen: "00ff7f",
|
|
steelblue: "4682b4",
|
|
tan: "d2b48c",
|
|
teal: "008080",
|
|
thistle: "d8bfd8",
|
|
tomato: "ff6347",
|
|
turquoise: "40e0d0",
|
|
violet: "ee82ee",
|
|
wheat: "f5deb3",
|
|
white: "fff",
|
|
whitesmoke: "f5f5f5",
|
|
yellow: "ff0",
|
|
yellowgreen: "9acd32"
|
|
}; // Make it easy to access colors via `hexNames[hex]`
|
|
|
|
var hexNames = tinycolor.hexNames = flip(names); // Utilities
|
|
// ---------
|
|
// `{ 'name1': 'val1' }` becomes `{ 'val1': 'name1' }`
|
|
|
|
function flip(o) {
|
|
var flipped = {};
|
|
|
|
for (var i in o) {
|
|
if (o.hasOwnProperty(i)) {
|
|
flipped[o[i]] = i;
|
|
}
|
|
}
|
|
|
|
return flipped;
|
|
} // Return a valid alpha value [0,1] with all invalid values being set to 1
|
|
|
|
|
|
function boundAlpha(a) {
|
|
a = parseFloat(a);
|
|
|
|
if (isNaN(a) || a < 0 || a > 1) {
|
|
a = 1;
|
|
}
|
|
|
|
return a;
|
|
} // Take input from [0, n] and return it as [0, 1]
|
|
|
|
|
|
function bound01(n, max) {
|
|
if (isOnePointZero(n)) {
|
|
n = "100%";
|
|
}
|
|
|
|
var processPercent = isPercentage(n);
|
|
n = mathMin(max, mathMax(0, parseFloat(n))); // Automatically convert percentage into number
|
|
|
|
if (processPercent) {
|
|
n = parseInt(n * max, 10) / 100;
|
|
} // Handle floating point rounding errors
|
|
|
|
|
|
if (Math.abs(n - max) < 0.000001) {
|
|
return 1;
|
|
} // Convert into [0, 1] range if it isn't already
|
|
|
|
|
|
return n % max / parseFloat(max);
|
|
} // Force a number between 0 and 1
|
|
|
|
|
|
function clamp01(val) {
|
|
return mathMin(1, mathMax(0, val));
|
|
} // Parse a base-16 hex value into a base-10 integer
|
|
|
|
|
|
function parseIntFromHex(val) {
|
|
return parseInt(val, 16);
|
|
} // Need to handle 1.0 as 100%, since once it is a number, there is no difference between it and 1
|
|
// <http://stackoverflow.com/questions/7422072/javascript-how-to-detect-number-as-a-decimal-including-1-0>
|
|
|
|
|
|
function isOnePointZero(n) {
|
|
return typeof n == "string" && n.indexOf('.') != -1 && parseFloat(n) === 1;
|
|
} // Check to see if string passed in is a percentage
|
|
|
|
|
|
function isPercentage(n) {
|
|
return typeof n === "string" && n.indexOf('%') != -1;
|
|
} // Force a hex value to have 2 characters
|
|
|
|
|
|
function pad2(c) {
|
|
return c.length == 1 ? '0' + c : '' + c;
|
|
} // Replace a decimal with it's percentage value
|
|
|
|
|
|
function convertToPercentage(n) {
|
|
if (n <= 1) {
|
|
n = n * 100 + "%";
|
|
}
|
|
|
|
return n;
|
|
} // Converts a decimal to a hex value
|
|
|
|
|
|
function convertDecimalToHex(d) {
|
|
return Math.round(parseFloat(d) * 255).toString(16);
|
|
} // Converts a hex value to a decimal
|
|
|
|
|
|
function convertHexToDecimal(h) {
|
|
return parseIntFromHex(h) / 255;
|
|
}
|
|
|
|
var matchers = function () {
|
|
// <http://www.w3.org/TR/css3-values/#integers>
|
|
var CSS_INTEGER = "[-\\+]?\\d+%?"; // <http://www.w3.org/TR/css3-values/#number-value>
|
|
|
|
var CSS_NUMBER = "[-\\+]?\\d*\\.\\d+%?"; // Allow positive/negative integer/number. Don't capture the either/or, just the entire outcome.
|
|
|
|
var CSS_UNIT = "(?:" + CSS_NUMBER + ")|(?:" + CSS_INTEGER + ")"; // Actual matching.
|
|
// Parentheses and commas are optional, but not required.
|
|
// Whitespace can take the place of commas or opening paren
|
|
|
|
var PERMISSIVE_MATCH3 = "[\\s|\\(]+(" + CSS_UNIT + ")[,|\\s]+(" + CSS_UNIT + ")[,|\\s]+(" + CSS_UNIT + ")\\s*\\)?";
|
|
var PERMISSIVE_MATCH4 = "[\\s|\\(]+(" + CSS_UNIT + ")[,|\\s]+(" + CSS_UNIT + ")[,|\\s]+(" + CSS_UNIT + ")[,|\\s]+(" + CSS_UNIT + ")\\s*\\)?";
|
|
return {
|
|
CSS_UNIT: new RegExp(CSS_UNIT),
|
|
rgb: new RegExp("rgb" + PERMISSIVE_MATCH3),
|
|
rgba: new RegExp("rgba" + PERMISSIVE_MATCH4),
|
|
hsl: new RegExp("hsl" + PERMISSIVE_MATCH3),
|
|
hsla: new RegExp("hsla" + PERMISSIVE_MATCH4),
|
|
hsv: new RegExp("hsv" + PERMISSIVE_MATCH3),
|
|
hsva: new RegExp("hsva" + PERMISSIVE_MATCH4),
|
|
hex3: /^#?([0-9a-fA-F]{1})([0-9a-fA-F]{1})([0-9a-fA-F]{1})$/,
|
|
hex6: /^#?([0-9a-fA-F]{2})([0-9a-fA-F]{2})([0-9a-fA-F]{2})$/,
|
|
hex4: /^#?([0-9a-fA-F]{1})([0-9a-fA-F]{1})([0-9a-fA-F]{1})([0-9a-fA-F]{1})$/,
|
|
hex8: /^#?([0-9a-fA-F]{2})([0-9a-fA-F]{2})([0-9a-fA-F]{2})([0-9a-fA-F]{2})$/
|
|
};
|
|
}(); // `isValidCSSUnit`
|
|
// Take in a single string / number and check to see if it looks like a CSS unit
|
|
// (see `matchers` above for definition).
|
|
|
|
|
|
function isValidCSSUnit(color) {
|
|
return !!matchers.CSS_UNIT.exec(color);
|
|
} // `stringInputToObject`
|
|
// Permissive string parsing. Take in a number of formats, and output an object
|
|
// based on detected format. Returns `{ r, g, b }` or `{ h, s, l }` or `{ h, s, v}`
|
|
|
|
|
|
function stringInputToObject(color) {
|
|
color = color.replace(trimLeft, '').replace(trimRight, '').toLowerCase();
|
|
var named = false;
|
|
|
|
if (names[color]) {
|
|
color = names[color];
|
|
named = true;
|
|
} else if (color == 'transparent') {
|
|
return {
|
|
r: 0,
|
|
g: 0,
|
|
b: 0,
|
|
a: 0,
|
|
format: "name"
|
|
};
|
|
} // Try to match string input using regular expressions.
|
|
// Keep most of the number bounding out of this function - don't worry about [0,1] or [0,100] or [0,360]
|
|
// Just return an object and let the conversion functions handle that.
|
|
// This way the result will be the same whether the tinycolor is initialized with string or object.
|
|
|
|
|
|
var match;
|
|
|
|
if (match = matchers.rgb.exec(color)) {
|
|
return {
|
|
r: match[1],
|
|
g: match[2],
|
|
b: match[3]
|
|
};
|
|
}
|
|
|
|
if (match = matchers.rgba.exec(color)) {
|
|
return {
|
|
r: match[1],
|
|
g: match[2],
|
|
b: match[3],
|
|
a: match[4]
|
|
};
|
|
}
|
|
|
|
if (match = matchers.hsl.exec(color)) {
|
|
return {
|
|
h: match[1],
|
|
s: match[2],
|
|
l: match[3]
|
|
};
|
|
}
|
|
|
|
if (match = matchers.hsla.exec(color)) {
|
|
return {
|
|
h: match[1],
|
|
s: match[2],
|
|
l: match[3],
|
|
a: match[4]
|
|
};
|
|
}
|
|
|
|
if (match = matchers.hsv.exec(color)) {
|
|
return {
|
|
h: match[1],
|
|
s: match[2],
|
|
v: match[3]
|
|
};
|
|
}
|
|
|
|
if (match = matchers.hsva.exec(color)) {
|
|
return {
|
|
h: match[1],
|
|
s: match[2],
|
|
v: match[3],
|
|
a: match[4]
|
|
};
|
|
}
|
|
|
|
if (match = matchers.hex8.exec(color)) {
|
|
return {
|
|
r: parseIntFromHex(match[1]),
|
|
g: parseIntFromHex(match[2]),
|
|
b: parseIntFromHex(match[3]),
|
|
a: convertHexToDecimal(match[4]),
|
|
format: named ? "name" : "hex8"
|
|
};
|
|
}
|
|
|
|
if (match = matchers.hex6.exec(color)) {
|
|
return {
|
|
r: parseIntFromHex(match[1]),
|
|
g: parseIntFromHex(match[2]),
|
|
b: parseIntFromHex(match[3]),
|
|
format: named ? "name" : "hex"
|
|
};
|
|
}
|
|
|
|
if (match = matchers.hex4.exec(color)) {
|
|
return {
|
|
r: parseIntFromHex(match[1] + '' + match[1]),
|
|
g: parseIntFromHex(match[2] + '' + match[2]),
|
|
b: parseIntFromHex(match[3] + '' + match[3]),
|
|
a: convertHexToDecimal(match[4] + '' + match[4]),
|
|
format: named ? "name" : "hex8"
|
|
};
|
|
}
|
|
|
|
if (match = matchers.hex3.exec(color)) {
|
|
return {
|
|
r: parseIntFromHex(match[1] + '' + match[1]),
|
|
g: parseIntFromHex(match[2] + '' + match[2]),
|
|
b: parseIntFromHex(match[3] + '' + match[3]),
|
|
format: named ? "name" : "hex"
|
|
};
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
function validateWCAG2Parms(parms) {
|
|
// return valid WCAG2 parms for isReadable.
|
|
// If input parms are invalid, return {"level":"AA", "size":"small"}
|
|
var level, size;
|
|
parms = parms || {
|
|
"level": "AA",
|
|
"size": "small"
|
|
};
|
|
level = (parms.level || "AA").toUpperCase();
|
|
size = (parms.size || "small").toLowerCase();
|
|
|
|
if (level !== "AA" && level !== "AAA") {
|
|
level = "AA";
|
|
}
|
|
|
|
if (size !== "small" && size !== "large") {
|
|
size = "small";
|
|
}
|
|
|
|
return {
|
|
"level": level,
|
|
"size": size
|
|
};
|
|
}
|
|
/*// Node: Export function
|
|
if (typeof module !== "undefined" && module.exports) {
|
|
module.exports = tinycolor;
|
|
}
|
|
// AMD/requirejs: Define the module
|
|
else if (typeof define === 'function' && define.amd) {
|
|
define(function () {return tinycolor;});
|
|
}
|
|
// Browser: Expose to window
|
|
else {
|
|
window.tinycolor = tinycolor;
|
|
}*/
|
|
|
|
|
|
return tinycolor;
|
|
}(Math);
|
|
|
|
/* harmony default export */ __webpack_exports__["default"] = (tinycolor);
|
|
|
|
/***/ })
|
|
|
|
}]);
|
|
//# sourceMappingURL=addon-default-entry.js.map
|