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:
Victor Seiji Hariki 2022-11-20 14:59:11 -03:00
parent 8ebf273bfa
commit fe508f8b21
4 changed files with 178 additions and 2 deletions

View file

@ -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);

View file

@ -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
View 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);
}
);

View file

@ -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;
} }