@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.
Files changed (78) hide show
  1. package/dist/CrudService.d.ts.map +1 -1
  2. package/dist/CrudService.js +2 -2
  3. package/dist/CrudStore.d.ts.map +1 -1
  4. package/dist/FieldRegistry.d.ts +3 -30
  5. package/dist/FieldRegistry.d.ts.map +1 -1
  6. package/dist/adapters/FunctionsAdapter.d.ts.map +1 -1
  7. package/dist/adapters/FunctionsAdapter.js +1 -1
  8. package/dist/components/CrudCard.d.ts.map +1 -1
  9. package/dist/components/EntityFilters.d.ts +1 -34
  10. package/dist/components/EntityFilters.d.ts.map +1 -1
  11. package/dist/components/EntityFilters.js +1 -1
  12. package/dist/components/__tests__/EntityFilters.test.js +1 -1
  13. package/dist/components/__tests__/FormFieldRenderer.test.js +1 -1
  14. package/dist/components/controlled/complex/ControlledFieldArrayField.d.ts +18 -0
  15. package/dist/components/controlled/complex/ControlledFieldArrayField.d.ts.map +1 -0
  16. package/dist/components/controlled/complex/ControlledFieldArrayField.js +1 -0
  17. package/dist/components/controlled/complex/index.d.ts +1 -0
  18. package/dist/components/controlled/complex/index.d.ts.map +1 -1
  19. package/dist/components/controlled/complex/index.js +1 -1
  20. package/dist/components/controlled/types.d.ts +19 -5
  21. package/dist/components/controlled/types.d.ts.map +1 -1
  22. package/dist/components/form/fields/DocumentFieldComponent.d.ts.map +1 -1
  23. package/dist/components/form/fields/DocumentFieldComponent.js +1 -1
  24. package/dist/components/form/fields/FileFieldComponent.d.ts.map +1 -1
  25. package/dist/components/form/fields/FileFieldComponent.js +1 -1
  26. package/dist/components/form/fields/internal/TiptapEditor.d.ts.map +1 -1
  27. package/dist/components/index.d.ts +2 -1
  28. package/dist/components/index.d.ts.map +1 -1
  29. package/dist/components/index.js +1 -1
  30. package/dist/forms/hooks/useEntityField.d.ts +15 -24
  31. package/dist/forms/hooks/useEntityField.d.ts.map +1 -1
  32. package/dist/forms/hooks/useEntityField.js +1 -1
  33. package/dist/forms/hooks/useEntityForm.d.ts +3 -2
  34. package/dist/forms/hooks/useEntityForm.d.ts.map +1 -1
  35. package/dist/forms/hooks/useEntityForm.js +1 -1
  36. package/dist/forms/types.d.ts +13 -8
  37. package/dist/forms/types.d.ts.map +1 -1
  38. package/dist/forms/utils/buildInitialValues.d.ts.map +1 -1
  39. package/dist/forms/utils/buildInitialValues.js +1 -1
  40. package/dist/hooks/index.d.ts +1 -1
  41. package/dist/hooks/index.d.ts.map +1 -1
  42. package/dist/hooks/index.js +1 -1
  43. package/dist/hooks/useFileUpload.d.ts.map +1 -1
  44. package/dist/hooks/useFileUpload.js +1 -1
  45. package/dist/hooks/useUnsavedChangesWarning.d.ts +3 -24
  46. package/dist/hooks/useUnsavedChangesWarning.d.ts.map +1 -1
  47. package/dist/hooks/useUnsavedChangesWarning.js +1 -1
  48. package/dist/index.d.ts +5 -1
  49. package/dist/index.d.ts.map +1 -1
  50. package/dist/index.js +1 -1
  51. package/dist/registerBuiltinFieldTypes.d.ts.map +1 -1
  52. package/dist/registerBuiltinFieldTypes.js +1 -1
  53. package/dist/stores/UploadStore.d.ts.map +1 -1
  54. package/dist/tsconfig.tsbuildinfo +1 -1
  55. package/dist/useBaseCrudList.d.ts.map +1 -1
  56. package/dist/useCrud.js +1 -1
  57. package/dist/useCrudCardList.d.ts +14 -19
  58. package/dist/useCrudCardList.d.ts.map +1 -1
  59. package/dist/useCrudCardList.js +1 -1
  60. package/dist/useCrudList.d.ts +19 -18
  61. package/dist/useCrudList.d.ts.map +1 -1
  62. package/dist/useCrudList.js +1 -1
  63. package/dist/utils/clientListProcessing.d.ts +82 -0
  64. package/dist/utils/clientListProcessing.d.ts.map +1 -0
  65. package/dist/utils/clientListProcessing.js +1 -0
  66. package/dist/utils/fileStorage.d.ts.map +1 -1
  67. package/dist/utils/imageStorage.d.ts.map +1 -1
  68. package/dist/utils/imageStorage.js +1 -1
  69. package/dist/utils/matchesFilter.d.ts +45 -0
  70. package/dist/utils/matchesFilter.d.ts.map +1 -0
  71. package/dist/utils/matchesFilter.js +1 -0
  72. package/dist/utils/mergeWithOptimistic.d.ts.map +1 -1
  73. package/dist/utils/uploadValidation.d.ts.map +1 -1
  74. package/dist/utils/uploadValidation.js +1 -1
  75. package/package.json +5 -8
  76. package/dist/hooks/useFormNavigationGuard.d.ts +0 -34
  77. package/dist/hooks/useFormNavigationGuard.d.ts.map +0 -1
  78. 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,EAAE,MAAM,EAAE,aAAa,EAAE,cAAc,EAAe,YAAY,EAAE,MAAM,gBAAgB,CAAC;AAQvG,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,CAwO1B"}
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 Q,useMemo as w,useCallback as a}from"react";import{useFeatureConsent as V,FRAMEWORK_FEATURES as W,FEATURE_STATUS as m,handleError as Z,isClient as $,createSchemas as B,DEGRADED_CRUD_API as H}from"@donotdev/core";import{getCrudService as J}from"./CrudService";import{useCrudStore as u}from"./CrudStore";import{injectScope as F,injectScopeFilter as X}from"./utils/scopeUtils";const ce={},re={};let p=null,E=null,g=!1;function o(){return p}function ne(l,d={}){const h=V(W.CRUD),P=u(e=>e.crudService),T=h&&!P,v=h?T?m.INITIALIZING:m.READY:m.DEGRADED;if(l==="status")return v;const S=typeof l=="object"?l:d.entity,s=typeof l=="string"?l:l.collection,i=S?.scope,A=d.noCache??!1,C=d.staleTime,f=w(()=>({noCache:A,staleTime:C}),[A,C]),t=w(()=>{if(d.schema){const e=d.schema;return{create:e,draft:e,update:e,get:e,list:e,listCard:e,delete:e}}if(S)return B(S)},[d.schema,S?.name,s]),x=u(e=>e.collections[s]?.loading||!1),_=u(e=>e.collections[s]?.error||null),M=null;Q(()=>{if(!$()||!h||!t)return;const{crudService:e}=u.getState();e||p||E||(E=(async()=>{try{if(p||u.getState().crudService)return;const c=J();p=c,c.setStore(u),await c.initialize(),g=!1,u.getState().setCrudService(c)}catch(c){g||(Z(c,{userMessage:"Failed to initialize CRUD service.",context:{collection:s},severity:"error"}),g=!0)}finally{E=null}})())},[h,s,t?"hasSchema":"noSchema"]);const b=T||x,N=v===m.READY,z=a(async e=>{const c=o();return!c||!t?null:c.get(s,e,t.get,f)},[s,t?"hasSchema":"noSchema",f]),Y=a(async(e,c,r)=>{const n=o();if(!n||!t)return;const D=F(c,i),k=D?.status==="draft"?t.draft:t.create;await n.set(s,e,D,k,r)},[s,t?"hasSchema":"noSchema",i?.provider]),j=a(async(e,c,r)=>{const n=o();n&&await n.update(s,e,c,t?.update,r)},[s,t]),q=a(async(e,c)=>{const r=o();r&&await r.delete(s,e,c)},[s]),G=a(async(e,c)=>{const r=o();if(!r||!t)return"";const n=F(e,i),U=n?.status==="draft"?t.draft:t.create;return r.add(s,n,U,c)},[s,t?"hasSchema":"noSchema",i?.provider]),L=a(async e=>{const c=o();if(!c||!t)return[];const r=X(e,i);return(await c.query(s,r,t.list,f)).items},[s,t?"hasSchema":"noSchema",f,i?.provider]),I=a((e,c)=>{const r=o();return!r||!t?()=>{}:r.subscribe(s,e,c,t.get)},[s,t?"hasSchema":"noSchema"]),R=a((e,c)=>{const r=o();return!r||!t?()=>{}:r.subscribeToCollection(s,e,c,t.list)},[s,t?"hasSchema":"noSchema"]),y=a(async()=>{const e=o();e&&await e.invalidateCollection(s)},[s]);return N?{status:m.READY,data:M,loading:b,error:_,get:z,set:Y,update:j,delete:q,add:G,query:L,subscribe:I,subscribeToCollection:R,invalidate:y,isAvailable:!0,_collection:s,_schemas:t,_cacheOptions:f,_scope:i}:{...H,status:v,loading:b,error:_,subscribe:I,subscribeToCollection:R,invalidate:y,_collection:s,_schemas:t,_cacheOptions:f,_scope:i}}export{ce as EMPTY_DATA,re as EMPTY_OPTIMISTIC,o as getCrudServiceInstance,ne as useCrud};
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
- * const { items, loading, refresh } = useCrudCardList(carEntity);
52
- *
53
- * if (loading) return <Skeletons />;
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":"AAEA;;;;;;;;GAQG;AAEH,OAAO,KAAK,EAAE,MAAM,EAAE,aAAa,EAAE,YAAY,EAAE,MAAM,gBAAgB,CAAC;AAM1E,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,WAAW,CAAC;AAEhD,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;CAC7B;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,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;;;;;;;;;;;;;;;;;;;;;;;;;GAyBG;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,CA+B1B"}
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"}
@@ -1 +1 @@
1
- import{useBaseCrudList as r}from"./useBaseCrudList";function l(a,t={}){const s={enabled:t.enabled,queryOptions:t.queryOptions,schema:t.schema,entity:t.entity,staleTime:t.staleTime,noCache:t.noCache},e=r(a,"listCard",null,s);return{status:e.status,data:e.data?{items:e.data.items}:null,items:e.items,loading:e.loading,fetching:e.fetching,error:e.error,mutate:e.mutate,refresh:e.refresh,isAvailable:e.isAvailable}}export{l as useCrudCardList};
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};
@@ -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
- * const { items, loading, refresh } = useCrudList(carEntity);
60
- *
61
- * if (loading) return <Loader />;
59
+ * // Basic usage processed has filters + search + sort applied
60
+ * const { processed, loading, refresh } = useCrudList(carEntity, {
61
+ * searchQuery: search,
62
+ * });
62
63
  *
