@radix-ng/primitives 0.11.1 → 0.12.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 (449) hide show
  1. package/.compodocrc.json +12 -0
  2. package/CHANGELOG.md +182 -0
  3. package/LICENSE +21 -0
  4. package/accordion/__tests__/accordion-content.directive.spec.ts +8 -0
  5. package/accordion/__tests__/accordion-header.directive.spec.ts +8 -0
  6. package/accordion/__tests__/accordion-item.directive.spec.ts +8 -0
  7. package/accordion/__tests__/accordion-root.directive.spec.ts +8 -0
  8. package/accordion/__tests__/accordion-trigger.directive.spec.ts +8 -0
  9. package/accordion/ng-package.json +5 -0
  10. package/accordion/src/accordion-content.directive.ts +46 -0
  11. package/accordion/src/accordion-header.directive.ts +15 -0
  12. package/accordion/src/accordion-item.directive.ts +216 -0
  13. package/accordion/src/accordion-root.directive.ts +205 -0
  14. package/accordion/src/accordion-trigger.directive.ts +37 -0
  15. package/accordion/stories/accordion.docs.mdx +77 -0
  16. package/accordion/stories/accordion.stories.ts +340 -0
  17. package/alert-dialog/{index.d.ts → index.ts} +1 -0
  18. package/alert-dialog/ng-package.json +5 -0
  19. package/alert-dialog/src/alert-dialog-cancel.directive.ts +17 -0
  20. package/alert-dialog/src/alert-dialog-content.directive.ts +24 -0
  21. package/alert-dialog/src/alert-dialog-root.directive.ts +15 -0
  22. package/alert-dialog/src/alert-dialog-title.directive.ts +7 -0
  23. package/alert-dialog/src/alert-dialog-trigger.directive.ts +17 -0
  24. package/alert-dialog/src/alert-dialog.service.ts +51 -0
  25. package/alert-dialog/stories/alert-dialog.stories.ts +139 -0
  26. package/avatar/__tests__/avatar-fallback.directive.spec.ts +31 -0
  27. package/avatar/__tests__/avatar-image.directive.spec.ts +36 -0
  28. package/avatar/ng-package.json +5 -0
  29. package/avatar/src/avatar-fallback.directive.ts +62 -0
  30. package/avatar/src/avatar-image.directive.ts +55 -0
  31. package/avatar/src/avatar-root.directive.ts +35 -0
  32. package/avatar/src/avatar.config.ts +29 -0
  33. package/avatar/stories/avatar.docs.mdx +37 -0
  34. package/avatar/stories/avatar.stories.ts +87 -0
  35. package/button/__tests__/.gitkeep +0 -0
  36. package/button/src/button-abstract.directive.ts +46 -0
  37. package/checkbox/ng-package.json +5 -0
  38. package/checkbox/src/checkbox-button.directive.ts +26 -0
  39. package/checkbox/src/checkbox-indicator.directive.ts +16 -0
  40. package/checkbox/src/checkbox-input.directive.ts +36 -0
  41. package/checkbox/src/checkbox.directive.ts +146 -0
  42. package/checkbox/src/checkbox.token.ts +8 -0
  43. package/checkbox/stories/checkbox-group.component.ts +87 -0
  44. package/checkbox/stories/checkbox-group.styles.scss +49 -0
  45. package/checkbox/stories/checkbox-indeterminate.component.ts +52 -0
  46. package/checkbox/stories/checkbox.docs.mdx +58 -0
  47. package/checkbox/stories/checkbox.stories.ts +105 -0
  48. package/collapsible/__tests__/collapsible-content.directive.spec.ts +30 -0
  49. package/collapsible/__tests__/collapsible-root.directive.spec.ts +27 -0
  50. package/collapsible/__tests__/collapsible-trigger.directive.spec.ts +30 -0
  51. package/collapsible/ng-package.json +5 -0
  52. package/collapsible/src/collapsible-content.directive.ts +34 -0
  53. package/collapsible/src/collapsible-content.token.ts +6 -0
  54. package/collapsible/src/collapsible-root.directive.ts +120 -0
  55. package/collapsible/src/collapsible-trigger.directive.ts +44 -0
  56. package/collapsible/stories/collapsible-animation.component.ts +116 -0
  57. package/collapsible/stories/collapsible-external-triggering.component.ts +86 -0
  58. package/collapsible/stories/collapsible.docs.mdx +53 -0
  59. package/collapsible/stories/collapsible.stories.ts +151 -0
  60. package/context-menu/README.md +1 -0
  61. package/context-menu/index.ts +10 -0
  62. package/context-menu/ng-package.json +5 -0
  63. package/context-menu/src/context-menu-content.directive.ts +47 -0
  64. package/context-menu/src/context-menu-item-checkbox.directive.ts +30 -0
  65. package/context-menu/src/context-menu-item-indicator.directive.ts +14 -0
  66. package/context-menu/src/context-menu-item-radio-group.directive.ts +31 -0
  67. package/context-menu/src/context-menu-item-radio.directive.ts +69 -0
  68. package/context-menu/src/context-menu-item-selectable.ts +18 -0
  69. package/context-menu/src/context-menu-item.directive.ts +65 -0
  70. package/context-menu/src/context-menu-label.directive.ts +7 -0
  71. package/context-menu/src/context-menu-separator.directive.ts +13 -0
  72. package/context-menu/src/context-menu-trigger.directive.ts +82 -0
  73. package/context-menu/stories/context-menu.docs.mdx +23 -0
  74. package/context-menu/stories/context-menu.stories.ts +253 -0
  75. package/core/index.ts +3 -0
  76. package/core/src/accessor/provide-value-accessor.ts +20 -0
  77. package/core/src/auto-focus.directive.ts +81 -0
  78. package/core/src/inject-ng-control.ts +28 -0
  79. package/core/src/mount.ts +27 -0
  80. package/dialog/__tests__/dialog-content.directive.spec.ts +77 -0
  81. package/dialog/__tests__/dialog-trigger.directive.spec.ts +85 -0
  82. package/dialog/index.ts +31 -0
  83. package/dialog/ng-package.json +5 -0
  84. package/dialog/src/dialog-close.directive.ts +18 -0
  85. package/dialog/src/dialog-content.directive.ts +45 -0
  86. package/dialog/src/dialog-description.directive.ts +7 -0
  87. package/dialog/src/dialog-dismiss.directive.ts +18 -0
  88. package/dialog/src/dialog-ref.ts +70 -0
  89. package/dialog/src/dialog-title.directive.ts +7 -0
  90. package/dialog/src/dialog-trigger.directive.ts +52 -0
  91. package/dialog/src/dialog.config.ts +55 -0
  92. package/dialog/src/dialog.injectors.ts +12 -0
  93. package/dialog/src/dialog.providers.ts +27 -0
  94. package/dialog/src/dialog.service.ts +94 -0
  95. package/dialog/stories/dialog.docs.mdx +32 -0
  96. package/dialog/stories/dialog.stories.ts +233 -0
  97. package/dropdown-menu/ng-package.json +5 -0
  98. package/dropdown-menu/src/dropdown-menu-content.directive.ts +47 -0
  99. package/dropdown-menu/src/dropdown-menu-item-checkbox.directive.ts +30 -0
  100. package/dropdown-menu/src/dropdown-menu-item-indicator.directive.ts +14 -0
  101. package/dropdown-menu/src/dropdown-menu-item-radio-group.directive.ts +31 -0
  102. package/dropdown-menu/src/dropdown-menu-item-radio.directive.ts +72 -0
  103. package/dropdown-menu/src/dropdown-menu-item-selectable.ts +18 -0
  104. package/dropdown-menu/src/dropdown-menu-item.directive.ts +66 -0
  105. package/dropdown-menu/src/dropdown-menu-label.directive.ts +7 -0
  106. package/dropdown-menu/src/dropdown-menu-separator.directive.ts +13 -0
  107. package/dropdown-menu/src/dropdown-menu-trigger.directive.ts +185 -0
  108. package/dropdown-menu/stories/dropdown-menu-item-checkbox.component.ts +104 -0
  109. package/dropdown-menu/stories/dropdown-menu-item-checkbox.styles.scss +106 -0
  110. package/dropdown-menu/stories/dropdown-menu-item-radio.component.ts +95 -0
  111. package/dropdown-menu/stories/dropdown-menu-item-radio.styles.scss +106 -0
  112. package/dropdown-menu/stories/dropdown.docs.mdx +27 -0
  113. package/dropdown-menu/stories/dropdown.stories.ts +212 -0
  114. package/form-field/index.ts +1 -0
  115. package/form-field/src/.gitkeep +0 -0
  116. package/jest.config.ts +21 -0
  117. package/label/__tests__/label-root.directive.spec.ts +99 -0
  118. package/label/ng-package.json +5 -0
  119. package/label/src/label.directive.ts +58 -0
  120. package/label/stories/label.docs.mdx +40 -0
  121. package/label/stories/label.stories.ts +76 -0
  122. package/menu/index.ts +29 -0
  123. package/menu/ng-package.json +5 -0
  124. package/menu/src/menu-content.directive.ts +9 -0
  125. package/menu/src/menu-directive.ts +10 -0
  126. package/menu/src/menu-group.directive.ts +12 -0
  127. package/menu/src/menu-item.directive.ts +44 -0
  128. package/menu/src/menu-label.directive.ts +7 -0
  129. package/menu/src/menu-separator.directive.ts +13 -0
  130. package/menubar/index.ts +38 -0
  131. package/menubar/ng-package.json +5 -0
  132. package/menubar/src/menubar-content.directive.ts +9 -0
  133. package/menubar/src/menubar-item-checkbox.directive.ts +32 -0
  134. package/menubar/src/menubar-item-indicator.directive.ts +10 -0
  135. package/menubar/src/menubar-item-radio.directive.ts +33 -0
  136. package/menubar/src/menubar-item.directive.ts +12 -0
  137. package/menubar/src/menubar-radio-group.directive.ts +9 -0
  138. package/menubar/src/menubar-root.directive.ts +15 -0
  139. package/menubar/src/menubar-separator.directive.ts +9 -0
  140. package/menubar/src/menubar-trigger.directive.ts +40 -0
  141. package/menubar/stories/menubar.stories.ts +229 -0
  142. package/ng-package.json +8 -0
  143. package/package.json +4 -118
  144. package/portal/stories/portal.docs.mdx +85 -0
  145. package/presence/__test__/presence-test.component.ts +51 -0
  146. package/presence/__test__/presence.spec.ts +50 -0
  147. package/presence/index.ts +4 -0
  148. package/presence/src/presence.ts +119 -0
  149. package/presence/src/transitions/transition.collapse.ts +99 -0
  150. package/presence/src/transitions/transition.toast.ts +27 -0
  151. package/presence/src/types.ts +20 -0
  152. package/presence/src/utils.ts +63 -0
  153. package/presence/stories/presence-story.componen.ts +69 -0
  154. package/presence/stories/presence.docs.mdx +40 -0
  155. package/presence/stories/presence.stories.ts +29 -0
  156. package/progress/__test__/progress.spec.ts +55 -0
  157. package/progress/{index.d.ts → index.ts} +1 -0
  158. package/progress/ng-package.json +5 -0
  159. package/progress/src/progress-indicator.directive.ts +26 -0
  160. package/progress/src/progress-root.directive.ts +134 -0
  161. package/progress/stories/progress.docs.mdx +65 -0
  162. package/progress/stories/progress.stories.ts +66 -0
  163. package/project.json +39 -0
  164. package/radio/{index.d.ts → index.ts} +1 -0
  165. package/radio/ng-package.json +5 -0
  166. package/radio/src/radio-indicator.directive.ts +17 -0
  167. package/radio/src/radio-item.directive.ts +68 -0
  168. package/radio/src/radio-root.directive.ts +207 -0
  169. package/radio/src/{radio-tokens.d.ts → radio-tokens.ts} +5 -1
  170. package/radio/stories/radio-group.component.ts +39 -0
  171. package/radio/stories/radio-group.styles.scss +70 -0
  172. package/radio/stories/radio.docs.mdx +68 -0
  173. package/radio/stories/radio.stories.ts +155 -0
  174. package/separator/__tests__/separator.directive.spec.ts +58 -0
  175. package/separator/ng-package.json +5 -0
  176. package/separator/src/separator.directive.ts +35 -0
  177. package/separator/stories/separator.docs.mdx +37 -0
  178. package/separator/stories/separator.stories.ts +82 -0
  179. package/slider/src/slider-input.directive.ts +0 -0
  180. package/slider/src/slider-thumb.directives.ts +60 -0
  181. package/slider/src/slider-track.directive.ts +11 -0
  182. package/slider/src/slider.directive.ts +59 -0
  183. package/slider/src/slider.types.ts +4 -0
  184. package/switch/index.ts +22 -0
  185. package/switch/ng-package.json +5 -0
  186. package/switch/src/switch-input.directive.ts +24 -0
  187. package/switch/src/switch-root.directive.ts +127 -0
  188. package/switch/src/switch-thumb.directive.ts +15 -0
  189. package/switch/stories/switch.docs.mdx +83 -0
  190. package/switch/stories/switch.stories.ts +149 -0
  191. package/tabs/__tests__/tabs-context.service.spec.ts +35 -0
  192. package/tabs/index.ts +26 -0
  193. package/tabs/ng-package.json +5 -0
  194. package/tabs/src/tabs-content.directive.ts +23 -0
  195. package/tabs/src/tabs-context.service.ts +43 -0
  196. package/tabs/src/tabs-list.directive.ts +21 -0
  197. package/tabs/src/tabs-root.directive.ts +70 -0
  198. package/tabs/src/tabs-trigger.directive.ts +55 -0
  199. package/tabs/stories/tabs.stories.ts +213 -0
  200. package/test-setup.ts +1 -0
  201. package/toggle/__tests__/toggle.directive.spec.ts +87 -0
  202. package/toggle/ng-package.json +5 -0
  203. package/toggle/src/toggle.directive.ts +49 -0
  204. package/toggle/stories/toggle.docs.mdx +60 -0
  205. package/toggle/stories/toggle.stories.ts +84 -0
  206. package/toggle-group/ng-package.json +5 -0
  207. package/toggle-group/src/toggle-group-button.directive.ts +73 -0
  208. package/toggle-group/src/toggle-group-button.token.ts +8 -0
  209. package/toggle-group/src/toggle-group-multi.directive.ts +158 -0
  210. package/toggle-group/src/toggle-group.directive.ts +148 -0
  211. package/toggle-group/src/toggle-group.token.ts +11 -0
  212. package/toggle-group/stories/toggle-group.docs.mdx +87 -0
  213. package/toggle-group/stories/toggle-group.stories.ts +95 -0
  214. package/tsconfig.doc.json +11 -0
  215. package/tsconfig.json +29 -0
  216. package/tsconfig.lib.json +19 -0
  217. package/tsconfig.lib.prod.json +12 -0
  218. package/tsconfig.spec.json +11 -0
  219. package/visually-hidden/README.md +3 -0
  220. package/visually-hidden/stories/visually-hidden.docs.mdx +36 -0
  221. package/accordion/src/accordion-content.directive.d.ts +0 -11
  222. package/accordion/src/accordion-header.directive.d.ts +0 -7
  223. package/accordion/src/accordion-item.directive.d.ts +0 -64
  224. package/accordion/src/accordion-root.directive.d.ts +0 -78
  225. package/accordion/src/accordion-trigger.directive.d.ts +0 -15
  226. package/alert-dialog/src/alert-dialog-cancel.directive.d.ts +0 -7
  227. package/alert-dialog/src/alert-dialog-content.directive.d.ts +0 -9
  228. package/alert-dialog/src/alert-dialog-root.directive.d.ts +0 -9
  229. package/alert-dialog/src/alert-dialog-title.directive.d.ts +0 -5
  230. package/alert-dialog/src/alert-dialog-trigger.directive.d.ts +0 -7
  231. package/alert-dialog/src/alert-dialog.service.d.ts +0 -14
  232. package/avatar/src/avatar-fallback.directive.d.ts +0 -27
  233. package/avatar/src/avatar-image.directive.d.ts +0 -16
  234. package/avatar/src/avatar-root.directive.d.ts +0 -21
  235. package/avatar/src/avatar.config.d.ts +0 -13
  236. package/checkbox/src/checkbox-button.directive.d.ts +0 -8
  237. package/checkbox/src/checkbox-indicator.directive.d.ts +0 -6
  238. package/checkbox/src/checkbox-input.directive.d.ts +0 -9
  239. package/checkbox/src/checkbox.directive.d.ts +0 -75
  240. package/checkbox/src/checkbox.token.d.ts +0 -4
  241. package/collapsible/src/collapsible-content.directive.d.ts +0 -17
  242. package/collapsible/src/collapsible-content.token.d.ts +0 -3
  243. package/collapsible/src/collapsible-root.directive.d.ts +0 -55
  244. package/collapsible/src/collapsible-trigger.directive.d.ts +0 -26
  245. package/compodoc/documentation.json +0 -11759
  246. package/dropdown-menu/src/dropdown-menu-content.directive.d.ts +0 -15
  247. package/dropdown-menu/src/dropdown-menu-item-checkbox.directive.d.ts +0 -9
  248. package/dropdown-menu/src/dropdown-menu-item-indicator.directive.d.ts +0 -7
  249. package/dropdown-menu/src/dropdown-menu-item-radio-group.directive.d.ts +0 -12
  250. package/dropdown-menu/src/dropdown-menu-item-radio.directive.d.ts +0 -19
  251. package/dropdown-menu/src/dropdown-menu-item-selectable.d.ts +0 -12
  252. package/dropdown-menu/src/dropdown-menu-item.directive.d.ts +0 -17
  253. package/dropdown-menu/src/dropdown-menu-label.directive.d.ts +0 -5
  254. package/dropdown-menu/src/dropdown-menu-separator.directive.d.ts +0 -6
  255. package/dropdown-menu/src/dropdown-menu-trigger.directive.d.ts +0 -43
  256. package/esm2022/accordion/index.mjs +0 -6
  257. package/esm2022/accordion/radix-ng-primitives-accordion.mjs +0 -5
  258. package/esm2022/accordion/src/accordion-content.directive.mjs +0 -42
  259. package/esm2022/accordion/src/accordion-header.directive.mjs +0 -23
  260. package/esm2022/accordion/src/accordion-item.directive.mjs +0 -175
  261. package/esm2022/accordion/src/accordion-root.directive.mjs +0 -160
  262. package/esm2022/accordion/src/accordion-trigger.directive.mjs +0 -42
  263. package/esm2022/alert-dialog/index.mjs +0 -7
  264. package/esm2022/alert-dialog/radix-ng-primitives-alert-dialog.mjs +0 -5
  265. package/esm2022/alert-dialog/src/alert-dialog-cancel.directive.mjs +0 -24
  266. package/esm2022/alert-dialog/src/alert-dialog-content.directive.mjs +0 -34
  267. package/esm2022/alert-dialog/src/alert-dialog-root.directive.mjs +0 -25
  268. package/esm2022/alert-dialog/src/alert-dialog-title.directive.mjs +0 -14
  269. package/esm2022/alert-dialog/src/alert-dialog-trigger.directive.mjs +0 -24
  270. package/esm2022/alert-dialog/src/alert-dialog.service.mjs +0 -47
  271. package/esm2022/avatar/index.mjs +0 -5
  272. package/esm2022/avatar/radix-ng-primitives-avatar.mjs +0 -5
  273. package/esm2022/avatar/src/avatar-fallback.directive.mjs +0 -54
  274. package/esm2022/avatar/src/avatar-image.directive.mjs +0 -48
  275. package/esm2022/avatar/src/avatar-root.directive.mjs +0 -38
  276. package/esm2022/avatar/src/avatar.config.mjs +0 -17
  277. package/esm2022/checkbox/index.mjs +0 -6
  278. package/esm2022/checkbox/radix-ng-primitives-checkbox.mjs +0 -5
  279. package/esm2022/checkbox/src/checkbox-button.directive.mjs +0 -33
  280. package/esm2022/checkbox/src/checkbox-indicator.directive.mjs +0 -24
  281. package/esm2022/checkbox/src/checkbox-input.directive.mjs +0 -41
  282. package/esm2022/checkbox/src/checkbox.directive.mjs +0 -141
  283. package/esm2022/checkbox/src/checkbox.token.mjs +0 -6
  284. package/esm2022/collapsible/index.mjs +0 -4
  285. package/esm2022/collapsible/radix-ng-primitives-collapsible.mjs +0 -5
  286. package/esm2022/collapsible/src/collapsible-content.directive.mjs +0 -45
  287. package/esm2022/collapsible/src/collapsible-content.token.mjs +0 -3
  288. package/esm2022/collapsible/src/collapsible-root.directive.mjs +0 -118
  289. package/esm2022/collapsible/src/collapsible-trigger.directive.mjs +0 -49
  290. package/esm2022/dialog/index.mjs +0 -2
  291. package/esm2022/dialog/radix-ng-primitives-dialog.mjs +0 -5
  292. package/esm2022/dropdown-menu/index.mjs +0 -11
  293. package/esm2022/dropdown-menu/radix-ng-primitives-dropdown-menu.mjs +0 -5
  294. package/esm2022/dropdown-menu/src/dropdown-menu-content.directive.mjs +0 -56
  295. package/esm2022/dropdown-menu/src/dropdown-menu-item-checkbox.directive.mjs +0 -36
  296. package/esm2022/dropdown-menu/src/dropdown-menu-item-indicator.directive.mjs +0 -22
  297. package/esm2022/dropdown-menu/src/dropdown-menu-item-radio-group.directive.mjs +0 -37
  298. package/esm2022/dropdown-menu/src/dropdown-menu-item-radio.directive.mjs +0 -64
  299. package/esm2022/dropdown-menu/src/dropdown-menu-item-selectable.mjs +0 -31
  300. package/esm2022/dropdown-menu/src/dropdown-menu-item.directive.mjs +0 -71
  301. package/esm2022/dropdown-menu/src/dropdown-menu-label.directive.mjs +0 -14
  302. package/esm2022/dropdown-menu/src/dropdown-menu-separator.directive.mjs +0 -21
  303. package/esm2022/dropdown-menu/src/dropdown-menu-trigger.directive.mjs +0 -176
  304. package/esm2022/index.mjs +0 -2
  305. package/esm2022/label/index.mjs +0 -2
  306. package/esm2022/label/radix-ng-primitives-label.mjs +0 -5
  307. package/esm2022/label/src/label.directive.mjs +0 -59
  308. package/esm2022/menu/index.mjs +0 -45
  309. package/esm2022/menu/radix-ng-primitives-menu.mjs +0 -5
  310. package/esm2022/menu/src/menu-content.directive.mjs +0 -17
  311. package/esm2022/menu/src/menu-directive.mjs +0 -18
  312. package/esm2022/menu/src/menu-group.directive.mjs +0 -20
  313. package/esm2022/menu/src/menu-item.directive.mjs +0 -46
  314. package/esm2022/menu/src/menu-label.directive.mjs +0 -14
  315. package/esm2022/menu/src/menu-separator.directive.mjs +0 -21
  316. package/esm2022/menubar/index.mjs +0 -60
  317. package/esm2022/menubar/radix-ng-primitives-menubar.mjs +0 -5
  318. package/esm2022/menubar/src/menubar-content.directive.mjs +0 -17
  319. package/esm2022/menubar/src/menubar-item-checkbox.directive.mjs +0 -34
  320. package/esm2022/menubar/src/menubar-item-indicator.directive.mjs +0 -17
  321. package/esm2022/menubar/src/menubar-item-radio.directive.mjs +0 -35
  322. package/esm2022/menubar/src/menubar-item.directive.mjs +0 -20
  323. package/esm2022/menubar/src/menubar-radio-group.directive.mjs +0 -17
  324. package/esm2022/menubar/src/menubar-root.directive.mjs +0 -24
  325. package/esm2022/menubar/src/menubar-separator.directive.mjs +0 -17
  326. package/esm2022/menubar/src/menubar-trigger.directive.mjs +0 -45
  327. package/esm2022/progress/index.mjs +0 -3
  328. package/esm2022/progress/radix-ng-primitives-progress.mjs +0 -5
  329. package/esm2022/progress/src/progress-indicator.directive.mjs +0 -34
  330. package/esm2022/progress/src/progress-root.directive.mjs +0 -127
  331. package/esm2022/radio/index.mjs +0 -4
  332. package/esm2022/radio/radix-ng-primitives-radio.mjs +0 -5
  333. package/esm2022/radio/src/radio-indicator.directive.mjs +0 -25
  334. package/esm2022/radio/src/radio-item.directive.mjs +0 -70
  335. package/esm2022/radio/src/radio-root.directive.mjs +0 -194
  336. package/esm2022/radio/src/radio-tokens.mjs +0 -3
  337. package/esm2022/radix-ng-primitives.mjs +0 -5
  338. package/esm2022/separator/index.mjs +0 -2
  339. package/esm2022/separator/radix-ng-primitives-separator.mjs +0 -5
  340. package/esm2022/separator/src/separator.directive.mjs +0 -31
  341. package/esm2022/switch/index.mjs +0 -30
  342. package/esm2022/switch/radix-ng-primitives-switch.mjs +0 -5
  343. package/esm2022/switch/src/switch-input.directive.mjs +0 -32
  344. package/esm2022/switch/src/switch-root.directive.mjs +0 -95
  345. package/esm2022/switch/src/switch-thumb.directive.mjs +0 -23
  346. package/esm2022/tabs/index.mjs +0 -38
  347. package/esm2022/tabs/radix-ng-primitives-tabs.mjs +0 -5
  348. package/esm2022/tabs/src/tabs-content.directive.mjs +0 -29
  349. package/esm2022/tabs/src/tabs-context.service.mjs +0 -43
  350. package/esm2022/tabs/src/tabs-list.directive.mjs +0 -23
  351. package/esm2022/tabs/src/tabs-root.directive.mjs +0 -54
  352. package/esm2022/tabs/src/tabs-trigger.directive.mjs +0 -52
  353. package/esm2022/toggle/index.mjs +0 -2
  354. package/esm2022/toggle/radix-ng-primitives-toggle.mjs +0 -5
  355. package/esm2022/toggle/src/toggle.directive.mjs +0 -39
  356. package/esm2022/toggle-group/index.mjs +0 -6
  357. package/esm2022/toggle-group/radix-ng-primitives-toggle-group.mjs +0 -5
  358. package/esm2022/toggle-group/src/toggle-group-button.directive.mjs +0 -75
  359. package/esm2022/toggle-group/src/toggle-group-button.token.mjs +0 -6
  360. package/esm2022/toggle-group/src/toggle-group-multi.directive.mjs +0 -143
  361. package/esm2022/toggle-group/src/toggle-group.directive.mjs +0 -134
  362. package/esm2022/toggle-group/src/toggle-group.token.mjs +0 -6
  363. package/fesm2022/radix-ng-primitives-accordion.mjs +0 -431
  364. package/fesm2022/radix-ng-primitives-accordion.mjs.map +0 -1
  365. package/fesm2022/radix-ng-primitives-alert-dialog.mjs +0 -161
  366. package/fesm2022/radix-ng-primitives-alert-dialog.mjs.map +0 -1
  367. package/fesm2022/radix-ng-primitives-avatar.mjs +0 -156
  368. package/fesm2022/radix-ng-primitives-avatar.mjs.map +0 -1
  369. package/fesm2022/radix-ng-primitives-checkbox.mjs +0 -241
  370. package/fesm2022/radix-ng-primitives-checkbox.mjs.map +0 -1
  371. package/fesm2022/radix-ng-primitives-collapsible.mjs +0 -213
  372. package/fesm2022/radix-ng-primitives-collapsible.mjs.map +0 -1
  373. package/fesm2022/radix-ng-primitives-dialog.mjs +0 -4
  374. package/fesm2022/radix-ng-primitives-dialog.mjs.map +0 -1
  375. package/fesm2022/radix-ng-primitives-dropdown-menu.mjs +0 -502
  376. package/fesm2022/radix-ng-primitives-dropdown-menu.mjs.map +0 -1
  377. package/fesm2022/radix-ng-primitives-label.mjs +0 -66
  378. package/fesm2022/radix-ng-primitives-label.mjs.map +0 -1
  379. package/fesm2022/radix-ng-primitives-menu.mjs +0 -158
  380. package/fesm2022/radix-ng-primitives-menu.mjs.map +0 -1
  381. package/fesm2022/radix-ng-primitives-menubar.mjs +0 -245
  382. package/fesm2022/radix-ng-primitives-menubar.mjs.map +0 -1
  383. package/fesm2022/radix-ng-primitives-progress.mjs +0 -165
  384. package/fesm2022/radix-ng-primitives-progress.mjs.map +0 -1
  385. package/fesm2022/radix-ng-primitives-radio.mjs +0 -289
  386. package/fesm2022/radix-ng-primitives-radio.mjs.map +0 -1
  387. package/fesm2022/radix-ng-primitives-separator.mjs +0 -38
  388. package/fesm2022/radix-ng-primitives-separator.mjs.map +0 -1
  389. package/fesm2022/radix-ng-primitives-switch.mjs +0 -173
  390. package/fesm2022/radix-ng-primitives-switch.mjs.map +0 -1
  391. package/fesm2022/radix-ng-primitives-tabs.mjs +0 -222
  392. package/fesm2022/radix-ng-primitives-tabs.mjs.map +0 -1
  393. package/fesm2022/radix-ng-primitives-toggle-group.mjs +0 -358
  394. package/fesm2022/radix-ng-primitives-toggle-group.mjs.map +0 -1
  395. package/fesm2022/radix-ng-primitives-toggle.mjs +0 -46
  396. package/fesm2022/radix-ng-primitives-toggle.mjs.map +0 -1
  397. package/fesm2022/radix-ng-primitives.mjs +0 -4
  398. package/fesm2022/radix-ng-primitives.mjs.map +0 -1
  399. package/index.d.ts +0 -1
  400. package/label/src/label.directive.d.ts +0 -29
  401. package/menu/index.d.ts +0 -18
  402. package/menu/src/menu-content.directive.d.ts +0 -6
  403. package/menu/src/menu-directive.d.ts +0 -6
  404. package/menu/src/menu-group.directive.d.ts +0 -6
  405. package/menu/src/menu-item.directive.d.ts +0 -12
  406. package/menu/src/menu-label.directive.d.ts +0 -5
  407. package/menu/src/menu-separator.directive.d.ts +0 -6
  408. package/menubar/index.d.ts +0 -24
  409. package/menubar/src/menubar-content.directive.d.ts +0 -6
  410. package/menubar/src/menubar-item-checkbox.directive.d.ts +0 -14
  411. package/menubar/src/menubar-item-indicator.directive.d.ts +0 -5
  412. package/menubar/src/menubar-item-radio.directive.d.ts +0 -14
  413. package/menubar/src/menubar-item.directive.d.ts +0 -8
  414. package/menubar/src/menubar-radio-group.directive.d.ts +0 -6
  415. package/menubar/src/menubar-root.directive.d.ts +0 -7
  416. package/menubar/src/menubar-separator.directive.d.ts +0 -6
  417. package/menubar/src/menubar-trigger.directive.d.ts +0 -11
  418. package/progress/src/progress-indicator.directive.d.ts +0 -16
  419. package/progress/src/progress-root.directive.d.ts +0 -63
  420. package/radio/src/radio-indicator.directive.d.ts +0 -9
  421. package/radio/src/radio-item.directive.d.ts +0 -21
  422. package/radio/src/radio-root.directive.d.ts +0 -72
  423. package/separator/src/separator.directive.d.ts +0 -22
  424. package/switch/index.d.ts +0 -13
  425. package/switch/src/switch-input.directive.d.ts +0 -6
  426. package/switch/src/switch-root.directive.d.ts +0 -51
  427. package/switch/src/switch-thumb.directive.d.ts +0 -6
  428. package/tabs/index.d.ts +0 -15
  429. package/tabs/src/tabs-content.directive.d.ts +0 -8
  430. package/tabs/src/tabs-context.service.d.ts +0 -22
  431. package/tabs/src/tabs-list.directive.d.ts +0 -6
  432. package/tabs/src/tabs-root.directive.d.ts +0 -37
  433. package/tabs/src/tabs-trigger.directive.d.ts +0 -19
  434. package/toggle/src/toggle.directive.d.ts +0 -30
  435. package/toggle-group/src/toggle-group-button.directive.d.ts +0 -39
  436. package/toggle-group/src/toggle-group-button.token.d.ts +0 -4
  437. package/toggle-group/src/toggle-group-multi.directive.d.ts +0 -93
  438. package/toggle-group/src/toggle-group.directive.d.ts +0 -84
  439. package/toggle-group/src/toggle-group.token.d.ts +0 -5
  440. /package/accordion/{index.d.ts → index.ts} +0 -0
  441. /package/avatar/{index.d.ts → index.ts} +0 -0
  442. /package/checkbox/{index.d.ts → index.ts} +0 -0
  443. /package/collapsible/{index.d.ts → index.ts} +0 -0
  444. /package/dropdown-menu/{index.d.ts → index.ts} +0 -0
  445. /package/{dialog/index.d.ts → index.ts} +0 -0
  446. /package/label/{index.d.ts → index.ts} +0 -0
  447. /package/separator/{index.d.ts → index.ts} +0 -0
  448. /package/toggle/{index.d.ts → index.ts} +0 -0
  449. /package/toggle-group/{index.d.ts → index.ts} +0 -0
