初始化提交
This commit is contained in:
157
common/blockly-core/plugins/screenshot.js
Normal file
157
common/blockly-core/plugins/screenshot.js
Normal file
@@ -0,0 +1,157 @@
|
||||
/**
|
||||
* @license
|
||||
* Copyright 2019 Google LLC
|
||||
* SPDX-License-Identifier: Apache-2.0
|
||||
*/
|
||||
|
||||
/**
|
||||
* @fileoverview Download screenshot.
|
||||
*/
|
||||
'use strict';
|
||||
goog.require('Blockly');
|
||||
goog.provide('Blockly.Screenshot');
|
||||
|
||||
/**
|
||||
* Convert an SVG datauri into a PNG datauri.
|
||||
* @param {string} data SVG datauri.
|
||||
* @param {number} width Image width.
|
||||
* @param {number} height Image height.
|
||||
* @param {!Function} callback Callback.
|
||||
*/
|
||||
Blockly.Screenshot.svgToPng_ = function (data, width, height, callback) {
|
||||
const canvas = document.createElement('canvas');
|
||||
const context = canvas.getContext('2d');
|
||||
const img = new Image();
|
||||
const maxSize = Math.max(width, height);
|
||||
const pixelDensity = Math.max(Math.min(10, 15000 / maxSize), window.devicePixelRatio || 1);
|
||||
canvas.width = width * pixelDensity;
|
||||
canvas.height = height * pixelDensity;
|
||||
img.onload = function () {
|
||||
context.drawImage(
|
||||
img,
|
||||
0,
|
||||
0,
|
||||
width,
|
||||
height,
|
||||
0,
|
||||
0,
|
||||
canvas.width,
|
||||
canvas.height
|
||||
);
|
||||
try {
|
||||
const dataUri = canvas.toDataURL('image/png');
|
||||
callback(dataUri);
|
||||
} catch (err) {
|
||||
console.warn('Error converting the workspace svg to a png');
|
||||
callback('');
|
||||
}
|
||||
};
|
||||
img.src = data;
|
||||
}
|
||||
|
||||
Blockly.Screenshot.getBase64Image_ = function(image) {
|
||||
return new Promise(resolve => {
|
||||
const img = new Image();
|
||||
img.crossOrigin = '';
|
||||
const href = image.getAttribute('xlink:href');
|
||||
if (!href) {
|
||||
resolve();
|
||||
}
|
||||
img.src = href;
|
||||
img.onload = function () {
|
||||
const canvas = document.createElement('canvas');
|
||||
canvas.width = img.width;
|
||||
canvas.height = img.height;
|
||||
const ctx = canvas.getContext('2d');
|
||||
ctx?.drawImage(img, 0, 0, img.width, img.height);
|
||||
const ext = img.src.substring(img.src.lastIndexOf('.') + 1).toLowerCase();
|
||||
const dataURL = canvas.toDataURL('image/' + ext);
|
||||
image.setAttribute('xlink:href', dataURL);
|
||||
resolve();
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* Create an SVG of the blocks on the workspace.
|
||||
* @param {!Blockly.WorkspaceSvg} workspace The workspace.
|
||||
* @param {!Function} callback Callback.
|
||||
* @param {string=} customCss Custom CSS to append to the SVG.
|
||||
*/
|
||||
Blockly.Screenshot.workspaceToSvg_ = function (workspace, callback, customCss) {
|
||||
// Go through all text areas and set their value.
|
||||
const textAreas = document.getElementsByTagName('textarea');
|
||||
for (let i = 0; i < textAreas.length; i++) {
|
||||
textAreas[i].innerHTML = textAreas[i].value;
|
||||
}
|
||||
|
||||
const bBox = workspace.getBlocksBoundingBox();
|
||||
const x = bBox.x || bBox.left;
|
||||
const y = bBox.y || bBox.top;
|
||||
const width = bBox.width || bBox.right - x;
|
||||
const height = bBox.height || bBox.bottom - y;
|
||||
|
||||
const blockCanvas = workspace.getCanvas();
|
||||
const clone = blockCanvas.cloneNode(true);
|
||||
clone.removeAttribute('transform');
|
||||
const $images = $(clone).find('image');
|
||||
let resolveList = [];
|
||||
for (let i = 0; i < $images.length; i++) {
|
||||
resolveList.push(this.getBase64Image_($images[i]));
|
||||
}
|
||||
|
||||
Promise.all(resolveList)
|
||||
.then(() => {
|
||||
const svg = document.createElementNS('http://www.w3.org/2000/svg', 'svg');
|
||||
svg.setAttribute('xmlns', 'http://www.w3.org/2000/svg');
|
||||
svg.appendChild(clone);
|
||||
svg.setAttribute('viewBox', x + ' ' + y + ' ' + width + ' ' + height);
|
||||
|
||||
svg.setAttribute(
|
||||
'class',
|
||||
'blocklySvg ' +
|
||||
(workspace.options.renderer || 'geras') +
|
||||
'-renderer ' +
|
||||
(workspace.getTheme ? workspace.getTheme().name + '-theme' : '')
|
||||
);
|
||||
svg.setAttribute('width', width);
|
||||
svg.setAttribute('height', height);
|
||||
svg.setAttribute('style', 'background-color: transparent');
|
||||
|
||||
const css = [].slice
|
||||
.call(document.head.querySelectorAll('style'))
|
||||
.filter(
|
||||
(el) =>
|
||||
/\.blocklySvg/.test(el.innerText) || el.id.indexOf('blockly-') === 0
|
||||
)
|
||||
.map((el) => el.innerText)
|
||||
.join('\n');
|
||||
const style = document.createElement('style');
|
||||
style.innerHTML = css + '\n' + customCss;
|
||||
svg.insertBefore(style, svg.firstChild);
|
||||
|
||||
let svgAsXML = new XMLSerializer().serializeToString(svg);
|
||||
svgAsXML = svgAsXML.replace(/ /g, ' ');
|
||||
const data = 'data:image/svg+xml,' + encodeURIComponent(svgAsXML);
|
||||
|
||||
Blockly.Screenshot.svgToPng_(data, width, height, callback);
|
||||
});
|
||||
}
|
||||
|
||||
/* eslint-disable no-unused-vars */
|
||||
/**
|
||||
* Download a screenshot of the blocks on a Blockly workspace.
|
||||
* @param {!Blockly.WorkspaceSvg} workspace The Blockly workspace.
|
||||
*/
|
||||
Blockly.Screenshot.downloadScreenshot = function (workspace) {
|
||||
Blockly.Screenshot.workspaceToSvg_(workspace, function (datauri) {
|
||||
const a = document.createElement('a');
|
||||
a.download = 'screenshot.png';
|
||||
a.target = '_self';
|
||||
a.href = datauri;
|
||||
document.body.appendChild(a);
|
||||
a.click();
|
||||
a.parentNode.removeChild(a);
|
||||
});
|
||||
}
|
||||
/* eslint-enable */
|
||||
Reference in New Issue
Block a user