The InteractiveCursor
is a Svelte 5 component that provides a customizable, interactive cursor effect. It dynamically changes its position and size based on user interactions within specified trigger areas. This component is ideal for enhancing user experiences with visually engaging cursor animations.
You can install the InteractiveCursor
component using npm or pnpm:
npm install @lostisworld/svelte-interactive-cursor
pnpm add @lostisworld/svelte-interactive-cursor
useDataElementRect
property.scaleOnActive
.KeyframeAnimationOptions
.children
property.activeDataValue
to track the active interactive element and its name dynamically.ScaleOnActiveElement
type ScaleOnActiveElement = {
element: string; // The name of the element (value of `data-interactive-cursor`).
scaleMultiplicator?: number; // Scale factor to apply when the element is active.
};
InteractiveCursorOptions
interface InteractiveCursorOptions {
defaultSize?: number; // Default cursor size in pixels.
scaleOnActive?: ScaleOnActiveElement[]; // Elements with scale factors.
duration?: number; // Animation duration in milliseconds.
useDataElementRect?: string[]; // Elements that trigger cursor resizing.
}
Import the InteractiveCursor
component and include it in your Svelte application:
<script lang="ts">
import InteractiveCursor from '@lostisworld/svelte-interactive-cursor';
</script>
<div data-interactive-cursor-area>
<button data-interactive-cursor="btn">Hover me!</button>
</div>
<InteractiveCursor
defaultSize={40}
duration={300}
scaleOnActive={[{ element: 'btn', scaleMultiplicator: 2 }]}
useDataElementRect={['btn']}
/>
Here is an example with custom cursor behavior and styles:
<script lang="ts">
import InteractiveCursor, {
type ScaleOnActiveElement
} from '@lostisworld/svelte-interactive-cursor';
let currentCursorState = $state({ activeDataName: '', activeDataElement: null });
// Custom cursor props
const scaleOnActive: ScaleOnActiveElement[] = [
{ element: 'image' },
{ element: 'video', scaleMultiplicator: 4 },
{ element: 'link' },
{ element: 'mixblend', scaleMultiplicator: 8 },
{ element: 'prevslide', scaleMultiplicator: 5 },
{ element: 'nextslide', scaleMultiplicator: 5 }
];
const customCursorProps = [
{ data: 'image', icon: '<svg>...</svg>' },
{ data: 'video', icon: '<svg>...</svg>', cursorClass: 'bg-red-500 text-white' },
{ data: 'link', icon: '<svg>...</svg>', cursorClass: 'bg-sky-500 text-white' },
{ data: 'tablist', cursorClass: 'rounded-none outline outline-2 outline-purple-500' }
];
</script>
<div>
<!-- Interactive Cursor Target Areas -->
<section data-interactive-cursor-area>
<div data-interactive-cursor="image">Image</div>
<div data-interactive-cursor="video">Video</div>
<div data-interactive-cursor="link">Link</div>
<ul data-interactive-cursor="tablist">
<li>Tab 1</li>
<li>Tab 2</li>
</ul>
</section>
<!-- Interactive Cursor Component -->
<InteractiveCursor
bind:activeDataValue={currentCursorState}
{scaleOnActive}
useDataElementRect={['tablist']}
class="rounded-full flex items-center justify-center {currentCursorState.activeDataName === ''
? 'bg-white text-black'
: customCursorProps.find((state) => state.data === currentCursorState.activeDataName)
?.cursorClass || 'bg-white text-black'}"
>
{#each customCursorProps as { icon, data }}
{#if data === currentCursorState.activeDataName && icon}
{@html icon}
{/if}
{/each}
</InteractiveCursor>
</div>
Property | Type | Default | Description |
---|---|---|---|
defaultSize |
number |
32 |
The default size (in pixels) of the cursor. |
scaleOnActive |
ScaleOnActiveElement[] |
[] |
Array of objects specifying elements and their respective scaling factors. |
duration |
number |
500 |
Duration of the cursor's animation in milliseconds. |
useDataElementRect |
string[] |
[] |
Array of element names (matched by data-interactive-cursor ) for which the cursor dynamically resizes and aligns to their bounding rectangle. |
class |
string |
'' |
Additional classes to apply to the cursor element. |
children |
Snippet |
undefined |
Custom content to render inside the cursor. |
activeDataValue |
{ activeDataName: string; activeDataElement: HTMLElement } |
{ activeDataName: '', activeDataElement: null } |
Tracks the currently active interactive element's name and DOM reference. |
data-interactive-cursor-area
to define areas where the cursor can interact.data-interactive-cursor="value"
to target specific elements and associate them with custom cursor behaviors.Example:
<div data-interactive-cursor-area>
<div data-interactive-cursor="image">Image Element</div>
<div data-interactive-cursor="video">Video Element</div>
</div>
To make the cursor scale when hovering over specific elements, define those elements using the data-interactive-cursor
attribute.
<main data-interactive-cursor-area>
<button data-interactive-cursor="button">Hover Me</button>
<InteractiveCursor
defaultSize={50}
scaleOnActive={[{ element: 'button', scaleMultiplicator: 2 }]}
/>
</main>
Enable the cursor to adapt its size and position to match specific elements.
<main data-interactive-cursor-area>
<div class="card" data-interactive-cursor="card">Hover me!</div>
<InteractiveCursor useDataElementRect={['card']} />
</main>
The InteractiveCursor
component includes default styles that can be customized using the class
prop or overriding CSS variables.
.lw-interactive-cursor
: Base cursor styles..lw-interactive-cursor.active
: Active state styles..lw-interactive-cursor {
background-color: white;
color: black;
}
.lw-interactive-cursor.active {
background-color: blue;
color: white;
}
interactiveCursor
For advanced customization, you can use the interactiveCursor
function to programmatically control the cursor.
Parameter | Type | Description |
---|---|---|
cursor |
HTMLElement |
Reference to the cursor DOM element. |
options |
InteractiveCursorOptions |
Configuration options for the cursor (see table below). |
Option | Type | Default | Description |
---|---|---|---|
defaultSize |
number |
32 |
Default cursor size in pixels. |
scaleOnActive |
ScaleOnActiveElement[] |
[] |
Elements that trigger scaling when hovered over. |
duration |
number |
500 |
Animation duration in milliseconds. |
useDataElementRect |
string[] |
[] |
Elements for which bounding rect sizes should be used. |
isActive
: Boolean indicating whether the cursor is currently active.activeDataValue
: Tracks the current data-interactive-cursor
name and element.init()
: Initializes event listeners and cursor tracking.destroy()
: Cleans up event listeners and animations.data-interactive-cursor-area
attribute is present on interactive parent elements.This documentation provides clear guidance on integrating and customizing the InteractiveCursor
component for a variety of use cases. Let me know if you'd like further refinements!
Contributions are welcome! Please ensure all changes are well-documented and tested.
This project is licensed under the MIT License.
Here’s the updated documentation for your InteractiveCursor
component based on the provided code: