@headless-tree/core 0.0.1 → 0.0.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (115) hide show
  1. package/CHANGELOG.md +7 -0
  2. package/lib/{features → cjs/features}/drag-and-drop/feature.js +0 -46
  3. package/lib/{features → cjs/features}/drag-and-drop/types.d.ts +1 -1
  4. package/lib/{features → cjs/features}/drag-and-drop/utils.d.ts +0 -1
  5. package/lib/{features → cjs/features}/drag-and-drop/utils.js +1 -60
  6. package/lib/{features → cjs/features}/hotkeys-core/feature.js +0 -1
  7. package/lib/{features → cjs/features}/tree/feature.js +0 -2
  8. package/lib/esm/core/create-tree.d.ts +2 -0
  9. package/lib/esm/core/create-tree.js +112 -0
  10. package/lib/esm/data-adapters/nested-data-adapter.d.ts +9 -0
  11. package/lib/esm/data-adapters/nested-data-adapter.js +28 -0
  12. package/lib/esm/data-adapters/types.d.ts +7 -0
  13. package/lib/esm/data-adapters/types.js +1 -0
  14. package/lib/esm/features/async-data-loader/feature.d.ts +5 -0
  15. package/lib/esm/features/async-data-loader/feature.js +77 -0
  16. package/lib/esm/features/async-data-loader/types.d.ts +41 -0
  17. package/lib/esm/features/async-data-loader/types.js +1 -0
  18. package/lib/esm/features/drag-and-drop/feature.d.ts +3 -0
  19. package/lib/esm/features/drag-and-drop/feature.js +95 -0
  20. package/lib/esm/features/drag-and-drop/types.d.ts +49 -0
  21. package/lib/esm/features/drag-and-drop/types.js +6 -0
  22. package/lib/esm/features/drag-and-drop/utils.d.ts +6 -0
  23. package/lib/esm/features/drag-and-drop/utils.js +55 -0
  24. package/lib/esm/features/expand-all/feature.d.ts +6 -0
  25. package/lib/esm/features/expand-all/feature.js +36 -0
  26. package/lib/esm/features/expand-all/types.d.ts +17 -0
  27. package/lib/esm/features/expand-all/types.js +1 -0
  28. package/lib/esm/features/hotkeys-core/feature.d.ts +4 -0
  29. package/lib/esm/features/hotkeys-core/feature.js +69 -0
  30. package/lib/esm/features/hotkeys-core/types.d.ts +25 -0
  31. package/lib/esm/features/hotkeys-core/types.js +1 -0
  32. package/lib/esm/features/main/types.d.ts +36 -0
  33. package/lib/esm/features/main/types.js +1 -0
  34. package/lib/esm/features/renaming/feature.d.ts +5 -0
  35. package/lib/esm/features/renaming/feature.js +62 -0
  36. package/lib/esm/features/renaming/types.d.ts +27 -0
  37. package/lib/esm/features/renaming/types.js +1 -0
  38. package/lib/esm/features/search/feature.d.ts +5 -0
  39. package/lib/esm/features/search/feature.js +87 -0
  40. package/lib/esm/features/search/types.d.ts +33 -0
  41. package/lib/esm/features/search/types.js +1 -0
  42. package/lib/esm/features/selection/feature.d.ts +5 -0
  43. package/lib/esm/features/selection/feature.js +109 -0
  44. package/lib/esm/features/selection/types.d.ts +21 -0
  45. package/lib/esm/features/selection/types.js +1 -0
  46. package/lib/esm/features/sync-data-loader/feature.d.ts +4 -0
  47. package/lib/esm/features/sync-data-loader/feature.js +6 -0
  48. package/lib/esm/features/sync-data-loader/types.d.ts +19 -0
  49. package/lib/esm/features/sync-data-loader/types.js +1 -0
  50. package/lib/esm/features/tree/feature.d.ts +6 -0
  51. package/lib/esm/features/tree/feature.js +211 -0
  52. package/lib/esm/features/tree/types.d.ts +57 -0
  53. package/lib/esm/features/tree/types.js +1 -0
  54. package/lib/esm/index.d.ts +21 -0
  55. package/lib/esm/index.js +21 -0
  56. package/lib/esm/mddocs-entry.d.ts +21 -0
  57. package/lib/esm/mddocs-entry.js +1 -0
  58. package/lib/esm/types/core.d.ts +68 -0
  59. package/lib/esm/types/core.js +1 -0
  60. package/lib/esm/types/deep-merge.d.ts +13 -0
  61. package/lib/esm/types/deep-merge.js +1 -0
  62. package/lib/esm/utils.d.ts +9 -0
  63. package/lib/esm/utils.js +96 -0
  64. package/package.json +11 -4
  65. package/tsconfig.json +7 -7
  66. /package/lib/{core → cjs/core}/create-tree.d.ts +0 -0
  67. /package/lib/{core → cjs/core}/create-tree.js +0 -0
  68. /package/lib/{data-adapters → cjs/data-adapters}/nested-data-adapter.d.ts +0 -0
  69. /package/lib/{data-adapters → cjs/data-adapters}/nested-data-adapter.js +0 -0
  70. /package/lib/{data-adapters → cjs/data-adapters}/types.d.ts +0 -0
  71. /package/lib/{data-adapters → cjs/data-adapters}/types.js +0 -0
  72. /package/lib/{features → cjs/features}/async-data-loader/feature.d.ts +0 -0
  73. /package/lib/{features → cjs/features}/async-data-loader/feature.js +0 -0
  74. /package/lib/{features → cjs/features}/async-data-loader/types.d.ts +0 -0
  75. /package/lib/{features → cjs/features}/async-data-loader/types.js +0 -0
  76. /package/lib/{features → cjs/features}/drag-and-drop/feature.d.ts +0 -0
  77. /package/lib/{features → cjs/features}/drag-and-drop/types.js +0 -0
  78. /package/lib/{features → cjs/features}/expand-all/feature.d.ts +0 -0
  79. /package/lib/{features → cjs/features}/expand-all/feature.js +0 -0
  80. /package/lib/{features → cjs/features}/expand-all/types.d.ts +0 -0
  81. /package/lib/{features → cjs/features}/expand-all/types.js +0 -0
  82. /package/lib/{features → cjs/features}/hotkeys-core/feature.d.ts +0 -0
  83. /package/lib/{features → cjs/features}/hotkeys-core/types.d.ts +0 -0
  84. /package/lib/{features → cjs/features}/hotkeys-core/types.js +0 -0
  85. /package/lib/{features → cjs/features}/main/types.d.ts +0 -0
  86. /package/lib/{features → cjs/features}/main/types.js +0 -0
  87. /package/lib/{features → cjs/features}/renaming/feature.d.ts +0 -0
  88. /package/lib/{features → cjs/features}/renaming/feature.js +0 -0
  89. /package/lib/{features → cjs/features}/renaming/types.d.ts +0 -0
  90. /package/lib/{features → cjs/features}/renaming/types.js +0 -0
  91. /package/lib/{features → cjs/features}/search/feature.d.ts +0 -0
  92. /package/lib/{features → cjs/features}/search/feature.js +0 -0
  93. /package/lib/{features → cjs/features}/search/types.d.ts +0 -0
  94. /package/lib/{features → cjs/features}/search/types.js +0 -0
  95. /package/lib/{features → cjs/features}/selection/feature.d.ts +0 -0
  96. /package/lib/{features → cjs/features}/selection/feature.js +0 -0
  97. /package/lib/{features → cjs/features}/selection/types.d.ts +0 -0
  98. /package/lib/{features → cjs/features}/selection/types.js +0 -0
  99. /package/lib/{features → cjs/features}/sync-data-loader/feature.d.ts +0 -0
  100. /package/lib/{features → cjs/features}/sync-data-loader/feature.js +0 -0
  101. /package/lib/{features → cjs/features}/sync-data-loader/types.d.ts +0 -0
  102. /package/lib/{features → cjs/features}/sync-data-loader/types.js +0 -0
  103. /package/lib/{features → cjs/features}/tree/feature.d.ts +0 -0
  104. /package/lib/{features → cjs/features}/tree/types.d.ts +0 -0
  105. /package/lib/{features → cjs/features}/tree/types.js +0 -0
  106. /package/lib/{index.d.ts → cjs/index.d.ts} +0 -0
  107. /package/lib/{index.js → cjs/index.js} +0 -0
  108. /package/lib/{mddocs-entry.d.ts → cjs/mddocs-entry.d.ts} +0 -0
  109. /package/lib/{mddocs-entry.js → cjs/mddocs-entry.js} +0 -0
  110. /package/lib/{types → cjs/types}/core.d.ts +0 -0
  111. /package/lib/{types → cjs/types}/core.js +0 -0
  112. /package/lib/{types → cjs/types}/deep-merge.d.ts +0 -0
  113. /package/lib/{types → cjs/types}/deep-merge.js +0 -0
  114. /package/lib/{utils.d.ts → cjs/utils.d.ts} +0 -0
  115. /package/lib/{utils.js → cjs/utils.js} +0 -0
@@ -0,0 +1,55 @@
1
+ import { DropTargetPosition } from "./types";
2
+ export const getDragCode = ({ item, childIndex }) => `${item.getId()}__${childIndex !== null && childIndex !== void 0 ? childIndex : "none"}`;
3
+ export const getDropOffset = (e, item) => {
4
+ var _a;
5
+ const bb = (_a = item.getElement()) === null || _a === void 0 ? void 0 : _a.getBoundingClientRect();
6
+ return bb ? (e.pageY - bb.top) / bb.height : 0.5;
7
+ };
8
+ export const canDrop = (dataTransfer, target, tree) => {
9
+ var _a, _b, _c, _d;
10
+ const draggedItems = (_a = tree.getState().dnd) === null || _a === void 0 ? void 0 : _a.draggedItems;
11
+ const config = tree.getConfig();
12
+ if (draggedItems && !((_c = (_b = config.canDrop) === null || _b === void 0 ? void 0 : _b.call(config, draggedItems, target)) !== null && _c !== void 0 ? _c : true)) {
13
+ return false;
14
+ }
15
+ if (!draggedItems &&
16
+ dataTransfer &&
17
+ !((_d = config.canDropForeignDragObject) === null || _d === void 0 ? void 0 : _d.call(config, dataTransfer, target))) {
18
+ return false;
19
+ }
20
+ return true;
21
+ };
22
+ const getDropTargetPosition = (offset, topLinePercentage, bottomLinePercentage) => {
23
+ if (offset < topLinePercentage) {
24
+ return DropTargetPosition.Top;
25
+ }
26
+ if (offset > bottomLinePercentage) {
27
+ return DropTargetPosition.Bottom;
28
+ }
29
+ return DropTargetPosition.Item;
30
+ };
31
+ export const getDropTarget = (e, item, tree) => {
32
+ var _a, _b;
33
+ const config = tree.getConfig();
34
+ const offset = getDropOffset(e, item);
35
+ const dropOnItemTarget = { item, childIndex: null };
36
+ const pos = getDropTargetPosition(offset, (_a = config.topLinePercentage) !== null && _a !== void 0 ? _a : 0.3, (_b = config.bottomLinePercentage) !== null && _b !== void 0 ? _b : 0.7);
37
+ const inbetweenPos = getDropTargetPosition(offset, 0.5, 0.5);
38
+ if (!config.canDropInbetween) {
39
+ return dropOnItemTarget;
40
+ }
41
+ if (!canDrop(e.dataTransfer, dropOnItemTarget, tree)) {
42
+ return {
43
+ item: item.getParent(),
44
+ childIndex: item.getIndexInParent() +
45
+ (inbetweenPos === DropTargetPosition.Top ? 0 : 1),
46
+ };
47
+ }
48
+ if (pos === DropTargetPosition.Item) {
49
+ return dropOnItemTarget;
50
+ }
51
+ return {
52
+ item: item.getParent(),
53
+ childIndex: item.getIndexInParent() + (pos === DropTargetPosition.Top ? 0 : 1),
54
+ };
55
+ };
@@ -0,0 +1,6 @@
1
+ import { FeatureImplementation } from "../../types/core";
2
+ import { ExpandAllFeatureDef } from "./types";
3
+ import { MainFeatureDef } from "../main/types";
4
+ import { TreeFeatureDef } from "../tree/types";
5
+ import { SyncDataLoaderFeatureDef } from "../sync-data-loader/types";
6
+ export declare const expandAllFeature: FeatureImplementation<any, ExpandAllFeatureDef, MainFeatureDef | TreeFeatureDef<any> | SyncDataLoaderFeatureDef<any> | ExpandAllFeatureDef>;
@@ -0,0 +1,36 @@
1
+ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
2
+ function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
3
+ return new (P || (P = Promise))(function (resolve, reject) {
4
+ function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
5
+ function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
6
+ function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
7
+ step((generator = generator.apply(thisArg, _arguments || [])).next());
8
+ });
9
+ };
10
+ import { poll } from "../../utils";
11
+ export const expandAllFeature = {
12
+ key: "expand-all",
13
+ dependingFeatures: ["main", "tree"],
14
+ createTreeInstance: (prev, tree) => (Object.assign(Object.assign({}, prev), { expandAll: (cancelToken) => __awaiter(void 0, void 0, void 0, function* () {
15
+ yield Promise.all(tree.getItems().map((item) => item.expandAll(cancelToken)));
16
+ }), collapseAll: () => __awaiter(void 0, void 0, void 0, function* () {
17
+ var _a, _b;
18
+ (_b = (_a = tree.getConfig()).setExpandedItems) === null || _b === void 0 ? void 0 : _b.call(_a, []);
19
+ }) })),
20
+ createItemInstance: (prev, item, tree) => (Object.assign(Object.assign({}, prev), { expandAll: (cancelToken) => __awaiter(void 0, void 0, void 0, function* () {
21
+ if (cancelToken === null || cancelToken === void 0 ? void 0 : cancelToken.current) {
22
+ return;
23
+ }
24
+ item.expand();
25
+ yield poll(() => !tree.getState().loadingItems.includes(item.getId()));
26
+ yield Promise.all(item.getChildren().map((child) => __awaiter(void 0, void 0, void 0, function* () {
27
+ yield poll(() => !tree.getState().loadingItems.includes(child.getId()));
28
+ yield (child === null || child === void 0 ? void 0 : child.expandAll(cancelToken));
29
+ })));
30
+ }), collapseAll: () => __awaiter(void 0, void 0, void 0, function* () {
31
+ yield Promise.all(item.getChildren().map((child) => __awaiter(void 0, void 0, void 0, function* () {
32
+ yield (child === null || child === void 0 ? void 0 : child.collapseAll());
33
+ })));
34
+ item.collapse();
35
+ }) })),
36
+ };
@@ -0,0 +1,17 @@
1
+ export type ExpandAllFeatureDef = {
2
+ state: {};
3
+ config: {};
4
+ treeInstance: {
5
+ expandAll: (cancelToken?: {
6
+ current: boolean;
7
+ }) => Promise<void>;
8
+ collapseAll: () => Promise<void>;
9
+ };
10
+ itemInstance: {
11
+ expandAll: (cancelToken?: {
12
+ current: boolean;
13
+ }) => Promise<void>;
14
+ collapseAll: () => Promise<void>;
15
+ };
16
+ hotkeys: never;
17
+ };
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,4 @@
1
+ import { FeatureImplementation } from "../../types/core";
2
+ import { HotkeysCoreFeatureDef } from "./types";
3
+ import { MainFeatureDef } from "../main/types";
4
+ export declare const hotkeysCoreFeature: FeatureImplementation<any, HotkeysCoreFeatureDef<any>, MainFeatureDef | HotkeysCoreFeatureDef<any>>;
@@ -0,0 +1,69 @@
1
+ const specialKeys = {
2
+ Letter: /^[a-z]$/,
3
+ LetterOrNumber: /^[a-z0-9]$/,
4
+ };
5
+ const testHotkeyMatch = (pressedKeys, tree, hotkey) => {
6
+ const supposedKeys = hotkey.hotkey.split("+");
7
+ const doKeysMatch = supposedKeys.every((key) => key in specialKeys
8
+ ? [...pressedKeys].some((pressedKey) => specialKeys[key].test(pressedKey))
9
+ : pressedKeys.has(key));
10
+ const isEnabled = !hotkey.isEnabled || hotkey.isEnabled(tree);
11
+ const equalCounts = pressedKeys.size === supposedKeys.length;
12
+ return doKeysMatch && isEnabled && equalCounts;
13
+ };
14
+ const findHotkeyMatch = (pressedKeys, tree, config1, config2) => {
15
+ var _a;
16
+ return (_a = Object.entries(Object.assign(Object.assign({}, config1), config2)).find(([, hotkey]) => testHotkeyMatch(pressedKeys, tree, hotkey))) === null || _a === void 0 ? void 0 : _a[0];
17
+ };
18
+ export const hotkeysCoreFeature = {
19
+ key: "hotkeys-core",
20
+ dependingFeatures: ["main", "tree"],
21
+ onTreeMount: (tree, element) => {
22
+ const data = tree.getDataRef();
23
+ const keydown = (e) => {
24
+ var _a, _b;
25
+ var _c;
26
+ (_a = (_c = data.current).pressedKeys) !== null && _a !== void 0 ? _a : (_c.pressedKeys = new Set());
27
+ const newMatch = !data.current.pressedKeys.has(e.key);
28
+ data.current.pressedKeys.add(e.key);
29
+ const hotkeyName = findHotkeyMatch(data.current.pressedKeys, tree, tree.getHotkeyPresets(), tree.getConfig().hotkeys);
30
+ if (!hotkeyName)
31
+ return;
32
+ const hotkeyConfig = Object.assign(Object.assign({}, tree.getHotkeyPresets()[hotkeyName]), (_b = tree.getConfig().hotkeys) === null || _b === void 0 ? void 0 : _b[hotkeyName]);
33
+ if (!hotkeyConfig)
34
+ return;
35
+ if (!hotkeyConfig.allowWhenInputFocused &&
36
+ e.target instanceof HTMLInputElement)
37
+ return;
38
+ if (!hotkeyConfig.canRepeat && !newMatch)
39
+ return;
40
+ if (hotkeyConfig.preventDefault)
41
+ e.preventDefault();
42
+ hotkeyConfig.handler(e, tree);
43
+ };
44
+ const keyup = (e) => {
45
+ var _a;
46
+ var _b;
47
+ (_a = (_b = data.current).pressedKeys) !== null && _a !== void 0 ? _a : (_b.pressedKeys = new Set());
48
+ data.current.pressedKeys.delete(e.key);
49
+ };
50
+ // keyup is registered on document, because some hotkeys shift
51
+ // the focus away from the tree (i.e. search)
52
+ // and then we wouldn't get the keyup event anymore
53
+ element.addEventListener("keydown", keydown);
54
+ document.addEventListener("keyup", keyup);
55
+ data.current.keydownHandler = keydown;
56
+ data.current.keyupHandler = keyup;
57
+ },
58
+ onTreeUnmount: (tree, element) => {
59
+ const data = tree.getDataRef();
60
+ if (data.current.keyupHandler) {
61
+ document.removeEventListener("keyup", data.current.keyupHandler);
62
+ delete data.current.keyupHandler;
63
+ }
64
+ if (data.current.keydownHandler) {
65
+ element.removeEventListener("keydown", data.current.keydownHandler);
66
+ delete data.current.keydownHandler;
67
+ }
68
+ },
69
+ };
@@ -0,0 +1,25 @@
1
+ import { CustomHotkeysConfig, ItemInstance, TreeInstance } from "../../types/core";
2
+ export interface HotkeyConfig<T> {
3
+ hotkey: string;
4
+ canRepeat?: boolean;
5
+ allowWhenInputFocused?: boolean;
6
+ isEnabled?: (tree: TreeInstance<T>) => boolean;
7
+ preventDefault?: boolean;
8
+ handler: (e: KeyboardEvent, tree: TreeInstance<T>) => void;
9
+ }
10
+ export type HotkeysCoreDataRef = {
11
+ keydownHandler?: (e: KeyboardEvent) => void;
12
+ keyupHandler?: (e: KeyboardEvent) => void;
13
+ pressedKeys: Set<string>;
14
+ };
15
+ export type HotkeysCoreFeatureDef<T> = {
16
+ state: {};
17
+ config: {
18
+ hotkeys?: CustomHotkeysConfig<T>;
19
+ onTreeHotkey?: (name: string, element: HTMLElement) => void;
20
+ onItemHotkey?: (name: string, item: ItemInstance<T>, element: HTMLElement) => void;
21
+ };
22
+ treeInstance: {};
23
+ itemInstance: {};
24
+ hotkeys: never;
25
+ };
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,36 @@
1
+ import { FeatureImplementation, HotkeysConfig, ItemInstance, SetStateFn, TreeConfig, TreeState } from "../../types/core";
2
+ import { ItemMeta } from "../tree/types";
3
+ export type MainFeatureDef<T = any> = {
4
+ state: {};
5
+ config: {
6
+ features?: FeatureImplementation<any>[];
7
+ state?: Partial<TreeState<T>>;
8
+ setState?: SetStateFn<TreeState<T>>;
9
+ };
10
+ treeInstance: {
11
+ setState: SetStateFn<TreeState<T>>;
12
+ getState: () => TreeState<T>;
13
+ setConfig: SetStateFn<TreeConfig<T>>;
14
+ getConfig: () => TreeConfig<T>;
15
+ getItemInstance: (itemId: string) => ItemInstance<T>;
16
+ getItems: () => ItemInstance<T>[];
17
+ registerElement: (element: HTMLElement | null) => void;
18
+ getElement: () => HTMLElement | undefined | null;
19
+ /** @internal */
20
+ getDataRef: <D>() => {
21
+ current: D;
22
+ };
23
+ getHotkeyPresets: () => HotkeysConfig<T>;
24
+ rebuildTree: () => void;
25
+ };
26
+ itemInstance: {
27
+ registerElement: (element: HTMLElement | null) => void;
28
+ getItemMeta: () => ItemMeta;
29
+ getElement: () => HTMLElement | undefined | null;
30
+ /** @internal */
31
+ getDataRef: <D>() => {
32
+ current: D;
33
+ };
34
+ };
35
+ hotkeys: never;
36
+ };
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,5 @@
1
+ import { FeatureImplementation } from "../../types/core";
2
+ import { RenamingFeatureDef } from "./types";
3
+ import { MainFeatureDef } from "../main/types";
4
+ import { TreeFeatureDef } from "../tree/types";
5
+ export declare const renamingFeature: FeatureImplementation<any, RenamingFeatureDef<any>, MainFeatureDef | TreeFeatureDef<any> | RenamingFeatureDef<any>>;
@@ -0,0 +1,62 @@
1
+ import { makeStateUpdater } from "../../utils";
2
+ export const renamingFeature = {
3
+ key: "renaming",
4
+ dependingFeatures: ["main", "tree"],
5
+ getDefaultConfig: (defaultConfig, tree) => (Object.assign({ setRenamingItem: makeStateUpdater("renamingItem", tree), setRenamingValue: makeStateUpdater("renamingValue", tree), canRename: () => true }, defaultConfig)),
6
+ createTreeInstance: (prev, instance) => (Object.assign(Object.assign({}, prev), { startRenamingItem: (itemId) => {
7
+ var _a, _b;
8
+ const config = instance.getConfig();
9
+ const item = instance.getItemInstance(itemId);
10
+ if (!item.canRename()) {
11
+ return;
12
+ }
13
+ (_a = config.setRenamingItem) === null || _a === void 0 ? void 0 : _a.call(config, itemId);
14
+ (_b = config.setRenamingValue) === null || _b === void 0 ? void 0 : _b.call(config, item.getItemName());
15
+ }, getRenamingItem: () => {
16
+ const itemId = instance.getState().renamingItem;
17
+ return itemId ? instance.getItemInstance(itemId) : null;
18
+ }, getRenamingValue: () => instance.getState().renamingValue || "", abortRenaming: () => {
19
+ var _a, _b;
20
+ (_b = (_a = instance.getConfig()).setRenamingItem) === null || _b === void 0 ? void 0 : _b.call(_a, null);
21
+ }, completeRenaming: () => {
22
+ var _a, _b, _c;
23
+ const config = instance.getConfig();
24
+ const item = instance.getRenamingItem();
25
+ if (item) {
26
+ (_a = config.onRename) === null || _a === void 0 ? void 0 : _a.call(config, item, instance.getState().renamingValue || "");
27
+ }
28
+ (_c = (_b = instance.getConfig()).setRenamingItem) === null || _c === void 0 ? void 0 : _c.call(_b, null);
29
+ }, isRenamingItem: () => !!instance.getState().renamingItem })),
30
+ createItemInstance: (prev, instance, tree) => (Object.assign(Object.assign({}, prev), { getRenameInputProps: () => ({
31
+ onBlur: () => tree.abortRenaming(),
32
+ value: tree.getRenamingValue(),
33
+ onChange: (e) => {
34
+ var _a, _b;
35
+ (_b = (_a = tree.getConfig()).setRenamingValue) === null || _b === void 0 ? void 0 : _b.call(_a, e.target.value);
36
+ },
37
+ }), canRename: () => { var _a, _b, _c; return (_c = (_b = (_a = tree.getConfig()).canRename) === null || _b === void 0 ? void 0 : _b.call(_a, instance)) !== null && _c !== void 0 ? _c : true; }, isRenaming: () => instance.getId() === tree.getState().renamingItem })),
38
+ hotkeys: {
39
+ renameItem: {
40
+ hotkey: "F2",
41
+ handler: (e, tree) => {
42
+ tree.startRenamingItem(tree.getFocusedItem().getId());
43
+ },
44
+ },
45
+ abortRenaming: {
46
+ hotkey: "Escape",
47
+ allowWhenInputFocused: true,
48
+ isEnabled: (tree) => tree.isRenamingItem(),
49
+ handler: (e, tree) => {
50
+ tree.abortRenaming();
51
+ },
52
+ },
53
+ completeRenaming: {
54
+ hotkey: "Enter",
55
+ allowWhenInputFocused: true,
56
+ isEnabled: (tree) => tree.isRenamingItem(),
57
+ handler: (e, tree) => {
58
+ tree.completeRenaming();
59
+ },
60
+ },
61
+ },
62
+ };
@@ -0,0 +1,27 @@
1
+ import { ItemInstance, SetStateFn } from "../../types/core";
2
+ export type RenamingFeatureDef<T> = {
3
+ state: {
4
+ renamingItem?: string | null;
5
+ renamingValue?: string;
6
+ };
7
+ config: {
8
+ setRenamingItem?: SetStateFn<string | null>;
9
+ setRenamingValue?: SetStateFn<string | undefined>;
10
+ canRename?: (item: ItemInstance<T>) => boolean;
11
+ onRename?: (item: ItemInstance<T>, value: string) => void;
12
+ };
13
+ treeInstance: {
14
+ startRenamingItem: (itemId: string) => void;
15
+ getRenamingItem: () => ItemInstance<T> | null;
16
+ getRenamingValue: () => string;
17
+ abortRenaming: () => void;
18
+ completeRenaming: () => void;
19
+ isRenamingItem: () => boolean;
20
+ };
21
+ itemInstance: {
22
+ getRenameInputProps: () => any;
23
+ canRename: () => boolean;
24
+ isRenaming: () => boolean;
25
+ };
26
+ hotkeys: "renameItem" | "abortRenaming" | "completeRenaming";
27
+ };
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,5 @@
1
+ import { FeatureImplementation } from "../../types/core";
2
+ import { SearchFeatureDef } from "./types";
3
+ import { MainFeatureDef } from "../main/types";
4
+ import { TreeFeatureDef } from "../tree/types";
5
+ export declare const searchFeature: FeatureImplementation<any, SearchFeatureDef<any>, MainFeatureDef | TreeFeatureDef<any> | SearchFeatureDef<any>>;
@@ -0,0 +1,87 @@
1
+ import { makeStateUpdater, memo } from "../../utils";
2
+ export const searchFeature = {
3
+ key: "search",
4
+ dependingFeatures: ["main", "tree"],
5
+ getInitialState: (initialState) => (Object.assign({ search: null }, initialState)),
6
+ getDefaultConfig: (defaultConfig, tree) => (Object.assign({ setSearch: makeStateUpdater("search", tree), isSearchMatchingItem: (search, item) => search.length > 0 &&
7
+ item.getItemName().toLowerCase().includes(search.toLowerCase()) }, defaultConfig)),
8
+ createTreeInstance: (prev, instance) => (Object.assign(Object.assign({}, prev), { setSearch: (search) => {
9
+ var _a, _b, _c;
10
+ (_b = (_a = instance.getConfig()).setSearch) === null || _b === void 0 ? void 0 : _b.call(_a, search);
11
+ (_c = instance
12
+ .getItems()
13
+ .find((item) => {
14
+ var _a, _b;
15
+ return (_b = (_a = instance
16
+ .getConfig()).isSearchMatchingItem) === null || _b === void 0 ? void 0 : _b.call(_a, instance.getSearchValue(), item);
17
+ })) === null || _c === void 0 ? void 0 : _c.setFocused();
18
+ }, openSearch: (initialValue = "") => {
19
+ instance.setSearch(initialValue);
20
+ setTimeout(() => {
21
+ var _a;
22
+ (_a = instance
23
+ .getDataRef()
24
+ .current.searchInput) === null || _a === void 0 ? void 0 : _a.focus();
25
+ });
26
+ }, closeSearch: () => {
27
+ instance.setSearch(null);
28
+ instance.updateDomFocus();
29
+ }, isSearchOpen: () => instance.getState().search !== null, getSearchValue: () => instance.getState().search || "", registerSearchInputElement: (element) => {
30
+ const dataRef = instance.getDataRef();
31
+ dataRef.current.searchInput = element;
32
+ if (element && dataRef.current.keydownHandler) {
33
+ element.addEventListener("keydown", dataRef.current.keydownHandler);
34
+ }
35
+ }, getSearchInputElement: () => { var _a; return (_a = instance.getDataRef().current.searchInput) !== null && _a !== void 0 ? _a : null; }, getSearchInputElementProps: () => ({
36
+ value: instance.getSearchValue(),
37
+ onChange: (e) => instance.setSearch(e.target.value),
38
+ onBlur: () => instance.closeSearch(),
39
+ }), getSearchMatchingItems: memo((search, items) => items.filter((item) => { var _a, _b; return (_b = (_a = instance.getConfig()).isSearchMatchingItem) === null || _b === void 0 ? void 0 : _b.call(_a, search, item); }), () => [instance.getSearchValue(), instance.getItems()]) })),
40
+ createItemInstance: (prev, item, tree) => (Object.assign(Object.assign({}, prev), { isMatchingSearch: () => tree.getSearchMatchingItems().some((i) => i.getId() === item.getId()) })),
41
+ hotkeys: {
42
+ openSearch: {
43
+ hotkey: "LetterOrNumber",
44
+ preventDefault: true,
45
+ isEnabled: (tree) => !tree.isSearchOpen(),
46
+ handler: (e, tree) => {
47
+ e.stopPropagation();
48
+ tree.openSearch(e.key);
49
+ },
50
+ },
51
+ closeSearch: {
52
+ // TODO allow multiple, i.e. Enter
53
+ hotkey: "Escape",
54
+ allowWhenInputFocused: true,
55
+ isEnabled: (tree) => tree.isSearchOpen(),
56
+ handler: (e, tree) => {
57
+ tree.closeSearch();
58
+ },
59
+ },
60
+ nextSearchItem: {
61
+ hotkey: "ArrowDown",
62
+ allowWhenInputFocused: true,
63
+ isEnabled: (tree) => tree.isSearchOpen(),
64
+ handler: (e, tree) => {
65
+ // TODO scroll into view
66
+ const focusItem = tree
67
+ .getSearchMatchingItems()
68
+ .find((item) => item.getItemMeta().index >
69
+ tree.getFocusedItem().getItemMeta().index);
70
+ focusItem === null || focusItem === void 0 ? void 0 : focusItem.setFocused();
71
+ },
72
+ },
73
+ previousSearchItem: {
74
+ hotkey: "ArrowUp",
75
+ allowWhenInputFocused: true,
76
+ isEnabled: (tree) => tree.isSearchOpen(),
77
+ handler: (e, tree) => {
78
+ // TODO scroll into view
79
+ const focusItem = [...tree.getSearchMatchingItems()]
80
+ .reverse()
81
+ .find((item) => item.getItemMeta().index <
82
+ tree.getFocusedItem().getItemMeta().index);
83
+ focusItem === null || focusItem === void 0 ? void 0 : focusItem.setFocused();
84
+ },
85
+ },
86
+ },
87
+ };
@@ -0,0 +1,33 @@
1
+ import { ItemInstance, SetStateFn } from "../../types/core";
2
+ import { HotkeysCoreDataRef } from "../hotkeys-core/types";
3
+ export type SearchFeatureDataRef<T = any> = HotkeysCoreDataRef & {
4
+ matchingItems: ItemInstance<T>[];
5
+ searchInput: HTMLInputElement | null;
6
+ };
7
+ export type SearchFeatureDef<T> = {
8
+ state: {
9
+ search: string | null;
10
+ };
11
+ config: {
12
+ setSearch?: SetStateFn<string | null>;
13
+ onOpenSearch?: () => void;
14
+ onCloseSearch?: () => void;
15
+ onSearchMatchesItems?: (search: string, items: ItemInstance<T>[]) => void;
16
+ isSearchMatchingItem?: (search: string, item: ItemInstance<T>) => boolean;
17
+ };
18
+ treeInstance: {
19
+ setSearch: (search: string | null) => void;
20
+ openSearch: (initialValue?: string) => void;
21
+ closeSearch: () => void;
22
+ isSearchOpen: () => boolean;
23
+ getSearchValue: () => string;
24
+ registerSearchInputElement: (element: HTMLInputElement | null) => void;
25
+ getSearchInputElement: () => HTMLInputElement | null;
26
+ getSearchInputElementProps: () => any;
27
+ getSearchMatchingItems: () => ItemInstance<T>[];
28
+ };
29
+ itemInstance: {
30
+ isMatchingSearch: () => boolean;
31
+ };
32
+ hotkeys: "openSearch" | "closeSearch" | "nextSearchItem" | "previousSearchItem";
33
+ };
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,5 @@
1
+ import { FeatureImplementation } from "../../types/core";
2
+ import { SelectionFeatureDef } from "./types";
3
+ import { MainFeatureDef } from "../main/types";
4
+ import { TreeFeatureDef } from "../tree/types";
5
+ export declare const selectionFeature: FeatureImplementation<any, SelectionFeatureDef<any>, MainFeatureDef | TreeFeatureDef<any> | SelectionFeatureDef<any>>;
@@ -0,0 +1,109 @@
1
+ import { makeStateUpdater } from "../../utils";
2
+ export const selectionFeature = {
3
+ key: "selection",
4
+ dependingFeatures: ["main", "tree"],
5
+ getInitialState: (initialState) => (Object.assign({ selectedItems: [] }, initialState)),
6
+ getDefaultConfig: (defaultConfig, tree) => (Object.assign({ setSelectedItems: makeStateUpdater("selectedItems", tree) }, defaultConfig)),
7
+ createTreeInstance: (prev, instance) => (Object.assign(Object.assign({}, prev), { setSelectedItems: (selectedItems) => {
8
+ var _a, _b;
9
+ (_b = (_a = instance.getConfig()).setSelectedItems) === null || _b === void 0 ? void 0 : _b.call(_a, selectedItems);
10
+ },
11
+ // TODO memo
12
+ getSelectedItems: () => {
13
+ return instance.getState().selectedItems.map(instance.getItemInstance);
14
+ } })),
15
+ createItemInstance: (prev, item, tree) => (Object.assign(Object.assign({}, prev), { select: () => {
16
+ const { selectedItems } = tree.getState();
17
+ tree.setSelectedItems(selectedItems.includes(item.getItemMeta().itemId)
18
+ ? selectedItems
19
+ : [...selectedItems, item.getItemMeta().itemId]);
20
+ }, deselect: () => {
21
+ const { selectedItems } = tree.getState();
22
+ tree.setSelectedItems(selectedItems.filter((id) => id !== item.getItemMeta().itemId));
23
+ }, isSelected: () => {
24
+ const { selectedItems } = tree.getState();
25
+ return selectedItems.includes(item.getItemMeta().itemId);
26
+ }, selectUpTo: (ctrl) => {
27
+ const indexA = item.getItemMeta().index;
28
+ // TODO dont use focused item as anchor, but last primary-clicked item
29
+ const indexB = tree.getFocusedItem().getItemMeta().index;
30
+ const [a, b] = indexA < indexB ? [indexA, indexB] : [indexB, indexA];
31
+ const newSelectedItems = tree
32
+ .getItems()
33
+ .slice(a, b + 1)
34
+ .map((item) => item.getItemMeta().itemId);
35
+ if (!ctrl) {
36
+ tree.setSelectedItems(newSelectedItems);
37
+ return;
38
+ }
39
+ const { selectedItems } = tree.getState();
40
+ const uniqueSelectedItems = [
41
+ ...new Set([...selectedItems, ...newSelectedItems]),
42
+ ];
43
+ tree.setSelectedItems(uniqueSelectedItems);
44
+ }, toggleSelect: () => {
45
+ if (item.isSelected()) {
46
+ item.deselect();
47
+ }
48
+ else {
49
+ item.select();
50
+ }
51
+ }, getProps: () => (Object.assign(Object.assign({}, prev.getProps()), { onClick: (e) => {
52
+ var _a, _b;
53
+ if (e.shiftKey) {
54
+ item.selectUpTo(e.ctrlKey || e.metaKey);
55
+ }
56
+ else if (e.ctrlKey || e.metaKey) {
57
+ item.toggleSelect();
58
+ }
59
+ else {
60
+ tree.setSelectedItems([item.getItemMeta().itemId]);
61
+ }
62
+ (_b = (_a = prev.getProps()).onClick) === null || _b === void 0 ? void 0 : _b.call(_a, e);
63
+ } })) })),
64
+ hotkeys: {
65
+ // setSelectedItem: {
66
+ // hotkey: "space",
67
+ // handler: (e, tree) => {
68
+ // tree.setSelectedItems([tree.getFocusedItem().getId()]);
69
+ // },
70
+ // },
71
+ toggleSelectItem: {
72
+ hotkey: "ctrl+space",
73
+ handler: (e, tree) => {
74
+ tree.getFocusedItem().toggleSelect();
75
+ },
76
+ },
77
+ selectUpwards: {
78
+ hotkey: "shift+ArrowUp",
79
+ handler: () => {
80
+ // TODO
81
+ },
82
+ },
83
+ selectDownwards: {
84
+ hotkey: "shift+ArrowDown",
85
+ handler: () => {
86
+ // TODO
87
+ },
88
+ },
89
+ selectUpwardsCtrl: {
90
+ hotkey: "shift+ctrl+ArrowUp",
91
+ handler: () => {
92
+ // TODO
93
+ },
94
+ },
95
+ selectDownwardsCtrl: {
96
+ hotkey: "shift+ctrl+ArrowUp",
97
+ handler: () => {
98
+ // TODO
99
+ },
100
+ },
101
+ selectAll: {
102
+ hotkey: "Control+a",
103
+ preventDefault: true,
104
+ handler: (e, tree) => {
105
+ tree.setSelectedItems(tree.getItems().map((item) => item.getId()));
106
+ },
107
+ },
108
+ },
109
+ };
@@ -0,0 +1,21 @@
1
+ import { ItemInstance, SetStateFn } from "../../types/core";
2
+ export type SelectionFeatureDef<T> = {
3
+ state: {
4
+ selectedItems: string[];
5
+ };
6
+ config: {
7
+ setSelectedItems?: SetStateFn<string[]>;
8
+ };
9
+ treeInstance: {
10
+ setSelectedItems: (selectedItems: string[]) => void;
11
+ getSelectedItems: () => ItemInstance<T>[];
12
+ };
13
+ itemInstance: {
14
+ select: () => void;
15
+ deselect: () => void;
16
+ toggleSelect: () => void;
17
+ isSelected: () => boolean;
18
+ selectUpTo: (ctrl: boolean) => void;
19
+ };
20
+ hotkeys: "toggleSelectItem" | "selectUpwards" | "selectDownwards" | "selectUpwardsCtrl" | "selectDownwardsCtrl" | "selectAll";
21
+ };
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,4 @@
1
+ import { FeatureImplementation } from "../../types/core";
2
+ import { SyncDataLoaderFeatureDef } from "./types";
3
+ import { MainFeatureDef } from "../main/types";
4
+ export declare const syncDataLoaderFeature: FeatureImplementation<any, SyncDataLoaderFeatureDef<any>, MainFeatureDef | SyncDataLoaderFeatureDef<any>>;
@@ -0,0 +1,6 @@
1
+ export const syncDataLoaderFeature = {
2
+ key: "sync-data-loader",
3
+ dependingFeatures: ["main"],
4
+ createTreeInstance: (prev, instance) => (Object.assign(Object.assign({}, prev), { retrieveItemData: (itemId) => instance.getConfig().dataLoader.getItem(itemId), retrieveChildrenIds: (itemId) => instance.getConfig().dataLoader.getChildren(itemId) })),
5
+ createItemInstance: (prev) => (Object.assign(Object.assign({}, prev), { isLoading: () => false })),
6
+ };