1
0
mirror of https://github.com/tooot-app/app synced 2025-04-05 14:11:16 +02:00
This commit is contained in:
xmflsct 2022-12-09 21:09:00 +01:00
parent 44f83e44b9
commit 1b58bcad3e
25 changed files with 267 additions and 254 deletions

View File

@ -30,6 +30,7 @@ declare namespace Mastodon {
bot: boolean bot: boolean
source?: Source source?: Source
suspended?: boolean suspended?: boolean
role?: Role
} }
type Announcement = { type Announcement = {
@ -384,6 +385,8 @@ declare namespace Mastodon {
mention: boolean mention: boolean
poll: boolean poll: boolean
status: boolean status: boolean
'admin.sign_up': boolean
'admin.report': boolean
} }
server_key: string server_key: string
} }
@ -409,6 +412,18 @@ declare namespace Mastodon {
hashtags?: Tag[] hashtags?: Tag[]
} }
type Role = {
// Added since 4.0
id: string
name: string
color: string
position: number
permissions: string
highlighted: boolean
created_at: string
updated_at: string
}
type Status = { type Status = {
// Base // Base
id: string id: string
@ -479,25 +494,4 @@ declare namespace Mastodon {
history: { day: string; accounts: string; uses: string }[] history: { day: string; accounts: string; uses: string }[]
following: boolean // Since v4.0 following: boolean // Since v4.0
} }
type WebSocketStream =
| 'user'
| 'public'
| 'public:local'
| 'hashtag'
| 'hashtag:local'
| 'list'
| 'direct'
type WebSocket =
| {
stream: WebSocketStream[]
event: 'update'
payload: string // Status
}
| { stream: WebSocketStream[]; event: 'delete'; payload: Status['id'] }
| {
stream: WebSocketStream[]
event: 'notification'
payload: string // Notification
}
} }

View File

@ -135,10 +135,7 @@ const Screens: React.FC<Props> = ({ localCorrupt }) => {
instance => paths[0] === `@${instance.account.acct}@${instance.uri}` instance => paths[0] === `@${instance.account.acct}@${instance.uri}`
) )
if (instanceIndex !== -1 && instanceActive !== instanceIndex) { if (instanceIndex !== -1 && instanceActive !== instanceIndex) {
initQuery({ initQuery({ instance: instances[instanceIndex] })
instance: instances[instanceIndex],
prefetch: { enabled: true }
})
} }
} }
} }

View File

@ -12,11 +12,7 @@ interface Props {
additionalActions?: () => void additionalActions?: () => void
} }
const AccountButton: React.FC<Props> = ({ const AccountButton: React.FC<Props> = ({ instance, selected = false, additionalActions }) => {
instance,
selected = false,
additionalActions
}) => {
const navigation = useNavigation() const navigation = useNavigation()
return ( return (
@ -27,12 +23,10 @@ const AccountButton: React.FC<Props> = ({
marginBottom: StyleConstants.Spacing.M, marginBottom: StyleConstants.Spacing.M,
marginRight: StyleConstants.Spacing.M marginRight: StyleConstants.Spacing.M
}} }}
content={`@${instance.account.acct}@${instance.uri}${ content={`@${instance.account.acct}@${instance.uri}${selected ? ' ✓' : ''}`}
selected ? ' ✓' : ''
}`}
onPress={() => { onPress={() => {
haptics('Light') haptics('Light')
initQuery({ instance, prefetch: { enabled: true } }) initQuery({ instance })
navigation.goBack() navigation.goBack()
if (additionalActions) { if (additionalActions) {
additionalActions() additionalActions()

View File

@ -19,6 +19,11 @@
"version": 3.5, "version": 3.5,
"reference": "https://github.com/mastodon/mastodon/releases/tag/v3.5.0" "reference": "https://github.com/mastodon/mastodon/releases/tag/v3.5.0"
}, },
{
"feature": "notification_type_admin_signup",
"version": 3.5,
"reference": "https://github.com/mastodon/mastodon/releases/tag/v3.5.0"
},
{ {
"feature": "notification_types_positive_filter", "feature": "notification_types_positive_filter",
"version": 3.5, "version": 3.5,
@ -33,5 +38,10 @@
"feature": "follow_tags", "feature": "follow_tags",
"version": 4.0, "version": 4.0,
"reference": "https://github.com/mastodon/mastodon/releases/tag/v4.0.0" "reference": "https://github.com/mastodon/mastodon/releases/tag/v4.0.0"
},
{
"feature": "notification_type_admin_report",
"version": 4.0,
"reference": "https://github.com/mastodon/mastodon/releases/tag/v3.5.0"
} }
] ]

View File

@ -0,0 +1,2 @@
export const PERMISSION_MANAGE_REPORTS = 0x0000000000000010
export const PERMISSION_MANAGE_USERS = 0x0000000000000400

View File

@ -214,6 +214,12 @@
"status": { "status": {
"heading": "Toot from subscribed users" "heading": "Toot from subscribed users"
}, },
"admin.sign_up": {
"heading": "Admin: sign up"
},
"admin.report": {
"heading": "Admin: reports"
},
"howitworks": "Learn how routing works" "howitworks": "Learn how routing works"
}, },
"root": { "root": {
@ -225,10 +231,8 @@
} }
}, },
"push": { "push": {
"content": { "content_true": "Enabled",
"enabled": "Enabled", "content_false": "Disabled"
"disabled": "Disabled"
}
}, },
"update": { "update": {
"title": "Update to latest version" "title": "Update to latest version"

View File

@ -21,7 +21,7 @@ const TabMeProfileRoot: React.FC<
const { showActionSheetWithOptions } = useActionSheet() const { showActionSheetWithOptions } = useActionSheet()
const { data, isLoading } = useProfileQuery({}) const { data, isLoading } = useProfileQuery()
const { mutateAsync } = useProfileMutation() const { mutateAsync } = useProfileMutation()
const dispatch = useAppDispatch() const dispatch = useAppDispatch()

View File

@ -17,7 +17,7 @@ const ProfileAvatarHeader: React.FC<Props> = ({ type, messageRef }) => {
const { showActionSheetWithOptions } = useActionSheet() const { showActionSheetWithOptions } = useActionSheet()
const query = useProfileQuery({}) const query = useProfileQuery()
const mutation = useProfileMutation() const mutation = useProfileMutation()
return ( return (

View File

@ -3,28 +3,48 @@ import Icon from '@components/Icon'
import { MenuContainer, MenuRow } from '@components/Menu' import { MenuContainer, MenuRow } from '@components/Menu'
import CustomText from '@components/Text' import CustomText from '@components/Text'
import browserPackage from '@helpers/browserPackage' import browserPackage from '@helpers/browserPackage'
import { PERMISSION_MANAGE_REPORTS, PERMISSION_MANAGE_USERS } from '@helpers/permissions'
import { useAppDispatch } from '@root/store' import { useAppDispatch } from '@root/store'
import { isDevelopment } from '@utils/checkEnvironment' import { isDevelopment } from '@utils/checkEnvironment'
import { useProfileQuery } from '@utils/queryHooks/profile'
import { getExpoToken } from '@utils/slices/appSlice' import { getExpoToken } from '@utils/slices/appSlice'
import { updateInstancePush } from '@utils/slices/instances/updatePush' import { updateInstancePush } from '@utils/slices/instances/updatePush'
import { updateInstancePushAlert } from '@utils/slices/instances/updatePushAlert' import { updateInstancePushAlert } from '@utils/slices/instances/updatePushAlert'
import { updateInstancePushDecode } from '@utils/slices/instances/updatePushDecode' import { updateInstancePushDecode } from '@utils/slices/instances/updatePushDecode'
import { import { getInstanceAccount, getInstancePush, getInstanceUri } from '@utils/slices/instancesSlice'
clearPushLoading,
getInstanceAccount,
getInstancePush,
getInstanceUri
} from '@utils/slices/instancesSlice'
import { StyleConstants } from '@utils/styles/constants' import { StyleConstants } from '@utils/styles/constants'
import layoutAnimation from '@utils/styles/layoutAnimation' import layoutAnimation from '@utils/styles/layoutAnimation'
import { useTheme } from '@utils/styles/ThemeManager' import { useTheme } from '@utils/styles/ThemeManager'
import * as Notifications from 'expo-notifications' import * as Notifications from 'expo-notifications'
import * as WebBrowser from 'expo-web-browser' import * as WebBrowser from 'expo-web-browser'
import React, { useState, useEffect, useMemo } from 'react' import React, { useState, useEffect } from 'react'
import { useTranslation } from 'react-i18next' import { useTranslation } from 'react-i18next'
import { AppState, Linking, ScrollView, View } from 'react-native' import { AppState, Linking, ScrollView, View } from 'react-native'
import { useSelector } from 'react-redux' import { useSelector } from 'react-redux'
export const PUSH_DEFAULT: [
'follow',
'follow_request',
'favourite',
'reblog',
'mention',
'poll',
'status'
] = ['follow', 'follow_request', 'favourite', 'reblog', 'mention', 'poll', 'status']
export const PUSH_ADMIN: { type: 'admin.sign_up' | 'admin.report'; permission: number }[] = [
{ type: 'admin.sign_up', permission: PERMISSION_MANAGE_USERS },
{ type: 'admin.report', permission: PERMISSION_MANAGE_REPORTS }
]
export const checkPushAdminPermission = (
permission: number,
permissions?: string | number
): boolean =>
permissions
? !!(
(typeof permissions === 'string' ? parseInt(permissions || '0') : permissions) & permission
)
: false
const TabMePush: React.FC = () => { const TabMePush: React.FC = () => {
const { colors } = useTheme() const { colors } = useTheme()
const { t } = useTranslation('screenTabs') const { t } = useTranslation('screenTabs')
@ -58,40 +78,21 @@ const TabMePush: React.FC = () => {
} }
}, []) }, [])
useEffect(() => { const alerts = () =>
dispatch(clearPushLoading()) instancePush?.alerts
}, []) ? PUSH_DEFAULT.map(alert => (
const isLoading = instancePush?.global.loading || instancePush?.decode.loading
const alerts = useMemo(() => {
return instancePush?.alerts
? (
['follow', 'follow_request', 'favourite', 'reblog', 'mention', 'poll', 'status'] as [
'follow',
'follow_request',
'favourite',
'reblog',
'mention',
'poll',
'status'
]
).map(alert => (
<MenuRow <MenuRow
key={alert} key={alert}
title={t(`me.push.${alert}.heading`)} title={t(`me.push.${alert}.heading`)}
switchDisabled={!pushEnabled || !instancePush.global.value || isLoading} switchDisabled={!pushEnabled || !instancePush.global}
switchValue={instancePush?.alerts[alert].value} switchValue={instancePush?.alerts[alert]}
switchOnValueChange={() => switchOnValueChange={() =>
dispatch( dispatch(
updateInstancePushAlert({ updateInstancePushAlert({
changed: alert, changed: alert,
alerts: { alerts: {
...instancePush?.alerts, ...instancePush?.alerts,
[alert]: { [alert]: instancePush?.alerts[alert]
...instancePush?.alerts[alert],
value: !instancePush?.alerts[alert].value
}
} }
}) })
) )
@ -99,7 +100,32 @@ const TabMePush: React.FC = () => {
/> />
)) ))
: null : null
}, [pushEnabled, instancePush?.global, instancePush?.alerts, isLoading])
const profileQuery = useProfileQuery()
const adminAlerts = () =>
profileQuery.data?.role?.permissions
? PUSH_ADMIN.map(({ type, permission }) =>
checkPushAdminPermission(permission, profileQuery.data.role?.permissions) ? (
<MenuRow
key={type}
title={t(`me.push.${type}.heading`)}
switchDisabled={!pushEnabled || !instancePush.global}
switchValue={instancePush?.alerts[type]}
switchOnValueChange={() =>
dispatch(
updateInstancePushAlert({
changed: type,
alerts: {
...instancePush?.alerts,
[type]: instancePush?.alerts[type]
}
})
)
}
/>
) : null
)
: null
return ( return (
<ScrollView> <ScrollView>
@ -134,22 +160,19 @@ const TabMePush: React.FC = () => {
acct: `@${instanceAccount?.acct}@${instanceUri}` acct: `@${instanceAccount?.acct}@${instanceUri}`
})} })}
description={t('me.push.global.description')} description={t('me.push.global.description')}
loading={instancePush?.global.loading} switchDisabled={!pushEnabled}
switchDisabled={!pushEnabled || isLoading} switchValue={pushEnabled === false ? false : instancePush?.global}
switchValue={pushEnabled === false ? false : instancePush?.global.value} switchOnValueChange={() => dispatch(updateInstancePush(!instancePush?.global))}
switchOnValueChange={() => dispatch(updateInstancePush(!instancePush?.global.value))}
/> />
</MenuContainer> </MenuContainer>
<MenuContainer> <MenuContainer>
<MenuRow <MenuRow
title={t('me.push.decode.heading')} title={t('me.push.decode.heading')}
description={t('me.push.decode.description')} description={t('me.push.decode.description')}
loading={instancePush?.decode.loading} loading={instancePush?.decode}
switchDisabled={!pushEnabled || !instancePush?.global.value || isLoading} switchDisabled={!pushEnabled || !instancePush?.global}
switchValue={instancePush?.decode.value} switchValue={instancePush?.decode}
switchOnValueChange={() => switchOnValueChange={() => dispatch(updateInstancePushDecode(!instancePush?.decode))}
dispatch(updateInstancePushDecode(!instancePush?.decode.value))
}
/> />
<MenuRow <MenuRow
title={t('me.push.howitworks')} title={t('me.push.howitworks')}
@ -161,7 +184,8 @@ const TabMePush: React.FC = () => {
} }
/> />
</MenuContainer> </MenuContainer>
<MenuContainer>{alerts}</MenuContainer> <MenuContainer children={alerts()} />
<MenuContainer children={adminAlerts()} />
</> </>
) : ( ) : (
<View <View
@ -169,11 +193,19 @@ const TabMePush: React.FC = () => {
flex: 1, flex: 1,
minHeight: '100%', minHeight: '100%',
justifyContent: 'center', justifyContent: 'center',
alignItems: 'center' alignItems: 'center',
paddingHorizontal: StyleConstants.Spacing.Global.PagePadding
}} }}
> >
<Icon name='Frown' size={StyleConstants.Font.Size.L} color={colors.primaryDefault} /> <Icon name='Frown' size={StyleConstants.Font.Size.L} color={colors.primaryDefault} />
<CustomText fontStyle='M' style={{ color: colors.primaryDefault }}> <CustomText
fontStyle='M'
style={{
color: colors.primaryDefault,
textAlign: 'center',
marginTop: StyleConstants.Spacing.S
}}
>
{t('me.push.notAvailable')} {t('me.push.notAvailable')}
</CustomText> </CustomText>
</View> </View>

View File

@ -50,10 +50,7 @@ const Collections: React.FC = () => {
} }
}, [announcementsQuery.data]) }, [announcementsQuery.data])
const instancePush = useSelector( const instancePush = useSelector(getInstancePush, (prev, next) => prev?.global === next?.global)
getInstancePush,
(prev, next) => prev?.global.value === next?.global.value
)
return ( return (
<MenuContainer> <MenuContainer>
@ -102,11 +99,7 @@ const Collections: React.FC = () => {
iconFront={instancePush ? 'Bell' : 'BellOff'} iconFront={instancePush ? 'Bell' : 'BellOff'}
iconBack='ChevronRight' iconBack='ChevronRight'
title={t('me.stacks.push.name')} title={t('me.stacks.push.name')}
content={ content={t('me.root.push.content', { context: instancePush.global.toString() })}
instancePush.global.value
? t('me.root.push.content.enabled')
: t('me.root.push.content.disabled')
}
onPress={() => navigation.navigate('Tab-Me-Push')} onPress={() => navigation.navigate('Tab-Me-Push')}
/> />
</MenuContainer> </MenuContainer>

View File

@ -2,6 +2,7 @@ import haptics from '@components/haptics'
import { MenuContainer, MenuRow } from '@components/Menu' import { MenuContainer, MenuRow } from '@components/Menu'
import { LOCALES } from '@root/i18n/locales' import { LOCALES } from '@root/i18n/locales'
import { TabMeStackScreenProps } from '@utils/navigation/navigators' import { TabMeStackScreenProps } from '@utils/navigation/navigators'
import { useProfileQuery } from '@utils/queryHooks/profile'
import androidDefaults from '@utils/slices/instances/push/androidDefaults' import androidDefaults from '@utils/slices/instances/push/androidDefaults'
import { getInstances } from '@utils/slices/instancesSlice' import { getInstances } from '@utils/slices/instancesSlice'
import { changeLanguage } from '@utils/slices/settingsSlice' import { changeLanguage } from '@utils/slices/settingsSlice'
@ -10,15 +11,18 @@ import React from 'react'
import { useTranslation } from 'react-i18next' import { useTranslation } from 'react-i18next'
import { FlatList, Platform } from 'react-native' import { FlatList, Platform } from 'react-native'
import { useDispatch, useSelector } from 'react-redux' import { useDispatch, useSelector } from 'react-redux'
import { checkPushAdminPermission, PUSH_ADMIN, PUSH_DEFAULT } from './Push'
const TabMeSettingsLanguage: React.FC< const TabMeSettingsLanguage: React.FC<TabMeStackScreenProps<'Tab-Me-Settings-Language'>> = ({
TabMeStackScreenProps<'Tab-Me-Settings-Language'> navigation
> = ({ navigation }) => { }) => {
const { i18n, t } = useTranslation('screenTabs') const { i18n, t } = useTranslation('screenTabs')
const languages = Object.entries(LOCALES) const languages = Object.entries(LOCALES)
const instances = useSelector(getInstances) const instances = useSelector(getInstances)
const dispatch = useDispatch() const dispatch = useDispatch()
const profileQuery = useProfileQuery({ options: { enabled: Platform.OS === 'android' } })
const change = (lang: string) => { const change = (lang: string) => {
haptics('Success') haptics('Success')
@ -29,41 +33,29 @@ const TabMeSettingsLanguage: React.FC<
if (Platform.OS === 'android') { if (Platform.OS === 'android') {
instances.forEach(instance => { instances.forEach(instance => {
const accountFull = `@${instance.account.acct}@${instance.uri}` const accountFull = `@${instance.account.acct}@${instance.uri}`
if (instance.push.decode.value === false) { if (instance.push.decode === false) {
Notifications.setNotificationChannelAsync(`${accountFull}_default`, { Notifications.setNotificationChannelAsync(`${accountFull}_default`, {
groupId: accountFull, groupId: accountFull,
name: t('me.push.default.heading'), name: t('me.push.default.heading'),
...androidDefaults ...androidDefaults
}) })
} else { } else {
Notifications.setNotificationChannelAsync(`${accountFull}_follow`, { for (const push of PUSH_DEFAULT) {
groupId: accountFull, Notifications.setNotificationChannelAsync(`${accountFull}_${push}`, {
name: t('me.push.follow.heading'),
...androidDefaults
})
Notifications.setNotificationChannelAsync(
`${accountFull}_favourite`,
{
groupId: accountFull, groupId: accountFull,
name: t('me.push.favourite.heading'), name: t(`me.push.${push}.heading`),
...androidDefaults ...androidDefaults
})
}
for (const { type, permission } of PUSH_ADMIN) {
if (checkPushAdminPermission(permission, profileQuery.data?.role?.permissions)) {
Notifications.setNotificationChannelAsync(`${accountFull}_${type}`, {
groupId: accountFull,
name: t(`me.push.${type}.heading`),
...androidDefaults
})
} }
) }
Notifications.setNotificationChannelAsync(`${accountFull}_reblog`, {
groupId: accountFull,
name: t('me.push.reblog.heading'),
...androidDefaults
})
Notifications.setNotificationChannelAsync(`${accountFull}_mention`, {
groupId: accountFull,
name: t('me.push.mention.heading'),
...androidDefaults
})
Notifications.setNotificationChannelAsync(`${accountFull}_poll`, {
groupId: accountFull,
name: t('me.push.poll.heading'),
...androidDefaults
})
} }
}) })
} }

View File

@ -1,13 +1,8 @@
import apiInstance from '@api/instance' import apiInstance from '@api/instance'
import NetInfo from '@react-native-community/netinfo' import NetInfo from '@react-native-community/netinfo'
import { store } from '@root/store' import { store } from '@root/store'
import initQuery from '@utils/initQuery'
import { getPreviousTab } from '@utils/slices/contextsSlice'
import removeInstance from '@utils/slices/instances/remove' import removeInstance from '@utils/slices/instances/remove'
import { import { getInstance, updateInstanceAccount } from '@utils/slices/instancesSlice'
getInstance,
updateInstanceAccount
} from '@utils/slices/instancesSlice'
import { onlineManager } from 'react-query' import { onlineManager } from 'react-query'
import log from './log' import log from './log'
@ -22,9 +17,7 @@ const netInfo = async (): Promise<{
onlineManager.setEventListener(setOnline => { onlineManager.setEventListener(setOnline => {
return NetInfo.addEventListener(state => { return NetInfo.addEventListener(state => {
setOnline( setOnline(typeof state.isConnected === 'boolean' ? state.isConnected : undefined)
typeof state.isConnected === 'boolean' ? state.isConnected : undefined
)
}) })
}) })
@ -60,23 +53,6 @@ const netInfo = async (): Promise<{
}) })
) )
if (instance.timelinesLookback) {
const previousTab = getPreviousTab(store.getState())
let loadPage:
| Extract<App.Pages, 'Following' | 'Local' | 'LocalPublic'>
| undefined = undefined
if (previousTab === 'Tab-Local') {
loadPage = 'Following'
} else if (previousTab === 'Tab-Public') {
loadPage = 'LocalPublic'
}
// await initQuery({
// instance,
// prefetch: { enabled: true, page: loadPage }
// })
}
return Promise.resolve({ connected: true }) return Promise.resolve({ connected: true })
} }
} else { } else {

View File

@ -46,7 +46,7 @@ const instancesPersistConfig = {
key: 'instances', key: 'instances',
prefix, prefix,
storage: Platform.OS === 'ios' ? secureStorage : AsyncStorage, storage: Platform.OS === 'ios' ? secureStorage : AsyncStorage,
version: 10, version: 11,
// @ts-ignore // @ts-ignore
migrate: createMigrate(instancesMigration) migrate: createMigrate(instancesMigration)
} }

View File

@ -1,33 +1,11 @@
import queryClient from '@helpers/queryClient' import queryClient from '@helpers/queryClient'
import { store } from '@root/store' import { store } from '@root/store'
import { InstanceLatest } from './migrations/instances/migration' import { InstanceLatest } from './migrations/instances/migration'
// import { prefetchTimelineQuery } from './queryHooks/timeline'
import { updateInstanceActive } from './slices/instancesSlice' import { updateInstanceActive } from './slices/instancesSlice'
const initQuery = async ({ const initQuery = async ({ instance }: { instance: InstanceLatest }) => {
instance,
prefetch
}: {
instance: InstanceLatest
prefetch?: { enabled: boolean; page?: 'Following' | 'LocalPublic' }
}) => {
store.dispatch(updateInstanceActive(instance)) store.dispatch(updateInstanceActive(instance))
await queryClient.resetQueries() await queryClient.resetQueries()
// if (prefetch?.enabled && instance.timelinesLookback) {
// if (
// prefetch.page &&
// instance.timelinesLookback[prefetch.page]?.ids?.length > 0
// ) {
// await prefetchTimelineQuery(instance.timelinesLookback[prefetch.page])
// }
// for (const page of Object.keys(instance.timelinesLookback)) {
// if (page !== prefetch.page) {
// prefetchTimelineQuery(instance.timelinesLookback[page])
// }
// }
// }
} }
export default initQuery export default initQuery

View File

@ -6,6 +6,7 @@ import { InstanceV7 } from './v7'
import { InstanceV8 } from './v8' import { InstanceV8 } from './v8'
import { InstanceV9 } from './v9' import { InstanceV9 } from './v9'
import { InstanceV10 } from './v10' import { InstanceV10 } from './v10'
import { InstanceV11 } from './v11'
const instancesMigration = { const instancesMigration = {
4: (state: InstanceV3): InstanceV4 => { 4: (state: InstanceV3): InstanceV4 => {
@ -128,9 +129,34 @@ const instancesMigration = {
} }
}) })
} }
},
11: (state: { instances: InstanceV10[] }): { instances: InstanceV11[] } => {
return {
instances: state.instances.map(instance => {
return {
...instance,
push: {
...instance.push,
global: instance.push.global.value,
decode: instance.push.decode.value,
alerts: {
follow: instance.push.alerts.follow.value,
follow_request: instance.push.alerts.follow_request.value,
favourite: instance.push.alerts.favourite.value,
reblog: instance.push.alerts.reblog.value,
mention: instance.push.alerts.mention.value,
poll: instance.push.alerts.poll.value,
status: instance.push.alerts.status.value,
'admin.sign_up': false,
'admin.report': false
}
}
}
})
}
} }
} }
export { InstanceV10 as InstanceLatest } export { InstanceV11 as InstanceLatest }
export default instancesMigration export default instancesMigration

View File

@ -0,0 +1,60 @@
import { ComposeStateDraft } from '@screens/Compose/utils/types'
import { QueryKeyTimeline } from '@utils/queryHooks/timeline'
export type InstanceV11 = {
active: boolean
appData: {
clientId: string
clientSecret: string
}
url: string
token: string
uri: Mastodon.Instance['uri']
urls: Mastodon.Instance['urls']
account: {
id: Mastodon.Account['id']
acct: Mastodon.Account['acct']
avatarStatic: Mastodon.Account['avatar_static']
preferences: Mastodon.Preferences
}
version: string
configuration?: Mastodon.Instance['configuration']
filters: Mastodon.Filter[]
notifications_filter: {
follow: boolean
follow_request: boolean
favourite: boolean
reblog: boolean
mention: boolean
poll: boolean
status: boolean
update: boolean
}
push: {
global: boolean
decode: boolean
alerts: Mastodon.PushSubscription['alerts']
keys: {
auth?: string
public?: string // legacy
private?: string // legacy
}
}
timelinesLookback?: {
[key: string]: {
queryKey: QueryKeyTimeline
ids: Mastodon.Status['id'][]
}
}
mePage: {
lists: { shown: boolean }
announcements: { shown: boolean; unread: number }
}
drafts: ComposeStateDraft[]
frequentEmojis: {
emoji: Pick<Mastodon.Emoji, 'shortcode' | 'url' | 'static_url'>
score: number
count: number
lastUsed: number
}[]
}

View File

