202 lines
6.3 KiB
TypeScript
202 lines
6.3 KiB
TypeScript
/*
|
|
Copyright 2022-2024 New Vector Ltd.
|
|
|
|
SPDX-License-Identifier: AGPL-3.0-only
|
|
Please see LICENSE in the repository root for full details.
|
|
*/
|
|
|
|
import { FC, FormEventHandler, ReactNode, useCallback, useState } from "react";
|
|
import { MatrixClient } from "matrix-js-sdk/src/client";
|
|
import { Trans, useTranslation } from "react-i18next";
|
|
import { useHistory } from "react-router-dom";
|
|
import { Button, Heading, Text } from "@vector-im/compound-web";
|
|
|
|
import styles from "./CallEndedView.module.css";
|
|
import feedbackStyle from "../input/FeedbackInput.module.css";
|
|
import { useProfile } from "../profile/useProfile";
|
|
import { Header, HeaderLogo, LeftNav, RightNav } from "../Header";
|
|
import { PosthogAnalytics } from "../analytics/PosthogAnalytics";
|
|
import { FieldRow, InputField } from "../input/Input";
|
|
import { StarRatingInput } from "../input/StarRatingInput";
|
|
import { RageshakeButton } from "../settings/RageshakeButton";
|
|
import { Link } from "../button/Link";
|
|
import { LinkButton } from "../button";
|
|
|
|
interface Props {
|
|
client: MatrixClient;
|
|
isPasswordlessUser: boolean;
|
|
confineToRoom: boolean;
|
|
endedCallId: string;
|
|
leaveError?: Error;
|
|
reconnect: () => void;
|
|
}
|
|
|
|
export const CallEndedView: FC<Props> = ({
|
|
client,
|
|
isPasswordlessUser,
|
|
confineToRoom,
|
|
endedCallId,
|
|
leaveError,
|
|
reconnect,
|
|
}) => {
|
|
const { t } = useTranslation();
|
|
const history = useHistory();
|
|
|
|
const { displayName } = useProfile(client);
|
|
const [surveySubmitted, setSurveySubmitted] = useState(false);
|
|
const [starRating, setStarRating] = useState(0);
|
|
const [submitting, setSubmitting] = useState(false);
|
|
const [submitDone, setSubmitDone] = useState(false);
|
|
const submitSurvey: FormEventHandler<HTMLFormElement> = useCallback(
|
|
(e) => {
|
|
e.preventDefault();
|
|
const data = new FormData(e.target as HTMLFormElement);
|
|
const feedbackText = data.get("feedbackText") as string;
|
|
|
|
PosthogAnalytics.instance.eventQualitySurvey.track(
|
|
endedCallId,
|
|
feedbackText,
|
|
starRating,
|
|
);
|
|
|
|
setSubmitting(true);
|
|
|
|
setTimeout(() => {
|
|
setSubmitDone(true);
|
|
|
|
setTimeout(() => {
|
|
if (isPasswordlessUser) {
|
|
// setting this renders the callEndedView with the invitation to create an account
|
|
setSurveySubmitted(true);
|
|
} else if (!confineToRoom) {
|
|
// if the user already has an account immediately go back to the home screen
|
|
history.push("/");
|
|
}
|
|
}, 1000);
|
|
}, 1000);
|
|
},
|
|
[endedCallId, history, isPasswordlessUser, confineToRoom, starRating],
|
|
);
|
|
|
|
const createAccountDialog = isPasswordlessUser && (
|
|
<div className={styles.callEndedContent}>
|
|
<Trans i18nKey="call_ended_view.create_account_prompt">
|
|
<p>Why not finish by setting up a password to keep your account?</p>
|
|
<p>
|
|
You'll be able to keep your name and set an avatar for use on future
|
|
calls
|
|
</p>
|
|
</Trans>
|
|
<LinkButton className={styles.callEndedButton} to="/register">
|
|
{t("call_ended_view.create_account_button")}
|
|
</LinkButton>
|
|
</div>
|
|
);
|
|
|
|
const qualitySurveyDialog = (
|
|
<div className={styles.callEndedContent}>
|
|
<Trans i18nKey="call_ended_view.feedback_prompt">
|
|
<p>
|
|
We'd love to hear your feedback so we can improve your experience.
|
|
</p>
|
|
</Trans>
|
|
<form onSubmit={submitSurvey}>
|
|
<FieldRow>
|
|
<StarRatingInput starCount={5} onChange={setStarRating} required />
|
|
</FieldRow>
|
|
<FieldRow>
|
|
<InputField
|
|
className={feedbackStyle.feedback}
|
|
id="feedbackText"
|
|
name="feedbackText"
|
|
label={t("settings.feedback_tab_description_label")}
|
|
placeholder={t("settings.feedback_tab_description_label")}
|
|
type="textarea"
|
|
/>
|
|
</FieldRow>{" "}
|
|
<FieldRow>
|
|
{submitDone ? (
|
|
<Trans i18nKey="call_ended_view.feedback_done">
|
|
<p>Thanks for your feedback!</p>
|
|
</Trans>
|
|
) : (
|
|
<Button
|
|
type="submit"
|
|
className={styles.submitButton}
|
|
data-testid="home_go"
|
|
>
|
|
{submitting ? t("submitting") : t("action.submit")}
|
|
</Button>
|
|
)}
|
|
</FieldRow>
|
|
</form>
|
|
</div>
|
|
);
|
|
|
|
const renderBody = (): ReactNode => {
|
|
if (leaveError) {
|
|
return (
|
|
<>
|
|
<main className={styles.main}>
|
|
<Heading size="xl" weight="semibold" className={styles.headline}>
|
|
<Trans i18nKey="call_ended_view.body">
|
|
You were disconnected from the call
|
|
</Trans>
|
|
</Heading>
|
|
<div className={styles.disconnectedButtons}>
|
|
<Button onClick={reconnect}>
|
|
{t("call_ended_view.reconnect_button")}
|
|
</Button>
|
|
<div className={styles.rageshakeButton}>
|
|
<RageshakeButton description="***Call disconnected***" />
|
|
</div>
|
|
</div>
|
|
</main>
|
|
{!confineToRoom && (
|
|
<Text className={styles.footer}>
|
|
<Link to="/"> {t("return_home_button")} </Link>
|
|
</Text>
|
|
)}
|
|
</>
|
|
);
|
|
} else {
|
|
return (
|
|
<>
|
|
<main className={styles.main}>
|
|
<Heading size="xl" weight="semibold" className={styles.headline}>
|
|
{surveySubmitted
|
|
? t("call_ended_view.headline", {
|
|
displayName,
|
|
})
|
|
: t("call_ended_view.headline", {
|
|
displayName,
|
|
}) +
|
|
"\n" +
|
|
t("call_ended_view.survey_prompt")}
|
|
</Heading>
|
|
{(!surveySubmitted || confineToRoom) &&
|
|
PosthogAnalytics.instance.isEnabled()
|
|
? qualitySurveyDialog
|
|
: createAccountDialog}
|
|
</main>
|
|
{!confineToRoom && (
|
|
<Text className={styles.footer}>
|
|
<Link to="/"> {t("call_ended_view.not_now_button")} </Link>
|
|
</Text>
|
|
)}
|
|
</>
|
|
);
|
|
}
|
|
};
|
|
|
|
return (
|
|
<>
|
|
<Header>
|
|
<LeftNav>{!confineToRoom && <HeaderLogo />}</LeftNav>
|
|
<RightNav />
|
|
</Header>
|
|
<div className={styles.container}>{renderBody()}</div>
|
|
</>
|
|
);
|
|
};
|