* Add round and square buttons * Allow some style for buttons, add icons * Change text selection color * Center button text, whoops * Outer layout, some primitive updates * WIP * onboarding feed prefs (#2590) * add `style` to toggle label to modify text style * Revert "add `style` to toggle label to modify text style" This reverts commit 8f4b517b8585ca64a4bf44f6cb40ac070ece8932. * following feed prefs * remove unnecessary memo * reusable divider component * org imports * add finished screen * Theme SelectedAccountCard * Require at least 3 interests * Placeholder save logic * WIP algo feeds * Improve lineHeight handling, add RichText, improve Link by adding InlineLink * Inherit lineHeight in heading comps * Algo feeds mostly good * Topical feeds ish * Layout cleanup * Improve button styles * moderation prefs for onboarding (#2594) * WIP algo feeds * modify controlalbelgroup typing for easy .map() * adjust padding on button * add moderation screen * add moderation screen * add moderation screen --------- Co-authored-by: Eric Bailey <git@esb.lol> * Fix toggle button styles * A11y props on outer portal * Put it all on red * New data shape * Handle mock data * Bulk write (not yet) * Remove interests validation * Clean up interests * i18n layout and first step * Clean up suggested follows screen * Clean up following step * Clean up algo feeds step * Clean up topical feeds * Add skeleton for feed card * WIP moderation step * cleanup moderation styles (#2605) * cleanup moderation styles * fix(?) toggle button group styles * adjust toggle to fit any screen * Some more cleanup * Icons * ToggleButton tweaks * Reset * Hook up data * Better suggestions * Bulk write * Some logging * Use new api * Concat topical feeds * Metrics * Disable links in RichText, feedcards * Tweak primary feed cards * Update metrics * Fix layout shift * Fix ToggleButton again, whoops * Error state * Bump api package, ensure interests are saved * Better fix for autofill * i18n, button positions * Remove unused export * Add default prefs object * Fix overflow in user cards * Add 2 lines of bios to suggested accounts cards * Nits * Don't resolve facets by default * Update storybook * Disable flag for now * Remove age dialog from moderations step * Improvements and tweaks to new onboarding --------- Co-authored-by: Hailey <153161762+haileyok@users.noreply.github.com> Co-authored-by: Paul Frazee <pfrazee@gmail.com>
263 lines
6.7 KiB
TypeScript
263 lines
6.7 KiB
TypeScript
import React from 'react'
|
|
import {
|
|
GestureResponderEvent,
|
|
Linking,
|
|
TouchableWithoutFeedback,
|
|
} from 'react-native'
|
|
import {
|
|
useLinkProps,
|
|
useNavigation,
|
|
StackActions,
|
|
} from '@react-navigation/native'
|
|
import {sanitizeUrl} from '@braintree/sanitize-url'
|
|
|
|
import {useInteractionState} from '#/components/hooks/useInteractionState'
|
|
import {isWeb} from '#/platform/detection'
|
|
import {useTheme, web, flatten, TextStyleProp} from '#/alf'
|
|
import {Button, ButtonProps} from '#/components/Button'
|
|
import {AllNavigatorParams, NavigationProp} from '#/lib/routes/types'
|
|
import {
|
|
convertBskyAppUrlIfNeeded,
|
|
isExternalUrl,
|
|
linkRequiresWarning,
|
|
} from '#/lib/strings/url-helpers'
|
|
import {useModalControls} from '#/state/modals'
|
|
import {router} from '#/routes'
|
|
import {Text} from '#/components/Typography'
|
|
|
|
/**
|
|
* Only available within a `Link`, since that inherits from `Button`.
|
|
* `InlineLink` provides no context.
|
|
*/
|
|
export {useButtonContext as useLinkContext} from '#/components/Button'
|
|
|
|
type BaseLinkProps = Pick<
|
|
Parameters<typeof useLinkProps<AllNavigatorParams>>[0],
|
|
'to'
|
|
> & {
|
|
/**
|
|
* The React Navigation `StackAction` to perform when the link is pressed.
|
|
*/
|
|
action?: 'push' | 'replace' | 'navigate'
|
|
|
|
/**
|
|
* If true, will warn the user if the link text does not match the href.
|
|
*
|
|
* Note: atm this only works for `InlineLink`s with a string child.
|
|
*/
|
|
warnOnMismatchingTextChild?: boolean
|
|
}
|
|
|
|
export function useLink({
|
|
to,
|
|
displayText,
|
|
action = 'push',
|
|
warnOnMismatchingTextChild,
|
|
}: BaseLinkProps & {
|
|
displayText: string
|
|
}) {
|
|
const navigation = useNavigation<NavigationProp>()
|
|
const {href} = useLinkProps<AllNavigatorParams>({
|
|
to:
|
|
typeof to === 'string' ? convertBskyAppUrlIfNeeded(sanitizeUrl(to)) : to,
|
|
})
|
|
const isExternal = isExternalUrl(href)
|
|
const {openModal, closeModal} = useModalControls()
|
|
|
|
const onPress = React.useCallback(
|
|
(e: GestureResponderEvent) => {
|
|
const requiresWarning = Boolean(
|
|
warnOnMismatchingTextChild &&
|
|
displayText &&
|
|
isExternal &&
|
|
linkRequiresWarning(href, displayText),
|
|
)
|
|
|
|
if (requiresWarning) {
|
|
e.preventDefault()
|
|
|
|
openModal({
|
|
name: 'link-warning',
|
|
text: displayText,
|
|
href: href,
|
|
})
|
|
} else {
|
|
e.preventDefault()
|
|
|
|
if (isExternal) {
|
|
Linking.openURL(href)
|
|
} else {
|
|
/**
|
|
* A `GestureResponderEvent`, but cast to `any` to avoid using a bunch
|
|
* of @ts-ignore below.
|
|
*/
|
|
const event = e as any
|
|
const isMiddleClick = isWeb && event.button === 1
|
|
const isMetaKey =
|
|
isWeb &&
|
|
(event.metaKey || event.altKey || event.ctrlKey || event.shiftKey)
|
|
const shouldOpenInNewTab = isMetaKey || isMiddleClick
|
|
|
|
if (
|
|
shouldOpenInNewTab ||
|
|
href.startsWith('http') ||
|
|
href.startsWith('mailto')
|
|
) {
|
|
Linking.openURL(href)
|
|
} else {
|
|
closeModal() // close any active modals
|
|
|
|
if (action === 'push') {
|
|
navigation.dispatch(StackActions.push(...router.matchPath(href)))
|
|
} else if (action === 'replace') {
|
|
navigation.dispatch(
|
|
StackActions.replace(...router.matchPath(href)),
|
|
)
|
|
} else if (action === 'navigate') {
|
|
// @ts-ignore
|
|
navigation.navigate(...router.matchPath(href))
|
|
} else {
|
|
throw Error('Unsupported navigator action.')
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
[
|
|
href,
|
|
isExternal,
|
|
warnOnMismatchingTextChild,
|
|
navigation,
|
|
action,
|
|
displayText,
|
|
closeModal,
|
|
openModal,
|
|
],
|
|
)
|
|
|
|
return {
|
|
isExternal,
|
|
href,
|
|
onPress,
|
|
}
|
|
}
|
|
|
|
export type LinkProps = Omit<BaseLinkProps, 'warnOnMismatchingTextChild'> &
|
|
Omit<ButtonProps, 'style' | 'onPress' | 'disabled' | 'label'> & {
|
|
/**
|
|
* Label for a11y. Defaults to the href.
|
|
*/
|
|
label?: string
|
|
}
|
|
|
|
/**
|
|
* A interactive element that renders as a `<a>` tag on the web. On mobile it
|
|
* will translate the `href` to navigator screens and params and dispatch a
|
|
* navigation action.
|
|
*
|
|
* Intended to behave as a web anchor tag. For more complex routing, use a
|
|
* `Button`.
|
|
*/
|
|
export function Link({children, to, action = 'push', ...rest}: LinkProps) {
|
|
const {href, isExternal, onPress} = useLink({
|
|
to,
|
|
displayText: typeof children === 'string' ? children : '',
|
|
action,
|
|
})
|
|
|
|
return (
|
|
<Button
|
|
label={href}
|
|
{...rest}
|
|
role="link"
|
|
accessibilityRole="link"
|
|
href={href}
|
|
onPress={onPress}
|
|
{...web({
|
|
hrefAttrs: {
|
|
target: isExternal ? 'blank' : undefined,
|
|
rel: isExternal ? 'noopener noreferrer' : undefined,
|
|
},
|
|
dataSet: {
|
|
// default to no underline, apply this ourselves
|
|
noUnderline: '1',
|
|
},
|
|
})}>
|
|
{children}
|
|
</Button>
|
|
)
|
|
}
|
|
|
|
export type InlineLinkProps = React.PropsWithChildren<
|
|
BaseLinkProps &
|
|
TextStyleProp & {
|
|
/**
|
|
* Label for a11y. Defaults to the href.
|
|
*/
|
|
label?: string
|
|
}
|
|
>
|
|
|
|
export function InlineLink({
|
|
children,
|
|
to,
|
|
action = 'push',
|
|
warnOnMismatchingTextChild,
|
|
style,
|
|
...rest
|
|
}: InlineLinkProps) {
|
|
const t = useTheme()
|
|
const stringChildren = typeof children === 'string'
|
|
const {href, isExternal, onPress} = useLink({
|
|
to,
|
|
displayText: stringChildren ? children : '',
|
|
action,
|
|
warnOnMismatchingTextChild,
|
|
})
|
|
const {state: focused, onIn: onFocus, onOut: onBlur} = useInteractionState()
|
|
const {
|
|
state: pressed,
|
|
onIn: onPressIn,
|
|
onOut: onPressOut,
|
|
} = useInteractionState()
|
|
|
|
return (
|
|
<TouchableWithoutFeedback
|
|
accessibilityRole="button"
|
|
onPress={onPress}
|
|
onPressIn={onPressIn}
|
|
onPressOut={onPressOut}
|
|
onFocus={onFocus}
|
|
onBlur={onBlur}>
|
|
<Text
|
|
label={href}
|
|
{...rest}
|
|
style={[
|
|
{color: t.palette.primary_500},
|
|
(focused || pressed) && {
|
|
outline: 0,
|
|
textDecorationLine: 'underline',
|
|
textDecorationColor: t.palette.primary_500,
|
|
},
|
|
flatten(style),
|
|
]}
|
|
role="link"
|
|
accessibilityRole="link"
|
|
href={href}
|
|
{...web({
|
|
hrefAttrs: {
|
|
target: isExternal ? 'blank' : undefined,
|
|
rel: isExternal ? 'noopener noreferrer' : undefined,
|
|
},
|
|
dataSet: stringChildren
|
|
? {}
|
|
: {
|
|
// default to no underline, apply this ourselves
|
|
noUnderline: '1',
|
|
},
|
|
})}>
|
|
{children}
|
|
</Text>
|
|
</TouchableWithoutFeedback>
|
|
)
|
|
}
|