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

2130 lines
90 KiB
JavaScript

(window["webpackJsonpGUI"] = window["webpackJsonpGUI"] || []).push([["addon-entry-debugger"],{
/***/ "./node_modules/css-loader/index.js!./src/addons/addons/debugger/style.css":
/*!************************************************************************!*\
!*** ./node_modules/css-loader!./src/addons/addons/debugger/style.css ***!
\************************************************************************/
/*! no static exports found */
/***/ (function(module, exports, __webpack_require__) {
var escape = __webpack_require__(/*! ../../../../node_modules/css-loader/lib/url/escape.js */ "./node_modules/css-loader/lib/url/escape.js");
exports = module.exports = __webpack_require__(/*! ../../../../node_modules/css-loader/lib/css-base.js */ "./node_modules/css-loader/lib/css-base.js")(false);
// imports
exports.i(__webpack_require__(/*! -!../../../../node_modules/css-loader!../editor-theme3/compatibility.css */ "./node_modules/css-loader/index.js!./src/addons/addons/editor-theme3/compatibility.css"), "");
// module
exports.push([module.i, "[dir=\"ltr\"] .sa-debugger-container {\n margin-right: 0.2rem;\n}\n\n[dir=\"rtl\"] .sa-debugger-container {\n margin-left: 0.2rem;\n}\n\n.sa-debugger-small .sa-debugger-container {\n display: none !important;\n}\n\n.sa-debugger-container [class*=\"button_content_\"] {\n position: relative;\n}\n\n.sa-debugger-unread::after {\n content: \"\";\n position: absolute;\n top: 1px;\n right: 0;\n display: block;\n width: 6px;\n height: 6px;\n background-color: var(--editorDarkMode-highlightText, #4d97ff);\n border-radius: 50%;\n}\n\n.sa-debugger-interface {\n display: none;\n position: absolute;\n z-index: 492;\n background-color: white;\n width: 565px;\n height: 25rem;\n}\n[theme=\"dark\"] .sa-debugger-interface {\n background: var(--ui-primary);\n}\n\n.sa-debugger-interface [class*=\"card_header-buttons_\"] {\n background-color: #29beb8;\n border-color: #3aa8a4;\n}\n\n.sa-debugger-interface h1 {\n padding: 10px;\n z-index: 10;\n width: calc(100% - 20px);\n font-size: 20px;\n}\n\n.sa-debugger-tabs {\n margin: 0;\n display: flex;\n align-items: center;\n padding: 0 15px;\n font-size: 0.75rem;\n}\n.sa-debugger-tabs li {\n margin: 0;\n display: flex;\n align-items: center;\n padding: 0.5em 1em;\n background-color: rgba(0, 0, 0, 0.1);\n border: 1px solid rgba(0, 0, 0, 0.15);\n border-radius: 1rem;\n color: white;\n cursor: pointer;\n}\n.sa-debugger-tabs li + li {\n margin-inline-start: 10px;\n}\n.sa-debugger-tabs li:hover {\n background-color: rgba(0, 0, 0, 0.15);\n}\n.sa-debugger-tabs li.sa-debugger-tab-selected {\n background-color: white;\n background-clip: padding-box;\n border-color: rgba(0, 0, 0, 0.25);\n color: #4d97ff;\n}\n.sa-debugger-tabs li img {\n margin: 0;\n margin-right: 0.25rem;\n width: 1rem;\n filter: brightness(0) invert(1);\n}\n.sa-debugger-tabs li.sa-debugger-tab-selected img {\n filter: none;\n}\n\n.sa-debugger-header-buttons img {\n width: 20px;\n height: 20px;\n}\n\n.sa-debugger-unpause {\n animation: saDebuggerUnpause 2s infinite alternate;\n}\n\n@keyframes saDebuggerUnpause {\n 0% {\n background-color: rgba(0, 0, 0, 0.15);\n }\n 100% {\n background-color: rgba(0, 0, 0, 0);\n }\n}\n\n.sa-debugger-tab-content {\n width: 100%;\n height: 100%;\n overflow: auto;\n cursor: auto;\n}\n\n.sa-debugger-chart {\n width: 100%;\n height: 100%;\n}\n\n.sa-performance-tab-content {\n padding: 15px;\n}\n\n.sa-debugger-log-outer {\n height: 100%;\n}\n\n.sa-debugger-log-inner {\n position: relative;\n overflow-y: auto;\n font-size: 12px;\n line-height: 1.2;\n height: 100%;\n contain: strict;\n}\n\n.sa-debugger-log-empty {\n display: flex;\n align-items: center;\n justify-content: center;\n width: 100%;\n height: 100%;\n font-size: 20px;\n font-style: italic;\n}\n\n.sa-debugger-log-end {\n position: absolute;\n top: 0;\n left: 0;\n width: 1px;\n height: 1px;\n}\n\n.sa-debugger-log {\n position: absolute;\n top: 0;\n left: 0;\n width: 100%;\n height: 20px;\n box-sizing: border-box;\n display: flex;\n align-items: center;\n border-bottom: 1px solid rgba(0, 0, 0, 0.15);\n padding-left: 4px;\n font-family: monospace;\n color: #000;\n}\n[theme=\"dark\"] .sa-debugger-log {\n color: var(--text-primary);\n border-color: rgba(255, 255, 255, 0.15);\n}\n.sa-debugger-log[data-type=\"warn\"] {\n border-color: hsl(50deg, 100%, 75%);\n color: hsl(39deg 100% 18%);\n background-color: hsl(50deg 100% 95%);\n}\n.sa-debugger-log[data-type=\"error\"] {\n border-color: hsl(0deg 100% 92%);\n color: red;\n background-color: hsl(0deg 100% 95%);\n}\n[theme=\"dark\"] .sa-debugger-log[data-type=\"warn\"] {\n border-color: hsl(50deg, 100%, 15%);\n color: hsl(39deg 100% 90%);\n background-color: hsl(50deg 100% 10%);\n}\n[theme=\"dark\"] .sa-debugger-log[data-type=\"error\"] {\n border-color: hsl(0deg 100% 15%);\n color: hsl(0deg 100% 77%);\n background-color: hsl(0deg 100% 10%);\n}\n\n.sa-debugger-log-repeats {\n background-color: hsla(163, 85%, 40%, 1);\n color: white;\n border-radius: 100px;\n padding: 1px 6px;\n margin-right: 4px;\n}\n[theme=\"dark\"] .sa-debugger-log-repeats {\n color: var(--ui-primary);\n}\n\n.sa-debugger-log-icon {\n width: 16px;\n height: 16px;\n margin-right: 4px;\n}\n[data-type=\"warn\"] .sa-debugger-log-icon {\n background-image: url(" + escape(__webpack_require__(/*! ./icons/warning.svg */ "./src/addons/addons/debugger/icons/warning.svg")) + ");\n}\n[data-type=\"error\"] .sa-debugger-log-icon {\n background-image: url(" + escape(__webpack_require__(/*! ./icons/error.svg */ "./src/addons/addons/debugger/icons/error.svg")) + ");\n}\n.sa-debugger-threads .sa-debugger-log-icon {\n background-image: url(" + escape(__webpack_require__(/*! ./icons/subthread.svg */ "./src/addons/addons/debugger/icons/subthread.svg")) + ");\n}\n\n.sa-debugger-log-link {\n color: inherit;\n cursor: pointer;\n opacity: 0.5;\n text-decoration: underline;\n float: right;\n text-align: right;\n max-width: 100%;\n padding-left: 4px;\n margin-right: 4px;\n margin-left: auto;\n}\n[theme=\"dark\"] .sa-debugger-log-link {\n color: inherit;\n}\n.sa-debugger-log-link:hover {\n text-decoration: underline;\n color: #4d97ff;\n opacity: 1;\n}\n.sa-debugger-log-link-unknown {\n pointer-events: none;\n}\n\n.sa-debugger-log-text {\n overflow: hidden;\n text-overflow: ellipsis;\n white-space: pre;\n}\n.sa-debugger-log-text-empty {\n font-style: italic;\n}\n.sa-debugger-log-internal .sa-debugger-log-text {\n font-style: italic;\n}\n\n.sa-debugger-thread-indent {\n width: calc(16px * var(--level));\n margin-right: 4px;\n}\n.sa-debugger-thread-title .sa-debugger-thread-indent {\n margin: 0;\n}\n.sa-debugger-thread-target-name {\n font-weight: bold;\n margin-right: 8px;\n}\n.sa-debugger-thread-running {\n background-color: rgba(255, 187, 0, 0.233);\n font-weight: bold;\n}\n\n.sa-debugger-block-preview {\n padding: 1px 6px;\n margin-right: 4px;\n background-color: var(--sa-block-colored-background);\n color: var(--sa-block-text);\n}\n.sa-debugger-block-preview[data-shape=\"round\"] {\n border-radius: 100px;\n}\n.sa-debugger-block-preview[data-shape=\"stacked\"] {\n border-radius: 3px;\n}\n\n.sa-debugger-thread-compiled {\n font-style: italic;\n}\n\n.sa-debugger-compiler-warning {\n position: relative;\n display: block;\n text-align: center;\n height: 24px;\n color: #2121bf;\n}\n.sa-debugger-compiler-warning[hidden] {\n display: none;\n}\n[theme=\"dark\"] .sa-debugger-compiler-warning {\n color: #bdbdf9;\n}\n", ""]);
// exports
/***/ }),
/***/ "./node_modules/css-loader/index.js!./src/addons/addons/editor-theme3/compatibility.css":
/*!*************************************************************************************!*\
!*** ./node_modules/css-loader!./src/addons/addons/editor-theme3/compatibility.css ***!
\*************************************************************************************/
/*! no static exports found */
/***/ (function(module, exports, __webpack_require__) {
exports = module.exports = __webpack_require__(/*! ../../../../node_modules/css-loader/lib/css-base.js */ "./node_modules/css-loader/lib/css-base.js")(false);
// imports
// module
exports.push([module.i, "/* Imported by other addons */\n\n.sa-block-color {\n --sa-block-colored-background: var(--sa-block-background-primary);\n --sa-block-colored-background-secondary: var(--sa-block-field-background);\n --sa-block-bright-background: var(--sa-block-background-primary);\n --sa-block-text: white;\n --sa-block-gray-text: white;\n --sa-block-colored-text: var(--sa-block-background-primary);\n --sa-block-text-on-bright-background: white;\n}\n\n.sa-block-color-motion {\n --sa-block-background-primary: var(--editorTheme3-motion-primary, #4c97ff);\n --sa-block-background-secondary: var(--editorTheme3-motion-secondary, #4280d7);\n --sa-block-background-tertiary: var(--editorTheme3-motion-tertiary, #3373cc);\n --sa-block-field-background: var(--editorTheme3-motion-field, #3373cc);\n}\n\n.sa-block-color-looks {\n --sa-block-background-primary: var(--editorTheme3-looks-primary, #9966ff);\n --sa-block-background-secondary: var(--editorTheme3-looks-secondary, #855cd6);\n --sa-block-background-tertiary: var(--editorTheme3-looks-tertiary, #774dcb);\n --sa-block-field-background: var(--editorTheme3-looks-field, #774dcb);\n}\n\n.sa-block-color-sounds {\n --sa-block-background-primary: var(--editorTheme3-sounds-primary, #cf63cf);\n --sa-block-background-secondary: var(--editorTheme3-sounds-secondary, #c94fc9);\n --sa-block-background-tertiary: var(--editorTheme3-sounds-tertiary, #bd42bd);\n --sa-block-field-background: var(--editorTheme3-sounds-field, #bd42bd);\n}\n\n.sa-block-color-events {\n --sa-block-background-primary: var(--editorTheme3-event-primary, #ffbf00);\n --sa-block-background-secondary: var(--editorTheme3-event-secondary, #e6ac00);\n --sa-block-background-tertiary: var(--editorTheme3-event-tertiary, #cc9900);\n --sa-block-field-background: var(--editorTheme3-event-field, #cc9900);\n}\n\n.sa-block-color-control {\n --sa-block-background-primary: var(--editorTheme3-control-primary, #ffab19);\n --sa-block-background-secondary: var(--editorTheme3-control-secondary, #ec9c13);\n --sa-block-background-tertiary: var(--editorTheme3-control-tertiary, #cf8b17);\n --sa-block-field-background: var(--editorTheme3-control-field, #cf8b17);\n}\n\n.sa-block-color-sensing {\n --sa-block-background-primary: var(--editorTheme3-sensing-primary, #5cb1d6);\n --sa-block-background-secondary: var(--editorTheme3-sensing-secondary, #47a8d1);\n --sa-block-background-tertiary: var(--editorTheme3-sensing-tertiary, #2e8eb8);\n --sa-block-field-background: var(--editorTheme3-sensing-field, #2e8eb8);\n}\n\n.sa-block-color-operators {\n --sa-block-background-primary: var(--editorTheme3-operators-primary, #59c059);\n --sa-block-background-secondary: var(--editorTheme3-operators-secondary, #46b946);\n --sa-block-background-tertiary: var(--editorTheme3-operators-tertiary, #389438);\n --sa-block-field-background: var(--editorTheme3-operators-field, #389438);\n}\n\n.sa-block-color-data {\n --sa-block-background-primary: var(--editorTheme3-data-primary, #ff8c1a);\n --sa-block-background-secondary: var(--editorTheme3-data-secondary, #ff8000);\n --sa-block-background-tertiary: var(--editorTheme3-data-tertiary, #db6e00);\n --sa-block-field-background: var(--editorTheme3-data-field, #db6e00);\n}\n\n.sa-block-color-data-lists,\n.sa-block-color-list {\n --sa-block-background-primary: var(--editorTheme3-data_lists-primary, #ff661a);\n --sa-block-background-secondary: var(--editorTheme3-data_lists-secondary, #ff5500);\n --sa-block-background-tertiary: var(--editorTheme3-data_lists-tertiary, #e64d00);\n --sa-block-field-background: var(--editorTheme3-data_lists-field, #e64d00);\n}\n\n.sa-block-color-more,\n.sa-block-color-null {\n --sa-block-background-primary: var(--editorTheme3-more-primary, #ff6680);\n --sa-block-background-secondary: var(--editorTheme3-more-secondary, #ff4d6a);\n --sa-block-background-tertiary: var(--editorTheme3-more-tertiary, #ff3355);\n --sa-block-field-background: var(--editorTheme3-more-field, #ff3355);\n}\n\n.sa-block-color-pen {\n --sa-block-background-primary: var(--editorTheme3-pen-primary, #0fbd8c);\n --sa-block-background-secondary: var(--editorTheme3-pen-secondary, #0da57a);\n --sa-block-background-tertiary: var(--editorTheme3-pen-tertiary, #0b8e69);\n --sa-block-field-background: var(--editorTheme3-pen-field, #0b8e69);\n}\n\n.sa-block-color-addon-custom-block {\n --sa-block-background-primary: var(--editorTheme3-sa-primary, #29beb8);\n --sa-block-background-secondary: var(--editorTheme3-sa-secondary, #3aa8a4);\n --sa-block-background-tertiary: var(--editorTheme3-sa-tertiary, #3aa8a4);\n --sa-block-field-background: var(--editorTheme3-sa-field, #3aa8a4);\n}\n\n.sa-block-color-TurboWarp {\n --sa-block-background-primary: var(--editorTheme3-tw-primary, #ff4c4c);\n --sa-block-background-secondary: var(--editorTheme3-tw-secondary, #e64444);\n --sa-block-background-tertiary: var(--editorTheme3-tw-tertiary, #e64444);\n --sa-block-field-background: var(--editorTheme3-tw-field, #e64444);\n}\n", ""]);
// exports
/***/ }),
/***/ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/debugger/icons/close.svg":
/*!******************************************************************************************!*\
!*** ./node_modules/url-loader/dist/cjs.js!./src/addons/addons/debugger/icons/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/debugger/icons/debug.svg":
/*!******************************************************************************************!*\
!*** ./node_modules/url-loader/dist/cjs.js!./src/addons/addons/debugger/icons/debug.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/debugger/icons/delete.svg":
/*!*******************************************************************************************!*\
!*** ./node_modules/url-loader/dist/cjs.js!./src/addons/addons/debugger/icons/delete.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/debugger/icons/download-white.svg":
/*!***************************************************************************************************!*\
!*** ./node_modules/url-loader/dist/cjs.js!./src/addons/addons/debugger/icons/download-white.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/debugger/icons/error.svg":
/*!******************************************************************************************!*\
!*** ./node_modules/url-loader/dist/cjs.js!./src/addons/addons/debugger/icons/error.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/debugger/icons/logs.svg":
/*!*****************************************************************************************!*\
!*** ./node_modules/url-loader/dist/cjs.js!./src/addons/addons/debugger/icons/logs.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/debugger/icons/performance.svg":
/*!************************************************************************************************!*\
!*** ./node_modules/url-loader/dist/cjs.js!./src/addons/addons/debugger/icons/performance.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/debugger/icons/play.svg":
/*!*****************************************************************************************!*\
!*** ./node_modules/url-loader/dist/cjs.js!./src/addons/addons/debugger/icons/play.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/debugger/icons/step.svg":
/*!*****************************************************************************************!*\
!*** ./node_modules/url-loader/dist/cjs.js!./src/addons/addons/debugger/icons/step.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/debugger/icons/subthread.svg":
/*!**********************************************************************************************!*\
!*** ./node_modules/url-loader/dist/cjs.js!./src/addons/addons/debugger/icons/subthread.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/debugger/icons/threads.svg":
/*!********************************************************************************************!*\
!*** ./node_modules/url-loader/dist/cjs.js!./src/addons/addons/debugger/icons/threads.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/debugger/icons/warning.svg":
/*!********************************************************************************************!*\
!*** ./node_modules/url-loader/dist/cjs.js!./src/addons/addons/debugger/icons/warning.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/debugger/_runtime_entry.js":
/*!******************************************************!*\
!*** ./src/addons/addons/debugger/_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/debugger/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/debugger/style.css");
/* harmony import */ var _css_loader_style_css__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_css_loader_style_css__WEBPACK_IMPORTED_MODULE_1__);
/* harmony import */ var _url_loader_icons_close_svg__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! url-loader!./icons/close.svg */ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/debugger/icons/close.svg");
/* harmony import */ var _url_loader_icons_debug_svg__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! url-loader!./icons/debug.svg */ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/debugger/icons/debug.svg");
/* harmony import */ var _url_loader_icons_delete_svg__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! url-loader!./icons/delete.svg */ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/debugger/icons/delete.svg");
/* harmony import */ var _url_loader_icons_download_white_svg__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! url-loader!./icons/download-white.svg */ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/debugger/icons/download-white.svg");
/* harmony import */ var _url_loader_icons_error_svg__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(/*! url-loader!./icons/error.svg */ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/debugger/icons/error.svg");
/* harmony import */ var _url_loader_icons_logs_svg__WEBPACK_IMPORTED_MODULE_7__ = __webpack_require__(/*! url-loader!./icons/logs.svg */ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/debugger/icons/logs.svg");
/* harmony import */ var _url_loader_icons_performance_svg__WEBPACK_IMPORTED_MODULE_8__ = __webpack_require__(/*! url-loader!./icons/performance.svg */ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/debugger/icons/performance.svg");
/* harmony import */ var _url_loader_icons_play_svg__WEBPACK_IMPORTED_MODULE_9__ = __webpack_require__(/*! url-loader!./icons/play.svg */ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/debugger/icons/play.svg");
/* harmony import */ var _url_loader_icons_step_svg__WEBPACK_IMPORTED_MODULE_10__ = __webpack_require__(/*! url-loader!./icons/step.svg */ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/debugger/icons/step.svg");
/* harmony import */ var _url_loader_icons_subthread_svg__WEBPACK_IMPORTED_MODULE_11__ = __webpack_require__(/*! url-loader!./icons/subthread.svg */ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/debugger/icons/subthread.svg");
/* harmony import */ var _url_loader_icons_threads_svg__WEBPACK_IMPORTED_MODULE_12__ = __webpack_require__(/*! url-loader!./icons/threads.svg */ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/debugger/icons/threads.svg");
/* harmony import */ var _url_loader_icons_warning_svg__WEBPACK_IMPORTED_MODULE_13__ = __webpack_require__(/*! url-loader!./icons/warning.svg */ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/debugger/icons/warning.svg");
/* generated by pull.js */
const resources = {
"userscript.js": _userscript_js__WEBPACK_IMPORTED_MODULE_0__["default"],
"style.css": _css_loader_style_css__WEBPACK_IMPORTED_MODULE_1___default.a,
"icons/close.svg": _url_loader_icons_close_svg__WEBPACK_IMPORTED_MODULE_2__["default"],
"icons/debug.svg": _url_loader_icons_debug_svg__WEBPACK_IMPORTED_MODULE_3__["default"],
"icons/delete.svg": _url_loader_icons_delete_svg__WEBPACK_IMPORTED_MODULE_4__["default"],
"icons/download-white.svg": _url_loader_icons_download_white_svg__WEBPACK_IMPORTED_MODULE_5__["default"],
"icons/error.svg": _url_loader_icons_error_svg__WEBPACK_IMPORTED_MODULE_6__["default"],
"icons/logs.svg": _url_loader_icons_logs_svg__WEBPACK_IMPORTED_MODULE_7__["default"],
"icons/performance.svg": _url_loader_icons_performance_svg__WEBPACK_IMPORTED_MODULE_8__["default"],
"icons/play.svg": _url_loader_icons_play_svg__WEBPACK_IMPORTED_MODULE_9__["default"],
"icons/step.svg": _url_loader_icons_step_svg__WEBPACK_IMPORTED_MODULE_10__["default"],
"icons/subthread.svg": _url_loader_icons_subthread_svg__WEBPACK_IMPORTED_MODULE_11__["default"],
"icons/threads.svg": _url_loader_icons_threads_svg__WEBPACK_IMPORTED_MODULE_12__["default"],
"icons/warning.svg": _url_loader_icons_warning_svg__WEBPACK_IMPORTED_MODULE_13__["default"]
};
/***/ }),
/***/ "./src/addons/addons/debugger/icons/error.svg":
/*!****************************************************!*\
!*** ./src/addons/addons/debugger/icons/error.svg ***!
\****************************************************/
/*! no static exports found */
/***/ (function(module, exports, __webpack_require__) {
module.exports = __webpack_require__.p + "static/assets/76b6cb627b95d79705c0b41664064f0e.svg";
/***/ }),
/***/ "./src/addons/addons/debugger/icons/subthread.svg":
/*!********************************************************!*\
!*** ./src/addons/addons/debugger/icons/subthread.svg ***!
\********************************************************/
/*! no static exports found */
/***/ (function(module, exports, __webpack_require__) {
module.exports = __webpack_require__.p + "static/assets/c846a442121113b1d04f6b9d50912038.svg";
/***/ }),
/***/ "./src/addons/addons/debugger/icons/warning.svg":
/*!******************************************************!*\
!*** ./src/addons/addons/debugger/icons/warning.svg ***!
\******************************************************/
/*! no static exports found */
/***/ (function(module, exports, __webpack_require__) {
module.exports = __webpack_require__.p + "static/assets/0e009d6e684951615b31a267baa37636.svg";
/***/ }),
/***/ "./src/addons/addons/debugger/log-view.js":
/*!************************************************!*\
!*** ./src/addons/addons/debugger/log-view.js ***!
\************************************************/
/*! exports provided: default */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
const clamp = (i, min, max) => Math.max(min, Math.min(max, i));
const appendSortedElement = (parent, newChild) => {
const newChildIndex = +newChild.dataset.index;
let foundSpot = false;
for (const existingChild of parent.children) {
const existingChildIndex = +existingChild.dataset.index;
if (existingChildIndex > newChildIndex) {
foundSpot = true;
parent.insertBefore(newChild, existingChild);
break;
}
}
if (!foundSpot) {
parent.appendChild(newChild);
}
};
/**
* LogView: A virtualized row viewer.
* It efficiently manages row rendering and scrolling.
*
* 1. .logs is the place where all the rows live. This is an array of any arbitrary object.
* 2. Implement generateRow(row). This takes a row from .logs as an argument. This should return
* an object with a bunch of DOM elements on it. The "root" property must be set, nothing else
* is required. This is called when a row becomes visible. It can be called any number of times.
* This is where you should setup elements that are immutable for a given row. LogView will
* move the root element to the right spot for you.
* 3. Implement renderRow(elements, row). This will be called with the result returned by
* generateRow() and the row in .logs any time a row is changed, including the first render.
* It can be called any number of times. This is where you should update any dynamic elements.
* 4. Whenever you update .logs without using the helper methods such as append(), call
* queueUpdateContent().
*/
class LogView {
constructor() {
this.rows = [];
this.canAutoScrollToEnd = true;
this.rowHeight = 20;
this.outerElement = document.createElement("div");
this.outerElement.className = "sa-debugger-log-outer";
this.innerElement = document.createElement("div");
this.innerElement.className = "sa-debugger-log-inner";
this.outerElement.appendChild(this.innerElement);
this.innerElement.addEventListener("scroll", this._handleScroll.bind(this), {
passive: true
});
this.innerElement.addEventListener("wheel", this._handleWheel.bind(this), {
passive: true
});
this.endElement = document.createElement("div");
this.endElement.className = "sa-debugger-log-end";
this.endElement.dataset.index = "-1";
this.innerElement.appendChild(this.endElement);
this.placeholderElement = document.createElement("div");
this.placeholderElement.className = "sa-debugger-log-empty";
this.visible = false;
this.isScrolledToEnd = true;
this.scrollTopWhenHidden = "end";
this.scrollTop = 0;
this.updateContentQueued = false;
this.scrollToEndQueued = false;
this.oldLength = -1;
this.rowToMetadata = new Map();
}
append(log) {
this.queueUpdateContent();
this._queueScrollToEnd();
this.rows.push(log);
const MAX_LOGS = 200000;
while (this.rows.length > MAX_LOGS) {
this.rows.shift();
}
}
clear() {
this.rows.length = 0;
this.scrollTop = 0;
this.isScrolledToEnd = true;
this.queueUpdateContent();
}
show() {
this.visible = true;
this.height = this.innerElement.offsetHeight;
this.queueUpdateContent();
if (this.scrollTopWhenHidden === "end") {
this._queueScrollToEnd();
} else {
this.innerElement.scrollTop = this.scrollTopWhenHidden;
}
}
hide() {
this.visible = false;
this.scrollTopWhenHidden = this.isScrolledToEnd ? "end" : this.scrollTop;
}
_handleScroll(e) {
this.scrollTop = e.target.scrollTop;
this.isScrolledToEnd = e.target.scrollTop + 5 >= e.target.scrollHeight - e.target.clientHeight;
this.queueUpdateContent();
}
_handleWheel(e) {
if (e.deltaY < 0) {
this.isScrolledToEnd = false;
}
}
scrollIntoView(index) {
const distanceFromTop = index * this.rowHeight;
const viewportStart = this.scrollTop;
const viewportEnd = this.scrollTop + this.height;
const isInView = distanceFromTop > viewportStart && distanceFromTop < viewportEnd;
if (!isInView) {
this.scrollTop = distanceFromTop;
this.innerElement.scrollTop = distanceFromTop;
}
}
_queueScrollToEnd() {
if (this.visible && this.canAutoScrollToEnd && this.isScrolledToEnd && !this.scrollToEndQueued) {
this.scrollToEndQueued = true;
queueMicrotask(() => {
this.scrollToEndQueued = false;
if (this.isScrolledToEnd) {
const scrollEnd = this.innerElement.scrollHeight - this.innerElement.offsetHeight;
this.innerElement.scrollTop = scrollEnd;
this.scrollTop = scrollEnd;
}
});
}
}
queueUpdateContent() {
if (this.visible && !this.updateContentQueued) {
this.updateContentQueued = true;
queueMicrotask(() => {
this.updateContentQueued = false;
this.updateContent();
});
}
}
generateRow(row) {// to be implemented by users
}
renderRow(elements, row) {// to be implemented by users
}
updateContent() {
if (this.rows.length !== this.oldLength) {
this.oldLength = this.rows.length;
const totalHeight = this.rows.length * this.rowHeight;
this.endElement.style.transform = "translateY(".concat(totalHeight, "px)");
if (this.rows.length) {
this.placeholderElement.remove();
} else {
this.innerElement.appendChild(this.placeholderElement);
for (const metadata of this.rowToMetadata.values()) {
metadata.elements.root.remove();
}
this.rowToMetadata.clear();
}
}
if (this.rows.length === 0) {
return;
} // For better compatibility with asynchronous scrolling, we'll render a few extra rows in either direction.
const EXTRA_ROWS_ABOVE = 5;
const EXTRA_ROWS_BELOW = 5;
const scrollStartIndex = Math.floor(this.scrollTop / this.rowHeight);
const rowsVisible = Math.ceil(this.height / this.rowHeight);
const startIndex = clamp(scrollStartIndex - EXTRA_ROWS_BELOW, 0, this.rows.length);
const endIndex = clamp(scrollStartIndex + rowsVisible + EXTRA_ROWS_ABOVE, 0, this.rows.length);
const allVisibleRows = new Set();
const newElements = [];
for (let i = startIndex; i < endIndex; i++) {
const row = this.rows[i];
allVisibleRows.add(row);
let metadata = this.rowToMetadata.get(row);
if (!metadata) {
const elements = this.generateRow(row);
newElements.push(elements.root);
metadata = {
stringify: null,
elements
};
this.rowToMetadata.set(row, metadata);
}
const currentStringify = JSON.stringify(row);
if (currentStringify !== metadata.stringify) {
metadata.stringify = currentStringify;
this.renderRow(metadata.elements, row);
}
const root = metadata.elements.root;
root.style.transform = "translateY(".concat(i * this.rowHeight, "px)");
root.dataset.index = i;
}
for (const [row, metadata] of this.rowToMetadata.entries()) {
if (!allVisibleRows.has(row)) {
metadata.elements.root.remove();
this.rowToMetadata.delete(row);
}
}
for (const root of newElements) {
appendSortedElement(this.innerElement, root);
}
}
}
/* harmony default export */ __webpack_exports__["default"] = (LogView);
/***/ }),
/***/ "./src/addons/addons/debugger/logs.js":
/*!********************************************!*\
!*** ./src/addons/addons/debugger/logs.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 createLogsTab; });
/* harmony import */ var _libraries_common_cs_download_blob_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../../libraries/common/cs/download-blob.js */ "./src/addons/libraries/common/cs/download-blob.js");
/* harmony import */ var _log_view_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./log-view.js */ "./src/addons/addons/debugger/log-view.js");
async function createLogsTab({
debug,
addon,
console,
msg
}) {
const vm = addon.tab.traps.vm;
const tab = debug.createHeaderTab({
text: msg("tab-logs"),
icon: addon.self.getResource("/icons/logs.svg")
/* rewritten by pull.js */
});
const logView = new _log_view_js__WEBPACK_IMPORTED_MODULE_1__["default"]();
logView.placeholderElement.textContent = msg("no-logs");
const getInputOfBlock = (targetId, blockId) => {
var _Object$values$;
const target = vm.runtime.getTargetById(targetId);
if (!target) {
return null;
}
const block = debug.getBlock(target, blockId);
if (!block) {
return null;
}
return (_Object$values$ = Object.values(block.inputs)[0]) === null || _Object$values$ === void 0 ? void 0 : _Object$values$.block;
};
logView.generateRow = row => {
const root = document.createElement("div");
root.className = "sa-debugger-log";
if (row.internal) {
root.classList.add("sa-debugger-log-internal");
}
root.dataset.type = row.type;
const icon = document.createElement("div");
icon.className = "sa-debugger-log-icon";
if (row.type === "warn" || row.type === "error") {
icon.title = msg("icon-" + row.type);
}
root.appendChild(icon);
const repeats = document.createElement("div");
repeats.className = "sa-debugger-log-repeats";
repeats.style.display = "none";
root.appendChild(repeats);
if (row.preview && row.blockId && row.targetInfo) {
const originalId = row.targetInfo.originalId;
const inputBlock = getInputOfBlock(originalId, row.blockId);
if (inputBlock) {
const preview = debug.createBlockPreview(originalId, inputBlock);
if (preview) {
root.appendChild(preview);
}
}
}
const text = document.createElement("div");
text.className = "sa-debugger-log-text";
if (row.text.length === 0) {
text.classList.add("sa-debugger-log-text-empty");
text.textContent = msg("empty-string");
} else {
text.textContent = row.text;
text.title = row.text;
}
root.appendChild(text);
if (row.targetInfo && row.blockId) {
root.appendChild(debug.createBlockLink(row.targetInfo, row.blockId));
}
return {
root,
repeats
};
};
logView.renderRow = (elements, row) => {
const {
repeats
} = elements;
if (row.count > 1) {
repeats.style.display = "";
repeats.textContent = row.count;
}
};
const exportButton = debug.createHeaderButton({
text: msg("export"),
icon: addon.self.getResource("/icons/download-white.svg")
/* rewritten by pull.js */
,
description: msg("export-desc")
});
const downloadText = (filename, text) => {
Object(_libraries_common_cs_download_blob_js__WEBPACK_IMPORTED_MODULE_0__["default"])(filename, new Blob([text], {
type: "text/plain"
}));
};
exportButton.element.addEventListener("click", async e => {
const defaultFormat = "{sprite}: {content} ({type})";
const exportFormat = e.shiftKey ? await addon.tab.prompt(msg("export"), msg("enter-format"), defaultFormat, {
useEditorClasses: true
}) : defaultFormat;
if (!exportFormat) return;
const file = logView.rows.map(({
text,
targetInfo,
type,
count
}) => (exportFormat.replace(/\{(sprite|type|content)\}/g, (_, match) => ({
sprite: targetInfo ? targetInfo.name : msg("unknown-sprite"),
type,
content: text
})[match]) + "\n").repeat(count)).join("");
downloadText("logs.txt", file);
});
const trashButton = debug.createHeaderButton({
text: msg("clear"),
icon: addon.self.getResource("/icons/delete.svg")
/* rewritten by pull.js */
});
trashButton.element.addEventListener("click", () => {
clearLogs();
});
const areLogsEqual = (a, b) => a.text === b.text && a.type === b.type && a.internal === b.internal && a.blockId === b.blockId && a.targetId === b.targetId;
const addLog = (text, thread, type) => {
const log = {
text,
type,
count: 1,
preview: true
};
if (thread) {
log.blockId = thread.peekStack();
const targetId = thread.target.id;
log.targetId = targetId;
log.targetInfo = debug.getTargetInfoById(targetId);
}
if (type === "internal") {
log.internal = true;
log.preview = false;
log.type = "log";
}
if (type === "internal-warn") {
log.internal = true;
log.type = "warn";
}
const previousLog = logView.rows[logView.rows.length - 1];
if (previousLog && areLogsEqual(log, previousLog)) {
previousLog.count++;
logView.queueUpdateContent();
} else {
logView.append(log);
}
if (!logView.visible && !log.internal) {
debug.setHasUnreadMessage(true);
}
};
const clearLogs = () => {
logView.clear();
};
const show = () => {
logView.show();
debug.setHasUnreadMessage(false);
};
const hide = () => {
logView.hide();
};
return {
tab,
content: logView.outerElement,
buttons: [exportButton, trashButton],
show,
hide,
addLog,
clearLogs
};
}
/***/ }),
/***/ "./src/addons/addons/debugger/performance.js":
/*!***************************************************!*\
!*** ./src/addons/addons/debugger/performance.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 createPerformanceTab; });
/* harmony import */ var _module_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./module.js */ "./src/addons/addons/debugger/module.js");
async function createPerformanceTab({
debug,
addon,
console,
msg
}) {
const vm = addon.tab.traps.vm;
await addon.tab.loadScript(addon.self.getResource("/thirdparty/cs/chart.min.js"))
/* rewritten by pull.js */
;
const tab = debug.createHeaderTab({
text: msg("tab-performance"),
icon: addon.self.getResource("/icons/performance.svg")
/* rewritten by pull.js */
});
const content = Object.assign(document.createElement("div"), {
className: "sa-performance-tab-content"
});
const createChart = ({
title
}) => {
const titleElement = Object.assign(document.createElement("h2"), {
textContent: title
});
const canvas = Object.assign(document.createElement("canvas"), {
className: "sa-debugger-chart"
});
return {
title: titleElement,
canvas
};
};
const now = () => performance.now();
const getMaxFps = () => Math.round(1000 / vm.runtime.currentStepTime);
const NUMBER_OF_POINTS = 20; // An array like [9, 8, 7, 6, 5, 4, 3, 2, 1, 0]
const labels = Array.from(Array(NUMBER_OF_POINTS).keys()).reverse();
const fpsElements = createChart({
title: msg("performance-framerate-title")
});
const fpsChart = new Chart(fpsElements.canvas.getContext("2d"), {
type: "line",
data: {
labels,
datasets: [{
data: Array(NUMBER_OF_POINTS).fill(-1),
borderWidth: 1,
fill: true,
backgroundColor: "#29beb8"
}]
},
options: {
scales: {
y: {
max: getMaxFps(),
min: 0
}
},
plugins: {
legend: {
display: false
},
tooltip: {
callbacks: {
label: context => msg("performance-framerate-graph-tooltip", {
fps: context.parsed.y
})
}
}
}
}
});
const clonesElements = createChart({
title: msg("performance-clonecount-title")
});
const performanceClonesChart = new Chart(clonesElements.canvas.getContext("2d"), {
type: "line",
data: {
labels,
datasets: [{
data: Array(NUMBER_OF_POINTS).fill(-1),
borderWidth: 1,
fill: true,
backgroundColor: "#29beb8"
}]
},
options: {
scales: {
y: {
max: 300,
min: 0
}
},
plugins: {
legend: {
display: false
},
tooltip: {
callbacks: {
label: context => msg("performance-clonecount-graph-tooltip", {
clones: context.parsed.y
})
}
}
}
}
}); // Holds the times of each frame drawn in the last second.
// The length of this list is effectively the FPS.
const renderTimes = []; // The last time we pushed a new datapoint to the graph
let lastFpsTime = now() + 3000;
debug.addAfterStepCallback(() => {
if (Object(_module_js__WEBPACK_IMPORTED_MODULE_0__["isPaused"])()) {
return;
}
const time = now(); // Remove all frame times older than 1 second in renderTimes
while (renderTimes.length > 0 && renderTimes[0] <= time - 1000) renderTimes.shift();
renderTimes.push(time);
if (time - lastFpsTime > 1000) {
lastFpsTime = time;
const maxFps = getMaxFps();
const fpsData = fpsChart.data.datasets[0].data;
fpsData.shift();
fpsData.push(Math.min(renderTimes.length, maxFps)); // Incase we switch between 30FPS and 60FPS, update the max height of the chart.
fpsChart.options.scales.y.max = maxFps;
const clonesData = performanceClonesChart.data.datasets[0].data;
clonesData.shift();
clonesData.push(vm.runtime._cloneCounter);
if (isVisible) {
fpsChart.update();
performanceClonesChart.update();
}
}
});
content.appendChild(fpsElements.title);
content.appendChild(fpsElements.canvas);
content.appendChild(clonesElements.title);
content.appendChild(clonesElements.canvas);
let pauseTime = 0;
Object(_module_js__WEBPACK_IMPORTED_MODULE_0__["onPauseChanged"])(paused => {
if (paused) {
pauseTime = now();
} else {
const dt = now() - pauseTime;
lastFpsTime += dt;
for (var i = 0; i < renderTimes.length; i++) {
renderTimes[i] += dt;
}
}
});
let isVisible = false;
const show = () => {
isVisible = true;
};
const hide = () => {
isVisible = false;
};
return {
tab,
content,
buttons: [],
show,
hide
};
}
/***/ }),
/***/ "./src/addons/addons/debugger/threads.js":
/*!***********************************************!*\
!*** ./src/addons/addons/debugger/threads.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 createThreadsTab; });
/* harmony import */ var _module_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./module.js */ "./src/addons/addons/debugger/module.js");
/* harmony import */ var _log_view_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./log-view.js */ "./src/addons/addons/debugger/log-view.js");
/* harmony import */ var _editor_stepping_highlighter_js__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ../editor-stepping/highlighter.js */ "./src/addons/addons/editor-stepping/highlighter.js");
const concatInPlace = (copyInto, copyFrom) => {
for (const i of copyFrom) {
copyInto.push(i);
}
};
async function createThreadsTab({
debug,
addon,
console,
msg
}) {
const vm = addon.tab.traps.vm;
const tab = debug.createHeaderTab({
text: msg("tab-threads"),
icon: addon.self.getResource("/icons/threads.svg")
/* rewritten by pull.js */
});
const logView = new _log_view_js__WEBPACK_IMPORTED_MODULE_1__["default"]();
logView.canAutoScrollToEnd = false;
logView.outerElement.classList.add("sa-debugger-threads");
logView.placeholderElement.textContent = msg("no-threads-running");
const highlighter = new _editor_stepping_highlighter_js__WEBPACK_IMPORTED_MODULE_2__["default"](10, "#ff0000");
logView.generateRow = row => {
const root = document.createElement("div");
root.className = "sa-debugger-log";
const isHeader = row.type === "thread-header";
const indenter = document.createElement("div");
indenter.className = "sa-debugger-thread-indent";
indenter.style.setProperty("--level", isHeader ? row.depth : row.depth + 1);
root.appendChild(indenter);
if (isHeader) {
root.classList.add("sa-debugger-thread-title");
if (row.depth > 0) {
const icon = document.createElement("div");
icon.className = "sa-debugger-log-icon";
root.appendChild(icon);
}
const name = document.createElement("div");
name.textContent = row.targetName;
name.className = "sa-debugger-thread-target-name";
root.appendChild(name);
const id = document.createElement("div");
id.className = "sa-debugger-thread-id";
id.textContent = msg("thread", {
id: row.id
});
root.appendChild(id);
}
if (row.type === "thread-stack") {
const preview = debug.createBlockPreview(row.targetId, row.blockId);
if (preview) {
root.appendChild(preview);
}
}
if (row.type === "compiled") {
const el = document.createElement('div');
el.className = "sa-debugger-thread-compiled";
el.textContent = "Compiled threads can't be stepped and have no stack information.";
root.appendChild(el);
}
if (row.targetId && row.blockId) {
root.appendChild(debug.createBlockLink(debug.getTargetInfoById(row.targetId), row.blockId));
}
return {
root
};
};
logView.renderRow = (elements, row) => {
const {
root
} = elements;
root.classList.toggle("sa-debugger-thread-running", !!row.running);
};
let threadInfoCache = new WeakMap();
const allThreadIds = new WeakMap();
let nextThreadId = 1;
const getThreadId = thread => {
if (!allThreadIds.has(thread)) {
allThreadIds.set(thread, nextThreadId++);
}
return allThreadIds.get(thread);
};
const updateContent = () => {
if (!logView.visible) {
return;
}
const newRows = [];
const threads = vm.runtime.threads;
const visitedThreads = new Set();
const createThreadInfo = (thread, depth) => {
if (visitedThreads.has(thread)) {
return [];
}
visitedThreads.add(thread);
const id = getThreadId(thread);
const target = thread.target;
if (!threadInfoCache.has(thread)) {
threadInfoCache.set(thread, {
headerItem: {
type: "thread-header",
depth,
targetName: target.getName(),
id
},
compiledItem: thread.isCompiled ? {
type: "compiled",
depth: 1
} : null,
blockCache: new WeakMap()
});
}
const cacheInfo = threadInfoCache.get(thread);
const runningThread = Object(_module_js__WEBPACK_IMPORTED_MODULE_0__["getRunningThread"])();
const createBlockInfo = (block, stackFrameIdx) => {
const blockId = block.id;
if (!block) return;
const stackFrame = thread.stackFrames[stackFrameIdx];
if (!cacheInfo.blockCache.has(block)) {
cacheInfo.blockCache.set(block, {});
}
const blockInfoMap = cacheInfo.blockCache.get(block);
let blockInfo = blockInfoMap[stackFrameIdx];
if (!blockInfo) {
blockInfo = blockInfoMap[stackFrameIdx] = {
type: "thread-stack",
depth,
targetId: target.id,
blockId
};
}
blockInfo.running = thread === runningThread && (thread.isCompiled || blockId === runningThread.peekStack() && stackFrameIdx === runningThread.stackFrames.length - 1);
const result = [blockInfo];
if (stackFrame && stackFrame.executionContext && stackFrame.executionContext.startedThreads) {
for (const thread of stackFrame.executionContext.startedThreads) {
concatInPlace(result, createThreadInfo(thread, depth + 1));
}
}
return result;
};
const topBlock = debug.getBlock(thread.target, thread.topBlock);
const result = [cacheInfo.headerItem];
if (topBlock) {
concatInPlace(result, createBlockInfo(topBlock, 0));
for (let i = 0; i < thread.stack.length; i++) {
const blockId = thread.stack[i];
if (blockId === topBlock.id) continue;
const block = debug.getBlock(thread.target, blockId);
if (block) {
concatInPlace(result, createBlockInfo(block, i));
}
}
}
if (cacheInfo.compiledItem) {
result.push(cacheInfo.compiledItem);
}
return result;
};
for (let i = 0; i < threads.length; i++) {
const thread = threads[i]; // Do not display threads used to update variable and list monitors.
if (thread.updateMonitor) {
continue;
}
concatInPlace(newRows, createThreadInfo(thread, 0));
}
logView.rows = newRows;
logView.queueUpdateContent();
};
debug.addAfterStepCallback(() => {
updateContent();
const runningThread = Object(_module_js__WEBPACK_IMPORTED_MODULE_0__["getRunningThread"])();
if (runningThread) {
highlighter.setGlowingThreads([runningThread]);
} else {
highlighter.setGlowingThreads([]);
}
});
const stepButton = debug.createHeaderButton({
text: msg("step"),
icon: addon.self.getResource("/icons/step.svg")
/* rewritten by pull.js */
,
description: msg("step-desc")
});
stepButton.element.addEventListener("click", () => {
Object(_module_js__WEBPACK_IMPORTED_MODULE_0__["singleStep"])();
});
const handlePauseChanged = paused => {
stepButton.element.style.display = paused ? "" : "none";
updateContent();
};
handlePauseChanged(Object(_module_js__WEBPACK_IMPORTED_MODULE_0__["isPaused"])());
Object(_module_js__WEBPACK_IMPORTED_MODULE_0__["onPauseChanged"])(handlePauseChanged);
Object(_module_js__WEBPACK_IMPORTED_MODULE_0__["onSingleStep"])(updateContent);
const show = () => {
logView.show();
updateContent();
};
const hide = () => {
logView.hide();
};
return {
tab,
content: logView.outerElement,
buttons: [stepButton],
show,
hide
};
}
/***/ }),
/***/ "./src/addons/addons/debugger/userscript.js":
/*!**************************************************!*\
!*** ./src/addons/addons/debugger/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/debugger/module.js");
/* harmony import */ var _logs_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./logs.js */ "./src/addons/addons/debugger/logs.js");
/* harmony import */ var _threads_js__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./threads.js */ "./src/addons/addons/debugger/threads.js");
/* harmony import */ var _performance_js__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./performance.js */ "./src/addons/addons/debugger/performance.js");
/* harmony import */ var _find_bar_blockly_Utils_js__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ../find-bar/blockly/Utils.js */ "./src/addons/addons/find-bar/blockly/Utils.js");
const removeAllChildren = element => {
while (element.firstChild) {
element.removeChild(element.firstChild);
}
};
/* harmony default export */ __webpack_exports__["default"] = (async function ({
addon,
console,
msg
}) {
Object(_module_js__WEBPACK_IMPORTED_MODULE_0__["setup"])(addon.tab.traps.vm);
let logsTab;
const messagesLoggedBeforeLogsTabLoaded = [];
const logMessage = (...args) => {
if (logsTab) {
logsTab.addLog(...args);
} else {
messagesLoggedBeforeLogsTabLoaded.push(args);
}
};
let hasLoggedPauseError = false;
const pause = (_, thread) => {
if (addon.tab.redux.state.scratchGui.mode.isPlayerOnly) {
if (!hasLoggedPauseError) {
logMessage(msg("cannot-pause-player"), thread, "error");
hasLoggedPauseError = true;
}
return;
}
Object(_module_js__WEBPACK_IMPORTED_MODULE_0__["setPaused"])(true);
setInterfaceVisible(true);
};
addon.tab.addBlock("\u200B\u200Bbreakpoint\u200B\u200B", {
args: [],
displayName: msg("block-breakpoint"),
callback: pause
});
addon.tab.addBlock("\u200B\u200Blog\u200B\u200B %s", {
args: ["content"],
displayName: msg("block-log"),
callback: ({
content
}, thread) => {
logMessage(content, thread, "log");
}
});
addon.tab.addBlock("\u200B\u200Bwarn\u200B\u200B %s", {
args: ["content"],
displayName: msg("block-warn"),
callback: ({
content
}, thread) => {
logMessage(content, thread, "warn");
}
});
addon.tab.addBlock("\u200B\u200Berror\u200B\u200B %s", {
args: ["content"],
displayName: msg("block-error"),
callback: ({
content
}, thread) => {
logMessage(content, thread, "error");
}
});
const vm = addon.tab.traps.vm;
await new Promise((resolve, reject) => {
if (vm.editingTarget) return resolve();
vm.runtime.once("PROJECT_LOADED", resolve);
});
const ScratchBlocks = await addon.tab.traps.getBlockly();
const debuggerButtonOuter = document.createElement("div");
debuggerButtonOuter.className = "sa-debugger-container";
const debuggerButton = document.createElement("div");
debuggerButton.className = addon.tab.scratchClass("button_outlined-button", "stage-header_stage-button");
const debuggerButtonContent = document.createElement("div");
debuggerButtonContent.className = addon.tab.scratchClass("button_content");
const debuggerButtonImage = document.createElement("img");
debuggerButtonImage.className = addon.tab.scratchClass("stage-header_stage-button-icon");
debuggerButtonImage.draggable = false;
debuggerButtonImage.src = addon.self.getResource("/icons/debug.svg")
/* rewritten by pull.js */
;
debuggerButtonContent.appendChild(debuggerButtonImage);
debuggerButton.appendChild(debuggerButtonContent);
debuggerButtonOuter.appendChild(debuggerButton);
debuggerButton.addEventListener("click", () => setInterfaceVisible(true));
const setHasUnreadMessage = unreadMessage => {
debuggerButtonContent.classList.toggle("sa-debugger-unread", unreadMessage);
};
const interfaceContainer = Object.assign(document.createElement("div"), {
className: addon.tab.scratchClass("card_card", {
others: "sa-debugger-interface"
})
});
const interfaceHeader = Object.assign(document.createElement("div"), {
className: addon.tab.scratchClass("card_header-buttons")
});
const tabListElement = Object.assign(document.createElement("ul"), {
className: "sa-debugger-tabs"
});
const buttonContainerElement = Object.assign(document.createElement("div"), {
className: addon.tab.scratchClass("card_header-buttons-right", {
others: "sa-debugger-header-buttons"
})
});
const tabContentContainer = Object.assign(document.createElement("div"), {
className: "sa-debugger-tab-content"
});
const compilerWarning = document.createElement("a");
compilerWarning.addEventListener("click", () => {
addon.tab.redux.dispatch({
type: "scratch-gui/modals/OPEN_MODAL",
modal: "settingsModal"
});
});
compilerWarning.className = "sa-debugger-log sa-debugger-compiler-warning";
compilerWarning.textContent = "The debugger works best when the compiler is disabled.";
const updateCompilerWarningVisibility = () => {
compilerWarning.hidden = !vm.runtime.compilerOptions.enabled;
};
vm.on("COMPILER_OPTIONS_CHANGED", updateCompilerWarningVisibility);
updateCompilerWarningVisibility();
let isInterfaceVisible = false;
const setInterfaceVisible = _isVisible => {
isInterfaceVisible = _isVisible;
interfaceContainer.style.display = isInterfaceVisible ? "flex" : "";
if (isInterfaceVisible) {
activeTab.show();
} else {
activeTab.hide();
}
};
let mouseOffsetX = 0;
let mouseOffsetY = 0;
let lastX = 0;
let lastY = 0;
const handleStartDrag = e => {
e.preventDefault();
mouseOffsetX = e.clientX - interfaceContainer.offsetLeft;
mouseOffsetY = e.clientY - interfaceContainer.offsetTop;
lastX = e.clientX;
lastY = e.clientY;
document.addEventListener("mouseup", handleStopDrag);
document.addEventListener("mousemove", handleDragInterface);
};
const handleStopDrag = () => {
document.removeEventListener("mouseup", handleStopDrag);
document.removeEventListener("mousemove", handleDragInterface);
};
const moveInterface = (x, y) => {
lastX = x;
lastY = y;
const width = (document.documentElement.clientWidth || document.body.clientWidth) - 1;
const height = (document.documentElement.clientHeight || document.body.clientHeight) - 1;
const clampedX = Math.max(0, Math.min(x - mouseOffsetX, width - interfaceContainer.offsetWidth));
const clampedY = Math.max(0, Math.min(y - mouseOffsetY, height - interfaceContainer.offsetHeight));
interfaceContainer.style.left = clampedX + "px";
interfaceContainer.style.top = clampedY + "px";
};
const handleDragInterface = e => {
e.preventDefault();
moveInterface(e.clientX, e.clientY);
};
window.addEventListener("resize", () => {
moveInterface(lastX, lastY);
});
interfaceHeader.addEventListener("mousedown", handleStartDrag);
interfaceHeader.append(tabListElement, buttonContainerElement);
interfaceContainer.append(interfaceHeader, compilerWarning, tabContentContainer);
document.body.append(interfaceContainer);
const createHeaderButton = ({
text,
icon,
description
}) => {
const button = Object.assign(document.createElement("div"), {
className: addon.tab.scratchClass("card_shrink-expand-button"),
draggable: false
});
if (description) {
button.title = description;
}
const imageElement = Object.assign(document.createElement("img"), {
src: icon,
draggable: false
});
const textElement = Object.assign(document.createElement("span"), {
textContent: text
});
button.appendChild(imageElement);
button.appendChild(textElement);
return {
element: button,
image: imageElement,
text: textElement
};
};
const createHeaderTab = ({
text,
icon
}) => {
const tab = document.createElement("li");
const imageElement = Object.assign(document.createElement("img"), {
src: icon,
draggable: false
});
const textElement = Object.assign(document.createElement("span"), {
textContent: text
});
tab.appendChild(imageElement);
tab.appendChild(textElement);
return {
element: tab,
image: imageElement,
text: textElement
};
};
const unpauseButton = createHeaderButton({
text: msg("unpause"),
icon: addon.self.getResource("/icons/play.svg")
/* rewritten by pull.js */
});
unpauseButton.element.classList.add("sa-debugger-unpause");
unpauseButton.element.addEventListener("click", () => Object(_module_js__WEBPACK_IMPORTED_MODULE_0__["setPaused"])(false));
const updateUnpauseVisibility = paused => {
unpauseButton.element.style.display = paused ? "" : "none";
};
updateUnpauseVisibility(Object(_module_js__WEBPACK_IMPORTED_MODULE_0__["isPaused"])());
Object(_module_js__WEBPACK_IMPORTED_MODULE_0__["onPauseChanged"])(updateUnpauseVisibility);
const closeButton = createHeaderButton({
text: msg("close"),
icon: addon.self.getResource("/icons/close.svg")
/* rewritten by pull.js */
});
closeButton.element.addEventListener("click", () => setInterfaceVisible(false));
const originalStep = vm.runtime._step;
const afterStepCallbacks = [];
vm.runtime._step = function (...args) {
const ret = originalStep.call(this, ...args);
for (const cb of afterStepCallbacks) {
cb();
}
return ret;
};
const addAfterStepCallback = cb => {
afterStepCallbacks.push(cb);
};
const getBlock = (target, id) => target.blocks.getBlock(id) || vm.runtime.flyoutBlocks.getBlock(id);
const getTargetInfoById = id => {
const target = vm.runtime.getTargetById(id);
if (target) {
let name = target.getName();
let original = target;
if (!target.isOriginal) {
name = msg("clone-of", {
sprite: name
});
original = target.sprite.clones[0];
}
return {
exists: true,
originalId: original.id,
name
};
}
return {
exists: false,
original: null,
name: msg("unknown-sprite")
};
};
const createBlockLink = (targetInfo, blockId) => {
const link = document.createElement("a");
link.className = "sa-debugger-log-link";
const {
exists,
name,
originalId
} = targetInfo;
link.textContent = name;
if (exists) {
// We use mousedown instead of click so that you can still go to blocks when logs are rapidly scrolling
link.addEventListener("mousedown", () => {
switchToSprite(originalId);
activateCodeTab();
goToBlock(blockId);
});
} else {
link.classList.add("sa-debugger-log-link-unknown");
}
return link;
};
const switchToSprite = targetId => {
if (targetId !== vm.editingTarget.id) {
if (vm.runtime.getTargetById(targetId)) {
vm.setEditingTarget(targetId);
}
}
};
const activateCodeTab = () => {
const redux = addon.tab.redux;
if (redux.state.scratchGui.editorTab.activeTabIndex !== 0) {
redux.dispatch({
type: "scratch-gui/navigation/ACTIVATE_TAB",
activeTabIndex: 0
});
}
};
const goToBlock = blockId => {
const workspace = Blockly.getMainWorkspace();
const block = workspace.getBlockById(blockId);
if (!block) return; // Don't scroll to blocks in the flyout
if (block.workspace.isFlyout) return;
new _find_bar_blockly_Utils_js__WEBPACK_IMPORTED_MODULE_4__["default"](addon).scrollBlockIntoView(blockId);
};
/**
* @param {string} procedureCode
* @returns {string}
*/
const formatProcedureCode = procedureCode => {
const customBlock = addon.tab.getCustomBlock(procedureCode);
if (customBlock) {
procedureCode = customBlock.displayName;
} // May be slightly incorrect in some edge cases.
return procedureCode.replace(/%[nbs]/g, "()");
}; // May be slightly incorrect in some edge cases.
const formatBlocklyBlockData = jsonData => {
// For sample jsonData, see:
// https://github.com/LLK/scratch-blocks/blob/0bd1a17e66a779ec5d11f4a00c43784e3ac7a7b8/blocks_vertical/motion.js
// https://github.com/LLK/scratch-blocks/blob/0bd1a17e66a779ec5d11f4a00c43784e3ac7a7b8/blocks_vertical/control.js
const processSegment = index => {
const message = jsonData["message".concat(index)];
const args = jsonData["args".concat(index)];
if (!message) {
return null;
}
const parts = message.split(/%\d+/g);
let formattedMessage = "";
for (let i = 0; i < parts.length; i++) {
formattedMessage += parts[i];
const argInfo = args && args[i];
if (argInfo) {
const type = argInfo.type;
if (type === "field_vertical_separator") {// no-op
} else if (type === "field_image") {
const src = argInfo.src;
if (src.endsWith("rotate-left.svg")) {
formattedMessage += "↩";
} else if (src.endsWith("rotate-right.svg")) {
formattedMessage += "↪";
}
} else {
formattedMessage += "()";
}
}
}
return formattedMessage;
};
const parts = [];
let i = 0; // The jsonData doesn't directly tell us how many segments it has, so we have to
// just keep looping until one doesn't exist.
while (true) {
const nextSegment = processSegment(i);
if (nextSegment) {
parts.push(nextSegment);
} else {
break;
}
i++;
}
return parts.join(" ");
};
const createBlockPreview = (targetId, blockId) => {
const target = vm.runtime.getTargetById(targetId);
if (!target) {
return null;
}
const block = getBlock(target, blockId);
if (!block || block.opcode === "text") {
return null;
}
let text;
let category;
let shape;
if (block.opcode === "data_variable" || block.opcode === "data_listcontents" || block.opcode === "argument_reporter_string_number" || block.opcode === "argument_reporter_boolean") {
text = Object.values(block.fields)[0].value;
if (block.opcode === "data_variable") {
category = "data";
} else if (block.opcode === "data_listcontents") {
category = "list";
} else {
category = "more";
}
shape = "round";
} else if (block.opcode === "procedures_call") {
const proccode = block.mutation.proccode;
text = formatProcedureCode(proccode);
const customBlock = addon.tab.getCustomBlock(proccode);
if (customBlock) {
category = "addon-custom-block";
} else {
category = "more";
}
} else if (block.opcode === "procedures_definition") {
const prototypeBlockId = block.inputs.custom_block.block;
const prototypeBlock = getBlock(target, prototypeBlockId);
const proccode = prototypeBlock.mutation.proccode;
text = ScratchBlocks.ScratchMsgs.translate("PROCEDURES_DEFINITION", "define %1").replace("%1", formatProcedureCode(proccode));
category = "more";
} else {
var _jsonData$extensions;
// Try to call things like https://github.com/LLK/scratch-blocks/blob/0bd1a17e66a779ec5d11f4a00c43784e3ac7a7b8/blocks_vertical/operators.js#L36
var jsonData;
const fakeBlock = {
jsonInit(data) {
jsonData = data;
}
};
const blockConstructor = ScratchBlocks.Blocks[block.opcode];
if (blockConstructor) {
try {
blockConstructor.init.call(fakeBlock);
} catch (e) {// ignore
}
}
if (!jsonData) {
return null;
}
text = formatBlocklyBlockData(jsonData);
if (!text) {
return null;
} // jsonData.extensions is not guaranteed to exist
category = (_jsonData$extensions = jsonData.extensions) !== null && _jsonData$extensions !== void 0 && _jsonData$extensions.includes("scratch_extension") ? "pen" : jsonData.category;
const isStatement = jsonData.extensions && (jsonData.extensions.includes("shape_statement") || jsonData.extensions.includes("shape_hat") || jsonData.extensions.includes("shape_end")) || "previousStatement" in jsonData || "nextStatement" in jsonData;
shape = isStatement ? "stacked" : "round";
}
if (!text || !category) {
return null;
}
const element = document.createElement("span");
element.className = "sa-debugger-block-preview sa-block-color";
element.textContent = text;
element.dataset.shape = shape;
element.classList.add("sa-block-color-".concat(category));
return element;
};
const api = {
debug: {
createHeaderButton,
createHeaderTab,
setHasUnreadMessage,
addAfterStepCallback,
getBlock,
getTargetInfoById,
createBlockLink,
createBlockPreview
},
addon,
msg,
console
};
logsTab = await Object(_logs_js__WEBPACK_IMPORTED_MODULE_1__["default"])(api);
const threadsTab = await Object(_threads_js__WEBPACK_IMPORTED_MODULE_2__["default"])(api);
const allTabs = [logsTab, threadsTab];
for (const message of messagesLoggedBeforeLogsTabLoaded) {
logsTab.addLog(...message);
}
messagesLoggedBeforeLogsTabLoaded.length = 0;
let activeTab;
const setActiveTab = tab => {
if (tab === activeTab) return;
const selectedClass = "sa-debugger-tab-selected";
if (activeTab) {
activeTab.hide();
activeTab.tab.element.classList.remove(selectedClass);
}
tab.tab.element.classList.add(selectedClass);
activeTab = tab;
removeAllChildren(tabContentContainer);
tabContentContainer.appendChild(tab.content);
removeAllChildren(buttonContainerElement);
buttonContainerElement.appendChild(unpauseButton.element);
for (const button of tab.buttons) {
buttonContainerElement.appendChild(button.element);
}
buttonContainerElement.appendChild(closeButton.element);
if (isInterfaceVisible) {
activeTab.show();
}
};
for (const tab of allTabs) {
tab.tab.element.addEventListener("click", () => {
setActiveTab(tab);
});
tabListElement.appendChild(tab.tab.element);
}
setActiveTab(allTabs[0]);
if (addon.tab.redux.state && addon.tab.redux.state.scratchGui.stageSize.stageSize === "small") {
document.body.classList.add("sa-debugger-small");
}
document.addEventListener("click", e => {
if (e.target.closest("[class*='stage-header_stage-button-first']:not(.sa-hide-stage-button)")) {
document.body.classList.add("sa-debugger-small");
} else if (e.target.closest("[class*='stage-header_stage-button-last']") || e.target.closest(".sa-hide-stage-button")) {
document.body.classList.remove("sa-debugger-small");
}
}, {
capture: true
});
const ogGreenFlag = vm.runtime.greenFlag;
vm.runtime.greenFlag = function (...args) {
if (addon.settings.get("log_clear_greenflag")) {
logsTab.clearLogs();
}
if (addon.settings.get("log_greenflag")) {
logsTab.addLog(msg("log-msg-flag-clicked"), null, "internal");
}
return ogGreenFlag.call(this, ...args);
};
const ogMakeClone = vm.runtime.targets[0].constructor.prototype.makeClone;
vm.runtime.targets[0].constructor.prototype.makeClone = function (...args) {
if (addon.settings.get("log_failed_clone_creation") && !vm.runtime.clonesAvailable()) {
logsTab.addLog(msg("log-msg-clone-cap", {
sprite: this.getName()
}), vm.runtime.sequencer.activeThread, "internal-warn");
}
var clone = ogMakeClone.call(this, ...args);
if (addon.settings.get("log_clone_create") && clone) {
logsTab.addLog(msg("log-msg-clone-created", {
sprite: this.getName()
}), vm.runtime.sequencer.activeThread, "internal");
}
return clone;
};
const ogStartHats = vm.runtime.startHats;
vm.runtime.startHats = function (hat, optMatchFields, ...args) {
if (addon.settings.get("log_broadcasts") && hat === "event_whenbroadcastreceived") {
logsTab.addLog(msg("log-msg-broadcasted", {
broadcast: optMatchFields.BROADCAST_OPTION
}), vm.runtime.sequencer.activeThread, "internal");
}
return ogStartHats.call(this, hat, optMatchFields, ...args);
};
while (true) {
await addon.tab.waitForElement('[class*="stage-header_stage-size-row"]', {
markAsSeen: true,
reduxEvents: ["scratch-gui/mode/SET_PLAYER", "scratch-gui/mode/SET_FULL_SCREEN", "fontsLoaded/SET_FONTS_LOADED", "scratch-gui/locales/SELECT_LOCALE"]
});
if (addon.tab.editorMode === "editor") {
addon.tab.appendToSharedSpace({
space: "stageHeader",
element: debuggerButtonOuter,
order: 0
});
} else {
setInterfaceVisible(false);
}
}
});
/***/ }),
/***/ "./src/addons/addons/editor-stepping/highlighter.js":
/*!**********************************************************!*\
!*** ./src/addons/addons/editor-stepping/highlighter.js ***!
\**********************************************************/
/*! exports provided: default */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
const SVG_NS = "http://www.w3.org/2000/svg";
const containerSvg = document.createElementNS(SVG_NS, "svg"); // unfortunately we can't use display: none on this as that breaks filters
containerSvg.style.position = "fixed";
containerSvg.style.top = "-999999px";
containerSvg.style.width = "0";
containerSvg.style.height = "0";
document.body.appendChild(containerSvg);
let nextGlowerId = 0;
const highlightsPerElement = new WeakMap();
const getHighlightersForElement = element => {
if (!highlightsPerElement.get(element)) {
highlightsPerElement.set(element, new Set());
}
return highlightsPerElement.get(element);
};
const updateHighlight = (element, highlighters) => {
let result;
for (const i of highlighters) {
if (!result || i.priority > result.priority) {
result = i;
}
}
if (result) {
element.style.filter = result.filter;
} else {
element.style.filter = "";
}
};
const addHighlight = (element, highlighter) => {
const highlighters = getHighlightersForElement(element);
highlighters.add(highlighter);
updateHighlight(element, highlighters);
};
const removeHighlight = (element, highlighter) => {
const highlighters = getHighlightersForElement(element);
highlighters.delete(highlighter);
updateHighlight(element, highlighters);
};
class Highlighter {
constructor(priority, color) {
this.priority = priority;
const id = "sa_glower_filter".concat(nextGlowerId++);
this.filter = "url(\"#".concat(id, "\")");
this.previousElements = new Set();
const filterElement = document.createElementNS(SVG_NS, "filter");
filterElement.id = id;
filterElement.setAttribute("width", "180%");
filterElement.setAttribute("height", "160%");
filterElement.setAttribute("x", "-40%");
filterElement.setAttribute("y", "-30%");
const filterBlur = document.createElementNS(SVG_NS, "feGaussianBlur");
filterBlur.setAttribute("in", "SourceGraphic");
filterBlur.setAttribute("stdDeviation", "4");
filterElement.appendChild(filterBlur);
const filterTransfer = document.createElementNS(SVG_NS, "feComponentTransfer");
filterTransfer.setAttribute("result", "outBlur");
filterElement.appendChild(filterTransfer);
const filterTransferTable = document.createElementNS(SVG_NS, "feFuncA");
filterTransferTable.setAttribute("type", "table");
filterTransferTable.setAttribute("tableValues", "0 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1");
filterTransfer.appendChild(filterTransferTable);
const filterFlood = document.createElementNS(SVG_NS, "feFlood");
filterFlood.setAttribute("flood-opacity", "1");
filterFlood.setAttribute("result", "outColor");
filterElement.appendChild(filterFlood);
this.filterFlood = filterFlood;
const filterComposite = document.createElementNS(SVG_NS, "feComposite");
filterComposite.setAttribute("in", "outColor");
filterComposite.setAttribute("in2", "outBlur");
filterComposite.setAttribute("operator", "in");
filterComposite.setAttribute("result", "outGlow");
filterElement.appendChild(filterComposite);
const filterFinalComposite = document.createElementNS(SVG_NS, "feComposite");
filterFinalComposite.setAttribute("in", "SourceGraphic");
filterFinalComposite.setAttribute("in2", "outGlow");
filterFinalComposite.setAttribute("operator", "over");
filterElement.appendChild(filterFinalComposite);
containerSvg.appendChild(filterElement);
this.setColor(color);
}
setColor(color) {
this.filterFlood.setAttribute("flood-color", color);
}
setGlowingThreads(threads) {
const elementsToHighlight = new Set();
const workspace = Blockly.getMainWorkspace();
if (workspace) {
for (const thread of threads) {
thread.stack.forEach(blockId => {
const block = workspace.getBlockById(blockId);
if (!block) {
return;
}
const childblock = thread.stack.find(i => {
let b = block;
while (b.childBlocks_.length) {
b = b.childBlocks_[b.childBlocks_.length - 1];
if (i === b.id) return true;
}
return false;
});
if (!childblock && block.svgPath_) {
const svgPath = block.svgPath_;
elementsToHighlight.add(svgPath);
}
});
}
}
for (const element of this.previousElements) {
if (!elementsToHighlight.has(element)) {
removeHighlight(element, this);
}
}
for (const element of elementsToHighlight) {
if (!this.previousElements.has(element)) {
addHighlight(element, this);
}
}
this.previousElements = elementsToHighlight;
}
}
/* harmony default export */ __webpack_exports__["default"] = (Highlighter);
/***/ }),
/***/ "./src/addons/libraries/common/cs/download-blob.js":
/*!*********************************************************!*\
!*** ./src/addons/libraries/common/cs/download-blob.js ***!
\*********************************************************/
/*! exports provided: default */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
// From https://github.com/LLK/scratch-gui/blob/develop/src/lib/download-blob.js
/* harmony default export */ __webpack_exports__["default"] = ((filename, blob) => {
const downloadLink = document.createElement("a");
document.body.appendChild(downloadLink); // Use special ms version if available to get it working on Edge.
if (navigator.msSaveOrOpenBlob) {
navigator.msSaveOrOpenBlob(blob, filename);
return;
}
if ("download" in HTMLAnchorElement.prototype) {
const url = window.URL.createObjectURL(blob);
downloadLink.href = url;
downloadLink.download = filename;
downloadLink.type = blob.type;
downloadLink.click(); // remove the link after a timeout to prevent a crash on iOS 13 Safari
window.setTimeout(() => {
document.body.removeChild(downloadLink);
window.URL.revokeObjectURL(url);
}, 1000);
} else {
// iOS 12 Safari, open a new page and set href to data-uri
let popup = window.open("", "_blank");
const reader = new FileReader();
reader.onloadend = function () {
popup.location.href = reader.result;
popup = null;
};
reader.readAsDataURL(blob);
}
});
/***/ })
}]);
//# sourceMappingURL=addon-entry-debugger.js.map