create-du-app 0.1.3 → 0.1.4

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (414) hide show
  1. package/README.md +10 -7
  2. package/package.json +6 -5
  3. package/src/index.js +8 -8
  4. package/src/prompts.js +1 -1
  5. package/templates/mobile/expo/.env.example +5 -0
  6. package/templates/mobile/expo/.eslintrc.js +7 -0
  7. package/templates/mobile/expo/.prettierrc.js +7 -0
  8. package/templates/mobile/expo/.svgrrc.js +9 -0
  9. package/templates/mobile/expo/README.md +42 -7
  10. package/templates/mobile/expo/_gitignore +20 -0
  11. package/templates/mobile/expo/_package.json +62 -1
  12. package/templates/mobile/expo/app.json +18 -0
  13. package/templates/mobile/expo/babel.config.js +21 -0
  14. package/templates/mobile/expo/index.js +5 -0
  15. package/templates/mobile/expo/metro.config.js +31 -0
  16. package/templates/mobile/expo/src/app/App.tsx +24 -0
  17. package/templates/mobile/expo/src/app/app-provider.tsx +36 -0
  18. package/templates/mobile/expo/src/app/config/translation.ts +26 -0
  19. package/templates/mobile/expo/src/app/index.ts +1 -0
  20. package/templates/mobile/expo/src/app/navigation/app-route-type.ts +27 -0
  21. package/templates/mobile/expo/src/app/navigation/bottom-tabs.tsx +34 -0
  22. package/templates/mobile/expo/src/app/navigation/index.tsx +14 -0
  23. package/templates/mobile/expo/src/app/stores/auth.store.ts +33 -0
  24. package/templates/mobile/expo/src/app/stores/common.store.ts +19 -0
  25. package/templates/mobile/expo/src/app/stores/index.ts +3 -0
  26. package/templates/mobile/expo/src/app/stores/loading.store.ts +22 -0
  27. package/templates/mobile/expo/src/assets/Images/empty-list.png +0 -0
  28. package/templates/mobile/expo/src/assets/Images/index.ts +5 -0
  29. package/templates/mobile/expo/src/assets/Images/screen-bg-gradian.png +0 -0
  30. package/templates/mobile/expo/src/assets/i18n/en.json +12 -0
  31. package/templates/mobile/expo/src/assets/i18n/fr.json +12 -0
  32. package/templates/mobile/expo/src/assets/lotties/index.ts +3 -0
  33. package/templates/mobile/expo/src/assets/lotties/loading.json +1 -0
  34. package/templates/mobile/expo/src/assets/svgs/arrow-left.svg +3 -0
  35. package/templates/mobile/expo/src/assets/svgs/arrow-right.svg +3 -0
  36. package/templates/mobile/expo/src/assets/svgs/calendar.svg +12 -0
  37. package/templates/mobile/expo/src/assets/svgs/check.svg +3 -0
  38. package/templates/mobile/expo/src/assets/svgs/close.svg +3 -0
  39. package/templates/mobile/expo/src/assets/svgs/eye-hide.svg +3 -0
  40. package/templates/mobile/expo/src/assets/svgs/eye.svg +4 -0
  41. package/templates/mobile/expo/src/assets/svgs/index.ts +29 -0
  42. package/templates/mobile/expo/src/assets/svgs/minus.svg +3 -0
  43. package/templates/mobile/expo/src/assets/svgs/plus.svg +3 -0
  44. package/templates/mobile/expo/src/assets/svgs/search.svg +3 -0
  45. package/templates/mobile/expo/src/assets/svgs/toast-error.svg +5 -0
  46. package/templates/mobile/expo/src/assets/svgs/toast-success.svg +4 -0
  47. package/templates/mobile/expo/src/core/api/endpoints.ts +8 -0
  48. package/templates/mobile/expo/src/core/api/example.api.ts +53 -0
  49. package/templates/mobile/expo/src/core/api/index.ts +4 -0
  50. package/templates/mobile/expo/src/core/components/common/index.ts +1 -0
  51. package/templates/mobile/expo/src/core/components/common/list-empty/index.ts +2 -0
  52. package/templates/mobile/expo/src/core/components/common/list-empty/list-empty.tsx +30 -0
  53. package/templates/mobile/expo/src/core/components/common/list-empty/list-empty.type.ts +5 -0
  54. package/templates/mobile/expo/src/core/components/forms/hf-date-time.tsx +301 -0
  55. package/templates/mobile/expo/src/core/components/forms/hf-password-input.tsx +153 -0
  56. package/templates/mobile/expo/src/core/components/forms/hf-text-input.tsx +59 -0
  57. package/templates/mobile/expo/src/core/components/forms/hf-time-picker.tsx +117 -0
  58. package/templates/mobile/expo/src/core/components/forms/index.ts +4 -0
  59. package/templates/mobile/expo/src/core/components/index.ts +5 -0
  60. package/templates/mobile/expo/src/core/components/loading/index.ts +1 -0
  61. package/templates/mobile/expo/src/core/components/loading/loading-app/index.ts +1 -0
  62. package/templates/mobile/expo/src/core/components/loading/loading-app/loading-app.tsx +50 -0
  63. package/templates/mobile/expo/src/core/components/offline/index.ts +1 -0
  64. package/templates/mobile/expo/src/core/components/offline/offline-banner.tsx +186 -0
  65. package/templates/mobile/expo/src/core/components/screen/index.ts +1 -0
  66. package/templates/mobile/expo/src/core/components/screen/screen-container/index.ts +1 -0
  67. package/templates/mobile/expo/src/core/components/screen/screen-container/screen-container.tsx +252 -0
  68. package/templates/mobile/expo/src/core/components/splash/index.ts +1 -0
  69. package/templates/mobile/expo/src/core/components/splash/splash-overlay/index.ts +1 -0
  70. package/templates/mobile/expo/src/core/components/splash/splash-overlay/splash-overlay.tsx +93 -0
  71. package/templates/mobile/expo/src/core/components/ui/animated-list-item/animated-list-item.tsx +48 -0
  72. package/templates/mobile/expo/src/core/components/ui/animated-list-item/animated-list-item.type.ts +10 -0
  73. package/templates/mobile/expo/src/core/components/ui/animated-list-item/index.ts +2 -0
  74. package/templates/mobile/expo/src/core/components/ui/app-image/app-image.tsx +104 -0
  75. package/templates/mobile/expo/src/core/components/ui/app-image/app-image.type.ts +19 -0
  76. package/templates/mobile/expo/src/core/components/ui/app-image/index.ts +2 -0
  77. package/templates/mobile/expo/src/core/components/ui/asset-placeholder/asset-placeholder.tsx +76 -0
  78. package/templates/mobile/expo/src/core/components/ui/asset-placeholder/index.ts +1 -0
  79. package/templates/mobile/expo/src/core/components/ui/avatar-image/avatar-image.tsx +90 -0
  80. package/templates/mobile/expo/src/core/components/ui/avatar-image/index.ts +1 -0
  81. package/templates/mobile/expo/src/core/components/ui/bottom-sheet/bottom-sheet.tsx +145 -0
  82. package/templates/mobile/expo/src/core/components/ui/bottom-sheet/bottom-sheet.type.ts +10 -0
  83. package/templates/mobile/expo/src/core/components/ui/bottom-sheet/index.ts +2 -0
  84. package/templates/mobile/expo/src/core/components/ui/button/button.style.ts +146 -0
  85. package/templates/mobile/expo/src/core/components/ui/button/button.tsx +97 -0
  86. package/templates/mobile/expo/src/core/components/ui/button/button.type.ts +47 -0
  87. package/templates/mobile/expo/src/core/components/ui/button/index.ts +4 -0
  88. package/templates/mobile/expo/src/core/components/ui/checkbox/checkbox.tsx +127 -0
  89. package/templates/mobile/expo/src/core/components/ui/checkbox/checkbox.type.ts +24 -0
  90. package/templates/mobile/expo/src/core/components/ui/checkbox/index.ts +2 -0
  91. package/templates/mobile/expo/src/core/components/ui/collapsible-section/collapsible-section.tsx +141 -0
  92. package/templates/mobile/expo/src/core/components/ui/collapsible-section/collapsible-section.type.ts +10 -0
  93. package/templates/mobile/expo/src/core/components/ui/collapsible-section/index.ts +2 -0
  94. package/templates/mobile/expo/src/core/components/ui/components.registry.json +313 -0
  95. package/templates/mobile/expo/src/core/components/ui/field/field-frame.tsx +62 -0
  96. package/templates/mobile/expo/src/core/components/ui/field/field.shared.ts +31 -0
  97. package/templates/mobile/expo/src/core/components/ui/header/header.tsx +196 -0
  98. package/templates/mobile/expo/src/core/components/ui/header/header.type.ts +30 -0
  99. package/templates/mobile/expo/src/core/components/ui/header/index.ts +2 -0
  100. package/templates/mobile/expo/src/core/components/ui/icon-button/icon-button.style.ts +23 -0
  101. package/templates/mobile/expo/src/core/components/ui/icon-button/icon-button.tsx +66 -0
  102. package/templates/mobile/expo/src/core/components/ui/icon-button/icon-button.type.ts +25 -0
  103. package/templates/mobile/expo/src/core/components/ui/icon-button/index.ts +2 -0
  104. package/templates/mobile/expo/src/core/components/ui/image-slider/image-slider.tsx +107 -0
  105. package/templates/mobile/expo/src/core/components/ui/image-slider/image-slider.type.ts +10 -0
  106. package/templates/mobile/expo/src/core/components/ui/image-slider/index.ts +2 -0
  107. package/templates/mobile/expo/src/core/components/ui/index.ts +25 -0
  108. package/templates/mobile/expo/src/core/components/ui/label/index.ts +2 -0
  109. package/templates/mobile/expo/src/core/components/ui/label/label.tsx +36 -0
  110. package/templates/mobile/expo/src/core/components/ui/label/label.type.ts +12 -0
  111. package/templates/mobile/expo/src/core/components/ui/modal/index.ts +2 -0
  112. package/templates/mobile/expo/src/core/components/ui/modal/modal.tsx +62 -0
  113. package/templates/mobile/expo/src/core/components/ui/modal/modal.type.ts +11 -0
  114. package/templates/mobile/expo/src/core/components/ui/otp-input/index.ts +2 -0
  115. package/templates/mobile/expo/src/core/components/ui/otp-input/otp-input.tsx +129 -0
  116. package/templates/mobile/expo/src/core/components/ui/otp-input/otp-input.type.ts +20 -0
  117. package/templates/mobile/expo/src/core/components/ui/page-dots/index.ts +1 -0
  118. package/templates/mobile/expo/src/core/components/ui/page-dots/page-dots.tsx +60 -0
  119. package/templates/mobile/expo/src/core/components/ui/radio/index.ts +2 -0
  120. package/templates/mobile/expo/src/core/components/ui/radio/radio.tsx +121 -0
  121. package/templates/mobile/expo/src/core/components/ui/radio/radio.type.ts +20 -0
  122. package/templates/mobile/expo/src/core/components/ui/screen/index.ts +1 -0
  123. package/templates/mobile/expo/src/core/components/ui/screen/screen-gradient.tsx +33 -0
  124. package/templates/mobile/expo/src/core/components/ui/search-box/index.ts +2 -0
  125. package/templates/mobile/expo/src/core/components/ui/search-box/search-box.tsx +162 -0
  126. package/templates/mobile/expo/src/core/components/ui/search-box/search-box.type.ts +26 -0
  127. package/templates/mobile/expo/src/core/components/ui/segmented-control/index.ts +2 -0
  128. package/templates/mobile/expo/src/core/components/ui/segmented-control/segmented-control.tsx +86 -0
  129. package/templates/mobile/expo/src/core/components/ui/segmented-control/segmented-control.type.ts +22 -0
  130. package/templates/mobile/expo/src/core/components/ui/skeleton/index.ts +2 -0
  131. package/templates/mobile/expo/src/core/components/ui/skeleton/skeleton.tsx +106 -0
  132. package/templates/mobile/expo/src/core/components/ui/skeleton/skeleton.type.ts +8 -0
  133. package/templates/mobile/expo/src/core/components/ui/success-state/index.ts +1 -0
  134. package/templates/mobile/expo/src/core/components/ui/success-state/success-state.tsx +68 -0
  135. package/templates/mobile/expo/src/core/components/ui/tabs/index.ts +2 -0
  136. package/templates/mobile/expo/src/core/components/ui/tabs/tabs.tsx +273 -0
  137. package/templates/mobile/expo/src/core/components/ui/tabs/tabs.type.ts +21 -0
  138. package/templates/mobile/expo/src/core/components/ui/tag-input/index.ts +2 -0
  139. package/templates/mobile/expo/src/core/components/ui/tag-input/tag-input.tsx +146 -0
  140. package/templates/mobile/expo/src/core/components/ui/tag-input/tag-input.type.ts +22 -0
  141. package/templates/mobile/expo/src/core/components/ui/text-area/index.ts +2 -0
  142. package/templates/mobile/expo/src/core/components/ui/text-area/text-area.tsx +90 -0
  143. package/templates/mobile/expo/src/core/components/ui/text-area/text-area.type.ts +20 -0
  144. package/templates/mobile/expo/src/core/components/ui/text-field/index.ts +2 -0
  145. package/templates/mobile/expo/src/core/components/ui/text-field/text-field.tsx +116 -0
  146. package/templates/mobile/expo/src/core/components/ui/text-field/text-field.type.ts +21 -0
  147. package/templates/mobile/expo/src/core/components/ui/toggle/index.ts +2 -0
  148. package/templates/mobile/expo/src/core/components/ui/toggle/toggle.tsx +110 -0
  149. package/templates/mobile/expo/src/core/components/ui/toggle/toggle.type.ts +19 -0
  150. package/templates/mobile/expo/src/core/constants/external-urls.constant.ts +5 -0
  151. package/templates/mobile/expo/src/core/constants/hard-data.constant.ts +0 -0
  152. package/templates/mobile/expo/src/core/constants/index.ts +2 -0
  153. package/templates/mobile/expo/src/core/constants/type.constant.ts +3 -0
  154. package/templates/mobile/expo/src/core/context/index.ts +1 -0
  155. package/templates/mobile/expo/src/core/context/shared-transition-context.tsx +35 -0
  156. package/templates/mobile/expo/src/core/hook/index.ts +8 -0
  157. package/templates/mobile/expo/src/core/hook/useActiveRouteName.ts +63 -0
  158. package/templates/mobile/expo/src/core/hook/useAppNavigation.tsx +7 -0
  159. package/templates/mobile/expo/src/core/hook/useBottomInset.tsx +26 -0
  160. package/templates/mobile/expo/src/core/hook/useDebounce.tsx +16 -0
  161. package/templates/mobile/expo/src/core/hook/useEndReached.tsx +46 -0
  162. package/templates/mobile/expo/src/core/hook/useManualRefetch.ts +56 -0
  163. package/templates/mobile/expo/src/core/hook/useNetworkStatus.ts +68 -0
  164. package/templates/mobile/expo/src/core/hook/useTimeout.tsx +30 -0
  165. package/templates/mobile/expo/src/core/index.ts +7 -0
  166. package/templates/mobile/expo/src/core/services/api.service.ts +230 -0
  167. package/templates/mobile/expo/src/core/services/device-id.service.ts +23 -0
  168. package/templates/mobile/expo/src/core/services/index.ts +3 -0
  169. package/templates/mobile/expo/src/core/services/session-end.bridge.ts +26 -0
  170. package/templates/mobile/expo/src/core/theme/dark.theme.ts +10 -0
  171. package/templates/mobile/expo/src/core/theme/index.ts +5 -0
  172. package/templates/mobile/expo/src/core/theme/light.theme.ts +44 -0
  173. package/templates/mobile/expo/src/core/theme/theme-context.tsx +82 -0
  174. package/templates/mobile/expo/src/core/theme/theme.types.ts +26 -0
  175. package/templates/mobile/expo/src/core/theme/use-themed-styles.ts +25 -0
  176. package/templates/mobile/expo/src/core/utils/color.util.tsx +198 -0
  177. package/templates/mobile/expo/src/core/utils/date.util.ts +97 -0
  178. package/templates/mobile/expo/src/core/utils/device-locale.util.ts +22 -0
  179. package/templates/mobile/expo/src/core/utils/emitter/index.ts +161 -0
  180. package/templates/mobile/expo/src/core/utils/emitter/type.ts +40 -0
  181. package/templates/mobile/expo/src/core/utils/enum.util.tsx +15 -0
  182. package/templates/mobile/expo/src/core/utils/font.util.tsx +42 -0
  183. package/templates/mobile/expo/src/core/utils/func.util.ts +48 -0
  184. package/templates/mobile/expo/src/core/utils/greeting.util.ts +20 -0
  185. package/templates/mobile/expo/src/core/utils/image-format.util.ts +117 -0
  186. package/templates/mobile/expo/src/core/utils/image-picker.util.ts +84 -0
  187. package/templates/mobile/expo/src/core/utils/index.ts +18 -0
  188. package/templates/mobile/expo/src/core/utils/linking.util.ts +16 -0
  189. package/templates/mobile/expo/src/core/utils/navigation.util.tsx +100 -0
  190. package/templates/mobile/expo/src/core/utils/number-format.ts +28 -0
  191. package/templates/mobile/expo/src/core/utils/query-client.util.ts +35 -0
  192. package/templates/mobile/expo/src/core/utils/query-persister.util.ts +36 -0
  193. package/templates/mobile/expo/src/core/utils/schema.util.tsx +2713 -0
  194. package/templates/mobile/expo/src/core/utils/size.util.tsx +74 -0
  195. package/templates/mobile/expo/src/core/utils/storage.util.tsx +53 -0
  196. package/templates/mobile/expo/src/core/utils/toast.util.tsx +151 -0
  197. package/templates/mobile/expo/src/core/utils/translator.util.tsx +23 -0
  198. package/templates/mobile/expo/src/core/utils/typography.util.tsx +69 -0
  199. package/templates/mobile/expo/src/core/utils/validate.util.tsx +13 -0
  200. package/templates/mobile/expo/src/declarations.d.ts +54 -0
  201. package/templates/mobile/expo/src/modules/home/home.screen.tsx +33 -0
  202. package/templates/mobile/expo/src/modules/profile/profile.screen.tsx +29 -0
  203. package/templates/mobile/expo/src/scripts/link-fonts.js +60 -0
  204. package/templates/mobile/expo/src/scripts/sync-images.js +56 -0
  205. package/templates/mobile/expo/src/scripts/sync-svgs.js +50 -0
  206. package/templates/mobile/expo/src/types/models.d.ts +24 -0
  207. package/templates/mobile/expo/tsconfig.json +19 -0
  208. package/templates/mobile/rn/.env.example +5 -0
  209. package/templates/mobile/rn/.eslintrc.js +7 -0
  210. package/templates/mobile/rn/.prettierrc.js +7 -0
  211. package/templates/mobile/rn/.svgrrc.js +9 -0
  212. package/templates/mobile/rn/README.md +40 -7
  213. package/templates/mobile/rn/_gitignore +24 -0
  214. package/templates/mobile/rn/_package.json +67 -1
  215. package/templates/mobile/rn/app.json +4 -0
  216. package/templates/mobile/rn/babel.config.js +18 -0
  217. package/templates/mobile/rn/index.js +8 -0
  218. package/templates/mobile/rn/metro.config.js +33 -0
  219. package/templates/mobile/rn/src/app/App.tsx +24 -0
  220. package/templates/mobile/rn/src/app/app-provider.tsx +36 -0
  221. package/templates/mobile/rn/src/app/config/translation.ts +26 -0
  222. package/templates/mobile/rn/src/app/index.ts +1 -0
  223. package/templates/mobile/rn/src/app/navigation/app-route-type.ts +27 -0
  224. package/templates/mobile/rn/src/app/navigation/bottom-tabs.tsx +34 -0
  225. package/templates/mobile/rn/src/app/navigation/index.tsx +14 -0
  226. package/templates/mobile/rn/src/app/stores/auth.store.ts +33 -0
  227. package/templates/mobile/rn/src/app/stores/common.store.ts +19 -0
  228. package/templates/mobile/rn/src/app/stores/index.ts +3 -0
  229. package/templates/mobile/rn/src/app/stores/loading.store.ts +22 -0
  230. package/templates/mobile/rn/src/assets/Images/empty-list.png +0 -0
  231. package/templates/mobile/rn/src/assets/Images/index.ts +5 -0
  232. package/templates/mobile/rn/src/assets/Images/screen-bg-gradian.png +0 -0
  233. package/templates/mobile/rn/src/assets/i18n/en.json +12 -0
  234. package/templates/mobile/rn/src/assets/i18n/fr.json +12 -0
  235. package/templates/mobile/rn/src/assets/lotties/index.ts +3 -0
  236. package/templates/mobile/rn/src/assets/lotties/loading.json +1 -0
  237. package/templates/mobile/rn/src/assets/svgs/arrow-left.svg +3 -0
  238. package/templates/mobile/rn/src/assets/svgs/arrow-right.svg +3 -0
  239. package/templates/mobile/rn/src/assets/svgs/calendar.svg +12 -0
  240. package/templates/mobile/rn/src/assets/svgs/check.svg +3 -0
  241. package/templates/mobile/rn/src/assets/svgs/close.svg +3 -0
  242. package/templates/mobile/rn/src/assets/svgs/eye-hide.svg +3 -0
  243. package/templates/mobile/rn/src/assets/svgs/eye.svg +4 -0
  244. package/templates/mobile/rn/src/assets/svgs/index.ts +29 -0
  245. package/templates/mobile/rn/src/assets/svgs/minus.svg +3 -0
  246. package/templates/mobile/rn/src/assets/svgs/plus.svg +3 -0
  247. package/templates/mobile/rn/src/assets/svgs/search.svg +3 -0
  248. package/templates/mobile/rn/src/assets/svgs/toast-error.svg +5 -0
  249. package/templates/mobile/rn/src/assets/svgs/toast-success.svg +4 -0
  250. package/templates/mobile/rn/src/core/api/endpoints.ts +8 -0
  251. package/templates/mobile/rn/src/core/api/example.api.ts +53 -0
  252. package/templates/mobile/rn/src/core/api/index.ts +4 -0
  253. package/templates/mobile/rn/src/core/components/common/index.ts +1 -0
  254. package/templates/mobile/rn/src/core/components/common/list-empty/index.ts +2 -0
  255. package/templates/mobile/rn/src/core/components/common/list-empty/list-empty.tsx +30 -0
  256. package/templates/mobile/rn/src/core/components/common/list-empty/list-empty.type.ts +5 -0
  257. package/templates/mobile/rn/src/core/components/forms/hf-date-time.tsx +301 -0
  258. package/templates/mobile/rn/src/core/components/forms/hf-password-input.tsx +153 -0
  259. package/templates/mobile/rn/src/core/components/forms/hf-text-input.tsx +59 -0
  260. package/templates/mobile/rn/src/core/components/forms/hf-time-picker.tsx +117 -0
  261. package/templates/mobile/rn/src/core/components/forms/index.ts +4 -0
  262. package/templates/mobile/rn/src/core/components/index.ts +5 -0
  263. package/templates/mobile/rn/src/core/components/loading/index.ts +1 -0
  264. package/templates/mobile/rn/src/core/components/loading/loading-app/index.ts +1 -0
  265. package/templates/mobile/rn/src/core/components/loading/loading-app/loading-app.tsx +50 -0
  266. package/templates/mobile/rn/src/core/components/offline/index.ts +1 -0
  267. package/templates/mobile/rn/src/core/components/offline/offline-banner.tsx +186 -0
  268. package/templates/mobile/rn/src/core/components/screen/index.ts +1 -0
  269. package/templates/mobile/rn/src/core/components/screen/screen-container/index.ts +1 -0
  270. package/templates/mobile/rn/src/core/components/screen/screen-container/screen-container.tsx +252 -0
  271. package/templates/mobile/rn/src/core/components/splash/index.ts +1 -0
  272. package/templates/mobile/rn/src/core/components/splash/splash-overlay/index.ts +1 -0
  273. package/templates/mobile/rn/src/core/components/splash/splash-overlay/splash-overlay.tsx +93 -0
  274. package/templates/mobile/rn/src/core/components/ui/animated-list-item/animated-list-item.tsx +48 -0
  275. package/templates/mobile/rn/src/core/components/ui/animated-list-item/animated-list-item.type.ts +10 -0
  276. package/templates/mobile/rn/src/core/components/ui/animated-list-item/index.ts +2 -0
  277. package/templates/mobile/rn/src/core/components/ui/app-image/app-image.tsx +104 -0
  278. package/templates/mobile/rn/src/core/components/ui/app-image/app-image.type.ts +19 -0
  279. package/templates/mobile/rn/src/core/components/ui/app-image/index.ts +2 -0
  280. package/templates/mobile/rn/src/core/components/ui/asset-placeholder/asset-placeholder.tsx +76 -0
  281. package/templates/mobile/rn/src/core/components/ui/asset-placeholder/index.ts +1 -0
  282. package/templates/mobile/rn/src/core/components/ui/avatar-image/avatar-image.tsx +90 -0
  283. package/templates/mobile/rn/src/core/components/ui/avatar-image/index.ts +1 -0
  284. package/templates/mobile/rn/src/core/components/ui/bottom-sheet/bottom-sheet.tsx +145 -0
  285. package/templates/mobile/rn/src/core/components/ui/bottom-sheet/bottom-sheet.type.ts +10 -0
  286. package/templates/mobile/rn/src/core/components/ui/bottom-sheet/index.ts +2 -0
  287. package/templates/mobile/rn/src/core/components/ui/button/button.style.ts +146 -0
  288. package/templates/mobile/rn/src/core/components/ui/button/button.tsx +97 -0
  289. package/templates/mobile/rn/src/core/components/ui/button/button.type.ts +47 -0
  290. package/templates/mobile/rn/src/core/components/ui/button/index.ts +4 -0
  291. package/templates/mobile/rn/src/core/components/ui/checkbox/checkbox.tsx +127 -0
  292. package/templates/mobile/rn/src/core/components/ui/checkbox/checkbox.type.ts +24 -0
  293. package/templates/mobile/rn/src/core/components/ui/checkbox/index.ts +2 -0
  294. package/templates/mobile/rn/src/core/components/ui/collapsible-section/collapsible-section.tsx +141 -0
  295. package/templates/mobile/rn/src/core/components/ui/collapsible-section/collapsible-section.type.ts +10 -0
  296. package/templates/mobile/rn/src/core/components/ui/collapsible-section/index.ts +2 -0
  297. package/templates/mobile/rn/src/core/components/ui/components.registry.json +313 -0
  298. package/templates/mobile/rn/src/core/components/ui/field/field-frame.tsx +62 -0
  299. package/templates/mobile/rn/src/core/components/ui/field/field.shared.ts +31 -0
  300. package/templates/mobile/rn/src/core/components/ui/header/header.tsx +196 -0
  301. package/templates/mobile/rn/src/core/components/ui/header/header.type.ts +30 -0
  302. package/templates/mobile/rn/src/core/components/ui/header/index.ts +2 -0
  303. package/templates/mobile/rn/src/core/components/ui/icon-button/icon-button.style.ts +23 -0
  304. package/templates/mobile/rn/src/core/components/ui/icon-button/icon-button.tsx +66 -0
  305. package/templates/mobile/rn/src/core/components/ui/icon-button/icon-button.type.ts +25 -0
  306. package/templates/mobile/rn/src/core/components/ui/icon-button/index.ts +2 -0
  307. package/templates/mobile/rn/src/core/components/ui/image-slider/image-slider.tsx +107 -0
  308. package/templates/mobile/rn/src/core/components/ui/image-slider/image-slider.type.ts +10 -0
  309. package/templates/mobile/rn/src/core/components/ui/image-slider/index.ts +2 -0
  310. package/templates/mobile/rn/src/core/components/ui/index.ts +25 -0
  311. package/templates/mobile/rn/src/core/components/ui/label/index.ts +2 -0
  312. package/templates/mobile/rn/src/core/components/ui/label/label.tsx +36 -0
  313. package/templates/mobile/rn/src/core/components/ui/label/label.type.ts +12 -0
  314. package/templates/mobile/rn/src/core/components/ui/modal/index.ts +2 -0
  315. package/templates/mobile/rn/src/core/components/ui/modal/modal.tsx +62 -0
  316. package/templates/mobile/rn/src/core/components/ui/modal/modal.type.ts +11 -0
  317. package/templates/mobile/rn/src/core/components/ui/otp-input/index.ts +2 -0
  318. package/templates/mobile/rn/src/core/components/ui/otp-input/otp-input.tsx +129 -0
  319. package/templates/mobile/rn/src/core/components/ui/otp-input/otp-input.type.ts +20 -0
  320. package/templates/mobile/rn/src/core/components/ui/page-dots/index.ts +1 -0
  321. package/templates/mobile/rn/src/core/components/ui/page-dots/page-dots.tsx +60 -0
  322. package/templates/mobile/rn/src/core/components/ui/radio/index.ts +2 -0
  323. package/templates/mobile/rn/src/core/components/ui/radio/radio.tsx +121 -0
  324. package/templates/mobile/rn/src/core/components/ui/radio/radio.type.ts +20 -0
  325. package/templates/mobile/rn/src/core/components/ui/screen/index.ts +1 -0
  326. package/templates/mobile/rn/src/core/components/ui/screen/screen-gradient.tsx +33 -0
  327. package/templates/mobile/rn/src/core/components/ui/search-box/index.ts +2 -0
  328. package/templates/mobile/rn/src/core/components/ui/search-box/search-box.tsx +162 -0
  329. package/templates/mobile/rn/src/core/components/ui/search-box/search-box.type.ts +26 -0
  330. package/templates/mobile/rn/src/core/components/ui/segmented-control/index.ts +2 -0
  331. package/templates/mobile/rn/src/core/components/ui/segmented-control/segmented-control.tsx +86 -0
  332. package/templates/mobile/rn/src/core/components/ui/segmented-control/segmented-control.type.ts +22 -0
  333. package/templates/mobile/rn/src/core/components/ui/skeleton/index.ts +2 -0
  334. package/templates/mobile/rn/src/core/components/ui/skeleton/skeleton.tsx +106 -0
  335. package/templates/mobile/rn/src/core/components/ui/skeleton/skeleton.type.ts +8 -0
  336. package/templates/mobile/rn/src/core/components/ui/success-state/index.ts +1 -0
  337. package/templates/mobile/rn/src/core/components/ui/success-state/success-state.tsx +68 -0
  338. package/templates/mobile/rn/src/core/components/ui/tabs/index.ts +2 -0
  339. package/templates/mobile/rn/src/core/components/ui/tabs/tabs.tsx +273 -0
  340. package/templates/mobile/rn/src/core/components/ui/tabs/tabs.type.ts +21 -0
  341. package/templates/mobile/rn/src/core/components/ui/tag-input/index.ts +2 -0
  342. package/templates/mobile/rn/src/core/components/ui/tag-input/tag-input.tsx +146 -0
  343. package/templates/mobile/rn/src/core/components/ui/tag-input/tag-input.type.ts +22 -0
  344. package/templates/mobile/rn/src/core/components/ui/text-area/index.ts +2 -0
  345. package/templates/mobile/rn/src/core/components/ui/text-area/text-area.tsx +90 -0
  346. package/templates/mobile/rn/src/core/components/ui/text-area/text-area.type.ts +20 -0
  347. package/templates/mobile/rn/src/core/components/ui/text-field/index.ts +2 -0
  348. package/templates/mobile/rn/src/core/components/ui/text-field/text-field.tsx +116 -0
  349. package/templates/mobile/rn/src/core/components/ui/text-field/text-field.type.ts +21 -0
  350. package/templates/mobile/rn/src/core/components/ui/toggle/index.ts +2 -0
  351. package/templates/mobile/rn/src/core/components/ui/toggle/toggle.tsx +110 -0
  352. package/templates/mobile/rn/src/core/components/ui/toggle/toggle.type.ts +19 -0
  353. package/templates/mobile/rn/src/core/constants/external-urls.constant.ts +5 -0
  354. package/templates/mobile/rn/src/core/constants/hard-data.constant.ts +0 -0
  355. package/templates/mobile/rn/src/core/constants/index.ts +2 -0
  356. package/templates/mobile/rn/src/core/constants/type.constant.ts +3 -0
  357. package/templates/mobile/rn/src/core/context/index.ts +1 -0
  358. package/templates/mobile/rn/src/core/context/shared-transition-context.tsx +35 -0
  359. package/templates/mobile/rn/src/core/hook/index.ts +8 -0
  360. package/templates/mobile/rn/src/core/hook/useActiveRouteName.ts +63 -0
  361. package/templates/mobile/rn/src/core/hook/useAppNavigation.tsx +7 -0
  362. package/templates/mobile/rn/src/core/hook/useBottomInset.tsx +26 -0
  363. package/templates/mobile/rn/src/core/hook/useDebounce.tsx +16 -0
  364. package/templates/mobile/rn/src/core/hook/useEndReached.tsx +46 -0
  365. package/templates/mobile/rn/src/core/hook/useManualRefetch.ts +56 -0
  366. package/templates/mobile/rn/src/core/hook/useNetworkStatus.ts +68 -0
  367. package/templates/mobile/rn/src/core/hook/useTimeout.tsx +30 -0
  368. package/templates/mobile/rn/src/core/index.ts +7 -0
  369. package/templates/mobile/rn/src/core/services/api.service.ts +230 -0
  370. package/templates/mobile/rn/src/core/services/device-id.service.ts +23 -0
  371. package/templates/mobile/rn/src/core/services/index.ts +3 -0
  372. package/templates/mobile/rn/src/core/services/session-end.bridge.ts +26 -0
  373. package/templates/mobile/rn/src/core/theme/dark.theme.ts +10 -0
  374. package/templates/mobile/rn/src/core/theme/index.ts +5 -0
  375. package/templates/mobile/rn/src/core/theme/light.theme.ts +44 -0
  376. package/templates/mobile/rn/src/core/theme/theme-context.tsx +82 -0
  377. package/templates/mobile/rn/src/core/theme/theme.types.ts +26 -0
  378. package/templates/mobile/rn/src/core/theme/use-themed-styles.ts +25 -0
  379. package/templates/mobile/rn/src/core/utils/color.util.tsx +198 -0
  380. package/templates/mobile/rn/src/core/utils/date.util.ts +97 -0
  381. package/templates/mobile/rn/src/core/utils/device-locale.util.ts +22 -0
  382. package/templates/mobile/rn/src/core/utils/emitter/index.ts +161 -0
  383. package/templates/mobile/rn/src/core/utils/emitter/type.ts +40 -0
  384. package/templates/mobile/rn/src/core/utils/enum.util.tsx +15 -0
  385. package/templates/mobile/rn/src/core/utils/font.util.tsx +42 -0
  386. package/templates/mobile/rn/src/core/utils/func.util.ts +48 -0
  387. package/templates/mobile/rn/src/core/utils/greeting.util.ts +20 -0
  388. package/templates/mobile/rn/src/core/utils/image-format.util.ts +117 -0
  389. package/templates/mobile/rn/src/core/utils/image-picker.util.ts +84 -0
  390. package/templates/mobile/rn/src/core/utils/index.ts +18 -0
  391. package/templates/mobile/rn/src/core/utils/linking.util.ts +16 -0
  392. package/templates/mobile/rn/src/core/utils/navigation.util.tsx +100 -0
  393. package/templates/mobile/rn/src/core/utils/number-format.ts +28 -0
  394. package/templates/mobile/rn/src/core/utils/query-client.util.ts +35 -0
  395. package/templates/mobile/rn/src/core/utils/query-persister.util.ts +36 -0
  396. package/templates/mobile/rn/src/core/utils/schema.util.tsx +2713 -0
  397. package/templates/mobile/rn/src/core/utils/size.util.tsx +74 -0
  398. package/templates/mobile/rn/src/core/utils/storage.util.tsx +53 -0
  399. package/templates/mobile/rn/src/core/utils/toast.util.tsx +151 -0
  400. package/templates/mobile/rn/src/core/utils/translator.util.tsx +23 -0
  401. package/templates/mobile/rn/src/core/utils/typography.util.tsx +69 -0
  402. package/templates/mobile/rn/src/core/utils/validate.util.tsx +13 -0
  403. package/templates/mobile/rn/src/declarations.d.ts +54 -0
  404. package/templates/mobile/rn/src/modules/home/home.screen.tsx +33 -0
  405. package/templates/mobile/rn/src/modules/profile/profile.screen.tsx +29 -0
  406. package/templates/mobile/rn/src/scripts/link-fonts.js +60 -0
  407. package/templates/mobile/rn/src/scripts/sync-images.js +56 -0
  408. package/templates/mobile/rn/src/scripts/sync-svgs.js +50 -0
  409. package/templates/mobile/rn/src/types/models.d.ts +24 -0
  410. package/templates/mobile/rn/tsconfig.json +21 -0
  411. package/templates/shared/src/api-endpoints.ts +8 -0
  412. package/templates/shared/src/enums.ts +34 -0
  413. package/templates/shared/src/external-urls.ts +5 -0
  414. package/templates/shared/src/index.ts +6 -3
