xertica-ui 2.4.0 → 2.4.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 (524) hide show
  1. package/CHANGELOG.md +572 -564
  2. package/README.md +417 -417
  3. package/assets/xertica-logo.svg +37 -37
  4. package/assets/xertica-x-logo.svg +20 -20
  5. package/bin/cli.ts +1244 -1244
  6. package/bin/language-config.ts +358 -358
  7. package/components/assistant/code-block/CodeBlock.tsx +268 -268
  8. package/components/assistant/formatted-document/FormattedDocument.tsx +147 -147
  9. package/components/assistant/modern-chat-input/ModernChatInput.tsx +564 -564
  10. package/components/assistant/xertica-assistant/parts/AssistantCollapsedView.tsx +99 -99
  11. package/components/assistant/xertica-assistant/parts/AssistantConversationList.tsx +104 -104
  12. package/components/assistant/xertica-assistant/parts/AssistantDocumentEditor.tsx +81 -81
  13. package/components/assistant/xertica-assistant/parts/AssistantFeedbackDialog.tsx +88 -88
  14. package/components/assistant/xertica-assistant/parts/AssistantHeader.tsx +75 -75
  15. package/components/assistant/xertica-assistant/parts/AssistantMessageBubble.tsx +564 -564
  16. package/components/assistant/xertica-assistant/parts/AssistantTabBar.tsx +67 -67
  17. package/components/assistant/xertica-assistant/parts/AssistantWelcomeScreen.tsx +103 -103
  18. package/components/assistant/xertica-assistant/use-assistant.ts +615 -615
  19. package/components/assistant/xertica-assistant/xertica-assistant.tsx +611 -611
  20. package/components/blocks/card-patterns/ActivityCard.tsx +100 -100
  21. package/components/blocks/card-patterns/ActivityCardSkeleton.tsx +56 -56
  22. package/components/blocks/card-patterns/FeatureCardSkeleton.tsx +58 -58
  23. package/components/blocks/card-patterns/NotificationCard.tsx +140 -140
  24. package/components/blocks/card-patterns/NotificationCardSkeleton.tsx +81 -81
  25. package/components/blocks/card-patterns/ProfileCard.tsx +112 -112
  26. package/components/blocks/card-patterns/ProfileCardSkeleton.tsx +69 -69
  27. package/components/blocks/card-patterns/ProjectCard.tsx +123 -123
  28. package/components/blocks/card-patterns/ProjectCardSkeleton.tsx +67 -67
  29. package/components/blocks/card-patterns/QuickActionCardSkeleton.tsx +44 -44
  30. package/components/blocks/card-patterns/card-patterns.stories.tsx +594 -594
  31. package/components/blocks/card-patterns/index.ts +29 -29
  32. package/components/brand/language-selector/LanguageSelector.tsx +102 -102
  33. package/components/brand/language-selector/language-selector.stories.tsx +111 -111
  34. package/components/brand/language-selector/language-selector.test.tsx +101 -101
  35. package/components/brand/theme-toggle/ThemeToggle.tsx +74 -74
  36. package/components/brand/xertica-provider/XerticaProvider.tsx +109 -109
  37. package/components/brand/xertica-provider/xertica-provider.mdx +61 -61
  38. package/components/index.ts +86 -86
  39. package/components/layout/sidebar/sidebar.mdx +1 -1
  40. package/components/layout/sidebar/sidebar.tsx +1079 -1079
  41. package/components/media/FloatingMediaWrapper.tsx +371 -371
  42. package/components/media/audio-player/AudioPlayer.tsx +768 -768
  43. package/components/media/video-player/VideoPlayer.tsx +310 -310
  44. package/components/pages/forgot-password-page/ForgotPasswordPage.tsx +188 -188
  45. package/components/pages/home-content/HomeContent.tsx +120 -120
  46. package/components/pages/home-content/home-content.mdx +62 -62
  47. package/components/pages/home-page/HomePage.tsx +78 -78
  48. package/components/pages/home-page/home-page.mdx +53 -53
  49. package/components/pages/login-page/LoginPage.tsx +218 -218
  50. package/components/pages/reset-password-page/ResetPasswordPage.tsx +243 -243
  51. package/components/pages/template-content/TemplateContent.tsx +1354 -1354
  52. package/components/pages/template-content/template-content.mdx +61 -61
  53. package/components/pages/template-page/template-page.mdx +53 -53
  54. package/components/pages/verify-email-page/VerifyEmailPage.tsx +206 -206
  55. package/components/shared/error-boundary.stories.tsx +114 -114
  56. package/components/shared/error-boundary.tsx +150 -150
  57. package/components/shared/error-fallbacks.tsx +222 -222
  58. package/components/ui/accordion/accordion.mdx +8 -8
  59. package/components/ui/alert/alert.mdx +8 -8
  60. package/components/ui/alert-dialog/alert-dialog.mdx +8 -8
  61. package/components/ui/aspect-ratio/aspect-ratio.mdx +8 -8
  62. package/components/ui/assistant-chart/assistant-chart.mdx +8 -8
  63. package/components/ui/avatar/avatar.mdx +8 -8
  64. package/components/ui/badge/badge.mdx +8 -8
  65. package/components/ui/breadcrumb/breadcrumb.mdx +8 -8
  66. package/components/ui/button/button.mdx +8 -8
  67. package/components/ui/calendar/calendar.mdx +8 -8
  68. package/components/ui/card/card.mdx +8 -8
  69. package/components/ui/carousel/carousel.mdx +8 -8
  70. package/components/ui/chart/chart.mdx +8 -8
  71. package/components/ui/checkbox/checkbox.mdx +8 -8
  72. package/components/ui/collapsible/collapsible.mdx +8 -8
  73. package/components/ui/command/command.mdx +8 -8
  74. package/components/ui/context-menu/context-menu.mdx +8 -8
  75. package/components/ui/dialog/dialog.mdx +8 -8
  76. package/components/ui/drawer/drawer.mdx +8 -8
  77. package/components/ui/dropdown-menu/dropdown-menu.mdx +8 -8
  78. package/components/ui/empty/empty.mdx +8 -8
  79. package/components/ui/file-upload/file-upload.mdx +8 -8
  80. package/components/ui/hover-card/hover-card.mdx +8 -8
  81. package/components/ui/input/input.mdx +8 -8
  82. package/components/ui/input-otp/input-otp.mdx +8 -8
  83. package/components/ui/label/label.mdx +8 -8
  84. package/components/ui/map/map.mdx +8 -8
  85. package/components/ui/menubar/menubar.mdx +8 -8
  86. package/components/ui/navigation-menu/navigation-menu.mdx +8 -8
  87. package/components/ui/notification-badge/notification-badge.mdx +8 -8
  88. package/components/ui/pagination/pagination.mdx +8 -8
  89. package/components/ui/popover/popover.mdx +8 -8
  90. package/components/ui/progress/progress.mdx +8 -8
  91. package/components/ui/radio-group/radio-group.mdx +8 -8
  92. package/components/ui/rating/rating.mdx +8 -8
  93. package/components/ui/resizable/resizable.mdx +8 -8
  94. package/components/ui/route-map/route-map.mdx +8 -8
  95. package/components/ui/scroll-area/scroll-area.mdx +8 -8
  96. package/components/ui/search/search.mdx +8 -8
  97. package/components/ui/select/select.mdx +8 -8
  98. package/components/ui/separator/separator.mdx +8 -8
  99. package/components/ui/sheet/sheet.mdx +8 -8
  100. package/components/ui/simple-map/simple-map.mdx +8 -8
  101. package/components/ui/skeleton/skeleton.mdx +8 -8
  102. package/components/ui/slider/slider.mdx +8 -8
  103. package/components/ui/sonner/sonner.mdx +8 -8
  104. package/components/ui/stats-card/index.ts +2 -2
  105. package/components/ui/stats-card/stats-card-skeleton.tsx +60 -60
  106. package/components/ui/stats-card/stats-card.mdx +8 -8
  107. package/components/ui/stats-card/stats-card.stories.tsx +117 -99
  108. package/components/ui/stats-card/stats-card.tsx +18 -2
  109. package/components/ui/stepper/stepper.mdx +8 -8
  110. package/components/ui/switch/switch.mdx +8 -8
  111. package/components/ui/table/table.mdx +8 -8
  112. package/components/ui/tabs/tabs.mdx +8 -8
  113. package/components/ui/textarea/textarea.mdx +8 -8
  114. package/components/ui/timeline/timeline.mdx +8 -8
  115. package/components/ui/toggle/toggle.mdx +8 -8
  116. package/components/ui/toggle-group/toggle-group.mdx +8 -8
  117. package/components/ui/tooltip/tooltip.mdx +8 -8
  118. package/components/ui/tree-view/tree-view.mdx +8 -8
  119. package/components.json +153 -533
  120. package/contexts/AuthContext.tsx +121 -121
  121. package/contexts/LanguageContext.test.tsx +121 -121
  122. package/contexts/LanguageContext.tsx +250 -250
  123. package/dist/AssistantChart-BKVtGUKF.js +3383 -0
  124. package/dist/{AssistantChart-BAx9VQvb.cjs → AssistantChart-Bdd44uBn.cjs} +388 -127
  125. package/dist/{AssistantChart-CVko2A1W.js → AssistantChart-CFhDdGyU.js} +391 -130
  126. package/dist/{AssistantChart-CVzmmhx4.js → AssistantChart-C_hwFRRr.js} +4 -4
  127. package/dist/{AssistantChart-BAudAfne.cjs → AssistantChart-CldVCVDe.cjs} +5 -5
  128. package/dist/{AssistantChart-BP8upjMk.js → AssistantChart-Cu3m7RBo.js} +5 -5
  129. package/dist/AssistantChart-CxGjH7Qk.js +3477 -0
  130. package/dist/AssistantChart-DIpshm3i.js +4784 -0
  131. package/dist/AssistantChart-D_PTeu8P.cjs +3503 -0
  132. package/dist/{AssistantChart-9w31gdAb.cjs → AssistantChart-DoZCyS5r.cjs} +4 -4
  133. package/dist/AssistantChart-WeycT5Pd.cjs +3551 -0
  134. package/dist/AssistantChart-zjsy2GaZ.cjs +4810 -0
  135. package/dist/AudioPlayer-B1lt5cPl.cjs +989 -0
  136. package/dist/AudioPlayer-BZ7bibzU.cjs +982 -0
  137. package/dist/AudioPlayer-BpRPS4-1.cjs +1277 -0
  138. package/dist/AudioPlayer-C12BjQBV.cjs +997 -0
  139. package/dist/{AudioPlayer-1ypwE2Wh.cjs → AudioPlayer-CFeV8t-5.cjs} +1 -1
  140. package/dist/{AudioPlayer-DuKXrCfy.js → AudioPlayer-CGRUtUdN.js} +1 -1
  141. package/dist/AudioPlayer-Coly3q5R.js +1278 -0
  142. package/dist/AudioPlayer-CySJIyvL.js +937 -0
  143. package/dist/AudioPlayer-DMcG_c7L.js +990 -0
  144. package/dist/AudioPlayer-DcFKRJE_.js +998 -0
  145. package/dist/AudioPlayer-IAU5q5T1.cjs +936 -0
  146. package/dist/AudioPlayer-e8LfNoqO.js +983 -0
  147. package/dist/BrandColorsContext-565dDHd5.js +660 -0
  148. package/dist/BrandColorsContext-BcJbtkqn.cjs +659 -0
  149. package/dist/{xertica-assistant-Qp3ydksa.cjs → CodeBlock-7TTgmdGG.cjs} +263 -51
  150. package/dist/{xertica-assistant-gnCJdcZY.js → CodeBlock-BeSt1h5P.js} +219 -7
  151. package/dist/CodeBlock-BgfYL_rD.cjs +2094 -0
  152. package/dist/CodeBlock-BlcqlA9M.cjs +2094 -0
  153. package/dist/CodeBlock-Bnmeu5ez.cjs +2094 -0
  154. package/dist/CodeBlock-BtfPlbAI.js +2078 -0
  155. package/dist/CodeBlock-CIySIuYr.js +2078 -0
  156. package/dist/CodeBlock-CuPtUM-7.cjs +2094 -0
  157. package/dist/CodeBlock-D6ffWXgc.js +2078 -0
  158. package/dist/CodeBlock-D8dcwbit.cjs +2094 -0
  159. package/dist/CodeBlock-DMZrFnlw.cjs +2094 -0
  160. package/dist/CodeBlock-DlBehYN8.js +2078 -0
  161. package/dist/CodeBlock-DnYNI8rQ.js +2078 -0
  162. package/dist/CodeBlock-DvKWbSnE.cjs +2094 -0
  163. package/dist/CodeBlock-DwMCfkFY.js +2078 -0
  164. package/dist/CodeBlock-Dy6CNYyj.js +2078 -0
  165. package/dist/CodeBlock-U1pPOQI7.cjs +2094 -0
  166. package/dist/CodeBlock-f_GpNhEB.js +2078 -0
  167. package/dist/CodeBlock-oB6u8nI1.js +2078 -0
  168. package/dist/CodeBlock-tZC31B73.cjs +2094 -0
  169. package/dist/FeatureCard-CxC-7C-C.cjs +300 -0
  170. package/dist/FeatureCard-DbHWCb4E.js +301 -0
  171. package/dist/ImageWithFallback-CGtidP6B.cjs +4542 -0
  172. package/dist/ImageWithFallback-lsg3pdFg.js +4508 -0
  173. package/dist/{LanguageContext-DvUt5jBg.cjs → LanguageContext-B_KFTCzT.cjs} +2 -2
  174. package/dist/{LanguageContext-BwhwC3G2.js → LanguageContext-CS14yCpi.js} +2 -2
  175. package/dist/{XerticaXLogo-DHz5SugF.js → LanguageSelector-B5YfbHra.js} +115 -136
  176. package/dist/{XerticaXLogo-DTee_y8X.cjs → LanguageSelector-D6uacAIM.cjs} +115 -136
  177. package/dist/LayoutContext-B45-e9DI.cjs +93 -0
  178. package/dist/LayoutContext-BAql6ZRY.js +97 -0
  179. package/dist/LayoutContext-Bav3UMEA.js +94 -0
  180. package/dist/LayoutContext-BvK-ggDa.cjs +96 -0
  181. package/dist/{ThemeContext-Bo-W2WZH.js → ThemeContext-BWq9ACPo.js} +8 -13
  182. package/dist/{ThemeContext-ept8jhXI.js → ThemeContext-BXjrgUjW.js} +261 -200
  183. package/dist/{ThemeContext-BblcjQup.cjs → ThemeContext-Bmod0Cg2.cjs} +8 -13
  184. package/dist/ThemeContext-BoH4NLfN.js +734 -0
  185. package/dist/{ThemeContext-BbBNoFTG.js → ThemeContext-C2EwAPDt.js} +2 -2
  186. package/dist/{ThemeContext-U4dEYc6C.cjs → ThemeContext-CGk3KK0k.cjs} +1 -8
  187. package/dist/{ThemeContext-D3LzacmG.js → ThemeContext-CQSo4Iwc.js} +1 -8
  188. package/dist/{ThemeContext-CP3a0jxy.cjs → ThemeContext-j5aGtPky.cjs} +262 -193
  189. package/dist/ThemeContext-r69W20Xg.cjs +733 -0
  190. package/dist/{ThemeContext-Cmr8Ex8H.cjs → ThemeContext-vTjumZeM.cjs} +2 -2
  191. package/dist/{VerifyEmailPage-BRSP-Pwt.cjs → VerifyEmailPage--1Vurewl.cjs} +3 -3
  192. package/dist/{VerifyEmailPage-CbgjOF0v.js → VerifyEmailPage-1WwWczAn.js} +12 -22
  193. package/dist/{VerifyEmailPage-DF2ilhum.cjs → VerifyEmailPage-B4peJjAT.cjs} +356 -334
  194. package/dist/{VerifyEmailPage-CR7kb5df.cjs → VerifyEmailPage-BComraR7.cjs} +12 -22
  195. package/dist/{VerifyEmailPage-u_Dn7t1U.cjs → VerifyEmailPage-Bp1XXl3H.cjs} +4 -4
  196. package/dist/{VerifyEmailPage-CkBYfsNy.cjs → VerifyEmailPage-By3Jf__L.cjs} +4 -4
  197. package/dist/{VerifyEmailPage-Bv8Ah_TK.cjs → VerifyEmailPage-ByerOcm4.cjs} +20 -23
  198. package/dist/{VerifyEmailPage-BE-L9mB7.js → VerifyEmailPage-C0c2e5n0.js} +7 -7
  199. package/dist/{VerifyEmailPage-EhudUdqF.js → VerifyEmailPage-C5TNQTBa.js} +355 -343
  200. package/dist/{VerifyEmailPage-Dt7zgA4w.cjs → VerifyEmailPage-CFLMls1p.cjs} +4 -4
  201. package/dist/{VerifyEmailPage-Cyl55sJb.js → VerifyEmailPage-CJLz3jrn.js} +20 -23
  202. package/dist/VerifyEmailPage-COiyNl1y.js +2825 -0
  203. package/dist/{VerifyEmailPage-DMBh4NM9.cjs → VerifyEmailPage-CYXtbKi3.cjs} +1 -1
  204. package/dist/{VerifyEmailPage-DTtFfC-J.js → VerifyEmailPage-CgMxRb4z.js} +3 -3
  205. package/dist/VerifyEmailPage-CqKsR2v8.js +2827 -0
  206. package/dist/{VerifyEmailPage-Bae2cBXT.cjs → VerifyEmailPage-Cwi3kbol.cjs} +7 -7
  207. package/dist/{VerifyEmailPage-X14vhdyl.js → VerifyEmailPage-DGhuIqkb.js} +4 -4
  208. package/dist/{VerifyEmailPage-BIBOKV7Z.js → VerifyEmailPage-DSBMRHtl.js} +36 -41
  209. package/dist/{VerifyEmailPage-D-FRj5TU.cjs → VerifyEmailPage-De6bQjrz.cjs} +36 -41
  210. package/dist/{VerifyEmailPage-BJjAMUTW.js → VerifyEmailPage-DgIid028.js} +4 -4
  211. package/dist/VerifyEmailPage-DjQKRlUS.cjs +2824 -0
  212. package/dist/{VerifyEmailPage-CdYPSJoO.js → VerifyEmailPage-DvMLZgFt.js} +1 -1
  213. package/dist/{VerifyEmailPage-C_ihbcth.js → VerifyEmailPage-MTD7AG1Z.js} +4 -4
  214. package/dist/VerifyEmailPage-s-1X3LDJ.cjs +2826 -0
  215. package/dist/XerticaOrbe-KL1RBHzw.cjs +1354 -0
  216. package/dist/XerticaOrbe-zwS1p2a8.js +1355 -0
  217. package/dist/XerticaProvider-6btlAlzc.js +17 -0
  218. package/dist/{XerticaProvider-siSt9uG2.js → XerticaProvider-B7EVH-NF.js} +2 -2
  219. package/dist/{XerticaProvider-AbWlr7Af.cjs → XerticaProvider-BIrqfZ-i.cjs} +11 -8
  220. package/dist/XerticaProvider-BNoNOxQ5.cjs +16 -0
  221. package/dist/XerticaProvider-BlY2limY.cjs +38 -0
  222. package/dist/{XerticaProvider-CWgby5mY.js → XerticaProvider-C1DKnvLh.js} +4 -4
  223. package/dist/{XerticaProvider-AChwphCO.cjs → XerticaProvider-CBGc4EMA.cjs} +4 -4
  224. package/dist/{XerticaProvider-BITjgC5p.js → XerticaProvider-CEoWMTxu.js} +2 -2
  225. package/dist/{XerticaProvider-By8q3Roe.cjs → XerticaProvider-CllrbMEJ.cjs} +2 -2
  226. package/dist/{XerticaProvider-B8CaV7xu.cjs → XerticaProvider-D-yNhF94.cjs} +1 -1
  227. package/dist/XerticaProvider-DDuiIcKo.js +39 -0
  228. package/dist/{XerticaProvider-DQtvJU7m.js → XerticaProvider-DYq4JWtg.js} +1 -1
  229. package/dist/{XerticaProvider-CWs6EwNa.js → XerticaProvider-Dt5HEzbQ.js} +10 -10
  230. package/dist/{XerticaProvider-CW9hpCdF.cjs → XerticaProvider-ET0ihewn.cjs} +2 -2
  231. package/dist/XerticaProvider-cI9hSs27.cjs +38 -0
  232. package/dist/XerticaProvider-hSwhNQex.js +39 -0
  233. package/dist/{XerticaXLogo-ChryA6xj.js → XerticaXLogo-B7xQ5dhi.js} +1 -1
  234. package/dist/{XerticaXLogo-CziKMQil.cjs → XerticaXLogo-CQUUjXoH.cjs} +8 -8
  235. package/dist/{XerticaXLogo-DfUvz-lD.js → XerticaXLogo-Cmsp-Eey.js} +9 -9
  236. package/dist/{XerticaXLogo-CFuIlYFH.js → XerticaXLogo-DZbo4vOE.js} +12 -12
  237. package/dist/{XerticaXLogo-8TTzBjHw.cjs → XerticaXLogo-Zw2B276b.cjs} +1 -1
  238. package/dist/{XerticaXLogo-kslQ8Tk_.cjs → XerticaXLogo-bvZSgwGF.cjs} +13 -7
  239. package/dist/alert-dialog-BOje--vD.js +847 -0
  240. package/dist/alert-dialog-BtEuQqrg.cjs +870 -0
  241. package/dist/{alert-dialog-yckpaOpy.cjs → alert-dialog-DSKByiKZ.cjs} +3 -3
  242. package/dist/{alert-dialog-iDe5VE5o.js → alert-dialog-s-vmNkJ_.js} +3 -3
  243. package/dist/breadcrumb-CqJ7bHY5.js +161 -0
  244. package/dist/breadcrumb-m9Hb2_XN.cjs +177 -0
  245. package/dist/components/assistant/xertica-assistant/hooks/index.d.ts +6 -0
  246. package/dist/components/assistant/xertica-assistant/hooks/use-assistant-conversations.d.ts +21 -0
  247. package/dist/components/assistant/xertica-assistant/hooks/use-assistant-messages.d.ts +49 -0
  248. package/dist/components/assistant/xertica-assistant/hooks/use-assistant-suggestions.d.ts +16 -0
  249. package/dist/components/blocks/audio-player/AudioPlayer.d.ts +35 -0
  250. package/dist/components/blocks/audio-player/index.d.ts +1 -0
  251. package/dist/components/blocks/document-editor/DocumentEditor.d.ts +26 -0
  252. package/dist/components/blocks/document-editor/index.d.ts +1 -0
  253. package/dist/components/blocks/podcast-player/PodcastPlayer.d.ts +41 -0
  254. package/dist/components/blocks/podcast-player/index.d.ts +1 -0
  255. package/dist/components/ui/chart/parts/chart-dashboard.d.ts +113 -0
  256. package/dist/components/ui/chart/parts/chart-metric.d.ts +118 -0
  257. package/dist/components/ui/chart/parts/chart-primitives.d.ts +101 -0
  258. package/dist/components/ui/chart/parts/chart-shared.d.ts +20 -0
  259. package/dist/components/ui/chart/parts/chart-utils.d.ts +12 -0
  260. package/dist/components/ui/chart/parts/index.d.ts +5 -0
  261. package/dist/components/ui/stats-card/stats-card.d.ts +10 -0
  262. package/dist/dropdown-menu-BDB5CmQs.cjs +247 -0
  263. package/dist/dropdown-menu-DQidbKBD.js +231 -0
  264. package/dist/google-maps-loader-BFWp6VPd.js +287 -0
  265. package/dist/google-maps-loader-BKcdgFbu.cjs +312 -0
  266. package/dist/{google-maps-loader-t2IlYBzw.js → google-maps-loader-CTYySAun.js} +4 -0
  267. package/dist/google-maps-loader-CumCNXeG.js +312 -0
  268. package/dist/{google-maps-loader-BqsYL48U.cjs → google-maps-loader-Y-QkD-Li.cjs} +5 -0
  269. package/dist/google-maps-loader-eS3uQ5TA.cjs +287 -0
  270. package/dist/header-Cgy6vYPk.cjs +731 -0
  271. package/dist/header-DRlT4jgI.js +715 -0
  272. package/dist/header-Dux00SI4.cjs +731 -0
  273. package/dist/header-EkGKXPsD.js +715 -0
  274. package/dist/header-WfEywpyc.cjs +731 -0
  275. package/dist/header-tifNQn2U.js +715 -0
  276. package/dist/index-BhapVLVj.js +8 -0
  277. package/dist/{index-D3RLKRAs.cjs → index-COtD8bRW.cjs} +1 -1
  278. package/dist/index-D6fxYEY8.cjs +7 -0
  279. package/dist/index-DAIp0_HK.js +8 -0
  280. package/dist/index-DW5tYe26.js +8 -0
  281. package/dist/index-GA__GvnG.cjs +7 -0
  282. package/dist/index.cjs.js +2 -2
  283. package/dist/index.es.js +2 -2
  284. package/dist/index.umd.js +1043 -470
  285. package/dist/input-2R4loU86.js +127 -0
  286. package/dist/input-DWANSKGb.cjs +145 -0
  287. package/dist/pages.cjs.js +1 -1
  288. package/dist/pages.es.js +1 -1
  289. package/dist/progress-DPtzoVV8.js +175 -0
  290. package/dist/progress-EeaoqqUs.cjs +191 -0
  291. package/dist/rich-text-editor-0mraWT5y.cjs +2376 -0
  292. package/dist/rich-text-editor-B-IkcPD0.js +2874 -0
  293. package/dist/rich-text-editor-B6jMRLzk.cjs +1939 -0
  294. package/dist/rich-text-editor-B8_oYcIR.js +1730 -0
  295. package/dist/rich-text-editor-B9UbSXNb.js +1203 -0
  296. package/dist/rich-text-editor-BYuRBNBU.js +2373 -0
  297. package/dist/rich-text-editor-Bb9pySTs.cjs +2374 -0
  298. package/dist/rich-text-editor-BcL6L3cm.cjs +2374 -0
  299. package/dist/rich-text-editor-BoVZYtTs.cjs +2391 -0
  300. package/dist/rich-text-editor-Bp3zQqMC.js +2954 -0
  301. package/dist/rich-text-editor-CMgSN_w2.js +1189 -0
  302. package/dist/rich-text-editor-CPV1lEPH.cjs +1748 -0
  303. package/dist/rich-text-editor-CeucBdIv.cjs +2971 -0
  304. package/dist/rich-text-editor-CoKqbCtu.cjs +1799 -0
  305. package/dist/rich-text-editor-Cw56T_mB.js +2356 -0
  306. package/dist/rich-text-editor-Cyt8qs2b.js +1921 -0
  307. package/dist/rich-text-editor-D6H84OcX.cjs +1220 -0
  308. package/dist/rich-text-editor-D76gD-QI.js +2328 -0
  309. package/dist/rich-text-editor-DKkokOnA.js +1781 -0
  310. package/dist/rich-text-editor-DNsdpN64.cjs +2359 -0
  311. package/dist/rich-text-editor-DfG8bCyY.js +2358 -0
  312. package/dist/rich-text-editor-Dxjw31Z4.js +2341 -0
  313. package/dist/rich-text-editor-DzP0Epmb.js +2356 -0
  314. package/dist/rich-text-editor-bRkNoeZY.cjs +2891 -0
  315. package/dist/rich-text-editor-lyYE2ZG5.cjs +1207 -0
  316. package/dist/rich-text-editor-skplNlBM.cjs +2345 -0
  317. package/dist/select-Bkbr0f-Z.cjs +162 -0
  318. package/dist/select-CvIVdX2n.js +145 -0
  319. package/dist/{sidebar-CA6_ek3f.js → sidebar-B6SlKZYN.js} +40 -49
  320. package/dist/{sidebar-CmvwjnVb.js → sidebar-BViy8Eeu.js} +17 -9
  321. package/dist/{sidebar-Dz7bd3zP.js → sidebar-BbVIQvlP.js} +1 -1
  322. package/dist/{sidebar-CVUGHOS_.cjs → sidebar-BxGXsDAd.cjs} +16 -8
  323. package/dist/sidebar-CK_0ZQHj.cjs +803 -0
  324. package/dist/sidebar-CUuOvYhK.js +787 -0
  325. package/dist/{sidebar-KIS0C2JH.js → sidebar-CrQDDdcz.js} +24 -33
  326. package/dist/{sidebar-zowjejT2.cjs → sidebar-DAaY8bRU.cjs} +24 -33
  327. package/dist/{sidebar-B3EYhli0.cjs → sidebar-DQj1z3jG.cjs} +227 -269
  328. package/dist/{sidebar-B9NR0lCe.cjs → sidebar-Djn5syhi.cjs} +295 -309
  329. package/dist/{sidebar-CplprZpM.js → sidebar-LluMXfam.js} +227 -269
  330. package/dist/sidebar-_rT7rBMk.js +787 -0
  331. package/dist/{sidebar-BvF5I2Ue.cjs → sidebar-nzPoVHBQ.cjs} +41 -46
  332. package/dist/{sidebar-C5B_LHek.cjs → sidebar-q7P2Godd.cjs} +1 -1
  333. package/dist/slider-Bc5Hd0y1.js +56 -0
  334. package/dist/slider-N7hFFj6X.cjs +73 -0
  335. package/dist/tooltip-Ded96neP.cjs +137 -0
  336. package/dist/tooltip-HDOoD2-0.js +120 -0
  337. package/dist/ui.cjs.js +1 -1
  338. package/dist/ui.es.js +1 -1
  339. package/dist/use-audio-player-B31J-aqh.cjs +187 -0
  340. package/dist/use-audio-player-BkmEmj8Q.js +185 -0
  341. package/dist/use-audio-player-CLFTWFW1.cjs +184 -0
  342. package/dist/use-audio-player-CLLn00I6.js +188 -0
  343. package/dist/{use-audio-player-Dn1NR9xN.cjs → use-audio-player-NKsWyjWu.cjs} +7 -3
  344. package/dist/{use-audio-player-Bkh23vQ3.js → use-audio-player-nv8ZSGa1.js} +7 -3
  345. package/dist/use-file-upload-BcjEo2S5.js +404 -0
  346. package/dist/use-file-upload-CRJR68Tj.cjs +403 -0
  347. package/dist/use-mobile-B0hNy_Y6.cjs +4303 -0
  348. package/dist/use-mobile-BXuYROXM.js +4202 -0
  349. package/dist/use-mobile-Bbd51ASU.cjs +4392 -0
  350. package/dist/use-mobile-Bk6CX-TC.js +4359 -0
  351. package/dist/use-mobile-BvYdisLP.js +4202 -0
  352. package/dist/use-mobile-BzuxjzNX.cjs +4392 -0
  353. package/dist/use-mobile-CG2-SdXV.cjs +4235 -0
  354. package/dist/use-mobile-CKb5pqTs.js +4269 -0
  355. package/dist/use-mobile-CYuAuGDl.js +4202 -0
  356. package/dist/use-mobile-CaENcqm-.js +4508 -0
  357. package/dist/use-mobile-CbrYgJGJ.js +4203 -0
  358. package/dist/use-mobile-Cd4xPrKq.cjs +46 -0
  359. package/dist/use-mobile-DMOvImGQ.cjs +4542 -0
  360. package/dist/use-mobile-DRB3BQgD.cjs +4235 -0
  361. package/dist/use-mobile-DZvv7QMR.js +4359 -0
  362. package/dist/use-mobile-DdI_TXam.cjs +4235 -0
  363. package/dist/use-mobile-DlceKf8a.js +4359 -0
  364. package/dist/use-mobile-DsOnow1o.cjs +4236 -0
  365. package/dist/use-mobile-Kcj6jSnK.cjs +4392 -0
  366. package/dist/use-mobile-bnKcua_i.js +4202 -0
  367. package/dist/use-mobile-j4w2Jrf1.js +30 -0
  368. package/dist/use-mobile-ncXBeE2z.cjs +4235 -0
  369. package/dist/use-rich-text-editor-DjiddBGv.js +282 -0
  370. package/dist/use-rich-text-editor-lpeswbCs.cjs +281 -0
  371. package/dist/xertica-assistant-BdiZag0h.js +2187 -0
  372. package/dist/xertica-assistant-CrgTb6Hs.cjs +2155 -0
  373. package/dist/xertica-assistant-DCsnQyi5.js +2156 -0
  374. package/dist/xertica-assistant-DUBpmEgo.cjs +2186 -0
  375. package/dist/{xertica-assistant-Bj3vBCq_.cjs → xertica-assistant-V_IdW4WF.cjs} +27 -9
  376. package/dist/{xertica-assistant-BMqdyRVi.js → xertica-assistant-ciJaWqm1.js} +28 -10
  377. package/dist/{xertica-assistant-B1IaHXnB.cjs → xertica-assistant-dyP7KHM5.cjs} +533 -392
  378. package/dist/{xertica-assistant-DPsESB6t.js → xertica-assistant-yX1CFBBo.js} +535 -394
  379. package/dist/xertica-ui.css +2 -2
  380. package/docs/architecture-improvements.md +456 -456
  381. package/docs/architecture.md +312 -312
  382. package/docs/components/assistant.md +428 -428
  383. package/docs/components/branding.md +252 -252
  384. package/docs/components/card-patterns.md +447 -447
  385. package/docs/components/error-boundary.md +201 -201
  386. package/docs/components/hooks.md +432 -432
  387. package/docs/components/language-selector.md +176 -176
  388. package/docs/components/pages.md +323 -323
  389. package/docs/components/stats-card.md +20 -2
  390. package/docs/doc-audit.md +244 -244
  391. package/docs/getting-started.md +616 -616
  392. package/docs/guidelines.md +330 -330
  393. package/docs/i18n.md +480 -480
  394. package/docs/installation.md +268 -268
  395. package/docs/llms.md +295 -295
  396. package/docs/state-management.md +289 -289
  397. package/guidelines/Guidelines.md +409 -409
  398. package/llms-compact.txt +1 -1
  399. package/llms-full.txt +11553 -7133
  400. package/llms.txt +1 -1
  401. package/package.json +219 -219
  402. package/styles/xertica/base.css +90 -90
  403. package/templates/.prettierignore +4 -4
  404. package/templates/.prettierrc +10 -10
  405. package/templates/CLAUDE.md +180 -180
  406. package/templates/guidelines/Guidelines.md +577 -577
  407. package/templates/package.json +69 -69
  408. package/templates/src/app/App.tsx +46 -46
  409. package/templates/src/app/components/AuthGuard.tsx +131 -131
  410. package/templates/src/features/assistant/data/mock.ts +75 -75
  411. package/templates/src/features/assistant/hooks/useAssistantConfig.ts +20 -20
  412. package/templates/src/features/assistant/index.ts +5 -5
  413. package/templates/src/features/auth/ui/AuthPageShell.tsx +32 -32
  414. package/templates/src/features/auth/ui/ForgotPasswordContent.tsx +70 -70
  415. package/templates/src/features/auth/ui/LoginContent.tsx +92 -92
  416. package/templates/src/features/auth/ui/ResetPasswordContent.tsx +183 -183
  417. package/templates/src/features/auth/ui/SocialLoginButtons.tsx +78 -78
  418. package/templates/src/features/auth/ui/VerifyEmailContent.tsx +80 -80
  419. package/templates/src/features/home/data/mock.ts +41 -41
  420. package/templates/src/features/home/hooks/useFeatureCards.ts +20 -20
  421. package/templates/src/features/home/index.ts +11 -11
  422. package/templates/src/features/home/ui/HomeContent.tsx +117 -117
  423. package/templates/src/features/template/ui/CrudTemplate.tsx +112 -112
  424. package/templates/src/features/template/ui/DashboardTemplate.tsx +110 -110
  425. package/templates/src/features/template/ui/FormTemplate.tsx +117 -117
  426. package/templates/src/features/template/ui/LoginTemplate.tsx +59 -59
  427. package/templates/src/features/template/ui/TemplateContent.tsx +1322 -1322
  428. package/templates/src/i18n.ts +124 -124
  429. package/templates/src/locales/en/common.json +21 -21
  430. package/templates/src/locales/en/components/activityCard.json +10 -10
  431. package/templates/src/locales/en/components/assistant.json +119 -119
  432. package/templates/src/locales/en/components/media.json +29 -29
  433. package/templates/src/locales/en/components/notificationCard.json +5 -5
  434. package/templates/src/locales/en/components/profileCard.json +8 -8
  435. package/templates/src/locales/en/components/projectCard.json +10 -10
  436. package/templates/src/locales/en/components/sidebar.json +14 -14
  437. package/templates/src/locales/en/components/stats.json +8 -8
  438. package/templates/src/locales/en/components/team.json +14 -14
  439. package/templates/src/locales/en/errors.json +9 -9
  440. package/templates/src/locales/en/languageSelector.json +7 -7
  441. package/templates/src/locales/en/nav.json +6 -6
  442. package/templates/src/locales/en/pages/crudTemplate.json +25 -25
  443. package/templates/src/locales/en/pages/dashboardTemplate.json +20 -20
  444. package/templates/src/locales/en/pages/forgotPassword.json +10 -10
  445. package/templates/src/locales/en/pages/formTemplate.json +16 -16
  446. package/templates/src/locales/en/pages/home.json +7 -7
  447. package/templates/src/locales/en/pages/login.json +15 -15
  448. package/templates/src/locales/en/pages/loginTemplate.json +9 -9
  449. package/templates/src/locales/en/pages/resetPassword.json +18 -18
  450. package/templates/src/locales/en/pages/templates.json +317 -317
  451. package/templates/src/locales/en/pages/verifyEmail.json +12 -12
  452. package/templates/src/locales/en/themeToggle.json +6 -6
  453. package/templates/src/locales/es/common.json +21 -21
  454. package/templates/src/locales/es/components/activityCard.json +10 -10
  455. package/templates/src/locales/es/components/assistant.json +119 -119
  456. package/templates/src/locales/es/components/media.json +29 -29
  457. package/templates/src/locales/es/components/notificationCard.json +5 -5
  458. package/templates/src/locales/es/components/profileCard.json +8 -8
  459. package/templates/src/locales/es/components/projectCard.json +10 -10
  460. package/templates/src/locales/es/components/sidebar.json +14 -14
  461. package/templates/src/locales/es/components/stats.json +8 -8
  462. package/templates/src/locales/es/components/team.json +14 -14
  463. package/templates/src/locales/es/errors.json +9 -9
  464. package/templates/src/locales/es/languageSelector.json +7 -7
  465. package/templates/src/locales/es/nav.json +6 -6
  466. package/templates/src/locales/es/pages/crudTemplate.json +25 -25
  467. package/templates/src/locales/es/pages/dashboardTemplate.json +20 -20
  468. package/templates/src/locales/es/pages/forgotPassword.json +10 -10
  469. package/templates/src/locales/es/pages/formTemplate.json +16 -16
  470. package/templates/src/locales/es/pages/home.json +7 -7
  471. package/templates/src/locales/es/pages/login.json +15 -15
  472. package/templates/src/locales/es/pages/loginTemplate.json +9 -9
  473. package/templates/src/locales/es/pages/resetPassword.json +18 -18
  474. package/templates/src/locales/es/pages/templates.json +317 -317
  475. package/templates/src/locales/es/pages/verifyEmail.json +12 -12
  476. package/templates/src/locales/es/themeToggle.json +6 -6
  477. package/templates/src/locales/pt-BR/common.json +21 -21
  478. package/templates/src/locales/pt-BR/components/activityCard.json +10 -10
  479. package/templates/src/locales/pt-BR/components/assistant.json +119 -119
  480. package/templates/src/locales/pt-BR/components/media.json +29 -29
  481. package/templates/src/locales/pt-BR/components/notificationCard.json +5 -5
  482. package/templates/src/locales/pt-BR/components/profileCard.json +8 -8
  483. package/templates/src/locales/pt-BR/components/projectCard.json +10 -10
  484. package/templates/src/locales/pt-BR/components/sidebar.json +14 -14
  485. package/templates/src/locales/pt-BR/components/stats.json +8 -8
  486. package/templates/src/locales/pt-BR/components/team.json +14 -14
  487. package/templates/src/locales/pt-BR/errors.json +9 -9
  488. package/templates/src/locales/pt-BR/languageSelector.json +7 -7
  489. package/templates/src/locales/pt-BR/nav.json +6 -6
  490. package/templates/src/locales/pt-BR/pages/crudTemplate.json +25 -25
  491. package/templates/src/locales/pt-BR/pages/dashboardTemplate.json +20 -20
  492. package/templates/src/locales/pt-BR/pages/forgotPassword.json +10 -10
  493. package/templates/src/locales/pt-BR/pages/formTemplate.json +16 -16
  494. package/templates/src/locales/pt-BR/pages/home.json +7 -7
  495. package/templates/src/locales/pt-BR/pages/login.json +15 -15
  496. package/templates/src/locales/pt-BR/pages/loginTemplate.json +9 -9
  497. package/templates/src/locales/pt-BR/pages/resetPassword.json +18 -18
  498. package/templates/src/locales/pt-BR/pages/templates.json +317 -317
  499. package/templates/src/locales/pt-BR/pages/verifyEmail.json +12 -12
  500. package/templates/src/locales/pt-BR/themeToggle.json +6 -6
  501. package/templates/src/pages/AssistantPage.tsx +470 -470
  502. package/templates/src/pages/HomePage.tsx +53 -53
  503. package/templates/src/shared/error-boundary.tsx +150 -150
  504. package/templates/src/shared/error-fallbacks.tsx +222 -222
  505. package/templates/vite.config.js +20 -20
  506. package/templates/vite.config.ts +55 -55
  507. package/dist/ThemeContext-CpqYShLq.cjs +0 -324
  508. package/dist/ThemeContext-Du2nE1PL.js +0 -325
  509. package/dist/ThemeContext-GeEBTJ3q.cjs +0 -1621
  510. package/dist/ThemeContext-JyLK9B1o.js +0 -1622
  511. package/dist/VerifyEmailPage-BiRm7Nh4.cjs +0 -3213
  512. package/dist/VerifyEmailPage-Bvfv8HVQ.js +0 -3214
  513. package/dist/VerifyEmailPage-hdB8JQGv.cjs +0 -3213
  514. package/dist/VerifyEmailPage-vYHbYK3q.js +0 -3214
  515. package/dist/XerticaProvider-CUYJZc32.js +0 -49
  516. package/dist/XerticaProvider-CjQAQPcn.cjs +0 -48
  517. package/dist/XerticaProvider-D5lLumH-.js +0 -49
  518. package/dist/XerticaProvider-qQUDop71.cjs +0 -48
  519. package/dist/XerticaXLogo-BWaag64t.js +0 -252
  520. package/dist/XerticaXLogo-CU-U-GP4.cjs +0 -251
  521. package/dist/index-CkTUgOwX.js +0 -8
  522. package/dist/sidebar-OTO_up7Z.js +0 -801
  523. package/dist/{rich-text-editor-BmsjY03B.js → rich-text-editor-DgF8s7xW.js} +26 -26
  524. package/dist/{rich-text-editor-GS2kpTAK.cjs → rich-text-editor-mWoaSCE4.cjs} +26 -26
