primer_view_components 0.43.5 → 0.44.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 (117) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG.md +20 -0
  3. data/app/assets/javascripts/components/primer/alpha/tree_view/tree_view.d.ts +39 -0
  4. data/app/assets/javascripts/components/primer/alpha/tree_view/tree_view_icon_pair_element.d.ts +15 -0
  5. data/app/assets/javascripts/components/primer/alpha/tree_view/tree_view_include_fragment_element.d.ts +9 -0
  6. data/app/assets/javascripts/components/primer/alpha/tree_view/tree_view_roving_tab_index.d.ts +3 -0
  7. data/app/assets/javascripts/components/primer/alpha/tree_view/tree_view_sub_tree_node_element.d.ts +42 -0
  8. data/app/assets/javascripts/components/primer/primer.d.ts +4 -0
  9. data/app/assets/javascripts/components/primer/shared_events.d.ts +15 -0
  10. data/app/assets/javascripts/primer_view_components.js +1 -1
  11. data/app/assets/javascripts/primer_view_components.js.map +1 -1
  12. data/app/assets/styles/primer_view_components.css +1 -1
  13. data/app/assets/styles/primer_view_components.css.map +1 -1
  14. data/app/components/primer/alpha/file_tree_view/directory_node.html.erb +5 -0
  15. data/app/components/primer/alpha/file_tree_view/directory_node.rb +24 -0
  16. data/app/components/primer/alpha/file_tree_view/file_node.html.erb +2 -0
  17. data/app/components/primer/alpha/file_tree_view/file_node.rb +14 -0
  18. data/app/components/primer/alpha/file_tree_view.rb +15 -0
  19. data/app/components/primer/alpha/skeleton_box.css +1 -0
  20. data/app/components/primer/alpha/skeleton_box.css.json +6 -0
  21. data/app/components/primer/alpha/skeleton_box.css.map +1 -0
  22. data/app/components/primer/alpha/skeleton_box.html.erb +1 -0
  23. data/app/components/primer/alpha/skeleton_box.pcss +30 -0
  24. data/app/components/primer/alpha/skeleton_box.rb +29 -0
  25. data/app/components/primer/alpha/tree_view/icon.html.erb +1 -0
  26. data/app/components/primer/alpha/tree_view/icon.rb +22 -0
  27. data/app/components/primer/alpha/tree_view/icon_pair.html.erb +13 -0
  28. data/app/components/primer/alpha/tree_view/icon_pair.rb +42 -0
  29. data/app/components/primer/alpha/tree_view/leading_action.html.erb +3 -0
  30. data/app/components/primer/alpha/tree_view/leading_action.rb +18 -0
  31. data/app/components/primer/alpha/tree_view/leaf_node.html.erb +18 -0
  32. data/app/components/primer/alpha/tree_view/leaf_node.rb +96 -0
  33. data/app/components/primer/alpha/tree_view/loading_failure_message.html.erb +13 -0
  34. data/app/components/primer/alpha/tree_view/loading_failure_message.rb +31 -0
  35. data/app/components/primer/alpha/tree_view/node.html.erb +32 -0
  36. data/app/components/primer/alpha/tree_view/node.rb +194 -0
  37. data/app/components/primer/alpha/tree_view/skeleton_loader.html.erb +23 -0
  38. data/app/components/primer/alpha/tree_view/skeleton_loader.rb +36 -0
  39. data/app/components/primer/alpha/tree_view/spinner_loader.html.erb +20 -0
  40. data/app/components/primer/alpha/tree_view/spinner_loader.rb +33 -0
  41. data/app/components/primer/alpha/tree_view/sub_tree.html.erb +21 -0
  42. data/app/components/primer/alpha/tree_view/sub_tree.rb +113 -0
  43. data/app/components/primer/alpha/tree_view/sub_tree_container.html.erb +3 -0
  44. data/app/components/primer/alpha/tree_view/sub_tree_container.rb +39 -0
  45. data/app/components/primer/alpha/tree_view/sub_tree_node.html.erb +49 -0
  46. data/app/components/primer/alpha/tree_view/sub_tree_node.rb +188 -0
  47. data/app/components/primer/alpha/tree_view/tree_view.d.ts +39 -0
  48. data/app/components/primer/alpha/tree_view/tree_view.js +363 -0
  49. data/app/components/primer/alpha/tree_view/tree_view.ts +396 -0
  50. data/app/components/primer/alpha/tree_view/tree_view_icon_pair_element.d.ts +15 -0
  51. data/app/components/primer/alpha/tree_view/tree_view_icon_pair_element.js +62 -0
  52. data/app/components/primer/alpha/tree_view/tree_view_icon_pair_element.ts +56 -0
  53. data/app/components/primer/alpha/tree_view/tree_view_include_fragment_element.d.ts +9 -0
  54. data/app/components/primer/alpha/tree_view/tree_view_include_fragment_element.js +28 -0
  55. data/app/components/primer/alpha/tree_view/tree_view_include_fragment_element.ts +28 -0
  56. data/app/components/primer/alpha/tree_view/tree_view_roving_tab_index.d.ts +3 -0
  57. data/app/components/primer/alpha/tree_view/tree_view_roving_tab_index.js +130 -0
  58. data/app/components/primer/alpha/tree_view/tree_view_roving_tab_index.ts +161 -0
  59. data/app/components/primer/alpha/tree_view/tree_view_sub_tree_node_element.d.ts +42 -0
  60. data/app/components/primer/alpha/tree_view/tree_view_sub_tree_node_element.js +418 -0
  61. data/app/components/primer/alpha/tree_view/tree_view_sub_tree_node_element.ts +470 -0
  62. data/app/components/primer/alpha/tree_view/visual.html.erb +14 -0
  63. data/app/components/primer/alpha/tree_view/visual.rb +27 -0
  64. data/app/components/primer/alpha/tree_view.css +1 -0
  65. data/app/components/primer/alpha/tree_view.css.json +52 -0
  66. data/app/components/primer/alpha/tree_view.css.map +1 -0
  67. data/app/components/primer/alpha/tree_view.html.erb +12 -0
  68. data/app/components/primer/alpha/tree_view.pcss +373 -0
  69. data/app/components/primer/alpha/tree_view.rb +439 -0
  70. data/app/components/primer/beta/breadcrumbs.css +1 -1
  71. data/app/components/primer/beta/breadcrumbs.css.json +0 -1
  72. data/app/components/primer/beta/breadcrumbs.css.map +1 -1
  73. data/app/components/primer/beta/breadcrumbs.pcss +2 -8
  74. data/app/components/primer/beta/progress_bar.css +1 -1
  75. data/app/components/primer/beta/progress_bar.css.map +1 -1
  76. data/app/components/primer/beta/progress_bar.pcss +3 -2
  77. data/app/components/primer/beta/relative_time.rb +3 -0
  78. data/app/components/primer/beta/spinner.html.erb +1 -1
  79. data/app/components/primer/beta/spinner.rb +2 -0
  80. data/app/components/primer/primer.d.ts +4 -0
  81. data/app/components/primer/primer.js +4 -0
  82. data/app/components/primer/primer.pcss +2 -0
  83. data/app/components/primer/primer.ts +4 -0
  84. data/app/components/primer/shared_events.d.ts +15 -0
  85. data/app/components/primer/shared_events.ts +19 -0
  86. data/app/controllers/primer/view_components/tree_view_items.json +293 -0
  87. data/app/controllers/primer/view_components/tree_view_items_controller.rb +55 -0
  88. data/app/forms/check_box_with_nested_form.rb +10 -10
  89. data/app/forms/radio_button_with_nested_form.rb +16 -16
  90. data/app/views/primer/view_components/tree_view_items/async_alpha.html_fragment.erb +23 -0
  91. data/app/views/primer/view_components/tree_view_items/index.html_fragment.erb +24 -0
  92. data/config/routes.rb +2 -0
  93. data/lib/primer/view_components/version.rb +2 -2
  94. data/previews/primer/alpha/file_tree_view_preview/default.html.erb +16 -0
  95. data/previews/primer/alpha/file_tree_view_preview/playground.html.erb +4 -0
  96. data/previews/primer/alpha/file_tree_view_preview.rb +69 -0
  97. data/previews/primer/alpha/skeleton_box_preview.rb +20 -0
  98. data/previews/primer/alpha/tree_view_preview/async_alpha.html.erb +12 -0
  99. data/previews/primer/alpha/tree_view_preview/buttons.html.erb +10 -0
  100. data/previews/primer/alpha/tree_view_preview/default.html.erb +24 -0
  101. data/previews/primer/alpha/tree_view_preview/empty.html.erb +10 -0
  102. data/previews/primer/alpha/tree_view_preview/form_input.html.erb +14 -0
  103. data/previews/primer/alpha/tree_view_preview/leaf_node_playground.html.erb +15 -0
  104. data/previews/primer/alpha/tree_view_preview/links.html.erb +17 -0
  105. data/previews/primer/alpha/tree_view_preview/loading_failure.html.erb +36 -0
  106. data/previews/primer/alpha/tree_view_preview/loading_skeleton.html.erb +12 -0
  107. data/previews/primer/alpha/tree_view_preview/loading_spinner.html.erb +12 -0
  108. data/previews/primer/alpha/tree_view_preview/playground.html.erb +4 -0
  109. data/previews/primer/alpha/tree_view_preview.rb +208 -0
  110. data/static/arguments.json +456 -0
  111. data/static/audited_at.json +17 -0
  112. data/static/classes.json +15 -0
  113. data/static/constants.json +101 -0
  114. data/static/info_arch.json +1410 -56
  115. data/static/previews.json +232 -0
  116. data/static/statuses.json +17 -0
  117. metadata +89 -8
