[Statsig] Track likes, reposts, follows (#3195)
* [Statsig] Track likes * Move tracking to intent * Track repost/unrepost * Track profile follows/unfollows * Less copy paste * Reorder
This commit is contained in:
		
							parent
							
								
									db79c918b2
								
							
						
					
					
						commit
						7eaa573b57
					
				
					 15 changed files with 125 additions and 30 deletions
				
			
		|  | @ -1,5 +1,35 @@ | ||||||
| export type Events = { | export type LogEvents = { | ||||||
|   init: { |   init: { | ||||||
|     initMs: number |     initMs: number | ||||||
|   } |   } | ||||||
|  |   'post:like': { | ||||||
|  |     logContext: 'FeedItem' | 'PostThreadItem' | 'Post' | ||||||
|  |   } | ||||||
|  |   'post:repost': { | ||||||
|  |     logContext: 'FeedItem' | 'PostThreadItem' | 'Post' | ||||||
|  |   } | ||||||
|  |   'post:unlike': { | ||||||
|  |     logContext: 'FeedItem' | 'PostThreadItem' | 'Post' | ||||||
|  |   } | ||||||
|  |   'post:unrepost': { | ||||||
|  |     logContext: 'FeedItem' | 'PostThreadItem' | 'Post' | ||||||
|  |   } | ||||||
|  |   'profile:follow': { | ||||||
|  |     logContext: | ||||||
|  |       | 'RecommendedFollowsItem' | ||||||
|  |       | 'PostThreadItem' | ||||||
|  |       | 'ProfileCard' | ||||||
|  |       | 'ProfileHeader' | ||||||
|  |       | 'ProfileHeaderSuggestedFollows' | ||||||
|  |       | 'ProfileMenu' | ||||||
|  |   } | ||||||
|  |   'profile:unfollow': { | ||||||
|  |     logContext: | ||||||
|  |       | 'RecommendedFollowsItem' | ||||||
|  |       | 'PostThreadItem' | ||||||
|  |       | 'ProfileCard' | ||||||
|  |       | 'ProfileHeader' | ||||||
|  |       | 'ProfileHeaderSuggestedFollows' | ||||||
|  |       | 'ProfileMenu' | ||||||
|  |   } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -6,7 +6,9 @@ import { | ||||||
| } from 'statsig-react-native-expo' | } from 'statsig-react-native-expo' | ||||||
| import {useSession} from '../../state/session' | import {useSession} from '../../state/session' | ||||||
| import {sha256} from 'js-sha256' | import {sha256} from 'js-sha256' | ||||||
| import {Events} from './events' | import {LogEvents} from './events' | ||||||
|  | 
 | ||||||
|  | export type {LogEvents} | ||||||
| 
 | 
 | ||||||
| const statsigOptions = { | const statsigOptions = { | ||||||
|   environment: { |   environment: { | ||||||
|  | @ -31,9 +33,9 @@ export function attachRouteToLogEvents( | ||||||
|   getCurrentRouteName = getRouteName |   getCurrentRouteName = getRouteName | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export function logEvent<E extends keyof Events>( | export function logEvent<E extends keyof LogEvents>( | ||||||
|   eventName: E & string, |   eventName: E & string, | ||||||
|   rawMetadata?: Events[E] & FlatJSONRecord, |   rawMetadata: LogEvents[E] & FlatJSONRecord, | ||||||
| ) { | ) { | ||||||
|   const fullMetadata = { |   const fullMetadata = { | ||||||
|     ...rawMetadata, |     ...rawMetadata, | ||||||
|  |  | ||||||
|  | @ -5,6 +5,7 @@ import {Shadow} from '#/state/cache/types' | ||||||
| import {getAgent} from '#/state/session' | import {getAgent} from '#/state/session' | ||||||
| import {updatePostShadow} from '#/state/cache/post-shadow' | import {updatePostShadow} from '#/state/cache/post-shadow' | ||||||
| import {track} from '#/lib/analytics/analytics' | import {track} from '#/lib/analytics/analytics' | ||||||
|  | import {logEvent, LogEvents} from '#/lib/statsig/statsig' | ||||||
| import {useToggleMutationQueue} from '#/lib/hooks/useToggleMutationQueue' | import {useToggleMutationQueue} from '#/lib/hooks/useToggleMutationQueue' | ||||||
| 
 | 
 | ||||||
| export const RQKEY = (postUri: string) => ['post', postUri] | export const RQKEY = (postUri: string) => ['post', postUri] | ||||||
|  | @ -58,12 +59,14 @@ export function useGetPost() { | ||||||
| 
 | 
 | ||||||
| export function usePostLikeMutationQueue( | export function usePostLikeMutationQueue( | ||||||
|   post: Shadow<AppBskyFeedDefs.PostView>, |   post: Shadow<AppBskyFeedDefs.PostView>, | ||||||
|  |   logContext: LogEvents['post:like']['logContext'] & | ||||||
|  |     LogEvents['post:unlike']['logContext'], | ||||||
| ) { | ) { | ||||||
|   const postUri = post.uri |   const postUri = post.uri | ||||||
|   const postCid = post.cid |   const postCid = post.cid | ||||||
|   const initialLikeUri = post.viewer?.like |   const initialLikeUri = post.viewer?.like | ||||||
|   const likeMutation = usePostLikeMutation() |   const likeMutation = usePostLikeMutation(logContext) | ||||||
|   const unlikeMutation = usePostUnlikeMutation() |   const unlikeMutation = usePostUnlikeMutation(logContext) | ||||||
| 
 | 
 | ||||||
|   const queueToggle = useToggleMutationQueue({ |   const queueToggle = useToggleMutationQueue({ | ||||||
|     initialState: initialLikeUri, |     initialState: initialLikeUri, | ||||||
|  | @ -111,22 +114,30 @@ export function usePostLikeMutationQueue( | ||||||
|   return [queueLike, queueUnlike] |   return [queueLike, queueUnlike] | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| function usePostLikeMutation() { | function usePostLikeMutation(logContext: LogEvents['post:like']['logContext']) { | ||||||
|   return useMutation< |   return useMutation< | ||||||
|     {uri: string}, // responds with the uri of the like
 |     {uri: string}, // responds with the uri of the like
 | ||||||
|     Error, |     Error, | ||||||
|     {uri: string; cid: string} // the post's uri and cid
 |     {uri: string; cid: string} // the post's uri and cid
 | ||||||
|   >({ |   >({ | ||||||
|     mutationFn: post => getAgent().like(post.uri, post.cid), |     mutationFn: post => { | ||||||
|  |       logEvent('post:like', {logContext}) | ||||||
|  |       return getAgent().like(post.uri, post.cid) | ||||||
|  |     }, | ||||||
|     onSuccess() { |     onSuccess() { | ||||||
|       track('Post:Like') |       track('Post:Like') | ||||||
|     }, |     }, | ||||||
|   }) |   }) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| function usePostUnlikeMutation() { | function usePostUnlikeMutation( | ||||||
|  |   logContext: LogEvents['post:unlike']['logContext'], | ||||||
|  | ) { | ||||||
|   return useMutation<void, Error, {postUri: string; likeUri: string}>({ |   return useMutation<void, Error, {postUri: string; likeUri: string}>({ | ||||||
|     mutationFn: ({likeUri}) => getAgent().deleteLike(likeUri), |     mutationFn: ({likeUri}) => { | ||||||
|  |       logEvent('post:unlike', {logContext}) | ||||||
|  |       return getAgent().deleteLike(likeUri) | ||||||
|  |     }, | ||||||
|     onSuccess() { |     onSuccess() { | ||||||
|       track('Post:Unlike') |       track('Post:Unlike') | ||||||
|     }, |     }, | ||||||
|  | @ -135,12 +146,14 @@ function usePostUnlikeMutation() { | ||||||
| 
 | 
 | ||||||
| export function usePostRepostMutationQueue( | export function usePostRepostMutationQueue( | ||||||
|   post: Shadow<AppBskyFeedDefs.PostView>, |   post: Shadow<AppBskyFeedDefs.PostView>, | ||||||
|  |   logContext: LogEvents['post:repost']['logContext'] & | ||||||
|  |     LogEvents['post:unrepost']['logContext'], | ||||||
| ) { | ) { | ||||||
|   const postUri = post.uri |   const postUri = post.uri | ||||||
|   const postCid = post.cid |   const postCid = post.cid | ||||||
|   const initialRepostUri = post.viewer?.repost |   const initialRepostUri = post.viewer?.repost | ||||||
|   const repostMutation = usePostRepostMutation() |   const repostMutation = usePostRepostMutation(logContext) | ||||||
|   const unrepostMutation = usePostUnrepostMutation() |   const unrepostMutation = usePostUnrepostMutation(logContext) | ||||||
| 
 | 
 | ||||||
|   const queueToggle = useToggleMutationQueue({ |   const queueToggle = useToggleMutationQueue({ | ||||||
|     initialState: initialRepostUri, |     initialState: initialRepostUri, | ||||||
|  | @ -188,22 +201,32 @@ export function usePostRepostMutationQueue( | ||||||
|   return [queueRepost, queueUnrepost] |   return [queueRepost, queueUnrepost] | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| function usePostRepostMutation() { | function usePostRepostMutation( | ||||||
|  |   logContext: LogEvents['post:repost']['logContext'], | ||||||
|  | ) { | ||||||
|   return useMutation< |   return useMutation< | ||||||
|     {uri: string}, // responds with the uri of the repost
 |     {uri: string}, // responds with the uri of the repost
 | ||||||
|     Error, |     Error, | ||||||
|     {uri: string; cid: string} // the post's uri and cid
 |     {uri: string; cid: string} // the post's uri and cid
 | ||||||
|   >({ |   >({ | ||||||
|     mutationFn: post => getAgent().repost(post.uri, post.cid), |     mutationFn: post => { | ||||||
|  |       logEvent('post:repost', {logContext}) | ||||||
|  |       return getAgent().repost(post.uri, post.cid) | ||||||
|  |     }, | ||||||
|     onSuccess() { |     onSuccess() { | ||||||
|       track('Post:Repost') |       track('Post:Repost') | ||||||
|     }, |     }, | ||||||
|   }) |   }) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| function usePostUnrepostMutation() { | function usePostUnrepostMutation( | ||||||
|  |   logContext: LogEvents['post:unrepost']['logContext'], | ||||||
|  | ) { | ||||||
|   return useMutation<void, Error, {postUri: string; repostUri: string}>({ |   return useMutation<void, Error, {postUri: string; repostUri: string}>({ | ||||||
|     mutationFn: ({repostUri}) => getAgent().deleteRepost(repostUri), |     mutationFn: ({repostUri}) => { | ||||||
|  |       logEvent('post:unrepost', {logContext}) | ||||||
|  |       return getAgent().deleteRepost(repostUri) | ||||||
|  |     }, | ||||||
|     onSuccess() { |     onSuccess() { | ||||||
|       track('Post:Unrepost') |       track('Post:Unrepost') | ||||||
|     }, |     }, | ||||||
|  |  | ||||||
|  | @ -26,6 +26,7 @@ import {RQKEY as RQKEY_MY_MUTED} from './my-muted-accounts' | ||||||
| import {RQKEY as RQKEY_MY_BLOCKED} from './my-blocked-accounts' | import {RQKEY as RQKEY_MY_BLOCKED} from './my-blocked-accounts' | ||||||
| import {STALE} from '#/state/queries' | import {STALE} from '#/state/queries' | ||||||
| import {track} from '#/lib/analytics/analytics' | import {track} from '#/lib/analytics/analytics' | ||||||
|  | import {logEvent, LogEvents} from '#/lib/statsig/statsig' | ||||||
| import {ThreadNode} from './post-thread' | import {ThreadNode} from './post-thread' | ||||||
| 
 | 
 | ||||||
| export const RQKEY = (did: string) => ['profile', did] | export const RQKEY = (did: string) => ['profile', did] | ||||||
|  | @ -186,11 +187,13 @@ export function useProfileUpdateMutation() { | ||||||
| 
 | 
 | ||||||
| export function useProfileFollowMutationQueue( | export function useProfileFollowMutationQueue( | ||||||
|   profile: Shadow<AppBskyActorDefs.ProfileViewDetailed>, |   profile: Shadow<AppBskyActorDefs.ProfileViewDetailed>, | ||||||
|  |   logContext: LogEvents['profile:follow']['logContext'] & | ||||||
|  |     LogEvents['profile:unfollow']['logContext'], | ||||||
| ) { | ) { | ||||||
|   const did = profile.did |   const did = profile.did | ||||||
|   const initialFollowingUri = profile.viewer?.following |   const initialFollowingUri = profile.viewer?.following | ||||||
|   const followMutation = useProfileFollowMutation() |   const followMutation = useProfileFollowMutation(logContext) | ||||||
|   const unfollowMutation = useProfileUnfollowMutation() |   const unfollowMutation = useProfileUnfollowMutation(logContext) | ||||||
| 
 | 
 | ||||||
|   const queueToggle = useToggleMutationQueue({ |   const queueToggle = useToggleMutationQueue({ | ||||||
|     initialState: initialFollowingUri, |     initialState: initialFollowingUri, | ||||||
|  | @ -237,9 +240,12 @@ export function useProfileFollowMutationQueue( | ||||||
|   return [queueFollow, queueUnfollow] |   return [queueFollow, queueUnfollow] | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| function useProfileFollowMutation() { | function useProfileFollowMutation( | ||||||
|  |   logContext: LogEvents['profile:follow']['logContext'], | ||||||
|  | ) { | ||||||
|   return useMutation<{uri: string; cid: string}, Error, {did: string}>({ |   return useMutation<{uri: string; cid: string}, Error, {did: string}>({ | ||||||
|     mutationFn: async ({did}) => { |     mutationFn: async ({did}) => { | ||||||
|  |       logEvent('profile:follow', {logContext}) | ||||||
|       return await getAgent().follow(did) |       return await getAgent().follow(did) | ||||||
|     }, |     }, | ||||||
|     onSuccess(data, variables) { |     onSuccess(data, variables) { | ||||||
|  | @ -248,9 +254,12 @@ function useProfileFollowMutation() { | ||||||
|   }) |   }) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| function useProfileUnfollowMutation() { | function useProfileUnfollowMutation( | ||||||
|  |   logContext: LogEvents['profile:unfollow']['logContext'], | ||||||
|  | ) { | ||||||
|   return useMutation<void, Error, {did: string; followUri: string}>({ |   return useMutation<void, Error, {did: string; followUri: string}>({ | ||||||
|     mutationFn: async ({followUri}) => { |     mutationFn: async ({followUri}) => { | ||||||
|  |       logEvent('profile:unfollow', {logContext}) | ||||||
|       track('Profile:Unfollow', {username: followUri}) |       track('Profile:Unfollow', {username: followUri}) | ||||||
|       return await getAgent().deleteFollow(followUri) |       return await getAgent().deleteFollow(followUri) | ||||||
|     }, |     }, | ||||||
|  |  | ||||||
|  | @ -56,7 +56,7 @@ export function RecommendedFollowsItem({ | ||||||
|   ) |   ) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export function ProfileCard({ | function ProfileCard({ | ||||||
|   profile, |   profile, | ||||||
|   onFollowStateChange, |   onFollowStateChange, | ||||||
|   moderation, |   moderation, | ||||||
|  | @ -72,7 +72,10 @@ export function ProfileCard({ | ||||||
|   const pal = usePalette('default') |   const pal = usePalette('default') | ||||||
|   const [addingMoreSuggestions, setAddingMoreSuggestions] = |   const [addingMoreSuggestions, setAddingMoreSuggestions] = | ||||||
|     React.useState(false) |     React.useState(false) | ||||||
|   const [queueFollow, queueUnfollow] = useProfileFollowMutationQueue(profile) |   const [queueFollow, queueUnfollow] = useProfileFollowMutationQueue( | ||||||
|  |     profile, | ||||||
|  |     'RecommendedFollowsItem', | ||||||
|  |   ) | ||||||
| 
 | 
 | ||||||
|   const onToggleFollow = React.useCallback(async () => { |   const onToggleFollow = React.useCallback(async () => { | ||||||
|     try { |     try { | ||||||
|  |  | ||||||
|  | @ -42,7 +42,10 @@ function PostThreadFollowBtnLoaded({ | ||||||
|   const {isTabletOrDesktop} = useWebMediaQueries() |   const {isTabletOrDesktop} = useWebMediaQueries() | ||||||
|   const profile: Shadow<AppBskyActorDefs.ProfileViewBasic> = |   const profile: Shadow<AppBskyActorDefs.ProfileViewBasic> = | ||||||
|     useProfileShadow(profileUnshadowed) |     useProfileShadow(profileUnshadowed) | ||||||
|   const [queueFollow, queueUnfollow] = useProfileFollowMutationQueue(profile) |   const [queueFollow, queueUnfollow] = useProfileFollowMutationQueue( | ||||||
|  |     profile, | ||||||
|  |     'PostThreadItem', | ||||||
|  |   ) | ||||||
|   const requireAuth = useRequireAuth() |   const requireAuth = useRequireAuth() | ||||||
| 
 | 
 | ||||||
|   const isFollowing = !!profile.viewer?.following |   const isFollowing = !!profile.viewer?.following | ||||||
|  |  | ||||||
|  | @ -407,6 +407,7 @@ let PostThreadItemLoaded = ({ | ||||||
|                 record={record} |                 record={record} | ||||||
|                 richText={richText} |                 richText={richText} | ||||||
|                 onPressReply={onPressReply} |                 onPressReply={onPressReply} | ||||||
|  |                 logContext="PostThreadItem" | ||||||
|               /> |               /> | ||||||
|             </View> |             </View> | ||||||
|           </View> |           </View> | ||||||
|  | @ -560,6 +561,7 @@ let PostThreadItemLoaded = ({ | ||||||
|                   record={record} |                   record={record} | ||||||
|                   richText={richText} |                   richText={richText} | ||||||
|                   onPressReply={onPressReply} |                   onPressReply={onPressReply} | ||||||
|  |                   logContext="PostThreadItem" | ||||||
|                 /> |                 /> | ||||||
|               </View> |               </View> | ||||||
|             </View> |             </View> | ||||||
|  |  | ||||||
|  | @ -220,6 +220,7 @@ function PostInner({ | ||||||
|             record={record} |             record={record} | ||||||
|             richText={richText} |             richText={richText} | ||||||
|             onPressReply={onPressReply} |             onPressReply={onPressReply} | ||||||
|  |             logContext="Post" | ||||||
|           /> |           /> | ||||||
|         </View> |         </View> | ||||||
|       </View> |       </View> | ||||||
|  |  | ||||||
|  | @ -310,6 +310,7 @@ let FeedItemInner = ({ | ||||||
|             showAppealLabelItem={ |             showAppealLabelItem={ | ||||||
|               post.author.did === currentAccount?.did && isModeratedPost |               post.author.did === currentAccount?.did && isModeratedPost | ||||||
|             } |             } | ||||||
|  |             logContext="FeedItem" | ||||||
|           /> |           /> | ||||||
|         </View> |         </View> | ||||||
|       </View> |       </View> | ||||||
|  |  | ||||||
|  | @ -13,13 +13,18 @@ export function FollowButton({ | ||||||
|   followedType = 'default', |   followedType = 'default', | ||||||
|   profile, |   profile, | ||||||
|   labelStyle, |   labelStyle, | ||||||
|  |   logContext, | ||||||
| }: { | }: { | ||||||
|   unfollowedType?: ButtonType |   unfollowedType?: ButtonType | ||||||
|   followedType?: ButtonType |   followedType?: ButtonType | ||||||
|   profile: Shadow<AppBskyActorDefs.ProfileViewBasic> |   profile: Shadow<AppBskyActorDefs.ProfileViewBasic> | ||||||
|   labelStyle?: StyleProp<TextStyle> |   labelStyle?: StyleProp<TextStyle> | ||||||
|  |   logContext: 'ProfileCard' | ||||||
| }) { | }) { | ||||||
|   const [queueFollow, queueUnfollow] = useProfileFollowMutationQueue(profile) |   const [queueFollow, queueUnfollow] = useProfileFollowMutationQueue( | ||||||
|  |     profile, | ||||||
|  |     logContext, | ||||||
|  |   ) | ||||||
|   const {_} = useLingui() |   const {_} = useLingui() | ||||||
| 
 | 
 | ||||||
|   const onPressFollow = async () => { |   const onPressFollow = async () => { | ||||||
|  |  | ||||||
|  | @ -230,7 +230,9 @@ export function ProfileCardWithFollowBtn({ | ||||||
|       renderButton={ |       renderButton={ | ||||||
|         isMe |         isMe | ||||||
|           ? undefined |           ? undefined | ||||||
|           : profileShadow => <FollowButton profile={profileShadow} /> |           : profileShadow => ( | ||||||
|  |               <FollowButton profile={profileShadow} logContext="ProfileCard" /> | ||||||
|  |             ) | ||||||
|       } |       } | ||||||
|     /> |     /> | ||||||
|   ) |   ) | ||||||
|  |  | ||||||
|  | @ -103,7 +103,10 @@ let ProfileHeader = ({ | ||||||
|   const invalidHandle = isInvalidHandle(profile.handle) |   const invalidHandle = isInvalidHandle(profile.handle) | ||||||
|   const {isDesktop} = useWebMediaQueries() |   const {isDesktop} = useWebMediaQueries() | ||||||
|   const [showSuggestedFollows, setShowSuggestedFollows] = React.useState(false) |   const [showSuggestedFollows, setShowSuggestedFollows] = React.useState(false) | ||||||
|   const [queueFollow, queueUnfollow] = useProfileFollowMutationQueue(profile) |   const [queueFollow, queueUnfollow] = useProfileFollowMutationQueue( | ||||||
|  |     profile, | ||||||
|  |     'ProfileHeader', | ||||||
|  |   ) | ||||||
|   const [__, queueUnblock] = useProfileBlockMutationQueue(profile) |   const [__, queueUnblock] = useProfileBlockMutationQueue(profile) | ||||||
|   const unblockPromptControl = Prompt.usePromptControl() |   const unblockPromptControl = Prompt.usePromptControl() | ||||||
|   const moderation = useMemo( |   const moderation = useMemo( | ||||||
|  |  | ||||||
|  | @ -170,7 +170,10 @@ function SuggestedFollow({ | ||||||
|   const pal = usePalette('default') |   const pal = usePalette('default') | ||||||
|   const moderationOpts = useModerationOpts() |   const moderationOpts = useModerationOpts() | ||||||
|   const profile = useProfileShadow(profileUnshadowed) |   const profile = useProfileShadow(profileUnshadowed) | ||||||
|   const [queueFollow, queueUnfollow] = useProfileFollowMutationQueue(profile) |   const [queueFollow, queueUnfollow] = useProfileFollowMutationQueue( | ||||||
|  |     profile, | ||||||
|  |     'ProfileHeaderSuggestedFollows', | ||||||
|  |   ) | ||||||
| 
 | 
 | ||||||
|   const onPressFollow = React.useCallback(async () => { |   const onPressFollow = React.useCallback(async () => { | ||||||
|     try { |     try { | ||||||
|  |  | ||||||
|  | @ -52,7 +52,10 @@ let ProfileMenu = ({ | ||||||
| 
 | 
 | ||||||
|   const [queueMute, queueUnmute] = useProfileMuteMutationQueue(profile) |   const [queueMute, queueUnmute] = useProfileMuteMutationQueue(profile) | ||||||
|   const [queueBlock, queueUnblock] = useProfileBlockMutationQueue(profile) |   const [queueBlock, queueUnblock] = useProfileBlockMutationQueue(profile) | ||||||
|   const [, queueUnfollow] = useProfileFollowMutationQueue(profile) |   const [, queueUnfollow] = useProfileFollowMutationQueue( | ||||||
|  |     profile, | ||||||
|  |     'ProfileMenu', | ||||||
|  |   ) | ||||||
| 
 | 
 | ||||||
|   const blockPromptControl = Prompt.usePromptControl() |   const blockPromptControl = Prompt.usePromptControl() | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -44,6 +44,7 @@ let PostCtrls = ({ | ||||||
|   showAppealLabelItem, |   showAppealLabelItem, | ||||||
|   style, |   style, | ||||||
|   onPressReply, |   onPressReply, | ||||||
|  |   logContext, | ||||||
| }: { | }: { | ||||||
|   big?: boolean |   big?: boolean | ||||||
|   post: Shadow<AppBskyFeedDefs.PostView> |   post: Shadow<AppBskyFeedDefs.PostView> | ||||||
|  | @ -52,13 +53,17 @@ let PostCtrls = ({ | ||||||
|   showAppealLabelItem?: boolean |   showAppealLabelItem?: boolean | ||||||
|   style?: StyleProp<ViewStyle> |   style?: StyleProp<ViewStyle> | ||||||
|   onPressReply: () => void |   onPressReply: () => void | ||||||
|  |   logContext: 'FeedItem' | 'PostThreadItem' | 'Post' | ||||||
| }): React.ReactNode => { | }): React.ReactNode => { | ||||||
|   const theme = useTheme() |   const theme = useTheme() | ||||||
|   const {_} = useLingui() |   const {_} = useLingui() | ||||||
|   const {openComposer} = useComposerControls() |   const {openComposer} = useComposerControls() | ||||||
|   const {closeModal} = useModalControls() |   const {closeModal} = useModalControls() | ||||||
|   const [queueLike, queueUnlike] = usePostLikeMutationQueue(post) |   const [queueLike, queueUnlike] = usePostLikeMutationQueue(post, logContext) | ||||||
|   const [queueRepost, queueUnrepost] = usePostRepostMutationQueue(post) |   const [queueRepost, queueUnrepost] = usePostRepostMutationQueue( | ||||||
|  |     post, | ||||||
|  |     logContext, | ||||||
|  |   ) | ||||||
|   const requireAuth = useRequireAuth() |   const requireAuth = useRequireAuth() | ||||||
| 
 | 
 | ||||||
|   const defaultCtrlColor = React.useMemo( |   const defaultCtrlColor = React.useMemo( | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue