feat(aio): set “avoid” class by convention (filename has “.avoid.”)
Most important for StyleGuide which lost example css class in migration. Also hides copy for “avoid” files.
This commit is contained in:
parent
673d8ae583
commit
9ca2b4c967
|
@ -12,7 +12,7 @@
|
||||||
}
|
}
|
||||||
</code-pane>
|
</code-pane>
|
||||||
<code-pane title='HTML content file' language='html'><h1>Heading</h1></code-pane>
|
<code-pane title='HTML content file' language='html'><h1>Heading</h1></code-pane>
|
||||||
<code-pane title='JSON data file' language='json' class='is-anti-pattern'>{ "key": "value" }</code-pane>
|
<code-pane title='JSON data file' language='json' class='avoid'>{ "key": "value" }</code-pane>
|
||||||
</code-tabs>
|
</code-tabs>
|
||||||
<p></p>
|
<p></p>
|
||||||
|
|
||||||
|
@ -24,7 +24,7 @@
|
||||||
}
|
}
|
||||||
</code-pane>
|
</code-pane>
|
||||||
<code-pane title='HTML content file' language='html'><h1>Heading</h1></code-pane>
|
<code-pane title='HTML content file' language='html'><h1>Heading</h1></code-pane>
|
||||||
<code-pane title='JSON data file' language='json' class='is-anti-pattern'>{ "key": "value" }</code-pane>
|
<code-pane title='JSON data file' language='json' class='avoid'>{ "key": "value" }</code-pane>
|
||||||
</code-tabs>
|
</code-tabs>
|
||||||
<p></p>
|
<p></p>
|
||||||
|
|
||||||
|
@ -36,7 +36,7 @@
|
||||||
}
|
}
|
||||||
</code-pane>
|
</code-pane>
|
||||||
<code-pane title='HTML content file' language='html' linenums='true'><h1>Heading</h1></code-pane>
|
<code-pane title='HTML content file' language='html' linenums='true'><h1>Heading</h1></code-pane>
|
||||||
<code-pane title='JSON data file' language='json' class='is-anti-pattern'>{ "key": "value" }</code-pane>
|
<code-pane title='JSON data file' language='json' class='avoid'>{ "key": "value" }</code-pane>
|
||||||
</code-tabs>
|
</code-tabs>
|
||||||
<p></p>
|
<p></p>
|
||||||
|
|
||||||
|
@ -76,7 +76,7 @@
|
||||||
</code-example>
|
</code-example>
|
||||||
|
|
||||||
<p>An "avoid" header on this one.</p>
|
<p>An "avoid" header on this one.</p>
|
||||||
<code-example class="is-anti-pattern" title="hero-details.component.ts (Avoid)">
|
<code-example class="avoid" title="hero-details.component.ts (Avoid)">
|
||||||
<hero-details <em>nghost-pmm-5>
|
<hero-details <em>nghost-pmm-5>
|
||||||
<h2 </em>ngcontent-pmm-5>Mister Fantastic</h2>
|
<h2 </em>ngcontent-pmm-5>Mister Fantastic</h2>
|
||||||
<hero-team <em>ngcontent-pmm-5 </em>nghost-pmm-6>
|
<hero-team <em>ngcontent-pmm-5 </em>nghost-pmm-6>
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/* tslint:disable component-selector */
|
/* tslint:disable component-selector */
|
||||||
import { Component, ElementRef, OnInit } from '@angular/core';
|
import { Component, ElementRef, HostBinding, OnInit } from '@angular/core';
|
||||||
import { getBoolFromAttribute } from 'app/shared/attribute-utils';
|
import { getBoolFromAttribute } from 'app/shared/attribute-utils';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -17,12 +17,13 @@ import { getBoolFromAttribute } from 'app/shared/attribute-utils';
|
||||||
selector: 'code-example',
|
selector: 'code-example',
|
||||||
template: `
|
template: `
|
||||||
<header *ngIf="title">{{title}}</header>
|
<header *ngIf="title">{{title}}</header>
|
||||||
<aio-code [ngClass]="{'headed-code':title, 'simple-code':!title}" [code]="code"
|
<aio-code [ngClass]="classes" [code]="code"
|
||||||
[language]="language" [linenums]="linenums" [path]="path" [region]="region" [hideCopy]="hideCopy"></aio-code>
|
[language]="language" [linenums]="linenums" [path]="path" [region]="region" [hideCopy]="hideCopy"></aio-code>
|
||||||
`
|
`
|
||||||
})
|
})
|
||||||
export class CodeExampleComponent implements OnInit {
|
export class CodeExampleComponent implements OnInit {
|
||||||
|
|
||||||
|
classes: {};
|
||||||
code: string;
|
code: string;
|
||||||
language: string;
|
language: string;
|
||||||
linenums: boolean | number;
|
linenums: boolean | number;
|
||||||
|
@ -31,6 +32,9 @@ export class CodeExampleComponent implements OnInit {
|
||||||
title: string;
|
title: string;
|
||||||
hideCopy: boolean;
|
hideCopy: boolean;
|
||||||
|
|
||||||
|
@HostBinding('class.avoidFile')
|
||||||
|
isAvoid = false;
|
||||||
|
|
||||||
constructor(private elementRef: ElementRef) {
|
constructor(private elementRef: ElementRef) {
|
||||||
const element = this.elementRef.nativeElement;
|
const element = this.elementRef.nativeElement;
|
||||||
|
|
||||||
|
@ -39,7 +43,14 @@ export class CodeExampleComponent implements OnInit {
|
||||||
this.path = element.getAttribute('path') || '';
|
this.path = element.getAttribute('path') || '';
|
||||||
this.region = element.getAttribute('region') || '';
|
this.region = element.getAttribute('region') || '';
|
||||||
this.title = element.getAttribute('title') || '';
|
this.title = element.getAttribute('title') || '';
|
||||||
this.hideCopy = getBoolFromAttribute(element, ['hidecopy', 'hide-copy']);
|
|
||||||
|
this.isAvoid = this.path.indexOf('.avoid.') !== -1;
|
||||||
|
this.hideCopy = this.isAvoid || getBoolFromAttribute(element, ['hidecopy', 'hide-copy']);
|
||||||
|
|
||||||
|
this.classes = {
|
||||||
|
'headed-code': !!this.title,
|
||||||
|
'simple-code': !this.title,
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
|
|
|
@ -11,7 +11,7 @@ describe('Attribute Utilities', () => {
|
||||||
testEl = div.querySelector('div');
|
testEl = div.querySelector('div');
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('#getAttrs', () => {
|
describe('getAttrs', () => {
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
this.expectedMap = {
|
this.expectedMap = {
|
||||||
|
@ -34,7 +34,7 @@ describe('Attribute Utilities', () => {
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('#getAttrValue', () => {
|
describe('getAttrValue', () => {
|
||||||
let attrMap: { [index: string]: string };
|
let attrMap: { [index: string]: string };
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
|
@ -46,7 +46,7 @@ describe('Attribute Utilities', () => {
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should return empty string value for attribute "A"', () => {
|
it('should return empty string value for attribute "A"', () => {
|
||||||
expect(getAttrValue(attrMap, 'a')).toBe('');
|
expect(getAttrValue(attrMap, 'A')).toBe('');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should return "true" for attribute "b"', () => {
|
it('should return "true" for attribute "b"', () => {
|
||||||
|
@ -54,11 +54,10 @@ describe('Attribute Utilities', () => {
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should return empty string value for attribute "d-E"', () => {
|
it('should return empty string value for attribute "d-E"', () => {
|
||||||
expect(getAttrValue(attrMap, 'd-e')).toBe('');
|
expect(getAttrValue(attrMap, 'd-E')).toBe('');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should return empty string for attribute ["d-e"]', () => {
|
it('should return empty string for attribute ["d-e"]', () => {
|
||||||
// because d-e will be found before d
|
|
||||||
expect(getAttrValue(attrMap, ['d-e'])).toBe('');
|
expect(getAttrValue(attrMap, ['d-e'])).toBe('');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -82,48 +81,48 @@ describe('Attribute Utilities', () => {
|
||||||
|
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('#boolFromValue', () => {
|
describe('boolFromValue', () => {
|
||||||
let attrMap: { [index: string]: string };
|
|
||||||
|
|
||||||
beforeEach(() => {
|
it('should return true for "" as in present but unassigned attr "a"', () => {
|
||||||
attrMap = getAttrs(testEl);
|
expect(boolFromValue('')).toBe(true);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should return true for present but unassigned attr "a"', () => {
|
it('should return false for "false" as in attr "c"', () => {
|
||||||
expect(boolFromValue(getAttrValue(attrMap, 'a'))).toBe(true);
|
expect(boolFromValue('false')).toBe(false);
|
||||||
|
});
|
||||||
|
it('should return true for "true" as in attr "b"', () => {
|
||||||
|
expect(boolFromValue('true')).toBe(true);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should return true for attr "b" which is "true"', () => {
|
it('should return true for something other than "false"', () => {
|
||||||
expect(boolFromValue(getAttrValue(attrMap, 'b'))).toBe(true);
|
expect(boolFromValue('foo')).toBe(true);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should return false for attr "c" which is "false"', () => {
|
it('should return true for "False" because case-sensitive', () => {
|
||||||
expect(boolFromValue(getAttrValue(attrMap, 'c'))).toBe(false);
|
expect(boolFromValue('False')).toBe(true);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should return false by default for undefined attr "x"', () => {
|
|
||||||
expect(boolFromValue(getAttrValue(attrMap, 'x'))).toBe(false);
|
it('should return false by default as in undefined attr "x"', () => {
|
||||||
|
expect(boolFromValue(undefined)).toBe(false);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should return true for undefined attr "x" when default is true', () => {
|
it('should return true for undefined value when default is true', () => {
|
||||||
const value = getAttrValue(attrMap, 'x');
|
expect(boolFromValue(undefined, true)).toBe(true);
|
||||||
expect(boolFromValue(value, true)).toBe(true);
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should return false for undefined attr "x" when default is false', () => {
|
it('should return false for undefined value when default is false', () => {
|
||||||
const value = getAttrValue(attrMap, 'x');
|
expect(boolFromValue(undefined, false)).toBe(false);
|
||||||
expect(boolFromValue(value, false)).toBe(false);
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should return true for present but unassigned attr "a" even when default is false', () => {
|
it('should return true for "" as in unassigned attr "a" even when default is false', () => {
|
||||||
// default value is only applied when the attribute is missing
|
// default value is only applied when the attribute is missing
|
||||||
const value = getAttrValue(attrMap, 'a');
|
expect(boolFromValue('', false)).toBe(true);
|
||||||
expect(boolFromValue(value, false)).toBe(true);
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// Combines the three utilities for convenience.
|
// Combines the three utilities for convenience.
|
||||||
describe('#getBoolFromAttribute', () => {
|
describe('getBoolFromAttribute', () => {
|
||||||
it('should return true for present but unassigned attr "a"', () => {
|
it('should return true for present but unassigned attr "a"', () => {
|
||||||
expect(getBoolFromAttribute(testEl, 'a')).toBe(true);
|
expect(getBoolFromAttribute(testEl, 'a')).toBe(true);
|
||||||
});
|
});
|
||||||
|
|
|
@ -17,16 +17,19 @@ export function getAttrs(el: HTMLElement | ElementRef): StringMap {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Return the attribute that matches `atty`.
|
* Return the attribute that matches `attr`.
|
||||||
* @param atty Name of the attribute or a string of candidate attribute names
|
* @param attr Name of the attribute or a string of candidate attribute names
|
||||||
*/
|
*/
|
||||||
export function getAttrValue(attrs: StringMap, attr: string | string[] = ''): string {
|
export function getAttrValue(attrs: StringMap, attr: string | string[] = ''): string {
|
||||||
return attrs[typeof attr === 'string' ? attr : attr.find(a => attrs[a] !== undefined)];
|
return attrs[typeof attr === 'string' ?
|
||||||
|
attr.toLowerCase() :
|
||||||
|
attr.find(a => attrs[a.toLowerCase()] !== undefined)
|
||||||
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Return the boolean state of an attribute value (if supplied)
|
* Return the boolean state of an attribute value (if supplied)
|
||||||
* @param attyValue The string value of some attribute (or undefined if attribute not present)
|
* @param attrValue The string value of some attribute (or undefined if attribute not present)
|
||||||
* @param def Default boolean value when attribute is undefined.
|
* @param def Default boolean value when attribute is undefined.
|
||||||
*/
|
*/
|
||||||
export function boolFromValue(attrValue: string, def: boolean = false) {
|
export function boolFromValue(attrValue: string, def: boolean = false) {
|
||||||
|
|
|
@ -28,13 +28,16 @@ code-example header {
|
||||||
margin: -17px;
|
margin: -17px;
|
||||||
}
|
}
|
||||||
|
|
||||||
code-example.is-anti-pattern header {
|
code-example.avoid header,
|
||||||
|
code-example.avoidFile header {
|
||||||
border: 2px solid $anti-pattern;
|
border: 2px solid $anti-pattern;
|
||||||
background: $anti-pattern;
|
background: $anti-pattern;
|
||||||
}
|
}
|
||||||
|
|
||||||
code-example.is-anti-pattern,
|
code-example.avoid,
|
||||||
code-tabs.is-anti-pattern md-tab-body {
|
code-example.avoidFile,
|
||||||
|
code-tabs.avoid md-tab-body,
|
||||||
|
code-tabs.avoidFile md-tab-body {
|
||||||
border: 0.5px solid $anti-pattern;
|
border: 0.5px solid $anti-pattern;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -223,4 +226,4 @@ code-tabs md-tab-group *.mat-ripple-element, code-tabs md-tab-group *.mat-tab-bo
|
||||||
|
|
||||||
[role="tabpanel"] {
|
[role="tabpanel"] {
|
||||||
transition: none;
|
transition: none;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue