uBlock/src/js/settings.js
Raymond Hill 46ea5519c1
Add checkboxes to "My filters" pane
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.
2024-03-11 11:39:31 -04:00

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
);
/******************************************************************************/