mirror of
https://github.com/logto-io/logto.git
synced 2025-03-31 22:51:25 -05:00
refactor(console): use function-declaration for named components (#3520)
* refactor(console): use function-declaration for named components Application Insights can only automatically track component names from function declaration. update convention to make tracking easier. * refactor(console): fix lint errors
This commit is contained in:
parent
a43e72d177
commit
540c41ff64
287 changed files with 1069 additions and 1028 deletions
|
@ -115,7 +115,16 @@
|
|||
"@mdx/components/*": "./src/mdx-components/$1"
|
||||
},
|
||||
"eslintConfig": {
|
||||
"extends": "@silverhand/react"
|
||||
"extends": "@silverhand/react",
|
||||
"rules": {
|
||||
"react/function-component-definition": [
|
||||
"error",
|
||||
{
|
||||
"namedComponents": "function-declaration",
|
||||
"unnamedComponents": "arrow-function"
|
||||
}
|
||||
]
|
||||
}
|
||||
},
|
||||
"stylelint": {
|
||||
"extends": "@silverhand/eslint-config-react/.stylelintrc"
|
||||
|
|
|
@ -28,7 +28,7 @@ import setTitle from './utils/set-title';
|
|||
void initI18n();
|
||||
setTitle();
|
||||
|
||||
const Content = () => {
|
||||
function Content() {
|
||||
const { tenants, isSettle, currentTenantId } = useContext(TenantsContext);
|
||||
|
||||
const resources = deduplicate(
|
||||
|
@ -78,14 +78,14 @@ const Content = () => {
|
|||
</AppThemeProvider>
|
||||
</LogtoProvider>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
const App = () => {
|
||||
function App() {
|
||||
return (
|
||||
<TenantsProvider>
|
||||
<Content />
|
||||
</TenantsProvider>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default App;
|
||||
|
|
|
@ -7,7 +7,7 @@ import Main from './pages/Main';
|
|||
import SocialDemoCallback from './pages/SocialDemoCallback';
|
||||
import { CloudRoute } from './types';
|
||||
|
||||
const App = () => {
|
||||
function App() {
|
||||
return (
|
||||
<BrowserRouter>
|
||||
<div className={styles.app}>
|
||||
|
@ -20,6 +20,6 @@ const App = () => {
|
|||
</div>
|
||||
</BrowserRouter>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default App;
|
||||
|
|
|
@ -4,7 +4,7 @@ import { useNavigate } from 'react-router-dom';
|
|||
import AppLoading from '@/components/AppLoading';
|
||||
import { getUserTenantId } from '@/consts/tenants';
|
||||
|
||||
const Callback = () => {
|
||||
function Callback() {
|
||||
const navigate = useNavigate();
|
||||
const { error } = useHandleSignInCallback(() => {
|
||||
navigate('/' + getUserTenantId(), { replace: true });
|
||||
|
@ -21,6 +21,6 @@ const Callback = () => {
|
|||
}
|
||||
|
||||
return <AppLoading />;
|
||||
};
|
||||
}
|
||||
|
||||
export default Callback;
|
||||
|
|
|
@ -12,7 +12,7 @@ type Props = {
|
|||
toTenantId: string;
|
||||
};
|
||||
|
||||
const Redirect = ({ tenants, toTenantId }: Props) => {
|
||||
function Redirect({ tenants, toTenantId }: Props) {
|
||||
const { getAccessToken, signIn } = useLogto();
|
||||
const tenant = tenants.find(({ id }) => id === toTenantId);
|
||||
const { setIsSettle } = useContext(TenantsContext);
|
||||
|
@ -40,6 +40,6 @@ const Redirect = ({ tenants, toTenantId }: Props) => {
|
|||
}
|
||||
|
||||
return <AppLoading />;
|
||||
};
|
||||
}
|
||||
|
||||
export default Redirect;
|
||||
|
|
|
@ -14,7 +14,7 @@ type Props = {
|
|||
onAdd: (tenant: TenantInfo) => void;
|
||||
};
|
||||
|
||||
const Tenants = ({ data, onAdd }: Props) => {
|
||||
function Tenants({ data, onAdd }: Props) {
|
||||
const api = useCloudApi();
|
||||
const { navigate } = useContext(TenantsContext);
|
||||
|
||||
|
@ -57,6 +57,6 @@ const Tenants = ({ data, onAdd }: Props) => {
|
|||
}
|
||||
|
||||
return <AppLoading />;
|
||||
};
|
||||
}
|
||||
|
||||
export default Tenants;
|
||||
|
|
|
@ -12,7 +12,7 @@ import { TenantsContext } from '@/contexts/TenantsProvider';
|
|||
import Redirect from './Redirect';
|
||||
import Tenants from './Tenants';
|
||||
|
||||
const Protected = () => {
|
||||
function Protected() {
|
||||
const api = useCloudApi();
|
||||
const { tenants, setTenants, currentTenantId } = useContext(TenantsContext);
|
||||
|
||||
|
@ -43,9 +43,9 @@ const Protected = () => {
|
|||
}
|
||||
|
||||
return <AppLoading />;
|
||||
};
|
||||
}
|
||||
|
||||
const Main = () => {
|
||||
function Main() {
|
||||
const [searchParameters] = useSearchParams();
|
||||
const { isAuthenticated, isLoading, signIn } = useLogto();
|
||||
const { currentTenantId } = useContext(TenantsContext);
|
||||
|
@ -66,6 +66,6 @@ const Main = () => {
|
|||
}
|
||||
|
||||
return <Protected />;
|
||||
};
|
||||
}
|
||||
|
||||
export default Main;
|
||||
|
|
|
@ -8,7 +8,7 @@ import useTheme from '@/hooks/use-theme';
|
|||
|
||||
import * as styles from './index.module.scss';
|
||||
|
||||
const SocialDemoCallback = () => {
|
||||
function SocialDemoCallback() {
|
||||
const theme = useTheme();
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
|
||||
|
||||
|
@ -21,6 +21,6 @@ const SocialDemoCallback = () => {
|
|||
</Card>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default SocialDemoCallback;
|
||||
|
|
|
@ -20,14 +20,14 @@ type Props = {
|
|||
isDropdownFullWidth?: boolean;
|
||||
};
|
||||
|
||||
const ActionMenu = ({
|
||||
function ActionMenu({
|
||||
children,
|
||||
buttonProps,
|
||||
title,
|
||||
dropdownHorizontalAlign,
|
||||
dropdownClassName,
|
||||
isDropdownFullWidth = false,
|
||||
}: Props) => {
|
||||
}: Props) {
|
||||
const [isOpen, setIsOpen] = useState(false);
|
||||
const anchorReference = useRef<HTMLDivElement>(null);
|
||||
|
||||
|
@ -56,6 +56,6 @@ const ActionMenu = ({
|
|||
</Dropdown>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default ActionMenu;
|
||||
|
|
|
@ -19,7 +19,7 @@ type Props = {
|
|||
className?: string;
|
||||
};
|
||||
|
||||
const Alert = ({
|
||||
function Alert({
|
||||
children,
|
||||
action,
|
||||
href,
|
||||
|
@ -27,7 +27,7 @@ const Alert = ({
|
|||
severity = 'info',
|
||||
variant = 'plain',
|
||||
className,
|
||||
}: Props) => {
|
||||
}: Props) {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
|
||||
|
||||
return (
|
||||
|
@ -44,6 +44,6 @@ const Alert = ({
|
|||
)}
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default Alert;
|
||||
|
|
|
@ -19,7 +19,7 @@ type Props = {
|
|||
children?: React.ReactNode;
|
||||
};
|
||||
|
||||
const AppError = ({ title, errorCode, errorMessage, callStack, children }: Props) => {
|
||||
function AppError({ title, errorCode, errorMessage, callStack, children }: Props) {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
|
||||
const [isDetailsOpen, setIsDetailsOpen] = useState(false);
|
||||
const theme = useTheme();
|
||||
|
@ -55,6 +55,6 @@ const AppError = ({ title, errorCode, errorMessage, callStack, children }: Props
|
|||
{children}
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default AppError;
|
||||
|
|
|
@ -7,7 +7,7 @@ import useTheme from '@/hooks/use-theme';
|
|||
|
||||
import * as styles from './index.module.scss';
|
||||
|
||||
const AppLoading = () => {
|
||||
function AppLoading() {
|
||||
const theme = useTheme();
|
||||
|
||||
return (
|
||||
|
@ -16,6 +16,6 @@ const AppLoading = () => {
|
|||
<Spinner />
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default AppLoading;
|
||||
|
|
|
@ -9,12 +9,12 @@ type Props = {
|
|||
className?: string;
|
||||
};
|
||||
|
||||
const ApplicationIcon = ({ type, className }: Props) => {
|
||||
function ApplicationIcon({ type, className }: Props) {
|
||||
const theme = useTheme();
|
||||
const isLightMode = theme === Theme.Light;
|
||||
const Icon = isLightMode ? lightModeApplicationIconMap[type] : darkModeApplicationIconMap[type];
|
||||
|
||||
return <Icon className={className} />;
|
||||
};
|
||||
}
|
||||
|
||||
export default ApplicationIcon;
|
||||
|
|
|
@ -11,7 +11,7 @@ type Props = {
|
|||
isLink?: boolean;
|
||||
};
|
||||
|
||||
const ApplicationName = ({ applicationId, isLink = false }: Props) => {
|
||||
function ApplicationName({ applicationId, isLink = false }: Props) {
|
||||
const isAdminConsole = applicationId === adminConsoleApplicationId;
|
||||
|
||||
const { data } = useSWR<Application>(!isAdminConsole && `api/applications/${applicationId}`);
|
||||
|
@ -28,6 +28,6 @@ const ApplicationName = ({ applicationId, isLink = false }: Props) => {
|
|||
}
|
||||
|
||||
return <span>{name}</span>;
|
||||
};
|
||||
}
|
||||
|
||||
export default ApplicationName;
|
||||
|
|
|
@ -10,7 +10,7 @@ type Props = {
|
|||
onChange: (value?: string) => void;
|
||||
};
|
||||
|
||||
const ApplicationSelector = ({ value, onChange }: Props) => {
|
||||
function ApplicationSelector({ value, onChange }: Props) {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
|
||||
const { data } = useSWR<Application[]>('api/applications');
|
||||
const options =
|
||||
|
@ -28,6 +28,6 @@ const ApplicationSelector = ({ value, onChange }: Props) => {
|
|||
onChange={onChange}
|
||||
/>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default ApplicationSelector;
|
||||
|
|
|
@ -13,7 +13,7 @@ type Props = {
|
|||
to?: string;
|
||||
};
|
||||
|
||||
const EventName = ({ eventKey, isSuccess, to }: Props) => {
|
||||
function EventName({ eventKey, isSuccess, to }: Props) {
|
||||
const title = logEventTitle[eventKey] ?? eventKey;
|
||||
|
||||
return (
|
||||
|
@ -35,6 +35,6 @@ const EventName = ({ eventKey, isSuccess, to }: Props) => {
|
|||
{!to && <div className={styles.title}>{title}</div>}
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default EventName;
|
||||
|
|
|
@ -8,7 +8,7 @@ type Props = {
|
|||
onChange: (value?: string) => void;
|
||||
};
|
||||
|
||||
const EventSelector = ({ value, onChange }: Props) => {
|
||||
function EventSelector({ value, onChange }: Props) {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
|
||||
const options = Object.entries(logEventTitle).map(([value, title]) => ({
|
||||
value,
|
||||
|
@ -24,6 +24,6 @@ const EventSelector = ({ value, onChange }: Props) => {
|
|||
onChange={onChange}
|
||||
/>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default EventSelector;
|
||||
|
|
|
@ -25,7 +25,7 @@ type Props = {
|
|||
className?: string;
|
||||
};
|
||||
|
||||
const AuditLogTable = ({ userId, className }: Props) => {
|
||||
function AuditLogTable({ userId, className }: Props) {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
|
||||
const { pathname } = useLocation();
|
||||
const pageSize = defaultPageSize;
|
||||
|
@ -132,6 +132,6 @@ const AuditLogTable = ({ userId, className }: Props) => {
|
|||
onRetry={async () => mutate(undefined, true)}
|
||||
/>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default AuditLogTable;
|
||||
|
|
|
@ -39,7 +39,7 @@ type IconButtonProps = BaseProps & {
|
|||
|
||||
export type Props = TitleButtonProps | IconButtonProps;
|
||||
|
||||
const Button = ({
|
||||
function Button({
|
||||
htmlType = 'button',
|
||||
type = 'default',
|
||||
size = 'medium',
|
||||
|
@ -51,7 +51,7 @@ const Button = ({
|
|||
onClick,
|
||||
trailingIcon,
|
||||
...rest
|
||||
}: Props) => {
|
||||
}: Props) {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
|
||||
const [showSpinner, setShowSpinner] = useState(false);
|
||||
const timerRef = useRef<number>();
|
||||
|
@ -96,6 +96,6 @@ const Button = ({
|
|||
{trailingIcon && <span className={styles.trailingIcon}>{trailingIcon}</span>}
|
||||
</button>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default Button;
|
||||
|
|
|
@ -9,7 +9,7 @@ type Props = {
|
|||
className?: string;
|
||||
};
|
||||
|
||||
const Card = (props: Props, reference?: Ref<HTMLDivElement>) => {
|
||||
function Card(props: Props, reference?: Ref<HTMLDivElement>) {
|
||||
const { children, className } = props;
|
||||
|
||||
return (
|
||||
|
@ -17,6 +17,6 @@ const Card = (props: Props, reference?: Ref<HTMLDivElement>) => {
|
|||
{children}
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default forwardRef<HTMLDivElement, Props>(Card);
|
||||
|
|
|
@ -19,14 +19,14 @@ export type Props = {
|
|||
/**
|
||||
* Always use this component to render CardTitle, with built-in i18n support.
|
||||
*/
|
||||
const CardTitle = ({
|
||||
function CardTitle({
|
||||
title,
|
||||
subtitle,
|
||||
size = 'large',
|
||||
isWordWrapEnabled = false,
|
||||
learnMoreLink,
|
||||
className,
|
||||
}: Props) => {
|
||||
}: Props) {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
|
||||
|
||||
return (
|
||||
|
@ -46,6 +46,6 @@ const CardTitle = ({
|
|||
)}
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default CardTitle;
|
||||
|
|
|
@ -19,7 +19,7 @@ type Props = {
|
|||
tooltip?: ReactNode;
|
||||
};
|
||||
|
||||
const Checkbox = ({
|
||||
function Checkbox({
|
||||
checked,
|
||||
disabled = false,
|
||||
indeterminate,
|
||||
|
@ -27,7 +27,7 @@ const Checkbox = ({
|
|||
label,
|
||||
className,
|
||||
tooltip,
|
||||
}: Props) => {
|
||||
}: Props) {
|
||||
const [isIndeterminate, setIsIndeterminate] = useState(indeterminate);
|
||||
|
||||
useLayoutEffect(() => {
|
||||
|
@ -106,6 +106,6 @@ const Checkbox = ({
|
|||
</div>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default Checkbox;
|
||||
|
|
|
@ -21,7 +21,7 @@ type Props = {
|
|||
placeholder?: string;
|
||||
};
|
||||
|
||||
const CodeEditor = ({
|
||||
function CodeEditor({
|
||||
className,
|
||||
language,
|
||||
isReadonly = false,
|
||||
|
@ -31,7 +31,7 @@ const CodeEditor = ({
|
|||
hasError,
|
||||
errorMessage,
|
||||
placeholder,
|
||||
}: Props) => {
|
||||
}: Props) {
|
||||
const textareaRef = useRef<HTMLTextAreaElement>(null);
|
||||
const editorRef = useRef<HTMLDivElement>(null);
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
|
||||
|
@ -142,6 +142,6 @@ const CodeEditor = ({
|
|||
{hasError && <div className={styles.errorMessage}>{finalErrorMessage}</div>}
|
||||
</>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default CodeEditor;
|
||||
|
|
|
@ -9,7 +9,7 @@ type Props = {
|
|||
onChange?: (value: string) => void;
|
||||
};
|
||||
|
||||
const ColorPicker = ({ onChange, value = '#000000' }: Props) => {
|
||||
function ColorPicker({ onChange, value = '#000000' }: Props) {
|
||||
const [id, setId] = useState(nanoid());
|
||||
|
||||
const handleChange: ChangeEventHandler<HTMLInputElement> = (event) => {
|
||||
|
@ -22,5 +22,5 @@ const ColorPicker = ({ onChange, value = '#000000' }: Props) => {
|
|||
<label htmlFor={id}>{value.toUpperCase()}</label>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
export default ColorPicker;
|
||||
|
|
|
@ -24,7 +24,7 @@ export type ConfirmModalProps = {
|
|||
onConfirm?: () => void;
|
||||
};
|
||||
|
||||
const ConfirmModal = ({
|
||||
function ConfirmModal({
|
||||
children,
|
||||
className,
|
||||
title = 'general.reminder',
|
||||
|
@ -36,7 +36,7 @@ const ConfirmModal = ({
|
|||
isLoading = false,
|
||||
onCancel,
|
||||
onConfirm,
|
||||
}: ConfirmModalProps) => {
|
||||
}: ConfirmModalProps) {
|
||||
return (
|
||||
<ReactModal
|
||||
shouldCloseOnEsc
|
||||
|
@ -68,6 +68,6 @@ const ConfirmModal = ({
|
|||
</ModalLayout>
|
||||
</ReactModal>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default ConfirmModal;
|
||||
|
|
|
@ -13,7 +13,7 @@ type Props = {
|
|||
size?: 'small' | 'medium' | 'large';
|
||||
};
|
||||
|
||||
const ConnectorLogo = ({ className, data, size = 'medium' }: Props) => {
|
||||
function ConnectorLogo({ className, data, size = 'medium' }: Props) {
|
||||
const theme = useTheme();
|
||||
|
||||
return (
|
||||
|
@ -24,6 +24,6 @@ const ConnectorLogo = ({ className, data, size = 'medium' }: Props) => {
|
|||
src={theme === Theme.Dark && data.logoDark ? data.logoDark : data.logo}
|
||||
/>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default ConnectorLogo;
|
||||
|
|
|
@ -23,13 +23,13 @@ type Props = {
|
|||
|
||||
type CopyState = TFuncKey<'translation', 'admin_console.general'>;
|
||||
|
||||
const CopyToClipboard = ({
|
||||
function CopyToClipboard({
|
||||
value,
|
||||
className,
|
||||
hasVisibilityToggle,
|
||||
variant = 'contained',
|
||||
size = 'default',
|
||||
}: Props) => {
|
||||
}: Props) {
|
||||
const copyIconReference = useRef<HTMLButtonElement>(null);
|
||||
const [copyState, setCopyState] = useState<CopyState>('copy');
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console.general' });
|
||||
|
@ -101,6 +101,6 @@ const CopyToClipboard = ({
|
|||
</div>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default CopyToClipboard;
|
||||
|
|
|
@ -4,6 +4,8 @@ type Props = {
|
|||
children: ReactNode;
|
||||
};
|
||||
|
||||
const DangerousRaw = ({ children }: Props) => <span>{children}</span>;
|
||||
function DangerousRaw({ children }: Props) {
|
||||
return <span>{children}</span>;
|
||||
}
|
||||
|
||||
export default DangerousRaw;
|
||||
|
|
|
@ -5,7 +5,7 @@ type Props = {
|
|||
children: Nullable<string | number>;
|
||||
};
|
||||
|
||||
const DateTime = ({ children }: Props) => {
|
||||
function DateTime({ children }: Props) {
|
||||
const date = children && new Date(children);
|
||||
|
||||
if (!date || !isValid(date)) {
|
||||
|
@ -13,6 +13,6 @@ const DateTime = ({ children }: Props) => {
|
|||
}
|
||||
|
||||
return <span>{date.toLocaleDateString()}</span>;
|
||||
};
|
||||
}
|
||||
|
||||
export default DateTime;
|
||||
|
|
|
@ -15,7 +15,7 @@ type Props = {
|
|||
onConfirm: () => void;
|
||||
};
|
||||
|
||||
const DeleteConfirmModal = ({
|
||||
function DeleteConfirmModal({
|
||||
isOpen,
|
||||
isLoading = false,
|
||||
expectedInput,
|
||||
|
@ -24,7 +24,7 @@ const DeleteConfirmModal = ({
|
|||
className,
|
||||
onCancel,
|
||||
onConfirm,
|
||||
}: Props) => {
|
||||
}: Props) {
|
||||
const [input, setInput] = useState('');
|
||||
const isConfirmBlocked = Boolean(expectedInput) && input !== expectedInput;
|
||||
|
||||
|
@ -52,6 +52,6 @@ const DeleteConfirmModal = ({
|
|||
)}
|
||||
</ConfirmModal>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default DeleteConfirmModal;
|
||||
|
|
|
@ -13,14 +13,14 @@ type Props = {
|
|||
children: ReactNode;
|
||||
};
|
||||
|
||||
const DetailsForm = ({
|
||||
function DetailsForm({
|
||||
autoComplete,
|
||||
isDirty,
|
||||
isSubmitting,
|
||||
onSubmit,
|
||||
onDiscard,
|
||||
children,
|
||||
}: Props) => {
|
||||
}: Props) {
|
||||
return (
|
||||
<form
|
||||
className={classNames(styles.container, isDirty && styles.withSubmitActionBar)}
|
||||
|
@ -35,6 +35,6 @@ const DetailsForm = ({
|
|||
/>
|
||||
</form>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default DetailsForm;
|
||||
|
|
|
@ -6,36 +6,38 @@ import Spacer from '@/components/Spacer';
|
|||
|
||||
import * as styles from './index.module.scss';
|
||||
|
||||
const DetailsSkeleton = () => (
|
||||
<div className={styles.container}>
|
||||
<div className={styles.header}>
|
||||
<div className={styles.icon} />
|
||||
<div className={styles.wrapper}>
|
||||
<div className={styles.title} />
|
||||
<div className={styles.tags} />
|
||||
function DetailsSkeleton() {
|
||||
return (
|
||||
<div className={styles.container}>
|
||||
<div className={styles.header}>
|
||||
<div className={styles.icon} />
|
||||
<div className={styles.wrapper}>
|
||||
<div className={styles.title} />
|
||||
<div className={styles.tags} />
|
||||
</div>
|
||||
<Spacer />
|
||||
<div className={styles.button} />
|
||||
</div>
|
||||
<Spacer />
|
||||
<div className={styles.button} />
|
||||
</div>
|
||||
<div className={styles.tabBar} />
|
||||
<Card className={classNames(formCardStyles.container, styles.content)}>
|
||||
<div className={classNames(formCardStyles.introduction, styles.introduction)}>
|
||||
<div className={styles.title} />
|
||||
<div className={styles.description}>
|
||||
{Array.from({ length: 2 }).map((_, index) => (
|
||||
<div className={styles.tabBar} />
|
||||
<Card className={classNames(formCardStyles.container, styles.content)}>
|
||||
<div className={classNames(formCardStyles.introduction, styles.introduction)}>
|
||||
<div className={styles.title} />
|
||||
<div className={styles.description}>
|
||||
{Array.from({ length: 2 }).map((_, index) => (
|
||||
// eslint-disable-next-line react/no-array-index-key
|
||||
<div key={index} className={styles.text} />
|
||||
))}
|
||||
</div>
|
||||
</div>
|
||||
<div className={classNames(formCardStyles.form, styles.form)}>
|
||||
{Array.from({ length: 4 }).map((_, index) => (
|
||||
// eslint-disable-next-line react/no-array-index-key
|
||||
<div key={index} className={styles.text} />
|
||||
<div key={index} className={styles.field} />
|
||||
))}
|
||||
</div>
|
||||
</div>
|
||||
<div className={classNames(formCardStyles.form, styles.form)}>
|
||||
{Array.from({ length: 4 }).map((_, index) => (
|
||||
// eslint-disable-next-line react/no-array-index-key
|
||||
<div key={index} className={styles.field} />
|
||||
))}
|
||||
</div>
|
||||
</Card>
|
||||
</div>
|
||||
);
|
||||
</Card>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
export default DetailsSkeleton;
|
||||
|
|
|
@ -6,6 +6,8 @@ type Props = {
|
|||
className?: string;
|
||||
};
|
||||
|
||||
const Divider = ({ className }: Props) => <div className={classNames(styles.divider, className)} />;
|
||||
function Divider({ className }: Props) {
|
||||
return <div className={classNames(styles.divider, className)} />;
|
||||
}
|
||||
|
||||
export default Divider;
|
||||
|
|
|
@ -19,7 +19,7 @@ export const DragDropContext = createContext<DragDropContextProps>({
|
|||
isDragging: false,
|
||||
});
|
||||
|
||||
const DragDropProvider = ({ children }: Props) => {
|
||||
function DragDropProvider({ children }: Props) {
|
||||
const [isDragging, setIsDragging] = useState(false);
|
||||
|
||||
const memorizedContext = useMemo(() => ({ isDragging, setIsDragging }), [isDragging]);
|
||||
|
@ -31,6 +31,6 @@ const DragDropProvider = ({ children }: Props) => {
|
|||
</DndProvider>
|
||||
</DragDropContext.Provider>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default DragDropProvider;
|
||||
|
|
|
@ -21,14 +21,14 @@ type DragItemProps = {
|
|||
type: string;
|
||||
};
|
||||
|
||||
const DraggableItem = ({
|
||||
function DraggableItem({
|
||||
id,
|
||||
children,
|
||||
sortIndex,
|
||||
moveItem,
|
||||
dragType = 'DraggableItem',
|
||||
className,
|
||||
}: Props) => {
|
||||
}: Props) {
|
||||
const ref = useRef<HTMLDivElement>(null);
|
||||
const { setIsDragging } = useContext(DragDropContext);
|
||||
const [{ handlerId }, drop] = useDrop<DragItemProps, void, { handlerId: Nullable<Identifier> }>({
|
||||
|
@ -110,6 +110,6 @@ const DraggableItem = ({
|
|||
{children}
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default DraggableItem;
|
||||
|
|
|
@ -16,7 +16,7 @@ type Props = {
|
|||
onClose?: () => void;
|
||||
};
|
||||
|
||||
const Drawer = ({ title, subtitle, isOpen, children, onClose }: Props) => {
|
||||
function Drawer({ title, subtitle, isOpen, children, onClose }: Props) {
|
||||
return (
|
||||
<ReactModal
|
||||
shouldCloseOnOverlayClick
|
||||
|
@ -43,6 +43,6 @@ const Drawer = ({ title, subtitle, isOpen, children, onClose }: Props) => {
|
|||
</div>
|
||||
</ReactModal>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default Drawer;
|
||||
|
|
|
@ -14,24 +14,26 @@ type Props = {
|
|||
type?: 'default' | 'danger';
|
||||
};
|
||||
|
||||
const DropdownItem = ({
|
||||
function DropdownItem({
|
||||
onClick,
|
||||
className,
|
||||
children,
|
||||
icon,
|
||||
iconClassName,
|
||||
type = 'default',
|
||||
}: Props) => (
|
||||
<div
|
||||
className={classNames(styles.item, styles[type], className)}
|
||||
role="menuitem"
|
||||
tabIndex={0}
|
||||
onKeyDown={onKeyDownHandler(onClick)}
|
||||
onClick={onClick}
|
||||
>
|
||||
{icon && <span className={classNames(styles.icon, iconClassName)}>{icon}</span>}
|
||||
{children}
|
||||
</div>
|
||||
);
|
||||
}: Props) {
|
||||
return (
|
||||
<div
|
||||
className={classNames(styles.item, styles[type], className)}
|
||||
role="menuitem"
|
||||
tabIndex={0}
|
||||
onKeyDown={onKeyDownHandler(onClick)}
|
||||
onClick={onClick}
|
||||
>
|
||||
{icon && <span className={classNames(styles.icon, iconClassName)}>{icon}</span>}
|
||||
{children}
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
export default DropdownItem;
|
||||
|
|
|
@ -31,14 +31,14 @@ type Props = {
|
|||
hasOverflowContent?: boolean;
|
||||
};
|
||||
|
||||
const Div = ({
|
||||
function Div({
|
||||
children,
|
||||
...rest
|
||||
}: PropsWithChildren<DetailedHTMLProps<HTMLAttributes<HTMLDivElement>, HTMLDivElement>>) => (
|
||||
<div {...rest}>{children}</div>
|
||||
);
|
||||
}: PropsWithChildren<DetailedHTMLProps<HTMLAttributes<HTMLDivElement>, HTMLDivElement>>) {
|
||||
return <div {...rest}>{children}</div>;
|
||||
}
|
||||
|
||||
const Dropdown = ({
|
||||
function Dropdown({
|
||||
children,
|
||||
title,
|
||||
isOpen,
|
||||
|
@ -49,7 +49,7 @@ const Dropdown = ({
|
|||
titleClassName,
|
||||
horizontalAlign = 'end',
|
||||
hasOverflowContent,
|
||||
}: Props) => {
|
||||
}: Props) {
|
||||
const overlayRef = useRef<HTMLDivElement>(null);
|
||||
|
||||
const { position, positionState, mutate } = usePosition({
|
||||
|
@ -95,6 +95,6 @@ const Dropdown = ({
|
|||
</div>
|
||||
</ReactModal>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default Dropdown;
|
||||
|
|
|
@ -13,7 +13,7 @@ export type Props = {
|
|||
size?: 'large' | 'medium' | 'small';
|
||||
};
|
||||
|
||||
const EmptyDataPlaceholder = ({ title, size = 'medium' }: Props) => {
|
||||
function EmptyDataPlaceholder({ title, size = 'medium' }: Props) {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
|
||||
const theme = useTheme();
|
||||
const EmptyImage = theme === Theme.Light ? Empty : EmptyDark;
|
||||
|
@ -24,6 +24,6 @@ const EmptyDataPlaceholder = ({ title, size = 'medium' }: Props) => {
|
|||
<div className={styles.title}>{title ?? t('errors.empty')}</div>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default EmptyDataPlaceholder;
|
||||
|
|
|
@ -13,7 +13,7 @@ type Props = {
|
|||
children: ReactNode;
|
||||
};
|
||||
|
||||
const FormCard = ({ title, description, learnMoreLink, children }: Props) => {
|
||||
function FormCard({ title, description, learnMoreLink, children }: Props) {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
|
||||
|
||||
return (
|
||||
|
@ -37,6 +37,6 @@ const FormCard = ({ title, description, learnMoreLink, children }: Props) => {
|
|||
<div className={styles.form}>{children}</div>
|
||||
</Card>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default FormCard;
|
||||
|
|
|
@ -22,7 +22,7 @@ export type Props = {
|
|||
tip?: ToggleTipProps['content'];
|
||||
};
|
||||
|
||||
const FormField = ({
|
||||
function FormField({
|
||||
title,
|
||||
children,
|
||||
isRequired,
|
||||
|
@ -30,7 +30,7 @@ const FormField = ({
|
|||
className,
|
||||
tip,
|
||||
headlineClassName,
|
||||
}: Props) => {
|
||||
}: Props) {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
|
||||
|
||||
return (
|
||||
|
@ -55,6 +55,6 @@ const FormField = ({
|
|||
{children}
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default FormField;
|
||||
|
|
|
@ -9,10 +9,10 @@ export type Props = Omit<HTMLProps<HTMLButtonElement>, 'size' | 'type'> & {
|
|||
iconClassName?: string;
|
||||
};
|
||||
|
||||
const IconButton = (
|
||||
function IconButton(
|
||||
{ size = 'medium', children, className, iconClassName, ...rest }: Props,
|
||||
reference: ForwardedRef<HTMLButtonElement>
|
||||
) => {
|
||||
) {
|
||||
const tipRef = useRef<HTMLDivElement>(null);
|
||||
|
||||
return (
|
||||
|
@ -27,6 +27,6 @@ const IconButton = (
|
|||
</div>
|
||||
</button>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default forwardRef(IconButton);
|
||||
|
|
|
@ -11,14 +11,14 @@ type Props = {
|
|||
fallbackElement?: ReactElement;
|
||||
} & ImgHTMLAttributes<HTMLImageElement>;
|
||||
|
||||
const ImageWithErrorFallback = ({
|
||||
function ImageWithErrorFallback({
|
||||
src,
|
||||
alt,
|
||||
className,
|
||||
containerClassName,
|
||||
fallbackElement,
|
||||
...props
|
||||
}: Props) => {
|
||||
}: Props) {
|
||||
const [hasError, setHasError] = useState(false);
|
||||
const theme = useTheme();
|
||||
const DefaultFallback = theme === Theme.Light ? FallbackImageLight : FallbackImageDark;
|
||||
|
@ -40,6 +40,6 @@ const ImageWithErrorFallback = ({
|
|||
<img className={className} src={src} alt={alt} onError={errorHandler} {...props} />
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default ImageWithErrorFallback;
|
||||
|
|
|
@ -11,17 +11,19 @@ type Props = {
|
|||
isComplete?: boolean;
|
||||
};
|
||||
|
||||
const Index = ({ className, index, isActive, isComplete }: Props) => (
|
||||
<div
|
||||
className={classNames(
|
||||
styles.container,
|
||||
className,
|
||||
isActive && styles.active,
|
||||
isComplete && styles.completed
|
||||
)}
|
||||
>
|
||||
{isComplete ? <Tick /> : index}
|
||||
</div>
|
||||
);
|
||||
function Index({ className, index, isActive, isComplete }: Props) {
|
||||
return (
|
||||
<div
|
||||
className={classNames(
|
||||
styles.container,
|
||||
className,
|
||||
isActive && styles.active,
|
||||
isComplete && styles.completed
|
||||
)}
|
||||
>
|
||||
{isComplete ? <Tick /> : index}
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
export default Index;
|
||||
|
|
|
@ -13,7 +13,7 @@ type Props = {
|
|||
size?: 'default' | 'compact';
|
||||
};
|
||||
|
||||
const ItemPreview = ({ title, subtitle, icon, to, size = 'default' }: Props) => {
|
||||
function ItemPreview({ title, subtitle, icon, to, size = 'default' }: Props) {
|
||||
return (
|
||||
<div className={classNames(styles.item, styles[size])}>
|
||||
{icon}
|
||||
|
@ -34,6 +34,6 @@ const ItemPreview = ({ title, subtitle, icon, to, size = 'default' }: Props) =>
|
|||
</div>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default ItemPreview;
|
||||
|
|
|
@ -18,7 +18,7 @@ type Props = {
|
|||
isDisabled: boolean;
|
||||
};
|
||||
|
||||
const LivePreviewButton = ({ size, type, isDisabled }: Props) => {
|
||||
function LivePreviewButton({ size, type, isDisabled }: Props) {
|
||||
const { configs, updateConfigs } = useConfigs();
|
||||
const { userEndpoint } = useContext(AppEndpointsContext);
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
|
||||
|
@ -48,6 +48,6 @@ const LivePreviewButton = ({ size, type, isDisabled }: Props) => {
|
|||
/>
|
||||
</Tooltip>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default LivePreviewButton;
|
||||
|
|
|
@ -3,7 +3,7 @@ import { useRef, useState } from 'react';
|
|||
|
||||
const githubRawUrlPrefix = 'https://raw.githubusercontent.com/logto-io/logto/master';
|
||||
|
||||
const GithubRawImage = ({ src, alt }: HTMLProps<HTMLImageElement>) => {
|
||||
function GithubRawImage({ src, alt }: HTMLProps<HTMLImageElement>) {
|
||||
const imgRef = useRef<HTMLImageElement>(null);
|
||||
const [width, setWidth] = useState(0);
|
||||
|
||||
|
@ -30,6 +30,6 @@ const GithubRawImage = ({ src, alt }: HTMLProps<HTMLImageElement>) => {
|
|||
onLoad={onLoad}
|
||||
/>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default GithubRawImage;
|
||||
|
|
|
@ -14,7 +14,7 @@ type Props = {
|
|||
children: string;
|
||||
};
|
||||
|
||||
const Markdown = ({ className, children }: Props) => {
|
||||
function Markdown({ className, children }: Props) {
|
||||
const tocIdSet = useRef<Set<string>>(new Set());
|
||||
|
||||
const generateTocId = (text: string): Optional<string> => {
|
||||
|
@ -78,6 +78,6 @@ const Markdown = ({ className, children }: Props) => {
|
|||
{children}
|
||||
</ReactMarkdown>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default memo(Markdown);
|
||||
|
|
|
@ -17,14 +17,14 @@ type Props = {
|
|||
size?: 'medium' | 'large' | 'xlarge';
|
||||
} & Pick<CardTitleProps, 'learnMoreLink' | 'title' | 'subtitle' | 'isWordWrapEnabled'>;
|
||||
|
||||
const ModalLayout = ({
|
||||
function ModalLayout({
|
||||
children,
|
||||
footer,
|
||||
onClose,
|
||||
className,
|
||||
size = 'medium',
|
||||
...cardTitleProps
|
||||
}: Props) => {
|
||||
}: Props) {
|
||||
return (
|
||||
<Card className={classNames(styles.container, styles[size])}>
|
||||
<div className={styles.header}>
|
||||
|
@ -43,6 +43,6 @@ const ModalLayout = ({
|
|||
{footer && <div className={styles.footer}>{footer}</div>}
|
||||
</Card>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default ModalLayout;
|
||||
|
|
|
@ -25,7 +25,7 @@ export type Props = {
|
|||
className?: string;
|
||||
};
|
||||
|
||||
const MultiTextInput = ({
|
||||
function MultiTextInput({
|
||||
title,
|
||||
value,
|
||||
onChange,
|
||||
|
@ -33,7 +33,7 @@ const MultiTextInput = ({
|
|||
error,
|
||||
placeholder,
|
||||
className,
|
||||
}: Props) => {
|
||||
}: Props) {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
|
||||
|
||||
const [deleteFieldIndex, setDeleteFieldIndex] = useState<number>();
|
||||
|
@ -127,6 +127,6 @@ const MultiTextInput = ({
|
|||
</ConfirmModal>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default MultiTextInput;
|
||||
|
|
|
@ -12,23 +12,25 @@ type Props = MultiTextInputProps &
|
|||
formFieldClassName?: FormFieldProps['className'];
|
||||
};
|
||||
|
||||
const MultiTextInputField = ({
|
||||
function MultiTextInputField({
|
||||
title,
|
||||
isRequired,
|
||||
tip,
|
||||
formFieldClassName,
|
||||
value,
|
||||
...rest
|
||||
}: Props) => (
|
||||
<FormField
|
||||
title={title}
|
||||
isRequired={isRequired}
|
||||
tip={tip}
|
||||
className={formFieldClassName}
|
||||
headlineClassName={conditional(value && value.length > 1 && styles.headlineWithMultiInputs)}
|
||||
>
|
||||
<MultiTextInput title={title} value={value} {...rest} />
|
||||
</FormField>
|
||||
);
|
||||
}: Props) {
|
||||
return (
|
||||
<FormField
|
||||
title={title}
|
||||
isRequired={isRequired}
|
||||
tip={tip}
|
||||
className={formFieldClassName}
|
||||
headlineClassName={conditional(value && value.length > 1 && styles.headlineWithMultiInputs)}
|
||||
>
|
||||
<MultiTextInput title={title} value={value} {...rest} />
|
||||
</FormField>
|
||||
);
|
||||
}
|
||||
|
||||
export default MultiTextInputField;
|
||||
|
|
|
@ -1,11 +1,13 @@
|
|||
import type { OverlayScrollbarsComponentProps } from 'overlayscrollbars-react';
|
||||
import { OverlayScrollbarsComponent } from 'overlayscrollbars-react';
|
||||
|
||||
const OverlayScrollbar = (props: OverlayScrollbarsComponentProps) => (
|
||||
<OverlayScrollbarsComponent
|
||||
options={{ scrollbars: { autoHide: 'leave', autoHideDelay: 0 } }}
|
||||
{...props}
|
||||
/>
|
||||
);
|
||||
function OverlayScrollbar(props: OverlayScrollbarsComponentProps) {
|
||||
return (
|
||||
<OverlayScrollbarsComponent
|
||||
options={{ scrollbars: { autoHide: 'leave', autoHideDelay: 0 } }}
|
||||
{...props}
|
||||
/>
|
||||
);
|
||||
}
|
||||
|
||||
export default OverlayScrollbar;
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
const Next = () => {
|
||||
function Next() {
|
||||
return (
|
||||
<svg width="20" height="20" viewBox="0 0 20 20" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||
<path
|
||||
|
@ -7,6 +7,6 @@ const Next = () => {
|
|||
/>
|
||||
</svg>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default Next;
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
const Previous = () => {
|
||||
function Previous() {
|
||||
return (
|
||||
<svg width="20" height="20" viewBox="0 0 20 20" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||
<path
|
||||
|
@ -7,6 +7,6 @@ const Previous = () => {
|
|||
/>
|
||||
</svg>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default Previous;
|
||||
|
|
|
@ -19,14 +19,7 @@ export type Props = {
|
|||
onChange?: (pageIndex: number) => void;
|
||||
};
|
||||
|
||||
const Pagination = ({
|
||||
page,
|
||||
totalCount,
|
||||
pageSize,
|
||||
className,
|
||||
mode = 'normal',
|
||||
onChange,
|
||||
}: Props) => {
|
||||
function Pagination({ page, totalCount, pageSize, className, mode = 'normal', onChange }: Props) {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
|
||||
|
||||
/**
|
||||
|
@ -77,6 +70,6 @@ const Pagination = ({
|
|||
/>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default Pagination;
|
||||
|
|
|
@ -44,7 +44,7 @@ type Props = {
|
|||
retryHandler: () => void;
|
||||
};
|
||||
|
||||
const PermissionsTable = ({
|
||||
function PermissionsTable({
|
||||
scopes,
|
||||
isLoading,
|
||||
errorMessage,
|
||||
|
@ -58,7 +58,7 @@ const PermissionsTable = ({
|
|||
createHandler,
|
||||
deleteHandler,
|
||||
retryHandler,
|
||||
}: Props) => {
|
||||
}: Props) {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
|
||||
const { getDocumentationUrl } = useDocumentationUrl();
|
||||
|
||||
|
@ -182,6 +182,6 @@ const PermissionsTable = ({
|
|||
onRetry={retryHandler}
|
||||
/>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default PermissionsTable;
|
||||
|
|
|
@ -7,14 +7,16 @@ import { useTranslation } from 'react-i18next';
|
|||
import type DangerousRaw from '../DangerousRaw';
|
||||
import * as styles from './Radio.module.scss';
|
||||
|
||||
const Check = () => (
|
||||
<svg width="17" height="16" viewBox="0 0 17 16" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||
<path
|
||||
d="M8.66666 1.33334C4.99999 1.33334 1.99999 4.33334 1.99999 8.00001C1.99999 11.6667 4.99999 14.6667 8.66666 14.6667C12.3333 14.6667 15.3333 11.6667 15.3333 8.00001C15.3333 4.33334 12.3333 1.33334 8.66666 1.33334ZM11.4667 6.86668L8.26666 10.0667C7.99999 10.3333 7.59999 10.3333 7.33333 10.0667L5.86666 8.60001C5.59999 8.33334 5.59999 7.93334 5.86666 7.66668C6.13333 7.40001 6.53333 7.40001 6.79999 7.66668L7.79999 8.66668L10.5333 5.93334C10.8 5.66668 11.2 5.66668 11.4667 5.93334C11.7333 6.20001 11.7333 6.60001 11.4667 6.86668Z"
|
||||
fill="#5D34F2"
|
||||
/>
|
||||
</svg>
|
||||
);
|
||||
function Check() {
|
||||
return (
|
||||
<svg width="17" height="16" viewBox="0 0 17 16" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||
<path
|
||||
d="M8.66666 1.33334C4.99999 1.33334 1.99999 4.33334 1.99999 8.00001C1.99999 11.6667 4.99999 14.6667 8.66666 14.6667C12.3333 14.6667 15.3333 11.6667 15.3333 8.00001C15.3333 4.33334 12.3333 1.33334 8.66666 1.33334ZM11.4667 6.86668L8.26666 10.0667C7.99999 10.3333 7.59999 10.3333 7.33333 10.0667L5.86666 8.60001C5.59999 8.33334 5.59999 7.93334 5.86666 7.66668C6.13333 7.40001 6.53333 7.40001 6.79999 7.66668L7.79999 8.66668L10.5333 5.93334C10.8 5.66668 11.2 5.66668 11.4667 5.93334C11.7333 6.20001 11.7333 6.60001 11.4667 6.86668Z"
|
||||
fill="#5D34F2"
|
||||
/>
|
||||
</svg>
|
||||
);
|
||||
}
|
||||
|
||||
export type Props = {
|
||||
className?: string;
|
||||
|
@ -31,7 +33,7 @@ export type Props = {
|
|||
icon?: ReactNode;
|
||||
};
|
||||
|
||||
const Radio = ({
|
||||
function Radio({
|
||||
className,
|
||||
value,
|
||||
title,
|
||||
|
@ -44,7 +46,7 @@ const Radio = ({
|
|||
isDisabled,
|
||||
disabledLabel,
|
||||
icon,
|
||||
}: Props) => {
|
||||
}: Props) {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
|
||||
|
||||
const handleKeyPress: KeyboardEventHandler<HTMLDivElement> = useCallback(
|
||||
|
@ -95,6 +97,6 @@ const Radio = ({
|
|||
</div>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default Radio;
|
||||
|
|
|
@ -24,10 +24,10 @@ type Props = {
|
|||
onChange?: (value: string) => void;
|
||||
};
|
||||
|
||||
const RadioGroup = (
|
||||
function RadioGroup(
|
||||
{ name, children, value, className, onChange, type = 'plain' }: Props,
|
||||
ref?: Ref<HTMLDivElement>
|
||||
) => {
|
||||
) {
|
||||
return (
|
||||
<div ref={ref} className={classNames(styles.radioGroup, styles[type], className)}>
|
||||
{Children.map(children, (child) => {
|
||||
|
@ -56,7 +56,7 @@ const RadioGroup = (
|
|||
})}
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default forwardRef<HTMLDivElement, Props>(RadioGroup);
|
||||
export { default as Radio } from './Radio';
|
||||
|
|
|
@ -17,7 +17,7 @@ type Props = {
|
|||
className?: string;
|
||||
};
|
||||
|
||||
const RequestDataError = ({ error, onRetry, className }: Props) => {
|
||||
function RequestDataError({ error, onRetry, className }: Props) {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
|
||||
const theme = useTheme();
|
||||
const errorMessage = error.body?.message ?? error.message;
|
||||
|
@ -34,6 +34,6 @@ const RequestDataError = ({ error, onRetry, className }: Props) => {
|
|||
{onRetry && <Button title="general.retry" onClick={onRetry} />}
|
||||
</Card>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default RequestDataError;
|
||||
|
|
|
@ -20,7 +20,7 @@ type Props = {
|
|||
onSelectScope: (scope: ScopeResponse) => void;
|
||||
};
|
||||
|
||||
const ResourceItem = ({ resource, selectedScopes, onSelectResource, onSelectScope }: Props) => {
|
||||
function ResourceItem({ resource, selectedScopes, onSelectResource, onSelectScope }: Props) {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
|
||||
const { name, scopes } = resource;
|
||||
const selectedScopesCount = selectedScopes.length;
|
||||
|
@ -69,6 +69,6 @@ const ResourceItem = ({ resource, selectedScopes, onSelectResource, onSelectScop
|
|||
</div>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default ResourceItem;
|
||||
|
|
|
@ -11,28 +11,30 @@ type Props = {
|
|||
onSelect: (scope: ScopeResponse) => void;
|
||||
};
|
||||
|
||||
const SourceScopeItem = ({ scope, scope: { name }, isSelected, onSelect }: Props) => (
|
||||
<div className={styles.sourceScopeItem}>
|
||||
<Checkbox
|
||||
checked={isSelected}
|
||||
onChange={() => {
|
||||
onSelect(scope);
|
||||
}}
|
||||
/>
|
||||
<div
|
||||
className={styles.name}
|
||||
role="button"
|
||||
tabIndex={0}
|
||||
onKeyDown={onKeyDownHandler(() => {
|
||||
onSelect(scope);
|
||||
})}
|
||||
onClick={() => {
|
||||
onSelect(scope);
|
||||
}}
|
||||
>
|
||||
{name}
|
||||
function SourceScopeItem({ scope, scope: { name }, isSelected, onSelect }: Props) {
|
||||
return (
|
||||
<div className={styles.sourceScopeItem}>
|
||||
<Checkbox
|
||||
checked={isSelected}
|
||||
onChange={() => {
|
||||
onSelect(scope);
|
||||
}}
|
||||
/>
|
||||
<div
|
||||
className={styles.name}
|
||||
role="button"
|
||||
tabIndex={0}
|
||||
onKeyDown={onKeyDownHandler(() => {
|
||||
onSelect(scope);
|
||||
})}
|
||||
onClick={() => {
|
||||
onSelect(scope);
|
||||
}}
|
||||
>
|
||||
{name}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
);
|
||||
}
|
||||
|
||||
export default SourceScopeItem;
|
||||
|
|
|
@ -23,7 +23,7 @@ type Props = {
|
|||
onChange: (value: ScopeResponse[]) => void;
|
||||
};
|
||||
|
||||
const SourceScopesBox = ({ roleId, selectedScopes, onChange }: Props) => {
|
||||
function SourceScopesBox({ roleId, selectedScopes, onChange }: Props) {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
|
||||
|
||||
const { data: allResources, error: fetchAllResourcesError } = useSWR<
|
||||
|
@ -154,6 +154,6 @@ const SourceScopesBox = ({ roleId, selectedScopes, onChange }: Props) => {
|
|||
</div>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default SourceScopesBox;
|
||||
|
|
|
@ -10,7 +10,7 @@ export type Props = {
|
|||
onDelete: (scope: ScopeResponse) => void;
|
||||
};
|
||||
|
||||
const TargetScopeItem = ({ scope, onDelete }: Props) => {
|
||||
function TargetScopeItem({ scope, onDelete }: Props) {
|
||||
const {
|
||||
name,
|
||||
resource: { name: resourceName },
|
||||
|
@ -33,6 +33,6 @@ const TargetScopeItem = ({ scope, onDelete }: Props) => {
|
|||
</IconButton>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default TargetScopeItem;
|
||||
|
|
|
@ -11,7 +11,7 @@ type Props = {
|
|||
onChange: (value: ScopeResponse[]) => void;
|
||||
};
|
||||
|
||||
const TargetScopesBox = ({ selectedScopes, onChange }: Props) => {
|
||||
function TargetScopesBox({ selectedScopes, onChange }: Props) {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
|
||||
|
||||
return (
|
||||
|
@ -34,6 +34,6 @@ const TargetScopesBox = ({ selectedScopes, onChange }: Props) => {
|
|||
</div>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default TargetScopesBox;
|
||||
|
|
|
@ -13,12 +13,14 @@ type Props = {
|
|||
onChange: (value: ScopeResponse[]) => void;
|
||||
};
|
||||
|
||||
const RoleScopesTransfer = ({ roleId, value, onChange }: Props) => (
|
||||
<div className={classNames(transferLayout.container, styles.roleScopesTransfer)}>
|
||||
<SourceScopesBox roleId={roleId} selectedScopes={value} onChange={onChange} />
|
||||
<div className={transferLayout.verticalBar} />
|
||||
<TargetScopesBox selectedScopes={value} onChange={onChange} />
|
||||
</div>
|
||||
);
|
||||
function RoleScopesTransfer({ roleId, value, onChange }: Props) {
|
||||
return (
|
||||
<div className={classNames(transferLayout.container, styles.roleScopesTransfer)}>
|
||||
<SourceScopesBox roleId={roleId} selectedScopes={value} onChange={onChange} />
|
||||
<div className={transferLayout.verticalBar} />
|
||||
<TargetScopesBox selectedScopes={value} onChange={onChange} />
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
export default RoleScopesTransfer;
|
||||
|
|
|
@ -13,7 +13,7 @@ type Props = {
|
|||
onSelect: () => void;
|
||||
};
|
||||
|
||||
const SourceUserItem = ({ user, isSelected, onSelect }: Props) => {
|
||||
function SourceUserItem({ user, isSelected, onSelect }: Props) {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
|
||||
|
||||
return (
|
||||
|
@ -38,6 +38,6 @@ const SourceUserItem = ({ user, isSelected, onSelect }: Props) => {
|
|||
<div className={styles.name}>{user.name ?? t('users.unnamed')}</div>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default SourceUserItem;
|
||||
|
|
|
@ -27,7 +27,7 @@ type Props = {
|
|||
|
||||
const pageSize = defaultPageSize;
|
||||
|
||||
const SourceUsersBox = ({ roleId, selectedUsers, onChange }: Props) => {
|
||||
function SourceUsersBox({ roleId, selectedUsers, onChange }: Props) {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
|
||||
const [page, setPage] = useState(1);
|
||||
const [keyword, setKeyword] = useState('');
|
||||
|
@ -105,5 +105,5 @@ const SourceUsersBox = ({ roleId, selectedUsers, onChange }: Props) => {
|
|||
/>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
export default SourceUsersBox;
|
||||
|
|
|
@ -12,7 +12,7 @@ type Props = {
|
|||
onDelete: () => void;
|
||||
};
|
||||
|
||||
const TargetUserItem = ({ user, onDelete }: Props) => {
|
||||
function TargetUserItem({ user, onDelete }: Props) {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
|
||||
|
||||
return (
|
||||
|
@ -30,6 +30,6 @@ const TargetUserItem = ({ user, onDelete }: Props) => {
|
|||
</IconButton>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default TargetUserItem;
|
||||
|
|
|
@ -11,7 +11,7 @@ type Props = {
|
|||
onChange: (value: User[]) => void;
|
||||
};
|
||||
|
||||
const TargetUsersBox = ({ selectedUsers, onChange }: Props) => {
|
||||
function TargetUsersBox({ selectedUsers, onChange }: Props) {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
|
||||
|
||||
return (
|
||||
|
@ -35,6 +35,6 @@ const TargetUsersBox = ({ selectedUsers, onChange }: Props) => {
|
|||
</div>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default TargetUsersBox;
|
||||
|
|
|
@ -13,12 +13,14 @@ type Props = {
|
|||
onChange: (value: User[]) => void;
|
||||
};
|
||||
|
||||
const RoleUsersTransfer = ({ roleId, value, onChange }: Props) => (
|
||||
<div className={classNames(transferLayout.container, styles.roleUsersTransfer)}>
|
||||
<SourceUsersBox roleId={roleId} selectedUsers={value} onChange={onChange} />
|
||||
<div className={transferLayout.verticalBar} />
|
||||
<TargetUsersBox selectedUsers={value} onChange={onChange} />
|
||||
</div>
|
||||
);
|
||||
function RoleUsersTransfer({ roleId, value, onChange }: Props) {
|
||||
return (
|
||||
<div className={classNames(transferLayout.container, styles.roleUsersTransfer)}>
|
||||
<SourceUsersBox roleId={roleId} selectedUsers={value} onChange={onChange} />
|
||||
<div className={transferLayout.verticalBar} />
|
||||
<TargetUsersBox selectedUsers={value} onChange={onChange} />
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
export default RoleUsersTransfer;
|
||||
|
|
|
@ -17,14 +17,14 @@ type Props = {
|
|||
onClearSearch?: () => void;
|
||||
};
|
||||
|
||||
const Search = ({
|
||||
function Search({
|
||||
defaultValue = '',
|
||||
isClearable = false,
|
||||
placeholder,
|
||||
inputClassName,
|
||||
onSearch,
|
||||
onClearSearch,
|
||||
}: Props) => {
|
||||
}: Props) {
|
||||
const [inputValue, setInputValue] = useState<string>(defaultValue);
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
|
||||
|
||||
|
@ -58,6 +58,6 @@ const Search = ({
|
|||
)}
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default Search;
|
||||
|
|
|
@ -28,7 +28,7 @@ type Props<T> = {
|
|||
size?: 'small' | 'medium' | 'large';
|
||||
};
|
||||
|
||||
const Select = <T extends string>({
|
||||
function Select<T extends string>({
|
||||
className,
|
||||
value,
|
||||
options,
|
||||
|
@ -38,7 +38,7 @@ const Select = <T extends string>({
|
|||
placeholder,
|
||||
isClearable,
|
||||
size = 'large',
|
||||
}: Props<T>) => {
|
||||
}: Props<T>) {
|
||||
const [isOpen, setIsOpen] = useState(false);
|
||||
const anchorRef = useRef<HTMLInputElement>(null);
|
||||
const current = options.find((option) => value && option.value === value);
|
||||
|
@ -116,6 +116,6 @@ const Select = <T extends string>({
|
|||
</Dropdown>
|
||||
</>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default Select;
|
||||
|
|
|
@ -10,7 +10,7 @@ type Props = {
|
|||
callbackHref?: string;
|
||||
};
|
||||
|
||||
const SessionExpired = ({ callbackHref = '/callback', error }: Props) => {
|
||||
function SessionExpired({ callbackHref = '/callback', error }: Props) {
|
||||
const { signIn, signOut } = useLogto();
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
|
||||
|
||||
|
@ -31,6 +31,6 @@ const SessionExpired = ({ callbackHref = '/callback', error }: Props) => {
|
|||
/>
|
||||
</AppError>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default SessionExpired;
|
||||
|
|
|
@ -16,13 +16,13 @@ type Props = {
|
|||
size?: ButtonProps['size'];
|
||||
};
|
||||
|
||||
const ToggleUiThemeButton = ({
|
||||
function ToggleUiThemeButton({
|
||||
value,
|
||||
onToggle,
|
||||
className,
|
||||
iconClassName,
|
||||
size = 'medium',
|
||||
}: Props) => {
|
||||
}: Props) {
|
||||
const ThemeIcon = value === Theme.Light ? Sun : Moon;
|
||||
|
||||
return (
|
||||
|
@ -37,6 +37,6 @@ const ToggleUiThemeButton = ({
|
|||
/>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default ToggleUiThemeButton;
|
||||
|
|
|
@ -25,7 +25,7 @@ type Props = {
|
|||
signInExperience?: SignInExperience;
|
||||
};
|
||||
|
||||
const SignInExperiencePreview = ({ platform, mode, language = 'en', signInExperience }: Props) => {
|
||||
function SignInExperiencePreview({ platform, mode, language = 'en', signInExperience }: Props) {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
|
||||
|
||||
const { customPhrases } = useUiLanguages();
|
||||
|
@ -139,6 +139,6 @@ const SignInExperiencePreview = ({ platform, mode, language = 'en', signInExperi
|
|||
</div>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default SignInExperiencePreview;
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
import * as styles from './index.module.scss';
|
||||
|
||||
const Spacer = () => {
|
||||
function Spacer() {
|
||||
return <div className={styles.spacer} />;
|
||||
};
|
||||
}
|
||||
|
||||
export default Spacer;
|
||||
|
|
|
@ -6,133 +6,137 @@ type Props = {
|
|||
className?: string;
|
||||
};
|
||||
|
||||
const Ring = ({ className }: Props) => (
|
||||
<svg
|
||||
className={classNames(styles.ring, className)}
|
||||
width="16"
|
||||
height="16"
|
||||
viewBox="0 0 16 16"
|
||||
fill="none"
|
||||
xmlns="http://www.w3.org/2000/svg"
|
||||
>
|
||||
<path
|
||||
d="M9.33566 14.8714C9.44104 15.4135 9.08652 15.9451 8.53547 15.9821C7.4048 16.0579 6.2669 15.8929 5.19834 15.4934C3.81639 14.9767 2.60425 14.0879 1.69591 12.9253C0.78758 11.7627 0.218443 10.3715 0.0514252 8.90563C-0.115592 7.43973 0.126015 5.9562 0.749537 4.61905C1.37306 3.28191 2.35421 2.14323 3.5845 1.32891C4.8148 0.514598 6.24632 0.0563637 7.7208 0.00487344C9.19528 -0.0466168 10.6553 0.310643 11.9394 1.03715C12.9323 1.59891 13.7901 2.36452 14.4588 3.27942C14.7847 3.72531 14.6054 4.33858 14.1223 4.60633C13.6393 4.87408 13.0366 4.69278 12.6924 4.26086C12.2154 3.66218 11.6262 3.15785 10.9545 2.77787C9.99146 2.23298 8.89646 1.96504 7.7906 2.00366C6.68474 2.04227 5.6111 2.38595 4.68838 2.99669C3.76565 3.60742 3.02979 4.46143 2.56215 5.46429C2.09451 6.46715 1.91331 7.5798 2.03857 8.67922C2.16383 9.77864 2.59069 10.822 3.27194 11.694C3.95319 12.5659 4.8623 13.2325 5.89876 13.62C6.62154 13.8903 7.38663 14.0175 8.15188 13.9981C8.70399 13.9841 9.23028 14.3293 9.33566 14.8714Z"
|
||||
fill="currentColor"
|
||||
/>
|
||||
</svg>
|
||||
);
|
||||
function Ring({ className }: Props) {
|
||||
return (
|
||||
<svg
|
||||
className={classNames(styles.ring, className)}
|
||||
width="16"
|
||||
height="16"
|
||||
viewBox="0 0 16 16"
|
||||
fill="none"
|
||||
xmlns="http://www.w3.org/2000/svg"
|
||||
>
|
||||
<path
|
||||
d="M9.33566 14.8714C9.44104 15.4135 9.08652 15.9451 8.53547 15.9821C7.4048 16.0579 6.2669 15.8929 5.19834 15.4934C3.81639 14.9767 2.60425 14.0879 1.69591 12.9253C0.78758 11.7627 0.218443 10.3715 0.0514252 8.90563C-0.115592 7.43973 0.126015 5.9562 0.749537 4.61905C1.37306 3.28191 2.35421 2.14323 3.5845 1.32891C4.8148 0.514598 6.24632 0.0563637 7.7208 0.00487344C9.19528 -0.0466168 10.6553 0.310643 11.9394 1.03715C12.9323 1.59891 13.7901 2.36452 14.4588 3.27942C14.7847 3.72531 14.6054 4.33858 14.1223 4.60633C13.6393 4.87408 13.0366 4.69278 12.6924 4.26086C12.2154 3.66218 11.6262 3.15785 10.9545 2.77787C9.99146 2.23298 8.89646 1.96504 7.7906 2.00366C6.68474 2.04227 5.6111 2.38595 4.68838 2.99669C3.76565 3.60742 3.02979 4.46143 2.56215 5.46429C2.09451 6.46715 1.91331 7.5798 2.03857 8.67922C2.16383 9.77864 2.59069 10.822 3.27194 11.694C3.95319 12.5659 4.8623 13.2325 5.89876 13.62C6.62154 13.8903 7.38663 14.0175 8.15188 13.9981C8.70399 13.9841 9.23028 14.3293 9.33566 14.8714Z"
|
||||
fill="currentColor"
|
||||
/>
|
||||
</svg>
|
||||
);
|
||||
}
|
||||
|
||||
const Daisy = ({ className }: Props) => (
|
||||
<svg
|
||||
className={classNames(styles.daisy, className)}
|
||||
width="32"
|
||||
height="32"
|
||||
viewBox="0 0 32 32"
|
||||
fill="none"
|
||||
xmlns="http://www.w3.org/2000/svg"
|
||||
>
|
||||
<rect opacity="0.48" x="15" width="2" height="8" rx="1" fill="currentColor" />
|
||||
<rect opacity="0.96" x="15" y="24" width="2" height="8" rx="1" fill="currentColor" />
|
||||
<rect
|
||||
opacity="0.72"
|
||||
y="17"
|
||||
width="2"
|
||||
height="8"
|
||||
rx="1"
|
||||
transform="rotate(-90 0 17)"
|
||||
fill="currentColor"
|
||||
/>
|
||||
<rect
|
||||
opacity="0.24"
|
||||
x="24"
|
||||
y="17"
|
||||
width="2"
|
||||
height="8"
|
||||
rx="1"
|
||||
transform="rotate(-90 24 17)"
|
||||
fill="currentColor"
|
||||
/>
|
||||
<rect
|
||||
opacity="0.32"
|
||||
x="29.3564"
|
||||
y="7.13403"
|
||||
width="2"
|
||||
height="8"
|
||||
rx="1"
|
||||
transform="rotate(60 29.3564 7.13403)"
|
||||
fill="currentColor"
|
||||
/>
|
||||
<rect
|
||||
opacity="0.8"
|
||||
x="8.57227"
|
||||
y="19.134"
|
||||
width="2"
|
||||
height="8"
|
||||
rx="1"
|
||||
transform="rotate(60 8.57227 19.134)"
|
||||
fill="currentColor"
|
||||
/>
|
||||
<rect
|
||||
opacity="0.64"
|
||||
x="1.64355"
|
||||
y="8.86597"
|
||||
width="2"
|
||||
height="8"
|
||||
rx="1"
|
||||
transform="rotate(-60 1.64355 8.86597)"
|
||||
fill="currentColor"
|
||||
/>
|
||||
<rect
|
||||
opacity="0.16"
|
||||
x="22.4277"
|
||||
y="20.866"
|
||||
width="2"
|
||||
height="8"
|
||||
rx="1"
|
||||
transform="rotate(-60 22.4277 20.866)"
|
||||
fill="currentColor"
|
||||
/>
|
||||
<rect
|
||||
opacity="0.4"
|
||||
x="23.1338"
|
||||
y="1.64355"
|
||||
width="2"
|
||||
height="8"
|
||||
rx="1"
|
||||
transform="rotate(30 23.1338 1.64355)"
|
||||
fill="currentColor"
|
||||
/>
|
||||
<rect
|
||||
opacity="0.88"
|
||||
x="11.1338"
|
||||
y="22.4282"
|
||||
width="2"
|
||||
height="8"
|
||||
rx="1"
|
||||
transform="rotate(30 11.1338 22.4282)"
|
||||
fill="currentColor"
|
||||
/>
|
||||
<rect
|
||||
opacity="0.56"
|
||||
x="7.13379"
|
||||
y="2.64355"
|
||||
width="2"
|
||||
height="8"
|
||||
rx="1"
|
||||
transform="rotate(-30 7.13379 2.64355)"
|
||||
fill="currentColor"
|
||||
/>
|
||||
<rect
|
||||
opacity="0.08"
|
||||
x="19.1338"
|
||||
y="23.4282"
|
||||
width="2"
|
||||
height="8"
|
||||
rx="1"
|
||||
transform="rotate(-30 19.1338 23.4282)"
|
||||
fill="currentColor"
|
||||
/>
|
||||
</svg>
|
||||
);
|
||||
function Daisy({ className }: Props) {
|
||||
return (
|
||||
<svg
|
||||
className={classNames(styles.daisy, className)}
|
||||
width="32"
|
||||
height="32"
|
||||
viewBox="0 0 32 32"
|
||||
fill="none"
|
||||
xmlns="http://www.w3.org/2000/svg"
|
||||
>
|
||||
<rect opacity="0.48" x="15" width="2" height="8" rx="1" fill="currentColor" />
|
||||
<rect opacity="0.96" x="15" y="24" width="2" height="8" rx="1" fill="currentColor" />
|
||||
<rect
|
||||
opacity="0.72"
|
||||
y="17"
|
||||
width="2"
|
||||
height="8"
|
||||
rx="1"
|
||||
transform="rotate(-90 0 17)"
|
||||
fill="currentColor"
|
||||
/>
|
||||
<rect
|
||||
opacity="0.24"
|
||||
x="24"
|
||||
y="17"
|
||||
width="2"
|
||||
height="8"
|
||||
rx="1"
|
||||
transform="rotate(-90 24 17)"
|
||||
fill="currentColor"
|
||||
/>
|
||||
<rect
|
||||
opacity="0.32"
|
||||
x="29.3564"
|
||||
y="7.13403"
|
||||
width="2"
|
||||
height="8"
|
||||
rx="1"
|
||||
transform="rotate(60 29.3564 7.13403)"
|
||||
fill="currentColor"
|
||||
/>
|
||||
<rect
|
||||
opacity="0.8"
|
||||
x="8.57227"
|
||||
y="19.134"
|
||||
width="2"
|
||||
height="8"
|
||||
rx="1"
|
||||
transform="rotate(60 8.57227 19.134)"
|
||||
fill="currentColor"
|
||||
/>
|
||||
<rect
|
||||
opacity="0.64"
|
||||
x="1.64355"
|
||||
y="8.86597"
|
||||
width="2"
|
||||
height="8"
|
||||
rx="1"
|
||||
transform="rotate(-60 1.64355 8.86597)"
|
||||
fill="currentColor"
|
||||
/>
|
||||
<rect
|
||||
opacity="0.16"
|
||||
x="22.4277"
|
||||
y="20.866"
|
||||
width="2"
|
||||
height="8"
|
||||
rx="1"
|
||||
transform="rotate(-60 22.4277 20.866)"
|
||||
fill="currentColor"
|
||||
/>
|
||||
<rect
|
||||
opacity="0.4"
|
||||
x="23.1338"
|
||||
y="1.64355"
|
||||
width="2"
|
||||
height="8"
|
||||
rx="1"
|
||||
transform="rotate(30 23.1338 1.64355)"
|
||||
fill="currentColor"
|
||||
/>
|
||||
<rect
|
||||
opacity="0.88"
|
||||
x="11.1338"
|
||||
y="22.4282"
|
||||
width="2"
|
||||
height="8"
|
||||
rx="1"
|
||||
transform="rotate(30 11.1338 22.4282)"
|
||||
fill="currentColor"
|
||||
/>
|
||||
<rect
|
||||
opacity="0.56"
|
||||
x="7.13379"
|
||||
y="2.64355"
|
||||
width="2"
|
||||
height="8"
|
||||
rx="1"
|
||||
transform="rotate(-30 7.13379 2.64355)"
|
||||
fill="currentColor"
|
||||
/>
|
||||
<rect
|
||||
opacity="0.08"
|
||||
x="19.1338"
|
||||
y="23.4282"
|
||||
width="2"
|
||||
height="8"
|
||||
rx="1"
|
||||
transform="rotate(-30 19.1338 23.4282)"
|
||||
fill="currentColor"
|
||||
/>
|
||||
</svg>
|
||||
);
|
||||
}
|
||||
|
||||
export { Ring, Daisy };
|
||||
|
|
|
@ -9,11 +9,13 @@ type Props = {
|
|||
variant?: 'plain' | 'outlined';
|
||||
};
|
||||
|
||||
const Status = ({ status, children, variant = 'plain' }: Props) => (
|
||||
<div className={classNames(styles.status, styles[status], styles[variant])}>
|
||||
<div className={styles.icon} />
|
||||
<div>{children}</div>
|
||||
</div>
|
||||
);
|
||||
function Status({ status, children, variant = 'plain' }: Props) {
|
||||
return (
|
||||
<div className={classNames(styles.status, styles[status], styles[variant])}>
|
||||
<div className={styles.icon} />
|
||||
<div>{children}</div>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
export default Status;
|
||||
|
|
|
@ -10,26 +10,28 @@ type Props = {
|
|||
onDiscard: () => void;
|
||||
};
|
||||
|
||||
const SubmitFormChangesActionBar = ({ isOpen, isSubmitting, onSubmit, onDiscard }: Props) => (
|
||||
<div className={classNames(styles.container, isOpen && styles.active)}>
|
||||
<div className={styles.actionBar}>
|
||||
<Button
|
||||
size="medium"
|
||||
title="general.discard"
|
||||
disabled={isSubmitting}
|
||||
onClick={() => {
|
||||
onDiscard();
|
||||
}}
|
||||
/>
|
||||
<Button
|
||||
isLoading={isSubmitting}
|
||||
type="primary"
|
||||
size="medium"
|
||||
title="general.save_changes"
|
||||
onClick={async () => onSubmit()}
|
||||
/>
|
||||
function SubmitFormChangesActionBar({ isOpen, isSubmitting, onSubmit, onDiscard }: Props) {
|
||||
return (
|
||||
<div className={classNames(styles.container, isOpen && styles.active)}>
|
||||
<div className={styles.actionBar}>
|
||||
<Button
|
||||
size="medium"
|
||||
title="general.discard"
|
||||
disabled={isSubmitting}
|
||||
onClick={() => {
|
||||
onDiscard();
|
||||
}}
|
||||
/>
|
||||
<Button
|
||||
isLoading={isSubmitting}
|
||||
type="primary"
|
||||
size="medium"
|
||||
title="general.save_changes"
|
||||
onClick={async () => onSubmit()}
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
);
|
||||
}
|
||||
|
||||
export default SubmitFormChangesActionBar;
|
||||
|
|
|
@ -7,7 +7,7 @@ type Props = HTMLProps<HTMLInputElement> & {
|
|||
label?: ReactNode;
|
||||
};
|
||||
|
||||
const Switch = ({ label, ...rest }: Props, ref?: Ref<HTMLInputElement>) => {
|
||||
function Switch({ label, ...rest }: Props, ref?: Ref<HTMLInputElement>) {
|
||||
return (
|
||||
<div className={styles.wrapper}>
|
||||
<div className={styles.label}>{label}</div>
|
||||
|
@ -17,6 +17,6 @@ const Switch = ({ label, ...rest }: Props, ref?: Ref<HTMLInputElement>) => {
|
|||
</label>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default forwardRef<HTMLInputElement, Props>(Switch);
|
||||
|
|
|
@ -24,7 +24,7 @@ type Props =
|
|||
| (BaseProps & LinkStyleProps & Partial<Record<keyof TabStyleProps, undefined>>)
|
||||
| (BaseProps & TabStyleProps & Partial<Record<keyof LinkStyleProps, undefined>>);
|
||||
|
||||
const TabNavItem = ({ children, href, isActive, errorCount = 0, onClick }: Props) => {
|
||||
function TabNavItem({ children, href, isActive, errorCount = 0, onClick }: Props) {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
|
||||
const location = useLocation();
|
||||
const selected = href ? location.pathname === href : isActive;
|
||||
|
@ -46,6 +46,6 @@ const TabNavItem = ({ children, href, isActive, errorCount = 0, onClick }: Props
|
|||
)}
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default TabNavItem;
|
||||
|
|
|
@ -9,8 +9,8 @@ type Props = {
|
|||
children: React.ReactNode;
|
||||
};
|
||||
|
||||
const TabNav = ({ className, children }: Props) => (
|
||||
<nav className={classNames(styles.nav, className)}>{children}</nav>
|
||||
);
|
||||
function TabNav({ className, children }: Props) {
|
||||
return <nav className={classNames(styles.nav, className)}>{children}</nav>;
|
||||
}
|
||||
|
||||
export default TabNav;
|
||||
|
|
|
@ -7,12 +7,14 @@ type Props = {
|
|||
children: ReactNode;
|
||||
};
|
||||
|
||||
const TableEmptyWrapper = ({ columns, children }: Props) => (
|
||||
<tr>
|
||||
<td colSpan={columns} className={styles.tableEmptyWrapper}>
|
||||
<div className={styles.content}>{children}</div>
|
||||
</td>
|
||||
</tr>
|
||||
);
|
||||
function TableEmptyWrapper({ columns, children }: Props) {
|
||||
return (
|
||||
<tr>
|
||||
<td colSpan={columns} className={styles.tableEmptyWrapper}>
|
||||
<div className={styles.content}>{children}</div>
|
||||
</td>
|
||||
</tr>
|
||||
);
|
||||
}
|
||||
|
||||
export default TableEmptyWrapper;
|
||||
|
|
|
@ -15,7 +15,7 @@ type Props = {
|
|||
columns: number;
|
||||
};
|
||||
|
||||
const TableError = ({ title, content, onRetry, columns }: Props) => {
|
||||
function TableError({ title, content, onRetry, columns }: Props) {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
|
||||
const theme = useTheme();
|
||||
|
||||
|
@ -31,6 +31,6 @@ const TableError = ({ title, content, onRetry, columns }: Props) => {
|
|||
</td>
|
||||
</tr>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default TableError;
|
||||
|
|
|
@ -4,7 +4,7 @@ type Props = {
|
|||
columnSpans: number[];
|
||||
};
|
||||
|
||||
const TableLoading = ({ columnSpans }: Props) => {
|
||||
function TableLoading({ columnSpans }: Props) {
|
||||
return (
|
||||
<>
|
||||
{Array.from({ length: 8 }).map((_, rowIndex) => (
|
||||
|
@ -29,6 +29,6 @@ const TableLoading = ({ columnSpans }: Props) => {
|
|||
))}
|
||||
</>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default TableLoading;
|
||||
|
|
|
@ -17,14 +17,7 @@ type Props = {
|
|||
action: ReactNode;
|
||||
};
|
||||
|
||||
const TablePlaceholder = ({
|
||||
image,
|
||||
imageDark,
|
||||
title,
|
||||
description,
|
||||
learnMoreLink,
|
||||
action,
|
||||
}: Props) => {
|
||||
function TablePlaceholder({ image, imageDark, title, description, learnMoreLink, action }: Props) {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
|
||||
const theme = useTheme();
|
||||
|
||||
|
@ -46,6 +39,6 @@ const TablePlaceholder = ({
|
|||
{action && <div className={styles.action}>{action}</div>}
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default TablePlaceholder;
|
||||
|
|
|
@ -36,7 +36,7 @@ type Props<
|
|||
onRetry?: () => void;
|
||||
};
|
||||
|
||||
const Table = <
|
||||
function Table<
|
||||
TFieldValues extends FieldValues = FieldValues,
|
||||
TName extends FieldPath<TFieldValues> = FieldPath<TFieldValues>
|
||||
>({
|
||||
|
@ -56,7 +56,7 @@ const Table = <
|
|||
errorMessage,
|
||||
hasBorder,
|
||||
onRetry,
|
||||
}: Props<TFieldValues, TName>) => {
|
||||
}: Props<TFieldValues, TName>) {
|
||||
const totalColspan = columns.reduce((result, { colSpan }) => {
|
||||
return result + (colSpan ?? 1);
|
||||
}, 0);
|
||||
|
@ -152,6 +152,6 @@ const Table = <
|
|||
{pagination && <Pagination className={styles.pagination} {...pagination} />}
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default Table;
|
||||
|
|
|
@ -11,7 +11,7 @@ type Props = Omit<HTMLProps<HTMLInputElement>, 'size'> & {
|
|||
suffix?: ReactElement;
|
||||
};
|
||||
|
||||
const TextInput = (
|
||||
function TextInput(
|
||||
{
|
||||
errorMessage,
|
||||
hasError = Boolean(errorMessage),
|
||||
|
@ -23,7 +23,7 @@ const TextInput = (
|
|||
...rest
|
||||
}: Props,
|
||||
reference: ForwardedRef<HTMLInputElement>
|
||||
) => {
|
||||
) {
|
||||
return (
|
||||
<div className={className}>
|
||||
<div
|
||||
|
@ -45,6 +45,6 @@ const TextInput = (
|
|||
{hasError && errorMessage && <div className={styles.errorMessage}>{errorMessage}</div>}
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default forwardRef(TextInput);
|
||||
|
|
|
@ -11,7 +11,7 @@ type Props = AnchorHTMLAttributes<HTMLAnchorElement> &
|
|||
isTrailingIcon?: boolean;
|
||||
};
|
||||
|
||||
const TextLink = ({ to, children, icon, isTrailingIcon = false, className, ...rest }: Props) => {
|
||||
function TextLink({ to, children, icon, isTrailingIcon = false, className, ...rest }: Props) {
|
||||
if (to) {
|
||||
return (
|
||||
<Link
|
||||
|
@ -36,6 +36,6 @@ const TextLink = ({ to, children, icon, isTrailingIcon = false, className, ...re
|
|||
<>{children}</>
|
||||
</a>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default TextLink;
|
||||
|
|
|
@ -9,15 +9,15 @@ type Props = HTMLProps<HTMLTextAreaElement> & {
|
|||
hasError?: boolean;
|
||||
};
|
||||
|
||||
const Textarea = (
|
||||
function Textarea(
|
||||
{ className, hasError, ...rest }: Props,
|
||||
reference: ForwardedRef<HTMLTextAreaElement>
|
||||
) => {
|
||||
) {
|
||||
return (
|
||||
<div className={classNames(styles.container, hasError && styles.error, className)}>
|
||||
<textarea {...rest} ref={reference} />
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default forwardRef(Textarea);
|
||||
|
|
|
@ -21,7 +21,7 @@ type Props = HTMLProps<HTMLDivElement> & {
|
|||
|
||||
const supportHorizontalAlignmentPlacements = new Set<TipBubblePlacement>(['top', 'bottom']);
|
||||
|
||||
const TipBubble = (
|
||||
function TipBubble(
|
||||
{
|
||||
children,
|
||||
position,
|
||||
|
@ -33,7 +33,7 @@ const TipBubble = (
|
|||
...rest
|
||||
}: Props,
|
||||
reference: ForwardedRef<HTMLDivElement>
|
||||
) => {
|
||||
) {
|
||||
if (!anchorRef.current) {
|
||||
return null;
|
||||
}
|
||||
|
@ -64,6 +64,6 @@ const TipBubble = (
|
|||
<div className={styles.arrow} style={{ ...arrowPosition }} />
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default forwardRef(TipBubble);
|
||||
|
|
|
@ -25,14 +25,14 @@ export type Props = {
|
|||
content?: ((closeTip: () => void) => ReactNode) | ReactNode;
|
||||
};
|
||||
|
||||
const ToggleTip = ({
|
||||
function ToggleTip({
|
||||
children,
|
||||
className,
|
||||
anchorClassName,
|
||||
placement = 'top',
|
||||
horizontalAlign = 'center',
|
||||
content,
|
||||
}: Props) => {
|
||||
}: Props) {
|
||||
const tipBubbleRef = useRef<HTMLDivElement>(null);
|
||||
const anchorRef = useRef<HTMLDivElement>(null);
|
||||
|
||||
|
@ -95,6 +95,6 @@ const ToggleTip = ({
|
|||
</ReactModal>
|
||||
</>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default ToggleTip;
|
||||
|
|
|
@ -27,7 +27,7 @@ type Props = {
|
|||
content?: ReactNode;
|
||||
};
|
||||
|
||||
const Tooltip = ({
|
||||
function Tooltip({
|
||||
className,
|
||||
isKeepOpen = false,
|
||||
isSuccessful = false,
|
||||
|
@ -36,7 +36,7 @@ const Tooltip = ({
|
|||
anchorClassName,
|
||||
children,
|
||||
content,
|
||||
}: Props) => {
|
||||
}: Props) {
|
||||
const [tooltipDom, setTooltipDom] = useState<HTMLDivElement>();
|
||||
const anchorRef = useRef<HTMLDivElement>(null);
|
||||
const tooltipRef = useRef<HTMLDivElement>(null);
|
||||
|
@ -146,6 +146,6 @@ const Tooltip = ({
|
|||
)}
|
||||
</>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default Tooltip;
|
||||
|
|
|
@ -6,7 +6,7 @@ import Success from '@/assets/images/toast-success.svg';
|
|||
|
||||
import * as styles from './index.module.scss';
|
||||
|
||||
const Toast = () => {
|
||||
function Toast() {
|
||||
return (
|
||||
<Toaster
|
||||
toastOptions={{
|
||||
|
@ -31,6 +31,6 @@ const Toast = () => {
|
|||
)}
|
||||
</Toaster>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default Toast;
|
||||
|
|
|
@ -5,7 +5,7 @@ type Props = {
|
|||
className?: string;
|
||||
};
|
||||
|
||||
const UnnamedTrans = ({ resource, className }: Props) => {
|
||||
function UnnamedTrans({ resource, className }: Props) {
|
||||
const {
|
||||
i18n: { languages },
|
||||
} = useTranslation();
|
||||
|
@ -16,6 +16,6 @@ const UnnamedTrans = ({ resource, className }: Props) => {
|
|||
}
|
||||
|
||||
return <span className={className}>{resource[matchedLanguage]}</span>;
|
||||
};
|
||||
}
|
||||
|
||||
export default UnnamedTrans;
|
||||
|
|
|
@ -23,7 +23,7 @@ type Props = {
|
|||
parentPath?: string;
|
||||
};
|
||||
|
||||
const UnsavedChangesAlertModal = ({ hasUnsavedChanges, parentPath }: Props) => {
|
||||
function UnsavedChangesAlertModal({ hasUnsavedChanges, parentPath }: Props) {
|
||||
const { navigator } = useContext(UNSAFE_NavigationContext);
|
||||
|
||||
/**
|
||||
|
@ -96,6 +96,6 @@ const UnsavedChangesAlertModal = ({ hasUnsavedChanges, parentPath }: Props) => {
|
|||
{t('general.unsaved_changes_warning')}
|
||||
</ConfirmModal>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default UnsavedChangesAlertModal;
|
||||
|
|
|
@ -20,13 +20,13 @@ export type Props = {
|
|||
onUploadErrorChange: (errorMessage?: string) => void;
|
||||
};
|
||||
|
||||
const FileUploader = ({
|
||||
function FileUploader({
|
||||
maxSize,
|
||||
allowedMimeTypes,
|
||||
actionDescription,
|
||||
onCompleted,
|
||||
onUploadErrorChange,
|
||||
}: Props) => {
|
||||
}: Props) {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
|
||||
const [isUploading, setIsUploading] = useState(false);
|
||||
const [uploadError, setUploadError] = useState<string>();
|
||||
|
@ -119,6 +119,6 @@ const FileUploader = ({
|
|||
</div>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default FileUploader;
|
||||
|
|
|
@ -22,8 +22,8 @@ export type Props = Omit<FileUploaderProps, 'maxSize' | 'allowedMimeTypes'> & {
|
|||
onDelete: () => void;
|
||||
};
|
||||
|
||||
const ImageUploader = ({ name, value, onDelete, ...rest }: Props) =>
|
||||
value ? (
|
||||
function ImageUploader({ name, value, onDelete, ...rest }: Props) {
|
||||
return value ? (
|
||||
<div className={styles.imageUploader}>
|
||||
<img alt={name} src={value} />
|
||||
<IconButton
|
||||
|
@ -38,5 +38,6 @@ const ImageUploader = ({ name, value, onDelete, ...rest }: Props) =>
|
|||
) : (
|
||||
<FileUploader allowedMimeTypes={allowedImageMimeTypes} maxSize={maxImageSizeLimit} {...rest} />
|
||||
);
|
||||
}
|
||||
|
||||
export default ImageUploader;
|
||||
|
|
|
@ -12,7 +12,7 @@ type Props = Pick<ImageUploaderProps, 'name' | 'value' | 'actionDescription'> &
|
|||
onChange: (value: string) => void;
|
||||
};
|
||||
|
||||
const ImageUploaderField = ({ onChange, ...rest }: Props) => {
|
||||
function ImageUploaderField({ onChange, ...rest }: Props) {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
|
||||
|
||||
const limitDescription = t('components.uploader.image_limit', {
|
||||
|
@ -37,6 +37,6 @@ const ImageUploaderField = ({ onChange, ...rest }: Props) => {
|
|||
</div>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default ImageUploaderField;
|
||||
|
|
|
@ -22,7 +22,7 @@ type Props = {
|
|||
confirmButtonTitle?: AdminConsoleKey;
|
||||
};
|
||||
|
||||
const UserAccountInformation = ({
|
||||
function UserAccountInformation({
|
||||
username,
|
||||
password,
|
||||
title,
|
||||
|
@ -30,7 +30,7 @@ const UserAccountInformation = ({
|
|||
onConfirm,
|
||||
passwordLabel,
|
||||
confirmButtonTitle,
|
||||
}: Props) => {
|
||||
}: Props) {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
|
||||
const [passwordVisible, setPasswordVisible] = useState(false);
|
||||
|
||||
|
@ -94,6 +94,6 @@ const UserAccountInformation = ({
|
|||
</ModalLayout>
|
||||
</ReactModal>
|
||||
);
|
||||
};
|
||||
}
|
||||
|
||||
export default UserAccountInformation;
|
||||
|
|
Some files were not shown because too many files have changed in this diff Show more
Loading…
Add table
Reference in a new issue