mirror of
https://github.com/tusen-ai/naive-ui.git
synced 2025-01-12 12:25:16 +08:00
245 lines
5.3 KiB
Vue
245 lines
5.3 KiB
Vue
<i18n>
|
|
{
|
|
"zh-CN": {
|
|
"dark": "深色",
|
|
"light": "浅色",
|
|
"searchPlaceholder": "搜索组件",
|
|
"home": "首页",
|
|
"doc": "文档",
|
|
"common": "常规",
|
|
"debug": "调试"
|
|
},
|
|
"en-US": {
|
|
"dark": "Dark",
|
|
"light": "Light",
|
|
"searchPlaceholder": "Search Components",
|
|
"home": "Home",
|
|
"doc": "Documentation",
|
|
"common": "Common",
|
|
"debug": "Debug"
|
|
}
|
|
}
|
|
</i18n>
|
|
|
|
<template>
|
|
<div class="nav">
|
|
<div class="ui-logo" @click="handleLogoClick">
|
|
<img src="./assets/images/naivelogo.svg">
|
|
Naive UI ({{ version }})
|
|
</div>
|
|
<div style=" margin-left: 56px; display: flex; align-items: center;">
|
|
<n-auto-complete
|
|
v-model="searchInputValue"
|
|
style="width: 216px;"
|
|
:z-index="3001"
|
|
:placeholder="$t('searchPlaceholder')"
|
|
:options="searchOptions"
|
|
clear-after-select
|
|
blur-after-select
|
|
@select="handleSelect"
|
|
/>
|
|
<div class="nav-menu">
|
|
<n-menu mode="horizontal" :value="menuValue" @select="handleMenuSelect">
|
|
<n-menu-item :title="$t('home')" name="home" />
|
|
<n-menu-item :title="$t('doc')" name="doc" />
|
|
</n-menu>
|
|
</div>
|
|
</div>
|
|
<div class="theme-picker">
|
|
<n-select
|
|
v-model="NConfigProvider.$parent.theme"
|
|
:z-index="3001"
|
|
size="small"
|
|
:options="options"
|
|
/>
|
|
</div>
|
|
<div class="lang-picker">
|
|
<n-select
|
|
:z-index="3001"
|
|
:value="lang"
|
|
size="small"
|
|
:options="langOptions"
|
|
@change="handleLanguageChange"
|
|
/>
|
|
</div>
|
|
<div v-if="env==='development'" class="mode-picker">
|
|
<n-select
|
|
:z-index="3001"
|
|
:value="mode"
|
|
size="small"
|
|
:options="modeOptions"
|
|
@change="handleModeChange"
|
|
/>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<script>
|
|
import { version } from '../package.json'
|
|
import withapp from '../src/_mixins/withapp'
|
|
|
|
function match (pattern, string) {
|
|
if (!pattern.length) return true
|
|
if (!string.length) return false
|
|
if (pattern[0] === string[0]) return match(pattern.slice(1), string.slice(1))
|
|
return match(pattern, string.slice(1))
|
|
}
|
|
|
|
export default {
|
|
mixins: [withapp],
|
|
props: {
|
|
lang: {
|
|
type: String,
|
|
required: true
|
|
},
|
|
items: {
|
|
type: Array,
|
|
default: () => []
|
|
},
|
|
env: {
|
|
type: String,
|
|
default: null
|
|
}
|
|
},
|
|
data () {
|
|
return {
|
|
searchInputValue: '',
|
|
version,
|
|
theme: 'dark',
|
|
mode: 'debug',
|
|
langOptions: [
|
|
{
|
|
label: '中文',
|
|
value: 'zh-CN'
|
|
},
|
|
{
|
|
label: 'English',
|
|
value: 'en-US'
|
|
}
|
|
]
|
|
}
|
|
},
|
|
computed: {
|
|
showDebug () {
|
|
return this.mode === 'development'
|
|
},
|
|
menuValue () {
|
|
if (/^(\/[^/]+){2}\/doc/.test(this.$route.path)) return 'doc'
|
|
else if (this.$route.name === 'home') return 'home'
|
|
return null
|
|
},
|
|
searchOptions () {
|
|
function getLabel (item) {
|
|
if (item.title) return item.title + (item.titleExtra ? (' ' + item.titleExtra) : '')
|
|
return item.name
|
|
}
|
|
if (!this.searchInputValue) return []
|
|
const replaceRegex = / |-/g
|
|
return this.items.filter(item => {
|
|
const pattern = this.searchInputValue.toLowerCase().replace(replaceRegex, '').slice(0, 20)
|
|
const label = getLabel(item).toLowerCase().replace(replaceRegex, '')
|
|
return match(pattern, label)
|
|
}).map(item => ({
|
|
label: getLabel(item),
|
|
value: item.path
|
|
}))
|
|
},
|
|
options: function () {
|
|
return [
|
|
{
|
|
label: this.$t('dark'),
|
|
value: 'dark'
|
|
},
|
|
{
|
|
label: this.$t('light'),
|
|
value: 'light'
|
|
}
|
|
]
|
|
},
|
|
modeOptions: function () {
|
|
return [
|
|
{
|
|
label: this.$t('common'),
|
|
value: 'common'
|
|
},
|
|
{
|
|
label: this.$t('debug'),
|
|
value: 'debug'
|
|
}
|
|
]
|
|
}
|
|
},
|
|
methods: {
|
|
handleLogoClick () {
|
|
this.$router.push(
|
|
/^(\/[^/]+){3}/.exec(this.$route.path)[0]
|
|
)
|
|
},
|
|
handleSelect (value) {
|
|
this.$router.push(value)
|
|
},
|
|
handleMenuSelect (value) {
|
|
if (value === 'home') {
|
|
this.$router.push(
|
|
/^(\/[^/]+){3}/.exec(this.$route.path)[0]
|
|
)
|
|
} if (value === 'doc') {
|
|
if (/^(\/[^/]+){3}\/doc/.test(this.$route.path)) {
|
|
|
|
} else {
|
|
this.$router.push(
|
|
/^(\/[^/]+){3}/.exec(this.$route.path)[0] + '/doc/start'
|
|
)
|
|
}
|
|
}
|
|
},
|
|
handleThemeChange (theme) {
|
|
this.NConfigProvider.$parent.theme = theme
|
|
},
|
|
handleLanguageChange (lang) {
|
|
this.$emit('lang-change', lang)
|
|
},
|
|
handleModeChange (mode) {
|
|
this.mode = mode
|
|
this.$emit('mode-change', mode)
|
|
}
|
|
}
|
|
}
|
|
</script>
|
|
|
|
<style lang="scss" scoped>
|
|
.nav {
|
|
display: grid;
|
|
grid-template-columns: 288px 1fr auto 140px 200px;
|
|
grid-template-rows: 63px;
|
|
align-items: center;
|
|
}
|
|
.ui-logo {
|
|
cursor: pointer;
|
|
display: flex;
|
|
align-items: center;
|
|
padding-left: 36px;
|
|
font-size: 18px;
|
|
}
|
|
.ui-logo > img {
|
|
margin-right: 12px;
|
|
height: 32px;
|
|
width: 32px;
|
|
}
|
|
.nav-menu {
|
|
padding-left: 32px;
|
|
}
|
|
.nav-menu .n-menu-item {
|
|
height: 63px !important;
|
|
}
|
|
.theme-picker {
|
|
padding-right: 16px;
|
|
}
|
|
.lang-picker {
|
|
padding-right: 16px;
|
|
}
|
|
.mode-picker {
|
|
padding-right: 48px;
|
|
}
|
|
</style>
|