潘志宝
2024-11-04 23beab5084d97f25fb40ee57055436a1c84757f1
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
<script lang="ts" setup>
import { propTypes } from '@/utils/propTypes'
import Iconify from '@purge-icons/generated'
import { useDesign } from '@/hooks/web/useDesign'
 
defineOptions({ name: 'Icon' })
 
const { getPrefixCls } = useDesign()
 
const prefixCls = getPrefixCls('icon')
 
const props = defineProps({
  // icon name
  icon: propTypes.string,
  // icon color
  color: propTypes.string,
  // icon size
  size: propTypes.number.def(16),
  // icon svg class
  svgClass: propTypes.string.def('')
})
 
const elRef = ref<ElRef>(null)
 
const isLocal = computed(() => props.icon?.startsWith('svg-icon:'))
 
const symbolId = computed(() => {
  return unref(isLocal) ? `#icon-${props.icon.split('svg-icon:')[1]}` : props.icon
})
 
const getIconifyStyle = computed(() => {
  const { color, size } = props
  return {
    fontSize: `${size}px`,
    height: '1em',
    color
  }
})
 
const getSvgClass = computed(() => {
  const { svgClass } = props
  return `iconify ${svgClass}`
})
 
const updateIcon = async (icon: string) => {
  if (unref(isLocal)) return
 
  const el = unref(elRef)
  if (!el) return
 
  await nextTick()
 
  if (!icon) return
 
  const svg = Iconify.renderSVG(icon, {})
  if (svg) {
    el.textContent = ''
    el.appendChild(svg)
  } else {
    const span = document.createElement('span')
    span.className = 'iconify'
    span.dataset.icon = icon
    el.textContent = ''
    el.appendChild(span)
  }
}
 
watch(
  () => props.icon,
  (icon: string) => {
    updateIcon(icon)
  }
)
</script>
 
<template>
  <ElIcon :class="prefixCls" :color="color" :size="size">
    <svg v-if="isLocal" :class="getSvgClass" aria-hidden="true">
      <use :xlink:href="symbolId" />
    </svg>
 
    <span v-else ref="elRef" :class="$attrs.class" :style="getIconifyStyle">
      <span :class="getSvgClass" :data-icon="symbolId"></span>
    </span>
  </ElIcon>
</template>