@julianoczkowski/create-trimble-app 2.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (450) hide show
  1. package/README.md +191 -0
  2. package/bin/create-trimble-app.js +9 -0
  3. package/package.json +67 -0
  4. package/src/cli.js +134 -0
  5. package/src/frameworks.js +28 -0
  6. package/src/scaffold.js +209 -0
  7. package/src/utils/file.js +47 -0
  8. package/src/utils/git.js +140 -0
  9. package/src/utils/install.js +25 -0
  10. package/src/utils/logger.js +124 -0
  11. package/templates/angular/.cursor/commands/remove-dev-content.md +394 -0
  12. package/templates/angular/.cursor/mcp.json +13 -0
  13. package/templates/angular/.cursor/rules/modus-angular-20.mdc +82 -0
  14. package/templates/angular/.cursor/rules/modus-angular-accordion-state-management-short.mdc +45 -0
  15. package/templates/angular/.cursor/rules/modus-angular-accordion-state-management.mdc +322 -0
  16. package/templates/angular/.cursor/rules/modus-angular-best-practices.mdc +472 -0
  17. package/templates/angular/.cursor/rules/modus-angular-border-usage-short.mdc +48 -0
  18. package/templates/angular/.cursor/rules/modus-angular-border-usage.mdc +286 -0
  19. package/templates/angular/.cursor/rules/modus-angular-button-group-usage-short.mdc +47 -0
  20. package/templates/angular/.cursor/rules/modus-angular-button-group-usage.mdc +263 -0
  21. package/templates/angular/.cursor/rules/modus-angular-checkbox-value-inversion-short.mdc +36 -0
  22. package/templates/angular/.cursor/rules/modus-angular-checkbox-value-inversion.mdc +92 -0
  23. package/templates/angular/.cursor/rules/modus-angular-chrome-devtools-testing-short.mdc +34 -0
  24. package/templates/angular/.cursor/rules/modus-angular-chrome-devtools-testing.mdc +185 -0
  25. package/templates/angular/.cursor/rules/modus-angular-color-usage-short.mdc +56 -0
  26. package/templates/angular/.cursor/rules/modus-angular-color-usage.mdc +208 -0
  27. package/templates/angular/.cursor/rules/modus-angular-components-reference.mdc +114 -0
  28. package/templates/angular/.cursor/rules/modus-angular-design-system.mdc +273 -0
  29. package/templates/angular/.cursor/rules/modus-angular-development-workflow-short.mdc +43 -0
  30. package/templates/angular/.cursor/rules/modus-angular-development-workflow.mdc +145 -0
  31. package/templates/angular/.cursor/rules/modus-angular-essentials.mdc +272 -0
  32. package/templates/angular/.cursor/rules/modus-angular-forms-validation-short.mdc +56 -0
  33. package/templates/angular/.cursor/rules/modus-angular-forms-validation.mdc +124 -0
  34. package/templates/angular/.cursor/rules/modus-angular-icon-names.mdc +70 -0
  35. package/templates/angular/.cursor/rules/modus-angular-icons-short.mdc +40 -0
  36. package/templates/angular/.cursor/rules/modus-angular-icons.mdc +137 -0
  37. package/templates/angular/.cursor/rules/modus-angular-implementation-guides-short.mdc +36 -0
  38. package/templates/angular/.cursor/rules/modus-angular-implementation-guides.mdc +301 -0
  39. package/templates/angular/.cursor/rules/modus-angular-integration-short.mdc +60 -0
  40. package/templates/angular/.cursor/rules/modus-angular-integration.mdc +1096 -0
  41. package/templates/angular/.cursor/rules/modus-angular-master.mdc +164 -0
  42. package/templates/angular/.cursor/rules/modus-angular-modal-usage-short.mdc +51 -0
  43. package/templates/angular/.cursor/rules/modus-angular-modal-usage.mdc +115 -0
  44. package/templates/angular/.cursor/rules/modus-angular-navbar-usage-short.mdc +49 -0
  45. package/templates/angular/.cursor/rules/modus-angular-navbar-usage.mdc +146 -0
  46. package/templates/angular/.cursor/rules/modus-angular-no-emojis.mdc +66 -0
  47. package/templates/angular/.cursor/rules/modus-angular-opacity-utilities-short.mdc +43 -0
  48. package/templates/angular/.cursor/rules/modus-angular-opacity-utilities.mdc +160 -0
  49. package/templates/angular/.cursor/rules/modus-angular-select-vs-dropdown-menu-short.mdc +51 -0
  50. package/templates/angular/.cursor/rules/modus-angular-select-vs-dropdown-menu.mdc +83 -0
  51. package/templates/angular/.cursor/rules/modus-angular-semantic-html-short.mdc +36 -0
  52. package/templates/angular/.cursor/rules/modus-angular-semantic-html.mdc +81 -0
  53. package/templates/angular/.cursor/rules/modus-angular-side-navigation-usage-short.mdc +50 -0
  54. package/templates/angular/.cursor/rules/modus-angular-side-navigation-usage.mdc +136 -0
  55. package/templates/angular/.cursor/rules/modus-angular-table-usage-short.mdc +52 -0
  56. package/templates/angular/.cursor/rules/modus-angular-table-usage.mdc +151 -0
  57. package/templates/angular/.cursor/rules/modus-angular-tailwind-usage-short.mdc +44 -0
  58. package/templates/angular/.cursor/rules/modus-angular-tailwind-usage.mdc +242 -0
  59. package/templates/angular/.cursor/rules/modus-angular-themes-short.mdc +54 -0
  60. package/templates/angular/.cursor/rules/modus-angular-themes.mdc +222 -0
  61. package/templates/angular/.cursor/rules/modus-angular-utility-panel-usage-short.mdc +52 -0
  62. package/templates/angular/.cursor/rules/modus-angular-utility-panel-usage.mdc +130 -0
  63. package/templates/angular/.cursor/rules/ux/gestalt-laws-detailed.mdc +514 -0
  64. package/templates/angular/.cursor/rules/ux/ux-ui-foundations.mdc +235 -0
  65. package/templates/angular/.cursor/skills/run-lint-checks/SKILL.md +169 -0
  66. package/templates/angular/.editorconfig +17 -0
  67. package/templates/angular/.github/dependabot.yml +19 -0
  68. package/templates/angular/.github/workflows/a11y-check.yml +135 -0
  69. package/templates/angular/.github/workflows/ci.yml +44 -0
  70. package/templates/angular/.husky/pre-commit +32 -0
  71. package/templates/angular/.postcssrc.json +5 -0
  72. package/templates/angular/.vscode/extensions.json +4 -0
  73. package/templates/angular/.vscode/launch.json +20 -0
  74. package/templates/angular/.vscode/tasks.json +42 -0
  75. package/templates/angular/CLAUDE.md +148 -0
  76. package/templates/angular/README.md +92 -0
  77. package/templates/angular/amplify.yml +25 -0
  78. package/templates/angular/angular.json +106 -0
  79. package/templates/angular/data/modusIcons.ts +861 -0
  80. package/templates/angular/package-lock.json +11030 -0
  81. package/templates/angular/package.json +66 -0
  82. package/templates/angular/public/angular-icon.svg +12 -0
  83. package/templates/angular/public/favicon.ico +0 -0
  84. package/templates/angular/public/modus-icons.css +49 -0
  85. package/templates/angular/public/vite.svg +1 -0
  86. package/templates/angular/scripts/README.md +410 -0
  87. package/templates/angular/scripts/check-border-violations.js +352 -0
  88. package/templates/angular/scripts/check-icon-names.js +402 -0
  89. package/templates/angular/scripts/check-inline-styles.js +292 -0
  90. package/templates/angular/scripts/check-modus-colors.js +282 -0
  91. package/templates/angular/scripts/check-modus-icons.js +263 -0
  92. package/templates/angular/scripts/check-opacity-utilities.js +426 -0
  93. package/templates/angular/scripts/check-semantic-html.js +452 -0
  94. package/templates/angular/scripts/check-typescript.js +109 -0
  95. package/templates/angular/src/app/app.config.ts +29 -0
  96. package/templates/angular/src/app/app.css +0 -0
  97. package/templates/angular/src/app/app.html +4 -0
  98. package/templates/angular/src/app/app.routes.ts +351 -0
  99. package/templates/angular/src/app/app.spec.ts +27 -0
  100. package/templates/angular/src/app/app.ts +47 -0
  101. package/templates/angular/src/app/components/README.md +77 -0
  102. package/templates/angular/src/app/components/index.ts +53 -0
  103. package/templates/angular/src/app/components/modus-accordion.component.ts +50 -0
  104. package/templates/angular/src/app/components/modus-alert.component.ts +133 -0
  105. package/templates/angular/src/app/components/modus-autocomplete.component.ts +262 -0
  106. package/templates/angular/src/app/components/modus-avatar.component.ts +75 -0
  107. package/templates/angular/src/app/components/modus-badge.component.ts +84 -0
  108. package/templates/angular/src/app/components/modus-breadcrumbs.component.ts +65 -0
  109. package/templates/angular/src/app/components/modus-button-group.component.ts +82 -0
  110. package/templates/angular/src/app/components/modus-button.component.ts +292 -0
  111. package/templates/angular/src/app/components/modus-card.component.ts +73 -0
  112. package/templates/angular/src/app/components/modus-checkbox.component.ts +117 -0
  113. package/templates/angular/src/app/components/modus-chip.component.ts +97 -0
  114. package/templates/angular/src/app/components/modus-collapse.component.ts +118 -0
  115. package/templates/angular/src/app/components/modus-date.component.ts +165 -0
  116. package/templates/angular/src/app/components/modus-dropdown-menu.component.ts +108 -0
  117. package/templates/angular/src/app/components/modus-file-dropzone.component.ts +121 -0
  118. package/templates/angular/src/app/components/modus-handle.component.ts +96 -0
  119. package/templates/angular/src/app/components/modus-icon.component.ts +81 -0
  120. package/templates/angular/src/app/components/modus-input-feedback.component.ts +54 -0
  121. package/templates/angular/src/app/components/modus-input-label.component.ts +62 -0
  122. package/templates/angular/src/app/components/modus-loader.component.ts +48 -0
  123. package/templates/angular/src/app/components/modus-logo.component.ts +115 -0
  124. package/templates/angular/src/app/components/modus-menu-item.component.ts +116 -0
  125. package/templates/angular/src/app/components/modus-menu.component.ts +58 -0
  126. package/templates/angular/src/app/components/modus-modal.component.ts +70 -0
  127. package/templates/angular/src/app/components/modus-navbar.component.ts +303 -0
  128. package/templates/angular/src/app/components/modus-number-input.component.ts +174 -0
  129. package/templates/angular/src/app/components/modus-pagination.component.ts +74 -0
  130. package/templates/angular/src/app/components/modus-panel.component.ts +61 -0
  131. package/templates/angular/src/app/components/modus-progress.component.ts +62 -0
  132. package/templates/angular/src/app/components/modus-radio.component.ts +102 -0
  133. package/templates/angular/src/app/components/modus-rating.component.ts +80 -0
  134. package/templates/angular/src/app/components/modus-select.component.ts +131 -0
  135. package/templates/angular/src/app/components/modus-side-navigation.component.ts +90 -0
  136. package/templates/angular/src/app/components/modus-skeleton.component.ts +54 -0
  137. package/templates/angular/src/app/components/modus-slider.component.ts +132 -0
  138. package/templates/angular/src/app/components/modus-stepper.component.ts +65 -0
  139. package/templates/angular/src/app/components/modus-switch.component.ts +118 -0
  140. package/templates/angular/src/app/components/modus-table.component.ts +204 -0
  141. package/templates/angular/src/app/components/modus-tabs.component.ts +82 -0
  142. package/templates/angular/src/app/components/modus-text-input.component.ts +221 -0
  143. package/templates/angular/src/app/components/modus-textarea.component.ts +168 -0
  144. package/templates/angular/src/app/components/modus-theme-switcher.component.ts +45 -0
  145. package/templates/angular/src/app/components/modus-time-input.component.ts +172 -0
  146. package/templates/angular/src/app/components/modus-toast.component.ts +63 -0
  147. package/templates/angular/src/app/components/modus-toolbar.component.ts +43 -0
  148. package/templates/angular/src/app/components/modus-tooltip.component.ts +83 -0
  149. package/templates/angular/src/app/components/modus-typography.component.ts +79 -0
  150. package/templates/angular/src/app/components/modus-utility-panel.component.ts +275 -0
  151. package/templates/angular/src/app/components/theme-demo.component.ts +1242 -0
  152. package/templates/angular/src/app/data/component-demos.ts +360 -0
  153. package/templates/angular/src/app/data/modus-icons.ts +806 -0
  154. package/templates/angular/src/app/demos/accordion/accordion-demo.component.ts +212 -0
  155. package/templates/angular/src/app/demos/alert/alert-demo.component.ts +108 -0
  156. package/templates/angular/src/app/demos/autocomplete/autocomplete-demo.component.ts +174 -0
  157. package/templates/angular/src/app/demos/avatar/avatar-demo.component.ts +149 -0
  158. package/templates/angular/src/app/demos/badge/badge-demo.component.ts +148 -0
  159. package/templates/angular/src/app/demos/breadcrumbs/breadcrumbs-demo.component.ts +96 -0
  160. package/templates/angular/src/app/demos/button/button-demo.component.ts +256 -0
  161. package/templates/angular/src/app/demos/button-group/button-group-demo.component.ts +215 -0
  162. package/templates/angular/src/app/demos/card/card-demo.component.ts +180 -0
  163. package/templates/angular/src/app/demos/checkbox/checkbox-demo.component.ts +71 -0
  164. package/templates/angular/src/app/demos/chip/chip-demo.component.ts +183 -0
  165. package/templates/angular/src/app/demos/date/date-demo.component.ts +193 -0
  166. package/templates/angular/src/app/demos/dropdown/dropdown-demo.component.ts +196 -0
  167. package/templates/angular/src/app/demos/file-dropzone/file-dropzone-demo.component.ts +176 -0
  168. package/templates/angular/src/app/demos/handle/handle-demo.component.ts +265 -0
  169. package/templates/angular/src/app/demos/icon/icon-demo.component.ts +65 -0
  170. package/templates/angular/src/app/demos/input-feedback/input-feedback-demo.component.ts +189 -0
  171. package/templates/angular/src/app/demos/input-label/input-label-demo.component.ts +330 -0
  172. package/templates/angular/src/app/demos/loader/loader-demo.component.ts +143 -0
  173. package/templates/angular/src/app/demos/logo/logo-demo.component.ts +191 -0
  174. package/templates/angular/src/app/demos/menu/menu-demo.component.ts +72 -0
  175. package/templates/angular/src/app/demos/modal/modal-demo.component.ts +278 -0
  176. package/templates/angular/src/app/demos/navbar/navbar-demo.component.ts +135 -0
  177. package/templates/angular/src/app/demos/number-input/number-input-demo.component.ts +175 -0
  178. package/templates/angular/src/app/demos/pagination/pagination-demo.component.ts +134 -0
  179. package/templates/angular/src/app/demos/panel/panel-demo.component.ts +235 -0
  180. package/templates/angular/src/app/demos/progress/progress-demo.component.ts +169 -0
  181. package/templates/angular/src/app/demos/radio/radio-demo.component.ts +161 -0
  182. package/templates/angular/src/app/demos/rating/rating-demo.component.ts +97 -0
  183. package/templates/angular/src/app/demos/select/select-demo.component.ts +107 -0
  184. package/templates/angular/src/app/demos/shared/demo-example-clean.component.ts +41 -0
  185. package/templates/angular/src/app/demos/shared/demo-example.component.ts +42 -0
  186. package/templates/angular/src/app/demos/shared/demo-page.component.ts +97 -0
  187. package/templates/angular/src/app/demos/shared/index.ts +3 -0
  188. package/templates/angular/src/app/demos/side-navigation/side-navigation-demo.component.ts +524 -0
  189. package/templates/angular/src/app/demos/skeleton/skeleton-demo.component.ts +112 -0
  190. package/templates/angular/src/app/demos/slider/slider-demo.component.ts +76 -0
  191. package/templates/angular/src/app/demos/stepper/stepper-demo.component.ts +79 -0
  192. package/templates/angular/src/app/demos/switch/switch-demo.component.ts +113 -0
  193. package/templates/angular/src/app/demos/table/table-demo.component.ts +405 -0
  194. package/templates/angular/src/app/demos/tabs/tabs-demo.component.ts +318 -0
  195. package/templates/angular/src/app/demos/text-input/text-input-demo.component.ts +160 -0
  196. package/templates/angular/src/app/demos/textarea/textarea-demo.component.ts +95 -0
  197. package/templates/angular/src/app/demos/theme-switcher/theme-switcher-demo.component.ts +38 -0
  198. package/templates/angular/src/app/demos/time-input/time-input-demo.component.ts +130 -0
  199. package/templates/angular/src/app/demos/toast/toast-demo.component.ts +258 -0
  200. package/templates/angular/src/app/demos/toolbar/toolbar-demo.component.ts +54 -0
  201. package/templates/angular/src/app/demos/tooltip/tooltip-demo.component.ts +163 -0
  202. package/templates/angular/src/app/demos/utility-panel/utility-panel-demo.component.ts +197 -0
  203. package/templates/angular/src/app/dev/dev-config.ts +119 -0
  204. package/templates/angular/src/app/dev/dev-panel/dev-panel.component.ts +215 -0
  205. package/templates/angular/src/app/dev/dev-panel.service.ts +63 -0
  206. package/templates/angular/src/app/dev/index.ts +8 -0
  207. package/templates/angular/src/app/dev/theme-switcher-dropdown/theme-switcher-dropdown.component.ts +134 -0
  208. package/templates/angular/src/app/dev-pages/color-palette/color-palette.component.ts +229 -0
  209. package/templates/angular/src/app/dev-pages/components-gallery/components-gallery.component.ts +486 -0
  210. package/templates/angular/src/app/dev-pages/icons/icons.component.ts +149 -0
  211. package/templates/angular/src/app/pages/home/home.component.ts +251 -0
  212. package/templates/angular/src/app/services/README.md +57 -0
  213. package/templates/angular/src/app/services/theme.service.ts +163 -0
  214. package/templates/angular/src/environments/environment.development.ts +8 -0
  215. package/templates/angular/src/environments/environment.ts +8 -0
  216. package/templates/angular/src/index.html +14 -0
  217. package/templates/angular/src/main.ts +6 -0
  218. package/templates/angular/src/styles.css +1328 -0
  219. package/templates/angular/tsconfig.app.json +15 -0
  220. package/templates/angular/tsconfig.json +35 -0
  221. package/templates/angular/tsconfig.spec.json +14 -0
  222. package/templates/config.json +23 -0
  223. package/templates/react/.cursor/commands/remove-dev-content.md +311 -0
  224. package/templates/react/.cursor/mcp.json +13 -0
  225. package/templates/react/.cursor/rules/README.md +240 -0
  226. package/templates/react/.cursor/rules/border-usage-guidelines-short.mdc +22 -0
  227. package/templates/react/.cursor/rules/border-usage-guidelines.mdc +380 -0
  228. package/templates/react/.cursor/rules/chrome-devtools-testing-react-short.mdc +23 -0
  229. package/templates/react/.cursor/rules/development-workflow-react-short.mdc +23 -0
  230. package/templates/react/.cursor/rules/development-workflow-react.mdc +292 -0
  231. package/templates/react/.cursor/rules/implementation-guides-react-short.mdc +23 -0
  232. package/templates/react/.cursor/rules/implementation-guides-react.mdc +446 -0
  233. package/templates/react/.cursor/rules/modus-accordion-state-management-react-short.mdc +23 -0
  234. package/templates/react/.cursor/rules/modus-accordion-state-management-react.mdc +445 -0
  235. package/templates/react/.cursor/rules/modus-button-group-usage-react-short.mdc +23 -0
  236. package/templates/react/.cursor/rules/modus-button-group-usage-react.mdc +117 -0
  237. package/templates/react/.cursor/rules/modus-checkbox-value-inversion-react-short.mdc +23 -0
  238. package/templates/react/.cursor/rules/modus-checkbox-value-inversion-react.mdc +492 -0
  239. package/templates/react/.cursor/rules/modus-color-usage-react-short.mdc +23 -0
  240. package/templates/react/.cursor/rules/modus-color-usage-react.mdc +420 -0
  241. package/templates/react/.cursor/rules/modus-components-reference.mdc +366 -0
  242. package/templates/react/.cursor/rules/modus-icon-names.mdc +63 -0
  243. package/templates/react/.cursor/rules/modus-icons-react-short.mdc +24 -0
  244. package/templates/react/.cursor/rules/modus-icons-react.mdc +402 -0
  245. package/templates/react/.cursor/rules/modus-modal-implementation-react-short.mdc +23 -0
  246. package/templates/react/.cursor/rules/modus-modal-implementation-react.mdc +831 -0
  247. package/templates/react/.cursor/rules/modus-navbar-side-navigation-react-short.mdc +23 -0
  248. package/templates/react/.cursor/rules/modus-navbar-side-navigation-react.mdc +247 -0
  249. package/templates/react/.cursor/rules/modus-no-emojis-react-short.mdc +23 -0
  250. package/templates/react/.cursor/rules/modus-opacity-utilities-react-short.mdc +70 -0
  251. package/templates/react/.cursor/rules/modus-opacity-utilities-react.mdc +208 -0
  252. package/templates/react/.cursor/rules/modus-react-best-practices-short.mdc +23 -0
  253. package/templates/react/.cursor/rules/modus-react-best-practices.mdc +508 -0
  254. package/templates/react/.cursor/rules/modus-react-essentials.mdc +209 -0
  255. package/templates/react/.cursor/rules/modus-react-integration-short.mdc +23 -0
  256. package/templates/react/.cursor/rules/modus-react-integration.mdc +509 -0
  257. package/templates/react/.cursor/rules/modus-react-key-warnings-short.mdc +23 -0
  258. package/templates/react/.cursor/rules/modus-react-key-warnings.mdc +805 -0
  259. package/templates/react/.cursor/rules/modus-react-master.mdc +160 -0
  260. package/templates/react/.cursor/rules/modus-select-vs-dropdown-menu-react-short.mdc +23 -0
  261. package/templates/react/.cursor/rules/modus-select-vs-dropdown-menu-react.mdc +442 -0
  262. package/templates/react/.cursor/rules/modus-semantic-html-react-short.mdc +23 -0
  263. package/templates/react/.cursor/rules/modus-semantic-html-react.mdc +427 -0
  264. package/templates/react/.cursor/rules/modus-tailwind-usage-react-short.mdc +23 -0
  265. package/templates/react/.cursor/rules/modus-tailwind-usage-react.mdc +642 -0
  266. package/templates/react/.cursor/rules/modus-themes-react-short.mdc +23 -0
  267. package/templates/react/.cursor/rules/modus-themes-react.mdc +506 -0
  268. package/templates/react/.cursor/rules/ux/gestalt-laws-detailed.mdc +456 -0
  269. package/templates/react/.cursor/rules/ux/ux-ui-foundations.mdc +211 -0
  270. package/templates/react/.cursor/skills/create-modus-form-component/SKILL.md +518 -0
  271. package/templates/react/.cursor/skills/create-modus-wrapper-component/SKILL.md +252 -0
  272. package/templates/react/.cursor/skills/fix-modus-component-event-issues/SKILL.md +345 -0
  273. package/templates/react/.cursor/skills/handle-modus-checkbox-value-bug/SKILL.md +202 -0
  274. package/templates/react/.cursor/skills/implement-modus-modal-with-refs/SKILL.md +386 -0
  275. package/templates/react/.cursor/skills/integrate-modus-icons/SKILL.md +300 -0
  276. package/templates/react/.cursor/skills/run-lint-checks/SKILL.md +235 -0
  277. package/templates/react/.cursor/skills/set-up-modus-event-listeners/SKILL.md +284 -0
  278. package/templates/react/.cursor/skills/style-modus-components-with-tailwind/SKILL.md +382 -0
  279. package/templates/react/.env.development +3 -0
  280. package/templates/react/.env.production +3 -0
  281. package/templates/react/.github/CODEOWNERS +28 -0
  282. package/templates/react/.github/ISSUE_TEMPLATE/bug_report.yml +176 -0
  283. package/templates/react/.github/ISSUE_TEMPLATE/config.yml +20 -0
  284. package/templates/react/.github/ISSUE_TEMPLATE/documentation.yml +115 -0
  285. package/templates/react/.github/ISSUE_TEMPLATE/feature_request.yml +171 -0
  286. package/templates/react/.github/ISSUE_TEMPLATE/question.yml +139 -0
  287. package/templates/react/.github/copilot-instructions.md +80 -0
  288. package/templates/react/.github/instructions/components.instructions.md +82 -0
  289. package/templates/react/.github/instructions/demos.instructions.md +82 -0
  290. package/templates/react/.github/instructions/pages.instructions.md +76 -0
  291. package/templates/react/.github/instructions/styles.instructions.md +77 -0
  292. package/templates/react/.github/instructions/typescript.instructions.md +101 -0
  293. package/templates/react/.github/pull_request_template.md +188 -0
  294. package/templates/react/.github/workflows/ci.yml +43 -0
  295. package/templates/react/.github/workflows/claude-code-review.yml +44 -0
  296. package/templates/react/.github/workflows/claude.yml +50 -0
  297. package/templates/react/.husky/pre-commit +28 -0
  298. package/templates/react/.vscode/extensions.json +8 -0
  299. package/templates/react/CLAUDE.md +119 -0
  300. package/templates/react/CODE_OF_CONDUCT.md +79 -0
  301. package/templates/react/CONTRIBUTING.md +65 -0
  302. package/templates/react/LICENSE +21 -0
  303. package/templates/react/README.md +728 -0
  304. package/templates/react/SECURITY.md +50 -0
  305. package/templates/react/eslint.config.js +23 -0
  306. package/templates/react/index.html +13 -0
  307. package/templates/react/package-lock.json +5209 -0
  308. package/templates/react/package.json +49 -0
  309. package/templates/react/postcss.config.js +6 -0
  310. package/templates/react/public/react.svg +1 -0
  311. package/templates/react/public/vite.svg +1 -0
  312. package/templates/react/readme_assets/getting_started_header.png +0 -0
  313. package/templates/react/readme_assets/hero.png +0 -0
  314. package/templates/react/readme_assets/modus_comp.png +0 -0
  315. package/templates/react/readme_assets/modus_figma_mcp.png +0 -0
  316. package/templates/react/readme_assets/teaser_comp.gif +0 -0
  317. package/templates/react/scripts/README.md +343 -0
  318. package/templates/react/scripts/check-border-violations.js +483 -0
  319. package/templates/react/scripts/check-icon-names.js +486 -0
  320. package/templates/react/scripts/check-inline-styles.js +364 -0
  321. package/templates/react/scripts/check-modus-colors.js +247 -0
  322. package/templates/react/scripts/check-modus-icons.js +256 -0
  323. package/templates/react/scripts/check-opacity-utilities.js +481 -0
  324. package/templates/react/scripts/check-semantic-html.js +476 -0
  325. package/templates/react/scripts/check-typescript.js +109 -0
  326. package/templates/react/src/App.css +42 -0
  327. package/templates/react/src/App.tsx +54 -0
  328. package/templates/react/src/assets/react.svg +1 -0
  329. package/templates/react/src/components/DemoExample.tsx +61 -0
  330. package/templates/react/src/components/DemoPage.tsx +81 -0
  331. package/templates/react/src/components/ModusAccordion.tsx +89 -0
  332. package/templates/react/src/components/ModusAlert.tsx +85 -0
  333. package/templates/react/src/components/ModusAutocomplete.tsx +207 -0
  334. package/templates/react/src/components/ModusAvatar.tsx +50 -0
  335. package/templates/react/src/components/ModusBadge.tsx +82 -0
  336. package/templates/react/src/components/ModusBreadcrumbs.tsx +75 -0
  337. package/templates/react/src/components/ModusButton.tsx +244 -0
  338. package/templates/react/src/components/ModusButtonGroup.tsx +91 -0
  339. package/templates/react/src/components/ModusCard.tsx +70 -0
  340. package/templates/react/src/components/ModusCheckbox.tsx +168 -0
  341. package/templates/react/src/components/ModusChip.tsx +93 -0
  342. package/templates/react/src/components/ModusDate.tsx +154 -0
  343. package/templates/react/src/components/ModusDropdownMenu.tsx +148 -0
  344. package/templates/react/src/components/ModusFileDropzone.tsx +140 -0
  345. package/templates/react/src/components/ModusHandle.tsx +101 -0
  346. package/templates/react/src/components/ModusIcon.tsx +49 -0
  347. package/templates/react/src/components/ModusInputFeedback.tsx +52 -0
  348. package/templates/react/src/components/ModusInputLabel.tsx +50 -0
  349. package/templates/react/src/components/ModusLoader.tsx +42 -0
  350. package/templates/react/src/components/ModusLogo.tsx +102 -0
  351. package/templates/react/src/components/ModusMenu.tsx +119 -0
  352. package/templates/react/src/components/ModusMenuItem.tsx +86 -0
  353. package/templates/react/src/components/ModusModal.tsx +145 -0
  354. package/templates/react/src/components/ModusNavbar.tsx +504 -0
  355. package/templates/react/src/components/ModusNumberInput.tsx +230 -0
  356. package/templates/react/src/components/ModusPagination.tsx +94 -0
  357. package/templates/react/src/components/ModusPanel.tsx +92 -0
  358. package/templates/react/src/components/ModusProgress.tsx +70 -0
  359. package/templates/react/src/components/ModusProvider.tsx +18 -0
  360. package/templates/react/src/components/ModusRadio.tsx +114 -0
  361. package/templates/react/src/components/ModusRating.tsx +108 -0
  362. package/templates/react/src/components/ModusSelect.tsx +171 -0
  363. package/templates/react/src/components/ModusSideNavigation.tsx +149 -0
  364. package/templates/react/src/components/ModusSkeleton.tsx +42 -0
  365. package/templates/react/src/components/ModusSlider.tsx +128 -0
  366. package/templates/react/src/components/ModusStepper.tsx +85 -0
  367. package/templates/react/src/components/ModusSwitch.tsx +130 -0
  368. package/templates/react/src/components/ModusTable.tsx +309 -0
  369. package/templates/react/src/components/ModusTabs.tsx +114 -0
  370. package/templates/react/src/components/ModusTextInput.tsx +179 -0
  371. package/templates/react/src/components/ModusTextarea.tsx +164 -0
  372. package/templates/react/src/components/ModusThemeSwitcher.tsx +58 -0
  373. package/templates/react/src/components/ModusTimeInput.tsx +176 -0
  374. package/templates/react/src/components/ModusToast.tsx +207 -0
  375. package/templates/react/src/components/ModusToolbar.tsx +70 -0
  376. package/templates/react/src/components/ModusTooltip.tsx +97 -0
  377. package/templates/react/src/components/ModusUtilityPanel.tsx +198 -0
  378. package/templates/react/src/components/ThemeSwitcherDropdown.tsx +117 -0
  379. package/templates/react/src/components/ThemeToggleSimple.tsx +157 -0
  380. package/templates/react/src/config/routes.ts +196 -0
  381. package/templates/react/src/contexts/ThemeContext.tsx +81 -0
  382. package/templates/react/src/contexts/ThemeContextData.tsx +89 -0
  383. package/templates/react/src/data/modusIcons.ts +865 -0
  384. package/templates/react/src/demos/accordion-demo/page.tsx +236 -0
  385. package/templates/react/src/demos/alert-demo/page.tsx +94 -0
  386. package/templates/react/src/demos/autocomplete-demo/page.tsx +166 -0
  387. package/templates/react/src/demos/avatar-demo/page.tsx +135 -0
  388. package/templates/react/src/demos/badge-demo/page.tsx +174 -0
  389. package/templates/react/src/demos/breadcrumbs-demo/page.tsx +88 -0
  390. package/templates/react/src/demos/button-demo/page.tsx +261 -0
  391. package/templates/react/src/demos/button-group-demo/page.tsx +231 -0
  392. package/templates/react/src/demos/card-demo/page.tsx +241 -0
  393. package/templates/react/src/demos/checkbox-demo/page.tsx +79 -0
  394. package/templates/react/src/demos/chip-demo/page.tsx +197 -0
  395. package/templates/react/src/demos/date-demo/page.tsx +179 -0
  396. package/templates/react/src/demos/dropdown-demo/page.tsx +150 -0
  397. package/templates/react/src/demos/file-dropzone-demo/page.tsx +186 -0
  398. package/templates/react/src/demos/handle-demo/page.tsx +313 -0
  399. package/templates/react/src/demos/icon-demo/page.tsx +72 -0
  400. package/templates/react/src/demos/input-feedback-demo/page.tsx +202 -0
  401. package/templates/react/src/demos/input-label-demo/page.tsx +392 -0
  402. package/templates/react/src/demos/loader-demo/page.tsx +138 -0
  403. package/templates/react/src/demos/logo-demo/page.tsx +292 -0
  404. package/templates/react/src/demos/menu-demo/page.tsx +70 -0
  405. package/templates/react/src/demos/modal-demo/page.tsx +332 -0
  406. package/templates/react/src/demos/navbar-demo/page.tsx +141 -0
  407. package/templates/react/src/demos/number-input-demo/page.tsx +180 -0
  408. package/templates/react/src/demos/pagination-demo/page.tsx +147 -0
  409. package/templates/react/src/demos/panel-demo/page.tsx +376 -0
  410. package/templates/react/src/demos/progress-demo/page.tsx +185 -0
  411. package/templates/react/src/demos/radio-demo/page.tsx +242 -0
  412. package/templates/react/src/demos/rating-demo/page.tsx +97 -0
  413. package/templates/react/src/demos/select-demo/page.tsx +111 -0
  414. package/templates/react/src/demos/side-navigation-demo/page.tsx +775 -0
  415. package/templates/react/src/demos/skeleton-demo/page.tsx +107 -0
  416. package/templates/react/src/demos/slider-demo/page.tsx +78 -0
  417. package/templates/react/src/demos/stepper-demo/page.tsx +86 -0
  418. package/templates/react/src/demos/switch-demo/page.tsx +146 -0
  419. package/templates/react/src/demos/table-demo/page.tsx +489 -0
  420. package/templates/react/src/demos/tabs-demo/page.tsx +187 -0
  421. package/templates/react/src/demos/text-input-demo/page.tsx +151 -0
  422. package/templates/react/src/demos/textarea-demo/page.tsx +73 -0
  423. package/templates/react/src/demos/theme-switcher-demo/page.tsx +26 -0
  424. package/templates/react/src/demos/time-input-demo/page.tsx +148 -0
  425. package/templates/react/src/demos/toast-demo/page.tsx +302 -0
  426. package/templates/react/src/demos/toolbar-demo/page.tsx +49 -0
  427. package/templates/react/src/demos/tooltip-demo/page.tsx +209 -0
  428. package/templates/react/src/demos/typography-test/page.tsx +28 -0
  429. package/templates/react/src/demos/utility-panel-demo/page.tsx +197 -0
  430. package/templates/react/src/dev/DevPanel.tsx +219 -0
  431. package/templates/react/src/dev/DevPanelContext.ts +14 -0
  432. package/templates/react/src/dev/DevPanelProvider.tsx +63 -0
  433. package/templates/react/src/dev/DevRoutes.tsx +98 -0
  434. package/templates/react/src/dev/config.ts +127 -0
  435. package/templates/react/src/dev/index.ts +8 -0
  436. package/templates/react/src/dev/useDevPanel.ts +17 -0
  437. package/templates/react/src/dev-pages/ColorPalettePage.tsx +347 -0
  438. package/templates/react/src/dev-pages/ComponentsGalleryPage.tsx +489 -0
  439. package/templates/react/src/dev-pages/IconsPage.tsx +137 -0
  440. package/templates/react/src/dev-pages/index.ts +3 -0
  441. package/templates/react/src/hooks/useTheme.ts +15 -0
  442. package/templates/react/src/index.css +635 -0
  443. package/templates/react/src/main.tsx +14 -0
  444. package/templates/react/src/pages/HomePage.tsx +283 -0
  445. package/templates/react/src/vite-env.d.ts +9 -0
  446. package/templates/react/tailwind.config.js +58 -0
  447. package/templates/react/tsconfig.app.json +27 -0
  448. package/templates/react/tsconfig.json +7 -0
  449. package/templates/react/tsconfig.node.json +25 -0
  450. package/templates/react/vite.config.ts +18 -0
