这一版本优化了很多

This commit is contained in:
王利强
2026-06-03 10:16:37 +08:00
parent 8046316216
commit 2af9f1fd59
954 changed files with 58194 additions and 1609 deletions

View File

@@ -0,0 +1,203 @@
<!--
* @Author: North
* @Date: 2026-01-01
* @LastEditTime: 2026-01-01
* @LastEditors: North
* @Description: Avatar 头像组件支持图片文本或图标展示
* @FilePath: /wot-design-uni/src/uni_modules/wot-design-uni/components/wd-avatar/wd-avatar.vue
-->
<template>
<view v-if="isShow" :class="rootClass" :style="rootStyle" @click="handleClick">
<!-- 默认插槽优先 -->
<slot v-if="hasDefaultSlot"></slot>
<!-- 图片 -->
<wd-img v-else-if="src" :src="src" :width="imgSize" :height="imgSize" :mode="props.mode" custom-class="wd-avatar__img" @error="handleError" />
<!-- 文本 -->
<text v-else-if="text" class="wd-avatar__text">{{ text }}</text>
<!-- 图标 -->
<wd-icon v-else-if="icon" :name="icon" custom-class="wd-avatar__icon" />
</view>
</template>
<script lang="ts">
export default {
name: 'wd-avatar',
options: {
addGlobalClass: true,
virtualHost: true,
styleIsolation: 'shared'
}
}
</script>
<script lang="ts" setup>
import { computed, useSlots, type CSSProperties, ref } from 'vue'
import wdIcon from '../wd-icon/wd-icon.vue'
import wdImg from '../wd-img/wd-img.vue'
import { addUnit, isDef, objToStyle } from '../common/util'
import { avatarProps, type AvatarSize } from './types'
import { useParent } from '../composables/useParent'
import { AVATAR_GROUP_KEY } from '../wd-avatar-group/types'
const props = defineProps(avatarProps)
const emit = defineEmits(['error', 'click'])
const slots = useSlots()
// _internal 用于 avatar-group 内部的溢出计数头像,跳过父组件上下文
const { parent, index } = props._internal ? { parent: null, index: ref(-1) } : useParent(AVATAR_GROUP_KEY)
const SIZE_MAP: Record<AvatarSize, number> = {
large: 76,
medium: 64,
normal: 54,
small: 48
}
/**
* 是否显示该头像
*/
const isShow = computed(() => {
if (!parent) {
return true
}
// 在 avatar-group 中,根据 maxCount 判断
const maxCount = parent.props.maxCount
if (!isDef(maxCount)) {
return true
}
const count = typeof maxCount === 'number' ? maxCount : parseInt(maxCount, 10)
// 检查 count 是否为有效数字
if (isNaN(count) || count <= 0) {
return true
}
return index.value < count
})
/**
* 获取实际尺寸
* 在 avatar-group 中优先使用 parent 的 size
*/
const actualSize = computed(() => {
if (parent && isDef(parent.props.size)) {
return parent.props.size
}
return props.size
})
/**
* 获取实际像素尺寸
*/
const imgSize = computed(() => {
const size = actualSize.value
if (!isDef(size)) {
return SIZE_MAP.normal
}
if (typeof size === 'string' && size in SIZE_MAP) {
return SIZE_MAP[size as AvatarSize]
}
return size
})
const hasDefaultSlot = computed(() => !!slots.default)
const rootClass = computed(() => {
const classes = ['wd-avatar', props.customClass]
// 形状类 - 在 avatar-group 中优先使用 parent 的 shape
const shape = parent && isDef(parent.props.shape) ? parent.props.shape : props.shape
classes.push(`wd-avatar--${shape}`)
// 尺寸类仅预设尺寸
const size = actualSize.value
if (typeof size === 'string' && ['large', 'medium', 'normal', 'small'].includes(size)) {
classes.push(`wd-avatar--${size}`)
}
// 在 avatar-group 中时,添加 item 类
if (parent) {
classes.push('wd-avatar-group__item')
}
return classes.join(' ')
})
/**
* 根节点样式
*/
const rootStyle = computed(() => {
const style: CSSProperties = {}
let size = ''
const sizeValue = actualSize.value
if (typeof sizeValue === 'string' && sizeValue in SIZE_MAP) {
size = addUnit(SIZE_MAP[sizeValue as AvatarSize])
} else if (isDef(sizeValue)) {
size = addUnit(sizeValue)
}
if (size) {
style.width = size
style.height = size
style.fontSize = `calc(${size} * 0.45)`
if (parent) {
style['--wot-avatar-group-overlap' as any] = `calc(${size} * -0.22)`
}
}
// 形状 - 在 avatar-group 中优先使用 parent 的 shape
const shape = parent && isDef(parent.props.shape) ? parent.props.shape : props.shape
if (shape === 'round') {
style.borderRadius = '50%'
}
// 处理层叠效果的 z-index
if (parent) {
const cascading = parent.props.cascading
if (cascading === 'left-up') {
// 左侧在上,越后面越大
style.zIndex = index.value + 1
} else if (cascading === 'right-up') {
// 右侧在上,越前面越大
const maxCount = parent.props.maxCount
let count = parent.children?.length ?? 0
if (isDef(maxCount)) {
const parsedCount = typeof maxCount === 'number' ? maxCount : parseInt(maxCount, 10)
if (!isNaN(parsedCount) && parsedCount > 0) {
count = parsedCount
}
}
style.zIndex = count - index.value
}
}
if (props.color) {
style.color = props.color
}
if (props.bgColor) {
style.backgroundColor = props.bgColor
// 有背景色但无文字色时,默认白色
if (!props.color) {
style.color = '#fff'
}
}
return `${objToStyle(style)} ${props.customStyle}`
})
const handleError = (event: any) => {
emit('error', event)
}
const handleClick = () => {
emit('click')
}
</script>
<style lang="scss" scoped>
@import './index.scss';
</style>