File: upgrade-custom-element-error-event.html

package info (click to toggle)
thunderbird 1%3A140.4.0esr-1
  • links: PTS, VCS
  • area: main
  • in suites: forky
  • size: 4,609,432 kB
  • sloc: cpp: 7,672,442; javascript: 5,901,613; ansic: 3,898,954; python: 1,413,343; xml: 653,997; asm: 462,286; java: 180,927; sh: 113,489; makefile: 20,460; perl: 14,288; objc: 13,059; yacc: 4,583; pascal: 3,352; lex: 1,720; ruby: 1,222; exp: 762; sql: 715; awk: 580; php: 436; lisp: 430; sed: 70; csh: 10
file content (84 lines) | stat: -rw-r--r-- 5,157 bytes parent folder | download | duplicates (11)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
<!DOCTYPE html>
<link rel="help" href="https://dom.spec.whatwg.org/#concept-create-element">
<script src="/resources/testharness.js"></script>
<script src="/resources/testharnessreport.js"></script>
<script src="../resources/custom-elements-helpers.js"></script>
<body>
<script>
// Returns a promise which will resolve with the next error event fired at any
// of `windows`, after the invocation of this function. Once one does, this
// function removes its listeners and produces that error event so that it can
// be examined (most notably for which global proxy it was targeted at).
async function nextErrorEvent(windows) {
  let listener;
  let p = new Promise((resolve, reject) => {
    listener = (event) => { resolve(event); event.preventDefault(); };
  });
  for (let w of windows) {
    w.addEventListener('error', listener);
  }
  try {
    return await p;
  } finally {
    for (let w of windows) {
      w.removeEventListener('error', listener);
    }
  }
}
const supportsCustomizedBuiltinElements = (function() {
  let supported = false;
  customElements.define('feature-detect', class extends HTMLDivElement {}, { get extends() { supported = true; return "div"; } });
  return supported;
})();

promise_test(async t => {
  let w = await create_window_in_test(t, `<script>self.MyElement = class extends HTMLElement { constructor() { throw new Error(); } };</`+`script>`);
  let w2 = await create_window_in_test(t);
  w2.customElements.define('my-element', w.MyElement);
  let nextErrorPromise = nextErrorEvent([self, w, w2]);
  let elem = w2.document.createElement('my-element');
  let errorEvent = await nextErrorPromise;
  assert_true(elem instanceof w2.HTMLUnknownElement, 'element should be an HTMLUnknownElement');
  if (errorEvent.error) {
    assert_true(errorEvent.error instanceof w.Error, 'error should be an instance created inside the MyElement constructor (if it was not muted)');
  }
  assert_equals(errorEvent.target, w, `error event should target definition's global but instead targets ${event.target === w2 ? 'document\'s global' : 'test harness global'}`);
}, 'autonomous: exception thrown in constructor is reported to definition\'s global');

promise_test(async t => {
  let w = await create_window_in_test(t, `<script>self.MyElement = class extends HTMLElement { constructor() { this.textContent = 'hello!'; } };</`+`script>`);
  let w2 = await create_window_in_test(t);
  w2.customElements.define('my-element', w.MyElement);
  let nextErrorPromise = nextErrorEvent([self, w, w2]);
  let elem = w2.document.createElement('my-element');
  let errorEvent = await nextErrorPromise;
  assert_true(elem instanceof w2.HTMLUnknownElement, 'element should be an HTMLUnknownElement');
  assert_equals(errorEvent.target, w, `error event should target definition's global but instead targets ${event.currentTarget === w2 ? 'document\'s global' : 'test harness global'}`);
}, 'autonomous: exception thrown by spec due to invalid element state is reported to definition\'s global');

promise_test(async t => {
  assert_implements(supportsCustomizedBuiltinElements, 'customized built-in elements not supported');
  let w = await create_window_in_test(t, `<script>self.MyDivElement = class extends HTMLDivElement { constructor() { throw new Error(); } };</`+`script>`);
  let w2 = await create_window_in_test(t);
  w2.customElements.define('my-div', w.MyDivElement, {extends: 'div'});
  let nextErrorPromise = nextErrorEvent([self, w, w2]);
  let elem = w2.document.createElement('div', {is: 'my-div'});
  let errorEvent = await nextErrorPromise;
  assert_equals(Object.getPrototypeOf(elem), w2.HTMLDivElement.prototype, 'element should be an HTMLDivElement (and not inherited from it)');
  assert_true(errorEvent.error instanceof w.Error, 'error should be an instance created inside the MyElement constructor');
  assert_equals(errorEvent.target, w, `error event should target definition's global but instead targets ${event.target === w2 ? 'document\'s global' : 'test harness global'}`);
}, 'customized built-in: exception thrown in constructor is reported to definition\'s global');

promise_test(async t => {
  assert_implements(supportsCustomizedBuiltinElements, 'customized built-in elements not supported');
  let w = await create_window_in_test(t, `<script>self.MyDivElement = class extends HTMLDivElement { constructor() { return document.createElement('div'); } };</`+`script>`);
  let w2 = await create_window_in_test(t);
  w2.customElements.define('my-div', w.MyDivElement, {extends: 'div'});
  let nextErrorPromise = nextErrorEvent([self, w, w2]);
  let elem = w2.document.createElement('div', {is: 'my-div'});
  let errorEvent = await nextErrorPromise;
  assert_equals(Object.getPrototypeOf(elem), w2.HTMLDivElement.prototype, 'element should be an HTMLDivElement (and not inherited from it)');
  assert_equals(errorEvent.target, w, `error event should target definition's global but instead targets ${event.currentTarget === w2 ? 'document\'s global' : 'test harness global'}`);
}, 'customized built-in: exception thrown by spec due to not returning the same value is reported to definition\'s global');
</script>
</body>