Create <TrueFocus /> text animation

This commit is contained in:
Utkarsh-Singhal-26
2025-07-11 18:18:45 +05:30
parent 064233db20
commit 0b97a8c75a
5 changed files with 400 additions and 0 deletions

View File

@@ -19,6 +19,7 @@ export const CATEGORIES = [
'Falling Text', 'Falling Text',
'Text Cursor', 'Text Cursor',
'Decrypted Text', 'Decrypted Text',
'True Focus'
] ]
}, },
{ {

View File

@@ -23,6 +23,7 @@ const textAnimations = {
'falling-text': () => import("../demo/TextAnimations/FallingTextDemo.vue"), 'falling-text': () => import("../demo/TextAnimations/FallingTextDemo.vue"),
'text-cursor': () => import("../demo/TextAnimations/TextCursorDemo.vue"), 'text-cursor': () => import("../demo/TextAnimations/TextCursorDemo.vue"),
'decrypted-text': () => import("../demo/TextAnimations/DecryptedTextDemo.vue"), 'decrypted-text': () => import("../demo/TextAnimations/DecryptedTextDemo.vue"),
'true-focus': () => import("../demo/TextAnimations/TrueFocusDemo.vue"),
}; };
const components = { const components = {

View File

@@ -0,0 +1,22 @@
import code from "@/content/TextAnimations/TrueFocus/TrueFocus.vue?raw";
import type { CodeObject } from "../../../types/code";
export const trueFocus: CodeObject = {
cli: `npx jsrepo add https://vue-bits.dev/ui/TextAnimations/TrueFocus`,
installation: `npm install motion-v`,
usage: `<template>
<TrueFocus
sentence="True Focus"
manualMode="false"
blurAmount="5"
borderColor="red"
animationDuration="2"
pauseBetweenAnimations="1"
/>
</template>
<script setup lang="ts">
import TrueFocus from "./TrueFocus.vue";
</script>`,
code,
};

View File

@@ -0,0 +1,218 @@
<script setup lang="ts">
import { ref, onUnmounted, watch, nextTick, computed } from "vue";
interface TrueFocusProps {
sentence?: string;
manualMode?: boolean;
blurAmount?: number;
borderColor?: string;
glowColor?: string;
animationDuration?: number;
pauseBetweenAnimations?: number;
}
const props = withDefaults(defineProps<TrueFocusProps>(), {
sentence: "True Focus",
manualMode: false,
blurAmount: 5,
borderColor: "green",
glowColor: "rgba(0, 255, 0, 0.6)",
animationDuration: 0.5,
pauseBetweenAnimations: 1,
});
const words = computed(() => props.sentence.split(" "));
const currentIndex = ref(0);
const lastActiveIndex = ref<number | null>(null);
const containerRef = ref<HTMLDivElement>();
const wordRefs = ref<HTMLSpanElement[]>([]);
const focusRect = ref({ x: 0, y: 0, width: 0, height: 0 });
let interval: number | null = null;
watch(
[
() => props.manualMode,
() => props.animationDuration,
() => props.pauseBetweenAnimations,
words,
],
() => {
if (interval) {
clearInterval(interval);
interval = null;
}
if (!props.manualMode) {
interval = setInterval(
() => {
currentIndex.value = (currentIndex.value + 1) % words.value.length;
},
(props.animationDuration + props.pauseBetweenAnimations) * 1000,
);
}
},
{ immediate: true },
);
watch(
[currentIndex, words],
async () => {
if (currentIndex.value === null || currentIndex.value === -1) return;
if (!wordRefs.value[currentIndex.value] || !containerRef.value) return;
await nextTick();
const parentRect = containerRef.value.getBoundingClientRect();
const activeRect = wordRefs.value[currentIndex.value].getBoundingClientRect();
focusRect.value = {
x: activeRect.left - parentRect.left,
y: activeRect.top - parentRect.top,
width: activeRect.width,
height: activeRect.height,
};
},
{ immediate: true },
);
const handleMouseEnter = (index: number) => {
if (props.manualMode) {
lastActiveIndex.value = index;
currentIndex.value = index;
}
};
const handleMouseLeave = () => {
if (props.manualMode) {
currentIndex.value = lastActiveIndex.value || 0;
}
};
const setWordRef = (el: HTMLSpanElement | null, index: number) => {
if (el) {
wordRefs.value[index] = el;
}
};
onUnmounted(() => {
if (interval) {
clearInterval(interval);
}
});
</script>
<template>
<div class="focus-container" ref="containerRef">
<span
v-for="(word, index) in words"
:key="index"
:ref="(el) => setWordRef(el as HTMLSpanElement, index)"
:class="[
'focus-word',
{ manual: manualMode },
{ active: index === currentIndex && !manualMode },
]"
:style="{
filter: index === currentIndex ? 'blur(0px)' : `blur(${blurAmount}px)`,
'--border-color': borderColor,
'--glow-color': glowColor,
transition: `filter ${animationDuration}s ease`,
}"
@mouseenter="handleMouseEnter(index)"
@mouseleave="handleMouseLeave"
>
{{ word }}
</span>
<div
class="focus-frame"
:style="{
'--border-color': borderColor,
'--glow-color': glowColor,
transform: `translate(${focusRect.x}px, ${focusRect.y}px)`,
width: `${focusRect.width}px`,
height: `${focusRect.height}px`,
opacity: currentIndex >= 0 ? 1 : 0,
transition: `all ${animationDuration}s ease`,
}"
>
<span class="top-left corner"></span>
<span class="top-right corner"></span>
<span class="bottom-left corner"></span>
<span class="bottom-right corner"></span>
</div>
</div>
</template>
<style scoped>
.focus-container {
position: relative;
display: flex;
gap: 1em;
justify-content: center;
align-items: center;
flex-wrap: wrap;
}
.focus-word {
position: relative;
font-size: 3rem;
font-weight: 900;
cursor: pointer;
transition:
filter 0.3s ease,
color 0.3s ease;
}
.focus-word.active {
filter: blur(0);
}
.focus-frame {
position: absolute;
top: 0;
left: 0;
pointer-events: none;
box-sizing: content-box;
border: none;
}
.corner {
position: absolute;
width: 1rem;
height: 1rem;
border: 3px solid var(--border-color, #fff);
filter: drop-shadow(0px 0px 4px var(--border-color, #fff));
border-radius: 3px;
transition: none;
}
.top-left {
top: -10px;
left: -10px;
border-right: none;
border-bottom: none;
}
.top-right {
top: -10px;
right: -10px;
border-left: none;
border-bottom: none;
}
.bottom-left {
bottom: -10px;
left: -10px;
border-right: none;
border-top: none;
}
.bottom-right {
bottom: -10px;
right: -10px;
border-left: none;
border-top: none;
}
</style>

View File

@@ -0,0 +1,158 @@
<template>
<div class="truefocus-demo">
<TabbedLayout>
<template #preview>
<div class="relative py-6 overflow-hidden demo-container" style="min-height: 200px">
<RefreshButton @click="forceRerender" />
<div :key="key" class="flex flex-col justify-center items-center m-8 pl-6 w-full">
<TrueFocus v-bind="config" />
</div>
</div>
<Customize>
<PreviewColor title="Text Color" v-model="borderColor" @update:model-value="forceRerender" />
<PreviewSwitch
title="Hover Mode"
v-model="manualMode"
@update:model-value="forceRerender"
/>
<PreviewSlider
title="Blur Amount"
v-model="blurAmount"
:min="0"
:max="15"
:step="0.5"
value-unit="px"
@update:model-value="forceRerender"
/>
<PreviewSlider
title="Animation Duration"
v-model="animationDuration"
:min="0.1"
:max="3"
:step="0.1"
value-unit="s"
:disabled="!manualMode"
@update:model-value="forceRerender"
/>
<PreviewSlider
title="Pause Between Animations"
v-model="pauseBetweenAnimations"
:min="0"
:max="5"
:step="0.5"
value-unit="s"
:disabled="manualMode"
@update:model-value="forceRerender"
/>
</Customize>
<PropTable :data="propData" />
<Dependencies :dependency-list="['framer-motion']" />
</template>
<template #code>
<CodeExample :code-object="trueFocus" />
</template>
<template #cli>
<CliInstallation :command="trueFocus.cli" />
</template>
</TabbedLayout>
</div>
</template>
<script setup lang="ts">
import { ref, computed } from "vue";
import TabbedLayout from "../../components/common/TabbedLayout.vue";
import PropTable from "../../components/common/PropTable.vue";
import CliInstallation from "../../components/code/CliInstallation.vue";
import CodeExample from "../../components/code/CodeExample.vue";
import Dependencies from "../../components/code/Dependencies.vue";
import Customize from "../../components/common/Customize.vue";
import PreviewColor from "../../components/common/PreviewColor.vue";
import PreviewSlider from "../../components/common/PreviewSlider.vue";
import PreviewSwitch from "../../components/common/PreviewSwitch.vue";
import RefreshButton from "../../components/common/RefreshButton.vue";
import TrueFocus from "../../content/TextAnimations/TrueFocus/TrueFocus.vue";
import { trueFocus } from "../../constants/code/TextAnimations/trueFocusCode";
import { useForceRerender } from "@/composables/useForceRerender";
const { rerenderKey: key, forceRerender } = useForceRerender();
const manualMode = ref(false);
const blurAmount = ref(5);
const animationDuration = ref(0.5);
const pauseBetweenAnimations = ref(1);
const borderColor = ref("#5227FF");
const config = computed(() => ({
sentence: "True Focus",
manualMode: manualMode.value,
blurAmount: blurAmount.value,
borderColor: borderColor.value,
animationDuration: animationDuration.value,
pauseBetweenAnimations: pauseBetweenAnimations.value,
}));
const propData = [
{
name: "sentence",
type: "string",
default: "'True Focus'",
description: "The text to display with the focus animation.",
},
{
name: "manualMode",
type: "boolean",
default: "false",
description: "Disables automatic animation when set to true.",
},
{
name: "blurAmount",
type: "number",
default: "5",
description: "The amount of blur applied to non-active words.",
},
{
name: "borderColor",
type: "string",
default: "'green'",
description: "The color of the focus borders.",
},
{
name: "glowColor",
type: "string",
default: "'rgba(0, 255, 0, 0.6)'",
description: "The color of the glowing effect on the borders.",
},
{
name: "animationDuration",
type: "number",
default: "0.5",
description: "The duration of the animation for each word.",
},
{
name: "pauseBetweenAnimations",
type: "number",
default: "1",
description: "Time to pause between focusing on each word (in auto mode).",
},
];
</script>
<style scoped>
.truefocus-demo {
width: 100%;
}
.demo-container {
display: flex;
align-items: center;
justify-content: center;
}
</style>