@growsober/sdk 1.0.2 → 1.0.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/api/mutations/auth.js +7 -7
- package/dist/api/mutations/event-chat.d.ts +10 -27
- package/dist/api/mutations/index.d.ts +1 -0
- package/dist/api/mutations/index.js +2 -1
- package/dist/api/mutations/user-pins.d.ts +132 -0
- package/dist/api/mutations/user-pins.js +158 -0
- package/dist/api/queries/index.d.ts +1 -0
- package/dist/api/queries/index.js +2 -1
- package/dist/api/queries/user-pins.d.ts +134 -0
- package/dist/api/queries/user-pins.js +141 -0
- package/package.json +1 -1
- package/src/api/mutations/auth.ts +6 -6
- package/src/api/mutations/index.ts +1 -0
- package/src/api/mutations/user-pins.ts +186 -0
- package/src/api/queries/index.ts +1 -0
- package/src/api/queries/user-pins.ts +193 -0
|
@@ -66,7 +66,7 @@ function useRegister(options) {
|
|
|
66
66
|
return (0, react_query_1.useMutation)({
|
|
67
67
|
mutationFn: async (data) => {
|
|
68
68
|
const client = (0, client_1.getApiClient)();
|
|
69
|
-
const response = await client.post('/auth/register', data);
|
|
69
|
+
const response = await client.post('/api/v1/auth/register', data);
|
|
70
70
|
return response.data;
|
|
71
71
|
},
|
|
72
72
|
onSuccess: (data, variables, context) => {
|
|
@@ -137,7 +137,7 @@ function useLogin(options) {
|
|
|
137
137
|
return (0, react_query_1.useMutation)({
|
|
138
138
|
mutationFn: async (data) => {
|
|
139
139
|
const client = (0, client_1.getApiClient)();
|
|
140
|
-
const response = await client.post('/auth/login', data);
|
|
140
|
+
const response = await client.post('/api/v1/auth/login', data);
|
|
141
141
|
return response.data;
|
|
142
142
|
},
|
|
143
143
|
onSuccess: (data, variables, context) => {
|
|
@@ -225,7 +225,7 @@ function useRefreshAuthToken(options) {
|
|
|
225
225
|
return (0, react_query_1.useMutation)({
|
|
226
226
|
mutationFn: async (data) => {
|
|
227
227
|
const client = (0, client_1.getApiClient)();
|
|
228
|
-
const response = await client.post('/auth/refresh', data);
|
|
228
|
+
const response = await client.post('/api/v1/auth/refresh', data);
|
|
229
229
|
return response.data;
|
|
230
230
|
},
|
|
231
231
|
...options,
|
|
@@ -320,7 +320,7 @@ function useFirebaseAuth(options) {
|
|
|
320
320
|
return (0, react_query_1.useMutation)({
|
|
321
321
|
mutationFn: async (data) => {
|
|
322
322
|
const client = (0, client_1.getApiClient)();
|
|
323
|
-
const response = await client.post('/auth/firebase', data);
|
|
323
|
+
const response = await client.post('/api/v1/auth/firebase', data);
|
|
324
324
|
return response.data;
|
|
325
325
|
},
|
|
326
326
|
onSuccess: (data, variables, context) => {
|
|
@@ -370,7 +370,7 @@ function useSendOtp(options) {
|
|
|
370
370
|
return (0, react_query_1.useMutation)({
|
|
371
371
|
mutationFn: async (data) => {
|
|
372
372
|
const client = (0, client_1.getApiClient)();
|
|
373
|
-
const response = await client.post('/auth/phone/send-otp', data);
|
|
373
|
+
const response = await client.post('/api/v1/auth/phone/send-otp', data);
|
|
374
374
|
return response.data;
|
|
375
375
|
},
|
|
376
376
|
...options,
|
|
@@ -420,7 +420,7 @@ function useVerifyOtp(options) {
|
|
|
420
420
|
return (0, react_query_1.useMutation)({
|
|
421
421
|
mutationFn: async (data) => {
|
|
422
422
|
const client = (0, client_1.getApiClient)();
|
|
423
|
-
const response = await client.post('/auth/phone/verify-otp', data);
|
|
423
|
+
const response = await client.post('/api/v1/auth/phone/verify-otp', data);
|
|
424
424
|
return response.data;
|
|
425
425
|
},
|
|
426
426
|
onSuccess: (data, variables, context) => {
|
|
@@ -430,4 +430,4 @@ function useVerifyOtp(options) {
|
|
|
430
430
|
...options,
|
|
431
431
|
});
|
|
432
432
|
}
|
|
433
|
-
//# 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  SendOtpRequest,\n  OtpSentResponse,\n  VerifyOtpRequest,\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\n/**\n * Response type for verify OTP including isNewUser flag\n */\nexport type VerifyOtpResponse = AuthResponse & { isNewUser: boolean };\n\n/**\n * Send OTP to phone number\n *\n * @description\n * Sends a verification code to the specified phone number via SMS or voice call.\n * Used for passwordless authentication.\n *\n * @endpoint POST /auth/phone/send-otp\n *\n * @example\n * ```tsx\n * import { useSendOtp } from '@growsober/sdk';\n *\n * function PhoneInputScreen() {\n *   const { mutate: sendOtp, isPending, error } = useSendOtp({\n *     onSuccess: (data) => {\n *       console.log('OTP sent to:', data.phone);\n *       navigation.navigate('VerifyOtp', { phone });\n *     },\n *     onError: (error) => {\n *       Alert.alert('Error', error.message);\n *     },\n *   });\n *\n *   const handleSend = () => {\n *     sendOtp({ phone: '+1234567890', channel: 'sms' });\n *   };\n *\n *   return <Button onPress={handleSend} disabled={isPending} />;\n * }\n * ```\n *\n * @param options - TanStack Query mutation options\n * @returns TanStack Query mutation result\n */\nexport function useSendOtp(\n  options?: Omit<UseMutationOptions<OtpSentResponse, Error, SendOtpRequest>, 'mutationFn'>\n): UseMutationResult<OtpSentResponse, Error, SendOtpRequest> {\n  return useMutation({\n    mutationFn: async (data: SendOtpRequest): Promise<OtpSentResponse> => {\n      const client = getApiClient();\n      const response = await client.post<OtpSentResponse>('/auth/phone/send-otp', data);\n      return response.data;\n    },\n    ...options,\n  });\n}\n\n/**\n * Verify OTP and authenticate\n *\n * @description\n * Verifies the OTP code sent to the phone number and authenticates the user.\n * If the user doesn't exist, a new account is created automatically.\n * Returns authentication tokens and user information.\n *\n * @endpoint POST /auth/phone/verify-otp\n *\n * @example\n * ```tsx\n * import { useVerifyOtp } from '@growsober/sdk';\n *\n * function VerifyOtpScreen({ phone }) {\n *   const { mutate: verifyOtp, isPending, error } = useVerifyOtp({\n *     onSuccess: async (data) => {\n *       await SecureStore.setItemAsync('accessToken', data.accessToken);\n *       await SecureStore.setItemAsync('refreshToken', data.refreshToken);\n *\n *       if (data.isNewUser) {\n *         navigation.navigate('Onboarding');\n *       } else {\n *         navigation.navigate('Home');\n *       }\n *     },\n *   });\n *\n *   const handleVerify = (code: string) => {\n *     verifyOtp({ phone, code });\n *   };\n *\n *   return <OtpInput onComplete={handleVerify} disabled={isPending} />;\n * }\n * ```\n *\n * @param options - TanStack Query mutation options\n * @returns TanStack Query mutation result\n */\nexport function useVerifyOtp(\n  options?: Omit<UseMutationOptions<VerifyOtpResponse, Error, VerifyOtpRequest>, 'mutationFn'>\n): UseMutationResult<VerifyOtpResponse, Error, VerifyOtpRequest> {\n  const queryClient = useQueryClient();\n\n  return useMutation({\n    mutationFn: async (data: VerifyOtpRequest): Promise<VerifyOtpResponse> => {\n      const client = getApiClient();\n      const response = await client.post<VerifyOtpResponse>('/auth/phone/verify-otp', 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    },\n    ...options,\n  });\n}\n"]}
|
|
433
|
+
//# 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  SendOtpRequest,\n  OtpSentResponse,\n  VerifyOtpRequest,\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>('/api/v1/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>('/api/v1/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>('/api/v1/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>('/api/v1/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\n/**\n * Response type for verify OTP including isNewUser flag\n */\nexport type VerifyOtpResponse = AuthResponse & { isNewUser: boolean };\n\n/**\n * Send OTP to phone number\n *\n * @description\n * Sends a verification code to the specified phone number via SMS or voice call.\n * Used for passwordless authentication.\n *\n * @endpoint POST /auth/phone/send-otp\n *\n * @example\n * ```tsx\n * import { useSendOtp } from '@growsober/sdk';\n *\n * function PhoneInputScreen() {\n *   const { mutate: sendOtp, isPending, error } = useSendOtp({\n *     onSuccess: (data) => {\n *       console.log('OTP sent to:', data.phone);\n *       navigation.navigate('VerifyOtp', { phone });\n *     },\n *     onError: (error) => {\n *       Alert.alert('Error', error.message);\n *     },\n *   });\n *\n *   const handleSend = () => {\n *     sendOtp({ phone: '+1234567890', channel: 'sms' });\n *   };\n *\n *   return <Button onPress={handleSend} disabled={isPending} />;\n * }\n * ```\n *\n * @param options - TanStack Query mutation options\n * @returns TanStack Query mutation result\n */\nexport function useSendOtp(\n  options?: Omit<UseMutationOptions<OtpSentResponse, Error, SendOtpRequest>, 'mutationFn'>\n): UseMutationResult<OtpSentResponse, Error, SendOtpRequest> {\n  return useMutation({\n    mutationFn: async (data: SendOtpRequest): Promise<OtpSentResponse> => {\n      const client = getApiClient();\n      const response = await client.post<OtpSentResponse>('/api/v1/auth/phone/send-otp', data);\n      return response.data;\n    },\n    ...options,\n  });\n}\n\n/**\n * Verify OTP and authenticate\n *\n * @description\n * Verifies the OTP code sent to the phone number and authenticates the user.\n * If the user doesn't exist, a new account is created automatically.\n * Returns authentication tokens and user information.\n *\n * @endpoint POST /auth/phone/verify-otp\n *\n * @example\n * ```tsx\n * import { useVerifyOtp } from '@growsober/sdk';\n *\n * function VerifyOtpScreen({ phone }) {\n *   const { mutate: verifyOtp, isPending, error } = useVerifyOtp({\n *     onSuccess: async (data) => {\n *       await SecureStore.setItemAsync('accessToken', data.accessToken);\n *       await SecureStore.setItemAsync('refreshToken', data.refreshToken);\n *\n *       if (data.isNewUser) {\n *         navigation.navigate('Onboarding');\n *       } else {\n *         navigation.navigate('Home');\n *       }\n *     },\n *   });\n *\n *   const handleVerify = (code: string) => {\n *     verifyOtp({ phone, code });\n *   };\n *\n *   return <OtpInput onComplete={handleVerify} disabled={isPending} />;\n * }\n * ```\n *\n * @param options - TanStack Query mutation options\n * @returns TanStack Query mutation result\n */\nexport function useVerifyOtp(\n  options?: Omit<UseMutationOptions<VerifyOtpResponse, Error, VerifyOtpRequest>, 'mutationFn'>\n): UseMutationResult<VerifyOtpResponse, Error, VerifyOtpRequest> {\n  const queryClient = useQueryClient();\n\n  return useMutation({\n    mutationFn: async (data: VerifyOtpRequest): Promise<VerifyOtpResponse> => {\n      const client = getApiClient();\n      const response = await client.post<VerifyOtpResponse>('/api/v1/auth/phone/verify-otp', 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    },\n    ...options,\n  });\n}\n"]}
|
|
@@ -62,45 +62,28 @@ export declare function useMarkMessagesAsRead(eventId: string): import("@tanstac
|
|
|
62
62
|
*/
|
|
63
63
|
export declare function useSendEventChatMessage(eventId: string): import("@tanstack/react-query").UseMutationResult<{
|
|
64
64
|
id: string;
|
|
65
|
-
|
|
66
|
-
|
|
65
|
+
conversationId: string;
|
|
66
|
+
role: "USER" | "ASSISTANT" | "SYSTEM";
|
|
67
67
|
content: string;
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
isEdited: boolean;
|
|
72
|
-
isDeleted: boolean;
|
|
68
|
+
model?: Record<string, never>;
|
|
69
|
+
promptTokens?: Record<string, never>;
|
|
70
|
+
completionTokens?: Record<string, never>;
|
|
73
71
|
createdAt: string;
|
|
74
|
-
updatedAt?: string;
|
|
75
|
-
userName?: string;
|
|
76
|
-
userImage?: string;
|
|
77
|
-
replyToContent?: string;
|
|
78
|
-
replyToUserName?: string;
|
|
79
72
|
}, Error, {
|
|
80
73
|
content: string;
|
|
81
|
-
messageType: "TEXT" | "IMAGE" | "SYSTEM" | "ANNOUNCEMENT";
|
|
82
|
-
imageUrl?: string;
|
|
83
|
-
replyToId?: string;
|
|
84
74
|
}, unknown>;
|
|
85
75
|
/**
|
|
86
76
|
* Update a message
|
|
87
77
|
*/
|
|
88
78
|
export declare function useUpdateEventChatMessage(eventId: string): import("@tanstack/react-query").UseMutationResult<{
|
|
89
79
|
id: string;
|
|
90
|
-
|
|
91
|
-
|
|
80
|
+
conversationId: string;
|
|
81
|
+
role: "USER" | "ASSISTANT" | "SYSTEM";
|
|
92
82
|
content: string;
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
isEdited: boolean;
|
|
97
|
-
isDeleted: boolean;
|
|
83
|
+
model?: Record<string, never>;
|
|
84
|
+
promptTokens?: Record<string, never>;
|
|
85
|
+
completionTokens?: Record<string, never>;
|
|
98
86
|
createdAt: string;
|
|
99
|
-
updatedAt?: string;
|
|
100
|
-
userName?: string;
|
|
101
|
-
userImage?: string;
|
|
102
|
-
replyToContent?: string;
|
|
103
|
-
replyToUserName?: string;
|
|
104
87
|
}, Error, {
|
|
105
88
|
messageId: string;
|
|
106
89
|
} & {
|
|
@@ -36,4 +36,5 @@ __exportStar(require("./ambassadors"), exports);
|
|
|
36
36
|
__exportStar(require("./grow90"), exports);
|
|
37
37
|
__exportStar(require("./matching"), exports);
|
|
38
38
|
__exportStar(require("./event-chat"), exports);
|
|
39
|
-
|
|
39
|
+
__exportStar(require("./user-pins"), exports);
|
|
40
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvYXBpL211dGF0aW9ucy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUE7Ozs7R0FJRzs7Ozs7Ozs7Ozs7Ozs7OztBQUVILDBDQUF3QjtBQUN4Qix5Q0FBdUI7QUFDdkIsNkNBQTJCO0FBQzNCLDJDQUF5QjtBQUN6Qix5Q0FBdUI7QUFDdkIsNENBQTBCO0FBQzFCLHdDQUFzQjtBQUN0QixrREFBZ0M7QUFDaEMsMkNBQXlCO0FBQ3pCLGtEQUFnQztBQUNoQyw0Q0FBMEI7QUFDMUIsMENBQXdCO0FBQ3hCLHlDQUF1QjtBQUN2QixnREFBOEI7QUFDOUIsMkNBQXlCO0FBQ3pCLDZDQUEyQjtBQUMzQiwrQ0FBNkI7QUFDN0IsOENBQTRCIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBNdXRhdGlvbiBIb29rc1xuICpcbiAqIFJlLWV4cG9ydHMgYWxsIG11dGF0aW9uIGhvb2tzIGZvciBBUEkgZW5kcG9pbnRzLlxuICovXG5cbmV4cG9ydCAqIGZyb20gJy4vYWRtaW4nO1xuZXhwb3J0ICogZnJvbSAnLi9hdXRoJztcbmV4cG9ydCAqIGZyb20gJy4vYm9va2luZ3MnO1xuZXhwb3J0ICogZnJvbSAnLi9ldmVudHMnO1xuZXhwb3J0ICogZnJvbSAnLi9odWJzJztcbmV4cG9ydCAqIGZyb20gJy4vbGlicmFyeSc7XG5leHBvcnQgKiBmcm9tICcuL21hcCc7XG5leHBvcnQgKiBmcm9tICcuL25vdGlmaWNhdGlvbnMnO1xuZXhwb3J0ICogZnJvbSAnLi9vZmZlcnMnO1xuZXhwb3J0ICogZnJvbSAnLi9zdWJzY3JpcHRpb25zJztcbmV4cG9ydCAqIGZyb20gJy4vc3VwcG9ydCc7XG5leHBvcnQgKiBmcm9tICcuL3VzZXJzJztcbmV4cG9ydCAqIGZyb20gJy4vamFjayc7XG5leHBvcnQgKiBmcm9tICcuL2FtYmFzc2Fkb3JzJztcbmV4cG9ydCAqIGZyb20gJy4vZ3JvdzkwJztcbmV4cG9ydCAqIGZyb20gJy4vbWF0Y2hpbmcnO1xuZXhwb3J0ICogZnJvbSAnLi9ldmVudC1jaGF0JztcbmV4cG9ydCAqIGZyb20gJy4vdXNlci1waW5zJztcbiJdfQ==
|
|
@@ -0,0 +1,132 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* User Pins Mutation Hooks
|
|
3
|
+
*
|
|
4
|
+
* TanStack Query mutation hooks for user pin operations.
|
|
5
|
+
* These hooks handle creating and deleting user pins
|
|
6
|
+
* ("I'm here now" / "I'll be there").
|
|
7
|
+
*
|
|
8
|
+
* @module api/mutations/user-pins
|
|
9
|
+
*/
|
|
10
|
+
import { UseMutationOptions, UseMutationResult } from '@tanstack/react-query';
|
|
11
|
+
import { UserPinResponse, PinType } from '../queries/user-pins';
|
|
12
|
+
export interface CreatePinRequest {
|
|
13
|
+
type: PinType;
|
|
14
|
+
latitude: number;
|
|
15
|
+
longitude: number;
|
|
16
|
+
locationName?: string;
|
|
17
|
+
activity: string;
|
|
18
|
+
venueId?: string;
|
|
19
|
+
scheduledTime?: string;
|
|
20
|
+
duration?: number;
|
|
21
|
+
}
|
|
22
|
+
/**
|
|
23
|
+
* Create a new user pin
|
|
24
|
+
*
|
|
25
|
+
* @description
|
|
26
|
+
* Creates a new pin showing where the user is or will be.
|
|
27
|
+
* Any existing active pin is automatically deactivated.
|
|
28
|
+
*
|
|
29
|
+
* Two types of pins:
|
|
30
|
+
* - HERE_NOW: User is currently at this location (default 1 hour duration)
|
|
31
|
+
* - SCHEDULED: User will be at this location at a specific time (default 2 hour duration)
|
|
32
|
+
*
|
|
33
|
+
* @endpoint POST /api/v1/user-pins
|
|
34
|
+
*
|
|
35
|
+
* @example
|
|
36
|
+
* Drop a "I'm here now" pin:
|
|
37
|
+
* ```tsx
|
|
38
|
+
* import { useCreateUserPin } from '@growsober/sdk';
|
|
39
|
+
*
|
|
40
|
+
* function DropPinButton({ location }) {
|
|
41
|
+
* const createPin = useCreateUserPin();
|
|
42
|
+
*
|
|
43
|
+
* const handleDropPin = () => {
|
|
44
|
+
* createPin.mutate({
|
|
45
|
+
* type: 'HERE_NOW',
|
|
46
|
+
* latitude: location.lat,
|
|
47
|
+
* longitude: location.lng,
|
|
48
|
+
* locationName: 'Fabrica Coffee',
|
|
49
|
+
* activity: 'Working remotely',
|
|
50
|
+
* duration: 120, // 2 hours
|
|
51
|
+
* });
|
|
52
|
+
* };
|
|
53
|
+
*
|
|
54
|
+
* return (
|
|
55
|
+
* <Button onPress={handleDropPin} loading={createPin.isPending}>
|
|
56
|
+
* Drop Pin
|
|
57
|
+
* </Button>
|
|
58
|
+
* );
|
|
59
|
+
* }
|
|
60
|
+
* ```
|
|
61
|
+
*
|
|
62
|
+
* @example
|
|
63
|
+
* Schedule a pin for later:
|
|
64
|
+
* ```tsx
|
|
65
|
+
* import { useCreateUserPin } from '@growsober/sdk';
|
|
66
|
+
*
|
|
67
|
+
* function SchedulePinForm() {
|
|
68
|
+
* const createPin = useCreateUserPin({
|
|
69
|
+
* onSuccess: () => {
|
|
70
|
+
* toast.show('Pin scheduled!');
|
|
71
|
+
* },
|
|
72
|
+
* });
|
|
73
|
+
*
|
|
74
|
+
* const handleSchedule = (data) => {
|
|
75
|
+
* createPin.mutate({
|
|
76
|
+
* type: 'SCHEDULED',
|
|
77
|
+
* latitude: data.venue.lat,
|
|
78
|
+
* longitude: data.venue.lng,
|
|
79
|
+
* locationName: data.venue.name,
|
|
80
|
+
* venueId: data.venue.id,
|
|
81
|
+
* activity: 'Coffee meetup',
|
|
82
|
+
* scheduledTime: data.time.toISOString(),
|
|
83
|
+
* duration: 90,
|
|
84
|
+
* });
|
|
85
|
+
* };
|
|
86
|
+
*
|
|
87
|
+
* return <ScheduleForm onSubmit={handleSchedule} />;
|
|
88
|
+
* }
|
|
89
|
+
* ```
|
|
90
|
+
*
|
|
91
|
+
* @param options - TanStack Query mutation options
|
|
92
|
+
* @returns TanStack Query mutation result
|
|
93
|
+
*/
|
|
94
|
+
export declare function useCreateUserPin(options?: Omit<UseMutationOptions<UserPinResponse, Error, CreatePinRequest>, 'mutationFn'>): UseMutationResult<UserPinResponse, Error, CreatePinRequest>;
|
|
95
|
+
/**
|
|
96
|
+
* Delete a user pin
|
|
97
|
+
*
|
|
98
|
+
* @description
|
|
99
|
+
* Deactivates a user pin. Users can only delete their own pins.
|
|
100
|
+
*
|
|
101
|
+
* @endpoint DELETE /api/v1/user-pins/:id
|
|
102
|
+
*
|
|
103
|
+
* @example
|
|
104
|
+
* ```tsx
|
|
105
|
+
* import { useDeleteUserPin } from '@growsober/sdk';
|
|
106
|
+
*
|
|
107
|
+
* function MyPinCard({ pin }) {
|
|
108
|
+
* const deletePin = useDeleteUserPin({
|
|
109
|
+
* onSuccess: () => {
|
|
110
|
+
* toast.show('Pin removed');
|
|
111
|
+
* },
|
|
112
|
+
* });
|
|
113
|
+
*
|
|
114
|
+
* return (
|
|
115
|
+
* <Card>
|
|
116
|
+
* <Text>{pin.activity}</Text>
|
|
117
|
+
* <Text>At {pin.locationName}</Text>
|
|
118
|
+
* <Button
|
|
119
|
+
* onPress={() => deletePin.mutate(pin.id)}
|
|
120
|
+
* loading={deletePin.isPending}
|
|
121
|
+
* >
|
|
122
|
+
* Remove Pin
|
|
123
|
+
* </Button>
|
|
124
|
+
* </Card>
|
|
125
|
+
* );
|
|
126
|
+
* }
|
|
127
|
+
* ```
|
|
128
|
+
*
|
|
129
|
+
* @param options - TanStack Query mutation options
|
|
130
|
+
* @returns TanStack Query mutation result
|
|
131
|
+
*/
|
|
132
|
+
export declare function useDeleteUserPin(options?: Omit<UseMutationOptions<void, Error, string>, 'mutationFn'>): UseMutationResult<void, Error, string>;
|
|
@@ -0,0 +1,158 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* User Pins Mutation Hooks
|
|
4
|
+
*
|
|
5
|
+
* TanStack Query mutation hooks for user pin operations.
|
|
6
|
+
* These hooks handle creating and deleting user pins
|
|
7
|
+
* ("I'm here now" / "I'll be there").
|
|
8
|
+
*
|
|
9
|
+
* @module api/mutations/user-pins
|
|
10
|
+
*/
|
|
11
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
12
|
+
exports.useCreateUserPin = useCreateUserPin;
|
|
13
|
+
exports.useDeleteUserPin = useDeleteUserPin;
|
|
14
|
+
const react_query_1 = require("@tanstack/react-query");
|
|
15
|
+
const client_1 = require("../client");
|
|
16
|
+
const user_pins_1 = require("../queries/user-pins");
|
|
17
|
+
// ============================================================================
|
|
18
|
+
// MUTATION HOOKS
|
|
19
|
+
// ============================================================================
|
|
20
|
+
/**
|
|
21
|
+
* Create a new user pin
|
|
22
|
+
*
|
|
23
|
+
* @description
|
|
24
|
+
* Creates a new pin showing where the user is or will be.
|
|
25
|
+
* Any existing active pin is automatically deactivated.
|
|
26
|
+
*
|
|
27
|
+
* Two types of pins:
|
|
28
|
+
* - HERE_NOW: User is currently at this location (default 1 hour duration)
|
|
29
|
+
* - SCHEDULED: User will be at this location at a specific time (default 2 hour duration)
|
|
30
|
+
*
|
|
31
|
+
* @endpoint POST /api/v1/user-pins
|
|
32
|
+
*
|
|
33
|
+
* @example
|
|
34
|
+
* Drop a "I'm here now" pin:
|
|
35
|
+
* ```tsx
|
|
36
|
+
* import { useCreateUserPin } from '@growsober/sdk';
|
|
37
|
+
*
|
|
38
|
+
* function DropPinButton({ location }) {
|
|
39
|
+
* const createPin = useCreateUserPin();
|
|
40
|
+
*
|
|
41
|
+
* const handleDropPin = () => {
|
|
42
|
+
* createPin.mutate({
|
|
43
|
+
* type: 'HERE_NOW',
|
|
44
|
+
* latitude: location.lat,
|
|
45
|
+
* longitude: location.lng,
|
|
46
|
+
* locationName: 'Fabrica Coffee',
|
|
47
|
+
* activity: 'Working remotely',
|
|
48
|
+
* duration: 120, // 2 hours
|
|
49
|
+
* });
|
|
50
|
+
* };
|
|
51
|
+
*
|
|
52
|
+
* return (
|
|
53
|
+
* <Button onPress={handleDropPin} loading={createPin.isPending}>
|
|
54
|
+
* Drop Pin
|
|
55
|
+
* </Button>
|
|
56
|
+
* );
|
|
57
|
+
* }
|
|
58
|
+
* ```
|
|
59
|
+
*
|
|
60
|
+
* @example
|
|
61
|
+
* Schedule a pin for later:
|
|
62
|
+
* ```tsx
|
|
63
|
+
* import { useCreateUserPin } from '@growsober/sdk';
|
|
64
|
+
*
|
|
65
|
+
* function SchedulePinForm() {
|
|
66
|
+
* const createPin = useCreateUserPin({
|
|
67
|
+
* onSuccess: () => {
|
|
68
|
+
* toast.show('Pin scheduled!');
|
|
69
|
+
* },
|
|
70
|
+
* });
|
|
71
|
+
*
|
|
72
|
+
* const handleSchedule = (data) => {
|
|
73
|
+
* createPin.mutate({
|
|
74
|
+
* type: 'SCHEDULED',
|
|
75
|
+
* latitude: data.venue.lat,
|
|
76
|
+
* longitude: data.venue.lng,
|
|
77
|
+
* locationName: data.venue.name,
|
|
78
|
+
* venueId: data.venue.id,
|
|
79
|
+
* activity: 'Coffee meetup',
|
|
80
|
+
* scheduledTime: data.time.toISOString(),
|
|
81
|
+
* duration: 90,
|
|
82
|
+
* });
|
|
83
|
+
* };
|
|
84
|
+
*
|
|
85
|
+
* return <ScheduleForm onSubmit={handleSchedule} />;
|
|
86
|
+
* }
|
|
87
|
+
* ```
|
|
88
|
+
*
|
|
89
|
+
* @param options - TanStack Query mutation options
|
|
90
|
+
* @returns TanStack Query mutation result
|
|
91
|
+
*/
|
|
92
|
+
function useCreateUserPin(options) {
|
|
93
|
+
const queryClient = (0, react_query_1.useQueryClient)();
|
|
94
|
+
return (0, react_query_1.useMutation)({
|
|
95
|
+
mutationFn: async (data) => {
|
|
96
|
+
const client = (0, client_1.getApiClient)();
|
|
97
|
+
const response = await client.post('/api/v1/user-pins', data);
|
|
98
|
+
return response.data;
|
|
99
|
+
},
|
|
100
|
+
onSuccess: () => {
|
|
101
|
+
// Invalidate pin queries to reflect new pin
|
|
102
|
+
queryClient.invalidateQueries({ queryKey: user_pins_1.userPinKeys.all });
|
|
103
|
+
},
|
|
104
|
+
...options,
|
|
105
|
+
});
|
|
106
|
+
}
|
|
107
|
+
/**
|
|
108
|
+
* Delete a user pin
|
|
109
|
+
*
|
|
110
|
+
* @description
|
|
111
|
+
* Deactivates a user pin. Users can only delete their own pins.
|
|
112
|
+
*
|
|
113
|
+
* @endpoint DELETE /api/v1/user-pins/:id
|
|
114
|
+
*
|
|
115
|
+
* @example
|
|
116
|
+
* ```tsx
|
|
117
|
+
* import { useDeleteUserPin } from '@growsober/sdk';
|
|
118
|
+
*
|
|
119
|
+
* function MyPinCard({ pin }) {
|
|
120
|
+
* const deletePin = useDeleteUserPin({
|
|
121
|
+
* onSuccess: () => {
|
|
122
|
+
* toast.show('Pin removed');
|
|
123
|
+
* },
|
|
124
|
+
* });
|
|
125
|
+
*
|
|
126
|
+
* return (
|
|
127
|
+
* <Card>
|
|
128
|
+
* <Text>{pin.activity}</Text>
|
|
129
|
+
* <Text>At {pin.locationName}</Text>
|
|
130
|
+
* <Button
|
|
131
|
+
* onPress={() => deletePin.mutate(pin.id)}
|
|
132
|
+
* loading={deletePin.isPending}
|
|
133
|
+
* >
|
|
134
|
+
* Remove Pin
|
|
135
|
+
* </Button>
|
|
136
|
+
* </Card>
|
|
137
|
+
* );
|
|
138
|
+
* }
|
|
139
|
+
* ```
|
|
140
|
+
*
|
|
141
|
+
* @param options - TanStack Query mutation options
|
|
142
|
+
* @returns TanStack Query mutation result
|
|
143
|
+
*/
|
|
144
|
+
function useDeleteUserPin(options) {
|
|
145
|
+
const queryClient = (0, react_query_1.useQueryClient)();
|
|
146
|
+
return (0, react_query_1.useMutation)({
|
|
147
|
+
mutationFn: async (pinId) => {
|
|
148
|
+
const client = (0, client_1.getApiClient)();
|
|
149
|
+
await client.delete(`/api/v1/user-pins/${pinId}`);
|
|
150
|
+
},
|
|
151
|
+
onSuccess: () => {
|
|
152
|
+
// Invalidate pin queries to reflect deletion
|
|
153
|
+
queryClient.invalidateQueries({ queryKey: user_pins_1.userPinKeys.all });
|
|
154
|
+
},
|
|
155
|
+
...options,
|
|
156
|
+
});
|
|
157
|
+
}
|
|
158
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"user-pins.js","sourceRoot":"","sources":["../../../src/api/mutations/user-pins.ts"],"names":[],"mappings":";AAAA;;;;;;;;GAQG;;AAsGH,4CAoBC;AAuCD,4CAgBC;AA/KD,uDAK+B;AAC/B,sCAAyC;AACzC,oDAA6E;AAiB7E,+EAA+E;AAC/E,iBAAiB;AACjB,+EAA+E;AAE/E;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAuEG;AACH,SAAgB,gBAAgB,CAC9B,OAGC;IAED,MAAM,WAAW,GAAG,IAAA,4BAAc,GAAE,CAAC;IAErC,OAAO,IAAA,yBAAW,EAAC;QACjB,UAAU,EAAE,KAAK,EAAE,IAAsB,EAA4B,EAAE;YACrE,MAAM,MAAM,GAAG,IAAA,qBAAY,GAAE,CAAC;YAC9B,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,IAAI,CAAkB,mBAAmB,EAAE,IAAI,CAAC,CAAC;YAC/E,OAAO,QAAQ,CAAC,IAAI,CAAC;QACvB,CAAC;QACD,SAAS,EAAE,GAAG,EAAE;YACd,4CAA4C;YAC5C,WAAW,CAAC,iBAAiB,CAAC,EAAE,QAAQ,EAAE,uBAAW,CAAC,GAAG,EAAE,CAAC,CAAC;QAC/D,CAAC;QACD,GAAG,OAAO;KACX,CAAC,CAAC;AACL,CAAC;AAED;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAoCG;AACH,SAAgB,gBAAgB,CAC9B,OAAqE;IAErE,MAAM,WAAW,GAAG,IAAA,4BAAc,GAAE,CAAC;IAErC,OAAO,IAAA,yBAAW,EAAC;QACjB,UAAU,EAAE,KAAK,EAAE,KAAa,EAAiB,EAAE;YACjD,MAAM,MAAM,GAAG,IAAA,qBAAY,GAAE,CAAC;YAC9B,MAAM,MAAM,CAAC,MAAM,CAAC,qBAAqB,KAAK,EAAE,CAAC,CAAC;QACpD,CAAC;QACD,SAAS,EAAE,GAAG,EAAE;YACd,6CAA6C;YAC7C,WAAW,CAAC,iBAAiB,CAAC,EAAE,QAAQ,EAAE,uBAAW,CAAC,GAAG,EAAE,CAAC,CAAC;QAC/D,CAAC;QACD,GAAG,OAAO;KACX,CAAC,CAAC;AACL,CAAC","sourcesContent":["/**\n * User Pins Mutation Hooks\n *\n * TanStack Query mutation hooks for user pin operations.\n * These hooks handle creating and deleting user pins\n * (\"I'm here now\" / \"I'll be there\").\n *\n * @module api/mutations/user-pins\n */\n\nimport {\n  useMutation,\n  UseMutationOptions,\n  UseMutationResult,\n  useQueryClient,\n} from '@tanstack/react-query';\nimport { getApiClient } from '../client';\nimport { userPinKeys, UserPinResponse, PinType } from '../queries/user-pins';\n\n// ============================================================================\n// REQUEST TYPES\n// ============================================================================\n\nexport interface CreatePinRequest {\n  type: PinType;\n  latitude: number;\n  longitude: number;\n  locationName?: string;\n  activity: string;\n  venueId?: string;\n  scheduledTime?: string; // ISO date string, required for SCHEDULED type\n  duration?: number; // minutes, default 60 for HERE_NOW, 120 for SCHEDULED\n}\n\n// ============================================================================\n// MUTATION HOOKS\n// ============================================================================\n\n/**\n * Create a new user pin\n *\n * @description\n * Creates a new pin showing where the user is or will be.\n * Any existing active pin is automatically deactivated.\n *\n * Two types of pins:\n * - HERE_NOW: User is currently at this location (default 1 hour duration)\n * - SCHEDULED: User will be at this location at a specific time (default 2 hour duration)\n *\n * @endpoint POST /api/v1/user-pins\n *\n * @example\n * Drop a \"I'm here now\" pin:\n * ```tsx\n * import { useCreateUserPin } from '@growsober/sdk';\n *\n * function DropPinButton({ location }) {\n *   const createPin = useCreateUserPin();\n *\n *   const handleDropPin = () => {\n *     createPin.mutate({\n *       type: 'HERE_NOW',\n *       latitude: location.lat,\n *       longitude: location.lng,\n *       locationName: 'Fabrica Coffee',\n *       activity: 'Working remotely',\n *       duration: 120, // 2 hours\n *     });\n *   };\n *\n *   return (\n *     <Button onPress={handleDropPin} loading={createPin.isPending}>\n *       Drop Pin\n *     </Button>\n *   );\n * }\n * ```\n *\n * @example\n * Schedule a pin for later:\n * ```tsx\n * import { useCreateUserPin } from '@growsober/sdk';\n *\n * function SchedulePinForm() {\n *   const createPin = useCreateUserPin({\n *     onSuccess: () => {\n *       toast.show('Pin scheduled!');\n *     },\n *   });\n *\n *   const handleSchedule = (data) => {\n *     createPin.mutate({\n *       type: 'SCHEDULED',\n *       latitude: data.venue.lat,\n *       longitude: data.venue.lng,\n *       locationName: data.venue.name,\n *       venueId: data.venue.id,\n *       activity: 'Coffee meetup',\n *       scheduledTime: data.time.toISOString(),\n *       duration: 90,\n *     });\n *   };\n *\n *   return <ScheduleForm onSubmit={handleSchedule} />;\n * }\n * ```\n *\n * @param options - TanStack Query mutation options\n * @returns TanStack Query mutation result\n */\nexport function useCreateUserPin(\n  options?: Omit<\n    UseMutationOptions<UserPinResponse, Error, CreatePinRequest>,\n    'mutationFn'\n  >\n): UseMutationResult<UserPinResponse, Error, CreatePinRequest> {\n  const queryClient = useQueryClient();\n\n  return useMutation({\n    mutationFn: async (data: CreatePinRequest): Promise<UserPinResponse> => {\n      const client = getApiClient();\n      const response = await client.post<UserPinResponse>('/api/v1/user-pins', data);\n      return response.data;\n    },\n    onSuccess: () => {\n      // Invalidate pin queries to reflect new pin\n      queryClient.invalidateQueries({ queryKey: userPinKeys.all });\n    },\n    ...options,\n  });\n}\n\n/**\n * Delete a user pin\n *\n * @description\n * Deactivates a user pin. Users can only delete their own pins.\n *\n * @endpoint DELETE /api/v1/user-pins/:id\n *\n * @example\n * ```tsx\n * import { useDeleteUserPin } from '@growsober/sdk';\n *\n * function MyPinCard({ pin }) {\n *   const deletePin = useDeleteUserPin({\n *     onSuccess: () => {\n *       toast.show('Pin removed');\n *     },\n *   });\n *\n *   return (\n *     <Card>\n *       <Text>{pin.activity}</Text>\n *       <Text>At {pin.locationName}</Text>\n *       <Button\n *         onPress={() => deletePin.mutate(pin.id)}\n *         loading={deletePin.isPending}\n *       >\n *         Remove Pin\n *       </Button>\n *     </Card>\n *   );\n * }\n * ```\n *\n * @param options - TanStack Query mutation options\n * @returns TanStack Query mutation result\n */\nexport function useDeleteUserPin(\n  options?: Omit<UseMutationOptions<void, Error, string>, 'mutationFn'>\n): UseMutationResult<void, Error, string> {\n  const queryClient = useQueryClient();\n\n  return useMutation({\n    mutationFn: async (pinId: string): Promise<void> => {\n      const client = getApiClient();\n      await client.delete(`/api/v1/user-pins/${pinId}`);\n    },\n    onSuccess: () => {\n      // Invalidate pin queries to reflect deletion\n      queryClient.invalidateQueries({ queryKey: userPinKeys.all });\n    },\n    ...options,\n  });\n}\n"]}
|
|
@@ -38,4 +38,5 @@ __exportStar(require("./ambassadors"), exports);
|
|
|
38
38
|
__exportStar(require("./grow90"), exports);
|
|
39
39
|
__exportStar(require("./matching"), exports);
|
|
40
40
|
__exportStar(require("./event-chat"), exports);
|
|
41
|
-
|
|
41
|
+
__exportStar(require("./user-pins"), exports);
|
|
42
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvYXBpL3F1ZXJpZXMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOzs7O0dBSUc7Ozs7Ozs7Ozs7Ozs7Ozs7QUFFSCwwQ0FBd0I7QUFDeEIseUNBQXVCO0FBQ3ZCLDZDQUEyQjtBQUMzQiwrQ0FBNkI7QUFDN0IsMkNBQXlCO0FBQ3pCLDJDQUF5QjtBQUN6Qix5Q0FBdUI7QUFDdkIsNENBQTBCO0FBQzFCLHdDQUFzQjtBQUN0QixrREFBZ0M7QUFDaEMsMkNBQXlCO0FBQ3pCLGtEQUFnQztBQUNoQyw0Q0FBMEI7QUFDMUIsMENBQXdCO0FBQ3hCLHlDQUF1QjtBQUN2QixnREFBOEI7QUFDOUIsMkNBQXlCO0FBQ3pCLDZDQUEyQjtBQUMzQiwrQ0FBNkI7QUFDN0IsOENBQTRCIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBRdWVyeSBIb29rc1xuICpcbiAqIFJlLWV4cG9ydHMgYWxsIHF1ZXJ5IGhvb2tzIGZvciBBUEkgZW5kcG9pbnRzLlxuICovXG5cbmV4cG9ydCAqIGZyb20gJy4vYWRtaW4nO1xuZXhwb3J0ICogZnJvbSAnLi9hdXRoJztcbmV4cG9ydCAqIGZyb20gJy4vYm9va2luZ3MnO1xuZXhwb3J0ICogZnJvbSAnLi9idXNpbmVzc2VzJztcbmV4cG9ydCAqIGZyb20gJy4vY2l0aWVzJztcbmV4cG9ydCAqIGZyb20gJy4vZXZlbnRzJztcbmV4cG9ydCAqIGZyb20gJy4vaHVicyc7XG5leHBvcnQgKiBmcm9tICcuL2xpYnJhcnknO1xuZXhwb3J0ICogZnJvbSAnLi9tYXAnO1xuZXhwb3J0ICogZnJvbSAnLi9ub3RpZmljYXRpb25zJztcbmV4cG9ydCAqIGZyb20gJy4vb2ZmZXJzJztcbmV4cG9ydCAqIGZyb20gJy4vc3Vic2NyaXB0aW9ucyc7XG5leHBvcnQgKiBmcm9tICcuL3N1cHBvcnQnO1xuZXhwb3J0ICogZnJvbSAnLi91c2Vycyc7XG5leHBvcnQgKiBmcm9tICcuL2phY2snO1xuZXhwb3J0ICogZnJvbSAnLi9hbWJhc3NhZG9ycyc7XG5leHBvcnQgKiBmcm9tICcuL2dyb3c5MCc7XG5leHBvcnQgKiBmcm9tICcuL21hdGNoaW5nJztcbmV4cG9ydCAqIGZyb20gJy4vZXZlbnQtY2hhdCc7XG5leHBvcnQgKiBmcm9tICcuL3VzZXItcGlucyc7XG4iXX0=
|
|
@@ -0,0 +1,134 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* User Pins Query Hooks
|
|
3
|
+
*
|
|
4
|
+
* TanStack Query hooks for user pin read operations.
|
|
5
|
+
* These hooks handle fetching user pins for the map feature
|
|
6
|
+
* ("I'm here now" / "I'll be there").
|
|
7
|
+
*
|
|
8
|
+
* @module api/queries/user-pins
|
|
9
|
+
*/
|
|
10
|
+
import { UseQueryOptions, UseQueryResult } from '@tanstack/react-query';
|
|
11
|
+
export type PinType = 'HERE_NOW' | 'SCHEDULED';
|
|
12
|
+
export interface UserPinResponse {
|
|
13
|
+
id: string;
|
|
14
|
+
userId: string;
|
|
15
|
+
type: PinType;
|
|
16
|
+
latitude: number;
|
|
17
|
+
longitude: number;
|
|
18
|
+
locationName: string | null;
|
|
19
|
+
activity: string;
|
|
20
|
+
venueId: string | null;
|
|
21
|
+
scheduledTime: string | null;
|
|
22
|
+
expiresAt: string;
|
|
23
|
+
duration: number | null;
|
|
24
|
+
isActive: boolean;
|
|
25
|
+
createdAt: string;
|
|
26
|
+
user?: {
|
|
27
|
+
id: string;
|
|
28
|
+
name: string | null;
|
|
29
|
+
profileImage: string | null;
|
|
30
|
+
};
|
|
31
|
+
venue?: {
|
|
32
|
+
id: string;
|
|
33
|
+
name: string;
|
|
34
|
+
slug: string | null;
|
|
35
|
+
};
|
|
36
|
+
}
|
|
37
|
+
export interface NearbyPinsResponse {
|
|
38
|
+
pins: UserPinResponse[];
|
|
39
|
+
total: number;
|
|
40
|
+
}
|
|
41
|
+
/**
|
|
42
|
+
* Query key factory for user pin queries
|
|
43
|
+
*/
|
|
44
|
+
export declare const userPinKeys: {
|
|
45
|
+
all: readonly ["user-pins"];
|
|
46
|
+
nearby: (params: NearbyPinsParams) => readonly ["user-pins", "nearby", NearbyPinsParams];
|
|
47
|
+
me: () => readonly ["user-pins", "me"];
|
|
48
|
+
};
|
|
49
|
+
export interface NearbyPinsParams {
|
|
50
|
+
lat: number;
|
|
51
|
+
lng: number;
|
|
52
|
+
radius?: number;
|
|
53
|
+
}
|
|
54
|
+
/**
|
|
55
|
+
* Get nearby user pins
|
|
56
|
+
*
|
|
57
|
+
* @description
|
|
58
|
+
* Retrieves active user pins within a radius of a location.
|
|
59
|
+
* Shows where other GrowSober members are or will be.
|
|
60
|
+
*
|
|
61
|
+
* @endpoint GET /api/v1/user-pins/nearby
|
|
62
|
+
*
|
|
63
|
+
* @example
|
|
64
|
+
* ```tsx
|
|
65
|
+
* import { useNearbyPins } from '@growsober/sdk';
|
|
66
|
+
*
|
|
67
|
+
* function MapWithPins() {
|
|
68
|
+
* const { data, isLoading } = useNearbyPins({
|
|
69
|
+
* lat: 38.7097,
|
|
70
|
+
* lng: -9.1367,
|
|
71
|
+
* radius: 5, // 5km radius
|
|
72
|
+
* });
|
|
73
|
+
*
|
|
74
|
+
* return (
|
|
75
|
+
* <Map>
|
|
76
|
+
* {data?.pins.map(pin => (
|
|
77
|
+
* <Marker
|
|
78
|
+
* key={pin.id}
|
|
79
|
+
* position={[pin.latitude, pin.longitude]}
|
|
80
|
+
* icon={pin.type === 'HERE_NOW' ? 'here' : 'scheduled'}
|
|
81
|
+
* >
|
|
82
|
+
* <Popup>
|
|
83
|
+
* <p>{pin.user?.name}</p>
|
|
84
|
+
* <p>{pin.activity}</p>
|
|
85
|
+
* </Popup>
|
|
86
|
+
* </Marker>
|
|
87
|
+
* ))}
|
|
88
|
+
* </Map>
|
|
89
|
+
* );
|
|
90
|
+
* }
|
|
91
|
+
* ```
|
|
92
|
+
*
|
|
93
|
+
* @param params - Location and radius parameters
|
|
94
|
+
* @param options - TanStack Query options
|
|
95
|
+
* @returns TanStack Query result with nearby pins
|
|
96
|
+
*/
|
|
97
|
+
export declare function useNearbyPins(params: NearbyPinsParams, options?: Omit<UseQueryOptions<NearbyPinsResponse>, 'queryKey' | 'queryFn'>): UseQueryResult<NearbyPinsResponse>;
|
|
98
|
+
/**
|
|
99
|
+
* Get current user's active pins
|
|
100
|
+
*
|
|
101
|
+
* @description
|
|
102
|
+
* Retrieves the authenticated user's active pins.
|
|
103
|
+
* Users can have at most one active pin at a time.
|
|
104
|
+
*
|
|
105
|
+
* @endpoint GET /api/v1/user-pins/me
|
|
106
|
+
*
|
|
107
|
+
* @example
|
|
108
|
+
* ```tsx
|
|
109
|
+
* import { useMyPins } from '@growsober/sdk';
|
|
110
|
+
*
|
|
111
|
+
* function MyPinStatus() {
|
|
112
|
+
* const { data: pins, isLoading } = useMyPins();
|
|
113
|
+
*
|
|
114
|
+
* if (isLoading) return <Spinner />;
|
|
115
|
+
*
|
|
116
|
+
* if (!pins?.length) {
|
|
117
|
+
* return <Text>You haven't dropped a pin yet</Text>;
|
|
118
|
+
* }
|
|
119
|
+
*
|
|
120
|
+
* const pin = pins[0];
|
|
121
|
+
* return (
|
|
122
|
+
* <View>
|
|
123
|
+
* <Text>You're at {pin.locationName}</Text>
|
|
124
|
+
* <Text>Activity: {pin.activity}</Text>
|
|
125
|
+
* <Text>Expires: {formatTime(pin.expiresAt)}</Text>
|
|
126
|
+
* </View>
|
|
127
|
+
* );
|
|
128
|
+
* }
|
|
129
|
+
* ```
|
|
130
|
+
*
|
|
131
|
+
* @param options - TanStack Query options
|
|
132
|
+
* @returns TanStack Query result with user's active pins
|
|
133
|
+
*/
|
|
134
|
+
export declare function useMyPins(options?: Omit<UseQueryOptions<UserPinResponse[]>, 'queryKey' | 'queryFn'>): UseQueryResult<UserPinResponse[]>;
|
|
@@ -0,0 +1,141 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* User Pins Query Hooks
|
|
4
|
+
*
|
|
5
|
+
* TanStack Query hooks for user pin read operations.
|
|
6
|
+
* These hooks handle fetching user pins for the map feature
|
|
7
|
+
* ("I'm here now" / "I'll be there").
|
|
8
|
+
*
|
|
9
|
+
* @module api/queries/user-pins
|
|
10
|
+
*/
|
|
11
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
12
|
+
exports.userPinKeys = void 0;
|
|
13
|
+
exports.useNearbyPins = useNearbyPins;
|
|
14
|
+
exports.useMyPins = useMyPins;
|
|
15
|
+
const react_query_1 = require("@tanstack/react-query");
|
|
16
|
+
const client_1 = require("../client");
|
|
17
|
+
// ============================================================================
|
|
18
|
+
// QUERY KEYS
|
|
19
|
+
// ============================================================================
|
|
20
|
+
/**
|
|
21
|
+
* Query key factory for user pin queries
|
|
22
|
+
*/
|
|
23
|
+
exports.userPinKeys = {
|
|
24
|
+
all: ['user-pins'],
|
|
25
|
+
nearby: (params) => [...exports.userPinKeys.all, 'nearby', params],
|
|
26
|
+
me: () => [...exports.userPinKeys.all, 'me'],
|
|
27
|
+
};
|
|
28
|
+
// ============================================================================
|
|
29
|
+
// QUERY HOOKS
|
|
30
|
+
// ============================================================================
|
|
31
|
+
/**
|
|
32
|
+
* Get nearby user pins
|
|
33
|
+
*
|
|
34
|
+
* @description
|
|
35
|
+
* Retrieves active user pins within a radius of a location.
|
|
36
|
+
* Shows where other GrowSober members are or will be.
|
|
37
|
+
*
|
|
38
|
+
* @endpoint GET /api/v1/user-pins/nearby
|
|
39
|
+
*
|
|
40
|
+
* @example
|
|
41
|
+
* ```tsx
|
|
42
|
+
* import { useNearbyPins } from '@growsober/sdk';
|
|
43
|
+
*
|
|
44
|
+
* function MapWithPins() {
|
|
45
|
+
* const { data, isLoading } = useNearbyPins({
|
|
46
|
+
* lat: 38.7097,
|
|
47
|
+
* lng: -9.1367,
|
|
48
|
+
* radius: 5, // 5km radius
|
|
49
|
+
* });
|
|
50
|
+
*
|
|
51
|
+
* return (
|
|
52
|
+
* <Map>
|
|
53
|
+
* {data?.pins.map(pin => (
|
|
54
|
+
* <Marker
|
|
55
|
+
* key={pin.id}
|
|
56
|
+
* position={[pin.latitude, pin.longitude]}
|
|
57
|
+
* icon={pin.type === 'HERE_NOW' ? 'here' : 'scheduled'}
|
|
58
|
+
* >
|
|
59
|
+
* <Popup>
|
|
60
|
+
* <p>{pin.user?.name}</p>
|
|
61
|
+
* <p>{pin.activity}</p>
|
|
62
|
+
* </Popup>
|
|
63
|
+
* </Marker>
|
|
64
|
+
* ))}
|
|
65
|
+
* </Map>
|
|
66
|
+
* );
|
|
67
|
+
* }
|
|
68
|
+
* ```
|
|
69
|
+
*
|
|
70
|
+
* @param params - Location and radius parameters
|
|
71
|
+
* @param options - TanStack Query options
|
|
72
|
+
* @returns TanStack Query result with nearby pins
|
|
73
|
+
*/
|
|
74
|
+
function useNearbyPins(params, options) {
|
|
75
|
+
return (0, react_query_1.useQuery)({
|
|
76
|
+
queryKey: exports.userPinKeys.nearby(params),
|
|
77
|
+
queryFn: async () => {
|
|
78
|
+
const client = (0, client_1.getApiClient)();
|
|
79
|
+
const response = await client.get('/api/v1/user-pins/nearby', {
|
|
80
|
+
params: {
|
|
81
|
+
lat: params.lat,
|
|
82
|
+
lng: params.lng,
|
|
83
|
+
radius: params.radius,
|
|
84
|
+
},
|
|
85
|
+
});
|
|
86
|
+
return response.data;
|
|
87
|
+
},
|
|
88
|
+
enabled: params.lat !== undefined && params.lng !== undefined,
|
|
89
|
+
staleTime: 30 * 1000, // 30 seconds - pins change frequently
|
|
90
|
+
...options,
|
|
91
|
+
});
|
|
92
|
+
}
|
|
93
|
+
/**
|
|
94
|
+
* Get current user's active pins
|
|
95
|
+
*
|
|
96
|
+
* @description
|
|
97
|
+
* Retrieves the authenticated user's active pins.
|
|
98
|
+
* Users can have at most one active pin at a time.
|
|
99
|
+
*
|
|
100
|
+
* @endpoint GET /api/v1/user-pins/me
|
|
101
|
+
*
|
|
102
|
+
* @example
|
|
103
|
+
* ```tsx
|
|
104
|
+
* import { useMyPins } from '@growsober/sdk';
|
|
105
|
+
*
|
|
106
|
+
* function MyPinStatus() {
|
|
107
|
+
* const { data: pins, isLoading } = useMyPins();
|
|
108
|
+
*
|
|
109
|
+
* if (isLoading) return <Spinner />;
|
|
110
|
+
*
|
|
111
|
+
* if (!pins?.length) {
|
|
112
|
+
* return <Text>You haven't dropped a pin yet</Text>;
|
|
113
|
+
* }
|
|
114
|
+
*
|
|
115
|
+
* const pin = pins[0];
|
|
116
|
+
* return (
|
|
117
|
+
* <View>
|
|
118
|
+
* <Text>You're at {pin.locationName}</Text>
|
|
119
|
+
* <Text>Activity: {pin.activity}</Text>
|
|
120
|
+
* <Text>Expires: {formatTime(pin.expiresAt)}</Text>
|
|
121
|
+
* </View>
|
|
122
|
+
* );
|
|
123
|
+
* }
|
|
124
|
+
* ```
|
|
125
|
+
*
|
|
126
|
+
* @param options - TanStack Query options
|
|
127
|
+
* @returns TanStack Query result with user's active pins
|
|
128
|
+
*/
|
|
129
|
+
function useMyPins(options) {
|
|
130
|
+
return (0, react_query_1.useQuery)({
|
|
131
|
+
queryKey: exports.userPinKeys.me(),
|
|
132
|
+
queryFn: async () => {
|
|
133
|
+
const client = (0, client_1.getApiClient)();
|
|
134
|
+
const response = await client.get('/api/v1/user-pins/me');
|
|
135
|
+
return response.data;
|
|
136
|
+
},
|
|
137
|
+
staleTime: 60 * 1000, // 1 minute
|
|
138
|
+
...options,
|
|
139
|
+
});
|
|
140
|
+
}
|
|
141
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"user-pins.js","sourceRoot":"","sources":["../../../src/api/queries/user-pins.ts"],"names":[],"mappings":";AAAA;;;;;;;;GAQG;;;AAgHH,sCAqBC;AAsCD,8BAaC;AAtLD,uDAAkF;AAClF,sCAAyC;AAuCzC,+EAA+E;AAC/E,aAAa;AACb,+EAA+E;AAE/E;;GAEG;AACU,QAAA,WAAW,GAAG;IACzB,GAAG,EAAE,CAAC,WAAW,CAAU;IAC3B,MAAM,EAAE,CAAC,MAAwB,EAAE,EAAE,CAAC,CAAC,GAAG,mBAAW,CAAC,GAAG,EAAE,QAAQ,EAAE,MAAM,CAAU;IACrF,EAAE,EAAE,GAAG,EAAE,CAAC,CAAC,GAAG,mBAAW,CAAC,GAAG,EAAE,IAAI,CAAU;CAC9C,CAAC;AAYF,+EAA+E;AAC/E,cAAc;AACd,+EAA+E;AAE/E;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA0CG;AACH,SAAgB,aAAa,CAC3B,MAAwB,EACxB,OAA2E;IAE3E,OAAO,IAAA,sBAAQ,EAAC;QACd,QAAQ,EAAE,mBAAW,CAAC,MAAM,CAAC,MAAM,CAAC;QACpC,OAAO,EAAE,KAAK,IAAiC,EAAE;YAC/C,MAAM,MAAM,GAAG,IAAA,qBAAY,GAAE,CAAC;YAC9B,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,GAAG,CAAqB,0BAA0B,EAAE;gBAChF,MAAM,EAAE;oBACN,GAAG,EAAE,MAAM,CAAC,GAAG;oBACf,GAAG,EAAE,MAAM,CAAC,GAAG;oBACf,MAAM,EAAE,MAAM,CAAC,MAAM;iBACtB;aACF,CAAC,CAAC;YACH,OAAO,QAAQ,CAAC,IAAI,CAAC;QACvB,CAAC;QACD,OAAO,EAAE,MAAM,CAAC,GAAG,KAAK,SAAS,IAAI,MAAM,CAAC,GAAG,KAAK,SAAS;QAC7D,SAAS,EAAE,EAAE,GAAG,IAAI,EAAE,sCAAsC;QAC5D,GAAG,OAAO;KACX,CAAC,CAAC;AACL,CAAC;AAED;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAmCG;AACH,SAAgB,SAAS,CACvB,OAA0E;IAE1E,OAAO,IAAA,sBAAQ,EAAC;QACd,QAAQ,EAAE,mBAAW,CAAC,EAAE,EAAE;QAC1B,OAAO,EAAE,KAAK,IAAgC,EAAE;YAC9C,MAAM,MAAM,GAAG,IAAA,qBAAY,GAAE,CAAC;YAC9B,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,GAAG,CAAoB,sBAAsB,CAAC,CAAC;YAC7E,OAAO,QAAQ,CAAC,IAAI,CAAC;QACvB,CAAC;QACD,SAAS,EAAE,EAAE,GAAG,IAAI,EAAE,WAAW;QACjC,GAAG,OAAO;KACX,CAAC,CAAC;AACL,CAAC","sourcesContent":["/**\n * User Pins Query Hooks\n *\n * TanStack Query hooks for user pin read operations.\n * These hooks handle fetching user pins for the map feature\n * (\"I'm here now\" / \"I'll be there\").\n *\n * @module api/queries/user-pins\n */\n\nimport { useQuery, UseQueryOptions, UseQueryResult } from '@tanstack/react-query';\nimport { getApiClient } from '../client';\n\n// ============================================================================\n// TYPES\n// ============================================================================\n\nexport type PinType = 'HERE_NOW' | 'SCHEDULED';\n\nexport interface UserPinResponse {\n  id: string;\n  userId: string;\n  type: PinType;\n  latitude: number;\n  longitude: number;\n  locationName: string | null;\n  activity: string;\n  venueId: string | null;\n  scheduledTime: string | null;\n  expiresAt: string;\n  duration: number | null;\n  isActive: boolean;\n  createdAt: string;\n  user?: {\n    id: string;\n    name: string | null;\n    profileImage: string | null;\n  };\n  venue?: {\n    id: string;\n    name: string;\n    slug: string | null;\n  };\n}\n\nexport interface NearbyPinsResponse {\n  pins: UserPinResponse[];\n  total: number;\n}\n\n// ============================================================================\n// QUERY KEYS\n// ============================================================================\n\n/**\n * Query key factory for user pin queries\n */\nexport const userPinKeys = {\n  all: ['user-pins'] as const,\n  nearby: (params: NearbyPinsParams) => [...userPinKeys.all, 'nearby', params] as const,\n  me: () => [...userPinKeys.all, 'me'] as const,\n};\n\n// ============================================================================\n// PARAM TYPES\n// ============================================================================\n\nexport interface NearbyPinsParams {\n  lat: number;\n  lng: number;\n  radius?: number; // km, default 5\n}\n\n// ============================================================================\n// QUERY HOOKS\n// ============================================================================\n\n/**\n * Get nearby user pins\n *\n * @description\n * Retrieves active user pins within a radius of a location.\n * Shows where other GrowSober members are or will be.\n *\n * @endpoint GET /api/v1/user-pins/nearby\n *\n * @example\n * ```tsx\n * import { useNearbyPins } from '@growsober/sdk';\n *\n * function MapWithPins() {\n *   const { data, isLoading } = useNearbyPins({\n *     lat: 38.7097,\n *     lng: -9.1367,\n *     radius: 5, // 5km radius\n *   });\n *\n *   return (\n *     <Map>\n *       {data?.pins.map(pin => (\n *         <Marker\n *           key={pin.id}\n *           position={[pin.latitude, pin.longitude]}\n *           icon={pin.type === 'HERE_NOW' ? 'here' : 'scheduled'}\n *         >\n *           <Popup>\n *             <p>{pin.user?.name}</p>\n *             <p>{pin.activity}</p>\n *           </Popup>\n *         </Marker>\n *       ))}\n *     </Map>\n *   );\n * }\n * ```\n *\n * @param params - Location and radius parameters\n * @param options - TanStack Query options\n * @returns TanStack Query result with nearby pins\n */\nexport function useNearbyPins(\n  params: NearbyPinsParams,\n  options?: Omit<UseQueryOptions<NearbyPinsResponse>, 'queryKey' | 'queryFn'>\n): UseQueryResult<NearbyPinsResponse> {\n  return useQuery({\n    queryKey: userPinKeys.nearby(params),\n    queryFn: async (): Promise<NearbyPinsResponse> => {\n      const client = getApiClient();\n      const response = await client.get<NearbyPinsResponse>('/api/v1/user-pins/nearby', {\n        params: {\n          lat: params.lat,\n          lng: params.lng,\n          radius: params.radius,\n        },\n      });\n      return response.data;\n    },\n    enabled: params.lat !== undefined && params.lng !== undefined,\n    staleTime: 30 * 1000, // 30 seconds - pins change frequently\n    ...options,\n  });\n}\n\n/**\n * Get current user's active pins\n *\n * @description\n * Retrieves the authenticated user's active pins.\n * Users can have at most one active pin at a time.\n *\n * @endpoint GET /api/v1/user-pins/me\n *\n * @example\n * ```tsx\n * import { useMyPins } from '@growsober/sdk';\n *\n * function MyPinStatus() {\n *   const { data: pins, isLoading } = useMyPins();\n *\n *   if (isLoading) return <Spinner />;\n *\n *   if (!pins?.length) {\n *     return <Text>You haven't dropped a pin yet</Text>;\n *   }\n *\n *   const pin = pins[0];\n *   return (\n *     <View>\n *       <Text>You're at {pin.locationName}</Text>\n *       <Text>Activity: {pin.activity}</Text>\n *       <Text>Expires: {formatTime(pin.expiresAt)}</Text>\n *     </View>\n *   );\n * }\n * ```\n *\n * @param options - TanStack Query options\n * @returns TanStack Query result with user's active pins\n */\nexport function useMyPins(\n  options?: Omit<UseQueryOptions<UserPinResponse[]>, 'queryKey' | 'queryFn'>\n): UseQueryResult<UserPinResponse[]> {\n  return useQuery({\n    queryKey: userPinKeys.me(),\n    queryFn: async (): Promise<UserPinResponse[]> => {\n      const client = getApiClient();\n      const response = await client.get<UserPinResponse[]>('/api/v1/user-pins/me');\n      return response.data;\n    },\n    staleTime: 60 * 1000, // 1 minute\n    ...options,\n  });\n}\n"]}
|
package/package.json
CHANGED
|
@@ -75,7 +75,7 @@ export function useRegister(
|
|
|
75
75
|
return useMutation({
|
|
76
76
|
mutationFn: async (data: RegisterRequest): Promise<AuthResponse> => {
|
|
77
77
|
const client = getApiClient();
|
|
78
|
-
const response = await client.post<AuthResponse>('/auth/register', data);
|
|
78
|
+
const response = await client.post<AuthResponse>('/api/v1/auth/register', data);
|
|
79
79
|
return response.data;
|
|
80
80
|
},
|
|
81
81
|
onSuccess: (data, variables, context) => {
|
|
@@ -150,7 +150,7 @@ export function useLogin(
|
|
|
150
150
|
return useMutation({
|
|
151
151
|
mutationFn: async (data: LoginRequest): Promise<AuthResponse> => {
|
|
152
152
|
const client = getApiClient();
|
|
153
|
-
const response = await client.post<AuthResponse>('/auth/login', data);
|
|
153
|
+
const response = await client.post<AuthResponse>('/api/v1/auth/login', data);
|
|
154
154
|
return response.data;
|
|
155
155
|
},
|
|
156
156
|
onSuccess: (data, variables, context) => {
|
|
@@ -241,7 +241,7 @@ export function useRefreshAuthToken(
|
|
|
241
241
|
return useMutation({
|
|
242
242
|
mutationFn: async (data: RefreshTokenRequest): Promise<TokenResponse> => {
|
|
243
243
|
const client = getApiClient();
|
|
244
|
-
const response = await client.post<TokenResponse>('/auth/refresh', data);
|
|
244
|
+
const response = await client.post<TokenResponse>('/api/v1/auth/refresh', data);
|
|
245
245
|
return response.data;
|
|
246
246
|
},
|
|
247
247
|
...options,
|
|
@@ -340,7 +340,7 @@ export function useFirebaseAuth(
|
|
|
340
340
|
return useMutation({
|
|
341
341
|
mutationFn: async (data: FirebaseAuthRequest): Promise<AuthResponse> => {
|
|
342
342
|
const client = getApiClient();
|
|
343
|
-
const response = await client.post<AuthResponse>('/auth/firebase', data);
|
|
343
|
+
const response = await client.post<AuthResponse>('/api/v1/auth/firebase', data);
|
|
344
344
|
return response.data;
|
|
345
345
|
},
|
|
346
346
|
onSuccess: (data, variables, context) => {
|
|
@@ -398,7 +398,7 @@ export function useSendOtp(
|
|
|
398
398
|
return useMutation({
|
|
399
399
|
mutationFn: async (data: SendOtpRequest): Promise<OtpSentResponse> => {
|
|
400
400
|
const client = getApiClient();
|
|
401
|
-
const response = await client.post<OtpSentResponse>('/auth/phone/send-otp', data);
|
|
401
|
+
const response = await client.post<OtpSentResponse>('/api/v1/auth/phone/send-otp', data);
|
|
402
402
|
return response.data;
|
|
403
403
|
},
|
|
404
404
|
...options,
|
|
@@ -452,7 +452,7 @@ export function useVerifyOtp(
|
|
|
452
452
|
return useMutation({
|
|
453
453
|
mutationFn: async (data: VerifyOtpRequest): Promise<VerifyOtpResponse> => {
|
|
454
454
|
const client = getApiClient();
|
|
455
|
-
const response = await client.post<VerifyOtpResponse>('/auth/phone/verify-otp', data);
|
|
455
|
+
const response = await client.post<VerifyOtpResponse>('/api/v1/auth/phone/verify-otp', data);
|
|
456
456
|
return response.data;
|
|
457
457
|
},
|
|
458
458
|
onSuccess: (data, variables, context) => {
|
|
@@ -0,0 +1,186 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* User Pins Mutation Hooks
|
|
3
|
+
*
|
|
4
|
+
* TanStack Query mutation hooks for user pin operations.
|
|
5
|
+
* These hooks handle creating and deleting user pins
|
|
6
|
+
* ("I'm here now" / "I'll be there").
|
|
7
|
+
*
|
|
8
|
+
* @module api/mutations/user-pins
|
|
9
|
+
*/
|
|
10
|
+
|
|
11
|
+
import {
|
|
12
|
+
useMutation,
|
|
13
|
+
UseMutationOptions,
|
|
14
|
+
UseMutationResult,
|
|
15
|
+
useQueryClient,
|
|
16
|
+
} from '@tanstack/react-query';
|
|
17
|
+
import { getApiClient } from '../client';
|
|
18
|
+
import { userPinKeys, UserPinResponse, PinType } from '../queries/user-pins';
|
|
19
|
+
|
|
20
|
+
// ============================================================================
|
|
21
|
+
// REQUEST TYPES
|
|
22
|
+
// ============================================================================
|
|
23
|
+
|
|
24
|
+
export interface CreatePinRequest {
|
|
25
|
+
type: PinType;
|
|
26
|
+
latitude: number;
|
|
27
|
+
longitude: number;
|
|
28
|
+
locationName?: string;
|
|
29
|
+
activity: string;
|
|
30
|
+
venueId?: string;
|
|
31
|
+
scheduledTime?: string; // ISO date string, required for SCHEDULED type
|
|
32
|
+
duration?: number; // minutes, default 60 for HERE_NOW, 120 for SCHEDULED
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
// ============================================================================
|
|
36
|
+
// MUTATION HOOKS
|
|
37
|
+
// ============================================================================
|
|
38
|
+
|
|
39
|
+
/**
|
|
40
|
+
* Create a new user pin
|
|
41
|
+
*
|
|
42
|
+
* @description
|
|
43
|
+
* Creates a new pin showing where the user is or will be.
|
|
44
|
+
* Any existing active pin is automatically deactivated.
|
|
45
|
+
*
|
|
46
|
+
* Two types of pins:
|
|
47
|
+
* - HERE_NOW: User is currently at this location (default 1 hour duration)
|
|
48
|
+
* - SCHEDULED: User will be at this location at a specific time (default 2 hour duration)
|
|
49
|
+
*
|
|
50
|
+
* @endpoint POST /api/v1/user-pins
|
|
51
|
+
*
|
|
52
|
+
* @example
|
|
53
|
+
* Drop a "I'm here now" pin:
|
|
54
|
+
* ```tsx
|
|
55
|
+
* import { useCreateUserPin } from '@growsober/sdk';
|
|
56
|
+
*
|
|
57
|
+
* function DropPinButton({ location }) {
|
|
58
|
+
* const createPin = useCreateUserPin();
|
|
59
|
+
*
|
|
60
|
+
* const handleDropPin = () => {
|
|
61
|
+
* createPin.mutate({
|
|
62
|
+
* type: 'HERE_NOW',
|
|
63
|
+
* latitude: location.lat,
|
|
64
|
+
* longitude: location.lng,
|
|
65
|
+
* locationName: 'Fabrica Coffee',
|
|
66
|
+
* activity: 'Working remotely',
|
|
67
|
+
* duration: 120, // 2 hours
|
|
68
|
+
* });
|
|
69
|
+
* };
|
|
70
|
+
*
|
|
71
|
+
* return (
|
|
72
|
+
* <Button onPress={handleDropPin} loading={createPin.isPending}>
|
|
73
|
+
* Drop Pin
|
|
74
|
+
* </Button>
|
|
75
|
+
* );
|
|
76
|
+
* }
|
|
77
|
+
* ```
|
|
78
|
+
*
|
|
79
|
+
* @example
|
|
80
|
+
* Schedule a pin for later:
|
|
81
|
+
* ```tsx
|
|
82
|
+
* import { useCreateUserPin } from '@growsober/sdk';
|
|
83
|
+
*
|
|
84
|
+
* function SchedulePinForm() {
|
|
85
|
+
* const createPin = useCreateUserPin({
|
|
86
|
+
* onSuccess: () => {
|
|
87
|
+
* toast.show('Pin scheduled!');
|
|
88
|
+
* },
|
|
89
|
+
* });
|
|
90
|
+
*
|
|
91
|
+
* const handleSchedule = (data) => {
|
|
92
|
+
* createPin.mutate({
|
|
93
|
+
* type: 'SCHEDULED',
|
|
94
|
+
* latitude: data.venue.lat,
|
|
95
|
+
* longitude: data.venue.lng,
|
|
96
|
+
* locationName: data.venue.name,
|
|
97
|
+
* venueId: data.venue.id,
|
|
98
|
+
* activity: 'Coffee meetup',
|
|
99
|
+
* scheduledTime: data.time.toISOString(),
|
|
100
|
+
* duration: 90,
|
|
101
|
+
* });
|
|
102
|
+
* };
|
|
103
|
+
*
|
|
104
|
+
* return <ScheduleForm onSubmit={handleSchedule} />;
|
|
105
|
+
* }
|
|
106
|
+
* ```
|
|
107
|
+
*
|
|
108
|
+
* @param options - TanStack Query mutation options
|
|
109
|
+
* @returns TanStack Query mutation result
|
|
110
|
+
*/
|
|
111
|
+
export function useCreateUserPin(
|
|
112
|
+
options?: Omit<
|
|
113
|
+
UseMutationOptions<UserPinResponse, Error, CreatePinRequest>,
|
|
114
|
+
'mutationFn'
|
|
115
|
+
>
|
|
116
|
+
): UseMutationResult<UserPinResponse, Error, CreatePinRequest> {
|
|
117
|
+
const queryClient = useQueryClient();
|
|
118
|
+
|
|
119
|
+
return useMutation({
|
|
120
|
+
mutationFn: async (data: CreatePinRequest): Promise<UserPinResponse> => {
|
|
121
|
+
const client = getApiClient();
|
|
122
|
+
const response = await client.post<UserPinResponse>('/api/v1/user-pins', data);
|
|
123
|
+
return response.data;
|
|
124
|
+
},
|
|
125
|
+
onSuccess: () => {
|
|
126
|
+
// Invalidate pin queries to reflect new pin
|
|
127
|
+
queryClient.invalidateQueries({ queryKey: userPinKeys.all });
|
|
128
|
+
},
|
|
129
|
+
...options,
|
|
130
|
+
});
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
/**
|
|
134
|
+
* Delete a user pin
|
|
135
|
+
*
|
|
136
|
+
* @description
|
|
137
|
+
* Deactivates a user pin. Users can only delete their own pins.
|
|
138
|
+
*
|
|
139
|
+
* @endpoint DELETE /api/v1/user-pins/:id
|
|
140
|
+
*
|
|
141
|
+
* @example
|
|
142
|
+
* ```tsx
|
|
143
|
+
* import { useDeleteUserPin } from '@growsober/sdk';
|
|
144
|
+
*
|
|
145
|
+
* function MyPinCard({ pin }) {
|
|
146
|
+
* const deletePin = useDeleteUserPin({
|
|
147
|
+
* onSuccess: () => {
|
|
148
|
+
* toast.show('Pin removed');
|
|
149
|
+
* },
|
|
150
|
+
* });
|
|
151
|
+
*
|
|
152
|
+
* return (
|
|
153
|
+
* <Card>
|
|
154
|
+
* <Text>{pin.activity}</Text>
|
|
155
|
+
* <Text>At {pin.locationName}</Text>
|
|
156
|
+
* <Button
|
|
157
|
+
* onPress={() => deletePin.mutate(pin.id)}
|
|
158
|
+
* loading={deletePin.isPending}
|
|
159
|
+
* >
|
|
160
|
+
* Remove Pin
|
|
161
|
+
* </Button>
|
|
162
|
+
* </Card>
|
|
163
|
+
* );
|
|
164
|
+
* }
|
|
165
|
+
* ```
|
|
166
|
+
*
|
|
167
|
+
* @param options - TanStack Query mutation options
|
|
168
|
+
* @returns TanStack Query mutation result
|
|
169
|
+
*/
|
|
170
|
+
export function useDeleteUserPin(
|
|
171
|
+
options?: Omit<UseMutationOptions<void, Error, string>, 'mutationFn'>
|
|
172
|
+
): UseMutationResult<void, Error, string> {
|
|
173
|
+
const queryClient = useQueryClient();
|
|
174
|
+
|
|
175
|
+
return useMutation({
|
|
176
|
+
mutationFn: async (pinId: string): Promise<void> => {
|
|
177
|
+
const client = getApiClient();
|
|
178
|
+
await client.delete(`/api/v1/user-pins/${pinId}`);
|
|
179
|
+
},
|
|
180
|
+
onSuccess: () => {
|
|
181
|
+
// Invalidate pin queries to reflect deletion
|
|
182
|
+
queryClient.invalidateQueries({ queryKey: userPinKeys.all });
|
|
183
|
+
},
|
|
184
|
+
...options,
|
|
185
|
+
});
|
|
186
|
+
}
|
package/src/api/queries/index.ts
CHANGED
|
@@ -0,0 +1,193 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* User Pins Query Hooks
|
|
3
|
+
*
|
|
4
|
+
* TanStack Query hooks for user pin read operations.
|
|
5
|
+
* These hooks handle fetching user pins for the map feature
|
|
6
|
+
* ("I'm here now" / "I'll be there").
|
|
7
|
+
*
|
|
8
|
+
* @module api/queries/user-pins
|
|
9
|
+
*/
|
|
10
|
+
|
|
11
|
+
import { useQuery, UseQueryOptions, UseQueryResult } from '@tanstack/react-query';
|
|
12
|
+
import { getApiClient } from '../client';
|
|
13
|
+
|
|
14
|
+
// ============================================================================
|
|
15
|
+
// TYPES
|
|
16
|
+
// ============================================================================
|
|
17
|
+
|
|
18
|
+
export type PinType = 'HERE_NOW' | 'SCHEDULED';
|
|
19
|
+
|
|
20
|
+
export interface UserPinResponse {
|
|
21
|
+
id: string;
|
|
22
|
+
userId: string;
|
|
23
|
+
type: PinType;
|
|
24
|
+
latitude: number;
|
|
25
|
+
longitude: number;
|
|
26
|
+
locationName: string | null;
|
|
27
|
+
activity: string;
|
|
28
|
+
venueId: string | null;
|
|
29
|
+
scheduledTime: string | null;
|
|
30
|
+
expiresAt: string;
|
|
31
|
+
duration: number | null;
|
|
32
|
+
isActive: boolean;
|
|
33
|
+
createdAt: string;
|
|
34
|
+
user?: {
|
|
35
|
+
id: string;
|
|
36
|
+
name: string | null;
|
|
37
|
+
profileImage: string | null;
|
|
38
|
+
};
|
|
39
|
+
venue?: {
|
|
40
|
+
id: string;
|
|
41
|
+
name: string;
|
|
42
|
+
slug: string | null;
|
|
43
|
+
};
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
export interface NearbyPinsResponse {
|
|
47
|
+
pins: UserPinResponse[];
|
|
48
|
+
total: number;
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
// ============================================================================
|
|
52
|
+
// QUERY KEYS
|
|
53
|
+
// ============================================================================
|
|
54
|
+
|
|
55
|
+
/**
|
|
56
|
+
* Query key factory for user pin queries
|
|
57
|
+
*/
|
|
58
|
+
export const userPinKeys = {
|
|
59
|
+
all: ['user-pins'] as const,
|
|
60
|
+
nearby: (params: NearbyPinsParams) => [...userPinKeys.all, 'nearby', params] as const,
|
|
61
|
+
me: () => [...userPinKeys.all, 'me'] as const,
|
|
62
|
+
};
|
|
63
|
+
|
|
64
|
+
// ============================================================================
|
|
65
|
+
// PARAM TYPES
|
|
66
|
+
// ============================================================================
|
|
67
|
+
|
|
68
|
+
export interface NearbyPinsParams {
|
|
69
|
+
lat: number;
|
|
70
|
+
lng: number;
|
|
71
|
+
radius?: number; // km, default 5
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
// ============================================================================
|
|
75
|
+
// QUERY HOOKS
|
|
76
|
+
// ============================================================================
|
|
77
|
+
|
|
78
|
+
/**
|
|
79
|
+
* Get nearby user pins
|
|
80
|
+
*
|
|
81
|
+
* @description
|
|
82
|
+
* Retrieves active user pins within a radius of a location.
|
|
83
|
+
* Shows where other GrowSober members are or will be.
|
|
84
|
+
*
|
|
85
|
+
* @endpoint GET /api/v1/user-pins/nearby
|
|
86
|
+
*
|
|
87
|
+
* @example
|
|
88
|
+
* ```tsx
|
|
89
|
+
* import { useNearbyPins } from '@growsober/sdk';
|
|
90
|
+
*
|
|
91
|
+
* function MapWithPins() {
|
|
92
|
+
* const { data, isLoading } = useNearbyPins({
|
|
93
|
+
* lat: 38.7097,
|
|
94
|
+
* lng: -9.1367,
|
|
95
|
+
* radius: 5, // 5km radius
|
|
96
|
+
* });
|
|
97
|
+
*
|
|
98
|
+
* return (
|
|
99
|
+
* <Map>
|
|
100
|
+
* {data?.pins.map(pin => (
|
|
101
|
+
* <Marker
|
|
102
|
+
* key={pin.id}
|
|
103
|
+
* position={[pin.latitude, pin.longitude]}
|
|
104
|
+
* icon={pin.type === 'HERE_NOW' ? 'here' : 'scheduled'}
|
|
105
|
+
* >
|
|
106
|
+
* <Popup>
|
|
107
|
+
* <p>{pin.user?.name}</p>
|
|
108
|
+
* <p>{pin.activity}</p>
|
|
109
|
+
* </Popup>
|
|
110
|
+
* </Marker>
|
|
111
|
+
* ))}
|
|
112
|
+
* </Map>
|
|
113
|
+
* );
|
|
114
|
+
* }
|
|
115
|
+
* ```
|
|
116
|
+
*
|
|
117
|
+
* @param params - Location and radius parameters
|
|
118
|
+
* @param options - TanStack Query options
|
|
119
|
+
* @returns TanStack Query result with nearby pins
|
|
120
|
+
*/
|
|
121
|
+
export function useNearbyPins(
|
|
122
|
+
params: NearbyPinsParams,
|
|
123
|
+
options?: Omit<UseQueryOptions<NearbyPinsResponse>, 'queryKey' | 'queryFn'>
|
|
124
|
+
): UseQueryResult<NearbyPinsResponse> {
|
|
125
|
+
return useQuery({
|
|
126
|
+
queryKey: userPinKeys.nearby(params),
|
|
127
|
+
queryFn: async (): Promise<NearbyPinsResponse> => {
|
|
128
|
+
const client = getApiClient();
|
|
129
|
+
const response = await client.get<NearbyPinsResponse>('/api/v1/user-pins/nearby', {
|
|
130
|
+
params: {
|
|
131
|
+
lat: params.lat,
|
|
132
|
+
lng: params.lng,
|
|
133
|
+
radius: params.radius,
|
|
134
|
+
},
|
|
135
|
+
});
|
|
136
|
+
return response.data;
|
|
137
|
+
},
|
|
138
|
+
enabled: params.lat !== undefined && params.lng !== undefined,
|
|
139
|
+
staleTime: 30 * 1000, // 30 seconds - pins change frequently
|
|
140
|
+
...options,
|
|
141
|
+
});
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
/**
|
|
145
|
+
* Get current user's active pins
|
|
146
|
+
*
|
|
147
|
+
* @description
|
|
148
|
+
* Retrieves the authenticated user's active pins.
|
|
149
|
+
* Users can have at most one active pin at a time.
|
|
150
|
+
*
|
|
151
|
+
* @endpoint GET /api/v1/user-pins/me
|
|
152
|
+
*
|
|
153
|
+
* @example
|
|
154
|
+
* ```tsx
|
|
155
|
+
* import { useMyPins } from '@growsober/sdk';
|
|
156
|
+
*
|
|
157
|
+
* function MyPinStatus() {
|
|
158
|
+
* const { data: pins, isLoading } = useMyPins();
|
|
159
|
+
*
|
|
160
|
+
* if (isLoading) return <Spinner />;
|
|
161
|
+
*
|
|
162
|
+
* if (!pins?.length) {
|
|
163
|
+
* return <Text>You haven't dropped a pin yet</Text>;
|
|
164
|
+
* }
|
|
165
|
+
*
|
|
166
|
+
* const pin = pins[0];
|
|
167
|
+
* return (
|
|
168
|
+
* <View>
|
|
169
|
+
* <Text>You're at {pin.locationName}</Text>
|
|
170
|
+
* <Text>Activity: {pin.activity}</Text>
|
|
171
|
+
* <Text>Expires: {formatTime(pin.expiresAt)}</Text>
|
|
172
|
+
* </View>
|
|
173
|
+
* );
|
|
174
|
+
* }
|
|
175
|
+
* ```
|
|
176
|
+
*
|
|
177
|
+
* @param options - TanStack Query options
|
|
178
|
+
* @returns TanStack Query result with user's active pins
|
|
179
|
+
*/
|
|
180
|
+
export function useMyPins(
|
|
181
|
+
options?: Omit<UseQueryOptions<UserPinResponse[]>, 'queryKey' | 'queryFn'>
|
|
182
|
+
): UseQueryResult<UserPinResponse[]> {
|
|
183
|
+
return useQuery({
|
|
184
|
+
queryKey: userPinKeys.me(),
|
|
185
|
+
queryFn: async (): Promise<UserPinResponse[]> => {
|
|
186
|
+
const client = getApiClient();
|
|
187
|
+
const response = await client.get<UserPinResponse[]>('/api/v1/user-pins/me');
|
|
188
|
+
return response.data;
|
|
189
|
+
},
|
|
190
|
+
staleTime: 60 * 1000, // 1 minute
|
|
191
|
+
...options,
|
|
192
|
+
});
|
|
193
|
+
}
|