mirror of
https://github.com/DavidHDev/vue-bits.git
synced 2026-03-07 06:29:30 -07:00
59 lines
1.5 KiB
Vue
59 lines
1.5 KiB
Vue
<template>
|
|
<div ref="divRef" @mousemove="handleMouseMove" @focus="handleFocus" @blur="handleBlur" @mouseenter="handleMouseEnter"
|
|
@mouseleave="handleMouseLeave" :class="[
|
|
'relative rounded-3xl border overflow-hidden p-8',
|
|
className
|
|
]">
|
|
<div class="pointer-events-none absolute inset-0 opacity-0 transition-opacity duration-500 ease-in-out" :style="{
|
|
opacity,
|
|
background: `radial-gradient(circle at ${position.x}px ${position.y}px, ${spotlightColor}, transparent 80%)`,
|
|
}" />
|
|
<slot />
|
|
</div>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
import { ref } from 'vue'
|
|
|
|
interface Position {
|
|
x: number
|
|
y: number
|
|
}
|
|
|
|
interface SpotlightCardProps {
|
|
className?: string
|
|
spotlightColor?: string
|
|
}
|
|
|
|
const { className = '', spotlightColor = 'rgba(255, 255, 255, 0.25)' } = defineProps<SpotlightCardProps>()
|
|
|
|
const divRef = ref<HTMLDivElement>()
|
|
const isFocused = ref<boolean>(false)
|
|
const position = ref<Position>({ x: 0, y: 0 })
|
|
const opacity = ref<number>(0)
|
|
|
|
const handleMouseMove = (e: MouseEvent) => {
|
|
if (!divRef.value || isFocused.value) return
|
|
|
|
const rect = divRef.value.getBoundingClientRect()
|
|
position.value = { x: e.clientX - rect.left, y: e.clientY - rect.top }
|
|
}
|
|
|
|
const handleFocus = () => {
|
|
isFocused.value = true
|
|
opacity.value = 0.6
|
|
}
|
|
|
|
const handleBlur = () => {
|
|
isFocused.value = false
|
|
opacity.value = 0
|
|
}
|
|
|
|
const handleMouseEnter = () => {
|
|
opacity.value = 0.6
|
|
}
|
|
|
|
const handleMouseLeave = () => {
|
|
opacity.value = 0
|
|
}
|
|
</script> |