!function(e){if("object"==typeof exports&&"undefined"!=typeof module)module.exports=e();else if("function"==typeof define&&define.amd)define([],e);else{var f;"undefined"!=typeof window?f=window:"undefined"!=typeof global?f=global:"undefined"!=typeof self&&(f=self),f.keyboardJS=e()}}(function(){var define,module,exports;return (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 -1) { otherSubCombo.splice(index, 1); } } if (otherSubCombo.length !== 0) { return false; } } return true; }; KeyCombo._splitStr = function(str, deliminator) { var s = str; var d = deliminator; var c = ''; var ca = []; for (var ci = 0; ci < s.length; ci += 1) { if (ci > 0 && s[ci] === d && s[ci - 1] !== '\\') { ca.push(c.trim()); c = ''; ci += 1; } c += s[ci]; } if (c) { ca.push(c.trim()); } return ca; }; KeyCombo.prototype._checkSubCombo = function(subCombo, startingKeyNameIndex, pressedKeyNames) { subCombo = subCombo.slice(0); pressedKeyNames = pressedKeyNames.slice(startingKeyNameIndex); var endIndex = startingKeyNameIndex; for (var i = 0; i < subCombo.length; i += 1) { var keyName = subCombo[i]; if (keyName[0] === '\\') { var escapedKeyName = keyName.slice(1); if ( escapedKeyName === KeyCombo.comboDeliminator || escapedKeyName === KeyCombo.keyDeliminator ) { keyName = escapedKeyName; } } var index = pressedKeyNames.indexOf(keyName); if (index > -1) { subCombo.splice(i, 1); i -= 1; if (index > endIndex) { endIndex = index; } if (subCombo.length === 0) { return endIndex; } } } return -1; }; module.exports = KeyCombo; },{}],3:[function(require,module,exports){ (function (global){ var Locale = require('./locale'); var KeyCombo = require('./key-combo'); function Keyboard(targetWindow, targetElement, platform, userAgent) { this._locale = null; this._currentContext = null; this._contexts = {}; this._listeners = []; this._appliedListeners = []; this._locales = {}; this._targetElement = null; this._targetWindow = null; this._targetPlatform = ''; this._targetUserAgent = ''; this._isModernBrowser = false; this._targetKeyDownBinding = null; this._targetKeyUpBinding = null; this._targetResetBinding = null; this._paused = false; this.setContext('global'); this.watch(targetWindow, targetElement, platform, userAgent); } Keyboard.prototype.setLocale = function(localeName, localeBuilder) { var locale = null; if (typeof localeName === 'string') { if (localeBuilder) { locale = new Locale(localeName); localeBuilder(locale, this._targetPlatform, this._targetUserAgent); } else { locale = this._locales[localeName] || null; } } else { locale = localeName; localeName = locale._localeName; } this._locale = locale; this._locales[localeName] = locale; if (locale) { this._locale.pressedKeys = locale.pressedKeys; } }; Keyboard.prototype.getLocale = function(localName) { localName || (localName = this._locale.localeName); return this._locales[localName] || null; }; Keyboard.prototype.bind = function(keyComboStr, pressHandler, releaseHandler, preventRepeatByDefault) { if (keyComboStr === null || typeof keyComboStr === 'function') { preventRepeatByDefault = releaseHandler; releaseHandler = pressHandler; pressHandler = keyComboStr; keyComboStr = null; } if ( keyComboStr && typeof keyComboStr === 'object' && typeof keyComboStr.length === 'number' ) { for (var i = 0; i < keyComboStr.length; i += 1) { this.bind(keyComboStr[i], pressHandler, releaseHandler); } return; } this._listeners.push({ keyCombo : keyComboStr ? new KeyCombo(keyComboStr) : null, pressHandler : pressHandler || null, releaseHandler : releaseHandler || null, preventRepeat : preventRepeatByDefault || false, preventRepeatByDefault : preventRepeatByDefault || false }); }; Keyboard.prototype.addListener = Keyboard.prototype.bind; Keyboard.prototype.on = Keyboard.prototype.bind; Keyboard.prototype.unbind = function(keyComboStr, pressHandler, releaseHandler) { if (keyComboStr === null || typeof keyComboStr === 'function') { releaseHandler = pressHandler; pressHandler = keyComboStr; keyComboStr = null; } if ( keyComboStr && typeof keyComboStr === 'object' && typeof keyComboStr.length === 'number' ) { for (var i = 0; i < keyComboStr.length; i += 1) { this.unbind(keyComboStr[i], pressHandler, releaseHandler); } return; } for (var i = 0; i < this._listeners.length; i += 1) { var listener = this._listeners[i]; var comboMatches = !keyComboStr && !listener.keyCombo || listener.keyCombo && listener.keyCombo.isEqual(keyComboStr); var pressHandlerMatches = !pressHandler && !releaseHandler || !pressHandler && !listener.pressHandler || pressHandler === listener.pressHandler; var releaseHandlerMatches = !pressHandler && !releaseHandler || !releaseHandler && !listener.releaseHandler || releaseHandler === listener.releaseHandler; if (comboMatches && pressHandlerMatches && releaseHandlerMatches) { this._listeners.splice(i, 1); i -= 1; } } }; Keyboard.prototype.removeListener = Keyboard.prototype.unbind; Keyboard.prototype.off = Keyboard.prototype.unbind; Keyboard.prototype.setContext = function(contextName) { if(this._locale) { this.releaseAllKeys(); } if (!this._contexts[contextName]) { this._contexts[contextName] = []; } this._listeners = this._contexts[contextName]; this._currentContext = contextName; }; Keyboard.prototype.getContext = function() { return this._currentContext; }; Keyboard.prototype.withContext = function(contextName, callback) { var previousContextName = this.getContext(); this.setContext(contextName); callback(); this.setContext(previousContextName); }; Keyboard.prototype.watch = function(targetWindow, targetElement, targetPlatform, targetUserAgent) { var _this = this; this.stop(); if (!targetWindow) { if (!global.addEventListener && !global.attachEvent) { throw new Error('Cannot find global functions addEventListener or attachEvent.'); } targetWindow = global; } if (typeof targetWindow.nodeType === 'number') { targetUserAgent = targetPlatform; targetPlatform = targetElement; targetElement = targetWindow; targetWindow = global; } if (!targetWindow.addEventListener && !targetWindow.attachEvent) { throw new Error('Cannot find addEventListener or attachEvent methods on targetWindow.'); } this._isModernBrowser = !!targetWindow.addEventListener; var userAgent = targetWindow.navigator && targetWindow.navigator.userAgent || ''; var platform = targetWindow.navigator && targetWindow.navigator.platform || ''; targetElement && targetElement !== null || (targetElement = targetWindow.document); targetPlatform && targetPlatform !== null || (targetPlatform = platform); targetUserAgent && targetUserAgent !== null || (targetUserAgent = userAgent); this._targetKeyDownBinding = function(event) { _this.pressKey(event.keyCode, event); }; this._targetKeyUpBinding = function(event) { _this.releaseKey(event.keyCode, event); }; this._targetResetBinding = function(event) { _this.releaseAllKeys(event) }; this._bindEvent(targetElement, 'keydown', this._targetKeyDownBinding); this._bindEvent(targetElement, 'keyup', this._targetKeyUpBinding); this._bindEvent(targetWindow, 'focus', this._targetResetBinding); this._bindEvent(targetWindow, 'blur', this._targetResetBinding); this._targetElement = targetElement; this._targetWindow = targetWindow; this._targetPlatform = targetPlatform; this._targetUserAgent = targetUserAgent; }; Keyboard.prototype.stop = function() { var _this = this; if (!this._targetElement || !this._targetWindow) { return; } this._unbindEvent(this._targetElement, 'keydown', this._targetKeyDownBinding); this._unbindEvent(this._targetElement, 'keyup', this._targetKeyUpBinding); this._unbindEvent(this._targetWindow, 'focus', this._targetResetBinding); this._unbindEvent(this._targetWindow, 'blur', this._targetResetBinding); this._targetWindow = null; this._targetElement = null; }; Keyboard.prototype.pressKey = function(keyCode, event) { if (this._paused) { return; } if (!this._locale) { throw new Error('Locale not set'); } this._locale.pressKey(keyCode); this._applyBindings(event); }; Keyboard.prototype.releaseKey = function(keyCode, event) { if (this._paused) { return; } if (!this._locale) { throw new Error('Locale not set'); } this._locale.releaseKey(keyCode); this._clearBindings(event); }; Keyboard.prototype.releaseAllKeys = function(event) { if (this._paused) { return; } if (!this._locale) { throw new Error('Locale not set'); } this._locale.pressedKeys.length = 0; this._clearBindings(event); }; Keyboard.prototype.pause = function() { if (this._paused) { return; } if (this._locale) { this.releaseAllKeys(); } this._paused = true; }; Keyboard.prototype.resume = function() { this._paused = false; }; Keyboard.prototype.reset = function() { this.releaseAllKeys(); this._listeners.length = 0; }; Keyboard.prototype._bindEvent = function(targetElement, eventName, handler) { return this._isModernBrowser ? targetElement.addEventListener(eventName, handler, false) : targetElement.attachEvent('on' + eventName, handler); }; Keyboard.prototype._unbindEvent = function(targetElement, eventName, handler) { return this._isModernBrowser ? targetElement.removeEventListener(eventName, handler, false) : targetElement.detachEvent('on' + eventName, handler); }; Keyboard.prototype._getGroupedListeners = function() { var listenerGroups = []; var listenerGroupMap = []; var listeners = this._listeners; if (this._currentContext !== 'global') { listeners = [].concat(listeners, this._contexts.global); } listeners.sort(function(a, b) { return (b.keyCombo ? b.keyCombo.keyNames.length : 0) - (a.keyCombo ? a.keyCombo.keyNames.length : 0); }).forEach(function(l) { var mapIndex = -1; for (var i = 0; i < listenerGroupMap.length; i += 1) { if (listenerGroupMap[i] === null && l.keyCombo === null || listenerGroupMap[i] !== null && listenerGroupMap[i].isEqual(l.keyCombo)) { mapIndex = i; } } if (mapIndex === -1) { mapIndex = listenerGroupMap.length; listenerGroupMap.push(l.keyCombo); } if (!listenerGroups[mapIndex]) { listenerGroups[mapIndex] = []; } listenerGroups[mapIndex].push(l); }); return listenerGroups; }; Keyboard.prototype._applyBindings = function(event) { var preventRepeat = false; event || (event = {}); event.preventRepeat = function() { preventRepeat = true; }; event.pressedKeys = this._locale.pressedKeys.slice(0); var pressedKeys = this._locale.pressedKeys.slice(0); var listenerGroups = this._getGroupedListeners(); for (var i = 0; i < listenerGroups.length; i += 1) { var listeners = listenerGroups[i]; var keyCombo = listeners[0].keyCombo; if (keyCombo === null || keyCombo.check(pressedKeys)) { for (var j = 0; j < listeners.length; j += 1) { var listener = listeners[j]; if (keyCombo === null) { listener = { keyCombo : new KeyCombo(pressedKeys.join('+')), pressHandler : listener.pressHandler, releaseHandler : listener.releaseHandler, preventRepeat : listener.preventRepeat, preventRepeatByDefault : listener.preventRepeatByDefault }; } if (listener.pressHandler && !listener.preventRepeat) { listener.pressHandler.call(this, event); if (preventRepeat) { listener.preventRepeat = preventRepeat; preventRepeat = false; } } if (listener.releaseHandler && this._appliedListeners.indexOf(listener) === -1) { this._appliedListeners.push(listener); } } if (keyCombo) { for (var j = 0; j < keyCombo.keyNames.length; j += 1) { var index = pressedKeys.indexOf(keyCombo.keyNames[j]); if (index !== -1) { pressedKeys.splice(index, 1); j -= 1; } } } } } }; Keyboard.prototype._clearBindings = function(event) { event || (event = {}); for (var i = 0; i < this._appliedListeners.length; i += 1) { var listener = this._appliedListeners[i]; var keyCombo = listener.keyCombo; if (keyCombo === null || !keyCombo.check(this._locale.pressedKeys)) { listener.preventRepeat = listener.preventRepeatByDefault; listener.releaseHandler.call(this, event); this._appliedListeners.splice(i, 1); i -= 1; } } }; module.exports = Keyboard; }).call(this,typeof global !== "undefined" ? global : typeof self !== "undefined" ? self : typeof window !== "undefined" ? window : {}) //# sourceMappingURL=data:application/json;charset:utf-8;base64,{"version":3,"sources":["lib/keyboard.js"],"names":[],"mappings":";AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["\nvar Locale = require('./locale');\nvar KeyCombo = require('./key-combo');\n\n\nfunction Keyboard(targetWindow, targetElement, platform, userAgent) {\n  this._locale               = null;\n  this._currentContext       = null;\n  this._contexts             = {};\n  this._listeners            = [];\n  this._appliedListeners     = [];\n  this._locales              = {};\n  this._targetElement        = null;\n  this._targetWindow         = null;\n  this._targetPlatform       = '';\n  this._targetUserAgent      = '';\n  this._isModernBrowser      = false;\n  this._targetKeyDownBinding = null;\n  this._targetKeyUpBinding   = null;\n  this._targetResetBinding   = null;\n  this._paused               = false;\n\n  this.setContext('global');\n  this.watch(targetWindow, targetElement, platform, userAgent);\n}\n\nKeyboard.prototype.setLocale = function(localeName, localeBuilder) {\n  var locale = null;\n  if (typeof localeName === 'string') {\n\n    if (localeBuilder) {\n      locale = new Locale(localeName);\n      localeBuilder(locale, this._targetPlatform, this._targetUserAgent);\n    } else {\n      locale = this._locales[localeName] || null;\n    }\n  } else {\n    locale     = localeName;\n    localeName = locale._localeName;\n  }\n\n  this._locale              = locale;\n  this._locales[localeName] = locale;\n  if (locale) {\n    this._locale.pressedKeys = locale.pressedKeys;\n  }\n};\n\nKeyboard.prototype.getLocale = function(localName) {\n  localName || (localName = this._locale.localeName);\n  return this._locales[localName] || null;\n};\n\nKeyboard.prototype.bind = function(keyComboStr, pressHandler, releaseHandler, preventRepeatByDefault) {\n  if (keyComboStr === null || typeof keyComboStr === 'function') {\n    preventRepeatByDefault = releaseHandler;\n    releaseHandler         = pressHandler;\n    pressHandler           = keyComboStr;\n    keyComboStr            = null;\n  }\n\n  if (\n    keyComboStr &&\n    typeof keyComboStr === 'object' &&\n    typeof keyComboStr.length === 'number'\n  ) {\n    for (var i = 0; i < keyComboStr.length; i += 1) {\n      this.bind(keyComboStr[i], pressHandler, releaseHandler);\n    }\n    return;\n  }\n\n  this._listeners.push({\n    keyCombo               : keyComboStr ? new KeyCombo(keyComboStr) : null,\n    pressHandler           : pressHandler           || null,\n    releaseHandler         : releaseHandler         || null,\n    preventRepeat          : preventRepeatByDefault || false,\n    preventRepeatByDefault : preventRepeatByDefault || false\n  });\n};\nKeyboard.prototype.addListener = Keyboard.prototype.bind;\nKeyboard.prototype.on          = Keyboard.prototype.bind;\n\nKeyboard.prototype.unbind = function(keyComboStr, pressHandler, releaseHandler) {\n  if (keyComboStr === null || typeof keyComboStr === 'function') {\n    releaseHandler = pressHandler;\n    pressHandler   = keyComboStr;\n    keyComboStr = null;\n  }\n\n  if (\n    keyComboStr &&\n    typeof keyComboStr === 'object' &&\n    typeof keyComboStr.length === 'number'\n  ) {\n    for (var i = 0; i < keyComboStr.length; i += 1) {\n      this.unbind(keyComboStr[i], pressHandler, releaseHandler);\n    }\n    return;\n  }\n\n  for (var i = 0; i < this._listeners.length; i += 1) {\n    var listener = this._listeners[i];\n\n    var comboMatches          = !keyComboStr && !listener.keyCombo ||\n                                listener.keyCombo && listener.keyCombo.isEqual(keyComboStr);\n    var pressHandlerMatches   = !pressHandler && !releaseHandler ||\n                                !pressHandler && !listener.pressHandler ||\n                                pressHandler === listener.pressHandler;\n    var releaseHandlerMatches = !pressHandler && !releaseHandler ||\n                                !releaseHandler && !listener.releaseHandler ||\n                                releaseHandler === listener.releaseHandler;\n\n    if (comboMatches && pressHandlerMatches && releaseHandlerMatches) {\n      this._listeners.splice(i, 1);\n      i -= 1;\n    }\n  }\n};\nKeyboard.prototype.removeListener = Keyboard.prototype.unbind;\nKeyboard.prototype.off            = Keyboard.prototype.unbind;\n\nKeyboard.prototype.setContext = function(contextName) {\n  if(this._locale) { this.releaseAllKeys(); }\n\n  if (!this._contexts[contextName]) {\n    this._contexts[contextName] = [];\n  }\n  this._listeners      = this._contexts[contextName];\n  this._currentContext = contextName;\n};\n\nKeyboard.prototype.getContext = function() {\n  return this._currentContext;\n};\n\nKeyboard.prototype.withContext = function(contextName, callback) {\n  var previousContextName = this.getContext();\n  this.setContext(contextName);\n\n  callback();\n\n  this.setContext(previousContextName);\n};\n\nKeyboard.prototype.watch = function(targetWindow, targetElement, targetPlatform, targetUserAgent) {\n  var _this = this;\n\n  this.stop();\n\n  if (!targetWindow) {\n    if (!global.addEventListener && !global.attachEvent) {\n      throw new Error('Cannot find global functions addEventListener or attachEvent.');\n    }\n    targetWindow = global;\n  }\n\n  if (typeof targetWindow.nodeType === 'number') {\n    targetUserAgent = targetPlatform;\n    targetPlatform  = targetElement;\n    targetElement   = targetWindow;\n    targetWindow    = global;\n  }\n\n  if (!targetWindow.addEventListener && !targetWindow.attachEvent) {\n    throw new Error('Cannot find addEventListener or attachEvent methods on targetWindow.');\n  }\n\n  this._isModernBrowser = !!targetWindow.addEventListener;\n\n  var userAgent = targetWindow.navigator && targetWindow.navigator.userAgent || '';\n  var platform  = targetWindow.navigator && targetWindow.navigator.platform  || '';\n\n  targetElement   && targetElement   !== null || (targetElement   = targetWindow.document);\n  targetPlatform  && targetPlatform  !== null || (targetPlatform  = platform);\n  targetUserAgent && targetUserAgent !== null || (targetUserAgent = userAgent);\n\n  this._targetKeyDownBinding = function(event) {\n    _this.pressKey(event.keyCode, event);\n  };\n  this._targetKeyUpBinding = function(event) {\n    _this.releaseKey(event.keyCode, event);\n  };\n  this._targetResetBinding = function(event) {\n    _this.releaseAllKeys(event)\n  };\n\n  this._bindEvent(targetElement, 'keydown', this._targetKeyDownBinding);\n  this._bindEvent(targetElement, 'keyup',   this._targetKeyUpBinding);\n  this._bindEvent(targetWindow,  'focus',   this._targetResetBinding);\n  this._bindEvent(targetWindow,  'blur',    this._targetResetBinding);\n\n  this._targetElement   = targetElement;\n  this._targetWindow    = targetWindow;\n  this._targetPlatform  = targetPlatform;\n  this._targetUserAgent = targetUserAgent;\n};\n\nKeyboard.prototype.stop = function() {\n  var _this = this;\n\n  if (!this._targetElement || !this._targetWindow) { return; }\n\n  this._unbindEvent(this._targetElement, 'keydown', this._targetKeyDownBinding);\n  this._unbindEvent(this._targetElement, 'keyup',   this._targetKeyUpBinding);\n  this._unbindEvent(this._targetWindow,  'focus',   this._targetResetBinding);\n  this._unbindEvent(this._targetWindow,  'blur',    this._targetResetBinding);\n\n  this._targetWindow  = null;\n  this._targetElement = null;\n};\n\nKeyboard.prototype.pressKey = function(keyCode, event) {\n  if (this._paused) { return; }\n  if (!this._locale) { throw new Error('Locale not set'); }\n\n  this._locale.pressKey(keyCode);\n  this._applyBindings(event);\n};\n\nKeyboard.prototype.releaseKey = function(keyCode, event) {\n  if (this._paused) { return; }\n  if (!this._locale) { throw new Error('Locale not set'); }\n\n  this._locale.releaseKey(keyCode);\n  this._clearBindings(event);\n};\n\nKeyboard.prototype.releaseAllKeys = function(event) {\n  if (this._paused) { return; }\n  if (!this._locale) { throw new Error('Locale not set'); }\n\n  this._locale.pressedKeys.length = 0;\n  this._clearBindings(event);\n};\n\nKeyboard.prototype.pause = function() {\n  if (this._paused) { return; }\n  if (this._locale) { this.releaseAllKeys(); }\n  this._paused = true;\n};\n\nKeyboard.prototype.resume = function() {\n  this._paused = false;\n};\n\nKeyboard.prototype.reset = function() {\n  this.releaseAllKeys();\n  this._listeners.length = 0;\n};\n\nKeyboard.prototype._bindEvent = function(targetElement, eventName, handler) {\n  return this._isModernBrowser ?\n    targetElement.addEventListener(eventName, handler, false) :\n    targetElement.attachEvent('on' + eventName, handler);\n};\n\nKeyboard.prototype._unbindEvent = function(targetElement, eventName, handler) {\n  return this._isModernBrowser ?\n    targetElement.removeEventListener(eventName, handler, false) :\n    targetElement.detachEvent('on' + eventName, handler);\n};\n\nKeyboard.prototype._getGroupedListeners = function() {\n  var listenerGroups   = [];\n  var listenerGroupMap = [];\n\n  var listeners = this._listeners;\n  if (this._currentContext !== 'global') {\n    listeners = [].concat(listeners, this._contexts.global);\n  }\n\n  listeners.sort(function(a, b) {\n    return (b.keyCombo ? b.keyCombo.keyNames.length : 0) - (a.keyCombo ? a.keyCombo.keyNames.length : 0);\n  }).forEach(function(l) {\n    var mapIndex = -1;\n    for (var i = 0; i < listenerGroupMap.length; i += 1) {\n      if (listenerGroupMap[i] === null && l.keyCombo === null ||\n          listenerGroupMap[i] !== null && listenerGroupMap[i].isEqual(l.keyCombo)) {\n        mapIndex = i;\n      }\n    }\n    if (mapIndex === -1) {\n      mapIndex = listenerGroupMap.length;\n      listenerGroupMap.push(l.keyCombo);\n    }\n    if (!listenerGroups[mapIndex]) {\n      listenerGroups[mapIndex] = [];\n    }\n    listenerGroups[mapIndex].push(l);\n  });\n  return listenerGroups;\n};\n\nKeyboard.prototype._applyBindings = function(event) {\n  var preventRepeat = false;\n\n  event || (event = {});\n  event.preventRepeat = function() { preventRepeat = true; };\n  event.pressedKeys   = this._locale.pressedKeys.slice(0);\n\n  var pressedKeys    = this._locale.pressedKeys.slice(0);\n  var listenerGroups = this._getGroupedListeners();\n\n\n  for (var i = 0; i < listenerGroups.length; i += 1) {\n    var listeners = listenerGroups[i];\n    var keyCombo  = listeners[0].keyCombo;\n\n    if (keyCombo === null || keyCombo.check(pressedKeys)) {\n      for (var j = 0; j < listeners.length; j += 1) {\n        var listener = listeners[j];\n\n        if (keyCombo === null) {\n          listener = {\n            keyCombo               : new KeyCombo(pressedKeys.join('+')),\n            pressHandler           : listener.pressHandler,\n            releaseHandler         : listener.releaseHandler,\n            preventRepeat          : listener.preventRepeat,\n            preventRepeatByDefault : listener.preventRepeatByDefault\n          };\n        }\n\n        if (listener.pressHandler && !listener.preventRepeat) {\n          listener.pressHandler.call(this, event);\n          if (preventRepeat) {\n            listener.preventRepeat = preventRepeat;\n            preventRepeat          = false;\n          }\n        }\n\n        if (listener.releaseHandler && this._appliedListeners.indexOf(listener) === -1) {\n          this._appliedListeners.push(listener);\n        }\n      }\n\n      if (keyCombo) {\n        for (var j = 0; j < keyCombo.keyNames.length; j += 1) {\n          var index = pressedKeys.indexOf(keyCombo.keyNames[j]);\n          if (index !== -1) {\n            pressedKeys.splice(index, 1);\n            j -= 1;\n          }\n        }\n      }\n    }\n  }\n};\n\nKeyboard.prototype._clearBindings = function(event) {\n  event || (event = {});\n\n  for (var i = 0; i < this._appliedListeners.length; i += 1) {\n    var listener = this._appliedListeners[i];\n    var keyCombo = listener.keyCombo;\n    if (keyCombo === null || !keyCombo.check(this._locale.pressedKeys)) {\n      listener.preventRepeat = listener.preventRepeatByDefault;\n      listener.releaseHandler.call(this, event);\n      this._appliedListeners.splice(i, 1);\n      i -= 1;\n    }\n  }\n};\n\nmodule.exports = Keyboard;\n"]} },{"./key-combo":2,"./locale":4}],4:[function(require,module,exports){ var KeyCombo = require('./key-combo'); function Locale(name) { this.localeName = name; this.pressedKeys = []; this._appliedMacros = []; this._keyMap = {}; this._killKeyCodes = []; this._macros = []; } Locale.prototype.bindKeyCode = function(keyCode, keyNames) { if (typeof keyNames === 'string') { keyNames = [keyNames]; } this._keyMap[keyCode] = keyNames; }; Locale.prototype.bindMacro = function(keyComboStr, keyNames) { if (typeof keyNames === 'string') { keyNames = [ keyNames ]; } var handler = null; if (typeof keyNames === 'function') { handler = keyNames; keyNames = null; } var macro = { keyCombo : new KeyCombo(keyComboStr), keyNames : keyNames, handler : handler }; this._macros.push(macro); }; Locale.prototype.getKeyCodes = function(keyName) { var keyCodes = []; for (var keyCode in this._keyMap) { var index = this._keyMap[keyCode].indexOf(keyName); if (index > -1) { keyCodes.push(keyCode|0); } } return keyCodes; }; Locale.prototype.getKeyNames = function(keyCode) { return this._keyMap[keyCode] || []; }; Locale.prototype.setKillKey = function(keyCode) { if (typeof keyCode === 'string') { var keyCodes = this.getKeyCodes(keyCode); for (var i = 0; i < keyCodes.length; i += 1) { this.setKillKey(keyCodes[i]); } return; } this._killKeyCodes.push(keyCode); }; Locale.prototype.pressKey = function(keyCode) { if (typeof keyCode === 'string') { var keyCodes = this.getKeyCodes(keyCode); for (var i = 0; i < keyCodes.length; i += 1) { this.pressKey(keyCodes[i]); } return; } var keyNames = this.getKeyNames(keyCode); for (var i = 0; i < keyNames.length; i += 1) { if (this.pressedKeys.indexOf(keyNames[i]) === -1) { this.pressedKeys.push(keyNames[i]); } } this._applyMacros(); }; Locale.prototype.releaseKey = function(keyCode) { if (typeof keyCode === 'string') { var keyCodes = this.getKeyCodes(keyCode); for (var i = 0; i < keyCodes.length; i += 1) { this.releaseKey(keyCodes[i]); } } else { var keyNames = this.getKeyNames(keyCode); var killKeyCodeIndex = this._killKeyCodes.indexOf(keyCode); if (killKeyCodeIndex > -1) { this.pressedKeys.length = 0; } else { for (var i = 0; i < keyNames.length; i += 1) { var index = this.pressedKeys.indexOf(keyNames[i]); if (index > -1) { this.pressedKeys.splice(index, 1); } } } this._clearMacros(); } }; Locale.prototype._applyMacros = function() { var macros = this._macros.slice(0); for (var i = 0; i < macros.length; i += 1) { var macro = macros[i]; if (macro.keyCombo.check(this.pressedKeys)) { if (macro.handler) { macro.keyNames = macro.handler(this.pressedKeys); } for (var j = 0; j < macro.keyNames.length; j += 1) { if (this.pressedKeys.indexOf(macro.keyNames[j]) === -1) { this.pressedKeys.push(macro.keyNames[j]); } } this._appliedMacros.push(macro); } } }; Locale.prototype._clearMacros = function() { for (var i = 0; i < this._appliedMacros.length; i += 1) { var macro = this._appliedMacros[i]; if (!macro.keyCombo.check(this.pressedKeys)) { for (var j = 0; j < macro.keyNames.length; j += 1) { var index = this.pressedKeys.indexOf(macro.keyNames[j]); if (index > -1) { this.pressedKeys.splice(index, 1); } } if (macro.handler) { macro.keyNames = null; } this._appliedMacros.splice(i, 1); i -= 1; } } }; module.exports = Locale; },{"./key-combo":2}],5:[function(require,module,exports){ module.exports = function(locale, platform, userAgent) { // general locale.bindKeyCode(3, ['cancel']); locale.bindKeyCode(8, ['backspace']); locale.bindKeyCode(9, ['tab']); locale.bindKeyCode(12, ['clear']); locale.bindKeyCode(13, ['enter']); locale.bindKeyCode(16, ['shift']); locale.bindKeyCode(17, ['ctrl']); locale.bindKeyCode(18, ['alt', 'menu']); locale.bindKeyCode(19, ['pause', 'break']); locale.bindKeyCode(20, ['capslock']); locale.bindKeyCode(27, ['escape', 'esc']); locale.bindKeyCode(32, ['space', 'spacebar']); locale.bindKeyCode(33, ['pageup']); locale.bindKeyCode(34, ['pagedown']); locale.bindKeyCode(35, ['end']); locale.bindKeyCode(36, ['home']); locale.bindKeyCode(37, ['left']); locale.bindKeyCode(38, ['up']); locale.bindKeyCode(39, ['right']); locale.bindKeyCode(40, ['down']); locale.bindKeyCode(41, ['select']); locale.bindKeyCode(42, ['printscreen']); locale.bindKeyCode(43, ['execute']); locale.bindKeyCode(44, ['snapshot']); locale.bindKeyCode(45, ['insert', 'ins']); locale.bindKeyCode(46, ['delete', 'del']); locale.bindKeyCode(47, ['help']); locale.bindKeyCode(145, ['scrolllock', 'scroll']); locale.bindKeyCode(187, ['equal', 'equalsign', '=']); locale.bindKeyCode(188, ['comma', ',']); locale.bindKeyCode(190, ['period', '.']); locale.bindKeyCode(191, ['slash', 'forwardslash', '/']); locale.bindKeyCode(192, ['graveaccent', '`']); locale.bindKeyCode(219, ['openbracket', '[']); locale.bindKeyCode(220, ['backslash', '\\']); locale.bindKeyCode(221, ['closebracket', ']']); locale.bindKeyCode(222, ['apostrophe', '\'']); // 0-9 locale.bindKeyCode(48, ['zero', '0']); locale.bindKeyCode(49, ['one', '1']); locale.bindKeyCode(50, ['two', '2']); locale.bindKeyCode(51, ['three', '3']); locale.bindKeyCode(52, ['four', '4']); locale.bindKeyCode(53, ['five', '5']); locale.bindKeyCode(54, ['six', '6']); locale.bindKeyCode(55, ['seven', '7']); locale.bindKeyCode(56, ['eight', '8']); locale.bindKeyCode(57, ['nine', '9']); // numpad locale.bindKeyCode(96, ['numzero', 'num0']); locale.bindKeyCode(97, ['numone', 'num1']); locale.bindKeyCode(98, ['numtwo', 'num2']); locale.bindKeyCode(99, ['numthree', 'num3']); locale.bindKeyCode(100, ['numfour', 'num4']); locale.bindKeyCode(101, ['numfive', 'num5']); locale.bindKeyCode(102, ['numsix', 'num6']); locale.bindKeyCode(103, ['numseven', 'num7']); locale.bindKeyCode(104, ['numeight', 'num8']); locale.bindKeyCode(105, ['numnine', 'num9']); locale.bindKeyCode(106, ['nummultiply', 'num*']); locale.bindKeyCode(107, ['numadd', 'num+']); locale.bindKeyCode(108, ['numenter']); locale.bindKeyCode(109, ['numsubtract', 'num-']); locale.bindKeyCode(110, ['numdecimal', 'num.']); locale.bindKeyCode(111, ['numdivide', 'num/']); locale.bindKeyCode(144, ['numlock', 'num']); // function keys locale.bindKeyCode(112, ['f1']); locale.bindKeyCode(113, ['f2']); locale.bindKeyCode(114, ['f3']); locale.bindKeyCode(115, ['f4']); locale.bindKeyCode(116, ['f5']); locale.bindKeyCode(117, ['f6']); locale.bindKeyCode(118, ['f7']); locale.bindKeyCode(119, ['f8']); locale.bindKeyCode(120, ['f9']); locale.bindKeyCode(121, ['f10']); locale.bindKeyCode(122, ['f11']); locale.bindKeyCode(123, ['f12']); // secondary key symbols locale.bindMacro('shift + `', ['tilde', '~']); locale.bindMacro('shift + 1', ['exclamation', 'exclamationpoint', '!']); locale.bindMacro('shift + 2', ['at', '@']); locale.bindMacro('shift + 3', ['number', '#']); locale.bindMacro('shift + 4', ['dollar', 'dollars', 'dollarsign', '$']); locale.bindMacro('shift + 5', ['percent', '%']); locale.bindMacro('shift + 6', ['caret', '^']); locale.bindMacro('shift + 7', ['ampersand', 'and', '&']); locale.bindMacro('shift + 8', ['asterisk', '*']); locale.bindMacro('shift + 9', ['openparen', '(']); locale.bindMacro('shift + 0', ['closeparen', ')']); locale.bindMacro('shift + -', ['underscore', '_']); locale.bindMacro('shift + =', ['plus', '+']); locale.bindMacro('shift + [', ['opencurlybrace', 'opencurlybracket', '{']); locale.bindMacro('shift + ]', ['closecurlybrace', 'closecurlybracket', '}']); locale.bindMacro('shift + \\', ['verticalbar', '|']); locale.bindMacro('shift + ;', ['colon', ':']); locale.bindMacro('shift + \'', ['quotationmark', '\'']); locale.bindMacro('shift + !,', ['openanglebracket', '<']); locale.bindMacro('shift + .', ['closeanglebracket', '>']); locale.bindMacro('shift + /', ['questionmark', '?']); //a-z and A-Z for (var keyCode = 65; keyCode <= 90; keyCode += 1) { var keyName = String.fromCharCode(keyCode + 32); var capitalKeyName = String.fromCharCode(keyCode); locale.bindKeyCode(keyCode, keyName); locale.bindMacro('shift + ' + keyName, capitalKeyName); locale.bindMacro('capslock + ' + keyName, capitalKeyName); } // browser caveats var semicolonKeyCode = userAgent.match('Firefox') ? 59 : 186; var dashKeyCode = userAgent.match('Firefox') ? 173 : 189; var leftCommandKeyCode; var rightCommandKeyCode; if (platform.match('Mac') && (userAgent.match('Safari') || userAgent.match('Chrome'))) { leftCommandKeyCode = 91; rightCommandKeyCode = 93; } else if(platform.match('Mac') && userAgent.match('Opera')) { leftCommandKeyCode = 17; rightCommandKeyCode = 17; } else if(platform.match('Mac') && userAgent.match('Firefox')) { leftCommandKeyCode = 224; rightCommandKeyCode = 224; } locale.bindKeyCode(semicolonKeyCode, ['semicolon', ';']); locale.bindKeyCode(dashKeyCode, ['dash', '-']); locale.bindKeyCode(leftCommandKeyCode, ['command', 'windows', 'win', 'super', 'leftcommand', 'leftwindows', 'leftwin', 'leftsuper']); locale.bindKeyCode(rightCommandKeyCode, ['command', 'windows', 'win', 'super', 'rightcommand', 'rightwindows', 'rightwin', 'rightsuper']); // kill keys locale.setKillKey('command'); }; },{}]},{},[1])(1) }); //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["node_modules/browser-pack/_prelude.js","index.js","lib/key-combo.js","lib/keyboard.js","lib/locale.js","locales/us.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACbA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChXA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;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})","\nvar Keyboard = require('./lib/keyboard');\nvar Locale   = require('./lib/locale');\nvar KeyCombo = require('./lib/key-combo');\n\nvar keyboard = new Keyboard();\n\nkeyboard.setLocale('us', require('./locales/us'));\n\nexports          = module.exports = keyboard;\nexports.Keyboard = Keyboard;\nexports.Locale   = Locale;\nexports.KeyCombo = KeyCombo;\n","\nfunction KeyCombo(keyComboStr) {\n  this.sourceStr = keyComboStr;\n  this.subCombos = KeyCombo.parseComboStr(keyComboStr);\n  this.keyNames  = this.subCombos.reduce(function(memo, nextSubCombo) {\n    return memo.concat(nextSubCombo);\n  });\n}\n\n// TODO: Add support for key combo sequences\nKeyCombo.sequenceDeliminator = '>>';\nKeyCombo.comboDeliminator    = '>';\nKeyCombo.keyDeliminator      = '+';\n\nKeyCombo.parseComboStr = function(keyComboStr) {\n  var subComboStrs = KeyCombo._splitStr(keyComboStr, KeyCombo.comboDeliminator);\n  var combo        = [];\n\n  for (var i = 0 ; i < subComboStrs.length; i += 1) {\n    combo.push(KeyCombo._splitStr(subComboStrs[i], KeyCombo.keyDeliminator));\n  }\n  return combo;\n};\n\nKeyCombo.prototype.check = function(pressedKeyNames) {\n  var startingKeyNameIndex = 0;\n  for (var i = 0; i < this.subCombos.length; i += 1) {\n    startingKeyNameIndex = this._checkSubCombo(\n      this.subCombos[i],\n      startingKeyNameIndex,\n      pressedKeyNames\n    );\n    if (startingKeyNameIndex === -1) { return false; }\n  }\n  return true;\n};\n\nKeyCombo.prototype.isEqual = function(otherKeyCombo) {\n  if (\n    !otherKeyCombo ||\n    typeof otherKeyCombo !== 'string' &&\n    typeof otherKeyCombo !== 'object'\n  ) { return false; }\n\n  if (typeof otherKeyCombo === 'string') {\n    otherKeyCombo = new KeyCombo(otherKeyCombo);\n  }\n\n  if (this.subCombos.length !== otherKeyCombo.subCombos.length) {\n    return false;\n  }\n  for (var i = 0; i < this.subCombos.length; i += 1) {\n    if (this.subCombos[i].length !== otherKeyCombo.subCombos[i].length) {\n      return false;\n    }\n  }\n\n  for (var i = 0; i < this.subCombos.length; i += 1) {\n    var subCombo      = this.subCombos[i];\n    var otherSubCombo = otherKeyCombo.subCombos[i].slice(0);\n\n    for (var j = 0; j < subCombo.length; j += 1) {\n      var keyName = subCombo[j];\n      var index   = otherSubCombo.indexOf(keyName);\n\n      if (index > -1) {\n        otherSubCombo.splice(index, 1);\n      }\n    }\n    if (otherSubCombo.length !== 0) {\n      return false;\n    }\n  }\n\n  return true;\n};\n\nKeyCombo._splitStr = function(str, deliminator) {\n  var s  = str;\n  var d  = deliminator;\n  var c  = '';\n  var ca = [];\n\n  for (var ci = 0; ci < s.length; ci += 1) {\n    if (ci > 0 && s[ci] === d && s[ci - 1] !== '\\\\') {\n      ca.push(c.trim());\n      c = '';\n      ci += 1;\n    }\n    c += s[ci];\n  }\n  if (c) { ca.push(c.trim()); }\n\n  return ca;\n};\n\nKeyCombo.prototype._checkSubCombo = function(subCombo, startingKeyNameIndex, pressedKeyNames) {\n  subCombo = subCombo.slice(0);\n  pressedKeyNames = pressedKeyNames.slice(startingKeyNameIndex);\n\n  var endIndex = startingKeyNameIndex;\n  for (var i = 0; i < subCombo.length; i += 1) {\n\n    var keyName = subCombo[i];\n    if (keyName[0] === '\\\\') {\n      var escapedKeyName = keyName.slice(1);\n      if (\n        escapedKeyName === KeyCombo.comboDeliminator ||\n        escapedKeyName === KeyCombo.keyDeliminator\n      ) {\n        keyName = escapedKeyName;\n      }\n    }\n\n    var index = pressedKeyNames.indexOf(keyName);\n    if (index > -1) {\n      subCombo.splice(i, 1);\n      i -= 1;\n      if (index > endIndex) {\n        endIndex = index;\n      }\n      if (subCombo.length === 0) {\n        return endIndex;\n      }\n    }\n  }\n  return -1;\n};\n\n\nmodule.exports = KeyCombo;\n","(function (global){\n\nvar Locale = require('./locale');\nvar KeyCombo = require('./key-combo');\n\n\nfunction Keyboard(targetWindow, targetElement, platform, userAgent) {\n  this._locale               = null;\n  this._currentContext       = null;\n  this._contexts             = {};\n  this._listeners            = [];\n  this._appliedListeners     = [];\n  this._locales              = {};\n  this._targetElement        = null;\n  this._targetWindow         = null;\n  this._targetPlatform       = '';\n  this._targetUserAgent      = '';\n  this._isModernBrowser      = false;\n  this._targetKeyDownBinding = null;\n  this._targetKeyUpBinding   = null;\n  this._targetResetBinding   = null;\n  this._paused               = false;\n\n  this.setContext('global');\n  this.watch(targetWindow, targetElement, platform, userAgent);\n}\n\nKeyboard.prototype.setLocale = function(localeName, localeBuilder) {\n  var locale = null;\n  if (typeof localeName === 'string') {\n\n    if (localeBuilder) {\n      locale = new Locale(localeName);\n      localeBuilder(locale, this._targetPlatform, this._targetUserAgent);\n    } else {\n      locale = this._locales[localeName] || null;\n    }\n  } else {\n    locale     = localeName;\n    localeName = locale._localeName;\n  }\n\n  this._locale              = locale;\n  this._locales[localeName] = locale;\n  if (locale) {\n    this._locale.pressedKeys = locale.pressedKeys;\n  }\n};\n\nKeyboard.prototype.getLocale = function(localName) {\n  localName || (localName = this._locale.localeName);\n  return this._locales[localName] || null;\n};\n\nKeyboard.prototype.bind = function(keyComboStr, pressHandler, releaseHandler, preventRepeatByDefault) {\n  if (keyComboStr === null || typeof keyComboStr === 'function') {\n    preventRepeatByDefault = releaseHandler;\n    releaseHandler         = pressHandler;\n    pressHandler           = keyComboStr;\n    keyComboStr            = null;\n  }\n\n  if (\n    keyComboStr &&\n    typeof keyComboStr === 'object' &&\n    typeof keyComboStr.length === 'number'\n  ) {\n    for (var i = 0; i < keyComboStr.length; i += 1) {\n      this.bind(keyComboStr[i], pressHandler, releaseHandler);\n    }\n    return;\n  }\n\n  this._listeners.push({\n    keyCombo               : keyComboStr ? new KeyCombo(keyComboStr) : null,\n    pressHandler           : pressHandler           || null,\n    releaseHandler         : releaseHandler         || null,\n    preventRepeat          : preventRepeatByDefault || false,\n    preventRepeatByDefault : preventRepeatByDefault || false\n  });\n};\nKeyboard.prototype.addListener = Keyboard.prototype.bind;\nKeyboard.prototype.on          = Keyboard.prototype.bind;\n\nKeyboard.prototype.unbind = function(keyComboStr, pressHandler, releaseHandler) {\n  if (keyComboStr === null || typeof keyComboStr === 'function') {\n    releaseHandler = pressHandler;\n    pressHandler   = keyComboStr;\n    keyComboStr = null;\n  }\n\n  if (\n    keyComboStr &&\n    typeof keyComboStr === 'object' &&\n    typeof keyComboStr.length === 'number'\n  ) {\n    for (var i = 0; i < keyComboStr.length; i += 1) {\n      this.unbind(keyComboStr[i], pressHandler, releaseHandler);\n    }\n    return;\n  }\n\n  for (var i = 0; i < this._listeners.length; i += 1) {\n    var listener = this._listeners[i];\n\n    var comboMatches          = !keyComboStr && !listener.keyCombo ||\n                                listener.keyCombo && listener.keyCombo.isEqual(keyComboStr);\n    var pressHandlerMatches   = !pressHandler && !releaseHandler ||\n                                !pressHandler && !listener.pressHandler ||\n                                pressHandler === listener.pressHandler;\n    var releaseHandlerMatches = !pressHandler && !releaseHandler ||\n                                !releaseHandler && !listener.releaseHandler ||\n                                releaseHandler === listener.releaseHandler;\n\n    if (comboMatches && pressHandlerMatches && releaseHandlerMatches) {\n      this._listeners.splice(i, 1);\n      i -= 1;\n    }\n  }\n};\nKeyboard.prototype.removeListener = Keyboard.prototype.unbind;\nKeyboard.prototype.off            = Keyboard.prototype.unbind;\n\nKeyboard.prototype.setContext = function(contextName) {\n  if(this._locale) { this.releaseAllKeys(); }\n\n  if (!this._contexts[contextName]) {\n    this._contexts[contextName] = [];\n  }\n  this._listeners      = this._contexts[contextName];\n  this._currentContext = contextName;\n};\n\nKeyboard.prototype.getContext = function() {\n  return this._currentContext;\n};\n\nKeyboard.prototype.withContext = function(contextName, callback) {\n  var previousContextName = this.getContext();\n  this.setContext(contextName);\n\n  callback();\n\n  this.setContext(previousContextName);\n};\n\nKeyboard.prototype.watch = function(targetWindow, targetElement, targetPlatform, targetUserAgent) {\n  var _this = this;\n\n  this.stop();\n\n  if (!targetWindow) {\n    if (!global.addEventListener && !global.attachEvent) {\n      throw new Error('Cannot find global functions addEventListener or attachEvent.');\n    }\n    targetWindow = global;\n  }\n\n  if (typeof targetWindow.nodeType === 'number') {\n    targetUserAgent = targetPlatform;\n    targetPlatform  = targetElement;\n    targetElement   = targetWindow;\n    targetWindow    = global;\n  }\n\n  if (!targetWindow.addEventListener && !targetWindow.attachEvent) {\n    throw new Error('Cannot find addEventListener or attachEvent methods on targetWindow.');\n  }\n\n  this._isModernBrowser = !!targetWindow.addEventListener;\n\n  var userAgent = targetWindow.navigator && targetWindow.navigator.userAgent || '';\n  var platform  = targetWindow.navigator && targetWindow.navigator.platform  || '';\n\n  targetElement   && targetElement   !== null || (targetElement   = targetWindow.document);\n  targetPlatform  && targetPlatform  !== null || (targetPlatform  = platform);\n  targetUserAgent && targetUserAgent !== null || (targetUserAgent = userAgent);\n\n  this._targetKeyDownBinding = function(event) {\n    _this.pressKey(event.keyCode, event);\n  };\n  this._targetKeyUpBinding = function(event) {\n    _this.releaseKey(event.keyCode, event);\n  };\n  this._targetResetBinding = function(event) {\n    _this.releaseAllKeys(event)\n  };\n\n  this._bindEvent(targetElement, 'keydown', this._targetKeyDownBinding);\n  this._bindEvent(targetElement, 'keyup',   this._targetKeyUpBinding);\n  this._bindEvent(targetWindow,  'focus',   this._targetResetBinding);\n  this._bindEvent(targetWindow,  'blur',    this._targetResetBinding);\n\n  this._targetElement   = targetElement;\n  this._targetWindow    = targetWindow;\n  this._targetPlatform  = targetPlatform;\n  this._targetUserAgent = targetUserAgent;\n};\n\nKeyboard.prototype.stop = function() {\n  var _this = this;\n\n  if (!this._targetElement || !this._targetWindow) { return; }\n\n  this._unbindEvent(this._targetElement, 'keydown', this._targetKeyDownBinding);\n  this._unbindEvent(this._targetElement, 'keyup',   this._targetKeyUpBinding);\n  this._unbindEvent(this._targetWindow,  'focus',   this._targetResetBinding);\n  this._unbindEvent(this._targetWindow,  'blur',    this._targetResetBinding);\n\n  this._targetWindow  = null;\n  this._targetElement = null;\n};\n\nKeyboard.prototype.pressKey = function(keyCode, event) {\n  if (this._paused) { return; }\n  if (!this._locale) { throw new Error('Locale not set'); }\n\n  this._locale.pressKey(keyCode);\n  this._applyBindings(event);\n};\n\nKeyboard.prototype.releaseKey = function(keyCode, event) {\n  if (this._paused) { return; }\n  if (!this._locale) { throw new Error('Locale not set'); }\n\n  this._locale.releaseKey(keyCode);\n  this._clearBindings(event);\n};\n\nKeyboard.prototype.releaseAllKeys = function(event) {\n  if (this._paused) { return; }\n  if (!this._locale) { throw new Error('Locale not set'); }\n\n  this._locale.pressedKeys.length = 0;\n  this._clearBindings(event);\n};\n\nKeyboard.prototype.pause = function() {\n  if (this._paused) { return; }\n  if (this._locale) { this.releaseAllKeys(); }\n  this._paused = true;\n};\n\nKeyboard.prototype.resume = function() {\n  this._paused = false;\n};\n\nKeyboard.prototype.reset = function() {\n  this.releaseAllKeys();\n  this._listeners.length = 0;\n};\n\nKeyboard.prototype._bindEvent = function(targetElement, eventName, handler) {\n  return this._isModernBrowser ?\n    targetElement.addEventListener(eventName, handler, false) :\n    targetElement.attachEvent('on' + eventName, handler);\n};\n\nKeyboard.prototype._unbindEvent = function(targetElement, eventName, handler) {\n  return this._isModernBrowser ?\n    targetElement.removeEventListener(eventName, handler, false) :\n    targetElement.detachEvent('on' + eventName, handler);\n};\n\nKeyboard.prototype._getGroupedListeners = function() {\n  var listenerGroups   = [];\n  var listenerGroupMap = [];\n\n  var listeners = this._listeners;\n  if (this._currentContext !== 'global') {\n    listeners = [].concat(listeners, this._contexts.global);\n  }\n\n  listeners.sort(function(a, b) {\n    return (b.keyCombo ? b.keyCombo.keyNames.length : 0) - (a.keyCombo ? a.keyCombo.keyNames.length : 0);\n  }).forEach(function(l) {\n    var mapIndex = -1;\n    for (var i = 0; i < listenerGroupMap.length; i += 1) {\n      if (listenerGroupMap[i] === null && l.keyCombo === null ||\n          listenerGroupMap[i] !== null && listenerGroupMap[i].isEqual(l.keyCombo)) {\n        mapIndex = i;\n      }\n    }\n    if (mapIndex === -1) {\n      mapIndex = listenerGroupMap.length;\n      listenerGroupMap.push(l.keyCombo);\n    }\n    if (!listenerGroups[mapIndex]) {\n      listenerGroups[mapIndex] = [];\n    }\n    listenerGroups[mapIndex].push(l);\n  });\n  return listenerGroups;\n};\n\nKeyboard.prototype._applyBindings = function(event) {\n  var preventRepeat = false;\n\n  event || (event = {});\n  event.preventRepeat = function() { preventRepeat = true; };\n  event.pressedKeys   = this._locale.pressedKeys.slice(0);\n\n  var pressedKeys    = this._locale.pressedKeys.slice(0);\n  var listenerGroups = this._getGroupedListeners();\n\n\n  for (var i = 0; i < listenerGroups.length; i += 1) {\n    var listeners = listenerGroups[i];\n    var keyCombo  = listeners[0].keyCombo;\n\n    if (keyCombo === null || keyCombo.check(pressedKeys)) {\n      for (var j = 0; j < listeners.length; j += 1) {\n        var listener = listeners[j];\n\n        if (keyCombo === null) {\n          listener = {\n            keyCombo               : new KeyCombo(pressedKeys.join('+')),\n            pressHandler           : listener.pressHandler,\n            releaseHandler         : listener.releaseHandler,\n            preventRepeat          : listener.preventRepeat,\n            preventRepeatByDefault : listener.preventRepeatByDefault\n          };\n        }\n\n        if (listener.pressHandler && !listener.preventRepeat) {\n          listener.pressHandler.call(this, event);\n          if (preventRepeat) {\n            listener.preventRepeat = preventRepeat;\n            preventRepeat          = false;\n          }\n        }\n\n        if (listener.releaseHandler && this._appliedListeners.indexOf(listener) === -1) {\n          this._appliedListeners.push(listener);\n        }\n      }\n\n      if (keyCombo) {\n        for (var j = 0; j < keyCombo.keyNames.length; j += 1) {\n          var index = pressedKeys.indexOf(keyCombo.keyNames[j]);\n          if (index !== -1) {\n            pressedKeys.splice(index, 1);\n            j -= 1;\n          }\n        }\n      }\n    }\n  }\n};\n\nKeyboard.prototype._clearBindings = function(event) {\n  event || (event = {});\n\n  for (var i = 0; i < this._appliedListeners.length; i += 1) {\n    var listener = this._appliedListeners[i];\n    var keyCombo = listener.keyCombo;\n    if (keyCombo === null || !keyCombo.check(this._locale.pressedKeys)) {\n      listener.preventRepeat = listener.preventRepeatByDefault;\n      listener.releaseHandler.call(this, event);\n      this._appliedListeners.splice(i, 1);\n      i -= 1;\n    }\n  }\n};\n\nmodule.exports = Keyboard;\n\n}).call(this,typeof global !== \"undefined\" ? global : typeof self !== \"undefined\" ? self : typeof window !== \"undefined\" ? window : {})\n//# sourceMappingURL=data:application/json;charset:utf-8;base64,{"version":3,"sources":["lib/keyboard.js"],"names":[],"mappings":";AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["\nvar Locale = require('./locale');\nvar KeyCombo = require('./key-combo');\n\n\nfunction Keyboard(targetWindow, targetElement, platform, userAgent) {\n  this._locale               = null;\n  this._currentContext       = null;\n  this._contexts             = {};\n  this._listeners            = [];\n  this._appliedListeners     = [];\n  this._locales              = {};\n  this._targetElement        = null;\n  this._targetWindow         = null;\n  this._targetPlatform       = '';\n  this._targetUserAgent      = '';\n  this._isModernBrowser      = false;\n  this._targetKeyDownBinding = null;\n  this._targetKeyUpBinding   = null;\n  this._targetResetBinding   = null;\n  this._paused               = false;\n\n  this.setContext('global');\n  this.watch(targetWindow, targetElement, platform, userAgent);\n}\n\nKeyboard.prototype.setLocale = function(localeName, localeBuilder) {\n  var locale = null;\n  if (typeof localeName === 'string') {\n\n    if (localeBuilder) {\n      locale = new Locale(localeName);\n      localeBuilder(locale, this._targetPlatform, this._targetUserAgent);\n    } else {\n      locale = this._locales[localeName] || null;\n    }\n  } else {\n    locale     = localeName;\n    localeName = locale._localeName;\n  }\n\n  this._locale              = locale;\n  this._locales[localeName] = locale;\n  if (locale) {\n    this._locale.pressedKeys = locale.pressedKeys;\n  }\n};\n\nKeyboard.prototype.getLocale = function(localName) {\n  localName || (localName = this._locale.localeName);\n  return this._locales[localName] || null;\n};\n\nKeyboard.prototype.bind = function(keyComboStr, pressHandler, releaseHandler, preventRepeatByDefault) {\n  if (keyComboStr === null || typeof keyComboStr === 'function') {\n    preventRepeatByDefault = releaseHandler;\n    releaseHandler         = pressHandler;\n    pressHandler           = keyComboStr;\n    keyComboStr            = null;\n  }\n\n  if (\n    keyComboStr &&\n    typeof keyComboStr === 'object' &&\n    typeof keyComboStr.length === 'number'\n  ) {\n    for (var i = 0; i < keyComboStr.length; i += 1) {\n      this.bind(keyComboStr[i], pressHandler, releaseHandler);\n    }\n    return;\n  }\n\n  this._listeners.push({\n    keyCombo               : keyComboStr ? new KeyCombo(keyComboStr) : null,\n    pressHandler           : pressHandler           || null,\n    releaseHandler         : releaseHandler         || null,\n    preventRepeat          : preventRepeatByDefault || false,\n    preventRepeatByDefault : preventRepeatByDefault || false\n  });\n};\nKeyboard.prototype.addListener = Keyboard.prototype.bind;\nKeyboard.prototype.on          = Keyboard.prototype.bind;\n\nKeyboard.prototype.unbind = function(keyComboStr, pressHandler, releaseHandler) {\n  if (keyComboStr === null || typeof keyComboStr === 'function') {\n    releaseHandler = pressHandler;\n    pressHandler   = keyComboStr;\n    keyComboStr = null;\n  }\n\n  if (\n    keyComboStr &&\n    typeof keyComboStr === 'object' &&\n    typeof keyComboStr.length === 'number'\n  ) {\n    for (var i = 0; i < keyComboStr.length; i += 1) {\n      this.unbind(keyComboStr[i], pressHandler, releaseHandler);\n    }\n    return;\n  }\n\n  for (var i = 0; i < this._listeners.length; i += 1) {\n    var listener = this._listeners[i];\n\n    var comboMatches          = !keyComboStr && !listener.keyCombo ||\n                                listener.keyCombo && listener.keyCombo.isEqual(keyComboStr);\n    var pressHandlerMatches   = !pressHandler && !releaseHandler ||\n                                !pressHandler && !listener.pressHandler ||\n                                pressHandler === listener.pressHandler;\n    var releaseHandlerMatches = !pressHandler && !releaseHandler ||\n                                !releaseHandler && !listener.releaseHandler ||\n                                releaseHandler === listener.releaseHandler;\n\n    if (comboMatches && pressHandlerMatches && releaseHandlerMatches) {\n      this._listeners.splice(i, 1);\n      i -= 1;\n    }\n  }\n};\nKeyboard.prototype.removeListener = Keyboard.prototype.unbind;\nKeyboard.prototype.off            = Keyboard.prototype.unbind;\n\nKeyboard.prototype.setContext = function(contextName) {\n  if(this._locale) { this.releaseAllKeys(); }\n\n  if (!this._contexts[contextName]) {\n    this._contexts[contextName] = [];\n  }\n  this._listeners      = this._contexts[contextName];\n  this._currentContext = contextName;\n};\n\nKeyboard.prototype.getContext = function() {\n  return this._currentContext;\n};\n\nKeyboard.prototype.withContext = function(contextName, callback) {\n  var previousContextName = this.getContext();\n  this.setContext(contextName);\n\n  callback();\n\n  this.setContext(previousContextName);\n};\n\nKeyboard.prototype.watch = function(targetWindow, targetElement, targetPlatform, targetUserAgent) {\n  var _this = this;\n\n  this.stop();\n\n  if (!targetWindow) {\n    if (!global.addEventListener && !global.attachEvent) {\n      throw new Error('Cannot find global functions addEventListener or attachEvent.');\n    }\n    targetWindow = global;\n  }\n\n  if (typeof targetWindow.nodeType === 'number') {\n    targetUserAgent = targetPlatform;\n    targetPlatform  = targetElement;\n    targetElement   = targetWindow;\n    targetWindow    = global;\n  }\n\n  if (!targetWindow.addEventListener && !targetWindow.attachEvent) {\n    throw new Error('Cannot find addEventListener or attachEvent methods on targetWindow.');\n  }\n\n  this._isModernBrowser = !!targetWindow.addEventListener;\n\n  var userAgent = targetWindow.navigator && targetWindow.navigator.userAgent || '';\n  var platform  = targetWindow.navigator && targetWindow.navigator.platform  || '';\n\n  targetElement   && targetElement   !== null || (targetElement   = targetWindow.document);\n  targetPlatform  && targetPlatform  !== null || (targetPlatform  = platform);\n  targetUserAgent && targetUserAgent !== null || (targetUserAgent = userAgent);\n\n  this._targetKeyDownBinding = function(event) {\n    _this.pressKey(event.keyCode, event);\n  };\n  this._targetKeyUpBinding = function(event) {\n    _this.releaseKey(event.keyCode, event);\n  };\n  this._targetResetBinding = function(event) {\n    _this.releaseAllKeys(event)\n  };\n\n  this._bindEvent(targetElement, 'keydown', this._targetKeyDownBinding);\n  this._bindEvent(targetElement, 'keyup',   this._targetKeyUpBinding);\n  this._bindEvent(targetWindow,  'focus',   this._targetResetBinding);\n  this._bindEvent(targetWindow,  'blur',    this._targetResetBinding);\n\n  this._targetElement   = targetElement;\n  this._targetWindow    = targetWindow;\n  this._targetPlatform  = targetPlatform;\n  this._targetUserAgent = targetUserAgent;\n};\n\nKeyboard.prototype.stop = function() {\n  var _this = this;\n\n  if (!this._targetElement || !this._targetWindow) { return; }\n\n  this._unbindEvent(this._targetElement, 'keydown', this._targetKeyDownBinding);\n  this._unbindEvent(this._targetElement, 'keyup',   this._targetKeyUpBinding);\n  this._unbindEvent(this._targetWindow,  'focus',   this._targetResetBinding);\n  this._unbindEvent(this._targetWindow,  'blur',    this._targetResetBinding);\n\n  this._targetWindow  = null;\n  this._targetElement = null;\n};\n\nKeyboard.prototype.pressKey = function(keyCode, event) {\n  if (this._paused) { return; }\n  if (!this._locale) { throw new Error('Locale not set'); }\n\n  this._locale.pressKey(keyCode);\n  this._applyBindings(event);\n};\n\nKeyboard.prototype.releaseKey = function(keyCode, event) {\n  if (this._paused) { return; }\n  if (!this._locale) { throw new Error('Locale not set'); }\n\n  this._locale.releaseKey(keyCode);\n  this._clearBindings(event);\n};\n\nKeyboard.prototype.releaseAllKeys = function(event) {\n  if (this._paused) { return; }\n  if (!this._locale) { throw new Error('Locale not set'); }\n\n  this._locale.pressedKeys.length = 0;\n  this._clearBindings(event);\n};\n\nKeyboard.prototype.pause = function() {\n  if (this._paused) { return; }\n  if (this._locale) { this.releaseAllKeys(); }\n  this._paused = true;\n};\n\nKeyboard.prototype.resume = function() {\n  this._paused = false;\n};\n\nKeyboard.prototype.reset = function() {\n  this.releaseAllKeys();\n  this._listeners.length = 0;\n};\n\nKeyboard.prototype._bindEvent = function(targetElement, eventName, handler) {\n  return this._isModernBrowser ?\n    targetElement.addEventListener(eventName, handler, false) :\n    targetElement.attachEvent('on' + eventName, handler);\n};\n\nKeyboard.prototype._unbindEvent = function(targetElement, eventName, handler) {\n  return this._isModernBrowser ?\n    targetElement.removeEventListener(eventName, handler, false) :\n    targetElement.detachEvent('on' + eventName, handler);\n};\n\nKeyboard.prototype._getGroupedListeners = function() {\n  var listenerGroups   = [];\n  var listenerGroupMap = [];\n\n  var listeners = this._listeners;\n  if (this._currentContext !== 'global') {\n    listeners = [].concat(listeners, this._contexts.global);\n  }\n\n  listeners.sort(function(a, b) {\n    return (b.keyCombo ? b.keyCombo.keyNames.length : 0) - (a.keyCombo ? a.keyCombo.keyNames.length : 0);\n  }).forEach(function(l) {\n    var mapIndex = -1;\n    for (var i = 0; i < listenerGroupMap.length; i += 1) {\n      if (listenerGroupMap[i] === null && l.keyCombo === null ||\n          listenerGroupMap[i] !== null && listenerGroupMap[i].isEqual(l.keyCombo)) {\n        mapIndex = i;\n      }\n    }\n    if (mapIndex === -1) {\n      mapIndex = listenerGroupMap.length;\n      listenerGroupMap.push(l.keyCombo);\n    }\n    if (!listenerGroups[mapIndex]) {\n      listenerGroups[mapIndex] = [];\n    }\n    listenerGroups[mapIndex].push(l);\n  });\n  return listenerGroups;\n};\n\nKeyboard.prototype._applyBindings = function(event) {\n  var preventRepeat = false;\n\n  event || (event = {});\n  event.preventRepeat = function() { preventRepeat = true; };\n  event.pressedKeys   = this._locale.pressedKeys.slice(0);\n\n  var pressedKeys    = this._locale.pressedKeys.slice(0);\n  var listenerGroups = this._getGroupedListeners();\n\n\n  for (var i = 0; i < listenerGroups.length; i += 1) {\n    var listeners = listenerGroups[i];\n    var keyCombo  = listeners[0].keyCombo;\n\n    if (keyCombo === null || keyCombo.check(pressedKeys)) {\n      for (var j = 0; j < listeners.length; j += 1) {\n        var listener = listeners[j];\n\n        if (keyCombo === null) {\n          listener = {\n            keyCombo               : new KeyCombo(pressedKeys.join('+')),\n            pressHandler           : listener.pressHandler,\n            releaseHandler         : listener.releaseHandler,\n            preventRepeat          : listener.preventRepeat,\n            preventRepeatByDefault : listener.preventRepeatByDefault\n          };\n        }\n\n        if (listener.pressHandler && !listener.preventRepeat) {\n          listener.pressHandler.call(this, event);\n          if (preventRepeat) {\n            listener.preventRepeat = preventRepeat;\n            preventRepeat          = false;\n          }\n        }\n\n        if (listener.releaseHandler && this._appliedListeners.indexOf(listener) === -1) {\n          this._appliedListeners.push(listener);\n        }\n      }\n\n      if (keyCombo) {\n        for (var j = 0; j < keyCombo.keyNames.length; j += 1) {\n          var index = pressedKeys.indexOf(keyCombo.keyNames[j]);\n          if (index !== -1) {\n            pressedKeys.splice(index, 1);\n            j -= 1;\n          }\n        }\n      }\n    }\n  }\n};\n\nKeyboard.prototype._clearBindings = function(event) {\n  event || (event = {});\n\n  for (var i = 0; i < this._appliedListeners.length; i += 1) {\n    var listener = this._appliedListeners[i];\n    var keyCombo = listener.keyCombo;\n    if (keyCombo === null || !keyCombo.check(this._locale.pressedKeys)) {\n      listener.preventRepeat = listener.preventRepeatByDefault;\n      listener.releaseHandler.call(this, event);\n      this._appliedListeners.splice(i, 1);\n      i -= 1;\n    }\n  }\n};\n\nmodule.exports = Keyboard;\n"]}","\nvar KeyCombo = require('./key-combo');\n\n\nfunction Locale(name) {\n  this.localeName     = name;\n  this.pressedKeys    = [];\n  this._appliedMacros = [];\n  this._keyMap        = {};\n  this._killKeyCodes  = [];\n  this._macros        = [];\n}\n\nLocale.prototype.bindKeyCode = function(keyCode, keyNames) {\n  if (typeof keyNames === 'string') {\n    keyNames = [keyNames];\n  }\n\n  this._keyMap[keyCode] = keyNames;\n};\n\nLocale.prototype.bindMacro = function(keyComboStr, keyNames) {\n  if (typeof keyNames === 'string') {\n    keyNames = [ keyNames ];\n  }\n\n  var handler = null;\n  if (typeof keyNames === 'function') {\n    handler = keyNames;\n    keyNames = null;\n  }\n\n  var macro = {\n    keyCombo : new KeyCombo(keyComboStr),\n    keyNames : keyNames,\n    handler  : handler\n  };\n\n  this._macros.push(macro);\n};\n\nLocale.prototype.getKeyCodes = function(keyName) {\n  var keyCodes = [];\n  for (var keyCode in this._keyMap) {\n    var index = this._keyMap[keyCode].indexOf(keyName);\n    if (index > -1) { keyCodes.push(keyCode|0); }\n  }\n  return keyCodes;\n};\n\nLocale.prototype.getKeyNames = function(keyCode) {\n  return this._keyMap[keyCode] || [];\n};\n\nLocale.prototype.setKillKey = function(keyCode) {\n  if (typeof keyCode === 'string') {\n    var keyCodes = this.getKeyCodes(keyCode);\n    for (var i = 0; i < keyCodes.length; i += 1) {\n      this.setKillKey(keyCodes[i]);\n    }\n    return;\n  }\n\n  this._killKeyCodes.push(keyCode);\n};\n\nLocale.prototype.pressKey = function(keyCode) {\n  if (typeof keyCode === 'string') {\n    var keyCodes = this.getKeyCodes(keyCode);\n    for (var i = 0; i < keyCodes.length; i += 1) {\n      this.pressKey(keyCodes[i]);\n    }\n    return;\n  }\n\n  var keyNames = this.getKeyNames(keyCode);\n  for (var i = 0; i < keyNames.length; i += 1) {\n    if (this.pressedKeys.indexOf(keyNames[i]) === -1) {\n      this.pressedKeys.push(keyNames[i]);\n    }\n  }\n\n  this._applyMacros();\n};\n\nLocale.prototype.releaseKey = function(keyCode) {\n  if (typeof keyCode === 'string') {\n    var keyCodes = this.getKeyCodes(keyCode);\n    for (var i = 0; i < keyCodes.length; i += 1) {\n      this.releaseKey(keyCodes[i]);\n    }\n  }\n\n  else {\n    var keyNames         = this.getKeyNames(keyCode);\n    var killKeyCodeIndex = this._killKeyCodes.indexOf(keyCode);\n    \n    if (killKeyCodeIndex > -1) {\n      this.pressedKeys.length = 0;\n    } else {\n      for (var i = 0; i < keyNames.length; i += 1) {\n        var index = this.pressedKeys.indexOf(keyNames[i]);\n        if (index > -1) {\n          this.pressedKeys.splice(index, 1);\n        }\n      }\n    }\n\n    this._clearMacros();\n  }\n};\n\nLocale.prototype._applyMacros = function() {\n  var macros = this._macros.slice(0);\n  for (var i = 0; i < macros.length; i += 1) {\n    var macro = macros[i];\n    if (macro.keyCombo.check(this.pressedKeys)) {\n      if (macro.handler) {\n        macro.keyNames = macro.handler(this.pressedKeys);\n      }\n      for (var j = 0; j < macro.keyNames.length; j += 1) {\n        if (this.pressedKeys.indexOf(macro.keyNames[j]) === -1) {\n          this.pressedKeys.push(macro.keyNames[j]);\n        }\n      }\n      this._appliedMacros.push(macro);\n    }\n  }\n};\n\nLocale.prototype._clearMacros = function() {\n  for (var i = 0; i < this._appliedMacros.length; i += 1) {\n    var macro = this._appliedMacros[i];\n    if (!macro.keyCombo.check(this.pressedKeys)) {\n      for (var j = 0; j < macro.keyNames.length; j += 1) {\n        var index = this.pressedKeys.indexOf(macro.keyNames[j]);\n        if (index > -1) {\n          this.pressedKeys.splice(index, 1);\n        }\n      }\n      if (macro.handler) {\n        macro.keyNames = null;\n      }\n      this._appliedMacros.splice(i, 1);\n      i -= 1;\n    }\n  }\n};\n\n\nmodule.exports = Locale;\n","\nmodule.exports = function(locale, platform, userAgent) {\n\n  // general\n  locale.bindKeyCode(3,   ['cancel']);\n  locale.bindKeyCode(8,   ['backspace']);\n  locale.bindKeyCode(9,   ['tab']);\n  locale.bindKeyCode(12,  ['clear']);\n  locale.bindKeyCode(13,  ['enter']);\n  locale.bindKeyCode(16,  ['shift']);\n  locale.bindKeyCode(17,  ['ctrl']);\n  locale.bindKeyCode(18,  ['alt', 'menu']);\n  locale.bindKeyCode(19,  ['pause', 'break']);\n  locale.bindKeyCode(20,  ['capslock']);\n  locale.bindKeyCode(27,  ['escape', 'esc']);\n  locale.bindKeyCode(32,  ['space', 'spacebar']);\n  locale.bindKeyCode(33,  ['pageup']);\n  locale.bindKeyCode(34,  ['pagedown']);\n  locale.bindKeyCode(35,  ['end']);\n  locale.bindKeyCode(36,  ['home']);\n  locale.bindKeyCode(37,  ['left']);\n  locale.bindKeyCode(38,  ['up']);\n  locale.bindKeyCode(39,  ['right']);\n  locale.bindKeyCode(40,  ['down']);\n  locale.bindKeyCode(41,  ['select']);\n  locale.bindKeyCode(42,  ['printscreen']);\n  locale.bindKeyCode(43,  ['execute']);\n  locale.bindKeyCode(44,  ['snapshot']);\n  locale.bindKeyCode(45,  ['insert', 'ins']);\n  locale.bindKeyCode(46,  ['delete', 'del']);\n  locale.bindKeyCode(47,  ['help']);\n  locale.bindKeyCode(145, ['scrolllock', 'scroll']);\n  locale.bindKeyCode(187, ['equal', 'equalsign', '=']);\n  locale.bindKeyCode(188, ['comma', ',']);\n  locale.bindKeyCode(190, ['period', '.']);\n  locale.bindKeyCode(191, ['slash', 'forwardslash', '/']);\n  locale.bindKeyCode(192, ['graveaccent', '`']);\n  locale.bindKeyCode(219, ['openbracket', '[']);\n  locale.bindKeyCode(220, ['backslash', '\\\\']);\n  locale.bindKeyCode(221, ['closebracket', ']']);\n  locale.bindKeyCode(222, ['apostrophe', '\\'']);\n\n  // 0-9\n  locale.bindKeyCode(48, ['zero', '0']);\n  locale.bindKeyCode(49, ['one', '1']);\n  locale.bindKeyCode(50, ['two', '2']);\n  locale.bindKeyCode(51, ['three', '3']);\n  locale.bindKeyCode(52, ['four', '4']);\n  locale.bindKeyCode(53, ['five', '5']);\n  locale.bindKeyCode(54, ['six', '6']);\n  locale.bindKeyCode(55, ['seven', '7']);\n  locale.bindKeyCode(56, ['eight', '8']);\n  locale.bindKeyCode(57, ['nine', '9']);\n\n  // numpad\n  locale.bindKeyCode(96, ['numzero', 'num0']);\n  locale.bindKeyCode(97, ['numone', 'num1']);\n  locale.bindKeyCode(98, ['numtwo', 'num2']);\n  locale.bindKeyCode(99, ['numthree', 'num3']);\n  locale.bindKeyCode(100, ['numfour', 'num4']);\n  locale.bindKeyCode(101, ['numfive', 'num5']);\n  locale.bindKeyCode(102, ['numsix', 'num6']);\n  locale.bindKeyCode(103, ['numseven', 'num7']);\n  locale.bindKeyCode(104, ['numeight', 'num8']);\n  locale.bindKeyCode(105, ['numnine', 'num9']);\n  locale.bindKeyCode(106, ['nummultiply', 'num*']);\n  locale.bindKeyCode(107, ['numadd', 'num+']);\n  locale.bindKeyCode(108, ['numenter']);\n  locale.bindKeyCode(109, ['numsubtract', 'num-']);\n  locale.bindKeyCode(110, ['numdecimal', 'num.']);\n  locale.bindKeyCode(111, ['numdivide', 'num/']);\n  locale.bindKeyCode(144, ['numlock', 'num']);\n\n  // function keys\n  locale.bindKeyCode(112, ['f1']);\n  locale.bindKeyCode(113, ['f2']);\n  locale.bindKeyCode(114, ['f3']);\n  locale.bindKeyCode(115, ['f4']);\n  locale.bindKeyCode(116, ['f5']);\n  locale.bindKeyCode(117, ['f6']);\n  locale.bindKeyCode(118, ['f7']);\n  locale.bindKeyCode(119, ['f8']);\n  locale.bindKeyCode(120, ['f9']);\n  locale.bindKeyCode(121, ['f10']);\n  locale.bindKeyCode(122, ['f11']);\n  locale.bindKeyCode(123, ['f12']);\n\n  // secondary key symbols\n  locale.bindMacro('shift + `', ['tilde', '~']);\n  locale.bindMacro('shift + 1', ['exclamation', 'exclamationpoint', '!']);\n  locale.bindMacro('shift + 2', ['at', '@']);\n  locale.bindMacro('shift + 3', ['number', '#']);\n  locale.bindMacro('shift + 4', ['dollar', 'dollars', 'dollarsign', '$']);\n  locale.bindMacro('shift + 5', ['percent', '%']);\n  locale.bindMacro('shift + 6', ['caret', '^']);\n  locale.bindMacro('shift + 7', ['ampersand', 'and', '&']);\n  locale.bindMacro('shift + 8', ['asterisk', '*']);\n  locale.bindMacro('shift + 9', ['openparen', '(']);\n  locale.bindMacro('shift + 0', ['closeparen', ')']);\n  locale.bindMacro('shift + -', ['underscore', '_']);\n  locale.bindMacro('shift + =', ['plus', '+']);\n  locale.bindMacro('shift + [', ['opencurlybrace', 'opencurlybracket', '{']);\n  locale.bindMacro('shift + ]', ['closecurlybrace', 'closecurlybracket', '}']);\n  locale.bindMacro('shift + \\\\', ['verticalbar', '|']);\n  locale.bindMacro('shift + ;', ['colon', ':']);\n  locale.bindMacro('shift + \\'', ['quotationmark', '\\'']);\n  locale.bindMacro('shift + !,', ['openanglebracket', '<']);\n  locale.bindMacro('shift + .', ['closeanglebracket', '>']);\n  locale.bindMacro('shift + /', ['questionmark', '?']);\n\n  //a-z and A-Z\n  for (var keyCode = 65; keyCode <= 90; keyCode += 1) {\n    var keyName = String.fromCharCode(keyCode + 32);\n    var capitalKeyName = String.fromCharCode(keyCode);\n  \tlocale.bindKeyCode(keyCode, keyName);\n  \tlocale.bindMacro('shift + ' + keyName, capitalKeyName);\n  \tlocale.bindMacro('capslock + ' + keyName, capitalKeyName);\n  }\n\n  // browser caveats\n  var semicolonKeyCode = userAgent.match('Firefox') ? 59  : 186;\n  var dashKeyCode      = userAgent.match('Firefox') ? 173 : 189;\n  var leftCommandKeyCode;\n  var rightCommandKeyCode;\n  if (platform.match('Mac') && (userAgent.match('Safari') || userAgent.match('Chrome'))) {\n    leftCommandKeyCode  = 91;\n    rightCommandKeyCode = 93;\n  } else if(platform.match('Mac') && userAgent.match('Opera')) {\n    leftCommandKeyCode  = 17;\n    rightCommandKeyCode = 17;\n  } else if(platform.match('Mac') && userAgent.match('Firefox')) {\n    leftCommandKeyCode  = 224;\n    rightCommandKeyCode = 224;\n  }\n  locale.bindKeyCode(semicolonKeyCode,    ['semicolon', ';']);\n  locale.bindKeyCode(dashKeyCode,         ['dash', '-']);\n  locale.bindKeyCode(leftCommandKeyCode,  ['command', 'windows', 'win', 'super', 'leftcommand', 'leftwindows', 'leftwin', 'leftsuper']);\n  locale.bindKeyCode(rightCommandKeyCode, ['command', 'windows', 'win', 'super', 'rightcommand', 'rightwindows', 'rightwin', 'rightsuper']);\n\n  // kill keys\n  locale.setKillKey('command');\n};\n"]}