feat: local svg icon

This commit is contained in:
chansee97 2024-06-10 12:38:27 +08:00
parent 57739f960b
commit 46766a54fb
6 changed files with 108 additions and 17 deletions

View File

@ -1,4 +1,6 @@
<script setup lang="ts">
import { mapEntries } from 'radash'
interface Props {
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 () => {
iconLists.value = await fetchIconAllList(nameList)
iconList.value = await fetchIconAllList(nameList)
LocalIconList.value = generateLocalIconList()
})
// tab
@ -70,9 +96,9 @@ function handleSelectIconTag(icon: string) {
const icons = computed(() => {
const hasTag = !!currentTag.value
if (hasTag)
return iconLists.value[currentTab.value]?.categories[currentTag.value]
return iconList.value[currentTab.value]?.categories[currentTag.value]
else
return iconLists.value[currentTab.value].icons
return iconList.value[currentTab.value].icons
})
//
@ -119,7 +145,19 @@ function clearIcon() {
</template>
<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 size="small">
<n-tag
@ -136,7 +174,7 @@ function clearIcon() {
:placeholder="$t('components.iconSelector.searchPlaceholder')"
/>
<div class="h-410px">
<div>
<n-flex :size="2">
<n-el
v-for="(icon) in visibleIcons" :key="icon"

View File

@ -13,18 +13,40 @@ interface iconPorps {
}
const props = withDefaults(defineProps<iconPorps>(), {
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>
<template>
<n-icon
v-if="props.icon"
:size="props.size"
:depth="props.depth"
:color="props.color"
v-if="icon && !isLocal"
:size="size"
:depth="depth"
:color="color"
>
<Icon :icon="props.icon" />
<Icon :icon="icon" />
</n-icon>
<n-icon
v-if="icon && isLocal"
:size="size"
:depth="depth"
:color="color"
v-html="getLocalIcon(icon)"
/>
</template>
<style scoped></style>

View File

@ -183,7 +183,9 @@ function handleMouseEnter(index: number) {
@mousemove="setKeyboardFalse"
>
<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>
<icon-park-outline-right class="row-span-2 place-self-center" />
<span class="op-70">{{ option.value }}</span>

View File

@ -208,7 +208,7 @@ export const staticRoutes: AppRoute.RowRoute[] = [
path: '/demo/icons',
title: '图标',
requiresAuth: true,
icon: 'icon-park-outline:winking-face-with-open-eyes',
icon: 'local:cool',
componentPath: '/demo/icons/index.vue',
id: 22,
pid: 13,

View File

@ -1,4 +1,4 @@
import { createMenus, createRoutes } from './helper'
import { createMenus, createRoutes, generateCacheRoutes } from './helper'
import { $t, local } from '@/utils'
import { router } from '@/router'
import { fetchUserRoutes } from '@/service'
@ -16,9 +16,9 @@ export const useRouteStore = defineStore('route-store', {
state: (): RoutesStatus => {
return {
isInitAuthRoute: false,
activeMenu: null,
menus: [],
rowRoutes: [],
activeMenu: null,
cacheRoutes: [],
}
},
@ -79,6 +79,9 @@ export const useRouteStore = defineStore('route-store', {
// Generate side menu
this.menus = createMenus(rowRoutes)
// Generate the route cache
this.cacheRoutes = generateCacheRoutes(rowRoutes)
this.isInitAuthRoute = true
},
},

View File

@ -1,6 +1,32 @@
import { Icon } from '@iconify/vue'
import { NIcon } from 'naive-ui'
export function renderIcon(icon: string, props?: import('naive-ui').IconProps) {
return () => h(NIcon, props, { default: () => h(Icon, { icon }) })
export function renderIcon(icon?: string, props?: import('naive-ui').IconProps) {
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
}