* rrdom: add a diff function for properties * implement diffChildren function and unit tests * finish basic functions of diff algorithm * fix several bugs in the diff algorithm * replace the virtual parent optimization in applyMutation() * fix: moveAndHover after the diff algorithm is executed * replace virtual style map with rrdom cssom version has to be above 0.5.0 to pass virtual style tests * fix: failed virtual style tests in replayer.test.ts * fix: failed polyfill tests caused by nodejs compatibility of different versions * fix: svg viewBox attribute doesn't work Cause the attribute viewBox is case sensitive, set value for viewbox doesn't work * feat: replace treeIndex optimization with rrdom * fix bug of diffProps and disable smooth scrolling animation in fast-forward mode * feat: add iframe support * fix: @rollup/plugin-typescript build errors in rrweb-player Error: @rollup/plugin-typescript TS1371: This import is never used as a value and must use 'import type' because the 'importsNotUsedAsValues' is set to 'error' * fix: bug when fast-forward input events and add test for it * add test for fast-forward scroll events * fix: custom style rules don't get inserted into some iframe elements * code style tweak * fix: enable to diff iframe elements * fix the jest error "Unexpected token 'export'" * try to fix build error of rrweb-player * correct the attributes definition in rrdom * fix: custom style rules are not inserted in some iframes * add support for shadow dom * add support for MediaInteraction * add canvas support * fix unit test error in rrdom * add support for Text, Comment * try to refactor RRDom * refactor RRDom to reduce duplicate code * rename document-browser to virtual-dom * increase the test coverage for document.ts and add ownerDocument for it * Merge branch 'master' into virtual-dom * add more test for virtual-dom.ts * use cssstyle in document-nodejs * fix: bundle error * improve document-nodejs * enable to diff scroll positions of an element * rename rrdom to virtualDom for more readability and make the tree public * revert unknown change * improve the css style parser for comments * improve code style * update typings * add handling for the case where legacy_missingNodeRetryMap is not empty * only import types from rrweb into rrdom * Apply suggestions from code review Co-authored-by: Justin Halsall <Juice10@users.noreply.github.com> * Apply suggestions from code review * fix building error in rrweb * add a method setDefaultSN to set a default value for a RRNode's __sn * fix rrweb test error and bump up other packages * add support for custom property of css styles * add a switch for virtual-dom optimization * Apply suggestions from code review 1. add an enum type for NodeType 2. rename nodeType from rrweb-snapshot to RRNodeType 3. rename notSerializedId to unserializedId 4. add comments for some confusing variables * adapt changes of #865 to virtual-dom and improve the test case for more coverage * apply review suggestions https://github.com/rrweb-io/rrweb/pull/853#pullrequestreview-922474953 * tweak the diff algorithm * add description of the flag useVirtualDom and remove outdated logConfig * Remove console.log * Contain changes to document * Upgrade rollup to 2.70.2 * Revert "Upgrade rollup to 2.70.2" This reverts commit b1be81a2a76565935c9dc391f31beb7f64d25956. * Fix type checking rrdom * Fix typing error while bundling * Fix tslib error on build Rollup would output the following error: `semantic error TS2343: This syntax requires an imported helper named '__spreadArray' which does not exist in 'tslib'. Consider upgrading your version of 'tslib'.` * Increase memory limit for rollup * Use esbuild for bundling Speeds up bundling significantly * Avoid circular dependencies and import un-bundled rrdom * Fix imports * Revert back to pre-esbuild This reverts the following commits: b7b3c8dbaa551a0129da1477136b1baaad28e6e1 72e23b8e27f9030d911358d3a17fe5ad1b3b5d4f 85d600a20c56cfa764cf1f858932ba14e67b1d23 61e1a5d323212ca8fbe0569e0b3062ddd53fc612 * Set node to lts (12 is no longer supported) * Speed up bundling and use less memory This fixes the out of memory errors happening while bundling * remove __sn from rrdom * fix typo * test: add a test case for StyleSheet mutation exceptions while fast-forwarding * rename Array.prototype.slice.call() to Array.from() * improve test cases * fix: PR #887 in 'virtual-dom' branch * apply justin's suggestion on 'Array.from' refactor related commit 0f6729d27a323260b36fbe79485a86715c0bc98a * improve import code structure Co-authored-by: Yun Feng <yun.feng@anu.edu.au>
923 lines
36 KiB
TypeScript
923 lines
36 KiB
TypeScript
/**
|
|
* @jest-environment jsdom
|
|
*/
|
|
import { NodeType as RRNodeType } from 'rrweb-snapshot';
|
|
import {
|
|
BaseRRDocumentImpl,
|
|
BaseRRDocumentTypeImpl,
|
|
BaseRRElementImpl,
|
|
BaseRRMediaElementImpl,
|
|
BaseRRNode,
|
|
IRRDocumentType,
|
|
} from '../src/document';
|
|
|
|
describe('Basic RRDocument implementation', () => {
|
|
const RRNode = BaseRRNode;
|
|
const RRDocument = BaseRRDocumentImpl(RRNode);
|
|
const RRDocumentType = BaseRRDocumentTypeImpl(RRNode);
|
|
const RRElement = BaseRRElementImpl(RRNode);
|
|
class RRMediaElement extends BaseRRMediaElementImpl(RRElement) {}
|
|
|
|
describe('Basic RRNode implementation', () => {
|
|
it('should have basic properties', () => {
|
|
const node = new RRNode();
|
|
expect(node.parentNode).toEqual(null);
|
|
expect(node.parentElement).toEqual(null);
|
|
expect(node.childNodes).toBeInstanceOf(Array);
|
|
expect(node.childNodes.length).toBe(0);
|
|
expect(node.ownerDocument).toBeUndefined();
|
|
expect(node.textContent).toBeUndefined();
|
|
expect(node.RRNodeType).toBeUndefined();
|
|
expect(node.nodeType).toBeUndefined();
|
|
expect(node.nodeName).toBeUndefined();
|
|
expect(node.ELEMENT_NODE).toBe(document.ELEMENT_NODE);
|
|
expect(node.TEXT_NODE).toBe(document.TEXT_NODE);
|
|
expect(node.firstChild).toBeNull();
|
|
expect(node.lastChild).toBeNull();
|
|
expect(node.nextSibling).toBeNull();
|
|
expect(node.contains).toBeDefined();
|
|
expect(node.appendChild).toBeDefined();
|
|
expect(node.insertBefore).toBeDefined();
|
|
expect(node.removeChild).toBeDefined();
|
|
expect(node.toString()).toEqual('RRNode');
|
|
});
|
|
|
|
it('can get first child node', () => {
|
|
const parentNode = new RRNode();
|
|
const childNode1 = new RRNode();
|
|
const childNode2 = new RRNode();
|
|
expect(parentNode.firstChild).toBeNull();
|
|
parentNode.childNodes = [childNode1];
|
|
expect(parentNode.firstChild).toBe(childNode1);
|
|
parentNode.childNodes = [childNode1, childNode2];
|
|
expect(parentNode.firstChild).toBe(childNode1);
|
|
parentNode.childNodes = [childNode2, childNode1];
|
|
expect(parentNode.firstChild).toBe(childNode2);
|
|
});
|
|
|
|
it('can get last child node', () => {
|
|
const parentNode = new RRNode();
|
|
const childNode1 = new RRNode();
|
|
const childNode2 = new RRNode();
|
|
expect(parentNode.lastChild).toBeNull();
|
|
parentNode.childNodes = [childNode1];
|
|
expect(parentNode.lastChild).toBe(childNode1);
|
|
parentNode.childNodes = [childNode1, childNode2];
|
|
expect(parentNode.lastChild).toBe(childNode2);
|
|
parentNode.childNodes = [childNode2, childNode1];
|
|
expect(parentNode.lastChild).toBe(childNode1);
|
|
});
|
|
|
|
it('can get nextSibling', () => {
|
|
const parentNode = new RRNode();
|
|
const childNode1 = new RRNode();
|
|
const childNode2 = new RRNode();
|
|
expect(parentNode.nextSibling).toBeNull();
|
|
expect(childNode1.nextSibling).toBeNull();
|
|
childNode1.parentNode = parentNode;
|
|
parentNode.childNodes = [childNode1];
|
|
expect(childNode1.nextSibling).toBeNull();
|
|
childNode2.parentNode = parentNode;
|
|
parentNode.childNodes = [childNode1, childNode2];
|
|
expect(childNode1.nextSibling).toBe(childNode2);
|
|
expect(childNode2.nextSibling).toBeNull();
|
|
});
|
|
|
|
it('should return whether the node contains another node', () => {
|
|
const parentNode = new RRNode();
|
|
const childNode1 = new RRNode();
|
|
const childNode2 = new RRNode();
|
|
parentNode.childNodes = [childNode1];
|
|
expect(parentNode.contains(childNode1)).toBeTruthy();
|
|
expect(parentNode.contains(childNode2)).toBeFalsy();
|
|
childNode1.childNodes = [childNode2];
|
|
expect(parentNode.contains(childNode2)).toBeTruthy();
|
|
});
|
|
|
|
it('should not implement appendChild', () => {
|
|
const parentNode = new RRNode();
|
|
const childNode = new RRNode();
|
|
expect(() =>
|
|
parentNode.appendChild(childNode),
|
|
).toThrowErrorMatchingInlineSnapshot(
|
|
`"RRDomException: Failed to execute 'appendChild' on 'RRNode': This RRNode type does not support this method."`,
|
|
);
|
|
});
|
|
|
|
it('should not implement insertBefore', () => {
|
|
const parentNode = new RRNode();
|
|
const childNode = new RRNode();
|
|
expect(() =>
|
|
parentNode.insertBefore(childNode, null),
|
|
).toThrowErrorMatchingInlineSnapshot(
|
|
`"RRDomException: Failed to execute 'insertBefore' on 'RRNode': This RRNode type does not support this method."`,
|
|
);
|
|
});
|
|
|
|
it('should not implement removeChild', () => {
|
|
const parentNode = new RRNode();
|
|
const childNode = new RRNode();
|
|
expect(() =>
|
|
parentNode.removeChild(childNode),
|
|
).toThrowErrorMatchingInlineSnapshot(
|
|
`"RRDomException: Failed to execute 'removeChild' on 'RRNode': This RRNode type does not support this method."`,
|
|
);
|
|
});
|
|
});
|
|
|
|
describe('Basic RRDocument implementation', () => {
|
|
it('should have basic properties', () => {
|
|
const node = new RRDocument();
|
|
expect(node.toString()).toEqual('RRDocument');
|
|
expect(node.parentNode).toEqual(null);
|
|
expect(node.parentElement).toEqual(null);
|
|
expect(node.childNodes).toBeInstanceOf(Array);
|
|
expect(node.childNodes.length).toBe(0);
|
|
expect(node.ownerDocument).toBeUndefined();
|
|
expect(node.textContent).toBeNull();
|
|
expect(node.RRNodeType).toBe(RRNodeType.Document);
|
|
expect(node.nodeType).toBe(document.nodeType);
|
|
expect(node.nodeName).toBe('#document');
|
|
expect(node.compatMode).toBe('CSS1Compat');
|
|
expect(node.ELEMENT_NODE).toBe(document.ELEMENT_NODE);
|
|
expect(node.TEXT_NODE).toBe(document.TEXT_NODE);
|
|
expect(node.firstChild).toBeNull();
|
|
expect(node.lastChild).toBeNull();
|
|
expect(node.nextSibling).toBeNull();
|
|
expect(node.contains).toBeDefined();
|
|
expect(node.appendChild).toBeDefined();
|
|
expect(node.insertBefore).toBeDefined();
|
|
expect(node.removeChild).toBeDefined();
|
|
expect(node.documentElement).toBeNull();
|
|
expect(node.body).toBeNull();
|
|
expect(node.head).toBeNull();
|
|
expect(node.implementation).toBe(node);
|
|
expect(node.firstElementChild).toBeNull();
|
|
expect(node.createDocument).toBeDefined();
|
|
expect(node.createDocumentType).toBeDefined();
|
|
expect(node.createElement).toBeDefined();
|
|
expect(node.createElementNS).toBeDefined();
|
|
expect(node.createTextNode).toBeDefined();
|
|
expect(node.createComment).toBeDefined();
|
|
expect(node.createCDATASection).toBeDefined();
|
|
expect(node.open).toBeDefined();
|
|
expect(node.close).toBeDefined();
|
|
expect(node.write).toBeDefined();
|
|
expect(node.toString()).toEqual('RRDocument');
|
|
});
|
|
|
|
it('can get documentElement', () => {
|
|
const node = new RRDocument();
|
|
expect(node.documentElement).toBeNull();
|
|
const element = node.createElement('html');
|
|
node.appendChild(element);
|
|
expect(node.documentElement).toBe(element);
|
|
});
|
|
|
|
it('can get head', () => {
|
|
const node = new RRDocument();
|
|
expect(node.head).toBeNull();
|
|
const element = node.createElement('html');
|
|
node.appendChild(element);
|
|
expect(node.head).toBeNull();
|
|
const head = node.createElement('head');
|
|
element.appendChild(head);
|
|
expect(node.head).toBe(head);
|
|
});
|
|
|
|
it('can get body', () => {
|
|
const node = new RRDocument();
|
|
expect(node.body).toBeNull();
|
|
const element = node.createElement('html');
|
|
node.appendChild(element);
|
|
expect(node.body).toBeNull();
|
|
const body = node.createElement('body');
|
|
element.appendChild(body);
|
|
expect(node.body).toBe(body);
|
|
const head = node.createElement('head');
|
|
element.appendChild(head);
|
|
expect(node.body).toBe(body);
|
|
});
|
|
|
|
it('can get firstElementChild', () => {
|
|
const node = new RRDocument();
|
|
expect(node.firstElementChild).toBeNull();
|
|
const element = node.createElement('html');
|
|
node.appendChild(element);
|
|
expect(node.firstElementChild).toBe(element);
|
|
});
|
|
|
|
it('can append child', () => {
|
|
const node = new RRDocument();
|
|
expect(node.firstElementChild).toBeNull();
|
|
|
|
const documentType = node.createDocumentType('html', '', '');
|
|
expect(node.appendChild(documentType)).toBe(documentType);
|
|
expect(node.childNodes[0]).toEqual(documentType);
|
|
expect(documentType.parentElement).toBeNull();
|
|
expect(documentType.parentNode).toBe(node);
|
|
expect(() =>
|
|
node.appendChild(documentType),
|
|
).toThrowErrorMatchingInlineSnapshot(
|
|
`"RRDomException: Failed to execute 'appendChild' on 'RRNode': Only one RRDoctype on RRDocument allowed."`,
|
|
);
|
|
|
|
const element = node.createElement('html');
|
|
expect(node.appendChild(element)).toBe(element);
|
|
expect(node.childNodes[1]).toEqual(element);
|
|
expect(element.parentElement).toBeNull();
|
|
expect(element.parentNode).toBe(node);
|
|
const div = node.createElement('div');
|
|
expect(() => node.appendChild(div)).toThrowErrorMatchingInlineSnapshot(
|
|
`"RRDomException: Failed to execute 'appendChild' on 'RRNode': Only one RRElement on RRDocument allowed."`,
|
|
);
|
|
});
|
|
|
|
it('can insert new child before an existing child', () => {
|
|
const node = new RRDocument();
|
|
const docType = node.createDocumentType('', '', '');
|
|
expect(() =>
|
|
node.insertBefore(node, docType),
|
|
).toThrowErrorMatchingInlineSnapshot(
|
|
`"Failed to execute 'insertBefore' on 'RRNode': The RRNode before which the new node is to be inserted is not a child of this RRNode."`,
|
|
);
|
|
expect(node.insertBefore(docType, null)).toBe(docType);
|
|
expect(() =>
|
|
node.insertBefore(docType, null),
|
|
).toThrowErrorMatchingInlineSnapshot(
|
|
`"RRDomException: Failed to execute 'insertBefore' on 'RRNode': Only one RRDoctype on RRDocument allowed."`,
|
|
);
|
|
node.removeChild(docType);
|
|
|
|
const documentElement = node.createElement('html');
|
|
expect(() =>
|
|
node.insertBefore(documentElement, docType),
|
|
).toThrowErrorMatchingInlineSnapshot(
|
|
`"Failed to execute 'insertBefore' on 'RRNode': The RRNode before which the new node is to be inserted is not a child of this RRNode."`,
|
|
);
|
|
expect(node.insertBefore(documentElement, null)).toBe(documentElement);
|
|
expect(() =>
|
|
node.insertBefore(documentElement, null),
|
|
).toThrowErrorMatchingInlineSnapshot(
|
|
`"RRDomException: Failed to execute 'insertBefore' on 'RRNode': Only one RRElement on RRDocument allowed."`,
|
|
);
|
|
expect(node.insertBefore(docType, documentElement)).toBe(docType);
|
|
expect(node.childNodes[0]).toBe(docType);
|
|
expect(node.childNodes[1]).toBe(documentElement);
|
|
expect(docType.parentElement).toBeNull();
|
|
expect(documentElement.parentElement).toBeNull();
|
|
expect(docType.parentNode).toBe(node);
|
|
expect(documentElement.parentNode).toBe(node);
|
|
});
|
|
|
|
it('can remove an existing child', () => {
|
|
const node = new RRDocument();
|
|
const documentType = node.createDocumentType('html', '', '');
|
|
const documentElement = node.createElement('html');
|
|
node.appendChild(documentType);
|
|
node.appendChild(documentElement);
|
|
expect(documentType.parentNode).toBe(node);
|
|
expect(documentElement.parentNode).toBe(node);
|
|
|
|
expect(() =>
|
|
node.removeChild(node.createElement('div')),
|
|
).toThrowErrorMatchingInlineSnapshot(
|
|
`"Failed to execute 'removeChild' on 'RRDocument': The RRNode to be removed is not a child of this RRNode."`,
|
|
);
|
|
expect(node.removeChild(documentType)).toBe(documentType);
|
|
expect(documentType.parentNode).toBeNull();
|
|
expect(node.removeChild(documentElement)).toBe(documentElement);
|
|
expect(documentElement.parentNode).toBeNull();
|
|
});
|
|
|
|
it('should implement create node functions', () => {
|
|
const node = new RRDocument();
|
|
expect(node.createDocument(null, '', null).RRNodeType).toEqual(
|
|
RRNodeType.Document,
|
|
);
|
|
expect(node.createDocumentType('', '', '').RRNodeType).toEqual(
|
|
RRNodeType.DocumentType,
|
|
);
|
|
expect(node.createElement('html').RRNodeType).toEqual(RRNodeType.Element);
|
|
expect(node.createElementNS('', 'html').RRNodeType).toEqual(
|
|
RRNodeType.Element,
|
|
);
|
|
expect(node.createTextNode('text').RRNodeType).toEqual(RRNodeType.Text);
|
|
expect(node.createComment('comment').RRNodeType).toEqual(
|
|
RRNodeType.Comment,
|
|
);
|
|
expect(node.createCDATASection('data').RRNodeType).toEqual(
|
|
RRNodeType.CDATA,
|
|
);
|
|
});
|
|
|
|
it('can close and open a RRDocument', () => {
|
|
const node = new RRDocument();
|
|
const documentType = node.createDocumentType('html', '', '');
|
|
node.appendChild(documentType);
|
|
expect(node.childNodes[0]).toBe(documentType);
|
|
expect(node.close());
|
|
expect(node.open());
|
|
expect(node.childNodes.length).toEqual(0);
|
|
});
|
|
|
|
it('can cover the usage of write() in rrweb-snapshot', () => {
|
|
const node = new RRDocument();
|
|
node.write(
|
|
'<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "">',
|
|
);
|
|
expect(node.childNodes.length).toBe(1);
|
|
let doctype = node.childNodes[0] as IRRDocumentType;
|
|
expect(doctype.RRNodeType).toEqual(RRNodeType.DocumentType);
|
|
expect(doctype.parentNode).toEqual(node);
|
|
expect(doctype.name).toEqual('html');
|
|
expect(doctype.publicId).toEqual(
|
|
'-//W3C//DTD XHTML 1.0 Transitional//EN',
|
|
);
|
|
expect(doctype.systemId).toEqual('');
|
|
|
|
node.write(
|
|
'<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN" "">',
|
|
);
|
|
expect(node.childNodes.length).toBe(1);
|
|
doctype = node.childNodes[0] as IRRDocumentType;
|
|
expect(doctype.RRNodeType).toEqual(RRNodeType.DocumentType);
|
|
expect(doctype.parentNode).toEqual(node);
|
|
expect(doctype.name).toEqual('html');
|
|
expect(doctype.publicId).toEqual('-//W3C//DTD HTML 4.0 Transitional//EN');
|
|
expect(doctype.systemId).toEqual('');
|
|
});
|
|
});
|
|
|
|
describe('Basic RRDocumentType implementation', () => {
|
|
it('should have basic properties', () => {
|
|
const name = 'name',
|
|
publicId = 'publicId',
|
|
systemId = 'systemId';
|
|
const node = new RRDocumentType(name, publicId, systemId);
|
|
|
|
expect(node.parentNode).toEqual(null);
|
|
expect(node.parentElement).toEqual(null);
|
|
expect(node.childNodes).toBeInstanceOf(Array);
|
|
expect(node.childNodes.length).toBe(0);
|
|
expect(node.ownerDocument).toBeUndefined();
|
|
expect(node.textContent).toBeNull();
|
|
expect(node.RRNodeType).toBe(RRNodeType.DocumentType);
|
|
expect(node.nodeType).toBe(document.DOCUMENT_TYPE_NODE);
|
|
expect(node.nodeName).toBe(name);
|
|
expect(node.ELEMENT_NODE).toBe(document.ELEMENT_NODE);
|
|
expect(node.TEXT_NODE).toBe(document.TEXT_NODE);
|
|
expect(node.firstChild).toBeNull();
|
|
expect(node.lastChild).toBeNull();
|
|
expect(node.nextSibling).toBeNull();
|
|
expect(node.contains).toBeDefined();
|
|
expect(node.appendChild).toBeDefined();
|
|
expect(node.insertBefore).toBeDefined();
|
|
expect(node.removeChild).toBeDefined();
|
|
expect(node.name).toBe(name);
|
|
expect(node.publicId).toBe(publicId);
|
|
expect(node.systemId).toBe(systemId);
|
|
expect(node.toString()).toEqual('RRDocumentType');
|
|
});
|
|
});
|
|
|
|
describe('Basic RRElement implementation', () => {
|
|
const document = new RRDocument();
|
|
|
|
it('should have basic properties', () => {
|
|
const node = document.createElement('div');
|
|
|
|
node.scrollLeft = 100;
|
|
node.scrollTop = 200;
|
|
node.attributes.id = 'id';
|
|
node.attributes.class = 'className';
|
|
expect(node.parentNode).toEqual(null);
|
|
expect(node.parentElement).toEqual(null);
|
|
expect(node.childNodes).toBeInstanceOf(Array);
|
|
expect(node.childNodes.length).toBe(0);
|
|
expect(node.ownerDocument).toBe(document);
|
|
expect(node.textContent).toEqual('');
|
|
expect(node.RRNodeType).toBe(RRNodeType.Element);
|
|
expect(node.nodeType).toBe(document.ELEMENT_NODE);
|
|
expect(node.nodeName).toBe('DIV');
|
|
expect(node.ELEMENT_NODE).toBe(document.ELEMENT_NODE);
|
|
expect(node.TEXT_NODE).toBe(document.TEXT_NODE);
|
|
expect(node.firstChild).toBeNull();
|
|
expect(node.lastChild).toBeNull();
|
|
expect(node.nextSibling).toBeNull();
|
|
expect(node.contains).toBeDefined();
|
|
expect(node.appendChild).toBeDefined();
|
|
expect(node.insertBefore).toBeDefined();
|
|
expect(node.removeChild).toBeDefined();
|
|
expect(node.tagName).toEqual('DIV');
|
|
expect(node.attributes).toEqual({ id: 'id', class: 'className' });
|
|
expect(node.shadowRoot).toBeNull();
|
|
expect(node.scrollLeft).toEqual(100);
|
|
expect(node.scrollTop).toEqual(200);
|
|
expect(node.id).toEqual('id');
|
|
expect(node.className).toEqual('className');
|
|
expect(node.classList).toBeDefined();
|
|
expect(node.style).toBeDefined();
|
|
expect(node.getAttribute).toBeDefined();
|
|
expect(node.setAttribute).toBeDefined();
|
|
expect(node.setAttributeNS).toBeDefined();
|
|
expect(node.removeAttribute).toBeDefined();
|
|
expect(node.attachShadow).toBeDefined();
|
|
expect(node.dispatchEvent).toBeDefined();
|
|
expect(node.dispatchEvent((null as unknown) as Event)).toBeTruthy();
|
|
expect(node.toString()).toEqual('DIV id="id" class="className" ');
|
|
});
|
|
|
|
it('can get textContent', () => {
|
|
const node = document.createElement('div');
|
|
node.appendChild(document.createTextNode('text1 '));
|
|
node.appendChild(document.createTextNode('text2'));
|
|
expect(node.textContent).toEqual('text1 text2');
|
|
});
|
|
|
|
it('can set textContent', () => {
|
|
const node = document.createElement('div');
|
|
node.appendChild(document.createTextNode('text1 '));
|
|
node.appendChild(document.createTextNode('text2'));
|
|
expect(node.textContent).toEqual('text1 text2');
|
|
node.textContent = 'new text';
|
|
expect(node.textContent).toEqual('new text');
|
|
});
|
|
|
|
it('can get id', () => {
|
|
const node = document.createElement('div');
|
|
expect(node.id).toEqual('');
|
|
node.attributes.id = 'idName';
|
|
expect(node.id).toEqual('idName');
|
|
});
|
|
|
|
it('can get className', () => {
|
|
const node = document.createElement('div');
|
|
expect(node.className).toEqual('');
|
|
node.attributes.class = 'className';
|
|
expect(node.className).toEqual('className');
|
|
});
|
|
|
|
it('can get classList', () => {
|
|
const node = document.createElement('div');
|
|
const classList = node.classList;
|
|
expect(classList.add).toBeDefined();
|
|
expect(classList.remove).toBeDefined();
|
|
});
|
|
|
|
it('classList can add class name', () => {
|
|
const node = document.createElement('div');
|
|
expect(node.className).toEqual('');
|
|
const classList = node.classList;
|
|
classList.add('c1');
|
|
expect(node.className).toEqual('c1');
|
|
classList.add('c2');
|
|
expect(node.className).toEqual('c1 c2');
|
|
classList.add('c2');
|
|
expect(node.className).toEqual('c1 c2');
|
|
});
|
|
|
|
it('classList can remove class name', () => {
|
|
const node = document.createElement('div');
|
|
expect(node.className).toEqual('');
|
|
const classList = node.classList;
|
|
classList.add('c1', 'c2', 'c3');
|
|
expect(node.className).toEqual('c1 c2 c3');
|
|
classList.remove('c2');
|
|
expect(node.className).toEqual('c1 c3');
|
|
classList.remove('c3');
|
|
expect(node.className).toEqual('c1');
|
|
classList.remove('c1');
|
|
expect(node.className).toEqual('');
|
|
classList.remove('c1');
|
|
expect(node.className).toEqual('');
|
|
});
|
|
|
|
it('classList can remove duplicate class names', () => {
|
|
const node = document.createElement('div');
|
|
expect(node.className).toEqual('');
|
|
node.setAttribute('class', 'c1 c1 c1');
|
|
expect(node.className).toEqual('c1 c1 c1');
|
|
const classList = node.classList;
|
|
classList.remove('c1');
|
|
expect(node.className).toEqual('');
|
|
});
|
|
|
|
it('can get CSS style declaration', () => {
|
|
const node = document.createElement('div');
|
|
const style = node.style;
|
|
expect(style).toBeDefined();
|
|
expect(style.setProperty).toBeDefined();
|
|
expect(style.removeProperty).toBeDefined();
|
|
|
|
node.attributes.style =
|
|
'color: blue; background-color: red; width: 78%; height: 50vh !important;';
|
|
expect(node.style.color).toBe('blue');
|
|
expect(node.style.backgroundColor).toBe('red');
|
|
expect(node.style.width).toBe('78%');
|
|
expect(node.style.height).toBe('50vh !important');
|
|
});
|
|
|
|
it('can set CSS property', () => {
|
|
const node = document.createElement('div');
|
|
const style = node.style;
|
|
style.setProperty('color', 'red');
|
|
expect(node.attributes.style).toEqual('color: red;');
|
|
// camelCase style is unacceptable
|
|
style.setProperty('backgroundColor', 'blue');
|
|
expect(node.attributes.style).toEqual('color: red;');
|
|
style.setProperty('height', '50vh', 'important');
|
|
expect(node.attributes.style).toEqual(
|
|
'color: red; height: 50vh !important;',
|
|
);
|
|
|
|
// kebab-case
|
|
style.setProperty('background-color', 'red');
|
|
expect(node.attributes.style).toEqual(
|
|
'color: red; height: 50vh !important; background-color: red;',
|
|
);
|
|
|
|
// remove the property
|
|
style.setProperty('background-color', null);
|
|
expect(node.attributes.style).toEqual(
|
|
'color: red; height: 50vh !important;',
|
|
);
|
|
});
|
|
|
|
it('can remove CSS property', () => {
|
|
const node = document.createElement('div');
|
|
node.attributes.style =
|
|
'color: blue; background-color: red; width: 78%; height: 50vh !important;';
|
|
const style = node.style;
|
|
expect(style.removeProperty('color')).toEqual('blue');
|
|
expect(node.attributes.style).toEqual(
|
|
'background-color: red; width: 78%; height: 50vh !important;',
|
|
);
|
|
expect(style.removeProperty('height')).toEqual('50vh !important');
|
|
expect(node.attributes.style).toEqual(
|
|
'background-color: red; width: 78%;',
|
|
);
|
|
// kebab-case
|
|
expect(style.removeProperty('background-color')).toEqual('red');
|
|
expect(node.attributes.style).toEqual('width: 78%;');
|
|
style.setProperty('background-color', 'red');
|
|
expect(node.attributes.style).toEqual(
|
|
'width: 78%; background-color: red;',
|
|
);
|
|
expect(style.removeProperty('backgroundColor')).toEqual('');
|
|
expect(node.attributes.style).toEqual(
|
|
'width: 78%; background-color: red;',
|
|
);
|
|
// remove a non-exist property
|
|
expect(style.removeProperty('margin')).toEqual('');
|
|
});
|
|
|
|
it('can parse more inline styles correctly', () => {
|
|
const node = document.createElement('div');
|
|
// general
|
|
node.attributes.style =
|
|
'display: inline-block; margin: 0 auto; border: 5px solid #BADA55; font-size: .75em; position:absolute;width: 33.3%; z-index:1337; font-family: "Goudy Bookletter 1911", Gill Sans Extrabold, sans-serif;';
|
|
|
|
let style = node.style;
|
|
expect(style.display).toEqual('inline-block');
|
|
expect(style.margin).toEqual('0 auto');
|
|
expect(style.border).toEqual('5px solid #BADA55');
|
|
expect(style.fontSize).toEqual('.75em');
|
|
expect(style.position).toEqual('absolute');
|
|
expect(style.width).toEqual('33.3%');
|
|
expect(style.zIndex).toEqual('1337');
|
|
expect(style.fontFamily).toEqual(
|
|
'"Goudy Bookletter 1911", Gill Sans Extrabold, sans-serif',
|
|
);
|
|
|
|
// multiple of same property
|
|
node.attributes.style = 'color: rgba(0,0,0,1);color:white';
|
|
style = node.style;
|
|
expect(style.color).toEqual('white');
|
|
|
|
// url
|
|
node.attributes.style =
|
|
'background-image: url("http://example.com/img.png")';
|
|
expect(node.style.backgroundImage).toEqual(
|
|
'url("http://example.com/img.png")',
|
|
);
|
|
|
|
// vendor prefixes
|
|
node.attributes.style = `
|
|
-moz-border-radius: 10px 5px;
|
|
-webkit-border-top-left-radius: 10px;
|
|
-webkit-border-bottom-left-radius: 5px;
|
|
border-radius: 10px 5px;
|
|
`;
|
|
style = node.style;
|
|
expect(style.MozBorderRadius).toEqual('10px 5px');
|
|
expect(style.WebkitBorderTopLeftRadius).toEqual('10px');
|
|
expect(style.WebkitBorderBottomLeftRadius).toEqual('5px');
|
|
expect(style.borderRadius).toEqual('10px 5px');
|
|
|
|
// comment
|
|
node.attributes.style =
|
|
'top: 0; /* comment1 */ bottom: /* comment2 */42rem;';
|
|
expect(node.style.top).toEqual('0');
|
|
expect(node.style.bottom).toEqual('42rem');
|
|
// empty comment
|
|
node.attributes.style = 'top: /**/0;';
|
|
expect(node.style.top).toEqual('0');
|
|
|
|
// custom property (variable)
|
|
node.attributes.style = '--custom-property: value';
|
|
expect(node.style['--custom-property']).toEqual('value');
|
|
|
|
// incomplete
|
|
node.attributes.style = 'overflow:';
|
|
expect(node.style.overflow).toBeUndefined();
|
|
});
|
|
|
|
it('can get attribute', () => {
|
|
const node = document.createElement('div');
|
|
node.attributes.class = 'className';
|
|
expect(node.getAttribute('class')).toEqual('className');
|
|
expect(node.getAttribute('id')).toEqual(null);
|
|
node.attributes.id = 'id';
|
|
expect(node.getAttribute('id')).toEqual('id');
|
|
});
|
|
|
|
it('can set attribute', () => {
|
|
const node = document.createElement('div');
|
|
expect(node.getAttribute('class')).toEqual(null);
|
|
node.setAttribute('class', 'className');
|
|
expect(node.getAttribute('class')).toEqual('className');
|
|
expect(node.getAttribute('id')).toEqual(null);
|
|
node.setAttribute('id', 'id');
|
|
expect(node.getAttribute('id')).toEqual('id');
|
|
});
|
|
|
|
it('can setAttributeNS', () => {
|
|
const node = document.createElement('div');
|
|
expect(node.getAttribute('class')).toEqual(null);
|
|
node.setAttributeNS('namespace', 'class', 'className');
|
|
expect(node.getAttribute('class')).toEqual('className');
|
|
expect(node.getAttribute('id')).toEqual(null);
|
|
node.setAttributeNS('namespace', 'id', 'id');
|
|
expect(node.getAttribute('id')).toEqual('id');
|
|
});
|
|
|
|
it('can remove attribute', () => {
|
|
const node = document.createElement('div');
|
|
node.setAttribute('class', 'className');
|
|
expect(node.getAttribute('class')).toEqual('className');
|
|
node.removeAttribute('class');
|
|
expect(node.getAttribute('class')).toEqual(null);
|
|
node.removeAttribute('id');
|
|
expect(node.getAttribute('id')).toEqual(null);
|
|
});
|
|
|
|
it('can attach shadow dom', () => {
|
|
const node = document.createElement('div');
|
|
expect(node.shadowRoot).toBeNull();
|
|
node.attachShadow({ mode: 'open' });
|
|
expect(node.shadowRoot).not.toBeNull();
|
|
expect(node.shadowRoot!.RRNodeType).toBe(RRNodeType.Element);
|
|
expect(node.shadowRoot!.tagName).toBe('SHADOWROOT');
|
|
expect(node.parentNode).toBeNull();
|
|
});
|
|
|
|
it('can append child', () => {
|
|
const node = document.createElement('div');
|
|
expect(node.childNodes.length).toBe(0);
|
|
|
|
const child1 = document.createComment('span');
|
|
expect(node.appendChild(child1)).toBe(child1);
|
|
expect(node.childNodes[0]).toEqual(child1);
|
|
expect(child1.parentElement).toBe(node);
|
|
expect(child1.parentNode).toBe(node);
|
|
|
|
const child2 = document.createElement('p');
|
|
expect(node.appendChild(child2)).toBe(child2);
|
|
expect(node.childNodes[1]).toEqual(child2);
|
|
expect(child2.parentElement).toBe(node);
|
|
expect(child2.parentNode).toBe(node);
|
|
});
|
|
|
|
it('can insert new child before an existing child', () => {
|
|
const node = document.createElement('div');
|
|
const child1 = document.createElement('h1');
|
|
const child2 = document.createElement('h2');
|
|
expect(() =>
|
|
node.insertBefore(node, child1),
|
|
).toThrowErrorMatchingInlineSnapshot(
|
|
`"Failed to execute 'insertBefore' on 'RRNode': The RRNode before which the new node is to be inserted is not a child of this RRNode."`,
|
|
);
|
|
expect(node.insertBefore(child1, null)).toBe(child1);
|
|
expect(node.childNodes[0]).toBe(child1);
|
|
expect(child1.parentNode).toBe(node);
|
|
expect(child1.parentElement).toBe(node);
|
|
|
|
expect(node.insertBefore(child2, child1)).toBe(child2);
|
|
expect(node.childNodes.length).toBe(2);
|
|
expect(node.childNodes[0]).toBe(child2);
|
|
expect(node.childNodes[1]).toBe(child1);
|
|
expect(child2.parentNode).toBe(node);
|
|
expect(child2.parentElement).toBe(node);
|
|
});
|
|
|
|
it('can remove an existing child', () => {
|
|
const node = document.createElement('div');
|
|
const child1 = document.createElement('h1');
|
|
const child2 = document.createElement('h2');
|
|
node.appendChild(child1);
|
|
node.appendChild(child2);
|
|
expect(node.childNodes.length).toBe(2);
|
|
expect(child1.parentNode).toBe(node);
|
|
expect(child2.parentNode).toBe(node);
|
|
expect(child1.parentElement).toBe(node);
|
|
expect(child2.parentElement).toBe(node);
|
|
|
|
expect(() =>
|
|
node.removeChild(document.createElement('div')),
|
|
).toThrowErrorMatchingInlineSnapshot(
|
|
`"Failed to execute 'removeChild' on 'RRElement': The RRNode to be removed is not a child of this RRNode."`,
|
|
);
|
|
expect(node.removeChild(child1)).toBe(child1);
|
|
expect(child1.parentNode).toBeNull();
|
|
expect(child1.parentElement).toBeNull();
|
|
expect(node.childNodes.length).toBe(1);
|
|
expect(node.removeChild(child2)).toBe(child2);
|
|
expect(node.childNodes.length).toBe(0);
|
|
expect(child2.parentNode).toBeNull();
|
|
expect(child2.parentElement).toBeNull();
|
|
});
|
|
});
|
|
|
|
describe('Basic RRText implementation', () => {
|
|
const dom = new RRDocument();
|
|
|
|
it('should have basic properties', () => {
|
|
const node = dom.createTextNode('text');
|
|
|
|
expect(node.parentNode).toEqual(null);
|
|
expect(node.parentElement).toEqual(null);
|
|
expect(node.childNodes).toBeInstanceOf(Array);
|
|
expect(node.childNodes.length).toBe(0);
|
|
expect(node.ownerDocument).toBe(dom);
|
|
expect(node.textContent).toEqual('text');
|
|
expect(node.RRNodeType).toBe(RRNodeType.Text);
|
|
expect(node.nodeType).toBe(document.TEXT_NODE);
|
|
expect(node.nodeName).toBe('#text');
|
|
expect(node.ELEMENT_NODE).toBe(document.ELEMENT_NODE);
|
|
expect(node.TEXT_NODE).toBe(document.TEXT_NODE);
|
|
expect(node.firstChild).toBeNull();
|
|
expect(node.lastChild).toBeNull();
|
|
expect(node.nextSibling).toBeNull();
|
|
expect(node.contains).toBeDefined();
|
|
expect(node.appendChild).toBeDefined();
|
|
expect(node.insertBefore).toBeDefined();
|
|
expect(node.removeChild).toBeDefined();
|
|
expect(node.toString()).toEqual('RRText text="text"');
|
|
});
|
|
|
|
it('can set textContent', () => {
|
|
const node = dom.createTextNode('text');
|
|
expect(node.textContent).toEqual('text');
|
|
node.textContent = 'new text';
|
|
expect(node.textContent).toEqual('new text');
|
|
});
|
|
});
|
|
|
|
describe('Basic RRComment implementation', () => {
|
|
const dom = new RRDocument();
|
|
|
|
it('should have basic properties', () => {
|
|
const node = dom.createComment('comment');
|
|
|
|
expect(node.parentNode).toEqual(null);
|
|
expect(node.parentElement).toEqual(null);
|
|
expect(node.childNodes).toBeInstanceOf(Array);
|
|
expect(node.childNodes.length).toBe(0);
|
|
expect(node.ownerDocument).toBe(dom);
|
|
expect(node.textContent).toEqual('comment');
|
|
expect(node.RRNodeType).toBe(RRNodeType.Comment);
|
|
expect(node.nodeType).toBe(document.COMMENT_NODE);
|
|
expect(node.nodeName).toBe('#comment');
|
|
expect(node.ELEMENT_NODE).toBe(document.ELEMENT_NODE);
|
|
expect(node.TEXT_NODE).toBe(document.TEXT_NODE);
|
|
expect(node.firstChild).toBeNull();
|
|
expect(node.lastChild).toBeNull();
|
|
expect(node.nextSibling).toBeNull();
|
|
expect(node.contains).toBeDefined();
|
|
expect(node.appendChild).toBeDefined();
|
|
expect(node.insertBefore).toBeDefined();
|
|
expect(node.removeChild).toBeDefined();
|
|
expect(node.toString()).toEqual('RRComment text="comment"');
|
|
});
|
|
|
|
it('can set textContent', () => {
|
|
const node = dom.createComment('comment');
|
|
expect(node.textContent).toEqual('comment');
|
|
node.textContent = 'new comment';
|
|
expect(node.textContent).toEqual('new comment');
|
|
});
|
|
});
|
|
|
|
describe('Basic RRCDATASection implementation', () => {
|
|
const dom = new RRDocument();
|
|
|
|
it('should have basic properties', () => {
|
|
const node = dom.createCDATASection('data');
|
|
|
|
expect(node.parentNode).toEqual(null);
|
|
expect(node.parentElement).toEqual(null);
|
|
expect(node.childNodes).toBeInstanceOf(Array);
|
|
expect(node.childNodes.length).toBe(0);
|
|
expect(node.ownerDocument).toBe(dom);
|
|
expect(node.textContent).toEqual('data');
|
|
expect(node.RRNodeType).toBe(RRNodeType.CDATA);
|
|
expect(node.nodeType).toBe(document.CDATA_SECTION_NODE);
|
|
expect(node.nodeName).toBe('#cdata-section');
|
|
expect(node.ELEMENT_NODE).toBe(document.ELEMENT_NODE);
|
|
expect(node.TEXT_NODE).toBe(document.TEXT_NODE);
|
|
expect(node.firstChild).toBeNull();
|
|
expect(node.lastChild).toBeNull();
|
|
expect(node.nextSibling).toBeNull();
|
|
expect(node.contains).toBeDefined();
|
|
expect(node.appendChild).toBeDefined();
|
|
expect(node.insertBefore).toBeDefined();
|
|
expect(node.removeChild).toBeDefined();
|
|
expect(node.toString()).toEqual('RRCDATASection data="data"');
|
|
});
|
|
|
|
it('can set textContent', () => {
|
|
const node = dom.createCDATASection('data');
|
|
expect(node.textContent).toEqual('data');
|
|
node.textContent = 'new data';
|
|
expect(node.textContent).toEqual('new data');
|
|
});
|
|
});
|
|
|
|
describe('Basic RRMediaElement implementation', () => {
|
|
it('should have basic properties', () => {
|
|
const node = new RRMediaElement('video');
|
|
node.scrollLeft = 100;
|
|
node.scrollTop = 200;
|
|
expect(node.parentNode).toEqual(null);
|
|
expect(node.parentElement).toEqual(null);
|
|
expect(node.childNodes).toBeInstanceOf(Array);
|
|
expect(node.childNodes.length).toBe(0);
|
|
expect(node.ownerDocument).toBeUndefined();
|
|
expect(node.textContent).toEqual('');
|
|
expect(node.RRNodeType).toBe(RRNodeType.Element);
|
|
expect(node.nodeType).toBe(document.ELEMENT_NODE);
|
|
expect(node.ELEMENT_NODE).toBe(document.ELEMENT_NODE);
|
|
expect(node.TEXT_NODE).toBe(document.TEXT_NODE);
|
|
expect(node.firstChild).toBeNull();
|
|
expect(node.nextSibling).toBeNull();
|
|
expect(node.contains).toBeDefined();
|
|
expect(node.appendChild).toBeDefined();
|
|
expect(node.insertBefore).toBeDefined();
|
|
expect(node.removeChild).toBeDefined();
|
|
expect(node.tagName).toEqual('VIDEO');
|
|
expect(node.attributes).toEqual({});
|
|
expect(node.shadowRoot).toBeNull();
|
|
expect(node.scrollLeft).toEqual(100);
|
|
expect(node.scrollTop).toEqual(200);
|
|
expect(node.id).toEqual('');
|
|
expect(node.className).toEqual('');
|
|
expect(node.classList).toBeDefined();
|
|
expect(node.style).toBeDefined();
|
|
expect(node.getAttribute).toBeDefined();
|
|
expect(node.setAttribute).toBeDefined();
|
|
expect(node.setAttributeNS).toBeDefined();
|
|
expect(node.removeAttribute).toBeDefined();
|
|
expect(node.attachShadow).toBeDefined();
|
|
expect(node.dispatchEvent).toBeDefined();
|
|
expect(node.currentTime).toBeUndefined();
|
|
expect(node.volume).toBeUndefined();
|
|
expect(node.paused).toBeUndefined();
|
|
expect(node.muted).toBeUndefined();
|
|
expect(node.play).toBeDefined();
|
|
expect(node.pause).toBeDefined();
|
|
expect(node.toString()).toEqual('VIDEO ');
|
|
});
|
|
|
|
it('can play and pause the media', () => {
|
|
const node = new RRMediaElement('video');
|
|
expect(node.paused).toBeUndefined();
|
|
node.play();
|
|
expect(node.paused).toBeFalsy();
|
|
node.pause();
|
|
expect(node.paused).toBeTruthy();
|
|
node.play();
|
|
expect(node.paused).toBeFalsy();
|
|
});
|
|
|
|
it('should not support attachShadow function', () => {
|
|
const node = new RRMediaElement('video');
|
|
expect(() =>
|
|
node.attachShadow({ mode: 'open' }),
|
|
).toThrowErrorMatchingInlineSnapshot(
|
|
`"RRDomException: Failed to execute 'attachShadow' on 'RRElement': This RRElement does not support attachShadow"`,
|
|
);
|
|
});
|
|
});
|
|
});
|