Skip to content

Commit 3f84826

Browse files
committed
fix(form-field): update label gap for outline style
1 parent 3a84259 commit 3f84826

File tree

4 files changed

+34
-38
lines changed

4 files changed

+34
-38
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: 23 additions & 33 deletions
Original file line numberDiff line numberDiff line change
@@ -21,7 +21,6 @@ import {
2121
Inject,
2222
InjectionToken,
2323
Input,
24-
NgZone,
2524
Optional,
2625
QueryList,
2726
ViewChild,
@@ -148,11 +147,6 @@ export class MatFormField extends _MatFormFieldMixinBase
148147
return this._appearance || this._defaultOptions && this._defaultOptions.appearance || 'legacy';
149148
}
150149
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-
156150
this._appearance = value;
157151
}
158152
_appearance: MatFormFieldAppearance;
@@ -219,15 +213,14 @@ export class MatFormField extends _MatFormFieldMixinBase
219213

220214
_outlineGapWidth = 0;
221215
_outlineGapStart = 0;
222-
_initialGapCalculated = false;
223216

224217
/**
225218
* @deprecated
226219
* @breaking-change 7.0.0
227220
*/
228221
@ViewChild('underline') underlineRef: ElementRef;
229222

230-
@ViewChild('connectionContainer') _connectionContainerRef: ElementRef;
223+
@ViewChild('connectionContainer') _connectionContainerRef: ElementRef<HTMLDivElement>;
231224
@ViewChild('inputContainer') _inputContainerRef: ElementRef;
232225
@ViewChild('label') private _label: ElementRef;
233226
@ContentChild(MatFormFieldControl) _control: MatFormFieldControl<any>;
@@ -245,9 +238,8 @@ export class MatFormField extends _MatFormFieldMixinBase
245238
@Optional() private _dir: Directionality,
246239
@Optional() @Inject(MAT_FORM_FIELD_DEFAULT_OPTIONS) private _defaultOptions:
247240
MatFormFieldDefaultOptions,
248-
// @breaking-change 7.0.0 _platform, _ngZone and _animationMode to be made required.
241+
// @breaking-change 7.0.0 _platform and _animationMode to be made required.
249242
private _platform?: Platform,
250-
private _ngZone?: NgZone,
251243
@Optional() @Inject(ANIMATION_MODULE_TYPE) _animationMode?: string) {
252244
super(_elementRef);
253245

@@ -298,20 +290,7 @@ export class MatFormField extends _MatFormFieldMixinBase
298290

299291
ngAfterContentChecked() {
300292
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-
}
293+
this.updateOutlineGap();
315294
}
316295

317296
ngAfterViewInit() {
@@ -456,10 +435,19 @@ export class MatFormField extends _MatFormFieldMixinBase
456435
* appearance.
457436
*/
458437
updateOutlineGap() {
459-
if (this.appearance === 'outline' && this._label && this._label.nativeElement.children.length) {
438+
if (this.appearance !== 'outline') {
439+
return;
440+
}
441+
442+
let startWidth = 0;
443+
let gapWidth = 0;
444+
const startEls = this._connectionContainerRef.nativeElement.querySelectorAll<HTMLDivElement>(
445+
'.mat-form-field-outline > .mat-form-field-outline-start');
446+
const gapEls = this._connectionContainerRef.nativeElement.querySelectorAll<HTMLDivElement>(
447+
'.mat-form-field-outline > .mat-form-field-outline-gap');
448+
if (this._label && this._label.nativeElement.children.length) {
460449
if (this._platform && !this._platform.isBrowser) {
461450
// getBoundingClientRect isn't available on the server.
462-
this._initialGapCalculated = true;
463451
return;
464452
}
465453
if (!document.documentElement.contains(this._elementRef.nativeElement)) {
@@ -474,14 +462,16 @@ export class MatFormField extends _MatFormFieldMixinBase
474462
for (const child of this._label.nativeElement.children) {
475463
labelWidth += child.offsetWidth;
476464
}
477-
this._outlineGapStart = labelStart - containerStart - outlineGapPadding;
478-
this._outlineGapWidth = labelWidth * floatingLabelScale + outlineGapPadding * 2;
479-
} else {
480-
this._outlineGapStart = 0;
481-
this._outlineGapWidth = 0;
465+
startWidth = labelStart - containerStart - outlineGapPadding;
466+
gapWidth = labelWidth * floatingLabelScale + outlineGapPadding * 2;
467+
}
468+
469+
for (let i = 0; i < startEls.length; i++) {
470+
startEls.item(i).style.width = `${startWidth}px`;
471+
}
472+
for (let i = 0; i < gapEls.length; i++) {
473+
gapEls.item(i).style.width = `${gapWidth}px`;
482474
}
483-
this._initialGapCalculated = true;
484-
this._changeDetectorRef.markForCheck();
485475
}
486476

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

0 commit comments

Comments
 (0)