(window["webpackJsonpGUI"] = window["webpackJsonpGUI"] || []).push([["addon-entry-blocks2image"],{ /***/ "./src/addons/addons/blocks2image/_runtime_entry.js": /*!**********************************************************!*\ !*** ./src/addons/addons/blocks2image/_runtime_entry.js ***! \**********************************************************/ /*! exports provided: resources */ /***/ (function(module, __webpack_exports__, __webpack_require__) { "use strict"; __webpack_require__.r(__webpack_exports__); /* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "resources", function() { return resources; }); /* harmony import */ var _userscript_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./userscript.js */ "./src/addons/addons/blocks2image/userscript.js"); /* generated by pull.js */ const resources = { "userscript.js": _userscript_js__WEBPACK_IMPORTED_MODULE_0__["default"] }; /***/ }), /***/ "./src/addons/addons/blocks2image/userscript.js": /*!******************************************************!*\ !*** ./src/addons/addons/blocks2image/userscript.js ***! \******************************************************/ /*! exports provided: default */ /***/ (function(module, __webpack_exports__, __webpack_require__) { "use strict"; __webpack_require__.r(__webpack_exports__); /* harmony default export */ __webpack_exports__["default"] = (async function ({ addon, console, msg }) { function makeStyle() { let style = document.createElement("style"); style.textContent = "\n .blocklyText {\n fill: #fff;\n font-family: \"Helvetica Neue\", Helvetica, sans-serif;\n font-size: 12pt;\n font-weight: 500;\n }\n .blocklyNonEditableText>text, .blocklyEditableText>text {\n fill: #575E75;\n }\n .blocklyDropdownText {\n fill: #fff !important;\n }\n "; for (let userstyle of document.querySelectorAll(".scratch-addons-style[data-addons*='editor-theme3']")) { if (userstyle.disabled) continue; style.textContent += userstyle.textContent; } return style; } function setCSSVars(element) { for (let property of document.documentElement.style) { if (property.startsWith("--editorTheme3-")) element.style.setProperty(property, document.documentElement.style.getPropertyValue(property)); } } let exSVG = document.createElementNS("http://www.w3.org/2000/svg", "svg"); exSVG.setAttribute("xmlns:html", "http://www.w3.org/1999/xhtml"); exSVG.setAttribute("xmlns:xlink", "http://www.w3.org/1999/xlink"); exSVG.setAttribute("version", "1.1"); // blocks-media as base64 for svg inline image let blocksMedia = new Map(); blocksMedia.set("repeat.svg", ""); blocksMedia.set("green-flag.svg", ""); blocksMedia.set("rotate-left.svg", ""); blocksMedia.set("rotate-right.svg", ""); blocksMedia.set("dropdown-arrow.svg", ""); addon.tab.createBlockContextMenu(items => { var _svgchild$childNodes, _svgchild$childNodes2; if (addon.self.disabled) return items; let svgchild = document.querySelector("svg.blocklySvg g.blocklyBlockCanvas"); const pasteItemIndex = items.findIndex(obj => obj._isDevtoolsFirstItem); const insertBeforeIndex = pasteItemIndex !== -1 ? // If "paste" button exists, add own items before it pasteItemIndex : // If there's no such button, insert at end items.length; items.splice(insertBeforeIndex, 0, { enabled: !!(svgchild !== null && svgchild !== void 0 && (_svgchild$childNodes = svgchild.childNodes) !== null && _svgchild$childNodes !== void 0 && _svgchild$childNodes.length), text: msg("export_all_to_SVG"), callback: () => { exportBlock(false); }, separator: true }, { enabled: !!(svgchild !== null && svgchild !== void 0 && (_svgchild$childNodes2 = svgchild.childNodes) !== null && _svgchild$childNodes2 !== void 0 && _svgchild$childNodes2.length), text: msg("export_all_to_PNG"), callback: () => { exportBlock(true); }, separator: false }); return items; }, { workspace: true }); addon.tab.createBlockContextMenu((items, block) => { if (addon.self.disabled) return items; const makeSpaceItemIndex = items.findIndex(obj => obj._isDevtoolsFirstItem); const insertBeforeIndex = makeSpaceItemIndex !== -1 ? // If "make space" button exists, add own items before it makeSpaceItemIndex : // If there's no such button, insert at end items.length; items.splice(insertBeforeIndex, 0, { enabled: true, text: msg("export_selected_to_SVG"), callback: () => { exportBlock(false, block); }, separator: true }, { enabled: true, text: msg("export_selected_to_PNG"), callback: () => { exportBlock(true, block); }, separator: false }); return items; }, { blocks: true }); function exportBlock(isExportPNG, block) { let svg; if (block) { svg = selectedBlocks(isExportPNG, block); } else { svg = allBlocks(isExportPNG); } // resolve nbsp whitespace svg.querySelectorAll("text").forEach(text => { text.innerHTML = text.innerHTML.replace(/ /g, " "); }); // resolve image path let scratchURL = window.location.origin; svg.querySelectorAll("image").forEach(item => { let builtinSvgData = blocksMedia.get(item.getAttribute("xlink:href").substring(item.getAttribute("xlink:href").lastIndexOf("/") + 1)); if (builtinSvgData) { // replace svg file path (official) to inline svg item.setAttribute("xlink:href", builtinSvgData); } else if (item.getAttribute("xlink:href").indexOf("/static/") === 0) { // replace link path for third party website item.setAttribute("xlink:href", scratchURL + item.getAttribute("xlink:href").slice(0)); } else if (item.getAttribute("xlink:href").indexOf("./static/") === 0) { item.setAttribute("xlink:href", scratchURL + item.getAttribute("xlink:href").slice(1)); } else if (item.getAttribute("xlink:href").indexOf("static/") === 0) { item.setAttribute("xlink:href", scratchURL + "/" + item.getAttribute("xlink:href")); } }); if (!isExportPNG) { exportData(new XMLSerializer().serializeToString(svg)); } else { exportPNG(svg); } } function selectedBlocks(isExportPNG, block) { let svg = exSVG.cloneNode(); let svgchild = block.svgGroup_; svgchild = svgchild.cloneNode(true); let dataShapes = svgchild.getAttribute("data-shapes"); svgchild.setAttribute("transform", "translate(0,".concat(dataShapes === "hat" ? "18" : "0", ") ").concat(isExportPNG ? "scale(2)" : "")); setCSSVars(svg); svg.append(makeStyle()); svg.append(svgchild); return svg; } function allBlocks(isExportPNG) { let svg = exSVG.cloneNode(); let svgchild = document.querySelector("svg.blocklySvg g.blocklyBlockCanvas"); svgchild = svgchild.cloneNode(true); let xArr = []; let yArr = []; svgchild.childNodes.forEach(g => { let x = g.getAttribute("transform").match(/translate\((.*?),(.*?)\)/)[1] || 0; let y = g.getAttribute("transform").match(/translate\((.*?),(.*?)\)/)[2] || 0; xArr.push(x * (isExportPNG ? 2 : 1)); yArr.push(y * (isExportPNG ? 2 : 1)); g.style.display = ""; // because of TW scratch-blocks changes }); svgchild.setAttribute("transform", "translate(".concat(-Math.min(...xArr), ",").concat(-Math.min(...yArr) + 18 * (isExportPNG ? 2 : 1), ") ").concat(isExportPNG ? "scale(2)" : "")); setCSSVars(svg); svg.append(makeStyle()); svg.append(svgchild); return svg; } function exportData(text) { const saveLink = document.createElement("a"); document.body.appendChild(saveLink); const data = new Blob([text], { type: "text" }); const url = window.URL.createObjectURL(data); saveLink.href = url; // File name: project-DATE-TIME const date = new Date(); const timestamp = "".concat(date.toLocaleDateString(), "-").concat(date.toLocaleTimeString()); saveLink.download = "block_".concat(timestamp, ".svg"); saveLink.click(); window.URL.revokeObjectURL(url); document.body.removeChild(saveLink); } function exportPNG(svg) { const serializer = new XMLSerializer(); const iframe = document.createElement("iframe"); // iframe.style.display = "none" document.body.append(iframe); iframe.contentDocument.write(serializer.serializeToString(svg)); let { width, height } = iframe.contentDocument.body.querySelector("svg g").getBoundingClientRect(); height = height + 20 * 2; // hat block height restore svg.setAttribute("width", width + "px"); svg.setAttribute("height", height + "px"); let canvas = document.createElement("canvas"); let ctx = canvas.getContext("2d"); let img = document.createElement("img"); img.setAttribute("src", "data:image/svg+xml;base64," + btoa(unescape(encodeURIComponent(serializer.serializeToString(svg))))); img.onload = function () { canvas.height = img.height; canvas.width = img.width; ctx.drawImage(img, 0, 0, img.width, img.height); // Now is done let dataURL = canvas.toDataURL("image/png"); let link = document.createElement("a"); const date = new Date(); const timestamp = "".concat(date.toLocaleDateString(), "-").concat(date.toLocaleTimeString()); link.download = "block_".concat(timestamp, ".png"); link.href = dataURL; link.click(); iframe.remove(); }; } }); /***/ }) }]); //# sourceMappingURL=addon-entry-blocks2image.js.map