/* Copyright 2026 Adobe. All rights reserved. This file is licensed to you under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. */ import { createContext, useContext, ReactNode, useMemo } from 'react'; import { useColorScheme } from 'react-native'; import { Theme, Themes } from './Theme'; interface ThemeProviderProps { children: ReactNode; customThemes: Themes; } const defaultTheme: Themes = { light: { isDark: false, colors: { primary: '#007AFF', secondary: '#5856D6', background: '#FFFFFF', textPrimary: '#000000', textSecondary: '#8E8E93', imagePlaceholder: '#C7C7CC', buttonTextColor: 'dodgerblue', activeColor: '#0a7ea4', inactiveColor: '#687076', dotColor: '#FF4444', imageContainerColor: '#f0f0f0' } }, dark: { isDark: true, colors: { primary: '#0A84FF', secondary: '#5E5CE6', background: '#262626', textPrimary: '#FFFFFF', textSecondary: '#8E8E93', imagePlaceholder: '#48484A', buttonTextColor: 'dodgerblue', activeColor: '#fff', inactiveColor: '#9BA1A6', dotColor: '#FF6B6B', imageContainerColor: '#f0f0f0' } } }; const ThemeContext = createContext(undefined); /** * ThemeProvider component that provides the theme to the children components. * * @param children - The children components. * @param customThemes - The custom themes to override the default themes. * @returns The ThemeProvider component. */ export const ThemeProvider = ({ children, customThemes }: ThemeProviderProps) => { const systemColorScheme = useColorScheme(); // Memoize the merged themes to avoid recreation on every render const mergedThemes: Themes = useMemo(() => { return { light: { ...defaultTheme.light, colors: { ...defaultTheme.light.colors, ...(customThemes?.light?.colors || {}) } }, dark: { ...defaultTheme.dark, colors: { ...defaultTheme.dark.colors, ...(customThemes?.dark?.colors || {}) } } }; }, [customThemes]); // Memoize the active theme const activeTheme = useMemo(() => { const scheme = systemColorScheme ?? 'light'; return mergedThemes[scheme]; }, [mergedThemes, systemColorScheme]); // Memoize the context value to prevent unnecessary re-renders const contextValue: Theme = useMemo(() => activeTheme, [activeTheme]); return ( {children} ); }; /** * useTheme hook that returns the theme context. * @returns The theme context. */ export const useTheme = (): Theme => { const context = useContext(ThemeContext); const systemColorScheme = useColorScheme(); if (context === undefined) { return defaultTheme[systemColorScheme ?? 'light']; } return context; };