Spaces:
Running
Running
(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__) { | |
; | |
__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__) { | |
; | |
__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__) { | |
; | |
__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__) { | |
; | |
__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__) { | |
; | |
__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__) { | |
; | |
__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__) { | |
; | |
__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__) { | |
; | |
__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__) { | |
; | |
__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__) { | |
; | |
__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__) { | |
; | |
__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__) { | |
; | |
__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__) { | |
; | |
__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__) { | |
; | |
__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__) { | |
; | |
__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(_ref) { | |
let { | |
debug, | |
addon, | |
console, | |
msg | |
} = _ref; | |
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(_ref2 => { | |
let { | |
text, | |
targetInfo, | |
type, | |
count | |
} = _ref2; | |
return (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 ? thread.peekStack() : thread.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/module.js": | |
/*!**********************************************!*\ | |
!*** ./src/addons/addons/debugger/module.js ***! | |
\**********************************************/ | |
/*! exports provided: isPaused, setPaused, onPauseChanged, onSingleStep, getRunningThread, singleStep, setup */ | |
/***/ (function(module, __webpack_exports__, __webpack_require__) { | |
; | |
__webpack_require__.r(__webpack_exports__); | |
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "isPaused", function() { return isPaused; }); | |
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "setPaused", function() { return setPaused; }); | |
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "onPauseChanged", function() { return onPauseChanged; }); | |
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "onSingleStep", function() { return onSingleStep; }); | |
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "getRunningThread", function() { return getRunningThread; }); | |
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "singleStep", function() { return singleStep; }); | |
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "setup", function() { return setup; }); | |
/* harmony import */ var _event_target_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../../event-target.js */ "./src/addons/event-target.js"); | |
/* inserted by pull.js */ | |
// https://github.com/LLK/scratch-vm/blob/bb352913b57991713a5ccf0b611fda91056e14ec/src/engine/thread.js#L198 | |
const STATUS_RUNNING = 0; | |
const STATUS_PROMISE_WAIT = 1; | |
const STATUS_YIELD = 2; | |
const STATUS_YIELD_TICK = 3; | |
const STATUS_DONE = 4; | |
let vm; | |
let paused = false; | |
let pausedThreadState = new WeakMap(); | |
let pauseNewThreads = false; | |
let steppingThread = null; | |
const eventTarget = new _event_target_js__WEBPACK_IMPORTED_MODULE_0__["default"](); | |
let audioContextStateChange = Promise.resolve(); | |
const isPaused = () => paused; | |
const pauseThread = thread => { | |
if (thread.updateMonitor || pausedThreadState.has(thread)) { | |
// Thread is already paused or shouldn't be paused. | |
return; | |
} | |
const pauseState = { | |
time: vm.runtime.currentMSecs, | |
status: thread.status | |
}; | |
pausedThreadState.set(thread, pauseState); | |
// Pausing a thread now works by just setting its status to STATUS_PROMISE_WAIT. | |
// At the start of each frame, we make sure each paused thread is still paused. | |
// This is really the best way to implement this. | |
// Converting thread.status into a getter/setter causes Scratch's sequencer to permanently | |
// perform significantly slower in some projects. I think this is because it causes some | |
// very hot functions to be deoptimized. | |
// Trapping sequencer.stepThread to no-op for a paused thread causes Scratch's sequencer | |
// to waste 24ms of CPU time every frame because it thinks a thread is running. | |
thread.status = STATUS_PROMISE_WAIT; | |
}; | |
const ensurePausedThreadIsStillPaused = thread => { | |
if (thread.status === STATUS_DONE) { | |
// If a paused thread is finished by single stepping, let it keep being done. | |
return; | |
} | |
const pauseState = pausedThreadState.get(thread); | |
if (pauseState) { | |
if (thread.status !== STATUS_PROMISE_WAIT) { | |
// We'll record the change so we can properly resume the thread, but the thread must still be paused for now. | |
pauseState.status = thread.status; | |
thread.status = STATUS_PROMISE_WAIT; | |
} | |
} | |
}; | |
const setSteppingThread = thread => { | |
steppingThread = thread; | |
}; | |
const compensateForTimePassedWhilePaused = (thread, pauseState) => { | |
// TW: Compiled threads store their timer in a different place. | |
if (thread.timer) { | |
thread.timer.startTime += vm.runtime.currentMSecs - pauseState.time; | |
} | |
const stackFrame = thread.peekStackFrame(); | |
if (stackFrame && stackFrame.executionContext && stackFrame.executionContext.timer) { | |
stackFrame.executionContext.timer.startTime += vm.runtime.currentMSecs - pauseState.time; | |
} | |
}; | |
const stepUnsteppedThreads = lastSteppedThread => { | |
// If we paused in the middle of a tick, we need to make sure to step the scripts that didn't get | |
// stepped in that tick to avoid affecting project behavior. | |
const threads = vm.runtime.threads; | |
const startingIndex = getThreadIndex(lastSteppedThread); | |
if (startingIndex !== -1) { | |
for (let i = startingIndex; i < threads.length; i++) { | |
const thread = threads[i]; | |
const status = thread.status; | |
if (status === STATUS_RUNNING || status === STATUS_YIELD || status === STATUS_YIELD_TICK) { | |
vm.runtime.sequencer.activeThread = thread; | |
vm.runtime.sequencer.stepThread(thread); | |
} | |
} | |
} | |
}; | |
const setPaused = _paused => { | |
const didChange = paused !== _paused; | |
if (didChange) { | |
paused = _paused; | |
eventTarget.dispatchEvent(new CustomEvent("change")); | |
} | |
// Don't check didChange as new threads could've started that we need to pause. | |
if (paused) { | |
audioContextStateChange = audioContextStateChange.then(() => { | |
return vm.runtime.audioEngine.audioContext.suspend(); | |
}); | |
if (!vm.runtime.ioDevices.clock._paused) { | |
vm.runtime.ioDevices.clock.pause(); | |
} | |
vm.runtime.threads.forEach(pauseThread); | |
const activeThread = vm.runtime.sequencer.activeThread; | |
if (activeThread) { | |
setSteppingThread(activeThread); | |
eventTarget.dispatchEvent(new CustomEvent("step")); | |
} | |
} | |
// Only run unpausing logic when pause state changed to avoid unnecessary work | |
if (!paused && didChange) { | |
audioContextStateChange = audioContextStateChange.then(() => { | |
return vm.runtime.audioEngine.audioContext.resume(); | |
}); | |
vm.runtime.ioDevices.clock.resume(); | |
for (const thread of vm.runtime.threads) { | |
const pauseState = pausedThreadState.get(thread); | |
if (pauseState) { | |
compensateForTimePassedWhilePaused(thread, pauseState); | |
thread.status = pauseState.status; | |
} | |
} | |
pausedThreadState = new WeakMap(); | |
const lastSteppedThread = steppingThread; | |
// This must happen after the "change" event is fired to fix https://github.com/ScratchAddons/ScratchAddons/issues/4281 | |
stepUnsteppedThreads(lastSteppedThread); | |
steppingThread = null; | |
} | |
}; | |
const onPauseChanged = listener => { | |
eventTarget.addEventListener("change", () => listener(paused)); | |
}; | |
const onSingleStep = listener => { | |
eventTarget.addEventListener("step", listener); | |
}; | |
const getRunningThread = () => steppingThread; | |
// A modified version of this function | |
// https://github.com/LLK/scratch-vm/blob/0e86a78a00db41af114df64255e2cd7dd881329f/src/engine/sequencer.js#L179 | |
// Returns if we should continue executing this thread. | |
const singleStepThread = thread => { | |
if (thread.status === STATUS_DONE) { | |
return false; | |
} | |
// TW: Can't single-step compiled threads | |
if (thread.isCompiled) { | |
return false; | |
} | |
const currentBlockId = thread.peekStack(); | |
if (!currentBlockId) { | |
thread.popStack(); | |
if (thread.stack.length === 0) { | |
thread.status = STATUS_DONE; | |
return false; | |
} | |
} | |
pauseNewThreads = true; | |
vm.runtime.sequencer.activeThread = thread; | |
/* | |
We need to call execute(this, thread) like the original sequencer. We don't | |
have access to that method, so we need to force the original stepThread to run | |
execute for us then exit before it tries to run more blocks. | |
So, we make `thread.blockGlowInFrame = ...` throw an exception, so this line: | |
https://github.com/LLK/scratch-vm/blob/bb352913b57991713a5ccf0b611fda91056e14ec/src/engine/sequencer.js#L214 | |
will end the function early. We then have to set it back to normal afterward. | |
Why are we here just to suffer? | |
*/ | |
const specialError = ["special error used by Scratch Addons for implementing single-stepping"]; | |
Object.defineProperty(thread, "blockGlowInFrame", { | |
set(_block) { | |
throw specialError; | |
} | |
}); | |
try { | |
thread.status = STATUS_RUNNING; | |
// Restart the warp timer on each step. | |
// If we don't do this, Scratch will think a lot of time has passed and may yield this thread. | |
if (thread.warpTimer) { | |
thread.warpTimer.start(); | |
} | |
try { | |
vm.runtime.sequencer.stepThread(thread); | |
} catch (err) { | |
if (err !== specialError) throw err; | |
} | |
if (thread.status !== STATUS_RUNNING) { | |
return false; | |
} | |
if (thread.peekStack() === currentBlockId) { | |
thread.goToNextBlock(); | |
} | |
while (!thread.peekStack()) { | |
thread.popStack(); | |
if (thread.stack.length === 0) { | |
thread.status = STATUS_DONE; | |
return false; | |
} | |
const stackFrame = thread.peekStackFrame(); | |
if (stackFrame.isLoop) { | |
if (thread.peekStackFrame().warpMode) { | |
continue; | |
} else { | |
return false; | |
} | |
} else if (stackFrame.waitingReporter) { | |
return false; | |
} | |
thread.goToNextBlock(); | |
} | |
return true; | |
} finally { | |
pauseNewThreads = false; | |
vm.runtime.sequencer.activeThread = null; | |
Object.defineProperty(thread, "blockGlowInFrame", { | |
value: currentBlockId, | |
configurable: true, | |
enumerable: true, | |
writable: true | |
}); | |
// Strictly this doesn't seem to be necessary, but let's make sure the thread is still paused after we step it. | |
if (thread.status !== STATUS_DONE) { | |
thread.status = STATUS_PROMISE_WAIT; | |
} | |
} | |
}; | |
const getRealStatus = thread => { | |
const pauseState = pausedThreadState.get(thread); | |
if (pauseState) { | |
return pauseState.status; | |
} | |
return thread.status; | |
}; | |
const getThreadIndex = thread => { | |
// We can't use vm.runtime.threads.indexOf(thread) because threads can be restarted. | |
// This can happens when, for example, a "when I receive message1" script broadcasts message1. | |
// The object in runtime.threads is replaced when this happens. | |
if (!thread) return -1; | |
return vm.runtime.threads.findIndex(otherThread => otherThread.target === thread.target && otherThread.topBlock === thread.topBlock && otherThread.stackClick === thread.stackClick && otherThread.updateMonitor === thread.updateMonitor); | |
}; | |
const findNewSteppingThread = startingIndex => { | |
const threads = vm.runtime.threads; | |
for (let i = startingIndex; i < threads.length; i++) { | |
const possibleNewThread = threads[i]; | |
if (possibleNewThread.updateMonitor) { | |
// Never single-step monitor update threads. | |
continue; | |
} | |
// TW: Can't single-step compiled threads | |
if (possibleNewThread.isCompiled) { | |
continue; | |
} | |
const status = getRealStatus(possibleNewThread); | |
if (status === STATUS_RUNNING || status === STATUS_YIELD || status === STATUS_YIELD_TICK) { | |
// Thread must not be running for single stepping to work. | |
pauseThread(possibleNewThread); | |
return possibleNewThread; | |
} | |
} | |
return null; | |
}; | |
const singleStep = () => { | |
if (steppingThread) { | |
const pauseState = pausedThreadState.get(steppingThread); | |
// We can assume pauseState is defined as any single stepping threads must already be paused. | |
// Make it look like no time has passed | |
compensateForTimePassedWhilePaused(steppingThread, pauseState); | |
pauseState.time = vm.runtime.currentMSecs; | |
// Execute the block | |
const continueExecuting = singleStepThread(steppingThread); | |
if (!continueExecuting) { | |
// Try to move onto the next thread | |
steppingThread = findNewSteppingThread(getThreadIndex(steppingThread) + 1); | |
} | |
} | |
// If we don't have a thread, than we are between VM steps and should search for a new thread | |
if (!steppingThread) { | |
setSteppingThread(findNewSteppingThread(0)); | |
// End of VM step, emulate one frame of time passing. | |
vm.runtime.ioDevices.clock._pausedTime += vm.runtime.currentStepTime; | |
// Skip all sounds forward by vm.runtime.currentStepTime milliseconds so it's as | |
// if they where playing for one frame. | |
const audioContext = vm.runtime.audioEngine.audioContext; | |
for (const target of vm.runtime.targets) { | |
for (const soundId of Object.keys(target.sprite.soundBank.soundPlayers)) { | |
const soundPlayer = target.sprite.soundBank.soundPlayers[soundId]; | |
if (soundPlayer.outputNode) { | |
soundPlayer.outputNode.stop(audioContext.currentTime); | |
soundPlayer._createSource(); | |
soundPlayer.outputNode.start(audioContext.currentTime, audioContext.currentTime - soundPlayer.startingUntil + vm.runtime.currentStepTime / 1000); | |
soundPlayer.startingUntil -= vm.runtime.currentStepTime / 1000; | |
} | |
} | |
} | |
// Move all threads forward one frame in time. For blocks like `wait () seconds` | |
for (const thread of vm.runtime.threads) { | |
if (pausedThreadState.has(thread)) { | |
pausedThreadState.get(thread).time += vm.runtime.currentStepTime; | |
} | |
} | |
// Try to run edge activated hats | |
pauseNewThreads = true; | |
const hats = vm.runtime._hats; | |
for (const hatType in hats) { | |
if (!Object.prototype.hasOwnProperty.call(hats, hatType)) continue; | |
const hat = hats[hatType]; | |
if (hat.edgeActivated) { | |
vm.runtime.startHats(hatType); | |
} | |
} | |
pauseNewThreads = false; | |
} | |
eventTarget.dispatchEvent(new CustomEvent("step")); | |
}; | |
const setup = _vm => { | |
if (vm) { | |
return; | |
} | |
vm = _vm; | |
const originalStepThreads = vm.runtime.sequencer.stepThreads; | |
vm.runtime.sequencer.stepThreads = function () { | |
if (isPaused()) { | |
for (const thread of this.runtime.threads) { | |
ensurePausedThreadIsStillPaused(thread); | |
} | |
} | |
return originalStepThreads.call(this); | |
}; | |
// Unpause when green flag | |
const originalGreenFlag = vm.runtime.greenFlag; | |
vm.runtime.greenFlag = function () { | |
setPaused(false); | |
return originalGreenFlag.call(this); | |
}; | |
// Disable edge-activated hats and hats like "when key pressed" while paused. | |
const originalStartHats = vm.runtime.startHats; | |
vm.runtime.startHats = function () { | |
for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) { | |
args[_key] = arguments[_key]; | |
} | |
const hat = args[0]; | |
// These hats can be manually started by the user when paused or while single stepping. | |
const isUserInitiated = hat === "event_whenbroadcastreceived" || hat === "control_start_as_clone"; | |
if (pauseNewThreads) { | |
if (!isUserInitiated && !this.getIsEdgeActivatedHat(hat)) { | |
return []; | |
} | |
const newThreads = originalStartHats.apply(this, args); | |
for (const thread of newThreads) { | |
pauseThread(thread); | |
} | |
return newThreads; | |
} else if (paused && !isUserInitiated) { | |
return []; | |
} | |
return originalStartHats.apply(this, args); | |
}; | |
// Paused threads should not be counted as running when updating GUI state. | |
const originalGetMonitorThreadCount = vm.runtime._getMonitorThreadCount; | |
vm.runtime._getMonitorThreadCount = function (threads) { | |
let count = originalGetMonitorThreadCount.call(this, threads); | |
if (paused) { | |
for (const thread of threads) { | |
if (pausedThreadState.has(thread)) { | |
count++; | |
} | |
} | |
} | |
return count; | |
}; | |
}; | |
/***/ }), | |
/***/ "./src/addons/addons/debugger/performance.js": | |
/*!***************************************************!*\ | |
!*** ./src/addons/addons/debugger/performance.js ***! | |
\***************************************************/ | |
/*! exports provided: default */ | |
/***/ (function(module, __webpack_exports__, __webpack_require__) { | |
; | |
__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(_ref) { | |
let { | |
debug, | |
addon, | |
console, | |
msg | |
} = _ref; | |
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 = _ref2 => { | |
let { | |
title | |
} = _ref2; | |
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__) { | |
; | |
__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(_ref) { | |
let { | |
debug, | |
addon, | |
console, | |
msg | |
} = _ref; | |
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); | |
} | |
} | |
// pm: this isnt very useful if every thread will be compiled | |
// 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__) { | |
; | |
__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 (_ref) { | |
let { | |
addon, | |
console, | |
msg | |
} = _ref; | |
Object(_module_js__WEBPACK_IMPORTED_MODULE_0__["setup"])(addon.tab.traps.vm); | |
let logsTab; | |
const messagesLoggedBeforeLogsTabLoaded = []; | |
const logMessage = function logMessage() { | |
for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) { | |
args[_key] = arguments[_key]; | |
} | |
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: (_ref2, thread) => { | |
let { | |
content | |
} = _ref2; | |
logMessage(content, thread, "log"); | |
} | |
}); | |
addon.tab.addBlock("\u200B\u200Bwarn\u200B\u200B %s", { | |
args: ["content"], | |
displayName: msg("block-warn"), | |
callback: (_ref3, thread) => { | |
let { | |
content | |
} = _ref3; | |
logMessage(content, thread, "warn"); | |
} | |
}); | |
addon.tab.addBlock("\u200B\u200Berror\u200B\u200B %s", { | |
args: ["content"], | |
displayName: msg("block-error"), | |
callback: (_ref4, thread) => { | |
let { | |
content | |
} = _ref4; | |
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; | |
compilerWarning.hidden = true; | |
}; | |
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 = _ref5 => { | |
let { | |
text, | |
icon, | |
description | |
} = _ref5; | |
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 = _ref6 => { | |
let { | |
text, | |
icon | |
} = _ref6; | |
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 () { | |
for (var _len2 = arguments.length, args = new Array(_len2), _key2 = 0; _key2 < _len2; _key2++) { | |
args[_key2] = arguments[_key2]; | |
} | |
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" || block.opcode === "procedures_definition_return") { | |
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 () { | |
if (addon.settings.get("log_clear_greenflag")) { | |
logsTab.clearLogs(); | |
} | |
if (addon.settings.get("log_greenflag")) { | |
logsTab.addLog(msg("log-msg-flag-clicked"), null, "internal"); | |
} | |
for (var _len3 = arguments.length, args = new Array(_len3), _key3 = 0; _key3 < _len3; _key3++) { | |
args[_key3] = arguments[_key3]; | |
} | |
return ogGreenFlag.call(this, ...args); | |
}; | |
const ogMakeClone = vm.runtime.targets[0].constructor.prototype.makeClone; | |
vm.runtime.targets[0].constructor.prototype.makeClone = function () { | |
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"); | |
} | |
for (var _len4 = arguments.length, args = new Array(_len4), _key4 = 0; _key4 < _len4; _key4++) { | |
args[_key4] = arguments[_key4]; | |
} | |
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) { | |
if (addon.settings.get("log_broadcasts") && hat === "event_whenbroadcastreceived") { | |
logsTab.addLog(msg("log-msg-broadcasted", { | |
broadcast: optMatchFields.BROADCAST_OPTION | |
}), vm.runtime.sequencer.activeThread, "internal"); | |
} | |
for (var _len5 = arguments.length, args = new Array(_len5 > 2 ? _len5 - 2 : 0), _key5 = 2; _key5 < _len5; _key5++) { | |
args[_key5 - 2] = arguments[_key5]; | |
} | |
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__) { | |
; | |
__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__) { | |
; | |
__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 |