Spaces:
Running
Running
(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__) { | |
; | |
__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__) { | |
; | |
__webpack_require__.r(__webpack_exports__); | |
/* harmony default export */ __webpack_exports__["default"] = (async function (_ref) { | |
let { | |
addon, | |
console, | |
msg | |
} = _ref; | |
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("blue-flag.svg", ""); | |
blocksMedia.set("control_stop.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 |