如何让导入的css字体/图标对阴影dom中的元素产生影响?

How to let imported css font / icons have effects on elements in the shadow dom?

假设我想使用阴影创建自定义元素 dom。模板中的某些元素在链接的 css 文件中指定了 class 名称。现在我想让 css 规则对元素产生影响。但由于阴影 dom 样式边界,我无法实现。

<link rel="stylesheet" type="text/css"
href="https://maxcdn.bootstrapcdn.com/font-awesome/4.5.0/css/font-awesome.min.css">
<template id="blog-header">
<header>
    <h1>DreamLine</h1>
    <nav>
        <ul>
            <li><a href="#0">Tour</a></li>
            <li><a href="#0">Blog</a></li>
            <li><a href="#0">Contact</a></li>
            <li><a href="#0">Error</a></li>
            <li><a href="#0"><i class="fa fa-search"></i>Search</a></li>
        </ul>
    </nav>
</header>
</template>
<script type="text/javascript">
var importDoc = document.currentScript.ownerDocument;
var proto = Object.create(HTMLElement.prototype, {
    createdCallback: {
        value: function () {
            var t = importDoc.querySelector("#blog-header");
            var clone = document.importNode(t.content, true);
            this.createShadowRoot().appendChild(clone);
        }
    }
});
document.registerElement("blog-header", {
    prototype: proto
});
</script>

你看,fa-search 是 font-awesome css 文件中定义的 class,我如何设置 <i> 元素的样式?

要在 Shadow DOM 中使用导入的字体(例如 FontAwesome),您应该:

1°声明字体

首先,在主文档中包含 <link rel="stylesheet"> 元素。它将声明一个 @font-face CSS 规则,使该字体可用于文档中的所有文本。

2° 导入样式表

然后,在 <template> 节点中导入带有 @import url CSS 规则的同一文件,使 .fa-* 类 选择器可从 Shadow DOM :

<link rel="stylesheet" href="https://use.fontawesome.com/releases/v5.7.1/css/all.css" integrity="sha384-fnmOCqbTlWIlj8LyTjo7mOUStjsKC4pOpQbqyi7RrhN7udi9RwhKkMHpvLbHG9Sr" crossorigin="anonymous">

<template id="blog-header">
  <style>
    @import url("https://use.fontawesome.com/releases/v5.7.1/css/all.css")
  </style>
  <header>
    <h1>DreamLine</h1>
    //...
  </header>
</template>

var importDoc = document.currentScript.ownerDocument;
var proto = Object.create(HTMLElement.prototype, {
 createdCallback: {
  value: function() {
   var t = importDoc.querySelector("#blog-header");
   var clone = document.importNode(t.content, true);
   this.createShadowRoot().appendChild(clone);
  }
 }
});
document.registerElement("blog-header", {prototype: proto});
/* 
@font-face {
    font-family: "FontAwesome";
    src: url("https://maxcdn.bootstrapcdn.com/font-awesome/4.5.0/css/font-awesome-webfont.woff2?v=4.5.0") format('woff2');
  }
*/
<link rel="stylesheet" href="https://use.fontawesome.com/releases/v5.7.1/css/all.css" integrity="sha384-fnmOCqbTlWIlj8LyTjo7mOUStjsKC4pOpQbqyi7RrhN7udi9RwhKkMHpvLbHG9Sr" crossorigin="anonymous">

<template id="blog-header">
  <style>
    @import url("https://use.fontawesome.com/releases/v5.7.1/css/all.css")
  </style>
 <header>
  <h1>DreamLine</h1>
  <nav>
   <ul>
    <li><a href="#0">Tour</a></li>
    <li><a href="#0">Blog</a></li>
    <li><a href="#0">Contact</a></li>
    <li><a href="#0">Error</a></li>
    <li><a href="#0"><i class="fa fa-search"></i>Search</a></li>
   </ul>
  </nav>
 </header>
</template>

<blog-header></blog-header>

2019 年更新

现在您可以在阴影 DOM 中使用 <link rel="stylesheet"> 而不是 @import url()

我已经将一个带有 mdi 图标的示例项目作为子模块导入到一个更大的项目中。我的示例项目在传单地图中使用了 mdi 图标,当使用 dom-to-image 包在更大的项目中下载地图图像时,这些图标没有被下载。 我必须将它添加到 vue 文件中以使其工作的方式如下:

 <template>
  <div>
    <link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/MaterialDesign-Webfont/5.6.55/css/materialdesignicons.min.css" crossorigin="anonymous">