Skip to content

Commit

Permalink
feat(accordion): add aria-selected
Browse files Browse the repository at this point in the history
Closes #1109
  • Loading branch information
coutin authored and pkozlowski-opensource committed Dec 13, 2016
1 parent d1ae5eb commit d1abd37
Show file tree
Hide file tree
Showing 2 changed files with 64 additions and 20 deletions.
39 changes: 38 additions & 1 deletion src/accordion/accordion.spec.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import {TestBed, ComponentFixture, inject} from '@angular/core/testing';
import {By} from '@angular/platform-browser';
import {createGenericTestComponent} from '../test/common';

import {Component} from '@angular/core';
Expand Down Expand Up @@ -30,6 +31,14 @@ function expectOpenPanels(nativeEl: HTMLElement, openPanelsDef: boolean[]) {
expect(result).toEqual(openPanelsDef);
}

function expectAriaSelected(nativeEl: HTMLElement, ariaSelectedPanelsDef: boolean[]) {
const panels = getPanels(nativeEl);
expect(panels.length).toBe(ariaSelectedPanelsDef.length);

const result = panels.map(panel => (panel.getAttribute('aria-selected') === 'true'));
expect(result).toEqual(ariaSelectedPanelsDef);
}

describe('ngb-accordion', () => {
let html = `
<ngb-accordion #acc="ngbAccordion" [closeOthers]="closeOthers" [activeIds]="activeIds"
Expand Down Expand Up @@ -57,8 +66,10 @@ describe('ngb-accordion', () => {
it('should have no open panels', () => {
const fixture = TestBed.createComponent(TestComponent);
const accordionEl = fixture.nativeElement.children[0];
const el = fixture.nativeElement;
fixture.detectChanges();
expectOpenPanels(fixture.nativeElement, [false, false, false]);
expectOpenPanels(el, [false, false, false]);
expectAriaSelected(el, [false, false, false]);
expect(accordionEl.getAttribute('role')).toBe('tablist');
expect(accordionEl.getAttribute('aria-multiselectable')).toBe('true');
});
Expand Down Expand Up @@ -318,6 +329,32 @@ describe('ngb-accordion', () => {
expect(el[2]).toHaveCssClass('card-warning');
});

it('should toggle aria-selected attribute of the focused panel', () => {
const fixture = TestBed.createComponent(TestComponent);
fixture.detectChanges();

const headingLinks = fixture.debugElement.queryAll(By.css('.card-header a'));

headingLinks[0].triggerEventHandler('focus', {});
fixture.detectChanges();
expectAriaSelected(fixture.nativeElement, [true, false, false]);

headingLinks[0].triggerEventHandler('blur', {});
headingLinks[1].triggerEventHandler('focus', {});
fixture.detectChanges();
expectAriaSelected(fixture.nativeElement, [false, true, false]);

headingLinks[1].triggerEventHandler('blur', {});
headingLinks[2].triggerEventHandler('focus', {});
fixture.detectChanges();
expectAriaSelected(fixture.nativeElement, [false, false, true]);

headingLinks[2].triggerEventHandler('blur', {});
headingLinks[1].triggerEventHandler('focus', {});
fixture.detectChanges();
expectAriaSelected(fixture.nativeElement, [false, true, false]);
});

describe('Custom config', () => {
let config: NgbAccordionConfig;

Expand Down
45 changes: 26 additions & 19 deletions src/accordion/accordion.ts
Original file line number Diff line number Diff line change
@@ -1,16 +1,18 @@
import {
AfterContentChecked,
Component,
Input,
QueryList,
ContentChild,
ContentChildren,
Directive,
TemplateRef,
ContentChild,
Output,
EventEmitter,
AfterContentChecked
Input,
Output,
QueryList,
TemplateRef
} from '@angular/core';

import {isString} from '../util/util';

import {NgbAccordionConfig} from './accordion-config';

let nextId = 0;
Expand All @@ -37,6 +39,11 @@ export class NgbPanelContent {
*/
@Directive({selector: 'ngb-panel'})
export class NgbPanel {
/**
* Defines if the tab control is focused
*/
focused: boolean = false;

/**
* A flag determining whether the panel is disabled or not.
* When disabled, the panel cannot be toggled.
Expand Down Expand Up @@ -95,9 +102,10 @@ export interface NgbPanelChangeEvent {
template: `
<div class="card">
<template ngFor let-panel [ngForOf]="panels">
<div role="tab" id="{{panel.id}}-header"
<div role="tab" id="{{panel.id}}-header" [attr.aria-selected]="panel.focused"
[class]="'card-header ' + (panel.type ? 'card-'+panel.type: type ? 'card-'+type : '')" [class.active]="isOpen(panel.id)">
<a href (click)="!!toggle(panel.id)" [class.text-muted]="panel.disabled"
<a href (click)="!!toggle(panel.id)" (focus)="panel.focused = true"
(blur)="panel.focused = false" [class.text-muted]="panel.disabled"
[attr.aria-expanded]="isOpen(panel.id)" [attr.aria-controls]="panel.id">
{{panel.title}}<template [ngTemplateOutlet]="panel.titleTpl?.templateRef"></template>
</a>
Expand All @@ -110,6 +118,16 @@ export interface NgbPanelChangeEvent {
`
})
export class NgbAccordion implements AfterContentChecked {
/**
* A map that stores each panel state
*/
private _states: Map<string, boolean> = new Map<string, boolean>();

/**
* A map that stores references to all panels
*/
private _panelRefs: Map<string, NgbPanel> = new Map<string, NgbPanel>();

@ContentChildren(NgbPanel) panels: QueryList<NgbPanel>;

/**
Expand All @@ -128,22 +146,11 @@ export class NgbAccordion implements AfterContentChecked {
*/
@Input() type: string;


/**
* A panel change event fired right before the panel toggle happens. See NgbPanelChangeEvent for payload details
*/
@Output() panelChange = new EventEmitter<NgbPanelChangeEvent>();

/**
* A map that stores each panel state
*/
private _states: Map<string, boolean> = new Map<string, boolean>();

/**
* A map that stores references to all panels
*/
private _panelRefs: Map<string, NgbPanel> = new Map<string, NgbPanel>();

constructor(config: NgbAccordionConfig) {
this.type = config.type;
this.closeOtherPanels = config.closeOthers;
Expand Down

0 comments on commit d1abd37

Please sign in to comment.