File: event-handler-body.js

package info (click to toggle)
thunderbird 1%3A143.0.1-1
  • links: PTS, VCS
  • area: main
  • in suites: experimental
  • size: 4,703,968 kB
  • sloc: cpp: 7,770,492; javascript: 5,943,842; ansic: 3,918,754; python: 1,418,263; xml: 653,354; asm: 474,045; java: 183,079; sh: 111,238; makefile: 20,410; perl: 14,359; objc: 13,059; yacc: 4,583; pascal: 3,405; lex: 1,720; ruby: 999; exp: 762; sql: 715; awk: 580; php: 436; lisp: 430; sed: 69; csh: 10
file content (77 lines) | stat: -rw-r--r-- 3,250 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
const windowReflectingBodyElementEventHandlerSet =
  new Set(['blur', 'error', 'focus', 'load', 'resize', 'scroll']);

function handlersInInterface(mainIDL, name) {
  return mainIDL.find(idl => idl.name === name).members.map(member => member.name.slice(2));
}

const handlersListPromise = fetch("/interfaces/html.idl").then(res => res.text()).then(htmlIDL => {
  const parsedHTMLIDL = WebIDL2.parse(htmlIDL);
  const windowEventHandlers = handlersInInterface(parsedHTMLIDL, "WindowEventHandlers");
  const globalEventHandlers = handlersInInterface(parsedHTMLIDL, "GlobalEventHandlers");

  const shadowedHandlers = [
    ...windowReflectingBodyElementEventHandlerSet,
    ...windowEventHandlers
  ];
  const notShadowedHandlers = globalEventHandlers.filter(name => !windowReflectingBodyElementEventHandlerSet.has(name));
  return {
    shadowedHandlers,
    notShadowedHandlers
  };
});

function eventHandlerTest(shadowedHandlers, notShadowedHandlers, element) {
  const altBody = document.createElement(element);
  for (const [des, obj1, obj2, obj3, des1, des2, des3] of [
    ["document.body", document.body, altBody, window, "body", "alternative body", "window"],
    [`document.createElement("${element}")`, altBody, document.body, window, "alternative body", "body", "window"],
    ["window", window, document.body, altBody, "window", "body", "alternative body"]
  ]) {
    const f = () => 0;

    shadowedHandlers.forEach(handler => {
      const eventHandler = obj1['on' + handler];
      test(() => {
        obj1['on' + handler] = f;
        assert_equals(obj2['on' + handler], f, `${des2} should reflect`);
        assert_equals(obj3['on' + handler], f, `${des3} should reflect`);
      }, `shadowed ${handler} (${des})`);
      obj1['on' + handler] = eventHandler;
    });

    notShadowedHandlers.forEach(handler => {
      const eventHandler = obj1['on' + handler];
      test(() => {
        obj1['on' + handler] = f;
        assert_equals(obj2['on' + handler], null, `${des2} should reflect`);
        assert_equals(obj3['on' + handler], null, `${des3} should reflect`);
      }, `not shadowed ${handler} (${des})`);
      obj1['on' + handler] = eventHandler;
    });

    shadowedHandlers.forEach(handler => {
      test(() => {
        assert_equals(obj1['on' + handler], null, `${des1} should reflect changes to itself`);
        assert_equals(obj2['on' + handler], null, `${des2} should reflect`);
        assert_equals(obj3['on' + handler], null, `${des3} should reflect`);
      }, `shadowed ${handler} removal (${des})`);
    });

    shadowedHandlers.forEach(handler => {
      // Cannot test the error and unhandledrejection events as the test harness listens for those.
      if (des != "document.body" || handler == "error" || handler == "unhandledrejection") {
        return;
      }
      test(t => {
        t.add_cleanup(() => {
          obj1.removeAttribute('on' + handler);
          window[`on${handler}Happened`] = undefined;
        });
        obj1.setAttribute('on' + handler, `window.on${handler}Happened = true`);
        obj3.dispatchEvent(new Event(handler));
        assert_true(window[`on${handler}Happened`]);
      }, `shadowed ${handler} on body fires when event dispatched on window`);
    });
  }
}