diff --git a/packages/console/src/hooks/use-console-routes/routes/organizations.tsx b/packages/console/src/hooks/use-console-routes/routes/organizations.tsx
index ae42dc987..2bb589026 100644
--- a/packages/console/src/hooks/use-console-routes/routes/organizations.tsx
+++ b/packages/console/src/hooks/use-console-routes/routes/organizations.tsx
@@ -1,8 +1,11 @@
import { condArray } from '@silverhand/essentials';
-import { type RouteObject } from 'react-router-dom';
+import { Navigate, type RouteObject } from 'react-router-dom';
import { isDevFeaturesEnabled } from '@/consts/env';
import OrganizationDetails from '@/pages/OrganizationDetails';
+import Members from '@/pages/OrganizationDetails/Members';
+import Settings from '@/pages/OrganizationDetails/Settings';
+import { OrganizationDetailsTabs } from '@/pages/OrganizationDetails/types';
import Organizations from '@/pages/Organizations';
export const organizations: RouteObject = {
@@ -14,6 +17,14 @@ export const organizations: RouteObject = {
path: 'template',
element: ,
},
- { path: ':id/*', element: }
+ {
+ path: ':id/*',
+ element: ,
+ children: [
+ { index: true, element: },
+ { path: OrganizationDetailsTabs.Settings, element: },
+ { path: OrganizationDetailsTabs.Members, element: },
+ ],
+ }
),
};
diff --git a/packages/console/src/pages/OrganizationDetails/Members/index.tsx b/packages/console/src/pages/OrganizationDetails/Members/index.tsx
index 537592bda..cfcdcb4fe 100644
--- a/packages/console/src/pages/OrganizationDetails/Members/index.tsx
+++ b/packages/console/src/pages/OrganizationDetails/Members/index.tsx
@@ -1,6 +1,7 @@
-import { type UserWithOrganizationRoles, type Organization } from '@logto/schemas';
+import { type UserWithOrganizationRoles } from '@logto/schemas';
import { useState } from 'react';
import { useTranslation } from 'react-i18next';
+import { useOutletContext } from 'react-router-dom';
import useSWR from 'swr';
import Plus from '@/assets/icons/plus.svg';
@@ -19,17 +20,16 @@ import useActionTranslation from '@/hooks/use-action-translation';
import useApi, { type RequestError } from '@/hooks/use-api';
import { buildUrl } from '@/utils/url';
+import { type OrganizationDetailsOutletContext } from '../types';
+
import AddMembersToOrganization from './AddMembersToOrganization';
import EditOrganizationRolesModal from './EditOrganizationRolesModal';
import * as styles from './index.module.scss';
const pageSize = defaultPageSize;
-type Props = {
- organization: Organization;
-};
-
-function Members({ organization }: Props) {
+function Members() {
+ const { data: organization } = useOutletContext();
const api = useApi();
const [keyword, setKeyword] = useState('');
const [page, setPage] = useState(1);
diff --git a/packages/console/src/pages/OrganizationDetails/Settings/index.tsx b/packages/console/src/pages/OrganizationDetails/Settings/index.tsx
index b393543b1..a1489161f 100644
--- a/packages/console/src/pages/OrganizationDetails/Settings/index.tsx
+++ b/packages/console/src/pages/OrganizationDetails/Settings/index.tsx
@@ -2,6 +2,7 @@ import { type Organization } from '@logto/schemas';
import { useForm } from 'react-hook-form';
import { toast } from 'react-hot-toast';
import { useTranslation } from 'react-i18next';
+import { useOutletContext } from 'react-router-dom';
import DetailsForm from '@/components/DetailsForm';
import FormCard from '@/components/FormCard';
@@ -11,17 +12,10 @@ import TextInput from '@/ds-components/TextInput';
import useApi from '@/hooks/use-api';
import { trySubmitSafe } from '@/utils/form';
-type Props = {
- /**
- * Whether the organization is being deleted, this is used to disable the unsaved
- * changes alert modal.
- */
- isDeleting: boolean;
- data: Organization;
- onUpdated: (data: Organization) => void;
-};
+import { type OrganizationDetailsOutletContext } from '../types';
-function Settings({ isDeleting, data, onUpdated }: Props) {
+function Settings() {
+ const { isDeleting, data, onUpdated } = useOutletContext();
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
const {
register,
diff --git a/packages/console/src/pages/OrganizationDetails/index.tsx b/packages/console/src/pages/OrganizationDetails/index.tsx
index f7fd6db6a..df01f929b 100644
--- a/packages/console/src/pages/OrganizationDetails/index.tsx
+++ b/packages/console/src/pages/OrganizationDetails/index.tsx
@@ -1,9 +1,7 @@
import { type Organization } from '@logto/schemas';
import { useCallback, useState } from 'react';
import { useTranslation } from 'react-i18next';
-// FIXME: @gao
-// eslint-disable-next-line no-restricted-imports
-import { Navigate, Route, Routes, useParams } from 'react-router-dom';
+import { Outlet, useParams } from 'react-router-dom';
import useSWR from 'swr';
import Delete from '@/assets/icons/delete.svg';
@@ -23,15 +21,10 @@ import useTenantPathname from '@/hooks/use-tenant-pathname';
import Introduction from '../Organizations/Guide/Introduction';
-import Members from './Members';
-import Settings from './Settings';
import * as styles from './index.module.scss';
+import { OrganizationDetailsTabs, type OrganizationDetailsOutletContext } from './types';
const pathname = '/organizations';
-const tabs = Object.freeze({
- settings: 'settings',
- members: 'members',
-});
function OrganizationDetails() {
const { id } = useParams();
@@ -111,27 +104,22 @@ function OrganizationDetails() {
{t('organization_details.delete_confirmation')}
-
+
{t('general.settings_nav')}
-
+
{t('organizations.members')}
-
- } />
- mutate(data)}
- />
- }
- />
- } />
-
+ mutate(data),
+ } satisfies OrganizationDetailsOutletContext
+ }
+ />
>
)}
diff --git a/packages/console/src/pages/OrganizationDetails/types.ts b/packages/console/src/pages/OrganizationDetails/types.ts
new file mode 100644
index 000000000..6eb9254cb
--- /dev/null
+++ b/packages/console/src/pages/OrganizationDetails/types.ts
@@ -0,0 +1,16 @@
+import { type Organization } from '@logto/schemas';
+
+export type OrganizationDetailsOutletContext = {
+ data: Organization;
+ /**
+ * Whether the organization is being deleted, this is used to disable the unsaved
+ * changes alert modal.
+ */
+ isDeleting: boolean;
+ onUpdated: (data: Organization) => void;
+};
+
+export enum OrganizationDetailsTabs {
+ Settings = 'settings',
+ Members = 'members',
+}