mirror of
https://gitee.com/vant-contrib/vant.git
synced 2025-04-06 03:57:59 +08:00
feat(@vant/cli): mobile site support dark mode
This commit is contained in:
parent
966af9dd68
commit
df24a3b683
@ -1,7 +1,4 @@
|
|||||||
/**
|
import { ref } from 'vue';
|
||||||
* 同步父窗口和 iframe 的 vue-router 状态
|
|
||||||
*/
|
|
||||||
|
|
||||||
import { config } from 'site-desktop-shared';
|
import { config } from 'site-desktop-shared';
|
||||||
|
|
||||||
let queue = [];
|
let queue = [];
|
||||||
@ -62,6 +59,40 @@ export function syncPathToChild() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function syncThemeToChild(theme) {
|
||||||
|
const iframe = document.querySelector('iframe');
|
||||||
|
if (iframe) {
|
||||||
|
iframeReady(() => {
|
||||||
|
iframe.contentWindow.postMessage(
|
||||||
|
{
|
||||||
|
type: 'updateTheme',
|
||||||
|
value: theme,
|
||||||
|
},
|
||||||
|
'*'
|
||||||
|
);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getDefaultTheme() {
|
||||||
|
return window.localStorage.getItem('vantTheme') || 'light';
|
||||||
|
}
|
||||||
|
|
||||||
|
export function useCurrentTheme() {
|
||||||
|
const theme = ref(getDefaultTheme());
|
||||||
|
|
||||||
|
window.addEventListener('message', (event) => {
|
||||||
|
if (event.data?.type !== 'updateTheme') {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const newTheme = event.data?.value || '';
|
||||||
|
theme.value = newTheme;
|
||||||
|
});
|
||||||
|
|
||||||
|
return theme;
|
||||||
|
}
|
||||||
|
|
||||||
export function listenToSyncPath(router) {
|
export function listenToSyncPath(router) {
|
||||||
window.addEventListener('message', (event) => {
|
window.addEventListener('message', (event) => {
|
||||||
if (event.data?.type !== 'replacePath') {
|
if (event.data?.type !== 'replacePath') {
|
@ -1,4 +1,4 @@
|
|||||||
:root {
|
body {
|
||||||
// colors
|
// colors
|
||||||
--van-doc-black: #1a1a1a;
|
--van-doc-black: #1a1a1a;
|
||||||
--van-doc-white: #fff;
|
--van-doc-white: #fff;
|
||||||
@ -55,9 +55,9 @@
|
|||||||
--van-doc-link-color: #1bb5fe;
|
--van-doc-link-color: #1bb5fe;
|
||||||
|
|
||||||
// background
|
// background
|
||||||
--van-doc-background: var(--van-doc-black);
|
--van-doc-background: #202124;
|
||||||
--van-doc-background-light: rgba(255, 255, 255, 0.06);
|
--van-doc-background-light: rgba(255, 255, 255, 0.06);
|
||||||
--van-doc-header-background: #011428;
|
--van-doc-header-background: rgba(1, 31, 60, 0.3);
|
||||||
--van-doc-border-color: #3a3a3c;
|
--van-doc-border-color: #3a3a3c;
|
||||||
--van-doc-shadow-color: transparent;
|
--van-doc-shadow-color: transparent;
|
||||||
|
|
||||||
|
@ -8,7 +8,7 @@
|
|||||||
:simulator="simulator"
|
:simulator="simulator"
|
||||||
:has-simulator="hasSimulator"
|
:has-simulator="hasSimulator"
|
||||||
:lang-configs="langConfigs"
|
:lang-configs="langConfigs"
|
||||||
:support-dark-mode="supportDarkMode"
|
:dark-mode-class="darkModeClass"
|
||||||
>
|
>
|
||||||
<router-view />
|
<router-view />
|
||||||
</van-doc>
|
</van-doc>
|
||||||
@ -28,7 +28,7 @@ export default {
|
|||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
hasSimulator: true,
|
hasSimulator: true,
|
||||||
supportDarkMode: config.site.supportDarkMode,
|
darkModeClass: config.site.darkModeClass,
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
|
|
||||||
@ -72,18 +72,18 @@ export default {
|
|||||||
watch: {
|
watch: {
|
||||||
// eslint-disable-next-line
|
// eslint-disable-next-line
|
||||||
'$route.path'() {
|
'$route.path'() {
|
||||||
this.setTitleAndToogleSimulator();
|
this.setTitleAndToggleSimulator();
|
||||||
},
|
},
|
||||||
|
|
||||||
lang(val) {
|
lang(val) {
|
||||||
setLang(val);
|
setLang(val);
|
||||||
this.setTitleAndToogleSimulator();
|
this.setTitleAndToggleSimulator();
|
||||||
},
|
},
|
||||||
|
|
||||||
config: {
|
config: {
|
||||||
handler(val) {
|
handler(val) {
|
||||||
if (val) {
|
if (val) {
|
||||||
this.setTitleAndToogleSimulator();
|
this.setTitleAndToggleSimulator();
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
immediate: true,
|
immediate: true,
|
||||||
@ -102,7 +102,7 @@ export default {
|
|||||||
},
|
},
|
||||||
|
|
||||||
methods: {
|
methods: {
|
||||||
setTitleAndToogleSimulator() {
|
setTitleAndToggleSimulator() {
|
||||||
let { title } = this.config;
|
let { title } = this.config;
|
||||||
|
|
||||||
const navItems = this.config.nav.reduce(
|
const navItems = this.config.nav.reduce(
|
||||||
|
@ -21,7 +21,7 @@
|
|||||||
</a>
|
</a>
|
||||||
</li>
|
</li>
|
||||||
|
|
||||||
<li v-if="supportDarkMode" class="van-doc-header__top-nav-item">
|
<li v-if="darkModeClass" class="van-doc-header__top-nav-item">
|
||||||
<a
|
<a
|
||||||
class="van-doc-header__link"
|
class="van-doc-header__link"
|
||||||
target="_blank"
|
target="_blank"
|
||||||
@ -76,6 +76,7 @@
|
|||||||
<script>
|
<script>
|
||||||
import SearchInput from './SearchInput.vue';
|
import SearchInput from './SearchInput.vue';
|
||||||
import { packageVersion } from 'site-desktop-shared';
|
import { packageVersion } from 'site-desktop-shared';
|
||||||
|
import { getDefaultTheme, syncThemeToChild } from '../../common/iframe-sync';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
name: 'VanDocHeader',
|
name: 'VanDocHeader',
|
||||||
@ -89,12 +90,12 @@ export default {
|
|||||||
config: Object,
|
config: Object,
|
||||||
versions: Array,
|
versions: Array,
|
||||||
langConfigs: Array,
|
langConfigs: Array,
|
||||||
supportDarkMode: Boolean,
|
darkModeClass: String,
|
||||||
},
|
},
|
||||||
|
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
currentTheme: this.getDefaultTheme(),
|
currentTheme: getDefaultTheme(),
|
||||||
packageVersion,
|
packageVersion,
|
||||||
showVersionPop: false,
|
showVersionPop: false,
|
||||||
};
|
};
|
||||||
@ -136,16 +137,13 @@ export default {
|
|||||||
window.localStorage.setItem('vantTheme', newVal);
|
window.localStorage.setItem('vantTheme', newVal);
|
||||||
document.body.classList.remove(`van-doc-theme-${oldVal}`);
|
document.body.classList.remove(`van-doc-theme-${oldVal}`);
|
||||||
document.body.classList.add(`van-doc-theme-${newVal}`);
|
document.body.classList.add(`van-doc-theme-${newVal}`);
|
||||||
|
syncThemeToChild(newVal);
|
||||||
},
|
},
|
||||||
immediate: true,
|
immediate: true,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
methods: {
|
methods: {
|
||||||
getDefaultTheme() {
|
|
||||||
return window.localStorage.getItem('vantTheme') || 'light';
|
|
||||||
},
|
|
||||||
|
|
||||||
toggleTheme() {
|
toggleTheme() {
|
||||||
this.currentTheme = this.currentTheme === 'light' ? 'dark' : 'light';
|
this.currentTheme = this.currentTheme === 'light' ? 'dark' : 'light';
|
||||||
},
|
},
|
||||||
|
@ -5,7 +5,7 @@
|
|||||||
:config="config"
|
:config="config"
|
||||||
:versions="versions"
|
:versions="versions"
|
||||||
:lang-configs="langConfigs"
|
:lang-configs="langConfigs"
|
||||||
:support-dark-mode="supportDarkMode"
|
:dark-mode-class="darkModeClass"
|
||||||
@switch-version="$emit('switch-version', $event)"
|
@switch-version="$emit('switch-version', $event)"
|
||||||
/>
|
/>
|
||||||
<doc-nav :lang="lang" :nav-config="config.nav" />
|
<doc-nav :lang="lang" :nav-config="config.nav" />
|
||||||
@ -42,7 +42,7 @@ export default {
|
|||||||
simulator: String,
|
simulator: String,
|
||||||
langConfigs: Array,
|
langConfigs: Array,
|
||||||
hasSimulator: Boolean,
|
hasSimulator: Boolean,
|
||||||
supportDarkMode: Boolean,
|
darkModeClass: String,
|
||||||
config: {
|
config: {
|
||||||
type: Object,
|
type: Object,
|
||||||
required: true,
|
required: true,
|
||||||
|
@ -3,7 +3,7 @@ import { createRouter, createWebHashHistory } from 'vue-router';
|
|||||||
import { isMobile, decamelize } from '../common';
|
import { isMobile, decamelize } from '../common';
|
||||||
import { config, documents } from 'site-desktop-shared';
|
import { config, documents } from 'site-desktop-shared';
|
||||||
import { getLang, setDefaultLang } from '../common/locales';
|
import { getLang, setDefaultLang } from '../common/locales';
|
||||||
import { listenToSyncPath, syncPathToChild } from '../common/iframe-router';
|
import { listenToSyncPath, syncPathToChild } from '../common/iframe-sync';
|
||||||
|
|
||||||
if (isMobile) {
|
if (isMobile) {
|
||||||
location.replace('mobile.html' + location.hash);
|
location.replace('mobile.html' + location.hash);
|
||||||
|
@ -10,10 +10,31 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
|
import { watch } from 'vue';
|
||||||
import DemoNav from './components/DemoNav.vue';
|
import DemoNav from './components/DemoNav.vue';
|
||||||
|
import { useCurrentTheme } from '../common/iframe-sync';
|
||||||
|
import { config } from 'site-mobile-shared';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
components: { DemoNav },
|
components: { DemoNav },
|
||||||
|
|
||||||
|
setup() {
|
||||||
|
const theme = useCurrentTheme();
|
||||||
|
|
||||||
|
watch(
|
||||||
|
theme,
|
||||||
|
(newVal, oldVal) => {
|
||||||
|
document.body.classList.remove(`van-doc-theme-${oldVal}`);
|
||||||
|
document.body.classList.add(`van-doc-theme-${newVal}`);
|
||||||
|
|
||||||
|
const { darkModeClass } = config.site;
|
||||||
|
if (darkModeClass) {
|
||||||
|
document.body.classList.toggle(darkModeClass, newVal === 'dark');
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{ immediate: true }
|
||||||
|
);
|
||||||
|
},
|
||||||
};
|
};
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
@ -24,6 +45,10 @@ body {
|
|||||||
min-width: 100vw;
|
min-width: 100vw;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.van-doc-theme-dark {
|
||||||
|
background-color: var(--van-doc-background);
|
||||||
|
}
|
||||||
|
|
||||||
::-webkit-scrollbar {
|
::-webkit-scrollbar {
|
||||||
width: 0;
|
width: 0;
|
||||||
background: transparent;
|
background: transparent;
|
||||||
|
@ -42,7 +42,7 @@ export default {
|
|||||||
align-items: center;
|
align-items: center;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
height: 56px;
|
height: 56px;
|
||||||
background-color: #fff;
|
background-color: var(--van-doc-background-light);
|
||||||
|
|
||||||
&__title {
|
&__title {
|
||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
|
@ -4,7 +4,7 @@ import DemoHome from './components/DemoHome.vue';
|
|||||||
import { decamelize } from '../common';
|
import { decamelize } from '../common';
|
||||||
import { demos, config } from 'site-mobile-shared';
|
import { demos, config } from 'site-mobile-shared';
|
||||||
import { getLang, setDefaultLang } from '../common/locales';
|
import { getLang, setDefaultLang } from '../common/locales';
|
||||||
import { listenToSyncPath, syncPathToParent } from '../common/iframe-router';
|
import { listenToSyncPath, syncPathToParent } from '../common/iframe-sync';
|
||||||
|
|
||||||
const { locales, defaultLang } = config.site;
|
const { locales, defaultLang } = config.site;
|
||||||
|
|
||||||
|
@ -16,7 +16,7 @@ export default {
|
|||||||
},
|
},
|
||||||
site: {
|
site: {
|
||||||
defaultLang: 'en-US',
|
defaultLang: 'en-US',
|
||||||
supportDarkMode: true,
|
darkModeClass: 'van-theme-dark',
|
||||||
versions: [
|
versions: [
|
||||||
{ label: 'v1', link: '/vant/v1/' },
|
{ label: 'v1', link: '/vant/v1/' },
|
||||||
{ label: 'v2', link: '/vant/' },
|
{ label: 'v2', link: '/vant/' },
|
||||||
|
Loading…
x
Reference in New Issue
Block a user