
Documentation
Webflow
Code
Setup: External Scripts
External Scripts in Webflow
Make sure to always put the External Scripts before the Javascript step of the resource.
In this video you learn where to put these in your Webflow project? Or how to include a paid GSAP Club plugin in your project?
HTML
<script src="https://cdn.jsdelivr.net/npm/gsap@3.13.0/dist/gsap.min.js"></script>
<script src="https://cdn.jsdelivr.net/npm/gsap@3.13.0/dist/DrawSVGPlugin.min.js"></script>
Step 1: Copy structure to Webflow
Copy structure to Webflow
In the video below we described how you can copy + paste the structure of this resource to your Webflow project.
Copy to Webflow
Webflow structure is not required for this resource.
Step 1: Add HTML
HTML
<a data-draw-line="" href="#" class="text-draw">
<span class="text-draw__span">Hover me</span>
<div data-draw-line-box="" class="text-draw__box"></div>
</a>
HTML structure is not required for this resource.
Step 2: Add CSS
CSS
.text-draw {
cursor: pointer;
margin-left: 1em;
margin-right: 1em;
padding-top: .625em;
font-size: 2em;
text-decoration: none;
}
.text-draw__span {
margin-bottom: 0;
font-size: 1.5em;
font-weight: 500;
line-height: 1.1;
}
.text-draw__box {
color: #e55050;
width: 100%;
height: .625em;
position: relative;
}
.text-draw__box-svg {
width: 100%;
height: 100%;
position: absolute;
overflow: visible !important;
}
Step 2: Add custom Javascript
Custom Javascript in Webflow
In this video, Ilja gives you some guidance about using JavaScript in Webflow:
Step 2: Add Javascript
Step 3: Add Javascript
Javascript
gsap.registerPlugin(DrawSVGPlugin);
function initDrawRandomUnderline() {
const svgVariants = [
`<svg width="310" height="40" viewBox="0 0 310 40" fill="none" xmlns="http://www.w3.org/2000/svg"><path d="M5 20.9999C26.7762 16.2245 49.5532 11.5572 71.7979 14.6666C84.9553 16.5057 97.0392 21.8432 109.987 24.3888C116.413 25.6523 123.012 25.5143 129.042 22.6388C135.981 19.3303 142.586 15.1422 150.092 13.3333C156.799 11.7168 161.702 14.6225 167.887 16.8333C181.562 21.7212 194.975 22.6234 209.252 21.3888C224.678 20.0548 239.912 17.991 255.42 18.3055C272.027 18.6422 288.409 18.867 305 17.9999" stroke="currentColor" stroke-width="10" stroke-linecap="round"/></svg>`,
`<svg width="310" height="40" viewBox="0 0 310 40" fill="none" xmlns="http://www.w3.org/2000/svg"><path d="M5 24.2592C26.233 20.2879 47.7083 16.9968 69.135 13.8421C98.0469 9.5853 128.407 4.02322 158.059 5.14674C172.583 5.69708 187.686 8.66104 201.598 11.9696C207.232 13.3093 215.437 14.9471 220.137 18.3619C224.401 21.4596 220.737 25.6575 217.184 27.6168C208.309 32.5097 197.199 34.281 186.698 34.8486C183.159 35.0399 147.197 36.2657 155.105 26.5837C158.11 22.9053 162.993 20.6229 167.764 18.7924C178.386 14.7164 190.115 12.1115 201.624 10.3984C218.367 7.90626 235.528 7.06127 252.521 7.49276C258.455 7.64343 264.389 7.92791 270.295 8.41825C280.321 9.25056 296 10.8932 305 13.0242" stroke="#E55050" stroke-width="10" stroke-linecap="round"/></svg>`,
`<svg width="310" height="40" viewBox="0 0 310 40" fill="none" xmlns="http://www.w3.org/2000/svg"><path d="M5 29.5014C9.61174 24.4515 12.9521 17.9873 20.9532 17.5292C23.7742 17.3676 27.0987 17.7897 29.6575 19.0014C33.2644 20.7093 35.6481 24.0004 39.4178 25.5014C48.3911 29.0744 55.7503 25.7731 63.3048 21.0292C67.9902 18.0869 73.7668 16.1366 79.3721 17.8903C85.1682 19.7036 88.2173 26.2464 94.4121 27.2514C102.584 28.5771 107.023 25.5064 113.276 20.6125C119.927 15.4067 128.83 12.3333 137.249 15.0014C141.418 16.3225 143.116 18.7528 146.581 21.0014C149.621 22.9736 152.78 23.6197 156.284 24.2514C165.142 25.8479 172.315 17.5185 179.144 13.5014C184.459 10.3746 191.785 8.74853 195.868 14.5292C199.252 19.3205 205.597 22.9057 211.621 22.5014C215.553 22.2374 220.183 17.8356 222.979 15.5569C225.4 13.5845 227.457 11.1105 230.742 10.5292C232.718 10.1794 234.784 12.9691 236.164 14.0014C238.543 15.7801 240.717 18.4775 243.356 19.8903C249.488 23.1729 255.706 21.2551 261.079 18.0014C266.571 14.6754 270.439 11.5202 277.146 13.6125C280.725 14.7289 283.221 17.209 286.393 19.0014C292.321 22.3517 298.255 22.5014 305 22.5014" stroke="#E55050" stroke-width="10" stroke-linecap="round"/></svg>`,
`<svg width="310" height="40" viewBox="0 0 310 40" fill="none" xmlns="http://www.w3.org/2000/svg"><path d="M17.0039 32.6826C32.2307 32.8412 47.4552 32.8277 62.676 32.8118C67.3044 32.807 96.546 33.0555 104.728 32.0775C113.615 31.0152 104.516 28.3028 102.022 27.2826C89.9573 22.3465 77.3751 19.0254 65.0451 15.0552C57.8987 12.7542 37.2813 8.49399 44.2314 6.10216C50.9667 3.78422 64.2873 5.81914 70.4249 5.96641C105.866 6.81677 141.306 7.58809 176.75 8.59886C217.874 9.77162 258.906 11.0553 300 14.4892" stroke="#E55050" stroke-width="10" stroke-linecap="round"/></svg>`,
`<svg width="310" height="40" viewBox="0 0 310 40" fill="none" xmlns="http://www.w3.org/2000/svg"><path d="M4.99805 20.9998C65.6267 17.4649 126.268 13.845 187.208 12.8887C226.483 12.2723 265.751 13.2796 304.998 13.9998" stroke="currentColor" stroke-width="10" stroke-linecap="round"/></svg>`,
`<svg width="310" height="40" viewBox="0 0 310 40" fill="none" xmlns="http://www.w3.org/2000/svg"><path d="M5 29.8857C52.3147 26.9322 99.4329 21.6611 146.503 17.1765C151.753 16.6763 157.115 15.9505 162.415 15.6551C163.28 15.6069 165.074 15.4123 164.383 16.4275C161.704 20.3627 157.134 23.7551 153.95 27.4983C153.209 28.3702 148.194 33.4751 150.669 34.6605C153.638 36.0819 163.621 32.6063 165.039 32.2029C178.55 28.3608 191.49 23.5968 204.869 19.5404C231.903 11.3436 259.347 5.83254 288.793 5.12258C294.094 4.99476 299.722 4.82265 305 5.45025" stroke="#E55050" stroke-width="10" stroke-linecap="round"/></svg>`
];
// Add attributes to <svg> elements
function decorateSVG(svgEl) {
svgEl.setAttribute('class', 'text-draw__box-svg');
svgEl.setAttribute('preserveAspectRatio', 'none');
svgEl.querySelectorAll('path').forEach(path => {
path.setAttribute('stroke', 'currentColor');
});
}
let nextIndex = null;
document.querySelectorAll('[data-draw-line]').forEach(container => {
const box = container.querySelector('[data-draw-line-box]');
if (!box) return;
let enterTween = null;
let leaveTween = null;
container.addEventListener('mouseenter', () => {
// Don't restart if still playing
if (enterTween && enterTween.isActive()) return;
if (leaveTween && leaveTween.isActive()) leaveTween.kill();
// Random Start
if (nextIndex === null) {
nextIndex = Math.floor(Math.random() * svgVariants.length);
}
// Animate Draw
box.innerHTML = svgVariants[nextIndex];
const svg = box.querySelector('svg');
if (svg) {
decorateSVG(svg);
const path = svg.querySelector('path');
if (path) {
gsap.set(path, { drawSVG: '0%' });
enterTween = gsap.to(path, {
duration: 0.5,
drawSVG: '100%',
ease: 'power2.inOut',
onComplete: () => { enterTween = null; }
});
}
}
// Advance for next hover across all items
nextIndex = (nextIndex + 1) % svgVariants.length;
});
container.addEventListener('mouseleave', () => {
const path = box.querySelector('path');
if (!path) return;
const playOut = () => {
// Don't restart if still drawing out
if (leaveTween && leaveTween.isActive()) return;
leaveTween = gsap.to(path, {
duration: 0.5,
drawSVG: '100% 100%',
ease: 'power2.inOut',
onComplete: () => {
leaveTween = null;
box.innerHTML = ''; // remove SVG when done
}
});
};
if (enterTween && enterTween.isActive()) {
// Wait until draw-in finishes
enterTween.eventCallback('onComplete', playOut);
} else {
playOut();
}
});
});
}
// Initialize Draw Random Underline
document.addEventListener('DOMContentLoaded', function() {
initDrawRandomUnderline();
});
Step 3: Add custom CSS
Step 2: Add custom CSS
Custom CSS in Webflow
Curious about where to put custom CSS in Webflow? Ilja explains it in the below video:
CSS
Implementation
Draw line element
Use the [data-draw-line]
attribute to specify which element should trigger the line drawing animation on hover.
Draw line box
The script will look for the [data-draw-line-box]
attribute and append (random) one of the <svg> elements defined in the svgVariants
javascript variable.
Creating your own lines in Figma
We used Figma’s Pencil tool to draw this set of <svg> lines. Just be sure to start from the left—that’s how the script determines the correct draw direction.
