@govtechsg/sgds-web-component 0.0.10 → 0.0.11

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 (251) hide show
  1. package/package.json +7 -64
  2. package/.github/workflows/publish-latest.yml +0 -22
  3. package/.github/workflows/publish-pr.yml +0 -28
  4. package/.husky/commit-msg +0 -4
  5. package/.husky/prepare-commit-msg +0 -8
  6. package/.storybook/main.js +0 -16
  7. package/.storybook/preview-head.html +0 -11
  8. package/.storybook/preview.js +0 -9
  9. package/.vscode/settings.json +0 -7
  10. package/CONTRIBUTING.md +0 -56
  11. package/LICENSE +0 -20
  12. package/amplify.yml +0 -22
  13. package/commitlint.config.js +0 -1
  14. package/coverage/lcov-report/base.css +0 -224
  15. package/coverage/lcov-report/block-navigation.js +0 -87
  16. package/coverage/lcov-report/button-element.scss.html +0 -112
  17. package/coverage/lcov-report/button-element.ts.html +0 -145
  18. package/coverage/lcov-report/favicon.png +0 -0
  19. package/coverage/lcov-report/index.html +0 -116
  20. package/coverage/lcov-report/prettify.css +0 -1
  21. package/coverage/lcov-report/prettify.js +0 -2
  22. package/coverage/lcov-report/sort-arrow-sprite.png +0 -0
  23. package/coverage/lcov-report/sorter.js +0 -196
  24. package/coverage/lcov.info +0 -32
  25. package/index.html +0 -430
  26. package/mocks/dropdown.d.ts +0 -4
  27. package/mocks/dropdown.ts +0 -27
  28. package/mocks/link.d.ts +0 -3
  29. package/mocks/link.ts +0 -6
  30. package/rollup.config.js +0 -73
  31. package/rollup.test.config.js +0 -42
  32. package/scripts/buildUtils.js +0 -30
  33. package/scripts/frankBuild.js +0 -49
  34. package/src/Button/index.ts +0 -1
  35. package/src/Button/sgds-button.scss +0 -28
  36. package/src/Button/sgds-button.ts +0 -153
  37. package/src/Card/index.ts +0 -1
  38. package/src/Card/sgds-action-card.scss +0 -27
  39. package/src/Card/sgds-action-card.ts +0 -115
  40. package/src/Checkbox/index.ts +0 -1
  41. package/src/Checkbox/sgds-checkbox.scss +0 -4
  42. package/src/Checkbox/sgds-checkbox.ts +0 -149
  43. package/src/Dropdown/index.ts +0 -3
  44. package/src/Dropdown/sgds-dropdown-item.ts +0 -39
  45. package/src/Dropdown/sgds-dropdown.scss +0 -5
  46. package/src/Dropdown/sgds-dropdown.ts +0 -54
  47. package/src/Footer/index.ts +0 -3
  48. package/src/Footer/sgds-footer.scss +0 -5
  49. package/src/Footer/sgds-footer.ts +0 -121
  50. package/src/Input/index.ts +0 -1
  51. package/src/Input/sgds-input.scss +0 -20
  52. package/src/Input/sgds-input.ts +0 -178
  53. package/src/Mainnav/index.ts +0 -4
  54. package/src/Mainnav/sgds-mainnav-dropdown.scss +0 -13
  55. package/src/Mainnav/sgds-mainnav-dropdown.ts +0 -45
  56. package/src/Mainnav/sgds-mainnav-item.scss +0 -24
  57. package/src/Mainnav/sgds-mainnav-item.ts +0 -8
  58. package/src/Mainnav/sgds-mainnav.scss +0 -39
  59. package/src/Mainnav/sgds-mainnav.ts +0 -183
  60. package/src/Masthead/index.ts +0 -1
  61. package/src/Masthead/sgds-masthead.scss +0 -217
  62. package/src/Masthead/sgds-masthead.ts +0 -189
  63. package/src/Modal/index.ts +0 -1
  64. package/src/Modal/sgds-modal.scss +0 -128
  65. package/src/Modal/sgds-modal.ts +0 -309
  66. package/src/QuantityToggle/index.ts +0 -1
  67. package/src/QuantityToggle/sgds-quantitytoggle.scss +0 -10
  68. package/src/QuantityToggle/sgds-quantitytoggle.ts +0 -130
  69. package/src/Radio/index.ts +0 -2
  70. package/src/Radio/sgds-radio.scss +0 -5
  71. package/src/Radio/sgds-radio.ts +0 -120
  72. package/src/Radio/sgds-radiogroup.scss +0 -22
  73. package/src/Radio/sgds-radiogroup.ts +0 -221
  74. package/src/Sidenav/index.ts +0 -4
  75. package/src/Sidenav/sgds-sidenav-item.scss +0 -73
  76. package/src/Sidenav/sgds-sidenav-item.ts +0 -145
  77. package/src/Sidenav/sgds-sidenav-link.scss +0 -25
  78. package/src/Sidenav/sgds-sidenav-link.ts +0 -8
  79. package/src/Sidenav/sgds-sidenav.scss +0 -6
  80. package/src/Sidenav/sgds-sidenav.ts +0 -33
  81. package/src/Tab/index.ts +0 -3
  82. package/src/Tab/sgds-tab.scss +0 -84
  83. package/src/Tab/sgds-tab.ts +0 -87
  84. package/src/Tab/sgds-tabgroup.scss +0 -198
  85. package/src/Tab/sgds-tabgroup.ts +0 -295
  86. package/src/Tab/sgds-tabpanel.scss +0 -12
  87. package/src/Tab/sgds-tabpanel.ts +0 -55
  88. package/src/Textarea/index.ts +0 -1
  89. package/src/Textarea/sgds-textarea.scss +0 -23
  90. package/src/Textarea/sgds-textarea.ts +0 -201
  91. package/src/index.ts +0 -16
  92. package/src/utils/animate.ts +0 -69
  93. package/src/utils/animation-registry.ts +0 -71
  94. package/src/utils/base.scss +0 -14
  95. package/src/utils/breakpoints.ts +0 -5
  96. package/src/utils/card-element.ts +0 -42
  97. package/src/utils/components.style.scss +0 -531
  98. package/src/utils/defaultvalue.ts +0 -51
  99. package/src/utils/dropdown-element.ts +0 -244
  100. package/src/utils/event.ts +0 -13
  101. package/src/utils/form.ts +0 -183
  102. package/src/utils/generateId.ts +0 -4
  103. package/src/utils/link-element.ts +0 -34
  104. package/src/utils/mergeDeep.ts +0 -22
  105. package/src/utils/modal.ts +0 -64
  106. package/src/utils/object.ts +0 -2
  107. package/src/utils/offset.ts +0 -6
  108. package/src/utils/scroll.ts +0 -57
  109. package/src/utils/sgds-element.ts +0 -18
  110. package/src/utils/slot.ts +0 -102
  111. package/src/utils/tabbable.ts +0 -81
  112. package/src/utils/watch.ts +0 -62
  113. package/stories/ActionCard.stories.mdx +0 -199
  114. package/stories/Button.stories.mdx +0 -194
  115. package/stories/Checkbox.stories.mdx +0 -196
  116. package/stories/Dropdown.stories.mdx +0 -152
  117. package/stories/Footer.stories.mdx +0 -261
  118. package/stories/Input.stories.mdx +0 -236
  119. package/stories/MainNav.stories.mdx +0 -169
  120. package/stories/Masthead.stories.mdx +0 -112
  121. package/stories/Modal.stories.mdx +0 -103
  122. package/stories/QuantityToggle.stories.mdx +0 -97
  123. package/stories/Radio.stories.mdx +0 -262
  124. package/stories/Sample.stories.js +0 -29
  125. package/stories/Sample.stories.mdx +0 -33
  126. package/stories/SideNav.stories.mdx +0 -245
  127. package/stories/common.js +0 -185
  128. package/stories/textarea.stories.mdx +0 -253
  129. package/test/button.element.test.ts +0 -185
  130. package/test/checkbox.test.ts +0 -240
  131. package/test/dropdown.test.ts +0 -637
  132. package/test/footer.test.ts +0 -181
  133. package/test/generateId.test.ts +0 -18
  134. package/test/input.element.test.ts +0 -316
  135. package/test/link-element.test.ts +0 -38
  136. package/test/mainnav.test.ts +0 -313
  137. package/test/masthead.test.ts +0 -116
  138. package/test/modal.test.ts +0 -149
  139. package/test/quantitytoggle.test.ts +0 -76
  140. package/test/radio.test.ts +0 -310
  141. package/test/selectable-card.test.ts +0 -159
  142. package/test/sidenav.test.ts +0 -390
  143. package/test/tab.test.ts +0 -76
  144. package/test/textarea.test.ts +0 -126
  145. package/tsconfig.json +0 -26
  146. package/tsconfig.test.json +0 -24
  147. package/typings/scss.d.ts +0 -5
  148. package/web-dev-server.config.mjs +0 -7
  149. package/web-test-runner.config.mjs +0 -47
  150. /package/{lib/Button → Button}/index.d.ts +0 -0
  151. /package/{lib/Button → Button}/index.js +0 -0
  152. /package/{lib/Button → Button}/index.js.map +0 -0
  153. /package/{lib/Button → Button}/package.json +0 -0
  154. /package/{lib/Button → Button}/sgds-button.d.ts +0 -0
  155. /package/{lib/Card → Card}/index.d.ts +0 -0
  156. /package/{lib/Card → Card}/index.js +0 -0
  157. /package/{lib/Card → Card}/index.js.map +0 -0
  158. /package/{lib/Card → Card}/package.json +0 -0
  159. /package/{lib/Card → Card}/sgds-action-card.d.ts +0 -0
  160. /package/{lib/Checkbox → Checkbox}/index.d.ts +0 -0
  161. /package/{lib/Checkbox → Checkbox}/index.js +0 -0
  162. /package/{lib/Checkbox → Checkbox}/index.js.map +0 -0
  163. /package/{lib/Checkbox → Checkbox}/package.json +0 -0
  164. /package/{lib/Checkbox → Checkbox}/sgds-checkbox.d.ts +0 -0
  165. /package/{lib/Dropdown → Dropdown}/index.d.ts +0 -0
  166. /package/{lib/Dropdown → Dropdown}/index.js +0 -0
  167. /package/{lib/Dropdown → Dropdown}/index.js.map +0 -0
  168. /package/{lib/Dropdown → Dropdown}/package.json +0 -0
  169. /package/{lib/Dropdown → Dropdown}/sgds-dropdown-item.d.ts +0 -0
  170. /package/{lib/Dropdown → Dropdown}/sgds-dropdown.d.ts +0 -0
  171. /package/{lib/Footer → Footer}/index.d.ts +0 -0
  172. /package/{lib/Footer → Footer}/index.js +0 -0
  173. /package/{lib/Footer → Footer}/index.js.map +0 -0
  174. /package/{lib/Footer → Footer}/package.json +0 -0
  175. /package/{lib/Footer → Footer}/sgds-footer.d.ts +0 -0
  176. /package/{lib/Input → Input}/index.d.ts +0 -0
  177. /package/{lib/Input → Input}/index.js +0 -0
  178. /package/{lib/Input → Input}/index.js.map +0 -0
  179. /package/{lib/Input → Input}/package.json +0 -0
  180. /package/{lib/Input → Input}/sgds-input.d.ts +0 -0
  181. /package/{lib/Mainnav → Mainnav}/index.d.ts +0 -0
  182. /package/{lib/Mainnav → Mainnav}/index.js +0 -0
  183. /package/{lib/Mainnav → Mainnav}/index.js.map +0 -0
  184. /package/{lib/Mainnav → Mainnav}/package.json +0 -0
  185. /package/{lib/Mainnav → Mainnav}/sgds-mainnav-dropdown.d.ts +0 -0
  186. /package/{lib/Mainnav → Mainnav}/sgds-mainnav-item.d.ts +0 -0
  187. /package/{lib/Mainnav → Mainnav}/sgds-mainnav.d.ts +0 -0
  188. /package/{lib/Masthead → Masthead}/index.d.ts +0 -0
  189. /package/{lib/Masthead → Masthead}/index.js +0 -0
  190. /package/{lib/Masthead → Masthead}/index.js.map +0 -0
  191. /package/{lib/Masthead → Masthead}/package.json +0 -0
  192. /package/{lib/Masthead → Masthead}/sgds-masthead.d.ts +0 -0
  193. /package/{lib/Modal → Modal}/index.d.ts +0 -0
  194. /package/{lib/Modal → Modal}/index.js +0 -0
  195. /package/{lib/Modal → Modal}/index.js.map +0 -0
  196. /package/{lib/Modal → Modal}/package.json +0 -0
  197. /package/{lib/Modal → Modal}/sgds-modal.d.ts +0 -0
  198. /package/{lib/QuantityToggle → QuantityToggle}/index.d.ts +0 -0
  199. /package/{lib/QuantityToggle → QuantityToggle}/index.js +0 -0
  200. /package/{lib/QuantityToggle → QuantityToggle}/index.js.map +0 -0
  201. /package/{lib/QuantityToggle → QuantityToggle}/package.json +0 -0
  202. /package/{lib/QuantityToggle → QuantityToggle}/sgds-quantitytoggle.d.ts +0 -0
  203. /package/{lib/Radio → Radio}/index.d.ts +0 -0
  204. /package/{lib/Radio → Radio}/index.js +0 -0
  205. /package/{lib/Radio → Radio}/index.js.map +0 -0
  206. /package/{lib/Radio → Radio}/package.json +0 -0
  207. /package/{lib/Radio → Radio}/sgds-radio.d.ts +0 -0
  208. /package/{lib/Radio → Radio}/sgds-radiogroup.d.ts +0 -0
  209. /package/{lib/Sidenav → Sidenav}/index.d.ts +0 -0
  210. /package/{lib/Sidenav → Sidenav}/index.js +0 -0
  211. /package/{lib/Sidenav → Sidenav}/index.js.map +0 -0
  212. /package/{lib/Sidenav → Sidenav}/package.json +0 -0
  213. /package/{lib/Sidenav → Sidenav}/sgds-sidenav-item.d.ts +0 -0
  214. /package/{lib/Sidenav → Sidenav}/sgds-sidenav-link.d.ts +0 -0
  215. /package/{lib/Sidenav → Sidenav}/sgds-sidenav.d.ts +0 -0
  216. /package/{lib/Tab → Tab}/index.d.ts +0 -0
  217. /package/{lib/Tab → Tab}/index.js +0 -0
  218. /package/{lib/Tab → Tab}/index.js.map +0 -0
  219. /package/{lib/Tab → Tab}/package.json +0 -0
  220. /package/{lib/Tab → Tab}/sgds-tab.d.ts +0 -0
  221. /package/{lib/Tab → Tab}/sgds-tabgroup.d.ts +0 -0
  222. /package/{lib/Tab → Tab}/sgds-tabpanel.d.ts +0 -0
  223. /package/{lib/Textarea → Textarea}/index.d.ts +0 -0
  224. /package/{lib/Textarea → Textarea}/index.js +0 -0
  225. /package/{lib/Textarea → Textarea}/index.js.map +0 -0
  226. /package/{lib/Textarea → Textarea}/package.json +0 -0
  227. /package/{lib/Textarea → Textarea}/sgds-textarea.d.ts +0 -0
  228. /package/{lib/index.d.ts → index.d.ts} +0 -0
  229. /package/{lib/index.js → index.js} +0 -0
  230. /package/{lib/index.js.map → index.js.map} +0 -0
  231. /package/{lib/umd → umd}/index.js +0 -0
  232. /package/{lib/umd → umd}/index.js.map +0 -0
  233. /package/{lib/utils → utils}/animate.d.ts +0 -0
  234. /package/{lib/utils → utils}/animation-registry.d.ts +0 -0
  235. /package/{lib/utils → utils}/breakpoints.d.ts +0 -0
  236. /package/{lib/utils → utils}/card-element.d.ts +0 -0
  237. /package/{lib/utils → utils}/defaultvalue.d.ts +0 -0
  238. /package/{lib/utils → utils}/dropdown-element.d.ts +0 -0
  239. /package/{lib/utils → utils}/event.d.ts +0 -0
  240. /package/{lib/utils → utils}/form.d.ts +0 -0
  241. /package/{lib/utils → utils}/generateId.d.ts +0 -0
  242. /package/{lib/utils → utils}/link-element.d.ts +0 -0
  243. /package/{lib/utils → utils}/mergeDeep.d.ts +0 -0
  244. /package/{lib/utils → utils}/modal.d.ts +0 -0
  245. /package/{lib/utils → utils}/object.d.ts +0 -0
  246. /package/{lib/utils → utils}/offset.d.ts +0 -0
  247. /package/{lib/utils → utils}/scroll.d.ts +0 -0
  248. /package/{lib/utils → utils}/sgds-element.d.ts +0 -0
  249. /package/{lib/utils → utils}/slot.d.ts +0 -0
  250. /package/{lib/utils → utils}/tabbable.d.ts +0 -0
  251. /package/{lib/utils → utils}/watch.d.ts +0 -0
