MediaWiki:Common.js: Perbedaan antara revisi
Tampilan
Tidak ada ringkasan suntingan Tanda: Suntingan perangkat seluler Suntingan peramban seluler |
Tidak ada ringkasan suntingan Tanda: Suntingan perangkat seluler Suntingan peramban seluler |
||
| Baris 203: | Baris 203: | ||
/* ========================================================== | /* ========================================================== | ||
🔗 MIPPEDIA DATA - | 🔗 MIPPEDIA DATA - ULTIMATE SITELINK CONNECTOR (PRO VERSION) | ||
Status: | Status: INDUSTRIAL GRADE (Anti-Crash & Protection Aware) | ||
========================================================== */ | ========================================================== */ | ||
(function() { | (function() { | ||
$(document).ready(function() { | $(document).ready(function() { | ||
// Hanya jalan di halaman artikel utama | |||
if (mw.config.get('wgAction') !== 'view' || mw.config.get('wgNamespaceNumber') !== 0) return; | if (mw.config.get('wgAction') !== 'view' || mw.config.get('wgNamespaceNumber') !== 0) return; | ||
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 isProtected = mw.config.get('wgRestrictionEdit') && mw.config.get('wgRestrictionEdit').includes('sysop'); | |||
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 219: | Baris 223: | ||
}; | }; | ||
// 1. Tampilkan Box Utama | |||
var editIcon = (isProtected && !isAdmin) ? '' : '<span id="mip-edit-links" style="cursor: pointer; color: #6a5acd; font-size: 18px; transition: 0.3s;" title="Edit Koneksi">✎</span>'; | |||
var containerHtml = | var containerHtml = | ||
'<div id="mip-sitelink-box" style="background: #fff; border: 1px solid #ddd; border-radius: | '<div id="mip-sitelink-box" style="background: #fff; border: 1px solid #ddd; border-radius: 12px; padding: 18px; margin-top: 25px; font-family: sans-serif; box-shadow: 0 4px 15px rgba(0,0,0,0.05);">' + | ||
'<div style="display: flex; justify-content: space-between; align-items: center; margin-bottom: | '<div style="display: flex; justify-content: space-between; align-items: center; margin-bottom: 15px;">' + | ||
'<strong style="font-size: | '<strong style="font-size: 14px; color: #333; letter-spacing: 0.3px;">🔗 Koneksi Ekosistem Mippedia</strong>' + | ||
editIcon + | |||
'</div>' + | '</div>' + | ||
'<div id="mip-link-display" style="display: flex; gap: | '<div id="mip-link-display" style="display: flex; gap: 10px; flex-wrap: wrap;"><div class="mip-loader"></div></div>' + | ||
'<div id="mip-link-editor" style="display: none; margin-top: | '<div id="mip-link-editor" style="display: none; margin-top: 15px; border-top: 1px solid #eee; padding-top: 15px; animation: fadeIn 0.4s;">' + | ||
Object.keys(projects).map(p => | Object.keys(projects).map(p => | ||
'<div style="margin-bottom: | '<div style="margin-bottom: 10px; display: flex; gap: 8px;">' + | ||
'<span style="width: | '<span style="width: 60px; font-size: 10px; font-weight: 800; background: #f0f0f0; display: flex; align-items: center; justify-content: center; border-radius: 6px; color: #555;">' + projects[p].name + '</span>' + | ||
'<input type="text" id="input-' + p + '" placeholder="Judul artikel..." style="flex: 1; padding: | '<input type="text" id="input-' + p + '" placeholder="Judul artikel..." style="flex: 1; padding: 7px 12px; font-size: 13px; border: 1px solid #ddd; border-radius: 6px; outline: none; transition: 0.2s;">' + | ||
'</div>' | '</div>' | ||
).join('') + | ).join('') + | ||
'<button id="mip-save-links" style="width: 100%; margin-top: | '<button id="mip-save-links" style="width: 100%; margin-top: 10px; padding: 10px; background: #6a5acd; color: white; border: none; border-radius: 8px; font-size: 13px; cursor: pointer; font-weight: bold; transition: 0.3s;">Sinkronkan Data</button>' + | ||
'</div>' + | '</div>' + | ||
'</div>'; | '</div>' + | ||
'<style>' + | |||
'.mip-loader { border: 2px solid #f3f3f3; border-top: 2px solid #6a5acd; border-radius: 50%; width: 15px; height: 15px; animation: spin 1s linear infinite; }' + | |||
'@keyframes spin { 0% { transform: rotate(0deg); } 100% { transform: rotate(360deg); } }' + | |||
'@keyframes fadeIn { from { opacity: 0; transform: translateY(-5px); } to { opacity: 1; transform: translateY(0); } }' + | |||
'#mip-save-links:hover { background: #5a4db8; transform: translateY(-1px); box-shadow: 0 4px 8px rgba(106, 90, 205, 0.3); }' + | |||
'#mip-edit-links:hover { transform: scale(1.2) rotate(15deg); }' + | |||
'.mip-pill { text-decoration: none !important; color: white !important; padding: 6px 14px; border-radius: 20px; font-size: 11px; font-weight: bold; transition: 0.3s; display: flex; align-items: center; box-shadow: 0 2px 4px rgba(0,0,0,0.1); }' + | |||
'.mip-pill:hover { transform: translateY(-2px); box-shadow: 0 4px 8px rgba(0,0,0,0.15); opacity: 0.9; }' + | |||
'</style>'; | |||
$('#mip-data-footer').before(containerHtml); | $('#mip-data-footer').before(containerHtml); | ||
// | // 2. Fungsi Load Data (Global & Public) | ||
function | function loadData() { | ||
$.getJSON(mw.util.wikiScript('api'), { | |||
var | action: 'query', prop: 'revisions', titles: dataPage, rvprop: 'content', formatversion: 2, origin: location.origin | ||
renderLinks( | }).done(function(data) { | ||
var raw = data.query.pages[0].revisions ? data.query.pages[0].revisions[0].content : "{}"; | |||
var fullData = JSON.parse(raw); | |||
renderLinks(fullData[pageTitle] || {}); | |||
}).fail(function() { | |||
$('#mip-link-display').html('<small style="color:red;">Gagal memuat data server.</small>'); | |||
}); | }); | ||
} | } | ||
function renderLinks( | function renderLinks(links) { | ||
var $display = $('#mip-link-display') | var $display = $('#mip-link-display').empty(); | ||
var count = 0; | |||
var | for (var p in projects) { | ||
if (links[p]) { | |||
for (var p in | $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>'); | ||
if ( | $('#input-' + p).val(links[p]); | ||
count++; | |||
$display.append('<a href="' + projects[p].base + encodeURIComponent( | |||
$('#input-' + p).val( | |||
} | } | ||
} | } | ||
if ( | if (count === 0) $display.html('<em style="color: #bbb; font-size: 12px;">Entitas ini belum terhubung ke artikel manapun.</em>'); | ||
} | } | ||
// --- | // 3. Fungsi Animasi Sukses (Pop-up Profesional) | ||
function showNotif(msg, type) { | |||
var color = type === 'success' ? '#2ecc71' : '#e74c3c'; | |||
var $notif = $('<div style="position:fixed; top:20px; left:50%; transform:translateX(-50%); background:'+color+'; color:white; padding:12px 25px; border-radius:50px; font-size:14px; font-weight:bold; z-index:10000; box-shadow:0 10px 20px rgba(0,0,0,0.2); animation: fadeIn 0.3s;">'+msg+'</div>').appendTo('body'); | |||
setTimeout(function() { $notif.fadeOut(function() { $(this).remove(); }); }, 3000); | |||
} | |||
// 4. Sinkronisasi (Save/Delete) | |||
$(document).on('click', '#mip-save-links', function() { | $(document).on('click', '#mip-save-links', function() { | ||
var btn = $(this) | var btn = $(this).text('Proses Mendata...').prop('disabled', true); | ||
var api = new mw.Api(); | var api = new mw.Api(); | ||
api.get({ action: 'query', prop: 'revisions', titles: dataPage, rvprop: 'content', formatversion: 2 }).done(function(res) { | 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) : {}; | var fullData = res.query.pages[0].revisions ? JSON.parse(res.query.pages[0].revisions[0].content) : {}; | ||
var currentEntry = {}; | var currentEntry = {}; | ||
var isDeleting = true; | |||
Object.keys(projects).forEach(p => { | Object.keys(projects).forEach(p => { | ||
var val = $('#input-' + p).val().trim(); | var val = $('#input-' + p).val().trim(); | ||
if (val) currentEntry[p] = val; | if (val) { | ||
currentEntry[p] = val; | |||
isDeleting = false; | |||
} | |||
}); | }); | ||
fullData[pageTitle] = currentEntry; | if (isDeleting) delete fullData[pageTitle]; | ||
else fullData[pageTitle] = currentEntry; | |||
api.postWithEditToken({ | api.postWithEditToken({ | ||
action: 'edit', | action: 'edit', title: dataPage, text: JSON.stringify(fullData, null, 2), | ||
summary: 'Update Sitelink: ' + pageTitle + ' [Automated System]', format: 'json' | |||
summary: ' | |||
}).done(function() { | }).done(function() { | ||
showNotif(isDeleting ? '✅ Koneksi berhasil dihapus!' : '🚀 Berhasil menghubungkan data!', 'success'); | |||
location.reload(); | setTimeout(function() { location.reload(); }, 1000); | ||
}).fail(function(code) { | |||
showNotif('❌ Gagal: ' + code + '. Pastikan Anda login atau memiliki izin.', 'error'); | |||
btn.text('Sinkronkan Data').prop('disabled', false); | |||
}); | }); | ||
}); | }); | ||
| Baris 294: | Baris 324: | ||
$(document).on('click', '#mip-edit-links', function() { $('#mip-link-editor').slideToggle(); }); | $(document).on('click', '#mip-edit-links', function() { $('#mip-link-editor').slideToggle(); }); | ||
loadData(); | |||
}); | }); | ||
})(); | })(); | ||
Revisi per 16 April 2026 10.18
/* ==========================================================
🧠 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 - TOP PERMANENT HEADER
Fungsi: Menampilkan informasi identitas project di atas data
========================================================== */
(function() {
$(document).ready(function() {
// Cek apakah ini halaman artikel (bukan halaman edit/special)
if (mw.config.get('wgAction') !== 'view' || mw.config.get('wgNamespaceNumber') !== 0) return;
var headerHtml =
'<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;">' +
'<div style="display: flex; align-items: center; margin-bottom: 5px;">' +
'<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>' +
'<strong style="color: #333; font-size: 14px;">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>. ' +
'Informasi yang ditampilkan di bawah ini adalah metadata entitas yang disinkronisasi secara otomatis untuk keperluan referensi terbuka.' +
'</p>' +
'</div>';
// Sisipkan di atas konten utama (di atas infobox)
$('#mw-content-text').prepend(headerHtml);
});
})();
/* ==========================================================
🛡️ MIPPEDIA DATA - BOTTOM PERMANENT FOOTER
Fungsi: Menampilkan informasi lisensi di bawah data
========================================================== */
(function() {
$(document).ready(function() {
// Cek apakah ini halaman artikel
if (mw.config.get('wgAction') !== 'view' || mw.config.get('wgNamespaceNumber') !== 0) return;
var footerHtml =
'<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; 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; 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>';
// Sisipkan di paling bawah konten utama
$('#mw-content-text').append(footerHtml);
});
})();
/* ==========================================================
🔗 MIPPEDIA DATA - ULTIMATE SITELINK CONNECTOR (PRO VERSION)
Status: INDUSTRIAL GRADE (Anti-Crash & Protection Aware)
========================================================== */
(function() {
$(document).ready(function() {
// Hanya jalan di halaman artikel utama
if (mw.config.get('wgAction') !== 'view' || mw.config.get('wgNamespaceNumber') !== 0) return;
var pageTitle = mw.config.get('wgPageName');
var dataPage = 'MediaWiki:Sitelinks-Data.json';
var isProtected = mw.config.get('wgRestrictionEdit') && mw.config.get('wgRestrictionEdit').includes('sysop');
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' }
};
// 1. Tampilkan Box Utama
var editIcon = (isProtected && !isAdmin) ? '' : '<span id="mip-edit-links" style="cursor: pointer; color: #6a5acd; font-size: 18px; transition: 0.3s;" title="Edit Koneksi">✎</span>';
var containerHtml =
'<div id="mip-sitelink-box" style="background: #fff; border: 1px solid #ddd; border-radius: 12px; padding: 18px; margin-top: 25px; font-family: sans-serif; box-shadow: 0 4px 15px rgba(0,0,0,0.05);">' +
'<div style="display: flex; justify-content: space-between; align-items: center; margin-bottom: 15px;">' +
'<strong style="font-size: 14px; color: #333; letter-spacing: 0.3px;">🔗 Koneksi Ekosistem Mippedia</strong>' +
editIcon +
'</div>' +
'<div id="mip-link-display" style="display: flex; gap: 10px; flex-wrap: wrap;"><div class="mip-loader"></div></div>' +
'<div id="mip-link-editor" style="display: none; margin-top: 15px; border-top: 1px solid #eee; padding-top: 15px; animation: fadeIn 0.4s;">' +
Object.keys(projects).map(p =>
'<div style="margin-bottom: 10px; display: flex; gap: 8px;">' +
'<span style="width: 60px; font-size: 10px; font-weight: 800; background: #f0f0f0; display: flex; align-items: center; justify-content: center; border-radius: 6px; color: #555;">' + projects[p].name + '</span>' +
'<input type="text" id="input-' + p + '" placeholder="Judul artikel..." style="flex: 1; padding: 7px 12px; font-size: 13px; border: 1px solid #ddd; border-radius: 6px; outline: none; transition: 0.2s;">' +
'</div>'
).join('') +
'<button id="mip-save-links" style="width: 100%; margin-top: 10px; padding: 10px; background: #6a5acd; color: white; border: none; border-radius: 8px; font-size: 13px; cursor: pointer; font-weight: bold; transition: 0.3s;">Sinkronkan Data</button>' +
'</div>' +
'</div>' +
'<style>' +
'.mip-loader { border: 2px solid #f3f3f3; border-top: 2px solid #6a5acd; border-radius: 50%; width: 15px; height: 15px; animation: spin 1s linear infinite; }' +
'@keyframes spin { 0% { transform: rotate(0deg); } 100% { transform: rotate(360deg); } }' +
'@keyframes fadeIn { from { opacity: 0; transform: translateY(-5px); } to { opacity: 1; transform: translateY(0); } }' +
'#mip-save-links:hover { background: #5a4db8; transform: translateY(-1px); box-shadow: 0 4px 8px rgba(106, 90, 205, 0.3); }' +
'#mip-edit-links:hover { transform: scale(1.2) rotate(15deg); }' +
'.mip-pill { text-decoration: none !important; color: white !important; padding: 6px 14px; border-radius: 20px; font-size: 11px; font-weight: bold; transition: 0.3s; display: flex; align-items: center; box-shadow: 0 2px 4px rgba(0,0,0,0.1); }' +
'.mip-pill:hover { transform: translateY(-2px); box-shadow: 0 4px 8px rgba(0,0,0,0.15); opacity: 0.9; }' +
'</style>';
$('#mip-data-footer').before(containerHtml);
// 2. Fungsi Load Data (Global & Public)
function loadData() {
$.getJSON(mw.util.wikiScript('api'), {
action: 'query', prop: 'revisions', titles: dataPage, rvprop: 'content', formatversion: 2, origin: location.origin
}).done(function(data) {
var raw = data.query.pages[0].revisions ? data.query.pages[0].revisions[0].content : "{}";
var fullData = JSON.parse(raw);
renderLinks(fullData[pageTitle] || {});
}).fail(function() {
$('#mip-link-display').html('<small style="color:red;">Gagal memuat data server.</small>');
});
}
function renderLinks(links) {
var $display = $('#mip-link-display').empty();
var count = 0;
for (var p in projects) {
if (links[p]) {
$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>');
$('#input-' + p).val(links[p]);
count++;
}
}
if (count === 0) $display.html('<em style="color: #bbb; font-size: 12px;">Entitas ini belum terhubung ke artikel manapun.</em>');
}
// 3. Fungsi Animasi Sukses (Pop-up Profesional)
function showNotif(msg, type) {
var color = type === 'success' ? '#2ecc71' : '#e74c3c';
var $notif = $('<div style="position:fixed; top:20px; left:50%; transform:translateX(-50%); background:'+color+'; color:white; padding:12px 25px; border-radius:50px; font-size:14px; font-weight:bold; z-index:10000; box-shadow:0 10px 20px rgba(0,0,0,0.2); animation: fadeIn 0.3s;">'+msg+'</div>').appendTo('body');
setTimeout(function() { $notif.fadeOut(function() { $(this).remove(); }); }, 3000);
}
// 4. Sinkronisasi (Save/Delete)
$(document).on('click', '#mip-save-links', function() {
var btn = $(this).text('Proses Mendata...').prop('disabled', true);
var api = new mw.Api();
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) : {};
var currentEntry = {};
var isDeleting = true;
Object.keys(projects).forEach(p => {
var val = $('#input-' + p).val().trim();
if (val) {
currentEntry[p] = val;
isDeleting = false;
}
});
if (isDeleting) delete fullData[pageTitle];
else fullData[pageTitle] = currentEntry;
api.postWithEditToken({
action: 'edit', title: dataPage, text: JSON.stringify(fullData, null, 2),
summary: 'Update Sitelink: ' + pageTitle + ' [Automated System]', format: 'json'
}).done(function() {
showNotif(isDeleting ? '✅ Koneksi berhasil dihapus!' : '🚀 Berhasil menghubungkan data!', 'success');
setTimeout(function() { location.reload(); }, 1000);
}).fail(function(code) {
showNotif('❌ Gagal: ' + code + '. Pastikan Anda login atau memiliki izin.', 'error');
btn.text('Sinkronkan Data').prop('disabled', false);
});
});
});
$(document).on('click', '#mip-edit-links', function() { $('#mip-link-editor').slideToggle(); });
loadData();
});
})();