var wmCantWork = 'Викификатор не может работать в вашем браузере',
wmFullText = 'Викификатор обработает ВЕСЬ текст на этой странице. Продолжить?',
wmTalkPage = 'Викификатор не обрабатывает страницы обсуждения целиком.\n\nВыделите ваше сообщение — обработано будет только оно';
window.wfPlugins = window.wfPlugins || [];
function Wikify() {
'use strict';
var txt = '',
hidden = [],
wpTextbox1 = document.editform.wpTextbox1,
winScroll = document.documentElement.scrollTop;
// FUNCTIONS
function r(r1, r2) {
txt = txt.replace(r1, r2);
}
function hide(re) {
r(re, function (s) {
return '\x01' + hidden.push(s) + '\x02';
});
}
function hideTag(tag) {
hide(new RegExp('<' + tag + '( [^>]+)?>[\\s\\S]+?<\\/' + tag + '>', 'gi'));
}
function hideTemplates() {
var pos = 0,
stack = [],
tpl,
left,
right;
while (true) {
left = txt.indexOf('{{', pos);
right = txt.indexOf('}}', pos);
if (left === -1 && right === -1 && !stack.length) {
break;
}
if (left !== -1 && (left < right || right === -1)) {
stack.push(left);
pos = left + 2;
}
else {
left = stack.pop();
if (typeof left === 'undefined') {
if (right === -1) {
pos += 2;
continue;
}
else {
left = 0;
}
}
if (right === -1) {
right = txt.length;
}
right += 2;
tpl = txt.substring(left, right);
txt = txt.substring(0, left) + '\x01' + hidden.push(tpl) + '\x02' + txt.substr(right);
pos = right - tpl.length;
}
}
}
function processText() {
var i,
u = '\u00A0'; // unbreakable space
if (window.wgNamespaceNumber % 2 || window.wgNamespaceNumber === 4) { // is talk page
u = ' ';
var sigs = txt.match(/\d\d:\d\d, \d\d? \S{3,8} 20\d\d \(UTC\)/g);
if (sigs && sigs.length > 1) {
alert(wmTalkPage);
return;
}
}
hideTag('nowiki');
hideTag('pre');
hideTag('source');
hideTag('syntaxhighlight');
hideTag('code');
hideTag('tt');
hideTag('math');
hideTag('timeline');
r(/( |\n|\r)+\{\{(·|•|\*)\}\}/g, '{{$2}}'); // before {{·/•/*}}, usually in templates
r(/\{\{\s*[Шш]аблон:([\s\S]+?)\}\}/g, '{{$1}}');
r(/(\{\{\s*)reflist(\s*[\|\}])/ig, '$1примечания$2');
hideTemplates();
hide(/^ .*/mg);
hide(/(https?|ftp|news|nntp|telnet|irc|gopher):\/\/[^\s\[\]<>"]+ ?/gi);
hide(/^#(redirect|перенапр(авление)?)/i);
hideTag('gallery');
r(/ +(\n|\r)/g, '$1'); // spaces at EOL
txt = '\n' + txt + '\n';
// LINKS
r(/(\[\[:?)(category|категория):( *)/ig, '$1Категория:');
r(/(\[\[:?)(image|изображение|file):( *)/ig, '$1Файл:');
// Linked years, centuries and ranges
r(/(\(|\s)(\[\[[12]?\d{3}\]\])[\u00A0 ]?(-{1,3}|–|—) ?(\[\[[12]?\d{3}\]\])(\W)/g, '$1$2—$4$5');
r(/(\[\[[12]?\d{3}\]\]) ?(гг?\.)/g, '$1' + u + '$2');
r(/(\(|\s)(\[\[[IVX]{1,5}\]\])[\u00A0 ]?(-{1,3}|–|—) ?(\[\[[IVX]{1,5}\]\])(\W)/g, '$1$2—$4$5');
r(/(\[\[[IVX]{1,5}\]\]) ?(вв?\.)/g, '$1' + u + '$2');
r(/\[\[(\d+)\]\]\sгод/g, '[[$1' + u + 'год]]');
r(/\[\[(\d+)\sгод\|\1\]\]\sгод/g, '[[$1' + u + 'год]]');
r(/\[\[(\d+)\sгод\|\1\sгод([а-я]{0,3})\]\]/g, '[[$1' + u + 'год]]$2');
r(/\[\[((\d+)(?: (?:год )?в [\wa-яёА-ЯЁ ]+\|\2)?)\]\][\u00A0 ](год[а-яё]*)/g, '[[$1' + u + '$3]]');
r(/\[\[([XVI]+)\]\]\sвек/g, '[[$1' + u + 'век]]');
r(/\[\[([XVI]+)\sвек\|\1\]\]\sвек/g, '[[$1' + u + 'век]]');
r(/\[\[([XVI]+)\sвек\|\1\sвек([а-я]{0,3})\]\]/g, '[[$1' + u + 'век]]$2');
r(/\[\[(([XVI]+) век\|\2)\]\][\u00A0 ]век/g, '[[$2' + u + 'век]]');
// Nice links
r(/(\[\[[^|\[\]]*)[\u00AD\u200E\u200F]+([^\[\]]*\]\])/g, '$1$2'); // Soft Hyphen & DirMark
r(/\[\[ *([^|\[\]]+) *\| *(\1)([a-zа-яё]*) *\]\]/g, '[[$2]]$3');
r(/\[\[ *([^|\[\]]+)([^|\[\]()]+) *\| *\1 *\]\]\2/g, '[[$1$2]]'); // text repetition after link
r(/\[\[ *(?!Файл:|Категория:)([a-zA-Zа-яёА-ЯЁ\u00A0-\u00FF %!\"$&'()*,\-—.\/0-9:;=?\\@\^_`’~]+) *\| *([^\|\[\]]+) *\]\]([a-zа-яё]+)/g, '[[$1|$2$3]]'); // "
hide(/\[\[[^\]|]+/g); // only link part
// TAGS
r(/<<(\S.+\S)>>/g, '"$1"'); // << >>
r(/(su[pb]>)-(\d)/g, '$1−$2'); // ->minus
r(/<(b|strong)>(.*?)<\/(b|strong)>/gi, "'''$2'''");
r(/<(i|em)>(.*?)<\/(i|em)>/gi, "''$2''");
r(/^<hr ?\/?>/gim, '----');
r(/<[\/\\]?(hr|br)( [^\/\\>]+?)? ?[\/\\]?>/gi, '<$1$2 />');
r(/[\u00A0 \t]*<ref(?:\s+name="")?(\s|>)/gi, '<ref$1');
r(/(\n== *[a-zа-я\s\.:]+ *==\n+)<references *\/>/ig, '$1{' + '{примечания}}');
hide(/<[a-z][^>]*?>/gi);
hide(/^(\{\||\|\-).*/mg); // table/row def
hide(/(^\||^!|!!|\|\|) *[a-z]+=[^|]+\|(?!\|)/mgi); // cell style
hide(/\| +/g); // formatted cell
r(/[ \t\u00A0]+/g, ' '); // double spaces
// Entities etc. → Unicode chars
if (window.wgNamespaceNumber !== 10) {
r(/&(#x[0-9a-f]{2,4}|#[0-9]{3,4}|[0-9a-z]{2,8});/gi, function (s) {
var t = document.createElement('textarea');
t.innerHTML = s;
var c = t.value;
if (c.length === 1 && c.charCodeAt(0) > 127) {
return c;
}
return s;
});
}
r(/\(tm\)/gi, '™');
r(/\.\.\./g, '…');
r(/(^|[^+])\+-(?!\+|-)/g, '$1±');
r(/~=/g, '≈');
r(/\^2(\D)/g, '²$1');
r(/\^3(\D)/g, '³$1');
r(/(\s)кв\.\s*(дм|см|мм|мкм|нм|км|м)(\s)/g, '$1' + u + '$2²$3');
r(/(\s)куб\.\s*(дм|см|мм|мкм|нм|км|м)(\s)/g, '$1' + u + '$2³$3');
r(/((?:^|[\s"])\d+(?:[\.,]\d+)?)\s*[xх]\s*(\d+(?:[\.,]\d+)?)\s*([мm]{1,2}(?:[\s"\.,;?!]|$))/g, '$1×$2' + u + '$3');
r(/([\wа-яА-ЯёЁ])'([\wа-яА-ЯёЁ])/g, '$1’$2'); //'
r(/№№/g, '№');
// Headings
r(/^(=+)[ \t\f\v]*(.*?)[ \t\f\v]*=+$/gm, '$1 $2 $1'); //add spaces inside
r(/([^\r\n])(\r?\n==.*==\r?\n)/g, '$1\n$2'); //add empty line before
r(/^== см(\.?|отри|отрите) ?также ==$/gmi, '== См. также ==');
r(/^== сноски ==$/gmi, '== Примечания ==');
r(/^== внешние\sссылки ==$/gmi, '== Ссылки ==');
r(/^== (.+)[.:] ==$/gm, '== $1 ==');
r(/^== '''(?!.*'''.*''')(.+)''' ==$/gm, '== $1 ==');
r(/«|»|“|”|„/g, '"'); // temp
// Hyphens and en dashes to pretty dashes
r(/–/g, '-'); // – -> hyphen
r(/(\s)-{1,3} /g, '$1— '); // hyphen -> —
r(/(\d)--(\d)/g, '$1—$2'); // -> —
r(/(\s)-(\d)/g, '$1−$2'); // hyphen -> minus
// Year and century ranges
r(/(\(|\s)([12]?\d{3})[\u00A0 ]?(-{1,3}|—) ?([12]?\d{3})(?![\wА-ЯЁа-яё]|-[^ех]|-[ех][\wА-ЯЁа-яё])/g, '$1$2—$4');
r(/([12]?\d{3}) ?(гг?\.)/g, '$1' + u + '$2');
r(/(\(|\s)([IVX]{1,5})[\u00A0 ]?(-{1,3}|—) ?([IVX]{1,5})(?![\w\-])/g, '$1$2—$4');
r(/([IVX]{1,5}) ?(вв?\.)/g, '$1' + u + '$2');
// Reductions
r(/(Т|т)\.\s?е\./g, '$1о есть');
r(/(Т|т)\.\s?к\./g, '$1ак как');
r(/(В|в)\sт\. ?ч\./g, '$1 том числе');
r(/(И|и)\sт\.\s?д\./g, '$1' + u + 'т.' + u + 'д.');
r(/(И|и)\sт\.\s?п\./g, '$1' + u + 'т.' + u + 'п.');
r(/(Т|т)\.\s?н\./g, '$1.' + u + 'н.');
r(/(И|и)\.\s?о\./g, '$1.' + u + 'о.');
r(/н\.\s?э(\.|(?=\s))/g, 'н.' + u + 'э.');
r(/(Д|д)(о|\.)\sн\.\s?э\./g, '$1о' + u + 'н.' + u + 'э.');
r(/(\d)[\u00A0 ]?(млн|млрд|трлн|(?:м|с|д|к)?м|[км]г)\.?(?=[,;.]| "?[а-яё\-])/g, '$1' + u + '$2');
r(/(\d)[\u00A0 ](тыс)([^\.А-Яа-яЁё])/g, '$1' + u + '$2.$3');
r(/ISBN:\s?(?=[\d\-]{8,17})/, 'ISBN ');
// Insert/delete spaces
r(/^([#*:]+)[ \t\f\v]*(?!\{\|)([^ \t\f\v*#:;])/gm, '$1 $2'); // space after #*: unless before table
r(/(\S)[\u00A0 \t](-{1,3}|—)[\u00A0 \t](\S)/g, '$1' + u + '— $3');
r(/([А-ЯЁ]\.) ?([А-ЯЁ]\.) ?([А-ЯЁ][а-яё])/g, '$1' + u + '$2' + u + '$3');
r(/([А-ЯЁ]\.)([А-ЯЁ]\.)/g, '$1 $2');
r(/([а-яё]"?\)?[\.\?!:])((?:\x01\d+\x02\|)?[A-ZА-ЯЁ])/g, '$1 $2'); // word. word
r(/([)"a-zа-яё\]²³])\s*([,:])([\[("a-zа-яё])/g, '$1$2 $3'); // word, word
r(/([)"a-zа-яё\]²³])\s([,;])\s([\[("a-zа-яё])/g, '$1$2 $3');
r(/([^%\/\wА-Яа-яЁё]\d+?(?:[\.,]\d+?)?) ?([%‰])(?!-[А-Яа-яЁё])/g, '$1' + u + '$2'); //5 %
r(/(\d) ([%‰])(?=-[А-Яа-яЁё])/g, '$1$2'); //5%-й
r(/([№§])(\s*)(\d)/g, '$1' + u + '$3');
// inside ()
r(/\( +/g, '(');
r(/ +\)/g, ')');
// Temperature
r(/([\s\d=≈≠≤≥<>—("'|])([+±−\-]?\d+?(?:[.,]\d+?)?)(([ °\^*]| [°\^*])(C|F))(?=[\s"').,;!?|\x01])/gm, '$1$2' + u + '°$5'); //'
// Dot → comma in numbers
r(/(\s\d+)\.(\d+[\u00A0 ]*[%‰°×])/gi, '$1,$2');
// Plugins
for (i in window.wfPlugins) {
if (window.wfPlugins.hasOwnProperty(i)) {
window.wfPlugins[i](txt, r);
}
}
// "" → «»
for (i = 1; i <= 2; i++) {
r(/([\s\x02!|#'"\/(;+\-])"([^"]*)([^\s"(|])"([^a-zа-яё])/ig, '$1«$2$3»$4'); //"
}
while (/«[^»]*«/.test(txt)) {
r(/«([^»]*)«([^»]*)»/g, '«$1„$2“');
}
if ('0'.replace('0', '$$') === '$') { ////$ in replacing string is special, except in IE
for (i = 0; i < hidden.length; i++) {
hidden[i] = hidden[i].replace(/\$/g, '$$$$');
}
}
while (hidden.length > 0) {
r('\x01' + hidden.length + '\x02', hidden.pop());
}
txt = txt.substr(1, txt.length - 2);
}
function processAllText() {
txt = wpTextbox1.value;
processText();
r(/^[\n\r]+/, '');
wpTextbox1.value = txt;
txt = '';
if (window.auto_comment && window.insertSummary && !document.editform.wpSection.value) {
window.insertSummary('викификатор');
}
}
// Check regexp support
try {
txt = 'ая'.replace(/а/g, 'б').replace(/б(?=я)/, 'в');
}
catch (e) {}
if (txt !== 'вя') {
alert(wmCantWork);
return;
}
wpTextbox1.focus();
// Modern browsers
if (typeof wpTextbox1.selectionStart !== 'undefined') {
var textScroll = wpTextbox1.scrollTop,
startPos = wpTextbox1.selectionStart,
endPos = wpTextbox1.selectionEnd;
txt = wpTextbox1.value.substring(startPos, endPos);
if (txt === '') {
processAllText();
}
else {
processText();
wpTextbox1.value = wpTextbox1.value.substring(0, startPos) + txt + wpTextbox1.value.substring(endPos);
}
wpTextbox1.selectionStart = startPos;
wpTextbox1.selectionEnd = startPos + txt.length;
wpTextbox1.scrollTop = textScroll;
}
// IE
else if (document.selection && document.selection.createRange) {
var range = document.selection.createRange();
txt = range.text;
if (txt === '') {
processAllText();
}
else {
processText();
range.text = txt;
if (range.moveStart) {
range.moveStart('character', -txt.length);
}
range.select();
}
}
// Other browsers
else if (confirm(wmFullText)) {
processAllText();
}
document.documentElement.scrollTop = winScroll; // scroll back, for IE/Opera
}