@growsober/sdk 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (129) hide show
  1. package/README.md +276 -0
  2. package/dist/__tests__/e2e.test.d.ts +7 -0
  3. package/dist/__tests__/e2e.test.js +472 -0
  4. package/dist/api/client.d.ts +11 -0
  5. package/dist/api/client.js +61 -0
  6. package/dist/api/mutations/admin.d.ts +167 -0
  7. package/dist/api/mutations/admin.js +326 -0
  8. package/dist/api/mutations/ambassadors.d.ts +52 -0
  9. package/dist/api/mutations/ambassadors.js +148 -0
  10. package/dist/api/mutations/auth.d.ts +267 -0
  11. package/dist/api/mutations/auth.js +332 -0
  12. package/dist/api/mutations/bookings.d.ts +59 -0
  13. package/dist/api/mutations/bookings.js +143 -0
  14. package/dist/api/mutations/event-chat.d.ts +35 -0
  15. package/dist/api/mutations/event-chat.js +147 -0
  16. package/dist/api/mutations/events.d.ts +87 -0
  17. package/dist/api/mutations/events.js +205 -0
  18. package/dist/api/mutations/grow90.d.ts +36 -0
  19. package/dist/api/mutations/grow90.js +132 -0
  20. package/dist/api/mutations/hubs.d.ts +111 -0
  21. package/dist/api/mutations/hubs.js +240 -0
  22. package/dist/api/mutations/index.d.ts +22 -0
  23. package/dist/api/mutations/index.js +39 -0
  24. package/dist/api/mutations/jack.d.ts +61 -0
  25. package/dist/api/mutations/jack.js +104 -0
  26. package/dist/api/mutations/library.d.ts +67 -0
  27. package/dist/api/mutations/library.js +168 -0
  28. package/dist/api/mutations/map.d.ts +153 -0
  29. package/dist/api/mutations/map.js +181 -0
  30. package/dist/api/mutations/matching.d.ts +130 -0
  31. package/dist/api/mutations/matching.js +204 -0
  32. package/dist/api/mutations/notifications.d.ts +63 -0
  33. package/dist/api/mutations/notifications.js +106 -0
  34. package/dist/api/mutations/offers.d.ts +26 -0
  35. package/dist/api/mutations/offers.js +47 -0
  36. package/dist/api/mutations/subscriptions.d.ts +127 -0
  37. package/dist/api/mutations/subscriptions.js +140 -0
  38. package/dist/api/mutations/support.d.ts +165 -0
  39. package/dist/api/mutations/support.js +307 -0
  40. package/dist/api/mutations/users.d.ts +211 -0
  41. package/dist/api/mutations/users.js +261 -0
  42. package/dist/api/queries/admin.d.ts +257 -0
  43. package/dist/api/queries/admin.js +320 -0
  44. package/dist/api/queries/ambassadors.d.ts +53 -0
  45. package/dist/api/queries/ambassadors.js +98 -0
  46. package/dist/api/queries/auth.d.ts +16 -0
  47. package/dist/api/queries/auth.js +25 -0
  48. package/dist/api/queries/bookings.d.ts +91 -0
  49. package/dist/api/queries/bookings.js +102 -0
  50. package/dist/api/queries/businesses.d.ts +212 -0
  51. package/dist/api/queries/businesses.js +154 -0
  52. package/dist/api/queries/event-chat.d.ts +19 -0
  53. package/dist/api/queries/event-chat.js +75 -0
  54. package/dist/api/queries/events.d.ts +322 -0
  55. package/dist/api/queries/events.js +221 -0
  56. package/dist/api/queries/grow90.d.ts +26 -0
  57. package/dist/api/queries/grow90.js +85 -0
  58. package/dist/api/queries/hubs.d.ts +165 -0
  59. package/dist/api/queries/hubs.js +143 -0
  60. package/dist/api/queries/index.d.ts +23 -0
  61. package/dist/api/queries/index.js +40 -0
  62. package/dist/api/queries/jack.d.ts +63 -0
  63. package/dist/api/queries/jack.js +92 -0
  64. package/dist/api/queries/library.d.ts +132 -0
  65. package/dist/api/queries/library.js +120 -0
  66. package/dist/api/queries/map.d.ts +216 -0
  67. package/dist/api/queries/map.js +278 -0
  68. package/dist/api/queries/matching.d.ts +136 -0
  69. package/dist/api/queries/matching.js +161 -0
  70. package/dist/api/queries/notifications.d.ts +78 -0
  71. package/dist/api/queries/notifications.js +88 -0
  72. package/dist/api/queries/offers.d.ts +91 -0
  73. package/dist/api/queries/offers.js +103 -0
  74. package/dist/api/queries/subscriptions.d.ts +56 -0
  75. package/dist/api/queries/subscriptions.js +73 -0
  76. package/dist/api/queries/support.d.ts +106 -0
  77. package/dist/api/queries/support.js +202 -0
  78. package/dist/api/queries/users.d.ts +293 -0
  79. package/dist/api/queries/users.js +370 -0
  80. package/dist/api/types.d.ts +464 -0
  81. package/dist/api/types.js +9 -0
  82. package/dist/hooks/useAuth.d.ts +5 -0
  83. package/dist/hooks/useAuth.js +39 -0
  84. package/dist/hooks/useUser.d.ts +43 -0
  85. package/dist/hooks/useUser.js +44 -0
  86. package/dist/index.d.ts +36 -0
  87. package/dist/index.js +67 -0
  88. package/package.json +62 -0
  89. package/src/__tests__/e2e.test.ts +502 -0
  90. package/src/api/client.ts +71 -0
  91. package/src/api/mutations/admin.ts +531 -0
  92. package/src/api/mutations/ambassadors.ts +185 -0
  93. package/src/api/mutations/auth.ts +350 -0
  94. package/src/api/mutations/bookings.ts +190 -0
  95. package/src/api/mutations/event-chat.ts +177 -0
  96. package/src/api/mutations/events.ts +273 -0
  97. package/src/api/mutations/grow90.ts +169 -0
  98. package/src/api/mutations/hubs.ts +385 -0
  99. package/src/api/mutations/index.ts +23 -0
  100. package/src/api/mutations/jack.ts +130 -0
  101. package/src/api/mutations/library.ts +212 -0
  102. package/src/api/mutations/map.ts +230 -0
  103. package/src/api/mutations/matching.ts +271 -0
  104. package/src/api/mutations/notifications.ts +114 -0
  105. package/src/api/mutations/offers.ts +73 -0
  106. package/src/api/mutations/subscriptions.ts +162 -0
  107. package/src/api/mutations/support.ts +390 -0
  108. package/src/api/mutations/users.ts +271 -0
  109. package/src/api/queries/admin.ts +480 -0
  110. package/src/api/queries/ambassadors.ts +139 -0
  111. package/src/api/queries/auth.ts +24 -0
  112. package/src/api/queries/bookings.ts +135 -0
  113. package/src/api/queries/businesses.ts +203 -0
  114. package/src/api/queries/event-chat.ts +78 -0
  115. package/src/api/queries/events.ts +272 -0
  116. package/src/api/queries/grow90.ts +98 -0
  117. package/src/api/queries/hubs.ts +211 -0
  118. package/src/api/queries/index.ts +24 -0
  119. package/src/api/queries/jack.ts +127 -0
  120. package/src/api/queries/library.ts +166 -0
  121. package/src/api/queries/map.ts +331 -0
  122. package/src/api/queries/matching.ts +238 -0
  123. package/src/api/queries/notifications.ts +103 -0
  124. package/src/api/queries/offers.ts +136 -0
  125. package/src/api/queries/subscriptions.ts +91 -0
  126. package/src/api/queries/support.ts +235 -0
  127. package/src/api/queries/users.ts +393 -0
  128. package/src/api/types.ts +596 -0
  129. package/src/index.ts +57 -0
