PIXI.ParticleContainer로 많은 양의 스프라이트를 고성능으로 처리해보자
PIXI.ParticleContainer에 담으면 많은 수의 스프라이트를 좀더 가볍게 처리할 수 있다
단점은 제한된 기능만 사용할 수 있다
그렇다해도 화면표현에 대해서는 꽤 많은것을 할 수 있으니 큰 제약이 되지 않는다
bGV0IGdldF9hbmdsZV9pbl9yYWRpYW5fYmV0d2Vlbl90d29fcG9pbnRzID0gZnVuY3Rpb24gKHBvaW50MSwgcG9pbnQyKSB7CiAgIHJldHVybiBNYXRoLmF0YW4yKHBvaW50Mi55IC0gcG9pbnQxLnksIHBvaW50Mi54IC0gcG9pbnQxLngpOwp9OwpsZXQgZ2V0X2Rpc3RhbmNlX2JldHdlZW5fdHdvX3BvaW50ID0gZnVuY3Rpb24gKHBvaW50MSwgcG9pbnQyKSB7CiAgIHZhciBhID0gcG9pbnQxLnggLSBwb2ludDIueDsKICAgdmFyIGIgPSBwb2ludDEueSAtIHBvaW50Mi55OwogICByZXR1cm4gTWF0aC5zcXJ0KGEgKiBhICsgYiAqIGIpOwp9CmZ1bmN0aW9uIGdldF9wb2ludF9ieV9yYWRpYW4oeCwgeSwgbGVuZ3RoLCBhbmdsZSkgewogICByZXR1cm4gWwogICAgICB4ICsgTWF0aC5jb3MoYW5nbGUpICogbGVuZ3RoLAogICAgICB5ICsgTWF0aC5zaW4oYW5nbGUpICogbGVuZ3RoCiAgIF07Cn0KCndpbmRvdy5hZGRFdmVudExpc3RlbmVyKCdsb2FkJywgZnVuY3Rpb24gKCkgewogICAoZnVuY3Rpb24gKCkgeyB2YXIgc2NyaXB0ID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudCgnc2NyaXB0Jyk7IHNjcmlwdC5vbmxvYWQgPSBmdW5jdGlvbiAoKSB7IHZhciBzdGF0cyA9IG5ldyBTdGF0cygpOyBkb2N1bWVudC5ib2R5LmFwcGVuZENoaWxkKHN0YXRzLmRvbSk7IHJlcXVlc3RBbmltYXRpb25GcmFtZShmdW5jdGlvbiBsb29wKCkgeyBzdGF0cy51cGRhdGUoKTsgcmVxdWVzdEFuaW1hdGlvbkZyYW1lKGxvb3ApIH0pOyB9OyBzY3JpcHQuc3JjID0gJy8vbXJkb29iLmdpdGh1Yi5pby9zdGF0cy5qcy9idWlsZC9zdGF0cy5taW4uanMnOyBkb2N1bWVudC5oZWFkLmFwcGVuZENoaWxkKHNjcmlwdCk7IH0pKCk7CgoKICAgZnVuY3Rpb24gbWFrZVBpeGkobW9kZSkgewogICAgICBsZXQgcGFyZW50ID0gZG9jdW1lbnQucXVlcnlTZWxlY3RvcignI2NvbnRhaW5lcl93aW50ZXInKTsKICAgICAgLy8gcGFyZW50LmlubmVySFRNTCA9ICcnOwogICAgICBsZXQgcGFyZW50U2l6ZSA9IHBhcmVudC5nZXRCb3VuZGluZ0NsaWVudFJlY3QoKTsKICAgICAgbGV0IHdpZHRoID0gcGFyZW50U2l6ZS53aWR0aCAvIHdpbmRvdy5kZXZpY2VQaXhlbFJhdGlvOwogICAgICBsZXQgaGVpZ2h0ID0gcGFyZW50U2l6ZS53aWR0aCAvIHdpbmRvdy5kZXZpY2VQaXhlbFJhdGlvOwogICAgICBsZXQgYXBwID0gbmV3IFBJWEkuQXBwbGljYXRpb24oewogICAgICAgICB3aWR0aCwKICAgICAgICAgaGVpZ2h0LAogICAgICAgICBhbnRpYWxpYXM6IHRydWUsCiAgICAgICAgIHRyYW5zcGFyZW50OiB0cnVlLAogICAgICAgICByZXNvbHV0aW9uOiB3aW5kb3cuZGV2aWNlUGl4ZWxSYXRpbywKICAgICAgfSk7CgogICAgICBhcHAudmlldy5zZXRBdHRyaWJ1dGUoJ2lkJywgJ3NjcmVlbicpOwogICAgICBwYXJlbnQuYXBwZW5kQ2hpbGQoYXBwLnZpZXcpOwoKICAgICAgY29uc3Qgc2VnID0gMTAwMDA7CiAgICAgIGNvbnN0IGNvbnRhaW5lciA9IG5ldyBQSVhJLlBhcnRpY2xlQ29udGFpbmVyKHNlZywgeyBhbHBoYTogdHJ1ZSwgcm90YXRpb246IHRydWUgfSk7CiAgICAgIGFwcC5zdGFnZS5hZGRDaGlsZChjb250YWluZXIpOwoKICAgICAgY2xhc3MgUmVjdGFuZ2xlIGV4dGVuZHMgUElYSS5TcHJpdGUgewogICAgICAgICBjb25zdHJ1Y3RvcigpIHsKICAgICAgICAgICAgLy9QSVhJLlRleHR1cmUuV0hJVEUKICAgICAgICAgICAgY29uc3QgdGV4dHVyZSA9IFBJWEkuVGV4dHVyZS5mcm9tKCdkYXRhOmltYWdlL3BuZztiYXNlNjQsaVZCT1J3MEtHZ29BQUFBTlNVaEVVZ0FBQUFFQUFBQUJDQVFBQUFDMUhBd0NBQUFBQzBsRVFWUjQybVA4L3g4QUF3TUNBTytpcDFzQUFBQUFTVVZPUks1Q1lJST0nKTsKICAgICAgICAgICAgc3VwZXIodGV4dHVyZSk7CiAgICAgICAgIH0KICAgICAgfQogICAgICBjbGFzcyBMaW5lIGV4dGVuZHMgUmVjdGFuZ2xlIHsKICAgICAgICAgc3BlYyA9IHt9CiAgICAgICAgIGNvbnN0cnVjdG9yKCkgewogICAgICAgICAgICBzdXBlcigpOwogICAgICAgICAgICB0aGlzLmFuY2hvci55ID0gMC41OwogICAgICAgICAgICB0aGlzLmF4aXMgPSAwLjU7CiAgICAgICAgICAgIHRoaXMudGhpY2tuZXNzID0gMTsKICAgICAgICAgfQogICAgICAgICBzZXQgdGhpY2tuZXNzKHYpIHsgdGhpcy5oZWlnaHQgPSB2OyB9CiAgICAgICAgIGdldCB0aGlja25lc3MoKSB7IHJldHVybiB0aGlzLmhlaWdodDsgfQogICAgICAgICBzZXQgYXhpcyh2KSB7IHRoaXMuYW5jaG9yLnggPSB2OyB9CiAgICAgICAgIGdldCBheGlzKCkgeyByZXR1cm4gdGhpcy5hbmNob3IueDsgfQogICAgICAgICBzZXQgYXgodikgeyB0aGlzLnNwZWMuYXggPSB2OyB0aGlzLmNhbGNwb3MoKTsgfQogICAgICAgICBnZXQgYXgoKSB7IHJldHVybiB0aGlzLnNwZWMuYXggPz8gMDsgfQogICAgICAgICBzZXQgYXkodikgeyB0aGlzLnNwZWMuYXkgPSB2OyB0aGlzLmNhbGNwb3MoKTsgfQogICAgICAgICBnZXQgYXkoKSB7IHJldHVybiB0aGlzLnNwZWMuYXkgPz8gMDsgfQogICAgICAgICBzZXQgYngodikgeyB0aGlzLnNwZWMuYnggPSB2OyB0aGlzLmNhbGNwb3MoKTsgfQogICAgICAgICBnZXQgYngoKSB7IHJldHVybiB0aGlzLnNwZWMuYnggPz8gMDsgfQogICAgICAgICBzZXQgYnkodikgeyB0aGlzLnNwZWMuYnkgPSB2OyB0aGlzLmNhbGNwb3MoKTsgfQogICAgICAgICBnZXQgYnkoKSB7IHJldHVybiB0aGlzLnNwZWMuYnkgPz8gMDsgfQogICAgICAgICBnZXQgbGVuZ3RoKCkgewogICAgICAgICAgICBsZXQgYSA9IHRoaXMuYXggLSB0aGlzLmJ4OwogICAgICAgICAgICBsZXQgYiA9IHRoaXMuYXkgLSB0aGlzLmJ5OwogICAgICAgICAgICByZXR1cm4gTWF0aC5zcXJ0KGEgKiBhICsgYiAqIGIpOwogICAgICAgICB9CiAgICAgICAgIGNhbGNwb3MoKSB7CiAgICAgICAgICAgIHRoaXMud2lkdGggPSB0aGlzLmxlbmd0aDsKICAgICAgICAgICAgdGhpcy54ID0gdGhpcy5ieCAtICgodGhpcy5ieCAtIHRoaXMuYXgpICogKDEgLSB0aGlzLmFuY2hvci54KSk7CiAgICAgICAgICAgIHRoaXMueSA9IHRoaXMuYnkgLSAoKHRoaXMuYnkgLSB0aGlzLmF5KSAqICgxIC0gdGhpcy5hbmNob3IueCkpOwogICAgICAgICAgICB0aGlzLnJvdGF0aW9uID0gTWF0aC5hdGFuMih0aGlzLmJ5IC0gdGhpcy5heSwgdGhpcy5ieCAtIHRoaXMuYXgpOwogICAgICAgICB9CiAgICAgIH0KICAgICAgY29uc3QgbGlzdCA9IFtdOwogICAgICBjb25zdCBsZW4gPSB3aWR0aCAqIDAuNTsKICAgICAgZm9yIChsZXQgaSA9IDA7IGkgPCBzZWc7IGkrKykgewogICAgICAgICBsZXQgbG4gPSBuZXcgTGluZSgpOwogICAgICAgICBsbi50aW50ID0gMHgwMDAwMDA7CiAgICAgICAgIGxuLnRoaWNrbmVzcyA9IDAuMDM7CiAgICAgICAgIGxuLmF4ID0gMC41ICogd2lkdGg7CiAgICAgICAgIGxuLmF5ID0gMC41ICogd2lkdGg7CiAgICAgICAgIGxldCBicmFkaWFuID0gKE1hdGguUEkgKiAyIC8gc2VnKSAqIGk7CiAgICAgICAgIGxldCByb3RjbnQgPSAwOwogICAgICAgICBsbi5icG9pbnQgPSAoKSA9PiB7CiAgICAgICAgICAgIGxldCBbeCwgeV0gPSBnZXRfcG9pbnRfYnlfcmFkaWFuKGxuLmF4LCBsbi5heSwgbGVuLCBicmFkaWFuICsgcm90Y250KTsKICAgICAgICAgICAgbG4uYnggPSB4OwogICAgICAgICAgICBsbi5ieSA9IHk7CiAgICAgICAgIH0KICAgICAgICAgbG4uYW5pID0gKCkgPT4gewogICAgICAgICAgICByb3RjbnQgKz0gMC4wMDAwNTsKICAgICAgICAgICAgbG4uYnBvaW50KCk7CiAgICAgICAgIH0KICAgICAgICAgbG4uYnBvaW50KCk7CiAgICAgICAgIGxpc3QucHVzaChsbik7CiAgICAgICAgIChtb2RlID8gY29udGFpbmVyIDogYXBwLnN0YWdlKS5hZGRDaGlsZChsbik7CiAgICAgIH0KICAgICAgbGV0IHVwZGF0ZSA9IGZ1bmN0aW9uICgpIHsKICAgICAgICAgLy8gaWYgKCFhcHAudmlldy5wYXJlbnROb2RlKSByZXR1cm4gZmFsc2U7CiAgICAgICAgIGZvciAobGV0IGkgPSAwOyBpIDwgbGlzdC5sZW5ndGg7IGkrKykgewogICAgICAgICAgICBsZXQgbG4gPSBsaXN0W2ldOwogICAgICAgICAgICBsbi5hbmkoKTsKICAgICAgICAgfQogICAgICB9OwogICAgICBQSVhJLlRpY2tlci5zaGFyZWQuYWRkKHVwZGF0ZSk7CgogICAgICByZXR1cm4gewogICAgICAgICBkcm9wKCkgewogICAgICAgICAgICBQSVhJLlRpY2tlci5zaGFyZWQucmVtb3ZlKHVwZGF0ZSk7CiAgICAgICAgICAgIC8vIGFwcC5zdGFnZS5kZXN0cm95KHRydWUpOwogICAgICAgICAgICBjb250YWluZXIuY2hpbGRyZW4uZm9yRWFjaChmdW5jdGlvbiAoYykgeyBjb250YWluZXIucmVtb3ZlQ2hpbGQoYykgfSkKICAgICAgICAgICAgYXBwLnN0YWdlLmNoaWxkcmVuLmZvckVhY2goZnVuY3Rpb24gKGMpIHsgYXBwLnN0YWdlLnJlbW92ZUNoaWxkKGMpIH0pCgogICAgICAgICAgICBhcHAuZGVzdHJveSh0cnVlKTsKICAgICAgICAgICAgLy8gYXBwLnN0YWdlID0gbnVsbDsKCiAgICAgICAgICAgIGFwcCA9IG51bGw7CiAgICAgICAgICAgIHdoaWxlIChsaXN0Lmxlbmd0aCkgbGlzdC5zcGxpY2UoMCwgbGlzdC5sZW5ndGgpOwogICAgICAgICB9CiAgICAgIH0KICAgfQogICBsZXQgcGl4aTsKICAgZG9jdW1lbnQucXVlcnlTZWxlY3RvcignI2J0bmZhc3QnKS5hZGRFdmVudExpc3RlbmVyKCdjbGljaycsIGUgPT4gewogICAgICBwaXhpICYmIHBpeGkuZHJvcCgpOwogICAgICBwaXhpID0gbWFrZVBpeGkoIWZhbHNlKTsKICAgfSkKICAgZG9jdW1lbnQucXVlcnlTZWxlY3RvcignI2J0bnNsb3cnKS5hZGRFdmVudExpc3RlbmVyKCdjbGljaycsIGUgPT4gewogICAgICBwaXhpICYmIHBpeGkuZHJvcCgpOwogICAgICBwaXhpID0gbWFrZVBpeGkoIXRydWUpOwogICB9KQogICBidG5mYXN0LmNsaWNrKCk7Cgp9KQ==
let get_angle_in_radian_between_two_points = function (point1, point2) {
return Math.atan2(point2.y - point1.y, point2.x - point1.x);
};
let get_distance_between_two_point = function (point1, point2) {
var a = point1.x - point2.x;
var b = point1.y - point2.y;
return Math.sqrt(a * a + b * b);
}
function get_point_by_radian(x, y, length, angle) {
return [
x + Math.cos(angle) * length,
y + Math.sin(angle) * length
];
}
window.addEventListener('load', function () {
(function () { var script = document.createElement('script'); script.onload = function () { var stats = new Stats(); document.body.appendChild(stats.dom); requestAnimationFrame(function loop() { stats.update(); requestAnimationFrame(loop) }); }; script.src = '//mrdoob.github.io/stats.js/build/stats.min.js'; document.head.appendChild(script); })();
function makePixi(mode) {
let parent = document.querySelector('#container_winter');
// parent.innerHTML = '';
let parentSize = parent.getBoundingClientRect();
let width = parentSize.width / window.devicePixelRatio;
let height = parentSize.width / window.devicePixelRatio;
let app = new PIXI.Application({
width,
height,
antialias: true,
transparent: true,
resolution: window.devicePixelRatio,
});
app.view.setAttribute('id', 'screen');
parent.appendChild(app.view);
const seg = 10000;
const container = new PIXI.ParticleContainer(seg, { alpha: true, rotation: true });
app.stage.addChild(container);
class Rectangle extends PIXI.Sprite {
constructor() {
//PIXI.Texture.WHITE
const texture = PIXI.Texture.from('');
super(texture);
}
}
class Line extends Rectangle {
spec = {}
constructor() {
super();
this.anchor.y = 0.5;
this.axis = 0.5;
this.thickness = 1;
}
set thickness(v) { this.height = v; }
get thickness() { return this.height; }
set axis(v) { this.anchor.x = v; }
get axis() { return this.anchor.x; }
set ax(v) { this.spec.ax = v; this.calcpos(); }
get ax() { return this.spec.ax ?? 0; }
set ay(v) { this.spec.ay = v; this.calcpos(); }
get ay() { return this.spec.ay ?? 0; }
set bx(v) { this.spec.bx = v; this.calcpos(); }
get bx() { return this.spec.bx ?? 0; }
set by(v) { this.spec.by = v; this.calcpos(); }
get by() { return this.spec.by ?? 0; }
get length() {
let a = this.ax - this.bx;
let b = this.ay - this.by;
return Math.sqrt(a * a + b * b);
}
calcpos() {
this.width = this.length;
this.x = this.bx - ((this.bx - this.ax) * (1 - this.anchor.x));
this.y = this.by - ((this.by - this.ay) * (1 - this.anchor.x));
this.rotation = Math.atan2(this.by - this.ay, this.bx - this.ax);
}
}
const list = [];
const len = width * 0.5;
for (let i = 0; i < seg; i++) {
let ln = new Line();
ln.tint = 0x000000;
ln.thickness = 0.03;
ln.ax = 0.5 * width;
ln.ay = 0.5 * width;
let bradian = (Math.PI * 2 / seg) * i;
let rotcnt = 0;
ln.bpoint = () => {
let [x, y] = get_point_by_radian(ln.ax, ln.ay, len, bradian + rotcnt);
ln.bx = x;
ln.by = y;
}
ln.ani = () => {
rotcnt += 0.00005;
ln.bpoint();
}
ln.bpoint();
list.push(ln);
(mode ? container : app.stage).addChild(ln);
}
let update = function () {
// if (!app.view.parentNode) return false;
for (let i = 0; i < list.length; i++) {
let ln = list[i];
ln.ani();
}
};
PIXI.Ticker.shared.add(update);
return {
drop() {
PIXI.Ticker.shared.remove(update);
// app.stage.destroy(true);
container.children.forEach(function (c) { container.removeChild(c) })
app.stage.children.forEach(function (c) { app.stage.removeChild(c) })
app.destroy(true);
// app.stage = null;
app = null;
while (list.length) list.splice(0, list.length);
}
}
}
let pixi;
document.querySelector('#btnfast').addEventListener('click', e => {
pixi && pixi.drop();
pixi = makePixi(!false);
})
document.querySelector('#btnslow').addEventListener('click', e => {
pixi && pixi.drop();
pixi = makePixi(!true);
})
btnfast.click();
})