Malayala Kit

MalayalaKit: JavaScript library for creating customizable web menus with tabs, buttons, switches, and input elements.

このスクリプトは単体で利用できません。右のようなメタデータを含むスクリプトから、ライブラリとして読み込まれます: // @require https://update.greasyfork.org/scripts/482771/1323949/Malayala%20Kit.js

このスクリプトの質問や評価の投稿はこちら通報はこちらへお寄せください。
/*
Created by anonimbiri
*/

var MalayalaKit = (function () {
    const usedClasses = new Set();

    const generateRandomClass = (toLowerCase) => {
        const characters = 'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ';
        const length = Math.floor(Math.random() * 6) + 5;

        let newClass;
        do {
            newClass = Array.from({ length }, () => characters[Math.floor(Math.random() * characters.length)]).join('');
        } while (usedClasses.has(toLowerCase ? newClass.toLowerCase() : newClass));

        usedClasses.add(toLowerCase ? newClass.toLowerCase() : newClass);

        return toLowerCase ? newClass.toLowerCase() : newClass;
    };

    const classList = Object.fromEntries(
        ['menu-list', 'menu-container', 'border', 'tabs-container', 'tab', 'active', 'tab-underline', 'content-container', 'row', 'label', 'button', 'white-border', 'switch-container', 'switch-label', 'inner-circle', 'input', 'menu-title', 'close-button', 'slider-container', 'color', 'color-picker-container', 'color-picker-element', 'circular-area', 'toast-container', 'position', 'top', 'bottom', 'left', 'right', 'center', 'toast', 'info', 'warning', 'error', 'progress'].map(key =>
            [key, key === 'position' ? generateRandomClass(true) : generateRandomClass(false)]
        )
    );

    var cssStyles = `
        .${classList['menu-list']} {
            font-family: Arial, sans-serif;
            align-items: center;
            justify-content: center;
            overflow: hidden auto;
            -webkit-backdrop-filter: blur(2px);
            backdrop-filter: blur(2px);
            position: fixed;
            inset: 0;
            z-index: 1000;
            overflow: auto;
            outline: 0;
            -webkit-overflow-scrolling: touch;
            overflow: hidden;
            user-select: none;
        }

        .${classList['menu-container']} {
            background-color: #121212;
            box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1);
            border-radius: 8px;
            overflow: hidden;
            width: 300px;
            display: flex;
            flex-direction: column;
            position: fixed;
        }

        @keyframes borderAnimation {
            0%{background-position:0% 50%}
            50%{background-position:100% 50%}
            100%{background-position:0% 50%}
        }

        .${classList['border']} {
            background: linear-gradient(268deg, #ff3939, #eeff30, #37ff30, #30ffd6, #ff30f4);
            background-size: 1000% 1000%;
            animation: borderAnimation 30s ease infinite;
            width: calc(101%);
            height: calc(1%);
        }

        .${classList['tabs-container']} {
            display: flex;
            background: linear-gradient(to bottom, #222222, #111111);
            color: #fff;
            cursor: grab;
            padding: 0 24px;
            position: relative;
            background-color: rgba(0,0,0,0.4);
            border-bottom: 1px solid rgba(250, 250, 250, 0.12);
        }

        .${classList['tab']} {
            cursor: pointer;
            padding: 10px;
            margin-right: 10px;
            position: relative;
            transition: background 0.2s;
        }

        .${classList['tab']}:hover {
            color: #ffffff!important;
            background: rgba(255, 255, 255, 0.06);
            border-radius: 5px;
        }

        .${classList['tab-underline']} {
            height: 2px;
            position: absolute;
            background: #eeeeee;
            pointer-events: none;
            bottom: 0;
            box-sizing: border-box;
            display: block;
            white-space: nowrap;
            transition: width 0.3s, left 0.3s, right 0.3s;
        }

        .${classList['tab']}.${classList['active']} .${classList['tab-underline']} {
            transform: scaleX(1);
        }

        .${classList['content-container']} {
            margin: 20px;
            overflow: auto;
            scrollbar-width: thin;
            scrollbar-color: #6b6b6b #f1f1f1;
            padding-right: calc(10px + 5px);
            padding-bottom: 0.9vw;
        }
        
        .${classList['content-container']}::-webkit-scrollbar {
            width: 12px;
            cursor: pointer;
            width: 4px;
            height: 4px;
            background-color: transparent;
        }
        
        .${classList['content-container']}::-webkit-scrollbar-corner {
            display: none;
            width: 0;
            height: 0;
        }
        
        .${classList['content-container']}::-webkit-scrollbar-thumb {
            cursor: pointer;
            background-color: transparent;
            border-radius: 2px;
            transition: background-color 0.5s cubic-bezier(0.215, 0.61, 0.355, 1);
        }
        
        .${classList['content-container']}:hover::-webkit-scrollbar-thumb {
            background-color: rgba(255, 255, 255, 0.16);
        }
        
        .${classList['content-container']}::-webkit-scrollbar-thumb:hover {
            background-color: rgb(255, 255, 255);
        }

        .${classList['row']} {
            align-items: center;
            display: flex;
            padding-left: 10px;
            padding-right: 10px;
            padding-top: 20px;
            justify-content: space-between;
        }

        .${classList['row']} .${classList['label']} {
            display: inline-block;
            color: white;
        }

        .${classList['button']} {
            height: 30px;
            border-radius: 5px;
            color: #fff;
            text-align: center;
            display: inline-block;
            background-color: #111111;
            border: none;
            cursor: pointer;
            transition: background 0.2s;
            position: relative;
            z-index: 0;
        }

        .${classList['button']}:hover {
            background-color: rgba(255, 255, 255, 0.10);
        }

        /*.rgb-style:after { 
            content: '';
            position: absolute;
            background: #111111;
        }*/

        .${classList['white-border']}:before {
            content: '';
            border: 1px solid white;
            border-radius: 5px;
            position: absolute;
            top: -0.9px;
            left: -1.6px;
            z-index: -1;
            width: calc(100% + 1px);
            height: calc(100% + 1px);
        }
        
        /*.rgb-style:before {
            content: '';
            background: linear-gradient(268deg, #ff3939, #eeff30, #37ff30, #30ffd6, #ff30f4);
            position: absolute;
            top: -0.2px;
            left: 0px;
            z-index: -1;
            filter: blur(10px);
            width: calc(100% + 1px);
            height: calc(100% + 1px);
            background-size: 1000% 1000%;
            animation: borderAnimation 30s ease infinite;
            border-radius: 5px;
        }*/

        .${classList['switch-container']} {
            display: flex;
            align-items: center;
            margin-bottom: 10px;
        }

        .${classList['switch-container']} input {
            display: none;
        }

        .${classList['switch-container']} input + .${classList['switch-label']} {
            transition: background-color 0.2s;
        }

        .${classList['switch-container']} input:checked + .${classList['switch-label']} {
            background: #fff;
        }    

        .${classList['switch-container']} input:checked + .${classList['switch-label']} .${classList['inner-circle']} {
            transform: translateX(20px);
        }

        .${classList['switch-label']} {
            position: relative;
            display: inline-block;
            width: 40px;
            height: 20px;
            background: #555555;
            border-radius: 100px;
            cursor: pointer;
        }

        .${classList['switch-label']} .${classList['inner-circle']} {
            position: absolute;
            top: 2px;
            left: 2px;
            width: 16px;
            height: 16px;
            background: rgb(17, 17, 17);
            border-radius: 50%;
            transition: transform 0.2s;
        }

        .${classList['input']} {
            position: relative;
            display: inline-flex;
            width: 40%;
            min-width: 0;
            padding: 6px 11px;
            color: #ffffff;
            font-size: 14px;
            line-height: 1.5714285714285714;
            background-color: #111111;
            background-image: none;
            border-width: 1px;
            border-style: solid;
            border-color: #333333;
            border-radius: 5px;
            transition: all 0.2s;
        }

        .${classList['input']}:hover {
            border-color: #eeeeee;
            box-shadow: 0 0 0 2px rgba(17, 17, 17, 0.01);
            outline: 0;
        }

        .${classList['input']}:focus {
            border-color: #eeeeee;
            box-shadow: 0 0 0 2px rgba(17, 17, 17, 0.01);
            outline: 0;
        }

        .${classList['menu-title']} {
            display: flex;
            justify-content: space-between;
            align-items: center;
            padding: 10px;
            background: #121212;
            color: #fff;
            cursor: grab;
        }

        .${classList['menu-title']} img {
            width: 30px;
            height: 30px;
            margin-right: 10px;
            -webkit-user-drag: none;
            -khtml-user-drag: none;
            -moz-user-drag: none;
            -o-user-drag: none;
            user-drag: none;
        }

        .${classList['menu-title']} a {
            text-align: center;
            font-size: unset;
            font-weight: unset;
            font-family: revert;
            color: #fff;
            background-color: transparent;
            -webkit-user-drag: none;
            -khtml-user-drag: none;
            -moz-user-drag: none;
            -o-user-drag: none;
            user-drag: none;
        }

        .${classList['close-button']} {
            cursor: pointer;
            border: none;
            margin-right: 10px;
            position: relative;
            height: 30px;
            width: 30px;
            border-radius: 8px;
            background: transparent;
            align-items: center;
            transition: background 0.2s;
        }

        .${classList['close-button']}:hover {
            background-color: rgba(255, 255, 255, 0.10);
        }

        .${classList['close-button']} svg {
            width: 100%;
            height: 100%;
        }

        .${classList['slider-container']} {
            display: flex;
            justify-content: flex-end;
            align-items: center;
            width: 100%;
            margin: 10px 0;
        }
        
        .${classList['slider-container']} .${classList['input']} {
            width: 15%;
        }
        
        .${classList['slider-container']} input[type="range"] {
            width: 50%;
            -webkit-appearance: none;
            appearance: none;
            height: 5px;
            border-radius: 8px;
            outline: none;
            -webkit-transition: opacity 0.2s;
            transition: opacity 0.2s;
            cursor: pointer;
            margin-right: 10px;
        }
        
        .${classList['slider-container']} input[type="range"]::-webkit-slider-thumb {
            -webkit-appearance: none;
            appearance: none;
            width: 8px;
            height: 16px;
            border-radius: 2px;
            background: #fff;
            cursor: pointer;
        }
        
        .${classList['slider-container']} input[type="range"]::-moz-range-thumb {
            width: 15px;
            height: 15px;
            border-radius: 50%;
            background: #fff;
            cursor: pointer;
        }

        .${classList['slider-container']} input[type="range"]::-webkit-slider-thumb:hover,
        .${classList['slider-container']} input[type="range"]::-moz-range-thumb:hover {
            background: #ff0000;
        }

        .${classList['color']} {
            display: none;
        }

        .${classList['color-picker-container']} {
            display: flex;
            justify-content: flex-end;
            align-items: center;
        }

        .${classList['color-picker-container']} .${classList['input']} {
            width: 30%;
        }
        
        .${classList['color-picker-element']},
        .${classList['circular-area']} {
            margin-right: 10px;
        }
        
        .${classList['circular-area']} {
            transition: all 0.2s;
            border-radius: 50%;
            cursor: pointer;
            width: 25px;
            height: 25px;
            background-color: transparent;
        }
        
        .${classList['circular-area']}:hover {
            box-shadow: 0 0 0 2px #ffffff;
        }

        .${classList['toast-container']} {
            position: fixed;
            overflow: hidden;
            z-index: 1000;
        }

        .${classList['toast-container']}[data-${classList['position']}^="${classList['top']}-"] {
            top: 20px;
        }

        .${classList['toast-container']}[data-${classList['position']}^="${classList['bottom']}-"] {
            bottom: 20px;
        }

        .${classList['toast-container']}[data-${classList['position']}$="-${classList['left']}"] {
            left: 20px;
        }

        .${classList['toast-container']}[data-${classList['position']}$="-${classList['right']}"] {
            right: 20px;
        }

        .${classList['toast-container']}[data-${classList['position']}$="-${classList['center']}"] {
            left: 50%;
            transform: translateX(-50%);
        }
        
        .${classList['toast']} {
            position: relative;
            background: #111111;
            color: #fff;
            padding: 10px;
            margin-bottom: 8px;
            border-radius: 4px;
            box-shadow: 0 0 10px rgba(0, 0, 0, 0.2);
            font-size: 14px;
            font-weight: bold;
            font-family: monospace;
            pointer-events: auto;
            display: flex;
            align-items: center;
            overflow: hidden;
            user-select: none;
            justify-content: space-between;
            transition: opacity 0.3s ease-in-out;
        }

        .${classList['toast']}.${classList['info']} {
            background: #002126;
            color: rgb(138, 232, 255);
        }

        .${classList['toast']}.${classList['warning']} {
            background: #271a00;
            color: rgb(255, 203, 71);
        }

        .${classList['toast']}.${classList['error']} {
            background: #34001d;
            color: rgb(240, 79, 136);
        }

        .${classList['toast']}.${classList['info']} .${classList['close-button']} svg {
            stroke: rgb(138, 232, 255);
        }

        .${classList['toast']}.${classList['warning']} .${classList['close-button']} svg {
            stroke: rgb(255, 203, 71);
        }

        .${classList['toast']}.${classList['error']} .${classList['close-button']} svg {
            stroke: rgb(240, 79, 136);
        }

        .${classList['toast']} .${classList['close-button']} {
            margin-right: 0px;
            margin-left: 10px;
        }

        .${classList['toast-container']}[data-${classList['position']}="${classList['top']}-${classList['left']}"] .${classList['toast']},
        .${classList['toast-container']}[data-${classList['position']}="${classList['bottom']}-${classList['left']}"] .${classList['toast']} {
            animation: slideInLeft 0.3s ease-in-out;
        }

        .${classList['toast-container']}[data-${classList['position']}="${classList['top']}-${classList['center']}"] .${classList['toast']} {
            animation: slideInTopCenter 0.3s ease-in-out;
        }

        .${classList['toast-container']}[data-${classList['position']}="${classList['top']}-${classList['right']}"] .${classList['toast']}, 
        .${classList['toast-container']}[data-${classList['position']}="${classList['bottom']}-${classList['right']}"] .${classList['toast']} {
            animation: slideInRight 0.3s ease-in-out;
        }

        .${classList['toast-container']}[data-${classList['position']}="${classList['bottom']}-${classList['center']}"] .${classList['toast']} {
            animation: slideInBottomCenter 0.3s ease-in-out;
        }

        .${classList['toast']}.${classList['progress']}::before {
            content: "";
            position: absolute;
            height: 2px;
            width: calc(100%* var(--progress));
            background-color: #fff;
            bottom: 0;
            left: 0;
            right: 0;
            margin-right: auto;
        }

        .${classList['toast']}.${classList['info']}.${classList['progress']}::before {
            background-color: rgb(138, 232, 255);
        }

        .${classList['toast']}.${classList['warning']}.${classList['progress']}::before {
            background-color: rgb(255, 203, 71);
        }

        .${classList['toast']}.${classList['error']}.${classList['progress']}::before {
            background-color: rgb(240, 79, 136);
        }

        @keyframes slideInRight {
            from {
                transform: translateX(200%);
            }
            to {
                transform: translateX(0);
            }
        }

        @keyframes slideInLeft {
            from {
                transform: translateX(-200%);
            }
            to {
                transform: translateX(0);
            }
        }

        @keyframes slideInTopCenter {
            from {
                transform: translateY(-200%);
            }
            to {
                transform: translateY(0);
            }
        }

        @keyframes slideInBottomCenter {
            from {
                transform: translateY(200%);
            }
            to {
                transform: translateY(0);
            }
        }
    `;

    var style = document.createElement('style');
    style.type = 'text/css';
    style.innerHTML = cssStyles;
    document.head.appendChild(style);

    var menuList = document.createElement('div');
    menuList.className = classList['menu-list'];
    document.body.appendChild(menuList);

    var translations = {
        'en': {
            'pressKey': 'Press a key...',
            'setHotkey': 'Set Hotkey'
        },
        'tr': {
            'pressKey': 'Bir tuşa basın...',
            'setHotkey': 'Kısayol Ayarla'
        },
        'ml': {
            'pressKey': 'ഒരു കീ അമർത്തുക...',
            'setHotkey': 'ഹോട്ട്കീ സജ്ജീകരിക്കുക'
        },
        'ta': {
            'pressKey': 'ஒரு விசில் அழுத்தவும்...',
            'setHotkey': 'ஹாட்கீ அமைக்கவும்'
        },
        'hi': {
            'pressKey': 'एक कुंजी दबाएं...',
            'setHotkey': 'हॉटकी सेट करें'
        },
        'ko': {
            'pressKey': '키를 누르세요...',
            'setHotkey': '단축키 설정'
        },
        'ja': {
            'pressKey': 'キーを押してください...',
            'setHotkey': 'ホットキーを設定'
        },
        'it': {
            'pressKey': 'Premi un tasto...',
            'setHotkey': 'Imposta Hotkey'
        },
        'ar': {
            'pressKey': 'اضغط على مفتاح...',
            'setHotkey': 'تعيين مفتاح الاختصار'
        },
        'fr': {
            'pressKey': 'Appuyez sur une touche...',
            'setHotkey': 'Définir le raccourci'
        },
    };

    var selectedLanguage = navigator.language.split('-')[0];

    function translate(key) {
        return translations[selectedLanguage][key] || key;
    }

    function setLanguage(lang) {
        if (translations[lang]) {
            selectedLanguage = lang;
        } else {
            console.error('Translation not available for the selected language.');
        }
    }

    function CreateMenu(options) {
        this.title = options.title;
        this.icon = options.icon;
        this.isOpen = true;
        this.tabs = [];

        this.render = function () {
            var windowHeight = window.innerHeight || document.documentElement.clientHeight || document.body.clientHeight;
            var windowWidth = window.innerWidth || document.documentElement.clientWidth || document.body.clientWidth;
            this.container = document.createElement('div');
            this.container.className = classList['menu-container'];
            this.container.style.width = (options.size && options.size.width) ? options.size.width + 'px' : '500px';
            this.container.style.height = (options.size && options.size.height) ? options.size.height + 'px' : '400px';
            this.container.style.top = (options.position && options.position.top) ? options.position.top + 'px' : (windowHeight / 2 - this.container.clientHeight / 2) + 'px';
            this.container.style.left = (options.position && options.position.left) ? options.position.left + 'px' : (windowWidth / 2 - this.container.clientWidth / 2) + 'px';

            this.border = document.createElement('div');
            this.border.className = classList['border'];

            this.tabsContainer = document.createElement('div');
            this.tabsContainer.className = classList['tabs-container'];
            this.tabsContainer.style.cursor = 'grab';

            this.underline = document.createElement('div');
            this.underline.className = classList['tab-underline'];

            this.contentContainer = document.createElement('div');
            this.contentContainer.className = classList['content-container'];

            this.menuTitleContainer = document.createElement('div');
            this.menuTitleContainer.className = classList['menu-title'];
            this.menuIcon = document.createElement('img');
            this.menuIcon.setAttribute('src', this.icon);

            this.menuIcon.addEventListener('error', function () {
                this.menuIcon.setAttribute('src', '');
            }.bind(this));

            this.menuTitleContainer.appendChild(this.menuIcon);

            this.menuTitle = document.createElement("a");
            this.menuTitle.textContent = this.title || 'MalayalaKit';
            this.menuTitleContainer.appendChild(this.menuTitle);

            this.menuTitleContainer.addEventListener('mousedown', function (e) {
                isDragging = true;
                offsetX = e.clientX - this.container.offsetLeft;
                offsetY = e.clientY - this.container.offsetTop;
                this.menuTitleContainer.style.cursor = 'grabbing';
            }.bind(this));

            this.menuTitleContainer.addEventListener('mouseup', function () {
                isDragging = false;
                this.menuTitleContainer.style.cursor = 'grab';
            }.bind(this));

            this.closeButton = document.createElement('button');
            this.closeButton.className = classList['close-button'];
            var closeIconSvg = `
        <svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" stroke="white" stroke-width="2">
            <line x1="18" y1="6" x2="6" y2="18" />
            <line x1="6" y1="6" x2="18" y2="18" />
        </svg>
        `;

            this.closeButton.innerHTML = closeIconSvg;
            this.closeButton.addEventListener('click', function () {
                this.toggleMenu();
            }.bind(this));

            this.menuTitleContainer.appendChild(this.closeButton);
            this.container.appendChild(this.border);
            this.container.appendChild(this.menuTitleContainer);
            this.tabsContainer.appendChild(this.underline);
            this.container.appendChild(this.tabsContainer);
            this.container.appendChild(this.contentContainer);

            menuList.appendChild(this.container);

            var isDragging = false;
            var offsetX, offsetY;

            this.tabsContainer.addEventListener('mousedown', function (e) {
                isDragging = true;
                offsetX = e.clientX - this.container.offsetLeft;
                offsetY = e.clientY - this.container.offsetTop;
                this.tabsContainer.style.cursor = 'grabbing';
            }.bind(this));

            document.addEventListener('mousemove', function (e) {
                if (isDragging) {
                    this.container.style.left = e.clientX - offsetX + 'px';
                    this.container.style.top = e.clientY - offsetY + 'px';
                }
            }.bind(this));

            document.addEventListener('mouseup', function () {
                isDragging = false;
                this.tabsContainer.style.cursor = 'grab';
            }.bind(this));

            document.addEventListener('keydown', function (event) {
                if (isHotkeyPressed(options.hotkey, event)) {
                    this.toggleMenu();
                }
            }.bind(this));

            //this.tabsContainer.innerHTML = '';
            this.tabs.forEach(function (tab, index) {
                var tabElement = document.createElement('div');
                tabElement.className = classList['tab'];
                tabElement.textContent = tab.label;

                tabElement.addEventListener('click', function () {
                    this.contentContainer.innerHTML = '';
                    tab.renderContent(this.contentContainer);
                    const containerRect = this.tabsContainer.getBoundingClientRect();
                    const tabElementRect = tabElement.getBoundingClientRect();
                    this.underline.style.width = tabElementRect.width + 'px';
                    this.underline.style.left = tabElementRect.left - containerRect.left + 'px';
                }.bind(this));

                this.tabsContainer.insertBefore(tabElement, this.underline);

                if (index === 0) {
                    tabElement.click();
                }

            }.bind(this));
        }

        function toggleMenuVisibility(isOpen) {
            menuList.style.display = isOpen ? 'none' : 'flex';

            if (options.pointerLock) {
                const pointerLockElement = document.pointerLockElement || document.mozPointerLockElement || document.webkitPointerLockElement;

                if (pointerLockElement || !isOpen) {
                    document.exitPointerLock?.();
                } else {
                    document.body.requestPointerLock?.();
                }
            }
        }

        this.toggleMenu = function () {
            this.isOpen = !this.isOpen;
            toggleMenuVisibility(this.isOpen);
        };

        this.hideMenu = function () {
            this.isOpen = false;
            toggleMenuVisibility(false);
        };

        this.showMenu = function () {
            this.isOpen = true;
            toggleMenuVisibility(true);
        };


        function isHotkeyPressed(hotkey, event) {
            return event.keyCode === hotkey.keyCode && event.ctrlKey === hotkey.ctrlKey && event.altKey === hotkey.altKey && event.shiftKey === hotkey.shiftKey;
        }

        this.addTab = function (tab) {
            this.tabs.push(tab);
        };
    }

    function Tab(label) {
        this.label = label;
        this.content = [];

        this.addButton = function (button) {
            this.content.push({ type: 'button', data: button });
        };

        this.addSwitch = function (switchButton) {
            this.content.push({ type: 'switch', data: switchButton });
        };

        this.addInput = function (input) {
            this.content.push({ type: 'input', data: input });
        };

        this.addHotkey = function (hotkey) {
            this.content.push({ type: 'hotkey', data: hotkey });
        };

        this.addSlider = function (hotkey) {
            this.content.push({ type: 'slider', data: hotkey });
        };

        this.addColorPicker = function (hotkey) {
            this.content.push({ type: 'colorPicker', data: hotkey });
        };

        this.renderContent = function (container) {
            this.content.forEach(function (item) {
                var row = document.createElement('div');
                row.className = classList['row'];

                var textElement = document.createElement('span');
                textElement.className = classList['label'];
                textElement.textContent = item.data.label || '';
                row.appendChild(textElement);

                switch (item.type) {
                    case 'button':
                        var buttonElement = document.createElement('button');
                        buttonElement.textContent = item.data.buttonLabel || '';
                        buttonElement.className = classList['button'];
                        if (item.data.style === "border") {
                            buttonElement.classList.add(classList['white-border']);
                        } else if (item.data.style === "rgb") {
                            buttonElement.classList.add('rgb-style');
                        }
                        buttonElement.addEventListener('click', function () {
                            if (item.data.onclick) {
                                item.data.onclick();
                            }
                        });
                        row.appendChild(buttonElement);
                        break;

                    case 'switch':
                        var switchContainer = document.createElement('div');
                        switchContainer.className = classList['switch-container'];

                        var switchElement = document.createElement('input');
                        switchElement.type = 'checkbox';
                        switchElement.checked = item.data.value || false;
                        switchElement.addEventListener('change', function () {
                            item.data.value = switchElement.checked;
                            if (item.data.onchange) {
                                item.data.onchange(switchElement.checked);
                            }
                        });

                        var switchLabel = document.createElement('label');
                        switchLabel.className = classList['switch-label'];

                        var innerCircle = document.createElement('div');
                        innerCircle.className = classList['inner-circle'];

                        switchLabel.addEventListener('click', function () {
                            switchElement.checked = !switchElement.checked;
                            item.data.value = switchElement.checked;
                            if (item.data.onchange) {
                                item.data.onchange(switchElement.checked);
                            }
                        });

                        switchLabel.appendChild(innerCircle);
                        switchContainer.appendChild(switchElement);
                        switchContainer.appendChild(switchLabel);
                        row.appendChild(switchContainer);
                        break;

                    case 'input':
                        var inputElement = document.createElement('input');
                        inputElement.className = classList['input'];
                        inputElement.placeholder = item.data.placeholder || '';
                        inputElement.type = item.data.type || 'text';
                        inputElement.value = item.data.type === 'number' ? (item.data.value || 0) : (item.data.value || '');

                        if (item.data.onchange) {
                            inputElement.addEventListener('change', function () {
                                item.data.value = inputElement.value;
                                item.data.onchange(inputElement.value);
                            });
                        }

                        row.appendChild(inputElement);
                        break;

                    case 'hotkey':
                        var buttonElement = document.createElement('button');
                        buttonElement.className = classList['button'];
                        buttonElement.textContent = item.data.value ? formatHotkey(item.data.value) : translate('setHotkey');

                        if (item.data.style === "border") {
                            buttonElement.classList.add(classList['white-border']);
                        } else if (item.data.style === "rgb") {
                            buttonElement.classList.add('rgb-style');
                        }

                        buttonElement.addEventListener('click', function () {
                            buttonElement.textContent = translate('pressKey');
                            document.addEventListener('keyup', function (event) {
                                event.preventDefault();
                                if (document.activeElement.value !== undefined && document.activeElement !== buttonElement) return;
                                item.data.value = event;
                                buttonElement.textContent = formatHotkey(event);
                                if (item.data.onlistener) {
                                    item.data.onlistener(event);
                                }
                                document.removeEventListener('keyup', arguments.callee);
                            });
                        });

                        row.appendChild(buttonElement);
                        break;

                    case 'slider':
                        var sliderContainer = document.createElement('div');
                        sliderContainer.className = classList['slider-container'];

                        var sliderElement = document.createElement('input');
                        sliderElement.type = 'range';
                        sliderElement.min = item.data.min || 0;
                        sliderElement.max = item.data.max || 100;
                        sliderElement.value = item.data.value || 0;
                        sliderElement.style.background = 'linear-gradient(to right, #fff ' + sliderElement.value + '%, rgba(255, 255, 255, 0.10) ' + sliderElement.value + '%, rgba(255, 255, 255, 0.10) 100%)';

                        var valueElement = document.createElement('input');
                        valueElement.type = 'number';
                        valueElement.className = classList['input'];
                        valueElement.min = item.data.min || 0;
                        valueElement.max = item.data.max || 100;
                        valueElement.value = item.data.value || 0;

                        sliderElement.addEventListener('input', function () {
                            item.data.value = sliderElement.value;
                            valueElement.value = sliderElement.value;
                            sliderElement.style.background = 'linear-gradient(to right, #fff ' + sliderElement.value + '%, rgba(255, 255, 255, 0.10) ' + sliderElement.value + '%, rgba(255, 255, 255, 0.10) 100%)';
                        });

                        sliderElement.addEventListener('change', function () {
                            if (item.data.onchange) {
                                item.data.onchange(sliderElement.value);
                            }
                        });

                        valueElement.addEventListener('input', function () {
                            sliderElement.value = valueElement.value;
                            sliderElement.style.background = 'linear-gradient(to right, #fff ' + sliderElement.value + '%, rgba(255, 255, 255, 0.10) ' + sliderElement.value + '%, rgba(255, 255, 255, 0.10) 100%)';
                        });

                        valueElement.addEventListener('change', function () {
                            if (parseFloat(valueElement.value) < parseFloat(sliderElement.min) || parseFloat(valueElement.value) > parseFloat(sliderElement.max)) {
                                valueElement.value = item.data.value;
                                sliderElement.value = item.data.value;
                                sliderElement.style.background = 'linear-gradient(to right, #fff ' + item.data.value + '%, rgba(255, 255, 255, 0.10) ' + item.data.value + '%, rgba(255, 255, 255, 0.10) 100%)';
                            } else if (!valueElement.value) {
                                valueElement.value = sliderElement.min;
                                sliderElement.value = sliderElement.min;
                                sliderElement.style.background = 'linear-gradient(to right, #fff ' + sliderElement.min + '%, rgba(255, 255, 255, 0.10) ' + sliderElement.min + '%, rgba(255, 255, 255, 0.10) 100%)';
                            } else {
                                item.data.value = valueElement.value;
                            }
                            if (item.data.onchange) {
                                item.data.onchange(valueElement.value);
                            }
                        });

                        sliderContainer.appendChild(sliderElement);
                        sliderContainer.appendChild(valueElement);
                        row.appendChild(sliderContainer);
                        break;

                    case 'colorPicker':
                        var colorPickerContainer = document.createElement('div');
                        colorPickerContainer.className = classList['color-picker-container'];

                        var colorPickerElement = document.createElement('input');
                        colorPickerElement.type = 'color';
                        colorPickerElement.className = classList['color'];
                        colorPickerElement.value = item.data.value || '#000000';

                        var colorCodeElement = document.createElement('input');
                        colorCodeElement.type = 'text';
                        colorCodeElement.className = classList['input'];
                        colorCodeElement.value = item.data.value || '#000000';

                        var circularArea = document.createElement('div');
                        circularArea.className = classList['circular-area'];
                        circularArea.style.backgroundColor = colorPickerElement.value;

                        colorPickerElement.addEventListener('input', function () {
                            item.data.value = colorPickerElement.value;
                            colorCodeElement.value = colorPickerElement.value;
                            circularArea.style.backgroundColor = colorPickerElement.value;
                        });

                        colorPickerElement.addEventListener('change', function () {
                            if (item.data.onchange) {
                                item.data.onchange(colorPickerElement.value);
                            }
                        });

                        colorCodeElement.addEventListener('input', function () {
                            item.data.value = colorCodeElement.value;
                            colorPickerElement.value = colorCodeElement.value;
                            circularArea.style.backgroundColor = colorCodeElement.value;
                        });

                        colorCodeElement.addEventListener('change', function () {
                            if (/^#([0-9a-fA-F]{3}|[0-9a-fA-F]{6})$/.test(colorPickerElement.value)) colorCodeElement.value =
                                colorPickerElement.value;
                            if (item.data.onchange) {
                                item.data.onchange(colorCodeElement.value);
                            }
                        });

                        circularArea.addEventListener('click', function () {
                            colorPickerElement.click();
                        });

                        colorPickerContainer.appendChild(colorPickerElement);
                        colorPickerContainer.appendChild(circularArea);
                        colorPickerContainer.appendChild(colorCodeElement);
                        row.appendChild(colorPickerContainer);
                        break;

                    default:
                        break;
                }

                container.appendChild(row);
            });
        };

        function formatHotkey(event) {
            var modifiers = [];
            if (event.ctrlKey) modifiers.push('Ctrl');
            if (event.altKey) modifiers.push('Alt');
            if (event.shiftKey) modifiers.push('Shift');
            var key = event.key || String.fromCharCode(event.keyCode);

            return modifiers.length > 0 ? modifiers.join('+') + ' + ' + key : key;
        }
    }

    function ToastManager(options) {
        if (options && options.position) this.position = `${classList[options.position.split('-')[0]]}-${classList[options.position.split('-')[1]]}`
        else this.position = `${classList['bottom']}-${classList['right']}`;

        this.toastContainer = document.createElement('div');
        this.toastContainer.className = classList['toast-container'];
        this.toastContainer.dataset[classList['position']] = this.position;
        document.body.appendChild(this.toastContainer);

        this.showToast = function ({ message, duration = 3000, type, showProgress = false }) {
            var toast = document.createElement('div');
            toast.className = `${classList['toast']} ${classList[type]}`;
            toast.classList.toggle(classList['progress'], showProgress);
            toast.style.setProperty("--progress", 1);
            toast.textContent = message;

            var closeButton = document.createElement('button');
            closeButton.className = classList['close-button'];
            var closeIconSvg = `
        <svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" stroke="white" stroke-width="2">
            <line x1="18" y1="6" x2="6" y2="18" />
            <line x1="6" y1="6" x2="18" y2="18" />
        </svg>
        `;
            closeButton.innerHTML = closeIconSvg;
            closeButton.addEventListener('click', function () {
                this.removeToast(toast);
            }.bind(this));

            toast.appendChild(closeButton);

            toast.addEventListener('mouseover', function () {
                this.handleMouseOver(toast);
            }.bind(this));

            toast.addEventListener('mouseout', function () {
                this.handleMouseOut(toast, duration);
            }.bind(this));

            this.toastContainer[options.position.split('-')[0] === 'top' ? 'prepend' : 'appendChild'](toast);

            this.handleMouseOut(toast, duration);

            return toast;
        };

        this.removeToast = function (toast) {
            toast.style.opacity = 0;
            setTimeout(function () {
                this.toastContainer.removeChild(toast);
            }.bind(this), 300);
        };

        this.handleMouseOver = function (toast) {
            clearTimeout(toast.timeoutId);
            clearInterval(toast.intervalId);
        };
        this.handleMouseOut = function (toast, duration) {
            let lastTimeCalled = new Date();
            toast.intervalId = setInterval(() => {
                var timeVisible = + new Date() - lastTimeCalled;
                toast.style.setProperty(
                    "--progress",
                    timeVisible / duration
                );
            }, 10);
            toast.timeoutId = setTimeout(function () {
                this.removeToast(toast);
            }.bind(this), duration);
        };
    }

    return {
        CreateMenu: CreateMenu,
        Tab: Tab,
        ToastManager: ToastManager,
        setLanguage: setLanguage
    };
})();