我试图在一个web组件中使用字体awsome js/svg库,但是图标不显示。这个是可能的吗?
我正在尝试在一个现有的webforms项目中实现一个angular组件,而不是css和脚本“出血”,还有其他的建议吗?iframe不是一个选项。
<html>
<head>
<script src="https://polygit.org/components/webcomponentsjs/webcomponents-loader.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/5.13.0/js/all.min.js" defer>
</script>
<script>
customElements.define('my-holder', class extends HTMLElement {
constructor() {
super();
console.log("constructor");
let shadowRoot = this.attachShadow({
mode: 'open'
});
const t = document.querySelector('#holder');
const instance = t.content.cloneNode(true);
shadowRoot.appendChild(instance);
}
connectedCallback() {
console.log("callback");
}
});
</script>
</head>
<body>
<div id="outside">
light dom
<div class="fa-4x">
<span class="fa-layers fa-fw" style="background:MistyRose">
<i class="fas fa-circle" style="color:Tomato"></i>
<i class="fa-inverse fas fa-times" data-fa-transform="shrink-6"></i>
</span>
</div>
</div>
<template id="holder">
<script src="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/5.13.0/js/all.min.js" defer></script>
dark shadow dom
<div class="fa-4x">
<span class="fa-layers fa-fw" style="background:MistyRose">
<i class="fas fa-circle" style="color:Tomato"></i>
<i class="fa-inverse fas fa-times" data-fa-transform="shrink-6"></i>
</span>
</div>
</template>
<div id="inside">
<my-holder></my-holder>
</div>
</body>
</html>发布于 2020-05-04 17:45:21
许多旧的those库使用document.来访问主DOM。
所以他们不能对shadowDOM中的内容做任何事情
这意味着你的目标是不放出脚本是不可能的。
字体-令人敬畏(脚本和样式)必须加载到主DOM中。
如果你不想在shadowDOM之外放出样式,你必须遵守规则:
中
的(主DOM)原版
(如果元素本身在另一个shadowDOM中,则为其shadowDOM容器)
<slot></slot>(仅对其插槽进行了反射,仅反映在其插槽中)
(你也可以完全不使用自定义元素)
<span class="fa-4x fa-layers fa-fw">
<i class="fas fa-circle"></i>
<i class="fa-inverse fas fa-times" data-fa-transform="shrink-6"></i>
</span> 解决方案:
编写自定义元素,该元素
<slot></slot> (reflected!not moved!) <awesome-icon background="lightcoral" color="red"></awesome-icon>
<awesome-icon background="lightgreen" color="green"></awesome-icon>
<awesome-icon></awesome-icon>JSFidlle:
<script src="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/5.13.0/js/all.min.js" defer></script>
<script>
customElements.define('awesome-icon', class extends HTMLElement {
constructor() {
super().attachShadow({mode: 'open'})
.append(document.getElementById(this.nodeName).content.cloneNode(true));
}
connectedCallback() {
let setProperty =
(prop, value)=>this.shadowRoot.host.style.setProperty('--' + prop, value);
setProperty('fa-background', this.getAttribute('background'));
setProperty('fa-color', this.getAttribute('color'));
// move icon HTML back to lightDOM so FontAwesome can style it
this.innerHTML = this.shadowRoot.querySelector('#ICON').innerHTML;
}
});
</script>
<template id="AWESOME-ICON">
<style>
::slotted(*) {
/* lightDOM SPAN has higher Specificity, only way out is using !important */
background: var(--fa-background,grey) !important;
color: var(--fa-color,darkgrey) !important;
}
</style>
<template id="ICON">
<span class="fa-4x fa-layers fa-fw">
<i class="fas fa-circle"></i>
<i class="fa-inverse fas fa-times" data-fa-transform="shrink-6"></i>
</span>
</template>
<slot><!--lightDOM REFLECTED here--></slot>
</template>
<awesome-icon><!-- lightDOM CREATED here --></awesome-icon>
<awesome-icon background="lightcoral" color="red"></awesome-icon>
<awesome-icon background="lightgreen" color="green"></awesome-icon>
<style>
span{
background:lightblue; /* !important inside shadowDOM overrules these settings */
color:red;
}
</style>
使用shadowDOM和插槽时不是
并依赖于限定作用域的CSS属性,使代码更简单:
<script>
customElements.define('awesome-icon', class extends HTMLElement {
connectedCallback() {
this.append(document.getElementById(this.nodeName).content.cloneNode(true));
this.style.setProperty('--fa-background', this.getAttribute('background') );
this.style.setProperty('--fa-color' , this.getAttribute('color') );
}
});
</script>
<template id="AWESOME-ICON">
<span class="fa-4x fa-layers fa-fw">
<i class="fas fa-circle"></i>
<i class="fa-inverse fas fa-times" data-fa-transform="shrink-6"></i>
</span>
</template>
<awesome-icon background="lightcoral" color="red"></awesome-icon>
<awesome-icon background="lightgreen" color="green"></awesome-icon>
<awesome-icon></awesome-icon>
<style>
span {
background: var(--fa-background);
color: var(--fa-color );
}
</style>发布于 2021-09-18 23:03:06
我发现了一种方法来渲染SVG字体可怕的图标在阴影dom。
在docs中,@fortawesome/fontawesome-svg-core包提供了更多的控制
下面是我使用该https://codesandbox.io/s/romantic-panka-40xqm创建的示例
基本思想是,我们可以使用icon(faCamera).html[0]获得SVG HTML,并可以在卷影dom root中使用它。
import { icon } from "@fortawesome/fontawesome-svg-core";
import { faCamera } from "@fortawesome/free-solid-svg-icons";
icon(faCamera).html[0]发布于 2021-12-03 02:59:57
Font Awesome有一个内置的方法来解析/搜索给定元素中的图标标签,以生成SVG。下面的代码应该适用于您的示例,或者非常接近您正在寻找的内容:
FontAwesome.dom.i2svg({
node: document.querySelector('my-holder').shadowRoot
})或者,如果你想变得更花哨,让它自动热加载任何字体,更棒的图标在一个更动态的web组件中:
connectedCallback() {
if (this.isConnected) {
FontAwesome.dom.i2svg({
node: this.shadowRoot
})
FontAwesome.dom.watch({
autoReplaceSvgRoot: this,
observeMutationsRoot: this.shadowRoot
})
/** Other web component init on connected code here */
}
}FontAwesome是由font- name脚本设置的全局变量的名称。
不带参数的i2svg()会重新解析整个轻量级dom (注意,这是不必要的,因为默认情况下font watches已经监视dom的变化了)。有了一个参数,你就可以传递一个带有node属性的js对象,该对象指向某个元素(光或阴影),它会根据你的需要生成你的SVG。
对于那些想要在SPA/编译的js前端中做到这一点的人,请查看FA documentation,以获得更简单的导入解决方案。
https://stackoverflow.com/questions/61587210
复制相似问题