MediaWiki:Wikificator.js — различия между версиями

Материал из Ролевая энциклопедии
Перейти к: навигация, поиск
(обновление?)
 
Строка 1: Строка 1:
// основано на википедийном викификаторе; последние 5 авторов Alex Smotrov, Putnik, Kalan, Александр Сигачёв, Solon; условия распространения - GFDL; ссылка на источник http://ru.wikipedia.org/wiki/MediaWiki:Wikificator.js
+
var wmCantWork = 'Викификатор не может работать в вашем браузере',
var wmVersion = '2010-04-03'
+
  wmFullText = 'Викификатор обработает ВЕСЬ текст на этой странице. Продолжить?',
var wmCantWork = 'Викификатор не может работать в вашем браузере\n\nWikificator can not work in your browser'
+
  wmTalkPage = 'Викификатор не обрабатывает страницы обсуждения целиком.\n\nВыделите ваше сообщение — обработано будет только оно';
var wmFullText = 'Викификатор обработает ВЕСЬ текст на этой странице. Продолжить?'
+
window.wfPlugins = window.wfPlugins || [];
var wmTalkPage = 'Викификатор не обрабатывает страницы обсуждения целиком.\n\nВыделите ваше сообщение — обработано будет только оно'
 
  
function Wikify(){
+
function Wikify() {
var txt='', hidden = [], wpTextbox1 = document.editform.wpTextbox1
+
  'use strict';
var winScroll = document.documentElement.scrollTop
+
  var txt = '',
 +
    hidden = [],
 +
    wpTextbox1 = document.editform.wpTextbox1,
 +
    winScroll = document.documentElement.scrollTop;
  
try {txt='ая'.replace(/а/g,'б').replace(/б(?=я)/,'в')} catch(e){}//check regexp support
 
if (txt != 'вя' ||
 
  (navigator.appName=='Netscape' && navigator.appVersion.substr (0, 1) < 5))
 
  { alert(wmCantWork); return }
 
  
wpTextbox1.focus()
+
  // FUNCTIONS
  
if (typeof wpTextbox1.selectionStart != 'undefined'  
+
  function r(r1, r2) {
     && (navigator.productSub > 20031000 || is_safari || is_opera)) { //Mozilla/Opera/Safari3
+
    txt = txt.replace(r1, r2);
     var textScroll = wpTextbox1.scrollTop
+
  }
     var startPos = wpTextbox1.selectionStart
+
 
    var endPos = wpTextbox1.selectionEnd
+
  function hide(re) {
    txt = wpTextbox1.value.substring(startPos, endPos)
+
    r(re, function (s) {
    if (txt == '') processAllText()
+
      return '\x01' + hidden.push(s) + '\x02';
    else{
+
    });
      processText()
+
  }
      wpTextbox1.value = wpTextbox1.value.substring(0, startPos) + txt + wpTextbox1.value.substring(endPos)
+
 
 +
  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;
 +
      }
 
     }
 
     }
    wpTextbox1.selectionStart = startPos
+
  }
    wpTextbox1.selectionEnd = startPos + txt.length
 
    wpTextbox1.scrollTop = textScroll
 
  
}else if (document.selection && document.selection.createRange) { //IE
+
  function processText() {
  var range = document.selection.createRange()
+
    var i,
  txt = range.text
+
      u = '\u00A0'; // unbreakable space
  if (txt == '') processAllText()
+
    if (window.wgNamespaceNumber % 2 || window.wgNamespaceNumber === 4) { // is talk page
  else{
+
      u = ' ';
    processText()
+
      var sigs = txt.match(/\d\d:\d\d, \d\d? \S{3,8} 20\d\d \(UTC\)/g);
    range.text = txt
+
      if (sigs && sigs.length > 1) {
    if (range.moveStart) range.moveStart('character', - txt.length)
+
        alert(wmTalkPage);
    range.select()  
+
        return;
  }
+
      }
 
+
    }
}else // other browsers
 
  if (confirm(wmFullText)) processAllText()
 
  
document.documentElement.scrollTop = winScroll // scroll back, for IE/Opera
+
    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');
  
//functions
+
    hideTemplates();
 +
    hide(/^ .*/mg);
 +
    hide(/(https?|ftp|news|nntp|telnet|irc|gopher):\/\/[^\s\[\]<>"]+ ?/gi);
 +
    hide(/^#(redirect|перенапр(авление)?)/i);
 +
    hideTag('gallery');
  
function processAllText(){
 
txt = wpTextbox1.value
 
if (txt=='version') alert('Викификатор '+wmVersion)
 
processText()
 
r(/^[\n\r]+/, '')
 
wpTextbox1.value = txt
 
txt = ''
 
if (window.auto_comment && window.insertSummary && !document.editform.wpSection.value)
 
  insertSummary('викификатор')
 
}
 
  
function processText(){
+
    r(/ +(\n|\r)/g, '$1'); // spaces at EOL
 +
    txt = '\n' + txt + '\n';
  
var u = '\u00A0' //unbreakable space
 
if (wgNamespaceNumber % 2 || 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')
+
    // LINKS
hideTag('pre')
+
    r(/(\[\[:?)(category|категория):( *)/ig, '$1Категория:');
hideTag('source')
+
    r(/(\[\[:?)(image|изображение|file):( *)/ig, '$1Файл:');
hideTag('code')
+
    // Linked years, centuries and ranges
hideTag('tt')
+
    r(/(\(|\s)(\[\[[12]?\d{3}\]\])[\u00A0 ]?(-{1,3}|–|—) ?(\[\[[12]?\d{3}\]\])(\W)/g, '$1$2—$4$5');
hideTag('math')
+
    r(/(\[\[[12]?\d{3}\]\]) ?(гг?\.)/g, '$1' + u + '$2');
r(/( |\n|\r)+\{\{(·|•|\*)\}\}/g, '{{$2}}'); //before {{·//*}}, usually in templates
+
    r(/(\(|\s)(\[\[[IVX]{1,5}\]\])[\u00A0 ]?(-{1,3}|–|—) ?(\[\[[IVX]{1,5}\]\])(\W)/g, '$1$2—$4$5');
hide(/{\{[\s\S]+?}}/g)//templates
+
    r(/(\[\[[IVX]{1,5}\]\]) ?(вв?\.)/g, '$1' + u + '$2');
hide(/^ .*/mg)
+
    r(/\[\[(\d+)\]\]\sгод/g, '[[$1' + u + 'год]]');
hide(/(https?|ftp|news|nntp|telnet|irc|gopher):\/\/[^\s\[\]<>"]+ ?/gi)
+
    r(/\[\[(\d+)\sгод\|\1\]\]\sгод/g, '[[$1' + u + 'год]]');
hide(/^#(redirect|перенапр(авление)?)/i)
+
    r(/\[\[(\d+)\sгод\|\1\sгод([а-я]{0,3})\]\]/g, '[[$1' + u + 'год]]$2');
hideTag('gallery')
+
    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
  
  
r(/ +(\n|\r)/g,'$1')//spaces at EOL
+
    // TAGS
txt = '\n'+txt+'\n'
+
    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
  
//LINKS
+
    // Entities etc. → Unicode chars
r(/(\[\[:?)(category|категория):( *)/ig, '$1Категория:')
+
    if (window.wgNamespaceNumber !== 10) {
r(/(\[\[:?)(image|изображение|file):( *)/ig, '$1Файл:')
+
      r(/&(#x[0-9a-f]{2,4}|#[0-9]{3,4}|[0-9a-z]{2,8});/gi, function (s) {
//Linked years, centuries and ranges
+
        var t = document.createElement('textarea');
r(/(\(|\s)(\[\[[12]?\d{3}\]\])[\u00A0 ]?(-{1,3}|–|—) ?(\[\[[12]?\d{3}\]\])(\W)/g, '$1$2—$4$5')
+
        t.innerHTML = s;
r(/(\[\[[12]?\d{3}\]\]) ?(гг?\.)/g, '$1'+u+'$2')
+
        var c = t.value;
r(/(\(|\s)(\[\[[IVX]{1,5}\]\])[\u00A0 ]?(-{1,3}|–|—) ?(\[\[[IVX]{1,5}\]\])(\W)/g, '$1$2—$4$5')
+
        if (c.length === 1 && c.charCodeAt(0) > 127) {
r(/(\[\[[IVX]{1,5}\]\]) ?(вв?\.)/g, '$1'+u+'$2')
+
          return c;
r(/\[\[(\d+)\]\]\sгод/g, '[[$1'+u+'год]]')
+
        }
r(/\[\[(\d+)\sгод\|\1\]\]\sгод/g, '[[$1'+u+'год]]')
+
        return s;
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(/\(tm\)/gi, '');
r(/\[\[([XVI]+)\sвек\|\1\]\]\sвек/g, '[[$1'+u+'век]]')
+
    r(/\.\.\./g, '');
r(/\[\[([XVI]+)\sвек\|\1\sвек([а-я]{0,3})\]\]/g, '[[$1'+u+'век]]$2')
+
    r(/(^|[^+])\+-(?!\+|-)/g, '$');
r(/\[\[(([XVI]+) век\|\2)\]\][\u00A0 ]век/g, '[[$2'+u+'век]]')
+
    r(/~=/g, '');
// Nice links
+
    r(/\^2(\D)/g, '²$1');
r(/(\[\[[^|\[\]]*)[\u00AD\u200E\u200F]+([^\[\]]*\]\])/g, '$1$2') // Soft Hyphen & DirMark
+
    r(/\^3(\D)/g, '³$1');
r(/\[\[ *([a-zA-Zа-яёА-ЯЁ\u00A0-\u00FF %!\"$&'()*,\-—.\/0-9:;=?\\@\^_`’~]+) *\| *(\1)([a-zа-яё]*) *\]\]/g, '[[$2]]$3') // "
+
    r(/(\s)кв\.\s*(дм|см|мм|мкм|нм|км|м)(\s)/g, '$1' + u + '$2²$3');
r(/\[\[ *([^|\[\]]+)([^|\[\]]+) *\| *\1 *\]\]\2/g, '[[$1$2]]') // -повтор текста за ссылкой
+
    r(/(\s)куб\.\s*(дм|см|мм|мкм|нм|км|м)(\s)/g, '$1' + u + '$2³$3');
r(/\[\[ *(?!Файл:|Категория:)([a-zA-Zа-яёА-ЯЁ\u00A0-\u00FF %!\"$&'()*,\-—.\/0-9:;=?\\@\^_`’~]+) *\| *([^|[\]]+) *\]\]([a--яё]+)/g, '[[$1|$2$3]]') // "
+
    r(/((?:^|[\s"])\d+(?:[\.,]\d+)?)\s*[xх]\s*(\d+(?:[\.,]\d+)?)\s*([мm]{1,2}(?:[\s"\.,;?!]|$))/g, '$$2' + u + '$3');
hide(/\[\[[^\]|]+/g)//only link part
+
    r(/([\-яА-ЯёЁ])'([\-яА-ЯёЁ])/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 ==');
  
//TAGS
+
    r(/«|»|||/g, '"'); // temp
r(/<<(\S.+\S)>>/g, '"$1"') //<< >>
 
r(/(sup>|sub>|\s)-(\d)/g, '$1−$2') //minus
 
r(/&sup2;/gi, '²')
 
r(/&sup3;/gi, '³')
 
r(/<(b|strong)>(.*?)<\/(b|strong)>/gi,"'''$2'''")
 
r(/<(i|em)>(.*?)<\/(i|em)>/gi,"''$2''")
 
r(/^<hr ?\/?>/gim, '----')
 
r(/<\/?(hr|br)( [^\/>]+?)? ?\/?>/gi, '<$1$2 />')
 
r(/(\n== *[a-zа-я\s\.:]+ *==\n+)<references *\/>/ig,'$1{\{примечания}}')
 
hide(/<[a-z][^>]*?>/gi)
 
  
hide(/^({\||\|-).*/mg)//table/row def
+
    // Hyphens and en dashes to pretty dashes
hide(/(^\||^!|!!|\|\|) *[a-z]+=[^|]+\|(?!\|)/mgi)//cell style
+
    r(/–/g, '-'); // &ndash; -> hyphen
hide(/\| +/g)//formatted cell
+
    r(/(\s)-{1,3} /g, '$1— '); // hyphen -> &mdash;
 +
    r(/(\d)--(\d)/g, '$1—$2'); // -> &mdash;
 +
    r(/(\s)-(\d)/g, '$1−$2'); // hyphen -> minus
  
r(/[ \t]+/g,' ')//double spaces
+
    // 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');
  
// Headings
+
    // Reductions
r(/^(=+)[ \t\f\v]*(.*?)[ \t\f\v]*=+$/gm, '$1 $2 $1') //add spaces inside
+
    r(/(Т|т)\.\s?е\./g, '$1о есть');
r(/([^\r\n])(\r?\n==.*==\r?\n)/g, '$1\n$2') //add empty line before
+
    r(/(Т|т)\.\s?к\./g, '$1ак как');
r(/^== см(\.?|отри|отрите) ?также ==$/gmi, '== См. также ==')
+
    r(/(В|в)\\. ?ч\./g, '$1 том числе');
r(/^== сноски ==$/gmi, '== Примечания ==')
+
    r(/(И|и)\sт\.\s?д\./g, '$1' + u + 'т.' + u + 'д.');
r(/^== внешние\sссылки ==$/gmi, '== Ссылки ==')
+
    r(/(И|и)\\.\s?п\./g, '$1' + u + 'т.' + u + 'п.');
r(/^== (.+)[.:] ==$/gm, '== $1 ==')
+
    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 ');
  
r(/«|»||”|„/g, '"')//temp
+
    // 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, ')');
  
// Hyphens and en dashes to pretty dashes
+
    // Temperature
r(/–/g, '-') //&ndash; ->  hyphen
+
    r(/([\s\d=≈≠≤≥<>—("'|])([+±−\-]?\d+?(?:[.,]\d+?)?)(([ °\^*]| \^*])(C|F))(?=[\s"').,;!?|\x01])/gm, '$1$2' + u + '°$5'); //'
r(/&(#151|[nm]dash);/g, '—') // -> &mdash;
 
r(/(&nbsp;|\s)-{1,3} /g, '$1— ') // hyphen -> &mdash;
 
r(/(\d)--(\d)/g, '$1—$2') // -> &mdash;
 
  
// Entities etc. Unicode chars
+
    // Dot comma in numbers
r(/&#x([0-9a-f]{1,4});/gi, function(n,a){return String.fromCharCode(eval('0x'+a.substr(-4)))})  //&#x301;
+
    r(/(\s\d+)\.(\d+[\u00A0 ]*[%‰°×])/gi, '$1,$2');
r(/&copy;/gi,'©')
 
r(/&reg;/gi,'®')
 
r(/&sect;/gi,'§')
 
r(/&euro;/gi,'€')
 
r(/&yen;/gi,'¥')
 
r(/&pound;/gi,'£')
 
r(/&deg;/g,'°')
 
r(/\(tm\)|&trade;/gi,'™')
 
r(/\.\.\.|&hellip;/g,'…')
 
r(/\+-(?!\+|-)|&plusmn;/g,'±')
 
r(/~=/g,'≈')
 
r(/\^2(\D)/g,'²$1')
 
r(/\^3(\D)/g,'³$1')
 
r(/&((la|ra|bd|ld)quo|quot);/g,'"')
 
r(/([\wа-яА-ЯёЁ])'([\wа-яА-ЯёЁ])/g,'$1’$2') //'
 
r(/№№/g,'№')
 
  
// Year and century ranges
+
    // Plugins
r(/(\(|\s)([12]?\d{3})[\u00A0 ]?(-{1,3}|—) ?([12]?\d{3})(?![\w-])/g, '$1$2—$4')
+
    for (i in window.wfPlugins) {
r(/([12]?\d{3}) ?(гг?\.)/g, '$1'+u+'$2')
+
      if (window.wfPlugins.hasOwnProperty(i)) {
r(/(\(|\s)([IVX]{1,5})[\u00A0 ]?(-{1,3}|—) ?([IVX]{1,5})(?![\w-])/g, '$1$2—$4')
+
        window.wfPlugins[i](txt, r);
r(/([IVX]{1,5}) ?(вв?\.)/g, '$1'+u+'$2')
+
      }
 +
    }
  
// Reductions
+
    // "" → «»
r(/(Т|т)\.\s?е\./g, '$1о есть')
+
    for (i = 1; i <= 2; i++) {
r(/(Т|т)\.\s?к\./g, '$1ак как')
+
      r(/([\s\x02!|#'"\/(;+\-])"([^"]*)([^\s"(|])"([^a-zа-яё])/ig, '$1«$2$3»$4'); //"
r(/(В|в)\sт\. ?ч\./g, '$1 том числе')
+
    }
r(/(И|и)\sт\.\s?д\./g, '$1'+u+'т\.'+u+'д\.')
+
    while (/«[]*«/.test(txt)) {
r(/(И|и)\sт\.\s?п\./g, '$1'+u+'т\.'+u+'п\.')
+
      r(/«([]*)«([^»]*)»/g, '«$1„$2“');
r(/(Т|т)\.\s?н\./g, '$1\.'+u+'н\.')
+
    }
r(/(И|и)\.\s?о\./g, '$1\.'+u+'о\.')
 
r(/н\.\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
+
    if ('0'.replace('0', '$$') === '$') { ////$ in replacing string is special, except in IE
r(/^([#*:]+)[ \t\f\v]*(?!\{\|)([^ \t\f\v*#:;])/gm, '$1 $2') //space after #*: unless before table
+
      for (i = 0; i < hidden.length; i++) {
r(/(\S) (-{1,3}|—) (\S)/g, '$1'+u+'— $3')
+
        hidden[i] = hidden[i].replace(/\$/g, '$$$$');
r(/([А-Я]\.) ?([А-Я]\.) ?([А-Я][а-я])/g, '$1'+u+'$2'+u+'$3')
+
      }
r(/([А-Я]\.)([А-Я]\.)/g, '$1 $2')
+
    }
r(/([а-я]\.)([А-ЯA-Z])/g, '$1 $2') // word. word
+
    while (hidden.length > 0) {
r(/([)"а-яa-z\]])\s*,([\[("а-яa-z])/g, '$1, $2') // word, word
+
      r('\x01' + hidden.length + '\x02', hidden.pop());
r(/([)"а-яa-z\]])\s([,;])\s([\[("а-яa-z])/g, '$1$2 $3')
+
    }
r(/([^%\/\w]\d+?(?:[.,]\d+?)?) ?([%‰])(?!-[А-Яа-яЁё])/g, '$1'+u+'$2') //5 %
+
    txt = txt.substr(1, txt.length - 2);
r(/(\d) ([%‰])(?=-[А-Яа-яЁё])/g, '$1$2') //5%-й
+
  }
r(/([№§])(\s*)(\d)/g, '$1'+u+'$3')
 
r(/\( +/g, '('); r(/ +\)/g, ')') //inside ()
 
  
//Temperature
+
  function processAllText() {
r(/([\s\d=≈≠≤≥<>—("'|])([+±−-]?\d+?(?:[.,]\d+?)?)(([ °^*]| [°^*])C)(?=[\s"').,;!?|])/gm, '$1$2'+u+'°C') //'
+
    txt = wpTextbox1.value;
r(/([\s\d=≈≠≤≥<>—("'|])([+±−-]?\d+?(?:[.,]\d+?)?)(([ °^*]| [°^*])F)(?=[\s"').,;|!?])/gm, '$1$2'+u+'°F') //'
+
    processText();
 +
    r(/^[\n\r]+/, '');
 +
    wpTextbox1.value = txt;
 +
    txt = '';
 +
    if (window.auto_comment && window.insertSummary && !document.editform.wpSection.value) {
 +
      window.insertSummary('викификатор');
 +
    }
 +
  }
  
//Dot → comma in numbers
 
r(/(\s\d+)\.(\d+[\u00A0 ]*[%‰°])/gi, '$1,$2')
 
  
//"" → «»
+
  // Check regexp support
for (var i=1; i<=2; i++)
+
  try {
r(/([\s\x02!|#'"\/(;+-])"([^"]*)([^\s"(|])"([^a-zа-яё])/ig, '$1«$2$3»$4') //"
+
    txt = 'ая'.replace(/а/g, 'б').replace((?=я)/, 'в');
while (/«[^»]*«/.test(txt))
+
  }
r(/«([^»]*)«([^»]*)»/g, '«$1„$2“')
+
  catch (e) {}
 +
  if (txt !== 'вя') {
 +
    alert(wmCantWork);
 +
    return;
 +
  }
  
txt=txt.substr(1, txt.length-2)
+
  wpTextbox1.focus();
if ('0'.replace('0','$$') == '$') ////$ in replacing string is special, except in IE
 
for (var i=0; i<hidden.length; i++) hidden[i] = hidden[i].replace(/\$/g, '$$$$')
 
while (hidden.length>0)
 
r('\x01'+hidden.length+'\x02', hidden.pop())
 
  
}
+
  // Modern browsers
 
+
  if (typeof wpTextbox1.selectionStart !== 'undefined') {
function r(r1, r2){ txt = txt.replace(r1, r2) }
+
    var textScroll = wpTextbox1.scrollTop,
function hide(re){ r(re, function(s){return '\x01'+hidden.push(s)+'\x02'})}
+
      startPos = wpTextbox1.selectionStart,
function hideTag(tag){ hide(RegExp('<' + tag + '( [^>]+)?>[\\s\\S]+?<\\/' + tag + '>','gi')) }
+
      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
 
}
 
}

Текущая версия на 10:33, 30 января 2024

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, '-'); // &ndash; -> hyphen
    r(/(\s)-{1,3} /g, '$1— '); // hyphen -> &mdash;
    r(/(\d)--(\d)/g, '$1—$2'); // -> &mdash;
    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
}