diff --git a/Games/Tether/404.html b/Games/Tether/404.html new file mode 100644 index 0000000000..829eda8fdc --- /dev/null +++ b/Games/Tether/404.html @@ -0,0 +1,33 @@ + + + + + + Page Not Found + + + + +
+

404

+

Page Not Found

+

The specified file was not found on this website. Please check the URL for mistakes and try again.

+

Why am I seeing this?

+

This page was generated by the Firebase Command-Line Interface. To modify it, edit the 404.html file in your project's configured public directory.

+
+ + diff --git a/Games/Tether/favicon.ico b/Games/Tether/favicon.ico new file mode 100644 index 0000000000..52e6cdabbf Binary files /dev/null and b/Games/Tether/favicon.ico differ diff --git a/Games/Tether/icons/android-icon-144x144.png b/Games/Tether/icons/android-icon-144x144.png new file mode 100644 index 0000000000..4a480bf354 Binary files /dev/null and b/Games/Tether/icons/android-icon-144x144.png differ diff --git a/Games/Tether/icons/android-icon-192x192.png b/Games/Tether/icons/android-icon-192x192.png new file mode 100644 index 0000000000..ea1cb0ea3d Binary files /dev/null and b/Games/Tether/icons/android-icon-192x192.png differ diff --git a/Games/Tether/icons/android-icon-36x36.png b/Games/Tether/icons/android-icon-36x36.png new file mode 100644 index 0000000000..9ba08ddf31 Binary files /dev/null and b/Games/Tether/icons/android-icon-36x36.png differ diff --git a/Games/Tether/icons/android-icon-48x48.png b/Games/Tether/icons/android-icon-48x48.png new file mode 100644 index 0000000000..b36441c231 Binary files /dev/null and b/Games/Tether/icons/android-icon-48x48.png differ diff --git a/Games/Tether/icons/android-icon-72x72.png b/Games/Tether/icons/android-icon-72x72.png new file mode 100644 index 0000000000..cd1e9e06f2 Binary files /dev/null and b/Games/Tether/icons/android-icon-72x72.png differ diff --git a/Games/Tether/icons/android-icon-96x96.png b/Games/Tether/icons/android-icon-96x96.png new file mode 100644 index 0000000000..a4d8c1a4e5 Binary files /dev/null and b/Games/Tether/icons/android-icon-96x96.png differ diff --git a/Games/Tether/icons/apple-icon-114x114.png b/Games/Tether/icons/apple-icon-114x114.png new file mode 100644 index 0000000000..10cbc85bd9 Binary files /dev/null and b/Games/Tether/icons/apple-icon-114x114.png differ diff --git a/Games/Tether/icons/apple-icon-120x120.png b/Games/Tether/icons/apple-icon-120x120.png new file mode 100644 index 0000000000..724fc4e1d4 Binary files /dev/null and b/Games/Tether/icons/apple-icon-120x120.png differ diff --git a/Games/Tether/icons/apple-icon-144x144.png b/Games/Tether/icons/apple-icon-144x144.png new file mode 100644 index 0000000000..4a480bf354 Binary files /dev/null and b/Games/Tether/icons/apple-icon-144x144.png differ diff --git a/Games/Tether/icons/apple-icon-152x152.png b/Games/Tether/icons/apple-icon-152x152.png new file mode 100644 index 0000000000..396eb957f1 Binary files /dev/null and b/Games/Tether/icons/apple-icon-152x152.png differ diff --git a/Games/Tether/icons/apple-icon-180x180.png b/Games/Tether/icons/apple-icon-180x180.png new file mode 100644 index 0000000000..d1dead95cb Binary files /dev/null and b/Games/Tether/icons/apple-icon-180x180.png differ diff --git a/Games/Tether/icons/apple-icon-57x57.png b/Games/Tether/icons/apple-icon-57x57.png new file mode 100644 index 0000000000..bf9f720502 Binary files /dev/null and b/Games/Tether/icons/apple-icon-57x57.png differ diff --git a/Games/Tether/icons/apple-icon-60x60.png b/Games/Tether/icons/apple-icon-60x60.png new file mode 100644 index 0000000000..0c89467cad Binary files /dev/null and b/Games/Tether/icons/apple-icon-60x60.png differ diff --git a/Games/Tether/icons/apple-icon-72x72.png b/Games/Tether/icons/apple-icon-72x72.png new file mode 100644 index 0000000000..cd1e9e06f2 Binary files /dev/null and b/Games/Tether/icons/apple-icon-72x72.png differ diff --git a/Games/Tether/icons/apple-icon-76x76.png b/Games/Tether/icons/apple-icon-76x76.png new file mode 100644 index 0000000000..9711ea98a4 Binary files /dev/null and b/Games/Tether/icons/apple-icon-76x76.png differ diff --git a/Games/Tether/icons/apple-icon-precomposed.png b/Games/Tether/icons/apple-icon-precomposed.png new file mode 100644 index 0000000000..8c574eef59 Binary files /dev/null and b/Games/Tether/icons/apple-icon-precomposed.png differ diff --git a/Games/Tether/icons/apple-icon.png b/Games/Tether/icons/apple-icon.png new file mode 100644 index 0000000000..8c574eef59 Binary files /dev/null and b/Games/Tether/icons/apple-icon.png differ diff --git a/Games/Tether/icons/browserconfig.xml b/Games/Tether/icons/browserconfig.xml new file mode 100644 index 0000000000..c554148223 --- /dev/null +++ b/Games/Tether/icons/browserconfig.xml @@ -0,0 +1,2 @@ + +#ffffff \ No newline at end of file diff --git a/Games/Tether/icons/favicon-16x16.png b/Games/Tether/icons/favicon-16x16.png new file mode 100644 index 0000000000..b2ace7ee26 Binary files /dev/null and b/Games/Tether/icons/favicon-16x16.png differ diff --git a/Games/Tether/icons/favicon-32x32.png b/Games/Tether/icons/favicon-32x32.png new file mode 100644 index 0000000000..6c3622a968 Binary files /dev/null and b/Games/Tether/icons/favicon-32x32.png differ diff --git a/Games/Tether/icons/favicon-96x96.png b/Games/Tether/icons/favicon-96x96.png new file mode 100644 index 0000000000..a4d8c1a4e5 Binary files /dev/null and b/Games/Tether/icons/favicon-96x96.png differ diff --git a/Games/Tether/icons/ms-icon-144x144.png b/Games/Tether/icons/ms-icon-144x144.png new file mode 100644 index 0000000000..4a480bf354 Binary files /dev/null and b/Games/Tether/icons/ms-icon-144x144.png differ diff --git a/Games/Tether/icons/ms-icon-150x150.png b/Games/Tether/icons/ms-icon-150x150.png new file mode 100644 index 0000000000..7d39a0aeca Binary files /dev/null and b/Games/Tether/icons/ms-icon-150x150.png differ diff --git a/Games/Tether/icons/ms-icon-310x310.png b/Games/Tether/icons/ms-icon-310x310.png new file mode 100644 index 0000000000..123739785d Binary files /dev/null and b/Games/Tether/icons/ms-icon-310x310.png differ diff --git a/Games/Tether/icons/ms-icon-70x70.png b/Games/Tether/icons/ms-icon-70x70.png new file mode 100644 index 0000000000..d96d70bc4d Binary files /dev/null and b/Games/Tether/icons/ms-icon-70x70.png differ diff --git a/Games/Tether/index.html b/Games/Tether/index.html new file mode 100644 index 0000000000..e8551a7775 --- /dev/null +++ b/Games/Tether/index.html @@ -0,0 +1 @@ +tether! \ No newline at end of file diff --git a/Games/Tether/public/tether_cover.png b/Games/Tether/public/tether_cover.png new file mode 100644 index 0000000000..cc6983a347 Binary files /dev/null and b/Games/Tether/public/tether_cover.png differ diff --git a/Games/Tether/public/tether_logo-v2.png b/Games/Tether/public/tether_logo-v2.png new file mode 100644 index 0000000000..472c9b4105 Binary files /dev/null and b/Games/Tether/public/tether_logo-v2.png differ diff --git a/Games/Tether/public/tether_logo.png b/Games/Tether/public/tether_logo.png new file mode 100644 index 0000000000..8e668f3b22 Binary files /dev/null and b/Games/Tether/public/tether_logo.png differ diff --git a/Games/Tether/public/tether_opengraphimage.png b/Games/Tether/public/tether_opengraphimage.png new file mode 100644 index 0000000000..6266926eb5 Binary files /dev/null and b/Games/Tether/public/tether_opengraphimage.png differ diff --git a/Games/Tether/public/tether_twittercardimage.png b/Games/Tether/public/tether_twittercardimage.png new file mode 100644 index 0000000000..59cf575484 Binary files /dev/null and b/Games/Tether/public/tether_twittercardimage.png differ diff --git a/Games/Tether/source/game.js b/Games/Tether/source/game.js new file mode 100644 index 0000000000..f3ba5123b7 --- /dev/null +++ b/Games/Tether/source/game.js @@ -0,0 +1,2488 @@ +document.body.classList.add('game'); + +var storage = (function () { + var uid = new Date; + var storage; + var result; + try { + (storage = window.localStorage).setItem(uid, uid); + result = storage.getItem(uid) == uid; + storage.removeItem(uid); + return result && storage; + } catch (exception) { } + storage = function () { console.log('localStorage Disabled.') }; + storage.getItem = function () { console.log('localStorage Disabled.') }; + storage.setItem = function () { console.log('localStorage Disabled.') }; + return storage; +}()); + +this.top.location !== this.location && (this.top.location = this.location); + +var DEBUG = window.location.hash === '#DEBUG', + INFO = DEBUG || window.location.hash === '#INFO', + game, + music, + canvas, + ctx, + devicePixelRatio = window.devicePixelRatio || 1, + width = window.innerWidth, + height = window.innerHeight, + muteButtonPosition, + muteButtonProximityThreshold = 30, + playButtonPosition, + playButtonProximityThreshold = 30, + maximumPossibleDistanceBetweenTwoMasses, + highScoreCookieKey = 'tetherHighScore', + highScore = storage.getItem(highScoreCookieKey) ?? 0, + musicMutedCookieKey = 'tetherMusicMuted', + lastDayCookieKey = 'tetherLastDate', + streakCountCookieKey = 'tetherStreakCount', + streakCount = storage.getItem(streakCountCookieKey) ?? 0, + subtitleText = "", + lastDate = new Date(Number(storage.getItem(lastDayCookieKey))), + lastTouchStart, + uidCookieKey = 'tetherId', + uid, + playerRGB = [20, 20, 200], + hslVal = 0, + paused = false, + shouldUnmuteImmediately = false, + cookieExpiryDate = new Date(); + +if (window.location.pathname === '/source/') subtitleText = 'Source Development Mode. #OpenSource'; +else subtitleText = 'Swing around a ball and cause pure destruction.'; + +window.addEventListener('offline', () => { + window.location.href = '/offline/'; +}); + +cookieExpiryDate.setFullYear(cookieExpiryDate.getFullYear() + 50); +var cookieSuffix = '; expires=' + cookieExpiryDate.toUTCString(); + +function extend(base, sub) { + sub.prototype = Object.create(base.prototype); + sub.prototype.constructor = sub; + Object.defineProperty(sub.prototype, 'constructor', { + enumerable: false, + value: sub, + }); +} + +function choice(array) { + return array[Math.floor(Math.random() * array.length)]; +} + +function somewhereInTheViewport() { + return { + x: Math.random() * width, + y: Math.random() * height, + }; +} + +function somewhereJustOutsideTheViewport(buffer) { + var somewhere = somewhereInTheViewport(); + var edgeSeed = Math.random(); + + if (edgeSeed < 0.25) somewhere.x = -buffer; + else if (edgeSeed < 0.5) somewhere.x = width + buffer; + else if (edgeSeed < 0.75) somewhere.y = -buffer; + else somewhere.y = height + buffer; + + return somewhere; +} + +function closestWithinViewport(position) { + var newPos = { x: position.x, y: position.y }; + newPos = forXAndY([newPos, { x: 0, y: 0 }], forXAndY.theGreater); + newPos = forXAndY([newPos, { x: width, y: height }], forXAndY.theLesser); + return newPos; +} + +function getAttributeFromAllObjs(objs, attr) { + var attrs = []; + for (var i = 0; i < objs.length; i++) { + attrs.push(objs[i][attr]); + } + return attrs; +} + +function forXAndY(objs, func) { + return { + x: func.apply(null, getAttributeFromAllObjs(objs, 'x')), + y: func.apply(null, getAttributeFromAllObjs(objs, 'y')), + }; +} + +forXAndY.aPlusHalfB = function (a, b) { + return a + b * 5; +}; +forXAndY.aPlusBTimesSpeed = function (a, b) { + return a + b * game.timeDelta; +}; +forXAndY.subtract = function (a, b) { + return a - b; +}; +forXAndY.invSubtract = function (a, b) { + return b - a; +}; +forXAndY.theGreater = function (a, b) { + return a > b ? a : b; +}; +forXAndY.theLesser = function (a, b) { + return a < b ? a : b; +}; +forXAndY.add = function () { + var s = 0; + for (var i = 0; i < arguments.length; i++) s += arguments[i]; + return s; +}; +forXAndY.multiply = function () { + var p = 1; + for (var i = 0; i < arguments.length; i++) p *= arguments[i]; + return p; +}; + +function randomisedVector(vector, potentialMagnitude) { + var angle = Math.random() * Math.PI * 2; + var magnitude = Math.random() * potentialMagnitude; + return forXAndY([vector, vectorAt(angle, magnitude)], forXAndY.add); +} + +function getIntersection(line1, line2) { + var denominator, + a, + b, + numerator1, + numerator2, + result = { + x: null, + y: null, + onLine1: false, + onLine2: false, + }; + + denominator = + (line2[1].y - line2[0].y) * (line1[1].x - line1[0].x) - + (line2[1].x - line2[0].x) * (line1[1].y - line1[0].y); + + if (denominator === 0) { + return result; + } + + a = line1[0].y - line2[0].y; + b = line1[0].x - line2[0].x; + numerator1 = (line2[1].x - line2[0].x) * a - (line2[1].y - line2[0].y) * b; + numerator2 = (line1[1].x - line1[0].x) * a - (line1[1].y - line1[0].y) * b; + a = numerator1 / denominator; + b = numerator2 / denominator; + + result.x = line1[0].x + a * (line1[1].x - line1[0].x); + result.y = line1[0].y + a * (line1[1].y - line1[0].y); + + if (a > 0 && a < 1) { + result.onLine1 = true; + } + if (b > 0 && b < 1) { + result.onLine2 = true; + } + return result; +} + +function pointInPolygon(point, polygon) { + var i, j; + var c = 0; + var numberOfPoints = polygon.length; + for (i = 0, j = numberOfPoints - 1; i < numberOfPoints; j = i++) { + if ( + ((polygon[i].y <= point.y && point.y < polygon[j].y) || + (polygon[j].y <= point.y && point.y < polygon[i].y)) && + point.x < + ((polygon[j].x - polygon[i].x) * (point.y - polygon[i].y)) / + (polygon[j].y - polygon[i].y) + + polygon[i].x + ) { + c = !c; + } + } + + return c; +} + +function vectorMagnitude(vector) { + return Math.abs( + Math.pow(Math.pow(vector.x, 2) + Math.pow(vector.y, 2), 1 / 2), + ); +} + +function vectorAngle(vector) { + theta = Math.atan(vector.y / vector.x); + if (vector.x < 0) theta += Math.PI; + return theta; +} + +function vectorAt(angle, magnitude) { + return { + x: Math.cos(angle) * magnitude, + y: Math.sin(angle) * magnitude, + }; +} + +function inverseVector(vector) { + var angle = vectorAngle(vector); + var mag = vectorMagnitude(vector); + return vectorAt(angle, 1 / mag); +} + +function linesFromPolygon(polygon) { + var polyLine = []; + for (var i = 1; i < polygon.length; i++) { + polyLine.push([polygon[i - 1], polygon[i]]); + } + return polyLine; +} + +function lineAngle(line) { + return vectorAngle({ + x: line[1].x - line[0].x, + y: line[1].y - line[0].y, + }); +} + +function lineDelta(line) { + return forXAndY(line, forXAndY.invSubtract); +} + +function rgbWithOpacity(rgb, opacity) { + var rgbStrings = []; + for (var i = 0; i < rgb.length; rgbStrings.push(rgb[i++].toFixed(0))); + return 'rgba(' + rgbStrings.join(',') + ',' + opacity.toFixed(2) + ')'; +} + +function hsl(hsl) { + return 'hsl(' + hsl + ', 100%, 50%)'; +} + +function draw(opts) { + for (var defaultKey in draw.defaults) { + if (!(defaultKey in opts)) opts[defaultKey] = draw.defaults[defaultKey]; + } + + if (DEBUG) { + for (var key in opts) { + if (!(key in draw.defaults)) throw key + ' is not a valid option to draw()'; + } + } + + ctx.fillStyle = opts.fillStyle; + ctx.strokeStyle = opts.strokeStyle; + ctx.lineWidth = opts.lineWidth; + + ctx.beginPath(); + + if (opts.type === 'arc') draw.arc(opts); + else if (opts.type === 'line') draw.line(opts); + else if (opts.type === 'text') draw.text(opts); + else if (opts.type === 'rect') draw.rect(opts); + else if (opts.type === 'clear') draw.clear(opts); + else throw opts.type + ' is not an implemented draw type'; + + if (opts.fill) ctx.fill(); + if (opts.stroke) ctx.stroke(); +} + +draw.defaults = { + type: null, + fill: false, + stroke: false, + + linePaths: [], + + arcCenter: undefined, + arcRadius: 0, + arcStart: 0, + arcFinish: 2 * Math.PI, + + text: '', + textPosition: undefined, + fontFamily: 'Tulpen One', + fontFallback: 'sans-serif', + textAlign: 'center', + textBaseline: 'middle', + fontSize: 20, + + rectBounds: [], + + lineWidth: 1, + fillStyle: '#000', + strokeStyle: '#000', +}; + +draw.arc = function (opts) { + ctx.arc( + opts.arcCenter.x, + opts.arcCenter.y, + opts.arcRadius, + opts.arcStart, + opts.arcFinish, + ); +}; + +draw.line = function (opts) { + for (var ipath = 0; ipath < opts.linePaths.length; ipath++) { + var path = opts.linePaths[ipath]; + + ctx.moveTo(path[0].x, path[0].y); + + for (var ipos = 1; ipos < path.length; ipos++) { + var position = path[ipos]; + ctx.lineTo(position.x, position.y); + } + } +}; + +draw.rect = function (opts) { + ctx.fillRect.apply(ctx, opts.rectBounds); +}; + +draw.text = function (opts) { + ctx.font = + opts.fontSize.toString() + + 'px "' + + opts.fontFamily + + '", ' + + opts.fontFallback; + ctx.textAlign = opts.textAlign; + ctx.textBaseline = opts.textBaseline; + + ctx.fillText(opts.text, opts.textPosition.x, opts.textPosition.y); +}; + +draw.clear = function () { + ctx.clearRect(0, 0, width, height); +}; + +function scaleCanvas(ratio) { + canvas.width = width * ratio; + canvas.height = height * ratio; + + ctx.scale(ratio, ratio); +} + +var achievements = { + die: { + name: "You're coming with me", + description: 'Take solace in your mutual destruction', + }, + introduction: { + name: 'How to play', + description: 'Die with one point', + }, + kill: { + name: 'Weapon of choice', + description: 'Kill an enemy without dying yourself', + }, + impact: { + name: 'Concussion', + description: 'Feel the impact', + }, + quickdraw: { + name: 'Quick draw', + description: 'Kill an enemy within a few moments of it spawning', + }, + omnicide: { + name: 'Omnicide', + description: 'Kill every type of enemy in one game', + }, + panic: { + name: 'Panic', + description: 'Be alive while fifteen enemies are on screen', + }, + lowRes: { + name: 'Cramped', + description: + 'Score ten points at 500x500px or less (currently ' + + width + + 'x' + + height + + ')', + }, + handsFree: { + name: 'Hands-free', + description: 'Score five points in a row without moving the tether', + }, +}; + +function initCanvas() { + var later24Hours = lastDate.getTime() + 86400000; + var later48Hours = lastDate.getTime() + 2 * 86400000; + var currentDate = new Date(); + + var streak = Number(storage.getItem(streakCountCookieKey)); + + if ( + !Number(storage.getItem(lastDayCookieKey)) || + Number.isNaN(lastDate) + ) { + saveCookie(lastDayCookieKey, currentDate.getTime()); + saveCookie(streakCountCookieKey, 0); + } else if ( + later48Hours > Number(new Date()) && + Number(new Date()) > later24Hours + ) { + saveCookie(streakCountCookieKey, (streak += 1)); + saveCookie(lastDayCookieKey, currentDate.getTime()); + } else if (Number(new Date()) < later24Hours) { + } else { + saveCookie(streakCountCookieKey, 0); + saveCookie(lastDayCookieKey, currentDate.getTime()); + } + + switch (streak) { + case 0: + break; + case 1: + playerRGB = [206, 125, 165]; + break; + case 2: + playerRGB = [50, 147, 165]; + break; + case 3: + playerRGB = [223, 41, 53]; + break; + case 4: + playerRGB = [223, 41, 53]; + break; + case 5: + playerRGB = [39, 38, 53]; + break; + case 6: + playerRGB = [255, 231, 76]; + break; + case 7: + case 8: + case 9: + playerRGB = [15, 14, 14]; + break; + default: + case 10: + playerRGB = 'Rainbow'; + console.log('Congrats on your 10 day streak!!'); + break; + } + + width = window.innerWidth; + height = window.innerHeight; + muteButtonPosition = { x: 32, y: height - 28 }; + playButtonPosition = { x: 32, y: height - 28 }; + + maximumPossibleDistanceBetweenTwoMasses = vectorMagnitude({ + x: width, + y: height, + }); + + canvas = document.getElementById('game'); + ctx = canvas.getContext('2d'); + + canvas.style.width = width.toString() + 'px'; + canvas.style.height = height.toString() + 'px'; + + canvas.requestPointerLock = + canvas.requestPointerLock || canvas.mozRequestPointerLock; + document.exitPointerLock = + document.exitPointerLock || document.mozExitPointerLock; + + for (var key in storage) { + var value = storage.getItem(key); + if ( + achievements[key] || + key === musicMutedCookieKey || + key === highScoreCookieKey + ) { + saveCookie(key, value); + if (achievements[key]) { + achievements[key].unlocked = new Date(Number(value)); + } + } + } + + scaleCanvas(devicePixelRatio); +} + +window.addEventListener('resize', function (event) { + canvas = document.getElementById('game'); + + width = window.innerWidth; + height = window.innerHeight; + maximumPossibleDistanceBetweenTwoMasses = vectorMagnitude({ + x: width, + y: height, + }); + muteButtonPosition = { x: 32, y: height - 28 }; + playButtonPosition = { x: 32, y: height - 28 }; + devicePixelRatio = window.devicePixelRatio || 1; + + canvas.style.width = width + 'px'; + canvas.style.height = height + 'px'; + + if (!game.started) { + game.tether.teleportTo({ + x: width / 2, + y: (height / 3) * 2, + }); + } + scaleCanvas(devicePixelRatio); +}); + +function timeToNextClaim() { + var deadline = lastDate.getTime() + 86400000; + var timeRemaining = deadline - new Date(); + var formattedTime = new Date(timeRemaining); + + if (formattedTime > 0) { + return `${ + formattedTime.getHours() > 9 ? '' : '0' + }${formattedTime.getHours()}:${ + formattedTime.getMinutes() > 9 ? '' : '0' + }${formattedTime.getMinutes()}:${ + formattedTime.getSeconds() > 9 ? '' : '0' + }${formattedTime.getSeconds()}`; + } else { + return 'Right Now!'; + } +} + +function edgesOfCanvas() { + return linesFromPolygon([ + { x: 0, y: 0 }, + { x: 0, y: height }, + { x: width, y: height }, + { x: width, y: 0 }, + { x: 0, y: 0 }, + ]); +} + +initCanvas(); + +function Music() { + var self = this, + path; + + if (INFO) path = '../tether_theme.mp3'; + else path = '../tether_theme.mp3'; + + self.element = new Audio(path); + + if (typeof self.element.loop === 'boolean') { + if (INFO) console.log('using element.loop for looping'); + self.element.loop = true; + } else { + if (INFO) console.log('using event listener for looping'); + self.element.addEventListener('ended', function () { + self.element.currentTime = 0; + }); + } + + self.timeSignature = 4; + + if (shouldUnmuteImmediately) self.element.play(); +} + +Music.prototype = { + bpm: 90, + url: 'tether_theme.mp3', + delayCompensation: 0.03, + + totalBeat: function () { + return ((this.element.currentTime + this.delayCompensation) / 60) * this.bpm; + }, + + measure: function () { + return this.totalBeat() / this.timeSignature; + }, + + beat: function () { + return music.totalBeat() % this.timeSignature; + }, + + timeSinceBeat: function () { + return this.beat() % 1; + }, +}; + +function Mass() { + this.seed = Math.random(); +} + +Mass.prototype = { + position: { x: 0, y: 0 }, + positionOnPreviousFrame: { x: 0, y: 0 }, + velocity: { x: 0, y: 0 }, + force: { x: 0, y: 0 }, + mass: 1, + lubricant: 1, + radius: 0, + visibleRadius: null, + dashInterval: 1 / 8, + walls: false, + bounciness: 0, + rgb: [60, 60, 60], + reactsToForce: true, + + journeySincePreviousFrame: function () { + return [this.positionOnPreviousFrame, this.position]; + }, + + bounceInDimension: function (d, max) { + var distanceFromFarEdge = max - this.radius - this.position[d]; + var distanceFromNearEdge = this.position[d] - this.radius; + + if (distanceFromNearEdge < 0) { + this.velocity[d] *= -this.bounciness; + this.position[d] = distanceFromNearEdge * this.bounciness + this.radius; + this.bounceCallback(); + } else if (distanceFromFarEdge < 0) { + this.velocity[d] *= -this.bounciness; + this.position[d] = max - distanceFromFarEdge * this.bounciness - this.radius; + this.bounceCallback(); + } + }, + + bounceCallback: function () { }, + + collideWithWalls: function () { + if (!this.walls) return; + this.bounceInDimension('x', width); + this.bounceInDimension('y', height); + }, + + setPosition: function (position) { + this.positionOnPreviousFrame = this.position; + this.position = position; + }, + + teleportTo: function (position) { + this.positionOnPreviousFrame = position; + this.position = position; + }, + + reactToVelocity: function () { + this.setPosition( + forXAndY([this.position, this.velocity], forXAndY.aPlusBTimesSpeed), + ); + this.collideWithWalls(); + }, + + velocityDelta: function () { + var self = this; + return forXAndY([this.force], function (force) { + return force / self.mass; + }); + }, + + reactToForce: function () { + var self = this; + var projectedVelocity = forXAndY( + [this.velocity, this.velocityDelta()], + forXAndY.aPlusBTimesSpeed, + ); + + this.velocity = forXAndY([projectedVelocity], function (projected) { + return projected * Math.pow(self.lubricant, game.timeDelta); + }); + + this.reactToVelocity(); + }, + + step: function () { + if (this.reactsToForce) this.reactToForce(); + }, + + getOpacity: function () { + var opacity; + if (!this.died) opacity = 1; + else opacity = 1 / Math.max(1, game.timeElapsed - this.died); + return opacity; + }, + + getCurrentColor: function () { + if (this.rgb === 'Rainbow') { + if (hslVal !== 360) hslVal++; + else hslVal = 0; + } + + return this.rgb === 'Rainbow' + ? hsl(hslVal) + : rgbWithOpacity(this.rgb, this.getOpacity()); + }, + + draw: function () { + var radius = this.radius; + if (this.visibleRadius !== null) radius = this.visibleRadius; + + draw({ + type: 'arc', + arcRadius: radius, + arcCenter: this.position, + fillStyle: this.getCurrentColor(), + fill: true, + }); + }, + + drawDottedOutline: function () { + for (var i = 0; i < 1; i += this.dashInterval) { + var startAngle = game.timeElapsed / 100 + i * Math.PI * 2; + draw({ + type: 'arc', + stroke: true, + strokeStyle: this.getCurrentColor(), + arcCenter: this.position, + arcStart: startAngle, + arcFinish: startAngle + Math.PI * this.dashInterval * 0.7, + arcRadius: this.radius, + }); + } + }, + + explode: function () { + for (i = 0; i < 50; i++) { + var angle = Math.random() * Math.PI * 2; + var magnitude = Math.random() * 40; + var velocity = forXAndY( + [vectorAt(angle, magnitude), this.velocity], + forXAndY.add, + ); + new FireParticle(this.position, velocity); + } + }, + + focusSegment: function (offset) { + var baseAngle = game.timeElapsed / 30 + Math.cos(game.timeElapsed / 10) * 0.2; + + draw({ + type: 'arc', + stroke: true, + arcCenter: this.position, + arcStart: baseAngle + offset, + arcFinish: baseAngle + Math.PI * 0.5 + offset, + arcRadius: 40 + Math.sin(game.timeElapsed / 10) * 10, + strokeStyle: rgbWithOpacity([0, 0, 0], 0.6), + }); + }, + + focus: function () { + this.focusSegment(0); + this.focusSegment(Math.PI); + }, +}; + +function BackgroundPart(i) { + Mass.call(this); + this.i = i; + this.baseRadius = (2 * Math.max(width, height)) / i; + this.radius = 1; + this.bounciness = 1; + this.velocity = vectorAt(Math.PI * 2 * Math.random(), i * Math.random()); + this.teleportTo(somewhereInTheViewport()); + this.walls = true; +} +extend(Mass, BackgroundPart); + +BackgroundPart.prototype.getCurrentColor = function () { + return this.color; +}; + +BackgroundPart.prototype.step = function () { + this.color = rgbWithOpacity([127, 127, 127], 0.005 * this.i); + + if (game.clickShouldMute && music.element.paused) { + this.color = rgbWithOpacity([255, 255, 255], 0.05 * this.i); + this.visibleRadius = this.baseRadius + Math.random() * this.baseRadius; + } else if (!music.element.paused) { + this.visibleRadius = (1 / music.timeSinceBeat()) * 20 + this.baseRadius; + } else { + this.visibleRadius = this.baseRadius; + } + + Mass.prototype.step.call(this); +}; + +function Background() { + this.parts = []; + for (var i = 0; i < 10; i++) { + this.parts.push(new BackgroundPart(i)); + } +} + +Background.prototype.draw = function () { + if (game.clickShouldMute && music.element.paused) { + draw({ + type: 'rect', + rectBounds: [0, 0, width, height], + fillStyle: rgbWithOpacity([0, 0, 0], 1), + }); + } + + for (var i = 0; i < this.parts.length; this.parts[i++].draw()); +}; + +Background.prototype.step = function () { + for (var i = 0; i < this.parts.length; this.parts[i++].step()); +}; + +function Tether() { + Mass.call(this); + this.radius = 5; + + this.locked = true; + this.unlockable = true; + this.rgb = playerRGB ?? [20, 20, 200]; + + this.teleportTo({ + x: width / 2, + y: (height / 3) * 2, + }); + + this.lastInteraction = null; + this.pointsScoredSinceLastInteraction = 0; + + var self = this; + + document.addEventListener('mousemove', function (e) { + if ( + self.lastInteraction === 'mouse' && + document.pointerLockElement !== canvas + ) + game.lastMousePosition = { x: e.layerX, y: e.layerY }; + self.lastInteraction = 'mouse'; + }); + + document.addEventListener('touchend', function (e) { + self.locked = true; + }); + + function exitTether() { + if ( + document.pointerLockElement === canvas || + document.mozPointerLockElement === canvas + ) + self.locked = false; + else self.locked = true; + } + + if ('onpointerlockchange' in document) + document.addEventListener('pointerlockchange', exitTether); + else if ('onmozpointerlockchange' in document) + document.addEventListener('mozpointerlockchange', exitTether); + + function handleTouch(e) { + e.preventDefault(); + self.lastInteraction = 'touch'; + if (document.pointerLockElement) document.exitPointerLock(); + touch = e.changedTouches[0]; + game.lastMousePosition = { x: touch.clientX, y: touch.clientY }; + } + + document.addEventListener('touchstart', handleTouch, { passive: false }); + document.addEventListener('touchmove', handleTouch, { passive: false }); + + return this; +} +extend(Mass, Tether); + +Tether.prototype.setPosition = function (position) { + if (this.lastInteraction !== 'mouse' || document.pointerLockElement === canvas) + Mass.prototype.setPosition.call(this, position); + if (this.position !== this.positionOnPreviousFrame) { + this.pointsScoredSinceLastInteraction = 0; + } +}; + +Tether.prototype.step = function () { + var leniency = this.lastInteraction === 'touch' ? 50 : 30; + + if ( + this.unlockable && + vectorMagnitude( + forXAndY([this.position, game.lastMousePosition], forXAndY.subtract), + ) < leniency + ) { + if (canvas.requestPointerLock) canvas.requestPointerLock(); + if ( + !(this.lastInteraction !== 'mouse' || document.pointerLockElement === canvas) + ) + return; + + this.locked = false; + + if (!game.started) { + game.start(); + } + } + + if (!this.locked) { + this.setPosition(closestWithinViewport(game.lastMousePosition)); + } else { + this.setPosition(this.position); + } +}; + +Tether.prototype.draw = function () { + if (this.locked && this.unlockable) this.focus(); + Mass.prototype.draw.call(this); +}; + +function Player(tether) { + Mass.call(this); + this.mass = 50; + this.onceGameHasStartedLubricant = 0.99; + this.lubricant = 1; + this.radius = 10; + this.walls = true; + this.teleportTo({ + x: Math.min((width / 10) * 9, width / 2 + 200), + y: 5 * (height / 9), + }); + this.velocity = { x: 0, y: -height / 80 }; + this.bounciness = 0.4; + + this.tether = tether; + this.rgb = playerRGB ?? [20, 20, 200]; +} +extend(Mass, Player); + +Player.prototype.step = function () { + this.force = forXAndY( + [this.tether.position, this.position], + forXAndY.subtract, + ); + Mass.prototype.step.call(this); +}; + +function Cable(tether, player) { + var self = this; + + self.areaCoveredThisStep = function () { + return [ + tether.positionOnPreviousFrame, + player.positionOnPreviousFrame, + player.position, + tether.position, + ]; + }; + + self.line = function () { + return [tether.position, player.position]; + }; + + self.draw = function () { + draw({ + type: 'line', + stroke: true, + strokeStyle: `${ + playerRGB === 'Rainbow' + ? `${hsl(hslVal)}` + : `rgba(${playerRGB[0] ?? 20}, ${playerRGB[1] ?? 20}, ${ + playerRGB[2] ?? 200 + }, 1)` + }`, + linePaths: [self.line()], + }); + + if (DEBUG) self.drawAreaCoveredThisStep(); + }; + + self.drawAreaCoveredThisStep = function () { + draw({ + type: 'line', + fill: true, + fillStyle: rgbWithOpacity([127, 127, 255], 0.3), + linePaths: [self.areaCoveredThisStep()], + }); + }; +} + +function Enemy(opts) { + Mass.call(this); + this.died = null; + this.exhausts = []; + this.spawned = false; + + this.spawnAt = opts.spawnAt; + this.wave = opts.wave; + this.target = this.getTarget(); +} +extend(Mass, Enemy); + +Enemy.prototype.getTarget = function () { + return game.player; +}; + +Enemy.prototype.randomSpawnPosition = function () { + return somewhereInTheViewport(this.radius); +}; + +Enemy.prototype.getTargetVector = function () { + return forXAndY([this.target.position, this.position], forXAndY.subtract); +}; + +Enemy.prototype.step = function () { + if ( + this.force.x !== 0 && + this.force.y !== 0 && + Math.random() < game.timeDelta * vectorMagnitude(this.velocityDelta()) + ) { + new Exhaust(this); + } + + Mass.prototype.step.call(this); +}; + +Enemy.prototype.die = function (playerDeservesAchievement) { + if (this.died) { + if (INFO) console.log('tried to kill enemy that already died'); + return; + } + if (playerDeservesAchievement) { + unlockAchievement('kill'); + + var name = this.constructor.name; + + if (game.enemyTypesKilled.indexOf(name) === -1) { + game.enemyTypesKilled.push(name); + if (INFO) console.log(game.enemyTypesKilled); + if (game.enemyTypesKilled.length === enemyPool.length) { + unlockAchievement('omnicide'); + } + } + + if (this.died - this.spawnAt < 5) unlockAchievement('quickdraw'); + } + this.explode(); + this.died = game.timeElapsed; + if (game.ended) return; + + game.incrementScore(1); +}; + +Enemy.prototype.draw = function () { + if (DEBUG && !this.died) this.drawTargetVector(); + + Mass.prototype.draw.call(this); +}; + +Enemy.prototype.drawTargetVector = function () { + draw({ + type: 'line', + stroke: true, + strokeStyle: rgbWithOpacity([255, 127, 127], 0.7), + linePaths: [[this.position, this.target.position]], + }); +}; + +Enemy.prototype.drawWarning = function () { + var timeUntilSpawn = + (this.spawnAt - game.timeElapsed) / this.wave.spawnWarningDuration; + + draw({ + type: 'arc', + stroke: true, + arcCenter: this.position, + arcRadius: + (this.visibleRadius || this.radius) / 2 + Math.pow(timeUntilSpawn, 2) * 700, + lineWidth: + ((2 * (this.visibleRadius || this.radius)) / 2) * + Math.pow(1 - timeUntilSpawn, 3), + strokeStyle: rgbWithOpacity( + this.rgbWarning || this.rgb, + (1 - timeUntilSpawn) * this.getOpacity(), + ), + }); +}; + +function Drifter(opts) { + Enemy.call(this, opts); + this.radius = 10; + this.rgb = [30, 150, 150]; + this.thrustAngle = undefined; + this.walls = true; + this.bounciness = 1; + this.power = 0.3; + this.lubricant = 0.8; + this.curvature = Math.max(width, height); +} +extend(Enemy, Drifter); + +Drifter.prototype.getTarget = function () { + return game.tether; +}; + +Drifter.prototype.randomSpawnPosition = function () { + var somewhere = somewhereInTheViewport(); + somewhere.x = (somewhere.x * 2) / 3 + width / 6; + somewhere.y = (somewhere.y * 2) / 3 + height / 6; + return somewhere; +}; + +Drifter.prototype.step = function () { + if (this.thrustAngle === undefined) { + this.thrustAngle = vectorAngle(this.getTargetVector()); + + var error = Math.random() + 1; + if (Math.random() > 0.5) error *= -1; + this.thrustAngle += error / 5; + } + + if (!this.died) { + this.force = vectorAt(this.thrustAngle, this.power); + } else this.force = { x: 0, y: 0 }; + + Enemy.prototype.step.call(this); +}; + +Drifter.prototype.bounceCallback = function () { + this.thrustAngle = vectorAngle(this.velocity); +}; + +function Eye(opts) { + Enemy.call(this, opts); + + var size = opts.size || 0.75 + Math.random() / 1.5; + + this.mass = size * (1500 / maximumPossibleDistanceBetweenTwoMasses); + + this.lubricant = 0.9; + this.radius = size * 10; + this.shadowRadius = this.radius + 3; + this.shadowOpacity = 0.5; + this.rgb = [255, 255, 255]; + this.rgbWarning = [50, 50, 50]; +} +extend(Enemy, Eye); + +Eye.prototype.step = function () { + if (!this.died) { + var targetVector = this.getTargetVector(); + targetVectorMagnitude = vectorMagnitude(targetVector); + this.force = forXAndY([targetVector], function (target) { + return target * (1 / targetVectorMagnitude); + }); + } else this.force = { x: 0, y: 0 }; + + Enemy.prototype.step.call(this); +}; + +Eye.prototype.getRelativeDistance = function () { + var targetVector = this.getTargetVector(); + return vectorMagnitude(targetVector) / maximumPossibleDistanceBetweenTwoMasses; +}; + +Eye.prototype.getCalmness = function () { + return 1 / Math.pow(1 / this.getRelativeDistance(), 1 / 4); +}; + +Eye.prototype.drawWarning = function () { + var timeUntilSpawn = + (this.spawnAt - game.timeElapsed) / this.wave.spawnWarningDuration; + + draw({ + type: 'arc', + stroke: true, + lineWidth: ((2 * this.shadowRadius) / 2) * Math.pow(1 - timeUntilSpawn, 3), + strokeStyle: rgbWithOpacity( + this.rgbWarning || this.rgb, + (1 - timeUntilSpawn) * this.getOpacity() * this.shadowOpacity, + ), + arcCenter: this.position, + arcRadius: this.shadowRadius / 2 + Math.pow(timeUntilSpawn, 2) * 700, + }); +}; + +Eye.prototype.getIrisColor = function () { + var red = 0; + if (Math.random() < Math.pow(1 - this.getCalmness(), 4) * game.timeDelta) + red = 255; + return rgbWithOpacity([red, 0, 0], this.getOpacity()); +}; + +Eye.prototype.awakeness = function () { + var timeAlive = game.timeElapsed - this.spawnAt; + return 1 - 1 / (timeAlive / 3 + 1); +}; + +Eye.prototype.drawIris = function () { + var awakeness = this.awakeness(); + var targetVector = this.getTargetVector(); + var relativeDistance = this.getRelativeDistance(); + + var irisVector = vectorAt( + vectorAngle(targetVector), + awakeness * this.radius * Math.pow(relativeDistance, 1 / 2) * 0.7, + ); + + var centreOfIris = forXAndY([this.position, irisVector], forXAndY.add); + + var irisRadius = ((this.radius * 1) / 3) * awakeness; + + draw({ + type: 'arc', + fill: true, + fillStyle: this.getIrisColor(), + arcCenter: centreOfIris, + arcRadius: irisRadius, + }); +}; + +Eye.prototype.draw = function () { + draw({ + type: 'arc', + fill: true, + fillStyle: rgbWithOpacity([0, 0, 0], this.getOpacity() * this.shadowOpacity), + arcCenter: this.position, + arcRadius: this.shadowRadius, + }); + + this.visibleRadius = this.radius * Math.pow(this.awakeness(), 1 / 6); + Enemy.prototype.draw.call(this); + + if (this.died) return; + + this.drawIris(); +}; + +function Twitchy(opts) { + Enemy.call(this, opts); + this.charging = false; + + this.mass = 100; + this.lubricant = 0.92; + this.chargeRate = 0.01; + this.dischargeRate = 0.1; + this.radius = 5; + + this.fuel = 0.9; + this.rgbDischarging = [200, 30, 30]; + this.rgbWarning = this.rgbDischarging; +} +extend(Enemy, Twitchy); + +Twitchy.prototype.step = function () { + if (this.died || this.charging) { + this.force = { x: 0, y: 0 }; + if (this.charging) { + this.fuel += game.timeDelta * this.chargeRate; + if (this.fuel >= 1) { + this.fuel = 1; + this.charging = false; + } + } + } else { + this.force = this.getTargetVector(); + this.fuel -= game.timeDelta * this.dischargeRate; + + if (this.fuel <= 0) { + this.fuel = 0; + this.charging = true; + } + } + + Enemy.prototype.step.call(this); +}; + +Twitchy.prototype.getCurrentColor = function () { + if (this.charging) { + var brightness = 255; + var whiteness = Math.pow(this.fuel, 1 / 40); + + if (0.98 < this.fuel || (0.94 < this.fuel && this.fuel < 0.96)) { + brightness = 0; + } + + this.rgb = [brightness, brightness * whiteness, brightness * whiteness]; + } else this.rgb = this.rgbDischarging; + + return Enemy.prototype.getCurrentColor.call(this); +}; + +Twitchy.prototype.draw = function () { + if (this.charging && this.fuel >= 0) { + draw({ + type: 'arc', + fill: true, + fillStyle: rgbWithOpacity([30, 30, 30], this.getOpacity() * this.fuel), + arcRadius: (this.radius * 1.2) / this.fuel, + arcCenter: this.position, + }); + } + + Enemy.prototype.draw.call(this); +}; + +function Particle() { + Mass.call(this); + game.particles.push(this); +} +extend(Mass, Particle); +Particle.prototype.isWorthDestroying = function () { + return Math.abs(this.velocity.x) < 0.001 && Math.abs(this.velocity.y) < 0.001; +}; + +function FireParticle(position, velocity) { + Particle.call(this); + this.lubricant = 0.9; + this.created = game.timeElapsed; + this.teleportTo(position); + this.velocity = velocity; + this.red = 1; + this.green = 1; + this.blue = 0; + this.opacity = 1; + + this.initialIntensity = velocity.x * (2 * Math.random()); +} +extend(Particle, FireParticle); + +FireParticle.prototype.getCurrentColor = function () { + var intensity = this.velocity.x / this.initialIntensity; + return rgbWithOpacity( + this.rgbForIntensity(intensity), + Math.pow(intensity, 0.25) * this.opacity, + ); +}; + +FireParticle.prototype.rgbForIntensity = function (intensity) { + return [Math.pow(intensity, 0.2) * 255, intensity * 200, 0]; +}; + +FireParticle.prototype.draw = function () { + if (Math.random() < 0.1 * game.timeDelta) return; + + var timeAlive = game.timeElapsed - this.created; + var maturity = 1 - 1 / (timeAlive / 3 + 1); + var velocityButSmallerWhenYoung = forXAndY( + [this.velocity, { x: maturity, y: maturity }], + forXAndY.multiply, + ); + + draw({ + type: 'line', + stroke: true, + strokeStyle: this.getCurrentColor(), + linePaths: [ + [ + this.position, + forXAndY([this.position, velocityButSmallerWhenYoung], forXAndY.aPlusHalfB), + ], + ], + }); +}; + +function Exhaust(source) { + var position = source.position; + + var delta = source.velocityDelta(); + var baseVelocity = forXAndY([source.velocity, delta], function (v, d) { + return 0.3 * v - d * 20; + }); + + var deltaMagnitude = vectorMagnitude(delta); + var velocity = forXAndY([baseVelocity], function (b) { + return b * (1 + (Math.random() - 0.5) * (0.8 + deltaMagnitude * 0.1)); + }); + + FireParticle.call(this, position, velocity); + + this.opacity = 0.7; +} +extend(FireParticle, Exhaust); + +Exhaust.prototype.rgbForIntensity = function (intensity) { + return [intensity * 200, 50 + intensity * 100, 50 + intensity * 100]; +}; + +function TeleportDust(source) { + var randomDelta = vectorAt( + Math.random() * Math.PI * 2, + Math.random() * source.radius * 0.1, + ); + + var velocityMultiplier = (Math.random() * 1) / 10; + var baseVelocity = forXAndY( + [source.teleportDelta, { x: velocityMultiplier, y: velocityMultiplier }], + forXAndY.multiply, + ); + var velocity = forXAndY([baseVelocity, randomDelta], forXAndY.add); + + var distanceFromStart = Math.random(); + var vectorFromStart = forXAndY( + [source.teleportDelta, { x: distanceFromStart, y: distanceFromStart }], + forXAndY.multiply, + ); + var basePosition = forXAndY([source.position, vectorFromStart], forXAndY.add); + var position = forXAndY([basePosition, randomDelta], forXAndY.add); + + FireParticle.call(this, position, velocity); +} +extend(FireParticle, TeleportDust); + +TeleportDust.prototype.rgbForIntensity = function (intensity) { + return [100 + intensity * 100, intensity * 200, 60 + intensity * 150]; +}; + +function Wave() { + this.enemies = []; + this.complete = false; + this.doneSpawningEnemies = false; + this.spawnWarningDuration = 50; + this.boredomCompensation = 0; + this.startedAt = game.timeElapsed; +} + +Wave.prototype.step = function () { + this.spawnEnemies(); + + this.remainingLivingEnemies = 0; + + for (var i = 0; i < this.enemies.length; i++) { + var enemy = this.enemies[i]; + if (enemy.spawned) enemy.step(); + else if (enemy.spawnAt <= game.timeElapsed) enemy.spawned = true; + + if (!enemy.died) this.remainingLivingEnemies++; + } + + if (this.remainingLivingEnemies >= 15) unlockAchievement('panic'); + if ( + this.doneSpawningEnemies && + this.remainingLivingEnemies === 0 && + !this.hasEnemiesWorthDrawing + ) + this.complete = true; +}; + +Wave.prototype.draw = function () { + this.hasEnemiesWorthDrawing = false; + + for (var i = 0; i < this.enemies.length; i++) { + var enemy = this.enemies[i]; + var opacity = enemy.getOpacity(); + if (opacity > 0.01) { + if (enemy.spawned) enemy.draw(); + else enemy.drawWarning(); + + this.hasEnemiesWorthDrawing = true; + } + } +}; + +Wave.prototype.spawnEnemies = function () { + if (this.doneSpawningEnemies) return; + + var remaininUnspawnedEnemies = 0; + var totalDelay = this.boredomCompensation; + var compensatedForBoredom = false; + + for (var i = 0; i < this.spawns.length; i++) { + var spawn = this.spawns[i]; + + totalDelay += spawn.delay; + + if (spawn.spawned) continue; + + var timeUntilSpawn = totalDelay - (game.timeElapsed - this.startedAt); + + if (!compensatedForBoredom && this.remainingLivingEnemies === 0) { + compensatedForBoredom = true; + this.boredomCompensation += timeUntilSpawn; + timeUntilSpawn -= this.boredomCompensation; + } + + if (timeUntilSpawn <= 0) { + var opts = spawn.opts || {}; + + opts.spawnAt = game.timeElapsed + this.spawnWarningDuration; + opts.wave = this; + + var enemy = new spawn.type(opts); + + if (spawn.pos) { + enemy.teleportTo({ + x: spawn.pos[0] * width, + y: spawn.pos[1] * height, + }); + } else enemy.teleportTo(enemy.randomSpawnPosition()); + + this.enemies.push(enemy); + + spawn.spawned = true; + } else { + remaininUnspawnedEnemies++; + } + } + + if (remaininUnspawnedEnemies === 0) this.doneSpawningEnemies = true; +}; + +function tutorialFor(enemyType, enemyOpts) { + function Tutorial() { + Wave.call(this); + this.spawns = [ + { + delay: 0, + type: enemyType, + pos: [1 / 2, 1 / 5], + opts: enemyOpts || {}, + }, + ]; + } + extend(Wave, Tutorial); + return Tutorial; +} + +function aBunchOf(enemyType, count, interval) { + function ABunch() { + Wave.call(this); + this.spawns = []; + + for (var i = 0; i < count; i++) { + this.spawns.push({ + delay: interval * (i + 1), + type: enemyType, + }); + } + } + extend(Wave, ABunch); + return ABunch; +} + +function autoWave(difficulty) { + var totalSpawns; + var localEnemyPool; + + if (difficulty % 2) { + totalSpawns = 15 + difficulty; + localEnemyPool = enemyPool; + } else { + localEnemyPool = [enemyPool[(difficulty / 2) % enemyPool.length]]; + totalSpawns = 10 + difficulty; + } + + function AutoWave() { + Wave.call(this); + this.spawns = []; + + for (var i = 0; i < totalSpawns; i++) { + this.spawns.push({ + delay: (Math.pow(Math.random(), 1 / 2) * 400) / (difficulty + 7), + type: choice(localEnemyPool), + }); + } + } + + extend(Wave, AutoWave); + return AutoWave; +} + +function saveCookie(key, value) { + storage.setItem(key, value); + document.cookie = key + '=' + value + cookieSuffix; +} + +function unlockAchievement(slug) { + var achievement = achievements[slug]; + if (!achievement.unlocked) { + achievement.unlocked = new Date(); + saveCookie(slug, achievement.unlocked.getTime().toString()); + } +} + +function logScore(score) { + if (score > highScore) { + highScore = score; + saveCookie(highScoreCookieKey, score.toString()); + } +} + +function getUnlockedAchievements(invert) { + var unlockedAchievements = []; + invert = invert || false; + + for (var key in achievements) { + var achievement = achievements[key]; + if (invert ^ (achievement.unlocked !== undefined)) + unlockedAchievements.push(achievement); + } + + return unlockedAchievements; +} + +function getLockedAchievements() { + return getUnlockedAchievements(true); +} + +function Game() { + var self = this; + + self.lastMousePosition = { x: NaN, y: NaN }; + + self.reset = function (waveIndex) { + if (document.pointerLockElement) document.exitPointerLock(); + + self.background = new Background(); + self.ended = null; + self.score = 0; + self.enemyTypesKilled = []; + self.lastPointScoredAt = 0; + self.timeElapsed = 0; + self.normalSpeed = 0.04; + self.slowSpeed = self.normalSpeed / 100; + self.setSpeed(self.normalSpeed); + + self.started = false; + + self.waveIndex = waveIndex || 0; + self.waves = [ + tutorialFor(Drifter), + aBunchOf(Drifter, 2, 5), + + tutorialFor(Eye, { size: 1.5 }), + aBunchOf(Eye, 4, 100), + aBunchOf(Eye, 5, 10), + + tutorialFor(Twitchy), + aBunchOf(Twitchy, 4, 50), + aBunchOf(Twitchy, 5, 10), + ]; + self.wave = undefined; + + self.particles = []; + + self.tether = new Tether(); + self.player = new Player(self.tether); + self.cable = new Cable(self.tether, self.player); + }; + + self.setSpeed = function (speed) { + self.speed = speed; + }; + + self.start = function () { + self.tether.locked = false; + self.player.lubricant = self.player.onceGameHasStartedLubricant; + self.started = true; + self.timeElapsed = 0; + }; + + self.pickNextWave = function () { + var waveType = self.waves[self.waveIndex++]; + + if (waveType === undefined) { + waveType = autoWave(self.waveIndex - self.waves.length); + } + + self.wave = new waveType(); + }; + + self.incrementScore = function (incr) { + self.lastPointScoredAt = self.timeElapsed; + self.score += incr; + self.tether.pointsScoredSinceLastInteraction += incr; + + if (self.score >= 10 && width <= 500 && height <= 500) { + unlockAchievement('lowRes'); + } + + if (self.tether.pointsScoredSinceLastInteraction >= 5) { + unlockAchievement('handsFree'); + } + }; + + self.getIntensity = function () { + return 1 / (1 + (self.timeElapsed - self.lastPointScoredAt)); + }; + + self.stepParticles = function () { + for (var i = 0; i < self.particles.length; i++) { + if (self.particles[i] === undefined) { + continue; + } else if (self.particles[i].isWorthDestroying()) { + delete self.particles[i]; + } else { + self.particles[i].step(); + } + } + }; + + self.step = function () { + if (DEBUG) draw({ type: 'clear' }); + + var now = new Date().getTime(); + + if (!self.lastStepped) { + self.lastStepped = now; + return; + } else { + self.realTimeDelta = now - self.lastStepped; + + self.timeDelta = Math.min(self.realTimeDelta, 1000 / 20) * self.speed; + + self.timeElapsed += self.timeDelta; + self.lastStepped = now; + } + + if (isNaN(self.lastMousePosition.x)) { + self.proximityToMuteButton = maximumPossibleDistanceBetweenTwoMasses; + self.proximityToPlayButton = maximumPossibleDistanceBetweenTwoMasses; + } else { + self.proximityToMuteButton = vectorMagnitude( + forXAndY([muteButtonPosition, self.lastMousePosition], forXAndY.subtract), + ); + self.proximityToPlayButton = vectorMagnitude( + forXAndY([playButtonPosition, self.lastMousePosition], forXAndY.subtract), + ); + } + self.clickShouldMute = + (!self.started || self.ended) && + self.proximityToMuteButton < muteButtonProximityThreshold + ? true + : false; + self.clickShouldPlay = + self.started && + !self.ended && + self.proximityToPlayButton < playButtonProximityThreshold + ? true + : false; + if (self.clickShouldMute !== canvas.classList.contains('buttonhover')) + canvas.classList.toggle('buttonhover'); + if (self.clickShouldPlay !== canvas.classList.contains('buttonhover')) + canvas.classList.toggle('buttonhover'); + + self.background.step(); + self.tether.step(); + self.player.step(); + + if (self.started) { + if (self.wave === undefined || self.wave.complete) self.pickNextWave(); + self.wave.step(); + + if (!self.ended) self.checkForEnemyContact(); + self.checkForCableContact(); + } + + self.stepParticles(); + + self.draw(); + }; + + self.checkForCableContact = function () { + var cableAreaCovered = self.cable.areaCoveredThisStep(); + + for (var i = 0; i < self.wave.enemies.length; i++) { + var enemy = self.wave.enemies[i]; + + if (enemy.died || !enemy.spawned) { + continue; + } + + var journey = enemy.journeySincePreviousFrame(); + var cableLines = linesFromPolygon(cableAreaCovered); + + if (pointInPolygon(enemy.position, cableAreaCovered)) { + enemy.die(true); + continue; + } + + for (var ci = 0; ci < cableLines.length; ci++) { + var intersection = getIntersection(journey, cableLines[ci]); + + if (intersection.onLine1 && intersection.onLine2) { + enemy.position = intersection; + enemy.die(true); + break; + } + } + } + }; + + self.checkForEnemyContactWith = function (mass) { + var massPositionDelta = lineDelta([ + mass.positionOnPreviousFrame, + mass.position, + ]); + + var colChecks = []; + + for (var i = 0; i < self.wave.enemies.length; i++) { + var enemy = self.wave.enemies[i]; + + if (enemy.died || !enemy.spawned) { + continue; + } + + var enemyPositionDelta = lineDelta([ + enemy.positionOnPreviousFrame, + enemy.position, + ]); + + for ( + var progress = 0; + progress < 1; + progress += + Math.min(enemy.radius, mass.radius) / + (3 * + Math.max( + enemyPositionDelta.x, + enemyPositionDelta.y, + massPositionDelta.x, + massPositionDelta.y, + 1, + )) + ) { + enemyPosition = { + x: enemy.positionOnPreviousFrame.x + enemyPositionDelta.x * progress, + y: enemy.positionOnPreviousFrame.y + enemyPositionDelta.y * progress, + }; + + massPosition = { + x: mass.positionOnPreviousFrame.x + massPositionDelta.x * progress, + y: mass.positionOnPreviousFrame.y + massPositionDelta.y * progress, + }; + + if (INFO) this.collisionChecks += 1; + if (DEBUG) colChecks.push([enemyPosition, massPosition]); + + var distance = lineDelta([enemyPosition, massPosition]); + + if ( + Math.pow(distance.x, 2) + Math.pow(distance.y, 2) < + Math.pow(enemy.radius + mass.radius, 2) + ) { + enemy.position = enemyPosition; + mass.position = massPosition; + enemy.die(false); + + if (mass === this.player) { + var relativeVelocity = lineDelta([mass.velocity, enemy.velocity]); + var impact = + vectorMagnitude(relativeVelocity) / + maximumPossibleDistanceBetweenTwoMasses; + + if (impact > 0.04) unlockAchievement('impact'); + if (INFO) console.log('impact: ' + impact.toString()); + } + + return mass; + } + } + } + + if (DEBUG) + draw({ + type: 'line', + stroke: true, + linePaths: colChecks, + strokeStyle: rgbWithOpacity([0, 127, 0], 0.3), + }); + }; + + self.checkForEnemyContact = function () { + if (INFO) this.collisionChecks = 0; + var deadMass = + self.checkForEnemyContactWith(self.tether) || + self.checkForEnemyContactWith(self.player); + if (deadMass) { + deadMass.rgb = [200, 20, 20]; + deadMass.explode(); + unlockAchievement('die'); + if (game.score === 1) unlockAchievement('introduction'); + game.end(); + } + }; + + self.drawScore = function () { + if (self.score === 0) return; + + var intensity = self.getIntensity(); + + draw({ + type: 'text', + text: self.score.toString(), + fontSize: intensity * height * 5, + fillStyle: rgbWithOpacity([0, 0, 0], intensity), + textPosition: { x: width / 2, y: height / 2 }, + }); + }; + + self.drawParticles = function () { + for (var i = 0; i < this.particles.length; i++) { + if (this.particles[i] !== undefined) { + this.particles[i].draw(); + } + } + }; + + self.drawLogo = function () { + var opacity = game.started ? Math.pow(1 - game.timeElapsed / 50, 3) : 1; + if (opacity < 0.001) return; + + draw({ + type: 'text', + text: 'tether!', + fillStyle: rgbWithOpacity([0, 0, 0], opacity), + fontSize: 100, + textPosition: { + x: width / 2, + y: height / 3, + }, + }); + + draw({ + type: 'text', + text: subtitleText ?? 'Swing around a ball and cause pure destruction.', + fillStyle: rgbWithOpacity([0, 0, 0], opacity), + fontSize: 30, + textPosition: { + x: width / 2, + y: height / 3 + 55, + }, + }); + + draw({ + type: 'text', + text: + ({ touch: 'tap', mouse: 'click' }[self.tether.lastInteraction] ?? 'click') + + ' to start', + fillStyle: rgbWithOpacity([0, 0, 0], opacity), + fontSize: 24, + textPosition: { + x: width / 2, + y: (height / 4) * 3 + 80, + }, + }); + }; + + self.drawRestartTutorial = function () { + if (!self.ended) return; + + var opacity = -Math.sin((game.timeElapsed - game.ended) * 3); + if (opacity < 0) opacity = 0; + + var fontSize = Math.min(width / 5, height / 8); + + draw({ + type: 'text', + text: + ({ touch: 'tap', mouse: 'click' }[self.tether.lastInteraction] ?? 'click') + + ' to retry', + fontSize: fontSize, + textPosition: { x: width / 2, y: height / 2 - fontSize / 2 }, + fillStyle: rgbWithOpacity([0, 0, 0], opacity), + }); + }; + + self.drawAchievementNotifications = function () { + var now = new Date().getTime(); + var recentAchievements = []; + var animationDuration = 7000; + + for (var slug in achievements) { + var achievement = achievements[slug]; + if (achievement.unlocked === undefined) continue; + + var unlocked = achievement.unlocked.getTime(); + + if (now > unlocked && now < unlocked + animationDuration) { + recentAchievements.push(achievement); + } + } + + for (var i = 0; i < recentAchievements.length; i++) { + var recentAchievement = recentAchievements[i]; + var progress = (now - recentAchievement.unlocked) / animationDuration; + + var visibility = 1; + var buffer = 0.2; + + var easing = 6; + + if (progress < buffer) visibility = Math.pow(progress / buffer, 1 / easing); + else if (progress > 1 - buffer) + visibility = Math.pow((1 - progress) / buffer, easing); + + var sink = -50 * (1 - visibility); + var notificationHeight = 60; + var baseNotificationHeight = 20 + notificationHeight * i; + + var drawArgs = { + type: 'text', + text: 'Achievement Unlocked', + textAlign: 'right', + textBaseline: 'top', + fillStyle: rgbWithOpacity([0, 0, 0], visibility), + fontFamily: 'Quantico', + fontSize: 17, + textPosition: { + x: width - 25, + y: visibility * baseNotificationHeight + sink, + }, + }; + + draw(drawArgs); + + drawArgs.fontSize = 25; + drawArgs.text = recentAchievement.name; + drawArgs.textPosition = { + x: width - 25, + y: 19 + visibility * baseNotificationHeight + sink, + }; + + draw(drawArgs); + } + }; + + self.drawAchievements = function ( + achievementList, + fromBottom, + fromRight, + headingText, + fillStyle, + ) { + if (achievementList.length === 0) return fromBottom; + + var drawOpts = { + type: 'text', + fillStyle: fillStyle, + textAlign: 'right', + fontFamily: 'Quantico', + textBaseline: 'alphabetic', + }; + var xPos = width - fromRight; + + for (var i = 0; i < achievementList.length; i++) { + var achievement = achievementList[i]; + + drawOpts.text = achievement.name; + drawOpts.fontSize = 18; + drawOpts.textPosition = { x: xPos, y: height - fromBottom - 16 }; + draw(drawOpts); + + drawOpts.text = achievement.description; + drawOpts.fontSize = 13; + drawOpts.textPosition = { x: xPos, y: height - fromBottom }; + draw(drawOpts); + + fromBottom += 45; + } + + drawOpts.text = headingText; + drawOpts.fontSize = 20; + drawOpts.textPosition = { x: xPos, y: height - fromBottom }; + draw(drawOpts); + + fromBottom += 55; + return fromBottom; + }; + + self.drawPauseMessage = function () { + var fontSize = Math.min(width / 5, height / 8); + draw({ + type: 'text', + text: + ({ touch: 'tap', mouse: 'click' }[self.tether.lastInteraction] ?? 'click') + + ' to unpause', + fillStyle: '#000', + fontSize: fontSize, + textPosition: { + x: width / 2, + y: height / 2 - fontSize / 2, + }, + }); + }; + + self.drawAchievementUI = function () { + var unlockedAchievements = getUnlockedAchievements(); + if (unlockedAchievements.length > 0) { + var indicatedPosition = { x: 0, y: 0 }; + if (isNaN(game.lastMousePosition.x)) { + indicatedPosition = { x: 0, y: 0 }; + } else { + indicatedPosition = game.lastMousePosition; + } + var distanceFromCorner = vectorMagnitude( + lineDelta([indicatedPosition, { x: width, y: height }]), + ); + var distanceRange = [ + maximumPossibleDistanceBetweenTwoMasses / 10, + maximumPossibleDistanceBetweenTwoMasses / 4, + ]; + var hintOpacity; + + if (distanceFromCorner > distanceRange[1]) hintOpacity = 1; + else if (distanceFromCorner > distanceRange[0]) + hintOpacity = + (distanceFromCorner - distanceRange[0]) / + (distanceRange[1] - distanceRange[0]); + else hintOpacity = 0; + + var listingOpacity = 1 - hintOpacity; + + draw({ + type: 'text', + text: 'Achievements…', + fillStyle: (fillStyle = rgbWithOpacity([0, 0, 0], hintOpacity)), + fontSize: 16, + textPosition: { x: width - 5, y: height - 8 }, + textAlign: 'right', + textBaseline: 'alphabetic', + fontFamily: 'Quantico', + }); + + if (highScore) { + draw({ + type: 'text', + text: 'Best Score: ' + highScore.toString(), + fillStyle: (fillStyle = rgbWithOpacity([0, 0, 0], hintOpacity)), + fontSize: 16, + textPosition: { x: width - 6, y: height - 56 }, + textAlign: 'right', + textBaseline: 'bottom', + fontFamily: 'Quantico', + }); + } + + draw({ + type: 'text', + text: 'Login Streak: ' + streakCount.toString(), + fillStyle: (fillStyle = rgbWithOpacity([0, 0, 0], hintOpacity)), + fontSize: 16, + textPosition: { x: width - 6, y: height - 38 }, + textAlign: 'right', + textBaseline: 'bottom', + fontFamily: 'Quantico', + }); + + draw({ + type: 'text', + text: 'Next Day: ' + timeToNextClaim(), + fillStyle: (fillStyle = rgbWithOpacity([0, 0, 0], hintOpacity)), + fontSize: 16, + textPosition: { x: width - 6, y: height - 20 }, + textAlign: 'right', + textBaseline: 'bottom', + fontFamily: 'Quantico', + }); + + draw({ + type: 'rect', + rectBounds: [0, 0, width, height], + fillStyle: rgbWithOpacity([255, 255, 255], listingOpacity * 0.9), + }); + + var heightNeeded = 500; + var widthNeeded = 500; + var fromBottom = + ((game.lastMousePosition.y - height) / height) * heightNeeded + 40; + var fromRight = + ((game.lastMousePosition.x - width) / width) * widthNeeded + 35; + fromBottom = this.drawAchievements( + getLockedAchievements(), + fromBottom, + fromRight, + 'Locked', + rgbWithOpacity([0, 0, 0], listingOpacity * 0.5), + ); + this.drawAchievements( + unlockedAchievements, + fromBottom, + fromRight, + 'Unlocked', + rgbWithOpacity([0, 0, 0], listingOpacity), + ); + } + }; + + self.eventShouldMute = function (e) { + var position; + + if (e.changedTouches) { + var touch = e.changedTouches[0]; + position = { x: touch.pageX, y: touch.pageY }; + } else { + position = { x: e.layerX, y: e.layerY }; + } + + return self.positionShouldMute(position); + }; + + self.positionShouldMute = function (position) { + if (self.started || self.ended) return false; + self.proximityToMuteButton = vectorMagnitude( + forXAndY([muteButtonPosition, position], forXAndY.subtract), + ); + return self.proximityToMuteButton < muteButtonProximityThreshold; + }; + + self.eventShouldPlay = function (e) { + var position; + + if (e.changedTouches) { + var touch = e.changedTouches[0]; + position = { x: touch.pageX, y: touch.pageY }; + } else { + position = game.lastMousePosition || { x: e.layerX, y: e.layerY }; + } + + return self.positionShouldPlay(position); + }; + + self.positionShouldPlay = function (position) { + if (!(self.started && !self.ended)) return false; + if (paused) return true; + self.proximityToPlayButton = vectorMagnitude( + forXAndY([playButtonPosition, position], forXAndY.subtract), + ); + return self.proximityToPlayButton < playButtonProximityThreshold; + }; + + self.drawMuteButton = function () { + if (!self.clickShouldMute && music.element.paused) { + xNoise = (Math.random() - 0.5) * (500 / self.proximityToMuteButton); + yNoise = (Math.random() - 0.5) * (500 / self.proximityToMuteButton); + visiblePosition = { + x: xNoise + muteButtonPosition.x, + y: yNoise + muteButtonPosition.y + Math.sin(new Date().getTime() / 250) * 3, + }; + } else { + visiblePosition = { x: muteButtonPosition.x, y: muteButtonPosition.y }; + } + + if (!music.element.paused) { + visiblePosition.x = visiblePosition.x - 5; + visiblePosition.y = visiblePosition.y - 2; + } + + var opacity = 1; + + if (self.clickShouldMute && !music.element.paused) opacity = 0.5; + + draw({ + type: 'text', + text: music.element.paused ? '\uf025' : '\uf026', + fontFamily: 'FontAwesome', + fontSize: 30, + textAlign: 'center', + textBaseline: 'middle', + fillStyle: rgbWithOpacity([0, 0, 0], opacity), + textPosition: visiblePosition, + }); + }; + + self.drawPlayButton = function () { + if (!self.clickShouldPlay && paused) { + xNoise = (Math.random() - 0.5) * (500 / self.proximityToPlayButton); + yNoise = (Math.random() - 0.5) * (500 / self.proximityToPlayButton); + visiblePosition = { + x: xNoise + playButtonPosition.x, + y: yNoise + playButtonPosition.y + Math.sin(new Date().getTime() / 250) * 3, + }; + } else { + visiblePosition = { x: playButtonPosition.x, y: playButtonPosition.y }; + } + + var opacity = 1; + + if (self.clickShouldPlay && !paused) opacity = 0.5; + + draw({ + type: 'text', + text: paused ? '\uf04b' : '\uf04c', + fontFamily: 'FontAwesome', + fontSize: 30, + textAlign: 'center', + textBaseline: 'middle', + fillStyle: rgbWithOpacity([0, 0, 0], opacity), + textPosition: visiblePosition, + }); + }; + + self.drawInfo = function () { + var fromBottom = 7; + var info = { + beat: Math.floor(music.beat()), + measure: Math.floor(music.measure()) + 1, + time: self.timeElapsed.toFixed(2), + fps: (1000 / self.realTimeDelta).toFixed(), + score: game.score, + }; + + if (self.started) { + info.wave = this.waveIndex.toString() + ' - ' + this.wave.constructor.name; + info.colchecks = self.collisionChecks.toFixed(); + } + + for (var key in info) { + draw({ + type: 'text', + text: key + ': ' + info[key], + fontFamily: 'Monaco', + fontFallback: 'monospace', + fontSize: 12, + textAlign: 'left', + textBaseline: 'alphabetic', + fillStyle: rgbWithOpacity([0, 0, 0], 1), + textPosition: { x: 5, y: height - fromBottom }, + }); + + fromBottom += 15; + } + }; + + self.draw = function () { + if (!DEBUG) draw({ type: 'clear' }); + + self.background.draw(); + self.drawScore(); + self.drawParticles(); + + if (self.started) self.wave.draw(); + self.cable.draw(); + self.tether.draw(); + self.player.draw(); + + self.drawLogo(); + self.drawRestartTutorial(); + + self.drawAchievementNotifications(); + + if (!self.started || self.ended) self.drawMuteButton(); + if (self.started && !self.ended) self.drawPlayButton(); + + if ((self.tether.lastInteraction === 'mouse' && self.ended) || !self.started) + self.drawAchievementUI(); + + if (INFO) self.drawInfo(); + }; + + self.end = function () { + if (document.pointerLockElement) document.exitPointerLock(); + logScore(self.score); + self.ended = self.timeElapsed; + self.tether.locked = true; + self.tether.unlockable = false; + self.setSpeed(self.slowSpeed); + }; + + self.reset(0); +} + +var enemyPool = [Drifter, Eye, Twitchy]; + +music = new Music(); +game = new Game(); + +function handleClick(e) { + if (game.eventShouldMute(e)) { + if (music.element.paused) { + console.log('play'); + music.element.play(); + saveCookie(musicMutedCookieKey, 'true'); + } else { + console.log('pause'); + music.element.pause(); + saveCookie(musicMutedCookieKey, 'false'); + } + } else if (game.eventShouldPlay(e)) { + paused = !paused; + } else if (game.ended) { + game.reset(0); + } +} + +var konamiLength = 0; +var konamiSequence = ['ArrowUp', 'ArrowUp', 'ArrowDown', 'ArrowDown', 'ArrowLeft', 'ArrowRight', 'ArrowLeft', 'ArrowRight', 'KeyB', 'KeyA', 'Space']; + +function konamiSeq(requiredKey, givenKey) { + if (requiredKey === givenKey) konamiLength++; + else konamiLength = 0; + + if (konamiLength === 11) { + subtitleText = 'Special Cheats Activated. Have fun!'; + playerRGB = 'Rainbow'; + } +} + +function handleKey(e) { + konamiSeq(konamiSequence[konamiLength], e.code); + if (self.started && !self.ended && e.code === 'KeyP') paused = !paused; +} + +document.addEventListener('click', handleClick); +document.addEventListener('keydown', handleKey); + +canvas.addEventListener('mousemove', function (e) { + if (game.tether.lastInteraction === 'touch' && document.pointerLockElement) + document.exitPointerLock(); + else if (document.pointerLockElement === canvas) { + if (game.tether.locked) game.tether.locked = false; + + game.lastMousePosition.x += e.movementX; + game.lastMousePosition.y += e.movementY; + + if (game.lastMousePosition.x < 0) game.lastMousePosition.x = 0; + else if (game.lastMousePosition.x > width) game.lastMousePosition.x = width; + + if (game.lastMousePosition.y < 0) game.lastMousePosition.y = 0; + else if (game.lastMousePosition.y > height) game.lastMousePosition.y = height; + } +}); + +document.addEventListener('touchstart', function (e) { + lastTouchStart = new Date().getTime(); +}); +document.addEventListener('touchend', function (e) { + if ( + lastTouchStart !== undefined && + new Date().getTime() - lastTouchStart < 300 + ) { + handleClick(e); + } +}); + +window.requestFrame = + window.requestAnimationFrame || + window.webkitRequestAnimationFrame || + window.mozRequestAnimationFrame || + function (callback) { + window.setTimeout(callback, 1000 / 60); + }; + +var pauseDelay = 0; +function animate() { + requestFrame(animate); + if (!paused) { + game.step(); + if (pauseDelay !== 0) { + pauseDelay = 0; + if (canvas.requestPointerLock) canvas.requestPointerLock(); + game.player.teleportTo({ + x: game.lastMousePosition.x + 50, + y: game.lastMousePosition.y + 50, + }); + } + } else if (paused && pauseDelay !== 1) { + game.step(); + game.drawPauseMessage(); + if (document.pointerLockElement) document.exitPointerLock(); + pauseDelay++; + } +} + +var scrollTimeout; +window.addEventListener('scroll', function (e) { + clearTimeout(scrollTimeout); + scrollTimeout = setTimeout(function () { + window.scrollTo(0, 0); + }, 500); +}); +window.scrollTo(0, 0); + +animate(); \ No newline at end of file diff --git a/Games/Tether/source/index.html b/Games/Tether/source/index.html new file mode 100644 index 0000000000..a1255daec1 --- /dev/null +++ b/Games/Tether/source/index.html @@ -0,0 +1,187 @@ + + + + + + + + + + + + tether! + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/Games/Tether/source/templates/main.template.html b/Games/Tether/source/templates/main.template.html new file mode 100644 index 0000000000..086273e2c9 --- /dev/null +++ b/Games/Tether/source/templates/main.template.html @@ -0,0 +1 @@ +tether! \ No newline at end of file diff --git a/Games/Tether/source/templates/offline.template.html b/Games/Tether/source/templates/offline.template.html new file mode 100644 index 0000000000..27cfc521bd --- /dev/null +++ b/Games/Tether/source/templates/offline.template.html @@ -0,0 +1 @@ +tether! \ No newline at end of file diff --git a/Games/Tether/splashscreens/ipad_splash.png b/Games/Tether/splashscreens/ipad_splash.png new file mode 100644 index 0000000000..f7e0086c8f Binary files /dev/null and b/Games/Tether/splashscreens/ipad_splash.png differ diff --git a/Games/Tether/splashscreens/ipadpro1_splash.png b/Games/Tether/splashscreens/ipadpro1_splash.png new file mode 100644 index 0000000000..b00a641aeb Binary files /dev/null and b/Games/Tether/splashscreens/ipadpro1_splash.png differ diff --git a/Games/Tether/splashscreens/ipadpro2_splash.png b/Games/Tether/splashscreens/ipadpro2_splash.png new file mode 100644 index 0000000000..ca3092f51b Binary files /dev/null and b/Games/Tether/splashscreens/ipadpro2_splash.png differ diff --git a/Games/Tether/splashscreens/ipadpro3_splash.png b/Games/Tether/splashscreens/ipadpro3_splash.png new file mode 100644 index 0000000000..1008356f2f Binary files /dev/null and b/Games/Tether/splashscreens/ipadpro3_splash.png differ diff --git a/Games/Tether/splashscreens/iphone5_splash.png b/Games/Tether/splashscreens/iphone5_splash.png new file mode 100644 index 0000000000..ea436f468f Binary files /dev/null and b/Games/Tether/splashscreens/iphone5_splash.png differ diff --git a/Games/Tether/splashscreens/iphone6_splash.png b/Games/Tether/splashscreens/iphone6_splash.png new file mode 100644 index 0000000000..85d9d071e9 Binary files /dev/null and b/Games/Tether/splashscreens/iphone6_splash.png differ diff --git a/Games/Tether/splashscreens/iphoneplus_splash.png b/Games/Tether/splashscreens/iphoneplus_splash.png new file mode 100644 index 0000000000..3c29a9556a Binary files /dev/null and b/Games/Tether/splashscreens/iphoneplus_splash.png differ diff --git a/Games/Tether/splashscreens/iphonex_splash.png b/Games/Tether/splashscreens/iphonex_splash.png new file mode 100644 index 0000000000..7d24358790 Binary files /dev/null and b/Games/Tether/splashscreens/iphonex_splash.png differ diff --git a/Games/Tether/splashscreens/iphonexr_splash.png b/Games/Tether/splashscreens/iphonexr_splash.png new file mode 100644 index 0000000000..e17d977b21 Binary files /dev/null and b/Games/Tether/splashscreens/iphonexr_splash.png differ diff --git a/Games/Tether/splashscreens/iphonexsmax_splash.png b/Games/Tether/splashscreens/iphonexsmax_splash.png new file mode 100644 index 0000000000..233a6ba7e1 Binary files /dev/null and b/Games/Tether/splashscreens/iphonexsmax_splash.png differ diff --git a/Games/Tether/tether.webmanifest b/Games/Tether/tether.webmanifest new file mode 100644 index 0000000000..3ed310747c --- /dev/null +++ b/Games/Tether/tether.webmanifest @@ -0,0 +1,52 @@ +{ + "name": "tether! | Swing Around a Ball of Destruction!", + "short_name": "tether!", + "lang": "en-US", + "start_url": ".", + "display": "standalone", + "background_color": "#FFF", + "description": "A game about swinging a ball around and sheer destruction.", + "categories": [ + "game", + "mobile", + "fun" + ], + "icons": [ + { + "src": "/icons/android-icon-36x36.png", + "sizes": "36x36", + "type": "image/png", + "density": "0.75" + }, + { + "src": "/icons/android-icon-48x48.png", + "sizes": "48x48", + "type": "image/png", + "density": "1.0" + }, + { + "src": "/icons/android-icon-72x72.png", + "sizes": "72x72", + "type": "image/png", + "density": "1.5" + }, + { + "src": "/icons/android-icon-96x96.png", + "sizes": "96x96", + "type": "image/png", + "density": "2.0" + }, + { + "src": "/icons/android-icon-144x144.png", + "sizes": "144x144", + "type": "image/png", + "density": "3.0" + }, + { + "src": "/icons/android-icon-192x192.png", + "sizes": "192x192", + "type": "image/png", + "density": "4.0" + } + ] +} \ No newline at end of file diff --git a/Games/Tether/tether_theme.mp3 b/Games/Tether/tether_theme.mp3 new file mode 100644 index 0000000000..9cc1b7b09a Binary files /dev/null and b/Games/Tether/tether_theme.mp3 differ diff --git a/README.md b/README.md index 1d79be8b91..a18be6e7af 100644 --- a/README.md +++ b/README.md @@ -881,6 +881,7 @@ This repository also provides one such platforms where contributers come over an |[Idle_miner](https://github.com/kunjgit/GameZone/tree/main/Games/Idle_miner)| |[Five_Nights_at_Freddys](https://github.com/kunjgit/GameZone/tree/main/Games/Five_Nights_at_Freddys)| |[Snake_Gun_Water](https://github.com/kunjgit/GameZone/tree/main/Games/Snake_Gun_Water)| +|[Tether](https://github.com/kunjgit/GameZone/tree/main/Games/Tether)|
diff --git a/assets/images/Tether.png b/assets/images/Tether.png new file mode 100644 index 0000000000..8411e324b4 Binary files /dev/null and b/assets/images/Tether.png differ