mirror of
https://github.com/tooot-app/app
synced 2025-04-04 21:51:12 +02:00
Android build success
This commit is contained in:
parent
247f7a7982
commit
253ddee319
1
.gitignore
vendored
1
.gitignore
vendored
@ -51,6 +51,7 @@ build/
|
|||||||
.gradle
|
.gradle
|
||||||
local.properties
|
local.properties
|
||||||
*.iml
|
*.iml
|
||||||
|
*.hprof
|
||||||
|
|
||||||
# node.js
|
# node.js
|
||||||
#
|
#
|
||||||
|
@ -0,0 +1,37 @@
|
|||||||
|
package com.xmflsct.app.tooot.generated;
|
||||||
|
|
||||||
|
import java.util.Arrays;
|
||||||
|
import java.util.List;
|
||||||
|
import org.unimodules.core.interfaces.Package;
|
||||||
|
|
||||||
|
public class BasePackageList {
|
||||||
|
public List<Package> getPackageList() {
|
||||||
|
return Arrays.<Package>asList(
|
||||||
|
new expo.modules.application.ApplicationPackage(),
|
||||||
|
new expo.modules.av.AVPackage(),
|
||||||
|
new expo.modules.constants.ConstantsPackage(),
|
||||||
|
new expo.modules.crypto.CryptoPackage(),
|
||||||
|
new expo.modules.device.DevicePackage(),
|
||||||
|
new expo.modules.errorrecovery.ErrorRecoveryPackage(),
|
||||||
|
new expo.modules.filesystem.FileSystemPackage(),
|
||||||
|
new expo.modules.firebase.analytics.FirebaseAnalyticsPackage(),
|
||||||
|
new expo.modules.firebase.core.FirebaseCorePackage(),
|
||||||
|
new expo.modules.font.FontLoaderPackage(),
|
||||||
|
new expo.modules.haptics.HapticsPackage(),
|
||||||
|
new expo.modules.imageloader.ImageLoaderPackage(),
|
||||||
|
new expo.modules.imagepicker.ImagePickerPackage(),
|
||||||
|
new expo.modules.keepawake.KeepAwakePackage(),
|
||||||
|
new expo.modules.lineargradient.LinearGradientPackage(),
|
||||||
|
new expo.modules.localization.LocalizationPackage(),
|
||||||
|
new expo.modules.location.LocationPackage(),
|
||||||
|
new expo.modules.permissions.PermissionsPackage(),
|
||||||
|
new expo.modules.securestore.SecureStorePackage(),
|
||||||
|
new expo.modules.splashscreen.SplashScreenPackage(),
|
||||||
|
new expo.modules.sqlite.SQLitePackage(),
|
||||||
|
new expo.modules.storereview.StoreReviewPackage(),
|
||||||
|
new expo.modules.updates.UpdatesPackage(),
|
||||||
|
new expo.modules.videothumbnails.VideoThumbnailsPackage(),
|
||||||
|
new expo.modules.webbrowser.WebBrowserPackage()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
@ -27,3 +27,8 @@ android.enableJetifier=true
|
|||||||
|
|
||||||
# Version of flipper SDK to use with React Native
|
# Version of flipper SDK to use with React Native
|
||||||
FLIPPER_VERSION=0.54.0
|
FLIPPER_VERSION=0.54.0
|
||||||
|
|
||||||
|
org.gradle.jvmargs=-Xmx4096m -XX:MaxPermSize=4096m -XX:+HeapDumpOnOutOfMemoryError
|
||||||
|
org.gradle.daemon=true
|
||||||
|
org.gradle.parallel=true
|
||||||
|
org.gradle.configureondemand=true
|
@ -36,15 +36,6 @@ export default (): ExpoConfig => ({
|
|||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
ios: {
|
|
||||||
buildNumber: '4',
|
|
||||||
config: { usesNonExemptEncryption: false },
|
|
||||||
bundleIdentifier: 'com.xmflsct.app.tooot',
|
|
||||||
googleServicesFile: './configs/GoogleService-Info.plist',
|
|
||||||
infoPlist: {
|
|
||||||
CFBundleAllowMixedLocalizations: true
|
|
||||||
}
|
|
||||||
},
|
|
||||||
locales: {
|
locales: {
|
||||||
en: './src/i18n/en/system.json',
|
en: './src/i18n/en/system.json',
|
||||||
zh: './src/i18n/zh-Hans/system.json'
|
zh: './src/i18n/zh-Hans/system.json'
|
||||||
@ -54,18 +45,5 @@ export default (): ExpoConfig => ({
|
|||||||
package: 'com.xmflsct.app.tooot',
|
package: 'com.xmflsct.app.tooot',
|
||||||
googleServicesFile: './configs/google-services.json',
|
googleServicesFile: './configs/google-services.json',
|
||||||
permissions: ['CAMERA', 'VIBRATE']
|
permissions: ['CAMERA', 'VIBRATE']
|
||||||
},
|
|
||||||
web: {
|
|
||||||
config: {
|
|
||||||
firebase: {
|
|
||||||
apiKey: 'AIzaSyAnvo0jyD1WB0tv2FLenz-CSDS-RgaWWR4',
|
|
||||||
authDomain: 'xmflsct-mastodon-app.firebaseapp.com',
|
|
||||||
projectId: 'xmflsct-mastodon-app',
|
|
||||||
storageBucket: 'xmflsct-mastodon-app.appspot.com',
|
|
||||||
messagingSenderId: '661638997772',
|
|
||||||
appId: '1:661638997772:web:1e7aab28be7dc06d9f8b29',
|
|
||||||
measurementId: 'G-3J0FS8WV5J'
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
@ -212,9 +212,6 @@ const Timeline: React.FC<Props> = ({
|
|||||||
ListEmptyComponent={flItemEmptyComponent}
|
ListEmptyComponent={flItemEmptyComponent}
|
||||||
{...(!disableRefresh && { refreshControl })}
|
{...(!disableRefresh && { refreshControl })}
|
||||||
ItemSeparatorComponent={ItemSeparatorComponent}
|
ItemSeparatorComponent={ItemSeparatorComponent}
|
||||||
{...(queryKey &&
|
|
||||||
queryKey[1].page === 'RemotePublic' &&
|
|
||||||
!publicRemoteNotice && { ListHeaderComponent })}
|
|
||||||
{...(toot && isSuccess && { onScrollToIndexFailed })}
|
{...(toot && isSuccess && { onScrollToIndexFailed })}
|
||||||
maintainVisibleContentPosition={{
|
maintainVisibleContentPosition={{
|
||||||
minIndexForVisible: 0,
|
minIndexForVisible: 0,
|
||||||
|
@ -1,130 +0,0 @@
|
|||||||
import analytics from '@components/analytics'
|
|
||||||
import haptics from '@components/haptics'
|
|
||||||
import { MenuContainer, MenuHeader, MenuRow } from '@components/Menu'
|
|
||||||
import { toast } from '@components/toast'
|
|
||||||
import {
|
|
||||||
MutationVarsTimelineUpdateAccountProperty,
|
|
||||||
QueryKeyTimeline,
|
|
||||||
useTimelineMutation
|
|
||||||
} from '@utils/queryHooks/timeline'
|
|
||||||
import React from 'react'
|
|
||||||
import { useTranslation } from 'react-i18next'
|
|
||||||
import { useQueryClient } from 'react-query'
|
|
||||||
|
|
||||||
export interface Props {
|
|
||||||
queryKey?: QueryKeyTimeline
|
|
||||||
account: Mastodon.Account
|
|
||||||
setBottomSheetVisible: React.Dispatch<React.SetStateAction<boolean>>
|
|
||||||
}
|
|
||||||
|
|
||||||
const HeaderActionsAccount: React.FC<Props> = ({
|
|
||||||
queryKey,
|
|
||||||
account,
|
|
||||||
setBottomSheetVisible
|
|
||||||
}) => {
|
|
||||||
const { t } = useTranslation('componentTimeline')
|
|
||||||
|
|
||||||
const queryClient = useQueryClient()
|
|
||||||
const mutateion = useTimelineMutation({
|
|
||||||
queryClient,
|
|
||||||
onSuccess: (_, params) => {
|
|
||||||
const theParams = params as MutationVarsTimelineUpdateAccountProperty
|
|
||||||
haptics('Success')
|
|
||||||
toast({
|
|
||||||
type: 'success',
|
|
||||||
message: t('common:toastMessage.success.message', {
|
|
||||||
function: t(
|
|
||||||
`shared.header.actions.account.${theParams.payload.property}.function`,
|
|
||||||
{
|
|
||||||
acct: account.acct
|
|
||||||
}
|
|
||||||
)
|
|
||||||
})
|
|
||||||
})
|
|
||||||
},
|
|
||||||
onError: (err: any, params) => {
|
|
||||||
const theParams = params as MutationVarsTimelineUpdateAccountProperty
|
|
||||||
haptics('Error')
|
|
||||||
toast({
|
|
||||||
type: 'error',
|
|
||||||
message: t('common:toastMessage.error.message', {
|
|
||||||
function: t(
|
|
||||||
`shared.header.actions.account.${theParams.payload.property}.function`
|
|
||||||
)
|
|
||||||
}),
|
|
||||||
...(err.status &&
|
|
||||||
typeof err.status === 'number' &&
|
|
||||||
err.data &&
|
|
||||||
err.data.error &&
|
|
||||||
typeof err.data.error === 'string' && {
|
|
||||||
description: err.data.error
|
|
||||||
})
|
|
||||||
})
|
|
||||||
},
|
|
||||||
onSettled: () => {
|
|
||||||
queryKey && queryClient.invalidateQueries(queryKey)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
return (
|
|
||||||
<MenuContainer>
|
|
||||||
<MenuHeader heading={t('shared.header.actions.account.heading')} />
|
|
||||||
<MenuRow
|
|
||||||
onPress={() => {
|
|
||||||
analytics('timeline_shared_headeractions_account_mute_press', {
|
|
||||||
page: queryKey && queryKey[1].page
|
|
||||||
})
|
|
||||||
setBottomSheetVisible(false)
|
|
||||||
mutateion.mutate({
|
|
||||||
type: 'updateAccountProperty',
|
|
||||||
queryKey,
|
|
||||||
id: account.id,
|
|
||||||
payload: { property: 'mute' }
|
|
||||||
})
|
|
||||||
}}
|
|
||||||
iconFront='EyeOff'
|
|
||||||
title={t('shared.header.actions.account.mute.button', {
|
|
||||||
acct: account.acct
|
|
||||||
})}
|
|
||||||
/>
|
|
||||||
<MenuRow
|
|
||||||
onPress={() => {
|
|
||||||
analytics('timeline_shared_headeractions_account_block_press', {
|
|
||||||
page: queryKey && queryKey[1].page
|
|
||||||
})
|
|
||||||
setBottomSheetVisible(false)
|
|
||||||
mutateion.mutate({
|
|
||||||
type: 'updateAccountProperty',
|
|
||||||
queryKey,
|
|
||||||
id: account.id,
|
|
||||||
payload: { property: 'block' }
|
|
||||||
})
|
|
||||||
}}
|
|
||||||
iconFront='XCircle'
|
|
||||||
title={t('shared.header.actions.account.block.button', {
|
|
||||||
acct: account.acct
|
|
||||||
})}
|
|
||||||
/>
|
|
||||||
<MenuRow
|
|
||||||
onPress={() => {
|
|
||||||
analytics('timeline_shared_headeractions_account_reports_press', {
|
|
||||||
page: queryKey && queryKey[1].page
|
|
||||||
})
|
|
||||||
setBottomSheetVisible(false)
|
|
||||||
mutateion.mutate({
|
|
||||||
type: 'updateAccountProperty',
|
|
||||||
queryKey,
|
|
||||||
id: account.id,
|
|
||||||
payload: { property: 'reports' }
|
|
||||||
})
|
|
||||||
}}
|
|
||||||
iconFront='Flag'
|
|
||||||
title={t('shared.header.actions.account.reports.button', {
|
|
||||||
acct: account.acct
|
|
||||||
})}
|
|
||||||
/>
|
|
||||||
</MenuContainer>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
export default HeaderActionsAccount
|
|
@ -1,87 +0,0 @@
|
|||||||
import analytics from '@components/analytics'
|
|
||||||
import MenuContainer from '@components/Menu/Container'
|
|
||||||
import MenuHeader from '@components/Menu/Header'
|
|
||||||
import MenuRow from '@components/Menu/Row'
|
|
||||||
import { toast } from '@components/toast'
|
|
||||||
import {
|
|
||||||
QueryKeyTimeline,
|
|
||||||
useTimelineMutation
|
|
||||||
} from '@utils/queryHooks/timeline'
|
|
||||||
import React from 'react'
|
|
||||||
import { useTranslation } from 'react-i18next'
|
|
||||||
import { Alert } from 'react-native'
|
|
||||||
import { useQueryClient } from 'react-query'
|
|
||||||
|
|
||||||
export interface Props {
|
|
||||||
queryKey: QueryKeyTimeline
|
|
||||||
domain: string
|
|
||||||
setBottomSheetVisible: React.Dispatch<React.SetStateAction<boolean>>
|
|
||||||
}
|
|
||||||
|
|
||||||
const HeaderActionsDomain: React.FC<Props> = ({
|
|
||||||
queryKey,
|
|
||||||
domain,
|
|
||||||
setBottomSheetVisible
|
|
||||||
}) => {
|
|
||||||
const { t } = useTranslation('componentTimeline')
|
|
||||||
const queryClient = useQueryClient()
|
|
||||||
const mutation = useTimelineMutation({
|
|
||||||
queryClient,
|
|
||||||
onSettled: () => {
|
|
||||||
toast({
|
|
||||||
type: 'success',
|
|
||||||
message: t('common:toastMessage.success.message', {
|
|
||||||
function: t(`shared.header.actions.domain.block.function`)
|
|
||||||
})
|
|
||||||
})
|
|
||||||
queryClient.invalidateQueries(queryKey)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
return (
|
|
||||||
<MenuContainer>
|
|
||||||
<MenuHeader heading={t(`shared.header.actions.domain.heading`)} />
|
|
||||||
<MenuRow
|
|
||||||
onPress={() => {
|
|
||||||
analytics('timeline_shared_headeractions_domain_block_press', {
|
|
||||||
page: queryKey[1].page
|
|
||||||
})
|
|
||||||
Alert.alert(
|
|
||||||
t('shared.header.actions.domain.alert.title', { domain }),
|
|
||||||
t('shared.header.actions.domain.alert.message'),
|
|
||||||
[
|
|
||||||
{
|
|
||||||
text: t('shared.header.actions.domain.alert.buttons.cancel'),
|
|
||||||
style: 'cancel'
|
|
||||||
},
|
|
||||||
{
|
|
||||||
text: t('shared.header.actions.domain.alert.buttons.confirm'),
|
|
||||||
style: 'destructive',
|
|
||||||
onPress: () => {
|
|
||||||
analytics(
|
|
||||||
'timeline_shared_headeractions_domain_block_confirm',
|
|
||||||
{
|
|
||||||
page: queryKey && queryKey[1].page
|
|
||||||
}
|
|
||||||
)
|
|
||||||
setBottomSheetVisible(false)
|
|
||||||
mutation.mutate({
|
|
||||||
type: 'domainBlock',
|
|
||||||
queryKey,
|
|
||||||
domain: domain
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
)
|
|
||||||
}}
|
|
||||||
iconFront='CloudOff'
|
|
||||||
title={t(`shared.header.actions.domain.block.button`, {
|
|
||||||
domain
|
|
||||||
})}
|
|
||||||
/>
|
|
||||||
</MenuContainer>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
export default HeaderActionsDomain
|
|
@ -1,110 +0,0 @@
|
|||||||
import analytics from '@components/analytics'
|
|
||||||
import BottomSheet from '@screens/Tabs/Shared/node_modules/@screens/Actions/BottomSheet'
|
|
||||||
import Icon from '@components/Icon'
|
|
||||||
import { QueryKeyTimeline } from '@utils/queryHooks/timeline'
|
|
||||||
import { getLocalAccount, getLocalUrl } from '@utils/slices/instancesSlice'
|
|
||||||
import { StyleConstants } from '@utils/styles/constants'
|
|
||||||
import { useTheme } from '@utils/styles/ThemeManager'
|
|
||||||
import React, { useCallback, useMemo, useState } from 'react'
|
|
||||||
import { Pressable, StyleSheet } from 'react-native'
|
|
||||||
import { useSelector } from 'react-redux'
|
|
||||||
import HeaderActionsAccount from './Account'
|
|
||||||
import HeaderActionsDomain from './Domain'
|
|
||||||
import HeaderActionsShare from './Share'
|
|
||||||
import HeaderActionsStatus from './Status'
|
|
||||||
|
|
||||||
export interface Props {
|
|
||||||
queryKey: QueryKeyTimeline
|
|
||||||
status: Mastodon.Status
|
|
||||||
url?: string
|
|
||||||
type?: 'status' | 'account'
|
|
||||||
}
|
|
||||||
|
|
||||||
const HeaderActions = React.memo(
|
|
||||||
({ queryKey, status, url, type }: Props) => {
|
|
||||||
const { theme } = useTheme()
|
|
||||||
|
|
||||||
const localAccount = useSelector(getLocalAccount)
|
|
||||||
const sameAccount = localAccount?.id === status.account.id
|
|
||||||
|
|
||||||
const localDomain = useSelector(getLocalUrl)
|
|
||||||
const statusDomain = status.uri
|
|
||||||
? status.uri.split(new RegExp(/\/\/(.*?)\//))[1]
|
|
||||||
: ''
|
|
||||||
const sameDomain = localDomain === statusDomain
|
|
||||||
|
|
||||||
const [modalVisible, setBottomSheetVisible] = useState(false)
|
|
||||||
const onPress = useCallback(() => {
|
|
||||||
analytics('bottomsheet_open_press', {
|
|
||||||
page: queryKey[1].page
|
|
||||||
})
|
|
||||||
setBottomSheetVisible(true)
|
|
||||||
}, [])
|
|
||||||
const children = useMemo(
|
|
||||||
() => (
|
|
||||||
<Icon
|
|
||||||
name='MoreHorizontal'
|
|
||||||
color={theme.secondary}
|
|
||||||
size={StyleConstants.Font.Size.L}
|
|
||||||
/>
|
|
||||||
),
|
|
||||||
[]
|
|
||||||
)
|
|
||||||
|
|
||||||
return (
|
|
||||||
<>
|
|
||||||
<Pressable style={styles.base} onPress={onPress} children={children} />
|
|
||||||
{modalVisible && (
|
|
||||||
<BottomSheet
|
|
||||||
visible={modalVisible}
|
|
||||||
handleDismiss={() => setBottomSheetVisible(false)}
|
|
||||||
>
|
|
||||||
{!sameAccount && (
|
|
||||||
<HeaderActionsAccount
|
|
||||||
queryKey={queryKey}
|
|
||||||
account={status.account}
|
|
||||||
setBottomSheetVisible={setBottomSheetVisible}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{sameAccount && (
|
|
||||||
<HeaderActionsStatus
|
|
||||||
queryKey={queryKey}
|
|
||||||
status={status}
|
|
||||||
setBottomSheetVisible={setBottomSheetVisible}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{!sameDomain && (
|
|
||||||
<HeaderActionsDomain
|
|
||||||
queryKey={queryKey}
|
|
||||||
domain={statusDomain}
|
|
||||||
setBottomSheetVisible={setBottomSheetVisible}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{url && type ? (
|
|
||||||
<HeaderActionsShare
|
|
||||||
url={url}
|
|
||||||
type={type}
|
|
||||||
setBottomSheetVisible={setBottomSheetVisible}
|
|
||||||
/>
|
|
||||||
) : null}
|
|
||||||
</BottomSheet>
|
|
||||||
)}
|
|
||||||
</>
|
|
||||||
)
|
|
||||||
},
|
|
||||||
() => true
|
|
||||||
)
|
|
||||||
|
|
||||||
const styles = StyleSheet.create({
|
|
||||||
base: {
|
|
||||||
flex: 1,
|
|
||||||
flexDirection: 'row',
|
|
||||||
justifyContent: 'center',
|
|
||||||
paddingBottom: StyleConstants.Spacing.S
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
export default HeaderActions
|
|
@ -1,49 +0,0 @@
|
|||||||
import analytics from '@components/analytics'
|
|
||||||
import MenuContainer from '@components/Menu/Container'
|
|
||||||
import MenuHeader from '@components/Menu/Header'
|
|
||||||
import MenuRow from '@components/Menu/Row'
|
|
||||||
import React from 'react'
|
|
||||||
import { useTranslation } from 'react-i18next'
|
|
||||||
import { Platform, Share } from 'react-native'
|
|
||||||
|
|
||||||
export interface Props {
|
|
||||||
type: 'status' | 'account'
|
|
||||||
url: string
|
|
||||||
setBottomSheetVisible: React.Dispatch<React.SetStateAction<boolean>>
|
|
||||||
}
|
|
||||||
|
|
||||||
const HeaderActionsShare: React.FC<Props> = ({
|
|
||||||
type,
|
|
||||||
url,
|
|
||||||
setBottomSheetVisible
|
|
||||||
}) => {
|
|
||||||
const { t } = useTranslation('componentTimeline')
|
|
||||||
|
|
||||||
return (
|
|
||||||
<MenuContainer>
|
|
||||||
<MenuHeader heading={t(`shared.header.actions.share.${type}.heading`)} />
|
|
||||||
<MenuRow
|
|
||||||
iconFront='Share2'
|
|
||||||
title={t(`shared.header.actions.share.${type}.button`)}
|
|
||||||
onPress={async () => {
|
|
||||||
analytics('timeline_shared_headeractions_share_press')
|
|
||||||
switch (Platform.OS) {
|
|
||||||
case 'ios':
|
|
||||||
await Share.share({
|
|
||||||
url
|
|
||||||
})
|
|
||||||
break
|
|
||||||
case 'android':
|
|
||||||
await Share.share({
|
|
||||||
message: url
|
|
||||||
})
|
|
||||||
break
|
|
||||||
}
|
|
||||||
setBottomSheetVisible(false)
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
</MenuContainer>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
export default HeaderActionsShare
|
|
@ -1,180 +0,0 @@
|
|||||||
import React from 'react'
|
|
||||||
import { useTranslation } from 'react-i18next'
|
|
||||||
import { Alert } from 'react-native'
|
|
||||||
import { useQueryClient } from 'react-query'
|
|
||||||
import { MenuContainer, MenuHeader, MenuRow } from '@components/Menu'
|
|
||||||
import { toast } from '@components/toast'
|
|
||||||
import { useNavigation } from '@react-navigation/native'
|
|
||||||
import {
|
|
||||||
MutationVarsTimelineUpdateStatusProperty,
|
|
||||||
QueryKeyTimeline,
|
|
||||||
useTimelineMutation
|
|
||||||
} from '@utils/queryHooks/timeline'
|
|
||||||
import analytics from '@components/analytics'
|
|
||||||
|
|
||||||
export interface Props {
|
|
||||||
queryKey: QueryKeyTimeline
|
|
||||||
status: Mastodon.Status
|
|
||||||
setBottomSheetVisible: React.Dispatch<React.SetStateAction<boolean>>
|
|
||||||
}
|
|
||||||
|
|
||||||
const HeaderActionsStatus: React.FC<Props> = ({
|
|
||||||
queryKey,
|
|
||||||
status,
|
|
||||||
setBottomSheetVisible
|
|
||||||
}) => {
|
|
||||||
const navigation = useNavigation()
|
|
||||||
const { t } = useTranslation('componentTimeline')
|
|
||||||
|
|
||||||
const queryClient = useQueryClient()
|
|
||||||
const mutation = useTimelineMutation({
|
|
||||||
queryClient,
|
|
||||||
onMutate: true,
|
|
||||||
onError: (err: any, params, oldData) => {
|
|
||||||
const theFunction = (params as MutationVarsTimelineUpdateStatusProperty)
|
|
||||||
.payload
|
|
||||||
? (params as MutationVarsTimelineUpdateStatusProperty).payload.property
|
|
||||||
: 'delete'
|
|
||||||
toast({
|
|
||||||
type: 'error',
|
|
||||||
message: t('common:toastMessage.error.message', {
|
|
||||||
function: t(`shared.header.actions.status.${theFunction}.function`)
|
|
||||||
}),
|
|
||||||
...(err.status &&
|
|
||||||
typeof err.status === 'number' &&
|
|
||||||
err.data &&
|
|
||||||
err.data.error &&
|
|
||||||
typeof err.data.error === 'string' && {
|
|
||||||
description: err.data.error
|
|
||||||
})
|
|
||||||
})
|
|
||||||
queryClient.setQueryData(queryKey, oldData)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
return (
|
|
||||||
<MenuContainer>
|
|
||||||
<MenuHeader heading={t('shared.header.actions.status.heading')} />
|
|
||||||
<MenuRow
|
|
||||||
onPress={() => {
|
|
||||||
analytics('timeline_shared_headeractions_status_delete_press', {
|
|
||||||
page: queryKey && queryKey[1].page
|
|
||||||
})
|
|
||||||
setBottomSheetVisible(false)
|
|
||||||
mutation.mutate({
|
|
||||||
type: 'deleteItem',
|
|
||||||
source: 'statuses',
|
|
||||||
queryKey,
|
|
||||||
id: status.id
|
|
||||||
})
|
|
||||||
}}
|
|
||||||
iconFront='Trash'
|
|
||||||
title={t('shared.header.actions.status.delete.button')}
|
|
||||||
/>
|
|
||||||
<MenuRow
|
|
||||||
onPress={() => {
|
|
||||||
analytics('timeline_shared_headeractions_status_deleteedit_press', {
|
|
||||||
page: queryKey && queryKey[1].page
|
|
||||||
})
|
|
||||||
Alert.alert(
|
|
||||||
t('shared.header.actions.status.edit.alert.title'),
|
|
||||||
t('shared.header.actions.status.edit.alert.message'),
|
|
||||||
[
|
|
||||||
{
|
|
||||||
text: t(
|
|
||||||
'shared.header.actions.status.edit.alert.buttons.cancel'
|
|
||||||
),
|
|
||||||
style: 'cancel'
|
|
||||||
},
|
|
||||||
{
|
|
||||||
text: t(
|
|
||||||
'shared.header.actions.status.edit.alert.buttons.confirm'
|
|
||||||
),
|
|
||||||
style: 'destructive',
|
|
||||||
onPress: async () => {
|
|
||||||
analytics(
|
|
||||||
'timeline_shared_headeractions_status_deleteedit_confirm',
|
|
||||||
{
|
|
||||||
page: queryKey && queryKey[1].page
|
|
||||||
}
|
|
||||||
)
|
|
||||||
setBottomSheetVisible(false)
|
|
||||||
const res = await mutation.mutateAsync({
|
|
||||||
type: 'deleteItem',
|
|
||||||
source: 'statuses',
|
|
||||||
queryKey,
|
|
||||||
id: status.id
|
|
||||||
})
|
|
||||||
if (res.id) {
|
|
||||||
navigation.navigate('Screen-Compose', {
|
|
||||||
type: 'edit',
|
|
||||||
incomingStatus: res,
|
|
||||||
queryKey
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
)
|
|
||||||
}}
|
|
||||||
iconFront='Edit'
|
|
||||||
title={t('shared.header.actions.status.edit.button')}
|
|
||||||
/>
|
|
||||||
<MenuRow
|
|
||||||
onPress={() => {
|
|
||||||
analytics('timeline_shared_headeractions_status_mute_press', {
|
|
||||||
page: queryKey && queryKey[1].page
|
|
||||||
})
|
|
||||||
setBottomSheetVisible(false)
|
|
||||||
mutation.mutate({
|
|
||||||
type: 'updateStatusProperty',
|
|
||||||
queryKey,
|
|
||||||
id: status.id,
|
|
||||||
payload: {
|
|
||||||
property: 'muted',
|
|
||||||
currentValue: status.muted,
|
|
||||||
propertyCount: undefined,
|
|
||||||
countValue: undefined
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}}
|
|
||||||
iconFront='VolumeX'
|
|
||||||
title={
|
|
||||||
status.muted
|
|
||||||
? t('shared.header.actions.status.mute.button.negative')
|
|
||||||
: t('shared.header.actions.status.mute.button.positive')
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
{/* Also note that reblogs cannot be pinned. */}
|
|
||||||
{(status.visibility === 'public' || status.visibility === 'unlisted') && (
|
|
||||||
<MenuRow
|
|
||||||
onPress={() => {
|
|
||||||
analytics('timeline_shared_headeractions_status_pin_press', {
|
|
||||||
page: queryKey && queryKey[1].page
|
|
||||||
})
|
|
||||||
setBottomSheetVisible(false)
|
|
||||||
mutation.mutate({
|
|
||||||
type: 'updateStatusProperty',
|
|
||||||
queryKey,
|
|
||||||
id: status.id,
|
|
||||||
payload: {
|
|
||||||
property: 'pinned',
|
|
||||||
currentValue: status.pinned,
|
|
||||||
propertyCount: undefined,
|
|
||||||
countValue: undefined
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}}
|
|
||||||
iconFront='Anchor'
|
|
||||||
title={
|
|
||||||
status.pinned
|
|
||||||
? t('shared.header.actions.status.pin.button.negative')
|
|
||||||
: t('shared.header.actions.status.pin.button.positive')
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
</MenuContainer>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
export default HeaderActionsStatus
|
|
@ -1,15 +1,19 @@
|
|||||||
import { RelationshipOutgoing } from '@components/Relationship'
|
import Icon from '@components/Icon'
|
||||||
|
import {
|
||||||
|
RelationshipIncoming,
|
||||||
|
RelationshipOutgoing
|
||||||
|
} from '@components/Relationship'
|
||||||
|
import { useNavigation } from '@react-navigation/native'
|
||||||
|
import { QueryKeyTimeline } from '@utils/queryHooks/timeline'
|
||||||
import { StyleConstants } from '@utils/styles/constants'
|
import { StyleConstants } from '@utils/styles/constants'
|
||||||
|
import { useTheme } from '@utils/styles/ThemeManager'
|
||||||
import React, { useMemo } from 'react'
|
import React, { useMemo } from 'react'
|
||||||
import { StyleSheet, View } from 'react-native'
|
import { Pressable, StyleSheet, View } from 'react-native'
|
||||||
import HeaderSharedAccount from './HeaderShared/Account'
|
import HeaderSharedAccount from './HeaderShared/Account'
|
||||||
import HeaderSharedApplication from './HeaderShared/Application'
|
import HeaderSharedApplication from './HeaderShared/Application'
|
||||||
import HeaderSharedCreated from './HeaderShared/Created'
|
import HeaderSharedCreated from './HeaderShared/Created'
|
||||||
import HeaderSharedVisibility from './HeaderShared/Visibility'
|
|
||||||
import RelationshipIncoming from '@root/components/Relationship/Incoming'
|
|
||||||
import HeaderSharedMuted from './HeaderShared/Muted'
|
import HeaderSharedMuted from './HeaderShared/Muted'
|
||||||
import { QueryKeyTimeline } from '@utils/queryHooks/timeline'
|
import HeaderSharedVisibility from './HeaderShared/Visibility'
|
||||||
import ScreenActions from '@screens/Actions'
|
|
||||||
|
|
||||||
export interface Props {
|
export interface Props {
|
||||||
queryKey: QueryKeyTimeline
|
queryKey: QueryKeyTimeline
|
||||||
@ -20,6 +24,9 @@ const TimelineHeaderNotification: React.FC<Props> = ({
|
|||||||
queryKey,
|
queryKey,
|
||||||
notification
|
notification
|
||||||
}) => {
|
}) => {
|
||||||
|
const navigation = useNavigation()
|
||||||
|
const { theme } = useTheme()
|
||||||
|
|
||||||
const actions = useMemo(() => {
|
const actions = useMemo(() => {
|
||||||
switch (notification.type) {
|
switch (notification.type) {
|
||||||
case 'follow':
|
case 'follow':
|
||||||
@ -27,11 +34,36 @@ const TimelineHeaderNotification: React.FC<Props> = ({
|
|||||||
case 'follow_request':
|
case 'follow_request':
|
||||||
return <RelationshipIncoming id={notification.account.id} />
|
return <RelationshipIncoming id={notification.account.id} />
|
||||||
default:
|
default:
|
||||||
return notification.status ? (
|
if (notification.status) {
|
||||||
<ScreenActions queryKey={queryKey} status={notification.status} />
|
return (
|
||||||
) : null
|
<Pressable
|
||||||
|
style={{
|
||||||
|
flex: 1,
|
||||||
|
flexDirection: 'row',
|
||||||
|
justifyContent: 'center',
|
||||||
|
paddingBottom: StyleConstants.Spacing.S
|
||||||
|
}}
|
||||||
|
onPress={() =>
|
||||||
|
navigation.navigate('Screen-Actions', {
|
||||||
|
queryKey,
|
||||||
|
status,
|
||||||
|
url: notification.status?.url || notification.status?.uri,
|
||||||
|
type: 'status'
|
||||||
|
})
|
||||||
|
}
|
||||||
|
children={
|
||||||
|
<Icon
|
||||||
|
name='MoreHorizontal'
|
||||||
|
color={theme.secondary}
|
||||||
|
size={StyleConstants.Font.Size.L}
|
||||||
|
/>
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}, [notification.type])
|
}, [notification.type])
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<View style={styles.base}>
|
<View style={styles.base}>
|
||||||
<View
|
<View
|
||||||
|
@ -122,7 +122,7 @@ const ScreenActions = React.memo(
|
|||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{sameAccount && (
|
{sameAccount && status && (
|
||||||
<ActionsStatus
|
<ActionsStatus
|
||||||
navigation={navigation}
|
navigation={navigation}
|
||||||
queryKey={queryKey}
|
queryKey={queryKey}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user