Skip to content

Commit e021a8f

Browse files
authored
TreeView: Add truncate prop (#5122)
* Add "opt in" prop for `TreeView` to opt in to truncation of text * Make non-truncate opt-in * Add extra CSS, stories * Add changeset * Cleanup stories * Remove `truncate` from story * Add back prop to story * test(vrt): update snapshots * Fix story typos * test(vrt): update snapshots --------- Co-authored-by: TylerJDev <[email protected]>
1 parent 688ed1e commit e021a8f

22 files changed

+30
-14
lines changed

.changeset/slow-shoes-wait.md

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
---
2+
'@primer/react': minor
3+
---
4+
5+
TreeView: Adds prop `truncate`, keeps default behavior of truncation but allows for text to wrap when turned off.
Loading
Loading

packages/react/src/TreeView/TreeView.docs.json

+6
Original file line numberDiff line numberDiff line change
@@ -69,6 +69,12 @@
6969
"name": "onSelect",
7070
"type": "(event: React.MouseEvent<HTMLElement> | React.KeyboardEvent<HTMLElement>) => void"
7171
},
72+
{
73+
"name": "truncate",
74+
"type": "boolean",
75+
"defaultValue": "true",
76+
"description": "Truncate the item's text content with an ellipsis."
77+
},
7278
{
7379
"name": "ref",
7480
"type": "React.Ref<HTMLElement>"

packages/react/src/TreeView/TreeView.features.stories.tsx

+9-13
Original file line numberDiff line numberDiff line change
@@ -34,7 +34,7 @@ const meta: Meta = {
3434

3535
export const Files: StoryFn = () => (
3636
<nav aria-label="Files">
37-
<TreeView aria-label="Files">
37+
<TreeView aria-label="Files" truncate={false}>
3838
<TreeView.Item id="src" defaultExpanded>
3939
<TreeView.LeadingVisual>
4040
<TreeView.DirectoryIcon />
@@ -112,7 +112,7 @@ export const Files: StoryFn = () => (
112112
export const FilesChanged: StoryFn = () => {
113113
return (
114114
<nav aria-label="Files">
115-
<TreeView aria-label="Files">
115+
<TreeView aria-label="Files" truncate={false}>
116116
<TreeView.Item id="src" defaultExpanded>
117117
<TreeView.LeadingVisual>
118118
<TreeView.DirectoryIcon />
@@ -786,7 +786,7 @@ export const ContainIntrinsicSize: StoryFn = () => {
786786
export const InitialFocus: StoryFn = () => (
787787
<div>
788788
<Button>Focusable element before TreeView</Button>
789-
<TreeView aria-label="Test tree">
789+
<TreeView aria-label="Test tree" truncate={false}>
790790
<TreeView.Item id="src" defaultExpanded>
791791
<TreeView.LeadingVisual>
792792
<TreeView.DirectoryIcon />
@@ -891,11 +891,11 @@ export const InitialFocus: StoryFn = () => (
891891
</TreeView.Item>
892892
</TreeView.SubTree>
893893
</TreeView.Item>
894-
<TreeView.Item id="src/ReallyLongFileNameThatShouldBeTruncated.tsx">
894+
<TreeView.Item id="src/ReallyLongFileNameThatShouldNotBeTruncated.tsx">
895895
<TreeView.LeadingVisual>
896896
<FileIcon />
897897
</TreeView.LeadingVisual>
898-
ReallyLongFileNameThatShouldBeTruncated.tsx
898+
ReallyLongFileNameThatShouldNotBeTruncated.tsx
899899
</TreeView.Item>
900900
</TreeView.SubTree>
901901
</TreeView.Item>
@@ -1047,15 +1047,13 @@ export const LeadingAction: StoryFn = () => {
10471047

10481048
export const MultilineItems: StoryFn = () => (
10491049
<nav aria-label="Files changed">
1050-
<TreeView aria-label="Files changed">
1050+
<TreeView aria-label="Files changed" truncate={false}>
10511051
<TreeView.Item id="src" defaultExpanded>
10521052
<TreeView.LeadingVisual>
10531053
<TreeView.DirectoryIcon />
10541054
</TreeView.LeadingVisual>
1055-
<div style={{whiteSpace: 'wrap'}}>
1056-
this is a very long directory name that we have intentionally allowed to wrap over multiple lines to
1057-
demonstrate alignment
1058-
</div>
1055+
this is a very long directory name that we have intentionally allowed to wrap over multiple lines to demonstrate
1056+
alignment
10591057
<TreeView.SubTree>
10601058
<TreeView.Item id="src/Avatar.tsx">
10611059
<TreeView.LeadingVisual>
@@ -1072,9 +1070,7 @@ export const MultilineItems: StoryFn = () => (
10721070
<TreeView.LeadingVisual>
10731071
<TreeView.DirectoryIcon />
10741072
</TreeView.LeadingVisual>
1075-
<div style={{whiteSpace: 'wrap'}}>
1076-
this is a medium directory name that we wrap over 2 lines to demonstrate alignment
1077-
</div>
1073+
this is a medium directory name that we wrap over 2 lines to demonstrate alignment
10781074
<TreeView.TrailingVisual label="Added">
10791075
<Octicon icon={DiffAddedIcon} color="success.fg" />
10801076
</TreeView.TrailingVisual>

packages/react/src/TreeView/TreeView.tsx

+10-1
Original file line numberDiff line numberDiff line change
@@ -66,6 +66,7 @@ export type TreeViewProps = {
6666
'aria-labelledby'?: React.AriaAttributes['aria-labelledby']
6767
children: React.ReactNode
6868
flat?: boolean
69+
truncate?: boolean
6970
className?: string
7071
}
7172

@@ -205,14 +206,20 @@ const UlBox = styled.ul<SxProp>`
205206
}
206207
207208
.PRIVATE_TreeView-item-content-text {
208-
/* Truncate text label */
209209
flex: 1 1 auto;
210210
width: 0;
211+
}
212+
213+
&[data-truncate-text='true'] .PRIVATE_TreeView-item-content-text {
211214
overflow: hidden;
212215
white-space: nowrap;
213216
text-overflow: ellipsis;
214217
}
215218
219+
&[data-truncate-text='false'] .PRIVATE_TreeView-item-content-text {
220+
word-break: break-word;
221+
}
222+
216223
.PRIVATE_TreeView-item-visual {
217224
display: flex;
218225
align-items: center;
@@ -282,6 +289,7 @@ const Root: React.FC<TreeViewProps> = ({
282289
'aria-labelledby': ariaLabelledby,
283290
children,
284291
flat,
292+
truncate = true,
285293
className,
286294
}) => {
287295
const containerRef = React.useRef<HTMLUListElement>(null)
@@ -338,6 +346,7 @@ const Root: React.FC<TreeViewProps> = ({
338346
aria-label={ariaLabel}
339347
aria-labelledby={ariaLabelledby}
340348
data-omit-spacer={flat}
349+
data-truncate-text={truncate || false}
341350
onMouseDown={onMouseDown}
342351
className={className}
343352
>

0 commit comments

Comments
 (0)