2024-07-19 00:21:33 +02:00
|
|
|
// Copyright 2024, Command Line Inc.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
2024-07-30 20:44:19 +02:00
|
|
|
import { Modal } from "@/app/modals/modal";
|
2024-07-19 00:21:33 +02:00
|
|
|
import { Markdown } from "@/element/markdown";
|
2024-07-30 20:44:19 +02:00
|
|
|
import { modalsModel } from "@/store/modalmodel";
|
2024-07-19 00:21:33 +02:00
|
|
|
import * as keyutil from "@/util/keyutil";
|
2024-12-06 03:09:54 +01:00
|
|
|
import { fireAndForget } from "@/util/util";
|
2024-09-06 00:35:57 +02:00
|
|
|
import { useCallback, useEffect, useMemo, useRef, useState } from "react";
|
2024-12-06 03:09:54 +01:00
|
|
|
import { UserInputService } from "../store/services";
|
2024-11-22 01:05:04 +01:00
|
|
|
import "./userinputmodal.scss";
|
2024-07-19 00:21:33 +02:00
|
|
|
|
2024-07-30 20:44:19 +02:00
|
|
|
const UserInputModal = (userInputRequest: UserInputRequest) => {
|
2024-09-06 00:35:57 +02:00
|
|
|
const [responseText, setResponseText] = useState("");
|
|
|
|
const [countdown, setCountdown] = useState(Math.floor(userInputRequest.timeoutms / 1000));
|
2024-10-11 00:50:46 +02:00
|
|
|
const checkboxRef = useRef<HTMLInputElement>();
|
2024-07-19 00:21:33 +02:00
|
|
|
|
2024-11-28 01:52:00 +01:00
|
|
|
const handleSendErrResponse = useCallback(() => {
|
2024-12-06 03:09:54 +01:00
|
|
|
fireAndForget(() =>
|
|
|
|
UserInputService.SendUserInputResponse({
|
|
|
|
type: "userinputresp",
|
|
|
|
requestid: userInputRequest.requestid,
|
|
|
|
errormsg: "Canceled by the user",
|
|
|
|
})
|
|
|
|
);
|
2024-07-30 20:44:19 +02:00
|
|
|
modalsModel.popModal();
|
2024-07-19 00:21:33 +02:00
|
|
|
}, [responseText, userInputRequest]);
|
|
|
|
|
2024-09-06 00:35:57 +02:00
|
|
|
const handleSendText = useCallback(() => {
|
2024-12-06 03:09:54 +01:00
|
|
|
fireAndForget(() =>
|
|
|
|
UserInputService.SendUserInputResponse({
|
|
|
|
type: "userinputresp",
|
|
|
|
requestid: userInputRequest.requestid,
|
|
|
|
text: responseText,
|
|
|
|
checkboxstat: checkboxRef?.current?.checked ?? false,
|
|
|
|
})
|
|
|
|
);
|
2024-07-30 20:44:19 +02:00
|
|
|
modalsModel.popModal();
|
2024-07-19 00:21:33 +02:00
|
|
|
}, [responseText, userInputRequest]);
|
2024-11-28 01:52:00 +01:00
|
|
|
console.log("bar");
|
2024-07-19 00:21:33 +02:00
|
|
|
|
2024-11-28 01:52:00 +01:00
|
|
|
const handleSendConfirm = useCallback(
|
|
|
|
(response: boolean) => {
|
2024-12-06 03:09:54 +01:00
|
|
|
fireAndForget(() =>
|
|
|
|
UserInputService.SendUserInputResponse({
|
|
|
|
type: "userinputresp",
|
|
|
|
requestid: userInputRequest.requestid,
|
|
|
|
confirm: response,
|
|
|
|
checkboxstat: checkboxRef?.current?.checked ?? false,
|
|
|
|
})
|
|
|
|
);
|
2024-11-28 01:52:00 +01:00
|
|
|
modalsModel.popModal();
|
|
|
|
},
|
|
|
|
[userInputRequest]
|
|
|
|
);
|
2024-07-19 00:21:33 +02:00
|
|
|
|
2024-09-06 00:35:57 +02:00
|
|
|
const handleSubmit = useCallback(() => {
|
2024-07-19 00:21:33 +02:00
|
|
|
switch (userInputRequest.responsetype) {
|
|
|
|
case "text":
|
|
|
|
handleSendText();
|
|
|
|
break;
|
|
|
|
case "confirm":
|
2024-11-28 01:52:00 +01:00
|
|
|
handleSendConfirm(true);
|
2024-07-19 00:21:33 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}, [handleSendConfirm, handleSendText, userInputRequest.responsetype]);
|
2024-11-28 01:52:00 +01:00
|
|
|
console.log("baz");
|
2024-07-19 00:21:33 +02:00
|
|
|
|
2024-09-06 00:35:57 +02:00
|
|
|
const handleKeyDown = useCallback(
|
2024-07-19 00:21:33 +02:00
|
|
|
(waveEvent: WaveKeyboardEvent): boolean => {
|
|
|
|
if (keyutil.checkKeyPressed(waveEvent, "Escape")) {
|
2024-11-28 01:52:00 +01:00
|
|
|
handleSendErrResponse();
|
2024-07-19 00:21:33 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (keyutil.checkKeyPressed(waveEvent, "Enter")) {
|
|
|
|
handleSubmit();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
},
|
2024-11-28 01:52:00 +01:00
|
|
|
[handleSendErrResponse, handleSubmit]
|
2024-07-19 00:21:33 +02:00
|
|
|
);
|
|
|
|
|
2024-09-06 00:35:57 +02:00
|
|
|
const queryText = useMemo(() => {
|
2024-07-19 00:21:33 +02:00
|
|
|
if (userInputRequest.markdown) {
|
|
|
|
return <Markdown text={userInputRequest.querytext} className="userinput-markdown" />;
|
|
|
|
}
|
|
|
|
return <span className="userinput-text">{userInputRequest.querytext}</span>;
|
|
|
|
}, [userInputRequest.markdown, userInputRequest.querytext]);
|
2024-11-28 01:52:00 +01:00
|
|
|
console.log("foobarbaz");
|
2024-07-19 00:21:33 +02:00
|
|
|
|
2024-09-06 00:35:57 +02:00
|
|
|
const inputBox = useMemo(() => {
|
2024-07-19 00:21:33 +02:00
|
|
|
if (userInputRequest.responsetype === "confirm") {
|
|
|
|
return <></>;
|
|
|
|
}
|
|
|
|
return (
|
|
|
|
<input
|
|
|
|
type={userInputRequest.publictext ? "text" : "password"}
|
|
|
|
onChange={(e) => setResponseText(e.target.value)}
|
|
|
|
value={responseText}
|
|
|
|
maxLength={400}
|
|
|
|
className="userinput-inputbox"
|
|
|
|
autoFocus={true}
|
|
|
|
onKeyDown={(e) => keyutil.keydownWrapper(handleKeyDown)(e)}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
}, [userInputRequest.responsetype, userInputRequest.publictext, responseText, handleKeyDown, setResponseText]);
|
2024-11-28 01:52:00 +01:00
|
|
|
console.log("mem1");
|
2024-07-19 00:21:33 +02:00
|
|
|
|
2024-10-11 00:50:46 +02:00
|
|
|
const optionalCheckbox = useMemo(() => {
|
|
|
|
if (userInputRequest.checkboxmsg == "") {
|
|
|
|
return <></>;
|
|
|
|
}
|
|
|
|
return (
|
|
|
|
<div className="userinput-checkbox-container">
|
2024-11-28 01:52:00 +01:00
|
|
|
<div className="userinput-checkbox-row">
|
|
|
|
<input
|
|
|
|
type="checkbox"
|
|
|
|
id={`uicheckbox-${userInputRequest.requestid}`}
|
|
|
|
className="userinput-checkbox"
|
|
|
|
ref={checkboxRef}
|
|
|
|
/>
|
|
|
|
<label htmlFor={`uicheckbox-${userInputRequest.requestid}}`}>{userInputRequest.checkboxmsg}</label>
|
|
|
|
</div>
|
2024-10-11 00:50:46 +02:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}, []);
|
2024-11-28 01:52:00 +01:00
|
|
|
console.log("mem2");
|
2024-10-11 00:50:46 +02:00
|
|
|
|
2024-09-06 00:35:57 +02:00
|
|
|
useEffect(() => {
|
2024-07-19 00:21:33 +02:00
|
|
|
let timeout: ReturnType<typeof setTimeout>;
|
2024-09-06 13:15:42 +02:00
|
|
|
if (countdown <= 0) {
|
2024-07-19 00:21:33 +02:00
|
|
|
timeout = setTimeout(() => {
|
2024-11-28 01:52:00 +01:00
|
|
|
handleSendErrResponse();
|
2024-07-19 00:21:33 +02:00
|
|
|
}, 300);
|
|
|
|
} else {
|
|
|
|
timeout = setTimeout(() => {
|
|
|
|
setCountdown(countdown - 1);
|
|
|
|
}, 1000);
|
|
|
|
}
|
|
|
|
return () => clearTimeout(timeout);
|
|
|
|
}, [countdown]);
|
2024-11-28 01:52:00 +01:00
|
|
|
console.log("count");
|
|
|
|
|
|
|
|
const handleNegativeResponse = useCallback(() => {
|
|
|
|
switch (userInputRequest.responsetype) {
|
|
|
|
case "text":
|
|
|
|
handleSendErrResponse();
|
|
|
|
break;
|
|
|
|
case "confirm":
|
|
|
|
handleSendConfirm(false);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}, [userInputRequest.responsetype, handleSendErrResponse, handleSendConfirm]);
|
|
|
|
console.log("before end");
|
2024-07-19 00:21:33 +02:00
|
|
|
|
|
|
|
return (
|
2024-11-28 01:52:00 +01:00
|
|
|
<Modal
|
|
|
|
onOk={() => handleSubmit()}
|
|
|
|
onCancel={() => handleNegativeResponse()}
|
|
|
|
onClose={() => handleSendErrResponse()}
|
|
|
|
okLabel={userInputRequest.oklabel}
|
|
|
|
cancelLabel={userInputRequest.cancellabel}
|
|
|
|
>
|
2024-09-06 13:15:42 +02:00
|
|
|
<div className="userinput-header">{userInputRequest.title + ` (${countdown}s)`}</div>
|
2024-07-19 00:21:33 +02:00
|
|
|
<div className="userinput-body">
|
|
|
|
{queryText}
|
|
|
|
{inputBox}
|
2024-10-11 00:50:46 +02:00
|
|
|
{optionalCheckbox}
|
2024-07-19 00:21:33 +02:00
|
|
|
</div>
|
2024-07-30 20:44:19 +02:00
|
|
|
</Modal>
|
2024-07-19 00:21:33 +02:00
|
|
|
);
|
|
|
|
};
|
2024-07-30 20:44:19 +02:00
|
|
|
|
|
|
|
UserInputModal.displayName = "UserInputModal";
|
|
|
|
|
|
|
|
export { UserInputModal };
|