@@ -17,39 +17,17 @@ import {when} from 'lit/directives/when.js';
17
17
import { ARIAMixinStrict } from '../../internal/aria/aria.js' ;
18
18
import { requestUpdateOnAriaChange } from '../../internal/aria/delegate.js' ;
19
19
import { dispatchActivationClick , isActivationClick , redispatchEvent } from '../../internal/controller/events.js' ;
20
- import { FormController , getFormValue } from '../../internal/controller/form-controller.js' ;
21
- import { stringConverter } from '../../internal/controller/string-converter.js' ;
22
20
import { MdRipple } from '../../ripple/ripple.js' ;
23
21
24
22
// Disable warning for classMap with destructuring
25
23
// tslint:disable:quoted-properties-on-dictionary
26
24
27
- function inBounds ( { x, y} : PointerEvent , element ?: HTMLElement | null ) {
28
- if ( ! element ) {
29
- return false ;
30
- }
31
- const { top, left, bottom, right} = element . getBoundingClientRect ( ) ;
32
- return x >= left && x <= right && y >= top && y <= bottom ;
33
- }
34
-
35
- function isOverlapping ( elA : Element | null , elB : Element | null ) {
36
- if ( ! ( elA && elB ) ) {
37
- return false ;
38
- }
39
- const a = elA . getBoundingClientRect ( ) ;
40
- const b = elB . getBoundingClientRect ( ) ;
41
- return ! (
42
- a . top > b . bottom || a . right < b . left || a . bottom < b . top ||
43
- a . left > b . right ) ;
44
- }
45
-
46
- interface Action {
47
- canFlip : boolean ;
48
- flipped : boolean ;
49
- target : HTMLInputElement ;
50
- fixed : HTMLInputElement ;
51
- values : Map < HTMLInputElement | undefined , number | undefined > ;
52
- }
25
+ /** The default value for a continuous slider. */
26
+ const DEFAULT_VALUE = 50 ;
27
+ /** The default start value for a range slider. */
28
+ const DEFAULT_VALUE_START = 25 ;
29
+ /** The default end value for a range slider. */
30
+ const DEFAULT_VALUE_END = 75 ;
53
31
54
32
/**
55
33
* Slider component.
@@ -86,17 +64,19 @@ export class Slider extends LitElement {
86
64
/**
87
65
* The slider value displayed when range is false.
88
66
*/
89
- @property ( { type : Number } ) value = 50 ;
67
+ @property ( { type : Number } ) value = DEFAULT_VALUE ;
90
68
91
69
/**
92
70
* The slider start value displayed when range is true.
93
71
*/
94
- @property ( { type : Number } ) valueStart = 25 ;
72
+ @property ( { type : Number , attribute : 'value-start' } )
73
+ valueStart = DEFAULT_VALUE_START ;
95
74
96
75
/**
97
76
* The slider end value displayed when range is true.
98
77
*/
99
- @property ( { type : Number } ) valueEnd = 75 ;
78
+ @property ( { type : Number , attribute : 'value-end' } )
79
+ valueEnd = DEFAULT_VALUE_END ;
100
80
101
81
/**
102
82
* An optional label for the slider's value displayed when range is
@@ -153,13 +133,49 @@ export class Slider extends LitElement {
153
133
/**
154
134
* The HTML name to use in form submission.
155
135
*/
156
- @property ( { reflect : true , converter : stringConverter } ) name = '' ;
136
+ get name ( ) {
137
+ return this . getAttribute ( 'name' ) ?? '' ;
138
+ }
139
+ set name ( name : string ) {
140
+ this . setAttribute ( 'name' , name ) ;
141
+ }
142
+
143
+ /**
144
+ * The HTML name to use in form submission for a range slider's starting
145
+ * value. Use `name` instead if both the start and end values should use the
146
+ * same name.
147
+ */
148
+ get nameStart ( ) {
149
+ return this . getAttribute ( 'name-start' ) ?? this . name ;
150
+ }
151
+ set nameStart ( name : string ) {
152
+ this . setAttribute ( 'name-start' , name ) ;
153
+ }
154
+
155
+ /**
156
+ * The HTML name to use in form submission for a range slider's ending value.
157
+ * Use `name` instead if both the start and end values should use the same
158
+ * name.
159
+ */
160
+ get nameEnd ( ) {
161
+ return this . getAttribute ( 'name-end' ) ?? this . nameStart ;
162
+ }
163
+ set nameEnd ( name : string ) {
164
+ this . setAttribute ( 'name-end' , name ) ;
165
+ }
157
166
158
167
/**
159
168
* The associated form element with which this element's value will submit.
160
169
*/
161
170
get form ( ) {
162
- return this . closest ( 'form' ) ;
171
+ return this . internals . form ;
172
+ }
173
+
174
+ /**
175
+ * The labels this element is associated with.
176
+ */
177
+ get labels ( ) {
178
+ return this . internals . labels ;
163
179
}
164
180
165
181
@query ( 'input.start' ) private readonly inputStart ! : HTMLInputElement | null ;
@@ -193,9 +209,11 @@ export class Slider extends LitElement {
193
209
194
210
private action ?: Action ;
195
211
212
+ private readonly internals =
213
+ ( this as HTMLElement /* needed for closure */ ) . attachInternals ( ) ;
214
+
196
215
constructor ( ) {
197
216
super ( ) ;
198
- this . addController ( new FormController ( this ) ) ;
199
217
if ( ! isServer ) {
200
218
this . addEventListener ( 'click' , ( event : MouseEvent ) => {
201
219
if ( ! isActivationClick ( event ) || ! this . inputEnd ) {
@@ -211,12 +229,6 @@ export class Slider extends LitElement {
211
229
this . inputEnd ?. focus ( ) ;
212
230
}
213
231
214
- // value coerced to a string
215
- [ getFormValue ] ( ) {
216
- return this . range ? `${ this . valueStart } , ${ this . valueEnd } ` :
217
- `${ this . value } ` ;
218
- }
219
-
220
232
protected override willUpdate ( changed : PropertyValues ) {
221
233
this . renderValueStart = changed . has ( 'valueStart' ) ?
222
234
this . valueStart :
@@ -235,6 +247,22 @@ export class Slider extends LitElement {
235
247
}
236
248
}
237
249
250
+ protected override update ( changed : PropertyValues < Slider > ) {
251
+ if ( changed . has ( 'value' ) || changed . has ( 'range' ) ||
252
+ changed . has ( 'valueStart' ) || changed . has ( 'valueEnd' ) ) {
253
+ if ( this . range ) {
254
+ const data = new FormData ( ) ;
255
+ data . append ( this . nameStart , String ( this . valueStart ) ) ;
256
+ data . append ( this . nameEnd , String ( this . valueEnd ) ) ;
257
+ this . internals . setFormValue ( data ) ;
258
+ } else {
259
+ this . internals . setFormValue ( String ( this . value ) ) ;
260
+ }
261
+ }
262
+
263
+ super . update ( changed ) ;
264
+ }
265
+
238
266
protected override updated ( changed : PropertyValues ) {
239
267
// Validate input rendered value and re-render if necessary. This ensures
240
268
// the rendred handle stays in sync with the input thumb which is used for
@@ -590,4 +618,58 @@ export class Slider extends LitElement {
590
618
// ensure keyboard triggered change clears action.
591
619
this . finishAction ( e ) ;
592
620
}
621
+
622
+ /** @private */
623
+ formResetCallback ( ) {
624
+ if ( this . range ) {
625
+ this . valueStart =
626
+ Number ( this . getAttribute ( 'value-start' ) ?? DEFAULT_VALUE_START ) ;
627
+ this . valueEnd =
628
+ Number ( this . getAttribute ( 'value-end' ) ?? DEFAULT_VALUE_END ) ;
629
+ return ;
630
+ }
631
+
632
+ this . value = Number ( this . getAttribute ( 'value' ) ?? DEFAULT_VALUE ) ;
633
+ }
634
+
635
+ /** @private */
636
+ formStateRestoreCallback ( state : string | Array < [ string , string ] > | null ) {
637
+ if ( Array . isArray ( state ) ) {
638
+ const [ [ , valueStart ] , [ , valueEnd ] ] = state ;
639
+ this . valueStart = Number ( valueStart ?? DEFAULT_VALUE_START ) ;
640
+ this . valueEnd = Number ( valueEnd ?? DEFAULT_VALUE_START ) ;
641
+ this . range = true ;
642
+ return ;
643
+ }
644
+
645
+ this . value = Number ( state ?? DEFAULT_VALUE ) ;
646
+ this . range = false ;
647
+ }
648
+ }
649
+
650
+ function inBounds ( { x, y} : PointerEvent , element ?: HTMLElement | null ) {
651
+ if ( ! element ) {
652
+ return false ;
653
+ }
654
+ const { top, left, bottom, right} = element . getBoundingClientRect ( ) ;
655
+ return x >= left && x <= right && y >= top && y <= bottom ;
656
+ }
657
+
658
+ function isOverlapping ( elA : Element | null , elB : Element | null ) {
659
+ if ( ! ( elA && elB ) ) {
660
+ return false ;
661
+ }
662
+ const a = elA . getBoundingClientRect ( ) ;
663
+ const b = elB . getBoundingClientRect ( ) ;
664
+ return ! (
665
+ a . top > b . bottom || a . right < b . left || a . bottom < b . top ||
666
+ a . left > b . right ) ;
667
+ }
668
+
669
+ interface Action {
670
+ canFlip : boolean ;
671
+ flipped : boolean ;
672
+ target : HTMLInputElement ;
673
+ fixed : HTMLInputElement ;
674
+ values : Map < HTMLInputElement | undefined , number | undefined > ;
593
675
}
0 commit comments