@triptease/tt-date-picker 6.1.3 → 6.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (40) hide show
  1. package/dist/cjs/package.json +3 -0
  2. package/dist/cjs/src/TtDatePicker.js +220 -0
  3. package/dist/{src → cjs/src}/TtDatePicker.js.map +1 -1
  4. package/dist/cjs/src/helpers.js +9 -0
  5. package/dist/cjs/src/helpers.js.map +1 -0
  6. package/dist/cjs/src/index.js +21 -0
  7. package/dist/cjs/src/index.js.map +1 -0
  8. package/dist/cjs/src/styles.js +70 -0
  9. package/dist/cjs/src/styles.js.map +1 -0
  10. package/dist/cjs/src/tt-date-picker.js +23 -0
  11. package/dist/cjs/src/tt-date-picker.js.map +1 -0
  12. package/dist/cjs/src/types.js +3 -0
  13. package/dist/cjs/src/types.js.map +1 -0
  14. package/dist/esm/src/TtDatePicker.js.map +1 -0
  15. package/dist/esm/src/helpers.js.map +1 -0
  16. package/dist/esm/src/index.js.map +1 -0
  17. package/dist/{src → esm/src}/styles.js +1 -1
  18. package/dist/esm/src/styles.js.map +1 -0
  19. package/dist/esm/src/tt-date-picker.js.map +1 -0
  20. package/dist/esm/src/types.js.map +1 -0
  21. package/dist/esm/test/date-picker.test.d.ts +1 -0
  22. package/dist/esm/test/date-picker.test.js +57 -0
  23. package/dist/esm/test/date-picker.test.js.map +1 -0
  24. package/package.json +18 -13
  25. package/dist/src/helpers.js.map +0 -1
  26. package/dist/src/index.js.map +0 -1
  27. package/dist/src/styles.js.map +0 -1
  28. package/dist/src/tt-date-picker.js.map +0 -1
  29. package/dist/src/types.js.map +0 -1
  30. /package/dist/{src → esm/src}/TtDatePicker.d.ts +0 -0
  31. /package/dist/{src → esm/src}/TtDatePicker.js +0 -0
  32. /package/dist/{src → esm/src}/helpers.d.ts +0 -0
  33. /package/dist/{src → esm/src}/helpers.js +0 -0
  34. /package/dist/{src → esm/src}/index.d.ts +0 -0
  35. /package/dist/{src → esm/src}/index.js +0 -0
  36. /package/dist/{src → esm/src}/styles.d.ts +0 -0
  37. /package/dist/{src → esm/src}/tt-date-picker.d.ts +0 -0
  38. /package/dist/{src → esm/src}/tt-date-picker.js +0 -0
  39. /package/dist/{src → esm/src}/types.d.ts +0 -0
  40. /package/dist/{src → esm/src}/types.js +0 -0
@@ -0,0 +1,3 @@
1
+ {
2
+ "type": "commonjs"
3
+ }
@@ -0,0 +1,220 @@
1
+ "use strict";
2
+ var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
3
+ var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
4
+ if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
5
+ else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
6
+ return c > 3 && r && Object.defineProperty(target, key, r), r;
7
+ };
8
+ Object.defineProperty(exports, "__esModule", { value: true });
9
+ exports.TtDatePicker = void 0;
10
+ const lit_1 = require("lit");
11
+ const decorators_js_1 = require("lit/decorators.js");
12
+ const unsafe_svg_js_1 = require("lit/directives/unsafe-svg.js");
13
+ const luxon_1 = require("luxon");
14
+ const icons_1 = require("@triptease/icons");
15
+ const date_selection_context_js_1 = require("@triptease/tt-calendar/date-selection-context.js");
16
+ const helpers_js_1 = require("./helpers.js");
17
+ const styles_js_1 = require("./styles.js");
18
+ class TtDatePicker extends lit_1.LitElement {
19
+ // Internal DateTime properties for calculations
20
+ get _valueDateTime() {
21
+ return this.value ? luxon_1.DateTime.fromISO(this.value) : undefined;
22
+ }
23
+ get _minDateTime() {
24
+ return this.minDate ? luxon_1.DateTime.fromJSDate(this.minDate) : undefined;
25
+ }
26
+ get _maxDateTime() {
27
+ return this.maxDate ? luxon_1.DateTime.fromJSDate(this.maxDate) : undefined;
28
+ }
29
+ get _formValue() {
30
+ return this.value ?? '';
31
+ }
32
+ constructor() {
33
+ super();
34
+ this.openLeft = false;
35
+ this.disabled = false;
36
+ this._open = false;
37
+ this._onDialogOpen = () => {
38
+ this._open = true;
39
+ };
40
+ this._onDialogClose = () => {
41
+ this._open = false;
42
+ };
43
+ this.onFocusOut = (event) => {
44
+ if (!this._open)
45
+ return;
46
+ const target = (event.relatedTarget ?? event.explicitOriginalTarget);
47
+ if (!this.contains(target)) {
48
+ this.dialog.hide();
49
+ }
50
+ };
51
+ this._onFocus = () => {
52
+ this.dateInput.focus();
53
+ };
54
+ this.internals = this.attachInternals();
55
+ }
56
+ onDateChange(e) {
57
+ if (date_selection_context_js_1.DateSelectionEvent.is(e)) {
58
+ this.value = e.detail;
59
+ }
60
+ else if (e.type === 'change') {
61
+ this.value = e.target.value;
62
+ }
63
+ this.dispatchEvent(new InputEvent('change'));
64
+ this.dialog.hide();
65
+ }
66
+ get form() {
67
+ return this.internals.form;
68
+ }
69
+ updated(changedProperties) {
70
+ super.updated(changedProperties);
71
+ if (changedProperties.has('value')) {
72
+ this.internals.setFormValue(this._formValue);
73
+ this.validate();
74
+ }
75
+ if (changedProperties.has('disabled')) {
76
+ if (this.disabled) {
77
+ this.internals.setFormValue(null);
78
+ this.internals.states.add('disabled');
79
+ this.inert = true;
80
+ }
81
+ else {
82
+ this.internals.states.delete('disabled');
83
+ this.internals.setFormValue(this._formValue);
84
+ this.inert = false;
85
+ }
86
+ }
87
+ }
88
+ validate() {
89
+ const wasValid = this.internals.validity.valid;
90
+ const oldMessage = this.internals.validationMessage;
91
+ this.internals.setValidity({});
92
+ if (this._valueDateTime) {
93
+ const minDateIsValid = !this._minDateTime || this._valueDateTime > this._minDateTime || this._valueDateTime.equals(this._minDateTime);
94
+ if (!minDateIsValid) {
95
+ this.internals.setValidity({ rangeUnderflow: true }, `Date should be equal or after ${this._minDateTime?.toLocaleString({
96
+ month: 'short',
97
+ day: 'numeric',
98
+ year: 'numeric',
99
+ })}`);
100
+ return;
101
+ }
102
+ const maxDateIsValid = !this._maxDateTime || this._valueDateTime.equals(this._maxDateTime) || this._valueDateTime < this._maxDateTime;
103
+ if (!maxDateIsValid) {
104
+ this.internals.setValidity({ rangeOverflow: true }, `Date should be before or equal to ${this._maxDateTime?.toLocaleString({
105
+ month: 'short',
106
+ day: 'numeric',
107
+ year: 'numeric',
108
+ })}`);
109
+ return;
110
+ }
111
+ }
112
+ if (wasValid !== this.internals.validity.valid || oldMessage !== this.internals.validationMessage) {
113
+ this.requestUpdate();
114
+ }
115
+ }
116
+ connectedCallback() {
117
+ super.connectedCallback();
118
+ this.addEventListener('date-selection', this.onDateChange);
119
+ this.addEventListener('dialog-open', this._onDialogOpen);
120
+ this.addEventListener('dialog-close', this._onDialogClose);
121
+ this.addEventListener('focusout', this.onFocusOut);
122
+ this.addEventListener('focus', this._onFocus);
123
+ }
124
+ disconnectedCallback() {
125
+ super.disconnectedCallback();
126
+ this.removeEventListener('date-selection', this.onDateChange);
127
+ this.removeEventListener('dialog-open', this._onDialogOpen);
128
+ this.removeEventListener('dialog-close', this._onDialogClose);
129
+ this.removeEventListener('focusout', this.onFocusOut);
130
+ this.removeEventListener('focus', this._onFocus);
131
+ }
132
+ showCalendar() {
133
+ if (!this._open) {
134
+ this.dialog.show();
135
+ if (this.openLeft) {
136
+ this.style.setProperty('--calendar-left-distance', '-192px');
137
+ }
138
+ }
139
+ else {
140
+ this.dialog.hide();
141
+ }
142
+ }
143
+ render() {
144
+ return (0, lit_1.html) `
145
+ <div part="controls" aria-invalid=${!this.internals.validity.valid} aria-disabled="${this.disabled}">
146
+ <div name="container" part="container">
147
+ <tt-date-input @change=${this.onDateChange} .value=${this.value} .disabled=${this.disabled}></tt-date-input>
148
+ </div>
149
+ <button
150
+ name="Show calendar"
151
+ @click=${this.showCalendar}
152
+ class="show-calendar"
153
+ aria-label="Choose date, ${this}"
154
+ ?disabled=${this.disabled}
155
+ >
156
+ ${(0, unsafe_svg_js_1.unsafeSVG)(icons_1.calendarIcon)}
157
+ </button>
158
+ </div>
159
+ <tt-dialog name="calendar" aria-label="Choose date" part="dialog">
160
+ <tt-calendar
161
+ .value="${this.value}"
162
+ .minDate=${this.minDate}
163
+ .maxDate=${this.maxDate}
164
+ part="calendar"
165
+ ></tt-calendar>
166
+ </tt-dialog>
167
+ <div class="errormessage" id="error-msg-${this.id}" role="alert" aria-atomic="true" part="error">
168
+ ${this.internals.validity.valid ? lit_1.nothing : this.internals.validationMessage}
169
+ </div>
170
+ `;
171
+ }
172
+ }
173
+ exports.TtDatePicker = TtDatePicker;
174
+ TtDatePicker.styles = [
175
+ styles_js_1.styles,
176
+ (0, lit_1.css) `
177
+ [part='controls'] {
178
+ min-width: 19.1ch;
179
+ }
180
+ `,
181
+ ];
182
+ TtDatePicker.formAssociated = true;
183
+ TtDatePicker.shadowRootOptions = {
184
+ ...lit_1.LitElement.shadowRootOptions,
185
+ delegatesFocus: true,
186
+ };
187
+ __decorate([
188
+ (0, decorators_js_1.query)('tt-calendar')
189
+ ], TtDatePicker.prototype, "calendar", void 0);
190
+ __decorate([
191
+ (0, decorators_js_1.query)('tt-dialog')
192
+ ], TtDatePicker.prototype, "dialog", void 0);
193
+ __decorate([
194
+ (0, decorators_js_1.property)({ type: String })
195
+ ], TtDatePicker.prototype, "value", void 0);
196
+ __decorate([
197
+ (0, decorators_js_1.property)({
198
+ type: Date,
199
+ converter: helpers_js_1.dateConverter,
200
+ })
201
+ ], TtDatePicker.prototype, "minDate", void 0);
202
+ __decorate([
203
+ (0, decorators_js_1.property)({
204
+ type: Date,
205
+ converter: helpers_js_1.dateConverter,
206
+ })
207
+ ], TtDatePicker.prototype, "maxDate", void 0);
208
+ __decorate([
209
+ (0, decorators_js_1.property)({ type: Boolean, attribute: 'open-left' })
210
+ ], TtDatePicker.prototype, "openLeft", void 0);
211
+ __decorate([
212
+ (0, decorators_js_1.property)({ type: Boolean })
213
+ ], TtDatePicker.prototype, "disabled", void 0);
214
+ __decorate([
215
+ (0, decorators_js_1.state)()
216
+ ], TtDatePicker.prototype, "_open", void 0);
217
+ __decorate([
218
+ (0, decorators_js_1.query)('tt-date-input')
219
+ ], TtDatePicker.prototype, "dateInput", void 0);
220
+ //# sourceMappingURL=TtDatePicker.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"TtDatePicker.js","sourceRoot":"","sources":["../../src/TtDatePicker.ts"],"names":[],"mappings":"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","sourcesContent":["import { css, html, LitElement, nothing, PropertyValues } from 'lit';\nimport { property, query, state } from 'lit/decorators.js';\nimport { unsafeSVG } from 'lit/directives/unsafe-svg.js';\nimport { DateTime } from 'luxon';\nimport { Calendar } from '@triptease/tt-calendar';\nimport { calendarIcon } from '@triptease/icons';\nimport { TtDialog } from '@triptease/tt-dialog';\nimport { DateInput } from '@triptease/tt-date-input';\nimport { DateSelectionEvent } from '@triptease/tt-calendar/date-selection-context.js';\nimport { dateConverter } from './helpers.js';\nimport { styles } from './styles.js';\n\nexport class TtDatePicker extends LitElement {\n static styles = [\n styles,\n css`\n [part='controls'] {\n min-width: 19.1ch;\n }\n `,\n ];\n\n static formAssociated = true;\n\n static shadowRootOptions = {\n ...LitElement.shadowRootOptions,\n delegatesFocus: true,\n };\n\n @query('tt-calendar')\n calendar!: Calendar;\n\n @query('tt-dialog')\n dialog!: TtDialog;\n\n @property({ type: String })\n public value?: string;\n\n @property({\n type: Date,\n converter: dateConverter,\n })\n public minDate?: Date;\n\n @property({\n type: Date,\n converter: dateConverter,\n })\n public maxDate?: Date;\n\n @property({ type: Boolean, attribute: 'open-left' })\n public openLeft = false;\n\n @property({ type: Boolean })\n public disabled = false;\n\n // Internal DateTime properties for calculations\n private get _valueDateTime(): DateTime | undefined {\n return this.value ? DateTime.fromISO(this.value) : undefined;\n }\n\n private get _minDateTime(): DateTime | undefined {\n return this.minDate ? DateTime.fromJSDate(this.minDate) : undefined;\n }\n\n private get _maxDateTime(): DateTime | undefined {\n return this.maxDate ? DateTime.fromJSDate(this.maxDate) : undefined;\n }\n\n @state()\n private _open: boolean = false;\n\n @query('tt-date-input')\n private dateInput!: DateInput;\n\n public internals: ReturnType<typeof this.attachInternals>;\n\n private get _formValue(): string {\n return this.value ?? '';\n }\n\n constructor() {\n super();\n this.internals = this.attachInternals();\n }\n\n onDateChange(e: Event) {\n if (DateSelectionEvent.is(e)) {\n this.value = e.detail;\n } else if (e.type === 'change') {\n this.value = (e.target as HTMLInputElement).value;\n }\n\n this.dispatchEvent(new InputEvent('change'));\n this.dialog.hide();\n }\n\n get form(): HTMLFormElement | null {\n return this.internals.form;\n }\n\n public updated(changedProperties: PropertyValues) {\n super.updated(changedProperties);\n\n if (changedProperties.has('value')) {\n this.internals.setFormValue(this._formValue);\n this.validate();\n }\n\n if (changedProperties.has('disabled')) {\n if (this.disabled) {\n this.internals.setFormValue(null);\n this.internals.states.add('disabled');\n this.inert = true;\n } else {\n this.internals.states.delete('disabled');\n this.internals.setFormValue(this._formValue);\n this.inert = false;\n }\n }\n }\n\n private validate() {\n const wasValid = this.internals.validity.valid;\n const oldMessage = this.internals.validationMessage;\n this.internals.setValidity({});\n\n if (this._valueDateTime) {\n const minDateIsValid =\n !this._minDateTime || this._valueDateTime > this._minDateTime || this._valueDateTime.equals(this._minDateTime);\n if (!minDateIsValid) {\n this.internals.setValidity(\n { rangeUnderflow: true },\n `Date should be equal or after ${this._minDateTime?.toLocaleString({\n month: 'short',\n day: 'numeric',\n year: 'numeric',\n })}`\n );\n return;\n }\n const maxDateIsValid =\n !this._maxDateTime || this._valueDateTime.equals(this._maxDateTime) || this._valueDateTime < this._maxDateTime;\n if (!maxDateIsValid) {\n this.internals.setValidity(\n { rangeOverflow: true },\n `Date should be before or equal to ${this._maxDateTime?.toLocaleString({\n month: 'short',\n day: 'numeric',\n year: 'numeric',\n })}`\n );\n return;\n }\n }\n if (wasValid !== this.internals.validity.valid || oldMessage !== this.internals.validationMessage) {\n this.requestUpdate();\n }\n }\n\n public connectedCallback() {\n super.connectedCallback();\n\n this.addEventListener('date-selection', this.onDateChange);\n this.addEventListener('dialog-open', this._onDialogOpen);\n this.addEventListener('dialog-close', this._onDialogClose);\n this.addEventListener('focusout', this.onFocusOut);\n this.addEventListener('focus', this._onFocus);\n }\n\n private _onDialogOpen = () => {\n this._open = true;\n };\n\n private _onDialogClose = () => {\n this._open = false;\n };\n\n public onFocusOut = (event: FocusEvent) => {\n if (!this._open) return;\n\n const target = (event.relatedTarget ?? event.explicitOriginalTarget) as HTMLElement | null;\n\n if (!this.contains(target)) {\n this.dialog.hide();\n }\n };\n\n private _onFocus = () => {\n this.dateInput.focus();\n };\n\n public disconnectedCallback() {\n super.disconnectedCallback();\n this.removeEventListener('date-selection', this.onDateChange);\n this.removeEventListener('dialog-open', this._onDialogOpen);\n this.removeEventListener('dialog-close', this._onDialogClose);\n this.removeEventListener('focusout', this.onFocusOut);\n this.removeEventListener('focus', this._onFocus);\n }\n\n showCalendar() {\n if (!this._open) {\n this.dialog.show();\n\n if (this.openLeft) {\n this.style.setProperty('--calendar-left-distance', '-192px');\n }\n } else {\n this.dialog.hide();\n }\n }\n\n render() {\n return html`\n <div part=\"controls\" aria-invalid=${!this.internals.validity.valid} aria-disabled=\"${this.disabled}\">\n <div name=\"container\" part=\"container\">\n <tt-date-input @change=${this.onDateChange} .value=${this.value} .disabled=${this.disabled}></tt-date-input>\n </div>\n <button\n name=\"Show calendar\"\n @click=${this.showCalendar}\n class=\"show-calendar\"\n aria-label=\"Choose date, ${this}\"\n ?disabled=${this.disabled}\n >\n ${unsafeSVG(calendarIcon)}\n </button>\n </div>\n <tt-dialog name=\"calendar\" aria-label=\"Choose date\" part=\"dialog\">\n <tt-calendar\n .value=\"${this.value}\"\n .minDate=${this.minDate}\n .maxDate=${this.maxDate}\n part=\"calendar\"\n ></tt-calendar>\n </tt-dialog>\n <div class=\"errormessage\" id=\"error-msg-${this.id}\" role=\"alert\" aria-atomic=\"true\" part=\"error\">\n ${this.internals.validity.valid ? nothing : this.internals.validationMessage}\n </div>\n `;\n }\n}\n\ndeclare global {\n interface HTMLElementTagNameMap {\n 'tt-date-picker': TtDatePicker;\n }\n}\n"]}
1
+ {"version":3,"file":"TtDatePicker.js","sourceRoot":"","sources":["../../../src/TtDatePicker.ts"],"names":[],"mappings":"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","sourcesContent":["import { css, html, LitElement, nothing, PropertyValues } from 'lit';\nimport { property, query, state } from 'lit/decorators.js';\nimport { unsafeSVG } from 'lit/directives/unsafe-svg.js';\nimport { DateTime } from 'luxon';\nimport { Calendar } from '@triptease/tt-calendar';\nimport { calendarIcon } from '@triptease/icons';\nimport { TtDialog } from '@triptease/tt-dialog';\nimport { DateInput } from '@triptease/tt-date-input';\nimport { DateSelectionEvent } from '@triptease/tt-calendar/date-selection-context.js';\nimport { dateConverter } from './helpers.js';\nimport { styles } from './styles.js';\n\nexport class TtDatePicker extends LitElement {\n static styles = [\n styles,\n css`\n [part='controls'] {\n min-width: 19.1ch;\n }\n `,\n ];\n\n static formAssociated = true;\n\n static shadowRootOptions = {\n ...LitElement.shadowRootOptions,\n delegatesFocus: true,\n };\n\n @query('tt-calendar')\n calendar!: Calendar;\n\n @query('tt-dialog')\n dialog!: TtDialog;\n\n @property({ type: String })\n public value?: string;\n\n @property({\n type: Date,\n converter: dateConverter,\n })\n public minDate?: Date;\n\n @property({\n type: Date,\n converter: dateConverter,\n })\n public maxDate?: Date;\n\n @property({ type: Boolean, attribute: 'open-left' })\n public openLeft = false;\n\n @property({ type: Boolean })\n public disabled = false;\n\n // Internal DateTime properties for calculations\n private get _valueDateTime(): DateTime | undefined {\n return this.value ? DateTime.fromISO(this.value) : undefined;\n }\n\n private get _minDateTime(): DateTime | undefined {\n return this.minDate ? DateTime.fromJSDate(this.minDate) : undefined;\n }\n\n private get _maxDateTime(): DateTime | undefined {\n return this.maxDate ? DateTime.fromJSDate(this.maxDate) : undefined;\n }\n\n @state()\n private _open: boolean = false;\n\n @query('tt-date-input')\n private dateInput!: DateInput;\n\n public internals: ReturnType<typeof this.attachInternals>;\n\n private get _formValue(): string {\n return this.value ?? '';\n }\n\n constructor() {\n super();\n this.internals = this.attachInternals();\n }\n\n onDateChange(e: Event) {\n if (DateSelectionEvent.is(e)) {\n this.value = e.detail;\n } else if (e.type === 'change') {\n this.value = (e.target as HTMLInputElement).value;\n }\n\n this.dispatchEvent(new InputEvent('change'));\n this.dialog.hide();\n }\n\n get form(): HTMLFormElement | null {\n return this.internals.form;\n }\n\n public updated(changedProperties: PropertyValues) {\n super.updated(changedProperties);\n\n if (changedProperties.has('value')) {\n this.internals.setFormValue(this._formValue);\n this.validate();\n }\n\n if (changedProperties.has('disabled')) {\n if (this.disabled) {\n this.internals.setFormValue(null);\n this.internals.states.add('disabled');\n this.inert = true;\n } else {\n this.internals.states.delete('disabled');\n this.internals.setFormValue(this._formValue);\n this.inert = false;\n }\n }\n }\n\n private validate() {\n const wasValid = this.internals.validity.valid;\n const oldMessage = this.internals.validationMessage;\n this.internals.setValidity({});\n\n if (this._valueDateTime) {\n const minDateIsValid =\n !this._minDateTime || this._valueDateTime > this._minDateTime || this._valueDateTime.equals(this._minDateTime);\n if (!minDateIsValid) {\n this.internals.setValidity(\n { rangeUnderflow: true },\n `Date should be equal or after ${this._minDateTime?.toLocaleString({\n month: 'short',\n day: 'numeric',\n year: 'numeric',\n })}`\n );\n return;\n }\n const maxDateIsValid =\n !this._maxDateTime || this._valueDateTime.equals(this._maxDateTime) || this._valueDateTime < this._maxDateTime;\n if (!maxDateIsValid) {\n this.internals.setValidity(\n { rangeOverflow: true },\n `Date should be before or equal to ${this._maxDateTime?.toLocaleString({\n month: 'short',\n day: 'numeric',\n year: 'numeric',\n })}`\n );\n return;\n }\n }\n if (wasValid !== this.internals.validity.valid || oldMessage !== this.internals.validationMessage) {\n this.requestUpdate();\n }\n }\n\n public connectedCallback() {\n super.connectedCallback();\n\n this.addEventListener('date-selection', this.onDateChange);\n this.addEventListener('dialog-open', this._onDialogOpen);\n this.addEventListener('dialog-close', this._onDialogClose);\n this.addEventListener('focusout', this.onFocusOut);\n this.addEventListener('focus', this._onFocus);\n }\n\n private _onDialogOpen = () => {\n this._open = true;\n };\n\n private _onDialogClose = () => {\n this._open = false;\n };\n\n public onFocusOut = (event: FocusEvent) => {\n if (!this._open) return;\n\n const target = (event.relatedTarget ?? event.explicitOriginalTarget) as HTMLElement | null;\n\n if (!this.contains(target)) {\n this.dialog.hide();\n }\n };\n\n private _onFocus = () => {\n this.dateInput.focus();\n };\n\n public disconnectedCallback() {\n super.disconnectedCallback();\n this.removeEventListener('date-selection', this.onDateChange);\n this.removeEventListener('dialog-open', this._onDialogOpen);\n this.removeEventListener('dialog-close', this._onDialogClose);\n this.removeEventListener('focusout', this.onFocusOut);\n this.removeEventListener('focus', this._onFocus);\n }\n\n showCalendar() {\n if (!this._open) {\n this.dialog.show();\n\n if (this.openLeft) {\n this.style.setProperty('--calendar-left-distance', '-192px');\n }\n } else {\n this.dialog.hide();\n }\n }\n\n render() {\n return html`\n <div part=\"controls\" aria-invalid=${!this.internals.validity.valid} aria-disabled=\"${this.disabled}\">\n <div name=\"container\" part=\"container\">\n <tt-date-input @change=${this.onDateChange} .value=${this.value} .disabled=${this.disabled}></tt-date-input>\n </div>\n <button\n name=\"Show calendar\"\n @click=${this.showCalendar}\n class=\"show-calendar\"\n aria-label=\"Choose date, ${this}\"\n ?disabled=${this.disabled}\n >\n ${unsafeSVG(calendarIcon)}\n </button>\n </div>\n <tt-dialog name=\"calendar\" aria-label=\"Choose date\" part=\"dialog\">\n <tt-calendar\n .value=\"${this.value}\"\n .minDate=${this.minDate}\n .maxDate=${this.maxDate}\n part=\"calendar\"\n ></tt-calendar>\n </tt-dialog>\n <div class=\"errormessage\" id=\"error-msg-${this.id}\" role=\"alert\" aria-atomic=\"true\" part=\"error\">\n ${this.internals.validity.valid ? nothing : this.internals.validationMessage}\n </div>\n `;\n }\n}\n\ndeclare global {\n interface HTMLElementTagNameMap {\n 'tt-date-picker': TtDatePicker;\n }\n}\n"]}
@@ -0,0 +1,9 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.dateTimeConverter = exports.dateConverter = void 0;
4
+ const luxon_1 = require("luxon");
5
+ const dateConverter = (value) => value ? luxon_1.DateTime.fromISO(value).toJSDate() : undefined;
6
+ exports.dateConverter = dateConverter;
7
+ const dateTimeConverter = (value) => value ? luxon_1.DateTime.fromISO(value) : undefined;
8
+ exports.dateTimeConverter = dateTimeConverter;
9
+ //# sourceMappingURL=helpers.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"helpers.js","sourceRoot":"","sources":["../../../src/helpers.ts"],"names":[],"mappings":";;;AAAA,iCAAiC;AAE1B,MAAM,aAAa,GAAG,CAAC,KAAoB,EAAoB,EAAE,CACtE,KAAK,CAAC,CAAC,CAAC,gBAAQ,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,QAAQ,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC;AAD5C,QAAA,aAAa,iBAC+B;AAElD,MAAM,iBAAiB,GAAG,CAAC,KAAoB,EAAwB,EAAE,CAC9E,KAAK,CAAC,CAAC,CAAC,gBAAQ,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;AADjC,QAAA,iBAAiB,qBACgB","sourcesContent":["import { DateTime } from 'luxon';\n\nexport const dateConverter = (value: string | null): Date | undefined =>\n value ? DateTime.fromISO(value).toJSDate() : undefined;\n\nexport const dateTimeConverter = (value: string | null): DateTime | undefined =>\n value ? DateTime.fromISO(value) : undefined;\n"]}
@@ -0,0 +1,21 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __exportStar = (this && this.__exportStar) || function(m, exports) {
14
+ for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
15
+ };
16
+ Object.defineProperty(exports, "__esModule", { value: true });
17
+ exports.TtDatePicker = void 0;
18
+ __exportStar(require("./types.js"), exports);
19
+ var tt_date_picker_js_1 = require("./tt-date-picker.js");
20
+ Object.defineProperty(exports, "TtDatePicker", { enumerable: true, get: function () { return tt_date_picker_js_1.TtDatePicker; } });
21
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/index.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;AAAA,6CAA2B;AAC3B,yDAAmD;AAA1C,iHAAA,YAAY,OAAA","sourcesContent":["export * from './types.js';\nexport { TtDatePicker } from './tt-date-picker.js';\n"]}
@@ -0,0 +1,70 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.styles = void 0;
4
+ const lit_1 = require("lit");
5
+ exports.styles = (0, lit_1.css) `
6
+ :host {
7
+ position: relative;
8
+ }
9
+
10
+ :host(:state(disabled)) {
11
+ pointer-events: none;
12
+ opacity: 0.5;
13
+ }
14
+
15
+ tt-side-panel::part(side-panel) {
16
+ border-right: 1px solid var(--color-border-200);
17
+ }
18
+
19
+ tt-side-panel:empty {
20
+ display: none;
21
+ }
22
+
23
+ [part='container'] {
24
+ padding: var(--space-scale-1) var(--space-scale-1-5);
25
+ gap: var(--space-scale-1);
26
+ }
27
+
28
+ [part='controls'] {
29
+ display: flex;
30
+ border: 1px solid var(--color-border-200);
31
+ border-radius: var(--border-radius);
32
+ width: fit-content;
33
+ height: var(--date-picker-height, fit-content);
34
+ background-color: var(--color-surface-100);
35
+ font-size: var(--font-size-200);
36
+ color: var(--color-text-400);
37
+ align-items: stretch;
38
+ justify-content: space-around;
39
+ padding: 0;
40
+
41
+ &[aria-invalid='true'] {
42
+ outline: 2px solid var(--color-alert-strong);
43
+ }
44
+
45
+ button {
46
+ border: 0;
47
+ padding-inline: var(--space-scale-1);
48
+ background-color: transparent;
49
+ cursor: pointer;
50
+ display: flex;
51
+ align-items: center;
52
+ justify-content: center;
53
+ align-self: stretch;
54
+ border-left: 1px solid var(--color-border-200);
55
+ box-sizing: border-box;
56
+ color: var(--color-text-400);
57
+ }
58
+
59
+ &:has(+ tt-dialog[open]) .show-calendar svg,
60
+ .show-calendar:hover svg {
61
+ color: var(--color-primary-400);
62
+ }
63
+ }
64
+
65
+ [part='dialog']::part(dialog) {
66
+ margin-top: var(--space-scale-1);
67
+ flex-direction: row-reverse;
68
+ }
69
+ `;
70
+ //# sourceMappingURL=styles.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"styles.js","sourceRoot":"","sources":["../../../src/styles.ts"],"names":[],"mappings":";;;AAAA,6BAA0B;AAEb,QAAA,MAAM,GAAG,IAAA,SAAG,EAAA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;CAgExB,CAAC","sourcesContent":["import { css } from 'lit';\n\nexport const styles = css`\n :host {\n position: relative;\n }\n\n :host(:state(disabled)) {\n pointer-events: none;\n opacity: 0.5;\n }\n\n tt-side-panel::part(side-panel) {\n border-right: 1px solid var(--color-border-200);\n }\n\n tt-side-panel:empty {\n display: none;\n }\n\n [part='container'] {\n padding: var(--space-scale-1) var(--space-scale-1-5);\n gap: var(--space-scale-1);\n }\n\n [part='controls'] {\n display: flex;\n border: 1px solid var(--color-border-200);\n border-radius: var(--border-radius);\n width: fit-content;\n height: var(--date-picker-height, fit-content);\n background-color: var(--color-surface-100);\n font-size: var(--font-size-200);\n color: var(--color-text-400);\n align-items: stretch;\n justify-content: space-around;\n padding: 0;\n\n &[aria-invalid='true'] {\n outline: 2px solid var(--color-alert-strong);\n }\n\n button {\n border: 0;\n padding-inline: var(--space-scale-1);\n background-color: transparent;\n cursor: pointer;\n display: flex;\n align-items: center;\n justify-content: center;\n align-self: stretch;\n border-left: 1px solid var(--color-border-200);\n box-sizing: border-box;\n color: var(--color-text-400);\n }\n\n &:has(+ tt-dialog[open]) .show-calendar svg,\n .show-calendar:hover svg {\n color: var(--color-primary-400);\n }\n }\n\n [part='dialog']::part(dialog) {\n margin-top: var(--space-scale-1);\n flex-direction: row-reverse;\n }\n`;\n"]}
@@ -0,0 +1,23 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.TtDatePicker = void 0;
4
+ const tt_calendar_js_1 = require("@triptease/tt-calendar/tt-calendar.js");
5
+ const tt_date_input_js_1 = require("@triptease/tt-date-input/tt-date-input.js");
6
+ const tt_dialog_js_1 = require("@triptease/tt-dialog/tt-dialog.js");
7
+ const TtDatePicker_js_1 = require("./TtDatePicker.js");
8
+ Object.defineProperty(exports, "TtDatePicker", { enumerable: true, get: function () { return TtDatePicker_js_1.TtDatePicker; } });
9
+ if (typeof window !== 'undefined') {
10
+ if (!window.customElements.get('tt-calendar')) {
11
+ window.customElements.define('tt-calendar', tt_calendar_js_1.Calendar);
12
+ }
13
+ if (!window.customElements.get('tt-date-input')) {
14
+ window.customElements.define('tt-date-input', tt_date_input_js_1.DateInput);
15
+ }
16
+ if (!window.customElements.get('tt-dialog')) {
17
+ window.customElements.define('tt-dialog', tt_dialog_js_1.TtDialog);
18
+ }
19
+ if (!window.customElements.get('tt-date-picker')) {
20
+ window.customElements.define('tt-date-picker', TtDatePicker_js_1.TtDatePicker);
21
+ }
22
+ }
23
+ //# sourceMappingURL=tt-date-picker.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tt-date-picker.js","sourceRoot":"","sources":["../../../src/tt-date-picker.ts"],"names":[],"mappings":";;;AAAA,0EAAiE;AACjE,gFAAsE;AACtE,oEAA6D;AAC7D,uDAAiD;AAmBxC,6FAnBA,8BAAY,OAmBA;AAjBrB,IAAI,OAAO,MAAM,KAAK,WAAW,EAAE,CAAC;IAClC,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,GAAG,CAAC,aAAa,CAAC,EAAE,CAAC;QAC9C,MAAM,CAAC,cAAc,CAAC,MAAM,CAAC,aAAa,EAAE,yBAAQ,CAAC,CAAC;IACxD,CAAC;IACD,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,GAAG,CAAC,eAAe,CAAC,EAAE,CAAC;QAChD,MAAM,CAAC,cAAc,CAAC,MAAM,CAAC,eAAe,EAAE,4BAAS,CAAC,CAAC;IAC3D,CAAC;IAED,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,GAAG,CAAC,WAAW,CAAC,EAAE,CAAC;QAC5C,MAAM,CAAC,cAAc,CAAC,MAAM,CAAC,WAAW,EAAE,uBAAQ,CAAC,CAAC;IACtD,CAAC;IAED,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,GAAG,CAAC,gBAAgB,CAAC,EAAE,CAAC;QACjD,MAAM,CAAC,cAAc,CAAC,MAAM,CAAC,gBAAgB,EAAE,8BAAY,CAAC,CAAC;IAC/D,CAAC;AACH,CAAC","sourcesContent":["import { Calendar } from '@triptease/tt-calendar/tt-calendar.js';\nimport { DateInput } from '@triptease/tt-date-input/tt-date-input.js';\nimport { TtDialog } from '@triptease/tt-dialog/tt-dialog.js';\nimport { TtDatePicker } from './TtDatePicker.js';\n\nif (typeof window !== 'undefined') {\n if (!window.customElements.get('tt-calendar')) {\n window.customElements.define('tt-calendar', Calendar);\n }\n if (!window.customElements.get('tt-date-input')) {\n window.customElements.define('tt-date-input', DateInput);\n }\n\n if (!window.customElements.get('tt-dialog')) {\n window.customElements.define('tt-dialog', TtDialog);\n }\n\n if (!window.customElements.get('tt-date-picker')) {\n window.customElements.define('tt-date-picker', TtDatePicker);\n }\n}\n\nexport { TtDatePicker };\n"]}
@@ -0,0 +1,3 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ //# sourceMappingURL=types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.js","sourceRoot":"","sources":["../../../src/types.ts"],"names":[],"mappings":"","sourcesContent":["import { TtDatePicker } from './TtDatePicker.js';\n\ninterface TtDatePickerExternalAttributes {\n id?: string;\n value?: string;\n disabled?: boolean;\n 'open-left'?: boolean;\n name?: string;\n minDate?: string;\n maxDate?: string;\n}\n\ndeclare global {\n interface HTMLElementTagNameMap {\n 'tt-date-picker': TtDatePicker;\n }\n\n namespace JSX {\n interface IntrinsicElements {\n 'tt-date-picker': TtDatePickerExternalAttributes & { style?: string };\n }\n }\n\n namespace React {\n namespace JSX {\n interface IntrinsicElements {\n 'tt-date-picker': TtDatePickerExternalAttributes & {\n ref?: React.Ref<unknown>;\n style?: React.CSSProperties;\n };\n }\n }\n }\n}\n"]}
@@ -0,0 +1 @@
1
+ {"version":3,"file":"TtDatePicker.js","sourceRoot":"","sources":["../../../src/TtDatePicker.ts"],"names":[],"mappings":"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","sourcesContent":["import { css, html, LitElement, nothing, PropertyValues } from 'lit';\nimport { property, query, state } from 'lit/decorators.js';\nimport { unsafeSVG } from 'lit/directives/unsafe-svg.js';\nimport { DateTime } from 'luxon';\nimport { Calendar } from '@triptease/tt-calendar';\nimport { calendarIcon } from '@triptease/icons';\nimport { TtDialog } from '@triptease/tt-dialog';\nimport { DateInput } from '@triptease/tt-date-input';\nimport { DateSelectionEvent } from '@triptease/tt-calendar/date-selection-context.js';\nimport { dateConverter } from './helpers.js';\nimport { styles } from './styles.js';\n\nexport class TtDatePicker extends LitElement {\n static styles = [\n styles,\n css`\n [part='controls'] {\n min-width: 19.1ch;\n }\n `,\n ];\n\n static formAssociated = true;\n\n static shadowRootOptions = {\n ...LitElement.shadowRootOptions,\n delegatesFocus: true,\n };\n\n @query('tt-calendar')\n calendar!: Calendar;\n\n @query('tt-dialog')\n dialog!: TtDialog;\n\n @property({ type: String })\n public value?: string;\n\n @property({\n type: Date,\n converter: dateConverter,\n })\n public minDate?: Date;\n\n @property({\n type: Date,\n converter: dateConverter,\n })\n public maxDate?: Date;\n\n @property({ type: Boolean, attribute: 'open-left' })\n public openLeft = false;\n\n @property({ type: Boolean })\n public disabled = false;\n\n // Internal DateTime properties for calculations\n private get _valueDateTime(): DateTime | undefined {\n return this.value ? DateTime.fromISO(this.value) : undefined;\n }\n\n private get _minDateTime(): DateTime | undefined {\n return this.minDate ? DateTime.fromJSDate(this.minDate) : undefined;\n }\n\n private get _maxDateTime(): DateTime | undefined {\n return this.maxDate ? DateTime.fromJSDate(this.maxDate) : undefined;\n }\n\n @state()\n private _open: boolean = false;\n\n @query('tt-date-input')\n private dateInput!: DateInput;\n\n public internals: ReturnType<typeof this.attachInternals>;\n\n private get _formValue(): string {\n return this.value ?? '';\n }\n\n constructor() {\n super();\n this.internals = this.attachInternals();\n }\n\n onDateChange(e: Event) {\n if (DateSelectionEvent.is(e)) {\n this.value = e.detail;\n } else if (e.type === 'change') {\n this.value = (e.target as HTMLInputElement).value;\n }\n\n this.dispatchEvent(new InputEvent('change'));\n this.dialog.hide();\n }\n\n get form(): HTMLFormElement | null {\n return this.internals.form;\n }\n\n public updated(changedProperties: PropertyValues) {\n super.updated(changedProperties);\n\n if (changedProperties.has('value')) {\n this.internals.setFormValue(this._formValue);\n this.validate();\n }\n\n if (changedProperties.has('disabled')) {\n if (this.disabled) {\n this.internals.setFormValue(null);\n this.internals.states.add('disabled');\n this.inert = true;\n } else {\n this.internals.states.delete('disabled');\n this.internals.setFormValue(this._formValue);\n this.inert = false;\n }\n }\n }\n\n private validate() {\n const wasValid = this.internals.validity.valid;\n const oldMessage = this.internals.validationMessage;\n this.internals.setValidity({});\n\n if (this._valueDateTime) {\n const minDateIsValid =\n !this._minDateTime || this._valueDateTime > this._minDateTime || this._valueDateTime.equals(this._minDateTime);\n if (!minDateIsValid) {\n this.internals.setValidity(\n { rangeUnderflow: true },\n `Date should be equal or after ${this._minDateTime?.toLocaleString({\n month: 'short',\n day: 'numeric',\n year: 'numeric',\n })}`\n );\n return;\n }\n const maxDateIsValid =\n !this._maxDateTime || this._valueDateTime.equals(this._maxDateTime) || this._valueDateTime < this._maxDateTime;\n if (!maxDateIsValid) {\n this.internals.setValidity(\n { rangeOverflow: true },\n `Date should be before or equal to ${this._maxDateTime?.toLocaleString({\n month: 'short',\n day: 'numeric',\n year: 'numeric',\n })}`\n );\n return;\n }\n }\n if (wasValid !== this.internals.validity.valid || oldMessage !== this.internals.validationMessage) {\n this.requestUpdate();\n }\n }\n\n public connectedCallback() {\n super.connectedCallback();\n\n this.addEventListener('date-selection', this.onDateChange);\n this.addEventListener('dialog-open', this._onDialogOpen);\n this.addEventListener('dialog-close', this._onDialogClose);\n this.addEventListener('focusout', this.onFocusOut);\n this.addEventListener('focus', this._onFocus);\n }\n\n private _onDialogOpen = () => {\n this._open = true;\n };\n\n private _onDialogClose = () => {\n this._open = false;\n };\n\n public onFocusOut = (event: FocusEvent) => {\n if (!this._open) return;\n\n const target = (event.relatedTarget ?? event.explicitOriginalTarget) as HTMLElement | null;\n\n if (!this.contains(target)) {\n this.dialog.hide();\n }\n };\n\n private _onFocus = () => {\n this.dateInput.focus();\n };\n\n public disconnectedCallback() {\n super.disconnectedCallback();\n this.removeEventListener('date-selection', this.onDateChange);\n this.removeEventListener('dialog-open', this._onDialogOpen);\n this.removeEventListener('dialog-close', this._onDialogClose);\n this.removeEventListener('focusout', this.onFocusOut);\n this.removeEventListener('focus', this._onFocus);\n }\n\n showCalendar() {\n if (!this._open) {\n this.dialog.show();\n\n if (this.openLeft) {\n this.style.setProperty('--calendar-left-distance', '-192px');\n }\n } else {\n this.dialog.hide();\n }\n }\n\n render() {\n return html`\n <div part=\"controls\" aria-invalid=${!this.internals.validity.valid} aria-disabled=\"${this.disabled}\">\n <div name=\"container\" part=\"container\">\n <tt-date-input @change=${this.onDateChange} .value=${this.value} .disabled=${this.disabled}></tt-date-input>\n </div>\n <button\n name=\"Show calendar\"\n @click=${this.showCalendar}\n class=\"show-calendar\"\n aria-label=\"Choose date, ${this}\"\n ?disabled=${this.disabled}\n >\n ${unsafeSVG(calendarIcon)}\n </button>\n </div>\n <tt-dialog name=\"calendar\" aria-label=\"Choose date\" part=\"dialog\">\n <tt-calendar\n .value=\"${this.value}\"\n .minDate=${this.minDate}\n .maxDate=${this.maxDate}\n part=\"calendar\"\n ></tt-calendar>\n </tt-dialog>\n <div class=\"errormessage\" id=\"error-msg-${this.id}\" role=\"alert\" aria-atomic=\"true\" part=\"error\">\n ${this.internals.validity.valid ? nothing : this.internals.validationMessage}\n </div>\n `;\n }\n}\n\ndeclare global {\n interface HTMLElementTagNameMap {\n 'tt-date-picker': TtDatePicker;\n }\n}\n"]}
@@ -0,0 +1 @@
1
+ {"version":3,"file":"helpers.js","sourceRoot":"","sources":["../../../src/helpers.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,OAAO,CAAC;AAEjC,MAAM,CAAC,MAAM,aAAa,GAAG,CAAC,KAAoB,EAAoB,EAAE,CACtE,KAAK,CAAC,CAAC,CAAC,QAAQ,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,QAAQ,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC;AAEzD,MAAM,CAAC,MAAM,iBAAiB,GAAG,CAAC,KAAoB,EAAwB,EAAE,CAC9E,KAAK,CAAC,CAAC,CAAC,QAAQ,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC","sourcesContent":["import { DateTime } from 'luxon';\n\nexport const dateConverter = (value: string | null): Date | undefined =>\n value ? DateTime.fromISO(value).toJSDate() : undefined;\n\nexport const dateTimeConverter = (value: string | null): DateTime | undefined =>\n value ? DateTime.fromISO(value) : undefined;\n"]}
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/index.ts"],"names":[],"mappings":"AAAA,cAAc,YAAY,CAAC;AAC3B,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC","sourcesContent":["export * from './types.js';\nexport { TtDatePicker } from './tt-date-picker.js';\n"]}
@@ -36,7 +36,7 @@ export const styles = css `
36
36
  padding: 0;
37
37
 
38
38
  &[aria-invalid='true'] {
39
- outline: 2px solid var(--color-alert-300, red);
39
+ outline: 2px solid var(--color-alert-strong);
40
40
  }
41
41
 
42
42
  button {
@@ -0,0 +1 @@
1
+ {"version":3,"file":"styles.js","sourceRoot":"","sources":["../../../src/styles.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,GAAG,EAAE,MAAM,KAAK,CAAC;AAE1B,MAAM,CAAC,MAAM,MAAM,GAAG,GAAG,CAAA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;CAgExB,CAAC","sourcesContent":["import { css } from 'lit';\n\nexport const styles = css`\n :host {\n position: relative;\n }\n\n :host(:state(disabled)) {\n pointer-events: none;\n opacity: 0.5;\n }\n\n tt-side-panel::part(side-panel) {\n border-right: 1px solid var(--color-border-200);\n }\n\n tt-side-panel:empty {\n display: none;\n }\n\n [part='container'] {\n padding: var(--space-scale-1) var(--space-scale-1-5);\n gap: var(--space-scale-1);\n }\n\n [part='controls'] {\n display: flex;\n border: 1px solid var(--color-border-200);\n border-radius: var(--border-radius);\n width: fit-content;\n height: var(--date-picker-height, fit-content);\n background-color: var(--color-surface-100);\n font-size: var(--font-size-200);\n color: var(--color-text-400);\n align-items: stretch;\n justify-content: space-around;\n padding: 0;\n\n &[aria-invalid='true'] {\n outline: 2px solid var(--color-alert-strong);\n }\n\n button {\n border: 0;\n padding-inline: var(--space-scale-1);\n background-color: transparent;\n cursor: pointer;\n display: flex;\n align-items: center;\n justify-content: center;\n align-self: stretch;\n border-left: 1px solid var(--color-border-200);\n box-sizing: border-box;\n color: var(--color-text-400);\n }\n\n &:has(+ tt-dialog[open]) .show-calendar svg,\n .show-calendar:hover svg {\n color: var(--color-primary-400);\n }\n }\n\n [part='dialog']::part(dialog) {\n margin-top: var(--space-scale-1);\n flex-direction: row-reverse;\n }\n`;\n"]}
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tt-date-picker.js","sourceRoot":"","sources":["../../../src/tt-date-picker.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,uCAAuC,CAAC;AACjE,OAAO,EAAE,SAAS,EAAE,MAAM,2CAA2C,CAAC;AACtE,OAAO,EAAE,QAAQ,EAAE,MAAM,mCAAmC,CAAC;AAC7D,OAAO,EAAE,YAAY,EAAE,MAAM,mBAAmB,CAAC;AAEjD,IAAI,OAAO,MAAM,KAAK,WAAW,EAAE,CAAC;IAClC,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,GAAG,CAAC,aAAa,CAAC,EAAE,CAAC;QAC9C,MAAM,CAAC,cAAc,CAAC,MAAM,CAAC,aAAa,EAAE,QAAQ,CAAC,CAAC;IACxD,CAAC;IACD,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,GAAG,CAAC,eAAe,CAAC,EAAE,CAAC;QAChD,MAAM,CAAC,cAAc,CAAC,MAAM,CAAC,eAAe,EAAE,SAAS,CAAC,CAAC;IAC3D,CAAC;IAED,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,GAAG,CAAC,WAAW,CAAC,EAAE,CAAC;QAC5C,MAAM,CAAC,cAAc,CAAC,MAAM,CAAC,WAAW,EAAE,QAAQ,CAAC,CAAC;IACtD,CAAC;IAED,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,GAAG,CAAC,gBAAgB,CAAC,EAAE,CAAC;QACjD,MAAM,CAAC,cAAc,CAAC,MAAM,CAAC,gBAAgB,EAAE,YAAY,CAAC,CAAC;IAC/D,CAAC;AACH,CAAC;AAED,OAAO,EAAE,YAAY,EAAE,CAAC","sourcesContent":["import { Calendar } from '@triptease/tt-calendar/tt-calendar.js';\nimport { DateInput } from '@triptease/tt-date-input/tt-date-input.js';\nimport { TtDialog } from '@triptease/tt-dialog/tt-dialog.js';\nimport { TtDatePicker } from './TtDatePicker.js';\n\nif (typeof window !== 'undefined') {\n if (!window.customElements.get('tt-calendar')) {\n window.customElements.define('tt-calendar', Calendar);\n }\n if (!window.customElements.get('tt-date-input')) {\n window.customElements.define('tt-date-input', DateInput);\n }\n\n if (!window.customElements.get('tt-dialog')) {\n window.customElements.define('tt-dialog', TtDialog);\n }\n\n if (!window.customElements.get('tt-date-picker')) {\n window.customElements.define('tt-date-picker', TtDatePicker);\n }\n}\n\nexport { TtDatePicker };\n"]}
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.js","sourceRoot":"","sources":["../../../src/types.ts"],"names":[],"mappings":"","sourcesContent":["import { TtDatePicker } from './TtDatePicker.js';\n\ninterface TtDatePickerExternalAttributes {\n id?: string;\n value?: string;\n disabled?: boolean;\n 'open-left'?: boolean;\n name?: string;\n minDate?: string;\n maxDate?: string;\n}\n\ndeclare global {\n interface HTMLElementTagNameMap {\n 'tt-date-picker': TtDatePicker;\n }\n\n namespace JSX {\n interface IntrinsicElements {\n 'tt-date-picker': TtDatePickerExternalAttributes & { style?: string };\n }\n }\n\n namespace React {\n namespace JSX {\n interface IntrinsicElements {\n 'tt-date-picker': TtDatePickerExternalAttributes & {\n ref?: React.Ref<unknown>;\n style?: React.CSSProperties;\n };\n }\n }\n }\n}\n"]}
@@ -0,0 +1 @@
1
+ import '../src/index.js';
@@ -0,0 +1,57 @@
1
+ import { elementUpdated, expect, fixture, oneDefaultPreventedEvent, oneEvent } from '@open-wc/testing';
2
+ import { html } from 'lit';
3
+ import { Settings } from 'luxon';
4
+ import '../src/index.js'; // This handles the registration
5
+ describe('TtDatePicker', () => {
6
+ afterEach(() => {
7
+ Settings.defaultZone = 'system';
8
+ });
9
+ ['Australia/Brisbane', 'America/New_York', 'Europe/London', 'Asia/Tokyo', 'Etc/UTC'].forEach((timezone) => {
10
+ it('should include date value in form submission data', async () => {
11
+ Settings.defaultZone = timezone;
12
+ const form = await fixture(html `
13
+ <form method="post" action="#">
14
+ <input type="text" name="other_field" value="test_value" />
15
+ <tt-date-picker name="booking_date" value="2025-04-16"></tt-date-picker>
16
+ <button type="submit">Submit</button>
17
+ </form>
18
+ `);
19
+ const datePicker = form.querySelector('tt-date-picker');
20
+ expect(datePicker).to.exist;
21
+ await elementUpdated(datePicker);
22
+ setTimeout(() => form.querySelector('button').click());
23
+ await oneDefaultPreventedEvent(form, 'submit');
24
+ const formData = new FormData(form);
25
+ const dateValue = formData.get('booking_date');
26
+ const expectedDate = `2025-04-16`;
27
+ expect(dateValue).to.equal(expectedDate);
28
+ });
29
+ it(`should emit change event on date selection for ${timezone}`, async () => {
30
+ Settings.defaultZone = timezone;
31
+ const datePicker = await fixture(html ` <tt-date-picker value="2025-05-01"></tt-date-picker> `);
32
+ const showCalendarButton = datePicker.shadowRoot.querySelector('[name="Show calendar"]');
33
+ showCalendarButton.click();
34
+ await elementUpdated(datePicker);
35
+ const changeEventPromise = oneEvent(datePicker, 'change');
36
+ const calendar = datePicker.shadowRoot.querySelector('tt-calendar');
37
+ const desiredDate = calendar.shadowRoot.querySelector('[role="gridcell"][data-date="2025-05-02"]');
38
+ desiredDate.click();
39
+ await elementUpdated(datePicker);
40
+ const event = await changeEventPromise;
41
+ const eventValue = event.target.value;
42
+ const expectedEndDate = '2025-05-02';
43
+ expect(eventValue).to.equal(expectedEndDate);
44
+ });
45
+ });
46
+ it('should disable the date picker when disabled attribute is set', async () => {
47
+ const datePicker = await fixture(html `
48
+ <tt-date-picker value="2025-05-01" disabled></tt-date-picker>
49
+ `);
50
+ expect(datePicker.inert).to.be.true;
51
+ const showCalendarButton = datePicker.shadowRoot.querySelector('button.show-calendar');
52
+ showCalendarButton.click();
53
+ const dialog = datePicker.shadowRoot.querySelector('tt-dialog');
54
+ expect(dialog.open).to.be.false;
55
+ });
56
+ });
57
+ //# sourceMappingURL=date-picker.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"date-picker.test.js","sourceRoot":"","sources":["../../../test/date-picker.test.ts"],"names":[],"mappings":"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","sourcesContent":["import { elementUpdated, expect, fixture, oneDefaultPreventedEvent, oneEvent } from '@open-wc/testing';\nimport { html } from 'lit';\nimport { Settings } from 'luxon';\nimport { TtDialog } from '@triptease/tt-dialog';\nimport '../src/index.js'; // This handles the registration\nimport { TtDatePicker } from '../src/index.js';\n\ndescribe('TtDatePicker', () => {\n afterEach(() => {\n Settings.defaultZone = 'system';\n });\n\n ['Australia/Brisbane', 'America/New_York', 'Europe/London', 'Asia/Tokyo', 'Etc/UTC'].forEach((timezone) => {\n it('should include date value in form submission data', async () => {\n Settings.defaultZone = timezone;\n const form = await fixture<HTMLFormElement>(html`\n <form method=\"post\" action=\"#\">\n <input type=\"text\" name=\"other_field\" value=\"test_value\" />\n <tt-date-picker name=\"booking_date\" value=\"2025-04-16\"></tt-date-picker>\n <button type=\"submit\">Submit</button>\n </form>\n `);\n\n const datePicker = form.querySelector('tt-date-picker') as TtDatePicker;\n expect(datePicker).to.exist;\n\n await elementUpdated(datePicker);\n\n setTimeout(() => form.querySelector('button')!.click());\n await oneDefaultPreventedEvent(form, 'submit');\n\n const formData = new FormData(form);\n const dateValue = formData.get('booking_date');\n const expectedDate = `2025-04-16`;\n\n expect(dateValue).to.equal(expectedDate);\n });\n\n it(`should emit change event on date selection for ${timezone}`, async () => {\n Settings.defaultZone = timezone;\n const datePicker = await fixture<TtDatePicker>(html` <tt-date-picker value=\"2025-05-01\"></tt-date-picker> `);\n\n const showCalendarButton = datePicker.shadowRoot!.querySelector('[name=\"Show calendar\"]') as HTMLButtonElement;\n showCalendarButton.click();\n await elementUpdated(datePicker);\n\n const changeEventPromise = oneEvent(datePicker, 'change');\n\n const calendar = datePicker.shadowRoot!.querySelector('tt-calendar')!;\n const desiredDate = calendar.shadowRoot!.querySelector(\n '[role=\"gridcell\"][data-date=\"2025-05-02\"]'\n ) as HTMLLIElement;\n desiredDate.click();\n await elementUpdated(datePicker);\n\n const event = await changeEventPromise;\n const eventValue = (event.target as TtDatePicker).value;\n\n const expectedEndDate = '2025-05-02';\n\n expect(eventValue).to.equal(expectedEndDate);\n });\n });\n\n it('should disable the date picker when disabled attribute is set', async () => {\n const datePicker = await fixture<TtDatePicker>(html`\n <tt-date-picker value=\"2025-05-01\" disabled></tt-date-picker>\n `);\n\n expect(datePicker.inert).to.be.true;\n\n const showCalendarButton = datePicker.shadowRoot!.querySelector('button.show-calendar') as HTMLButtonElement;\n showCalendarButton.click();\n\n const dialog = datePicker.shadowRoot!.querySelector('tt-dialog') as TtDialog;\n expect(dialog.open).to.be.false;\n });\n});\n"]}
package/package.json CHANGED
@@ -3,28 +3,33 @@
3
3
  "description": "Webcomponent tt-date-picker following open-wc recommendations",
