YM DuoMax Lite

XP + Gems + Real Streak farming tool for Duolingo with compact UI

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Userscripts ,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name         YM DuoMax Lite
// @namespace    http://tampermonkey.net/
// @version      v1.0.1_LITE
// @description  XP + Gems + Real Streak farming tool for Duolingo with compact UI
// @author       ´꒳`ⓎⒶⓂⒾⓈⒸⓇⒾⓅⓉ×͜×
// @match        https://www.duolingo.com/*
// @grant        none
// @run-at       document-idle
// @icon         https://www.google.com/s2/favicons?sz=64&domain=duolingo.com
// ==/UserScript==

(function () {
    'use strict';

    const SERVER_ID = '1377275722342858973';
    const WIDGET_URL = `https://discord.com/widget?id=${SERVER_ID}&theme=dark`;
    const VERSION = 'v1.0.1_LITE';
    let lang = 'en';

    const LANG = {
       en: {
            header: 'DuoMax LITE', farmLabel: 'FARM ENGINE', start: '🚀 START',
            done: 'Farm finished', copied: 'Token copied!', placeholder: 'Enter target amount',
            settings: 'Settings', support: 'Support', profile: 'Profile', discord: 'Discord',
            version: 'Version', madeby: 'Made by'
        }
    };
    const t = k => LANG[lang][k];

    const ICONS = {
        xp: "",
        gem: "",
        streak: "",
        quest: "",
        league: "",
        close: "",
        minimize: "",
        settings: "",
        discord: ""
    };

    // Settings state
    const settings = {
        animations: true,
        debugMode: false,
        autoClearLogs: true
    };

    // Load saved settings
    try {
        const saved = localStorage.getItem('duomax_settings');
        if (saved) {
            Object.assign(settings, JSON.parse(saved));
        }
    } catch (e) {
        console.log('No saved settings found');
    }

    const style = document.createElement('style');
    style.innerHTML = `
        #duocheat-border-box {
            position: fixed;
            top: 50%;
            left: 50%;
            transform: translate(-50%, -50%);
            width: 580px;
            padding: 2px;
            border-radius: 16px;
            z-index: 99999;
            overflow: hidden;
            display: flex;
            align-items: center;
            justify-content: center;
            box-shadow: 0 12px 36px rgba(0,0,0,0.6);
            transition: all 0.3s ease;
            background: linear-gradient(135deg, #3b82f6, #2dd4bf, #ef4444);
        }
        #duocheat-border-box.animations-off {
            background: #1e293b !important;
        }
        #duocheat-border-box.minimized {
            width: 140px;
            height: 36px;
            top: 12px;
            left: 12px;
            transform: translate(0, 0);
            border-radius: 8px;
        }
        #duocheat-border-box.minimized .hide-on-min {
            display: none !important;
        }
        #duocheat-container {
            position: relative;
            width: 100%;
            height: 100%;
            border-radius: 14px;
            padding: 16px;
            font-family: 'Inter', 'Segoe UI', sans-serif;
            z-index: 2;
            color: #f1f5f9;
            background: #0a0f1c;
            overflow: hidden;
        }
        #duocheat-border-box.minimized #duocheat-container {
            padding: 6px 10px;
            border-radius: 6px;
        }
        #motion-canvas-bg, #border-canvas-engine {
            position: absolute;
            top: 0;
            left: 0;
            width: 100%;
            height: 100%;
            z-index: 1;
            pointer-events: none;
            transition: opacity 0.3s;
        }
        .animations-off #motion-canvas-bg,
        .animations-off #border-canvas-engine {
            opacity: 0 !important;
        }
        .header-top {
            display: flex;
            justify-content: space-between;
            align-items: center;
            margin-bottom: 10px;
            position: relative;
            z-index: 3;
            border-bottom: 1px solid rgba(255,255,255,0.1);
            padding-bottom: 8px;
        }
        #duocheat-border-box.minimized .header-top {
            margin-bottom: 0;
            border: none;
            padding-bottom: 0;
        }
        .logo-text {
            font-size: 16px;
            font-weight: 900;
            background: linear-gradient(90deg, #3b82f6, #2dd4bf);
            -webkit-background-clip: text;
            -webkit-text-fill-color: transparent;
        }
        #duocheat-border-box.minimized .logo-text {
            font-size: 12px;
        }
        .nav-controls {
            display: flex;
            gap: 6px;
            align-items: center;
        }
        .nav-icon {
            width: 14px;
            height: 14px;
            cursor: pointer;
            opacity: 0.6;
            transition: 0.2s;
        }
        .nav-icon:hover {
            opacity: 1;
            transform: scale(1.1);
        }
        .user-profile {
            display: flex;
            align-items: center;
            gap: 8px;
            background: rgba(255, 255, 255, 0.05);
            padding: 6px;
            border-radius: 12px;
            margin-bottom: 10px;
            border: 1px solid rgba(255, 255, 255, 0.1);
            position: relative;
            z-index: 3;
        }
        .user-avatar {
            width: 32px;
            height: 32px;
            border-radius: 50%;
            border: 1px solid #3b82f6;
            object-fit: cover;
            background: #1e293b;
            display: flex;
            align-items: center;
            justify-content: center;
            font-weight: bold;
            font-size: 12px;
            color: white;
        }
        .user-info {
            display: flex;
            flex-direction: column;
            flex: 1;
        }
        .user-info b {
            font-size: 11px;
            color: #f1f5f9;
            margin-bottom: 1px;
        }
        .user-info span {
            font-size: 9px;
            color: #94a3b8;
        }
        .stats-wrapper {
            display: grid;
            grid-template-columns: repeat(3, 1fr);
            gap: 8px;
            margin-bottom: 10px;
            position: relative;
            z-index: 3;
        }
        .stat-card-new {
            background: rgba(30, 41, 59, 0.6);
            border: 1px solid rgba(255,255,255,0.1);
            padding: 8px;
            border-radius: 10px;
            text-align: center;
            display: flex;
            flex-direction: column;
            align-items: center;
            gap: 3px;
        }
        .icon-small {
            width: 20px;
            height: 20px;
        }
        .stat-val {
            font-size: 14px;
            font-weight: 800;
        }
        .stat-lbl {
            font-size: 6px;
            text-transform: uppercase;
            color: #94a3b8;
            letter-spacing: 0.5px;
        }
        .section-title {
            font-size: 8px;
            font-weight: 700;
            color: #3b82f6;
            text-transform: uppercase;
            margin: 8px 0 4px;
            position: relative;
            z-index: 3;
            display: flex;
            align-items: center;
        }
        .section-title::after {
            content: "";
            flex: 1;
            height: 1px;
            background: rgba(59, 130, 246, 0.3);
            margin-left: 8px;
        }
        .mode-selection, .grid-options {
            display: grid;
            grid-template-columns: repeat(2, 1fr);
            gap: 4px;
            position: relative;
            z-index: 3;
            margin-bottom: 8px;
        }
        .grid-options {
            grid-template-columns: repeat(3, 1fr);
        }
        .mode-btn, .option-item {
            display: flex;
            align-items: center;
            justify-content: center;
            gap: 3px;
            padding: 6px;
            background: rgba(51, 65, 85, 0.4);
            border: 1px solid rgba(255,255,255,0.1);
            border-radius: 8px;
            cursor: pointer;
            font-size: 9px;
            font-weight: 600;
            transition: 0.3s;
            color: #cbd5e1;
        }
        .mode-btn.active {
            border-color: #3b82f6;
            background: rgba(59, 130, 246, 0.2);
            color: #60a5fa;
        }
        .option-item.active {
            border-color: #2dd4bf;
            color: #2dd4bf;
            background: rgba(45, 212, 191, 0.15);
        }
        .amount-input-section {
            background: rgba(15, 23, 42, 0.8);
            border: 1px solid rgba(59, 130, 246, 0.3);
            border-radius: 10px;
            padding: 8px 10px;
            margin-bottom: 8px;
            position: relative;
            z-index: 3;
        }
        .streak-info {
            display: flex;
            justify-content: space-between;
            align-items: center;
            margin-bottom: 3px;
        }
        .streak-current {
            font-size: 8px;
            color: #94a3b8;
        }
        .streak-target {
            font-size: 8px;
            color: #3b82f6;
            font-weight: bold;
        }
        .amount-label {
            font-size: 8px;
            font-weight: 700;
            color: #3b82f6;
            text-transform: uppercase;
            margin-bottom: 3px;
            display: block;
        }
        .amount-input {
            width: 100%;
            padding: 6px;
            background: rgba(30, 41, 59, 0.6);
            border: 1px solid rgba(59, 130, 246, 0.3);
            border-radius: 6px;
            color: #f1f5f9;
            font-size: 11px;
            font-weight: 600;
            text-align: center;
            outline: none;
            transition: 0.2s;
        }
        .amount-input:focus {
            border-color: #3b82f6;
            box-shadow: 0 0 0 1px rgba(59, 130, 246, 0.2);
        }
        .log-container {
            background: rgba(0, 0, 0, 0.3);
            border: 1px solid rgba(255, 255, 255, 0.1);
            border-radius: 8px;
            padding: 6px;
            margin: 6px 0;
            position: relative;
            z-index: 3;
        }
        .log-header {
            display: flex;
            justify-content: space-between;
            align-items: center;
            margin-bottom: 4px;
        }
        .log-title {
            font-size: 8px;
            font-weight: 700;
            color: #3b82f6;
            text-transform: uppercase;
        }
        .log-status {
            font-size: 8px;
            color: #94a3b8;
        }
        .log-box {
            max-height: 60px;
            overflow-y: auto;
            font-size: 8px;
            font-family: monospace;
            color: #94a3b8;
            background: rgba(0, 0, 0, 0.2);
            border-radius: 4px;
            padding: 4px;
        }
        .log-entry {
            margin: 1px 0;
            padding: 2px 4px;
            border-radius: 3px;
            font-size: 8px;
            line-height: 1.1;
        }
        .log-success {
            background: rgba(34, 197, 94, 0.15);
            color: #86efac;
            border-left: 2px solid #4ade80;
        }
        .log-error {
            background: rgba(239, 68, 68, 0.15);
            color: #fca5a5;
            border-left: 2px solid #f87171;
        }
        .log-info {
            background: rgba(59, 130, 246, 0.15);
            color: #93c5fd;
            border-left: 2px solid #3b82f6;
        }
        .log-warning {
            background: rgba(245, 158, 11, 0.15);
            color: #fcd34d;
            border-left: 2px solid #f59e0b;
        }
        .btn-test {
            margin-top: 6px;
            padding: 5px;
            background: rgba(59, 130, 246, 0.2);
            border: 1px solid #3b82f6;
            border-radius: 5px;
            color: #60a5fa;
            cursor: pointer;
            font-size: 9px;
            width: 100%;
            font-weight: 600;
        }
        .farm-mode-indicator {
            background: rgba(45, 212, 191, 0.1);
            border: 1px solid #2dd4bf;
            border-radius: 5px;
            padding: 4px 6px;
            margin: 4px 0;
            font-size: 8px;
            color: #2dd4bf;
            text-align: center;
            font-weight: bold;
        }
        .data-panel {
            background: rgba(15, 23, 42, 0.8);
            border: 1px solid rgba(59, 130, 246, 0.3);
            border-radius: 10px;
            padding: 6px 8px;
            margin-bottom: 8px;
            display: flex;
            justify-content: space-between;
            position: relative;
            z-index: 3;
        }
        .data-item {
            display: flex;
            flex-direction: column;
            align-items: center;
            flex: 1;
        }
        .data-item span {
            font-size: 6px;
            color: #64748b;
            margin-bottom: 1px;
        }
        .data-item b {
            font-size: 10px;
        }
        .progress-bar-container {
            height: 3px;
            background: rgba(30, 41, 59, 0.6);
            border-radius: 1px;
            overflow: hidden;
            margin-top: 4px;
            position: relative;
            z-index: 3;
        }
        .progress-bar {
            height: 100%;
            background: linear-gradient(90deg, #3b82f6, #2dd4bf);
            width: 0%;
            transition: width 0.3s ease;
        }
        .progress-text {
            font-size: 9px;
            color: #94a3b8;
            margin-top: 1px;
            text-align: center;
            position: relative;
            z-index: 3;
        }
        .btn-main-farm {
            width: 100%;
            padding: 10px;
            background: linear-gradient(135deg, #3b82f6, #2563eb);
            border: none;
            border-radius: 10px;
            color: white;
            font-weight: 800;
            font-size: 12px;
            cursor: pointer;
            text-transform: uppercase;
            letter-spacing: 0.5px;
            position: relative;
            z-index: 3;
            transition: 0.3s;
            margin-top: 6px;
        }
        .btn-main-farm.farming {
            background: linear-gradient(135deg, #ef4444, #b91c1c);
            animation: pulse 1.5s infinite;
        }
        .animations-off .btn-main-farm.farming {
            animation: none;
        }
        @keyframes pulse {
            0% { opacity: 1; }
            50% { opacity: 0.8; }
            100% { opacity: 1; }
        }
        .external-buttons {
            position: fixed;
            bottom: 10px;
            right: 10px;
            display: flex;
            flex-direction: column;
            gap: 5px;
            z-index: 99998;
        }
        .external-btn {
            width: 36px;
            height: 36px;
            border-radius: 50%;
            display: flex;
            align-items: center;
            justify-content: center;
            font-size: 14px;
            cursor: pointer;
            box-shadow: 0 3px 8px rgba(0,0,0,.3);
            transition: 0.3s;
            border: none;
        }
        .discord-btn {
            background: #5865F2;
            color: white;
        }
        .notify-btn {
            background: #ffb703;
            color: #000;
        }
        .toggle-btn {
            background: linear-gradient(135deg, #58cc02, #1cb0f6);
            color: white;
            font-size: 16px;
        }
        .version-tag {
            position: fixed;
            bottom: 50px;
            right: 16px;
            background: rgba(0,0,0,.5);
            color: #fff;
            font-size: 7px;
            padding: 1px 3px;
            border-radius: 3px;
            z-index: 10000;
            font-family: monospace;
        }

        /* Settings Panel */
        .settings-panel {
            position: fixed;
            top: 50%;
            left: 50%;
            transform: translate(-50%, -50%);
            width: 320px;
            background: #0a0f1c;
            border-radius: 12px;
            padding: 16px;
            z-index: 100000;
            box-shadow: 0 10px 30px rgba(0,0,0,0.8);
            border: 1px solid rgba(59, 130, 246, 0.3);
            display: none;
        }
        .settings-panel.active {
            display: block;
        }
        .settings-header {
            display: flex;
            justify-content: space-between;
            align-items: center;
            margin-bottom: 12px;
            padding-bottom: 8px;
            border-bottom: 1px solid rgba(255,255,255,0.1);
        }
        .settings-title {
            font-size: 14px;
            font-weight: 800;
            color: #3b82f6;
        }
        .settings-close {
            width: 14px;
            height: 14px;
            cursor: pointer;
            opacity: 0.6;
            transition: 0.2s;
        }
        .settings-close:hover {
            opacity: 1;
            transform: scale(1.1);
        }
        .settings-group {
            margin-bottom: 12px;
        }
        .settings-label {
            font-size: 10px;
            font-weight: 700;
            color: #94a3b8;
            margin-bottom: 6px;
            text-transform: uppercase;
            letter-spacing: 0.5px;
        }
        .settings-option {
            display: flex;
            align-items: center;
            justify-content: space-between;
            padding: 8px 10px;
            background: rgba(30, 41, 59, 0.6);
            border: 1px solid rgba(255,255,255,0.1);
            border-radius: 8px;
            margin-bottom: 6px;
            cursor: pointer;
            transition: 0.2s;
        }
        .settings-option:hover {
            background: rgba(51, 65, 85, 0.8);
        }
        .settings-option-label {
            font-size: 10px;
            color: #cbd5e1;
            display: flex;
            align-items: center;
            gap: 6px;
        }
        .settings-option-label img {
            width: 12px;
            height: 12px;
        }
        .settings-toggle {
            width: 24px;
            height: 12px;
            background: rgba(100, 116, 139, 0.5);
            border-radius: 6px;
            position: relative;
            cursor: pointer;
            transition: 0.3s;
        }
        .settings-toggle.active {
            background: #3b82f6;
        }
        .settings-toggle::after {
            content: '';
            position: absolute;
            width: 10px;
            height: 10px;
            background: white;
            border-radius: 50%;
            top: 1px;
            left: 1px;
            transition: 0.3s;
        }
        .settings-toggle.active::after {
            left: 13px;
        }
        .settings-btn {
            width: 100%;
            padding: 8px;
            background: rgba(59, 130, 246, 0.2);
            border: 1px solid #3b82f6;
            border-radius: 6px;
            color: #60a5fa;
            cursor: pointer;
            font-size: 10px;
            font-weight: 600;
            text-align: center;
            margin-top: 8px;
            transition: 0.2s;
        }
        .settings-btn:hover {
            background: rgba(59, 130, 246, 0.3);
        }
        .settings-btn.danger {
            background: rgba(239, 68, 68, 0.2);
            border-color: #ef4444;
            color: #fca5a5;
        }
        .settings-btn.danger:hover {
            background: rgba(239, 68, 68, 0.3);
        }
        .settings-btn.success {
            background: rgba(34, 197, 94, 0.2);
            border-color: #22c55e;
            color: #86efac;
        }
        .settings-btn.success:hover {
            background: rgba(34, 197, 94, 0.3);
        }
        .settings-overlay {
            position: fixed;
            top: 0;
            left: 0;
            width: 100%;
            height: 100%;
            background: rgba(0, 0, 0, 0.7);
            z-index: 99999;
            display: none;
        }
        .settings-overlay.active {
            display: block;
        }
    `;
    document.head.appendChild(style);

    // === MAIN UI CONTAINER ===
    const borderBox = document.createElement('div');
    borderBox.id = 'duocheat-border-box';

    const borderCanvas = document.createElement('canvas');
    borderCanvas.id = 'border-canvas-engine';

    const container = document.createElement('div');
    container.id = 'duocheat-container';
    container.innerHTML = `
        <canvas id="motion-canvas-bg"></canvas>
        <div class="header-top">
            <div class="logo-text" id="logo-main">DuoMax LITE<span style="font-size:8px; color:#64748b" class="hide-on-min">${VERSION}</span></div>
            <div class="nav-controls">
                <img src="${ICONS.settings}" class="nav-icon" id="btn-settings" title="Settings">
                <img src="${ICONS.minimize}" class="nav-icon" id="btn-minimize-toggle" title="Minimize/Maximize">
                <img src="${ICONS.close}" class="nav-icon" id="btn-close-ui" title="Close">
            </div>
        </div>
        <div class="hide-on-min">
            <div id="user-display" class="user-profile">
                <div class="user-avatar"></div>
                <div class="user-info">
                    <b>Loading User...</b>
                    <span id="user-status">Checking login...</span>
                </div>
            </div>
            <div class="stats-wrapper">
                <div class="stat-card-new">
                    <img src="${ICONS.xp}" class="icon-small">
                    <span class="stat-val" id="st-xp">0</span>
                    <span class="stat-lbl">⚡XP</span>
                </div>
                <div class="stat-card-new">
                    <img src="${ICONS.gem}" class="icon-small">
                    <span class="stat-val" style="color:#0ea5e9" id="st-gems">0</span>
                    <span class="stat-lbl">GEMS</span>
                </div>
                <div class="stat-card-new">
                    <img src="${ICONS.streak}" class="icon-small">
                    <span class="stat-val" style="color:#ff9600" id="st-streak">0</span>
                    <span class="stat-lbl">🔥STREAK</span>
                </div>
            </div>

            <div class="section-title">FARM ENGINE MODE</div>
            <div class="mode-selection">
                <div class="mode-btn active" id="sp-safe">🛡️ SAFE</div>
                <div class="mode-btn" id="sp-fast">⚡ FAST</div>
            </div>

            <div class="section-title">TARGET MODULES</div>
            <div class="grid-options">
                <div class="option-item active" id="opt-xp">
                    <img src="${ICONS.xp}" style="width:10px; height:10px"> XP
                </div>
                <div class="option-item" id="opt-gems">
                    <img src="${ICONS.gem}" style="width:10px; height:10px"> Gems
                </div>
                <div class="option-item" id="opt-streak">
                    <img src="${ICONS.streak}" style="width:10px; height:10px"> Streak
                </div>
                <div class="option-item" id="opt-league">
                    <img src="${ICONS.league}" style="width:10px; height:10px"> League
                </div>
                <div class="option-item" id="opt-quest">
                    <img src="${ICONS.quest}" style="width:10px; height:10px"> Quests
                </div>
                <div class="option-item" id="opt-all">🔥 All-In</div>
            </div>

            <div class="amount-input-section">
                <div class="streak-info">
                    <span class="streak-current" id="current-streak-display">Current: 0 days</span>
                    <span class="streak-target" id="target-streak-display">Target: 0 days</span>
                </div>
                <span class="amount-label">🎯 AMOUNT TO FARM</span>
                <input type="number" id="targetValue" class="amount-input" placeholder="e.g., 1000 XP" value="1000" min="1" max="999999">
            </div>

            <button class="btn-test" id="btn-test-api">Test API Connection</button>

            <div class="log-container">
                <div class="log-header">
                    <div class="log-title">ACTIVITY LOG</div>
                    <div class="log-status" id="log-status">Last: System started</div>
                </div>
                <div class="log-box" id="log-box">
                    <div class="log-entry log-info">System initialized. Ready to farm.</div>
                </div>
            </div>

            <div class="farm-mode-indicator" id="farm-mode-indicator">
                🛡️ SAFE MODE ACTIVE | DELAY: 3s
            </div>

            <div class="section-title">FARMING STATUS</div>
            <div class="data-panel">
                <div class="data-item">
                    <span>RUNTIME</span>
                    <b id="data-time">00:00:00</b>
                </div>
                <div class="data-item">
                    <span>STREAK DAYS</span>
                    <b id="data-streak-days">0</b>
                </div>
                <div class="data-item">
                    <span>STATUS</span>
                    <b id="data-status" style="color:#ef4444">IDLE</b>
                </div>
            </div>
            <div class="progress-bar-container">
                <div id="progressBar" class="progress-bar"></div>
            </div>
            <div class="progress-text" id="progressText">❌ Not started</div>
            <button class="btn-main-farm" id="btn-master-farm">${t('start')}</button>
        </div>
    `;

    borderBox.appendChild(borderCanvas);
    borderBox.appendChild(container);
    document.body.appendChild(borderBox);

    // === SETTINGS PANEL ===
    const settingsOverlay = document.createElement('div');
    settingsOverlay.className = 'settings-overlay';

    const settingsPanel = document.createElement('div');
    settingsPanel.className = 'settings-panel';
    settingsPanel.innerHTML = `
        <div class="settings-header">
            <div class="settings-title">⚙️ SETTINGS</div>
            <img src="${ICONS.close}" class="settings-close" id="settings-close" title="Close">
        </div>
        <div class="settings-group">
            <div class="settings-label">VISUAL SETTINGS</div>
            <div class="settings-option" id="toggle-animations">
                <div class="settings-option-label">
                    <span>🎨 Animations</span>
                </div>
                <div class="settings-toggle ${settings.animations ? 'active' : ''}"></div>
            </div>
            <div class="settings-option" id="toggle-debug">
                <div class="settings-option-label">
                    <span>🐛 Debug Mode</span>
                </div>
                <div class="settings-toggle ${settings.debugMode ? 'active' : ''}"></div>
            </div>
            <div class="settings-option" id="toggle-autoclear">
                <div class="settings-option-label">
                    <span>🧹 Auto-clear Logs</span>
                </div>
                <div class="settings-toggle ${settings.autoClearLogs ? 'active' : ''}"></div>
            </div>
        </div>
        <div class="settings-group">
            <div class="settings-label">MAINTENANCE</div>
            <button class="settings-btn" id="clear-logs-btn">
                🗑️ Clear Activity Logs
            </button>
            <button class="settings-btn" id="clear-cache-btn">
                🧹 Clear UI Cache
            </button>
            <button class="settings-btn" id="reset-settings-btn">
                🔄 Reset to Default
            </button>
            <button class="settings-btn danger" id="refresh-ui-btn">
                ♻️ Refresh UI
            </button>
            <button class="settings-btn success" id="export-settings-btn">
                📤 Export Settings
            </button>
        </div>
        <div class="settings-group">
            <div class="settings-label">INFORMATION</div>
            <div style="font-size:9px; color:#94a3b8; padding:6px 10px; background:rgba(30,41,59,0.6); border-radius:6px; margin-bottom:8px;">
                <strong>Version:</strong> ${VERSION}<br>
                <strong>UI Size:</strong> Compact (580px)<br>
                <strong>Animations:</strong> ${settings.animations ? 'ON' : 'OFF'}
            </div>
        </div>
    `;

    document.body.appendChild(settingsOverlay);
    document.body.appendChild(settingsPanel);

    // === EXTERNAL BUTTONS ===
    const externalButtons = document.createElement('div');
    externalButtons.className = 'external-buttons';
    externalButtons.innerHTML = `
        <button class="external-btn discord-btn" id="external-discord" title="Discord Chat">💬</button>
        <button class="external-btn notify-btn" id="external-notify" title="Notifications">🔔</button>
        <button class="external-btn toggle-btn" id="external-toggle" title="Toggle Panel">⚙️</button>
    `;
    document.body.appendChild(externalButtons);

    const versionTag = Object.assign(document.createElement('div'), {
        className: 'version-tag',
        textContent: VERSION
    });
    document.body.appendChild(versionTag);

    // === DISCORD CHAT IFRAME ===
    const discordChat = document.createElement('div');
    discordChat.className = 'duo-discord-chat';
    discordChat.style.cssText = 'position:fixed;bottom:70px;right:12px;width:280px;height:350px;background:#fff;border:2px solid #5865F2;border-radius:6px;overflow:hidden;z-index:99997;display:none;box-shadow:0 4px 10px rgba(0,0,0,.3);';

    const iframe = document.createElement('iframe');
    iframe.src = WIDGET_URL;
    iframe.style.cssText = 'border:none;width:100%;height:100%;';
    iframe.setAttribute('allowtransparency', 'true');
    iframe.setAttribute('frameborder', '0');
    iframe.setAttribute('sandbox', 'allow-popups allow-popups-to-escape-sandbox allow-same-origin allow-scripts');
    discordChat.appendChild(iframe);
    document.body.appendChild(discordChat);

    // === ANIMATION ===
    const ctxB = borderCanvas.getContext('2d');
    const canvasM = document.getElementById('motion-canvas-bg');
    const ctxM = canvasM.getContext('2d');
    let angle = 0;
    let animationId = null;

    function animate() {
        if (!document.getElementById('duocheat-border-box')) return;
        borderCanvas.width = borderBox.offsetWidth;
        borderCanvas.height = borderBox.offsetHeight;
        ctxB.clearRect(0, 0, borderCanvas.width, borderCanvas.height);
        ctxB.translate(borderCanvas.width/2, borderCanvas.height/2);
        ctxB.rotate(angle);
        const grad = ctxB.createConicGradient(0, 0, 0);
        grad.addColorStop(0, '#3b82f6');
        grad.addColorStop(0.2, '#2dd4bf');
        grad.addColorStop(0.5, '#ef4444');
        grad.addColorStop(1, '#3b82f6');
        ctxB.fillStyle = grad;
        ctxB.fillRect(-borderCanvas.width, -borderCanvas.height, borderCanvas.width*2, borderCanvas.height*2);
        ctxB.resetTransform();

        canvasM.width = container.offsetWidth;
        canvasM.height = container.offsetHeight;
        ctxM.fillStyle = '#0a0f1c';
        ctxM.fillRect(0, 0, canvasM.width, canvasM.height);
        const radial = ctxM.createRadialGradient(canvasM.width/2, canvasM.height/2, 0, canvasM.width/2, canvasM.height/2, 240);
        radial.addColorStop(0, 'rgba(59, 130, 246, 0.1)');
        radial.addColorStop(1, 'transparent');
        ctxM.fillStyle = radial;
        ctxM.fillRect(0, 0, canvasM.width, canvasM.height);
        angle += 0.015;

        if (settings.animations) {
            animationId = requestAnimationFrame(animate);
        }
    }

    function startAnimation() {
        if (settings.animations && !animationId) {
            animate();
        }
    }

    function stopAnimation() {
        if (animationId) {
            cancelAnimationFrame(animationId);
            animationId = null;
        }
    }

    // Apply initial animation settings
    if (!settings.animations) {
        borderBox.classList.add('animations-off');
    } else {
        startAnimation();
    }

    // === LOG MANAGEMENT ===
    let logCount = 0;
    const MAX_LOGS = 15;

    function addLog(message, type = 'info') {
        const logBox = document.getElementById('log-box');
        const logStatus = document.getElementById('log-status');

        // Truncate message if too long
        const maxLength = 60;
        const displayMsg = message.length > maxLength ?
            message.substring(0, maxLength) + '...' : message;

        const logEntry = document.createElement('div');
        logEntry.className = `log-entry log-${type}`;
        logEntry.textContent = `[${new Date().toLocaleTimeString([], {hour: '2-digit', minute:'2-digit', second:'2-digit'})}] ${displayMsg}`;

        // Add tooltip with full message
        logEntry.title = message;

        logBox.appendChild(logEntry);
        logCount++;

        // Limit number of logs
        if (logCount > MAX_LOGS && settings.autoClearLogs) {
            logBox.removeChild(logBox.firstChild);
            logCount--;
        }

        logBox.scrollTop = logBox.scrollHeight;

        // Update status with last log
        logStatus.textContent = `Last: ${type.toUpperCase()}`;
        logStatus.style.color = {
            'success': '#4ade80',
            'error': '#f87171',
            'info': '#60a5fa',
            'warning': '#fbbf24'
        }[type] || '#94a3b8';

        if (settings.debugMode || type === 'error') {
            console.log(`[DuoMax LITE ${type.toUpperCase()}] ${message}`);
        }
    }

    // === SETTINGS FUNCTIONS ===
    function saveSettings() {
        try {
            localStorage.setItem('duomax_settings', JSON.stringify(settings));
            addLog('Settings saved', 'success');
        } catch (e) {
            addLog('Failed to save settings: ' + e.message, 'error');
        }
    }

    function toggleSetting(settingName) {
        settings[settingName] = !settings[settingName];
        saveSettings();

        // Apply setting changes
        switch(settingName) {
            case 'animations':
                if (settings.animations) {
                    borderBox.classList.remove('animations-off');
                    startAnimation();
                } else {
                    borderBox.classList.add('animations-off');
                    stopAnimation();
                }
                addLog(`Animations ${settings.animations ? 'enabled' : 'disabled'}`, 'info');
                break;

            case 'debugMode':
                addLog(`Debug mode ${settings.debugMode ? 'enabled' : 'disabled'}`, 'info');
                break;

            case 'autoClearLogs':
                addLog(`Auto-clear logs ${settings.autoClearLogs ? 'enabled' : 'disabled'}`, 'info');
                break;
        }
    }

    function clearLogs() {
        const logBox = document.getElementById('log-box');
        logBox.innerHTML = '<div class="log-entry log-info">Logs cleared</div>';
        logCount = 0;
        addLog('Activity logs cleared', 'success');
    }

    function clearCache() {
        try {
            // Clear various caches
            localStorage.removeItem('duomax_ui_state');
            sessionStorage.removeItem('duomax_temp');
            addLog('UI cache cleared', 'success');

            // Refresh some UI elements
            document.getElementById('log-status').textContent = 'Last: Cache cleared';

        } catch (e) {
            addLog('Failed to clear cache: ' + e.message, 'error');
        }
    }

    function resetSettings() {
        if (confirm('Reset all settings to default?')) {
            // Reset to default settings
            settings.animations = true;
            settings.debugMode = false;
            settings.autoClearLogs = true;

            saveSettings();

            // Apply changes
            borderBox.classList.remove('animations-off');
            startAnimation();

            // Update toggle buttons
            document.querySelectorAll('.settings-toggle').forEach((toggle, index) => {
                const settingNames = ['animations', 'debugMode', 'autoClearLogs'];
                toggle.classList.toggle('active', settings[settingNames[index]]);
            });

            addLog('Settings reset to default', 'success');
        }
    }

    function refreshUI() {
        if (confirm('Refresh UI? This will restart the interface.')) {
            location.reload();
        }
    }

    function exportSettings() {
        const dataStr = JSON.stringify(settings, null, 2);
        const dataUri = 'data:application/json;charset=utf-8,'+ encodeURIComponent(dataStr);

        const exportFileDefaultName = `duomax_settings_${new Date().toISOString().split('T')[0]}.json`;

        const linkElement = document.createElement('a');
        linkElement.setAttribute('href', dataUri);
        linkElement.setAttribute('download', exportFileDefaultName);
        linkElement.click();

        addLog('Settings exported', 'success');
    }

    // === UI CONTROLS ===
    document.getElementById('btn-close-ui').onclick = () => {
        borderBox.remove();
        externalButtons.remove();
        versionTag.remove();
        discordChat.remove();
        settingsOverlay.remove();
        settingsPanel.remove();
        stopAnimation();
    };

    document.getElementById('btn-minimize-toggle').onclick = () => {
        borderBox.classList.toggle('minimized');
        document.getElementById('logo-main').innerHTML = borderBox.classList.contains('minimized')
            ? "DUOMAX LITE"
            : `DUOMAX LITE <span style="font-size:8px; color:#64748b" class="hide-on-min">${VERSION}</span>`;
        navigator.vibrate?.(20);
    };

    // Settings button
    document.getElementById('btn-settings').onclick = () => {
        settingsOverlay.classList.add('active');
        settingsPanel.classList.add('active');
        navigator.vibrate?.(20);
    };

    // Settings panel controls
    document.getElementById('settings-close').onclick = () => {
        settingsOverlay.classList.remove('active');
        settingsPanel.classList.remove('active');
    };

    settingsOverlay.onclick = (e) => {
        if (e.target === settingsOverlay) {
            settingsOverlay.classList.remove('active');
            settingsPanel.classList.remove('active');
        }
    };

    // Toggle buttons
    document.getElementById('toggle-animations').onclick = () => {
        toggleSetting('animations');
        document.querySelector('#toggle-animations .settings-toggle').classList.toggle('active');
    };

    document.getElementById('toggle-debug').onclick = () => {
        toggleSetting('debugMode');
        document.querySelector('#toggle-debug .settings-toggle').classList.toggle('active');
    };

    document.getElementById('toggle-autoclear').onclick = () => {
        toggleSetting('autoClearLogs');
        document.querySelector('#toggle-autoclear .settings-toggle').classList.toggle('active');
    };

    // Action buttons
    document.getElementById('clear-logs-btn').onclick = clearLogs;
    document.getElementById('clear-cache-btn').onclick = clearCache;
    document.getElementById('reset-settings-btn').onclick = resetSettings;
    document.getElementById('refresh-ui-btn').onclick = refreshUI;
    document.getElementById('export-settings-btn').onclick = exportSettings;

    // External buttons control
    let chatVisible = false;
    document.getElementById('external-discord').onclick = () => {
        chatVisible = !chatVisible;
        discordChat.style.display = chatVisible ? 'block' : 'none';
        navigator.vibrate?.(30);
    };

    document.getElementById('external-notify').onclick = () => {
        alert('📢 DuoMax LITE v1.0.5\nXP, GEMS, and REAL STREAK farming active!\n(ADMIN: YAMISCRIPT_DEV)');
        navigator.vibrate?.(30);
    };

    document.getElementById('external-toggle').onclick = () => {
        const isMinimized = borderBox.classList.contains('minimized');
        if (isMinimized) {
            borderBox.classList.remove('minimized');
            borderBox.style.top = '50%';
            borderBox.style.left = '50%';
            borderBox.style.transform = 'translate(-50%, -50%)';
        } else {
            borderBox.style.top = '12px';
            borderBox.style.left = '12px';
            borderBox.style.transform = 'translate(0, 0)';
        }
        navigator.vibrate?.(40);
    };

    // === UTILITY FUNCTIONS ===
    function getJwtToken() {
        try {
            // Method 1: Check localStorage
            const state = localStorage.getItem('duo-state');
            if (state) {
                const parsed = JSON.parse(state);
                if (parsed?.user?.jwt) {
                    return parsed.user.jwt;
                }
            }

            // Method 2: Check cookies
            const cookies = document.cookie.split(';');
            for (let cookie of cookies) {
                cookie = cookie.trim();
                if (cookie.startsWith('jwt_token=')) {
                    return cookie.substring(10);
                }
                if (cookie.startsWith('jwtToken=')) {
                    return cookie.substring(9);
                }
            }

            // Method 3: Check window object
            if (window.duo?.user?.jwt) {
                return window.duo.user.jwt;
            }

            // Method 4: Check sessionStorage
            const sessionJwt = sessionStorage.getItem('duolingo_jwt');
            if (sessionJwt) return sessionJwt;

            return null;
        } catch (e) {
            console.error('Error getting JWT:', e);
            return null;
        }
    }

    function getUserId() {
        try {
            const jwt = getJwtToken();
            if (!jwt) return null;

            // Decode JWT payload
            const payload = jwt.split('.')[1];
            const decoded = JSON.parse(atob(payload.replace(/-/g, '+').replace(/_/g, '/')));
            return decoded.sub || decoded.user_id || null;
        } catch (e) {
            return null;
        }
    }

    // === SYNC USER FUNCTION ===
    async function syncUser() {
        try {
            const jwt = getJwtToken();
            if (!jwt) {
                throw new Error("Not logged in - No JWT token found");
            }

            const userId = getUserId();
            if (!userId) {
                throw new Error("No user ID found in JWT");
            }

            const headers = {
                'Authorization': `Bearer ${jwt}`,
                'Content-Type': 'application/json',
                'Accept': 'application/json'
            };

            addLog(`Fetching user data for ID: ${userId}`, 'info');

            // Get user info from Duolingo API (2017-06-30)
            const userRes = await fetch(`https://www.duolingo.com/2017-06-30/users/${userId}?fields=username,streak,totalXp,gems,profilePicture,fromLanguage,learningLanguage`, { headers });

            if (!userRes.ok) {
                throw new Error(`API Error: ${userRes.status} ${userRes.statusText}`);
            }

            const userData = await userRes.json();
            addLog(`User data received: ${userData.username}`, 'success');

            // Get user info from v2 API for better data
            try {
                const v2Res = await fetch(`https://www.duolingo.com/api/2/users/${userId}?fields=name,avatar,courses`, {
                    headers: {
                        'Authorization': `Bearer ${jwt}`,
                        'Accept': 'application/json'
                    }
                });

                if (v2Res.ok) {
                    const v2Data = await v2Res.json();
                    await updateUserDisplayWithV2Data(userData, v2Data);
                } else {
                    // Fallback to v1 API
                    await tryV1API(userData, userId);
                }
            } catch (v2Error) {
                console.log('V2 API failed, trying v1:', v2Error);
                await tryV1API(userData, userId);
            }

            return userData;

        } catch (e) {
            console.error("Sync User Error:", e);
            addLog(`Sync Error: ${e.message}`, 'error');

            // Show guest state
            document.getElementById('user-display').innerHTML = `
                <div class="user-avatar" style="background:linear-gradient(135deg, #3b82f6, #2dd4bf)"></div>
                <div class="user-info">
                    <b>Guest User</b>
                    <span id="user-status">Login to sync data</span>
                </div>
            `;
            document.getElementById('user-status').textContent = '❌ Not logged in • Please log in';
            return null;
        }
    }

    async function tryV1API(userData, userId) {
        try {
            // Try with userId first
            const v1Res = await fetch(`https://www.duolingo.com/api/1/users/show?id=${userId}`);
            if (v1Res.ok) {
                const v1Data = await v1Res.json();
                await updateUserDisplayWithV1Data(userData, v1Data);
                return;
            }

            // Try with username
            const usernameRes = await fetch(`https://www.duolingo.com/api/1/users/show?username=${userData.username}`);
            if (usernameRes.ok) {
                const usernameData = await usernameRes.json();
                await updateUserDisplayWithV1Data(userData, usernameData);
                return;
            }

            // If all API calls fail, use basic data
            await updateUserDisplayBasic(userData);

        } catch (v1Error) {
            console.log('V1 API failed:', v1Error);
            await updateUserDisplayBasic(userData);
        }
    }

    async function updateUserDisplayWithV2Data(userData, v2Data) {
        // Get display name
        const displayName = v2Data.name || userData.username;

        // Get avatar - try different formats
        let avatarUrl = '';
        if (v2Data.avatar) {
            if (typeof v2Data.avatar === 'string') {
                avatarUrl = v2Data.avatar;
            } else if (v2Data.avatar.large) {
                avatarUrl = v2Data.avatar.large;
            } else if (v2Data.avatar.medium) {
                avatarUrl = v2Data.avatar.medium;
            } else if (v2Data.avatar.small) {
                avatarUrl = v2Data.avatar.small;
            }
        }

        await updateUserDisplay(displayName, userData.username, avatarUrl, userData.streak, userData.totalXp, userData.gems);
    }

    async function updateUserDisplayWithV1Data(userData, v1Data) {
        // Get display name
        const displayName = v1Data.fullname || v1Data.name || userData.username;

        // Get avatar - v1 API returns URLs like "https://dl-web.duolingo.com/avatars/abc123/large"
        let avatarUrl = '';
        if (v1Data.avatar) {
            if (typeof v1Data.avatar === 'string') {
                avatarUrl = v1Data.avatar;
                // Ensure it's a complete URL
                if (avatarUrl && !avatarUrl.startsWith('http')) {
                    avatarUrl = `https://dl-web.duolingo.com${avatarUrl}`;
                }
            }
        }

        await updateUserDisplay(displayName, userData.username, avatarUrl, userData.streak, userData.totalXp, userData.gems);
    }

    async function updateUserDisplayBasic(userData) {
        await updateUserDisplay(userData.username, userData.username, '', userData.streak, userData.totalXp, userData.gems);
    }

    // Helper function to update user display
    async function updateUserDisplay(displayName, username, avatarUrl, streak, xp, gems) {
        // Create avatar element
        const userDisplay = document.getElementById('user-display');

        // Clear existing content
        userDisplay.innerHTML = '';

        // Create avatar container
        const avatarContainer = document.createElement('div');
        avatarContainer.className = 'user-avatar';

        // Handle avatar URL
        if (avatarUrl) {
            // Make sure avatar URL is complete
            if (!avatarUrl.startsWith('http')) {
                avatarUrl = `https://dl-web.duolingo.com${avatarUrl}`;
            }

            // Create img element
            const avatarImg = document.createElement('img');
            avatarImg.className = 'user-avatar';
            avatarImg.src = avatarUrl;
            avatarImg.onerror = function() {
                // If image fails to load, use gradient background
                this.style.display = 'none';
                avatarContainer.style.background = 'linear-gradient(135deg, #3b82f6, #2dd4bf)';
                avatarContainer.style.display = 'flex';
                avatarContainer.style.alignItems = 'center';
                avatarContainer.style.justifyContent = 'center';
                avatarContainer.innerHTML = `<span style="font-weight:bold;color:white;font-size:10px">${displayName.charAt(0).toUpperCase()}</span>`;
            };

            avatarContainer.appendChild(avatarImg);
        } else {
            // Use gradient background with initial
            avatarContainer.style.background = 'linear-gradient(135deg, #3b82f6, #2dd4bf)';
            avatarContainer.style.display = 'flex';
            avatarContainer.style.alignItems = 'center';
            avatarContainer.style.justifyContent = 'center';
            avatarContainer.innerHTML = `<span style="font-weight:bold;color:white;font-size:10px">${displayName.charAt(0).toUpperCase()}</span>`;
        }

        // Create user info container
        const userInfo = document.createElement('div');
        userInfo.className = 'user-info';
        userInfo.innerHTML = `
            <b>${displayName}</b>
            <span id="user-status">@${username} • Streak: ${streak || 0} days</span>
        `;

        // Append to user display
        userDisplay.appendChild(avatarContainer);
        userDisplay.appendChild(userInfo);

        // Update stats
        document.getElementById('st-streak').innerText = streak || 0;
        document.getElementById('st-xp').innerText = xp || 0;
        document.getElementById('st-gems').innerText = gems || 0;
        document.getElementById('current-streak-display').textContent = `Current: ${streak || 0} days`;

        // Set user status
        const userStatus = document.getElementById('user-status');
        if (userStatus) {
            userStatus.textContent = `✅ @${username} • ${streak || 0} day streak`;
        }

        addLog(`User synced: ${displayName} (@${username})`, 'success');
    }

    // === TEST API FUNCTION ===
    document.getElementById('btn-test-api').onclick = async () => {
        addLog('Testing API connection...', 'info');

        const jwt = getJwtToken();
        if (!jwt) {
            addLog('ERROR: No JWT token found. Please log in to Duolingo.', 'error');
            alert('❌ Not logged in! Please log in to Duolingo first.');
            return;
        }

        addLog(`JWT token found: ${jwt.substring(0, 20)}...`, 'success');

        const userId = getUserId();
        if (!userId) {
            addLog('ERROR: Cannot extract user ID from token', 'error');
            return;
        }

        addLog(`User ID: ${userId}`, 'success');

        // Test basic API call
        try {
            const headers = {
                'Authorization': `Bearer ${jwt}`,
                'Content-Type': 'application/json'
            };

            const testRes = await fetch(`https://www.duolingo.com/2017-06-30/users/${userId}?fields=username,streak,totalXp,gems`, { headers });

            if (testRes.ok) {
                const data = await testRes.json();
                addLog(`✅ API Connection SUCCESS! Username: ${data.username}, Streak: ${data.streak || 0}, XP: ${data.totalXp || 0}`, 'success');
                alert(`✅ API Connection Successful!\nUsername: ${data.username}\nStreak: ${data.streak || 0} days\nXP: ${data.totalXp || 0}\nGems: ${data.gems || 0}`);

                // Update UI using sync function
                await syncUser();

            } else {
                addLog(`❌ API Error: ${testRes.status} ${testRes.statusText}`, 'error');
                alert(`❌ API Error: ${testRes.status}`);
            }
        } catch (error) {
            addLog(`❌ Network Error: ${error.message}`, 'error');
            alert('❌ Network error. Check console for details.');
        }
    };

    // === REAL STREAK FARMING FUNCTION ===
    async function farmStreak(targetDays) {
        const jwt = getJwtToken();
        if (!jwt) {
            addLog('ERROR: No authentication token', 'error');
            alert('❌ Please log in to Duolingo first!');
            return;
        }

        const userId = getUserId();
        if (!userId) {
            addLog('ERROR: No user ID', 'error');
            return;
        }

        const headers = {
            'Authorization': `Bearer ${jwt}`,
            'Content-Type': 'application/json'
        };

        try {
            // Get current streak info
            const userRes = await fetch(`https://www.duolingo.com/2017-06-30/users/${userId}?fields=streak,streakData,fromLanguage,learningLanguage`, { headers });
            if (!userRes.ok) {
                addLog('Failed to get user streak info', 'error');
                return;
            }

            const userData = await userRes.json();
            const currentStreak = userData.streak || 0;
            const hasStreak = userData.streakData?.currentStreak;
            const startStreakDate = hasStreak ? userData.streakData.currentStreak.startDate : new Date();
            const startFarmStreakTimestamp = Math.floor(new Date(startStreakDate).getTime() / 1000);
            let currentTimestamp = hasStreak ? startFarmStreakTimestamp - 86400 : startFarmStreakTimestamp;

            addLog(`Starting streak farming: Current streak ${currentStreak} days`, 'info');
            addLog(`Will simulate sessions starting from ${new Date(currentTimestamp * 1000).toLocaleDateString()}`, 'info');
            addLog(`Target: Add ${targetDays} days (Total: ${currentStreak + targetDays} days)`, 'info');

            document.getElementById('progressText').textContent = `🔥 Starting streak farming...`;
            document.getElementById('target-streak-display').textContent = `Target: ${currentStreak + targetDays} days`;

            let streakAdded = 0;

            while (streakAdded < targetDays && isFarming) {
                try {
                    addLog(`Processing streak day ${streakAdded + 1}/${targetDays}`, 'info');

                    // Create a practice session
                    const sessionPayload = {
                        challengeTypes: ['translate', 'assist'],
                        fromLanguage: userData.fromLanguage || 'en',
                        learningLanguage: userData.learningLanguage || 'es',
                        type: 'GLOBAL_PRACTICE',
                        difficulty: 'EASY',
                        startTime: currentTimestamp,
                        endTime: currentTimestamp + 300
                    };

                    const sessionRes = await fetch('https://www.duolingo.com/2017-06-30/sessions', {
                        method: 'POST',
                        headers,
                        body: JSON.stringify(sessionPayload)
                    });

                    if (!sessionRes.ok) {
                        throw new Error(`Session creation failed: ${sessionRes.status}`);
                    }

                    const sessionData = await sessionRes.json();

                    // Complete the session with good results
                    const completePayload = {
                        ...sessionData,
                        heartsLeft: 5,
                        failed: false,
                        maxInLessonStreak: 10,
                        shouldLearnThings: true,
                        xpGain: 15,
                        startTime: currentTimestamp,
                        endTime: currentTimestamp + 300
                    };

                    const completeRes = await fetch(`https://www.duolingo.com/2017-06-30/sessions/${sessionData.id}`, {
                        method: 'PUT',
                        headers,
                        body: JSON.stringify(completePayload)
                    });

                    if (!completeRes.ok) {
                        throw new Error(`Session completion failed: ${completeRes.status}`);
                    }

                    // Move to previous day for next session
                    currentTimestamp -= 86400;
                    streakAdded++;

                    // Calculate new streak total
                    const newStreak = currentStreak + streakAdded;

                    // Update UI
                    document.getElementById('st-streak').innerText = newStreak;
                    document.getElementById('data-streak-days').innerText = streakAdded;
                    document.getElementById('progressBar').style.width = `${(streakAdded/targetDays)*100}%`;
                    document.getElementById('progressText').textContent = `🔥 Day ${streakAdded}/${targetDays}: Streak at ${newStreak} days`;
                    document.getElementById('current-streak-display').textContent = `Current: ${newStreak} days`;

                    // Also add XP for the lesson
                    const currentXP = parseInt(document.getElementById('st-xp').innerText) || 0;
                    document.getElementById('st-xp').innerText = currentXP + 15;

                    addLog(`✅ Streak day ${streakAdded} completed. Total: ${newStreak} days`, 'success');

                    // Delay between days based on mode
                    const delayTime = mult === 1 ? 5000 : 2000;
                    await new Promise(r => setTimeout(r, delayTime));

                } catch (dayError) {
                    addLog(`⚠️ Session failed for day ${streakAdded + 1}: ${dayError.message}`, 'error');
                    await new Promise(r => setTimeout(r, 3000));
                }
            }

            if (streakAdded >= targetDays) {
                addLog(`✅ Streak farming completed! Added ${streakAdded} days`, 'success');
                document.getElementById('progressText').textContent = `✅ Streak farming completed! Added ${streakAdded} days`;

                // Verify the new streak
                try {
                    const verifyRes = await fetch(`https://www.duolingo.com/2017-06-30/users/${userId}?fields=streak`, { headers });
                    if (verifyRes.ok) {
                        const verifyData = await verifyRes.json();
                        const finalStreak = verifyData.streak || 0;
                        addLog(`Final streak verification: ${finalStreak} days`, 'success');

                        if (finalStreak >= currentStreak + streakAdded) {
                            alert(`🎉 SUCCESS! Your streak is now ${finalStreak} days!`);
                        } else {
                            alert(`⚠️ Partial success: Streak increased from ${currentStreak} to ${finalStreak} days`);
                        }
                    }
                } catch (verifyError) {
                    addLog('Could not verify final streak', 'info');
                    alert(`✅ Streak farming completed! Check your Duolingo profile.`);
                }
            }

        } catch (error) {
            addLog(`❌ Streak farming error: ${error.message}`, 'error');
            document.getElementById('progressText').textContent = '❌ Error farming streak!';
            alert('❌ Error farming streak. Please try again.');
        }
    }

    // === XP FARMING FUNCTION ===
    async function farmXP_Simple(amount) {
        addLog(`Starting XP farm: ${amount} XP`, 'info');

        const jwt = getJwtToken();
        if (!jwt) {
            addLog('ERROR: No authentication token', 'error');
            return;
        }

        const userId = getUserId();
        if (!userId) {
            addLog('ERROR: No user ID', 'error');
            return;
        }

        const headers = {
            'Authorization': `Bearer ${jwt}`,
            'Content-Type': 'application/json'
        };

        let farmed = 0;
        const batchSize = 10; // XP per session

        while (farmed < amount && isFarming) {
            try {
                // Get user language info
                const userInfo = await fetch(`https://www.duolingo.com/2017-06-30/users/${userId}?fields=fromLanguage,learningLanguage`, { headers });
                const userData = await userInfo.json();

                // Create a simple practice session
                const sessionData = {
                    challengeTypes: ['translate'],
                    fromLanguage: userData.fromLanguage || 'en',
                    learningLanguage: userData.learningLanguage || 'es',
                    type: 'GLOBAL_PRACTICE',
                    difficulty: 'EASY'
                };

                const sessionRes = await fetch('https://www.duolingo.com/2017-06-30/sessions', {
                    method: 'POST',
                    headers,
                    body: JSON.stringify(sessionData)
                });

                if (!sessionRes.ok) {
                    addLog(`Session creation failed: ${sessionRes.status}`, 'error');
                    await new Promise(r => setTimeout(r, 2000));
                    continue;
                }

                const session = await sessionRes.json();

                // Complete the session
                const completeData = {
                    ...session,
                    heartsLeft: 5,
                    startTime: Math.floor(Date.now()/1000) - 60,
                    endTime: Math.floor(Date.now()/1000),
                    failed: false,
                    maxInLessonStreak: 10,
                    shouldLearnThings: true,
                    xpGain: batchSize
                };

                const completeRes = await fetch(`https://www.duolingo.com/2017-06-30/sessions/${session.id}`, {
                    method: 'PUT',
                    headers,
                    body: JSON.stringify(completeData)
                });

                if (completeRes.ok) {
                    farmed += batchSize;

                    // Update UI
                    const currentXP = parseInt(document.getElementById('st-xp').innerText) || 0;
                    document.getElementById('st-xp').innerText = currentXP + batchSize;
                    document.getElementById('progressBar').style.width = `${(farmed/amount)*100}%`;
                    document.getElementById('progressText').textContent = `🧠 Farming ${farmed}/${amount} XP`;

                    addLog(`+${batchSize} XP (Total: ${farmed}/${amount})`, 'success');

                    // Wait based on mode
                    await new Promise(r => setTimeout(r, mult === 1 ? 3000 : 1000));
                } else {
                    addLog(`Session completion failed: ${completeRes.status}`, 'error');
                }

            } catch (error) {
                addLog(`Error in XP farm: ${error.message}`, 'error');
                await new Promise(r => setTimeout(r, 3000));
            }
        }

        if (farmed >= amount) {
            addLog(`✅ XP farming completed: ${farmed} XP gained`, 'success');
            document.getElementById('progressText').textContent = '✅ XP farming completed!';
        }
    }

    // === GEMS FARMING FUNCTION ===
    async function farmGems_Simple(amount) {
        addLog(`Starting Gems farm: ${amount} gems`, 'info');

        const jwt = getJwtToken();
        if (!jwt) {
            addLog('ERROR: No authentication token', 'error');
            alert('❌ Please log in to Duolingo first!');
            return;
        }

        const userId = getUserId();
        if (!userId) {
            addLog('ERROR: No user ID', 'error');
            return;
        }

        const headers = {
            'Authorization': `Bearer ${jwt}`,
            'Content-Type': 'application/json',
            'Accept': 'application/json'
        };

        let farmed = 0;

        while (farmed < amount && isFarming) {
            try {
                addLog(`Attempting to earn gem ${farmed + 1}/${amount}...`, 'info');

                // Get user language info
                const userInfo = await fetch(`https://www.duolingo.com/2017-06-30/users/${userId}?fields=fromLanguage,learningLanguage`, { headers });
                const userData = await userInfo.json();

                // Create a practice session
                const sessionData = {
                    challengeTypes: ['translate', 'assist'],
                    fromLanguage: userData.fromLanguage || 'en',
                    learningLanguage: userData.learningLanguage || 'es',
                    type: 'GLOBAL_PRACTICE',
                    difficulty: 'EASY'
                };

                const sessionRes = await fetch('https://www.duolingo.com/2017-06-30/sessions', {
                    method: 'POST',
                    headers,
                    body: JSON.stringify(sessionData)
                });

                if (!sessionRes.ok) {
                    throw new Error(`Session creation failed: ${sessionRes.status}`);
                }

                const session = await sessionRes.json();

                // Complete the session with gem reward
                const completeData = {
                    ...session,
                    heartsLeft: 5,
                    failed: false,
                    maxInLessonStreak: 10,
                    shouldLearnThings: true,
                    xpGain: 10,
                    gemGain: 2,
                    startTime: Math.floor(Date.now()/1000) - 60,
                    endTime: Math.floor(Date.now()/1000)
                };

                const completeRes = await fetch(`https://www.duolingo.com/2017-06-30/sessions/${session.id}`, {
                    method: 'PUT',
                    headers,
                    body: JSON.stringify(completeData)
                });

                if (completeRes.ok) {
                    farmed += 2;

                    // Update UI
                    const currentGems = parseInt(document.getElementById('st-gems').innerText) || 0;
                    document.getElementById('st-gems').innerText = currentGems + 2;
                    document.getElementById('progressBar').style.width = `${(farmed/amount)*100}%`;
                    document.getElementById('progressText').textContent = `💎 Farming ${farmed}/${amount} gems`;

                    // Also add XP for consistency
                    const currentXP = parseInt(document.getElementById('st-xp').innerText) || 0;
                    document.getElementById('st-xp').innerText = currentXP + 10;

                    addLog(`+2 Gems via lesson completion (Total: ${farmed}/${amount})`, 'success');

                    // Wait based on mode
                    await new Promise(r => setTimeout(r, mult === 1 ? 4000 : 2000));

                } else {
                    // Alternative method: Try to update via gem transaction
                    addLog(`Standard method failed, trying alternative...`, 'warning');

                    try {
                        // Get current gems first
                        const currentRes = await fetch(`https://www.duolingo.com/2017-06-30/users/${userId}?fields=gems`, { headers });
                        if (currentRes.ok) {
                            const currentData = await currentRes.json();
                            const currentGems = currentData.gems || 0;

                            // Try direct PATCH update
                            const updateRes = await fetch(`https://www.duolingo.com/2017-06-30/users/${userId}`, {
                                method: 'PATCH',
                                headers,
                                body: JSON.stringify({
                                    gems: currentGems + 2
                                })
                            });

                            if (updateRes.ok) {
                                farmed += 2;
                                document.getElementById('st-gems').innerText = currentGems + 2;
                                document.getElementById('progressBar').style.width = `${(farmed/amount)*100}%`;
                                document.getElementById('progressText').textContent = `💎 Farming ${farmed}/${amount} gems`;
                                addLog(`+2 Gems via direct update (Total: ${farmed}/${amount})`, 'success');
                            } else {
                                addLog(`Direct update failed: ${updateRes.status}`, 'error');
                            }
                        }
                    } catch (altError) {
                        addLog(`Alternative method error: ${altError.message}`, 'error');
                    }

                    await new Promise(r => setTimeout(r, 3000));
                }

            } catch (error) {
                addLog(`Error in gems farm: ${error.message}`, 'error');
                await new Promise(r => setTimeout(r, 3000));
            }
        }

        if (farmed >= amount) {
            addLog(`✅ Gems farming completed: ${farmed} gems gained`, 'success');
            document.getElementById('progressText').textContent = '✅ Gems farming completed!';

            // Verify final gems count
            try {
                const verifyRes = await fetch(`https://www.duolingo.com/2017-06-30/users/${userId}?fields=gems`, { headers });
                if (verifyRes.ok) {
                    const verifyData = await verifyRes.json();
                    const finalGems = verifyData.gems || 0;
                    addLog(`Final gems verification: ${finalGems} gems`, 'success');
                    alert(`✅ Gems farming completed! You now have ${finalGems} gems!`);
                }
            } catch (verifyError) {
                addLog('Could not verify final gems count', 'info');
                alert(`✅ Gems farming completed! Check your Duolingo profile.`);
            }
        }
    }

    // === FARMING CONTROL ===
    let isFarming = false;
    let seconds = 0;
    let mult = 1;

    // Target modules selection
    ['xp', 'gems', 'streak', 'league', 'quest', 'all'].forEach(k => {
        const el = document.getElementById('opt-' + k);
        el.onclick = () => {
            if (k === 'all') {
                // Toggle all except "all"
                ['xp', 'gems', 'streak', 'league', 'quest'].forEach(t => {
                    const item = document.getElementById('opt-' + t);
                    item.classList.toggle('active');
                });
            } else {
                el.classList.toggle('active');
            }

            if (['xp', 'gems', 'streak'].includes(k)) {
                // Update placeholder for input
                const targetInput = document.getElementById('targetValue');
                switch(k) {
                    case 'xp':
                        targetInput.placeholder = 'e.g., 1000 XP';
                        targetInput.value = '1000';
                        targetInput.max = '999999';
                        break;
                    case 'gems':
                        targetInput.placeholder = 'e.g., 500 Gems';
                        targetInput.value = '500';
                        targetInput.max = '9999';
                        break;
                    case 'streak':
                        targetInput.placeholder = 'e.g., 30 days';
                        targetInput.value = '30';
                        targetInput.max = '365';
                        break;
                }
            }
        };
    });

    // Mode selection
    document.getElementById('sp-safe').onclick = function() {
        mult = 1;
        this.classList.add('active');
        document.getElementById('sp-fast').classList.remove('active');
        document.getElementById('farm-mode-indicator').innerHTML = '🛡️ SAFE MODE | DELAY: 3s';
        addLog('Switched to Safe Mode', 'info');
    };

    document.getElementById('sp-fast').onclick = function() {
        mult = 3;
        this.classList.add('active');
        document.getElementById('sp-safe').classList.remove('active');
        document.getElementById('farm-mode-indicator').innerHTML = '⚡ FAST MODE | DELAY: 1s';
        addLog('Switched to Fast Mode', 'info');
    };

    // Main farming control
    const masterBtn = document.getElementById('btn-master-farm');
    masterBtn.onclick = async function() {
        if (isFarming) {
            // Stop farming
            isFarming = false;
            this.innerText = t('start');
            this.classList.remove('farming');
            document.getElementById('data-status').innerText = "IDLE";
            document.getElementById('data-status').style.color = "#ef4444";
            document.getElementById('progressText').textContent = '⏸️ Farming stopped';
            addLog('Farming stopped by user', 'info');
            return;
        }

        // Check if logged in
        const jwt = getJwtToken();
        if (!jwt) {
            alert('❌ Please log in to Duolingo first!\n\n1. Make sure you are on Duolingo website\n2. Log in with your account\n3. Refresh the page');
            addLog('ERROR: User not logged in', 'error');
            return;
        }

        const amount = parseInt(document.getElementById('targetValue').value || '0');
        if (!amount || amount <= 0) {
            alert('❌ Please enter a valid amount!');
            document.getElementById('targetValue').focus();
            return;
        }

        // Check what's selected
        const selectedModules = [];
        ['xp', 'gems', 'streak'].forEach(type => {
            if (document.getElementById('opt-' + type).classList.contains('active')) {
                selectedModules.push(type);
            }
        });

        if (selectedModules.length === 0) {
            alert('❌ Please select at least one farming module!');
            return;
        }

        const confirmMsg = `Start farming ${amount} ${selectedModules.join(', ').toUpperCase()} in ${mult === 1 ? 'SAFE' : 'FAST'} mode?\n\nMake sure you are logged into Duolingo.`;
        if (!confirm(confirmMsg)) return;

        // Start farming
        isFarming = true;
        seconds = 0;
        this.innerText = "STOP FARMING";
        this.classList.add('farming');
        document.getElementById('data-status').innerText = "RUNNING";
        document.getElementById('data-status').style.color = "#2dd4bf";
        document.getElementById('progressText').textContent = '⏳ Starting farm...';
        document.getElementById('progressBar').style.width = '0%';
        document.getElementById('data-streak-days').innerText = '0';

        addLog(`Starting farming session: ${selectedModules.join(', ')} x${amount}`, 'info');

        try {
            // Execute selected modules
            for (const module of selectedModules) {
                if (!isFarming) break;

                addLog(`Starting ${module.toUpperCase()} farming...`, 'info');

                switch(module) {
                    case 'xp':
                        await farmXP_Simple(amount);
                        break;
                    case 'gems':
                        await farmGems_Simple(amount);
                        break;
                    case 'streak':
                        await farmStreak(amount);
                        break;
                }

                // Reset for next module if continuing
                if (isFarming && selectedModules.indexOf(module) < selectedModules.length - 1) {
                    document.getElementById('progressText').textContent = `⏭️ Switching to next module...`;
                    document.getElementById('progressBar').style.width = '0%';
                    document.getElementById('data-streak-days').innerText = '0';
                    await new Promise(r => setTimeout(r, 1000));
                }
            }

        } catch(error) {
            console.error('Farming error:', error);
            addLog(`Farming error: ${error.message}`, 'error');
            document.getElementById('progressText').textContent = '❌ Farming error occurred';
        }

        if (isFarming) {
            isFarming = false;
            this.innerText = t('start');
            this.classList.remove('farming');
            document.getElementById('data-status').innerText = "IDLE";
            document.getElementById('data-status').style.color = "#ef4444";
            addLog('Farming session completed', 'success');
        }
    };

    // Timer for runtime display
    setInterval(() => {
        if (!isFarming) return;
        seconds++;
        const hrs = String(Math.floor(seconds / 3600)).padStart(2, '0');
        const mins = String(Math.floor((seconds % 3600) / 60)).padStart(2, '0');
        const secs = String(seconds % 60).padStart(2, '0');
        document.getElementById('data-time').innerText = `${hrs}:${mins}:${secs}`;
    }, 1000);

    // Enter key support
    document.getElementById('targetValue').addEventListener('keypress', (e) => {
        if (e.key === 'Enter') {
            document.getElementById('btn-master-farm').click();
        }
    });

    // === INITIALIZATION ===
    addLog('DuoMax LITE v1.0.5 initialized', 'info');
    addLog(`UI Size: Compact (580px)`, 'info');
    addLog(`Animations: ${settings.animations ? 'ON' : 'OFF'}`, 'info');

    // Auto-detect login and sync user data
    setTimeout(async () => {
        const jwt = getJwtToken();
        if (jwt) {
            addLog('Auto-detected: User is logged in', 'success');
            document.getElementById('user-status').textContent = '✅ Logged in • Syncing data...';

            // Sync user data
            await syncUser();

            // Update streak target display
            const currentStreak = parseInt(document.getElementById('st-streak').innerText) || 0;
            const targetInput = document.getElementById('targetValue');
            const targetValue = parseInt(targetInput.value) || 30;
            document.getElementById('target-streak-display').textContent = `Target: ${currentStreak + targetValue} days`;

        } else {
            addLog('User not logged in. Please log into Duolingo.', 'error');
            document.getElementById('user-status').textContent = '❌ Not logged in • Please log in';
        }
    }, 2000);

    // Update target display when value changes
    document.getElementById('targetValue').addEventListener('input', function() {
        const currentStreak = parseInt(document.getElementById('st-streak').innerText) || 0;
        const targetValue = parseInt(this.value) || 0;
        document.getElementById('target-streak-display').textContent = `Target: ${currentStreak + targetValue} days`;
    });
})();