1
1
import * as React from 'react'
2
2
import { createContext , forwardRef , useCallback , useContext , useEffect , useState } from 'react'
3
3
4
+ import { Icon as CanaryIcon , ScrollArea } from '@/components'
4
5
import * as AccordionPrimitive from '@radix-ui/react-accordion'
5
- import { ScrollArea } from '@radix-ui/react-scroll-area'
6
6
import { cn } from '@utils/cn'
7
7
import { ExecutionState } from '@views/repo/pull-request'
8
8
9
- import { Icon as CanaryIcon } from '../components/icon'
10
-
11
9
type ExecutionDetail = {
12
10
status : ExecutionState
13
11
/* formatted duration */
@@ -17,14 +15,18 @@ type ExecutionDetail = {
17
15
const getStatusIcon = ( status : ExecutionState ) : React . ReactElement => {
18
16
switch ( status ) {
19
17
case ExecutionState . RUNNING :
20
- return < CanaryIcon size = { 16 } name = "running" className = "animate-spin text-warning" />
18
+ return < CanaryIcon size = { 20 } name = "running" className = "animate-spin text-warning" />
21
19
case ExecutionState . SUCCESS :
22
20
return < CanaryIcon name = "success" size = { 16 } className = "text-foreground-success" />
23
21
case ExecutionState . FAILURE :
24
22
return < CanaryIcon name = "fail" size = { 16 } />
25
23
case ExecutionState . WAITING_ON_DEPENDENCIES :
26
24
case ExecutionState . PENDING :
27
- return < CanaryIcon name = "pending-clock" size = { 16 } />
25
+ return (
26
+ < div className = "flex size-5 items-center justify-center" >
27
+ < span className = "size-4 rounded-full border border-icons-7" />
28
+ </ div >
29
+ )
28
30
case ExecutionState . SKIPPED :
29
31
case ExecutionState . UNKNOWN :
30
32
default :
@@ -46,8 +48,6 @@ type TreeContextProps = {
46
48
handleExpand : ( id : string ) => void
47
49
selectItem : ( id : string ) => void
48
50
setExpendedItems ?: React . Dispatch < React . SetStateAction < string [ ] | undefined > >
49
- openIcon ?: React . ReactNode
50
- closeIcon ?: React . ReactNode
51
51
direction : 'rtl' | 'ltr'
52
52
}
53
53
@@ -73,21 +73,7 @@ type TreeViewProps = {
73
73
} & React . HTMLAttributes < HTMLDivElement >
74
74
75
75
const Tree = forwardRef < HTMLDivElement , TreeViewProps > (
76
- (
77
- {
78
- className,
79
- elements,
80
- initialSelectedId,
81
- initialExpendedItems,
82
- children,
83
- indicator = true ,
84
- openIcon,
85
- closeIcon,
86
- dir,
87
- ...props
88
- } ,
89
- ref
90
- ) => {
76
+ ( { className, elements, initialSelectedId, initialExpendedItems, children, indicator = true , dir, ...props } ) => {
91
77
const [ selectedId , setSelectedId ] = useState < string | undefined > ( initialSelectedId )
92
78
const [ expendedItems , setExpendedItems ] = useState < string [ ] | undefined > ( initialExpendedItems )
93
79
@@ -149,26 +135,24 @@ const Tree = forwardRef<HTMLDivElement, TreeViewProps>(
149
135
selectItem,
150
136
setExpendedItems,
151
137
indicator,
152
- openIcon,
153
- closeIcon,
154
138
direction
155
139
} }
156
140
>
157
- < div className = { cn ( 'size-full' , className ) } >
158
- < ScrollArea ref = { ref } className = "relative h -full px-2" dir = { dir as Direction } >
141
+ < ScrollArea className = "pt-4" dir = { dir as Direction } >
142
+ < div className = { cn ( 'size -full' , className ) } >
159
143
< AccordionPrimitive . Root
160
144
{ ...props }
161
145
type = "multiple"
162
146
defaultValue = { expendedItems }
163
147
value = { expendedItems }
164
- className = "flex flex-col gap-1 "
148
+ className = "flex flex-col gap-3 "
165
149
onValueChange = { value => setExpendedItems ( prev => [ ...( prev ?? [ ] ) , value [ 0 ] ] ) }
166
150
dir = { dir as Direction }
167
151
>
168
152
{ children }
169
153
</ AccordionPrimitive . Root >
170
- </ ScrollArea >
171
- </ div >
154
+ </ div >
155
+ </ ScrollArea >
172
156
</ TreeContext . Provider >
173
157
)
174
158
}
@@ -201,45 +185,51 @@ type FolderProps = {
201
185
element : string
202
186
isSelectable ?: boolean
203
187
isSelect ?: boolean
188
+ level : number
204
189
} & React . ComponentPropsWithoutRef < typeof AccordionPrimitive . Item > &
205
190
ExecutionDetail
206
191
207
192
const Folder = forwardRef < HTMLDivElement , FolderProps & React . HTMLAttributes < HTMLDivElement > > (
208
- ( { className, element, value, isSelectable = true , isSelect, children, status, duration, ...props } ) => {
209
- const { direction, handleExpand, expendedItems, indicator , setExpendedItems, openIcon , closeIcon } = useTree ( )
193
+ ( { className, element, value, isSelectable = true , isSelect, children, status, duration, level , ...props } ) => {
194
+ const { direction, handleExpand, expendedItems, setExpendedItems } = useTree ( )
210
195
211
196
return (
212
- < AccordionPrimitive . Item { ...props } value = { value } className = "relative size-full overflow-hidden" >
197
+ < AccordionPrimitive . Item { ...props } value = { value } className = "relative -mb-3 size-full overflow-hidden pb-3 " >
213
198
< AccordionPrimitive . Trigger
214
- className = { cn ( `flex w-full items-center gap-1 rounded-md pb-1.5 text-sm` , className , {
215
- 'rounded-md' : isSelect && isSelectable ,
216
- 'cursor-pointer' : isSelectable ,
217
- 'cursor-not-allowed opacity-50' : ! isSelectable
218
- } ) }
199
+ className = { cn (
200
+ `flex w-full items-center gap-1 rounded-md text-sm px-5` ,
201
+ className ,
202
+ {
203
+ 'rounded-md' : isSelect && isSelectable ,
204
+ 'cursor-pointer' : isSelectable ,
205
+ 'cursor-not-allowed opacity-50' : ! isSelectable
206
+ } ,
207
+ level >= 2 && 'pl-14' ,
208
+ level === 1 && 'pl-7'
209
+ ) }
219
210
disabled = { ! isSelectable }
220
211
onClick = { ( ) => handleExpand ( value ) }
221
212
>
222
- < div className = "mt-1 pt-1" >
223
- { expendedItems ?. includes ( value )
224
- ? ( openIcon ?? < CanaryIcon name = "chevron-down" className = "size-4" height = { 12 } /> )
225
- : ( closeIcon ?? < CanaryIcon name = "chevron-right" className = " size-4" height = { 12 } /> ) }
226
- </ div >
227
- < div className = "mr-1 mt-1 flex w-full items-baseline justify-between" >
228
- < div className = "flex items-baseline " >
229
- < div className = "mr-1 flex self-center" > { getStatusIcon ( status ) } </ div >
230
- < span className = "ml-1 text-sm font-normal " >
231
- { element } < span className = "text-muted- foreground" > ({ React . Children . count ( children ) } )</ span >
213
+ < CanaryIcon
214
+ name = "chevron-right"
215
+ className = { cn ( 'text-icons-1' , expendedItems ?. includes ( value ) && 'rotate-90' ) }
216
+ size = { 12 }
217
+ / >
218
+ < div className = "flex w-full items-center justify-between" >
219
+ < div className = "flex items-center " >
220
+ { getStatusIcon ( status ) }
221
+ < span className = "ml-1 leading-tight text-foreground-8 " >
222
+ { element } < span className = "text-foreground-5 " > ({ React . Children . count ( children ) } )</ span >
232
223
</ span >
233
224
</ div >
234
- < span className = "text-muted- foreground" > { duration ?? '--' } </ span >
225
+ < span className = "text-foreground-4 " > { duration ?? '--' } </ span >
235
226
</ div >
236
227
</ AccordionPrimitive . Trigger >
237
- < AccordionPrimitive . Content className = "relative h-full overflow-hidden text-sm data-[state=closed]:animate-accordion-up data-[state=open]:animate-accordion-down" >
238
- { element && indicator && < TreeIndicator aria-hidden = "true" /> }
228
+ < AccordionPrimitive . Content className = "relative h-full overflow-visible px-5 text-sm data-[state=closed]:animate-accordion-up data-[state=open]:animate-accordion-down" >
239
229
< AccordionPrimitive . Root
240
230
dir = { direction }
241
231
type = "multiple"
242
- className = "flex flex-col gap-1 pl-5 rtl:mr-5"
232
+ className = "mt-3 flex flex-col gap-3 rtl:mr-5"
243
233
defaultValue = { expendedItems }
244
234
value = { expendedItems }
245
235
onValueChange = { value => {
@@ -264,6 +254,7 @@ const File = forwardRef<
264
254
isSelectable ?: boolean
265
255
isSelect ?: boolean
266
256
fileIcon ?: React . ReactNode
257
+ level : number
267
258
} & React . ComponentPropsWithoutRef < typeof AccordionPrimitive . Trigger > &
268
259
ExecutionDetail
269
260
> (
@@ -278,6 +269,7 @@ const File = forwardRef<
278
269
children,
279
270
status,
280
271
duration,
272
+ level,
281
273
...props
282
274
} ,
283
275
ref
@@ -293,22 +285,27 @@ const File = forwardRef<
293
285
disabled = { ! isSelectable }
294
286
aria-label = "File"
295
287
className = { cn (
296
- 'flex w-full cursor-pointer items-center gap-1 rounded-md px-1 py-1 text-sm duration-200 ease-in-out rtl:pl-1 rtl:pr-0' ,
297
- { [ 'bg-[#18181B] px-2' ] : isSelected } ,
288
+ 'flex relative w-full cursor-pointer items-center gap-1 rounded-md text-sm duration-200 ease-in-out rtl:pl-1 rtl:pr-0' ,
289
+ {
290
+ [ 'after:absolute after:bg-background-4 after:-inset-x-1 after:-inset-y-1.5 after:-z-10 after:rounded' ] :
291
+ isSelected
292
+ } ,
298
293
isSelectable ? 'cursor-pointer' : 'cursor-not-allowed opacity-50' ,
294
+ level >= 2 && 'pl-14' ,
295
+ level === 1 && 'pl-7' ,
299
296
className
300
297
) }
301
298
onClick = { ( ) => {
302
299
handleSelect ?.( value )
303
300
selectItem ( value )
304
301
} }
305
302
>
306
- < div className = "flex w-full items-baseline justify-between" >
307
- < div className = "flex items-baseline " >
308
- < div className = "mr-1 flex size-4 self-center" > { getStatusIcon ( status ) } </ div >
309
- < span className = "ml-1 text-sm font-normal " > { children } </ span >
303
+ < div className = "relative flex w-full items-center justify-between pl-4 " >
304
+ < div className = "flex items-center " >
305
+ { getStatusIcon ( status ) }
306
+ < span className = "ml-1 leading-tight text-foreground-8 " > { children } </ span >
310
307
</ div >
311
- < span className = "text-muted- foreground" > { duration ?? '--' } </ span >
308
+ < span className = "text-foreground-4 " > { duration ?? '--' } </ span >
312
309
</ div >
313
310
</ AccordionPrimitive . Trigger >
314
311
</ AccordionPrimitive . Item >
0 commit comments