Skip to content

Commit c38d84e

Browse files
test(topappbar): Add Jasmine component test. (material-components#5459)
* Project import generated by Copybara. PiperOrigin-RevId: 289465443 * Remove Mocha component test. Co-authored-by: Material Web Copybara Robot <59487319+material-web-copybara@users.noreply.github.com>
1 parent aca8e6c commit c38d84e

File tree

3 files changed

+320
-267
lines changed

3 files changed

+320
-267
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,319 @@
1+
/**
2+
* @license
3+
* Copyright 2020 Google Inc.
4+
*
5+
* Permission is hereby granted, free of charge, to any person obtaining a copy
6+
* of this software and associated documentation files (the "Software"), to deal
7+
* in the Software without restriction, including without limitation the rights
8+
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9+
* copies of the Software, and to permit persons to whom the Software is
10+
* furnished to do so, subject to the following conditions:
11+
*
12+
* The above copyright notice and this permission notice shall be included in
13+
* all copies or substantial portions of the Software.
14+
*
15+
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16+
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17+
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18+
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19+
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20+
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
21+
* THE SOFTWARE.
22+
*/
23+
24+
import {MDCRipple} from '../../mdc-ripple/component';
25+
import {emitEvent} from '../../../testing/dom/events';
26+
import {createMockFoundation} from '../../../testing/helpers/foundation';
27+
import {strings} from '../constants';
28+
import {MDCFixedTopAppBarFoundation} from '../fixed/foundation';
29+
import {MDCTopAppBar} from '../index';
30+
import {MDCShortTopAppBarFoundation} from '../short/foundation';
31+
import {MDCTopAppBarFoundation} from '../standard/foundation';
32+
33+
function getFixture(removeIcon = false) {
34+
const wrapper = document.createElement('div');
35+
wrapper.innerHTML = `
36+
<div>
37+
<header class="mdc-top-app-bar">
38+
<div class="mdc-top-app-bar__row">
39+
<section class="mdc-top-app-bar__section mdc-top-app-bar__section--align-start">
40+
<a href="#" class="material-icons mdc-top-app-bar__navigation-icon">menu</a>
41+
<span class="mdc-top-app-bar__title">Title</span>
42+
</section>
43+
<section class="mdc-top-app-bar__section mdc-top-app-bar__section--align-end"
44+
role="top-app-bar">
45+
<a href="#" class="material-icons mdc-top-app-bar__action-item" aria-label="Download" alt="Download">
46+
file_download</a>
47+
<a href="#" class="material-icons mdc-top-app-bar__action-item"
48+
aria-label="Print this page" alt="Print this page">
49+
print</a>
50+
<a href="#" class="material-icons mdc-top-app-bar__action-item" aria-label="Bookmark this page"
51+
alt="Bookmark this page">bookmark</a>
52+
<div class="mdc-menu-anchor">
53+
<div class="mdc-menu" tabindex="-1" id="demo-menu">
54+
<ul class="mdc-menu__items mdc-list" role="menu" aria-hidden="true" style="transform: scale(1, 1);">
55+
</ul>
56+
</div>
57+
</section>
58+
</div>
59+
</header>
60+
<main class="mdc-top-app-bar-fixed-adjust">
61+
</main>
62+
</div>
63+
`;
64+
65+
const el = wrapper.firstElementChild as HTMLElement;
66+
wrapper.removeChild(el);
67+
68+
if (removeIcon) {
69+
const icon =
70+
el.querySelector(strings.NAVIGATION_ICON_SELECTOR) as HTMLElement;
71+
(icon.parentNode as HTMLElement).removeChild(icon);
72+
}
73+
74+
return el;
75+
}
76+
77+
class FakeRipple {
78+
destroy: Function = jasmine.createSpy('.destroy');
79+
unbounded: boolean|null = null;
80+
}
81+
82+
function setupTest(
83+
removeIcon = false,
84+
rippleFactory = () => new FakeRipple()) {
85+
const fixture = getFixture(removeIcon);
86+
const root = fixture.querySelector(strings.ROOT_SELECTOR) as HTMLElement;
87+
const mockFoundation = createMockFoundation(MDCTopAppBarFoundation);
88+
mockFoundation.handleNavigationClick = jasmine.createSpy();
89+
mockFoundation.handleTargetScroll = jasmine.createSpy();
90+
mockFoundation.handleWindowResize = jasmine.createSpy();
91+
92+
const icon =
93+
root.querySelector(strings.NAVIGATION_ICON_SELECTOR) as HTMLElement;
94+
const component = new MDCTopAppBar(root, mockFoundation, rippleFactory);
95+
96+
return {root, component, icon, mockFoundation, fixture};
97+
}
98+
99+
describe('MDCTopAppBar', () => {
100+
it('attachTo initializes and returns an MDCTopAppBar instance', () => {
101+
expect(MDCTopAppBar.attachTo(getFixture()) instanceof MDCTopAppBar)
102+
.toBe(true);
103+
});
104+
105+
it('constructor instantiates icon ripples for all icons', () => {
106+
const rippleFactory = jasmine.createSpy('');
107+
rippleFactory.withArgs(jasmine.anything())
108+
.and.callFake(() => new FakeRipple());
109+
setupTest(/** removeIcon */ false, rippleFactory);
110+
});
111+
112+
it('constructor does not instantiate ripple for nav icon when not present',
113+
() => {
114+
const rippleFactory = jasmine.createSpy('');
115+
rippleFactory.withArgs(jasmine.anything())
116+
.and.callFake(() => new FakeRipple());
117+
setupTest(/** removeIcon */ true, rippleFactory);
118+
});
119+
120+
it('navIcon click event calls #foundation.handleNavigationClick', () => {
121+
const {root, mockFoundation} = setupTest();
122+
const navIcon =
123+
root.querySelector('.mdc-top-app-bar__navigation-icon') as HTMLElement;
124+
emitEvent(navIcon, 'click');
125+
expect(mockFoundation.handleNavigationClick)
126+
.toHaveBeenCalledWith(jasmine.any(Object));
127+
expect(mockFoundation.handleNavigationClick).toHaveBeenCalledTimes(1);
128+
});
129+
130+
it('scroll event triggers #foundation.handleTargetScroll', () => {
131+
const {mockFoundation} = setupTest();
132+
emitEvent(window, 'scroll');
133+
expect(mockFoundation.handleTargetScroll)
134+
.toHaveBeenCalledWith(jasmine.any(Object));
135+
expect(mockFoundation.handleTargetScroll).toHaveBeenCalledTimes(1);
136+
});
137+
138+
it('resize event triggers #foundation.handleWindowResize', () => {
139+
const {mockFoundation} = setupTest();
140+
emitEvent(window, 'resize');
141+
expect(mockFoundation.handleWindowResize)
142+
.toHaveBeenCalledWith(jasmine.any(Object));
143+
expect(mockFoundation.handleWindowResize).toHaveBeenCalledTimes(1);
144+
});
145+
146+
it('destroy destroys icon ripples', () => {
147+
const {component} = setupTest();
148+
component.destroy();
149+
(component as any).iconRipples_.forEach((icon: MDCRipple) => {
150+
expect(icon.destroy).toHaveBeenCalled();
151+
});
152+
});
153+
154+
it('destroy destroys scroll event handler', () => {
155+
const {mockFoundation, component} = setupTest();
156+
component.destroy();
157+
emitEvent(window, 'scroll');
158+
expect(mockFoundation.handleTargetScroll)
159+
.not.toHaveBeenCalledWith(jasmine.any(Object));
160+
});
161+
162+
it('destroy destroys resize event handler', () => {
163+
const {mockFoundation, component} = setupTest();
164+
component.destroy();
165+
emitEvent(window, 'resize');
166+
expect(mockFoundation.handleWindowResize)
167+
.not.toHaveBeenCalledWith(jasmine.any(Object));
168+
});
169+
170+
it('destroy destroys handleNavigationClick handler', () => {
171+
const {mockFoundation, component, root} = setupTest();
172+
const navIcon =
173+
root.querySelector('.mdc-top-app-bar__navigation-icon') as HTMLElement;
174+
component.destroy();
175+
emitEvent(navIcon, 'resize');
176+
expect(mockFoundation.handleNavigationClick)
177+
.not.toHaveBeenCalledWith(jasmine.any(Object));
178+
});
179+
180+
it('#setScrollTarget deregisters and registers scroll handler on provided target',
181+
() => {
182+
const {component} = setupTest();
183+
const fakeTarget1 = document.createElement('div');
184+
const fakeTarget2 = document.createElement('div');
185+
186+
component.setScrollTarget(fakeTarget1);
187+
expect((component as any).scrollTarget_).toEqual(fakeTarget1);
188+
189+
component.setScrollTarget(fakeTarget2);
190+
191+
expect((component as any).scrollTarget_).toEqual(fakeTarget2);
192+
});
193+
194+
it('getDefaultFoundation returns the appropriate foundation for default',
195+
() => {
196+
const fixture = getFixture();
197+
const root = fixture.querySelector(strings.ROOT_SELECTOR) as HTMLElement;
198+
const component = new MDCTopAppBar(
199+
root, undefined, () => new FakeRipple());
200+
expect((component as any).foundation_ instanceof MDCTopAppBarFoundation)
201+
.toBe(true);
202+
expect(
203+
(component as any).foundation_ instanceof
204+
MDCShortTopAppBarFoundation)
205+
.toBe(false);
206+
expect(
207+
(component as any).foundation_ instanceof
208+
MDCFixedTopAppBarFoundation)
209+
.toBe(false);
210+
});
211+
212+
it('getDefaultFoundation returns the appropriate foundation for fixed',
213+
() => {
214+
const fixture = getFixture();
215+
const root = fixture.querySelector(strings.ROOT_SELECTOR) as HTMLElement;
216+
root.classList.add(MDCTopAppBarFoundation.cssClasses.FIXED_CLASS);
217+
const component = new MDCTopAppBar(
218+
root, undefined, () => new FakeRipple());
219+
expect(
220+
(component as any).foundation_ instanceof
221+
MDCShortTopAppBarFoundation)
222+
.toBe(false);
223+
expect(
224+
(component as any).foundation_ instanceof
225+
MDCFixedTopAppBarFoundation)
226+
.toBe(true);
227+
});
228+
229+
it('getDefaultFoundation returns the appropriate foundation for short',
230+
() => {
231+
const fixture = getFixture();
232+
const root = fixture.querySelector(strings.ROOT_SELECTOR) as HTMLElement;
233+
root.classList.add(MDCTopAppBarFoundation.cssClasses.SHORT_CLASS);
234+
const component = new MDCTopAppBar(
235+
root, undefined, () => new FakeRipple());
236+
expect(
237+
(component as any).foundation_ instanceof
238+
MDCShortTopAppBarFoundation)
239+
.toBe(true);
240+
expect(
241+
(component as any).foundation_ instanceof
242+
MDCFixedTopAppBarFoundation)
243+
.toBe(false);
244+
});
245+
246+
it('adapter#hasClass returns true if the root element has specified class',
247+
() => {
248+
const {root, component} = setupTest();
249+
root.classList.add('foo');
250+
expect(
251+
(component.getDefaultFoundation() as any).adapter_.hasClass('foo'))
252+
.toBe(true);
253+
});
254+
255+
it('adapter#hasClass returns false if the root element does not have specified class',
256+
() => {
257+
const {component} = setupTest();
258+
expect(
259+
(component.getDefaultFoundation() as any).adapter_.hasClass('foo'))
260+
.toBe(false);
261+
});
262+
263+
it('adapter#addClass adds a class to the root element', () => {
264+
const {root, component} = setupTest();
265+
(component.getDefaultFoundation() as any).adapter_.addClass('foo');
266+
expect(root.classList.contains('foo')).toBe(true);
267+
});
268+
269+
it('adapter#removeClass removes a class from the root element', () => {
270+
const {root, component} = setupTest();
271+
root.classList.add('foo');
272+
(component.getDefaultFoundation() as any).adapter_.removeClass('foo');
273+
expect(root.classList.contains('foo')).toBe(false);
274+
});
275+
276+
it('adapter#setStyle sets a style attribute on the root element', () => {
277+
const {root, component} = setupTest();
278+
expect(root.style.getPropertyValue('top') === '1px').toBe(false);
279+
(component.getDefaultFoundation() as any).adapter_.setStyle('top', '1px');
280+
expect(root.style.getPropertyValue('top') === '1px').toBe(true);
281+
});
282+
283+
it('adapter#getViewportScrollY returns scroll distance', () => {
284+
const {component} = setupTest();
285+
expect(
286+
(component.getDefaultFoundation() as any).adapter_.getViewportScrollY())
287+
.toEqual(window.pageYOffset);
288+
});
289+
290+
it('adapter#getViewportScrollY returns scroll distance when scrollTarget_ is not window',
291+
() => {
292+
const {component} = setupTest();
293+
const mockContent = {addEventListener: () => {}, scrollTop: 20} as any;
294+
component.setScrollTarget(mockContent);
295+
expect((component.getDefaultFoundation() as any)
296+
.adapter_.getViewportScrollY())
297+
.toEqual(mockContent.scrollTop);
298+
});
299+
300+
it('adapter#getTotalActionItems returns the number of action items on the opposite side of the menu',
301+
() => {
302+
const {root, component} = setupTest();
303+
const adapterReturn = (component.getDefaultFoundation() as any)
304+
.adapter_.getTotalActionItems();
305+
const actual =
306+
root.querySelectorAll(strings.ACTION_ITEM_SELECTOR).length;
307+
expect(adapterReturn).toEqual(actual);
308+
});
309+
310+
it('adapter#notifyNavigationIconClicked emits the NAVIGATION_EVENT', () => {
311+
const {component} = setupTest();
312+
const callback = jasmine.createSpy('');
313+
component.listen(strings.NAVIGATION_EVENT, callback);
314+
(component.getDefaultFoundation() as any)
315+
.adapter_.notifyNavigationIconClicked();
316+
expect(callback).toHaveBeenCalledWith(jasmine.any(Object));
317+
expect(callback).toHaveBeenCalledTimes(1);
318+
});
319+
});

0 commit comments

Comments
 (0)