chromium/third_party/blink/web_tests/shadow-dom/resources/shadow-dom.js

function removeWhiteSpaceOnlyTextNodes(node) {
  for (var i = 0; i < node.childNodes.length; i++) {
    var child = node.childNodes[i];
    if (child.nodeType === Node.TEXT_NODE &&
        child.nodeValue.trim().length == 0) {
      node.removeChild(child);
      i--;
    } else if (
        child.nodeType === Node.ELEMENT_NODE ||
        child.nodeType === Node.DOCUMENT_FRAGMENT_NODE) {
      removeWhiteSpaceOnlyTextNodes(child);
    }
  }
  if (node.shadowRoot) {
    removeWhiteSpaceOnlyTextNodes(node.shadowRoot);
  }
}

function convertTemplatesToShadowRootsWithin(node) {
  var nodes = node.querySelectorAll('template');
  for (var i = 0; i < nodes.length; ++i) {
    var template = nodes[i];
    var mode = template.getAttribute('data-mode');
    var delegatesFocus = template.hasAttribute('data-delegatesFocus');
    var parent = template.parentNode;
    parent.removeChild(template);
    var shadowRoot;
    if (!mode || mode == 'v0') {
      shadowRoot = parent.attachShadow({mode: 'open'});
    } else {
      shadowRoot =
          parent.attachShadow({'mode': mode, 'delegatesFocus': delegatesFocus});
    }
    var expose = template.getAttribute('data-expose-as');
    if (expose)
      window[expose] = shadowRoot;
    if (template.id)
      shadowRoot.id = template.id;
    var fragments = document.importNode(template.content, true);
    shadowRoot.appendChild(fragments);

    convertTemplatesToShadowRootsWithin(shadowRoot);
  }
}

function isShadowHost(node) {
  return node && node.nodeType == Node.ELEMENT_NODE && node.shadowRoot;
}

function isIFrameElement(element) {
  return element && element.nodeName == 'IFRAME';
}

// Returns node from shadow/iframe tree "path".
function getNodeInComposedTree(path) {
  var ids = path.split('/');
  var node = document.getElementById(ids[0]);
  for (var i = 1; node != null && i < ids.length; ++i) {
    if (isIFrameElement(node))
      node = node.contentDocument.getElementById(ids[i]);
    else if (isShadowHost(node))
      node = node.shadowRoot.getElementById(ids[i]);
    else
      return null;
  }
  return node;
}

function createTestTree(node) {
  let ids = {};

  function attachShadowFromTemplate(template) {
    let parent = template.parentNode;
    parent.removeChild(template);
    let shadowRoot;
    if (template.getAttribute('data-mode') === 'v0') {
      // For legacy Shadow DOM
      shadowRoot = parent.attachShadow({mode: 'open'});
    } else if (template.getAttribute('data-slot-assignment') === 'manual') {
       shadowRoot =
          parent.attachShadow({mode: template.getAttribute('data-mode'),
                               slotAssignment: 'manual'});
    } else {
      shadowRoot =
          parent.attachShadow({mode: template.getAttribute('data-mode')});
    }
    let id = template.id;
    if (id) {
      shadowRoot.id = id;
      ids[id] = shadowRoot;
    }
    shadowRoot.appendChild(document.importNode(template.content, true));
    return shadowRoot;
  }

  function walk(root) {
    if (root.id) {
      ids[root.id] = root;
    }
    for (let e of Array.from(root.querySelectorAll('[id]'))) {
      ids[e.id] = e;
    }
    for (let e of Array.from(root.querySelectorAll('template'))) {
      walk(attachShadowFromTemplate(e));
    }
  }

  walk(node.cloneNode(true));
  return ids;
}

function dispatchEventWithLog(nodes, target, event) {
  function labelFor(e) {
    return e.id || e.tagName;
  }

  let log = [];
  let attachedNodes = [];
  for (let label in nodes) {
    let startingNode = nodes[label];
    for (let node = startingNode; node; node = node.parentNode) {
      if (attachedNodes.indexOf(node) >= 0)
        continue;
      let id = node.id;
      if (!id)
        continue;
      attachedNodes.push(node);
      node.addEventListener(event.type, (e) => {
        // Record [currentTarget, target, relatedTarget, composedPath()]
        log.push([
          id, labelFor(e.target),
          e.relatedTarget ? labelFor(e.relatedTarget) : null,
          e.composedPath().map((n) => {
            return labelFor(n);
          })
        ]);
      });
    }
  }
  target.dispatchEvent(event);
  return log;
}

function debugEventLog(log) {
  for (let i = 0; i < log.length; i++) {
    console.log(
        '[' + i + '] currentTarget: ' + log[i][0] + ' target: ' + log[i][1] +
        ' relatedTarget: ' + log[i][2] + ' composedPath(): ' + log[i][3]);
  }
}

function debugCreateTestTree(nodes) {
  for (let k in nodes) {
    console.log(k + ' -> ' + nodes[k]);
  }
}

// This function assumes that testharness.js is available.
function assert_event_path_equals(actual, expected) {
  assert_equals(actual.length, expected.length);
  for (let i = 0; i < actual.length; ++i) {
    assert_equals(
        actual[i][0], expected[i][0],
        'currentTarget at ' + i + ' should be same');
    assert_equals(
        actual[i][1], expected[i][1], 'target at ' + i + ' should be same');
    assert_equals(
        actual[i][2], expected[i][2],
        'relatedTarget at ' + i + ' should be same');
    assert_array_equals(
        actual[i][3], expected[i][3],
        'composedPath at ' + i + ' should be same');
  }
}

function assert_background_color(path, color) {
  assert_equals(
      window.getComputedStyle(getNodeInComposedTree(path)).backgroundColor,
      color, 'backgroundColor for ' + path + ' should be ' + color);
}