forked from extern/easydiffusion
commit
4fafc8aa67
4
ui/media/images/fa-eraser.svg
Normal file
4
ui/media/images/fa-eraser.svg
Normal file
@ -0,0 +1,4 @@
|
||||
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 576 576" width="24" height="24">
|
||||
<!--! Font Awesome Pro 6.2.1 by @fontawesome - https://fontawesome.com License - https://fontawesome.com/license (Commercial License) Copyright 2022 Fonticons, Inc.-->
|
||||
<path style="filter: drop-shadow(0px 0px 20px white)" d="M290.7 57.4 57.4 290.7c-25 25-25 65.5 0 90.5l80 80c12 12 28.3 18.7 45.3 18.7H512c17.7 0 32-14.3 32-32s-14.3-32-32-32H387.9l130.7-130.6c25-25 25-65.5 0-90.5L381.3 57.4c-25-25-65.5-25-90.5 0zm6.7 358.6H182.6l-80-80 124.7-124.7 137.4 137.4-67.3 67.3z"/>
|
||||
</svg>
|
After Width: | Height: | Size: 571 B |
4
ui/media/images/fa-eye-dropper.svg
Normal file
4
ui/media/images/fa-eye-dropper.svg
Normal file
@ -0,0 +1,4 @@
|
||||
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512" width="24" height="24">
|
||||
<!--! Font Awesome Pro 6.2.1 by @fontawesome - https://fontawesome.com License - https://fontawesome.com/license (Commercial License) Copyright 2022 Fonticons, Inc.-->
|
||||
<path style="filter: drop-shadow(0px 0px 20px white)" d="M341.6 29.2 240.1 130.8l-9.4-9.4c-12.5-12.5-32.8-12.5-45.3 0s-12.5 32.8 0 45.3l160 160c12.5 12.5 32.8 12.5 45.3 0s12.5-32.8 0-45.3l-9.4-9.4 101.5-101.6c39-39 39-102.2 0-141.1s-102.2-39-141.1 0zM55.4 323.3c-15 15-23.4 35.4-23.4 56.6v42.4L5.4 462.2c-8.5 12.7-6.8 29.6 4 40.4s27.7 12.5 40.4 4L89.7 480h42.4c21.2 0 41.6-8.4 56.6-23.4l120.7-120.7-45.3-45.3-120.7 120.7c-3 3-7.1 4.7-11.3 4.7H96v-36.1c0-4.2 1.7-8.3 4.7-11.3l120.7-120.7-45.3-45.3L55.4 323.3z"/>
|
||||
</svg>
|
After Width: | Height: | Size: 775 B |
4
ui/media/images/fa-fill.svg
Normal file
4
ui/media/images/fa-fill.svg
Normal file
@ -0,0 +1,4 @@
|
||||
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 576 576" width="24" height="24">
|
||||
<!--! Font Awesome Pro 6.2.1 by @fontawesome - https://fontawesome.com License - https://fontawesome.com/license (Commercial License) Copyright 2022 Fonticons, Inc.-->
|
||||
<path style="filter: drop-shadow(0px 0px 20px white)" d="M118.6 9.4c-12.5-12.5-32.7-12.5-45.2 0s-12.5 32.8 0 45.3l81.3 81.3-92.1 92.1c-37.5 37.5-37.5 98.3 0 135.8l117.5 117.5c37.5 37.5 98.3 37.5 135.8 0l190.4-190.5c28.1-28.1 28.1-73.7 0-101.8L354.9 37.7c-28.1-28.1-73.7-28.1-101.8 0l-53.1 53-81.4-81.3zM200 181.3l49.4 49.4c12.5 12.5 32.8 12.5 45.3 0s12.5-32.8 0-45.3L245.3 136l53.1-53.1c3.1-3.1 8.2-3.1 11.3 0l151.4 151.4c3.1 3.1 3.1 8.2 0 11.3L418.7 288H99.5c1.4-5.4 4.2-10.4 8.4-14.6l92.1-92.1z"/>
|
||||
</svg>
|
After Width: | Height: | Size: 763 B |
4
ui/media/images/fa-pencil.svg
Normal file
4
ui/media/images/fa-pencil.svg
Normal file
@ -0,0 +1,4 @@
|
||||
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512" width="24" height="24">
|
||||
<!--! Font Awesome Pro 6.2.1 by @fontawesome - https://fontawesome.com License - https://fontawesome.com/license (Commercial License) Copyright 2022 Fonticons, Inc.-->
|
||||
<path style="filter: drop-shadow(0px 0px 20px white)" d="m410.3 231 11.3-11.3-33.9-33.9-62.1-62.1-33.9-33.9-11.3 11.3-22.6 22.6L58.6 322.9c-10.4 10.4-18 23.3-22.2 37.4L1 480.7c-2.5 8.4-.2 17.5 6.1 23.7s15.3 8.5 23.7 6.1l120.3-35.4c14.1-4.2 27-11.8 37.4-22.2l199.2-199.2 22.6-22.7zM160 399.4l-9.1 22.7c-4 3.1-8.5 5.4-13.3 6.9l-78.2 23 23-78.1c1.4-4.9 3.8-9.4 6.9-13.3l22.7-9.1v32c0 8.8 7.2 16 16 16h32zM362.7 18.7l-14.4 14.5-22.6 22.6-11.4 11.3 33.9 33.9 62.1 62.1 33.9 33.9 11.3-11.3 22.6-22.6 14.5-14.5c25-25 25-65.5 0-90.5l-39.3-39.4c-25-25-65.5-25-90.5 0zm-47.4 168-144 144c-6.2 6.2-16.4 6.2-22.6 0s-6.2-16.4 0-22.6l144-144c6.2-6.2 16.4-6.2 22.6 0s6.2 16.4 0 22.6z"/>
|
||||
</svg>
|
After Width: | Height: | Size: 934 B |
@ -36,13 +36,14 @@ const defaultToolEnd = (editor, ctx, x, y, is_overlay = false) => {
|
||||
ctx.clearRect(0, 0, editor.width, editor.height)
|
||||
}
|
||||
}
|
||||
const toolDoNothing = (editor, ctx, x, y, is_overlay = false) => {}
|
||||
|
||||
const IMAGE_EDITOR_TOOLS = [
|
||||
{
|
||||
id: "draw",
|
||||
name: "Draw",
|
||||
icon: "fa-solid fa-pencil",
|
||||
cursor: "url(/media/images/fa-pencil.png) 0 24, pointer",
|
||||
cursor: "url(/media/images/fa-pencil.svg) 0 24, pointer",
|
||||
begin: defaultToolBegin,
|
||||
move: defaultToolMove,
|
||||
end: defaultToolEnd
|
||||
@ -51,7 +52,7 @@ const IMAGE_EDITOR_TOOLS = [
|
||||
id: "erase",
|
||||
name: "Erase",
|
||||
icon: "fa-solid fa-eraser",
|
||||
cursor: "url(/media/images/fa-eraser.png) 0 18, pointer",
|
||||
cursor: "url(/media/images/fa-eraser.svg) 0 14, pointer",
|
||||
begin: defaultToolBegin,
|
||||
move: (editor, ctx, x, y, is_overlay = false) => {
|
||||
ctx.lineTo(x, y)
|
||||
@ -78,27 +79,56 @@ const IMAGE_EDITOR_TOOLS = [
|
||||
}
|
||||
},
|
||||
{
|
||||
id: "colorpicker",
|
||||
name: "Color Picker",
|
||||
icon: "fa-solid fa-eye-dropper",
|
||||
cursor: "url(/media/images/fa-eye-dropper.png) 0 24, pointer",
|
||||
id: "fill",
|
||||
name: "Fill",
|
||||
icon: "fa-solid fa-fill",
|
||||
cursor: "url(/media/images/fa-fill.svg) 20 6, pointer",
|
||||
begin: (editor, ctx, x, y, is_overlay = false) => {
|
||||
var img_rgb = editor.layers.background.ctx.getImageData(x, y, 1, 1).data
|
||||
var drawn_rgb = editor.ctx_current.getImageData(x, y, 1, 1).data
|
||||
var drawn_opacity = drawn_rgb[3] / 255
|
||||
editor.custom_color_input.value = rgbToHex({
|
||||
r: (drawn_rgb[0] * drawn_opacity) + (img_rgb[0] * (1 - drawn_opacity)),
|
||||
g: (drawn_rgb[1] * drawn_opacity) + (img_rgb[1] * (1 - drawn_opacity)),
|
||||
b: (drawn_rgb[2] * drawn_opacity) + (img_rgb[2] * (1 - drawn_opacity)),
|
||||
})
|
||||
editor.custom_color_input.dispatchEvent(new Event("change"))
|
||||
if (!is_overlay) {
|
||||
var color = hexToRgb(ctx.fillStyle)
|
||||
color.a = parseInt(ctx.globalAlpha * 255) // layer.ctx.globalAlpha
|
||||
flood_fill(editor, ctx, parseInt(x), parseInt(y), color)
|
||||
}
|
||||
},
|
||||
move: (editor, ctx, x, y, is_overlay = false) => {},
|
||||
end: (editor, ctx, x, y, is_overlay = false) => {}
|
||||
move: toolDoNothing,
|
||||
end: toolDoNothing
|
||||
},
|
||||
{
|
||||
id: "colorpicker",
|
||||
name: "Picker",
|
||||
icon: "fa-solid fa-eye-dropper",
|
||||
cursor: "url(/media/images/fa-eye-dropper.svg) 0 24, pointer",
|
||||
begin: (editor, ctx, x, y, is_overlay = false) => {
|
||||
if (!is_overlay) {
|
||||
var img_rgb = editor.layers.background.ctx.getImageData(x, y, 1, 1).data
|
||||
var drawn_rgb = editor.ctx_current.getImageData(x, y, 1, 1).data
|
||||
var drawn_opacity = drawn_rgb[3] / 255
|
||||
editor.custom_color_input.value = rgbToHex({
|
||||
r: (drawn_rgb[0] * drawn_opacity) + (img_rgb[0] * (1 - drawn_opacity)),
|
||||
g: (drawn_rgb[1] * drawn_opacity) + (img_rgb[1] * (1 - drawn_opacity)),
|
||||
b: (drawn_rgb[2] * drawn_opacity) + (img_rgb[2] * (1 - drawn_opacity)),
|
||||
})
|
||||
editor.custom_color_input.dispatchEvent(new Event("change"))
|
||||
}
|
||||
},
|
||||
move: toolDoNothing,
|
||||
end: toolDoNothing
|
||||
}
|
||||
]
|
||||
|
||||
const IMAGE_EDITOR_ACTIONS = [
|
||||
{
|
||||
id: "fill_all",
|
||||
name: "Fill all",
|
||||
icon: "fa-solid fa-paint-roller",
|
||||
handler: (editor) => {
|
||||
editor.ctx_current.globalCompositeOperation = "source-over"
|
||||
editor.ctx_current.rect(0, 0, editor.width, editor.height)
|
||||
editor.ctx_current.fill()
|
||||
editor.setBrush()
|
||||
},
|
||||
trackHistory: true
|
||||
},
|
||||
{
|
||||
id: "clear",
|
||||
name: "Clear",
|
||||
@ -404,7 +434,6 @@ class ImageEditor {
|
||||
|
||||
if (this.inpainter) {
|
||||
this.selectOption("color", IMAGE_EDITOR_SECTIONS.find(s => s.name == "color").options.indexOf("#ffffff"))
|
||||
this.selectOption("opacity", IMAGE_EDITOR_SECTIONS.find(s => s.name == "opacity").options.indexOf(0.4))
|
||||
}
|
||||
|
||||
// initialize the right-side controls
|
||||
@ -467,8 +496,8 @@ class ImageEditor {
|
||||
width = (multiplier * width).toFixed()
|
||||
height = (multiplier * height).toFixed()
|
||||
}
|
||||
this.width = width
|
||||
this.height = height
|
||||
this.width = parseInt(width)
|
||||
this.height = parseInt(height)
|
||||
|
||||
this.container.style.width = width + "px"
|
||||
this.container.style.height = height + "px"
|
||||
@ -494,8 +523,10 @@ class ImageEditor {
|
||||
}
|
||||
setImage(url, width, height) {
|
||||
this.setSize(width, height)
|
||||
this.layers.drawing.ctx.clearRect(0, 0, this.width, this.height)
|
||||
this.layers.background.ctx.clearRect(0, 0, this.width, this.height)
|
||||
if (!(url && this.inpainter)) {
|
||||
this.layers.drawing.ctx.clearRect(0, 0, this.width, this.height)
|
||||
}
|
||||
if (url) {
|
||||
var image = new Image()
|
||||
image.onload = () => {
|
||||
@ -685,14 +716,6 @@ class ImageEditor {
|
||||
}
|
||||
}
|
||||
|
||||
function rgbToHex(rgb) {
|
||||
function componentToHex(c) {
|
||||
var hex = parseInt(c).toString(16)
|
||||
return hex.length == 1 ? "0" + hex : hex
|
||||
}
|
||||
return "#" + componentToHex(rgb.r) + componentToHex(rgb.g) + componentToHex(rgb.b)
|
||||
}
|
||||
|
||||
const imageEditor = new ImageEditor(document.getElementById("image-editor"))
|
||||
const imageInpainter = new ImageEditor(document.getElementById("image-inpainter"), true)
|
||||
|
||||
@ -707,3 +730,107 @@ document.getElementById("init_image_button_inpaint").addEventListener("click", (
|
||||
})
|
||||
|
||||
img2imgUnload() // no init image when the app starts
|
||||
|
||||
|
||||
function rgbToHex(rgb) {
|
||||
function componentToHex(c) {
|
||||
var hex = parseInt(c).toString(16)
|
||||
return hex.length == 1 ? "0" + hex : hex
|
||||
}
|
||||
return "#" + componentToHex(rgb.r) + componentToHex(rgb.g) + componentToHex(rgb.b)
|
||||
}
|
||||
|
||||
function hexToRgb(hex) {
|
||||
var result = /^#?([a-f\d]{2})([a-f\d]{2})([a-f\d]{2})$/i.exec(hex);
|
||||
return result ? {
|
||||
r: parseInt(result[1], 16),
|
||||
g: parseInt(result[2], 16),
|
||||
b: parseInt(result[3], 16)
|
||||
} : null;
|
||||
}
|
||||
|
||||
function pixelCompare(int1, int2) {
|
||||
return Math.abs(int1 - int2) < 4
|
||||
}
|
||||
|
||||
// adapted from https://ben.akrin.com/canvas_fill/fill_04.html
|
||||
function flood_fill(editor, the_canvas_context, x, y, color) {
|
||||
pixel_stack = [{x:x, y:y}] ;
|
||||
pixels = the_canvas_context.getImageData( 0, 0, editor.width, editor.height ) ;
|
||||
var linear_cords = ( y * editor.width + x ) * 4 ;
|
||||
var original_color = {r:pixels.data[linear_cords],
|
||||
g:pixels.data[linear_cords+1],
|
||||
b:pixels.data[linear_cords+2],
|
||||
a:pixels.data[linear_cords+3]} ;
|
||||
|
||||
var opacity = color.a / 255;
|
||||
var new_color = {
|
||||
r: parseInt((color.r * opacity) + (original_color.r * (1 - opacity))),
|
||||
g: parseInt((color.g * opacity) + (original_color.g * (1 - opacity))),
|
||||
b: parseInt((color.b * opacity) + (original_color.b * (1 - opacity)))
|
||||
}
|
||||
|
||||
if ((pixelCompare(new_color.r, original_color.r) &&
|
||||
pixelCompare(new_color.g, original_color.g) &&
|
||||
pixelCompare(new_color.b, original_color.b)))
|
||||
{
|
||||
return; // This color is already the color we want, so do nothing
|
||||
}
|
||||
var max_stack_size = editor.width * editor.height;
|
||||
while( pixel_stack.length > 0 && pixel_stack.length < max_stack_size ) {
|
||||
new_pixel = pixel_stack.shift() ;
|
||||
x = new_pixel.x ;
|
||||
y = new_pixel.y ;
|
||||
|
||||
linear_cords = ( y * editor.width + x ) * 4 ;
|
||||
while( y-->=0 &&
|
||||
(pixelCompare(pixels.data[linear_cords], original_color.r) &&
|
||||
pixelCompare(pixels.data[linear_cords+1], original_color.g) &&
|
||||
pixelCompare(pixels.data[linear_cords+2], original_color.b))) {
|
||||
linear_cords -= editor.width * 4 ;
|
||||
}
|
||||
linear_cords += editor.width * 4 ;
|
||||
y++ ;
|
||||
|
||||
var reached_left = false ;
|
||||
var reached_right = false ;
|
||||
while( y++<editor.height &&
|
||||
(pixelCompare(pixels.data[linear_cords], original_color.r) &&
|
||||
pixelCompare(pixels.data[linear_cords+1], original_color.g) &&
|
||||
pixelCompare(pixels.data[linear_cords+2], original_color.b))) {
|
||||
pixels.data[linear_cords] = new_color.r ;
|
||||
pixels.data[linear_cords+1] = new_color.g ;
|
||||
pixels.data[linear_cords+2] = new_color.b ;
|
||||
pixels.data[linear_cords+3] = 255 ;
|
||||
|
||||
if( x>0 ) {
|
||||
if( pixelCompare(pixels.data[linear_cords-4], original_color.r) &&
|
||||
pixelCompare(pixels.data[linear_cords-4+1], original_color.g) &&
|
||||
pixelCompare(pixels.data[linear_cords-4+2], original_color.b)) {
|
||||
if( !reached_left ) {
|
||||
pixel_stack.push( {x:x-1, y:y} ) ;
|
||||
reached_left = true ;
|
||||
}
|
||||
} else if( reached_left ) {
|
||||
reached_left = false ;
|
||||
}
|
||||
}
|
||||
|
||||
if( x<editor.width-1 ) {
|
||||
if( pixelCompare(pixels.data[linear_cords+4], original_color.r) &&
|
||||
pixelCompare(pixels.data[linear_cords+4+1], original_color.g) &&
|
||||
pixelCompare(pixels.data[linear_cords+4+2], original_color.b)) {
|
||||
if( !reached_right ) {
|
||||
pixel_stack.push( {x:x+1,y:y} ) ;
|
||||
reached_right = true ;
|
||||
}
|
||||
} else if( reached_right ) {
|
||||
reached_right = false ;
|
||||
}
|
||||
}
|
||||
|
||||
linear_cords += editor.width * 4 ;
|
||||
}
|
||||
}
|
||||
the_canvas_context.putImageData( pixels, 0, 0 ) ;
|
||||
}
|
||||
|
@ -13,8 +13,15 @@ function initTheme() {
|
||||
.filter(sheet => sheet.href?.startsWith(window.location.origin))
|
||||
.flatMap(sheet => Array.from(sheet.cssRules))
|
||||
.forEach(rule => {
|
||||
var selector = rule.selectorText; // TODO: also do selector == ":root", re-run un-set props
|
||||
var selector = rule.selectorText;
|
||||
if (selector && selector.startsWith(".theme-") && !selector.includes(" ")) {
|
||||
if (DEFAULT_THEME) { // re-add props that dont change (css needs this so they update correctly)
|
||||
Array.from(DEFAULT_THEME.rule.style)
|
||||
.filter(cssVariable => !Array.from(rule.style).includes(cssVariable))
|
||||
.forEach(cssVariable => {
|
||||
rule.style.setProperty(cssVariable, DEFAULT_THEME.rule.style.getPropertyValue(cssVariable));
|
||||
});
|
||||
}
|
||||
var theme_key = selector.substring(1);
|
||||
THEMES.push({
|
||||
key: theme_key,
|
||||
@ -62,12 +69,6 @@ function themeFieldChanged() {
|
||||
var theme = THEMES.find(t => t.key == theme_key);
|
||||
let borderColor = undefined
|
||||
if (theme) {
|
||||
// refresh variables incase they are back referencing
|
||||
Array.from(DEFAULT_THEME.rule.style)
|
||||
.filter(cssVariable => !Array.from(theme.rule.style).includes(cssVariable))
|
||||
.forEach(cssVariable => {
|
||||
body.style.setProperty(cssVariable, DEFAULT_THEME.rule.style.getPropertyValue(cssVariable));
|
||||
});
|
||||
borderColor = theme.rule.style.getPropertyValue('--input-border-color').trim()
|
||||
if (!borderColor.startsWith('#')) {
|
||||
borderColor = theme.rule.style.getPropertyValue('--theme-color-fallback')
|
||||
|
Loading…
Reference in New Issue
Block a user