angular-docs-cn/packages/common/test/viewport_scroller_spec.ts

148 lines
4.8 KiB
TypeScript

/**
* @license
* Copyright Google LLC All Rights Reserved.
*
* 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
*/
import {describe, expect, it} from '@angular/core/testing/src/testing_internal';
import {browserDetection} from '@angular/platform-browser/testing/src/browser_util';
import {BrowserViewportScroller, ViewportScroller} from '../src/viewport_scroller';
describe('BrowserViewportScroller', () => {
describe('setHistoryScrollRestoration', () => {
let scroller: ViewportScroller;
let windowSpy: any;
beforeEach(() => {
windowSpy =
jasmine.createSpyObj('window', ['history', 'scrollTo', 'pageXOffset', 'pageYOffset']);
windowSpy.history.scrollRestoration = 'auto';
scroller = new BrowserViewportScroller(document, windowSpy);
});
function createNonWritableScrollRestoration() {
Object.defineProperty(windowSpy.history, 'scrollRestoration', {
value: 'auto',
configurable: true,
});
}
it('should not crash when scrollRestoration is not writable', () => {
createNonWritableScrollRestoration();
expect(() => scroller.setHistoryScrollRestoration('manual')).not.toThrow();
});
it('should still allow scrolling if scrollRestoration is not writable', () => {
createNonWritableScrollRestoration();
scroller.scrollToPosition([10, 10]);
expect(windowSpy.scrollTo as jasmine.Spy).toHaveBeenCalledWith(10, 10);
});
});
describe('scrollToAnchor', () => {
// Testing scroll behavior does not make sense outside a browser
if (isNode) return;
const anchor = 'anchor';
let scroller: BrowserViewportScroller;
beforeEach(() => {
scroller = new BrowserViewportScroller(document, window);
scroller.scrollToPosition([0, 0]);
});
it('should scroll when element with matching id is found', () => {
const {anchorNode, cleanup} = createTallElement();
anchorNode.id = anchor;
scroller.scrollToAnchor(anchor);
expect(scroller.getScrollPosition()[1]).not.toEqual(0);
cleanup();
});
it('should scroll when anchor with matching name is found', () => {
const {anchorNode, cleanup} = createTallElement();
anchorNode.name = anchor;
scroller.scrollToAnchor(anchor);
expect(scroller.getScrollPosition()[1]).not.toEqual(0);
cleanup();
});
it('should not scroll when no matching element is found', () => {
const {cleanup} = createTallElement();
scroller.scrollToAnchor(anchor);
expect(scroller.getScrollPosition()[1]).toEqual(0);
cleanup();
});
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;
}
});
});