mysterypond / WhatsApp Web Notification Sound Changer

// ==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();
  }
});