mirror of
https://github.com/zhom/donutbrowser.git
synced 2026-06-12 09:47:51 +02:00
518 lines
14 KiB
TypeScript
518 lines
14 KiB
TypeScript
"use client";
|
|
|
|
import Color from "color";
|
|
import { Slider } from "radix-ui";
|
|
import {
|
|
type ComponentProps,
|
|
createContext,
|
|
type HTMLAttributes,
|
|
memo,
|
|
useCallback,
|
|
useContext,
|
|
useEffect,
|
|
useMemo,
|
|
useRef,
|
|
useState,
|
|
} from "react";
|
|
import { LuPipette } from "react-icons/lu";
|
|
import { Button } from "@/components/ui/button";
|
|
import { Input } from "@/components/ui/input";
|
|
import {
|
|
Select,
|
|
SelectContent,
|
|
SelectItem,
|
|
SelectTrigger,
|
|
SelectValue,
|
|
} from "@/components/ui/select";
|
|
import { cn } from "@/lib/utils";
|
|
|
|
interface ColorPickerContextValue {
|
|
hue: number;
|
|
saturation: number;
|
|
lightness: number;
|
|
alpha: number;
|
|
mode: string;
|
|
setHue: (hue: number) => void;
|
|
setSaturation: (saturation: number) => void;
|
|
setLightness: (lightness: number) => void;
|
|
setAlpha: (alpha: number) => void;
|
|
setMode: (mode: string) => void;
|
|
}
|
|
|
|
const ColorPickerContext = createContext<ColorPickerContextValue | undefined>(
|
|
undefined,
|
|
);
|
|
|
|
export const useColorPicker = () => {
|
|
const context = useContext(ColorPickerContext);
|
|
|
|
if (!context) {
|
|
throw new Error("useColorPicker must be used within a ColorPickerProvider");
|
|
}
|
|
|
|
return context;
|
|
};
|
|
|
|
export type ColorPickerProps = Omit<
|
|
HTMLAttributes<HTMLDivElement>,
|
|
"onChange"
|
|
> & {
|
|
value?: Parameters<typeof Color>[0];
|
|
defaultValue?: Parameters<typeof Color>[0];
|
|
onColorChange?: (value: [number, number, number, number]) => void;
|
|
};
|
|
|
|
export const ColorPicker = ({
|
|
value,
|
|
defaultValue = "#000000",
|
|
onColorChange,
|
|
className,
|
|
children,
|
|
...props
|
|
}: ColorPickerProps) => {
|
|
const selectedColor = Color(value ?? defaultValue);
|
|
const defaultColor = Color(defaultValue);
|
|
|
|
const initialHue = Number.isFinite(selectedColor.hue())
|
|
? selectedColor.hue()
|
|
: Number.isFinite(defaultColor.hue())
|
|
? defaultColor.hue()
|
|
: 0;
|
|
const initialSaturation = Number.isFinite(selectedColor.saturationl())
|
|
? selectedColor.saturationl()
|
|
: Number.isFinite(defaultColor.saturationl())
|
|
? defaultColor.saturationl()
|
|
: 100;
|
|
const initialLightness = Number.isFinite(selectedColor.lightness())
|
|
? selectedColor.lightness()
|
|
: Number.isFinite(defaultColor.lightness())
|
|
? defaultColor.lightness()
|
|
: 50;
|
|
const initialAlpha = Number.isFinite(selectedColor.alpha())
|
|
? Math.round(selectedColor.alpha() * 100)
|
|
: Math.round(defaultColor.alpha() * 100);
|
|
|
|
const [hue, setHue] = useState(initialHue);
|
|
const [saturation, setSaturation] = useState(initialSaturation);
|
|
const [lightness, setLightness] = useState(initialLightness);
|
|
const [alpha, setAlpha] = useState(initialAlpha);
|
|
const [mode, setMode] = useState("hex");
|
|
const lastEmittedRef = useRef<string>(
|
|
`${Math.round(initialHue)}|${Math.round(initialSaturation)}|${Math.round(initialLightness)}|${Math.round(initialAlpha)}`,
|
|
);
|
|
|
|
// Update color when controlled value changes
|
|
useEffect(() => {
|
|
if (value !== undefined) {
|
|
const c = Color(value).hsl();
|
|
const nextHue = Number.isFinite(c.hue()) ? c.hue() : 0;
|
|
const nextSat = Number.isFinite(c.saturationl()) ? c.saturationl() : 0;
|
|
const nextLight = Number.isFinite(c.lightness()) ? c.lightness() : 0;
|
|
const nextAlpha = Math.round(
|
|
(Number.isFinite(c.alpha()) ? c.alpha() : 1) * 100,
|
|
);
|
|
|
|
// Update internal state unconditionally when value prop changes
|
|
setHue(nextHue);
|
|
setSaturation(nextSat);
|
|
setLightness(nextLight);
|
|
setAlpha(nextAlpha);
|
|
}
|
|
}, [value]); // Remove state values from dependency array to prevent infinite loop
|
|
|
|
// Notify parent of changes
|
|
useEffect(() => {
|
|
if (onColorChange) {
|
|
const key = `${Math.round(hue)}|${Math.round(saturation)}|${Math.round(lightness)}|${Math.round(alpha)}`;
|
|
if (key === lastEmittedRef.current) {
|
|
return;
|
|
}
|
|
lastEmittedRef.current = key;
|
|
|
|
const color = Color.hsl(hue, saturation, lightness).alpha(alpha / 100);
|
|
const rgba = color.rgb().array();
|
|
onColorChange([rgba[0], rgba[1], rgba[2], alpha / 100]);
|
|
}
|
|
}, [hue, saturation, lightness, alpha, onColorChange]);
|
|
|
|
return (
|
|
<ColorPickerContext.Provider
|
|
value={{
|
|
hue,
|
|
saturation,
|
|
lightness,
|
|
alpha,
|
|
mode,
|
|
setHue,
|
|
setSaturation,
|
|
setLightness,
|
|
setAlpha,
|
|
setMode,
|
|
}}
|
|
>
|
|
<div
|
|
className={cn("flex flex-col gap-4 size-full", className)}
|
|
{...props}
|
|
>
|
|
{children}
|
|
</div>
|
|
</ColorPickerContext.Provider>
|
|
);
|
|
};
|
|
|
|
export type ColorPickerSelectionProps = HTMLAttributes<HTMLDivElement>;
|
|
|
|
export const ColorPickerSelection = memo(
|
|
({ className, ...props }: ColorPickerSelectionProps) => {
|
|
const containerRef = useRef<HTMLDivElement>(null);
|
|
const [isDragging, setIsDragging] = useState(false);
|
|
const [positionX, setPositionX] = useState(0);
|
|
const [positionY, setPositionY] = useState(0);
|
|
const { hue, saturation, lightness, setSaturation, setLightness } =
|
|
useColorPicker();
|
|
|
|
const backgroundGradient = useMemo(() => {
|
|
return `linear-gradient(0deg, rgba(0,0,0,1), rgba(0,0,0,0)),
|
|
linear-gradient(90deg, rgba(255,255,255,1), rgba(255,255,255,0)),
|
|
hsl(${hue}, 100%, 50%)`;
|
|
}, [hue]);
|
|
|
|
// Update position indicators when saturation/lightness change externally
|
|
useEffect(() => {
|
|
if (!isDragging) {
|
|
const x = saturation / 100;
|
|
const topLightness = x < 0.01 ? 100 : 50 + 50 * (1 - x);
|
|
const y = topLightness > 0 ? 1 - lightness / topLightness : 0;
|
|
setPositionX(x);
|
|
setPositionY(Math.max(0, Math.min(1, y)));
|
|
}
|
|
}, [saturation, lightness, isDragging]);
|
|
|
|
const handlePointerMove = useCallback(
|
|
(event: PointerEvent) => {
|
|
if (!(isDragging && containerRef.current)) {
|
|
return;
|
|
}
|
|
const rect = containerRef.current.getBoundingClientRect();
|
|
const x = Math.max(
|
|
0,
|
|
Math.min(1, (event.clientX - rect.left) / rect.width),
|
|
);
|
|
const y = Math.max(
|
|
0,
|
|
Math.min(1, (event.clientY - rect.top) / rect.height),
|
|
);
|
|
setPositionX(x);
|
|
setPositionY(y);
|
|
setSaturation(x * 100);
|
|
const topLightness = x < 0.01 ? 100 : 50 + 50 * (1 - x);
|
|
const lightness = topLightness * (1 - y);
|
|
|
|
setLightness(lightness);
|
|
},
|
|
[isDragging, setSaturation, setLightness],
|
|
);
|
|
|
|
useEffect(() => {
|
|
const handlePointerUp = () => setIsDragging(false);
|
|
|
|
if (isDragging) {
|
|
window.addEventListener("pointermove", handlePointerMove);
|
|
window.addEventListener("pointerup", handlePointerUp);
|
|
}
|
|
|
|
return () => {
|
|
window.removeEventListener("pointermove", handlePointerMove);
|
|
window.removeEventListener("pointerup", handlePointerUp);
|
|
};
|
|
}, [isDragging, handlePointerMove]);
|
|
|
|
return (
|
|
<div
|
|
className={cn("relative rounded cursor-pointer size-full", className)}
|
|
onPointerDown={(e) => {
|
|
e.preventDefault();
|
|
setIsDragging(true);
|
|
handlePointerMove(e.nativeEvent);
|
|
}}
|
|
ref={containerRef}
|
|
style={{
|
|
background: backgroundGradient,
|
|
}}
|
|
{...props}
|
|
>
|
|
<div
|
|
className="absolute w-4 h-4 rounded-full border-2 border-white -translate-x-1/2 -translate-y-1/2 pointer-events-none"
|
|
style={{
|
|
left: `${positionX * 100}%`,
|
|
top: `${positionY * 100}%`,
|
|
boxShadow: "0 0 0 1px rgba(0,0,0,0.5)",
|
|
}}
|
|
/>
|
|
</div>
|
|
);
|
|
},
|
|
);
|
|
|
|
ColorPickerSelection.displayName = "ColorPickerSelection";
|
|
|
|
export type ColorPickerHueProps = ComponentProps<typeof Slider.Root>;
|
|
|
|
export const ColorPickerHue = ({
|
|
className,
|
|
...props
|
|
}: ColorPickerHueProps) => {
|
|
const { hue, setHue } = useColorPicker();
|
|
|
|
return (
|
|
<Slider.Root
|
|
className={cn("flex relative w-full h-4 touch-none", className)}
|
|
max={360}
|
|
onValueChange={([hue]) => setHue(hue)}
|
|
step={1}
|
|
value={[hue]}
|
|
{...props}
|
|
>
|
|
<Slider.Track className="relative my-0.5 h-3 w-full grow rounded-full bg-[linear-gradient(90deg,#FF0000,#FFFF00,#00FF00,#00FFFF,#0000FF,#FF00FF,#FF0000)]">
|
|
<Slider.Range className="absolute h-full" />
|
|
</Slider.Track>
|
|
<Slider.Thumb className="block w-4 h-4 rounded-full border shadow transition-colors border-primary/50 bg-background focus-visible:outline-none focus-visible:ring-1 focus-visible:ring-ring disabled:pointer-events-none disabled:opacity-50" />
|
|
</Slider.Root>
|
|
);
|
|
};
|
|
|
|
export type ColorPickerAlphaProps = ComponentProps<typeof Slider.Root>;
|
|
|
|
export const ColorPickerAlpha = ({
|
|
className,
|
|
...props
|
|
}: ColorPickerAlphaProps) => {
|
|
const { alpha, setAlpha } = useColorPicker();
|
|
|
|
return (
|
|
<Slider.Root
|
|
className={cn("flex relative w-full h-4 touch-none", className)}
|
|
max={100}
|
|
onValueChange={([alpha]) => setAlpha(alpha)}
|
|
step={1}
|
|
value={[alpha]}
|
|
{...props}
|
|
>
|
|
<Slider.Track
|
|
className="relative my-0.5 h-3 w-full grow rounded-full"
|
|
style={{
|
|
background:
|
|
'url("data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAAMUlEQVQ4T2NkYGAQYcAP3uCTZhw1gGGYhAGBZIA/nYDCgBDAm9BGDWAAJyRCgLaBCAAgXwixzAS0pgAAAABJRU5ErkJggg==") left center',
|
|
}}
|
|
>
|
|
<div className="absolute inset-0 bg-gradient-to-r from-transparent rounded-full to-black/50" />
|
|
<Slider.Range className="absolute h-full bg-transparent rounded-full" />
|
|
</Slider.Track>
|
|
<Slider.Thumb className="block w-4 h-4 rounded-full border shadow transition-colors border-primary/50 bg-background focus-visible:outline-none focus-visible:ring-1 focus-visible:ring-ring disabled:pointer-events-none disabled:opacity-50" />
|
|
</Slider.Root>
|
|
);
|
|
};
|
|
|
|
export type ColorPickerEyeDropperProps = ComponentProps<typeof Button>;
|
|
|
|
export const ColorPickerEyeDropper = ({
|
|
className,
|
|
...props
|
|
}: ColorPickerEyeDropperProps) => {
|
|
const { setHue, setSaturation, setLightness, setAlpha } = useColorPicker();
|
|
|
|
const handleEyeDropper = async () => {
|
|
try {
|
|
// @ts-expect-error - EyeDropper API is experimental
|
|
const eyeDropper = new EyeDropper();
|
|
const result = await eyeDropper.open();
|
|
const color = Color(result.sRGBHex);
|
|
const [h, s, l] = color.hsl().array();
|
|
|
|
setHue(h);
|
|
setSaturation(s);
|
|
setLightness(l);
|
|
setAlpha(100);
|
|
} catch (error) {
|
|
console.error("EyeDropper failed:", error);
|
|
}
|
|
};
|
|
|
|
return (
|
|
<Button
|
|
className={cn("shrink-0 text-muted-foreground", className)}
|
|
onClick={handleEyeDropper}
|
|
size="icon"
|
|
variant="outline"
|
|
type="button"
|
|
{...props}
|
|
>
|
|
<LuPipette size={16} />
|
|
</Button>
|
|
);
|
|
};
|
|
|
|
export type ColorPickerOutputProps = ComponentProps<typeof SelectTrigger>;
|
|
|
|
const formats = ["hex", "rgb", "css", "hsl"];
|
|
|
|
export const ColorPickerOutput = ({
|
|
className,
|
|
...props
|
|
}: ColorPickerOutputProps) => {
|
|
const { mode, setMode } = useColorPicker();
|
|
|
|
return (
|
|
<Select onValueChange={setMode} value={mode}>
|
|
<SelectTrigger className="w-20 h-8 text-xs shrink-0" {...props}>
|
|
<SelectValue placeholder="Mode" />
|
|
</SelectTrigger>
|
|
<SelectContent>
|
|
{formats.map((format) => (
|
|
<SelectItem className="text-xs" key={format} value={format}>
|
|
{format.toUpperCase()}
|
|
</SelectItem>
|
|
))}
|
|
</SelectContent>
|
|
</Select>
|
|
);
|
|
};
|
|
|
|
type PercentageInputProps = ComponentProps<typeof Input>;
|
|
|
|
const PercentageInput = ({ className, ...props }: PercentageInputProps) => {
|
|
return (
|
|
<div className="relative">
|
|
<Input
|
|
readOnly
|
|
type="text"
|
|
{...props}
|
|
className={cn(
|
|
"h-8 w-[3.25rem] rounded-l-none bg-secondary px-2 text-xs shadow-none",
|
|
className,
|
|
)}
|
|
/>
|
|
<span className="absolute right-2 top-1/2 text-xs -translate-y-1/2 text-muted-foreground">
|
|
%
|
|
</span>
|
|
</div>
|
|
);
|
|
};
|
|
|
|
export type ColorPickerFormatProps = HTMLAttributes<HTMLDivElement>;
|
|
|
|
export const ColorPickerFormat = ({
|
|
className,
|
|
...props
|
|
}: ColorPickerFormatProps) => {
|
|
const { hue, saturation, lightness, alpha, mode } = useColorPicker();
|
|
const color = Color.hsl(hue, saturation, lightness, alpha / 100);
|
|
|
|
if (mode === "hex") {
|
|
const hex = color.hex();
|
|
|
|
return (
|
|
<div
|
|
className={cn(
|
|
"flex relative items-center -space-x-px w-full rounded-md shadow-sm",
|
|
className,
|
|
)}
|
|
{...props}
|
|
>
|
|
<Input
|
|
className="px-2 h-8 text-xs rounded-r-none shadow-none bg-secondary"
|
|
readOnly
|
|
type="text"
|
|
value={hex}
|
|
/>
|
|
<PercentageInput value={alpha} />
|
|
</div>
|
|
);
|
|
}
|
|
|
|
if (mode === "rgb") {
|
|
const rgb = color
|
|
.rgb()
|
|
.array()
|
|
.map((value) => Math.round(value));
|
|
|
|
return (
|
|
<div
|
|
className={cn(
|
|
"flex items-center -space-x-px rounded-md shadow-sm",
|
|
className,
|
|
)}
|
|
{...props}
|
|
>
|
|
{rgb.map((value, index) => (
|
|
<Input
|
|
className={cn(
|
|
"h-8 rounded-r-none bg-secondary px-2 text-xs shadow-none",
|
|
index && "rounded-l-none",
|
|
className,
|
|
)}
|
|
key={`rgb-${value.toString()}`}
|
|
readOnly
|
|
type="text"
|
|
value={value}
|
|
/>
|
|
))}
|
|
<PercentageInput value={alpha} />
|
|
</div>
|
|
);
|
|
}
|
|
|
|
if (mode === "css") {
|
|
const rgb = color
|
|
.rgb()
|
|
.array()
|
|
.map((value) => Math.round(value));
|
|
|
|
return (
|
|
<div className={cn("w-full rounded-md shadow-sm", className)} {...props}>
|
|
<Input
|
|
className="px-2 w-full h-8 text-xs shadow-none bg-secondary"
|
|
readOnly
|
|
type="text"
|
|
value={`rgba(${rgb.join(", ")}, ${alpha}%)`}
|
|
{...props}
|
|
/>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
if (mode === "hsl") {
|
|
const hsl = color
|
|
.hsl()
|
|
.array()
|
|
.map((value) => Math.round(value));
|
|
|
|
return (
|
|
<div
|
|
className={cn(
|
|
"flex items-center -space-x-px rounded-md shadow-sm",
|
|
className,
|
|
)}
|
|
{...props}
|
|
>
|
|
{hsl.map((value, index) => (
|
|
<Input
|
|
className={cn(
|
|
"h-8 rounded-r-none bg-secondary px-2 text-xs shadow-none",
|
|
index && "rounded-l-none",
|
|
className,
|
|
)}
|
|
key={`hsl-${value.toString()}`}
|
|
readOnly
|
|
type="text"
|
|
value={value}
|
|
/>
|
|
))}
|
|
<PercentageInput value={alpha} />
|
|
</div>
|
|
);
|
|
}
|
|
|
|
return null;
|
|
};
|