threex.keyboardstate.js 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162
  1. // THREEx.KeyboardState.js keep the current state of the keyboard.
  2. // It is possible to query it at any time. No need of an event.
  3. // This is particularly convenient in loop driven case, like in
  4. // 3D demos or games.
  5. //
  6. // # Usage
  7. //
  8. // **Step 1**: Create the object
  9. //
  10. // ```var keyboard = new THREEx.KeyboardState();```
  11. //
  12. // **Step 2**: Query the keyboard state
  13. //
  14. // This will return true if shift and A are pressed, false otherwise
  15. //
  16. // ```keyboard.pressed("shift+A")```
  17. //
  18. // **Step 3**: Stop listening to the keyboard
  19. //
  20. // ```keyboard.destroy()```
  21. //
  22. // NOTE: this library may be nice as standaline. independant from three.js
  23. // - rename it keyboardForGame
  24. //
  25. // # Code
  26. //
  27. /** @namespace */
  28. var THREEx = THREEx || {};
  29. /**
  30. * - NOTE: it would be quite easy to push event-driven too
  31. * - microevent.js for events handling
  32. * - in this._onkeyChange, generate a string from the DOM event
  33. * - use this as event name
  34. */
  35. THREEx.KeyboardState = function(domElement)
  36. {
  37. this.domElement= domElement || document;
  38. // to store the current state
  39. this.keyCodes = {};
  40. this.modifiers = {};
  41. // create callback to bind/unbind keyboard events
  42. var _this = this;
  43. this._onKeyDown = function(event){ _this._onKeyChange(event) }
  44. this._onKeyUp = function(event){ _this._onKeyChange(event) }
  45. // bind keyEvents
  46. this.domElement.addEventListener("keydown", this._onKeyDown, false);
  47. this.domElement.addEventListener("keyup", this._onKeyUp, false);
  48. // create callback to bind/unbind window blur event
  49. this._onBlur = function(){
  50. for(var prop in _this.keyCodes) _this.keyCodes[prop] = false;
  51. for(var prop in _this.modifiers) _this.modifiers[prop] = false;
  52. }
  53. // bind window blur
  54. window.addEventListener("blur", this._onBlur, false);
  55. }
  56. /**
  57. * To stop listening of the keyboard events
  58. */
  59. THREEx.KeyboardState.prototype.destroy = function()
  60. {
  61. // unbind keyEvents
  62. this.domElement.removeEventListener("keydown", this._onKeyDown, false);
  63. this.domElement.removeEventListener("keyup", this._onKeyUp, false);
  64. // unbind window blur event
  65. window.removeEventListener("blur", this._onBlur, false);
  66. }
  67. THREEx.KeyboardState.MODIFIERS = ['shift', 'ctrl', 'alt', 'meta'];
  68. THREEx.KeyboardState.ALIAS = {
  69. 'left' : 37,
  70. 'up' : 38,
  71. 'right' : 39,
  72. 'down' : 40,
  73. 'space' : 32,
  74. 'pageup' : 33,
  75. 'pagedown' : 34,
  76. 'tab' : 9,
  77. 'escape' : 27
  78. };
  79. /**
  80. * to process the keyboard dom event
  81. */
  82. THREEx.KeyboardState.prototype._onKeyChange = function(event)
  83. {
  84. // log to debug
  85. //console.log("onKeyChange", event, event.keyCode, event.shiftKey, event.ctrlKey, event.altKey, event.metaKey)
  86. // update this.keyCodes
  87. var keyCode = event.keyCode
  88. var pressed = event.type === 'keydown' ? true : false
  89. this.keyCodes[keyCode] = pressed
  90. // update this.modifiers
  91. this.modifiers['shift'] = event.shiftKey
  92. this.modifiers['ctrl'] = event.ctrlKey
  93. this.modifiers['alt'] = event.altKey
  94. this.modifiers['meta'] = event.metaKey
  95. }
  96. /**
  97. * query keyboard state to know if a key is pressed of not
  98. *
  99. * @param {String} keyDesc the description of the key. format : modifiers+key e.g shift+A
  100. * @returns {Boolean} true if the key is pressed, false otherwise
  101. */
  102. THREEx.KeyboardState.prototype.pressed = function(keyDesc){
  103. var keys = keyDesc.split("+");
  104. for(var i = 0; i < keys.length; i++){
  105. var key = keys[i]
  106. var pressed = false
  107. if( THREEx.KeyboardState.MODIFIERS.indexOf( key ) !== -1 ){
  108. pressed = this.modifiers[key];
  109. }else if( Object.keys(THREEx.KeyboardState.ALIAS).indexOf( key ) != -1 ){
  110. pressed = this.keyCodes[ THREEx.KeyboardState.ALIAS[key] ];
  111. }else {
  112. pressed = this.keyCodes[key.toUpperCase().charCodeAt(0)]
  113. }
  114. if( !pressed) return false;
  115. };
  116. return true;
  117. }
  118. /**
  119. * return true if an event match a keyDesc
  120. * @param {KeyboardEvent} event keyboard event
  121. * @param {String} keyDesc string description of the key
  122. * @return {Boolean} true if the event match keyDesc, false otherwise
  123. */
  124. THREEx.KeyboardState.prototype.eventMatches = function(event, keyDesc) {
  125. var aliases = THREEx.KeyboardState.ALIAS
  126. var aliasKeys = Object.keys(aliases)
  127. var keys = keyDesc.split("+")
  128. // log to debug
  129. // console.log("eventMatches", event, event.keyCode, event.shiftKey, event.ctrlKey, event.altKey, event.metaKey)
  130. for(var i = 0; i < keys.length; i++){
  131. var key = keys[i];
  132. var pressed = false;
  133. if( key === 'shift' ){
  134. pressed = (event.shiftKey ? true : false)
  135. }else if( key === 'ctrl' ){
  136. pressed = (event.ctrlKey ? true : false)
  137. }else if( key === 'alt' ){
  138. pressed = (event.altKey ? true : false)
  139. }else if( key === 'meta' ){
  140. pressed = (event.metaKey ? true : false)
  141. }else if( aliasKeys.indexOf( key ) !== -1 ){
  142. pressed = (event.keyCode === aliases[key] ? true : false);
  143. }else if( event.keyCode === key.toUpperCase().charCodeAt(0) ){
  144. pressed = true;
  145. }
  146. if( !pressed ) return false;
  147. }
  148. return true;
  149. }