Alt text for gifs (#3876)

* add alt text dialog

* multiline alt text input

* add pressable alt text badge

* rename `ALT: ` to `Alt text: ` to avoid including old bad ones

* reuse alt text reminder

* reuse alt text reminder in gallery

* add alt text reminder in the dialog itself

* autofocus text input

* reorder components to fix tab order

* fix close btn position
zio/stable
Samuel Newman 2024-05-06 17:28:38 +01:00 committed by GitHub
parent ae7626ce6e
commit c33c3b7d1e
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
9 changed files with 344 additions and 47 deletions

View File

@ -43,7 +43,9 @@ export function Outer({
<Dialog.ScrollableInner <Dialog.ScrollableInner
accessibilityLabelledBy={titleId} accessibilityLabelledBy={titleId}
accessibilityDescribedBy={descriptionId} accessibilityDescribedBy={descriptionId}
style={[gtMobile ? {width: 'auto', maxWidth: 400} : a.w_full]}> style={[
gtMobile ? {width: 'auto', maxWidth: 400, minWidth: 200} : a.w_full,
]}>
{children} {children}
</Dialog.ScrollableInner> </Dialog.ScrollableInner>
</Context.Provider> </Context.Provider>

View File

@ -37,12 +37,12 @@ export function MutedWordsDialog() {
return ( return (
<Dialog.Outer control={control}> <Dialog.Outer control={control}>
<Dialog.Handle /> <Dialog.Handle />
<MutedWordsInner control={control} /> <MutedWordsInner />
</Dialog.Outer> </Dialog.Outer>
) )
} }
function MutedWordsInner({}: {control: Dialog.DialogOuterProps['control']}) { function MutedWordsInner() {
const t = useTheme() const t = useTheme()
const {_} = useLingui() const {_} = useLingui()
const {gtMobile} = useBreakpoints() const {gtMobile} = useBreakpoints()

View File

@ -59,6 +59,7 @@ import * as Toast from '../util/Toast'
import {UserAvatar} from '../util/UserAvatar' import {UserAvatar} from '../util/UserAvatar'
import {CharProgress} from './char-progress/CharProgress' import {CharProgress} from './char-progress/CharProgress'
import {ExternalEmbed} from './ExternalEmbed' import {ExternalEmbed} from './ExternalEmbed'
import {GifAltText} from './GifAltText'
import {LabelsBtn} from './labels/LabelsBtn' import {LabelsBtn} from './labels/LabelsBtn'
import {Gallery} from './photos/Gallery' import {Gallery} from './photos/Gallery'
import {OpenCameraBtn} from './photos/OpenCameraBtn' import {OpenCameraBtn} from './photos/OpenCameraBtn'
@ -327,7 +328,7 @@ export const ComposePost = observer(function ComposePost({
image: gif.media_formats.preview.url, image: gif.media_formats.preview.url,
likelyType: LikelyType.HTML, likelyType: LikelyType.HTML,
title: gif.content_description, title: gif.content_description,
description: `ALT: ${gif.content_description}`, description: '',
}, },
}) })
setExtGif(gif) setExtGif(gif)
@ -335,6 +336,26 @@ export const ComposePost = observer(function ComposePost({
[setExtLink], [setExtLink],
) )
const handleChangeGifAltText = useCallback(
(altText: string) => {
setExtLink(ext =>
ext && ext.meta
? {
...ext,
meta: {
...ext?.meta,
description:
altText.trim().length === 0
? ''
: `Alt text: ${altText.trim()}`,
},
}
: ext,
)
},
[setExtLink],
)
return ( return (
<KeyboardAvoidingView <KeyboardAvoidingView
testID="composePostView" testID="composePostView"
@ -474,6 +495,7 @@ export const ComposePost = observer(function ComposePost({
<Gallery gallery={gallery} /> <Gallery gallery={gallery} />
{gallery.isEmpty && extLink && ( {gallery.isEmpty && extLink && (
<View style={a.relative}>
<ExternalEmbed <ExternalEmbed
link={extLink} link={extLink}
gif={extGif} gif={extGif}
@ -482,6 +504,12 @@ export const ComposePost = observer(function ComposePost({
setExtGif(undefined) setExtGif(undefined)
}} }}
/> />
<GifAltText
link={extLink}
gif={extGif}
onSubmit={handleChangeGifAltText}
/>
</View>
)} )}
{quote ? ( {quote ? (
<View style={[s.mt5, isWeb && s.mb10]}> <View style={[s.mt5, isWeb && s.mb10]}>

View File

@ -46,7 +46,12 @@ export const ExternalEmbed = ({
: undefined : undefined
return ( return (
<View style={[a.mb_xl, a.overflow_hidden, t.atoms.border_contrast_medium]}> <View
style={[
!gif && a.mb_xl,
a.overflow_hidden,
t.atoms.border_contrast_medium,
]}>
{link.isLoading ? ( {link.isLoading ? (
<Container style={loadingStyle}> <Container style={loadingStyle}>
<Loader size="xl" /> <Loader size="xl" />
@ -62,7 +67,7 @@ export const ExternalEmbed = ({
</Container> </Container>
) : linkInfo ? ( ) : linkInfo ? (
<View style={{pointerEvents: !gif ? 'none' : 'auto'}}> <View style={{pointerEvents: !gif ? 'none' : 'auto'}}>
<ExternalLinkEmbed link={linkInfo} /> <ExternalLinkEmbed link={linkInfo} hideAlt />
</View> </View>
) : null} ) : null}
<TouchableOpacity <TouchableOpacity

View File

@ -0,0 +1,177 @@
import React, {useCallback, useState} from 'react'
import {TouchableOpacity, View} from 'react-native'
import {AppBskyEmbedExternal} from '@atproto/api'
import {msg, Trans} from '@lingui/macro'
import {useLingui} from '@lingui/react'
import {ExternalEmbedDraft} from '#/lib/api'
import {HITSLOP_10, MAX_ALT_TEXT} from '#/lib/constants'
import {
EmbedPlayerParams,
parseEmbedPlayerFromUrl,
} from '#/lib/strings/embed-player'
import {enforceLen} from '#/lib/strings/helpers'
import {isAndroid} from '#/platform/detection'
import {Gif} from '#/state/queries/tenor'
import {atoms as a, native, useTheme} from '#/alf'
import {Button, ButtonText} from '#/components/Button'
import * as Dialog from '#/components/Dialog'
import * as TextField from '#/components/forms/TextField'
import {Check_Stroke2_Corner0_Rounded as Check} from '#/components/icons/Check'
import {PlusSmall_Stroke2_Corner0_Rounded as Plus} from '#/components/icons/Plus'
import {Text} from '#/components/Typography'
import {GifEmbed} from '../util/post-embeds/GifEmbed'
import {AltTextReminder} from './photos/Gallery'
export function GifAltText({
link: linkProp,
gif,
onSubmit,
}: {
link: ExternalEmbedDraft
gif?: Gif
onSubmit: (alt: string) => void
}) {
const control = Dialog.useDialogControl()
const {_} = useLingui()
const t = useTheme()
const {link, params} = React.useMemo(() => {
return {
link: {
title: linkProp.meta?.title ?? linkProp.uri,
uri: linkProp.uri,
description: linkProp.meta?.description ?? '',
thumb: linkProp.localThumb?.path,
},
params: parseEmbedPlayerFromUrl(linkProp.uri),
}
}, [linkProp])
const onPressSubmit = useCallback(
(alt: string) => {
control.close(() => {
onSubmit(alt)
})
},
[onSubmit, control],
)
if (!gif || !params) return null
return (
<>
<TouchableOpacity
testID="altTextButton"
accessibilityRole="button"
accessibilityLabel={_(msg`Add alt text`)}
accessibilityHint=""
hitSlop={HITSLOP_10}
onPress={control.open}
style={[
a.absolute,
{top: 20, left: 12},
{borderRadius: 6},
a.pl_xs,
a.pr_sm,
a.py_2xs,
a.flex_row,
a.gap_xs,
a.align_center,
{backgroundColor: 'rgba(0, 0, 0, 0.75)'},
]}>
{link.description ? (
<Check size="xs" fill={t.palette.white} style={a.ml_xs} />
) : (
<Plus size="sm" fill={t.palette.white} />
)}
<Text
style={[a.font_bold, {color: t.palette.white}]}
accessible={false}>
<Trans>ALT</Trans>
</Text>
</TouchableOpacity>
<AltTextReminder />
<Dialog.Outer
control={control}
nativeOptions={isAndroid ? {sheet: {snapPoints: ['100%']}} : {}}>
<Dialog.Handle />
<AltTextInner
onSubmit={onPressSubmit}
link={link}
params={params}
initalValue={link.description.replace('Alt text: ', '')}
key={link.uri}
/>
</Dialog.Outer>
</>
)
}
function AltTextInner({
onSubmit,
link,
params,
initalValue,
}: {
onSubmit: (text: string) => void
link: AppBskyEmbedExternal.ViewExternal
params: EmbedPlayerParams
initalValue: string
}) {
const {_} = useLingui()
const [altText, setAltText] = useState(initalValue)
const onPressSubmit = useCallback(() => {
onSubmit(altText)
}, [onSubmit, altText])
return (
<Dialog.ScrollableInner label={_(msg`Add alt text`)}>
<View style={a.flex_col_reverse}>
<View style={[a.mt_md, a.gap_md]}>
<View>
<TextField.LabelText>
<Trans>Descriptive alt text</Trans>
</TextField.LabelText>
<TextField.Root>
<Dialog.Input
label={_(msg`Alt text`)}
placeholder={link.title}
onChangeText={text =>
setAltText(enforceLen(text, MAX_ALT_TEXT))
}
value={altText}
multiline
numberOfLines={3}
autoFocus
/>
</TextField.Root>
</View>
<Button
label={_(msg`Save`)}
size="medium"
color="primary"
variant="solid"
onPress={onPressSubmit}>
<ButtonText>
<Trans>Save</Trans>
</ButtonText>
</Button>
</View>
{/* below the text input to force tab order */}
<View>
<Text style={[a.text_2xl, a.font_bold, a.leading_tight, a.pb_sm]}>
<Trans>Add ALT text</Trans>
</Text>
<View style={[a.w_full, a.align_center, native({maxHeight: 200})]}>
<GifEmbed link={link} params={params} hideAlt />
</View>
</View>
</View>
<Dialog.Close />
</Dialog.ScrollableInner>
)
}

View File

@ -1,19 +1,20 @@
import React, {useState} from 'react' import React, {useState} from 'react'
import {ImageStyle, Keyboard, LayoutChangeEvent} from 'react-native' import {ImageStyle, Keyboard, LayoutChangeEvent} from 'react-native'
import {GalleryModel} from 'state/models/media/gallery'
import {observer} from 'mobx-react-lite'
import {FontAwesomeIcon} from '@fortawesome/react-native-fontawesome'
import {s, colors} from 'lib/styles'
import {StyleSheet, TouchableOpacity, View} from 'react-native' import {StyleSheet, TouchableOpacity, View} from 'react-native'
import {Image} from 'expo-image' import {Image} from 'expo-image'
import {Text} from 'view/com/util/text/Text' import {FontAwesomeIcon} from '@fortawesome/react-native-fontawesome'
import {Dimensions} from 'lib/media/types' import {msg, Trans} from '@lingui/macro'
import {usePalette} from 'lib/hooks/usePalette'
import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries'
import {Trans, msg} from '@lingui/macro'
import {useLingui} from '@lingui/react' import {useLingui} from '@lingui/react'
import {observer} from 'mobx-react-lite'
import {useModalControls} from '#/state/modals' import {useModalControls} from '#/state/modals'
import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries'
import {Dimensions} from 'lib/media/types'
import {colors, s} from 'lib/styles'
import {isNative} from 'platform/detection' import {isNative} from 'platform/detection'
import {GalleryModel} from 'state/models/media/gallery'
import {Text} from 'view/com/util/text/Text'
import {useTheme} from '#/alf'
const IMAGE_GAP = 8 const IMAGE_GAP = 8
@ -49,10 +50,10 @@ const GalleryInner = observer(function GalleryImpl({
gallery, gallery,
containerInfo, containerInfo,
}: GalleryInnerProps) { }: GalleryInnerProps) {
const pal = usePalette('default')
const {_} = useLingui() const {_} = useLingui()
const {isMobile} = useWebMediaQueries() const {isMobile} = useWebMediaQueries()
const {openModal} = useModalControls() const {openModal} = useModalControls()
const t = useTheme()
let side: number let side: number
@ -126,16 +127,22 @@ const GalleryInner = observer(function GalleryImpl({
}) })
}} }}
style={[styles.altTextControl, altTextControlStyle]}> style={[styles.altTextControl, altTextControlStyle]}>
<Text style={styles.altTextControlLabel} accessible={false}>
<Trans>ALT</Trans>
</Text>
{image.altText.length > 0 ? ( {image.altText.length > 0 ? (
<FontAwesomeIcon <FontAwesomeIcon
icon="check" icon="check"
size={10} size={10}
style={{color: colors.green3}} style={{color: t.palette.white}}
/> />
) : undefined} ) : (
<FontAwesomeIcon
icon="plus"
size={10}
style={{color: t.palette.white}}
/>
)}
<Text style={styles.altTextControlLabel} accessible={false}>
<Trans>ALT</Trans>
</Text>
</TouchableOpacity> </TouchableOpacity>
<View style={imageControlsStyle}> <View style={imageControlsStyle}>
<TouchableOpacity <TouchableOpacity
@ -201,20 +208,27 @@ const GalleryInner = observer(function GalleryImpl({
</View> </View>
))} ))}
</View> </View>
<AltTextReminder />
</>
) : null
})
export function AltTextReminder() {
const t = useTheme()
return (
<View style={[styles.reminder]}> <View style={[styles.reminder]}>
<View style={[styles.infoIcon, pal.viewLight]}> <View style={[styles.infoIcon, t.atoms.bg_contrast_25]}>
<FontAwesomeIcon icon="info" size={12} color={pal.colors.text} /> <FontAwesomeIcon icon="info" size={12} color={t.atoms.text.color} />
</View> </View>
<Text type="sm" style={[pal.textLight, s.flex1]}> <Text type="sm" style={[t.atoms.text_contrast_medium, s.flex1]}>
<Trans> <Trans>
Alt text describes images for blind and low-vision users, and helps Alt text describes images for blind and low-vision users, and helps
give context to everyone. give context to everyone.
</Trans> </Trans>
</Text> </Text>
</View> </View>
</> )
) : null }
})
const styles = StyleSheet.create({ const styles = StyleSheet.create({
gallery: { gallery: {
@ -244,6 +258,7 @@ const styles = StyleSheet.create({
paddingVertical: 3, paddingVertical: 3,
flexDirection: 'row', flexDirection: 'row',
alignItems: 'center', alignItems: 'center',
gap: 4,
}, },
altTextControlLabel: { altTextControlLabel: {
color: 'white', color: 'white',

View File

@ -1,9 +1,10 @@
import {AppBskyEmbedImages} from '@atproto/api'
import React, {ComponentProps, FC} from 'react' import React, {ComponentProps, FC} from 'react'
import {StyleSheet, Text, Pressable, View} from 'react-native' import {Pressable, StyleSheet, Text, View} from 'react-native'
import {Image} from 'expo-image' import {Image} from 'expo-image'
import {AppBskyEmbedImages} from '@atproto/api'
import {msg} from '@lingui/macro' import {msg} from '@lingui/macro'
import {useLingui} from '@lingui/react' import {useLingui} from '@lingui/react'
import {isWeb} from 'platform/detection' import {isWeb} from 'platform/detection'
type EventFunction = (index: number) => void type EventFunction = (index: number) => void

View File

@ -20,9 +20,11 @@ import {Text} from '../text/Text'
export const ExternalLinkEmbed = ({ export const ExternalLinkEmbed = ({
link, link,
style, style,
hideAlt,
}: { }: {
link: AppBskyEmbedExternal.ViewExternal link: AppBskyEmbedExternal.ViewExternal
style?: StyleProp<ViewStyle> style?: StyleProp<ViewStyle>
hideAlt?: boolean
}) => { }) => {
const pal = usePalette('default') const pal = usePalette('default')
const {isMobile} = useWebMediaQueries() const {isMobile} = useWebMediaQueries()
@ -37,7 +39,7 @@ export const ExternalLinkEmbed = ({
}, [link.uri, externalEmbedPrefs]) }, [link.uri, externalEmbedPrefs])
if (embedPlayerParams?.source === 'tenor') { if (embedPlayerParams?.source === 'tenor') {
return <GifEmbed params={embedPlayerParams} link={link} /> return <GifEmbed params={embedPlayerParams} link={link} hideAlt={hideAlt} />
} }
return ( return (

View File

@ -1,14 +1,18 @@
import React from 'react' import React from 'react'
import {Pressable, View} from 'react-native' import {Pressable, StyleSheet, TouchableOpacity, View} from 'react-native'
import {AppBskyEmbedExternal} from '@atproto/api' import {AppBskyEmbedExternal} from '@atproto/api'
import {FontAwesomeIcon} from '@fortawesome/react-native-fontawesome' import {FontAwesomeIcon} from '@fortawesome/react-native-fontawesome'
import {msg} from '@lingui/macro' import {msg, Trans} from '@lingui/macro'
import {useLingui} from '@lingui/react' import {useLingui} from '@lingui/react'
import {HITSLOP_10} from '#/lib/constants'
import {isWeb} from '#/platform/detection'
import {EmbedPlayerParams} from 'lib/strings/embed-player' import {EmbedPlayerParams} from 'lib/strings/embed-player'
import {useAutoplayDisabled} from 'state/preferences' import {useAutoplayDisabled} from 'state/preferences'
import {atoms as a, useTheme} from '#/alf' import {atoms as a, useTheme} from '#/alf'
import {Loader} from '#/components/Loader' import {Loader} from '#/components/Loader'
import * as Prompt from '#/components/Prompt'
import {Text} from '#/components/Typography'
import {GifView} from '../../../../../modules/expo-bluesky-gif-view' import {GifView} from '../../../../../modules/expo-bluesky-gif-view'
import {GifViewStateChangeEvent} from '../../../../../modules/expo-bluesky-gif-view/src/GifView.types' import {GifViewStateChangeEvent} from '../../../../../modules/expo-bluesky-gif-view/src/GifView.types'
@ -82,9 +86,11 @@ function PlaybackControls({
export function GifEmbed({ export function GifEmbed({
params, params,
link, link,
hideAlt,
}: { }: {
params: EmbedPlayerParams params: EmbedPlayerParams
link: AppBskyEmbedExternal.ViewExternal link: AppBskyEmbedExternal.ViewExternal
hideAlt?: boolean
}) { }) {
const {_} = useLingui() const {_} = useLingui()
const autoplayDisabled = useAutoplayDisabled() const autoplayDisabled = useAutoplayDisabled()
@ -111,7 +117,8 @@ export function GifEmbed({
}, []) }, [])
return ( return (
<View style={[a.rounded_sm, a.overflow_hidden, a.mt_sm]}> <View
style={[a.rounded_sm, a.overflow_hidden, a.mt_sm, {maxWidth: '100%'}]}>
<View <View
style={[ style={[
a.rounded_sm, a.rounded_sm,
@ -133,9 +140,69 @@ export function GifEmbed({
onPlayerStateChange={onPlayerStateChange} onPlayerStateChange={onPlayerStateChange}
ref={playerRef} ref={playerRef}
accessibilityHint={_(msg`Animated GIF`)} accessibilityHint={_(msg`Animated GIF`)}
accessibilityLabel={link.description.replace('ALT: ', '')} accessibilityLabel={link.description.replace('Alt text: ', '')}
/> />
{!hideAlt && link.description.startsWith('Alt text: ') && (
<AltText text={link.description.replace('Alt text: ', '')} />
)}
</View> </View>
</View> </View>
) )
} }
function AltText({text}: {text: string}) {
const control = Prompt.usePromptControl()
const {_} = useLingui()
return (
<>
<TouchableOpacity
testID="altTextButton"
accessibilityRole="button"
accessibilityLabel={_(msg`Show alt text`)}
accessibilityHint=""
hitSlop={HITSLOP_10}
onPress={control.open}
style={styles.altContainer}>
<Text style={styles.alt} accessible={false}>
<Trans>ALT</Trans>
</Text>
</TouchableOpacity>
<Prompt.Outer control={control}>
<Prompt.TitleText>
<Trans>Alt Text</Trans>
</Prompt.TitleText>
<Prompt.DescriptionText>{text}</Prompt.DescriptionText>
<Prompt.Actions>
<Prompt.Action
onPress={control.close}
cta={_(msg`Close`)}
color="secondary"
/>
</Prompt.Actions>
</Prompt.Outer>
</>
)
}
const styles = StyleSheet.create({
altContainer: {
backgroundColor: 'rgba(0, 0, 0, 0.75)',
borderRadius: 6,
paddingHorizontal: 6,
paddingVertical: 3,
position: 'absolute',
// Related to margin/gap hack. This keeps the alt label in the same position
// on all platforms
left: isWeb ? 8 : 5,
bottom: isWeb ? 8 : 5,
zIndex: 2,
},
alt: {
color: 'white',
fontSize: 10,
fontWeight: 'bold',
},
})