anvil/striker-ui/components/Text/SensitiveText.tsx

117 lines
2.8 KiB
TypeScript
Raw Normal View History

import { Button, styled } from '@mui/material';
import {
createElement,
FC,
ReactElement,
ReactNode,
useCallback,
useMemo,
useState,
} from 'react';
import { BORDER_RADIUS, EERIE_BLACK } from '../../lib/consts/DEFAULT_THEME';
2023-02-28 06:33:18 +00:00
import BodyText from './BodyText';
import FlexBox from '../FlexBox';
import IconButton from '../IconButton';
import MonoText from './MonoText';
const InlineButton = styled(Button)({
backgroundColor: EERIE_BLACK,
borderRadius: BORDER_RADIUS,
minWidth: 'initial',
padding: '0 .6em',
textTransform: 'none',
':hover': {
backgroundColor: `${EERIE_BLACK}F0`,
},
});
2023-02-28 06:33:18 +00:00
const SensitiveText: FC<SensitiveTextProps> = ({
children,
inline: isInline = false,
2023-02-28 06:33:18 +00:00
monospaced: isMonospaced = false,
revealButtonProps,
2023-02-28 06:33:18 +00:00
revealInitially: isRevealInitially = false,
textLineHeight = 2.8,
2023-02-28 06:33:18 +00:00
textProps,
}) => {
const [isReveal, setIsReveal] = useState<boolean>(isRevealInitially);
const clickEventHandler = useCallback(() => {
setIsReveal((previous) => !previous);
}, []);
const textSxLineHeight = useMemo<number | string | undefined>(
() => (isInline ? undefined : textLineHeight || undefined),
[isInline, textLineHeight],
);
2023-02-28 06:33:18 +00:00
const textElementType = useMemo(
() => (isMonospaced ? MonoText : BodyText),
[isMonospaced],
);
const contentElement = useMemo(() => {
let content: ReactNode;
if (isReveal) {
content =
typeof children === 'string'
? createElement(
textElementType,
{
sx: {
lineHeight: textSxLineHeight,
2023-02-28 06:33:18 +00:00
maxWidth: '20em',
overflowY: 'scroll',
whiteSpace: 'nowrap',
},
...textProps,
},
children,
)
: children;
} else {
content = createElement(
textElementType,
{
sx: {
lineHeight: textSxLineHeight,
},
...textProps,
},
'*****',
);
2023-02-28 06:33:18 +00:00
}
return content;
}, [children, isReveal, textElementType, textProps, textSxLineHeight]);
const rootElement = useMemo<ReactElement>(
() =>
isInline ? (
<InlineButton onClick={clickEventHandler}>
{contentElement}
</InlineButton>
) : (
<FlexBox row spacing=".5em">
{contentElement}
<IconButton
edge="end"
mapPreset="visibility"
onClick={clickEventHandler}
state={String(isReveal)}
sx={{ marginRight: '-.2em', padding: '.2em' }}
variant="normal"
{...revealButtonProps}
/>
</FlexBox>
),
[clickEventHandler, contentElement, revealButtonProps, isInline, isReveal],
2023-02-28 06:33:18 +00:00
);
return rootElement;
2023-02-28 06:33:18 +00:00
};
export default SensitiveText;