@donotdev/crud 0.0.15 → 0.0.17
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/dist/CrudService.d.ts.map +1 -1
- package/dist/CrudService.js +2 -2
- package/dist/CrudStore.d.ts.map +1 -1
- package/dist/FieldRegistry.d.ts +3 -30
- package/dist/FieldRegistry.d.ts.map +1 -1
- package/dist/adapters/FunctionsAdapter.d.ts.map +1 -1
- package/dist/adapters/FunctionsAdapter.js +1 -1
- package/dist/components/CrudCard.d.ts.map +1 -1
- package/dist/components/EntityFilters.d.ts +1 -34
- package/dist/components/EntityFilters.d.ts.map +1 -1
- package/dist/components/EntityFilters.js +1 -1
- package/dist/components/__tests__/EntityFilters.test.js +1 -1
- package/dist/components/__tests__/FormFieldRenderer.test.js +1 -1
- package/dist/components/controlled/complex/ControlledFieldArrayField.d.ts +18 -0
- package/dist/components/controlled/complex/ControlledFieldArrayField.d.ts.map +1 -0
- package/dist/components/controlled/complex/ControlledFieldArrayField.js +1 -0
- package/dist/components/controlled/complex/index.d.ts +1 -0
- package/dist/components/controlled/complex/index.d.ts.map +1 -1
- package/dist/components/controlled/complex/index.js +1 -1
- package/dist/components/controlled/types.d.ts +19 -5
- package/dist/components/controlled/types.d.ts.map +1 -1
- package/dist/components/form/fields/DocumentFieldComponent.d.ts.map +1 -1
- package/dist/components/form/fields/DocumentFieldComponent.js +1 -1
- package/dist/components/form/fields/FileFieldComponent.d.ts.map +1 -1
- package/dist/components/form/fields/FileFieldComponent.js +1 -1
- package/dist/components/form/fields/internal/TiptapEditor.d.ts.map +1 -1
- package/dist/components/index.d.ts +2 -1
- package/dist/components/index.d.ts.map +1 -1
- package/dist/components/index.js +1 -1
- package/dist/forms/hooks/useEntityField.d.ts +15 -24
- package/dist/forms/hooks/useEntityField.d.ts.map +1 -1
- package/dist/forms/hooks/useEntityField.js +1 -1
- package/dist/forms/hooks/useEntityForm.d.ts +3 -2
- package/dist/forms/hooks/useEntityForm.d.ts.map +1 -1
- package/dist/forms/hooks/useEntityForm.js +1 -1
- package/dist/forms/types.d.ts +13 -8
- package/dist/forms/types.d.ts.map +1 -1
- package/dist/forms/utils/buildInitialValues.d.ts.map +1 -1
- package/dist/forms/utils/buildInitialValues.js +1 -1
- package/dist/hooks/index.d.ts +1 -1
- package/dist/hooks/index.d.ts.map +1 -1
- package/dist/hooks/index.js +1 -1
- package/dist/hooks/useFileUpload.d.ts.map +1 -1
- package/dist/hooks/useFileUpload.js +1 -1
- package/dist/hooks/useUnsavedChangesWarning.d.ts +3 -24
- package/dist/hooks/useUnsavedChangesWarning.d.ts.map +1 -1
- package/dist/hooks/useUnsavedChangesWarning.js +1 -1
- package/dist/index.d.ts +5 -1
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +1 -1
- package/dist/registerBuiltinFieldTypes.d.ts.map +1 -1
- package/dist/registerBuiltinFieldTypes.js +1 -1
- package/dist/stores/UploadStore.d.ts.map +1 -1
- package/dist/tsconfig.tsbuildinfo +1 -1
- package/dist/useBaseCrudList.d.ts.map +1 -1
- package/dist/useCrud.js +1 -1
- package/dist/useCrudCardList.d.ts +14 -19
- package/dist/useCrudCardList.d.ts.map +1 -1
- package/dist/useCrudCardList.js +1 -1
- package/dist/useCrudList.d.ts +19 -18
- package/dist/useCrudList.d.ts.map +1 -1
- package/dist/useCrudList.js +1 -1
- package/dist/utils/clientListProcessing.d.ts +82 -0
- package/dist/utils/clientListProcessing.d.ts.map +1 -0
- package/dist/utils/clientListProcessing.js +1 -0
- package/dist/utils/fileStorage.d.ts.map +1 -1
- package/dist/utils/imageStorage.d.ts.map +1 -1
- package/dist/utils/imageStorage.js +1 -1
- package/dist/utils/matchesFilter.d.ts +45 -0
- package/dist/utils/matchesFilter.d.ts.map +1 -0
- package/dist/utils/matchesFilter.js +1 -0
- package/dist/utils/mergeWithOptimistic.d.ts.map +1 -1
- package/dist/utils/uploadValidation.d.ts.map +1 -1
- package/dist/utils/uploadValidation.js +1 -1
- package/package.json +5 -8
- package/dist/hooks/useFormNavigationGuard.d.ts +0 -34
- package/dist/hooks/useFormNavigationGuard.d.ts.map +0 -1
- package/dist/hooks/useFormNavigationGuard.js +0 -1
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"useBaseCrudList.d.ts","sourceRoot":"","sources":["../src/useBaseCrudList.ts"],"names":[],"mappings":"AAeA,OAAO,KAAK,
|
|
1
|
+
{"version":3,"file":"useBaseCrudList.d.ts","sourceRoot":"","sources":["../src/useBaseCrudList.ts"],"names":[],"mappings":"AAeA,OAAO,KAAK,EACV,MAAM,EACN,aAAa,EACb,cAAc,EAEd,YAAY,EACb,MAAM,gBAAgB,CAAC;AAQxB,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,WAAW,CAAC;AAKhD,+BAA+B;AAC/B,MAAM,WAAW,gBAAgB;IAC/B,uEAAuE;IACvE,IAAI,EAAE,QAAQ,GAAG,QAAQ,CAAC;IAC1B,yDAAyD;IACzD,IAAI,EAAE,MAAM,CAAC;IACb,2CAA2C;IAC3C,QAAQ,EAAE,MAAM,CAAC;CAClB;AAED,kCAAkC;AAClC,MAAM,WAAW,sBAAsB,CAAC,CAAC,CAAE,SAAQ,cAAc,CAAC,CAAC,CAAC;IAClE,wDAAwD;IACxD,OAAO,CAAC,EAAE,OAAO,CAAC;IAClB,0CAA0C;IAC1C,YAAY,CAAC,EAAE,YAAY,CAAC;CAC7B;AAED,sCAAsC;AACtC,MAAM,WAAW,qBAAqB,CAAC,CAAC;IACtC,MAAM,EAAE,aAAa,CAAC;IACtB,IAAI,EAAE;QAAE,KAAK,EAAE,CAAC,EAAE,CAAC;QAAC,KAAK,CAAC,EAAE,MAAM,CAAA;KAAE,GAAG,IAAI,CAAC;IAC5C,mDAAmD;IACnD,KAAK,EAAE,CAAC,EAAE,CAAC;IACX,iDAAiD;IACjD,OAAO,EAAE,OAAO,CAAC;IACjB,6DAA6D;IAC7D,QAAQ,EAAE,OAAO,CAAC;IAClB,KAAK,EAAE,KAAK,GAAG,IAAI,CAAC;IACpB,8BAA8B;IAC9B,MAAM,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC;IAC5B,wCAAwC;IACxC,OAAO,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC;IAC7B,WAAW,EAAE,OAAO,CAAC;CACtB;AAED;;;;;GAKG;AACH,wBAAgB,eAAe,CAAC,CAAC,SAAS;IAAE,EAAE,EAAE,MAAM,CAAA;CAAE,GAAG;IAAE,EAAE,EAAE,MAAM,CAAA;CAAE,EACvE,kBAAkB,EAAE,MAAM,GAAG,MAAM,EACnC,UAAU,EAAE,cAAc,EAC1B,UAAU,EAAE,gBAAgB,GAAG,IAAI,EACnC,OAAO,GAAE,sBAAsB,CAAC,CAAC,CAAM,GACtC,qBAAqB,CAAC,CAAC,CAAC,CA2O1B"}
|
package/dist/useCrud.js
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
"use client";import{useEffect as
|
|
1
|
+
"use client";import{useEffect as V,useMemo as F,useCallback as a}from"react";import{useFeatureConsent as W,FRAMEWORK_FEATURES as Z,FEATURE_STATUS as h,handleError as $,isClient as B,createSchemas as H,DEGRADED_CRUD_API as J}from"@donotdev/core";import{getCrudService as X}from"./CrudService";import{useCrudStore as u}from"./CrudStore";import{injectScope as P,injectScopeFilter as O}from"./utils/scopeUtils";const re={},ne={};let v=null,_=null,g=!1;function i(){return v}function ae(l,d={}){const S=W(Z.CRUD),x=u(e=>e.crudService),C=S&&!x,D=S?C?h.INITIALIZING:h.READY:h.DEGRADED,m=l==="status",p=!m&&typeof l=="object"?l:d.entity,s=m?"__status__":typeof l=="string"?l:l.collection,o=p?.scope,T=d.noCache??!1,A=d.staleTime,f=F(()=>({noCache:T,staleTime:A}),[T,A]),t=F(()=>{if(!m){if(d.schema){const e=d.schema;return{create:e,draft:e,update:e,get:e,list:e,listCard:e,delete:e}}if(p)return H(p)}},[m,d.schema,p?.name,s]),M=u(e=>e.collections[s]?.loading||!1),b=u(e=>e.collections[s]?.error||null),N=null;V(()=>{if(!B()||!S||!t)return;const{crudService:e}=u.getState();e||v||_||(_=(async()=>{try{if(v||u.getState().crudService)return;const c=X();v=c,c.setStore(u),await c.initialize(),g=!1,u.getState().setCrudService(c)}catch(c){g||($(c,{userMessage:"Failed to initialize CRUD service.",context:{collection:s},severity:"error"}),g=!0)}finally{_=null}})())},[S,s,t?"hasSchema":"noSchema"]);const I=C||M,z=D===h.READY,Y=a(async e=>{const c=i();return!c||!t?null:c.get(s,e,t.get,f)},[s,t?"hasSchema":"noSchema",f]),j=a(async(e,c,r)=>{const n=i();if(!n||!t)return;const E=P(c,o),Q=E?.status==="draft"?t.draft:t.create;await n.set(s,e,E,Q,r)},[s,t?"hasSchema":"noSchema",o?.provider]),k=a(async(e,c,r)=>{const n=i();n&&await n.update(s,e,c,t?.update,r)},[s,t]),q=a(async(e,c)=>{const r=i();r&&await r.delete(s,e,c)},[s]),G=a(async(e,c)=>{const r=i();if(!r||!t)return"";const n=P(e,o),w=n?.status==="draft"?t.draft:t.create;return r.add(s,n,w,c)},[s,t?"hasSchema":"noSchema",o?.provider]),L=a(async e=>{const c=i();if(!c||!t)return[];const r=O(e,o);return(await c.query(s,r,t.list,f)).items},[s,t?"hasSchema":"noSchema",f,o?.provider]),R=a((e,c)=>{const r=i();return!r||!t?()=>{}:r.subscribe(s,e,c,t.get)},[s,t?"hasSchema":"noSchema"]),y=a((e,c)=>{const r=i();return!r||!t?()=>{}:r.subscribeToCollection(s,e,c,t.list)},[s,t?"hasSchema":"noSchema"]),U=a(async()=>{const e=i();e&&await e.invalidateCollection(s)},[s]);return m?D:z?{status:h.READY,data:N,loading:I,error:b,get:Y,set:j,update:k,delete:q,add:G,query:L,subscribe:R,subscribeToCollection:y,invalidate:U,isAvailable:!0,_collection:s,_schemas:t,_cacheOptions:f,_scope:o}:{...J,status:D,loading:I,error:b,subscribe:R,subscribeToCollection:y,invalidate:U,_collection:s,_schemas:t,_cacheOptions:f,_scope:o}}export{re as EMPTY_DATA,ne as EMPTY_OPTIMISTIC,i as getCrudServiceInstance,ae as useCrud};
|
|
@@ -1,27 +1,27 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @fileoverview useCrudCardList Hook - Optimized Card List Management
|
|
3
|
-
* @description Specialized hook for fetching lists of entities optimized for card views.
|
|
4
|
-
* Fetches only the fields defined in `listCardFields` (or `listFields`) to reduce payload size.
|
|
5
|
-
*
|
|
6
|
-
* @version 0.0.10
|
|
7
|
-
* @since 0.0.1
|
|
8
|
-
* @author AMBROISE PARK Consulting
|
|
9
|
-
*/
|
|
10
1
|
import type { Entity, FeatureStatus, QueryOptions } from '@donotdev/core';
|
|
11
2
|
import type { UseCrudOptions } from './useCrud';
|
|
3
|
+
import type { ClientSortConfig } from './utils/clientListProcessing';
|
|
12
4
|
export interface UseCrudCardListOptions<T> extends UseCrudOptions<T> {
|
|
13
5
|
/** Automatically fetch data on mount (default: true) */
|
|
14
6
|
enabled?: boolean;
|
|
15
7
|
/** Where/orderBy/limit passed to listCard query */
|
|
16
8
|
queryOptions?: QueryOptions;
|
|
9
|
+
/** Client-side search query. Applied after filters. */
|
|
10
|
+
searchQuery?: string;
|
|
11
|
+
/** Client-side sort config. Defaults to entity.defaultSort. Pass null to disable. */
|
|
12
|
+
clientSort?: ClientSortConfig | null;
|
|
17
13
|
}
|
|
18
14
|
export interface UseCrudCardListReturn<T> {
|
|
19
15
|
status: FeatureStatus;
|
|
20
16
|
data: {
|
|
21
17
|
items: T[];
|
|
22
18
|
} | null;
|
|
23
|
-
/** Convenience: items array (data?.items ?? []) */
|
|
19
|
+
/** Convenience: items array (data?.items ?? []) — raw + optimistic, no client processing */
|
|
24
20
|
items: T[];
|
|
21
|
+
/** Items after client-side filters + search + sort — ready to render */
|
|
22
|
+
processed: T[];
|
|
23
|
+
/** Count of items after filters + search (before pagination) */
|
|
24
|
+
filteredTotal: number;
|
|
25
25
|
/** True during initial fetch (show skeletons) */
|
|
26
26
|
loading: boolean;
|
|
27
27
|
/** True during background refetch (show subtle indicator) */
|
|
@@ -48,15 +48,10 @@ export interface UseCrudCardListReturn<T> {
|
|
|
48
48
|
*
|
|
49
49
|
* @example
|
|
50
50
|
* ```typescript
|
|
51
|
-
*
|
|
52
|
-
*
|
|
53
|
-
*
|
|
54
|
-
*
|
|
55
|
-
* return (
|
|
56
|
-
* <Grid>
|
|
57
|
-
* {items.map(car => <Card key={car.id} title={car.make} />)}
|
|
58
|
-
* </Grid>
|
|
59
|
-
* );
|
|
51
|
+
* // processed has filters + search + sort applied
|
|
52
|
+
* const { processed, loading } = useCrudCardList(carEntity, {
|
|
53
|
+
* searchQuery: search,
|
|
54
|
+
* });
|
|
60
55
|
* ```
|
|
61
56
|
*/
|
|
62
57
|
export declare function useCrudCardList<T extends {
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"useCrudCardList.d.ts","sourceRoot":"","sources":["../src/useCrudCardList.ts"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"useCrudCardList.d.ts","sourceRoot":"","sources":["../src/useCrudCardList.ts"],"names":[],"mappings":"AAcA,OAAO,KAAK,EAAE,MAAM,EAAE,aAAa,EAAE,YAAY,EAAE,MAAM,gBAAgB,CAAC;AAM1E,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,WAAW,CAAC;AAQhD,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,8BAA8B,CAAC;AAErE,MAAM,WAAW,sBAAsB,CAAC,CAAC,CAAE,SAAQ,cAAc,CAAC,CAAC,CAAC;IAClE,wDAAwD;IACxD,OAAO,CAAC,EAAE,OAAO,CAAC;IAClB,mDAAmD;IACnD,YAAY,CAAC,EAAE,YAAY,CAAC;IAC5B,uDAAuD;IACvD,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,qFAAqF;IACrF,UAAU,CAAC,EAAE,gBAAgB,GAAG,IAAI,CAAC;CACtC;AAED,MAAM,WAAW,qBAAqB,CAAC,CAAC;IACtC,MAAM,EAAE,aAAa,CAAC;IACtB,IAAI,EAAE;QAAE,KAAK,EAAE,CAAC,EAAE,CAAA;KAAE,GAAG,IAAI,CAAC;IAC5B,4FAA4F;IAC5F,KAAK,EAAE,CAAC,EAAE,CAAC;IACX,wEAAwE;IACxE,SAAS,EAAE,CAAC,EAAE,CAAC;IACf,gEAAgE;IAChE,aAAa,EAAE,MAAM,CAAC;IACtB,iDAAiD;IACjD,OAAO,EAAE,OAAO,CAAC;IACjB,6DAA6D;IAC7D,QAAQ,EAAE,OAAO,CAAC;IAClB,KAAK,EAAE,KAAK,GAAG,IAAI,CAAC;IACpB,8BAA8B;IAC9B,MAAM,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC;IAC5B,wCAAwC;IACxC,OAAO,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC;IAC7B,WAAW,EAAE,OAAO,CAAC;CACtB;AAED;;;;;;;;;;;;;;;;;;;;GAoBG;AACH,wBAAgB,eAAe,CAAC,CAAC,SAAS;IAAE,EAAE,EAAE,MAAM,CAAA;CAAE,GAAG;IAAE,EAAE,EAAE,MAAM,CAAA;CAAE,EACvE,kBAAkB,EAAE,MAAM,GAAG,MAAM,EACnC,OAAO,GAAE,sBAAsB,CAAC,CAAC,CAAM,GACtC,qBAAqB,CAAC,CAAC,CAAC,CAuE1B"}
|
package/dist/useCrudCardList.js
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
import{useBaseCrudList as
|
|
1
|
+
import{useMemo as o}from"react";import{useBaseCrudList as f}from"./useBaseCrudList";import{useCrudFilters as h}from"./hooks/useCrudFilters";import{applyFilters as y,applySearch as g,applySort as p}from"./utils/clientListProcessing";function v(l,t={}){const u={enabled:t.enabled,queryOptions:t.queryOptions,schema:t.schema,entity:t.entity,staleTime:t.staleTime,noCache:t.noCache},e=f(l,"listCard",null,u),s=typeof l=="string"?null:l,c=s?.collection??l,{filters:a}=h({collection:c}),i=t.clientSort===null?null:t.clientSort??s?.defaultSort??null,{processed:n,filteredTotal:m}=o(()=>{let r=e.items;s&&Object.keys(a).length>0&&(r=y(r,a,s)),s&&t.searchQuery&&(r=g(r,t.searchQuery,s));const d=r.length;return i&&(r=p(r,i)),{processed:r,filteredTotal:d}},[e.items,a,t.searchQuery,i,s]);return{status:e.status,data:e.data?{items:e.data.items}:null,items:e.items,processed:n,filteredTotal:m,loading:e.loading,fetching:e.fetching,error:e.error,mutate:e.mutate,refresh:e.refresh,isAvailable:e.isAvailable}}export{v as useCrudCardList};
|
package/dist/useCrudList.d.ts
CHANGED
|
@@ -1,15 +1,7 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @fileoverview useCrudList Hook - Collection Management with Pagination
|
|
3
|
-
* @description Specialized hook for fetching and managing lists of entities.
|
|
4
|
-
* Supports both client-side and server-side pagination.
|
|
5
|
-
*
|
|
6
|
-
* @version 0.0.10
|
|
7
|
-
* @since 0.0.1
|
|
8
|
-
* @author AMBROISE PARK Consulting
|
|
9
|
-
*/
|
|
10
1
|
import type { Entity, FeatureStatus } from '@donotdev/core';
|
|
11
2
|
import type { QueryOptions } from '@donotdev/core';
|
|
12
3
|
import type { UseCrudOptions } from './useCrud';
|
|
4
|
+
import type { ClientSortConfig } from './utils/clientListProcessing';
|
|
13
5
|
export interface UseCrudListOptions<T> extends UseCrudOptions<T> {
|
|
14
6
|
/** Automatically fetch data on mount (default: true) */
|
|
15
7
|
enabled?: boolean;
|
|
@@ -21,6 +13,10 @@ export interface UseCrudListOptions<T> extends UseCrudOptions<T> {
|
|
|
21
13
|
pageSize?: number;
|
|
22
14
|
/** Where/orderBy/limit passed to list query (merged with pagination on server) */
|
|
23
15
|
queryOptions?: QueryOptions;
|
|
16
|
+
/** Client-side search query. Applied after filters. */
|
|
17
|
+
searchQuery?: string;
|
|
18
|
+
/** Client-side sort config. Defaults to entity.defaultSort. Pass null to disable. */
|
|
19
|
+
clientSort?: ClientSortConfig | null;
|
|
24
20
|
}
|
|
25
21
|
export interface UseCrudListReturn<T> {
|
|
26
22
|
status: FeatureStatus;
|
|
@@ -28,8 +24,12 @@ export interface UseCrudListReturn<T> {
|
|
|
28
24
|
items: T[];
|
|
29
25
|
total?: number;
|
|
30
26
|
} | null;
|
|
31
|
-
/** Convenience: items array (data?.items ?? []) */
|
|
27
|
+
/** Convenience: items array (data?.items ?? []) — raw + optimistic, no client processing */
|
|
32
28
|
items: T[];
|
|
29
|
+
/** Items after client-side filters + search + sort — ready to render */
|
|
30
|
+
processed: T[];
|
|
31
|
+
/** Count of items after filters + search (before pagination) */
|
|
32
|
+
filteredTotal: number;
|
|
33
33
|
/** True during initial fetch (show skeletons) */
|
|
34
34
|
loading: boolean;
|
|
35
35
|
/** True during background refetch (show subtle indicator) */
|
|
@@ -56,15 +56,16 @@ export interface UseCrudListReturn<T> {
|
|
|
56
56
|
*
|
|
57
57
|
* @example
|
|
58
58
|
* ```typescript
|
|
59
|
-
*
|
|
60
|
-
*
|
|
61
|
-
*
|
|
59
|
+
* // Basic usage — processed has filters + search + sort applied
|
|
60
|
+
* const { processed, loading, refresh } = useCrudList(carEntity, {
|
|
61
|
+
* searchQuery: search,
|
|
62
|
+
* });
|
|
62
63
|
*
|
|
63
|
-
*
|
|
64
|
-
*
|
|
65
|
-
*
|
|
66
|
-
*
|
|
67
|
-
* );
|
|
64
|
+
* // With explicit sort
|
|
65
|
+
* const { processed } = useCrudList(carEntity, {
|
|
66
|
+
* searchQuery: search,
|
|
67
|
+
* clientSort: { field: 'price', direction: 'asc' },
|
|
68
|
+
* });
|
|
68
69
|
* ```
|
|
69
70
|
*/
|
|
70
71
|
export declare function useCrudList<T extends {
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"useCrudList.d.ts","sourceRoot":"","sources":["../src/useCrudList.ts"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"useCrudList.d.ts","sourceRoot":"","sources":["../src/useCrudList.ts"],"names":[],"mappings":"AAcA,OAAO,KAAK,EAAE,MAAM,EAAE,aAAa,EAAE,MAAM,gBAAgB,CAAC;AAO5D,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,gBAAgB,CAAC;AACnD,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,WAAW,CAAC;AAQhD,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,8BAA8B,CAAC;AAErE,MAAM,WAAW,kBAAkB,CAAC,CAAC,CAAE,SAAQ,cAAc,CAAC,CAAC,CAAC;IAC9D,wDAAwD;IACxD,OAAO,CAAC,EAAE,OAAO,CAAC;IAClB,iFAAiF;IACjF,UAAU,CAAC,EAAE,QAAQ,GAAG,QAAQ,CAAC;IACjC,qEAAqE;IACrE,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,yDAAyD;IACzD,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,kFAAkF;IAClF,YAAY,CAAC,EAAE,YAAY,CAAC;IAC5B,uDAAuD;IACvD,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,qFAAqF;IACrF,UAAU,CAAC,EAAE,gBAAgB,GAAG,IAAI,CAAC;CACtC;AAED,MAAM,WAAW,iBAAiB,CAAC,CAAC;IAClC,MAAM,EAAE,aAAa,CAAC;IACtB,IAAI,EAAE;QAAE,KAAK,EAAE,CAAC,EAAE,CAAC;QAAC,KAAK,CAAC,EAAE,MAAM,CAAA;KAAE,GAAG,IAAI,CAAC;IAC5C,4FAA4F;IAC5F,KAAK,EAAE,CAAC,EAAE,CAAC;IACX,wEAAwE;IACxE,SAAS,EAAE,CAAC,EAAE,CAAC;IACf,gEAAgE;IAChE,aAAa,EAAE,MAAM,CAAC;IACtB,iDAAiD;IACjD,OAAO,EAAE,OAAO,CAAC;IACjB,6DAA6D;IAC7D,QAAQ,EAAE,OAAO,CAAC;IAClB,KAAK,EAAE,KAAK,GAAG,IAAI,CAAC;IACpB,8BAA8B;IAC9B,MAAM,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC;IAC5B,wCAAwC;IACxC,OAAO,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC;IAC7B,WAAW,EAAE,OAAO,CAAC;CACtB;AAED;;;;;;;;;;;;;;;;;;;;;;;;;;GA0BG;AACH,wBAAgB,WAAW,CAAC,CAAC,SAAS;IAAE,EAAE,EAAE,MAAM,CAAA;CAAE,GAAG;IAAE,EAAE,EAAE,MAAM,CAAA;CAAE,EACnE,kBAAkB,EAAE,MAAM,GAAG,MAAM,EACnC,OAAO,GAAE,kBAAkB,CAAC,CAAC,CAAM,GAClC,iBAAiB,CAAC,CAAC,CAAC,CAwEtB"}
|
package/dist/useCrudList.js
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
import{useBaseCrudList as
|
|
1
|
+
import{useMemo as h}from"react";import{useBaseCrudList as y}from"./useBaseCrudList";import{useCrudFilters as S}from"./hooks/useCrudFilters";import{applyFilters as b,applySearch as T,applySort as Q}from"./utils/clientListProcessing";function L(s,e={}){const n=e.pagination??"client",u=e.page??1,i=e.pageSize??10,o={mode:n,page:u,pageSize:i},m={enabled:e.enabled,queryOptions:e.queryOptions,schema:e.schema,entity:e.entity,staleTime:e.staleTime,noCache:e.noCache},a=y(s,"list",o,m),r=typeof s=="string"?null:s,d=r?.collection??s,{filters:l}=S({collection:d}),c=e.clientSort===null?null:e.clientSort??r?.defaultSort??null,{processed:f,filteredTotal:p}=h(()=>{let t=a.items;r&&Object.keys(l).length>0&&(t=b(t,l,r)),r&&e.searchQuery&&(t=T(t,e.searchQuery,r));const g=t.length;return c&&(t=Q(t,c)),{processed:t,filteredTotal:g}},[a.items,l,e.searchQuery,c,r]);return{...a,processed:f,filteredTotal:p}}export{L as useCrudList};
|
|
@@ -0,0 +1,82 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @fileoverview Client-side list processing utilities
|
|
3
|
+
* @description Pure functions for filtering, searching, and sorting entity items.
|
|
4
|
+
* Used by useCrudList/useCrudCardList hooks and UI components.
|
|
5
|
+
*
|
|
6
|
+
* @version 0.0.1
|
|
7
|
+
* @since 0.0.1
|
|
8
|
+
* @author AMBROISE PARK Consulting
|
|
9
|
+
*/
|
|
10
|
+
import type { Entity } from '@donotdev/core';
|
|
11
|
+
import type { FilterState } from '../types';
|
|
12
|
+
/**
|
|
13
|
+
* Sort configuration for client-side sorting.
|
|
14
|
+
*
|
|
15
|
+
* @example
|
|
16
|
+
* ```typescript
|
|
17
|
+
* const sort: ClientSortConfig = { field: 'price', direction: 'desc' };
|
|
18
|
+
* ```
|
|
19
|
+
*/
|
|
20
|
+
export interface ClientSortConfig {
|
|
21
|
+
field: string;
|
|
22
|
+
direction?: 'asc' | 'desc';
|
|
23
|
+
}
|
|
24
|
+
/**
|
|
25
|
+
* Apply filter state to items using entity field type metadata.
|
|
26
|
+
* Typically called automatically by `useCrudList`/`useCrudCardList` via `processed`.
|
|
27
|
+
* Export available for advanced use cases (e.g. filtering outside React).
|
|
28
|
+
*
|
|
29
|
+
* @param items - Raw items array
|
|
30
|
+
* @param filters - Current filter state from CrudStore
|
|
31
|
+
* @param entity - Entity definition (for field type lookup)
|
|
32
|
+
* @returns Filtered items
|
|
33
|
+
*
|
|
34
|
+
* @example
|
|
35
|
+
* ```typescript
|
|
36
|
+
* import { applyFilters } from '@donotdev/crud';
|
|
37
|
+
* const filtered = applyFilters(items, { status: 'active' }, productEntity);
|
|
38
|
+
* ```
|
|
39
|
+
*/
|
|
40
|
+
export declare function applyFilters<T extends Record<string, any>>(items: T[], filters: FilterState, entity: Entity): T[];
|
|
41
|
+
/**
|
|
42
|
+
* Returns the list of searchable field names for an entity.
|
|
43
|
+
* Uses entity.search.fields if defined, otherwise auto-detects text-like visible fields.
|
|
44
|
+
*
|
|
45
|
+
* @param entity - Entity definition
|
|
46
|
+
* @returns Array of field names to search
|
|
47
|
+
*/
|
|
48
|
+
export declare function getSearchableFields(entity: Entity): string[];
|
|
49
|
+
/**
|
|
50
|
+
* Apply search query to items.
|
|
51
|
+
* Searches fields defined by `entity.search.fields` or auto-detected text-like visible fields.
|
|
52
|
+
* Typically called automatically by `useCrudList`/`useCrudCardList` when `searchQuery` option is set.
|
|
53
|
+
*
|
|
54
|
+
* @param items - Items to search
|
|
55
|
+
* @param query - Search query string (case-insensitive contains)
|
|
56
|
+
* @param entity - Entity definition (for field config and search.fields lookup)
|
|
57
|
+
* @returns Filtered items matching search
|
|
58
|
+
*
|
|
59
|
+
* @example
|
|
60
|
+
* ```typescript
|
|
61
|
+
* import { applySearch } from '@donotdev/crud';
|
|
62
|
+
* const results = applySearch(items, 'bmw', carEntity);
|
|
63
|
+
* ```
|
|
64
|
+
*/
|
|
65
|
+
export declare function applySearch<T extends Record<string, any>>(items: T[], query: string, entity: Entity): T[];
|
|
66
|
+
/**
|
|
67
|
+
* Apply client-side sorting to items.
|
|
68
|
+
* Handles null/undefined values by sorting them last.
|
|
69
|
+
* Supports number, string, Date, and price ({ amount }) comparisons.
|
|
70
|
+
*
|
|
71
|
+
* @param items - Items to sort
|
|
72
|
+
* @param sort - Sort configuration (field + direction)
|
|
73
|
+
* @returns Sorted items (new array, original unchanged)
|
|
74
|
+
*
|
|
75
|
+
* @example
|
|
76
|
+
* ```typescript
|
|
77
|
+
* import { applySort } from '@donotdev/crud';
|
|
78
|
+
* const sorted = applySort(items, { field: 'createdAt', direction: 'desc' });
|
|
79
|
+
* ```
|
|
80
|
+
*/
|
|
81
|
+
export declare function applySort<T extends Record<string, any>>(items: T[], sort: ClientSortConfig): T[];
|
|
82
|
+
//# sourceMappingURL=clientListProcessing.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"clientListProcessing.d.ts","sourceRoot":"","sources":["../../src/utils/clientListProcessing.ts"],"names":[],"mappings":"AAEA;;;;;;;;GAQG;AAEH,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,gBAAgB,CAAC;AAI7C,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,UAAU,CAAC;AAE5C;;;;;;;GAOG;AACH,MAAM,WAAW,gBAAgB;IAC/B,KAAK,EAAE,MAAM,CAAC;IACd,SAAS,CAAC,EAAE,KAAK,GAAG,MAAM,CAAC;CAC5B;AAED;;;;;;;;;;;;;;;GAeG;AACH,wBAAgB,YAAY,CAAC,CAAC,SAAS,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,EACxD,KAAK,EAAE,CAAC,EAAE,EACV,OAAO,EAAE,WAAW,EACpB,MAAM,EAAE,MAAM,GACb,CAAC,EAAE,CAWL;AAED;;;;;;GAMG;AACH,wBAAgB,mBAAmB,CAAC,MAAM,EAAE,MAAM,GAAG,MAAM,EAAE,CAyB5D;AAED;;;;;;;;;;;;;;;GAeG;AACH,wBAAgB,WAAW,CAAC,CAAC,SAAS,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,EACvD,KAAK,EAAE,CAAC,EAAE,EACV,KAAK,EAAE,MAAM,EACb,MAAM,EAAE,MAAM,GACb,CAAC,EAAE,CA+BL;AAED;;;;;;;;;;;;;;GAcG;AACH,wBAAgB,SAAS,CAAC,CAAC,SAAS,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,EACrD,KAAK,EAAE,CAAC,EAAE,EACV,IAAI,EAAE,gBAAgB,GACrB,CAAC,EAAE,CAoCL"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
import{matchesFilter as u}from"./matchesFilter";import{getValueType as c}from"../fieldTypeRegistry";function g(t,l,s){const o=Object.entries(l);return o.length===0?t:t.filter(r=>o.every(([i,f])=>{const n=s.fields[i]?.type||"text";return u(r[i],f,n)}))}function a(t){if(t.search?.fields&&t.search.fields.length>0)return t.search.fields;const l=new Set(["string"]),s=[];for(const[o,r]of Object.entries(t.fields)){if(r.visibility==="hidden"||r.visibility==="technical")continue;const i=c(r.type);i&&l.has(i)&&s.push(o)}return s}function h(t,l,s){if(!l||l.trim()==="")return t;const o=l.toLowerCase(),r=a(s);return r.length===0?t.filter(i=>Object.values(i).some(f=>f==null?!1:String(f).toLowerCase().includes(o))):t.filter(i=>r.some(f=>{const e=i[f];return e==null?!1:typeof e=="object"&&"formatted_address"in e?String(e.formatted_address).toLowerCase().includes(o):String(e).toLowerCase().includes(o)}))}function b(t,l){const{field:s,direction:o="asc"}=l,r=o==="desc"?-1:1;return[...t].sort((i,f)=>{const e=i[s],n=f[s];return e==null&&n==null?0:e==null?1:n==null?-1:typeof e=="number"&&typeof n=="number"?(e-n)*r:typeof e=="object"&&"amount"in e&&typeof n=="object"&&"amount"in n?(e.amount-n.amount)*r:e instanceof Date&&n instanceof Date?(e.getTime()-n.getTime())*r:String(e).localeCompare(String(n))*r})}export{g as applyFilters,h as applySearch,b as applySort,a as getSearchableFields};
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"fileStorage.d.ts","sourceRoot":"","sources":["../../src/utils/fileStorage.ts"],"names":[],"mappings":"AAgBA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAC;AAEhD,iGAAiG;AACjG,KAAK,sBAAsB,GAAG,CAAC,QAAQ,EAAE;IACvC,gBAAgB,EAAE,MAAM,CAAC;IACzB,UAAU,EAAE,MAAM,CAAC;IACnB,QAAQ,EAAE,MAAM,CAAC;CAClB,KAAK,IAAI,CAAC;AAEX,YAAY,EAAE,SAAS,EAAE,CAAC;AAE1B,MAAM,WAAW,cAAc;IAC7B,gBAAgB,EAAE,MAAM,CAAC;IACzB,UAAU,EAAE,MAAM,CAAC;IACnB,QAAQ,EAAE,MAAM,CAAC;CAClB;AAED,MAAM,WAAW,iBAAiB;IAChC,8CAA8C;IAC9C,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,wDAAwD;IACxD,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,wBAAwB;IACxB,UAAU,CAAC,EAAE,sBAAsB,CAAC;CACrC;AAYD;;;GAGG;AACH,wBAAsB,UAAU,CAC9B,IAAI,EAAE,IAAI,EACV,OAAO,GAAE,iBAAsB,GAC9B,OAAO,CAAC,SAAS,CAAC,
|
|
1
|
+
{"version":3,"file":"fileStorage.d.ts","sourceRoot":"","sources":["../../src/utils/fileStorage.ts"],"names":[],"mappings":"AAgBA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAC;AAEhD,iGAAiG;AACjG,KAAK,sBAAsB,GAAG,CAAC,QAAQ,EAAE;IACvC,gBAAgB,EAAE,MAAM,CAAC;IACzB,UAAU,EAAE,MAAM,CAAC;IACnB,QAAQ,EAAE,MAAM,CAAC;CAClB,KAAK,IAAI,CAAC;AAEX,YAAY,EAAE,SAAS,EAAE,CAAC;AAE1B,MAAM,WAAW,cAAc;IAC7B,gBAAgB,EAAE,MAAM,CAAC;IACzB,UAAU,EAAE,MAAM,CAAC;IACnB,QAAQ,EAAE,MAAM,CAAC;CAClB;AAED,MAAM,WAAW,iBAAiB;IAChC,8CAA8C;IAC9C,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,wDAAwD;IACxD,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,wBAAwB;IACxB,UAAU,CAAC,EAAE,sBAAsB,CAAC;CACrC;AAYD;;;GAGG;AACH,wBAAsB,UAAU,CAC9B,IAAI,EAAE,IAAI,EACV,OAAO,GAAE,iBAAsB,GAC9B,OAAO,CAAC,SAAS,CAAC,CAmDpB;AAED;;GAEG;AACH,wBAAsB,WAAW,CAC/B,KAAK,EAAE,IAAI,EAAE,EACb,OAAO,GAAE,iBAAsB,GAC9B,OAAO,CAAC,SAAS,EAAE,CAAC,CA4BtB;AAED;;GAEG;AACH,wBAAsB,UAAU,CAAC,KAAK,EAAE,SAAS,GAAG,OAAO,CAAC,IAAI,CAAC,CAgBhE;AAED;;GAEG;AACH,wBAAgB,cAAc,IAAI,MAAM,CAEvC;AAED;;GAEG;AACH,wBAAgB,gBAAgB,CAAC,QAAQ,EAAE,MAAM,GAAG,MAAM,CAGzD;AAED;;GAEG;AACH,wBAAgB,WAAW,CAAC,QAAQ,CAAC,EAAE,MAAM,EAAE,QAAQ,CAAC,EAAE,MAAM,GAAG,MAAM,CA4CxE;AAED;;GAEG;AACH,wBAAgB,cAAc,CAAC,KAAK,EAAE,MAAM,GAAG,MAAM,CAMpD;AAED;;GAEG;AACH,eAAO,MAAM,mBAAmB,UAY/B,CAAC;AAEF;;GAEG;AACH,eAAO,MAAM,mBAAmB,UAY/B,CAAC;AAEF;;GAEG;AACH,wBAAgB,uBAAuB,IAAI,MAAM,CAEhD"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"imageStorage.d.ts","sourceRoot":"","sources":["../../src/utils/imageStorage.ts"],"names":[],"mappings":"AAeA,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,gBAAgB,CAAC;AAE9C,iGAAiG;AACjG,KAAK,sBAAsB,GAAG,CAAC,QAAQ,EAAE;IACvC,gBAAgB,EAAE,MAAM,CAAC;IACzB,UAAU,EAAE,MAAM,CAAC;IACnB,QAAQ,EAAE,MAAM,CAAC;CAClB,KAAK,IAAI,CAAC;AAEX,YAAY,EAAE,OAAO,EAAE,CAAC;AAExB,MAAM,WAAW,cAAc;IAC7B,gBAAgB,EAAE,MAAM,CAAC;IACzB,UAAU,EAAE,MAAM,CAAC;IACnB,QAAQ,EAAE,MAAM,CAAC;CAClB;AAED,MAAM,WAAW,kBAAkB;IACjC,+CAA+C;IAC/C,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,wDAAwD;IACxD,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,wBAAwB;IACxB,UAAU,CAAC,EAAE,sBAAsB,CAAC;CACrC;AAYD;;;GAGG;AACH,wBAAsB,WAAW,CAC/B,QAAQ,EAAE,IAAI,EACd,SAAS,EAAE,IAAI,EACf,gBAAgB,EAAE,MAAM,EACxB,OAAO,GAAE,kBAAuB,GAC/B,OAAO,CAAC,OAAO,CAAC,
|
|
1
|
+
{"version":3,"file":"imageStorage.d.ts","sourceRoot":"","sources":["../../src/utils/imageStorage.ts"],"names":[],"mappings":"AAeA,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,gBAAgB,CAAC;AAE9C,iGAAiG;AACjG,KAAK,sBAAsB,GAAG,CAAC,QAAQ,EAAE;IACvC,gBAAgB,EAAE,MAAM,CAAC;IACzB,UAAU,EAAE,MAAM,CAAC;IACnB,QAAQ,EAAE,MAAM,CAAC;CAClB,KAAK,IAAI,CAAC;AAEX,YAAY,EAAE,OAAO,EAAE,CAAC;AAExB,MAAM,WAAW,cAAc;IAC7B,gBAAgB,EAAE,MAAM,CAAC;IACzB,UAAU,EAAE,MAAM,CAAC;IACnB,QAAQ,EAAE,MAAM,CAAC;CAClB;AAED,MAAM,WAAW,kBAAkB;IACjC,+CAA+C;IAC/C,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,wDAAwD;IACxD,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,wBAAwB;IACxB,UAAU,CAAC,EAAE,sBAAsB,CAAC;CACrC;AAYD;;;GAGG;AACH,wBAAsB,WAAW,CAC/B,QAAQ,EAAE,IAAI,EACd,SAAS,EAAE,IAAI,EACf,gBAAgB,EAAE,MAAM,EACxB,OAAO,GAAE,kBAAuB,GAC/B,OAAO,CAAC,OAAO,CAAC,CA0HlB;AAED;;GAEG;AACH,wBAAsB,WAAW,CAAC,OAAO,EAAE,OAAO,GAAG,OAAO,CAAC,IAAI,CAAC,CAqBjE"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
import{handleError as F,hasProvider as
|
|
1
|
+
import{handleError as F,hasProvider as f,getProvider as h}from"@donotdev/core";function R(e){return e.replace(/[^a-zA-Z0-9.-]/g,"_").replace(/_{2,}/g,"_").toLowerCase()}async function B(e,t,o,y={}){const{storagePath:n="uploads/images",filename:P,onProgress:a}=y,U=Date.now(),l=(P||`${U}_${R(o)}`).replace(/\.[^/.]+$/,""),d=`${n}/${l}_full.webp`,g=`${n}/${l}_thumb.webp`;try{const c=new File([e],`${l}_full.webp`,{type:"image/webp"}),s=new File([t],`${l}_thumb.webp`,{type:"image/webp"});let i,u;if(f("storage")){const r=h("storage");i=(await r.upload(c,{storagePath:n,filename:`${l}_full.webp`,onProgress:a?b=>a({bytesTransferred:0,totalBytes:0,progress:b*.5}):void 0})).url,u=(await r.upload(s,{storagePath:n,filename:`${l}_thumb.webp`,onProgress:a?b=>a({bytesTransferred:0,totalBytes:0,progress:50+b*.5}):void 0})).url}else{const{uploadFileResumable:r}=await import("@donotdev/firebase"),w=a?m=>{a({bytesTransferred:m.bytesTransferred,totalBytes:m.totalBytes*2,progress:m.progress*.5})}:void 0,{promise:p}=r(c,{basePath:n,filename:`${l}_full.webp`,onProgress:w,metadata:{contentType:"image/webp"}});i=(await p).url;const $=a?m=>{a({bytesTransferred:m.bytesTransferred+e.size,totalBytes:e.size+t.size,progress:50+m.progress*.5})}:void 0,{promise:_}=r(s,{basePath:n,filename:`${l}_thumb.webp`,onProgress:$,metadata:{contentType:"image/webp"}});u=(await _).url}return{fullUrl:i,thumbUrl:u}}catch(c){try{if(f("storage")){const s=h("storage");await s.delete(d).catch(()=>{}),await s.delete(g).catch(()=>{})}else{const{getFileUrl:s,deleteFileByUrl:i}=await import("@donotdev/firebase"),u=await s(d).catch(()=>null),r=await s(g).catch(()=>null);u&&await i(u).catch(()=>{}),r&&await i(r).catch(()=>{})}}catch{}throw c}}async function v(e){try{if(f("storage")){const t=h("storage"),o=[t.delete(e.fullUrl)];e.thumbUrl&&o.push(t.delete(e.thumbUrl)),await Promise.all(o)}else{const{deleteFileByUrl:t}=await import("@donotdev/firebase"),o=[t(e.fullUrl)];e.thumbUrl&&o.push(t(e.thumbUrl)),await Promise.all(o)}}catch(t){throw F(t,{userMessage:"Failed to delete image",severity:"error",context:{fullUrl:e.fullUrl,thumbUrl:e.thumbUrl}}),t}}export{v as deleteImage,B as uploadImage};
|
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Pure filter matching utility — checks if an item value matches a filter value.
|
|
3
|
+
*
|
|
4
|
+
* Supports all built-in field types: text (contains), number/price (range),
|
|
5
|
+
* date/datetime (range), select (exact), multiselect (contains), address (formatted_address).
|
|
6
|
+
*
|
|
7
|
+
* Null/undefined handling for range filters:
|
|
8
|
+
* - MIN only → exclude null/undefined
|
|
9
|
+
* - MAX only → include null/undefined
|
|
10
|
+
* - MIN + MAX → exclude null/undefined
|
|
11
|
+
*
|
|
12
|
+
* @param itemValue - The item's field value to test
|
|
13
|
+
* @param filterValue - The filter criterion (string, range, date, array)
|
|
14
|
+
* @param fieldType - Entity field type (e.g. 'text', 'number', 'date', 'price')
|
|
15
|
+
* @returns true if item passes the filter
|
|
16
|
+
*
|
|
17
|
+
* @example
|
|
18
|
+
* ```typescript
|
|
19
|
+
* import { matchesFilter } from '@donotdev/crud';
|
|
20
|
+
*
|
|
21
|
+
* matchesFilter('hello world', 'hello', 'text'); // true (contains)
|
|
22
|
+
* matchesFilter(42, { min: '10', max: '50' }, 'number'); // true (in range)
|
|
23
|
+
* matchesFilter(null, { min: '10' }, 'number'); // false (null excluded)
|
|
24
|
+
* ```
|
|
25
|
+
*/
|
|
26
|
+
export declare function matchesFilter(itemValue: any, filterValue: string | {
|
|
27
|
+
min?: string;
|
|
28
|
+
max?: string;
|
|
29
|
+
} | string[] | {
|
|
30
|
+
date: string;
|
|
31
|
+
time?: string;
|
|
32
|
+
} | {
|
|
33
|
+
min?: {
|
|
34
|
+
date: string;
|
|
35
|
+
time?: string;
|
|
36
|
+
};
|
|
37
|
+
max?: {
|
|
38
|
+
date: string;
|
|
39
|
+
time?: string;
|
|
40
|
+
};
|
|
41
|
+
} | {
|
|
42
|
+
date: string;
|
|
43
|
+
time?: string;
|
|
44
|
+
}[], fieldType: string): boolean;
|
|
45
|
+
//# sourceMappingURL=matchesFilter.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"matchesFilter.d.ts","sourceRoot":"","sources":["../../src/utils/matchesFilter.ts"],"names":[],"mappings":"AAcA;;;;;;;;;;;;;;;;;;;;;;;;GAwBG;AACH,wBAAgB,aAAa,CAC3B,SAAS,EAAE,GAAG,EAEd,WAAW,EACP,MAAM,GACN;IAAE,GAAG,CAAC,EAAE,MAAM,CAAC;IAAC,GAAG,CAAC,EAAE,MAAM,CAAA;CAAE,GAC9B,MAAM,EAAE,GACR;IAAE,IAAI,EAAE,MAAM,CAAC;IAAC,IAAI,CAAC,EAAE,MAAM,CAAA;CAAE,GAC/B;IACE,GAAG,CAAC,EAAE;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,IAAI,CAAC,EAAE,MAAM,CAAA;KAAE,CAAC;IACtC,GAAG,CAAC,EAAE;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,IAAI,CAAC,EAAE,MAAM,CAAA;KAAE,CAAC;CACvC,GACD;IAAE,IAAI,EAAE,MAAM,CAAC;IAAC,IAAI,CAAC,EAAE,MAAM,CAAA;CAAE,EAAE,EAErC,SAAS,EAAE,MAAM,GAChB,OAAO,CAiVT"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
import{getFilterType as w}from"../fieldTypeRegistry";function T(n,t,e){if(!t)return!0;if(Array.isArray(t)){if(t.length===0)return!0;if(!(w(e)==="range"&&(e==="date"||e==="datetime-local"||e==="timestamp")))return t.some(s=>{if(typeof s=="object"&&"date"in s){const i=s.date;return(n instanceof Date?n.toISOString().split("T")[0]:new Date(n).toISOString().split("T")[0])===i}return String(n)===String(s)});const f=n instanceof Date?n.toISOString().split("T")[0]:new Date(n).toISOString().split("T")[0];return t.some(s=>typeof s=="string"?f===s:typeof s=="object"&&"date"in s?f===s.date:!1)}if(typeof t=="object"&&"date"in t&&!("min"in t)&&!Array.isArray(t)){if(!(w(e)==="range"&&(e==="date"||e==="datetime-local"||e==="timestamp")))return!1;const f=t.date;if(!f)return!0;if(n==null)return!1;const s=n instanceof Date?n.toISOString().split("T")[0]:new Date(n).toISOString().split("T")[0];if((e==="datetime-local"||e==="timestamp")&&t.time){const i=n instanceof Date?n.toISOString().slice(11,16):new Date(n).toISOString().slice(11,16);return s===f&&i===t.time}return s===f}if(typeof t=="object"&&"min"in t){const D=w(e);if(!D)return!0;const p=D==="range"&&(e==="date"||e==="datetime-local"||e==="timestamp"),f=D==="range"&&!p,s=e==="price",i=s&&n!=null&&typeof n=="object"?Number(n.amount):f&&n!=null?typeof n=="number"?n:Number(n):NaN;if(f||s){if(typeof t.min=="object"&&t.min!==null&&"date"in t.min)return!1;const r=t,g=typeof r.min=="string"&&r.min!=="",S=typeof r.max=="string"&&r.max!=="";return g&&!S?n==null||isNaN(i)?!1:i>=Number(r.min):!g&&S?n==null?!0:isNaN(i)?!1:i<=Number(r.max):g&&S?n==null||isNaN(i)?!1:i>=Number(r.min)&&i<=Number(r.max):!0}if(p){if(typeof t.min=="object"&&t.min!==null&&"date"in t.min&&!("min"in t.min)){const o=t,d=o.min,a=o.max,u=d?.date&&d.date!=="",b=a?.date&&a.date!=="";if(u&&!b){if(n==null)return!1;const c=n instanceof Date?n:new Date(n);if(isNaN(c.getTime()))return!1;const N=new Date(d.date+(d.time?`T${d.time}:00`:"T00:00:00"));return c>=N}if(!u&&b&&a){if(n==null)return!0;const c=n instanceof Date?n:new Date(n);if(isNaN(c.getTime()))return!1;const N=new Date(a.date+(a.time?`T${a.time}:00`:"T23:59:59"));return c<=N}if(u&&b&&a){if(n==null)return!1;const c=n instanceof Date?n:new Date(n);if(isNaN(c.getTime()))return!1;const N=new Date(d.date+(d.time?`T${d.time}:00`:"T00:00:00")),x=new Date(a.date+(a.time?`T${a.time}:00`:"T23:59:59"));return c>=N&&c<=x}return!0}const r=t,g=typeof r.min=="string"&&r.min!=="",S=typeof r.max=="string"&&r.max!=="";if(g&&!S){if(n==null)return!1;const o=n instanceof Date?n:new Date(n);return isNaN(o.getTime())?!1:o>=new Date(r.min)}if(!g&&S){if(n==null)return!0;const o=n instanceof Date?n:new Date(n);return isNaN(o.getTime())?!1:o<=new Date(r.max)}if(g&&S){if(n==null)return!1;const o=n instanceof Date?n:new Date(n);return isNaN(o.getTime())?!1:o>=new Date(r.min)&&o<=new Date(r.max)}return!0}}const m=w(e);if(!m)return!0;if(m==="address")return n==null?!1:typeof n=="object"&&"formatted_address"in n?String(n.formatted_address).toLowerCase().includes(String(t).toLowerCase()):!1;if(m==="multiselect"){if(!Array.isArray(n)||n.length===0)return!1;const D=String(t).toLowerCase();return n.some(p=>String(p).toLowerCase().includes(D))}return n==null?!1:String(n).toLowerCase().includes(String(t).toLowerCase())}export{T as matchesFilter};
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"mergeWithOptimistic.d.ts","sourceRoot":"","sources":["../../src/utils/mergeWithOptimistic.ts"],"names":[],"mappings":"AAEA;;;;;;;GAOG;AAEH,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,UAAU,CAAC;AAG/C,UAAU,MAAM;IACd,EAAE,EAAE,MAAM,CAAC;IACX,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAC;CACxB;AAED;;;;;;;;;GASG;AACH,wBAAgB,mBAAmB,CAAC,CAAC,SAAS,MAAM,EAClD,SAAS,EAAE,CAAC,EAAE,EACd,UAAU,EAAE,MAAM,CAAC,MAAM,EAAE,cAAc,CAAC,GACzC,CAAC,EAAE,
|
|
1
|
+
{"version":3,"file":"mergeWithOptimistic.d.ts","sourceRoot":"","sources":["../../src/utils/mergeWithOptimistic.ts"],"names":[],"mappings":"AAEA;;;;;;;GAOG;AAEH,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,UAAU,CAAC;AAG/C,UAAU,MAAM;IACd,EAAE,EAAE,MAAM,CAAC;IACX,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAC;CACxB;AAED;;;;;;;;;GASG;AACH,wBAAgB,mBAAmB,CAAC,CAAC,SAAS,MAAM,EAClD,SAAS,EAAE,CAAC,EAAE,EACd,UAAU,EAAE,MAAM,CAAC,MAAM,EAAE,cAAc,CAAC,GACzC,CAAC,EAAE,CAgCL"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"uploadValidation.d.ts","sourceRoot":"","sources":["../../src/utils/uploadValidation.ts"],"names":[],"mappings":"AAGA;;;;;;;;GAQG;AAEH,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,gBAAgB,CAAC;AAE9C;;;;GAIG;AACH,wBAAgB,YAAY,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAGjD;AAED;;;;;GAKG;AACH,wBAAgB,eAAe,CAAC,OAAO,EAAE,OAAO,GAAG,OAAO,CAMzD;AAED;;;;;;GAMG;AACH,wBAAgB,gBAAgB,CAAC,IAAI,EAAE,OAAO,EAAE,IAAI,SAAK,GAAG,MAAM,EAAE,
|
|
1
|
+
{"version":3,"file":"uploadValidation.d.ts","sourceRoot":"","sources":["../../src/utils/uploadValidation.ts"],"names":[],"mappings":"AAGA;;;;;;;;GAQG;AAEH,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,gBAAgB,CAAC;AAE9C;;;;GAIG;AACH,wBAAgB,YAAY,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAGjD;AAED;;;;;GAKG;AACH,wBAAgB,eAAe,CAAC,OAAO,EAAE,OAAO,GAAG,OAAO,CAMzD;AAED;;;;;;GAMG;AACH,wBAAgB,gBAAgB,CAAC,IAAI,EAAE,OAAO,EAAE,IAAI,SAAK,GAAG,MAAM,EAAE,CAqDnE;AAED;;;;GAIG;AACH,wBAAgB,WAAW,CAAC,IAAI,EAAE,OAAO,GAAG,OAAO,CAElD"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
"use client";function
|
|
1
|
+
"use client";function s(r){return!r||typeof r!="string"?!1:r.startsWith("https://")&&!r.startsWith("blob:")}function c(r){return!(!r||typeof r!="object"||!s(r.fullUrl)||r.thumbUrl&&!s(r.thumbUrl))}function u(r,o=""){const t=[];if(r==null)return t;if(typeof r=="string")return r.startsWith("blob:")&&t.push(o||"root"),t;if(typeof r!="object")return t;if(Array.isArray(r))r.forEach((e,n)=>{const l=o?`${o}[${n}]`:`[${n}]`;t.push(...u(e,l))});else{const e=r;if(e.fullUrl!==void 0&&e.thumbUrl!==void 0){const n=e.fullUrl,l=e.thumbUrl,f=typeof n=="string"&&!s(n),b=l&&typeof l=="string"&&!s(l);if(f||b){const i=o||"root";t.push(`${i}.fullUrl or ${i}.thumbUrl contains blob URL`)}}else Object.entries(e).forEach(([n,l])=>{const f=o?`${o}.${n}`:n;t.push(...u(l,f))})}return t}function h(r){return u(r).length>0}export{u as checkForBlobUrls,h as hasBlobUrls,s as isStorageUrl,c as validatePicture};
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@donotdev/crud",
|
|
3
|
-
"version": "0.0.
|
|
3
|
+
"version": "0.0.17",
|
|
4
4
|
"private": false,
|
|
5
5
|
"type": "module",
|
|
6
6
|
"license": "SEE LICENSE IN LICENSE.md",
|
|
@@ -47,19 +47,19 @@
|
|
|
47
47
|
"dev": "tsc --noEmit --watch --listFiles false --listEmittedFiles false",
|
|
48
48
|
"clean": "rimraf dist tsconfig.tsbuildinfo",
|
|
49
49
|
"lint": "eslint src/",
|
|
50
|
-
"type-check": "tsc --noEmit"
|
|
50
|
+
"type-check": "bunx tsc --noEmit"
|
|
51
51
|
},
|
|
52
52
|
"dependencies": {
|
|
53
53
|
"@dnd-kit/core": "^6.3.1",
|
|
54
54
|
"@dnd-kit/sortable": "^10.0.0",
|
|
55
55
|
"@dnd-kit/utilities": "^3.2.2",
|
|
56
|
-
"@donotdev/components": "^0.0.
|
|
57
|
-
"@donotdev/core": "^0.0.
|
|
56
|
+
"@donotdev/components": "^0.0.20",
|
|
57
|
+
"@donotdev/core": "^0.0.26",
|
|
58
58
|
"@hookform/resolvers": "^5.2.2",
|
|
59
59
|
"react-easy-crop": "^5.5.6"
|
|
60
60
|
},
|
|
61
61
|
"peerDependencies": {
|
|
62
|
-
"@donotdev/firebase": "^0.0.
|
|
62
|
+
"@donotdev/firebase": "^0.0.12",
|
|
63
63
|
"@tiptap/extension-placeholder": "^3.19.0",
|
|
64
64
|
"@tiptap/pm": "^3.19.0",
|
|
65
65
|
"@tiptap/react": "^3.19.0",
|
|
@@ -89,9 +89,6 @@
|
|
|
89
89
|
},
|
|
90
90
|
"firebase": {
|
|
91
91
|
"optional": true
|
|
92
|
-
},
|
|
93
|
-
"tiptap": {
|
|
94
|
-
"optional": true
|
|
95
92
|
}
|
|
96
93
|
}
|
|
97
94
|
}
|
|
@@ -1,34 +0,0 @@
|
|
|
1
|
-
export interface UseFormNavigationGuardOptions {
|
|
2
|
-
/** Whether the guard is enabled (default: true) */
|
|
3
|
-
enabled?: boolean;
|
|
4
|
-
/** Custom message for navigation confirmation */
|
|
5
|
-
message?: string;
|
|
6
|
-
}
|
|
7
|
-
/**
|
|
8
|
-
* Hook to block SPA navigation when forms are dirty.
|
|
9
|
-
* Uses FormStore as single source of truth.
|
|
10
|
-
*
|
|
11
|
-
* For React Router: Uses useBlocker
|
|
12
|
-
* For Next.js: Intercepts router.push calls
|
|
13
|
-
*
|
|
14
|
-
* @example
|
|
15
|
-
* ```tsx
|
|
16
|
-
* // In your app root/layout
|
|
17
|
-
* useFormNavigationGuard({
|
|
18
|
-
* message: 'You have unsaved changes. Are you sure you want to leave?',
|
|
19
|
-
* });
|
|
20
|
-
* ```
|
|
21
|
-
*/
|
|
22
|
-
export declare function useFormNavigationGuard({ enabled, message, }?: UseFormNavigationGuardOptions): void;
|
|
23
|
-
/**
|
|
24
|
-
* Check if navigation should be blocked and get confirmation.
|
|
25
|
-
* Used by router navigation hooks.
|
|
26
|
-
*
|
|
27
|
-
* NOTE: For safe access from @donotdev/ui, use checkFormNavigationSafe from useFormStoreSafe.
|
|
28
|
-
* This function is for direct use within CRUD package.
|
|
29
|
-
*
|
|
30
|
-
* @returns Promise<boolean> - true if navigation should proceed, false if blocked
|
|
31
|
-
*/
|
|
32
|
-
export declare function checkFormNavigation(message?: string): Promise<boolean>;
|
|
33
|
-
export default useFormNavigationGuard;
|
|
34
|
-
//# sourceMappingURL=useFormNavigationGuard.d.ts.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"useFormNavigationGuard.d.ts","sourceRoot":"","sources":["../../src/hooks/useFormNavigationGuard.ts"],"names":[],"mappings":"AAmBA,MAAM,WAAW,6BAA6B;IAC5C,mDAAmD;IACnD,OAAO,CAAC,EAAE,OAAO,CAAC;IAClB,iDAAiD;IACjD,OAAO,CAAC,EAAE,MAAM,CAAC;CAClB;AAED;;;;;;;;;;;;;;GAcG;AACH,wBAAgB,sBAAsB,CAAC,EACrC,OAAc,EACd,OAAqE,GACtE,GAAE,6BAAkC,GAAG,IAAI,CA0B3C;AAED;;;;;;;;GAQG;AACH,wBAAsB,mBAAmB,CACvC,OAAO,SAA4C,GAClD,OAAO,CAAC,OAAO,CAAC,CAOlB;AAED,eAAe,sBAAsB,CAAC"}
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
"use client";import{useEffect as a}from"react";import{isClient as u}from"@donotdev/core";import{useFormStore as i}from"../stores";function s({enabled:r=!0,message:e="You have unsaved changes. Are you sure you want to leave?"}={}){const n=i(t=>Object.values(t.forms).some(o=>o.isDirty));a(()=>{if(!r||!n||!u())return;const t=o=>(o.preventDefault(),o.returnValue=e,e);return window.addEventListener("beforeunload",t),()=>{window.removeEventListener("beforeunload",t)}},[r,n,e])}async function d(r="You have unsaved changes. Discard them?"){return!u()||!i.getState().hasDirtyForms()?!0:window.confirm(r)}var h=s;export{d as checkFormNavigation,h as default,s as useFormNavigationGuard};
|