feat(cli): support locales

This commit is contained in:
陈嘉涵 2019-12-06 15:51:20 +08:00
parent 6bd858fac5
commit dc6cc6c5af
16 changed files with 336 additions and 135 deletions

View File

@ -0,0 +1,30 @@
const ZH_CN = 'zh-CN';
const EN_US = 'en-US';
const CACHE_KEY = 'vant-cli-lang';
let currentLang = ZH_CN;
export function getLang() {
return currentLang;
}
export function setLang(lang) {
currentLang = lang;
localStorage.setItem(CACHE_KEY, lang);
}
export function setDefaultLang(langFromConfig) {
const cached = localStorage.getItem(CACHE_KEY);
if (cached) {
currentLang = cached;
return;
}
if (navigator.language && navigator.language.indexOf('zh-') !== -1) {
currentLang = ZH_CN;
return;
}
currentLang = langFromConfig || EN_US;
}

View File

@ -10,21 +10,38 @@
import VanDoc from './components'; import VanDoc from './components';
import { config } from 'site-desktop-shared'; import { config } from 'site-desktop-shared';
function getPublicPath() {
const { site } = config.build || {};
if (process.env.NODE_ENV === 'production') {
return (site && site.publicPath) || '/';
}
return '/';
}
export default { export default {
components: { components: {
VanDoc VanDoc
}, },
data() { data() {
const { site } = config.build || {};
const isProd = process.env.NODE_ENV === 'production';
const prodPublicPath = (site && site.publicPath) || '/';
const publicPath = isProd ? prodPublicPath : '/';
return { return {
config: config.site, simulator: `${getPublicPath()}mobile.html${location.hash}`
simulator: `${publicPath}mobile.html${location.hash}`
}; };
},
computed: {
config() {
const { locales } = config.site;
if (locales) {
const { lang } = this.$route.meta;
return locales[lang];
}
return config.site;
}
} }
}; };
</script> </script>
@ -32,7 +49,7 @@ export default {
<style lang="less"> <style lang="less">
.van-doc-intro { .van-doc-intro {
padding-top: 20px; padding-top: 20px;
font-family: "Dosis", "Source Sans Pro", "Helvetica Neue", Arial, sans-serif; font-family: 'Dosis', 'Source Sans Pro', 'Helvetica Neue', Arial, sans-serif;
text-align: center; text-align: center;
p { p {

View File

@ -58,6 +58,7 @@ export default {
}, },
data() { data() {
console.log(this.config);
return { return {
showVersionPop: false showVersionPop: false
}; };

View File

@ -1,38 +1,11 @@
import Vue from 'vue'; import Vue from 'vue';
import VueRouter from 'vue-router';
import App from './App'; import App from './App';
import { routes } from './router'; import { router } from './router';
import { isMobile } from '../common';
import '../common/iframe-router';
if (process.env.NODE_ENV !== 'production') { if (process.env.NODE_ENV !== 'production') {
Vue.config.productionTip = false; Vue.config.productionTip = false;
} }
Vue.use(VueRouter);
if (isMobile) {
location.replace('mobile.html' + location.hash);
}
const router = new VueRouter({
mode: 'hash',
routes,
scrollBehavior(to) {
if (to.hash) {
return { selector: to.hash };
}
return { x: 0, y: 0 };
}
});
router.afterEach(() => {
Vue.nextTick(() => window.syncPath());
});
window.vueRouter = router;
new Vue({ new Vue({
el: '#app', el: '#app',
mounted() { mounted() {

View File

@ -1,28 +1,98 @@
import Vue from 'vue';
import VueRouter from 'vue-router';
import decamelize from 'decamelize'; import decamelize from 'decamelize';
import { documents } from 'site-desktop-shared'; import { isMobile } from '../common';
import { config, documents } from 'site-desktop-shared';
import { getLang, setDefaultLang } from '../common/locales';
import '../common/iframe-router';
const routes = []; if (isMobile) {
const names = Object.keys(documents); location.replace('mobile.html' + location.hash);
}
routes.push({ const { locales, defaultLang } = config.site;
path: '/home',
component: documents.Home
});
routes.push({ setDefaultLang(defaultLang);
path: '*',
redirect: '/home'
});
names.forEach(name => { function parseName(name) {
routes.push({ if (name.indexOf('_') !== -1) {
name, const pairs = name.split('_');
component: documents[name], const component = pairs.shift();
path: `/${decamelize(name, '-')}`,
meta: { return {
name component: `${decamelize(component, '-')}`,
lang: pairs.join('-')
};
}
return {
component: `${decamelize(name, '-')}`,
lang: ''
};
}
function getRoutes() {
const routes = [];
const names = Object.keys(documents);
if (locales) {
routes.push({
path: '*',
redirect: `/${getLang()}/`
});
} else {
routes.push({
path: '*',
redirect: '/'
});
}
function addHomeRoute(Home, lang) {
routes.push({
name: lang,
path: `/${lang || ''}`,
component: Home,
meta: { lang }
});
}
names.forEach(name => {
const { component, lang } = parseName(name);
if (component === 'home') {
addHomeRoute(documents[name], lang);
} }
routes.push({
name: `${lang}/${component}`,
path: `/${lang}/${component}`,
component: documents[name],
meta: {
lang,
name: component
}
});
}); });
return routes;
}
Vue.use(VueRouter);
export const router = new VueRouter({
mode: 'hash',
routes: getRoutes(),
scrollBehavior(to) {
if (to.hash) {
return { selector: to.hash };
}
return { x: 0, y: 0 };
}
}); });
export { routes }; router.afterEach(() => {
Vue.nextTick(() => window.syncPath());
});
window.vueRouter = router;

View File

@ -8,7 +8,7 @@
<template v-for="(group, index) in config.nav"> <template v-for="(group, index) in config.nav">
<demo-home-nav <demo-home-nav
:group="group" :group="group"
:base="$vantLang" :lang="lang"
:key="index" :key="index"
/> />
</template> </template>
@ -24,10 +24,21 @@ export default {
DemoHomeNav DemoHomeNav
}, },
data() { computed: {
return { lang() {
config: config.site const { lang } = this.$route.meta || {};
}; return lang;
},
config() {
const { locales } = config.site;
if (locales) {
return locales[this.lang];
}
return config.site;
}
} }
}; };
</script> </script>

View File

@ -18,7 +18,7 @@
<template v-for="(navItem, index) in group.items"> <template v-for="(navItem, index) in group.items">
<van-cell <van-cell
:key="index" :key="index"
:to="'/' + navItem.path" :to="`${base}/${navItem.path}`"
:title="navItem.title" :title="navItem.title"
is-link is-link
/> />
@ -39,7 +39,7 @@ export default {
}, },
props: { props: {
base: String, lang: String,
group: Object group: Object
}, },
@ -47,6 +47,12 @@ export default {
return { return {
active: [] active: []
}; };
},
computed: {
base() {
return this.lang ? `/${this.lang}` : '';
}
} }
}; };
</script> </script>

View File

@ -19,8 +19,7 @@ export default {
computed: { computed: {
title() { title() {
const route = this.$route || {}; const { name } = this.$route.meta || {};
const { name } = route.meta || {};
return name ? name.replace(/-/g, '') : ''; return name ? name.replace(/-/g, '') : '';
} }
}, },

View File

@ -1,34 +1,17 @@
import Vue from 'vue'; import Vue from 'vue';
import VueRouter from 'vue-router';
import DemoBlock from './components/DemoBlock'; import DemoBlock from './components/DemoBlock';
import DemoSection from './components/DemoSection'; import DemoSection from './components/DemoSection';
import { routes } from './router'; import { router } from './router';
import App from './App'; import App from './App';
import '@vant/touch-emulator'; import '@vant/touch-emulator';
import '../common/iframe-router';
if (process.env.NODE_ENV !== 'production') { if (process.env.NODE_ENV !== 'production') {
Vue.config.productionTip = false; Vue.config.productionTip = false;
} }
Vue.use(VueRouter);
Vue.component(DemoBlock.name, DemoBlock); Vue.component(DemoBlock.name, DemoBlock);
Vue.component(DemoSection.name, DemoSection); Vue.component(DemoSection.name, DemoSection);
const router = new VueRouter({
mode: 'hash',
routes,
scrollBehavior: (to, from, savedPosition) => savedPosition || { x: 0, y: 0 }
});
router.afterEach(() => {
if (!router.currentRoute.redirectedFrom) {
Vue.nextTick(window.syncPath);
}
});
window.vueRouter = router;
new Vue({ new Vue({
el: '#app', el: '#app',
render: h => h(App), render: h => h(App),

View File

@ -1,29 +1,86 @@
import Vue from 'vue';
import VueRouter from 'vue-router';
import decamelize from 'decamelize'; import decamelize from 'decamelize';
import DemoHome from './components/DemoHome'; import DemoHome from './components/DemoHome';
import { demos } from 'site-mobile-shared'; import { demos, config } from 'site-mobile-shared';
import { getLang, setDefaultLang } from '../common/locales';
import '../common/iframe-router';
const routes = []; const { locales, defaultLang } = config.site;
const names = Object.keys(demos);
routes.push({ setDefaultLang(defaultLang);
path: '/home',
component: DemoHome
});
routes.push({ function getRoutes() {
path: '*', const routes = [];
redirect: '/home' const names = Object.keys(demos);
}); const langs = locales ? Object.keys(locales) : [];
names.forEach(name => { if (langs.length) {
routes.push({ routes.push({
name, path: '*',
component: demos[name], redirect: () => `/${getLang()}/`
path: `/${decamelize(name, '-')}`, });
meta: {
name langs.forEach(lang => {
routes.push({
path: `/${lang}`,
component: DemoHome,
meta: { lang }
});
});
} else {
routes.push({
path: '*',
redirect: () => '/'
});
routes.push({
path: '',
component: DemoHome
});
}
names.forEach(name => {
const component = decamelize(name, '-');
if (langs.length) {
langs.forEach(lang => {
routes.push({
name: `${lang}/${component}`,
path: `/${lang}/${component}`,
component: demos[name],
meta: {
name
}
});
});
} else {
routes.push({
name,
path: `/${component}`,
component: demos[name],
meta: {
name
}
});
} }
}); });
return routes;
}
Vue.use(VueRouter);
export const router = new VueRouter({
mode: 'hash',
routes: getRoutes(),
scrollBehavior: (to, from, savedPosition) => savedPosition || { x: 0, y: 0 }
}); });
export { routes }; router.afterEach(() => {
if (!router.currentRoute.redirectedFrom) {
Vue.nextTick(window.syncPath);
}
});
window.vueRouter = router;

View File

@ -67,12 +67,8 @@ type CompileSfcOptions = {
skipStyle?: boolean; skipStyle?: boolean;
}; };
export async function compileSfc( export function parseSfc(filePath: string) {
filePath: string,
options: CompileSfcOptions = {}
): Promise<any> {
const source = readFileSync(filePath, 'utf-8'); const source = readFileSync(filePath, 'utf-8');
const jsFilePath = replaceExt(filePath, '.js');
const descriptor = compileUtils.parse({ const descriptor = compileUtils.parse({
source, source,
@ -80,9 +76,17 @@ export async function compileSfc(
needMap: false needMap: false
} as any); } as any);
const { template, styles } = descriptor; return descriptor;
}
export async function compileSfc(
filePath: string,
options: CompileSfcOptions = {}
): Promise<any> {
const tasks = [remove(filePath)]; const tasks = [remove(filePath)];
const jsFilePath = replaceExt(filePath, '.js');
const descriptor = parseSfc(filePath);
const { template, styles } = descriptor;
// compile js part // compile js part
if (descriptor.script) { if (descriptor.script) {

View File

@ -4,6 +4,7 @@ import { existsSync } from 'fs-extra';
import { import {
pascalize, pascalize,
removeExt, removeExt,
getVantConfig,
getComponents, getComponents,
smartOutputFile smartOutputFile
} from '../common'; } from '../common';
@ -19,23 +20,59 @@ type DocumentItem = {
path: string; path: string;
}; };
function formatName(component: string, lang?: string) {
component = pascalize(component);
if (lang) {
return `${component}_${lang.replace('-', '_')}`;
}
return component;
}
/**
* i18n mode:
* - action-sheet/README.md => ActionSheet_EnUS
* - action-sheet/README.zh-CN.md => ActionSheet_ZhCN
*
* default mode:
* - action-sheet/README.md => ActionSheet
*/
function resolveDocuments(components: string[]): DocumentItem[] { function resolveDocuments(components: string[]): DocumentItem[] {
const componentDocs = components const vantConfig = getVantConfig();
.filter(component => { const { locales, defaultLang } = vantConfig.site;
const absolutePath = join(SRC_DIR, component, 'README.md');
return existsSync(absolutePath);
})
.map(component => ({
name: pascalize(component),
path: join(SRC_DIR, component, 'README.md')
}));
const staticDocs = glob.sync(join(DOCS_DIR, '**/*.md')).map(path => ({ const docs: DocumentItem[] = [];
name: pascalize(parse(path).name),
path
}));
return [...componentDocs, ...staticDocs]; if (locales) {
const langs = Object.keys(locales);
langs.forEach(lang => {
const fileName = lang === defaultLang ? 'README.md' : `README.${lang}.md`;
components.forEach(component => {
docs.push({
name: formatName(component, lang),
path: join(SRC_DIR, component, fileName)
});
});
});
} else {
components.forEach(component => {
docs.push({
name: formatName(component),
path: join(SRC_DIR, component, 'README.md')
});
});
}
const staticDocs = glob.sync(join(DOCS_DIR, '**/*.md')).map(path => {
const pairs = parse(path).name.split('.');
return {
name: formatName(pairs[0], pairs[1] || defaultLang),
path
};
});
return [...staticDocs, ...docs.filter(item => existsSync(item.path))];
} }
function genImportDocuments(items: DocumentItem[]) { function genImportDocuments(items: DocumentItem[]) {

View File

@ -40,12 +40,25 @@ function genConfig(demos: DemoItem[]) {
const vantConfig = getVantConfig(); const vantConfig = getVantConfig();
const demoNames = demos.map(item => decamelize(item.name, '-')); const demoNames = demos.map(item => decamelize(item.name, '-'));
vantConfig.site.nav = vantConfig.site.nav.filter((group: any) => { function demoFilter(nav: any[]) {
group.items = group.items.filter((item: any) => return nav.filter(group => {
demoNames.includes(item.path) group.items = group.items.filter((item: any) =>
); demoNames.includes(item.path)
return group.items.length; );
}); return group.items.length;
});
}
const { nav, locales } = vantConfig.site;
if (locales) {
Object.keys(locales).forEach((lang: string) => {
if (locales[lang].nav) {
locales[lang].nav = demoFilter(locales[lang].nav);
}
});
} else if (nav) {
vantConfig.site.nav = demoFilter(nav);
}
return `export const config = ${JSON.stringify(vantConfig, null, 2)}`; return `export const config = ${JSON.stringify(vantConfig, null, 2)}`;
} }

View File

@ -14,11 +14,11 @@ module.exports = {
logo: 'https://img.yzcdn.cn/vant/logo.png', logo: 'https://img.yzcdn.cn/vant/logo.png',
links: [ links: [
{ {
image: 'https://b.yzcdn.cn/vant/logo/weapp.svg', logo: 'https://b.yzcdn.cn/vant/logo/weapp.svg',
url: 'vant-weapp' url: '/vant-weapp'
}, },
{ {
image: 'https://b.yzcdn.cn/vant/logo/github.svg', logo: 'https://b.yzcdn.cn/vant/logo/github.svg',
url: 'https://github.com/youzan/vant' url: 'https://github.com/youzan/vant'
} }
], ],
@ -27,7 +27,7 @@ module.exports = {
title: '开发指南', title: '开发指南',
items: [ items: [
{ {
path: 'intro', path: 'home',
title: '介绍' title: '介绍'
}, },
{ {
@ -352,11 +352,11 @@ module.exports = {
logo: 'https://img.yzcdn.cn/vant/logo.png', logo: 'https://img.yzcdn.cn/vant/logo.png',
links: [ links: [
{ {
image: 'https://b.yzcdn.cn/vant/logo/weapp.svg', logo: 'https://b.yzcdn.cn/vant/logo/weapp.svg',
url: 'vant-weapp' url: '/vant-weapp'
}, },
{ {
image: 'https://b.yzcdn.cn/vant/logo/github.svg', logo: 'https://b.yzcdn.cn/vant/logo/github.svg',
url: 'https://github.com/youzan/vant' url: 'https://github.com/youzan/vant'
} }
], ],
@ -365,7 +365,7 @@ module.exports = {
title: 'Essentials', title: 'Essentials',
items: [ items: [
{ {
path: 'intro', path: 'home',
title: 'Introduction' title: 'Introduction'
}, },
{ {