91 lines
3.0 KiB
TypeScript
91 lines
3.0 KiB
TypeScript
import { useState } from "react";
|
|
|
|
import { getLayout } from "@calcom/features/settings/layouts/SettingsLayout";
|
|
import { useLocale } from "@calcom/lib/hooks/useLocale";
|
|
import { trpc } from "@calcom/trpc/react";
|
|
import type { RouterOutputs } from "@calcom/trpc/react";
|
|
import { Meta, showToast, SettingsToggle, SkeletonContainer, SkeletonText } from "@calcom/ui";
|
|
|
|
import PageWrapper from "@components/PageWrapper";
|
|
|
|
const SkeletonLoader = ({ title, description }: { title: string; description: string }) => {
|
|
return (
|
|
<SkeletonContainer>
|
|
<Meta title={title} description={description} borderInShellHeader={true} />
|
|
<div className="border-subtle space-y-6 border border-t-0 px-4 py-8 sm:px-6">
|
|
<SkeletonText className="h-8 w-full" />
|
|
</div>
|
|
</SkeletonContainer>
|
|
);
|
|
};
|
|
|
|
const ProfileImpersonationView = ({ user }: { user: RouterOutputs["viewer"]["me"] }) => {
|
|
const { t } = useLocale();
|
|
const utils = trpc.useContext();
|
|
const [disableImpersonation, setDisableImpersonation] = useState<boolean | undefined>(
|
|
user?.disableImpersonation
|
|
);
|
|
|
|
const mutation = trpc.viewer.updateProfile.useMutation({
|
|
onSuccess: () => {
|
|
showToast(t("profile_updated_successfully"), "success");
|
|
},
|
|
onSettled: () => {
|
|
utils.viewer.me.invalidate();
|
|
},
|
|
onMutate: async ({ disableImpersonation }) => {
|
|
await utils.viewer.me.cancel();
|
|
const previousValue = utils.viewer.me.getData();
|
|
|
|
setDisableImpersonation(disableImpersonation);
|
|
|
|
return { previousValue };
|
|
},
|
|
onError: (error, variables, context) => {
|
|
if (context?.previousValue) {
|
|
utils.viewer.me.setData(undefined, context.previousValue);
|
|
setDisableImpersonation(context.previousValue?.disableImpersonation);
|
|
}
|
|
showToast(`${t("error")}, ${error.message}`, "error");
|
|
},
|
|
});
|
|
|
|
return (
|
|
<>
|
|
<Meta
|
|
title={t("impersonation")}
|
|
description={t("impersonation_description")}
|
|
borderInShellHeader={true}
|
|
/>
|
|
<div>
|
|
<SettingsToggle
|
|
toggleSwitchAtTheEnd={true}
|
|
title={t("user_impersonation_heading")}
|
|
description={t("user_impersonation_description")}
|
|
checked={!disableImpersonation}
|
|
onCheckedChange={(checked) => {
|
|
mutation.mutate({ disableImpersonation: !checked });
|
|
}}
|
|
disabled={mutation.isLoading}
|
|
switchContainerClassName="py-6 px-4 sm:px-6 border-subtle rounded-b-xl border border-t-0"
|
|
/>
|
|
</div>
|
|
</>
|
|
);
|
|
};
|
|
|
|
const ProfileImpersonationViewWrapper = () => {
|
|
const { data: user, isLoading } = trpc.viewer.me.useQuery();
|
|
const { t } = useLocale();
|
|
|
|
if (isLoading || !user)
|
|
return <SkeletonLoader title={t("impersonation")} description={t("impersonation_description")} />;
|
|
|
|
return <ProfileImpersonationView user={user} />;
|
|
};
|
|
|
|
ProfileImpersonationViewWrapper.getLayout = getLayout;
|
|
ProfileImpersonationViewWrapper.PageWrapper = PageWrapper;
|
|
|
|
export default ProfileImpersonationViewWrapper;
|