mirror of
https://github.com/wassname/Open-Assistant.git
synced 2026-06-28 16:20:34 +08:00
177 lines
5.0 KiB
TypeScript
177 lines
5.0 KiB
TypeScript
import { useRef, useState } from "react";
|
|
import { TaskControls } from "src/components/Survey/TaskControls";
|
|
import { CreateTask } from "src/components/Tasks/CreateTask";
|
|
import { EvaluateTask } from "src/components/Tasks/EvaluateTask";
|
|
import { LabelTask } from "src/components/Tasks/LabelTask";
|
|
import { TaskCategory, TaskInfo, TaskInfos } from "src/components/Tasks/TaskTypes";
|
|
import { UnchangedWarning } from "src/components/Tasks/UnchangedWarning";
|
|
import { post } from "src/lib/api";
|
|
import { TaskContent } from "src/types/Task";
|
|
import { TaskReplyState } from "src/types/TaskReplyState";
|
|
import useSWRMutation from "swr/mutation";
|
|
|
|
export type TaskStatus = "NOT_SUBMITTABLE" | "DEFAULT" | "VALID" | "REVIEW" | "SUBMITTED";
|
|
|
|
export interface TaskSurveyProps<T> {
|
|
// we need a task type
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
task: any;
|
|
taskType: TaskInfo;
|
|
isEditable: boolean;
|
|
isDisabled?: boolean;
|
|
onReplyChanged: (state: TaskReplyState<T>) => void;
|
|
}
|
|
|
|
export const Task = ({ frontendId, task, trigger, mutate }) => {
|
|
const [taskStatus, setTaskStatus] = useState<TaskStatus>("NOT_SUBMITTABLE");
|
|
const replyContent = useRef<TaskContent>(null);
|
|
const [showUnchangedWarning, setShowUnchangedWarning] = useState(false);
|
|
|
|
const rootEl = useRef<HTMLDivElement>(null);
|
|
|
|
const taskType = TaskInfos.find((taskType) => taskType.type === task.type && taskType.mode === task.mode);
|
|
|
|
const { trigger: sendRejection } = useSWRMutation("/api/reject_task", post, {
|
|
onSuccess: async () => {
|
|
mutate();
|
|
},
|
|
});
|
|
|
|
const rejectTask = (reason: string) => {
|
|
sendRejection({
|
|
id: frontendId,
|
|
reason,
|
|
});
|
|
};
|
|
|
|
const onReplyChanged = useRef((state: TaskReplyState<TaskContent>) => {
|
|
if (taskStatus === "SUBMITTED") return;
|
|
|
|
replyContent.current = state?.content;
|
|
if (state === null) {
|
|
if (taskStatus !== "NOT_SUBMITTABLE") setTaskStatus("NOT_SUBMITTABLE");
|
|
} else if (state.state === "DEFAULT") {
|
|
if (taskStatus !== "DEFAULT") setTaskStatus("DEFAULT");
|
|
} else if (state.state === "VALID") {
|
|
if (taskStatus !== "VALID") setTaskStatus("VALID");
|
|
} else if (state.state === "INVALID") {
|
|
setTaskStatus("NOT_SUBMITTABLE");
|
|
}
|
|
}).current;
|
|
|
|
const reviewResponse = () => {
|
|
switch (taskStatus) {
|
|
case "DEFAULT":
|
|
setShowUnchangedWarning(true);
|
|
break;
|
|
case "VALID":
|
|
setTaskStatus("REVIEW");
|
|
break;
|
|
default:
|
|
return;
|
|
}
|
|
};
|
|
|
|
const editResponse = () => {
|
|
switch (taskStatus) {
|
|
case "REVIEW":
|
|
setTaskStatus("VALID");
|
|
break;
|
|
default:
|
|
return;
|
|
}
|
|
};
|
|
|
|
const submitResponse = () => {
|
|
switch (taskStatus) {
|
|
case "REVIEW": {
|
|
trigger({
|
|
id: frontendId,
|
|
update_type: taskType.update_type,
|
|
content: replyContent.current,
|
|
});
|
|
setTaskStatus("SUBMITTED");
|
|
scrollToTop(rootEl.current);
|
|
break;
|
|
}
|
|
default:
|
|
return;
|
|
}
|
|
};
|
|
|
|
const edit_mode = taskStatus === "NOT_SUBMITTABLE" || taskStatus === "DEFAULT" || taskStatus === "VALID";
|
|
const submitted = taskStatus === "SUBMITTED";
|
|
|
|
function taskTypeComponent() {
|
|
switch (taskType.category) {
|
|
case TaskCategory.Create:
|
|
return (
|
|
<CreateTask
|
|
key={task.id}
|
|
task={task}
|
|
taskType={taskType}
|
|
isEditable={edit_mode}
|
|
isDisabled={submitted}
|
|
onReplyChanged={onReplyChanged}
|
|
/>
|
|
);
|
|
case TaskCategory.Evaluate:
|
|
return (
|
|
<EvaluateTask
|
|
key={task.id}
|
|
task={task}
|
|
taskType={taskType}
|
|
isEditable={edit_mode}
|
|
isDisabled={submitted}
|
|
onReplyChanged={onReplyChanged}
|
|
/>
|
|
);
|
|
case TaskCategory.Label:
|
|
return (
|
|
<LabelTask
|
|
key={task.id}
|
|
task={task}
|
|
taskType={taskType}
|
|
isEditable={edit_mode}
|
|
isDisabled={submitted}
|
|
onReplyChanged={onReplyChanged}
|
|
/>
|
|
);
|
|
}
|
|
}
|
|
|
|
return (
|
|
<div ref={rootEl}>
|
|
{taskTypeComponent()}
|
|
<TaskControls
|
|
task={task}
|
|
taskStatus={taskStatus}
|
|
onEdit={editResponse}
|
|
onReview={reviewResponse}
|
|
onSubmit={submitResponse}
|
|
onSkip={rejectTask}
|
|
/>
|
|
<UnchangedWarning
|
|
show={showUnchangedWarning}
|
|
title={taskType.unchanged_title || "No changes"}
|
|
message={taskType.unchanged_message || "Are you sure you would like to continue?"}
|
|
continueButtonText={"Continue anyway"}
|
|
onClose={() => setShowUnchangedWarning(false)}
|
|
onContinueAnyway={() => {
|
|
if (taskStatus === "DEFAULT") {
|
|
setTaskStatus("REVIEW");
|
|
setShowUnchangedWarning(false);
|
|
}
|
|
}}
|
|
/>
|
|
</div>
|
|
);
|
|
};
|
|
|
|
const scrollToTop = (element: HTMLElement) => {
|
|
while (element) {
|
|
element.scrollTop = 0;
|
|
element = element.parentElement;
|
|
}
|
|
};
|