/*Downloaded from https://www.codeseek.co/iseeyou911/image-exploader-FKAgv */
var ctx, imageObj, processImage, leafFactory, createChilds;
ctx = document.getElementById('_canvasMa').getContext('2d');
imageObj = new Image();
/*
Leaf = {
children : [] (Leaf*4; top-left, top-right, bottom-left, bottom-right),
parent : Leaf
}
*/
getNeigborhood = function(leaf, root, dir/*l,r,t,b*/) {
var x, y;
switch(dir) {
case 'l' : {
x = leaf.x - 1;
y = leaf.y + 1;
break;
}
case 'r' : {
x = leaf.x + leaf.w + 1;
y = leaf.y + 1;
break;
}
case 'b' : {
x = leaf.x + 1;
y = leaf.y + leaf.h + 1;
break;
}
case 't' : {
x = leaf.x + 1;
y = leaf.y - 1;
break;
}
}
return getLeafByCoords(root, x, y);
};
getLeafByCoords = function(fromLeaf, x, y) {
var child, i, result;
for (i = 0; i < fromLeaf.children.length; i++) {
child = fromLeaf.children[i];
if (x >= child.x && x <= child.x + child.w && y >= child.y && y <= child.y + child.h) {
if (child.children.length === 0) {
return child;
} else {
result = getLeafByCoords(child,x, y);
if (result) {
return result;
}
}
}
}
return null;
};
leafFactory = function(parent, children, w, h, level) {
return {
getNeigborhood : function (root, dir) {
return getNeigborhood(this, root, dir);
},
level : level || 0,
w : Math.round(w),
h : Math.round(h),
x : 0,
y : 0,
parent : parent,
children : children || [],
setBl : function(value) {
value.x = value.parent.x;
value.y = value.parent.y + value.h;
this.children[1] = value;
},
setBr : function(value) {
value.x = value.parent.x + value.w;
value.y = value.parent.y + value.h;
this.children[3] = value;
},
setTl : function(value) {
value.x = value.parent.x;
value.y = value.parent.y;
this.children[0] = value;
},
setTr : function(value) {
value.x = value.parent.x + value.w;
value.y = value.parent.y;
this.children[2] = value;
},
tr : function() {
return this.children[1];
},
br : function() {
return this.children[3];
},
tl : function() {
return this.children[0];
},
bl : function() {
return this.children[2];
}
};
};
createChilds = function(parent) {
parent.setTl(leafFactory(parent, [], parent.w / 2, parent.h / 2, parent.level + 1));
parent.setTr(leafFactory(parent, [], parent.w / 2, parent.h / 2, parent.level + 1));
parent.setBl(leafFactory(parent, [], parent.w / 2, parent.h / 2, parent.level + 1));
parent.setBr(leafFactory(parent, [], parent.w / 2, parent.h / 2, parent.level + 1));
return parent.children;
};
processImage = function() {
var buildTree, travers, needDivid, root, color, imageData, data, width, height, minWidth, minHeight, i, j, r, g, b, a, leafList;
minHeight = 0.5;
minWidth = 0.5;
width = 600;
height = 600;
ctx.drawImage(imageObj, width / 2 - imageObj.width / 2, height / 2 - imageObj.height / 2);
imageData = ctx.getImageData(0, 0, width, height);
data = imageData.data;
squares = [];
offset = 0;
root = leafFactory(null, null, 600, 600, 0);
needDivid = function(leaf) {
var offset, i, j, r, g, b, a, isBlack, isWhite, black;
isBlack = undefined;
for (i = leaf.y; i < leaf.y + leaf.h; i++) {
for (j = leaf.x; j < leaf.x + leaf.w; j++) {
offset = i * height * 4 + j * 4;
r = data[offset];
g = data[offset + 1];
b = data[offset + 2];
a = data[offset + 3];
black = (r < 200 && g < 200 && b < 200);
if ((isBlack === true && !black) || (isBlack === false && black)) {
return true;
} else {
isBlack = black;
}
}
}
leaf.isBlack = isBlack;
return false;
};
buildTree = function(parent) {
if (parent.w * 2 >= minWidth && parent.h * 2 >= minHeight) {
createChilds(parent).forEach(function(child) {
if (needDivid(child)) {
buildTree(child);
}
});
}
};
buildTree(root);
travers = function(leaf, callback) {
callback(leaf);
leaf.children.forEach(function(child) {
travers(child, callback);
});
};
leafList = [];
travers(root, function(leaf) {
var offset, i, j, r, g, b, a, color;
if (leaf.children.length !== 0 || !leaf.isBlack) {
return;
}
leafList.push(leaf);
});
simpleQuadTest = function(leaf, dirArray, callback, walked) {
var i, _leaf, result;
i = 0;
result = [leaf];
_leaf = leaf;
while(dirArray.length > i) {
_leaf = _leaf.getNeigborhood(root, dirArray[i].side ? dirArray[i].side : dirArray[i]);
if (_leaf && walked.indexOf(_leaf) === -1 && _leaf.isBlack && _leaf && _leaf.level === leaf.level - (dirArray[i].lvlAdd || 0)) {
result.push(_leaf);
} else {
return null;
}
i++;
}
callback(result);
return result;
};
var squares = [];
group = function(list) {
var walked, process, merge;
walked = [];
process = function(array) {
walked = walked.concat(array);
};
merge = function(array) {
var newLeaf, x1, y1, x2, y2, lvl;
array.forEach(function(leaf) {
if (leaf.level <= lvl || lvl === undefined) {
lvl = leaf.level;
}
if (leaf.x <= x1 || x1 === undefined) {
x1 = leaf.x;
}
if (leaf.x + leaf.w >= x2 || x2 === undefined) {
x2 = leaf.x + leaf.w;
}
if (leaf.y <= y1 || y1 === undefined) {
y1 = leaf.y;
}
if (leaf.y + leaf.h >= y2 || y2 === undefined) {
y2 = leaf.y + leaf.h;
}
});
newLeaf = leafFactory(null, null, x2 - x1, y2 - y1, lvl);
newLeaf.x = x1;
newLeaf.y = y1;
return newLeaf;
};
list.forEach(function(leaf) {
var i, pathList, grouped;
pathList= [
['l', 'b', 'r'],
['t', 'l', 'b'],
['l', 'l', 'b', 'b', {side : 'r', lvlAdd : 1}],
['r', 'r', 't', 't', {side : 'l', lvlAdd : 1}],
['b', 'b', 'r', 'r', {side : 't', lvlAdd : 1}],
['b', 'b', 'r', 'r', {side : 't', lvlAdd : 1}]
];
if (leaf.level > 9 || walked.indexOf(leaf) !== -1) {
return;
}
i = 0;
grouped = false;
while (i < pathList.length && !grouped) {
grouped = simpleQuadTest(leaf, pathList[i], process, walked);
i++;
}
if (grouped) {
grouped.forEach(function(_leaf) {
var index;
index = squares.indexOf(_leaf);
if (index !== -1) {
squares = squares.slice(0, index).concat(squares.slice(index + 1));
}
});
squares.push(merge(grouped));
} else {
squares.push(leaf);
}
});
};
group(leafList);
squares.forEach(function(leaf) {
var offset, i, j, r, g, b, a, color;
for (i = leaf.y; i < leaf.y + leaf.h; i++) {
for (j = leaf.x; j < leaf.x + leaf.w; j++) {
offset = i * height * 4 + j * 4;
color = i === leaf.y + leaf.h || j === leaf.x + leaf.w || i === leaf.y || j === leaf.x ? 0 : 255 * leaf.level / 10;
data[offset] = color;
data[offset + 1] = 0;
data[offset + 2] = 0;
data[offset + 3] = 255;
}
}
});
var impulse = 0.2, lifeTime = 1000, t = 0, center;
center = [
{x: 0, y :0, i : 126, d: 100, lt: 2000, timer : 0},
{x: 200, y : 50, i : 226, d: 100, lt: 2000, timer : 0},
{x: 350, y : 550, i :126, d:250, lt: 2000, timer : 0},
{x: 150, y :300, i : 17, d: 1500, lt: 1000, timer : 0},
{x: 250, y : 300, i : 18, d: 210, lt: 1000, timer : 0},
{x: 300, y : 300, i : 18, d:1400, lt: 1000, timer : 0},
{x: 400, y : 300, i : 118, d:1300, lt: 1000, timer : 0},
{x: 550, y : 300, i : 224, d:450, lt: 1000, timer : 0}
];
window.addEventListener('click' ,function (e){
console.log(e)
center.push({x: e.pageX, y :e.pageY, i : 1000 * Math.random(), d: t + 10, lt: 1000, timer : 0});
});
setInterval(function explode () {
ctx.clearRect(0, 0, width, height);
ctx.beginPath();
squares.forEach(function(leaf) {
var x, y, offset, i, j, r, g, b, a, color, gravity, Fm, r, r2;
leaf.xC = leaf.xC === undefined ? leaf.x + leaf.w / 2 : leaf.xC;
leaf.yC = leaf.yC === undefined ? leaf.y + leaf.h / 2: leaf.yC;
for (i = 0; i < center.length; i++) {
if (t < center[i].d || center[i].isEnded) {
continue;
}
r = (Math.pow(center[i].x - leaf.xC, 2) + Math.pow(center[i].y - leaf.yC, 2));
r2 = (Math.pow(center[i].x - leaf.x, 2) + Math.pow(center[i].y - leaf.y, 2));
Fm = Math.random() * center[i].i * ((center[i].lt / center[i].timer) / center[i].lt) / r2 / 0.00016 / (12 - leaf.level);
Fm = isFinite(Fm) && !!Fm ? Fm : 0;
leaf.xC = leaf.xC + (leaf.xC - center[i].x) / Math.sqrt(r) * Fm;
leaf.yC = leaf.yC + (leaf.yC - center[i].y) / Math.sqrt(r) * Fm;
}
r2 = Math.sqrt((Math.pow(leaf.x + leaf.w / 2 - leaf.xC, 2) + Math.pow(leaf.y + leaf.h / 2 - leaf.yC, 2)));
gravity = (1 - Math.exp(-(r2 < 2 ? 0 : r2 * (12 - leaf.level))));
gravity = isFinite(gravity) ? gravity : 0;
x = leaf.xC = leaf.xC + (((-leaf.xC + leaf.x + leaf.w / 2) / r2 * gravity) || 0);
y = leaf.yC = leaf.yC + (((-leaf.yC + leaf.y + leaf.h / 2) / r2 * gravity) || 0) ;
ctx.rect(x - leaf.w / 2, y - leaf.h / 2, leaf.w, leaf.h);
});
ctx.fillStyle = 'black';
ctx.fill();
var explCenter;
for (i = 0; i < center.length; i++) {
explCenter = center[i];
if (t < explCenter.d) {
continue;
} else if (explCenter.timer > 0 && !explCenter.isEnded){
//ctx.beginPath();
//ctx.arc( explCenter.x, explCenter.y, explCenter.i * 0.2 * explCenter.timer / explCenter.lt, 0, 2 * Math.PI);
//ctx.fillStyle = 'red';
//ctx.fill();
}
if (explCenter.timer <= explCenter.lt) {
explCenter.timer += 10;
} else {
explCenter.isEnded = true;
}
}
t += 10;
} ,10);
}
imageObj.onload = processImage;
imageObj.src = '';