mirror of
https://gitee.com/h_mo/uniapp-vue3-vite-ts-template
synced 2025-04-05 19:41:44 +08:00
209 lines
6.0 KiB
Vue
209 lines
6.0 KiB
Vue
<script lang="ts" setup name="FontAwesomeIcon">
|
|
/**
|
|
* 基于 Font Awesome Icon v6.2.0
|
|
* solid 支持
|
|
* regular 支持
|
|
* brands 支持
|
|
* light 支持
|
|
* sharp 支持
|
|
* thin 支持
|
|
* duotone 支持
|
|
*/
|
|
import { FontAwesomeIconProps } from '@/components/FontAwesomeIcon/props';
|
|
import { computed, ref } from 'vue';
|
|
const props = defineProps(FontAwesomeIconProps);
|
|
|
|
const emit = defineEmits<{
|
|
(e: 'click'): void;
|
|
}>();
|
|
|
|
const singleBeat = computed(() => props.beat && !props.fade);
|
|
const singleFade = computed(() => !props.beat && props.fade);
|
|
const beatFade = computed(() => props.beat && props.fade);
|
|
const count = computed(() => {
|
|
return props.counter > props.counterMax
|
|
? `${props.counterMax}+`
|
|
: props.counter;
|
|
});
|
|
const wrapClassObject = computed(() => {
|
|
return [
|
|
props.float ? `fa-pull-${props.float}` : '',
|
|
props.border ? 'fa-border' : '',
|
|
props.rotate ? 'fa-rotate-by' : '',
|
|
props.rotateFlip ? `fa-flip-${props.rotateFlip}` : '',
|
|
singleBeat.value ? `fa-beat` : '',
|
|
singleFade.value ? `fa-fade` : '',
|
|
beatFade.value ? `fa-beat-fade` : '',
|
|
props.bounce ? 'fa-bounce' : '',
|
|
props.flip ? 'fa-flip' : '',
|
|
props.shake ? 'fa-shake' : '',
|
|
props.spin ? 'fa-spin' : '',
|
|
props.spinReverse ? 'fa-spin-reverse' : '',
|
|
props.spinPulse ? 'fa-spin-pulse' : '',
|
|
props.stack ? 'fa-stack' : '',
|
|
props.stackChild ? `fa-stack-${props.stackX}x` : '',
|
|
props.stackInverse ? 'fa-inverse' : '',
|
|
props.counter ? 'layers' : '',
|
|
];
|
|
});
|
|
const wrapStyleObject = computed(() => {
|
|
return Object.assign(
|
|
{
|
|
'--fa-animation-duration': `${props.duration}s`,
|
|
},
|
|
props.bgColor
|
|
? {
|
|
'background-color': props.bgColor,
|
|
}
|
|
: {},
|
|
props.rotate ? { '--fa-rotate-angle': `${props.rotate}deg` } : {},
|
|
singleBeat.value ? { '--fa-beat-scale': `${props.scale}` } : {},
|
|
singleFade.value ? { '--fa-fade-opacity': `${props.opacity}` } : {},
|
|
beatFade.value
|
|
? {
|
|
'--fa-beat-fade-scale': `${props.scale}`,
|
|
'--fa-beat-fade-opacity': `${props.opacity}`,
|
|
}
|
|
: {},
|
|
props.bounce
|
|
? {
|
|
'--fa-bounce-rebound': props.bounceConfig?.rebound || '-0.125em',
|
|
'--fa-bounce-height': props.bounceConfig?.height || '-0.5em',
|
|
'--fa-bounce-start-scale-x':
|
|
props.bounceConfig?.startScaleX || '1.1',
|
|
'--fa-bounce-start-scale-y':
|
|
props.bounceConfig?.startScaleY || '0.9',
|
|
'--fa-bounce-jump-scale-x': props.bounceConfig?.jumpScaleX || '0.9',
|
|
'--fa-bounce-jump-scale-y': props.bounceConfig?.jumpScaleY || '1.1',
|
|
'--fa-bounce-land-scale-x':
|
|
props.bounceConfig?.landScaleX || '1.05',
|
|
'--fa-bounce-land-scale-y':
|
|
props.bounceConfig?.landScaleY || '0.95',
|
|
}
|
|
: {},
|
|
props.flip
|
|
? {
|
|
'--fa-flip-x': props.flipConfig?.x || '0',
|
|
'--fa-flip-y': props.flipConfig?.y || '1',
|
|
'--fa-flip-z': props.flipConfig?.z || '0',
|
|
'--fa-flip-angle': props.flipConfig?.angle || '-180deg',
|
|
}
|
|
: {},
|
|
props.spin
|
|
? {
|
|
'--fa-animation-delay': props.spinDelay,
|
|
'--fa-animation-iteration-count': props.spinIterationCount,
|
|
'--fa-animation-timing': props.spinTiming,
|
|
}
|
|
: {},
|
|
props.border
|
|
? {
|
|
'--fa-border-color': props.borderConfig?.color || '#eee',
|
|
'--fa-border-padding': props.borderConfig?.padding || '.04em .26em',
|
|
'--fa-border-radius': props.borderConfig?.radius || '.1em',
|
|
'--fa-border-style': props.borderConfig?.style || 'solid',
|
|
'--fa-border-width': props.borderConfig?.width || '.08em',
|
|
}
|
|
: {},
|
|
props.float
|
|
? {
|
|
'--fa-pull-margin': props.pullMargin,
|
|
}
|
|
: {},
|
|
props.stackChild
|
|
? {
|
|
'--fa-stack-z-index': props.stackZIndex,
|
|
}
|
|
: {},
|
|
props.stackChild && props.stackInverse
|
|
? {
|
|
'--fa-inverse': props.stackInverseColor,
|
|
}
|
|
: {},
|
|
);
|
|
});
|
|
|
|
const iconClassObject = computed(() => {
|
|
return [
|
|
`fa-${props.mode}`,
|
|
`fa-${props.name}`,
|
|
props.frameSize ? `fa-flip-${props.frameSize}` : '',
|
|
props.sharp ? 'fass' : '',
|
|
];
|
|
});
|
|
const iconStyleObject = computed(() => {
|
|
return Object.assign(
|
|
{},
|
|
props.color
|
|
? {
|
|
color: props.color,
|
|
}
|
|
: {},
|
|
props.size
|
|
? {
|
|
'font-size': `${props.size}rpx`,
|
|
}
|
|
: {},
|
|
);
|
|
});
|
|
|
|
const counterStyleObject = computed(() => {
|
|
return {
|
|
'--fa-counter-background-color': props.counterMgColor,
|
|
};
|
|
});
|
|
|
|
const onClick = () => {
|
|
emit('click');
|
|
};
|
|
</script>
|
|
<template>
|
|
<view
|
|
@click="onClick"
|
|
class="icon-wrap"
|
|
:class="wrapClassObject"
|
|
:style="wrapStyleObject"
|
|
>
|
|
<template v-if="props.stack">
|
|
<slot :stack="props.stack"></slot>
|
|
</template>
|
|
<template v-else>
|
|
<text class="icon" :style="iconStyleObject" :class="iconClassObject" />
|
|
</template>
|
|
<text v-if="props.counter" class="counter" :style="counterStyleObject">{{
|
|
count
|
|
}}</text>
|
|
</view>
|
|
</template>
|
|
<style lang="scss" scoped>
|
|
.icon-wrap {
|
|
display: inline-block;
|
|
&.layers {
|
|
position: relative;
|
|
text-align: center;
|
|
}
|
|
.icon {
|
|
display: inline-block;
|
|
color: inherit;
|
|
}
|
|
.counter {
|
|
display: inline-block;
|
|
background-color: var(--fa-counter-background-color, #ff253a);
|
|
border-radius: 1em;
|
|
box-sizing: border-box;
|
|
color: #fff;
|
|
line-height: 1;
|
|
max-width: 5em;
|
|
min-width: 1.5em;
|
|
overflow: hidden;
|
|
padding: 0.25em 0.5em;
|
|
position: absolute;
|
|
right: -0.35em;
|
|
text-overflow: ellipsis;
|
|
top: 0;
|
|
transform: scale(0.35);
|
|
transform-origin: top right;
|
|
}
|
|
}
|
|
</style>
|