@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.
- package/README.md +276 -0
- package/dist/__tests__/e2e.test.d.ts +7 -0
- package/dist/__tests__/e2e.test.js +472 -0
- package/dist/api/client.d.ts +11 -0
- package/dist/api/client.js +61 -0
- package/dist/api/mutations/admin.d.ts +167 -0
- package/dist/api/mutations/admin.js +326 -0
- package/dist/api/mutations/ambassadors.d.ts +52 -0
- package/dist/api/mutations/ambassadors.js +148 -0
- package/dist/api/mutations/auth.d.ts +267 -0
- package/dist/api/mutations/auth.js +332 -0
- package/dist/api/mutations/bookings.d.ts +59 -0
- package/dist/api/mutations/bookings.js +143 -0
- package/dist/api/mutations/event-chat.d.ts +35 -0
- package/dist/api/mutations/event-chat.js +147 -0
- package/dist/api/mutations/events.d.ts +87 -0
- package/dist/api/mutations/events.js +205 -0
- package/dist/api/mutations/grow90.d.ts +36 -0
- package/dist/api/mutations/grow90.js +132 -0
- package/dist/api/mutations/hubs.d.ts +111 -0
- package/dist/api/mutations/hubs.js +240 -0
- package/dist/api/mutations/index.d.ts +22 -0
- package/dist/api/mutations/index.js +39 -0
- package/dist/api/mutations/jack.d.ts +61 -0
- package/dist/api/mutations/jack.js +104 -0
- package/dist/api/mutations/library.d.ts +67 -0
- package/dist/api/mutations/library.js +168 -0
- package/dist/api/mutations/map.d.ts +153 -0
- package/dist/api/mutations/map.js +181 -0
- package/dist/api/mutations/matching.d.ts +130 -0
- package/dist/api/mutations/matching.js +204 -0
- package/dist/api/mutations/notifications.d.ts +63 -0
- package/dist/api/mutations/notifications.js +106 -0
- package/dist/api/mutations/offers.d.ts +26 -0
- package/dist/api/mutations/offers.js +47 -0
- package/dist/api/mutations/subscriptions.d.ts +127 -0
- package/dist/api/mutations/subscriptions.js +140 -0
- package/dist/api/mutations/support.d.ts +165 -0
- package/dist/api/mutations/support.js +307 -0
- package/dist/api/mutations/users.d.ts +211 -0
- package/dist/api/mutations/users.js +261 -0
- package/dist/api/queries/admin.d.ts +257 -0
- package/dist/api/queries/admin.js +320 -0
- package/dist/api/queries/ambassadors.d.ts +53 -0
- package/dist/api/queries/ambassadors.js +98 -0
- package/dist/api/queries/auth.d.ts +16 -0
- package/dist/api/queries/auth.js +25 -0
- package/dist/api/queries/bookings.d.ts +91 -0
- package/dist/api/queries/bookings.js +102 -0
- package/dist/api/queries/businesses.d.ts +212 -0
- package/dist/api/queries/businesses.js +154 -0
- package/dist/api/queries/event-chat.d.ts +19 -0
- package/dist/api/queries/event-chat.js +75 -0
- package/dist/api/queries/events.d.ts +322 -0
- package/dist/api/queries/events.js +221 -0
- package/dist/api/queries/grow90.d.ts +26 -0
- package/dist/api/queries/grow90.js +85 -0
- package/dist/api/queries/hubs.d.ts +165 -0
- package/dist/api/queries/hubs.js +143 -0
- package/dist/api/queries/index.d.ts +23 -0
- package/dist/api/queries/index.js +40 -0
- package/dist/api/queries/jack.d.ts +63 -0
- package/dist/api/queries/jack.js +92 -0
- package/dist/api/queries/library.d.ts +132 -0
- package/dist/api/queries/library.js +120 -0
- package/dist/api/queries/map.d.ts +216 -0
- package/dist/api/queries/map.js +278 -0
- package/dist/api/queries/matching.d.ts +136 -0
- package/dist/api/queries/matching.js +161 -0
- package/dist/api/queries/notifications.d.ts +78 -0
- package/dist/api/queries/notifications.js +88 -0
- package/dist/api/queries/offers.d.ts +91 -0
- package/dist/api/queries/offers.js +103 -0
- package/dist/api/queries/subscriptions.d.ts +56 -0
- package/dist/api/queries/subscriptions.js +73 -0
- package/dist/api/queries/support.d.ts +106 -0
- package/dist/api/queries/support.js +202 -0
- package/dist/api/queries/users.d.ts +293 -0
- package/dist/api/queries/users.js +370 -0
- package/dist/api/types.d.ts +464 -0
- package/dist/api/types.js +9 -0
- package/dist/hooks/useAuth.d.ts +5 -0
- package/dist/hooks/useAuth.js +39 -0
- package/dist/hooks/useUser.d.ts +43 -0
- package/dist/hooks/useUser.js +44 -0
- package/dist/index.d.ts +36 -0
- package/dist/index.js +67 -0
- package/package.json +62 -0
- package/src/__tests__/e2e.test.ts +502 -0
- package/src/api/client.ts +71 -0
- package/src/api/mutations/admin.ts +531 -0
- package/src/api/mutations/ambassadors.ts +185 -0
- package/src/api/mutations/auth.ts +350 -0
- package/src/api/mutations/bookings.ts +190 -0
- package/src/api/mutations/event-chat.ts +177 -0
- package/src/api/mutations/events.ts +273 -0
- package/src/api/mutations/grow90.ts +169 -0
- package/src/api/mutations/hubs.ts +385 -0
- package/src/api/mutations/index.ts +23 -0
- package/src/api/mutations/jack.ts +130 -0
- package/src/api/mutations/library.ts +212 -0
- package/src/api/mutations/map.ts +230 -0
- package/src/api/mutations/matching.ts +271 -0
- package/src/api/mutations/notifications.ts +114 -0
- package/src/api/mutations/offers.ts +73 -0
- package/src/api/mutations/subscriptions.ts +162 -0
- package/src/api/mutations/support.ts +390 -0
- package/src/api/mutations/users.ts +271 -0
- package/src/api/queries/admin.ts +480 -0
- package/src/api/queries/ambassadors.ts +139 -0
- package/src/api/queries/auth.ts +24 -0
- package/src/api/queries/bookings.ts +135 -0
- package/src/api/queries/businesses.ts +203 -0
- package/src/api/queries/event-chat.ts +78 -0
- package/src/api/queries/events.ts +272 -0
- package/src/api/queries/grow90.ts +98 -0
- package/src/api/queries/hubs.ts +211 -0
- package/src/api/queries/index.ts +24 -0
- package/src/api/queries/jack.ts +127 -0
- package/src/api/queries/library.ts +166 -0
- package/src/api/queries/map.ts +331 -0
- package/src/api/queries/matching.ts +238 -0
- package/src/api/queries/notifications.ts +103 -0
- package/src/api/queries/offers.ts +136 -0
- package/src/api/queries/subscriptions.ts +91 -0
- package/src/api/queries/support.ts +235 -0
- package/src/api/queries/users.ts +393 -0
- package/src/api/types.ts +596 -0
- package/src/index.ts +57 -0
|
@@ -0,0 +1,332 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* Auth Mutation Hooks
|
|
4
|
+
*
|
|
5
|
+
* TanStack Query mutation hooks for authentication-related write operations.
|
|
6
|
+
* These hooks handle user registration, login, token refresh, and Firebase authentication.
|
|
7
|
+
*
|
|
8
|
+
* @module api/mutations/auth
|
|
9
|
+
*/
|
|
10
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
11
|
+
exports.useRegister = useRegister;
|
|
12
|
+
exports.useLogin = useLogin;
|
|
13
|
+
exports.useRefreshAuthToken = useRefreshAuthToken;
|
|
14
|
+
exports.useFirebaseAuth = useFirebaseAuth;
|
|
15
|
+
const react_query_1 = require("@tanstack/react-query");
|
|
16
|
+
const client_1 = require("../client");
|
|
17
|
+
const users_1 = require("../queries/users");
|
|
18
|
+
// ============================================================================
|
|
19
|
+
// MUTATION HOOKS
|
|
20
|
+
// ============================================================================
|
|
21
|
+
/**
|
|
22
|
+
* Register a new user account
|
|
23
|
+
*
|
|
24
|
+
* @description
|
|
25
|
+
* Creates a new user account with email/phone and password.
|
|
26
|
+
* Returns authentication tokens and user information upon successful registration.
|
|
27
|
+
*
|
|
28
|
+
* @endpoint POST /auth/register
|
|
29
|
+
*
|
|
30
|
+
* @example
|
|
31
|
+
* ```tsx
|
|
32
|
+
* import { useRegister } from '@growsober/sdk';
|
|
33
|
+
*
|
|
34
|
+
* function RegisterForm() {
|
|
35
|
+
* const { mutate: register, isPending, error } = useRegister({
|
|
36
|
+
* onSuccess: (data) => {
|
|
37
|
+
* // Store tokens securely
|
|
38
|
+
* await SecureStore.setItemAsync('accessToken', data.accessToken);
|
|
39
|
+
* await SecureStore.setItemAsync('refreshToken', data.refreshToken);
|
|
40
|
+
* navigation.navigate('Onboarding');
|
|
41
|
+
* },
|
|
42
|
+
* onError: (error) => {
|
|
43
|
+
* Alert.alert('Registration failed', error.message);
|
|
44
|
+
* },
|
|
45
|
+
* });
|
|
46
|
+
*
|
|
47
|
+
* const handleSubmit = () => {
|
|
48
|
+
* register({
|
|
49
|
+
* email: 'user@example.com',
|
|
50
|
+
* password: 'SecurePassword123!',
|
|
51
|
+
* name: 'John Doe',
|
|
52
|
+
* });
|
|
53
|
+
* };
|
|
54
|
+
*
|
|
55
|
+
* return <Button onPress={handleSubmit} disabled={isPending} />;
|
|
56
|
+
* }
|
|
57
|
+
* ```
|
|
58
|
+
*
|
|
59
|
+
* @param options - TanStack Query mutation options
|
|
60
|
+
* @returns TanStack Query mutation result
|
|
61
|
+
*/
|
|
62
|
+
function useRegister(options) {
|
|
63
|
+
const queryClient = (0, react_query_1.useQueryClient)();
|
|
64
|
+
return (0, react_query_1.useMutation)({
|
|
65
|
+
mutationFn: async (data) => {
|
|
66
|
+
const client = (0, client_1.getApiClient)();
|
|
67
|
+
const response = await client.post('/auth/register', data);
|
|
68
|
+
return response.data;
|
|
69
|
+
},
|
|
70
|
+
onSuccess: (data, variables, context) => {
|
|
71
|
+
// Invalidate current user query to trigger refetch with new token
|
|
72
|
+
queryClient.invalidateQueries({ queryKey: users_1.userKeys.me() });
|
|
73
|
+
// User's onSuccess is handled by spreading options
|
|
74
|
+
},
|
|
75
|
+
...options,
|
|
76
|
+
});
|
|
77
|
+
}
|
|
78
|
+
/**
|
|
79
|
+
* Login with email/phone and password
|
|
80
|
+
*
|
|
81
|
+
* @description
|
|
82
|
+
* Authenticates an existing user with their credentials.
|
|
83
|
+
* Returns authentication tokens and user information upon successful login.
|
|
84
|
+
*
|
|
85
|
+
* @endpoint POST /auth/login
|
|
86
|
+
*
|
|
87
|
+
* @example
|
|
88
|
+
* ```tsx
|
|
89
|
+
* import { useLogin } from '@growsober/sdk';
|
|
90
|
+
*
|
|
91
|
+
* function LoginForm() {
|
|
92
|
+
* const { mutate: login, isPending, error } = useLogin({
|
|
93
|
+
* onSuccess: (data) => {
|
|
94
|
+
* // Store tokens securely
|
|
95
|
+
* await SecureStore.setItemAsync('accessToken', data.accessToken);
|
|
96
|
+
* await SecureStore.setItemAsync('refreshToken', data.refreshToken);
|
|
97
|
+
* navigation.navigate('Home');
|
|
98
|
+
* },
|
|
99
|
+
* });
|
|
100
|
+
*
|
|
101
|
+
* const handleSubmit = () => {
|
|
102
|
+
* login({
|
|
103
|
+
* email: 'user@example.com',
|
|
104
|
+
* password: 'SecurePassword123!',
|
|
105
|
+
* });
|
|
106
|
+
* };
|
|
107
|
+
*
|
|
108
|
+
* return (
|
|
109
|
+
* <form onSubmit={handleSubmit}>
|
|
110
|
+
* <input type="email" name="email" />
|
|
111
|
+
* <input type="password" name="password" />
|
|
112
|
+
* <button type="submit" disabled={isPending}>
|
|
113
|
+
* {isPending ? 'Logging in...' : 'Login'}
|
|
114
|
+
* </button>
|
|
115
|
+
* {error && <p className="error">{error.message}</p>}
|
|
116
|
+
* </form>
|
|
117
|
+
* );
|
|
118
|
+
* }
|
|
119
|
+
* ```
|
|
120
|
+
*
|
|
121
|
+
* @example
|
|
122
|
+
* Login with phone number:
|
|
123
|
+
* ```tsx
|
|
124
|
+
* login({
|
|
125
|
+
* phone: '+1234567890',
|
|
126
|
+
* password: 'SecurePassword123!',
|
|
127
|
+
* });
|
|
128
|
+
* ```
|
|
129
|
+
*
|
|
130
|
+
* @param options - TanStack Query mutation options
|
|
131
|
+
* @returns TanStack Query mutation result
|
|
132
|
+
*/
|
|
133
|
+
function useLogin(options) {
|
|
134
|
+
const queryClient = (0, react_query_1.useQueryClient)();
|
|
135
|
+
return (0, react_query_1.useMutation)({
|
|
136
|
+
mutationFn: async (data) => {
|
|
137
|
+
const client = (0, client_1.getApiClient)();
|
|
138
|
+
const response = await client.post('/auth/login', data);
|
|
139
|
+
return response.data;
|
|
140
|
+
},
|
|
141
|
+
onSuccess: (data, variables, context) => {
|
|
142
|
+
// Invalidate current user query to trigger refetch with new token
|
|
143
|
+
queryClient.invalidateQueries({ queryKey: users_1.userKeys.me() });
|
|
144
|
+
// User's onSuccess is handled by spreading options
|
|
145
|
+
},
|
|
146
|
+
...options,
|
|
147
|
+
});
|
|
148
|
+
}
|
|
149
|
+
/**
|
|
150
|
+
* Refresh access token using refresh token
|
|
151
|
+
*
|
|
152
|
+
* @description
|
|
153
|
+
* Obtains a new access token using a valid refresh token.
|
|
154
|
+
* Should be called when the access token expires.
|
|
155
|
+
*
|
|
156
|
+
* @endpoint POST /auth/refresh
|
|
157
|
+
*
|
|
158
|
+
* @example
|
|
159
|
+
* ```tsx
|
|
160
|
+
* import { useRefreshAuthToken } from '@growsober/sdk';
|
|
161
|
+
*
|
|
162
|
+
* function useTokenRefresh() {
|
|
163
|
+
* const { mutateAsync: refreshToken } = useRefreshAuthToken();
|
|
164
|
+
*
|
|
165
|
+
* const handleTokenExpired = async () => {
|
|
166
|
+
* const storedRefreshToken = await SecureStore.getItemAsync('refreshToken');
|
|
167
|
+
*
|
|
168
|
+
* if (!storedRefreshToken) {
|
|
169
|
+
* navigation.navigate('Login');
|
|
170
|
+
* return;
|
|
171
|
+
* }
|
|
172
|
+
*
|
|
173
|
+
* try {
|
|
174
|
+
* const { accessToken, refreshToken: newRefreshToken } = await refreshToken({
|
|
175
|
+
* refreshToken: storedRefreshToken,
|
|
176
|
+
* });
|
|
177
|
+
*
|
|
178
|
+
* // Store new tokens
|
|
179
|
+
* await SecureStore.setItemAsync('accessToken', accessToken);
|
|
180
|
+
* await SecureStore.setItemAsync('refreshToken', newRefreshToken);
|
|
181
|
+
* } catch (error) {
|
|
182
|
+
* // Refresh token is invalid or expired
|
|
183
|
+
* navigation.navigate('Login');
|
|
184
|
+
* }
|
|
185
|
+
* };
|
|
186
|
+
*
|
|
187
|
+
* return { handleTokenExpired };
|
|
188
|
+
* }
|
|
189
|
+
* ```
|
|
190
|
+
*
|
|
191
|
+
* @example
|
|
192
|
+
* Integrate with SDK configuration:
|
|
193
|
+
* ```tsx
|
|
194
|
+
* import { configureSDK } from '@growsober/sdk';
|
|
195
|
+
*
|
|
196
|
+
* configureSDK({
|
|
197
|
+
* baseURL: 'https://api.growsober.app',
|
|
198
|
+
* getAccessToken: async () => {
|
|
199
|
+
* return await SecureStore.getItemAsync('accessToken');
|
|
200
|
+
* },
|
|
201
|
+
* refreshAccessToken: async () => {
|
|
202
|
+
* const refreshToken = await SecureStore.getItemAsync('refreshToken');
|
|
203
|
+
* const { accessToken, refreshToken: newRefreshToken } = await fetch('/auth/refresh', {
|
|
204
|
+
* method: 'POST',
|
|
205
|
+
* body: JSON.stringify({ refreshToken }),
|
|
206
|
+
* }).then(r => r.json());
|
|
207
|
+
*
|
|
208
|
+
* await SecureStore.setItemAsync('accessToken', accessToken);
|
|
209
|
+
* await SecureStore.setItemAsync('refreshToken', newRefreshToken);
|
|
210
|
+
*
|
|
211
|
+
* return accessToken;
|
|
212
|
+
* },
|
|
213
|
+
* onUnauthorized: () => {
|
|
214
|
+
* navigation.navigate('Login');
|
|
215
|
+
* },
|
|
216
|
+
* });
|
|
217
|
+
* ```
|
|
218
|
+
*
|
|
219
|
+
* @param options - TanStack Query mutation options
|
|
220
|
+
* @returns TanStack Query mutation result
|
|
221
|
+
*/
|
|
222
|
+
function useRefreshAuthToken(options) {
|
|
223
|
+
return (0, react_query_1.useMutation)({
|
|
224
|
+
mutationFn: async (data) => {
|
|
225
|
+
const client = (0, client_1.getApiClient)();
|
|
226
|
+
const response = await client.post('/auth/refresh', data);
|
|
227
|
+
return response.data;
|
|
228
|
+
},
|
|
229
|
+
...options,
|
|
230
|
+
});
|
|
231
|
+
}
|
|
232
|
+
/**
|
|
233
|
+
* Authenticate with Firebase ID token
|
|
234
|
+
*
|
|
235
|
+
* @description
|
|
236
|
+
* Authenticates a user using a Firebase ID token.
|
|
237
|
+
* Creates a new user account if one doesn't exist, or logs in an existing user.
|
|
238
|
+
* Returns GrowSober authentication tokens and user information.
|
|
239
|
+
*
|
|
240
|
+
* @endpoint POST /auth/firebase
|
|
241
|
+
*
|
|
242
|
+
* @example
|
|
243
|
+
* ```tsx
|
|
244
|
+
* import { useFirebaseAuth } from '@growsober/sdk';
|
|
245
|
+
* import { signInWithPhoneNumber } from 'firebase/auth';
|
|
246
|
+
*
|
|
247
|
+
* function PhoneAuthScreen() {
|
|
248
|
+
* const { mutate: firebaseAuth, isPending } = useFirebaseAuth({
|
|
249
|
+
* onSuccess: (data) => {
|
|
250
|
+
* // Store GrowSober tokens
|
|
251
|
+
* await SecureStore.setItemAsync('accessToken', data.accessToken);
|
|
252
|
+
* await SecureStore.setItemAsync('refreshToken', data.refreshToken);
|
|
253
|
+
*
|
|
254
|
+
* if (data.user.onboardingCompleted) {
|
|
255
|
+
* navigation.navigate('Home');
|
|
256
|
+
* } else {
|
|
257
|
+
* navigation.navigate('Onboarding');
|
|
258
|
+
* }
|
|
259
|
+
* },
|
|
260
|
+
* onError: (error) => {
|
|
261
|
+
* Alert.alert('Authentication failed', error.message);
|
|
262
|
+
* },
|
|
263
|
+
* });
|
|
264
|
+
*
|
|
265
|
+
* const handlePhoneAuth = async (phoneNumber: string) => {
|
|
266
|
+
* try {
|
|
267
|
+
* // Firebase authentication flow
|
|
268
|
+
* const confirmation = await signInWithPhoneNumber(auth, phoneNumber);
|
|
269
|
+
* const code = await promptUserForCode(); // Your UI to get verification code
|
|
270
|
+
* const credential = await confirmation.confirm(code);
|
|
271
|
+
*
|
|
272
|
+
* // Get Firebase ID token
|
|
273
|
+
* const idToken = await credential.user.getIdToken();
|
|
274
|
+
*
|
|
275
|
+
* // Authenticate with GrowSober backend
|
|
276
|
+
* firebaseAuth({ idToken });
|
|
277
|
+
* } catch (error) {
|
|
278
|
+
* console.error('Phone auth error:', error);
|
|
279
|
+
* }
|
|
280
|
+
* };
|
|
281
|
+
*
|
|
282
|
+
* return <PhoneInput onSubmit={handlePhoneAuth} disabled={isPending} />;
|
|
283
|
+
* }
|
|
284
|
+
* ```
|
|
285
|
+
*
|
|
286
|
+
* @example
|
|
287
|
+
* With Google Sign-In:
|
|
288
|
+
* ```tsx
|
|
289
|
+
* import { GoogleSignin } from '@react-native-google-signin/google-signin';
|
|
290
|
+
*
|
|
291
|
+
* const handleGoogleSignIn = async () => {
|
|
292
|
+
* const { idToken } = await GoogleSignin.signIn();
|
|
293
|
+
* firebaseAuth({ idToken });
|
|
294
|
+
* };
|
|
295
|
+
* ```
|
|
296
|
+
*
|
|
297
|
+
* @example
|
|
298
|
+
* With Apple Sign-In:
|
|
299
|
+
* ```tsx
|
|
300
|
+
* import * as AppleAuthentication from 'expo-apple-authentication';
|
|
301
|
+
*
|
|
302
|
+
* const handleAppleSignIn = async () => {
|
|
303
|
+
* const credential = await AppleAuthentication.signInAsync({
|
|
304
|
+
* requestedScopes: [
|
|
305
|
+
* AppleAuthentication.AppleAuthenticationScope.FULL_NAME,
|
|
306
|
+
* AppleAuthentication.AppleAuthenticationScope.EMAIL,
|
|
307
|
+
* ],
|
|
308
|
+
* });
|
|
309
|
+
* firebaseAuth({ idToken: credential.identityToken });
|
|
310
|
+
* };
|
|
311
|
+
* ```
|
|
312
|
+
*
|
|
313
|
+
* @param options - TanStack Query mutation options
|
|
314
|
+
* @returns TanStack Query mutation result
|
|
315
|
+
*/
|
|
316
|
+
function useFirebaseAuth(options) {
|
|
317
|
+
const queryClient = (0, react_query_1.useQueryClient)();
|
|
318
|
+
return (0, react_query_1.useMutation)({
|
|
319
|
+
mutationFn: async (data) => {
|
|
320
|
+
const client = (0, client_1.getApiClient)();
|
|
321
|
+
const response = await client.post('/auth/firebase', data);
|
|
322
|
+
return response.data;
|
|
323
|
+
},
|
|
324
|
+
onSuccess: (data, variables, context) => {
|
|
325
|
+
// Invalidate current user query to trigger refetch with new token
|
|
326
|
+
queryClient.invalidateQueries({ queryKey: users_1.userKeys.me() });
|
|
327
|
+
// User's onSuccess is handled by spreading options
|
|
328
|
+
},
|
|
329
|
+
...options,
|
|
330
|
+
});
|
|
331
|
+
}
|
|
332
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"auth.js","sourceRoot":"","sources":["../../../src/api/mutations/auth.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Auth Mutation Hooks\n *\n * TanStack Query mutation hooks for authentication-related write operations.\n * These hooks handle user registration, login, token refresh, and Firebase authentication.\n *\n * @module api/mutations/auth\n */\n\nimport { useMutation, UseMutationOptions, UseMutationResult, useQueryClient } from '@tanstack/react-query';\nimport { getApiClient } from '../client';\nimport { userKeys } from '../queries/users';\nimport type {\n  RegisterRequest,\n  LoginRequest,\n  RefreshTokenRequest,\n  FirebaseAuthRequest,\n  AuthResponse,\n  TokenResponse,\n} from '../types';\n\n// ============================================================================\n// MUTATION HOOKS\n// ============================================================================\n\n/**\n * Register a new user account\n *\n * @description\n * Creates a new user account with email/phone and password.\n * Returns authentication tokens and user information upon successful registration.\n *\n * @endpoint POST /auth/register\n *\n * @example\n * ```tsx\n * import { useRegister } from '@growsober/sdk';\n *\n * function RegisterForm() {\n *   const { mutate: register, isPending, error } = useRegister({\n *     onSuccess: (data) => {\n *       // Store tokens securely\n *       await SecureStore.setItemAsync('accessToken', data.accessToken);\n *       await SecureStore.setItemAsync('refreshToken', data.refreshToken);\n *       navigation.navigate('Onboarding');\n *     },\n *     onError: (error) => {\n *       Alert.alert('Registration failed', error.message);\n *     },\n *   });\n *\n *   const handleSubmit = () => {\n *     register({\n *       email: 'user@example.com',\n *       password: 'SecurePassword123!',\n *       name: 'John Doe',\n *     });\n *   };\n *\n *   return <Button onPress={handleSubmit} disabled={isPending} />;\n * }\n * ```\n *\n * @param options - TanStack Query mutation options\n * @returns TanStack Query mutation result\n */\nexport function useRegister(\n  options?: Omit<UseMutationOptions<AuthResponse, Error, RegisterRequest>, 'mutationFn'>\n): UseMutationResult<AuthResponse, Error, RegisterRequest> {\n  const queryClient = useQueryClient();\n\n  return useMutation({\n    mutationFn: async (data: RegisterRequest): Promise<AuthResponse> => {\n      const client = getApiClient();\n      const response = await client.post<AuthResponse>('/auth/register', data);\n      return response.data;\n    },\n    onSuccess: (data, variables, context) => {\n      // Invalidate current user query to trigger refetch with new token\n      queryClient.invalidateQueries({ queryKey: userKeys.me() });\n      // User's onSuccess is handled by spreading options\n    },\n    ...options,\n  });\n}\n\n/**\n * Login with email/phone and password\n *\n * @description\n * Authenticates an existing user with their credentials.\n * Returns authentication tokens and user information upon successful login.\n *\n * @endpoint POST /auth/login\n *\n * @example\n * ```tsx\n * import { useLogin } from '@growsober/sdk';\n *\n * function LoginForm() {\n *   const { mutate: login, isPending, error } = useLogin({\n *     onSuccess: (data) => {\n *       // Store tokens securely\n *       await SecureStore.setItemAsync('accessToken', data.accessToken);\n *       await SecureStore.setItemAsync('refreshToken', data.refreshToken);\n *       navigation.navigate('Home');\n *     },\n *   });\n *\n *   const handleSubmit = () => {\n *     login({\n *       email: 'user@example.com',\n *       password: 'SecurePassword123!',\n *     });\n *   };\n *\n *   return (\n *     <form onSubmit={handleSubmit}>\n *       <input type=\"email\" name=\"email\" />\n *       <input type=\"password\" name=\"password\" />\n *       <button type=\"submit\" disabled={isPending}>\n *         {isPending ? 'Logging in...' : 'Login'}\n *       </button>\n *       {error && <p className=\"error\">{error.message}</p>}\n *     </form>\n *   );\n * }\n * ```\n *\n * @example\n * Login with phone number:\n * ```tsx\n * login({\n *   phone: '+1234567890',\n *   password: 'SecurePassword123!',\n * });\n * ```\n *\n * @param options - TanStack Query mutation options\n * @returns TanStack Query mutation result\n */\nexport function useLogin(\n  options?: Omit<UseMutationOptions<AuthResponse, Error, LoginRequest>, 'mutationFn'>\n): UseMutationResult<AuthResponse, Error, LoginRequest> {\n  const queryClient = useQueryClient();\n\n  return useMutation({\n    mutationFn: async (data: LoginRequest): Promise<AuthResponse> => {\n      const client = getApiClient();\n      const response = await client.post<AuthResponse>('/auth/login', data);\n      return response.data;\n    },\n    onSuccess: (data, variables, context) => {\n      // Invalidate current user query to trigger refetch with new token\n      queryClient.invalidateQueries({ queryKey: userKeys.me() });\n      // User's onSuccess is handled by spreading options\n    },\n    ...options,\n  });\n}\n\n/**\n * Refresh access token using refresh token\n *\n * @description\n * Obtains a new access token using a valid refresh token.\n * Should be called when the access token expires.\n *\n * @endpoint POST /auth/refresh\n *\n * @example\n * ```tsx\n * import { useRefreshAuthToken } from '@growsober/sdk';\n *\n * function useTokenRefresh() {\n *   const { mutateAsync: refreshToken } = useRefreshAuthToken();\n *\n *   const handleTokenExpired = async () => {\n *     const storedRefreshToken = await SecureStore.getItemAsync('refreshToken');\n *\n *     if (!storedRefreshToken) {\n *       navigation.navigate('Login');\n *       return;\n *     }\n *\n *     try {\n *       const { accessToken, refreshToken: newRefreshToken } = await refreshToken({\n *         refreshToken: storedRefreshToken,\n *       });\n *\n *       // Store new tokens\n *       await SecureStore.setItemAsync('accessToken', accessToken);\n *       await SecureStore.setItemAsync('refreshToken', newRefreshToken);\n *     } catch (error) {\n *       // Refresh token is invalid or expired\n *       navigation.navigate('Login');\n *     }\n *   };\n *\n *   return { handleTokenExpired };\n * }\n * ```\n *\n * @example\n * Integrate with SDK configuration:\n * ```tsx\n * import { configureSDK } from '@growsober/sdk';\n *\n * configureSDK({\n *   baseURL: 'https://api.growsober.app',\n *   getAccessToken: async () => {\n *     return await SecureStore.getItemAsync('accessToken');\n *   },\n *   refreshAccessToken: async () => {\n *     const refreshToken = await SecureStore.getItemAsync('refreshToken');\n *     const { accessToken, refreshToken: newRefreshToken } = await fetch('/auth/refresh', {\n *       method: 'POST',\n *       body: JSON.stringify({ refreshToken }),\n *     }).then(r => r.json());\n *\n *     await SecureStore.setItemAsync('accessToken', accessToken);\n *     await SecureStore.setItemAsync('refreshToken', newRefreshToken);\n *\n *     return accessToken;\n *   },\n *   onUnauthorized: () => {\n *     navigation.navigate('Login');\n *   },\n * });\n * ```\n *\n * @param options - TanStack Query mutation options\n * @returns TanStack Query mutation result\n */\nexport function useRefreshAuthToken(\n  options?: Omit<UseMutationOptions<TokenResponse, Error, RefreshTokenRequest>, 'mutationFn'>\n): UseMutationResult<TokenResponse, Error, RefreshTokenRequest> {\n  return useMutation({\n    mutationFn: async (data: RefreshTokenRequest): Promise<TokenResponse> => {\n      const client = getApiClient();\n      const response = await client.post<TokenResponse>('/auth/refresh', data);\n      return response.data;\n    },\n    ...options,\n  });\n}\n\n/**\n * Authenticate with Firebase ID token\n *\n * @description\n * Authenticates a user using a Firebase ID token.\n * Creates a new user account if one doesn't exist, or logs in an existing user.\n * Returns GrowSober authentication tokens and user information.\n *\n * @endpoint POST /auth/firebase\n *\n * @example\n * ```tsx\n * import { useFirebaseAuth } from '@growsober/sdk';\n * import { signInWithPhoneNumber } from 'firebase/auth';\n *\n * function PhoneAuthScreen() {\n *   const { mutate: firebaseAuth, isPending } = useFirebaseAuth({\n *     onSuccess: (data) => {\n *       // Store GrowSober tokens\n *       await SecureStore.setItemAsync('accessToken', data.accessToken);\n *       await SecureStore.setItemAsync('refreshToken', data.refreshToken);\n *\n *       if (data.user.onboardingCompleted) {\n *         navigation.navigate('Home');\n *       } else {\n *         navigation.navigate('Onboarding');\n *       }\n *     },\n *     onError: (error) => {\n *       Alert.alert('Authentication failed', error.message);\n *     },\n *   });\n *\n *   const handlePhoneAuth = async (phoneNumber: string) => {\n *     try {\n *       // Firebase authentication flow\n *       const confirmation = await signInWithPhoneNumber(auth, phoneNumber);\n *       const code = await promptUserForCode(); // Your UI to get verification code\n *       const credential = await confirmation.confirm(code);\n *\n *       // Get Firebase ID token\n *       const idToken = await credential.user.getIdToken();\n *\n *       // Authenticate with GrowSober backend\n *       firebaseAuth({ idToken });\n *     } catch (error) {\n *       console.error('Phone auth error:', error);\n *     }\n *   };\n *\n *   return <PhoneInput onSubmit={handlePhoneAuth} disabled={isPending} />;\n * }\n * ```\n *\n * @example\n * With Google Sign-In:\n * ```tsx\n * import { GoogleSignin } from '@react-native-google-signin/google-signin';\n *\n * const handleGoogleSignIn = async () => {\n *   const { idToken } = await GoogleSignin.signIn();\n *   firebaseAuth({ idToken });\n * };\n * ```\n *\n * @example\n * With Apple Sign-In:\n * ```tsx\n * import * as AppleAuthentication from 'expo-apple-authentication';\n *\n * const handleAppleSignIn = async () => {\n *   const credential = await AppleAuthentication.signInAsync({\n *     requestedScopes: [\n *       AppleAuthentication.AppleAuthenticationScope.FULL_NAME,\n *       AppleAuthentication.AppleAuthenticationScope.EMAIL,\n *     ],\n *   });\n *   firebaseAuth({ idToken: credential.identityToken });\n * };\n * ```\n *\n * @param options - TanStack Query mutation options\n * @returns TanStack Query mutation result\n */\nexport function useFirebaseAuth(\n  options?: Omit<UseMutationOptions<AuthResponse, Error, FirebaseAuthRequest>, 'mutationFn'>\n): UseMutationResult<AuthResponse, Error, FirebaseAuthRequest> {\n  const queryClient = useQueryClient();\n\n  return useMutation({\n    mutationFn: async (data: FirebaseAuthRequest): Promise<AuthResponse> => {\n      const client = getApiClient();\n      const response = await client.post<AuthResponse>('/auth/firebase', data);\n      return response.data;\n    },\n    onSuccess: (data, variables, context) => {\n      // Invalidate current user query to trigger refetch with new token\n      queryClient.invalidateQueries({ queryKey: userKeys.me() });\n      // User's onSuccess is handled by spreading options\n    },\n    ...options,\n  });\n}\n"]}
|
|
@@ -0,0 +1,59 @@
|
|
|
1
|
+
import { UseMutationOptions, UseMutationResult } from '@tanstack/react-query';
|
|
2
|
+
import type { BookingResponse } from '../types';
|
|
3
|
+
/**
|
|
4
|
+
* Create a new booking for an event
|
|
5
|
+
*
|
|
6
|
+
* @param options - TanStack Query mutation options
|
|
7
|
+
*
|
|
8
|
+
* @example
|
|
9
|
+
* ```tsx
|
|
10
|
+
* const { mutate, isPending } = useCreateBooking();
|
|
11
|
+
*
|
|
12
|
+
* // Simple usage - just pass eventId
|
|
13
|
+
* mutate({ eventId: 'event-123' });
|
|
14
|
+
*
|
|
15
|
+
* // With optional data
|
|
16
|
+
* mutate({
|
|
17
|
+
* eventId: 'event-123',
|
|
18
|
+
* ticketCount: 2,
|
|
19
|
+
* notes: 'Vegetarian meal preferred'
|
|
20
|
+
* });
|
|
21
|
+
* ```
|
|
22
|
+
*/
|
|
23
|
+
export declare function useCreateBooking(options?: Omit<UseMutationOptions<BookingResponse, Error, {
|
|
24
|
+
eventId: string;
|
|
25
|
+
ticketCount?: number;
|
|
26
|
+
notes?: string;
|
|
27
|
+
}>, 'mutationFn'>): UseMutationResult<BookingResponse, Error, {
|
|
28
|
+
eventId: string;
|
|
29
|
+
ticketCount?: number;
|
|
30
|
+
notes?: string;
|
|
31
|
+
}>;
|
|
32
|
+
/**
|
|
33
|
+
* Cancel a booking
|
|
34
|
+
*
|
|
35
|
+
* @param options - TanStack Query mutation options
|
|
36
|
+
*
|
|
37
|
+
* @example
|
|
38
|
+
* ```tsx
|
|
39
|
+
* const { mutate, isPending } = useCancelBooking();
|
|
40
|
+
*
|
|
41
|
+
* mutate('booking-123');
|
|
42
|
+
* ```
|
|
43
|
+
*/
|
|
44
|
+
export declare function useCancelBooking(options?: Omit<UseMutationOptions<void, Error, string>, 'mutationFn'>): UseMutationResult<void, Error, string>;
|
|
45
|
+
/**
|
|
46
|
+
* Check in a booking
|
|
47
|
+
*
|
|
48
|
+
* Note: This is typically used by event hosts/ambassadors to check in attendees
|
|
49
|
+
*
|
|
50
|
+
* @param options - TanStack Query mutation options
|
|
51
|
+
*
|
|
52
|
+
* @example
|
|
53
|
+
* ```tsx
|
|
54
|
+
* const { mutate, isPending } = useCheckInBooking();
|
|
55
|
+
*
|
|
56
|
+
* mutate('booking-123');
|
|
57
|
+
* ```
|
|
58
|
+
*/
|
|
59
|
+
export declare function useCheckInBooking(options?: Omit<UseMutationOptions<BookingResponse, Error, string>, 'mutationFn'>): UseMutationResult<BookingResponse, Error, string>;
|
|
@@ -0,0 +1,143 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.useCreateBooking = useCreateBooking;
|
|
4
|
+
exports.useCancelBooking = useCancelBooking;
|
|
5
|
+
exports.useCheckInBooking = useCheckInBooking;
|
|
6
|
+
const react_query_1 = require("@tanstack/react-query");
|
|
7
|
+
const client_1 = require("../client");
|
|
8
|
+
const bookings_1 = require("../queries/bookings");
|
|
9
|
+
const events_1 = require("../queries/events");
|
|
10
|
+
// ============================================================================
|
|
11
|
+
// MUTATION HOOKS
|
|
12
|
+
// ============================================================================
|
|
13
|
+
/**
|
|
14
|
+
* Create a new booking for an event
|
|
15
|
+
*
|
|
16
|
+
* @param options - TanStack Query mutation options
|
|
17
|
+
*
|
|
18
|
+
* @example
|
|
19
|
+
* ```tsx
|
|
20
|
+
* const { mutate, isPending } = useCreateBooking();
|
|
21
|
+
*
|
|
22
|
+
* // Simple usage - just pass eventId
|
|
23
|
+
* mutate({ eventId: 'event-123' });
|
|
24
|
+
*
|
|
25
|
+
* // With optional data
|
|
26
|
+
* mutate({
|
|
27
|
+
* eventId: 'event-123',
|
|
28
|
+
* ticketCount: 2,
|
|
29
|
+
* notes: 'Vegetarian meal preferred'
|
|
30
|
+
* });
|
|
31
|
+
* ```
|
|
32
|
+
*/
|
|
33
|
+
function useCreateBooking(options) {
|
|
34
|
+
const queryClient = (0, react_query_1.useQueryClient)();
|
|
35
|
+
return (0, react_query_1.useMutation)({
|
|
36
|
+
mutationFn: async ({ eventId, ticketCount, notes, }) => {
|
|
37
|
+
const client = (0, client_1.getApiClient)();
|
|
38
|
+
const data = {};
|
|
39
|
+
if (ticketCount)
|
|
40
|
+
data.ticketCount = ticketCount;
|
|
41
|
+
if (notes)
|
|
42
|
+
data.notes = notes;
|
|
43
|
+
const response = await client.post(`/api/v1/events/${eventId}/book`, data);
|
|
44
|
+
return response.data;
|
|
45
|
+
},
|
|
46
|
+
onSuccess: (newBooking, variables, context) => {
|
|
47
|
+
// Invalidate user's bookings list
|
|
48
|
+
queryClient.invalidateQueries({ queryKey: bookings_1.bookingKeys.mine() });
|
|
49
|
+
// Invalidate all booking lists
|
|
50
|
+
queryClient.invalidateQueries({ queryKey: bookings_1.bookingKeys.lists() });
|
|
51
|
+
// Invalidate the event to reflect updated booking count
|
|
52
|
+
queryClient.invalidateQueries({
|
|
53
|
+
queryKey: events_1.eventKeys.detail(variables.eventId)
|
|
54
|
+
});
|
|
55
|
+
// Invalidate event bookings if applicable
|
|
56
|
+
queryClient.invalidateQueries({
|
|
57
|
+
queryKey: events_1.eventKeys.bookings(variables.eventId)
|
|
58
|
+
});
|
|
59
|
+
// Invalidate ambient events
|
|
60
|
+
queryClient.invalidateQueries({ queryKey: events_1.eventKeys.ambient() });
|
|
61
|
+
// Call user's onSuccess if provided
|
|
62
|
+
},
|
|
63
|
+
...options,
|
|
64
|
+
});
|
|
65
|
+
}
|
|
66
|
+
/**
|
|
67
|
+
* Cancel a booking
|
|
68
|
+
*
|
|
69
|
+
* @param options - TanStack Query mutation options
|
|
70
|
+
*
|
|
71
|
+
* @example
|
|
72
|
+
* ```tsx
|
|
73
|
+
* const { mutate, isPending } = useCancelBooking();
|
|
74
|
+
*
|
|
75
|
+
* mutate('booking-123');
|
|
76
|
+
* ```
|
|
77
|
+
*/
|
|
78
|
+
function useCancelBooking(options) {
|
|
79
|
+
const queryClient = (0, react_query_1.useQueryClient)();
|
|
80
|
+
return (0, react_query_1.useMutation)({
|
|
81
|
+
mutationFn: async (id) => {
|
|
82
|
+
const client = (0, client_1.getApiClient)();
|
|
83
|
+
await client.delete(`/api/v1/bookings/${id}`);
|
|
84
|
+
},
|
|
85
|
+
onSuccess: (data, bookingId, context) => {
|
|
86
|
+
// Remove the booking from cache
|
|
87
|
+
queryClient.removeQueries({ queryKey: bookings_1.bookingKeys.detail(bookingId) });
|
|
88
|
+
// Invalidate user's bookings list
|
|
89
|
+
queryClient.invalidateQueries({ queryKey: bookings_1.bookingKeys.mine() });
|
|
90
|
+
// Invalidate all booking lists
|
|
91
|
+
queryClient.invalidateQueries({ queryKey: bookings_1.bookingKeys.lists() });
|
|
92
|
+
// Invalidate QR code cache
|
|
93
|
+
queryClient.removeQueries({ queryKey: bookings_1.bookingKeys.qrCode(bookingId) });
|
|
94
|
+
// Invalidate events to reflect updated booking counts
|
|
95
|
+
queryClient.invalidateQueries({ queryKey: events_1.eventKeys.lists() });
|
|
96
|
+
queryClient.invalidateQueries({ queryKey: events_1.eventKeys.upcoming() });
|
|
97
|
+
queryClient.invalidateQueries({ queryKey: events_1.eventKeys.ambient() });
|
|
98
|
+
// Call user's onSuccess if provided
|
|
99
|
+
},
|
|
100
|
+
...options,
|
|
101
|
+
});
|
|
102
|
+
}
|
|
103
|
+
/**
|
|
104
|
+
* Check in a booking
|
|
105
|
+
*
|
|
106
|
+
* Note: This is typically used by event hosts/ambassadors to check in attendees
|
|
107
|
+
*
|
|
108
|
+
* @param options - TanStack Query mutation options
|
|
109
|
+
*
|
|
110
|
+
* @example
|
|
111
|
+
* ```tsx
|
|
112
|
+
* const { mutate, isPending } = useCheckInBooking();
|
|
113
|
+
*
|
|
114
|
+
* mutate('booking-123');
|
|
115
|
+
* ```
|
|
116
|
+
*/
|
|
117
|
+
function useCheckInBooking(options) {
|
|
118
|
+
const queryClient = (0, react_query_1.useQueryClient)();
|
|
119
|
+
return (0, react_query_1.useMutation)({
|
|
120
|
+
mutationFn: async (id) => {
|
|
121
|
+
const client = (0, client_1.getApiClient)();
|
|
122
|
+
const response = await client.post(`/api/v1/bookings/${id}/check-in`);
|
|
123
|
+
return response.data;
|
|
124
|
+
},
|
|
125
|
+
onSuccess: (checkedInBooking, bookingId, context) => {
|
|
126
|
+
// Update the booking in cache
|
|
127
|
+
queryClient.setQueryData(bookings_1.bookingKeys.detail(bookingId), checkedInBooking);
|
|
128
|
+
// Invalidate user's bookings list to reflect check-in status
|
|
129
|
+
queryClient.invalidateQueries({ queryKey: bookings_1.bookingKeys.mine() });
|
|
130
|
+
// Invalidate all booking lists
|
|
131
|
+
queryClient.invalidateQueries({ queryKey: bookings_1.bookingKeys.lists() });
|
|
132
|
+
// Invalidate event bookings if applicable
|
|
133
|
+
if (checkedInBooking.eventId) {
|
|
134
|
+
queryClient.invalidateQueries({
|
|
135
|
+
queryKey: events_1.eventKeys.bookings(checkedInBooking.eventId)
|
|
136
|
+
});
|
|
137
|
+
}
|
|
138
|
+
// Call user's onSuccess if provided
|
|
139
|
+
},
|
|
140
|
+
...options,
|
|
141
|
+
});
|
|
142
|
+
}
|
|
143
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bookings.js","sourceRoot":"","sources":["../../../src/api/mutations/bookings.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  useMutation,\n  useQueryClient,\n  UseMutationOptions,\n  UseMutationResult,\n} from '@tanstack/react-query';\nimport { getApiClient } from '../client';\nimport type { BookingResponse } from '../types';\nimport { bookingKeys } from '../queries/bookings';\nimport { eventKeys } from '../queries/events';\n\n// ============================================================================\n// MUTATION HOOKS\n// ============================================================================\n\n/**\n * Create a new booking for an event\n *\n * @param options - TanStack Query mutation options\n *\n * @example\n * ```tsx\n * const { mutate, isPending } = useCreateBooking();\n *\n * // Simple usage - just pass eventId\n * mutate({ eventId: 'event-123' });\n *\n * // With optional data\n * mutate({\n *   eventId: 'event-123',\n *   ticketCount: 2,\n *   notes: 'Vegetarian meal preferred'\n * });\n * ```\n */\nexport function useCreateBooking(\n  options?: Omit<\n    UseMutationOptions<\n      BookingResponse,\n      Error,\n      { eventId: string; ticketCount?: number; notes?: string }\n    >,\n    'mutationFn'\n  >\n): UseMutationResult<\n  BookingResponse,\n  Error,\n  { eventId: string; ticketCount?: number; notes?: string }\n> {\n  const queryClient = useQueryClient();\n\n  return useMutation({\n    mutationFn: async ({\n      eventId,\n      ticketCount,\n      notes,\n    }: {\n      eventId: string;\n      ticketCount?: number;\n      notes?: string;\n    }): Promise<BookingResponse> => {\n      const client = getApiClient();\n      const data: any = {};\n      if (ticketCount) data.ticketCount = ticketCount;\n      if (notes) data.notes = notes;\n      const response = await client.post(`/api/v1/events/${eventId}/book`, data);\n      return response.data;\n    },\n    onSuccess: (newBooking, variables, context) => {\n      // Invalidate user's bookings list\n      queryClient.invalidateQueries({ queryKey: bookingKeys.mine() });\n\n      // Invalidate all booking lists\n      queryClient.invalidateQueries({ queryKey: bookingKeys.lists() });\n\n      // Invalidate the event to reflect updated booking count\n      queryClient.invalidateQueries({\n        queryKey: eventKeys.detail(variables.eventId)\n      });\n\n      // Invalidate event bookings if applicable\n      queryClient.invalidateQueries({\n        queryKey: eventKeys.bookings(variables.eventId)\n      });\n\n      // Invalidate ambient events\n      queryClient.invalidateQueries({ queryKey: eventKeys.ambient() });\n\n      // Call user's onSuccess if provided\n    },\n    ...options,\n  });\n}\n\n/**\n * Cancel a booking\n *\n * @param options - TanStack Query mutation options\n *\n * @example\n * ```tsx\n * const { mutate, isPending } = useCancelBooking();\n *\n * mutate('booking-123');\n * ```\n */\nexport function useCancelBooking(\n  options?: Omit<UseMutationOptions<void, Error, string>, 'mutationFn'>\n): UseMutationResult<void, Error, string> {\n  const queryClient = useQueryClient();\n\n  return useMutation({\n    mutationFn: async (id: string): Promise<void> => {\n      const client = getApiClient();\n      await client.delete(`/api/v1/bookings/${id}`);\n    },\n    onSuccess: (data, bookingId, context) => {\n      // Remove the booking from cache\n      queryClient.removeQueries({ queryKey: bookingKeys.detail(bookingId) });\n\n      // Invalidate user's bookings list\n      queryClient.invalidateQueries({ queryKey: bookingKeys.mine() });\n\n      // Invalidate all booking lists\n      queryClient.invalidateQueries({ queryKey: bookingKeys.lists() });\n\n      // Invalidate QR code cache\n      queryClient.removeQueries({ queryKey: bookingKeys.qrCode(bookingId) });\n\n      // Invalidate events to reflect updated booking counts\n      queryClient.invalidateQueries({ queryKey: eventKeys.lists() });\n      queryClient.invalidateQueries({ queryKey: eventKeys.upcoming() });\n      queryClient.invalidateQueries({ queryKey: eventKeys.ambient() });\n\n      // Call user's onSuccess if provided\n    },\n    ...options,\n  });\n}\n\n/**\n * Check in a booking\n *\n * Note: This is typically used by event hosts/ambassadors to check in attendees\n *\n * @param options - TanStack Query mutation options\n *\n * @example\n * ```tsx\n * const { mutate, isPending } = useCheckInBooking();\n *\n * mutate('booking-123');\n * ```\n */\nexport function useCheckInBooking(\n  options?: Omit<UseMutationOptions<BookingResponse, Error, string>, 'mutationFn'>\n): UseMutationResult<BookingResponse, Error, string> {\n  const queryClient = useQueryClient();\n\n  return useMutation({\n    mutationFn: async (id: string): Promise<BookingResponse> => {\n      const client = getApiClient();\n      const response = await client.post(`/api/v1/bookings/${id}/check-in`);\n      return response.data;\n    },\n    onSuccess: (checkedInBooking, bookingId, context) => {\n      // Update the booking in cache\n      queryClient.setQueryData(\n        bookingKeys.detail(bookingId),\n        checkedInBooking\n      );\n\n      // Invalidate user's bookings list to reflect check-in status\n      queryClient.invalidateQueries({ queryKey: bookingKeys.mine() });\n\n      // Invalidate all booking lists\n      queryClient.invalidateQueries({ queryKey: bookingKeys.lists() });\n\n      // Invalidate event bookings if applicable\n      if (checkedInBooking.eventId) {\n        queryClient.invalidateQueries({\n          queryKey: eventKeys.bookings(checkedInBooking.eventId)\n        });\n      }\n\n      // Call user's onSuccess if provided\n    },\n    ...options,\n  });\n}\n"]}
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
import type { ChatMemberResponse, MessageResponse, EventChatResponse, SendMessageRequest, UpdateMessageRequest, UpdateChatSettingsRequest, UpdateMemberSettingsRequest } from '../types';
|
|
2
|
+
/**
|
|
3
|
+
* Update chat settings (hosts only)
|
|
4
|
+
*/
|
|
5
|
+
export declare function useUpdateChatSettings(eventId: string): import("@tanstack/react-query").UseMutationResult<EventChatResponse, Error, UpdateChatSettingsRequest, unknown>;
|
|
6
|
+
/**
|
|
7
|
+
* Join event chat
|
|
8
|
+
*/
|
|
9
|
+
export declare function useJoinEventChat(eventId: string): import("@tanstack/react-query").UseMutationResult<ChatMemberResponse, Error, void, unknown>;
|
|
10
|
+
/**
|
|
11
|
+
* Leave event chat
|
|
12
|
+
*/
|
|
13
|
+
export declare function useLeaveEventChat(eventId: string): import("@tanstack/react-query").UseMutationResult<void, Error, void, unknown>;
|
|
14
|
+
/**
|
|
15
|
+
* Update member settings (mute, nickname)
|
|
16
|
+
*/
|
|
17
|
+
export declare function useUpdateMemberSettings(eventId: string): import("@tanstack/react-query").UseMutationResult<ChatMemberResponse, Error, UpdateMemberSettingsRequest, unknown>;
|
|
18
|
+
/**
|
|
19
|
+
* Mark messages as read
|
|
20
|
+
*/
|
|
21
|
+
export declare function useMarkMessagesAsRead(eventId: string): import("@tanstack/react-query").UseMutationResult<void, Error, void, unknown>;
|
|
22
|
+
/**
|
|
23
|
+
* Send a message
|
|
24
|
+
*/
|
|
25
|
+
export declare function useSendEventChatMessage(eventId: string): import("@tanstack/react-query").UseMutationResult<MessageResponse, Error, SendMessageRequest, unknown>;
|
|
26
|
+
/**
|
|
27
|
+
* Update a message
|
|
28
|
+
*/
|
|
29
|
+
export declare function useUpdateEventChatMessage(eventId: string): import("@tanstack/react-query").UseMutationResult<MessageResponse, Error, {
|
|
30
|
+
messageId: string;
|
|
31
|
+
} & UpdateMessageRequest, unknown>;
|
|
32
|
+
/**
|
|
33
|
+
* Delete a message
|
|
34
|
+
*/
|
|
35
|
+
export declare function useDeleteEventChatMessage(eventId: string): import("@tanstack/react-query").UseMutationResult<void, Error, string, unknown>;
|