Version 3.18.1
Show:

File: datatable/js/keynav.js

  1. /**
  2. Provides keyboard navigation of DataTable cells and support for adding other
  3. keyboard actions.
  4. @module datatable
  5. @submodule datatable-keynav
  6. */
  7. var arrEach = Y.Array.each,
  8. /**
  9. A DataTable class extension that provides navigation via keyboard, based on
  10. WAI-ARIA recommendation for the [Grid widget](http://www.w3.org/WAI/PF/aria-practices/#grid)
  11. and extensible to support other actions.
  12. @class DataTable.KeyNav
  13. @for DataTable
  14. */
  15. DtKeyNav = function (){};
  16. /**
  17. Mapping of key codes to friendly key names that can be used in the
  18. [keyActions](#property_keyActions) property and [ARIA_ACTIONS](#property_ARIA_ACTIONS)
  19. property.
  20. It contains aliases for the following keys:
  21. <ul>
  22. <li>backspace</li>
  23. <li>tab</li>
  24. <li>enter</li>
  25. <li>esc</li>
  26. <li>space</li>
  27. <li>pgup</li>
  28. <li>pgdown</li>
  29. <li>end</li>
  30. <li>home</li>
  31. <li>left</li>
  32. <li>up</li>
  33. <li>right</li>
  34. <li>down</li>
  35. <li>f1 .. f12</li>
  36. </ul>
  37. @property KEY_NAMES
  38. @type {Object}
  39. @static
  40. **/
  41. DtKeyNav.KEY_NAMES = {
  42. 8: 'backspace',
  43. 9: 'tab',
  44. 13: 'enter',
  45. 27: 'esc',
  46. 32: 'space',
  47. 33: 'pgup',
  48. 34: 'pgdown',
  49. 35: 'end',
  50. 36: 'home',
  51. 37: 'left',
  52. 38: 'up',
  53. 39: 'right',
  54. 40: 'down',
  55. 112:'f1',
  56. 113:'f2',
  57. 114:'f3',
  58. 115:'f4',
  59. 116:'f5',
  60. 117:'f6',
  61. 118:'f7',
  62. 119:'f8',
  63. 120:'f9',
  64. 121:'f10',
  65. 122:'f11',
  66. 123:'f12'
  67. };
  68. /**
  69. Mapping of key codes to actions according to the WAI-ARIA suggestion for the
  70. [Grid Widget](http://www.w3.org/WAI/PF/aria-practices/#grid).
  71. The key for each entry is a key-code or [keyName](#property_KEY_NAMES) while the
  72. value can be a function that performs the action or a string. If a string,
  73. it can either correspond to the name of a method in this module (or any
  74. method in a DataTable instance) or the name of an event to fire.
  75. @property ARIA_ACTIONS
  76. @type Object
  77. @static
  78. */
  79. DtKeyNav.ARIA_ACTIONS = {
  80. left: '_keyMoveLeft',
  81. right: '_keyMoveRight',
  82. up: '_keyMoveUp',
  83. down: '_keyMoveDown',
  84. home: '_keyMoveRowStart',
  85. end: '_keyMoveRowEnd',
  86. pgup: '_keyMoveColTop',
  87. pgdown: '_keyMoveColBottom'
  88. };
  89. DtKeyNav.ATTRS = {
  90. /**
  91. Cell that's currently either focused or
  92. focusable when the DataTable gets the focus.
  93. @attribute focusedCell
  94. @type Node
  95. @default first cell in the table.
  96. **/
  97. focusedCell: {
  98. setter: '_focusedCellSetter'
  99. },
  100. /**
  101. Determines whether it is possible to navigate into the header area.
  102. The examples referenced in the document show both behaviors so it seems
  103. it is optional.
  104. @attribute keyIntoHeaders
  105. @type Boolean
  106. @default true
  107. */
  108. keyIntoHeaders: {
  109. value: true
  110. }
  111. };
  112. Y.mix( DtKeyNav.prototype, {
  113. /**
  114. Table of actions to be performed for each key. It is loaded with a clone
  115. of [ARIA_ACTIONS](#property_ARIA_ACTIONS) by default.
  116. The key for each entry is either a key-code or an alias from the
  117. [KEY_NAMES](#property_KEY_NAMES) table. They can be prefixed with any combination
  118. of the modifier keys `alt`, `ctrl`, `meta` or `shift` each followed by a hyphen,
  119. such as `"ctrl-shift-up"` (modifiers, if more than one, should appear in alphabetical order).
  120. The value for each entry should be a function or the name of a method in
  121. the DataTable instance. The method will receive the original keyboard
  122. EventFacade as its only argument.
  123. If the value is a string and it cannot be resolved into a method,
  124. it will be assumed to be the name of an event to fire. The listener for that
  125. event will receive an EventFacade containing references to the cell that has the focus,
  126. the row, column and, unless it is a header row, the record it corresponds to.
  127. The second argument will be the original EventFacade for the keyboard event.
  128. @property keyActions
  129. @type {Object}
  130. @default Y.DataTable.keyNav.ARIA_ACTIONS
  131. */
  132. keyActions: null,
  133. /**
  134. Array containing the event handles to any event that might need to be detached
  135. on destruction.
  136. @property _keyNavSubscr
  137. @type Array
  138. @default null,
  139. @private
  140. */
  141. _keyNavSubscr: null,
  142. /**
  143. Reference to the THead section that holds the headers for the datatable.
  144. For a Scrolling DataTable, it is the one visible to the user.
  145. @property _keyNavTHead
  146. @type Node
  147. @default: null
  148. @private
  149. */
  150. _keyNavTHead: null,
  151. /**
  152. Indicates if the headers of the table are nested or not.
  153. Nested headers makes navigation in the headers much harder.
  154. @property _keyNavNestedHeaders
  155. @default false
  156. @private
  157. */
  158. _keyNavNestedHeaders: false,
  159. /**
  160. CSS class name prefix for columns, used to search for a cell by key.
  161. @property _keyNavColPrefix
  162. @type String
  163. @default null (initialized via getClassname() )
  164. @private
  165. */
  166. _keyNavColPrefix:null,
  167. /**
  168. Regular expression to extract the column key from a cell via its CSS class name.
  169. @property _keyNavColRegExp
  170. @type RegExp
  171. @default null (initialized based on _keyNavColPrefix)
  172. @private
  173. */
  174. _keyNavColRegExp:null,
  175. initializer: function () {
  176. this.onceAfter('render', this._afterKeyNavRender);
  177. this._keyNavSubscr = [
  178. this.after('focusedCellChange', this._afterKeyNavFocusedCellChange),
  179. this.after('focusedChange', this._afterKeyNavFocusedChange)
  180. ];
  181. this._keyNavColPrefix = this.getClassName('col', '');
  182. this._keyNavColRegExp = new RegExp(this._keyNavColPrefix + '(.+?)(\\s|$)');
  183. this.keyActions = Y.clone(DtKeyNav.ARIA_ACTIONS);
  184. },
  185. destructor: function () {
  186. arrEach(this._keyNavSubscr, function (evHandle) {
  187. if (evHandle && evHandle.detach) {
  188. evHandle.detach();
  189. }
  190. });
  191. },
  192. /**
  193. Sets the tabIndex on the focused cell and, if the DataTable has the focus,
  194. sets the focus on it.
  195. @method _afterFocusedCellChange
  196. @param e {EventFacade}
  197. @private
  198. */
  199. _afterKeyNavFocusedCellChange: function (e) {
  200. var newVal = e.newVal,
  201. prevVal = e.prevVal;
  202. if (prevVal) {
  203. prevVal.set('tabIndex', -1);
  204. }
  205. if (newVal) {
  206. newVal.set('tabIndex', 0);
  207. if (this.get('focused')) {
  208. newVal.scrollIntoView();
  209. newVal.focus();
  210. }
  211. } else {
  212. this.set('focused', null);
  213. }
  214. },
  215. /**
  216. When the DataTable gets the focus, it ensures the correct cell regains
  217. the focus.
  218. @method _afterKeyNavFocusedChange
  219. @param e {EventFacade}
  220. @private
  221. */
  222. _afterKeyNavFocusedChange: function (e) {
  223. var cell = this.get('focusedCell');
  224. if (e.newVal) {
  225. if (cell) {
  226. cell.scrollIntoView();
  227. cell.focus();
  228. } else {
  229. this._keyMoveFirst();
  230. }
  231. } else {
  232. if (cell) {
  233. cell.blur();
  234. }
  235. }
  236. },
  237. /**
  238. Subscribes to the events on the DataTable elements once they have been rendered,
  239. finds out the header section and makes the top-left element focusable.
  240. @method _afterKeyNavRender
  241. @private
  242. */
  243. _afterKeyNavRender: function () {
  244. var cbx = this.get('contentBox');
  245. this._keyNavSubscr.push(
  246. cbx.on('keydown', this._onKeyNavKeyDown, this),
  247. cbx.on('click', this._onKeyNavClick, this)
  248. );
  249. this._keyNavTHead = (this._yScrollHeader || this._tableNode).one('thead');
  250. this._keyMoveFirst();
  251. // determine if we have nested headers
  252. this._keyNavNestedHeaders = (this.get('columns').length !== this.head.theadNode.all('th').size());
  253. },
  254. /**
  255. In response to a click event, it sets the focus on the clicked cell
  256. @method _onKeyNavClick
  257. @param e {EventFacade}
  258. @private
  259. */
  260. _onKeyNavClick: function (e) {
  261. var cell = e.target.ancestor((this.get('keyIntoHeaders') ? 'td, th': 'td'), true);
  262. if (cell) {
  263. this.focus();
  264. this.set('focusedCell', cell);
  265. }
  266. },
  267. /**
  268. Responds to a key down event by executing the action set in the
  269. [keyActions](#property_keyActions) table.
  270. @method _onKeyNavKeyDown
  271. @param e {EventFacade}
  272. @private
  273. */
  274. _onKeyNavKeyDown: function (e) {
  275. var keyCode = e.keyCode,
  276. keyName = DtKeyNav.KEY_NAMES[keyCode] || keyCode,
  277. action;
  278. arrEach(['alt', 'ctrl', 'meta', 'shift'], function (modifier) {
  279. if (e[modifier + 'Key']) {
  280. keyCode = modifier + '-' + keyCode;
  281. keyName = modifier + '-' + keyName;
  282. }
  283. });
  284. action = this.keyActions[keyCode] || this.keyActions[keyName];
  285. if (typeof action === 'string') {
  286. if (this[action]) {
  287. this[action].call(this, e);
  288. } else {
  289. this._keyNavFireEvent(action, e);
  290. }
  291. } else {
  292. action.call(this, e);
  293. }
  294. },
  295. /**
  296. If the action associated to a key combination is a string and no method
  297. by that name was found in this instance, this method will
  298. fire an event using that string and provides extra information
  299. to the listener.
  300. @method _keyNavFireEvent
  301. @param action {String} Name of the event to fire
  302. @param e {EventFacade} Original facade from the keydown event.
  303. @private
  304. */
  305. _keyNavFireEvent: function (action, e) {
  306. var cell = e.target.ancestor('td, th', true);
  307. if (cell) {
  308. this.fire(action, {
  309. cell: cell,
  310. row: cell.ancestor('tr'),
  311. record: this.getRecord(cell),
  312. column: this.getColumn(cell.get('cellIndex'))
  313. }, e);
  314. }
  315. },
  316. /**
  317. Sets the focus on the very first cell in the header of the table.
  318. @method _keyMoveFirst
  319. @private
  320. */
  321. _keyMoveFirst: function () {
  322. this.set('focusedCell' , (this.get('keyIntoHeaders') ? this._keyNavTHead.one('th') : this._tbodyNode.one('td')), {src:'keyNav'});
  323. },
  324. /**
  325. Sets the focus on the cell to the left of the currently focused one.
  326. Does not wrap, following the WAI-ARIA recommendation.
  327. @method _keyMoveLeft
  328. @param e {EventFacade} Event Facade for the keydown event
  329. @private
  330. */
  331. _keyMoveLeft: function (e) {
  332. var cell = this.get('focusedCell'),
  333. index = cell.get('cellIndex'),
  334. row = cell.ancestor();
  335. e.preventDefault();
  336. if (index === 0) {
  337. return;
  338. }
  339. cell = row.get('cells').item(index - 1);
  340. this.set('focusedCell', cell , {src:'keyNav'});
  341. },
  342. /**
  343. Sets the focus on the cell to the right of the currently focused one.
  344. Does not wrap, following the WAI-ARIA recommendation.
  345. @method _keyMoveRight
  346. @param e {EventFacade} Event Facade for the keydown event
  347. @private
  348. */
  349. _keyMoveRight: function (e) {
  350. var cell = this.get('focusedCell'),
  351. row = cell.ancestor('tr'),
  352. section = row.ancestor(),
  353. inHead = section === this._keyNavTHead,
  354. nextCell,
  355. parent;
  356. e.preventDefault();
  357. // a little special with nested headers
  358. /*
  359. +-------------+-------+
  360. | ABC | DE |
  361. +-------+-----+---+---+
  362. | AB | | | |
  363. +---+---+ | | |
  364. | A | B | C | D | E |
  365. +---+---+-----+---+---+
  366. */
  367. nextCell = cell.next();
  368. if (row.get('rowIndex') !== 0 && inHead && this._keyNavNestedHeaders) {
  369. if (nextCell) {
  370. cell = nextCell;
  371. } else { //-- B -> C
  372. parent = this._getTHParent(cell);
  373. if (parent && parent.next()) {
  374. cell = parent.next();
  375. } else { //-- E -> ...
  376. return;
  377. }
  378. }
  379. } else {
  380. if (!nextCell) {
  381. return;
  382. } else {
  383. cell = nextCell;
  384. }
  385. }
  386. this.set('focusedCell', cell, { src:'keyNav' });
  387. },
  388. /**
  389. Sets the focus on the cell above the currently focused one.
  390. It will move into the headers when the top of the data rows is reached.
  391. Does not wrap, following the WAI-ARIA recommendation.
  392. @method _keyMoveUp
  393. @param e {EventFacade} Event Facade for the keydown event
  394. @private
  395. */
  396. _keyMoveUp: function (e) {
  397. var cell = this.get('focusedCell'),
  398. cellIndex = cell.get('cellIndex'),
  399. row = cell.ancestor('tr'),
  400. rowIndex = row.get('rowIndex'),
  401. section = row.ancestor(),
  402. sectionRows = section.get('rows'),
  403. inHead = section === this._keyNavTHead,
  404. parent;
  405. e.preventDefault();
  406. if (!inHead) {
  407. rowIndex -= section.get('firstChild').get('rowIndex');
  408. }
  409. if (rowIndex === 0) {
  410. if (inHead || !this.get('keyIntoHeaders')) {
  411. return;
  412. }
  413. section = this._keyNavTHead;
  414. sectionRows = section.get('rows');
  415. if (this._keyNavNestedHeaders) {
  416. key = this._getCellColumnName(cell);
  417. cell = section.one('.' + this._keyNavColPrefix + key);
  418. cellIndex = cell.get('cellIndex');
  419. row = cell.ancestor('tr');
  420. } else {
  421. row = section.get('firstChild');
  422. cell = row.get('cells').item(cellIndex);
  423. }
  424. } else {
  425. if (inHead && this._keyNavNestedHeaders) {
  426. key = this._getCellColumnName(cell);
  427. parent = this._columnMap[key]._parent;
  428. if (parent) {
  429. cell = section.one('#' + parent.id);
  430. }
  431. } else {
  432. row = sectionRows.item(rowIndex -1);
  433. cell = row.get('cells').item(cellIndex);
  434. }
  435. }
  436. this.set('focusedCell', cell);
  437. },
  438. /**
  439. Sets the focus on the cell below the currently focused one.
  440. It will move into the data rows when the bottom of the header rows is reached.
  441. Does not wrap, following the WAI-ARIA recommendation.
  442. @method _keyMoveDown
  443. @param e {EventFacade} Event Facade for the keydown event
  444. @private
  445. */
  446. _keyMoveDown: function (e) {
  447. var cell = this.get('focusedCell'),
  448. cellIndex = cell.get('cellIndex'),
  449. row = cell.ancestor('tr'),
  450. rowIndex = row.get('rowIndex') + 1,
  451. section = row.ancestor(),
  452. inHead = section === this._keyNavTHead,
  453. tbody = (this.body && this.body.tbodyNode),
  454. sectionRows = section.get('rows'),
  455. key,
  456. children;
  457. e.preventDefault();
  458. if (inHead) { // focused cell is in the header
  459. if (this._keyNavNestedHeaders) { // the header is nested
  460. key = this._getCellColumnName(cell);
  461. children = this._columnMap[key].children;
  462. rowIndex += (cell.getAttribute('rowspan') || 1) - 1;
  463. if (children) {
  464. // stay in thead
  465. cell = section.one('#' + children[0].id);
  466. } else {
  467. // moving into tbody
  468. cell = tbody.one('.' + this._keyNavColPrefix + key);
  469. section = tbody;
  470. sectionRows = section.get('rows');
  471. }
  472. cellIndex = cell.get('cellIndex');
  473. } else { // the header is not nested
  474. row = tbody.one('tr');
  475. cell = row.get('cells').item(cellIndex);
  476. }
  477. }
  478. // offset row index to tbody
  479. rowIndex -= sectionRows.item(0).get('rowIndex');
  480. if (rowIndex >= sectionRows.size()) {
  481. if (!inHead) { // last row in tbody
  482. return;
  483. }
  484. section = tbody;
  485. row = section.one('tr');
  486. } else {
  487. row = sectionRows.item(rowIndex);
  488. }
  489. this.set('focusedCell', row.get('cells').item(cellIndex));
  490. },
  491. /**
  492. Sets the focus on the left-most cell of the row containing the currently focused cell.
  493. @method _keyMoveRowStart
  494. @param e {EventFacade} Event Facade for the keydown event
  495. @private
  496. */
  497. _keyMoveRowStart: function (e) {
  498. var row = this.get('focusedCell').ancestor();
  499. this.set('focusedCell', row.get('firstChild'), {src:'keyNav'});
  500. e.preventDefault();
  501. },
  502. /**
  503. Sets the focus on the right-most cell of the row containing the currently focused cell.
  504. @method _keyMoveRowEnd
  505. @param e {EventFacade} Event Facade for the keydown event
  506. @private
  507. */
  508. _keyMoveRowEnd: function (e) {
  509. var row = this.get('focusedCell').ancestor();
  510. this.set('focusedCell', row.get('lastChild'), {src:'keyNav'});
  511. e.preventDefault();
  512. },
  513. /**
  514. Sets the focus on the top-most cell of the column containing the currently focused cell.
  515. It would normally be a header cell.
  516. @method _keyMoveColTop
  517. @param e {EventFacade} Event Facade for the keydown event
  518. @private
  519. */
  520. _keyMoveColTop: function (e) {
  521. var cell = this.get('focusedCell'),
  522. cellIndex = cell.get('cellIndex'),
  523. key, header;
  524. e.preventDefault();
  525. if (this._keyNavNestedHeaders && this.get('keyIntoHeaders')) {
  526. key = this._getCellColumnName(cell);
  527. header = this._columnMap[key];
  528. while (header._parent) {
  529. header = header._parent;
  530. }
  531. cell = this._keyNavTHead.one('#' + header.id);
  532. } else {
  533. cell = (this.get('keyIntoHeaders') ? this._keyNavTHead: this._tbodyNode).get('firstChild').get('cells').item(cellIndex);
  534. }
  535. this.set('focusedCell', cell , {src:'keyNav'});
  536. },
  537. /**
  538. Sets the focus on the last cell of the column containing the currently focused cell.
  539. @method _keyMoveColBottom
  540. @param e {EventFacade} Event Facade for the keydown event
  541. @private
  542. */
  543. _keyMoveColBottom: function (e) {
  544. var cell = this.get('focusedCell'),
  545. cellIndex = cell.get('cellIndex');
  546. this.set('focusedCell', this._tbodyNode.get('lastChild').get('cells').item(cellIndex), {src:'keyNav'});
  547. e.preventDefault();
  548. },
  549. /**
  550. Setter method for the [focusedCell](#attr_focusedCell) attribute.
  551. Checks that the passed value is a Node, either a TD or TH and is
  552. contained within the DataTable contentBox.
  553. @method _focusedCellSetter
  554. @param cell {Node} DataTable cell to receive the focus
  555. @return cell or Y.Attribute.INVALID_VALUE
  556. @private
  557. */
  558. _focusedCellSetter: function (cell) {
  559. if (cell instanceof Y.Node) {
  560. var tag = cell.get('tagName').toUpperCase();
  561. if ((tag === 'TD' || tag === 'TH') && this.get('contentBox').contains(cell) ) {
  562. return cell;
  563. }
  564. } else if (cell === null) {
  565. return cell;
  566. }
  567. return Y.Attribute.INVALID_VALUE;
  568. },
  569. /**
  570. Retrieves the parent cell of the given TH cell. If there is no parent for
  571. the provided cell, null is returned.
  572. @protected
  573. @method _getTHParent
  574. @param {Node} thCell Cell to find parent of
  575. @return {Node} Parent of the cell provided or null
  576. */
  577. _getTHParent: function (thCell) {
  578. var key = this._getCellColumnName(thCell),
  579. parent = this._columnMap[key] && this._columnMap[key]._parent;
  580. if (parent) {
  581. return thCell.ancestor().ancestor().one('.' + this._keyNavColPrefix + parent.key);
  582. }
  583. return null;
  584. },
  585. /**
  586. Retrieves the column name based from the data attribute on the cell if
  587. available. Other wise, extracts the column name from the classname
  588. @protected
  589. @method _getCellColumnName
  590. @param {Node} cell Cell to get column name from
  591. @return String Column name of the provided cell
  592. */
  593. _getCellColumnName: function (cell) {
  594. return cell.getData('yui3-col-id') || this._keyNavColRegExp.exec(cell.get('className'))[1];
  595. }
  596. });
  597. Y.DataTable.KeyNav = DtKeyNav;
  598. Y.Base.mix(Y.DataTable, [DtKeyNav]);