@spotsdev/sdk 1.2.0 → 1.3.3
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/entities.d.ts +8 -0
- package/dist/api/entities.js +1 -1
- package/dist/api/mutations/index.d.ts +1 -0
- package/dist/api/mutations/index.js +3 -1
- package/dist/api/mutations/posts.js +4 -3
- package/dist/api/mutations/redemptions.d.ts +33 -0
- package/dist/api/mutations/redemptions.js +63 -0
- package/dist/api/queries/index.d.ts +2 -0
- package/dist/api/queries/index.js +6 -2
- package/dist/api/queries/wallet.d.ts +109 -0
- package/dist/api/queries/wallet.js +136 -0
- package/package.json +2 -2
- package/src/api/entities.ts +9 -0
- package/src/api/mutations/index.ts +3 -0
- package/src/api/mutations/posts.ts +3 -2
- package/src/api/mutations/redemptions.ts +91 -0
- package/src/api/queries/index.ts +4 -0
- package/src/api/queries/wallet.ts +240 -0
package/dist/api/entities.d.ts
CHANGED
|
@@ -12,6 +12,13 @@ export type ResponseStatus = 'PENDING' | 'ACCEPTED' | 'DECLINED';
|
|
|
12
12
|
export type ReportReason = 'SPAM' | 'HARASSMENT' | 'INAPPROPRIATE' | 'SCAM' | 'OTHER';
|
|
13
13
|
export type ReportStatus = 'PENDING' | 'REVIEWED' | 'RESOLVED' | 'DISMISSED';
|
|
14
14
|
export type NotificationType = 'POST_RESPONSE' | 'POST_REPLY' | 'POST_UPVOTE' | 'RESPONSE_ACCEPTED' | 'RESPONSE_DECLINED' | 'NEW_MESSAGE' | 'SYSTEM';
|
|
15
|
+
export interface UserVibe {
|
|
16
|
+
id: string;
|
|
17
|
+
name: string;
|
|
18
|
+
emoji: string | null;
|
|
19
|
+
slug?: string;
|
|
20
|
+
category?: string;
|
|
21
|
+
}
|
|
15
22
|
export interface User {
|
|
16
23
|
id: string;
|
|
17
24
|
phone: string | null;
|
|
@@ -21,6 +28,7 @@ export interface User {
|
|
|
21
28
|
avatarUrl: string | null;
|
|
22
29
|
status: AccountStatus;
|
|
23
30
|
onboardingCompleted: boolean;
|
|
31
|
+
vibes?: UserVibe[];
|
|
24
32
|
lastActiveAt: Date | null;
|
|
25
33
|
createdAt: Date;
|
|
26
34
|
updatedAt: Date;
|
package/dist/api/entities.js
CHANGED
|
@@ -6,4 +6,4 @@
|
|
|
6
6
|
* This allows the SDK to work in React Native where Prisma is not available.
|
|
7
7
|
*/
|
|
8
8
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
9
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"entities.js","sourceRoot":"","sources":["../../src/api/entities.ts"],"names":[],"mappings":";AAAA;;;;;GAKG","sourcesContent":["/**\n * Spots SDK Entity Types\n *\n * These types mirror the Prisma schema entities but without @prisma/client dependency.\n * This allows the SDK to work in React Native where Prisma is not available.\n */\n\n// ============================================================================\n// ENUMS\n// ============================================================================\n\nexport type SpotType =\n  | 'CAFE'\n  | 'RESTAURANT'\n  | 'BAR'\n  | 'PARK'\n  | 'BEACH'\n  | 'COWORKING'\n  | 'LIBRARY'\n  | 'GYM'\n  | 'MUSEUM'\n  | 'GALLERY'\n  | 'THEATER'\n  | 'CINEMA'\n  | 'MALL'\n  | 'MARKET'\n  | 'HOTEL'\n  | 'HOSTEL'\n  | 'OTHER'\n\nexport type SpotPostType =\n  | 'CHECK_IN'\n  | 'LOOKING_FOR'\n  | 'OFFERING'\n  | 'QUESTION'\n  | 'EVENT'\n  | 'RECOMMENDATION'\n  | 'GENERAL'\n\nexport type AccountStatus = 'ACTIVE' | 'SUSPENDED' | 'DELETED' | 'PENDING'\n\nexport type ClaimStatus = 'PENDING' | 'APPROVED' | 'REJECTED'\n\nexport type ResponseStatus = 'PENDING' | 'ACCEPTED' | 'DECLINED'\n\nexport type ReportReason =\n  | 'SPAM'\n  | 'HARASSMENT'\n  | 'INAPPROPRIATE'\n  | 'SCAM'\n  | 'OTHER'\n\nexport type ReportStatus = 'PENDING' | 'REVIEWED' | 'RESOLVED' | 'DISMISSED'\n\nexport type NotificationType =\n  | 'POST_RESPONSE'\n  | 'POST_REPLY'\n  | 'POST_UPVOTE'\n  | 'RESPONSE_ACCEPTED'\n  | 'RESPONSE_DECLINED'\n  | 'NEW_MESSAGE'\n  | 'SYSTEM'\n\n// ============================================================================\n// ENTITY INTERFACES\n// ============================================================================\n\nexport interface User {\n  id: string\n  phone: string | null\n  email: string | null\n  name: string | null\n  bio: string | null\n  avatarUrl: string | null\n  status: AccountStatus\n  onboardingCompleted: boolean\n  lastActiveAt: Date | null\n  createdAt: Date\n  updatedAt: Date\n}\n\nexport interface Spot {\n  id: string\n  name: string\n  slug: string\n  description: string | null\n  address: string | null\n  city: string | null\n  country: string | null\n  latitude: number | null\n  longitude: number | null\n  // Aliases for backwards compatibility\n  lat?: number | null\n  lng?: number | null\n  type: SpotType\n  googlePlaceId: string | null\n  website: string | null\n  phone: string | null\n  imageUrl: string | null\n  ownerId: string | null\n  isVerified: boolean\n  // Dates as strings for JSON serialization compatibility\n  createdAt: Date | string\n  updatedAt: Date | string\n}\n\nexport interface SpotPost {\n  id: string\n  spotId: string\n  userId: string\n  postType: SpotPostType\n  title: string\n  content: string | null\n  emoji: string | null\n  expiresAt: Date | null\n  status: 'ACTIVE' | 'FULFILLED' | 'EXPIRED' | 'DELETED'\n  upvoteCount: number\n  responseCount: number\n  replyCount: number\n  createdAt: Date\n  updatedAt: Date\n  // Relations (optional, populated by API)\n  user?: User\n  spot?: Spot\n}\n\nexport interface PostReply {\n  id: string\n  postId: string\n  userId: string\n  content: string\n  createdAt: Date\n  updatedAt: Date\n  user?: User\n}\n\nexport interface PostResponse {\n  id: string\n  postId: string\n  userId: string\n  note: string | null\n  status: ResponseStatus\n  createdAt: Date\n  updatedAt: Date\n  user?: User\n  post?: SpotPost\n}\n\nexport interface PostUpvote {\n  id: string\n  postId: string\n  userId: string\n  createdAt: Date\n}\n\n/**\n * PostUpvoter - user who upvoted a post (from /posts/:id/upvotes endpoint)\n */\nexport interface PostUpvoter {\n  id: string\n  userId: string\n  name: string | null\n  avatarUrl: string | null\n  createdAt: Date\n}\n\n/**\n * PostUpvotesResponse - response from /posts/:id/upvotes endpoint\n */\nexport interface PostUpvotesResponse {\n  data: PostUpvoter[]\n  meta: {\n    total: number\n    limit: number\n    offset: number\n    hasMore: boolean\n    currentUserUpvoted: boolean\n    isPostAuthor: boolean\n  }\n}\n\nexport interface PostReport {\n  id: string\n  postId: string\n  userId: string\n  reason: ReportReason\n  details: string | null\n  status: ReportStatus\n  createdAt: Date\n  updatedAt: Date\n}\n\nexport interface Club {\n  id: string\n  name: string\n  slug: string\n  description: string | null\n  imageUrl: string | null\n  isPrivate: boolean\n  createdById: string\n  createdAt: Date\n  updatedAt: Date\n}\n\nexport interface ClubMember {\n  id: string\n  clubId: string\n  userId: string\n  role: 'MEMBER' | 'ADMIN' | 'OWNER'\n  joinedAt: Date\n}\n\nexport interface Conversation {\n  id: string\n  createdAt: Date\n  updatedAt: Date\n  participants?: User[]\n  messages?: Message[]\n}\n\nexport interface Message {\n  id: string\n  conversationId: string\n  senderId: string\n  content: string\n  createdAt: Date\n  sender?: User\n}\n\nexport interface Notification {\n  id: string\n  userId: string\n  type: NotificationType\n  title: string\n  body: string | null\n  data: Record<string, unknown> | null\n  isRead: boolean\n  createdAt: Date\n}\n\nexport interface PostTemplate {\n  id: string\n  postType: SpotPostType\n  emoji: string\n  title: string\n  placeholder: string | null\n  isActive: boolean\n  sortOrder: number\n  createdAt: Date\n  updatedAt: Date\n}\n\nexport interface Vibe {\n  id: string\n  name: string\n  emoji: string | null\n  createdAt: Date\n}\n\nexport interface Interest {\n  id: string\n  name: string\n  emoji: string | null\n  category: string | null\n  createdAt: Date\n}\n\nexport interface Intention {\n  id: string\n  name: string\n  emoji: string | null\n  createdAt: Date\n}\n\nexport interface LifeSituation {\n  id: string\n  name: string\n  emoji: string | null\n  createdAt: Date\n}\n\nexport interface FavoriteSpot {\n  id: string\n  userId: string\n  spotId: string\n  createdAt: Date\n  spot?: Spot\n}\n\nexport interface SpotImage {\n  id: string\n  spotId: string\n  url: string\n  caption: string | null\n  sortOrder: number\n  createdAt: Date\n}\n\nexport interface SpotVibe {\n  id: string\n  spotId: string\n  vibeId: string\n  vibe?: Vibe\n}\n\nexport interface SpotIntention {\n  id: string\n  spotId: string\n  intentionId: string\n  intention?: Intention\n}\n\nexport interface SpotClaim {\n  id: string\n  spotId: string\n  userId: string\n  status: ClaimStatus\n  verificationMethod: string | null\n  createdAt: Date\n  updatedAt: Date\n}\n\nexport interface SpotSubscription {\n  id: string\n  spotId: string\n  userId: string\n  createdAt: Date\n  spot?: Spot\n}\n\nexport interface City {\n  id: string\n  name: string\n  country: string\n  latitude: number\n  longitude: number\n  timezone: string | null\n  createdAt: Date\n}\n\n// ============================================================================\n// API RESPONSE TYPES\n// ============================================================================\n\n// NOTE: ApiResponse is re-exported from @spots/types in types.ts\n\n/**\n * Paginated response with meta information\n */\nexport interface PaginatedResponse<T> {\n  data: T[]\n  meta: PaginationMeta\n}\n\nexport interface PaginationMeta {\n  total: number\n  page: number\n  limit: number\n  totalPages: number\n  hasNextPage: boolean\n  hasPreviousPage: boolean\n}\n\n// ============================================================================\n// MARKETPLACE TYPES\n// ============================================================================\n\nexport type ProductType = 'PHYSICAL' | 'DIGITAL' | 'SERVICE' | 'BUNDLE'\n\nexport type ProductStatus = 'DRAFT' | 'ACTIVE' | 'SOLD_OUT' | 'ARCHIVED'\n\nexport type OrderStatus =\n  | 'PENDING'\n  | 'CONFIRMED'\n  | 'PREPARING'\n  | 'READY'\n  | 'COMPLETED'\n  | 'CANCELLED'\n  | 'REFUNDED'\n\nexport interface Product {\n  id: string\n  spotId: string\n  name: string\n  slug: string\n  description: string | null\n  type: ProductType\n  status: ProductStatus\n  price: number\n  currency: string\n  imageUrl: string | null\n  images: string[]\n  stock: number | null\n  maxQuantity: number\n  sortOrder: number\n  metadata: Record<string, unknown> | null\n  createdAt: Date\n  updatedAt: Date\n}\n\nexport interface Order {\n  id: string\n  spotId: string\n  userId: string\n  status: OrderStatus\n  subtotal: number\n  tax: number\n  total: number\n  currency: string\n  notes: string | null\n  metadata: Record<string, unknown> | null\n  createdAt: Date\n  updatedAt: Date\n}\n\nexport interface OrderItem {\n  id: string\n  orderId: string\n  productId: string\n  quantity: number\n  unitPrice: number\n  total: number\n  createdAt: Date\n}\n"]}
|
|
9
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"entities.js","sourceRoot":"","sources":["../../src/api/entities.ts"],"names":[],"mappings":";AAAA;;;;;GAKG","sourcesContent":["/**\n * Spots SDK Entity Types\n *\n * These types mirror the Prisma schema entities but without @prisma/client dependency.\n * This allows the SDK to work in React Native where Prisma is not available.\n */\n\n// ============================================================================\n// ENUMS\n// ============================================================================\n\nexport type SpotType =\n  | 'CAFE'\n  | 'RESTAURANT'\n  | 'BAR'\n  | 'PARK'\n  | 'BEACH'\n  | 'COWORKING'\n  | 'LIBRARY'\n  | 'GYM'\n  | 'MUSEUM'\n  | 'GALLERY'\n  | 'THEATER'\n  | 'CINEMA'\n  | 'MALL'\n  | 'MARKET'\n  | 'HOTEL'\n  | 'HOSTEL'\n  | 'OTHER'\n\nexport type SpotPostType =\n  | 'CHECK_IN'\n  | 'LOOKING_FOR'\n  | 'OFFERING'\n  | 'QUESTION'\n  | 'EVENT'\n  | 'RECOMMENDATION'\n  | 'GENERAL'\n\nexport type AccountStatus = 'ACTIVE' | 'SUSPENDED' | 'DELETED' | 'PENDING'\n\nexport type ClaimStatus = 'PENDING' | 'APPROVED' | 'REJECTED'\n\nexport type ResponseStatus = 'PENDING' | 'ACCEPTED' | 'DECLINED'\n\nexport type ReportReason =\n  | 'SPAM'\n  | 'HARASSMENT'\n  | 'INAPPROPRIATE'\n  | 'SCAM'\n  | 'OTHER'\n\nexport type ReportStatus = 'PENDING' | 'REVIEWED' | 'RESOLVED' | 'DISMISSED'\n\nexport type NotificationType =\n  | 'POST_RESPONSE'\n  | 'POST_REPLY'\n  | 'POST_UPVOTE'\n  | 'RESPONSE_ACCEPTED'\n  | 'RESPONSE_DECLINED'\n  | 'NEW_MESSAGE'\n  | 'SYSTEM'\n\n// ============================================================================\n// ENTITY INTERFACES\n// ============================================================================\n\nexport interface UserVibe {\n  id: string\n  name: string\n  emoji: string | null\n  slug?: string\n  category?: string\n}\n\nexport interface User {\n  id: string\n  phone: string | null\n  email: string | null\n  name: string | null\n  bio: string | null\n  avatarUrl: string | null\n  status: AccountStatus\n  onboardingCompleted: boolean\n  vibes?: UserVibe[]\n  lastActiveAt: Date | null\n  createdAt: Date\n  updatedAt: Date\n}\n\nexport interface Spot {\n  id: string\n  name: string\n  slug: string\n  description: string | null\n  address: string | null\n  city: string | null\n  country: string | null\n  latitude: number | null\n  longitude: number | null\n  // Aliases for backwards compatibility\n  lat?: number | null\n  lng?: number | null\n  type: SpotType\n  googlePlaceId: string | null\n  website: string | null\n  phone: string | null\n  imageUrl: string | null\n  ownerId: string | null\n  isVerified: boolean\n  // Dates as strings for JSON serialization compatibility\n  createdAt: Date | string\n  updatedAt: Date | string\n}\n\nexport interface SpotPost {\n  id: string\n  spotId: string\n  userId: string\n  postType: SpotPostType\n  title: string\n  content: string | null\n  emoji: string | null\n  expiresAt: Date | null\n  status: 'ACTIVE' | 'FULFILLED' | 'EXPIRED' | 'DELETED'\n  upvoteCount: number\n  responseCount: number\n  replyCount: number\n  createdAt: Date\n  updatedAt: Date\n  // Relations (optional, populated by API)\n  user?: User\n  spot?: Spot\n}\n\nexport interface PostReply {\n  id: string\n  postId: string\n  userId: string\n  content: string\n  createdAt: Date\n  updatedAt: Date\n  user?: User\n}\n\nexport interface PostResponse {\n  id: string\n  postId: string\n  userId: string\n  note: string | null\n  status: ResponseStatus\n  createdAt: Date\n  updatedAt: Date\n  user?: User\n  post?: SpotPost\n}\n\nexport interface PostUpvote {\n  id: string\n  postId: string\n  userId: string\n  createdAt: Date\n}\n\n/**\n * PostUpvoter - user who upvoted a post (from /posts/:id/upvotes endpoint)\n */\nexport interface PostUpvoter {\n  id: string\n  userId: string\n  name: string | null\n  avatarUrl: string | null\n  createdAt: Date\n}\n\n/**\n * PostUpvotesResponse - response from /posts/:id/upvotes endpoint\n */\nexport interface PostUpvotesResponse {\n  data: PostUpvoter[]\n  meta: {\n    total: number\n    limit: number\n    offset: number\n    hasMore: boolean\n    currentUserUpvoted: boolean\n    isPostAuthor: boolean\n  }\n}\n\nexport interface PostReport {\n  id: string\n  postId: string\n  userId: string\n  reason: ReportReason\n  details: string | null\n  status: ReportStatus\n  createdAt: Date\n  updatedAt: Date\n}\n\nexport interface Club {\n  id: string\n  name: string\n  slug: string\n  description: string | null\n  imageUrl: string | null\n  isPrivate: boolean\n  createdById: string\n  createdAt: Date\n  updatedAt: Date\n}\n\nexport interface ClubMember {\n  id: string\n  clubId: string\n  userId: string\n  role: 'MEMBER' | 'ADMIN' | 'OWNER'\n  joinedAt: Date\n}\n\nexport interface Conversation {\n  id: string\n  createdAt: Date\n  updatedAt: Date\n  participants?: User[]\n  messages?: Message[]\n}\n\nexport interface Message {\n  id: string\n  conversationId: string\n  senderId: string\n  content: string\n  createdAt: Date\n  sender?: User\n}\n\nexport interface Notification {\n  id: string\n  userId: string\n  type: NotificationType\n  title: string\n  body: string | null\n  data: Record<string, unknown> | null\n  isRead: boolean\n  createdAt: Date\n}\n\nexport interface PostTemplate {\n  id: string\n  postType: SpotPostType\n  emoji: string\n  title: string\n  placeholder: string | null\n  isActive: boolean\n  sortOrder: number\n  createdAt: Date\n  updatedAt: Date\n}\n\nexport interface Vibe {\n  id: string\n  name: string\n  emoji: string | null\n  createdAt: Date\n}\n\nexport interface Interest {\n  id: string\n  name: string\n  emoji: string | null\n  category: string | null\n  createdAt: Date\n}\n\nexport interface Intention {\n  id: string\n  name: string\n  emoji: string | null\n  createdAt: Date\n}\n\nexport interface LifeSituation {\n  id: string\n  name: string\n  emoji: string | null\n  createdAt: Date\n}\n\nexport interface FavoriteSpot {\n  id: string\n  userId: string\n  spotId: string\n  createdAt: Date\n  spot?: Spot\n}\n\nexport interface SpotImage {\n  id: string\n  spotId: string\n  url: string\n  caption: string | null\n  sortOrder: number\n  createdAt: Date\n}\n\nexport interface SpotVibe {\n  id: string\n  spotId: string\n  vibeId: string\n  vibe?: Vibe\n}\n\nexport interface SpotIntention {\n  id: string\n  spotId: string\n  intentionId: string\n  intention?: Intention\n}\n\nexport interface SpotClaim {\n  id: string\n  spotId: string\n  userId: string\n  status: ClaimStatus\n  verificationMethod: string | null\n  createdAt: Date\n  updatedAt: Date\n}\n\nexport interface SpotSubscription {\n  id: string\n  spotId: string\n  userId: string\n  createdAt: Date\n  spot?: Spot\n}\n\nexport interface City {\n  id: string\n  name: string\n  country: string\n  latitude: number\n  longitude: number\n  timezone: string | null\n  createdAt: Date\n}\n\n// ============================================================================\n// API RESPONSE TYPES\n// ============================================================================\n\n// NOTE: ApiResponse is re-exported from @spots/types in types.ts\n\n/**\n * Paginated response with meta information\n */\nexport interface PaginatedResponse<T> {\n  data: T[]\n  meta: PaginationMeta\n}\n\nexport interface PaginationMeta {\n  total: number\n  page: number\n  limit: number\n  totalPages: number\n  hasNextPage: boolean\n  hasPreviousPage: boolean\n}\n\n// ============================================================================\n// MARKETPLACE TYPES\n// ============================================================================\n\nexport type ProductType = 'PHYSICAL' | 'DIGITAL' | 'SERVICE' | 'BUNDLE'\n\nexport type ProductStatus = 'DRAFT' | 'ACTIVE' | 'SOLD_OUT' | 'ARCHIVED'\n\nexport type OrderStatus =\n  | 'PENDING'\n  | 'CONFIRMED'\n  | 'PREPARING'\n  | 'READY'\n  | 'COMPLETED'\n  | 'CANCELLED'\n  | 'REFUNDED'\n\nexport interface Product {\n  id: string\n  spotId: string\n  name: string\n  slug: string\n  description: string | null\n  type: ProductType\n  status: ProductStatus\n  price: number\n  currency: string\n  imageUrl: string | null\n  images: string[]\n  stock: number | null\n  maxQuantity: number\n  sortOrder: number\n  metadata: Record<string, unknown> | null\n  createdAt: Date\n  updatedAt: Date\n}\n\nexport interface Order {\n  id: string\n  spotId: string\n  userId: string\n  status: OrderStatus\n  subtotal: number\n  tax: number\n  total: number\n  currency: string\n  notes: string | null\n  metadata: Record<string, unknown> | null\n  createdAt: Date\n  updatedAt: Date\n}\n\nexport interface OrderItem {\n  id: string\n  orderId: string\n  productId: string\n  quantity: number\n  unitPrice: number\n  total: number\n  createdAt: Date\n}\n"]}
|
|
@@ -35,4 +35,6 @@ __exportStar(require("./notifications"), exports);
|
|
|
35
35
|
__exportStar(require("./products"), exports);
|
|
36
36
|
// Orders (marketplace)
|
|
37
37
|
__exportStar(require("./orders"), exports);
|
|
38
|
-
|
|
38
|
+
// Redemptions (wallet)
|
|
39
|
+
__exportStar(require("./redemptions"), exports);
|
|
40
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvYXBpL211dGF0aW9ucy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUE7Ozs7R0FJRzs7Ozs7Ozs7Ozs7Ozs7OztBQUVILFFBQVE7QUFDUiwwQ0FBdUI7QUFFdkIsUUFBUTtBQUNSLDBDQUF1QjtBQUV2QixRQUFRO0FBQ1IsMENBQXVCO0FBRXZCLGdCQUFnQjtBQUNoQixrREFBK0I7QUFFL0IsUUFBUTtBQUNSLDBDQUF1QjtBQUV2QixnQkFBZ0I7QUFDaEIsa0RBQStCO0FBRS9CLHlCQUF5QjtBQUN6Qiw2Q0FBMEI7QUFFMUIsdUJBQXVCO0FBQ3ZCLDJDQUF3QjtBQUV4Qix1QkFBdUI7QUFDdkIsZ0RBQTZCIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBTcG90cyBTREsgTXV0YXRpb24gSG9va3MgSW5kZXhcbiAqXG4gKiBSZS1leHBvcnRzIGFsbCBtdXRhdGlvbiBob29rcy5cbiAqL1xuXG4vLyBVc2Vyc1xuZXhwb3J0ICogZnJvbSAnLi91c2VycydcblxuLy8gUG9zdHNcbmV4cG9ydCAqIGZyb20gJy4vcG9zdHMnXG5cbi8vIFNwb3RzXG5leHBvcnQgKiBmcm9tICcuL3Nwb3RzJ1xuXG4vLyBDb252ZXJzYXRpb25zXG5leHBvcnQgKiBmcm9tICcuL2NvbnZlcnNhdGlvbnMnXG5cbi8vIENsdWJzXG5leHBvcnQgKiBmcm9tICcuL2NsdWJzJ1xuXG4vLyBOb3RpZmljYXRpb25zXG5leHBvcnQgKiBmcm9tICcuL25vdGlmaWNhdGlvbnMnXG5cbi8vIFByb2R1Y3RzIChtYXJrZXRwbGFjZSlcbmV4cG9ydCAqIGZyb20gJy4vcHJvZHVjdHMnXG5cbi8vIE9yZGVycyAobWFya2V0cGxhY2UpXG5leHBvcnQgKiBmcm9tICcuL29yZGVycydcblxuLy8gUmVkZW1wdGlvbnMgKHdhbGxldClcbmV4cG9ydCAqIGZyb20gJy4vcmVkZW1wdGlvbnMnXG4iXX0=
|
|
@@ -51,9 +51,10 @@ function useCreatePost(options) {
|
|
|
51
51
|
function useUpdatePost(options) {
|
|
52
52
|
const queryClient = (0, react_query_1.useQueryClient)();
|
|
53
53
|
return (0, react_query_1.useMutation)({
|
|
54
|
-
mutationFn: async ({ postId,
|
|
54
|
+
mutationFn: async ({ postId, title, content }) => {
|
|
55
55
|
const client = (0, client_1.getApiClient)();
|
|
56
|
-
|
|
56
|
+
// API expects 'description' field, not 'content'
|
|
57
|
+
const response = await client.put(`/posts/${postId}`, { title, description: content });
|
|
57
58
|
return response.data.data;
|
|
58
59
|
},
|
|
59
60
|
onSuccess: (data, variables) => {
|
|
@@ -238,4 +239,4 @@ function useUpdatePostStatus(options) {
|
|
|
238
239
|
...options,
|
|
239
240
|
});
|
|
240
241
|
}
|
|
241
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"posts.js","sourceRoot":"","sources":["../../../src/api/mutations/posts.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Posts Mutation Hooks\n *\n * TanStack Query hooks for post/board mutation operations.\n */\n\nimport {\n  useMutation,\n  type UseMutationOptions,\n  type UseMutationResult,\n  useQueryClient,\n} from '@tanstack/react-query'\n\nimport {getApiClient} from '../client'\nimport {postKeys} from '../queries/posts'\nimport {\n  type ApiResponse,\n  type CreatePostRequest,\n  type CreateReplyRequest,\n  type Post,\n  type PostResponse,\n  type PostStatusDto,\n  type RespondToPostRequest,\n  type UpdateResponseRequest,\n  type UpvoteResponse,\n} from '../types'\n\n// ============================================================================\n// MUTATION HOOKS\n// ============================================================================\n\n/**\n * Create a new post\n *\n * @endpoint POST /spots/{spotId}/posts\n */\nexport function useCreatePost(\n  options?: Omit<\n    UseMutationOptions<Post, Error, {spotId: string} & CreatePostRequest>,\n    'mutationFn'\n  >,\n): UseMutationResult<Post, Error, {spotId: string} & CreatePostRequest> {\n  const queryClient = useQueryClient()\n\n  return useMutation({\n    mutationFn: async ({spotId, ...data}): Promise<Post> => {\n      const client = getApiClient()\n      const response = await client.post<ApiResponse<Post>>(\n        `/spots/${spotId}/posts`,\n        data,\n      )\n      return response.data.data\n    },\n    onSuccess: (_, variables) => {\n      queryClient.invalidateQueries({\n        queryKey: postKeys.bySpot(variables.spotId),\n      })\n    },\n    ...options,\n  })\n}\n\n/**\n * Update a post\n *\n * @endpoint PUT /posts/{postId}\n */\nexport function useUpdatePost(\n  options?: Omit<\n    UseMutationOptions<\n      Post,\n      Error,\n      {postId: string; title?: string; content?: string}\n    >,\n    'mutationFn'\n  >,\n): UseMutationResult<\n  Post,\n  Error,\n  {postId: string; title?: string; content?: string}\n> {\n  const queryClient = useQueryClient()\n\n  return useMutation({\n    mutationFn: async ({postId, ...data}): Promise<Post> => {\n      const client = getApiClient()\n      const response = await client.put<ApiResponse<Post>>(\n        `/posts/${postId}`,\n        data,\n      )\n      return response.data.data\n    },\n    onSuccess: (data, variables) => {\n      queryClient.setQueryData(postKeys.detail(variables.postId), data)\n    },\n    ...options,\n  })\n}\n\n/**\n * Delete a post\n *\n * @endpoint DELETE /posts/{postId}\n */\nexport function useDeletePost(\n  options?: Omit<UseMutationOptions<void, Error, string>, 'mutationFn'>,\n): UseMutationResult<void, Error, string> {\n  const queryClient = useQueryClient()\n\n  return useMutation({\n    mutationFn: async (postId: string): Promise<void> => {\n      const client = getApiClient()\n      await client.delete(`/posts/${postId}`)\n    },\n    onSuccess: (_, postId) => {\n      queryClient.invalidateQueries({queryKey: postKeys.detail(postId)})\n      queryClient.invalidateQueries({queryKey: postKeys.lists()})\n    },\n    ...options,\n  })\n}\n\n/**\n * Fulfill a post (mark as fulfilled)\n *\n * @endpoint PUT /posts/{postId}/fulfill\n */\nexport function useFulfillPost(\n  options?: Omit<UseMutationOptions<Post, Error, string>, 'mutationFn'>,\n): UseMutationResult<Post, Error, string> {\n  const queryClient = useQueryClient()\n\n  return useMutation({\n    mutationFn: async (postId: string): Promise<Post> => {\n      const client = getApiClient()\n      const response = await client.put<ApiResponse<Post>>(\n        `/posts/${postId}/fulfill`,\n      )\n      return response.data.data\n    },\n    onSuccess: (data, postId) => {\n      queryClient.setQueryData(postKeys.detail(postId), data)\n    },\n    ...options,\n  })\n}\n\n/**\n * Upvote/unupvote a post\n *\n * @endpoint POST /posts/{postId}/upvote\n */\nexport function useUpvotePost(\n  options?: Omit<\n    UseMutationOptions<UpvoteResponse, Error, string>,\n    'mutationFn'\n  >,\n): UseMutationResult<UpvoteResponse, Error, string> {\n  const queryClient = useQueryClient()\n\n  return useMutation({\n    mutationFn: async (postId: string): Promise<UpvoteResponse> => {\n      const client = getApiClient()\n      const response = await client.post<ApiResponse<UpvoteResponse>>(\n        `/posts/${postId}/upvote`,\n      )\n      return response.data.data\n    },\n    onSuccess: (_, postId) => {\n      queryClient.invalidateQueries({queryKey: postKeys.detail(postId)})\n    },\n    ...options,\n  })\n}\n\n/**\n * Create a reply to a post\n *\n * @endpoint POST /posts/{postId}/reply\n */\nexport function useCreateReply(\n  options?: Omit<\n    UseMutationOptions<unknown, Error, {postId: string} & CreateReplyRequest>,\n    'mutationFn'\n  >,\n): UseMutationResult<unknown, Error, {postId: string} & CreateReplyRequest> {\n  const queryClient = useQueryClient()\n\n  return useMutation({\n    mutationFn: async ({postId, ...data}) => {\n      const client = getApiClient()\n      const response = await client.post<ApiResponse<unknown>>(\n        `/posts/${postId}/reply`,\n        data,\n      )\n      return response.data.data\n    },\n    onSuccess: (_, variables) => {\n      queryClient.invalidateQueries({\n        queryKey: postKeys.detail(variables.postId),\n      })\n    },\n    ...options,\n  })\n}\n\n/**\n * Delete a reply\n *\n * @endpoint DELETE /replies/{replyId}\n */\nexport function useDeleteReply(\n  options?: Omit<\n    UseMutationOptions<void, Error, {replyId: string; postId: string}>,\n    'mutationFn'\n  >,\n): UseMutationResult<void, Error, {replyId: string; postId: string}> {\n  const queryClient = useQueryClient()\n\n  return useMutation({\n    mutationFn: async ({replyId}): Promise<void> => {\n      const client = getApiClient()\n      await client.delete(`/replies/${replyId}`)\n    },\n    onSuccess: (_, variables) => {\n      queryClient.invalidateQueries({\n        queryKey: postKeys.detail(variables.postId),\n      })\n    },\n    ...options,\n  })\n}\n\n/**\n * Respond to a post (express interest)\n *\n * @endpoint POST /posts/{postId}/respond\n */\nexport function useRespondToPost(\n  options?: Omit<\n    UseMutationOptions<\n      PostResponse,\n      Error,\n      {postId: string} & RespondToPostRequest\n    >,\n    'mutationFn'\n  >,\n): UseMutationResult<\n  PostResponse,\n  Error,\n  {postId: string} & RespondToPostRequest\n> {\n  const queryClient = useQueryClient()\n\n  return useMutation({\n    mutationFn: async ({postId, ...data}): Promise<PostResponse> => {\n      const client = getApiClient()\n      const response = await client.post<ApiResponse<PostResponse>>(\n        `/posts/${postId}/respond`,\n        data,\n      )\n      return response.data.data\n    },\n    onSuccess: (_, variables) => {\n      queryClient.invalidateQueries({\n        queryKey: postKeys.detail(variables.postId),\n      })\n      queryClient.invalidateQueries({\n        queryKey: postKeys.responses(variables.postId),\n      })\n    },\n    ...options,\n  })\n}\n\n/**\n * Accept or decline a response\n *\n * @endpoint PUT /responses/{responseId}\n */\nexport function useUpdateResponse(\n  options?: Omit<\n    UseMutationOptions<\n      PostResponse,\n      Error,\n      {responseId: string; postId: string} & UpdateResponseRequest\n    >,\n    'mutationFn'\n  >,\n): UseMutationResult<\n  PostResponse,\n  Error,\n  {responseId: string; postId: string} & UpdateResponseRequest\n> {\n  const queryClient = useQueryClient()\n\n  return useMutation({\n    mutationFn: async ({\n      responseId,\n      postId: _,\n      ...data\n    }): Promise<PostResponse> => {\n      const client = getApiClient()\n      const response = await client.put<ApiResponse<PostResponse>>(\n        `/responses/${responseId}`,\n        data,\n      )\n      return response.data.data\n    },\n    onSuccess: (_, variables) => {\n      queryClient.invalidateQueries({\n        queryKey: postKeys.responses(variables.postId),\n      })\n    },\n    ...options,\n  })\n}\n\n/**\n * Report a post\n *\n * @endpoint POST /posts/{postId}/report\n */\nexport function useReportPost(\n  options?: Omit<\n    UseMutationOptions<\n      void,\n      Error,\n      {postId: string; reason: string; details?: string}\n    >,\n    'mutationFn'\n  >,\n): UseMutationResult<\n  void,\n  Error,\n  {postId: string; reason: string; details?: string}\n> {\n  return useMutation({\n    mutationFn: async ({postId, ...data}): Promise<void> => {\n      const client = getApiClient()\n      await client.post(`/posts/${postId}/report`, data)\n    },\n    ...options,\n  })\n}\n\n// PostStatusUpdate matches partial PostStatusDto for updates\ninterface PostStatusUpdate {\n  isRead?: boolean\n  isHidden?: boolean\n  isPinned?: boolean\n}\n\n/**\n * Update user's status for a post (read/hidden/pinned)\n *\n * @endpoint PUT /posts/{postId}/status\n */\nexport function useUpdatePostStatus(\n  options?: Omit<\n    UseMutationOptions<\n      PostStatusDto,\n      Error,\n      {postId: string} & PostStatusUpdate\n    >,\n    'mutationFn'\n  >,\n): UseMutationResult<\n  PostStatusDto,\n  Error,\n  {postId: string} & PostStatusUpdate\n> {\n  const queryClient = useQueryClient()\n\n  return useMutation({\n    mutationFn: async ({postId, ...data}): Promise<PostStatusDto> => {\n      const client = getApiClient()\n      const response = await client.put<ApiResponse<PostStatusDto>>(\n        `/posts/${postId}/status`,\n        data,\n      )\n      return response.data.data\n    },\n    onSuccess: (data, variables) => {\n      queryClient.setQueryData(postKeys.status(variables.postId), data)\n    },\n    ...options,\n  })\n}\n"]}
|
|
242
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"posts.js","sourceRoot":"","sources":["../../../src/api/mutations/posts.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Posts Mutation Hooks\n *\n * TanStack Query hooks for post/board mutation operations.\n */\n\nimport {\n  useMutation,\n  type UseMutationOptions,\n  type UseMutationResult,\n  useQueryClient,\n} from '@tanstack/react-query'\n\nimport {getApiClient} from '../client'\nimport {postKeys} from '../queries/posts'\nimport {\n  type ApiResponse,\n  type CreatePostRequest,\n  type CreateReplyRequest,\n  type Post,\n  type PostResponse,\n  type PostStatusDto,\n  type RespondToPostRequest,\n  type UpdateResponseRequest,\n  type UpvoteResponse,\n} from '../types'\n\n// ============================================================================\n// MUTATION HOOKS\n// ============================================================================\n\n/**\n * Create a new post\n *\n * @endpoint POST /spots/{spotId}/posts\n */\nexport function useCreatePost(\n  options?: Omit<\n    UseMutationOptions<Post, Error, {spotId: string} & CreatePostRequest>,\n    'mutationFn'\n  >,\n): UseMutationResult<Post, Error, {spotId: string} & CreatePostRequest> {\n  const queryClient = useQueryClient()\n\n  return useMutation({\n    mutationFn: async ({spotId, ...data}): Promise<Post> => {\n      const client = getApiClient()\n      const response = await client.post<ApiResponse<Post>>(\n        `/spots/${spotId}/posts`,\n        data,\n      )\n      return response.data.data\n    },\n    onSuccess: (_, variables) => {\n      queryClient.invalidateQueries({\n        queryKey: postKeys.bySpot(variables.spotId),\n      })\n    },\n    ...options,\n  })\n}\n\n/**\n * Update a post\n *\n * @endpoint PUT /posts/{postId}\n */\nexport function useUpdatePost(\n  options?: Omit<\n    UseMutationOptions<\n      Post,\n      Error,\n      {postId: string; title?: string; content?: string}\n    >,\n    'mutationFn'\n  >,\n): UseMutationResult<\n  Post,\n  Error,\n  {postId: string; title?: string; content?: string}\n> {\n  const queryClient = useQueryClient()\n\n  return useMutation({\n    mutationFn: async ({postId, title, content}): Promise<Post> => {\n      const client = getApiClient()\n      // API expects 'description' field, not 'content'\n      const response = await client.put<ApiResponse<Post>>(\n        `/posts/${postId}`,\n        {title, description: content},\n      )\n      return response.data.data\n    },\n    onSuccess: (data, variables) => {\n      queryClient.setQueryData(postKeys.detail(variables.postId), data)\n    },\n    ...options,\n  })\n}\n\n/**\n * Delete a post\n *\n * @endpoint DELETE /posts/{postId}\n */\nexport function useDeletePost(\n  options?: Omit<UseMutationOptions<void, Error, string>, 'mutationFn'>,\n): UseMutationResult<void, Error, string> {\n  const queryClient = useQueryClient()\n\n  return useMutation({\n    mutationFn: async (postId: string): Promise<void> => {\n      const client = getApiClient()\n      await client.delete(`/posts/${postId}`)\n    },\n    onSuccess: (_, postId) => {\n      queryClient.invalidateQueries({queryKey: postKeys.detail(postId)})\n      queryClient.invalidateQueries({queryKey: postKeys.lists()})\n    },\n    ...options,\n  })\n}\n\n/**\n * Fulfill a post (mark as fulfilled)\n *\n * @endpoint PUT /posts/{postId}/fulfill\n */\nexport function useFulfillPost(\n  options?: Omit<UseMutationOptions<Post, Error, string>, 'mutationFn'>,\n): UseMutationResult<Post, Error, string> {\n  const queryClient = useQueryClient()\n\n  return useMutation({\n    mutationFn: async (postId: string): Promise<Post> => {\n      const client = getApiClient()\n      const response = await client.put<ApiResponse<Post>>(\n        `/posts/${postId}/fulfill`,\n      )\n      return response.data.data\n    },\n    onSuccess: (data, postId) => {\n      queryClient.setQueryData(postKeys.detail(postId), data)\n    },\n    ...options,\n  })\n}\n\n/**\n * Upvote/unupvote a post\n *\n * @endpoint POST /posts/{postId}/upvote\n */\nexport function useUpvotePost(\n  options?: Omit<\n    UseMutationOptions<UpvoteResponse, Error, string>,\n    'mutationFn'\n  >,\n): UseMutationResult<UpvoteResponse, Error, string> {\n  const queryClient = useQueryClient()\n\n  return useMutation({\n    mutationFn: async (postId: string): Promise<UpvoteResponse> => {\n      const client = getApiClient()\n      const response = await client.post<ApiResponse<UpvoteResponse>>(\n        `/posts/${postId}/upvote`,\n      )\n      return response.data.data\n    },\n    onSuccess: (_, postId) => {\n      queryClient.invalidateQueries({queryKey: postKeys.detail(postId)})\n    },\n    ...options,\n  })\n}\n\n/**\n * Create a reply to a post\n *\n * @endpoint POST /posts/{postId}/reply\n */\nexport function useCreateReply(\n  options?: Omit<\n    UseMutationOptions<unknown, Error, {postId: string} & CreateReplyRequest>,\n    'mutationFn'\n  >,\n): UseMutationResult<unknown, Error, {postId: string} & CreateReplyRequest> {\n  const queryClient = useQueryClient()\n\n  return useMutation({\n    mutationFn: async ({postId, ...data}) => {\n      const client = getApiClient()\n      const response = await client.post<ApiResponse<unknown>>(\n        `/posts/${postId}/reply`,\n        data,\n      )\n      return response.data.data\n    },\n    onSuccess: (_, variables) => {\n      queryClient.invalidateQueries({\n        queryKey: postKeys.detail(variables.postId),\n      })\n    },\n    ...options,\n  })\n}\n\n/**\n * Delete a reply\n *\n * @endpoint DELETE /replies/{replyId}\n */\nexport function useDeleteReply(\n  options?: Omit<\n    UseMutationOptions<void, Error, {replyId: string; postId: string}>,\n    'mutationFn'\n  >,\n): UseMutationResult<void, Error, {replyId: string; postId: string}> {\n  const queryClient = useQueryClient()\n\n  return useMutation({\n    mutationFn: async ({replyId}): Promise<void> => {\n      const client = getApiClient()\n      await client.delete(`/replies/${replyId}`)\n    },\n    onSuccess: (_, variables) => {\n      queryClient.invalidateQueries({\n        queryKey: postKeys.detail(variables.postId),\n      })\n    },\n    ...options,\n  })\n}\n\n/**\n * Respond to a post (express interest)\n *\n * @endpoint POST /posts/{postId}/respond\n */\nexport function useRespondToPost(\n  options?: Omit<\n    UseMutationOptions<\n      PostResponse,\n      Error,\n      {postId: string} & RespondToPostRequest\n    >,\n    'mutationFn'\n  >,\n): UseMutationResult<\n  PostResponse,\n  Error,\n  {postId: string} & RespondToPostRequest\n> {\n  const queryClient = useQueryClient()\n\n  return useMutation({\n    mutationFn: async ({postId, ...data}): Promise<PostResponse> => {\n      const client = getApiClient()\n      const response = await client.post<ApiResponse<PostResponse>>(\n        `/posts/${postId}/respond`,\n        data,\n      )\n      return response.data.data\n    },\n    onSuccess: (_, variables) => {\n      queryClient.invalidateQueries({\n        queryKey: postKeys.detail(variables.postId),\n      })\n      queryClient.invalidateQueries({\n        queryKey: postKeys.responses(variables.postId),\n      })\n    },\n    ...options,\n  })\n}\n\n/**\n * Accept or decline a response\n *\n * @endpoint PUT /responses/{responseId}\n */\nexport function useUpdateResponse(\n  options?: Omit<\n    UseMutationOptions<\n      PostResponse,\n      Error,\n      {responseId: string; postId: string} & UpdateResponseRequest\n    >,\n    'mutationFn'\n  >,\n): UseMutationResult<\n  PostResponse,\n  Error,\n  {responseId: string; postId: string} & UpdateResponseRequest\n> {\n  const queryClient = useQueryClient()\n\n  return useMutation({\n    mutationFn: async ({\n      responseId,\n      postId: _,\n      ...data\n    }): Promise<PostResponse> => {\n      const client = getApiClient()\n      const response = await client.put<ApiResponse<PostResponse>>(\n        `/responses/${responseId}`,\n        data,\n      )\n      return response.data.data\n    },\n    onSuccess: (_, variables) => {\n      queryClient.invalidateQueries({\n        queryKey: postKeys.responses(variables.postId),\n      })\n    },\n    ...options,\n  })\n}\n\n/**\n * Report a post\n *\n * @endpoint POST /posts/{postId}/report\n */\nexport function useReportPost(\n  options?: Omit<\n    UseMutationOptions<\n      void,\n      Error,\n      {postId: string; reason: string; details?: string}\n    >,\n    'mutationFn'\n  >,\n): UseMutationResult<\n  void,\n  Error,\n  {postId: string; reason: string; details?: string}\n> {\n  return useMutation({\n    mutationFn: async ({postId, ...data}): Promise<void> => {\n      const client = getApiClient()\n      await client.post(`/posts/${postId}/report`, data)\n    },\n    ...options,\n  })\n}\n\n// PostStatusUpdate matches partial PostStatusDto for updates\ninterface PostStatusUpdate {\n  isRead?: boolean\n  isHidden?: boolean\n  isPinned?: boolean\n}\n\n/**\n * Update user's status for a post (read/hidden/pinned)\n *\n * @endpoint PUT /posts/{postId}/status\n */\nexport function useUpdatePostStatus(\n  options?: Omit<\n    UseMutationOptions<\n      PostStatusDto,\n      Error,\n      {postId: string} & PostStatusUpdate\n    >,\n    'mutationFn'\n  >,\n): UseMutationResult<\n  PostStatusDto,\n  Error,\n  {postId: string} & PostStatusUpdate\n> {\n  const queryClient = useQueryClient()\n\n  return useMutation({\n    mutationFn: async ({postId, ...data}): Promise<PostStatusDto> => {\n      const client = getApiClient()\n      const response = await client.put<ApiResponse<PostStatusDto>>(\n        `/posts/${postId}/status`,\n        data,\n      )\n      return response.data.data\n    },\n    onSuccess: (data, variables) => {\n      queryClient.setQueryData(postKeys.status(variables.postId), data)\n    },\n    ...options,\n  })\n}\n"]}
|
|
@@ -0,0 +1,33 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Redemption Mutation Hooks
|
|
3
|
+
*
|
|
4
|
+
* TanStack Query mutations for redemption operations.
|
|
5
|
+
*/
|
|
6
|
+
import { type UseMutationResult } from '@tanstack/react-query';
|
|
7
|
+
import { type WalletRedemption } from '../queries/wallet';
|
|
8
|
+
export interface RedeemRequest {
|
|
9
|
+
qrCode: string;
|
|
10
|
+
spotId?: string;
|
|
11
|
+
deviceInfo?: string;
|
|
12
|
+
latitude?: number;
|
|
13
|
+
longitude?: number;
|
|
14
|
+
notes?: string;
|
|
15
|
+
}
|
|
16
|
+
export interface VoidRedemptionRequest {
|
|
17
|
+
reason: string;
|
|
18
|
+
}
|
|
19
|
+
/**
|
|
20
|
+
* Redeem a QR code (spot owner/staff action)
|
|
21
|
+
*
|
|
22
|
+
* @endpoint POST /redemptions/redeem
|
|
23
|
+
*/
|
|
24
|
+
export declare function useRedeem(): UseMutationResult<WalletRedemption, Error, RedeemRequest>;
|
|
25
|
+
/**
|
|
26
|
+
* Void a redemption (spot owner action)
|
|
27
|
+
*
|
|
28
|
+
* @endpoint PUT /redemptions/:id/void
|
|
29
|
+
*/
|
|
30
|
+
export declare function useVoidRedemption(): UseMutationResult<WalletRedemption, Error, {
|
|
31
|
+
redemptionId: string;
|
|
32
|
+
reason: string;
|
|
33
|
+
}>;
|
|
@@ -0,0 +1,63 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* Redemption Mutation Hooks
|
|
4
|
+
*
|
|
5
|
+
* TanStack Query mutations for redemption operations.
|
|
6
|
+
*/
|
|
7
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
8
|
+
exports.useRedeem = useRedeem;
|
|
9
|
+
exports.useVoidRedemption = useVoidRedemption;
|
|
10
|
+
const react_query_1 = require("@tanstack/react-query");
|
|
11
|
+
const client_1 = require("../client");
|
|
12
|
+
const wallet_1 = require("../queries/wallet");
|
|
13
|
+
// ============================================================================
|
|
14
|
+
// MUTATION HOOKS
|
|
15
|
+
// ============================================================================
|
|
16
|
+
/**
|
|
17
|
+
* Redeem a QR code (spot owner/staff action)
|
|
18
|
+
*
|
|
19
|
+
* @endpoint POST /redemptions/redeem
|
|
20
|
+
*/
|
|
21
|
+
function useRedeem() {
|
|
22
|
+
const queryClient = (0, react_query_1.useQueryClient)();
|
|
23
|
+
return (0, react_query_1.useMutation)({
|
|
24
|
+
mutationFn: async (request) => {
|
|
25
|
+
const client = (0, client_1.getApiClient)();
|
|
26
|
+
const response = await client.post('/redemptions/redeem', request);
|
|
27
|
+
return response.data.data;
|
|
28
|
+
},
|
|
29
|
+
onSuccess: (_data, variables) => {
|
|
30
|
+
// Invalidate redemption lookup for this QR code
|
|
31
|
+
queryClient.invalidateQueries({
|
|
32
|
+
queryKey: wallet_1.walletKeys.lookup(variables.qrCode),
|
|
33
|
+
});
|
|
34
|
+
// Invalidate spot redemptions if spotId provided
|
|
35
|
+
if (variables.spotId) {
|
|
36
|
+
queryClient.invalidateQueries({
|
|
37
|
+
queryKey: wallet_1.walletKeys.spotRedemptions(variables.spotId),
|
|
38
|
+
});
|
|
39
|
+
}
|
|
40
|
+
},
|
|
41
|
+
});
|
|
42
|
+
}
|
|
43
|
+
/**
|
|
44
|
+
* Void a redemption (spot owner action)
|
|
45
|
+
*
|
|
46
|
+
* @endpoint PUT /redemptions/:id/void
|
|
47
|
+
*/
|
|
48
|
+
function useVoidRedemption() {
|
|
49
|
+
const queryClient = (0, react_query_1.useQueryClient)();
|
|
50
|
+
return (0, react_query_1.useMutation)({
|
|
51
|
+
mutationFn: async ({ redemptionId, reason, }) => {
|
|
52
|
+
const client = (0, client_1.getApiClient)();
|
|
53
|
+
const response = await client.put(`/redemptions/${redemptionId}/void`, { reason });
|
|
54
|
+
return response.data.data;
|
|
55
|
+
},
|
|
56
|
+
onSuccess: () => {
|
|
57
|
+
// Invalidate wallet queries
|
|
58
|
+
queryClient.invalidateQueries({ queryKey: wallet_1.walletKeys.all });
|
|
59
|
+
queryClient.invalidateQueries({ queryKey: wallet_1.walletKeys.redemptions() });
|
|
60
|
+
},
|
|
61
|
+
});
|
|
62
|
+
}
|
|
63
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVkZW1wdGlvbnMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvYXBpL211dGF0aW9ucy9yZWRlbXB0aW9ucy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUE7Ozs7R0FJRzs7QUFnQ0gsOEJBc0JDO0FBT0QsOENBeUJDO0FBcEZELHVEQUF5RjtBQUN6RixzQ0FBc0M7QUFDdEMsOENBQW1FO0FBbUJuRSwrRUFBK0U7QUFDL0UsaUJBQWlCO0FBQ2pCLCtFQUErRTtBQUUvRTs7OztHQUlHO0FBQ0gsU0FBZ0IsU0FBUztJQUN2QixNQUFNLFdBQVcsR0FBRyxJQUFBLDRCQUFjLEdBQUUsQ0FBQTtJQUVwQyxPQUFPLElBQUEseUJBQVcsRUFBQztRQUNqQixVQUFVLEVBQUUsS0FBSyxFQUFFLE9BQXNCLEVBQTZCLEVBQUU7WUFDdEUsTUFBTSxNQUFNLEdBQUcsSUFBQSxxQkFBWSxHQUFFLENBQUE7WUFDN0IsTUFBTSxRQUFRLEdBQUcsTUFBTSxNQUFNLENBQUMsSUFBSSxDQUFDLHFCQUFxQixFQUFFLE9BQU8sQ0FBQyxDQUFBO1lBQ2xFLE9BQU8sUUFBUSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUE7UUFDM0IsQ0FBQztRQUNELFNBQVMsRUFBRSxDQUFDLEtBQUssRUFBRSxTQUFTLEVBQUUsRUFBRTtZQUM5QixnREFBZ0Q7WUFDaEQsV0FBVyxDQUFDLGlCQUFpQixDQUFDO2dCQUM1QixRQUFRLEVBQUUsbUJBQVUsQ0FBQyxNQUFNLENBQUMsU0FBUyxDQUFDLE1BQU0sQ0FBQzthQUM5QyxDQUFDLENBQUE7WUFDRixpREFBaUQ7WUFDakQsSUFBSSxTQUFTLENBQUMsTUFBTSxFQUFFLENBQUM7Z0JBQ3JCLFdBQVcsQ0FBQyxpQkFBaUIsQ0FBQztvQkFDNUIsUUFBUSxFQUFFLG1CQUFVLENBQUMsZUFBZSxDQUFDLFNBQVMsQ0FBQyxNQUFNLENBQUM7aUJBQ3ZELENBQUMsQ0FBQTtZQUNKLENBQUM7UUFDSCxDQUFDO0tBQ0YsQ0FBQyxDQUFBO0FBQ0osQ0FBQztBQUVEOzs7O0dBSUc7QUFDSCxTQUFnQixpQkFBaUI7SUFLL0IsTUFBTSxXQUFXLEdBQUcsSUFBQSw0QkFBYyxHQUFFLENBQUE7SUFFcEMsT0FBTyxJQUFBLHlCQUFXLEVBQUM7UUFDakIsVUFBVSxFQUFFLEtBQUssRUFBRSxFQUNqQixZQUFZLEVBQ1osTUFBTSxHQUlQLEVBQTZCLEVBQUU7WUFDOUIsTUFBTSxNQUFNLEdBQUcsSUFBQSxxQkFBWSxHQUFFLENBQUE7WUFDN0IsTUFBTSxRQUFRLEdBQUcsTUFBTSxNQUFNLENBQUMsR0FBRyxDQUFDLGdCQUFnQixZQUFZLE9BQU8sRUFBRSxFQUFDLE1BQU0sRUFBQyxDQUFDLENBQUE7WUFDaEYsT0FBTyxRQUFRLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQTtRQUMzQixDQUFDO1FBQ0QsU0FBUyxFQUFFLEdBQUcsRUFBRTtZQUNkLDRCQUE0QjtZQUM1QixXQUFXLENBQUMsaUJBQWlCLENBQUMsRUFBQyxRQUFRLEVBQUUsbUJBQVUsQ0FBQyxHQUFHLEVBQUMsQ0FBQyxDQUFBO1lBQ3pELFdBQVcsQ0FBQyxpQkFBaUIsQ0FBQyxFQUFDLFFBQVEsRUFBRSxtQkFBVSxDQUFDLFdBQVcsRUFBRSxFQUFDLENBQUMsQ0FBQTtRQUNyRSxDQUFDO0tBQ0YsQ0FBQyxDQUFBO0FBQ0osQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogUmVkZW1wdGlvbiBNdXRhdGlvbiBIb29rc1xuICpcbiAqIFRhblN0YWNrIFF1ZXJ5IG11dGF0aW9ucyBmb3IgcmVkZW1wdGlvbiBvcGVyYXRpb25zLlxuICovXG5cbmltcG9ydCB7dXNlTXV0YXRpb24sIHVzZVF1ZXJ5Q2xpZW50LCB0eXBlIFVzZU11dGF0aW9uUmVzdWx0fSBmcm9tICdAdGFuc3RhY2svcmVhY3QtcXVlcnknXG5pbXBvcnQge2dldEFwaUNsaWVudH0gZnJvbSAnLi4vY2xpZW50J1xuaW1wb3J0IHt3YWxsZXRLZXlzLCB0eXBlIFdhbGxldFJlZGVtcHRpb259IGZyb20gJy4uL3F1ZXJpZXMvd2FsbGV0J1xuXG4vLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG4vLyBUWVBFU1xuLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuXG5leHBvcnQgaW50ZXJmYWNlIFJlZGVlbVJlcXVlc3Qge1xuICBxckNvZGU6IHN0cmluZ1xuICBzcG90SWQ/OiBzdHJpbmdcbiAgZGV2aWNlSW5mbz86IHN0cmluZ1xuICBsYXRpdHVkZT86IG51bWJlclxuICBsb25naXR1ZGU/OiBudW1iZXJcbiAgbm90ZXM/OiBzdHJpbmdcbn1cblxuZXhwb3J0IGludGVyZmFjZSBWb2lkUmVkZW1wdGlvblJlcXVlc3Qge1xuICByZWFzb246IHN0cmluZ1xufVxuXG4vLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG4vLyBNVVRBVElPTiBIT09LU1xuLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuXG4vKipcbiAqIFJlZGVlbSBhIFFSIGNvZGUgKHNwb3Qgb3duZXIvc3RhZmYgYWN0aW9uKVxuICpcbiAqIEBlbmRwb2ludCBQT1NUIC9yZWRlbXB0aW9ucy9yZWRlZW1cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIHVzZVJlZGVlbSgpOiBVc2VNdXRhdGlvblJlc3VsdDxXYWxsZXRSZWRlbXB0aW9uLCBFcnJvciwgUmVkZWVtUmVxdWVzdD4ge1xuICBjb25zdCBxdWVyeUNsaWVudCA9IHVzZVF1ZXJ5Q2xpZW50KClcblxuICByZXR1cm4gdXNlTXV0YXRpb24oe1xuICAgIG11dGF0aW9uRm46IGFzeW5jIChyZXF1ZXN0OiBSZWRlZW1SZXF1ZXN0KTogUHJvbWlzZTxXYWxsZXRSZWRlbXB0aW9uPiA9PiB7XG4gICAgICBjb25zdCBjbGllbnQgPSBnZXRBcGlDbGllbnQoKVxuICAgICAgY29uc3QgcmVzcG9uc2UgPSBhd2FpdCBjbGllbnQucG9zdCgnL3JlZGVtcHRpb25zL3JlZGVlbScsIHJlcXVlc3QpXG4gICAgICByZXR1cm4gcmVzcG9uc2UuZGF0YS5kYXRhXG4gICAgfSxcbiAgICBvblN1Y2Nlc3M6IChfZGF0YSwgdmFyaWFibGVzKSA9PiB7XG4gICAgICAvLyBJbnZhbGlkYXRlIHJlZGVtcHRpb24gbG9va3VwIGZvciB0aGlzIFFSIGNvZGVcbiAgICAgIHF1ZXJ5Q2xpZW50LmludmFsaWRhdGVRdWVyaWVzKHtcbiAgICAgICAgcXVlcnlLZXk6IHdhbGxldEtleXMubG9va3VwKHZhcmlhYmxlcy5xckNvZGUpLFxuICAgICAgfSlcbiAgICAgIC8vIEludmFsaWRhdGUgc3BvdCByZWRlbXB0aW9ucyBpZiBzcG90SWQgcHJvdmlkZWRcbiAgICAgIGlmICh2YXJpYWJsZXMuc3BvdElkKSB7XG4gICAgICAgIHF1ZXJ5Q2xpZW50LmludmFsaWRhdGVRdWVyaWVzKHtcbiAgICAgICAgICBxdWVyeUtleTogd2FsbGV0S2V5cy5zcG90UmVkZW1wdGlvbnModmFyaWFibGVzLnNwb3RJZCksXG4gICAgICAgIH0pXG4gICAgICB9XG4gICAgfSxcbiAgfSlcbn1cblxuLyoqXG4gKiBWb2lkIGEgcmVkZW1wdGlvbiAoc3BvdCBvd25lciBhY3Rpb24pXG4gKlxuICogQGVuZHBvaW50IFBVVCAvcmVkZW1wdGlvbnMvOmlkL3ZvaWRcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIHVzZVZvaWRSZWRlbXB0aW9uKCk6IFVzZU11dGF0aW9uUmVzdWx0PFxuICBXYWxsZXRSZWRlbXB0aW9uLFxuICBFcnJvcixcbiAge3JlZGVtcHRpb25JZDogc3RyaW5nOyByZWFzb246IHN0cmluZ31cbj4ge1xuICBjb25zdCBxdWVyeUNsaWVudCA9IHVzZVF1ZXJ5Q2xpZW50KClcblxuICByZXR1cm4gdXNlTXV0YXRpb24oe1xuICAgIG11dGF0aW9uRm46IGFzeW5jICh7XG4gICAgICByZWRlbXB0aW9uSWQsXG4gICAgICByZWFzb24sXG4gICAgfToge1xuICAgICAgcmVkZW1wdGlvbklkOiBzdHJpbmdcbiAgICAgIHJlYXNvbjogc3RyaW5nXG4gICAgfSk6IFByb21pc2U8V2FsbGV0UmVkZW1wdGlvbj4gPT4ge1xuICAgICAgY29uc3QgY2xpZW50ID0gZ2V0QXBpQ2xpZW50KClcbiAgICAgIGNvbnN0IHJlc3BvbnNlID0gYXdhaXQgY2xpZW50LnB1dChgL3JlZGVtcHRpb25zLyR7cmVkZW1wdGlvbklkfS92b2lkYCwge3JlYXNvbn0pXG4gICAgICByZXR1cm4gcmVzcG9uc2UuZGF0YS5kYXRhXG4gICAgfSxcbiAgICBvblN1Y2Nlc3M6ICgpID0+IHtcbiAgICAgIC8vIEludmFsaWRhdGUgd2FsbGV0IHF1ZXJpZXNcbiAgICAgIHF1ZXJ5Q2xpZW50LmludmFsaWRhdGVRdWVyaWVzKHtxdWVyeUtleTogd2FsbGV0S2V5cy5hbGx9KVxuICAgICAgcXVlcnlDbGllbnQuaW52YWxpZGF0ZVF1ZXJpZXMoe3F1ZXJ5S2V5OiB3YWxsZXRLZXlzLnJlZGVtcHRpb25zKCl9KVxuICAgIH0sXG4gIH0pXG59XG4iXX0=
|
|
@@ -19,7 +19,7 @@ var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
|
19
19
|
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
|
|
20
20
|
};
|
|
21
21
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
22
|
-
exports.orderKeys = exports.productKeys = exports.miscKeys = exports.notificationKeys = exports.templateKeys = exports.clubKeys = exports.conversationKeys = exports.postKeys = exports.spotKeys = exports.userKeys = void 0;
|
|
22
|
+
exports.walletKeys = exports.orderKeys = exports.productKeys = exports.miscKeys = exports.notificationKeys = exports.templateKeys = exports.clubKeys = exports.conversationKeys = exports.postKeys = exports.spotKeys = exports.userKeys = void 0;
|
|
23
23
|
// Auth (mutations, but grouped with queries for convenience)
|
|
24
24
|
__exportStar(require("./auth"), exports);
|
|
25
25
|
// Users
|
|
@@ -62,4 +62,8 @@ Object.defineProperty(exports, "productKeys", { enumerable: true, get: function
|
|
|
62
62
|
__exportStar(require("./orders"), exports);
|
|
63
63
|
var orders_1 = require("./orders");
|
|
64
64
|
Object.defineProperty(exports, "orderKeys", { enumerable: true, get: function () { return orders_1.orderKeys; } });
|
|
65
|
-
|
|
65
|
+
// Wallet (redemptions)
|
|
66
|
+
__exportStar(require("./wallet"), exports);
|
|
67
|
+
var wallet_1 = require("./wallet");
|
|
68
|
+
Object.defineProperty(exports, "walletKeys", { enumerable: true, get: function () { return wallet_1.walletKeys; } });
|
|
69
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvYXBpL3F1ZXJpZXMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOzs7O0dBSUc7Ozs7Ozs7Ozs7Ozs7Ozs7O0FBRUgsNkRBQTZEO0FBQzdELHlDQUFzQjtBQUV0QixRQUFRO0FBQ1IsMENBQXVCO0FBQ3ZCLGlDQUFnQztBQUF4QixpR0FBQSxRQUFRLE9BQUE7QUFFaEIsUUFBUTtBQUNSLDBDQUF1QjtBQUN2QixpQ0FBZ0M7QUFBeEIsaUdBQUEsUUFBUSxPQUFBO0FBRWhCLFFBQVE7QUFDUiwwQ0FBdUI7QUFDdkIsaUNBQWdDO0FBQXhCLGlHQUFBLFFBQVEsT0FBQTtBQUVoQixnQkFBZ0I7QUFDaEIsa0RBQStCO0FBQy9CLGlEQUFnRDtBQUF4QyxpSEFBQSxnQkFBZ0IsT0FBQTtBQUV4QixRQUFRO0FBQ1IsMENBQXVCO0FBQ3ZCLGlDQUFnQztBQUF4QixpR0FBQSxRQUFRLE9BQUE7QUFFaEIsWUFBWTtBQUNaLDhDQUEyQjtBQUMzQix5Q0FBd0M7QUFBaEMseUdBQUEsWUFBWSxPQUFBO0FBRXBCLGdCQUFnQjtBQUNoQixrREFBK0I7QUFDL0IsaURBQWdEO0FBQXhDLGlIQUFBLGdCQUFnQixPQUFBO0FBRXhCLCtCQUErQjtBQUMvQix5Q0FBc0I7QUFDdEIsK0JBQStCO0FBQXZCLGdHQUFBLFFBQVEsT0FBQTtBQUVoQix5QkFBeUI7QUFDekIsNkNBQTBCO0FBQzFCLHVDQUFzQztBQUE5Qix1R0FBQSxXQUFXLE9BQUE7QUFFbkIsdUJBQXVCO0FBQ3ZCLDJDQUF3QjtBQUN4QixtQ0FBa0M7QUFBMUIsbUdBQUEsU0FBUyxPQUFBO0FBRWpCLHVCQUF1QjtBQUN2QiwyQ0FBd0I7QUFDeEIsbUNBQW1DO0FBQTNCLG9HQUFBLFVBQVUsT0FBQSIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogU3BvdHMgU0RLIFF1ZXJ5IEhvb2tzIEluZGV4XG4gKlxuICogUmUtZXhwb3J0cyBhbGwgcXVlcnkgaG9va3MgYW5kIGtleXMuXG4gKi9cblxuLy8gQXV0aCAobXV0YXRpb25zLCBidXQgZ3JvdXBlZCB3aXRoIHF1ZXJpZXMgZm9yIGNvbnZlbmllbmNlKVxuZXhwb3J0ICogZnJvbSAnLi9hdXRoJ1xuXG4vLyBVc2Vyc1xuZXhwb3J0ICogZnJvbSAnLi91c2VycydcbmV4cG9ydCB7dXNlcktleXN9IGZyb20gJy4vdXNlcnMnXG5cbi8vIFNwb3RzXG5leHBvcnQgKiBmcm9tICcuL3Nwb3RzJ1xuZXhwb3J0IHtzcG90S2V5c30gZnJvbSAnLi9zcG90cydcblxuLy8gUG9zdHNcbmV4cG9ydCAqIGZyb20gJy4vcG9zdHMnXG5leHBvcnQge3Bvc3RLZXlzfSBmcm9tICcuL3Bvc3RzJ1xuXG4vLyBDb252ZXJzYXRpb25zXG5leHBvcnQgKiBmcm9tICcuL2NvbnZlcnNhdGlvbnMnXG5leHBvcnQge2NvbnZlcnNhdGlvbktleXN9IGZyb20gJy4vY29udmVyc2F0aW9ucydcblxuLy8gQ2x1YnNcbmV4cG9ydCAqIGZyb20gJy4vY2x1YnMnXG5leHBvcnQge2NsdWJLZXlzfSBmcm9tICcuL2NsdWJzJ1xuXG4vLyBUZW1wbGF0ZXNcbmV4cG9ydCAqIGZyb20gJy4vdGVtcGxhdGVzJ1xuZXhwb3J0IHt0ZW1wbGF0ZUtleXN9IGZyb20gJy4vdGVtcGxhdGVzJ1xuXG4vLyBOb3RpZmljYXRpb25zXG5leHBvcnQgKiBmcm9tICcuL25vdGlmaWNhdGlvbnMnXG5leHBvcnQge25vdGlmaWNhdGlvbktleXN9IGZyb20gJy4vbm90aWZpY2F0aW9ucydcblxuLy8gTWlzYyAoY2l0aWVzLCB2aWJlcywgZXZlbnRzKVxuZXhwb3J0ICogZnJvbSAnLi9taXNjJ1xuZXhwb3J0IHttaXNjS2V5c30gZnJvbSAnLi9taXNjJ1xuXG4vLyBQcm9kdWN0cyAobWFya2V0cGxhY2UpXG5leHBvcnQgKiBmcm9tICcuL3Byb2R1Y3RzJ1xuZXhwb3J0IHtwcm9kdWN0S2V5c30gZnJvbSAnLi9wcm9kdWN0cydcblxuLy8gT3JkZXJzIChtYXJrZXRwbGFjZSlcbmV4cG9ydCAqIGZyb20gJy4vb3JkZXJzJ1xuZXhwb3J0IHtvcmRlcktleXN9IGZyb20gJy4vb3JkZXJzJ1xuXG4vLyBXYWxsZXQgKHJlZGVtcHRpb25zKVxuZXhwb3J0ICogZnJvbSAnLi93YWxsZXQnXG5leHBvcnQge3dhbGxldEtleXN9IGZyb20gJy4vd2FsbGV0J1xuIl19
|
|
@@ -0,0 +1,109 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Wallet Query Hooks
|
|
3
|
+
*
|
|
4
|
+
* TanStack Query hooks for wallet and redemption operations.
|
|
5
|
+
*/
|
|
6
|
+
import { type UseQueryOptions, type UseQueryResult } from '@tanstack/react-query';
|
|
7
|
+
export type RedemptionStatus = 'PENDING' | 'REDEEMED' | 'EXPIRED' | 'CANCELLED' | 'VOID';
|
|
8
|
+
export interface WalletProduct {
|
|
9
|
+
id: string;
|
|
10
|
+
name: string;
|
|
11
|
+
type: string;
|
|
12
|
+
imageUrl: string | null;
|
|
13
|
+
}
|
|
14
|
+
export interface WalletOrderItem {
|
|
15
|
+
product: WalletProduct;
|
|
16
|
+
quantity: number;
|
|
17
|
+
unitPrice: number;
|
|
18
|
+
}
|
|
19
|
+
export interface WalletOrder {
|
|
20
|
+
id: string;
|
|
21
|
+
orderNumber: string;
|
|
22
|
+
spotId: string;
|
|
23
|
+
createdAt: string;
|
|
24
|
+
spot: {
|
|
25
|
+
id: string;
|
|
26
|
+
name: string;
|
|
27
|
+
slug: string;
|
|
28
|
+
address: string | null;
|
|
29
|
+
};
|
|
30
|
+
}
|
|
31
|
+
export interface WalletRedemption {
|
|
32
|
+
id: string;
|
|
33
|
+
qrCode: string;
|
|
34
|
+
qrCodeUrl: string | null;
|
|
35
|
+
status: RedemptionStatus;
|
|
36
|
+
validFrom: string;
|
|
37
|
+
validUntil: string;
|
|
38
|
+
redeemedAt: string | null;
|
|
39
|
+
createdAt: string;
|
|
40
|
+
orderItem: WalletOrderItem & {
|
|
41
|
+
order: WalletOrder;
|
|
42
|
+
};
|
|
43
|
+
}
|
|
44
|
+
export interface RedemptionLookup extends WalletRedemption {
|
|
45
|
+
redeemedBy: {
|
|
46
|
+
id: string;
|
|
47
|
+
name: string;
|
|
48
|
+
} | null;
|
|
49
|
+
spot: {
|
|
50
|
+
id: string;
|
|
51
|
+
name: string;
|
|
52
|
+
} | null;
|
|
53
|
+
orderItem: WalletOrderItem & {
|
|
54
|
+
order: WalletOrder & {
|
|
55
|
+
user: {
|
|
56
|
+
id: string;
|
|
57
|
+
name: string | null;
|
|
58
|
+
avatarUrl: string | null;
|
|
59
|
+
};
|
|
60
|
+
};
|
|
61
|
+
};
|
|
62
|
+
}
|
|
63
|
+
export interface WalletHistoryResponse {
|
|
64
|
+
data: WalletRedemption[];
|
|
65
|
+
meta: {
|
|
66
|
+
total: number;
|
|
67
|
+
page: number;
|
|
68
|
+
limit: number;
|
|
69
|
+
totalPages: number;
|
|
70
|
+
};
|
|
71
|
+
}
|
|
72
|
+
export declare const walletKeys: {
|
|
73
|
+
all: readonly ["wallet"];
|
|
74
|
+
active: () => readonly ["wallet", "active"];
|
|
75
|
+
history: (page?: number) => readonly ["wallet", "history", number | undefined];
|
|
76
|
+
redemptions: () => readonly ["redemptions"];
|
|
77
|
+
lookup: (qrCode: string) => readonly ["redemptions", "lookup", string];
|
|
78
|
+
spotRedemptions: (spotId: string, page?: number) => readonly ["redemptions", "spot", string, number | undefined];
|
|
79
|
+
};
|
|
80
|
+
/**
|
|
81
|
+
* Get user's active wallet items (pending redemptions)
|
|
82
|
+
*
|
|
83
|
+
* @endpoint GET /wallet
|
|
84
|
+
*/
|
|
85
|
+
export declare function useWallet(options?: Omit<UseQueryOptions<WalletRedemption[]>, 'queryKey' | 'queryFn'>): UseQueryResult<WalletRedemption[]>;
|
|
86
|
+
/**
|
|
87
|
+
* Get user's wallet history (all redemptions)
|
|
88
|
+
*
|
|
89
|
+
* @endpoint GET /wallet/history
|
|
90
|
+
*/
|
|
91
|
+
export declare function useWalletHistory(params?: {
|
|
92
|
+
page?: number;
|
|
93
|
+
limit?: number;
|
|
94
|
+
}, options?: Omit<UseQueryOptions<WalletHistoryResponse>, 'queryKey' | 'queryFn'>): UseQueryResult<WalletHistoryResponse>;
|
|
95
|
+
/**
|
|
96
|
+
* Lookup redemption by QR code (for scanner)
|
|
97
|
+
*
|
|
98
|
+
* @endpoint GET /redemptions/lookup/:qrCode
|
|
99
|
+
*/
|
|
100
|
+
export declare function useRedemptionLookup(qrCode: string, options?: Omit<UseQueryOptions<RedemptionLookup>, 'queryKey' | 'queryFn'>): UseQueryResult<RedemptionLookup>;
|
|
101
|
+
/**
|
|
102
|
+
* Get spot's redemption history (for spot owner)
|
|
103
|
+
*
|
|
104
|
+
* @endpoint GET /spots/:spotId/redemptions
|
|
105
|
+
*/
|
|
106
|
+
export declare function useSpotRedemptions(spotId: string, params?: {
|
|
107
|
+
page?: number;
|
|
108
|
+
limit?: number;
|
|
109
|
+
}, options?: Omit<UseQueryOptions<WalletHistoryResponse>, 'queryKey' | 'queryFn'>): UseQueryResult<WalletHistoryResponse>;
|
|
@@ -0,0 +1,136 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* Wallet Query Hooks
|
|
4
|
+
*
|
|
5
|
+
* TanStack Query hooks for wallet and redemption operations.
|
|
6
|
+
*/
|
|
7
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
8
|
+
exports.walletKeys = void 0;
|
|
9
|
+
exports.useWallet = useWallet;
|
|
10
|
+
exports.useWalletHistory = useWalletHistory;
|
|
11
|
+
exports.useRedemptionLookup = useRedemptionLookup;
|
|
12
|
+
exports.useSpotRedemptions = useSpotRedemptions;
|
|
13
|
+
const react_query_1 = require("@tanstack/react-query");
|
|
14
|
+
const client_1 = require("../client");
|
|
15
|
+
// ============================================================================
|
|
16
|
+
// HELPER FUNCTIONS
|
|
17
|
+
// ============================================================================
|
|
18
|
+
function extractArrayData(data) {
|
|
19
|
+
if (Array.isArray(data)) {
|
|
20
|
+
return data;
|
|
21
|
+
}
|
|
22
|
+
if (data && typeof data === 'object' && 'data' in data) {
|
|
23
|
+
const nested = data.data;
|
|
24
|
+
if (Array.isArray(nested)) {
|
|
25
|
+
return nested;
|
|
26
|
+
}
|
|
27
|
+
}
|
|
28
|
+
return [];
|
|
29
|
+
}
|
|
30
|
+
function extractObjectData(data) {
|
|
31
|
+
if (data &&
|
|
32
|
+
typeof data === 'object' &&
|
|
33
|
+
'data' in data &&
|
|
34
|
+
!Array.isArray(data)) {
|
|
35
|
+
const nested = data.data;
|
|
36
|
+
if (nested &&
|
|
37
|
+
typeof nested === 'object' &&
|
|
38
|
+
'data' in nested &&
|
|
39
|
+
!Array.isArray(nested)) {
|
|
40
|
+
return nested.data;
|
|
41
|
+
}
|
|
42
|
+
return nested;
|
|
43
|
+
}
|
|
44
|
+
return data;
|
|
45
|
+
}
|
|
46
|
+
// ============================================================================
|
|
47
|
+
// QUERY KEYS
|
|
48
|
+
// ============================================================================
|
|
49
|
+
exports.walletKeys = {
|
|
50
|
+
all: ['wallet'],
|
|
51
|
+
active: () => [...exports.walletKeys.all, 'active'],
|
|
52
|
+
history: (page) => [...exports.walletKeys.all, 'history', page],
|
|
53
|
+
redemptions: () => ['redemptions'],
|
|
54
|
+
lookup: (qrCode) => [...exports.walletKeys.redemptions(), 'lookup', qrCode],
|
|
55
|
+
spotRedemptions: (spotId, page) => [...exports.walletKeys.redemptions(), 'spot', spotId, page],
|
|
56
|
+
};
|
|
57
|
+
// ============================================================================
|
|
58
|
+
// QUERY HOOKS
|
|
59
|
+
// ============================================================================
|
|
60
|
+
/**
|
|
61
|
+
* Get user's active wallet items (pending redemptions)
|
|
62
|
+
*
|
|
63
|
+
* @endpoint GET /wallet
|
|
64
|
+
*/
|
|
65
|
+
function useWallet(options) {
|
|
66
|
+
return (0, react_query_1.useQuery)({
|
|
67
|
+
queryKey: exports.walletKeys.active(),
|
|
68
|
+
queryFn: async () => {
|
|
69
|
+
const client = (0, client_1.getApiClient)();
|
|
70
|
+
const response = await client.get('/wallet');
|
|
71
|
+
return extractArrayData(response.data.data);
|
|
72
|
+
},
|
|
73
|
+
...options,
|
|
74
|
+
});
|
|
75
|
+
}
|
|
76
|
+
/**
|
|
77
|
+
* Get user's wallet history (all redemptions)
|
|
78
|
+
*
|
|
79
|
+
* @endpoint GET /wallet/history
|
|
80
|
+
*/
|
|
81
|
+
function useWalletHistory(params, options) {
|
|
82
|
+
return (0, react_query_1.useQuery)({
|
|
83
|
+
queryKey: exports.walletKeys.history(params?.page),
|
|
84
|
+
queryFn: async () => {
|
|
85
|
+
const client = (0, client_1.getApiClient)();
|
|
86
|
+
const queryParams = new URLSearchParams();
|
|
87
|
+
if (params?.page)
|
|
88
|
+
queryParams.set('page', String(params.page));
|
|
89
|
+
if (params?.limit)
|
|
90
|
+
queryParams.set('limit', String(params.limit));
|
|
91
|
+
const response = await client.get(`/wallet/history?${queryParams}`);
|
|
92
|
+
return extractObjectData(response.data.data);
|
|
93
|
+
},
|
|
94
|
+
...options,
|
|
95
|
+
});
|
|
96
|
+
}
|
|
97
|
+
/**
|
|
98
|
+
* Lookup redemption by QR code (for scanner)
|
|
99
|
+
*
|
|
100
|
+
* @endpoint GET /redemptions/lookup/:qrCode
|
|
101
|
+
*/
|
|
102
|
+
function useRedemptionLookup(qrCode, options) {
|
|
103
|
+
return (0, react_query_1.useQuery)({
|
|
104
|
+
queryKey: exports.walletKeys.lookup(qrCode),
|
|
105
|
+
queryFn: async () => {
|
|
106
|
+
const client = (0, client_1.getApiClient)();
|
|
107
|
+
const response = await client.get(`/redemptions/lookup/${qrCode}`);
|
|
108
|
+
return extractObjectData(response.data.data);
|
|
109
|
+
},
|
|
110
|
+
enabled: !!qrCode && qrCode.length > 0,
|
|
111
|
+
...options,
|
|
112
|
+
});
|
|
113
|
+
}
|
|
114
|
+
/**
|
|
115
|
+
* Get spot's redemption history (for spot owner)
|
|
116
|
+
*
|
|
117
|
+
* @endpoint GET /spots/:spotId/redemptions
|
|
118
|
+
*/
|
|
119
|
+
function useSpotRedemptions(spotId, params, options) {
|
|
120
|
+
return (0, react_query_1.useQuery)({
|
|
121
|
+
queryKey: exports.walletKeys.spotRedemptions(spotId, params?.page),
|
|
122
|
+
queryFn: async () => {
|
|
123
|
+
const client = (0, client_1.getApiClient)();
|
|
124
|
+
const queryParams = new URLSearchParams();
|
|
125
|
+
if (params?.page)
|
|
126
|
+
queryParams.set('page', String(params.page));
|
|
127
|
+
if (params?.limit)
|
|
128
|
+
queryParams.set('limit', String(params.limit));
|
|
129
|
+
const response = await client.get(`/spots/${spotId}/redemptions?${queryParams}`);
|
|
130
|
+
return extractObjectData(response.data.data);
|
|
131
|
+
},
|
|
132
|
+
enabled: !!spotId,
|
|
133
|
+
...options,
|
|
134
|
+
});
|
|
135
|
+
}
|
|
136
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"wallet.js","sourceRoot":"","sources":["../../../src/api/queries/wallet.ts"],"names":[],"mappings":";AAAA;;;;GAIG;;;AAoJH,8BAYC;AAOD,4CAkBC;AAOD,kDAgBC;AAOD,gDAoBC;AAzOD,uDAI8B;AAE9B,sCAAsC;AA+EtC,+EAA+E;AAC/E,mBAAmB;AACnB,+EAA+E;AAE/E,SAAS,gBAAgB,CAAI,IAAa;IACxC,IAAI,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,CAAC;QACxB,OAAO,IAAI,CAAA;IACb,CAAC;IACD,IAAI,IAAI,IAAI,OAAO,IAAI,KAAK,QAAQ,IAAI,MAAM,IAAI,IAAI,EAAE,CAAC;QACvD,MAAM,MAAM,GAAI,IAAwB,CAAC,IAAI,CAAA;QAC7C,IAAI,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC;YAC1B,OAAO,MAAM,CAAA;QACf,CAAC;IACH,CAAC;IACD,OAAO,EAAE,CAAA;AACX,CAAC;AAED,SAAS,iBAAiB,CAAI,IAAa;IACzC,IACE,IAAI;QACJ,OAAO,IAAI,KAAK,QAAQ;QACxB,MAAM,IAAI,IAAI;QACd,CAAC,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,EACpB,CAAC;QACD,MAAM,MAAM,GAAI,IAAwB,CAAC,IAAI,CAAA;QAC7C,IACE,MAAM;YACN,OAAO,MAAM,KAAK,QAAQ;YAC1B,MAAM,IAAI,MAAM;YAChB,CAAC,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,EACtB,CAAC;YACD,OAAQ,MAAoB,CAAC,IAAI,CAAA;QACnC,CAAC;QACD,OAAO,MAAW,CAAA;IACpB,CAAC;IACD,OAAO,IAAS,CAAA;AAClB,CAAC;AAED,+EAA+E;AAC/E,aAAa;AACb,+EAA+E;AAElE,QAAA,UAAU,GAAG;IACxB,GAAG,EAAE,CAAC,QAAQ,CAAU;IACxB,MAAM,EAAE,GAAG,EAAE,CAAC,CAAC,GAAG,kBAAU,CAAC,GAAG,EAAE,QAAQ,CAAU;IACpD,OAAO,EAAE,CAAC,IAAa,EAAE,EAAE,CAAC,CAAC,GAAG,kBAAU,CAAC,GAAG,EAAE,SAAS,EAAE,IAAI,CAAU;IACzE,WAAW,EAAE,GAAG,EAAE,CAAC,CAAC,aAAa,CAAU;IAC3C,MAAM,EAAE,CAAC,MAAc,EAAE,EAAE,CAAC,CAAC,GAAG,kBAAU,CAAC,WAAW,EAAE,EAAE,QAAQ,EAAE,MAAM,CAAU;IACpF,eAAe,EAAE,CAAC,MAAc,EAAE,IAAa,EAAE,EAAE,CACjD,CAAC,GAAG,kBAAU,CAAC,WAAW,EAAE,EAAE,MAAM,EAAE,MAAM,EAAE,IAAI,CAAU;CAC/D,CAAA;AAED,+EAA+E;AAC/E,cAAc;AACd,+EAA+E;AAE/E;;;;GAIG;AACH,SAAgB,SAAS,CACvB,OAA2E;IAE3E,OAAO,IAAA,sBAAQ,EAAC;QACd,QAAQ,EAAE,kBAAU,CAAC,MAAM,EAAE;QAC7B,OAAO,EAAE,KAAK,IAAiC,EAAE;YAC/C,MAAM,MAAM,GAAG,IAAA,qBAAY,GAAE,CAAA;YAC7B,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,GAAG,CAAuB,SAAS,CAAC,CAAA;YAClE,OAAO,gBAAgB,CAAmB,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;QAC/D,CAAC;QACD,GAAG,OAAO;KACX,CAAC,CAAA;AACJ,CAAC;AAED;;;;GAIG;AACH,SAAgB,gBAAgB,CAC9B,MAAwC,EACxC,OAA8E;IAE9E,OAAO,IAAA,sBAAQ,EAAC;QACd,QAAQ,EAAE,kBAAU,CAAC,OAAO,CAAC,MAAM,EAAE,IAAI,CAAC;QAC1C,OAAO,EAAE,KAAK,IAAoC,EAAE;YAClD,MAAM,MAAM,GAAG,IAAA,qBAAY,GAAE,CAAA;YAC7B,MAAM,WAAW,GAAG,IAAI,eAAe,EAAE,CAAA;YACzC,IAAI,MAAM,EAAE,IAAI;gBAAE,WAAW,CAAC,GAAG,CAAC,MAAM,EAAE,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAA;YAC9D,IAAI,MAAM,EAAE,KAAK;gBAAE,WAAW,CAAC,GAAG,CAAC,OAAO,EAAE,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAA;YACjE,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,GAAG,CAC/B,mBAAmB,WAAW,EAAE,CACjC,CAAA;YACD,OAAO,iBAAiB,CAAwB,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;QACrE,CAAC;QACD,GAAG,OAAO;KACX,CAAC,CAAA;AACJ,CAAC;AAED;;;;GAIG;AACH,SAAgB,mBAAmB,CACjC,MAAc,EACd,OAAyE;IAEzE,OAAO,IAAA,sBAAQ,EAAC;QACd,QAAQ,EAAE,kBAAU,CAAC,MAAM,CAAC,MAAM,CAAC;QACnC,OAAO,EAAE,KAAK,IAA+B,EAAE;YAC7C,MAAM,MAAM,GAAG,IAAA,qBAAY,GAAE,CAAA;YAC7B,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,GAAG,CAC/B,uBAAuB,MAAM,EAAE,CAChC,CAAA;YACD,OAAO,iBAAiB,CAAmB,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;QAChE,CAAC;QACD,OAAO,EAAE,CAAC,CAAC,MAAM,IAAI,MAAM,CAAC,MAAM,GAAG,CAAC;QACtC,GAAG,OAAO;KACX,CAAC,CAAA;AACJ,CAAC;AAED;;;;GAIG;AACH,SAAgB,kBAAkB,CAChC,MAAc,EACd,MAAwC,EACxC,OAA8E;IAE9E,OAAO,IAAA,sBAAQ,EAAC;QACd,QAAQ,EAAE,kBAAU,CAAC,eAAe,CAAC,MAAM,EAAE,MAAM,EAAE,IAAI,CAAC;QAC1D,OAAO,EAAE,KAAK,IAAoC,EAAE;YAClD,MAAM,MAAM,GAAG,IAAA,qBAAY,GAAE,CAAA;YAC7B,MAAM,WAAW,GAAG,IAAI,eAAe,EAAE,CAAA;YACzC,IAAI,MAAM,EAAE,IAAI;gBAAE,WAAW,CAAC,GAAG,CAAC,MAAM,EAAE,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAA;YAC9D,IAAI,MAAM,EAAE,KAAK;gBAAE,WAAW,CAAC,GAAG,CAAC,OAAO,EAAE,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAA;YACjE,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,GAAG,CAC/B,UAAU,MAAM,gBAAgB,WAAW,EAAE,CAC9C,CAAA;YACD,OAAO,iBAAiB,CAAwB,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;QACrE,CAAC;QACD,OAAO,EAAE,CAAC,CAAC,MAAM;QACjB,GAAG,OAAO;KACX,CAAC,CAAA;AACJ,CAAC","sourcesContent":["/**\n * Wallet Query Hooks\n *\n * TanStack Query hooks for wallet and redemption operations.\n */\n\nimport {\n  useQuery,\n  type UseQueryOptions,\n  type UseQueryResult,\n} from '@tanstack/react-query'\n\nimport {getApiClient} from '../client'\nimport {type ApiResponse} from '../types'\n\n// ============================================================================\n// TYPES\n// ============================================================================\n\nexport type RedemptionStatus = 'PENDING' | 'REDEEMED' | 'EXPIRED' | 'CANCELLED' | 'VOID'\n\nexport interface WalletProduct {\n  id: string\n  name: string\n  type: string\n  imageUrl: string | null\n}\n\nexport interface WalletOrderItem {\n  product: WalletProduct\n  quantity: number\n  unitPrice: number\n}\n\nexport interface WalletOrder {\n  id: string\n  orderNumber: string\n  spotId: string\n  createdAt: string\n  spot: {\n    id: string\n    name: string\n    slug: string\n    address: string | null\n  }\n}\n\nexport interface WalletRedemption {\n  id: string\n  qrCode: string\n  qrCodeUrl: string | null\n  status: RedemptionStatus\n  validFrom: string\n  validUntil: string\n  redeemedAt: string | null\n  createdAt: string\n  orderItem: WalletOrderItem & {\n    order: WalletOrder\n  }\n}\n\nexport interface RedemptionLookup extends WalletRedemption {\n  redeemedBy: {\n    id: string\n    name: string\n  } | null\n  spot: {\n    id: string\n    name: string\n  } | null\n  orderItem: WalletOrderItem & {\n    order: WalletOrder & {\n      user: {\n        id: string\n        name: string | null\n        avatarUrl: string | null\n      }\n    }\n  }\n}\n\nexport interface WalletHistoryResponse {\n  data: WalletRedemption[]\n  meta: {\n    total: number\n    page: number\n    limit: number\n    totalPages: number\n  }\n}\n\n// ============================================================================\n// HELPER FUNCTIONS\n// ============================================================================\n\nfunction extractArrayData<T>(data: unknown): T[] {\n  if (Array.isArray(data)) {\n    return data\n  }\n  if (data && typeof data === 'object' && 'data' in data) {\n    const nested = (data as {data: unknown}).data\n    if (Array.isArray(nested)) {\n      return nested\n    }\n  }\n  return []\n}\n\nfunction extractObjectData<T>(data: unknown): T {\n  if (\n    data &&\n    typeof data === 'object' &&\n    'data' in data &&\n    !Array.isArray(data)\n  ) {\n    const nested = (data as {data: unknown}).data\n    if (\n      nested &&\n      typeof nested === 'object' &&\n      'data' in nested &&\n      !Array.isArray(nested)\n    ) {\n      return (nested as {data: T}).data\n    }\n    return nested as T\n  }\n  return data as T\n}\n\n// ============================================================================\n// QUERY KEYS\n// ============================================================================\n\nexport const walletKeys = {\n  all: ['wallet'] as const,\n  active: () => [...walletKeys.all, 'active'] as const,\n  history: (page?: number) => [...walletKeys.all, 'history', page] as const,\n  redemptions: () => ['redemptions'] as const,\n  lookup: (qrCode: string) => [...walletKeys.redemptions(), 'lookup', qrCode] as const,\n  spotRedemptions: (spotId: string, page?: number) =>\n    [...walletKeys.redemptions(), 'spot', spotId, page] as const,\n}\n\n// ============================================================================\n// QUERY HOOKS\n// ============================================================================\n\n/**\n * Get user's active wallet items (pending redemptions)\n *\n * @endpoint GET /wallet\n */\nexport function useWallet(\n  options?: Omit<UseQueryOptions<WalletRedemption[]>, 'queryKey' | 'queryFn'>,\n): UseQueryResult<WalletRedemption[]> {\n  return useQuery({\n    queryKey: walletKeys.active(),\n    queryFn: async (): Promise<WalletRedemption[]> => {\n      const client = getApiClient()\n      const response = await client.get<ApiResponse<unknown>>('/wallet')\n      return extractArrayData<WalletRedemption>(response.data.data)\n    },\n    ...options,\n  })\n}\n\n/**\n * Get user's wallet history (all redemptions)\n *\n * @endpoint GET /wallet/history\n */\nexport function useWalletHistory(\n  params?: {page?: number; limit?: number},\n  options?: Omit<UseQueryOptions<WalletHistoryResponse>, 'queryKey' | 'queryFn'>,\n): UseQueryResult<WalletHistoryResponse> {\n  return useQuery({\n    queryKey: walletKeys.history(params?.page),\n    queryFn: async (): Promise<WalletHistoryResponse> => {\n      const client = getApiClient()\n      const queryParams = new URLSearchParams()\n      if (params?.page) queryParams.set('page', String(params.page))\n      if (params?.limit) queryParams.set('limit', String(params.limit))\n      const response = await client.get<ApiResponse<WalletHistoryResponse>>(\n        `/wallet/history?${queryParams}`,\n      )\n      return extractObjectData<WalletHistoryResponse>(response.data.data)\n    },\n    ...options,\n  })\n}\n\n/**\n * Lookup redemption by QR code (for scanner)\n *\n * @endpoint GET /redemptions/lookup/:qrCode\n */\nexport function useRedemptionLookup(\n  qrCode: string,\n  options?: Omit<UseQueryOptions<RedemptionLookup>, 'queryKey' | 'queryFn'>,\n): UseQueryResult<RedemptionLookup> {\n  return useQuery({\n    queryKey: walletKeys.lookup(qrCode),\n    queryFn: async (): Promise<RedemptionLookup> => {\n      const client = getApiClient()\n      const response = await client.get<ApiResponse<unknown>>(\n        `/redemptions/lookup/${qrCode}`,\n      )\n      return extractObjectData<RedemptionLookup>(response.data.data)\n    },\n    enabled: !!qrCode && qrCode.length > 0,\n    ...options,\n  })\n}\n\n/**\n * Get spot's redemption history (for spot owner)\n *\n * @endpoint GET /spots/:spotId/redemptions\n */\nexport function useSpotRedemptions(\n  spotId: string,\n  params?: {page?: number; limit?: number},\n  options?: Omit<UseQueryOptions<WalletHistoryResponse>, 'queryKey' | 'queryFn'>,\n): UseQueryResult<WalletHistoryResponse> {\n  return useQuery({\n    queryKey: walletKeys.spotRedemptions(spotId, params?.page),\n    queryFn: async (): Promise<WalletHistoryResponse> => {\n      const client = getApiClient()\n      const queryParams = new URLSearchParams()\n      if (params?.page) queryParams.set('page', String(params.page))\n      if (params?.limit) queryParams.set('limit', String(params.limit))\n      const response = await client.get<ApiResponse<WalletHistoryResponse>>(\n        `/spots/${spotId}/redemptions?${queryParams}`,\n      )\n      return extractObjectData<WalletHistoryResponse>(response.data.data)\n    },\n    enabled: !!spotId,\n    ...options,\n  })\n}\n"]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@spotsdev/sdk",
|
|
3
|
-
"version": "1.
|
|
3
|
+
"version": "1.3.3",
|
|
4
4
|
"description": "Shared TypeScript SDK for Spots API - TanStack Query hooks, API client, and utilities",
|
|
5
5
|
"main": "dist/index.js",
|
|
6
6
|
"source": "src/index.ts",
|
|
@@ -32,7 +32,7 @@
|
|
|
32
32
|
"react": ">=18.0.0"
|
|
33
33
|
},
|
|
34
34
|
"dependencies": {
|
|
35
|
-
"@spotsdev/types": "^1.
|
|
35
|
+
"@spotsdev/types": "^1.1.0"
|
|
36
36
|
},
|
|
37
37
|
"devDependencies": {
|
|
38
38
|
"@types/node": "^20.0.0",
|
package/src/api/entities.ts
CHANGED
|
@@ -65,6 +65,14 @@ export type NotificationType =
|
|
|
65
65
|
// ENTITY INTERFACES
|
|
66
66
|
// ============================================================================
|
|
67
67
|
|
|
68
|
+
export interface UserVibe {
|
|
69
|
+
id: string
|
|
70
|
+
name: string
|
|
71
|
+
emoji: string | null
|
|
72
|
+
slug?: string
|
|
73
|
+
category?: string
|
|
74
|
+
}
|
|
75
|
+
|
|
68
76
|
export interface User {
|
|
69
77
|
id: string
|
|
70
78
|
phone: string | null
|
|
@@ -74,6 +82,7 @@ export interface User {
|
|
|
74
82
|
avatarUrl: string | null
|
|
75
83
|
status: AccountStatus
|
|
76
84
|
onboardingCompleted: boolean
|
|
85
|
+
vibes?: UserVibe[]
|
|
77
86
|
lastActiveAt: Date | null
|
|
78
87
|
createdAt: Date
|
|
79
88
|
updatedAt: Date
|
|
@@ -82,11 +82,12 @@ export function useUpdatePost(
|
|
|
82
82
|
const queryClient = useQueryClient()
|
|
83
83
|
|
|
84
84
|
return useMutation({
|
|
85
|
-
mutationFn: async ({postId,
|
|
85
|
+
mutationFn: async ({postId, title, content}): Promise<Post> => {
|
|
86
86
|
const client = getApiClient()
|
|
87
|
+
// API expects 'description' field, not 'content'
|
|
87
88
|
const response = await client.put<ApiResponse<Post>>(
|
|
88
89
|
`/posts/${postId}`,
|
|
89
|
-
|
|
90
|
+
{title, description: content},
|
|
90
91
|
)
|
|
91
92
|
return response.data.data
|
|
92
93
|
},
|
|
@@ -0,0 +1,91 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Redemption Mutation Hooks
|
|
3
|
+
*
|
|
4
|
+
* TanStack Query mutations for redemption operations.
|
|
5
|
+
*/
|
|
6
|
+
|
|
7
|
+
import {useMutation, useQueryClient, type UseMutationResult} from '@tanstack/react-query'
|
|
8
|
+
import {getApiClient} from '../client'
|
|
9
|
+
import {walletKeys, type WalletRedemption} from '../queries/wallet'
|
|
10
|
+
|
|
11
|
+
// ============================================================================
|
|
12
|
+
// TYPES
|
|
13
|
+
// ============================================================================
|
|
14
|
+
|
|
15
|
+
export interface RedeemRequest {
|
|
16
|
+
qrCode: string
|
|
17
|
+
spotId?: string
|
|
18
|
+
deviceInfo?: string
|
|
19
|
+
latitude?: number
|
|
20
|
+
longitude?: number
|
|
21
|
+
notes?: string
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
export interface VoidRedemptionRequest {
|
|
25
|
+
reason: string
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
// ============================================================================
|
|
29
|
+
// MUTATION HOOKS
|
|
30
|
+
// ============================================================================
|
|
31
|
+
|
|
32
|
+
/**
|
|
33
|
+
* Redeem a QR code (spot owner/staff action)
|
|
34
|
+
*
|
|
35
|
+
* @endpoint POST /redemptions/redeem
|
|
36
|
+
*/
|
|
37
|
+
export function useRedeem(): UseMutationResult<WalletRedemption, Error, RedeemRequest> {
|
|
38
|
+
const queryClient = useQueryClient()
|
|
39
|
+
|
|
40
|
+
return useMutation({
|
|
41
|
+
mutationFn: async (request: RedeemRequest): Promise<WalletRedemption> => {
|
|
42
|
+
const client = getApiClient()
|
|
43
|
+
const response = await client.post('/redemptions/redeem', request)
|
|
44
|
+
return response.data.data
|
|
45
|
+
},
|
|
46
|
+
onSuccess: (_data, variables) => {
|
|
47
|
+
// Invalidate redemption lookup for this QR code
|
|
48
|
+
queryClient.invalidateQueries({
|
|
49
|
+
queryKey: walletKeys.lookup(variables.qrCode),
|
|
50
|
+
})
|
|
51
|
+
// Invalidate spot redemptions if spotId provided
|
|
52
|
+
if (variables.spotId) {
|
|
53
|
+
queryClient.invalidateQueries({
|
|
54
|
+
queryKey: walletKeys.spotRedemptions(variables.spotId),
|
|
55
|
+
})
|
|
56
|
+
}
|
|
57
|
+
},
|
|
58
|
+
})
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
/**
|
|
62
|
+
* Void a redemption (spot owner action)
|
|
63
|
+
*
|
|
64
|
+
* @endpoint PUT /redemptions/:id/void
|
|
65
|
+
*/
|
|
66
|
+
export function useVoidRedemption(): UseMutationResult<
|
|
67
|
+
WalletRedemption,
|
|
68
|
+
Error,
|
|
69
|
+
{redemptionId: string; reason: string}
|
|
70
|
+
> {
|
|
71
|
+
const queryClient = useQueryClient()
|
|
72
|
+
|
|
73
|
+
return useMutation({
|
|
74
|
+
mutationFn: async ({
|
|
75
|
+
redemptionId,
|
|
76
|
+
reason,
|
|
77
|
+
}: {
|
|
78
|
+
redemptionId: string
|
|
79
|
+
reason: string
|
|
80
|
+
}): Promise<WalletRedemption> => {
|
|
81
|
+
const client = getApiClient()
|
|
82
|
+
const response = await client.put(`/redemptions/${redemptionId}/void`, {reason})
|
|
83
|
+
return response.data.data
|
|
84
|
+
},
|
|
85
|
+
onSuccess: () => {
|
|
86
|
+
// Invalidate wallet queries
|
|
87
|
+
queryClient.invalidateQueries({queryKey: walletKeys.all})
|
|
88
|
+
queryClient.invalidateQueries({queryKey: walletKeys.redemptions()})
|
|
89
|
+
},
|
|
90
|
+
})
|
|
91
|
+
}
|
package/src/api/queries/index.ts
CHANGED
|
@@ -0,0 +1,240 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Wallet Query Hooks
|
|
3
|
+
*
|
|
4
|
+
* TanStack Query hooks for wallet and redemption operations.
|
|
5
|
+
*/
|
|
6
|
+
|
|
7
|
+
import {
|
|
8
|
+
useQuery,
|
|
9
|
+
type UseQueryOptions,
|
|
10
|
+
type UseQueryResult,
|
|
11
|
+
} from '@tanstack/react-query'
|
|
12
|
+
|
|
13
|
+
import {getApiClient} from '../client'
|
|
14
|
+
import {type ApiResponse} from '../types'
|
|
15
|
+
|
|
16
|
+
// ============================================================================
|
|
17
|
+
// TYPES
|
|
18
|
+
// ============================================================================
|
|
19
|
+
|
|
20
|
+
export type RedemptionStatus = 'PENDING' | 'REDEEMED' | 'EXPIRED' | 'CANCELLED' | 'VOID'
|
|
21
|
+
|
|
22
|
+
export interface WalletProduct {
|
|
23
|
+
id: string
|
|
24
|
+
name: string
|
|
25
|
+
type: string
|
|
26
|
+
imageUrl: string | null
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
export interface WalletOrderItem {
|
|
30
|
+
product: WalletProduct
|
|
31
|
+
quantity: number
|
|
32
|
+
unitPrice: number
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
export interface WalletOrder {
|
|
36
|
+
id: string
|
|
37
|
+
orderNumber: string
|
|
38
|
+
spotId: string
|
|
39
|
+
createdAt: string
|
|
40
|
+
spot: {
|
|
41
|
+
id: string
|
|
42
|
+
name: string
|
|
43
|
+
slug: string
|
|
44
|
+
address: string | null
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
export interface WalletRedemption {
|
|
49
|
+
id: string
|
|
50
|
+
qrCode: string
|
|
51
|
+
qrCodeUrl: string | null
|
|
52
|
+
status: RedemptionStatus
|
|
53
|
+
validFrom: string
|
|
54
|
+
validUntil: string
|
|
55
|
+
redeemedAt: string | null
|
|
56
|
+
createdAt: string
|
|
57
|
+
orderItem: WalletOrderItem & {
|
|
58
|
+
order: WalletOrder
|
|
59
|
+
}
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
export interface RedemptionLookup extends WalletRedemption {
|
|
63
|
+
redeemedBy: {
|
|
64
|
+
id: string
|
|
65
|
+
name: string
|
|
66
|
+
} | null
|
|
67
|
+
spot: {
|
|
68
|
+
id: string
|
|
69
|
+
name: string
|
|
70
|
+
} | null
|
|
71
|
+
orderItem: WalletOrderItem & {
|
|
72
|
+
order: WalletOrder & {
|
|
73
|
+
user: {
|
|
74
|
+
id: string
|
|
75
|
+
name: string | null
|
|
76
|
+
avatarUrl: string | null
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
export interface WalletHistoryResponse {
|
|
83
|
+
data: WalletRedemption[]
|
|
84
|
+
meta: {
|
|
85
|
+
total: number
|
|
86
|
+
page: number
|
|
87
|
+
limit: number
|
|
88
|
+
totalPages: number
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
// ============================================================================
|
|
93
|
+
// HELPER FUNCTIONS
|
|
94
|
+
// ============================================================================
|
|
95
|
+
|
|
96
|
+
function extractArrayData<T>(data: unknown): T[] {
|
|
97
|
+
if (Array.isArray(data)) {
|
|
98
|
+
return data
|
|
99
|
+
}
|
|
100
|
+
if (data && typeof data === 'object' && 'data' in data) {
|
|
101
|
+
const nested = (data as {data: unknown}).data
|
|
102
|
+
if (Array.isArray(nested)) {
|
|
103
|
+
return nested
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
return []
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
function extractObjectData<T>(data: unknown): T {
|
|
110
|
+
if (
|
|
111
|
+
data &&
|
|
112
|
+
typeof data === 'object' &&
|
|
113
|
+
'data' in data &&
|
|
114
|
+
!Array.isArray(data)
|
|
115
|
+
) {
|
|
116
|
+
const nested = (data as {data: unknown}).data
|
|
117
|
+
if (
|
|
118
|
+
nested &&
|
|
119
|
+
typeof nested === 'object' &&
|
|
120
|
+
'data' in nested &&
|
|
121
|
+
!Array.isArray(nested)
|
|
122
|
+
) {
|
|
123
|
+
return (nested as {data: T}).data
|
|
124
|
+
}
|
|
125
|
+
return nested as T
|
|
126
|
+
}
|
|
127
|
+
return data as T
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
// ============================================================================
|
|
131
|
+
// QUERY KEYS
|
|
132
|
+
// ============================================================================
|
|
133
|
+
|
|
134
|
+
export const walletKeys = {
|
|
135
|
+
all: ['wallet'] as const,
|
|
136
|
+
active: () => [...walletKeys.all, 'active'] as const,
|
|
137
|
+
history: (page?: number) => [...walletKeys.all, 'history', page] as const,
|
|
138
|
+
redemptions: () => ['redemptions'] as const,
|
|
139
|
+
lookup: (qrCode: string) => [...walletKeys.redemptions(), 'lookup', qrCode] as const,
|
|
140
|
+
spotRedemptions: (spotId: string, page?: number) =>
|
|
141
|
+
[...walletKeys.redemptions(), 'spot', spotId, page] as const,
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
// ============================================================================
|
|
145
|
+
// QUERY HOOKS
|
|
146
|
+
// ============================================================================
|
|
147
|
+
|
|
148
|
+
/**
|
|
149
|
+
* Get user's active wallet items (pending redemptions)
|
|
150
|
+
*
|
|
151
|
+
* @endpoint GET /wallet
|
|
152
|
+
*/
|
|
153
|
+
export function useWallet(
|
|
154
|
+
options?: Omit<UseQueryOptions<WalletRedemption[]>, 'queryKey' | 'queryFn'>,
|
|
155
|
+
): UseQueryResult<WalletRedemption[]> {
|
|
156
|
+
return useQuery({
|
|
157
|
+
queryKey: walletKeys.active(),
|
|
158
|
+
queryFn: async (): Promise<WalletRedemption[]> => {
|
|
159
|
+
const client = getApiClient()
|
|
160
|
+
const response = await client.get<ApiResponse<unknown>>('/wallet')
|
|
161
|
+
return extractArrayData<WalletRedemption>(response.data.data)
|
|
162
|
+
},
|
|
163
|
+
...options,
|
|
164
|
+
})
|
|
165
|
+
}
|
|
166
|
+
|
|
167
|
+
/**
|
|
168
|
+
* Get user's wallet history (all redemptions)
|
|
169
|
+
*
|
|
170
|
+
* @endpoint GET /wallet/history
|
|
171
|
+
*/
|
|
172
|
+
export function useWalletHistory(
|
|
173
|
+
params?: {page?: number; limit?: number},
|
|
174
|
+
options?: Omit<UseQueryOptions<WalletHistoryResponse>, 'queryKey' | 'queryFn'>,
|
|
175
|
+
): UseQueryResult<WalletHistoryResponse> {
|
|
176
|
+
return useQuery({
|
|
177
|
+
queryKey: walletKeys.history(params?.page),
|
|
178
|
+
queryFn: async (): Promise<WalletHistoryResponse> => {
|
|
179
|
+
const client = getApiClient()
|
|
180
|
+
const queryParams = new URLSearchParams()
|
|
181
|
+
if (params?.page) queryParams.set('page', String(params.page))
|
|
182
|
+
if (params?.limit) queryParams.set('limit', String(params.limit))
|
|
183
|
+
const response = await client.get<ApiResponse<WalletHistoryResponse>>(
|
|
184
|
+
`/wallet/history?${queryParams}`,
|
|
185
|
+
)
|
|
186
|
+
return extractObjectData<WalletHistoryResponse>(response.data.data)
|
|
187
|
+
},
|
|
188
|
+
...options,
|
|
189
|
+
})
|
|
190
|
+
}
|
|
191
|
+
|
|
192
|
+
/**
|
|
193
|
+
* Lookup redemption by QR code (for scanner)
|
|
194
|
+
*
|
|
195
|
+
* @endpoint GET /redemptions/lookup/:qrCode
|
|
196
|
+
*/
|
|
197
|
+
export function useRedemptionLookup(
|
|
198
|
+
qrCode: string,
|
|
199
|
+
options?: Omit<UseQueryOptions<RedemptionLookup>, 'queryKey' | 'queryFn'>,
|
|
200
|
+
): UseQueryResult<RedemptionLookup> {
|
|
201
|
+
return useQuery({
|
|
202
|
+
queryKey: walletKeys.lookup(qrCode),
|
|
203
|
+
queryFn: async (): Promise<RedemptionLookup> => {
|
|
204
|
+
const client = getApiClient()
|
|
205
|
+
const response = await client.get<ApiResponse<unknown>>(
|
|
206
|
+
`/redemptions/lookup/${qrCode}`,
|
|
207
|
+
)
|
|
208
|
+
return extractObjectData<RedemptionLookup>(response.data.data)
|
|
209
|
+
},
|
|
210
|
+
enabled: !!qrCode && qrCode.length > 0,
|
|
211
|
+
...options,
|
|
212
|
+
})
|
|
213
|
+
}
|
|
214
|
+
|
|
215
|
+
/**
|
|
216
|
+
* Get spot's redemption history (for spot owner)
|
|
217
|
+
*
|
|
218
|
+
* @endpoint GET /spots/:spotId/redemptions
|
|
219
|
+
*/
|
|
220
|
+
export function useSpotRedemptions(
|
|
221
|
+
spotId: string,
|
|
222
|
+
params?: {page?: number; limit?: number},
|
|
223
|
+
options?: Omit<UseQueryOptions<WalletHistoryResponse>, 'queryKey' | 'queryFn'>,
|
|
224
|
+
): UseQueryResult<WalletHistoryResponse> {
|
|
225
|
+
return useQuery({
|
|
226
|
+
queryKey: walletKeys.spotRedemptions(spotId, params?.page),
|
|
227
|
+
queryFn: async (): Promise<WalletHistoryResponse> => {
|
|
228
|
+
const client = getApiClient()
|
|
229
|
+
const queryParams = new URLSearchParams()
|
|
230
|
+
if (params?.page) queryParams.set('page', String(params.page))
|
|
231
|
+
if (params?.limit) queryParams.set('limit', String(params.limit))
|
|
232
|
+
const response = await client.get<ApiResponse<WalletHistoryResponse>>(
|
|
233
|
+
`/spots/${spotId}/redemptions?${queryParams}`,
|
|
234
|
+
)
|
|
235
|
+
return extractObjectData<WalletHistoryResponse>(response.data.data)
|
|
236
|
+
},
|
|
237
|
+
enabled: !!spotId,
|
|
238
|
+
...options,
|
|
239
|
+
})
|
|
240
|
+
}
|