2023-07-21 01:12:01 -05:00
|
|
|
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(
|
2023-09-03 05:52:45 -05:00
|
|
|
(pathname: string, exact = false) => {
|
|
|
|
// Match relative pathnames directly
|
|
|
|
if (pathname.startsWith('.')) {
|
|
|
|
return (
|
|
|
|
matchPath(joinPath(location.pathname, pathname, exact ? '' : '*'), location.pathname) !==
|
|
|
|
null
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Match absolute pathnames with the tenant segment
|
|
|
|
return (
|
|
|
|
matchPath(joinPath(':tenantId', pathname, exact ? '' : '*'), location.pathname) !== null
|
|
|
|
);
|
|
|
|
},
|
2023-07-21 01:12:01 -05:00
|
|
|
[location.pathname]
|
|
|
|
);
|
|
|
|
|
|
|
|
/** Returns the pathname with the current tenant ID prepended. */
|
|
|
|
const getPathname = useCallback(
|
|
|
|
(pathname: string) => {
|
2023-08-17 03:18:33 -05:00
|
|
|
if (pathname.startsWith('/') && !pathname.startsWith(`/${tenantSegment}`)) {
|
2023-07-21 01:12:01 -05:00
|
|
|
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(
|
2023-07-23 22:39:55 -05:00
|
|
|
(pathname = '/') => appendPath(new URL(window.location.origin), href, tenantSegment, pathname),
|
2023-07-21 01:12:01 -05:00
|
|
|
[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;
|