Skip to content

Commit 7bd02aa

Browse files
committedMar 21, 2025·
refactor to have slotted start and end inputs
1 parent 225acf3 commit 7bd02aa

File tree

5 files changed

+423
-178
lines changed

5 files changed

+423
-178
lines changed
 

‎src/components/common/definitions/defineAllComponents.ts

+2
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,7 @@ import IgcSwitchComponent from '../../checkbox/switch.js';
2121
import IgcChipComponent from '../../chip/chip.js';
2222
import IgcComboComponent from '../../combo/combo.js';
2323
import IgcDatePickerComponent from '../../date-picker/date-picker.js';
24+
import IgcDateRangeInputComponent from '../../date-range-picker/date-range-input.js';
2425
import IgcDateTimeInputComponent from '../../date-time-input/date-time-input.js';
2526
import IgcDialogComponent from '../../dialog/dialog.js';
2627
import IgcDividerComponent from '../../divider/divider.js';
@@ -91,6 +92,7 @@ const allComponents: IgniteComponent[] = [
9192
IgcChipComponent,
9293
IgcComboComponent,
9394
IgcDatePickerComponent,
95+
IgcDateRangeInputComponent,
9496
IgcDateRangePickerComponent,
9597
IgcDropdownComponent,
9698
IgcDropdownGroupComponent,
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,172 @@
1+
import { LitElement, html, nothing } from 'lit';
2+
import { property, query, queryAssignedElements } from 'lit/decorators.js';
3+
import { ifDefined } from 'lit/directives/if-defined.js';
4+
import { convertToDate } from '../calendar/helpers.js';
5+
import { registerComponent } from '../common/definitions/register.js';
6+
import type { Constructor } from '../common/mixins/constructor.js';
7+
import { EventEmitterMixin } from '../common/mixins/event-emitter.js';
8+
import { createCounter } from '../common/util.js';
9+
import IgcDateTimeInputComponent from '../date-time-input/date-time-input.js';
10+
11+
export const setInputFormat = Symbol();
12+
export const setDisplayFormat = Symbol();
13+
14+
export interface IgcDatePickerComponentEventMap {
15+
igcToggleIconClicked: CustomEvent<void>;
16+
igcClearIconClicked: CustomEvent<void>;
17+
igcChange: CustomEvent<Date | null>;
18+
igcInput: CustomEvent<Date | null>;
19+
}
20+
21+
// should this be form associated?
22+
export default class IgcDateRangeInputComponent extends EventEmitterMixin<
23+
IgcDatePickerComponentEventMap,
24+
Constructor<LitElement>
25+
>(LitElement) {
26+
public static readonly tagName = 'igc-date-range-input';
27+
28+
private static readonly increment = createCounter();
29+
public inputId = `date-range-input-${IgcDateRangeInputComponent.increment()}`;
30+
31+
protected static shadowRootOptions = {
32+
...LitElement.shadowRootOptions,
33+
delegatesFocus: true,
34+
};
35+
36+
/* blazorSuppress */
37+
public static register() {
38+
registerComponent(IgcDateRangeInputComponent);
39+
}
40+
41+
@query(IgcDateTimeInputComponent.tagName)
42+
private _input!: IgcDateTimeInputComponent;
43+
44+
@queryAssignedElements({ slot: 'prefix' })
45+
private prefixes!: Array<HTMLElement>;
46+
47+
@queryAssignedElements({ slot: 'suffix' })
48+
private suffixes!: Array<HTMLElement>;
49+
50+
private _open = false;
51+
private _value: Date | null = null;
52+
53+
/**
54+
* The value of the picker
55+
* @attr
56+
*/
57+
@property({ converter: convertToDate })
58+
public set value(value: Date | string | null | undefined) {
59+
this._value = value as Date | null;
60+
}
61+
62+
public get value(): Date | null {
63+
return this._value;
64+
}
65+
66+
/**
67+
* The label of the datepicker.
68+
* @attr label
69+
*/
70+
@property()
71+
public label!: string;
72+
73+
protected override createRenderRoot() {
74+
const root = super.createRenderRoot();
75+
root.addEventListener('slotchange', () => this.requestUpdate());
76+
return root;
77+
}
78+
79+
private handleAnchorClick() {
80+
this.emitEvent('igcToggleIconClicked');
81+
}
82+
83+
private clearIconClick() {
84+
this.emitEvent('igcClearIconClicked');
85+
}
86+
87+
/** @private @hidden @internal */
88+
public async [setInputFormat](value: string) {
89+
this._input.inputFormat = value;
90+
}
91+
92+
/** @private @hidden @internal */
93+
public async [setDisplayFormat](value: string) {
94+
this._input.displayFormat = value;
95+
}
96+
97+
public clear() {
98+
this.value = null;
99+
this._input.clear();
100+
}
101+
102+
protected handleInputChangeEvent(event: CustomEvent<Date>) {
103+
event.stopPropagation();
104+
this.value = (event.target as IgcDateTimeInputComponent).value!;
105+
this.emitEvent('igcChange', { detail: this.value });
106+
}
107+
108+
protected handleInputEvent(event: CustomEvent<Date>) {
109+
event.stopPropagation();
110+
this.value = (event.target as IgcDateTimeInputComponent).value!;
111+
this.emitEvent('igcInput', { detail: this.value });
112+
}
113+
114+
private renderClearIcon() {
115+
return !this.value
116+
? nothing
117+
: html`
118+
<span slot="suffix" part="clear-icon" @click=${this.clearIconClick}>
119+
<slot name="clear-icon">
120+
<igc-icon
121+
name="input_clear"
122+
collection="default"
123+
aria-hidden="true"
124+
></igc-icon>
125+
</slot>
126+
</span>
127+
`;
128+
}
129+
130+
private renderCalendarIcon() {
131+
const defaultIcon = html`
132+
<igc-icon name="today" collection="default" aria-hidden="true"></igc-icon>
133+
`;
134+
135+
const state = this._open ? 'calendar-icon-open' : 'calendar-icon';
136+
137+
return html`
138+
<span slot="prefix" part=${state} @click=${this.handleAnchorClick}>
139+
<slot name=${state}>${defaultIcon}</slot>
140+
</span>
141+
`;
142+
}
143+
144+
protected override render() {
145+
return html`
146+
<igc-date-time-input
147+
id=${this.inputId}
148+
.value=${this.value}
149+
.label=${this.label}
150+
@igcChange=${this.handleInputChangeEvent}
151+
@igcInput=${this.handleInputEvent}
152+
>
153+
${this.renderCalendarIcon()}
154+
<slot
155+
name="prefix"
156+
slot=${ifDefined(!this.prefixes.length ? undefined : 'prefix')}
157+
></slot>
158+
${this.renderClearIcon()}
159+
<slot
160+
name="suffix"
161+
slot=${ifDefined(!this.suffixes.length ? undefined : 'suffix')}
162+
></slot>
163+
</igc-date-time-input>
164+
`;
165+
}
166+
}
167+
168+
declare global {
169+
interface HTMLElementTagNameMap {
170+
'igc-date-range-input': IgcDateRangeInputComponent;
171+
}
172+
}

0 commit comments

Comments
 (0)