Return Pikabu minus

Возвращает минусы на Pikabu, а также фильтрацию по рейтингу.

// ==UserScript==
// @name         Return Pikabu minus
// @version      0.9.6
// @namespace    pikabu-return-minus.pyxiion.ru
// @description  Возвращает минусы на Pikabu, а также фильтрацию по рейтингу.
// @author       PyXiion
// @match        *://pikabu.ru/*
// @connect      api.pikabu.ru
// @connect      pikabu.ru
// @connect      rpm.pyxiion.ru
// @connect      gollum.space
// @connect      isla-de-muerta.com
// @grant        GM.xmlHttpRequest
// @grant        GM.getValue
// @grant        GM.setValue
// @grant        GM.registerMenuCommand
// @require      https://openuserjs.org/src/libs/sizzle/GM_config.js
// @license      MIT
// ==/UserScript==
//#region Utils
function MD5(string) {
    function RotateLeft(lValue, iShiftBits) {
        return (lValue << iShiftBits) | (lValue >>> (32 - iShiftBits));
    }
    function AddUnsigned(lX, lY) {
        var lX4, lY4, lX8, lY8, lResult;
        lX8 = (lX & 0x80000000);
        lY8 = (lY & 0x80000000);
        lX4 = (lX & 0x40000000);
        lY4 = (lY & 0x40000000);
        lResult = (lX & 0x3FFFFFFF) + (lY & 0x3FFFFFFF);
        if (lX4 & lY4) {
            return (lResult ^ 0x80000000 ^ lX8 ^ lY8);
        }
        if (lX4 | lY4) {
            if (lResult & 0x40000000) {
                return (lResult ^ 0xC0000000 ^ lX8 ^ lY8);
            }
            else {
                return (lResult ^ 0x40000000 ^ lX8 ^ lY8);
            }
        }
        else {
            return (lResult ^ lX8 ^ lY8);
        }
    }
    function F(x, y, z) { return (x & y) | ((~x) & z); }
    function G(x, y, z) { return (x & z) | (y & (~z)); }
    function H(x, y, z) { return (x ^ y ^ z); }
    function I(x, y, z) { return (y ^ (x | (~z))); }
    function FF(a, b, c, d, x, s, ac) {
        a = AddUnsigned(a, AddUnsigned(AddUnsigned(F(b, c, d), x), ac));
        return AddUnsigned(RotateLeft(a, s), b);
    }
    ;
    function GG(a, b, c, d, x, s, ac) {
        a = AddUnsigned(a, AddUnsigned(AddUnsigned(G(b, c, d), x), ac));
        return AddUnsigned(RotateLeft(a, s), b);
    }
    ;
    function HH(a, b, c, d, x, s, ac) {
        a = AddUnsigned(a, AddUnsigned(AddUnsigned(H(b, c, d), x), ac));
        return AddUnsigned(RotateLeft(a, s), b);
    }
    ;
    function II(a, b, c, d, x, s, ac) {
        a = AddUnsigned(a, AddUnsigned(AddUnsigned(I(b, c, d), x), ac));
        return AddUnsigned(RotateLeft(a, s), b);
    }
    ;
    function ConvertToWordArray(string) {
        var lWordCount;
        var lMessageLength = string.length;
        var lNumberOfWords_temp1 = lMessageLength + 8;
        var lNumberOfWords_temp2 = (lNumberOfWords_temp1 - (lNumberOfWords_temp1 % 64)) / 64;
        var lNumberOfWords = (lNumberOfWords_temp2 + 1) * 16;
        var lWordArray = Array(lNumberOfWords - 1);
        var lBytePosition = 0;
        var lByteCount = 0;
        while (lByteCount < lMessageLength) {
            lWordCount = (lByteCount - (lByteCount % 4)) / 4;
            lBytePosition = (lByteCount % 4) * 8;
            lWordArray[lWordCount] = (lWordArray[lWordCount] | (string.charCodeAt(lByteCount) << lBytePosition));
            lByteCount++;
        }
        lWordCount = (lByteCount - (lByteCount % 4)) / 4;
        lBytePosition = (lByteCount % 4) * 8;
        lWordArray[lWordCount] = lWordArray[lWordCount] | (0x80 << lBytePosition);
        lWordArray[lNumberOfWords - 2] = lMessageLength << 3;
        lWordArray[lNumberOfWords - 1] = lMessageLength >>> 29;
        return lWordArray;
    }
    ;
    function WordToHex(lValue) {
        var WordToHexValue = "", WordToHexValue_temp = "", lByte, lCount;
        for (lCount = 0; lCount <= 3; lCount++) {
            lByte = (lValue >>> (lCount * 8)) & 255;
            WordToHexValue_temp = "0" + lByte.toString(16);
            WordToHexValue = WordToHexValue + WordToHexValue_temp.substr(WordToHexValue_temp.length - 2, 2);
        }
        return WordToHexValue;
    }
    ;
    function Utf8Encode(string) {
        string = string.replace(/\r\n/g, "\n");
        var utftext = "";
        for (var n = 0; n < string.length; n++) {
            var c = string.charCodeAt(n);
            if (c < 128) {
                utftext += String.fromCharCode(c);
            }
            else if ((c > 127) && (c < 2048)) {
                utftext += String.fromCharCode((c >> 6) | 192);
                utftext += String.fromCharCode((c & 63) | 128);
            }
            else {
                utftext += String.fromCharCode((c >> 12) | 224);
                utftext += String.fromCharCode(((c >> 6) & 63) | 128);
                utftext += String.fromCharCode((c & 63) | 128);
            }
        }
        return utftext;
    }
    ;
    var x = Array();
    var k, AA, BB, CC, DD, a, b, c, d;
    var S11 = 7, S12 = 12, S13 = 17, S14 = 22;
    var S21 = 5, S22 = 9, S23 = 14, S24 = 20;
    var S31 = 4, S32 = 11, S33 = 16, S34 = 23;
    var S41 = 6, S42 = 10, S43 = 15, S44 = 21;
    string = Utf8Encode(string);
    x = ConvertToWordArray(string);
    a = 0x67452301;
    b = 0xEFCDAB89;
    c = 0x98BADCFE;
    d = 0x10325476;
    for (k = 0; k < x.length; k += 16) {
        AA = a;
        BB = b;
        CC = c;
        DD = d;
        a = FF(a, b, c, d, x[k + 0], S11, 0xD76AA478);
        d = FF(d, a, b, c, x[k + 1], S12, 0xE8C7B756);
        c = FF(c, d, a, b, x[k + 2], S13, 0x242070DB);
        b = FF(b, c, d, a, x[k + 3], S14, 0xC1BDCEEE);
        a = FF(a, b, c, d, x[k + 4], S11, 0xF57C0FAF);
        d = FF(d, a, b, c, x[k + 5], S12, 0x4787C62A);
        c = FF(c, d, a, b, x[k + 6], S13, 0xA8304613);
        b = FF(b, c, d, a, x[k + 7], S14, 0xFD469501);
        a = FF(a, b, c, d, x[k + 8], S11, 0x698098D8);
        d = FF(d, a, b, c, x[k + 9], S12, 0x8B44F7AF);
        c = FF(c, d, a, b, x[k + 10], S13, 0xFFFF5BB1);
        b = FF(b, c, d, a, x[k + 11], S14, 0x895CD7BE);
        a = FF(a, b, c, d, x[k + 12], S11, 0x6B901122);
        d = FF(d, a, b, c, x[k + 13], S12, 0xFD987193);
        c = FF(c, d, a, b, x[k + 14], S13, 0xA679438E);
        b = FF(b, c, d, a, x[k + 15], S14, 0x49B40821);
        a = GG(a, b, c, d, x[k + 1], S21, 0xF61E2562);
        d = GG(d, a, b, c, x[k + 6], S22, 0xC040B340);
        c = GG(c, d, a, b, x[k + 11], S23, 0x265E5A51);
        b = GG(b, c, d, a, x[k + 0], S24, 0xE9B6C7AA);
        a = GG(a, b, c, d, x[k + 5], S21, 0xD62F105D);
        d = GG(d, a, b, c, x[k + 10], S22, 0x2441453);
        c = GG(c, d, a, b, x[k + 15], S23, 0xD8A1E681);
        b = GG(b, c, d, a, x[k + 4], S24, 0xE7D3FBC8);
        a = GG(a, b, c, d, x[k + 9], S21, 0x21E1CDE6);
        d = GG(d, a, b, c, x[k + 14], S22, 0xC33707D6);
        c = GG(c, d, a, b, x[k + 3], S23, 0xF4D50D87);
        b = GG(b, c, d, a, x[k + 8], S24, 0x455A14ED);
        a = GG(a, b, c, d, x[k + 13], S21, 0xA9E3E905);
        d = GG(d, a, b, c, x[k + 2], S22, 0xFCEFA3F8);
        c = GG(c, d, a, b, x[k + 7], S23, 0x676F02D9);
        b = GG(b, c, d, a, x[k + 12], S24, 0x8D2A4C8A);
        a = HH(a, b, c, d, x[k + 5], S31, 0xFFFA3942);
        d = HH(d, a, b, c, x[k + 8], S32, 0x8771F681);
        c = HH(c, d, a, b, x[k + 11], S33, 0x6D9D6122);
        b = HH(b, c, d, a, x[k + 14], S34, 0xFDE5380C);
        a = HH(a, b, c, d, x[k + 1], S31, 0xA4BEEA44);
        d = HH(d, a, b, c, x[k + 4], S32, 0x4BDECFA9);
        c = HH(c, d, a, b, x[k + 7], S33, 0xF6BB4B60);
        b = HH(b, c, d, a, x[k + 10], S34, 0xBEBFBC70);
        a = HH(a, b, c, d, x[k + 13], S31, 0x289B7EC6);
        d = HH(d, a, b, c, x[k + 0], S32, 0xEAA127FA);
        c = HH(c, d, a, b, x[k + 3], S33, 0xD4EF3085);
        b = HH(b, c, d, a, x[k + 6], S34, 0x4881D05);
        a = HH(a, b, c, d, x[k + 9], S31, 0xD9D4D039);
        d = HH(d, a, b, c, x[k + 12], S32, 0xE6DB99E5);
        c = HH(c, d, a, b, x[k + 15], S33, 0x1FA27CF8);
        b = HH(b, c, d, a, x[k + 2], S34, 0xC4AC5665);
        a = II(a, b, c, d, x[k + 0], S41, 0xF4292244);
        d = II(d, a, b, c, x[k + 7], S42, 0x432AFF97);
        c = II(c, d, a, b, x[k + 14], S43, 0xAB9423A7);
        b = II(b, c, d, a, x[k + 5], S44, 0xFC93A039);
        a = II(a, b, c, d, x[k + 12], S41, 0x655B59C3);
        d = II(d, a, b, c, x[k + 3], S42, 0x8F0CCC92);
        c = II(c, d, a, b, x[k + 10], S43, 0xFFEFF47D);
        b = II(b, c, d, a, x[k + 1], S44, 0x85845DD1);
        a = II(a, b, c, d, x[k + 8], S41, 0x6FA87E4F);
        d = II(d, a, b, c, x[k + 15], S42, 0xFE2CE6E0);
        c = II(c, d, a, b, x[k + 6], S43, 0xA3014314);
        b = II(b, c, d, a, x[k + 13], S44, 0x4E0811A1);
        a = II(a, b, c, d, x[k + 4], S41, 0xF7537E82);
        d = II(d, a, b, c, x[k + 11], S42, 0xBD3AF235);
        c = II(c, d, a, b, x[k + 2], S43, 0x2AD7D2BB);
        b = II(b, c, d, a, x[k + 9], S44, 0xEB86D391);
        a = AddUnsigned(a, AA);
        b = AddUnsigned(b, BB);
        c = AddUnsigned(c, CC);
        d = AddUnsigned(d, DD);
    }
    var temp = WordToHex(a) + WordToHex(b) + WordToHex(c) + WordToHex(d);
    return temp.toLowerCase();
}
;
class AbstractHttpRequest {
    constructor(url, responseType, additionalParameters = null) {
        this.url = url;
        this.httpMethod = "POST";
        this.headers = new Map();
        this.timeout = 15000;
        this.responseType = responseType;
        this.additionalParameters = {
            anonymous: true,
            fetch: true,
            ...additionalParameters
        };
    }
    addHeader(key, value) {
        this.headers.set(key, value);
        return this;
    }
    setHttpMethod(httpMethod) {
        this.httpMethod = httpMethod;
        return this;
    }
    execute(callback) {
        const data = this.getData();
        const details = {
            url: this.url,
            method: this.httpMethod,
            headers: Object.fromEntries(this.headers),
            data: data ? JSON.stringify(data) : null,
            timeout: this.timeout,
            responseType: this.responseType,
            onerror: callback.onError,
            onload: callback.onSuccess,
            // TODO: ontimeout
            onabort: callback.onError,
            ontimeout: callback.onError,
            ...this.additionalParameters
        };
        GM.xmlHttpRequest(details);
    }
    executeAsync() {
        const promise = new Promise((resolve, reject) => {
            this.execute({
                onError: reject,
                onSuccess: resolve
            });
        });
        promise.catch(error);
        return promise;
    }
}
class HttpRequest extends AbstractHttpRequest {
    constructor(url, method = "GET", responseType, additionalParameters = null) {
        super(url, responseType, additionalParameters);
        this.httpMethod = method;
    }
    setBody(body) {
        this.body = body;
    }
    getData() {
        return this.body;
    }
}
//#endregion
//#region Pikabu API
var Pikabu;
(function (Pikabu) {
    const DOMAIN = "https://api.pikabu.ru/";
    const API_V1 = DOMAIN + "v1/";
    const API_KEY = "kmq4!2cPl)peZ";
    class API {
        static getDeviceId() {
            return "0";
        }
    }
    API.USER_AGENT = "ru.pikabu.android/1.21.15 (SM-N975F Android 7.1.2)";
    API.COOKIE = "unqKms867=aba48a160c; rm5bH=8c68fbfe3dc5e5f5b23a9ec1a8f784f8";
    class Request extends AbstractHttpRequest {
        constructor(domain, controller, params) {
            super(domain + controller, "json");
            this.controller = controller;
            this.params = params;
            this.setHttpMethod("GET");
            this.addHeader("DeviceId", API.getDeviceId());
            this.addHeader("User-Agent", API.USER_AGENT);
            this.addHeader("Cookie", API.COOKIE);
            this.addHeader("Content-Type", "application/json");
        }
        setParam(key, value) {
            this.params[key] = value;
        }
        static getHash(data, controller, ms) {
            const join = Object.values(data).sort().join(",");
            const toHash = [API_KEY, controller, ms, join].join(",");
            const hashed = MD5(toHash);
            return btoa(hashed);
        }
        getData() {
            const ms = Date.now();
            const data = {
                new_sort: 1,
                ...this.params,
            };
            return {
                ...data,
                id: "iws",
                hash: Request.getHash(data, this.controller, ms),
                token: ms,
            };
        }
        async executeAsync() {
            const response = await super.executeAsync();
            const data = response.response;
            if (!("response" in data)) {
                throw new Error(data?.error?.message ?? "Unknown error");
            }
            return data.response;
        }
    }
    class PostRequest extends Request {
        constructor(controller, params) {
            super(API_V1, controller, params);
            this.setHttpMethod("POST");
        }
    }
    class RatingObject {
    }
    Pikabu.RatingObject = RatingObject;
    class Post extends RatingObject {
        constructor(payload) {
            super();
            this.videos = [];
            this.id = payload.story_id;
            this.pluses = payload.story_pluses ?? 0;
            this.minuses = payload.story_minuses ?? 0;
            this.rating = payload.story_digs ?? (this.pluses - this.minuses);
            this.parseData(payload.story_data);
        }
        parseData(dataArr) {
            for (let data of dataArr) {
                if (data.type.includes('v')) { // v means video (maybe)
                    data = data.data;
                    let urls = [];
                    let extensions = ['mp4', 'webm', 'av1'];
                    for (let ext of extensions) {
                        if (ext in data && data[ext].url) {
                            urls.push(data[ext].url);
                        }
                    }
                    this.videos.push(urls);
                }
            }
        }
    }
    Pikabu.Post = Post;
    class Comment extends RatingObject {
        constructor(payload) {
            super();
            this.id = payload.comment_id;
            this.parentId = payload.parent_id;
            this.rating = payload.comment_rating ?? 0;
            this.pluses = payload.comment_pluses ?? 0;
            this.minuses = payload.comment_minuses ?? 0;
            this.videos = (payload.comment_desc.videos ?? []).flatMap((v) => v.url);
        }
    }
    Pikabu.Comment = Comment;
    class StoryData {
        constructor(payload) {
            this.story = "story" in payload ? new Post(payload.story) : null;
        }
    }
    Pikabu.StoryData = StoryData;
    class CommentsData extends StoryData {
        constructor(payload) {
            super(payload);
            this.selectedCommentId = 0;
            this.comments = payload.comments.map((x) => new Comment(x));
            this.hasMoreComments = payload.has_next_page_comments;
        }
    }
    Pikabu.CommentsData = CommentsData;
    let DataService;
    (function (DataService) {
        async function fetchStory(storyId, commentsPage) {
            const params = {
                story_id: storyId,
                page: commentsPage,
            };
            try {
                const request = new PostRequest("story.get", params);
                const payload = (await request.executeAsync());
                const commentsData = new CommentsData(payload);
                return commentsData;
            }
            catch (e) {
                error(e);
                return null;
            }
        }
        DataService.fetchStory = fetchStory;
    })(DataService = Pikabu.DataService || (Pikabu.DataService = {}));
})(Pikabu || (Pikabu = {}));
//#endregion
//#region API/Nodes
var RPM;
(function (RPM) {
    let Service;
    (function (Service) {
        const DOMAIN = 'https://rpm.pyxiion.ru/';
        // const DOMAIN = 'http://localhost:8000/'
        const USER_REQUEST_QUEUE_PERIOD = 300;
        let PERIOD_MULTIPLIER = 1;
        const USER_REQUEST_QUEUE_AT_ONCE = 50;
        function isAuthorized() {
            return GM_config.get('uuid') !== '';
        }
        Service.isAuthorized = isAuthorized;
        async function register() {
            const response = (await post(DOMAIN + 'register', {}));
            return response.secret;
        }
        Service.register = register;
        async function getFeedbacks() {
            const response = (await get(DOMAIN + 'meta/feedback'));
            return response;
        }
        Service.getFeedbacks = getFeedbacks;
        const userInfoRequestQueue = new Map();
        let isQueueRunning = false;
        function getUserInfo(id) {
            return new Promise((resolve) => {
                if (!userInfoRequestQueue.has(id)) {
                    userInfoRequestQueue.set(id, [{ callback: resolve }]);
                }
                else {
                    userInfoRequestQueue.get(id).push({ callback: resolve });
                }
                workQueue(USER_REQUEST_QUEUE_PERIOD * PERIOD_MULTIPLIER);
            });
        }
        Service.getUserInfo = getUserInfo;
        async function getBunchOfUserInfo(ids) {
            const body = { ids };
            const uuid = GM_config.get('uuid');
            if (uuid)
                body.user_uuid = uuid;
            const response = (await post(DOMAIN + 'user/info_bunch', body));
            const users = response.users;
            for (const id in users) {
                postprocessUserInfo(users[id]);
            }
            return users;
        }
        async function workQueue(sleepTime = 0) {
            if (userInfoRequestQueue.size === 0 || isQueueRunning)
                return;
            isQueueRunning = true;
            await sleep(sleepTime);
            // Извлекаем до N уникальных запросов из очереди
            const requestsToProcess = Array.from(userInfoRequestQueue.keys()).slice(0, USER_REQUEST_QUEUE_AT_ONCE);
            const ids = requestsToProcess;
            try {
                const usersInfo = await getBunchOfUserInfo(ids);
                // Вызываем все callback для каждого запроса
                requestsToProcess.forEach(id => {
                    const userRequests = userInfoRequestQueue.get(id);
                    if (userRequests) {
                        const info = usersInfo[id] || null; // null если инфо не найдено
                        userRequests.forEach(req => req.callback(info));
                    }
                    userInfoRequestQueue.delete(id); // Удаляем обработанные запросы
                });
            }
            catch (error) {
                error("Error processing user info requests:", error);
                PERIOD_MULTIPLIER += 1;
            }
            finally {
                isQueueRunning = false;
            }
            // Повторный запуск, если есть еще запросы
            setTimeout(workQueue, USER_REQUEST_QUEUE_PERIOD * PERIOD_MULTIPLIER);
        }
        function postprocessUserInfo(info) {
            if (info.own_vote) {
                // Removes own vote from other votes
                info.pluses -= info.own_vote === 1 ? 1 : 0;
                info.minuses -= info.own_vote === -1 ? 1 : 0;
            }
        }
        function voteUser(id, vote) {
            if (!isAuthorized())
                return null;
            return post(DOMAIN + `user/${id}/vote`, { user_uuid: GM_config.get('uuid'), vote });
        }
        Service.voteUser = voteUser;
        async function post(url, json) {
            const request = new HttpRequest(url, "POST", "json");
            request.addHeader("Content-Type", "application/json");
            request.setBody(json);
            const response = await request.executeAsync();
            return response.response;
        }
        async function get(url) {
            const request = new HttpRequest(url, "GET", "json");
            const response = await request.executeAsync();
            return response.response;
        }
    })(Service = RPM.Service || (RPM.Service = {}));
    let Nodes;
    (function (Nodes) {
        function createLoadingIcon() {
            const loadingIcon = document.createElement('div');
            loadingIcon.classList.add('rpm-loading');
            return loadingIcon;
        }
        Nodes.createLoadingIcon = createLoadingIcon;
        function createPoweredNote(text) {
            const elem = document.createElement('span');
            elem.classList.add('rpm-powered');
            elem.innerHTML = text;
            return elem;
        }
        Nodes.createPoweredNote = createPoweredNote;
        function createUserRatingNode(uid, infoConsumer = null) {
            const elem = document.createElement('div');
            elem.classList.add('rpm-user-rating', 'hint', `rpm-user-rating-${uid}`);
            elem.setAttribute('aria-label', 'Рейтинг автора в RPM');
            elem.setAttribute('pikabu-user-id', uid.toString());
            function addSpan(cls) {
                const e = document.createElement('span');
                e.className = cls;
                elem.appendChild(e);
                e.innerText = '0';
                e.style.display = 'none';
                return e;
            }
            const loadingIcon = createLoadingIcon();
            elem.appendChild(loadingIcon);
            const plusElem = addSpan("rpm-pluses");
            addSpan("rpm-rating");
            const minusElem = addSpan("rpm-minuses");
            if (Service.isAuthorized()) {
                plusElem.addEventListener('click', () => UserRating.voteCallback(elem, uid, 1));
                minusElem.addEventListener('click', () => UserRating.voteCallback(elem, uid, -1));
            }
            else {
                const msgCallback = () => sendNotification('Ошибка', 'Авторизируйтесь в системе RPM в настройках скрипта, чтобы голосовать за авторов.');
                plusElem.addEventListener('click', msgCallback);
                minusElem.addEventListener('click', msgCallback);
            }
            UserRating.updateUserRatingElemAsync(elem, infoConsumer);
            return elem;
        }
        Nodes.createUserRatingNode = createUserRatingNode;
        let UserRating;
        (function (UserRating) {
            const userCache = new Map();
            function updateUserRatingElem(elem, info) {
                const pluses = info.pluses + (info.own_vote === 1 ? 1 : 0);
                const minuses = info.minuses + (info.own_vote === -1 ? 1 : 0);
                const rating = pluses - minuses + info.base_rating;
                if (info.own_vote !== undefined && info.own_vote !== null)
                    elem.setAttribute('rpm-own-vote', info.own_vote.toString());
                function set(sel, value) {
                    const e = elem.querySelector(sel);
                    e.style.display = '';
                    e.innerText = value.toString();
                }
                set('.rpm-pluses', pluses);
                set('.rpm-rating', rating);
                set('.rpm-minuses', minuses);
                elem.querySelector('.rpm-loading')?.remove();
            }
            async function updateUserRatingElemAsync(elem, infoConsumer = null) {
                const uid = parseInt(elem.getAttribute('pikabu-user-id'));
                if (uid === null || uid === undefined || Number.isNaN(uid))
                    return;
                let info = userCache.get(uid) ?? null;
                if (!info) {
                    info = await Service.getUserInfo(uid);
                    userCache.set(uid, info);
                }
                if (infoConsumer)
                    infoConsumer(info);
                updateUserRatingElem(elem, info);
            }
            UserRating.updateUserRatingElemAsync = updateUserRatingElemAsync;
            async function voteUser(uid, vote) {
                const info = userCache.get(uid);
                info.own_vote = vote;
                updateAll(uid, info);
                const response = await Service.voteUser(uid, vote);
            }
            function updateAll(uid, info) {
                getAllElementsOfUser(uid).forEach(e => updateUserRatingElem(e, info));
            }
            function getAllElementsOfUser(uid) {
                return document.querySelectorAll(`.rpm-user-rating-${uid}`);
            }
            async function voteCallback(elem, uid, btn) {
                if (!Service.isAuthorized()) {
                    sendNotification('Ошибка', 'Чтобы проголосовать за автора, нужно зарегистрироваться в системе RPM. Вы можете сделать это в настройках.');
                    return;
                }
                const ownVote = parseInt(elem.getAttribute('rpm-own-vote') ?? '0');
                let vote = ownVote;
                if (ownVote === btn)
                    vote = 0;
                else
                    vote += btn;
                info(vote);
                await voteUser(uid, vote);
                // TODO: check response
            }
            UserRating.voteCallback = voteCallback;
        })(UserRating || (UserRating = {}));
    })(Nodes = RPM.Nodes || (RPM.Nodes = {}));
})(RPM || (RPM = {}));
//#endregion
//#region Gollum
var Gollum;
(function (Gollum) {
    function createGollumTagsGetter(tagsType) {
        return async (userId) => {
            const request = new HttpRequest(`https://gollum.space/api/${userId}-${tagsType}`, "GET", "json");
            const response = await request.executeAsync();
            const data = response.response;
            return Object.values(data).map(x => x.TagRU);
        };
    }
    Gollum.storyTagsGetter = createGollumTagsGetter("PostTags");
    Gollum.commentTagsGetter = createGollumTagsGetter("CommentTags");
    const idsCache = new Map();
    async function getUserId(userName) {
        if (idsCache.has(userName)) {
            return idsCache.get(userName);
        }
        const request = new HttpRequest(`https://gollum.space/user/${userName.replace('.', '_')}-summary`, "GET", "text");
        const response = await request.executeAsync();
        const doc = response.responseText;
        // we need $.get("/api/2036358-usertotalposts"
        //                     ^^^^^^^
        const id = parseInt(doc.match(/\/api\/(\d+)-/i)[1]);
        idsCache.set(userName, id);
        return id;
    }
    let LastComments;
    (function (LastComments) {
        const CHUNK_SIZE = 5;
        const COMMENT_UP_DEPTH = 2;
        const COMMENT_DOWN_DEPTH = 2;
        const parser = new DOMParser();
        async function loadCommentFromPikabu(info, parent = false, disableMiniProfile = false) {
            const request = new HttpRequest(info.link, "GET", "arraybuffer", {
                anonymous: false
            });
            request.addHeader('User-Agent', 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.124 Safari/537.36 Edg/91.0.864.59');
            const response = await request.executeAsync();
            // Ебал я этот ваш Пикабу, нахуя он использует CP1251
            const buffer = response.response;
            const dec = new TextDecoder('windows-1251');
            const body = dec.decode(new Uint8Array(buffer));
            const htmlDoc = parser.parseFromString(body, 'text/html');
            const wantedElem = htmlDoc.querySelector(parent === false ? `#comment_${info.id}` : '.comments');
            const clone = wantedElem.cloneNode(true);
            return postprocessPikabuComment(clone, info.id, disableMiniProfile);
        }
        function postprocessPikabuComment(element, commentId, disableMiniProfile = false) {
            element.classList.add('rpm-comment-no-js');
            element.querySelectorAll('.comment').forEach(x => {
                x.classList.add(x.getAttribute('id'));
                x.removeAttribute('id');
                // Fix comment toggling
                const children = x.querySelector(':scope > .comment__children');
                const toggle = x.querySelector(':scope > .comment-toggle-children');
                if (children && toggle) {
                    toggle.addEventListener('click', (e) => {
                        e.preventDefault();
                        children.toggleAttribute('hidden');
                        toggle.classList.toggle('comment-toggle-children_collapse');
                    });
                }
            });
            const commentElem = element.querySelector(`.comment_${commentId}`);
            commentElem.classList.add('rpm-highlight-comment');
            const children = commentElem.querySelector(':scope > .comment__children');
            const toggle = commentElem.querySelector(':scope > .comment-toggle-children');
            if (children && toggle) {
                children.toggleAttribute('hidden');
                toggle.classList.toggle('comment-toggle-children_collapse');
            }
            // Depth control
            // commentElem.querySelectorAll(':scope' + ' > .comment__children > .comment'.repeat(COMMENT_DOWN_DEPTH + 1) + ', :scope' + ' > .comment__children'.repeat(COMMENT_DOWN_DEPTH) + '> .comment-toggle-children').forEach(x => {
            //   x.remove();
            // });
            let parent = commentElem;
            for (let i = 0; i < COMMENT_UP_DEPTH; ++i) {
                const container = parent.parentElement;
                if (container !== null && container.matches('.comment__children')) {
                    for (const child of container.childNodes) {
                        if (child.nodeType === Node.ELEMENT_NODE && !parent.isSameNode(child)) {
                            child.remove();
                            // (child as HTMLElement).style.opacity = '50%';
                        }
                    }
                    const parentComment = container.parentElement;
                    parent = parentComment;
                }
            }
            if (disableMiniProfile) {
                element.querySelectorAll('.comment__user').forEach((x) => {
                    x.removeAttribute('data-profile');
                });
            }
            element.querySelectorAll('img').forEach((x) => {
                if (x.hasAttribute('data-src')) {
                    x.src = x.getAttribute('data-src');
                    x.classList.add('image-loaded');
                }
            });
            return parent;
        }
        function createCommentContainer(info, autoload, disableMiniProfile) {
            const container = document.createElement('div');
            container.classList.add('rpm-comment-container');
            const title = document.createElement('a');
            title.href = info.link;
            title.target = "_blank";
            title.textContent = `${info.rating} | ${info.postTitle}`;
            const loadFull = async () => {
                const icon = RPM.Nodes.createLoadingIcon();
                preview.replaceWith(icon);
                try {
                    const pikabuComment = await loadCommentFromPikabu(info, true, disableMiniProfile);
                    icon.replaceWith(pikabuComment);
                }
                catch (e) {
                    error(e);
                    icon.replaceWith('Не удалось загрузить комментарий :(');
                }
            };
            const preview = createCommentPreview(info, loadFull);
            container.append(title, preview);
            if (autoload) {
                loadFull();
            }
            return container;
        }
        function createCommentPreview(info, loadCallback) {
            const container = document.createElement('div');
            container.classList.add('rpm-comment-preview');
            const buttonToLoad = document.createElement('button');
            buttonToLoad.textContent = 'Загрузить';
            buttonToLoad.addEventListener('click', loadCallback);
            container.append(buttonToLoad);
            return container;
        }
        async function loadCommentsFromGollum(userName) {
            userName = userName.replace('.', '_');
            const request = new HttpRequest(`https://gollum.space/user/${userName}-last`, "GET", "document");
            const response = await request.executeAsync();
            const htmlDoc = response.response;
            const comments = Array.from(htmlDoc.querySelectorAll('.comment-block'));
            return comments.map(elem => {
                const anchor = elem.querySelector('.comment-link');
                const [rating, postTitle] = elem.getAttribute('title').split('|').map(x => x.trim());
                return {
                    id: parseInt(anchor.textContent),
                    postId: parseInt(elem.getAttribute('post')),
                    postTitle,
                    rating,
                    link: anchor.href.replace('pikabu.ru', 'gollum.space')
                };
            });
        }
        function createLastCommentsSection(userName, autoloadGollum = false, autoloadCount = 0, disableMiniProfile = false) {
            const main = document.createElement('div');
            main.classList.add('rpm-last-comments');
            const title = document.createElement('h4');
            title.textContent = 'Последние комментарии';
            const containerOfComments = document.createElement('div');
            containerOfComments.classList.add('rpm-last-comments-container');
            const loadMoreBtn = document.createElement('button');
            loadMoreBtn.textContent = 'Загрузить';
            main.append(title, containerOfComments, loadMoreBtn);
            async function* loadComments() {
                loadMoreBtn.textContent = 'Больше комментариев';
                const loadingIcon = RPM.Nodes.createLoadingIcon();
                containerOfComments.append(loadingIcon);
                loadMoreBtn.style.display = 'none';
                const comments = await loadCommentsFromGollum(userName);
                loadMoreBtn.style.display = '';
                loadingIcon.remove();
                if (comments.length === 0) {
                    containerOfComments.append('Комментарии не найдены.');
                }
                else {
                    let count = 0;
                    for (const comment of comments) {
                        containerOfComments.append(createCommentContainer(comment, count < autoloadCount, disableMiniProfile));
                        if (++count % CHUNK_SIZE == 0) {
                            yield;
                        }
                    }
                }
                loadMoreBtn.remove();
            }
            const loader = loadComments();
            if (autoloadGollum)
                loader.next();
            loadMoreBtn.addEventListener('click', () => {
                loader.next();
            });
            return main;
        }
        LastComments.createLastCommentsSection = createLastCommentsSection;
    })(LastComments || (LastComments = {}));
    Gollum.createLastCommentsSection = LastComments.createLastCommentsSection;
    let TagsSection;
    (function (TagsSection) {
        function createTag(name) {
            const elem = document.createElement('span');
            elem.classList.add('rpm-tags-tag');
            elem.textContent = name;
            return elem;
        }
        function createTagsSection(userName, tagsGetter, autoload = false) {
            const elem = document.createElement('div');
            elem.classList.add('rpm-tags-list');
            elem.innerHTML = '<span class="rpm-clickable">Нажмите, чтобы загрузить.</span>';
            const loadTags = () => {
                const loadingIcon = RPM.Nodes.createLoadingIcon();
                elem.replaceChildren(loadingIcon);
                getUserId(userName).then(userId => {
                    tagsGetter(userId).then((tags) => {
                        const tagElems = tags.map(createTag);
                        if (tagElems.length > 0) {
                            elem.append(...tagElems);
                        }
                        else {
                            elem.append('Теги не найдены.');
                        }
                        loadingIcon.remove();
                    }, e => {
                        elem.textContent = 'Не удалось получить теги.';
                        error(e);
                    });
                }, e => {
                    elem.textContent = 'Не удалось достучаться до Голлума и получить ID пользователя.';
                    error(e);
                });
            };
            if (!autoload) {
                const onClick = () => {
                    loadTags();
                    elem.removeEventListener('click', onClick);
                };
                elem.addEventListener('click', onClick);
            }
            else {
                loadTags();
            }
            return elem;
        }
        TagsSection.createTagsSection = createTagsSection;
        function createTagsContainer(userName, name, getter, autoload = false) {
            const elem = document.createElement('div');
            elem.classList.add('rpm-tags');
            const nameElem = document.createElement('h4');
            nameElem.textContent = name;
            const tagsList = createTagsSection(userName, getter, autoload);
            elem.append(nameElem, tagsList);
            return elem;
        }
        TagsSection.createTagsContainer = createTagsContainer;
    })(TagsSection || (TagsSection = {}));
    Gollum.createTagsSection = TagsSection.createTagsSection;
    Gollum.createTagsContainer = TagsSection.createTagsContainer;
})(Gollum || (Gollum = {}));
//#endregion
class Deferred {
    constructor() {
        this.promise = new Promise((resolve, reject) => {
            this.reject = reject;
            this.resolve = resolve;
        });
    }
}
function waitForElement(selector, timeout = 5000, parent = null) {
    return new Promise((resolve, reject) => {
        const startTime = Date.now();
        parent = parent ?? document;
        const checkExistence = () => {
            const element = parent.querySelector(selector);
            if (element) {
                resolve(element);
            }
            else if (Date.now() - startTime >= timeout) {
                reject(new Error(`Element with selector "${selector}" not found within ${timeout}ms`));
            }
            else {
                setTimeout(checkExistence, 50);
            }
        };
        checkExistence();
    });
}
const STYLES = `.story__footer .story__rating-up {
  margin-right: 5px !important;
}
.prm-minuses {
  padding-left: 7px !important;
  margin: 0px !important;
}
.story__rating-count {
  margin: 7px 0 7px;
}
.rpm-story-summary {
  margin: 14px 0 4px;
}
.rpm-summary-comment {
  margin-right: 8px;
}
.comment__rating-down .comment__rating-count {
  margin-right: 8px;
}
.comment__rating-down {
  padding: 2px 8px;
}
.story__footer .story__rating-rpm-count {
  font-size: 13px;
  color: var(--color-black-700);
  margin: auto 7px auto 7px;
  line-height: 0;
  display: block;
}
.story__footer .rpm-summary,
.comment .rpm-summary {
  margin: auto 9px auto 0px;
  font-weight: 500;
}
.comment__rating-rpm-count {
  padding: 2px 8px;
  flex-shrink: 0;
  margin-left: auto;
}
.rpm-rating-bar {
  width: 5px;
  background: var(--color-danger-800);
  height: 90%;
  position: absolute;
  right: -9.5px;
  top: 5%;
  border-radius: 5px;
}
.rpm-rating-bar-inner {
  background: var(--color-primary-700);    /* width: 99%; */
  border-radius: 5px;
}
.comment__body {
  position: relative;
}
.comment .rpm-rating-bar {
  height: 70px;
  top: 15px;
  left: -10px;
}  /* old mobile interface */
.story__footer-rating .story__rating-minus {
  background-color:var(--color-black-300);
  border-radius:8px;
  overflow:hidden;
  padding:0;
  display:flex;
  align-items:center  ;
}
.story__footer-rating .story__rating-down {
  display:flex;
  align-items:center;
  justify-content:center;
  padding-left:2px  ;
}
.rpm-download-video-button {
  margin-left: 3px;
}
.rpm-block-author {
  overflow:hidden;
  margin-right:24px;
  cursor:pointer;
  display:flex;
  align-items:center;
  padding:0;
  background:0 0
}
.rpm-block-author:hover * {
  fill: var(--color-danger-800);
}
.story__footer-tools-inner .rpm-block-author {
  overflow: visible;
  margin-right: auto;
  margin-left:8px;
  transform: scale(1.3);
}
.rpm-open-settings-button {
  margin-top: 10px;
  text-align: center;
  width: 100%;
  font-size: 0.9em;
}
.rpm-video-list {
  text-align: center;
}
.rpm-video-list a {
  margin: 0 5px;
}
.rpm-story-icon {
  width: 24px;
  height: 24px;
  padding: 0 4px;
  margin-right: 6px;
  vertical-align: text-top;
  border-radius: 3px;
}
.rpm-story-icon svg {
  margin-top: auto;
  padding: 2px 0px;
  width: 16px;
  height: 16px;
  margin: 0;
  transition: all ease 300ms;
}
.rpm-story-icon:hover svg {
  width: 18px;
  height: 18px;
}
@media only screen and (max-width: 768px)  {
  #prm {
    left: 2.5% !important;
    width: 100% !important;
  }
}  /* Notifications */
@keyframes rpm-notification-intro {
  from {
    translate: -100%;
  }
  to {
    translate: 0%;
  }
}
@keyframes rpm-notification-outro {
  from {
    translate: 0 0%;
    opacity: 1.0;
  }
  to {
    translate: 0 -200%;
    opacity: 0;
  }
}
.rpm-notification {
  position: fixed;
  bottom: 15px;
  left: 15px;
  z-index: 9999;
  overflow: hidden;
  max-width: 250px;
  background: var(--color-black-100);
  border-radius: 10px;
  border: 1px solid var(--color-black-440);
  pointer-events: none;
}
.rpm-notification * {
  padding: 5px 15px;
}
.rpm-notification-header {
  background: var(--color-primary-800);
  color: white;
  font-weight: bolder;
}
.rpm-user-rating {
  font-size: 1.05em;
  padding: 0.3em 0.6em;
  border-radius: 1rem;
  display: inline-block;
  background-color: var(--color-black-alpha-005);
  user-select: none;
  white-space: nowrap;
  line-height: 1em;
}
.story__main .rpm-user-rating,
.rpm-placeholder .rpm-user-rating {
  margin-right: 10px;
}
.comment__header .rpm-user-rating {
  margin-left:auto;
  right: 0;
}
.rpm-user-rating + .comment__right {
  margin-left: unset;
}
.rpm-user-rating span {
  display: inline-block;
  min-width: 1.5ch;
  text-align: center;
}
.rpm-user-rating .rpm-rating {
  margin: 0 1ch;
}
.rpm-user-rating .rpm-pluses {
  color: var(--color-primary-700);
  cursor: pointer;
}
.rpm-user-rating .rpm-minuses {
  color: var(--color-danger-900);
  cursor: pointer;
}
.rpm-user-rating[rpm-own-vote="1"] {
  background-color: var(--color-primary-200)
}
.rpm-user-rating[rpm-own-vote="-1"] {
  background-color: var(--color-danger-200)
}
.rpm-placeholder {
  background-color: var(--color-bright-800);
  border: 1px solid var(--color-black-430);
  border-radius: 15px;
  width: max-content;
  height: max-content;
  text-align: center;
  margin: 10px auto 0;
  padding: 5px 20px;
  position: relative;
  max-width: 95%;
}
.rpm-placeholder .rpm-user-info-container {
  width: max-content;
  padding: 0.5em;
  border-radius: 10px;
  margin: 0 auto;
}
.rpm-placeholder .collapse-button {
  position: absolute;
  top: 0;
  left: -70px;
  translate: 0 -70%;
}
.mv .rpm-placeholder {
  font-size: 0.825em;
}
.mv .rpm-placeholder .collapse-button {
  display: inline flow-root list-item;
  position: initial;
  margin: 0 auto;
  translate: 0 0;
  transform: scale(0.75);
}
.rpm-placeholder:has(.collapse-button_active) + article {
  display: none;
}
.rpm-loading {
  display: block;
  width: 4px;
  height: 4px;
  border: 7px solid var(--color-primary-400);
  border-top: 7px solid var(--color-primary-700);
  border-radius: 50%;
  animation: spin 1.5s linear infinite;
  margin: auto;
}
.rpm-feedback-window {
  position: fixed;
  display: flex;
  left: 70px;
  top: 100px;
  width: 450px;
  height: 50%;
  border: 1px solid var(--color-black-430);
  border-radius: 8px;
  background-color: var(--color-bright-800);
  padding: 0;
  flex-direction: column;
}
.rpm-feedback-window iframe {
  width: 100%;
}
.comment__more:has(+.rpm-unroll-all),
.rpm-unroll-all {
  --gap: 10px;
  width: calc(50% - var(--gap) / 2);
  margin-right: var(--gap);
  text-align: center;
}
.rpm-unroll-all {
  display: none;
  margin: auto 0;
  background-color: var(--color-primary-700);
  color: var(--color-bright-900);
}
.comment__more + .rpm-unroll-all {
  display: inline-block;
}
#prm {
  background-color: var(--color-black-440);
  border-radius: 15px;
  border: none !important;
}
#prm .field_label {
  font-size: 14px;
  font-weight: bold;
}
#prm .radio_label {
  font-size: 14px;
}
#prm .config_var {
  margin: 0 0 4px;
}
#prm .section_header {
  color: #FFF;
  font-size: 4em;
  margin: 0;
}
#prm .section_desc {
  color: var(--color-black-700);
  font-size: 9pt;
  margin: 0 0 6px;
}
#prm_wrapper {
  --gap: 10px;
  box-sizing: border-box;
  padding: 15px;
  margin: 0;
  display: flex;
  flex-flow: row wrap;
  align-content: space-evenly;
  gap: var(--gap);
}
#prm_header {
  height: max-content;
  flex: 0 0 100%;
}
#prm_header p {
  font-size: x-large;
}
#prm_header a {
  font-size: large;
  text-decoration: underline;
  margin: 0 10px;
}
.section_header_holder {
  margin: 8px auto 0;
  display: block;
  padding: 10px;
  flex-basis: 100%;
  overflow: hidden;
  border-radius: 10px;
  background-color: var(--color-black-430);
}
#prm_section_1,
#prm_section_2,
#prm_section_5,
#prm_section_6 {
  flex: 1;
}
#prm_buttons_holder {
  flex-basis: 100%;
  display: flex;
  justify-content: flex-end;
  gap: 5px;
  flex-flow: row wrap;
}
#prm_buttons_holder .reset_holder {
  flex-basis: 100%;
  text-align: right;
}
#prm_wrapper .section_header.center {
  font-size: 17pt;
  display: block;
  width: calc(100% + 20px);
  text-align: center;
  margin: -10px -10px 0;
  background-color: var(--color-primary-700);
  padding: 4px 0;
}
#prm_wrapper .config_var {
  width: 100%;
  margin-top: 5px;
  font-size: medium;
}
#prm_wrapper .config_var input,
#prm_wrapper .config_var select {
  margin-right: 10px;
}
#prm_wrapper .config_var input[type=text] {
  background-color: var(--color-black-440);
  padding: 2px;
  border: solid 1px black;
  border-radius: 7px;
  padding: 4px;
  width: 30%;
}
#prm_wrapper input[type=button],
#prm_wrapper button {
  background-color: var(--color-black-440);
  color: var(--color-black-700);
  transition: background-color 200ms ease-out, filter 200ms, color 200ms;
  padding: 0px 20px;
  vertical-align: middle;
  box-sizing: border-box;
  border-radius: 8px;
  cursor: pointer;
  font-size: 14px;
  line-height: 32px;
  font-weight: 500;
  max-width: 100%;
  text-wrap: balance;
}
#prm_buttons_holder button {
  background-color: var(--color-black-300);
}
#prm_wrapper input[type=button]:hover {
  background-color: var(--color-black-500);
}
#prm_wrapper input[type=checkbox] {
  margin-right: 10px;
  width: 1.25em;
  height: 1.25em;
}
#prm_wrapper select {
  background-color: var(--color-black-440);
  color: var(--color-black-700);
  border: none;
  padding: 5px 10px;
  border-radius: 5px;
  outline: none;
}
#prm_section_3 .config_var {
  display: inline-block;
  width: calc(100%/7);
  min-width: max-content;
  padding: 0 5px;
  text-align: center;
}
#prm_section_5 .config_var input[type=text] {
  width: 30em;
  max-width: 100%;
}
@media only screen and (max-width: 768px)  {
  #prm {
    width: unset !important;
    height: unset !important;
    left: 10px !important;
    right: 10px !important;
    top: 10px !important;
    bottom: 10px !important;
  }
  #prm_section_1,
  #prm_section_2,
  #prm_section_5,
  #prm_section_6  {
    flex: unset;
  }
  #prm_section_5 {
    display: none;
  }
  #prm .config_var {
    margin-bottom: 20px;
  }
  #prm .config_var:last-child {
    margin-bottom: unset;
  }
}  /* Notifications */
@keyframes rpm-notification-intro {
  from {
    translate: -100%;
  }
  to {
    translate: 0%;
  }
}  /* Themes */
.rpm-theme-picker {
  flex: 1 0 0px;
}
.rpm-theme-picker:after {
  content: attr(data-name)
}
.theme-picker__buttons {
  flex-wrap: wrap;
  flex-direction: row;
  gap: 10px;
  justify-content: center;
}
.theme-picker__button {
  min-width: fit-content;
  max-width: max-content;
}
.theme-picker__button[data-type="default"] {
  max-width: unset;
  width: 35px;
}  /* SUNSET GLOW */
.rpm-theme-picker[data-type="sunset-glow"] {
  background: linear-gradient(135deg, #6b1d52, #b02e78, #f77fba);
  border: 2px solid #8e2465;
  border-radius: 8px;
  transition: transform 0.2s ease;
}
.rpm-theme-picker[data-type="sunset-glow"]:hover {
  transform: scale(1.1);
}
html[data-theme="sunset-glow"] {
  --color-primary-900: #6b1d52;
  --color-primary-800: #8e2465;
  --color-primary-700: #b02e78;
  --color-primary-500: #d14791;
  --color-primary-400: #f77fba;
  --color-primary-200: #fbc8e4;
  --color-primary-100: #fdeaf4;
}  /* OCEAN BREEZE */
.rpm-theme-picker[data-type="ocean-breeze"] {
  background: linear-gradient(135deg, #012a4a, #014f86, #61a5c2);
  border: 2px solid #013a63;
  border-radius: 50%;
  transition: transform 0.2s ease;
}
.rpm-theme-picker[data-type="ocean-breeze"]:hover {
  transform: scale(1.1);
}
html[data-theme="ocean-breeze"] {
  --color-primary-900: #012a4a;
  --color-primary-800: #013a63;
  --color-primary-700: #014f86;
  --color-primary-500: #2a6f97;
  --color-primary-400: #61a5c2;
  --color-primary-200: #a9d6e5;
  --color-primary-100: #d9f1f6;
}  /* FOREST */
.rpm-theme-picker[data-type="forest-whisper"] {
  background: linear-gradient(135deg, #143601, #275d03, #63b530);
  border: 2px solid #1c4a02;
  border-radius: 8px;
  transition: transform 0.2s ease;
}
.rpm-theme-picker[data-type="forest-whisper"]:hover {
  transform: scale(1.1);
}
html[data-theme="forest-whisper"] {
  --color-primary-900:rgb(27, 68, 3);
  --color-primary-800:rgb(32, 80, 4);
  --color-primary-700:rgb(72, 170, 7);
  --color-primary-500: #398d05;
  --color-primary-400: #63b530;
  --color-primary-200: #a9e8a4;
  --color-primary-100:rgb(227, 248, 227);
}  /* LAVENDER DREAMS */
.rpm-theme-picker[data-type="lavender-dreams"] {
  background: linear-gradient(135deg, #4c1d6f, #70308e, #cfa5e0);
  border: 2px solid #5e267e;
  border-radius: 50%;
  transition: transform 0.2s ease;
}
.rpm-theme-picker[data-type="lavender-dreams"]:hover {
  transform: scale(1.1);
}
html[data-theme="lavender-dreams"] {
  --color-primary-900:rgb(141, 92, 179);
  --color-primary-800:rgb(117, 57, 151);
  --color-primary-700:rgb(132, 73, 160);
  --color-primary-500: #9b5cb2;
  --color-primary-400: #cfa5e0;
  --color-primary-200:rgb(236, 223, 245);
  --color-primary-100: #f7ecfc;
}  /* FIRE EMBER */
.rpm-theme-picker[data-type="fire-ember"] {
  background: linear-gradient(135deg, #7f1d1d, #b91c1c, #f87171);
  border: 2px solid #991b1b;
  border-radius: 8px;
  transition: transform 0.2s ease;
}
.rpm-theme-picker[data-type="fire-ember"]:hover {
  transform: scale(1.1);
}
html[data-theme="fire-ember"] {
  --color-primary-900:rgb(145, 42, 42);
  --color-primary-800:rgb(172, 42, 42);
  --color-primary-700:rgb(211, 32, 32);
  --color-primary-500:rgb(241, 62, 62);
  --color-primary-400:rgb(252, 138, 138);
  --color-primary-200:rgb(253, 185, 185);
  --color-primary-100: #fee2e2;
}  /* POLKA (mosaic before) */
html[data-theme="mosaic"] .app {
  background-image:  radial-gradient(rgba(68, 77, 247, 0.5) 1.3px, transparent 1.3px), radial-gradient(rgba(68, 77, 247, 0.5) 1.3px, transparent 1.3px);
  background-repeat: repeat;
  background-size: 66px 66px;
  background-position: 0 0, 33px 33px;
}
.rpm-theme-picker[data-type="mosaic"] {
  background-image:  radial-gradient(rgba(68, 77, 247, 0.5) 1.3px, transparent 1.3px), radial-gradient(rgba(68, 77, 247, 0.5) 1.3px, rgba(0, 0, 0, 0.1) 1.3px);
  background-repeat: repeat;
  background-size: 10px 10px;
  background-position: 0 0, 5px 5px;
  border-radius: 8px;
  box-shadow: 0 4px 8px rgba(0, 0, 0, 0.2);
  transition: transform 0.2s ease, box-shadow 0.2s ease;
}
.rpm-theme-picker[data-type="mosaic"]:after {
  color: var(--color-black-800);
}
.rpm-theme-picker[data-type="mosaic"]:hover {
  transform: scale(1.1);
  box-shadow: 0 6px 12px rgba(0, 0, 0, 0.3);
}  /* WHITE TEXT */
html[data-theme="lavender-dreams"] .achievements-progress__bar,
html[data-theme="fire-ember"] .achievements-progress__bar,
html[data-theme="forest-whisper"] .achievements-progress__bar,
html[data-theme="ocean-breeze"] .achievements-progress__bar,
html[data-theme="sunset-glow"] .achievements-progress__bar {
  color: white;
}
/* MINI PROFILE */.rpm-mini-profile-note {
  resize: none;
}
.rpm-mini-profile-note-display {
  display: block;
  min-height: 10px;
}
.rpm-powered {
  text-align: right;
  display: inline-block;
  width: 100%;
  font-size: 0.8em;
  opacity: 80%;
}
.rpm-gollum-stats {
  width: 100%;
  padding: 15px;
}
.rpm-gollum-stats > * {
  margin-bottom: 10px;
}
.rpm-gollum-stats > *:last-child {
  margin-bottom: unset;
}
.rpm-tags p {
  width: 100%;
  text-align: center;
  font-weight: bold;
  margin-bottom: 5px;
}
.rpm-tags-list {
  justify-content: space-evenly;
  display: flex;
  flex-wrap: wrap;
  gap: 4px;
  font-size: 0.8rem;
}
.rpm-clickable {
  cursor: pointer;
}
.rpm-tags-tag {
  background: var(--color-primary-200);
  padding: 0.05em 0.6em;
  border-radius: 10px;
}
.mini-profile[rpm-affected] {
  width: 400px;
}
.rpm-last-comments-container {
  max-height: 150px;
  overflow-y: scroll;
}
.rpm-comment-no-js .comment__tools,
.rpm-comment-no-js .comment__controls,
.rpm-comment-no-js .comment-hidden-group {
  display: none;
}
.rpm-last-comments > p {
  font-weight: bold;
  text-align: center;
}
.rpm-last-comments > button {
  width: 100%;
  text-align: center;
}
.rpm-last-comments-container {
  max-height: 400px;
  height: max-content;
  resize: vertical;
  border-radius: 15px;
  margin-bottom: 10px;
}
section .rpm-last-comments-container {
  max-height: unset;
}
.rpm-comment-container {
  border-radius: 15px;
  background: var(--color-black-430);
  margin-bottom: 7px;
  padding: 5px 1em;
  overflow: hidden;
}
.rpm-comment-container > a {
  display: block;
  width: 100%;
  text-align: center;
  margin-bottom: 5px;
}
.rpm-comment-container .comment__tools > *:not([data-role="link"]) {
  display: none;
}
.rpm-comment-preview {
  text-align: center;
}
.rpm-highlight-comment > .comment__body  {
  border: 3px dotted var(--color-primary-400);
  border-radius: 10px;
  padding: 5px;
}
.rpm-comment-container > .comment,
.rpm-comment-container > .comments {
  background: var(--color-bright-800);
  border-radius: 15px;
  padding: 0 5px;
  padding-top: 5px;
}
`;
let enableFilters = null;
const isStoryPage = window.location.href.includes("/story/");
const currentStoryId = parseInt(["0", ...window.location.href.split('_')].pop());
function makeEval(args, str, defaultFunc) {
    try {
        return new Function(args, "return " + str);
    }
    catch {
        return defaultFunc;
    }
}
class Formats {
}
var SettingEnums;
(function (SettingEnums) {
    let UnrollComments;
    (function (UnrollComments) {
        UnrollComments["NONE"] = "\u0421\u0442\u0430\u043D\u0434\u0430\u0440\u0442\u043D\u0430\u044F \u043F\u0438\u043A\u0430\u0431\u0443\u0448\u043D\u0430\u044F \u043A\u043D\u043E\u043F\u043A\u0430";
        UnrollComments["UNROLL_ALL_BUTTON"] = "\u0414\u043E\u043F\u043E\u043B\u043D\u0438\u0442\u0435\u043B\u044C\u043D\u0430\u044F \u043A\u043D\u043E\u043F\u043A\u0430 \"\u0420\u0430\u0441\u043A\u0440\u044B\u0442\u044C \u0432\u0441\u0451\"";
        UnrollComments["AUTO_UNROLL"] = "\u0410\u0432\u0442\u043E\u043C\u0430\u0442\u0438\u0447\u0435\u0441\u043A\u0430\u044F \u0440\u0430\u0441\u043A\u0440\u0443\u0442\u043A\u0430 \u0432\u0441\u0435\u0445 \u043A\u043E\u043C\u043C\u0435\u043D\u0442\u0430\u0440\u0438\u0435\u0432";
    })(UnrollComments = SettingEnums.UnrollComments || (SettingEnums.UnrollComments = {}));
})(SettingEnums || (SettingEnums = {}));
const formats = new Formats();
let isConfigInit = false;
let frame = document.createElement('div');
document.body.appendChild(frame);
async function handleOldConfigFields() {
    const config = JSON.parse(await GM.getValue('prm', '{}'));
    let changed = false;
    if ('unrollCommentariesAutomatically' in config) {
        if (config.unrollCommentariesAutomatically) {
            config.unrollCommentaries = SettingEnums.UnrollComments.AUTO_UNROLL;
        }
        delete config.unrollCommentariesAutomatically;
    }
    if (changed)
        await GM.setValue('prm', JSON.stringify(config));
}
async function handleConfig() {
    await handleOldConfigFields();
    GM_config.init({
        id: "prm",
        title: (() => {
            const div = document.createElement('div');
            const p1 = document.createElement('p');
            p1.textContent = "Return Pikabu minus";
            const links = [];
            function addLink(text, url) {
                const link = document.createElement('a');
                link.href = url;
                link.textContent = text;
                links.push(link);
            }
            addLink("Телеграм", "https://t.me/return_pikabu");
            addLink("GitHub", "https://github.com/PyXiion/Pikabu-Return-Minus");
            div.append(p1, ...links);
            return div;
        })(),
        fields: {
            // ОБЩИЕ НАСТРОЙКИ
            summary: {
                section: [
                    "Общие настройки",
                ],
                type: "checkbox",
                default: true,
                label: "Отображение суммарного рейтинга у постов и комментариев.",
            },
            minRatesCountToShowRatingBar: {
                type: "int",
                default: 3,
                label: "Минимальное количество оценок у поста или комментария для отображения соотношения плюсов и минусов. " +
                    "Установите на 0, чтобы всегда показывать.",
            },
            noLinkTracking: {
                type: "checkbox",
                default: true,
                label: "Удалять трекеры с ссылок."
            },
            // НАСТРОЙКИ ПОСТОВ
            minStoryRating: {
                section: [
                    "Настройки постов",
                ],
                type: "int",
                default: 100,
                label: "Посты с рейтингом ниже указанного будут удаляться из ленты. Вы сможете увидеть удалённые посты в списке просмотренных.",
            },
            ratingBar: {
                type: "checkbox",
                default: true,
                label: "Отображение соотношения плюсов и минусов у постов. При отсутствии оценок у поста будет показано соотношение 1:1.",
            },
            showBlockAuthorForeverButton: {
                type: "checkbox",
                default: true,
                label: "Отображение кнопки, которая блокирует автора поста навсегда. То есть добавляет в игнор-лист. " +
                    "Вы должны быть авторизированы на сайте, иначе кнопка работать не будет.",
            },
            blockPaidAuthors: {
                type: "checkbox",
                default: true,
                label: "Удаляет из ленты посты от проплаченных авторов (которые с подпиской Пикабу+).",
            },
            videoDownloadButtons: {
                type: "checkbox",
                default: true,
                label: "Добавляет ко всем видео в постах ссылки на источники, если их возможно найти.",
            },
            socialLinks: {
                type: "checkbox",
                default: false,
                label: "Добавляет в начале заголовка поста значки Телеграма, ВК, Тиктока, если в посте есть соответствующие ссылки.",
            },
            // НАСТРОЙКИ КОММЕНТАРИЕВ
            ratingBarComments: {
                section: [
                    "Настройки комментариев",
                ],
                type: "checkbox",
                default: true,
                label: "Отображение соотношения плюсов и минусов у комментариев.",
            },
            allCommentsLoadedNotification: {
                type: "checkbox",
                default: false,
                label: "Показывать уведомление о загрузке всех комментариев под постом."
            },
            commentVideoDownloadButtons: {
                type: "checkbox",
                default: true,
                label: "Добавляет ко всем видео в комментариях ссылки на источники, если их возможно найти."
            },
            unrollCommentariesAutomatically: {
                type: "hidden",
                default: undefined
            },
            unrollCommentaries: {
                type: 'select',
                label: 'Раскрутка комментариев.',
                options: [
                    SettingEnums.UnrollComments.NONE,
                    SettingEnums.UnrollComments.UNROLL_ALL_BUTTON,
                    SettingEnums.UnrollComments.AUTO_UNROLL
                ],
                default: SettingEnums.UnrollComments.NONE
            },
            // ВКЛАДКИ ПИКАБ
            hotTab: {
                section: [
                    "Вкладки Пикабу",
                    "Включает/выключает вкладки сверху. Работает только на ПК"
                ],
                type: "checkbox",
                default: true,
                label: "Горячее",
            },
            bestTab: {
                type: "checkbox",
                default: true,
                label: "Лучшее",
            },
            newTab: {
                type: "checkbox",
                default: true,
                label: "Свежее",
            },
            subsTab: {
                type: "checkbox",
                default: true,
                label: "Подписки",
            },
            communitiesTab: {
                type: "checkbox",
                default: true,
                label: "Сообщества",
            },
            blogsTab: {
                type: "checkbox",
                default: true,
                label: "Блоги",
            },
            expertsTab: {
                type: "checkbox",
                default: true,
                label: "Эксперты",
            },
            // НАСТРОЙКИ RPM
            rpmEnabled: {
                section: ["Настройки RPM", "Дополнительные функции скрипта. Используется сервер rpm.pyxiion.ru"],
                type: "checkbox",
                default: true,
                label: "Включить для постов."
            },
            rpmMinStoryRating: {
                type: "int",
                default: 0,
                label: "Минимальный рейтинг автора в системе RPM. Если рейтинг автора меньше его значения, то его посты будут удалены из ленты."
            },
            rpmIgnoreDownvoted: {
                type: "checkbox",
                default: true,
                label: "Скрытие постов с вашим минусом в системе RPM. Типа игнор-листа."
            },
            rpmComments: {
                type: "checkbox",
                default: true,
                label: "Включить для комментариев."
            },
            registerRpm: {
                type: "button",
                label: "Зарегистрироваться в системе RPM. После нажатия страница перезагрузится.",
                async click() {
                    const uuid = GM_config.get('uuid');
                    if (uuid === null || uuid === undefined || uuid === '') {
                        GM_config.set('uuid', await RPM.Service.register());
                        GM_config.save();
                        sendNotification('Успешно', 'Вы успешно зарегистрировались. Или нет. Проверки успешности не существует.');
                        await sleep(300);
                        window.location.reload();
                    }
                    else {
                        sendNotification('Вы уже зарегистрированы', 'Вы не можете зарегистрироватся ещё раз.');
                    }
                }
            },
            // НАСТРОЙКИ МИНИ-ПРОФИЛЕЙ
            miniProfileEditableNote: {
                section: ["Настройки мини-профилей", "Дополнительные функции в мини-профилей пользователей, которые появляются при наведении на их ник."],
                type: "checkbox",
                default: true,
                label: "Редактирование заметки."
            },
            miniProfileStoryTags: {
                type: "checkbox",
                default: true,
                label: "Основные теги постов пользователя."
            },
            miniProfileСommentTags: {
                type: "checkbox",
                default: true,
                label: "Основные теги постов, где пользователь оставляет комментарии."
            },
            miniProfileСomments: {
                type: "checkbox",
                default: false,
                label: "Последние комментарии пользователя."
            },
            miniProfileAutoloadTags: {
                type: "checkbox",
                default: false,
                label: "Автозагрузка тегов."
            },
            miniProfileAutoloadComments: {
                type: "checkbox",
                default: false,
                label: "Автозагрузка предпросмотра последних комментариев."
            },
            miniProfileAutoloadPikabuCommentCount: {
                type: "number",
                default: 1,
                label: "Сколько комментариев загрузить без нажатия на кнопку Загрузить."
            },
            // СТРАНИЦА ПОЛЬЗОВАТЕЛЯ
            profileStoryTags: {
                section: ["Настройки профилей пользователей", "То же самое, но только на странице профиля."],
                type: "checkbox",
                default: true,
                label: "Основные теги постов пользователя."
            },
            profileСommentTags: {
                type: "checkbox",
                default: true,
                label: "Основные теги постов, где пользователь оставляет комментарии."
            },
            profileСomments: {
                type: "checkbox",
                default: true,
                label: "Последние комментарии пользователя."
            },
            profileAutoloadComments: {
                type: "checkbox",
                default: true,
                label: "Автозагрузка предпросмотра последних комментариев."
            },
            profileAutoloadPikabuCommentCount: {
                type: "number",
                default: 3,
                label: "Сколько комментариев загрузить без нажатия на кнопку Загрузить."
            },
            // БОЛЕЕ СЛОЖНЫЕ НАСТРОЙКИ
            filteringPageRegex: {
                section: ["Продвинутые настройки"],
                type: "text",
                label: "Страницы, на которых работает фильтрация по рейтингу (регулярное выражение).",
                default: "^https?:\\/\\/pikabu.ru\\/(|best|companies|browse|disputed|most-saved)$",
            },
            minusesPattern: {
                type: "text",
                default: "story.minuses",
                label: "Шаблон отображения минусов у постов (JS). Пример: `story.minuses * 5000`. story: {id, rating, pluses, minuses}. Внутри может выполняться любой код, поэтому используйте с осторожностью.\n" +
                    "Шаблоны гарантированно работают только на Tampermonkey.",
            },
            minusesCommentPattern: {
                type: "text",
                default: "comment.minuses",
                label: "Шаблон отображения минусов у комментариев (JS). Пример: `comment.minuses * 5000`. comment: {id, rating, pluses, minuses}.",
            },
            ownCommentPattern: {
                type: "text",
                default: "comment.pluses == 0 && comment.minuses == 0 ? 0 : comment.pluses == comment.minuses ? `+${comment.pluses} / -${comment.minuses}` : comment.pluses == 0 ? `-${comment.minuses}` : comment.minuses == 0 ? `+${comment.pluses}` : `+${comment.pluses} / ${comment.rating} / -${comment.minuses}`",
                label: "Шаблон отображения рейтинга у ВАШИХ комментариев (JS). Пример: `comment.minuses * 5000`. comment: {id, rating, pluses, minuses}.",
            },
            analytics: {
                type: "checkbox",
                label: "Отправка всякой информации на сервера RPM, если включено. Пока что никакой информации не собирается, но вы можете выключить это заранее, если таковая появится.",
                default: true
            },
            debug: {
                type: "checkbox",
                label: "Включить дополнительные логи в консоли. Для разработки и отладки.",
                default: false
            },
            uuid: {
                type: "hidden",
                // label:
                //   "Ваш уникальный UUID в системе рейтинга RPM. Позволяет вам оценивать профили других пользователей.",
                default: ''
            },
        },
        events: {
            init() {
                isConfigInit = true;
                formats.formatStoryMinuses = makeEval("story", this.get('minusesPattern'), (x) => x.minuses);
                formats.formatCommentMinuses = makeEval("comment", this.get('minusesCommentPattern'), (x) => x.minuses);
                formats.formatOwnComment = makeEval("comment", this.get('ownCommentPattern'), (x) => (x.pluses == 0 && x.minuses == 0) ? 0 : `${x.pluses}/${x.minuses}`);
                enableFilters = new RegExp(this.get('filteringPageRegex')).test(window.location.href);
                this.css.basic = [];
                if (this.get('unrollCommentariesAutomatically')) {
                    this.set('unrollCommentatries', SettingEnums.UnrollComments.AUTO_UNROLL);
                    this.set('unrollCommentariesAutomatically', null);
                    this.save();
                }
            }
        },
        frame: frame
    });
}
handleConfig();
const logPrefix = "[RPM]";
function info(...args) {
    if (GM_config === undefined || GM_config.get === undefined)
        return;
    if (GM_config.get('debug')) {
        console.info(logPrefix, ...args);
    }
}
function warn(...args) {
    if (GM_config === undefined || GM_config.get === undefined)
        return;
    if (GM_config.get('debug')) {
        console.warn(logPrefix, ...args);
    }
}
function error(...args) {
    if (GM_config === undefined || GM_config.get === undefined)
        return;
    if (GM_config.get('debug')) {
        console.error(logPrefix, ...args);
    }
}
const waitConfig = () => new Promise((resolve) => {
    let isInit = () => setTimeout(() => (isConfigInit ? resolve() : isInit()), 1);
    isInit();
});
const supportMenuCommands = GM.registerMenuCommand !== undefined;
GM.registerMenuCommand("Открыть настройки", () => {
    info("Открыты настройки.");
    GM_config.open();
});
function addCss(css) {
    const styleSheet = document.createElement("style");
    styleSheet.innerText = css;
    // is added to the end of the body because it must override some of the original styles
    document.body.appendChild(styleSheet);
    info("Добавлен CSS");
}
class CommentData {
    constructor(data) {
        this.id = data.id;
        this.rating = data.rating;
        this.pluses = data.pluses;
        this.minuses = data.minuses;
        this.videos = data.videos;
    }
}
// Variables
const cachedComments = new Map();
let oldInterface = null;
const deferredComments = new Map();
const cachedPostVideos = new Map();
const blockIconTemplate = (function () {
    const div = document.createElement("div");
    div.innerHTML = `<svg xmlns="http://www.w3.org/2000/svg" class="icon icon--ui__save"><use xlink:href="#icon--ui__ban"></use></svg>`;
    return div.firstChild;
})();
// UI functions
function sleep(ms) {
    return new Promise(resolve => setTimeout(resolve, ms));
}
async function sendNotification(title, description, timeout = 2000) {
    function construct() {
        const notification = document.createElement('div');
        notification.classList.add('rpm-notification');
        const header = document.createElement('div');
        header.classList.add('rpm-notification-header');
        header.textContent = title;
        const content = document.createElement('div');
        content.classList.add('rpm-notification-content');
        content.textContent = description;
        notification.append(header, content);
        return notification;
    }
    const notification = construct();
    document.body.append(notification);
    const animationTime = 600.0;
    notification.style.animationDuration = `${animationTime / 1000.0}s`;
    notification.style.animationTimingFunction = "cubic-bezier(.18,.89,.32,1.28)";
    // Intro
    notification.style.animationName = "rpm-notification-intro";
    await sleep(animationTime);
    await sleep(timeout);
    // Outro
    notification.style.animationTimingFunction = "linear";
    notification.style.animationName = "rpm-notification-outro";
    notification.style.opacity = "0";
    await sleep(animationTime);
    // Remove
    notification.remove();
}
// Functions
async function blockAuthorForever(button, authorId) {
    button.disabled = true;
    // const fetch = unsafeWindow.fetch;
    try {
        await fetch(`https://pikabu.ru/ajax/ignore_actions.php?authors=${authorId}&story_id=0&period=forever&action=add_rule`, {
            method: "POST",
        });
        button.remove();
        info("Автор с ID", authorId, "заблокирован");
    }
    catch {
        button.disabled = false;
        error("Не получилось заблокировать автора с ID", authorId, ", возможно отсутствует Интернет-соединение");
    }
}
function addBlockButton(story) {
    const saveButton = story.querySelector(".story__save");
    if (saveButton === null) {
        warn("Failed to add a block button to", story);
        return;
    }
    const button = document.createElement("button");
    button.classList.add("rpm-block-author", "hint");
    button.setAttribute("aria-label", "Заблокировать автора навсегда");
    button.appendChild(blockIconTemplate.cloneNode(true));
    const authorId = parseInt(story.getAttribute("data-author-id"));
    button.addEventListener("click", () => {
        blockAuthorForever(button, authorId);
    });
    saveButton.parentElement.insertBefore(button, saveButton);
}
function processComment(comment) {
    const commentElem = document.getElementById(`comment_${comment.id}`);
    if (commentElem === null) {
        if (comment instanceof Pikabu.Comment) {
            cachedComments[comment.id] = new CommentData(comment);
            info('Закэшировал комментарий', comment.id);
        }
        return;
    }
    const userElem = commentElem.querySelector(".comment__user");
    const ratingDown = commentElem.querySelector(".comment__rating-down");
    if (!userElem || !ratingDown) {
        // Defer comment
        info('У комментария', comment.id, ' нет юзера или кнопок рейтинга, кэширую его');
        // setTimeout(() => processComment(comment), 400);
        cachedComments[comment.id] = (comment instanceof Pikabu.Comment) ? new CommentData(comment) : comment;
        return;
    }
    if (userElem.hasAttribute("data-own") &&
        userElem.getAttribute("data-own") === "true") {
        const textRatingElem = commentElem.querySelector(".comment__rating-count");
        textRatingElem.innerText = formats.formatOwnComment(comment);
        info('Обработал "свой" комментарий', comment.id);
        return;
    }
    const minusesText = document.createElement("div");
    minusesText.classList.add("comment__rating-count");
    ratingDown.prepend(minusesText);
    minusesText.textContent = formats.formatCommentMinuses(comment);
    if (GM_config.get('summary')) {
        const summary = document.createElement("div");
        summary.classList.add("comment__rating-count", "rpm-summary");
        summary.textContent = comment.rating.toString();
        ratingDown.parentElement.insertBefore(summary, ratingDown);
    }
    const totalRates = comment.pluses + comment.minuses;
    if (GM_config.get('ratingBarComments') &&
        totalRates >= GM_config.get('minRatesCountToShowRatingBar')) {
        let ratio = 0.5;
        if (totalRates > 0)
            ratio = comment.pluses / totalRates;
        addRatingBar(commentElem, ratio);
    }
    // Comment videos
    if (GM_config.get('commentVideoDownloadButtons')) {
        const videoElements = commentElem.querySelectorAll(':scope > .comment__body .comment-external-video');
        const videoCount = Math.min(videoElements.length, comment.videos.length);
        for (let i = 0; i < videoCount; ++i) {
            const elem = videoElements[i];
            const url = comment.videos[i];
            const linkElem = document.createElement('a');
            linkElem.classList.add('rpm-download-video-button');
            linkElem.href = url;
            linkElem.text = 'Источник';
            linkElem.target = '_blank';
            elem.parentNode.insertBefore(linkElem, elem.nextSibling);
        }
    }
    info('Обработал комметарий', comment.id);
}
async function processStoryComments(storyId, storyData, page) {
    if (!isStoryPage || storyId != currentStoryId) {
        return;
    }
    for (const comment of storyData.comments) {
        processComment(comment);
    }
    if (storyData.hasMoreComments) {
        storyData = await Pikabu.DataService.fetchStory(storyId, page + 1);
        await processStoryComments(storyId, storyData, page + 1);
    }
    else if (GM_config.get('allCommentsLoadedNotification')) {
        sendNotification('Return Pikabu Minus', 'Все рейтинги комментариев загружены!');
    }
}
function addRatingBar(story, ratio) {
    const block = story.querySelector(".story__rating-block, .comment__body, .story__emotions");
    if (block !== null) {
        const bar = document.createElement("div");
        const inner = document.createElement("div");
        bar.append(inner);
        bar.classList.add("rpm-rating-bar");
        inner.classList.add("rpm-rating-bar-inner");
        inner.style.height = (ratio * 100).toFixed(1) + "%";
        block.prepend(bar);
    }
    else {
        // TODO mobile
    }
}
const linkTypes = [
    // Telegram
    {
        domains: [
            "t.me"
        ],
        iconHtml: `<svg xmlns="http://www.w3.org/2000/svg" class="rpm-story-icon icon icon--social__telegram"><use xlink:href="#icon--social__telegram"></use></svg>`,
        style: "fill: #24A1DE;",
    },
    // VK
    {
        domains: [
            "vk.com"
        ],
        iconHtml: `<svg xmlns="http://www.w3.org/2000/svg" class="icon icon--social__vk"><use xlink:href="#icon--social__vk"></use></svg>`,
        style: "fill: black;",
    },
    // TIKTOK
    {
        domains: [
            "tiktok.com"
        ],
        iconHtml: `<svg class="icon" fill="#000000" viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg" xml:space="preserve"><path d="M19.589 6.686a4.793 4.793 0 0 1-3.77-4.245V2h-3.445v13.672a2.896 2.896 0 0 1-5.201 1.743l-.002-.001.002.001a2.895 2.895 0 0 1 3.183-4.51v-3.5a6.329 6.329 0 0 0-5.394 10.692 6.33 6.33 0 0 0 10.857-4.424V8.687a8.182 8.182 0 0 0 4.773 1.526V6.79a4.831 4.831 0 0 1-1.003-.104z"/></svg>`
    },
    // Boosty
    {
        domains: [
            "boosty.to"
        ],
        iconHtml: `<svg class="icon" fill="#000000" xmlns="http://www.w3.org/2000/svg" x="0px" y="0px" viewBox="50 50 217.4 197.4">
                <style type="text/css">
                  .st0{fill:#242B2C;}
                  .st1{fill:url(#SVGID_1_);}
                </style>
                <g id="sign">
                  <g id="b_1_">
                    <linearGradient id="SVGID_1_" gradientUnits="userSpaceOnUse" x1="188.3014" y1="75.5591" x2="123.8106" y2="295.4895">
                      <stop offset="0" style="stop-color:#EF7829"/>
                      <stop offset="5.189538e-02" style="stop-color:#F07529"/>
                      <stop offset="0.3551" style="stop-color:#F0672B"/>
                      <stop offset="0.6673" style="stop-color:#F15E2C"/>
                      <stop offset="1" style="stop-color:#F15A2C"/>
                    </linearGradient>
                    <path class="st1" d="M87.5,163.9L120.2,51h50.1l-10.1,35c-0.1,0.2-0.2,0.4-0.3,0.6L133.3,179h24.8c-10.4,25.9-18.5,46.2-24.3,60.9    c-45.8-0.5-58.6-33.3-47.4-72.1 M133.9,240l60.4-86.9h-25.6l22.3-55.7c38.2,4,56.2,34.1,45.6,70.5C225.3,207,179.4,240,134.8,240    C134.5,240,134.2,240,133.9,240z"/>
                  </g>
                </g>
              </svg>`
    }
];
async function checkStoryLinks(story) {
    function addIcon(linkType, element) {
        const elem = element.cloneNode();
        elem.innerHTML = linkType.iconHtml.trim();
        if (linkType.style) {
            elem.setAttribute("style", linkType.style);
        }
        elem.classList.add('rpm-story-icon');
        // Add before the title
        const titleElem = story.querySelector('.story__title');
        titleElem.prepend(elem);
    }
    const linkElems = Array.from(story.querySelectorAll('.story__content a'));
    linkElems.reverse();
    // Iterate reversibly so that the last link is added to the icon list
    linkTypeFor: for (const linkType of linkTypes) {
        for (const domain of linkType.domains) {
            for (const linkElem of linkElems) {
                if (linkElem.href.includes(domain)) {
                    addIcon(linkType, linkElem);
                    continue linkTypeFor;
                }
            }
        }
    }
}
function removeStory(storyElem, reason, keepUser = false) {
    const titleElem = storyElem.querySelector('.story__title a.story__title-link');
    if (titleElem === null || storyElem.hasAttribute('rpm-deleted'))
        return;
    storyElem.setAttribute('rpm-deleted', '');
    const title = titleElem.textContent;
    const url = titleElem.href;
    const placeholder = document.createElement('div');
    placeholder.classList.add('rpm-placeholder');
    const urlElem = document.createElement('a');
    urlElem.textContent = title;
    urlElem.href = url;
    const userInfo = storyElem.querySelector('.story__user-info');
    if (keepUser && userInfo) {
        const userInfoContainer = document.createElement('div');
        userInfoContainer.append(userInfo.cloneNode(true));
        userInfoContainer.classList.add('rpm-user-info-container');
        // Update RPM ratings
        for (const ratingElem of userInfoContainer.querySelectorAll('.rpm-user-rating')) {
            const uid = parseInt(ratingElem.getAttribute('pikabu-user-id'));
            ratingElem.replaceWith(RPM.Nodes.createUserRatingNode(uid));
        }
        placeholder.append(urlElem, ` скрыт: ${reason}.`, userInfoContainer);
    }
    else {
        placeholder.append(urlElem, ` скрыт: ${reason}.`);
    }
    storyElem.parentElement.insertBefore(placeholder, storyElem);
    const collapseButton = document.createElement('div');
    collapseButton.classList.add("collapse-button", "collapse-button_active");
    collapseButton.append(document.createElement('div'), document.createElement('div'));
    collapseButton.addEventListener('click', () => {
        if (collapseButton.classList.contains('collapse-button_active'))
            collapseButton.classList.remove('collapse-button_active');
        else
            collapseButton.classList.add('collapse-button_active');
    });
    placeholder.prepend(collapseButton);
}
function processOldStory(story, storyData) {
    let ratingElem = story.querySelector(".story__footer-rating > div");
    let isMobile = false;
    if (ratingElem !== null) {
        // mobile
        isMobile = true;
    }
    else {
        // pc
        ratingElem = story.querySelector(".story__left .story__rating-block");
    }
    if (ratingElem === null) {
        return false;
    }
    oldInterface = true;
    let ratingDown = ratingElem.querySelector(".story__rating-minus, .story__rating-down");
    if (isMobile) {
        const buttonMinus = document.createElement("button");
        buttonMinus.classList.add("story__rating-minus");
        buttonMinus.innerHTML = `
    <span class="story__rating-rpm-count">${storyData.story.minuses}</span>
    <span type="button" class="tool story__rating-down" data-role="rating-down">
      <svg xmlns="http://www.w3.org/2000/svg" class="icon icon--ui__rating-down icon--ui__rating-down_story">
        <use xlink:href="#icon--ui__rating-down"></use>
      </svg>
    </span>`;
        ratingDown.replaceWith(buttonMinus);
        ratingDown = buttonMinus;
    }
    else {
        const minusesCounter = document.createElement("div");
        minusesCounter.classList.add("story__rating-count");
        minusesCounter.textContent = formats.formatStoryMinuses(storyData.story);
        ratingDown.prepend(minusesCounter);
    }
    if (GM_config.get('summary')) {
        const summary = document.createElement("div");
        if (isMobile)
            summary.classList.add("story__rating-rpm-count", "rpm-summary");
        else
            summary.classList.add("story__rating-count", "rpm-story-summary");
        summary.textContent = storyData.story.rating.toString();
        ratingDown.parentElement.insertBefore(summary, ratingDown);
    }
    const totalRates = storyData.story.pluses + storyData.story.minuses;
    if (GM_config.get('ratingBar') && totalRates >= GM_config.get('minRatesCountToShowRatingBar')) {
        let ratio = 0.5;
        if (totalRates > 0)
            ratio = storyData.story.pluses / totalRates;
        addRatingBar(story, ratio);
    }
    processStoryComments(storyData.story.id, storyData, 1);
    return true;
}
const trackedLinkPattern = /pikabu.ru.+\?[ut]=(.+?)&[ut]=.+/i;
function removeLinkTracker(link) {
    if (trackedLinkPattern.test(link.href)) {
        const realUrl = trackedLinkPattern.exec(link.href)[1];
        link.href = decodeURIComponent(realUrl);
    }
}
async function processStory(story, processComments) {
    // Block author button
    if (GM_config.get('showBlockAuthorForeverButton')) {
        addBlockButton(story);
    }
    // Links
    if (GM_config.get('socialLinks')) {
        checkStoryLinks(story);
    }
    // Remove pikabu trackers
    if (GM_config.get('noLinkTracking')) {
        const links = story.querySelectorAll('a');
        links.forEach(removeLinkTracker);
    }
    // Block paid stories
    if (enableFilters &&
        GM_config.get('blockPaidAuthors') &&
        story.querySelector(".user__label[data-type=\"pikabu-plus\"]") !== null) {
        removeStory(story, "подписка Пикабу+", true);
        info("Удалил пост", story, "как проплаченный");
    }
    // RPM
    if (GM_config.get('rpmEnabled')) {
        try {
            processStoryRpm(story);
        }
        catch (e) {
            error(e);
        }
    }
    const storyId = parseInt(story.getAttribute("data-story-id"));
    if (GM_config.get('rpmComments') && isStoryPage && currentStoryId === storyId) {
        document.querySelectorAll('.comment').forEach(processCommentRpm);
    }
    // get story data
    const storyData = await Pikabu.DataService.fetchStory(storyId, 1);
    if (storyData === null || storyData === undefined) {
        warn("Не удалось получить пост #", storyId);
        return;
    }
    // delete the story if its ratings < the min rating
    if (enableFilters &&
        storyData.story.rating < GM_config.get('minStoryRating')) {
        removeStory(story, `рейтинг поста (${storyData.story.rating})`);
        info("Удалил пост", story, "по фильтру рейтинга");
    }
    // videos
    if (GM_config.get('videoDownloadButtons'))
        processPostVideos(story, storyData);
    processOldStory(story, storyData);
}
async function processStoryRpm(story) {
    const uid = parseInt(story.getAttribute('data-author-id'));
    const userInfoRowElem = story.querySelector('.story__community_after-author-panel, .story__user-info');
    const footerElem = story.querySelector('.story__footer-tools .story__comments-link.story__to-comments');
    function ratingCallback(userInfo) {
        if (!enableFilters)
            return;
        const rating = userInfo.base_rating + userInfo.pluses - userInfo.minuses + (userInfo.own_vote ?? 0);
        const ownVote = userInfo.own_vote ?? 0;
        if (rating < GM_config.get('rpmMinStoryRating') && ownVote != 1) {
            removeStory(story, `RPM-рейтинг (${rating})`, true);
        }
        else if (ownVote === -1) {
            removeStory(story, `ваш минус пользователю в RPM`, true);
        }
    }
    const elem = RPM.Nodes.createUserRatingNode(uid, ratingCallback);
    if (userInfoRowElem)
        userInfoRowElem.prepend(elem);
    else
        footerElem.parentElement.insertBefore(elem, footerElem);
}
function getCommentAuthorId(comment) {
    if (comment.hasAttribute('data-author-id')) {
        return parseInt(comment.getAttribute('data-author-id'));
    }
    if (comment.hasAttribute('data-meta')) {
        return parseInt(comment.getAttribute('data-meta').match(/(?:^|;)aid=(\d+)(?:;|$)/)[1]);
    }
    return null;
}
function processCommentRpm(comment) {
    const uid = getCommentAuthorId(comment);
    if (!uid)
        return;
    const commentHeader = comment.querySelector('.comment__header');
    info(comment, uid);
    const elem = RPM.Nodes.createUserRatingNode(uid);
    commentHeader.insertBefore(elem, commentHeader.querySelector('.comment__right'));
}
async function processStories(stories) {
    for (const story of stories) {
        processStory(story, false);
    }
}
function processCached(commentElem) {
    const commentId = parseInt(commentElem.getAttribute("data-id"));
    if (commentId in cachedComments) {
        processComment(cachedComments[commentId]);
        delete cachedComments[commentId];
    }
}
function processPostVideos(story, storyData) {
    function getPostPlayers() {
        return Array.from(story.querySelectorAll('.story-block_type_video'));
    }
    function createUrl(text, url) {
        const urlElem = document.createElement('a');
        urlElem.target = '_blank';
        urlElem.textContent = text;
        urlElem.href = url;
        return urlElem;
    }
    function addUrlToPlayer(videoBlock, urls) {
        const player = videoBlock.querySelector('.player');
        // to check video origin
        const dataType = player.getAttribute("data-type");
        const urlListElem = document.createElement('p');
        urlListElem.classList.add('rpm-video-list');
        // if it's a pikabu video
        if (dataType == "video-file") {
            for (const url of urls) {
                const extension = '.' + url.split('.').pop();
                urlListElem.appendChild(createUrl(extension, url));
            }
        }
        else {
            // try get video url
            const dataSource = player.getAttribute('data-source');
            if (dataSource)
                urlListElem.appendChild(createUrl('Источник', dataSource));
        }
        if (urlListElem.hasChildNodes())
            videoBlock.parentElement.insertBefore(urlListElem, videoBlock.nextSibling);
        else
            urlListElem.remove();
    }
    const playerElements = getPostPlayers();
    for (const i in storyData.story.videos) {
        const player = playerElements[i];
        const videoUrls = storyData.story.videos[i];
        addUrlToPlayer(player, videoUrls);
    }
}
function addVideoDownloadButtons(postId, url) {
    const videoControls = Array.from(document.querySelectorAll(`.story[data-story-id="${postId}"] .player__controls`));
    function addButton(link, videoControls) {
        const a = document.createElement("a");
        a.classList.add("rpm-download-video-button");
        const name = link.split("/").pop(); // "https://example/com/some_cool_video.mp4" -> "some_cool_video.mp4"
        const extension = name.split(".").slice(1).join("."); // "some_cool_video.mp4" -> "mp4" (and "video.av1.mp4" -> "av1.mp4")
        a.href = link;
        a.download = name;
        a.textContent = extension;
        // add link to controls
        videoControls.append(a);
    }
    const videos = cachedPostVideos[postId];
    for (const i in videoControls) {
        for (const url of videos[i]) {
            addButton(url, videoControls[i]);
        }
    }
}
let customThemesDone = false;
function appendCustomThemesUi(themePicker) {
    if (customThemesDone)
        return;
    customThemesDone = true;
    // Dark mode buttons
    const darkModeLabel = themePicker.querySelector('.theme-picker__option_dark');
    const darkModeCheckbox = darkModeLabel.querySelector('span');
    const newCheckbox = darkModeCheckbox.cloneNode(true);
    darkModeCheckbox.replaceWith(newCheckbox);
    let isTimeouting = false;
    newCheckbox.addEventListener('click', () => {
        if (isTimeouting)
            return;
        isTimeouting = true;
        if (newCheckbox.classList.contains('checkbox_checked')) {
            newCheckbox.classList.remove('checkbox_checked');
            switchDarkMode(false);
        }
        else {
            newCheckbox.classList.add('checkbox_checked');
            switchDarkMode(true);
        }
        setTimeout(() => isTimeouting = false, 300);
    });
    const buttonsContainer = themePicker.querySelector('.theme-picker__buttons');
    const basicThemes = Array.from(buttonsContainer.children).map(x => x.getAttribute('data-type'));
    buttonsContainer.replaceChildren();
    // Custom theme buttons
    function createTheme(name, dataType, isBasic = false) {
        const themeButton = document.createElement('div');
        themeButton.classList.add('theme-picker__button');
        if (!isBasic)
            themeButton.classList.add('rpm-theme-picker');
        if (name !== null)
            themeButton.setAttribute('data-name', name);
        themeButton.setAttribute('data-type', dataType);
        const html = document.documentElement;
        let previousTheme = null;
        themeButton.addEventListener('mouseover', () => {
            previousTheme = html.getAttribute('data-theme');
            html.setAttribute('data-theme', dataType);
        });
        themeButton.addEventListener('mouseout', () => {
            if (previousTheme === null || previousTheme === 'default')
                html.removeAttribute('data-theme');
            else
                html.setAttribute('data-theme', previousTheme);
        });
        themeButton.addEventListener('click', () => {
            setTheme(dataType);
            previousTheme = dataType;
        });
        return themeButton;
    }
    // Add buttons
    const themes = [
        ...basicThemes.map(x => createTheme(null, x, true)),
        createTheme('Закат', 'sunset-glow'),
        createTheme('Бриз', 'ocean-breeze'),
        createTheme('Лес', 'forest-whisper'),
        createTheme('Лаванда', 'lavender-dreams'),
        createTheme('Огонь', 'fire-ember'),
        createTheme('Мозаика', 'mosaic'),
    ];
    buttonsContainer.append(...themes);
}
function updateTheme() {
    const theme = getTheme();
    const darkMode = isDarkMode();
    const html = document.documentElement;
    if (darkMode === null || !darkMode) {
        html.removeAttribute('data-theme-dark');
    }
    else {
        html.setAttribute('data-theme-dark', 'true');
    }
    if (theme === null || theme === 'default') {
        html.removeAttribute('data-theme');
    }
    else if (html.getAttribute('data-theme') !== theme) {
        html.setAttribute('data-theme', theme);
    }
}
function isDarkMode() {
    const config = JSON.parse(localStorage.getItem('pkb_theme'));
    return config !== null && config.d;
}
function switchDarkMode(enabled) {
    const config = JSON.parse(localStorage.getItem('pkb_theme'));
    if (config !== null) {
        config.d = enabled;
        localStorage.setItem('pkb_theme', JSON.stringify(config));
    }
    else {
        localStorage.setItem('rpm-dark-mode', JSON.stringify(enabled));
    }
    updateTheme();
}
function getTheme() {
    const rpmTheme = localStorage.getItem('rpm-theme');
    if (rpmTheme !== null)
        return rpmTheme;
    const config = JSON.parse(localStorage.getItem('pkb_theme'));
    if (config === null) {
        return 'default';
    }
    return config.t;
}
function setTheme(theme) {
    localStorage.setItem('rpm-theme', theme);
    updateTheme();
}
let csrfToken = null;
async function requestCsrfToken() {
    if (csrfToken !== null)
        return csrfToken;
    const appConfig = JSON.parse(document.querySelector('.app__config').textContent);
    csrfToken = appConfig.csrfToken;
    return csrfToken;
}
const newNotes = new Map();
function handleMiniProfile(element) {
    if (element.hasAttribute('rpm-affected'))
        return;
    element.setAttribute('rpm-affected', '');
    const userId = parseInt(element.getAttribute('data-user-id'));
    const userName = element.querySelector('.pkb-profile-username h2').textContent;
    function updateNoteElement() {
        let note = newNotes.get(userId);
        let noteHtml;
        const pkbNoteElem = element.querySelector('.mini-profile__note');
        if (pkbNoteElem) {
            if (note === undefined || note === null) {
                note = pkbNoteElem.textContent;
                noteHtml = pkbNoteElem.innerHTML;
                for (const link of pkbNoteElem.querySelectorAll('a')) {
                    note = note.replace(link.textContent, link.href);
                }
            }
            pkbNoteElem.remove();
        }
        const noteElem = document.createElement('div');
        noteElem.classList.add('input__box', 'mini-profile__note');
        const textareaElem = document.createElement('textarea');
        textareaElem.classList.add('input__input', 'profile-note__textarea', 'rpm-mini-profile-note');
        textareaElem.setAttribute('rows', '1');
        textareaElem.setAttribute('placeholder', 'Заметка об этом пользователе будет видна только вам. Нажмите, чтобы ввести текст.');
        textareaElem.value = note ?? '';
        const noteDisplayElem = document.createElement('cite');
        noteDisplayElem.classList.add('rpm-mini-profile-note-display');
        noteDisplayElem.innerHTML = noteHtml ?? note ?? 'Заметка об этом пользователе будет видна только вам. Нажмите, чтобы ввести текст.';
        noteElem.append(textareaElem, noteDisplayElem);
        const mainElem = element.querySelector('.mini-profile__main');
        mainElem.append(noteElem);
        const rpmPoweredElem = RPM.Nodes.createPoweredNote('Улучшено с помощью Return Pikabu minus');
        mainElem.append(rpmPoweredElem);
        function setTextareaActive(active) {
            if (active) {
                textareaElem.style.display = '';
                noteDisplayElem.style.display = 'none';
            }
            else {
                textareaElem.style.display = 'none';
                noteDisplayElem.style.display = '';
            }
        }
        setTextareaActive(note === '');
        let saved = true;
        async function save() {
            saved = true;
            newNotes.set(userId, note);
            const action = note !== '' ? 'note+' : 'note-';
            await fetch('/ajax/users_actions.php', {
                method: 'POST',
                headers: {
                    "X-Csrf-Token": await requestCsrfToken(),
                    "Accept": "application/json, text/javascript, */*; q=0.01",
                    "X-Requested-With": "XMLHttpRequest",
                    "Content-Type": "application/x-www-form-urlencoded; charset=UTF-8",
                    "Priority": "u=0"
                },
                body: `id=${userId}&action=${encodeURIComponent(action)}&message=${encodeURIComponent(note)}`
            });
            sendNotification('Успешно', 'Заметка сохранена');
        }
        let timeout = null;
        textareaElem.addEventListener('input', () => {
            saved = false;
            note = textareaElem.value;
            if (timeout !== null) {
                clearTimeout(timeout);
            }
            timeout = setTimeout(save, 1500);
        });
        const onUnfocus = () => {
            setTextareaActive(false);
            noteDisplayElem.textContent = textareaElem.value;
            if (timeout !== null && !saved) {
                clearTimeout(timeout);
                timeout = null;
                save();
            }
        };
        textareaElem.addEventListener('blur', onUnfocus);
        textareaElem.addEventListener('focusout', onUnfocus);
        noteDisplayElem.addEventListener('click', () => {
            setTextareaActive(true);
            textareaElem.focus();
        });
    }
    function gollumIntegration() {
        const main = document.createElement('div');
        main.classList.add('rpm-gollum-stats');
        let worked = false;
        if (GM_config.get('miniProfileStoryTags')) {
            const storyTagsElem = Gollum.createTagsContainer(userName, 'Теги постов', Gollum.storyTagsGetter, GM_config.get('miniProfileAutoloadTags'));
            main.append(storyTagsElem);
            worked = true;
        }
        if (GM_config.get('miniProfileСommentTags')) {
            const commentsTagsElem = Gollum.createTagsContainer(userName, 'Теги комментариев', Gollum.commentTagsGetter, GM_config.get('miniProfileAutoloadTags'));
            main.append(commentsTagsElem);
            worked = true;
        }
        if (GM_config.get('miniProfileСomments')) {
            const lastComments = Gollum.createLastCommentsSection(userName, GM_config.get('miniProfileAutoloadComments'), GM_config.get('miniProfileAutoloadPikabuCommentCount'), true);
            main.append(lastComments);
            worked = true;
        }
        if (worked) {
            const poweredElem = RPM.Nodes.createPoweredNote(`Данные получены с <a href="https://gollum.space/user/${userName.replace('.', '_')}-summary" target="_blank">gollum.space</a>`);
            main.append(poweredElem);
        }
        element.append(main);
    }
    if (GM_config.get('miniProfileEditableNote')) {
        updateNoteElement();
    }
    gollumIntegration();
}
function mutationsListener(mutationList, observer) {
    for (const mutation of mutationList) {
        if (mutation.type === 'childList') {
            for (const node of mutation.addedNodes) {
                if (!(node instanceof HTMLElement))
                    continue;
                if (node.hasAttribute('rpm-observer-ignore'))
                    continue;
                if (node.matches(".comment__header")) {
                    const commentElem = node.closest(".comment");
                    info('Поймал голову комментария!', commentElem);
                    processCached(commentElem);
                    // RPM
                    if (GM_config.get('rpmComments')) {
                        processCommentRpm(commentElem);
                    }
                }
                else if (node.matches("article.story")) {
                    const storyElem = node;
                    info('Поймал пост!', storyElem);
                    processStory(storyElem, false);
                }
                else if (node.matches(".comment__more:not(.rpm-unroll-all)")) {
                    commentMoreBtn();
                }
                else if (node.matches('.overlay')) {
                    info('Поймал .overlay!');
                    waitForElement('.theme-picker__popup, .mini-profile', 500, node).then((e) => {
                        if (e.matches('.mini-profile')) {
                            info('Поймал мини-профиль!', e);
                            handleMiniProfile(e);
                        }
                        else {
                            info('Поймал селектор типов!');
                            appendCustomThemesUi(e);
                        }
                    }, () => { });
                }
                else if (node.matches('.mini-profile')) {
                    handleMiniProfile(node);
                }
            }
        }
    }
}
var observer = null;
function addSettingsOpenButton() {
    let block = 
    // mobile version
    document.querySelector('.footer__links .accordion')
        // else PC version
        ?? document.querySelector(".sidebar .sidebar__inner");
    if (block === null) {
        error("Не удалось найти место для создания кнопки открытия настроек.");
        return;
    }
    const button = document.createElement('button');
    button.innerText = "Открыть настройки Return Pikabu minus";
    button.classList.add('rpm-open-settings-button');
    button.addEventListener('click', () => {
        button.disabled = true;
        GM_config.open();
        button.disabled = false;
    });
    block.appendChild(button);
}
var FeedbackManager;
(function (FeedbackManager) {
    async function init() {
        const settings = await getSettings();
        const isToday = settings.lastCheckDate.toDateString() === new Date().toDateString();
        if (!isToday) {
            const feedbacks = await RPM.Service.getFeedbacks();
            feedbacks.forEach(feedback => {
                if (settings.completed.includes(feedback.id))
                    return;
                if (settings.saved.find(([, fb]) => fb.id == feedback.id) !== undefined)
                    return;
                settings.saved.push([new Date(), feedback]);
                showFeedback(feedback);
            });
            settings.lastCheckDate = new Date();
            await updateSettings(settings);
        }
        else {
            showSavedFeedback();
        }
    }
    FeedbackManager.init = init;
    async function showSavedFeedback() {
        const settings = await getSettings();
        const now = new Date();
        const saved = settings.saved.filter(([date]) => date <= now).map(([, fb]) => fb);
        if (saved.length === 0)
            return;
        for (const fb of saved) {
            await showFeedback(fb);
        }
    }
    async function getSettings() {
        const settings = JSON.parse(await GM.getValue("rpm-feedback", JSON.stringify({
            lastCheckDate: (() => { const date = new Date(); date.setDate(date.getDate() - 1); return date; })(),
            completed: [],
            saved: []
        })));
        settings.lastCheckDate = new Date(settings.lastCheckDate);
        for (const entry of settings.saved) {
            entry[0] = new Date(entry[0]);
        }
        return settings;
    }
    function updateSettings(settings) {
        settings.saved = settings.saved.filter(fb => !settings.completed.includes(fb[1].id));
        return GM.setValue("rpm-feedback", JSON.stringify(settings));
    }
    async function markCompleted(feedbackId) {
        const settings = await getSettings();
        settings.completed.push(feedbackId);
        await updateSettings(settings);
    }
    let isFeedbackActive = false;
    function showFeedback(feedback) {
        if (isFeedbackActive)
            return Promise.resolve();
        isFeedbackActive = true;
        const deffered = new Deferred();
        const feedbackElem = document.createElement('div');
        feedbackElem.classList.add('rpm-feedback-window');
        const iframe = document.createElement('iframe');
        iframe.src = feedback.iframe_url;
        const completedButton = document.createElement('button');
        completedButton.classList.add('rpm-completed');
        completedButton.textContent = 'Закрыть';
        completedButton.addEventListener('click', () => {
            feedbackElem.remove();
            markCompleted(feedback.id);
            deffered.resolve();
        });
        feedbackElem.append(iframe, completedButton);
        document.body.append(feedbackElem);
        return deffered.promise;
    }
})(FeedbackManager || (FeedbackManager = {}));
async function processTabs() {
    const tabConfig = {
        hot: 'hotTab',
        best: 'bestTab',
        new: 'newTab',
        my_lent: 'subsTab',
        communities: 'communitiesTab',
        companies: 'blogsTab',
        experts: 'expertsTab'
    };
    await waitForElement('.header-menu__item, .pkb-tab-list');
    Object.entries(tabConfig).forEach(([key, field]) => {
        const selector = `.header-menu__item[data-feed-key="${key}"], .pkb-tab[data-feed-key="${key}"]`;
        if (!GM_config.get(field)) {
            const element = document.querySelector(selector);
            if (element) {
                element.remove();
            }
        }
    });
}
function init() {
    updateTheme();
    addCss(STYLES);
    observer = new MutationObserver(mutationsListener);
    observer.observe(document.body, {
        childList: true,
        subtree: true
    });
    window.addEventListener("load", onLoad);
    waitConfig().then(onConfig);
    if (window.location.pathname.startsWith('/@')) {
        onUserProfilePage();
    }
}
async function onConfig() {
    processTabs();
    if (GM_config.get('uuid')) {
        delete GM_config.fields['registerRpm'];
    }
}
function unrollComments(button) {
    button.click();
    setTimeout(() => {
        if (document.body.contains(button)) {
            unrollComments(button);
        }
    }, 500);
}
function commentMoreBtn() {
    const value = GM_config.get('unrollCommentaries');
    if (value === SettingEnums.UnrollComments.NONE)
        return;
    const moreButton = document.querySelector('.comment__more');
    if (!moreButton)
        return;
    if (value === SettingEnums.UnrollComments.UNROLL_ALL_BUTTON) {
        const btn = document.createElement('button');
        btn.textContent = 'Раскрыть все комментарии';
        btn.classList.add('rpm-unroll-all');
        btn.addEventListener('click', () => {
            btn.remove();
            unrollComments(moreButton);
        });
        moreButton.parentElement.append(btn);
    }
    else if (value === SettingEnums.UnrollComments.AUTO_UNROLL) {
        unrollComments(moreButton);
    }
}
async function onLoad() {
    updateTheme();
    processStories(document.querySelectorAll("article.story"));
    if (!supportMenuCommands)
        addSettingsOpenButton();
}
async function onUserProfilePage() {
    const userName = document.querySelector('.page-profile .profile__nick').textContent;
    const feedPanel = document.querySelector('.feed-panel, .user-filters');
    let lastSection = null;
    await waitConfig();
    if (GM_config.get('profileStoryTags')) {
        const section = document.createElement('section');
        section.append(Gollum.createTagsContainer(userName, 'Теги постов', Gollum.storyTagsGetter, true));
        feedPanel.parentElement.insertBefore(section, feedPanel);
        lastSection = section;
    }
    if (GM_config.get('profileСommentTags')) {
        const section = document.createElement('section');
        section.append(Gollum.createTagsContainer(userName, 'Теги комментариев', Gollum.commentTagsGetter, true));
        feedPanel.parentElement.insertBefore(section, feedPanel);
        lastSection = section;
    }
    if (GM_config.get('profileСomments')) {
        const section = document.createElement('section');
        section.append(Gollum.createLastCommentsSection(userName, GM_config.get('profileAutoloadComments'), GM_config.get('profileAutoloadPikabuCommentCount')));
        feedPanel.parentElement.insertBefore(section, feedPanel);
        lastSection = section;
    }
    // Powered
    if (lastSection !== null) {
        const poweredElem = RPM.Nodes.createPoweredNote(`Данные получены с <a href="https://gollum.space/user/${userName.replace('.', '_')}-summary" target="_blank">gollum.space</a>`);
        const rpmPoweredElem = RPM.Nodes.createPoweredNote('Работает с помощью Return Pikabu minus');
        lastSection.append(poweredElem, rpmPoweredElem);
    }
}
init();