foundation.ts 9.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268
  1. import BaseFoundation, { DefaultAdapter } from '../base/foundation';
  2. import NavItem from './NavItem';
  3. import { ItemProps, ItemKey } from './itemFoundation';
  4. import { strings } from './constants';
  5. import { get } from 'lodash';
  6. import isNullOrUndefined from '../utils/isNullOrUndefined';
  7. export interface ItemKey2ParentKeysMap {
  8. [key: string]: (string | number)[]
  9. }
  10. export interface OnClickData {
  11. itemKey: ItemKey;
  12. domEvent: any;
  13. isOpen: boolean
  14. }
  15. export interface OnSelectData extends OnClickData {
  16. selectedKeys: ItemKey[];
  17. selectedItems: ItemProps[]
  18. }
  19. export interface OnOpenChangeData extends OnClickData {
  20. openKeys: ItemKey[]
  21. }
  22. export interface NavItemType {
  23. props?: ItemProps;
  24. items?: NavItemType[];
  25. [key: string]: any
  26. }
  27. export interface NavigationAdapter<P = Record<string, any>, S = Record<string, any>> extends DefaultAdapter<P, S> {
  28. notifySelect(data: OnSelectData): void;
  29. notifyOpenChange(data: OnOpenChangeData): void;
  30. setIsCollapsed(isCollapsed: boolean): void;
  31. notifyCollapseChange(isCollapsed: boolean): void;
  32. updateItems(items: ItemProps[]): void;
  33. setItemKeysMap(map: { [key: string]: (string | number)[] }): void;
  34. addSelectedKeys(...keys: (string | number)[]): void;
  35. removeSelectedKeys(...keys: (string | number)[]): void;
  36. updateSelectedKeys(keys: (string | number)[], includeParentKeys?: boolean): void;
  37. updateOpenKeys(keys: (string | number)[]): void;
  38. addOpenKeys(...keys: (string | number)[]): void;
  39. removeOpenKeys(...keys: (string | number)[]): void;
  40. setItemsChanged(isChanged: boolean): void
  41. }
  42. export default class NavigationFoundation<P = Record<string, any>, S = Record<string, any>> extends BaseFoundation<NavigationAdapter<P, S>, P, S> {
  43. constructor(adapter: NavigationAdapter<P, S>) {
  44. super({ ...adapter });
  45. }
  46. /* istanbul ignore next */
  47. static getZeroParentKeys(itemKeysMap = {}, ...itemKeys: ItemKey[]) {
  48. const willAddKeys = [];
  49. if (itemKeys.length) {
  50. for (const itemKey of itemKeys) {
  51. if (Array.isArray(itemKeysMap[itemKey]) && itemKeysMap[itemKey].length) {
  52. const levelZeroParentKey = itemKeysMap[itemKey][0];
  53. if (!isNullOrUndefined(levelZeroParentKey)) {
  54. willAddKeys.push(levelZeroParentKey);
  55. }
  56. }
  57. }
  58. }
  59. return willAddKeys;
  60. }
  61. static buildItemKeysMap(items: NavItemType[] = [], keysMap = {}, parentKeys: (string | number)[] = [], keyPropName = 'itemKey') {
  62. if (Array.isArray(items) && items.length) {
  63. for (const item of items) {
  64. if (Array.isArray(item)) {
  65. NavigationFoundation.buildItemKeysMap(item, keysMap, [...parentKeys], keyPropName);
  66. } else {
  67. let itemKey;
  68. if (item && typeof item === 'object') {
  69. itemKey = item[keyPropName] || (item.props && item.props[keyPropName]);
  70. }
  71. if (itemKey) {
  72. keysMap[itemKey] = [...parentKeys];
  73. if (Array.isArray(item.items) && item.items.length) {
  74. NavigationFoundation.buildItemKeysMap(
  75. item.items,
  76. keysMap,
  77. [...parentKeys, itemKey],
  78. keyPropName
  79. );
  80. } else if (item.props && item.props.children) {
  81. const children = Array.isArray(item.props.children)
  82. ? item.props.children
  83. : [item.props.children];
  84. NavigationFoundation.buildItemKeysMap(
  85. children,
  86. keysMap,
  87. [...parentKeys, itemKey],
  88. keyPropName
  89. );
  90. }
  91. }
  92. }
  93. }
  94. }
  95. return keysMap;
  96. }
  97. /**
  98. * init is called in constructor and componentDidMount.
  99. * if you want to update state in constructor, please add it to return object;
  100. * if you want to update state in componentDidMount, please call adapter in else logic.
  101. * @param {*} lifecycle
  102. * @returns
  103. */
  104. init(lifecycle?: string) {
  105. const { defaultSelectedKeys, selectedKeys } = this.getProps();
  106. let willSelectedKeys = selectedKeys || defaultSelectedKeys || [];
  107. const { itemKeysMap, willOpenKeys, formattedItems } = this.getCalcState();
  108. const parentSelectKeys = this.selectLevelZeroParentKeys(itemKeysMap, willSelectedKeys);
  109. willSelectedKeys = willSelectedKeys.concat(parentSelectKeys);
  110. if (lifecycle === 'constructor') {
  111. return {
  112. selectedKeys: willSelectedKeys,
  113. itemKeysMap,
  114. openKeys: willOpenKeys,
  115. items: formattedItems,
  116. };
  117. } else {
  118. // already include parentSelectKeys, set second parameter to false
  119. this._adapter.updateSelectedKeys(willSelectedKeys, false);
  120. this._adapter.setItemKeysMap(itemKeysMap);
  121. this._adapter.updateOpenKeys(willOpenKeys);
  122. this._adapter.updateItems(formattedItems);
  123. this._adapter.setItemsChanged(true);
  124. }
  125. return undefined;
  126. }
  127. /**
  128. * Get the state to be calculated
  129. */
  130. getCalcState() {
  131. const { itemKeysMap, formattedItems } = this.getFormattedItems();
  132. const willOpenKeys = this.getWillOpenKeys(itemKeysMap);
  133. return { itemKeysMap, willOpenKeys, formattedItems };
  134. }
  135. /**
  136. * Calculate formatted items and itemsKeyMap
  137. */
  138. getFormattedItems() {
  139. const { items, children } = this.getProps();
  140. const formattedItems = this.formatItems(items);
  141. const willHandleItems = Array.isArray(items) && items.length ? formattedItems : children;
  142. const itemKeysMap = NavigationFoundation.buildItemKeysMap(willHandleItems);
  143. return {
  144. itemKeysMap,
  145. formattedItems
  146. };
  147. }
  148. /**
  149. * Calculate the keys that will need to be opened soon
  150. * @param {*} itemKeysMap
  151. */
  152. getWillOpenKeys(itemKeysMap: ItemKey2ParentKeysMap) {
  153. const { defaultOpenKeys, openKeys, defaultSelectedKeys, selectedKeys, mode } = this.getProps();
  154. const { openKeys: stateOpenKeys = [] } = this.getStates();
  155. let willOpenKeys = openKeys || defaultOpenKeys || [];
  156. if (
  157. !(Array.isArray(defaultOpenKeys) ||
  158. Array.isArray(openKeys)) && mode === strings.MODE_VERTICAL && (Array.isArray(defaultSelectedKeys) || Array.isArray(selectedKeys))
  159. ) {
  160. const currentSelectedKeys = Array.isArray(selectedKeys) ? selectedKeys : defaultSelectedKeys;
  161. willOpenKeys = stateOpenKeys.concat(this.getShouldOpenKeys(itemKeysMap, currentSelectedKeys));
  162. willOpenKeys = Array.from(new Set(willOpenKeys));
  163. }
  164. return [...willOpenKeys];
  165. }
  166. getShouldOpenKeys(itemKeysMap: ItemKey2ParentKeysMap = {}, selectedKeys: ItemKey[] = []) {
  167. const willOpenKeySet = new Set();
  168. if (Array.isArray(selectedKeys) && selectedKeys.length) {
  169. selectedKeys.forEach(item => {
  170. if (item) {
  171. const parentKeys = get(itemKeysMap, item);
  172. if (Array.isArray(parentKeys)) {
  173. parentKeys.forEach(k => willOpenKeySet.add(k));
  174. }
  175. }
  176. });
  177. }
  178. return [...willOpenKeySet];
  179. }
  180. destroy() {}
  181. selectLevelZeroParentKeys(itemKeysMap: ItemKey2ParentKeysMap, itemKeys: ItemKey[]) {
  182. const _itemKeysMap = isNullOrUndefined(itemKeysMap) ? this.getState('itemKeysMap') : itemKeysMap;
  183. // console.log(itemKeysMap);
  184. const willAddKeys = [];
  185. if (itemKeys.length) {
  186. for (const itemKey of itemKeys) {
  187. if (Array.isArray(_itemKeysMap[itemKey]) && _itemKeysMap[itemKey].length) {
  188. const levelZeroParentKey = _itemKeysMap[itemKey][0];
  189. if (!isNullOrUndefined(levelZeroParentKey)) {
  190. willAddKeys.push(levelZeroParentKey);
  191. }
  192. }
  193. }
  194. }
  195. if (willAddKeys.length) {
  196. return willAddKeys;
  197. }
  198. return [];
  199. }
  200. formatItems(items: ItemProps[] = []) {
  201. const formattedItems = [];
  202. for (const item of items) {
  203. formattedItems.push(new NavItem(item));
  204. }
  205. return formattedItems;
  206. }
  207. handleSelect(data: OnSelectData) {
  208. this._adapter.notifySelect(data);
  209. }
  210. /* istanbul ignore next */
  211. judgeIfOpen(openKeys: ItemKey[], items: NavItemType[]): boolean {
  212. let shouldBeOpen = false;
  213. const _openKeys = Array.isArray(openKeys) ? openKeys : openKeys && [openKeys];
  214. if (_openKeys && Array.isArray(items) && items.length) {
  215. for (const item of items) {
  216. shouldBeOpen = _openKeys.includes(item.itemKey) || this.judgeIfOpen(_openKeys, item.items);
  217. if (shouldBeOpen) {
  218. break;
  219. }
  220. }
  221. }
  222. return shouldBeOpen;
  223. }
  224. handleCollapseChange() {
  225. const isCollapsed = !this.getState('isCollapsed');
  226. if (!this._isControlledComponent('isCollapsed')) {
  227. this._adapter.setIsCollapsed(isCollapsed);
  228. }
  229. this._adapter.notifyCollapseChange(isCollapsed);
  230. }
  231. handleItemsChange(isChanged: boolean) {
  232. this._adapter.setItemsChanged(isChanged);
  233. }
  234. }