Skip to content

Instantly share code, notes, and snippets.

@perrygovier
Created April 9, 2019 15:17
Show Gist options
  • Save perrygovier/1822c20ba1de6ebd5d585370355c55ae to your computer and use it in GitHub Desktop.
Save perrygovier/1822c20ba1de6ebd5d585370355c55ae to your computer and use it in GitHub Desktop.
Example of combining Stencil.js and GSAP
import {
Component,
// Prop,
State,
Element
} from '@stencil/core';
import {
TweenLite
} from "gsap/TweenLite";
@Component({
tag: 'ionic-section-activator',
styleUrl: 'ionic-section-activator.scss',
shadow: false
})
export class IonicSectionActivator {
@State() $circles = [];
@State() $lis = [];
@Element() el;
duration = 6//seconds
quickDuration = .25 //seconds
r = 31; // radius
circumference: number;
gsRefs = [];
constructor() {
this.circumference = this.r * 2 * Math.PI;
this.animationSelect = this.animationSelect.bind(this);
this.animationStart = this.animationStart.bind(this);
}
componentDidLoad() {
setTimeout(this.animationStart, 2000, 0);
// setTimeout(this.animationStart, 3000, 2);
// setTimeout(this.animationStart, 4000, 3);
// recursive function to add all the circles
const addCircle = (li, i) => {
this.$lis[i] = li;
this.$circles[i] = li.querySelector('.progress-ring__circle');
if (li.nextElementSibling && li.nextElementSibling.nodeName === 'LI') {
addCircle(li.nextElementSibling, i + 1);
}
}
addCircle(this.el.querySelector('li:nth-child(1)'), 0);
}
animationStart(index) {
this.$lis[index].classList.add('active');
TweenLite.to(this.$circles[index], .4, {
opacity: 1
});
TweenLite.to(this.$circles[index], this.duration, {
strokeDashoffset: 0,
onCompleteScope: this,
onComplete: () => {
// fade out and start the next animation
this.animationStart(
index >= this.$circles.length - 1 ? 0 : index + 1
);
this.$lis[index].classList.remove('active');
TweenLite.to(this.$circles[index], .2, {
opacity: 0,
onCompleteScope: this,
onComplete: () => {
// jump back to starting position
TweenLite.to(this.$circles[index], 0, {
strokeDashoffset: this.circumference,
lazy:true
});
}
});
}
})
}
animationSelect(index) {
this.$lis[index].classList.add('active');
this.animationStopOthers(index)
TweenLite.to(this.$circles[index], this.quickDuration, {
strokeDashoffset: 0,
opacity: 1,
onCompleteScope: this,
onComplete: function() {
// this.animationStart(index);
}}
)
}
animationRestart(index) {
TweenLite.to(this.$circles[index], .8, {
strokeDashoffset: this.circumference * -1,
lazy: true,
onCompleteScope: this,
onComplete: function() {
TweenLite.to(this.$circles[index], 0, {
strokeDashoffset: this.circumference,
opacity: 0,
lazy: true,
onCompleteScope: this,
onComplete: function() {
this.animationStart(index);
}
})
}
})
}
animationStopOthers(index) {
const circles = [];
this.$circles.forEach((circle, i) => {
if (i != index ) {
circles.push(circle);
this.$lis[i].classList.remove('active');
}
});
TweenLite.to(circles, .2, {
opacity: 0,
lazy: true,
onCompleteScope: this,
onComplete: function() {
TweenLite.to(circles, 0, {
strokeDashoffset: this.circumference,
opacity: 0,
lazy: true
});
}
})
}
circle(percent = 0) {
return (
<svg class="progress-ring" height="64" width="64" >
<circle class="progress-ring__circle"
stroke="#6C89F7"
stroke-width="2"
fill="transparent"
r={this.r}
cx="32"
cy="32"
style={{
strokeDasharray:
`${this.circumference} ${this.circumference}`,
strokeDashoffset:
this.circumference - percent / 100 * this.circumference
}}/>
</svg>
)
}
render() {
return (
<nav>
<ul>
<li onMouseEnter={() => this.animationSelect(0)}
onMouseLeave={() => this.animationRestart(0)}>
{this.circle()}
<slot name="1"></slot>
</li>
<li onMouseEnter={() => this.animationSelect(1)}
onMouseLeave={() => this.animationRestart(1)}>
{this.circle()}
<slot name="2"></slot>
</li>
<li onMouseEnter={() => this.animationSelect(2)}
onMouseLeave={() => this.animationRestart(2)}>
{this.circle()}
<slot name="3"></slot>
</li>
</ul>
</nav>
);
}
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment