Skip to content

Commit bfcb136

Browse files
chore: remove personal branding if in an org and use org branding (#14872)
* chore: remove personal branding if in an org and use org branding * fix: minor fix * Update SettingsLayout.tsx * Update SettingsLayout.tsx * Update profile.tsx --------- Co-authored-by: Udit Takkar <53316345+Udit-takkar@users.noreply.github.com>
1 parent cd888be commit bfcb136

File tree

4 files changed

+127
-139
lines changed

4 files changed

+127
-139
lines changed

apps/web/pages/settings/my-account/appearance.tsx

+7-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,7 @@
11
"use client";
22

3+
import { useSession } from "next-auth/react";
4+
import { useRouter } from "next/navigation";
35
import { useState } from "react";
46
import { Controller, useForm } from "react-hook-form";
57
import type { z } from "zod";
@@ -430,7 +432,11 @@ const AppearanceView = ({
430432
const AppearanceViewWrapper = () => {
431433
const { data: user, isPending } = trpc.viewer.me.useQuery();
432434
const { isPending: isTeamPlanStatusLoading, hasPaidPlan } = useHasPaidPlan();
433-
435+
const session = useSession();
436+
const router = useRouter();
437+
if (!!session.data?.user?.org?.id) {
438+
router.replace("/settings/organizations/profile");
439+
}
434440
const { t } = useLocale();
435441

436442
if (isPending || isTeamPlanStatusLoading || !user)

packages/features/ee/organizations/pages/settings/appearance.tsx

+71-132
Original file line numberDiff line numberDiff line change
@@ -5,20 +5,17 @@ import { useRouter } from "next/navigation";
55
import { useState, useEffect } from "react";
66
import { useForm } from "react-hook-form";
77

8-
import LicenseRequired from "@calcom/features/ee/common/components/LicenseRequired";
98
import BrandColorsForm from "@calcom/features/ee/components/BrandColorsForm";
109
import { AppearanceSkeletonLoader } from "@calcom/features/ee/components/CommonSkeletonLoaders";
1110
import SectionBottomActions from "@calcom/features/settings/SectionBottomActions";
1211
import ThemeLabel from "@calcom/features/settings/ThemeLabel";
13-
import { getLayout } from "@calcom/features/settings/layouts/SettingsLayout";
1412
import { DEFAULT_LIGHT_BRAND_COLOR, DEFAULT_DARK_BRAND_COLOR } from "@calcom/lib/constants";
1513
import { APP_NAME } from "@calcom/lib/constants";
1614
import { useLocale } from "@calcom/lib/hooks/useLocale";
1715
import { MembershipRole } from "@calcom/prisma/enums";
1816
import { trpc } from "@calcom/trpc/react";
1917
import type { RouterOutputs } from "@calcom/trpc/react";
20-
import { Button, Form, Meta, showToast, SettingsToggle, Avatar, ImageUploader } from "@calcom/ui";
21-
import { Icon } from "@calcom/ui";
18+
import { Button, Form, showToast, SettingsToggle } from "@calcom/ui";
2219

2320
type BrandColorsFormValues = {
2421
brandColor: string;
@@ -27,7 +24,6 @@ type BrandColorsFormValues = {
2724

2825
const OrgAppearanceView = ({
2926
currentOrg,
30-
isAdminOrOwner,
3127
}: {
3228
currentOrg: RouterOutputs["viewer"]["organizations"]["listCurrent"];
3329
isAdminOrOwner: boolean;
@@ -79,134 +75,79 @@ const OrgAppearanceView = ({
7975
};
8076

8177
return (
82-
<LicenseRequired>
83-
<Meta
84-
title={t("appearance")}
85-
description={t("appearance_org_description")}
86-
borderInShellHeader={false}
87-
/>
88-
{isAdminOrOwner ? (
89-
<div>
90-
<div className="my-6">
91-
<div className="flex items-center text-sm">
92-
<Avatar
93-
alt="calVideoLogo"
94-
imageSrc={currentOrg?.calVideoLogo}
95-
fallback={<Icon name="plus" className="text-subtle h-6 w-6" />}
96-
size="lg"
97-
/>
98-
<div className="ms-4">
99-
<div className="flex gap-2">
100-
<ImageUploader
101-
target="avatar"
102-
id="cal-video-logo-upload"
103-
buttonMsg={
104-
currentOrg?.calVideoLogo ? t("update_cal_video_logo") : t("upload_cal_video_logo")
105-
}
106-
handleAvatarChange={(newLogo) => {
107-
mutation.mutate({
108-
calVideoLogo: newLogo,
109-
});
110-
}}
111-
disabled={mutation.isPending}
112-
imageSrc={currentOrg?.calVideoLogo ?? undefined}
113-
uploadInstruction={t("cal_video_logo_upload_instruction")}
114-
triggerButtonColor={currentOrg?.calVideoLogo ? "secondary" : "primary"}
115-
/>
116-
{currentOrg?.calVideoLogo && (
117-
<Button
118-
color="destructive"
119-
disabled={mutation.isPending}
120-
onClick={() => {
121-
mutation.mutate({
122-
calVideoLogo: null,
123-
});
124-
}}>
125-
{t("remove")}
126-
</Button>
127-
)}
128-
</div>
129-
</div>
130-
</div>
78+
<div>
79+
<Form
80+
form={themeForm}
81+
handleSubmit={(value) => {
82+
mutation.mutate({
83+
theme: value.theme ?? null,
84+
});
85+
}}>
86+
<div className="border-subtle mt-6 flex items-center rounded-t-xl border p-6 text-sm">
87+
<div>
88+
<p className="text-default text-base font-semibold">{t("theme")}</p>
89+
<p className="text-default">{t("theme_applies_note")}</p>
13190
</div>
132-
<Form
133-
form={themeForm}
134-
handleSubmit={(value) => {
135-
mutation.mutate({
136-
theme: value.theme ?? null,
137-
});
138-
}}>
139-
<div className="border-subtle mt-6 flex items-center rounded-t-xl border p-6 text-sm">
140-
<div>
141-
<p className="text-default text-base font-semibold">{t("theme")}</p>
142-
<p className="text-default">{t("theme_applies_note")}</p>
143-
</div>
144-
</div>
145-
<div className="border-subtle flex flex-col justify-between border-x px-6 py-8 sm:flex-row">
146-
<ThemeLabel
147-
variant="system"
148-
value={undefined}
149-
label={t("theme_system")}
150-
defaultChecked={currentOrg.theme === null}
151-
register={themeForm.register}
152-
/>
153-
<ThemeLabel
154-
variant="light"
155-
value="light"
156-
label={t("light")}
157-
defaultChecked={currentOrg.theme === "light"}
158-
register={themeForm.register}
159-
/>
160-
<ThemeLabel
161-
variant="dark"
162-
value="dark"
163-
label={t("dark")}
164-
defaultChecked={currentOrg.theme === "dark"}
165-
register={themeForm.register}
166-
/>
167-
</div>
168-
<SectionBottomActions className="mb-6" align="end">
169-
<Button
170-
disabled={isOrgThemeSubmitting || !isOrgThemeDirty}
171-
type="submit"
172-
data-testid="update-org-theme-btn"
173-
color="primary">
174-
{t("update")}
175-
</Button>
176-
</SectionBottomActions>
177-
</Form>
178-
179-
<Form
180-
form={brandColorsFormMethods}
181-
handleSubmit={(values) => {
182-
onBrandColorsFormSubmit(values);
183-
}}>
184-
<BrandColorsForm
185-
onSubmit={onBrandColorsFormSubmit}
186-
brandColor={currentOrg?.brandColor ?? DEFAULT_LIGHT_BRAND_COLOR}
187-
darkBrandColor={currentOrg?.darkBrandColor ?? DEFAULT_DARK_BRAND_COLOR}
188-
/>
189-
</Form>
190-
191-
<SettingsToggle
192-
toggleSwitchAtTheEnd={true}
193-
title={t("disable_cal_branding", { appName: APP_NAME })}
194-
disabled={mutation?.isPending}
195-
description={t("removes_cal_branding", { appName: APP_NAME })}
196-
checked={hideBrandingValue}
197-
onCheckedChange={(checked) => {
198-
setHideBrandingValue(checked);
199-
mutation.mutate({ hideBranding: checked });
200-
}}
201-
switchContainerClassName="mt-6"
202-
/>
20391
</div>
204-
) : (
205-
<div className="py-5">
206-
<span className="text-default text-sm">{t("only_owner_change")}</span>
92+
<div className="border-subtle flex flex-col justify-between border-x px-6 py-8 sm:flex-row">
93+
<ThemeLabel
94+
variant="system"
95+
value={undefined}
96+
label={t("theme_system")}
97+
defaultChecked={currentOrg.theme === null}
98+
register={themeForm.register}
99+
/>
100+
<ThemeLabel
101+
variant="light"
102+
value="light"
103+
label={t("light")}
104+
defaultChecked={currentOrg.theme === "light"}
105+
register={themeForm.register}
106+
/>
107+
<ThemeLabel
108+
variant="dark"
109+
value="dark"
110+
label={t("dark")}
111+
defaultChecked={currentOrg.theme === "dark"}
112+
register={themeForm.register}
113+
/>
207114
</div>
208-
)}
209-
</LicenseRequired>
115+
<SectionBottomActions className="mb-6" align="end">
116+
<Button
117+
disabled={isOrgThemeSubmitting || !isOrgThemeDirty}
118+
type="submit"
119+
data-testid="update-org-theme-btn"
120+
color="primary">
121+
{t("update")}
122+
</Button>
123+
</SectionBottomActions>
124+
</Form>
125+
126+
<Form
127+
form={brandColorsFormMethods}
128+
handleSubmit={(values) => {
129+
onBrandColorsFormSubmit(values);
130+
}}>
131+
<BrandColorsForm
132+
onSubmit={onBrandColorsFormSubmit}
133+
brandColor={currentOrg?.brandColor ?? DEFAULT_LIGHT_BRAND_COLOR}
134+
darkBrandColor={currentOrg?.darkBrandColor ?? DEFAULT_DARK_BRAND_COLOR}
135+
/>
136+
</Form>
137+
138+
<SettingsToggle
139+
toggleSwitchAtTheEnd={true}
140+
title={t("disable_cal_branding", { appName: APP_NAME })}
141+
disabled={mutation?.isPending}
142+
description={t("removes_cal_branding", { appName: APP_NAME })}
143+
checked={hideBrandingValue}
144+
onCheckedChange={(checked) => {
145+
setHideBrandingValue(checked);
146+
mutation.mutate({ hideBranding: checked });
147+
}}
148+
switchContainerClassName="mt-6"
149+
/>
150+
</div>
210151
);
211152
};
212153

@@ -237,6 +178,4 @@ const OrgAppearanceViewWrapper = () => {
237178
return <OrgAppearanceView currentOrg={currentOrg} isAdminOrOwner={isAdminOrOwner} />;
238179
};
239180

240-
OrgAppearanceViewWrapper.getLayout = getLayout;
241-
242181
export default OrgAppearanceViewWrapper;

packages/features/ee/organizations/pages/settings/profile.tsx

+45-1
Original file line numberDiff line numberDiff line change
@@ -9,13 +9,15 @@ import { z } from "zod";
99

1010
import LicenseRequired from "@calcom/features/ee/common/components/LicenseRequired";
1111
import { subdomainSuffix } from "@calcom/features/ee/organizations/lib/orgDomains";
12+
import OrgAppearanceViewWrapper from "@calcom/features/ee/organizations/pages/settings/appearance";
1213
import SectionBottomActions from "@calcom/features/settings/SectionBottomActions";
1314
import { getPlaceholderAvatar } from "@calcom/lib/defaultAvatarImage";
1415
import { useLocale } from "@calcom/lib/hooks/useLocale";
1516
import { md } from "@calcom/lib/markdownIt";
1617
import turndown from "@calcom/lib/turndownService";
1718
import { MembershipRole } from "@calcom/prisma/enums";
1819
import { trpc } from "@calcom/trpc/react";
20+
import { Icon } from "@calcom/ui";
1921
import {
2022
Avatar,
2123
BannerUploader,
@@ -125,7 +127,10 @@ const OrgProfileView = () => {
125127
<Meta title={t("profile")} description={t("profile_org_description")} borderInShellHeader={true} />
126128
<>
127129
{isOrgAdminOrOwner ? (
128-
<OrgProfileForm defaultValues={defaultValues} />
130+
<>
131+
<OrgProfileForm defaultValues={defaultValues} />
132+
<OrgAppearanceViewWrapper />
133+
</>
129134
) : (
130135
<div className="border-subtle flex rounded-b-md border border-t-0 px-4 py-8 sm:px-6">
131136
<div className="flex-grow">
@@ -165,6 +170,7 @@ const OrgProfileForm = ({ defaultValues }: { defaultValues: FormValues }) => {
165170
const utils = trpc.useUtils();
166171
const { t } = useLocale();
167172
const [firstRender, setFirstRender] = useState(true);
173+
const { data: currentOrg } = trpc.viewer.organizations.listCurrent.useQuery(undefined, {});
168174

169175
const form = useForm({
170176
defaultValues,
@@ -288,6 +294,44 @@ const OrgProfileForm = ({ defaultValues }: { defaultValues: FormValues }) => {
288294
}}
289295
/>
290296
</div>
297+
<div className="mt-2 flex items-center">
298+
<Avatar
299+
alt="calVideoLogo"
300+
imageSrc={currentOrg?.calVideoLogo}
301+
fallback={<Icon name="plus" className="text-subtle h-6 w-6" />}
302+
size="lg"
303+
/>
304+
<div className="ms-4">
305+
<div className="flex gap-2">
306+
<ImageUploader
307+
target="avatar"
308+
id="cal-video-logo-upload"
309+
buttonMsg={currentOrg?.calVideoLogo ? t("update_cal_video_logo") : t("upload_cal_video_logo")}
310+
handleAvatarChange={(newLogo) => {
311+
mutation.mutate({
312+
calVideoLogo: newLogo,
313+
});
314+
}}
315+
disabled={mutation.isPending}
316+
imageSrc={currentOrg?.calVideoLogo ?? undefined}
317+
uploadInstruction={t("cal_video_logo_upload_instruction")}
318+
triggerButtonColor={currentOrg?.calVideoLogo ? "secondary" : "primary"}
319+
/>
320+
{currentOrg?.calVideoLogo && (
321+
<Button
322+
color="destructive"
323+
disabled={mutation.isPending}
324+
onClick={() => {
325+
mutation.mutate({
326+
calVideoLogo: null,
327+
});
328+
}}>
329+
{t("remove")}
330+
</Button>
331+
)}
332+
</div>
333+
</div>
334+
</div>
291335

292336
<Controller
293337
control={form.control}

packages/features/settings/layouts/SettingsLayout.tsx

+4-5
Original file line numberDiff line numberDiff line change
@@ -84,10 +84,6 @@ const tabs: VerticalTabItemProps[] = [
8484
name: "privacy",
8585
href: "/settings/organizations/privacy",
8686
},
87-
{
88-
name: "appearance",
89-
href: "/settings/organizations/appearance",
90-
},
9187
{
9288
name: "billing",
9389
href: "/settings/organizations/billing",
@@ -148,7 +144,7 @@ tabs.find((tab) => {
148144
// The following keys are assigned to admin only
149145
const adminRequiredKeys = ["admin"];
150146
const organizationRequiredKeys = ["organization"];
151-
const organizationAdminKeys = ["privacy", "appearance", "billing", "OAuth Clients", "SSO", "directory_sync"];
147+
const organizationAdminKeys = ["privacy", "billing", "OAuth Clients", "SSO", "directory_sync"];
152148

153149
const useTabs = () => {
154150
const session = useSession();
@@ -161,6 +157,9 @@ const useTabs = () => {
161157
const processTabsMemod = useMemo(() => {
162158
const processedTabs = tabs.map((tab) => {
163159
if (tab.href === "/settings/my-account") {
160+
if (!!session.data?.user?.org?.id) {
161+
tab.children = tab?.children?.filter((child) => child.href !== "/settings/my-account/appearance");
162+
}
164163
return {
165164
...tab,
166165
name: user?.name || "my_account",

0 commit comments

Comments
 (0)