Skip to content

Commit

Permalink
Merge pull request #3445 from IgniteUI/avatar-box-sizing
Browse files Browse the repository at this point in the history
fix(avatar): allow for more robust styling of the avatar to fix border issues
  • Loading branch information
zdrawku authored Jan 3, 2019
2 parents 7ea2649 + d5bac95 commit c84e02e
Show file tree
Hide file tree
Showing 7 changed files with 242 additions and 119 deletions.
16 changes: 7 additions & 9 deletions projects/igniteui-angular/src/lib/avatar/avatar.component.html
Original file line number Diff line number Diff line change
@@ -1,19 +1,17 @@
<ng-template #defaultTemplate>
<ng-content></ng-content>
</ng-template>

<ng-template #imageTemplate>
<div #image [style.backgroundImage]="getSrcUrl()" class="igx-avatar__image" [style.backgroundColor]="bgColor" [attr.aria-roledescription]="roleDescription"
></div>
<div #image class="igx-avatar__image" [style.backgroundImage]="getSrcUrl()"></div>
</ng-template>

<ng-template #initialsTemplate>
<div class="igx-avatar__initials" [style.backgroundColor]="bgColor" [style.color]="color" [attr.aria-roledescription]="roleDescription">
<span>{{initials.substring(0, 2)}}</span>
</div>
<span>{{initials.substring(0, 2)}}</span>
</ng-template>

<ng-template #iconTemplate>
<span class="igx-avatar__icon" [style.backgroundColor]="bgColor" [style.color]="color" [attr.aria-roledescription]="roleDescription">
<igx-icon>{{icon}}</igx-icon>
</span>
<igx-icon>{{icon}}</igx-icon>
</ng-template>

<ng-container *ngTemplateOutlet="template"></ng-container>
<ng-content></ng-content>
201 changes: 137 additions & 64 deletions projects/igniteui-angular/src/lib/avatar/avatar.component.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -5,21 +5,33 @@ import {
} from '@angular/core/testing';
import { By } from '@angular/platform-browser';
import { IgxIconModule } from '../icon/index';
import { IgxAvatarComponent, IgxAvatarModule } from './avatar.component';
import { IgxAvatarComponent, AvatarType, Size } from './avatar.component';

import { configureTestSuite } from '../test-utils/configure-suite';

describe('Avatar', () => {
configureTestSuite();
const baseClass = 'igx-avatar';

const classes = {
default: `${baseClass}--default`,
round: `${baseClass}--rounded`,
small: `${baseClass}--small`,
medium: `${baseClass}--medium`,
large: `${baseClass}--large`,
image: `${baseClass}--image`,
initials: `${baseClass}--initials`,
icon: `${baseClass}--icon`
};

beforeEach(async(() => {
TestBed.configureTestingModule({
declarations: [
InitAvatarComponent,
AvatarWithAttribsComponent,
IgxAvatarComponent,
InitIconAvatarComponent,
InitImageAvatarComponent,
InitAvatarWithAriaComponent
InitImageAvatarComponent
],
imports: [IgxIconModule]
})
Expand All @@ -29,122 +41,183 @@ describe('Avatar', () => {
it('Initializes avatar with auto-incremented id', () => {
const fixture = TestBed.createComponent(InitAvatarComponent);
fixture.detectChanges();
const avatar = fixture.componentInstance.avatar;
const domAvatar = fixture.debugElement.query(By.css('igx-avatar')).nativeElement;
const instance = fixture.componentInstance.avatar;
const hostEl = fixture.debugElement.query(By.css(baseClass)).nativeElement;

expect(instance.id).toContain('igx-avatar-');
expect(hostEl.id).toContain('igx-avatar-');

instance.id = 'customAvatar';
fixture.detectChanges();

expect(instance.id).toBe('customAvatar');
expect(hostEl.id).toBe('customAvatar');
});

it('Initializes square and round avatar', () => {
const fixture = TestBed.createComponent(AvatarWithAttribsComponent);
fixture.detectChanges();
const instance = fixture.componentInstance.avatar;
const hostEl = fixture.debugElement.query(By.css(baseClass)).nativeElement;

expect(instance.roundShape).toBeTruthy();
expect(hostEl.classList).toContain(classes.round);

expect(avatar.id).toContain('igx-avatar-');
expect(domAvatar.id).toContain('igx-avatar-');
instance.roundShape = false;

fixture.detectChanges();
expect(instance.roundShape).toBeFalsy();
expect(hostEl.classList).not.toContain(classes.round);
});

it('Can change its size', () => {
const fixture = TestBed.createComponent(AvatarWithAttribsComponent);
fixture.detectChanges();
const instance = fixture.componentInstance.avatar;
const hostEl = fixture.debugElement.query(By.css(baseClass)).nativeElement;

expect(instance.size).toEqual(Size.SMALL);
expect(hostEl.classList).toContain(classes.small);

instance.size = Size.MEDIUM;
fixture.detectChanges();
expect(instance.size).toEqual(Size.MEDIUM);
expect(hostEl.classList).not.toContain(classes.medium);

avatar.id = 'customAvatar';
instance.size = Size.LARGE;
fixture.detectChanges();
expect(instance.size).toEqual(Size.LARGE);
expect(hostEl.classList).not.toContain(classes.large);

expect(avatar.id).toBe('customAvatar');
expect(domAvatar.id).toBe('customAvatar');
instance.size = 'nonsense';
fixture.detectChanges();
expect(instance.size).toEqual(Size.SMALL);
expect(hostEl.classList).toContain(classes.small);
});

it('Initializes avatar with initials', () => {
it('Initializes default avatar', () => {
const fixture = TestBed.createComponent(InitAvatarComponent);
fixture.detectChanges();
const avatar = fixture.componentInstance.avatar;

expect(fixture.debugElement.query(By.css('.igx-avatar__initials'))).toBeTruthy();
expect(avatar.roundShape).toEqual(false);
const instance = fixture.componentInstance.avatar;
const hostEl = fixture.debugElement.query(By.css(baseClass)).nativeElement;

expect(instance.type).toEqual(AvatarType.DEFAULT);
expect(instance.initials).toBeUndefined();
expect(instance.src).toBeUndefined();
expect(instance.icon).toBeUndefined();

expect(hostEl.textContent).toEqual('TEST');
expect(hostEl.classList).toContain(classes.default);
});

it('Initializes round avatar with initials', () => {

it('Initializes initials avatar', () => {
const fixture = TestBed.createComponent(AvatarWithAttribsComponent);
fixture.detectChanges();
const avatar = fixture.componentInstance.avatar;

expect(avatar.elementRef.nativeElement.classList.contains('igx-avatar--rounded')).toBeTruthy();
expect(fixture.debugElement.query(By.css('.igx-avatar__initials'))).toBeTruthy();
expect(avatar.roundShape).toBeTruthy();
const instance = fixture.componentInstance.avatar;
const hostEl = fixture.debugElement.query(By.css(baseClass)).nativeElement;

expect(instance.type).toEqual(AvatarType.INITIALS);
expect(instance.initials).toEqual('ZK');
expect(hostEl.querySelector('span').textContent).toEqual('ZK');
expect(hostEl.classList).toContain(classes.initials);
});

it('Initializes icon avatar', () => {
const fixture = TestBed.createComponent(InitIconAvatarComponent);
fixture.detectChanges();
const avatar = fixture.componentInstance.avatar;
const spanEl = avatar.elementRef.nativeElement.querySelector('.igx-avatar__icon');

expect(avatar.image === undefined).toBeTruthy();
expect(avatar.src).toBeFalsy();
expect(spanEl).toBeTruthy();
expect(avatar.elementRef.nativeElement.classList.contains('igx-avatar--small')).toBeTruthy();
expect(spanEl.classList.length === 1).toBeTruthy();
expect(avatar.roundShape).toBeFalsy();

// For ARIA
expect(spanEl.getAttribute('aria-roledescription') === 'icon type avatar').toBeTruthy();
const instance = fixture.componentInstance.avatar;
const hostEl = fixture.debugElement.query(By.css(baseClass)).nativeElement;

expect(instance.type).toEqual(AvatarType.ICON);
expect(instance.icon).toBeTruthy();
expect(hostEl.classList).toContain(classes.icon);
});

it('Initializes image avatar', () => {
const fixture = TestBed.createComponent(InitImageAvatarComponent);
fixture.detectChanges();
const avatar = fixture.componentInstance.avatar;
const instance = fixture.componentInstance.avatar;
const hostEl = fixture.debugElement.query(By.css(baseClass)).nativeElement;

expect(avatar.image).toBeTruthy();
expect(avatar.image.nativeElement.style.backgroundImage.length !== 0).toBeTruthy();
expect(avatar.elementRef.nativeElement.classList.contains('igx-avatar--large')).toBeTruthy();
expect(avatar.image.nativeElement.classList.length === 1).toBeTruthy();
expect(avatar.roundShape).toBeTruthy();
expect(instance.type).toEqual(AvatarType.IMAGE);
expect(instance.image).toBeTruthy();
expect(instance.image.nativeElement.style.backgroundImage).toBeDefined();

// For ARIA
expect(avatar.image.nativeElement.getAttribute('aria-roledescription') === 'image type avatar').toBeTruthy();
expect(avatar.roleDescription === 'image type avatar').toBeTruthy();
expect(instance.image.nativeElement.classList).toContain(`${baseClass}__image`);
expect(hostEl.classList).toContain(classes.image);
});

it('Should set ARIA attributes.', () => {
const fixture = TestBed.createComponent(InitAvatarWithAriaComponent);
it('Sets background and foreground colors', () => {
const fixture = TestBed.createComponent(AvatarWithAttribsComponent);
fixture.detectChanges();
const avatar = fixture.componentInstance.avatar;
const instance = fixture.componentInstance.avatar;
const hostEl = fixture.debugElement.query(By.css(baseClass)).nativeElement;

expect(hostEl.style.background).toEqual(instance.bgColor);
expect(hostEl.style.color).toEqual(instance.color);

expect(avatar.elementRef.nativeElement.querySelector('.igx-avatar__initials')
.getAttribute('aria-roledescription')).toMatch('initials type avatar');
instance.bgColor = '#000';
instance.color = '#fff';

fixture.detectChanges();
expect(hostEl.style.background).toEqual('rgb(0, 0, 0)');
expect(hostEl.style.color).toEqual('rgb(255, 255, 255)');
});

it('Sets ARIA attributes', () => {
const fixture = TestBed.createComponent(InitImageAvatarComponent);
fixture.detectChanges();
const instance = fixture.componentInstance.avatar;
const hostEl = fixture.debugElement.query(By.css(baseClass)).nativeElement;

expect(instance.roleDescription).toEqual('image avatar');
expect(hostEl.getAttribute('role')).toEqual('img');
expect(hostEl.getAttribute('aria-roledescription')).toEqual('image avatar');
expect(hostEl.getAttribute('aria-label')).toEqual('avatar');
});
});

@Component({
template: `<igx-avatar initials="PP" size="medium" [roundShape]="false"
bgColor="paleturquoise">
</igx-avatar>`})
template: `<igx-avatar>TEST</igx-avatar>`
})
class InitAvatarComponent {
@ViewChild(IgxAvatarComponent) public avatar: IgxAvatarComponent;
}

@Component({
template: `<igx-avatar [initials]="initials" [bgColor]="bgColor" size="small"
[roundShape]="roundShape"></igx-avatar>`})
template: `<igx-avatar
[initials]="initials"
[bgColor]="bgColor"
[color]="color"
size="small"
[roundShape]="roundShape">
</igx-avatar>`
})
class AvatarWithAttribsComponent {
@ViewChild(IgxAvatarComponent) public avatar: IgxAvatarComponent;

public initials = 'ZK';
public bgColor = 'lightblue';
public color = 'orange';
public bgColor = 'royalblue';
public roundShape = 'true';
}

@Component({
template: `<igx-avatar [roundShape]="false" icon="person"
bgColor="#0375be" size="someIncorectSize">
</igx-avatar>`})
template: `<igx-avatar icon="person"></igx-avatar>`
})
class InitIconAvatarComponent {
@ViewChild(IgxAvatarComponent) public avatar: IgxAvatarComponent;
}

@Component({
template: `<igx-avatar [roundShape]="true" bgColor="#0375be" size="large"
[src]="source">
</igx-avatar>`})
template: `<igx-avatar [src]="source"></igx-avatar>`
})
class InitImageAvatarComponent {
@ViewChild(IgxAvatarComponent) public avatar: IgxAvatarComponent;

// tslint:disable-next-line:max-line-length
public source = '';
}

@Component({
template: `<igx-avatar initials="PP" size="medium">
</igx-avatar>`})
class InitAvatarWithAriaComponent {
@ViewChild(IgxAvatarComponent) public avatar: IgxAvatarComponent;
}
Loading

0 comments on commit c84e02e

Please sign in to comment.