2019-03-28 18:00:51 -04:00
|
|
|
/**
|
|
|
|
* @license
|
2020-05-19 15:08:49 -04:00
|
|
|
* Copyright Google LLC All Rights Reserved.
|
2019-03-28 18:00:51 -04:00
|
|
|
*
|
|
|
|
* Use of this source code is governed by an MIT-style license that can be
|
|
|
|
* found in the LICENSE file at https://angular.io/license
|
|
|
|
*/
|
|
|
|
|
2021-04-15 11:01:25 -04:00
|
|
|
import {browserDetection} from '@angular/platform-browser/testing/src/browser_util';
|
2019-03-28 18:00:51 -04:00
|
|
|
import {BrowserViewportScroller, ViewportScroller} from '../src/viewport_scroller';
|
|
|
|
|
2020-07-16 04:58:17 -04:00
|
|
|
describe('BrowserViewportScroller', () => {
|
2020-12-22 15:50:51 -05:00
|
|
|
describe('setHistoryScrollRestoration', () => {
|
|
|
|
let scroller: ViewportScroller;
|
|
|
|
let windowSpy: any;
|
2019-05-23 08:44:46 -04:00
|
|
|
|
2020-12-22 15:50:51 -05:00
|
|
|
beforeEach(() => {
|
|
|
|
windowSpy =
|
|
|
|
jasmine.createSpyObj('window', ['history', 'scrollTo', 'pageXOffset', 'pageYOffset']);
|
|
|
|
windowSpy.history.scrollRestoration = 'auto';
|
|
|
|
scroller = new BrowserViewportScroller(document, windowSpy);
|
|
|
|
});
|
2019-05-23 08:44:46 -04:00
|
|
|
|
2020-08-14 13:32:55 -04:00
|
|
|
function createNonWritableScrollRestoration() {
|
2019-05-23 08:44:46 -04:00
|
|
|
Object.defineProperty(windowSpy.history, 'scrollRestoration', {
|
|
|
|
value: 'auto',
|
|
|
|
configurable: true,
|
|
|
|
});
|
2020-08-14 13:32:55 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
it('should not crash when scrollRestoration is not writable', () => {
|
|
|
|
createNonWritableScrollRestoration();
|
2019-05-23 08:44:46 -04:00
|
|
|
expect(() => scroller.setHistoryScrollRestoration('manual')).not.toThrow();
|
|
|
|
});
|
2020-08-14 13:32:55 -04:00
|
|
|
|
|
|
|
it('should still allow scrolling if scrollRestoration is not writable', () => {
|
|
|
|
createNonWritableScrollRestoration();
|
|
|
|
scroller.scrollToPosition([10, 10]);
|
|
|
|
expect(windowSpy.scrollTo as jasmine.Spy).toHaveBeenCalledWith(10, 10);
|
|
|
|
});
|
2020-07-16 04:58:17 -04:00
|
|
|
});
|
|
|
|
|
|
|
|
describe('scrollToAnchor', () => {
|
2020-12-22 15:50:51 -05:00
|
|
|
// Testing scroll behavior does not make sense outside a browser
|
|
|
|
if (isNode) return;
|
2020-07-16 04:58:17 -04:00
|
|
|
const anchor = 'anchor';
|
2020-12-22 15:50:51 -05:00
|
|
|
let scroller: BrowserViewportScroller;
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
scroller = new BrowserViewportScroller(document, window);
|
|
|
|
scroller.scrollToPosition([0, 0]);
|
|
|
|
});
|
2020-07-16 04:58:17 -04:00
|
|
|
|
2020-12-22 15:50:51 -05:00
|
|
|
it('should scroll when element with matching id is found', () => {
|
2021-04-15 11:01:25 -04:00
|
|
|
const {anchorNode, cleanup} = createTallElement();
|
|
|
|
anchorNode.id = anchor;
|
2020-07-16 04:58:17 -04:00
|
|
|
scroller.scrollToAnchor(anchor);
|
2020-12-22 15:50:51 -05:00
|
|
|
expect(scroller.getScrollPosition()[1]).not.toEqual(0);
|
2021-04-15 11:01:25 -04:00
|
|
|
cleanup();
|
2020-07-16 04:58:17 -04:00
|
|
|
});
|
|
|
|
|
2020-12-22 15:50:51 -05:00
|
|
|
it('should scroll when anchor with matching name is found', () => {
|
2021-04-15 11:01:25 -04:00
|
|
|
const {anchorNode, cleanup} = createTallElement();
|
|
|
|
anchorNode.name = anchor;
|
2020-07-16 04:58:17 -04:00
|
|
|
scroller.scrollToAnchor(anchor);
|
2020-12-22 15:50:51 -05:00
|
|
|
expect(scroller.getScrollPosition()[1]).not.toEqual(0);
|
2021-04-15 11:01:25 -04:00
|
|
|
cleanup();
|
2020-07-16 04:58:17 -04:00
|
|
|
});
|
2019-05-23 08:44:46 -04:00
|
|
|
|
2020-12-22 15:50:51 -05:00
|
|
|
it('should not scroll when no matching element is found', () => {
|
2021-04-15 11:01:25 -04:00
|
|
|
const {cleanup} = createTallElement();
|
2020-07-16 04:58:17 -04:00
|
|
|
scroller.scrollToAnchor(anchor);
|
2020-12-22 15:50:51 -05:00
|
|
|
expect(scroller.getScrollPosition()[1]).toEqual(0);
|
2021-04-15 11:01:25 -04:00
|
|
|
cleanup();
|
2019-03-28 18:00:51 -04:00
|
|
|
});
|
2021-04-15 11:01:25 -04:00
|
|
|
|
|
|
|
it('should scroll when element with matching id is found inside the shadow DOM', () => {
|
|
|
|
// This test is only relevant for browsers that support shadow DOM.
|
|
|
|
if (!browserDetection.supportsShadowDom) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const {anchorNode, cleanup} = createTallElementWithShadowRoot();
|
|
|
|
anchorNode.id = anchor;
|
|
|
|
scroller.scrollToAnchor(anchor);
|
|
|
|
expect(scroller.getScrollPosition()[1]).not.toEqual(0);
|
|
|
|
cleanup();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should scroll when anchor with matching name is found inside the shadow DOM', () => {
|
|
|
|
// This test is only relevant for browsers that support shadow DOM.
|
|
|
|
if (!browserDetection.supportsShadowDom) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const {anchorNode, cleanup} = createTallElementWithShadowRoot();
|
|
|
|
anchorNode.name = anchor;
|
|
|
|
scroller.scrollToAnchor(anchor);
|
|
|
|
expect(scroller.getScrollPosition()[1]).not.toEqual(0);
|
|
|
|
cleanup();
|
|
|
|
});
|
|
|
|
|
|
|
|
function createTallElement() {
|
|
|
|
const tallItem = document.createElement('div');
|
|
|
|
tallItem.style.height = '3000px';
|
|
|
|
document.body.appendChild(tallItem);
|
|
|
|
const anchorNode = createAnchorNode();
|
|
|
|
document.body.appendChild(anchorNode);
|
|
|
|
|
|
|
|
return {
|
|
|
|
anchorNode,
|
|
|
|
cleanup: () => {
|
|
|
|
document.body.removeChild(tallItem);
|
|
|
|
document.body.removeChild(anchorNode);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
function createTallElementWithShadowRoot() {
|
|
|
|
const tallItem = document.createElement('div');
|
|
|
|
tallItem.style.height = '3000px';
|
|
|
|
document.body.appendChild(tallItem);
|
|
|
|
|
|
|
|
const elementWithShadowRoot = document.createElement('div');
|
|
|
|
const shadowRoot = elementWithShadowRoot.attachShadow({mode: 'open'});
|
|
|
|
const anchorNode = createAnchorNode();
|
|
|
|
shadowRoot.appendChild(anchorNode);
|
|
|
|
document.body.appendChild(elementWithShadowRoot);
|
|
|
|
|
|
|
|
return {
|
|
|
|
anchorNode,
|
|
|
|
cleanup: () => {
|
|
|
|
document.body.removeChild(tallItem);
|
|
|
|
document.body.removeChild(elementWithShadowRoot);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
function createAnchorNode() {
|
|
|
|
const anchorNode = document.createElement('a');
|
|
|
|
anchorNode.innerText = 'some link';
|
|
|
|
anchorNode.href = '#';
|
|
|
|
return anchorNode;
|
|
|
|
}
|
2019-03-28 18:00:51 -04:00
|
|
|
});
|
2020-07-16 04:58:17 -04:00
|
|
|
});
|