vibefast-cli 1.2.1 → 1.3.1

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.
Files changed (348) hide show
  1. package/README.md +30 -95
  2. package/dist/__tests__/recipes.test.js +94 -91
  3. package/dist/__tests__/recipes.test.js.map +1 -1
  4. package/dist/commands/add.d.ts.map +1 -1
  5. package/dist/commands/add.js +301 -125
  6. package/dist/commands/add.js.map +1 -1
  7. package/dist/commands/checklist.d.ts.map +1 -1
  8. package/dist/commands/checklist.js +85 -44
  9. package/dist/commands/checklist.js.map +1 -1
  10. package/dist/commands/health.d.ts.map +1 -1
  11. package/dist/commands/health.js +13 -4
  12. package/dist/commands/health.js.map +1 -1
  13. package/dist/commands/init.d.ts.map +1 -1
  14. package/dist/commands/init.js +118 -26
  15. package/dist/commands/init.js.map +1 -1
  16. package/dist/commands/migrate.d.ts +3 -0
  17. package/dist/commands/migrate.d.ts.map +1 -0
  18. package/dist/commands/migrate.js +202 -0
  19. package/dist/commands/migrate.js.map +1 -0
  20. package/dist/commands/remove.d.ts.map +1 -1
  21. package/dist/commands/remove.js +61 -3
  22. package/dist/commands/remove.js.map +1 -1
  23. package/dist/core/auth.d.ts.map +1 -1
  24. package/dist/core/auth.js +20 -18
  25. package/dist/core/auth.js.map +1 -1
  26. package/dist/core/codemod.d.ts +33 -0
  27. package/dist/core/codemod.d.ts.map +1 -1
  28. package/dist/core/codemod.js +116 -0
  29. package/dist/core/codemod.js.map +1 -1
  30. package/dist/core/detect.d.ts.map +1 -1
  31. package/dist/core/detect.js +24 -7
  32. package/dist/core/detect.js.map +1 -1
  33. package/dist/core/journal.d.ts +1 -0
  34. package/dist/core/journal.d.ts.map +1 -1
  35. package/dist/core/journal.js.map +1 -1
  36. package/dist/core/recipes.d.ts.map +1 -1
  37. package/dist/core/recipes.js +25 -7
  38. package/dist/core/recipes.js.map +1 -1
  39. package/dist/index.js +2 -2
  40. package/dist/index.js.map +1 -1
  41. package/docs/architecture.md +50 -0
  42. package/docs/commands.md +78 -0
  43. package/docs/contributing.md +27 -0
  44. package/docs/quickstart.md +50 -0
  45. package/docs/recipes.md +57 -0
  46. package/docs/troubleshooting.md +31 -0
  47. package/package.json +2 -2
  48. package/recipes/0/apps/native/src/components/advanced-ui/timeline/demo.tsx +445 -0
  49. package/recipes/0/apps/native/src/components/advanced-ui/timeline/timeline-view.tsx +355 -0
  50. package/recipes/0/apps/native/src/components/advanced-ui/timeline/types.ts +31 -0
  51. package/recipes/0/recipe.json +18 -0
  52. package/recipes/animated-chip/apps/native/src/components/advanced-ui/chip/demo.tsx +2 -1
  53. package/recipes/animated-chip/recipe.json +5 -2
  54. package/recipes/animated-chip-native@latest.zip +0 -0
  55. package/recipes/animated-chip@latest.zip +0 -0
  56. package/recipes/animated-switch/apps/native/src/components/advanced-ui/switch/demo.tsx +1 -1
  57. package/recipes/animated-switch/recipe.json +5 -2
  58. package/recipes/animated-switch-native@latest.zip +0 -0
  59. package/recipes/animated-switch@latest.zip +0 -0
  60. package/recipes/audio-recorder/apps/native/src/features/audio-recorder/components/audio-recorder.tsx +2 -1
  61. package/recipes/audio-recorder/apps/native/src/features/audio-recorder/demo/with-recording-list-demo.tsx +2 -2
  62. package/recipes/audio-recorder/recipe.json +7 -2
  63. package/recipes/audio-recorder-native@latest.zip +0 -0
  64. package/recipes/audio-recorder-supabase/apps/native/src/features/audio-recorder/components/audio-recorder.tsx +2 -1
  65. package/recipes/audio-recorder-supabase/apps/native/src/features/audio-recorder/demo/with-recording-list-demo.tsx +2 -1
  66. package/recipes/audio-recorder-supabase/recipe.json +12 -16
  67. package/recipes/audio-recorder-supabase-native@latest.zip +0 -0
  68. package/recipes/audio-recorder-supabase@latest.zip +0 -0
  69. package/recipes/audio-recorder@latest.zip +0 -0
  70. package/recipes/charts/apps/native/src/app/charts/index.tsx +3 -0
  71. package/recipes/charts/apps/native/src/features/charts/components/bar-chart.tsx +3 -1
  72. package/recipes/charts/apps/native/src/features/charts/components/candlestick-chart.tsx +3 -1
  73. package/recipes/charts/apps/native/src/features/charts/components/column-chart.tsx +3 -1
  74. package/recipes/charts/apps/native/src/features/charts/components/doughnut-chart.tsx +3 -1
  75. package/recipes/charts/apps/native/src/features/charts/components/line-chart.tsx +3 -1
  76. package/recipes/charts/apps/native/src/features/charts/components/radar-chart.tsx +3 -1
  77. package/recipes/charts/apps/native/src/features/charts/components/stacked-bar-chart.tsx +3 -1
  78. package/recipes/charts/recipe.json +13 -4
  79. package/recipes/charts-native@latest.zip +0 -0
  80. package/recipes/charts@latest.zip +0 -0
  81. package/recipes/chatbot/apps/native/src/features/chatbot/components/chat-markdown.tsx +86 -86
  82. package/recipes/chatbot/apps/native/src/features/chatbot/components/markdown/code-block.tsx +4 -4
  83. package/recipes/chatbot/recipe.json +3 -40
  84. package/recipes/chatbot-native@latest.zip +0 -0
  85. package/recipes/chatbot-supabase/apps/native/src/features/chatbot/components/chat-markdown.tsx +4 -1
  86. package/recipes/chatbot-supabase/apps/native/src/features/chatbot/components/markdown/code-block.tsx +86 -53
  87. package/recipes/chatbot-supabase/recipe.json +3 -42
  88. package/recipes/chatbot-supabase-native@latest.zip +0 -0
  89. package/recipes/chatbot-supabase@latest.zip +0 -0
  90. package/recipes/chatbot@latest.zip +0 -0
  91. package/recipes/glowing-button/recipe.json +6 -2
  92. package/recipes/glowing-button-native@latest.zip +0 -0
  93. package/recipes/glowing-button@latest.zip +0 -0
  94. package/recipes/image-analysis/apps/native/src/app/analysis/[type]/_layout.tsx +5 -0
  95. package/recipes/image-analysis/apps/native/src/app/analysis/[type]/analysis-options.tsx +50 -0
  96. package/recipes/image-analysis/apps/native/src/app/analysis/[type]/camera.tsx +2 -0
  97. package/recipes/image-analysis/apps/native/src/app/analysis/[type]/index.tsx +50 -0
  98. package/recipes/image-analysis/apps/native/src/app/analysis/[type]/loading.tsx +50 -0
  99. package/recipes/image-analysis/apps/native/src/app/analysis/[type]/results.tsx +2 -0
  100. package/recipes/image-analysis/apps/native/src/app/analysis/[type]/trait-details.tsx +3 -0
  101. package/recipes/image-analysis/apps/native/src/features/image-analyzer/app/analysis-options-screen.tsx +2 -2
  102. package/recipes/image-analysis/apps/native/src/features/image-analyzer/app/camera.tsx +72 -65
  103. package/recipes/image-analysis/apps/native/src/features/image-analyzer/app/image-capture-screen.tsx +65 -47
  104. package/recipes/image-analysis/apps/native/src/features/image-analyzer/app/loading-screen.tsx +43 -2
  105. package/recipes/image-analysis/apps/native/src/features/image-analyzer/app/loading.tsx +34 -1
  106. package/recipes/image-analysis/apps/native/src/features/image-analyzer/hooks/use-image-analysis.ts +83 -2
  107. package/recipes/image-analysis/recipe.json +11 -19
  108. package/recipes/image-analysis-native@latest.zip +0 -0
  109. package/recipes/image-analysis-supabase/apps/native/src/app/analysis/[type]/_layout.tsx +5 -0
  110. package/recipes/image-analysis-supabase/apps/native/src/app/analysis/[type]/analysis-options.tsx +50 -0
  111. package/recipes/image-analysis-supabase/apps/native/src/app/analysis/[type]/camera.tsx +2 -0
  112. package/recipes/image-analysis-supabase/apps/native/src/app/analysis/[type]/index.tsx +50 -0
  113. package/recipes/image-analysis-supabase/apps/native/src/app/analysis/[type]/loading.tsx +50 -0
  114. package/recipes/image-analysis-supabase/apps/native/src/app/analysis/[type]/results.tsx +2 -0
  115. package/recipes/image-analysis-supabase/apps/native/src/app/analysis/[type]/trait-details.tsx +3 -0
  116. package/recipes/image-analysis-supabase/recipe.json +10 -37
  117. package/recipes/image-analysis-supabase-native@latest.zip +0 -0
  118. package/recipes/image-analysis-supabase@latest.zip +0 -0
  119. package/recipes/image-analysis@latest.zip +0 -0
  120. package/recipes/image-analyzer/apps/native/src/app/(root)/(protected)/image-analyzer/index.tsx +2 -0
  121. package/recipes/image-generator/apps/native/src/app/image-generator/gallery.tsx +3 -0
  122. package/recipes/image-generator/apps/native/src/app/image-generator/index.tsx +3 -0
  123. package/recipes/image-generator/recipe.json +8 -18
  124. package/recipes/image-generator-native@latest.zip +0 -0
  125. package/recipes/image-generator-supabase/recipe.json +6 -35
  126. package/recipes/image-generator-supabase-native@latest.zip +0 -0
  127. package/recipes/image-generator-supabase@latest.zip +0 -0
  128. package/recipes/image-generator@latest.zip +0 -0
  129. package/recipes/ios-widget/recipe.json +18 -119
  130. package/recipes/ios-widget-native@latest.zip +0 -0
  131. package/recipes/ios-widget@latest.zip +0 -0
  132. package/recipes/number-stepper/apps/native/src/components/advanced-ui/stepper/demo.tsx +1 -1
  133. package/recipes/number-stepper/recipe.json +5 -2
  134. package/recipes/number-stepper-native@latest.zip +0 -0
  135. package/recipes/number-stepper@latest.zip +0 -0
  136. package/recipes/onboarding/apps/native/src/features/onboarding/expense-tracker/components/interactive-onboarding.tsx +11 -18
  137. package/recipes/onboarding/apps/native/src/features/onboarding/expense-tracker/components/steps/ai-tone-step.tsx +5 -7
  138. package/recipes/onboarding/apps/native/src/features/onboarding/expense-tracker/components/steps/currency-step.tsx +9 -7
  139. package/recipes/onboarding/apps/native/src/features/onboarding/expense-tracker/components/steps/feature-ai-step.tsx +8 -7
  140. package/recipes/onboarding/apps/native/src/features/onboarding/expense-tracker/components/steps/feature-chatbot-step.tsx +6 -5
  141. package/recipes/onboarding/apps/native/src/features/onboarding/expense-tracker/components/steps/feature-manual-step.tsx +4 -3
  142. package/recipes/onboarding/apps/native/src/features/onboarding/expense-tracker/components/steps/feature-scan-step.tsx +6 -5
  143. package/recipes/onboarding/apps/native/src/features/onboarding/expense-tracker/components/steps/main-reason-step.tsx +5 -7
  144. package/recipes/onboarding/apps/native/src/features/onboarding/expense-tracker/components/steps/notification-step.tsx +7 -6
  145. package/recipes/onboarding/apps/native/src/features/onboarding/expense-tracker/components/steps/overspend-step.tsx +5 -7
  146. package/recipes/onboarding/apps/native/src/features/onboarding/expense-tracker/components/steps/personalizing-step.tsx +8 -7
  147. package/recipes/onboarding/apps/native/src/features/onboarding/expense-tracker/components/steps/rating-step.tsx +6 -5
  148. package/recipes/onboarding/apps/native/src/features/onboarding/expense-tracker/components/steps/reminder-step.tsx +5 -6
  149. package/recipes/onboarding/apps/native/src/features/onboarding/expense-tracker/components/steps/safety-step.tsx +5 -4
  150. package/recipes/onboarding/apps/native/src/features/onboarding/expense-tracker/components/steps/struggle-step.tsx +5 -7
  151. package/recipes/onboarding/apps/native/src/features/onboarding/expense-tracker/components/steps/welcome-step.tsx +7 -6
  152. package/recipes/onboarding/apps/native/src/features/onboarding/expense-tracker/components/ui/onboarding-header.tsx +4 -3
  153. package/recipes/onboarding/recipe.json +9 -6
  154. package/recipes/onboarding-native@latest.zip +0 -0
  155. package/recipes/onboarding@latest.zip +0 -0
  156. package/recipes/payments/apps/native/src/app/paywall/index.tsx +74 -0
  157. package/recipes/payments/apps/native/src/app/paywall/local.tsx +25 -0
  158. package/recipes/payments/apps/native/src/app/paywall/remote.tsx +23 -0
  159. package/recipes/payments/packages/backend/convex/payments.ts +21 -3
  160. package/recipes/payments/recipe.json +14 -34
  161. package/recipes/payments-native@latest.zip +0 -0
  162. package/recipes/payments-supabase/apps/native/src/app/paywall/index.tsx +74 -0
  163. package/recipes/payments-supabase/apps/native/src/app/paywall/local.tsx +25 -0
  164. package/recipes/payments-supabase/apps/native/src/app/paywall/remote.tsx +23 -0
  165. package/recipes/payments-supabase/recipe.json +16 -23
  166. package/recipes/payments-supabase-native@latest.zip +0 -0
  167. package/recipes/payments-supabase@latest.zip +0 -0
  168. package/recipes/payments@latest.zip +0 -0
  169. package/recipes/posthog/apps/native/src/components/analytics/navigation-tracker.tsx +14 -0
  170. package/recipes/posthog/apps/native/src/lib/hooks/use-navigation-analytics.ts +44 -0
  171. package/recipes/posthog/apps/native/src/providers/posthog-provider.tsx +51 -0
  172. package/recipes/posthog/recipe.json +60 -0
  173. package/recipes/posthog-native@latest.zip +0 -0
  174. package/recipes/progress-circle/apps/native/src/components/advanced-ui/progress-bars/progress-circle-page.tsx +1 -1
  175. package/recipes/progress-circle/recipe.json +5 -2
  176. package/recipes/progress-circle-native@latest.zip +0 -0
  177. package/recipes/progress-circle@latest.zip +0 -0
  178. package/recipes/quiz/apps/native/src/app/quiz/index.tsx +47 -0
  179. package/recipes/quiz/recipe.json +9 -6
  180. package/recipes/quiz-native@latest.zip +0 -0
  181. package/recipes/quiz@latest.zip +0 -0
  182. package/recipes/screen-kits/apps/native/src/app/screen-kits/_layout.tsx +12 -0
  183. package/recipes/screen-kits/apps/native/src/app/screen-kits/index.tsx +114 -0
  184. package/recipes/screen-kits/apps/native/src/features/screen-kits/index.ts +1 -0
  185. package/recipes/screen-kits/apps/native/src/features/screen-kits/types.ts +28 -0
  186. package/recipes/screen-kits/recipe.json +26 -0
  187. package/recipes/screen-kits-duolingo/apps/native/src/app/screen-kits/duolingo/_layout.tsx +12 -0
  188. package/recipes/screen-kits-duolingo/apps/native/src/app/screen-kits/duolingo/home.tsx +5 -0
  189. package/recipes/screen-kits-duolingo/apps/native/src/app/screen-kits/duolingo/index.tsx +5 -0
  190. package/recipes/screen-kits-duolingo/apps/native/src/app/screen-kits/duolingo/lesson-complete.tsx +5 -0
  191. package/recipes/screen-kits-duolingo/apps/native/src/app/screen-kits/duolingo/lesson-fail.tsx +5 -0
  192. package/recipes/screen-kits-duolingo/apps/native/src/app/screen-kits/duolingo/lesson.tsx +5 -0
  193. package/recipes/screen-kits-duolingo/apps/native/src/app/screen-kits/duolingo/skill-tree.tsx +5 -0
  194. package/recipes/screen-kits-duolingo/apps/native/src/features/screen-kits/duolingo/components/duo-button.tsx +174 -0
  195. package/recipes/screen-kits-duolingo/apps/native/src/features/screen-kits/duolingo/components/skill-button.tsx +186 -0
  196. package/recipes/screen-kits-duolingo/apps/native/src/features/screen-kits/duolingo/components/xp-header.tsx +115 -0
  197. package/recipes/screen-kits-duolingo/apps/native/src/features/screen-kits/duolingo/constants.ts +89 -0
  198. package/recipes/screen-kits-duolingo/apps/native/src/features/screen-kits/duolingo/index.ts +3 -0
  199. package/recipes/screen-kits-duolingo/apps/native/src/features/screen-kits/duolingo/screens/home-screen.tsx +225 -0
  200. package/recipes/screen-kits-duolingo/apps/native/src/features/screen-kits/duolingo/screens/lesson-complete-screen.tsx +485 -0
  201. package/recipes/screen-kits-duolingo/apps/native/src/features/screen-kits/duolingo/screens/lesson-fail-screen.tsx +105 -0
  202. package/recipes/screen-kits-duolingo/apps/native/src/features/screen-kits/duolingo/screens/lesson-screen.tsx +384 -0
  203. package/recipes/screen-kits-duolingo/recipe.json +58 -0
  204. package/recipes/screen-kits-duolingo-native@latest.zip +0 -0
  205. package/recipes/screen-kits-finance/apps/native/src/app/screen-kits/finance/_layout.tsx +45 -0
  206. package/recipes/screen-kits-finance/apps/native/src/app/screen-kits/finance/asset-detail.tsx +3 -0
  207. package/recipes/screen-kits-finance/apps/native/src/app/screen-kits/finance/notifications.tsx +3 -0
  208. package/recipes/screen-kits-finance/apps/native/src/app/screen-kits/finance/receive.tsx +3 -0
  209. package/recipes/screen-kits-finance/apps/native/src/app/screen-kits/finance/send.tsx +3 -0
  210. package/recipes/screen-kits-finance/apps/native/src/app/screen-kits/finance/swap.tsx +3 -0
  211. package/recipes/screen-kits-finance/apps/native/src/app/screen-kits/finance/wallet.tsx +3 -0
  212. package/recipes/screen-kits-finance/apps/native/src/features/screen-kits/finance/components/ActionButtons.tsx +78 -0
  213. package/recipes/screen-kits-finance/apps/native/src/features/screen-kits/finance/components/AssetRow.tsx +94 -0
  214. package/recipes/screen-kits-finance/apps/native/src/features/screen-kits/finance/components/BalanceCard.tsx +118 -0
  215. package/recipes/screen-kits-finance/apps/native/src/features/screen-kits/finance/constants.ts +85 -0
  216. package/recipes/screen-kits-finance/apps/native/src/features/screen-kits/finance/screens/asset-detail.tsx +378 -0
  217. package/recipes/screen-kits-finance/apps/native/src/features/screen-kits/finance/screens/notifications.tsx +210 -0
  218. package/recipes/screen-kits-finance/apps/native/src/features/screen-kits/finance/screens/receive-modal.tsx +317 -0
  219. package/recipes/screen-kits-finance/apps/native/src/features/screen-kits/finance/screens/send-modal.tsx +420 -0
  220. package/recipes/screen-kits-finance/apps/native/src/features/screen-kits/finance/screens/swap-modal.tsx +363 -0
  221. package/recipes/screen-kits-finance/apps/native/src/features/screen-kits/finance/screens/wallet-dashboard.tsx +281 -0
  222. package/recipes/screen-kits-finance/recipe.json +46 -0
  223. package/recipes/screen-kits-finance-native@latest.zip +0 -0
  224. package/recipes/screen-kits-fitness/apps/native/assets/sounds/timer-beep.wav +0 -0
  225. package/recipes/screen-kits-fitness/apps/native/src/app/screen-kits/fitness/_layout.tsx +10 -0
  226. package/recipes/screen-kits-fitness/apps/native/src/app/screen-kits/fitness/index.tsx +6 -0
  227. package/recipes/screen-kits-fitness/apps/native/src/app/screen-kits/fitness/timer.tsx +3 -0
  228. package/recipes/screen-kits-fitness/apps/native/src/app/screen-kits/fitness/workout.tsx +3 -0
  229. package/recipes/screen-kits-fitness/apps/native/src/features/screen-kits/fitness/components/timer-components.tsx +500 -0
  230. package/recipes/screen-kits-fitness/apps/native/src/features/screen-kits/fitness/components/timer-settings-modal.tsx +352 -0
  231. package/recipes/screen-kits-fitness/apps/native/src/features/screen-kits/fitness/components/workout-card.tsx +105 -0
  232. package/recipes/screen-kits-fitness/apps/native/src/features/screen-kits/fitness/constants.ts +189 -0
  233. package/recipes/screen-kits-fitness/apps/native/src/features/screen-kits/fitness/hooks/use-timer.ts +307 -0
  234. package/recipes/screen-kits-fitness/apps/native/src/features/screen-kits/fitness/index.ts +1 -0
  235. package/recipes/screen-kits-fitness/apps/native/src/features/screen-kits/fitness/screens/timer-screen.tsx +278 -0
  236. package/recipes/screen-kits-fitness/apps/native/src/features/screen-kits/fitness/screens/workout-dashboard.tsx +350 -0
  237. package/recipes/screen-kits-fitness/recipe.json +63 -0
  238. package/recipes/screen-kits-fitness-native@latest.zip +0 -0
  239. package/recipes/screen-kits-habits/apps/native/src/app/screen-kits/productivity/habits.tsx +1 -0
  240. package/recipes/screen-kits-habits/apps/native/src/app/screen-kits/productivity/kanban.tsx +1 -0
  241. package/recipes/screen-kits-habits/apps/native/src/app/screen-kits/productivity/routes.ts +4 -0
  242. package/recipes/screen-kits-habits/apps/native/src/features/screen-kits/productivity/components/AddTaskModal.tsx +246 -0
  243. package/recipes/screen-kits-habits/apps/native/src/features/screen-kits/productivity/components/DraggableTaskCard.tsx +92 -0
  244. package/recipes/screen-kits-habits/apps/native/src/features/screen-kits/productivity/components/KanbanColumn.tsx +238 -0
  245. package/recipes/screen-kits-habits/apps/native/src/features/screen-kits/productivity/components/TaskCard.tsx +144 -0
  246. package/recipes/screen-kits-habits/apps/native/src/features/screen-kits/productivity/components/add-habit-modal.tsx +271 -0
  247. package/recipes/screen-kits-habits/apps/native/src/features/screen-kits/productivity/constants.ts +295 -0
  248. package/recipes/screen-kits-habits/apps/native/src/features/screen-kits/productivity/kanban-utils.ts +62 -0
  249. package/recipes/screen-kits-habits/apps/native/src/features/screen-kits/productivity/screens/habit-tracker.tsx +1160 -0
  250. package/recipes/screen-kits-habits/apps/native/src/features/screen-kits/productivity/screens/kanban-board.tsx +432 -0
  251. package/recipes/screen-kits-habits/recipe.json +52 -0
  252. package/recipes/screen-kits-habits-native@latest.zip +0 -0
  253. package/recipes/screen-kits-native@latest.zip +0 -0
  254. package/recipes/sentry/apps/native/src/providers/sentry-provider.tsx +64 -0
  255. package/recipes/sentry/recipe.json +39 -0
  256. package/recipes/sentry-native@latest.zip +0 -0
  257. package/recipes/swipe-slider/apps/native/src/components/advanced-ui/sliders/swipe-slider-page.tsx +1 -1
  258. package/recipes/swipe-slider/recipe.json +5 -2
  259. package/recipes/swipe-slider-native@latest.zip +0 -0
  260. package/recipes/swipe-slider@latest.zip +0 -0
  261. package/recipes/timeline/apps/native/src/components/advanced-ui/timeline/demo.tsx +2 -1
  262. package/recipes/timeline/recipe.json +5 -2
  263. package/recipes/timeline-native@latest.zip +0 -0
  264. package/recipes/timeline@latest.zip +0 -0
  265. package/recipes/tracker-app/apps/native/src/app/tracker-app/index.tsx +1 -0
  266. package/recipes/tracker-app/recipe.json +10 -7
  267. package/recipes/tracker-app-native@latest.zip +0 -0
  268. package/recipes/tracker-app@latest.zip +0 -0
  269. package/recipes/upload-all.sh +8 -31
  270. package/recipes/voice-bot/apps/native/src/app/voice-bot/index.tsx +56 -0
  271. package/recipes/voice-bot/recipe.json +31 -7
  272. package/recipes/voice-bot-native@latest.zip +0 -0
  273. package/recipes/voice-bot@latest.zip +0 -0
  274. package/recipes/wake-word/apps/native/src/app/{(root)/(protected)/test-wake-word.tsx → test-wake-word.tsx} +43 -4
  275. package/recipes/wake-word/recipe.json +16 -26
  276. package/recipes/wake-word-native@latest.zip +0 -0
  277. package/recipes/wake-word@latest.zip +0 -0
  278. package/scripts/create-advanced-ui-recipes.sh +46 -19
  279. package/scripts/create-recipes.mjs +471 -117
  280. package/scripts/package-recipes.mjs +76 -0
  281. package/scripts/publish-all.sh +6 -2
  282. package/CHANGELOG.md +0 -198
  283. package/docs/archive/AUTO-DETECT-DEPS.md +0 -607
  284. package/docs/archive/FINAL-PACKAGE-STRATEGY.md +0 -583
  285. package/docs/archive/FINAL-SIMPLE-PLAN.md +0 -487
  286. package/docs/archive/FINAL-STATUS.md +0 -144
  287. package/docs/archive/FLOW-DIAGRAM.md +0 -1629
  288. package/docs/archive/GOTCHAS-AND-RISKS.md +0 -801
  289. package/docs/archive/IMPLEMENTATION-PLAN.md +0 -1360
  290. package/docs/archive/PLAN.md +0 -453
  291. package/docs/archive/PRODUCTION-READINESS.md +0 -684
  292. package/docs/archive/PRODUCTION-TEST-RESULTS.md +0 -465
  293. package/docs/archive/SIMPLIFIED-PLAN.md +0 -578
  294. package/docs/archive/STATUS.md +0 -199
  295. package/docs/archive/SUCCESS.md +0 -259
  296. package/docs/archive/TEST-SUMMARY.md +0 -261
  297. package/docs/archive/TESTING-CHECKLIST.md +0 -450
  298. package/docs/archive/USER-MODIFICATIONS.md +0 -448
  299. package/docs/decisions.md +0 -55
  300. package/docs/manual-testing.md +0 -91
  301. package/docs/next-steps.md +0 -12
  302. package/recipes/README.md +0 -156
  303. package/recipes/audio-recorder-supabase/packages/backend/src/services/recordings.ts +0 -369
  304. package/recipes/chatbot/apps/native/src/api-client/chatbot.ts +0 -83
  305. package/recipes/chatbot/packages/backend/convex/agents.ts +0 -115
  306. package/recipes/chatbot/packages/backend/convex/tools/index.ts +0 -18
  307. package/recipes/chatbot/packages/backend/convex/tools/knowledgeRetrieval.ts +0 -97
  308. package/recipes/chatbot/packages/backend/convex/tools/tavilySearch.ts +0 -83
  309. package/recipes/chatbot/packages/backend/convex/tools/userProfile.ts +0 -72
  310. package/recipes/chatbot-supabase/apps/native/src/api-client/supabase/chatbot.ts +0 -515
  311. package/recipes/chatbot-supabase/packages/backend/src/services/conversations.ts +0 -243
  312. package/recipes/chatbot-supabase/packages/backend/src/services/messages.ts +0 -327
  313. package/recipes/image-analysis/apps/native/src/api-client/image-analyzer.ts +0 -62
  314. package/recipes/image-analysis-supabase/packages/backend/src/services/image-analyses.ts +0 -132
  315. package/recipes/image-generator/apps/native/src/api-client/image-generator.ts +0 -34
  316. package/recipes/payments/apps/native/src/api-client/payments.ts +0 -44
  317. package/recipes/payments-supabase/packages/backend/src/services/payments.ts +0 -201
  318. package/recipes/posthog.json +0 -47
  319. package/recipes/revenuecat.json +0 -43
  320. package/recipes/sentry.json +0 -47
  321. package/recipes/wake-word/apps/native/assets/vosk-model/README.md +0 -103
  322. package/recipes/wake-word/apps/native/scripts/download-vosk-model.mjs +0 -127
  323. /package/recipes/{audio-recorder/apps/native/src/app/(root)/(protected) → audio-recorder-supabase/apps/native/src/app}/audio-recorder/index.tsx +0 -0
  324. /package/recipes/ios-widget/{targets → apps/native/targets}/widget/AppIntent.swift +0 -0
  325. /package/recipes/ios-widget/{targets → apps/native/targets}/widget/Assets.xcassets/AppIcon.appiconset/App-Icon-20x20@1x.png +0 -0
  326. /package/recipes/ios-widget/{targets → apps/native/targets}/widget/Assets.xcassets/AppIcon.appiconset/App-Icon-20x20@2x.png +0 -0
  327. /package/recipes/ios-widget/{targets → apps/native/targets}/widget/Assets.xcassets/AppIcon.appiconset/App-Icon-20x20@3x.png +0 -0
  328. /package/recipes/ios-widget/{targets → apps/native/targets}/widget/Assets.xcassets/AppIcon.appiconset/App-Icon-29x29@1x.png +0 -0
  329. /package/recipes/ios-widget/{targets → apps/native/targets}/widget/Assets.xcassets/AppIcon.appiconset/App-Icon-29x29@2x.png +0 -0
  330. /package/recipes/ios-widget/{targets → apps/native/targets}/widget/Assets.xcassets/AppIcon.appiconset/App-Icon-29x29@3x.png +0 -0
  331. /package/recipes/ios-widget/{targets → apps/native/targets}/widget/Assets.xcassets/AppIcon.appiconset/App-Icon-40x40@1x.png +0 -0
  332. /package/recipes/ios-widget/{targets → apps/native/targets}/widget/Assets.xcassets/AppIcon.appiconset/App-Icon-40x40@2x.png +0 -0
  333. /package/recipes/ios-widget/{targets → apps/native/targets}/widget/Assets.xcassets/AppIcon.appiconset/App-Icon-40x40@3x.png +0 -0
  334. /package/recipes/ios-widget/{targets → apps/native/targets}/widget/Assets.xcassets/AppIcon.appiconset/App-Icon-60x60@2x.png +0 -0
  335. /package/recipes/ios-widget/{targets → apps/native/targets}/widget/Assets.xcassets/AppIcon.appiconset/App-Icon-60x60@3x.png +0 -0
  336. /package/recipes/ios-widget/{targets → apps/native/targets}/widget/Assets.xcassets/AppIcon.appiconset/App-Icon-76x76@1x.png +0 -0
  337. /package/recipes/ios-widget/{targets → apps/native/targets}/widget/Assets.xcassets/AppIcon.appiconset/App-Icon-76x76@2x.png +0 -0
  338. /package/recipes/ios-widget/{targets → apps/native/targets}/widget/Assets.xcassets/AppIcon.appiconset/App-Icon-83.5x83.5@2x.png +0 -0
  339. /package/recipes/ios-widget/{targets → apps/native/targets}/widget/Assets.xcassets/AppIcon.appiconset/Contents.json +0 -0
  340. /package/recipes/ios-widget/{targets → apps/native/targets}/widget/Assets.xcassets/AppIcon.appiconset/ItunesArtwork@2x.png +0 -0
  341. /package/recipes/ios-widget/{targets → apps/native/targets}/widget/CalorieTrackerWidget.swift +0 -0
  342. /package/recipes/ios-widget/{targets → apps/native/targets}/widget/HabitTrackerWidget.swift +0 -0
  343. /package/recipes/ios-widget/{targets → apps/native/targets}/widget/Info.plist +0 -0
  344. /package/recipes/ios-widget/{targets → apps/native/targets}/widget/WidgetLiveActivity.swift +0 -0
  345. /package/recipes/ios-widget/{targets → apps/native/targets}/widget/expo-target.config.js +0 -0
  346. /package/recipes/ios-widget/{targets → apps/native/targets}/widget/generated.entitlements +0 -0
  347. /package/recipes/ios-widget/{targets → apps/native/targets}/widget/index.swift +0 -0
  348. /package/recipes/ios-widget/{targets → apps/native/targets}/widget/widgets.swift +0 -0
