subNavFoundation.ts 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136
  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. this.clearDelayTimer();
  47. }
  48. clearDelayTimer() {
  49. if (this._timer) {
  50. clearTimeout(this._timer);
  51. this._timer = null;
  52. }
  53. }
  54. isValidKey(itemKey: string | number) {
  55. return itemKey != null && (typeof itemKey === 'number' || typeof itemKey === 'string');
  56. }
  57. handleDropdownVisibleChange(visible: boolean) {
  58. const itemKey = this.getProp('itemKey');
  59. const openKeysIsControlled = this._adapter.getOpenKeysIsControlled();
  60. const canUpdateOpenKeys = this._adapter.getCanUpdateOpenKeys();
  61. const rawOpenKeys = this._adapter.getOpenKeys();
  62. const openKeys = visible ? addKeys(rawOpenKeys, itemKey) : removeKeys(rawOpenKeys, itemKey);
  63. this.clearDelayTimer();
  64. if (!openKeysIsControlled) {
  65. if (canUpdateOpenKeys) {
  66. this._adapter.updateOpen(visible);
  67. }
  68. // this._adapter.updateIsHovered(visible);
  69. }
  70. this._adapter.notifyGlobalOpenChange({ itemKey, openKeys, isOpen: visible });
  71. }
  72. /**
  73. *
  74. * @param {Event} e
  75. * @param {HTMLElement} titleRef
  76. */
  77. handleClick(e: any, titleRef: any) {
  78. const { itemKey, disabled } = this.getProps();
  79. if (disabled) {
  80. return;
  81. }
  82. // this.log(e, titleRef, titleRef.contains(e.target));
  83. const clickedDomIsTitle = titleRef && titleRef.contains(e.target);
  84. let isOpen = Boolean(this._adapter.getIsOpen());
  85. if (!clickedDomIsTitle) {
  86. isOpen = false;
  87. } else {
  88. isOpen = !isOpen;
  89. }
  90. const openKeys = isOpen
  91. ? addKeys(this._adapter.getOpenKeys(), itemKey)
  92. : removeKeys(this._adapter.getOpenKeys(), itemKey);
  93. const cbVal = { itemKey, openKeys, isOpen, domEvent: e };
  94. const openKeysIsControlled = this._adapter.getOpenKeysIsControlled();
  95. const canUpdateOpenKeys = this._adapter.getCanUpdateOpenKeys();
  96. if (!openKeysIsControlled && canUpdateOpenKeys) {
  97. this._adapter.updateOpen(isOpen);
  98. }
  99. this._adapter.notifyGlobalOpenChange(cbVal);
  100. this._adapter.notifyGlobalOnClick(cbVal);
  101. }
  102. /**
  103. * A11y: simulate sub nav click
  104. * @param e
  105. * @param titleRef
  106. */
  107. handleKeyPress(e: any, titleRef: any) {
  108. if (isEnterPress(e)) {
  109. this.handleClick(e, titleRef);
  110. }
  111. }
  112. }