미디어위키:Gadget-ProtectCGI.js
참고: 설정을 저장한 후에 바뀐 점을 확인하기 위해서는 브라우저의 캐시를 새로 고쳐야 합니다.
- 파이어폭스 / 사파리: Shift 키를 누르면서 새로 고침을 클릭하거나, Ctrl-F5 또는 Ctrl-R을 입력 (Mac에서는 ⌘-R)
- 구글 크롬: Ctrl-Shift-R키를 입력 (Mac에서는 ⌘-Shift-R)
- 인터넷 익스플로러 / 엣지: Ctrl 키를 누르면서 새로 고침을 클릭하거나, Ctrl-F5를 입력.
- 오페라: Ctrl-F5를 입력.
/**
* [[틀:CGI보호]]
* CGI나 주소를 조작하지 못하게 합니다.
* 제작자: [[사용자:hsl0|hsl0]]
**/
(function protectCGI() {
function getTitle(pagename) {
pagename = pagename.split('/')[0].split(':');
var ns = pagename[0].replace(/talk|토론/gi, '');
return ns + (ns && ':') + pagename[1];
}
function compareObject(a, b) {
var aKey = Object.keys(a);
var bKey = Object.keys(b);
if(aKey.length !== bKey.length) return false;
for(var key in a) {
if(a[key] !== b[key]) return false;
}
return true;
}
function filterLink() {
var href = new URL(this.href, location);
var params = geturlSearch(href);
if(href.host !== location.host) return false;
if(params.action || params.oldid) return false;
if(href.pathname.startsWith('/wiki/')) params.title = href.pathname.slice(6);
if(getTitle(params.title) !== title) return false;
$(this).data('params', params);
return true;
}
var params = geturlSearch();
params.title = mw.config.get('wgPageName');
var title = getTitle(params.title);
var protect = document.getElementById('protectCGI');
var cheat = null;
var temp = localStorage.getItem('protectCGI-temp');
var data = sessionStorage.getItem('protectCGI');
var link;
registerRenderer(function() {
$('.protectCGI-link-container a').filter(filterLink).addClass('protectCGI-link');
$('.protectCGI-link-container a:not(.protectCGI-link)').addClass('protectCGI-link-no');
});
registerHandler(function() {
function hijack(link) {
var title;
if(link.href && link.href != location.href) location.replace(link.href);
else if(data) {
data = JSON.parse(data);
title = data.title;
delete data.title;
location.replace(mw.util.getUrl(title, data));
} else if(document.referrer) history.back();
else location.replace(mw.util.getUrl(getTitle(mw.config.get('wgPageName'))));
}
if(protect) (function() {
var clicked;
link = $(protect).find('a').click(function(event) {
event.preventDefault();
if(this === clicked) {
if('wait' in this.dataset) {
if(this.dataset.wait <= 0) {
event.stopImmediatePropagation();
hijack(this);
}
} else hijack(this);
} else {
clicked = this;
this.click();
}
});
if(params.action || params.oldid) return;
else cheat = false;
if(data) {
data = JSON.parse(data);
cheat = !compareObject(data, params);
} else cheat = true;
})();
else if(!(params.action || params.oldid)) cheat = false;
if(cheat === false) $('.protectCGI-link').click(function(event) {
sessionStorage.setItem('protectCGI', JSON.stringify($(this).data('params')));
if(protect) {
event.preventDefault();
location.replace(this.href);
}
});
window.addEventListener('hashchange', function(event) {
var data, dataObj;
if(location.hash.endsWith('##emergency-save')) {
data = sessionStorage.getItem('protectCGI');
if(data) dataObj = JSON.parse(data);
if(data && dataObj.title === params.title) localStorage.setItem('protectCGI-temp', data);
}
});
});
registerTrigger(function() {
if(temp) {
if(!data) {
sessionStorage.setItem('protectCGI', temp);
data = temp;
mw.loader.using('oojs-ui-windows').then(function() {
OO.ui.confirm('비상 저장된 세션이 있습니다. 다시 돌아가겠습니까?').then(function(res) {
var data = JSON.parse(temp);
var title;
if(res && !compareObject(params, data)) {
title = data.title;
delete data.title;
location.href = mw.util.getUrl(data.title, data);
}
});
});
}
localStorage.removeItem('protectCGI-temp');
}
if(cheat) {
if(link.length) link.click();
else if(data) {
title = data.title;
delete data.title;
location.replace(mw.util.getUrl(title, data));
}
else if(document.referrer) history.back();
else location.replace(mw.util.getUrl(getTitle(mw.config.get('wgPageName'))));
}
if(cheat === false) {
document.body.classList.add('protectCGI-done');
$('.protectCGI-only').show();
}
});
})();