@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,261 @@
1
+ "use strict";
2
+ /**
3
+ * Users Mutation Hooks
4
+ *
5
+ * TanStack Query mutation hooks for user-related write operations.
6
+ * These hooks handle user profile updates and onboarding completion.
7
+ *
8
+ * @module api/mutations/users
9
+ */
10
+ Object.defineProperty(exports, "__esModule", { value: true });
11
+ exports.useUpdateCurrentUser = useUpdateCurrentUser;
12
+ exports.useCompleteOnboarding = useCompleteOnboarding;
13
+ const react_query_1 = require("@tanstack/react-query");
14
+ const client_1 = require("../client");
15
+ const users_1 = require("../queries/users");
16
+ // ============================================================================
17
+ // MUTATION HOOKS
18
+ // ============================================================================
19
+ /**
20
+ * Update current user's profile
21
+ *
22
+ * @description
23
+ * Updates the authenticated user's profile information.
24
+ * This can include name, bio, avatar, location, sober date, and other profile fields.
25
+ * Automatically invalidates relevant user queries to refresh the UI.
26
+ *
27
+ * @endpoint PUT /api/v1/users/me
28
+ *
29
+ * @example
30
+ * ```tsx
31
+ * import { useUpdateCurrentUser } from '@growsober/sdk';
32
+ *
33
+ * function EditProfileForm() {
34
+ * const { mutate: updateProfile, isPending, error } = useUpdateCurrentUser({
35
+ * onSuccess: (data) => {
36
+ * toast.success('Profile updated successfully');
37
+ * navigation.goBack();
38
+ * },
39
+ * onError: (error) => {
40
+ * toast.error('Failed to update profile: ' + error.message);
41
+ * },
42
+ * });
43
+ *
44
+ * const handleSubmit = (formData: FormData) => {
45
+ * updateProfile({
46
+ * name: formData.name,
47
+ * bio: formData.bio,
48
+ * avatar: formData.avatar,
49
+ * city: formData.city,
50
+ * soberDate: formData.soberDate,
51
+ * });
52
+ * };
53
+ *
54
+ * return (
55
+ * <form onSubmit={handleSubmit}>
56
+ * <input name="name" />
57
+ * <textarea name="bio" />
58
+ * <button type="submit" disabled={isPending}>
59
+ * {isPending ? 'Saving...' : 'Save Profile'}
60
+ * </button>
61
+ * {error && <p className="error">{error.message}</p>}
62
+ * </form>
63
+ * );
64
+ * }
65
+ * ```
66
+ *
67
+ * @example
68
+ * Update specific fields:
69
+ * ```tsx
70
+ * function AvatarUploader() {
71
+ * const { mutateAsync: updateProfile } = useUpdateCurrentUser();
72
+ *
73
+ * const handleAvatarChange = async (file: File) => {
74
+ * const uploadedUrl = await uploadImage(file);
75
+ * await updateProfile({ avatar: uploadedUrl });
76
+ * toast.success('Avatar updated!');
77
+ * };
78
+ *
79
+ * return <ImagePicker onSelect={handleAvatarChange} />;
80
+ * }
81
+ * ```
82
+ *
83
+ * @example
84
+ * Update sober date:
85
+ * ```tsx
86
+ * function SoberDatePicker() {
87
+ * const { mutate: updateProfile } = useUpdateCurrentUser();
88
+ * const { data: user } = useCurrentUser();
89
+ *
90
+ * const handleDateChange = (date: Date) => {
91
+ * updateProfile({
92
+ * soberDate: date.toISOString(),
93
+ * });
94
+ * };
95
+ *
96
+ * return (
97
+ * <DatePicker
98
+ * value={user?.soberDate}
99
+ * onChange={handleDateChange}
100
+ * />
101
+ * );
102
+ * }
103
+ * ```
104
+ *
105
+ * @param options - TanStack Query mutation options
106
+ * @returns TanStack Query mutation result
107
+ */
108
+ function useUpdateCurrentUser(options) {
109
+ const queryClient = (0, react_query_1.useQueryClient)();
110
+ return (0, react_query_1.useMutation)({
111
+ mutationFn: async (data) => {
112
+ const client = (0, client_1.getApiClient)();
113
+ const response = await client.put('/api/v1/users/me', data);
114
+ return response.data;
115
+ },
116
+ onSuccess: (data, variables, context) => {
117
+ // Invalidate current user query to trigger refetch with updated data
118
+ queryClient.invalidateQueries({ queryKey: users_1.userKeys.me() });
119
+ // Also invalidate the detail query for this user if they've viewed their own profile
120
+ if (data.id) {
121
+ queryClient.invalidateQueries({ queryKey: users_1.userKeys.detail(data.id) });
122
+ queryClient.invalidateQueries({ queryKey: users_1.userKeys.public(data.id) });
123
+ }
124
+ // Call user's onSuccess if provided
125
+ // User's onSuccess is handled by spreading options
126
+ },
127
+ ...options,
128
+ });
129
+ }
130
+ /**
131
+ * Complete user onboarding
132
+ *
133
+ * @description
134
+ * Marks the user's onboarding process as complete.
135
+ * This is typically called after a user completes all required onboarding steps
136
+ * (e.g., setting up profile, selecting interests, setting sober date).
137
+ * Updates the user's onboardingCompleted flag and unlocks full app access.
138
+ *
139
+ * @endpoint POST /api/v1/users/me/complete-onboarding
140
+ *
141
+ * @example
142
+ * ```tsx
143
+ * import { useCompleteOnboarding } from '@growsober/sdk';
144
+ *
145
+ * function OnboardingFinalStep() {
146
+ * const { mutate: completeOnboarding, isPending } = useCompleteOnboarding({
147
+ * onSuccess: () => {
148
+ * toast.success('Welcome to GrowSober!');
149
+ * navigation.navigate('Home');
150
+ * },
151
+ * onError: (error) => {
152
+ * toast.error('Failed to complete onboarding: ' + error.message);
153
+ * },
154
+ * });
155
+ *
156
+ * const handleFinish = () => {
157
+ * completeOnboarding();
158
+ * };
159
+ *
160
+ * return (
161
+ * <div>
162
+ * <h1>You're All Set!</h1>
163
+ * <p>Ready to start your sober journey?</p>
164
+ * <button onClick={handleFinish} disabled={isPending}>
165
+ * {isPending ? 'Setting up...' : 'Get Started'}
166
+ * </button>
167
+ * </div>
168
+ * );
169
+ * }
170
+ * ```
171
+ *
172
+ * @example
173
+ * Multi-step onboarding with validation:
174
+ * ```tsx
175
+ * function OnboardingFlow() {
176
+ * const [currentStep, setCurrentStep] = useState(1);
177
+ * const { mutateAsync: completeOnboarding } = useCompleteOnboarding();
178
+ * const { data: user } = useCurrentUser();
179
+ *
180
+ * const handleComplete = async () => {
181
+ * // Validate all required fields are filled
182
+ * if (!user?.name || !user?.soberDate || !user?.city) {
183
+ * toast.error('Please complete all required fields');
184
+ * return;
185
+ * }
186
+ *
187
+ * try {
188
+ * await completeOnboarding();
189
+ * // User will be redirected via onSuccess callback
190
+ * } catch (error) {
191
+ * console.error('Onboarding error:', error);
192
+ * }
193
+ * };
194
+ *
195
+ * return (
196
+ * <div>
197
+ * {currentStep === 1 && <ProfileSetup onNext={() => setCurrentStep(2)} />}
198
+ * {currentStep === 2 && <InterestSelection onNext={() => setCurrentStep(3)} />}
199
+ * {currentStep === 3 && <SoberDateSetup onNext={handleComplete} />}
200
+ * </div>
201
+ * );
202
+ * }
203
+ * ```
204
+ *
205
+ * @example
206
+ * Check onboarding status:
207
+ * ```tsx
208
+ * function AppNavigator() {
209
+ * const { data: user, isLoading } = useCurrentUser();
210
+ *
211
+ * if (isLoading) return <SplashScreen />;
212
+ *
213
+ * // Redirect to onboarding if not completed
214
+ * if (user && !user.onboardingCompleted) {
215
+ * return <OnboardingFlow />;
216
+ * }
217
+ *
218
+ * return <MainApp />;
219
+ * }
220
+ * ```
221
+ *
222
+ * @example
223
+ * With analytics tracking:
224
+ * ```tsx
225
+ * const { mutate: completeOnboarding } = useCompleteOnboarding({
226
+ * onSuccess: (data) => {
227
+ * analytics.track('Onboarding Completed', {
228
+ * userId: data.id,
229
+ * completedAt: new Date().toISOString(),
230
+ * profileComplete: !!data.bio && !!data.avatar,
231
+ * });
232
+ * navigation.navigate('Home');
233
+ * },
234
+ * });
235
+ * ```
236
+ *
237
+ * @param options - TanStack Query mutation options
238
+ * @returns TanStack Query mutation result
239
+ */
240
+ function useCompleteOnboarding(options) {
241
+ const queryClient = (0, react_query_1.useQueryClient)();
242
+ return (0, react_query_1.useMutation)({
243
+ mutationFn: async () => {
244
+ const client = (0, client_1.getApiClient)();
245
+ const response = await client.post('/api/v1/users/me/complete-onboarding');
246
+ return response.data;
247
+ },
248
+ onSuccess: (data, variables, context) => {
249
+ // Invalidate current user query to trigger refetch with updated onboarding status
250
+ queryClient.invalidateQueries({ queryKey: users_1.userKeys.me() });
251
+ // Also invalidate the detail query for this user
252
+ if (data.id) {
253
+ queryClient.invalidateQueries({ queryKey: users_1.userKeys.detail(data.id) });
254
+ }
255
+ // Call user's onSuccess if provided
256
+ // User's onSuccess is handled by spreading options
257
+ },
258
+ ...options,
259
+ });
260
+ }
261
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"users.js","sourceRoot":"","sources":["../../../src/api/mutations/users.ts"],"names":[],"mappings":";AAAA;;;;;;;GAOG;;AAoGH,oDA0BC;AAgHD,sDAyBC;AArQD,uDAA2G;AAC3G,sCAAyC;AACzC,4CAA4C;AAG5C,+EAA+E;AAC/E,iBAAiB;AACjB,+EAA+E;AAE/E;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAwFG;AACH,SAAgB,oBAAoB,CAClC,OAAwF;IAExF,MAAM,WAAW,GAAG,IAAA,4BAAc,GAAE,CAAC;IAErC,OAAO,IAAA,yBAAW,EAAC;QACjB,UAAU,EAAE,KAAK,EAAE,IAAuB,EAAyB,EAAE;YACnE,MAAM,MAAM,GAAG,IAAA,qBAAY,GAAE,CAAC;YAC9B,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,GAAG,CAAe,kBAAkB,EAAE,IAAI,CAAC,CAAC;YAC1E,OAAO,QAAQ,CAAC,IAAI,CAAC;QACvB,CAAC;QACD,SAAS,EAAE,CAAC,IAAI,EAAE,SAAS,EAAE,OAAO,EAAE,EAAE;YACtC,qEAAqE;YACrE,WAAW,CAAC,iBAAiB,CAAC,EAAE,QAAQ,EAAE,gBAAQ,CAAC,EAAE,EAAE,EAAE,CAAC,CAAC;YAE3D,qFAAqF;YACrF,IAAI,IAAI,CAAC,EAAE,EAAE,CAAC;gBACZ,WAAW,CAAC,iBAAiB,CAAC,EAAE,QAAQ,EAAE,gBAAQ,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC;gBACtE,WAAW,CAAC,iBAAiB,CAAC,EAAE,QAAQ,EAAE,gBAAQ,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC;YACxE,CAAC;YAED,oCAAoC;YACpC,mDAAmD;QACrD,CAAC;QACD,GAAG,OAAO;KACX,CAAC,CAAC;AACL,CAAC;AAED;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA6GG;AACH,SAAgB,qBAAqB,CACnC,OAA2E;IAE3E,MAAM,WAAW,GAAG,IAAA,4BAAc,GAAE,CAAC;IAErC,OAAO,IAAA,yBAAW,EAAC;QACjB,UAAU,EAAE,KAAK,IAA2B,EAAE;YAC5C,MAAM,MAAM,GAAG,IAAA,qBAAY,GAAE,CAAC;YAC9B,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,IAAI,CAAe,sCAAsC,CAAC,CAAC;YACzF,OAAO,QAAQ,CAAC,IAAI,CAAC;QACvB,CAAC;QACD,SAAS,EAAE,CAAC,IAAI,EAAE,SAAS,EAAE,OAAO,EAAE,EAAE;YACtC,kFAAkF;YAClF,WAAW,CAAC,iBAAiB,CAAC,EAAE,QAAQ,EAAE,gBAAQ,CAAC,EAAE,EAAE,EAAE,CAAC,CAAC;YAE3D,iDAAiD;YACjD,IAAI,IAAI,CAAC,EAAE,EAAE,CAAC;gBACZ,WAAW,CAAC,iBAAiB,CAAC,EAAE,QAAQ,EAAE,gBAAQ,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC;YACxE,CAAC;YAED,oCAAoC;YACpC,mDAAmD;QACrD,CAAC;QACD,GAAG,OAAO;KACX,CAAC,CAAC;AACL,CAAC","sourcesContent":["/**\n * Users Mutation Hooks\n *\n * TanStack Query mutation hooks for user-related write operations.\n * These hooks handle user profile updates and onboarding completion.\n *\n * @module api/mutations/users\n */\n\nimport { useMutation, UseMutationOptions, UseMutationResult, useQueryClient } from '@tanstack/react-query';\nimport { getApiClient } from '../client';\nimport { userKeys } from '../queries/users';\nimport type { UpdateUserRequest, UserResponse } from '../types';\n\n// ============================================================================\n// MUTATION HOOKS\n// ============================================================================\n\n/**\n * Update current user's profile\n *\n * @description\n * Updates the authenticated user's profile information.\n * This can include name, bio, avatar, location, sober date, and other profile fields.\n * Automatically invalidates relevant user queries to refresh the UI.\n *\n * @endpoint PUT /api/v1/users/me\n *\n * @example\n * ```tsx\n * import { useUpdateCurrentUser } from '@growsober/sdk';\n *\n * function EditProfileForm() {\n *   const { mutate: updateProfile, isPending, error } = useUpdateCurrentUser({\n *     onSuccess: (data) => {\n *       toast.success('Profile updated successfully');\n *       navigation.goBack();\n *     },\n *     onError: (error) => {\n *       toast.error('Failed to update profile: ' + error.message);\n *     },\n *   });\n *\n *   const handleSubmit = (formData: FormData) => {\n *     updateProfile({\n *       name: formData.name,\n *       bio: formData.bio,\n *       avatar: formData.avatar,\n *       city: formData.city,\n *       soberDate: formData.soberDate,\n *     });\n *   };\n *\n *   return (\n *     <form onSubmit={handleSubmit}>\n *       <input name=\"name\" />\n *       <textarea name=\"bio\" />\n *       <button type=\"submit\" disabled={isPending}>\n *         {isPending ? 'Saving...' : 'Save Profile'}\n *       </button>\n *       {error && <p className=\"error\">{error.message}</p>}\n *     </form>\n *   );\n * }\n * ```\n *\n * @example\n * Update specific fields:\n * ```tsx\n * function AvatarUploader() {\n *   const { mutateAsync: updateProfile } = useUpdateCurrentUser();\n *\n *   const handleAvatarChange = async (file: File) => {\n *     const uploadedUrl = await uploadImage(file);\n *     await updateProfile({ avatar: uploadedUrl });\n *     toast.success('Avatar updated!');\n *   };\n *\n *   return <ImagePicker onSelect={handleAvatarChange} />;\n * }\n * ```\n *\n * @example\n * Update sober date:\n * ```tsx\n * function SoberDatePicker() {\n *   const { mutate: updateProfile } = useUpdateCurrentUser();\n *   const { data: user } = useCurrentUser();\n *\n *   const handleDateChange = (date: Date) => {\n *     updateProfile({\n *       soberDate: date.toISOString(),\n *     });\n *   };\n *\n *   return (\n *     <DatePicker\n *       value={user?.soberDate}\n *       onChange={handleDateChange}\n *     />\n *   );\n * }\n * ```\n *\n * @param options - TanStack Query mutation options\n * @returns TanStack Query mutation result\n */\nexport function useUpdateCurrentUser(\n  options?: Omit<UseMutationOptions<UserResponse, Error, UpdateUserRequest>, 'mutationFn'>\n): UseMutationResult<UserResponse, Error, UpdateUserRequest> {\n  const queryClient = useQueryClient();\n\n  return useMutation({\n    mutationFn: async (data: UpdateUserRequest): Promise<UserResponse> => {\n      const client = getApiClient();\n      const response = await client.put<UserResponse>('/api/v1/users/me', data);\n      return response.data;\n    },\n    onSuccess: (data, variables, context) => {\n      // Invalidate current user query to trigger refetch with updated data\n      queryClient.invalidateQueries({ queryKey: userKeys.me() });\n\n      // Also invalidate the detail query for this user if they've viewed their own profile\n      if (data.id) {\n        queryClient.invalidateQueries({ queryKey: userKeys.detail(data.id) });\n        queryClient.invalidateQueries({ queryKey: userKeys.public(data.id) });\n      }\n\n      // Call user's onSuccess if provided\n      // User's onSuccess is handled by spreading options\n    },\n    ...options,\n  });\n}\n\n/**\n * Complete user onboarding\n *\n * @description\n * Marks the user's onboarding process as complete.\n * This is typically called after a user completes all required onboarding steps\n * (e.g., setting up profile, selecting interests, setting sober date).\n * Updates the user's onboardingCompleted flag and unlocks full app access.\n *\n * @endpoint POST /api/v1/users/me/complete-onboarding\n *\n * @example\n * ```tsx\n * import { useCompleteOnboarding } from '@growsober/sdk';\n *\n * function OnboardingFinalStep() {\n *   const { mutate: completeOnboarding, isPending } = useCompleteOnboarding({\n *     onSuccess: () => {\n *       toast.success('Welcome to GrowSober!');\n *       navigation.navigate('Home');\n *     },\n *     onError: (error) => {\n *       toast.error('Failed to complete onboarding: ' + error.message);\n *     },\n *   });\n *\n *   const handleFinish = () => {\n *     completeOnboarding();\n *   };\n *\n *   return (\n *     <div>\n *       <h1>You're All Set!</h1>\n *       <p>Ready to start your sober journey?</p>\n *       <button onClick={handleFinish} disabled={isPending}>\n *         {isPending ? 'Setting up...' : 'Get Started'}\n *       </button>\n *     </div>\n *   );\n * }\n * ```\n *\n * @example\n * Multi-step onboarding with validation:\n * ```tsx\n * function OnboardingFlow() {\n *   const [currentStep, setCurrentStep] = useState(1);\n *   const { mutateAsync: completeOnboarding } = useCompleteOnboarding();\n *   const { data: user } = useCurrentUser();\n *\n *   const handleComplete = async () => {\n *     // Validate all required fields are filled\n *     if (!user?.name || !user?.soberDate || !user?.city) {\n *       toast.error('Please complete all required fields');\n *       return;\n *     }\n *\n *     try {\n *       await completeOnboarding();\n *       // User will be redirected via onSuccess callback\n *     } catch (error) {\n *       console.error('Onboarding error:', error);\n *     }\n *   };\n *\n *   return (\n *     <div>\n *       {currentStep === 1 && <ProfileSetup onNext={() => setCurrentStep(2)} />}\n *       {currentStep === 2 && <InterestSelection onNext={() => setCurrentStep(3)} />}\n *       {currentStep === 3 && <SoberDateSetup onNext={handleComplete} />}\n *     </div>\n *   );\n * }\n * ```\n *\n * @example\n * Check onboarding status:\n * ```tsx\n * function AppNavigator() {\n *   const { data: user, isLoading } = useCurrentUser();\n *\n *   if (isLoading) return <SplashScreen />;\n *\n *   // Redirect to onboarding if not completed\n *   if (user && !user.onboardingCompleted) {\n *     return <OnboardingFlow />;\n *   }\n *\n *   return <MainApp />;\n * }\n * ```\n *\n * @example\n * With analytics tracking:\n * ```tsx\n * const { mutate: completeOnboarding } = useCompleteOnboarding({\n *   onSuccess: (data) => {\n *     analytics.track('Onboarding Completed', {\n *       userId: data.id,\n *       completedAt: new Date().toISOString(),\n *       profileComplete: !!data.bio && !!data.avatar,\n *     });\n *     navigation.navigate('Home');\n *   },\n * });\n * ```\n *\n * @param options - TanStack Query mutation options\n * @returns TanStack Query mutation result\n */\nexport function useCompleteOnboarding(\n  options?: Omit<UseMutationOptions<UserResponse, Error, void>, 'mutationFn'>\n): UseMutationResult<UserResponse, Error, void> {\n  const queryClient = useQueryClient();\n\n  return useMutation({\n    mutationFn: async (): Promise<UserResponse> => {\n      const client = getApiClient();\n      const response = await client.post<UserResponse>('/api/v1/users/me/complete-onboarding');\n      return response.data;\n    },\n    onSuccess: (data, variables, context) => {\n      // Invalidate current user query to trigger refetch with updated onboarding status\n      queryClient.invalidateQueries({ queryKey: userKeys.me() });\n\n      // Also invalidate the detail query for this user\n      if (data.id) {\n        queryClient.invalidateQueries({ queryKey: userKeys.detail(data.id) });\n      }\n\n      // Call user's onSuccess if provided\n      // User's onSuccess is handled by spreading options\n    },\n    ...options,\n  });\n}\n"]}
@@ -0,0 +1,257 @@
1
+ /**
2
+ * Admin Query Hooks
3
+ *
4
+ * TanStack Query hooks for admin-related read operations.
5
+ * These hooks handle fetching data for the admin dashboard including
6
+ * users, hubs, events, businesses, content, and analytics.
7
+ *
8
+ * @module api/queries/admin
9
+ */
10
+ import { UseQueryOptions, UseQueryResult } from '@tanstack/react-query';
11
+ import type { UserResponse, HubResponse, EventResponse, BusinessResponse, LibraryContentResponse } from '../types';
12
+ export interface AdminDashboardStats {
13
+ totalUsers: number;
14
+ activeUsers: number;
15
+ newUsersToday: number;
16
+ newUsersThisWeek: number;
17
+ totalHubs: number;
18
+ totalEvents: number;
19
+ upcomingEvents: number;
20
+ totalBusinesses: number;
21
+ verifiedBusinesses: number;
22
+ premiumUsers: number;
23
+ revenue: {
24
+ today: number;
25
+ thisWeek: number;
26
+ thisMonth: number;
27
+ };
28
+ }
29
+ export interface AdminAnalyticsData {
30
+ period: string;
31
+ userGrowth: Array<{
32
+ date: string;
33
+ count: number;
34
+ }>;
35
+ eventAttendance: Array<{
36
+ date: string;
37
+ count: number;
38
+ }>;
39
+ hubActivity: Array<{
40
+ date: string;
41
+ count: number;
42
+ }>;
43
+ revenue: Array<{
44
+ date: string;
45
+ amount: number;
46
+ }>;
47
+ topHubs: Array<{
48
+ id: string;
49
+ name: string;
50
+ memberCount: number;
51
+ }>;
52
+ topEvents: Array<{
53
+ id: string;
54
+ title: string;
55
+ attendeeCount: number;
56
+ }>;
57
+ }
58
+ export interface PaginatedResponse<T> {
59
+ data: T[];
60
+ meta: {
61
+ total: number;
62
+ page: number;
63
+ limit: number;
64
+ totalPages: number;
65
+ };
66
+ }
67
+ export interface AdminUserFilters {
68
+ page?: number;
69
+ limit?: number;
70
+ search?: string;
71
+ status?: 'active' | 'suspended' | 'deleted';
72
+ isPremium?: boolean;
73
+ sortBy?: string;
74
+ sortOrder?: 'asc' | 'desc';
75
+ }
76
+ export interface AdminHubFilters {
77
+ page?: number;
78
+ limit?: number;
79
+ search?: string;
80
+ status?: 'active' | 'inactive';
81
+ sortBy?: string;
82
+ sortOrder?: 'asc' | 'desc';
83
+ }
84
+ export interface AdminEventFilters {
85
+ page?: number;
86
+ limit?: number;
87
+ search?: string;
88
+ status?: 'draft' | 'published' | 'cancelled';
89
+ hubId?: string;
90
+ upcoming?: boolean;
91
+ sortBy?: string;
92
+ sortOrder?: 'asc' | 'desc';
93
+ }
94
+ export interface AdminBusinessFilters {
95
+ page?: number;
96
+ limit?: number;
97
+ search?: string;
98
+ isVerified?: boolean;
99
+ type?: string;
100
+ sortBy?: string;
101
+ sortOrder?: 'asc' | 'desc';
102
+ }
103
+ export interface AdminContentFilters {
104
+ page?: number;
105
+ limit?: number;
106
+ search?: string;
107
+ type?: string;
108
+ category?: string;
109
+ isFeatured?: boolean;
110
+ sortBy?: string;
111
+ sortOrder?: 'asc' | 'desc';
112
+ }
113
+ export declare const adminKeys: {
114
+ all: readonly ["admin"];
115
+ dashboard: () => readonly ["admin", "dashboard"];
116
+ analytics: (period?: string) => readonly ["admin", "analytics", string | undefined];
117
+ users: {
118
+ all: () => readonly ["admin", "users"];
119
+ list: (filters?: AdminUserFilters) => readonly ["admin", "users", "list", AdminUserFilters | undefined];
120
+ detail: (id: string) => readonly ["admin", "users", "detail", string];
121
+ };
122
+ hubs: {
123
+ all: () => readonly ["admin", "hubs"];
124
+ list: (filters?: AdminHubFilters) => readonly ["admin", "hubs", "list", AdminHubFilters | undefined];
125
+ detail: (id: string) => readonly ["admin", "hubs", "detail", string];
126
+ };
127
+ events: {
128
+ all: () => readonly ["admin", "events"];
129
+ list: (filters?: AdminEventFilters) => readonly ["admin", "events", "list", AdminEventFilters | undefined];
130
+ detail: (id: string) => readonly ["admin", "events", "detail", string];
131
+ };
132
+ businesses: {
133
+ all: () => readonly ["admin", "businesses"];
134
+ list: (filters?: AdminBusinessFilters) => readonly ["admin", "businesses", "list", AdminBusinessFilters | undefined];
135
+ detail: (id: string) => readonly ["admin", "businesses", "detail", string];
136
+ };
137
+ content: {
138
+ all: () => readonly ["admin", "content"];
139
+ list: (filters?: AdminContentFilters) => readonly ["admin", "content", "list", AdminContentFilters | undefined];
140
+ detail: (id: string) => readonly ["admin", "content", "detail", string];
141
+ };
142
+ };
143
+ /**
144
+ * Get admin dashboard statistics
145
+ *
146
+ * @description
147
+ * Retrieves key metrics for the admin dashboard including
148
+ * user counts, hub stats, event stats, and revenue data.
149
+ *
150
+ * @endpoint GET /api/v1/admin/dashboard
151
+ *
152
+ * @example
153
+ * ```tsx
154
+ * import { useAdminDashboard } from '@growsober/sdk';
155
+ *
156
+ * function AdminDashboard() {
157
+ * const { data: stats, isLoading } = useAdminDashboard();
158
+ *
159
+ * if (isLoading) return <Spinner />;
160
+ *
161
+ * return (
162
+ * <div>
163
+ * <StatCard label="Total Users" value={stats.totalUsers} />
164
+ * <StatCard label="Active Users" value={stats.activeUsers} />
165
+ * <StatCard label="Premium Users" value={stats.premiumUsers} />
166
+ * <StatCard label="Revenue This Month" value={stats.revenue.thisMonth} />
167
+ * </div>
168
+ * );
169
+ * }
170
+ * ```
171
+ */
172
+ export declare function useAdminDashboard(options?: Omit<UseQueryOptions<AdminDashboardStats>, 'queryKey' | 'queryFn'>): UseQueryResult<AdminDashboardStats>;
173
+ /**
174
+ * Get admin analytics data
175
+ *
176
+ * @description
177
+ * Retrieves detailed analytics data for charts and reports.
178
+ *
179
+ * @endpoint GET /api/v1/admin/analytics
180
+ *
181
+ * @example
182
+ * ```tsx
183
+ * import { useAdminAnalytics } from '@growsober/sdk';
184
+ *
185
+ * function AnalyticsPage() {
186
+ * const { data: analytics } = useAdminAnalytics('30d');
187
+ *
188
+ * return (
189
+ * <div>
190
+ * <LineChart data={analytics?.userGrowth} />
191
+ * <BarChart data={analytics?.eventAttendance} />
192
+ * </div>
193
+ * );
194
+ * }
195
+ * ```
196
+ */
197
+ export declare function useAdminAnalytics(period?: string, options?: Omit<UseQueryOptions<AdminAnalyticsData>, 'queryKey' | 'queryFn'>): UseQueryResult<AdminAnalyticsData>;
198
+ /**
199
+ * List users for admin
200
+ *
201
+ * @endpoint GET /api/v1/admin/users
202
+ */
203
+ export declare function useAdminUsers(filters?: AdminUserFilters, options?: Omit<UseQueryOptions<PaginatedResponse<UserResponse>>, 'queryKey' | 'queryFn'>): UseQueryResult<PaginatedResponse<UserResponse>>;
204
+ /**
205
+ * Get user detail for admin
206
+ *
207
+ * @endpoint GET /api/v1/admin/users/{id}
208
+ */
209
+ export declare function useAdminUser(id: string, options?: Omit<UseQueryOptions<UserResponse>, 'queryKey' | 'queryFn'>): UseQueryResult<UserResponse>;
210
+ /**
211
+ * List hubs for admin
212
+ *
213
+ * @endpoint GET /api/v1/admin/hubs
214
+ */
215
+ export declare function useAdminHubs(filters?: AdminHubFilters, options?: Omit<UseQueryOptions<PaginatedResponse<HubResponse>>, 'queryKey' | 'queryFn'>): UseQueryResult<PaginatedResponse<HubResponse>>;
216
+ /**
217
+ * Get hub detail for admin
218
+ *
219
+ * @endpoint GET /api/v1/admin/hubs/{id}
220
+ */
221
+ export declare function useAdminHub(id: string, options?: Omit<UseQueryOptions<HubResponse>, 'queryKey' | 'queryFn'>): UseQueryResult<HubResponse>;
222
+ /**
223
+ * List events for admin
224
+ *
225
+ * @endpoint GET /api/v1/admin/events
226
+ */
227
+ export declare function useAdminEvents(filters?: AdminEventFilters, options?: Omit<UseQueryOptions<PaginatedResponse<EventResponse>>, 'queryKey' | 'queryFn'>): UseQueryResult<PaginatedResponse<EventResponse>>;
228
+ /**
229
+ * Get event detail for admin
230
+ *
231
+ * @endpoint GET /api/v1/admin/events/{id}
232
+ */
233
+ export declare function useAdminEvent(id: string, options?: Omit<UseQueryOptions<EventResponse>, 'queryKey' | 'queryFn'>): UseQueryResult<EventResponse>;
234
+ /**
235
+ * List businesses for admin
236
+ *
237
+ * @endpoint GET /api/v1/admin/businesses
238
+ */
239
+ export declare function useAdminBusinesses(filters?: AdminBusinessFilters, options?: Omit<UseQueryOptions<PaginatedResponse<BusinessResponse>>, 'queryKey' | 'queryFn'>): UseQueryResult<PaginatedResponse<BusinessResponse>>;
240
+ /**
241
+ * Get business detail for admin
242
+ *
243
+ * @endpoint GET /api/v1/admin/businesses/{id}
244
+ */
245
+ export declare function useAdminBusiness(id: string, options?: Omit<UseQueryOptions<BusinessResponse>, 'queryKey' | 'queryFn'>): UseQueryResult<BusinessResponse>;
246
+ /**
247
+ * List content for admin
248
+ *
249
+ * @endpoint GET /api/v1/admin/content
250
+ */
251
+ export declare function useAdminContent(filters?: AdminContentFilters, options?: Omit<UseQueryOptions<PaginatedResponse<LibraryContentResponse>>, 'queryKey' | 'queryFn'>): UseQueryResult<PaginatedResponse<LibraryContentResponse>>;
252
+ /**
253
+ * Get content detail for admin
254
+ *
255
+ * @endpoint GET /api/v1/admin/content/{id}
256
+ */
257
+ export declare function useAdminContentItem(id: string, options?: Omit<UseQueryOptions<LibraryContentResponse>, 'queryKey' | 'queryFn'>): UseQueryResult<LibraryContentResponse>;