2019-08-30 07:04:28 +00:00
|
|
|
import Diff from "scripts/Diff"
|
|
|
|
|
|
|
|
export default class SVGIcon extends HTMLElement {
|
2019-08-31 11:29:03 +00:00
|
|
|
static cache = new Map<string, Promise<Node>>()
|
2019-08-30 07:04:28 +00:00
|
|
|
|
|
|
|
static get observedAttributes() {
|
|
|
|
return ["name"]
|
|
|
|
}
|
|
|
|
|
|
|
|
attributeChangedCallback(attrName) {
|
|
|
|
if(attrName === "name") {
|
|
|
|
this.render()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async render() {
|
2019-08-31 04:45:42 +00:00
|
|
|
let cache = SVGIcon.cache.get(this.name)
|
2019-08-30 07:04:28 +00:00
|
|
|
|
|
|
|
if(cache) {
|
2019-08-31 11:29:03 +00:00
|
|
|
let svg = await cache
|
|
|
|
|
|
|
|
Diff.mutations.queue(() => {
|
|
|
|
// Remove all existing child nodes
|
|
|
|
while(this.firstChild) {
|
|
|
|
this.removeChild(this.firstChild)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Append a clone of the cached SVG node
|
|
|
|
this.appendChild(svg.cloneNode(true))
|
|
|
|
})
|
|
|
|
|
2019-08-30 07:04:28 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-08-31 04:45:42 +00:00
|
|
|
SVGIcon.cache.set(this.name, new Promise(async (resolve, reject) => {
|
2019-08-30 07:04:28 +00:00
|
|
|
let url = `//media.notify.moe/images/icons/${this.name}.svg`
|
|
|
|
let response = await fetch(url)
|
|
|
|
|
|
|
|
if(!response.ok) {
|
|
|
|
console.warn(`Failed loading SVG icon: ${url}`)
|
|
|
|
reject(response.statusText)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
let text = await response.text()
|
2019-08-31 11:29:03 +00:00
|
|
|
|
|
|
|
Diff.mutations.queue(() => {
|
|
|
|
this.innerHTML = text
|
|
|
|
let svg = this.firstChild
|
|
|
|
|
|
|
|
if(!svg) {
|
|
|
|
console.warn("Invalid SVG icon:", svg)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
resolve(svg)
|
|
|
|
})
|
2019-08-31 04:45:42 +00:00
|
|
|
}))
|
2019-08-30 07:04:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
get name() {
|
|
|
|
return this.getAttribute("name") || ""
|
|
|
|
}
|
|
|
|
|
|
|
|
set name(value: string) {
|
|
|
|
this.setAttribute("name", value)
|
|
|
|
}
|
|
|
|
}
|