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 153: | Baris 153: | ||
/* ========================================================== | /* ========================================================== | ||
🚀 MIPPEDIA DATA - TOTAL INTERFACE | 🚀 MIPPEDIA DATA - TOTAL INTERFACE (FINAL BUNDLE V6) | ||
Status: MOBILE OPTIMIZED & SMART PERMISSION | |||
Rules: Namespace 0, Not Main Page, Responsive Notif | |||
========================================================== */ | ========================================================== */ | ||
(function() { | (function() { | ||
$(document).ready(function() { | $(document).ready(function() { | ||
// --- 1. FILTER GLOBAL | // --- 1. FILTER GLOBAL --- | ||
if (mw.config.get('wgNamespaceNumber') !== 0 || | if (mw.config.get('wgNamespaceNumber') !== 0 || | ||
mw.config.get('wgIsMainPage') || | mw.config.get('wgIsMainPage') || | ||
| Baris 169: | Baris 169: | ||
var userGroups = mw.config.get('wgUserGroups'); | var userGroups = mw.config.get('wgUserGroups'); | ||
var isAdmin = userGroups.includes('sysop') || userGroups.includes('interface-admin'); | var isAdmin = userGroups.includes('sysop') || userGroups.includes('interface-admin'); | ||
// Cek apakah halaman ini dilindungi (protected) | |||
var isProtected = mw.config.get('wgRestrictionEdit') && mw.config.get('wgRestrictionEdit').includes('sysop'); | |||
var projects = { | var projects = { | ||
| Baris 176: | Baris 179: | ||
}; | }; | ||
// --- 2. | // --- 2. NOTIFIKASI KOTAK (RESPONSIVE HP & PC) --- | ||
function showBoxNotif(title, msg, type) { | function showBoxNotif(title, msg, type) { | ||
$('.mip-box-notif').remove(); | $('.mip-box-notif').remove(); | ||
| Baris 182: | Baris 185: | ||
{bg:'#fff5f5', border:'#f56565', text:'#742a2a', icon:'🚫'}; | {bg:'#fff5f5', border:'#f56565', text:'#742a2a', icon:'🚫'}; | ||
var $notif = $('<div class="mip-box-notif" style="position:fixed; | // Logika Responsif via JS: Tengah di HP, Pojok di PC | ||
var isMobile = $(window).width() < 768; | |||
var positionStyle = isMobile ? 'left:10px; right:10px; top:20px;' : 'right:25px; top:25px; width:340px;'; | |||
var $notif = $('<div class="mip-box-notif" style="position:fixed; '+positionStyle+' background:'+theme.bg+'; border-left:6px solid '+theme.border+'; padding:16px; z-index:99999; border-radius:12px; box-shadow:0 15px 35px rgba(0,0,0,0.2); font-family:sans-serif; animation:mipSlideIn 0.4s ease-out;">' + | |||
'<div style="display:flex; align-items:center; gap:12px; margin-bottom:6px;">' + | '<div style="display:flex; align-items:center; gap:12px; margin-bottom:6px;">' + | ||
'<span style="font-size:18px;">'+theme.icon+'</span>' + | '<span style="font-size:18px;">'+theme.icon+'</span>' + | ||
'<strong style="color:'+theme.text+'; font-size:14px | '<strong style="color:'+theme.text+'; font-size:14px;">'+title+'</strong>' + | ||
'</div>' + | '</div>' + | ||
'<p style="margin:0; font-size:12.5px; color:'+theme.text+'; line-height:1.5 | '<p style="margin:0; font-size:12.5px; color:'+theme.text+'; line-height:1.5;">'+msg+'</p>' + | ||
'</div>').appendTo('body'); | '</div>').appendTo('body'); | ||
setTimeout(function() { $notif.fadeOut(function(){ $(this).remove(); }); }, | setTimeout(function() { $notif.fadeOut(function(){ $(this).remove(); }); }, 4500); | ||
} | } | ||
// --- 3. UI GENERATOR | // --- 3. UI GENERATOR --- | ||
// A. Header ( | // A. Header (Identitas) | ||
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;">' + | 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: | '<div style="display:flex; align-items:center; gap:8px; margin-bottom:5px;">' + | ||
'<span style="background:#6a5acd; color:white; padding:2px | '<span style="background:#6a5acd; color:white; padding:2px 6px; border-radius:3px; font-size:9px; font-weight:bold;">DATA</span>' + | ||
'<strong style="color:#1a1a1a | '<strong style="font-size:14px; color:#1a1a1a;">Mippedia Data Project</strong></div>' + | ||
'<p style="margin:0; font-size: | '<p style="margin:0; font-size:12px; color:#555;">Metadata repositori terstruktur ekosistem Mippedia.</p></div>'; | ||
// B. Connector ( | // B. Connector (Hanya munculkan ikon pensil jika USER ADMIN DAN HALAMAN TIDAK PROTECTED) | ||
var editBtn = '<span id="mip-open-editor" style="cursor:pointer; font-size:18px; color:#6a5acd; opacity:0. | // Jika halaman diproteksi, ikon pensil hilang. | ||
var connectorHtml = '<div id="mip-connector" style="background:#fff; border:1px solid #e0e0e0; border-radius:14px; padding: | var editBtn = (isAdmin && !isProtected) ? '<span id="mip-open-editor" style="cursor:pointer; font-size:18px; color:#6a5acd; opacity:0.8;">✎</span>' : ''; | ||
'<div style="display:flex; justify-content:space-between; align-items:center; margin-bottom: | |||
'<div id="mip-display" style="display:flex; gap: | var connectorHtml = '<div id="mip-connector" style="background:#fff; border:1px solid #e0e0e0; border-radius:14px; padding:18px; margin-top:20px; box-shadow:0 4px 15px rgba(0,0,0,0.05);">' + | ||
'<div id="mip-editor" style="display:none; margin-top: | '<div style="display:flex; justify-content:space-between; align-items:center; margin-bottom:15px;"><strong style="font-size:13px; color:#1a1a1a;">🔗 Koneksi Ekosistem</strong>'+editBtn+'</div>' + | ||
Object.keys(projects).map(p => '<div style="margin-bottom: | '<div id="mip-display" style="display:flex; gap:8px; flex-wrap:wrap;"><small style="color:#bbb;">Memproses...</small></div>' + | ||
'<button id="mip-save" style="width:100%; padding:12px; background:#6a5acd; color:white; border:none; border-radius:10px | '<div id="mip-editor" style="display:none; margin-top:15px; border-top:1px solid #f0f0f0; padding-top:15px;">' + | ||
Object.keys(projects).map(p => '<div style="margin-bottom:10px; display:flex; gap:8px;"><span style="width:60px; font-size:10px; font-weight:bold; background:#f5f5f5; display:flex; align-items:center; justify-content:center; border-radius:6px; color:#666;">'+projects[p].name+'</span><input type="text" id="in-'+p+'" placeholder="Judul artikel..." style="flex:1; padding:10px; font-size:13px; border:1.5px solid #eee; border-radius:8px; outline:none; -webkit-appearance:none;"></div>').join('') + | |||
'<button id="mip-save" style="width:100%; padding:12px; background:#6a5acd; color:white; border:none; border-radius:10px; font-weight:bold; cursor:pointer;">Sinkronkan Data</button>' + | |||
'</div></div>'; | '</div></div>'; | ||
// C. Footer ( | // C. Footer (Lisensi) | ||
var footerHtml = '<div id="mip-footer" style="background:#fff; border:1px dashed #ccc; padding: | var footerHtml = '<div id="mip-footer" style="background:#fff; border:1px dashed #ccc; padding:12px; margin-top:20px; text-align:center; border-radius:8px;">' + | ||
'<div style="font-size: | '<div style="font-size:10px; color:#999; font-weight:bold; text-transform:uppercase; margin-bottom:4px;">Lisensi Data</div>' + | ||
'<p style="margin:0; font-size: | '<p style="margin:0; font-size:11.5px; color:#666;">Tersedia di bawah lisensi <a href="https://creativecommons.org/licenses/by-sa/4.0/" target="_blank" style="color:#6a5acd; font-weight:bold; text-decoration:none;">CC BY-SA 4.0</a></p></div>'; | ||
// Injeksi | // Injeksi | ||
$('#mw-content-text').prepend(headerHtml); | $('#mw-content-text').prepend(headerHtml); | ||
$('#mw-content-text').append(connectorHtml).append(footerHtml); | $('#mw-content-text').append(connectorHtml).append(footerHtml); | ||
// --- 4. LOGIC | // --- 4. LOGIC CORE --- | ||
function loadLinks() { | function loadLinks() { | ||
new mw.Api().get({ action: 'query', prop: 'revisions', titles: dataPage, rvprop: 'content', formatversion: 2 }).done(function(d) { | new mw.Api().get({ action: 'query', prop: 'revisions', titles: dataPage, rvprop: 'content', formatversion: 2 }).done(function(d) { | ||
| Baris 231: | Baris 240: | ||
if (data[p]) { | if (data[p]) { | ||
found = true; | found = true; | ||
$box.append('<a style="background:'+projects[p].color+'; color:white; padding: | $box.append('<a style="background:'+projects[p].color+'; color:white !important; padding:8px 14px; border-radius:10px; font-size:11px; font-weight:bold; text-decoration:none !important;" href="'+projects[p].base+encodeURIComponent(data[p])+'" target="_blank">Mippedia '+projects[p].name+': '+data[p]+'</a>'); | ||
$('#in-' + p).val(data[p]); | $('#in-' + p).val(data[p]); | ||
} | } | ||
}); | }); | ||
if (!found) $box.html('<em style="color:# | if (!found) $box.html('<em style="color:#ccc; font-size:11.5px;">Belum terhubung ke ekosistem.</em>'); | ||
}); | }); | ||
} | } | ||
$(document).on('click', '#mip-save', function() { | $(document).on('click', '#mip-save', function() { | ||
if (!isAdmin) { | if (!isAdmin) { | ||
showBoxNotif('Akses | showBoxNotif('Akses Ditolak', 'Hanya grup <b>Pengurus</b> yang diizinkan mengedit metadata ini.', 'error'); | ||
return; | return; | ||
} | } | ||
var $btn = $(this).text(' | var $btn = $(this).text('Memeriksa...').prop('disabled', true); | ||
var api = new mw.Api(); | var api = new mw.Api(); | ||
var newData = {}, promises = [], invalid = [], isDel = true; | var newData = {}, promises = [], invalid = [], isDel = true; | ||
| Baris 262: | Baris 270: | ||
$.when.apply($, promises).then(function() { | $.when.apply($, promises).then(function() { | ||
if (invalid.length > 0) { | if (invalid.length > 0) { | ||
showBoxNotif(' | showBoxNotif('Judul Salah', 'Artikel tidak ada di Mippedia ' + invalid.join(', ') + '.', '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(r) { | 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) : {}; | 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; | if (isDel) delete full[pageTitle]; else full[pageTitle] = newData; | ||
api.postWithEditToken({ action: 'edit', title: dataPage, text: JSON.stringify(full, null, 2), summary: 'Sync: ' + pageTitle }).done(function() { | |||
api.postWithEditToken({ action: 'edit', title: dataPage, text: JSON.stringify(full, null, 2), summary: 'Sync | showBoxNotif('Berhasil!', 'Data ekosistem diperbarui.', 'success'); | ||
showBoxNotif(' | |||
setTimeout(function(){ location.reload(); }, 1500); | setTimeout(function(){ location.reload(); }, 1500); | ||
}); | }); | ||
}); | }); | ||
Revisi per 16 April 2026 11.37
/* ==========================================================
🧠 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 (FINAL BUNDLE V6)
Status: MOBILE OPTIMIZED & SMART PERMISSION
Rules: Namespace 0, Not Main Page, Responsive Notif
========================================================== */
(function() {
$(document).ready(function() {
// --- 1. FILTER GLOBAL ---
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');
// Cek apakah halaman ini dilindungi (protected)
var isProtected = mw.config.get('wgRestrictionEdit') && mw.config.get('wgRestrictionEdit').includes('sysop');
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. NOTIFIKASI KOTAK (RESPONSIVE HP & PC) ---
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:'🚫'};
// Logika Responsif via JS: Tengah di HP, Pojok di PC
var isMobile = $(window).width() < 768;
var positionStyle = isMobile ? 'left:10px; right:10px; top:20px;' : 'right:25px; top:25px; width:340px;';
var $notif = $('<div class="mip-box-notif" style="position:fixed; '+positionStyle+' background:'+theme.bg+'; border-left:6px solid '+theme.border+'; padding:16px; z-index:99999; border-radius:12px; box-shadow:0 15px 35px rgba(0,0,0,0.2); 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;">'+title+'</strong>' +
'</div>' +
'<p style="margin:0; font-size:12.5px; color:'+theme.text+'; line-height:1.5;">'+msg+'</p>' +
'</div>').appendTo('body');
setTimeout(function() { $notif.fadeOut(function(){ $(this).remove(); }); }, 4500);
}
// --- 3. UI GENERATOR ---
// A. Header (Identitas)
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:8px; margin-bottom:5px;">' +
'<span style="background:#6a5acd; color:white; padding:2px 6px; border-radius:3px; font-size:9px; font-weight:bold;">DATA</span>' +
'<strong style="font-size:14px; color:#1a1a1a;">Mippedia Data Project</strong></div>' +
'<p style="margin:0; font-size:12px; color:#555;">Metadata repositori terstruktur ekosistem Mippedia.</p></div>';
// B. Connector (Hanya munculkan ikon pensil jika USER ADMIN DAN HALAMAN TIDAK PROTECTED)
// Jika halaman diproteksi, ikon pensil hilang.
var editBtn = (isAdmin && !isProtected) ? '<span id="mip-open-editor" style="cursor:pointer; font-size:18px; color:#6a5acd; opacity:0.8;">✎</span>' : '';
var connectorHtml = '<div id="mip-connector" style="background:#fff; border:1px solid #e0e0e0; border-radius:14px; padding:18px; margin-top:20px; 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:13px; color:#1a1a1a;">🔗 Koneksi Ekosistem</strong>'+editBtn+'</div>' +
'<div id="mip-display" style="display:flex; gap:8px; flex-wrap:wrap;"><small style="color:#bbb;">Memproses...</small></div>' +
'<div id="mip-editor" style="display:none; margin-top:15px; border-top:1px solid #f0f0f0; padding-top:15px;">' +
Object.keys(projects).map(p => '<div style="margin-bottom:10px; display:flex; gap:8px;"><span style="width:60px; font-size:10px; font-weight:bold; background:#f5f5f5; display:flex; align-items:center; justify-content:center; border-radius:6px; color:#666;">'+projects[p].name+'</span><input type="text" id="in-'+p+'" placeholder="Judul artikel..." style="flex:1; padding:10px; font-size:13px; border:1.5px solid #eee; border-radius:8px; outline:none; -webkit-appearance:none;"></div>').join('') +
'<button id="mip-save" style="width:100%; padding:12px; background:#6a5acd; color:white; border:none; border-radius:10px; font-weight:bold; cursor:pointer;">Sinkronkan Data</button>' +
'</div></div>';
// C. Footer (Lisensi)
var footerHtml = '<div id="mip-footer" style="background:#fff; border:1px dashed #ccc; padding:12px; margin-top:20px; text-align:center; border-radius:8px;">' +
'<div style="font-size:10px; color:#999; font-weight:bold; text-transform:uppercase; margin-bottom:4px;">Lisensi Data</div>' +
'<p style="margin:0; font-size:11.5px; color:#666;">Tersedia di bawah lisensi <a href="https://creativecommons.org/licenses/by-sa/4.0/" target="_blank" style="color:#6a5acd; font-weight:bold; text-decoration:none;">CC BY-SA 4.0</a></p></div>';
// Injeksi
$('#mw-content-text').prepend(headerHtml);
$('#mw-content-text').append(connectorHtml).append(footerHtml);
// --- 4. LOGIC CORE ---
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 !important; padding:8px 14px; border-radius:10px; font-size:11px; font-weight:bold; text-decoration:none !important;" 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:#ccc; font-size:11.5px;">Belum terhubung ke ekosistem.</em>');
});
}
$(document).on('click', '#mip-save', function() {
if (!isAdmin) {
showBoxNotif('Akses Ditolak', 'Hanya grup <b>Pengurus</b> yang diizinkan mengedit metadata ini.', 'error');
return;
}
var $btn = $(this).text('Memeriksa...').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('Judul Salah', 'Artikel tidak ada di Mippedia ' + invalid.join(', ') + '.', '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: ' + pageTitle }).done(function() {
showBoxNotif('Berhasil!', 'Data ekosistem diperbarui.', 'success');
setTimeout(function(){ location.reload(); }, 1500);
});
});
});
});
$(document).on('click', '#mip-open-editor', function() { $('#mip-editor').slideToggle(); });
loadLinks();
});
})();