mirror of
https://github.com/streetwriters/notesnook.git
synced 2025-12-16 19:57:52 +01:00
319 lines
8.9 KiB
JavaScript
319 lines
8.9 KiB
JavaScript
import React, {useEffect, useState} from 'react';
|
|
import {Dimensions, Platform, View} from 'react-native';
|
|
import ScrollableTabView from 'react-native-scrollable-tab-view';
|
|
import {notesnook} from './e2e/test.ids';
|
|
import ContextMenu from './src/components/ContextMenu';
|
|
import {DialogManager} from './src/components/DialogManager';
|
|
import {DummyText} from './src/components/DummyText';
|
|
import {Menu} from './src/components/Menu';
|
|
import {Toast} from './src/components/Toast';
|
|
import {NavigationStack} from './src/navigation/Drawer';
|
|
import {NavigatorStack} from './src/navigation/NavigatorStack';
|
|
import {useTracked} from './src/provider';
|
|
import {Actions} from './src/provider/Actions';
|
|
import {DDS} from './src/services/DeviceDetection';
|
|
import {
|
|
eSendEvent,
|
|
eSubscribeEvent,
|
|
eUnSubscribeEvent,
|
|
} from './src/services/EventManager';
|
|
import {editing, setWidthHeight} from './src/utils';
|
|
import {
|
|
eCloseFullscreenEditor,
|
|
eCloseSideMenu,
|
|
eOnLoadNote,
|
|
eOpenFullscreenEditor,
|
|
eOpenSideMenu,
|
|
} from './src/utils/Events';
|
|
import {editorRef, tabBarRef} from './src/utils/Refs';
|
|
import {EditorWrapper} from './src/views/Editor/EditorWrapper';
|
|
import {getIntent, getNote, post} from './src/views/Editor/Functions';
|
|
|
|
let {width, height} = Dimensions.get('window');
|
|
let movedAway = true;
|
|
let layoutTimer = null;
|
|
let currentTab = 0;
|
|
const onChangeTab = async (obj) => {
|
|
if (obj.i === 1) {
|
|
eSendEvent(eCloseSideMenu);
|
|
if (getIntent()) return;
|
|
movedAway = false;
|
|
currentTab = 1;
|
|
if (!editing.currentlyEditing || !getNote()) {
|
|
eSendEvent(eOnLoadNote, {type: 'new'});
|
|
editing.currentlyEditing = true;
|
|
}
|
|
} else {
|
|
if (obj.from === 1) {
|
|
movedAway = true;
|
|
post('blur');
|
|
}
|
|
currentTab = 0;
|
|
eSendEvent(eOpenSideMenu);
|
|
}
|
|
};
|
|
|
|
export const RootView = () => {
|
|
const [state] = useTracked();
|
|
const {colors} = state;
|
|
|
|
return (
|
|
<>
|
|
<NavigationStack component={AppStack} />
|
|
<Toast />
|
|
<ContextMenu />
|
|
<DummyText />
|
|
<DialogManager colors={colors} />
|
|
</>
|
|
);
|
|
};
|
|
|
|
let updatedDimensions = {
|
|
width: width,
|
|
height: height,
|
|
};
|
|
|
|
let currentScroll = 0;
|
|
let startLocation = 0;
|
|
let startLocationX = 0;
|
|
const _responder = (e) => {
|
|
startLocation = e.nativeEvent.pageY;
|
|
startLocationX = e.nativeEvent.pageX;
|
|
_handleTouch();
|
|
return false;
|
|
};
|
|
const _moveResponder = (e) => {
|
|
_handleTouch();
|
|
return false;
|
|
};
|
|
|
|
const _handleTouch = () => {
|
|
{
|
|
if (currentTab === 1 && startLocation > updatedDimensions.height - 70) {
|
|
if (currentScroll === 0 || currentScroll === 1) {
|
|
tabBarRef.current?.setScrollEnabled(false);
|
|
}
|
|
} else if (currentTab === 0 && Platform.OS === 'ios') {
|
|
if (currentScroll === 0 && startLocationX < 150) {
|
|
tabBarRef.current?.setScrollEnabled(false);
|
|
} else {
|
|
tabBarRef.current?.setScrollEnabled(true);
|
|
}
|
|
} else {
|
|
console.log('blocking');
|
|
tabBarRef.current?.setScrollEnabled(true);
|
|
}
|
|
}
|
|
};
|
|
|
|
const _onTouchEnd = (e) => {
|
|
console.log('touch ended');
|
|
startLocation = 0;
|
|
tabBarRef.current?.setScrollEnabled(true);
|
|
};
|
|
|
|
const AppStack = React.memo(
|
|
() => {
|
|
const [state, dispatch] = useTracked();
|
|
const {colors} = state;
|
|
const [mode, setMode] = useState(
|
|
DDS.isLargeTablet()
|
|
? 'tablet'
|
|
: DDS.isSmallTab
|
|
? 'smallTablet'
|
|
: 'mobile',
|
|
);
|
|
const [dimensions, setDimensions] = useState({width, height});
|
|
|
|
const showFullScreenEditor = () => {
|
|
dispatch({type: Actions.FULLSCREEN, state: true});
|
|
editorRef.current?.setNativeProps({
|
|
style: {
|
|
position: 'absolute',
|
|
width: dimensions.width,
|
|
zIndex: 999,
|
|
paddingHorizontal: dimensions.width * 0.15,
|
|
backgroundColor: colors.bg,
|
|
},
|
|
});
|
|
};
|
|
|
|
const closeFullScreenEditor = () => {
|
|
dispatch({type: Actions.FULLSCREEN, state: false});
|
|
editorRef.current?.setNativeProps({
|
|
style: {
|
|
position: 'relative',
|
|
width: dimensions.width * 0.55,
|
|
zIndex: null,
|
|
paddingHorizontal: 0,
|
|
},
|
|
});
|
|
};
|
|
|
|
useEffect(() => {
|
|
eSubscribeEvent(eOpenFullscreenEditor, showFullScreenEditor);
|
|
eSubscribeEvent(eCloseFullscreenEditor, closeFullScreenEditor);
|
|
|
|
return () => {
|
|
eUnSubscribeEvent(eOpenFullscreenEditor, showFullScreenEditor);
|
|
eUnSubscribeEvent(eCloseFullscreenEditor, closeFullScreenEditor);
|
|
};
|
|
}, []);
|
|
|
|
const _onLayout = async (event) => {
|
|
if (layoutTimer) {
|
|
clearTimeout(layoutTimer);
|
|
layoutTimer = null;
|
|
}
|
|
|
|
let size = event?.nativeEvent?.layout;
|
|
updatedDimensions = size;
|
|
if (!size || (size.width === dimensions.width && mode !== null)) {
|
|
dispatch({type: Actions.DEVICE_MODE, state: mode});
|
|
return;
|
|
}
|
|
|
|
updatedDimensions = size;
|
|
layoutTimer = setTimeout(async () => {
|
|
checkDeviceType(size);
|
|
}, 500);
|
|
};
|
|
|
|
function checkDeviceType(size) {
|
|
setDimensions({
|
|
width: size.width,
|
|
height: size.height,
|
|
});
|
|
|
|
setWidthHeight(size);
|
|
DDS.setSize(size);
|
|
DDS.checkSmallTab(size.width > size.height ? 'LANDSCAPE' : 'PORTRAIT');
|
|
|
|
if (DDS.isLargeTablet()) {
|
|
setDeviceMode('tablet', size);
|
|
} else if (DDS.isSmallTab) {
|
|
setDeviceMode('smallTablet', size);
|
|
} else {
|
|
setDeviceMode('mobile', size);
|
|
}
|
|
}
|
|
|
|
function setDeviceMode(current, size) {
|
|
eSendEvent(current !== 'mobile' ? eCloseSideMenu : eOpenSideMenu);
|
|
setMode(current);
|
|
dispatch({type: Actions.DEVICE_MODE, state: current});
|
|
dispatch({type: Actions.FULLSCREEN, state: false});
|
|
editorRef.current?.setNativeProps({
|
|
style: {
|
|
position: 'relative',
|
|
width: current === 'tablet' ? size.width * 0.55 : size.width,
|
|
zIndex: null,
|
|
paddingHorizontal: 0,
|
|
},
|
|
});
|
|
if (!movedAway && current !== 'tablet') {
|
|
tabBarRef.current?.goToPage(1);
|
|
}
|
|
}
|
|
|
|
return (
|
|
<View
|
|
onLayout={_onLayout}
|
|
testID={notesnook.ids.default.root}
|
|
style={{
|
|
width: '100%',
|
|
height: '100%',
|
|
backgroundColor: colors.bg,
|
|
}}
|
|
onMoveShouldSetResponderCapture={_moveResponder}
|
|
onTouchEnd={_onTouchEnd}
|
|
onStartShouldSetResponderCapture={_responder}>
|
|
{mode && (
|
|
<ScrollableTabView
|
|
ref={tabBarRef}
|
|
style={{
|
|
zIndex: 1,
|
|
}}
|
|
onScroll={(event) => (currentScroll = event)}
|
|
initialPage={0}
|
|
prerenderingSiblingsNumber={Infinity}
|
|
onChangeTab={onChangeTab}
|
|
renderTabBar={() => <></>}>
|
|
{mode !== 'tablet' && (
|
|
<View
|
|
style={{
|
|
width: dimensions.width,
|
|
height: '100%',
|
|
borderRightColor: colors.nav,
|
|
borderRightWidth: 1,
|
|
flexDirection: 'row',
|
|
alignItems: 'center',
|
|
justifyContent: 'flex-start',
|
|
}}>
|
|
{mode === 'smallTablet' && (
|
|
<View
|
|
style={{
|
|
height: '100%',
|
|
width: dimensions.width * 0.35,
|
|
}}>
|
|
<Menu />
|
|
</View>
|
|
)}
|
|
|
|
<View
|
|
style={{
|
|
height: '100%',
|
|
width:
|
|
mode === 'mobile'
|
|
? dimensions.width
|
|
: dimensions.width * 0.65,
|
|
}}>
|
|
<NavigatorStack />
|
|
</View>
|
|
</View>
|
|
)}
|
|
|
|
<View
|
|
style={{
|
|
width: '100%',
|
|
height: '100%',
|
|
flexDirection: 'row',
|
|
backgroundColor: colors.bg,
|
|
}}>
|
|
{mode === 'tablet' && (
|
|
<View
|
|
style={{
|
|
width: dimensions.width * 0.45,
|
|
height: '100%',
|
|
borderRightColor: colors.nav,
|
|
borderRightWidth: 1,
|
|
flexDirection: 'row',
|
|
alignItems: 'center',
|
|
}}>
|
|
<View
|
|
style={{
|
|
height: '100%',
|
|
width: dimensions.width * 0.15,
|
|
}}>
|
|
<Menu />
|
|
</View>
|
|
|
|
<View
|
|
style={{
|
|
height: '100%',
|
|
width: dimensions.width * 0.3,
|
|
}}>
|
|
<NavigatorStack />
|
|
</View>
|
|
</View>
|
|
)}
|
|
<EditorWrapper dimensions={dimensions} />
|
|
</View>
|
|
</ScrollableTabView>
|
|
)}
|
|
</View>
|
|
);
|
|
},
|
|
() => true,
|
|
);
|