|
|
|
@ -6,18 +6,70 @@ import "./styles.css";
|
|
|
|
|
|
|
|
|
|
var elements = [];
|
|
|
|
|
|
|
|
|
|
function newElement(type, x, y) {
|
|
|
|
|
function newElement(type, x, y, width = 0, height = 0) {
|
|
|
|
|
const element = {
|
|
|
|
|
type: type,
|
|
|
|
|
x: x,
|
|
|
|
|
y: y,
|
|
|
|
|
width: 0,
|
|
|
|
|
height: 0,
|
|
|
|
|
width: width,
|
|
|
|
|
height: height,
|
|
|
|
|
isSelected: false
|
|
|
|
|
};
|
|
|
|
|
return element;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function exportAsPNG({ background, visibleOnly, padding = 10 }) {
|
|
|
|
|
clearSelection();
|
|
|
|
|
drawScene();
|
|
|
|
|
|
|
|
|
|
let subCanvasX1 = Infinity;
|
|
|
|
|
let subCanvasX2 = 0;
|
|
|
|
|
let subCanvasY1 = Infinity;
|
|
|
|
|
let subCanvasY2 = 0;
|
|
|
|
|
|
|
|
|
|
elements.forEach(element => {
|
|
|
|
|
subCanvasX1 = Math.min(subCanvasX1, getElementAbsoluteX1(element));
|
|
|
|
|
subCanvasX2 = Math.max(subCanvasX2, getElementAbsoluteX2(element));
|
|
|
|
|
subCanvasY1 = Math.min(subCanvasY1, getElementAbsoluteY1(element));
|
|
|
|
|
subCanvasY2 = Math.max(subCanvasY2, getElementAbsoluteY2(element));
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
let targetCanvas = canvas;
|
|
|
|
|
|
|
|
|
|
if ( visibleOnly ) {
|
|
|
|
|
targetCanvas = document.createElement('canvas');
|
|
|
|
|
targetCanvas.style.display = 'none';
|
|
|
|
|
document.body.appendChild(targetCanvas);
|
|
|
|
|
targetCanvas.width = subCanvasX2 - subCanvasX1 + padding * 2;
|
|
|
|
|
targetCanvas.height = subCanvasY2 - subCanvasY1 + padding * 2;
|
|
|
|
|
const targetCanvas_ctx = targetCanvas.getContext('2d');
|
|
|
|
|
|
|
|
|
|
if ( background ) {
|
|
|
|
|
targetCanvas_ctx.fillStyle = "#FFF";
|
|
|
|
|
targetCanvas_ctx.fillRect(0, 0, canvas.width, canvas.height);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
targetCanvas_ctx.drawImage(
|
|
|
|
|
canvas,
|
|
|
|
|
subCanvasX1 - padding, // x
|
|
|
|
|
subCanvasY1 - padding, // y
|
|
|
|
|
subCanvasX2 - subCanvasX1 + padding * 2, // width
|
|
|
|
|
subCanvasY2 - subCanvasY1 + padding * 2, // height
|
|
|
|
|
0,
|
|
|
|
|
0,
|
|
|
|
|
targetCanvas.width,
|
|
|
|
|
targetCanvas.height
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const link = document.createElement('a');
|
|
|
|
|
link.setAttribute('download', 'excalibur.png');
|
|
|
|
|
link.setAttribute('href', targetCanvas.toDataURL("image/png"));
|
|
|
|
|
link.click();
|
|
|
|
|
link.remove();
|
|
|
|
|
if ( targetCanvas !== canvas ) targetCanvas.remove();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function rotate(x1, y1, x2, y2, angle) {
|
|
|
|
|
// 𝑎′𝑥=(𝑎𝑥−𝑐𝑥)cos𝜃−(𝑎𝑦−𝑐𝑦)sin𝜃+𝑐𝑥
|
|
|
|
|
// 𝑎′𝑦=(𝑎𝑥−𝑐𝑥)sin𝜃+(𝑎𝑦−𝑐𝑦)cos𝜃+𝑐𝑦.
|
|
|
|
@ -150,7 +202,7 @@ function clearSelection() {
|
|
|
|
|
class App extends React.Component {
|
|
|
|
|
componentDidMount() {
|
|
|
|
|
this.onKeyDown = event => {
|
|
|
|
|
if (event.key === "Backspace") {
|
|
|
|
|
if (event.key === "Backspace" && event.target.nodeName !== "INPUT") {
|
|
|
|
|
for (var i = elements.length - 1; i >= 0; --i) {
|
|
|
|
|
if (elements[i].isSelected) {
|
|
|
|
|
elements.splice(i, 1);
|
|
|
|
@ -188,7 +240,10 @@ class App extends React.Component {
|
|
|
|
|
super();
|
|
|
|
|
this.state = {
|
|
|
|
|
draggingElement: null,
|
|
|
|
|
elementType: "selection"
|
|
|
|
|
elementType: "selection",
|
|
|
|
|
exportBackground: false,
|
|
|
|
|
exportVisibleOnly: true,
|
|
|
|
|
exportPadding: 10
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@ -210,7 +265,40 @@ class App extends React.Component {
|
|
|
|
|
);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
return (
|
|
|
|
|
return <>
|
|
|
|
|
<div className="exportWrapper">
|
|
|
|
|
<button onClick={() => {
|
|
|
|
|
exportAsPNG({
|
|
|
|
|
background: this.state.exportBackground,
|
|
|
|
|
visibleOnly: this.state.exportVisibleOnly,
|
|
|
|
|
padding: this.state.exportPadding
|
|
|
|
|
})
|
|
|
|
|
}}>Export to png</button>
|
|
|
|
|
<label>
|
|
|
|
|
<input type="checkbox"
|
|
|
|
|
checked={this.state.exportBackground}
|
|
|
|
|
onChange={e => {
|
|
|
|
|
this.setState({ exportBackground: e.target.checked })
|
|
|
|
|
}}
|
|
|
|
|
/> background
|
|
|
|
|
</label>
|
|
|
|
|
<label>
|
|
|
|
|
<input type="checkbox"
|
|
|
|
|
checked={this.state.exportVisibleOnly}
|
|
|
|
|
onChange={e => {
|
|
|
|
|
this.setState({ exportVisibleOnly: e.target.checked })
|
|
|
|
|
}}
|
|
|
|
|
/>
|
|
|
|
|
visible area only
|
|
|
|
|
</label>
|
|
|
|
|
(padding:
|
|
|
|
|
<input type="number" value={this.state.exportPadding}
|
|
|
|
|
onChange={e => {
|
|
|
|
|
this.setState({ exportPadding: e.target.value });
|
|
|
|
|
}}
|
|
|
|
|
disabled={!this.state.exportVisibleOnly}/>
|
|
|
|
|
px)
|
|
|
|
|
</div>
|
|
|
|
|
<div>
|
|
|
|
|
{/* Can't use the <ElementOption> form because ElementOption is re-defined
|
|
|
|
|
on every render, which would blow up and re-create the entire DOM tree,
|
|
|
|
@ -352,7 +440,7 @@ class App extends React.Component {
|
|
|
|
|
}}
|
|
|
|
|
/>
|
|
|
|
|
</div>
|
|
|
|
|
);
|
|
|
|
|
</>;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|