mirror of
https://github.com/gorhill/uBlock.git
synced 2026-03-11 09:04:36 +00:00
Related issues: - https://github.com/uBlockOrigin/uBlock-issues/issues/3161 - https://github.com/uBlockOrigin/uBlock-issues/discussions/2895#discussioncomment-8504374 Two checkboxes have been added to the "My filters "pane: 1. A checkbox to wholly disable/enable "My filters". This is equivalent to the checkbox for "My filters" in "Filter lists" pane. 2. A checkbox to enable/disable the trustworthiness of the content of "My filters". Default to untrusted. Since toggling these checkboxes requires reloading all filter lists, their new state must be committed through the "Apply changes" button. Additionally: a "book" icon has been added to the top-right of the dashboard, which is a link to the wiki according to whichever pane is currently active.
328 lines
10 KiB
JavaScript
328 lines
10 KiB
JavaScript
/*******************************************************************************
|
|
|
|
uBlock Origin - a comprehensive, efficient content blocker
|
|
Copyright (C) 2014-present Raymond Hill
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program. If not, see {http://www.gnu.org/licenses/}.
|
|
|
|
Home: https://github.com/gorhill/uBlock
|
|
*/
|
|
|
|
'use strict';
|
|
|
|
import { i18n$ } from './i18n.js';
|
|
import { dom, qs$, qsa$ } from './dom.js';
|
|
import { setAccentColor, setTheme } from './theme.js';
|
|
|
|
/******************************************************************************/
|
|
|
|
function handleImportFilePicker() {
|
|
const file = this.files[0];
|
|
if ( file === undefined || file.name === '' ) { return; }
|
|
|
|
const reportError = ( ) => {
|
|
window.alert(i18n$('aboutRestoreDataError'));
|
|
};
|
|
|
|
const expectedFileTypes = [
|
|
'text/plain',
|
|
'application/json',
|
|
];
|
|
if ( expectedFileTypes.includes(file.type) === false ) {
|
|
return reportError();
|
|
}
|
|
|
|
const filename = file.name;
|
|
const fr = new FileReader();
|
|
|
|
fr.onload = function() {
|
|
let userData;
|
|
try {
|
|
userData = JSON.parse(this.result);
|
|
if ( typeof userData !== 'object' ) {
|
|
throw 'Invalid';
|
|
}
|
|
if ( typeof userData.userSettings !== 'object' ) {
|
|
throw 'Invalid';
|
|
}
|
|
if (
|
|
Array.isArray(userData.whitelist) === false &&
|
|
typeof userData.netWhitelist !== 'string'
|
|
) {
|
|
throw 'Invalid';
|
|
}
|
|
if (
|
|
typeof userData.filterLists !== 'object' &&
|
|
Array.isArray(userData.selectedFilterLists) === false
|
|
) {
|
|
throw 'Invalid';
|
|
}
|
|
}
|
|
catch (e) {
|
|
userData = undefined;
|
|
}
|
|
if ( userData === undefined ) {
|
|
return reportError();
|
|
}
|
|
const time = new Date(userData.timeStamp);
|
|
const msg = i18n$('aboutRestoreDataConfirm')
|
|
.replace('{{time}}', time.toLocaleString());
|
|
const proceed = window.confirm(msg);
|
|
if ( proceed !== true ) { return; }
|
|
vAPI.messaging.send('dashboard', {
|
|
what: 'restoreUserData',
|
|
userData,
|
|
file: filename,
|
|
});
|
|
};
|
|
|
|
fr.readAsText(file);
|
|
}
|
|
|
|
/******************************************************************************/
|
|
|
|
function startImportFilePicker() {
|
|
const input = qs$('#restoreFilePicker');
|
|
// Reset to empty string, this will ensure an change event is properly
|
|
// triggered if the user pick a file, even if it is the same as the last
|
|
// one picked.
|
|
input.value = '';
|
|
input.click();
|
|
}
|
|
|
|
/******************************************************************************/
|
|
|
|
async function exportToFile() {
|
|
const response = await vAPI.messaging.send('dashboard', {
|
|
what: 'backupUserData',
|
|
});
|
|
if (
|
|
response instanceof Object === false ||
|
|
response.userData instanceof Object === false
|
|
) {
|
|
return;
|
|
}
|
|
vAPI.download({
|
|
'url': 'data:text/plain;charset=utf-8,' +
|
|
encodeURIComponent(JSON.stringify(response.userData, null, ' ')),
|
|
'filename': response.localData.lastBackupFile
|
|
});
|
|
onLocalDataReceived(response.localData);
|
|
}
|
|
|
|
/******************************************************************************/
|
|
|
|
function onLocalDataReceived(details) {
|
|
let v, unit;
|
|
if ( typeof details.storageUsed === 'number' ) {
|
|
v = details.storageUsed;
|
|
if ( v < 1e3 ) {
|
|
unit = 'genericBytes';
|
|
} else if ( v < 1e6 ) {
|
|
v /= 1e3;
|
|
unit = 'KB';
|
|
} else if ( v < 1e9 ) {
|
|
v /= 1e6;
|
|
unit = 'MB';
|
|
} else {
|
|
v /= 1e9;
|
|
unit = 'GB';
|
|
}
|
|
} else {
|
|
v = '?';
|
|
unit = '';
|
|
}
|
|
dom.text(
|
|
'#storageUsed',
|
|
i18n$('storageUsed')
|
|
.replace('{{value}}', v.toLocaleString(undefined, { maximumSignificantDigits: 3 }))
|
|
.replace('{{unit}}', unit && i18n$(unit) || '')
|
|
);
|
|
|
|
const timeOptions = {
|
|
weekday: 'long',
|
|
year: 'numeric',
|
|
month: 'long',
|
|
day: 'numeric',
|
|
hour: 'numeric',
|
|
minute: 'numeric',
|
|
timeZoneName: 'short'
|
|
};
|
|
|
|
const lastBackupFile = details.lastBackupFile || '';
|
|
if ( lastBackupFile !== '' ) {
|
|
const dt = new Date(details.lastBackupTime);
|
|
const text = i18n$('settingsLastBackupPrompt');
|
|
const node = qs$('#settingsLastBackupPrompt');
|
|
node.textContent = text + '\xA0' + dt.toLocaleString('fullwide', timeOptions);
|
|
node.style.display = '';
|
|
}
|
|
|
|
const lastRestoreFile = details.lastRestoreFile || '';
|
|
if ( lastRestoreFile !== '' ) {
|
|
const dt = new Date(details.lastRestoreTime);
|
|
const text = i18n$('settingsLastRestorePrompt');
|
|
const node = qs$('#settingsLastRestorePrompt');
|
|
node.textContent = text + '\xA0' + dt.toLocaleString('fullwide', timeOptions);
|
|
node.style.display = '';
|
|
}
|
|
|
|
if ( details.cloudStorageSupported === false ) {
|
|
dom.attr('[data-setting-name="cloudStorageEnabled"]', 'disabled', '');
|
|
}
|
|
|
|
if ( details.privacySettingsSupported === false ) {
|
|
dom.attr('[data-setting-name="prefetchingDisabled"]', 'disabled', '');
|
|
dom.attr('[data-setting-name="hyperlinkAuditingDisabled"]', 'disabled', '');
|
|
dom.attr('[data-setting-name="webrtcIPAddressHidden"]', 'disabled', '');
|
|
}
|
|
}
|
|
|
|
/******************************************************************************/
|
|
|
|
function resetUserData() {
|
|
const msg = i18n$('aboutResetDataConfirm');
|
|
const proceed = window.confirm(msg);
|
|
if ( proceed !== true ) { return; }
|
|
vAPI.messaging.send('dashboard', {
|
|
what: 'resetUserData',
|
|
});
|
|
}
|
|
|
|
/******************************************************************************/
|
|
|
|
function synchronizeDOM() {
|
|
dom.cl.toggle(
|
|
dom.body,
|
|
'advancedUser',
|
|
qs$('[data-setting-name="advancedUserEnabled"]').checked === true
|
|
);
|
|
}
|
|
|
|
/******************************************************************************/
|
|
|
|
function changeUserSettings(name, value) {
|
|
vAPI.messaging.send('dashboard', {
|
|
what: 'userSettings',
|
|
name,
|
|
value,
|
|
});
|
|
|
|
// Maybe reflect some changes immediately
|
|
switch ( name ) {
|
|
case 'uiTheme':
|
|
setTheme(value, true);
|
|
break;
|
|
case 'uiAccentCustom':
|
|
case 'uiAccentCustom0':
|
|
setAccentColor(
|
|
qs$('[data-setting-name="uiAccentCustom"]').checked,
|
|
qs$('[data-setting-name="uiAccentCustom0"]').value,
|
|
true
|
|
);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
/******************************************************************************/
|
|
|
|
function onValueChanged(ev) {
|
|
const input = ev.target;
|
|
const name = dom.attr(input, 'data-setting-name');
|
|
let value = input.value;
|
|
// Maybe sanitize value
|
|
switch ( name ) {
|
|
case 'largeMediaSize':
|
|
value = Math.min(Math.max(Math.floor(parseInt(value, 10) || 0), 0), 1000000);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
if ( value !== input.value ) {
|
|
input.value = value;
|
|
}
|
|
|
|
changeUserSettings(name, value);
|
|
}
|
|
|
|
/******************************************************************************/
|
|
|
|
// TODO: use data-* to declare simple settings
|
|
|
|
function onUserSettingsReceived(details) {
|
|
const checkboxes = qsa$('[data-setting-type="bool"]');
|
|
const onchange = ev => {
|
|
const checkbox = ev.target;
|
|
const name = checkbox.dataset.settingName || '';
|
|
changeUserSettings(name, checkbox.checked);
|
|
synchronizeDOM();
|
|
};
|
|
for ( const checkbox of checkboxes ) {
|
|
const name = dom.attr(checkbox, 'data-setting-name') || '';
|
|
if ( details[name] === undefined ) {
|
|
dom.attr(checkbox.closest('.checkbox'), 'disabled', '');
|
|
dom.attr(checkbox, 'disabled', '');
|
|
continue;
|
|
}
|
|
checkbox.checked = details[name] === true;
|
|
dom.on(checkbox, 'change', onchange);
|
|
}
|
|
|
|
if ( details.canLeakLocalIPAddresses === true ) {
|
|
qs$('[data-setting-name="webrtcIPAddressHidden"]')
|
|
.closest('div.li')
|
|
.style.display = '';
|
|
}
|
|
|
|
qsa$('[data-setting-type="value"]').forEach(function(elem) {
|
|
elem.value = details[dom.attr(elem, 'data-setting-name')];
|
|
dom.on(elem, 'change', onValueChanged);
|
|
});
|
|
|
|
dom.on('#export', 'click', ( ) => { exportToFile(); });
|
|
dom.on('#import', 'click', startImportFilePicker);
|
|
dom.on('#reset', 'click', resetUserData);
|
|
dom.on('#restoreFilePicker', 'change', handleImportFilePicker);
|
|
|
|
synchronizeDOM();
|
|
}
|
|
|
|
/******************************************************************************/
|
|
|
|
self.wikilink = 'https://github.com/gorhill/uBlock/wiki/Dashboard:-Settings';
|
|
|
|
self.hasUnsavedData = function() {
|
|
return false;
|
|
};
|
|
|
|
/******************************************************************************/
|
|
|
|
vAPI.messaging.send('dashboard', { what: 'userSettings' }).then(result => {
|
|
onUserSettingsReceived(result);
|
|
});
|
|
|
|
vAPI.messaging.send('dashboard', { what: 'getLocalData' }).then(result => {
|
|
onLocalDataReceived(result);
|
|
});
|
|
|
|
// https://github.com/uBlockOrigin/uBlock-issues/issues/591
|
|
dom.on(
|
|
'[data-i18n-title="settingsAdvancedUserSettings"]',
|
|
'click',
|
|
self.uBlockDashboard.openOrSelectPage
|
|
);
|
|
|
|
/******************************************************************************/
|