├── README.md ├── bower.json ├── bug-min.js ├── bug.js ├── chrome-extension ├── bug extension.crx ├── bug-icon-128.png ├── bug-icon-16.png ├── bug-icon-48.png ├── bug.js ├── manifest.json ├── options.html └── options.js ├── example.html ├── fly-sprite.png ├── originals ├── fly-sprite-10x.psd └── spider2.psd └── spider-sprite.png /README.md: -------------------------------------------------------------------------------- 1 | Bug 2 | =========== 3 | 4 | Add bugs to your page. 5 | 6 | 7 | 8 | Features 9 | -------- 10 | 11 | * Creates multiple fruit flys which fly and walk around the browser window. 12 | * Creates multiple spiders which walk around the browser window. 13 | * Flys are responsive to mouse movements (optional) and mouseover events. 14 | 15 | 16 | Demo 17 | ---- 18 | 19 | See project page: http://auz.github.io/Bug/ 20 | 21 | 22 | Dependencies 23 | ------------ 24 | 25 | None, all native js code 26 | 27 | 28 | Compatibility 29 | ------------- 30 | 31 | Works on all browsers that support CSS3 transforms, even mobile (that I've tested). 32 | 33 | See http://caniuse.com/transforms2d 34 | 35 | 36 | How to use 37 | ---------- 38 | 39 | Include the JS somewhere, and then initialize with 40 | ```js 41 | new BugController(); 42 | ``` 43 | or 44 | ```js 45 | new BugController({'minBugs':10, 'maxBugs':50, 'mouseOver':'die'}); 46 | ``` 47 | You can use SpiderController() as a shortcut for loading options and the sprite for the spiders. 48 | 49 | ``` 50 | new SpiderController({'minBugs':2, 'maxBugs':6}); 51 | ``` 52 | 53 | See example.html 54 | 55 | BugController constructor can optionally take an object of options. To make this js more async friendly, you can adjust the default options at the top of bug.js, and then instantiate at the bottom of the file as above. This will allow one to wrap the entire script in a closure to prevent any global window name space overlaps. 56 | 57 | Async code: 58 | ``` 59 | var targethead = window.document.getElementsByTagName("head")[0], 60 | loadedSpiders = false, 61 | jst = window.document.createElement("script"); 62 | jst.async = true; 63 | jst.type = "text/javascript"; 64 | jst.src = "/path/to/bug-min.js"; 65 | jst.onload = jst.onreadystatechange = function() { 66 | if (!loadedSpiders && (!this.readyState || this.readyState == 'complete')) { 67 | loadedSpiders = true; 68 | // start fire the JS. 69 | new BugController(); 70 | } 71 | }; 72 | targethead.appendChild(jst); 73 | ``` 74 | 75 | Options 76 | ------- 77 | 78 | * minDelay - Minimum delay before a bug will appear on the page. (default: 500) 79 | * maxDelay - Maximum delay before a bug will appear on the page. (default: 10000) 80 | * minBugs - Minumum number of bugs to show. (default: 1) 81 | * maxBugs - Maximum number of bugs to show. (default: 20) 82 | * minSpeed - Minimum speed of a bug, in no particular units. (default: 1) 83 | * maxSpeed - Maximum speed of a bug, in no particular units. (default: 3) 84 | * imageSprite - Location of the fly sprite sheet. (default: 'fly-sprite.png') 85 | * bugWidth - The width of the fly sprite cell, and also div width. (default: 13) 86 | * bugHeight - The height of the fly sprite cell, and also div height. (default: 14) 87 | * num_frames - The number of frames in the sprite walk animation. (default: 5) 88 | * monitorMouseMovement - If enabled, a mousemove event will be added to the window, and used to detect if the cursor is near a fly. Probably best to leave this one off. (default: false) 89 | * eventDistanceToBug - If monitorMouseMovemenet is enabled, this is the distance from the bug in pixels which will trigger the near bug event. (default: 40) 90 | * minTimeBetweenMultipy - When in 'multiply' mode, this is the minimum time in ms between a multiply event. (default: 1000) 91 | * mouseOver - What to do when the mouse is over (or near) a fly. Can be 'fly', 'flyoff' (if we the bug canFly), 'die', 'multiply', or 'random'. See Modes. (default: random) 92 | * canFly - Whether or not to allow fly modes, and to use wings open animation (second row of sprite, default: true). 93 | * canDie - Whether or not to allow the bug to 'die' - need bottom row of sprite with dead version. (default: true) 94 | * zoom - Minimum amount to scale the bug, out of 10. So a zoom of 5 would randomly choose a zoom (css scale) value between 1/2 and 1 for each bug size. (default: 10 - no zooming) 95 | * maxLargeTurnDeg - When making a large turn, the maximum number of degrees to turn. (default: 150) 96 | * maxSmallTurnDeg - When making a smaller turn, the maximum number of degrees to turn. (default: 10) 97 | * maxWiggleDeg - When wiggling around the screen, this is the maximum number of degrees to turn. (default: 5), 98 | 99 | Modes 100 | ----- 101 | 102 | * random: Randomly pick one of the other modes on each mouse over/near event 103 | * fly: The bug will fly away to another random point on the page (if 'canFly' is true) 104 | * flyoff: The bug will fly off the screen... and slowly work its way back (if 'canFly' is true) 105 | * multiply: The bug will spawn a new bug and both will fly away to other parts of the page 106 | * nothing: Do nothing 107 | * die: The bug will be struck dead, and fall to the bottom of the page 108 | 109 | Credits 110 | ------- 111 | 112 | Original Screen Bug http://screen-bug.googlecode.com/git/index.html 113 | 114 | Released under WTFPL license. 115 | -------------------------------------------------------------------------------- /bower.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "bug", 3 | "main": "bug-min.js", 4 | "homepage": "https://github.com/Auz/Bug", 5 | "description": "Native JS library which adds bugs and/or spiders to walk around your window", 6 | "moduleType": [ 7 | "globals" 8 | ], 9 | "keywords": [ 10 | "bug", 11 | "spider", 12 | "fly", 13 | "flies", 14 | "fun", 15 | "creepy" 16 | ], 17 | "authors": [ 18 | "Graham McNicoll" 19 | ], 20 | "license": "MIT", 21 | "ignore": [ 22 | "**/.*", 23 | "node_modules", 24 | "bower_components", 25 | "test", 26 | "tests" 27 | ] 28 | } 29 | -------------------------------------------------------------------------------- /bug-min.js: -------------------------------------------------------------------------------- 1 | /* 2 | Bug.js - https://github.com/Auz/Bug 3 | Released under MIT-style license. 4 | Original Screen Bug http://screen-bug.googlecode.com/git/index.html 5 | */ 6 | var BugDispatch={options:{minDelay:500,maxDelay:1E4,minBugs:2,maxBugs:20,minSpeed:5,maxSpeed:10,maxLargeTurnDeg:150,maxSmallTurnDeg:10,maxWiggleDeg:5,imageSprite:"fly-sprite.png",bugWidth:13,bugHeight:14,num_frames:5,zoom:10,canFly:!0,canDie:!0,numDeathTypes:3,monitorMouseMovement:!1,eventDistanceToBug:40,minTimeBetweenMultipy:1E3,mouseOver:"random"},initialize:function(a){this.options=mergeOptions(this.options,a);this.options.minBugs>this.options.maxBugs&&(this.options.minBugs=this.options.maxBugs); 7 | this.modes=["multiply","nothing"];this.options.canFly&&this.modes.push("fly","flyoff");this.options.canDie&&this.modes.push("die");-1==this.modes.indexOf(this.options.mouseOver)&&(this.options.mouseOver="random");this.transform=null;this.transforms={Moz:function(a){this.bug.style.MozTransform=a},webkit:function(a){this.bug.style.webkitTransform=a},O:function(a){this.bug.style.OTransform=a},ms:function(a){this.bug.style.msTransform=a},Khtml:function(a){this.bug.style.KhtmlTransform=a},w3c:function(a){this.bug.style.transform= 8 | a}};if("transform"in document.documentElement.style)this.transform=this.transforms.w3c;else{var b=["Moz","webkit","O","ms","Khtml"],c=0;for(c=0;cb?d=b:dc||(200=--this.toggle_stationary_counter&&this.toggleStationary(),this.stationary))){if(0>=--this.edge_test_counter&&this.bug_near_window_edge()&&(this.angle_deg%=360,0>this.angle_deg&&(this.angle_deg+=360),15=--this.large_turn_counter&&(this.large_turn_angle_deg=this.random(1,this.options.maxLargeTurnDeg,!0),this.next_large_turn());if(0>=--this.small_turn_counter)this.angle_deg+=this.random(1,this.options.maxSmallTurnDeg),this.next_small_turn();else{a=this.random(1,this.options.maxWiggleDeg,!0);if(0a||0>this.large_turn_angle_deg&&0=this.options.num_frames&&(this.walkIndex=0)},fly:function(a){var b=this.bug.top,c=this.bug.left,d=c-a.left,e=b-a.top,f=Math.atan(e/d);50>Math.abs(d)+Math.abs(e)&&(this.bug.style.backgroundPosition=-2*this.options.bugWidth+ 26 | "px -"+2*this.options.bugHeight+"px");30>Math.abs(d)+Math.abs(e)&&(this.bug.style.backgroundPosition=-1*this.options.bugWidth+"px -"+2*this.options.bugHeight+"px");if(10>Math.abs(d)+Math.abs(e))this.bug.style.backgroundPosition="0 0",this.stop(),this.go();else{var g=Math.cos(f)*this.options.flySpeed;f=Math.sin(f)*this.options.flySpeed;if(c>a.left&&0a.left&&0>g)g*=-1,Math.abs(d)f||b>a.top&&0a&&(a=0);0===a?(a=-2*this.options.bugHeight,b*=Math.random()):1===a?(a=Math.random()*c,b+=2*this.options.bugWidth): 29 | 2===a?(a=c+2*this.options.bugHeight,b*=Math.random()):(a=Math.random()*c,b=-3*this.options.bugWidth);this.bug.style.backgroundPosition=-3*this.options.bugWidth+"px "+(this.wingsOpen?"0":"-"+this.options.bugHeight+"px");this.bug.top=a;this.bug.left=b;this.drawBug();a={};a.top=this.random(this.options.edge_resistance,document.documentElement.clientHeight-this.options.edge_resistance);a.left=this.random(this.options.edge_resistance,document.documentElement.clientWidth-this.options.edge_resistance);this.startFlying(a)}}, 30 | walkIn:function(){this.bug||this.makeBug();if(this.bug){this.stop();var a=Math.round(4*Math.random()-.5),b=document,c=b.documentElement,d=b.getElementsByTagName("body")[0];b=window.innerWidth||c.clientWidth||d.clientWidth;c=window.innerHeight||c.clientHeight||d.clientHeight;3a&&(a=0);0===a?(a=-1.3*this.options.bugHeight,b*=Math.random()):1===a?(a=Math.random()*c,b+=.3*this.options.bugWidth):2===a?(a=c+.3*this.options.bugHeight,b*=Math.random()):(a=Math.random()*c,b=-1.3*this.options.bugWidth); 31 | this.bug.style.backgroundPosition=-3*this.options.bugWidth+"px "+(this.wingsOpen?"0":"-"+this.options.bugHeight+"px");this.bug.top=a;this.bug.left=b;this.drawBug();this.go()}},flyOff:function(){this.stop();var a=this.random(0,3),b={},c=document,d=c.documentElement,e=c.getElementsByTagName("body")[0];c=window.innerWidth||d.clientWidth||e.clientWidth;d=window.innerHeight||d.clientHeight||e.clientHeight;0===a?(b.top=-200,b.left=Math.random()*c):1===a?(b.top=Math.random()*d,b.left=c+200):2===a?(b.top= 32 | d+200,b.left=Math.random()*c):(b.top=Math.random()*d,b.left=-200);this.startFlying(b)},die:function(){this.stop();var a=this.random(0,this.options.numDeathTypes-1);this.alive=!1;this.drop(a)},drop:function(a){var b=this.bug.top,c=document,d=c.documentElement;c=c.getElementsByTagName("body")[0];var e=window.innerHeight||d.clientHeight||c.clientHeight;e-=this.options.bugHeight;var f=this.random(0,20,!0);Date.now();var g=this;this.bug.classList.add("bug-dead");this.dropTimer=requestAnimFrame(function(c){g._lastTimestamp= 33 | c;g.dropping(c,b,e,f,a)})},dropping:function(a,b,c,d,e){a-=this._lastTimestamp;var f=b+.002*a*a,g=this;f>=c?(f=c,clearTimeout(this.dropTimer),this.angle_deg=0,this.angle_rad=this.deg2rad(this.angle_deg),this.transform("rotate("+(90-this.angle_deg)+"deg) scale("+this.zoom+")"),this.bug.style.top=null,this.bug.style.bottom=Math.ceil((this.options.bugWidth*this.zoom-this.options.bugHeight*this.zoom)/2-this.options.bugHeight/2*(1-this.zoom))+"px",this.bug.style.left=this.bug.left+"px",this.bug.style.backgroundPosition= 34 | "-"+2*e*this.options.bugWidth+"px 100%",this.twitch(e)):(this.dropTimer=requestAnimFrame(function(a){g.dropping(a,b,c,d,e)}),20>a||(this.angle_deg=(this.angle_deg+d)%360,this.angle_rad=this.deg2rad(this.angle_deg),this.moveBug(this.bug.left,f,this.angle_deg)))},twitch:function(a,b){b||(b=0);var c=this;if(0===a||1===a)c.twitchTimer=setTimeout(function(){c.bug.style.backgroundPosition="-"+(2*a+b%2)*c.options.bugWidth+"px 100%";c.twitchTimer=setTimeout(function(){b++;c.bug.style.backgroundPosition="-"+ 35 | (2*a+b%2)*c.options.bugWidth+"px 100%";c.twitch(a,++b)},c.random(300,800))},this.random(1E3,1E4))},rad2deg:function(a){return a*this.rad2deg_k},deg2rad:function(a){return a*this.deg2rad_k},random:function(a,b,c){if(a==b)return a;a=Math.round(a-.5+Math.random()*(b-a+1));return c?.5document.documentElement.clientHeight-this.options.edge_resistance&&(this.near_edge|=this.NEAR_BOTTOM_EDGE);this.bug.leftdocument.documentElement.clientWidth-this.options.edge_resistance&&(this.near_edge|=this.NEAR_RIGHT_EDGE);return this.near_edge},getPos:function(){return this.inserted&& 37 | this.bug&&this.bug.style?{top:parseInt(this.bug.top,10),left:parseInt(this.bug.left,10)}:null}},SpawnBug=function(){var a={},b;for(b in Bug)Bug.hasOwnProperty(b)&&(a[b]=Bug[b]);return a},mergeOptions=function(a,b,c){"undefined"==typeof c&&(c=!0);a=c?cloneOf(a):a;for(var d in b)b.hasOwnProperty(d)&&(a[d]=b[d]);return a},cloneOf=function(a){if(null==a||"object"!=typeof a)return a;var b=a.constructor(),c;for(c in a)a.hasOwnProperty(c)&&(b[c]=cloneOf(a[c]));return b}; 38 | window.requestAnimFrame=function(){return window.requestAnimationFrame||window.webkitRequestAnimationFrame||window.mozRequestAnimationFrame||window.oRequestAnimationFrame||window.msRequestAnimationFrame||function(a,b){window.setTimeout(a,1E3/60)}}(); 39 | -------------------------------------------------------------------------------- /bug.js: -------------------------------------------------------------------------------- 1 | // ==ClosureCompiler== 2 | // @compilation_level SIMPLE_OPTIMIZATIONS 3 | // @output_file_name bug-min.js 4 | // ==/ClosureCompiler== 5 | /** 6 | * @preserve Bug.js - https://github.com/Auz/Bug 7 | * Released under MIT-style license. 8 | * Original Screen Bug http://screen-bug.googlecode.com/git/index.html 9 | */ 10 | /** 11 | * Bug.js - Add bugs to your page 12 | * 13 | * https://github.com/Auz/Bug 14 | * 15 | * license: MIT-style license. 16 | * copyright: Copyright (c) 2016 Graham McNicoll 17 | * 18 | * 19 | * Created for an aprils fool joke at Education.com 2013. I knew there was probably a script 20 | * that did it already, and there was: http://screen-bug.googlecode.com/git/index.html. 21 | * I used this as the starting point and heavily modified it, used sprite image animation, 22 | * and added many new features. 23 | * 24 | * 25 | * Original Screen Bug http://screen-bug.googlecode.com/git/index.html 26 | * Copyright ©2011 Kernc (kerncece ^_^ gmail) 27 | * Released under WTFPL license. 28 | * 29 | */ 30 | "use strict"; 31 | 32 | 33 | var BugDispatch = { 34 | 35 | options: { 36 | minDelay: 500, 37 | maxDelay: 10000, 38 | minBugs: 2, 39 | maxBugs: 20, 40 | minSpeed: 5, 41 | maxSpeed: 10, 42 | maxLargeTurnDeg: 150, 43 | maxSmallTurnDeg: 10, 44 | maxWiggleDeg: 5, 45 | imageSprite: 'fly-sprite.png', 46 | bugWidth: 13, 47 | bugHeight: 14, 48 | num_frames: 5, 49 | zoom: 10, // random zoom variation from 1 to 10 - 10 being full size. 50 | canFly: true, 51 | canDie: true, 52 | numDeathTypes: 3, 53 | monitorMouseMovement: false, 54 | eventDistanceToBug: 40, 55 | minTimeBetweenMultipy: 1000, 56 | mouseOver: 'random' // can be 'fly', 'flyoff' (if the bug can fly), die', 'multiply', 'nothing' or 'random' 57 | }, 58 | 59 | initialize: function(options) { 60 | 61 | this.options = mergeOptions(this.options, options); 62 | 63 | // sanity check: 64 | if (this.options.minBugs > this.options.maxBugs) { 65 | this.options.minBugs = this.options.maxBugs; 66 | } 67 | 68 | this.modes = ['multiply', 'nothing']; 69 | 70 | if (this.options.canFly) { 71 | this.modes.push('fly', 'flyoff'); 72 | } 73 | if (this.options.canDie) { 74 | this.modes.push('die'); 75 | } 76 | 77 | if (this.modes.indexOf(this.options.mouseOver) == -1) { 78 | // invalid mode: use random: 79 | this.options.mouseOver = 'random'; 80 | } 81 | 82 | // can we transform? 83 | this.transform = null; 84 | 85 | this.transforms = { 86 | 'Moz': function(s) { 87 | this.bug.style.MozTransform = s; 88 | }, 89 | 'webkit': function(s) { 90 | this.bug.style.webkitTransform = s; 91 | }, 92 | 'O': function(s) { 93 | this.bug.style.OTransform = s; 94 | }, 95 | 'ms': function(s) { 96 | this.bug.style.msTransform = s; 97 | }, 98 | 'Khtml': function(s) { 99 | this.bug.style.KhtmlTransform = s; 100 | }, 101 | 'w3c': function(s) { 102 | this.bug.style.transform = s; 103 | } 104 | }; 105 | 106 | 107 | // check to see if it is a modern browser: 108 | 109 | if ('transform' in document.documentElement.style) { 110 | this.transform = this.transforms.w3c; 111 | } else { 112 | 113 | // feature detection for the other transforms: 114 | var vendors = ['Moz', 'webkit', 'O', 'ms', 'Khtml'], 115 | i = 0; 116 | 117 | for (i = 0; i < vendors.length; i++) { 118 | if (vendors[i] + 'Transform' in document.documentElement.style) { 119 | this.transform = this.transforms[vendors[i]]; 120 | break; 121 | } 122 | } 123 | } 124 | 125 | // dont support transforms... quit 126 | if (!this.transform) return; 127 | 128 | // make bugs: 129 | this.bugs = []; 130 | var numBugs = (this.options.mouseOver === 'multiply') ? this.options.minBugs : this.random(this.options.minBugs, this.options.maxBugs, true), 131 | i = 0, 132 | that = this; 133 | 134 | for (i = 0; i < numBugs; i++) { 135 | var options = JSON.parse(JSON.stringify(this.options)), 136 | b = SpawnBug(); 137 | 138 | options.wingsOpen = (this.options.canFly) ? ((Math.random() > 0.5) ? true : false) : true, 139 | options.walkSpeed = this.random(this.options.minSpeed, this.options.maxSpeed), 140 | 141 | b.initialize(this.transform, options); 142 | this.bugs.push(b); 143 | } 144 | 145 | // fly them in staggered: 146 | this.spawnDelay = []; 147 | for (i = 0; i < numBugs; i++) { 148 | var delay = this.random(this.options.minDelay, this.options.maxDelay, true), 149 | thebug = this.bugs[i]; 150 | // fly the bug onto the page: 151 | this.spawnDelay[i] = setTimeout((function(thebug) { 152 | return function() { 153 | if (that.options.canFly) { 154 | thebug.flyIn(); 155 | } else { 156 | thebug.walkIn(); 157 | } 158 | 159 | }; 160 | }(thebug)), delay); 161 | 162 | // add mouse over events: 163 | that.add_events_to_bug(thebug); 164 | } 165 | 166 | // add window event if required: 167 | if (this.options.monitorMouseMovement) { 168 | window.onmousemove = function() { 169 | that.check_if_mouse_close_to_bug(); 170 | }; 171 | } 172 | 173 | }, 174 | 175 | stop: function() { 176 | for (var i = 0; i < this.bugs.length; i++) { 177 | if(this.spawnDelay[i]) clearTimeout(this.spawnDelay[i]); 178 | this.bugs[i].stop(); 179 | } 180 | }, 181 | 182 | end: function() { 183 | for (var i = 0; i < this.bugs.length; i++) { 184 | if(this.spawnDelay[i]) clearTimeout(this.spawnDelay[i]); 185 | this.bugs[i].stop(); 186 | this.bugs[i].remove(); 187 | } 188 | }, 189 | 190 | reset: function() { 191 | this.stop(); 192 | for (var i = 0; i < this.bugs.length; i++) { 193 | this.bugs[i].reset(); 194 | this.bugs[i].walkIn(); 195 | } 196 | }, 197 | 198 | killAll: function() { 199 | for (var i = 0; i < this.bugs.length; i++) { 200 | if(this.spawnDelay[i]) clearTimeout(this.spawnDelay[i]); 201 | this.bugs[i].die(); 202 | } 203 | }, 204 | 205 | add_events_to_bug: function(thebug) { 206 | var that = this; 207 | if (thebug.bug) { 208 | if (thebug.bug.addEventListener) { 209 | thebug.bug.addEventListener('mouseover', function(e) { 210 | that.on_bug(thebug); 211 | }); 212 | } else if (thebug.bug.attachEvent) { 213 | thebug.bug.attachEvent('onmouseover', function(e) { 214 | that.on_bug(thebug); 215 | }); 216 | } 217 | } 218 | }, 219 | 220 | check_if_mouse_close_to_bug: function(e) { 221 | e = e || window.event; 222 | if (!e) { 223 | return; 224 | } 225 | 226 | var posx = 0, 227 | posy = 0; 228 | if (e.client && e.client.x) { 229 | posx = e.client.x; 230 | posy = e.client.y; 231 | } else if (e.clientX) { 232 | posx = e.clientX; 233 | posy = e.clientY; 234 | } else if (e.page && e.page.x) { 235 | posx = e.page.x - (document.body.scrollLeft + document.documentElement.scrollLeft); 236 | posy = e.page.y - (document.body.scrollTop + document.documentElement.scrollTop); 237 | } else if (e.pageX) { 238 | posx = e.pageX - (document.body.scrollLeft + document.documentElement.scrollLeft); 239 | posy = e.pageY - (document.body.scrollTop + document.documentElement.scrollTop); 240 | } 241 | var numBugs = this.bugs.length, 242 | i = 0; 243 | for (i = 0; i < numBugs; i++) { 244 | var pos = this.bugs[i].getPos(); 245 | if (pos) { 246 | if (Math.abs(pos.top - posy) + Math.abs(pos.left - posx) < this.options.eventDistanceToBug && !this.bugs[i].flyperiodical) { 247 | this.near_bug(this.bugs[i]); 248 | } 249 | } 250 | } 251 | 252 | }, 253 | 254 | near_bug: function(bug) { 255 | this.on_bug(bug); 256 | }, 257 | 258 | on_bug: function(bug) { 259 | if (!bug.alive) { 260 | return; 261 | } 262 | 263 | var mode = this.options.mouseOver; 264 | 265 | if (mode === 'random') { 266 | mode = this.modes[(this.random(0, (this.modes.length - 1), true))]; 267 | } 268 | 269 | if (mode === 'fly') { 270 | // fly away! 271 | bug.stop(); 272 | bug.flyRand(); 273 | } else if (mode === 'nothing') { 274 | return; 275 | } else if (mode === 'flyoff') { 276 | // fly away and off the page 277 | bug.stop(); 278 | bug.flyOff(); 279 | } else if (mode === 'die') { 280 | // drop dead! 281 | bug.die(); 282 | } else if (mode === 'multiply') { 283 | if (!this.multiplyDelay && this.bugs.length < this.options.maxBugs) { 284 | // spawn another: 285 | // create new bug: 286 | var b = SpawnBug(), 287 | options = JSON.parse(JSON.stringify(this.options)), 288 | pos = bug.getPos(), 289 | that = this; 290 | 291 | options.wingsOpen = (this.options.canFly) ? ((Math.random() > 0.5) ? true : false) : true; 292 | options.walkSpeed = this.random(this.options.minSpeed, this.options.maxSpeed); 293 | 294 | b.initialize(this.transform, options); 295 | b.drawBug(pos.top, pos.left); 296 | // fly them both away: 297 | if (options.canFly) { 298 | b.flyRand(); 299 | bug.flyRand(); 300 | } else { 301 | b.go(); 302 | bug.go(); 303 | } 304 | // store new bug: 305 | this.bugs.push(b); 306 | // watch out for spawning too quickly: 307 | this.multiplyDelay = true; 308 | setTimeout(function() { 309 | // add event to this bug: 310 | that.add_events_to_bug(b); 311 | that.multiplyDelay = false; 312 | }, this.options.minTimeBetweenMultipy); 313 | } 314 | 315 | } 316 | }, 317 | 318 | random: function(min, max, round) { 319 | if (min == max) return ((round) ? Math.round(min) : min); 320 | 321 | var result = ((min - 0.5) + (Math.random() * (max - min + 1))); 322 | if (result > max) { 323 | result = max; 324 | } else if (result < min) { 325 | result = min; 326 | } 327 | return ((round) ? Math.round(result) : result); 328 | } 329 | 330 | 331 | }; 332 | 333 | var BugController = function() { 334 | this.initialize.apply(this, arguments); 335 | } 336 | BugController.prototype = BugDispatch; 337 | 338 | var SpiderController = function() { 339 | var spiderOptions = { 340 | imageSprite: 'spider-sprite.png', 341 | bugWidth: 69, 342 | bugHeight: 90, 343 | num_frames: 7, 344 | canFly: false, 345 | canDie: true, 346 | numDeathTypes: 2, 347 | zoom: 6, 348 | minDelay: 200, 349 | maxDelay: 3000, 350 | minSpeed: 6, 351 | maxSpeed: 13, 352 | minBugs: 3, 353 | maxBugs: 10 354 | }; 355 | this.options = mergeOptions(this.options, spiderOptions); 356 | this.initialize.apply(this, arguments); 357 | 358 | } 359 | SpiderController.prototype = BugDispatch; 360 | 361 | /***************/ 362 | /** Bug **/ 363 | /***************/ 364 | 365 | var Bug = { 366 | 367 | options: { 368 | wingsOpen: false, 369 | walkSpeed: 2, 370 | flySpeed: 40, 371 | edge_resistance: 50, 372 | zoom: 10 373 | 374 | }, 375 | 376 | initialize: function(transform, options) { 377 | 378 | this.options = mergeOptions(this.options, options); 379 | 380 | this.NEAR_TOP_EDGE = 1; 381 | this.NEAR_BOTTOM_EDGE = 2; 382 | this.NEAR_LEFT_EDGE = 4; 383 | this.NEAR_RIGHT_EDGE = 8; 384 | this.directions = {}; // 0 degrees starts on the East 385 | this.directions[this.NEAR_TOP_EDGE] = 270; 386 | this.directions[this.NEAR_BOTTOM_EDGE] = 90; 387 | this.directions[this.NEAR_LEFT_EDGE] = 0; 388 | this.directions[this.NEAR_RIGHT_EDGE] = 180; 389 | this.directions[this.NEAR_TOP_EDGE + this.NEAR_LEFT_EDGE] = 315; 390 | this.directions[this.NEAR_TOP_EDGE + this.NEAR_RIGHT_EDGE] = 225; 391 | this.directions[this.NEAR_BOTTOM_EDGE + this.NEAR_LEFT_EDGE] = 45; 392 | this.directions[this.NEAR_BOTTOM_EDGE + this.NEAR_RIGHT_EDGE] = 135; 393 | 394 | this.angle_deg = 0; 395 | this.angle_rad = 0; 396 | this.large_turn_angle_deg = 0; 397 | this.near_edge = false; 398 | this.edge_test_counter = 10; 399 | this.small_turn_counter = 0; 400 | this.large_turn_counter = 0; 401 | this.fly_counter = 0; 402 | this.toggle_stationary_counter = Math.random() * 50; 403 | this.zoom = this.random(this.options.zoom, 10) / 10; 404 | 405 | this.stationary = false; 406 | this.bug = null; 407 | this.active = true; 408 | this.wingsOpen = this.options.wingsOpen; 409 | this.transform = transform; 410 | this.walkIndex = 0; 411 | this.flyIndex = 0; 412 | this.alive = true; 413 | this.twitchTimer = null; 414 | 415 | this.rad2deg_k = 180 / Math.PI; 416 | this.deg2rad_k = Math.PI / 180; 417 | 418 | this.makeBug(); 419 | 420 | this.angle_rad = this.deg2rad(this.angle_deg); 421 | 422 | this.angle_deg = this.random(0, 360, true); 423 | 424 | }, 425 | 426 | go: function() { 427 | if (this.transform) { 428 | this.drawBug(); 429 | var that = this; 430 | 431 | this.animating = true; 432 | 433 | this.going = requestAnimFrame(function(t) { 434 | that.animate(t); 435 | }); 436 | } 437 | }, 438 | 439 | stop: function() { 440 | this.animating = false; 441 | if (this.going) { 442 | clearTimeout(this.going); 443 | this.going = null; 444 | } 445 | if (this.flyperiodical) { 446 | clearTimeout(this.flyperiodical); 447 | this.flyperiodical = null; 448 | } 449 | if (this.twitchTimer) { 450 | clearTimeout(this.twitchTimer); 451 | this.twitchTimer = null; 452 | } 453 | }, 454 | 455 | remove: function() { 456 | this.active = false; 457 | if (this.inserted && this.bug.parentNode) { 458 | this.bug.parentNode.removeChild(this.bug); 459 | this.inserted = false; 460 | } 461 | }, 462 | 463 | reset: function() { 464 | this.alive = true; 465 | this.active = true; 466 | this.bug.style.bottom = ''; 467 | this.bug.style.top = 0; 468 | this.bug.style.left = 0; 469 | this.bug.classList.remove('bug-dead'); 470 | }, 471 | 472 | animate: function(t) { 473 | 474 | if (!this.animating || !this.alive || !this.active) return; 475 | 476 | var that = this; 477 | this.going = requestAnimFrame(function(t) { 478 | that.animate(t); 479 | }); 480 | 481 | if (!('_lastTimestamp' in this)) this._lastTimestamp = t; 482 | 483 | var delta = t - this._lastTimestamp; 484 | 485 | if (delta < 40) return; // don't animate too frequently 486 | 487 | // sometimes if the browser doesnt have focus, or the delta in request animation 488 | // frame can be very large. We set a sensible max so that the bugs dont spaz out. 489 | 490 | if (delta > 200) delta = 200; 491 | 492 | this._lastTimestamp = t; 493 | 494 | if (--this.toggle_stationary_counter <= 0) { 495 | this.toggleStationary(); 496 | } 497 | if (this.stationary) { 498 | return; 499 | } 500 | 501 | 502 | if (--this.edge_test_counter <= 0 && this.bug_near_window_edge()) { 503 | // if near edge, go away from edge 504 | this.angle_deg %= 360; 505 | if (this.angle_deg < 0) this.angle_deg += 360; 506 | 507 | if (Math.abs(this.directions[this.near_edge] - this.angle_deg) > 15) { 508 | var angle1 = this.directions[this.near_edge] - this.angle_deg; 509 | var angle2 = (360 - this.angle_deg) + this.directions[this.near_edge]; 510 | this.large_turn_angle_deg = (Math.abs(angle1) < Math.abs(angle2) ? angle1 : angle2); 511 | 512 | this.edge_test_counter = 10; 513 | this.large_turn_counter = 100; 514 | this.small_turn_counter = 30; 515 | } 516 | } 517 | if (--this.large_turn_counter <= 0) { 518 | this.large_turn_angle_deg = this.random(1, this.options.maxLargeTurnDeg, true); 519 | this.next_large_turn(); 520 | } 521 | if (--this.small_turn_counter <= 0) { 522 | this.angle_deg += this.random(1, this.options.maxSmallTurnDeg); 523 | this.next_small_turn(); 524 | } else { 525 | var dangle = this.random(1, this.options.maxWiggleDeg, true); 526 | if ((this.large_turn_angle_deg > 0 && dangle < 0) || (this.large_turn_angle_deg < 0 && dangle > 0)) { 527 | dangle = -dangle; // ensures both values either + or - 528 | } 529 | this.large_turn_angle_deg -= dangle; 530 | this.angle_deg += dangle; 531 | } 532 | 533 | this.angle_rad = this.deg2rad(this.angle_deg); 534 | 535 | var dx = Math.cos(this.angle_rad) * this.options.walkSpeed * (delta / 100); 536 | var dy = -Math.sin(this.angle_rad) * this.options.walkSpeed * (delta / 100); 537 | 538 | this.moveBug((this.bug.left + dx), (this.bug.top + dy), (90 - this.angle_deg)); 539 | this.walkFrame(); 540 | 541 | }, 542 | 543 | makeBug: function() { 544 | if (!this.bug && this.active) { 545 | var row = (this.wingsOpen) ? '0' : '-' + this.options.bugHeight + 'px', 546 | bug = document.createElement('div'); 547 | bug.className = 'bug'; 548 | bug.style.background = 'transparent url(' + this.options.imageSprite + ') no-repeat 0 ' + row; 549 | bug.style.width = this.options.bugWidth + 'px'; 550 | bug.style.height = this.options.bugHeight + 'px'; 551 | bug.style.position = 'fixed'; 552 | bug.style.top = 0; 553 | bug.style.left = 0; 554 | bug.style.zIndex = '9999999'; 555 | 556 | this.bug = bug; 557 | this.setPos(); 558 | 559 | } 560 | 561 | }, 562 | 563 | setPos: function(top, left) { 564 | this.bug.top = top || this.random(this.options.edge_resistance, document.documentElement.clientHeight - this.options.edge_resistance); 565 | 566 | this.bug.left = left || this.random(this.options.edge_resistance, document.documentElement.clientWidth - this.options.edge_resistance); 567 | 568 | this.moveBug(this.bug.left, this.bug.top, (90 - this.angle_deg)); 569 | }, 570 | 571 | moveBug: function(x, y, deg) { 572 | // keep track of where we are: 573 | this.bug.left = x; 574 | this.bug.top = y; 575 | 576 | // transform: 577 | var trans = "translate(" + parseInt(x) + "px," + parseInt(y) + "px)"; 578 | if (deg) { 579 | //console.log("translate("+(x)+"px, "+(y)+"px) rotate("+deg+"deg)"); 580 | trans += " rotate(" + deg + "deg)"; 581 | } 582 | trans += " scale(" + this.zoom + ")"; 583 | 584 | this.transform(trans); 585 | 586 | }, 587 | 588 | drawBug: function(top, left) { 589 | 590 | if (!this.bug) { 591 | this.makeBug(); 592 | } 593 | if(!this.bug) return; 594 | 595 | if (top && left) { 596 | this.setPos(top, left); 597 | } else { 598 | this.setPos(this.bug.top, this.bug.left) 599 | } 600 | if (!this.inserted) { 601 | this.inserted = true; 602 | document.body.appendChild(this.bug); 603 | } 604 | }, 605 | 606 | toggleStationary: function() { 607 | this.stationary = !this.stationary; 608 | this.next_stationary(); 609 | var ypos = (this.wingsOpen) ? '0' : '-' + this.options.bugHeight + 'px'; 610 | if (this.stationary) { 611 | 612 | this.bug.style.backgroundPosition = '0 ' + ypos; 613 | } else { 614 | this.bug.style.backgroundPosition = '-' + this.options.bugWidth + 'px ' + ypos; 615 | } 616 | }, 617 | 618 | walkFrame: function() { 619 | var xpos = (-1 * (this.walkIndex * this.options.bugWidth)) + 'px', 620 | ypos = (this.wingsOpen) ? '0' : '-' + this.options.bugHeight + 'px'; 621 | this.bug.style.backgroundPosition = xpos + ' ' + ypos; 622 | this.walkIndex++; 623 | if (this.walkIndex >= this.options.num_frames) this.walkIndex = 0; 624 | }, 625 | 626 | fly: function(landingPosition) { 627 | var currentTop = this.bug.top, 628 | currentLeft = this.bug.left, 629 | diffx = (currentLeft - landingPosition.left), 630 | diffy = (currentTop - landingPosition.top), 631 | angle = Math.atan(diffy / diffx); 632 | 633 | if (Math.abs(diffx) + Math.abs(diffy) < 50) { 634 | this.bug.style.backgroundPosition = (-2 * this.options.bugWidth) + 'px -' + (2 * this.options.bugHeight) + 'px'; 635 | } 636 | if (Math.abs(diffx) + Math.abs(diffy) < 30) { 637 | this.bug.style.backgroundPosition = (-1 * this.options.bugWidth) + 'px -' + (2 * this.options.bugHeight) + 'px'; 638 | } 639 | if (Math.abs(diffx) + Math.abs(diffy) < 10) { 640 | // close enough: 641 | this.bug.style.backgroundPosition = '0 0'; //+row+'px')); 642 | 643 | this.stop(); 644 | this.go(); 645 | //this.go.delay(100, this); 646 | 647 | return; 648 | 649 | } 650 | 651 | // make it wiggle: disabled becuase its just too fast to see... better would be to make its path wiggly. 652 | //angle = angle - (this.deg2rad(this.random(0,10))); 653 | //console.log('angle: ',this.rad2deg(angle)); 654 | 655 | var dx = Math.cos(angle) * this.options.flySpeed, 656 | dy = Math.sin(angle) * this.options.flySpeed; 657 | 658 | if ((currentLeft > landingPosition.left && dx > 0) || (currentLeft > landingPosition.left && dx < 0)) { 659 | // make sure angle is right way 660 | dx = -1 * dx; 661 | if (Math.abs(diffx) < Math.abs(dx)) { 662 | dx = dx / 4; 663 | } 664 | } 665 | if ((currentTop < landingPosition.top && dy < 0) || (currentTop > landingPosition.top && dy > 0)) { 666 | dy = -1 * dy; 667 | if (Math.abs(diffy) < Math.abs(dy)) { 668 | dy = dy / 4; 669 | } 670 | } 671 | 672 | this.moveBug((currentLeft + dx), (currentTop + dy)); 673 | 674 | }, 675 | 676 | flyRand: function() { 677 | this.stop(); 678 | var landingPosition = {}; 679 | landingPosition.top = this.random(this.options.edge_resistance, document.documentElement.clientHeight - this.options.edge_resistance); 680 | landingPosition.left = this.random(this.options.edge_resistance, document.documentElement.clientWidth - this.options.edge_resistance); 681 | 682 | this.startFlying(landingPosition); 683 | }, 684 | 685 | startFlying: function(landingPosition) { 686 | 687 | var currentTop = this.bug.top, 688 | currentLeft = this.bug.left, 689 | diffx = (landingPosition.left - currentLeft), 690 | diffy = (landingPosition.top - currentTop); 691 | 692 | this.bug.left = landingPosition.left; 693 | this.bug.top = landingPosition.top; 694 | 695 | this.angle_rad = Math.atan(diffy / diffx); 696 | this.angle_deg = this.rad2deg(this.angle_rad); 697 | 698 | if (diffx > 0) { 699 | // going left: quadrant 1 or 2 700 | this.angle_deg = 90 + this.angle_deg; 701 | } else { 702 | // going right: quadrant 3 or 4 703 | this.angle_deg = 270 + this.angle_deg; 704 | } 705 | 706 | this.moveBug(currentLeft, currentTop, this.angle_deg); 707 | 708 | // start animation: 709 | var that = this; 710 | this.flyperiodical = setInterval(function() { 711 | that.fly(landingPosition); 712 | }, 10); 713 | }, 714 | 715 | flyIn: function() { 716 | if (!this.bug) { 717 | this.makeBug(); 718 | } 719 | 720 | if(!this.bug) return; 721 | 722 | this.stop(); 723 | // pick a random side: 724 | var side = Math.round(Math.random() * 4 - 0.5), 725 | d = document, 726 | e = d.documentElement, 727 | g = d.getElementsByTagName('body')[0], 728 | windowX = window.innerWidth || e.clientWidth || g.clientWidth, 729 | windowY = window.innerHeight || e.clientHeight || g.clientHeight; 730 | if (side > 3) side = 3; 731 | if (side < 0) side = 0; 732 | var style = {}, 733 | s; 734 | if (side === 0) { 735 | // top: 736 | style.top = (-2 * this.options.bugHeight); 737 | style.left = Math.random() * windowX; 738 | } else if (side === 1) { 739 | // right: 740 | style.top = Math.random() * windowY; 741 | style.left = windowX + (2 * this.options.bugWidth); 742 | } else if (side === 2) { 743 | // bottom: 744 | style.top = windowY + (2 * this.options.bugHeight); 745 | style.left = Math.random() * windowX; 746 | } else { 747 | // left: 748 | style.top = Math.random() * windowY; 749 | style.left = (-3 * this.options.bugWidth); 750 | } 751 | var row = (this.wingsOpen) ? '0' : '-' + this.options.bugHeight + 'px'; 752 | this.bug.style.backgroundPosition = (-3 * this.options.bugWidth) + 'px ' + row; 753 | this.bug.top = style.top 754 | this.bug.left = style.left 755 | 756 | this.drawBug(); 757 | 758 | // landing position: 759 | var landingPosition = {}; 760 | landingPosition.top = this.random(this.options.edge_resistance, document.documentElement.clientHeight - this.options.edge_resistance); 761 | landingPosition.left = this.random(this.options.edge_resistance, document.documentElement.clientWidth - this.options.edge_resistance); 762 | 763 | this.startFlying(landingPosition); 764 | }, 765 | 766 | walkIn: function() { 767 | if (!this.bug) { 768 | this.makeBug(); 769 | } 770 | 771 | if(!this.bug) return; 772 | 773 | this.stop(); 774 | // pick a random side: 775 | var side = Math.round(Math.random() * 4 - 0.5), 776 | d = document, 777 | e = d.documentElement, 778 | g = d.getElementsByTagName('body')[0], 779 | windowX = window.innerWidth || e.clientWidth || g.clientWidth, 780 | windowY = window.innerHeight || e.clientHeight || g.clientHeight; 781 | if (side > 3) side = 3; 782 | if (side < 0) side = 0; 783 | var style = {}, 784 | s; 785 | if (side === 0) { 786 | // top: 787 | style.top = (-1.3 * this.options.bugHeight); 788 | style.left = Math.random() * windowX; 789 | } else if (side === 1) { 790 | // right: 791 | style.top = Math.random() * windowY; 792 | style.left = windowX + (0.3 * this.options.bugWidth); 793 | } else if (side === 2) { 794 | // bottom: 795 | style.top = windowY + (0.3 * this.options.bugHeight); 796 | style.left = Math.random() * windowX; 797 | } else { 798 | // left: 799 | style.top = Math.random() * windowY; 800 | style.left = (-1.3 * this.options.bugWidth); 801 | } 802 | var row = (this.wingsOpen) ? '0' : '-' + this.options.bugHeight + 'px'; 803 | this.bug.style.backgroundPosition = (-3 * this.options.bugWidth) + 'px ' + row; 804 | this.bug.top = style.top 805 | this.bug.left = style.left 806 | 807 | this.drawBug(); 808 | 809 | // start walking: 810 | this.go(); 811 | 812 | }, 813 | 814 | flyOff: function() { 815 | this.stop(); 816 | // pick a random side to fly off to, where 0 is top and continuing clockwise. 817 | var side = this.random(0, 3), 818 | style = {}, 819 | d = document, 820 | e = d.documentElement, 821 | g = d.getElementsByTagName('body')[0], 822 | windowX = window.innerWidth || e.clientWidth || g.clientWidth, 823 | windowY = window.innerHeight || e.clientHeight || g.clientHeight; 824 | 825 | if (side === 0) { 826 | // top: 827 | style.top = -200; 828 | style.left = Math.random() * windowX; 829 | } else if (side === 1) { 830 | // right: 831 | style.top = Math.random() * windowY; 832 | style.left = windowX + 200; 833 | } else if (side === 2) { 834 | //bottom: 835 | style.top = windowY + 200; 836 | style.left = Math.random() * windowX; 837 | } else { 838 | // left: 839 | style.top = Math.random() * windowY; 840 | style.left = -200; 841 | } 842 | this.startFlying(style); 843 | }, 844 | 845 | die: function() { 846 | this.stop(); 847 | //pick death style: 848 | var deathType = this.random(0, this.options.numDeathTypes - 1); 849 | 850 | this.alive = false; 851 | this.drop(deathType); 852 | }, 853 | 854 | drop: function(deathType) { 855 | var startPos = this.bug.top, 856 | d = document, 857 | e = d.documentElement, 858 | g = d.getElementsByTagName('body')[0], 859 | finalPos = window.innerHeight || e.clientHeight || g.clientHeight, 860 | finalPos = finalPos - this.options.bugHeight, 861 | rotationRate = this.random(0, 20, true), 862 | startTime = Date.now(), 863 | that = this; 864 | 865 | this.bug.classList.add('bug-dead'); 866 | 867 | this.dropTimer = requestAnimFrame(function(t) { 868 | that._lastTimestamp = t; 869 | that.dropping(t, startPos, finalPos, rotationRate, deathType); 870 | }); 871 | 872 | }, 873 | 874 | dropping: function(t, startPos, finalPos, rotationRate, deathType) { 875 | var elapsedTime = t - this._lastTimestamp, 876 | deltaPos = (0.002 * (elapsedTime * elapsedTime)), 877 | newPos = startPos + deltaPos; 878 | //console.log(t, elapsedTime, deltaPos, newPos); 879 | 880 | var that = this; 881 | 882 | 883 | if (newPos >= finalPos) { 884 | newPos = finalPos; 885 | clearTimeout(this.dropTimer); 886 | 887 | 888 | 889 | this.angle_deg = 0; 890 | this.angle_rad = this.deg2rad(this.angle_deg); 891 | this.transform("rotate(" + (90 - this.angle_deg) + "deg) scale(" + this.zoom + ")"); 892 | this.bug.style.top = null; 893 | // because it is (or might be) zoomed and rotated, we cannot just just bottom = 0. Figure out real bottom position: 894 | var rotationOffset = ((this.options.bugWidth * this.zoom) - (this.options.bugHeight * this.zoom)) / 2; 895 | var zoomOffset = ((this.options.bugHeight) / 2) * (1 - this.zoom); 896 | this.bug.style.bottom = Math.ceil((rotationOffset - zoomOffset)) + 'px'; // because its rotated and zoomed. 897 | this.bug.style.left = this.bug.left + 'px'; 898 | this.bug.style.backgroundPosition = '-' + ((deathType * 2) * this.options.bugWidth) + 'px 100%'; 899 | 900 | 901 | this.twitch(deathType); 902 | 903 | return; 904 | } 905 | 906 | this.dropTimer = requestAnimFrame(function(t) { 907 | that.dropping(t, startPos, finalPos, rotationRate, deathType); 908 | }); 909 | 910 | if (elapsedTime < 20) return; 911 | 912 | this.angle_deg = ((this.angle_deg + rotationRate) % 360); 913 | this.angle_rad = this.deg2rad(this.angle_deg); 914 | 915 | this.moveBug(this.bug.left, newPos, this.angle_deg); 916 | }, 917 | 918 | twitch: function(deathType, legPos) { 919 | //this.bug.style.back 920 | if (!legPos) legPos = 0; 921 | var that = this; 922 | if (deathType === 0 || deathType === 1) { 923 | that.twitchTimer = setTimeout(function() { 924 | that.bug.style.backgroundPosition = '-' + ((deathType * 2 + (legPos % 2)) * that.options.bugWidth) + 'px 100%'; 925 | that.twitchTimer = setTimeout(function() { 926 | legPos++; 927 | that.bug.style.backgroundPosition = '-' + ((deathType * 2 + (legPos % 2)) * that.options.bugWidth) + 'px 100%'; 928 | that.twitch(deathType, ++legPos); 929 | }, that.random(300, 800)); 930 | }, this.random(1000, 10000)); 931 | } 932 | }, 933 | 934 | /* helper methods: */ 935 | rad2deg: function(rad) { 936 | return rad * this.rad2deg_k; 937 | }, 938 | deg2rad: function(deg) { 939 | return deg * this.deg2rad_k; 940 | }, 941 | random: function(min, max, plusminus) { 942 | if (min == max) return min; 943 | var result = Math.round(min - 0.5 + (Math.random() * (max - min + 1))); 944 | if (plusminus) return Math.random() > 0.5 ? result : -result; 945 | return result; 946 | }, 947 | 948 | next_small_turn: function() { 949 | this.small_turn_counter = Math.round(Math.random() * 10); 950 | }, 951 | next_large_turn: function() { 952 | this.large_turn_counter = Math.round(Math.random() * 40); 953 | }, 954 | next_stationary: function() { 955 | this.toggle_stationary_counter = this.random(50, 300); 956 | }, 957 | 958 | bug_near_window_edge: function() { 959 | this.near_edge = 0; 960 | if (this.bug.top < this.options.edge_resistance) 961 | this.near_edge |= this.NEAR_TOP_EDGE; 962 | else if (this.bug.top > document.documentElement.clientHeight - this.options.edge_resistance) 963 | this.near_edge |= this.NEAR_BOTTOM_EDGE; 964 | if (this.bug.left < this.options.edge_resistance) 965 | this.near_edge |= this.NEAR_LEFT_EDGE; 966 | else if (this.bug.left > document.documentElement.clientWidth - this.options.edge_resistance) 967 | this.near_edge |= this.NEAR_RIGHT_EDGE; 968 | return this.near_edge; 969 | }, 970 | 971 | getPos: function() { 972 | if (this.inserted && this.bug && this.bug.style) { 973 | return { 974 | 'top': parseInt(this.bug.top, 10), 975 | 'left': parseInt(this.bug.left, 10) 976 | }; 977 | } 978 | return null; 979 | } 980 | 981 | }; 982 | 983 | var SpawnBug = function() { 984 | var newBug = {}, 985 | prop; 986 | for (prop in Bug) { 987 | if (Bug.hasOwnProperty(prop)) { 988 | newBug[prop] = Bug[prop]; 989 | } 990 | } 991 | return newBug; 992 | }; 993 | 994 | // debated about which pattern to use to instantiate each bug... 995 | // see http://jsperf.com/obj-vs-prototype-vs-other 996 | 997 | 998 | 999 | /** 1000 | * Helper methods: 1001 | **/ 1002 | 1003 | var mergeOptions = function(obj1, obj2, clone) { 1004 | if (typeof(clone) == 'undefined') { 1005 | clone = true; 1006 | } 1007 | var newobj = (clone) ? cloneOf(obj1) : obj1; 1008 | for (var key in obj2) { 1009 | if (obj2.hasOwnProperty(key)) { 1010 | newobj[key] = obj2[key]; 1011 | } 1012 | } 1013 | return newobj; 1014 | }; 1015 | 1016 | var cloneOf = function(obj) { 1017 | if (obj == null || typeof(obj) != 'object') 1018 | return obj; 1019 | 1020 | var temp = obj.constructor(); // changed 1021 | 1022 | for (var key in obj) { 1023 | if (obj.hasOwnProperty(key)) { 1024 | temp[key] = cloneOf(obj[key]); 1025 | } 1026 | } 1027 | return temp; 1028 | } 1029 | 1030 | /* Request animation frame polyfill */ 1031 | /* http://paulirish.com/2011/requestanimationframe-for-smart-animating/ */ 1032 | window.requestAnimFrame = (function() { 1033 | return window.requestAnimationFrame || 1034 | window.webkitRequestAnimationFrame || 1035 | window.mozRequestAnimationFrame || 1036 | window.oRequestAnimationFrame || 1037 | window.msRequestAnimationFrame || function( /* function */ callback, /* DOMElement */ element) { 1038 | window.setTimeout(callback, 1000 / 60); 1039 | }; 1040 | })(); 1041 | -------------------------------------------------------------------------------- /chrome-extension/bug extension.crx: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Auz/Bug/8eac7b7337604fb3b8c8f0be1116c8de0984057e/chrome-extension/bug extension.crx -------------------------------------------------------------------------------- /chrome-extension/bug-icon-128.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Auz/Bug/8eac7b7337604fb3b8c8f0be1116c8de0984057e/chrome-extension/bug-icon-128.png -------------------------------------------------------------------------------- /chrome-extension/bug-icon-16.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Auz/Bug/8eac7b7337604fb3b8c8f0be1116c8de0984057e/chrome-extension/bug-icon-16.png -------------------------------------------------------------------------------- /chrome-extension/bug-icon-48.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Auz/Bug/8eac7b7337604fb3b8c8f0be1116c8de0984057e/chrome-extension/bug-icon-48.png -------------------------------------------------------------------------------- /chrome-extension/bug.js: -------------------------------------------------------------------------------- 1 | // ==ClosureCompiler== 2 | // @compilation_level SIMPLE_OPTIMIZATIONS 3 | // @output_file_name bug-min.js 4 | // ==/ClosureCompiler== 5 | /** 6 | * @preserve Bug.js - https://github.com/Auz/Bug 7 | * Released under MIT-style license. 8 | * Original Screen Bug http://screen-bug.googlecode.com/git/index.html 9 | */ 10 | /** 11 | * Bug.js - Add bugs to your page 12 | * 13 | * https://github.com/Auz/Bug 14 | * 15 | * license: MIT-style license. 16 | * copyright: Copyright (c) 2016 Graham McNicoll 17 | * 18 | * 19 | * Created for an aprils fool joke at Education.com 2013. I knew there was probably a script 20 | * that did it already, and there was: http://screen-bug.googlecode.com/git/index.html. 21 | * I used this as the starting point and heavily modified it, used sprite image animation, 22 | * and added many new features. 23 | * 24 | * 25 | * Original Screen Bug http://screen-bug.googlecode.com/git/index.html 26 | * Copyright ©2011 Kernc (kerncece ^_^ gmail) 27 | * Released under WTFPL license. 28 | * 29 | */ 30 | "use strict"; 31 | 32 | 33 | var BugDispatch = { 34 | 35 | options: { 36 | minDelay: 500, 37 | maxDelay: 10000, 38 | minBugs: 2, 39 | maxBugs: 20, 40 | minSpeed: 5, 41 | maxSpeed: 10, 42 | maxLargeTurnDeg: 150, 43 | maxSmallTurnDeg: 10, 44 | maxWiggleDeg: 5, 45 | imageSprite: '', 46 | bugWidth: 13, 47 | bugHeight: 14, 48 | num_frames: 5, 49 | zoom: 10, // random zoom variation from 1 to 10 - 10 being full size. 50 | canFly: true, 51 | canDie: true, 52 | numDeathTypes: 3, 53 | monitorMouseMovement: false, 54 | eventDistanceToBug: 40, 55 | minTimeBetweenMultipy: 1000, 56 | mouseOver: 'random' // can be 'fly', 'flyoff' (if the bug can fly), die', 'multiply', 'nothing' or 'random' 57 | }, 58 | 59 | initialize: function(options) { 60 | 61 | this.options = mergeOptions(this.options, options); 62 | 63 | // sanity check: 64 | if (this.options.minBugs > this.options.maxBugs) { 65 | this.options.minBugs = this.options.maxBugs; 66 | } 67 | 68 | this.modes = ['multiply', 'nothing']; 69 | 70 | if (this.options.canFly) { 71 | this.modes.push('fly', 'flyoff'); 72 | } 73 | if (this.options.canDie) { 74 | this.modes.push('die'); 75 | } 76 | 77 | if (this.modes.indexOf(this.options.mouseOver) == -1) { 78 | // invalid mode: use random: 79 | this.options.mouseOver = 'random'; 80 | } 81 | 82 | // can we transform? 83 | this.transform = null; 84 | 85 | this.transforms = { 86 | 'Moz': function(s) { 87 | this.bug.style.MozTransform = s; 88 | }, 89 | 'webkit': function(s) { 90 | this.bug.style.webkitTransform = s; 91 | }, 92 | 'O': function(s) { 93 | this.bug.style.OTransform = s; 94 | }, 95 | 'ms': function(s) { 96 | this.bug.style.msTransform = s; 97 | }, 98 | 'Khtml': function(s) { 99 | this.bug.style.KhtmlTransform = s; 100 | }, 101 | 'w3c': function(s) { 102 | this.bug.style.transform = s; 103 | } 104 | }; 105 | 106 | 107 | // check to see if it is a modern browser: 108 | 109 | if ('transform' in document.documentElement.style) { 110 | this.transform = this.transforms.w3c; 111 | } else { 112 | 113 | // feature detection for the other transforms: 114 | var vendors = ['Moz', 'webkit', 'O', 'ms', 'Khtml'], 115 | i = 0; 116 | 117 | for (i = 0; i < vendors.length; i++) { 118 | if (vendors[i] + 'Transform' in document.documentElement.style) { 119 | this.transform = this.transforms[vendors[i]]; 120 | break; 121 | } 122 | } 123 | } 124 | 125 | // dont support transforms... quit 126 | if (!this.transform) return; 127 | 128 | // make bugs: 129 | this.bugs = []; 130 | var numBugs = (this.options.mouseOver === 'multiply') ? this.options.minBugs : this.random(this.options.minBugs, this.options.maxBugs, true), 131 | i = 0, 132 | that = this; 133 | 134 | for (i = 0; i < numBugs; i++) { 135 | var options = JSON.parse(JSON.stringify(this.options)), 136 | b = SpawnBug(); 137 | 138 | options.wingsOpen = (this.options.canFly) ? ((Math.random() > 0.5) ? true : false) : true, 139 | options.walkSpeed = this.random(this.options.minSpeed, this.options.maxSpeed), 140 | 141 | b.initialize(this.transform, options); 142 | this.bugs.push(b); 143 | } 144 | 145 | // fly them in staggered: 146 | this.spawnDelay = []; 147 | for (i = 0; i < numBugs; i++) { 148 | var delay = this.random(this.options.minDelay, this.options.maxDelay, true), 149 | thebug = this.bugs[i]; 150 | // fly the bug onto the page: 151 | this.spawnDelay[i] = setTimeout((function(thebug) { 152 | return function() { 153 | if (that.options.canFly) { 154 | thebug.flyIn(); 155 | } else { 156 | thebug.walkIn(); 157 | } 158 | 159 | }; 160 | }(thebug)), delay); 161 | 162 | // add mouse over events: 163 | that.add_events_to_bug(thebug); 164 | } 165 | 166 | // add window event if required: 167 | if (this.options.monitorMouseMovement) { 168 | window.onmousemove = function() { 169 | that.check_if_mouse_close_to_bug(); 170 | }; 171 | } 172 | 173 | }, 174 | 175 | stop: function() { 176 | for (var i = 0; i < this.bugs.length; i++) { 177 | if(this.spawnDelay[i]) clearTimeout(this.spawnDelay[i]); 178 | this.bugs[i].stop(); 179 | } 180 | }, 181 | 182 | end: function() { 183 | for (var i = 0; i < this.bugs.length; i++) { 184 | if(this.spawnDelay[i]) clearTimeout(this.spawnDelay[i]); 185 | this.bugs[i].stop(); 186 | this.bugs[i].remove(); 187 | } 188 | }, 189 | 190 | reset: function() { 191 | this.stop(); 192 | for (var i = 0; i < this.bugs.length; i++) { 193 | this.bugs[i].reset(); 194 | this.bugs[i].walkIn(); 195 | } 196 | }, 197 | 198 | killAll: function() { 199 | for (var i = 0; i < this.bugs.length; i++) { 200 | if(this.spawnDelay[i]) clearTimeout(this.spawnDelay[i]); 201 | this.bugs[i].die(); 202 | } 203 | }, 204 | 205 | add_events_to_bug: function(thebug) { 206 | var that = this; 207 | if (thebug.bug) { 208 | if (thebug.bug.addEventListener) { 209 | thebug.bug.addEventListener('mouseover', function(e) { 210 | that.on_bug(thebug); 211 | }); 212 | } else if (thebug.bug.attachEvent) { 213 | thebug.bug.attachEvent('onmouseover', function(e) { 214 | that.on_bug(thebug); 215 | }); 216 | } 217 | } 218 | }, 219 | 220 | check_if_mouse_close_to_bug: function(e) { 221 | e = e || window.event; 222 | if (!e) { 223 | return; 224 | } 225 | 226 | var posx = 0, 227 | posy = 0; 228 | if (e.client && e.client.x) { 229 | posx = e.client.x; 230 | posy = e.client.y; 231 | } else if (e.clientX) { 232 | posx = e.clientX; 233 | posy = e.clientY; 234 | } else if (e.page && e.page.x) { 235 | posx = e.page.x - (document.body.scrollLeft + document.documentElement.scrollLeft); 236 | posy = e.page.y - (document.body.scrollTop + document.documentElement.scrollTop); 237 | } else if (e.pageX) { 238 | posx = e.pageX - (document.body.scrollLeft + document.documentElement.scrollLeft); 239 | posy = e.pageY - (document.body.scrollTop + document.documentElement.scrollTop); 240 | } 241 | var numBugs = this.bugs.length, 242 | i = 0; 243 | for (i = 0; i < numBugs; i++) { 244 | var pos = this.bugs[i].getPos(); 245 | if (pos) { 246 | if (Math.abs(pos.top - posy) + Math.abs(pos.left - posx) < this.options.eventDistanceToBug && !this.bugs[i].flyperiodical) { 247 | this.near_bug(this.bugs[i]); 248 | } 249 | } 250 | } 251 | 252 | }, 253 | 254 | near_bug: function(bug) { 255 | this.on_bug(bug); 256 | }, 257 | 258 | on_bug: function(bug) { 259 | if (!bug.alive) { 260 | return; 261 | } 262 | 263 | var mode = this.options.mouseOver; 264 | 265 | if (mode === 'random') { 266 | mode = this.modes[(this.random(0, (this.modes.length - 1), true))]; 267 | } 268 | 269 | if (mode === 'fly') { 270 | // fly away! 271 | bug.stop(); 272 | bug.flyRand(); 273 | } else if (mode === 'nothing') { 274 | return; 275 | } else if (mode === 'flyoff') { 276 | // fly away and off the page 277 | bug.stop(); 278 | bug.flyOff(); 279 | } else if (mode === 'die') { 280 | // drop dead! 281 | bug.die(); 282 | } else if (mode === 'multiply') { 283 | if (!this.multiplyDelay && this.bugs.length < this.options.maxBugs) { 284 | // spawn another: 285 | // create new bug: 286 | var b = SpawnBug(), 287 | options = JSON.parse(JSON.stringify(this.options)), 288 | pos = bug.getPos(), 289 | that = this; 290 | 291 | options.wingsOpen = (this.options.canFly) ? ((Math.random() > 0.5) ? true : false) : true; 292 | options.walkSpeed = this.random(this.options.minSpeed, this.options.maxSpeed); 293 | 294 | b.initialize(this.transform, options); 295 | b.drawBug(pos.top, pos.left); 296 | // fly them both away: 297 | if (options.canFly) { 298 | b.flyRand(); 299 | bug.flyRand(); 300 | } else { 301 | b.go(); 302 | bug.go(); 303 | } 304 | // store new bug: 305 | this.bugs.push(b); 306 | // watch out for spawning too quickly: 307 | this.multiplyDelay = true; 308 | setTimeout(function() { 309 | // add event to this bug: 310 | that.add_events_to_bug(b); 311 | that.multiplyDelay = false; 312 | }, this.options.minTimeBetweenMultipy); 313 | } 314 | 315 | } 316 | }, 317 | 318 | random: function(min, max, round) { 319 | if (min == max) return ((round) ? Math.round(min) : min); 320 | 321 | var result = ((min - 0.5) + (Math.random() * (max - min + 1))); 322 | if (result > max) { 323 | result = max; 324 | } else if (result < min) { 325 | result = min; 326 | } 327 | return ((round) ? Math.round(result) : result); 328 | } 329 | 330 | 331 | }; 332 | 333 | var BugController = function() { 334 | this.initialize.apply(this, arguments); 335 | } 336 | BugController.prototype = BugDispatch; 337 | 338 | var SpiderController = function() { 339 | var spiderOptions = { 340 | imageSprite: '', 341 | bugWidth: 69, 342 | bugHeight: 90, 343 | num_frames: 7, 344 | canFly: false, 345 | canDie: true, 346 | numDeathTypes: 2, 347 | zoom: 6, 348 | minDelay: 200, 349 | maxDelay: 3000, 350 | minSpeed: 6, 351 | maxSpeed: 13, 352 | minBugs: 3, 353 | maxBugs: 10 354 | }; 355 | this.options = mergeOptions(this.options, spiderOptions); 356 | this.initialize.apply(this, arguments); 357 | 358 | } 359 | SpiderController.prototype = BugDispatch; 360 | 361 | /***************/ 362 | /** Bug **/ 363 | /***************/ 364 | 365 | var Bug = { 366 | 367 | options: { 368 | wingsOpen: false, 369 | walkSpeed: 2, 370 | flySpeed: 40, 371 | edge_resistance: 50, 372 | zoom: 10 373 | 374 | }, 375 | 376 | initialize: function(transform, options) { 377 | 378 | this.options = mergeOptions(this.options, options); 379 | 380 | this.NEAR_TOP_EDGE = 1; 381 | this.NEAR_BOTTOM_EDGE = 2; 382 | this.NEAR_LEFT_EDGE = 4; 383 | this.NEAR_RIGHT_EDGE = 8; 384 | this.directions = {}; // 0 degrees starts on the East 385 | this.directions[this.NEAR_TOP_EDGE] = 270; 386 | this.directions[this.NEAR_BOTTOM_EDGE] = 90; 387 | this.directions[this.NEAR_LEFT_EDGE] = 0; 388 | this.directions[this.NEAR_RIGHT_EDGE] = 180; 389 | this.directions[this.NEAR_TOP_EDGE + this.NEAR_LEFT_EDGE] = 315; 390 | this.directions[this.NEAR_TOP_EDGE + this.NEAR_RIGHT_EDGE] = 225; 391 | this.directions[this.NEAR_BOTTOM_EDGE + this.NEAR_LEFT_EDGE] = 45; 392 | this.directions[this.NEAR_BOTTOM_EDGE + this.NEAR_RIGHT_EDGE] = 135; 393 | 394 | this.angle_deg = 0; 395 | this.angle_rad = 0; 396 | this.large_turn_angle_deg = 0; 397 | this.near_edge = false; 398 | this.edge_test_counter = 10; 399 | this.small_turn_counter = 0; 400 | this.large_turn_counter = 0; 401 | this.fly_counter = 0; 402 | this.toggle_stationary_counter = Math.random() * 50; 403 | this.zoom = this.random(this.options.zoom, 10) / 10; 404 | 405 | this.stationary = false; 406 | this.bug = null; 407 | this.active = true; 408 | this.wingsOpen = this.options.wingsOpen; 409 | this.transform = transform; 410 | this.walkIndex = 0; 411 | this.flyIndex = 0; 412 | this.alive = true; 413 | this.twitchTimer = null; 414 | 415 | this.rad2deg_k = 180 / Math.PI; 416 | this.deg2rad_k = Math.PI / 180; 417 | 418 | this.makeBug(); 419 | 420 | this.angle_rad = this.deg2rad(this.angle_deg); 421 | 422 | this.angle_deg = this.random(0, 360, true); 423 | 424 | }, 425 | 426 | go: function() { 427 | if (this.transform) { 428 | this.drawBug(); 429 | var that = this; 430 | 431 | this.animating = true; 432 | 433 | this.going = requestAnimFrame(function(t) { 434 | that.animate(t); 435 | }); 436 | } 437 | }, 438 | 439 | stop: function() { 440 | this.animating = false; 441 | if (this.going) { 442 | clearTimeout(this.going); 443 | this.going = null; 444 | } 445 | if (this.flyperiodical) { 446 | clearTimeout(this.flyperiodical); 447 | this.flyperiodical = null; 448 | } 449 | if (this.twitchTimer) { 450 | clearTimeout(this.twitchTimer); 451 | this.twitchTimer = null; 452 | } 453 | }, 454 | 455 | remove: function() { 456 | this.active = false; 457 | if (this.inserted && this.bug.parentNode) { 458 | this.bug.parentNode.removeChild(this.bug); 459 | this.inserted = false; 460 | } 461 | }, 462 | 463 | reset: function() { 464 | this.alive = true; 465 | this.active = true; 466 | this.bug.style.bottom = ''; 467 | this.bug.style.top = 0; 468 | this.bug.style.left = 0; 469 | }, 470 | 471 | animate: function(t) { 472 | 473 | if (!this.animating || !this.alive || !this.active) return; 474 | 475 | var that = this; 476 | this.going = requestAnimFrame(function(t) { 477 | that.animate(t); 478 | }); 479 | 480 | if (!('_lastTimestamp' in this)) this._lastTimestamp = t; 481 | 482 | var delta = t - this._lastTimestamp; 483 | 484 | if (delta < 40) return; // don't animate too frequently 485 | 486 | // sometimes if the browser doesnt have focus, or the delta in request animation 487 | // frame can be very large. We set a sensible max so that the bugs dont spaz out. 488 | 489 | if (delta > 200) delta = 200; 490 | 491 | this._lastTimestamp = t; 492 | 493 | if (--this.toggle_stationary_counter <= 0) { 494 | this.toggleStationary(); 495 | } 496 | if (this.stationary) { 497 | return; 498 | } 499 | 500 | 501 | if (--this.edge_test_counter <= 0 && this.bug_near_window_edge()) { 502 | // if near edge, go away from edge 503 | this.angle_deg %= 360; 504 | if (this.angle_deg < 0) this.angle_deg += 360; 505 | 506 | if (Math.abs(this.directions[this.near_edge] - this.angle_deg) > 15) { 507 | var angle1 = this.directions[this.near_edge] - this.angle_deg; 508 | var angle2 = (360 - this.angle_deg) + this.directions[this.near_edge]; 509 | this.large_turn_angle_deg = (Math.abs(angle1) < Math.abs(angle2) ? angle1 : angle2); 510 | 511 | this.edge_test_counter = 10; 512 | this.large_turn_counter = 100; 513 | this.small_turn_counter = 30; 514 | } 515 | } 516 | if (--this.large_turn_counter <= 0) { 517 | this.large_turn_angle_deg = this.random(1, this.options.maxLargeTurnDeg, true); 518 | this.next_large_turn(); 519 | } 520 | if (--this.small_turn_counter <= 0) { 521 | this.angle_deg += this.random(1, this.options.maxSmallTurnDeg); 522 | this.next_small_turn(); 523 | } else { 524 | var dangle = this.random(1, this.options.maxWiggleDeg, true); 525 | if ((this.large_turn_angle_deg > 0 && dangle < 0) || (this.large_turn_angle_deg < 0 && dangle > 0)) { 526 | dangle = -dangle; // ensures both values either + or - 527 | } 528 | this.large_turn_angle_deg -= dangle; 529 | this.angle_deg += dangle; 530 | } 531 | 532 | this.angle_rad = this.deg2rad(this.angle_deg); 533 | 534 | var dx = Math.cos(this.angle_rad) * this.options.walkSpeed * (delta / 100); 535 | var dy = -Math.sin(this.angle_rad) * this.options.walkSpeed * (delta / 100); 536 | 537 | this.moveBug((this.bug.left + dx), (this.bug.top + dy), (90 - this.angle_deg)); 538 | this.walkFrame(); 539 | 540 | }, 541 | 542 | makeBug: function() { 543 | if (!this.bug && this.active) { 544 | var row = (this.wingsOpen) ? '0' : '-' + this.options.bugHeight + 'px', 545 | bug = document.createElement('div'); 546 | bug.className = 'bug'; 547 | bug.style.background = 'transparent url(' + this.options.imageSprite + ') no-repeat 0 ' + row; 548 | bug.style.width = this.options.bugWidth + 'px'; 549 | bug.style.height = this.options.bugHeight + 'px'; 550 | bug.style.position = 'fixed'; 551 | bug.style.top = 0; 552 | bug.style.left = 0; 553 | bug.style.zIndex = '9999999'; 554 | 555 | this.bug = bug; 556 | this.setPos(); 557 | 558 | } 559 | 560 | }, 561 | 562 | setPos: function(top, left) { 563 | this.bug.top = top || this.random(this.options.edge_resistance, document.documentElement.clientHeight - this.options.edge_resistance); 564 | 565 | this.bug.left = left || this.random(this.options.edge_resistance, document.documentElement.clientWidth - this.options.edge_resistance); 566 | 567 | this.moveBug(this.bug.left, this.bug.top, (90 - this.angle_deg)); 568 | }, 569 | 570 | moveBug: function(x, y, deg) { 571 | // keep track of where we are: 572 | this.bug.left = x; 573 | this.bug.top = y; 574 | 575 | // transform: 576 | var trans = "translate(" + parseInt(x) + "px," + parseInt(y) + "px)"; 577 | if (deg) { 578 | //console.log("translate("+(x)+"px, "+(y)+"px) rotate("+deg+"deg)"); 579 | trans += " rotate(" + deg + "deg)"; 580 | } 581 | trans += " scale(" + this.zoom + ")"; 582 | 583 | this.transform(trans); 584 | 585 | }, 586 | 587 | drawBug: function(top, left) { 588 | 589 | if (!this.bug) { 590 | this.makeBug(); 591 | } 592 | if(!this.bug) return; 593 | 594 | if (top && left) { 595 | this.setPos(top, left); 596 | } else { 597 | this.setPos(this.bug.top, this.bug.left) 598 | } 599 | if (!this.inserted) { 600 | this.inserted = true; 601 | document.body.appendChild(this.bug); 602 | } 603 | }, 604 | 605 | toggleStationary: function() { 606 | this.stationary = !this.stationary; 607 | this.next_stationary(); 608 | var ypos = (this.wingsOpen) ? '0' : '-' + this.options.bugHeight + 'px'; 609 | if (this.stationary) { 610 | 611 | this.bug.style.backgroundPosition = '0 ' + ypos; 612 | } else { 613 | this.bug.style.backgroundPosition = '-' + this.options.bugWidth + 'px ' + ypos; 614 | } 615 | }, 616 | 617 | walkFrame: function() { 618 | var xpos = (-1 * (this.walkIndex * this.options.bugWidth)) + 'px', 619 | ypos = (this.wingsOpen) ? '0' : '-' + this.options.bugHeight + 'px'; 620 | this.bug.style.backgroundPosition = xpos + ' ' + ypos; 621 | this.walkIndex++; 622 | if (this.walkIndex >= this.options.num_frames) this.walkIndex = 0; 623 | }, 624 | 625 | fly: function(landingPosition) { 626 | var currentTop = this.bug.top, 627 | currentLeft = this.bug.left, 628 | diffx = (currentLeft - landingPosition.left), 629 | diffy = (currentTop - landingPosition.top), 630 | angle = Math.atan(diffy / diffx); 631 | 632 | if (Math.abs(diffx) + Math.abs(diffy) < 50) { 633 | this.bug.style.backgroundPosition = (-2 * this.options.bugWidth) + 'px -' + (2 * this.options.bugHeight) + 'px'; 634 | } 635 | if (Math.abs(diffx) + Math.abs(diffy) < 30) { 636 | this.bug.style.backgroundPosition = (-1 * this.options.bugWidth) + 'px -' + (2 * this.options.bugHeight) + 'px'; 637 | } 638 | if (Math.abs(diffx) + Math.abs(diffy) < 10) { 639 | // close enough: 640 | this.bug.style.backgroundPosition = '0 0'; //+row+'px')); 641 | 642 | this.stop(); 643 | this.go(); 644 | //this.go.delay(100, this); 645 | 646 | return; 647 | 648 | } 649 | 650 | // make it wiggle: disabled becuase its just too fast to see... better would be to make its path wiggly. 651 | //angle = angle - (this.deg2rad(this.random(0,10))); 652 | //console.log('angle: ',this.rad2deg(angle)); 653 | 654 | var dx = Math.cos(angle) * this.options.flySpeed, 655 | dy = Math.sin(angle) * this.options.flySpeed; 656 | 657 | if ((currentLeft > landingPosition.left && dx > 0) || (currentLeft > landingPosition.left && dx < 0)) { 658 | // make sure angle is right way 659 | dx = -1 * dx; 660 | if (Math.abs(diffx) < Math.abs(dx)) { 661 | dx = dx / 4; 662 | } 663 | } 664 | if ((currentTop < landingPosition.top && dy < 0) || (currentTop > landingPosition.top && dy > 0)) { 665 | dy = -1 * dy; 666 | if (Math.abs(diffy) < Math.abs(dy)) { 667 | dy = dy / 4; 668 | } 669 | } 670 | 671 | this.moveBug((currentLeft + dx), (currentTop + dy)); 672 | 673 | }, 674 | 675 | flyRand: function() { 676 | this.stop(); 677 | var landingPosition = {}; 678 | landingPosition.top = this.random(this.options.edge_resistance, document.documentElement.clientHeight - this.options.edge_resistance); 679 | landingPosition.left = this.random(this.options.edge_resistance, document.documentElement.clientWidth - this.options.edge_resistance); 680 | 681 | this.startFlying(landingPosition); 682 | }, 683 | 684 | startFlying: function(landingPosition) { 685 | 686 | var currentTop = this.bug.top, 687 | currentLeft = this.bug.left, 688 | diffx = (landingPosition.left - currentLeft), 689 | diffy = (landingPosition.top - currentTop); 690 | 691 | this.bug.left = landingPosition.left; 692 | this.bug.top = landingPosition.top; 693 | 694 | this.angle_rad = Math.atan(diffy / diffx); 695 | this.angle_deg = this.rad2deg(this.angle_rad); 696 | 697 | if (diffx > 0) { 698 | // going left: quadrant 1 or 2 699 | this.angle_deg = 90 + this.angle_deg; 700 | } else { 701 | // going right: quadrant 3 or 4 702 | this.angle_deg = 270 + this.angle_deg; 703 | } 704 | 705 | this.moveBug(currentLeft, currentTop, this.angle_deg); 706 | 707 | // start animation: 708 | var that = this; 709 | this.flyperiodical = setInterval(function() { 710 | that.fly(landingPosition); 711 | }, 10); 712 | }, 713 | 714 | flyIn: function() { 715 | if (!this.bug) { 716 | this.makeBug(); 717 | } 718 | 719 | if(!this.bug) return; 720 | 721 | this.stop(); 722 | // pick a random side: 723 | var side = Math.round(Math.random() * 4 - 0.5), 724 | d = document, 725 | e = d.documentElement, 726 | g = d.getElementsByTagName('body')[0], 727 | windowX = window.innerWidth || e.clientWidth || g.clientWidth, 728 | windowY = window.innerHeight || e.clientHeight || g.clientHeight; 729 | if (side > 3) side = 3; 730 | if (side < 0) side = 0; 731 | var style = {}, 732 | s; 733 | if (side === 0) { 734 | // top: 735 | style.top = (-2 * this.options.bugHeight); 736 | style.left = Math.random() * windowX; 737 | } else if (side === 1) { 738 | // right: 739 | style.top = Math.random() * windowY; 740 | style.left = windowX + (2 * this.options.bugWidth); 741 | } else if (side === 2) { 742 | // bottom: 743 | style.top = windowY + (2 * this.options.bugHeight); 744 | style.left = Math.random() * windowX; 745 | } else { 746 | // left: 747 | style.top = Math.random() * windowY; 748 | style.left = (-3 * this.options.bugWidth); 749 | } 750 | var row = (this.wingsOpen) ? '0' : '-' + this.options.bugHeight + 'px'; 751 | this.bug.style.backgroundPosition = (-3 * this.options.bugWidth) + 'px ' + row; 752 | this.bug.top = style.top 753 | this.bug.left = style.left 754 | 755 | this.drawBug(); 756 | 757 | // landing position: 758 | var landingPosition = {}; 759 | landingPosition.top = this.random(this.options.edge_resistance, document.documentElement.clientHeight - this.options.edge_resistance); 760 | landingPosition.left = this.random(this.options.edge_resistance, document.documentElement.clientWidth - this.options.edge_resistance); 761 | 762 | this.startFlying(landingPosition); 763 | }, 764 | 765 | walkIn: function() { 766 | if (!this.bug) { 767 | this.makeBug(); 768 | } 769 | 770 | if(!this.bug) return; 771 | 772 | this.stop(); 773 | // pick a random side: 774 | var side = Math.round(Math.random() * 4 - 0.5), 775 | d = document, 776 | e = d.documentElement, 777 | g = d.getElementsByTagName('body')[0], 778 | windowX = window.innerWidth || e.clientWidth || g.clientWidth, 779 | windowY = window.innerHeight || e.clientHeight || g.clientHeight; 780 | if (side > 3) side = 3; 781 | if (side < 0) side = 0; 782 | var style = {}, 783 | s; 784 | if (side === 0) { 785 | // top: 786 | style.top = (-1.3 * this.options.bugHeight); 787 | style.left = Math.random() * windowX; 788 | } else if (side === 1) { 789 | // right: 790 | style.top = Math.random() * windowY; 791 | style.left = windowX + (0.3 * this.options.bugWidth); 792 | } else if (side === 2) { 793 | // bottom: 794 | style.top = windowY + (0.3 * this.options.bugHeight); 795 | style.left = Math.random() * windowX; 796 | } else { 797 | // left: 798 | style.top = Math.random() * windowY; 799 | style.left = (-1.3 * this.options.bugWidth); 800 | } 801 | var row = (this.wingsOpen) ? '0' : '-' + this.options.bugHeight + 'px'; 802 | this.bug.style.backgroundPosition = (-3 * this.options.bugWidth) + 'px ' + row; 803 | this.bug.top = style.top 804 | this.bug.left = style.left 805 | 806 | this.drawBug(); 807 | 808 | // start walking: 809 | this.go(); 810 | 811 | }, 812 | 813 | flyOff: function() { 814 | this.stop(); 815 | // pick a random side to fly off to, where 0 is top and continuing clockwise. 816 | var side = this.random(0, 3), 817 | style = {}, 818 | d = document, 819 | e = d.documentElement, 820 | g = d.getElementsByTagName('body')[0], 821 | windowX = window.innerWidth || e.clientWidth || g.clientWidth, 822 | windowY = window.innerHeight || e.clientHeight || g.clientHeight; 823 | 824 | if (side === 0) { 825 | // top: 826 | style.top = -200; 827 | style.left = Math.random() * windowX; 828 | } else if (side === 1) { 829 | // right: 830 | style.top = Math.random() * windowY; 831 | style.left = windowX + 200; 832 | } else if (side === 2) { 833 | //bottom: 834 | style.top = windowY + 200; 835 | style.left = Math.random() * windowX; 836 | } else { 837 | // left: 838 | style.top = Math.random() * windowY; 839 | style.left = -200; 840 | } 841 | this.startFlying(style); 842 | }, 843 | 844 | die: function() { 845 | this.stop(); 846 | //pick death style: 847 | var deathType = this.random(0, this.options.numDeathTypes - 1); 848 | 849 | this.alive = false; 850 | this.drop(deathType); 851 | }, 852 | 853 | drop: function(deathType) { 854 | var startPos = this.bug.top, 855 | d = document, 856 | e = d.documentElement, 857 | g = d.getElementsByTagName('body')[0], 858 | finalPos = window.innerHeight || e.clientHeight || g.clientHeight, 859 | finalPos = finalPos - this.options.bugHeight, 860 | rotationRate = this.random(0, 20, true), 861 | startTime = Date.now(), 862 | that = this; 863 | 864 | this.dropTimer = requestAnimFrame(function(t) { 865 | that._lastTimestamp = t; 866 | that.dropping(t, startPos, finalPos, rotationRate, deathType); 867 | }); 868 | 869 | }, 870 | 871 | dropping: function(t, startPos, finalPos, rotationRate, deathType) { 872 | var elapsedTime = t - this._lastTimestamp, 873 | deltaPos = (0.002 * (elapsedTime * elapsedTime)), 874 | newPos = startPos + deltaPos; 875 | //console.log(t, elapsedTime, deltaPos, newPos); 876 | 877 | var that = this; 878 | 879 | 880 | if (newPos >= finalPos) { 881 | newPos = finalPos; 882 | clearTimeout(this.dropTimer); 883 | 884 | 885 | 886 | this.angle_deg = 0; 887 | this.angle_rad = this.deg2rad(this.angle_deg); 888 | this.transform("rotate(" + (90 - this.angle_deg) + "deg) scale(" + this.zoom + ")"); 889 | this.bug.style.top = null; 890 | // because it is (or might be) zoomed and rotated, we cannot just just bottom = 0. Figure out real bottom position: 891 | var rotationOffset = ((this.options.bugWidth * this.zoom) - (this.options.bugHeight * this.zoom)) / 2; 892 | var zoomOffset = ((this.options.bugHeight) / 2) * (1 - this.zoom); 893 | this.bug.style.bottom = Math.ceil((rotationOffset - zoomOffset)) + 'px'; // because its rotated and zoomed. 894 | this.bug.style.left = this.bug.left + 'px'; 895 | this.bug.style.backgroundPosition = '-' + ((deathType * 2) * this.options.bugWidth) + 'px 100%'; 896 | 897 | 898 | this.twitch(deathType); 899 | 900 | return; 901 | } 902 | 903 | this.dropTimer = requestAnimFrame(function(t) { 904 | that.dropping(t, startPos, finalPos, rotationRate, deathType); 905 | }); 906 | 907 | if (elapsedTime < 20) return; 908 | 909 | this.angle_deg = ((this.angle_deg + rotationRate) % 360); 910 | this.angle_rad = this.deg2rad(this.angle_deg); 911 | 912 | this.moveBug(this.bug.left, newPos, this.angle_deg); 913 | }, 914 | 915 | twitch: function(deathType, legPos) { 916 | //this.bug.style.back 917 | if (!legPos) legPos = 0; 918 | var that = this; 919 | if (deathType === 0 || deathType === 1) { 920 | that.twitchTimer = setTimeout(function() { 921 | that.bug.style.backgroundPosition = '-' + ((deathType * 2 + (legPos % 2)) * that.options.bugWidth) + 'px 100%'; 922 | that.twitchTimer = setTimeout(function() { 923 | legPos++; 924 | that.bug.style.backgroundPosition = '-' + ((deathType * 2 + (legPos % 2)) * that.options.bugWidth) + 'px 100%'; 925 | that.twitch(deathType, ++legPos); 926 | }, that.random(300, 800)); 927 | }, this.random(1000, 10000)); 928 | } 929 | }, 930 | 931 | /* helper methods: */ 932 | rad2deg: function(rad) { 933 | return rad * this.rad2deg_k; 934 | }, 935 | deg2rad: function(deg) { 936 | return deg * this.deg2rad_k; 937 | }, 938 | random: function(min, max, plusminus) { 939 | if (min == max) return min; 940 | var result = Math.round(min - 0.5 + (Math.random() * (max - min + 1))); 941 | if (plusminus) return Math.random() > 0.5 ? result : -result; 942 | return result; 943 | }, 944 | 945 | next_small_turn: function() { 946 | this.small_turn_counter = Math.round(Math.random() * 10); 947 | }, 948 | next_large_turn: function() { 949 | this.large_turn_counter = Math.round(Math.random() * 40); 950 | }, 951 | next_stationary: function() { 952 | this.toggle_stationary_counter = this.random(50, 300); 953 | }, 954 | 955 | bug_near_window_edge: function() { 956 | this.near_edge = 0; 957 | if (this.bug.top < this.options.edge_resistance) 958 | this.near_edge |= this.NEAR_TOP_EDGE; 959 | else if (this.bug.top > document.documentElement.clientHeight - this.options.edge_resistance) 960 | this.near_edge |= this.NEAR_BOTTOM_EDGE; 961 | if (this.bug.left < this.options.edge_resistance) 962 | this.near_edge |= this.NEAR_LEFT_EDGE; 963 | else if (this.bug.left > document.documentElement.clientWidth - this.options.edge_resistance) 964 | this.near_edge |= this.NEAR_RIGHT_EDGE; 965 | return this.near_edge; 966 | }, 967 | 968 | getPos: function() { 969 | if (this.inserted && this.bug && this.bug.style) { 970 | return { 971 | 'top': parseInt(this.bug.top, 10), 972 | 'left': parseInt(this.bug.left, 10) 973 | }; 974 | } 975 | return null; 976 | } 977 | 978 | }; 979 | 980 | var SpawnBug = function() { 981 | var newBug = {}, 982 | prop; 983 | for (prop in Bug) { 984 | if (Bug.hasOwnProperty(prop)) { 985 | newBug[prop] = Bug[prop]; 986 | } 987 | } 988 | return newBug; 989 | }; 990 | 991 | // debated about which pattern to use to instantiate each bug... 992 | // see http://jsperf.com/obj-vs-prototype-vs-other 993 | 994 | 995 | 996 | /** 997 | * Helper methods: 998 | **/ 999 | 1000 | var mergeOptions = function(obj1, obj2, clone) { 1001 | if (typeof(clone) == 'undefined') { 1002 | clone = true; 1003 | } 1004 | var newobj = (clone) ? cloneOf(obj1) : obj1; 1005 | for (var key in obj2) { 1006 | if (obj2.hasOwnProperty(key)) { 1007 | newobj[key] = obj2[key]; 1008 | } 1009 | } 1010 | return newobj; 1011 | }; 1012 | 1013 | var cloneOf = function(obj) { 1014 | if (obj == null || typeof(obj) != 'object') 1015 | return obj; 1016 | 1017 | var temp = obj.constructor(); // changed 1018 | 1019 | for (var key in obj) { 1020 | if (obj.hasOwnProperty(key)) { 1021 | temp[key] = cloneOf(obj[key]); 1022 | } 1023 | } 1024 | return temp; 1025 | } 1026 | 1027 | /* Request animation frame polyfill */ 1028 | /* http://paulirish.com/2011/requestanimationframe-for-smart-animating/ */ 1029 | window.requestAnimFrame = (function() { 1030 | return window.requestAnimationFrame || 1031 | window.webkitRequestAnimationFrame || 1032 | window.mozRequestAnimationFrame || 1033 | window.oRequestAnimationFrame || 1034 | window.msRequestAnimationFrame || function( /* function */ callback, /* DOMElement */ element) { 1035 | window.setTimeout(callback, 1000 / 60); 1036 | }; 1037 | })(); 1038 | 1039 | chrome.storage.sync.get({ 1040 | bugBugs: false, 1041 | bugSpiders: true, 1042 | bugMax: 3, 1043 | bugMin: 1, 1044 | bugDelay: 2.5, 1045 | bugOdds: 100 1046 | }, function(items) { 1047 | 1048 | var rand = Math.random()*100; 1049 | 1050 | if(items.bugOdds > rand) { 1051 | if(items.bugMin > items.bugMax) items.bugMin = items.bugMax; 1052 | 1053 | if(items.bugBugs && items.bugSpiders) { 1054 | // need to split: 1055 | var splitPercentage = Math.random(); 1056 | var minSpiders = parseInt(splitPercentage*items.bugMin); 1057 | var minBugs = items.bugMin - minSpiders; 1058 | var maxSpiders = parseInt(splitPercentage*items.bugMax); 1059 | var maxBugs = items.bugMax - maxSpiders; 1060 | if(maxBugs > 0) { 1061 | new BugController({'minBugs':minBugs, 'maxBugs':maxBugs, 'minDelay':items.bugDelay*1000}); 1062 | } 1063 | if(maxSpiders > 0) { 1064 | new SpiderController({'minBugs':minSpiders, 'maxBugs':maxSpiders, 'minDelay':items.bugDelay*1000}); 1065 | } 1066 | } else if(items.bugBugs) { 1067 | new BugController({'minBugs':items.bugMin, 'maxBugs':items.bugMax, 'minDelay':items.bugDelay*1000}); 1068 | } else if(items.bugSpiders) { 1069 | new SpiderController({'minBugs':items.bugMin, 'maxBugs':items.bugMax, 'minDelay':items.bugDelay*1000}); 1070 | } 1071 | } 1072 | }); -------------------------------------------------------------------------------- /chrome-extension/manifest.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "Bug", 3 | "short_name": "Bug", 4 | "version": "1.0", 5 | "manifest_version": 2, 6 | "description": "Adds very realistic animated bugs (spiders and/or flies) to your chrome browser.", 7 | "homepage_url": "https://auz.github.io/Bug/", 8 | 9 | "options_ui": { 10 | "page": "options.html", 11 | "open_in_tab": false 12 | }, 13 | 14 | "content_scripts": [ 15 | { 16 | "run_at" :"document_end", 17 | "matches": ["http://*/*", "https://*/*"], 18 | "js": ["bug.js"] 19 | } 20 | ], 21 | "browser_action": { 22 | "default_popup": "options.html" 23 | }, 24 | "permissions": [ 25 | "storage" 26 | ], 27 | "icons": { 28 | "16": "bug-icon-16.png", 29 | "48": "bug-icon-48.png", 30 | "128": "bug-icon-128.png" 31 | } 32 | 33 | } 34 | -------------------------------------------------------------------------------- /chrome-extension/options.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | Bug Options 4 | 5 | 6 | 10 |
11 | 15 |
16 | 19 |
20 | 23 |
24 | 27 |
28 | 31 |
32 |
33 | 34 | 35 | 36 | 37 | -------------------------------------------------------------------------------- /chrome-extension/options.js: -------------------------------------------------------------------------------- 1 | // Saves options to chrome.storage 2 | function save_options() { 3 | var bugs = document.getElementById('bugs').checked; 4 | var spiders = document.getElementById('spiders').checked; 5 | 6 | var max = parseInt(document.getElementById('maxnum').value); 7 | var min = parseInt(document.getElementById('minnum').value); 8 | var delay = document.getElementById('delay').value; 9 | var odds = parseInt(document.getElementById('odds').value); 10 | 11 | chrome.storage.sync.set({ 12 | bugBugs: bugs, 13 | bugSpiders: spiders, 14 | bugMax: max, 15 | bugMin: min, 16 | bugDelay: delay, 17 | bugOdds: odds 18 | }, function() { 19 | // Update status to let user know options were saved. 20 | var status = document.getElementById('status'); 21 | status.textContent = 'Options saved.'; 22 | setTimeout(function() { 23 | status.textContent = ''; 24 | }, 750); 25 | }); 26 | } 27 | 28 | // Restores select box and checkbox state using the preferences 29 | // stored in chrome.storage. 30 | function restore_options() { 31 | // Use default value color = 'red' and likesColor = true. 32 | chrome.storage.sync.get({ 33 | bugBugs: false, 34 | bugSpiders: true, 35 | bugMax: 3, 36 | bugMin: 1, 37 | bugDelay: 3.5, 38 | bugOdds: 70 39 | }, function(items) { 40 | document.getElementById('bugs').checked = items.bugBugs; 41 | document.getElementById('spiders').checked = items.bugSpiders; 42 | document.getElementById('maxnum').value = items.bugMax; 43 | document.getElementById('minnum').value = items.bugMin; 44 | document.getElementById('delay').value = items.bugDelay; 45 | document.getElementById('odds').value = items.bugOdds; 46 | }); 47 | } 48 | document.addEventListener('DOMContentLoaded', restore_options); 49 | document.getElementById('save').addEventListener('click', 50 | save_options); -------------------------------------------------------------------------------- /example.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 16 | 17 | 18 | -------------------------------------------------------------------------------- /fly-sprite.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Auz/Bug/8eac7b7337604fb3b8c8f0be1116c8de0984057e/fly-sprite.png -------------------------------------------------------------------------------- /originals/fly-sprite-10x.psd: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Auz/Bug/8eac7b7337604fb3b8c8f0be1116c8de0984057e/originals/fly-sprite-10x.psd -------------------------------------------------------------------------------- /originals/spider2.psd: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Auz/Bug/8eac7b7337604fb3b8c8f0be1116c8de0984057e/originals/spider2.psd -------------------------------------------------------------------------------- /spider-sprite.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Auz/Bug/8eac7b7337604fb3b8c8f0be1116c8de0984057e/spider-sprite.png --------------------------------------------------------------------------------