diff options
Diffstat (limited to 'src/components')
-rw-r--r-- | src/components/Link.tsx | 52 | ||||
-rw-r--r-- | src/components/RichText.tsx | 17 |
2 files changed, 52 insertions, 17 deletions
diff --git a/src/components/Link.tsx b/src/components/Link.tsx index 7d0e83332..1a494626f 100644 --- a/src/components/Link.tsx +++ b/src/components/Link.tsx @@ -1,23 +1,24 @@ import React from 'react' import {GestureResponderEvent} from 'react-native' -import {useLinkProps, StackActions} from '@react-navigation/native' import {sanitizeUrl} from '@braintree/sanitize-url' +import {StackActions, useLinkProps} from '@react-navigation/native' -import {useInteractionState} from '#/components/hooks/useInteractionState' -import {isWeb} from '#/platform/detection' -import {useTheme, web, flatten, TextStyleProp, atoms as a} from '#/alf' -import {Button, ButtonProps} from '#/components/Button' import {AllNavigatorParams} from '#/lib/routes/types' +import {shareUrl} from '#/lib/sharing' import { convertBskyAppUrlIfNeeded, isExternalUrl, linkRequiresWarning, } from '#/lib/strings/url-helpers' +import {isNative, isWeb} from '#/platform/detection' import {useModalControls} from '#/state/modals' -import {router} from '#/routes' -import {Text, TextProps} from '#/components/Typography' -import {useOpenLink} from 'state/preferences/in-app-browser' +import {useOpenLink} from '#/state/preferences/in-app-browser' import {useNavigationDeduped} from 'lib/hooks/useNavigationDeduped' +import {atoms as a, flatten, TextStyleProp, useTheme, web} from '#/alf' +import {Button, ButtonProps} from '#/components/Button' +import {useInteractionState} from '#/components/hooks/useInteractionState' +import {Text, TextProps} from '#/components/Typography' +import {router} from '#/routes' /** * Only available within a `Link`, since that inherits from `Button`. @@ -60,6 +61,11 @@ type BaseLinkProps = Pick< * Web-only attribute. Sets `download` attr on web. */ download?: string + + /** + * Native-only attribute. If true, will open the share sheet on long press. + */ + shareOnLongPress?: boolean } export function useLink({ @@ -68,6 +74,7 @@ export function useLink({ action = 'push', disableMismatchWarning, onPress: outerOnPress, + shareOnLongPress, }: BaseLinkProps & { displayText: string }) { @@ -157,10 +164,34 @@ export function useLink({ ], ) + const handleLongPress = React.useCallback(() => { + const requiresWarning = Boolean( + !disableMismatchWarning && + displayText && + isExternal && + linkRequiresWarning(href, displayText), + ) + + if (requiresWarning) { + openModal({ + name: 'link-warning', + text: displayText, + href: href, + share: true, + }) + } else { + shareUrl(href) + } + }, [disableMismatchWarning, displayText, href, isExternal, openModal]) + + const onLongPress = + isNative && isExternal && shareOnLongPress ? handleLongPress : undefined + return { isExternal, href, onPress, + onLongPress, } } @@ -229,16 +260,18 @@ export function InlineLink({ download, selectable, label, + shareOnLongPress, ...rest }: InlineLinkProps) { const t = useTheme() const stringChildren = typeof children === 'string' - const {href, isExternal, onPress} = useLink({ + const {href, isExternal, onPress, onLongPress} = useLink({ to, displayText: stringChildren ? children : '', action, disableMismatchWarning, onPress: outerOnPress, + shareOnLongPress, }) const { state: hovered, @@ -270,6 +303,7 @@ export function InlineLink({ ]} role="link" onPress={download ? undefined : onPress} + onLongPress={onLongPress} onPressIn={onPressIn} onPressOut={onPressOut} onFocus={onFocus} diff --git a/src/components/RichText.tsx b/src/components/RichText.tsx index 1a14415cf..0e89e13b7 100644 --- a/src/components/RichText.tsx +++ b/src/components/RichText.tsx @@ -1,15 +1,15 @@ import React from 'react' -import {RichText as RichTextAPI, AppBskyRichtextFacet} from '@atproto/api' -import {useLingui} from '@lingui/react' +import {AppBskyRichtextFacet, RichText as RichTextAPI} from '@atproto/api' import {msg} from '@lingui/macro' +import {useLingui} from '@lingui/react' -import {atoms as a, TextStyleProp, flatten, useTheme, web, native} from '#/alf' -import {InlineLink} from '#/components/Link' -import {Text, TextProps} from '#/components/Typography' -import {toShortUrl} from 'lib/strings/url-helpers' -import {TagMenu, useTagMenuControl} from '#/components/TagMenu' +import {toShortUrl} from '#/lib/strings/url-helpers' import {isNative} from '#/platform/detection' +import {atoms as a, flatten, native, TextStyleProp, useTheme, web} from '#/alf' import {useInteractionState} from '#/components/hooks/useInteractionState' +import {InlineLink} from '#/components/Link' +import {TagMenu, useTagMenuControl} from '#/components/TagMenu' +import {Text, TextProps} from '#/components/Typography' const WORD_WRAP = {wordWrap: 1} @@ -105,7 +105,8 @@ export function RichText({ to={link.uri} style={[...styles, {pointerEvents: 'auto'}]} // @ts-ignore TODO - dataSet={WORD_WRAP}> + dataSet={WORD_WRAP} + shareOnLongPress> {toShortUrl(segment.text)} </InlineLink>, ) |