diff --git a/.jshintrc b/.jshintrc new file mode 100644 index 0000000..62bb2d4 --- /dev/null +++ b/.jshintrc @@ -0,0 +1,86 @@ +{ + // See http://jshint.com/docs/ for details + + "maxerr" : 50, // {int} Maximum error before stopping + + // Enforcing + "bitwise" : true, // true: Prohibit bitwise operators (&, |, ^, etc.) + "camelcase" : false, // true: Identifiers must be in camelCase + "curly" : true, // true: Require {} for every new block or scope + "eqeqeq" : true, // true: Require triple equals (===) for comparison + "freeze" : true, // true: prohibits overwriting prototypes of native objects such as Array, Date etc. + "forin" : true, // true: Require filtering for..in loops with obj.hasOwnProperty() + "immed" : false, // true: Require immediate invocations to be wrapped in parens e.g. `(function () { } ());` + "indent" : 4, // {int} Number of spaces to use for indentation + "latedef" : false, // true: Require variables/functions to be defined before being used + "newcap" : false, // true: Require capitalization of all constructor functions e.g. `new F()` + "noarg" : true, // true: Prohibit use of `arguments.caller` and `arguments.callee` + "noempty" : true, // true: Prohibit use of empty blocks + "nonbsp" : true, // true: Prohibit "non-breaking whitespace" characters. + "nonew" : false, // true: Prohibit use of constructors for side-effects (without assignment) + "plusplus" : false, // true: Prohibit use of `++` & `--` + "quotmark" : false, // Quotation mark consistency: + // false : do nothing (default) + // true : ensure whatever is used is consistent + // "single" : require single quotes + // "double" : require double quotes + "undef" : true, // true: Require all non-global variables to be declared (prevents global leaks) + "unused" : true, // true: Require all defined variables be used + "strict" : true, // true: Requires all functions run in ES5 Strict Mode + "maxparams" : false, // {int} Max number of formal params allowed per function + "maxdepth" : false, // {int} Max depth of nested blocks (within functions) + "maxstatements" : false, // {int} Max number statements per function + "maxcomplexity" : false, // {int} Max cyclomatic complexity per function + "maxlen" : 79, // {int} Max number of characters per line + + // Relaxing + "asi" : false, // true: Tolerate Automatic Semicolon Insertion (no semicolons) + "boss" : false, // true: Tolerate assignments where comparisons would be expected + "debug" : false, // true: Allow debugger statements e.g. browser breakpoints. + "eqnull" : false, // true: Tolerate use of `== null` + "es5" : false, // true: Allow ES5 syntax (ex: getters and setters) + "esnext" : false, // true: Allow ES.next (ES6) syntax (ex: `const`) + "moz" : false, // true: Allow Mozilla specific syntax (extends and overrides esnext features) + // (ex: `for each`, multiple try/catch, function expression…) + "evil" : false, // true: Tolerate use of `eval` and `new Function()` + "expr" : false, // true: Tolerate `ExpressionStatement` as Programs + "funcscope" : false, // true: Tolerate defining variables inside control statements + "globalstrict" : false, // true: Allow global "use strict" (also enables 'strict') + "iterator" : false, // true: Tolerate using the `__iterator__` property + "lastsemic" : false, // true: Tolerate omitting a semicolon for the last statement of a 1-line block + "laxbreak" : false, // true: Tolerate possibly unsafe line breakings + "laxcomma" : false, // true: Tolerate comma-first style coding + "loopfunc" : false, // true: Tolerate functions being defined in loops + "multistr" : false, // true: Tolerate multi-line strings + "noyield" : false, // true: Tolerate generator functions with no yield statement in them. + "notypeof" : false, // true: Tolerate invalid typeof operator values + "proto" : false, // true: Tolerate using the `__proto__` property + "scripturl" : false, // true: Tolerate script-targeted URLs + "shadow" : false, // true: Allows re-define variables later in code e.g. `var x=1; x=2;` + "sub" : false, // true: Tolerate using `[]` notation when it can still be expressed in dot notation + "supernew" : false, // true: Tolerate `new function () { ... };` and `new Object;` + "validthis" : false, // true: Tolerate using this in a non-constructor function + + // Environments + "browser" : true, // Web Browser (window, document, etc) + "browserify" : true, // Browserify (node.js code in the browser) + "couch" : false, // CouchDB + "devel" : true, // Development/debugging (alert, confirm, etc) + "dojo" : false, // Dojo Toolkit + "jasmine" : false, // Jasmine + "jquery" : false, // jQuery + "mocha" : true, // Mocha + "mootools" : false, // MooTools + "node" : false, // Node.js + "nonstandard" : false, // Widely adopted globals (escape, unescape, etc) + "prototypejs" : false, // Prototype and Scriptaculous + "qunit" : false, // QUnit + "rhino" : false, // Rhino + "shelljs" : false, // ShellJS + "worker" : false, // Web Workers + "wsh" : false, // Windows Scripting Host + "yui" : false, // Yahoo User Interface + + // Custom Globals + "globals" : {} // additional predefined global variables +} diff --git a/dist/js/gamepad-client.js b/dist/js/gamepad-client.js index 5649ec3..e83b7b0 100755 --- a/dist/js/gamepad-client.js +++ b/dist/js/gamepad-client.js @@ -10,7 +10,7 @@ var error = utils.error; var trace = utils.trace; -utils.polyfill(window); +utils.polyfill(); utils.lockOrientation('landscape-primary'); @@ -29,8 +29,8 @@ document.addEventListener('keyup', function (e) { trace('User pressed "F"; entering/exiting fullscreen'); return utils.toggleFullScreen(); case 78: // Pressing NF (really just N) should toggle full-screen mode. - trace('User pressed "NF"; exiting fullscreen and will not automatically ' + - 'open next time'); + trace('User pressed "NF"; exiting fullscreen and will not ' + + 'automatically open next time'); localStorage.disableAutoFullScreen = '1'; return utils.toggleFullScreen(); } @@ -52,7 +52,7 @@ document.addEventListener('click', function (e) { var peerId = utils.getPeerId(); -var peer = new Peer('controller_' + peerId, { +var peer = new window.Peer('controller_' + peerId, { key: settings.PEERJS_KEY, debug: settings.DEBUG ? 3 : 0 }); @@ -79,7 +79,8 @@ conn.on('open', function () { function send(msg) { if (settings.DEBUG) { - console.log('Sent: ' + (typeof msg === 'object' ? JSON.stringify(msg) : msg)); + console.log('Sent: ' + + (typeof msg === 'object' ? JSON.stringify(msg) : msg)); } conn.send(msg); } @@ -116,7 +117,8 @@ function angularShape(canvas, coords) { } function linearFill(shape, color1, color2, coords) { - var bg = shape.createLinearGradient(coords[0], coords[1], coords[2], coords[3]); + var bg = shape.createLinearGradient(coords[0], coords[1], coords[2], + coords[3]); bg.addColorStop(0, color1); bg.addColorStop(1, color2); shape.fillStyle = bg; @@ -187,15 +189,16 @@ var gamepadState = { function bindPress(button, eventName, isPressed) { - document.querySelector('#' + button).addEventListener(eventName, function (e) { - // Handle D-pad presses. - if (e.target && e.target.parentNode === dpad) { - dpad.classList.toggle(this.id); - } + document.querySelector('#' + button) + .addEventListener(eventName, function (e) { + // Handle D-pad presses. + if (e.target && e.target.parentNode === dpad) { + dpad.classList.toggle(this.id); + } - gamepadState[button] = isPressed; - send({type: 'state', data: gamepadState}); - }); + gamepadState[button] = isPressed; + send({type: 'state', data: gamepadState}); + }); } @@ -290,8 +293,12 @@ bindKeyPresses('keyup', false); })(window, document); },{"./lib/utils":2,"./settings":3}],2:[function(require,module,exports){ +module.exports = function (window, document) { +'use strict'; + function trace(text, level) { - console[level || 'log']((window.performance.now() / 1000).toFixed(3) + ': ' + text); + console[level || 'log']( + (window.performance.now() / 1000).toFixed(3) + ': ' + text); } @@ -305,17 +312,18 @@ function warn(text) { } -function polyfill(win) { - if (!('performance' in win)) { - win.performance = { +function polyfill() { + if (!('performance' in window)) { + window.performance = { now: function () { return +new Date(); } }; } - if (('origin' in win.location)) { - win.location.origin = win.location.protocol + '//' + win.location.host; + if (('origin' in window.location)) { + window.location.origin = (window.location.protocol + '//' + + window.location.host); } } @@ -343,7 +351,7 @@ function fieldFocused(e) { function hasTouchEvents() { return ('ontouchstart' in window || - window.DocumentTouch && document instanceof DocumentTouch); + window.DocumentTouch && document instanceof window.DocumentTouch); } function injectCSS(opts) { @@ -384,7 +392,8 @@ function toggleFullScreen() { } else if (document.documentElement.mozRequestFullScreen) { document.documentElement.mozRequestFullScreen(); } else if (document.documentElement.webkitRequestFullscreen) { - document.documentElement.webkitRequestFullscreen(Element.ALLOW_KEYBOARD_INPUT); + document.documentElement.webkitRequestFullscreen( + Element.ALLOW_KEYBOARD_INPUT); } else if (document.documentElement.msRequestFullscreen) { document.documentElement.msRequestFullscreen(); } @@ -403,16 +412,16 @@ function toggleFullScreen() { } -function lockOrientation() { - var lo = (screen.LockOrientation || - screen.mozLockOrientation || - screen.webkitLockOrientation || - screen.msLockOrientation); +function lockOrientation(orientation) { + var lo = (window.screen.LockOrientation || + window.screen.mozLockOrientation || + window.screen.webkitLockOrientation || + window.screen.msLockOrientation); if (!lo) { return warn('Orientation could not be locked'); } - lo(orientation); + return lo(orientation); } @@ -424,37 +433,46 @@ function triggerEvent(type) { } -module.exports.trace = trace; -module.exports.error = error; -module.exports.warn = warn; -module.exports.polyfill = polyfill; -module.exports.getPeerId = getPeerId; -module.exports.fieldFocused = fieldFocused; -module.exports.hasTouchEvents = hasTouchEvents; -module.exports.injectCSS = injectCSS; -module.exports.escape = escape; -module.exports.isFullScreen = isFullScreen; -module.exports.toggleFullScreen = toggleFullScreen; -module.exports.lockOrientation = lockOrientation; -module.exports.triggerEvent = triggerEvent; +return { + trace: trace, + error: error, + warn: warn, + polyfill: polyfill, + getPeerId: getPeerId, + fieldFocused: fieldFocused, + hasTouchEvents: hasTouchEvents, + injectCSS: injectCSS, + escape: escape, + isFullScreen: isFullScreen, + toggleFullScreen: toggleFullScreen, + lockOrientation: lockOrientation, + triggerEvent: triggerEvent +}; + +}; },{}],3:[function(require,module,exports){ +'use strict'; + var settings_local = {}; try { settings_local = require('./settings_local.js'); } catch (e) { } + var settings = { - API_URL: 'http://localhost:5000', // This URL to the Galaxy API. No trailing slash. + API_URL: 'http://localhost:5000', // Galaxy API URL. No trailing slash. DEBUG: false, PEERJS_KEY: '', // Sign up for a key at http://peerjs.com/peerserver VERSION: '0.0.1' // Version of the `gamepad.js` script }; -for (var key in settings_local) { - settings[key] = settings_local[key]; -} +// Override each default setting with user-defined setting. +Object.keys(settings_local).forEach(function (key) { + settings[key] = settings_local[key]; +}); + module.exports = settings; @@ -464,4 +482,5 @@ module.exports = { PEERJS_KEY: 'rovu5xmqo69wwmi' }; -},{}] \ No newline at end of file +},{}]},{},[1]) +//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["/opt/galaxy.js-mobile-gamepad/node_modules/browserify/node_modules/browser-pack/_prelude.js","./src/js/client.js","/opt/galaxy.js-mobile-gamepad/src/js/lib/utils.js","/opt/galaxy.js-mobile-gamepad/src/js/settings.js","/opt/galaxy.js-mobile-gamepad/src/js/settings_local.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpSA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7JA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvBA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","(function (window, document) {\n'use strict';\n\n// var peer = require('./lib/peer');\n// var Promise = require('./lib/promise-1.0.0');  // jshint ignore:line\nvar settings = require('./settings');\nvar utils = require('./lib/utils');\nvar error = utils.error;\nvar trace = utils.trace;\n\n\nutils.polyfill();\n\n\nutils.lockOrientation('landscape-primary');\nfunction wantsAutoFullScreen() {\n  return !('disableAutoFullScreen' in localStorage);\n}\n\n\ndocument.addEventListener('keyup', function (e) {\n  if (utils.fieldFocused(e)) {\n    return;\n  }\n\n  switch (e.keyCode) {\n    case 70:  // Pressing F should toggle full-screen mode.\n      trace('User pressed \"F\"; entering/exiting fullscreen');\n      return utils.toggleFullScreen();\n    case 78:  // Pressing NF (really just N) should toggle full-screen mode.\n      trace('User pressed \"NF\"; exiting fullscreen and will not ' +\n        'automatically open next time');\n      localStorage.disableAutoFullScreen = '1';\n      return utils.toggleFullScreen();\n  }\n});\n\n\ndocument.addEventListener('click', function (e) {\n  if (utils.fieldFocused(e) || !wantsAutoFullScreen()) {\n    return;\n  }\n  trace('Automatically entering fullscreen');\n  utils.toggleFullScreen();\n});\n\n\n// if there's not a pin, tell the user to open the game on another device\n// first. instead of relegating mobile to be always a controller, allow the\n// game to mirror the desktop (à la WiiU).\n\nvar peerId = utils.getPeerId();\n\nvar peer = new window.Peer('controller_' + peerId, {\n  key: settings.PEERJS_KEY,\n  debug: settings.DEBUG ? 3 : 0\n});\n\nwindow.addEventListener('beforeunload', function () {\n  peer.destroy();\n});\n\nvar conn = peer.connect(peerId);\n\nconn.on('open', function () {\n  trace('My peer ID: ' + peer.id);\n  trace('My connection ID: ' + conn.id);\n\n  conn.on('data', function (data) {\n    trace('Received: ' + data);\n  });\n\n  conn.on('error', function (err) {\n    error(err.message);\n  });\n});\n\n\nfunction send(msg) {\n  if (settings.DEBUG) {\n    console.log('Sent: ' +\n      (typeof msg === 'object' ? JSON.stringify(msg) : msg));\n  }\n  conn.send(msg);\n}\n\n\n/**\n * Traditional, NES-inspired gamepad.\n */\nvar dpad = document.querySelector('#dpad');\nvar selectButton = document.querySelector('#select');\nvar startButton = document.querySelector('#start');\nvar bButton = document.querySelector('#b');\nvar aButton = document.querySelector('#a');\n\n\n/**\n * Draw D-pad.\n */\nvar canvas = document.getElementById('dpad-body');\n\nfunction angularShape(canvas, coords) {\n  var shape = canvas.getContext('2d');\n  var i = 0;\n  shape.beginPath();\n  shape.moveTo(coords[0][0], coords[0][1]);\n  coords.slice(1);\n\n  for (; i < coords.length; i++) {\n    shape.lineTo(coords[i][0], coords[i][1]);\n  }\n\n  shape.closePath();\n  return shape;\n}\n\nfunction linearFill(shape, color1, color2, coords) {\n  var bg = shape.createLinearGradient(coords[0], coords[1], coords[2],\n    coords[3]);\n  bg.addColorStop(0, color1);\n  bg.addColorStop(1, color2);\n  shape.fillStyle = bg;\n  shape.fill();\n}\n\nfunction ySide(canvas, y, xFrom, xTo) {\n  var shape = angularShape(canvas, [\n    [y, xFrom],\n    [y + 5, xFrom + 3.5],\n    [y + 5, xTo + 3.5],\n    [y, xTo]\n  ]);\n  linearFill(shape, '#666', '#000', [y, xFrom, y + 15, xFrom]);\n}\n\nfunction xSide(canvas, x, yFrom, yTo) {\n  var shape = angularShape(canvas, [\n    [yFrom, x],\n    [yFrom + 5, x + 3.5],\n    [yTo + 5, x + 3.5],\n    [yTo, x]\n  ]);\n  linearFill(shape, '#666', '#000', [yFrom, x, yFrom, x + 15]);\n}\n\n// Draw the sides first.\nxSide(canvas, 63.5, 0, 100);\nxSide(canvas, 100, 36.5, 63.5);\nySide(canvas, 63.5, 0, 36.5);\nySide(canvas, 63.5, 63.5, 100);\nySide(canvas, 100, 36.5, 63.5);\n\n// Draw the D-pad.\nvar plus = angularShape(canvas, [\n  [0, 36.5],\n  [36.5, 36.5],\n  [36.5, 0],\n  [63.5, 0],\n  [63.5, 36.5],\n  [100, 36.5],\n  [100, 63.5],\n  [63.5, 63.5],\n  [63.5, 100],\n  [36.5, 100],\n  [36.5, 63],\n  [0, 63.5]\n]);\n\nplus.fillStyle = '#1a1a1a';\nplus.shadowColor = 'rgba(0,0,0,.6)';\nplus.shadowBlur = 15;\nplus.shadowOffsetX = 20;\nplus.shadowOffsetY = 10;\nplus.fill();\n\n\nvar gamepadState = {\n  up: false,\n  right: false,\n  down: false,\n  left: false,\n  select: false,\n  start: false,\n  b: false,\n  a: false\n};\n\n\nfunction bindPress(button, eventName, isPressed) {\n  document.querySelector('#' + button)\n    .addEventListener(eventName, function (e) {\n      // Handle D-pad presses.\n      if (e.target && e.target.parentNode === dpad) {\n        dpad.classList.toggle(this.id);\n      }\n\n      gamepadState[button] = isPressed;\n      send({type: 'state', data: gamepadState});\n    });\n}\n\n\nfunction bindKeyPresses(eventName, isPressed) {\n  document.addEventListener(eventName, function (e) {\n    if (utils.fieldFocused(e)) {\n      return;\n    }\n\n    switch (e.keyCode) {\n      case 38:\n        // Send event only once.\n        if (isPressed && gamepadState.up) {\n          return;\n        }\n        gamepadState.up = isPressed;\n        dpad.className = isPressed ? 'up' : '';\n        break;\n      case 39:\n        if (isPressed && gamepadState.right) {\n          return;\n        }\n        gamepadState.right = isPressed;\n        dpad.className = isPressed ? 'right' : '';\n        break;\n      case 40:\n        if (isPressed && gamepadState.down) {\n          return;\n        }\n        gamepadState.down = isPressed;\n        dpad.className = isPressed ? 'down' : '';\n        break;\n      case 37:\n        if (isPressed && gamepadState.left) {\n          return;\n        }\n        gamepadState.left = isPressed;\n        dpad.className = isPressed ? 'left' : '';\n        break;\n      case 13:\n        if (isPressed && gamepadState.start) {\n          return;\n        }\n        gamepadState.start = isPressed;\n        startButton.dataset.pressed = +isPressed;\n        break;\n      case 65:\n        if (isPressed && gamepadState.a) {\n          return;\n        }\n        gamepadState.a = isPressed;\n        aButton.dataset.pressed = +isPressed;\n        break;\n      case 66:\n        if (isPressed && gamepadState.b) {\n          return;\n        }\n        gamepadState.b = isPressed;\n        bButton.dataset.pressed = +isPressed;\n        break;\n      default:\n        if (e.shiftKey || (!isPressed && gamepadState.select)) {\n          // If the Shift key was pressed or unpressed, toggle its state.\n          gamepadState.select = isPressed;\n          selectButton.dataset.pressed = +isPressed;\n        } else {\n          // Otherwise (i.e., any other key was pressed), bail.\n          return;\n        }\n    }\n\n    send(gamepadState);\n  });\n}\n\n\nObject.keys(gamepadState).forEach(function (button) {\n  if (utils.hasTouchEvents()) {\n    bindPress(button, 'touchstart', true);\n    bindPress(button, 'touchend', false);\n  } else {\n    bindPress(button, 'mousedown', true);\n    bindPress(button, 'mouseup', false);\n  }\n});\n\n\nbindKeyPresses('keydown', true);\nbindKeyPresses('keyup', false);\n\n\n})(window, document);\n","module.exports = function (window, document) {\n'use strict';\n\nfunction trace(text, level) {\n  console[level || 'log'](\n    (window.performance.now() / 1000).toFixed(3) + ': ' + text);\n}\n\n\nfunction error(text) {\n  return trace(text, 'error');\n}\n\n\nfunction warn(text) {\n  return trace(text, 'warn');\n}\n\n\nfunction polyfill() {\n  if (!('performance' in window)) {\n    window.performance = {\n      now: function () {\n        return +new Date();\n      }\n    };\n  }\n\n  if (('origin' in window.location)) {\n    window.location.origin = (window.location.protocol + '//' +\n      window.location.host);\n  }\n}\n\n\nfunction getPeerId() {\n  return (window.location.pathname.indexOf('.html') ?\n    window.location.search.substr(1) : window.location.pathname.substr(1));\n}\n\n\nvar FIELD_FOCUSED_TAGS = [\n  'input',\n  'keygen',\n  'meter',\n  'option',\n  'output',\n  'progress',\n  'select',\n  'textarea'\n];\nfunction fieldFocused(e) {\n  return FIELD_FOCUSED_TAGS.indexOf(e.target.nodeName.toLowerCase()) !== -1;\n}\n\n\nfunction hasTouchEvents() {\n  return ('ontouchstart' in window ||\n    window.DocumentTouch && document instanceof window.DocumentTouch);\n}\n\nfunction injectCSS(opts) {\n  var link = document.createElement('link');\n  link.href = opts.href;\n  link.media = 'all';\n  link.rel = 'stylesheet';\n  link.type = 'text/css';\n  Object.keys(opts || {}).forEach(function (prop) {\n    link[prop] = opts[prop];\n  });\n  document.querySelector('head').appendChild(link);\n}\n\nfunction escape(text) {\n  if (!text) {\n    return text;\n  }\n  return text.replace(/&/g, '&amp;')\n             .replace(/</g, '&lt;')\n             .replace(/>/g, '&gt;')\n             .replace(/'/g, '&#39;')\n             .replace(/\"/g, '&#34;');\n}\n\nfunction isFullScreen() {\n  return (!document.fullscreenElement &&  // standard method\n    !document.mozFullScreenElement &&\n    !document.webkitFullscreenElement &&\n    !document.msFullscreenElement);  // vendor-prefixed methods\n}\n\nfunction toggleFullScreen() {\n  if (isFullScreen()) {\n    trace('Entering full screen');\n    if (document.documentElement.requestFullscreen) {\n      document.documentElement.requestFullscreen();\n    } else if (document.documentElement.mozRequestFullScreen) {\n      document.documentElement.mozRequestFullScreen();\n    } else if (document.documentElement.webkitRequestFullscreen) {\n      document.documentElement.webkitRequestFullscreen(\n        Element.ALLOW_KEYBOARD_INPUT);\n    } else if (document.documentElement.msRequestFullscreen) {\n      document.documentElement.msRequestFullscreen();\n    }\n  } else {\n    trace('Exiting full screen');\n    if (document.exitFullscreen) {\n      document.exitFullscreen();\n    } else if (document.mozCancelFullScreen) {\n      document.mozCancelFullScreen();\n    } else if (document.webkitExitFullscreen) {\n      document.webkitExitFullscreen();\n    } else if (document.msExitFullscreen) {\n      document.msExitFullscreen();\n    }\n  }\n}\n\n\nfunction lockOrientation(orientation) {\n  var lo = (window.screen.LockOrientation ||\n    window.screen.mozLockOrientation ||\n    window.screen.webkitLockOrientation ||\n    window.screen.msLockOrientation);\n  if (!lo) {\n    return warn('Orientation could not be locked');\n  }\n\n  return lo(orientation);\n}\n\n\nfunction triggerEvent(type) {\n  var event = document.createEvent('HTMLEvents');\n  event.initEvent(type, true, true);\n  event.eventName = type;\n  (document.body || window).dispatchEvent(event);\n}\n\n\nreturn {\n  trace: trace,\n  error: error,\n  warn: warn,\n  polyfill: polyfill,\n  getPeerId: getPeerId,\n  fieldFocused: fieldFocused,\n  hasTouchEvents: hasTouchEvents,\n  injectCSS: injectCSS,\n  escape: escape,\n  isFullScreen: isFullScreen,\n  toggleFullScreen: toggleFullScreen,\n  lockOrientation: lockOrientation,\n  triggerEvent: triggerEvent\n};\n\n};\n","'use strict';\n\nvar settings_local = {};\ntry {\n  settings_local = require('./settings_local.js');\n} catch (e) {\n}\n\n\nvar settings = {\n  API_URL: 'http://localhost:5000',  // Galaxy API URL. No trailing slash.\n  DEBUG: false,\n  PEERJS_KEY: '',  // Sign up for a key at http://peerjs.com/peerserver\n  VERSION: '0.0.1'  // Version of the `gamepad.js` script\n};\n\n// Override each default setting with user-defined setting.\nObject.keys(settings_local).forEach(function (key) {\n\tsettings[key] = settings_local[key];\n});\n\n\nmodule.exports = settings;\n","module.exports = {\n  DEBUG: true,\n  PEERJS_KEY: 'rovu5xmqo69wwmi'\n};\n"]} diff --git a/dist/js/gamepad-client.min.js b/dist/js/gamepad-client.min.js index fa6f6b3..9a3e944 100755 --- a/dist/js/gamepad-client.min.js +++ b/dist/js/gamepad-client.min.js @@ -1 +1 @@ -!function e(t,n,r){function o(l,u){if(!n[l]){if(!t[l]){var i="function"==typeof require&&require;if(!u&&i)return i(l,!0);if(c)return c(l,!0);var s=new Error("Cannot find module '"+l+"'");throw s.code="MODULE_NOT_FOUND",s}var a=n[l]={exports:{}};t[l][0].call(a.exports,function(e){var n=t[l][1][e];return o(n?n:e)},a,a.exports,e,t,n,r)}return n[l].exports}for(var c="function"==typeof require&&require,l=0;l/g,">").replace(/'/g,"'").replace(/"/g,"""):e}function d(){return!(document.fullscreenElement||document.mozFullScreenElement||document.webkitFullscreenElement||document.msFullscreenElement)}function f(){d()?(n("Entering full screen"),document.documentElement.requestFullscreen?document.documentElement.requestFullscreen():document.documentElement.mozRequestFullScreen?document.documentElement.mozRequestFullScreen():document.documentElement.webkitRequestFullscreen?document.documentElement.webkitRequestFullscreen(Element.ALLOW_KEYBOARD_INPUT):document.documentElement.msRequestFullscreen&&document.documentElement.msRequestFullscreen()):(n("Exiting full screen"),document.exitFullscreen?document.exitFullscreen():document.mozCancelFullScreen?document.mozCancelFullScreen():document.webkitExitFullscreen?document.webkitExitFullscreen():document.msExitFullscreen&&document.msExitFullscreen())}function m(){var e=screen.LockOrientation||screen.mozLockOrientation||screen.webkitLockOrientation||screen.msLockOrientation;return e?(e(orientation),void 0):o("Orientation could not be locked")}function p(e){var t=document.createEvent("HTMLEvents");t.initEvent(e,!0,!0),t.eventName=e,(document.body||window).dispatchEvent(t)}var g=["input","keygen","meter","option","output","progress","select","textarea"];t.exports.trace=n,t.exports.error=r,t.exports.warn=o,t.exports.polyfill=c,t.exports.getPeerId=l,t.exports.fieldFocused=u,t.exports.hasTouchEvents=i,t.exports.injectCSS=s,t.exports.escape=a,t.exports.isFullScreen=d,t.exports.toggleFullScreen=f,t.exports.lockOrientation=m,t.exports.triggerEvent=p},{}],3:[function(e,t){var n={};try{n=e("./settings_local.js")}catch(r){}var o={API_URL:"http://localhost:5000",DEBUG:!1,PEERJS_KEY:"",VERSION:"0.0.1"};for(var c in n)o[c]=n[c];t.exports=o},{"./settings_local.js":4}],4:[function(e,t){t.exports={DEBUG:!0,PEERJS_KEY:"rovu5xmqo69wwmi"}},{}]},{},[1]); \ No newline at end of file +!function e(t,n,r){function o(l,i){if(!n[l]){if(!t[l]){var u="function"==typeof require&&require;if(!i&&u)return u(l,!0);if(c)return c(l,!0);var a=new Error("Cannot find module '"+l+"'");throw a.code="MODULE_NOT_FOUND",a}var s=n[l]={exports:{}};t[l][0].call(s.exports,function(e){var n=t[l][1][e];return o(n?n:e)},s,s.exports,e,t,n,r)}return n[l].exports}for(var c="function"==typeof require&&require,l=0;l/g,">").replace(/'/g,"'").replace(/"/g,"""):e}function f(){return!(t.fullscreenElement||t.mozFullScreenElement||t.webkitFullscreenElement||t.msFullscreenElement)}function d(){f()?(n("Entering full screen"),t.documentElement.requestFullscreen?t.documentElement.requestFullscreen():t.documentElement.mozRequestFullScreen?t.documentElement.mozRequestFullScreen():t.documentElement.webkitRequestFullscreen?t.documentElement.webkitRequestFullscreen(Element.ALLOW_KEYBOARD_INPUT):t.documentElement.msRequestFullscreen&&t.documentElement.msRequestFullscreen()):(n("Exiting full screen"),t.exitFullscreen?t.exitFullscreen():t.mozCancelFullScreen?t.mozCancelFullScreen():t.webkitExitFullscreen?t.webkitExitFullscreen():t.msExitFullscreen&&t.msExitFullscreen())}function m(t){var n=e.screen.LockOrientation||e.screen.mozLockOrientation||e.screen.webkitLockOrientation||e.screen.msLockOrientation;return n?n(t):o("Orientation could not be locked")}function p(n){var r=t.createEvent("HTMLEvents");r.initEvent(n,!0,!0),r.eventName=n,(t.body||e).dispatchEvent(r)}var g=["input","keygen","meter","option","output","progress","select","textarea"];return{trace:n,error:r,warn:o,polyfill:c,getPeerId:l,fieldFocused:i,hasTouchEvents:u,injectCSS:a,escape:s,isFullScreen:f,toggleFullScreen:d,lockOrientation:m,triggerEvent:p}}},{}],3:[function(e,t){"use strict";var n={};try{n=e("./settings_local.js")}catch(r){}var o={API_URL:"http://localhost:5000",DEBUG:!1,PEERJS_KEY:"",VERSION:"0.0.1"};Object.keys(n).forEach(function(e){o[e]=n[e]}),t.exports=o},{"./settings_local.js":4}],4:[function(e,t){t.exports={DEBUG:!0,PEERJS_KEY:"rovu5xmqo69wwmi"}},{}]},{},[1]); \ No newline at end of file diff --git a/dist/js/gamepad-host.js b/dist/js/gamepad-host.js index 65591a2..32b7331 100755 --- a/dist/js/gamepad-host.js +++ b/dist/js/gamepad-host.js @@ -4,14 +4,14 @@ // var peer = require('./lib/peer'); // var Promise = require('./lib/promise-1.0.0.js'); // jshint ignore:line -var Modal = require('./lib/modal'); +var Modal = require('./lib/modal')(window, document); var settings = require('./settings'); -var utils = require('./lib/utils'); +var utils = require('./lib/utils')(window, document); var error = utils.error; var trace = utils.trace; -utils.polyfill(window); +utils.polyfill(); /** @@ -40,12 +40,12 @@ gamepad.state = {}; * @memberOf gamepad */ gamepad.peerHandshake = function (peerId) { - return new Promise(function (resolve, reject) { + return new Promise(function (resolve) { if (!peerId) { peerId = utils.getPeerId(); // The host ID. } - var peer = new Peer(peerId, { + var peer = new window.Peer(peerId, { key: settings.PEERJS_KEY, debug: settings.DEBUG ? 3 : 0 }); @@ -79,11 +79,13 @@ gamepad.peerConnect = function (peer) { gamepad._updateState(data.data); break; default: - console.warn('WebRTC message received of unknown type: "' + data.type + '"'); + console.warn( + 'WebRTC message received of unknown type: "' + data.type + '"'); break; } - trace('Received: ' + (typeof data === 'object' ? JSON.stringify(data) : '')); + trace('Received: ' + + (typeof data === 'object' ? JSON.stringify(data) : '')); }); conn.on('error', function (err) { @@ -110,16 +112,20 @@ gamepad.pair = function (peerId) { return new Promise(function (resolve) { return gamepad.peerHandshake(peerId).then(function (peer) { - var pairId = peer.id; // This should be the same as `peerId`, but this comes from PeerJS, which is the source of truth. + // `pairId` should be the same as `peerId`, + // but `peer.id` is the source of truth. + var pairId = peer.id; var pairIdEsc = encodeURIComponent(pairId); var pairUrl = galaxyOrigin + '/client.html?' + pairIdEsc; // Update the querystring in the address bar. - window.history.replaceState(null, null, window.location.pathname + '?' + pairIdEsc); + window.history.replaceState(null, null, + window.location.pathname + '?' + pairIdEsc); var content = ( '' ); @@ -220,7 +226,7 @@ gamepad._bind = function (eventName, listener) { * @param {Function} [listener] (Optional) The listener function to remove. * @return {Boolean} Was unbinding the listener successful. */ -Gamepad.prototype.unbind = function (eventName, listener) { +gamepad.prototype.unbind = function (eventName, listener) { // Remove everything for all event types. if (typeof eventName === 'undefined') { this.listeners = {}; @@ -273,6 +279,9 @@ module.exports = gamepad; })(window, document); },{"./lib/modal":2,"./lib/utils":3,"./settings":4}],2:[function(require,module,exports){ +module.exports = function (window, document) { +'use strict'; + var utils = require('./utils'); @@ -340,11 +349,17 @@ Modal.prototype.open = function () { }; -module.exports = Modal; +return Modal; + +}; },{"./utils":3}],3:[function(require,module,exports){ +module.exports = function (window, document) { +'use strict'; + function trace(text, level) { - console[level || 'log']((window.performance.now() / 1000).toFixed(3) + ': ' + text); + console[level || 'log']( + (window.performance.now() / 1000).toFixed(3) + ': ' + text); } @@ -358,17 +373,18 @@ function warn(text) { } -function polyfill(win) { - if (!('performance' in win)) { - win.performance = { +function polyfill() { + if (!('performance' in window)) { + window.performance = { now: function () { return +new Date(); } }; } - if (('origin' in win.location)) { - win.location.origin = win.location.protocol + '//' + win.location.host; + if (('origin' in window.location)) { + window.location.origin = (window.location.protocol + '//' + + window.location.host); } } @@ -396,7 +412,7 @@ function fieldFocused(e) { function hasTouchEvents() { return ('ontouchstart' in window || - window.DocumentTouch && document instanceof DocumentTouch); + window.DocumentTouch && document instanceof window.DocumentTouch); } function injectCSS(opts) { @@ -437,7 +453,8 @@ function toggleFullScreen() { } else if (document.documentElement.mozRequestFullScreen) { document.documentElement.mozRequestFullScreen(); } else if (document.documentElement.webkitRequestFullscreen) { - document.documentElement.webkitRequestFullscreen(Element.ALLOW_KEYBOARD_INPUT); + document.documentElement.webkitRequestFullscreen( + Element.ALLOW_KEYBOARD_INPUT); } else if (document.documentElement.msRequestFullscreen) { document.documentElement.msRequestFullscreen(); } @@ -456,16 +473,16 @@ function toggleFullScreen() { } -function lockOrientation() { - var lo = (screen.LockOrientation || - screen.mozLockOrientation || - screen.webkitLockOrientation || - screen.msLockOrientation); +function lockOrientation(orientation) { + var lo = (window.screen.LockOrientation || + window.screen.mozLockOrientation || + window.screen.webkitLockOrientation || + window.screen.msLockOrientation); if (!lo) { return warn('Orientation could not be locked'); } - lo(orientation); + return lo(orientation); } @@ -477,18 +494,55 @@ function triggerEvent(type) { } -module.exports.trace = trace; -module.exports.error = error; -module.exports.warn = warn; -module.exports.polyfill = polyfill; -module.exports.getPeerId = getPeerId; -module.exports.fieldFocused = fieldFocused; -module.exports.hasTouchEvents = hasTouchEvents; -module.exports.injectCSS = injectCSS; -module.exports.escape = escape; -module.exports.isFullScreen = isFullScreen; -module.exports.toggleFullScreen = toggleFullScreen; -module.exports.lockOrientation = lockOrientation; -module.exports.triggerEvent = triggerEvent; +return { + trace: trace, + error: error, + warn: warn, + polyfill: polyfill, + getPeerId: getPeerId, + fieldFocused: fieldFocused, + hasTouchEvents: hasTouchEvents, + injectCSS: injectCSS, + escape: escape, + isFullScreen: isFullScreen, + toggleFullScreen: toggleFullScreen, + lockOrientation: lockOrientation, + triggerEvent: triggerEvent +}; -},{}] \ No newline at end of file +}; + +},{}],4:[function(require,module,exports){ +'use strict'; + +var settings_local = {}; +try { + settings_local = require('./settings_local.js'); +} catch (e) { +} + + +var settings = { + API_URL: 'http://localhost:5000', // Galaxy API URL. No trailing slash. + DEBUG: false, + PEERJS_KEY: '', // Sign up for a key at http://peerjs.com/peerserver + VERSION: '0.0.1' // Version of the `gamepad.js` script +}; + +// Override each default setting with user-defined setting. +Object.keys(settings_local).forEach(function (key) { + settings[key] = settings_local[key]; +}); + + +module.exports = settings; + +},{"./settings_local.js":5}],5:[function(require,module,exports){ +module.exports = { + DEBUG: true, + PEERJS_KEY: 'rovu5xmqo69wwmi' +}; + +},{}]},{},[1])(1) +}); +//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["/opt/galaxy.js-mobile-gamepad/node_modules/browserify/node_modules/browser-pack/_prelude.js","./src/js/host.js","/opt/galaxy.js-mobile-gamepad/src/js/lib/modal.js","/opt/galaxy.js-mobile-gamepad/src/js/lib/utils.js","/opt/galaxy.js-mobile-gamepad/src/js/settings.js","/opt/galaxy.js-mobile-gamepad/src/js/settings_local.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtRA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7JA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvBA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","(function (window, document) {\n'use strict';\n\n// var peer = require('./lib/peer');\n// var Promise = require('./lib/promise-1.0.0.js');  // jshint ignore:line\nvar Modal = require('./lib/modal')(window, document);\nvar settings = require('./settings');\nvar utils = require('./lib/utils')(window, document);\nvar error = utils.error;\nvar trace = utils.trace;\n\n\nutils.polyfill();\n\n\n/**\n * A library for controlling an HTML5 game using WebRTC.\n *\n * @exports gamepad\n * @namespace gamepad\n */\nfunction gamepad() {\n}\n\n\ngamepad.listeners = {};\ngamepad.state = {};\n\n\n/**\n * Does a handshake with PeerJS' WebSocket server to get a peer ID.\n *\n * Once we have the peer ID, we can tell the controller how to find us. Then\n * all communication between the host and the controller is peer-to-peer via\n * WebRTC data channels.\n *\n * @param {String} peerId The peer ID.\n * @returns {Promise}\n * @memberOf gamepad\n */\ngamepad.peerHandshake = function (peerId) {\n  return new Promise(function (resolve) {\n    if (!peerId) {\n      peerId = utils.getPeerId();  // The host ID.\n    }\n\n    var peer = new window.Peer(peerId, {\n      key: settings.PEERJS_KEY,\n      debug: settings.DEBUG ? 3 : 0\n    });\n\n    window.addEventListener('beforeunload', function () {\n      peer.destroy();\n    });\n\n    peer.on('open', function () {\n      trace('My peer ID: ' + peer.id);\n      resolve(peer);\n    });\n  });\n};\n\n\n/**\n * Listens for a peer connection with the controller via WebRTC data channels.\n *\n * If one is given, we will tell PeerJS to use the peer ID the query-string.\n *\n * @returns {Promise}\n * @memberOf gamepad\n */\ngamepad.peerConnect = function (peer) {\n  return new Promise(function (resolve, reject) {\n    peer.on('connection', function (conn) {\n      conn.on('data', function (data) {\n        switch (data.type) {\n          case 'state':\n            gamepad._updateState(data.data);\n            break;\n          default:\n            console.warn(\n              'WebRTC message received of unknown type: \"' + data.type + '\"');\n            break;\n        }\n\n        trace('Received: ' +\n          (typeof data === 'object' ? JSON.stringify(data) : ''));\n      });\n\n      conn.on('error', function (err) {\n        error(err.message);\n        reject(err);\n      });\n\n      // We've connected to a controller.\n      resolve(conn);\n    });\n  });\n};\n\n\n/**\n * Connects to a peer (controller).\n *\n * Establishes connection with peer.\n *\n * @returns {Promise}\n * @memberOf gamepad\n */\ngamepad.pair = function (peerId) {\n  return new Promise(function (resolve) {\n\n    return gamepad.peerHandshake(peerId).then(function (peer) {\n      // `pairId` should be the same as `peerId`,\n      // but `peer.id` is the source of truth.\n      var pairId = peer.id;\n      var pairIdEsc = encodeURIComponent(pairId);\n      var pairUrl = galaxyOrigin + '/client.html?' + pairIdEsc;\n\n      // Update the querystring in the address bar.\n      window.history.replaceState(null, null,\n        window.location.pathname + '?' + pairIdEsc);\n\n      var content = (\n        '<div class=\"modal-inner modal-pair\">' +\n          '<h2>URL</h2><p><a href=\"' + pairUrl +\n            '\" class=\"pair-url\" target=\"_blank\">' + pairUrl + '</a></p>' +\n          '<h2>Code</h2><p class=\"pair-code\">' + pairIdEsc + '</p>' +\n        '</div>'\n      );\n\n      var modal = new Modal({\n        id: 'pairing-screen',\n        classes: 'slim',\n        title: 'Pair your mobile phone',\n        content: content\n      }, true);\n\n      // todo: replace `setTimeout`s with `transitionend` event listeners.\n      window.setTimeout(function () {\n        // Waiting for the transition to end.\n        modal.open();\n      }, 150);\n\n      [\n        'https://fonts.googleapis.com/css?family=Source+Sans+Pro:300,400,700',\n        '/css/modal.css'  // todo: do not hardcode absolute path\n      ].forEach(function (stylesheet) {\n        utils.injectCSS({href: stylesheet});\n      });\n\n      gamepad.peerConnect(peer).then(function (conn) {\n        console.log('Peer connected');\n        modal.close();\n        resolve(conn);\n      });\n\n    }).catch(console.error.bind(console));\n  });\n};\n\n\ngamepad._updateState = function (data) {\n Object.keys(data || {}).forEach(function (key) {\n   if (!this.state[key] && data[key]) {\n     // Button pushed.\n     gamepad._emit('buttondown', key);\n     gamepad._emit('buttondown.' + key, key);\n   } else if (this.state[key] && !data[key]) {\n     // Button released.\n     gamepad._emit('buttonup', key);\n     gamepad._emit('buttonup.' + key, key);\n   }\n }.bind(this));\n};\n\n\ngamepad.hidePairingScreen = function () {\n  Modal.closeAll();\n};\n\n\n/**\n * Fires an internal event with given data.\n *\n * @method _fire\n * @param {String} eventName Name of event to fire (e.g., `buttondown`).\n * @param {*} data Data to pass to the listener.\n * @private\n */\ngamepad._emit = function (eventName, data) {\n  (this.listeners[eventName] || []).forEach(function (listener) {\n    listener.apply(listener, [data]);\n  });\n};\n\n\n/**\n * Binds a listener to a gamepad event.\n *\n * @method bind\n * @param {String} eventName Event to bind to (e.g., `buttondown`).\n * @param {Function} listener Listener to call when given event occurs.\n * @return {Gamepad} Self\n */\ngamepad._bind = function (eventName, listener) {\n  if (typeof(this.listeners[event]) === 'undefined') {\n    this.listeners[event] = [];\n  }\n\n  this.listeners[event].push(listener);\n\n  return this;\n};\n\n\n/**\n * Removes listener of given type.\n *\n * If no type is given, all listeners are removed. If no listener is given, all\n * listeners of given type are removed.\n *\n * @method unbind\n * @param {String} [type] Type of listener to remove.\n * @param {Function} [listener] (Optional) The listener function to remove.\n * @return {Boolean} Was unbinding the listener successful.\n */\ngamepad.prototype.unbind = function (eventName, listener) {\n  // Remove everything for all event types.\n  if (typeof eventName === 'undefined') {\n    this.listeners = {};\n    return;\n  }\n\n  // Remove all listener functions for that event type.\n  if (typeof listener === 'undefined') {\n    this.listeners[eventName] = [];\n    return;\n  }\n\n  if (typeof this.listeners[eventName] === 'undefined') {\n    return false;\n  }\n\n  this.listeners[eventName].forEach(function (value, idx) {\n    // Remove only the listener function passed to this method.\n    if (value === listener) {\n      this.listeners[eventName].splice(idx, 1);\n      return true;\n    }\n  });\n\n  return false;\n};\n\n\n\n// todo: these are mapped directly to NES controller. fix this.\ngamepad.buttons = {\n  a: {\n    clicked: gamepad._bind\n  }\n};\n\n\ngamepad.version = settings.VERSION;\n\n\nvar galaxyOrigin = window.location.origin;\nvar dataOrigin = document.querySelector('[data-galaxy-origin]');\nif (dataOrigin) {\n  gamepad.galaxyOrigin = dataOrigin.dataset.galaxyOrigin;\n}\n\n\nmodule.exports = gamepad;\n\n})(window, document);\n","module.exports = function (window, document) {\n'use strict';\n\nvar utils = require('./utils');\n\n\nfunction Modal(opts, inject) {\n  // Create properties for `id`, `classes`, `title`, and `content`.\n  Object.keys(opts).forEach(function (key) {\n    this[key] = opts[key];\n  }.bind(this));\n\n  if (inject) {\n    this.inject();\n  }\n}\n\nModal.closeAll = Modal.prototype.close = function () {\n  // Close any open modal.\n  var openedModal = document.querySelector('.md-show');\n  if (openedModal) {\n    openedModal.classList.remove('md-show');\n  }\n  // TODO: Wait until transition end.\n  setTimeout(function () {\n    document.body.classList.remove('galaxy-overlayed');\n  }, 150);\n};\n\nModal.injectOverlay = function () {\n  // Inject the overlay we use for overlaying it behind modals.\n  if (!document.querySelector('.md-overlay')) {\n    var d = document.createElement('div');\n    d.className = 'md-overlay';\n    document.body.appendChild(d);\n  }\n};\n\nModal.prototype.html = function () {\n  var d = document.createElement('div');\n  d.id = 'modal-' + this.id;\n  d.className = 'md-modal md-effect-1 ' + (this.classes || '');\n  d.style.display = 'none';\n  d.innerHTML = (\n    '<div class=\"md-content\">' +\n      '<h3>' + utils.escape(this.title) + '</h3> ' +\n      '<a class=\"md-close\" title=\"Close\"><span><div>Close</div></span></a>' +\n      '<div>' + this.content + '</div>' +\n    '</div>'\n  );\n  return d;\n};\n\nModal.prototype.inject = function () {\n  Modal.injectOverlay();\n\n  this.el = this.html();\n  this.el.style.display = 'block';\n\n  document.body.appendChild(this.el);\n  document.body.classList.add('galaxy-overlayed');\n\n  return this.el;\n};\n\nModal.prototype.open = function () {\n  this.el.classList.add('md-show');\n};\n\n\nreturn Modal;\n\n};\n","module.exports = function (window, document) {\n'use strict';\n\nfunction trace(text, level) {\n  console[level || 'log'](\n    (window.performance.now() / 1000).toFixed(3) + ': ' + text);\n}\n\n\nfunction error(text) {\n  return trace(text, 'error');\n}\n\n\nfunction warn(text) {\n  return trace(text, 'warn');\n}\n\n\nfunction polyfill() {\n  if (!('performance' in window)) {\n    window.performance = {\n      now: function () {\n        return +new Date();\n      }\n    };\n  }\n\n  if (('origin' in window.location)) {\n    window.location.origin = (window.location.protocol + '//' +\n      window.location.host);\n  }\n}\n\n\nfunction getPeerId() {\n  return (window.location.pathname.indexOf('.html') ?\n    window.location.search.substr(1) : window.location.pathname.substr(1));\n}\n\n\nvar FIELD_FOCUSED_TAGS = [\n  'input',\n  'keygen',\n  'meter',\n  'option',\n  'output',\n  'progress',\n  'select',\n  'textarea'\n];\nfunction fieldFocused(e) {\n  return FIELD_FOCUSED_TAGS.indexOf(e.target.nodeName.toLowerCase()) !== -1;\n}\n\n\nfunction hasTouchEvents() {\n  return ('ontouchstart' in window ||\n    window.DocumentTouch && document instanceof window.DocumentTouch);\n}\n\nfunction injectCSS(opts) {\n  var link = document.createElement('link');\n  link.href = opts.href;\n  link.media = 'all';\n  link.rel = 'stylesheet';\n  link.type = 'text/css';\n  Object.keys(opts || {}).forEach(function (prop) {\n    link[prop] = opts[prop];\n  });\n  document.querySelector('head').appendChild(link);\n}\n\nfunction escape(text) {\n  if (!text) {\n    return text;\n  }\n  return text.replace(/&/g, '&amp;')\n             .replace(/</g, '&lt;')\n             .replace(/>/g, '&gt;')\n             .replace(/'/g, '&#39;')\n             .replace(/\"/g, '&#34;');\n}\n\nfunction isFullScreen() {\n  return (!document.fullscreenElement &&  // standard method\n    !document.mozFullScreenElement &&\n    !document.webkitFullscreenElement &&\n    !document.msFullscreenElement);  // vendor-prefixed methods\n}\n\nfunction toggleFullScreen() {\n  if (isFullScreen()) {\n    trace('Entering full screen');\n    if (document.documentElement.requestFullscreen) {\n      document.documentElement.requestFullscreen();\n    } else if (document.documentElement.mozRequestFullScreen) {\n      document.documentElement.mozRequestFullScreen();\n    } else if (document.documentElement.webkitRequestFullscreen) {\n      document.documentElement.webkitRequestFullscreen(\n        Element.ALLOW_KEYBOARD_INPUT);\n    } else if (document.documentElement.msRequestFullscreen) {\n      document.documentElement.msRequestFullscreen();\n    }\n  } else {\n    trace('Exiting full screen');\n    if (document.exitFullscreen) {\n      document.exitFullscreen();\n    } else if (document.mozCancelFullScreen) {\n      document.mozCancelFullScreen();\n    } else if (document.webkitExitFullscreen) {\n      document.webkitExitFullscreen();\n    } else if (document.msExitFullscreen) {\n      document.msExitFullscreen();\n    }\n  }\n}\n\n\nfunction lockOrientation(orientation) {\n  var lo = (window.screen.LockOrientation ||\n    window.screen.mozLockOrientation ||\n    window.screen.webkitLockOrientation ||\n    window.screen.msLockOrientation);\n  if (!lo) {\n    return warn('Orientation could not be locked');\n  }\n\n  return lo(orientation);\n}\n\n\nfunction triggerEvent(type) {\n  var event = document.createEvent('HTMLEvents');\n  event.initEvent(type, true, true);\n  event.eventName = type;\n  (document.body || window).dispatchEvent(event);\n}\n\n\nreturn {\n  trace: trace,\n  error: error,\n  warn: warn,\n  polyfill: polyfill,\n  getPeerId: getPeerId,\n  fieldFocused: fieldFocused,\n  hasTouchEvents: hasTouchEvents,\n  injectCSS: injectCSS,\n  escape: escape,\n  isFullScreen: isFullScreen,\n  toggleFullScreen: toggleFullScreen,\n  lockOrientation: lockOrientation,\n  triggerEvent: triggerEvent\n};\n\n};\n","'use strict';\n\nvar settings_local = {};\ntry {\n  settings_local = require('./settings_local.js');\n} catch (e) {\n}\n\n\nvar settings = {\n  API_URL: 'http://localhost:5000',  // Galaxy API URL. No trailing slash.\n  DEBUG: false,\n  PEERJS_KEY: '',  // Sign up for a key at http://peerjs.com/peerserver\n  VERSION: '0.0.1'  // Version of the `gamepad.js` script\n};\n\n// Override each default setting with user-defined setting.\nObject.keys(settings_local).forEach(function (key) {\n\tsettings[key] = settings_local[key];\n});\n\n\nmodule.exports = settings;\n","module.exports = {\n  DEBUG: true,\n  PEERJS_KEY: 'rovu5xmqo69wwmi'\n};\n"]} diff --git a/dist/js/gamepad-host.min.js b/dist/js/gamepad-host.min.js index e65151f..be6338b 100755 --- a/dist/js/gamepad-host.min.js +++ b/dist/js/gamepad-host.min.js @@ -1 +1 @@ -!function(e){if("object"==typeof exports&&"undefined"!=typeof module)module.exports=e();else if("function"==typeof define&&define.amd)define([],e);else{var t;"undefined"!=typeof window?t=window:"undefined"!=typeof global?t=global:"undefined"!=typeof self&&(t=self),t.gamepad=e()}}(function(){return function e(t,n,o){function i(c,s){if(!n[c]){if(!t[c]){var l="function"==typeof require&&require;if(!s&&l)return l(c,!0);if(r)return r(c,!0);var u=new Error("Cannot find module '"+c+"'");throw u.code="MODULE_NOT_FOUND",u}var a=n[c]={exports:{}};t[c][0].call(a.exports,function(e){var n=t[c][1][e];return i(n?n:e)},a,a.exports,e,t,n,o)}return n[c].exports}for(var r="function"==typeof require&&require,c=0;c'+l+'

Code

'+c+"

",d=new r({id:"pairing-screen",classes:"slim",title:"Pair your mobile phone",content:u},!0);n.setTimeout(function(){d.open()},150),["https://fonts.googleapis.com/css?family=Source+Sans+Pro:300,400,700","/css/modal.css"].forEach(function(e){s.injectCSS({href:e})}),i.peerConnect(e).then(function(e){console.log("Peer connected"),d.close(),t(e)})}).catch(console.error.bind(console))})},i._updateState=function(e){Object.keys(e||{}).forEach(function(t){!this.state[t]&&e[t]?(i._emit("buttondown",t),i._emit("buttondown."+t,t)):this.state[t]&&!e[t]&&(i._emit("buttonup",t),i._emit("buttonup."+t,t))}.bind(this))},i.hidePairingScreen=function(){r.closeAll()},i._emit=function(e,t){(this.listeners[e]||[]).forEach(function(e){e.apply(e,[t])})},i._bind=function(e,t){return"undefined"==typeof this.listeners[event]&&(this.listeners[event]=[]),this.listeners[event].push(t),this},Gamepad.prototype.unbind=function(e,t){return"undefined"==typeof e?(this.listeners={},void 0):"undefined"==typeof t?(this.listeners[e]=[],void 0):"undefined"==typeof this.listeners[e]?!1:(this.listeners[e].forEach(function(n,o){return n===t?(this.listeners[e].splice(o,1),!0):void 0}),!1)},i.buttons={a:{clicked:i._bind}},i.version=c.VERSION;var a=n.location.origin,d=o.querySelector("[data-galaxy-origin]");d&&(i.galaxyOrigin=d.dataset.galaxyOrigin),t.exports=i}(window,document)},{"./lib/modal":2,"./lib/utils":3,"./settings":4}],2:[function(e,t){function n(e,t){Object.keys(e).forEach(function(t){this[t]=e[t]}.bind(this)),t&&this.inject()}var o=e("./utils");n.closeAll=n.prototype.close=function(){var e=document.querySelector(".md-show");e&&e.classList.remove("md-show"),setTimeout(function(){document.body.classList.remove("galaxy-overlayed")},150)},n.injectOverlay=function(){if(!document.querySelector(".md-overlay")){var e=document.createElement("div");e.className="md-overlay",document.body.appendChild(e)}},n.prototype.html=function(){var e=document.createElement("div");return e.id="modal-"+this.id,e.className="md-modal md-effect-1 "+(this.classes||""),e.style.display="none",e.innerHTML='

'+o.escape(this.title)+'

Close
'+this.content+"
",e},n.prototype.inject=function(){return n.injectOverlay(),this.el=this.html(),this.el.style.display="block",document.body.appendChild(this.el),document.body.classList.add("galaxy-overlayed"),this.el},n.prototype.open=function(){this.el.classList.add("md-show")},t.exports=n},{"./utils":3}],3:[function(e,t){function n(e,t){console[t||"log"]((window.performance.now()/1e3).toFixed(3)+": "+e)}function o(e){return n(e,"error")}function i(e){return n(e,"warn")}function r(e){"performance"in e||(e.performance={now:function(){return+new Date}}),"origin"in e.location&&(e.location.origin=e.location.protocol+"//"+e.location.host)}function c(){return window.location.pathname.indexOf(".html")?window.location.search.substr(1):window.location.pathname.substr(1)}function s(e){return-1!==h.indexOf(e.target.nodeName.toLowerCase())}function l(){return"ontouchstart"in window||window.DocumentTouch&&document instanceof DocumentTouch}function u(e){var t=document.createElement("link");t.href=e.href,t.media="all",t.rel="stylesheet",t.type="text/css",Object.keys(e||{}).forEach(function(n){t[n]=e[n]}),document.querySelector("head").appendChild(t)}function a(e){return e?e.replace(/&/g,"&").replace(//g,">").replace(/'/g,"'").replace(/"/g,"""):e}function d(){return!(document.fullscreenElement||document.mozFullScreenElement||document.webkitFullscreenElement||document.msFullscreenElement)}function m(){d()?(n("Entering full screen"),document.documentElement.requestFullscreen?document.documentElement.requestFullscreen():document.documentElement.mozRequestFullScreen?document.documentElement.mozRequestFullScreen():document.documentElement.webkitRequestFullscreen?document.documentElement.webkitRequestFullscreen(Element.ALLOW_KEYBOARD_INPUT):document.documentElement.msRequestFullscreen&&document.documentElement.msRequestFullscreen()):(n("Exiting full screen"),document.exitFullscreen?document.exitFullscreen():document.mozCancelFullScreen?document.mozCancelFullScreen():document.webkitExitFullscreen?document.webkitExitFullscreen():document.msExitFullscreen&&document.msExitFullscreen())}function f(){var e=screen.LockOrientation||screen.mozLockOrientation||screen.webkitLockOrientation||screen.msLockOrientation;return e?(e(orientation),void 0):i("Orientation could not be locked")}function p(e){var t=document.createEvent("HTMLEvents");t.initEvent(e,!0,!0),t.eventName=e,(document.body||window).dispatchEvent(t)}var h=["input","keygen","meter","option","output","progress","select","textarea"];t.exports.trace=n,t.exports.error=o,t.exports.warn=i,t.exports.polyfill=r,t.exports.getPeerId=c,t.exports.fieldFocused=s,t.exports.hasTouchEvents=l,t.exports.injectCSS=u,t.exports.escape=a,t.exports.isFullScreen=d,t.exports.toggleFullScreen=m,t.exports.lockOrientation=f,t.exports.triggerEvent=p},{}],4:[function(e,t){var n={};try{n=e("./settings_local.js")}catch(o){}var i={API_URL:"http://localhost:5000",DEBUG:!1,PEERJS_KEY:"",VERSION:"0.0.1"};for(var r in n)i[r]=n[r];t.exports=i},{"./settings_local.js":5}],5:[function(e,t){t.exports={DEBUG:!0,PEERJS_KEY:"rovu5xmqo69wwmi"}},{}]},{},[1])(1)}); \ No newline at end of file +!function(e){if("object"==typeof exports&&"undefined"!=typeof module)module.exports=e();else if("function"==typeof define&&define.amd)define([],e);else{var t;"undefined"!=typeof window?t=window:"undefined"!=typeof global?t=global:"undefined"!=typeof self&&(t=self),t.gamepad=e()}}(function(){return function e(t,n,o){function i(s,c){if(!n[s]){if(!t[s]){var l="function"==typeof require&&require;if(!c&&l)return l(s,!0);if(r)return r(s,!0);var u=new Error("Cannot find module '"+s+"'");throw u.code="MODULE_NOT_FOUND",u}var a=n[s]={exports:{}};t[s][0].call(a.exports,function(e){var n=t[s][1][e];return i(n?n:e)},a,a.exports,e,t,n,o)}return n[s].exports}for(var r="function"==typeof require&&require,s=0;s'+l+'

Code

'+s+"

",d=new r({id:"pairing-screen",classes:"slim",title:"Pair your mobile phone",content:u},!0);n.setTimeout(function(){d.open()},150),["https://fonts.googleapis.com/css?family=Source+Sans+Pro:300,400,700","/css/modal.css"].forEach(function(e){c.injectCSS({href:e})}),i.peerConnect(e).then(function(e){console.log("Peer connected"),d.close(),t(e)})}).catch(console.error.bind(console))})},i._updateState=function(e){Object.keys(e||{}).forEach(function(t){!this.state[t]&&e[t]?(i._emit("buttondown",t),i._emit("buttondown."+t,t)):this.state[t]&&!e[t]&&(i._emit("buttonup",t),i._emit("buttonup."+t,t))}.bind(this))},i.hidePairingScreen=function(){r.closeAll()},i._emit=function(e,t){(this.listeners[e]||[]).forEach(function(e){e.apply(e,[t])})},i._bind=function(e,t){return"undefined"==typeof this.listeners[event]&&(this.listeners[event]=[]),this.listeners[event].push(t),this},i.prototype.unbind=function(e,t){return"undefined"==typeof e?(this.listeners={},void 0):"undefined"==typeof t?(this.listeners[e]=[],void 0):"undefined"==typeof this.listeners[e]?!1:(this.listeners[e].forEach(function(n,o){return n===t?(this.listeners[e].splice(o,1),!0):void 0}),!1)},i.buttons={a:{clicked:i._bind}},i.version=s.VERSION;var a=n.location.origin,d=o.querySelector("[data-galaxy-origin]");d&&(i.galaxyOrigin=d.dataset.galaxyOrigin),t.exports=i}(window,document)},{"./lib/modal":2,"./lib/utils":3,"./settings":4}],2:[function(e,t){t.exports=function(t,n){"use strict";function o(e,t){Object.keys(e).forEach(function(t){this[t]=e[t]}.bind(this)),t&&this.inject()}var i=e("./utils");return o.closeAll=o.prototype.close=function(){var e=n.querySelector(".md-show");e&&e.classList.remove("md-show"),setTimeout(function(){n.body.classList.remove("galaxy-overlayed")},150)},o.injectOverlay=function(){if(!n.querySelector(".md-overlay")){var e=n.createElement("div");e.className="md-overlay",n.body.appendChild(e)}},o.prototype.html=function(){var e=n.createElement("div");return e.id="modal-"+this.id,e.className="md-modal md-effect-1 "+(this.classes||""),e.style.display="none",e.innerHTML='

'+i.escape(this.title)+'

Close
'+this.content+"
",e},o.prototype.inject=function(){return o.injectOverlay(),this.el=this.html(),this.el.style.display="block",n.body.appendChild(this.el),n.body.classList.add("galaxy-overlayed"),this.el},o.prototype.open=function(){this.el.classList.add("md-show")},o}},{"./utils":3}],3:[function(e,t){t.exports=function(e,t){"use strict";function n(t,n){console[n||"log"]((e.performance.now()/1e3).toFixed(3)+": "+t)}function o(e){return n(e,"error")}function i(e){return n(e,"warn")}function r(){"performance"in e||(e.performance={now:function(){return+new Date}}),"origin"in e.location&&(e.location.origin=e.location.protocol+"//"+e.location.host)}function s(){return e.location.pathname.indexOf(".html")?e.location.search.substr(1):e.location.pathname.substr(1)}function c(e){return-1!==h.indexOf(e.target.nodeName.toLowerCase())}function l(){return"ontouchstart"in e||e.DocumentTouch&&t instanceof e.DocumentTouch}function u(e){var n=t.createElement("link");n.href=e.href,n.media="all",n.rel="stylesheet",n.type="text/css",Object.keys(e||{}).forEach(function(t){n[t]=e[t]}),t.querySelector("head").appendChild(n)}function a(e){return e?e.replace(/&/g,"&").replace(//g,">").replace(/'/g,"'").replace(/"/g,"""):e}function d(){return!(t.fullscreenElement||t.mozFullScreenElement||t.webkitFullscreenElement||t.msFullscreenElement)}function f(){d()?(n("Entering full screen"),t.documentElement.requestFullscreen?t.documentElement.requestFullscreen():t.documentElement.mozRequestFullScreen?t.documentElement.mozRequestFullScreen():t.documentElement.webkitRequestFullscreen?t.documentElement.webkitRequestFullscreen(Element.ALLOW_KEYBOARD_INPUT):t.documentElement.msRequestFullscreen&&t.documentElement.msRequestFullscreen()):(n("Exiting full screen"),t.exitFullscreen?t.exitFullscreen():t.mozCancelFullScreen?t.mozCancelFullScreen():t.webkitExitFullscreen?t.webkitExitFullscreen():t.msExitFullscreen&&t.msExitFullscreen())}function p(t){var n=e.screen.LockOrientation||e.screen.mozLockOrientation||e.screen.webkitLockOrientation||e.screen.msLockOrientation;return n?n(t):i("Orientation could not be locked")}function m(n){var o=t.createEvent("HTMLEvents");o.initEvent(n,!0,!0),o.eventName=n,(t.body||e).dispatchEvent(o)}var h=["input","keygen","meter","option","output","progress","select","textarea"];return{trace:n,error:o,warn:i,polyfill:r,getPeerId:s,fieldFocused:c,hasTouchEvents:l,injectCSS:u,escape:a,isFullScreen:d,toggleFullScreen:f,lockOrientation:p,triggerEvent:m}}},{}],4:[function(e,t){"use strict";var n={};try{n=e("./settings_local.js")}catch(o){}var i={API_URL:"http://localhost:5000",DEBUG:!1,PEERJS_KEY:"",VERSION:"0.0.1"};Object.keys(n).forEach(function(e){i[e]=n[e]}),t.exports=i},{"./settings_local.js":5}],5:[function(e,t){t.exports={DEBUG:!0,PEERJS_KEY:"rovu5xmqo69wwmi"}},{}]},{},[1])(1)}); \ No newline at end of file diff --git a/src/js/client.js b/src/js/client.js index e0711b1..05ab5df 100644 --- a/src/js/client.js +++ b/src/js/client.js @@ -9,7 +9,7 @@ var error = utils.error; var trace = utils.trace; -utils.polyfill(window); +utils.polyfill(); utils.lockOrientation('landscape-primary'); @@ -28,8 +28,8 @@ document.addEventListener('keyup', function (e) { trace('User pressed "F"; entering/exiting fullscreen'); return utils.toggleFullScreen(); case 78: // Pressing NF (really just N) should toggle full-screen mode. - trace('User pressed "NF"; exiting fullscreen and will not automatically ' + - 'open next time'); + trace('User pressed "NF"; exiting fullscreen and will not ' + + 'automatically open next time'); localStorage.disableAutoFullScreen = '1'; return utils.toggleFullScreen(); } @@ -51,7 +51,7 @@ document.addEventListener('click', function (e) { var peerId = utils.getPeerId(); -var peer = new Peer('controller_' + peerId, { +var peer = new window.Peer('controller_' + peerId, { key: settings.PEERJS_KEY, debug: settings.DEBUG ? 3 : 0 }); @@ -78,7 +78,8 @@ conn.on('open', function () { function send(msg) { if (settings.DEBUG) { - console.log('Sent: ' + (typeof msg === 'object' ? JSON.stringify(msg) : msg)); + console.log('Sent: ' + + (typeof msg === 'object' ? JSON.stringify(msg) : msg)); } conn.send(msg); } @@ -115,7 +116,8 @@ function angularShape(canvas, coords) { } function linearFill(shape, color1, color2, coords) { - var bg = shape.createLinearGradient(coords[0], coords[1], coords[2], coords[3]); + var bg = shape.createLinearGradient(coords[0], coords[1], coords[2], + coords[3]); bg.addColorStop(0, color1); bg.addColorStop(1, color2); shape.fillStyle = bg; @@ -186,15 +188,16 @@ var gamepadState = { function bindPress(button, eventName, isPressed) { - document.querySelector('#' + button).addEventListener(eventName, function (e) { - // Handle D-pad presses. - if (e.target && e.target.parentNode === dpad) { - dpad.classList.toggle(this.id); - } + document.querySelector('#' + button) + .addEventListener(eventName, function (e) { + // Handle D-pad presses. + if (e.target && e.target.parentNode === dpad) { + dpad.classList.toggle(this.id); + } - gamepadState[button] = isPressed; - send({type: 'state', data: gamepadState}); - }); + gamepadState[button] = isPressed; + send({type: 'state', data: gamepadState}); + }); } diff --git a/src/js/host.js b/src/js/host.js index 92a1922..57d6497 100644 --- a/src/js/host.js +++ b/src/js/host.js @@ -3,14 +3,14 @@ // var peer = require('./lib/peer'); // var Promise = require('./lib/promise-1.0.0.js'); // jshint ignore:line -var Modal = require('./lib/modal'); +var Modal = require('./lib/modal')(window, document); var settings = require('./settings'); -var utils = require('./lib/utils'); +var utils = require('./lib/utils')(window, document); var error = utils.error; var trace = utils.trace; -utils.polyfill(window); +utils.polyfill(); /** @@ -39,12 +39,12 @@ gamepad.state = {}; * @memberOf gamepad */ gamepad.peerHandshake = function (peerId) { - return new Promise(function (resolve, reject) { + return new Promise(function (resolve) { if (!peerId) { peerId = utils.getPeerId(); // The host ID. } - var peer = new Peer(peerId, { + var peer = new window.Peer(peerId, { key: settings.PEERJS_KEY, debug: settings.DEBUG ? 3 : 0 }); @@ -78,11 +78,13 @@ gamepad.peerConnect = function (peer) { gamepad._updateState(data.data); break; default: - console.warn('WebRTC message received of unknown type: "' + data.type + '"'); + console.warn( + 'WebRTC message received of unknown type: "' + data.type + '"'); break; } - trace('Received: ' + (typeof data === 'object' ? JSON.stringify(data) : '')); + trace('Received: ' + + (typeof data === 'object' ? JSON.stringify(data) : '')); }); conn.on('error', function (err) { @@ -109,16 +111,20 @@ gamepad.pair = function (peerId) { return new Promise(function (resolve) { return gamepad.peerHandshake(peerId).then(function (peer) { - var pairId = peer.id; // This should be the same as `peerId`, but this comes from PeerJS, which is the source of truth. + // `pairId` should be the same as `peerId`, + // but `peer.id` is the source of truth. + var pairId = peer.id; var pairIdEsc = encodeURIComponent(pairId); var pairUrl = galaxyOrigin + '/client.html?' + pairIdEsc; // Update the querystring in the address bar. - window.history.replaceState(null, null, window.location.pathname + '?' + pairIdEsc); + window.history.replaceState(null, null, + window.location.pathname + '?' + pairIdEsc); var content = ( '' ); @@ -219,7 +225,7 @@ gamepad._bind = function (eventName, listener) { * @param {Function} [listener] (Optional) The listener function to remove. * @return {Boolean} Was unbinding the listener successful. */ -Gamepad.prototype.unbind = function (eventName, listener) { +gamepad.prototype.unbind = function (eventName, listener) { // Remove everything for all event types. if (typeof eventName === 'undefined') { this.listeners = {}; diff --git a/src/js/lib/modal.js b/src/js/lib/modal.js index 533701c..a120ce3 100644 --- a/src/js/lib/modal.js +++ b/src/js/lib/modal.js @@ -1,3 +1,6 @@ +module.exports = function (window, document) { +'use strict'; + var utils = require('./utils'); @@ -65,4 +68,6 @@ Modal.prototype.open = function () { }; -module.exports = Modal; +return Modal; + +}; diff --git a/src/js/lib/utils.js b/src/js/lib/utils.js index c61df42..b64c783 100644 --- a/src/js/lib/utils.js +++ b/src/js/lib/utils.js @@ -1,5 +1,9 @@ +module.exports = function (window, document) { +'use strict'; + function trace(text, level) { - console[level || 'log']((window.performance.now() / 1000).toFixed(3) + ': ' + text); + console[level || 'log']( + (window.performance.now() / 1000).toFixed(3) + ': ' + text); } @@ -13,17 +17,18 @@ function warn(text) { } -function polyfill(win) { - if (!('performance' in win)) { - win.performance = { +function polyfill() { + if (!('performance' in window)) { + window.performance = { now: function () { return +new Date(); } }; } - if (('origin' in win.location)) { - win.location.origin = win.location.protocol + '//' + win.location.host; + if (('origin' in window.location)) { + window.location.origin = (window.location.protocol + '//' + + window.location.host); } } @@ -51,7 +56,7 @@ function fieldFocused(e) { function hasTouchEvents() { return ('ontouchstart' in window || - window.DocumentTouch && document instanceof DocumentTouch); + window.DocumentTouch && document instanceof window.DocumentTouch); } function injectCSS(opts) { @@ -92,7 +97,8 @@ function toggleFullScreen() { } else if (document.documentElement.mozRequestFullScreen) { document.documentElement.mozRequestFullScreen(); } else if (document.documentElement.webkitRequestFullscreen) { - document.documentElement.webkitRequestFullscreen(Element.ALLOW_KEYBOARD_INPUT); + document.documentElement.webkitRequestFullscreen( + Element.ALLOW_KEYBOARD_INPUT); } else if (document.documentElement.msRequestFullscreen) { document.documentElement.msRequestFullscreen(); } @@ -111,16 +117,16 @@ function toggleFullScreen() { } -function lockOrientation() { - var lo = (screen.LockOrientation || - screen.mozLockOrientation || - screen.webkitLockOrientation || - screen.msLockOrientation); +function lockOrientation(orientation) { + var lo = (window.screen.LockOrientation || + window.screen.mozLockOrientation || + window.screen.webkitLockOrientation || + window.screen.msLockOrientation); if (!lo) { return warn('Orientation could not be locked'); } - lo(orientation); + return lo(orientation); } @@ -132,16 +138,20 @@ function triggerEvent(type) { } -module.exports.trace = trace; -module.exports.error = error; -module.exports.warn = warn; -module.exports.polyfill = polyfill; -module.exports.getPeerId = getPeerId; -module.exports.fieldFocused = fieldFocused; -module.exports.hasTouchEvents = hasTouchEvents; -module.exports.injectCSS = injectCSS; -module.exports.escape = escape; -module.exports.isFullScreen = isFullScreen; -module.exports.toggleFullScreen = toggleFullScreen; -module.exports.lockOrientation = lockOrientation; -module.exports.triggerEvent = triggerEvent; +return { + trace: trace, + error: error, + warn: warn, + polyfill: polyfill, + getPeerId: getPeerId, + fieldFocused: fieldFocused, + hasTouchEvents: hasTouchEvents, + injectCSS: injectCSS, + escape: escape, + isFullScreen: isFullScreen, + toggleFullScreen: toggleFullScreen, + lockOrientation: lockOrientation, + triggerEvent: triggerEvent +}; + +}; diff --git a/src/js/settings.js b/src/js/settings.js index 07fc72e..f0a710b 100644 --- a/src/js/settings.js +++ b/src/js/settings.js @@ -1,18 +1,23 @@ +'use strict'; + var settings_local = {}; try { settings_local = require('./settings_local.js'); } catch (e) { } + var settings = { - API_URL: 'http://localhost:5000', // This URL to the Galaxy API. No trailing slash. + API_URL: 'http://localhost:5000', // Galaxy API URL. No trailing slash. DEBUG: false, PEERJS_KEY: '', // Sign up for a key at http://peerjs.com/peerserver VERSION: '0.0.1' // Version of the `gamepad.js` script }; -for (var key in settings_local) { - settings[key] = settings_local[key]; -} +// Override each default setting with user-defined setting. +Object.keys(settings_local).forEach(function (key) { + settings[key] = settings_local[key]; +}); + module.exports = settings;