4
4
  "license": "MIT",
5
5
  "author": "@triptease",
6
- "version": "6.1.3",
6
+ "version": "6.2.0",
7
7
  "type": "module",
8
- "main": "dist/src/index.js",
9
- "module": "dist/src/index.js",
8
+ "main": "dist/esm/src/index.js",
9
+ "module": "dist/esm/src/index.js",
10
10
  "exports": {
11
11
  ".": {
12
- "import": "./dist/src/index.js",
13
- "types": "./dist/src/types.d.ts"
12
+ "types": "./dist/esm/src/index.d.ts",
13
+ "import": "./dist/esm/src/index.js",
14
+ "require": "./dist/cjs/src/index.js"
14
15
  },
15
16
  "./tt-date-picker.js": {
16
- "import": "./dist/src/tt-date-picker.js",
17
- "types": "./dist/src/types.d.ts"
17
+ "types": "./dist/esm/src/tt-date-picker.d.ts",
18
+ "import": "./dist/esm/src/tt-date-picker.js",
19
+ "require": "./dist/cjs/src/tt-date-picker.js"
18
20
  }
19
21
  },
20
22
  "files": [
21
- "dist/src"
23
+ "dist/esm",
24
+ "dist/cjs"
22
25
  ],
23
26
  "scripts": {
24
27
  "analyze": "cem analyze --litelement",
25
28
  "start": "tsc && concurrently -k -r \"tsc --watch --preserveWatchOutput\" \"web-dev-server\"",
26
29
  "build": "yarn build:node && yarn build:web && npm run analyze -- --exclude dist",
27
- "build:node": "tsc",
30
+ "build:esm": "tsc",
31
+ "build:cjs": "tsc -p tsconfig.cjs.json && node ../../scripts/create-cjs-package.mjs",
32
+ "build:node": "yarn build:esm && yarn build:cjs",
28
33
  "build:node:watch": "tsc --watch",
29
34
  "build:web": "node ../../scripts/esbuild.mjs",
30
35
  "prepublish": "tsc && npm run analyze -- --exclude dist",
@@ -32,10 +37,10 @@
32
37
  "test:watch": "tsc && concurrently -k -r \"tsc --watch --preserveWatchOutput\" \"wtr --watch\""
33
38
  },
