forked from ruffle-rs/ruffle
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathutils.ts
139 lines (119 loc) · 4.31 KB
/
utils.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
import type { Options } from "./common";
import { Config } from "ruffle-core";
const DEFAULT_OPTIONS: Required<Options> = {
...Config.DEFAULT_CONFIG,
ruffleEnable: true,
ignoreOptout: false,
autostart: false,
showReloadButton: false,
swfTakeover: true,
};
// TODO: Once https://crbug.com/798169 is addressed, just use browser.
// We have to wait until whatever version of Chromium supports that
// is old enough to be the oldest version we want to support.
export let i18n: typeof browser.i18n | typeof chrome.i18n;
type ScriptingType = (typeof browser.scripting | typeof chrome.scripting) & {
ExecutionWorld: {
MAIN: string | undefined;
ISOLATED: string;
};
};
export let scripting: ScriptingType;
export let storage: typeof browser.storage | typeof chrome.storage;
export let tabs: typeof browser.tabs | typeof chrome.tabs;
export let runtime: typeof browser.runtime | typeof chrome.runtime;
export let permissions: typeof browser.permissions | typeof chrome.permissions;
export let declarativeNetRequest:
| typeof browser.declarativeNetRequest
| typeof chrome.declarativeNetRequest;
function promisify<T>(
func: (callback: (result: T) => void) => void,
): Promise<T> {
return new Promise((resolve, reject) => {
func((result) => {
const error = chrome.runtime.lastError;
if (error) {
reject(error);
} else {
resolve(result);
}
});
});
}
if (typeof browser !== "undefined") {
i18n = browser.i18n;
scripting = browser.scripting as ScriptingType;
storage = browser.storage;
tabs = browser.tabs;
runtime = browser.runtime;
permissions = browser.permissions;
declarativeNetRequest = browser.declarativeNetRequest;
} else if (typeof chrome !== "undefined") {
i18n = chrome.i18n;
scripting = chrome.scripting as ScriptingType;
storage = chrome.storage;
tabs = chrome.tabs;
runtime = chrome.runtime;
permissions = chrome.permissions;
declarativeNetRequest = chrome.declarativeNetRequest;
} else {
throw new Error("Extension API not found.");
}
export const openOptionsPage: () => Promise<void> = () =>
runtime.openOptionsPage();
export const openPlayerPage: () => Promise<void> = () =>
promisify((cb: () => void) => tabs.create({ url: "/player.html" }, cb));
export const openOnboardPage: () => Promise<void> = () =>
promisify((cb: () => void) => tabs.create({ url: "/onboard.html" }, cb));
export async function getOptions(): Promise<Options> {
const options = await storage.sync.get();
// Copy over default options if they don't exist yet.
return { ...DEFAULT_OPTIONS, ...options };
}
/**
* Gets the options that are explicitly different from the defaults.
*
* In the future we should just not store options we don't want to set.
*/
export async function getExplicitOptions(): Promise<Options> {
const options = await getOptions();
const defaultOptions = DEFAULT_OPTIONS;
for (const key in defaultOptions) {
// @ts-expect-error: Element implicitly has an any type
if (key in options && defaultOptions[key] === options[key]) {
// @ts-expect-error: Element implicitly has an any type
// eslint-disable-next-line @typescript-eslint/no-dynamic-delete
delete options[key];
}
}
// This value is specific to the internal extension pages, and is always "default"
if ("responseHeadersUnsupported" in options) {
delete options["responseHeadersUnsupported"];
}
return options;
}
export const hasAllUrlsPermission = async () => {
const allPermissions = await permissions.getAll();
return allPermissions.origins?.includes("<all_urls>") ?? false;
};
export async function hasHostPermissionForSpecifiedTab(
origin: string | undefined,
) {
try {
return origin
? await permissions.contains({
origins: [origin],
})
: await hasAllUrlsPermission();
} catch {
// catch error that occurs for special urls like about:
return false;
}
}
export async function hasHostPermissionForActiveTab() {
const [activeTab] = await tabs.query({
active: true,
currentWindow: true,
});
return await hasHostPermissionForSpecifiedTab(activeTab?.url);
}