@@ -1,198 +0,0 @@
1
- @import '../utils/components.style';
2
- @import '../utils/base.scss';
3
- :host {
4
- --indicator-color: #0f71bb;
5
- --track-color: transparent;
6
- --track-width: 2px;
7
- display: block;
8
-
9
- }
10
- .tab-group {
11
- display: flex;
12
- border-radius: 0;
13
- }
14
- .tab-group__tabs {
15
- display: flex;
16
- position: relative;
17
- }
18
- .tab-group__indicator {
19
- position: absolute;
20
- // transition: var(--sl-transition-fast) translate ease, var(--sl-transition-fast) width ease;
21
- }
22
- .tab-group--has-scroll-controls .tab-group__nav-container {
23
- position: relative;
24
- padding: 0 var(--sl-spacing-x-large);
25
- }
26
- .tab-group__body {
27
- display: block;
28
- overflow: auto;
29
- }
30
- .tab-group__scroll-button {
31
- display: flex;
32
- align-items: center;
33
- justify-content: center;
34
- position: absolute;
35
- top: 0;
36
- bottom: 0;
37
- width: var(--sl-spacing-x-large);
38
- }
39
- .tab-group__scroll-button--start {
40
- left: 0;
41
- }
42
- .tab-group__scroll-button--end {
43
- right: 0;
44
- }
45
- .tab-group--rtl .tab-group__scroll-button--start {
46
- left: auto;
47
- right: 0;
48
- }
49
- .tab-group--rtl .tab-group__scroll-button--end {
50
- left: 0;
51
- right: auto;
52
- }
53
- /*
54
- * Top
55
- */
56
- .tab-group--top {
57
- flex-direction: column;
58
-
59
- }
60
- .tab-group--top .tab-group__nav-container {
61
- order: 1;
62
- }
63
- .tab-group--top .tab-group__nav {
64
- display: flex;
65
- overflow-x: auto;
66
- /* Hide scrollbar in Firefox */
67
- scrollbar-width: none;
68
- }
69
- /* Hide scrollbar in Chrome/Safari */
70
- .tab-group--top .tab-group__nav::-webkit-scrollbar {
71
- width: 0;
72
- height: 0;
73
- }
74
- .tab-group--top .tab-group__tabs {
75
- flex: 1 1 auto;
76
- position: relative;
77
- flex-direction: row;
78
- border-bottom: solid $nav-tabs-border-width transparent;
79
- gap: 1rem;
80
-
81
- }
82
- .tab-group--top .tab-group__indicator {
83
- bottom: calc(-1 * #{$nav-tabs-border-width});
84
- // border-bottom: solid $nav-tabs-border-width $blue-500;
85
- }
86
- .tab-group--top .tab-group__body {
87
- order: 2;
88
- }
89
- .tab-group--top ::slotted(sl-tab-panel) {
90
- --padding: var(--sl-spacing-medium) 0;
91
- }
92
- /*
93
- * Bottom
94
- */
95
- .tab-group--bottom {
96
- flex-direction: column;
97
- }
98
- .tab-group--bottom .tab-group__nav-container {
99
- order: 2;
100
- }
101
- .tab-group--bottom .tab-group__nav {
102
- display: flex;
103
- overflow-x: auto;
104
- /* Hide scrollbar in Firefox */
105
- scrollbar-width: none;
106
- }
107
- /* Hide scrollbar in Chrome/Safari */
108
- .tab-group--bottom .tab-group__nav::-webkit-scrollbar {
109
- width: 0;
110
- height: 0;
111
- }
112
- .tab-group--bottom .tab-group__tabs {
113
- flex: 1 1 auto;
114
- position: relative;
115
- flex-direction: row;
116
- border-top: solid $nav-tabs-border-width transparent;
117
- }
118
- .tab-group--bottom .tab-group__indicator {
119
- top: calc(-1 * #{$nav-tabs-border-width});
120
- border-top: solid $nav-tabs-border-width $blue-500;
121
- }
122
- .tab-group--bottom .tab-group__body {
123
- order: 1;
124
- }
125
- .tab-group--bottom ::slotted(sl-tab-panel) {
126
- --padding: var(--sl-spacing-medium) 0;
127
- }
128
- /*
129
- * Start
130
- */
131
- .tab-group--start {
132
- flex-direction: row;
133
- }
134
- .tab-group--start .tab-group__nav-container {
135
- order: 1;
136
- }
137
- .tab-group--start .tab-group__tabs {
138
- flex: 0 0 auto;
139
- flex-direction: column;
140
- border-inline-end: solid $nav-tabs-border-width transparent;
141
- }
142
- .tab-group--start .tab-group__indicator {
143
- right: calc(-1 * #{$nav-tabs-border-width});
144
- border-right: solid $nav-tabs-border-width $blue-500;
145
- }
146
- .tab-group--start.tab-group--rtl .tab-group__indicator {
147
- right: auto;
148
- left: calc(-1 * #{$nav-tabs-border-width});
149
- }
150
- .tab-group--start .tab-group__body {
151
- flex: 1 1 auto;
152
- order: 2;
153
- }
154
- .tab-group--start ::slotted(sl-tab-panel) {
155
- --padding: 0 var(--sl-spacing-medium);
156
- }
157
- /*
158
- * End
159
- */
160
- .tab-group--end {
161
- flex-direction: row;
162
- }
163
- .tab-group--end .tab-group__nav-container {
164
- order: 2;
165
- }
166
- .tab-group--end .tab-group__tabs {
167
- flex: 0 0 auto;
168
- flex-direction: column;
169
- border-left: solid $nav-tabs-border-width transparent;
170
- }
171
- .tab-group--end .tab-group__indicator {
172
- left: calc(-1 * #{$nav-tabs-border-width});
173
- border-inline-start: solid $nav-tabs-border-width $blue-500;
174
- }
175
- .tab-group--end.tab-group--rtl .tab-group__indicator {
176
- right: calc(-1 * #{$nav-tabs-border-width});
177
- left: auto;
178
- }
179
- .tab-group--end .tab-group__body {
180
- flex: 1 1 auto;
181
- order: 1;
182
- }
183
- .tab-group--end ::slotted(sl-tab-panel) {
184
- --padding: 0 var(--sl-spacing-medium);
185
- }
186
-
187
- .tab-group--top{
188
- &.tab-group--basic-toggle .tab-group__tabs {
189
- gap:0;
190
- ::slotted(sgds-tab){
191
- margin-right: -2px;
192
-
193
- }
194
- }
195
- }
196
-
197
-
198
-
@@ -1,295 +0,0 @@
1
- import { html } from 'lit';
2
- import { customElement, property, query, state } from 'lit/decorators.js';
3
- import { classMap } from 'lit/directives/class-map.js';
4
- import { scrollIntoView } from '../utils/scroll';
5
- import SgdsElement from "../utils/sgds-element";
6
- import { watch } from "../utils/watch";
7
- import styles from "./sgds-tabgroup.scss";
8
- import { SgdsTabPanel } from './sgds-tabpanel';
9
- import { SgdsTab } from './sgds-tab';
10
-
11
- @customElement('sgds-tab-group')
12
- export class SgdsTabGroup extends SgdsElement {
13
- static styles = styles;
14
-
15
- @query('.tab-group') tabGroup: HTMLElement;
16
- @query('.tab-group__body') body: HTMLSlotElement;
17
- @query('.tab-group__nav') nav: HTMLElement;
18
-
19
- private activeTab?: SgdsTab;
20
- private mutationObserver: MutationObserver;
21
- private resizeObserver: ResizeObserver;
22
- private tabs: SgdsTab[] = [];
23
- private panels: SgdsTabPanel[] = [];
24
-
25
- @state() private hasScrollControls = false;
26
-
27
- /** The placement of the tabs. */
28
- @property() placement: 'top' | 'bottom' | 'start' | 'end' = 'top';
29
-
30
- @property({reflect: true}) TabVariant: 'basic-toggle' | 'info-toggle';
31
- /**
32
- * When set to auto, navigating tabs with the arrow keys will instantly show the corresponding tab panel. When set to
33
- * manual, the tab will receive focus but will not show until the user presses spacebar or enter.
34
- */
35
- @property() activation: 'auto' | 'manual' = 'auto';
36
-
37
- /** Disables the scroll arrows that appear when tabs overflow. */
38
- @property({ attribute: 'no-scroll-controls', type: Boolean }) noScrollControls = false;
39
-
40
- connectedCallback() {
41
- super.connectedCallback();
42
-
43
- this.resizeObserver = new ResizeObserver(() => {
44
- this.updateScrollControls();
45
- });
46
-
47
- this.mutationObserver = new MutationObserver(mutations => {
48
- // Update aria labels when the DOM changes
49
- if (mutations.some(m => !['aria-labelledby', 'aria-controls'].includes(m.attributeName!))) {
50
- setTimeout(() => this.setAriaLabels());
51
- }
52
-
53
- // Sync tabs when disabled states change
54
- if (mutations.some(m => m.attributeName === 'disabled')) {
55
- this.syncTabsAndPanels();
56
- }
57
- });
58
-
59
- this.updateComplete.then(() => {
60
- this.syncTabsAndPanels();
61
- this.mutationObserver.observe(this, { attributes: true, childList: true, subtree: true });
62
- this.resizeObserver.observe(this.nav);
63
-
64
- // Set initial tab state when the tabs first become visible
65
- const intersectionObserver = new IntersectionObserver((entries, observer) => {
66
- if (entries[0].intersectionRatio > 0) {
67
- this.setAriaLabels();
68
- this.setTabVariant();
69
- this.setActiveTab(this.getActiveTab() ?? this.tabs[0], { emitEvents: false });
70
- observer.unobserve(entries[0].target);
71
- }
72
- });
73
- intersectionObserver.observe(this.tabGroup);
74
- });
75
- }
76
-
77
- disconnectedCallback() {
78
- this.mutationObserver.disconnect();
79
- this.resizeObserver.unobserve(this.nav);
80
- }
81
-
82
- /** Shows the specified tab panel. */
83
- show(panel: string) {
84
- const tab = this.tabs.find(el => el.panel === panel);
85
-
86
- if (tab) {
87
- this.setActiveTab(tab, { scrollBehavior: 'smooth' });
88
- }
89
- }
90
-
91
- getAllTabs(options: { includeDisabled: boolean } = { includeDisabled: true }) {
92
- const slot = this.shadowRoot!.querySelector<HTMLSlotElement>('slot[name="nav"]')!;
93
-
94
- return [...(slot.assignedElements() as SgdsTab[])].filter(el => {
95
- return options.includeDisabled
96
- ? el.tagName.toLowerCase() === 'sgds-tab'
97
- : el.tagName.toLowerCase() === 'sgds-tab' && !el.disabled;
98
- });
99
- }
100
-
101
- getAllPanels() {
102
- return [...this.body.assignedElements()].filter(el => el.tagName.toLowerCase() === 'sgds-tab-panel') as [SgdsTabPanel];
103
- }
104
-
105
- getActiveTab() {
106
- return this.tabs.find(el => el.active);
107
- }
108
-
109
- handleClick(event: MouseEvent) {
110
- const target = event.target as HTMLElement;
111
- const tab = target.closest('sgds-tab') as SgdsTab;
112
- const tabGroup = tab?.closest('sgds-tab-group');
113
-
114
- // Ensure the target tab is in this tab group
115
- if (tabGroup !== this) {
116
- return;
117
- }
118
-
119
- if (tab !== null) {
120
- this.setActiveTab(tab, { scrollBehavior: 'smooth' });
121
- }
122
- }
123
-
124
- handleKeyDown(event: KeyboardEvent) {
125
- const target = event.target as HTMLElement;
126
- const tab = target.closest('sgds-tab') as SgdsTab;
127
- const tabGroup = tab?.closest('sgds-tab-group');
128
-
129
- // Ensure the target tab is in this tab group
130
- if (tabGroup !== this) {
131
- return;
132
- }
133
-
134
- // Activate a tab
135
- if (['Enter', ' '].includes(event.key)) {
136
- if (tab !== null) {
137
- this.setActiveTab(tab, { scrollBehavior: 'smooth' });
138
- event.preventDefault();
139
- }
140
- }
141
-
142
- // Move focus left or right
143
- if (['ArrowLeft', 'ArrowRight', 'ArrowUp', 'ArrowDown', 'Home', 'End'].includes(event.key)) {
144
- const activeEl = this.tabs.find(t => t.matches(':focus'));
145
-
146
-
147
- if (activeEl?.tagName.toLowerCase() === 'sgds-tab') {
148
- let index = this.tabs.indexOf(activeEl);
149
-
150
- if (event.key === 'Home') {
151
- index = 0;
152
- } else if (event.key === 'End') {
153
- index = this.tabs.length - 1;
154
- } else if (
155
- ['top', 'bottom'].includes(this.placement) ||
156
- (['start', 'end'].includes(this.placement) && event.key === 'ArrowUp')
157
- ) {
158
- index--;
159
- } else if (
160
- ['top', 'bottom'].includes(this.placement) ||
161
- (['start', 'end'].includes(this.placement) && event.key === 'ArrowDown')
162
- ) {
163
- index++;
164
- }
165
-
166
- if (index < 0) {
167
- index = this.tabs.length - 1;
168
- }
169
-
170
- if (index > this.tabs.length - 1) {
171
- index = 0;
172
- }
173
-
174
- this.tabs[index].focus({ preventScroll: true });
175
-
176
- if (this.activation === 'auto') {
177
- this.setActiveTab(this.tabs[index], { scrollBehavior: 'smooth' });
178
- }
179
-
180
- if (['top', 'bottom'].includes(this.placement)) {
181
- scrollIntoView(this.tabs[index], this.nav, 'horizontal');
182
- }
183
-
184
- event.preventDefault();
185
- }
186
- }
187
- }
188
-
189
- handleScrollToStart() {
190
- this.nav.scroll({
191
- left: this.nav.scrollLeft - this.nav.clientWidth,
192
- behavior: 'smooth'
193
- });
194
- }
195
-
196
- handleScrollToEnd() {
197
- this.nav.scroll({
198
- left: this.nav.scrollLeft + this.nav.clientWidth,
199
- behavior: 'smooth'
200
- });
201
- }
202
-
203
- @watch('noScrollControls', { waitUntilFirstUpdate: true })
204
- updateScrollControls() {
205
- if (this.noScrollControls) {
206
- this.hasScrollControls = false;
207
- } else {
208
- this.hasScrollControls =
209
- ['top', 'bottom'].includes(this.placement) && this.nav.scrollWidth > this.nav.clientWidth;
210
- }
211
- }
212
-
213
- setActiveTab(tab: SgdsTab, options?: { emitEvents?: boolean; scrollBehavior?: 'auto' | 'smooth' }) {
214
- options = {
215
- emitEvents: true,
216
- scrollBehavior: 'auto',
217
- ...options
218
- };
219
-
220
- if (tab !== this.activeTab && !tab.disabled) {
221
- const previousTab = this.activeTab;
222
- this.activeTab = tab;
223
-
224
- // Sync active tab and panel
225
- this.tabs.map(el => (el.active = el === this.activeTab));
226
- this.panels.map(el => (el.active = el.name === this.activeTab?.panel));
227
-
228
-
229
- if (['top', 'bottom'].includes(this.placement)) {
230
- scrollIntoView(this.activeTab, this.nav, 'horizontal', options.scrollBehavior);
231
- }
232
-
233
- // Emit events
234
- if (options.emitEvents) {
235
- if (previousTab) {
236
- this.emit('sgds-tab-hide', { detail: { name: previousTab.panel } });
237
- }
238
-
239
- this.emit('sgds-tab-show', { detail: { name: this.activeTab.panel } });
240
- }
241
- }
242
- }
243
-
244
- setAriaLabels() {
245
- // Link each tab with its corresponding panel
246
- this.tabs.forEach(tab => {
247
- const panel = this.panels.find(el => el.name === tab.panel);
248
- if (panel) {
249
- tab.setAttribute('aria-controls', panel.getAttribute('id')!);
250
- panel.setAttribute('aria-labelledby', tab.getAttribute('id')!);
251
- }
252
- });
253
- }
254
-
255
- setTabVariant() {
256
- // Link each tab with its corresponding panel
257
- this.tabs.forEach(tab => {
258
- tab.setAttribute('variant', this.TabVariant);
259
- });
260
- }
261
-
262
- // This stores tabs and panels so we can refer to a cache instead of calling querySelectorAll() multiple times.
263
- syncTabsAndPanels() {
264
- this.tabs = this.getAllTabs({ includeDisabled: false });
265
- this.panels = this.getAllPanels();
266
- }
267
-
268
- render() {
269
- return html`
270
- <div
271
- part="base"
272
- class=${classMap({
273
- 'tab-group': true,
274
- 'tab-group--top': this.placement === 'top',
275
- 'tab-group--bottom': this.placement === 'bottom',
276
- 'tab-group--start': this.placement === 'start',
277
- 'tab-group--end': this.placement === 'end',
278
- 'tab-group--basic-toggle': this.TabVariant === 'basic-toggle',
279
- 'tab-group--info-toggle': this.TabVariant === 'info-toggle',
280
- })}
281
- @click=${this.handleClick}
282
- @keydown=${this.handleKeyDown}
283
- >
284
- <div class="tab-group__nav-container" part="nav">
285
- <div class="tab-group__nav">
286
- <div part="tabs" class="tab-group__tabs" role="tablist">
287
- <slot name="nav" @slotchange=${this.syncTabsAndPanels}></slot>
288
- </div>
289
- </div>
290
- </div>
291
- <slot part="body" class="tab-group__body" @slotchange=${this.syncTabsAndPanels}></slot>
292
- </div>
293
- `;
294
- }
295
- }
@@ -1,12 +0,0 @@
1
- @import '../utils/components.style';
2
- :host {
3
- --padding: 1rem;
4
- display: block;
5
- }
6
- .tab-panel {
7
- display: block;
8
- padding: var(--padding) 0;
9
- }
10
- .tab-panel:not(.tab-panel--active) {
11
- display: none;
12
- }
@@ -1,55 +0,0 @@
1
- import { html } from 'lit';
2
- import { customElement, property } from 'lit/decorators.js';
3
- import { classMap } from 'lit/directives/class-map.js';
4
- import SgdsElement from "../utils/sgds-element";
5
- import { watch } from "../utils/watch";
6
- import styles from "./sgds-tabpanel.scss";
7
- let id = 0;
8
-
9
- /**
10
- * @summary Tab panels are used inside [tab groups](/components/tab-group) to display tabbed content.
11
- *
12
- * @since 2.0
13
- * @status stable
14
- *
15
- * @slot - The tab panel's content.
16
- *
17
- * @csspart base - The component's base wrapper.
18
- *
19
- * @cssproperty --padding - The tab panel's padding.
20
- */
21
- @customElement('sgds-tab-panel')
22
- export class SgdsTabPanel extends SgdsElement {
23
- static styles = styles;
24
- private readonly attrId = ++id;
25
- private readonly componentId = `sgds-tab-panel-${this.attrId}`;
26
-
27
- /** The tab panel's name. */
28
- @property({ reflect: true }) name = '';
29
-
30
- /** When true, the tab panel will be shown. */
31
- @property({ type: Boolean, reflect: true }) active = false;
32
-
33
- connectedCallback() {
34
- super.connectedCallback();
35
- this.id = this.id.length > 0 ? this.id : this.componentId;
36
- this.setAttribute('role', 'tabpanel');
37
- }
38
-
39
- @watch('active')
40
- handleActiveChange() {
41
- this.setAttribute('aria-hidden', this.active ? 'false' : 'true');
42
- }
43
-
44
- render() {
45
- return html`
46
- <slot
47
- part="base"
48
- class=${classMap({
49
- 'tab-panel': true,
50
- 'tab-panel--active': this.active
51
- })}
52
- ></slot>
53
- `;
54
- }
55
- }
@@ -1 +0,0 @@
1
- export { SgdsTextArea } from "./sgds-textarea";
@@ -1,23 +0,0 @@
1
-
2
- @import '../utils/base.scss';
3
- @import "~@govtechsg/sgds/sass/forms/labels";
4
- @import "~@govtechsg/sgds/sass/forms/form-text";
5
- @import "~@govtechsg/sgds/sass/forms/form-control";
6
- @import "~@govtechsg/sgds/sass/forms/form-control-group";
7
- @import "~@govtechsg/sgds/sass/forms/input-group";
8
- @import "~@govtechsg/sgds/sass/forms/validation";
9
-
10
- :host{
11
- --sgds-body-line-height: 2rem;
12
- & > .text-area-label-wrapper{
13
- line-height: var(--sgds-body-line-height);
14
- }
15
-
16
- .textarea-resize-none{
17
- resize: none;
18
- }
19
- .textarea-resize-vertical {
20
- resize: vertical;
21
- }
22
- }
23
-