foundation.ts 9.8 KB

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