Lompat ke isi

MediaWiki:Common.js: Perbedaan antara revisi

Dari Mippedia Data, basis data terbuka
Tidak ada ringkasan suntingan
Tanda: Suntingan perangkat seluler Suntingan peramban seluler
Tidak ada ringkasan suntingan
Tanda: Suntingan perangkat seluler Suntingan peramban seluler
Baris 153: Baris 153:


/* ==========================================================
/* ==========================================================
   🚀 MIPPEDIA DATA - CORE INTERFACE (FINAL BUNDLE)
   🚀 MIPPEDIA DATA - TOTAL INTERFACE BUNDLE (FINAL V5)
   Includes: Header, Sitelink V4 (Smart API), and Footer
   Fungsi: Header, Connector (Admin-Only), & Footer Lisensi
   Rules: Only Namespace 0 & NOT Main Page
   Aturan: Hanya Namespace 0 & Bukan Halaman Utama
   ========================================================== */
   ========================================================== */


(function() {
(function() {
     $(document).ready(function() {
     $(document).ready(function() {
         // --- PROTEKSI GLOBAL ---
         // --- 1. FILTER GLOBAL NAMESPACE & HALAMAN UTAMA ---
        // 1. Harus Namespace Utama (0)
        // 2. Bukan Halaman Utama (Main Page)
        // 3. Hanya mode View (bukan edit/history)
         if (mw.config.get('wgNamespaceNumber') !== 0 ||  
         if (mw.config.get('wgNamespaceNumber') !== 0 ||  
             mw.config.get('wgIsMainPage') ||  
             mw.config.get('wgIsMainPage') ||  
Baris 170: Baris 167:
         var pageTitle = mw.config.get('wgPageName');
         var pageTitle = mw.config.get('wgPageName');
         var dataPage = 'MediaWiki:Sitelinks-Data.json';
         var dataPage = 'MediaWiki:Sitelinks-Data.json';
        var userGroups = mw.config.get('wgUserGroups');
        var isAdmin = userGroups.includes('sysop') || userGroups.includes('interface-admin');
         var projects = {
         var projects = {
             'id': { name: 'ID', url: 'https://id.mippedia.org/api.php', base: 'https://id.mippedia.org/wiki/', color: '#6a5acd' },
             'id': { name: 'ID', url: 'https://id.mippedia.org/api.php', base: 'https://id.mippedia.org/wiki/', color: '#6a5acd' },
Baris 176: Baris 176:
         };
         };


         // --- 1. HEADER (ATAS) ---
         // --- 2. FUNGSI NOTIFIKASI KOTAK PROFESIONAL ---
        var headerHtml =  
        function showBoxNotif(title, msg, type) {
             '<div id="mip-data-header" style="background: #f8f9fa; border-left: 5px solid #6a5acd; border-right: 1px solid #ddd; border-top: 1px solid #ddd; border-bottom: 1px solid #ddd; padding: 12px; margin-bottom: 15px; border-radius: 0 4px 4px 0; font-family: sans-serif;">' +
            $('.mip-box-notif').remove();
                 '<div style="display: flex; align-items: center; margin-bottom: 5px;">' +
            var theme = type === 'success' ? {bg:'#e6fffa', border:'#38b2ac', text:'#234e52', icon:'✅'} :
                     '<span style="background: #6a5acd; color: white; padding: 2px 8px; border-radius: 3px; font-size: 10px; font-weight: bold; margin-right: 8px; text-transform: uppercase;">Basis Data</span>' +
                        {bg:'#fff5f5', border:'#f56565', text:'#742a2a', icon:'🚫'};
                     '<strong style="color: #333; font-size: 14px;">Mippedia Data Project</strong>' +
              
            var $notif = $('<div class="mip-box-notif" style="position:fixed; top:25px; right:25px; width:340px; background:'+theme.bg+'; border-left:6px solid '+theme.border+'; padding:16px; z-index:10001; border-radius:10px; box-shadow:0 12px 30px rgba(0,0,0,0.18); font-family:sans-serif; animation:mipSlideIn 0.4s ease-out;">' +
                 '<div style="display:flex; align-items:center; gap:12px; margin-bottom:6px;">' +
                     '<span style="font-size:18px;">'+theme.icon+'</span>' +
                     '<strong style="color:'+theme.text+'; font-size:14px; letter-spacing:-0.2px;">'+title+'</strong>' +
                 '</div>' +
                 '</div>' +
                 '<p style="margin: 0; font-size: 12.5px; color: #555; line-height: 1.5;">' +
                 '<p style="margin:0; font-size:12.5px; color:'+theme.text+'; line-height:1.5; opacity:0.9;">'+msg+'</p>' +
                    'Halaman ini merupakan repositori data terstruktur dari ekosistem <strong>Mippedia</strong>. ' +
             '</div>').appendTo('body');
                    'Informasi yang ditampilkan di bawah ini adalah metadata entitas yang disinkronisasi secara otomatis untuk keperluan referensi terbuka.' +
                '</p>' +
             '</div>';
        $('#mw-content-text').prepend(headerHtml);


        // --- 2. SITELINK V4 (SMART CONNECTOR) ---
            setTimeout(function() { $notif.fadeOut(function(){ $(this).remove(); }); }, 5000);
        function showFancyNotif(msg, type) {
            $('.mip-notif').remove();
            var color = type === 'success' ? '#00ffa3' : '#ff4d4d';
            var icon = type === 'success' ? '✓' : '✕';
            var $notif = $('<div class="mip-notif" style="position:fixed; top:40px; left:50%; transform:translateX(-50%); background:rgba(15,15,15,0.92); backdrop-filter:blur(12px); color:white; padding:14px 28px; border-radius:12px; font-size:13px; font-weight:600; z-index:10000; display:flex; align-items:center; gap:12px; box-shadow:0 15px 35px rgba(0,0,0,0.5); border:1px solid rgba(255,255,255,0.1); border-bottom:3px solid '+color+'; animation:mipSlideDown 0.5s cubic-bezier(0.19, 1, 0.22, 1);">'+
                '<span style="background:'+color+'; color:#000; width:22px; height:22px; display:flex; align-items:center; justify-content:center; border-radius:50%; font-size:12px; font-weight:900;">'+icon+'</span>'+
                '<span>'+msg+'</span></div>').appendTo('body');
            setTimeout(function() { $notif.fadeOut(); }, 3800);
         }
         }


         var editIcon = mw.config.get('wgUserGroups').includes('sysop') || !mw.config.get('wgRestrictionEdit').includes('sysop') ? '<span id="mip-edit-links" style="cursor:pointer; color:#6a5acd; font-size:18px; opacity:0.7;">✎</span>' : '';
         // --- 3. UI GENERATOR (HEADER, CONNECTOR, FOOTER) ---
         var sitelinkHtml = '<div id="mip-sitelink-box" style="background:#fff; border:1px solid #e0e0e0; border-radius:14px; padding:20px; margin-top:25px; box-shadow:0 8px 24px rgba(0,0,0,0.04);">' +
       
             '<div style="display:flex; justify-content:space-between; align-items:center; margin-bottom:18px;"><strong style="font-size:14px; color:#1a1a1a;">🔗 Koneksi Ekosistem Mippedia</strong>' + editIcon + '</div>' +
        // A. Header (Paling Atas)
            '<div id="mip-link-display" style="display:flex; gap:10px; flex-wrap:wrap;"><small style="color:#888;">Memuat koneksi...</small></div>' +
         var headerHtml = '<div id="mip-header" style="background:#f8f9fa; border:1px solid #ddd; border-left:5px solid #6a5acd; padding:15px; margin-bottom:20px; border-radius:4px;">' +
             '<div id="mip-link-editor" style="display:none; margin-top:18px; border-top:1px solid #f0f0f0; padding-top:18px;">' +
             '<div style="display:flex; align-items:center; gap:10px; margin-bottom:5px;">' +
            Object.keys(projects).map(p => '<div style="margin-bottom:10px; display:flex; gap:10px;"><span style="width:65px; font-size:10px; font-weight:800; background:#f5f5f5; display:flex; align-items:center; justify-content:center; border-radius:8px;">'+projects[p].name+'</span><input type="text" id="input-'+p+'" placeholder="Ketik judul..." style="flex:1; padding:9px; font-size:13px; border:1.5px solid #eee; border-radius:8px; outline:none;"></div>').join('') +
             '<span style="background:#6a5acd; color:white; padding:2px 8px; border-radius:3px; font-size:10px; font-weight:bold; text-transform:uppercase;">Basis Data</span>' +
             '<button id="mip-save-links" style="width:100%; margin-top:12px; padding:12px; background:#6a5acd; color:white; border:none; border-radius:10px; font-size:13px; font-weight:700; cursor:pointer;">Sinkronkan Data</button></div></div>' +
            '<strong style="color:#1a1a1a; font-size:15px;">Mippedia Data Project</strong></div>' +
            '<style>@keyframes mipSlideDown{from{top:0; opacity:0;} to{top:40px; opacity:1;}} .mip-pill{color:white !important; padding:7px 15px; border-radius:8px; font-size:11px; font-weight:700; text-decoration:none !important; border-bottom:2px solid rgba(0,0,0,0.15); transition:0.2s;} .mip-pill:hover{transform:translateY(-2px); opacity:0.9;}</style>';
             '<p style="margin:0; font-size:12.5px; color:#555; line-height:1.5;">Halaman ini merupakan repositori data terstruktur dari ekosistem <strong>Mippedia</strong> untuk keperluan referensi terbuka.</p></div>';


         $('#mw-content-text').append(sitelinkHtml); // Sitelink muncul setelah konten
         // B. Connector (Bawah Konten)
        var editBtn = '<span id="mip-open-editor" style="cursor:pointer; font-size:18px; color:#6a5acd; opacity:0.7; transition:0.2s;" title="Edit Sitelink">✎</span>';
        var connectorHtml = '<div id="mip-connector" style="background:#fff; border:1px solid #e0e0e0; border-radius:14px; padding:20px; margin-top:25px; box-shadow:0 4px 15px rgba(0,0,0,0.05);">' +
            '<div style="display:flex; justify-content:space-between; align-items:center; margin-bottom:18px;"><strong style="font-size:14px; color:#1a1a1a;">🔗 Koneksi Ekosistem Mippedia</strong>'+editBtn+'</div>' +
            '<div id="mip-display" style="display:flex; gap:10px; flex-wrap:wrap;"><small style="color:#999;">Sinkronisasi...</small></div>' +
            '<div id="mip-editor" style="display:none; margin-top:18px; border-top:1px solid #f0f0f0; padding-top:18px;">' +
                Object.keys(projects).map(p => '<div style="margin-bottom:12px; display:flex; gap:10px;"><span style="width:65px; font-size:10px; font-weight:800; background:#f5f5f5; display:flex; align-items:center; justify-content:center; border-radius:8px; color:#666;">'+projects[p].name+'</span><input type="text" id="in-'+p+'" placeholder="Judul artikel..." style="flex:1; padding:9px; font-size:13px; border:1.5px solid #eee; border-radius:8px; outline:none;"></div>').join('') +
                '<button id="mip-save" style="width:100%; padding:12px; background:#6a5acd; color:white; border:none; border-radius:10px; font-size:13px; font-weight:bold; cursor:pointer; transition:0.3s;">Sinkronkan Data</button>' +
            '</div></div>';


         // --- 3. FOOTER (LISENSI) ---
         // C. Footer (Paling Bawah)
         var footerHtml =  
         var footerHtml = '<div id="mip-footer" style="background:#fff; border:1px dashed #ccc; padding:15px; margin-top:25px; text-align:center; border-radius:8px;">' +
            '<div id="mip-data-footer" style="background: #fff; border: 1px dashed #ccc; padding: 10px; margin-top: 20px; text-align: center; border-radius: 6px;">' +
            '<div style="font-size:11px; color:#888; font-weight:bold; text-transform:uppercase; margin-bottom:5px;">Lisensi & Atribusi</div>' +
                '<div style="font-size: 11px; color: #888; margin-bottom: 4px; font-weight: bold; text-transform: uppercase; letter-spacing: 0.5px;">Lisensi & Penggunaan Data</div>' +
            '<p style="margin:0; font-size:12px; color:#666;">Data tersedia di bawah lisensi <a href="https://creativecommons.org/licenses/by-sa/4.0/" target="_blank" style="color:#6a5acd; text-decoration:none; font-weight:bold;">CC BY-SA 4.0</a>.</p></div>';
                '<p style="margin: 0; font-size: 12px; color: #666; line-height: 1.4;">' +
                    'Seluruh data di dalam <strong>Mippedia Data</strong> tersedia di bawah lisensi ' +
                    '<a href="https://creativecommons.org/licenses/by-sa/4.0/" target="_blank" style="color: #6a5acd; text-decoration: none;">CC BY-SA 4.0</a>. ' +
                    'Data ini dapat digunakan kembali dengan mencantumkan atribusi ke Mippedia Community.' +
                '</p>' +
            '</div>';
        $('#mw-content-text').append(footerHtml);


         // --- LOGIC: LOAD, VERIFY, & SAVE ---
         // Injeksi ke Halaman
         function loadData() {
        $('#mw-content-text').prepend(headerHtml);
             new mw.Api().get({ action: 'query', prop: 'revisions', titles: dataPage, rvprop: 'content', formatversion: 2 }).done(function(data) {
        $('#mw-content-text').append(connectorHtml).append(footerHtml);
                 var content = data.query.pages[0].revisions ? JSON.parse(data.query.pages[0].revisions[0].content) : {};
 
                 var links = content[pageTitle] || {};
        // --- 4. LOGIC SISTEM ---
                 var $display = $('#mip-link-display').empty();
         function loadLinks() {
                 var hasAny = false;
             new mw.Api().get({ action: 'query', prop: 'revisions', titles: dataPage, rvprop: 'content', formatversion: 2 }).done(function(d) {
                 var json = d.query.pages[0].revisions ? JSON.parse(d.query.pages[0].revisions[0].content) : {};
                 var data = json[pageTitle] || {};
                 var $box = $('#mip-display').empty();
                 var found = false;
                 Object.keys(projects).forEach(p => {
                 Object.keys(projects).forEach(p => {
                     if (links[p]) {
                     if (data[p]) {
                         hasAny = true;
                         found = true;
                         $display.append('<a class="mip-pill" style="background:'+projects[p].color+'" href="'+projects[p].base+encodeURIComponent(links[p])+'" target="_blank">Mippedia '+projects[p].name+': '+links[p]+'</a>');
                         $box.append('<a style="background:'+projects[p].color+'; color:white; padding:7px 15px; border-radius:8px; font-size:11px; font-weight:bold; text-decoration:none; transition:0.2s;" href="'+projects[p].base+encodeURIComponent(data[p])+'" target="_blank">Mippedia '+projects[p].name+': '+data[p]+'</a>');
                         $('#input-' + p).val(links[p]);
                         $('#in-' + p).val(data[p]);
                     }
                     }
                 });
                 });
                 if (!hasAny) $display.html('<em style="color:#888; font-size:12px;">Belum terkoneksi.</em>');
                 if (!found) $box.html('<em style="color:#bbb; font-size:12px;">Entitas ini belum terhubung ke ekosistem.</em>');
             });
             });
         }
         }


         $(document).on('click', '#mip-save-links', function() {
         $(document).on('click', '#mip-save', function() {
             var $btn = $(this).text('Memverifikasi...').prop('disabled', true);
            // PROTEKSI: Tolak jika bukan Admin
            if (!isAdmin) {
                showBoxNotif('Akses Terbatas', 'Hanya grup <b>Pengurus</b> yang dapat melakukan sinkronisasi metadata permanen.', 'error');
                return;
            }
 
             var $btn = $(this).text('Memverifikasi Judul...').prop('disabled', true);
             var api = new mw.Api();
             var api = new mw.Api();
             var newData = {}, checkPromises = [], invalid = [], isDel = true;
             var newData = {}, promises = [], invalid = [], isDel = true;


             Object.keys(projects).forEach(p => {
             Object.keys(projects).forEach(p => {
                 var v = $('#input-' + p).val().trim();
                 var v = $('#in-' + p).val().trim();
                 if (v) {
                 if (v) {
                     isDel = false;
                     isDel = false;
                     checkPromises.push($.ajax({ url: projects[p].url, data: { action: 'query', titles: v, format: 'json', origin: '*' }, dataType: 'json' }).done(function(d){
                     promises.push($.ajax({ url: projects[p].url, data: { action: 'query', titles: v, format: 'json', origin: '*' }, dataType: 'json' }).done(function(res){
                         if (d.query.pages["-1"]) invalid.push(projects[p].name); else newData[p] = v;
                         if (res.query.pages["-1"]) invalid.push(projects[p].name); else newData[p] = v;
                     }));
                     }));
                 }
                 }
             });
             });


             $.when.apply($, checkPromises).then(function() {
             $.when.apply($, promises).then(function() {
                 if (invalid.length > 0) {
                 if (invalid.length > 0) {
                     showFancyNotif('Judul tidak ada di: ' + invalid.join(', '), 'error');
                     showBoxNotif('Artikel Ghaib', 'Artikel tidak ditemukan di Mippedia ' + invalid.join(', ') + '. Cek kembali ejaannya.', 'error');
                     $btn.text('Sinkronkan Data').prop('disabled', false); return;
                     $btn.text('Sinkronkan Data').prop('disabled', false); return;
                 }
                 }
                 api.get({ action: 'query', prop: 'revisions', titles: dataPage, rvprop: 'content', formatversion: 2 }).done(function(res) {
               
                     var fullData = res.query.pages[0].revisions ? JSON.parse(res.query.pages[0].revisions[0].content) : {};
                 api.get({ action: 'query', prop: 'revisions', titles: dataPage, rvprop: 'content', formatversion: 2 }).done(function(r) {
                     if (isDel) delete fullData[pageTitle]; else fullData[pageTitle] = newData;
                     var full = r.query.pages[0].revisions ? JSON.parse(r.query.pages[0].revisions[0].content) : {};
                     api.postWithEditToken({ action: 'edit', title: dataPage, text: JSON.stringify(fullData, null, 2), summary: 'Update Sitelink: ' + pageTitle }).done(function() {
                     if (isDel) delete full[pageTitle]; else full[pageTitle] = newData;
                         showFancyNotif(isDel ? 'Koneksi Dihapus' : 'Data Terkoneksi!', 'success');
                   
                         setTimeout(function(){ location.reload(); }, 1200);
                     api.postWithEditToken({ action: 'edit', title: dataPage, text: JSON.stringify(full, null, 2), summary: 'Sync Sitelink: ' + pageTitle })
                    .done(function() {
                         showBoxNotif('Sinkron Berhasil!', 'Metadata ekosistem telah diperbarui secara permanen.', 'success');
                         setTimeout(function(){ location.reload(); }, 1500);
                    }).fail(function(c) {
                        showBoxNotif('Sistem Sibuk', 'Gagal menulis data ke server MediaWiki. Kode: ' + c, 'error');
                        $btn.text('Sinkronkan Data').prop('disabled', false);
                     });
                     });
                 });
                 });
Baris 273: Baris 282:
         });
         });


         $(document).on('click', '#mip-edit-links', function() { $('#mip-link-editor').slideToggle(); });
         $(document).on('click', '#mip-open-editor', function() { $('#mip-editor').slideToggle(); });
         loadData();
         loadLinks();
     });
     });
})();
})();

Revisi per 16 April 2026 11.27

/* ==========================================================
   🧠 MIPPEDIA DATA - SMART CONTEXT + AUTO THUMBNAIL
   Features: Smart Sync, Auto-Age, Auto-Bold, Smart Links,
             Dynamic Context, AND NEW: Auto Thumbnail Image
   ========================================================== */
(function() {
    $(document).ready(function() {
        var $descSection = $('#mip-desc-section'), $descBox = $('#mip-auto-description'),
            $sourceInfo = $('#mip-source-info'), $portalLinks = $('#mip-portal-links'),
            $projectPortal = $('#mip-project-portal');
        
        if (!$descBox.length) return;

        var pageTitle = mw.config.get('wgPageName').replace(/_/g, ' '),
            cacheKey = 'mip_smart_context_thumb_' + pageTitle, now = new Date().getTime(),
            currentYear = 2026;

        var projects = [
            { id: 'id', name: 'Mippedia bahasa Indonesia', url: 'https://id.mippedia.org/api.php', base: 'https://id.mippedia.org/wiki/', label: 'Bahasa Indonesia' },
            { id: 'en', name: 'Mippedia bahasa Inggris', url: 'https://en.mippedia.org/api.php', base: 'https://en.mippedia.org/wiki/', label: 'Bahasa Inggris' },
            { id: 'concise', name: 'Mippedia bahasa Indonesia ringkas', url: 'https://concise.mippedia.org/api.php', base: 'https://concise.mippedia.org/wiki/', label: 'Versi Ringkas' }
        ];

        // --- SEMUA FUNGSI AWAL (TETAP UTUH) ---
        function applyAutoBold(text) { return text.replace(new RegExp('(' + pageTitle + ')', 'gi'), '<strong>$1</strong>'); }
        function cleanExtract(text) { return text.replace(/\[\d+\]/g, '').replace(/\{\{[^}]+\}\}/g, '').replace(/\(\s*\)/g, '').replace(/\s\s+/g, ' ').trim(); }
        
        function applyDynamicContext(text) {
            var date = new Date();
            var hari = ["Minggu", "Senin", "Selasa", "Rabu", "Kamis", "Jumat", "Sabtu"];
            var bulan = ["Januari", "Februari", "Maret", "April", "Mei", "Juni", "Juli", "Agustus", "September", "Oktober", "November", "Desember"];
            var tglSkrg = hari[date.getDay()] + ", " + date.getDate() + " " + bulan[date.getMonth()] + " " + date.getFullYear();

            return text
                .replace(/(hari ini|saat ini|sekarang)/gi, '$1 (' + tglSkrg + ')')
                .replace(/(\d{1,2}\s(?:Januari|Februari|Maret|April|Mei|Juni|Juli|Agustus|September|Oktober|November|Desember)\s(\d{4}))/gi, function(m, f, y) { 
                    return f + " <span style='color: #444; font-weight: bold;'>– usia " + (currentYear - parseInt(y)) + " tahun</span>"; 
                })
                .replace(/(sejak|tahun)\s(\d{4})/gi, function(m, k, y) {
                    var gap = currentYear - parseInt(y);
                    if (gap > 0 && gap < 100) return k + " " + y + " <small style='color: #888;'>(" + gap + " thn lalu)</small>";
                    return m;
                });
        }

        function applySummary(text) {
            var limit = 250;
            if (text.length <= limit) return '<span>' + text + '</span>';
            return '<span>' + text.substring(0, limit) + '</span><span class="mip-dots">... </span><span class="mip-more" style="display:none;">' + text.substring(limit) + '</span><span class="mip-read-btn" style="color: #6a5acd; cursor: pointer; font-weight: bold; margin-left: 5px;">Baca selengkapnya</span>';
        }

        function buildSmartLinks(currentId) {
            $portalLinks.empty();
            var validLinks = [];
            var checkRequests = projects.filter(p => p.id !== currentId).map(function(p) {
                return $.ajax({ url: p.url, data: { action: 'query', titles: pageTitle, format: 'json', origin: '*' }, dataType: 'json' })
                    .then(function(data) { if (data.query.pages["-1"] === undefined) { validLinks.push('<a href="' + p.base + encodeURIComponent(pageTitle) + '" target="_blank" style="color: #6a5acd; text-decoration: underline; font-weight: bold;">' + p.label + '</a>'); } });
            });
            $.when.apply($, checkRequests).done(function() {
                if (validLinks.length > 0) { $portalLinks.html(validLinks.join(' <span style="color:#ccc; margin: 0 5px;">•</span> ')); $projectPortal.fadeIn(); }
            });
        }

        // --- SISTEM SMART CHECK DENGAN SUPPORT THUMBNAIL ---
        var cached = JSON.parse(localStorage.getItem(cacheKey) || "{}");

        function init() {
            var pCheck = projects[Math.floor(Math.random() * projects.length)];
            $.ajax({
                url: pCheck.url,
                data: { action: 'query', prop: 'revisions|pageimages', rvprop: 'timestamp', piprop: 'thumbnail', pithumbsize: 150, titles: pageTitle, format: 'json', origin: '*' },
                dataType: 'json',
                success: function(res) {
                    var pg = res.query.pages, id = Object.keys(pg)[0];
                    var latestTS = (id != "-1") ? pg[id].revisions[0].timestamp : "0";
                    // Ambil thumbnail kalau ada
                    var thumbUrl = (id != "-1" && pg[id].thumbnail) ? pg[id].thumbnail.source : "";

                    if (cached.content && cached.ts === latestTS) {
                        renderAll(cached.p, cached.content, cached.isTrans, cached.orig, cached.img);
                    } else { 
                        fetchFreshData(latestTS, thumbUrl); 
                    }
                },
                error: function() { if (cached.content) renderAll(cached.p, cached.content, cached.isTrans, cached.orig, cached.img); }
            });
        }

        function fetchFreshData(newTS, img) {
            var found = false;
            var shuffled = projects.sort(() => Math.random() - 0.5);
            shuffled.forEach(function(p) {
                if (found) return;
                $.ajax({
                    url: p.url,
                    data: { action: 'query', prop: 'extracts', exintro: 1, explaintext: 1, titles: pageTitle, format: 'json', origin: '*' },
                    dataType: 'json',
                    success: function(data) {
                        if (found) return;
                        var pages = data.query.pages, pageId = Object.keys(pages)[0];
                        if (pageId != "-1" && pages[pageId].extract) {
                            found = true;
                            var extract = cleanExtract(pages[pageId].extract);
                            if (p.id === 'en') {
                                $.ajax({ url: "https://translate.googleapis.com/translate_a/single?client=gtx&sl=en&tl=id&dt=t&q=" + encodeURIComponent(extract), success: function(res) { 
                                    var trans = ""; res[0].forEach(s => { if (s[0]) trans += s[0]; });
                                    finalize(p, trans, true, extract, newTS, img);
                                }});
                            } else { 
                                finalize(p, extract, false, "", newTS, img); 
                            }
                        }
                    }
                });
            });
        }

        function finalize(p, c, t, o, ts, img) {
            localStorage.setItem(cacheKey, JSON.stringify({p:p, content:c, isTrans:t, orig:o, ts:ts, img:img}));
            renderAll(p, c, t, o, img);
        }

        function renderAll(p, currentText, isTranslated, originalText, img) {
            var processedText = applyAutoBold(applyDynamicContext(currentText));
            
            // Layouting Thumbnail (Tetap ringan: hanya render jika gambar ada)
            var thumbHtml = img ? '<div style="float: right; margin-left: 15px; margin-bottom: 10px; border: 1px solid #ddd; padding: 3px; background: #fff; border-radius: 4px; box-shadow: 0 2px 4px rgba(0,0,0,0.1);"><img src="'+img+'" style="max-width: 100px; display: block; height: auto;"></div>' : '';

            $descSection.show();
            // Masukkan thumbnail ke dalam konten box
            $descBox.show().html(thumbHtml + applySummary(processedText) + '<div style="clear:both;"></div>');
            
            var footer = '<div style="font-size: 0.9em; color: #777;">Sumber Dari : <a href="' + p.base + encodeURIComponent(pageTitle) + '" target="_blank" style="color: #6a5acd; font-weight: bold; text-decoration: none;">' + p.name + '.</a>';
            if (isTranslated) { footer += '<br><span style="font-size: 0.85em; font-style: italic;">(Diterjemahkan secara otomatis)</span> <span id="mip-toggle-orig" style="color: #6a5acd; cursor: pointer; text-decoration: underline; margin-left: 5px;">Tampilkan versi asli</span>'; }
            $sourceInfo.show().html(footer);
            
            buildSmartLinks(p.id);
            
            // Handlers
            $(document).off('click', '.mip-read-btn').on('click', '.mip-read-btn', function() { $(this).hide(); $('.mip-dots').hide(); $('.mip-more').fadeIn(); });
            $(document).off('click', '#mip-toggle-orig').on('click', '#mip-toggle-orig', function() {
                var isOrig = ($(this).text() === 'Tampilkan versi asli');
                var textToDisplay = applyAutoBold(applyDynamicContext(isOrig ? originalText : currentText));
                // Pastikan thumbnail tetap ada saat toggle versi asli
                $descBox.html(thumbHtml + applySummary(textToDisplay) + '<div style="clear:both;"></div>');
                $(this).text(isOrig ? 'Tampilkan terjemahan' : 'Tampilkan versi asli');
            });
        }

        init();
    });
})();

/* ==========================================================
   🚀 MIPPEDIA DATA - TOTAL INTERFACE BUNDLE (FINAL V5)
   Fungsi: Header, Connector (Admin-Only), & Footer Lisensi
   Aturan: Hanya Namespace 0 & Bukan Halaman Utama
   ========================================================== */

(function() {
    $(document).ready(function() {
        // --- 1. FILTER GLOBAL NAMESPACE & HALAMAN UTAMA ---
        if (mw.config.get('wgNamespaceNumber') !== 0 || 
            mw.config.get('wgIsMainPage') || 
            mw.config.get('wgAction') !== 'view') return;

        var pageTitle = mw.config.get('wgPageName');
        var dataPage = 'MediaWiki:Sitelinks-Data.json';
        var userGroups = mw.config.get('wgUserGroups');
        var isAdmin = userGroups.includes('sysop') || userGroups.includes('interface-admin');

        var projects = {
            'id': { name: 'ID', url: 'https://id.mippedia.org/api.php', base: 'https://id.mippedia.org/wiki/', color: '#6a5acd' },
            'en': { name: 'EN', url: 'https://en.mippedia.org/api.php', base: 'https://en.mippedia.org/wiki/', color: '#4169e1' },
            'concise': { name: 'Concise', url: 'https://concise.mippedia.org/api.php', base: 'https://concise.mippedia.org/wiki/', color: '#20b2aa' }
        };

        // --- 2. FUNGSI NOTIFIKASI KOTAK PROFESIONAL ---
        function showBoxNotif(title, msg, type) {
            $('.mip-box-notif').remove();
            var theme = type === 'success' ? {bg:'#e6fffa', border:'#38b2ac', text:'#234e52', icon:'✅'} : 
                        {bg:'#fff5f5', border:'#f56565', text:'#742a2a', icon:'🚫'};
            
            var $notif = $('<div class="mip-box-notif" style="position:fixed; top:25px; right:25px; width:340px; background:'+theme.bg+'; border-left:6px solid '+theme.border+'; padding:16px; z-index:10001; border-radius:10px; box-shadow:0 12px 30px rgba(0,0,0,0.18); font-family:sans-serif; animation:mipSlideIn 0.4s ease-out;">' +
                '<div style="display:flex; align-items:center; gap:12px; margin-bottom:6px;">' +
                    '<span style="font-size:18px;">'+theme.icon+'</span>' +
                    '<strong style="color:'+theme.text+'; font-size:14px; letter-spacing:-0.2px;">'+title+'</strong>' +
                '</div>' +
                '<p style="margin:0; font-size:12.5px; color:'+theme.text+'; line-height:1.5; opacity:0.9;">'+msg+'</p>' +
            '</div>').appendTo('body');

            setTimeout(function() { $notif.fadeOut(function(){ $(this).remove(); }); }, 5000);
        }

        // --- 3. UI GENERATOR (HEADER, CONNECTOR, FOOTER) ---
        
        // A. Header (Paling Atas)
        var headerHtml = '<div id="mip-header" style="background:#f8f9fa; border:1px solid #ddd; border-left:5px solid #6a5acd; padding:15px; margin-bottom:20px; border-radius:4px;">' +
            '<div style="display:flex; align-items:center; gap:10px; margin-bottom:5px;">' +
            '<span style="background:#6a5acd; color:white; padding:2px 8px; border-radius:3px; font-size:10px; font-weight:bold; text-transform:uppercase;">Basis Data</span>' +
            '<strong style="color:#1a1a1a; font-size:15px;">Mippedia Data Project</strong></div>' +
            '<p style="margin:0; font-size:12.5px; color:#555; line-height:1.5;">Halaman ini merupakan repositori data terstruktur dari ekosistem <strong>Mippedia</strong> untuk keperluan referensi terbuka.</p></div>';

        // B. Connector (Bawah Konten)
        var editBtn = '<span id="mip-open-editor" style="cursor:pointer; font-size:18px; color:#6a5acd; opacity:0.7; transition:0.2s;" title="Edit Sitelink">✎</span>';
        var connectorHtml = '<div id="mip-connector" style="background:#fff; border:1px solid #e0e0e0; border-radius:14px; padding:20px; margin-top:25px; box-shadow:0 4px 15px rgba(0,0,0,0.05);">' +
            '<div style="display:flex; justify-content:space-between; align-items:center; margin-bottom:18px;"><strong style="font-size:14px; color:#1a1a1a;">🔗 Koneksi Ekosistem Mippedia</strong>'+editBtn+'</div>' +
            '<div id="mip-display" style="display:flex; gap:10px; flex-wrap:wrap;"><small style="color:#999;">Sinkronisasi...</small></div>' +
            '<div id="mip-editor" style="display:none; margin-top:18px; border-top:1px solid #f0f0f0; padding-top:18px;">' +
                Object.keys(projects).map(p => '<div style="margin-bottom:12px; display:flex; gap:10px;"><span style="width:65px; font-size:10px; font-weight:800; background:#f5f5f5; display:flex; align-items:center; justify-content:center; border-radius:8px; color:#666;">'+projects[p].name+'</span><input type="text" id="in-'+p+'" placeholder="Judul artikel..." style="flex:1; padding:9px; font-size:13px; border:1.5px solid #eee; border-radius:8px; outline:none;"></div>').join('') +
                '<button id="mip-save" style="width:100%; padding:12px; background:#6a5acd; color:white; border:none; border-radius:10px; font-size:13px; font-weight:bold; cursor:pointer; transition:0.3s;">Sinkronkan Data</button>' +
            '</div></div>';

        // C. Footer (Paling Bawah)
        var footerHtml = '<div id="mip-footer" style="background:#fff; border:1px dashed #ccc; padding:15px; margin-top:25px; text-align:center; border-radius:8px;">' +
            '<div style="font-size:11px; color:#888; font-weight:bold; text-transform:uppercase; margin-bottom:5px;">Lisensi & Atribusi</div>' +
            '<p style="margin:0; font-size:12px; color:#666;">Data tersedia di bawah lisensi <a href="https://creativecommons.org/licenses/by-sa/4.0/" target="_blank" style="color:#6a5acd; text-decoration:none; font-weight:bold;">CC BY-SA 4.0</a>.</p></div>';

        // Injeksi ke Halaman
        $('#mw-content-text').prepend(headerHtml);
        $('#mw-content-text').append(connectorHtml).append(footerHtml);

        // --- 4. LOGIC SISTEM ---
        function loadLinks() {
            new mw.Api().get({ action: 'query', prop: 'revisions', titles: dataPage, rvprop: 'content', formatversion: 2 }).done(function(d) {
                var json = d.query.pages[0].revisions ? JSON.parse(d.query.pages[0].revisions[0].content) : {};
                var data = json[pageTitle] || {};
                var $box = $('#mip-display').empty();
                var found = false;
                Object.keys(projects).forEach(p => {
                    if (data[p]) {
                        found = true;
                        $box.append('<a style="background:'+projects[p].color+'; color:white; padding:7px 15px; border-radius:8px; font-size:11px; font-weight:bold; text-decoration:none; transition:0.2s;" href="'+projects[p].base+encodeURIComponent(data[p])+'" target="_blank">Mippedia '+projects[p].name+': '+data[p]+'</a>');
                        $('#in-' + p).val(data[p]);
                    }
                });
                if (!found) $box.html('<em style="color:#bbb; font-size:12px;">Entitas ini belum terhubung ke ekosistem.</em>');
            });
        }

        $(document).on('click', '#mip-save', function() {
            // PROTEKSI: Tolak jika bukan Admin
            if (!isAdmin) {
                showBoxNotif('Akses Terbatas', 'Hanya grup <b>Pengurus</b> yang dapat melakukan sinkronisasi metadata permanen.', 'error');
                return;
            }

            var $btn = $(this).text('Memverifikasi Judul...').prop('disabled', true);
            var api = new mw.Api();
            var newData = {}, promises = [], invalid = [], isDel = true;

            Object.keys(projects).forEach(p => {
                var v = $('#in-' + p).val().trim();
                if (v) {
                    isDel = false;
                    promises.push($.ajax({ url: projects[p].url, data: { action: 'query', titles: v, format: 'json', origin: '*' }, dataType: 'json' }).done(function(res){
                        if (res.query.pages["-1"]) invalid.push(projects[p].name); else newData[p] = v;
                    }));
                }
            });

            $.when.apply($, promises).then(function() {
                if (invalid.length > 0) {
                    showBoxNotif('Artikel Ghaib', 'Artikel tidak ditemukan di Mippedia ' + invalid.join(', ') + '. Cek kembali ejaannya.', 'error');
                    $btn.text('Sinkronkan Data').prop('disabled', false); return;
                }
                
                api.get({ action: 'query', prop: 'revisions', titles: dataPage, rvprop: 'content', formatversion: 2 }).done(function(r) {
                    var full = r.query.pages[0].revisions ? JSON.parse(r.query.pages[0].revisions[0].content) : {};
                    if (isDel) delete full[pageTitle]; else full[pageTitle] = newData;
                    
                    api.postWithEditToken({ action: 'edit', title: dataPage, text: JSON.stringify(full, null, 2), summary: 'Sync Sitelink: ' + pageTitle })
                    .done(function() {
                        showBoxNotif('Sinkron Berhasil!', 'Metadata ekosistem telah diperbarui secara permanen.', 'success');
                        setTimeout(function(){ location.reload(); }, 1500);
                    }).fail(function(c) {
                        showBoxNotif('Sistem Sibuk', 'Gagal menulis data ke server MediaWiki. Kode: ' + c, 'error');
                        $btn.text('Sinkronkan Data').prop('disabled', false);
                    });
                });
            });
        });

        $(document).on('click', '#mip-open-editor', function() { $('#mip-editor').slideToggle(); });
        loadLinks();
    });
})();