From 9d43fc46baebe527b24cd59417864b1f73f2b129 Mon Sep 17 00:00:00 2001 From: Niklas Kiefer Date: Fri, 22 Mar 2019 18:46:46 +0100 Subject: [PATCH 1/2] feat(collision-detection): move to web worker --- .eslintignore | 1 + .eslintrc | 1 + lib/canvas/canvas.js | 57 ---------- lib/collision-detection/collisionDetection.js | 87 +++++++++++++++ lib/main/main.js | 21 +++- lib/threadify/threadify.min.js | 1 + test/canvas/canvasSpec.js | 71 ------------ .../collisionDetectionSpec.js | 102 ++++++++++++++++++ 8 files changed, 211 insertions(+), 130 deletions(-) create mode 100644 lib/collision-detection/collisionDetection.js create mode 100755 lib/threadify/threadify.min.js create mode 100644 test/collision-detection/collisionDetectionSpec.js diff --git a/.eslintignore b/.eslintignore index c3651b8..c6a4e07 100644 --- a/.eslintignore +++ b/.eslintignore @@ -1,4 +1,5 @@ dist/ node_modules/ lib/jquery/ +lib/threadify coverage/ \ No newline at end of file diff --git a/.eslintrc b/.eslintrc index ccf2ddf..37b9414 100644 --- a/.eslintrc +++ b/.eslintrc @@ -8,6 +8,7 @@ "no-trailing-spaces": ["error", { "skipBlankLines": true }], "padded-blocks": ["error", "always"], "no-unused-vars": 0, + "new-cap": 0, "no-console": "error", "no-only-tests/no-only-tests": 2 }, diff --git a/lib/canvas/canvas.js b/lib/canvas/canvas.js index dea3289..1ee8d53 100644 --- a/lib/canvas/canvas.js +++ b/lib/canvas/canvas.js @@ -220,63 +220,6 @@ class Canvas { } - /** - * checks collision between obstacles and bird object - */ - collisionDetection () { - - let birdShape = this.bird.shape; - let obstacles = this.obstacles; - let birdWidth = birdShape.x + birdShape.width; - - const _hitAction = () => { - - if (typeof this.onClear === 'function') { - - this.onClear(); - - } - - }; - - obstacles.forEach(obstacle => { - - let obstacleShape = obstacle.shape; - let obstacleHeight = obstacleShape.y + obstacleShape.height; - let obstacleWidth = obstacleShape.x + obstacleShape.width; - - if ( - (birdShape.x >= obstacleShape.x && birdWidth <= obstacleWidth) || - (birdWidth >= obstacleShape.x && birdWidth <= obstacleWidth) || - (birdShape.x === obstacleWidth) || - (birdShape.x === obstacleShape.x) || - (birdWidth === obstacleWidth) - ) { - - if (obstacleShape.y === 0) { // check upper obstacle - - if (birdShape.y <= obstacleHeight) { - - _hitAction(); - - } - - } else { // check lower obstacle - - if (birdShape.y + birdShape.height >= obstacleShape.y) { - - _hitAction(); - - } - - } - - } - - }); - - } - /** * completely deletes canvas and all its shapes */ diff --git a/lib/collision-detection/collisionDetection.js b/lib/collision-detection/collisionDetection.js new file mode 100644 index 0000000..b224a8c --- /dev/null +++ b/lib/collision-detection/collisionDetection.js @@ -0,0 +1,87 @@ + +/* global threadify */ +class CollisionDetection { + + /** + * @param {Bird} options.bird + * @param {Array} options.obstacle + */ + static collisionDetection ({ bird, obstacles }) { + + const thread = this; + + let birdWidth = bird.x + bird.width; + + obstacles.forEach(obstacle => { + + let obstacleHeight = obstacle.y + obstacle.height; + let obstacleWidth = obstacle.x + obstacle.width; + + if ( + (bird.x >= obstacle.x && birdWidth <= obstacleWidth) || + (birdWidth >= obstacle.x && birdWidth <= obstacleWidth) || + (bird.x === obstacleWidth) || + (bird.x === obstacle.x) || + (birdWidth === obstacleWidth) + ) { + + if (obstacle.y === 0) { // check upper obstacle + + if (bird.y <= obstacleHeight) { + + thread.return(true); + + } + + } else { // check lower obstacle + + if (bird.y + bird.height >= obstacle.y) { + + thread.return(true); + + } + + } + + } + + }); + + thread.return(false); + + } + + static wrapped (canvas) { + + // cleanup objects for later use + const bird = { + x: canvas.bird.shape.x, + y: canvas.bird.shape.y, + width: canvas.bird.shape.width, + height: canvas.bird.shape.height + }; + + const obstacles = canvas.obstacles.map(o => { + + return { + x: o.shape.x, + y: o.shape.y, + width: o.shape.width, + height: o.shape.height + }; + + }); + + const wrappedFn = threadify(this.collisionDetection); + + return { + collisionDetection: wrappedFn, + collideCheckObject: { + bird, + obstacles + } + }; + + } + +} diff --git a/lib/main/main.js b/lib/main/main.js index efda47d..a2005f7 100644 --- a/lib/main/main.js +++ b/lib/main/main.js @@ -1,4 +1,4 @@ -/* global Canvas, Footer, KeyboardHandler */ +/* global Canvas, Footer, KeyboardHandler, CollisionDetection */ let mrflap, canvas, bird; function appendFooter () { @@ -93,7 +93,24 @@ function spawnObstacles () { setInterval(() => { canvas.moveObstacles(); - canvas.collisionDetection(); + + // execute collision detection inside web worker + const { + collisionDetection, + collideCheckObject + } = CollisionDetection.wrapped(canvas); + + const job = collisionDetection(collideCheckObject); + + job.done = (isCollided) => { + + if (isCollided) { + + canvas.onClear(); + + } + + }; }, 1000 / 30); diff --git a/lib/threadify/threadify.min.js b/lib/threadify/threadify.min.js new file mode 100755 index 0000000..54b1a21 --- /dev/null +++ b/lib/threadify/threadify.min.js @@ -0,0 +1 @@ +!(function(e){if("object"==typeof exports&&"undefined"!=typeof module)module.exports=e();else if("function"==typeof define&&define.amd)define([],e);else{("undefined"!=typeof window?window:"undefined"!=typeof global?global:"undefined"!=typeof self?self:this).threadify=e()}}(function(){return function i(o,s,f){function u(r,e){if(!s[r]){if(!o[r]){var t="function"==typeof require&&require;if(!e&&t)return t(r,!0);if(l)return l(r,!0);var n=new Error("Cannot find module '"+r+"'");throw n.code="MODULE_NOT_FOUND",n}var a=s[r]={exports:{}};o[r][0].call(a.exports,function(e){return u(o[r][1][e]||e)},a,a.exports,i,o,s,f)}return s[r].exports}for(var l="function"==typeof require&&require,e=0;e'); + + }; + + beforeEach(function () { + + document.body.innerHTML = __html__['test.html']; + _initPlayground(); + mrflapDiv = $('.mrflap-playground'); + + }); + describe('#collisionDetection', function () { + + let canvas, bird; + + beforeEach(function () { + + canvas = new Canvas({ + mrflapDiv: mrflapDiv + }); + + bird = canvas.drawBird(); + + }); + + it('should hit lower obstacle', function () { + + // given + canvas.drawObstacle({ + heightBottom: 50 + }); + + // when + for (let i = 0; i < 110; i++) { + + canvas.moveObstacles(); + + } + + // when + const { + collisionDetection, + collideCheckObject + } = CollisionDetection.wrapped(canvas); + + const job = collisionDetection(collideCheckObject); + + job.done = (isCollided) => { + + // then + expect(isCollided).to.be.true; + + }; + + }); + + it('should hit upper obstacle', function () { + + // given + canvas.drawObstacle({ + heightBottom: 50 + }); + + bird.moveUp({ + speed: 100 + }); + + // when + for (let i = 0; i < 110; i++) { + + canvas.moveObstacles(); + + } + + // when + const { + collisionDetection, + collideCheckObject + } = CollisionDetection.wrapped(canvas); + + const job = collisionDetection(collideCheckObject); + + job.done = (isCollided) => { + + // then + expect(isCollided).to.be.true; + + }; + + }); + + }); + +}); From dd84ff6e095c045fa3b5b16dd7b35a7af17ba358 Mon Sep 17 00:00:00 2001 From: Niklas Kiefer Date: Fri, 22 Mar 2019 19:10:39 +0100 Subject: [PATCH 2/2] feat(project): introduce test disabler --- gulpfile.js | 2 ++ lib/main/main.js | 14 +++++++++++++- test/main/mainSpec.js | 16 +++++++++++++++- test/test.html | 3 +++ 4 files changed, 33 insertions(+), 2 deletions(-) diff --git a/gulpfile.js b/gulpfile.js index eef876b..2bb385c 100644 --- a/gulpfile.js +++ b/gulpfile.js @@ -140,6 +140,8 @@ gulp.task('default', gulp.series(['clean', 'html', 'css', 'js', 'vendor'])); gulp.task('test', gulp.series(['default'], function (done) { + process.env.NODE_ENV = 'test'; + return new KarmaServ({ configFile: `${configuration.paths.test}/config/karma.unit.js`, singleRun: true diff --git a/lib/main/main.js b/lib/main/main.js index a2005f7..8206c7a 100644 --- a/lib/main/main.js +++ b/lib/main/main.js @@ -94,6 +94,12 @@ function spawnObstacles () { canvas.moveObstacles(); + if (_isTestMode()) { + + return; + + } + // execute collision detection inside web worker const { collisionDetection, @@ -102,7 +108,7 @@ function spawnObstacles () { const job = collisionDetection(collideCheckObject); - job.done = (isCollided) => { + job.done = function (isCollided) { if (isCollided) { @@ -170,6 +176,12 @@ function _reset () { } +function _isTestMode () { + + return $('#test-mode').length > 0; + +} + document.addEventListener('DOMContentLoaded', event => { _init(); diff --git a/test/main/mainSpec.js b/test/main/mainSpec.js index 85beb5c..312eb31 100644 --- a/test/main/mainSpec.js +++ b/test/main/mainSpec.js @@ -1,4 +1,4 @@ -/* global it, describe, expect, before, __html__, _init, _reset, bird, canvas, sinon */ +/* global it, describe, expect, before, __html__, _init, _reset, bird, canvas, sinon, _isTestMode */ /* eslint-disable no-unused-expressions */ describe('Main', function () { @@ -108,6 +108,20 @@ describe('Main', function () { }); + describe('#_isTestMode', function () { + + it('should return true in test mode', function () { + + // given + const isTestMode = _isTestMode(); + + // then + expect(isTestMode).to.be.true; + + }); + + }); + describe('#_reset', function () { before(function () { diff --git a/test/test.html b/test/test.html index a82d875..691b5f3 100644 --- a/test/test.html +++ b/test/test.html @@ -3,5 +3,8 @@ + +
+ \ No newline at end of file