@@ -0,0 +1,197 @@
1
+ import { Component, signal } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { DemoPageComponent } from '../shared/demo-page.component';
4
+ import { DemoExampleComponent } from '../shared/demo-example.component';
5
+ import { ModusButtonComponent } from '../../components/modus-button.component';
6
+ import { ModusTextInputComponent } from '../../components/modus-text-input.component';
7
+ import { ModusSwitchComponent } from '../../components/modus-switch.component';
8
+ import { ModusUtilityPanelComponent } from '../../components/modus-utility-panel.component';
9
+
10
+ interface UtilityPanelFormState {
11
+ projectName: string;
12
+ email: string;
13
+ notifications: boolean;
14
+ autoSave: boolean;
15
+ }
16
+
17
+ /**
18
+ * Demo page showcasing the Modus Utility Panel component.
19
+ */
20
+ @Component({
21
+ selector: 'app-utility-panel-demo-page',
22
+ standalone: true,
23
+ imports: [
24
+ CommonModule,
25
+ DemoPageComponent,
26
+ DemoExampleComponent,
27
+ ModusButtonComponent,
28
+ ModusTextInputComponent,
29
+ ModusSwitchComponent,
30
+ ModusUtilityPanelComponent,
31
+ ],
32
+ template: `
33
+ <demo-page
34
+ title="Modus Utility Panel"
35
+ description="Utility panels slide in contextual information or controls without leaving the page. Keep content focused and provide clear actions."
36
+ >
37
+ <demo-example
38
+ title="Settings Panel with Form Controls"
39
+ description="A utility panel containing form inputs and controls. The panel can be toggled to show or hide additional settings without leaving the main workspace."
40
+ >
41
+ <div class="space-y-6">
42
+ <!-- Toggle controls section -->
43
+ <div class="flex gap-4 items-center">
44
+ <modus-button
45
+ color="primary"
46
+ size="sm"
47
+ (buttonClick)="togglePanel()"
48
+ >
49
+ {{ expanded() ? 'Close' : 'Open' }} Settings Panel
50
+ </modus-button>
51
+ <div class="text-sm text-muted-foreground">
52
+ Panel state: {{ expanded() ? 'Open' : 'Closed' }}
53
+ </div>
54
+ </div>
55
+
56
+ <!-- Main content area -->
57
+ <div class="p-6 bg-card rounded-lg border-default">
58
+ <div class="text-xl font-semibold text-foreground mb-3">Main Workspace</div>
59
+ <div class="text-muted-foreground mb-4">
60
+ This is the primary content area. The utility panel slides in from the right to
61
+ provide additional controls and settings without leaving this workspace.
62
+ </div>
63
+ <div class="space-y-2 text-sm">
64
+ <div class="text-foreground">
65
+ Panel Position: <span class="text-muted-foreground">Right side (default)</span>
66
+ </div>
67
+ <div class="text-foreground">
68
+ Panel Width: <span class="text-muted-foreground">312px (default)</span>
69
+ </div>
70
+ <div class="text-foreground">
71
+ Overlay Mode: <span class="text-muted-foreground">Yes</span>
72
+ </div>
73
+ <div class="text-foreground">
74
+ Current Form Data:
75
+ <div class="text-muted-foreground mt-2 bg-muted rounded-lg p-3 border-default">
76
+ {{ formData() | json }}
77
+ </div>
78
+ </div>
79
+ </div>
80
+ </div>
81
+
82
+ @if (saveMessage()) {
83
+ <div class="text-sm text-success">{{ saveMessage() }}</div>
84
+ }
85
+ </div>
86
+ </demo-example>
87
+ </demo-page>
88
+
89
+ <!-- Fixed positioned utility panel - outside the demo content flow -->
90
+ <modus-utility-panel
91
+ [expanded]="expanded()"
92
+ position="right"
93
+ panelWidth="312px"
94
+ ariaLabel="Project settings panel"
95
+ className="fixed-utility-panel"
96
+ >
97
+ <div
98
+ slot="header"
99
+ class="flex items-center justify-between w-full min-w-full max-w-full"
100
+ >
101
+ <div class="text-xl font-bold text-foreground">Project Settings</div>
102
+ <modus-button
103
+ size="sm"
104
+ color="secondary"
105
+ variant="borderless"
106
+ shape="circle"
107
+ icon="close"
108
+ iconPosition="only"
109
+ ariaLabel="Close panel"
110
+ (buttonClick)="closePanel()"
111
+ />
112
+ </div>
113
+
114
+ <div class="space-y-4">
115
+ <div>
116
+ <modus-text-input
117
+ label="Project Name"
118
+ [value]="formData().projectName"
119
+ placeholder="Enter project name"
120
+ (inputChange)="handleInputChange('projectName', $event)"
121
+ />
122
+ </div>
123
+
124
+ <div>
125
+ <modus-text-input
126
+ label="Email"
127
+ type="email"
128
+ [value]="formData().email"
129
+ placeholder="Enter email address"
130
+ (inputChange)="handleInputChange('email', $event)"
131
+ />
132
+ </div>
133
+
134
+ <div class="space-y-3">
135
+ <modus-switch
136
+ label="Enable Notifications"
137
+ [value]="formData().notifications"
138
+ (inputChange)="handleSwitchChange('notifications', $event)"
139
+ />
140
+
141
+ <modus-switch
142
+ label="Auto-save Changes"
143
+ [value]="formData().autoSave"
144
+ (inputChange)="handleSwitchChange('autoSave', $event)"
145
+ />
146
+ </div>
147
+
148
+ <div class="pt-2 text-xs text-muted-foreground">
149
+ <div>• Panel slides over main content</div>
150
+ <div>• Form state is preserved when toggling</div>
151
+ <div>• Settings are applied immediately</div>
152
+ </div>
153
+ </div>
154
+
155
+ <div slot="footer" class="flex gap-2 justify-end">
156
+ <modus-button size="md" color="secondary" (buttonClick)="closePanel()">
157
+ Close
158
+ </modus-button>
159
+ <modus-button size="md" color="primary" (buttonClick)="handleSave()">
160
+ Save Settings
161
+ </modus-button>
162
+ </div>
163
+ </modus-utility-panel>
164
+ `,
165
+ })
166
+ export class UtilityPanelDemoPageComponent {
167
+ readonly expanded = signal<boolean>(false);
168
+ readonly saveMessage = signal<string>('');
169
+ readonly formData = signal<UtilityPanelFormState>({
170
+ projectName: '',
171
+ email: '',
172
+ notifications: true,
173
+ autoSave: false,
174
+ });
175
+
176
+ togglePanel(): void {
177
+ this.expanded.set(!this.expanded());
178
+ }
179
+
180
+ closePanel(): void {
181
+ this.expanded.set(false);
182
+ }
183
+
184
+ handleInputChange(field: 'projectName' | 'email', event: InputEvent): void {
185
+ const target = event.target as HTMLInputElement;
186
+ this.formData.update((data) => ({ ...data, [field]: target.value }));
187
+ }
188
+
189
+ handleSwitchChange(field: 'notifications' | 'autoSave', event: InputEvent): void {
190
+ const target = event.target as HTMLInputElement;
191
+ this.formData.update((data) => ({ ...data, [field]: target.checked }));
192
+ }
193
+
194
+ handleSave(): void {
195
+ this.saveMessage.set('Settings saved. Check the form data above for current values.');
196
+ }
197
+ }
@@ -0,0 +1,119 @@
1
+ /**
2
+ * Dev Panel configuration
3
+ */
4
+
5
+ export interface DevNavItem {
6
+ label: string;
7
+ path: string;
8
+ icon?: string;
9
+ description?: string;
10
+ }
11
+
12
+ export interface DevNavCategory {
13
+ label: string;
14
+ items: DevNavItem[];
15
+ }
16
+
17
+ /**
18
+ * Main navigation items for the Dev Panel
19
+ */
20
+ export const devNavItems: DevNavItem[] = [
21
+ {
22
+ label: 'Colors',
23
+ path: '/dev/colors',
24
+ icon: 'palette',
25
+ description: 'Color palette reference',
26
+ },
27
+ {
28
+ label: 'Icons',
29
+ path: '/dev/icons',
30
+ icon: 'apps',
31
+ description: 'Icon library browser',
32
+ },
33
+ {
34
+ label: 'Components',
35
+ path: '/dev/components',
36
+ icon: 'widgets',
37
+ description: 'Component gallery',
38
+ },
39
+ ];
40
+
41
+ /**
42
+ * Component demo categories for the Dev Panel navigation.
43
+ * Categories and items are sorted alphabetically.
44
+ */
45
+ export const demoCategories: DevNavCategory[] = [
46
+ {
47
+ label: 'Display',
48
+ items: [
49
+ { label: 'Avatar', path: '/dev/demos/avatar' },
50
+ { label: 'Badge', path: '/dev/demos/badge' },
51
+ { label: 'Chip', path: '/dev/demos/chip' },
52
+ { label: 'Icon', path: '/dev/demos/icon' },
53
+ { label: 'Logo', path: '/dev/demos/logo' },
54
+ { label: 'Pagination', path: '/dev/demos/pagination' },
55
+ { label: 'Skeleton', path: '/dev/demos/skeleton' },
56
+ { label: 'Stepper', path: '/dev/demos/stepper' },
57
+ { label: 'Table', path: '/dev/demos/table' },
58
+ ],
59
+ },
60
+ {
61
+ label: 'Feedback',
62
+ items: [
63
+ { label: 'Alert', path: '/dev/demos/alert' },
64
+ { label: 'Handle', path: '/dev/demos/handle' },
65
+ { label: 'Input Feedback', path: '/dev/demos/input-feedback' },
66
+ { label: 'Input Label', path: '/dev/demos/input-label' },
67
+ { label: 'Loader', path: '/dev/demos/loader' },
68
+ { label: 'Modal', path: '/dev/demos/modal' },
69
+ { label: 'Progress', path: '/dev/demos/progress' },
70
+ { label: 'Rating', path: '/dev/demos/rating' },
71
+ { label: 'Toast', path: '/dev/demos/toast' },
72
+ { label: 'Tooltip', path: '/dev/demos/tooltip' },
73
+ ],
74
+ },
75
+ {
76
+ label: 'Forms',
77
+ items: [
78
+ { label: 'Autocomplete', path: '/dev/demos/autocomplete' },
79
+ { label: 'Button', path: '/dev/demos/button' },
80
+ { label: 'Button Group', path: '/dev/demos/button-group' },
81
+ { label: 'Checkbox', path: '/dev/demos/checkbox' },
82
+ { label: 'Date', path: '/dev/demos/date' },
83
+ { label: 'Dropdown', path: '/dev/demos/dropdown' },
84
+ { label: 'File Dropzone', path: '/dev/demos/file-dropzone' },
85
+ { label: 'Number Input', path: '/dev/demos/number-input' },
86
+ { label: 'Radio', path: '/dev/demos/radio' },
87
+ { label: 'Select', path: '/dev/demos/select' },
88
+ { label: 'Slider', path: '/dev/demos/slider' },
89
+ { label: 'Switch', path: '/dev/demos/switch' },
90
+ { label: 'Text Input', path: '/dev/demos/text-input' },
91
+ { label: 'Textarea', path: '/dev/demos/textarea' },
92
+ { label: 'Time Input', path: '/dev/demos/time-input' },
93
+ ],
94
+ },
95
+ {
96
+ label: 'Layout',
97
+ items: [
98
+ { label: 'Accordion', path: '/dev/demos/accordion' },
99
+ { label: 'Card', path: '/dev/demos/card' },
100
+ { label: 'Panel', path: '/dev/demos/panel' },
101
+ { label: 'Utility Panel', path: '/dev/demos/utility-panel' },
102
+ ],
103
+ },
104
+ {
105
+ label: 'Navigation',
106
+ items: [
107
+ { label: 'Breadcrumbs', path: '/dev/demos/breadcrumbs' },
108
+ { label: 'Menu', path: '/dev/demos/menu' },
109
+ { label: 'Navbar', path: '/dev/demos/navbar' },
110
+ { label: 'Side Navigation', path: '/dev/demos/side-navigation' },
111
+ { label: 'Tabs', path: '/dev/demos/tabs' },
112
+ { label: 'Toolbar', path: '/dev/demos/toolbar' },
113
+ ],
114
+ },
115
+ {
116
+ label: 'System',
117
+ items: [{ label: 'Theme Switcher', path: '/dev/demos/theme-switcher' }],
118
+ },
119
+ ];
@@ -0,0 +1,215 @@
1
+ import { ChangeDetectionStrategy, Component, inject, signal } from '@angular/core';
2
+ import { Router, RouterModule } from '@angular/router';
3
+ import { DevPanelService } from '../dev-panel.service';
4
+ import { devNavItems, demoCategories } from '../dev-config';
5
+ import { ThemeSwitcherDropdownComponent } from '../theme-switcher-dropdown/theme-switcher-dropdown.component';
6
+ import { ModusButtonComponent } from '../../components/modus-button.component';
7
+
8
+ /**
9
+ * Floating Dev Panel component.
10
+ *
11
+ * Provides access to development tools, demos, and theme switching.
12
+ * Only renders in development mode when devPanel environment flag is true.
13
+ *
14
+ * Features:
15
+ * - Floating toggle button (bottom-right)
16
+ * - Slide-out panel from right
17
+ * - Theme switcher
18
+ * - Navigation to reference pages (Colors, Icons, Components)
19
+ * - Accordion navigation to component demos
20
+ * - Keyboard shortcuts (Ctrl+Shift+D toggle, Escape close)
21
+ */
22
+ @Component({
23
+ selector: 'app-dev-panel',
24
+ standalone: true,
25
+ imports: [RouterModule, ModusButtonComponent, ThemeSwitcherDropdownComponent],
26
+ changeDetection: ChangeDetectionStrategy.OnPush,
27
+ template: `
28
+ <!-- Floating Toggle Button - always visible -->
29
+ <div class="fixed bottom-4 right-4 z-[9999]">
30
+ <modus-button color="primary" size="lg" (buttonClick)="toggle()">
31
+ <div class="flex items-center gap-2">
32
+ <i class="modus-icons text-xl">{{ isOpen() ? 'close' : 'code' }}</i>
33
+ <div class="hidden sm:block">Dev</div>
34
+ </div>
35
+ </modus-button>
36
+ </div>
37
+
38
+ <!-- Panel - Only render when open to avoid blocking clicks -->
39
+ @if (isOpen()) {
40
+ <div class="fixed inset-0 z-[9998]">
41
+ <!-- Backdrop -->
42
+ <div class="absolute inset-0 bg-foreground-20 animate-fade-in" (click)="close()"></div>
43
+
44
+ <!-- Panel -->
45
+ <div
46
+ class="absolute top-0 right-0 h-full w-[360px] max-w-[90vw] bg-card shadow-xl animate-slide-in-right overflow-hidden flex flex-col"
47
+ >
48
+ <!-- Header -->
49
+ <div class="flex items-center justify-between p-4 border-bottom-default shrink-0">
50
+ <div class="text-lg font-semibold text-foreground">Dev Panel</div>
51
+ <modus-button
52
+ variant="borderless"
53
+ color="secondary"
54
+ size="sm"
55
+ (buttonClick)="close()"
56
+ ariaLabel="Close Dev Panel"
57
+ >
58
+ <i class="modus-icons">close</i>
59
+ </modus-button>
60
+ </div>
61
+
62
+ <!-- Body -->
63
+ <div class="flex-1 overflow-y-auto p-4">
64
+ <div class="space-y-6">
65
+ <!-- Theme Switcher Section -->
66
+ <div class="space-y-2">
67
+ <div class="text-sm font-semibold text-foreground-60 uppercase tracking-wide">
68
+ Theme
69
+ </div>
70
+ <app-theme-switcher-dropdown />
71
+ </div>
72
+
73
+ <!-- Main Navigation -->
74
+ <div class="space-y-2">
75
+ <div class="text-sm font-semibold text-foreground-60 uppercase tracking-wide">
76
+ Reference
77
+ </div>
78
+ <div class="space-y-1">
79
+ <!-- Home Button -->
80
+ <div
81
+ role="button"
82
+ tabindex="0"
83
+ (click)="handleNavClick('/')"
84
+ (keydown.enter)="handleNavClick('/')"
85
+ class="w-full flex items-center gap-3 px-3 py-2 rounded-lg text-left transition-colors cursor-pointer"
86
+ [class.bg-primary]="isActive('/')"
87
+ [class.text-primary-foreground]="isActive('/')"
88
+ [class.hover:bg-muted]="!isActive('/')"
89
+ [class.text-foreground]="!isActive('/')"
90
+ >
91
+ <i class="modus-icons text-lg">home</i>
92
+ <div>Home</div>
93
+ </div>
94
+
95
+ @for (item of navItems; track item.path) {
96
+ <div
97
+ role="button"
98
+ tabindex="0"
99
+ (click)="handleNavClick(item.path)"
100
+ (keydown.enter)="handleNavClick(item.path)"
101
+ class="w-full flex items-center gap-3 px-3 py-2 rounded-lg text-left transition-colors cursor-pointer"
102
+ [class.bg-primary]="isActive(item.path)"
103
+ [class.text-primary-foreground]="isActive(item.path)"
104
+ [class.hover:bg-muted]="!isActive(item.path)"
105
+ [class.text-foreground]="!isActive(item.path)"
106
+ >
107
+ @if (item.icon) {
108
+ <i class="modus-icons text-lg">{{ item.icon }}</i>
109
+ }
110
+ <div>{{ item.label }}</div>
111
+ </div>
112
+ }
113
+ </div>
114
+ </div>
115
+
116
+ <!-- Component Demos -->
117
+ <div class="space-y-2">
118
+ <div class="text-sm font-semibold text-foreground-60 uppercase tracking-wide">
119
+ Component Demos
120
+ </div>
121
+ <div class="border-default rounded-lg overflow-hidden">
122
+ @for (category of demoCategories; track category.label) {
123
+ <div class="border-bottom-default last:border-b-0">
124
+ <div
125
+ role="button"
126
+ tabindex="0"
127
+ (click)="toggleCategory(category.label)"
128
+ (keydown.enter)="toggleCategory(category.label)"
129
+ class="w-full flex items-center justify-between px-3 py-2 hover:bg-muted transition-colors text-left cursor-pointer"
130
+ >
131
+ <div class="font-medium text-foreground">{{ category.label }}</div>
132
+ <i class="modus-icons text-foreground-60">
133
+ {{ isCategoryExpanded(category.label) ? 'expand_less' : 'expand_more' }}
134
+ </i>
135
+ </div>
136
+ @if (isCategoryExpanded(category.label)) {
137
+ <div class="pl-4 pb-2 space-y-1">
138
+ @for (item of category.items; track item.path) {
139
+ <div
140
+ role="button"
141
+ tabindex="0"
142
+ (click)="handleNavClick(item.path)"
143
+ (keydown.enter)="handleNavClick(item.path)"
144
+ class="w-full flex items-center gap-2 px-3 py-1.5 rounded text-sm text-left transition-colors cursor-pointer"
145
+ [class.bg-primary]="isActive(item.path)"
146
+ [class.text-primary-foreground]="isActive(item.path)"
147
+ [class.hover:bg-muted]="!isActive(item.path)"
148
+ [class.text-foreground]="!isActive(item.path)"
149
+ >
150
+ {{ item.label }}
151
+ </div>
152
+ }
153
+ </div>
154
+ }
155
+ </div>
156
+ }
157
+ </div>
158
+ </div>
159
+
160
+ <!-- Keyboard Shortcut Hint -->
161
+ <div class="pt-4 border-top-default">
162
+ <div class="text-xs text-foreground-60 text-center">
163
+ Press
164
+ <span class="inline px-1.5 py-0.5 bg-muted rounded text-foreground text-xs font-mono"
165
+ >Ctrl+Shift+D</span
166
+ >
167
+ to toggle
168
+ </div>
169
+ </div>
170
+ </div>
171
+ </div>
172
+ </div>
173
+ </div>
174
+ }
175
+ `,
176
+ })
177
+ export class DevPanelComponent {
178
+ private readonly devPanelService = inject(DevPanelService);
179
+ private readonly router = inject(Router);
180
+
181
+ readonly isOpen = this.devPanelService.isOpen;
182
+ readonly navItems = devNavItems;
183
+ readonly demoCategories = demoCategories;
184
+
185
+ private readonly expandedCategories = signal<string[]>([]);
186
+
187
+ toggle(): void {
188
+ this.devPanelService.toggle();
189
+ }
190
+
191
+ close(): void {
192
+ this.devPanelService.close();
193
+ }
194
+
195
+ handleNavClick(path: string): void {
196
+ this.router.navigate([path]);
197
+ this.close();
198
+ }
199
+
200
+ isActive(path: string): boolean {
201
+ return this.router.url === path;
202
+ }
203
+
204
+ toggleCategory(label: string): void {
205
+ this.expandedCategories.update((categories) =>
206
+ categories.includes(label)
207
+ ? categories.filter((l) => l !== label)
208
+ : [...categories, label],
209
+ );
210
+ }
211
+
212
+ isCategoryExpanded(label: string): boolean {
213
+ return this.expandedCategories().includes(label);
214
+ }
215
+ }
@@ -0,0 +1,63 @@
1
+ import { Injectable, signal, computed } from '@angular/core';
2
+ import { environment } from '../../environments/environment.development';
3
+
4
+ /**
5
+ * Service for managing Dev Panel state.
6
+ *
7
+ * Provides:
8
+ * - Signal-based state management for panel open/close
9
+ * - Environment-based feature flag
10
+ * - Keyboard shortcuts (Ctrl+Shift+D to toggle, Escape to close)
11
+ */
12
+ @Injectable({ providedIn: 'root' })
13
+ export class DevPanelService {
14
+ private readonly _isOpen = signal(false);
15
+
16
+ /** Read-only signal for panel open state */
17
+ readonly isOpen = this._isOpen.asReadonly();
18
+
19
+ /** Computed signal to check if Dev Panel feature is enabled */
20
+ readonly isEnabled = computed(() => environment.devPanel);
21
+
22
+ constructor() {
23
+ this.initKeyboardShortcuts();
24
+ }
25
+
26
+ /** Toggle the panel open/close state */
27
+ toggle(): void {
28
+ this._isOpen.update((open) => !open);
29
+ }
30
+
31
+ /** Open the panel */
32
+ open(): void {
33
+ this._isOpen.set(true);
34
+ }
35
+
36
+ /** Close the panel */
37
+ close(): void {
38
+ this._isOpen.set(false);
39
+ }
40
+
41
+ /**
42
+ * Initialize keyboard shortcuts for the Dev Panel.
43
+ * - Ctrl+Shift+D (or Cmd+Shift+D on Mac): Toggle panel
44
+ * - Escape: Close panel when open
45
+ */
46
+ private initKeyboardShortcuts(): void {
47
+ if (typeof window === 'undefined') return;
48
+
49
+ window.addEventListener('keydown', (event: KeyboardEvent) => {
50
+ // Toggle with Ctrl+Shift+D or Cmd+Shift+D
51
+ if ((event.ctrlKey || event.metaKey) && event.shiftKey && event.key.toLowerCase() === 'd') {
52
+ event.preventDefault();
53
+ this.toggle();
54
+ }
55
+
56
+ // Close with Escape when open
57
+ if (event.key === 'Escape' && this._isOpen()) {
58
+ event.preventDefault();
59
+ this.close();
60
+ }
61
+ });
62
+ }
63
+ }
@@ -0,0 +1,8 @@
1
+ // Dev Panel Infrastructure
2
+ export { DevPanelService } from './dev-panel.service';
3
+ export { DevPanelComponent } from './dev-panel/dev-panel.component';
4
+ export { ThemeSwitcherDropdownComponent } from './theme-switcher-dropdown/theme-switcher-dropdown.component';
5
+
6
+ // Configuration
7
+ export { devNavItems, demoCategories } from './dev-config';
8
+ export type { DevNavItem, DevNavCategory } from './dev-config';