vk voice message download

Allows you to download voice messages!

// ==UserScript==
// @name            vk voice message download
// @name:ru         Загрузка голосовых сообщений из ВК
// @namespace       http://tampermonkey.net/
// @version         0.1.0
// @description     Allows you to download voice messages!
// @description:ru  Позволяет скачивать голосовые сообщения из ВК!
// @author          VEGAMETA
// @match           *://vk.com/im*
// @match           *://m.vk.com/mail*
// @icon            
// @grant           none
// @license         MIT
// ==/UserScript==

(function() {
    'use strict';

    function createButton(link) {
        const icon = document.createElement('div');
        icon.style.position = 'absolute';
        icon.style.width = '18px';
        icon.style.height = '18px';
        icon.style.marginTop = '22px';
        icon.style.marginLeft = '2px';
        icon.style.backgroundImage = 'url("")';
        icon.style.backgroundSize = 'contain';
        icon.style.zIndex = 1;
        icon.addEventListener('click', async () => {
            event.stopPropagation();
            try {
                const response = await fetch(link);
                const blob = await response.blob();
                const url = window.URL.createObjectURL(blob);
                const anchor = document.createElement('a');
                anchor.href = url;
                anchor.setAttribute('download', 'voicemessage.mp3'); // задать имя файла для скачивания
                document.body.appendChild(anchor);
                anchor.click();
                document.body.removeChild(anchor);
                window.URL.revokeObjectURL(url);
            } catch (error) {
                console.error('Download failed:', error);
            }
        });
        return icon;
    }

    function processBlocks() {
        const isMobileSite = window.location.hostname.startsWith('m.');
        const messageSelector = isMobileSite ? '.im_msg_audiomsg' : '.im_msg_media_audio_message';

        const blocks = document.querySelectorAll(messageSelector);
        blocks.forEach(block => {
            const vmsg = block.querySelector('.audio-msg-track');
            const existingIcon = block.querySelector('.download-button');
            if (existingIcon) return;
            const link = vmsg.getAttribute('data-mp3');
            if (!link) return;
            const icon = createButton(link);
            icon.classList.add('download-button');
            block.insertBefore(icon, block.firstChild);
        });
    }

    processBlocks();

    const observer = new MutationObserver(processBlocks);
    observer.observe(document.body, {
        childList: true,
        subtree: true
    });
})();