subNavFoundation.ts 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134
  1. import BaseFoundation, { DefaultAdapter } from '../base/foundation';
  2. import isEnterPress from '../utils/isEnterPress';
  3. const addKeys = function addKeys(originKeys: (string | number)[] = [], ...willAddKeys: (string | number)[]) {
  4. const keySet = new Set(originKeys);
  5. willAddKeys.forEach(key => key && keySet.add(key));
  6. return Array.from(keySet);
  7. };
  8. const removeKeys = function removeKeys(originKeys: (string | number)[] = [], ...willRemoveKeys: (string | number)[]) {
  9. const keySet = new Set(originKeys);
  10. willRemoveKeys.forEach(key => key && keySet.delete(key));
  11. return Array.from(keySet);
  12. };
  13. export interface OnOpenChangeData {
  14. itemKey: string | number;
  15. openKeys: (string | number)[];
  16. isOpen: boolean
  17. }
  18. export interface OnClickData extends OnOpenChangeData {
  19. domEvent: any
  20. }
  21. export interface OnSelectData extends OnOpenChangeData {
  22. domEvent: any
  23. }
  24. export interface SubNavAdapter<P = Record<string, any>, S = Record<string, any>> extends DefaultAdapter<P, S> {
  25. updateIsHovered(isHovered: boolean): void;
  26. getOpenKeys(): (string | number)[];
  27. getOpenKeysIsControlled(): boolean;
  28. getCanUpdateOpenKeys(): boolean;
  29. updateOpen(isOpen: boolean): void;
  30. notifyGlobalOpenChange(data: OnOpenChangeData): void;
  31. notifyGlobalOnSelect(data: OnSelectData): void;
  32. notifyGlobalOnClick(data: OnClickData): void;
  33. getIsSelected(itemKey: string | number): boolean;
  34. getIsOpen(): boolean
  35. }
  36. export default class SubNavFoundation<P = Record<string, any>, S = Record<string, any>> extends BaseFoundation<SubNavAdapter<P, S>, P, S> {
  37. constructor(adapter: SubNavAdapter<P, S>) {
  38. super({ ...adapter });
  39. }
  40. _timer: number;
  41. init() {
  42. // this.log('invoke SubNavFoundation init()');
  43. this._timer = null;
  44. }
  45. destroy() {}
  46. clearDelayTimer() {
  47. if (this._timer) {
  48. clearTimeout(this._timer);
  49. this._timer = null;
  50. }
  51. }
  52. isValidKey(itemKey: string | number) {
  53. return itemKey != null && (typeof itemKey === 'number' || typeof itemKey === 'string');
  54. }
  55. handleDropdownVisibleChange(visible: boolean) {
  56. const itemKey = this.getProp('itemKey');
  57. const openKeysIsControlled = this._adapter.getOpenKeysIsControlled();
  58. const canUpdateOpenKeys = this._adapter.getCanUpdateOpenKeys();
  59. const rawOpenKeys = this._adapter.getOpenKeys();
  60. const openKeys = visible ? addKeys(rawOpenKeys, itemKey) : removeKeys(rawOpenKeys, itemKey);
  61. this.clearDelayTimer();
  62. if (!openKeysIsControlled) {
  63. if (canUpdateOpenKeys) {
  64. this._adapter.updateOpen(visible);
  65. }
  66. // this._adapter.updateIsHovered(visible);
  67. }
  68. this._adapter.notifyGlobalOpenChange({ itemKey, openKeys, isOpen: visible });
  69. }
  70. /**
  71. *
  72. * @param {Event} e
  73. * @param {HTMLElement} titleRef
  74. */
  75. handleClick(e: any, titleRef: any) {
  76. const { itemKey, disabled } = this.getProps();
  77. if (disabled) {
  78. return;
  79. }
  80. // this.log(e, titleRef, titleRef.contains(e.target));
  81. const clickedDomIsTitle = titleRef && titleRef.contains(e.target);
  82. let isOpen = Boolean(this._adapter.getIsOpen());
  83. if (!clickedDomIsTitle) {
  84. isOpen = false;
  85. } else {
  86. isOpen = !isOpen;
  87. }
  88. const openKeys = isOpen
  89. ? addKeys(this._adapter.getOpenKeys(), itemKey)
  90. : removeKeys(this._adapter.getOpenKeys(), itemKey);
  91. const cbVal = { itemKey, openKeys, isOpen, domEvent: e };
  92. const openKeysIsControlled = this._adapter.getOpenKeysIsControlled();
  93. const canUpdateOpenKeys = this._adapter.getCanUpdateOpenKeys();
  94. if (!openKeysIsControlled && canUpdateOpenKeys) {
  95. this._adapter.updateOpen(isOpen);
  96. }
  97. this._adapter.notifyGlobalOpenChange(cbVal);
  98. this._adapter.notifyGlobalOnClick(cbVal);
  99. }
  100. /**
  101. * A11y: simulate sub nav click
  102. * @param e
  103. * @param titleRef
  104. */
  105. handleKeyPress(e: any, titleRef: any) {
  106. if (isEnterPress(e)) {
  107. this.handleClick(e, titleRef);
  108. }
  109. }
  110. }