blob: 0149bd5a1aa44103a451c51846ef1f1392713a8b [file] [log] [blame]
/**
* @license
* Copyright (C) 2016 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
import * as MockInteractions from '@polymer/iron-test-helpers/mock-interactions';
import '../../../test/common-test-setup-karma';
import './gr-button';
import {addListener} from '@polymer/polymer/lib/utils/gestures';
import {appContext} from '../../../services/app-context';
import {html} from '@polymer/polymer/lib/utils/html-tag';
import {GrButton} from './gr-button';
import {queryAndAssert} from '../../../test/test-utils';
import {PaperButtonElement} from '@polymer/paper-button';
const basicFixture = fixtureFromElement('gr-button');
const nestedFixture = fixtureFromTemplate(html`
<div id="test">
<gr-button class="testBtn"></gr-button>
</div>
`);
const tabindexFixture = fixtureFromTemplate(html`
<gr-button tabindex="3"></gr-button>
`);
suite('gr-button tests', () => {
let element: GrButton;
const addSpyOn = function (eventName: string) {
const spy = sinon.spy();
if (eventName === 'tap') {
addListener(element, eventName, spy);
} else {
element.addEventListener(eventName, spy);
}
return spy;
};
setup(async () => {
element = basicFixture.instantiate();
await element.updateComplete;
});
test('disabled is set by disabled', async () => {
const paperBtn = queryAndAssert<PaperButtonElement>(
element,
'paper-button'
);
assert.isFalse(paperBtn.disabled);
element.disabled = true;
await element.updateComplete;
assert.isTrue(paperBtn.disabled);
element.disabled = false;
await element.updateComplete;
assert.isFalse(paperBtn.disabled);
});
test('loading set from listener', async () => {
let resolve: Function;
element.addEventListener('click', e => {
const target = e.target as HTMLElement;
target.setAttribute('loading', 'true');
resolve = () => target.removeAttribute('loading');
});
const paperBtn = queryAndAssert<PaperButtonElement>(
element,
'paper-button'
);
assert.isFalse(paperBtn.disabled);
MockInteractions.tap(element);
await element.updateComplete;
assert.isTrue(paperBtn.disabled);
assert.isTrue(element.hasAttribute('loading'));
resolve!();
await element.updateComplete;
assert.isFalse(paperBtn.disabled);
assert.isFalse(element.hasAttribute('loading'));
});
test('tabindex should be -1 if disabled', async () => {
element.disabled = true;
await element.updateComplete;
assert.equal(element.getAttribute('tabindex'), '-1');
});
// Regression tests for Issue: 11969
test('tabindex should be reset to 0 if enabled', async () => {
element.disabled = false;
await element.updateComplete;
assert.equal(element.getAttribute('tabindex'), '0');
element.disabled = true;
await element.updateComplete;
assert.equal(element.getAttribute('tabindex'), '-1');
element.disabled = false;
await element.updateComplete;
assert.equal(element.getAttribute('tabindex'), '0');
});
test('tabindex should be preserved', async () => {
const tabIndexElement = tabindexFixture.instantiate() as GrButton;
tabIndexElement.disabled = false;
await element.updateComplete;
assert.equal(tabIndexElement.getAttribute('tabindex'), '3');
tabIndexElement.disabled = true;
await element.updateComplete;
assert.equal(tabIndexElement.getAttribute('tabindex'), '-1');
tabIndexElement.disabled = false;
await element.updateComplete;
assert.equal(tabIndexElement.getAttribute('tabindex'), '3');
});
// 'tap' event is tested so we don't loose backward compatibility with older
// plugins who didn't move to on-click which is faster and well supported.
test('dispatches click event', () => {
const spy = addSpyOn('click');
MockInteractions.click(element);
assert.isTrue(spy.calledOnce);
});
test('dispatches tap event', () => {
const spy = addSpyOn('tap');
MockInteractions.tap(element);
assert.isTrue(spy.calledOnce);
});
test('dispatches click from tap event', () => {
const spy = addSpyOn('click');
MockInteractions.tap(element);
assert.isTrue(spy.calledOnce);
});
// Keycodes: 32 for Space, 13 for Enter.
for (const key of [32, 13]) {
test(`dispatches click event on keycode ${key}`, () => {
const tapSpy = sinon.spy();
element.addEventListener('click', tapSpy);
MockInteractions.pressAndReleaseKeyOn(element, key);
assert.isTrue(tapSpy.calledOnce);
});
test(`dispatches no click event with modifier on keycode ${key}`, () => {
const tapSpy = sinon.spy();
element.addEventListener('click', tapSpy);
MockInteractions.pressAndReleaseKeyOn(element, key, 'shift');
MockInteractions.pressAndReleaseKeyOn(element, key, 'ctrl');
MockInteractions.pressAndReleaseKeyOn(element, key, 'meta');
MockInteractions.pressAndReleaseKeyOn(element, key, 'alt');
assert.isFalse(tapSpy.calledOnce);
});
}
suite('disabled', () => {
setup(async () => {
element.disabled = true;
await element.updateComplete;
});
for (const eventName of ['tap', 'click']) {
test('stops ' + eventName + ' event', () => {
const spy = addSpyOn(eventName);
MockInteractions.tap(element);
assert.isFalse(spy.called);
});
}
// Keycodes: 32 for Space, 13 for Enter.
for (const key of [32, 13]) {
test(`stops click event on keycode ${key}`, () => {
const tapSpy = sinon.spy();
element.addEventListener('click', tapSpy);
MockInteractions.pressAndReleaseKeyOn(element, key);
assert.isFalse(tapSpy.called);
});
}
});
suite('reporting', () => {
let reportStub: sinon.SinonStub;
setup(() => {
reportStub = sinon.stub(appContext.reportingService, 'reportInteraction');
reportStub.reset();
});
test('report event after click', () => {
MockInteractions.click(element);
assert.isTrue(reportStub.calledOnce);
assert.equal(reportStub.lastCall.args[0], 'button-click');
assert.deepEqual(reportStub.lastCall.args[1], {
path: `html>body>test-fixture#${element.parentElement!.id}>gr-button`,
});
});
test('report event after click on nested', () => {
const nestedElement = nestedFixture.instantiate() as HTMLDivElement;
MockInteractions.click(queryAndAssert(nestedElement, 'gr-button'));
assert.isTrue(reportStub.calledOnce);
assert.equal(reportStub.lastCall.args[0], 'button-click');
assert.deepEqual(reportStub.lastCall.args[1], {
path:
`html>body>test-fixture#${nestedElement.parentElement!.id}` +
'>div#test>gr-button.testBtn',
});
});
});
});