@grantcodes/ui 2.0.0-beta4 → 2.0.2

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 (310) hide show
  1. package/CHANGELOG.md +125 -0
  2. package/README.md +6 -8
  3. package/custom-elements.json +4273 -0
  4. package/package.json +79 -88
  5. package/src/components/app-bar/app-bar.component.js +90 -0
  6. package/src/components/app-bar/app-bar.js +8 -0
  7. package/src/components/app-bar/app-bar.stories.js +84 -0
  8. package/src/components/app-bar/app-bar.styles.js +227 -0
  9. package/src/components/app-bar/app-bar.test.js +174 -0
  10. package/src/components/app-bar/index.js +3 -0
  11. package/src/components/avatar/avatar.component.js +78 -0
  12. package/src/components/avatar/avatar.js +18 -0
  13. package/src/components/avatar/avatar.stories.js +45 -0
  14. package/src/components/avatar/avatar.styles.js +42 -0
  15. package/src/components/avatar/avatar.test.js +85 -0
  16. package/src/components/avatar/index.js +13 -0
  17. package/src/components/badge/badge.component.js +36 -0
  18. package/src/components/badge/badge.js +8 -0
  19. package/src/components/badge/badge.stories.js +46 -0
  20. package/src/components/badge/badge.styles.js +94 -0
  21. package/src/components/badge/badge.test.js +70 -0
  22. package/src/components/badge/index.js +3 -0
  23. package/src/components/breadcrumb/breadcrumb.component.js +110 -0
  24. package/src/components/breadcrumb/breadcrumb.js +12 -0
  25. package/src/components/breadcrumb/breadcrumb.stories.js +25 -0
  26. package/src/components/breadcrumb/breadcrumb.styles.js +96 -0
  27. package/src/components/breadcrumb/breadcrumb.test.js +144 -0
  28. package/src/components/breadcrumb/index.js +3 -0
  29. package/src/components/button/button.component.js +64 -0
  30. package/src/components/button/button.js +6 -0
  31. package/src/components/button/button.stories.js +78 -0
  32. package/src/components/button/button.styles.js +97 -0
  33. package/src/components/button/button.test.js +98 -0
  34. package/src/components/button/index.js +1 -0
  35. package/src/components/button-group/button-group.component.js +27 -0
  36. package/src/components/button-group/button-group.js +6 -0
  37. package/src/components/button-group/button-group.stories.js +33 -0
  38. package/src/components/button-group/button-group.styles.js +43 -0
  39. package/src/components/button-group/button-group.test.js +57 -0
  40. package/src/components/button-group/index.js +1 -0
  41. package/src/components/card/card.component.js +17 -0
  42. package/src/components/card/card.js +6 -0
  43. package/src/components/card/card.stories.js +36 -0
  44. package/src/components/card/card.styles.js +128 -0
  45. package/src/components/card/card.test.js +59 -0
  46. package/src/components/card/index.js +1 -0
  47. package/src/components/code-preview/code-preview.component.js +53 -0
  48. package/src/components/code-preview/code-preview.js +7 -0
  49. package/src/components/code-preview/code-preview.stories.js +67 -0
  50. package/src/components/code-preview/code-preview.styles.js +18 -0
  51. package/src/components/code-preview/code-preview.test.js +118 -0
  52. package/src/components/code-preview/index.js +1 -0
  53. package/src/components/container/container.component.js +38 -0
  54. package/src/components/container/container.js +7 -0
  55. package/src/components/container/container.stories.js +59 -0
  56. package/src/components/container/container.styles.js +43 -0
  57. package/src/components/container/container.test.js +84 -0
  58. package/src/components/container/index.js +1 -0
  59. package/src/components/dialog/dialog.component.js +78 -0
  60. package/src/components/dialog/dialog.js +7 -0
  61. package/src/components/dialog/dialog.stories.js +43 -0
  62. package/src/components/dialog/dialog.styles.js +74 -0
  63. package/src/components/dialog/dialog.test.js +97 -0
  64. package/src/components/dialog/index.js +1 -0
  65. package/src/components/dropdown/dropdown.component.js +225 -0
  66. package/src/components/dropdown/dropdown.js +12 -0
  67. package/src/components/dropdown/dropdown.stories.js +107 -0
  68. package/src/components/dropdown/dropdown.styles.js +128 -0
  69. package/src/components/dropdown/dropdown.test.js +144 -0
  70. package/src/components/dropdown/index.js +3 -0
  71. package/src/components/dropzone/dropzone.component.js +141 -0
  72. package/src/components/dropzone/dropzone.js +6 -0
  73. package/src/components/dropzone/dropzone.stories.js +41 -0
  74. package/src/components/dropzone/dropzone.styles.js +64 -0
  75. package/src/components/dropzone/dropzone.test.js +112 -0
  76. package/src/components/dropzone/index.js +1 -0
  77. package/src/components/footer/footer-column.component.js +15 -0
  78. package/src/components/footer/footer-column.styles.js +51 -0
  79. package/src/components/footer/footer.component.js +38 -0
  80. package/src/components/footer/footer.js +9 -0
  81. package/src/components/footer/footer.stories.js +143 -0
  82. package/src/components/footer/footer.styles.js +90 -0
  83. package/src/components/footer/footer.test.js +107 -0
  84. package/src/components/footer/index.js +2 -0
  85. package/src/components/form-field/form-field.component.js +173 -0
  86. package/src/components/form-field/form-field.js +7 -0
  87. package/src/components/form-field/form-field.stories.js +104 -0
  88. package/src/components/form-field/form-field.styles.js +47 -0
  89. package/src/components/form-field/form-field.test.js +118 -0
  90. package/src/components/form-field/index.js +1 -0
  91. package/src/components/gallery/gallery-image.component.js +52 -0
  92. package/src/components/gallery/gallery-image.js +7 -0
  93. package/src/components/gallery/gallery.component.js +25 -0
  94. package/src/components/gallery/gallery.js +7 -0
  95. package/src/components/gallery/gallery.stories.js +60 -0
  96. package/src/components/gallery/gallery.styles.js +56 -0
  97. package/src/components/gallery/gallery.test.js +58 -0
  98. package/src/components/gallery/index.js +2 -0
  99. package/src/components/icon/icon.component.js +14 -0
  100. package/src/components/icon/icon.js +7 -0
  101. package/src/components/icon/icon.stories.js +26 -0
  102. package/src/components/icon/icon.styles.js +28 -0
  103. package/src/components/icon/icon.test.js +57 -0
  104. package/src/components/icon/index.js +1 -0
  105. package/src/components/loading/index.js +1 -0
  106. package/src/components/loading/loading.component.js +21 -0
  107. package/src/components/loading/loading.js +7 -0
  108. package/src/components/loading/loading.stories.js +25 -0
  109. package/src/components/loading/loading.styles.js +43 -0
  110. package/src/components/loading/loading.test.js +57 -0
  111. package/src/components/notice/index.js +1 -0
  112. package/src/components/notice/notice.component.js +77 -0
  113. package/src/components/notice/notice.js +7 -0
  114. package/src/components/notice/notice.stories.js +122 -0
  115. package/src/components/notice/notice.styles.js +72 -0
  116. package/src/components/notice/notice.test.js +146 -0
  117. package/src/components/pagination/index.js +1 -0
  118. package/src/components/pagination/pagination.component.js +62 -0
  119. package/src/components/pagination/pagination.js +7 -0
  120. package/src/components/pagination/pagination.stories.js +34 -0
  121. package/src/components/pagination/pagination.styles.js +19 -0
  122. package/src/components/pagination/pagination.test.js +98 -0
  123. package/src/components/sidebar/index.js +3 -0
  124. package/src/components/sidebar/sidebar.component.js +165 -0
  125. package/src/components/sidebar/sidebar.js +8 -0
  126. package/src/components/sidebar/sidebar.stories.js +155 -0
  127. package/src/components/sidebar/sidebar.styles.js +192 -0
  128. package/src/components/sidebar/sidebar.test.js +196 -0
  129. package/src/components/tabs/index.js +2 -0
  130. package/src/components/tabs/internal/tabs-button.component.js +39 -0
  131. package/src/components/tabs/internal/tabs-button.js +7 -0
  132. package/src/components/tabs/internal/tabs-item.component.js +39 -0
  133. package/src/components/tabs/tab.component.js +20 -0
  134. package/src/components/tabs/tab.js +7 -0
  135. package/src/components/tabs/tabs.component.js +130 -0
  136. package/src/components/tabs/tabs.js +7 -0
  137. package/src/components/tabs/tabs.stories.js +39 -0
  138. package/src/components/tabs/tabs.styles.js +88 -0
  139. package/src/components/tabs/tabs.test.js +148 -0
  140. package/src/components/toast/index.js +3 -0
  141. package/src/components/toast/toast.component.js +187 -0
  142. package/src/components/toast/toast.js +9 -0
  143. package/src/components/toast/toast.stories.js +169 -0
  144. package/src/components/toast/toast.styles.js +207 -0
  145. package/src/components/toast/toast.test.js +196 -0
  146. package/src/components/tooltip/index.js +1 -0
  147. package/src/components/tooltip/tooltip.component.js +70 -0
  148. package/src/components/tooltip/tooltip.js +7 -0
  149. package/src/components/tooltip/tooltip.stories.js +33 -0
  150. package/src/components/tooltip/tooltip.styles.js +78 -0
  151. package/src/components/tooltip/tooltip.test.js +150 -0
  152. package/src/css/all.css +1 -0
  153. package/src/css/base.css +31 -0
  154. package/src/css/colors.stories.js +192 -0
  155. package/src/css/elements/a.css +50 -0
  156. package/src/css/elements/forms/button.css +15 -0
  157. package/src/css/elements/forms/input.css +183 -0
  158. package/src/css/elements/forms/label.css +5 -0
  159. package/src/css/elements/media/image.css +3 -0
  160. package/src/css/elements.css +5 -0
  161. package/src/css/elements.stories.js +108 -0
  162. package/src/css/helpers.css +16 -0
  163. package/src/css/themes/grantcodes.css +3 -0
  164. package/src/css/themes/todomap.css +2 -0
  165. package/src/css/themes/wireframe.css +2 -0
  166. package/src/css/tokens.stories.js +183 -0
  167. package/src/css/typography.css +64 -0
  168. package/src/css/typography.stories.js +179 -0
  169. package/src/css/util/functions.css +16 -0
  170. package/src/css/util/index.css +2 -0
  171. package/src/css/util/mixins.css +63 -0
  172. package/src/icons.js +3 -0
  173. package/src/lib/classnames.js +61 -0
  174. package/src/lib/generate-id.js +10 -0
  175. package/src/main.js +17 -0
  176. package/src/test-utils/assert-helpers.js +150 -0
  177. package/src/test-utils/events.js +88 -0
  178. package/src/test-utils/fixture.js +77 -0
  179. package/src/test-utils/index.js +7 -0
  180. package/dist/components/avatar/avatar.component.js +0 -3
  181. package/dist/components/avatar/avatar.js +0 -1
  182. package/dist/components/avatar/avatar.react.js +0 -1
  183. package/dist/components/avatar/avatar.scss.js +0 -1
  184. package/dist/components/avatar/index.js +0 -1
  185. package/dist/components/button/button.component.js +0 -5
  186. package/dist/components/button/button.js +0 -1
  187. package/dist/components/button/button.react.js +0 -1
  188. package/dist/components/button/button.scss.js +0 -1
  189. package/dist/components/button/index.js +0 -1
  190. package/dist/components/button-group/button-group.component.js +0 -5
  191. package/dist/components/button-group/button-group.js +0 -1
  192. package/dist/components/button-group/button-group.react.js +0 -1
  193. package/dist/components/button-group/button-group.scss.js +0 -1
  194. package/dist/components/button-group/index.js +0 -1
  195. package/dist/components/card/card.component.js +0 -8
  196. package/dist/components/card/card.js +0 -1
  197. package/dist/components/card/card.react.js +0 -1
  198. package/dist/components/card/card.scss.js +0 -1
  199. package/dist/components/card/index.js +0 -1
  200. package/dist/components/code-preview/code-preview.component.js +0 -1
  201. package/dist/components/code-preview/code-preview.js +0 -1
  202. package/dist/components/code-preview/code-preview.react.js +0 -1
  203. package/dist/components/code-preview/code-preview.scss.js +0 -1
  204. package/dist/components/code-preview/index.js +0 -1
  205. package/dist/components/container/container.component.js +0 -5
  206. package/dist/components/container/container.js +0 -1
  207. package/dist/components/container/container.react.js +0 -1
  208. package/dist/components/container/container.scss.js +0 -1
  209. package/dist/components/container/index.js +0 -1
  210. package/dist/components/dialog/dialog.component.js +0 -23
  211. package/dist/components/dialog/dialog.js +0 -1
  212. package/dist/components/dialog/dialog.react.js +0 -1
  213. package/dist/components/dialog/dialog.scss.js +0 -1
  214. package/dist/components/dialog/index.js +0 -1
  215. package/dist/components/dropzone/dropzone.component.js +0 -11
  216. package/dist/components/dropzone/dropzone.js +0 -1
  217. package/dist/components/dropzone/dropzone.react.js +0 -1
  218. package/dist/components/dropzone/dropzone.scss.js +0 -1
  219. package/dist/components/dropzone/index.js +0 -1
  220. package/dist/components/form-field/form-field.component.js +0 -22
  221. package/dist/components/form-field/form-field.js +0 -1
  222. package/dist/components/form-field/form-field.react.js +0 -1
  223. package/dist/components/form-field/form-field.scss.js +0 -1
  224. package/dist/components/form-field/index.js +0 -1
  225. package/dist/components/gallery/gallery-image.component.js +0 -14
  226. package/dist/components/gallery/gallery-image.js +0 -1
  227. package/dist/components/gallery/gallery.component.js +0 -5
  228. package/dist/components/gallery/gallery.js +0 -1
  229. package/dist/components/gallery/gallery.react.js +0 -1
  230. package/dist/components/gallery/gallery.scss.js +0 -1
  231. package/dist/components/gallery/index.js +0 -1
  232. package/dist/components/icon/icon.component.js +0 -1
  233. package/dist/components/icon/icon.js +0 -1
  234. package/dist/components/icon/icon.react.js +0 -1
  235. package/dist/components/icon/icon.scss.js +0 -1
  236. package/dist/components/icon/index.js +0 -1
  237. package/dist/components/loading/index.js +0 -1
  238. package/dist/components/loading/loading.component.js +0 -5
  239. package/dist/components/loading/loading.js +0 -1
  240. package/dist/components/loading/loading.react.js +0 -1
  241. package/dist/components/loading/loading.scss.js +0 -1
  242. package/dist/components/notice/index.js +0 -1
  243. package/dist/components/notice/notice.component.js +0 -17
  244. package/dist/components/notice/notice.js +0 -1
  245. package/dist/components/notice/notice.react.js +0 -1
  246. package/dist/components/notice/notice.scss.js +0 -1
  247. package/dist/components/pagination/index.js +0 -1
  248. package/dist/components/pagination/pagination.component.js +0 -13
  249. package/dist/components/pagination/pagination.js +0 -1
  250. package/dist/components/pagination/pagination.react.js +0 -1
  251. package/dist/components/pagination/pagination.scss.js +0 -1
  252. package/dist/components/tabs/index.js +0 -1
  253. package/dist/components/tabs/tab.component.js +0 -1
  254. package/dist/components/tabs/tab.js +0 -1
  255. package/dist/components/tabs/tabs-item.component.js +0 -1
  256. package/dist/components/tabs/tabs-panel.component.js +0 -10
  257. package/dist/components/tabs/tabs-panel.js +0 -1
  258. package/dist/components/tabs/tabs-tab.component.js +0 -12
  259. package/dist/components/tabs/tabs-tab.js +0 -1
  260. package/dist/components/tabs/tabs.component.js +0 -28
  261. package/dist/components/tabs/tabs.js +0 -1
  262. package/dist/components/tabs/tabs.react.js +0 -1
  263. package/dist/components/tabs/tabs.scss.js +0 -1
  264. package/dist/components/tooltip/index.js +0 -1
  265. package/dist/components/tooltip/tooltip.component.js +0 -10
  266. package/dist/components/tooltip/tooltip.js +0 -1
  267. package/dist/components/tooltip/tooltip.react.js +0 -1
  268. package/dist/components/tooltip/tooltip.scss.js +0 -1
  269. package/dist/css/base.css +0 -1
  270. package/dist/css/themes/grantcodes.css +0 -1
  271. package/dist/fonts/greycliff-bold-oblique.woff +0 -0
  272. package/dist/fonts/greycliff-bold-oblique.woff2 +0 -0
  273. package/dist/fonts/greycliff-bold.woff +0 -0
  274. package/dist/fonts/greycliff-bold.woff2 +0 -0
  275. package/dist/fonts/greycliff-demi-bold-oblique.woff +0 -0
  276. package/dist/fonts/greycliff-demi-bold-oblique.woff2 +0 -0
  277. package/dist/fonts/greycliff-demi-bold.woff +0 -0
  278. package/dist/fonts/greycliff-demi-bold.woff2 +0 -0
  279. package/dist/fonts/greycliff-extra-bold-oblique.woff +0 -0
  280. package/dist/fonts/greycliff-extra-bold-oblique.woff2 +0 -0
  281. package/dist/fonts/greycliff-extra-bold.woff +0 -0
  282. package/dist/fonts/greycliff-extra-bold.woff2 +0 -0
  283. package/dist/fonts/greycliff-extra-light-oblique.woff +0 -0
  284. package/dist/fonts/greycliff-extra-light-oblique.woff2 +0 -0
  285. package/dist/fonts/greycliff-extra-light.woff +0 -0
  286. package/dist/fonts/greycliff-extra-light.woff2 +0 -0
  287. package/dist/fonts/greycliff-heavy-oblique.woff +0 -0
  288. package/dist/fonts/greycliff-heavy-oblique.woff2 +0 -0
  289. package/dist/fonts/greycliff-heavy.woff +0 -0
  290. package/dist/fonts/greycliff-heavy.woff2 +0 -0
  291. package/dist/fonts/greycliff-light-oblique.woff +0 -0
  292. package/dist/fonts/greycliff-light-oblique.woff2 +0 -0
  293. package/dist/fonts/greycliff-light.woff +0 -0
  294. package/dist/fonts/greycliff-light.woff2 +0 -0
  295. package/dist/fonts/greycliff-medium-oblique.woff +0 -0
  296. package/dist/fonts/greycliff-medium-oblique.woff2 +0 -0
  297. package/dist/fonts/greycliff-medium.woff +0 -0
  298. package/dist/fonts/greycliff-medium.woff2 +0 -0
  299. package/dist/fonts/greycliff-regular-oblique.woff +0 -0
  300. package/dist/fonts/greycliff-regular-oblique.woff2 +0 -0
  301. package/dist/fonts/greycliff-regular.woff +0 -0
  302. package/dist/fonts/greycliff-regular.woff2 +0 -0
  303. package/dist/fonts/greycliff-thin-oblique.woff +0 -0
  304. package/dist/fonts/greycliff-thin-oblique.woff2 +0 -0
  305. package/dist/fonts/greycliff-thin.woff +0 -0
  306. package/dist/fonts/greycliff-thin.woff2 +0 -0
  307. package/dist/icons.js +0 -1
  308. package/dist/lib/generate-id.js +0 -1
  309. package/dist/main.js +0 -1
  310. package/dist/react.js +0 -1
