mirror of
https://github.com/DocNR/POWR.git
synced 2025-04-22 16:51:33 +00:00
256 lines
8.8 KiB
TypeScript
256 lines
8.8 KiB
TypeScript
// app/_layout.tsx
|
|
import 'expo-dev-client';
|
|
import '@/global.css';
|
|
import { DarkTheme, DefaultTheme, ThemeProvider } from '@react-navigation/native';
|
|
import { Stack } from 'expo-router';
|
|
import { StatusBar } from 'expo-status-bar';
|
|
import * as React from 'react';
|
|
import { View, Text, Platform, ActivityIndicator } from 'react-native';
|
|
import { NAV_THEME } from '@/lib/theme/constants';
|
|
import { useColorScheme } from '@/lib/theme/useColorScheme';
|
|
import { PortalHost } from '@rn-primitives/portal';
|
|
import { setAndroidNavigationBar } from '@/lib/android-navigation-bar';
|
|
import { GestureHandlerRootView } from 'react-native-gesture-handler';
|
|
import { DatabaseProvider } from '@/components/DatabaseProvider';
|
|
import { ErrorBoundary } from '@/components/ErrorBoundary';
|
|
import { SettingsDrawerProvider } from '@/lib/contexts/SettingsDrawerContext';
|
|
import SettingsDrawer from '@/components/SettingsDrawer';
|
|
import RelayInitializer from '@/components/RelayInitializer';
|
|
import OfflineIndicator from '@/components/OfflineIndicator';
|
|
import { useNDKStore } from '@/lib/stores/ndk';
|
|
import { useWorkoutStore } from '@/stores/workoutStore';
|
|
import { ConnectivityService } from '@/lib/db/services/ConnectivityService';
|
|
// Import splash screens with improved fallback mechanism
|
|
let SplashComponent: React.ComponentType<{onFinish: () => void}>;
|
|
let useVideoSplash = false;
|
|
|
|
// Determine if we should use video splash based on platform
|
|
if (Platform.OS === 'ios') {
|
|
// On iOS, try to use the video splash screen
|
|
try {
|
|
// Check if expo-av is available
|
|
require('expo-av');
|
|
useVideoSplash = true;
|
|
console.log('expo-av is available, will use VideoSplashScreen on iOS');
|
|
} catch (e) {
|
|
console.warn('expo-av not available on iOS:', e);
|
|
useVideoSplash = false;
|
|
}
|
|
} else {
|
|
// On Android, directly use SimpleSplashScreen to avoid issues
|
|
console.log('Android platform detected, using SimpleSplashScreen');
|
|
useVideoSplash = false;
|
|
}
|
|
|
|
// Import the appropriate splash screen component
|
|
if (useVideoSplash) {
|
|
try {
|
|
SplashComponent = require('@/components/VideoSplashScreen').default;
|
|
console.log('Successfully imported VideoSplashScreen');
|
|
} catch (e) {
|
|
console.warn('Failed to import VideoSplashScreen:', e);
|
|
useVideoSplash = false;
|
|
}
|
|
}
|
|
|
|
// If video splash is not available or failed to import, use simple splash
|
|
if (!useVideoSplash) {
|
|
try {
|
|
SplashComponent = require('@/components/SimpleSplashScreen').default;
|
|
console.log('Using SimpleSplashScreen');
|
|
} catch (simpleSplashError) {
|
|
console.warn('Failed to import SimpleSplashScreen:', simpleSplashError);
|
|
// Last resort fallback is an inline component
|
|
SplashComponent = ({onFinish}) => {
|
|
React.useEffect(() => {
|
|
// Call onFinish after a short delay
|
|
const timer = setTimeout(() => {
|
|
onFinish();
|
|
}, 1000);
|
|
return () => clearTimeout(timer);
|
|
}, [onFinish]);
|
|
|
|
return (
|
|
<View style={{
|
|
flex: 1,
|
|
backgroundColor: '#000000',
|
|
alignItems: 'center',
|
|
justifyContent: 'center',
|
|
}}>
|
|
<Text style={{
|
|
color: '#ffffff',
|
|
fontSize: 32,
|
|
fontWeight: 'bold',
|
|
}}>POWR</Text>
|
|
<ActivityIndicator
|
|
size="large"
|
|
color="#ffffff"
|
|
style={{ marginTop: 30 }}
|
|
/>
|
|
</View>
|
|
);
|
|
};
|
|
}
|
|
}
|
|
|
|
console.log('_layout.tsx loaded');
|
|
|
|
const LIGHT_THEME = {
|
|
...DefaultTheme,
|
|
colors: NAV_THEME.light,
|
|
};
|
|
|
|
const DARK_THEME = {
|
|
...DarkTheme,
|
|
colors: NAV_THEME.dark,
|
|
};
|
|
|
|
export default function RootLayout() {
|
|
const [isInitialized, setIsInitialized] = React.useState(false);
|
|
const [isSplashFinished, setIsSplashFinished] = React.useState(false);
|
|
const { colorScheme, isDarkColorScheme } = useColorScheme();
|
|
const { init } = useNDKStore();
|
|
const initializationPromise = React.useRef<Promise<void> | null>(null);
|
|
|
|
// Start app initialization immediately
|
|
React.useEffect(() => {
|
|
if (!initializationPromise.current) {
|
|
initializationPromise.current = (async () => {
|
|
try {
|
|
console.log('Starting app initialization in background...');
|
|
if (Platform.OS === 'web') {
|
|
document.documentElement.classList.add('bg-background');
|
|
}
|
|
setAndroidNavigationBar(colorScheme);
|
|
|
|
// Initialize connectivity service first
|
|
const connectivityService = ConnectivityService.getInstance();
|
|
const isOnline = await connectivityService.checkNetworkStatus();
|
|
console.log(`Network connectivity: ${isOnline ? 'online' : 'offline'}`);
|
|
|
|
// Start database initialization and NDK initialization in parallel
|
|
const initPromises = [];
|
|
|
|
// Initialize NDK with timeout
|
|
const ndkPromise = init().catch(error => {
|
|
console.error('NDK initialization error:', error);
|
|
// Continue even if NDK fails
|
|
return { offlineMode: true };
|
|
});
|
|
initPromises.push(ndkPromise);
|
|
|
|
// Load favorites from SQLite (local operation)
|
|
const favoritesPromise = useWorkoutStore.getState().loadFavorites()
|
|
.catch(error => {
|
|
console.error('Error loading favorites:', error);
|
|
// Continue even if loading favorites fails
|
|
});
|
|
initPromises.push(favoritesPromise);
|
|
|
|
// Wait for all initialization tasks with a timeout
|
|
const timeoutPromise = new Promise((_, reject) =>
|
|
setTimeout(() => reject(new Error('Initialization timeout')), 10000)
|
|
);
|
|
|
|
try {
|
|
// Use Promise.allSettled to continue even if some promises fail
|
|
await Promise.race([
|
|
Promise.allSettled(initPromises),
|
|
timeoutPromise
|
|
]);
|
|
} catch (error) {
|
|
console.warn('Some initialization tasks timed out, continuing anyway:', error);
|
|
}
|
|
|
|
console.log('App initialization completed!');
|
|
setIsInitialized(true);
|
|
} catch (error) {
|
|
console.error('Failed to initialize:', error);
|
|
// Still mark as initialized to prevent hanging
|
|
setIsInitialized(true);
|
|
}
|
|
})();
|
|
}
|
|
|
|
return () => {
|
|
// This is just for cleanup, the promise will continue executing
|
|
initializationPromise.current = null;
|
|
};
|
|
}, []);
|
|
|
|
// Function to handle splash finish - will check if initialization is also complete
|
|
const handleSplashFinish = React.useCallback(() => {
|
|
console.log('Splash video finished playing');
|
|
setIsSplashFinished(true);
|
|
|
|
// If initialization isn't done yet, we'll show a loading indicator
|
|
if (!isInitialized) {
|
|
console.log('Waiting for initialization to complete...');
|
|
}
|
|
}, [isInitialized]);
|
|
|
|
// Show splash screen if not finished
|
|
if (!isSplashFinished) {
|
|
try {
|
|
return <SplashComponent onFinish={handleSplashFinish} />;
|
|
} catch (e) {
|
|
console.error('Error rendering splash screen:', e);
|
|
// Skip splash screen if there's an error
|
|
if (!isInitialized) {
|
|
return (
|
|
<View className="flex-1 items-center justify-center bg-background">
|
|
<Text className="text-foreground">Loading...</Text>
|
|
</View>
|
|
);
|
|
}
|
|
// Force continue to main app
|
|
setIsSplashFinished(true);
|
|
return null;
|
|
}
|
|
}
|
|
|
|
// If splash is done but initialization isn't, show loading
|
|
if (!isInitialized) {
|
|
return (
|
|
<View className="flex-1 items-center justify-center bg-background">
|
|
<Text className="text-foreground">Finalizing setup...</Text>
|
|
</View>
|
|
);
|
|
}
|
|
|
|
// Main app UI wrapped in error boundary
|
|
return (
|
|
<ErrorBoundary>
|
|
<GestureHandlerRootView style={{ flex: 1 }}>
|
|
<DatabaseProvider>
|
|
<ThemeProvider value={isDarkColorScheme ? DARK_THEME : LIGHT_THEME}>
|
|
{/* Ensure SettingsDrawerProvider wraps everything */}
|
|
<SettingsDrawerProvider>
|
|
{/* Add RelayInitializer here - it loads relay data once NDK is available */}
|
|
<RelayInitializer />
|
|
|
|
{/* Add OfflineIndicator to show network status */}
|
|
<OfflineIndicator />
|
|
|
|
<StatusBar style={isDarkColorScheme ? 'light' : 'dark'} />
|
|
<Stack screenOptions={{ headerShown: false }}>
|
|
<Stack.Screen
|
|
name="(tabs)"
|
|
options={{
|
|
headerShown: false,
|
|
}}
|
|
/>
|
|
</Stack>
|
|
|
|
{/* Settings drawer needs to be outside the navigation stack */}
|
|
<SettingsDrawer />
|
|
|
|
<PortalHost />
|
|
</SettingsDrawerProvider>
|
|
</ThemeProvider>
|
|
</DatabaseProvider>
|
|
</GestureHandlerRootView>
|
|
</ErrorBoundary>
|
|
);
|
|
}
|