diff --git a/packages/manager/.changeset/pr-11418-tech-stories-1734276290091.md b/packages/manager/.changeset/pr-11418-tech-stories-1734276290091.md new file mode 100644 index 00000000000..98ef05cd9e5 --- /dev/null +++ b/packages/manager/.changeset/pr-11418-tech-stories-1734276290091.md @@ -0,0 +1,5 @@ +--- +"@linode/manager": Tech Stories +--- + +Refactor Domains Routing (Tanstack Router) ([#11418](https://github.com/linode/manager/pull/11418)) diff --git a/packages/manager/.eslintrc.cjs b/packages/manager/.eslintrc.cjs index 3e3b3de5a66..cc85bbb89d0 100644 --- a/packages/manager/.eslintrc.cjs +++ b/packages/manager/.eslintrc.cjs @@ -90,6 +90,7 @@ module.exports = { files: [ // for each new features added to the migration router, add its directory here 'src/features/Betas/**/*', + 'src/features/Domains/**/*', 'src/features/Volumes/**/*', ], rules: { diff --git a/packages/manager/cypress/e2e/core/domains/smoke-clone-domain.spec.ts b/packages/manager/cypress/e2e/core/domains/smoke-clone-domain.spec.ts index ac922358180..e168fb625b5 100644 --- a/packages/manager/cypress/e2e/core/domains/smoke-clone-domain.spec.ts +++ b/packages/manager/cypress/e2e/core/domains/smoke-clone-domain.spec.ts @@ -120,7 +120,7 @@ describe('Clone a Domain', () => { .click(); }); // After cloning a Domain, the user is redirected to the new Domain's details page - cy.url().should('endWith', 'domains'); + cy.url().should('match', /\/domains\/\d+$/); // Confirm that domain is cloned and cloned domains contain the same records as the original Domain. cy.visitWithLogin('/domains'); diff --git a/packages/manager/src/MainContent.tsx b/packages/manager/src/MainContent.tsx index 01d688f4799..dcdae2ec607 100644 --- a/packages/manager/src/MainContent.tsx +++ b/packages/manager/src/MainContent.tsx @@ -131,11 +131,6 @@ const LinodesRoutes = React.lazy(() => default: module.LinodesRoutes, })) ); -const Domains = React.lazy(() => - import('src/features/Domains').then((module) => ({ - default: module.DomainsRoutes, - })) -); const Images = React.lazy(() => import('src/features/Images')); const Kubernetes = React.lazy(() => import('src/features/Kubernetes').then((module) => ({ @@ -343,7 +338,6 @@ export const MainContent = () => { component={NodeBalancers} path="/nodebalancers" /> - diff --git a/packages/manager/src/features/Betas/BetaSignup.tsx b/packages/manager/src/features/Betas/BetaSignup.tsx index d62a9e5e278..90338c3c931 100644 --- a/packages/manager/src/features/Betas/BetaSignup.tsx +++ b/packages/manager/src/features/Betas/BetaSignup.tsx @@ -15,7 +15,7 @@ import { useCreateAccountBetaMutation } from 'src/queries/account/betas'; import { useBetaQuery } from 'src/queries/betas'; export const BetaSignup = () => { - const betaAgreement = `# Early Adopter Testing Program + const betaAgreement = `### Early Adopter Testing Program This Early Adopter Testing Program Service Level Agreement (the “EAP”) is between Linode LLC (“Linode”) and you, the customer who requests access and participation (the “Participant”) to the Linode Early Access Program (the “Program”). This EAP is attached to and amends the master services agreement between you and Linode diff --git a/packages/manager/src/features/Domains/CloneDomainDrawer.tsx b/packages/manager/src/features/Domains/CloneDomainDrawer.tsx index 891f536b9e8..8c1c102dac1 100644 --- a/packages/manager/src/features/Domains/CloneDomainDrawer.tsx +++ b/packages/manager/src/features/Domains/CloneDomainDrawer.tsx @@ -5,9 +5,9 @@ import { RadioGroup, TextField, } from '@linode/ui'; +import { useNavigate } from '@tanstack/react-router'; import { useFormik } from 'formik'; import React from 'react'; -import { useHistory } from 'react-router-dom'; import { ActionsPanel } from 'src/components/ActionsPanel/ActionsPanel'; import { Drawer } from 'src/components/Drawer'; @@ -18,12 +18,13 @@ import type { Domain } from '@linode/api-v4'; interface CloneDomainDrawerProps { domain: Domain | undefined; + isFetching: boolean; onClose: () => void; open: boolean; } export const CloneDomainDrawer = (props: CloneDomainDrawerProps) => { - const { domain, onClose: _onClose, open } = props; + const { domain, isFetching, onClose: _onClose, open } = props; const { data: profile } = useProfile(); const { data: grants } = useGrants(); @@ -32,14 +33,17 @@ export const CloneDomainDrawer = (props: CloneDomainDrawerProps) => { domain?.id ?? 0 ); - const history = useHistory(); + const navigate = useNavigate(); const formik = useFormik<{ domain: string }>({ initialValues: { domain: '' }, onSubmit: async (values) => { const newDomain = await cloneDomain(values); - history.push(`/domains/${newDomain.id}`); onClose(); + navigate({ + params: { domainId: newDomain.id }, + to: '/domains/$domainId', + }); }, }); @@ -52,7 +56,12 @@ export const CloneDomainDrawer = (props: CloneDomainDrawerProps) => { const noPermission = profile?.restricted && !grants?.global.add_domains; return ( - + {noPermission && ( You do not have permission to create new Domains. diff --git a/packages/manager/src/features/Domains/CreateDomain/CreateDomain.tsx b/packages/manager/src/features/Domains/CreateDomain/CreateDomain.tsx index 49cd6e60593..f75c545105d 100644 --- a/packages/manager/src/features/Domains/CreateDomain/CreateDomain.tsx +++ b/packages/manager/src/features/Domains/CreateDomain/CreateDomain.tsx @@ -11,11 +11,10 @@ import { import { createDomainSchema } from '@linode/validation/lib/domains.schema'; import { styled } from '@mui/material/styles'; import Grid from '@mui/material/Unstable_Grid2'; -import { createLazyRoute } from '@tanstack/react-router'; +import { useNavigate } from '@tanstack/react-router'; import { useFormik } from 'formik'; import { path } from 'ramda'; import * as React from 'react'; -import { useHistory } from 'react-router-dom'; import { ActionsPanel } from 'src/components/ActionsPanel/ActionsPanel'; import { DocumentTitleSegment } from 'src/components/DocumentTitle'; @@ -46,6 +45,7 @@ import type { } from '@linode/api-v4/lib/domains'; import type { NodeBalancer } from '@linode/api-v4/lib/nodebalancers'; import type { APIError } from '@linode/api-v4/lib/types'; +import type { DomainState } from 'src/routes/domains'; import type { ExtendedIP } from 'src/utilities/ipUtils'; interface DefaultRecordsSetting { @@ -65,7 +65,7 @@ export const CreateDomain = () => { // of the payload and must be handled separately. const [errors, setErrors] = React.useState(undefined); - const history = useHistory(); + const navigate = useNavigate(); const defaultRecords: DefaultRecordsSetting[] = [ { @@ -127,20 +127,24 @@ export const CreateDomain = () => { const isCreatingPrimaryDomain = values.type === 'master'; const isCreatingSecondaryDomain = values.type === 'slave'; - const redirect = (id: '' | number, state?: Record) => { + const redirect = (id: null | number, state?: DomainState) => { const returnPath = !!id ? `/domains/${id}` : '/domains'; - history.push(returnPath, state); + navigate({ + params: { domainId: Number(id) }, + state: (prev) => ({ ...prev, ...state }), + to: returnPath, + }); }; const redirectToLandingOrDetail = ( type: 'master' | 'slave', domainID: number, - state: Record = {} + state: DomainState = {} ) => { if (type === 'master' && domainID) { redirect(domainID, state); } else { - redirect('', state); + redirect(null, state); } }; @@ -449,10 +453,6 @@ export const CreateDomain = () => { ); }; -export const createDomainLazyRoute = createLazyRoute('/domains/create')({ - component: CreateDomain, -}); - const StyledGrid = styled(Grid, { label: 'StyledGrid' })({ width: '100%', }); diff --git a/packages/manager/src/features/Domains/DisableDomainDialog.tsx b/packages/manager/src/features/Domains/DisableDomainDialog.tsx index fd770db6110..5fb25010178 100644 --- a/packages/manager/src/features/Domains/DisableDomainDialog.tsx +++ b/packages/manager/src/features/Domains/DisableDomainDialog.tsx @@ -9,13 +9,14 @@ import type { Domain } from '@linode/api-v4'; interface DisableDomainDialogProps { domain: Domain | undefined; + isFetching: boolean; onClose: () => void; open: boolean; } export const DisableDomainDialog = React.memo( (props: DisableDomainDialogProps) => { - const { domain, onClose, open } = props; + const { domain, isFetching, onClose, open } = props; const { error, isPending, @@ -56,6 +57,7 @@ export const DisableDomainDialog = React.memo( /> } error={error?.[0]?.reason} + isFetching={isFetching} onClose={onClose} open={open} title={`Disable Domain ${domain?.domain}?`} diff --git a/packages/manager/src/features/Domains/DomainActionMenu.test.tsx b/packages/manager/src/features/Domains/DomainActionMenu.test.tsx index 4b7efc443a9..d69c967d38d 100644 --- a/packages/manager/src/features/Domains/DomainActionMenu.test.tsx +++ b/packages/manager/src/features/Domains/DomainActionMenu.test.tsx @@ -2,7 +2,7 @@ import { userEvent } from '@testing-library/user-event'; import * as React from 'react'; import { domainFactory } from 'src/factories/domain'; -import { renderWithTheme } from 'src/utilities/testHelpers'; +import { renderWithThemeAndRouter } from 'src/utilities/testHelpers'; import { DomainActionMenu } from './DomainActionMenu'; @@ -15,7 +15,7 @@ const props = { describe('Domain action menu', () => { it('should include basic Domain actions', async () => { - const { getByText, queryByLabelText } = renderWithTheme( + const { getByText, queryByLabelText } = await renderWithThemeAndRouter( ); diff --git a/packages/manager/src/features/Domains/DomainDetail/DomainDetail.tsx b/packages/manager/src/features/Domains/DomainDetail/DomainDetail.tsx index 194da9f2a17..6d59718b8ae 100644 --- a/packages/manager/src/features/Domains/DomainDetail/DomainDetail.tsx +++ b/packages/manager/src/features/Domains/DomainDetail/DomainDetail.tsx @@ -1,8 +1,8 @@ import { CircleProgress, Notice, Paper, Typography } from '@linode/ui'; import { styled } from '@mui/material/styles'; import Grid from '@mui/material/Unstable_Grid2'; +import { useLocation, useNavigate, useParams } from '@tanstack/react-router'; import * as React from 'react'; -import { useHistory, useLocation, useParams } from 'react-router-dom'; import { ErrorState } from 'src/components/ErrorState/ErrorState'; import { LandingHeader } from 'src/components/LandingHeader'; @@ -18,14 +18,18 @@ import { DeleteDomain } from '../DeleteDomain'; import DomainRecords from '../DomainRecords'; import { DownloadDNSZoneFileButton } from '../DownloadDNSZoneFileButton'; -export const DomainDetail = () => { - const params = useParams<{ domainId: string }>(); - const domainId = Number(params.domainId); - - const history = useHistory(); - const location = useLocation<{ recordError?: string }>(); +import type { DomainState } from 'src/routes/domains'; - const { data: domain, error, isLoading } = useDomainQuery(domainId); +export const DomainDetail = () => { + const navigate = useNavigate(); + const params = useParams({ from: '/domains/$domainId' }); + const domainId = params.domainId; + const location = useLocation(); + const locationState = location.state as DomainState; + const { data: domain, error, isLoading } = useDomainQuery( + domainId, + !!domainId + ); const { mutateAsync: updateDomain } = useUpdateDomainMutation(); const { data: records, @@ -110,8 +114,8 @@ export const DomainDetail = () => { docsLink="https://techdocs.akamai.com/cloud-computing/docs/dns-manager" title="Domain Details" /> - {location.state && location.state.recordError && ( - + {locationState?.recordError && ( + )} @@ -138,7 +142,7 @@ export const DomainDetail = () => { history.push('/domains')} + onSuccess={() => navigate({ to: '/domains' })} /> diff --git a/packages/manager/src/features/Domains/DomainDetail/index.tsx b/packages/manager/src/features/Domains/DomainDetail/index.tsx index fdb8bb62e4f..6eaa1b60106 100644 --- a/packages/manager/src/features/Domains/DomainDetail/index.tsx +++ b/packages/manager/src/features/Domains/DomainDetail/index.tsx @@ -1,7 +1,6 @@ import { CircleProgress } from '@linode/ui'; -import { createLazyRoute } from '@tanstack/react-router'; +import { useParams } from '@tanstack/react-router'; import * as React from 'react'; -import { useParams } from 'react-router-dom'; import { ErrorState } from 'src/components/ErrorState/ErrorState'; import { NotFound } from 'src/components/NotFound'; @@ -17,8 +16,8 @@ const DomainDetail = React.lazy(() => ); export const DomainDetailRouting = () => { - const params = useParams<{ domainId: string }>(); - const domainId = Number(params.domainId); + const params = useParams({ from: '/domains/$domainId' }); + const domainId = params.domainId; const { data: domain, error, isLoading } = useDomainQuery(domainId); @@ -43,7 +42,3 @@ export const DomainDetailRouting = () => { // page with an open drawer. return ; }; - -export const domainDetailLazyRoute = createLazyRoute('/domains/$domainId')({ - component: DomainDetailRouting, -}); diff --git a/packages/manager/src/features/Domains/DomainZoneImportDrawer.tsx b/packages/manager/src/features/Domains/DomainZoneImportDrawer.tsx index 2946893b578..80f2a1fe93e 100644 --- a/packages/manager/src/features/Domains/DomainZoneImportDrawer.tsx +++ b/packages/manager/src/features/Domains/DomainZoneImportDrawer.tsx @@ -1,7 +1,7 @@ import { Notice, TextField } from '@linode/ui'; import { useFormik } from 'formik'; import * as React from 'react'; -import { useHistory } from 'react-router-dom'; +import { useNavigate } from '@tanstack/react-router'; import { ActionsPanel } from 'src/components/ActionsPanel/ActionsPanel'; import { Drawer } from 'src/components/Drawer'; @@ -21,7 +21,7 @@ export const DomainZoneImportDrawer = (props: DomainZoneImportDrawerProps) => { const { data: profile } = useProfile(); const { data: grants } = useGrants(); - const history = useHistory(); + const navigate = useNavigate(); const { error, mutateAsync: importZone, reset } = useImportZoneMutation(); @@ -32,7 +32,10 @@ export const DomainZoneImportDrawer = (props: DomainZoneImportDrawerProps) => { }, onSubmit: async (values) => { const result = await importZone(values); - history.push(`/domains/${result.id}`); + navigate({ + params: { domainId: result.id }, + to: '/domains/$domainId', + }); }, }); diff --git a/packages/manager/src/features/Domains/DomainsLanding.test.tsx b/packages/manager/src/features/Domains/DomainsLanding.test.tsx index 97314ad8655..92ed3bbde17 100644 --- a/packages/manager/src/features/Domains/DomainsLanding.test.tsx +++ b/packages/manager/src/features/Domains/DomainsLanding.test.tsx @@ -1,12 +1,28 @@ import * as React from 'react'; -import { renderWithTheme } from 'src/utilities/testHelpers'; +import { migrationRouteTree } from 'src/routes'; +import { renderWithThemeAndRouter } from 'src/utilities/testHelpers'; import { DomainsLanding } from './DomainsLanding'; +vi.mock('src/queries/domains', async () => { + const actual = await vi.importActual('src/queries/domains'); + return { + ...actual, + useDomainsQuery: vi.fn().mockReturnValue({ + data: undefined, + error: null, + isLoading: true, + }), + }; +}); + describe('Domains Landing', () => { - it('should initially render a loading state', () => { - const { getByTestId } = renderWithTheme(); + it('should initially render a loading state', async () => { + const { getByTestId } = await renderWithThemeAndRouter(, { + initialRoute: '/domains', + routeTree: migrationRouteTree, + }); expect(getByTestId('circle-progress')).toBeInTheDocument(); }); }); diff --git a/packages/manager/src/features/Domains/DomainsLanding.tsx b/packages/manager/src/features/Domains/DomainsLanding.tsx index 47e4dae0884..f96eb352ccb 100644 --- a/packages/manager/src/features/Domains/DomainsLanding.tsx +++ b/packages/manager/src/features/Domains/DomainsLanding.tsx @@ -1,9 +1,8 @@ import { Button, CircleProgress, Notice } from '@linode/ui'; import { styled } from '@mui/material/styles'; -import { createLazyRoute } from '@tanstack/react-router'; +import { useLocation, useNavigate, useParams } from '@tanstack/react-router'; import { useSnackbar } from 'notistack'; import * as React from 'react'; -import { useHistory, useLocation } from 'react-router-dom'; import { DeletionDialog } from 'src/components/DeletionDialog/DeletionDialog'; import { DocumentTitleSegment } from 'src/components/DocumentTitle'; @@ -17,10 +16,12 @@ import { TableCell } from 'src/components/TableCell'; import { TableHead } from 'src/components/TableHead'; import { TableRow } from 'src/components/TableRow'; import { TableSortCell } from 'src/components/TableSortCell'; -import { useOrder } from 'src/hooks/useOrder'; -import { usePagination } from 'src/hooks/usePagination'; +import { useDialogData } from 'src/hooks/useDialogData'; +import { useOrderV2 } from 'src/hooks/useOrderV2'; +import { usePaginationV2 } from 'src/hooks/usePaginationV2'; import { useDeleteDomainMutation, + useDomainQuery, useDomainsQuery, useUpdateDomainMutation, } from 'src/queries/domains'; @@ -29,6 +30,11 @@ import { useProfile } from 'src/queries/profile/profile'; import { getAPIErrorOrDefault } from 'src/utilities/errorUtils'; import { CloneDomainDrawer } from './CloneDomainDrawer'; +import { + DOMAINS_TABLE_DEFAULT_ORDER, + DOMAINS_TABLE_DEFAULT_ORDER_BY, + DOMAINS_TABLE_PREFERENCE_KEY, +} from './constants'; import { DisableDomainDialog } from './DisableDomainDialog'; import { DomainBanner } from './DomainBanner'; import { DomainsEmptyLandingState } from './DomainsEmptyLandingPage'; @@ -38,6 +44,7 @@ import { EditDomainDrawer } from './EditDomainDrawer'; import type { Handlers as DomainHandlers } from './DomainActionMenu'; import type { Domain } from '@linode/api-v4'; +import type { DomainState } from 'src/routes/domains'; const DOMAIN_CREATE_ROUTE = '/domains/create'; @@ -47,24 +54,30 @@ interface DomainsLandingProps { domainForEditing?: Domain; } -const PREFERENCE_KEY = 'domains'; - export const DomainsLanding = (props: DomainsLandingProps) => { - const history = useHistory(); - const location = useLocation<{ recordError?: string }>(); + const navigate = useNavigate(); + const params = useParams({ strict: false }); + const location = useLocation(); + const locationState = location.state as DomainState; const { enqueueSnackbar } = useSnackbar(); const { data: profile } = useProfile(); - const pagination = usePagination(1, PREFERENCE_KEY); - - const { handleOrderChange, order, orderBy } = useOrder( - { - order: 'asc', - orderBy: 'domain', + const pagination = usePaginationV2({ + currentRoute: '/domains', + preferenceKey: DOMAINS_TABLE_PREFERENCE_KEY, + }); + + const { handleOrderChange, order, orderBy } = useOrderV2({ + initialRoute: { + defaultOrder: { + order: DOMAINS_TABLE_DEFAULT_ORDER, + orderBy: DOMAINS_TABLE_DEFAULT_ORDER_BY, + }, + from: '/domains', }, - `${PREFERENCE_KEY}-order` - ); + preferenceKey: `${DOMAINS_TABLE_PREFERENCE_KEY}-order`, + }); const filter = { ['+order']: order, @@ -87,15 +100,12 @@ export const DomainsLanding = (props: DomainsLandingProps) => { const { domainForEditing } = props; - const [importDrawerOpen, setImportDrawerOpen] = React.useState(false); - const [removeDialogOpen, setRemoveDialogOpen] = React.useState(false); - const [disableDialogOpen, setDisableDialogOpen] = React.useState(false); - const [cloneDialogOpen, setCloneDialogOpen] = React.useState(false); - const [editDialogOpen, setEditDialogOpen] = React.useState(false); - - const [selectedDomain, setSelectedDomain] = React.useState< - Domain | undefined - >(); + const { data: selectedDomain, isFetching: isFetchingDomain } = useDialogData({ + enabled: !!params.domainId, + paramKey: 'domainId', + queryHook: useDomainQuery, + redirectToOnNotFound: '/domains', + }); const { error: deleteError, @@ -105,51 +115,67 @@ export const DomainsLanding = (props: DomainsLandingProps) => { const { mutateAsync: updateDomain } = useUpdateDomainMutation(); - const onClone = (domain: Domain) => { - setSelectedDomain(domain); - setCloneDialogOpen(true); + const navigateToDomains = () => { + navigate({ + search: (prev) => prev, + to: '/domains', + }); + }; + + const handleClone = (domain: Domain) => { + navigate({ + params: { action: 'clone', domainId: domain.id }, + search: (prev) => prev, + to: `/domains/$domainId/$action`, + }); }; - const onEdit = (domain: Domain) => { - setSelectedDomain(domain); - setEditDialogOpen(true); + const handleEdit = (domain: Domain) => { + navigate({ + params: { action: 'edit', domainId: domain.id }, + search: (prev) => prev, + to: `/domains/$domainId/$action`, + }); }; React.useEffect(() => { // Open the "Edit Domain" drawer if so specified by this component's props. if (domainForEditing) { - onEdit(domainForEditing); + handleEdit(domainForEditing); } }, [domainForEditing]); const navigateToCreate = () => { - history.push(DOMAIN_CREATE_ROUTE); - }; - - const openImportZoneDrawer = () => { - setImportDrawerOpen(true); - }; - - const closeImportZoneDrawer = () => { - setImportDrawerOpen(false); + navigate({ + to: DOMAIN_CREATE_ROUTE, + }); }; - const onRemove = (domain: Domain) => { - setSelectedDomain(domain); - setRemoveDialogOpen(true); + const handleImport = () => { + navigate({ + search: (prev) => prev, + to: `/domains/import`, + }); }; - const closeRemoveDialog = () => { - setRemoveDialogOpen(false); + const handleDelete = (domain: Domain) => { + navigate({ + params: { action: 'delete', domainId: domain.id }, + search: (prev) => prev, + to: `/domains/$domainId/$action`, + }); }; const removeDomain = () => { deleteDomain().then(() => { - closeRemoveDialog(); + navigateToDomains(); }); }; - const onDisableOrEnable = (action: 'disable' | 'enable', domain: Domain) => { + const handleDisableOrEnable = ( + action: 'disable' | 'enable', + domain: Domain + ) => { if (action === 'enable') { updateDomain({ id: domain.id, @@ -164,16 +190,19 @@ export const DomainsLanding = (props: DomainsLandingProps) => { ); }); } else { - setSelectedDomain(domain); - setDisableDialogOpen(true); + navigate({ + params: { action: 'disable', domainId: domain.id }, + search: (prev) => prev, + to: `/domains/$domainId/$action`, + }); } }; const handlers: DomainHandlers = { - onClone, - onDisableOrEnable, - onEdit, - onRemove, + onClone: handleClone, + onDisableOrEnable: handleDisableOrEnable, + onEdit: handleEdit, + onRemove: handleDelete, }; if (isLoading) { @@ -192,11 +221,11 @@ export const DomainsLanding = (props: DomainsLandingProps) => { ); @@ -223,12 +252,16 @@ export const DomainsLanding = (props: DomainsLandingProps) => { <>