mirror of
https://github.com/chansee97/nova-admin.git
synced 2025-04-06 03:57:54 +08:00
feat: local svg icon
This commit is contained in:
parent
57739f960b
commit
46766a54fb
@ -1,4 +1,6 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
|
import { mapEntries } from 'radash'
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
disabled?: boolean
|
disabled?: boolean
|
||||||
}
|
}
|
||||||
@ -38,10 +40,34 @@ async function fetchIconAllList(nameList: string[]) {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
const iconLists = shallowRef<IconList[]>([])
|
// 获取所有本地图标
|
||||||
|
function generateLocalIconList() {
|
||||||
|
const localSvgList = import.meta.glob('@/assets/svg-icons/*.svg', {
|
||||||
|
query: '?raw',
|
||||||
|
import: 'default',
|
||||||
|
eager: true,
|
||||||
|
})
|
||||||
|
|
||||||
|
function getSvgName(path: string) {
|
||||||
|
const regex = /\/([^/]+)\.svg$/
|
||||||
|
const match = path.match(regex)
|
||||||
|
if (match) {
|
||||||
|
const fileName = match[1]
|
||||||
|
return fileName
|
||||||
|
}
|
||||||
|
return path
|
||||||
|
}
|
||||||
|
return mapEntries(localSvgList, (key, value) => {
|
||||||
|
return [getSvgName(key), value]
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
const iconList = shallowRef<IconList[]>([])
|
||||||
|
const LocalIconList = shallowRef({})
|
||||||
|
|
||||||
onMounted(async () => {
|
onMounted(async () => {
|
||||||
iconLists.value = await fetchIconAllList(nameList)
|
iconList.value = await fetchIconAllList(nameList)
|
||||||
|
LocalIconList.value = generateLocalIconList()
|
||||||
})
|
})
|
||||||
|
|
||||||
// 当前tab
|
// 当前tab
|
||||||
@ -70,9 +96,9 @@ function handleSelectIconTag(icon: string) {
|
|||||||
const icons = computed(() => {
|
const icons = computed(() => {
|
||||||
const hasTag = !!currentTag.value
|
const hasTag = !!currentTag.value
|
||||||
if (hasTag)
|
if (hasTag)
|
||||||
return iconLists.value[currentTab.value]?.categories[currentTag.value]
|
return iconList.value[currentTab.value]?.categories[currentTag.value]
|
||||||
else
|
else
|
||||||
return iconLists.value[currentTab.value].icons
|
return iconList.value[currentTab.value].icons
|
||||||
})
|
})
|
||||||
|
|
||||||
// 符合搜索条件的图标列表
|
// 符合搜索条件的图标列表
|
||||||
@ -119,7 +145,19 @@ function clearIcon() {
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<n-tabs :value="currentTab" type="line" animated placement="left" @update:value="handleChangeTab">
|
<n-tabs :value="currentTab" type="line" animated placement="left" @update:value="handleChangeTab">
|
||||||
<n-tab-pane v-for="(list, index) in iconLists" :key="list.prefix" :name="index" :tab="list.title">
|
<n-tab-pane name="local" tab="local">
|
||||||
|
<n-flex :size="2">
|
||||||
|
<n-el
|
||||||
|
v-for="(_icon, key) in LocalIconList" :key="key"
|
||||||
|
class="hover:(text-[var(--primary-color)] ring-1) ring-[var(--primary-color)] p-1 rounded flex-center"
|
||||||
|
:title="`local:${key}`"
|
||||||
|
@click="handleSelectIcon(`local:${key}`)"
|
||||||
|
>
|
||||||
|
<nova-icon :icon="`local:${key}`" :size="24" />
|
||||||
|
</n-el>
|
||||||
|
</n-flex>
|
||||||
|
</n-tab-pane>
|
||||||
|
<n-tab-pane v-for="(list, index) in iconList" :key="list.prefix" :name="index" :tab="list.title">
|
||||||
<n-flex vertical>
|
<n-flex vertical>
|
||||||
<n-flex size="small">
|
<n-flex size="small">
|
||||||
<n-tag
|
<n-tag
|
||||||
@ -136,7 +174,7 @@ function clearIcon() {
|
|||||||
:placeholder="$t('components.iconSelector.searchPlaceholder')"
|
:placeholder="$t('components.iconSelector.searchPlaceholder')"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<div class="h-410px">
|
<div>
|
||||||
<n-flex :size="2">
|
<n-flex :size="2">
|
||||||
<n-el
|
<n-el
|
||||||
v-for="(icon) in visibleIcons" :key="icon"
|
v-for="(icon) in visibleIcons" :key="icon"
|
||||||
|
@ -13,18 +13,40 @@ interface iconPorps {
|
|||||||
}
|
}
|
||||||
const props = withDefaults(defineProps<iconPorps>(), {
|
const props = withDefaults(defineProps<iconPorps>(), {
|
||||||
size: 18,
|
size: 18,
|
||||||
|
isLocal: false,
|
||||||
})
|
})
|
||||||
|
|
||||||
|
const isLocal = computed(() => {
|
||||||
|
return props.icon && props.icon.startsWith('local:')
|
||||||
|
})
|
||||||
|
|
||||||
|
function getLocalIcon(icon: string) {
|
||||||
|
const svgName = icon.replace('local:', '')
|
||||||
|
const svg = import.meta.glob('@/assets/svg-icons/*.svg', {
|
||||||
|
query: '?raw',
|
||||||
|
import: 'default',
|
||||||
|
eager: true,
|
||||||
|
})
|
||||||
|
return svg[`/src/assets/svg-icons/${svgName}.svg`]
|
||||||
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<n-icon
|
<n-icon
|
||||||
v-if="props.icon"
|
v-if="icon && !isLocal"
|
||||||
:size="props.size"
|
:size="size"
|
||||||
:depth="props.depth"
|
:depth="depth"
|
||||||
:color="props.color"
|
:color="color"
|
||||||
>
|
>
|
||||||
<Icon :icon="props.icon" />
|
<Icon :icon="icon" />
|
||||||
</n-icon>
|
</n-icon>
|
||||||
|
<n-icon
|
||||||
|
v-if="icon && isLocal"
|
||||||
|
:size="size"
|
||||||
|
:depth="depth"
|
||||||
|
:color="color"
|
||||||
|
v-html="getLocalIcon(icon)"
|
||||||
|
/>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<style scoped></style>
|
<style scoped></style>
|
||||||
|
@ -183,7 +183,9 @@ function handleMouseEnter(index: number) {
|
|||||||
@mousemove="setKeyboardFalse"
|
@mousemove="setKeyboardFalse"
|
||||||
>
|
>
|
||||||
<div class="grid grid-rows-2 grid-cols-[40px_1fr_30px] h-full p-2">
|
<div class="grid grid-rows-2 grid-cols-[40px_1fr_30px] h-full p-2">
|
||||||
<nova-icon :icon="option.icon" class="row-span-2 place-self-center" />
|
<div class="row-span-2 place-self-center">
|
||||||
|
<nova-icon :icon="option.icon" />
|
||||||
|
</div>
|
||||||
<span>{{ option.label }}</span>
|
<span>{{ option.label }}</span>
|
||||||
<icon-park-outline-right class="row-span-2 place-self-center" />
|
<icon-park-outline-right class="row-span-2 place-self-center" />
|
||||||
<span class="op-70">{{ option.value }}</span>
|
<span class="op-70">{{ option.value }}</span>
|
||||||
|
@ -208,7 +208,7 @@ export const staticRoutes: AppRoute.RowRoute[] = [
|
|||||||
path: '/demo/icons',
|
path: '/demo/icons',
|
||||||
title: '图标',
|
title: '图标',
|
||||||
requiresAuth: true,
|
requiresAuth: true,
|
||||||
icon: 'icon-park-outline:winking-face-with-open-eyes',
|
icon: 'local:cool',
|
||||||
componentPath: '/demo/icons/index.vue',
|
componentPath: '/demo/icons/index.vue',
|
||||||
id: 22,
|
id: 22,
|
||||||
pid: 13,
|
pid: 13,
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
import { createMenus, createRoutes } from './helper'
|
import { createMenus, createRoutes, generateCacheRoutes } from './helper'
|
||||||
import { $t, local } from '@/utils'
|
import { $t, local } from '@/utils'
|
||||||
import { router } from '@/router'
|
import { router } from '@/router'
|
||||||
import { fetchUserRoutes } from '@/service'
|
import { fetchUserRoutes } from '@/service'
|
||||||
@ -16,9 +16,9 @@ export const useRouteStore = defineStore('route-store', {
|
|||||||
state: (): RoutesStatus => {
|
state: (): RoutesStatus => {
|
||||||
return {
|
return {
|
||||||
isInitAuthRoute: false,
|
isInitAuthRoute: false,
|
||||||
|
activeMenu: null,
|
||||||
menus: [],
|
menus: [],
|
||||||
rowRoutes: [],
|
rowRoutes: [],
|
||||||
activeMenu: null,
|
|
||||||
cacheRoutes: [],
|
cacheRoutes: [],
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@ -79,6 +79,9 @@ export const useRouteStore = defineStore('route-store', {
|
|||||||
// Generate side menu
|
// Generate side menu
|
||||||
this.menus = createMenus(rowRoutes)
|
this.menus = createMenus(rowRoutes)
|
||||||
|
|
||||||
|
// Generate the route cache
|
||||||
|
this.cacheRoutes = generateCacheRoutes(rowRoutes)
|
||||||
|
|
||||||
this.isInitAuthRoute = true
|
this.isInitAuthRoute = true
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
@ -1,6 +1,32 @@
|
|||||||
import { Icon } from '@iconify/vue'
|
import { Icon } from '@iconify/vue'
|
||||||
import { NIcon } from 'naive-ui'
|
import { NIcon } from 'naive-ui'
|
||||||
|
|
||||||
export function renderIcon(icon: string, props?: import('naive-ui').IconProps) {
|
export function renderIcon(icon?: string, props?: import('naive-ui').IconProps) {
|
||||||
return () => h(NIcon, props, { default: () => h(Icon, { icon }) })
|
if (!icon)
|
||||||
|
return
|
||||||
|
|
||||||
|
return () => createIcon(icon, props)
|
||||||
|
}
|
||||||
|
|
||||||
|
export function createIcon(icon?: string, props?: import('naive-ui').IconProps) {
|
||||||
|
if (!icon)
|
||||||
|
return
|
||||||
|
|
||||||
|
const isLocal = icon.startsWith('local:')
|
||||||
|
let innerIcon: any
|
||||||
|
if (isLocal) {
|
||||||
|
const svgName = icon.replace('local:', '')
|
||||||
|
const svg = import.meta.glob('@/assets/svg-icons/*.svg', {
|
||||||
|
query: '?raw',
|
||||||
|
import: 'default',
|
||||||
|
eager: true,
|
||||||
|
})
|
||||||
|
const target = svg[`/src/assets/svg-icons/${svgName}.svg`]
|
||||||
|
innerIcon = h(NIcon, { ...props, innerHTML: target })
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
innerIcon = h(NIcon, props, { default: () => h(Icon, { icon }) })
|
||||||
|
}
|
||||||
|
|
||||||
|
return innerIcon
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user