@growsober/sdk 1.0.4 → 1.0.6
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/client.js +10 -3
- package/dist/api/mutations/user-pins.js +1 -1
- package/dist/api/queries/events.d.ts +10 -0
- package/dist/api/utils/eventGrouping.d.ts +104 -0
- package/dist/api/utils/eventGrouping.js +155 -0
- package/dist/index.d.ts +1 -0
- package/dist/index.js +5 -1
- package/package.json +1 -1
- package/src/api/client.ts +9 -2
- package/src/api/mutations/user-pins.ts +1 -1
- package/src/api/utils/eventGrouping.ts +181 -0
- package/src/index.ts +6 -0
package/dist/api/client.js
CHANGED
|
@@ -35,8 +35,15 @@ function createApiClient(sdkConfig) {
|
|
|
35
35
|
}
|
|
36
36
|
return requestConfig;
|
|
37
37
|
}, (error) => Promise.reject(error));
|
|
38
|
-
// Response interceptor - handle 401
|
|
39
|
-
client.interceptors.response.use((response) =>
|
|
38
|
+
// Response interceptor - unwrap API response and handle 401
|
|
39
|
+
client.interceptors.response.use((response) => {
|
|
40
|
+
// API wraps all responses in {data: ..., meta: {...}}
|
|
41
|
+
// Unwrap to return just the data portion
|
|
42
|
+
if (response.data && typeof response.data === 'object' && 'data' in response.data) {
|
|
43
|
+
response.data = response.data.data;
|
|
44
|
+
}
|
|
45
|
+
return response;
|
|
46
|
+
}, async (error) => {
|
|
40
47
|
if (error.response?.status === 401) {
|
|
41
48
|
if (sdkConfig.refreshAccessToken) {
|
|
42
49
|
try {
|
|
@@ -58,4 +65,4 @@ function createApiClient(sdkConfig) {
|
|
|
58
65
|
});
|
|
59
66
|
return client;
|
|
60
67
|
}
|
|
61
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
68
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xpZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2FwaS9jbGllbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7O0FBWUEsb0NBR0M7QUFFRCxvQ0FLQztBQXRCRCxrREFBcUY7QUFTckYsSUFBSSxNQUFNLEdBQXFCLElBQUksQ0FBQztBQUNwQyxJQUFJLFNBQVMsR0FBeUIsSUFBSSxDQUFDO0FBbUVsQyw4QkFBUztBQWpFbEIsU0FBZ0IsWUFBWSxDQUFDLFNBQW9CO0lBQy9DLE1BQU0sR0FBRyxTQUFTLENBQUM7SUFDbkIsb0JBQUEsU0FBUyxHQUFHLGVBQWUsQ0FBQyxTQUFTLENBQUMsQ0FBQztBQUN6QyxDQUFDO0FBRUQsU0FBZ0IsWUFBWTtJQUMxQixJQUFJLENBQUMsU0FBUyxFQUFFLENBQUM7UUFDZixNQUFNLElBQUksS0FBSyxDQUFDLGdEQUFnRCxDQUFDLENBQUM7SUFDcEUsQ0FBQztJQUNELE9BQU8sU0FBUyxDQUFDO0FBQ25CLENBQUM7QUFFRCxTQUFTLGVBQWUsQ0FBQyxTQUFvQjtJQUMzQyxNQUFNLE1BQU0sR0FBRyxlQUFLLENBQUMsTUFBTSxDQUFDO1FBQzFCLE9BQU8sRUFBRSxTQUFTLENBQUMsT0FBTztRQUMxQixPQUFPLEVBQUU7WUFDUCxjQUFjLEVBQUUsa0JBQWtCO1NBQ25DO0tBQ0YsQ0FBQyxDQUFDO0lBRUgsdUNBQXVDO0lBQ3ZDLE1BQU0sQ0FBQyxZQUFZLENBQUMsT0FBTyxDQUFDLEdBQUcsQ0FDN0IsS0FBSyxFQUFFLGFBQXlDLEVBQUUsRUFBRTtRQUNsRCxNQUFNLEtBQUssR0FBRyxNQUFNLFNBQVMsQ0FBQyxjQUFjLEVBQUUsQ0FBQztRQUMvQyxJQUFJLEtBQUssRUFBRSxDQUFDO1lBQ1YsYUFBYSxDQUFDLE9BQU8sQ0FBQyxhQUFhLEdBQUcsVUFBVSxLQUFLLEVBQUUsQ0FBQztRQUMxRCxDQUFDO1FBQ0QsT0FBTyxhQUFhLENBQUM7SUFDdkIsQ0FBQyxFQUNELENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FBQyxPQUFPLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUNqQyxDQUFDO0lBRUYsNERBQTREO0lBQzVELE1BQU0sQ0FBQyxZQUFZLENBQUMsUUFBUSxDQUFDLEdBQUcsQ0FDOUIsQ0FBQyxRQUFRLEVBQUUsRUFBRTtRQUNYLHNEQUFzRDtRQUN0RCx5Q0FBeUM7UUFDekMsSUFBSSxRQUFRLENBQUMsSUFBSSxJQUFJLE9BQU8sUUFBUSxDQUFDLElBQUksS0FBSyxRQUFRLElBQUksTUFBTSxJQUFJLFFBQVEsQ0FBQyxJQUFJLEVBQUUsQ0FBQztZQUNsRixRQUFRLENBQUMsSUFBSSxHQUFHLFFBQVEsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDO1FBQ3JDLENBQUM7UUFDRCxPQUFPLFFBQVEsQ0FBQztJQUNsQixDQUFDLEVBQ0QsS0FBSyxFQUFFLEtBQWlCLEVBQUUsRUFBRTtRQUMxQixJQUFJLEtBQUssQ0FBQyxRQUFRLEVBQUUsTUFBTSxLQUFLLEdBQUcsRUFBRSxDQUFDO1lBQ25DLElBQUksU0FBUyxDQUFDLGtCQUFrQixFQUFFLENBQUM7Z0JBQ2pDLElBQUksQ0FBQztvQkFDSCxNQUFNLFFBQVEsR0FBRyxNQUFNLFNBQVMsQ0FBQyxrQkFBa0IsRUFBRSxDQUFDO29CQUN0RCxJQUFJLFFBQVEsSUFBSSxLQUFLLENBQUMsTUFBTSxFQUFFLENBQUM7d0JBQzdCLEtBQUssQ0FBQyxNQUFNLENBQUMsT0FBTyxDQUFDLGFBQWEsR0FBRyxVQUFVLFFBQVEsRUFBRSxDQUFDO3dCQUMxRCxPQUFPLE1BQU0sQ0FBQyxPQUFPLENBQUMsS0FBSyxDQUFDLE1BQU0sQ0FBQyxDQUFDO29CQUN0QyxDQUFDO2dCQUNILENBQUM7Z0JBQUMsTUFBTSxDQUFDO29CQUNQLFNBQVMsQ0FBQyxjQUFjLEVBQUUsRUFBRSxDQUFDO2dCQUMvQixDQUFDO1lBQ0gsQ0FBQztpQkFBTSxDQUFDO2dCQUNOLFNBQVMsQ0FBQyxjQUFjLEVBQUUsRUFBRSxDQUFDO1lBQy9CLENBQUM7UUFDSCxDQUFDO1FBQ0QsT0FBTyxPQUFPLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFDO0lBQy9CLENBQUMsQ0FDRixDQUFDO0lBRUYsT0FBTyxNQUFNLENBQUM7QUFDaEIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBheGlvcywgeyBBeGlvc0luc3RhbmNlLCBBeGlvc0Vycm9yLCBJbnRlcm5hbEF4aW9zUmVxdWVzdENvbmZpZyB9IGZyb20gJ2F4aW9zJztcblxuZXhwb3J0IGludGVyZmFjZSBTREtDb25maWcge1xuICBiYXNlVVJMOiBzdHJpbmc7XG4gIGdldEFjY2Vzc1Rva2VuOiAoKSA9PiBzdHJpbmcgfCBudWxsIHwgUHJvbWlzZTxzdHJpbmcgfCBudWxsPjtcbiAgcmVmcmVzaEFjY2Vzc1Rva2VuPzogKCkgPT4gUHJvbWlzZTxzdHJpbmc+O1xuICBvblVuYXV0aG9yaXplZD86ICgpID0+IHZvaWQ7XG59XG5cbmxldCBjb25maWc6IFNES0NvbmZpZyB8IG51bGwgPSBudWxsO1xubGV0IGFwaUNsaWVudDogQXhpb3NJbnN0YW5jZSB8IG51bGwgPSBudWxsO1xuXG5leHBvcnQgZnVuY3Rpb24gY29uZmlndXJlU0RLKHNka0NvbmZpZzogU0RLQ29uZmlnKTogdm9pZCB7XG4gIGNvbmZpZyA9IHNka0NvbmZpZztcbiAgYXBpQ2xpZW50ID0gY3JlYXRlQXBpQ2xpZW50KHNka0NvbmZpZyk7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBnZXRBcGlDbGllbnQoKTogQXhpb3NJbnN0YW5jZSB7XG4gIGlmICghYXBpQ2xpZW50KSB7XG4gICAgdGhyb3cgbmV3IEVycm9yKCdTREsgbm90IGNvbmZpZ3VyZWQuIENhbGwgY29uZmlndXJlU0RLKCkgZmlyc3QuJyk7XG4gIH1cbiAgcmV0dXJuIGFwaUNsaWVudDtcbn1cblxuZnVuY3Rpb24gY3JlYXRlQXBpQ2xpZW50KHNka0NvbmZpZzogU0RLQ29uZmlnKTogQXhpb3NJbnN0YW5jZSB7XG4gIGNvbnN0IGNsaWVudCA9IGF4aW9zLmNyZWF0ZSh7XG4gICAgYmFzZVVSTDogc2RrQ29uZmlnLmJhc2VVUkwsXG4gICAgaGVhZGVyczoge1xuICAgICAgJ0NvbnRlbnQtVHlwZSc6ICdhcHBsaWNhdGlvbi9qc29uJyxcbiAgICB9LFxuICB9KTtcblxuICAvLyBSZXF1ZXN0IGludGVyY2VwdG9yIC0gYWRkIGF1dGggdG9rZW5cbiAgY2xpZW50LmludGVyY2VwdG9ycy5yZXF1ZXN0LnVzZShcbiAgICBhc3luYyAocmVxdWVzdENvbmZpZzogSW50ZXJuYWxBeGlvc1JlcXVlc3RDb25maWcpID0+IHtcbiAgICAgIGNvbnN0IHRva2VuID0gYXdhaXQgc2RrQ29uZmlnLmdldEFjY2Vzc1Rva2VuKCk7XG4gICAgICBpZiAodG9rZW4pIHtcbiAgICAgICAgcmVxdWVzdENvbmZpZy5oZWFkZXJzLkF1dGhvcml6YXRpb24gPSBgQmVhcmVyICR7dG9rZW59YDtcbiAgICAgIH1cbiAgICAgIHJldHVybiByZXF1ZXN0Q29uZmlnO1xuICAgIH0sXG4gICAgKGVycm9yKSA9PiBQcm9taXNlLnJlamVjdChlcnJvcilcbiAgKTtcblxuICAvLyBSZXNwb25zZSBpbnRlcmNlcHRvciAtIHVud3JhcCBBUEkgcmVzcG9uc2UgYW5kIGhhbmRsZSA0MDFcbiAgY2xpZW50LmludGVyY2VwdG9ycy5yZXNwb25zZS51c2UoXG4gICAgKHJlc3BvbnNlKSA9PiB7XG4gICAgICAvLyBBUEkgd3JhcHMgYWxsIHJlc3BvbnNlcyBpbiB7ZGF0YTogLi4uLCBtZXRhOiB7Li4ufX1cbiAgICAgIC8vIFVud3JhcCB0byByZXR1cm4ganVzdCB0aGUgZGF0YSBwb3J0aW9uXG4gICAgICBpZiAocmVzcG9uc2UuZGF0YSAmJiB0eXBlb2YgcmVzcG9uc2UuZGF0YSA9PT0gJ29iamVjdCcgJiYgJ2RhdGEnIGluIHJlc3BvbnNlLmRhdGEpIHtcbiAgICAgICAgcmVzcG9uc2UuZGF0YSA9IHJlc3BvbnNlLmRhdGEuZGF0YTtcbiAgICAgIH1cbiAgICAgIHJldHVybiByZXNwb25zZTtcbiAgICB9LFxuICAgIGFzeW5jIChlcnJvcjogQXhpb3NFcnJvcikgPT4ge1xuICAgICAgaWYgKGVycm9yLnJlc3BvbnNlPy5zdGF0dXMgPT09IDQwMSkge1xuICAgICAgICBpZiAoc2RrQ29uZmlnLnJlZnJlc2hBY2Nlc3NUb2tlbikge1xuICAgICAgICAgIHRyeSB7XG4gICAgICAgICAgICBjb25zdCBuZXdUb2tlbiA9IGF3YWl0IHNka0NvbmZpZy5yZWZyZXNoQWNjZXNzVG9rZW4oKTtcbiAgICAgICAgICAgIGlmIChuZXdUb2tlbiAmJiBlcnJvci5jb25maWcpIHtcbiAgICAgICAgICAgICAgZXJyb3IuY29uZmlnLmhlYWRlcnMuQXV0aG9yaXphdGlvbiA9IGBCZWFyZXIgJHtuZXdUb2tlbn1gO1xuICAgICAgICAgICAgICByZXR1cm4gY2xpZW50LnJlcXVlc3QoZXJyb3IuY29uZmlnKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9IGNhdGNoIHtcbiAgICAgICAgICAgIHNka0NvbmZpZy5vblVuYXV0aG9yaXplZD8uKCk7XG4gICAgICAgICAgfVxuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIHNka0NvbmZpZy5vblVuYXV0aG9yaXplZD8uKCk7XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICAgIHJldHVybiBQcm9taXNlLnJlamVjdChlcnJvcik7XG4gICAgfVxuICApO1xuXG4gIHJldHVybiBjbGllbnQ7XG59XG5cbmV4cG9ydCB7IGFwaUNsaWVudCB9O1xuIl19
|
|
@@ -155,4 +155,4 @@ function useDeleteUserPin(options) {
|
|
|
155
155
|
...options,
|
|
156
156
|
});
|
|
157
157
|
}
|
|
158
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
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; // hours (1-8), default 2 for HERE_NOW, 2 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"]}
|
|
@@ -97,6 +97,8 @@ export declare function useEvent(id: string, options?: Omit<UseQueryOptions<Even
|
|
|
97
97
|
isCancelled: boolean;
|
|
98
98
|
isAmbient: boolean;
|
|
99
99
|
ambientCategory?: Record<string, never>;
|
|
100
|
+
vibe?: "CHILL" | "ACTIVE" | "SOCIAL" | "CREATIVE" | "DEEP_TALK" | "ADVENTURE";
|
|
101
|
+
city?: import("@growsober/types/dist/generated").components["schemas"]["EventCityDto"];
|
|
100
102
|
bookingCount: number;
|
|
101
103
|
checkinCount: number;
|
|
102
104
|
createdAt: string;
|
|
@@ -147,6 +149,8 @@ export declare function useEventBySlug(slug: string, options?: Omit<UseQueryOpti
|
|
|
147
149
|
isCancelled: boolean;
|
|
148
150
|
isAmbient: boolean;
|
|
149
151
|
ambientCategory?: Record<string, never>;
|
|
152
|
+
vibe?: "CHILL" | "ACTIVE" | "SOCIAL" | "CREATIVE" | "DEEP_TALK" | "ADVENTURE";
|
|
153
|
+
city?: import("@growsober/types/dist/generated").components["schemas"]["EventCityDto"];
|
|
150
154
|
bookingCount: number;
|
|
151
155
|
checkinCount: number;
|
|
152
156
|
createdAt: string;
|
|
@@ -197,6 +201,8 @@ export declare function useUpcomingEvents(limit?: number, options?: Omit<UseQuer
|
|
|
197
201
|
isCancelled: boolean;
|
|
198
202
|
isAmbient: boolean;
|
|
199
203
|
ambientCategory?: Record<string, never>;
|
|
204
|
+
vibe?: "CHILL" | "ACTIVE" | "SOCIAL" | "CREATIVE" | "DEEP_TALK" | "ADVENTURE";
|
|
205
|
+
city?: import("@growsober/types/dist/generated").components["schemas"]["EventCityDto"];
|
|
200
206
|
bookingCount: number;
|
|
201
207
|
checkinCount: number;
|
|
202
208
|
createdAt: string;
|
|
@@ -249,6 +255,8 @@ export declare function useFeaturedEvents(limit?: number, options?: Omit<UseQuer
|
|
|
249
255
|
isCancelled: boolean;
|
|
250
256
|
isAmbient: boolean;
|
|
251
257
|
ambientCategory?: Record<string, never>;
|
|
258
|
+
vibe?: "CHILL" | "ACTIVE" | "SOCIAL" | "CREATIVE" | "DEEP_TALK" | "ADVENTURE";
|
|
259
|
+
city?: import("@growsober/types/dist/generated").components["schemas"]["EventCityDto"];
|
|
252
260
|
bookingCount: number;
|
|
253
261
|
checkinCount: number;
|
|
254
262
|
createdAt: string;
|
|
@@ -299,6 +307,8 @@ export declare function useAmbientEvents(filters?: AmbientEventFilters, options?
|
|
|
299
307
|
isCancelled: boolean;
|
|
300
308
|
isAmbient: boolean;
|
|
301
309
|
ambientCategory?: Record<string, never>;
|
|
310
|
+
vibe?: "CHILL" | "ACTIVE" | "SOCIAL" | "CREATIVE" | "DEEP_TALK" | "ADVENTURE";
|
|
311
|
+
city?: import("@growsober/types/dist/generated").components["schemas"]["EventCityDto"];
|
|
302
312
|
bookingCount: number;
|
|
303
313
|
checkinCount: number;
|
|
304
314
|
createdAt: string;
|
|
@@ -0,0 +1,104 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Event Grouping Utilities
|
|
3
|
+
*
|
|
4
|
+
* Utilities for organizing events by time of day.
|
|
5
|
+
* Used in the Gatherings screen to show events grouped by
|
|
6
|
+
* Morning, Afternoon, and Evening.
|
|
7
|
+
*
|
|
8
|
+
* @module api/utils/eventGrouping
|
|
9
|
+
*/
|
|
10
|
+
import { EventResponse } from '../types';
|
|
11
|
+
export type TimeOfDay = 'MORNING' | 'AFTERNOON' | 'EVENING';
|
|
12
|
+
export interface GroupedEvents {
|
|
13
|
+
morning: EventResponse[];
|
|
14
|
+
afternoon: EventResponse[];
|
|
15
|
+
evening: EventResponse[];
|
|
16
|
+
}
|
|
17
|
+
export interface TimeOfDayInfo {
|
|
18
|
+
key: TimeOfDay;
|
|
19
|
+
label: string;
|
|
20
|
+
emoji: string;
|
|
21
|
+
hours: {
|
|
22
|
+
start: number;
|
|
23
|
+
end: number;
|
|
24
|
+
};
|
|
25
|
+
}
|
|
26
|
+
export declare const TIME_OF_DAY_CONFIG: Record<TimeOfDay, TimeOfDayInfo>;
|
|
27
|
+
/**
|
|
28
|
+
* Get time of day for a given date
|
|
29
|
+
*
|
|
30
|
+
* @param date - Date string or Date object
|
|
31
|
+
* @returns TimeOfDay - MORNING, AFTERNOON, or EVENING
|
|
32
|
+
*
|
|
33
|
+
* @example
|
|
34
|
+
* ```tsx
|
|
35
|
+
* const timeOfDay = getTimeOfDay('2024-03-15T10:00:00Z');
|
|
36
|
+
* console.log(timeOfDay); // 'MORNING'
|
|
37
|
+
* ```
|
|
38
|
+
*/
|
|
39
|
+
export declare function getTimeOfDay(date: Date | string): TimeOfDay;
|
|
40
|
+
/**
|
|
41
|
+
* Get label for time of day
|
|
42
|
+
*
|
|
43
|
+
* @param timeOfDay - TimeOfDay enum value
|
|
44
|
+
* @returns Human-readable label
|
|
45
|
+
*
|
|
46
|
+
* @example
|
|
47
|
+
* ```tsx
|
|
48
|
+
* const label = getTimeOfDayLabel('MORNING');
|
|
49
|
+
* console.log(label); // 'Morning'
|
|
50
|
+
* ```
|
|
51
|
+
*/
|
|
52
|
+
export declare function getTimeOfDayLabel(timeOfDay: TimeOfDay): string;
|
|
53
|
+
/**
|
|
54
|
+
* Group events by time of day
|
|
55
|
+
*
|
|
56
|
+
* @param events - Array of events to group
|
|
57
|
+
* @returns GroupedEvents object with morning, afternoon, and evening arrays
|
|
58
|
+
*
|
|
59
|
+
* @example
|
|
60
|
+
* ```tsx
|
|
61
|
+
* import { useUpcomingEvents, groupEventsByTimeOfDay } from '@growsober/sdk';
|
|
62
|
+
*
|
|
63
|
+
* function GatheringsList() {
|
|
64
|
+
* const { data: events } = useUpcomingEvents(30);
|
|
65
|
+
* const grouped = useMemo(() => {
|
|
66
|
+
* if (!events) return null;
|
|
67
|
+
* return groupEventsByTimeOfDay(events);
|
|
68
|
+
* }, [events]);
|
|
69
|
+
*
|
|
70
|
+
* return (
|
|
71
|
+
* <SectionList
|
|
72
|
+
* sections={[
|
|
73
|
+
* { title: 'Morning', data: grouped?.morning || [] },
|
|
74
|
+
* { title: 'Afternoon', data: grouped?.afternoon || [] },
|
|
75
|
+
* { title: 'Evening', data: grouped?.evening || [] },
|
|
76
|
+
* ]}
|
|
77
|
+
* ...
|
|
78
|
+
* />
|
|
79
|
+
* );
|
|
80
|
+
* }
|
|
81
|
+
* ```
|
|
82
|
+
*/
|
|
83
|
+
export declare function groupEventsByTimeOfDay(events: EventResponse[]): GroupedEvents;
|
|
84
|
+
/**
|
|
85
|
+
* Convert grouped events to section list data format
|
|
86
|
+
*
|
|
87
|
+
* @param events - Array of events to group
|
|
88
|
+
* @returns Array of sections with title and data
|
|
89
|
+
*
|
|
90
|
+
* @example
|
|
91
|
+
* ```tsx
|
|
92
|
+
* const sections = getEventSections(events);
|
|
93
|
+
* // [
|
|
94
|
+
* // { title: 'Morning', data: [...] },
|
|
95
|
+
* // { title: 'Afternoon', data: [...] },
|
|
96
|
+
* // { title: 'Evening', data: [...] },
|
|
97
|
+
* // ]
|
|
98
|
+
* ```
|
|
99
|
+
*/
|
|
100
|
+
export declare function getEventSections(events: EventResponse[]): Array<{
|
|
101
|
+
title: string;
|
|
102
|
+
timeOfDay: TimeOfDay;
|
|
103
|
+
data: EventResponse[];
|
|
104
|
+
}>;
|
|
@@ -0,0 +1,155 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* Event Grouping Utilities
|
|
4
|
+
*
|
|
5
|
+
* Utilities for organizing events by time of day.
|
|
6
|
+
* Used in the Gatherings screen to show events grouped by
|
|
7
|
+
* Morning, Afternoon, and Evening.
|
|
8
|
+
*
|
|
9
|
+
* @module api/utils/eventGrouping
|
|
10
|
+
*/
|
|
11
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
12
|
+
exports.TIME_OF_DAY_CONFIG = void 0;
|
|
13
|
+
exports.getTimeOfDay = getTimeOfDay;
|
|
14
|
+
exports.getTimeOfDayLabel = getTimeOfDayLabel;
|
|
15
|
+
exports.groupEventsByTimeOfDay = groupEventsByTimeOfDay;
|
|
16
|
+
exports.getEventSections = getEventSections;
|
|
17
|
+
// ============================================================================
|
|
18
|
+
// CONSTANTS
|
|
19
|
+
// ============================================================================
|
|
20
|
+
exports.TIME_OF_DAY_CONFIG = {
|
|
21
|
+
MORNING: {
|
|
22
|
+
key: 'MORNING',
|
|
23
|
+
label: 'Morning',
|
|
24
|
+
emoji: '',
|
|
25
|
+
hours: { start: 5, end: 11 }, // 5:00 AM - 11:59 AM
|
|
26
|
+
},
|
|
27
|
+
AFTERNOON: {
|
|
28
|
+
key: 'AFTERNOON',
|
|
29
|
+
label: 'Afternoon',
|
|
30
|
+
emoji: '',
|
|
31
|
+
hours: { start: 12, end: 16 }, // 12:00 PM - 4:59 PM
|
|
32
|
+
},
|
|
33
|
+
EVENING: {
|
|
34
|
+
key: 'EVENING',
|
|
35
|
+
label: 'Evening',
|
|
36
|
+
emoji: '',
|
|
37
|
+
hours: { start: 17, end: 4 }, // 5:00 PM - 4:59 AM (next day)
|
|
38
|
+
},
|
|
39
|
+
};
|
|
40
|
+
// ============================================================================
|
|
41
|
+
// FUNCTIONS
|
|
42
|
+
// ============================================================================
|
|
43
|
+
/**
|
|
44
|
+
* Get time of day for a given date
|
|
45
|
+
*
|
|
46
|
+
* @param date - Date string or Date object
|
|
47
|
+
* @returns TimeOfDay - MORNING, AFTERNOON, or EVENING
|
|
48
|
+
*
|
|
49
|
+
* @example
|
|
50
|
+
* ```tsx
|
|
51
|
+
* const timeOfDay = getTimeOfDay('2024-03-15T10:00:00Z');
|
|
52
|
+
* console.log(timeOfDay); // 'MORNING'
|
|
53
|
+
* ```
|
|
54
|
+
*/
|
|
55
|
+
function getTimeOfDay(date) {
|
|
56
|
+
const d = typeof date === 'string' ? new Date(date) : date;
|
|
57
|
+
const hour = d.getHours();
|
|
58
|
+
if (hour >= 5 && hour < 12)
|
|
59
|
+
return 'MORNING';
|
|
60
|
+
if (hour >= 12 && hour < 17)
|
|
61
|
+
return 'AFTERNOON';
|
|
62
|
+
return 'EVENING';
|
|
63
|
+
}
|
|
64
|
+
/**
|
|
65
|
+
* Get label for time of day
|
|
66
|
+
*
|
|
67
|
+
* @param timeOfDay - TimeOfDay enum value
|
|
68
|
+
* @returns Human-readable label
|
|
69
|
+
*
|
|
70
|
+
* @example
|
|
71
|
+
* ```tsx
|
|
72
|
+
* const label = getTimeOfDayLabel('MORNING');
|
|
73
|
+
* console.log(label); // 'Morning'
|
|
74
|
+
* ```
|
|
75
|
+
*/
|
|
76
|
+
function getTimeOfDayLabel(timeOfDay) {
|
|
77
|
+
return exports.TIME_OF_DAY_CONFIG[timeOfDay].label;
|
|
78
|
+
}
|
|
79
|
+
/**
|
|
80
|
+
* Group events by time of day
|
|
81
|
+
*
|
|
82
|
+
* @param events - Array of events to group
|
|
83
|
+
* @returns GroupedEvents object with morning, afternoon, and evening arrays
|
|
84
|
+
*
|
|
85
|
+
* @example
|
|
86
|
+
* ```tsx
|
|
87
|
+
* import { useUpcomingEvents, groupEventsByTimeOfDay } from '@growsober/sdk';
|
|
88
|
+
*
|
|
89
|
+
* function GatheringsList() {
|
|
90
|
+
* const { data: events } = useUpcomingEvents(30);
|
|
91
|
+
* const grouped = useMemo(() => {
|
|
92
|
+
* if (!events) return null;
|
|
93
|
+
* return groupEventsByTimeOfDay(events);
|
|
94
|
+
* }, [events]);
|
|
95
|
+
*
|
|
96
|
+
* return (
|
|
97
|
+
* <SectionList
|
|
98
|
+
* sections={[
|
|
99
|
+
* { title: 'Morning', data: grouped?.morning || [] },
|
|
100
|
+
* { title: 'Afternoon', data: grouped?.afternoon || [] },
|
|
101
|
+
* { title: 'Evening', data: grouped?.evening || [] },
|
|
102
|
+
* ]}
|
|
103
|
+
* ...
|
|
104
|
+
* />
|
|
105
|
+
* );
|
|
106
|
+
* }
|
|
107
|
+
* ```
|
|
108
|
+
*/
|
|
109
|
+
function groupEventsByTimeOfDay(events) {
|
|
110
|
+
const grouped = {
|
|
111
|
+
morning: [],
|
|
112
|
+
afternoon: [],
|
|
113
|
+
evening: [],
|
|
114
|
+
};
|
|
115
|
+
for (const event of events) {
|
|
116
|
+
const timeOfDay = getTimeOfDay(event.startDate);
|
|
117
|
+
switch (timeOfDay) {
|
|
118
|
+
case 'MORNING':
|
|
119
|
+
grouped.morning.push(event);
|
|
120
|
+
break;
|
|
121
|
+
case 'AFTERNOON':
|
|
122
|
+
grouped.afternoon.push(event);
|
|
123
|
+
break;
|
|
124
|
+
case 'EVENING':
|
|
125
|
+
grouped.evening.push(event);
|
|
126
|
+
break;
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
return grouped;
|
|
130
|
+
}
|
|
131
|
+
/**
|
|
132
|
+
* Convert grouped events to section list data format
|
|
133
|
+
*
|
|
134
|
+
* @param events - Array of events to group
|
|
135
|
+
* @returns Array of sections with title and data
|
|
136
|
+
*
|
|
137
|
+
* @example
|
|
138
|
+
* ```tsx
|
|
139
|
+
* const sections = getEventSections(events);
|
|
140
|
+
* // [
|
|
141
|
+
* // { title: 'Morning', data: [...] },
|
|
142
|
+
* // { title: 'Afternoon', data: [...] },
|
|
143
|
+
* // { title: 'Evening', data: [...] },
|
|
144
|
+
* // ]
|
|
145
|
+
* ```
|
|
146
|
+
*/
|
|
147
|
+
function getEventSections(events) {
|
|
148
|
+
const grouped = groupEventsByTimeOfDay(events);
|
|
149
|
+
return [
|
|
150
|
+
{ title: 'Morning', timeOfDay: 'MORNING', data: grouped.morning },
|
|
151
|
+
{ title: 'Afternoon', timeOfDay: 'AFTERNOON', data: grouped.afternoon },
|
|
152
|
+
{ title: 'Evening', timeOfDay: 'EVENING', data: grouped.evening },
|
|
153
|
+
].filter(section => section.data.length > 0);
|
|
154
|
+
}
|
|
155
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"eventGrouping.js","sourceRoot":"","sources":["../../../src/api/utils/eventGrouping.ts"],"names":[],"mappings":";AAAA;;;;;;;;GAQG;;;AAgEH,oCAOC;AAcD,8CAEC;AAgCD,wDAuBC;AAkBD,4CAYC;AArJD,+EAA+E;AAC/E,YAAY;AACZ,+EAA+E;AAElE,QAAA,kBAAkB,GAAqC;IAClE,OAAO,EAAE;QACP,GAAG,EAAE,SAAS;QACd,KAAK,EAAE,SAAS;QAChB,KAAK,EAAE,EAAE;QACT,KAAK,EAAE,EAAE,KAAK,EAAE,CAAC,EAAE,GAAG,EAAE,EAAE,EAAE,EAAE,qBAAqB;KACpD;IACD,SAAS,EAAE;QACT,GAAG,EAAE,WAAW;QAChB,KAAK,EAAE,WAAW;QAClB,KAAK,EAAE,EAAE;QACT,KAAK,EAAE,EAAE,KAAK,EAAE,EAAE,EAAE,GAAG,EAAE,EAAE,EAAE,EAAE,qBAAqB;KACrD;IACD,OAAO,EAAE;QACP,GAAG,EAAE,SAAS;QACd,KAAK,EAAE,SAAS;QAChB,KAAK,EAAE,EAAE;QACT,KAAK,EAAE,EAAE,KAAK,EAAE,EAAE,EAAE,GAAG,EAAE,CAAC,EAAE,EAAE,+BAA+B;KAC9D;CACF,CAAC;AAEF,+EAA+E;AAC/E,YAAY;AACZ,+EAA+E;AAE/E;;;;;;;;;;;GAWG;AACH,SAAgB,YAAY,CAAC,IAAmB;IAC9C,MAAM,CAAC,GAAG,OAAO,IAAI,KAAK,QAAQ,CAAC,CAAC,CAAC,IAAI,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;IAC3D,MAAM,IAAI,GAAG,CAAC,CAAC,QAAQ,EAAE,CAAC;IAE1B,IAAI,IAAI,IAAI,CAAC,IAAI,IAAI,GAAG,EAAE;QAAE,OAAO,SAAS,CAAC;IAC7C,IAAI,IAAI,IAAI,EAAE,IAAI,IAAI,GAAG,EAAE;QAAE,OAAO,WAAW,CAAC;IAChD,OAAO,SAAS,CAAC;AACnB,CAAC;AAED;;;;;;;;;;;GAWG;AACH,SAAgB,iBAAiB,CAAC,SAAoB;IACpD,OAAO,0BAAkB,CAAC,SAAS,CAAC,CAAC,KAAK,CAAC;AAC7C,CAAC;AAED;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA6BG;AACH,SAAgB,sBAAsB,CAAC,MAAuB;IAC5D,MAAM,OAAO,GAAkB;QAC7B,OAAO,EAAE,EAAE;QACX,SAAS,EAAE,EAAE;QACb,OAAO,EAAE,EAAE;KACZ,CAAC;IAEF,KAAK,MAAM,KAAK,IAAI,MAAM,EAAE,CAAC;QAC3B,MAAM,SAAS,GAAG,YAAY,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;QAChD,QAAQ,SAAS,EAAE,CAAC;YAClB,KAAK,SAAS;gBACZ,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;gBAC5B,MAAM;YACR,KAAK,WAAW;gBACd,OAAO,CAAC,SAAS,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;gBAC9B,MAAM;YACR,KAAK,SAAS;gBACZ,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;gBAC5B,MAAM;QACV,CAAC;IACH,CAAC;IAED,OAAO,OAAO,CAAC;AACjB,CAAC;AAED;;;;;;;;;;;;;;;GAeG;AACH,SAAgB,gBAAgB,CAAC,MAAuB;IAKtD,MAAM,OAAO,GAAG,sBAAsB,CAAC,MAAM,CAAC,CAAC;IAE/C,OAAO;QACL,EAAE,KAAK,EAAE,SAAS,EAAE,SAAS,EAAE,SAAsB,EAAE,IAAI,EAAE,OAAO,CAAC,OAAO,EAAE;QAC9E,EAAE,KAAK,EAAE,WAAW,EAAE,SAAS,EAAE,WAAwB,EAAE,IAAI,EAAE,OAAO,CAAC,SAAS,EAAE;QACpF,EAAE,KAAK,EAAE,SAAS,EAAE,SAAS,EAAE,SAAsB,EAAE,IAAI,EAAE,OAAO,CAAC,OAAO,EAAE;KAC/E,CAAC,MAAM,CAAC,OAAO,CAAC,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;AAC/C,CAAC","sourcesContent":["/**\n * Event Grouping Utilities\n *\n * Utilities for organizing events by time of day.\n * Used in the Gatherings screen to show events grouped by\n * Morning, Afternoon, and Evening.\n *\n * @module api/utils/eventGrouping\n */\n\nimport { EventResponse } from '../types';\n\n// ============================================================================\n// TYPES\n// ============================================================================\n\nexport type TimeOfDay = 'MORNING' | 'AFTERNOON' | 'EVENING';\n\nexport interface GroupedEvents {\n  morning: EventResponse[];\n  afternoon: EventResponse[];\n  evening: EventResponse[];\n}\n\nexport interface TimeOfDayInfo {\n  key: TimeOfDay;\n  label: string;\n  emoji: string;\n  hours: { start: number; end: number };\n}\n\n// ============================================================================\n// CONSTANTS\n// ============================================================================\n\nexport const TIME_OF_DAY_CONFIG: Record<TimeOfDay, TimeOfDayInfo> = {\n  MORNING: {\n    key: 'MORNING',\n    label: 'Morning',\n    emoji: '',\n    hours: { start: 5, end: 11 }, // 5:00 AM - 11:59 AM\n  },\n  AFTERNOON: {\n    key: 'AFTERNOON',\n    label: 'Afternoon',\n    emoji: '',\n    hours: { start: 12, end: 16 }, // 12:00 PM - 4:59 PM\n  },\n  EVENING: {\n    key: 'EVENING',\n    label: 'Evening',\n    emoji: '',\n    hours: { start: 17, end: 4 }, // 5:00 PM - 4:59 AM (next day)\n  },\n};\n\n// ============================================================================\n// FUNCTIONS\n// ============================================================================\n\n/**\n * Get time of day for a given date\n *\n * @param date - Date string or Date object\n * @returns TimeOfDay - MORNING, AFTERNOON, or EVENING\n *\n * @example\n * ```tsx\n * const timeOfDay = getTimeOfDay('2024-03-15T10:00:00Z');\n * console.log(timeOfDay); // 'MORNING'\n * ```\n */\nexport function getTimeOfDay(date: Date | string): TimeOfDay {\n  const d = typeof date === 'string' ? new Date(date) : date;\n  const hour = d.getHours();\n\n  if (hour >= 5 && hour < 12) return 'MORNING';\n  if (hour >= 12 && hour < 17) return 'AFTERNOON';\n  return 'EVENING';\n}\n\n/**\n * Get label for time of day\n *\n * @param timeOfDay - TimeOfDay enum value\n * @returns Human-readable label\n *\n * @example\n * ```tsx\n * const label = getTimeOfDayLabel('MORNING');\n * console.log(label); // 'Morning'\n * ```\n */\nexport function getTimeOfDayLabel(timeOfDay: TimeOfDay): string {\n  return TIME_OF_DAY_CONFIG[timeOfDay].label;\n}\n\n/**\n * Group events by time of day\n *\n * @param events - Array of events to group\n * @returns GroupedEvents object with morning, afternoon, and evening arrays\n *\n * @example\n * ```tsx\n * import { useUpcomingEvents, groupEventsByTimeOfDay } from '@growsober/sdk';\n *\n * function GatheringsList() {\n *   const { data: events } = useUpcomingEvents(30);\n *   const grouped = useMemo(() => {\n *     if (!events) return null;\n *     return groupEventsByTimeOfDay(events);\n *   }, [events]);\n *\n *   return (\n *     <SectionList\n *       sections={[\n *         { title: 'Morning', data: grouped?.morning || [] },\n *         { title: 'Afternoon', data: grouped?.afternoon || [] },\n *         { title: 'Evening', data: grouped?.evening || [] },\n *       ]}\n *       ...\n *     />\n *   );\n * }\n * ```\n */\nexport function groupEventsByTimeOfDay(events: EventResponse[]): GroupedEvents {\n  const grouped: GroupedEvents = {\n    morning: [],\n    afternoon: [],\n    evening: [],\n  };\n\n  for (const event of events) {\n    const timeOfDay = getTimeOfDay(event.startDate);\n    switch (timeOfDay) {\n      case 'MORNING':\n        grouped.morning.push(event);\n        break;\n      case 'AFTERNOON':\n        grouped.afternoon.push(event);\n        break;\n      case 'EVENING':\n        grouped.evening.push(event);\n        break;\n    }\n  }\n\n  return grouped;\n}\n\n/**\n * Convert grouped events to section list data format\n *\n * @param events - Array of events to group\n * @returns Array of sections with title and data\n *\n * @example\n * ```tsx\n * const sections = getEventSections(events);\n * // [\n * //   { title: 'Morning', data: [...] },\n * //   { title: 'Afternoon', data: [...] },\n * //   { title: 'Evening', data: [...] },\n * // ]\n * ```\n */\nexport function getEventSections(events: EventResponse[]): Array<{\n  title: string;\n  timeOfDay: TimeOfDay;\n  data: EventResponse[];\n}> {\n  const grouped = groupEventsByTimeOfDay(events);\n\n  return [\n    { title: 'Morning', timeOfDay: 'MORNING' as TimeOfDay, data: grouped.morning },\n    { title: 'Afternoon', timeOfDay: 'AFTERNOON' as TimeOfDay, data: grouped.afternoon },\n    { title: 'Evening', timeOfDay: 'EVENING' as TimeOfDay, data: grouped.evening },\n  ].filter(section => section.data.length > 0);\n}\n"]}
|
package/dist/index.d.ts
CHANGED
package/dist/index.js
CHANGED
|
@@ -64,4 +64,8 @@ __exportStar(require("./api/queries"), exports);
|
|
|
64
64
|
// MUTATION HOOKS
|
|
65
65
|
// ============================================================================
|
|
66
66
|
__exportStar(require("./api/mutations"), exports);
|
|
67
|
-
|
|
67
|
+
// ============================================================================
|
|
68
|
+
// UTILITIES
|
|
69
|
+
// ============================================================================
|
|
70
|
+
__exportStar(require("./api/utils/eventGrouping"), exports);
|
|
71
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0E4Qkc7Ozs7Ozs7Ozs7Ozs7Ozs7O0FBRUgsK0VBQStFO0FBQy9FLGFBQWE7QUFDYiwrRUFBK0U7QUFFL0UsdUNBQTBEO0FBQWpELHNHQUFBLFlBQVksT0FBQTtBQUFFLHNHQUFBLFlBQVksT0FBQTtBQUduQywrRUFBK0U7QUFDL0UsUUFBUTtBQUNSLCtFQUErRTtBQUUvRSw4Q0FBNEI7QUFFNUIsK0VBQStFO0FBQy9FLGNBQWM7QUFDZCwrRUFBK0U7QUFFL0UsZ0RBQThCO0FBRTlCLCtFQUErRTtBQUMvRSxpQkFBaUI7QUFDakIsK0VBQStFO0FBRS9FLGtEQUFnQztBQUVoQywrRUFBK0U7QUFDL0UsWUFBWTtBQUNaLCtFQUErRTtBQUUvRSw0REFBMEMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEdyb3dTb2JlciBTREtcbiAqXG4gKiBUeXBlU2NyaXB0IFNESyBmb3IgdGhlIEdyb3dTb2JlciBBUEkgd2l0aCBUYW5TdGFjayBRdWVyeSBob29rcy5cbiAqXG4gKiBAZXhhbXBsZVxuICogYGBgdHN4XG4gKiBpbXBvcnQgeyBjb25maWd1cmVTREssIHVzZUN1cnJlbnRVc2VyLCB1c2VMb2dpbiB9IGZyb20gJ0Bncm93c29iZXIvc2RrJztcbiAqXG4gKiAvLyBDb25maWd1cmUgU0RLIHdpdGggeW91ciBBUEkgYmFzZSBVUkwgYW5kIHRva2VuIG1hbmFnZW1lbnRcbiAqIGNvbmZpZ3VyZVNESyh7XG4gKiAgIGJhc2VVUkw6ICdodHRwczovL2FwaS5ncm93c29iZXIuYXBwJyxcbiAqICAgZ2V0QWNjZXNzVG9rZW46ICgpID0+IGxvY2FsU3RvcmFnZS5nZXRJdGVtKCdhY2Nlc3NUb2tlbicpLFxuICogICByZWZyZXNoQWNjZXNzVG9rZW46IGFzeW5jICgpID0+IHtcbiAqICAgICAvLyBZb3VyIHRva2VuIHJlZnJlc2ggbG9naWNcbiAqICAgICByZXR1cm4gbmV3QWNjZXNzVG9rZW47XG4gKiAgIH0sXG4gKiAgIG9uVW5hdXRob3JpemVkOiAoKSA9PiB7XG4gKiAgICAgLy8gSGFuZGxlIHVuYXV0aG9yaXplZCBhY2Nlc3NcbiAqICAgICB3aW5kb3cubG9jYXRpb24uaHJlZiA9ICcvbG9naW4nO1xuICogICB9LFxuICogfSk7XG4gKlxuICogLy8gVXNlIGhvb2tzIGluIHlvdXIgY29tcG9uZW50c1xuICogZnVuY3Rpb24gQXBwKCkge1xuICogICBjb25zdCB7IGRhdGE6IHVzZXIgfSA9IHVzZUN1cnJlbnRVc2VyKCk7XG4gKiAgIGNvbnN0IHsgbXV0YXRlOiBsb2dpbiB9ID0gdXNlTG9naW4oKTtcbiAqICAgLy8gLi4uXG4gKiB9XG4gKiBgYGBcbiAqL1xuXG4vLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG4vLyBBUEkgQ0xJRU5UXG4vLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG5cbmV4cG9ydCB7IGNvbmZpZ3VyZVNESywgZ2V0QXBpQ2xpZW50IH0gZnJvbSAnLi9hcGkvY2xpZW50JztcbmV4cG9ydCB0eXBlIHsgU0RLQ29uZmlnIH0gZnJvbSAnLi9hcGkvY2xpZW50JztcblxuLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuLy8gVFlQRVNcbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cblxuZXhwb3J0ICogZnJvbSAnLi9hcGkvdHlwZXMnO1xuXG4vLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG4vLyBRVUVSWSBIT09LU1xuLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuXG5leHBvcnQgKiBmcm9tICcuL2FwaS9xdWVyaWVzJztcblxuLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuLy8gTVVUQVRJT04gSE9PS1Ncbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cblxuZXhwb3J0ICogZnJvbSAnLi9hcGkvbXV0YXRpb25zJztcblxuLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuLy8gVVRJTElUSUVTXG4vLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG5cbmV4cG9ydCAqIGZyb20gJy4vYXBpL3V0aWxzL2V2ZW50R3JvdXBpbmcnO1xuXG4iXX0=
|
package/package.json
CHANGED
package/src/api/client.ts
CHANGED
|
@@ -42,9 +42,16 @@ function createApiClient(sdkConfig: SDKConfig): AxiosInstance {
|
|
|
42
42
|
(error) => Promise.reject(error)
|
|
43
43
|
);
|
|
44
44
|
|
|
45
|
-
// Response interceptor - handle 401
|
|
45
|
+
// Response interceptor - unwrap API response and handle 401
|
|
46
46
|
client.interceptors.response.use(
|
|
47
|
-
(response) =>
|
|
47
|
+
(response) => {
|
|
48
|
+
// API wraps all responses in {data: ..., meta: {...}}
|
|
49
|
+
// Unwrap to return just the data portion
|
|
50
|
+
if (response.data && typeof response.data === 'object' && 'data' in response.data) {
|
|
51
|
+
response.data = response.data.data;
|
|
52
|
+
}
|
|
53
|
+
return response;
|
|
54
|
+
},
|
|
48
55
|
async (error: AxiosError) => {
|
|
49
56
|
if (error.response?.status === 401) {
|
|
50
57
|
if (sdkConfig.refreshAccessToken) {
|
|
@@ -29,7 +29,7 @@ export interface CreatePinRequest {
|
|
|
29
29
|
activity: string;
|
|
30
30
|
venueId?: string;
|
|
31
31
|
scheduledTime?: string; // ISO date string, required for SCHEDULED type
|
|
32
|
-
duration?: number; //
|
|
32
|
+
duration?: number; // hours (1-8), default 2 for HERE_NOW, 2 for SCHEDULED
|
|
33
33
|
}
|
|
34
34
|
|
|
35
35
|
// ============================================================================
|
|
@@ -0,0 +1,181 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Event Grouping Utilities
|
|
3
|
+
*
|
|
4
|
+
* Utilities for organizing events by time of day.
|
|
5
|
+
* Used in the Gatherings screen to show events grouped by
|
|
6
|
+
* Morning, Afternoon, and Evening.
|
|
7
|
+
*
|
|
8
|
+
* @module api/utils/eventGrouping
|
|
9
|
+
*/
|
|
10
|
+
|
|
11
|
+
import { EventResponse } from '../types';
|
|
12
|
+
|
|
13
|
+
// ============================================================================
|
|
14
|
+
// TYPES
|
|
15
|
+
// ============================================================================
|
|
16
|
+
|
|
17
|
+
export type TimeOfDay = 'MORNING' | 'AFTERNOON' | 'EVENING';
|
|
18
|
+
|
|
19
|
+
export interface GroupedEvents {
|
|
20
|
+
morning: EventResponse[];
|
|
21
|
+
afternoon: EventResponse[];
|
|
22
|
+
evening: EventResponse[];
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
export interface TimeOfDayInfo {
|
|
26
|
+
key: TimeOfDay;
|
|
27
|
+
label: string;
|
|
28
|
+
emoji: string;
|
|
29
|
+
hours: { start: number; end: number };
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
// ============================================================================
|
|
33
|
+
// CONSTANTS
|
|
34
|
+
// ============================================================================
|
|
35
|
+
|
|
36
|
+
export const TIME_OF_DAY_CONFIG: Record<TimeOfDay, TimeOfDayInfo> = {
|
|
37
|
+
MORNING: {
|
|
38
|
+
key: 'MORNING',
|
|
39
|
+
label: 'Morning',
|
|
40
|
+
emoji: '',
|
|
41
|
+
hours: { start: 5, end: 11 }, // 5:00 AM - 11:59 AM
|
|
42
|
+
},
|
|
43
|
+
AFTERNOON: {
|
|
44
|
+
key: 'AFTERNOON',
|
|
45
|
+
label: 'Afternoon',
|
|
46
|
+
emoji: '',
|
|
47
|
+
hours: { start: 12, end: 16 }, // 12:00 PM - 4:59 PM
|
|
48
|
+
},
|
|
49
|
+
EVENING: {
|
|
50
|
+
key: 'EVENING',
|
|
51
|
+
label: 'Evening',
|
|
52
|
+
emoji: '',
|
|
53
|
+
hours: { start: 17, end: 4 }, // 5:00 PM - 4:59 AM (next day)
|
|
54
|
+
},
|
|
55
|
+
};
|
|
56
|
+
|
|
57
|
+
// ============================================================================
|
|
58
|
+
// FUNCTIONS
|
|
59
|
+
// ============================================================================
|
|
60
|
+
|
|
61
|
+
/**
|
|
62
|
+
* Get time of day for a given date
|
|
63
|
+
*
|
|
64
|
+
* @param date - Date string or Date object
|
|
65
|
+
* @returns TimeOfDay - MORNING, AFTERNOON, or EVENING
|
|
66
|
+
*
|
|
67
|
+
* @example
|
|
68
|
+
* ```tsx
|
|
69
|
+
* const timeOfDay = getTimeOfDay('2024-03-15T10:00:00Z');
|
|
70
|
+
* console.log(timeOfDay); // 'MORNING'
|
|
71
|
+
* ```
|
|
72
|
+
*/
|
|
73
|
+
export function getTimeOfDay(date: Date | string): TimeOfDay {
|
|
74
|
+
const d = typeof date === 'string' ? new Date(date) : date;
|
|
75
|
+
const hour = d.getHours();
|
|
76
|
+
|
|
77
|
+
if (hour >= 5 && hour < 12) return 'MORNING';
|
|
78
|
+
if (hour >= 12 && hour < 17) return 'AFTERNOON';
|
|
79
|
+
return 'EVENING';
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
/**
|
|
83
|
+
* Get label for time of day
|
|
84
|
+
*
|
|
85
|
+
* @param timeOfDay - TimeOfDay enum value
|
|
86
|
+
* @returns Human-readable label
|
|
87
|
+
*
|
|
88
|
+
* @example
|
|
89
|
+
* ```tsx
|
|
90
|
+
* const label = getTimeOfDayLabel('MORNING');
|
|
91
|
+
* console.log(label); // 'Morning'
|
|
92
|
+
* ```
|
|
93
|
+
*/
|
|
94
|
+
export function getTimeOfDayLabel(timeOfDay: TimeOfDay): string {
|
|
95
|
+
return TIME_OF_DAY_CONFIG[timeOfDay].label;
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
/**
|
|
99
|
+
* Group events by time of day
|
|
100
|
+
*
|
|
101
|
+
* @param events - Array of events to group
|
|
102
|
+
* @returns GroupedEvents object with morning, afternoon, and evening arrays
|
|
103
|
+
*
|
|
104
|
+
* @example
|
|
105
|
+
* ```tsx
|
|
106
|
+
* import { useUpcomingEvents, groupEventsByTimeOfDay } from '@growsober/sdk';
|
|
107
|
+
*
|
|
108
|
+
* function GatheringsList() {
|
|
109
|
+
* const { data: events } = useUpcomingEvents(30);
|
|
110
|
+
* const grouped = useMemo(() => {
|
|
111
|
+
* if (!events) return null;
|
|
112
|
+
* return groupEventsByTimeOfDay(events);
|
|
113
|
+
* }, [events]);
|
|
114
|
+
*
|
|
115
|
+
* return (
|
|
116
|
+
* <SectionList
|
|
117
|
+
* sections={[
|
|
118
|
+
* { title: 'Morning', data: grouped?.morning || [] },
|
|
119
|
+
* { title: 'Afternoon', data: grouped?.afternoon || [] },
|
|
120
|
+
* { title: 'Evening', data: grouped?.evening || [] },
|
|
121
|
+
* ]}
|
|
122
|
+
* ...
|
|
123
|
+
* />
|
|
124
|
+
* );
|
|
125
|
+
* }
|
|
126
|
+
* ```
|
|
127
|
+
*/
|
|
128
|
+
export function groupEventsByTimeOfDay(events: EventResponse[]): GroupedEvents {
|
|
129
|
+
const grouped: GroupedEvents = {
|
|
130
|
+
morning: [],
|
|
131
|
+
afternoon: [],
|
|
132
|
+
evening: [],
|
|
133
|
+
};
|
|
134
|
+
|
|
135
|
+
for (const event of events) {
|
|
136
|
+
const timeOfDay = getTimeOfDay(event.startDate);
|
|
137
|
+
switch (timeOfDay) {
|
|
138
|
+
case 'MORNING':
|
|
139
|
+
grouped.morning.push(event);
|
|
140
|
+
break;
|
|
141
|
+
case 'AFTERNOON':
|
|
142
|
+
grouped.afternoon.push(event);
|
|
143
|
+
break;
|
|
144
|
+
case 'EVENING':
|
|
145
|
+
grouped.evening.push(event);
|
|
146
|
+
break;
|
|
147
|
+
}
|
|
148
|
+
}
|
|
149
|
+
|
|
150
|
+
return grouped;
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
/**
|
|
154
|
+
* Convert grouped events to section list data format
|
|
155
|
+
*
|
|
156
|
+
* @param events - Array of events to group
|
|
157
|
+
* @returns Array of sections with title and data
|
|
158
|
+
*
|
|
159
|
+
* @example
|
|
160
|
+
* ```tsx
|
|
161
|
+
* const sections = getEventSections(events);
|
|
162
|
+
* // [
|
|
163
|
+
* // { title: 'Morning', data: [...] },
|
|
164
|
+
* // { title: 'Afternoon', data: [...] },
|
|
165
|
+
* // { title: 'Evening', data: [...] },
|
|
166
|
+
* // ]
|
|
167
|
+
* ```
|
|
168
|
+
*/
|
|
169
|
+
export function getEventSections(events: EventResponse[]): Array<{
|
|
170
|
+
title: string;
|
|
171
|
+
timeOfDay: TimeOfDay;
|
|
172
|
+
data: EventResponse[];
|
|
173
|
+
}> {
|
|
174
|
+
const grouped = groupEventsByTimeOfDay(events);
|
|
175
|
+
|
|
176
|
+
return [
|
|
177
|
+
{ title: 'Morning', timeOfDay: 'MORNING' as TimeOfDay, data: grouped.morning },
|
|
178
|
+
{ title: 'Afternoon', timeOfDay: 'AFTERNOON' as TimeOfDay, data: grouped.afternoon },
|
|
179
|
+
{ title: 'Evening', timeOfDay: 'EVENING' as TimeOfDay, data: grouped.evening },
|
|
180
|
+
].filter(section => section.data.length > 0);
|
|
181
|
+
}
|
package/src/index.ts
CHANGED
|
@@ -55,3 +55,9 @@ export * from './api/queries';
|
|
|
55
55
|
|
|
56
56
|
export * from './api/mutations';
|
|
57
57
|
|
|
58
|
+
// ============================================================================
|
|
59
|
+
// UTILITIES
|
|
60
|
+
// ============================================================================
|
|
61
|
+
|
|
62
|
+
export * from './api/utils/eventGrouping';
|
|
63
|
+
|