-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsuper-text-input.js
383 lines (326 loc) · 11.5 KB
/
super-text-input.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
/**
* super text input for Home Assistant
* Provides a customizable text input field with optional buttons and r/t updates
*/
// note the super hacky way of restoring focus in r/t mode. HELP!
// Import utilities and constants from card-utils.js
import {
CARD_HEIGHT,
DEFAULT_PADDING,
DEFAULT_BUTTON_STYLE,
ButtonFactory,
handleAction,
debounce,
computeStateName,
} from "./card-utils.js";
// Add this new import
import "./editor.js";
// Get LitElement base class from Home Assistant frontend
const LitElement = customElements.get("home-assistant-main")
? Object.getPrototypeOf(customElements.get("home-assistant-main"))
: Object.getPrototypeOf(customElements.get("hui-view"));
const html = LitElement.prototype.html;
const css = LitElement.prototype.css;
class SuperTextInput extends LitElement {
static getConfigElement() {
return document.createElement("super-text-input-editor");
}
/**
* Define reactive properties
*/
static get properties() {
return {
label: { type: String },
value: { type: String },
minlength: { type: Number },
maxlength: { type: Number },
pattern: { type: String },
mode: { type: String },
stateObj: { type: Object },
_config: { type: Object },
_lastUpdate: { type: Number },
};
}
// Add to static constants
static DEFAULT_FOCUS_DELAY = 100;
// default initial values
static DEFAULT_CONFIG = {
entity: "",
name: "",
label: "",
placeholder: "",
update_mode: "blur",
debounce_time: 1000,
};
// Text field style constants
static TEXT_FIELD_STYLES = {
width: "100%",
height: "36px",
marginLeft: "0px",
marginRight: "0px",
offsetLeftMargin: "8px",
defaultTextLeftPadding: "6px",
defaultTextBottomMargin: "-3px",
};
// Add these new static methods here
static getConfigElement() {
return document.createElement("super-text-input-editor");
}
static getStubConfig() {
return SuperTextInput.DEFAULT_CONFIG;
}
static DEFAULT_DEBOUNCE_TIME = 1000;
_internalChange = false;
/**
* Initialize default values
*/
constructor() {
super();
this.label = "";
this.value = "";
this.minlength = 0;
this.maxlength = Infinity;
this.pattern = "";
this.mode = "";
this.stateObj = null;
this._config = null;
}
/**
* Set up card configuration
* @param {Object} config - Card configuration object
*/
setConfig(config) {
this._config = {
...SuperTextInput.DEFAULT_CONFIG,
...config,
};
if (config.entity) {
this._entityType = config.entity.split(".")[0];
}
this._debounce_time = config.debounce_time || SuperTextInput.DEFAULT_DEBOUNCE_TIME;
this._update_mode = config.update_mode || "blur";
if (config.buttons) {
this._buttonFactory = new ButtonFactory(config, this);
}
if (this._update_mode === "realtime") {
this._debouncedUpdate = debounce((value) => {
this.setValue(value);
}, this._debounce_time);
}
}
/**
* Handle Home Assistant state updates
* @param {Object} hass - Home Assistant instance
*/
set hass(hass) {
// Store current value to detect changes
const oldValue = this.value;
// Store Home Assistant object reference
this._hass = hass;
// Get the current state of our entity
this.stateObj = hass.states[this._config.entity];
if (this.stateObj) {
// Update all entity-based properties
this.value = this.stateObj.state;
this.minlength = this.stateObj.attributes.min;
this.maxlength = this.stateObj.attributes.max;
this.pattern = this.stateObj.attributes.pattern;
this.mode = this.stateObj.attributes.mode;
// Set name to entity friendly name if not configured
this._config.name = this._config.name || computeStateName(this.stateObj, this._config.entity);
// Set label to configured label or name
this.label = this._config.label || this._config.name;
// Handle external value changes and trigger change actions
// Only if change wasn't triggered internally and value actually changed
if (!this._internalChange && oldValue !== this.value && this._config.change_action) {
handleAction(this._config.change_action, this.value, this);
}
// Reset internal change flag
this._internalChange = false;
}
}
/**
* Apply styles to the card container
* @param {HTMLElement} card - Card element to style
*/
_getCardStyles(card) {
const cardStyle = this._config.style?.card || {};
card.style.display = "flex";
card.style.flexDirection = "row";
card.style.alignItems = "center";
card.style.height = cardStyle.height || CARD_HEIGHT;
card.style.padding = cardStyle.padding || DEFAULT_PADDING;
if (cardStyle.background) card.style.background = cardStyle.background;
if (cardStyle["border-radius"]) card.style.borderRadius = cardStyle["border-radius"];
if (cardStyle.border) card.style.border = cardStyle.border;
}
/**
* Apply styles to the text field
* @param {HTMLElement} textField - Text field element to style
*/
_getTextFieldStyles(textField) {
const style = this._config.style?.editor || {};
const hasLeadingButtons = this._config.buttons?.some((btn) => !btn.position || btn.position === "start");
const styles = { ...SuperTextInput.TEXT_FIELD_STYLES };
if (hasLeadingButtons) {
styles.marginLeft = styles.offsetLeftMargin;
}
styles.marginLeft = style["margin-left"] || styles.marginLeft;
styles.marginRight = style["margin-right"] || styles.marginRight;
Object.assign(textField.style, styles);
textField.updateComplete.then(() => {
const mdcTextField = textField.shadowRoot.querySelector(".mdc-text-field");
if (mdcTextField) {
mdcTextField.style.height = styles.height;
if (style.background) mdcTextField.style.background = style.background;
mdcTextField.style.paddingLeft = style["padding-left"] || styles.defaultTextLeftPadding;
}
const input = textField.shadowRoot.querySelector(".mdc-text-field__input");
if (input) {
input.style.alignSelf = "end";
input.style.marginBottom = style["margin-bottom"] || styles.defaultTextBottomMargin;
}
const label = textField.shadowRoot.querySelector(".mdc-floating-label");
if (label) {
label.style.setProperty("left", style["padding-left"] || styles.defaultTextLeftPadding, "important");
}
});
}
/**
* Create the text field element
* @returns {HTMLElement} Configured text field
*/
_createTextField() {
// Create Home Assistant's material design text field
const textField = document.createElement("ha-textfield");
// Set up basic field properties from component state
textField.label = this.label; // Display label above input
textField.value = this.value; // Current input value
textField.minlength = this.minlength; // Minimum text length validation
textField.maxlength = this.maxlength; // Maximum text length validation
textField.autoValidate = this.pattern; // Enable pattern validation
textField.pattern = this.pattern; // Regex pattern for validation
textField.type = this.mode; // Input type (text, password, etc)
textField.id = "textinput"; // ID for DOM queries
textField.placeholder = this._config.placeholder || ""; // Placeholder text
// Event Listeners for value changes:
// 'change' fires when focus leaves the field (blur)
textField.addEventListener("change", this.valueChanged.bind(this));
// 'input' fires on every keystroke for real-time updates
textField.addEventListener("input", this.inputChanged.bind(this));
// Apply configured styles to the field
this._getTextFieldStyles(textField);
// Focus Management:
// This code maintains cursor position and typing flow during real-time updates
if (this._update_mode === "realtime" && this._lastUpdate && Date.now() - this._lastUpdate < 1000) {
// Three-part check ensures optimal focus handling:
//
// 1. Real-time Mode Check (this._update_mode === "realtime")
// - Only needed during immediate keystroke updates
// - Blur mode updates happen after focus is lost, so no restoration needed
//
// 2. Update Timestamp Check (this._lastUpdate)
// - Verifies we have actually performed an update
// - Prevents unnecessary focus management on initial render
// - Timestamp is set in setValue() during real-time updates
//
// 3. Time Window Check (Date.now() - this._lastUpdate < 1000)
// - Creates 1-second window for focus restoration
// - Matches natural typing rhythm and update cycles
// - Prevents focus jumps during non-typing interactions
//
// Focus Restoration (setTimeout):
// - 100ms delay ensures DOM stability after updates
// - Arrow function maintains correct 'this' context
// - Returns cursor to input field for uninterrupted typing
// this._config.forced_focus_delay is an advanced prop to adjust focus delay
setTimeout(() => textField.focus(), this._config.forced_focus_delay || SuperTextInput.DEFAULT_FOCUS_DELAY);
}
return textField;
}
/**
* Render the card
* @returns {TemplateResult} Card template
*/
render() {
const card = document.createElement("ha-card");
this._getCardStyles(card);
console.log("rendering-card");
// create the leading buttons, if any
if (this._config.buttons) {
card.appendChild(this._buttonFactory.createButtonContainer(false, this._config.buttons));
}
// create the text field
card.appendChild(this._createTextField());
// create the trailing buttons, if any
if (this._config.buttons) {
card.appendChild(this._buttonFactory.createButtonContainer(true, this._config.buttons));
}
return html`${card}`;
}
/**
* Updates the entity value in Home Assistant and handles related actions
* @param {string} value - The new value to set
*/
setValue(value) {
// Only proceed if the value has actually changed
if (this.stateObj.state !== value) {
// Flag to prevent feedback loop when value updates come back from HA
this._internalChange = true;
// Call Home Assistant service to update the entity
// Uses the entity type (input_text, text, etc) to determine correct service
this._hass.callService(this._entityType, "set_value", {
entity_id: this._config.entity,
value: value,
});
// If a change action is configured (like a script or service call)
// trigger it with the new value
if (this._config.change_action) {
handleAction(this._config.change_action, value, this);
}
// In realtime mode, track when the last update occurred
// Used to force focus after updates
if (this._update_mode === "realtime") {
this._lastUpdate = Date.now();
}
}
}
/**
* Handles real-time input changes and debouncing
* @param {Event} ev - Input event from text field
*/
inputChanged(ev) {
// Only process changes in realtime mode
if (this._update_mode !== "realtime") return;
// Get the current input value
const value = ev.target.value;
// Empty values are updated immediately
if (value === "") {
this.setValue(value);
} else {
// Non-empty values use debounced update to prevent too frequent updates
this._debouncedUpdate(value);
}
}
/**
* Handle value changes on blur
* @param {Event} ev - Change event
*/
valueChanged(ev) {
const value = this.shadowRoot.querySelector("#textinput").value;
this.setValue(value);
}
}
// Register the custom element and editor xx
customElements.define("super-text-input", SuperTextInput);
// Register card for UI editor
window.customCards = window.customCards || [];
window.customCards.push({
type: "super-text-input",
name: "Super Text Input",
description: "A text input card with enhanced features - such as real-time input, icons, buttons and actions",
preview: "/local/community/super-text-input/preview.png",
configurable: true,
version: "0.1.0",
customElement: true,
});