@@ -0,0 +1,373 @@
1
+ /* stylelint-disable selector-max-type -- Copied from primer/react */
2
+
3
+ .TreeViewRootUlStyles {
4
+ padding: 0;
5
+ margin: 0;
6
+ list-style: none;
7
+
8
+ /*
9
+ * WARNING: This is a performance optimization.
10
+ *
11
+ * We define styles for the tree items at the root level of the tree
12
+ * to avoid recomputing the styles for each item when the tree updates.
13
+ * We're sacrificing maintainability for performance because TreeView
14
+ * needs to be performant enough to handle large trees (thousands of items).
15
+ *
16
+ * This is intended to be a temporary solution until we can improve the
17
+ * performance of our styling patterns.
18
+ *
19
+ * Do NOT copy this pattern without understanding the tradeoffs.
20
+ */
21
+ & .TreeViewItem {
22
+ outline: none;
23
+
24
+ &:focus-visible > div {
25
+ box-shadow: var(--boxShadow-thick) var(--fgColor-accent);
26
+
27
+ @media (forced-colors: active) {
28
+ outline: 2px solid HighlightText;
29
+ /* stylelint-disable-next-line declaration-property-value-no-unknown -- Copied from primer/react */
30
+ outline-offset: -2;
31
+ }
32
+ }
33
+
34
+ &[data-has-leading-action] {
35
+ --has-leading-action: 1;
36
+ }
37
+ }
38
+
39
+ & .TreeViewItemContainer {
40
+ --level: 1;
41
+ --toggle-width: 1rem;
42
+ --min-item-height: 2rem;
43
+
44
+ position: relative;
45
+ display: grid;
46
+ width: 100%;
47
+ font-size: var(--text-body-size-medium);
48
+ color: var(--fgColor-default);
49
+ border-radius: var(--borderRadius-medium);
50
+ grid-template-columns: var(--spacer-width) var(--leading-action-width) var(--toggle-width) 1fr;
51
+ grid-template-areas: 'spacer leadingAction toggle content';
52
+
53
+ --leading-action-width: calc(var(--has-leading-action, 0) * 1.5rem);
54
+ --spacer-width: calc(calc(var(--level) - 1) * (var(--toggle-width) / 2));
55
+
56
+ &:hover {
57
+ background-color: var(--control-transparent-bgColor-hover);
58
+
59
+ @media (forced-colors: active) {
60
+ outline: 2px solid transparent;
61
+ outline-offset: -2px;
62
+ }
63
+ }
64
+
65
+ @media (pointer: coarse) {
66
+ --toggle-width: 1.5rem;
67
+ --min-item-height: 2.75rem;
68
+ }
69
+
70
+ &:has(.TreeViewFailureMessage):hover {
71
+ cursor: default;
72
+ background-color: transparent;
73
+
74
+ @media (forced-colors: active) {
75
+ outline: none;
76
+ }
77
+ }
78
+
79
+ &:has([role='treeitem']:focus-visible) {
80
+ box-shadow: var(--boxShadow-thick) var(--fgColor-accent);
81
+ }
82
+ }
83
+
84
+ &:where([data-omit-spacer='true']) .TreeViewItemContainer {
85
+ grid-template-columns: 0 0 0 1fr;
86
+ }
87
+
88
+ /* stylelint-disable-next-line selector-max-specificity */
89
+ & .TreeViewItem > .TreeViewItemContainer:has(.TreeViewItemContent[aria-current='true']) {
90
+ background-color: var(--control-transparent-bgColor-selected);
91
+
92
+ /* Current item indicator */
93
+ /* stylelint-disable-next-line selector-max-specificity -- Copied from primer/react */
94
+ &::after {
95
+ position: absolute;
96
+ top: calc(50% - var(--base-size-12));
97
+ left: calc(-1 * var(--base-size-8));
98
+ width: 0.25rem;
99
+ height: 1.5rem;
100
+ content: '';
101
+
102
+ /*
103
+ * Use fgColor accent for consistency across all themes. Using the "correct" variable,
104
+ * --bgColor-accent-emphasis, causes vrt failures for dark high contrast mode
105
+ */
106
+ /* stylelint-disable-next-line primer/colors */
107
+ background-color: var(--fgColor-accent);
108
+ border-radius: var(--borderRadius-medium);
109
+
110
+ @media (forced-colors: active) {
111
+ background-color: HighlightText;
112
+ }
113
+ }
114
+ }
115
+
116
+ & .TreeViewItemToggle {
117
+ display: flex;
118
+ height: 100%;
119
+
120
+ /* The toggle should appear vertically centered for single-line items, but remain at the top for items that wrap
121
+ across more lines. */
122
+ /* stylelint-disable-next-line primer/spacing */
123
+ padding-top: calc(var(--min-item-height) / 2 - var(--base-size-12) / 2);
124
+ color: var(--fgColor-muted);
125
+ grid-area: toggle;
126
+ justify-content: center;
127
+ align-items: flex-start;
128
+ cursor: pointer;
129
+ }
130
+
131
+ & .TreeViewItemToggleHover:hover {
132
+ background-color: var(--control-transparent-bgColor-hover);
133
+ }
134
+
135
+ & .TreeViewItemToggleEnd {
136
+ border-top-left-radius: var(--borderRadius-medium);
137
+ border-bottom-left-radius: var(--borderRadius-medium);
138
+ }
139
+
140
+ /* stylelint-disable-next-line selector-no-qualifying-type */
141
+ & a.TreeViewItemContent:hover, button.TreeViewItemContent:hover {
142
+ text-decoration: underline;
143
+ text-decoration-color: var(--control-fgColor-rest);
144
+ }
145
+
146
+ & :has(.TreeViewItemContent[aria-disabled="true"]) {
147
+ cursor: not-allowed;
148
+ }
149
+
150
+ & .TreeViewItemContent {
151
+ display: flex;
152
+ height: 100%;
153
+ padding: 0 var(--base-size-8);
154
+ outline: none;
155
+ text-align: left;
156
+ user-select: none;
157
+ background-color: transparent;
158
+ border: none;
159
+ touch-action: manipulation;
160
+ -webkit-tap-highlight-color: transparent;
161
+ cursor: pointer;
162
+
163
+ /* The dynamic top and bottom padding to maintain the minimum item height for single line items */
164
+ /* stylelint-disable-next-line primer/spacing */
165
+ padding-top: calc((var(--min-item-height) - var(--custom-line-height, 1.3rem)) / 2);
166
+ /* stylelint-disable-next-line primer/spacing */
167
+ padding-bottom: calc((var(--min-item-height) - var(--custom-line-height, 1.3rem)) / 2);
168
+ line-height: var(--custom-line-height, var(--text-body-lineHeight-medium, 1.4285));
169
+ grid-area: content;
170
+ gap: var(--stack-gap-condensed);
171
+
172
+ & .TreeViewItemCheckbox {
173
+ position: relative;
174
+ color: var(--control-fgColor-rest);
175
+ text-align: left;
176
+ user-select: none;
177
+ background-color: transparent;
178
+ border: none;
179
+ border-radius: var(--borderRadius-medium);
180
+ transition: background 33.333ms linear;
181
+ touch-action: manipulation;
182
+ -webkit-tap-highlight-color: transparent;
183
+ }
184
+
185
+ &[aria-checked='true'] {
186
+ & .FormControl-checkbox {
187
+ background: var(--control-checked-bgColor-rest);
188
+ border-color: var(--control-checked-borderColor-rest);
189
+ transition: background-color, border-color 80ms cubic-bezier(0.32, 0, 0.67, 0) 0ms; /* unchecked -> checked */
190
+
191
+ /* stylelint-disable-next-line max-nesting-depth, selector-max-specificity -- Copied from primer/react */
192
+ &::before {
193
+ visibility: visible;
194
+ transition: visibility 0s linear 0s;
195
+ animation: checkmarkIn 80ms cubic-bezier(0.65, 0, 0.35, 1) forwards 80ms;
196
+ }
197
+ }
198
+ }
199
+
200
+ &[aria-checked='mixed'] {
201
+ & .FormControl-checkbox {
202
+ background: var(--control-checked-bgColor-rest);
203
+ border-color: var(--control-checked-borderColor-rest);
204
+ transition: background-color, border-color 80ms cubic-bezier(0.32, 0, 0.67, 0) 0ms; /* unchecked -> checked */
205
+
206
+ /* stylelint-disable-next-line max-nesting-depth, selector-max-specificity -- Copied from primer/react */
207
+ &::before {
208
+ visibility: visible;
209
+ mask-image: url('data:image/svg+xml;base64,PHN2ZyB3aWR0aD0iMTAiIGhlaWdodD0iMiIgdmlld0JveD0iMCAwIDEwIDIiIGZpbGw9Im5vbmUiIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyI+CjxwYXRoIGZpbGwtcnVsZT0iZXZlbm9kZCIgY2xpcC1ydWxlPSJldmVub2RkIiBkPSJNMCAxQzAgMC40NDc3MTUgMC40NDc3MTUgMCAxIDBIOUM5LjU1MjI5IDAgMTAgMC40NDc3MTUgMTAgMUMxMCAxLjU1MjI4IDkuNTUyMjkgMiA5IDJIMUMwLjQ0NzcxNSAyIDAgMS41NTIyOCAwIDFaIiBmaWxsPSJ3aGl0ZSIvPgo8L3N2Zz4K');
210
+ animation: checkmarkIn 80ms cubic-bezier(0.65, 0, 0.35, 1) forwards 80ms;
211
+ clip-path: none;
212
+ }
213
+ }
214
+ }
215
+
216
+ &[aria-disabled='true'] {
217
+ pointer-events: none;
218
+
219
+ & .TreeViewItemContentText {
220
+ color: var(--control-fgColor-disabled);
221
+ }
222
+
223
+ & .TreeViewItemVisual {
224
+ fill: var(--control-fgColor-disabled);
225
+
226
+ /* stylelint-disable-next-line max-nesting-depth, selector-max-compound-selectors, selector-max-specificity */
227
+ & svg {
228
+ fill: var(--control-fgColor-disabled);
229
+ }
230
+ }
231
+
232
+ & .FormControl-checkbox {
233
+ /* stylelint-disable-next-line max-nesting-depth */
234
+ @media (hover: hover) {
235
+ /* stylelint-disable-next-line max-nesting-depth, selector-max-specificity */
236
+ &:hover {
237
+ cursor: not-allowed;
238
+ }
239
+ }
240
+ }
241
+
242
+ @media (hover: hover) {
243
+ /* stylelint-disable-next-line max-nesting-depth */
244
+ &:hover {
245
+ cursor: not-allowed;
246
+ background-color: transparent;
247
+ }
248
+ }
249
+ }
250
+ }
251
+
252
+ & .TreeViewItemContentText {
253
+ color: var(--control-fgColor-rest);
254
+ flex: 1 1 auto;
255
+ width: 0;
256
+ }
257
+
258
+ &:where([data-truncate-text='true']) .TreeViewItemContentText {
259
+ overflow: hidden;
260
+ text-overflow: ellipsis;
261
+ white-space: nowrap;
262
+ }
263
+
264
+ &:where([data-truncate-text='false']) .TreeViewItemContentText {
265
+ /* stylelint-disable-next-line declaration-property-value-keyword-no-deprecated -- Copied from primer/react */
266
+ word-break: break-word;
267
+ }
268
+
269
+ & .TreeViewItemVisual {
270
+ display: flex;
271
+
272
+ /* The visual icons should appear vertically centered for single-line items, but remain at the top for items that wrap
273
+ across more lines. */
274
+ height: var(--custom-line-height, 1.3rem);
275
+ color: var(--fgColor-muted);
276
+ align-items: center;
277
+ }
278
+
279
+ & .TreeViewItemLeadingAction {
280
+ display: flex;
281
+ color: var(--fgColor-muted);
282
+ grid-area: leadingAction;
283
+
284
+ & > button {
285
+ flex-shrink: 1;
286
+ }
287
+ }
288
+
289
+ & .TreeViewItemLevelLine {
290
+ width: 100%;
291
+ height: 100%;
292
+
293
+ /*
294
+ * On devices without hover, the nesting indicator lines
295
+ * appear at all times.
296
+ */
297
+ border-color: var(--borderColor-muted);
298
+ border-right: var(--borderWidth-thin) solid;
299
+ }
300
+
301
+ /*
302
+ * On devices with :hover support, the nesting indicator lines
303
+ * fade in when the user mouses over the entire component,
304
+ * or when there's focus inside the component. This makes
305
+ * sure the component remains simple when not in use.
306
+ */
307
+ @media (hover: hover) {
308
+ .TreeViewItemLevelLine {
309
+ border-color: transparent;
310
+ }
311
+
312
+ &:hover .TreeViewItemLevelLine,
313
+ &:focus-within .TreeViewItemLevelLine {
314
+ border-color: var(--borderColor-muted);
315
+ }
316
+ }
317
+
318
+ & .TreeViewVisuallyHidden {
319
+ position: absolute;
320
+ width: 1px;
321
+ height: 1px;
322
+ padding: 0;
323
+ /* stylelint-disable-next-line primer/spacing */
324
+ margin: -1px;
325
+ overflow: hidden;
326
+ clip: rect(0, 0, 0, 0);
327
+ white-space: nowrap;
328
+ border-width: 0;
329
+ }
330
+ }
331
+
332
+ .TreeViewSkeletonItemContainerStyle {
333
+ display: flex;
334
+ align-items: center;
335
+ column-gap: 0.5rem;
336
+ height: 2rem;
337
+
338
+ @media (pointer: coarse) {
339
+ height: 2.75rem;
340
+ }
341
+
342
+ &:nth-of-type(5n + 1) {
343
+ --tree-item-loading-width: 67%;
344
+ }
345
+
346
+ &:nth-of-type(5n + 2) {
347
+ --tree-item-loading-width: 47%;
348
+ }
349
+
350
+ &:nth-of-type(5n + 3) {
351
+ --tree-item-loading-width: 73%;
352
+ }
353
+
354
+ &:nth-of-type(5n + 4) {
355
+ --tree-item-loading-width: 64%;
356
+ }
357
+
358
+ &:nth-of-type(5n + 5) {
359
+ --tree-item-loading-width: 50%;
360
+ }
361
+ }
362
+
363
+ .TreeItemSkeletonTextStyles {
364
+ width: var(--tree-item-loading-width, 67%);
365
+ }
366
+
367
+ .TreeViewFailureMessage {
368
+ display: grid;
369
+ grid-template-columns: auto 1fr;
370
+ gap: 0.5rem;
371
+ width: 100%;
372
+ align-items: center;
373
+ }