mirror of
https://github.com/zadam/trilium.git
synced 2025-11-21 16:14:23 +01:00
client/note color picker menu item: add support to select a custom color
This commit is contained in:
parent
01d6dee9fc
commit
5ecd8b41e5
@ -1,5 +1,6 @@
|
|||||||
import "./NoteColorPickerMenuItem.css";
|
import "./NoteColorPickerMenuItem.css";
|
||||||
import { useEffect, useState } from "preact/hooks";
|
import { useEffect, useRef, useState} from "preact/hooks";
|
||||||
|
import {ComponentChildren} from "preact";
|
||||||
import attributes from "../../services/attributes";
|
import attributes from "../../services/attributes";
|
||||||
import FNote from "../../entities/fnote";
|
import FNote from "../../entities/fnote";
|
||||||
import froca from "../../services/froca";
|
import froca from "../../services/froca";
|
||||||
@ -55,21 +56,87 @@ export default function NoteColorPickerMenuItem(props: NoteColorPickerMenuItemPr
|
|||||||
color={color}
|
color={color}
|
||||||
isSelected={(color === currentColor)}
|
isSelected={(color === currentColor)}
|
||||||
isDisabled={(note === null)}
|
isDisabled={(note === null)}
|
||||||
onClick={() => onColorCellClicked(color)} />
|
onSelect={() => onColorCellClicked(color)} />
|
||||||
))}
|
))}
|
||||||
|
|
||||||
|
<CustomColorCell color={currentColor} isSelected={false} onSelect={onColorCellClicked} />
|
||||||
</div>
|
</div>
|
||||||
}
|
}
|
||||||
|
|
||||||
interface ColorCellProps {
|
interface ColorCellProps {
|
||||||
|
children?: ComponentChildren,
|
||||||
|
className?: string;
|
||||||
color: string | null,
|
color: string | null,
|
||||||
isSelected: boolean,
|
isSelected: boolean,
|
||||||
isDisabled?: boolean,
|
isDisabled?: boolean,
|
||||||
onClick?: () => void
|
onSelect?: (color: string | null) => void
|
||||||
}
|
}
|
||||||
|
|
||||||
function ColorCell(props: ColorCellProps) {
|
function ColorCell(props: ColorCellProps) {
|
||||||
return <div class={`color-cell ${props.isSelected ? "selected" : ""} ${props.isDisabled ? "disabled-color-cell" : ""}`}
|
return <div class={`color-cell ${props.isSelected ? "selected" : ""} ${props.isDisabled ? "disabled-color-cell" : ""} ${props.className ?? ""}`}
|
||||||
style={`${(props.color !== null) ? `background-color: ${props.color}` : ""}`}
|
style={`${(props.color !== null) ? `background-color: ${props.color}` : ""}`}
|
||||||
onClick={props.onClick}>
|
onClick={() => props.onSelect?.(props.color)}>
|
||||||
|
{props.children}
|
||||||
</div>;
|
</div>;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function CustomColorCell(props: ColorCellProps) {
|
||||||
|
const colorInput = useRef<HTMLInputElement>(null);
|
||||||
|
let colorInputDebouncer: Debouncer<string | null>;
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
colorInputDebouncer = new Debouncer(500, (color) => {
|
||||||
|
props.onSelect?.(color);
|
||||||
|
});
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
colorInputDebouncer.destroy();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return <>
|
||||||
|
<ColorCell {...props}
|
||||||
|
className="custom-color-cell"
|
||||||
|
onSelect={() => {colorInput.current?.click()}}>
|
||||||
|
|
||||||
|
<input ref={colorInput}
|
||||||
|
type="color"
|
||||||
|
value={props.color ?? ""}
|
||||||
|
onChange={() => {colorInputDebouncer.updateValue(colorInput.current?.value ?? null)}}
|
||||||
|
style="width: 0; height: 0; opacity: 0" />
|
||||||
|
</ColorCell>
|
||||||
|
</>
|
||||||
|
}
|
||||||
|
|
||||||
|
type DebouncerCallback<T> = (value: T) => void;
|
||||||
|
|
||||||
|
class Debouncer<T> {
|
||||||
|
|
||||||
|
private debounceInterval: number;
|
||||||
|
private callback: DebouncerCallback<T>;
|
||||||
|
private lastValue: T | undefined;
|
||||||
|
private timeoutId: any | null = null;
|
||||||
|
|
||||||
|
constructor(debounceInterval: number, onUpdate: DebouncerCallback<T>) {
|
||||||
|
this.debounceInterval = debounceInterval;
|
||||||
|
this.callback = onUpdate;
|
||||||
|
}
|
||||||
|
|
||||||
|
updateValue(value: T) {
|
||||||
|
this.lastValue = value;
|
||||||
|
this.destroy();
|
||||||
|
this.timeoutId = setTimeout(this.reportUpdate.bind(this), this.debounceInterval);
|
||||||
|
}
|
||||||
|
|
||||||
|
destroy() {
|
||||||
|
if (this.timeoutId !== null) {
|
||||||
|
clearTimeout(this.timeoutId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private reportUpdate() {
|
||||||
|
if (this.lastValue !== undefined) {
|
||||||
|
this.callback(this.lastValue);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
Loading…
x
Reference in New Issue
Block a user