Files
AFFiNE-Mirror/blocksuite/affine/shared/src/services/theme-service.ts
zzj3720 5a5779c05a feat(editor): simple table block (#9740)
close: BS-2122, BS-2125, BS-2124, BS-2420, PD-2073, BS-2126, BS-2469, BS-2470, BS-2478, BS-2471
2025-01-24 10:07:57 +00:00

202 lines
5.1 KiB
TypeScript

import {
type Color,
ColorScheme,
DefaultTheme,
resolveColor,
} from '@blocksuite/affine-model';
import { type BlockStdScope, StdIdentifier } from '@blocksuite/block-std';
import { type Container, createIdentifier } from '@blocksuite/global/di';
import { Extension, type ExtensionType } from '@blocksuite/store';
import { type Signal, signal } from '@preact/signals-core';
import {
type AffineCssVariables,
combinedDarkCssVariables,
combinedLightCssVariables,
} from '@toeverything/theme';
import { isInsideEdgelessEditor } from '../utils/dom';
export const ThemeExtensionIdentifier = createIdentifier<ThemeExtension>(
'AffineThemeExtension'
);
export interface ThemeExtension {
getAppTheme?: () => Signal<ColorScheme>;
getEdgelessTheme?: (docId?: string) => Signal<ColorScheme>;
}
export function OverrideThemeExtension(service: ThemeExtension): ExtensionType {
return {
setup: di => {
di.override(ThemeExtensionIdentifier, () => service);
},
};
}
export const ThemeProvider = createIdentifier<ThemeService>(
'AffineThemeProvider'
);
export class ThemeService extends Extension {
app$: Signal<ColorScheme>;
edgeless$: Signal<ColorScheme>;
get appTheme() {
return this.app$.peek();
}
get edgelessTheme() {
return this.edgeless$.peek();
}
get theme() {
return this.theme$.peek();
}
get theme$() {
return isInsideEdgelessEditor(this.std.host) ? this.edgeless$ : this.app$;
}
constructor(private readonly std: BlockStdScope) {
super();
const extension = this.std.getOptional(ThemeExtensionIdentifier);
this.app$ = extension?.getAppTheme?.() || getThemeObserver().theme$;
this.edgeless$ =
extension?.getEdgelessTheme?.(this.std.store.id) ||
getThemeObserver().theme$;
}
static override setup(di: Container) {
di.addImpl(ThemeProvider, ThemeService, [StdIdentifier]);
}
/**
* Generates a CSS's color property with `var` or `light-dark` functions.
*
* Sometimes used to set the frame/note background.
*
* @param color - A color value.
* @param fallback - If color value processing fails, it will be used as a fallback.
* @returns - A color property string.
*
* @example
*
* ```
* `rgba(255,0,0)`
* `#fff`
* `light-dark(#fff, #000)`
* `var(--affine-palette-shape-blue)`
* ```
*/
generateColorProperty(
color: Color,
fallback: Color = DefaultTheme.transparent,
theme = this.theme
) {
const result = resolveColor(color, theme, resolveColor(fallback, theme));
// Compatible old data
if (result.startsWith('--')) {
return result.endsWith('transparent')
? DefaultTheme.transparent
: `var(${result})`;
}
return result;
}
/**
* Gets a color with the current theme.
*
* @param color - A color value.
* @param fallback - If color value processing fails, it will be used as a fallback.
* @param real - If true, it returns the computed style.
* @returns - A color property string.
*
* @example
*
* ```
* `rgba(255,0,0)`
* `#fff`
* `--affine-palette-shape-blue`
* ```
*/
getColorValue(
color: Color,
fallback: Color = DefaultTheme.transparent,
real = false,
theme = this.theme
) {
let result = resolveColor(color, theme, resolveColor(fallback, theme));
// Compatible old data
if (real && result.startsWith('--')) {
result = result.endsWith('transparent')
? DefaultTheme.transparent
: this.getCssVariableColor(result, theme);
}
return result ?? DefaultTheme.transparent;
}
getCssVariableColor(property: string, theme = this.theme) {
// Compatible old data
if (property.startsWith('--')) {
if (property.endsWith('transparent')) {
return DefaultTheme.transparent;
}
const key = property as keyof AffineCssVariables;
// V1 theme
const color =
theme === ColorScheme.Dark
? combinedDarkCssVariables[key]
: combinedLightCssVariables[key];
return color;
}
return property;
}
}
export class ThemeObserver {
private readonly observer: MutationObserver;
theme$: Signal<ColorScheme>;
constructor() {
const COLOR_SCHEMES: string[] = Object.values(ColorScheme);
// Set initial theme according to the data-theme attribute
const initialMode = document.documentElement.dataset.theme;
this.theme$ = signal(
initialMode && COLOR_SCHEMES.includes(initialMode)
? (initialMode as ColorScheme)
: ColorScheme.Light
);
this.observer = new MutationObserver(() => {
const mode = document.documentElement.dataset.theme;
if (!mode) return;
if (!COLOR_SCHEMES.includes(mode)) return;
if (mode === this.theme$.value) return;
this.theme$.value = mode as ColorScheme;
});
this.observer.observe(document.documentElement, {
attributes: true,
attributeFilter: ['data-theme'],
});
}
destroy() {
this.observer.disconnect();
}
}
export const getThemeObserver = (function () {
let observer: ThemeObserver;
return function () {
if (observer) return observer;
observer = new ThemeObserver();
return observer;
};
})();