Files
mattermost-mobile/app/context/theme/index.tsx
Elias Nahum 4a842f0129 Gekidou bugs fixes (#6473)
* 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>
2022-07-15 12:40:59 +02:00

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);