// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved. // See LICENSE.txt for license information. import React, {RefObject, useEffect, useRef, useState} from 'react'; import {AppState, Keyboard, NativeEventEmitter, NativeModules, Platform, View} from 'react-native'; import {useSafeAreaInsets} from 'react-native-safe-area-context'; import {Device} from '@constants'; import type {KeyboardTrackingViewRef} from 'react-native-keyboard-tracking-view'; const {SplitView} = NativeModules; const {isRunningInSplitView} = SplitView; const emitter = new NativeEventEmitter(SplitView); export function useSplitView() { const [isSplitView, setIsSplitView] = useState(false); useEffect(() => { if (Device.IS_TABLET) { isRunningInSplitView().then((result: SplitViewResult) => { if (result.isSplitView != null) { setIsSplitView(result.isSplitView); } }); } const listener = emitter.addListener('SplitViewChanged', (result: SplitViewResult) => { if (result.isSplitView != null) { setIsSplitView(result.isSplitView); } }); return () => listener.remove(); }, []); return isSplitView; } export function useAppState() { const [appState, setAppState] = useState(AppState.currentState); useEffect(() => { const listener = AppState.addEventListener('change', (nextState) => { setAppState(nextState); }); return () => listener.remove(); }, [appState]); return appState; } export function useIsTablet() { const isSplitView = useSplitView(); return Device.IS_TABLET && !isSplitView; } export function useKeyboardHeightWithDuration(keyboardTracker?: React.RefObject) { const [keyboardHeight, setKeyboardHeight] = useState({height: 0, duration: 0}); const updateTimeout = useRef(null); const insets = useSafeAreaInsets(); // This is a magic number. With tracking view, to properly get the final position, this had to be added. const KEYBOARD_TRACKINGVIEW_SEPARATION = 4; const updateValue = (height: number, duration: number) => { if (updateTimeout.current != null) { clearTimeout(updateTimeout.current); updateTimeout.current = null; } updateTimeout.current = setTimeout(() => { setKeyboardHeight({height, duration}); updateTimeout.current = null; }, 200); }; useEffect(() => { const show = Keyboard.addListener(Platform.select({ios: 'keyboardWillShow', default: 'keyboardDidShow'}), async (event) => { if (keyboardTracker?.current) { const props = await keyboardTracker.current.getNativeProps(); if (props.keyboardHeight) { updateValue((props.trackingViewHeight + props.keyboardHeight) - KEYBOARD_TRACKINGVIEW_SEPARATION, event.duration); } else { updateValue((props.trackingViewHeight + insets.bottom) - KEYBOARD_TRACKINGVIEW_SEPARATION, event.duration); } } else { setKeyboardHeight({height: event.endCoordinates.height, duration: event.duration}); } }); const hide = Keyboard.addListener(Platform.select({ios: 'keyboardWillHide', default: 'keyboardDidHide'}), (event) => { if (updateTimeout.current != null) { clearTimeout(updateTimeout.current); updateTimeout.current = null; } setKeyboardHeight({height: 0, duration: event.duration}); }); return () => { show.remove(); hide.remove(); }; }, [keyboardTracker && insets.bottom]); return keyboardHeight; } export function useKeyboardHeight(keyboardTracker?: React.RefObject) { const {height} = useKeyboardHeightWithDuration(keyboardTracker); return height; } export function useModalPosition(viewRef: RefObject, deps?: React.DependencyList) { const [modalPosition, setModalPosition] = useState(0); const isTablet = useIsTablet(); const height = useKeyboardHeight(); useEffect(() => { if (Platform.OS === 'ios' && isTablet) { viewRef.current?.measureInWindow((_, y) => { if (y !== modalPosition) { setModalPosition(y); } }); } }, [...(deps || []), isTablet, height]); return modalPosition; }