@@ -0,0 +1,207 @@
1
+ import { css } from "lit";
2
+
3
+ export const toastStyles = css`
4
+ *,
5
+ *::before,
6
+ *::after {
7
+ box-sizing: border-box;
8
+ }
9
+
10
+ :host {
11
+ display: block;
12
+ pointer-events: none;
13
+ }
14
+
15
+ /* Toast Container */
16
+ .toast-container {
17
+ position: fixed;
18
+ z-index: 9999;
19
+ display: flex;
20
+ flex-direction: column;
21
+ gap: 0.75rem;
22
+ padding: 1rem;
23
+ pointer-events: none;
24
+ max-inline-size: 100vw;
25
+ max-block-size: 100vh;
26
+ overflow: hidden;
27
+ }
28
+
29
+ /* Container positions */
30
+ .toast-container--top-left {
31
+ inset-block-start: 0;
32
+ inset-inline-start: 0;
33
+ }
34
+
35
+ .toast-container--top-center {
36
+ inset-block-start: 0;
37
+ inset-inline-start: 50%;
38
+ transform: translateX(-50%);
39
+ }
40
+
41
+ .toast-container--top-right {
42
+ inset-block-start: 0;
43
+ inset-inline-end: 0;
44
+ }
45
+
46
+ .toast-container--bottom-left {
47
+ inset-block-end: 0;
48
+ inset-inline-start: 0;
49
+ }
50
+
51
+ .toast-container--bottom-center {
52
+ inset-block-end: 0;
53
+ inset-inline-start: 50%;
54
+ transform: translateX(-50%);
55
+ }
56
+
57
+ .toast-container--bottom-right {
58
+ inset-block-end: 0;
59
+ inset-inline-end: 0;
60
+ }
61
+
62
+ /* Toast */
63
+ .toast {
64
+ display: flex;
65
+ align-items: flex-start;
66
+ gap: 0.75rem;
67
+ min-inline-size: 300px;
68
+ max-inline-size: 500px;
69
+ padding-block: 1rem;
70
+ padding-inline: 1rem;
71
+ border-radius: var(--g-theme-border-radius-md);
72
+ background-color: var(--g-theme-color-background-default);
73
+ border-inline-start-width: 4px;
74
+ border-inline-start-style: solid;
75
+ pointer-events: auto;
76
+ opacity: 0;
77
+ transform: translateY(-1rem);
78
+ transition:
79
+ opacity 0.3s ease,
80
+ transform 0.3s ease;
81
+ }
82
+
83
+ .toast--visible {
84
+ opacity: 1;
85
+ transform: translateY(0);
86
+ }
87
+
88
+ /* Variant colors */
89
+ .toast--info {
90
+ border-inline-start-color: var(--g-theme-color-background-utility-info);
91
+ }
92
+
93
+ .toast--success {
94
+ border-inline-start-color: var(--g-theme-color-background-utility-success);
95
+ }
96
+
97
+ .toast--warning {
98
+ border-inline-start-color: var(--g-theme-color-background-utility-warning);
99
+ }
100
+
101
+ .toast--error {
102
+ border-inline-start-color: var(--g-theme-color-background-utility-error);
103
+ }
104
+
105
+ /* Toast icon */
106
+ .toast__icon {
107
+ flex-shrink: 0;
108
+ inline-size: 1.5rem;
109
+ block-size: 1.5rem;
110
+ }
111
+
112
+ .toast--info .toast__icon {
113
+ color: var(--g-theme-color-background-utility-info);
114
+ }
115
+
116
+ .toast--success .toast__icon {
117
+ color: var(--g-theme-color-background-utility-success);
118
+ }
119
+
120
+ .toast--warning .toast__icon {
121
+ color: var(--g-theme-color-background-utility-warning);
122
+ }
123
+
124
+ .toast--error .toast__icon {
125
+ color: var(--g-theme-color-background-utility-error);
126
+ }
127
+
128
+ /* Toast content */
129
+ .toast__content {
130
+ flex: 1;
131
+ min-inline-size: 0;
132
+ }
133
+
134
+ .toast__title {
135
+ font-weight: 600;
136
+ font-size: var(--g-typography-font-size-16);
137
+ margin-block-end: 0.25rem;
138
+ color: var(--g-theme-color-content-default);
139
+ }
140
+
141
+ .toast__message {
142
+ font-size: var(--g-typography-font-size-14);
143
+ color: var(--g-theme-color-content-secondary);
144
+ line-height: 1.5;
145
+ }
146
+
147
+ /* Close button */
148
+ .toast__close {
149
+ all: unset;
150
+ flex-shrink: 0;
151
+ inline-size: 1.5rem;
152
+ block-size: 1.5rem;
153
+ display: inline-flex;
154
+ align-items: center;
155
+ justify-content: center;
156
+ border-radius: var(--g-theme-border-radius-md);
157
+ color: var(--g-theme-color-content-secondary);
158
+ cursor: pointer;
159
+ transition: all 0.2s ease;
160
+ margin-inline-start: 0.5rem;
161
+ }
162
+
163
+ .toast__close:hover {
164
+ color: var(--g-theme-color-content-default);
165
+ background-color: var(--g-theme-color-background-subtle-hover);
166
+ }
167
+
168
+ .toast__close:focus-visible {
169
+ outline: 2px solid var(--component-focus-ring-color);
170
+ outline-offset: 2px;
171
+ }
172
+
173
+ /* Bottom position toasts slide up instead of down */
174
+ .toast--bottom-left,
175
+ .toast--bottom-center,
176
+ .toast--bottom-right {
177
+ transform: translateY(1rem);
178
+ }
179
+
180
+ .toast--bottom-left.toast--visible,
181
+ .toast--bottom-center.toast--visible,
182
+ .toast--bottom-right.toast--visible {
183
+ transform: translateY(0);
184
+ }
185
+
186
+ /* Responsive */
187
+ @media (max-width: 640px) {
188
+ .toast-container {
189
+ inset-inline: 0;
190
+ transform: none;
191
+ }
192
+
193
+ .toast {
194
+ min-inline-size: auto;
195
+ inline-size: 100%;
196
+ }
197
+ }
198
+
199
+ /* View transitions */
200
+ @media (prefers-reduced-motion: no-preference) {
201
+ @supports (view-transition-name: auto) {
202
+ .toast {
203
+ view-transition-name: toast;
204
+ }
205
+ }
206
+ }
207
+ `;
@@ -0,0 +1,196 @@
1
+ import { describe, it, afterEach } from "node:test";
2
+ import { strict as assert } from "node:assert";
3
+ import { fixture, cleanup, click } from "../../test-utils/index.js";
4
+ import "./toast.js";
5
+
6
+ describe("Toast Component", () => {
7
+ let element;
8
+
9
+ afterEach(() => {
10
+ cleanup(element);
11
+ });
12
+
13
+ it("should render with default properties", async () => {
14
+ element = await fixture("grantcodes-toast");
15
+ assert.ok(element, "Element should be created");
16
+ assert.ok(element.shadowRoot, "Element should have shadow root");
17
+ });
18
+
19
+ it("should have info variant by default", async () => {
20
+ element = await fixture("grantcodes-toast");
21
+ assert.strictEqual(element.variant, "info", "Default variant should be info");
22
+ });
23
+
24
+ it("should have 5000ms duration by default", async () => {
25
+ element = await fixture("grantcodes-toast");
26
+ assert.strictEqual(element.duration, 5000, "Default duration should be 5000ms");
27
+ });
28
+
29
+ it("should have top-right position by default", async () => {
30
+ element = await fixture("grantcodes-toast");
31
+ assert.strictEqual(element.position, "top-right", "Default position should be top-right");
32
+ });
33
+
34
+ it("should be dismissible by default", async () => {
35
+ element = await fixture("grantcodes-toast");
36
+ assert.strictEqual(element.dismissible, true, "Should be dismissible by default");
37
+ });
38
+
39
+ it("should render toast with correct variant class", async () => {
40
+ element = await fixture("grantcodes-toast", {
41
+ variant: "success",
42
+ });
43
+
44
+ const toast = element.shadowRoot.querySelector(".toast--success");
45
+ assert.ok(toast, "Toast should have success variant class");
46
+ });
47
+
48
+ it("should render title when provided", async () => {
49
+ element = await fixture("grantcodes-toast", {
50
+ title: "Test Title",
51
+ });
52
+
53
+ const title = element.shadowRoot.querySelector(".toast__title");
54
+ assert.ok(title, "Title element should exist");
55
+ assert.strictEqual(title.textContent, "Test Title", "Title text should match");
56
+ });
57
+
58
+ it("should not render title when not provided", async () => {
59
+ element = await fixture("grantcodes-toast");
60
+
61
+ const title = element.shadowRoot.querySelector(".toast__title");
62
+ assert.ok(!title, "Title element should not exist");
63
+ });
64
+
65
+ it("should render dismiss button when dismissible", async () => {
66
+ element = await fixture("grantcodes-toast", {
67
+ dismissible: true,
68
+ });
69
+
70
+ const closeButton = element.shadowRoot.querySelector(".toast__close");
71
+ assert.ok(closeButton, "Close button should exist");
72
+ });
73
+
74
+ it("should not render dismiss button when not dismissible", async () => {
75
+ element = await fixture("grantcodes-toast", {
76
+ dismissible: false,
77
+ });
78
+
79
+ const closeButton = element.shadowRoot.querySelector(".toast__close");
80
+ assert.ok(!closeButton, "Close button should not exist");
81
+ });
82
+
83
+ it("should have role=status", async () => {
84
+ element = await fixture("grantcodes-toast");
85
+
86
+ const toast = element.shadowRoot.querySelector(".toast");
87
+ assert.strictEqual(toast.getAttribute("role"), "status", "Should have status role");
88
+ });
89
+
90
+ it("should have aria-live=polite", async () => {
91
+ element = await fixture("grantcodes-toast");
92
+
93
+ const toast = element.shadowRoot.querySelector(".toast");
94
+ assert.strictEqual(
95
+ toast.getAttribute("aria-live"),
96
+ "polite",
97
+ "Should have aria-live polite",
98
+ );
99
+ });
100
+
101
+ it("should render icon based on variant", async () => {
102
+ element = await fixture("grantcodes-toast", {
103
+ variant: "success",
104
+ });
105
+
106
+ const icon = element.shadowRoot.querySelector(".toast__icon");
107
+ assert.ok(icon, "Icon should exist");
108
+ });
109
+
110
+ it("should emit dismiss event when dismissed", async () => {
111
+ element = await fixture("grantcodes-toast", {
112
+ dismissible: true,
113
+ duration: 0, // Disable auto-dismiss
114
+ });
115
+
116
+ let dismissed = false;
117
+ element.addEventListener("dismiss", () => {
118
+ dismissed = true;
119
+ });
120
+
121
+ const closeButton = element.shadowRoot.querySelector(".toast__close");
122
+ click(closeButton);
123
+
124
+ // Wait a bit for animation
125
+ await new Promise((resolve) => setTimeout(resolve, 350));
126
+
127
+ assert.ok(dismissed, "Dismiss event should have fired");
128
+ });
129
+
130
+ it("should support all variant types", async () => {
131
+ const variants = ["info", "success", "warning", "error"];
132
+
133
+ for (const variant of variants) {
134
+ element = await fixture("grantcodes-toast", { variant });
135
+ const toast = element.shadowRoot.querySelector(`.toast--${variant}`);
136
+ assert.ok(toast, `Toast should have ${variant} variant class`);
137
+ cleanup(element);
138
+ }
139
+ });
140
+ });
141
+
142
+ describe("Toast Container Component", () => {
143
+ let element;
144
+
145
+ afterEach(() => {
146
+ cleanup(element);
147
+ });
148
+
149
+ it("should render toast container", async () => {
150
+ element = await fixture("grantcodes-toast-container");
151
+ assert.ok(element, "Element should be created");
152
+ assert.ok(element.shadowRoot, "Element should have shadow root");
153
+ });
154
+
155
+ it("should have top-right position by default", async () => {
156
+ element = await fixture("grantcodes-toast-container");
157
+ assert.strictEqual(element.position, "top-right", "Default position should be top-right");
158
+ });
159
+
160
+ it("should render container with correct position class", async () => {
161
+ element = await fixture("grantcodes-toast-container", {
162
+ position: "bottom-left",
163
+ });
164
+
165
+ const container = element.shadowRoot.querySelector(".toast-container--bottom-left");
166
+ assert.ok(container, "Container should have bottom-left position class");
167
+ });
168
+
169
+ it("should have slot for toast items", async () => {
170
+ element = await fixture("grantcodes-toast-container");
171
+ const slot = element.shadowRoot.querySelector("slot");
172
+ assert.ok(slot, "Slot should exist for toast items");
173
+ });
174
+
175
+ it("should support all position types", async () => {
176
+ const positions = [
177
+ "top-left",
178
+ "top-center",
179
+ "top-right",
180
+ "bottom-left",
181
+ "bottom-center",
182
+ "bottom-right",
183
+ ];
184
+
185
+ for (const position of positions) {
186
+ element = await fixture("grantcodes-toast-container", { position });
187
+ const container = element.shadowRoot.querySelector(
188
+ `.toast-container--${position}`,
189
+ );
190
+ assert.ok(container, `Container should have ${position} position class`);
191
+ cleanup(element);
192
+ }
193
+ });
194
+ });
195
+
196
+
@@ -0,0 +1 @@
1
+ export * from "./tooltip";
@@ -0,0 +1,70 @@
1
+ import { LitElement } from "lit";
2
+ import { html } from "lit/static-html.js";
3
+ import { tooltipStyles } from "./tooltip.styles.js";
4
+ import { generateId } from "../../lib/generate-id.js";
5
+
6
+ export class GrantCodesTooltip extends LitElement {
7
+ // Styles are scoped to this element: they won't conflict with styles
8
+ // on the main page or in other components. Styling API can be exposed
9
+ // via CSS custom properties.
10
+ static styles = [tooltipStyles];
11
+
12
+ static properties = {
13
+ label: { type: String },
14
+ description: { type: String },
15
+ };
16
+
17
+ constructor() {
18
+ super();
19
+
20
+ this.id = generateId("tooltip");
21
+ /**
22
+ * Label for the tooltip, used when the tooltip is the main label for the item.
23
+ */
24
+ this.label = "";
25
+
26
+ /**
27
+ * Description for the tooltip, used when the tooltip is additional descriptive text for the item.
28
+ */
29
+ this.description = "";
30
+ }
31
+
32
+ /** @type {Element[]} */
33
+ slotted = [];
34
+
35
+ firstUpdated() {
36
+ const slot = this.renderRoot.querySelector("slot");
37
+ this.slotted = slot ? slot.assignedElements() : [];
38
+ const labeledElement = this.slotted[0];
39
+ if (labeledElement) {
40
+ // Set up anchor name for CSS anchor positioning
41
+ const tooltipSlot = this.renderRoot.querySelector(".tooltip__slot");
42
+ if (tooltipSlot) {
43
+ tooltipSlot.style.anchorName = `--tooltip-${this.id}`;
44
+ }
45
+
46
+ if (this.label) {
47
+ labeledElement.setAttribute("aria-labelledby", this.id);
48
+ } else if (this.description) {
49
+ labeledElement.setAttribute("aria-describedby", this.id);
50
+ }
51
+ }
52
+ }
53
+
54
+ render() {
55
+ if (this.label && this.description) {
56
+ throw new Error("You cannot provide both a label and a description");
57
+ }
58
+
59
+ return html`
60
+ <div class="tooltip">
61
+ <div class="tooltip__slot">
62
+ <slot></slot>
63
+ </div>
64
+ <p id="${this.id}" class="tooltip__content" role="tooltip">
65
+ ${this.label || this.description}
66
+ </p>
67
+ </div>
68
+ `;
69
+ }
70
+ }
@@ -0,0 +1,7 @@
1
+ import { GrantCodesTooltip } from "./tooltip.component.js";
2
+
3
+ export * from "./tooltip.component.js";
4
+ export default GrantCodesTooltip;
5
+
6
+ customElements.define("grantcodes-tooltip", GrantCodesTooltip);
7
+
@@ -0,0 +1,33 @@
1
+ import { html } from "lit/static-html.js";
2
+ import { getStorybookHelpers } from "@wc-toolkit/storybook-helpers";
3
+ const { events, args, argTypes, template } =
4
+ getStorybookHelpers("grantcodes-tooltip");
5
+ import "./tooltip.js";
6
+
7
+ const meta = {
8
+ title: "Components/Tooltip",
9
+ component: "grantcodes-tooltip",
10
+ args: {
11
+ ...args,
12
+ label: "This is a tooltip label",
13
+ description: "",
14
+ },
15
+ argTypes,
16
+ render: (args) => template(args, html`<span>This has a tooltip</span>`),
17
+ parameters: {
18
+ actions: {
19
+ handles: events,
20
+ },
21
+ },
22
+ };
23
+
24
+ export default meta;
25
+
26
+ export const Tooltip = {};
27
+
28
+ export const Description = {
29
+ args: {
30
+ label: "",
31
+ description: "This is a tooltip description",
32
+ },
33
+ };
@@ -0,0 +1,78 @@
1
+ import { css } from "lit";
2
+
3
+ export const tooltipStyles = css`
4
+
5
+ :host,
6
+ .tooltip,
7
+ .tooltip__slot {
8
+ width: fit-content;
9
+ }
10
+
11
+ .tooltip {
12
+ position: relative;
13
+ }
14
+
15
+ .tooltip__slot {
16
+ padding: 0.5rem;
17
+ margin: -0.5rem;
18
+ /* Anchor name will be set via JavaScript to include unique ID */
19
+ }
20
+
21
+ .tooltip__content {
22
+ display: none;
23
+ width: fit-content;
24
+ margin: 0;
25
+ max-width: 100%;
26
+ min-width: 6rem;
27
+ z-index: 100;
28
+ border-radius: var(--g-theme-border-radius-md);
29
+ padding: 0.6em 1em;
30
+ font-size: var(--g-typography-font-size-14);
31
+ line-height: 1;
32
+ color: var(--g-color-brand-purple-900);
33
+ background-color: var(--g-color-brand-purple-100);
34
+ box-shadow: 0 1px 3px rgba(0, 0, 0, 0.12), 0 1px 2px rgba(0, 0, 0, 0.24);
35
+ user-select: none;
36
+ text-align: center;
37
+ opacity: 0;
38
+ transition: opacity 0.2s;
39
+ transition-behavior: allow-discrete;
40
+
41
+ /* Modern CSS Anchor Positioning API */
42
+ /* This positions the tooltip relative to its anchor (the slot) */
43
+ position: absolute;
44
+
45
+ /* Try anchor positioning first (modern browsers) */
46
+ position-anchor: var(--tooltip-anchor, none);
47
+ inset-area: block-end;
48
+ position-area: block-end;
49
+ margin-block-start: 0.5rem;
50
+
51
+ /* Fallback for browsers without anchor positioning support */
52
+ @supports not (inset-area: block-end) {
53
+ /* Traditional positioning as fallback */
54
+ position: absolute;
55
+ top: 100%;
56
+ left: 50%;
57
+ transform: translateX(-50%);
58
+ margin-top: 0.5rem;
59
+ }
60
+ }
61
+
62
+ .tooltip__slot:hover + .tooltip__content,
63
+ .tooltip__slot:focus + .tooltip__content,
64
+ .tooltip__slot:focus-within + .tooltip__content {
65
+ display: block;
66
+ opacity: 1;
67
+ }
68
+
69
+ /* Optional: Position tooltip above when near bottom of viewport */
70
+ @media (prefers-reduced-motion: no-preference) {
71
+ .tooltip__content {
72
+ @starting-style {
73
+ opacity: 0;
74
+ }
75
+ }
76
+ }
77
+
78
+ `;