0
Fork 0
mirror of https://github.com/logto-io/logto.git synced 2025-01-20 21:32:31 -05:00
logto/packages/console/src/hooks/use-tenant-pathname.ts
Gao Sun 4c9aef827b
refactor(console): use global router (#4191)
* refactor(console): use global router

* refactor(console): clean swr cache when switching tenant

* refactor(console): bug fix

* chore(test): increase integration test timeout

* refactor: update packages/console/src/App.tsx

Co-authored-by: Charles Zhao <charleszhao@silverhand.io>

---------

Co-authored-by: Charles Zhao <charleszhao@silverhand.io>
2023-07-21 06:12:01 +00:00

117 lines
3.2 KiB
TypeScript

import { ossConsolePath } from '@logto/schemas';
import { appendPath, joinPath } from '@silverhand/essentials';
import { useCallback, useContext, useMemo } from 'react';
import {
type NavigateOptions,
type To,
matchPath,
useLocation,
useNavigate,
useHref,
} from 'react-router-dom';
import { isCloud } from '@/consts/env';
import { TenantsContext } from '@/contexts/TenantsProvider';
type TenantPathname = {
/**
* A function that can be used to match a specific tenant pathname with
* the current location.
*
* @example
* ```ts
* // Current location: `/tenant-1/console`
* const match = useMatchTenantPath();
* match('/console'); // true
* match('/tenant-1/console'); // false
* match('/another-page'); // false
* ```
*
* @param pathname Pathname to match
* @param exact Whether to match exactly, defaults to `false`
*/
match: (pathname: string, exact?: boolean) => boolean;
/**
* Returns the pathname with the current tenant ID prepended if the pathname
* is an absolute pathname; otherwise, returns the pathname directly.
*/
getPathname: (pathname: string) => string;
/**
* Returns the `to` object with the current tenant ID prepended if the
* pathname is an absolute pathname; otherwise, returns the `to` object
* or the string directly.
*/
getTo: (to: To) => To;
/** Navigate to the given pathname in the current tenant. */
navigate: (to: To, options?: NavigateOptions) => void;
/** Returns the full URL with the current tenant ID prepended. */
getUrl: (pathname: string) => URL;
};
/**
* Returns a `TenantPathname` object that contains utilities for matching
* and generating tenant-specific pathnames.
*
* @see {@link TenantPathname}
*/
function useTenantPathname(): TenantPathname {
const location = useLocation();
const { currentTenantId } = useContext(TenantsContext);
const tenantSegment = useMemo(
() => (isCloud ? currentTenantId : ossConsolePath.slice(1)),
[currentTenantId]
);
const navigate = useNavigate();
const href = useHref('/');
const match = useCallback(
(pathname: string, exact = false) =>
matchPath(joinPath(':tenantId', pathname, exact ? '' : '*'), location.pathname) !== null,
[location.pathname]
);
/** Returns the pathname with the current tenant ID prepended. */
const getPathname = useCallback(
(pathname: string) => {
if (pathname.startsWith('/')) {
return joinPath(tenantSegment, pathname);
}
// Directly return the pathname if it's a relative pathname
return pathname;
},
[tenantSegment]
);
const getTo = useCallback(
(to: To): To => {
if (typeof to === 'string') {
return getPathname(to);
}
return { ...to, pathname: getPathname(to.pathname ?? '') };
},
[getPathname]
);
const getUrl = useCallback(
(pathname = '/') => {
return appendPath(new URL(href), tenantSegment, pathname);
},
[href, tenantSegment]
);
const data = useMemo(
() => ({
match,
navigate: (to: To, options?: NavigateOptions) => {
navigate(getTo(to), options);
},
getPathname,
getTo,
getUrl,
}),
[match, getPathname, getTo, navigate, getUrl]
);
return data;
}
export default useTenantPathname;