Adds a settings menu to change WhatsApp Web notification sound/volume
// ==UserScript==
// @name WhatsApp Web Notification Sound Changer
// @description Adds a settings menu to change WhatsApp Web notification sound/volume
// @version 1.2
// @license GPL-3.0-or-later
// @require https://openuserjs.org/src/libs/sizzle/GM_config.js
// @grant GM_getValue
// @grant GM_setValue
// @match *://web.whatsapp.com/*
// @run-at document-start
// @namespace https://greasyfork.org/users/751814
// ==/UserScript==
var config_image = "";
const sleep = ms => new Promise(res => setTimeout(res, ms));
function doStore()
{
let c = `
window.notificationAudio.volume = ${GM_config.get('volume')/100}
if(${GM_config.get('useSound')}){
window.notificationAudio.src = "${GM_config.get('sound')}"
} else {
window.notificationAudio.src = window.defaultNotificationAudioPath
}`;
unsafeWindow.eval(c);
}
GM_config.init(
{
'id': 'wa_web_cfg',
'title': 'Notification Settings',
'fields':
{
'volume': {
'label': 'Notification Volume Percent',
'type': 'int',
'min': 0,
'max': 100,
'default': 100
},
'sound': {
'label': 'Notification Sound File',
'type': 'fileupload'
},
'useSound': {
'label': 'Use custom sound',
'type': 'checkbox',
'default': false
},
'playButton': {
'label': 'Play notification sound',
'type': 'button',
'click': function() {
unsafeWindow.notificationAudio.play();
}
}
},
'events':
{
'save': doStore
},
'types':
{
'fileupload': {
'default': null,
toNode: function(configId) {
var field = this.settings,
id = this.id,
create = this.create,
retNode = create('div', { className: 'config_var',
id: configId + '_' + id + '_var',
title: field.title || '' });
retNode.appendChild(create('label', {
innerHTML: field.label,
id: configId + '_' + id + '_field_label',
for: configId + '_field_' + id,
className: 'field_label'
}));
var props = {
id: configId + '_field_' + id,
type: 'file'
};
let input = create('input', props);
input.onchange = function() {
let reader = new FileReader();
reader.readAsDataURL(input.files[0]);
reader.onload = function() {
input.setAttribute('datauri', reader.result);
};
};
retNode.appendChild(input);
return retNode;
},
toValue: function() {
var rval = null;
if (this.wrapper) {
var input = this.wrapper.getElementsByTagName('input')[0];
rval = input.getAttribute('datauri');
}
return rval;
},
reset: function() {
if (this.wrapper) {
var input = this.wrapper.getElementsByTagName('input')[0];
input.value = '';
}
}
}
}
});
document.addEventListener('readystatechange', async (event) => {
if(document.readyState === 'interactive') {
unsafeWindow.eval(`
window.Audio = class extends Audio
{
constructor(x){
super(x);
if(x != undefined && x.includes('notification')) {
window.notificationAudio = this;
window.defaultNotificationAudioPath = x;
}
}
};
` );
}
if(document.readyState === 'complete') {
var headers = [];
while (headers.length < 1) {
headers = document.querySelectorAll('header');
await sleep(200);
}
var config = document.createElement('img');
config.src = config_image;
config.style.height = "75%";
config.style.cursor = "pointer";
config.onclick = ()=>GM_config.open();
headers[0].appendChild(config);
doStore();
}
});