@headless-tree/core 0.0.1

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 (87) hide show
  1. package/lib/core/create-tree.d.ts +2 -0
  2. package/lib/core/create-tree.js +116 -0
  3. package/lib/data-adapters/nested-data-adapter.d.ts +9 -0
  4. package/lib/data-adapters/nested-data-adapter.js +32 -0
  5. package/lib/data-adapters/types.d.ts +7 -0
  6. package/lib/data-adapters/types.js +2 -0
  7. package/lib/features/async-data-loader/feature.d.ts +5 -0
  8. package/lib/features/async-data-loader/feature.js +80 -0
  9. package/lib/features/async-data-loader/types.d.ts +41 -0
  10. package/lib/features/async-data-loader/types.js +2 -0
  11. package/lib/features/drag-and-drop/feature.d.ts +3 -0
  12. package/lib/features/drag-and-drop/feature.js +144 -0
  13. package/lib/features/drag-and-drop/types.d.ts +49 -0
  14. package/lib/features/drag-and-drop/types.js +9 -0
  15. package/lib/features/drag-and-drop/utils.d.ts +7 -0
  16. package/lib/features/drag-and-drop/utils.js +121 -0
  17. package/lib/features/expand-all/feature.d.ts +6 -0
  18. package/lib/features/expand-all/feature.js +39 -0
  19. package/lib/features/expand-all/types.d.ts +17 -0
  20. package/lib/features/expand-all/types.js +2 -0
  21. package/lib/features/hotkeys-core/feature.d.ts +4 -0
  22. package/lib/features/hotkeys-core/feature.js +73 -0
  23. package/lib/features/hotkeys-core/types.d.ts +25 -0
  24. package/lib/features/hotkeys-core/types.js +2 -0
  25. package/lib/features/main/types.d.ts +36 -0
  26. package/lib/features/main/types.js +2 -0
  27. package/lib/features/renaming/feature.d.ts +5 -0
  28. package/lib/features/renaming/feature.js +65 -0
  29. package/lib/features/renaming/types.d.ts +27 -0
  30. package/lib/features/renaming/types.js +2 -0
  31. package/lib/features/search/feature.d.ts +5 -0
  32. package/lib/features/search/feature.js +90 -0
  33. package/lib/features/search/types.d.ts +33 -0
  34. package/lib/features/search/types.js +2 -0
  35. package/lib/features/selection/feature.d.ts +5 -0
  36. package/lib/features/selection/feature.js +112 -0
  37. package/lib/features/selection/types.d.ts +21 -0
  38. package/lib/features/selection/types.js +2 -0
  39. package/lib/features/sync-data-loader/feature.d.ts +4 -0
  40. package/lib/features/sync-data-loader/feature.js +9 -0
  41. package/lib/features/sync-data-loader/types.d.ts +19 -0
  42. package/lib/features/sync-data-loader/types.js +2 -0
  43. package/lib/features/tree/feature.d.ts +6 -0
  44. package/lib/features/tree/feature.js +216 -0
  45. package/lib/features/tree/types.d.ts +57 -0
  46. package/lib/features/tree/types.js +2 -0
  47. package/lib/index.d.ts +21 -0
  48. package/lib/index.js +37 -0
  49. package/lib/mddocs-entry.d.ts +21 -0
  50. package/lib/mddocs-entry.js +17 -0
  51. package/lib/types/core.d.ts +68 -0
  52. package/lib/types/core.js +2 -0
  53. package/lib/types/deep-merge.d.ts +13 -0
  54. package/lib/types/deep-merge.js +2 -0
  55. package/lib/utils.d.ts +9 -0
  56. package/lib/utils.js +105 -0
  57. package/package.json +15 -0
  58. package/src/core/create-tree.ts +195 -0
  59. package/src/data-adapters/nested-data-adapter.ts +48 -0
  60. package/src/data-adapters/types.ts +9 -0
  61. package/src/features/async-data-loader/feature.ts +117 -0
  62. package/src/features/async-data-loader/types.ts +41 -0
  63. package/src/features/drag-and-drop/feature.ts +153 -0
  64. package/src/features/drag-and-drop/types.ts +64 -0
  65. package/src/features/drag-and-drop/utils.ts +88 -0
  66. package/src/features/expand-all/feature.ts +62 -0
  67. package/src/features/expand-all/types.ts +13 -0
  68. package/src/features/hotkeys-core/feature.ts +111 -0
  69. package/src/features/hotkeys-core/types.ts +36 -0
  70. package/src/features/main/types.ts +41 -0
  71. package/src/features/renaming/feature.ts +102 -0
  72. package/src/features/renaming/types.ts +28 -0
  73. package/src/features/search/feature.ts +142 -0
  74. package/src/features/search/types.ts +39 -0
  75. package/src/features/selection/feature.ts +153 -0
  76. package/src/features/selection/types.ts +28 -0
  77. package/src/features/sync-data-loader/feature.ts +27 -0
  78. package/src/features/sync-data-loader/types.ts +20 -0
  79. package/src/features/tree/feature.ts +307 -0
  80. package/src/features/tree/types.ts +70 -0
  81. package/src/index.ts +23 -0
  82. package/src/mddocs-entry.ts +26 -0
  83. package/src/types/core.ts +182 -0
  84. package/src/types/deep-merge.ts +31 -0
  85. package/src/utils.ts +136 -0
  86. package/tsconfig.json +7 -0
  87. package/typedoc.json +4 -0
