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

Материал из Ролевая энциклопедии
Перейти к: навигация, поиск
(начало установки)
 
(давно пора новую версию нам перетянуть)
Строка 3: Строка 3:
 
/* <pre> */  
 
/* <pre> */  
  
var txt;
+
var wmVersion = '2010-04-03'
+
var wmCantWork = 'Викификатор не может работать в вашем браузере\n\nWikificator can not work in your browser'
var wmFullText = 'Wikificator will process ALL text on this page. Continue?';
+
var wmFullText = 'Викификатор обработает ВЕСЬ текст на этой странице. Продолжить?'
var wmCantWork = 'Wikificator cannot work in your browser';
+
var wmTalkPage = 'Викификатор не обрабатывает страницы обсуждения целиком.\n\nВыделите ваше сообщение — обработано будет только оно'
var wmWontWork = 'Wikificator will not work in Netscape 4.x and less';
 
  
 +
function Wikify(){
 +
var txt='', hidden = [], wpTextbox1 = document.editform.wpTextbox1
 +
var winScroll = document.documentElement.scrollTop
  
var wmCategoryNS = 'Категория';
+
try {txt='ая'.replace(/а/g,'б').replace(/б(?=я)/,'в')} catch(e){}//check regexp support
var wmTemplateNS = 'Шаблон';
+
if (txt != 'вя' ||
var wmUserNS = 'Участник';
+
  (navigator.appName=='Netscape' && navigator.appVersion.substr (0, 1) < 5))
var wmImageNS = 'Изображение';
+
  { alert(wmCantWork); return }
var wmMediaNS = 'Медиа';
 
  
var wmLocaleNS = new Array ( wmCategoryNS, wmTemplateNS, wmUserNS, wmImageNS, wmMediaNS );
+
wpTextbox1.focus()
var wmEnNS = new Array ( 'category', 'template', 'user', 'image', 'media');
 
  
 +
if (typeof wpTextbox1.selectionStart != 'undefined'
 +
    && (navigator.productSub > 20031000 || is_safari || is_opera)) { //Mozilla/Opera/Safari3
 +
    var textScroll = wpTextbox1.scrollTop
 +
    var startPos = wpTextbox1.selectionStart
 +
    var 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
  
if (window.event){
+
}else if (document.selection && document.selection.createRange) { //IE
  document.onkeypress = pressed;
+
  var range = document.selection.createRange()
}
+
  txt = range.text
 
+
  if (txt == '') processAllText()
function pressed() //On Ctrl+Enter (MSIE)
+
  else{
{key = window.event.keyCode;if (key==10){Wikify();}}
+
    processText()
//======================================
+
    range.text = txt
function Wikify()
+
    if (range.moveStart) range.moveStart('character', - txt.length)
{
+
     range.select()
check_regexp(); // Check whether regular expressions are supported
 
 
 
// Только если пользователь указал в персональном скрипте auto_comment = 1;
 
 
 
// Раскоментировал, теперь не должно ругаться (добавил определение переменной в общий монобук)
 
// - если все равно ругается откатите и забаньте меня
 
 
 
if (auto_comment == 1) {
 
  var wpS = document.editform.wpSummary;
 
  if (wpS.value != '' && wpS.value.charAt(wpS.value.length-2) != '/') {
 
     wpS.value += ', Викификатор'
 
  } else {
 
    wpS.value += 'Викификатор'
 
 
   }
 
   }
  }
+
 
 +
  }else // other browsers
 +
  if (confirm(wmFullText)) processAllText()
  
document.editform.wpTextbox1.focus();
+
  document.documentElement.scrollTop = winScroll // scroll back, for IE/Opera
var txtarea = document.editform.wpTextbox1;
 
if(document.selection && !is_gecko)/* IE */ {
 
txt = " "+document.selection.createRange().text;
 
if (txt == " ") {all_text();} // If nothing was selected;
 
else{
 
Process();
 
txt = txt.substr (1, txt.length-1);
 
document.selection.createRange().text = txt;
 
}
 
}
 
else if((txtarea.selectionStart || txtarea.selectionStart == '0')&&(navigator.productSub>20031000)) /*Gecko-browsers older then 10.2003*/  {
 
var startPos = txtarea.selectionStart;
 
var endPos = txtarea.selectionEnd;
 
var scrollTop=txtarea.scrollTop;
 
txt = " "+(txtarea.value).substring(startPos, endPos);
 
if (txt == " ") {all_text();} // If nothing was selected;
 
else{
 
Process();
 
txt = txt.substr (1, txt.length-1);
 
txtarea.value = txtarea.value.substring(0, startPos) + txt + txtarea.value.substring(endPos, txtarea.value.length);
 
txtarea.focus();
 
}
 
}
 
else{if (confirm(wmFullText)) {all_text();}} // Other browsers
 
}
 
//======================================
 
function all_text()// Process all text
 
{
 
txt = " "+document.editform.wpTextbox1.value;
 
Process();
 
txt = txt.substr (1, txt.length-1);
 
document.editform.wpTextbox1.value=txt;
 
}
 
//======================================
 
function check_regexp()// Check whether regular expressions are supported
 
{
 
var reg1 = "code";
 
reg1 = reg1.replace(/d/g, "r");
 
if (reg1 != "core"){alert(wmCantWork);exit;}
 
b_ver = navigator.appVersion.substr (0, 1);
 
if (navigator.appName=="Netscape"&&b_ver<5){alert(wmWontWork);exit;}
 
return ;
 
}
 
function Process()
 
// We have 2 more pairs of safe chars in \x1E — \x1F !
 
{
 
if (wgNamespaceNumber % 2 || wgNamespaceNumber==4) {
 
var sigs = txt.match(/\d\d:\d\d, \d\d? \S{3,9} 20\d\d \(UTC\)/g);
 
if (sigs && sigs.length > 1)
 
  if (!confirm('Пожалуйста, не обрабатывайте Викификатором реплики других участников. Вы уверены, что хотите продолжить?'))
 
    return;
 
}
 
//var nowiki = ReplaceElements( '\<math\>(.|\r|\n)+?\<\/math\>, "\x03", "\x04" );
 
//That variant will make ReplaceTags() function unnecessary, but it's so ugly...
 
  
var nowiki = ReplaceTags( 'nowiki', "\x03", "\x04" );
 
var pre = ReplaceTags( 'pre', "\x12", "\x13" );
 
var code = ReplaceTags( 'code', "\x1c", "\x1d" );
 
var math = ReplaceTags( 'math', "\x05", "\x06" );
 
var gallery = ReplaceTags( 'gallery', "\x14", "\x15" );
 
  
// Exclude lines starting with space
+
//functions
f_space = txt.substr (0, 1)
 
txt = txt.substr (1, txt.length-1)
 
var sp_lines = ReplaceElements( "^( )(.+)$", "\x16", "\x17" );
 
txt = f_space + txt
 
 
 
ProcessNS( wmEnNS , wmLocaleNS );
 
CorrectRanges();
 
 
 
// Exclude templates and internal links
 
var templates = ReplaceElements( "\\{\\{(.|\\r|\\n)+?\\}\\}", "\x18", "\x19" );
 
var links = ReplaceElements( "(\\[\\[)(.*?)(\\||\\]\\])", "\x10", "\x11" );
 
var ext_links = ReplaceElements( "\\[(http|https|ftp|tftp|news|nntp|telnet|irc|gopher)://(.*?)\\]", "\x1A", "\x1B");
 
 
 
HTML2Wiki();
 
 
 
// Exclude tags and tag attributes (all text in quotes after "=" sign)
 
var attrs = ReplaceElements( '(=)(\\s?)(\\' + '")(.*?)(\\")', "\x0E", "\x0F");
 
var tags = ReplaceElements( "<([^>]*?)>", "\x01", "\x02");
 
 
 
ProcessTypography();
 
ProcessTypography(); // Second call
 
 
 
//alert(txt);
 
 
 
RestoreElements( tags, "\x01", "\x02");
 
RestoreElements( attrs, "\x0E", "\x0F");
 
RestoreElements( ext_links, "\x1A", "\x1B" );
 
RestoreElements( links, "\x10", "\x11" );
 
RestoreElements( templates, "\x18", "\x19" );
 
RestoreElements( sp_lines, "\x16", "\x17" );
 
RestoreElements( gallery, "\x14", "\x15" );
 
RestoreElements( math, "\x05", "\x06" );
 
RestoreElements( code, "\x1c", "\x1d" );
 
RestoreElements( pre, "\x12", "\x13" );
 
RestoreElements( nowiki, "\x03", "\x04" );
 
  
 +
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 HTML2Wiki()
+
function processText(){
{
 
  
 
+
var u = '\u00A0' //unbreakable space
// Replace <b>, <strong> tags with ''' and <i>, <em> with ''
+
if (wgNamespaceNumber % 2 || wgNamespaceNumber==4) { //is talk page
txt = txt.replace(/\<\/?(b|strong)\>/gim, "\'\'\'")
+
u = ' '
txt = txt.replace(/\<\/?(i|em)\>/gim, "\'\'")
+
var sigs = txt.match(/\d\d:\d\d, \d\d? \S{3,8} 20\d\d \(UTC\)/g)
 
+
if (sigs && sigs.length > 1) {
// Replace <hr> tag with ----, improve <hr> and <br> tags
+
  alert(wmTalkPage); return
txt = txt.replace(/\<hr ?\/?\>/gi, "----")
+
}
txt = txt.replace(/\<hr ([^\>\/]+?) ?\/?\>/gi, "<hr $1 />")
 
txt = txt.replace(/\<br\/?\>/gi, "<br />")
 
txt = txt.replace(/\<br ([^\>\/]+?) ?\/?\>/gi, "<br $1 />")
 
 
 
// Replace small and big tags with inline styling
 
 
 
/// txt = txt.replace(/(\<small\>)/g, "<span style=\"font-size\:smaller\;\">");
 
/// txt = txt.replace(/(\<\/small\>)/g, "<\/span>");
 
/// txt = txt.replace(/(\<big\>)/g, "<span style=\"font-size\:1.25em\;\">");
 
/// txt = txt.replace(/(\<\/big\>)/g, "<\/span>");
 
 
 
}
 
 
 
// Process default namespaces
 
 
 
function ProcessNS( En_NS_List , Loc_NS_List )
 
{
 
 
 
for (i=0; i < En_NS_List.length; i++)
 
  {
 
  //alert("(\\[\\[)(:?)(" + En_NS_List[i] + "|" +
 
  //Loc_NS_List[i] + ")(:)( *)");
 
 
 
  txt = txt.replace( new RegExp( "(\\[\\[:?)(" + En_NS_List[i] + "|" +
 
  Loc_NS_List[i] + "):( *)" , "gi" ), "$1" + Loc_NS_List[i] + ":");
 
 
 
  }
 
 
}
 
}
  
//======================================
+
hideTag('nowiki')
// Replace '<replaced_tag> ... </replaced_tag>' (<nowiki> <br/> </nowiki>)
+
hideTag('pre')
// with 'opepening_char + tag's counter + closing_char' ('\x03'+1'+'\x04')
+
hideTag('source')
//======================================
+
hideTag('code')
 
+
hideTag('tt')
function ReplaceTags(replaced_tag, op_char, cl_char )
+
hideTag('math')
 
+
hide(/{\{[\s\S]+?}}/g)//templates
// @replaced_tag - tag to be replaced
+
hide(/^ .*/mg)
// @op_char, @cl_char (opening & closing chars) - "Safe" pair of
+
hide(/(https?|ftp|news|nntp|telnet|irc|gopher):\/\/[^\s\[\]<>"]+ ?/gi)
// unicode unprintable characters, that will be used in replacement
+
hide(/^#(redirect|перенапр(авление)?)/i)
 +
hideTag('gallery')
  
{
 
  
var counter = 0; //tags counter
+
r(/ +(\n|\r)/g,'$1')//spaces at EOL
 +
txt = '\n'+txt+'\n'
  
// RegExp pattern
 
var pattern = "\\<" + replaced_tag + "\\>(.|\r|\n)+?\<\\/" + replaced_tag + "\\>";
 
  
// RegExp template to be replaced (multiline text between
 
// <replaced_tag> and </replaced_tag> case sensitive tags)
 
var replaced_regexp = new RegExp(pattern , "im")
 
  
// Buffer for replaced matches storage. It's array of matching substrings -
+
//LINKS
// multiline text between <replaced_tag> and </replaced_tag> case sensitive tags ()
+
r(/(\[\[:?)(category|категория):( *)/ig, '$1Категория:')
matches_buffer = txt.match( new RegExp(pattern , "gim") );
+
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(/\[\[ *([a-zA-Zа-яёА-ЯЁ\u00A0-\u00FF %!\"$&'()*,\-—.\/0-9:;=?\\@\^_`’~]+) *\| *(\1)([a-zа-яё]*) *\]\]/g, '[[$2]]$3') // "
 +
r(/\[\[ *([^|\[\]]+)([^|\[\]]+) *\| *\1 *\]\]\2/g, '[[$1$2]]') // -повтор текста за ссылкой
 +
r(/\[\[ *(?!Файл:|Категория:)([a-zA-Zа-яёА-ЯЁ\u00A0-\u00FF %!\"$&'()*,\-—.\/0-9:;=?\\@\^_`’~]+) *\| *([^|[\]]+) *\]\]([a-zа-яё]+)/g, '[[$1|$2$3]]') // "
 +
hide(/\[\[[^\]|]+/g)//only link part
  
// while some substring of txt matches replaced_regexp...
 
while (replaced_regexp.test(txt))
 
  {
 
  txt = txt.replace(replaced_regexp, op_char + ++counter + cl_char );
 
  }
 
return matches_buffer;
 
}
 
  
//======================================
+
//TAGS
// Replace '<replaced_tag> ... </replaced_tag>' (<nowiki> <br/> </nowiki>)
+
r(/<<(\S.+\S)>>/g, '"$1"') //<< >>
// with 'opepening_char + tag's counter + closing_char' ('\x03'+1'+'\x04')
+
r(/(sup>|sub>|\s)-(\d)/g, '$1−$2') //minus
//======================================
+
r(/(<sup>2<\/sup>|&sup2;)/gi, '²');
 +
r(/(<sup>3<\/sup>|&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)
  
function ReplaceElements( req_exp, op_char, cl_char )
+
hide(/^({\||\|-).*/mg)//table/row def
 +
hide(/(^\||^!|!!|\|\|) *[a-z]+=[^|]+\|(?!\|)/mgi)//cell style
 +
hide(/\| +/g)//formatted cell
  
// @req_exp - reqular expression to be replaced
+
r(/[ \t]+/g,' ')//double spaces
// @op_char, @cl_char (opening & closing chars) - "Safe" pair of
 
// unicode unprintable characters, that will be used in replacement
 
  
{
+
// 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 ==')
  
var counter = 0; //tags counter
+
r(/«|»|“|”|„/g, '"')//temp
  
// RegExp template to be replaced (multiline, case sensitive)
+
// Hyphens and en dashes to pretty dashes
var replaced_regexp = new RegExp( req_exp , "m" )
+
r(/–/g, '-') //&ndash; ->  hyphen
 
+
r(/&(#151|[nm]dash);/g, '—') // -> &mdash;
// Buffer for replaced matches storage. It's array of matching substrings.
+
r(/(&nbsp;|\s)-{1,3} /g, '$1— ') // hyphen -> &mdash;
// (multiline, case sensitive, global)
+
r(/(\d)--(\d)/g, '$1—$2') // -> &mdash;
matches_buffer = txt.match( new RegExp( req_exp , "gm" ) );
 
 
 
// while some substring of txt matches replaced_regexp...
 
while (replaced_regexp.test(txt))
 
  {
 
  //alert(txt.match(replaced_regexp));
 
  txt = txt.replace(replaced_regexp, op_char + ++counter + cl_char );
 
  }
 
return matches_buffer;
 
}
 
  
//======================================
+
// Entities etc. → Unicode chars
// Restore text, that was damaged by replacing 3 chars with substring from array
+
r(/&#x([0-9a-f]{1,4});/gi, function(n,a){return String.fromCharCode(eval('0x'+a.substr(-4)))})  //&#x301;
//======================================
+
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,'№')
  
function RestoreElements( replaced_buffer, op_char, cl_char )
+
// Year and century ranges
 +
r(/(\(|\s)([12]?\d{3})[\u00A0 ]?(-{1,3}|—) ?([12]?\d{3})(?![\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')
  
// @replaced_buffer - array of replaced substrings.
+
// Reductions
// @op_char, @cl_char (opening & closing chars) - "Safe" pair
+
r(/(Т|т)\.\s?е\./g, '$1о есть')
// to be replaced with <replaced_tag> and </replaced_tag> accordingly
+
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?э\./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 #*:
 +
r(/(\S) (-{1,3}|—) (\S)/g, '$1'+u+'— $3')
 +
r(/([А-Я]\.) ?([А-Я]\.) ?([А-Я][а-я])/g, '$1'+u+'$2'+u+'$3')
 +
r(/([А-Я]\.)([А-Я]\.)/g, '$1 $2')
 +
r(/([а-я]\.)([А-ЯA-Z])/g, '$1 $2') // word. word
 +
r(/([)"а-яa-z\]])\s*,([\[("а-яa-z])/g, '$1, $2') // 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')
 +
r(/\( +/g, '('); r(/ +\)/g, ')') //inside ()
  
var counter = 0; //tags counter
+
//Temperature
 +
r(/([\s\d=≈≠≤≥<>—("'|])([+±−-]?\d+?(?:[.,]\d+?)?)(([ °^*]| [°^*])[CС])(?=[\s"').,;!?|])/gm, '$1$2'+u+'°C') //'
 +
r(/([\s\d=≈≠≤≥<>—("'|])([+±−-]?\d+?(?:[.,]\d+?)?)(([ °^*]| [°^*])F)(?=[\s"').,;|!?])/gm, '$1$2'+u+'°F') //'
  
// RegExp template to be replaced (3 chars: tag's counter
+
//Dot → comma in numbers
// surrunded by "Safe" pair)
+
r(/(\s\d+)\.(\d+[\u00A0 ]*[%‰°])/gi, '$1,$2')
var replaced_regexp = new RegExp("\\" +op_char+ "([0-9]*)\\" +cl_char );
 
  
//replaced_regexp = /\x03([0-9]*)\x04/
+
//"" → «»
 +
for (var i=1; i<=2; i++)
 +
r(/([\s\x02!|#'"\/(;+-])"([^"]*)([^\s"(|])"([^a-zа-яё])/ig, '$1«$2$3»$4') //"
 +
while (/«[^»]*«/.test(txt))  
 +
r(/«([^»]*)«([^»]*)»/g, '«$1„$2“')
  
// while some substring of txt matches replaced_regexp...
+
txt=txt.substr(1, txt.length-2)
while (replaced_regexp.test(txt))
+
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, '$$$$')
  txt = txt.replace(replaced_regexp, replaced_buffer[counter++]);
+
while (hidden.length>0)
}
+
r('\x01'+hidden.length+'\x02', hidden.pop())
return txt;
 
}
 
  
// Corrects year and century ranges (as links) in text
 
function CorrectRanges() {
 
// Correct year ranges
 
txt = txt.replace(/(?!ISBN)(\(|\s)(\[\[[12]?\d{3}\]\])[\u00A0 ]?(-|--|–|—) ?(\[\[[12]?\d{3}\]\])(\W)/g, "$1$2—$4$5")
 
txt = txt.replace(/(\[\[[12]?\d{3}\]\]) ?(г\.|гг\.)/g, "$1\u00A0$2")
 
// Correct century ranges
 
txt = txt.replace(/(\(|\s)(\[\[[IVX]{1,5}\]\])[\u00A0 ]?(-|--|–|—) ?(\[\[[IVX]{1,5}\]\])(\W)/g, "$1$2—$4$5")
 
txt = txt.replace(/(\[\[[IVX]{1,5}\]\]) ?(в\.|вв\.)/g, "$1\u00A0$2")
 
 
}
 
}
  
/***************************************************
+
function r(r1, r2){ txt = txt.replace(r1, r2) }
Typographical considerations
+
function hide(re){ r(re, function(s){return '\x01'+hidden.push(s)+'\x02'})}
***************************************************/
+
function hideTag(tag){ hide(RegExp('<' + tag + '( [^>]+)?>[\\s\\S]+?<\\/' + tag + '>','gi')) }
function ProcessTypography() {
 
 
 
// Insert spaces in titles
 
txt = txt.replace(/^(=+)([ \t\f\v]*)(.*?)([ \t\f\v]*)(=+)$/gm, "$1 $3 $1")
 
//======================================
 
// Use 1 character to display squaring and cubing
 
txt = txt.replace(/(<sup>2<\/sup>|&sup2;)/g, "²");
 
txt = txt.replace(/(<sup>3<\/sup>|&sup3;)/g, "³");
 
txt = txt.replace(/(\^2)(\D)/g, "²$2");
 
txt = txt.replace(/(\^3)(\D)/g, "³$2");
 
//======================================
 
// Replace right HTML symbols with wrong ones in order to process everything
 
txt = txt.replace(/–/g, "-")
 
txt = txt.replace(/(«|»|“|”|„|\&((la|ra|bd|ld)quo|#132|#147|#148|quot);)/g, "\"")
 
//======================================
 
// Replace double hyphen with a dash
 
txt = txt.replace(/(--)(\[\[Участник|\~\~\~)/g, "—$2")
 
//======================================
 
// Replace set of 'less then' or 'greater then' symbols (<< or >>) with usual double quotes
 
txt = txt.replace(/(<<)(\S.+\S)(>>)/g, "\"$2\"")
 
//======================================
 
// Process degree sign "°", "+-" and "~="
 
txt = txt.replace(/(\+[--])|(&plusmn;)/g, "±")
 
txt = txt.replace(/(~=)/g, "≈")
 
txt = txt.replace(/\&deg;/g, "°")
 
txt = txt.replace(/([ =≈≠≤≥<>("'|]|^)([+±−\-]?\d+?(?:[.,]\d+?)?)(([ °^*]| [°^*])[CС])(?=[ "').,;!?|]|$)/gm, "$1$2\u00A0°C")
 
txt = txt.replace(/([ =≈≠≤≥<>("'|]|^)([+±−\-]?\d+?(?:[.,]\d+?)?)(([ °^*]| [°^*])F)(?=[ "').,;|!?]|$)/gm, "$1$2\u00A0°F")
 
//======================================
 
// Replace "...", "&hellip;" and "&#133;" with ellipsis
 
txt = txt.replace(/(\.{3}|\&(hellip|#133);)/g, '…')
 
// Apostrophe handler
 
txt = txt.replace(/([\wа-яА-ЯёЁ])'([\wа-яА-ЯёЁ])/g, "$1’$2")
 
// Minus handler
 
txt = txt.replace(/(sup\>|sub\>|\s)-(\d)/g, "$1−$2")
 
//======================================
 
// Replace hyphens and en dashes with normal dashes
 
txt = txt.replace(/\&(#151|[nm]dash);/g, "—")
 
txt = txt.replace(/(&nbsp;|[\f\n\r\t\v\u00A0\u2028\u2029])(-|--|–) /g, "$1— ")
 
txt = txt.replace(/(\d)--(\d)/g, "$1—$2")
 
// Insert non-breaking space before dashes
 
txt = txt.replace(/(\S) (-|--|–|—) (\S)/g, "$1\u00A0— $3")
 
//======================================
 
// Special characters: ©, ®, ™, §, €, ¥ и £.
 
txt = txt.replace(/\&copy;/gi, "©")
 
txt = txt.replace(/\&reg;/gi, "®")
 
txt = txt.replace(/(\((tm|тм)\)|\&trade;)/gi, "™")
 
txt = txt.replace(/\&sect;/gi, "§")
 
txt = txt.replace (/\&euro;/gi, "€")
 
txt = txt.replace (/\&yen;/gi, "¥")
 
txt = txt.replace (/\&pound;/gi, "£")
 
//======================================
 
// Correct year ranges
 
txt = txt.replace(/(\(|\s)([12]?\d{3})[\u00A0 ]?(-|--|–|—) ?([12]?\d{3})(\W)/g, "$1$2—$4$5")
 
txt = txt.replace(/([12]?\d{3}) ?(г\.|гг\.)/g, "$1\u00A0$2")
 
// Correct century ranges
 
txt = txt.replace(/(\(|\s)([IVX]{1,5})[\u00A0 ]?(-|--|–|—) ?([IVX]{1,5})(\W)/g, "$1$2—$4$5")
 
txt = txt.replace(/([IVX]{1,5}) ?(в\.|вв\.)/g, "$1\u00A0$2")
 
// Correct the reductions
 
txt = txt.replace(/(Т|т)\. ?е\./g, "$1о есть")
 
txt = txt.replace(/(Т|т)\. ?к\./g, "$1ак как")
 
txt = txt.replace(/(В|в) т\. ?ч\./g, "$1 том числе")
 
txt = txt.replace(/и т\. ?д\./g, "и\u00A0т\.\u00A0д\.")
 
txt = txt.replace(/и т\. ?п\./g, "и\u00A0т\.\u00A0п\.")
 
txt = txt.replace(/(Т|т)\. ?н\./g, "$1\.\u00A0н\.")
 
txt = txt.replace(/н\. ?э\./g, "н\.\u00A0э\.")
 
txt = txt.replace(/(Д|д)(о|\.) н\. ?э\./g, "$1о\u00A0н\.\u00A0э\.")
 
txt = txt.replace(/(\d) ?(млн|млрд|трлн|(?:м|с|д|к)?м|[км]?г|с)\.?( ([^\.А-ЯЁ])|[,;.])(?!\[.*?\|[А-Я].*?\])/g, "$1\u00A0$2$3")
 
txt = txt.replace(/(\d) (тыс)([^\.А-Яа-яЁё])/g, "$1\u00A0$2.$3")
 
 
 
//txt = txt.replace(/(\d) (млн|млрд|трлн)([^А-Яа-яЁё])/g, "$1\u00A0$2$3")
 
// Insert missing and delete unnecessary spaces
 
txt = txt.replace(/([А-Я]\.) ?([А-Я]\.) ?([А-Я][а-я])/g, "$1\u00A0$2\u00A0$3")
 
txt = txt.replace(/([А-Я]\.)([А-Я]\.)/g, "$1 $2")
 
txt = txt.replace(/^([#\*:]+)([ \t\f\v]*)([^ \t\f\v\*#:])/gm, "$1 $3")
 
txt = txt.replace(/([а-я])(\.)([А-ЯA-Z])/g, "$1$2 $3")
 
txt = txt.replace(/([а-яa-z\)\»\“\"\]])(\s*)(\,)([а-яa-z\(\«\„\"\[])/g, "$1$3 $4")
 
txt = txt.replace(/([а-яa-z\)\»\“\"\]])(\s)([\,\;])(\s)([а-яa-z\(\«\„\"\[])/g, "$1$3 $5")
 
txt = txt.replace(/([^%\/\w]\d+?(?:[.,]\d+?)?) ?([%‰])(?!-[А-Яа-яЁё])/g, "$1\u00A0$2")
 
txt = txt.replace(/(\d) ([%‰])(?=-[А-Яа-яЁё])/g, "$1$2")
 
txt = txt.replace(/([№§])(\s*)(\d)/g, "$1\u00A0$3")
 
txt = txt.replace(/(^|[^ \t])([ \t]+)($|\n)/gm, "$1$3")
 
txt = txt.replace(/(\()( +)/g, "$1");
 
txt = txt.replace(/( +)(\))/g, "$2");
 
//======================================
 
// Avoid double spaces
 
txt = txt.substr (1, txt.length-1);
 
txt = txt.replace(/(\S)([ \t]{2,})([\S\r])/g, "$1 $3")
 
txt = " " + txt
 
//======================================
 
// Replace double quotes ("")  with double angle quotes («»)
 
txt = txt.replace(/([\x01-(\s\|\"]|\/|\+)(\")([^\"]{0,})([^\s\"(\|])(\")/g, "$1«\$3\$4»")
 
// Quotations in quotes
 
if (/"/.test(txt))
 
{
 
  txt = txt.replace(/([\x01(\s\"])(\")([^\"]{0,})([^\s\"(\|])(\")/g, "\$1«\$3\$4»")
 
  while (/(«)([^»]*)(«)/.test(txt))
 
    txt = txt.replace(/(«)([^»]*)(«)([^»]*)(»)/g, "\$1\$2„\$4“")
 
}
 
  
 
}
 
}
  
 
/* </pre> */
 
/* </pre> */

Версия 14:23, 30 апреля 2010

// Копия [[Wikipedia:ru:MediaWiki:Wikificator.js]] согласно [[Wikipedia:ru:ВП:ВФ-ИУ]].

/* <pre> */ 

var wmVersion = '2010-04-03'
var wmCantWork = 'Викификатор не может работать в вашем браузере\n\nWikificator can not work in your browser'
var wmFullText = 'Викификатор обработает ВЕСЬ текст на этой странице. Продолжить?'
var wmTalkPage = 'Викификатор не обрабатывает страницы обсуждения целиком.\n\nВыделите ваше сообщение — обработано будет только оно'

function Wikify(){
 var txt='', hidden = [], wpTextbox1 = document.editform.wpTextbox1
 var 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()

 if (typeof wpTextbox1.selectionStart != 'undefined' 
    && (navigator.productSub > 20031000 || is_safari || is_opera)) { //Mozilla/Opera/Safari3
    var textScroll = wpTextbox1.scrollTop
    var startPos = wpTextbox1.selectionStart
    var 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

 }else if (document.selection && document.selection.createRange) { //IE
   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() 
   }
  
 }else // other browsers
   if (confirm(wmFullText)) processAllText()

 document.documentElement.scrollTop = winScroll // scroll back, for IE/Opera


//functions

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(){

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')
hideTag('pre')
hideTag('source')
hideTag('code')
hideTag('tt')
hideTag('math')
hide(/{\{[\s\S]+?}}/g)//templates
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(/\[\[ *([a-zA-Zа-яёА-ЯЁ\u00A0-\u00FF %!\"$&'()*,\-—.\/0-9:;=?\\@\^_`’~]+) *\| *(\1)([a-zа-яё]*) *\]\]/g, '[[$2]]$3') // "
r(/\[\[ *([^|\[\]]+)([^|\[\]]+) *\| *\1 *\]\]\2/g, '[[$1$2]]') // -повтор текста за ссылкой
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(/(sup>|sub>|\s)-(\d)/g, '$1−$2') //minus
r(/(<sup>2<\/sup>|&sup2;)/gi, '²');
r(/(<sup>3<\/sup>|&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
hide(/(^\||^!|!!|\|\|) *[a-z]+=[^|]+\|(?!\|)/mgi)//cell style
hide(/\| +/g)//formatted cell

r(/[ \t]+/g,' ')//double spaces

// 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(/«|»|“|”|„/g, '"')//temp

// Hyphens and en dashes to pretty dashes
r(/–/g, '-') //&ndash; ->  hyphen
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
r(/&#x([0-9a-f]{1,4});/gi, function(n,a){return String.fromCharCode(eval('0x'+a.substr(-4)))})  //&#x301;
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
r(/(\(|\s)([12]?\d{3})[\u00A0 ]?(-{1,3}|—) ?([12]?\d{3})(?![\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?э\./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 #*:
r(/(\S) (-{1,3}|—) (\S)/g, '$1'+u+'— $3')
r(/([А-Я]\.) ?([А-Я]\.) ?([А-Я][а-я])/g, '$1'+u+'$2'+u+'$3')
r(/([А-Я]\.)([А-Я]\.)/g, '$1 $2')
r(/([а-я]\.)([А-ЯA-Z])/g, '$1 $2') // word. word
r(/([)"а-яa-z\]])\s*,([\[("а-яa-z])/g, '$1, $2') // 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')
r(/\( +/g, '('); r(/ +\)/g, ')') //inside ()

//Temperature
r(/([\s\d=≈≠≤≥<>—("'|])([+±−-]?\d+?(?:[.,]\d+?)?)(([ °^*]| [°^*])[CС])(?=[\s"').,;!?|])/gm, '$1$2'+u+'°C') //'
r(/([\s\d=≈≠≤≥<>—("'|])([+±−-]?\d+?(?:[.,]\d+?)?)(([ °^*]| [°^*])F)(?=[\s"').,;|!?])/gm, '$1$2'+u+'°F') //'

//Dot → comma in numbers
r(/(\s\d+)\.(\d+[\u00A0 ]*[%‰°])/gi, '$1,$2')

//"" → «»
for (var i=1; i<=2; i++)
 r(/([\s\x02!|#'"\/(;+-])"([^"]*)([^\s"(|])"([^a-zа-яё])/ig, '$1«$2$3»$4') //"
while (/«[^»]*«/.test(txt)) 
 r(/«([^»]*)«([^»]*)»/g, '«$1„$2“')

txt=txt.substr(1, txt.length-2)
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())

}

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(RegExp('<' + tag + '( [^>]+)?>[\\s\\S]+?<\\/' + tag + '>','gi')) }

}

/* </pre> */