import React from 'react'; import { StackNavigationOptions } from '@react-navigation/stack'; import I18n from '../i18n'; import { withTheme } from '../theme'; import { themes } from '../constants/colors'; import StatusBar from '../containers/StatusBar'; import * as List from '../containers/List'; import { supportSystemTheme } from '../utils/deviceInfo'; import SafeAreaView from '../containers/SafeAreaView'; import UserPreferences from '../lib/methods/userPreferences'; import { events, logEvent } from '../utils/log'; import { IThemePreference, TThemeMode, TDarkLevel } from '../definitions/ITheme'; import { THEME_PREFERENCES_KEY } from '../lib/constants'; const THEME_GROUP = 'THEME_GROUP'; const DARK_GROUP = 'DARK_GROUP'; const SYSTEM_THEME: ITheme = { label: 'Automatic', value: 'automatic', group: THEME_GROUP }; const THEMES: ITheme[] = [ { label: 'Light', value: 'light', group: THEME_GROUP }, { label: 'Dark', value: 'dark', group: THEME_GROUP }, { label: 'Black', value: 'black', group: DARK_GROUP }, { label: 'Dark', value: 'dark', group: DARK_GROUP } ]; if (supportSystemTheme()) { THEMES.unshift(SYSTEM_THEME); } const themeGroup = THEMES.filter(item => item.group === THEME_GROUP); const darkGroup = THEMES.filter(item => item.group === DARK_GROUP); interface ITheme { label: string; value: TThemeMode | TDarkLevel; group: string; } interface IThemeViewProps { theme: string; themePreferences: IThemePreference; setTheme(newTheme?: IThemePreference): void; } class ThemeView extends React.Component { static navigationOptions = (): StackNavigationOptions => ({ title: I18n.t('Theme') }); isSelected = (item: ITheme) => { const { themePreferences } = this.props; const { group } = item; const { darkLevel, currentTheme } = themePreferences; if (group === THEME_GROUP) { return item.value === currentTheme; } if (group === DARK_GROUP) { return item.value === darkLevel; } }; onClick = (item: ITheme) => { const { themePreferences } = this.props; const { darkLevel, currentTheme } = themePreferences; const { value, group } = item; let changes: Partial = {}; if (group === THEME_GROUP && currentTheme !== value) { logEvent(events.THEME_SET_THEME_GROUP, { theme_group: value }); changes = { currentTheme: value as TThemeMode }; } if (group === DARK_GROUP && darkLevel !== value) { logEvent(events.THEME_SET_DARK_LEVEL, { dark_level: value }); changes = { darkLevel: value as TDarkLevel }; } this.setTheme(changes); }; setTheme = (theme: Partial) => { const { setTheme, themePreferences } = this.props; const newTheme = { ...themePreferences, ...theme }; setTheme(newTheme); UserPreferences.setMap(THEME_PREFERENCES_KEY, newTheme); }; renderIcon = () => { const { theme } = this.props; return ; }; renderItem = ({ item }: { item: ITheme }) => { const { label, value } = item; return ( <> this.onClick(item)} testID={`theme-view-${value}`} right={() => (this.isSelected(item) ? this.renderIcon() : null)} /> ); }; render() { return ( <>{themeGroup.map(item => this.renderItem({ item }))} <>{darkGroup.map(item => this.renderItem({ item }))} ); } } export default withTheme(ThemeView);