DOM访问

无论是否开始沙箱,Haploid.js 都会将子应用渲染到如下模拟 DOM 结构中:

<haploid-html class="haploid-html">
  <haploid-head class="haploid-head">
    <haploid-title></haploid-title>
  </haploid-head>
  <haploid-body class="haploid-body"></haploid-body>
</haploid-html>

但一些 BOM API 的行为会有所不同。

比如 document.documentElement 会指向上面的 ≶haploid-html> 元素。如果开启 sandbox.enableHtmlPretending 选项,document.documentElement 就会尽可能表现得像原生 <html> 一样,包括:

document.documentElement.constructor === HTMLHtmlElement;
document.documentElement.tagName === "HTML";
document.documentElement.nodeName === "HTML";
document.documentElement.version === "";
document.documentElement.parentNode === document;
document.documentElement.parentElement === null;
document.documentElement instanceof HTMLHtmlElement === true;

上面的 document 是代理对象。

警告

这种伪装在有的场景下可能是致命的,比如由于其 parentElement 是 null,可能导致一些需要上溯到真正 document 的逻辑被破坏。

sandbox.enableBodyPretendingsandbox.enableHeadPretendingsandbox.enableTitlePretending 具有类似的效果。

更多细节可参考环境代理


你同样也可以使用 presetHeadHTMLpresetBodyHTML 来预设一些 DOM 元素。比如:

container.registerApp({
  presetHeadHTML: `<meta name="keywords" content="">`,
  presetBodyHTML: `<article><main></main></article>`,
});

将会产出这样的 DOM 结构:

<haploid-html class="haploid-html">
  <haploid-head class="haploid-head">
    <haploid-title></haploid-title>
    <meta name="keywords" content="" />
  </haploid-head>
  <haploid-body class="haploid-body">
    <article>
      <main></main>
    </article>
  </haploid-body>
</haploid-html>

如果开启了 preserveHTML 选项,而且 entry 是 HTML 格式,那么其中的 DOM 会经过安全过滤后拷贝过来。