vira 31.19.0 → 31.21.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.
@@ -23,3 +23,4 @@ export * from './vira-select.element.js';
23
23
  export * from './vira-tabs.element.js';
24
24
  export * from './vira-tag.element.js';
25
25
  export * from './vira-text-area.element.js';
26
+ export * from './vira-theme-switcher.element.js';
@@ -23,3 +23,4 @@ export * from './vira-select.element.js';
23
23
  export * from './vira-tabs.element.js';
24
24
  export * from './vira-tag.element.js';
25
25
  export * from './vira-text-area.element.js';
26
+ export * from './vira-theme-switcher.element.js';
@@ -17,10 +17,10 @@ export type ViraCheckboxInputs = {
17
17
  } & PartialWithUndefined<{
18
18
  stylePassthrough: Partial<Record<ViraCheckboxInnerElements, CSSResult>>;
19
19
  attributePassthrough: Partial<Record<ViraCheckboxInnerElements, AttributeValues>>;
20
- disabled: boolean;
20
+ isDisabled: boolean;
21
21
  label: string;
22
22
  hasError: boolean;
23
- horizontal: boolean;
23
+ useHorizontalLabel: boolean;
24
24
  /** The checkbox will be filled with a form selection color when it is checked. */
25
25
  fillWhenChecked: boolean;
26
26
  /** The checkbox will be filled with a form error color when it is unchecked. */
@@ -15,7 +15,7 @@ import { ViraIcon } from './vira-icon.element.js';
15
15
  export const ViraCheckbox = defineViraElement()({
16
16
  tagName: 'vira-checkbox',
17
17
  hostClasses: {
18
- 'vira-checkbox-horizontal': ({ inputs }) => !!inputs.horizontal,
18
+ 'vira-checkbox-horizontal': ({ inputs }) => !!inputs.useHorizontalLabel,
19
19
  'vira-checkbox-filled-checked': ({ inputs }) => !!inputs.fillWhenChecked,
20
20
  'vira-checkbox-filled-unchecked': ({ inputs }) => !!inputs.fillWhenUnchecked,
21
21
  },
@@ -126,8 +126,8 @@ export const ViraCheckbox = defineViraElement()({
126
126
  }
127
127
 
128
128
  ${hostClasses['vira-checkbox-horizontal'].selector} label {
129
- flex-direction: row-reverse;
130
- align-items: flex-start;
129
+ flex-direction: row;
130
+ align-items: center;
131
131
  gap: 8px;
132
132
 
133
133
  & .label-text {
@@ -140,7 +140,7 @@ export const ViraCheckbox = defineViraElement()({
140
140
  },
141
141
  render({ inputs, dispatch, events }) {
142
142
  function updateValue() {
143
- if (!inputs.disabled) {
143
+ if (!inputs.isDisabled) {
144
144
  dispatch(new events.valueChange(!inputs.value));
145
145
  }
146
146
  }
@@ -158,7 +158,7 @@ export const ViraCheckbox = defineViraElement()({
158
158
  return html `
159
159
  <label
160
160
  class=${classMap({
161
- disabled: !!inputs.disabled,
161
+ disabled: !!inputs.isDisabled,
162
162
  })}
163
163
  ${attributes(inputs.attributePassthrough?.label)}
164
164
  style=${ifDefined(inputs.stylePassthrough?.label)}
@@ -168,14 +168,14 @@ export const ViraCheckbox = defineViraElement()({
168
168
  <span
169
169
  class="custom-checkbox ${classMap({
170
170
  checked: inputs.value,
171
- disabled: !!inputs.disabled,
171
+ disabled: !!inputs.isDisabled,
172
172
  error: !!inputs.hasError,
173
173
  })}"
174
174
  role="checkbox"
175
175
  aria-label=${ifDefined(inputs.label || undefined)}
176
176
  aria-checked=${inputs.value ? 'true' : 'false'}
177
- aria-disabled=${inputs.disabled ? 'true' : 'false'}
178
- tabindex=${inputs.disabled ? '-1' : '0'}
177
+ aria-disabled=${inputs.isDisabled ? 'true' : 'false'}
178
+ tabindex=${inputs.isDisabled ? '-1' : '0'}
179
179
  ${attributes(inputs.attributePassthrough?.['custom-checkbox'])}
180
180
  style=${ifDefined(inputs.stylePassthrough?.['custom-checkbox'])}
181
181
  ${listenToActivate(updateValue)}
@@ -29,6 +29,21 @@ export declare const ViraForm: import("element-vir").DeclarativeElementDefinitio
29
29
  * @default false
30
30
  */
31
31
  horizontalCheckboxes: boolean;
32
+ /**
33
+ * When `true`, all form field labels render to the left of their inputs instead of above
34
+ * them.
35
+ *
36
+ * @default false
37
+ */
38
+ useHorizontalLabels: boolean;
39
+ /**
40
+ * When `true`, all fields in this form are prevented from user edits. Inputs, selects, and
41
+ * text areas render their current value as plain text; checkboxes render disabled since
42
+ * they have no native readonly mode.
43
+ *
44
+ * @default false
45
+ */
46
+ isReadonly: boolean;
32
47
  }>, {
33
48
  lastIsValid: boolean;
34
49
  }, {
@@ -1,5 +1,6 @@
1
1
  import { getObjectTypedEntries } from '@augment-vir/common';
2
- import { css, defineElementEvent, html, listen, nothing, testId } from 'element-vir';
2
+ import { css, defineElementEvent, html, listen, nothing, testId, } from 'element-vir';
3
+ import { viraFormCssVars } from '../styles/form-styles.js';
3
4
  import { defineViraElement } from '../util/define-vira-element.js';
4
5
  import { applyRequiredLabel, areFormFieldsValid, ViraFormFieldType, } from '../util/vira-form-fields.js';
5
6
  import { ViraCheckbox } from './vira-checkbox.element.js';
@@ -14,6 +15,11 @@ import { ViraTextArea } from './vira-text-area.element.js';
14
15
  */
15
16
  export const ViraForm = defineViraElement()({
16
17
  tagName: 'vira-form',
18
+ state() {
19
+ return {
20
+ lastIsValid: false,
21
+ };
22
+ },
17
23
  events: {
18
24
  valueChange: defineElementEvent(),
19
25
  validChange: defineElementEvent(),
@@ -34,12 +40,35 @@ export const ViraForm = defineViraElement()({
34
40
  width: unset;
35
41
  }
36
42
  }
43
+
44
+ .horizontal-fields {
45
+ width: 100%;
46
+ border-collapse: separate;
47
+ border-spacing: 0 10px;
48
+
49
+ & th,
50
+ & td {
51
+ padding: 0;
52
+ }
53
+
54
+ & th {
55
+ padding: 0 8px;
56
+ vertical-align: middle;
57
+ white-space: nowrap;
58
+ font-weight: ${viraFormCssVars['vira-form-label-font-weight'].value};
59
+ text-align: right;
60
+ }
61
+
62
+ & td {
63
+ width: 100%;
64
+ vertical-align: top;
65
+
66
+ & > ${ViraCheckbox}, & > ${ViraInput}, & > ${ViraSelect}, & > ${ViraTextArea} {
67
+ width: 100%;
68
+ }
69
+ }
70
+ }
37
71
  `,
38
- state() {
39
- return {
40
- lastIsValid: false,
41
- };
42
- },
43
72
  render({ inputs, dispatch, events, state, updateState }) {
44
73
  const currentIsValid = areFormFieldsValid(inputs.fields);
45
74
  if (currentIsValid !== state.lastIsValid) {
@@ -50,168 +79,236 @@ export const ViraForm = defineViraElement()({
50
79
  allFieldsAreValid: currentIsValid,
51
80
  }));
52
81
  }
82
+ function wrapFormField({ fieldTemplate, label, }) {
83
+ if (inputs.useHorizontalLabels) {
84
+ return html `
85
+ <tr>
86
+ <th scope="row">${label}</th>
87
+ <td>${fieldTemplate}</td>
88
+ </tr>
89
+ `;
90
+ }
91
+ else {
92
+ return fieldTemplate;
93
+ }
94
+ }
53
95
  const formFieldTemplates = getObjectTypedEntries(inputs.fields).map(([key, field,]) => {
96
+ const label = applyRequiredLabel(field.label, !!field.isRequired && !inputs.hideRequiredMarkers);
97
+ const isDisabled = !!(inputs.isDisabled || field.isDisabled);
98
+ const childLabel = inputs.useHorizontalLabels ? undefined : label;
99
+ const horizontalLabelAttributes = inputs.useHorizontalLabels && label
100
+ ? {
101
+ 'aria-label': label,
102
+ }
103
+ : {};
54
104
  if (field.isHidden) {
55
105
  return nothing;
56
106
  }
57
107
  else if (field.type === ViraFormFieldType.Checkbox) {
58
- return html `
59
- <${ViraCheckbox.assign({
60
- value: field.value || false,
61
- disabled: inputs.isDisabled || field.isDisabled,
62
- hasError: field.hasError,
63
- horizontal: inputs.horizontalCheckboxes,
64
- label: applyRequiredLabel(field.label, !!field.isRequired && !inputs.hideRequiredMarkers),
65
- })}
66
- ${field.testId ? testId(field.testId) : nothing}
67
- ${listen(ViraCheckbox.events.valueChange, (event) => {
68
- dispatch(new events.valueChange({
69
- key,
70
- ...field,
71
- value: event.detail,
72
- }));
73
- })}
74
- ></${ViraCheckbox}>
75
- `;
108
+ return wrapFormField({
109
+ label,
110
+ fieldTemplate: html `
111
+ <${ViraCheckbox.assign({
112
+ value: field.value || false,
113
+ isDisabled: !!(isDisabled || inputs.isReadonly),
114
+ hasError: field.hasError,
115
+ useHorizontalLabel: inputs.horizontalCheckboxes,
116
+ fillWhenChecked: field.fillWhenChecked,
117
+ fillWhenUnchecked: field.fillWhenUnchecked,
118
+ label: childLabel,
119
+ ...(inputs.useHorizontalLabels && label
120
+ ? {
121
+ attributePassthrough: {
122
+ 'custom-checkbox': horizontalLabelAttributes,
123
+ },
124
+ }
125
+ : {}),
126
+ })}
127
+ ${field.testId ? testId(field.testId) : nothing}
128
+ ${listen(ViraCheckbox.events.valueChange, (event) => {
129
+ dispatch(new events.valueChange({
130
+ key,
131
+ ...field,
132
+ value: event.detail,
133
+ }));
134
+ })}
135
+ ></${ViraCheckbox}>
136
+ `,
137
+ });
76
138
  }
77
139
  else if (field.type === ViraFormFieldType.Select) {
78
- return html `
79
- <${ViraSelect.assign({
80
- options: field.options,
81
- value: field.value,
82
- placeholder: field.placeholder,
83
- disabled: inputs.isDisabled || field.isDisabled,
84
- label: applyRequiredLabel(field.label, !!field.isRequired && !inputs.hideRequiredMarkers),
85
- hasError: field.hasError,
86
- icon: field.icon,
87
- })}
88
- ${field.testId ? testId(field.testId) : nothing}
89
- ${listen(ViraSelect.events.valueChange, (event) => {
90
- dispatch(new events.valueChange({
91
- key,
92
- ...field,
93
- value: event.detail,
94
- }));
95
- })}
96
- ></${ViraSelect}>
97
- `;
140
+ return wrapFormField({
141
+ label,
142
+ fieldTemplate: html `
143
+ <${ViraSelect.assign({
144
+ options: field.options,
145
+ value: field.value,
146
+ placeholder: field.placeholder,
147
+ disabled: isDisabled,
148
+ isReadonly: inputs.isReadonly,
149
+ label: childLabel,
150
+ hasError: field.hasError,
151
+ icon: field.icon,
152
+ ...(inputs.useHorizontalLabels && label
153
+ ? {
154
+ attributePassthrough: {
155
+ select: horizontalLabelAttributes,
156
+ },
157
+ }
158
+ : {}),
159
+ })}
160
+ ${field.testId ? testId(field.testId) : nothing}
161
+ ${listen(ViraSelect.events.valueChange, (event) => {
162
+ dispatch(new events.valueChange({
163
+ key,
164
+ ...field,
165
+ value: event.detail,
166
+ }));
167
+ })}
168
+ ></${ViraSelect}>
169
+ `,
170
+ });
98
171
  }
99
172
  else if (field.type === ViraFormFieldType.TextArea) {
100
- return html `
101
- <${ViraTextArea.assign({
102
- value: field.value || '',
103
- disabled: inputs.isDisabled || field.isDisabled,
104
- hasError: field.hasError,
105
- label: applyRequiredLabel(field.label, !!field.isRequired && !inputs.hideRequiredMarkers),
106
- placeholder: field.placeholder,
107
- rows: field.rows,
108
- preventResize: field.preventResize,
109
- })}
110
- ${field.testId ? testId(field.testId) : nothing}
111
- ${listen(ViraTextArea.events.valueChange, (event) => {
112
- dispatch(new events.valueChange({
113
- key,
114
- ...field,
115
- value: event.detail,
116
- }));
117
- })}
118
- ></${ViraTextArea}>
119
- `;
173
+ return wrapFormField({
174
+ label,
175
+ fieldTemplate: html `
176
+ <${ViraTextArea.assign({
177
+ value: field.value || '',
178
+ disabled: isDisabled,
179
+ hasError: field.hasError,
180
+ isReadonly: inputs.isReadonly,
181
+ label: childLabel,
182
+ placeholder: field.placeholder,
183
+ rows: field.rows,
184
+ preventResize: field.preventResize,
185
+ attributePassthrough: horizontalLabelAttributes,
186
+ })}
187
+ ${field.testId ? testId(field.testId) : nothing}
188
+ ${listen(ViraTextArea.events.valueChange, (event) => {
189
+ dispatch(new events.valueChange({
190
+ key,
191
+ ...field,
192
+ value: event.detail,
193
+ }));
194
+ })}
195
+ ></${ViraTextArea}>
196
+ `,
197
+ });
120
198
  }
121
199
  else if (field.type === ViraFormFieldType.Number) {
122
- return html `
123
- <${ViraInput.assign({
124
- value: field.value?.toString() || '',
125
- disabled: inputs.isDisabled || field.isDisabled,
126
- allowedInputs: /\d/,
127
- hasError: field.hasError,
128
- icon: field.icon,
129
- label: applyRequiredLabel(field.label, !!field.isRequired && !inputs.hideRequiredMarkers),
130
- placeholder: field.placeholder,
131
- showClearButton: inputs.showClearButtons,
132
- type: ViraInputType.Number,
133
- attributePassthrough: {
134
- ...(field.min === undefined
135
- ? {}
136
- : {
137
- min: String(field.min),
138
- }),
139
- ...(field.max === undefined
140
- ? {}
141
- : {
142
- max: String(field.max),
143
- }),
144
- ...(field.step === undefined
145
- ? {}
146
- : {
147
- step: String(field.step),
148
- }),
149
- },
150
- })}
151
- ${field.testId ? testId(field.testId) : nothing}
152
- ${listen(ViraInput.events.valueChange, (event) => {
153
- const numericValue = event.detail === '' ? undefined : Number(event.detail);
154
- dispatch(new events.valueChange({
155
- key,
156
- ...field,
157
- value: numericValue,
158
- }));
159
- })}
160
- ></${ViraInput}>
161
- `;
200
+ return wrapFormField({
201
+ label,
202
+ fieldTemplate: html `
203
+ <${ViraInput.assign({
204
+ value: field.value?.toString() || '',
205
+ disabled: isDisabled,
206
+ allowedInputs: /\d/,
207
+ hasError: field.hasError,
208
+ icon: field.icon,
209
+ isReadonly: inputs.isReadonly,
210
+ label: childLabel,
211
+ placeholder: field.placeholder,
212
+ showClearButton: inputs.showClearButtons,
213
+ type: ViraInputType.Number,
214
+ attributePassthrough: {
215
+ ...horizontalLabelAttributes,
216
+ ...(field.min === undefined
217
+ ? {}
218
+ : {
219
+ min: String(field.min),
220
+ }),
221
+ ...(field.max === undefined
222
+ ? {}
223
+ : {
224
+ max: String(field.max),
225
+ }),
226
+ ...(field.step === undefined
227
+ ? {}
228
+ : {
229
+ step: String(field.step),
230
+ }),
231
+ },
232
+ })}
233
+ ${field.testId ? testId(field.testId) : nothing}
234
+ ${listen(ViraInput.events.valueChange, (event) => {
235
+ const numericValue = event.detail === '' ? undefined : Number(event.detail);
236
+ dispatch(new events.valueChange({
237
+ key,
238
+ ...field,
239
+ value: numericValue,
240
+ }));
241
+ })}
242
+ ></${ViraInput}>
243
+ `,
244
+ });
162
245
  }
163
246
  else {
164
- return html `
165
- <${ViraInput.assign({
166
- value: field.value || '',
167
- disabled: inputs.isDisabled || field.isDisabled,
168
- hasError: field.hasError,
169
- icon: field.icon,
170
- label: applyRequiredLabel(field.label, !!field.isRequired && !inputs.hideRequiredMarkers),
171
- placeholder: field.placeholder,
172
- showClearButton: inputs.showClearButtons,
173
- attributePassthrough: field.isUsername
174
- ? {
175
- autocomplete: 'username',
176
- }
177
- : field.type === ViraFormFieldType.NewPassword
178
- ? {
179
- autocomplete: 'new-password',
180
- }
181
- : field.type === ViraFormFieldType.ExistingPassword
247
+ return wrapFormField({
248
+ label,
249
+ fieldTemplate: html `
250
+ <${ViraInput.assign({
251
+ value: field.value || '',
252
+ disabled: isDisabled,
253
+ hasError: field.hasError,
254
+ icon: field.icon,
255
+ isReadonly: inputs.isReadonly,
256
+ label: childLabel,
257
+ placeholder: field.placeholder,
258
+ showClearButton: inputs.showClearButtons,
259
+ attributePassthrough: {
260
+ ...horizontalLabelAttributes,
261
+ ...(field.isUsername
182
262
  ? {
183
- autocomplete: 'password',
263
+ autocomplete: 'username',
184
264
  }
185
- : field.type === ViraFormFieldType.Email
265
+ : field.type === ViraFormFieldType.NewPassword
186
266
  ? {
187
- autocomplete: 'email',
267
+ autocomplete: 'new-password',
188
268
  }
189
- : {},
190
- type: [
191
- ViraFormFieldType.NewPassword,
192
- ViraFormFieldType.ExistingPassword,
193
- ViraFormFieldType.PlainPassword,
194
- ].includes(field.type)
195
- ? ViraInputType.Password
196
- : field.type === ViraFormFieldType.Email
197
- ? ViraInputType.Email
198
- : ViraInputType.Default,
199
- })}
200
- ${field.testId ? testId(field.testId) : nothing}
201
- ${listen(ViraInput.events.valueChange, (event) => {
202
- dispatch(new events.valueChange({
203
- key,
204
- ...field,
205
- value: event.detail,
206
- }));
207
- })}
208
- ></${ViraInput}>
209
- `;
269
+ : field.type === ViraFormFieldType.ExistingPassword
270
+ ? {
271
+ autocomplete: 'password',
272
+ }
273
+ : field.type === ViraFormFieldType.Email
274
+ ? {
275
+ autocomplete: 'email',
276
+ }
277
+ : {}),
278
+ },
279
+ type: [
280
+ ViraFormFieldType.NewPassword,
281
+ ViraFormFieldType.ExistingPassword,
282
+ ViraFormFieldType.PlainPassword,
283
+ ].includes(field.type)
284
+ ? ViraInputType.Password
285
+ : field.type === ViraFormFieldType.Email
286
+ ? ViraInputType.Email
287
+ : ViraInputType.Default,
288
+ })}
289
+ ${field.testId ? testId(field.testId) : nothing}
290
+ ${listen(ViraInput.events.valueChange, (event) => {
291
+ dispatch(new events.valueChange({
292
+ key,
293
+ ...field,
294
+ value: event.detail,
295
+ }));
296
+ })}
297
+ ></${ViraInput}>
298
+ `,
299
+ });
210
300
  }
211
301
  });
302
+ const formFieldsWrapper = inputs.useHorizontalLabels
303
+ ? html `
304
+ <table class="horizontal-fields">
305
+ <tbody>${formFieldTemplates}</tbody>
306
+ </table>
307
+ `
308
+ : formFieldTemplates;
212
309
  return html `
213
310
  <form ${listen('submit', (event) => event.preventDefault())}>
214
- ${formFieldTemplates}
311
+ ${formFieldsWrapper}
215
312
  <slot></slot>
216
313
  </form>
217
314
  `;
@@ -35,6 +35,7 @@ export declare const ViraInput: import("element-vir").DeclarativeElementDefiniti
35
35
  } & PartialWithUndefined<{
36
36
  placeholder: string;
37
37
  disabled: boolean;
38
+ isReadonly: boolean;
38
39
  allowedInputs: string | RegExp;
39
40
  blockedInputs: string | RegExp;
40
41
  disableBrowserHelps: boolean;
@@ -156,6 +156,10 @@ export const ViraInput = defineViraElement()({
156
156
  margin-right: calc(${cssVars['vira-input-padding-horizontal'].value} - 4px);
157
157
  }
158
158
 
159
+ .readonly-value {
160
+ overflow-wrap: anywhere;
161
+ }
162
+
159
163
  input {
160
164
  ${noNativeFormStyles};
161
165
  cursor: text;
@@ -293,6 +297,22 @@ export const ViraInput = defineViraElement()({
293
297
  allowed: inputs.allowedInputs,
294
298
  blocked: inputs.blockedInputs,
295
299
  });
300
+ if (inputs.isReadonly) {
301
+ const readonlyValueTemplate = html `
302
+ <span class="readonly-value">${filteredValue}</span>
303
+ `;
304
+ if (inputs.label) {
305
+ return html `
306
+ <label>
307
+ <span class="input-label">${inputs.label}</span>
308
+ ${readonlyValueTemplate}
309
+ </label>
310
+ `;
311
+ }
312
+ else {
313
+ return readonlyValueTemplate;
314
+ }
315
+ }
296
316
  const iconTemplate = inputs.icon
297
317
  ? html `
298
318
  <${ViraIcon.assign({
@@ -290,7 +290,7 @@ export const ViraJsonForm = defineViraElement()({
290
290
  return html `
291
291
  <${ViraCheckbox.assign({
292
292
  value: value === true,
293
- disabled: isDisabled,
293
+ isDisabled,
294
294
  })}
295
295
  ${listen(ViraCheckbox.events.valueChange, (event) => {
296
296
  emitReplaceAt(path, event.detail);
@@ -20,6 +20,11 @@ export declare const ViraSelect: import("element-vir").DeclarativeElementDefinit
20
20
  /** If set to `true`, only minimal styles are applied. */
21
21
  rawSelect: boolean;
22
22
  disabled: boolean;
23
+ /**
24
+ * When `true`, the currently selected option's label is rendered as plain text with no
25
+ * wrapper, border, or focus styles.
26
+ */
27
+ isReadonly: boolean;
23
28
  attributePassthrough: Readonly<PartialWithUndefined<{
24
29
  label: AttributeValues;
25
30
  select: AttributeValues;
@@ -185,6 +185,10 @@ export const ViraSelect = defineViraElement()({
185
185
  }
186
186
  }
187
187
 
188
+ .readonly-value {
189
+ overflow-wrap: anywhere;
190
+ }
191
+
188
192
  ${hostClasses['vira-select-disabled'].selector} {
189
193
  cursor: not-allowed;
190
194
 
@@ -252,6 +256,27 @@ export const ViraSelect = defineViraElement()({
252
256
  },
253
257
  render({ inputs, state, dispatch, events }) {
254
258
  const value = inputs.value || undefined;
259
+ if (inputs.isReadonly) {
260
+ const selectedOption = inputs.options
261
+ .flatMap((entry) => (isViraSelectOptionGroup(entry) ? [...entry.options] : [entry]))
262
+ .find((option) => option.value === value);
263
+ const readonlyValueTemplate = html `
264
+ <span class="readonly-value">
265
+ ${selectedOption?.label || inputs.placeholder || ''}
266
+ </span>
267
+ `;
268
+ if (inputs.label) {
269
+ return html `
270
+ <label ${attributes(inputs.attributePassthrough?.label)}>
271
+ <span class="select-label">${inputs.label}</span>
272
+ ${readonlyValueTemplate}
273
+ </label>
274
+ `;
275
+ }
276
+ else {
277
+ return readonlyValueTemplate;
278
+ }
279
+ }
255
280
  const placeholderOptionTemplate = inputs.placeholder || value == undefined
256
281
  ? html `
257
282
  <option value="" disabled ?selected=${value == undefined}>
@@ -21,6 +21,7 @@ export declare const ViraTextArea: import("element-vir").DeclarativeElementDefin
21
21
  } & PartialWithUndefined<{
22
22
  placeholder: string;
23
23
  disabled: boolean;
24
+ isReadonly: boolean;
24
25
  allowedInputs: string | RegExp;
25
26
  blockedInputs: string | RegExp;
26
27
  disableBrowserHelps: boolean;
@@ -97,6 +97,11 @@ export const ViraTextArea = defineViraElement()({
97
97
  pointer-events: none;
98
98
  }
99
99
 
100
+ .readonly-value {
101
+ white-space: pre-wrap;
102
+ overflow-wrap: anywhere;
103
+ }
104
+
100
105
  ${hostClasses['vira-text-area-prevent-resize'].selector} textarea {
101
106
  resize: none;
102
107
  }
@@ -154,6 +159,22 @@ export const ViraTextArea = defineViraElement()({
154
159
  allowed: inputs.allowedInputs,
155
160
  blocked: inputs.blockedInputs,
156
161
  });
162
+ if (inputs.isReadonly) {
163
+ const readonlyValueTemplate = html `
164
+ <span class="readonly-value">${filteredValue}</span>
165
+ `;
166
+ if (inputs.label) {
167
+ return html `
168
+ <label>
169
+ <span class="text-area-label">${inputs.label}</span>
170
+ ${readonlyValueTemplate}
171
+ </label>
172
+ `;
173
+ }
174
+ else {
175
+ return readonlyValueTemplate;
176
+ }
177
+ }
157
178
  const textAreaTemplate = html `
158
179
  <span class="text-area-wrapper">
159
180
  <textarea
@@ -0,0 +1,17 @@
1
+ import { type PartialWithUndefined } from '@augment-vir/common';
2
+ import { ViraThemeClient, ViraThemeSelection } from '../util/vira-theme-client.js';
3
+ /**
4
+ * A row of buttons for selecting a {@link ViraThemeSelection} (light, dark, or auto). Fires a
5
+ * `themeSelect` event when the user picks an option; the consumer is responsible for applying the
6
+ * resulting theme.
7
+ *
8
+ * @category Elements
9
+ */
10
+ export declare const ViraThemeSwitcher: import("element-vir").DeclarativeElementDefinition<"vira-theme-switcher", PartialWithUndefined<{
11
+ themeClient: Readonly<ViraThemeClient>;
12
+ /** Override the default English button titles (used as `title` attributes for tooltips). */
13
+ labels: Readonly<Record<ViraThemeSelection, string>>;
14
+ }>, {
15
+ internalThemeClient: undefined | Readonly<ViraThemeClient>;
16
+ currentTheme: ViraThemeSelection;
17
+ }, {}, "vira-theme-switcher-", "vira-theme-switcher-", readonly [], readonly []>;
@@ -0,0 +1,104 @@
1
+ import { mapEnumToObject } from '@augment-vir/common';
2
+ import { classMap, css, html, listen } from 'element-vir';
3
+ import { AutoTheme24Icon, Moon24Icon, Sun24Icon } from '../icons/index.js';
4
+ import { viraFormCssVars } from '../styles/form-styles.js';
5
+ import { noNativeFormStyles, noNativeSpacing, viraTheme } from '../styles/index.js';
6
+ import { defineViraElement } from '../util/define-vira-element.js';
7
+ import { ViraThemeClient, ViraThemeSelection } from '../util/vira-theme-client.js';
8
+ import { ViraIcon } from './vira-icon.element.js';
9
+ const themeIcons = mapEnumToObject(ViraThemeSelection, (theme) => {
10
+ const map = {
11
+ [ViraThemeSelection.Light]: Sun24Icon,
12
+ [ViraThemeSelection.Dark]: Moon24Icon,
13
+ [ViraThemeSelection.Auto]: AutoTheme24Icon,
14
+ };
15
+ return map[theme];
16
+ });
17
+ const defaultThemeLabels = {
18
+ [ViraThemeSelection.Light]: 'Light',
19
+ [ViraThemeSelection.Dark]: 'Dark',
20
+ [ViraThemeSelection.Auto]: 'Auto',
21
+ };
22
+ /**
23
+ * A row of buttons for selecting a {@link ViraThemeSelection} (light, dark, or auto). Fires a
24
+ * `themeSelect` event when the user picks an option; the consumer is responsible for applying the
25
+ * resulting theme.
26
+ *
27
+ * @category Elements
28
+ */
29
+ export const ViraThemeSwitcher = defineViraElement()({
30
+ tagName: 'vira-theme-switcher',
31
+ styles: css `
32
+ :host {
33
+ display: inline-flex;
34
+ align-items: center;
35
+ box-sizing: border-box;
36
+ gap: 4px;
37
+ }
38
+
39
+ button {
40
+ ${noNativeSpacing};
41
+ ${noNativeFormStyles};
42
+ display: flex;
43
+ align-items: center;
44
+ justify-content: center;
45
+ border: 1px solid transparent;
46
+ border-radius: ${viraFormCssVars['vira-form-radius'].value};
47
+ padding: 2px;
48
+ cursor: pointer;
49
+ color: ${viraTheme.colors['vira-grey-foreground-placeholder'].foreground.value};
50
+
51
+ &:hover {
52
+ color: ${viraFormCssVars['vira-form-accent-primary-hover-color'].value};
53
+ border-color: currentColor;
54
+ }
55
+
56
+ &.selected {
57
+ pointer-events: none;
58
+ color: ${viraFormCssVars['vira-form-accent-primary-color'].value};
59
+ border-color: ${viraFormCssVars['vira-form-accent-primary-color'].value};
60
+ }
61
+ }
62
+
63
+ ${ViraIcon} {
64
+ width: 20px;
65
+ aspect-ratio: 1;
66
+ }
67
+ `,
68
+ state() {
69
+ return {
70
+ internalThemeClient: undefined,
71
+ currentTheme: ViraThemeSelection.Auto,
72
+ };
73
+ },
74
+ render({ inputs, state, updateState }) {
75
+ const themeClient = inputs.themeClient || state.internalThemeClient || new ViraThemeClient();
76
+ updateState({
77
+ internalThemeClient: themeClient,
78
+ currentTheme: themeClient.currentTheme,
79
+ });
80
+ const labels = inputs.labels || defaultThemeLabels;
81
+ return Object.values(ViraThemeSelection).map((theme) => {
82
+ return html `
83
+ <button
84
+ class=${classMap({
85
+ selected: themeClient.currentTheme === theme,
86
+ })}
87
+ title=${labels[theme]}
88
+ ${listen('click', (event) => {
89
+ event.stopPropagation();
90
+ themeClient.setSelectedTheme(theme);
91
+ updateState({
92
+ currentTheme: themeClient.currentTheme,
93
+ });
94
+ })}
95
+ >
96
+ <${ViraIcon.assign({
97
+ icon: themeIcons[theme],
98
+ fitContainer: true,
99
+ })}></${ViraIcon}>
100
+ </button>
101
+ `;
102
+ });
103
+ },
104
+ });
@@ -8,3 +8,4 @@ export * from './shared-text-input-logic.js';
8
8
  export * from './vira-form-fields.js';
9
9
  export * from './vira-json-schema.js';
10
10
  export * from './vira-select-option.js';
11
+ export * from './vira-theme-client.js';
@@ -8,3 +8,4 @@ export * from './shared-text-input-logic.js';
8
8
  export * from './vira-form-fields.js';
9
9
  export * from './vira-json-schema.js';
10
10
  export * from './vira-select-option.js';
11
+ export * from './vira-theme-client.js';
@@ -2,6 +2,7 @@
2
2
  * Creates an observer that monitors whether an element's content overflows its visible width. Uses
3
3
  * a ResizeObserver for size changes and a MutationObserver for DOM content changes.
4
4
  *
5
+ * @category Util
5
6
  * @returns A cleanup function that disconnects all observers.
6
7
  */
7
8
  export declare function createOverflowObserver({ element, widthElement, onChange, hysteresisPx, }: Readonly<{
@@ -2,6 +2,7 @@
2
2
  * Creates an observer that monitors whether an element's content overflows its visible width. Uses
3
3
  * a ResizeObserver for size changes and a MutationObserver for DOM content changes.
4
4
  *
5
+ * @category Util
5
6
  * @returns A cleanup function that disconnects all observers.
6
7
  */
7
8
  export function createOverflowObserver({ element, widthElement, onChange, hysteresisPx = 0, }) {
@@ -12,6 +12,7 @@ export type SharedTextInputElementInputs = {
12
12
  placeholder: string;
13
13
  /** Set to true to trigger disabled styles and to block all user input. */
14
14
  disabled: boolean;
15
+ isReadonly: boolean;
15
16
  /**
16
17
  * Only letters in the given string or matches to the given RegExp will be allowed.
17
18
  * blockedInputs takes precedence over this input.
@@ -77,7 +77,12 @@ export type ViraFormField = ({
77
77
  }> & CommonViraFormFields) | ({
78
78
  type: ViraFormFieldType.Checkbox;
79
79
  value: boolean | undefined;
80
- } & CommonViraFormFields) | ({
80
+ } & PartialWithUndefined<{
81
+ /** The checkbox will be filled with a form selection color when it is checked. */
82
+ fillWhenChecked: boolean;
83
+ /** The checkbox will be filled with a form error color when it is unchecked. */
84
+ fillWhenUnchecked: boolean;
85
+ }> & CommonViraFormFields) | ({
81
86
  type: ViraFormFieldType.Number;
82
87
  value: number | undefined;
83
88
  } & PartialWithUndefined<{
@@ -0,0 +1,83 @@
1
+ import { type MaybePromise, type PartialWithUndefined } from '@augment-vir/common';
2
+ import { LocalStorageClient } from '@electrovir/local-storage-client';
3
+ /**
4
+ * A user-facing selection of which theme to display. `Auto` follows the system color scheme; the
5
+ * other values force a specific theme regardless of system preference.
6
+ *
7
+ * @category Internal
8
+ */
9
+ export declare enum ViraThemeSelection {
10
+ Light = "light",
11
+ Dark = "dark",
12
+ Auto = "auto"
13
+ }
14
+ /**
15
+ * Used by {@link ViraThemeClient} to apply themes. By default, vira themes will be applied (via
16
+ * {@link defaultApplyThemeCallback}).
17
+ *
18
+ * @category Internal
19
+ * @default `defaultApplyThemeCallback`
20
+ */
21
+ export type ApplyThemeCallback = (params: Readonly<{
22
+ useDarkTheme: boolean;
23
+ }>) => MaybePromise<void>;
24
+ /**
25
+ * Default implementation of {@link ApplyThemeCallback}, which simply applies Vira themes.
26
+ *
27
+ * @category Internal
28
+ */
29
+ export declare const defaultApplyThemeCallback: ApplyThemeCallback;
30
+ /**
31
+ * Constructor params for {@link ViraThemeClient}.
32
+ *
33
+ * @category Internal
34
+ */
35
+ export type ViraThemeClientParams = PartialWithUndefined<{
36
+ /**
37
+ * Called whenever the effective theme should change. If not provided, the default Vira themes
38
+ * will be used.
39
+ */
40
+ applyTheme: ApplyThemeCallback;
41
+ /**
42
+ * Override the LocalStorage store name used for theme persistence. Useful if a single page
43
+ * hosts multiple isolated theme clients.
44
+ *
45
+ * @default 'vira-theme'
46
+ */
47
+ storeName: string;
48
+ }>;
49
+ declare const themeStorageShapes: {
50
+ selectedTheme: import("object-shape-tester").Shape<{
51
+ theme: import("object-shape-tester").Shape<import("@sinclair/typebox").TUnion<(import("@sinclair/typebox").TLiteral<ViraThemeSelection.Light> | import("@sinclair/typebox").TLiteral<ViraThemeSelection.Dark> | import("@sinclair/typebox").TLiteral<ViraThemeSelection.Auto>)[]>>;
52
+ }>;
53
+ };
54
+ /**
55
+ * Tracks the user's {@link ViraThemeSelection} and bridges it to a consumer-supplied `applyTheme`
56
+ * callback. Persists the selection in LocalStorage via an internal `LocalStorageClient`, and
57
+ * listens for system color-scheme changes to re-apply the theme in auto mode without persisting.
58
+ *
59
+ * The initial theme is applied during construction.
60
+ *
61
+ * @category Util
62
+ */
63
+ export declare class ViraThemeClient {
64
+ /** The callback that will be called to apply a new theme. */
65
+ protected readonly applyThemeCallback: ApplyThemeCallback;
66
+ /** Contains the user's last selected theme, saving and loading it to disk for persistence. */
67
+ protected readonly localStorageClient: LocalStorageClient<typeof themeStorageShapes>;
68
+ /** A callback to remove the global theme preference listener. */
69
+ protected readonly removeThemePreferenceListener: () => void;
70
+ constructor(params?: Readonly<ViraThemeClientParams>);
71
+ /**
72
+ * The currently selected theme. If you use multiple clients to set the same theme, this might
73
+ * get out of sync.
74
+ */
75
+ get currentTheme(): ViraThemeSelection;
76
+ /** Set the selected theme. */
77
+ setSelectedTheme(selection: ViraThemeSelection): void;
78
+ /** Cleanup internal state and listeners. */
79
+ destroy(): void;
80
+ /** Apply the currently selected theme. */
81
+ protected applySelection(selection: ViraThemeSelection): void;
82
+ }
83
+ export {};
@@ -0,0 +1,93 @@
1
+ import { assert } from '@augment-vir/assert';
2
+ import { LocalStorageClient } from '@electrovir/local-storage-client';
3
+ import { defineShape, enumShape } from 'object-shape-tester';
4
+ import { applyColorThemeViaStyleElement } from 'theme-vir';
5
+ import { listenTo } from 'typed-event-target';
6
+ import { viraTheme, viraThemeDarkOverride } from '../styles/vira-color-theme.js';
7
+ /**
8
+ * A user-facing selection of which theme to display. `Auto` follows the system color scheme; the
9
+ * other values force a specific theme regardless of system preference.
10
+ *
11
+ * @category Internal
12
+ */
13
+ export var ViraThemeSelection;
14
+ (function (ViraThemeSelection) {
15
+ ViraThemeSelection["Light"] = "light";
16
+ ViraThemeSelection["Dark"] = "dark";
17
+ ViraThemeSelection["Auto"] = "auto";
18
+ })(ViraThemeSelection || (ViraThemeSelection = {}));
19
+ /**
20
+ * Default implementation of {@link ApplyThemeCallback}, which simply applies Vira themes.
21
+ *
22
+ * @category Internal
23
+ */
24
+ export const defaultApplyThemeCallback = ({ useDarkTheme }) => {
25
+ applyColorThemeViaStyleElement(viraTheme, useDarkTheme ? viraThemeDarkOverride : undefined);
26
+ };
27
+ const darkSchemeMediaQuery = '(prefers-color-scheme: dark)';
28
+ const themeStorageShapes = {
29
+ selectedTheme: defineShape({
30
+ theme: enumShape(ViraThemeSelection),
31
+ }),
32
+ };
33
+ /**
34
+ * Tracks the user's {@link ViraThemeSelection} and bridges it to a consumer-supplied `applyTheme`
35
+ * callback. Persists the selection in LocalStorage via an internal `LocalStorageClient`, and
36
+ * listens for system color-scheme changes to re-apply the theme in auto mode without persisting.
37
+ *
38
+ * The initial theme is applied during construction.
39
+ *
40
+ * @category Util
41
+ */
42
+ export class ViraThemeClient {
43
+ /** The callback that will be called to apply a new theme. */
44
+ applyThemeCallback = defaultApplyThemeCallback;
45
+ /** Contains the user's last selected theme, saving and loading it to disk for persistence. */
46
+ localStorageClient;
47
+ /** A callback to remove the global theme preference listener. */
48
+ removeThemePreferenceListener = listenTo(globalThis.matchMedia(darkSchemeMediaQuery), 'change', (event) => {
49
+ assert.instanceOf(event, MediaQueryListEvent);
50
+ if (this.currentTheme === ViraThemeSelection.Auto) {
51
+ void this.applyThemeCallback({
52
+ useDarkTheme: event.matches,
53
+ });
54
+ }
55
+ });
56
+ constructor(params = {}) {
57
+ if (params.applyTheme) {
58
+ this.applyThemeCallback = params.applyTheme;
59
+ }
60
+ this.localStorageClient = new LocalStorageClient(themeStorageShapes, {
61
+ storeName: params.storeName || 'vira-theme',
62
+ });
63
+ this.applySelection(this.currentTheme);
64
+ }
65
+ /**
66
+ * The currently selected theme. If you use multiple clients to set the same theme, this might
67
+ * get out of sync.
68
+ */
69
+ get currentTheme() {
70
+ return this.localStorageClient.get.selectedTheme()?.theme || ViraThemeSelection.Auto;
71
+ }
72
+ /** Set the selected theme. */
73
+ setSelectedTheme(selection) {
74
+ this.applySelection(selection);
75
+ this.localStorageClient.set.selectedTheme({
76
+ theme: selection,
77
+ });
78
+ }
79
+ /** Cleanup internal state and listeners. */
80
+ destroy() {
81
+ this.removeThemePreferenceListener();
82
+ this.localStorageClient.destroy();
83
+ }
84
+ /** Apply the currently selected theme. */
85
+ applySelection(selection) {
86
+ const useDarkTheme = selection === ViraThemeSelection.Dark ||
87
+ (selection === ViraThemeSelection.Auto &&
88
+ globalThis.matchMedia(darkSchemeMediaQuery).matches);
89
+ void this.applyThemeCallback({
90
+ useDarkTheme,
91
+ });
92
+ }
93
+ }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "vira",
3
- "version": "31.19.0",
3
+ "version": "31.21.0",
4
4
  "description": "A simple and highly versatile design system using element-vir.",
5
5
  "keywords": [
6
6
  "design",
@@ -42,10 +42,12 @@
42
42
  "@augment-vir/common": "^31.70.1",
43
43
  "@augment-vir/web": "^31.70.1",
44
44
  "@electrovir/color": "^1.7.9",
45
+ "@electrovir/local-storage-client": "^0.1.0",
45
46
  "date-vir": "^8.3.2",
46
47
  "device-navigation": "^4.5.5",
47
48
  "json-schema-to-ts": "^3.1.1",
48
49
  "lit-css-vars": "^3.6.2",
50
+ "object-shape-tester": "^6.13.0",
49
51
  "observavir": "^2.3.2",
50
52
  "page-active": "^1.0.3",
51
53
  "spa-router-vir": "^6.6.0",