docs(template-syntax): ngClass and ngStyle binding examples not horrible (#3076)
closes #3072, whose criticism prompted these changes.
This commit is contained in:
parent
aff39d20a9
commit
48fb6adc42
|
@ -409,15 +409,23 @@ bindon-ngModel
|
||||||
<!-- NgClass binding -->
|
<!-- NgClass binding -->
|
||||||
<hr><h2 id="ngClass">NgClass Binding</h2>
|
<hr><h2 id="ngClass">NgClass Binding</h2>
|
||||||
|
|
||||||
<p>setClasses returns {{setClasses() | json}}</p>
|
<p>currentClasses returns {{currentClasses | json}}</p>
|
||||||
<!-- #docregion NgClass-1 -->
|
<!-- #docregion NgClass-1 -->
|
||||||
<div [ngClass]="setClasses()">This div is saveable and special</div>
|
<div [ngClass]="currentClasses">This div is initially saveable, unchanged, and special</div>
|
||||||
<!-- #enddocregion NgClass-1 -->
|
<!-- #enddocregion NgClass-1 -->
|
||||||
<div [ngClass]="setClasses()" #classDiv>
|
|
||||||
After setClasses(), the classes are "{{classDiv.className}}"
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- not used in chapter -->
|
<!-- not used in chapter -->
|
||||||
|
<br>
|
||||||
|
<label>saveable <input type="checkbox" [(ngModel)]="canSave"></label> |
|
||||||
|
<label>modified: <input type="checkbox" [value]="!isUnchanged" (change)="isUnchanged=!isUnchanged"></label> |
|
||||||
|
<label>special: <input type="checkbox" [(ngModel)]="isSpecial"></label>
|
||||||
|
<button (click)="setCurrentClasses()">Refresh currentClasses</button>
|
||||||
|
<br><br>
|
||||||
|
<div [ngClass]="currentClasses">
|
||||||
|
This div should be {{ canSave ? "": "not"}} saveable,
|
||||||
|
{{ isUnchanged ? "unchanged" : "modified" }} and,
|
||||||
|
{{ isSpecial ? "": "not"}} special after clicking "refresh".</div>
|
||||||
|
<br><br>
|
||||||
|
|
||||||
<div [ngClass]="isSpecial ? 'special' : ''">This div is special</div>
|
<div [ngClass]="isSpecial ? 'special' : ''">This div is special</div>
|
||||||
|
|
||||||
|
@ -429,38 +437,32 @@ bindon-ngModel
|
||||||
<!-- NgStyle binding -->
|
<!-- NgStyle binding -->
|
||||||
<hr><h2 id="ngStyle">NgStyle Binding</h2>
|
<hr><h2 id="ngStyle">NgStyle Binding</h2>
|
||||||
|
|
||||||
<!-- #docregion NgStyle -->
|
|
||||||
<div>
|
|
||||||
<p [ngStyle]="setStyle()" #styleP>Change style of this text!</p>
|
|
||||||
|
|
||||||
<label>Italic: <input type="checkbox" [(ngModel)]="isItalic"></label> |
|
|
||||||
<label>Bold: <input type="checkbox" [(ngModel)]="isBold"></label> |
|
|
||||||
<label>Size: <input type="text" [(ngModel)]="fontSize"></label>
|
|
||||||
|
|
||||||
<p>Style set to: <code>'{{styleP.style.cssText}}'</code></p>
|
|
||||||
</div>
|
|
||||||
<!-- #enddocregion NgStyle -->
|
|
||||||
|
|
||||||
<!-- #docregion NgStyle-1 -->
|
<!-- #docregion NgStyle-1 -->
|
||||||
<div [style.font-size]="isSpecial ? 'x-large' : 'smaller'" >
|
<div [style.font-size]="isSpecial ? 'x-large' : 'smaller'" >
|
||||||
This div is x-large.
|
This div is x-large.
|
||||||
</div>
|
</div>
|
||||||
<!-- #enddocregion NgStyle-1 -->
|
<!-- #enddocregion NgStyle-1 -->
|
||||||
|
|
||||||
<h3>Use setStyles() - CSS property names</h3>
|
<h3>[ngStyle] binding to `currentStyles` - CSS property names</h3>
|
||||||
<p>setStyles returns {{setStyles() | json}}</p>
|
<p>currentStyles returns {{currentStyles | json}}</p>
|
||||||
<!-- #docregion NgStyle-2 -->
|
<!-- #docregion NgStyle-2 -->
|
||||||
<div [ngStyle]="setStyles()">
|
<div [ngStyle]="currentStyles">
|
||||||
This div is italic, normal weight, and extra large (24px).
|
This div is initially italic, normal weight, and extra large (24px).
|
||||||
</div>
|
</div>
|
||||||
<!-- #enddocregion NgStyle-2 -->
|
<!-- #enddocregion NgStyle-2 -->
|
||||||
<p>After setStyles(), the DOM confirms that the styles are
|
|
||||||
<span [ngStyle]="setStyles()" #styleDiv>
|
|
||||||
{{getStyles(styleDiv)}}
|
|
||||||
</span>.
|
|
||||||
</p>
|
|
||||||
|
|
||||||
<!-- not used in chapter -->
|
<!-- not used in chapter -->
|
||||||
|
<br>
|
||||||
|
<label>italic: <input type="checkbox" [(ngModel)]="canSave"></label> |
|
||||||
|
<label>normal: <input type="checkbox" [(ngModel)]="isUnchanged"></label> |
|
||||||
|
<label>xlarge: <input type="checkbox" [(ngModel)]="isSpecial"></label>
|
||||||
|
<button (click)="setCurrentStyles()">Refresh currentStyles</button>
|
||||||
|
<br><br>
|
||||||
|
<div [ngStyle]="currentStyles">
|
||||||
|
This div should be {{ canSave ? "italic": "plain"}},
|
||||||
|
{{ isUnchanged ? "normal weight" : "bold" }} and,
|
||||||
|
{{ isSpecial ? "extra large": "normal size"}} after clicking "refresh".</div>
|
||||||
|
<br>
|
||||||
|
|
||||||
<a class="to-toc" href="#toc">top</a>
|
<a class="to-toc" href="#toc">top</a>
|
||||||
|
|
||||||
|
|
|
@ -25,6 +25,8 @@ export class AppComponent implements AfterViewInit, OnInit {
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
this.refreshHeroes();
|
this.refreshHeroes();
|
||||||
|
this.setCurrentClasses();
|
||||||
|
this.setCurrentStyles();
|
||||||
}
|
}
|
||||||
|
|
||||||
ngAfterViewInit() {
|
ngAfterViewInit() {
|
||||||
|
@ -56,14 +58,10 @@ export class AppComponent implements AfterViewInit, OnInit {
|
||||||
|
|
||||||
title = 'Template Syntax';
|
title = 'Template Syntax';
|
||||||
|
|
||||||
// DevMode memoization fields
|
|
||||||
private priorClasses: {};
|
|
||||||
private _priorStyles: {};
|
|
||||||
|
|
||||||
getStyles(el: Element) {
|
getStyles(el: Element) {
|
||||||
let styles = window.getComputedStyle(el);
|
let styles = window.getComputedStyle(el);
|
||||||
let showStyles = {};
|
let showStyles = {};
|
||||||
for (let p in this.setStyles()) {
|
for (let p in this.currentStyles) { // only interested in these styles
|
||||||
showStyles[p] = styles[p];
|
showStyles[p] = styles[p];
|
||||||
}
|
}
|
||||||
return JSON.stringify(showStyles);
|
return JSON.stringify(showStyles);
|
||||||
|
@ -141,57 +139,28 @@ export class AppComponent implements AfterViewInit, OnInit {
|
||||||
}
|
}
|
||||||
|
|
||||||
// #docregion setClasses
|
// #docregion setClasses
|
||||||
setClasses() {
|
currentClasses: {};
|
||||||
let classes = {
|
setCurrentClasses() {
|
||||||
saveable: this.canSave, // true
|
// CSS classes: added/removed per current state of component properties
|
||||||
modified: !this.isUnchanged, // false
|
this.currentClasses = {
|
||||||
special: this.isSpecial, // true
|
saveable: this.canSave,
|
||||||
|
modified: !this.isUnchanged,
|
||||||
|
special: this.isSpecial
|
||||||
};
|
};
|
||||||
// #enddocregion setClasses
|
|
||||||
// compensate for DevMode (sigh)
|
|
||||||
if (JSON.stringify(classes) === JSON.stringify(this.priorClasses)) {
|
|
||||||
return this.priorClasses;
|
|
||||||
}
|
|
||||||
this.priorClasses = classes;
|
|
||||||
// #docregion setClasses
|
|
||||||
return classes;
|
|
||||||
}
|
}
|
||||||
// #enddocregion setClasses
|
// #enddocregion setClasses
|
||||||
|
|
||||||
|
|
||||||
// #docregion setStyles
|
// #docregion setStyles
|
||||||
setStyles() {
|
currentStyles: {};
|
||||||
let styles = {
|
setCurrentStyles() {
|
||||||
// CSS property names
|
this.currentStyles = {
|
||||||
'font-style': this.canSave ? 'italic' : 'normal', // italic
|
// CSS styles: set per current state of component properties
|
||||||
'font-weight': !this.isUnchanged ? 'bold' : 'normal', // normal
|
'font-style': this.canSave ? 'italic' : 'normal',
|
||||||
'font-size': this.isSpecial ? '24px' : '8px', // 24px
|
'font-weight': !this.isUnchanged ? 'bold' : 'normal',
|
||||||
};
|
'font-size': this.isSpecial ? '24px' : '12px'
|
||||||
// #enddocregion setStyles
|
|
||||||
// compensate for DevMode (sigh)
|
|
||||||
if (JSON.stringify(styles) === JSON.stringify(this._priorStyles)) {
|
|
||||||
return this._priorStyles;
|
|
||||||
}
|
|
||||||
this._priorStyles = styles;
|
|
||||||
// #docregion setStyles
|
|
||||||
return styles;
|
|
||||||
}
|
|
||||||
// #enddocregion setStyles
|
|
||||||
|
|
||||||
// #docregion NgStyle
|
|
||||||
isItalic = false;
|
|
||||||
isBold = false;
|
|
||||||
fontSize: string = 'large';
|
|
||||||
fontSizePx: number | string = 14;
|
|
||||||
|
|
||||||
setStyle() {
|
|
||||||
return {
|
|
||||||
'font-style': this.isItalic ? 'italic' : 'normal',
|
|
||||||
'font-weight': this.isBold ? 'bold' : 'normal',
|
|
||||||
'font-size': this.fontSize
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
// #enddocregion NgStyle
|
// #enddocregion setStyles
|
||||||
|
|
||||||
toeChoice = '';
|
toeChoice = '';
|
||||||
toeChooser(picker: HTMLFieldSetElement) {
|
toeChooser(picker: HTMLFieldSetElement) {
|
||||||
|
@ -202,7 +171,6 @@ export class AppComponent implements AfterViewInit, OnInit {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// #docregion trackByHeroes
|
// #docregion trackByHeroes
|
||||||
trackByHeroes(index: number, hero: Hero) { return hero.id; }
|
trackByHeroes(index: number, hero: Hero) { return hero.id; }
|
||||||
// #enddocregion trackByHeroes
|
// #enddocregion trackByHeroes
|
||||||
|
|
|
@ -6,7 +6,8 @@ img {height: 100px;}
|
||||||
.parent-div {margin-top: 1em; font-weight: bold}
|
.parent-div {margin-top: 1em; font-weight: bold}
|
||||||
.special {font-weight:bold; font-size: x-large}
|
.special {font-weight:bold; font-size: x-large}
|
||||||
.bad {color: red;}
|
.bad {color: red;}
|
||||||
.curly {font-family: "Brush Script MT"}
|
.saveable {color: limegreen;}
|
||||||
|
.curly, .modified {font-family: "Brush Script MT"}
|
||||||
.toe {margin-left: 1em; font-style: italic;}
|
.toe {margin-left: 1em; font-style: italic;}
|
||||||
little-hero {color:blue; font-size: smaller; background-color: Turquoise }
|
little-hero {color:blue; font-size: smaller; background-color: Turquoise }
|
||||||
.to-toc {margin-top: 10px; display: block}
|
.to-toc {margin-top: 10px; display: block}
|
|
@ -997,15 +997,22 @@ figure.image-display
|
||||||
The `NgClass` directive may be the better choice
|
The `NgClass` directive may be the better choice
|
||||||
when we want to add or remove *many* CSS classes at the same time.
|
when we want to add or remove *many* CSS classes at the same time.
|
||||||
|
|
||||||
A good way to apply `NgClass` is by binding it to a key:value control !{__objectAsMap}. Each key of the object is a CSS class name; its value is `true` if the class should be added, `false` if it should be removed.
|
A good way to apply `NgClass` is by binding it to a key:value control !{__objectAsMap}.
|
||||||
|
Each key of the object is a CSS class name; its value is `true` if the class should be added,
|
||||||
|
`false` if it should be removed.
|
||||||
|
|
||||||
:marked
|
:marked
|
||||||
Consider a component method such as `setClasses` that manages the state of three CSS classes:
|
Consider a
|
||||||
|
`setCurrentClasses` component method that sets a component property, `currentClasses`
|
||||||
|
with an object that adds or removes three classes based on the
|
||||||
|
`true`/`false` state of three other component propertes:
|
||||||
+makeExample('template-syntax/ts/app/app.component.ts', 'setClasses')(format=".")
|
+makeExample('template-syntax/ts/app/app.component.ts', 'setClasses')(format=".")
|
||||||
:marked
|
:marked
|
||||||
Now we can add an `NgClass` property binding that calls `setClasses`
|
Adding an `NgClass` property binding to `currentClasses` sets the element's classes accordingly:
|
||||||
and sets the element's classes accordingly:
|
|
||||||
+makeExample('template-syntax/ts/app/app.component.html', 'NgClass-1')(format=".")
|
+makeExample('template-syntax/ts/app/app.component.html', 'NgClass-1')(format=".")
|
||||||
|
.l-sub-section
|
||||||
|
:marked
|
||||||
|
It's up to you to call `setCurrentClassess()`, both initially and when the dependent properties change.
|
||||||
|
|
||||||
<a id="ngStyle"></a>
|
<a id="ngStyle"></a>
|
||||||
.l-main-section
|
.l-main-section
|
||||||
|
@ -1023,12 +1030,15 @@ figure.image-display
|
||||||
We apply `NgStyle` by binding it to a key:value control !{__objectAsMap}.
|
We apply `NgStyle` by binding it to a key:value control !{__objectAsMap}.
|
||||||
Each key of the object is a style name; its value is whatever is appropriate for that style.
|
Each key of the object is a style name; its value is whatever is appropriate for that style.
|
||||||
|
|
||||||
Consider a component method such as `setStyles` that returns an object defining three styles:
|
Consider a `setCurrentStyles` component method that sets a component property, `currentStyles`
|
||||||
|
with an object that defines three styles, based on the state of three other component propertes:
|
||||||
+makeExample('template-syntax/ts/app/app.component.ts', 'setStyles')(format=".")
|
+makeExample('template-syntax/ts/app/app.component.ts', 'setStyles')(format=".")
|
||||||
:marked
|
:marked
|
||||||
Now we just add an `NgStyle` property binding that calls `setStyles`
|
Adding an `NgStyle` property binding to `currentStyles` sets the element's styles accordingly:
|
||||||
and sets the element's styles accordingly:
|
|
||||||
+makeExample('template-syntax/ts/app/app.component.html', 'NgStyle-2')(format=".")
|
+makeExample('template-syntax/ts/app/app.component.html', 'NgStyle-2')(format=".")
|
||||||
|
.l-sub-section
|
||||||
|
:marked
|
||||||
|
It's up to you to call `setCurrentStyles()`, both initially and when the dependent properties change.
|
||||||
|
|
||||||
<a id="ngIf"></a>
|
<a id="ngIf"></a>
|
||||||
.l-main-section
|
.l-main-section
|
||||||
|
|
Loading…
Reference in New Issue