|
1 | 1 | import { env } from "./utils";
|
2 | 2 |
|
3 |
| -function injectStylesheets(urlList: string[]) { |
4 |
| - urlList.forEach((url) => { |
5 |
| - if (!document.querySelector(`link[href="${url}"]`)) { |
6 |
| - const link = document.createElement("link"); |
7 |
| - link.rel = "stylesheet"; |
8 |
| - link.type = "text/css"; |
9 |
| - link.href = url; |
10 |
| - link.classList.add("SuperCSSInject"); |
11 |
| - |
12 |
| - document.head.appendChild(link); |
| 3 | +function main () { |
| 4 | + env.runtime.onMessage.addListener((message) => { |
| 5 | + if (message.action == "inject") { |
| 6 | + updateInjectedStylesheets(message.urlList); |
13 | 7 | }
|
14 | 8 | });
|
| 9 | + |
| 10 | + env.runtime.sendMessage({ action: "load" }); |
| 11 | + maintainStylesheetsOrder(); |
15 | 12 | }
|
16 | 13 |
|
17 |
| -function clearStylesheets(url: string) { |
| 14 | +function updateInjectedStylesheets (urlList: string[]) { |
18 | 15 | const links: NodeListOf<HTMLLinkElement> = document.querySelectorAll("link.SuperCSSInject");
|
19 |
| - |
20 |
| - if (links.length > 0) { |
21 |
| - links.forEach((link: HTMLLinkElement) => { |
22 |
| - if (link.href === url) { |
23 |
| - link.remove(); |
| 16 | + const currentList = Array.from(links).map((link) => link.href); |
| 17 | + |
| 18 | + if (currentList.length > urlList.length) { |
| 19 | + for (const url of currentList) { |
| 20 | + if (!urlList.includes(url)) { |
| 21 | + clearStylesheet(url); |
24 | 22 | }
|
25 |
| - }); |
| 23 | + } |
| 24 | + } else { |
| 25 | + for (const url of urlList) { |
| 26 | + if (!currentList.includes(url)) { |
| 27 | + injectStylesheet(url); |
| 28 | + } |
| 29 | + } |
26 | 30 | }
|
27 | 31 | }
|
28 | 32 |
|
29 |
| -function main() { |
30 |
| - env.runtime.onMessage.addListener((message) => { |
31 |
| - if (message.action == "inject") { |
32 |
| - injectStylesheets(message.urlList); |
33 |
| - } |
| 33 | +function clearStylesheet (url: string) { |
| 34 | + const link = document.querySelector(`link[href="${url}"].SuperCSSInject`); |
| 35 | + link && link.remove(); |
| 36 | +} |
| 37 | + |
| 38 | +function injectStylesheet (url: string) { |
| 39 | + const link = createLinkElement(url); |
| 40 | + document.head.append(link); |
| 41 | +} |
34 | 42 |
|
35 |
| - if (message.action == "clear") { |
36 |
| - clearStylesheets(message.url); |
| 43 | +function createLinkElement (url: string) { |
| 44 | + const link = document.createElement("link"); |
| 45 | + |
| 46 | + link.rel = "stylesheet"; |
| 47 | + link.type = "text/css"; |
| 48 | + link.href = url; |
| 49 | + link.classList.add("SuperCSSInject"); |
| 50 | + |
| 51 | + return link; |
| 52 | +} |
| 53 | + |
| 54 | +/** |
| 55 | + * Make sure the injected stylesheets are always placed last on the DOM |
| 56 | + * |
| 57 | + * This handles SPAs where is common for additional assets to be loaded after |
| 58 | + * the initial page load and ensures the injected styles retain priority. |
| 59 | + */ |
| 60 | +function maintainStylesheetsOrder () { |
| 61 | + const observer = new MutationObserver(() => { |
| 62 | + const injectedLinks: NodeListOf<HTMLLinkElement> = document.head.querySelectorAll("link.SuperCSSInject"); |
| 63 | + |
| 64 | + if (injectedLinks.length > 0) { |
| 65 | + const links: NodeListOf<HTMLLinkElement> = document.head.querySelectorAll("link[rel='stylesheet']"); |
| 66 | + const lastLink: HTMLLinkElement = links[links.length - 1]; |
| 67 | + const isInjectedStylesheetLast = lastLink.className === "SuperCSSInject"; |
| 68 | + |
| 69 | + if (!isInjectedStylesheetLast) { |
| 70 | + observer.disconnect(); |
| 71 | + moveInjectedStylesheets(); |
| 72 | + } |
37 | 73 | }
|
38 | 74 | });
|
39 | 75 |
|
40 |
| - env.runtime.sendMessage({ action: "pageLoad" }); |
| 76 | + observer.observe(document.head, { childList: true }); |
| 77 | +} |
| 78 | + |
| 79 | +function moveInjectedStylesheets () { |
| 80 | + const links: NodeListOf<HTMLLinkElement> = document.head.querySelectorAll("link.SuperCSSInject"); |
| 81 | + |
| 82 | + for (const link of links) { |
| 83 | + document.head.appendChild(link); |
| 84 | + } |
| 85 | + |
| 86 | + maintainStylesheetsOrder(); |
41 | 87 | }
|
42 | 88 |
|
43 | 89 | window.addEventListener("load", main);
|
44 | 90 |
|
45 | 91 | // This is just to make the TS compiler happy
|
46 |
| -export {}; |
| 92 | +export { }; |
0 commit comments