@apia/tree 0.1.0

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.
package/LICENSE.md ADDED
@@ -0,0 +1,21 @@
1
+ MIT License
2
+
3
+ Copyright (c) [year] [fullname]
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy
6
+ of this software and associated documentation files (the "Software"), to deal
7
+ in the Software without restriction, including without limitation the rights
8
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+ copies of the Software, and to permit persons to whom the Software is
10
+ furnished to do so, subject to the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be included in all
13
+ copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21
+ SOFTWARE.
package/README.md ADDED
@@ -0,0 +1,28 @@
1
+ # Initiator
2
+
3
+ Este package se creó con la única utilidad de copiarlo entero y pegarlo a la hora de crear un nuevo package.
4
+
5
+ Este iniciador permite crear un paquete que compila typescript y puede ser importado desde otros packages.
6
+
7
+ ## IMPORTANTE
8
+
9
+ Los archivos **tsconfig.json** y **tsup.config.ts** no son archivos únicos sino symlinks a archivos de configuración comunes a todos los packages. En caso de que sea necesario modificar alguno de ellos **que en la gran mayoría de los casos no sería necesario**, es necesario eliminar el archivo a modificar y crear uno nuevo.
10
+
11
+ ## Procedimiento
12
+
13
+ - Copiar la carpeta initiator y pegarla con otro nombre dentro de packages.
14
+ - Modificar el package.json:
15
+ - Eliminar la línea ```private: true```.
16
+ - Cambiar la ocurrencia `initiator` por el nombre del nuevo paquete.
17
+ - Agregar los scripts convenientes: dev, build, etc.
18
+ - Ejecutar el comando lerna bootstrap desde la carpeta raíz.
19
+
20
+ Luego de ejecutar estos pasos, el package estaría listo para comenzar a usarse.
21
+
22
+ Este package trae como dependencias por defecto theme-ui y react. Si se desea agregar más dependencias se debe ejecutar el comando ```lerna add --scope="@apia/packageName" dependencyName```. Ejemplo, si creamos un paquete con el nombre @apia/myPackage y queremos agregar lodash como dependencia, ejecutamos el comando ```lerna add --scope="@apia/myPackage" lodash```.
23
+
24
+ **Importante 1**: Dado que estamos desarrollando packages, es importante determinar si las dependencias que vamos a agregar son de tipo dependency o de tipo peerDependency.
25
+
26
+ **Importante 2**: lerna no permite instalar de a varias dependencias a la vez como lo hace npm, por lo tanto, si se desea agregar varias dependencias se debe ejecutar el comando anterior tantas veces como dependencias se quiera agregar.
27
+
28
+ **Importante 3**: React y theme-ui vienen instalados como peerDependencies. En caso de no ser necesarios pueden ser removidos, pero en cualquier caso asegurarse de que estas dependencias no sean instaladas como dependencies puesto que esto causaría comportamientos indeseados en la aplicación.
package/cleanDist.json ADDED
@@ -0,0 +1,3 @@
1
+ {
2
+ "cleanDist": 0.2318938201855587
3
+ }
@@ -0,0 +1,334 @@
1
+ import * as theme_ui_jsx_runtime from 'theme-ui/jsx-runtime';
2
+ import React, { RefObject } from 'react';
3
+ import { EventEmitter, PropsStore, TModify, TRequireOnlyOne, TUpdateFieldConfiguration, TPropsConfiguration } from '@apia/util';
4
+ import { TIconName } from '@apia/icons';
5
+
6
+ declare function useTreeDataController<TreeType extends TreeDataController<any, any> = TreeDataController<any, any>>(name: string): TreeType;
7
+ declare class TreeDataController<NodeProps extends Record<string, unknown> = Record<string, unknown>, NodeType extends TDataNode<NodeProps> = TDataNode<NodeProps>> extends EventEmitter<{
8
+ onArrowUpOnFirstElement: boolean;
9
+ expandNode: NodeType;
10
+ mustFocus: TId;
11
+ }> {
12
+ name: string;
13
+ propsStore: PropsStore<NodeType>;
14
+ _configuration: RefObject<TTreeDataControllerConfig<NodeProps, NodeType>>;
15
+ get configuration(): {
16
+ current: TTreeDataControllerConfig<NodeProps, NodeType> | null;
17
+ };
18
+ hasApendedFirstChild: boolean;
19
+ /**
20
+ * Este array se usa para conocer los padres faltantes al momento de la
21
+ * construcción del árbol, de forma de optimizar el proceso
22
+ */
23
+ missingParents: TId[];
24
+ nonEmittedUpdates: TId[];
25
+ get state(): TTreeDataControllerState;
26
+ stateKey: string;
27
+ get stateStore(): PropsStore<TTreeDataControllerState>;
28
+ constructor(name: string, configuration: RefObject<Partial<TTreeDataControllerConfig<NodeProps, NodeType>>>, propsStore?: PropsStore<NodeType>);
29
+ destructor(): void;
30
+ append(node: Omit<TModify<NodeType, {
31
+ children?: NodeType['children'];
32
+ parentId?: NodeType['parentId'];
33
+ }>, 'nodeProps' | 'actualParentId'> & Partial<Pick<NodeType, 'nodeProps'>>): void;
34
+ /**
35
+ * Cuando se quieren agregar muchos nodos, es conveniente llamar primero al
36
+ * método batchInit y luego de finalizar, al método batchFinish()
37
+ */
38
+ batchInit(): void;
39
+ /**
40
+ * Cuando se quieren agregar muchos nodos, es conveniente llamar primero al
41
+ * método batchInit y luego de finalizar, al método batchFinish()
42
+ */
43
+ batchFinish(): void;
44
+ config(newConf: Partial<TTreeDataControllerConfig<NodeProps, NodeType>>): void;
45
+ deselectAll(): void;
46
+ forceEmitUpdate(): void;
47
+ getInitialState(): TTreeDataControllerState;
48
+ getNodesAsArray(): NodeType[];
49
+ /**
50
+ * Devuelve un array que contiene el id de todos los nodos pertenecientes al
51
+ * árbol
52
+ */
53
+ getNodesIds(): string[];
54
+ getParentId(parentId: TId): TId;
55
+ /**
56
+ * Este método permite controlar el comportamiento con el teclado desde fuera
57
+ * del componente.
58
+ */
59
+ handleKey(ev: React.KeyboardEvent): void;
60
+ isNode(node: NodeType | TDataNodeContainer<NodeProps> | TId): node is NodeType;
61
+ isNodeContainer(node: NodeType | TDataNodeContainer<NodeProps> | TId): node is TDataNodeContainer<NodeProps>;
62
+ includes(searchNode: NodeType): boolean;
63
+ initRoot(): void;
64
+ move(moveNode: TId, destinationNode: TId, afterOrBefore?: TRequireOnlyOne<{
65
+ after: NodeType | TId;
66
+ before: NodeType | TId;
67
+ position: number;
68
+ }>): void;
69
+ /**
70
+ * Borra el nodo del árbol, y dependiendo del parámetro removeChildren, borra
71
+ * también sus hijos.
72
+ *
73
+ * @param removeChildren - Si se pasa en false, los nodos hijos son movidos al root
74
+ */
75
+ remove(removeNode: NodeType | TId, removeChildren?: boolean): void;
76
+ removeMultiple(nodes: TId[], removeChildren?: boolean): void;
77
+ /**
78
+ * Borra todos los nodos del árbol.
79
+ */
80
+ removeAll(): void;
81
+ selectAll(): void;
82
+ setFocusedNode(key: TId, avoidSelection?: boolean): void;
83
+ setSelectedNodes(nodes: TId[], force?: boolean): void;
84
+ setSelectedNodesByClickEvent(ev: React.MouseEvent): void;
85
+ setState(updateProps: Partial<TTreeDataControllerState>, conf?: TUpdateFieldConfiguration): void;
86
+ toggleNodeExpandedState(key: TId, shouldExpand?: boolean): void;
87
+ toggleNodeSelectedState(key: TId, isSelected?: boolean): void;
88
+ }
89
+ /**
90
+ * Este hook permite escuchar las props de un árbol, para ello es necesario
91
+ * pasar el handler.
92
+ *
93
+ * @param handler
94
+ * @param configuration
95
+ * @returns
96
+ */
97
+ declare function useTreeSelector<NodeProps extends Record<string, unknown> = Record<string, unknown>, NodeType extends TDataNode<NodeProps> = TDataNode<NodeProps>, Selected = Record<string, unknown>, PropsType extends Record<string, unknown> = TTreeDataControllerState>(handler: TreeDataController<NodeProps, NodeType>, configuration?: TPropsConfiguration<Selected, PropsType>): Selected;
98
+ /**
99
+ * Este hook permite escuchar las props de un árbol, para ello es necesario
100
+ * pasar el handler.
101
+ *
102
+ * @param handler
103
+ * @param configuration
104
+ * @returns
105
+ */
106
+ declare function useTreeSelectorByName<NodeProps extends Record<string, unknown> = Record<string, unknown>, NodeType extends TDataNode<NodeProps> = TDataNode<NodeProps>, Selected = Record<string, unknown>, PropsType extends Record<string, unknown> = TTreeDataControllerState>(treeName: string, configuration?: TPropsConfiguration<Selected, PropsType>): {
107
+ handler: TreeDataController<NodeProps, NodeType>;
108
+ selection: Selected;
109
+ };
110
+
111
+ type TId = string | number;
112
+ type TDataNodeContainer<NodeProps extends Record<string, unknown> = Record<string, unknown>> = Pick<TDataNode<NodeProps>, 'children' | 'id'>;
113
+ type TTreeDataControllerConfig<NodeProps extends Record<string, unknown> = Record<string, unknown>, NodeType extends TDataNode<NodeProps> = TDataNode<NodeProps>> = {
114
+ emitUpdates?: boolean;
115
+ } & Pick<TTreeProps<NodeProps, NodeType>, 'isMultiple' | 'onLoadData' | 'onExpand' | 'disableSelection' | 'selectionMode'>;
116
+ type TTreeDataControllerState = {
117
+ expandedNodes: TId[];
118
+ filteredNodes: TId[];
119
+ focusedNode: TId | null;
120
+ isLoading: boolean;
121
+ length: number;
122
+ selectedNodes: TId[];
123
+ };
124
+ type TTreeLabelRenderer<NodeProps extends Record<string, unknown> = Record<string, unknown>> = React.FunctionComponent<TDataNode<NodeProps> & {
125
+ level: number;
126
+ }>;
127
+ type TNodesExpandEvent<NodeProps extends Record<string, unknown> = Record<string, unknown>> = TDataNode<NodeProps>;
128
+ type TNodesSelectionEvent<NodeProps extends Record<string, unknown> = Record<string, unknown>> = TDataNode<NodeProps>[];
129
+ type TTreeProps<NodeProps extends Record<string, unknown> = Record<string, unknown>, NodeType extends TDataNode<NodeProps> = TDataNode<NodeProps>> = {
130
+ /**
131
+ * Debe ser un texto descriptivo, se usa para las personas no videntes
132
+ * principalmente.
133
+ */
134
+ label: string;
135
+ name: string;
136
+ } & Partial<{
137
+ className: string;
138
+ disableSelection: boolean;
139
+ /**
140
+ * Si se pasa un filterString !== undefined => los nodos del árbol cuyo label
141
+ * no coincida serán ocultados, a menos que alguno de sus hijos si coincida.
142
+ */
143
+ filterString: string;
144
+ /**
145
+ * Es posible alterar el comportamiento por defecto cada vez que el usuario
146
+ * selecciona un elemento de la lista mediante esta función, indicando al tree
147
+ * sobre qué elemento debe colocar el foco.
148
+ *
149
+ * Esta función debe tomar el liElement que recibe como parámetro y mediante
150
+ * querySelectors o el método que se considere correcto devolver la referencia
151
+ * al hijo sobre el cual se desea colocar el foco.
152
+ *
153
+ * @param liElement - Es el li que renderiza el nodo. Los hijos más importantes de este li son:
154
+ *
155
+ * - .tree__nodeItemLabel: Es el wrapper del botón de expandir, del ícono
156
+ * del nodo y del (custom?)Renderer. - ul[role="group"]: Es el wrapper de
157
+ * todos los hijos del nodo actual.*/
158
+ focusGetter: (liElement: HTMLElement) => HTMLElement;
159
+ /**
160
+ * Esta función es llamada cuando el árbol inicializó con la referencia al
161
+ * handler como parámetro.
162
+ */
163
+ getHandler: (handler: TreeDataController<NodeProps, NodeType>) => unknown;
164
+ /**
165
+ * Permite ocultar todos los nodos de tipo folder que no tengan hijos.
166
+ */
167
+ hideEmptyFolders: boolean;
168
+ /**
169
+ * Permite ocultar la etiqueta de búsqueda
170
+ */
171
+ hideSearchLabel: boolean;
172
+ initialNodes: NodeType[];
173
+ /**
174
+ * Pone el mismo spinner que mientras está en modo batch.
175
+ */
176
+ isLoading: boolean;
177
+ /**
178
+ * Habilita la selección múltiple, por defecto está deshabilitada.
179
+ */
180
+ isMultiple: boolean;
181
+ /**
182
+ * Este callback es llamado cuando se presiona la flecha arriba en el primer
183
+ * elemento del árbol. Está pensado para mover el foco hacia el cuadro de
184
+ * búsqueda.
185
+ */
186
+ onArrowUpOnFirstElement?: () => unknown;
187
+ onDeleteFilterString?: () => unknown;
188
+ onExpand: (ev: TNodesExpandEvent<NodeProps>) => unknown;
189
+ /**
190
+ * Esta función debe devolver una promesa. Cuando se expande un nodo que no
191
+ * tiene hijos, esta función es llamada y el nodo es puesto en estado de carga
192
+ * hasta que la promesa se resuelve.
193
+ */
194
+ onLoadData: (node: NodeType) => Promise<void>;
195
+ /**
196
+ * Cada vez que se hace click sobre un nodo o se presiona enter estando
197
+ * éste en foco, se llama este callback informando de la acción.
198
+ */
199
+ onNodeClick: (ev: React.KeyboardEvent | React.MouseEvent, node: NodeType) => unknown;
200
+ onSelect: (ev: TNodesSelectionEvent<NodeProps>) => unknown;
201
+ /**
202
+ * Si se pasa onFocus, el nodo será seleccionado al recibir foco (por
203
+ * ejemplo, al navegarlo con las flechas). Si se pasa explicit, solamente se
204
+ * hará seleccionará al hacerle click (o presionar enter o espacio).
205
+ */
206
+ selectionMode: 'onFocus' | 'explicit';
207
+ /**
208
+ * Si se pasa en false, los nodos padre solamente se podrán expandir/colapsar
209
+ * al hacer click en los íconos.
210
+ */
211
+ toggleNodesOnLabelClick: boolean;
212
+ variant: string;
213
+ }>;
214
+ type TTreeIconRenderer<NodeProps extends Record<string, unknown> = Record<string, unknown>> = React.FunctionComponent<TDataNode<NodeProps>>;
215
+ type TDataNode<NodeProps extends Record<string, unknown> = Record<string, unknown>> = {
216
+ actualParentId?: TId;
217
+ children: TId[];
218
+ id: TId;
219
+ label: string;
220
+ /**
221
+ * Un objeto arbitrario con propiedades de nodo que no están relacionados a
222
+ * la estructura del árbol, sino de la estructura de información que
223
+ * representa.
224
+ */
225
+ nodeProps: NodeProps;
226
+ parentId: TId;
227
+ } & Partial<{
228
+ /**
229
+ * Con esta prop se puede evitar que el nodo sea colapsado desde el label
230
+ */
231
+ allowToggleExpandedFromLabel?: boolean;
232
+ className?: string;
233
+ color?: string;
234
+ /**
235
+ * Esta prop marca que ya se pidieron los datos mediante onLoadData para este
236
+ * nodo.
237
+ */
238
+ hasLoaded: boolean;
239
+ /**
240
+ * El icon puede ser el nombre de uno de los elementos del mapa de íconos de
241
+ * Icon.tsx o un TTreeIconRenderer que permite utilizar un componente para
242
+ * renderizar el ícono.*/
243
+ icon: TIconName | TTreeIconRenderer<NodeProps>;
244
+ /**
245
+ * Este string se usa como height y width del ícono, de modo que se puede
246
+ * pasar cualquier expresión válida de ThemeUI.
247
+ */
248
+ iconSize: string;
249
+ /**
250
+ * Cuando un nodo está deshabilitado, no se pondrá foco en él.
251
+ */
252
+ isDisabled: boolean;
253
+ /**
254
+ * Indica si un nodo está o no expandido, en caso de que se establezca en
255
+ * true, el mismo será expandido. Esto puede ser útil para controlar el árbol
256
+ * desde fuera, en cuyo caso se deben modificar las propiedades mediante
257
+ * handler.propsStore.updateField(node.id, { newProps }).
258
+ */
259
+ isExpanded: boolean;
260
+ /**
261
+ * Indica que el nodo no debe mostrarse ya que no coincide con los criterios
262
+ * de filtrado.
263
+ */
264
+ isFiltered: boolean;
265
+ /**
266
+ * Indica si un nodo está en foco, es importante notar que alterar esta
267
+ * propiedad no va a alterar qué nodo del árbol está en foco ya que ese
268
+ * control lo hace internamente el árbol. En caso de querer modificar esto en
269
+ * forma externa, se deben usar los métodos correspondientes del handler.
270
+ */
271
+ isFocused: boolean;
272
+ /**
273
+ * Esta prop se usa para decir que un nodo es un sub-árbol, aún cuando no
274
+ * tiene hijos. Esto es especialmente útil cuando se quiere hacer carga
275
+ * asíncrona de datos, ya que al expandirse, se llamará al método onLoadData
276
+ * del tree.
277
+ */
278
+ isLeaf: boolean;
279
+ /**
280
+ * Se setea en true al momento de llamar a onLoadData y en false al finalizar
281
+ * la carga de datos.
282
+ */
283
+ isLoading: boolean;
284
+ /**
285
+ * Por defecto todos los nodos son seleccionables, se debe pasar false
286
+ * explícitamente para deshabilitar la selección.
287
+ */
288
+ isSelectable: boolean;
289
+ /**
290
+ * Indica si un nodo está o no seleccionado. Esto puede ser útil para
291
+ * controlar el árbol desde fuera, en cuyo caso se deben modificar las
292
+ * propiedades mediante handler.propsStore.updateField(node.id, { newProps }).
293
+ */
294
+ isSelected: boolean;
295
+ /**
296
+ * Si se pasa un labelRenderer se utilizará en lugar de
297
+ * DefaultLabelRenderer.tsx. El labelRenderer es el responsable de hacer caso
298
+ * de los eventos de click para la selección del nodo.
299
+ */
300
+ labelRenderer: TTreeLabelRenderer<NodeProps>;
301
+ title: string;
302
+ }>;
303
+
304
+ declare const _default: <NodeProps extends Record<string, unknown> = Record<string, unknown>, NodeType extends TDataNode<NodeProps> = TDataNode<NodeProps>>(props: TTreeProps<NodeProps, NodeType>) => theme_ui_jsx_runtime.JSX.Element;
305
+
306
+ type TTreeContext<NodeProps extends Record<string, unknown> = Record<string, unknown>, NodeType extends TDataNode<NodeProps> = TDataNode<NodeProps>> = {
307
+ forceUpdate: number;
308
+ name: string;
309
+ handler: TreeDataController<NodeProps, NodeType>;
310
+ treeProps: TTreeProps<NodeProps, NodeType>;
311
+ } & Pick<TTreeProps, 'focusGetter'>;
312
+ declare function useTreeContext<NodeProps extends Record<string, unknown> = Record<string, unknown>, NodeType extends TDataNode<NodeProps> = TDataNode<NodeProps>>(): TTreeContext<NodeProps, NodeType>;
313
+
314
+ interface IUseTreeData<NodeProps extends Record<string, unknown> = Record<string, unknown>, NodeType extends TDataNode<NodeProps> = TDataNode<NodeProps>> {
315
+ name: string;
316
+ treeProps: TTreeProps<NodeProps, NodeType>;
317
+ }
318
+ declare function useTreeData<NodeProps extends Record<string, unknown> = Record<string, unknown>, NodeType extends TDataNode<NodeProps> = TDataNode<NodeProps>>({ name, treeProps }: IUseTreeData<NodeProps, NodeType>): {
319
+ data: {
320
+ children: TId[];
321
+ forceUpdate: number;
322
+ };
323
+ handler: TreeDataController<NodeProps, NodeType>;
324
+ keyHandler: {
325
+ id: string;
326
+ onKeyDown: (ev: React.KeyboardEvent) => void;
327
+ onMouseDown: (ev: React.MouseEvent) => void;
328
+ onClick: (ev: React.MouseEvent) => void;
329
+ ref: React.MutableRefObject<HTMLDivElement | null>;
330
+ };
331
+ };
332
+
333
+ export { TDataNode, TTreeLabelRenderer, TTreeProps, _default as Tree, TreeDataController, useTreeContext, useTreeData, useTreeDataController, useTreeSelector, useTreeSelectorByName };
334
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sources":[],"sourcesContent":[],"names":[],"mappings":""}