254 lines
8.0 KiB
TypeScript
254 lines
8.0 KiB
TypeScript
import { useEffect, useState, type KeyboardEvent as ReactKeyboardEvent } from "react";
|
|
|
|
import {
|
|
getProjectDialogues,
|
|
type ProjectDialogue,
|
|
type ProjectDialogueLibrary
|
|
} from "../dialogues/project-dialogues";
|
|
|
|
interface ProjectDialoguesPanelProps {
|
|
dialogues: ProjectDialogueLibrary;
|
|
selectedDialogueId: string | null;
|
|
onSelectDialogue(dialogueId: string | null): void;
|
|
onAddDialogue(): void;
|
|
onDeleteDialogue(dialogueId: string): void;
|
|
onSetDialogueTitle(dialogueId: string, title: string): void;
|
|
onAddDialogueLine(dialogueId: string): void;
|
|
onDeleteDialogueLine(dialogueId: string, lineId: string): void;
|
|
onSetDialogueLineSpeaker(
|
|
dialogueId: string,
|
|
lineId: string,
|
|
speakerName: string | null
|
|
): void;
|
|
onSetDialogueLineText(dialogueId: string, lineId: string, text: string): void;
|
|
}
|
|
|
|
function commitOnEnter(
|
|
event: ReactKeyboardEvent<HTMLInputElement>,
|
|
commit: () => void
|
|
) {
|
|
if (event.key !== "Enter") {
|
|
return;
|
|
}
|
|
|
|
event.currentTarget.blur();
|
|
commit();
|
|
}
|
|
|
|
export function ProjectDialoguesPanel({
|
|
dialogues,
|
|
selectedDialogueId,
|
|
onSelectDialogue,
|
|
onAddDialogue,
|
|
onDeleteDialogue,
|
|
onSetDialogueTitle,
|
|
onAddDialogueLine,
|
|
onDeleteDialogueLine,
|
|
onSetDialogueLineSpeaker,
|
|
onSetDialogueLineText
|
|
}: ProjectDialoguesPanelProps) {
|
|
const dialogueList = getProjectDialogues(dialogues);
|
|
const selectedDialogue =
|
|
selectedDialogueId === null ? null : dialogues.dialogues[selectedDialogueId] ?? null;
|
|
const [titleDraft, setTitleDraft] = useState(selectedDialogue?.title ?? "");
|
|
const [lineDrafts, setLineDrafts] = useState<
|
|
Record<string, { speakerName: string; text: string }>
|
|
>({});
|
|
|
|
useEffect(() => {
|
|
setTitleDraft(selectedDialogue?.title ?? "");
|
|
setLineDrafts(
|
|
selectedDialogue === null
|
|
? {}
|
|
: Object.fromEntries(
|
|
selectedDialogue.lines.map((line) => [
|
|
line.id,
|
|
{
|
|
speakerName: line.speakerName ?? "",
|
|
text: line.text
|
|
}
|
|
])
|
|
)
|
|
);
|
|
}, [selectedDialogueId, selectedDialogue]);
|
|
|
|
const commitTitle = () => {
|
|
if (selectedDialogue === null) {
|
|
return;
|
|
}
|
|
|
|
onSetDialogueTitle(selectedDialogue.id, titleDraft);
|
|
};
|
|
|
|
const getLineDraft = (dialogue: ProjectDialogue, lineId: string) =>
|
|
lineDrafts[lineId] ??
|
|
(() => {
|
|
const line = dialogue.lines.find((candidate) => candidate.id === lineId);
|
|
return {
|
|
speakerName: line?.speakerName ?? "",
|
|
text: line?.text ?? ""
|
|
};
|
|
})();
|
|
|
|
return (
|
|
<div className="form-section">
|
|
<div className="label">Dialogues</div>
|
|
{dialogueList.length === 0 ? (
|
|
<div className="outliner-empty">
|
|
No project dialogues authored yet.
|
|
</div>
|
|
) : (
|
|
<div className="outliner-list">
|
|
{dialogueList.map((dialogue) => (
|
|
<div
|
|
key={dialogue.id}
|
|
className={`outliner-item outliner-item--compact ${
|
|
selectedDialogue?.id === dialogue.id
|
|
? "outliner-item--selected"
|
|
: ""
|
|
}`.trim()}
|
|
>
|
|
<div className="outliner-item__row">
|
|
<button
|
|
className="outliner-item__select"
|
|
type="button"
|
|
onClick={() => onSelectDialogue(dialogue.id)}
|
|
>
|
|
<span className="outliner-item__title">{dialogue.title}</span>
|
|
<span className="outliner-item__meta">
|
|
{dialogue.lines.length} line
|
|
{dialogue.lines.length === 1 ? "" : "s"}
|
|
</span>
|
|
</button>
|
|
<button
|
|
className="outliner-item__delete"
|
|
type="button"
|
|
aria-label={`Delete ${dialogue.title}`}
|
|
onClick={() => onDeleteDialogue(dialogue.id)}
|
|
>
|
|
x
|
|
</button>
|
|
</div>
|
|
</div>
|
|
))}
|
|
</div>
|
|
)}
|
|
|
|
<div className="inline-actions">
|
|
<button className="toolbar__button" type="button" onClick={onAddDialogue}>
|
|
Add Dialogue
|
|
</button>
|
|
</div>
|
|
|
|
{selectedDialogue === null ? (
|
|
<div className="outliner-empty">
|
|
Select a dialogue to edit its title and lines.
|
|
</div>
|
|
) : (
|
|
<div className="form-section">
|
|
<label className="form-field">
|
|
<span className="label">Title</span>
|
|
<input
|
|
className="text-input"
|
|
type="text"
|
|
value={titleDraft}
|
|
onChange={(event) => {
|
|
const nextValue = event.currentTarget.value;
|
|
setTitleDraft(nextValue);
|
|
}}
|
|
onBlur={commitTitle}
|
|
onKeyDown={(event) => commitOnEnter(event, commitTitle)}
|
|
/>
|
|
</label>
|
|
|
|
<div className="label">Lines</div>
|
|
<div className="outliner-list">
|
|
{selectedDialogue.lines.map((line, index) => {
|
|
const draft = getLineDraft(selectedDialogue, line.id);
|
|
|
|
return (
|
|
<div key={line.id} className="outliner-item">
|
|
<div className="outliner-item__row">
|
|
<div className="outliner-item__meta">{`Line ${index + 1}`}</div>
|
|
<button
|
|
className="outliner-item__delete"
|
|
type="button"
|
|
aria-label={`Delete line ${index + 1}`}
|
|
onClick={() => onDeleteDialogueLine(selectedDialogue.id, line.id)}
|
|
>
|
|
x
|
|
</button>
|
|
</div>
|
|
<label className="form-field">
|
|
<span className="label">Speaker</span>
|
|
<input
|
|
className="text-input"
|
|
type="text"
|
|
placeholder="Optional"
|
|
value={draft.speakerName}
|
|
onChange={(event) => {
|
|
const nextSpeakerName = event.currentTarget.value;
|
|
|
|
setLineDrafts((current) => ({
|
|
...current,
|
|
[line.id]: {
|
|
...draft,
|
|
speakerName: nextSpeakerName
|
|
}
|
|
}));
|
|
}}
|
|
onBlur={() =>
|
|
onSetDialogueLineSpeaker(
|
|
selectedDialogue.id,
|
|
line.id,
|
|
draft.speakerName
|
|
)
|
|
}
|
|
/>
|
|
</label>
|
|
<label className="form-field">
|
|
<span className="label">Text</span>
|
|
<textarea
|
|
className="text-input"
|
|
rows={3}
|
|
value={draft.text}
|
|
onChange={(event) => {
|
|
const nextText = event.currentTarget.value;
|
|
|
|
setLineDrafts((current) => ({
|
|
...current,
|
|
[line.id]: {
|
|
...draft,
|
|
text: nextText
|
|
}
|
|
}));
|
|
}}
|
|
onBlur={() =>
|
|
onSetDialogueLineText(
|
|
selectedDialogue.id,
|
|
line.id,
|
|
draft.text
|
|
)
|
|
}
|
|
/>
|
|
</label>
|
|
</div>
|
|
);
|
|
})}
|
|
</div>
|
|
|
|
<div className="inline-actions">
|
|
<button
|
|
className="toolbar__button"
|
|
type="button"
|
|
onClick={() => onAddDialogueLine(selectedDialogue.id)}
|
|
>
|
|
Add Line
|
|
</button>
|
|
</div>
|
|
</div>
|
|
)}
|
|
</div>
|
|
);
|
|
}
|