forked from Ivasoft/mattermost-mobile
* Timezone search results not updating on term edit MM-45621 * Fix update theme MM-45604 / MM-45591 * Prevent channel items from stacking when collapse/expand category MM-45590 * Remove current channel from unread section MM-45589 * Latex code block style MM-45572 * Update Post input style when theme update * Show tutorial only for the first item in the first section MM-45549 * Fix close user profile modal on iPad * Move style out of component Co-authored-by: Daniel Espino García <larkox@gmail.com>
104 lines
3.1 KiB
TypeScript
104 lines
3.1 KiB
TypeScript
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
// See LICENSE.txt for license information.
|
|
|
|
import withObservables from '@nozbe/with-observables';
|
|
import React, {ComponentType, createContext, useEffect, useState} from 'react';
|
|
import {Appearance} from 'react-native';
|
|
|
|
import {Preferences} from '@constants';
|
|
import {queryPreferencesByCategoryAndName} from '@queries/servers/preference';
|
|
import {observeCurrentTeamId} from '@queries/servers/system';
|
|
import {setThemeDefaults, updateThemeIfNeeded} from '@utils/theme';
|
|
|
|
import type {PreferenceModel} from '@database/models/server';
|
|
import type Database from '@nozbe/watermelondb/Database';
|
|
|
|
type Props = {
|
|
currentTeamId?: string;
|
|
children: React.ReactNode;
|
|
themes: PreferenceModel[];
|
|
}
|
|
|
|
type WithThemeProps = {
|
|
theme: Theme;
|
|
}
|
|
|
|
export function getDefaultThemeByAppearance(): Theme {
|
|
if (Appearance.getColorScheme() === 'dark') {
|
|
return Preferences.THEMES.onyx;
|
|
}
|
|
return Preferences.THEMES.denim;
|
|
}
|
|
|
|
export const ThemeContext = createContext(getDefaultThemeByAppearance());
|
|
const {Consumer, Provider} = ThemeContext;
|
|
|
|
const getTheme = (teamId: string | undefined, themes: PreferenceModel[]): Theme => {
|
|
if (teamId) {
|
|
const teamTheme = themes.find((t) => t.name === teamId) || themes[0];
|
|
if (teamTheme?.value) {
|
|
try {
|
|
const theme = setThemeDefaults(JSON.parse(teamTheme.value));
|
|
return theme;
|
|
} catch {
|
|
// no theme change
|
|
}
|
|
}
|
|
}
|
|
|
|
const defaultTheme = getDefaultThemeByAppearance();
|
|
|
|
return defaultTheme;
|
|
};
|
|
|
|
const ThemeProvider = ({currentTeamId, children, themes}: Props) => {
|
|
const [theme, setTheme] = useState(() => getTheme(currentTeamId, themes));
|
|
|
|
useEffect(() => {
|
|
const listener = Appearance.addChangeListener(() => {
|
|
const newTheme = getTheme(currentTeamId, themes);
|
|
if (theme !== newTheme) {
|
|
setTheme(newTheme);
|
|
}
|
|
});
|
|
|
|
return () => listener.remove();
|
|
}, [currentTeamId, themes, theme]);
|
|
|
|
useEffect(() => {
|
|
updateThemeIfNeeded(theme);
|
|
}, [theme]);
|
|
|
|
useEffect(() => {
|
|
setTheme(getTheme(currentTeamId, themes));
|
|
}, [currentTeamId, themes]);
|
|
|
|
return (<Provider value={theme}>{children}</Provider>);
|
|
};
|
|
|
|
export function withTheme<T extends WithThemeProps>(Component: ComponentType<T>): ComponentType<T> {
|
|
return function ThemeComponent(props) {
|
|
return (
|
|
<Consumer>
|
|
{(theme: Theme) => (
|
|
<Component
|
|
{...props}
|
|
theme={theme}
|
|
/>
|
|
)}
|
|
</Consumer>
|
|
);
|
|
};
|
|
}
|
|
|
|
export function useTheme(): Theme {
|
|
return React.useContext(ThemeContext);
|
|
}
|
|
|
|
const enhancedThemeProvider = withObservables([], ({database}: {database: Database}) => ({
|
|
currentTeamId: observeCurrentTeamId(database),
|
|
themes: queryPreferencesByCategoryAndName(database, Preferences.CATEGORY_THEME).observeWithColumns(['value']),
|
|
}));
|
|
|
|
export default enhancedThemeProvider(ThemeProvider);
|