SponsorBlock/src/config.ts

472 lines
13 KiB
TypeScript
Raw Normal View History

import * as CompileConfig from "../config.json";
import { CategorySelection, CategorySkipOption, PreviewBarOption, SponsorTime } from "./types";
import Utils from "./utils";
const utils = new Utils();
interface SBConfig {
userID: string,
segmentTimes: SBMap<string, SponsorTime[]>,
defaultCategory: string,
whitelistedChannels: string[],
forceChannelCheck: boolean,
startSponsorKeybind: string,
submitKeybind: string,
minutesSaved: number,
skipCount: number,
sponsorTimesContributed: number,
2020-06-06 03:38:38 +02:00
submissionCountSinceCategories: number, // New count used to show the "Read The Guidelines!!" message
showTimeWithSkips: boolean,
2020-06-07 17:27:35 +02:00
unsubmittedWarning: boolean,
disableSkipping: boolean,
trackViewCount: boolean,
dontShowNotice: boolean,
hideVideoPlayerControls: boolean,
hideInfoButtonPlayerControls: boolean,
hideDeleteButtonPlayerControls: boolean,
hideUploadButtonPlayerControls: boolean,
hideDiscordLaunches: number,
hideDiscordLink: boolean,
invidiousInstances: string[],
supportInvidious: boolean,
serverAddress: string,
minDuration: number,
audioNotificationOnSkip,
2020-02-24 02:39:13 +01:00
checkForUnlistedVideos: boolean,
2020-04-09 06:40:11 +02:00
testingServer: boolean,
2020-09-04 18:39:00 +02:00
hashPrefix: boolean,
refetchWhenNotFound: boolean,
2020-04-03 04:13:36 +02:00
// What categories should be skipped
2020-06-04 02:20:02 +02:00
categorySelections: CategorySelection[],
// Preview bar
barTypes: {
"preview-chooseACategory": PreviewBarOption,
2020-06-04 02:20:02 +02:00
"sponsor": PreviewBarOption,
"preview-sponsor": PreviewBarOption,
"intro": PreviewBarOption,
"preview-intro": PreviewBarOption,
"outro": PreviewBarOption,
"preview-outro": PreviewBarOption,
"interaction": PreviewBarOption,
"preview-interaction": PreviewBarOption,
"selfpromo": PreviewBarOption,
"preview-selfpromo": PreviewBarOption,
"music_offtopic": PreviewBarOption,
2020-07-03 03:37:38 +02:00
"preview-music_offtopic": PreviewBarOption,
2020-06-04 02:20:02 +02:00
}
}
2020-01-29 04:16:48 +01:00
interface SBObject {
configListeners: Array<Function>;
defaults: SBConfig;
localConfig: SBConfig;
config: SBConfig;
// Functions
encodeStoredItem<T>(data: T): T | Array<any>;
convertJSON(): void;
2020-01-29 04:16:48 +01:00
}
// Allows a SBMap to be conveted into json form
// Currently used for local storage
class SBMap<T, U> extends Map {
id: string;
constructor(id: string, entries?: [T, U][]) {
2020-02-01 22:53:33 +01:00
super();
this.id = id;
2020-02-01 22:53:33 +01:00
// Import all entries if they were given
if (entries !== undefined) {
for (const item of entries) {
2020-02-09 16:34:18 +01:00
super.set(item[0], item[1])
2020-02-01 22:53:33 +01:00
}
}
}
get(key): U {
return super.get(key);
}
rawSet(key, value) {
return super.set(key, value);
}
update() {
// Store updated SBMap locally
chrome.storage.sync.set({
[this.id]: encodeStoredItem(this)
});
}
set(key: T, value: U) {
const result = super.set(key, value);
this.update();
return result;
}
delete(key) {
const result = super.delete(key);
// Make sure there are no empty elements
for (const entry of this.entries()) {
if (entry[1].length === 0) {
super.delete(entry[0]);
}
}
this.update();
return result;
}
clear() {
const result = super.clear();
this.update();
return result;
}
2020-01-29 04:16:48 +01:00
}
const Config: SBObject = {
2020-01-10 02:09:32 +01:00
/**
* Callback function when an option is updated
*/
2020-01-29 04:16:48 +01:00
configListeners: [],
defaults: {
userID: null,
segmentTimes: new SBMap("segmentTimes"),
defaultCategory: "chooseACategory",
whitelistedChannels: [],
forceChannelCheck: false,
startSponsorKeybind: ";",
submitKeybind: "'",
minutesSaved: 0,
skipCount: 0,
sponsorTimesContributed: 0,
2020-06-06 03:38:38 +02:00
submissionCountSinceCategories: 0,
showTimeWithSkips: true,
2020-06-07 17:27:35 +02:00
unsubmittedWarning: true,
disableSkipping: false,
trackViewCount: true,
dontShowNotice: false,
hideVideoPlayerControls: false,
hideInfoButtonPlayerControls: false,
hideDeleteButtonPlayerControls: false,
hideUploadButtonPlayerControls: false,
hideDiscordLaunches: 0,
hideDiscordLink: false,
2020-09-05 03:02:07 +02:00
invidiousInstances: ["invidious.snopyta.org"],
2020-02-09 01:10:04 +01:00
supportInvidious: false,
serverAddress: CompileConfig.serverAddress,
minDuration: 0,
audioNotificationOnSkip: false,
2020-02-24 02:39:13 +01:00
checkForUnlistedVideos: false,
2020-04-09 06:40:11 +02:00
testingServer: false,
hashPrefix: false,
2020-09-04 18:39:00 +02:00
refetchWhenNotFound: true,
2020-04-09 06:40:11 +02:00
2020-04-03 04:13:36 +02:00
categorySelections: [{
name: "sponsor",
option: CategorySkipOption.AutoSkip
2020-06-04 02:20:02 +02:00
}],
// Preview bar
barTypes: {
"preview-chooseACategory": {
color: "#ffffff",
opacity: "0.7"
},
2020-06-04 02:20:02 +02:00
"sponsor": {
color: "#00d400",
opacity: "0.7"
},
"preview-sponsor": {
color: "#007800",
opacity: "0.7"
},
"intro": {
color: "#00ffff",
opacity: "0.7"
},
"preview-intro": {
color: "#008080",
opacity: "0.7"
},
"outro": {
color: "#0202ed",
opacity: "0.7"
},
"preview-outro": {
color: "#000070",
opacity: "0.7"
},
"interaction": {
color: "#cc00ff",
opacity: "0.7"
},
"preview-interaction": {
color: "#6c0087",
opacity: "0.7"
},
"selfpromo": {
color: "#ffff00",
opacity: "0.7"
},
"preview-selfpromo": {
color: "#bfbf35",
opacity: "0.7"
},
"music_offtopic": {
color: "#ff9900",
opacity: "0.7"
},
"preview-music_offtopic": {
color: "#a6634a",
opacity: "0.7"
}
}
2020-01-29 04:16:48 +01:00
},
2020-02-02 01:18:53 +01:00
localConfig: null,
config: null,
// Functions
encodeStoredItem,
convertJSON
2020-01-10 02:09:32 +01:00
};
2019-12-31 21:07:43 +01:00
2020-01-09 18:46:04 +01:00
// Function setup
/**
2020-01-29 04:16:48 +01:00
* A SBMap cannot be stored in the chrome storage.
2020-02-15 05:20:11 +01:00
* This data will be encoded into an array instead
2020-01-09 18:46:04 +01:00
*
2020-02-02 01:18:53 +01:00
* @param data
2020-01-09 18:46:04 +01:00
*/
function encodeStoredItem<T>(data: T): T | Array<any> {
2020-01-29 04:16:48 +01:00
// if data is SBMap convert to json for storing
if(!(data instanceof SBMap)) return data;
return Array.from(data.entries());
2020-01-09 18:12:49 +01:00
}
/**
* An SBMap cannot be stored in the chrome storage.
* This data will be decoded from the array it is stored in
*
* @param {*} data
*/
function decodeStoredItem<T>(id: string, data: T): T | SBMap<string, SponsorTime[]> {
if (!Config.defaults[id]) return data;
if (Config.defaults[id] instanceof SBMap) {
try {
if (!Array.isArray(data)) return data;
return new SBMap(id, data);
} catch(e) {
2020-02-15 05:20:11 +01:00
console.error("Failed to parse SBMap: " + id);
2020-02-09 02:08:34 +01:00
}
}
2020-02-15 05:20:11 +01:00
// If all else fails, return the data
return data;
2020-01-09 00:16:02 +01:00
}
2020-02-02 01:25:40 +01:00
function configProxy(): any {
2019-12-31 21:07:43 +01:00
chrome.storage.onChanged.addListener((changes, namespace) => {
2020-01-10 02:09:32 +01:00
for (const key in changes) {
Config.localConfig[key] = decodeStoredItem(key, changes[key].newValue);
2019-12-31 21:07:43 +01:00
}
2020-01-10 02:09:32 +01:00
for (const callback of Config.configListeners) {
2020-01-10 02:09:32 +01:00
callback(changes);
}
2019-12-31 21:07:43 +01:00
});
2020-01-09 17:39:23 +01:00
const handler: ProxyHandler<any> = {
set(obj, prop, value) {
Config.localConfig[prop] = value;
2020-01-09 19:16:27 +01:00
2019-12-31 21:07:43 +01:00
chrome.storage.sync.set({
2020-01-09 18:46:04 +01:00
[prop]: encodeStoredItem(value)
2020-01-08 04:59:50 +01:00
});
2020-01-29 04:16:48 +01:00
return true;
2019-12-31 21:07:43 +01:00
},
2020-01-29 04:16:48 +01:00
get(obj, prop): any {
const data = Config.localConfig[prop];
return obj[prop] || data;
},
deleteProperty(obj, prop) {
2020-01-29 04:16:48 +01:00
chrome.storage.sync.remove(<string> prop);
return true;
2019-12-31 21:07:43 +01:00
}
2020-01-16 19:43:49 +01:00
2019-12-31 21:07:43 +01:00
};
2020-01-08 04:59:50 +01:00
return new Proxy({handler}, handler);
2019-12-31 21:07:43 +01:00
}
function fetchConfig(): Promise<void> {
2020-01-09 18:46:04 +01:00
return new Promise((resolve, reject) => {
chrome.storage.sync.get(null, function(items) {
Config.localConfig = <SBConfig> <unknown> items; // Data is ready
2020-01-09 18:46:04 +01:00
resolve();
});
2019-12-31 21:07:43 +01:00
});
2020-01-09 18:46:04 +01:00
}
2019-12-31 21:07:43 +01:00
function migrateOldFormats(config: SBConfig) {
if (config["disableAutoSkip"]) {
for (const selection of config.categorySelections) {
2020-04-06 06:40:30 +02:00
if (selection.name === "sponsor") {
selection.option = CategorySkipOption.ManualSkip;
chrome.storage.sync.remove("disableAutoSkip");
}
2020-01-06 22:11:37 +01:00
}
}
2020-05-21 05:50:26 +02:00
// Auto vote removal
if (config["autoUpvote"]) {
2020-05-21 05:50:26 +02:00
chrome.storage.sync.remove("autoUpvote");
}
// mobileUpdateShowCount removal
if (config["mobileUpdateShowCount"] !== undefined) {
chrome.storage.sync.remove("mobileUpdateShowCount");
}
2020-07-03 03:13:07 +02:00
// categoryUpdateShowCount removal
if (config["categoryUpdateShowCount"] !== undefined) {
2020-07-03 03:13:07 +02:00
chrome.storage.sync.remove("categoryUpdateShowCount");
}
// Channel URLS
if (config.whitelistedChannels.length > 0 &&
(config.whitelistedChannels[0] == null || config.whitelistedChannels[0].includes("/"))) {
const channelURLFixer = async() => {
const newChannelList: string[] = [];
for (const item of config.whitelistedChannels) {
if (item != null) {
if (item.includes("/channel/")) {
newChannelList.push(item.split("/")[2]);
} else if (item.includes("/user/") && utils.isContentScript()) {
// Replace channel URL with channelID
const response = await utils.asyncRequestToCustomServer("GET", "https://sponsor.ajay.app/invidious/api/v1/channels/" + item.split("/")[2] + "?fields=authorId");
if (response.ok) {
newChannelList.push((JSON.parse(response.responseText)).authorId);
} else {
// Add it at the beginning so it gets converted later
newChannelList.unshift(item);
}
} else if (item.includes("/user/")) {
// Add it at the beginning so it gets converted later (The API can only be called in the content script due to CORS issues)
newChannelList.unshift(item);
} else {
newChannelList.push(item);
}
}
}
config.whitelistedChannels = newChannelList;
}
channelURLFixer();
}
// Check if off-topic category needs to be removed
for (let i = 0; i < config.categorySelections.length; i++) {
if (config.categorySelections[i].name === "offtopic") {
config.categorySelections.splice(i, 1);
// Call set listener
config.categorySelections = config.categorySelections;
break;
}
}
// Migrate old "sponsorTimes"
if (config["sponsorTimes"]) {
let jsonData: any = config["sponsorTimes"];
// Check if data is stored in the old format for SBMap (a JSON string)
if (typeof jsonData === "string") {
try {
jsonData = JSON.parse(jsonData);
} catch(e) {
// Continue normally (out of this if statement)
}
}
// Otherwise junk data
if (Array.isArray(jsonData)) {
const oldMap = new Map(jsonData);
oldMap.forEach((sponsorTimes: number[][], key) => {
const segmentTimes: SponsorTime[] = [];
for (const segment of sponsorTimes) {
segmentTimes.push({
segment: segment,
category: "sponsor",
UUID: null
});
}
config.segmentTimes.rawSet(key, segmentTimes);
});
config.segmentTimes.update();
}
chrome.storage.sync.remove("sponsorTimes");
}
2020-01-06 22:11:37 +01:00
}
2020-01-09 18:46:04 +01:00
async function setupConfig() {
2019-12-31 21:07:43 +01:00
await fetchConfig();
addDefaults();
convertJSON();
const config = configProxy();
migrateOldFormats(config);
Config.config = config;
2019-12-31 21:07:43 +01:00
}
2019-12-31 23:46:16 +01:00
// Reset config
function resetConfig() {
Config.config = Config.defaults;
}
2019-12-31 23:46:16 +01:00
function convertJSON(): void {
2020-02-09 02:05:31 +01:00
Object.keys(Config.localConfig).forEach(key => {
Config.localConfig[key] = decodeStoredItem(key, Config.localConfig[key]);
});
2020-01-09 17:39:23 +01:00
}
2020-01-09 18:46:04 +01:00
2019-12-31 23:46:16 +01:00
// Add defaults
function addDefaults() {
for (const key in Config.defaults) {
if(!Object.prototype.hasOwnProperty.call(Config.localConfig, key)) {
Config.localConfig[key] = Config.defaults[key];
2020-07-03 03:37:38 +02:00
} else if (key === "barTypes") {
for (const key2 in Config.defaults[key]) {
if(!Object.prototype.hasOwnProperty.call(Config.localConfig[key], key2)) {
2020-07-03 03:37:38 +02:00
Config.localConfig[key][key2] = Config.defaults[key][key2];
}
}
}
}
}
2019-12-31 23:46:16 +01:00
2019-12-31 21:07:43 +01:00
// Sync config
2020-01-09 18:46:04 +01:00
setupConfig();
2020-01-29 04:16:48 +01:00
2020-06-26 20:19:51 +02:00
export default Config;