mirror of
https://github.com/logto-io/logto.git
synced 2025-01-06 20:40:08 -05:00
feat(console): select
This commit is contained in:
parent
8d80106475
commit
b22b0cf22f
11 changed files with 292 additions and 52 deletions
|
@ -1,12 +1,11 @@
|
|||
import React, { MouseEventHandler, ReactNode, useRef, useState } from 'react';
|
||||
import ReactModal from 'react-modal';
|
||||
import React, { ReactNode, useRef, useState } from 'react';
|
||||
|
||||
import { Props as ButtonProps } from '../Button';
|
||||
import Dropdown from '../Dropdown';
|
||||
import ActionMenuButton from './ActionMenuButton';
|
||||
import * as styles from './index.module.scss';
|
||||
import usePosition from './use-position';
|
||||
|
||||
export { default as ActionMenuItem } from './ActionMenuItem';
|
||||
export { default as ActionMenuItem } from '../Dropdown/DropdownItem';
|
||||
|
||||
type Props = {
|
||||
children: ReactNode;
|
||||
|
@ -17,46 +16,26 @@ type Props = {
|
|||
const ActionMenu = ({ children, buttonProps, title }: Props) => {
|
||||
const [isOpen, setIsOpen] = useState(false);
|
||||
const anchorReference = useRef<HTMLDivElement>(null);
|
||||
const overlayReference = useRef<HTMLDivElement>(null);
|
||||
|
||||
const {
|
||||
position: { left, top },
|
||||
mutate,
|
||||
} = usePosition(anchorReference, overlayReference);
|
||||
|
||||
const handleClick: MouseEventHandler<HTMLButtonElement> = () => {
|
||||
setIsOpen(true);
|
||||
// Calc position after modal opened.
|
||||
setTimeout(() => {
|
||||
mutate();
|
||||
});
|
||||
};
|
||||
|
||||
return (
|
||||
<div className={styles.actionMenu}>
|
||||
<ActionMenuButton {...buttonProps} ref={anchorReference} onClick={handleClick} />
|
||||
<ReactModal
|
||||
shouldCloseOnOverlayClick
|
||||
isOpen={isOpen}
|
||||
style={{ content: { left: `${left}px`, top: `${top + 4}px` } }}
|
||||
className={styles.content}
|
||||
overlayClassName={styles.overlay}
|
||||
onRequestClose={() => {
|
||||
setIsOpen(false);
|
||||
}}
|
||||
>
|
||||
<div ref={overlayReference}>
|
||||
{title && <div className={styles.title}>{title}</div>}
|
||||
<ul
|
||||
className={styles.actionList}
|
||||
<ActionMenuButton
|
||||
{...buttonProps}
|
||||
ref={anchorReference}
|
||||
onClick={() => {
|
||||
setIsOpen(true);
|
||||
}}
|
||||
/>
|
||||
<Dropdown
|
||||
title={title}
|
||||
anchorRef={anchorReference}
|
||||
isOpen={isOpen}
|
||||
onClose={() => {
|
||||
setIsOpen(false);
|
||||
}}
|
||||
>
|
||||
{children}
|
||||
</ul>
|
||||
</div>
|
||||
</ReactModal>
|
||||
</Dropdown>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
|
|
|
@ -1,12 +1,14 @@
|
|||
@use '@/scss/underscore' as _;
|
||||
|
||||
.item {
|
||||
padding: _.unit(2) _.unit(4);
|
||||
padding: 6px _.unit(2);
|
||||
border-radius: _.unit(2);
|
||||
list-style: none;
|
||||
font: var(--font-body-medium);
|
||||
cursor: pointer;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
overflow: hidden;
|
||||
|
||||
&:hover {
|
||||
background: var(--color-neutral-95);
|
|
@ -1,7 +1,7 @@
|
|||
import classNames from 'classnames';
|
||||
import React from 'react';
|
||||
|
||||
import * as styles from './ActionMenuItem.module.scss';
|
||||
import * as styles from './DropdownItem.module.scss';
|
||||
|
||||
type Props = {
|
||||
onClick?: () => void;
|
||||
|
@ -10,7 +10,7 @@ type Props = {
|
|||
type?: 'default' | 'danger';
|
||||
};
|
||||
|
||||
const ActionMenuItem = ({ onClick, children, icon, type = 'default' }: Props) => (
|
||||
const DropdownItem = ({ onClick, children, icon, type = 'default' }: Props) => (
|
||||
<li
|
||||
className={classNames(styles.item, styles[type])}
|
||||
onClick={() => {
|
||||
|
@ -22,4 +22,4 @@ const ActionMenuItem = ({ onClick, children, icon, type = 'default' }: Props) =>
|
|||
</li>
|
||||
);
|
||||
|
||||
export default ActionMenuItem;
|
||||
export default DropdownItem;
|
33
packages/console/src/components/Dropdown/index.module.scss
Normal file
33
packages/console/src/components/Dropdown/index.module.scss
Normal file
|
@ -0,0 +1,33 @@
|
|||
@use '@/scss/underscore' as _;
|
||||
|
||||
.content {
|
||||
background: var(--color-layer-1);
|
||||
box-shadow: var(--shadow-light-s2);
|
||||
border-radius: _.unit(2);
|
||||
position: absolute;
|
||||
|
||||
&.onTop {
|
||||
box-shadow: var(--shadow-light-s2-reversed);
|
||||
}
|
||||
|
||||
&:focus {
|
||||
outline: none;
|
||||
}
|
||||
|
||||
.title {
|
||||
padding: _.unit(4) _.unit(4) 0 _.unit(4);
|
||||
font: var(--font-body-medium);
|
||||
color: var(--color-caption);
|
||||
}
|
||||
}
|
||||
|
||||
.overlay {
|
||||
background: transparent;
|
||||
position: fixed;
|
||||
inset: 0;
|
||||
}
|
||||
|
||||
ul.list {
|
||||
margin: 0;
|
||||
padding: _.unit(2) _.unit(1);
|
||||
}
|
52
packages/console/src/components/Dropdown/index.tsx
Normal file
52
packages/console/src/components/Dropdown/index.tsx
Normal file
|
@ -0,0 +1,52 @@
|
|||
import classNames from 'classnames';
|
||||
import React, { ReactNode, RefObject, useRef } from 'react';
|
||||
import ReactModal from 'react-modal';
|
||||
|
||||
import * as styles from './index.module.scss';
|
||||
import usePosition from './use-position';
|
||||
|
||||
export { default as DropdownItem } from './DropdownItem';
|
||||
|
||||
type Props = {
|
||||
children: ReactNode;
|
||||
title?: ReactNode;
|
||||
isOpen: boolean;
|
||||
onClose?: () => void;
|
||||
anchorRef: RefObject<HTMLElement>;
|
||||
isFullWidth?: boolean;
|
||||
};
|
||||
|
||||
const Dropdown = ({ children, title, isOpen, onClose, anchorRef, isFullWidth }: Props) => {
|
||||
const overlayRef = useRef<HTMLDivElement>(null);
|
||||
|
||||
const { position, mutate } = usePosition(anchorRef, overlayRef);
|
||||
|
||||
return (
|
||||
<ReactModal
|
||||
shouldCloseOnOverlayClick
|
||||
isOpen={isOpen}
|
||||
style={{
|
||||
content: position
|
||||
? {
|
||||
left: `${position.left}px`,
|
||||
top: `${position.top}px`,
|
||||
width: isFullWidth ? `${position.width}px` : undefined,
|
||||
}
|
||||
: { visibility: 'hidden' },
|
||||
}}
|
||||
className={classNames(styles.content, position?.isOnTop && styles.onTop)}
|
||||
overlayClassName={styles.overlay}
|
||||
onRequestClose={onClose}
|
||||
onAfterOpen={mutate}
|
||||
>
|
||||
<div ref={overlayRef}>
|
||||
{title && <div className={styles.title}>{title}</div>}
|
||||
<ul className={styles.list} onClick={onClose}>
|
||||
{children}
|
||||
</ul>
|
||||
</div>
|
||||
</ReactModal>
|
||||
);
|
||||
};
|
||||
|
||||
export default Dropdown;
|
45
packages/console/src/components/Dropdown/use-position.ts
Normal file
45
packages/console/src/components/Dropdown/use-position.ts
Normal file
|
@ -0,0 +1,45 @@
|
|||
import { RefObject, useCallback, useLayoutEffect, useState } from 'react';
|
||||
|
||||
type Position = {
|
||||
left: number;
|
||||
top: number;
|
||||
width: number;
|
||||
isOnTop?: boolean;
|
||||
};
|
||||
|
||||
// Leave space for box-shadow effect.
|
||||
const safePadding = 12;
|
||||
// The distance to anchor
|
||||
const distance = 4;
|
||||
|
||||
export default function usePosition(
|
||||
anchorRef: RefObject<HTMLElement>,
|
||||
overlayRef: RefObject<HTMLElement>
|
||||
) {
|
||||
const [position, setPosition] = useState<Position>();
|
||||
|
||||
const updatePosition = useCallback(() => {
|
||||
if (anchorRef.current && overlayRef.current) {
|
||||
const anchor = anchorRef.current.getBoundingClientRect();
|
||||
const overlay = overlayRef.current.getBoundingClientRect();
|
||||
const isOnTop = anchor.y + anchor.height + overlay.height > window.innerHeight - safePadding;
|
||||
const isOnLeft = anchor.x + overlay.width > window.innerWidth - safePadding;
|
||||
const left = isOnLeft ? anchor.x + anchor.width - overlay.width : anchor.x;
|
||||
const top = isOnTop
|
||||
? anchor.y - overlay.height - distance
|
||||
: anchor.y + anchor.height + distance;
|
||||
setPosition({ left, top, width: anchor.width, isOnTop });
|
||||
}
|
||||
}, [anchorRef, overlayRef]);
|
||||
|
||||
useLayoutEffect(() => {
|
||||
updatePosition();
|
||||
window.addEventListener('resize', updatePosition);
|
||||
|
||||
return () => {
|
||||
window.removeEventListener('resize', updatePosition);
|
||||
};
|
||||
}, [updatePosition]);
|
||||
|
||||
return { position, mutate: updatePosition };
|
||||
}
|
48
packages/console/src/components/Select/index.module.scss
Normal file
48
packages/console/src/components/Select/index.module.scss
Normal file
|
@ -0,0 +1,48 @@
|
|||
@use '@/scss/underscore' as _;
|
||||
|
||||
.select {
|
||||
background: var(--color-layer-1);
|
||||
border: 1px solid var(--color-border);
|
||||
border-radius: 6px;
|
||||
outline: 3px solid transparent;
|
||||
transition-property: outline, border;
|
||||
transition-timing-function: ease-in-out;
|
||||
transition-duration: 0.2s;
|
||||
padding: _.unit(2) _.unit(5) _.unit(2) _.unit(3);
|
||||
font: var(--font-body-medium);
|
||||
cursor: pointer;
|
||||
position: relative;
|
||||
overflow: hidden;
|
||||
text-overflow: ellipsis;
|
||||
white-space: nowrap;
|
||||
|
||||
&.open {
|
||||
border-color: var(--color-primary);
|
||||
outline-color: var(--color-focused-variant);
|
||||
}
|
||||
|
||||
&.readOnly {
|
||||
background: var(--color-inverse-on-surface);
|
||||
color: var(--color-text);
|
||||
border-color: var(--color-border);
|
||||
cursor: default;
|
||||
}
|
||||
|
||||
&.error {
|
||||
border-color: var(--color-error);
|
||||
|
||||
&.open {
|
||||
outline-color: var(--color-error-80);
|
||||
}
|
||||
}
|
||||
|
||||
.arrow {
|
||||
position: absolute;
|
||||
right: _.unit(2);
|
||||
top: 6px;
|
||||
|
||||
svg {
|
||||
fill: var(--color-icon);
|
||||
}
|
||||
}
|
||||
}
|
75
packages/console/src/components/Select/index.tsx
Normal file
75
packages/console/src/components/Select/index.tsx
Normal file
|
@ -0,0 +1,75 @@
|
|||
import classNames from 'classnames';
|
||||
import React, { ReactNode, useRef, useState } from 'react';
|
||||
|
||||
import { ArrowDown, ArrowUp } from '@/icons/Arrow';
|
||||
|
||||
import Dropdown, { DropdownItem } from '../Dropdown';
|
||||
import * as styles from './index.module.scss';
|
||||
|
||||
type Option = {
|
||||
value: string;
|
||||
title: ReactNode;
|
||||
};
|
||||
|
||||
type Props = {
|
||||
value?: string;
|
||||
options: Option[];
|
||||
onChange?: (value: string) => void;
|
||||
isReadOnly?: boolean;
|
||||
hasError?: boolean;
|
||||
};
|
||||
|
||||
const Select = ({ value, options, onChange, isReadOnly, hasError }: Props) => {
|
||||
const [isOpen, setIsOpen] = useState(false);
|
||||
const anchorRef = useRef<HTMLInputElement>(null);
|
||||
const current = options.find((option) => value && option.value === value);
|
||||
|
||||
const handleSelect = (value: string) => {
|
||||
onChange?.(value);
|
||||
setIsOpen(false);
|
||||
};
|
||||
|
||||
return (
|
||||
<>
|
||||
<div
|
||||
ref={anchorRef}
|
||||
className={classNames(
|
||||
styles.select,
|
||||
isOpen && styles.open,
|
||||
isReadOnly && styles.readOnly,
|
||||
hasError && styles.error
|
||||
)}
|
||||
role="button"
|
||||
onClick={() => {
|
||||
if (!isReadOnly) {
|
||||
setIsOpen(true);
|
||||
}
|
||||
}}
|
||||
>
|
||||
{current?.title}
|
||||
<div className={styles.arrow}>{isOpen ? <ArrowUp /> : <ArrowDown />}</div>
|
||||
</div>
|
||||
<Dropdown
|
||||
isFullWidth
|
||||
anchorRef={anchorRef}
|
||||
isOpen={isOpen}
|
||||
onClose={() => {
|
||||
setIsOpen(false);
|
||||
}}
|
||||
>
|
||||
{options.map(({ value, title }) => (
|
||||
<DropdownItem
|
||||
key={value}
|
||||
onClick={() => {
|
||||
handleSelect(value);
|
||||
}}
|
||||
>
|
||||
{title}
|
||||
</DropdownItem>
|
||||
))}
|
||||
</Dropdown>
|
||||
</>
|
||||
);
|
||||
};
|
||||
|
||||
export default Select;
|
|
@ -4,6 +4,7 @@ import { Controller, useFormContext } from 'react-hook-form';
|
|||
import { useTranslation } from 'react-i18next';
|
||||
|
||||
import FormField from '@/components/FormField';
|
||||
import Select from '@/components/Select';
|
||||
import Switch from '@/components/Switch';
|
||||
|
||||
import { SignInExperienceForm } from '../types';
|
||||
|
@ -21,14 +22,20 @@ const SignInMethodsForm = () => {
|
|||
<>
|
||||
<div className={styles.title}>{t('sign_in_exp.sign_in_methods.title')}</div>
|
||||
<FormField isRequired title="admin_console.sign_in_exp.sign_in_methods.primary">
|
||||
{/* TODO: LOG-2191 select component */}
|
||||
<select {...register('signInMethods.primary')}>
|
||||
{signInMethods.map((method) => (
|
||||
<option key={method} value={method}>
|
||||
{t('sign_in_exp.sign_in_methods.methods', { context: method })}
|
||||
</option>
|
||||
))}
|
||||
</select>
|
||||
<Controller
|
||||
name="signInMethods.primary"
|
||||
control={control}
|
||||
render={({ field: { value, onChange } }) => (
|
||||
<Select
|
||||
value={value}
|
||||
options={signInMethods.map((method) => ({
|
||||
value: method,
|
||||
title: t('sign_in_exp.sign_in_methods.methods', { context: method }),
|
||||
}))}
|
||||
onChange={onChange}
|
||||
/>
|
||||
)}
|
||||
/>
|
||||
</FormField>
|
||||
<FormField isRequired title="admin_console.sign_in_exp.sign_in_methods.enable_secondary">
|
||||
<Switch
|
||||
|
|
|
@ -1,3 +0,0 @@
|
|||
import { SignInMethods } from '@logto/schemas';
|
||||
|
||||
export const signInMethods: Array<keyof SignInMethods> = ['username', 'sms', 'email', 'social'];
|
|
@ -137,6 +137,7 @@
|
|||
--color-focused-variant: rgba(93, 52, 242, 16%); // 16% P40
|
||||
--shadow-light-s1: 0 4px 8px rgba(66, 41, 159, 8%);
|
||||
--shadow-light-s2: 0 4px 12px rgba(66, 41, 159, 12%);
|
||||
--shadow-light-s2-reversed: 0 -4px 12px rgba(66, 41, 159, 12%);
|
||||
--shadow-light-s3: 0 4px 16px rgba(66, 41, 159, 16%);
|
||||
|
||||
// Client specific variables
|
||||
|
@ -286,6 +287,7 @@
|
|||
--color-focused-variant: rgba(202, 190, 255, 16%); // 16% P40
|
||||
--shadow-light-s1: 0 4px 8px rgba(66, 41, 159, 8%);
|
||||
--shadow-light-s2: 0 4px 12px rgba(66, 41, 159, 12%);
|
||||
--shadow-light-s2-reversed: 0 -4px -12px rgba(66, 41, 159, 12%);
|
||||
--shadow-light-s3: 0 4px 16px rgba(66, 41, 159, 16%);
|
||||
|
||||
// Client specific variables
|
||||
|
|
Loading…
Reference in a new issue