@growsober/sdk 1.0.3 → 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.
@@ -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
- chatId: string;
66
- userId: string;
65
+ conversationId: string;
66
+ role: "USER" | "ASSISTANT" | "SYSTEM";
67
67
  content: string;
68
- messageType: "TEXT" | "IMAGE" | "SYSTEM" | "ANNOUNCEMENT";
69
- imageUrl?: string;
70
- replyToId?: string;
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
- chatId: string;
91
- userId: string;
80
+ conversationId: string;
81
+ role: "USER" | "ASSISTANT" | "SYSTEM";
92
82
  content: string;
93
- messageType: "TEXT" | "IMAGE" | "SYSTEM" | "ANNOUNCEMENT";
94
- imageUrl?: string;
95
- replyToId?: string;
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
  } & {
@@ -20,3 +20,4 @@ export * from './ambassadors';
20
20
  export * from './grow90';
21
21
  export * from './matching';
22
22
  export * from './event-chat';
23
+ export * from './user-pins';
@@ -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
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvYXBpL211dGF0aW9ucy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUE7Ozs7R0FJRzs7Ozs7Ozs7Ozs7Ozs7OztBQUVILDBDQUF3QjtBQUN4Qix5Q0FBdUI7QUFDdkIsNkNBQTJCO0FBQzNCLDJDQUF5QjtBQUN6Qix5Q0FBdUI7QUFDdkIsNENBQTBCO0FBQzFCLHdDQUFzQjtBQUN0QixrREFBZ0M7QUFDaEMsMkNBQXlCO0FBQ3pCLGtEQUFnQztBQUNoQyw0Q0FBMEI7QUFDMUIsMENBQXdCO0FBQ3hCLHlDQUF1QjtBQUN2QixnREFBOEI7QUFDOUIsMkNBQXlCO0FBQ3pCLDZDQUEyQjtBQUMzQiwrQ0FBNkIiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIE11dGF0aW9uIEhvb2tzXG4gKlxuICogUmUtZXhwb3J0cyBhbGwgbXV0YXRpb24gaG9va3MgZm9yIEFQSSBlbmRwb2ludHMuXG4gKi9cblxuZXhwb3J0ICogZnJvbSAnLi9hZG1pbic7XG5leHBvcnQgKiBmcm9tICcuL2F1dGgnO1xuZXhwb3J0ICogZnJvbSAnLi9ib29raW5ncyc7XG5leHBvcnQgKiBmcm9tICcuL2V2ZW50cyc7XG5leHBvcnQgKiBmcm9tICcuL2h1YnMnO1xuZXhwb3J0ICogZnJvbSAnLi9saWJyYXJ5JztcbmV4cG9ydCAqIGZyb20gJy4vbWFwJztcbmV4cG9ydCAqIGZyb20gJy4vbm90aWZpY2F0aW9ucyc7XG5leHBvcnQgKiBmcm9tICcuL29mZmVycyc7XG5leHBvcnQgKiBmcm9tICcuL3N1YnNjcmlwdGlvbnMnO1xuZXhwb3J0ICogZnJvbSAnLi9zdXBwb3J0JztcbmV4cG9ydCAqIGZyb20gJy4vdXNlcnMnO1xuZXhwb3J0ICogZnJvbSAnLi9qYWNrJztcbmV4cG9ydCAqIGZyb20gJy4vYW1iYXNzYWRvcnMnO1xuZXhwb3J0ICogZnJvbSAnLi9ncm93OTAnO1xuZXhwb3J0ICogZnJvbSAnLi9tYXRjaGluZyc7XG5leHBvcnQgKiBmcm9tICcuL2V2ZW50LWNoYXQnO1xuIl19
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"]}
@@ -22,3 +22,4 @@ export * from './ambassadors';
22
22
  export * from './grow90';
23
23
  export * from './matching';
24
24
  export * from './event-chat';
25
+ export * from './user-pins';
@@ -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
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvYXBpL3F1ZXJpZXMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOzs7O0dBSUc7Ozs7Ozs7Ozs7Ozs7Ozs7QUFFSCwwQ0FBd0I7QUFDeEIseUNBQXVCO0FBQ3ZCLDZDQUEyQjtBQUMzQiwrQ0FBNkI7QUFDN0IsMkNBQXlCO0FBQ3pCLDJDQUF5QjtBQUN6Qix5Q0FBdUI7QUFDdkIsNENBQTBCO0FBQzFCLHdDQUFzQjtBQUN0QixrREFBZ0M7QUFDaEMsMkNBQXlCO0FBQ3pCLGtEQUFnQztBQUNoQyw0Q0FBMEI7QUFDMUIsMENBQXdCO0FBQ3hCLHlDQUF1QjtBQUN2QixnREFBOEI7QUFDOUIsMkNBQXlCO0FBQ3pCLDZDQUEyQjtBQUMzQiwrQ0FBNkIiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIFF1ZXJ5IEhvb2tzXG4gKlxuICogUmUtZXhwb3J0cyBhbGwgcXVlcnkgaG9va3MgZm9yIEFQSSBlbmRwb2ludHMuXG4gKi9cblxuZXhwb3J0ICogZnJvbSAnLi9hZG1pbic7XG5leHBvcnQgKiBmcm9tICcuL2F1dGgnO1xuZXhwb3J0ICogZnJvbSAnLi9ib29raW5ncyc7XG5leHBvcnQgKiBmcm9tICcuL2J1c2luZXNzZXMnO1xuZXhwb3J0ICogZnJvbSAnLi9jaXRpZXMnO1xuZXhwb3J0ICogZnJvbSAnLi9ldmVudHMnO1xuZXhwb3J0ICogZnJvbSAnLi9odWJzJztcbmV4cG9ydCAqIGZyb20gJy4vbGlicmFyeSc7XG5leHBvcnQgKiBmcm9tICcuL21hcCc7XG5leHBvcnQgKiBmcm9tICcuL25vdGlmaWNhdGlvbnMnO1xuZXhwb3J0ICogZnJvbSAnLi9vZmZlcnMnO1xuZXhwb3J0ICogZnJvbSAnLi9zdWJzY3JpcHRpb25zJztcbmV4cG9ydCAqIGZyb20gJy4vc3VwcG9ydCc7XG5leHBvcnQgKiBmcm9tICcuL3VzZXJzJztcbmV4cG9ydCAqIGZyb20gJy4vamFjayc7XG5leHBvcnQgKiBmcm9tICcuL2FtYmFzc2Fkb3JzJztcbmV4cG9ydCAqIGZyb20gJy4vZ3JvdzkwJztcbmV4cG9ydCAqIGZyb20gJy4vbWF0Y2hpbmcnO1xuZXhwb3J0ICogZnJvbSAnLi9ldmVudC1jaGF0JztcbiJdfQ==
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
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@growsober/sdk",
3
- "version": "1.0.3",
3
+ "version": "1.0.4",
4
4
  "description": "Shared TypeScript SDK for GrowSober API - TanStack Query hooks, API client, and utilities",
5
5
  "main": "dist/index.js",
6
6
  "module": "dist/index.js",
@@ -21,3 +21,4 @@ export * from './ambassadors';
21
21
  export * from './grow90';
22
22
  export * from './matching';
23
23
  export * from './event-chat';
24
+ export * from './user-pins';
@@ -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
+ }
@@ -23,3 +23,4 @@ export * from './ambassadors';
23
23
  export * from './grow90';
24
24
  export * from './matching';
25
25
  export * from './event-chat';
26
+ export * from './user-pins';
@@ -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
+ }