6
6
7
7
import { ARIARole } from '@material/web/types/aria' ;
8
8
import { html , LitElement , PropertyValues , TemplateResult } from 'lit' ;
9
- import { queryAssignedElements } from 'lit/decorators' ;
9
+ import { property , query , queryAssignedElements } from 'lit/decorators' ;
10
10
11
11
import { ListItemInteractionEvent } from './listitem/constants' ;
12
12
import { ListItem } from './listitem/list-item' ;
13
13
14
+ const NAVIGATABLE_KEYS = {
15
+ ArrowDown : 'ArrowDown' ,
16
+ ArrowUp : 'ArrowUp' ,
17
+ Home : 'Home' ,
18
+ End : 'End' ,
19
+ } ;
20
+
14
21
/** @soyCompatible */
15
22
export class List extends LitElement {
16
23
static override shadowRootOptions :
17
24
ShadowRootInit = { mode : 'open' , delegatesFocus : true } ;
18
25
26
+ @property ( { type : Number } ) listTabIndex : number = 0 ;
27
+
19
28
items : ListItem [ ] = [ ] ;
29
+ activeListItem : ListItem | null = null ;
30
+
31
+ @query ( '.md3-list' ) listRoot ! : HTMLElement ;
20
32
21
33
@queryAssignedElements ( { flatten : true } )
22
34
protected assignedElements ! : HTMLElement [ ] | null ;
@@ -36,20 +48,76 @@ export class List extends LitElement {
36
48
override render ( ) : TemplateResult {
37
49
return html `
38
50
< ul class ="md3-list "
39
- tabindex =" 0 "
51
+ tabindex =${ this . listTabIndex }
40
52
role =${ this . getAriaRole ( ) }
41
- @list-item-interaction =${ this . handleItemInteraction } >
53
+ @list-item-interaction=${ this . handleItemInteraction }
54
+ @keydown=${ this . handleKeydown }
55
+ >
42
56
< slot > </ slot >
43
57
</ ul >
44
58
` ;
45
59
}
46
60
61
+ handleKeydown ( event : KeyboardEvent ) {
62
+ if ( Object . values ( NAVIGATABLE_KEYS ) . indexOf ( event . key ) === - 1 ) return ;
63
+
64
+ if ( event . key === NAVIGATABLE_KEYS . ArrowDown ) {
65
+ event . preventDefault ( ) ;
66
+ if ( this . activeListItem ) {
67
+ this . activeListItem = this . getNextItem ( this . activeListItem ) ;
68
+ } else {
69
+ this . activeListItem = this . getFirstItem ( ) ;
70
+ }
71
+ }
72
+
73
+ if ( event . key === NAVIGATABLE_KEYS . ArrowUp ) {
74
+ event . preventDefault ( ) ;
75
+ if ( this . activeListItem ) {
76
+ this . activeListItem = this . getPrevItem ( this . activeListItem ) ;
77
+ } else {
78
+ this . activeListItem = this . getLastItem ( ) ;
79
+ }
80
+ }
81
+
82
+ if ( event . key === NAVIGATABLE_KEYS . Home ) {
83
+ event . preventDefault ( ) ;
84
+ this . activeListItem = this . getFirstItem ( ) ;
85
+ }
86
+
87
+ if ( event . key === NAVIGATABLE_KEYS . End ) {
88
+ event . preventDefault ( ) ;
89
+ this . activeListItem = this . getLastItem ( ) ;
90
+ }
91
+
92
+ if ( ! this . activeListItem ) return ;
93
+
94
+ for ( const item of this . items ) {
95
+ item . deactivate ( ) ;
96
+ }
97
+
98
+ this . activeListItem . activate ( ) ;
99
+ }
100
+
47
101
handleItemInteraction ( event : ListItemInteractionEvent ) {
48
102
if ( event . detail . state . isSelected ) {
49
103
// TODO: manage selection state.
50
104
}
51
105
}
52
106
107
+ activateFirstItem ( ) {
108
+ this . activeListItem = this . getFirstItem ( ) ;
109
+ this . activeListItem . activate ( ) ;
110
+ }
111
+
112
+ activateLastItem ( ) {
113
+ this . activeListItem = this . getLastItem ( ) ;
114
+ this . activeListItem . activate ( ) ;
115
+ }
116
+
117
+ focusListRoot ( ) {
118
+ this . listRoot . focus ( ) ;
119
+ }
120
+
53
121
/** Updates `this.items` based on slot elements in the DOM. */
54
122
protected updateItems ( ) {
55
123
const elements = this . assignedElements || [ ] ;
@@ -64,4 +132,22 @@ export class List extends LitElement {
64
132
private isListItem ( element : Element ) : element is ListItem {
65
133
return element . tagName . toLowerCase ( ) === this . getListItemTagName ( ) ;
66
134
}
135
+
136
+ private getFirstItem ( ) : ListItem {
137
+ return this . items [ 0 ] ;
138
+ }
139
+
140
+ private getLastItem ( ) : ListItem {
141
+ return this . items [ this . items . length - 1 ] ;
142
+ }
143
+
144
+ private getPrevItem ( item : ListItem ) : ListItem {
145
+ const curIndex = this . items . indexOf ( item ) ;
146
+ return this . items [ curIndex === 0 ? this . items . length - 1 : curIndex - 1 ] ;
147
+ }
148
+
149
+ private getNextItem ( item : ListItem ) : ListItem {
150
+ const curIndex = this . items . indexOf ( item ) ;
151
+ return this . items [ ( curIndex + 1 ) % this . items . length ] ;
152
+ }
67
153
}
0 commit comments