【问题标题】:Override Element.prototype.attachShadow using Chrome Extension使用 Chrome 扩展覆盖 Element.prototype.attachShadow
【发布时间】:2019-07-24 01:29:21
【问题描述】:

我需要访问具有封闭 Shadow DOM 的 Web 组件的 DOM 以进行某些 Selenium 测试。我已经阅读了几个参考资料,您可以在文档启动时覆盖Element.prototype.attachShadow,以便将Shadow 从关闭更改为打开。为此,我创建了一个 Chrome 扩展程序。下面是我的manifest.json

{
    "name": "SeleniumTesting",
    "description": "Extension to open closed Shadow DOM for selenium testing",
    "version": "1",
    "author": "SeleniumTesting",
    "manifest_version": 2,
    "permissions": ["downloads", "<all_urls>"],
    "content_scripts": [{
        "matches": ["http://localhost:5000/*"],
        "run_at": "document_start",
        "all_frames": true,
        "js": ["shadowInject.js"]
    }]
}

还有我的 shadowInject.js

console.log('test');
Element.prototype._attachShadow = Element.prototype.attachShadow;
Element.prototype.attachShadow = function () {
    console.log('attachShadow');
    return this._attachShadow( { mode: "open" } );
};

为了测试它,我在一个 ASPNetCore MVC 项目中创建了我的组件。下面是我创建自定义组件的 javascript:

customElements.define('x-element', class extends HTMLElement {
    constructor() {
        super(); 
        this._shadowRoot = this.attachShadow({
            mode: 'closed'
        });
        this._shadowRoot.innerHTML = `<div class="wrapper">
        <a href="download/File.pdf" download>
            <button id="download">Download File</button>
        </a>
        <p>Link para download</p>
      </div>`;
    }
});

还有我使用它的 HTML 文件:

@page
@model IndexModel
@{
    ViewData["Title"] = "Home page";
}
<script src='~/js/componente.js'></script>

<div class="text-center">
    <h1 class="display-4">Welcome</h1>
    <p>Learn about <a href="https://docs.microsoft.com/aspnet/core">building Web apps with ASP.NET Core</a>.</p>
    <x-element id='comp'></x-element>
</div>

我将我的扩展程序加载到 Chrome 中,然后运行该页面。我得到控制台日志测试,但从未调用 attachShadow 方法,我仍然无法访问关闭的影子 DOM

对于我做错的事情,我非常感谢一些帮助。非常感谢。

最终解决方案

应用答案中的更改后,我需要对manifest.json 进行一些调整。以下是最终版本:

{
    "name": "SeleniumTesting",
    "description": "Extension to open closed Shadow DOM for selenium testing",
    "version": "1",
    "author": "SeleniumTesting",
    "manifest_version": 2,
    "permissions": ["downloads", "<all_urls>"],
    "content_scripts": [{
        "matches": ["http://localhost:5000/*"],
        "run_at": "document_start",
        "all_frames": true,
        "js": ["shadowInject.js"]
    }],
    "web_accessible_resources": ["injected.js"]
}

现在它工作了,Shadow DOM 变成了 open

【问题讨论】:

    标签: javascript google-chrome-extension shadow-dom


    【解决方案1】:

    您不应将代码放在 content_scripts 中,因为 content_scripts 与当前页面上下文不同。

    您尝试将shadowInject.js 代码更改为:

    const injectedScript = document.createElement('script');
    injectedScript.src = chrome.extension.getURL('injected.js');
    (document.head || document.documentElement).appendChild(injectedScript);
    

    然后在同目录下创建injected.js文件:

    文件内容为:

    console.log('test');
    Element.prototype._attachShadow = Element.prototype.attachShadow;
    Element.prototype.attachShadow = function () {
        console.log('attachShadow');
        return this._attachShadow( { mode: "open" } );
    };
    

    你可以试试。如果有什么问题,请告诉我

    【讨论】:

    • 我要试试。我应该如何更改 manifest.json 以触发 shadowInject.js,而不是 content_script?
    • manifest.json 不需要更改。您只需要更改shadowInject.js文件的内容并创建一个injected.js文件。
    • 实际上,我需要添加 "web_accessible_resources": ["injected.js"] 以便 getURL 可以识别和使用注入的.js,但是一个简单的谷歌让我明白了这一点。我已经发布了最终的 manifest.json 以帮助将来的任何人。再次感谢您的帮助
    【解决方案2】:

    正如 Black-Hole 的回答中提到的,内容脚本有自己的 DOM 对象版本,因此您必须注入额外的脚本才能在真实 DOM 中运行。

    如果您想尽可能少地触摸页面并保持页面其余部分的阴影关闭,您可以使用此脚本:

    {
      const shadows = new WeakMap();
      const original = Element.prototype.attachShadow;
      Element.prototype.attachShadow = function() {
        const shadow = original.apply(this, arguments);
        shadows.set(this, shadow);
        return shadow;
      };
    
      // Later...
      shadows.get(document.querySelector("x-element"));
    }
    

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2013-05-15
      • 1970-01-01
      • 2018-08-29
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多