Image exploader

In this example below you will see how to do a Image exploader with some HTML / CSS and Javascript

Transform image to binary color image, smart split image to squares and totaly explode it. You also may click to image to make some explode

Thumbnail
This awesome code was written by iseeyou911, you can see more from this user in the personal repository.
You can find the original code on Codepen.io
Copyright iseeyou911 ©
  • HTML
  • JavaScript
<!DOCTYPE html>
<html lang="en" >

<head>
  <meta charset="UTF-8">
  <title>Image exploader</title>
  <meta name="viewport" content="width=device-width, initial-scale=1">
  
  
  
  
</head>

<body>

  <canvas width="600" height="600" id="_canvasMa">

</canvas>
  
  

    <script  src="js/index.js"></script>




</body>

</html>

/*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 = '';

Comments