@spotsdev/sdk 1.1.0 → 1.3.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -35,6 +35,8 @@ export interface Spot {
35
35
  country: string | null;
36
36
  latitude: number | null;
37
37
  longitude: number | null;
38
+ lat?: number | null;
39
+ lng?: number | null;
38
40
  type: SpotType;
39
41
  googlePlaceId: string | null;
40
42
  website: string | null;
@@ -42,8 +44,8 @@ export interface Spot {
42
44
  imageUrl: string | null;
43
45
  ownerId: string | null;
44
46
  isVerified: boolean;
45
- createdAt: Date;
46
- updatedAt: Date;
47
+ createdAt: Date | string;
48
+ updatedAt: Date | string;
47
49
  }
48
50
  export interface SpotPost {
49
51
  id: string;
@@ -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  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  createdAt: Date\n  updatedAt: Date\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 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"]}
@@ -11,3 +11,4 @@ export * from './clubs';
11
11
  export * from './notifications';
12
12
  export * from './products';
13
13
  export * from './orders';
14
+ export * from './redemptions';
@@ -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
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvYXBpL211dGF0aW9ucy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUE7Ozs7R0FJRzs7Ozs7Ozs7Ozs7Ozs7OztBQUVILFFBQVE7QUFDUiwwQ0FBdUI7QUFFdkIsUUFBUTtBQUNSLDBDQUF1QjtBQUV2QixRQUFRO0FBQ1IsMENBQXVCO0FBRXZCLGdCQUFnQjtBQUNoQixrREFBK0I7QUFFL0IsUUFBUTtBQUNSLDBDQUF1QjtBQUV2QixnQkFBZ0I7QUFDaEIsa0RBQStCO0FBRS9CLHlCQUF5QjtBQUN6Qiw2Q0FBMEI7QUFFMUIsdUJBQXVCO0FBQ3ZCLDJDQUF3QiIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogU3BvdHMgU0RLIE11dGF0aW9uIEhvb2tzIEluZGV4XG4gKlxuICogUmUtZXhwb3J0cyBhbGwgbXV0YXRpb24gaG9va3MuXG4gKi9cblxuLy8gVXNlcnNcbmV4cG9ydCAqIGZyb20gJy4vdXNlcnMnXG5cbi8vIFBvc3RzXG5leHBvcnQgKiBmcm9tICcuL3Bvc3RzJ1xuXG4vLyBTcG90c1xuZXhwb3J0ICogZnJvbSAnLi9zcG90cydcblxuLy8gQ29udmVyc2F0aW9uc1xuZXhwb3J0ICogZnJvbSAnLi9jb252ZXJzYXRpb25zJ1xuXG4vLyBDbHVic1xuZXhwb3J0ICogZnJvbSAnLi9jbHVicydcblxuLy8gTm90aWZpY2F0aW9uc1xuZXhwb3J0ICogZnJvbSAnLi9ub3RpZmljYXRpb25zJ1xuXG4vLyBQcm9kdWN0cyAobWFya2V0cGxhY2UpXG5leHBvcnQgKiBmcm9tICcuL3Byb2R1Y3RzJ1xuXG4vLyBPcmRlcnMgKG1hcmtldHBsYWNlKVxuZXhwb3J0ICogZnJvbSAnLi9vcmRlcnMnXG4iXX0=
38
+ // Redemptions (wallet)
39
+ __exportStar(require("./redemptions"), exports);
40
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvYXBpL211dGF0aW9ucy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUE7Ozs7R0FJRzs7Ozs7Ozs7Ozs7Ozs7OztBQUVILFFBQVE7QUFDUiwwQ0FBdUI7QUFFdkIsUUFBUTtBQUNSLDBDQUF1QjtBQUV2QixRQUFRO0FBQ1IsMENBQXVCO0FBRXZCLGdCQUFnQjtBQUNoQixrREFBK0I7QUFFL0IsUUFBUTtBQUNSLDBDQUF1QjtBQUV2QixnQkFBZ0I7QUFDaEIsa0RBQStCO0FBRS9CLHlCQUF5QjtBQUN6Qiw2Q0FBMEI7QUFFMUIsdUJBQXVCO0FBQ3ZCLDJDQUF3QjtBQUV4Qix1QkFBdUI7QUFDdkIsZ0RBQTZCIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBTcG90cyBTREsgTXV0YXRpb24gSG9va3MgSW5kZXhcbiAqXG4gKiBSZS1leHBvcnRzIGFsbCBtdXRhdGlvbiBob29rcy5cbiAqL1xuXG4vLyBVc2Vyc1xuZXhwb3J0ICogZnJvbSAnLi91c2VycydcblxuLy8gUG9zdHNcbmV4cG9ydCAqIGZyb20gJy4vcG9zdHMnXG5cbi8vIFNwb3RzXG5leHBvcnQgKiBmcm9tICcuL3Nwb3RzJ1xuXG4vLyBDb252ZXJzYXRpb25zXG5leHBvcnQgKiBmcm9tICcuL2NvbnZlcnNhdGlvbnMnXG5cbi8vIENsdWJzXG5leHBvcnQgKiBmcm9tICcuL2NsdWJzJ1xuXG4vLyBOb3RpZmljYXRpb25zXG5leHBvcnQgKiBmcm9tICcuL25vdGlmaWNhdGlvbnMnXG5cbi8vIFByb2R1Y3RzIChtYXJrZXRwbGFjZSlcbmV4cG9ydCAqIGZyb20gJy4vcHJvZHVjdHMnXG5cbi8vIE9yZGVycyAobWFya2V0cGxhY2UpXG5leHBvcnQgKiBmcm9tICcuL29yZGVycydcblxuLy8gUmVkZW1wdGlvbnMgKHdhbGxldClcbmV4cG9ydCAqIGZyb20gJy4vcmVkZW1wdGlvbnMnXG4iXX0=
@@ -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=
@@ -24,3 +24,5 @@ export * from './products';
24
24
  export { productKeys } from './products';
25
25
  export * from './orders';
26
26
  export { orderKeys } from './orders';
27
+ export * from './wallet';
28
+ export { walletKeys } from './wallet';
@@ -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
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvYXBpL3F1ZXJpZXMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOzs7O0dBSUc7Ozs7Ozs7Ozs7Ozs7Ozs7O0FBRUgsNkRBQTZEO0FBQzdELHlDQUFzQjtBQUV0QixRQUFRO0FBQ1IsMENBQXVCO0FBQ3ZCLGlDQUFnQztBQUF4QixpR0FBQSxRQUFRLE9BQUE7QUFFaEIsUUFBUTtBQUNSLDBDQUF1QjtBQUN2QixpQ0FBZ0M7QUFBeEIsaUdBQUEsUUFBUSxPQUFBO0FBRWhCLFFBQVE7QUFDUiwwQ0FBdUI7QUFDdkIsaUNBQWdDO0FBQXhCLGlHQUFBLFFBQVEsT0FBQTtBQUVoQixnQkFBZ0I7QUFDaEIsa0RBQStCO0FBQy9CLGlEQUFnRDtBQUF4QyxpSEFBQSxnQkFBZ0IsT0FBQTtBQUV4QixRQUFRO0FBQ1IsMENBQXVCO0FBQ3ZCLGlDQUFnQztBQUF4QixpR0FBQSxRQUFRLE9BQUE7QUFFaEIsWUFBWTtBQUNaLDhDQUEyQjtBQUMzQix5Q0FBd0M7QUFBaEMseUdBQUEsWUFBWSxPQUFBO0FBRXBCLGdCQUFnQjtBQUNoQixrREFBK0I7QUFDL0IsaURBQWdEO0FBQXhDLGlIQUFBLGdCQUFnQixPQUFBO0FBRXhCLCtCQUErQjtBQUMvQix5Q0FBc0I7QUFDdEIsK0JBQStCO0FBQXZCLGdHQUFBLFFBQVEsT0FBQTtBQUVoQix5QkFBeUI7QUFDekIsNkNBQTBCO0FBQzFCLHVDQUFzQztBQUE5Qix1R0FBQSxXQUFXLE9BQUE7QUFFbkIsdUJBQXVCO0FBQ3ZCLDJDQUF3QjtBQUN4QixtQ0FBa0M7QUFBMUIsbUdBQUEsU0FBUyxPQUFBIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBTcG90cyBTREsgUXVlcnkgSG9va3MgSW5kZXhcbiAqXG4gKiBSZS1leHBvcnRzIGFsbCBxdWVyeSBob29rcyBhbmQga2V5cy5cbiAqL1xuXG4vLyBBdXRoIChtdXRhdGlvbnMsIGJ1dCBncm91cGVkIHdpdGggcXVlcmllcyBmb3IgY29udmVuaWVuY2UpXG5leHBvcnQgKiBmcm9tICcuL2F1dGgnXG5cbi8vIFVzZXJzXG5leHBvcnQgKiBmcm9tICcuL3VzZXJzJ1xuZXhwb3J0IHt1c2VyS2V5c30gZnJvbSAnLi91c2VycydcblxuLy8gU3BvdHNcbmV4cG9ydCAqIGZyb20gJy4vc3BvdHMnXG5leHBvcnQge3Nwb3RLZXlzfSBmcm9tICcuL3Nwb3RzJ1xuXG4vLyBQb3N0c1xuZXhwb3J0ICogZnJvbSAnLi9wb3N0cydcbmV4cG9ydCB7cG9zdEtleXN9IGZyb20gJy4vcG9zdHMnXG5cbi8vIENvbnZlcnNhdGlvbnNcbmV4cG9ydCAqIGZyb20gJy4vY29udmVyc2F0aW9ucydcbmV4cG9ydCB7Y29udmVyc2F0aW9uS2V5c30gZnJvbSAnLi9jb252ZXJzYXRpb25zJ1xuXG4vLyBDbHVic1xuZXhwb3J0ICogZnJvbSAnLi9jbHVicydcbmV4cG9ydCB7Y2x1YktleXN9IGZyb20gJy4vY2x1YnMnXG5cbi8vIFRlbXBsYXRlc1xuZXhwb3J0ICogZnJvbSAnLi90ZW1wbGF0ZXMnXG5leHBvcnQge3RlbXBsYXRlS2V5c30gZnJvbSAnLi90ZW1wbGF0ZXMnXG5cbi8vIE5vdGlmaWNhdGlvbnNcbmV4cG9ydCAqIGZyb20gJy4vbm90aWZpY2F0aW9ucydcbmV4cG9ydCB7bm90aWZpY2F0aW9uS2V5c30gZnJvbSAnLi9ub3RpZmljYXRpb25zJ1xuXG4vLyBNaXNjIChjaXRpZXMsIHZpYmVzLCBldmVudHMpXG5leHBvcnQgKiBmcm9tICcuL21pc2MnXG5leHBvcnQge21pc2NLZXlzfSBmcm9tICcuL21pc2MnXG5cbi8vIFByb2R1Y3RzIChtYXJrZXRwbGFjZSlcbmV4cG9ydCAqIGZyb20gJy4vcHJvZHVjdHMnXG5leHBvcnQge3Byb2R1Y3RLZXlzfSBmcm9tICcuL3Byb2R1Y3RzJ1xuXG4vLyBPcmRlcnMgKG1hcmtldHBsYWNlKVxuZXhwb3J0ICogZnJvbSAnLi9vcmRlcnMnXG5leHBvcnQge29yZGVyS2V5c30gZnJvbSAnLi9vcmRlcnMnXG4iXX0=
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.1.0",
3
+ "version": "1.3.0",
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.0.0"
35
+ "@spotsdev/types": "^1.1.0"
36
36
  },
37
37
  "devDependencies": {
38
38
  "@types/node": "^20.0.0",
@@ -89,6 +89,9 @@ export interface Spot {
89
89
  country: string | null
90
90
  latitude: number | null
91
91
  longitude: number | null
92
+ // Aliases for backwards compatibility
93
+ lat?: number | null
94
+ lng?: number | null
92
95
  type: SpotType
93
96
  googlePlaceId: string | null
94
97
  website: string | null
@@ -96,8 +99,9 @@ export interface Spot {
96
99
  imageUrl: string | null
97
100
  ownerId: string | null
98
101
  isVerified: boolean
99
- createdAt: Date
100
- updatedAt: Date
102
+ // Dates as strings for JSON serialization compatibility
103
+ createdAt: Date | string
104
+ updatedAt: Date | string
101
105
  }
102
106
 
103
107
  export interface SpotPost {
@@ -27,3 +27,6 @@ export * from './products'
27
27
 
28
28
  // Orders (marketplace)
29
29
  export * from './orders'
30
+
31
+ // Redemptions (wallet)
32
+ export * from './redemptions'
@@ -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
+ }
@@ -46,3 +46,7 @@ export {productKeys} from './products'
46
46
  // Orders (marketplace)
47
47
  export * from './orders'
48
48
  export {orderKeys} from './orders'
49
+
50
+ // Wallet (redemptions)
51
+ export * from './wallet'
52
+ export {walletKeys} from './wallet'
@@ -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
+ }