Skip to content

Commit 03527c6

Browse files
josephperrottmmalerba
authored andcommitted
fix(form-field): update label gap for outline style (#12555)
1 parent 7e74b5d commit 03527c6

File tree

4 files changed

+45
-40
lines changed

4 files changed

+45
-40
lines changed

src/lib/form-field/BUILD.bazel

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,7 @@ ng_module(
1818
deps = [
1919
"//src/lib/core",
2020
"//src/cdk/coercion",
21+
"//src/cdk/observers",
2122
"//src/cdk/platform",
2223
],
2324
tsconfig = "//src/lib:tsconfig-build.json",

src/lib/form-field/form-field-module.ts

Lines changed: 5 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@
88

99
import {CommonModule} from '@angular/common';
1010
import {NgModule} from '@angular/core';
11+
import {ObserversModule} from '@angular/cdk/observers';
1112
import {MatError} from './error';
1213
import {MatFormField} from './form-field';
1314
import {MatHint} from './hint';
@@ -27,7 +28,10 @@ import {MatSuffix} from './suffix';
2728
MatPrefix,
2829
MatSuffix,
2930
],
30-
imports: [CommonModule],
31+
imports: [
32+
CommonModule,
33+
ObserversModule,
34+
],
3135
exports: [
3236
MatError,
3337
MatFormField,

src/lib/form-field/form-field.html

Lines changed: 5 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -5,13 +5,13 @@
55
<!-- Outline used for outline appearance. -->
66
<ng-container *ngIf="appearance == 'outline'">
77
<div class="mat-form-field-outline">
8-
<div class="mat-form-field-outline-start" [style.width.px]="_outlineGapStart"></div>
9-
<div class="mat-form-field-outline-gap" [style.width.px]="_outlineGapWidth"></div>
8+
<div class="mat-form-field-outline-start"></div>
9+
<div class="mat-form-field-outline-gap"></div>
1010
<div class="mat-form-field-outline-end"></div>
1111
</div>
1212
<div class="mat-form-field-outline mat-form-field-outline-thick">
13-
<div class="mat-form-field-outline-start" [style.width.px]="_outlineGapStart"></div>
14-
<div class="mat-form-field-outline-gap" [style.width.px]="_outlineGapWidth"></div>
13+
<div class="mat-form-field-outline-start"></div>
14+
<div class="mat-form-field-outline-gap"></div>
1515
<div class="mat-form-field-outline-end"></div>
1616
</div>
1717
</ng-container>
@@ -27,6 +27,7 @@
2727
<!-- We add aria-owns as a workaround for an issue in JAWS & NVDA where the label isn't
2828
read if it comes before the control in the DOM. -->
2929
<label class="mat-form-field-label"
30+
(cdkObserveContent)="updateOutlineGap()"
3031
[id]="_labelId"
3132
[attr.for]="_control.id"
3233
[attr.aria-owns]="_control.id"

src/lib/form-field/form-field.ts

Lines changed: 34 additions & 35 deletions
Original file line numberDiff line numberDiff line change
@@ -148,12 +148,18 @@ export class MatFormField extends _MatFormFieldMixinBase
148148
return this._appearance || this._defaultOptions && this._defaultOptions.appearance || 'legacy';
149149
}
150150
set appearance(value: MatFormFieldAppearance) {
151-
// If we're switching to `outline` from another appearance, we have to recalculate the gap.
152-
if (value !== this._appearance && value === 'outline') {
153-
this._initialGapCalculated = false;
154-
}
155-
151+
const oldValue = this._appearance;
156152
this._appearance = value;
153+
if (this._appearance === 'outline' && oldValue !== value) {
154+
// @breaking-change 7.0.0 Remove this check and else block once _ngZone is required.
155+
if (this._ngZone) {
156+
this._ngZone!.onStable.pipe(take(1)).subscribe(() => {
157+
this._ngZone!.runOutsideAngular(() => this.updateOutlineGap());
158+
});
159+
} else {
160+
Promise.resolve().then(() => this.updateOutlineGap());
161+
}
162+
}
157163
}
158164
_appearance: MatFormFieldAppearance;
159165

@@ -217,17 +223,13 @@ export class MatFormField extends _MatFormFieldMixinBase
217223
/** Whether the Angular animations are enabled. */
218224
_animationsEnabled: boolean;
219225

220-
_outlineGapWidth = 0;
221-
_outlineGapStart = 0;
222-
_initialGapCalculated = false;
223-
224226
/**
225227
* @deprecated
226228
* @breaking-change 7.0.0
227229
*/
228230
@ViewChild('underline') underlineRef: ElementRef;
229231

230-
@ViewChild('connectionContainer') _connectionContainerRef: ElementRef;
232+
@ViewChild('connectionContainer') _connectionContainerRef: ElementRef<HTMLElement>;
231233
@ViewChild('inputContainer') _inputContainerRef: ElementRef;
232234
@ViewChild('label') private _label: ElementRef;
233235
@ContentChild(MatFormFieldControl) _control: MatFormFieldControl<any>;
@@ -298,20 +300,6 @@ export class MatFormField extends _MatFormFieldMixinBase
298300

299301
ngAfterContentChecked() {
300302
this._validateControlChild();
301-
302-
if (!this._initialGapCalculated) {
303-
// @breaking-change 7.0.0 Remove this check and else block once _ngZone is required.
304-
if (this._ngZone) {
305-
// It's important that we run this outside the `_ngZone`, because the `Promise.resolve`
306-
// can kick us into an infinite change detection loop, if the `_initialGapCalculated`
307-
// wasn't flipped on for some reason.
308-
this._ngZone.runOutsideAngular(() => {
309-
Promise.resolve().then(() => this.updateOutlineGap());
310-
});
311-
} else {
312-
Promise.resolve().then(() => this.updateOutlineGap());
313-
}
314-
}
315303
}
316304

317305
ngAfterViewInit() {
@@ -422,9 +410,9 @@ export class MatFormField extends _MatFormFieldMixinBase
422410
let ids: string[] = [];
423411

424412
if (this._getDisplayedMessages() === 'hint') {
425-
let startHint = this._hintChildren ?
413+
const startHint = this._hintChildren ?
426414
this._hintChildren.find(hint => hint.align === 'start') : null;
427-
let endHint = this._hintChildren ?
415+
const endHint = this._hintChildren ?
428416
this._hintChildren.find(hint => hint.align === 'end') : null;
429417

430418
if (startHint) {
@@ -456,10 +444,19 @@ export class MatFormField extends _MatFormFieldMixinBase
456444
* appearance.
457445
*/
458446
updateOutlineGap() {
459-
if (this.appearance === 'outline' && this._label && this._label.nativeElement.children.length) {
447+
if (this.appearance !== 'outline') {
448+
return;
449+
}
450+
451+
let startWidth = 0;
452+
let gapWidth = 0;
453+
const startEls = this._connectionContainerRef.nativeElement.querySelectorAll<HTMLElement>(
454+
'.mat-form-field-outline-start');
455+
const gapEls = this._connectionContainerRef.nativeElement.querySelectorAll<HTMLElement>(
456+
'.mat-form-field-outline-gap');
457+
if (this._label && this._label.nativeElement.children.length) {
460458
if (this._platform && !this._platform.isBrowser) {
461459
// getBoundingClientRect isn't available on the server.
462-
this._initialGapCalculated = true;
463460
return;
464461
}
465462
if (!document.documentElement.contains(this._elementRef.nativeElement)) {
@@ -474,14 +471,16 @@ export class MatFormField extends _MatFormFieldMixinBase
474471
for (const child of this._label.nativeElement.children) {
475472
labelWidth += child.offsetWidth;
476473
}
477-
this._outlineGapStart = labelStart - containerStart - outlineGapPadding;
478-
this._outlineGapWidth = labelWidth * floatingLabelScale + outlineGapPadding * 2;
479-
} else {
480-
this._outlineGapStart = 0;
481-
this._outlineGapWidth = 0;
474+
startWidth = labelStart - containerStart - outlineGapPadding;
475+
gapWidth = labelWidth * floatingLabelScale + outlineGapPadding * 2;
476+
}
477+
478+
for (let i = 0; i < startEls.length; i++) {
479+
startEls.item(i).style.width = `${startWidth}px`;
480+
}
481+
for (let i = 0; i < gapEls.length; i++) {
482+
gapEls.item(i).style.width = `${gapWidth}px`;
482483
}
483-
this._initialGapCalculated = true;
484-
this._changeDetectorRef.markForCheck();
485484
}
486485

487486
/** Gets the start end of the rect considering the current directionality. */

0 commit comments

Comments
 (0)