simple command pattern and edit history
Implements command pattern for providing edit history capabilities to editing. For now, no implementation is done to support keyboard shortcuts, so the buttons are the only way to navigate. Also, only image insertion is supported for now. Waiting for the masking updates to implement masking history. Signed-off-by: Victor Seiji Hariki <victorseijih@gmail.com>
This commit is contained in:
parent
8ebf273bfa
commit
fe508f8b21
4 changed files with 178 additions and 2 deletions
|
@ -76,6 +76,37 @@
|
||||||
border-color: black;
|
border-color: black;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.toolbar {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
}
|
||||||
|
|
||||||
|
.toolbar > .tool {
|
||||||
|
flex: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.toolbar > .tool:not(:last-child) {
|
||||||
|
margin-right: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
button.tool {
|
||||||
|
background-color: rgb(0, 0, 50);
|
||||||
|
color: rgb(255, 255, 255);
|
||||||
|
border-radius: 5px;
|
||||||
|
cursor: pointer;
|
||||||
|
border: none;
|
||||||
|
text-align: center;
|
||||||
|
outline: none;
|
||||||
|
font-size: 15px;
|
||||||
|
padding: 5px;
|
||||||
|
margin-top: 5px;
|
||||||
|
margin-bottom: 5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
button.tool:hover {
|
||||||
|
background-color: #667;
|
||||||
|
}
|
||||||
|
|
||||||
.collapsible {
|
.collapsible {
|
||||||
background-color: rgb(0, 0, 0);
|
background-color: rgb(0, 0, 0);
|
||||||
color: rgb(255, 255, 255);
|
color: rgb(255, 255, 255);
|
||||||
|
|
|
@ -110,6 +110,10 @@
|
||||||
<br />
|
<br />
|
||||||
<hr>
|
<hr>
|
||||||
</div>
|
</div>
|
||||||
|
<div class="toolbar">
|
||||||
|
<button type="button" onclick="commands.undo()" class="tool">undo</button>
|
||||||
|
<button type="button" onclick="commands.redo()" class="tool">redo</button>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
@ -166,6 +170,7 @@
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|
||||||
|
<script src="js/commands.js" type="text/javascript"></script>
|
||||||
<script src="js/index.js" type="text/javascript"></script>
|
<script src="js/index.js" type="text/javascript"></script>
|
||||||
<script src="js/settingsbar.js" type="text/javascript"></script>
|
<script src="js/settingsbar.js" type="text/javascript"></script>
|
||||||
</body>
|
</body>
|
||||||
|
|
132
js/commands.js
Normal file
132
js/commands.js
Normal file
|
@ -0,0 +1,132 @@
|
||||||
|
/**
|
||||||
|
* Command pattern to allow for editing history
|
||||||
|
*/
|
||||||
|
const commands = {
|
||||||
|
current: -1,
|
||||||
|
history: [],
|
||||||
|
undo(n = 1) {
|
||||||
|
for (var i = 0; i < n && this.current > -1; i++) {
|
||||||
|
this.history[this.current--].undo();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
redo(n = 1) {
|
||||||
|
for (var i = 0; i < n && this.current + 1 < this.history.length; i++) {
|
||||||
|
this.history[++this.current].redo();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* These are basic commands that can be done/undone
|
||||||
|
*
|
||||||
|
* They must contain a 'run' method that performs the action the first time,
|
||||||
|
* a 'undo' method that undoes that action and a 'redo' method that does the
|
||||||
|
* action again, but without requiring parameters. 'redo' is by default the
|
||||||
|
* same as 'run'.
|
||||||
|
*
|
||||||
|
* The 'run' and 'redo' functions will receive a 'options' parameter which will be
|
||||||
|
* forwarded directly to the operation, and a 'state' parameter that
|
||||||
|
* can be used to store state for undoing things.
|
||||||
|
*
|
||||||
|
* The 'state' object will be passed to the 'undo' function as well.
|
||||||
|
*/
|
||||||
|
createCommand(name, run, undo, redo = run) {
|
||||||
|
const command = function runWrapper(options) {
|
||||||
|
// Create copy of options and state object
|
||||||
|
const copy = {};
|
||||||
|
Object.assign(copy, options);
|
||||||
|
const state = {};
|
||||||
|
|
||||||
|
// Attempt to run command
|
||||||
|
try {
|
||||||
|
run(copy, state);
|
||||||
|
} catch (e) {
|
||||||
|
console.warn(
|
||||||
|
`Error while running command '${name}' with options:`
|
||||||
|
);
|
||||||
|
console.warn(copy);
|
||||||
|
console.warn(e);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const undoWrapper = () => {
|
||||||
|
console.debug(`Undoing ${name}, currently ${commands.current}`);
|
||||||
|
undo(state);
|
||||||
|
};
|
||||||
|
const redoWrapper = () => {
|
||||||
|
console.debug(`Redoing ${name}, currently ${commands.current}`);
|
||||||
|
redo(copy, state);
|
||||||
|
};
|
||||||
|
|
||||||
|
// Add to history
|
||||||
|
if (commands.history.length > commands.current + 1)
|
||||||
|
commands.history.splice(commands.current + 1);
|
||||||
|
commands.history.push({ undo: undoWrapper, redo: redoWrapper });
|
||||||
|
commands.current++;
|
||||||
|
};
|
||||||
|
|
||||||
|
this.types[name] = command;
|
||||||
|
|
||||||
|
return command;
|
||||||
|
},
|
||||||
|
runCommand(name, options) {
|
||||||
|
this.types[name](options);
|
||||||
|
},
|
||||||
|
types: {},
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Draw Image Command, used to draw a Image to a context
|
||||||
|
*/
|
||||||
|
commands.createCommand(
|
||||||
|
"drawImage",
|
||||||
|
(options, state) => {
|
||||||
|
if (
|
||||||
|
!options ||
|
||||||
|
options.image === undefined ||
|
||||||
|
options.x === undefined ||
|
||||||
|
options.y === undefined
|
||||||
|
)
|
||||||
|
throw "Command drawImage requires options in the format: {image, x, y, ctx?}";
|
||||||
|
|
||||||
|
// Check if we have state
|
||||||
|
if (!state.context) {
|
||||||
|
const context = options.ctx || imgCtx;
|
||||||
|
state.context = context;
|
||||||
|
|
||||||
|
// Saving what was in the canvas before the command
|
||||||
|
const imgData = context.getImageData(
|
||||||
|
options.x,
|
||||||
|
options.y,
|
||||||
|
options.image.width,
|
||||||
|
options.image.height
|
||||||
|
);
|
||||||
|
state.box = {
|
||||||
|
x: options.x,
|
||||||
|
y: options.y,
|
||||||
|
w: options.image.width,
|
||||||
|
h: options.image.height,
|
||||||
|
};
|
||||||
|
// Create Image
|
||||||
|
const cutout = document.createElement("canvas");
|
||||||
|
cutout.width = state.box.w;
|
||||||
|
cutout.height = state.box.h;
|
||||||
|
cutout.getContext("2d").putImageData(imgData, 0, 0);
|
||||||
|
state.original = new Image();
|
||||||
|
state.original.src = cutout.toDataURL();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Apply command
|
||||||
|
state.context.drawImage(options.image, state.box.x, state.box.y);
|
||||||
|
},
|
||||||
|
(state) => {
|
||||||
|
// Clear destination area
|
||||||
|
state.context.clearRect(
|
||||||
|
state.box.x,
|
||||||
|
state.box.y,
|
||||||
|
state.box.w,
|
||||||
|
state.box.h
|
||||||
|
);
|
||||||
|
// Undo
|
||||||
|
state.context.drawImage(state.original, state.box.x, state.box.y);
|
||||||
|
}
|
||||||
|
);
|
12
js/index.js
12
js/index.js
|
@ -173,7 +173,11 @@ function drop(imageParams) {
|
||||||
}
|
}
|
||||||
|
|
||||||
function writeArbitraryImage(img, x, y) {
|
function writeArbitraryImage(img, x, y) {
|
||||||
imgCtx.drawImage(img, x, y);
|
commands.runCommand('drawImage', {
|
||||||
|
x,
|
||||||
|
y,
|
||||||
|
image: img,
|
||||||
|
});
|
||||||
blockNewImages = false;
|
blockNewImages = false;
|
||||||
placingArbitraryImage = false;
|
placingArbitraryImage = false;
|
||||||
document.getElementById("preloadImage").files = null;
|
document.getElementById("preloadImage").files = null;
|
||||||
|
@ -329,7 +333,11 @@ function clearPaintedMask() {
|
||||||
function placeImage() {
|
function placeImage() {
|
||||||
const img = new Image();
|
const img = new Image();
|
||||||
img.onload = function () {
|
img.onload = function () {
|
||||||
imgCtx.drawImage(img, tmpImgXYWH.x, tmpImgXYWH.y);
|
commands.runCommand('drawImage', {
|
||||||
|
x: tmpImgXYWH.x,
|
||||||
|
y: tmpImgXYWH.y,
|
||||||
|
image: img,
|
||||||
|
});
|
||||||
tmpImgXYWH = {};
|
tmpImgXYWH = {};
|
||||||
returnedImages = null;
|
returnedImages = null;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue