forked from Ivasoft/mattermost-mobile
* Refactor app database queries to not require the app database as argument * Android Share Extension and fix notifications prompt * feedback review
133 lines
4.2 KiB
TypeScript
133 lines
4.2 KiB
TypeScript
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
// See LICENSE.txt for license information.
|
|
|
|
import {NavigationContainer} from '@react-navigation/native';
|
|
import {createStackNavigator} from '@react-navigation/stack';
|
|
import React, {useEffect, useMemo, useState} from 'react';
|
|
import {IntlProvider} from 'react-intl';
|
|
import {Appearance, BackHandler, NativeModules} from 'react-native';
|
|
|
|
import {getDefaultThemeByAppearance} from '@context/theme';
|
|
import {DEFAULT_LOCALE, getTranslations} from '@i18n';
|
|
import {initialize} from '@init/app';
|
|
import {extractStartLink, isValidUrl} from '@utils/url';
|
|
|
|
import ChannelsScreen from './screens/channels';
|
|
import ServersScreen from './screens/servers';
|
|
import ShareScreen from './screens/share';
|
|
|
|
const ShareModule: NativeShareExtension = NativeModules.MattermostShare;
|
|
|
|
const Stack = createStackNavigator();
|
|
|
|
const closeExtension = (data: ShareExtensionDataToSend | null = null) => {
|
|
ShareModule.close(data);
|
|
};
|
|
|
|
const ShareExtension = () => {
|
|
const [data, setData] = useState<SharedItem[]>();
|
|
const [theme, setTheme] = useState<Theme>(getDefaultThemeByAppearance());
|
|
|
|
const defaultNavigationOptions = useMemo(() => ({
|
|
headerStyle: {
|
|
backgroundColor: theme.sidebarHeaderBg,
|
|
},
|
|
headerTitleStyle: {
|
|
marginHorizontal: 0,
|
|
left: 0,
|
|
color: theme.sidebarHeaderTextColor,
|
|
},
|
|
headerBackTitleStyle: {
|
|
color: theme.sidebarHeaderTextColor,
|
|
margin: 0,
|
|
},
|
|
headerTintColor: theme.sidebarHeaderTextColor,
|
|
headerTopInsetEnabled: false,
|
|
cardStyle: {backgroundColor: theme.centerChannelBg},
|
|
}), [theme]);
|
|
|
|
const {text: message, link: linkPreviewUrl} = useMemo(() => {
|
|
let text = data?.filter((i) => i.isString)[0]?.value;
|
|
let link;
|
|
if (text) {
|
|
const first = extractStartLink(text);
|
|
if (first && isValidUrl(first)) {
|
|
link = first;
|
|
text = text.replace(first, '');
|
|
}
|
|
}
|
|
|
|
return {text, link};
|
|
}, [data]);
|
|
|
|
const files = useMemo(() => {
|
|
return data?.filter((i) => !i.isString) || [];
|
|
}, [data]);
|
|
|
|
useEffect(() => {
|
|
initialize().finally(async () => {
|
|
const items = await ShareModule.getSharedData();
|
|
setData(items);
|
|
});
|
|
|
|
const backListener = BackHandler.addEventListener('hardwareBackPress', () => {
|
|
const scene = ShareModule.getCurrentActivityName();
|
|
if (scene === 'ShareActivity') {
|
|
closeExtension();
|
|
return true;
|
|
}
|
|
return false;
|
|
});
|
|
|
|
const appearanceListener = Appearance.addChangeListener(() => {
|
|
setTheme(getDefaultThemeByAppearance());
|
|
});
|
|
|
|
return () => {
|
|
backListener.remove();
|
|
appearanceListener.remove();
|
|
};
|
|
}, []);
|
|
|
|
if (!data) {
|
|
return null;
|
|
}
|
|
|
|
return (
|
|
<IntlProvider
|
|
locale={DEFAULT_LOCALE}
|
|
messages={getTranslations(DEFAULT_LOCALE)}
|
|
>
|
|
<NavigationContainer>
|
|
<Stack.Navigator
|
|
initialRouteName='Share'
|
|
screenOptions={defaultNavigationOptions}
|
|
>
|
|
<Stack.Screen name='Share'>
|
|
{() => (
|
|
<ShareScreen
|
|
files={files}
|
|
linkPreviewUrl={linkPreviewUrl}
|
|
message={message}
|
|
theme={theme}
|
|
/>
|
|
)}
|
|
</Stack.Screen>
|
|
<Stack.Screen name='Servers'>
|
|
{() => (
|
|
<ServersScreen theme={theme}/>
|
|
)}
|
|
</Stack.Screen>
|
|
<Stack.Screen name='Channels'>
|
|
{() => (
|
|
<ChannelsScreen theme={theme}/>
|
|
)}
|
|
</Stack.Screen>
|
|
</Stack.Navigator>
|
|
</NavigationContainer>
|
|
</IntlProvider>
|
|
);
|
|
};
|
|
|
|
export default ShareExtension;
|