Skip to content

Commit

Permalink
fix(web-components): add current-value attribute to TextInput compo…
Browse files Browse the repository at this point in the history
…nent (#33144)
  • Loading branch information
radium-v authored Dec 5, 2024
1 parent f42240c commit df2eb61
Show file tree
Hide file tree
Showing 4 changed files with 115 additions and 10 deletions.
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
{
"type": "prerelease",
"comment": "add current-value attribute to text-input",
"packageName": "@fluentui/web-components",
"email": "[email protected]",
"dependentChangeType": "patch"
}
3 changes: 3 additions & 0 deletions packages/web-components/docs/api-report.md
Original file line number Diff line number Diff line change
Expand Up @@ -792,6 +792,9 @@ export class BaseTextInput extends FASTElement {
control: HTMLInputElement;
// @internal
controlLabel: HTMLLabelElement;
currentValue: string;
// @internal
currentValueChanged(prev: string, next: string): void;
// @internal
defaultSlottedNodes: Node[];
// @internal
Expand Down
81 changes: 81 additions & 0 deletions packages/web-components/src/text-input/text-input.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -833,4 +833,85 @@ test.describe('TextInput', () => {
await expect(control).toBeFocused();
});
});

test('should reset the value to an empty string when the form is reset', async ({ page }) => {
const element = page.locator('fluent-text-input');
const control = element.locator('input');
const reset = page.locator('button');

await page.setContent(/* html */ `
<form id="form" action="foo">
<fluent-text-input name="testinput"></fluent-text-input>
<button type="reset">Reset</button>
</form>
`);

await expect(control).toHaveValue('');

await control.fill('hello');

await reset.click();

await expect(control).toHaveValue('');
});

test('should change the `value` property when the `current-value` attribute changes', async ({ page }) => {
const element = page.locator('fluent-text-input');

await page.setContent(/* html */ `
<fluent-text-input></fluent-text-input>
`);

await element.evaluate(node => {
node.setAttribute('current-value', 'foo');
});

await expect(element).toHaveJSProperty('value', 'foo');
});

test('should change the `value` property when the `currentValue` property changes', async ({ page }) => {
const element = page.locator('fluent-text-input');

await page.setContent(/* html */ `
<fluent-text-input></fluent-text-input>
`);

await element.evaluate((node: TextInput) => {
node.currentValue = 'foo';
});

await expect(element).toHaveJSProperty('value', 'foo');
});

test('should set the `current-value` attribute to match the `value` property', async ({ page }) => {
const element = page.locator('fluent-text-input');

await page.setContent(/* html */ `
<fluent-text-input></fluent-text-input>
`);

await expect(element).not.toHaveAttribute('current-value');

await element.evaluate((node: TextInput) => {
node.value = 'foo';
});

await expect(element).toHaveAttribute('current-value', 'foo');
});

test('should set the `currentValue` property to match the `value` property', async ({ page }) => {
const element = page.locator('fluent-text-input');

await page.setContent(/* html */ `
<fluent-text-input></fluent-text-input>
`);

await expect(element).toHaveJSProperty('currentValue', undefined);

await element.evaluate((node: TextInput) => {
node.value = 'foo';
});

await expect(element).toHaveJSProperty('currentValue', 'foo');
});
});
34 changes: 24 additions & 10 deletions packages/web-components/src/text-input/text-input.ts
Original file line number Diff line number Diff line change
Expand Up @@ -44,6 +44,27 @@ export class BaseTextInput extends FASTElement {
@attr({ mode: 'boolean' })
public autofocus!: boolean;

/**
* The current value of the input.
* @public
* @remarks
* HTML Attribute: `current-value`
*/
@attr({ attribute: 'current-value' })
public currentValue!: string;

/**
* Tracks the current value of the input.
*
* @param prev - the previous value
* @param next - the next value
*
* @internal
*/
currentValueChanged(prev: string, next: string): void {
this.value = next;
}

/**
* The default slotted content. This is the content that appears in the text field label.
*
Expand Down Expand Up @@ -274,13 +295,6 @@ export class BaseTextInput extends FASTElement {
@attr
public type: TextInputType = TextInputType.text;

/**
* The current value of the input.
*
* @internal
*/
private _value: string = this.initialValue;

/**
* A reference to the internal input element.
*
Expand Down Expand Up @@ -346,14 +360,14 @@ export class BaseTextInput extends FASTElement {
*/
public get value(): string {
Observable.track(this, 'value');
return this._value;
return this.currentValue;
}

public set value(value: string) {
this._value = value;
this.currentValue = value;

if (this.$fastController.isConnected) {
this.control.value = value;
this.control.value = value ?? '';
this.setFormValue(value);
this.setValidity();
Observable.notify(this, 'value');
Expand Down

0 comments on commit df2eb61

Please sign in to comment.