feat(ConfigProvider): add theme-vars-scope props enable root affects (#12240)

* feat(ConfigProvider): add theme-vars-in-root props enable root affects

* feat(ConfigProvider): add theme-vars-in-root props enable root affects

* feat(ConfigProvider): add theme-vars-in-root props enable root affects
This commit is contained in:
ShuGang Zhou 2023-09-04 07:56:21 +08:00 committed by GitHub
parent ee2788796b
commit 5bcd41cfef
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 137 additions and 3 deletions

View File

@ -26,6 +26,8 @@ const [name, bem] = createNamespace('config-provider');
export type ConfigProviderTheme = 'light' | 'dark';
export type ConfigProviderThemeVarsScope = 'local' | 'global';
export type ConfigProviderProvide = {
iconPrefix?: string;
};
@ -42,6 +44,7 @@ export const configProviderProps = {
themeVars: Object as ThemeVars,
themeVarsDark: Object as ThemeVars,
themeVarsLight: Object as ThemeVars,
themeVarsScope: makeStringProp<ConfigProviderThemeVarsScope>('local'),
iconPrefix: String,
};
@ -55,6 +58,22 @@ function mapThemeVarsToCSSVars(themeVars: Record<string, Numeric>) {
return cssVars;
}
function syncThemeVarsOnRoot(
newStyle: Record<string, Numeric> = {},
oldStyle: Record<string, Numeric> = {},
) {
Object.keys(newStyle).forEach((key) => {
if (newStyle[key] !== oldStyle[key]) {
document.documentElement.style.setProperty(key, newStyle[key] as string);
}
});
Object.keys(oldStyle).forEach((key) => {
if (!newStyle[key]) {
document.documentElement.style.removeProperty(key);
}
});
}
export default defineComponent({
name,
@ -93,6 +112,38 @@ export default defineComponent({
onActivated(addTheme);
onDeactivated(removeTheme);
onBeforeUnmount(removeTheme);
watch(
style,
(newStyle, oldStyle) => {
if (props.themeVarsScope === 'global') {
syncThemeVarsOnRoot(
newStyle as Record<string, Numeric>,
oldStyle as Record<string, Numeric>,
);
}
},
{
immediate: true,
},
);
watch(
() => props.themeVarsScope,
(newScope, oldStyle) => {
if (oldStyle === 'global') {
// remove on Root
syncThemeVarsOnRoot({}, style.value as Record<string, Numeric>);
}
if (newScope === 'global') {
// add on root
syncThemeVarsOnRoot(style.value as Record<string, Numeric>, {});
}
},
{
immediate: true,
},
);
}
provide(CONFIG_PROVIDER_KEY, props);
@ -104,7 +155,10 @@ export default defineComponent({
});
return () => (
<props.tag class={bem()} style={style.value}>
<props.tag
class={bem()}
style={props.themeVarsScope === 'local' ? style.value : undefined}
>
{slots.default?.()}
</props.tag>
);

View File

@ -161,7 +161,7 @@ export default {
};
```
> Tips: ConfigProvider only affects its child components.
> Tips: ConfigProvider by default only affects its child components. set to `global` for the entire page to take effect.
#### Use In TypeScript
@ -316,6 +316,7 @@ There are all **Basic Variables** below, for component CSS Variables, please ref
| theme-vars | Theme variables | _object_ | - |
| theme-vars-dark | Theme variables that work in dark modewill override `theme-vars` | _object_ | - |
| theme-vars-light | Theme variables that work in light mode, will override `theme-vars` | _object_ | - |
| theme-vars-scope | by default only affects its child componentsset to `global` for the entire page to take effect | _ConfigProviderThemeVarsScope_ | `local` |
| z-index | Set the z-index of all popup components, this property takes effect globally | _number_ | `2000` |
| tag | HTML Tag of root element | _string_ | `div` |
| icon-prefix | Icon className prefix | _string_ | `van-icon` |
@ -329,5 +330,6 @@ import type {
ConfigProviderProps,
ConfigProviderTheme,
ConfigProviderThemeVars,
ConfigProviderThemeVarsScope,
} from 'vant';
```

View File

@ -159,7 +159,7 @@ export default {
};
```
> 注意ConfigProvider 仅影响它的子组件的样式,不影响全局 root 节点。
> 注意ConfigProvider 默认仅影响它的子组件的样式,不影响全局 root 节点。设置为 `global` 整个页面生效。
#### 在 TypeScript 中使用
@ -318,6 +318,7 @@ Vant 中的 CSS 变量分为 **基础变量** 和 **组件变量**。组件变
| theme-vars | 自定义主题变量,局部生效 | _object_ | - |
| theme-vars-dark | 仅在深色模式下生效的主题变量,优先级高于 `theme-vars` | _object_ | - |
| theme-vars-light | 仅在浅色模式下生效的主题变量,优先级高于 `theme-vars` | _object_ | - |
| theme-vars-scope | 默认仅影响子组件的样式,设置为 `global` 整个页面生效 | _ConfigProviderThemeVarsScope_ | `local` |
| tag | 根节点对应的 HTML 标签名 | _string_ | `div` |
| z-index | 设置所有弹窗类组件的 z-index该属性对全局生效 | _number_ | `2000` |
| icon-prefix | 所有图标的类名前缀,等同于 Icon 组件的 [class-prefix 属性](#/zh-CN/icon#props) | _string_ | `van-icon` |
@ -331,5 +332,6 @@ import type {
ConfigProviderProps,
ConfigProviderTheme,
ConfigProviderThemeVars,
ConfigProviderThemeVarsScope,
} from 'vant';
```

View File

@ -7,6 +7,7 @@ export { configProviderProps } from './ConfigProvider';
export type {
ConfigProviderProps,
ConfigProviderTheme,
ConfigProviderThemeVarsScope,
} from './ConfigProvider';
export type { ConfigProviderThemeVars } from './types';

View File

@ -78,3 +78,78 @@ test('should apply theme-vars-dark in dark mode', () => {
'--van-rate-icon-full-color: green;',
);
});
test('should apply theme-vars-scope enable root affects', async () => {
const wrapper = mount({
render() {
return (
<ConfigProvider
theme="dark"
themeVars={{ rateIconFullColor: 'red' }}
themeVarsDark={{ rateIconFullColor: 'green' }}
themeVarsLight={{ rateIconFullColor: 'blue' }}
themeVarsScope="global"
/>
);
},
});
expect(document.documentElement.getAttribute('style')).toEqual(
'--van-rate-icon-full-color: green;',
);
expect(
wrapper.element.getAttribute('style') ===
'--van-rate-icon-full-color: green;',
).toBeFalsy();
await wrapper.setProps({
themeVarsScope: 'local',
});
expect(wrapper.element.getAttribute('style')).toEqual(
'--van-rate-icon-full-color: green;',
);
expect(
document.documentElement.getAttribute('style') ===
'--van-rate-icon-full-color: green;',
).toBeFalsy();
});
test('should apply theme-vars-scope enable root affects and sync theme vars', async () => {
const wrapper = mount({
render() {
return (
<ConfigProvider
theme="dark"
themeVars={{ rateIconFullColor: 'red' }}
themeVarsScope="global"
/>
);
},
});
expect(document.documentElement.getAttribute('style')).toEqual(
'--van-rate-icon-full-color: red;',
);
await wrapper.setProps({
themeVars: {
rateIconFullColor: 'red',
buttonPrimaryColor: 'red',
},
});
expect(document.documentElement.getAttribute('style')).toEqual(
'--van-rate-icon-full-color: red; --van-button-primary-color: red;',
);
await wrapper.setProps({
themeVars: {
buttonPrimaryColor: 'red',
},
});
expect(document.documentElement.getAttribute('style')).toEqual(
'--van-button-primary-color: red;',
);
});