From ad02ddbb4733ec609c29c9d0337e56a7659c654d Mon Sep 17 00:00:00 2001 From: Banks Nussman Date: Mon, 9 Dec 2024 11:26:44 -0500 Subject: [PATCH 1/5] initial refactor --- .../src/components/PrimaryNav/PrimaryNav.tsx | 13 ++++--- .../LinodeResize/LinodeResize.tsx | 12 ++++--- .../src/queries/profile/preferences.ts | 35 ++++++++++--------- .../manager/src/types/ManagerPreferences.ts | 1 + packages/manager/src/utilities/theme.ts | 12 +++++-- 5 files changed, 43 insertions(+), 30 deletions(-) diff --git a/packages/manager/src/components/PrimaryNav/PrimaryNav.tsx b/packages/manager/src/components/PrimaryNav/PrimaryNav.tsx index f968da8fd05..6cac9b49d01 100644 --- a/packages/manager/src/components/PrimaryNav/PrimaryNav.tsx +++ b/packages/manager/src/components/PrimaryNav/PrimaryNav.tsx @@ -93,7 +93,12 @@ export const PrimaryNav = (props: PrimaryNavProps) => { const { isIAMEnabled, isIAMBeta } = useIsIAMEnabled(); - const { data: preferences } = usePreferences(); + const { data: collapsedSideNavPreference } = usePreferences( + (preferences) => preferences?.collapsedSideNavProductFamilies + ); + + const collapsedAccordions = collapsedSideNavPreference ?? []; + const { mutateAsync: updatePreferences } = useMutatePreferences(); const productFamilyLinkGroups: ProductFamilyLinkGroup< @@ -253,10 +258,6 @@ export const PrimaryNav = (props: PrimaryNavProps) => { ] ); - const [collapsedAccordions, setCollapsedAccordions] = React.useState< - number[] - >(preferences?.collapsedSideNavProductFamilies ?? []); - const accordionClicked = (index: number) => { let updatedCollapsedAccordions; if (collapsedAccordions.includes(index)) { @@ -266,13 +267,11 @@ export const PrimaryNav = (props: PrimaryNavProps) => { updatePreferences({ collapsedSideNavProductFamilies: updatedCollapsedAccordions, }); - setCollapsedAccordions(updatedCollapsedAccordions); } else { updatedCollapsedAccordions = [...collapsedAccordions, index]; updatePreferences({ collapsedSideNavProductFamilies: updatedCollapsedAccordions, }); - setCollapsedAccordions(updatedCollapsedAccordions); } }; diff --git a/packages/manager/src/features/Linodes/LinodesDetail/LinodeResize/LinodeResize.tsx b/packages/manager/src/features/Linodes/LinodesDetail/LinodeResize/LinodeResize.tsx index d0403f3bb59..9a38f9ff28e 100644 --- a/packages/manager/src/features/Linodes/LinodesDetail/LinodeResize/LinodeResize.tsx +++ b/packages/manager/src/features/Linodes/LinodesDetail/LinodeResize/LinodeResize.tsx @@ -72,7 +72,12 @@ export const LinodeResize = (props: Props) => { ); const { data: types } = useAllTypes(open); - const { data: preferences } = usePreferences(open); + + const { data: typeToConfirmPreference } = usePreferences( + (preferences) => preferences?.type_to_confirm, + open + ); + const { enqueueSnackbar } = useSnackbar(); const [confirmationText, setConfirmationText] = React.useState(''); const [resizeError, setResizeError] = React.useState(''); @@ -162,8 +167,7 @@ export const LinodeResize = (props: Props) => { const tableDisabled = hostMaintenance || isLinodesGrantReadOnly; const submitButtonDisabled = - preferences?.type_to_confirm !== false && - confirmationText !== linode?.label; + typeToConfirmPreference !== false && confirmationText !== linode?.label; const type = types?.find((t) => t.id === linode?.type); @@ -323,7 +327,7 @@ export const LinodeResize = (props: Props) => { title="Confirm" typographyStyle={{ marginBottom: 8 }} value={confirmationText} - visible={preferences?.type_to_confirm} + visible={typeToConfirmPreference} /> diff --git a/packages/manager/src/queries/profile/preferences.ts b/packages/manager/src/queries/profile/preferences.ts index 086b3e6fcfc..4f5594bf220 100644 --- a/packages/manager/src/queries/profile/preferences.ts +++ b/packages/manager/src/queries/profile/preferences.ts @@ -1,27 +1,25 @@ import { updateUserPreferences } from '@linode/api-v4'; -import { - QueryClient, - useMutation, - useQuery, - useQueryClient, -} from '@tanstack/react-query'; - -import { ManagerPreferences } from 'src/types/ManagerPreferences'; +import { useMutation, useQuery, useQueryClient } from '@tanstack/react-query'; import { queryPresets } from '../base'; import { profileQueries } from './profile'; import type { APIError } from '@linode/api-v4'; +import type { QueryClient } from '@tanstack/react-query'; +import type { ManagerPreferences } from 'src/types/ManagerPreferences'; -export const usePreferences = (enabled = true) => - useQuery({ +export const usePreferences = ( + select?: (data: ManagerPreferences | undefined) => TData, + enabled = true +) => + useQuery({ ...profileQueries.preferences, ...queryPresets.oneTimeFetch, enabled, + select, }); export const useMutatePreferences = (replace = false) => { - const { data: preferences } = usePreferences(!replace); const queryClient = useQueryClient(); return useMutation< @@ -29,11 +27,16 @@ export const useMutatePreferences = (replace = false) => { APIError[], Partial >({ - mutationFn: (data) => - updateUserPreferences({ - ...(!replace && preferences !== undefined ? preferences : {}), - ...data, - }), + async mutationFn(data) { + if (replace) { + return updateUserPreferences(data); + } else { + const existingPreferences = await queryClient.ensureQueryData( + profileQueries.preferences + ); + return updateUserPreferences({ ...existingPreferences, ...data }); + } + }, onMutate: (data) => updatePreferenceData(data, replace, queryClient), }); }; diff --git a/packages/manager/src/types/ManagerPreferences.ts b/packages/manager/src/types/ManagerPreferences.ts index 5c89f454313..7515e3c3b2c 100644 --- a/packages/manager/src/types/ManagerPreferences.ts +++ b/packages/manager/src/types/ManagerPreferences.ts @@ -17,6 +17,7 @@ export interface DismissedNotification { export interface ManagerPreferences extends UserPreferences { avatarColor?: string; backups_cta_dismissed?: boolean; + collapsedSideNavProductFamilies?: number[]; desktop_sidebar_open?: boolean; dismissed_notifications?: Record; domains_group_by_tag?: boolean; diff --git a/packages/manager/src/utilities/theme.ts b/packages/manager/src/utilities/theme.ts index 7de52ec1697..0155e58443d 100644 --- a/packages/manager/src/utilities/theme.ts +++ b/packages/manager/src/utilities/theme.ts @@ -52,13 +52,19 @@ export const getThemeFromPreferenceValue = ( }; export const useColorMode = () => { - // Make sure we are authenticated before we fetch preferences. const isAuthenticated = !!useAuthentication().token; - const { data: preferences } = usePreferences(isAuthenticated); + + const { data: themePreference } = usePreferences( + (preferences) => preferences?.theme, + // Make sure we are authenticated before we fetch preferences. + // If we don't, we get an authentication loop. + isAuthenticated + ); + const isSystemInDarkMode = useMediaQuery('(prefers-color-scheme: dark)'); const colorMode = getThemeFromPreferenceValue( - preferences?.theme, + themePreference, isSystemInDarkMode ); From a31c62c80daeaaa9709b98fd192dfdc1c052a2f5 Mon Sep 17 00:00:00 2001 From: Banks Nussman Date: Mon, 9 Dec 2024 13:19:28 -0500 Subject: [PATCH 2/5] fix type-safety issues --- .../PreferenceToggle/PreferenceToggle.tsx | 36 ++++++++++--------- .../Linodes/LinodesLanding/LinodesLanding.tsx | 9 +++-- .../manager/src/types/ManagerPreferences.ts | 5 +-- 3 files changed, 26 insertions(+), 24 deletions(-) diff --git a/packages/manager/src/components/PreferenceToggle/PreferenceToggle.tsx b/packages/manager/src/components/PreferenceToggle/PreferenceToggle.tsx index b4494ec9a26..f769d10b532 100644 --- a/packages/manager/src/components/PreferenceToggle/PreferenceToggle.tsx +++ b/packages/manager/src/components/PreferenceToggle/PreferenceToggle.tsx @@ -3,28 +3,30 @@ import { usePreferences, } from 'src/queries/profile/preferences'; -export interface PreferenceToggleProps { - preference: T; - togglePreference: () => T; -} +import type { ManagerPreferences } from 'src/types/ManagerPreferences'; interface RenderChildrenProps { - preference: T; - togglePreference: () => T; + preference: NonNullable; + togglePreference: () => NonNullable; } type RenderChildren = (props: RenderChildrenProps) => JSX.Element; -interface Props { - children: RenderChildren; - initialSetCallbackFn?: (value: T) => void; - preferenceKey: string; - preferenceOptions: [T, T]; - toggleCallbackFn?: (value: T) => void; - value?: T; +interface Props { + children: RenderChildren; + initialSetCallbackFn?: (value: ManagerPreferences[Key]) => void; + preferenceKey: Key; + preferenceOptions: [ManagerPreferences[Key], ManagerPreferences[Key]]; + toggleCallbackFn?: (value: ManagerPreferences[Key]) => void; + value?: ManagerPreferences[Key]; } -export const PreferenceToggle = (props: Props) => { +/** + * @deprecated There are more simple ways to use preferences. Look into using `usePreferences` directly. + */ +export const PreferenceToggle = ( + props: Props +) => { const { children, preferenceKey, @@ -38,7 +40,7 @@ export const PreferenceToggle = (props: Props) => { const { mutateAsync: updateUserPreferences } = useMutatePreferences(); const togglePreference = () => { - let newPreferenceToSet: T; + let newPreferenceToSet: ManagerPreferences[Key]; if (preferences?.[preferenceKey] === undefined) { // Because we default to preferenceOptions[0], toggling with no preference should pick preferenceOptions[1] @@ -58,11 +60,11 @@ export const PreferenceToggle = (props: Props) => { toggleCallbackFn(newPreferenceToSet); } - return newPreferenceToSet; + return newPreferenceToSet!; }; return children({ - preference: value ?? preferences?.[preferenceKey] ?? preferenceOptions[0], + preference: value ?? preferences?.[preferenceKey] ?? preferenceOptions[0]!, togglePreference, }); }; diff --git a/packages/manager/src/features/Linodes/LinodesLanding/LinodesLanding.tsx b/packages/manager/src/features/Linodes/LinodesLanding/LinodesLanding.tsx index 8c2b2aa11d1..cdd57ba4996 100644 --- a/packages/manager/src/features/Linodes/LinodesLanding/LinodesLanding.tsx +++ b/packages/manager/src/features/Linodes/LinodesLanding/LinodesLanding.tsx @@ -43,7 +43,6 @@ import type { ExtendedStatus } from './utils'; import type { Config } from '@linode/api-v4/lib/linodes/types'; import type { APIError } from '@linode/api-v4/lib/types'; import type { RouteComponentProps } from 'react-router-dom'; -import type { PreferenceToggleProps } from 'src/components/PreferenceToggle/PreferenceToggle'; import type { WithFeatureFlagProps } from 'src/containers/flags.container'; import type { WithProfileProps } from 'src/containers/profile.container'; import type { DialogType } from 'src/features/Linodes/types'; @@ -308,7 +307,7 @@ class ListLinodes extends React.Component { )} - + { {({ preference: linodesAreGrouped, togglePreference: toggleGroupLinodes, - }: PreferenceToggleProps) => { + }) => { return ( - + { {({ preference: linodeViewPreference, togglePreference: toggleLinodeView, - }: PreferenceToggleProps<'grid' | 'list'>) => { + }) => { return ( diff --git a/packages/manager/src/types/ManagerPreferences.ts b/packages/manager/src/types/ManagerPreferences.ts index 7515e3c3b2c..7cf133a7065 100644 --- a/packages/manager/src/types/ManagerPreferences.ts +++ b/packages/manager/src/types/ManagerPreferences.ts @@ -1,4 +1,4 @@ -import type { UserPreferences } from '@linode/api-v4'; +import type { AclpConfig } from '@linode/api-v4'; import type { Order } from 'src/hooks/useOrder'; import type { ThemeChoice } from 'src/utilities/theme'; @@ -14,7 +14,8 @@ export interface DismissedNotification { label?: string; } -export interface ManagerPreferences extends UserPreferences { +export interface ManagerPreferences { + aclpPreference?: AclpConfig; // Why is this type in @linode/api-v4? avatarColor?: string; backups_cta_dismissed?: boolean; collapsedSideNavProductFamilies?: number[]; From 605fa4f52270f5111d111c4c9e707b766b66b047 Mon Sep 17 00:00:00 2001 From: Banks Nussman <115251059+bnussman-akamai@users.noreply.github.com> Date: Wed, 11 Dec 2024 09:44:52 -0500 Subject: [PATCH 3/5] Update packages/manager/src/queries/profile/preferences.ts Co-authored-by: Purvesh Makode --- packages/manager/src/queries/profile/preferences.ts | 9 ++++----- 1 file changed, 4 insertions(+), 5 deletions(-) diff --git a/packages/manager/src/queries/profile/preferences.ts b/packages/manager/src/queries/profile/preferences.ts index 4f5594bf220..11c82b4092a 100644 --- a/packages/manager/src/queries/profile/preferences.ts +++ b/packages/manager/src/queries/profile/preferences.ts @@ -30,12 +30,11 @@ export const useMutatePreferences = (replace = false) => { async mutationFn(data) { if (replace) { return updateUserPreferences(data); - } else { - const existingPreferences = await queryClient.ensureQueryData( - profileQueries.preferences - ); - return updateUserPreferences({ ...existingPreferences, ...data }); } + const existingPreferences = await queryClient.ensureQueryData( + profileQueries.preferences + ); + return updateUserPreferences({ ...existingPreferences, ...data }); }, onMutate: (data) => updatePreferenceData(data, replace, queryClient), }); From d185cc412770692ad2f37b977bfa8cca859028eb Mon Sep 17 00:00:00 2001 From: Banks Nussman Date: Wed, 11 Dec 2024 12:07:21 -0500 Subject: [PATCH 4/5] fix re-render at maincontent level @mjac0bs --- packages/manager/src/MainContent.tsx | 10 ++++++---- packages/manager/src/Root.tsx | 8 +++++--- 2 files changed, 11 insertions(+), 7 deletions(-) diff --git a/packages/manager/src/MainContent.tsx b/packages/manager/src/MainContent.tsx index 82a2b3eb6d5..01d688f4799 100644 --- a/packages/manager/src/MainContent.tsx +++ b/packages/manager/src/MainContent.tsx @@ -31,6 +31,7 @@ import { sessionExpirationContext } from './context/sessionExpirationContext'; import { switchAccountSessionContext } from './context/switchAccountSessionContext'; import { useIsACLPEnabled } from './features/CloudPulse/Utils/utils'; import { useIsDatabasesEnabled } from './features/Databases/utilities'; +import { useIsIAMEnabled } from './features/IAM/Shared/utilities'; import { useIsPlacementGroupsEnabled } from './features/PlacementGroups/utils'; import { useGlobalErrors } from './hooks/useGlobalErrors'; import { useAccountSettings } from './queries/account/settings'; @@ -39,7 +40,6 @@ import { migrationRouter } from './routes'; import type { Theme } from '@mui/material/styles'; import type { AnyRouter } from '@tanstack/react-router'; -import { useIsIAMEnabled } from './features/IAM/Shared/utilities'; const useStyles = makeStyles()((theme: Theme) => ({ activationWrapper: { @@ -205,7 +205,9 @@ const IAM = React.lazy(() => export const MainContent = () => { const { classes, cx } = useStyles(); - const { data: preferences } = usePreferences(); + const { data: isDesktopSidebarOpenPreference } = usePreferences( + (preferences) => preferences?.desktop_sidebar_open + ); const { mutateAsync: updatePreferences } = useMutatePreferences(); const queryClient = useQueryClient(); @@ -287,11 +289,11 @@ export const MainContent = () => { return ; } - const desktopMenuIsOpen = preferences?.desktop_sidebar_open ?? false; + const desktopMenuIsOpen = isDesktopSidebarOpenPreference ?? false; const desktopMenuToggle = () => { updatePreferences({ - desktop_sidebar_open: !preferences?.desktop_sidebar_open, + desktop_sidebar_open: !isDesktopSidebarOpenPreference, }); }; diff --git a/packages/manager/src/Root.tsx b/packages/manager/src/Root.tsx index 3df94224ecb..64089d9c47e 100644 --- a/packages/manager/src/Root.tsx +++ b/packages/manager/src/Root.tsx @@ -31,7 +31,9 @@ import { useStyles } from './Root.styles'; export const Root = () => { const { classes, cx } = useStyles(); - const { data: preferences } = usePreferences(); + const { data: isDesktopSidebarOpenPreference } = usePreferences( + (preferences) => preferences?.desktop_sidebar_open + ); const { mutateAsync: updatePreferences } = useMutatePreferences(); const globalErrors = useGlobalErrors(); @@ -57,11 +59,11 @@ export const Root = () => { const { data: profile } = useProfile(); const username = profile?.username || ''; - const desktopMenuIsOpen = preferences?.desktop_sidebar_open ?? false; + const desktopMenuIsOpen = isDesktopSidebarOpenPreference ?? false; const desktopMenuToggle = () => { updatePreferences({ - desktop_sidebar_open: !preferences?.desktop_sidebar_open, + desktop_sidebar_open: !isDesktopSidebarOpenPreference, }); }; From f4f90706a4d072f79fde44da08dd2c839bb97644 Mon Sep 17 00:00:00 2001 From: Banks Nussman <115251059+bnussman-akamai@users.noreply.github.com> Date: Wed, 11 Dec 2024 20:17:47 -0500 Subject: [PATCH 5/5] Update packages/manager/src/queries/profile/preferences.ts Co-authored-by: Mariah Jacobs <114685994+mjac0bs@users.noreply.github.com> --- packages/manager/src/queries/profile/preferences.ts | 1 + 1 file changed, 1 insertion(+) diff --git a/packages/manager/src/queries/profile/preferences.ts b/packages/manager/src/queries/profile/preferences.ts index 11c82b4092a..e66fd726360 100644 --- a/packages/manager/src/queries/profile/preferences.ts +++ b/packages/manager/src/queries/profile/preferences.ts @@ -8,6 +8,7 @@ import type { APIError } from '@linode/api-v4'; import type { QueryClient } from '@tanstack/react-query'; import type { ManagerPreferences } from 'src/types/ManagerPreferences'; +// Reference for this pattern: https://tkdodo.eu/blog/react-query-data-transformations#3-using-the-select-option export const usePreferences = ( select?: (data: ManagerPreferences | undefined) => TData, enabled = true