63
- * return (
64
- * <ul>
65
- * {items.map(car => <li key={car.id}>{car.make}</li>)}
66
- * </ul>
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":"AAEA;;;;;;;;GAQG;AAEH,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;AAEhD,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;CAC7B;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,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;;;;;;;;;;;;;;;;;;;;;;;;;GAyBG;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,CA4BtB"}
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"}
@@ -1 +1 @@
1
- import{useBaseCrudList as r}from"./useBaseCrudList";function g(a,e={}){const t=e.pagination??"client",n=e.page??1,i=e.pageSize??10,c={mode:t,page:n,pageSize:i},s={enabled:e.enabled,queryOptions:e.queryOptions,schema:e.schema,entity:e.entity,staleTime:e.staleTime,noCache:e.noCache};return r(a,"list",c,s)}export{g as useCrudList};
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,CAkDpB;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
+ {"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,CA6GlB;AAED;;GAEG;AACH,wBAAsB,WAAW,CAAC,OAAO,EAAE,OAAO,GAAG,OAAO,CAAC,IAAI,CAAC,CAuBjE"}
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 b,getProvider as f}from"@donotdev/core";function R(e){return e.replace(/[^a-zA-Z0-9.-]/g,"_").replace(/_{2,}/g,"_").toLowerCase()}async function B(e,t,p,y={}){const{storagePath:o="uploads/images",filename:P,onProgress:a}=y,U=Date.now(),l=(P||`${U}_${R(p)}`).replace(/\.[^/.]+$/,""),h=`${o}/${l}_full.webp`,d=`${o}/${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 n,i;if(b("storage")){const r=f("storage");n=(await r.upload(c,{storagePath:o,filename:`${l}_full.webp`,onProgress:a?m=>a({bytesTransferred:0,totalBytes:0,progress:m*.5}):void 0})).url,i=(await r.upload(s,{storagePath:o,filename:`${l}_thumb.webp`,onProgress:a?m=>a({bytesTransferred:0,totalBytes:0,progress:50+m*.5}):void 0})).url}else{const{uploadFileResumable:r}=await import("@donotdev/firebase"),g=a?u=>{a({bytesTransferred:u.bytesTransferred,totalBytes:u.totalBytes*2,progress:u.progress*.5})}:void 0,{promise:w}=r(c,{basePath:o,filename:`${l}_full.webp`,onProgress:g,metadata:{contentType:"image/webp"}});n=(await w).url;const $=a?u=>{a({bytesTransferred:u.bytesTransferred+e.size,totalBytes:e.size+t.size,progress:50+u.progress*.5})}:void 0,{promise:_}=r(s,{basePath:o,filename:`${l}_thumb.webp`,onProgress:$,metadata:{contentType:"image/webp"}});i=(await _).url}return{fullUrl:n,thumbUrl:i}}catch(c){try{if(b("storage")){const s=f("storage");await s.delete(h).catch(()=>{}),await s.delete(d).catch(()=>{})}else{const{getFileUrl:s,deleteFileByUrl:n}=await import("@donotdev/firebase"),i=await s(h).catch(()=>null),r=await s(d).catch(()=>null);i&&await n(i).catch(()=>{}),r&&await n(r).catch(()=>{})}}catch{}throw c}}async function v(e){try{if(b("storage")){const t=f("storage");await Promise.all([t.delete(e.fullUrl),t.delete(e.thumbUrl)])}else{const{deleteFileByUrl:t}=await import("@donotdev/firebase");await Promise.all([t(e.fullUrl),t(e.thumbUrl)])}}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};
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,CA4BL"}
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,CA+CnE;AAED;;;;GAIG;AACH,wBAAgB,WAAW,CAAC,IAAI,EAAE,OAAO,GAAG,OAAO,CAElD"}
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 l(r){return!r||typeof r!="string"?!1:r.startsWith("https://")&&!r.startsWith("blob:")}function u(r){return!(!r||typeof r!="object"||!l(r.fullUrl)||r.thumbUrl&&!l(r.thumbUrl))}function f(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((n,e)=>{const s=o?`${o}[${e}]`:`[${e}]`;t.push(...f(n,s))});else{const n=r;if(n.fullUrl!==void 0&&n.thumbUrl!==void 0){if(!l(String(n.fullUrl))||!l(String(n.thumbUrl))){const e=o||"root";t.push(`${e}.fullUrl or ${e}.thumbUrl contains blob URL`)}}else Object.entries(n).forEach(([e,s])=>{const i=o?`${o}.${e}`:e;t.push(...f(s,i))})}return t}function b(r){return f(r).length>0}export{f as checkForBlobUrls,b as hasBlobUrls,l as isStorageUrl,u as validatePicture};
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.15",
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.18",
57
- "@donotdev/core": "^0.0.24",
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.10",
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};