/fp10/src/com/furusystems/dconsole2/core/input/KeyboardSequences.as

http://doomsdayconsole.googlecode.com/ · ActionScript · 328 lines · 190 code · 27 blank · 111 comment · 37 complexity · 41349af5a865ff57436fc5b98394a2e4 MD5 · raw file

  1. package com.furusystems.dconsole2.core.input
  2. {
  3. import flash.events.KeyboardEvent;
  4. /**
  5. * Maintains a list of keyboard sequences and dispatches the callback function when a sequence has been triggered.
  6. *
  7. * @author Cristobal Dabed
  8. * @version 0.2
  9. */
  10. public final class KeyboardSequences implements KeyboardList
  11. {
  12. // TODO: Add a more robust validator for validateKeyboardSequence
  13. /* Constants */
  14. public const KEYBOARD_SEQUENCES_MIN_LENGTH:uint = 1; // NOTE: Should min length be 2?
  15. /* Variables */
  16. private static var INSTANCE:KeyboardSequences = null;
  17. private var keyboardSequences:Vector.<KeyboardSequence> = new Vector.<KeyboardSequence>();
  18. /* @group API */
  19. /**
  20. * Instance
  21. *
  22. * @return
  23. * Returns the singleton representation of this class.
  24. */
  25. public static function get instance():KeyboardSequences
  26. {
  27. if (!INSTANCE)
  28. {
  29. INSTANCE = new KeyboardSequences();
  30. }
  31. return INSTANCE;
  32. }
  33. /**
  34. * Add
  35. *
  36. * @param keyCodes The list of keycodes to trigger a callback.
  37. * @param callback The callback function to call.
  38. * @param override Optional override for an existing keyboard sequence with the new function.
  39. */
  40. public function add(keyCodes:Array, callback:Function, override:Boolean = false, replaceAll:Boolean = true):Boolean
  41. {
  42. var success:Boolean = true;
  43. /*
  44. * 1. Must satisfy minimum of length 1
  45. * 2. Must satisfy a valid callback.
  46. */
  47. if (!validateKeyboardSequence(keyCodes))
  48. {
  49. throw new Error("A keyboard sequence can not have less than " + KEYBOARD_SEQUENCES_MIN_LENGTH + " elements");
  50. }
  51. if (typeof(callback) != "function")
  52. {
  53. throw new Error("Invalid callback function");
  54. }
  55. if (replaceAll)
  56. {
  57. keyboardSequences = new Vector.<KeyboardSequence>();
  58. }
  59. /*
  60. * 1. If the keyCodes are not registered yet add them.
  61. * 2. If the exist and set to override, remove the old one and add the new one.
  62. * 3. Warn the user that the keyCodes could not be added success is set to false.
  63. */
  64. if (!has(keyCodes))
  65. {
  66. keyboardSequences.push(new KeyboardSequence(keyCodes, callback));
  67. }
  68. else if (override)
  69. {
  70. remove(keyCodes);
  71. keyboardSequences.push(new KeyboardSequence(keyCodes, callback));
  72. }
  73. else
  74. {
  75. //trace("Warn: Keyboard sequence exists and was not set to be overriden.");
  76. success = false;
  77. }
  78. return success;
  79. }
  80. /**
  81. * Remove keyboard sequence.
  82. *
  83. * @param keyCodes The keyCodes to stop listening on.
  84. *
  85. * @return
  86. * Returns true or false depending on wether it successfully removed the keyCode sequence from the list or not.
  87. */
  88. public function remove(keyCodes:Array):Boolean
  89. {
  90. var success:Boolean = false;
  91. if (!isEmpty())
  92. {
  93. if (has(keyCodes))
  94. {
  95. var i:int = 0;
  96. for (var l:int = keyboardSequences.length; i < l; i++)
  97. {
  98. if (inKeyboardSequence(keyCodes, keyboardSequences[i]))
  99. {
  100. break;
  101. }
  102. }
  103. keyboardSequences.splice(i, 1);
  104. success = true;
  105. }
  106. }
  107. else
  108. {
  109. //trace("Warn: Empty keyboard sequences, none to remove");
  110. }
  111. return success;
  112. }
  113. /**
  114. * Has
  115. *
  116. * @param keyCodes The keyCodes to search on.
  117. *
  118. * @return
  119. * Returns true or false depending on wether the given keyCode sequence is in the list or not.
  120. */
  121. public function has(keyCodes:Array):Boolean
  122. {
  123. var success:Boolean = false;
  124. for (var i:int = 0, l:int = keyboardSequences.length; i < l; i++)
  125. {
  126. if (inKeyboardSequence(keyCodes, keyboardSequences[i]))
  127. {
  128. success = true;
  129. break;
  130. }
  131. }
  132. return success;
  133. }
  134. /**
  135. * Remove All
  136. */
  137. public function removeAll():void
  138. {
  139. // Reset the internal list.
  140. if (!isEmpty())
  141. {
  142. while (keyboardSequences.length > 0)
  143. {
  144. keyboardSequences.pop();
  145. }
  146. }
  147. }
  148. /**
  149. * Is empty
  150. *
  151. * @return
  152. * Returns true or false depending on wether there are any registered keyboard sequences or not.
  153. */
  154. public function isEmpty():Boolean
  155. {
  156. return (keyboardSequences.length == 0 ? true : false);
  157. }
  158. /**
  159. * Validate keyboard sequence
  160. * - For the moment only validate on the length.
  161. *
  162. * @return
  163. * Returns true or false wether the keyboard sequence is valid or not.
  164. */
  165. public function validateKeyboardSequence(keyCodes:Array):Boolean
  166. {
  167. return (keyCodes.length < KEYBOARD_SEQUENCES_MIN_LENGTH ? false : true);
  168. }
  169. /* @end */
  170. /* @group Delegates called irectly by the KeyboardManager */
  171. // We could have added custom events but do not to save some resources while keyboard input should be as fast as possible.
  172. /**
  173. * On key down.
  174. *
  175. * @param event The keyboard event.
  176. */
  177. public function onKeyDown(event:KeyboardEvent):Boolean
  178. {
  179. return false;
  180. // Nothing to do here, placeholder function only.
  181. }
  182. /**
  183. * On key up.
  184. *
  185. * @param event The keyboard event.
  186. */
  187. public function onKeyUp(event:KeyboardEvent):Boolean
  188. {
  189. var success:Boolean = false;
  190. if (!isEmpty())
  191. {
  192. var value:uint = event.charCode;
  193. var modifier:Boolean = false;
  194. /*
  195. * If the charCode is 0 use the keycode instead.
  196. * Since the charCode is not an unicode character or is either a modifier.
  197. */
  198. if (value == 0)
  199. {
  200. value = event.keyCode;
  201. }
  202. switch (value)
  203. {
  204. case KeyBindings.ALT:
  205. case KeyBindings.SHIFT:
  206. case KeyBindings.CTRL:
  207. modifier = true;
  208. break;
  209. }
  210. // Only compare with none modifiers.
  211. if (!modifier)
  212. {
  213. /*
  214. * Loop over the keyboard sequences.
  215. * If the value matches the current keyCode in the keystrokes for the given keyboard sequence
  216. * check if it is completed, if it is completed then set success to true and trigger the callback outside of the loop block.
  217. *
  218. * If the current value did not match and or the reset flag is set, then reset the given keyboard sequence
  219. * to the default keystrokes state.
  220. *
  221. * If we had a successful match break out of the loop block and execute the callback on the i-th element which matched.
  222. */
  223. for (var i:int = 0, l:int = keyboardSequences.length, reset:Boolean = true; i < l; i++, reset = true)
  224. {
  225. if (value == keyboardSequences[i].keystrokes.shift())
  226. {
  227. if (keyboardSequences[i].keystrokes.length == 0)
  228. {
  229. success = true;
  230. }
  231. else
  232. {
  233. reset = false;
  234. }
  235. }
  236. if (reset)
  237. {
  238. keyboardSequences[i].keystrokes = keyboardSequences[i].keyCodes.concat();
  239. }
  240. if (success)
  241. {
  242. break;
  243. }
  244. }
  245. if (success)
  246. {
  247. executeCallback(keyboardSequences[i].callback);
  248. }
  249. }
  250. }
  251. return success;
  252. }
  253. /* @end */
  254. /* @group Private Functions */
  255. /**
  256. * In keyboard sequence
  257. *
  258. * @param keyCodes The keyCodes to test for.
  259. * @param keyboardSequence The keyboardSequence to check on.
  260. */
  261. private function inKeyboardSequence(keyCodes:Array, keyboardSequence:KeyboardSequence):Boolean
  262. {
  263. var success:Boolean = true;
  264. if (keyCodes.length == keyboardSequence.keyCodes.length)
  265. {
  266. for (var i:int = 0, l:int = keyCodes.length; i < l; i++)
  267. {
  268. if (keyCodes[i] != keyboardSequence.keyCodes[i])
  269. {
  270. success = false;
  271. break;
  272. }
  273. }
  274. }
  275. return success;
  276. }
  277. /**
  278. * Execute callback.
  279. *
  280. * @param callback The callback to execute.
  281. */
  282. private function executeCallback(callback:Function):void
  283. {
  284. if (typeof(callback) == "function")
  285. {
  286. try
  287. {
  288. callback();
  289. }
  290. catch (error:Error)
  291. { /* supress warning. */
  292. }
  293. }
  294. else
  295. {
  296. //trace("Warn: could not execute the callback, perhaps not a valid callback function...");
  297. }
  298. }
  299. /* @end */
  300. }
  301. }