2023-01-11 09:17:55 -08:00
|
|
|
import * as React from 'react';
|
|
|
|
import { connect } from 'react-redux';
|
2023-02-18 15:05:48 +00:00
|
|
|
import NotesScreen from './screens/Notes';
|
2024-09-24 07:12:02 -07:00
|
|
|
import SearchScreen from './screens/SearchScreen';
|
2023-01-11 09:17:55 -08:00
|
|
|
import { Component } from 'react';
|
|
|
|
import { KeyboardAvoidingView, Keyboard, Platform, View, KeyboardEvent, Dimensions, EmitterSubscription } from 'react-native';
|
|
|
|
import { AppState } from '../utils/types';
|
2024-03-09 03:15:13 -08:00
|
|
|
import { themeStyle } from './global-style';
|
2017-07-14 18:49:14 +00:00
|
|
|
|
2023-01-11 09:17:55 -08:00
|
|
|
interface State {
|
|
|
|
autoCompletionBarExtraHeight: number;
|
|
|
|
floatingKeyboardEnabled: boolean;
|
|
|
|
}
|
|
|
|
|
|
|
|
interface Props {
|
2024-04-05 12:16:49 +01:00
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
2023-01-11 09:17:55 -08:00
|
|
|
route: any;
|
2024-04-05 12:16:49 +01:00
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
2023-01-11 09:17:55 -08:00
|
|
|
screens: any;
|
2024-04-05 12:16:49 +01:00
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
2023-01-11 09:17:55 -08:00
|
|
|
dispatch: (action: any)=> void;
|
|
|
|
themeId: number;
|
|
|
|
}
|
|
|
|
|
|
|
|
class AppNavComponent extends Component<Props, State> {
|
|
|
|
private previousRouteName_: string|null = null;
|
|
|
|
private keyboardDidShowListener: EmitterSubscription|null = null;
|
|
|
|
private keyboardDidHideListener: EmitterSubscription|null = null;
|
|
|
|
private keyboardWillChangeFrameListener: EmitterSubscription|null = null;
|
|
|
|
|
2023-03-06 14:22:01 +00:00
|
|
|
public constructor(props: Props) {
|
2023-01-11 09:17:55 -08:00
|
|
|
super(props);
|
|
|
|
|
2017-08-01 21:08:38 +02:00
|
|
|
this.previousRouteName_ = null;
|
2018-02-05 18:32:59 +00:00
|
|
|
this.state = {
|
|
|
|
autoCompletionBarExtraHeight: 0, // Extra padding for the auto completion bar at the top of the keyboard
|
2023-01-11 09:17:55 -08:00
|
|
|
floatingKeyboardEnabled: false,
|
2019-07-29 15:43:53 +02:00
|
|
|
};
|
2018-02-05 18:32:59 +00:00
|
|
|
}
|
|
|
|
|
2023-03-06 14:22:01 +00:00
|
|
|
public UNSAFE_componentWillMount() {
|
2018-02-05 18:32:59 +00:00
|
|
|
if (Platform.OS === 'ios') {
|
|
|
|
this.keyboardDidShowListener = Keyboard.addListener('keyboardDidShow', this.keyboardDidShow.bind(this));
|
|
|
|
this.keyboardDidHideListener = Keyboard.addListener('keyboardDidHide', this.keyboardDidHide.bind(this));
|
2023-01-11 09:17:55 -08:00
|
|
|
this.keyboardWillChangeFrameListener = Keyboard.addListener('keyboardWillChangeFrame', this.keyboardWillChangeFrame);
|
2018-02-05 18:32:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-06 14:22:01 +00:00
|
|
|
public componentWillUnmount() {
|
2023-01-11 09:17:55 -08:00
|
|
|
this.keyboardDidShowListener?.remove();
|
|
|
|
this.keyboardDidHideListener?.remove();
|
|
|
|
this.keyboardWillChangeFrameListener?.remove();
|
|
|
|
|
2018-02-05 18:32:59 +00:00
|
|
|
this.keyboardDidShowListener = null;
|
|
|
|
this.keyboardDidHideListener = null;
|
2023-01-11 09:17:55 -08:00
|
|
|
this.keyboardWillChangeFrameListener = null;
|
2018-02-05 18:32:59 +00:00
|
|
|
}
|
|
|
|
|
2023-03-06 14:22:01 +00:00
|
|
|
public keyboardDidShow() {
|
2019-07-29 15:43:53 +02:00
|
|
|
this.setState({ autoCompletionBarExtraHeight: 30 });
|
2018-02-05 18:32:59 +00:00
|
|
|
}
|
|
|
|
|
2023-03-06 14:22:01 +00:00
|
|
|
public keyboardDidHide() {
|
2019-07-29 15:43:53 +02:00
|
|
|
this.setState({ autoCompletionBarExtraHeight: 0 });
|
2017-08-01 21:08:38 +02:00
|
|
|
}
|
2017-08-01 17:59:01 +00:00
|
|
|
|
2023-03-06 14:22:01 +00:00
|
|
|
private keyboardWillChangeFrame = (evt: KeyboardEvent) => {
|
2023-01-11 09:17:55 -08:00
|
|
|
const windowWidth = Dimensions.get('window').width;
|
|
|
|
|
2024-02-26 10:16:23 +00:00
|
|
|
// If the keyboard isn't as wide as the window, the floating keyboard is disabled.
|
2023-01-11 09:17:55 -08:00
|
|
|
// See https://github.com/facebook/react-native/issues/29473#issuecomment-696658937
|
|
|
|
this.setState({
|
|
|
|
floatingKeyboardEnabled: evt.endCoordinates.width < windowWidth,
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2023-03-06 14:22:01 +00:00
|
|
|
public render() {
|
2017-07-14 18:49:14 +00:00
|
|
|
if (!this.props.route) throw new Error('Route must not be null');
|
|
|
|
|
2017-08-01 21:08:38 +02:00
|
|
|
// Note: certain screens are kept into memory, in particular Notes and Search
|
|
|
|
// so that the scroll position is not lost when the user navigate away from them.
|
|
|
|
|
2020-03-13 23:46:14 +00:00
|
|
|
const route = this.props.route;
|
2017-07-31 22:03:12 +02:00
|
|
|
let Screen = null;
|
|
|
|
let notesScreenVisible = false;
|
2017-08-01 21:08:38 +02:00
|
|
|
let searchScreenVisible = false;
|
2017-07-31 22:03:12 +02:00
|
|
|
|
2022-07-23 09:31:32 +02:00
|
|
|
if (route.routeName === 'Notes') {
|
2017-07-31 22:03:12 +02:00
|
|
|
notesScreenVisible = true;
|
2022-07-23 09:31:32 +02:00
|
|
|
} else if (route.routeName === 'Search') {
|
2017-08-01 21:08:38 +02:00
|
|
|
searchScreenVisible = true;
|
2017-07-31 22:03:12 +02:00
|
|
|
} else {
|
|
|
|
Screen = this.props.screens[route.routeName].screen;
|
|
|
|
}
|
2017-07-14 18:49:14 +00:00
|
|
|
|
2017-08-01 21:08:38 +02:00
|
|
|
// Keep the search screen loaded if the user is viewing a note from that search screen
|
|
|
|
// so that if the back button is pressed, the screen is still loaded. However, unload
|
|
|
|
// it if navigating away.
|
2022-07-23 09:31:32 +02:00
|
|
|
const searchScreenLoaded = searchScreenVisible || (this.previousRouteName_ === 'Search' && route.routeName === 'Note');
|
2017-08-01 21:08:38 +02:00
|
|
|
|
|
|
|
this.previousRouteName_ = route.routeName;
|
|
|
|
|
2020-09-15 14:01:07 +01:00
|
|
|
const theme = themeStyle(this.props.themeId);
|
2017-10-30 18:27:51 +00:00
|
|
|
|
2019-07-29 15:43:53 +02:00
|
|
|
const style = { flex: 1, backgroundColor: theme.backgroundColor };
|
2017-10-30 18:27:51 +00:00
|
|
|
|
2024-02-26 10:16:23 +00:00
|
|
|
// When the floating keyboard is enabled, the KeyboardAvoidingView can have a very small
|
2023-01-11 09:17:55 -08:00
|
|
|
// height. Don't use the KeyboardAvoidingView when the floating keyboard is enabled.
|
|
|
|
// See https://github.com/facebook/react-native/issues/29473
|
|
|
|
const keyboardAvoidingViewEnabled = !this.state.floatingKeyboardEnabled;
|
|
|
|
|
2017-07-14 18:49:14 +00:00
|
|
|
return (
|
2023-01-11 09:17:55 -08:00
|
|
|
<KeyboardAvoidingView
|
|
|
|
enabled={keyboardAvoidingViewEnabled}
|
|
|
|
behavior={Platform.OS === 'ios' ? 'padding' : null}
|
|
|
|
style={style}
|
|
|
|
>
|
2024-09-21 04:57:38 -07:00
|
|
|
<NotesScreen visible={notesScreenVisible} />
|
2024-09-21 10:28:33 -07:00
|
|
|
{searchScreenLoaded && <SearchScreen visible={searchScreenVisible} />}
|
2023-01-10 12:08:13 +00:00
|
|
|
{!notesScreenVisible && !searchScreenVisible && <Screen navigation={{ state: route }} themeId={this.props.themeId} dispatch={this.props.dispatch} />}
|
2018-02-05 18:32:59 +00:00
|
|
|
<View style={{ height: this.state.autoCompletionBarExtraHeight }} />
|
|
|
|
</KeyboardAvoidingView>
|
2017-07-14 18:49:14 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-01-11 09:17:55 -08:00
|
|
|
const AppNav = connect((state: AppState) => {
|
2019-07-29 15:43:53 +02:00
|
|
|
return {
|
|
|
|
route: state.route,
|
2020-09-15 14:01:07 +01:00
|
|
|
themeId: state.settings.theme,
|
2019-07-29 15:43:53 +02:00
|
|
|
};
|
|
|
|
})(AppNavComponent);
|
2017-07-14 18:49:14 +00:00
|
|
|
|
2019-07-29 15:43:53 +02:00
|
|
|
module.exports = { AppNav };
|