34
39
  "dependencies": {
35
- "@triptease/icons": "1.3.14",
36
- "@triptease/tt-calendar": "6.0.17",
37
- "@triptease/tt-date-input": "6.0.6",
38
- "@triptease/tt-dialog": "5.1.1",
40
+ "@triptease/icons": "1.4.0",
41
+ "@triptease/tt-calendar": "6.1.0",
42
+ "@triptease/tt-date-input": "6.1.0",
43
+ "@triptease/tt-dialog": "5.2.0",
39
44
  "@types/luxon": "^3.6.2",
40
45
  "lit": "^3.1.4",
41
46
  "luxon": "^3.6.1"
@@ -1 +0,0 @@
1
- {"version":3,"file":"helpers.js","sourceRoot":"","sources":["../../src/helpers.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,OAAO,CAAC;AAEjC,MAAM,CAAC,MAAM,aAAa,GAAG,CAAC,KAAoB,EAAoB,EAAE,CACtE,KAAK,CAAC,CAAC,CAAC,QAAQ,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,QAAQ,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC;AAEzD,MAAM,CAAC,MAAM,iBAAiB,GAAG,CAAC,KAAoB,EAAwB,EAAE,CAC9E,KAAK,CAAC,CAAC,CAAC,QAAQ,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC","sourcesContent":["import { DateTime } from 'luxon';\n\nexport const dateConverter = (value: string | null): Date | undefined =>\n value ? DateTime.fromISO(value).toJSDate() : undefined;\n\nexport const dateTimeConverter = (value: string | null): DateTime | undefined =>\n value ? DateTime.fromISO(value) : undefined;\n"]}
@@ -1 +0,0 @@
1
- {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/index.ts"],"names":[],"mappings":"AAAA,cAAc,YAAY,CAAC;AAC3B,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC","sourcesContent":["export * from './types.js';\nexport { TtDatePicker } from './tt-date-picker.js';\n"]}
@@ -1 +0,0 @@
1
- {"version":3,"file":"styles.js","sourceRoot":"","sources":["../../src/styles.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,GAAG,EAAE,MAAM,KAAK,CAAC;AAE1B,MAAM,CAAC,MAAM,MAAM,GAAG,GAAG,CAAA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;CAgExB,CAAC","sourcesContent":["import { css } from 'lit';\n\nexport const styles = css`\n :host {\n position: relative;\n }\n\n :host(:state(disabled)) {\n pointer-events: none;\n opacity: 0.5;\n }\n\n tt-side-panel::part(side-panel) {\n border-right: 1px solid var(--color-border-200);\n }\n\n tt-side-panel:empty {\n display: none;\n }\n\n [part='container'] {\n padding: var(--space-scale-1) var(--space-scale-1-5);\n gap: var(--space-scale-1);\n }\n\n [part='controls'] {\n display: flex;\n border: 1px solid var(--color-border-200);\n border-radius: var(--border-radius);\n width: fit-content;\n height: var(--date-picker-height, fit-content);\n background-color: var(--color-surface-100);\n font-size: var(--font-size-200);\n color: var(--color-text-400);\n align-items: stretch;\n justify-content: space-around;\n padding: 0;\n\n &[aria-invalid='true'] {\n outline: 2px solid var(--color-alert-300, red);\n }\n\n button {\n border: 0;\n padding-inline: var(--space-scale-1);\n background-color: transparent;\n cursor: pointer;\n display: flex;\n align-items: center;\n justify-content: center;\n align-self: stretch;\n border-left: 1px solid var(--color-border-200);\n box-sizing: border-box;\n color: var(--color-text-400);\n }\n\n &:has(+ tt-dialog[open]) .show-calendar svg,\n .show-calendar:hover svg {\n color: var(--color-primary-400);\n }\n }\n\n [part='dialog']::part(dialog) {\n margin-top: var(--space-scale-1);\n flex-direction: row-reverse;\n }\n`;\n"]}
@@ -1 +0,0 @@
1
- {"version":3,"file":"tt-date-picker.js","sourceRoot":"","sources":["../../src/tt-date-picker.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,uCAAuC,CAAC;AACjE,OAAO,EAAE,SAAS,EAAE,MAAM,2CAA2C,CAAC;AACtE,OAAO,EAAE,QAAQ,EAAE,MAAM,mCAAmC,CAAC;AAC7D,OAAO,EAAE,YAAY,EAAE,MAAM,mBAAmB,CAAC;AAEjD,IAAI,OAAO,MAAM,KAAK,WAAW,EAAE,CAAC;IAClC,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,GAAG,CAAC,aAAa,CAAC,EAAE,CAAC;QAC9C,MAAM,CAAC,cAAc,CAAC,MAAM,CAAC,aAAa,EAAE,QAAQ,CAAC,CAAC;IACxD,CAAC;IACD,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,GAAG,CAAC,eAAe,CAAC,EAAE,CAAC;QAChD,MAAM,CAAC,cAAc,CAAC,MAAM,CAAC,eAAe,EAAE,SAAS,CAAC,CAAC;IAC3D,CAAC;IAED,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,GAAG,CAAC,WAAW,CAAC,EAAE,CAAC;QAC5C,MAAM,CAAC,cAAc,CAAC,MAAM,CAAC,WAAW,EAAE,QAAQ,CAAC,CAAC;IACtD,CAAC;IAED,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,GAAG,CAAC,gBAAgB,CAAC,EAAE,CAAC;QACjD,MAAM,CAAC,cAAc,CAAC,MAAM,CAAC,gBAAgB,EAAE,YAAY,CAAC,CAAC;IAC/D,CAAC;AACH,CAAC;AAED,OAAO,EAAE,YAAY,EAAE,CAAC","sourcesContent":["import { Calendar } from '@triptease/tt-calendar/tt-calendar.js';\nimport { DateInput } from '@triptease/tt-date-input/tt-date-input.js';\nimport { TtDialog } from '@triptease/tt-dialog/tt-dialog.js';\nimport { TtDatePicker } from './TtDatePicker.js';\n\nif (typeof window !== 'undefined') {\n if (!window.customElements.get('tt-calendar')) {\n window.customElements.define('tt-calendar', Calendar);\n }\n if (!window.customElements.get('tt-date-input')) {\n window.customElements.define('tt-date-input', DateInput);\n }\n\n if (!window.customElements.get('tt-dialog')) {\n window.customElements.define('tt-dialog', TtDialog);\n }\n\n if (!window.customElements.get('tt-date-picker')) {\n window.customElements.define('tt-date-picker', TtDatePicker);\n }\n}\n\nexport { TtDatePicker };\n"]}
@@ -1 +0,0 @@
1
- {"version":3,"file":"types.js","sourceRoot":"","sources":["../../src/types.ts"],"names":[],"mappings":"","sourcesContent":["import { TtDatePicker } from './TtDatePicker.js';\n\ninterface TtDatePickerExternalAttributes {\n id?: string;\n value?: string;\n disabled?: boolean;\n 'open-left'?: boolean;\n name?: string;\n minDate?: string;\n maxDate?: string;\n}\n\ndeclare global {\n interface HTMLElementTagNameMap {\n 'tt-date-picker': TtDatePicker;\n }\n\n namespace JSX {\n interface IntrinsicElements {\n 'tt-date-picker': TtDatePickerExternalAttributes & { style?: string };\n }\n }\n\n namespace React {\n namespace JSX {\n interface IntrinsicElements {\n 'tt-date-picker': TtDatePickerExternalAttributes & {\n ref?: React.Ref<unknown>;\n style?: React.CSSProperties;\n };\n }\n }\n }\n}\n"]}
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes