* Small logic cleanups * Small logic cleanups (#3451) * remove a few things * oops * stop swallowing the error * queue callbacks * oops * log error if caught * no need to be nullable * move isClosing=true up * reset `isClosing` and `closeCallbacks` on close completion and open * run queued callbacks on `open` if there are any pending * rm unnecessary ref and check * ensure order of calls is always correct * call `snapToIndex()` on open * add tester to storybook --------- Co-authored-by: Hailey <me@haileyok.com>
200 lines
4.8 KiB
TypeScript
200 lines
4.8 KiB
TypeScript
import React from 'react'
|
|
import {View} from 'react-native'
|
|
import {msg} from '@lingui/macro'
|
|
import {useLingui} from '@lingui/react'
|
|
|
|
import {atoms as a, useBreakpoints, useTheme} from '#/alf'
|
|
import {Button, ButtonColor, ButtonText} from '#/components/Button'
|
|
import * as Dialog from '#/components/Dialog'
|
|
import {Text} from '#/components/Typography'
|
|
|
|
export {useDialogControl as usePromptControl} from '#/components/Dialog'
|
|
|
|
const Context = React.createContext<{
|
|
titleId: string
|
|
descriptionId: string
|
|
}>({
|
|
titleId: '',
|
|
descriptionId: '',
|
|
})
|
|
|
|
export function Outer({
|
|
children,
|
|
control,
|
|
testID,
|
|
}: React.PropsWithChildren<{
|
|
control: Dialog.DialogOuterProps['control']
|
|
testID?: string
|
|
}>) {
|
|
const {gtMobile} = useBreakpoints()
|
|
const titleId = React.useId()
|
|
const descriptionId = React.useId()
|
|
|
|
const context = React.useMemo(
|
|
() => ({titleId, descriptionId}),
|
|
[titleId, descriptionId],
|
|
)
|
|
|
|
return (
|
|
<Dialog.Outer control={control} testID={testID}>
|
|
<Context.Provider value={context}>
|
|
<Dialog.Handle />
|
|
|
|
<Dialog.ScrollableInner
|
|
accessibilityLabelledBy={titleId}
|
|
accessibilityDescribedBy={descriptionId}
|
|
style={[gtMobile ? {width: 'auto', maxWidth: 400} : a.w_full]}>
|
|
{children}
|
|
</Dialog.ScrollableInner>
|
|
</Context.Provider>
|
|
</Dialog.Outer>
|
|
)
|
|
}
|
|
|
|
export function TitleText({children}: React.PropsWithChildren<{}>) {
|
|
const {titleId} = React.useContext(Context)
|
|
return (
|
|
<Text nativeID={titleId} style={[a.text_2xl, a.font_bold, a.pb_sm]}>
|
|
{children}
|
|
</Text>
|
|
)
|
|
}
|
|
|
|
export function DescriptionText({children}: React.PropsWithChildren<{}>) {
|
|
const t = useTheme()
|
|
const {descriptionId} = React.useContext(Context)
|
|
return (
|
|
<Text
|
|
nativeID={descriptionId}
|
|
style={[a.text_md, a.leading_snug, t.atoms.text_contrast_high, a.pb_lg]}>
|
|
{children}
|
|
</Text>
|
|
)
|
|
}
|
|
|
|
export function Actions({children}: React.PropsWithChildren<{}>) {
|
|
const {gtMobile} = useBreakpoints()
|
|
|
|
return (
|
|
<View
|
|
style={[
|
|
a.w_full,
|
|
a.gap_md,
|
|
a.justify_end,
|
|
gtMobile
|
|
? [a.flex_row, a.flex_row_reverse, a.justify_start]
|
|
: [a.flex_col],
|
|
]}>
|
|
{children}
|
|
</View>
|
|
)
|
|
}
|
|
|
|
export function Cancel({
|
|
cta,
|
|
}: {
|
|
/**
|
|
* Optional i18n string. If undefined, it will default to "Cancel".
|
|
*/
|
|
cta?: string
|
|
}) {
|
|
const {_} = useLingui()
|
|
const {gtMobile} = useBreakpoints()
|
|
const {close} = Dialog.useDialogContext()
|
|
const onPress = React.useCallback(() => {
|
|
close()
|
|
}, [close])
|
|
|
|
return (
|
|
<Button
|
|
variant="solid"
|
|
color="secondary"
|
|
size={gtMobile ? 'small' : 'medium'}
|
|
label={cta || _(msg`Cancel`)}
|
|
onPress={onPress}>
|
|
<ButtonText>{cta || _(msg`Cancel`)}</ButtonText>
|
|
</Button>
|
|
)
|
|
}
|
|
|
|
export function Action({
|
|
onPress,
|
|
color = 'primary',
|
|
cta,
|
|
testID,
|
|
}: {
|
|
/**
|
|
* Callback to run when the action is pressed. The method is called _after_
|
|
* the dialog closes.
|
|
*
|
|
* Note: The dialog will close automatically when the action is pressed, you
|
|
* should NOT close the dialog as a side effect of this method.
|
|
*/
|
|
onPress: () => void
|
|
color?: ButtonColor
|
|
/**
|
|
* Optional i18n string. If undefined, it will default to "Confirm".
|
|
*/
|
|
cta?: string
|
|
testID?: string
|
|
}) {
|
|
const {_} = useLingui()
|
|
const {gtMobile} = useBreakpoints()
|
|
const {close} = Dialog.useDialogContext()
|
|
const handleOnPress = React.useCallback(() => {
|
|
close(onPress)
|
|
}, [close, onPress])
|
|
|
|
return (
|
|
<Button
|
|
variant="solid"
|
|
color={color}
|
|
size={gtMobile ? 'small' : 'medium'}
|
|
label={cta || _(msg`Confirm`)}
|
|
onPress={handleOnPress}
|
|
testID={testID}>
|
|
<ButtonText>{cta || _(msg`Confirm`)}</ButtonText>
|
|
</Button>
|
|
)
|
|
}
|
|
|
|
export function Basic({
|
|
control,
|
|
title,
|
|
description,
|
|
cancelButtonCta,
|
|
confirmButtonCta,
|
|
onConfirm,
|
|
confirmButtonColor,
|
|
}: React.PropsWithChildren<{
|
|
control: Dialog.DialogOuterProps['control']
|
|
title: string
|
|
description: string
|
|
cancelButtonCta?: string
|
|
confirmButtonCta?: string
|
|
/**
|
|
* Callback to run when the Confirm button is pressed. The method is called
|
|
* _after_ the dialog closes.
|
|
*
|
|
* Note: The dialog will close automatically when the action is pressed, you
|
|
* should NOT close the dialog as a side effect of this method.
|
|
*/
|
|
onConfirm: () => void
|
|
confirmButtonColor?: ButtonColor
|
|
}>) {
|
|
return (
|
|
<Outer control={control} testID="confirmModal">
|
|
<TitleText>{title}</TitleText>
|
|
<DescriptionText>{description}</DescriptionText>
|
|
<Actions>
|
|
<Action
|
|
cta={confirmButtonCta}
|
|
onPress={onConfirm}
|
|
color={confirmButtonColor}
|
|
testID="confirmBtn"
|
|
/>
|
|
<Cancel cta={cancelButtonCta} />
|
|
</Actions>
|
|
</Outer>
|
|
)
|
|
}
|