@ -23,7 +23,7 @@ const pushUseConnect = () => {
const expoToken = useSelector(getExpoToken) const expoToken = useSelector(getExpoToken)
const instances = useSelector(getInstances, (prev, next) => prev.length === next.length) const instances = useSelector(getInstances, (prev, next) => prev.length === next.length)
const pushEnabled = instances.filter(instance => instance.push.global.value) const pushEnabled = instances.filter(instance => instance.push.global)
const connect = () => { const connect = () => {
apiTooot({ apiTooot({
@ -60,7 +60,7 @@ const pushUseConnect = () => {
dispatch(disableAllPushes()) dispatch(disableAllPushes())
instances.forEach(instance => { instances.forEach(instance => {
if (instance.push.global.value) { if (instance.push.global) {
apiGeneral<{}>({ apiGeneral<{}>({
method: 'delete', method: 'delete',
domain: instance.url, domain: instance.url,

View File

@ -31,10 +31,7 @@ const pushUseReceive = () => {
description: notification.request.content.body!, description: notification.request.content.body!,
onPress: () => { onPress: () => {
if (notificationIndex !== -1) { if (notificationIndex !== -1) {
initQuery({ initQuery({ instance: instances[notificationIndex] })
instance: instances[notificationIndex],
prefetch: { enabled: true }
})
} }
pushUseNavigate(payloadData.notification_id) pushUseNavigate(payloadData.notification_id)
} }

View File

@ -27,10 +27,7 @@ const pushUseRespond = () => {
instance.account.id === payloadData.accountId instance.account.id === payloadData.accountId
) )
if (notificationIndex !== -1) { if (notificationIndex !== -1) {
initQuery({ initQuery({ instance: instances[notificationIndex] })
instance: instances[notificationIndex],
prefetch: { enabled: true }
})
} }
pushUseNavigate(payloadData.notification_id) pushUseNavigate(payloadData.notification_id)
} }

View File

@ -21,12 +21,12 @@ const queryFunction = async () => {
return res.body return res.body
} }
const useProfileQuery = ({ const useProfileQuery = (
options params: {
}: { options: UseQueryOptions<AccountWithSource, AxiosError>
options?: UseQueryOptions<AccountWithSource, AxiosError> } | void
}) => { ) => {
return useQuery(queryKey, queryFunction, options) return useQuery(queryKey, queryFunction, params?.options)
} }
type MutationVarsProfileBase = type MutationVarsProfileBase =

View File

@ -90,16 +90,18 @@ const addInstance = createAsyncThunk(
update: true update: true
}, },
push: { push: {
global: { loading: false, value: false }, global: false,
decode: { loading: false, value: false }, decode: false,
alerts: { alerts: {
follow: { loading: false, value: true }, follow: true,
follow_request: { loading: false, value: true }, follow_request: true,
favourite: { loading: false, value: true }, favourite: true,
reblog: { loading: false, value: true }, reblog: true,
mention: { loading: false, value: true }, mention: true,
poll: { loading: false, value: true }, poll: true,
status: { loading: false, value: true } status: true,
'admin.sign_up': false,
'admin.report': false
}, },
keys: { auth: undefined, public: undefined, private: undefined } keys: { auth: undefined, public: undefined, private: undefined }
}, },

View File

@ -6,7 +6,6 @@ import { RootState } from '@root/store'
import * as Sentry from '@sentry/react-native' import * as Sentry from '@sentry/react-native'
import { InstanceLatest } from '@utils/migrations/instances/migration' import { InstanceLatest } from '@utils/migrations/instances/migration'
import { getInstance } from '@utils/slices/instancesSlice' import { getInstance } from '@utils/slices/instancesSlice'
import { Theme } from '@utils/styles/themes'
import * as Notifications from 'expo-notifications' import * as Notifications from 'expo-notifications'
import * as Random from 'expo-random' import * as Random from 'expo-random'
import i18next from 'i18next' import i18next from 'i18next'
@ -97,7 +96,7 @@ const pushRegister = async (
accountId, accountId,
accountFull, accountFull,
serverKey: res.body.server_key, serverKey: res.body.server_key,
auth: instancePush.decode.value === false ? null : auth auth: instancePush.decode === false ? null : auth
}) })
if (Platform.OS === 'android') { if (Platform.OS === 'android') {
@ -106,7 +105,7 @@ const pushRegister = async (
...androidDefaults ...androidDefaults
}).then(group => { }).then(group => {
if (group) { if (group) {
if (instancePush.decode.value === false) { if (instancePush.decode === false) {
Notifications.setNotificationChannelAsync(`${group.id}_default`, { Notifications.setNotificationChannelAsync(`${group.id}_default`, {
groupId: group.id, groupId: group.id,
name: i18n.t('meSettingsPush:content.default.heading'), name: i18n.t('meSettingsPush:content.default.heading'),

View File

@ -6,7 +6,7 @@ import { updateInstancePush } from './updatePush'
const removeInstance = createAsyncThunk( const removeInstance = createAsyncThunk(
'instances/remove', 'instances/remove',
async (instance: InstanceLatest, { dispatch }): Promise<InstanceLatest> => { async (instance: InstanceLatest, { dispatch }): Promise<InstanceLatest> => {
if (instance.push.global.value) { if (instance.push.global) {
dispatch(updateInstancePush(false)) dispatch(updateInstancePush(false))
} }

View File

@ -13,7 +13,7 @@ export const updateInstancePushDecode = createAsyncThunk(
async ( async (
disable: boolean, disable: boolean,
{ getState } { getState }
): Promise<{ disable: InstanceLatest['push']['decode']['value'] }> => { ): Promise<{ disable: InstanceLatest['push']['decode'] }> => {
const state = getState() as RootState const state = getState() as RootState
const instance = getInstance(state) const instance = getInstance(state)
if (!instance?.url || !instance.account.id || !instance.push.keys) { if (!instance?.url || !instance.account.id || !instance.push.keys) {
@ -43,14 +43,11 @@ export const updateInstancePushDecode = createAsyncThunk(
name: i18n.t('meSettingsPush:content.follow.heading'), name: i18n.t('meSettingsPush:content.follow.heading'),
...androidDefaults ...androidDefaults
}) })
Notifications.setNotificationChannelAsync( Notifications.setNotificationChannelAsync(`${accountFull}_favourite`, {
`${accountFull}_favourite`, groupId: accountFull,
{ name: i18n.t('meSettingsPush:content.favourite.heading'),
groupId: accountFull, ...androidDefaults
name: i18n.t('meSettingsPush:content.favourite.heading'), })
...androidDefaults
}
)
Notifications.setNotificationChannelAsync(`${accountFull}_reblog`, { Notifications.setNotificationChannelAsync(`${accountFull}_reblog`, {
groupId: accountFull, groupId: accountFull,
name: i18n.t('meSettingsPush:content.reblog.heading'), name: i18n.t('meSettingsPush:content.reblog.heading'),
@ -74,9 +71,7 @@ export const updateInstancePushDecode = createAsyncThunk(
...androidDefaults ...androidDefaults
}) })
Notifications.deleteNotificationChannelAsync(`${accountFull}_follow`) Notifications.deleteNotificationChannelAsync(`${accountFull}_follow`)
Notifications.deleteNotificationChannelAsync( Notifications.deleteNotificationChannelAsync(`${accountFull}_favourite`)
`${accountFull}_favourite`
)
Notifications.deleteNotificationChannelAsync(`${accountFull}_reblog`) Notifications.deleteNotificationChannelAsync(`${accountFull}_reblog`)
Notifications.deleteNotificationChannelAsync(`${accountFull}_mention`) Notifications.deleteNotificationChannelAsync(`${accountFull}_mention`)
Notifications.deleteNotificationChannelAsync(`${accountFull}_poll`) Notifications.deleteNotificationChannelAsync(`${accountFull}_poll`)

View File

@ -73,18 +73,11 @@ const instancesSlice = createSlice({
}, },
clearPushLoading: ({ instances }) => { clearPushLoading: ({ instances }) => {
const activeIndex = findInstanceActive(instances) const activeIndex = findInstanceActive(instances)
instances[activeIndex].push.global.loading = false
instances[activeIndex].push.decode.loading = false
instances[activeIndex].push.alerts.favourite.loading = false
instances[activeIndex].push.alerts.follow.loading = false
instances[activeIndex].push.alerts.mention.loading = false
instances[activeIndex].push.alerts.poll.loading = false
instances[activeIndex].push.alerts.reblog.loading = false
}, },
disableAllPushes: ({ instances }) => { disableAllPushes: ({ instances }) => {
instances = instances.map(instance => { instances = instances.map(instance => {
let newInstance = instance let newInstance = instance
newInstance.push.global.value = false newInstance.push.global = false
return newInstance return newInstance
}) })
}, },
@ -238,48 +231,21 @@ const instancesSlice = createSlice({
// Update Instance Push Global // Update Instance Push Global
.addCase(updateInstancePush.fulfilled, (state, action) => { .addCase(updateInstancePush.fulfilled, (state, action) => {
const activeIndex = findInstanceActive(state.instances) const activeIndex = findInstanceActive(state.instances)
state.instances[activeIndex].push.global.loading = false state.instances[activeIndex].push.global = action.meta.arg
state.instances[activeIndex].push.global.value = action.meta.arg
state.instances[activeIndex].push.keys = { auth: action.payload } state.instances[activeIndex].push.keys = { auth: action.payload }
}) })
.addCase(updateInstancePush.rejected, state => {
const activeIndex = findInstanceActive(state.instances)
state.instances[activeIndex].push.global.loading = false
})
.addCase(updateInstancePush.pending, state => {
const activeIndex = findInstanceActive(state.instances)
state.instances[activeIndex].push.global.loading = true
})
// Update Instance Push Decode // Update Instance Push Decode
.addCase(updateInstancePushDecode.fulfilled, (state, action) => { .addCase(updateInstancePushDecode.fulfilled, (state, action) => {
const activeIndex = findInstanceActive(state.instances) const activeIndex = findInstanceActive(state.instances)
state.instances[activeIndex].push.decode.loading = false state.instances[activeIndex].push.decode = action.payload.disable
state.instances[activeIndex].push.decode.value = action.payload.disable
})
.addCase(updateInstancePushDecode.rejected, state => {
const activeIndex = findInstanceActive(state.instances)
state.instances[activeIndex].push.decode.loading = false
})
.addCase(updateInstancePushDecode.pending, state => {
const activeIndex = findInstanceActive(state.instances)
state.instances[activeIndex].push.decode.loading = true
}) })
// Update Instance Push Individual Alert // Update Instance Push Individual Alert
.addCase(updateInstancePushAlert.fulfilled, (state, action) => { .addCase(updateInstancePushAlert.fulfilled, (state, action) => {
const activeIndex = findInstanceActive(state.instances) const activeIndex = findInstanceActive(state.instances)
state.instances[activeIndex].push.alerts[action.meta.arg.changed].loading = false
state.instances[activeIndex].push.alerts = action.payload state.instances[activeIndex].push.alerts = action.payload
}) })
.addCase(updateInstancePushAlert.rejected, (state, action) => {
const activeIndex = findInstanceActive(state.instances)
state.instances[activeIndex].push.alerts[action.meta.arg.changed].loading = false
})
.addCase(updateInstancePushAlert.pending, (state, action) => {
const activeIndex = findInstanceActive(state.instances)
state.instances[activeIndex].push.alerts[action.meta.arg.changed].loading = true
})
// Check if frequently used emojis still exist // Check if frequently used emojis still exist
.addCase(checkEmojis.fulfilled, (state, action) => { .addCase(checkEmojis.fulfilled, (state, action) => {
@ -412,7 +378,6 @@ export const {
updateInstanceNotificationsFilter, updateInstanceNotificationsFilter,
updateInstanceDraft, updateInstanceDraft,
removeInstanceDraft, removeInstanceDraft,
clearPushLoading,
disableAllPushes, disableAllPushes,
updateInstanceTimelineLookback, updateInstanceTimelineLookback,
updateInstanceMePage, updateInstanceMePage,