Animating with Vue
data:image/s3,"s3://crabby-images/49778/497786bd00c09598b03f88d2db679807063d7db0" alt=""
Hello 👋
Ramona Bîscoveanu
data:image/s3,"s3://crabby-images/b46df/b46df0bc91a3caaecee897b72c2c703cf91233e6" alt=""
👩🏼💻 Senior Developer @ SAP
@CodesOfRa
🧀 ☕️🌱
⚠️ Motion⚠️
Remember the 90s?
data:image/s3,"s3://crabby-images/75a4e/75a4e882b0bd27c3cfd008a69fca9c68597433e3" alt=""
Why animate?
It's all about communication
data:image/s3,"s3://crabby-images/b1b72/b1b723f139fdba263a21df23df981029d2646ba1" alt=""
It's also that animations are
FUN 👯♀️
data:image/s3,"s3://crabby-images/09297/092974d0fd036087ff048722570b058a032aba33" alt=""
<Transition/>
<transition name="fade">
<p v-if="show">hello</p>
</transition>
.fade-enter-active,
.fade-leave-active {
transition: opacity 0.5s ease;
}
.fade-enter-from,
.fade-leave-to {
opacity: 0;
}
data:image/s3,"s3://crabby-images/2e1ad/2e1adaeba8fad51fdc4f101e49157773d60b831f" alt=""
- <transition> as a component's root will no longer trigger transitions when the component is toggled from the outside
- v-enter-from/v-leave-from
Vue 3
<TransitionGroup />
- <transition-group> no longer renders a root element by default, but can still create one with the tag attribute.
Vue 3
data:image/s3,"s3://crabby-images/ccb1a/ccb1a514396a9479fbc9c308ee809b77a0e0d350" alt=""
<transition-group tag="div" class="tile-section" name="list" appear>
<TileComp
v-for="(tile, i) in tiles"
:key="i + 'tile'"
:header="tile.title"
:content="tile.content"
:footer="tile.footer"
></TileComp>
</transition-group>
.list-enter-active,
.list-leave-active {
transition: all 1s ease;
}
.list-enter-from,
.list-leave-to {
opacity: 0;
transform: translateY(70px);
}
JavaScript Hooks
<Transition
@before-enter="onBeforeEnter"
@enter="onEnter"
@after-enter="onAfterEnter"
@enter-cancelled="onEnterCancelled"
@before-leave="onBeforeLeave"
@leave="onLeave"
@after-leave="onAfterLeave"
@leave-cancelled="onLeaveCancelled"
>
<!-- ... -->
</Transition>
data:image/s3,"s3://crabby-images/ad248/ad248594e92b087d05a147f24cc6755e8dc6da2d" alt=""
data:image/s3,"s3://crabby-images/39444/394448c278fc1f57100a621dd9b17048b80d2a88" alt=""
🎉
🎉
SVG
+
=
🎉
Tween
gsap.to(".selector", {toVars});
gsap.from(".selector", {fromVars});
gsap.fromTo(".selector", {fromVars}, {toVars});
// special properties (duration, ease, etc.) go in toVar
gsap.set(".selector", {toVars});
data:image/s3,"s3://crabby-images/9ccfc/9ccfc840ad8ee9d759b4452b017bd79659e26c4f" alt=""
<div class="tile-section">
<TileComp
v-for="(tile, i) in tiles"
:key="i + 'tile'"
:header="tile.title"
:content="tile.content"
:footer="tile.footer"
></TileComp>
</div>
import gsap from "gsap";
...
mounted() {
gsap.from(".tile", {
duration: 0.5,
opacity: 0,
scale: 0,
y: 200,
ease: "power2",
stagger: 0.1,
});
},
data:image/s3,"s3://crabby-images/c5329/c5329accb40ad3387fc44dd701fdb5209e454ab0" alt=""
https://twitter.com/hexagoncircle/status/1527715562956066817
data:image/s3,"s3://crabby-images/07f8c/07f8c2549f073c4270d548caf8d30f977971c4ca" alt=""
<h1 ref="celebrate" class="counter" :class="{ celebrate: isCelebrate }">
{{ this.numberWithCommas(value) }}
</h1>
const tl = gsap.timeline();
mounted() {
tl.fromTo(
".counter",
{
innerText: start,
scale: 0.8,
},
{
innerText: end,
snap: { innerText: 1 },
duration: 4,
ease: "linear",
onUpdate: () => {
this.$refs.celebrate.innerText = this.numberWithCommas(
this.$refs.celebrate.innerText
);
},
onComplete: () => {
this.celebrate();
},
}
).to(".counter", {
scale: 1,
ease: "elastic.out(1, 0.2)",
duration: 1.2,
});
},
restart() {
tl.restart();
this.isCelebrate = false;
}
Page transition
<router-view v-slot="{ Component }">
<transition name="fade">
<component :is="Component" />
</transition>
</router-view>
data:image/s3,"s3://crabby-images/3fd7b/3fd7b534b1b85f83522ed73103d1153c8a7a4764" alt=""
<router-view v-slot="{ Component }">
<transition
:key="$route.path"
@enter="onEnter"
@leave="onLeave"
:css="false"
>
<component :is="Component" />
</transition>
</router-view>
import gsap from "gsap";
import SplitText from "gsap/SplitText";
gsap.registerPlugin(SplitText);
gsap.defaults({
duration: 1,
ease: "power3.inOut",
});
mySplitText(el) {
return new SplitText(el, { type: "words,chars,lines" });
},
...
onEnter(el, done) {
const masterTL = gsap.timeline({
onComplete: () => {
done;
},
});
masterTL.add(
this.enterContentTextAnimation(
mySplitText(".content-text-header").chars
)
);
masterTL.add(
this.enterContentTextAnimation(".content-text-body"),
"-=0.9" //overlap with previous by 0.9s
);
masterTL.add(this.imgScaleOut(".content img"), "<");
//The start of previous animation
},
enterContentTextAnimation(id) {
const tl = gsap.timeline();
tl.fromTo(
id,
{
yPercent: "100",
opacity: 0,
},
{
yPercent: "0",
opacity: 1,
stagger: 0.03,
}
);
},
....
imgScaleOut(id) {
const tl = gsap.timeline();
tl.from(id, {
scale: 1.5,
});
return tl;
},
SVG
data:image/s3,"s3://crabby-images/518dc/518dcf78098d55c98374f666b5b4e9b28005a799" alt=""
watch: {
data(newValue, oldValue) {
newValue.map((data, index) => {
var id = "#arc" + index;
var d = this.calculateArc(data, index);
var oldValueD = this.calculateArc(oldValue[index]);
const tl = gsap.timeline();
tl.fromTo(
id,
{
attr: { d: oldValueD },
},
{
duration: 0.5,
attr: { d: d },
yoyo: true,
}
);
});
},
},
Thank you!
data:image/s3,"s3://crabby-images/9caef/9caef42eacf8763cd81a00781a723a88590fc197" alt=""
https://github.com/CodesOfRa/co2-radial-bar-chart
https://github.com/CodesOfRa/vue-animation-counter
Animating with Vue - Fundamental
By Ramona Biscoveanu
Animating with Vue - Fundamental
- 416