@@ -0,0 +1,39 @@
1
+ "use strict";
2
+ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
3
+ function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
4
+ return new (P || (P = Promise))(function (resolve, reject) {
5
+ function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
6
+ function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
7
+ function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
8
+ step((generator = generator.apply(thisArg, _arguments || [])).next());
9
+ });
10
+ };
11
+ Object.defineProperty(exports, "__esModule", { value: true });
12
+ exports.expandAllFeature = void 0;
13
+ const utils_1 = require("../../utils");
14
+ exports.expandAllFeature = {
15
+ key: "expand-all",
16
+ dependingFeatures: ["main", "tree"],
17
+ createTreeInstance: (prev, tree) => (Object.assign(Object.assign({}, prev), { expandAll: (cancelToken) => __awaiter(void 0, void 0, void 0, function* () {
18
+ yield Promise.all(tree.getItems().map((item) => item.expandAll(cancelToken)));
19
+ }), collapseAll: () => __awaiter(void 0, void 0, void 0, function* () {
20
+ var _a, _b;
21
+ (_b = (_a = tree.getConfig()).setExpandedItems) === null || _b === void 0 ? void 0 : _b.call(_a, []);
22
+ }) })),
23
+ createItemInstance: (prev, item, tree) => (Object.assign(Object.assign({}, prev), { expandAll: (cancelToken) => __awaiter(void 0, void 0, void 0, function* () {
24
+ if (cancelToken === null || cancelToken === void 0 ? void 0 : cancelToken.current) {
25
+ return;
26
+ }
27
+ item.expand();
28
+ yield (0, utils_1.poll)(() => !tree.getState().loadingItems.includes(item.getId()));
29
+ yield Promise.all(item.getChildren().map((child) => __awaiter(void 0, void 0, void 0, function* () {
30
+ yield (0, utils_1.poll)(() => !tree.getState().loadingItems.includes(child.getId()));
31
+ yield (child === null || child === void 0 ? void 0 : child.expandAll(cancelToken));
32
+ })));
33
+ }), collapseAll: () => __awaiter(void 0, void 0, void 0, function* () {
34
+ yield Promise.all(item.getChildren().map((child) => __awaiter(void 0, void 0, void 0, function* () {
35
+ yield (child === null || child === void 0 ? void 0 : child.collapseAll());
36
+ })));
37
+ item.collapse();
38
+ }) })),
39
+ };
@@ -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,2 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
@@ -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,73 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.hotkeysCoreFeature = void 0;
4
+ const specialKeys = {
5
+ Letter: /^[a-z]$/,
6
+ LetterOrNumber: /^[a-z0-9]$/,
7
+ };
8
+ const testHotkeyMatch = (pressedKeys, tree, hotkey) => {
9
+ const supposedKeys = hotkey.hotkey.split("+");
10
+ const doKeysMatch = supposedKeys.every((key) => key in specialKeys
11
+ ? [...pressedKeys].some((pressedKey) => specialKeys[key].test(pressedKey))
12
+ : pressedKeys.has(key));
13
+ const isEnabled = !hotkey.isEnabled || hotkey.isEnabled(tree);
14
+ const equalCounts = pressedKeys.size === supposedKeys.length;
15
+ return doKeysMatch && isEnabled && equalCounts;
16
+ };
17
+ const findHotkeyMatch = (pressedKeys, tree, config1, config2) => {
18
+ var _a;
19
+ return (_a = Object.entries(Object.assign(Object.assign({}, config1), config2)).find(([, hotkey]) => testHotkeyMatch(pressedKeys, tree, hotkey))) === null || _a === void 0 ? void 0 : _a[0];
20
+ };
21
+ exports.hotkeysCoreFeature = {
22
+ key: "hotkeys-core",
23
+ dependingFeatures: ["main", "tree"],
24
+ onTreeMount: (tree, element) => {
25
+ const data = tree.getDataRef();
26
+ const keydown = (e) => {
27
+ var _a, _b;
28
+ var _c;
29
+ (_a = (_c = data.current).pressedKeys) !== null && _a !== void 0 ? _a : (_c.pressedKeys = new Set());
30
+ const newMatch = !data.current.pressedKeys.has(e.key);
31
+ data.current.pressedKeys.add(e.key);
32
+ console.log(data.current.pressedKeys);
33
+ const hotkeyName = findHotkeyMatch(data.current.pressedKeys, tree, tree.getHotkeyPresets(), tree.getConfig().hotkeys);
34
+ if (!hotkeyName)
35
+ return;
36
+ const hotkeyConfig = Object.assign(Object.assign({}, tree.getHotkeyPresets()[hotkeyName]), (_b = tree.getConfig().hotkeys) === null || _b === void 0 ? void 0 : _b[hotkeyName]);
37
+ if (!hotkeyConfig)
38
+ return;
39
+ if (!hotkeyConfig.allowWhenInputFocused &&
40
+ e.target instanceof HTMLInputElement)
41
+ return;
42
+ if (!hotkeyConfig.canRepeat && !newMatch)
43
+ return;
44
+ if (hotkeyConfig.preventDefault)
45
+ e.preventDefault();
46
+ hotkeyConfig.handler(e, tree);
47
+ };
48
+ const keyup = (e) => {
49
+ var _a;
50
+ var _b;
51
+ (_a = (_b = data.current).pressedKeys) !== null && _a !== void 0 ? _a : (_b.pressedKeys = new Set());
52
+ data.current.pressedKeys.delete(e.key);
53
+ };
54
+ // keyup is registered on document, because some hotkeys shift
55
+ // the focus away from the tree (i.e. search)
56
+ // and then we wouldn't get the keyup event anymore
57
+ element.addEventListener("keydown", keydown);
58
+ document.addEventListener("keyup", keyup);
59
+ data.current.keydownHandler = keydown;
60
+ data.current.keyupHandler = keyup;
61
+ },
62
+ onTreeUnmount: (tree, element) => {
63
+ const data = tree.getDataRef();
64
+ if (data.current.keyupHandler) {
65
+ document.removeEventListener("keyup", data.current.keyupHandler);
66
+ delete data.current.keyupHandler;
67
+ }
68
+ if (data.current.keydownHandler) {
69
+ element.removeEventListener("keydown", data.current.keydownHandler);
70
+ delete data.current.keydownHandler;
71
+ }
72
+ },
73
+ };
@@ -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,2 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
@@ -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,2 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
@@ -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,65 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.renamingFeature = void 0;
4
+ const utils_1 = require("../../utils");
5
+ exports.renamingFeature = {
6
+ key: "renaming",
7
+ dependingFeatures: ["main", "tree"],
8
+ getDefaultConfig: (defaultConfig, tree) => (Object.assign({ setRenamingItem: (0, utils_1.makeStateUpdater)("renamingItem", tree), setRenamingValue: (0, utils_1.makeStateUpdater)("renamingValue", tree), canRename: () => true }, defaultConfig)),
9
+ createTreeInstance: (prev, instance) => (Object.assign(Object.assign({}, prev), { startRenamingItem: (itemId) => {
10
+ var _a, _b;
11
+ const config = instance.getConfig();
12
+ const item = instance.getItemInstance(itemId);
13
+ if (!item.canRename()) {
14
+ return;
15
+ }
16
+ (_a = config.setRenamingItem) === null || _a === void 0 ? void 0 : _a.call(config, itemId);
17
+ (_b = config.setRenamingValue) === null || _b === void 0 ? void 0 : _b.call(config, item.getItemName());
18
+ }, getRenamingItem: () => {
19
+ const itemId = instance.getState().renamingItem;
20
+ return itemId ? instance.getItemInstance(itemId) : null;
21
+ }, getRenamingValue: () => instance.getState().renamingValue || "", abortRenaming: () => {
22
+ var _a, _b;
23
+ (_b = (_a = instance.getConfig()).setRenamingItem) === null || _b === void 0 ? void 0 : _b.call(_a, null);
24
+ }, completeRenaming: () => {
25
+ var _a, _b, _c;
26
+ const config = instance.getConfig();
27
+ const item = instance.getRenamingItem();
28
+ if (item) {
29
+ (_a = config.onRename) === null || _a === void 0 ? void 0 : _a.call(config, item, instance.getState().renamingValue || "");
30
+ }
31
+ (_c = (_b = instance.getConfig()).setRenamingItem) === null || _c === void 0 ? void 0 : _c.call(_b, null);
32
+ }, isRenamingItem: () => !!instance.getState().renamingItem })),
33
+ createItemInstance: (prev, instance, tree) => (Object.assign(Object.assign({}, prev), { getRenameInputProps: () => ({
34
+ onBlur: () => tree.abortRenaming(),
35
+ value: tree.getRenamingValue(),
36
+ onChange: (e) => {
37
+ var _a, _b;
38
+ (_b = (_a = tree.getConfig()).setRenamingValue) === null || _b === void 0 ? void 0 : _b.call(_a, e.target.value);
39
+ },
40
+ }), 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 })),
41
+ hotkeys: {
42
+ renameItem: {
43
+ hotkey: "F2",
44
+ handler: (e, tree) => {
45
+ tree.startRenamingItem(tree.getFocusedItem().getId());
46
+ },
47
+ },
48
+ abortRenaming: {
49
+ hotkey: "Escape",
50
+ allowWhenInputFocused: true,
51
+ isEnabled: (tree) => tree.isRenamingItem(),
52
+ handler: (e, tree) => {
53
+ tree.abortRenaming();
54
+ },
55
+ },
56
+ completeRenaming: {
57
+ hotkey: "Enter",
58
+ allowWhenInputFocused: true,
59
+ isEnabled: (tree) => tree.isRenamingItem(),
60
+ handler: (e, tree) => {
61
+ tree.completeRenaming();
62
+ },
63
+ },
64
+ },
65
+ };
@@ -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,2 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
@@ -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,90 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.searchFeature = void 0;
4
+ const utils_1 = require("../../utils");
5
+ exports.searchFeature = {
6
+ key: "search",
7
+ dependingFeatures: ["main", "tree"],
8
+ getInitialState: (initialState) => (Object.assign({ search: null }, initialState)),
9
+ getDefaultConfig: (defaultConfig, tree) => (Object.assign({ setSearch: (0, utils_1.makeStateUpdater)("search", tree), isSearchMatchingItem: (search, item) => search.length > 0 &&
10
+ item.getItemName().toLowerCase().includes(search.toLowerCase()) }, defaultConfig)),
11
+ createTreeInstance: (prev, instance) => (Object.assign(Object.assign({}, prev), { setSearch: (search) => {
12
+ var _a, _b, _c;
13
+ (_b = (_a = instance.getConfig()).setSearch) === null || _b === void 0 ? void 0 : _b.call(_a, search);
14
+ (_c = instance
15
+ .getItems()
16
+ .find((item) => {
17
+ var _a, _b;
18
+ return (_b = (_a = instance
19
+ .getConfig()).isSearchMatchingItem) === null || _b === void 0 ? void 0 : _b.call(_a, instance.getSearchValue(), item);
20
+ })) === null || _c === void 0 ? void 0 : _c.setFocused();
21
+ }, openSearch: (initialValue = "") => {
22
+ instance.setSearch(initialValue);
23
+ setTimeout(() => {
24
+ var _a;
25
+ (_a = instance
26
+ .getDataRef()
27
+ .current.searchInput) === null || _a === void 0 ? void 0 : _a.focus();
28
+ });
29
+ }, closeSearch: () => {
30
+ instance.setSearch(null);
31
+ instance.updateDomFocus();
32
+ }, isSearchOpen: () => instance.getState().search !== null, getSearchValue: () => instance.getState().search || "", registerSearchInputElement: (element) => {
33
+ const dataRef = instance.getDataRef();
34
+ dataRef.current.searchInput = element;
35
+ if (element && dataRef.current.keydownHandler) {
36
+ element.addEventListener("keydown", dataRef.current.keydownHandler);
37
+ }
38
+ }, getSearchInputElement: () => { var _a; return (_a = instance.getDataRef().current.searchInput) !== null && _a !== void 0 ? _a : null; }, getSearchInputElementProps: () => ({
39
+ value: instance.getSearchValue(),
40
+ onChange: (e) => instance.setSearch(e.target.value),
41
+ onBlur: () => instance.closeSearch(),
42
+ }), getSearchMatchingItems: (0, utils_1.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()]) })),
43
+ createItemInstance: (prev, item, tree) => (Object.assign(Object.assign({}, prev), { isMatchingSearch: () => tree.getSearchMatchingItems().some((i) => i.getId() === item.getId()) })),
44
+ hotkeys: {
45
+ openSearch: {
46
+ hotkey: "LetterOrNumber",
47
+ preventDefault: true,
48
+ isEnabled: (tree) => !tree.isSearchOpen(),
49
+ handler: (e, tree) => {
50
+ e.stopPropagation();
51
+ tree.openSearch(e.key);
52
+ },
53
+ },
54
+ closeSearch: {
55
+ // TODO allow multiple, i.e. Enter
56
+ hotkey: "Escape",
57
+ allowWhenInputFocused: true,
58
+ isEnabled: (tree) => tree.isSearchOpen(),
59
+ handler: (e, tree) => {
60
+ tree.closeSearch();
61
+ },
62
+ },
63
+ nextSearchItem: {
64
+ hotkey: "ArrowDown",
65
+ allowWhenInputFocused: true,
66
+ isEnabled: (tree) => tree.isSearchOpen(),
67
+ handler: (e, tree) => {
68
+ // TODO scroll into view
69
+ const focusItem = tree
70
+ .getSearchMatchingItems()
71
+ .find((item) => item.getItemMeta().index >
72
+ tree.getFocusedItem().getItemMeta().index);
73
+ focusItem === null || focusItem === void 0 ? void 0 : focusItem.setFocused();
74
+ },
75
+ },
76
+ previousSearchItem: {
77
+ hotkey: "ArrowUp",
78
+ allowWhenInputFocused: true,
79
+ isEnabled: (tree) => tree.isSearchOpen(),
80
+ handler: (e, tree) => {
81
+ // TODO scroll into view
82
+ const focusItem = [...tree.getSearchMatchingItems()]
83
+ .reverse()
84
+ .find((item) => item.getItemMeta().index <
85
+ tree.getFocusedItem().getItemMeta().index);
86
+ focusItem === null || focusItem === void 0 ? void 0 : focusItem.setFocused();
87
+ },
88
+ },
89
+ },
90
+ };
@@ -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,2 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
@@ -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,112 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.selectionFeature = void 0;
4
+ const utils_1 = require("../../utils");
5
+ exports.selectionFeature = {
6
+ key: "selection",
7
+ dependingFeatures: ["main", "tree"],
8
+ getInitialState: (initialState) => (Object.assign({ selectedItems: [] }, initialState)),
9
+ getDefaultConfig: (defaultConfig, tree) => (Object.assign({ setSelectedItems: (0, utils_1.makeStateUpdater)("selectedItems", tree) }, defaultConfig)),
10
+ createTreeInstance: (prev, instance) => (Object.assign(Object.assign({}, prev), { setSelectedItems: (selectedItems) => {
11
+ var _a, _b;
12
+ (_b = (_a = instance.getConfig()).setSelectedItems) === null || _b === void 0 ? void 0 : _b.call(_a, selectedItems);
13
+ },
14
+ // TODO memo
15
+ getSelectedItems: () => {
16
+ return instance.getState().selectedItems.map(instance.getItemInstance);
17
+ } })),
18
+ createItemInstance: (prev, item, tree) => (Object.assign(Object.assign({}, prev), { select: () => {
19
+ const { selectedItems } = tree.getState();
20
+ tree.setSelectedItems(selectedItems.includes(item.getItemMeta().itemId)
21
+ ? selectedItems
22
+ : [...selectedItems, item.getItemMeta().itemId]);
23
+ }, deselect: () => {
24
+ const { selectedItems } = tree.getState();
25
+ tree.setSelectedItems(selectedItems.filter((id) => id !== item.getItemMeta().itemId));
26
+ }, isSelected: () => {
27
+ const { selectedItems } = tree.getState();
28
+ return selectedItems.includes(item.getItemMeta().itemId);
29
+ }, selectUpTo: (ctrl) => {
30
+ const indexA = item.getItemMeta().index;
31
+ // TODO dont use focused item as anchor, but last primary-clicked item
32
+ const indexB = tree.getFocusedItem().getItemMeta().index;
33
+ const [a, b] = indexA < indexB ? [indexA, indexB] : [indexB, indexA];
34
+ const newSelectedItems = tree
35
+ .getItems()
36
+ .slice(a, b + 1)
37
+ .map((item) => item.getItemMeta().itemId);
38
+ if (!ctrl) {
39
+ tree.setSelectedItems(newSelectedItems);
40
+ return;
41
+ }
42
+ const { selectedItems } = tree.getState();
43
+ const uniqueSelectedItems = [
44
+ ...new Set([...selectedItems, ...newSelectedItems]),
45
+ ];
46
+ tree.setSelectedItems(uniqueSelectedItems);
47
+ }, toggleSelect: () => {
48
+ if (item.isSelected()) {
49
+ item.deselect();
50
+ }
51
+ else {
52
+ item.select();
53
+ }
54
+ }, getProps: () => (Object.assign(Object.assign({}, prev.getProps()), { onClick: (e) => {
55
+ var _a, _b;
56
+ if (e.shiftKey) {
57
+ item.selectUpTo(e.ctrlKey || e.metaKey);
58
+ }
59
+ else if (e.ctrlKey || e.metaKey) {
60
+ item.toggleSelect();
61
+ }
62
+ else {
63
+ tree.setSelectedItems([item.getItemMeta().itemId]);
64
+ }
65
+ (_b = (_a = prev.getProps()).onClick) === null || _b === void 0 ? void 0 : _b.call(_a, e);
66
+ } })) })),
67
+ hotkeys: {
68
+ // setSelectedItem: {
69
+ // hotkey: "space",
70
+ // handler: (e, tree) => {
71
+ // tree.setSelectedItems([tree.getFocusedItem().getId()]);
72
+ // },
73
+ // },
74
+ toggleSelectItem: {
75
+ hotkey: "ctrl+space",
76
+ handler: (e, tree) => {
77
+ tree.getFocusedItem().toggleSelect();
78
+ },
79
+ },
80
+ selectUpwards: {
81
+ hotkey: "shift+ArrowUp",
82
+ handler: () => {
83
+ // TODO
84
+ },
85
+ },
86
+ selectDownwards: {
87
+ hotkey: "shift+ArrowDown",
88
+ handler: () => {
89
+ // TODO
90
+ },
91
+ },
92
+ selectUpwardsCtrl: {
93
+ hotkey: "shift+ctrl+ArrowUp",
94
+ handler: () => {
95
+ // TODO
96
+ },
97
+ },
98
+ selectDownwardsCtrl: {
99
+ hotkey: "shift+ctrl+ArrowUp",
100
+ handler: () => {
101
+ // TODO
102
+ },
103
+ },
104
+ selectAll: {
105
+ hotkey: "Control+a",
106
+ preventDefault: true,
107
+ handler: (e, tree) => {
108
+ tree.setSelectedItems(tree.getItems().map((item) => item.getId()));
109
+ },
110
+ },
111
+ },
112
+ };
@@ -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,2 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
@@ -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,9 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.syncDataLoaderFeature = void 0;
4
+ exports.syncDataLoaderFeature = {
5
+ key: "sync-data-loader",
6
+ dependingFeatures: ["main"],
7
+ createTreeInstance: (prev, instance) => (Object.assign(Object.assign({}, prev), { retrieveItemData: (itemId) => instance.getConfig().dataLoader.getItem(itemId), retrieveChildrenIds: (itemId) => instance.getConfig().dataLoader.getChildren(itemId) })),
8
+ createItemInstance: (prev) => (Object.assign(Object.assign({}, prev), { isLoading: () => false })),
9
+ };
@@ -0,0 +1,19 @@
1
+ export type SyncTreeDataLoader<T> = {
2
+ getItem: (itemId: string) => T;
3
+ getChildren: (itemId: string) => string[];
4
+ };
5
+ export type SyncDataLoaderFeatureDef<T> = {
6
+ state: {};
7
+ config: {
8
+ rootItemId: string;
9
+ dataLoader: SyncTreeDataLoader<T>;
10
+ };
11
+ treeInstance: {
12
+ retrieveItemData: (itemId: string) => T;
13
+ retrieveChildrenIds: (itemId: string) => string[];
14
+ };
15
+ itemInstance: {
16
+ isLoading: () => boolean;
17
+ };
18
+ hotkeys: never;
19
+ };
@@ -0,0 +1,2 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
@@ -0,0 +1,6 @@
1
+ import { FeatureImplementation } from "../../types/core";
2
+ import { TreeFeatureDef } from "./types";
3
+ import { MainFeatureDef } from "../main/types";
4
+ import { HotkeysCoreFeatureDef } from "../hotkeys-core/types";
5
+ import { SyncDataLoaderFeatureDef } from "../sync-data-loader/types";
6
+ export declare const treeFeature: FeatureImplementation<any, TreeFeatureDef<any>, MainFeatureDef | TreeFeatureDef<any> | HotkeysCoreFeatureDef<any> | SyncDataLoaderFeatureDef<any>>;