fix: note component in json-editor getting out of focus when typing and revert theme switch animation (#706)

* fix: note component in json-editor getting out of focus when typing. fixes JSON-Generator Notes Field #687

* revert: theme transition
This commit is contained in:
Bram Suurd 2024-12-06 15:12:35 -05:00 committed by GitHub
parent e2aa5fe627
commit c46f15cdc1
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
2 changed files with 54 additions and 95 deletions

View File

@ -12,7 +12,7 @@ import { cn } from "@/lib/utils";
import { PlusCircle, Trash2 } from "lucide-react"; import { PlusCircle, Trash2 } from "lucide-react";
import { z } from "zod"; import { z } from "zod";
import { ScriptSchema, type Script } from "../_schemas/schemas"; import { ScriptSchema, type Script } from "../_schemas/schemas";
import { memo, useCallback } from "react"; import { memo, useCallback, useRef } from "react";
type NoteProps = { type NoteProps = {
script: Script; script: Script;
@ -27,6 +27,8 @@ function Note({
setIsValid, setIsValid,
setZodErrors, setZodErrors,
}: NoteProps) { }: NoteProps) {
const inputRefs = useRef<(HTMLInputElement | null)[]>([]);
const addNote = useCallback(() => { const addNote = useCallback(() => {
setScript({ setScript({
...script, ...script,
@ -49,6 +51,12 @@ function Note({
setIsValid(result.success); setIsValid(result.success);
setZodErrors(result.success ? null : result.error); setZodErrors(result.success ? null : result.error);
setScript(updated); setScript(updated);
// Restore focus after state update
if (key === "text") {
setTimeout(() => {
inputRefs.current[index]?.focus();
}, 0);
}
}, [script, setScript, setIsValid, setZodErrors]); }, [script, setScript, setIsValid, setZodErrors]);
const removeNote = useCallback((index: number) => { const removeNote = useCallback((index: number) => {
@ -58,46 +66,51 @@ function Note({
}); });
}, [script, setScript]); }, [script, setScript]);
const NoteItem = memo(({ note, index }: { note: Script["notes"][number], index: number }) => ( const NoteItem = memo(
<div className="space-y-2 border p-4 rounded"> ({ note, index }: { note: Script["notes"][number]; index: number }) => (
<Input <div className="space-y-2 border p-4 rounded">
placeholder="Note Text" <Input
value={note.text} placeholder="Note Text"
onChange={(e) => updateNote(index, "text", e.target.value)} value={note.text}
/> onChange={(e) => updateNote(index, "text", e.target.value)}
<Select ref={(el) => {
value={note.type} inputRefs.current[index] = el;
onValueChange={(value) => updateNote(index, "type", value)} }}
> />
<SelectTrigger className="flex-1"> <Select
<SelectValue placeholder="Type" /> value={note.type}
</SelectTrigger> onValueChange={(value) => updateNote(index, "type", value)}
<SelectContent> >
{Object.keys(AlertColors).map((type) => ( <SelectTrigger className="flex-1">
<SelectItem key={type} value={type}> <SelectValue placeholder="Type" />
<span className="flex items-center gap-2"> </SelectTrigger>
{type.charAt(0).toUpperCase() + type.slice(1)}{" "} <SelectContent>
<div {Object.keys(AlertColors).map((type) => (
className={cn( <SelectItem key={type} value={type}>
"size-4 rounded-full border", <span className="flex items-center gap-2">
AlertColors[type as keyof typeof AlertColors], {type.charAt(0).toUpperCase() + type.slice(1)}{" "}
)} <div
/> className={cn(
</span> "size-4 rounded-full border",
</SelectItem> AlertColors[type as keyof typeof AlertColors],
))} )}
</SelectContent> />
</Select> </span>
<Button </SelectItem>
size="sm" ))}
variant="destructive" </SelectContent>
type="button" </Select>
onClick={() => removeNote(index)} <Button
> size="sm"
<Trash2 className="mr-2 h-4 w-4" /> Remove Note variant="destructive"
</Button> type="button"
</div> onClick={() => removeNote(index)}
)); >
<Trash2 className="mr-2 h-4 w-4" /> Remove Note
</Button>
</div>
),
);
NoteItem.displayName = 'NoteItem'; NoteItem.displayName = 'NoteItem';

View File

@ -30,24 +30,6 @@
--chart-3: 197 37% 24%; --chart-3: 197 37% 24%;
--chart-4: 43 74% 66%; --chart-4: 43 74% 66%;
--chart-5: 27 87% 67%; --chart-5: 27 87% 67%;
--expo-out: linear(
0 0%,
0.1684 2.66%,
0.3165 5.49%,
0.446 8.52%,
0.5581 11.78%,
0.6535 15.29%,
0.7341 19.11%,
0.8011 23.3%,
0.8557 27.93%,
0.8962 32.68%,
0.9283 38.01%,
0.9529 44.08%,
0.9711 51.14%,
0.9833 59.06%,
0.9915 68.74%,
1 100%
);
} }
::selection { ::selection {
@ -81,42 +63,6 @@
--chart-4: 280 65% 60%; --chart-4: 280 65% 60%;
--chart-5: 340 75% 55%; --chart-5: 340 75% 55%;
} }
::view-transition-group(root) {
animation-duration: 0.7bun s;
animation-timing-function: var(--expo-out);
}
::view-transition-new(root) {
animation-name: reveal-light;
}
::view-transition-old(root),
.dark::view-transition-old(root) {
animation: none;
z-index: -1;
}
.dark::view-transition-new(root) {
animation-name: reveal-dark;
}
@keyframes reveal-dark {
from {
clip-path: polygon(50% -71%, -50% 71%, -50% 71%, 50% -71%);
}
to {
clip-path: polygon(50% -71%, -50% 71%, 50% 171%, 171% 50%);
}
}
@keyframes reveal-light {
from {
clip-path: polygon(171% 50%, 50% 171%, 50% 171%, 171% 50%);
}
to {
clip-path: polygon(171% 50%, 50% 171%, -50% 71%, 50% -71%);
}
}
} }
@layer base { @layer base {