2022-11-27 01:35:16 +00:00
|
|
|
const setMask = (state) => {
|
2022-11-29 20:55:25 +00:00
|
|
|
const canvas = imageCollection.layers.mask.canvas;
|
2022-11-27 01:35:16 +00:00
|
|
|
switch (state) {
|
|
|
|
case "clear":
|
|
|
|
canvas.classList.remove("hold");
|
|
|
|
canvas.classList.add("display", "clear");
|
|
|
|
break;
|
|
|
|
case "hold":
|
|
|
|
canvas.classList.remove("clear");
|
|
|
|
canvas.classList.add("display", "hold");
|
|
|
|
break;
|
|
|
|
case "neutral":
|
|
|
|
canvas.classList.remove("clear", "hold");
|
|
|
|
canvas.classList.add("display");
|
|
|
|
break;
|
|
|
|
case "none":
|
|
|
|
canvas.classList.remove("display", "hold", "clear");
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
console.debug(`Invalid mask type: ${state}`);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2022-11-24 15:30:13 +00:00
|
|
|
const _mask_brush_draw_callback = (evn, state) => {
|
2022-11-29 20:55:25 +00:00
|
|
|
maskPaintCtx.globalCompositeOperation = "source-over";
|
|
|
|
maskPaintCtx.strokeStyle = "black";
|
2022-11-22 22:24:55 +00:00
|
|
|
|
2022-11-29 20:55:25 +00:00
|
|
|
maskPaintCtx.lineWidth = state.brushSize;
|
|
|
|
maskPaintCtx.beginPath();
|
|
|
|
maskPaintCtx.moveTo(
|
|
|
|
evn.px === undefined ? evn.x : evn.px,
|
|
|
|
evn.py === undefined ? evn.y : evn.py
|
|
|
|
);
|
|
|
|
maskPaintCtx.lineTo(evn.x, evn.y);
|
|
|
|
maskPaintCtx.lineJoin = maskPaintCtx.lineCap = "round";
|
|
|
|
maskPaintCtx.stroke();
|
2022-11-22 22:24:55 +00:00
|
|
|
};
|
|
|
|
|
2022-11-24 15:30:13 +00:00
|
|
|
const _mask_brush_erase_callback = (evn, state) => {
|
2022-11-29 20:55:25 +00:00
|
|
|
maskPaintCtx.globalCompositeOperation = "destination-out";
|
|
|
|
maskPaintCtx.strokeStyle = "black";
|
2022-11-22 22:24:55 +00:00
|
|
|
|
2022-11-29 20:55:25 +00:00
|
|
|
maskPaintCtx.lineWidth = state.brushSize;
|
|
|
|
maskPaintCtx.beginPath();
|
|
|
|
maskPaintCtx.moveTo(
|
|
|
|
evn.px === undefined ? evn.x : evn.px,
|
|
|
|
evn.py === undefined ? evn.y : evn.py
|
|
|
|
);
|
|
|
|
maskPaintCtx.lineTo(evn.x, evn.y);
|
|
|
|
maskPaintCtx.lineJoin = maskPaintCtx.lineCap = "round";
|
|
|
|
maskPaintCtx.stroke();
|
2022-11-22 22:24:55 +00:00
|
|
|
};
|
2022-11-24 15:30:13 +00:00
|
|
|
|
2022-12-01 21:05:14 +00:00
|
|
|
const _paint_mb_cursor = (state) => {
|
|
|
|
const v = state.brushSize;
|
|
|
|
state.cursorLayer.resize(v + 20, v + 20);
|
|
|
|
|
|
|
|
const ctx = state.cursorLayer.ctx;
|
|
|
|
|
|
|
|
ctx.clearRect(0, 0, v + 20, v + 20);
|
|
|
|
ctx.beginPath();
|
|
|
|
ctx.arc(
|
|
|
|
(v + 20) / 2,
|
|
|
|
(v + 20) / 2,
|
|
|
|
state.brushSize / 2,
|
|
|
|
0,
|
|
|
|
2 * Math.PI,
|
|
|
|
true
|
|
|
|
);
|
|
|
|
ctx.fillStyle = "#FFFFFF50";
|
|
|
|
|
|
|
|
ctx.fill();
|
|
|
|
|
|
|
|
if (state.preview) {
|
|
|
|
ctx.strokeStyle = "#000F";
|
|
|
|
ctx.setLineDash([4, 2]);
|
|
|
|
ctx.stroke();
|
|
|
|
ctx.setLineDash([]);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2022-11-24 15:30:13 +00:00
|
|
|
const maskBrushTool = () =>
|
|
|
|
toolbar.registerTool(
|
|
|
|
"res/icons/paintbrush.svg",
|
|
|
|
"Mask Brush",
|
|
|
|
(state, opt) => {
|
2022-12-01 21:05:14 +00:00
|
|
|
// New layer for the cursor
|
|
|
|
state.cursorLayer = imageCollection.registerLayer(null, {
|
|
|
|
after: maskPaintLayer,
|
|
|
|
bb: {x: 0, y: 0, w: state.brushSize + 20, h: state.brushSize + 20},
|
|
|
|
});
|
|
|
|
|
|
|
|
_paint_mb_cursor(state);
|
|
|
|
|
2022-11-24 15:30:13 +00:00
|
|
|
// Draw new cursor immediately
|
|
|
|
ovCtx.clearRect(0, 0, ovCanvas.width, ovCanvas.height);
|
2022-11-29 20:55:25 +00:00
|
|
|
state.movecb({...mouse.coords.world.pos});
|
2022-11-24 15:30:13 +00:00
|
|
|
|
|
|
|
// Start Listeners
|
2022-11-29 20:55:25 +00:00
|
|
|
mouse.listen.world.onmousemove.on(state.movecb);
|
|
|
|
mouse.listen.world.onwheel.on(state.wheelcb);
|
|
|
|
mouse.listen.world.btn.left.onpaintstart.on(state.drawcb);
|
|
|
|
mouse.listen.world.btn.left.onpaint.on(state.drawcb);
|
|
|
|
mouse.listen.world.btn.right.onpaintstart.on(state.erasecb);
|
|
|
|
mouse.listen.world.btn.right.onpaint.on(state.erasecb);
|
2022-11-27 01:35:16 +00:00
|
|
|
|
|
|
|
// Display Mask
|
|
|
|
setMask("neutral");
|
2022-11-24 15:30:13 +00:00
|
|
|
},
|
|
|
|
(state, opt) => {
|
2022-12-01 21:05:14 +00:00
|
|
|
// Don't want to keep hogging resources
|
|
|
|
imageCollection.deleteLayer(state.cursorLayer);
|
|
|
|
state.cursorLayer = null;
|
|
|
|
|
2022-11-24 15:30:13 +00:00
|
|
|
// Clear Listeners
|
2022-11-29 20:55:25 +00:00
|
|
|
mouse.listen.world.onmousemove.clear(state.movecb);
|
|
|
|
mouse.listen.world.onwheel.clear(state.wheelcb);
|
|
|
|
mouse.listen.world.btn.left.onpaintstart.clear(state.drawcb);
|
|
|
|
mouse.listen.world.btn.left.onpaint.clear(state.drawcb);
|
|
|
|
mouse.listen.world.btn.right.onpaintstart.clear(state.erasecb);
|
|
|
|
mouse.listen.world.btn.right.onpaint.clear(state.erasecb);
|
2022-11-27 01:35:16 +00:00
|
|
|
|
|
|
|
// Hide Mask
|
|
|
|
setMask("none");
|
2022-11-27 02:08:37 +00:00
|
|
|
state.ctxmenu.previewMaskButton.classList.remove("active");
|
2022-11-27 02:00:41 +00:00
|
|
|
maskPaintCanvas.classList.remove("opaque");
|
2022-11-27 02:08:37 +00:00
|
|
|
state.preview = false;
|
2022-11-24 15:30:13 +00:00
|
|
|
},
|
|
|
|
{
|
|
|
|
init: (state) => {
|
|
|
|
state.config = {
|
|
|
|
brushScrollSpeed: 1 / 4,
|
|
|
|
minBrushSize: 10,
|
|
|
|
maxBrushSize: 500,
|
|
|
|
};
|
|
|
|
|
|
|
|
state.brushSize = 64;
|
|
|
|
state.setBrushSize = (size) => {
|
|
|
|
state.brushSize = size;
|
|
|
|
state.ctxmenu.brushSizeRange.value = size;
|
|
|
|
state.ctxmenu.brushSizeText.value = size;
|
|
|
|
};
|
|
|
|
|
2022-11-27 02:08:37 +00:00
|
|
|
state.preview = false;
|
|
|
|
|
2022-12-01 21:05:14 +00:00
|
|
|
state.clearPrevCursor = () =>
|
2022-11-29 20:55:25 +00:00
|
|
|
ovCtx.clearRect(0, 0, ovCanvas.width, ovCanvas.height);
|
2022-11-27 02:08:37 +00:00
|
|
|
|
2022-12-01 21:05:14 +00:00
|
|
|
state.movecb = (evn) => {
|
|
|
|
state.cursorLayer.moveTo(
|
|
|
|
evn.x - state.brushSize / 2 - 10,
|
|
|
|
evn.y - state.brushSize / 2 - 10
|
|
|
|
);
|
2022-11-27 02:08:37 +00:00
|
|
|
|
2022-12-01 21:05:14 +00:00
|
|
|
state.clearPrevCursor = () =>
|
|
|
|
ovCtx.clearRect(
|
|
|
|
evn.x - state.brushSize / 2 - 10,
|
|
|
|
evn.y - state.brushSize / 2 - 10,
|
|
|
|
evn.x + state.brushSize / 2 + 10,
|
|
|
|
evn.y + state.brushSize / 2 + 10
|
|
|
|
);
|
2022-11-24 15:30:13 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
state.wheelcb = (evn) => {
|
2022-11-29 20:55:25 +00:00
|
|
|
if (!evn.evn.ctrlKey) {
|
2022-11-24 15:30:13 +00:00
|
|
|
state.brushSize = state.setBrushSize(
|
|
|
|
state.brushSize -
|
|
|
|
Math.floor(state.config.brushScrollSpeed * evn.delta)
|
|
|
|
);
|
|
|
|
state.movecb(evn);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
state.drawcb = (evn) => _mask_brush_draw_callback(evn, state);
|
|
|
|
state.erasecb = (evn) => _mask_brush_erase_callback(evn, state);
|
|
|
|
},
|
|
|
|
populateContextMenu: (menu, state) => {
|
|
|
|
if (!state.ctxmenu) {
|
|
|
|
state.ctxmenu = {};
|
|
|
|
const brushSizeSlider = _toolbar_input.slider(
|
|
|
|
state,
|
|
|
|
"brushSize",
|
|
|
|
"Brush Size",
|
2022-12-03 10:50:23 +00:00
|
|
|
{
|
|
|
|
min: state.config.minBrushSize,
|
|
|
|
max: state.config.maxBrushSize,
|
|
|
|
step: 5,
|
|
|
|
textStep: 1,
|
|
|
|
cb: (v) => {
|
|
|
|
if (!state.cursorLayer) return;
|
|
|
|
_paint_mb_cursor(state);
|
|
|
|
},
|
2022-12-01 21:05:14 +00:00
|
|
|
}
|
2022-11-24 15:30:13 +00:00
|
|
|
);
|
|
|
|
state.ctxmenu.brushSizeSlider = brushSizeSlider.slider;
|
|
|
|
state.setBrushSize = brushSizeSlider.setValue;
|
2022-11-27 02:00:41 +00:00
|
|
|
|
|
|
|
// Some mask-related action buttons
|
|
|
|
const actionArray = document.createElement("div");
|
|
|
|
actionArray.classList.add("button-array");
|
|
|
|
|
|
|
|
const clearMaskButton = document.createElement("button");
|
|
|
|
clearMaskButton.classList.add("button", "tool");
|
|
|
|
clearMaskButton.textContent = "Clear";
|
|
|
|
clearMaskButton.title = "Clears Painted Mask";
|
|
|
|
clearMaskButton.onclick = () => {
|
|
|
|
maskPaintCtx.clearRect(
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
maskPaintCanvas.width,
|
|
|
|
maskPaintCanvas.height
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
const previewMaskButton = document.createElement("button");
|
|
|
|
previewMaskButton.classList.add("button", "tool");
|
|
|
|
previewMaskButton.textContent = "Preview";
|
|
|
|
previewMaskButton.title = "Displays Mask with Full Opacity";
|
|
|
|
previewMaskButton.onclick = () => {
|
|
|
|
if (previewMaskButton.classList.contains("active")) {
|
|
|
|
maskPaintCanvas.classList.remove("opaque");
|
2022-11-27 02:08:37 +00:00
|
|
|
state.preview = false;
|
2022-12-01 21:05:14 +00:00
|
|
|
_paint_mb_cursor(state);
|
2022-11-27 02:00:41 +00:00
|
|
|
} else {
|
|
|
|
maskPaintCanvas.classList.add("opaque");
|
2022-11-27 02:08:37 +00:00
|
|
|
state.preview = true;
|
2022-12-01 21:05:14 +00:00
|
|
|
_paint_mb_cursor(state);
|
2022-11-27 02:00:41 +00:00
|
|
|
}
|
|
|
|
previewMaskButton.classList.toggle("active");
|
|
|
|
};
|
|
|
|
|
2022-11-27 02:08:37 +00:00
|
|
|
state.ctxmenu.previewMaskButton = previewMaskButton;
|
|
|
|
|
2022-11-27 02:00:41 +00:00
|
|
|
actionArray.appendChild(clearMaskButton);
|
|
|
|
actionArray.appendChild(previewMaskButton);
|
|
|
|
|
|
|
|
state.ctxmenu.actionArray = actionArray;
|
2022-11-24 15:30:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
menu.appendChild(state.ctxmenu.brushSizeSlider);
|
2022-11-27 02:00:41 +00:00
|
|
|
menu.appendChild(state.ctxmenu.actionArray);
|
2022-11-24 15:30:13 +00:00
|
|
|
},
|
|
|
|
shortcut: "M",
|
|
|
|
}
|
|
|
|
);
|