@@ -0,0 +1,278 @@
1
+ "use strict";
2
+ /**
3
+ * Map Query Hooks
4
+ *
5
+ * TanStack Query hooks for map-related read operations.
6
+ * These hooks handle fetching map data including users, events, and businesses
7
+ * for display on the member map feature.
8
+ *
9
+ * @module api/queries/map
10
+ */
11
+ Object.defineProperty(exports, "__esModule", { value: true });
12
+ exports.mapKeys = void 0;
13
+ exports.useMapMembers = useMapMembers;
14
+ exports.useMapEvents = useMapEvents;
15
+ exports.useMapHubs = useMapHubs;
16
+ exports.useMapBusinesses = useMapBusinesses;
17
+ const react_query_1 = require("@tanstack/react-query");
18
+ const client_1 = require("../client");
19
+ // ============================================================================
20
+ // QUERY KEYS
21
+ // ============================================================================
22
+ /**
23
+ * Query key factory for map-related queries
24
+ */
25
+ exports.mapKeys = {
26
+ all: ['map'],
27
+ members: (params) => [...exports.mapKeys.all, 'members', params],
28
+ events: (params) => [...exports.mapKeys.all, 'events', params],
29
+ hubs: (params) => [...exports.mapKeys.all, 'hubs', params],
30
+ businesses: (params) => [...exports.mapKeys.all, 'businesses', params],
31
+ };
32
+ // ============================================================================
33
+ // QUERY HOOKS
34
+ // ============================================================================
35
+ /**
36
+ * Get members for map display
37
+ *
38
+ * @description
39
+ * Retrieves users who have opted to be visible on the member map.
40
+ * Only shows users who have enabled "open to meet" and shared their location.
41
+ * Premium feature.
42
+ *
43
+ * @endpoint GET /api/v1/map/members
44
+ *
45
+ * @example
46
+ * ```tsx
47
+ * import { useMapMembers } from '@growsober/sdk';
48
+ *
49
+ * function MemberMap() {
50
+ * const { data: members, isLoading } = useMapMembers({
51
+ * lat: 51.5074,
52
+ * lng: -0.1278,
53
+ * radius: 50, // 50km radius
54
+ * });
55
+ *
56
+ * if (isLoading) return <MapLoader />;
57
+ *
58
+ * return (
59
+ * <Map>
60
+ * {members?.map(member => (
61
+ * <Marker
62
+ * key={member.id}
63
+ * position={[member.locationLat, member.locationLong]}
64
+ * >
65
+ * <Avatar src={member.avatar} />
66
+ * </Marker>
67
+ * ))}
68
+ * </Map>
69
+ * );
70
+ * }
71
+ * ```
72
+ *
73
+ * @param params - Location and filter parameters
74
+ * @param options - TanStack Query options
75
+ * @returns TanStack Query result with array of map users
76
+ */
77
+ function useMapMembers(params, options) {
78
+ return (0, react_query_1.useQuery)({
79
+ queryKey: exports.mapKeys.members(params),
80
+ queryFn: async () => {
81
+ const client = (0, client_1.getApiClient)();
82
+ const response = await client.get('/api/v1/map/members', {
83
+ params: {
84
+ lat: params.lat,
85
+ lng: params.lng,
86
+ radius: params.radius,
87
+ hubId: params.hubId,
88
+ openToMeetOnly: params.openToMeetOnly,
89
+ },
90
+ });
91
+ return response.data;
92
+ },
93
+ enabled: params.lat !== undefined && params.lng !== undefined,
94
+ ...options,
95
+ });
96
+ }
97
+ /**
98
+ * Get events for map display
99
+ *
100
+ * @description
101
+ * Retrieves events with location data for display on the map.
102
+ * Shows upcoming events within the specified radius.
103
+ *
104
+ * @endpoint GET /api/v1/map/events
105
+ *
106
+ * @example
107
+ * ```tsx
108
+ * import { useMapEvents } from '@growsober/sdk';
109
+ *
110
+ * function EventMap() {
111
+ * const { data: events, isLoading } = useMapEvents({
112
+ * lat: 51.5074,
113
+ * lng: -0.1278,
114
+ * radius: 25,
115
+ * upcoming: true,
116
+ * });
117
+ *
118
+ * return (
119
+ * <Map>
120
+ * {events?.map(event => (
121
+ * <Marker
122
+ * key={event.id}
123
+ * position={[event.locationLat, event.locationLong]}
124
+ * icon="event"
125
+ * >
126
+ * <EventPopup event={event} />
127
+ * </Marker>
128
+ * ))}
129
+ * </Map>
130
+ * );
131
+ * }
132
+ * ```
133
+ *
134
+ * @param params - Location and filter parameters
135
+ * @param options - TanStack Query options
136
+ * @returns TanStack Query result with array of map events
137
+ */
138
+ function useMapEvents(params, options) {
139
+ return (0, react_query_1.useQuery)({
140
+ queryKey: exports.mapKeys.events(params),
141
+ queryFn: async () => {
142
+ const client = (0, client_1.getApiClient)();
143
+ const response = await client.get('/api/v1/map/events', {
144
+ params: {
145
+ lat: params.lat,
146
+ lng: params.lng,
147
+ radius: params.radius,
148
+ hubId: params.hubId,
149
+ upcoming: params.upcoming,
150
+ },
151
+ });
152
+ return response.data;
153
+ },
154
+ enabled: params.lat !== undefined && params.lng !== undefined,
155
+ ...options,
156
+ });
157
+ }
158
+ /**
159
+ * Get hubs for map display
160
+ *
161
+ * @description
162
+ * Retrieves hubs with location data for display on the map.
163
+ * Shows active hubs within the specified radius.
164
+ *
165
+ * @endpoint GET /api/v1/map/hubs
166
+ *
167
+ * @example
168
+ * ```tsx
169
+ * import { useMapHubs } from '@growsober/sdk';
170
+ *
171
+ * function HubMap() {
172
+ * const { data: hubs, isLoading } = useMapHubs({
173
+ * lat: 51.5074,
174
+ * lng: -0.1278,
175
+ * radius: 50,
176
+ * });
177
+ *
178
+ * return (
179
+ * <Map>
180
+ * {hubs?.map(hub => (
181
+ * <Marker
182
+ * key={hub.id}
183
+ * position={[hub.locationLat, hub.locationLong]}
184
+ * icon="hub"
185
+ * >
186
+ * <HubPopup hub={hub} />
187
+ * </Marker>
188
+ * ))}
189
+ * </Map>
190
+ * );
191
+ * }
192
+ * ```
193
+ *
194
+ * @param params - Location and filter parameters
195
+ * @param options - TanStack Query options
196
+ * @returns TanStack Query result with array of map hubs
197
+ */
198
+ function useMapHubs(params, options) {
199
+ return (0, react_query_1.useQuery)({
200
+ queryKey: exports.mapKeys.hubs(params),
201
+ queryFn: async () => {
202
+ const client = (0, client_1.getApiClient)();
203
+ const response = await client.get('/api/v1/map/hubs', {
204
+ params: {
205
+ lat: params.lat,
206
+ lng: params.lng,
207
+ radius: params.radius,
208
+ city: params.city,
209
+ },
210
+ });
211
+ return response.data;
212
+ },
213
+ enabled: params.lat !== undefined && params.lng !== undefined,
214
+ ...options,
215
+ });
216
+ }
217
+ /**
218
+ * Get businesses for map display
219
+ *
220
+ * @description
221
+ * Retrieves partner businesses with location data for display on the map.
222
+ * Can filter by type and whether they serve alcohol-free drinks.
223
+ *
224
+ * @endpoint GET /api/v1/map/businesses
225
+ *
226
+ * @example
227
+ * ```tsx
228
+ * import { useMapBusinesses } from '@growsober/sdk';
229
+ *
230
+ * function BusinessMap() {
231
+ * const { data: businesses, isLoading } = useMapBusinesses({
232
+ * lat: 51.5074,
233
+ * lng: -0.1278,
234
+ * radius: 10,
235
+ * afDrinksOnly: true, // Only show AF-friendly venues
236
+ * });
237
+ *
238
+ * return (
239
+ * <Map>
240
+ * {businesses?.map(business => (
241
+ * <Marker
242
+ * key={business.id}
243
+ * position={[business.locationLat, business.locationLong]}
244
+ * icon={business.hasAfDrinks ? 'af-venue' : 'venue'}
245
+ * >
246
+ * <BusinessPopup business={business} />
247
+ * </Marker>
248
+ * ))}
249
+ * </Map>
250
+ * );
251
+ * }
252
+ * ```
253
+ *
254
+ * @param params - Location and filter parameters
255
+ * @param options - TanStack Query options
256
+ * @returns TanStack Query result with array of map businesses
257
+ */
258
+ function useMapBusinesses(params, options) {
259
+ return (0, react_query_1.useQuery)({
260
+ queryKey: exports.mapKeys.businesses(params),
261
+ queryFn: async () => {
262
+ const client = (0, client_1.getApiClient)();
263
+ const response = await client.get('/api/v1/map/businesses', {
264
+ params: {
265
+ lat: params.lat,
266
+ lng: params.lng,
267
+ radius: params.radius,
268
+ type: params.type,
269
+ afDrinksOnly: params.afDrinksOnly,
270
+ },
271
+ });
272
+ return response.data;
273
+ },
274
+ enabled: params.lat !== undefined && params.lng !== undefined,
275
+ ...options,
276
+ });
277
+ }
278
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"map.js","sourceRoot":"","sources":["../../../src/api/queries/map.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Map Query Hooks\n *\n * TanStack Query hooks for map-related read operations.\n * These hooks handle fetching map data including users, events, and businesses\n * for display on the member map feature.\n *\n * @module api/queries/map\n */\n\nimport { useQuery, UseQueryOptions, UseQueryResult } from '@tanstack/react-query';\nimport { getApiClient } from '../client';\nimport type {\n  MapUserResponse,\n  MapEventResponse,\n  MapHubResponse,\n  MapBusinessResponse,\n} from '../types';\n\n// ============================================================================\n// QUERY KEYS\n// ============================================================================\n\n/**\n * Query key factory for map-related queries\n */\nexport const mapKeys = {\n  all: ['map'] as const,\n  members: (params?: MapMembersParams) => [...mapKeys.all, 'members', params] as const,\n  events: (params?: MapEventsParams) => [...mapKeys.all, 'events', params] as const,\n  hubs: (params?: MapHubsParams) => [...mapKeys.all, 'hubs', params] as const,\n  businesses: (params?: MapBusinessesParams) => [...mapKeys.all, 'businesses', params] as const,\n};\n\n// ============================================================================\n// PARAM TYPES\n// ============================================================================\n\nexport interface MapMembersParams {\n  lat: number;\n  lng: number;\n  radius?: number; // km\n  hubId?: string;\n  openToMeetOnly?: boolean;\n}\n\nexport interface MapEventsParams {\n  lat: number;\n  lng: number;\n  radius?: number; // km\n  hubId?: string;\n  upcoming?: boolean;\n}\n\nexport interface MapHubsParams {\n  lat: number;\n  lng: number;\n  radius?: number; // km\n  city?: string;\n}\n\nexport interface MapBusinessesParams {\n  lat: number;\n  lng: number;\n  radius?: number; // km\n  type?: string;\n  afDrinksOnly?: boolean;\n}\n\n// ============================================================================\n// QUERY HOOKS\n// ============================================================================\n\n/**\n * Get members for map display\n *\n * @description\n * Retrieves users who have opted to be visible on the member map.\n * Only shows users who have enabled \"open to meet\" and shared their location.\n * Premium feature.\n *\n * @endpoint GET /api/v1/map/members\n *\n * @example\n * ```tsx\n * import { useMapMembers } from '@growsober/sdk';\n *\n * function MemberMap() {\n *   const { data: members, isLoading } = useMapMembers({\n *     lat: 51.5074,\n *     lng: -0.1278,\n *     radius: 50, // 50km radius\n *   });\n *\n *   if (isLoading) return <MapLoader />;\n *\n *   return (\n *     <Map>\n *       {members?.map(member => (\n *         <Marker\n *           key={member.id}\n *           position={[member.locationLat, member.locationLong]}\n *         >\n *           <Avatar src={member.avatar} />\n *         </Marker>\n *       ))}\n *     </Map>\n *   );\n * }\n * ```\n *\n * @param params - Location and filter parameters\n * @param options - TanStack Query options\n * @returns TanStack Query result with array of map users\n */\nexport function useMapMembers(\n  params: MapMembersParams,\n  options?: Omit<UseQueryOptions<MapUserResponse[]>, 'queryKey' | 'queryFn'>\n): UseQueryResult<MapUserResponse[]> {\n  return useQuery({\n    queryKey: mapKeys.members(params),\n    queryFn: async (): Promise<MapUserResponse[]> => {\n      const client = getApiClient();\n      const response = await client.get<MapUserResponse[]>('/api/v1/map/members', {\n        params: {\n          lat: params.lat,\n          lng: params.lng,\n          radius: params.radius,\n          hubId: params.hubId,\n          openToMeetOnly: params.openToMeetOnly,\n        },\n      });\n      return response.data;\n    },\n    enabled: params.lat !== undefined && params.lng !== undefined,\n    ...options,\n  });\n}\n\n/**\n * Get events for map display\n *\n * @description\n * Retrieves events with location data for display on the map.\n * Shows upcoming events within the specified radius.\n *\n * @endpoint GET /api/v1/map/events\n *\n * @example\n * ```tsx\n * import { useMapEvents } from '@growsober/sdk';\n *\n * function EventMap() {\n *   const { data: events, isLoading } = useMapEvents({\n *     lat: 51.5074,\n *     lng: -0.1278,\n *     radius: 25,\n *     upcoming: true,\n *   });\n *\n *   return (\n *     <Map>\n *       {events?.map(event => (\n *         <Marker\n *           key={event.id}\n *           position={[event.locationLat, event.locationLong]}\n *           icon=\"event\"\n *         >\n *           <EventPopup event={event} />\n *         </Marker>\n *       ))}\n *     </Map>\n *   );\n * }\n * ```\n *\n * @param params - Location and filter parameters\n * @param options - TanStack Query options\n * @returns TanStack Query result with array of map events\n */\nexport function useMapEvents(\n  params: MapEventsParams,\n  options?: Omit<UseQueryOptions<MapEventResponse[]>, 'queryKey' | 'queryFn'>\n): UseQueryResult<MapEventResponse[]> {\n  return useQuery({\n    queryKey: mapKeys.events(params),\n    queryFn: async (): Promise<MapEventResponse[]> => {\n      const client = getApiClient();\n      const response = await client.get<MapEventResponse[]>('/api/v1/map/events', {\n        params: {\n          lat: params.lat,\n          lng: params.lng,\n          radius: params.radius,\n          hubId: params.hubId,\n          upcoming: params.upcoming,\n        },\n      });\n      return response.data;\n    },\n    enabled: params.lat !== undefined && params.lng !== undefined,\n    ...options,\n  });\n}\n\n/**\n * Get hubs for map display\n *\n * @description\n * Retrieves hubs with location data for display on the map.\n * Shows active hubs within the specified radius.\n *\n * @endpoint GET /api/v1/map/hubs\n *\n * @example\n * ```tsx\n * import { useMapHubs } from '@growsober/sdk';\n *\n * function HubMap() {\n *   const { data: hubs, isLoading } = useMapHubs({\n *     lat: 51.5074,\n *     lng: -0.1278,\n *     radius: 50,\n *   });\n *\n *   return (\n *     <Map>\n *       {hubs?.map(hub => (\n *         <Marker\n *           key={hub.id}\n *           position={[hub.locationLat, hub.locationLong]}\n *           icon=\"hub\"\n *         >\n *           <HubPopup hub={hub} />\n *         </Marker>\n *       ))}\n *     </Map>\n *   );\n * }\n * ```\n *\n * @param params - Location and filter parameters\n * @param options - TanStack Query options\n * @returns TanStack Query result with array of map hubs\n */\nexport function useMapHubs(\n  params: MapHubsParams,\n  options?: Omit<UseQueryOptions<MapHubResponse[]>, 'queryKey' | 'queryFn'>\n): UseQueryResult<MapHubResponse[]> {\n  return useQuery({\n    queryKey: mapKeys.hubs(params),\n    queryFn: async (): Promise<MapHubResponse[]> => {\n      const client = getApiClient();\n      const response = await client.get<MapHubResponse[]>('/api/v1/map/hubs', {\n        params: {\n          lat: params.lat,\n          lng: params.lng,\n          radius: params.radius,\n          city: params.city,\n        },\n      });\n      return response.data;\n    },\n    enabled: params.lat !== undefined && params.lng !== undefined,\n    ...options,\n  });\n}\n\n/**\n * Get businesses for map display\n *\n * @description\n * Retrieves partner businesses with location data for display on the map.\n * Can filter by type and whether they serve alcohol-free drinks.\n *\n * @endpoint GET /api/v1/map/businesses\n *\n * @example\n * ```tsx\n * import { useMapBusinesses } from '@growsober/sdk';\n *\n * function BusinessMap() {\n *   const { data: businesses, isLoading } = useMapBusinesses({\n *     lat: 51.5074,\n *     lng: -0.1278,\n *     radius: 10,\n *     afDrinksOnly: true, // Only show AF-friendly venues\n *   });\n *\n *   return (\n *     <Map>\n *       {businesses?.map(business => (\n *         <Marker\n *           key={business.id}\n *           position={[business.locationLat, business.locationLong]}\n *           icon={business.hasAfDrinks ? 'af-venue' : 'venue'}\n *         >\n *           <BusinessPopup business={business} />\n *         </Marker>\n *       ))}\n *     </Map>\n *   );\n * }\n * ```\n *\n * @param params - Location and filter parameters\n * @param options - TanStack Query options\n * @returns TanStack Query result with array of map businesses\n */\nexport function useMapBusinesses(\n  params: MapBusinessesParams,\n  options?: Omit<UseQueryOptions<MapBusinessResponse[]>, 'queryKey' | 'queryFn'>\n): UseQueryResult<MapBusinessResponse[]> {\n  return useQuery({\n    queryKey: mapKeys.businesses(params),\n    queryFn: async (): Promise<MapBusinessResponse[]> => {\n      const client = getApiClient();\n      const response = await client.get<MapBusinessResponse[]>('/api/v1/map/businesses', {\n        params: {\n          lat: params.lat,\n          lng: params.lng,\n          radius: params.radius,\n          type: params.type,\n          afDrinksOnly: params.afDrinksOnly,\n        },\n      });\n      return response.data;\n    },\n    enabled: params.lat !== undefined && params.lng !== undefined,\n    ...options,\n  });\n}\n"]}
@@ -0,0 +1,136 @@
1
+ import { UseQueryOptions } from '@tanstack/react-query';
2
+ export declare const matchingKeys: {
3
+ all: readonly ["matching"];
4
+ discover: (filters?: MatchQueryFilters) => readonly ["matching", "discover", MatchQueryFilters | undefined];
5
+ matches: (filters?: MatchQueryFilters) => readonly ["matching", "matches", MatchQueryFilters | undefined];
6
+ match: (id: string) => readonly ["matching", "match", string];
7
+ buddies: () => readonly ["matching", "buddies"];
8
+ buddyRequests: () => readonly ["matching", "buddy-requests"];
9
+ stats: () => readonly ["matching", "stats"];
10
+ };
11
+ export interface MatchQueryFilters {
12
+ status?: 'PENDING' | 'ACCEPTED' | 'DECLINED' | 'BLOCKED';
13
+ limit?: number;
14
+ offset?: number;
15
+ }
16
+ export interface MatchResponse {
17
+ id: string;
18
+ userId: string;
19
+ matchedUserId: string;
20
+ status: 'PENDING' | 'ACCEPTED' | 'DECLINED' | 'BLOCKED';
21
+ matchScore?: number;
22
+ sharedInterests?: string[];
23
+ createdAt: string;
24
+ updatedAt: string;
25
+ matchedUser?: {
26
+ id: string;
27
+ name: string;
28
+ avatarUrl?: string;
29
+ bio?: string;
30
+ city?: string;
31
+ };
32
+ }
33
+ export interface DiscoverMatchResponse {
34
+ id: string;
35
+ name: string;
36
+ avatarUrl?: string;
37
+ bio?: string;
38
+ city?: string;
39
+ matchScore: number;
40
+ sharedInterests: string[];
41
+ distance?: number;
42
+ }
43
+ export interface BuddyResponse {
44
+ id: string;
45
+ userId: string;
46
+ buddyId: string;
47
+ status: 'PENDING' | 'ACTIVE' | 'ENDED';
48
+ startDate?: string;
49
+ endDate?: string;
50
+ lastActivity?: string;
51
+ buddy?: {
52
+ id: string;
53
+ name: string;
54
+ avatarUrl?: string;
55
+ };
56
+ }
57
+ export interface MatchingStatsResponse {
58
+ totalMatches: number;
59
+ pendingMatches: number;
60
+ acceptedMatches: number;
61
+ activeBuddies: number;
62
+ matchRate: number;
63
+ }
64
+ /**
65
+ * Discover potential matches based on shared interests and location
66
+ *
67
+ * @endpoint GET /api/v1/matching/discover
68
+ *
69
+ * @example
70
+ * ```tsx
71
+ * const { data: matches, isLoading } = useDiscoverMatches();
72
+ * ```
73
+ */
74
+ export declare function useDiscoverMatches(filters?: MatchQueryFilters, options?: Omit<UseQueryOptions<DiscoverMatchResponse[]>, 'queryKey' | 'queryFn'>): import("@tanstack/react-query").UseQueryResult<DiscoverMatchResponse[], Error>;
75
+ /**
76
+ * Get all my matches
77
+ *
78
+ * @endpoint GET /api/v1/matching/matches
79
+ *
80
+ * @example
81
+ * ```tsx
82
+ * const { data: matches, isLoading } = useMyMatches({ status: 'PENDING' });
83
+ * ```
84
+ */
85
+ export declare function useMyMatches(filters?: MatchQueryFilters, options?: Omit<UseQueryOptions<MatchResponse[]>, 'queryKey' | 'queryFn'>): import("@tanstack/react-query").UseQueryResult<MatchResponse[], Error>;
86
+ /**
87
+ * Get pending matches (convenience hook)
88
+ *
89
+ * @example
90
+ * ```tsx
91
+ * const { data: pendingMatches } = usePendingMatches();
92
+ * ```
93
+ */
94
+ export declare function usePendingMatches(options?: Omit<UseQueryOptions<MatchResponse[]>, 'queryKey' | 'queryFn'>): import("@tanstack/react-query").UseQueryResult<MatchResponse[], Error>;
95
+ /**
96
+ * Get accepted matches (convenience hook)
97
+ *
98
+ * @example
99
+ * ```tsx
100
+ * const { data: acceptedMatches } = useAcceptedMatches();
101
+ * ```
102
+ */
103
+ export declare function useAcceptedMatches(options?: Omit<UseQueryOptions<MatchResponse[]>, 'queryKey' | 'queryFn'>): import("@tanstack/react-query").UseQueryResult<MatchResponse[], Error>;
104
+ /**
105
+ * Get my accountability buddies
106
+ *
107
+ * @endpoint GET /api/v1/matching/buddies
108
+ *
109
+ * @example
110
+ * ```tsx
111
+ * const { data: buddies, isLoading } = useMyBuddies();
112
+ * ```
113
+ */
114
+ export declare function useMyBuddies(options?: Omit<UseQueryOptions<BuddyResponse[]>, 'queryKey' | 'queryFn'>): import("@tanstack/react-query").UseQueryResult<BuddyResponse[], Error>;
115
+ /**
116
+ * Get pending buddy requests
117
+ *
118
+ * @endpoint GET /api/v1/matching/buddies/requests
119
+ *
120
+ * @example
121
+ * ```tsx
122
+ * const { data: requests } = useBuddyRequests();
123
+ * ```
124
+ */
125
+ export declare function useBuddyRequests(options?: Omit<UseQueryOptions<BuddyResponse[]>, 'queryKey' | 'queryFn'>): import("@tanstack/react-query").UseQueryResult<BuddyResponse[], Error>;
126
+ /**
127
+ * Get matching statistics
128
+ *
129
+ * @endpoint GET /api/v1/matching/stats
130
+ *
131
+ * @example
132
+ * ```tsx
133
+ * const { data: stats } = useMatchingStats();
134
+ * ```
135
+ */
136
+ export declare function useMatchingStats(options?: Omit<UseQueryOptions<MatchingStatsResponse>, 'queryKey' | 'queryFn'>): import("@tanstack/react-query").UseQueryResult<MatchingStatsResponse, Error>;
@@ -0,0 +1,161 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.matchingKeys = void 0;
4
+ exports.useDiscoverMatches = useDiscoverMatches;
5
+ exports.useMyMatches = useMyMatches;
6
+ exports.usePendingMatches = usePendingMatches;
7
+ exports.useAcceptedMatches = useAcceptedMatches;
8
+ exports.useMyBuddies = useMyBuddies;
9
+ exports.useBuddyRequests = useBuddyRequests;
10
+ exports.useMatchingStats = useMatchingStats;
11
+ const react_query_1 = require("@tanstack/react-query");
12
+ const client_1 = require("../client");
13
+ // ============================================================================
14
+ // QUERY KEY FACTORY
15
+ // ============================================================================
16
+ exports.matchingKeys = {
17
+ all: ['matching'],
18
+ discover: (filters) => [...exports.matchingKeys.all, 'discover', filters],
19
+ matches: (filters) => [...exports.matchingKeys.all, 'matches', filters],
20
+ match: (id) => [...exports.matchingKeys.all, 'match', id],
21
+ buddies: () => [...exports.matchingKeys.all, 'buddies'],
22
+ buddyRequests: () => [...exports.matchingKeys.all, 'buddy-requests'],
23
+ stats: () => [...exports.matchingKeys.all, 'stats'],
24
+ };
25
+ // ============================================================================
26
+ // QUERY HOOKS
27
+ // ============================================================================
28
+ /**
29
+ * Discover potential matches based on shared interests and location
30
+ *
31
+ * @endpoint GET /api/v1/matching/discover
32
+ *
33
+ * @example
34
+ * ```tsx
35
+ * const { data: matches, isLoading } = useDiscoverMatches();
36
+ * ```
37
+ */
38
+ function useDiscoverMatches(filters, options) {
39
+ return (0, react_query_1.useQuery)({
40
+ queryKey: exports.matchingKeys.discover(filters),
41
+ queryFn: async () => {
42
+ const client = (0, client_1.getApiClient)();
43
+ const response = await client.get('/api/v1/matching/discover', {
44
+ params: filters,
45
+ });
46
+ return response.data?.data || response.data;
47
+ },
48
+ ...options,
49
+ });
50
+ }
51
+ /**
52
+ * Get all my matches
53
+ *
54
+ * @endpoint GET /api/v1/matching/matches
55
+ *
56
+ * @example
57
+ * ```tsx
58
+ * const { data: matches, isLoading } = useMyMatches({ status: 'PENDING' });
59
+ * ```
60
+ */
61
+ function useMyMatches(filters, options) {
62
+ return (0, react_query_1.useQuery)({
63
+ queryKey: exports.matchingKeys.matches(filters),
64
+ queryFn: async () => {
65
+ const client = (0, client_1.getApiClient)();
66
+ const response = await client.get('/api/v1/matching/matches', {
67
+ params: filters,
68
+ });
69
+ return response.data?.data || response.data;
70
+ },
71
+ staleTime: 30 * 1000,
72
+ ...options,
73
+ });
74
+ }
75
+ /**
76
+ * Get pending matches (convenience hook)
77
+ *
78
+ * @example
79
+ * ```tsx
80
+ * const { data: pendingMatches } = usePendingMatches();
81
+ * ```
82
+ */
83
+ function usePendingMatches(options) {
84
+ return useMyMatches({ status: 'PENDING' }, options);
85
+ }
86
+ /**
87
+ * Get accepted matches (convenience hook)
88
+ *
89
+ * @example
90
+ * ```tsx
91
+ * const { data: acceptedMatches } = useAcceptedMatches();
92
+ * ```
93
+ */
94
+ function useAcceptedMatches(options) {
95
+ return useMyMatches({ status: 'ACCEPTED' }, options);
96
+ }
97
+ /**
98
+ * Get my accountability buddies
99
+ *
100
+ * @endpoint GET /api/v1/matching/buddies
101
+ *
102
+ * @example
103
+ * ```tsx
104
+ * const { data: buddies, isLoading } = useMyBuddies();
105
+ * ```
106
+ */
107
+ function useMyBuddies(options) {
108
+ return (0, react_query_1.useQuery)({
109
+ queryKey: exports.matchingKeys.buddies(),
110
+ queryFn: async () => {
111
+ const client = (0, client_1.getApiClient)();
112
+ const response = await client.get('/api/v1/matching/buddies');
113
+ return response.data?.data || response.data;
114
+ },
115
+ ...options,
116
+ });
117
+ }
118
+ /**
119
+ * Get pending buddy requests
120
+ *
121
+ * @endpoint GET /api/v1/matching/buddies/requests
122
+ *
123
+ * @example
124
+ * ```tsx
125
+ * const { data: requests } = useBuddyRequests();
126
+ * ```
127
+ */
128
+ function useBuddyRequests(options) {
129
+ return (0, react_query_1.useQuery)({
130
+ queryKey: exports.matchingKeys.buddyRequests(),
131
+ queryFn: async () => {
132
+ const client = (0, client_1.getApiClient)();
133
+ const response = await client.get('/api/v1/matching/buddies/requests');
134
+ return response.data?.data || response.data;
135
+ },
136
+ ...options,
137
+ });
138
+ }
139
+ /**
140
+ * Get matching statistics
141
+ *
142
+ * @endpoint GET /api/v1/matching/stats
143
+ *
144
+ * @example
145
+ * ```tsx
146
+ * const { data: stats } = useMatchingStats();
147
+ * ```
148
+ */
149
+ function useMatchingStats(options) {
150
+ return (0, react_query_1.useQuery)({
151
+ queryKey: exports.matchingKeys.stats(),
152
+ queryFn: async () => {
153
+ const client = (0, client_1.getApiClient)();
154
+ const response = await client.get('/api/v1/matching/stats');
155
+ return response.data?.data || response.data;
156
+ },
157
+ staleTime: 5 * 60 * 1000,
158
+ ...options,
159
+ });
160
+ }
161
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"matching.js","sourceRoot":"","sources":["../../../src/api/queries/matching.ts"],"names":[],"mappings":"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","sourcesContent":["import { useQuery, UseQueryOptions } from '@tanstack/react-query';\nimport { getApiClient } from '../client';\n\n// ============================================================================\n// QUERY KEY FACTORY\n// ============================================================================\n\nexport const matchingKeys = {\n  all: ['matching'] as const,\n  discover: (filters?: MatchQueryFilters) => [...matchingKeys.all, 'discover', filters] as const,\n  matches: (filters?: MatchQueryFilters) => [...matchingKeys.all, 'matches', filters] as const,\n  match: (id: string) => [...matchingKeys.all, 'match', id] as const,\n  buddies: () => [...matchingKeys.all, 'buddies'] as const,\n  buddyRequests: () => [...matchingKeys.all, 'buddy-requests'] as const,\n  stats: () => [...matchingKeys.all, 'stats'] as const,\n};\n\n// ============================================================================\n// TYPES\n// ============================================================================\n\nexport interface MatchQueryFilters {\n  status?: 'PENDING' | 'ACCEPTED' | 'DECLINED' | 'BLOCKED';\n  limit?: number;\n  offset?: number;\n}\n\nexport interface MatchResponse {\n  id: string;\n  userId: string;\n  matchedUserId: string;\n  status: 'PENDING' | 'ACCEPTED' | 'DECLINED' | 'BLOCKED';\n  matchScore?: number;\n  sharedInterests?: string[];\n  createdAt: string;\n  updatedAt: string;\n  matchedUser?: {\n    id: string;\n    name: string;\n    avatarUrl?: string;\n    bio?: string;\n    city?: string;\n  };\n}\n\nexport interface DiscoverMatchResponse {\n  id: string;\n  name: string;\n  avatarUrl?: string;\n  bio?: string;\n  city?: string;\n  matchScore: number;\n  sharedInterests: string[];\n  distance?: number;\n}\n\nexport interface BuddyResponse {\n  id: string;\n  userId: string;\n  buddyId: string;\n  status: 'PENDING' | 'ACTIVE' | 'ENDED';\n  startDate?: string;\n  endDate?: string;\n  lastActivity?: string;\n  buddy?: {\n    id: string;\n    name: string;\n    avatarUrl?: string;\n  };\n}\n\nexport interface MatchingStatsResponse {\n  totalMatches: number;\n  pendingMatches: number;\n  acceptedMatches: number;\n  activeBuddies: number;\n  matchRate: number;\n}\n\n// ============================================================================\n// QUERY HOOKS\n// ============================================================================\n\n/**\n * Discover potential matches based on shared interests and location\n *\n * @endpoint GET /api/v1/matching/discover\n *\n * @example\n * ```tsx\n * const { data: matches, isLoading } = useDiscoverMatches();\n * ```\n */\nexport function useDiscoverMatches(\n  filters?: MatchQueryFilters,\n  options?: Omit<UseQueryOptions<DiscoverMatchResponse[]>, 'queryKey' | 'queryFn'>\n) {\n  return useQuery({\n    queryKey: matchingKeys.discover(filters),\n    queryFn: async (): Promise<DiscoverMatchResponse[]> => {\n      const client = getApiClient();\n      const response = await client.get('/api/v1/matching/discover', {\n        params: filters,\n      });\n      return response.data?.data || response.data;\n    },\n    ...options,\n  });\n}\n\n/**\n * Get all my matches\n *\n * @endpoint GET /api/v1/matching/matches\n *\n * @example\n * ```tsx\n * const { data: matches, isLoading } = useMyMatches({ status: 'PENDING' });\n * ```\n */\nexport function useMyMatches(\n  filters?: MatchQueryFilters,\n  options?: Omit<UseQueryOptions<MatchResponse[]>, 'queryKey' | 'queryFn'>\n) {\n  return useQuery({\n    queryKey: matchingKeys.matches(filters),\n    queryFn: async (): Promise<MatchResponse[]> => {\n      const client = getApiClient();\n      const response = await client.get('/api/v1/matching/matches', {\n        params: filters,\n      });\n      return response.data?.data || response.data;\n    },\n    staleTime: 30 * 1000,\n    ...options,\n  });\n}\n\n/**\n * Get pending matches (convenience hook)\n *\n * @example\n * ```tsx\n * const { data: pendingMatches } = usePendingMatches();\n * ```\n */\nexport function usePendingMatches(\n  options?: Omit<UseQueryOptions<MatchResponse[]>, 'queryKey' | 'queryFn'>\n) {\n  return useMyMatches({ status: 'PENDING' }, options);\n}\n\n/**\n * Get accepted matches (convenience hook)\n *\n * @example\n * ```tsx\n * const { data: acceptedMatches } = useAcceptedMatches();\n * ```\n */\nexport function useAcceptedMatches(\n  options?: Omit<UseQueryOptions<MatchResponse[]>, 'queryKey' | 'queryFn'>\n) {\n  return useMyMatches({ status: 'ACCEPTED' }, options);\n}\n\n/**\n * Get my accountability buddies\n *\n * @endpoint GET /api/v1/matching/buddies\n *\n * @example\n * ```tsx\n * const { data: buddies, isLoading } = useMyBuddies();\n * ```\n */\nexport function useMyBuddies(\n  options?: Omit<UseQueryOptions<BuddyResponse[]>, 'queryKey' | 'queryFn'>\n) {\n  return useQuery({\n    queryKey: matchingKeys.buddies(),\n    queryFn: async (): Promise<BuddyResponse[]> => {\n      const client = getApiClient();\n      const response = await client.get('/api/v1/matching/buddies');\n      return response.data?.data || response.data;\n    },\n    ...options,\n  });\n}\n\n/**\n * Get pending buddy requests\n *\n * @endpoint GET /api/v1/matching/buddies/requests\n *\n * @example\n * ```tsx\n * const { data: requests } = useBuddyRequests();\n * ```\n */\nexport function useBuddyRequests(\n  options?: Omit<UseQueryOptions<BuddyResponse[]>, 'queryKey' | 'queryFn'>\n) {\n  return useQuery({\n    queryKey: matchingKeys.buddyRequests(),\n    queryFn: async (): Promise<BuddyResponse[]> => {\n      const client = getApiClient();\n      const response = await client.get('/api/v1/matching/buddies/requests');\n      return response.data?.data || response.data;\n    },\n    ...options,\n  });\n}\n\n/**\n * Get matching statistics\n *\n * @endpoint GET /api/v1/matching/stats\n *\n * @example\n * ```tsx\n * const { data: stats } = useMatchingStats();\n * ```\n */\nexport function useMatchingStats(\n  options?: Omit<UseQueryOptions<MatchingStatsResponse>, 'queryKey' | 'queryFn'>\n) {\n  return useQuery({\n    queryKey: matchingKeys.stats(),\n    queryFn: async (): Promise<MatchingStatsResponse> => {\n      const client = getApiClient();\n      const response = await client.get('/api/v1/matching/stats');\n      return response.data?.data || response.data;\n    },\n    staleTime: 5 * 60 * 1000,\n    ...options,\n  });\n}\n"]}