Merge branch 'bluesky-social:main' into zh
This commit is contained in:
		
						commit
						a33e370d22
					
				
					 5 changed files with 323 additions and 160 deletions
				
			
		|  | @ -84,6 +84,7 @@ export const createHitslop = (size: number): Insets => ({ | ||||||
| export const HITSLOP_10 = createHitslop(10) | export const HITSLOP_10 = createHitslop(10) | ||||||
| export const HITSLOP_20 = createHitslop(20) | export const HITSLOP_20 = createHitslop(20) | ||||||
| export const HITSLOP_30 = createHitslop(30) | export const HITSLOP_30 = createHitslop(30) | ||||||
|  | export const POST_CTRL_HITSLOP = {top: 5, bottom: 10, left: 10, right: 10} | ||||||
| export const BACK_HITSLOP = HITSLOP_30 | export const BACK_HITSLOP = HITSLOP_30 | ||||||
| export const MAX_POST_LINES = 25 | export const MAX_POST_LINES = 25 | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -25,7 +25,7 @@ import {sanitizeHandle} from 'lib/strings/handles' | ||||||
| import {countLines} from 'lib/strings/helpers' | import {countLines} from 'lib/strings/helpers' | ||||||
| import {niceDate} from 'lib/strings/time' | import {niceDate} from 'lib/strings/time' | ||||||
| import {s} from 'lib/styles' | import {s} from 'lib/styles' | ||||||
| import {isNative, isWeb} from 'platform/detection' | import {isWeb} from 'platform/detection' | ||||||
| import {useSession} from 'state/session' | import {useSession} from 'state/session' | ||||||
| import {PostThreadFollowBtn} from 'view/com/post-thread/PostThreadFollowBtn' | import {PostThreadFollowBtn} from 'view/com/post-thread/PostThreadFollowBtn' | ||||||
| import {atoms as a} from '#/alf' | import {atoms as a} from '#/alf' | ||||||
|  | @ -35,7 +35,7 @@ import {LabelsOnMyPost} from '../../../components/moderation/LabelsOnMe' | ||||||
| import {PostAlerts} from '../../../components/moderation/PostAlerts' | import {PostAlerts} from '../../../components/moderation/PostAlerts' | ||||||
| import {PostHider} from '../../../components/moderation/PostHider' | import {PostHider} from '../../../components/moderation/PostHider' | ||||||
| import {getTranslatorLink, isPostInLanguage} from '../../../locale/helpers' | import {getTranslatorLink, isPostInLanguage} from '../../../locale/helpers' | ||||||
| import {WhoCanReply} from '../threadgate/WhoCanReply' | import {WhoCanReplyBlock, WhoCanReplyInline} from '../threadgate/WhoCanReply' | ||||||
| import {ErrorMessage} from '../util/error/ErrorMessage' | import {ErrorMessage} from '../util/error/ErrorMessage' | ||||||
| import {Link, TextLink} from '../util/Link' | import {Link, TextLink} from '../util/Link' | ||||||
| import {formatCount} from '../util/numeric/format' | import {formatCount} from '../util/numeric/format' | ||||||
|  | @ -340,6 +340,7 @@ let PostThreadItemLoaded = ({ | ||||||
|             </ContentHider> |             </ContentHider> | ||||||
|             <ExpandedPostDetails |             <ExpandedPostDetails | ||||||
|               post={post} |               post={post} | ||||||
|  |               isThreadAuthor={isThreadAuthor} | ||||||
|               translatorUrl={translatorUrl} |               translatorUrl={translatorUrl} | ||||||
|               needsTranslation={needsTranslation} |               needsTranslation={needsTranslation} | ||||||
|             /> |             /> | ||||||
|  | @ -396,11 +397,6 @@ let PostThreadItemLoaded = ({ | ||||||
|             </View> |             </View> | ||||||
|           </View> |           </View> | ||||||
|         </View> |         </View> | ||||||
|         <WhoCanReply |  | ||||||
|           post={post} |  | ||||||
|           isThreadAuthor={isThreadAuthor} |  | ||||||
|           style={{borderBottomWidth: isNative ? 1 : 0}} |  | ||||||
|         /> |  | ||||||
|       </> |       </> | ||||||
|     ) |     ) | ||||||
|   } else { |   } else { | ||||||
|  | @ -579,14 +575,7 @@ let PostThreadItemLoaded = ({ | ||||||
|             ) : undefined} |             ) : undefined} | ||||||
|           </PostHider> |           </PostHider> | ||||||
|         </PostOuterWrapper> |         </PostOuterWrapper> | ||||||
|         <WhoCanReply |         <WhoCanReplyBlock post={post} isThreadAuthor={isThreadAuthor} /> | ||||||
|           post={post} |  | ||||||
|           style={{ |  | ||||||
|             marginTop: 4, |  | ||||||
|             borderBottomWidth: 1, |  | ||||||
|           }} |  | ||||||
|           isThreadAuthor={isThreadAuthor} |  | ||||||
|         /> |  | ||||||
|       </> |       </> | ||||||
|     ) |     ) | ||||||
|   } |   } | ||||||
|  | @ -654,10 +643,12 @@ function PostOuterWrapper({ | ||||||
| 
 | 
 | ||||||
| function ExpandedPostDetails({ | function ExpandedPostDetails({ | ||||||
|   post, |   post, | ||||||
|  |   isThreadAuthor, | ||||||
|   needsTranslation, |   needsTranslation, | ||||||
|   translatorUrl, |   translatorUrl, | ||||||
| }: { | }: { | ||||||
|   post: AppBskyFeedDefs.PostView |   post: AppBskyFeedDefs.PostView | ||||||
|  |   isThreadAuthor: boolean | ||||||
|   needsTranslation: boolean |   needsTranslation: boolean | ||||||
|   translatorUrl: string |   translatorUrl: string | ||||||
| }) { | }) { | ||||||
|  | @ -670,14 +661,23 @@ function ExpandedPostDetails({ | ||||||
|   }, [openLink, translatorUrl]) |   }, [openLink, translatorUrl]) | ||||||
| 
 | 
 | ||||||
|   return ( |   return ( | ||||||
|     <View style={[s.flexRow, s.mt2, s.mb10]}> |     <View | ||||||
|       <Text style={pal.textLight}>{niceDate(post.indexedAt)}</Text> |       style={[ | ||||||
|  |         a.flex_row, | ||||||
|  |         a.align_center, | ||||||
|  |         a.flex_wrap, | ||||||
|  |         a.gap_sm, | ||||||
|  |         s.mt2, | ||||||
|  |         s.mb10, | ||||||
|  |       ]}> | ||||||
|  |       <Text style={[a.text_sm, pal.textLight]}>{niceDate(post.indexedAt)}</Text> | ||||||
|  |       <WhoCanReplyInline post={post} isThreadAuthor={isThreadAuthor} /> | ||||||
|       {needsTranslation && ( |       {needsTranslation && ( | ||||||
|         <> |         <> | ||||||
|           <Text style={pal.textLight}> · </Text> |           <Text style={[a.text_sm, pal.textLight]}>·</Text> | ||||||
| 
 | 
 | ||||||
|           <Text |           <Text | ||||||
|             style={pal.link} |             style={[a.text_sm, pal.link]} | ||||||
|             title={_(msg`Translate`)} |             title={_(msg`Translate`)} | ||||||
|             onPress={onTranslatePress}> |             onPress={onTranslatePress}> | ||||||
|             <Trans>Translate</Trans> |             <Trans>Translate</Trans> | ||||||
|  |  | ||||||
|  | @ -1,16 +1,20 @@ | ||||||
| import React from 'react' | import React from 'react' | ||||||
| import {Keyboard, StyleProp, View, ViewStyle} from 'react-native' | import {Keyboard, StyleProp, View, ViewStyle} from 'react-native' | ||||||
| import {AppBskyFeedDefs, AppBskyGraphDefs, AtUri} from '@atproto/api' | import { | ||||||
|  |   AppBskyFeedDefs, | ||||||
|  |   AppBskyFeedGetPostThread, | ||||||
|  |   AppBskyGraphDefs, | ||||||
|  |   AtUri, | ||||||
|  |   BskyAgent, | ||||||
|  | } from '@atproto/api' | ||||||
| import {msg, Trans} from '@lingui/macro' | import {msg, Trans} from '@lingui/macro' | ||||||
| import {useLingui} from '@lingui/react' | import {useLingui} from '@lingui/react' | ||||||
| import {useQueryClient} from '@tanstack/react-query' | import {useQueryClient} from '@tanstack/react-query' | ||||||
| 
 | 
 | ||||||
| import {useAnalytics} from '#/lib/analytics/analytics' |  | ||||||
| import {createThreadgate} from '#/lib/api' | import {createThreadgate} from '#/lib/api' | ||||||
| import {useColorSchemeStyle} from '#/lib/hooks/useColorSchemeStyle' | import {until} from '#/lib/async/until' | ||||||
| import {usePalette} from '#/lib/hooks/usePalette' | import {HITSLOP_10} from '#/lib/constants' | ||||||
| import {makeListLink, makeProfileLink} from '#/lib/routes/links' | import {makeListLink, makeProfileLink} from '#/lib/routes/links' | ||||||
| import {colors} from '#/lib/styles' |  | ||||||
| import {logger} from '#/logger' | import {logger} from '#/logger' | ||||||
| import {isNative} from '#/platform/detection' | import {isNative} from '#/platform/detection' | ||||||
| import {useModalControls} from '#/state/modals' | import {useModalControls} from '#/state/modals' | ||||||
|  | @ -21,84 +25,31 @@ import { | ||||||
| } from '#/state/queries/threadgate' | } from '#/state/queries/threadgate' | ||||||
| import {useAgent} from '#/state/session' | import {useAgent} from '#/state/session' | ||||||
| import * as Toast from 'view/com/util/Toast' | import * as Toast from 'view/com/util/Toast' | ||||||
|  | import {atoms as a, useTheme} from '#/alf' | ||||||
| import {Button} from '#/components/Button' | import {Button} from '#/components/Button' | ||||||
|  | import * as Dialog from '#/components/Dialog' | ||||||
|  | import {useDialogControl} from '#/components/Dialog' | ||||||
|  | import {CircleBanSign_Stroke2_Corner0_Rounded as CircleBanSign} from '#/components/icons/CircleBanSign' | ||||||
|  | import {Earth_Stroke2_Corner0_Rounded as Earth} from '#/components/icons/Globe' | ||||||
|  | import {Group3_Stroke2_Corner0_Rounded as Group} from '#/components/icons/Group' | ||||||
|  | import {Text} from '#/components/Typography' | ||||||
| import {TextLink} from '../util/Link' | import {TextLink} from '../util/Link' | ||||||
| import {Text} from '../util/text/Text' |  | ||||||
| 
 | 
 | ||||||
| export function WhoCanReply({ | interface WhoCanReplyProps { | ||||||
|   post, |  | ||||||
|   isThreadAuthor, |  | ||||||
|   style, |  | ||||||
| }: { |  | ||||||
|   post: AppBskyFeedDefs.PostView |   post: AppBskyFeedDefs.PostView | ||||||
|   isThreadAuthor: boolean |   isThreadAuthor: boolean | ||||||
|   style?: StyleProp<ViewStyle> |   style?: StyleProp<ViewStyle> | ||||||
| }) { | } | ||||||
|   const {track} = useAnalytics() |  | ||||||
|   const {_} = useLingui() |  | ||||||
|   const pal = usePalette('default') |  | ||||||
|   const agent = useAgent() |  | ||||||
|   const queryClient = useQueryClient() |  | ||||||
|   const {openModal} = useModalControls() |  | ||||||
|   const containerStyles = useColorSchemeStyle( |  | ||||||
|     { |  | ||||||
|       backgroundColor: pal.colors.unreadNotifBg, |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       backgroundColor: pal.colors.unreadNotifBg, |  | ||||||
|     }, |  | ||||||
|   ) |  | ||||||
|   const textStyles = useColorSchemeStyle( |  | ||||||
|     {color: colors.blue5}, |  | ||||||
|     {color: colors.blue1}, |  | ||||||
|   ) |  | ||||||
|   const hoverStyles = useColorSchemeStyle( |  | ||||||
|     { |  | ||||||
|       backgroundColor: colors.white, |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       backgroundColor: pal.colors.background, |  | ||||||
|     }, |  | ||||||
|   ) |  | ||||||
|   const settings = React.useMemo( |  | ||||||
|     () => threadgateViewToSettings(post.threadgate), |  | ||||||
|     [post], |  | ||||||
|   ) |  | ||||||
|   const isRootPost = !('reply' in post.record) |  | ||||||
| 
 | 
 | ||||||
|   const onPressEdit = () => { | export function WhoCanReplyInline({ | ||||||
|     track('Post:EditThreadgateOpened') |   post, | ||||||
|     if (isNative && Keyboard.isVisible()) { |   isThreadAuthor, | ||||||
|       Keyboard.dismiss() |   style, | ||||||
|     } | }: WhoCanReplyProps) { | ||||||
|     openModal({ |   const {_} = useLingui() | ||||||
|       name: 'threadgate', |   const t = useTheme() | ||||||
|       settings, |   const infoDialogControl = useDialogControl() | ||||||
|       async onConfirm(newSettings: ThreadgateSetting[]) { |   const {settings, isRootPost, onPressEdit} = useWhoCanReply(post) | ||||||
|         try { |  | ||||||
|           if (newSettings.length) { |  | ||||||
|             await createThreadgate(agent, post.uri, newSettings) |  | ||||||
|           } else { |  | ||||||
|             await agent.api.com.atproto.repo.deleteRecord({ |  | ||||||
|               repo: agent.session!.did, |  | ||||||
|               collection: 'app.bsky.feed.threadgate', |  | ||||||
|               rkey: new AtUri(post.uri).rkey, |  | ||||||
|             }) |  | ||||||
|           } |  | ||||||
|           Toast.show('Thread settings updated') |  | ||||||
|           queryClient.invalidateQueries({ |  | ||||||
|             queryKey: [POST_THREAD_RQKEY_ROOT], |  | ||||||
|           }) |  | ||||||
|           track('Post:ThreadgateEdited') |  | ||||||
|         } catch (err) { |  | ||||||
|           Toast.show( |  | ||||||
|             'There was an issue. Please check your internet connection and try again.', |  | ||||||
|           ) |  | ||||||
|           logger.error('Failed to edit threadgate', {message: err}) |  | ||||||
|         } |  | ||||||
|       }, |  | ||||||
|     }) |  | ||||||
|   } |  | ||||||
| 
 | 
 | ||||||
|   if (!isRootPost) { |   if (!isRootPost) { | ||||||
|     return null |     return null | ||||||
|  | @ -107,28 +58,183 @@ export function WhoCanReply({ | ||||||
|     return null |     return null | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|  |   const isEverybody = settings.length === 0 | ||||||
|  |   const isNobody = !!settings.find(gate => gate.type === 'nobody') | ||||||
|  |   const description = isEverybody | ||||||
|  |     ? _(msg`Everybody can reply`) | ||||||
|  |     : isNobody | ||||||
|  |     ? _(msg`Replies disabled`) | ||||||
|  |     : _(msg`Some people can reply`) | ||||||
|  | 
 | ||||||
|   return ( |   return ( | ||||||
|  |     <> | ||||||
|  |       <Button | ||||||
|  |         label={ | ||||||
|  |           isThreadAuthor ? _(msg`Edit who can reply`) : _(msg`Who can reply`) | ||||||
|  |         } | ||||||
|  |         onPress={isThreadAuthor ? onPressEdit : infoDialogControl.open} | ||||||
|  |         hitSlop={HITSLOP_10}> | ||||||
|  |         {({hovered}) => ( | ||||||
|  |           <View style={[a.flex_row, a.align_center, a.gap_xs, style]}> | ||||||
|  |             <Icon | ||||||
|  |               color={t.palette.contrast_400} | ||||||
|  |               width={16} | ||||||
|  |               settings={settings} | ||||||
|  |             /> | ||||||
|  |             <Text | ||||||
|  |               style={[ | ||||||
|  |                 a.text_sm, | ||||||
|  |                 a.leading_tight, | ||||||
|  |                 t.atoms.text_contrast_medium, | ||||||
|  |                 hovered && a.underline, | ||||||
|  |               ]}> | ||||||
|  |               {description} | ||||||
|  |             </Text> | ||||||
|  |           </View> | ||||||
|  |         )} | ||||||
|  |       </Button> | ||||||
|  |       <InfoDialog control={infoDialogControl} post={post} settings={settings} /> | ||||||
|  |     </> | ||||||
|  |   ) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function WhoCanReplyBlock({ | ||||||
|  |   post, | ||||||
|  |   isThreadAuthor, | ||||||
|  |   style, | ||||||
|  | }: WhoCanReplyProps) { | ||||||
|  |   const {_} = useLingui() | ||||||
|  |   const t = useTheme() | ||||||
|  |   const infoDialogControl = useDialogControl() | ||||||
|  |   const {settings, isRootPost, onPressEdit} = useWhoCanReply(post) | ||||||
|  | 
 | ||||||
|  |   if (!isRootPost) { | ||||||
|  |     return null | ||||||
|  |   } | ||||||
|  |   if (!settings.length && !isThreadAuthor) { | ||||||
|  |     return null | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   const isEverybody = settings.length === 0 | ||||||
|  |   const isNobody = !!settings.find(gate => gate.type === 'nobody') | ||||||
|  |   const description = isEverybody | ||||||
|  |     ? _(msg`Everybody can reply`) | ||||||
|  |     : isNobody | ||||||
|  |     ? _(msg`Replies on this thread are disabled`) | ||||||
|  |     : _(msg`Some people can reply`) | ||||||
|  | 
 | ||||||
|  |   return ( | ||||||
|  |     <> | ||||||
|  |       <Button | ||||||
|  |         label={ | ||||||
|  |           isThreadAuthor ? _(msg`Edit who can reply`) : _(msg`Who can reply`) | ||||||
|  |         } | ||||||
|  |         onPress={isThreadAuthor ? onPressEdit : infoDialogControl.open} | ||||||
|  |         hitSlop={HITSLOP_10}> | ||||||
|  |         {({hovered}) => ( | ||||||
|           <View |           <View | ||||||
|             style={[ |             style={[ | ||||||
|         { |               a.flex_1, | ||||||
|           flexDirection: 'row', |               a.flex_row, | ||||||
|           alignItems: 'center', |               a.align_center, | ||||||
|           gap: 10, |               a.py_sm, | ||||||
|           paddingLeft: 18, |               a.pr_lg, | ||||||
|           paddingRight: 14, |  | ||||||
|           paddingVertical: 10, |  | ||||||
|           borderTopWidth: 1, |  | ||||||
|         }, |  | ||||||
|         pal.border, |  | ||||||
|         containerStyles, |  | ||||||
|               style, |               style, | ||||||
|             ]}> |             ]}> | ||||||
|       <View style={{flex: 1, paddingVertical: 6}}> |             <View style={[{paddingLeft: 25, paddingRight: 18}]}> | ||||||
|         <Text type="sm" style={[{flexWrap: 'wrap'}, textStyles]}> |               <Icon color={t.palette.contrast_300} settings={settings} /> | ||||||
|  |             </View> | ||||||
|  |             <Text | ||||||
|  |               style={[ | ||||||
|  |                 a.text_sm, | ||||||
|  |                 a.leading_tight, | ||||||
|  |                 t.atoms.text_contrast_medium, | ||||||
|  |                 hovered && a.underline, | ||||||
|  |               ]}> | ||||||
|  |               {description} | ||||||
|  |             </Text> | ||||||
|  |           </View> | ||||||
|  |         )} | ||||||
|  |       </Button> | ||||||
|  |       <InfoDialog control={infoDialogControl} post={post} settings={settings} /> | ||||||
|  |     </> | ||||||
|  |   ) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | function Icon({ | ||||||
|  |   color, | ||||||
|  |   width, | ||||||
|  |   settings, | ||||||
|  | }: { | ||||||
|  |   color: string | ||||||
|  |   width?: number | ||||||
|  |   settings: ThreadgateSetting[] | ||||||
|  | }) { | ||||||
|  |   const isEverybody = settings.length === 0 | ||||||
|  |   const isNobody = !!settings.find(gate => gate.type === 'nobody') | ||||||
|  |   const IconComponent = isEverybody ? Earth : isNobody ? CircleBanSign : Group | ||||||
|  |   return <IconComponent fill={color} width={width} /> | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | function InfoDialog({ | ||||||
|  |   control, | ||||||
|  |   post, | ||||||
|  |   settings, | ||||||
|  | }: { | ||||||
|  |   control: Dialog.DialogControlProps | ||||||
|  |   post: AppBskyFeedDefs.PostView | ||||||
|  |   settings: ThreadgateSetting[] | ||||||
|  | }) { | ||||||
|  |   return ( | ||||||
|  |     <Dialog.Outer control={control}> | ||||||
|  |       <Dialog.Handle /> | ||||||
|  |       <InfoDialogInner post={post} settings={settings} /> | ||||||
|  |     </Dialog.Outer> | ||||||
|  |   ) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | function InfoDialogInner({ | ||||||
|  |   post, | ||||||
|  |   settings, | ||||||
|  | }: { | ||||||
|  |   post: AppBskyFeedDefs.PostView | ||||||
|  |   settings: ThreadgateSetting[] | ||||||
|  | }) { | ||||||
|  |   const {_} = useLingui() | ||||||
|  |   return ( | ||||||
|  |     <Dialog.ScrollableInner | ||||||
|  |       label={_(msg`Who can reply dialog`)} | ||||||
|  |       style={[{width: 'auto', maxWidth: 400, minWidth: 200}]}> | ||||||
|  |       <View style={[a.gap_sm]}> | ||||||
|  |         <Text style={[a.font_bold, a.text_xl]}> | ||||||
|  |           <Trans>Who can reply?</Trans> | ||||||
|  |         </Text> | ||||||
|  |         <Rules post={post} settings={settings} /> | ||||||
|  |       </View> | ||||||
|  |     </Dialog.ScrollableInner> | ||||||
|  |   ) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | function Rules({ | ||||||
|  |   post, | ||||||
|  |   settings, | ||||||
|  | }: { | ||||||
|  |   post: AppBskyFeedDefs.PostView | ||||||
|  |   settings: ThreadgateSetting[] | ||||||
|  | }) { | ||||||
|  |   const t = useTheme() | ||||||
|  |   return ( | ||||||
|  |     <Text | ||||||
|  |       style={[ | ||||||
|  |         a.text_md, | ||||||
|  |         a.leading_tight, | ||||||
|  |         a.flex_wrap, | ||||||
|  |         t.atoms.text_contrast_medium, | ||||||
|  |       ]}> | ||||||
|       {!settings.length ? ( |       {!settings.length ? ( | ||||||
|             <Trans>Everybody can reply.</Trans> |         <Trans>Everybody can reply</Trans> | ||||||
|       ) : settings[0].type === 'nobody' ? ( |       ) : settings[0].type === 'nobody' ? ( | ||||||
|             <Trans>Replies to this thread are disabled.</Trans> |         <Trans>Replies to this thread are disabled</Trans> | ||||||
|       ) : ( |       ) : ( | ||||||
|         <Trans> |         <Trans> | ||||||
|           Only{' '} |           Only{' '} | ||||||
|  | @ -143,29 +249,10 @@ export function WhoCanReply({ | ||||||
|               <Separator key={`sep-${i}`} i={i} length={settings.length} /> |               <Separator key={`sep-${i}`} i={i} length={settings.length} /> | ||||||
|             </> |             </> | ||||||
|           ))}{' '} |           ))}{' '} | ||||||
|               can reply. |           can reply | ||||||
|         </Trans> |         </Trans> | ||||||
|       )} |       )} | ||||||
|     </Text> |     </Text> | ||||||
|       </View> |  | ||||||
|       {isThreadAuthor && ( |  | ||||||
|         <View> |  | ||||||
|           <Button label={_(msg`Edit`)} onPress={onPressEdit}> |  | ||||||
|             {({hovered}) => ( |  | ||||||
|               <View |  | ||||||
|                 style={[ |  | ||||||
|                   hovered && hoverStyles, |  | ||||||
|                   {paddingVertical: 6, paddingHorizontal: 8, borderRadius: 8}, |  | ||||||
|                 ]}> |  | ||||||
|                 <Text type="sm" style={pal.link}> |  | ||||||
|                   <Trans>Edit</Trans> |  | ||||||
|                 </Text> |  | ||||||
|               </View> |  | ||||||
|             )} |  | ||||||
|           </Button> |  | ||||||
|         </View> |  | ||||||
|       )} |  | ||||||
|     </View> |  | ||||||
|   ) |   ) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -178,7 +265,7 @@ function Rule({ | ||||||
|   post: AppBskyFeedDefs.PostView |   post: AppBskyFeedDefs.PostView | ||||||
|   lists: AppBskyGraphDefs.ListViewBasic[] | undefined |   lists: AppBskyGraphDefs.ListViewBasic[] | undefined | ||||||
| }) { | }) { | ||||||
|   const pal = usePalette('default') |   const t = useTheme() | ||||||
|   if (rule.type === 'mention') { |   if (rule.type === 'mention') { | ||||||
|     return <Trans>mentioned users</Trans> |     return <Trans>mentioned users</Trans> | ||||||
|   } |   } | ||||||
|  | @ -190,7 +277,7 @@ function Rule({ | ||||||
|           type="sm" |           type="sm" | ||||||
|           href={makeProfileLink(post.author)} |           href={makeProfileLink(post.author)} | ||||||
|           text={`@${post.author.handle}`} |           text={`@${post.author.handle}`} | ||||||
|           style={pal.link} |           style={{color: t.palette.primary_500}} | ||||||
|         /> |         /> | ||||||
|       </Trans> |       </Trans> | ||||||
|     ) |     ) | ||||||
|  | @ -205,7 +292,7 @@ function Rule({ | ||||||
|             type="sm" |             type="sm" | ||||||
|             href={makeListLink(listUrip.hostname, listUrip.rkey)} |             href={makeListLink(listUrip.hostname, listUrip.rkey)} | ||||||
|             text={list.name} |             text={list.name} | ||||||
|             style={pal.link} |             style={{color: t.palette.primary_500}} | ||||||
|           />{' '} |           />{' '} | ||||||
|           members |           members | ||||||
|         </Trans> |         </Trans> | ||||||
|  | @ -227,3 +314,78 @@ function Separator({i, length}: {i: number; length: number}) { | ||||||
|   } |   } | ||||||
|   return <>, </> |   return <>, </> | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | function useWhoCanReply(post: AppBskyFeedDefs.PostView) { | ||||||
|  |   const agent = useAgent() | ||||||
|  |   const queryClient = useQueryClient() | ||||||
|  |   const {openModal} = useModalControls() | ||||||
|  | 
 | ||||||
|  |   const settings = React.useMemo( | ||||||
|  |     () => threadgateViewToSettings(post.threadgate), | ||||||
|  |     [post], | ||||||
|  |   ) | ||||||
|  |   const isRootPost = !('reply' in post.record) | ||||||
|  | 
 | ||||||
|  |   const onPressEdit = () => { | ||||||
|  |     if (isNative && Keyboard.isVisible()) { | ||||||
|  |       Keyboard.dismiss() | ||||||
|  |     } | ||||||
|  |     openModal({ | ||||||
|  |       name: 'threadgate', | ||||||
|  |       settings, | ||||||
|  |       async onConfirm(newSettings: ThreadgateSetting[]) { | ||||||
|  |         try { | ||||||
|  |           if (newSettings.length) { | ||||||
|  |             await createThreadgate(agent, post.uri, newSettings) | ||||||
|  |           } else { | ||||||
|  |             await agent.api.com.atproto.repo.deleteRecord({ | ||||||
|  |               repo: agent.session!.did, | ||||||
|  |               collection: 'app.bsky.feed.threadgate', | ||||||
|  |               rkey: new AtUri(post.uri).rkey, | ||||||
|  |             }) | ||||||
|  |           } | ||||||
|  |           await whenAppViewReady(agent, post.uri, res => { | ||||||
|  |             const thread = res.data.thread | ||||||
|  |             if (AppBskyFeedDefs.isThreadViewPost(thread)) { | ||||||
|  |               const fetchedSettings = threadgateViewToSettings( | ||||||
|  |                 thread.post.threadgate, | ||||||
|  |               ) | ||||||
|  |               return ( | ||||||
|  |                 JSON.stringify(fetchedSettings) === JSON.stringify(newSettings) | ||||||
|  |               ) | ||||||
|  |             } | ||||||
|  |             return false | ||||||
|  |           }) | ||||||
|  |           Toast.show('Thread settings updated') | ||||||
|  |           queryClient.invalidateQueries({ | ||||||
|  |             queryKey: [POST_THREAD_RQKEY_ROOT], | ||||||
|  |           }) | ||||||
|  |         } catch (err) { | ||||||
|  |           Toast.show( | ||||||
|  |             'There was an issue. Please check your internet connection and try again.', | ||||||
|  |           ) | ||||||
|  |           logger.error('Failed to edit threadgate', {message: err}) | ||||||
|  |         } | ||||||
|  |       }, | ||||||
|  |     }) | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   return {settings, isRootPost, onPressEdit} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | async function whenAppViewReady( | ||||||
|  |   agent: BskyAgent, | ||||||
|  |   uri: string, | ||||||
|  |   fn: (res: AppBskyFeedGetPostThread.Response) => boolean, | ||||||
|  | ) { | ||||||
|  |   await until( | ||||||
|  |     5, // 5 tries
 | ||||||
|  |     1e3, // 1s delay between tries
 | ||||||
|  |     fn, | ||||||
|  |     () => | ||||||
|  |       agent.app.bsky.feed.getPostThread({ | ||||||
|  |         uri, | ||||||
|  |         depth: 0, | ||||||
|  |       }), | ||||||
|  |   ) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @ -15,7 +15,7 @@ import { | ||||||
| import {msg, plural} from '@lingui/macro' | import {msg, plural} from '@lingui/macro' | ||||||
| import {useLingui} from '@lingui/react' | import {useLingui} from '@lingui/react' | ||||||
| 
 | 
 | ||||||
| import {HITSLOP_10, HITSLOP_20} from '#/lib/constants' | import {POST_CTRL_HITSLOP} from '#/lib/constants' | ||||||
| import {useHaptics} from '#/lib/haptics' | import {useHaptics} from '#/lib/haptics' | ||||||
| import {makeProfileLink} from '#/lib/routes/links' | import {makeProfileLink} from '#/lib/routes/links' | ||||||
| import {shareUrl} from '#/lib/sharing' | import {shareUrl} from '#/lib/sharing' | ||||||
|  | @ -215,7 +215,7 @@ let PostCtrls = ({ | ||||||
|             other: 'Reply (# replies)', |             other: 'Reply (# replies)', | ||||||
|           })} |           })} | ||||||
|           accessibilityHint="" |           accessibilityHint="" | ||||||
|           hitSlop={big ? HITSLOP_20 : HITSLOP_10}> |           hitSlop={POST_CTRL_HITSLOP}> | ||||||
|           <Bubble |           <Bubble | ||||||
|             style={[defaultCtrlColor, {pointerEvents: 'none'}]} |             style={[defaultCtrlColor, {pointerEvents: 'none'}]} | ||||||
|             width={big ? 22 : 18} |             width={big ? 22 : 18} | ||||||
|  | @ -258,7 +258,7 @@ let PostCtrls = ({ | ||||||
|                 }) |                 }) | ||||||
|           } |           } | ||||||
|           accessibilityHint="" |           accessibilityHint="" | ||||||
|           hitSlop={big ? HITSLOP_20 : HITSLOP_10}> |           hitSlop={POST_CTRL_HITSLOP}> | ||||||
|           {post.viewer?.like ? ( |           {post.viewer?.like ? ( | ||||||
|             <HeartIconFilled style={s.likeColor} width={big ? 22 : 18} /> |             <HeartIconFilled style={s.likeColor} width={big ? 22 : 18} /> | ||||||
|           ) : ( |           ) : ( | ||||||
|  | @ -299,7 +299,7 @@ let PostCtrls = ({ | ||||||
|               }} |               }} | ||||||
|               accessibilityLabel={_(msg`Share`)} |               accessibilityLabel={_(msg`Share`)} | ||||||
|               accessibilityHint="" |               accessibilityHint="" | ||||||
|               hitSlop={big ? HITSLOP_20 : HITSLOP_10}> |               hitSlop={POST_CTRL_HITSLOP}> | ||||||
|               <ArrowOutOfBox |               <ArrowOutOfBox | ||||||
|                 style={[defaultCtrlColor, {pointerEvents: 'none'}]} |                 style={[defaultCtrlColor, {pointerEvents: 'none'}]} | ||||||
|                 width={22} |                 width={22} | ||||||
|  | @ -325,7 +325,7 @@ let PostCtrls = ({ | ||||||
|           record={record} |           record={record} | ||||||
|           richText={richText} |           richText={richText} | ||||||
|           style={{padding: 5}} |           style={{padding: 5}} | ||||||
|           hitSlop={big ? HITSLOP_20 : HITSLOP_10} |           hitSlop={POST_CTRL_HITSLOP} | ||||||
|           timestamp={post.indexedAt} |           timestamp={post.indexedAt} | ||||||
|         /> |         /> | ||||||
|       </View> |       </View> | ||||||
|  |  | ||||||
|  | @ -3,7 +3,7 @@ import {View} from 'react-native' | ||||||
| import {msg, plural} from '@lingui/macro' | import {msg, plural} from '@lingui/macro' | ||||||
| import {useLingui} from '@lingui/react' | import {useLingui} from '@lingui/react' | ||||||
| 
 | 
 | ||||||
| import {HITSLOP_10, HITSLOP_20} from '#/lib/constants' | import {POST_CTRL_HITSLOP} from '#/lib/constants' | ||||||
| import {useHaptics} from '#/lib/haptics' | import {useHaptics} from '#/lib/haptics' | ||||||
| import {useRequireAuth} from '#/state/session' | import {useRequireAuth} from '#/state/session' | ||||||
| import {atoms as a, useTheme} from '#/alf' | import {atoms as a, useTheme} from '#/alf' | ||||||
|  | @ -67,7 +67,7 @@ let RepostButton = ({ | ||||||
|         shape="round" |         shape="round" | ||||||
|         variant="ghost" |         variant="ghost" | ||||||
|         color="secondary" |         color="secondary" | ||||||
|         hitSlop={big ? HITSLOP_20 : HITSLOP_10}> |         hitSlop={POST_CTRL_HITSLOP}> | ||||||
|         <Repost style={color} width={big ? 22 : 18} /> |         <Repost style={color} width={big ? 22 : 18} /> | ||||||
|         {typeof repostCount !== 'undefined' && repostCount > 0 ? ( |         {typeof repostCount !== 'undefined' && repostCount > 0 ? ( | ||||||
|           <Text |           <Text | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue