2022-02-21 00:49:19 +01:00
|
|
|
const sleep = (ms) => new Promise(resolve => setTimeout(resolve, ms));
|
|
|
|
|
|
|
|
|
|
|
|
|
2022-02-19 00:48:41 +01:00
|
|
|
$(function () {
|
|
|
|
|
|
|
|
const SEVENTV_GLOBAL_URL = "https://api.7tv.app/v2/emotes/global";
|
|
|
|
const SEVENTV_CHANNEL_URL = "https://api.7tv.app/v2/users/" + "..." + "/emotes";
|
|
|
|
const FFZ_GLOBAL_URL = "https://api.frankerfacez.com/v1/set/global";
|
|
|
|
const FFZ_CHANNEL_URL = "https://api.frankerfacez.com/v1/room/" + "...";
|
|
|
|
const BTTV_GLOBAL_URL = "https://api.betterttv.net/3/cached/emotes/global";
|
|
|
|
const BTTV_CHANNEL_URL = "https://api.betterttv.net/3/cached/users/twitch/" + "...";
|
|
|
|
|
|
|
|
const webSocket = window.socket;
|
|
|
|
const emote_str = ["peepo", 'nod']
|
2022-02-21 00:15:03 +01:00
|
|
|
const emote_url = ['https://cdn.7tv.app/emote/603cac391cd55c0014d989be/4x', 'https://cdn.7tv.app/emote/60ae4bb30e35477634610fda/4x']
|
2022-02-19 00:48:41 +01:00
|
|
|
var img_map = {};
|
|
|
|
|
|
|
|
var lastframe = 0;
|
|
|
|
var preloaded = false, initialized = false;
|
|
|
|
var mode = "rain" || "bounce" || "explosion" || "firework" || "...";
|
|
|
|
|
|
|
|
var particles = [];
|
|
|
|
|
|
|
|
|
|
|
|
const canvas = document.getElementById('wall');
|
|
|
|
const ctx = canvas.getContext('2d');
|
|
|
|
const resizeCanvas = () => {
|
2022-02-21 00:49:19 +01:00
|
|
|
canvas.width = window.innerWidth;
|
|
|
|
canvas.height = window.innerHeight;
|
2022-02-19 00:48:41 +01:00
|
|
|
}
|
|
|
|
window.addEventListener('resize', resizeCanvas, false);
|
|
|
|
resizeCanvas();
|
|
|
|
|
|
|
|
function loadImages(strs, urls) {
|
|
|
|
let loadcount = 0, loadtotal = urls.length;
|
|
|
|
preloaded = false;
|
|
|
|
var loadedimages = {};
|
|
|
|
for (let i=0; i<urls.length; i++) {
|
|
|
|
var image = new Image();
|
|
|
|
image.onload = function () {
|
|
|
|
loadcount++;
|
|
|
|
if (loadcount == loadtotal) preloaded = true;
|
|
|
|
};
|
|
|
|
image.src = urls[i];
|
|
|
|
loadedimages[strs[i]] = image;
|
|
|
|
}
|
|
|
|
return loadedimages;
|
|
|
|
}
|
|
|
|
img_map = loadImages(emote_str, emote_url)
|
|
|
|
|
|
|
|
|
|
|
|
const part_speed = 50;
|
|
|
|
const part_bounce_el = 0.9
|
2022-02-21 00:15:03 +01:00
|
|
|
const fdt = 1;
|
|
|
|
const drt = 8;
|
|
|
|
const tWave = (t) => (t<0 || t>drt )? 0 : (t<fdt ? t/fdt : (t>(drt-fdt) ? (drt-t)/fdt : 1))
|
|
|
|
const sWave = (t, s) => (t<0 || t>drt) ? 1 : (t<fdt ? t*s/fdt : (t>(drt-fdt) ? (drt-t)*s/fdt : s))
|
2022-02-19 00:48:41 +01:00
|
|
|
|
2022-02-21 00:15:03 +01:00
|
|
|
const randRange = (min,max) => (Math.random()* (max-min) + min)
|
2022-02-19 00:48:41 +01:00
|
|
|
|
|
|
|
class Particle {
|
|
|
|
constructor(emote,data){
|
|
|
|
this.size = 64;
|
|
|
|
this.data = data;
|
2022-02-21 00:15:03 +01:00
|
|
|
this.setPhysics(randRange(0,canvas.width),
|
|
|
|
randRange(0,canvas.height),
|
|
|
|
0,0,0,0);
|
2022-02-19 00:48:41 +01:00
|
|
|
this.emote = emote;
|
|
|
|
this.time = 0;
|
2022-02-21 00:15:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
setPhysics(x,y,vx,vy,ax,ay){
|
|
|
|
this.x = x || this.x || 0;
|
|
|
|
this.y = y || this.y || 0;
|
|
|
|
this.vx = vx || this.vx || 0;
|
|
|
|
this.vy = vy || this.vy || 0;
|
|
|
|
this.ax = ax || this.ax || 0;
|
|
|
|
this.ay = ay || this.ay || 0;
|
2022-02-19 00:48:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
draw(){
|
|
|
|
ctx.globalAlpha = tWave(this.time);
|
|
|
|
if(img_map[this.emote])
|
2022-02-21 00:15:03 +01:00
|
|
|
ctx.drawImage(img_map[this.emote], this.x, this.y,
|
|
|
|
sWave(this.time,this.size), sWave(this.time,this.size));
|
2022-02-19 00:48:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
update(tick){
|
|
|
|
this.time += tick;
|
|
|
|
this.vx += tick * this.ax;
|
|
|
|
this.vy += tick * this.ay;
|
|
|
|
this.x += tick * this.vx * part_speed;
|
|
|
|
this.y += tick * this.vy * part_speed;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
class RainP extends Particle {
|
|
|
|
constructor(emote,data) {
|
|
|
|
super(emote,data)
|
2022-02-21 00:15:03 +01:00
|
|
|
this.setPhysics(randRange(0, canvas.width),
|
2022-02-21 00:49:19 +01:00
|
|
|
randRange(0, -this.size),
|
|
|
|
0,randRange(10,15),
|
2022-02-21 00:15:03 +01:00
|
|
|
0,0)
|
2022-02-21 00:49:19 +01:00
|
|
|
this.size = randRange(this.size/2,this.size)
|
2022-02-21 00:15:03 +01:00
|
|
|
this.time = 5.5+this.y/canvas.height*2;
|
2022-02-19 00:48:41 +01:00
|
|
|
}
|
2022-02-21 00:49:19 +01:00
|
|
|
|
|
|
|
draw(){
|
|
|
|
ctx.globalAlpha = tWave(this.time);
|
|
|
|
if(img_map[this.emote])
|
|
|
|
ctx.drawImage(img_map[this.emote], this.x, this.y,
|
|
|
|
this.size, this.size);
|
2022-02-19 00:48:41 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class BounceP extends Particle {
|
|
|
|
constructor(emote,data) {
|
|
|
|
super(emote,data)
|
2022-02-21 00:15:03 +01:00
|
|
|
let speed = randRange(5,10);
|
|
|
|
let ang = randRange(0, 2*Math.PI);
|
|
|
|
this.setPhysics(randRange(0, canvas.width),
|
|
|
|
randRange(0, canvas.height),
|
|
|
|
Math.cos(ang)*speed,
|
|
|
|
Math.sin(ang)*speed,
|
|
|
|
0,0)
|
2022-02-19 00:48:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
update(tick){
|
|
|
|
super.update(tick);
|
|
|
|
if (this.x < 0) {
|
|
|
|
this.vx = this.vx < 0 ? -this.vx * part_bounce_el : this.vx;
|
|
|
|
this.x = 0;
|
|
|
|
} else if (this.x + this.size > canvas.width) {
|
|
|
|
this.vx = this.vx > 0 ? -this.vx * part_bounce_el : this.vx;
|
|
|
|
this.x = canvas.width - this.size;
|
|
|
|
}
|
|
|
|
if (this.y <0) {
|
|
|
|
this.vy = this.vy < 0 ? -this.vy * part_bounce_el : this.vy;
|
|
|
|
this.y = 0;
|
|
|
|
} else if (this.y + this.size > canvas.height) {
|
|
|
|
this.vy = this.vy > 0 ? -this.vy * part_bounce_el : this.vy;
|
|
|
|
this.y = canvas.height - this.size;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class BounceGP extends BounceP {
|
|
|
|
constructor(emote,data) {
|
|
|
|
super(emote,data)
|
2022-02-21 00:15:03 +01:00
|
|
|
let speed = randRange(5,10);
|
|
|
|
let ang = randRange(0,2*Math.PI);
|
|
|
|
this.setPhysics(randRange(0,canvas.width),
|
|
|
|
randRange(0,canvas.height),
|
|
|
|
Math.cos(ang)*speed,
|
|
|
|
Math.sin(ang)*speed,
|
|
|
|
0,10);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-21 00:49:19 +01:00
|
|
|
class ExplosionP extends RainP {
|
2022-02-21 00:15:03 +01:00
|
|
|
constructor(emote,data,args){
|
|
|
|
super(emote,data)
|
2022-02-21 00:49:19 +01:00
|
|
|
let speed = randRange(5,15);
|
2022-02-21 00:15:03 +01:00
|
|
|
let fang = 2*Math.PI* (args.angle/360)
|
|
|
|
let ang = randRange(-fang/2, +fang/2)+Math.PI/2 ;//+ 2*(angle/360)*Math.PI;
|
|
|
|
this.setPhysics(args.x,args.y,
|
|
|
|
Math.cos(ang)*speed,
|
|
|
|
-Math.sin(ang)*speed,
|
|
|
|
0,10);
|
2022-02-21 00:49:19 +01:00
|
|
|
this.time = drt - fdt*2;
|
|
|
|
this.size = randRange(this.size/2,this.size)
|
2022-02-21 00:15:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
class FireworkP extends Particle {
|
|
|
|
constructor(emote,data) {
|
|
|
|
super(emote,data)
|
|
|
|
this.setPhysics(canvas.width/2,canvas.height,
|
|
|
|
0,-15,
|
|
|
|
0,0);
|
|
|
|
this.time=fdt;
|
2022-02-19 00:48:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
update(tick){
|
|
|
|
super.update(tick);
|
2022-02-21 00:15:03 +01:00
|
|
|
if(this.y <= canvas.height/4 && this.time < drt){
|
|
|
|
this.time = drt;
|
|
|
|
genExplosion(this.x,this.y,360)
|
|
|
|
}
|
2022-02-19 00:48:41 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-21 00:15:03 +01:00
|
|
|
class BombP extends Particle {
|
2022-02-19 00:48:41 +01:00
|
|
|
constructor(emote,data) {
|
|
|
|
super(emote,data)
|
2022-02-21 00:15:03 +01:00
|
|
|
this.setPhysics(randRange(0, canvas.width-this.size),
|
|
|
|
-this.size,
|
|
|
|
0,15,
|
|
|
|
0,0)
|
|
|
|
this.time=fdt;
|
2022-02-19 00:48:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
update(tick){
|
|
|
|
super.update(tick);
|
2022-02-21 00:15:03 +01:00
|
|
|
if(this.y >= canvas.height - this.size && this.time < drt){
|
|
|
|
this.time = drt;
|
|
|
|
genExplosion(this.x,this.y,60)
|
2022-02-19 00:48:41 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-02-21 00:15:03 +01:00
|
|
|
const createParticule = (partP, emote, data, args) => {
|
|
|
|
let a = new partP(emote,data,args)
|
|
|
|
particles.push(a);
|
|
|
|
setTimeout(()=>{particles.pop()}, drt*1000+2000);
|
|
|
|
}
|
|
|
|
|
|
|
|
const genExplosion = (x,y,ang) =>{
|
2022-02-21 00:49:19 +01:00
|
|
|
for(let i=0; i<100; ++i) createParticule(ExplosionP, 'nod', {},{x:x,y:y,angle:ang});
|
2022-02-21 00:15:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
const genRain = () => {
|
2022-02-21 00:49:19 +01:00
|
|
|
let xpos = randRange(0, canvas.width);
|
|
|
|
let time = 0;
|
|
|
|
const interval = setInterval(()=>{
|
|
|
|
for(let i=0; i<10; ++i) createParticule(RainP, 'nod', {},{});
|
|
|
|
++time;
|
|
|
|
if(time > 50) clearInterval(interval);
|
|
|
|
}, 100);
|
2022-02-21 00:15:03 +01:00
|
|
|
}
|
2022-02-19 00:48:41 +01:00
|
|
|
|
2022-02-21 00:49:19 +01:00
|
|
|
const genVolcano = () => {
|
|
|
|
let xpos = randRange(0, canvas.width);
|
|
|
|
let time = 0;
|
|
|
|
const interval = setInterval(()=>{
|
|
|
|
for(let i=0; i<10; ++i) createParticule(ExplosionP, 'nod', {},{x:xpos,y:canvas.height,angle:30});
|
|
|
|
++time;
|
|
|
|
if(time > 50) clearInterval(interval);
|
|
|
|
}, 100);
|
|
|
|
}
|
2022-02-19 00:48:41 +01:00
|
|
|
|
|
|
|
|
2022-02-21 00:49:19 +01:00
|
|
|
|
|
|
|
async function init() {
|
|
|
|
main(0);
|
|
|
|
sleep(1000).then(()=>{
|
2022-02-21 00:15:03 +01:00
|
|
|
createParticule(BounceGP, "nod", {}, {})
|
|
|
|
createParticule(BounceP, "nod", {}, {})
|
|
|
|
createParticule(BounceP, "nod", {}, {})
|
|
|
|
createParticule(BombP, "nod", {}, {})
|
|
|
|
createParticule(FireworkP, "nod", {}, {})
|
2022-02-21 00:49:19 +01:00
|
|
|
genRain();
|
|
|
|
genVolcano();
|
|
|
|
})
|
2022-02-19 00:48:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
function main(tframe) {
|
|
|
|
window.requestAnimationFrame(main);
|
|
|
|
|
|
|
|
var dt = (tframe - lastframe) / 1000;
|
|
|
|
lastframe = tframe;
|
|
|
|
|
|
|
|
ctx.clearRect(0, 0, canvas.width, canvas.height);
|
|
|
|
if (!initialized) {
|
2022-02-21 00:15:03 +01:00
|
|
|
if (preloaded) setTimeout(function(){initialized = true;}, 1000);
|
2022-02-19 00:48:41 +01:00
|
|
|
} else {
|
2022-02-21 00:15:03 +01:00
|
|
|
particles.forEach(p => p.update(dt));
|
|
|
|
particles.forEach(p => p.draw());
|
2022-02-19 00:48:41 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
init()
|
|
|
|
|
|
|
|
const handleSocketMessage = (e)=>{
|
|
|
|
try {
|
|
|
|
let rawMessage = e.data,
|
|
|
|
message = JSON.parse(rawMessage);
|
|
|
|
|
|
|
|
if(!message.hasOwnProperty('eventFamily') || message.eventFamily != 'emotewall' ||
|
|
|
|
!message.hasOwnProperty('eventType') || !message.hasOwnProperty('data'))
|
|
|
|
return;
|
|
|
|
|
|
|
|
console.log(message.eventType, message.data)
|
|
|
|
if(message.eventType == 'message') {
|
|
|
|
//....
|
|
|
|
}
|
|
|
|
} catch (ex) {
|
|
|
|
console.log(ex)
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
jQuery(async ()=>{
|
|
|
|
|
|
|
|
try{
|
|
|
|
socket.addFamilyHandler("overlay", handleSocketMessage);
|
|
|
|
if(socket){
|
|
|
|
while(socket.getReadyState() === 0){
|
|
|
|
await new Promise(r => setTimeout(r, 500));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}catch(e) {console.log(e)}
|
|
|
|
})
|
|
|
|
|
|
|
|
});
|