사용자:기나ㅏㄴ/massmove.js

위키백과, 우리 모두의 백과사전.

참고: 설정을 저장한 후에 바뀐 점을 확인하기 위해서는 브라우저의 캐시를 새로 고쳐야 합니다. 구글 크롬, 파이어폭스, 마이크로소프트 엣지, 사파리: ⇧ Shift 키를 누른 채 "새로 고침" 버튼을 클릭하십시오. 더 자세한 정보를 보려면 위키백과:캐시 무시하기 항목을 참고하십시오.

//자세한 설명은 [[사용자:기나ㅏㄴ/massmove]]을 참고하시기 바랍니다.
var massMoveTitle = "대량 문서 이동기";

$.when(mw.loader.using(['mediawiki.util'], $.ready)).done( function() {
	mw.util.addPortletLink("p-tb", mw.config.get('wgArticlePath').replace('$1', "특수:Massmove"), '대량 문서 이동기' , "대량 문서 이동" , "대량 문서 이동");
});

if (mw.config.get('wgCanonicalNamespace')+':'+mw.config.get('wgTitle') === '특수:Massmove' && mw.config.get('wgAction') == 'view');

// Adapted from [[User:Animum/massdelete.js]]
function massMoveGetValues() {
		return {reason: document.getElementById("wpMassMoveReason").value,
			oldPrefix: document.getElementById("wpMassMovePrefix1").value,
			newPrefix: document.getElementById("wpMassMovePrefix2").value,
			oldSuffix: document.getElementById("wpMassMoveSuffix1").value,
			newSuffix: document.getElementById("wpMassMoveSuffix2").value,
			watch: document.getElementById("wpMassMoveWatch").value,
			pipeTrick: document.getElementById("wpMassMovePipeTrick").checked,
			leaveRedirect: document.getElementById("wpMassMoveLeaveRedirect").checked,
			noRatelimit: document.getElementById("wpMassMoveNoRatelimit").checked,
			moveTalk: document.getElementById("wpMassMoveMoveTalk").checked,
			moveSubPages: document.getElementById("wpMassMoveMoveSubPages").checked
		};
}

function massMoveReplace(s, values) {
	s = s.trim();
	if (values.pipeTrick) {
		s = s.replace(/^(?:\:)?(?:.*\:)?(.*?)(?:, .*)?$/,"$1").replace(/(.*?)(?: ?\(.*\))?$/, "$1");
	}
	if (s.substring(0,values.oldSuffix.length) == values.oldSuffix) {
		s = s.substring(values.oldPrefix.length);
	}
	if (s.substring(s.length - values.oldSuffix.length) == values.oldSuffix) {
		s = s.substring(0, s.length - values.oldSuffix.length);
	}
	return values.newPrefix + s + values.newSuffix;
}

function massMoveGetArticles() {
	var articles = document.getElementById("wpMassMovePages").value.split("\n");
	var ret = [];
	var i, len;
	for (i = 0, len = articles.length; i < len; i++) {
		var s = articles[i];
		s = s.trim();
		if (s) {
			ret.push(s);
		}
	}
	return ret;
}
	
function massMoveUpdatePreview() {
	var	articles = massMoveGetArticles();
	if (articles.length > 0) {
		var values = massMoveGetValues();
		var preview = [articles[0] + " → " + massMoveReplace(articles[0], values)];
		for (var i = 1, len = articles.length; i < len; i++) {
			preview.push(articles[i] + " → " + massMoveReplace(articles[i], values));
		}
		document.getElementById("wpMassMovePreview").value = preview.join("\n");
		document.getElementById("wpMassMoveSubmit").disabled = false;
	} else {
		document.getElementById("wpMassMovePreview").value = '';
		document.getElementById("wpMassMoveSubmit").disabled = true;
	}
}

jQuery(document).ready(function($) {
	var config = mw.config.get(['wgNamespaceNumber', 'wgTitle', 'wgUserGroups', 'skin']);
	
	function now() {
		return new Date().getTime();
	}

	function doMassMove() {
		var articles = massMoveGetArticles();
		if (!articles.length) {
			return;
		}
		var
			api = new mw.Api(),
			values = massMoveGetValues(),
			moved = 0,
			failed = [],
			error = [],
			deferreds = [],
			lastMoved = 0,
			onSuccess = function () {
				moved++;
				console.log(now() + ": Moved " + moved);
				mw.notify("Success! " + moved + " pages moved.", {type: 'success', tag: 'status', autoHide: true});
			};
		
		function delay(len) {
			return function() {
				return $.Deferred(function (deferred) {
					var interval = lastMoved + config.wait - now();
					if ( (len <= config.hits) || ((lastMoved + config.wait - now()) < 0) || values.noRatelimit ) {
						interval = 0;
					}
					console.log(now() + ': 대기 중 ' + interval + 'ms...');
					setTimeout(function () {
						console.log(now() + ': 완료 대기 중.');
						deferred.resolve();
					}, interval);
				});
			};
		}
		
		function makeMoveFunc(article) {
			return function () {
				return $.Deferred(function (deferred) {
					var options = {
						format: 'json',
						action: 'move',
						watchlist: values.watch,
						from: article,
						to: massMoveReplace(article, values),
						reason: values.reason + ' ([[사용자:기나ㅏㄴ/massmove|' + massMoveTitle + ']])'
					};
					if (!values.leaveRedirect) {
						options.noredirect = '';
					}
					if (values.moveTalk) {
						options.movetalk = '';
					}
					if (values.moveSubPages) {
						options.movesubpages = '';
					}
					console.log(now() + ": 다음을 이동 중: " + options.from + "→" + options.to);
					mw.notify("다음과 같이 이동 중: " + options.from + " → " + options.to, {type: 'info', tag: 'status', autoHide: true});
					lastMoved = now();
					var promise = api.postWithEditToken(options);
					promise.done(onSuccess);
					promise.fail(function (code, obj) {
						failed.push(article);
						error.push(obj.error.info);
						console.warn(now() + ": 이동 실패 (" + obj.error.info + ").");
						mw.notify("이동 실패:  " + obj.error.info, {type: 'error', autoHide: true});
					});
					promise.always(function () {
						deferred.resolve();
					});
				});
			};
		}

		// Make a chain of deferred objects. We chain them rather than execute them in
		// parallel so that we don't make 1000 simultaneous move requests and bring the
		// site down. We use deferred objects rather than the promise objects returned
		// from the API request so that the chain continues even if some articles gave
		// errors.
		var deferred = makeMoveFunc(articles[0])();
		for (var i = 1, len = articles.length; i < len; i++) {
			deferred = deferred.then(delay(len));
			deferred = deferred.then(makeMoveFunc(articles[i]));
		}

		// Show the output and do cleanup once all the requests are done.
		$.when(deferred).then(function () {
			console.log(now() + ": 완료! " + moved + " 이동");
			if (failed.length) {
				mw.notify("완료. " + moved + "개 문서 이동, " + failed.length + "개 에러.", {type: 'warn', tag: 'mainNotify', autoHide: false});
				var $failedList = $('<ul>');
				for(var x = 0; x < failed.length; x++) {
					// Link the titles in the "failed" array
					var failedTitle = mw.Title.newFromText(failed[x]);
					var $failedItem = $('<li>');
					if (failedTitle) {
						$failedItem.append( $('<a>')
							.attr('href', failedTitle.getUrl())
							.text(failed[x])
						);
					} else {
						$failedItem.text(failed[x]);
					}
					$failedItem.append(document.createTextNode(': ' + error[x]));
					$failedList.append($failedItem);
				}
				$('#wpMassMoveFailedContainer')
					.append($('<br />'))
					.append($('<b>')
						.text('이동 실패:')
					)
					.append($failedList);
			} else {
				mw.notify("완료. " + moved + " 문서 이동.", {type: 'success', tag: 'mainNotify', autoHide: false});
			}
			document.getElementById("wpMassMoveSubmit").value = "Move";
			$("*", "#wpMassMove").not("#wpMassMovePreview").prop('disabled',false);
		});
	}
 
 	function getWait() {
 		config.hits=8; // default rate limit is 8/minute
 		config.seconds=60; // default rate limit is 8/minute
		config.wait = Math.ceil(config.seconds/config.hits) * 1000;
		new mw.Api().get({
    		meta: 'userinfo',
    		uiprop: 'ratelimits'
		}).fail(function(code, error) {
			console.warn(error);
			doMassMove();
		}).done( function(d) {
			if (d && d.query && d.query.userinfo && d.query.userinfo.ratelimits
					&& d.query.userinfo.ratelimits.move)
			{
				for (const property in d.query.userinfo.ratelimits.move) {
					var rlm = d.query.userinfo.ratelimits.move[property];
					if (rlm && rlm.hits && rlm.seconds) {
						console.log(property + " 속도 제한: " + rlm.seconds + "초마다 "+ rlm.hits + "개 문서 이동.");
						var thisWait = Math.ceil(rlm.seconds/rlm.hits) * 1000;
						if (thisWait < config.wait) {
							config.hits = rlm.hits;
							config.seconds = rlm.seconds;
							config.wait = thisWait;
							console.log("Calculated " + config.wait + "-millisecond wait between queries");
						}
					}
				}
			}
			mw.notify("속도 제한: " + config.seconds + "초마다 " + config.hits + "개 문서 이동.", {type: 'info', tag: 'mainNotify', autoHide: false});
			doMassMove();
		});
	}
 
	function massMoveForm() {
		wpMassMoveStyle = document.createElement('style');
		wpMassMoveStyle.type = 'text/css';
		document.getElementsByTagName('head')[0].appendChild(wpMassMoveStyle);
		wpMassMoveStyle = document.styleSheets[document.styleSheets.length-1];
		if (typeof(wpMassMoveStyle.media) === 'string') { //IE compatability
			wpMassMoveStyle.addRule('td.mincol', 'width:1%; white-space:nowrap;');
			wpMassMoveStyle.addRule('td.maxcol', 'width:auto');
		} else if (typeof(wpMassMoveStyle.media) === 'object') { //Modern browsers
			wpMassMoveStyle.insertRule('td.mincol {width:1%; white-space:nowrap;}', wpMassMoveStyle.cssRules.length);
			wpMassMoveStyle.insertRule('td.maxcol {width:auto;}', wpMassMoveStyle.cssRules.length);
		}
		
		document.getElementById(config.bodyContent).innerHTML = config.wpMassMoveIntro +
			'<form id="wpMassMove" name="wpMassMove">' +
			'<b>도구 사용 미숙으로 인해 발생하는 문제에 대한 책임은 <i>전적으로 귀하</i>에게 있습니다.</b>' +
			'<div id="wpMassMoveFailedContainer"></div>' +
			'<br /><br />' +
				'이동할 문서 (한 줄에 한 문서):<br />' +
					'<textarea tabindex="1" accesskey="," name="wpMassMovePages" id="wpMassMovePages" rows="10" cols="80" oninput="massMoveUpdatePreview()"></textarea>' +
				'<br /><br /><table style="background-color:transparent">' +
				'<tr><td>이동 전 이름에 <a href="' + (mw.config.get('wgServer')+mw.config.get('wgArticlePath')).replace('$1','en:Help:Pipe_trick') + '">"Pipe Trick"</a> 사용:</td>' +
					'<td colspan="5"><input type="checkbox" id="wpMassMovePipeTrick" name="wpMassMovePipeTrick"/ oninput="massMoveUpdatePreview()"></td></tr>' +
				'<tr><td class="mincol">이동 전 이름의 접두어 (e.g., 틀:):</td>' +
					'<td class="mincol"><input type="text" id="wpMassMovePrefix1" name="wpMassMovePrefix1" maxlength="255"  oninput="massMoveUpdatePreview()"/></td>' +
					'<td class="maxcol">&nbsp;</td>' +
					'<td class="mincol">이동 후 이름의 접두어 (e.g., 사용자:홍길동/):</td>' +
					'<td class="mincol"><input type="text" id="wpMassMovePrefix2" name="wpMassMovePrefix2" maxlength="255"  oninput="massMoveUpdatePreview()"/></td>' +
					'<td class="maxcol">&nbsp;</td></tr>' +
				'<tr><td class="mincol">이동 전 이름의 접미어 (e.g., /연습장):</td>' +
					'<td class="mincol"><input type="text" id="wpMassMoveSuffix1" name="wpMassMoveSuffix1" maxlength="255"  oninput="massMoveUpdatePreview()"/></td>' +
					'<td class="maxcol">&nbsp;</td>' +
					'<td class="mincol">이동 후 이름의 접미어 (e.g., /보존1):</td>' +
					'<td class="mincol"><input type="text" id="wpMassMoveSuffix2" name="wpMassMoveSuffix2" maxlength="255"  oninput="massMoveUpdatePreview()"/></td>' +
					'<td class="maxcol">&nbsp;</td></tr>' +
				'<tr><td class="mincol">딸린 토론 문서도 이동하기:</td>' +
					'<td class="mincol"><input type="checkbox" id="wpMassMoveMoveTalk" name="wpMassMoveMoveTalk" checked/></td>' +
					'<td class="maxcol">&nbsp;</td>' +
					'<td class="mincol">이동한 뒤 넘겨주기를 남기기:</td>' +
					'<td class="mincol"><input type="checkbox" id="wpMassMoveLeaveRedirect" name="wpMassMoveLeaveRedirect" checked/></td></tr>' +
				'<tr><td class="mincol">하위 문서도 이동 (100개까지):</td>' +
					'<td class="mincol"><input type="checkbox" id="wpMassMoveMoveSubPages" name="wpMassMoveMoveSubPages" checked/></td>' +	
					'<td class="maxcol">&nbsp;</td>' +
					'<td class="mincol">속도 제한 무시하기 (에러 발생 가능):</td>' +
					'<td class="mincol"><input type="checkbox" id="wpMassMoveNoRatelimit" name="wpMassMoveNoRatelimit"/></td></tr>' +
				'<tr><td>	원래 문서와 대상 문서를 주시하기:</td>' +
						'<td colspan="5"><select id="wpMassMoveWatch">' +
							'<option value="nochange">변화 없음</option>' +
							'<option value="preferences">사용자 설정에 따라</option>' +
							'<option value="watch">주시하기</option>' +
							'<option value="unwatch">주시문서 목록에서 삭제하기</option>' +
						'</select></td></tr>' +
				'<tr><td>편집 요약:</td>' +
					'<td colspan="4"><input type="text" id="wpMassMoveReason" name="wpMassMoveReason" maxlength="500" style="width:100%;" /></td>' +
					'<td class="maxcol">&nbsp;</td></tr></table>' +
				'<br /><br />미리보기:<br />' +
					'<textarea disabled name="wpMassMovePreview" id="wpMassMovePreview" rows="10" cols="80"></textarea>' +
				'<br /><br /><input disabled type="button" id="wpMassMoveSubmit" name="wpMassMoveSubmit" value="이동" />' +
			'</form>';
		document.getElementById("wpMassMoveSubmit").addEventListener("click", function (e) {
			$('#wpMassMoveFailedContainer').empty();
			$("*", "#wpMassMove").prop('disabled',true);
			document.getElementById("wpMassMoveSubmit").value = "이동하는 중...";
			getWait();
		});
	}
	
	function massMoveError() {
		document.getElementById(config.bodyContent).innerHTML = config.wpMassMoveIntro +
			'더 많은 정보가 필요하다면 <a href="' + (mw.config.get('wgServer')+mw.config.get('wgArticlePath')).replace('$1','사용자:기나ㅏㄴ') + '">번역자</a>에게 문의하세요!';
	}
	 
	if(mw.config.get('wgNamespaceNumber') === -1 
		&& (mw.config.get('wgTitle') === "Massmove" || 
		mw.config.get('wgTitle') === "MassMove")
	) {
		document.getElementsByTagName("h1")[0].textContent = massMoveTitle;
		document.title = massMoveTitle + " - 위키백과, 우리 모두의 백과사전";
	
		config.wpMassMoveIntro = '<div id="siteSub">위키백과, 우리 모두의 백과사전</div>' +
			'<p>한국어 위키백과에 최적화된 대량 문서 이동기입니다.</p>' +
			'<br />' + '이 도구는 <code>sysop</code>이나 <code>extendedconfirmed</code> 권한을 가진 사용자만 사용할 수 있습니다.' +
			'<br />' + '당신의 권한은 다음과 같습니다: ' + mw.config.get('wgUserGroups') + '<br /><br />';

		if (config.skin == 'modern') {
			config.bodyContent = 'mw_contentholder';
		} else if (config.skin == 'cologneblue') {
			config.bodyContent = 'article';
		} else {
			config.bodyContent = 'bodyContent';
		}
		
		if (/sysop/.test(config.wgUserGroups) || /extendedconfirmed/.test(config.wgUserGroups)) {
			$.when( $.ready, mw.loader.using(['mediawiki.util'])).done( massMoveForm );
		} else {
			$.when( $.ready, mw.loader.using(['mediawiki.util'])).done( massMoveError );
		}
	}
});