@@ -1,6 +1,9 @@
1
+ import { MaterialCommunityIcons } from '@expo/vector-icons';
2
+ import * as Haptics from 'expo-haptics';
1
3
  import React, { useEffect } from 'react';
2
4
  import { View } from 'react-native';
3
5
  import Animated, {
6
+ Easing,
4
7
  FadeIn,
5
8
  FadeOut,
6
9
  useAnimatedStyle,
@@ -8,12 +11,10 @@ import Animated, {
8
11
  withRepeat,
9
12
  withSequence,
10
13
  withTiming,
11
- Easing,
12
14
  } from 'react-native-reanimated';
13
- import { MaterialCommunityIcons } from '@expo/vector-icons';
14
- import { Text, Button } from '@/components/ui';
15
+
16
+ import { Button, Text } from '@/components/ui';
15
17
  import { useThemeConfig } from '@/lib/use-theme-config';
16
- import * as Haptics from 'expo-haptics';
17
18
 
18
19
  const ScanDemo = () => {
19
20
  const theme = useThemeConfig();
@@ -37,7 +38,7 @@ const ScanDemo = () => {
37
38
  -1,
38
39
  true,
39
40
  );
40
- }, []);
41
+ }, [checkmarkScale, scanLineY]);
41
42
 
42
43
  const animatedLineStyle = useAnimatedStyle(() => ({
43
44
  transform: [{ translateY: scanLineY.value }],
@@ -1,10 +1,10 @@
1
- import React from 'react';
2
- import { View, ScrollView } from 'react-native';
3
- import Animated, { FadeIn, FadeOut, FadeInDown } from 'react-native-reanimated';
4
- import { Text, Button, Pressable } from '@/components/ui';
5
- import { useThemeConfig } from '@/lib/use-theme-config';
6
1
  import { MaterialCommunityIcons } from '@expo/vector-icons';
7
2
  import * as Haptics from 'expo-haptics';
3
+ import React from 'react';
4
+ import { ScrollView, View } from 'react-native';
5
+ import Animated, { FadeIn, FadeInDown, FadeOut } from 'react-native-reanimated';
6
+
7
+ import { Button, Pressable, Text } from '@/components/ui';
8
8
 
9
9
  interface MainReasonStepProps {
10
10
  selectedReason: string | null;
@@ -50,8 +50,6 @@ export function MainReasonStep({
50
50
  onSelectReason,
51
51
  onNext,
52
52
  }: MainReasonStepProps) {
53
- const theme = useThemeConfig();
54
-
55
53
  return (
56
54
  <Animated.View
57
55
  entering={FadeIn}
@@ -1,11 +1,12 @@
1
- import React, { useState } from 'react';
2
- import { View, Alert, Linking } from 'react-native';
3
- import Animated, { ZoomIn, FadeIn, FadeOut } from 'react-native-reanimated';
4
- import { Text, Button } from '@/components/ui';
5
- import { useThemeConfig } from '@/lib/use-theme-config';
6
1
  import { MaterialCommunityIcons } from '@expo/vector-icons';
7
- import * as Notifications from 'expo-notifications';
8
2
  import * as Haptics from 'expo-haptics';
3
+ import * as Notifications from 'expo-notifications';
4
+ import React, { useState } from 'react';
5
+ import { Alert, Linking, View } from 'react-native';
6
+ import Animated, { FadeIn, FadeOut, ZoomIn } from 'react-native-reanimated';
7
+
8
+ import { Button, Text } from '@/components/ui';
9
+ import { useThemeConfig } from '@/lib/use-theme-config';
9
10
 
10
11
  interface NotificationStepProps {
11
12
  onEnable: () => void;
@@ -1,10 +1,10 @@
1
- import React from 'react';
2
- import { View, ScrollView } from 'react-native';
3
- import Animated, { FadeIn, FadeOut, FadeInDown } from 'react-native-reanimated';
4
- import { Text, Button, Pressable } from '@/components/ui';
5
- import { useThemeConfig } from '@/lib/use-theme-config';
6
1
  import { MaterialCommunityIcons } from '@expo/vector-icons';
7
2
  import * as Haptics from 'expo-haptics';
3
+ import React from 'react';
4
+ import { ScrollView, View } from 'react-native';
5
+ import Animated, { FadeIn, FadeInDown, FadeOut } from 'react-native-reanimated';
6
+
7
+ import { Button, Pressable, Text } from '@/components/ui';
8
8
 
9
9
  interface OverspendStepProps {
10
10
  selectedCategories: string[];
@@ -57,8 +57,6 @@ export function OverspendStep({
57
57
  onToggleCategory,
58
58
  onNext,
59
59
  }: OverspendStepProps) {
60
- const theme = useThemeConfig();
61
-
62
60
  return (
63
61
  <Animated.View
64
62
  entering={FadeIn}
@@ -1,19 +1,20 @@
1
+ import { MaterialCommunityIcons } from '@expo/vector-icons';
2
+ import { LinearGradient } from 'expo-linear-gradient';
1
3
  import React, { useEffect, useState } from 'react';
2
- import { View, Dimensions } from 'react-native';
4
+ import { Dimensions, View } from 'react-native';
3
5
  import Animated, {
6
+ Easing,
4
7
  FadeIn,
8
+ interpolate,
5
9
  useAnimatedStyle,
6
10
  useSharedValue,
7
11
  withRepeat,
8
12
  withSequence,
9
13
  withTiming,
10
- Easing,
11
- interpolate,
12
14
  } from 'react-native-reanimated';
13
- import { MaterialCommunityIcons } from '@expo/vector-icons';
15
+
14
16
  import { Text } from '@/components/ui';
15
17
  import { useThemeConfig } from '@/lib/use-theme-config';
16
- import { LinearGradient } from 'expo-linear-gradient';
17
18
 
18
19
  interface PersonalizingStepProps {
19
20
  onComplete: () => void;
@@ -86,13 +87,13 @@ export function PersonalizingStep({ onComplete }: PersonalizingStepProps) {
86
87
  }, 2000);
87
88
 
88
89
  return () => clearInterval(interval);
89
- }, []);
90
+ }, [circleScale, iconOpacity, onComplete, ringRotation]);
90
91
 
91
92
  useEffect(() => {
92
93
  progress.value = withTiming((currentStep + 1) / STEPS.length, {
93
94
  duration: 500,
94
95
  });
95
- }, [currentStep]);
96
+ }, [currentStep, progress]);
96
97
 
97
98
  const animatedRingStyle = useAnimatedStyle(() => ({
98
99
  transform: [{ rotate: `${ringRotation.value}deg` }],
@@ -1,11 +1,12 @@
1
+ import { MaterialCommunityIcons } from '@expo/vector-icons';
2
+ import * as Haptics from 'expo-haptics';
3
+ import * as StoreReview from 'expo-store-review';
1
4
  import React, { useState } from 'react';
2
5
  import { View } from 'react-native';
3
- import Animated, { FadeInDown, FadeIn, FadeOut } from 'react-native-reanimated';
4
- import { Text, Button, Pressable } from '@/components/ui';
6
+ import Animated, { FadeIn, FadeInDown, FadeOut } from 'react-native-reanimated';
7
+
8
+ import { Button, Pressable, Text } from '@/components/ui';
5
9
  import { useThemeConfig } from '@/lib/use-theme-config';
6
- import { MaterialCommunityIcons } from '@expo/vector-icons';
7
- import * as StoreReview from 'expo-store-review';
8
- import * as Haptics from 'expo-haptics';
9
10
 
10
11
  interface RatingStepProps {
11
12
  onNext: () => void;
@@ -1,10 +1,10 @@
1
+ import * as Haptics from 'expo-haptics';
1
2
  import React, { useState } from 'react';
2
- import { View, ScrollView } from 'react-native';
3
- import Animated, { FadeInDown, FadeIn, FadeOut } from 'react-native-reanimated';
4
- import { Text, Button, Pressable } from '@/components/ui';
5
- import { useThemeConfig } from '@/lib/use-theme-config';
3
+ import { ScrollView, View } from 'react-native';
4
+ import Animated, { FadeIn, FadeInDown, FadeOut } from 'react-native-reanimated';
5
+
6
+ import { Button, Pressable, Text } from '@/components/ui';
6
7
  import { DateTimePickerComponent } from '@/components/ui/date-time-picker';
7
- import * as Haptics from 'expo-haptics';
8
8
 
9
9
  interface ReminderStepProps {
10
10
  selectedFrequency: string | null;
@@ -30,7 +30,6 @@ export function ReminderStep({
30
30
  onSelectTime,
31
31
  onNext,
32
32
  }: ReminderStepProps) {
33
- const theme = useThemeConfig();
34
33
  const showTimePicker =
35
34
  selectedFrequency === 'daily' || selectedFrequency === 'purchase';
36
35
  const [customDate, setCustomDate] = useState(new Date());
@@ -1,10 +1,11 @@
1
+ import { MaterialCommunityIcons } from '@expo/vector-icons';
2
+ import * as Haptics from 'expo-haptics';
1
3
  import React from 'react';
2
4
  import { View } from 'react-native';
3
- import Animated, { ZoomIn, FadeIn, FadeOut } from 'react-native-reanimated';
4
- import { MaterialCommunityIcons } from '@expo/vector-icons';
5
- import { Text, Button } from '@/components/ui';
5
+ import Animated, { FadeIn, FadeOut, ZoomIn } from 'react-native-reanimated';
6
+
7
+ import { Button, Text } from '@/components/ui';
6
8
  import { useThemeConfig } from '@/lib/use-theme-config';
7
- import * as Haptics from 'expo-haptics';
8
9
 
9
10
  interface SafetyStepProps {
10
11
  onComplete: () => void;
@@ -1,10 +1,10 @@
1
- import React from 'react';
2
- import { View, ScrollView } from 'react-native';
3
- import Animated, { FadeIn, FadeOut, FadeInDown } from 'react-native-reanimated';
4
- import { Text, Button, Pressable } from '@/components/ui';
5
- import { useThemeConfig } from '@/lib/use-theme-config';
6
1
  import { MaterialCommunityIcons } from '@expo/vector-icons';
7
2
  import * as Haptics from 'expo-haptics';
3
+ import React from 'react';
4
+ import { ScrollView, View } from 'react-native';
5
+ import Animated, { FadeIn, FadeInDown, FadeOut } from 'react-native-reanimated';
6
+
7
+ import { Button, Pressable, Text } from '@/components/ui';
8
8
 
9
9
  interface StruggleStepProps {
10
10
  selectedStruggle: string | null;
@@ -50,8 +50,6 @@ export function StruggleStep({
50
50
  onSelectStruggle,
51
51
  onNext,
52
52
  }: StruggleStepProps) {
53
- const theme = useThemeConfig();
54
-
55
53
  return (
56
54
  <Animated.View
57
55
  entering={FadeIn}
@@ -1,12 +1,13 @@
1
+ import { MaterialCommunityIcons } from '@expo/vector-icons';
2
+ import * as Clipboard from 'expo-clipboard';
3
+ import * as Haptics from 'expo-haptics';
4
+ import { LinearGradient } from 'expo-linear-gradient';
1
5
  import React, { useState } from 'react';
2
- import { View, ScrollView, TouchableOpacity } from 'react-native';
6
+ import { ScrollView, TouchableOpacity, View } from 'react-native';
3
7
  import Animated, { FadeIn, FadeOut, ZoomIn } from 'react-native-reanimated';
4
- import { LinearGradient } from 'expo-linear-gradient';
5
- import { MaterialCommunityIcons } from '@expo/vector-icons';
6
- import { Text, Button, Pressable } from '@/components/ui';
8
+
9
+ import { Button, Pressable, Text } from '@/components/ui';
7
10
  import { useThemeConfig } from '@/lib/use-theme-config';
8
- import * as Haptics from 'expo-haptics';
9
- import * as Clipboard from 'expo-clipboard';
10
11
 
11
12
  /**
12
13
  * Recommended theme colors for this onboarding (from CashPilot/Expense Tracker app)
@@ -1,13 +1,14 @@
1
+ import { MaterialCommunityIcons } from '@expo/vector-icons';
1
2
  import React from 'react';
2
3
  import { View } from 'react-native';
3
- import { MaterialCommunityIcons } from '@expo/vector-icons';
4
4
  import Animated, {
5
5
  useAnimatedStyle,
6
6
  useSharedValue,
7
7
  withTiming,
8
8
  } from 'react-native-reanimated';
9
- import { useThemeConfig } from '@/lib/use-theme-config';
9
+
10
10
  import { Pressable } from '@/components/ui';
11
+ import { useThemeConfig } from '@/lib/use-theme-config';
11
12
 
12
13
  interface OnboardingHeaderProps {
13
14
  onBack: () => void;
@@ -25,7 +26,7 @@ export function OnboardingHeader({
25
26
 
26
27
  React.useEffect(() => {
27
28
  progressValue.value = withTiming(progress, { duration: 500 });
28
- }, [progress]);
29
+ }, [progress, progressValue]);
29
30
 
30
31
  const animatedProgressStyle = useAnimatedStyle(() => ({
31
32
  width: `${progressValue.value * 100}%`,
@@ -1,15 +1,18 @@
1
1
  {
2
2
  "name": "onboarding",
3
3
  "version": "1.0.0",
4
- "description": "Premium onboarding flows (demo-one, expense-tracker)",
4
+ "description": "User onboarding flow",
5
5
  "copy": [
6
6
  {
7
7
  "from": "apps/native/src/features/onboarding",
8
8
  "to": "apps/native/src/features/onboarding"
9
9
  }
10
10
  ],
11
- "target": "native",
12
- "dependencies": {
13
- "expo": ["react-native-reanimated", "react-native-gesture-handler"]
14
- }
15
- }
11
+ "nav": {
12
+ "href": "/(root)/(protected)/onboarding",
13
+ "label": "Onboarding",
14
+ "icon": "🎯",
15
+ "color": "#84CC16"
16
+ },
17
+ "target": "native"
18
+ }
Binary file
@@ -0,0 +1,74 @@
1
+ import { router, Stack } from 'expo-router';
2
+ import React, { useMemo } from 'react';
3
+ import { ScrollView, View } from 'react-native';
4
+
5
+ import { FeatureButton, FocusAwareStatusBar } from '@/components/ui';
6
+ import { translate } from '@/lib';
7
+ import { useThemeConfig } from '@/lib/use-theme-config';
8
+
9
+ export default function PaywallSelection() {
10
+ const theme = useThemeConfig();
11
+ const optionsTitle = translate('paywall.options_title');
12
+ const screenOptions = useMemo(
13
+ () => ({
14
+ title: optionsTitle,
15
+ headerShown: true,
16
+ headerBackButtonDisplayMode: 'generic' as const,
17
+ }),
18
+ [optionsTitle],
19
+ );
20
+
21
+ const paywallOptions = [
22
+ {
23
+ id: 'remote',
24
+ title: translate('paywall.remote_title'),
25
+ icon: '💳',
26
+ color: '#FBBF24', // Amber
27
+ description: translate('paywall.remote_description'),
28
+ route: '/paywall/remote',
29
+ testID: 'remote-paywall-option',
30
+ },
31
+ {
32
+ id: 'local',
33
+ title: translate('paywall.local_title'),
34
+ icon: '🛒',
35
+ color: '#10B981', // Emerald
36
+ description: translate('paywall.local_description'),
37
+ route: '/paywall/local',
38
+ testID: 'local-paywall-option',
39
+ },
40
+ ];
41
+
42
+ const handleOptionPress = (route: string) => {
43
+ router.push(route as any);
44
+ };
45
+
46
+ return (
47
+ <>
48
+ <FocusAwareStatusBar />
49
+ <Stack.Screen options={screenOptions} />
50
+ <ScrollView
51
+ style={{ backgroundColor: theme.colors.background }}
52
+ className="flex-1"
53
+ showsVerticalScrollIndicator={false}
54
+ >
55
+ <View className="px-6 py-8">
56
+ <View className="-mx-2 flex-row flex-wrap">
57
+ {paywallOptions.map((option) => (
58
+ <View key={option.id} className="w-1/2 px-2 pb-4">
59
+ <FeatureButton
60
+ title={option.title}
61
+ icon={option.icon}
62
+ color={option.color}
63
+ description={option.description}
64
+ testID={option.testID}
65
+ onPress={() => handleOptionPress(option.route)}
66
+ />
67
+ </View>
68
+ ))}
69
+ </View>
70
+ </View>
71
+ </ScrollView>
72
+ </>
73
+ );
74
+ }
@@ -0,0 +1,25 @@
1
+ import { Stack } from 'expo-router';
2
+ import React, { useMemo } from 'react';
3
+
4
+ import LocalPaywall from '@/features/payments/app/local-paywall';
5
+ import { RevenueCatAdapter } from '@/features/payments/services/revenuecat-adapter';
6
+ import { translate } from '@/lib';
7
+
8
+ export default function LocalPaywallScreen() {
9
+ const paymentService = useMemo(() => new RevenueCatAdapter(), []);
10
+ const localTitle = translate('paywall.local_title');
11
+ const screenOptions = useMemo(
12
+ () => ({
13
+ title: localTitle,
14
+ headerShown: false,
15
+ }),
16
+ [localTitle],
17
+ );
18
+
19
+ return (
20
+ <>
21
+ <Stack.Screen options={screenOptions} />
22
+ <LocalPaywall paymentService={paymentService} />
23
+ </>
24
+ );
25
+ }
@@ -0,0 +1,23 @@
1
+ import { Stack } from 'expo-router';
2
+ import React, { useMemo } from 'react';
3
+
4
+ import RemotePaywall from '@/features/payments/app/remote-paywall';
5
+ import { translate } from '@/lib';
6
+
7
+ export default function RemotePaywallScreen() {
8
+ const remoteTitle = translate('paywall.remote_title');
9
+ const screenOptions = useMemo(
10
+ () => ({
11
+ title: remoteTitle,
12
+ headerShown: true,
13
+ }),
14
+ [remoteTitle],
15
+ );
16
+
17
+ return (
18
+ <>
19
+ <Stack.Screen options={screenOptions} />
20
+ <RemotePaywall />
21
+ </>
22
+ );
23
+ }
@@ -3,6 +3,24 @@ import { v } from 'convex/values';
3
3
 
4
4
  import { mutation, query } from './_generated/server';
5
5
 
6
+ // ============================================================================
7
+ // Payment Provider Exports
8
+ // All payment functions are delegated through the provider interface.
9
+ // Change ACTIVE_PROVIDER in payments/index.ts to switch providers.
10
+ // ============================================================================
11
+
12
+ export {
13
+ createCheckout,
14
+ createPortalSession,
15
+ getSubscription,
16
+ cancelSubscription,
17
+ getPayments,
18
+ } from './payments/index';
19
+
20
+ // ============================================================================
21
+ // Credits & Consumable Purchases
22
+ // ============================================================================
23
+
6
24
  /**
7
25
  * Record a consumable purchase (like credits) in the database
8
26
  */
@@ -35,8 +53,8 @@ export const recordConsumablePurchase = mutation({
35
53
  creditsToAdd = args.quantity * 2; // Premium products give 2x credits
36
54
  }
37
55
 
38
- // Update user credits - safely access credits field
39
- const currentCredits = (user as any).credits || 0;
56
+ // Update user credits - credits is optional in schema
57
+ const currentCredits = user.credits ?? 0;
40
58
  const newTotalCredits = currentCredits + creditsToAdd;
41
59
 
42
60
  await ctx.db.patch(userId, {
@@ -73,7 +91,7 @@ export const getUserCredits = query({
73
91
  }
74
92
  const user = await ctx.db.get(userId);
75
93
 
76
- return (user as any)?.credits || 0;
94
+ return user?.credits ?? 0;
77
95
  },
78
96
  });
79
97
 
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "payments",
3
3
  "version": "1.0.0",
4
- "description": "RevenueCat payments",
4
+ "description": "In-app purchases and subscriptions",
5
5
  "copy": [
6
6
  {
7
7
  "from": "apps/native/src/app/paywall",
@@ -12,57 +12,37 @@
12
12
  "to": "apps/native/src/features/payments"
13
13
  },
14
14
  {
15
- "from": "apps/native/src/api-client/payments.ts",
16
- "to": "apps/native/src/api-client/payments.ts"
15
+ "from": "packages/backend/convex/payments",
16
+ "to": "packages/backend/convex/payments"
17
17
  },
18
18
  {
19
19
  "from": "packages/backend/convex/payments.ts",
20
20
  "to": "packages/backend/convex/payments.ts"
21
- },
22
- {
23
- "from": "packages/backend/convex/payments",
24
- "to": "packages/backend/convex/payments"
25
21
  }
26
22
  ],
27
- "configuration": {
28
- "apiClient": {
29
- "exports": ["export { type PaymentsApi, paymentsApi } from './payments';"]
30
- }
31
- },
32
23
  "nav": {
33
24
  "href": "/(root)/(protected)/paywall",
34
25
  "label": "Payments",
35
26
  "icon": "💳",
36
- "color": "#EC4899"
27
+ "color": "#F59E0B"
37
28
  },
38
29
  "target": "native",
39
30
  "dependencies": {
40
- "expo": ["react-native-purchases", "react-native-purchases-ui"]
31
+ "expo": [
32
+ "react-native-purchases",
33
+ "react-native-purchases-ui"
34
+ ]
41
35
  },
42
- "env": [
43
- {
44
- "key": "REVENUECAT_API_KEY_APPLE",
45
- "description": "RevenueCat API key for iOS",
46
- "example": "appl_...",
47
- "link": "https://app.revenuecat.com",
48
- "required": true
49
- },
50
- {
51
- "key": "REVENUECAT_API_KEY_GOOGLE",
52
- "description": "RevenueCat API key for Android",
53
- "example": "goog_...",
54
- "link": "https://app.revenuecat.com",
55
- "required": true
56
- }
57
- ],
58
36
  "manualSteps": [
59
37
  {
60
- "title": "Configure RevenueCat",
61
- "description": "1. Create a project at https://app.revenuecat.com\\n2. Add your app (iOS/Android)\\n3. Configure products/offerings\\n4. Copy API keys to .env.local"
38
+ "title": "Enable RevenueCat initialization",
39
+ "description": "Uncomment the PaymentInitializer in root providers so the SDK is initialized on app startup.",
40
+ "file": "apps/native/src/providers/root-providers.tsx"
62
41
  },
63
42
  {
64
- "title": "Setup App Store / Play Store",
65
- "description": "1. Create in-app products in App Store Connect / Google Play Console\\n2. Link to RevenueCat\\n3. Configure entitlements"
43
+ "title": "Set RevenueCat API keys",
44
+ "description": "Update your .env files with REVENUECAT_API_KEY_APPLE and REVENUECAT_API_KEY_GOOGLE values from RevenueCat.",
45
+ "file": "apps/native/.env.local"
66
46
  }
67
47
  ]
68
48
  }
@@ -0,0 +1,74 @@
1
+ import { router, Stack } from 'expo-router';
2
+ import React, { useMemo } from 'react';
3
+ import { ScrollView, View } from 'react-native';
4
+
5
+ import { FeatureButton, FocusAwareStatusBar } from '@/components/ui';
6
+ import { translate } from '@/lib';
7
+ import { useThemeConfig } from '@/lib/use-theme-config';
8
+
9
+ export default function PaywallSelection() {
10
+ const theme = useThemeConfig();
11
+ const optionsTitle = translate('paywall.options_title');
12
+ const screenOptions = useMemo(
13
+ () => ({
14
+ title: optionsTitle,
15
+ headerShown: true,
16
+ headerBackButtonDisplayMode: 'generic' as const,
17
+ }),
18
+ [optionsTitle],
19
+ );
20
+
21
+ const paywallOptions = [
22
+ {
23
+ id: 'remote',
24
+ title: translate('paywall.remote_title'),
25
+ icon: '💳',
26
+ color: '#FBBF24', // Amber
27
+ description: translate('paywall.remote_description'),
28
+ route: '/paywall/remote',
29
+ testID: 'remote-paywall-option',
30
+ },
31
+ {
32
+ id: 'local',
33
+ title: translate('paywall.local_title'),
34
+ icon: '🛒',
35
+ color: '#10B981', // Emerald
36
+ description: translate('paywall.local_description'),
37
+ route: '/paywall/local',
38
+ testID: 'local-paywall-option',
39
+ },
40
+ ];
41
+
42
+ const handleOptionPress = (route: string) => {
43
+ router.push(route as any);
44
+ };
45
+
46
+ return (
47
+ <>
48
+ <FocusAwareStatusBar />
49
+ <Stack.Screen options={screenOptions} />
50
+ <ScrollView
51
+ style={{ backgroundColor: theme.colors.background }}
52
+ className="flex-1"
53
+ showsVerticalScrollIndicator={false}
54
+ >
55
+ <View className="px-6 py-8">
56
+ <View className="-mx-2 flex-row flex-wrap">
57
+ {paywallOptions.map((option) => (
58
+ <View key={option.id} className="w-1/2 px-2 pb-4">
59
+ <FeatureButton
60
+ title={option.title}
61
+ icon={option.icon}
62
+ color={option.color}
63
+ description={option.description}
64
+ testID={option.testID}
65
+ onPress={() => handleOptionPress(option.route)}
66
+ />
67
+ </View>
68
+ ))}
69
+ </View>
70
+ </View>
71
+ </ScrollView>
72
+ </>
73
+ );
74
+ }
@@ -0,0 +1,25 @@
1
+ import { Stack } from 'expo-router';
2
+ import React, { useMemo } from 'react';
3
+
4
+ import LocalPaywall from '@/features/payments/app/local-paywall';
5
+ import { RevenueCatAdapter } from '@/features/payments/services/revenuecat-adapter';
6
+ import { translate } from '@/lib';
7
+
8
+ export default function LocalPaywallScreen() {
9
+ const paymentService = useMemo(() => new RevenueCatAdapter(), []);
10
+ const localTitle = translate('paywall.local_title');
11
+ const screenOptions = useMemo(
12
+ () => ({
13
+ title: localTitle,
14
+ headerShown: false,
15
+ }),
16
+ [localTitle],
17
+ );
18
+
19
+ return (
20
+ <>
21
+ <Stack.Screen options={screenOptions} />
22
+ <LocalPaywall paymentService={paymentService} />
23
+ </>
24
+ );
25
+ }