diff --git a/packages/console/src/ds-components/Table/index.module.scss b/packages/console/src/ds-components/Table/index.module.scss
index fc8f3f123..0cd576d91 100644
--- a/packages/console/src/ds-components/Table/index.module.scss
+++ b/packages/console/src/ds-components/Table/index.module.scss
@@ -34,7 +34,6 @@
.headerTable {
background-color: var(--color-layer-1);
border-radius: 12px 12px 0 0;
- padding: 0 _.unit(3);
thead {
tr {
@@ -55,7 +54,6 @@
.bodyTable {
overflow-y: auto;
- padding: 0 _.unit(3) _.unit(3);
background-color: var(--color-layer-1);
border-radius: 0 0 12px 12px;
@@ -165,6 +163,12 @@
}
}
+.footer {
+ display: flex;
+ justify-content: space-between;
+ align-items: center;
+}
+
.pagination {
margin-top: _.unit(4);
}
diff --git a/packages/console/src/ds-components/Table/index.tsx b/packages/console/src/ds-components/Table/index.tsx
index b03f6e6f6..5105a839b 100644
--- a/packages/console/src/ds-components/Table/index.tsx
+++ b/packages/console/src/ds-components/Table/index.tsx
@@ -37,6 +37,14 @@ export type Props<
errorMessage?: string;
hasBorder?: boolean;
onRetry?: () => void;
+ /**
+ * The padding of the table container in px, excluding top padding.
+ *
+ * @default 12
+ */
+ padding?: number;
+ /** A footer that will be rendered on the bottom-left of the table. */
+ footer?: ReactNode;
};
function Table<
@@ -61,6 +69,8 @@ function Table<
errorMessage,
hasBorder,
onRetry,
+ padding = 12,
+ footer,
}: Props
) {
const totalColspan = columns.reduce((result, { colSpan }) => {
return result + (colSpan ?? 1);
@@ -85,6 +95,7 @@ function Table<
filter && styles.hideTopBorderRadius,
headerTableClassName
)}
+ style={{ padding: `0 ${padding}px` }}
>
@@ -102,6 +113,7 @@ function Table<
isEmpty && styles.empty,
bodyTableWrapperClassName
)}
+ style={{ padding: `0 ${padding}px ${padding}px` }}
>
@@ -160,7 +172,10 @@ function Table<
- {pagination && }
+
+ {footer}
+ {pagination &&
}
+
);
}
diff --git a/packages/console/src/ds-components/Tag/index.module.scss b/packages/console/src/ds-components/Tag/index.module.scss
index 8627563ee..e5a6063dd 100644
--- a/packages/console/src/ds-components/Tag/index.module.scss
+++ b/packages/console/src/ds-components/Tag/index.module.scss
@@ -21,6 +21,16 @@
color: var(--color-on-success-container);
}
+ // Distinguish from the info status
+ &.cell {
+ font-family: 'Roboto Mono', monospace;
+ font-size: 14px;
+ line-height: 20px;
+ border-radius: 6px;
+ background: var(--color-neutral-variant-90);
+ padding: _.unit(0.5) _.unit(2);
+ }
+
&.info {
.icon {
background: var(--color-on-info-container);
diff --git a/packages/console/src/ds-components/Tag/index.tsx b/packages/console/src/ds-components/Tag/index.tsx
index 6b429a114..200760132 100644
--- a/packages/console/src/ds-components/Tag/index.tsx
+++ b/packages/console/src/ds-components/Tag/index.tsx
@@ -1,17 +1,16 @@
import { conditional } from '@silverhand/essentials';
import classNames from 'classnames';
-import type { ReactNode } from 'react';
+import type { HTMLProps, ReactNode } from 'react';
import Failed from '@/assets/icons/failed.svg';
import Success from '@/assets/icons/success.svg';
import * as styles from './index.module.scss';
-export type Props = {
+export type Props = Pick, 'className' | 'onClick'> & {
type?: 'property' | 'state' | 'result';
status?: 'info' | 'success' | 'alert' | 'error';
- variant?: 'plain' | 'outlined';
- className?: string;
+ variant?: 'plain' | 'outlined' | 'cell';
children: ReactNode;
};
@@ -26,14 +25,15 @@ function Tag({
variant = 'outlined',
className,
children,
+ ...rest
}: Props) {
const ResultIcon = conditional(type === 'result' && ResultIconMap[status]);
return (
-
+
{type === 'state' &&
}
{ResultIcon &&
}
-
{children}
+ {children}
);
}
diff --git a/packages/console/src/pages/Connectors/ConnectorDeleteButton/index.tsx b/packages/console/src/pages/Connectors/ConnectorDeleteButton/index.tsx
index 7c4b1c3d1..4e1f62cbe 100644
--- a/packages/console/src/pages/Connectors/ConnectorDeleteButton/index.tsx
+++ b/packages/console/src/pages/Connectors/ConnectorDeleteButton/index.tsx
@@ -1,4 +1,3 @@
-import type { ConnectorResponse } from '@logto/schemas';
import { useState } from 'react';
import { toast } from 'react-hot-toast';
import { useTranslation } from 'react-i18next';
@@ -41,7 +40,7 @@ function ConnectorDeleteButton({ connectorGroup }: Props) {
try {
await Promise.all(
connectors.map(async (connector) => {
- await api.delete(`api/connectors/${connector.id}`).json
();
+ await api.delete(`api/connectors/${connector.id}`);
})
);
diff --git a/packages/console/src/pages/Organizations/CreatePermissionModal/index.tsx b/packages/console/src/pages/Organizations/CreatePermissionModal/index.tsx
new file mode 100644
index 000000000..a6b31f1ed
--- /dev/null
+++ b/packages/console/src/pages/Organizations/CreatePermissionModal/index.tsx
@@ -0,0 +1,86 @@
+import { useEffect, useState } from 'react';
+import { useForm } from 'react-hook-form';
+import { useTranslation } from 'react-i18next';
+import ReactModal from 'react-modal';
+
+import Button from '@/ds-components/Button';
+import FormField from '@/ds-components/FormField';
+import ModalLayout from '@/ds-components/ModalLayout';
+import TextInput from '@/ds-components/TextInput';
+import useApi from '@/hooks/use-api';
+import * as modalStyles from '@/scss/modal.module.scss';
+
+type Props = {
+ isOpen: boolean;
+ onFinish: () => void;
+};
+
+function CreatePermissionModal({ isOpen, onFinish }: Props) {
+ const api = useApi();
+ const [isLoading, setIsLoading] = useState(false);
+ const {
+ reset,
+ register,
+ handleSubmit,
+ formState: { errors },
+ } = useForm<{ name: string; description?: string }>({ defaultValues: { name: '' } });
+ const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
+
+ const addPermission = handleSubmit(async (json) => {
+ setIsLoading(true);
+ try {
+ await api.post('api/organization-scopes', {
+ json,
+ });
+ onFinish();
+ } finally {
+ setIsLoading(false);
+ }
+ });
+
+ useEffect(() => {
+ if (!isOpen) {
+ reset();
+ }
+ }, [isOpen, reset]);
+
+ return (
+
+
+ }
+ onClose={onFinish}
+ >
+
+
+
+
+
+
+
+
+ );
+}
+export default CreatePermissionModal;
diff --git a/packages/console/src/pages/Organizations/PermissionsField/index.tsx b/packages/console/src/pages/Organizations/PermissionsField/index.tsx
new file mode 100644
index 000000000..723893401
--- /dev/null
+++ b/packages/console/src/pages/Organizations/PermissionsField/index.tsx
@@ -0,0 +1,93 @@
+import { type OrganizationScope } from '@logto/schemas';
+import { useState } from 'react';
+import { useTranslation } from 'react-i18next';
+import useSWR from 'swr';
+
+import DeleteButton from '@/components/DeleteButton';
+import FormField from '@/ds-components/FormField';
+import useApi, { type RequestError } from '@/hooks/use-api';
+import { buildUrl } from '@/utils/url';
+
+import CreatePermissionModal from '../CreatePermissionModal';
+import TemplateTable, { pageSize } from '../TemplateTable';
+import * as styles from '../index.module.scss';
+
+/**
+ * Renders the permissions field that allows users to add, edit, and delete organization
+ * permissions.
+ */
+function PermissionsField() {
+ const [page, setPage] = useState(1);
+ const {
+ data: response,
+ error,
+ mutate,
+ } = useSWR<[OrganizationScope[], number], RequestError>(
+ buildUrl('api/organization-scopes', {
+ page: String(page),
+ page_size: String(pageSize),
+ })
+ );
+
+ const [data, totalCount] = response ?? [[], 0];
+ const api = useApi();
+ const [isModalOpen, setIsModalOpen] = useState(false);
+ const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
+
+ const isLoading = !response && !error;
+
+ if (isLoading) {
+ return <>loading>; // TODO: loading state
+ }
+
+ return (
+
+ {
+ setIsModalOpen(false);
+ void mutate();
+ }}
+ />
+ {name}
,
+ },
+ {
+ title: t('general.description'),
+ dataIndex: 'description',
+ colSpan: 6,
+ render: ({ description }) => description ?? '-',
+ },
+ {
+ title: null,
+ dataIndex: 'delete',
+ render: ({ id }) => (
+ {
+ await api.delete(`api/organization-scopes/${id}`);
+ void mutate();
+ }}
+ />
+ ),
+ },
+ ]}
+ onPageChange={setPage}
+ onAdd={() => {
+ setIsModalOpen(true);
+ }}
+ />
+
+ );
+}
+
+export default PermissionsField;
diff --git a/packages/console/src/pages/Organizations/Settings/index.tsx b/packages/console/src/pages/Organizations/Settings/index.tsx
new file mode 100644
index 000000000..5cade4e8e
--- /dev/null
+++ b/packages/console/src/pages/Organizations/Settings/index.tsx
@@ -0,0 +1,14 @@
+import FormCard from '@/components/FormCard';
+
+import PermissionsField from '../PermissionsField';
+
+export default function Settings() {
+ return (
+
+
+
+ );
+}
diff --git a/packages/console/src/pages/Organizations/TemplateTable/index.module.scss b/packages/console/src/pages/Organizations/TemplateTable/index.module.scss
new file mode 100644
index 000000000..c8cb91841
--- /dev/null
+++ b/packages/console/src/pages/Organizations/TemplateTable/index.module.scss
@@ -0,0 +1,9 @@
+@use '@/scss/underscore' as _;
+
+.addButton {
+ margin-top: _.unit(3);
+}
+
+.table {
+ margin-top: _.unit(3);
+}
diff --git a/packages/console/src/pages/Organizations/TemplateTable/index.tsx b/packages/console/src/pages/Organizations/TemplateTable/index.tsx
new file mode 100644
index 000000000..ce2cd5a75
--- /dev/null
+++ b/packages/console/src/pages/Organizations/TemplateTable/index.tsx
@@ -0,0 +1,86 @@
+import { type FieldValues, type FieldPath } from 'react-hook-form';
+
+import CirclePlus from '@/assets/icons/circle-plus.svg';
+import Plus from '@/assets/icons/plus.svg';
+import Button from '@/ds-components/Button';
+import Table from '@/ds-components/Table';
+import { type Column } from '@/ds-components/Table/types';
+
+import * as styles from './index.module.scss';
+
+type Props> = {
+ rowIndexKey: TName;
+ data: TFieldValues[];
+ columns: Array>;
+ totalCount: number;
+ page: number;
+ onPageChange: (page: number) => void;
+ onAdd?: () => void;
+};
+
+export const pageSize = 10;
+
+/**
+ * The table component for organization template editing, such as permissions and roles.
+ * If `onAdd` is provided, an add button will be rendered in the bottom.
+ */
+function TemplateTable<
+ TFieldValues extends FieldValues = FieldValues,
+ TName extends FieldPath = FieldPath,
+>({
+ rowIndexKey,
+ data,
+ columns,
+ onAdd,
+ totalCount,
+ page,
+ onPageChange,
+}: Props) {
+ const hasData = data.length > 0;
+
+ return (
+ <>
+ {hasData && (
+ }
+ title="general.create_another"
+ onClick={onAdd}
+ />
+ }
+ />
+ )}
+ {onAdd && !hasData && (
+ }
+ title="general.create"
+ onClick={onAdd}
+ />
+ )}
+ >
+ );
+}
+
+export default TemplateTable;
diff --git a/packages/console/src/pages/Organizations/index.module.scss b/packages/console/src/pages/Organizations/index.module.scss
new file mode 100644
index 000000000..d56af3c82
--- /dev/null
+++ b/packages/console/src/pages/Organizations/index.module.scss
@@ -0,0 +1,9 @@
+@use '@/scss/underscore' as _;
+
+.tabs {
+ margin: _.unit(4) 0;
+}
+
+.permission {
+ @include _.tag;
+}
diff --git a/packages/console/src/pages/Organizations/index.tsx b/packages/console/src/pages/Organizations/index.tsx
new file mode 100644
index 000000000..3d89019cb
--- /dev/null
+++ b/packages/console/src/pages/Organizations/index.tsx
@@ -0,0 +1,42 @@
+import { joinPath } from '@silverhand/essentials';
+import { useTranslation } from 'react-i18next';
+import { useParams } from 'react-router-dom';
+
+import CardTitle from '@/ds-components/CardTitle';
+import TabNav, { TabNavItem } from '@/ds-components/TabNav';
+import * as pageLayout from '@/scss/page-layout.module.scss';
+
+import Settings from './Settings';
+import * as styles from './index.module.scss';
+
+const pathnames = Object.freeze({
+ organizations: 'organizations',
+ settings: 'settings',
+});
+
+function Organizations() {
+ const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
+ const { tab } = useParams();
+
+ console.log('tab', tab);
+
+ return (
+
+
+
+
+
+
+ {t('organizations.title')}
+
+
+ {t('general.settings_nav')}
+
+
+ {!tab && <>Not found>}
+ {tab === pathnames.settings &&
}
+
+ );
+}
+
+export default Organizations;
diff --git a/packages/console/src/scss/_underscore.scss b/packages/console/src/scss/_underscore.scss
index 2471bc6e6..9ff87fff8 100644
--- a/packages/console/src/scss/_underscore.scss
+++ b/packages/console/src/scss/_underscore.scss
@@ -96,3 +96,14 @@
overflow: hidden;
text-overflow: ellipsis;
}
+
+/** Render a tag that has background color and border radius. */
+@mixin tag {
+ display: inline-block;
+ max-width: 100%;
+ vertical-align: bottom;
+ padding: unit(1) unit(2);
+ border-radius: 6px;
+ background: var(--color-neutral-95);
+ @include text-ellipsis;
+}
diff --git a/packages/core/src/queries/organizations.ts b/packages/core/src/queries/organizations.ts
index b5b220110..b4863d4bb 100644
--- a/packages/core/src/queries/organizations.ts
+++ b/packages/core/src/queries/organizations.ts
@@ -88,9 +88,9 @@ export default class OrganizationQueries extends SchemaQueries<
Organization
> {
/** Queries for roles in the organization template. */
- roles = new SchemaQueries(this.pool, OrganizationRoles);
+ roles = new SchemaQueries(this.pool, OrganizationRoles, { field: 'name', order: 'asc' });
/** Queries for scopes in the organization template. */
- scopes = new SchemaQueries(this.pool, OrganizationScopes);
+ scopes = new SchemaQueries(this.pool, OrganizationScopes, { field: 'name', order: 'asc' });
/** Queries for relations that connected with organization-related entities. */
relations = {
diff --git a/packages/core/src/utils/SchemaQueries.ts b/packages/core/src/utils/SchemaQueries.ts
index 03b2e33b8..263182349 100644
--- a/packages/core/src/utils/SchemaQueries.ts
+++ b/packages/core/src/utils/SchemaQueries.ts
@@ -32,10 +32,11 @@ export default class SchemaQueries<
constructor(
public readonly pool: CommonQueryMethods,
- public readonly schema: GeneratedSchema
+ public readonly schema: GeneratedSchema,
+ orderBy?: { field: Key | 'id'; order: 'asc' | 'desc' }
) {
this.#findTotalNumber = buildGetTotalRowCountWithPool(this.pool, this.schema.table);
- this.#findAll = buildFindAllEntitiesWithPool(this.pool)(this.schema);
+ this.#findAll = buildFindAllEntitiesWithPool(this.pool)(this.schema, orderBy && [orderBy]);
this.#findById = buildFindEntityByIdWithPool(this.pool)(this.schema);
this.#insert = buildInsertIntoWithPool(this.pool)(this.schema, { returning: true });
this.#updateById = buildUpdateWhereWithPool(this.pool)(this.schema, true);