@@ -0,0 +1,27 @@
1
+ import { DialogModule } from '@angular/cdk/dialog';
2
+ import { EnvironmentProviders, importProvidersFrom, makeEnvironmentProviders, Provider } from '@angular/core';
3
+ import { RdxDialogService } from './dialog.service';
4
+
5
+ /**
6
+ * Configures the RdxDialog module by providing necessary dependencies.
7
+ *
8
+ * This function sets up the environment providers required for the RdxDialog to function,
9
+ * specifically importing the Angular CDK's DialogModule.
10
+ *
11
+ * @returns {EnvironmentProviders} An EnvironmentProviders instance containing the DialogModule.
12
+ */
13
+ export function configureRdxDialog(): EnvironmentProviders {
14
+ return makeEnvironmentProviders([importProvidersFrom(DialogModule)]);
15
+ }
16
+
17
+ /**
18
+ * Provides the RdxDialogService for dependency injection.
19
+ *
20
+ * This function is used to make the RdxDialogService available for injection
21
+ * in components, directives, or other services that require dialog functionality.
22
+ *
23
+ * @returns {Provider} A provider for the RdxDialogService.
24
+ */
25
+ export function provideRdxDialog(): Provider {
26
+ return RdxDialogService;
27
+ }
@@ -0,0 +1,94 @@
1
+ import { Dialog } from '@angular/cdk/dialog';
2
+ import { inject, Injectable, Injector, Renderer2 } from '@angular/core';
3
+ import { filter, isObservable, merge, of, switchMap, take, takeUntil } from 'rxjs';
4
+ import { DISMISSED_VALUE, RdxDialogRef } from './dialog-ref';
5
+ import type { RdxDialogConfig, RdxDialogResult } from './dialog.config';
6
+
7
+ /**
8
+ * Modality control: When `isModal` is set to `true`, the dialog will:
9
+ *
10
+ * - Have a backdrop that blocks interaction with the rest of the page
11
+ * - Disable closing by clicking outside or pressing Escape
12
+ * - Set `aria-modal="true"` for screen readers
13
+ * - Automatically focus the first tabbable element in the dialog
14
+ * - Restore focus to the element that opened the dialog when it's closed
15
+ *
16
+ *
17
+ * When `isModal` is `false`, the dialog will:
18
+ *
19
+ * - Not have a backdrop, allowing interaction with the rest of the page
20
+ * - Allow closing by clicking outside or pressing Escape
21
+ * - Not set `aria-modal` attribute
22
+ * - Not automatically manage focus
23
+ */
24
+ @Injectable()
25
+ export class RdxDialogService {
26
+ #cdkDialog = inject(Dialog);
27
+ #injector = inject(Injector);
28
+
29
+ open<C>(config: RdxDialogConfig<C>): RdxDialogRef<C> {
30
+ let dialogRef: RdxDialogRef<C>;
31
+ let modeClasses: string[] = [];
32
+
33
+ switch (config.mode) {
34
+ case 'drawer':
35
+ modeClasses = ['mod-drawer'];
36
+ break;
37
+ case 'drawer-bottom':
38
+ modeClasses = ['mod-drawer', 'mod-bottom'];
39
+ break;
40
+ }
41
+
42
+ const cdkRef = this.#cdkDialog.open<RdxDialogResult<C> | typeof DISMISSED_VALUE, unknown, C>(config.content, {
43
+ ariaModal: config.modal ?? true,
44
+ hasBackdrop: config.modal ?? true,
45
+ data: 'data' in config ? config.data : null,
46
+ restoreFocus: true,
47
+ role: 'dialog',
48
+ disableClose: true,
49
+ closeOnDestroy: true,
50
+ injector: this.#injector,
51
+ backdropClass: config.backdropClass ? config.backdropClass : 'cdk-overlay-dark-backdrop',
52
+ panelClass: ['dialog', ...modeClasses, ...(config.panelClasses || [])],
53
+ autoFocus: config.autoFocus === 'first-input' ? 'dialog' : (config.autoFocus ?? 'first-tabbable'),
54
+ ariaLabel: config.ariaLabel,
55
+ templateContext: () => ({ dialogRef: dialogRef }),
56
+ providers: (ref) => {
57
+ dialogRef = new RdxDialogRef(ref, config);
58
+ return [
59
+ {
60
+ provide: RdxDialogRef,
61
+ useValue: dialogRef
62
+ }
63
+ ];
64
+ },
65
+ // @FIXME
66
+ ...(config.cdkConfigOverride || ({} as any))
67
+ });
68
+
69
+ if (cdkRef.componentRef) {
70
+ cdkRef.componentRef.injector
71
+ .get(Renderer2)
72
+ .setStyle(cdkRef.componentRef.location.nativeElement, 'display', 'contents');
73
+ }
74
+
75
+ merge(cdkRef.backdropClick, cdkRef.keydownEvents.pipe(filter((e) => e.key === 'Escape' && !e.defaultPrevented)))
76
+ .pipe(
77
+ filter(() => config.canCloseWithBackdrop ?? true),
78
+ switchMap(() => {
79
+ const canClose = (cdkRef.componentInstance && config.canClose?.(cdkRef.componentInstance)) ?? true;
80
+ const canClose$ = isObservable(canClose) ? canClose : of(canClose);
81
+ return canClose$.pipe(take(1));
82
+ }),
83
+
84
+ takeUntil(dialogRef!.closed$)
85
+ )
86
+ .subscribe((canClose) => {
87
+ if (canClose) {
88
+ cdkRef.close(DISMISSED_VALUE);
89
+ }
90
+ });
91
+
92
+ return dialogRef!;
93
+ }
94
+ }
@@ -0,0 +1,32 @@
1
+ import { Canvas, Meta } from '@storybook/blocks';
2
+ import * as DialogStories from './dialog.stories';
3
+
4
+ <Meta title="Primitives/Dialog" />
5
+
6
+ # Dialog
7
+
8
+ <Canvas sourceState="hidden" of={DialogStories.Default} />
9
+
10
+ ## Features
11
+ - ✅ Supports modal and non-modal modes.
12
+ - ✅ Focus is automatically trapped when modal.
13
+ - ✅ Can be controlled or uncontrolled.
14
+ - ✅ Esc closes the component automatically.
15
+
16
+ ### Anatomy
17
+ Import all parts and piece them together.
18
+
19
+ ```html
20
+
21
+ <button [rdxDialogTrigger]="dialog">Open Dialog</button>
22
+
23
+ <ng-template #dialog>
24
+ <div rdxDialogContent>
25
+ <h2 rdxDialogTitle></h2>
26
+ <p rdxDialogDescription></p>
27
+ <button rdxDialogClose></button>
28
+ <button rdxDialogDismiss>X</button>
29
+ </div>
30
+ </ng-template>
31
+
32
+ ```
@@ -0,0 +1,233 @@
1
+ import { applicationConfig, componentWrapperDecorator, Meta, moduleMetadata, StoryObj } from '@storybook/angular';
2
+ import { RdxDialogCloseDirective } from '../src/dialog-close.directive';
3
+ import { RdxDialogContentDirective } from '../src/dialog-content.directive';
4
+ import { RdxDialogDescriptionDirective } from '../src/dialog-description.directive';
5
+ import { RdxDialogTitleDirective } from '../src/dialog-title.directive';
6
+ import { RdxDialogTriggerDirective } from '../src/dialog-trigger.directive';
7
+ import { configureRdxDialog } from '../src/dialog.providers';
8
+
9
+ const html = String.raw;
10
+
11
+ export default {
12
+ title: 'Primitives/Dialog',
13
+ decorators: [
14
+ applicationConfig({
15
+ providers: [configureRdxDialog()]
16
+ }),
17
+ moduleMetadata({
18
+ imports: [
19
+ RdxDialogTriggerDirective,
20
+ RdxDialogContentDirective,
21
+ RdxDialogTitleDirective,
22
+ RdxDialogCloseDirective,
23
+ RdxDialogDescriptionDirective
24
+ ]
25
+ }),
26
+ componentWrapperDecorator(
27
+ (story) => `
28
+ <div class="radix-themes light light-theme radix-themes-default-fonts rt-Flex rt-r-ai-start rt-r-jc-center rt-r-position-relative"
29
+ data-accent-color="indigo"
30
+ data-radius="medium"
31
+ data-scaling="100%"
32
+ >
33
+ ${story}
34
+ </div>`
35
+ )
36
+ ]
37
+ } as Meta;
38
+
39
+ type Story = StoryObj;
40
+
41
+ export const Default: Story = {
42
+ argTypes: {
43
+ mode: {
44
+ options: ['default', 'drawer', 'drawer-bottom'],
45
+ control: {
46
+ type: 'select'
47
+ }
48
+ }
49
+ },
50
+ render: (args) => ({
51
+ props: {
52
+ config: args
53
+ },
54
+ template: html`
55
+ <button class="Button violet" [rdxDialogConfig]="config" [rdxDialogTrigger]="dialog">Open Dialog</button>
56
+
57
+ <ng-template #dialog>
58
+ <div class="DialogContent" rdxDialogContent>
59
+ <h2 class="DialogTitle" rdxDialogTitle>Edit profile</h2>
60
+ <p class="DialogDescription" rdxDialogDescription>
61
+ Make changes to your profile here. Click save when you're done.
62
+ </p>
63
+ <fieldset class="Fieldset">
64
+ <label class="Label" htmlFor="name">Name</label>
65
+ <input class="Input" id="name" defaultValue="Pedro Duarte" />
66
+ </fieldset>
67
+ <fieldset class="Fieldset">
68
+ <label class="Label" htmlFor="username">Username</label>
69
+ <input class="Input" id="username" defaultValue="@peduarte" />
70
+ </fieldset>
71
+ <div style="display:flex; margin-top: 25px; justify-content: flex-end;">
72
+ <button class="Button green" rdxDialogClose>Save changes</button>
73
+ </div>
74
+ <button class="IconButton" rdxDialogClose aria-label="Close">X</button>
75
+ </div>
76
+ </ng-template>
77
+
78
+ <style>
79
+ /* reset */
80
+ button,
81
+ fieldset,
82
+ input {
83
+ all: unset;
84
+ }
85
+
86
+ .DialogOverlay {
87
+ background-color: var(--black-a9);
88
+ position: fixed;
89
+ inset: 0;
90
+ animation: overlayShow 150ms cubic-bezier(0.16, 1, 0.3, 1);
91
+ }
92
+
93
+ .DialogContent {
94
+ background-color: white;
95
+ border-radius: 6px;
96
+ box-shadow:
97
+ hsl(206 22% 7% / 35%) 0px 10px 38px -10px,
98
+ hsl(206 22% 7% / 20%) 0px 10px 20px -15px;
99
+ position: fixed;
100
+ top: 50%;
101
+ left: 50%;
102
+ transform: translate(-50%, -50%);
103
+ width: 90vw;
104
+ max-width: 450px;
105
+ max-height: 85vh;
106
+ padding: 25px;
107
+ animation: contentShow 150ms cubic-bezier(0.16, 1, 0.3, 1);
108
+ }
109
+
110
+ .DialogContent:focus {
111
+ outline: none;
112
+ }
113
+
114
+ .DialogTitle {
115
+ margin: 0;
116
+ font-weight: 500;
117
+ color: var(--mauve-12);
118
+ font-size: 17px;
119
+ }
120
+
121
+ .DialogDescription {
122
+ margin: 10px 0 20px;
123
+ color: var(--mauve-11);
124
+ font-size: 15px;
125
+ line-height: 1.5;
126
+ }
127
+
128
+ .Button {
129
+ display: inline-flex;
130
+ align-items: center;
131
+ justify-content: center;
132
+ border-radius: 4px;
133
+ padding: 0 15px;
134
+ font-size: 15px;
135
+ line-height: 1;
136
+ font-weight: 500;
137
+ height: 35px;
138
+ }
139
+
140
+ .Button.violet {
141
+ background-color: white;
142
+ color: var(--violet-11);
143
+ box-shadow: 0 2px 10px var(--black-a7);
144
+ }
145
+
146
+ .Button.violet:hover {
147
+ background-color: var(--mauve-3);
148
+ }
149
+
150
+ .Button.violet:focus {
151
+ box-shadow: 0 0 0 2px black;
152
+ }
153
+
154
+ .Button.green {
155
+ background-color: var(--green-4);
156
+ color: var(--green-11);
157
+ }
158
+
159
+ .Button.green:hover {
160
+ background-color: var(--green-5);
161
+ }
162
+
163
+ .Button.green:focus {
164
+ box-shadow: 0 0 0 2px var(--green-7);
165
+ }
166
+
167
+ .IconButton {
168
+ font-family: inherit;
169
+ border-radius: 100%;
170
+ height: 25px;
171
+ width: 25px;
172
+ display: inline-flex;
173
+ align-items: center;
174
+ justify-content: center;
175
+ color: var(--violet-11);
176
+ position: absolute;
177
+ top: 10px;
178
+ right: 10px;
179
+ }
180
+
181
+ .IconButton:hover {
182
+ background-color: var(--violet-4);
183
+ }
184
+
185
+ .IconButton:focus {
186
+ box-shadow: 0 0 0 2px var(--violet-7);
187
+ }
188
+
189
+ .Fieldset {
190
+ display: flex;
191
+ gap: 20px;
192
+ align-items: center;
193
+ margin-bottom: 15px;
194
+ }
195
+
196
+ .Label {
197
+ font-size: 15px;
198
+ color: var(--violet-11);
199
+ width: 90px;
200
+ text-align: right;
201
+ }
202
+
203
+ .Input {
204
+ width: 100%;
205
+ flex: 1;
206
+ display: inline-flex;
207
+ align-items: center;
208
+ justify-content: center;
209
+ border-radius: 4px;
210
+ padding: 0 10px;
211
+ font-size: 15px;
212
+ line-height: 1;
213
+ color: var(--violet-11);
214
+ box-shadow: 0 0 0 1px var(--violet-7);
215
+ height: 35px;
216
+ }
217
+
218
+ .Input:focus {
219
+ box-shadow: 0 0 0 2px var(--violet-8);
220
+ }
221
+ </style>
222
+ `
223
+ })
224
+ };
225
+
226
+ export const DrawerStory: Story = {
227
+ render: (args) => ({
228
+ props: {
229
+ config: args
230
+ },
231
+ template: html``
232
+ })
233
+ };
@@ -0,0 +1,5 @@
1
+ {
2
+ "lib": {
3
+ "entryFile": "index.ts"
4
+ }
5
+ }
@@ -0,0 +1,47 @@
1
+ import { CdkMenu, CdkMenuItem } from '@angular/cdk/menu';
2
+ import { Directive, inject, Input } from '@angular/core';
3
+ import { pairwise, startWith, Subject } from 'rxjs';
4
+ import { RdxDropdownMenuItemDirective } from './dropdown-menu-item.directive';
5
+ import { RdxDropdownMenuTriggerDirective } from './dropdown-menu-trigger.directive';
6
+
7
+ @Directive({
8
+ selector: '[rdxDropdownMenuContent]',
9
+ standalone: true,
10
+ host: {
11
+ '[attr.data-state]': "menuTrigger.isOpen() ? 'open': 'closed'",
12
+ '[attr.data-align]': 'menuTrigger!.align',
13
+ '[attr.data-side]': 'menuTrigger!.side',
14
+ '[attr.data-orientation]': 'orientation'
15
+ },
16
+ providers: [
17
+ {
18
+ provide: CdkMenu,
19
+ useExisting: RdxDropdownMenuContentDirective
20
+ }
21
+ ]
22
+ })
23
+ export class RdxDropdownMenuContentDirective extends CdkMenu {
24
+ readonly highlighted = new Subject<RdxDropdownMenuItemDirective>();
25
+ readonly menuTrigger = inject(RdxDropdownMenuTriggerDirective, { optional: true });
26
+
27
+ @Input() onEscapeKeyDown: (event?: Event) => void = () => undefined;
28
+ @Input() closeOnEscape: boolean = true;
29
+
30
+ constructor() {
31
+ super();
32
+
33
+ this.highlighted.pipe(startWith(null), pairwise()).subscribe(([prev, item]) => {
34
+ if (prev) {
35
+ prev.highlighted = false;
36
+ }
37
+
38
+ if (item) {
39
+ item.highlighted = true;
40
+ }
41
+ });
42
+ }
43
+
44
+ updateActiveItem(item: CdkMenuItem) {
45
+ this.keyManager.updateActiveItem(item);
46
+ }
47
+ }
@@ -0,0 +1,30 @@
1
+ import { CDK_MENU, CdkMenuItem } from '@angular/cdk/menu';
2
+ import { Directive } from '@angular/core';
3
+ import { RdxDropdownMenuContentDirective } from './dropdown-menu-content.directive';
4
+ import { RdxDropdownMenuSelectable } from './dropdown-menu-item-selectable';
5
+ import { RdxDropdownMenuItemDirective } from './dropdown-menu-item.directive';
6
+
7
+ @Directive({
8
+ selector: '[rdxDropdownMenuItemCheckbox]',
9
+ standalone: true,
10
+ host: {
11
+ role: 'menuitemcheckbox'
12
+ },
13
+ providers: [
14
+ { provide: RdxDropdownMenuSelectable, useExisting: RdxDropdownMenuItemCheckboxDirective },
15
+ { provide: RdxDropdownMenuItemDirective, useExisting: RdxDropdownMenuSelectable },
16
+ { provide: CdkMenuItem, useExisting: RdxDropdownMenuItemDirective },
17
+ { provide: CDK_MENU, useExisting: RdxDropdownMenuContentDirective }
18
+ ]
19
+ })
20
+ export class RdxDropdownMenuItemCheckboxDirective extends RdxDropdownMenuSelectable {
21
+ override trigger(options?: { keepOpen: boolean }) {
22
+ if (!this.disabled) {
23
+ this.checked = !this.checked;
24
+
25
+ this.checkedChange.emit(this.checked);
26
+ }
27
+
28
+ super.trigger(options);
29
+ }
30
+ }
@@ -0,0 +1,14 @@
1
+ import { Directive, inject } from '@angular/core';
2
+ import { RdxDropdownMenuSelectable } from './dropdown-menu-item-selectable';
3
+
4
+ @Directive({
5
+ selector: '[rdxDropdownMenuItemIndicator]',
6
+ standalone: true,
7
+ host: {
8
+ '[style.display]': "item.checked ? 'block' : 'none'",
9
+ '[attr.data-state]': "item.checked ? 'checked' : 'unchecked'"
10
+ }
11
+ })
12
+ export class RdxDropdownMenuItemIndicatorDirective {
13
+ item = inject(RdxDropdownMenuSelectable);
14
+ }
@@ -0,0 +1,31 @@
1
+ import { UniqueSelectionDispatcher } from '@angular/cdk/collections';
2
+ import { AfterContentInit, Directive, EventEmitter, inject, Input, Output } from '@angular/core';
3
+
4
+ @Directive({
5
+ selector: '[rdxDropdownMenuItemRadioGroup]',
6
+ standalone: true,
7
+ host: {
8
+ role: 'group'
9
+ },
10
+ providers: [{ provide: UniqueSelectionDispatcher, useClass: UniqueSelectionDispatcher }]
11
+ })
12
+ export class RdxDropdownMenuItemRadioGroupDirective<T> implements AfterContentInit {
13
+ private readonly selectionDispatcher = inject(UniqueSelectionDispatcher);
14
+
15
+ @Input()
16
+ set value(id: T | null) {
17
+ this._value = id;
18
+ }
19
+
20
+ get value(): T | null {
21
+ return this._value;
22
+ }
23
+
24
+ private _value: T | null = null;
25
+
26
+ @Output() readonly valueChange = new EventEmitter();
27
+
28
+ ngAfterContentInit(): void {
29
+ this.selectionDispatcher.notify(this.value as string, '');
30
+ }
31
+ }
@@ -0,0 +1,72 @@
1
+ import { UniqueSelectionDispatcher } from '@angular/cdk/collections';
2
+ import { CDK_MENU, CdkMenuItem } from '@angular/cdk/menu';
3
+ import { AfterContentInit, Directive, inject, Input, OnDestroy } from '@angular/core';
4
+ import { RdxDropdownMenuContentDirective } from './dropdown-menu-content.directive';
5
+ import { RdxDropdownMenuItemRadioGroupDirective } from './dropdown-menu-item-radio-group.directive';
6
+ import { RdxDropdownMenuSelectable } from './dropdown-menu-item-selectable';
7
+ import { RdxDropdownMenuItemDirective } from './dropdown-menu-item.directive';
8
+
9
+ /** Counter used to set a unique id and name for a selectable item */
10
+ let nextId = 0;
11
+
12
+ @Directive({
13
+ selector: '[rdxDropdownMenuItemRadio]',
14
+ standalone: true,
15
+ host: {
16
+ role: 'menuitemradio'
17
+ },
18
+ providers: [
19
+ { provide: RdxDropdownMenuSelectable, useExisting: RdxDropdownMenuItemRadioDirective },
20
+ { provide: RdxDropdownMenuItemDirective, useExisting: RdxDropdownMenuSelectable },
21
+ { provide: CdkMenuItem, useExisting: RdxDropdownMenuItemDirective },
22
+ { provide: CDK_MENU, useExisting: RdxDropdownMenuContentDirective }
23
+ ]
24
+ })
25
+ export class RdxDropdownMenuItemRadioDirective
26
+ extends RdxDropdownMenuSelectable
27
+ implements AfterContentInit, OnDestroy
28
+ {
29
+ /** The unique selection dispatcher for this radio's `RdxDropdownMenuItemRadioGroupDirective`. */
30
+ private readonly selectionDispatcher = inject(UniqueSelectionDispatcher);
31
+
32
+ private readonly group = inject(RdxDropdownMenuItemRadioGroupDirective);
33
+
34
+ @Input()
35
+ get value() {
36
+ return this._value || this.id;
37
+ }
38
+
39
+ set value(value: string) {
40
+ this._value = value;
41
+ }
42
+
43
+ private _value: string | undefined;
44
+
45
+ /** An ID to identify this radio item to the `UniqueSelectionDispatcher`. */
46
+ private id = `${nextId++}`;
47
+
48
+ private removeDispatcherListener!: () => void;
49
+
50
+ constructor() {
51
+ super();
52
+
53
+ this.triggered.subscribe(() => {
54
+ if (!this.disabled) {
55
+ this.selectionDispatcher.notify(this.value, '');
56
+
57
+ this.group.valueChange.emit(this.value);
58
+ }
59
+ });
60
+ }
61
+
62
+ ngAfterContentInit() {
63
+ this.removeDispatcherListener = this.selectionDispatcher.listen((id: string) => {
64
+ this.checked = this.value === id;
65
+ });
66
+ }
67
+
68
+ override ngOnDestroy() {
69
+ super.ngOnDestroy();
70
+ this.removeDispatcherListener();
71
+ }
72
+ }
@@ -0,0 +1,18 @@
1
+ import { booleanAttribute, Directive, EventEmitter, Input, Output } from '@angular/core';
2
+ import { RdxDropdownMenuItemDirective } from './dropdown-menu-item.directive';
3
+
4
+ /** Base class providing checked state for selectable DropdownMenuItems. */
5
+ @Directive({
6
+ standalone: true,
7
+ host: {
8
+ '[attr.aria-checked]': '!!checked',
9
+ '[attr.aria-disabled]': 'disabled || null',
10
+ '[attr.data-state]': 'checked ? "checked" : "unchecked"'
11
+ }
12
+ })
13
+ export class RdxDropdownMenuSelectable extends RdxDropdownMenuItemDirective {
14
+ /** Whether the element is checked */
15
+ @Input({ transform: booleanAttribute }) checked: boolean = false;
16
+
17
+ @Output() readonly checkedChange = new EventEmitter<boolean>();
18
+ }
@@ -0,0 +1,66 @@
1
+ import { CDK_MENU, CdkMenuItem } from '@angular/cdk/menu';
2
+ import { booleanAttribute, Directive, ElementRef, EventEmitter, inject, Input, Output } from '@angular/core';
3
+ import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
4
+
5
+ import { RdxDropdownMenuContentDirective } from './dropdown-menu-content.directive';
6
+
7
+ @Directive({
8
+ selector: '[rdxDropdownMenuItem]',
9
+ standalone: true,
10
+ host: {
11
+ type: 'button',
12
+ // todo horizontal ?
13
+ '[attr.data-orientation]': '"vertical"',
14
+ '[attr.data-highlighted]': 'highlighted ? "" : null',
15
+ '[attr.data-disabled]': 'disabled ? "" : null',
16
+ '[attr.disabled]': 'disabled ? "" : null',
17
+ '(pointermove)': 'onPointerMove()',
18
+ '(focus)': 'menu.highlighted.next(this)',
19
+ '(keydown)': 'onKeydown($event)'
20
+ },
21
+ providers: [
22
+ { provide: CdkMenuItem, useExisting: RdxDropdownMenuItemDirective },
23
+ { provide: CDK_MENU, useExisting: RdxDropdownMenuContentDirective }
24
+ ]
25
+ })
26
+ export class RdxDropdownMenuItemDirective extends CdkMenuItem {
27
+ protected readonly menu = inject(RdxDropdownMenuContentDirective);
28
+ protected readonly nativeElement = inject(ElementRef).nativeElement;
29
+
30
+ highlighted = false;
31
+
32
+ @Input({ transform: booleanAttribute }) override disabled: boolean = false;
33
+
34
+ @Output() readonly onSelect = new EventEmitter<void>();
35
+
36
+ constructor() {
37
+ super();
38
+
39
+ this.menu.highlighted.pipe(takeUntilDestroyed()).subscribe((value) => {
40
+ if (value !== this) {
41
+ this.highlighted = false;
42
+ }
43
+ });
44
+
45
+ this.triggered.subscribe(this.onSelect);
46
+ }
47
+
48
+ protected onPointerMove() {
49
+ this.nativeElement.focus({ preventScroll: true });
50
+ this.menu.updateActiveItem(this);
51
+ }
52
+
53
+ protected onKeydown(event: KeyboardEvent) {
54
+ if (this.nativeElement.tagName !== 'BUTTON' && ['Enter', ' '].includes(event.key)) {
55
+ event.preventDefault();
56
+ }
57
+
58
+ if (event.key === 'Escape') {
59
+ if (!this.menu.closeOnEscape) {
60
+ event.stopPropagation();
61
+ } else {
62
+ this.menu.onEscapeKeyDown(event);
63
+ }
64
+ }
65
+ }
66
+ }
@@ -0,0 +1,7 @@
1
+ import { Directive } from '@angular/core';
2
+
3
+ @Directive({
4
+ selector: '[rdxDropdownMenuLabel]',
5
+ standalone: true
6
+ })
7
+ export class RdxDropdownMenuLabelDirective {}