package/bin/cli.ts CHANGED
@@ -1,1244 +1,1244 @@
1
- #!/usr/bin/env node
2
- import { Command } from 'commander';
3
- import prompts from 'prompts';
4
- import chalk from 'chalk';
5
- import ora from 'ora';
6
- import fs from 'fs-extra';
7
- import path from 'path';
8
- import { fileURLToPath } from 'url';
9
- import { execa } from 'execa';
10
- import { colorThemes } from '../contexts/theme-data';
11
- import { generateTokensCss } from './generate-tokens';
12
- import {
13
- SUPPORTED_LANGUAGES,
14
- DEFAULT_SELECTION,
15
- readLanguagesConfig,
16
- writeLanguagesConfig,
17
- syncLocaleFiles,
18
- generateI18nFile,
19
- generateAppTsx,
20
- } from './language-config';
21
-
22
- // ─────────────────────────────────────────────────────────────────────────────
23
- // Project config helpers (.xertica.json)
24
- // Persists per-project feature flags so `update` can read them later.
25
- // ─────────────────────────────────────────────────────────────────────────────
26
-
27
- const XERTICA_CONFIG_FILE = '.xertica.json';
28
-
29
- interface XerticaConfig {
30
- version: 1;
31
- hasAssistant: boolean;
32
- disableDarkMode?: boolean;
33
- }
34
-
35
- async function readXerticaConfig(targetDir: string): Promise<XerticaConfig | null> {
36
- const configPath = path.join(targetDir, XERTICA_CONFIG_FILE);
37
- if (!(await fs.pathExists(configPath))) return null;
38
- try {
39
- return (await fs.readJson(configPath)) as XerticaConfig;
40
- } catch {
41
- return null;
42
- }
43
- }
44
-
45
- async function writeXerticaConfig(
46
- targetDir: string,
47
- config: Partial<XerticaConfig>
48
- ): Promise<void> {
49
- const configPath = path.join(targetDir, XERTICA_CONFIG_FILE);
50
- const existing = (await readXerticaConfig(targetDir)) ?? {
51
- version: 1 as const,
52
- hasAssistant: false,
53
- };
54
- await fs.writeJson(configPath, { ...existing, ...config, version: 1 }, { spaces: 2 });
55
- }
56
-
57
- const __filename = fileURLToPath(import.meta.url);
58
- const __dirname = path.dirname(__filename);
59
-
60
- // Read CLI version from package.json so it always matches the published one,
61
- // instead of hardcoding a literal that drifts on every version bump.
62
- const pkgJson = JSON.parse(
63
- fs.readFileSync(path.resolve(__dirname, '../package.json'), 'utf-8')
64
- ) as { version: string };
65
-
66
- // ─────────────────────────────────────────────────────────────────────────────
67
- // AuthGuard generator
68
- // Generates src/app/components/AuthGuard.tsx based on the selected features.
69
- // ─────────────────────────────────────────────────────────────────────────────
70
-
71
- function generateAuthGuard({
72
- hasLogin,
73
- hasHome,
74
- hasTemplate,
75
- hasAssistant,
76
- firstProtectedPath,
77
- }: {
78
- hasLogin: boolean;
79
- hasHome: boolean;
80
- hasTemplate: boolean;
81
- hasAssistant: boolean;
82
- firstProtectedPath: string;
83
- }): string {
84
- return `import React from 'react';
85
- import { Routes, Route, Navigate } from 'react-router-dom';
86
- import { useAuth } from '../context/AuthContext';
87
-
88
- // ─── Lazy page imports ────────────────────────────────────────────────────────
89
-
90
- ${
91
- hasLogin
92
- ? `const LoginPage = React.lazy(() => import('../../pages/LoginPage').then(m => ({ default: m.LoginPage })));
93
- const ForgotPasswordPage = React.lazy(() => import('../../pages/ForgotPasswordPage').then(m => ({ default: m.ForgotPasswordPage })));
94
- const VerifyEmailPage = React.lazy(() => import('../../pages/VerifyEmailPage').then(m => ({ default: m.VerifyEmailPage })));
95
- const ResetPasswordPage = React.lazy(() => import('../../pages/ResetPasswordPage').then(m => ({ default: m.ResetPasswordPage })));`
96
- : ''
97
- }
98
-
99
- ${hasHome ? `const HomePage = React.lazy(() => import('../../pages/HomePage').then(m => ({ default: m.HomePage })));` : ''}
100
- ${hasTemplate ? `const TemplatePage = React.lazy(() => import('../../pages/TemplatePage').then(m => ({ default: m.TemplatePage })));` : ''}
101
- ${hasAssistant ? `const AssistantPage = React.lazy(() => import('../../pages/AssistantPage').then(m => ({ default: m.AssistantPage })));` : ''}
102
-
103
- // ─── Route guards ─────────────────────────────────────────────────────────────
104
-
105
- function ProtectedRoute({ children }: { children: React.ReactNode }) {
106
- const { user, isLoading } = useAuth();
107
- if (isLoading) return null;
108
- if (!user) return <Navigate to="${hasLogin ? '/login' : firstProtectedPath}" replace />;
109
- return <>{children}</>;
110
- }
111
-
112
- ${
113
- hasLogin
114
- ? `function GuestRoute({ children }: { children: React.ReactNode }) {
115
- const { user, isLoading } = useAuth();
116
- if (isLoading) return null;
117
- if (user) return <Navigate to="${firstProtectedPath}" replace />;
118
- return <>{children}</>;
119
- }
120
-
121
- function LoginPageWithAuth() {
122
- const { login } = useAuth();
123
- return <LoginPage onLogin={login} />;
124
- }`
125
- : ''
126
- }
127
-
128
- // ─── Route tree ───────────────────────────────────────────────────────────────
129
-
130
- export function AuthGuard() {
131
- const { user } = useAuth();
132
-
133
- return (
134
- <div className="min-h-screen bg-muted overflow-x-hidden max-w-full">
135
- <Routes>
136
- ${
137
- hasLogin
138
- ? ` <Route path="/login" element={<GuestRoute><LoginPageWithAuth /></GuestRoute>} />
139
- <Route path="/forgot-password" element={<GuestRoute><ForgotPasswordPage /></GuestRoute>} />
140
- <Route path="/verify-email" element={<GuestRoute><VerifyEmailPage /></GuestRoute>} />
141
- <Route path="/reset-password" element={<GuestRoute><ResetPasswordPage /></GuestRoute>} />`
142
- : ''
143
- }
144
-
145
- ${hasHome ? ` <Route path="/home" element={<ProtectedRoute><HomePage /></ProtectedRoute>} />` : ''}
146
- ${hasTemplate ? ` <Route path="/template" element={<ProtectedRoute><TemplatePage /></ProtectedRoute>} />` : ''}
147
- ${hasAssistant ? ` <Route path="/assistente" element={<ProtectedRoute><AssistantPage /></ProtectedRoute>} />` : ''}
148
-
149
- <Route path="/" element={<Navigate to={user ? '${firstProtectedPath}' : '${hasLogin ? '/login' : firstProtectedPath}'} replace />} />
150
- <Route path="*" element={<Navigate to={user ? '${firstProtectedPath}' : '${hasLogin ? '/login' : firstProtectedPath}'} replace />} />
151
- </Routes>
152
- </div>
153
- );
154
- }
155
- `;
156
- }
157
-
158
- // ─────────────────────────────────────────────────────────────────────────────
159
- // Page generators
160
- // Generate page files that vary based on whether the assistant is included.
161
- // ─────────────────────────────────────────────────────────────────────────────
162
-
163
- function generateHomePage(hasAssistant: boolean): string {
164
- if (hasAssistant) {
165
- return `import React from 'react';
166
- import { XerticaAssistant, generateDemoResponse } from 'xertica-ui/assistant';
167
- import { useLayout } from 'xertica-ui/hooks';
168
- import { useNavigate } from 'react-router-dom';
169
- import { AppLayout } from '../app/components/AppLayout';
170
- import { HomeContent } from '../features/home';
171
- import { useAssistantConfig, getMockRichSuggestions, getMockFeedbackOptions } from '../features/assistant';
172
-
173
- /**
174
- * Home page — thin layout shell.
175
- *
176
- * Assistant config (suggestions, feedback options) is fetched via React Query.
177
- * To connect to a real API replace \`fetchAssistantConfig\` in
178
- * \`features/assistant/data/mock.ts\`.
179
- */
180
- export function HomePage() {
181
- const { assistenteExpanded, toggleAssistente } = useLayout();
182
- const navigate = useNavigate();
183
-
184
- const { data: assistantConfig } = useAssistantConfig();
185
-
186
- return (
187
- <AppLayout
188
- assistant={
189
- <XerticaAssistant
190
- isExpanded={assistenteExpanded}
191
- onToggle={toggleAssistente}
192
- defaultTab="chat"
193
- demoMode={true}
194
- userName="Ariel Santos"
195
- responseGenerator={generateDemoResponse}
196
- suggestions={assistantConfig?.suggestions}
197
- richSuggestions={assistantConfig?.richSuggestions ?? getMockRichSuggestions()}
198
- feedbackOptions={assistantConfig?.feedbackOptions ?? getMockFeedbackOptions()}
199
- showHistory={false}
200
- showFavorites={false}
201
- onNavigateSettings={() => navigate('/settings')}
202
- onNavigateFullPage={() => navigate('/assistente')}
203
- onEvaluation={(messageId, type, reason) => {
204
- // Wire your feedback persistence logic here
205
- console.log(\`Avaliação: \${type} na mensagem \${messageId}. Motivo: \${reason}\`);
206
- }}
207
- />
208
- }
209
- >
210
- <HomeContent />
211
- </AppLayout>
212
- );
213
- }
214
- `;
215
- }
216
-
217
- return `import React from 'react';
218
- import { AppLayout } from '../app/components/AppLayout';
219
- import { HomeContent } from '../features/home';
220
-
221
- /**
222
- * Home page — thin layout shell.
223
- */
224
- export function HomePage() {
225
- return (
226
- <AppLayout>
227
- <HomeContent />
228
- </AppLayout>
229
- );
230
- }
231
- `;
232
- }
233
-
234
- function generateTemplatePage(hasAssistant: boolean): string {
235
- if (hasAssistant) {
236
- return `import React from 'react';
237
- import { XerticaAssistant } from 'xertica-ui/assistant';
238
- import { useLayout } from 'xertica-ui/hooks';
239
- import { AppLayout } from '../app/components/AppLayout';
240
- import { TemplateContent } from '../features/template';
241
-
242
- /**
243
- * Template page — thin layout shell.
244
- *
245
- * Auth state is consumed from \`AuthContext\` via \`AppLayout\` — no props needed.
246
- */
247
- export function TemplatePage() {
248
- const { assistenteExpanded, toggleAssistente } = useLayout();
249
-
250
- return (
251
- <AppLayout
252
- assistant={
253
- <XerticaAssistant
254
- isExpanded={assistenteExpanded}
255
- onToggle={toggleAssistente}
256
- onEvaluation={(id, type, reason) => console.log('Feedback:', id, type, reason)}
257
- />
258
- }
259
- >
260
- <TemplateContent />
261
- </AppLayout>
262
- );
263
- }
264
- `;
265
- }
266
-
267
- return `import React from 'react';
268
- import { AppLayout } from '../app/components/AppLayout';
269
- import { TemplateContent } from '../features/template';
270
-
271
- /**
272
- * Template page — thin layout shell.
273
- *
274
- * Auth state is consumed from \`AuthContext\` via \`AppLayout\` — no props needed.
275
- */
276
- export function TemplatePage() {
277
- return (
278
- <AppLayout>
279
- <TemplateContent />
280
- </AppLayout>
281
- );
282
- }
283
- `;
284
- }
285
-
286
- const program = new Command();
287
-
288
- program
289
- .name('xertica-ui')
290
- .description('CLI to initialize Xertica UI projects')
291
- .version(pkgJson.version);
292
-
293
- program
294
- .command('init')
295
- .description('Initialize a new Xertica UI project')
296
- .argument('[directory]', 'Directory to initialize in', '.')
297
- .action(async directory => {
298
- const targetDir = path.resolve(process.cwd(), directory);
299
- const templatesDir = path.resolve(__dirname, '../templates');
300
-
301
- console.log(chalk.blue(`🚀 Welcome to Xertica UI CLI! ${chalk.dim(`v${pkgJson.version}`)}`));
302
-
303
- const response = await prompts([
304
- {
305
- type: 'multiselect',
306
- name: 'pages',
307
- message: 'Which pages/templates to include?',
308
- choices: [
309
- {
310
- title: 'Login Page (+ Forgot / Verify / Reset Password)',
311
- value: 'login',
312
- selected: true,
313
- },
314
- { title: 'Home Page', value: 'home', selected: true },
315
- { title: 'Template Page (components showcase)', value: 'template', selected: true },
316
- ],
317
- },
318
- {
319
- type: 'multiselect',
320
- name: 'languages',
321
- message: 'Which languages should the app support?',
322
- instructions: false,
323
- hint: 'Select at least 1. Single-language apps auto-hide the LanguageSelector.',
324
- min: 1,
325
- choices: SUPPORTED_LANGUAGES.map(l => ({
326
- title: l.label,
327
- value: l.code,
328
- selected: true,
329
- })),
330
- },
331
- {
332
- type: 'select',
333
- name: 'theme',
334
- message: 'Select the default color theme for your project:',
335
- choices: colorThemes.map(t => ({
336
- title: t.name,
337
- description: t.description,
338
- value: t.id,
339
- })),
340
- initial: 0,
341
- },
342
- {
343
- type: 'confirm',
344
- name: 'hasAssistant',
345
- message: 'Include AI Assistant? (XerticaAssistant chat page + sidebar variant)',
346
- initial: true,
347
- },
348
- {
349
- type: 'confirm',
350
- name: 'enableDarkMode',
351
- message: 'Enable dark mode support?',
352
- initial: true,
353
- },
354
- {
355
- type: 'confirm',
356
- name: 'install',
357
- message: 'Install dependencies automatically?',
358
- initial: true,
359
- },
360
- ]);
361
-
362
- // Abort if the user cancelled any prompt (prompts returns undefined on Ctrl+C)
363
- if (!response.pages || !response.languages || !response.theme) return;
364
-
365
- const spinner = ora('Initializing project...').start();
366
-
367
- try {
368
- await fs.ensureDir(targetDir);
369
-
370
- const pages = response.pages || [];
371
- const hasLogin = pages.includes('login');
372
- const hasHome = pages.includes('home');
373
- const hasTemplate = pages.includes('template');
374
- const hasAssistant = response.hasAssistant ?? true;
375
-
376
- // Resolve selected languages — fall back to all defaults if the user
377
- // somehow ended up with an empty array (the prompt's min:1 should prevent
378
- // this, but we defend defensively).
379
- const selectedLanguages: string[] =
380
- Array.isArray(response.languages) && response.languages.length > 0
381
- ? response.languages
382
- : DEFAULT_SELECTION;
383
-
384
- // 1. Copy root config files
385
- const rootFilesToCopy = [
386
- 'index.html',
387
- 'vite.config.ts',
388
- 'tsconfig.json',
389
- 'tsconfig.node.json',
390
- 'postcss.config.js',
391
- 'vite-env.d.ts',
392
- 'eslint.config.js',
393
- '.env.example',
394
- 'guidelines',
395
- 'CLAUDE.md',
396
- ];
397
-
398
- for (const file of rootFilesToCopy) {
399
- const srcPath = path.join(templatesDir, file);
400
- if (await fs.pathExists(srcPath)) {
401
- await fs.copy(srcPath, path.join(targetDir, file));
402
- }
403
- }
404
-
405
- // 2. Copy package.json
406
- const pkgTemplatePath = path.join(templatesDir, 'package.json');
407
- if (await fs.pathExists(pkgTemplatePath)) {
408
- const pkgContent = await fs.readJson(pkgTemplatePath);
409
- const projectName = path.basename(targetDir) || 'my-xertica-app';
410
- pkgContent.name = projectName.toLowerCase().replace(/[^a-z0-9-]/g, '-');
411
- await fs.writeJson(path.join(targetDir, 'package.json'), pkgContent, { spaces: 2 });
412
- }
413
-
414
- // 3. Copy src/main.tsx
415
- await fs.copy(
416
- path.join(templatesDir, 'src', 'main.tsx'),
417
- path.join(targetDir, 'src', 'main.tsx')
418
- );
419
-
420
- const disableDarkMode = response.enableDarkMode === false;
421
-
422
- // 4. Generate src/app/App.tsx with the user's language selection
423
- // (instead of copying the static template, we inject `availableLanguages`)
424
- await fs.ensureDir(path.join(targetDir, 'src', 'app'));
425
- await fs.writeFile(
426
- path.join(targetDir, 'src', 'app', 'App.tsx'),
427
- generateAppTsx(selectedLanguages, disableDarkMode)
428
- );
429
-
430
- // 5. Copy src/app/components/AppLayout.tsx (always needed)
431
- await fs.ensureDir(path.join(targetDir, 'src', 'app', 'components'));
432
- await fs.copy(
433
- path.join(templatesDir, 'src', 'app', 'components', 'AppLayout.tsx'),
434
- path.join(targetDir, 'src', 'app', 'components', 'AppLayout.tsx')
435
- );
436
-
437
- // 6. Copy src/shared/ (always needed — auth helpers, navigation config, types)
438
- await fs.copy(
439
- path.join(templatesDir, 'src', 'shared'),
440
- path.join(targetDir, 'src', 'shared')
441
- );
442
-
443
- // 6.1 Generate i18n.ts with only the imports/resources for selected languages
444
- await fs.writeFile(
445
- path.join(targetDir, 'src', 'i18n.ts'),
446
- generateI18nFile(selectedLanguages)
447
- );
448
-
449
- // 6.2 Copy only the selected locale JSON files (no orphan locales)
450
- await syncLocaleFiles(templatesDir, targetDir, selectedLanguages, { pruneOthers: true });
451
-
452
- // 6.3 Persist the language selection so `update` can remember it
453
- await writeLanguagesConfig(targetDir, selectedLanguages);
454
-
455
- // 6.5 Persist project feature flags (.xertica.json)
456
- await writeXerticaConfig(targetDir, { hasAssistant, disableDarkMode });
457
-
458
- // 6.4 Copy context
459
- await fs.ensureDir(path.join(targetDir, 'src', 'app', 'context'));
460
- await fs.copy(
461
- path.join(templatesDir, 'src', 'app', 'context', 'AuthContext.tsx'),
462
- path.join(targetDir, 'src', 'app', 'context', 'AuthContext.tsx')
463
- );
464
-
465
- // 7. Copy features based on selections
466
- if (hasLogin) {
467
- await fs.copy(
468
- path.join(templatesDir, 'src', 'features', 'auth'),
469
- path.join(targetDir, 'src', 'features', 'auth')
470
- );
471
- }
472
- if (hasHome) {
473
- await fs.copy(
474
- path.join(templatesDir, 'src', 'features', 'home'),
475
- path.join(targetDir, 'src', 'features', 'home')
476
- );
477
- }
478
- if (hasTemplate) {
479
- await fs.copy(
480
- path.join(templatesDir, 'src', 'features', 'template'),
481
- path.join(targetDir, 'src', 'features', 'template')
482
- );
483
- }
484
- // Copy assistant feature only if selected
485
- if (hasAssistant) {
486
- await fs.copy(
487
- path.join(templatesDir, 'src', 'features', 'assistant'),
488
- path.join(targetDir, 'src', 'features', 'assistant')
489
- );
490
- }
491
-
492
- // 8. Copy pages based on selections
493
- await fs.ensureDir(path.join(targetDir, 'src', 'pages'));
494
-
495
- const pagesToCopy: string[] = [];
496
- if (hasLogin)
497
- pagesToCopy.push(
498
- 'LoginPage.tsx',
499
- 'ForgotPasswordPage.tsx',
500
- 'VerifyEmailPage.tsx',
501
- 'ResetPasswordPage.tsx'
502
- );
503
- if (hasHome) pagesToCopy.push('HomePage.tsx');
504
- if (hasTemplate) pagesToCopy.push('TemplatePage.tsx');
505
- if (hasAssistant) pagesToCopy.push('AssistantPage.tsx');
506
-
507
- for (const pageFile of pagesToCopy) {
508
- // HomePage and TemplatePage are generated (they vary by hasAssistant)
509
- if (pageFile === 'HomePage.tsx') {
510
- await fs.writeFile(
511
- path.join(targetDir, 'src', 'pages', 'HomePage.tsx'),
512
- generateHomePage(hasAssistant)
513
- );
514
- continue;
515
- }
516
- if (pageFile === 'TemplatePage.tsx') {
517
- await fs.writeFile(
518
- path.join(targetDir, 'src', 'pages', 'TemplatePage.tsx'),
519
- generateTemplatePage(hasAssistant)
520
- );
521
- continue;
522
- }
523
- const src = path.join(templatesDir, 'src', 'pages', pageFile);
524
- if (await fs.pathExists(src)) {
525
- await fs.copy(src, path.join(targetDir, 'src', 'pages', pageFile));
526
- }
527
- }
528
-
529
- // 9. Generate AuthGuard.tsx based on selected pages
530
- const firstProtectedPath = hasHome ? '/home' : hasTemplate ? '/template' : '/login';
531
- const authGuardContent = generateAuthGuard({
532
- hasLogin,
533
- hasHome,
534
- hasTemplate,
535
- hasAssistant,
536
- firstProtectedPath,
537
- });
538
-
539
- await fs.writeFile(
540
- path.join(targetDir, 'src', 'app', 'components', 'AuthGuard.tsx'),
541
- authGuardContent
542
- );
543
-
544
- // 10. Generate theme tokens
545
- const selectedTheme = colorThemes.find(t => t.id === response.theme) || colorThemes[0];
546
- const tokensDir = path.join(targetDir, 'src', 'styles', 'xertica');
547
- await fs.ensureDir(tokensDir);
548
- await fs.copy(
549
- path.join(templatesDir, 'src', 'styles', 'index.css'),
550
- path.join(targetDir, 'src', 'styles', 'index.css')
551
- );
552
- await fs.writeFile(path.join(tokensDir, 'tokens.css'), generateTokensCss(selectedTheme));
553
-
554
- spinner.succeed('Project initialized successfully!');
555
-
556
- if (response.install) {
557
- const installSpinner = ora('Installing dependencies...').start();
558
- await execa('npm', ['install'], { cwd: targetDir });
559
- installSpinner.succeed('Dependencies installed!');
560
- }
561
-
562
- console.log(chalk.green('\n✅ Done! Your Xertica UI project is ready.'));
563
- console.log(chalk.cyan(`\n cd ${directory}`));
564
- if (!response.install) {
565
- console.log(chalk.cyan(' npm install'));
566
- }
567
- console.log(chalk.cyan(' npm run dev'));
568
- console.log();
569
- console.log(chalk.gray(' Components are imported from the xertica-ui package.'));
570
- console.log(chalk.gray(' Customize the theme in src/styles/xertica/tokens.css'));
571
- const langLabels = SUPPORTED_LANGUAGES.filter(l => selectedLanguages.includes(l.code))
572
- .map(l => l.label)
573
- .join(', ');
574
- console.log(
575
- chalk.gray(
576
- ` Languages: ${langLabels}${selectedLanguages.length === 1 ? ' (monolingual — LanguageSelector hidden)' : ''}`
577
- )
578
- );
579
- console.log(chalk.gray(' To add/remove languages later: npx xertica-ui update → Languages'));
580
- console.log(
581
- chalk.gray(` AI Assistant: ${hasAssistant ? 'included (/assistente)' : 'not included'}`)
582
- );
583
- if (!hasAssistant) {
584
- console.log(chalk.gray(' To add the assistant later: npx xertica-ui update → Assistant'));
585
- }
586
- } catch (error) {
587
- spinner.fail('Failed to initialize project');
588
- console.error(error);
589
- }
590
- });
591
-
592
- program
593
- .command('update')
594
- .alias('update-theme')
595
- .description('Update theme or project files to the latest version')
596
- .action(async () => {
597
- const targetDir = process.cwd();
598
-
599
- console.log(chalk.blue(`🔧 Xertica UI CLI ${chalk.dim(`v${pkgJson.version}`)}`));
600
-
601
- const currentConfig = await readXerticaConfig(targetDir);
602
-
603
- const { updateType } = await prompts({
604
- type: 'select',
605
- name: 'updateType',
606
- message: 'What do you want to update?',
607
- choices: [
608
- {
609
- title: 'Theme only',
610
- description: 'Change the color tokens (tokens.css)',
611
- value: 'theme',
612
- },
613
- {
614
- title: 'Languages',
615
- description: 'Add or remove supported languages (pt-BR, en, es, …)',
616
- value: 'languages',
617
- },
618
- {
619
- title: 'Assistant',
620
- description: currentConfig?.hasAssistant
621
- ? 'Remove the AI Assistant from your project'
622
- : 'Add the AI Assistant to your project',
623
- value: 'assistant',
624
- },
625
- {
626
- title: 'Dark Mode',
627
- description: currentConfig?.disableDarkMode
628
- ? 'Enable dark mode support in your project'
629
- : 'Disable dark mode support in your project',
630
- value: 'darkmode',
631
- },
632
- {
633
- title: 'Project files',
634
- description: 'Update app shell, shared, features and pages to a specific version',
635
- value: 'project',
636
- },
637
- ],
638
- });
639
-
640
- if (!updateType) return;
641
-
642
- // ── Theme update ─────────────────────────────────────────────────────────
643
- if (updateType === 'theme') {
644
- const { theme } = await prompts({
645
- type: 'select',
646
- name: 'theme',
647
- message: 'Select the new color theme:',
648
- choices: colorThemes.map(t => ({
649
- title: t.name,
650
- description: t.description,
651
- value: t.id,
652
- })),
653
- initial: 0,
654
- });
655
-
656
- if (!theme) return;
657
-
658
- const spinner = ora('Updating theme...').start();
659
- try {
660
- const tokensPath = path.join(targetDir, 'src', 'styles', 'xertica', 'tokens.css');
661
- const selectedTheme = colorThemes.find(t => t.id === theme);
662
- if (selectedTheme) {
663
- await fs.ensureDir(path.dirname(tokensPath));
664
- await fs.writeFile(tokensPath, generateTokensCss(selectedTheme));
665
- spinner.succeed(`Theme updated to "${selectedTheme.name}" successfully!`);
666
- } else {
667
- spinner.fail('Theme not found.');
668
- }
669
- } catch (error) {
670
- spinner.fail('Failed to update theme');
671
- console.error(error);
672
- }
673
- return;
674
- }
675
-
676
- // ── Languages update (add / remove) ──────────────────────────────────────
677
- if (updateType === 'languages') {
678
- // Resolve current selection — read from .languages.json if present,
679
- // else inspect the locales/ folder to infer it (for projects scaffolded
680
- // before this feature shipped).
681
- const persistedCodes = await readLanguagesConfig(targetDir);
682
- let currentCodes: string[] = persistedCodes ?? [];
683
- if (currentCodes.length === 0) {
684
- const localesDir = path.join(targetDir, 'src', 'locales');
685
- if (await fs.pathExists(localesDir)) {
686
- const entries = await fs.readdir(localesDir);
687
- // Accept both the new folder layout (locales/<code>/) and the legacy
688
- // flat layout (locales/<code>.json) when inferring the current set.
689
- currentCodes = SUPPORTED_LANGUAGES.filter(
690
- l => entries.includes(l.jsonFile) || entries.includes(`${l.jsonFile}.json`)
691
- ).map(l => l.code);
692
- }
693
- if (currentCodes.length === 0) currentCodes = DEFAULT_SELECTION;
694
- }
695
-
696
- console.log(
697
- chalk.cyan(
698
- `\nCurrent languages: ${
699
- SUPPORTED_LANGUAGES.filter(l => currentCodes.includes(l.code))
700
- .map(l => l.label)
701
- .join(', ') || '(none)'
702
- }\n`
703
- )
704
- );
705
-
706
- const { newCodes } = await prompts({
707
- type: 'multiselect',
708
- name: 'newCodes',
709
- message: 'Select the languages this project should support:',
710
- instructions: false,
711
- hint: 'Press SPACE to toggle. At least 1 required. Single-language apps auto-hide the LanguageSelector.',
712
- min: 1,
713
- choices: SUPPORTED_LANGUAGES.map(l => ({
714
- title: l.label,
715
- value: l.code,
716
- selected: currentCodes.includes(l.code),
717
- })),
718
- });
719
-
720
- if (!Array.isArray(newCodes) || newCodes.length === 0) {
721
- console.log(chalk.gray('Update cancelled.'));
722
- return;
723
- }
724
-
725
- // Compute add/remove diff for the user-facing summary
726
- const toAdd = newCodes.filter((c: string) => !currentCodes.includes(c));
727
- const toRemove = currentCodes.filter(c => !newCodes.includes(c));
728
-
729
- if (toAdd.length === 0 && toRemove.length === 0) {
730
- console.log(chalk.gray('No changes — selection matches the current set.'));
731
- return;
732
- }
733
-
734
- const summary: string[] = [];
735
- if (toAdd.length > 0) summary.push(chalk.green(` + ${toAdd.join(', ')}`));
736
- if (toRemove.length > 0) summary.push(chalk.red(` - ${toRemove.join(', ')}`));
737
- console.log(`\n${summary.join('\n')}\n`);
738
-
739
- const { confirmed } = await prompts({
740
- type: 'confirm',
741
- name: 'confirmed',
742
- message: chalk.yellow(
743
- `⚠️ This will regenerate src/app/App.tsx and src/i18n.ts (preserving language-only changes). Continue?`
744
- ),
745
- initial: true,
746
- });
747
- if (!confirmed) {
748
- console.log(chalk.gray('Update cancelled.'));
749
- return;
750
- }
751
-
752
- const spinner = ora('Updating languages...').start();
753
- try {
754
- // The freshly-installed library may not be present in this flow, so we
755
- // read locale JSON sources from `node_modules/xertica-ui/templates`
756
- // (installed when the project was created) — fallback to package
757
- // directory lookup.
758
- const installedTemplatesDir = path.join(
759
- targetDir,
760
- 'node_modules',
761
- 'xertica-ui',
762
- 'templates'
763
- );
764
- const templatesSourceDir = (await fs.pathExists(installedTemplatesDir))
765
- ? installedTemplatesDir
766
- : path.resolve(__dirname, '../templates');
767
-
768
- // 1) Sync locale JSON files: copy newly-added, prune removed
769
- const { copied, removed } = await syncLocaleFiles(templatesSourceDir, targetDir, newCodes, {
770
- pruneOthers: true,
771
- });
772
-
773
- // 2) Regenerate i18n.ts so imports/resources reflect the new set
774
- await fs.writeFile(path.join(targetDir, 'src', 'i18n.ts'), generateI18nFile(newCodes));
775
-
776
- // 3) Regenerate App.tsx so the `availableLanguages` prop matches
777
- await fs.writeFile(
778
- path.join(targetDir, 'src', 'app', 'App.tsx'),
779
- generateAppTsx(newCodes, currentConfig?.disableDarkMode ?? false)
780
- );
781
-
782
- // 4) Persist the new selection
783
- await writeLanguagesConfig(targetDir, newCodes);
784
-
785
- spinner.succeed('Languages updated successfully!');
786
-
787
- if (copied.length > 0) console.log(chalk.green(` Copied: ${copied.join(', ')}`));
788
- if (removed.length > 0) console.log(chalk.red(` Removed: ${removed.join(', ')}`));
789
- if (newCodes.length === 1) {
790
- console.log(
791
- chalk.gray(` Project is now monolingual — the LanguageSelector will auto-hide.`)
792
- );
793
- }
794
- } catch (error) {
795
- spinner.fail('Failed to update languages');
796
- console.error(error);
797
- }
798
- return;
799
- }
800
-
801
- // ── Assistant add / remove ────────────────────────────────────────────────
802
- if (updateType === 'assistant') {
803
- // Determine current state: prefer persisted config, fall back to file presence
804
- // (handles projects scaffolded before .xertica.json existed).
805
- let currentlyHas: boolean;
806
- if (currentConfig !== null) {
807
- currentlyHas = currentConfig.hasAssistant;
808
- } else {
809
- const assistantFeatureDir = path.join(targetDir, 'src', 'features', 'assistant');
810
- const assistantPage = path.join(targetDir, 'src', 'pages', 'AssistantPage.tsx');
811
- currentlyHas =
812
- (await fs.pathExists(assistantFeatureDir)) || (await fs.pathExists(assistantPage));
813
- // Persist the inferred state so future runs don't need to infer again
814
- await writeXerticaConfig(targetDir, { hasAssistant: currentlyHas });
815
- }
816
-
817
- console.log(
818
- chalk.cyan(
819
- `\nAI Assistant is currently: ${currentlyHas ? chalk.green('enabled') : chalk.red('disabled')}\n`
820
- )
821
- );
822
-
823
- const { action } = await prompts({
824
- type: 'select',
825
- name: 'action',
826
- message: currentlyHas
827
- ? 'Remove the AI Assistant from your project?'
828
- : 'Add the AI Assistant to your project?',
829
- choices: currentlyHas
830
- ? [
831
- {
832
- title: 'Remove assistant',
833
- description: 'Deletes AssistantPage and assistant feature files',
834
- value: 'remove',
835
- },
836
- { title: 'Cancel', value: 'cancel' },
837
- ]
838
- : [
839
- {
840
- title: 'Add assistant',
841
- description: 'Copies AssistantPage and assistant feature files',
842
- value: 'add',
843
- },
844
- { title: 'Cancel', value: 'cancel' },
845
- ],
846
- });
847
-
848
- if (!action || action === 'cancel') {
849
- console.log(chalk.gray('Update cancelled.'));
850
- return;
851
- }
852
-
853
- const { confirmed } = await prompts({
854
- type: 'confirm',
855
- name: 'confirmed',
856
- message: chalk.yellow(
857
- action === 'remove'
858
- ? '⚠️ This will delete src/features/assistant/ and src/pages/AssistantPage.tsx and regenerate AuthGuard.tsx. Continue?'
859
- : '⚠️ This will copy src/features/assistant/ and src/pages/AssistantPage.tsx and regenerate AuthGuard.tsx. Continue?'
860
- ),
861
- initial: action === 'add',
862
- });
863
-
864
- if (!confirmed) {
865
- console.log(chalk.gray('Update cancelled.'));
866
- return;
867
- }
868
-
869
- const spinner = ora(
870
- action === 'add' ? 'Adding assistant...' : 'Removing assistant...'
871
- ).start();
872
-
873
- try {
874
- const installedTemplatesDir = path.join(
875
- targetDir,
876
- 'node_modules',
877
- 'xertica-ui',
878
- 'templates'
879
- );
880
- const templatesSourceDir = (await fs.pathExists(installedTemplatesDir))
881
- ? installedTemplatesDir
882
- : path.resolve(__dirname, '../templates');
883
-
884
- // Infer current page set from the pages directory
885
- const pagesDir = path.join(targetDir, 'src', 'pages');
886
- const existingPages = (await fs.pathExists(pagesDir)) ? await fs.readdir(pagesDir) : [];
887
- const hasLogin = existingPages.includes('LoginPage.tsx');
888
- const hasHome = existingPages.includes('HomePage.tsx');
889
- const hasTemplate = existingPages.includes('TemplatePage.tsx');
890
- const firstProtectedPath = hasHome ? '/home' : hasTemplate ? '/template' : '/login';
891
-
892
- // Read persisted language selection so AuthGuard can be regenerated correctly
893
- const persistedCodes = await readLanguagesConfig(targetDir);
894
- const selectedCodes =
895
- persistedCodes && persistedCodes.length > 0 ? persistedCodes : DEFAULT_SELECTION;
896
-
897
- if (action === 'add') {
898
- // Copy assistant feature
899
- await fs.copy(
900
- path.join(templatesSourceDir, 'src', 'features', 'assistant'),
901
- path.join(targetDir, 'src', 'features', 'assistant'),
902
- { overwrite: true }
903
- );
904
- // Copy AssistantPage
905
- await fs.copy(
906
- path.join(templatesSourceDir, 'src', 'pages', 'AssistantPage.tsx'),
907
- path.join(targetDir, 'src', 'pages', 'AssistantPage.tsx'),
908
- { overwrite: true }
909
- );
910
- } else {
911
- // Remove assistant feature
912
- await fs.remove(path.join(targetDir, 'src', 'features', 'assistant'));
913
- await fs.remove(path.join(targetDir, 'src', 'pages', 'AssistantPage.tsx'));
914
- }
915
-
916
- // Regenerate pages and AuthGuard reflecting the new assistant state
917
- const newHasAssistant = action === 'add';
918
-
919
- if (hasHome) {
920
- await fs.writeFile(
921
- path.join(targetDir, 'src', 'pages', 'HomePage.tsx'),
922
- generateHomePage(newHasAssistant)
923
- );
924
- }
925
- if (hasTemplate) {
926
- await fs.writeFile(
927
- path.join(targetDir, 'src', 'pages', 'TemplatePage.tsx'),
928
- generateTemplatePage(newHasAssistant)
929
- );
930
- }
931
-
932
- await fs.writeFile(
933
- path.join(targetDir, 'src', 'app', 'components', 'AuthGuard.tsx'),
934
- generateAuthGuard({
935
- hasLogin,
936
- hasHome,
937
- hasTemplate,
938
- hasAssistant: newHasAssistant,
939
- firstProtectedPath,
940
- })
941
- );
942
-
943
- // Persist the updated flag
944
- await writeXerticaConfig(targetDir, { hasAssistant: newHasAssistant });
945
-
946
- spinner.succeed(
947
- action === 'add'
948
- ? 'AI Assistant added successfully!'
949
- : 'AI Assistant removed successfully!'
950
- );
951
-
952
- if (action === 'add') {
953
- console.log(chalk.gray('\n Route /assistente is now available.'));
954
- console.log(chalk.gray(' Configure your Gemini API key in VITE_GEMINI_API_KEY.'));
955
- } else {
956
- console.log(chalk.gray('\n Assistant files removed and AuthGuard updated.'));
957
- }
958
- } catch (error) {
959
- spinner.fail('Failed to update assistant');
960
- console.error(error);
961
- }
962
- return;
963
- }
964
-
965
- // ── Dark Mode update (enable / disable) ──────────────────────────────────
966
- if (updateType === 'darkmode') {
967
- const currentlyDisabled = currentConfig?.disableDarkMode ?? false;
968
- const { enableDarkMode } = await prompts({
969
- type: 'confirm',
970
- name: 'enableDarkMode',
971
- message: currentlyDisabled
972
- ? 'Enable dark mode support in your project?'
973
- : 'Disable dark mode support in your project? (This will hide the toggle and force light mode)',
974
- initial: !currentlyDisabled,
975
- });
976
-
977
- if (enableDarkMode === undefined) return;
978
-
979
- const newDisableDarkMode = !enableDarkMode;
980
-
981
- const spinner = ora(
982
- newDisableDarkMode ? 'Disabling dark mode...' : 'Enabling dark mode...'
983
- ).start();
984
- try {
985
- // Persist the selection
986
- await writeXerticaConfig(targetDir, { disableDarkMode: newDisableDarkMode });
987
-
988
- // Regenerate App.tsx with the new dark mode flag
989
- const persistedCodes = await readLanguagesConfig(targetDir);
990
- const selectedCodes =
991
- persistedCodes && persistedCodes.length > 0 ? persistedCodes : DEFAULT_SELECTION;
992
-
993
- await fs.writeFile(
994
- path.join(targetDir, 'src', 'app', 'App.tsx'),
995
- generateAppTsx(selectedCodes, newDisableDarkMode)
996
- );
997
-
998
- spinner.succeed(
999
- newDisableDarkMode
1000
- ? 'Dark mode disabled successfully! (Locked to Light Mode)'
1001
- : 'Dark mode enabled successfully!'
1002
- );
1003
- } catch (error) {
1004
- spinner.fail('Failed to update dark mode configuration');
1005
- console.error(error);
1006
- }
1007
- return;
1008
- }
1009
-
1010
- // ── Project files update ──────────────────────────────────────────────────
1011
- const { versionType } = await prompts({
1012
- type: 'select',
1013
- name: 'versionType',
1014
- message: 'Which version do you want to update to?',
1015
- choices: [
1016
- { title: 'Latest', description: 'Install the latest published version', value: 'latest' },
1017
- {
1018
- title: 'Specific version',
1019
- description: 'Enter a version number (e.g. 2.0.2)',
1020
- value: 'specific',
1021
- },
1022
- ],
1023
- });
1024
-
1025
- if (!versionType) return;
1026
-
1027
- let targetVersion = 'latest';
1028
- if (versionType === 'specific') {
1029
- const { version } = await prompts({
1030
- type: 'text',
1031
- name: 'version',
1032
- message: 'Enter the version (e.g. 2.0.2):',
1033
- validate: v =>
1034
- /^\d+\.\d+\.\d+/.test(v.trim()) ? true : 'Enter a valid semver (e.g. 2.0.2)',
1035
- });
1036
- if (!version) return;
1037
- targetVersion = version.trim();
1038
- }
1039
-
1040
- const { filesToUpdate } = await prompts({
1041
- type: 'multiselect',
1042
- name: 'filesToUpdate',
1043
- message: 'Select which parts of the project to update:',
1044
- choices: [
1045
- {
1046
- title: 'App shell (src/app/)',
1047
- description: 'App.tsx, AppLayout.tsx',
1048
- value: 'app',
1049
- selected: true,
1050
- },
1051
- {
1052
- title: 'Shared utilities (src/shared/)',
1053
- description: 'auth.ts, navigation.ts, types',
1054
- value: 'shared',
1055
- selected: true,
1056
- },
1057
- {
1058
- title: 'Features (src/features/)',
1059
- description: 'auth, home, template UI components',
1060
- value: 'features',
1061
- selected: true,
1062
- },
1063
- {
1064
- title: 'Pages (src/pages/)',
1065
- description: 'Thin page wrapper components',
1066
- value: 'pages',
1067
- selected: true,
1068
- },
1069
- {
1070
- title: 'Root config files',
1071
- description: 'vite.config.ts, tsconfig.json, etc.',
1072
- value: 'config',
1073
- selected: false,
1074
- },
1075
- ],
1076
- });
1077
-
1078
- if (!filesToUpdate || filesToUpdate.length === 0) return;
1079
-
1080
- const { confirmed } = await prompts({
1081
- type: 'confirm',
1082
- name: 'confirmed',
1083
- message: chalk.yellow(
1084
- `⚠️ This will overwrite the selected files. Local changes will be lost. Continue?`
1085
- ),
1086
- initial: false,
1087
- });
1088
-
1089
- if (!confirmed) {
1090
- console.log(chalk.gray('Update cancelled.'));
1091
- return;
1092
- }
1093
-
1094
- const spinner = ora(`Installing xertica-ui@${targetVersion}...`).start();
1095
-
1096
- try {
1097
- // Install the target version in the consumer project
1098
- await execa('npm', ['install', `xertica-ui@${targetVersion}`], { cwd: targetDir });
1099
- spinner.text = 'Copying updated files...';
1100
-
1101
- // Templates now come from the freshly installed version
1102
- const updatedTemplatesDir = path.join(targetDir, 'node_modules', 'xertica-ui', 'templates');
1103
-
1104
- if (filesToUpdate.includes('app')) {
1105
- // AppLayout is always safe to overwrite (no per-project config baked in)
1106
- await fs.copy(
1107
- path.join(updatedTemplatesDir, 'src', 'app', 'components', 'AppLayout.tsx'),
1108
- path.join(targetDir, 'src', 'app', 'components', 'AppLayout.tsx'),
1109
- { overwrite: true }
1110
- );
1111
-
1112
- // For App.tsx and i18n.ts we must preserve the user's language selection.
1113
- // Read it from .languages.json (or fall back to inspecting locales/ for
1114
- // projects scaffolded before the file existed).
1115
- const persistedCodes = await readLanguagesConfig(targetDir);
1116
- let selectedCodes: string[] = persistedCodes ?? [];
1117
- if (selectedCodes.length === 0) {
1118
- const localesDir = path.join(targetDir, 'src', 'locales');
1119
- if (await fs.pathExists(localesDir)) {
1120
- const entries = await fs.readdir(localesDir);
1121
- // Accept both the new folder layout and the legacy flat layout
1122
- selectedCodes = SUPPORTED_LANGUAGES.filter(
1123
- l => entries.includes(l.jsonFile) || entries.includes(`${l.jsonFile}.json`)
1124
- ).map(l => l.code);
1125
- }
1126
- if (selectedCodes.length === 0) selectedCodes = DEFAULT_SELECTION;
1127
- // Persist the inferred selection so future updates have it cached
1128
- await writeLanguagesConfig(targetDir, selectedCodes);
1129
- }
1130
-
1131
- const projectConfig = await readXerticaConfig(targetDir);
1132
-
1133
- // Regenerate App.tsx and i18n.ts honoring the persisted language set
1134
- await fs.writeFile(
1135
- path.join(targetDir, 'src', 'app', 'App.tsx'),
1136
- generateAppTsx(selectedCodes, projectConfig?.disableDarkMode ?? false)
1137
- );
1138
- await fs.writeFile(path.join(targetDir, 'src', 'i18n.ts'), generateI18nFile(selectedCodes));
1139
-
1140
- // Refresh locale JSON files for the selected languages (keys grow over
1141
- // library updates) — but prune any orphans from prior selections.
1142
- await syncLocaleFiles(updatedTemplatesDir, targetDir, selectedCodes, {
1143
- pruneOthers: true,
1144
- });
1145
-
1146
- // Regenerate AuthGuard preserving the current page set and assistant flag
1147
- const pagesDir = path.join(targetDir, 'src', 'pages');
1148
- const existingPages = (await fs.pathExists(pagesDir)) ? await fs.readdir(pagesDir) : [];
1149
- const hasLoginP = existingPages.includes('LoginPage.tsx');
1150
- const hasHomeP = existingPages.includes('HomePage.tsx');
1151
- const hasTemplateP = existingPages.includes('TemplatePage.tsx');
1152
- const hasAssistantP =
1153
- projectConfig?.hasAssistant ?? existingPages.includes('AssistantPage.tsx');
1154
- const firstProtectedP = hasHomeP ? '/home' : hasTemplateP ? '/template' : '/login';
1155
- await fs.writeFile(
1156
- path.join(targetDir, 'src', 'app', 'components', 'AuthGuard.tsx'),
1157
- generateAuthGuard({
1158
- hasLogin: hasLoginP,
1159
- hasHome: hasHomeP,
1160
- hasTemplate: hasTemplateP,
1161
- hasAssistant: hasAssistantP,
1162
- firstProtectedPath: firstProtectedP,
1163
- })
1164
- );
1165
- }
1166
-
1167
- if (filesToUpdate.includes('shared')) {
1168
- await fs.copy(
1169
- path.join(updatedTemplatesDir, 'src', 'shared'),
1170
- path.join(targetDir, 'src', 'shared'),
1171
- { overwrite: true }
1172
- );
1173
- }
1174
-
1175
- if (filesToUpdate.includes('features')) {
1176
- // Only update feature directories that already exist in the project.
1177
- // 'assistant' is only present if it was included during init (or added via update → Assistant).
1178
- for (const feature of ['auth', 'home', 'template', 'assistant']) {
1179
- const destFeature = path.join(targetDir, 'src', 'features', feature);
1180
- const srcFeature = path.join(updatedTemplatesDir, 'src', 'features', feature);
1181
- if ((await fs.pathExists(destFeature)) && (await fs.pathExists(srcFeature))) {
1182
- await fs.copy(srcFeature, destFeature, { overwrite: true });
1183
- }
1184
- }
1185
- }
1186
-
1187
- if (filesToUpdate.includes('pages')) {
1188
- const pagesDir = path.join(targetDir, 'src', 'pages');
1189
- const srcPagesDir = path.join(updatedTemplatesDir, 'src', 'pages');
1190
- if ((await fs.pathExists(pagesDir)) && (await fs.pathExists(srcPagesDir))) {
1191
- const projectCfg = await readXerticaConfig(targetDir);
1192
- const existingPages = await fs.readdir(pagesDir);
1193
- const assistantEnabled =
1194
- projectCfg?.hasAssistant ?? existingPages.includes('AssistantPage.tsx');
1195
-
1196
- for (const pageFile of existingPages) {
1197
- // Generated pages: regenerate instead of copy so assistant imports stay correct
1198
- if (pageFile === 'HomePage.tsx') {
1199
- await fs.writeFile(
1200
- path.join(pagesDir, 'HomePage.tsx'),
1201
- generateHomePage(assistantEnabled)
1202
- );
1203
- continue;
1204
- }
1205
- if (pageFile === 'TemplatePage.tsx') {
1206
- await fs.writeFile(
1207
- path.join(pagesDir, 'TemplatePage.tsx'),
1208
- generateTemplatePage(assistantEnabled)
1209
- );
1210
- continue;
1211
- }
1212
- const src = path.join(srcPagesDir, pageFile);
1213
- if (await fs.pathExists(src)) {
1214
- await fs.copy(src, path.join(pagesDir, pageFile), { overwrite: true });
1215
- }
1216
- }
1217
- }
1218
- }
1219
-
1220
- if (filesToUpdate.includes('config')) {
1221
- // Config files are inside the templates/ directory (same level as src/)
1222
- const configFiles = [
1223
- 'vite.config.ts',
1224
- 'tsconfig.json',
1225
- 'tsconfig.node.json',
1226
- 'postcss.config.js',
1227
- ];
1228
- for (const file of configFiles) {
1229
- const src = path.join(updatedTemplatesDir, file);
1230
- if (await fs.pathExists(src)) {
1231
- await fs.copy(src, path.join(targetDir, file), { overwrite: true });
1232
- }
1233
- }
1234
- }
1235
-
1236
- spinner.succeed(`Project updated to xertica-ui@${targetVersion} successfully!`);
1237
- console.log(chalk.gray('\n Run npm run dev to start the development server.'));
1238
- } catch (error) {
1239
- spinner.fail('Failed to update project');
1240
- console.error(error);
1241
- }
1242
- });
1243
-
1244
- program.parse();
1
+ #!/usr/bin/env node
2
+ import { Command } from 'commander';
3
+ import prompts from 'prompts';
4
+ import chalk from 'chalk';
5
+ import ora from 'ora';
6
+ import fs from 'fs-extra';
7
+ import path from 'path';
8
+ import { fileURLToPath } from 'url';
9
+ import { execa } from 'execa';
10
+ import { colorThemes } from '../contexts/theme-data';
11
+ import { generateTokensCss } from './generate-tokens';
12
+ import {
13
+ SUPPORTED_LANGUAGES,
14
+ DEFAULT_SELECTION,
15
+ readLanguagesConfig,
16
+ writeLanguagesConfig,
17
+ syncLocaleFiles,
18
+ generateI18nFile,
19
+ generateAppTsx,
20
+ } from './language-config';
21
+
22
+ // ─────────────────────────────────────────────────────────────────────────────
23
+ // Project config helpers (.xertica.json)
24
+ // Persists per-project feature flags so `update` can read them later.
25
+ // ─────────────────────────────────────────────────────────────────────────────
26
+
27
+ const XERTICA_CONFIG_FILE = '.xertica.json';
28
+
29
+ interface XerticaConfig {
30
+ version: 1;
31
+ hasAssistant: boolean;
32
+ disableDarkMode?: boolean;
33
+ }
34
+
35
+ async function readXerticaConfig(targetDir: string): Promise<XerticaConfig | null> {
36
+ const configPath = path.join(targetDir, XERTICA_CONFIG_FILE);
37
+ if (!(await fs.pathExists(configPath))) return null;
38
+ try {
39
+ return (await fs.readJson(configPath)) as XerticaConfig;
40
+ } catch {
41
+ return null;
42
+ }
43
+ }
44
+
45
+ async function writeXerticaConfig(
46
+ targetDir: string,
47
+ config: Partial<XerticaConfig>
48
+ ): Promise<void> {
49
+ const configPath = path.join(targetDir, XERTICA_CONFIG_FILE);
50
+ const existing = (await readXerticaConfig(targetDir)) ?? {
51
+ version: 1 as const,
52
+ hasAssistant: false,
53
+ };
54
+ await fs.writeJson(configPath, { ...existing, ...config, version: 1 }, { spaces: 2 });
55
+ }
56
+
57
+ const __filename = fileURLToPath(import.meta.url);
58
+ const __dirname = path.dirname(__filename);
59
+
60
+ // Read CLI version from package.json so it always matches the published one,
61
+ // instead of hardcoding a literal that drifts on every version bump.
62
+ const pkgJson = JSON.parse(
63
+ fs.readFileSync(path.resolve(__dirname, '../package.json'), 'utf-8')
64
+ ) as { version: string };
65
+
66
+ // ─────────────────────────────────────────────────────────────────────────────
67
+ // AuthGuard generator
68
+ // Generates src/app/components/AuthGuard.tsx based on the selected features.
69
+ // ─────────────────────────────────────────────────────────────────────────────
70
+
71
+ function generateAuthGuard({
72
+ hasLogin,
73
+ hasHome,
74
+ hasTemplate,
75
+ hasAssistant,
76
+ firstProtectedPath,
77
+ }: {
78
+ hasLogin: boolean;
79
+ hasHome: boolean;
80
+ hasTemplate: boolean;
81
+ hasAssistant: boolean;
82
+ firstProtectedPath: string;
83
+ }): string {
84
+ return `import React from 'react';
85
+ import { Routes, Route, Navigate } from 'react-router-dom';
86
+ import { useAuth } from '../context/AuthContext';
87
+
88
+ // ─── Lazy page imports ────────────────────────────────────────────────────────
89
+
90
+ ${
91
+ hasLogin
92
+ ? `const LoginPage = React.lazy(() => import('../../pages/LoginPage').then(m => ({ default: m.LoginPage })));
93
+ const ForgotPasswordPage = React.lazy(() => import('../../pages/ForgotPasswordPage').then(m => ({ default: m.ForgotPasswordPage })));
94
+ const VerifyEmailPage = React.lazy(() => import('../../pages/VerifyEmailPage').then(m => ({ default: m.VerifyEmailPage })));
95
+ const ResetPasswordPage = React.lazy(() => import('../../pages/ResetPasswordPage').then(m => ({ default: m.ResetPasswordPage })));`
96
+ : ''
97
+ }
98
+
99
+ ${hasHome ? `const HomePage = React.lazy(() => import('../../pages/HomePage').then(m => ({ default: m.HomePage })));` : ''}
100
+ ${hasTemplate ? `const TemplatePage = React.lazy(() => import('../../pages/TemplatePage').then(m => ({ default: m.TemplatePage })));` : ''}
101
+ ${hasAssistant ? `const AssistantPage = React.lazy(() => import('../../pages/AssistantPage').then(m => ({ default: m.AssistantPage })));` : ''}
102
+
103
+ // ─── Route guards ─────────────────────────────────────────────────────────────
104
+
105
+ function ProtectedRoute({ children }: { children: React.ReactNode }) {
106
+ const { user, isLoading } = useAuth();
107
+ if (isLoading) return null;
108
+ if (!user) return <Navigate to="${hasLogin ? '/login' : firstProtectedPath}" replace />;
109
+ return <>{children}</>;
110
+ }
111
+
112
+ ${
113
+ hasLogin
114
+ ? `function GuestRoute({ children }: { children: React.ReactNode }) {
115
+ const { user, isLoading } = useAuth();
116
+ if (isLoading) return null;
117
+ if (user) return <Navigate to="${firstProtectedPath}" replace />;
118
+ return <>{children}</>;
119
+ }
120
+
121
+ function LoginPageWithAuth() {
122
+ const { login } = useAuth();
123
+ return <LoginPage onLogin={login} />;
124
+ }`
125
+ : ''
126
+ }
127
+
128
+ // ─── Route tree ───────────────────────────────────────────────────────────────
129
+
130
+ export function AuthGuard() {
131
+ const { user } = useAuth();
132
+
133
+ return (
134
+ <div className="min-h-screen bg-muted overflow-x-hidden max-w-full">
135
+ <Routes>
136
+ ${
137
+ hasLogin
138
+ ? ` <Route path="/login" element={<GuestRoute><LoginPageWithAuth /></GuestRoute>} />
139
+ <Route path="/forgot-password" element={<GuestRoute><ForgotPasswordPage /></GuestRoute>} />
140
+ <Route path="/verify-email" element={<GuestRoute><VerifyEmailPage /></GuestRoute>} />
141
+ <Route path="/reset-password" element={<GuestRoute><ResetPasswordPage /></GuestRoute>} />`
142
+ : ''
143
+ }
144
+
145
+ ${hasHome ? ` <Route path="/home" element={<ProtectedRoute><HomePage /></ProtectedRoute>} />` : ''}
146
+ ${hasTemplate ? ` <Route path="/template" element={<ProtectedRoute><TemplatePage /></ProtectedRoute>} />` : ''}
147
+ ${hasAssistant ? ` <Route path="/assistente" element={<ProtectedRoute><AssistantPage /></ProtectedRoute>} />` : ''}
148
+
149
+ <Route path="/" element={<Navigate to={user ? '${firstProtectedPath}' : '${hasLogin ? '/login' : firstProtectedPath}'} replace />} />
150
+ <Route path="*" element={<Navigate to={user ? '${firstProtectedPath}' : '${hasLogin ? '/login' : firstProtectedPath}'} replace />} />
151
+ </Routes>
152
+ </div>
153
+ );
154
+ }
155
+ `;
156
+ }
157
+
158
+ // ─────────────────────────────────────────────────────────────────────────────
159
+ // Page generators
160
+ // Generate page files that vary based on whether the assistant is included.
161
+ // ─────────────────────────────────────────────────────────────────────────────
162
+
163
+ function generateHomePage(hasAssistant: boolean): string {
164
+ if (hasAssistant) {
165
+ return `import React from 'react';
166
+ import { XerticaAssistant, generateDemoResponse } from 'xertica-ui/assistant';
167
+ import { useLayout } from 'xertica-ui/hooks';
168
+ import { useNavigate } from 'react-router-dom';
169
+ import { AppLayout } from '../app/components/AppLayout';
170
+ import { HomeContent } from '../features/home';
171
+ import { useAssistantConfig, getMockRichSuggestions, getMockFeedbackOptions } from '../features/assistant';
172
+
173
+ /**
174
+ * Home page — thin layout shell.
175
+ *
176
+ * Assistant config (suggestions, feedback options) is fetched via React Query.
177
+ * To connect to a real API replace \`fetchAssistantConfig\` in
178
+ * \`features/assistant/data/mock.ts\`.
179
+ */
180
+ export function HomePage() {
181
+ const { assistenteExpanded, toggleAssistente } = useLayout();
182
+ const navigate = useNavigate();
183
+
184
+ const { data: assistantConfig } = useAssistantConfig();
185
+
186
+ return (
187
+ <AppLayout
188
+ assistant={
189
+ <XerticaAssistant
190
+ isExpanded={assistenteExpanded}
191
+ onToggle={toggleAssistente}
192
+ defaultTab="chat"
193
+ demoMode={true}
194
+ userName="Ariel Santos"
195
+ responseGenerator={generateDemoResponse}
196
+ suggestions={assistantConfig?.suggestions}
197
+ richSuggestions={assistantConfig?.richSuggestions ?? getMockRichSuggestions()}
198
+ feedbackOptions={assistantConfig?.feedbackOptions ?? getMockFeedbackOptions()}
199
+ showHistory={false}
200
+ showFavorites={false}
201
+ onNavigateSettings={() => navigate('/settings')}
202
+ onNavigateFullPage={() => navigate('/assistente')}
203
+ onEvaluation={(messageId, type, reason) => {
204
+ // Wire your feedback persistence logic here
205
+ console.log(\`Avaliação: \${type} na mensagem \${messageId}. Motivo: \${reason}\`);
206
+ }}
207
+ />
208
+ }
209
+ >
210
+ <HomeContent />
211
+ </AppLayout>
212
+ );
213
+ }
214
+ `;
215
+ }
216
+
217
+ return `import React from 'react';
218
+ import { AppLayout } from '../app/components/AppLayout';
219
+ import { HomeContent } from '../features/home';
220
+
221
+ /**
222
+ * Home page — thin layout shell.
223
+ */
224
+ export function HomePage() {
225
+ return (
226
+ <AppLayout>
227
+ <HomeContent />
228
+ </AppLayout>
229
+ );
230
+ }
231
+ `;
232
+ }
233
+
234
+ function generateTemplatePage(hasAssistant: boolean): string {
235
+ if (hasAssistant) {
236
+ return `import React from 'react';
237
+ import { XerticaAssistant } from 'xertica-ui/assistant';
238
+ import { useLayout } from 'xertica-ui/hooks';
239
+ import { AppLayout } from '../app/components/AppLayout';
240
+ import { TemplateContent } from '../features/template';
241
+
242
+ /**
243
+ * Template page — thin layout shell.
244
+ *
245
+ * Auth state is consumed from \`AuthContext\` via \`AppLayout\` — no props needed.
246
+ */
247
+ export function TemplatePage() {
248
+ const { assistenteExpanded, toggleAssistente } = useLayout();
249
+
250
+ return (
251
+ <AppLayout
252
+ assistant={
253
+ <XerticaAssistant
254
+ isExpanded={assistenteExpanded}
255
+ onToggle={toggleAssistente}
256
+ onEvaluation={(id, type, reason) => console.log('Feedback:', id, type, reason)}
257
+ />
258
+ }
259
+ >
260
+ <TemplateContent />
261
+ </AppLayout>
262
+ );
263
+ }
264
+ `;
265
+ }
266
+
267
+ return `import React from 'react';
268
+ import { AppLayout } from '../app/components/AppLayout';
269
+ import { TemplateContent } from '../features/template';
270
+
271
+ /**
272
+ * Template page — thin layout shell.
273
+ *
274
+ * Auth state is consumed from \`AuthContext\` via \`AppLayout\` — no props needed.
275
+ */
276
+ export function TemplatePage() {
277
+ return (
278
+ <AppLayout>
279
+ <TemplateContent />
280
+ </AppLayout>
281
+ );
282
+ }
283
+ `;
284
+ }
285
+
286
+ const program = new Command();
287
+
288
+ program
289
+ .name('xertica-ui')
290
+ .description('CLI to initialize Xertica UI projects')
291
+ .version(pkgJson.version);
292
+
293
+ program
294
+ .command('init')
295
+ .description('Initialize a new Xertica UI project')
296
+ .argument('[directory]', 'Directory to initialize in', '.')
297
+ .action(async directory => {
298
+ const targetDir = path.resolve(process.cwd(), directory);
299
+ const templatesDir = path.resolve(__dirname, '../templates');
300
+
301
+ console.log(chalk.blue(`🚀 Welcome to Xertica UI CLI! ${chalk.dim(`v${pkgJson.version}`)}`));
302
+
303
+ const response = await prompts([
304
+ {
305
+ type: 'multiselect',
306
+ name: 'pages',
307
+ message: 'Which pages/templates to include?',
308
+ choices: [
309
+ {
310
+ title: 'Login Page (+ Forgot / Verify / Reset Password)',
311
+ value: 'login',
312
+ selected: true,
313
+ },
314
+ { title: 'Home Page', value: 'home', selected: true },
315
+ { title: 'Template Page (components showcase)', value: 'template', selected: true },
316
+ ],
317
+ },
318
+ {
319
+ type: 'multiselect',
320
+ name: 'languages',
321
+ message: 'Which languages should the app support?',
322
+ instructions: false,
323
+ hint: 'Select at least 1. Single-language apps auto-hide the LanguageSelector.',
324
+ min: 1,
325
+ choices: SUPPORTED_LANGUAGES.map(l => ({
326
+ title: l.label,
327
+ value: l.code,
328
+ selected: true,
329
+ })),
330
+ },
331
+ {
332
+ type: 'select',
333
+ name: 'theme',
334
+ message: 'Select the default color theme for your project:',
335
+ choices: colorThemes.map(t => ({
336
+ title: t.name,
337
+ description: t.description,
338
+ value: t.id,
339
+ })),
340
+ initial: 0,
341
+ },
342
+ {
343
+ type: 'confirm',
344
+ name: 'hasAssistant',
345
+ message: 'Include AI Assistant? (XerticaAssistant chat page + sidebar variant)',
346
+ initial: true,
347
+ },
348
+ {
349
+ type: 'confirm',
350
+ name: 'enableDarkMode',
351
+ message: 'Enable dark mode support?',
352
+ initial: true,
353
+ },
354
+ {
355
+ type: 'confirm',
356
+ name: 'install',
357
+ message: 'Install dependencies automatically?',
358
+ initial: true,
359
+ },
360
+ ]);
361
+
362
+ // Abort if the user cancelled any prompt (prompts returns undefined on Ctrl+C)
363
+ if (!response.pages || !response.languages || !response.theme) return;
364
+
365
+ const spinner = ora('Initializing project...').start();
366
+
367
+ try {
368
+ await fs.ensureDir(targetDir);
369
+
370
+ const pages = response.pages || [];
371
+ const hasLogin = pages.includes('login');
372
+ const hasHome = pages.includes('home');
373
+ const hasTemplate = pages.includes('template');
374
+ const hasAssistant = response.hasAssistant ?? true;
375
+
376
+ // Resolve selected languages — fall back to all defaults if the user
377
+ // somehow ended up with an empty array (the prompt's min:1 should prevent
378
+ // this, but we defend defensively).
379
+ const selectedLanguages: string[] =
380
+ Array.isArray(response.languages) && response.languages.length > 0
381
+ ? response.languages
382
+ : DEFAULT_SELECTION;
383
+
384
+ // 1. Copy root config files
385
+ const rootFilesToCopy = [
386
+ 'index.html',
387
+ 'vite.config.ts',
388
+ 'tsconfig.json',
389
+ 'tsconfig.node.json',
390
+ 'postcss.config.js',
391
+ 'vite-env.d.ts',
392
+ 'eslint.config.js',
393
+ '.env.example',
394
+ 'guidelines',
395
+ 'CLAUDE.md',
396
+ ];
397
+
398
+ for (const file of rootFilesToCopy) {
399
+ const srcPath = path.join(templatesDir, file);
400
+ if (await fs.pathExists(srcPath)) {
401
+ await fs.copy(srcPath, path.join(targetDir, file));
402
+ }
403
+ }
404
+
405
+ // 2. Copy package.json
406
+ const pkgTemplatePath = path.join(templatesDir, 'package.json');
407
+ if (await fs.pathExists(pkgTemplatePath)) {
408
+ const pkgContent = await fs.readJson(pkgTemplatePath);
409
+ const projectName = path.basename(targetDir) || 'my-xertica-app';
410
+ pkgContent.name = projectName.toLowerCase().replace(/[^a-z0-9-]/g, '-');
411
+ await fs.writeJson(path.join(targetDir, 'package.json'), pkgContent, { spaces: 2 });
412
+ }
413
+
414
+ // 3. Copy src/main.tsx
415
+ await fs.copy(
416
+ path.join(templatesDir, 'src', 'main.tsx'),
417
+ path.join(targetDir, 'src', 'main.tsx')
418
+ );
419
+
420
+ const disableDarkMode = response.enableDarkMode === false;
421
+
422
+ // 4. Generate src/app/App.tsx with the user's language selection
423
+ // (instead of copying the static template, we inject `availableLanguages`)
424
+ await fs.ensureDir(path.join(targetDir, 'src', 'app'));
425
+ await fs.writeFile(
426
+ path.join(targetDir, 'src', 'app', 'App.tsx'),
427
+ generateAppTsx(selectedLanguages, disableDarkMode)
428
+ );
429
+
430
+ // 5. Copy src/app/components/AppLayout.tsx (always needed)
431
+ await fs.ensureDir(path.join(targetDir, 'src', 'app', 'components'));
432
+ await fs.copy(
433
+ path.join(templatesDir, 'src', 'app', 'components', 'AppLayout.tsx'),
434
+ path.join(targetDir, 'src', 'app', 'components', 'AppLayout.tsx')
435
+ );
436
+
437
+ // 6. Copy src/shared/ (always needed — auth helpers, navigation config, types)
438
+ await fs.copy(
439
+ path.join(templatesDir, 'src', 'shared'),
440
+ path.join(targetDir, 'src', 'shared')
441
+ );
442
+
443
+ // 6.1 Generate i18n.ts with only the imports/resources for selected languages
444
+ await fs.writeFile(
445
+ path.join(targetDir, 'src', 'i18n.ts'),
446
+ generateI18nFile(selectedLanguages)
447
+ );
448
+
449
+ // 6.2 Copy only the selected locale JSON files (no orphan locales)
450
+ await syncLocaleFiles(templatesDir, targetDir, selectedLanguages, { pruneOthers: true });
451
+
452
+ // 6.3 Persist the language selection so `update` can remember it
453
+ await writeLanguagesConfig(targetDir, selectedLanguages);
454
+
455
+ // 6.5 Persist project feature flags (.xertica.json)
456
+ await writeXerticaConfig(targetDir, { hasAssistant, disableDarkMode });
457
+
458
+ // 6.4 Copy context
459
+ await fs.ensureDir(path.join(targetDir, 'src', 'app', 'context'));
460
+ await fs.copy(
461
+ path.join(templatesDir, 'src', 'app', 'context', 'AuthContext.tsx'),
462
+ path.join(targetDir, 'src', 'app', 'context', 'AuthContext.tsx')
463
+ );
464
+
465
+ // 7. Copy features based on selections
466
+ if (hasLogin) {
467
+ await fs.copy(
468
+ path.join(templatesDir, 'src', 'features', 'auth'),
469
+ path.join(targetDir, 'src', 'features', 'auth')
470
+ );
471
+ }
472
+ if (hasHome) {
473
+ await fs.copy(
474
+ path.join(templatesDir, 'src', 'features', 'home'),
475
+ path.join(targetDir, 'src', 'features', 'home')
476
+ );
477
+ }
478
+ if (hasTemplate) {
479
+ await fs.copy(
480
+ path.join(templatesDir, 'src', 'features', 'template'),
481
+ path.join(targetDir, 'src', 'features', 'template')
482
+ );
483
+ }
484
+ // Copy assistant feature only if selected
485
+ if (hasAssistant) {
486
+ await fs.copy(
487
+ path.join(templatesDir, 'src', 'features', 'assistant'),
488
+ path.join(targetDir, 'src', 'features', 'assistant')
489
+ );
490
+ }
491
+
492
+ // 8. Copy pages based on selections
493
+ await fs.ensureDir(path.join(targetDir, 'src', 'pages'));
494
+
495
+ const pagesToCopy: string[] = [];
496
+ if (hasLogin)
497
+ pagesToCopy.push(
498
+ 'LoginPage.tsx',
499
+ 'ForgotPasswordPage.tsx',
500
+ 'VerifyEmailPage.tsx',
501
+ 'ResetPasswordPage.tsx'
502
+ );
503
+ if (hasHome) pagesToCopy.push('HomePage.tsx');
504
+ if (hasTemplate) pagesToCopy.push('TemplatePage.tsx');
505
+ if (hasAssistant) pagesToCopy.push('AssistantPage.tsx');
506
+
507
+ for (const pageFile of pagesToCopy) {
508
+ // HomePage and TemplatePage are generated (they vary by hasAssistant)
509
+ if (pageFile === 'HomePage.tsx') {
510
+ await fs.writeFile(
511
+ path.join(targetDir, 'src', 'pages', 'HomePage.tsx'),
512
+ generateHomePage(hasAssistant)
513
+ );
514
+ continue;
515
+ }
516
+ if (pageFile === 'TemplatePage.tsx') {
517
+ await fs.writeFile(
518
+ path.join(targetDir, 'src', 'pages', 'TemplatePage.tsx'),
519
+ generateTemplatePage(hasAssistant)
520
+ );
521
+ continue;
522
+ }
523
+ const src = path.join(templatesDir, 'src', 'pages', pageFile);
524
+ if (await fs.pathExists(src)) {
525
+ await fs.copy(src, path.join(targetDir, 'src', 'pages', pageFile));
526
+ }
527
+ }
528
+
529
+ // 9. Generate AuthGuard.tsx based on selected pages
530
+ const firstProtectedPath = hasHome ? '/home' : hasTemplate ? '/template' : '/login';
531
+ const authGuardContent = generateAuthGuard({
532
+ hasLogin,
533
+ hasHome,
534
+ hasTemplate,
535
+ hasAssistant,
536
+ firstProtectedPath,
537
+ });
538
+
539
+ await fs.writeFile(
540
+ path.join(targetDir, 'src', 'app', 'components', 'AuthGuard.tsx'),
541
+ authGuardContent
542
+ );
543
+
544
+ // 10. Generate theme tokens
545
+ const selectedTheme = colorThemes.find(t => t.id === response.theme) || colorThemes[0];
546
+ const tokensDir = path.join(targetDir, 'src', 'styles', 'xertica');
547
+ await fs.ensureDir(tokensDir);
548
+ await fs.copy(
549
+ path.join(templatesDir, 'src', 'styles', 'index.css'),
550
+ path.join(targetDir, 'src', 'styles', 'index.css')
551
+ );
552
+ await fs.writeFile(path.join(tokensDir, 'tokens.css'), generateTokensCss(selectedTheme));
553
+
554
+ spinner.succeed('Project initialized successfully!');
555
+
556
+ if (response.install) {
557
+ const installSpinner = ora('Installing dependencies...').start();
558
+ await execa('npm', ['install'], { cwd: targetDir });
559
+ installSpinner.succeed('Dependencies installed!');
560
+ }
561
+
562
+ console.log(chalk.green('\n✅ Done! Your Xertica UI project is ready.'));
563
+ console.log(chalk.cyan(`\n cd ${directory}`));
564
+ if (!response.install) {
565
+ console.log(chalk.cyan(' npm install'));
566
+ }
567
+ console.log(chalk.cyan(' npm run dev'));
568
+ console.log();
569
+ console.log(chalk.gray(' Components are imported from the xertica-ui package.'));
570
+ console.log(chalk.gray(' Customize the theme in src/styles/xertica/tokens.css'));
571
+ const langLabels = SUPPORTED_LANGUAGES.filter(l => selectedLanguages.includes(l.code))
572
+ .map(l => l.label)
573
+ .join(', ');
574
+ console.log(
575
+ chalk.gray(
576
+ ` Languages: ${langLabels}${selectedLanguages.length === 1 ? ' (monolingual — LanguageSelector hidden)' : ''}`
577
+ )
578
+ );
579
+ console.log(chalk.gray(' To add/remove languages later: npx xertica-ui update → Languages'));
580
+ console.log(
581
+ chalk.gray(` AI Assistant: ${hasAssistant ? 'included (/assistente)' : 'not included'}`)
582
+ );
583
+ if (!hasAssistant) {
584
+ console.log(chalk.gray(' To add the assistant later: npx xertica-ui update → Assistant'));
585
+ }
586
+ } catch (error) {
587
+ spinner.fail('Failed to initialize project');
588
+ console.error(error);
589
+ }
590
+ });
591
+
592
+ program
593
+ .command('update')
594
+ .alias('update-theme')
595
+ .description('Update theme or project files to the latest version')
596
+ .action(async () => {
597
+ const targetDir = process.cwd();
598
+
599
+ console.log(chalk.blue(`🔧 Xertica UI CLI ${chalk.dim(`v${pkgJson.version}`)}`));
600
+
601
+ const currentConfig = await readXerticaConfig(targetDir);
602
+
603
+ const { updateType } = await prompts({
604
+ type: 'select',
605
+ name: 'updateType',
606
+ message: 'What do you want to update?',
607
+ choices: [
608
+ {
609
+ title: 'Theme only',
610
+ description: 'Change the color tokens (tokens.css)',
611
+ value: 'theme',
612
+ },
613
+ {
614
+ title: 'Languages',
615
+ description: 'Add or remove supported languages (pt-BR, en, es, …)',
616
+ value: 'languages',
617
+ },
618
+ {
619
+ title: 'Assistant',
620
+ description: currentConfig?.hasAssistant
621
+ ? 'Remove the AI Assistant from your project'
622
+ : 'Add the AI Assistant to your project',
623
+ value: 'assistant',
624
+ },
625
+ {
626
+ title: 'Dark Mode',
627
+ description: currentConfig?.disableDarkMode
628
+ ? 'Enable dark mode support in your project'
629
+ : 'Disable dark mode support in your project',
630
+ value: 'darkmode',
631
+ },
632
+ {
633
+ title: 'Project files',
634
+ description: 'Update app shell, shared, features and pages to a specific version',
635
+ value: 'project',
636
+ },
637
+ ],
638
+ });
639
+
640
+ if (!updateType) return;
641
+
642
+ // ── Theme update ─────────────────────────────────────────────────────────
643
+ if (updateType === 'theme') {
644
+ const { theme } = await prompts({
645
+ type: 'select',
646
+ name: 'theme',
647
+ message: 'Select the new color theme:',
648
+ choices: colorThemes.map(t => ({
649
+ title: t.name,
650
+ description: t.description,
651
+ value: t.id,
652
+ })),
653
+ initial: 0,
654
+ });
655
+
656
+ if (!theme) return;
657
+
658
+ const spinner = ora('Updating theme...').start();
659
+ try {
660
+ const tokensPath = path.join(targetDir, 'src', 'styles', 'xertica', 'tokens.css');
661
+ const selectedTheme = colorThemes.find(t => t.id === theme);
662
+ if (selectedTheme) {
663
+ await fs.ensureDir(path.dirname(tokensPath));
664
+ await fs.writeFile(tokensPath, generateTokensCss(selectedTheme));
665
+ spinner.succeed(`Theme updated to "${selectedTheme.name}" successfully!`);
666
+ } else {
667
+ spinner.fail('Theme not found.');
668
+ }
669
+ } catch (error) {
670
+ spinner.fail('Failed to update theme');
671
+ console.error(error);
672
+ }
673
+ return;
674
+ }
675
+
676
+ // ── Languages update (add / remove) ──────────────────────────────────────
677
+ if (updateType === 'languages') {
678
+ // Resolve current selection — read from .languages.json if present,
679
+ // else inspect the locales/ folder to infer it (for projects scaffolded
680
+ // before this feature shipped).
681
+ const persistedCodes = await readLanguagesConfig(targetDir);
682
+ let currentCodes: string[] = persistedCodes ?? [];
683
+ if (currentCodes.length === 0) {
684
+ const localesDir = path.join(targetDir, 'src', 'locales');
685
+ if (await fs.pathExists(localesDir)) {
686
+ const entries = await fs.readdir(localesDir);
687
+ // Accept both the new folder layout (locales/<code>/) and the legacy
688
+ // flat layout (locales/<code>.json) when inferring the current set.
689
+ currentCodes = SUPPORTED_LANGUAGES.filter(
690
+ l => entries.includes(l.jsonFile) || entries.includes(`${l.jsonFile}.json`)
691
+ ).map(l => l.code);
692
+ }
693
+ if (currentCodes.length === 0) currentCodes = DEFAULT_SELECTION;
694
+ }
695
+
696
+ console.log(
697
+ chalk.cyan(
698
+ `\nCurrent languages: ${
699
+ SUPPORTED_LANGUAGES.filter(l => currentCodes.includes(l.code))
700
+ .map(l => l.label)
701
+ .join(', ') || '(none)'
702
+ }\n`
703
+ )
704
+ );
705
+
706
+ const { newCodes } = await prompts({
707
+ type: 'multiselect',
708
+ name: 'newCodes',
709
+ message: 'Select the languages this project should support:',
710
+ instructions: false,
711
+ hint: 'Press SPACE to toggle. At least 1 required. Single-language apps auto-hide the LanguageSelector.',
712
+ min: 1,
713
+ choices: SUPPORTED_LANGUAGES.map(l => ({
714
+ title: l.label,
715
+ value: l.code,
716
+ selected: currentCodes.includes(l.code),
717
+ })),
718
+ });
719
+
720
+ if (!Array.isArray(newCodes) || newCodes.length === 0) {
721
+ console.log(chalk.gray('Update cancelled.'));
722
+ return;
723
+ }
724
+
725
+ // Compute add/remove diff for the user-facing summary
726
+ const toAdd = newCodes.filter((c: string) => !currentCodes.includes(c));
727
+ const toRemove = currentCodes.filter(c => !newCodes.includes(c));
728
+
729
+ if (toAdd.length === 0 && toRemove.length === 0) {
730
+ console.log(chalk.gray('No changes — selection matches the current set.'));
731
+ return;
732
+ }
733
+
734
+ const summary: string[] = [];
735
+ if (toAdd.length > 0) summary.push(chalk.green(` + ${toAdd.join(', ')}`));
736
+ if (toRemove.length > 0) summary.push(chalk.red(` - ${toRemove.join(', ')}`));
737
+ console.log(`\n${summary.join('\n')}\n`);
738
+
739
+ const { confirmed } = await prompts({
740
+ type: 'confirm',
741
+ name: 'confirmed',
742
+ message: chalk.yellow(
743
+ `⚠️ This will regenerate src/app/App.tsx and src/i18n.ts (preserving language-only changes). Continue?`
744
+ ),
745
+ initial: true,
746
+ });
747
+ if (!confirmed) {
748
+ console.log(chalk.gray('Update cancelled.'));
749
+ return;
750
+ }
751
+
752
+ const spinner = ora('Updating languages...').start();
753
+ try {
754
+ // The freshly-installed library may not be present in this flow, so we
755
+ // read locale JSON sources from `node_modules/xertica-ui/templates`
756
+ // (installed when the project was created) — fallback to package
757
+ // directory lookup.
758
+ const installedTemplatesDir = path.join(
759
+ targetDir,
760
+ 'node_modules',
761
+ 'xertica-ui',
762
+ 'templates'
763
+ );
764
+ const templatesSourceDir = (await fs.pathExists(installedTemplatesDir))
765
+ ? installedTemplatesDir
766
+ : path.resolve(__dirname, '../templates');
767
+
768
+ // 1) Sync locale JSON files: copy newly-added, prune removed
769
+ const { copied, removed } = await syncLocaleFiles(templatesSourceDir, targetDir, newCodes, {
770
+ pruneOthers: true,
771
+ });
772
+
773
+ // 2) Regenerate i18n.ts so imports/resources reflect the new set
774
+ await fs.writeFile(path.join(targetDir, 'src', 'i18n.ts'), generateI18nFile(newCodes));
775
+
776
+ // 3) Regenerate App.tsx so the `availableLanguages` prop matches
777
+ await fs.writeFile(
778
+ path.join(targetDir, 'src', 'app', 'App.tsx'),
779
+ generateAppTsx(newCodes, currentConfig?.disableDarkMode ?? false)
780
+ );
781
+
782
+ // 4) Persist the new selection
783
+ await writeLanguagesConfig(targetDir, newCodes);
784
+
785
+ spinner.succeed('Languages updated successfully!');
786
+
787
+ if (copied.length > 0) console.log(chalk.green(` Copied: ${copied.join(', ')}`));
788
+ if (removed.length > 0) console.log(chalk.red(` Removed: ${removed.join(', ')}`));
789
+ if (newCodes.length === 1) {
790
+ console.log(
791
+ chalk.gray(` Project is now monolingual — the LanguageSelector will auto-hide.`)
792
+ );
793
+ }
794
+ } catch (error) {
795
+ spinner.fail('Failed to update languages');
796
+ console.error(error);
797
+ }
798
+ return;
799
+ }
800
+
801
+ // ── Assistant add / remove ────────────────────────────────────────────────
802
+ if (updateType === 'assistant') {
803
+ // Determine current state: prefer persisted config, fall back to file presence
804
+ // (handles projects scaffolded before .xertica.json existed).
805
+ let currentlyHas: boolean;
806
+ if (currentConfig !== null) {
807
+ currentlyHas = currentConfig.hasAssistant;
808
+ } else {
809
+ const assistantFeatureDir = path.join(targetDir, 'src', 'features', 'assistant');
810
+ const assistantPage = path.join(targetDir, 'src', 'pages', 'AssistantPage.tsx');
811
+ currentlyHas =
812
+ (await fs.pathExists(assistantFeatureDir)) || (await fs.pathExists(assistantPage));
813
+ // Persist the inferred state so future runs don't need to infer again
814
+ await writeXerticaConfig(targetDir, { hasAssistant: currentlyHas });
815
+ }
816
+
817
+ console.log(
818
+ chalk.cyan(
819
+ `\nAI Assistant is currently: ${currentlyHas ? chalk.green('enabled') : chalk.red('disabled')}\n`
820
+ )
821
+ );
822
+
823
+ const { action } = await prompts({
824
+ type: 'select',
825
+ name: 'action',
826
+ message: currentlyHas
827
+ ? 'Remove the AI Assistant from your project?'
828
+ : 'Add the AI Assistant to your project?',
829
+ choices: currentlyHas
830
+ ? [
831
+ {
832
+ title: 'Remove assistant',
833
+ description: 'Deletes AssistantPage and assistant feature files',
834
+ value: 'remove',
835
+ },
836
+ { title: 'Cancel', value: 'cancel' },
837
+ ]
838
+ : [
839
+ {
840
+ title: 'Add assistant',
841
+ description: 'Copies AssistantPage and assistant feature files',
842
+ value: 'add',
843
+ },
844
+ { title: 'Cancel', value: 'cancel' },
845
+ ],
846
+ });
847
+
848
+ if (!action || action === 'cancel') {
849
+ console.log(chalk.gray('Update cancelled.'));
850
+ return;
851
+ }
852
+
853
+ const { confirmed } = await prompts({
854
+ type: 'confirm',
855
+ name: 'confirmed',
856
+ message: chalk.yellow(
857
+ action === 'remove'
858
+ ? '⚠️ This will delete src/features/assistant/ and src/pages/AssistantPage.tsx and regenerate AuthGuard.tsx. Continue?'
859
+ : '⚠️ This will copy src/features/assistant/ and src/pages/AssistantPage.tsx and regenerate AuthGuard.tsx. Continue?'
860
+ ),
861
+ initial: action === 'add',
862
+ });
863
+
864
+ if (!confirmed) {
865
+ console.log(chalk.gray('Update cancelled.'));
866
+ return;
867
+ }
868
+
869
+ const spinner = ora(
870
+ action === 'add' ? 'Adding assistant...' : 'Removing assistant...'
871
+ ).start();
872
+
873
+ try {
874
+ const installedTemplatesDir = path.join(
875
+ targetDir,
876
+ 'node_modules',
877
+ 'xertica-ui',
878
+ 'templates'
879
+ );
880
+ const templatesSourceDir = (await fs.pathExists(installedTemplatesDir))
881
+ ? installedTemplatesDir
882
+ : path.resolve(__dirname, '../templates');
883
+
884
+ // Infer current page set from the pages directory
885
+ const pagesDir = path.join(targetDir, 'src', 'pages');
886
+ const existingPages = (await fs.pathExists(pagesDir)) ? await fs.readdir(pagesDir) : [];
887
+ const hasLogin = existingPages.includes('LoginPage.tsx');
888
+ const hasHome = existingPages.includes('HomePage.tsx');
889
+ const hasTemplate = existingPages.includes('TemplatePage.tsx');
890
+ const firstProtectedPath = hasHome ? '/home' : hasTemplate ? '/template' : '/login';
891
+
892
+ // Read persisted language selection so AuthGuard can be regenerated correctly
893
+ const persistedCodes = await readLanguagesConfig(targetDir);
894
+ const selectedCodes =
895
+ persistedCodes && persistedCodes.length > 0 ? persistedCodes : DEFAULT_SELECTION;
896
+
897
+ if (action === 'add') {
898
+ // Copy assistant feature
899
+ await fs.copy(
900
+ path.join(templatesSourceDir, 'src', 'features', 'assistant'),
901
+ path.join(targetDir, 'src', 'features', 'assistant'),
902
+ { overwrite: true }
903
+ );
904
+ // Copy AssistantPage
905
+ await fs.copy(
906
+ path.join(templatesSourceDir, 'src', 'pages', 'AssistantPage.tsx'),
907
+ path.join(targetDir, 'src', 'pages', 'AssistantPage.tsx'),
908
+ { overwrite: true }
909
+ );
910
+ } else {
911
+ // Remove assistant feature
912
+ await fs.remove(path.join(targetDir, 'src', 'features', 'assistant'));
913
+ await fs.remove(path.join(targetDir, 'src', 'pages', 'AssistantPage.tsx'));
914
+ }
915
+
916
+ // Regenerate pages and AuthGuard reflecting the new assistant state
917
+ const newHasAssistant = action === 'add';
918
+
919
+ if (hasHome) {
920
+ await fs.writeFile(
921
+ path.join(targetDir, 'src', 'pages', 'HomePage.tsx'),
922
+ generateHomePage(newHasAssistant)
923
+ );
924
+ }
925
+ if (hasTemplate) {
926
+ await fs.writeFile(
927
+ path.join(targetDir, 'src', 'pages', 'TemplatePage.tsx'),
928
+ generateTemplatePage(newHasAssistant)
929
+ );
930
+ }
931
+
932
+ await fs.writeFile(
933
+ path.join(targetDir, 'src', 'app', 'components', 'AuthGuard.tsx'),
934
+ generateAuthGuard({
935
+ hasLogin,
936
+ hasHome,
937
+ hasTemplate,
938
+ hasAssistant: newHasAssistant,
939
+ firstProtectedPath,
940
+ })
941
+ );
942
+
943
+ // Persist the updated flag
944
+ await writeXerticaConfig(targetDir, { hasAssistant: newHasAssistant });
945
+
946
+ spinner.succeed(
947
+ action === 'add'
948
+ ? 'AI Assistant added successfully!'
949
+ : 'AI Assistant removed successfully!'
950
+ );
951
+
952
+ if (action === 'add') {
953
+ console.log(chalk.gray('\n Route /assistente is now available.'));
954
+ console.log(chalk.gray(' Configure your Gemini API key in VITE_GEMINI_API_KEY.'));
955
+ } else {
956
+ console.log(chalk.gray('\n Assistant files removed and AuthGuard updated.'));
957
+ }
958
+ } catch (error) {
959
+ spinner.fail('Failed to update assistant');
960
+ console.error(error);
961
+ }
962
+ return;
963
+ }
964
+
965
+ // ── Dark Mode update (enable / disable) ──────────────────────────────────
966
+ if (updateType === 'darkmode') {
967
+ const currentlyDisabled = currentConfig?.disableDarkMode ?? false;
968
+ const { enableDarkMode } = await prompts({
969
+ type: 'confirm',
970
+ name: 'enableDarkMode',
971
+ message: currentlyDisabled
972
+ ? 'Enable dark mode support in your project?'
973
+ : 'Disable dark mode support in your project? (This will hide the toggle and force light mode)',
974
+ initial: !currentlyDisabled,
975
+ });
976
+
977
+ if (enableDarkMode === undefined) return;
978
+
979
+ const newDisableDarkMode = !enableDarkMode;
980
+
981
+ const spinner = ora(
982
+ newDisableDarkMode ? 'Disabling dark mode...' : 'Enabling dark mode...'
983
+ ).start();
984
+ try {
985
+ // Persist the selection
986
+ await writeXerticaConfig(targetDir, { disableDarkMode: newDisableDarkMode });
987
+
988
+ // Regenerate App.tsx with the new dark mode flag
989
+ const persistedCodes = await readLanguagesConfig(targetDir);
990
+ const selectedCodes =
991
+ persistedCodes && persistedCodes.length > 0 ? persistedCodes : DEFAULT_SELECTION;
992
+
993
+ await fs.writeFile(
994
+ path.join(targetDir, 'src', 'app', 'App.tsx'),
995
+ generateAppTsx(selectedCodes, newDisableDarkMode)
996
+ );
997
+
998
+ spinner.succeed(
999
+ newDisableDarkMode
1000
+ ? 'Dark mode disabled successfully! (Locked to Light Mode)'
1001
+ : 'Dark mode enabled successfully!'
1002
+ );
1003
+ } catch (error) {
1004
+ spinner.fail('Failed to update dark mode configuration');
1005
+ console.error(error);
1006
+ }
1007
+ return;
1008
+ }
1009
+
1010
+ // ── Project files update ──────────────────────────────────────────────────
1011
+ const { versionType } = await prompts({
1012
+ type: 'select',
1013
+ name: 'versionType',
1014
+ message: 'Which version do you want to update to?',
1015
+ choices: [
1016
+ { title: 'Latest', description: 'Install the latest published version', value: 'latest' },
1017
+ {
1018
+ title: 'Specific version',
1019
+ description: 'Enter a version number (e.g. 2.0.2)',
1020
+ value: 'specific',
1021
+ },
1022
+ ],
1023
+ });
1024
+
1025
+ if (!versionType) return;
1026
+
1027
+ let targetVersion = 'latest';
1028
+ if (versionType === 'specific') {
1029
+ const { version } = await prompts({
1030
+ type: 'text',
1031
+ name: 'version',
1032
+ message: 'Enter the version (e.g. 2.0.2):',
1033
+ validate: v =>
1034
+ /^\d+\.\d+\.\d+/.test(v.trim()) ? true : 'Enter a valid semver (e.g. 2.0.2)',
1035
+ });
1036
+ if (!version) return;
1037
+ targetVersion = version.trim();
1038
+ }
1039
+
1040
+ const { filesToUpdate } = await prompts({
1041
+ type: 'multiselect',
1042
+ name: 'filesToUpdate',
1043
+ message: 'Select which parts of the project to update:',
1044
+ choices: [
1045
+ {
1046
+ title: 'App shell (src/app/)',
1047
+ description: 'App.tsx, AppLayout.tsx',
1048
+ value: 'app',
1049
+ selected: true,
1050
+ },
1051
+ {
1052
+ title: 'Shared utilities (src/shared/)',
1053
+ description: 'auth.ts, navigation.ts, types',
1054
+ value: 'shared',
1055
+ selected: true,
1056
+ },
1057
+ {
1058
+ title: 'Features (src/features/)',
1059
+ description: 'auth, home, template UI components',
1060
+ value: 'features',
1061
+ selected: true,
1062
+ },
1063
+ {
1064
+ title: 'Pages (src/pages/)',
1065
+ description: 'Thin page wrapper components',
1066
+ value: 'pages',
1067
+ selected: true,
1068
+ },
1069
+ {
1070
+ title: 'Root config files',
1071
+ description: 'vite.config.ts, tsconfig.json, etc.',
1072
+ value: 'config',
1073
+ selected: false,
1074
+ },
1075
+ ],
1076
+ });
1077
+
1078
+ if (!filesToUpdate || filesToUpdate.length === 0) return;
1079
+
1080
+ const { confirmed } = await prompts({
1081
+ type: 'confirm',
1082
+ name: 'confirmed',
1083
+ message: chalk.yellow(
1084
+ `⚠️ This will overwrite the selected files. Local changes will be lost. Continue?`
1085
+ ),
1086
+ initial: false,
1087
+ });
1088
+
1089
+ if (!confirmed) {
1090
+ console.log(chalk.gray('Update cancelled.'));
1091
+ return;
1092
+ }
1093
+
1094
+ const spinner = ora(`Installing xertica-ui@${targetVersion}...`).start();
1095
+
1096
+ try {
1097
+ // Install the target version in the consumer project
1098
+ await execa('npm', ['install', `xertica-ui@${targetVersion}`], { cwd: targetDir });
1099
+ spinner.text = 'Copying updated files...';
1100
+
1101
+ // Templates now come from the freshly installed version
1102
+ const updatedTemplatesDir = path.join(targetDir, 'node_modules', 'xertica-ui', 'templates');
1103
+
1104
+ if (filesToUpdate.includes('app')) {
1105
+ // AppLayout is always safe to overwrite (no per-project config baked in)
1106
+ await fs.copy(
1107
+ path.join(updatedTemplatesDir, 'src', 'app', 'components', 'AppLayout.tsx'),
1108
+ path.join(targetDir, 'src', 'app', 'components', 'AppLayout.tsx'),
1109
+ { overwrite: true }
1110
+ );
1111
+
1112
+ // For App.tsx and i18n.ts we must preserve the user's language selection.
1113
+ // Read it from .languages.json (or fall back to inspecting locales/ for
1114
+ // projects scaffolded before the file existed).
1115
+ const persistedCodes = await readLanguagesConfig(targetDir);
1116
+ let selectedCodes: string[] = persistedCodes ?? [];
1117
+ if (selectedCodes.length === 0) {
1118
+ const localesDir = path.join(targetDir, 'src', 'locales');
1119
+ if (await fs.pathExists(localesDir)) {
1120
+ const entries = await fs.readdir(localesDir);
1121
+ // Accept both the new folder layout and the legacy flat layout
1122
+ selectedCodes = SUPPORTED_LANGUAGES.filter(
1123
+ l => entries.includes(l.jsonFile) || entries.includes(`${l.jsonFile}.json`)
1124
+ ).map(l => l.code);
1125
+ }
1126
+ if (selectedCodes.length === 0) selectedCodes = DEFAULT_SELECTION;
1127
+ // Persist the inferred selection so future updates have it cached
1128
+ await writeLanguagesConfig(targetDir, selectedCodes);
1129
+ }
1130
+
1131
+ const projectConfig = await readXerticaConfig(targetDir);
1132
+
1133
+ // Regenerate App.tsx and i18n.ts honoring the persisted language set
1134
+ await fs.writeFile(
1135
+ path.join(targetDir, 'src', 'app', 'App.tsx'),
1136
+ generateAppTsx(selectedCodes, projectConfig?.disableDarkMode ?? false)
1137
+ );
1138
+ await fs.writeFile(path.join(targetDir, 'src', 'i18n.ts'), generateI18nFile(selectedCodes));
1139
+
1140
+ // Refresh locale JSON files for the selected languages (keys grow over
1141
+ // library updates) — but prune any orphans from prior selections.
1142
+ await syncLocaleFiles(updatedTemplatesDir, targetDir, selectedCodes, {
1143
+ pruneOthers: true,
1144
+ });
1145
+
1146
+ // Regenerate AuthGuard preserving the current page set and assistant flag
1147
+ const pagesDir = path.join(targetDir, 'src', 'pages');
1148
+ const existingPages = (await fs.pathExists(pagesDir)) ? await fs.readdir(pagesDir) : [];
1149
+ const hasLoginP = existingPages.includes('LoginPage.tsx');
1150
+ const hasHomeP = existingPages.includes('HomePage.tsx');
1151
+ const hasTemplateP = existingPages.includes('TemplatePage.tsx');
1152
+ const hasAssistantP =
1153
+ projectConfig?.hasAssistant ?? existingPages.includes('AssistantPage.tsx');
1154
+ const firstProtectedP = hasHomeP ? '/home' : hasTemplateP ? '/template' : '/login';
1155
+ await fs.writeFile(
1156
+ path.join(targetDir, 'src', 'app', 'components', 'AuthGuard.tsx'),
1157
+ generateAuthGuard({
1158
+ hasLogin: hasLoginP,
1159
+ hasHome: hasHomeP,
1160
+ hasTemplate: hasTemplateP,
1161
+ hasAssistant: hasAssistantP,
1162
+ firstProtectedPath: firstProtectedP,
1163
+ })
1164
+ );
1165
+ }
1166
+
1167
+ if (filesToUpdate.includes('shared')) {
1168
+ await fs.copy(
1169
+ path.join(updatedTemplatesDir, 'src', 'shared'),
1170
+ path.join(targetDir, 'src', 'shared'),
1171
+ { overwrite: true }
1172
+ );
1173
+ }
1174
+
1175
+ if (filesToUpdate.includes('features')) {
1176
+ // Only update feature directories that already exist in the project.
1177
+ // 'assistant' is only present if it was included during init (or added via update → Assistant).
1178
+ for (const feature of ['auth', 'home', 'template', 'assistant']) {
1179
+ const destFeature = path.join(targetDir, 'src', 'features', feature);
1180
+ const srcFeature = path.join(updatedTemplatesDir, 'src', 'features', feature);
1181
+ if ((await fs.pathExists(destFeature)) && (await fs.pathExists(srcFeature))) {
1182
+ await fs.copy(srcFeature, destFeature, { overwrite: true });
1183
+ }
1184
+ }
1185
+ }
1186
+
1187
+ if (filesToUpdate.includes('pages')) {
1188
+ const pagesDir = path.join(targetDir, 'src', 'pages');
1189
+ const srcPagesDir = path.join(updatedTemplatesDir, 'src', 'pages');
1190
+ if ((await fs.pathExists(pagesDir)) && (await fs.pathExists(srcPagesDir))) {
1191
+ const projectCfg = await readXerticaConfig(targetDir);
1192
+ const existingPages = await fs.readdir(pagesDir);
1193
+ const assistantEnabled =
1194
+ projectCfg?.hasAssistant ?? existingPages.includes('AssistantPage.tsx');
1195
+
1196
+ for (const pageFile of existingPages) {
1197
+ // Generated pages: regenerate instead of copy so assistant imports stay correct
1198
+ if (pageFile === 'HomePage.tsx') {
1199
+ await fs.writeFile(
1200
+ path.join(pagesDir, 'HomePage.tsx'),
1201
+ generateHomePage(assistantEnabled)
1202
+ );
1203
+ continue;
1204
+ }
1205
+ if (pageFile === 'TemplatePage.tsx') {
1206
+ await fs.writeFile(
1207
+ path.join(pagesDir, 'TemplatePage.tsx'),
1208
+ generateTemplatePage(assistantEnabled)
1209
+ );
1210
+ continue;
1211
+ }
1212
+ const src = path.join(srcPagesDir, pageFile);
1213
+ if (await fs.pathExists(src)) {
1214
+ await fs.copy(src, path.join(pagesDir, pageFile), { overwrite: true });
1215
+ }
1216
+ }
1217
+ }
1218
+ }
1219
+
1220
+ if (filesToUpdate.includes('config')) {
1221
+ // Config files are inside the templates/ directory (same level as src/)
1222
+ const configFiles = [
1223
+ 'vite.config.ts',
1224
+ 'tsconfig.json',
1225
+ 'tsconfig.node.json',
1226
+ 'postcss.config.js',
1227
+ ];
1228
+ for (const file of configFiles) {
1229
+ const src = path.join(updatedTemplatesDir, file);
1230
+ if (await fs.pathExists(src)) {
1231
+ await fs.copy(src, path.join(targetDir, file), { overwrite: true });
1232
+ }
1233
+ }
1234
+ }
1235
+
1236
+ spinner.succeed(`Project updated to xertica-ui@${targetVersion} successfully!`);
1237
+ console.log(chalk.gray('\n Run npm run dev to start the development server.'));
1238
+ } catch (error) {
1239
+ spinner.fail('Failed to update project');
1240
+ console.error(error);
1241
+ }
1242
+ });
1243
+
1244
+ program.parse();