@@ -0,0 +1,117 @@
1
+ export type SupportedImageMimeType = 'image/jpeg' | 'image/png';
2
+
3
+ export type ImageAsset = {
4
+ uri: string;
5
+ type?: string | null;
6
+ name?: string | null;
7
+ fileName?: string | null;
8
+ };
9
+
10
+ const SUPPORTED_MIME_TYPES = new Set<SupportedImageMimeType>([
11
+ 'image/jpeg',
12
+ 'image/png',
13
+ ]);
14
+
15
+ const MIME_TYPE_BY_EXTENSION: Record<string, SupportedImageMimeType> = {
16
+ jpg: 'image/jpeg',
17
+ jpeg: 'image/jpeg',
18
+ png: 'image/png',
19
+ };
20
+
21
+ const sanitizeValue = (value?: string | null): string =>
22
+ value?.split('?')[0]?.split('#')[0]?.trim() ?? '';
23
+
24
+ const getExtension = (value?: string | null): string | undefined => {
25
+ const sanitized = sanitizeValue(value);
26
+
27
+ if (!sanitized) {
28
+ return undefined;
29
+ }
30
+
31
+ const lastDotIndex = sanitized.lastIndexOf('.');
32
+
33
+ if (lastDotIndex === -1 || lastDotIndex === sanitized.length - 1) {
34
+ return undefined;
35
+ }
36
+
37
+ return sanitized.slice(lastDotIndex + 1).toLowerCase();
38
+ };
39
+
40
+ export const getSupportedImageMimeType = (
41
+ asset: ImageAsset,
42
+ ): SupportedImageMimeType | undefined => {
43
+ const normalizedType = sanitizeValue(asset.type).toLowerCase();
44
+
45
+ if (
46
+ normalizedType &&
47
+ SUPPORTED_MIME_TYPES.has(normalizedType as SupportedImageMimeType)
48
+ ) {
49
+ return normalizedType as SupportedImageMimeType;
50
+ }
51
+
52
+ const extension =
53
+ getExtension(asset.name) ??
54
+ getExtension(asset.fileName) ??
55
+ getExtension(asset.uri);
56
+
57
+ if (!extension) {
58
+ return undefined;
59
+ }
60
+
61
+ return MIME_TYPE_BY_EXTENSION[extension];
62
+ };
63
+
64
+ export const isSupportedImageAsset = (asset: ImageAsset): boolean =>
65
+ Boolean(asset.uri && getSupportedImageMimeType(asset));
66
+
67
+ const buildNormalizedFileName = (
68
+ asset: ImageAsset,
69
+ index: number,
70
+ mimeType: SupportedImageMimeType,
71
+ ): string => {
72
+ const extension = mimeType === 'image/png' ? 'png' : 'jpg';
73
+ const rawName = sanitizeValue(asset.name) || sanitizeValue(asset.fileName);
74
+
75
+ if (rawName) {
76
+ const baseName = rawName.replace(/\.[^/.]+$/, '');
77
+ return `${baseName}.${extension}`;
78
+ }
79
+
80
+ return `photo_${Date.now()}_${index}.${extension}`;
81
+ };
82
+
83
+ export const normalizeImageAsset = (
84
+ asset: ImageAsset,
85
+ index = 0,
86
+ ): { uri: string; type: SupportedImageMimeType; name: string } | null => {
87
+ const type = getSupportedImageMimeType(asset);
88
+
89
+ if (!asset.uri || !type) {
90
+ return null;
91
+ }
92
+
93
+ return {
94
+ uri: asset.uri,
95
+ type,
96
+ name: buildNormalizedFileName(asset, index, type),
97
+ };
98
+ };
99
+
100
+ export const filterSupportedImageAssets = (assets: ImageAsset[]) => {
101
+ const supported: ImageAsset[] = [];
102
+ const unsupported: ImageAsset[] = [];
103
+
104
+ assets.forEach(asset => {
105
+ if (isSupportedImageAsset(asset)) {
106
+ supported.push(asset);
107
+ return;
108
+ }
109
+
110
+ unsupported.push(asset);
111
+ });
112
+
113
+ return {
114
+ supported,
115
+ unsupported,
116
+ };
117
+ };
@@ -0,0 +1,84 @@
1
+ import { PermissionsAndroid, Platform } from 'react-native'
2
+ import {
3
+ CameraOptions,
4
+ ImageLibraryOptions,
5
+ launchCamera,
6
+ launchImageLibrary,
7
+ } from 'react-native-image-picker'
8
+
9
+ export class ImagePicker {
10
+ private static optionsDefault: CameraOptions | ImageLibraryOptions = {
11
+ mediaType: 'photo',
12
+ quality: 0.8,
13
+ maxHeight: 800,
14
+ maxWidth: 800,
15
+ }
16
+
17
+ public static async requestCameraPermission() {
18
+ if (Platform.OS === 'android') {
19
+ const granted = await PermissionsAndroid.request(
20
+ PermissionsAndroid.PERMISSIONS.CAMERA,
21
+ {
22
+ title: 'Camera Permission',
23
+ message: 'App needs access to your camera',
24
+ buttonNeutral: 'Ask Me Later',
25
+ buttonNegative: 'Cancel',
26
+ buttonPositive: 'OK',
27
+ },
28
+ )
29
+ return granted === PermissionsAndroid.RESULTS.GRANTED
30
+ }
31
+ return true
32
+ }
33
+
34
+ public static async lauchCamera(
35
+ options?: CameraOptions,
36
+ ): Promise<IMediaFormData | undefined> {
37
+ const granted = await this.requestCameraPermission()
38
+ if (!granted) {
39
+ return undefined
40
+ }
41
+
42
+ const result = await launchCamera({
43
+ ...this.optionsDefault,
44
+ ...options,
45
+ })
46
+ if (result?.assets?.[0]) {
47
+ const media = {
48
+ name:
49
+ result?.assets?.[0]?.fileName ||
50
+ `img-${new Date().getTime()}`,
51
+ type: result?.assets?.[0]?.type || '',
52
+ uri: result?.assets?.[0]?.uri || '',
53
+ }
54
+ return media
55
+ } else {
56
+ return undefined
57
+ }
58
+ }
59
+
60
+ public static async launchImagesLibrary(
61
+ options?: ImageLibraryOptions,
62
+ ): Promise<IMediaFormData[]> {
63
+ return new Promise((resolve, reject) => {
64
+ launchImageLibrary({
65
+ ...this.optionsDefault,
66
+ ...options,
67
+ })
68
+ .then(({ assets, didCancel, errorCode }) => {
69
+ if (didCancel || errorCode) {
70
+ return reject(didCancel || errorCode)
71
+ }
72
+ const media = (assets || []).map(
73
+ ({ fileName, uri, type }) => ({
74
+ name: fileName || `img-${new Date().getTime()}`,
75
+ type: type || '',
76
+ uri: uri || '',
77
+ }),
78
+ )
79
+ assets && resolve(media)
80
+ })
81
+ .catch(reject)
82
+ })
83
+ }
84
+ }
@@ -0,0 +1,18 @@
1
+ export * from './color.util';
2
+ export * from './greeting.util';
3
+ export * from './navigation.util';
4
+ export * from './schema.util';
5
+ export * from './size.util';
6
+ export * from './storage.util';
7
+ export * from './translator.util';
8
+ export * from './validate.util';
9
+ export * from './font.util';
10
+ export * from './typography.util';
11
+ export * from './image-picker.util';
12
+ export * from './func.util';
13
+ export * from './enum.util';
14
+ export * from './toast.util';
15
+ export * from './query-client.util';
16
+ export * from './query-persister.util';
17
+ export * from './date.util';
18
+ export * from './linking.util';
@@ -0,0 +1,16 @@
1
+ import { Linking } from 'react-native';
2
+ import { Translator } from './translator.util';
3
+ import { showErrorToast } from './toast.util';
4
+
5
+ export const openExternalUrl = async (url: string) => {
6
+ try {
7
+ const supported = await Linking.canOpenURL(url);
8
+ if (!supported) {
9
+ showErrorToast('', Translator.translate('common.cannotOpenUrl'));
10
+ return;
11
+ }
12
+ await Linking.openURL(url);
13
+ } catch {
14
+ showErrorToast('', Translator.translate('common.cannotOpenUrl'));
15
+ }
16
+ };
@@ -0,0 +1,100 @@
1
+ import { CommonActions, NavigationContainerRef, Route, StackActions } from '@react-navigation/native';
2
+ import { NavigatorParamList, Screen } from '@src/app/navigation/app-route-type';
3
+ import { createRef } from 'react';
4
+
5
+ export const navigationRef = createRef<NavigationContainerRef<NavigatorParamList>>();
6
+
7
+ type Args<RouteName extends keyof NavigatorParamList> = RouteName extends unknown
8
+ ? undefined extends NavigatorParamList[RouteName]
9
+ ? [screen: RouteName] | [screen: RouteName, params: NavigatorParamList[RouteName]]
10
+ : [screen: RouteName, params: NavigatorParamList[RouteName]]
11
+ : never;
12
+
13
+ export class NavigationUtil {
14
+ static reset(name?: Screen) {
15
+ if (navigationRef.current) {
16
+ navigationRef.current.reset({
17
+ index: 0,
18
+ routes: [{ name: name || Screen.SignIn }],
19
+ });
20
+ }
21
+ }
22
+
23
+ static navigate<RouteName extends keyof NavigatorParamList>(...args: Args<RouteName>) {
24
+ if (navigationRef.current) {
25
+ navigationRef.current.navigate(...args);
26
+ }
27
+ }
28
+
29
+ static push<RouteName extends keyof NavigatorParamList>(...args: Args<RouteName>) {
30
+ if (navigationRef.current) {
31
+ const [name, params] = args;
32
+ navigationRef.current.dispatch(StackActions.push(name, params));
33
+ }
34
+ }
35
+
36
+ static replace<RouteName extends keyof NavigatorParamList>(...args: Args<RouteName>) {
37
+ if (navigationRef.current) {
38
+ navigationRef.current.dispatch(StackActions.replace(args[0], args[1]));
39
+ }
40
+ }
41
+
42
+ static pop(count?: number) {
43
+ if (navigationRef.current) {
44
+ navigationRef.current.dispatch(StackActions.pop(count));
45
+ }
46
+ }
47
+
48
+ static popToTop() {
49
+ if (navigationRef.current) {
50
+ navigationRef.current.dispatch(StackActions.popToTop());
51
+ }
52
+ }
53
+
54
+ static goBack() {
55
+ if (navigationRef.current) {
56
+ navigationRef.current.goBack();
57
+ }
58
+ }
59
+
60
+ /**
61
+ * Reset to HomeTab with the specified tab active (preserving its internal
62
+ * history), then push the given screen on top of the root stack so the
63
+ * user can navigate back to the correct tab.
64
+ *
65
+ * @param tabIndex - which tab to activate inside HomeTab (default: 0)
66
+ */
67
+ static resetToHomeTabThenNavigate<RouteName extends keyof NavigatorParamList>(
68
+ tabIndex: number = 0,
69
+ ...args: Args<RouteName>
70
+ ) {
71
+ if (!navigationRef.current) return;
72
+ const [screen, params] = args;
73
+ const rootState = navigationRef.current.getRootState();
74
+ const homeTabRoute = rootState.routes[0]; // HomeTab is always the first route
75
+ const existingRoutes = [...(homeTabRoute.state?.routes ?? [])] as any[];
76
+
77
+ if (existingRoutes.length === 0) {
78
+ // HomeTab hasn't mounted yet — navigate normally
79
+ navigationRef.current.navigate(screen as any, params);
80
+ return;
81
+ }
82
+
83
+ navigationRef.current.dispatch(
84
+ CommonActions.reset({
85
+ ...rootState,
86
+ index: 1,
87
+ routes: [
88
+ {
89
+ ...homeTabRoute,
90
+ state: {
91
+ routes: existingRoutes,
92
+ index: tabIndex, // activate the requested tab
93
+ },
94
+ },
95
+ { name: screen as string, params: params as object | undefined },
96
+ ] as Omit<Route<string>, 'key'>[],
97
+ }),
98
+ );
99
+ }
100
+ }
@@ -0,0 +1,28 @@
1
+ export const priceFormatter = new Intl.NumberFormat('fr-FR', {
2
+ style: 'decimal',
3
+ minimumFractionDigits: 0,
4
+ maximumFractionDigits: 2,
5
+ });
6
+
7
+ export const normalizeNumber = (value?: string | number | null): number | null => {
8
+ if (value === null || value === undefined) return null;
9
+ const parsed = typeof value === 'number' ? value : Number(value);
10
+ return Number.isFinite(parsed) ? parsed : null;
11
+ };
12
+
13
+ export const formatPrice = (value?: string | number | null, fallbackPrice?: string | number | null): string => {
14
+ const normalized = normalizeNumber(value);
15
+
16
+ if (normalized === null || normalized === 0) {
17
+ const fb = normalizeNumber(fallbackPrice);
18
+ return fb !== null ? priceFormatter.format(fb) + ' €' : '—';
19
+ }
20
+
21
+ return priceFormatter.format(normalized) + ' €';
22
+ };
23
+
24
+ export const formatPriceLabel = (params: { price_label?: string | null; price?: string | number | null }): string => {
25
+ if (params.price_label) return params.price_label;
26
+ if (params.price) return formatPrice(params.price);
27
+ return '—';
28
+ };
@@ -0,0 +1,35 @@
1
+ import { QueryClient } from '@tanstack/react-query';
2
+
3
+ // Persisted queries are restored from disk on a cold start, so the cache must
4
+ // survive far longer than React Query's 5-minute default garbage-collection
5
+ // window — otherwise an entry would be pruned in memory and never re-written to
6
+ // disk, leaving nothing to show on a later offline launch. 24h keeps a day's
7
+ // worth of feeds/lists ready offline while still expiring stale data.
8
+ const OFFLINE_GC_TIME = 24 * 60 * 60 * 1000;
9
+
10
+ class AppQueryClient {
11
+ private static instance: QueryClient;
12
+
13
+ static getInstance(): QueryClient {
14
+ if (!AppQueryClient.instance) {
15
+ AppQueryClient.instance = new QueryClient({
16
+ defaultOptions: {
17
+ queries: {
18
+ gcTime: OFFLINE_GC_TIME,
19
+ // `onlineManager` (wired to NetInfo in app-provider) already pauses
20
+ // queries while offline, so a failure here is a real server/timeout
21
+ // error — retrying once is enough; more just delays the cached
22
+ // fallback from showing.
23
+ retry: 1,
24
+ },
25
+ },
26
+ });
27
+ }
28
+ return AppQueryClient.instance;
29
+ }
30
+ }
31
+
32
+ export const queryClient = AppQueryClient.getInstance();
33
+
34
+ export default AppQueryClient;
35
+
@@ -0,0 +1,36 @@
1
+ import { createMMKV } from 'react-native-mmkv';
2
+ import {
3
+ PersistedClient,
4
+ Persister,
5
+ } from '@tanstack/react-query-persist-client';
6
+
7
+ /**
8
+ * Dedicated MMKV store for the React Query cache. Kept separate from any other
9
+ * MMKV usage so clearing the query cache never touches unrelated app storage.
10
+ * MMKV is synchronous + native-fast, so persisting on every cache mutation is
11
+ * cheap enough to keep the on-disk snapshot fresh for an offline cold start.
12
+ */
13
+ const queryStorage = createMMKV({ id: 'react-query-cache' });
14
+
15
+ const CACHE_KEY = 'REACT_QUERY_OFFLINE_CACHE';
16
+
17
+ /**
18
+ * A {@link Persister} backed by MMKV. The persist client writes the whole
19
+ * dehydrated query cache under a single key; on launch it reads it straight
20
+ * back so the last-seen feeds/lists/profile are on screen before the network
21
+ * even resolves — the foundation of the offline-first experience.
22
+ */
23
+ export const mmkvQueryPersister: Persister = {
24
+ persistClient: (client: PersistedClient) => {
25
+ queryStorage.set(CACHE_KEY, JSON.stringify(client));
26
+ },
27
+ restoreClient: () => {
28
+ const cached = queryStorage.getString(CACHE_KEY);
29
+ return cached ? (JSON.parse(cached) as PersistedClient) : undefined;
30
+ },
31
+ removeClient: () => {
32
+ queryStorage.remove(CACHE_KEY);
33
+ },
34
+ };
35
+
36
+ export default mmkvQueryPersister;