„Benutzer:TMg/autoFormatter.js“ – Versionsunterschied
Inhalt gelöscht Inhalt hinzugefügt
TMg (Diskussion | Beiträge) Datum in Vorlage:Charts nicht ausschreiben, hatte ich am 2. Feb. 2011 schon mal behoben, ist am 3. Nov. 2011 wieder rein gerutscht | TMg (Diskussion | Beiträge) Freischaltung der Betaversion für alle Benutzer; da ich inzwischen sogar einen Softwaretext dafür habe, bin ich optimistisch; Dokumentation der Neuerungen vorläufig unter BD:TMg/autoFormatter.js/Beta.js | ||
Zeile 1: | Zeile 1: | ||
/** | /** | ||
* Blendet eine „Auto-Format“-Funktion | * Blendet eine „Auto-Format“-Funktion in der Werkzeugleiste ein, die viele typische Wikifizierungs-Fehler | ||
* automatisch korrigiert. Eine ausführliche Beschreibung ist auf der Diskussionsseite zu finden. | |||
* viele typische Wikifizierungs-Fehler automatisch korrigiert. | |||
* (<nowiki> zur Umgehung von [[bugzilla:8761]].) | |||
* Eine ausführliche Beschreibung ist auf der Diskussionsseite zu finden. | |||
*/ | */ | ||
function | (function($, mw) | ||
{ | { | ||
var | var autoFormatter = { | ||
click: function(a) | |||
var e = document.forms['editform'].elements; | |||
{ | |||
var b = e['wpTextbox1']; | |||
this.clickedElement = a; | |||
var t = b.value; | |||
this.locale = typeof window.autoFormatLocalisation == 'undefined' | |||
? mw.config.get('wgContentLanguage') == 'de' | |||
: window.autoFormatLocalisation; | |||
var elements = document.forms['editform'].elements; | |||
var textbox = elements['wpTextbox1'], summary = elements['wpSummary']; | |||
if (!textbox) return; | |||
this.isDisambiguation = /\{\{\s*[Bb]egriffsklärung\s*\}\}/.test(textbox.value); | |||
this.cleanElement(textbox); | |||
summary.value = this.cleanInternalLinks(summary.value); | |||
return false; | |||
}, | |||
isChanged: function(oldValue, newValue) | |||
{ | |||
/* Entfernte Leerräume am Textende zählen nie als Änderung */ | |||
oldValue = oldValue.replace(/\s+$/, ''); | |||
newValue = newValue.replace(/\s+$/, ''); | |||
/* Entfernte Leerräume am Zeilenende nicht als Änderung anzeigen, aber trotzdem ersetzen */ | |||
var changed = oldValue.replace(/[ \t\r]+\n/g, '\n') !== newValue; | |||
var a = this.clickedElement; | |||
if (!a || !a.nodeType || a.nodeName == 'IMG') | |||
{ | |||
var e = $(a && a.nodeType ? a : 'img[rel=autoFormatter]'); | |||
e.css('backgroundColor', changed ? '#DEF740' : ''); | |||
e.css('borderRadius', changed ? '3px' : ''); | |||
e.css('opacity', changed ? '' : '.4'); | |||
} | |||
else if (a) a.style.color = changed ? 'green' : 'silver'; | |||
/* Normalisierte Zeilenumbrüche nie als Änderung werten, das vermeidet Flackern */ | |||
return changed || oldValue.replace(/\r+\n/g, '\n') !== newValue; | |||
}, | |||
cleanElement: function(e) | |||
{ | |||
e.focus(); | |||
if (typeof e.selectionStart == 'number') | |||
{ | |||
var scroll = e.scrollTop; | |||
var s1 = e.selectionStart, s2 = e.selectionEnd; | |||
if (s2 > s1) | |||
{ | |||
var t = this.cleanText(e.value.substring(s1, s2)); | |||
if (t === false) return; | |||
var newValue = e.value.substr(0, s1) + t + e.value.substr(s2); | |||
e.value = newValue; | |||
s2 = s1 + t.length + (e.value.length - newValue.length); /* Fix for Opera */ | |||
} | |||
else if (!this.cleanAll(e)) return; | |||
e.selectionStart = s1, e.selectionEnd = s2; | |||
e.scrollTop = scroll; | |||
} | |||
else if (typeof document.selection == 'object') /* Internet Explorer */ | |||
{ | |||
var range = document.selection.createRange(); | |||
if (range.text.length > 0) | |||
{ | |||
var t = this.cleanText(range.text); | |||
if (t !== false) range.text = t; | |||
} | |||
else this.cleanAll(e); | |||
} | |||
else this.cleanAll(e); | |||
}, | |||
cleanAll: function(e) | |||
{ | |||
var t = this.cleanText(e.value); | |||
if (t !== false) e.value = t.replace(/\s+$/, ''); | |||
return t !== false; | |||
}, | |||
cleanText: function(t) | |||
{ | |||
var oldValue = t; | |||
t = t.replace(/[ \t\r]+\n/g, '\n'); | |||
/* Steuerzeichen, undefinierte Unicode-Nummern, LRM, ZERO WIDTH SPACE, BOM entfernen */ | |||
/* Überflüssige Leerräume entfernen, aber das allein rechtfertigt keinen Edit */ | |||
t = t.replace(/[\x00-\x08\x0C\x0E-\x1F\x7F-\x9F\u200B\u200E\uFEFF]+/g, ''); | |||
/* Unsichtbares weiches Trennzeichen sichtbar machen, egal wo */ | |||
t = t.replace(/[ \t\r]+\n/g, '\n'); | |||
t = t.replace(/\xAD+/g, '­'); | |||
b.value = t; | |||
t = this.backupNowikis(t); | |||
/* Steuerzeichen, undefinierte Unicode-Nummern, LRM, ZERO WIDTH SPACE, BOM entfernen */ | |||
t = t.replace(/[\x00-\x08\x0C\x0E-\x1F\x7F-\x9F\u200B\u200E\uFEFF]+/g, ''); | |||
/* Mehrfache Leerzeilen auf einzelne reduzieren */ | |||
if (t.indexOf('{{Begriffsklärung}}') < 0) | |||
t = t.replace(/([ \t\r]*\n){3,}/g, '\n\n'); | |||
/* Geschützte Leerzeichen einheitlich als " ", vereinfacht viele folgende Suchmuster */ | |||
t = t.replace(/&#(x0*A|0*16)0 *;/gi, ' '); | |||
/* Die verschiedensten Formen der Zeilenumbrüche durch korrekte ersetzen */ | |||
t = t.replace(/<[\s\/]*br\s*(\s\w[^>]*?)?[\s\/]*>/gi, '<br$1 />'); | |||
/* Mehrfache Leerzeilen auf einzelne reduzieren */ | |||
/* Dateinamen retten incl. Vereinheitlichung als "Datei:" */ | |||
if (!this.isDisambiguation) t = t.replace(/\n{3,}/g, '\n\n'); | |||
var regex = /(^|\[\[)\s*(Bild|Datei|File|Image)\s*:\s*([^|\]\n]*?)\s*([|\]\n])/gim; | |||
/* Geschützte Leerzeichen einheitlich als „ “, vereinfacht viele folgende Suchmuster, | |||
var files = [], match; | |||
muss vor backupFilenames() stehen */ | |||
while (match = regex.exec(t)) files[files.length] = match; | |||
t = t.replace(/&#(?:x0*A|0*16)0 *;/gi, ' '); | |||
for (i = files.length; i--; ) | |||
/* Verschiedenste Formen von HTML-Zeilenumbrüchen durch einheitliche ersetzen */ | |||
{ | |||
t = t.replace(/<(?:[\s\/]*br\b)+\s*(\s\w[^<>]*?)?[\s\/]*>/gi, '<br$1 />'); | |||
if (!files[i][1]) files[i][0] = (files[i][1] = '\n') + files[i][0]; | |||
/* Unnötige HTML-Zeilenumbrüche entfernen, wenn sowieso ein Absatz folgt */ | |||
/* Keine geschützten Leerzeichen/Unterstriche in Dateinamen */ | |||
t = t.replace(/ *<br \/>(\n[\n#*:;])/gi, '$1'); | |||
/* Einheitliche Schreibweise und Leerzeichenausgleich */ | |||
t = t.replace(files[i][0], files[i][1] + 'Datei:' + i + files[i][4]); | |||
} | |||
t = this.backupFilenames(t); | |||
/* == Überschriften == immer mit Leerzeichen, repariert auch kaputte Überschriften */ | |||
t = t.replace(/^(=+) *(.*[^ =]) *\1$/gm, '$1 $2 $1'); | |||
t = this.cleanHeadlines(t); | |||
/* Weblinks immer als Weblinks */ | |||
if (this.locale) | |||
t = t.replace(/^== *(External|Externer?)? *(Links?|Weblinks?|Webseiten?|Websites?) *==/gim, | |||
{ | |||
'== Weblinks =='); | |||
/* „Weblinks“ immer als „Weblinks“ */ | |||
t = t.replace(/^== *(?:Externer?|External)? *(?:Weblinks?|Links?|Webseiten?|Websites?) *==/gim, | |||
'== Weblinks =='); | |||
/* Einheitliche Schreibweisen für Schlüsselwörter incl. Leerzeichenausgleich */ | |||
t = t.replace(/\{\{\s*(SEITENTITEL|DISPLAYTITLE):\s*/gi, '{{SEITENTITEL:'); | |||
t = t.replace(/\s*#(WEITERLEITUNG|REDIRECT)\s*\[\[/gi, '#WEITERLEITUNG [['); | |||
t = t.replace(/\[\[\s*[KC]ategor[iy]e?\s*:\s*/gi, '[[Kategorie:'); | |||
t = t.replace(/(\[\[Datei:[^\[\]\n]*\S)\s*\|+\s*(?:(?:rechts|right)\s*\|+\s*)?(?:miniatur|thumb)\s*\|+\s*((rechts|right)\s*\|+\s*)?/gi, | |||
'$1|miniatur|'); | |||
t = t.replace(/(\[\[Datei:[^\]]*?)\s*\|+\s*(hochkant|upright)\s*((=)\s*([\d.]*))?\s*\|+\s*/gi, '$1|hochkant$4$5|'); | |||
t = t.replace(/(\[\[Datei:[^\[\]\n]*\S)\s*\|+\s*(?:rechts|right)\s*\|+\s*/gi, '$1|rechts|'); | |||
t = t.replace(/(\[\[Datei:[^\[\]\n]*\S)\s*\|+\s*(?:links|left)\s*\|+\s*/gi, '$1|links|'); | |||
t = t.replace(/(\[\[Datei:[^\[\]\n]*\S)\s*\|+\s*(?:hochkant|upright)\s*(?:(=)\s*([\d.]*)\s*)?\|+\s*/gi, | |||
'$1|hochkant$2$3|'); | |||
t = t.replace(/\{\{\s*Wikisource\s*\|\s*/gi, '{{Wikisource|'); | |||
} | |||
t = t.replace(/\{\{\s*Wiktionary\s*\|\s*/gi, '{{Wiktionary|'); | |||
t = t.replace(/\{\{\s*(DEFAULTSORT|SORTIERUNG)\s*:\s*/gi, '{{SORTIERUNG:'); | |||
t = t.replace(/\s*#(REDIRECT|WEITERLEITUNG)\s*\[\[/gi, '#WEITERLEITUNG [['); | |||
t = t.replace(/\s*<\s*\/\s*ref\s*>/gi, '</ref>'); | |||
t = t.replace(/<\s*references\s*\/\s*>/gi, '<references />'); | |||
t = t.replace(/class\s*=\s*("(\s*[^\s"]+\s)*)?\s*\bprettytable\b/g, 'class=$1wikitable'); | |||
t = this.cleanTemplates(t); | |||
/* Sortierung soll nie Umlaute enthalten */ | |||
t = t.replace(/\bclass\s*=\s*("( *[\w-]+ )*)? *\bprettytable\b/g, 'class=$1wikitable'); | |||
var sortSrc = '¢£¥©ª®²³¹ºÀÁÂÃÄÅÇÈÉÊËÌÍÎÏÐÑÒÓÔÕÖØÙÚÛÜÝàáâãäåçèéêëìíîïðñòóôõöøùúûüýÿ'; | |||
var sortDst = 'cLYCaR231oAAAAAACEEEEIIIIDNOOOOOOUUUUYaaaaaaceeeeiiiidnoooooouuuuyy'; | |||
regex = new RegExp('(\\{\\{SORTIERUNG:[^}]*?)([' + sortSrc + '])', 'g'); | |||
while (t.match(regex)) t = t.replace(regex, function($0, $1, $2) { return $1 + sortDst.charAt(sortSrc.indexOf($2)); }); | |||
/* Sortierung soll einheitliche Groß/Kleinschreibung haben */ | |||
regex = /(\{\{SORTIERUNG:[^}]*?)(\b[a-z])/g; | |||
while (t.match(regex)) t = t.replace(regex, function($0, $1, $2) { return $1 + $2.toUpperCase(); }); | |||
regex = /(\{\{SORTIERUNG:[^}]*?)(\B[A-Z]+)/g; | |||
while (t.match(regex)) t = t.replace(regex, function($0, $1, $2) { return $1 + $2.toLowerCase(); }); | |||
t = this.cleanReferences(t); | |||
/* Weblinks auf Wikipedia-Sprachversionen in Wikilinks umwandeln */ | |||
t = this.cleanDefaultSort(t); | |||
var interWikiReplace = function($0, $1, $2, $3) | |||
t = this.cleanInternalLinks(t); | |||
{ | |||
t = this.cleanExternalLinks(t); | |||
return '[[:' + $1 + ':' + $2.replace(/_/g, ' ') + (typeof $3 === 'string' ? '|' + $3 : '') + ']]'; | |||
} | |||
t = t.replace(/\[+ *\w+:\/+([a-z-]+)\.wikipedia\.org\/wiki\/([^?\[\]|]*?) *\|+ *([^\[\]\n]+?) *\]+/gi, interWikiReplace); | |||
t = t.replace(/\[+ *\w+:\/+([a-z-]+)\.wikipedia\.org\/wiki\/([^?\[\]\s]*) +([^\[\]\n]+?) *\]+/gi, interWikiReplace); | |||
t = t.replace(/\[+ *\w+:\/+([a-z-]+)\.wikipedia\.org\/wiki\/([^?\[\]\s]*) *\]+/gi, interWikiReplace); | |||
/* Wikilinks mit unnötigem Präfix w:de: oder :de: vereinfachen */ | |||
t = t.replace(/\[\[ *w? *: *de *: *([^\[\]\n]*?) *\]\]/gi, '[[$1]]'); | |||
/* Sonstige Weblinks mit senkrechtem Strich reparieren */ | |||
t = t.replace(/\[ *(https?:\/\/[^\s|\]]*?) *\| *([^\[\]\n]*?) *\]/gi, '[$1 $2]'); | |||
/* Doppelte eckige Klammern um Weblinks entfernen */ | |||
t = t.replace(/\[+ *(https?:\/\/[^\[\]\n]*?) *\]+/gi, '[$1]'); | |||
/* Jede Kategorie in einer eigenen Zeile */ | |||
t = t.replace(/(\[\[Kategorie:[^\[\]\n]*\]\]) *(?!\n|$)/gi, '$1\n'); | |||
t = t.replace(/([^\n])(\[\[Kategorie:[^\[\]\n]*\]\])/gi, '$1\n$2'); | |||
{ | |||
t = t.replace(/(\[\[Kategorie:[^\[\]\n]*\]\]\n)(?!\[\[Kategorie:|\n|$)/gi, '$1\n'); | |||
try | |||
{ | |||
$2 = decodeURIComponent($2.replace(/\.([289A-E][\dA-F]|[357][B-F]|40|60)/g, '%$1')). | |||
replace(/[%\[\]{|}]/g, function($0) | |||
{ | |||
return '%' + $0.charCodeAt(0).toString(16).toUpperCase(); | |||
}); | |||
} | |||
catch (ex) { /* keine Veränderung im Fehlerfall */ } | |||
return ($1 + $2).replace(/_/g, ' ') + $3; | |||
}); | |||
/* Versehentliche Links um ISO-Daten entfernen */ | |||
/* [[Link|Die]]s wird zu [[Link|Dies]] weil besser lesbar */ | |||
t = t.replace(/\[+([12]\d{3}-[01]\d-[0-3]\d)\]+/g, '$1'); | |||
t = this.cleanDuplicateLinks(t); | |||
/* [[Link|Link]]s werden zu [[Link]]s weil kürzer und besser lesbar */ | |||
t = t.replace(/\[\[([^\|\]]+)\|\1([^\[\]\n]*)\]\]/g, '[[$1]]$2'); | |||
/* Jede Kategorie in einer eigenen Zeile */ | |||
t = t.replace(/(\[\[Kategorie:[^\[\]\n]*\]\]) *(?!\n|$)/gi, '$1\n'); | |||
t = t.replace(/([^\n])(\[\[Kategorie:[^\[\]\n]*\]\])/gi, '$1\n$2'); | |||
t = t.replace(/(\[\[Kategorie:[^\[\]\n]*\]\]\n)(?!\[\[Kategorie:|\n|$)/gi, '$1\n'); | |||
/* Unnötige Leerzeichen bei HTML-Attributen, wichtig vor den Anführungszeichen */ | |||
/* Versehentliche Links um ISO-Daten entfernen */ | |||
while (/[<|][^<>|]*\b *= +"/i.test(t)) | |||
t = t.replace(/\[+([12]\d{3}-[01]\d-[0-3]\d)\]+/g, '$1'); | |||
t = t.replace(/([<|][^<>|]*)\b *= +"/gi, '$1="'); | |||
/* Doppelte Jahreszahlen entlinken */ | |||
/* Anführungszeichen */ | |||
var m = t.match(/\[\[[12]\d{3}\]\]/g); | |||
t = t.replace(/([\s!#'(*+\/:;>[|-])"([^\s"“”„][^\n"“”„]*)"(?=[\s!'),.\/:;<?\]}-])/g, '$1„$2“'); | |||
if (m && m.length > 1) | |||
/* Auslassungspunkte */ | |||
{ | |||
t = t.replace(/([([])\.\.\.(?=[)\]])/g, '$1…'); | |||
var p, pMax = t.lastIndexOf('{{Personendaten'); | |||
/* Bis-Striche bei Jahreszahlenbereichen */ | |||
for (i = m.length; i--; ) | |||
t = t.replace(/([\s(|][12]\d{3}) *[–-] *(?=[12]\d{3}[^\d\]–-])/g, '$1–'); | |||
/* ISSNs aber ohne Bis-Striche */ | |||
t = t.replace(/(IS\wN\W*\d+)–(?=\d)/g, '$1-'); | |||
t = this.cleanISBNs(t); | |||
t = this.cleanDates(t); | |||
var re = /([\w'\)\]\xC0-\xFF]) +- +(?=[\w'\(\[\xC0-\xFF])/gi; | |||
var i = t.search(/^\[\[ *[a-z-]{2,12} *:/m); //12 wegen [[zh-classical:…]] | |||
if (i >= 0) t = t.slice(0, i).replace(re, '$1 – ') + t.slice(i); | |||
else t = t.replace(re, '$1 – '); | |||
t = this.cleanUnits(t); | |||
/* Paragraf, Abs. und Satz mit geschützten Leerzeichen */ | |||
t = t.replace(/§(?: *| )(\d\w* +A[bsatz.]+)(?: *| )(\d+ +S[atz.]+) *(?=\d)/gi, | |||
'§ $1 $2 '); | |||
t = t.replace(/§(?: *| )(\d\w* +A[bsatz.]+) *(?=\d)/gi, | |||
'§ $1 '); | |||
t = t.replace(/§ *(?=\d)/gi, | |||
'§ '); | |||
t = this.executeUserReplacements(t); | |||
t = this.cleanRedundantTemplateParameters(t); | |||
t = this.cleanInfoboxVerwaltungseinheitInDeutschland(t); | |||
t = this.restoreFilenames(t); | |||
t = this.restoreNowikis(t); | |||
return this.isChanged(oldValue, t) ? t : false; | |||
}, | |||
cleanHeadlines: function(t) | |||
{ | { | ||
/* Keine geschützten Leerzeichen in Überschriften */ | |||
p = t.lastIndexOf(m[i], pMax > 0 ? pMax : t.length); | |||
while (/^=.* .*=$/im.test(t)) | |||
/* Je ein Duplikat entfernen, wenn vorhanden */ | |||
t = t.replace(/^(=.*) (?=.*=$)/gim, '$1 '); | |||
if (p > t.indexOf(m[i])) | |||
/* Fettung zumindest kompletter Überschriften ist unerwünscht */ | |||
t = t.substr(0, p) + m[i].replace(/[\[\]]+/g, '') + | |||
t = t.replace(/^(=+) *'''([^']+)''' *(?==+$)/gm, '$1 $2 '); | |||
/* Repariert kaputte Überschriften, entfernt Doppelpunkte, setzt Leerzeichen */ | |||
} | |||
t = t.replace(/^(=+) *(.*[^\s=:]):? *\1$/gm, '$1 $2 $1'); | |||
} | |||
return t; | |||
}, | |||
cleanInternalLinks: function(t) | |||
{ | |||
var lang = mw.config.get('wgContentLanguage'); | |||
var ns = mw.config.get('wgFormattedNamespaces')[-1]; | |||
/* Permanente Weblinks in Spezialseiten-Syntax umwandeln */ | |||
/* Unnötige Leerzeichen bei HTML-Attributen, wichtig vor den Anführungszeichen */ | |||
var permaLinkReplace = function($0, $1, $2, $3) | |||
while (t.match(/[<|][^|>]*\b *= +"/im)) | |||
{ | |||
t = t.replace(/([<|][^|>]*)\b *= +"/gim, '$1="'); | |||
return '[[:' + $1 + ':' + | |||
/* Anführungszeichen */ | |||
($1 == lang ? ns : 'Special') + ':' + | |||
t = t.replace(/([^\w={])["]([^\n"“”]+)["]([^\w|])/gm, '$1„$2“$3'); | |||
($1 == 'de' ? 'Permanenter Link' : 'PermanentLink') + '/' + $2 + | |||
/* Bis-Striche bei Jahreszahlen */ | |||
(typeof $3 == 'string' ? '|' + $3 : '') + ']]'; | |||
t = t.replace(/([\s(|][12]\d{3}) *[–-]{1,2} *([12]\d{3}[^\d\]–-])/g, '$1–$2'); | |||
} | |||
/* ISSNs aber ohne Bis-Striche */ | |||
/* Weblinks auf Sprachversionen (auch auf die eigene) in Wikilinks umwandeln */ | |||
t = t.replace(/(IS\wN\W*\d+)–(\d)/g, '$1-$2'); | |||
var interWikiReplace = function($0, $1, $2, $3) | |||
/* ISBNs mit Bindestrichen gliedern, aktuell für englische (0 und 1) und deutsche (3) Bücher */ | |||
{ | |||
t = t.replace(/\bISBN(?:-?1[03])?:?\s*((?:9-?7-?[89]-?)?0)([\d-]{8,}[\dX]\b)/gi, function($0, $1, $2) { | |||
return ' | return '[[:' + $1 + ':' + $2.replace(/_/g, ' ') + | ||
(typeof $3 == 'string' ? '|' + $3 : '') + ']]'; | |||
replace(/^([01]\d)(\d{6})/, '$1-$2-'). | |||
} | |||
replace(/^([2-6]\d\d)(\d{5})/, '$1-$2-'). | |||
/* Schreibweise [[Weblink#Anker mit Leerzeichen|Beschriftung]] reparieren */ | |||
replace(/^(7\d{3}|8[0-4]\d\d)(\d{4})/, '$1-$2-'). | |||
t = t.replace(/\[+ *(?:https?:)?\/\/([a-z-]+)\.wikipedia\.org\/w\/[\w.]*\?(?:title=[^&|\[\]\s]*&)?oldid=([^?|\[\]\n]+?) *\|+ *([^|\[\]\n]+?) *\]+/gi, permaLinkReplace); | |||
replace(/^(8[5-9]\d{3})(\d{3})/, '$1-$2-'). | |||
t = t.replace(/\[+ *(?:https?:)?\/\/([a-z-]+)\.wikipedia\.org\/wiki\/([^?|\[\]\n]*?) *\|+ *([^|\[\]\n]+?) *\]+/gi, interWikiReplace); | |||
replace(/^(9[0-4]\d{4})(\d\d)/, '$1-$2-'). | |||
/* Schreibweise [Weblink#Anker Beschriftung] umwandeln */ | |||
replace(/^(9[5-9]\d{5})(\d)/, '$1-$2-'); | |||
t = t.replace(/\[+ *(?:https?:)?\/\/([a-z-]+)\.wikipedia\.org\/w\/[\w.]*\?(?:title=[^&|\[\]\s]*&)?oldid=([^?|\[\]\s]+) +([^|\[\]\n]+?) *\]+/gi, permaLinkReplace); | |||
}); | |||
t = t.replace(/\[+ *(?:https?:)?\/\/([a-z-]+)\.wikipedia\.org\/wiki\/([^?|\[\]\s]*) +([^|\[\]\n]+?) *\]+/gi, interWikiReplace); | |||
/* Schreibweise [Weblink#Anker] umwandeln */ | |||
return 'ISBN ' + $1.replace(/^9\D*7\D*([89])\D*1/, '97$1-1') + '-' + $2.replace(/[^\dX]/gi, ''). | |||
t = t.replace(/\[+ *(?:https?:)?\/\/([a-z-]+)\.wikipedia\.org\/w\/[\w.]*\?(?:title=[^&|\[\]\s]*&)?oldid=([^?|\[\]\s]+) *\]+/gi, permaLinkReplace); | |||
replace(/^(0\d)(\d{6})/, '$1-$2-'). | |||
replace(/ | t = t.replace(/\[+ *(?:https?:)?\/\/([a-z-]+)\.wikipedia\.org\/wiki\/([^?|\[\]\s]*) *\]+/gi, interWikiReplace); | ||
/* Verbliebene projektinterne Weblinks ohne eckige Klammern ebenfalls umwandeln */ | |||
replace(/^(4\d{3}|5[0-4]\d\d)(\d{4})/, '$1-$2-'). | |||
replace(/ | t = t.replace(/(?:https?:)?\/\/([a-z-]+)\.wikipedia\.org\/w\/[\w.]*\?(?:title=[^&|\[\]\s]*&)?oldid=([^?|\[\]\s]*[^?|\[\]\s!"),.:;<>?\\])/gi, permaLinkReplace); | ||
replace(/ | t = t.replace(/(?:https?:)?\/\/([a-z-]+)\.wikipedia\.org\/wiki\/([^?|\[\]\s]*[^?|\[\]\s!"),.:;<>?\\])/gi, interWikiReplace); | ||
replace(/^(999\d{4})(\d)/, '$1-$2-'); | |||
}); | |||
t = t.replace(/\bISBN(?:-?1[03])?:?\s*((?:9-?7-?[89]-?)?3)([\d-]{8,}[\dX]\b)/gi, function($0, $1, $2) { | |||
return 'ISBN ' + $1.replace(/^9\D*7\D*([89])\D*3/, '97$1-3') + '-' + $2.replace(/[^\dX]/gi, ''). | |||
replace(/^(0[0-24-9]|1\d)(\d{6})/, '$1-$2-'). | |||
replace(/^(03[0-3]|[2-6]\d\d)(\d{5})/, '$1-$2-'). | |||
replace(/^(03[4-6]\d|7\d{3}|8[0-4]\d\d)(\d{4})/, '$1-$2-'). | |||
replace(/^(03[7-9]\d\d|8[5-9]\d{3}|95[4-9]\d\d|9[69]\d{3})(\d{3})/, '$1-$2-'). | |||
replace(/^(9[0-4]\d{4})(\d\d)/, '$1-$2-'). | |||
replace(/^(95[0-3]\d{4}|9[78]\d{5})(\d)/, '$1-$2-'); | |||
}); | |||
/* Wikilinks mit unnötigem Präfix „:w:de:“, „w:de:“ oder „:de:“ vereinfachen */ | |||
var months = ['Januar', 'Februar', 'März', 'April', 'Mai', 'Juni', | |||
t = t.replace(new RegExp('\\[\\[ *(?::? *w *)?: *' + lang + ' *: *((Kategorie *:|Category *:)?[^\\[\\]\\n]*\\S) *\\]\\]', 'gi'), function($0, $1, $2) | |||
'Juli', 'August', 'September', 'Oktober', 'November', 'Dezember']; | |||
{ | |||
/* Missverständliches deutsches Datumsformat durch Langform ersetzen */ | |||
return '[[' + ($2 ? ':' : '') + $1 + ']]'; | |||
t = t.replace(/([\s(])(3[01]|[12]\d|0?[1-9])\. *(1[012]|0?[1-9])\. *([12]\d{3}[^\d.–-])/g, | |||
}); | |||
function($0, $1, $2, $3, $4) | |||
/* Anker in internen Links dekodieren */ | |||
t = t.replace(/(\[\[[^\n#\[\]{|]*#)([^\n\[\]|]+)(?=\|?[^\n\[\]|]*\]\])/gi, function($0, $1, $2) | |||
{ | |||
try | |||
{ | |||
$2 = decodeURIComponent($2.replace(/\.([289A-E][\dA-F]|[357][B-F]|40|60)/g, '%$1')). | |||
replace(/[%\[\]{|}]/g, function($0) | |||
{ | |||
return '%' + $0.charCodeAt(0).toString(16).toUpperCase(); | |||
}); | |||
} | |||
catch (ex) { /* keine Veränderung im Fehlerfall */ } | |||
return ($1 + $2).replace(/_/g, ' '); | |||
}); | |||
/* [[Link|Die]]s wird zu [[Link|Dies]] weil besser lesbar */ | |||
t = t.replace(/(\[\[[^|\[\]\n]+\|[^|\[\]\n]+)\]\]([a-zäöüß]+)/gi, '$1$2]]'); | |||
/* [[Link|Link]]s werden zu [[Link]]s weil kürzer und besser lesbar */ | |||
t = t.replace(/\[\[([^|\[\]\n]+)\|\1([^|\[\]\n]*)\]\]/g, '[[$1]]$2'); | |||
return t; | |||
}, | |||
cleanExternalLinks: function(t) | |||
{ | { | ||
/* Doppelte eckige Klammern um Weblinks vereinfachen */ | |||
return $1 + 1 * $2 + '. ' + months[$3 - 1] + ' ' + $4; | |||
t = t.replace(/\[+ *(https?:\/\/[^\n[\]]*?) *\]+/gi, '[$1]'); | |||
}); | |||
/* Weblinks mit senkrechtem Strich reparieren */ | |||
/* In "1850–14. Januar" immer "bis" einsetzen */ | |||
t = t.replace(/(\[https?:\/\/[^\s[\]|]*?) *\| *(?=[^\n=[\]|]+\])/gi, '$1 '); | |||
/* Schrägstriche am Ende einfacher Domains ergänzen */ | |||
function($0, $1, $2, $3) | |||
t = t.replace(/(\[https?:\/\/(?:[\w-]+\.)+\w+) +/gi, '$1/ '); | |||
/* Domains klein schreiben, egal ob beschriftet oder nicht */ | |||
t = t.replace(/\bhttps?:\/\/[a-z.-]*[A-Z][\w.-]*/g, function($0) { return $0.toLowerCase(); }); | |||
/* Verbliebene projektinterne Weblinks protokollrelativ machen */ | |||
t = t.replace(/\[ *https?:\/+(?=[a-z-]+\.wikipedia\.org\b)/gi, '[//'); | |||
return t; | |||
}, | |||
cleanDuplicateLinks: function(t) | |||
{ | { | ||
/* Daten und Jahreszahlen in Personendaten nie verlinken */ | |||
for (i = months.length; i--; ) | |||
var re = /\{\{\s*[Pp]ersonendaten\s*\|\s*([^{}]*?)\s*\|\s*((?:GEBURTS|STERBE)DATUM\s*=[^\n=[\]{|}]*)\[+([^\n=[\]{|}]+)\]+/g; | |||
if ($3 === months[i]) return $1 + ' bis ' + $2 + '. ' + $3; | |||
while (re.test(t)) t = t.replace(re, '{{Personendaten\n|$1\n|$2$3'); | |||
return $0; | |||
}); | |||
/* 1980–90 immer als 1980–1990 */ | |||
t = t.replace(/([\s(|][12]\d{3}) *- *((?:1[3-9]|[2-9]\d)\b[^.=–-])/g, '$1–$2'); /* Bindestrich gefunden */ | |||
t = t.replace(/\b(([12]\d)\d{2}) *– *(\d{2}\b[^=–-])/g, '$1–$2$3'); /* Bis-Strich gefunden */ | |||
/* Gedankenstriche in <math> vermeiden */ | |||
t = t.replace(/(['\)a-z\]\xC0-\xFF]) +- +(['\(a-z\[\xC0-\xFF])/gi, '$1 – $2'); | |||
/* Doppelte Jahreszahlen entlinken */ | |||
/* Maßeinheiten immer mit Leerzeichen */ | |||
var re = /\[\[([12]\d{3})\]\]/g, found = [], a = []; | |||
t = t.replace(/(\b\d+) ?([mck]?m|kg|[KMG]iB|[kMG](B|Hz)|KB)\b(?![\xC0-\xFF-])/g, '$1 $2'); | |||
/* Jeweils ersten Fund eines Jahres merken, danach entlinken */ | |||
/* Prozentwerte erhalten seit Mitte 2007 automatisch ein geschütztes Leerzeichen */ | |||
while (m = re.exec(t)) found[m[1]] ? a.push(m) : found[m[1]] = true; | |||
t = t.replace(/([^=]\b\d+)( )?(%[^\w"%;])/gi, '$1 $3'); | |||
for (var i = a.length; i--; ) | |||
/* Paragraf, Abs. und Satz mit geschützten Leerzeichen */ | |||
t = t.slice(0, a[i].index) + a[i][1] + t.slice(a[i].index + a[i][0].length); | |||
t = t.replace(/§( | )+(\d+\w* +Abs\.)( | )+(\d+ +Satz) +(\d+)/gi, | |||
return t; | |||
'§ $2 $4 $5'); | |||
}, | |||
t = t.replace(/§( | )+(\d+\w* +Abs\.) +(\d+)/gi, | |||
cleanDates: function(t) | |||
'§ $2 $3'); | |||
{ | |||
t = t.replace(/§ +(\d+)/gi, | |||
var months = ['Januar', 'Februar', 'März', 'April', 'Mai', 'Juni', | |||
'§ $1'); | |||
'Juli', 'August', 'September', 'Oktober', 'November', 'Dezember']; | |||
/* Missverständliches deutsches Datumsformat durch Langform ersetzen */ | |||
t = t.replace(/([\s(])(3[01]|[12]\d|0?[1-9])\. *(1[012]|0?[1-9])\. *(?=[12]\d{3}[^\d.–-])/g, | |||
function($0, $1, $2, $3, $4) | |||
{ | |||
return $1 + 1 * $2 + '. ' + months[$3 - 1] + ' '; | |||
}); | |||
/* In „1850–14. Januar“ immer „bis“ einsetzen */ | |||
t = t.replace(/(\s[12]\d{3}) *[–-] *(3[01]|[12]\d|0?[1-9])\. *([A-Z][a-zä]+)/g, | |||
function($0, $1, $2, $3) | |||
{ | |||
for (var i = months.length; i--; ) | |||
if ($3 == months[i]) return $1 + ' bis ' + $2 + '. ' + $3; | |||
return $0; | |||
}); | |||
/* „1980–90“ immer als „1980–1990“ */ | |||
t = t.replace(/([\s(|][12]\d{3}) *- *(?=(?:1[3-9]|[2-9]\d)\b[^.=–-])/g, '$1–'); /* Bindestrich gefunden */ | |||
t = t.replace(/\b(([12]\d)\d\d) *– *(?=\d\d\b[^=–-])/g, '$1–$2'); /* Bis-Strich gefunden */ | |||
return t; | |||
}, | |||
cleanUnits: function(t) | |||
{ | |||
/* Maßeinheiten immer mit Leerzeichen */ | |||
t = t.replace(/(\b\d+) ?(?=(?:k[Bgm]|Ki?B|[MGT](i?B|Hz)|[mc]m|k?Hz|EUR|Euro|CHF|US[D$]|JPY|[gm€¥])[\s!'),.\/:;?²³“])/g, | |||
'$1 '); | |||
/* Prozentwerte erhalten seit Mitte 2007 automatisch ein geschütztes Leerzeichen */ | |||
t = t.replace(/([\s'(*+,\/:;>–„−-]\d+)(?: )?(?=%[\s!'),.\/:<?\]|“])/gi, '$1 '); | |||
return t; | |||
}, | |||
cleanISBNs: function(t) | |||
{ | |||
/* ISBNs mit Bindestrichen gliedern, Gruppennummern 0 und 1 für englischsprachige Bücher */ | |||
t = t.replace(/\bISBN(?:-?1[03])?:?\s*((?:9-?7-?[89]-?)?0)([\d-]{8,}[\dX]\b)/gi, function($0, $1, $2) { | |||
return 'ISBN ' + $1.replace(/^9\D*7\D*([89])\D*0/, '97$1-0') + '-' + $2.replace(/[^\dX]/gi, ''). | |||
replace(/^([01]\d)(\d{6})/, '$1-$2-'). | |||
replace(/^([2-6]\d\d)(\d{5})/, '$1-$2-'). | |||
replace(/^(7\d{3}|8[0-4]\d\d)(\d{4})/, '$1-$2-'). | |||
replace(/^(8[5-9]\d{3})(\d{3})/, '$1-$2-'). | |||
replace(/^(9[0-4]\d{4})(\d\d)/, '$1-$2-'). | |||
replace(/^(9[5-9]\d{5})(\d)/, '$1-$2-'); | |||
}); | |||
t = t.replace(/\bISBN(?:-?1[03])?:?\s*((?:9-?7-?[89]-?)?1)([\d-]{8,}[\dX]\b)/gi, function($0, $1, $2) { | |||
return 'ISBN ' + $1.replace(/^9\D*7\D*([89])\D*1/, '97$1-1') + '-' + $2.replace(/[^\dX]/gi, ''). | |||
replace(/^(0\d)(\d{6})/, '$1-$2-'). | |||
replace(/^([1-3]\d\d)(\d{5})/, '$1-$2-'). | |||
replace(/^(4\d{3}|5[0-4]\d\d)(\d{4})/, '$1-$2-'). | |||
replace(/^(5[5-9]\d{3}|[67]\d{4}|8[0-5]\d{3}|86[0-8]\d\d|869[0-7]\d)(\d{3})/, '$1-$2-'). | |||
replace(/^(869[89]\d\d|8[7-9]\d{4}|9[0-8]\d{4}|99[0-8]\d{3})(\d\d)/, '$1-$2-'). | |||
replace(/^(999\d{4})(\d)/, '$1-$2-'); | |||
}); | |||
/* Gruppennummer 3 für deutschsprachige Bücher */ | |||
t = t.replace(/\bISBN(?:-?1[03])?:?\s*((?:9-?7-?[89]-?)?3)([\d-]{8,}[\dX]\b)/gi, function($0, $1, $2) { | |||
return 'ISBN ' + $1.replace(/^9\D*7\D*([89])\D*3/, '97$1-3') + '-' + $2.replace(/[^\dX]/gi, ''). | |||
replace(/^(0[0-24-9]|1\d)(\d{6})/, '$1-$2-'). | |||
replace(/^(03[0-3]|[2-6]\d\d)(\d{5})/, '$1-$2-'). | |||
replace(/^(03[4-6]\d|7\d{3}|8[0-4]\d\d)(\d{4})/, '$1-$2-'). | |||
replace(/^(03[7-9]\d\d|8[5-9]\d{3}|95[4-9]\d\d|9[69]\d{3})(\d{3})/, '$1-$2-'). | |||
replace(/^(9[0-4]\d{4})(\d\d)/, '$1-$2-'). | |||
replace(/^(95[0-3]\d{4}|9[78]\d{5})(\d)/, '$1-$2-'); | |||
}); | |||
return t; | |||
}, | |||
cleanReferences: function(t) | |||
{ | |||
t = t.replace(/<\s*references\s*(\s\b[^<\/>]*?)?\s*\/\s*>/gi, '<references$1 />'); | |||
t = t.replace(/<\s*ref\s*(\s\b[^<\/>]*?)\s*\/\s*>/gi, '<ref$1 />'); | |||
t = t.replace(/<\s*ref\s*(\s\b[^<\/>]*?)?\s*>\s*/gi, '<ref$1>'); | |||
t = t.replace(/\s*<\s*\/\s*ref\s*>/gi, '</ref>'); | |||
/* Lücken zwischen Satzende und <ref> oder zwei <ref> entfernen */ | |||
t = t.replace(/([!,.;?]|<ref\b[^<\/>]*(?:\/|>[^<>]*<\/ref)>) *[ \n](?=<ref[ >])/gi, '$1'); | |||
/* Zwei gleiche Satzzeichen vor und nach einem <ref> auf eins kürzen */ | |||
t = t.replace(/([!,.;?])(<ref\b[^<\/>]*(?:\/|>[^<>]*<\/ref)>)\1/gi, '$1$2'); | |||
return t; | |||
}, | |||
cleanDefaultSort: function(t) | |||
{ | |||
if (this.locale) t = t.replace(/\{\{\s*(SORTIERUNG|DEFAULTSORT):\s*/gi, '{{SORTIERUNG:'); | |||
/* Sortierung soll nie Umlaute enthalten */ | |||
var autoFormatReplacements = window.autoFormatReplacements || [], from; | |||
t = t.replace(/(\{\{(?:SORTIERUNG|DEFAULTSORT):[^{}\n]*?)ß/, '$1ss'); | |||
for (from in autoFormatReplacements) | |||
var sortSrc = '¢£¥©ª®²³¹ºÀÁÂÃÄÅÇÈÉÊËÌÍÎÏÐÑÒÓÔÕÖØÙÚÛÜÝàáâãäåçèéêëìíîïðñòóôõöøùúûüýÿ'; | |||
{ | |||
var sortDst = 'cLYCaR231oAAAAAACEEEEIIIIDNOOOOOOUUUUYaaaaaaceeeeiiiidnoooooouuuuyy'; | |||
var to = autoFormatReplacements[from]; | |||
var re = new RegExp('(\\{\\{(?:SORTIERUNG|DEFAULTSORT):[^{}\n]*?)([' + sortSrc + '])', 'g'); | |||
from = from.replace(/([.+*?(){}<>|])/g, '\\$1'); //Regex-Zeichen maskieren | |||
while (re.test(t)) | |||
from = from.replace(/^(\w)/, '\\b$1').replace(/(\w)$/, '$1\\b'); //Wortgrenzen beachten | |||
t = t.replace(re, function($0, $1, $2) { return $1 + sortDst.charAt(sortSrc.indexOf($2)); }); | |||
for (i = 1; from.match(/\.\\d/); i++) | |||
/* Sortierung soll einheitliche Groß/Kleinschreibung haben */ | |||
re = /(\{\{(?:SORTIERUNG|DEFAULTSORT):[^{}\n]*?)(\b[a-z])/g; | |||
while (re.test(t)) t = t.replace(re, function($0, $1, $2) { return $1 + $2.toUpperCase(); }); | |||
re = /(\{\{(?:SORTIERUNG|DEFAULTSORT):[^{}\n]*?)(\B[A-Z]+)/g; | |||
while (re.test(t)) t = t.replace(re, function($0, $1, $2) { return $1 + $2.toLowerCase(); }); | |||
return t; | |||
}, | |||
cleanTemplates: function(t) | |||
{ | { | ||
t = t.replace(/\{\{\s*(?:Vorlage|Template)\s*:\s*/gi, '{{'); | |||
/* Unterstriche aus allen Vorlagennamen entfernen */ | |||
to = to.replace(/\\d/, '$' + i); | |||
while (/(?:^|[^{])\{\{[ 0-9a-z\xC0-\xFF-]+_/i.test(t)) | |||
} | |||
t = t.replace(/((?:^|[^{])\{\{[ 0-9a-z\xC0-\xFF-]+)_/gi, '$1 '); | |||
t = t.replace(new RegExp(from, 'g'), to); | |||
} | |||
/* {{Commons|Kategorie:…}} durch {{Commonscat|…}} austauschen */ | |||
/* Keine geschützten Leerzeichen in Überschriften */ | |||
t = t.replace(/\{\{\s*Commons(?:cat)?\s*\|\s*[KC]ategor[iy]e?\s*:\s*/gi, '{{Commonscat|'); | |||
while (t.match(/^=.* .*=$/m)) | |||
t = t.replace(/^(=.*) (.*=)$/gm, '$1 $2'); | |||
/* Projektweit einheitliche Schreibweisen für häufig verwendete Vorlagen */ | |||
t = t.replace(/\{\{\s*Normdaten\s*\|\s*PND\s*=\s*/g, '{{Normdaten|TYP=p|GND='); | |||
t = t.replace(/\{\{\s*Belege fehlen[ |]*(?=\s*(?:\||\}\}))/gi, '{{Belege fehlen'); | |||
t = t.replace(/\{\{\s*Commons(cat)?[\s|]*\}\}/gi, '{{Commons$1}}'); | |||
t = t.replace(/\{\{\s*Commons(cat)?\s*\|\s*/gi, '{{Commons$1|'); | |||
t = t.replace(/\{\{\s*Dieser Artikel\s*\|\s*/gi, '{{Dieser Artikel|'); | |||
t = t.replace(/\{\{\s*Dts(x?)\s*\|\s*/gi, '{{dts$1|'); | |||
t = t.replace(/\{\{\s*(?:Hauptartikel|Hauptseite|Main)\s*\|\s*/gi, '{{Hauptartikel|'); | |||
t = t.replace(/\{\{\s*(?:Internetquelle|Weblink)(?=\s*\|)/gi, '{{Internetquelle'); | |||
t = t.replace(/\{\{\s*Lang\s*\|\s*/gi, '{{lang|'); | |||
t = t.replace(/\{\{\s*Nts\s*\|\s*/gi, '{{nts|'); | |||
t = t.replace(/\{\{\s*Rotten ?Tomatoes\s*\|\s*/gi, '{{Rotten Tomatoes|'); | |||
t = t.replace(/\{\{\s*S(?:iehe auch|ee also)\s*\|\s*/gi, '{{Siehe auch|'); | |||
t = t.replace(/\{\{\s*Sort(Date|Key|KeyName)\s*\|\s*/gi, '{{Sort$1|'); | |||
t = t.replace(/\{\{\s*Wik(ibooks|inews|iquote|isource|tionary)[\s|]*(?=\||\}\})/gi, '{{Wik$1'); | |||
t = t.replace(/\{\{\s*Normdaten\s*\|\s*PND\s*=\s*/g, '{{Normdaten|TYP=p|GND='); | |||
var redundantTemplateParameters = window.redundantTemplateParameters || [ | |||
'Infobox Arcade|Titel', | |||
'Infobox Band|Name', | |||
'Infobox Berg|NAME', | |||
'Infobox Burg|Name', | |||
'Infobox Chemikalie|Name', | |||
'Infobox Computer- und Videospiel|Titel', | |||
'Infobox Eishockeyspieler|Name', | |||
'Infobox Flughafen|Name', | |||
'Infobox Flugzeug|Name', | |||
'Infobox Fluss|NAME', | |||
'Infobox Fußballspieler|kurzname', | |||
'Infobox Gemeinde in Österreich|Name', | |||
'Infobox Gemeinde in Deutschland|Name', | |||
'Infobox Gemeinde in Italien|nomeComune', | |||
'Infobox Gemeindeverband in Deutschland|Name', | |||
'Infobox Insel|NAME', | |||
'Infobox Landkreis|Name', | |||
'Infobox Musikalbum|Titel', | |||
'Infobox Ort in den Vereinigten Staaten|Name', | |||
'Infobox Ort in der Schweiz|NAME_ORT', | |||
'Infobox Ort in Polen|Ort', | |||
'Infobox Ort in Tschechien|Ort', | |||
'Infobox Ortsteil einer Gemeinde|Ortsteil', | |||
'Infobox Ortsteil einer Gemeinde in Deutschland|Ortsteil', | |||
'Infobox PKW-Modell|Modell', | |||
'Infobox Publikation|titel', | |||
'Infobox Schiff|Name', | |||
'Infobox See|NAME', | |||
'Infobox Software|Name', | |||
'Infobox Stadion|Name', | |||
'Infobox Unternehmen|Name']; | |||
var parameter = '\\s*=\\s*(' + | |||
wgTitle.replace(/([.\\+*?\[\](){}|])/g, '\\$1').replace(/\s+/g, '\\s+') + | |||
'|\\{+\\w*\\}+)?\\s*([}|])'; | |||
for (i = redundantTemplateParameters.length; i--; ) | |||
{ | |||
t = t.replace(new RegExp('(\\{\\{\\s*' + | |||
redundantTemplateParameters[i].replace(/[\s_]+/g, '[\\s_]+').replace('|', '\\s*(?:\\|[^|}]*)*)\\|\\s*') + | |||
parameter, 'g'), '$1$3'); | |||
} | |||
t = t.replace(/\{\{\s*WBA\s*\|\s*/gi, '{{Waybackarchiv|'); | |||
/* Spezielle Infobox-Ersetzungen */ | |||
t = t.replace(/\[ *\{\{\s*Wayback\w*\s*\|\s*url\s*=\s*[^\s\d{|}]*(\d{1,14})\/(\w+:[^\s{|}]*)[^{}]*\}\}\s+([^[\]|]*)\]/gi, | |||
if (t.match(/Infobox[ _](Gemeinde|Ort)[ _]in[ _]Deutschland/i)) | |||
'{{Webarchiv | url=$2 | wayback=$1 | text=$3}}'); | |||
{ | |||
return t; | |||
t = t.replace(/Infobox[ _](Gemeinde|Ort)[ _]in[ _]Deutschland([^}]*}})[\s\r\n]*/i, 'Infobox Gemeinde in Deutschland$2\n\n'); | |||
}, | |||
t = t.replace(/\|\s*lat_deg\s*=\s*(\S*)(\s*\|\s*lat_min\s*=\s*(\S*)(\s*\|\s*lat_sec\s*=\s*(\S*))?)?/ig, | |||
cleanRedundantTemplateParameters: function(t) | |||
'|Breitengrad = $1/$3/$5'); | |||
t = t.replace(/\|\s*lon_deg\s*=\s*(\S*)(\s*\|\s*lon_min\s*=\s*(\S*)(\s*\|\s*lon_sec\s*=\s*(\S*))?)?/ig, | |||
'|Längengrad = $1/$3/$5'); | |||
t = t.replace(/(\/0*)+[\r\n]+/g, '\n'); | |||
t = t.replace(/(\[http:\/\/[^\/\s\]]*)\s+/ig, '$1/ '); | |||
if (t.match(/Art\s*=\s*(Amt|(Samt|Verbands)gemeinde|Verwaltungsgemeinschaft|(Gemeinde)?verwaltungsverband)/i)) | |||
{ | { | ||
var parameters = window.redundantTemplateParameters || [ | |||
t = t.replace(/[ _]Gemeinde[ _]in[ _]Deutschland/i, ' Gemeindeverband in Deutschland'); | |||
'Infobox Arcade|Titel', | |||
t = t.replace(/\|\s*(PLZ(-alt)?|Vorwahl|Kfz)\s*=.*[\r\n]+/ig, ''); | |||
'Infobox Band|Name', | |||
t = t.replace(/\|\s*Bürgermeister( {0,5}|titel)?\s*=\s*/ig, '|Organwalter$1 = '); | |||
'Infobox Berg|NAME', | |||
if (!t.match(/\|\s*Gliederung\s*=/)) | |||
'Infobox Burg|Name', | |||
t = t.replace(/(\|\s*Adresse\s*=)/i, '|Gliederung = \n$1'); | |||
'Infobox Chemikalie|Name', | |||
t = t.replace(/\|\s*Gliederung\s*=\s*(\d*).*/i, '|Gliederung = $1'); | |||
'Infobox Computer- und Videospiel|Titel', | |||
e['wpSummary'].value += 'Einsatz der neuen [[Vorlage:Infobox Gemeindeverband in Deutschland]]'; | |||
'Infobox Eishockeyspieler|Name', | |||
e['wpMinoredit'].checked = true; | |||
'Infobox Flughafen|Name', | |||
'Infobox Flugzeug|Name', | |||
'Infobox Fluss|NAME', | |||
'Infobox Fußballspieler|kurzname', | |||
'Infobox Gemeinde in Österreich|Name', | |||
'Infobox Gemeinde in Deutschland|Name', | |||
'Infobox Gemeinde in Italien|nomeComune', | |||
'Infobox Gemeindeverband in Deutschland|Name', | |||
'Infobox Insel|NAME', | |||
'Infobox Landkreis|Name', | |||
'Infobox Musikalbum|Titel', | |||
'Infobox Ort in den Vereinigten Staaten|Name', | |||
'Infobox Ort in der Schweiz|NAME_ORT', | |||
'Infobox Ort in Polen|Ort', | |||
'Infobox Ort in Tschechien|Ort', | |||
'Infobox Ortsteil einer Gemeinde|Ortsteil', | |||
'Infobox Ortsteil einer Gemeinde in Deutschland|Ortsteil', | |||
'Infobox PKW-Modell|Modell', | |||
'Infobox Publikation|titel', | |||
'Infobox Schiff|Name', | |||
'Infobox See|NAME', | |||
'Infobox Software|Name', | |||
'Infobox Stadion|Name', | |||
'Infobox Unternehmen|Name']; | |||
var title = '\\s*=\\s*(?:' + | |||
mw.config.get('wgTitle').replace(/([.\\+*?\[\](){}|])/g, '\\$1').replace(/\s+/g, '\\s+') + | |||
'|\\{+\\w*\\}+)?\\s*(?=[}|])'; | |||
for (var i = parameters.length; i--; ) | |||
{ | |||
t = t.replace(new RegExp('(\\{\\{\\s*' + | |||
parameters[i].replace(/[\s_]+/g, '[\\s_]+').replace('|', '\\s*(?:\\|[^{}]*)?)\\|\\s*') + | |||
title, 'g'), '$1'); | |||
} | |||
return t; | |||
}, | |||
cleanInfoboxVerwaltungseinheitInDeutschland: function(t) | |||
{ | |||
t = t.replace(/\{\{\s*Infobox Ort in Deutschland/gi, '{{Infobox Gemeinde in Deutschland'); | |||
if (!/\{\{\s*Infobox Gemeinde in Deutschland/i.test(t)) return t; | |||
t = t.replace(/\|\s*lat_deg\s*=\s*([^\s{|}]*)(?:\s*\|\s*lat_min\s*=\s*([^\s{|}]*)(?:\s*\|\s*lat_sec\s*=\s*([^\s{|}]*))?)?/ig, | |||
'|Breitengrad = $1/$2/$3'); | |||
t = t.replace(/\|\s*lon_deg\s*=\s*([^\s{|}]*)(?:\s*\|\s*lon_min\s*=\s*([^\s{|}]*)(?:\s*\|\s*lon_sec\s*=\s*([^\s{|}]*))?)?/ig, | |||
'|Längengrad = $1/$2/$3'); | |||
t = t.replace(/(\|\s*(?:Breit|Läng)engrad\s*=\s*[\d.\/]*?)(?:\/0*)+(?=[\n|}])/g, '$1'); | |||
if (/Art\s*=\s*(?:Amt|(?:Samt|Verbands)gemeinde|Verwaltungsgemeinschaft|(?:Gemeinde)?verwaltungsverband)/i.test(t)) | |||
{ | |||
t = t.replace(/\{\{\s*Infobox Gemeinde in Deutschland/gi, '{{Infobox Gemeindeverband in Deutschland'); | |||
t = t.replace(/\|\s*(?:PLZ(?:-alt)?|Vorwahl|Kfz)\s*=[^{|}]*/ig, ''); | |||
t = t.replace(/\|\s*Bürgermeister( {0,5}|titel)?\s*=\s*/ig, '|Organwalter$1 = '); | |||
if (!/\|\s*Gliederung\s*=/.test(t)) | |||
t = t.replace(/(\|\s*Adresse\s*=)/i, '|Gliederung = \n$1'); | |||
t = t.replace(/\|\s*Gliederung\s*=\s*(\d*).*/i, '|Gliederung = $1'); | |||
} | |||
if (!/\|\s*Partei\s*=/.test(t)) | |||
t = t.replace(/((?:\|\s*(?:Bürgermeister|Organwalter)[^|}]*)+)/i, '$1|Partei =\n'); | |||
t = t.replace(/\n+\s*\|\s*([ \wäöüß-]{17})\s*=/ig, '\n|$1 ='); | |||
return t; | |||
}, | |||
executeUserReplacements: function(t) | |||
{ | |||
var from, replacements = window.autoFormatReplacements || {}; | |||
for (from in replacements) | |||
{ | |||
var to = replacements[from]; | |||
/* Wenn die Ersetzungen kein assoziatives Objekt sondern ein 2-dimensionales Array sind */ | |||
if (typeof to == 'object' && to.length > 1) from = to[0], to = to[1]; | |||
/* Wenn das Suchmuster schon ein regulärer Ausdruck ist */ | |||
if (typeof from == 'object') | |||
{ | |||
t = t.replace(from, to); | |||
continue; | |||
} | |||
/* Leere Suchmuster sicherheitshalber nicht zulassen */ | |||
if (/^\s*$/.test(from) || typeof to != 'string') continue; | |||
/* Die meisten Regex-Zeichen maskieren, außer Zeichenklassen */ | |||
from = from.replace(/(?=[.+*?(){}<>|])/g, '\\'); | |||
/* Wortgrenzen beachten */ | |||
from = from.replace(/^(?=\w|\\d)/, '\\b').replace(/(\w)$/, '$1\\b'); | |||
var a = []; | |||
for (var m, re = /\\[dw]/g, i = 1; m = re.exec(from); a.push(m)) | |||
to = to.replace(m[0], '$' + i++); | |||
for (var i = a.length; i--; ) | |||
from = from.slice(0, a[i].index) + (a[i][0] == '\\d' ? '(\\d+)' : | |||
'([A-Za-z\xB5\xC0-\xD6\xD8-\xF6\xF8-\u02AF]+)') + | |||
from.slice(a[i].index + 2); | |||
/* Look-ahead verwenden, wenn ein Platzhalter in Suchmuster und Ersatz am Ende steht */ | |||
if (/\+\)\\b$/.test(from) && new RegExp('\\$' + a.length + '$').test(to)) | |||
{ | |||
from = from.replace(/([^()]+)\)\\b$/, '?=$1\\b)'); | |||
to = to.replace(/\$\d+$/, ''); | |||
} | |||
/* Punkte im Suchmuster um optionale Leerzeichen erweitern */ | |||
from = from.replace(/\.(?=[\w\xC0-\xFF(])/g, '.(?: | )*'); | |||
t = t.replace(new RegExp(from, 'g'), to); | |||
} | |||
return t; | |||
}, | |||
backupNowikis: function(t) | |||
{ | |||
this.nowikis = []; | |||
var m, re = /<(includeonly|nowiki|pre|syntaxhighlight|source|math|code)\b[\s\S]*?<\/\1\s*>/gi; | |||
while (m = re.exec(t)) | |||
{ | |||
delete m.input; | |||
this.nowikis.push(m); | |||
} | |||
for (var i = this.nowikis.length; i--; ) | |||
{ | |||
var placeholder = '<nowiki>' + i + '</nowiki>'; | |||
t = t.slice(0, this.nowikis[i].index) + placeholder + | |||
t.slice(this.nowikis[i].index + this.nowikis[i][0].length); | |||
if (this.nowikis[i][1] == 'source') | |||
this.nowikis[i][0] = this.nowikis[i][0].replace(/^(<)\w+|\w+\s*(>)$/g, '$1syntaxhighlight$2'); | |||
this.nowikis[i][1] = placeholder; | |||
delete this.nowikis[i].index; | |||
} | |||
return t; | |||
}, | |||
restoreNowikis: function(t) | |||
{ | |||
for (var i = 0, len = this.nowikis.length, index = 0; i < len; i++) | |||
{ | |||
index = t.indexOf(this.nowikis[i][1], index); | |||
if (index >= 0) | |||
t = t.slice(0, index) + this.nowikis[i][0] + | |||
t.slice(index + this.nowikis[i][1].length); | |||
} | |||
delete this.nowikis; | |||
return t; | |||
}, | |||
backupFilenames: function(t) | |||
{ | |||
/* Dateinamen retten incl. Vereinheitlichung als „Datei:“ */ | |||
this.files = []; | |||
var m; | |||
/* Match <gallery> lines, [[File:Thumbnails]] and {{Template|Parameters.jpg}} */ | |||
var re = /(\n|\[\[:?)\s*(Bild|Datei|File|Image) *: *([^\n[\]|]*?) *([\n\]|])|(\|(?:[^=[\]{|}]*=)?\s*)([^\n[\]{|}]*\.(?:gif|jpe?g|ogg|png|svg))(\s*[|\}])/gi; | |||
while (m = re.exec(t)) | |||
{ | |||
/* Einheitliche Schreibweise und Leerzeichenausgleich */ | |||
m[1] = m[5] || m[1] + (this.locale ? 'Datei' : m[2]) + ':'; | |||
/* Keine geschützten Leerzeichen/Unterstriche in Dateinamen */ | |||
m[2] = (m[6] || m[3]).replace(/(_|%20| )/gi, ' '); | |||
m[3] = m[7] || m[4]; | |||
this.files.push(m); | |||
} | |||
for (var i = this.files.length; i--; ) | |||
t = t.slice(0, this.files[i].index) + this.files[i][1] + 'FILE' + i + | |||
this.files[i][3] + t.slice(this.files[i].index + this.files[i][0].length); | |||
return t; | |||
}, | |||
restoreFilenames: function(t) | |||
{ | |||
/* Gerettete Dateinamen wieder einsetzen */ | |||
for (var i = this.files.length; i--; ) | |||
{ | |||
t = t.replace(this.files[i][1] + 'FILE' + i + this.files[i][3], | |||
this.files[i][1] + this.files[i][2] + this.files[i][3]); | |||
} | |||
delete this.files; | |||
return t; | |||
} | } | ||
}; | |||
if (!t.match(/\|\s*Partei\s*=/)) | |||
t = t.replace(/((\|\s*(Bürgermeister|Organwalter)[^|}]*)+)/i, '$1|Partei =\n'); | |||
t = t.replace(/[\r\n]+\s*\|\s*(.{17})\s*=/ig, '\n|$1 ='); | |||
} | |||
if (typeof $ == 'function' && typeof mw == 'object' && mw.user.options.get('usebetatoolbar')) | |||
/* Gerettete Dateinamen wieder einsetzen */ | |||
for (i = files.length; i--; ) | |||
{ | { | ||
mw.loader.using('ext.wikiEditor.toolbar', function() | |||
t = t.replace(files[i][1] + 'Datei:' + i + files[i][4], files[i][1] + 'Datei:' + files[i][3] + files[i][4]); | |||
{ | |||
$(document).ready(function() | |||
{ | |||
$('#wpTextbox1').wikiEditor('addToToolbar', { | |||
'section': 'main', | |||
'group': 'format', | |||
'tools': { | |||
'autoFormatter': { | |||
'label': 'Auto-Format', | |||
'type': 'button', | |||
'icon': '//upload.wikimedia.org/wikipedia/commons/thumb/2/2c/Broom_icon.svg/22px-Broom_icon.svg.png', | |||
'action': { | |||
'type': 'callback', | |||
'execute': function() { return autoFormatter.click(this); } | |||
} | |||
} | |||
} | |||
}); | |||
}); | |||
}); | |||
} | } | ||
else if (typeof $ == 'function' && typeof mw == 'object' && mw.user.options.get('showtoolbar')) | |||
var changed = t != b.value.replace(/\r+\n/g, '\n'); | |||
if (changed) b.value = t.replace(/\s+$/, ''); | |||
if ((!a || !a.nodeType || a.nodeName === 'IMG') && $j) /* Vector/Monobook */ | |||
{ | { | ||
mw.loader.using('mediawiki.action.edit', function() | |||
var e = $j(a && a.nodeType ? a : 'img[rel=autoFormatter]'); | |||
{ | |||
e.css('backgroundColor', changed ? '#DEF740' : ''); | |||
mw.toolbar.addButton('//upload.wikimedia.org/wikipedia/commons/2/2e/Button_broom.png', | |||
e.css('borderRadius', changed ? '3px' : ''); | |||
'Auto-Format', '', '', '', 'mw-customeditbutton-autoFormatter'); | |||
e.css('opacity', changed ? '' : '0.4'); | |||
$(function() | |||
{ | |||
$('#mw-customeditbutton-autoFormatter').click(function() { return autoFormatter.click(this); }); | |||
}); | |||
}); | |||
} | } | ||
else if ( | else if (typeof hookEvent == 'function') | ||
{ | |||
return false; | |||
hookEvent('load', function() | |||
} | |||
{ | |||
/* Notfalls als Link unter dem Bearbeitungsfenster */ | |||
if (typeof $ === 'function' && typeof mw === 'object' && mw.user.options.get('usebetatoolbar')) /* Vector */ | |||
var e = document.getElementById('editform'); | |||
{ | |||
if (!e) return; | |||
mw.loader.using('ext.wikiEditor.toolbar', function() { | |||
var e = e.getElementsByTagName('SPAN'); | |||
$('#wpTextbox1').wikiEditor('addToToolbar', { | |||
for (var i = e.length; i--; ) | |||
'section': 'main', /* oder advanced */ | |||
if (e[i].className == 'editHelp') | |||
'group': 'format', | |||
{ | |||
var a = document.createElement('A'); | |||
'autoFormatter': { | |||
a.href = '#'; | |||
a.onclick = function() { return autoFormatter.click(this); } | |||
'type': 'button', | |||
a.appendChild(document.createTextNode('Auto-Format')); | |||
'icon': '//upload.wikimedia.org/wikipedia/commons/thumb/2/2c/Broom_icon.svg/22px-Broom_icon.svg.png', | |||
e[i].appendChild(document.createTextNode(' | ')); | |||
'action': { | |||
e[i].appendChild(a); | |||
'type': 'callback', | |||
break; | |||
'execute': function() { return doAutoFormat(this); } | |||
} | |||
} | } | ||
} | |||
}); | }); | ||
} | } | ||
})(jQuery, mediaWiki); | |||
} | |||
else if (typeof $ === 'function' && typeof mw === 'object' && mw.user.options.get('showtoolbar')) /* Monobook */ | |||
{ | |||
mw.loader.using('mediawiki.action.edit', function() { | |||
mw.toolbar.addButton('//upload.wikimedia.org/wikipedia/commons/2/2e/Button_broom.png', | |||
'Auto-Format', '', '', '', 'mw-customeditbutton-autoFormatter'); | |||
$(function() { | |||
$('#mw-customeditbutton-autoFormatter').click(function() { return doAutoFormat(this); }) | |||
}); | |||
}); | |||
} | |||
else if (typeof hookEvent === 'function') | |||
{ | |||
hookEvent('load', function() | |||
{ | |||
/* Notfalls als Link unter dem Bearbeitungsfenster */ | |||
f = document.getElementById('editform'); | |||
if (!f) return; | |||
var a = document.createElement('A'); | |||
a.href = '#'; | |||
a.onclick = function() { return doAutoFormat(this); } | |||
a.appendChild(document.createTextNode('Auto-Format')); | |||
var s = f.getElementsByTagName('SPAN'); | |||
for (var i = s.length - 1; i >= 0; i--) if (s[i].className === 'editHelp') { s = s[i]; break; } | |||
s.appendChild(document.createTextNode(' | ')); | |||
s.appendChild(a); | |||
}); | |||
} |
Version vom 10. Dezember 2012, 23:01 Uhr
/**
* Blendet eine „Auto-Format“-Funktion in der Werkzeugleiste ein, die viele typische Wikifizierungs-Fehler
* automatisch korrigiert. Eine ausführliche Beschreibung ist auf der Diskussionsseite zu finden.
* (<nowiki> zur Umgehung von [[bugzilla:8761]].)
*/
(function($, mw)
{
var autoFormatter = {
click: function(a)
{
this.clickedElement = a;
this.locale = typeof window.autoFormatLocalisation == 'undefined'
? mw.config.get('wgContentLanguage') == 'de'
: window.autoFormatLocalisation;
var elements = document.forms['editform'].elements;
var textbox = elements['wpTextbox1'], summary = elements['wpSummary'];
if (!textbox) return;
this.isDisambiguation = /\{\{\s*[Bb]egriffsklärung\s*\}\}/.test(textbox.value);
this.cleanElement(textbox);
summary.value = this.cleanInternalLinks(summary.value);
return false;
},
isChanged: function(oldValue, newValue)
{
/* Entfernte Leerräume am Textende zählen nie als Änderung */
oldValue = oldValue.replace(/\s+$/, '');
newValue = newValue.replace(/\s+$/, '');
/* Entfernte Leerräume am Zeilenende nicht als Änderung anzeigen, aber trotzdem ersetzen */
var changed = oldValue.replace(/[ \t\r]+\n/g, '\n') !== newValue;
var a = this.clickedElement;
if (!a || !a.nodeType || a.nodeName == 'IMG')
{
var e = $(a && a.nodeType ? a : 'img[rel=autoFormatter]');
e.css('backgroundColor', changed ? '#DEF740' : '');
e.css('borderRadius', changed ? '3px' : '');
e.css('opacity', changed ? '' : '.4');
}
else if (a) a.style.color = changed ? 'green' : 'silver';
/* Normalisierte Zeilenumbrüche nie als Änderung werten, das vermeidet Flackern */
return changed || oldValue.replace(/\r+\n/g, '\n') !== newValue;
},
cleanElement: function(e)
{
e.focus();
if (typeof e.selectionStart == 'number')
{
var scroll = e.scrollTop;
var s1 = e.selectionStart, s2 = e.selectionEnd;
if (s2 > s1)
{
var t = this.cleanText(e.value.substring(s1, s2));
if (t === false) return;
var newValue = e.value.substr(0, s1) + t + e.value.substr(s2);
e.value = newValue;
s2 = s1 + t.length + (e.value.length - newValue.length); /* Fix for Opera */
}
else if (!this.cleanAll(e)) return;
e.selectionStart = s1, e.selectionEnd = s2;
e.scrollTop = scroll;
}
else if (typeof document.selection == 'object') /* Internet Explorer */
{
var range = document.selection.createRange();
if (range.text.length > 0)
{
var t = this.cleanText(range.text);
if (t !== false) range.text = t;
}
else this.cleanAll(e);
}
else this.cleanAll(e);
},
cleanAll: function(e)
{
var t = this.cleanText(e.value);
if (t !== false) e.value = t.replace(/\s+$/, '');
return t !== false;
},
cleanText: function(t)
{
var oldValue = t;
t = t.replace(/[ \t\r]+\n/g, '\n');
/* Steuerzeichen, undefinierte Unicode-Nummern, LRM, ZERO WIDTH SPACE, BOM entfernen */
t = t.replace(/[\x00-\x08\x0C\x0E-\x1F\x7F-\x9F\u200B\u200E\uFEFF]+/g, '');
/* Unsichtbares weiches Trennzeichen sichtbar machen, egal wo */
t = t.replace(/\xAD+/g, '­');
t = this.backupNowikis(t);
/* Mehrfache Leerzeilen auf einzelne reduzieren */
if (!this.isDisambiguation) t = t.replace(/\n{3,}/g, '\n\n');
/* Geschützte Leerzeichen einheitlich als „ “, vereinfacht viele folgende Suchmuster,
muss vor backupFilenames() stehen */
t = t.replace(/&#(?:x0*A|0*16)0 *;/gi, ' ');
/* Verschiedenste Formen von HTML-Zeilenumbrüchen durch einheitliche ersetzen */
t = t.replace(/<(?:[\s\/]*br\b)+\s*(\s\w[^<>]*?)?[\s\/]*>/gi, '<br$1 />');
/* Unnötige HTML-Zeilenumbrüche entfernen, wenn sowieso ein Absatz folgt */
t = t.replace(/ *<br \/>(\n[\n#*:;])/gi, '$1');
t = this.backupFilenames(t);
t = this.cleanHeadlines(t);
if (this.locale)
{
/* „Weblinks“ immer als „Weblinks“ */
t = t.replace(/^== *(?:Externer?|External)? *(?:Weblinks?|Links?|Webseiten?|Websites?) *==/gim,
'== Weblinks ==');
/* Einheitliche Schreibweisen für Schlüsselwörter incl. Leerzeichenausgleich */
t = t.replace(/\{\{\s*(SEITENTITEL|DISPLAYTITLE):\s*/gi, '{{SEITENTITEL:');
t = t.replace(/\s*#(WEITERLEITUNG|REDIRECT)\s*\[\[/gi, '#WEITERLEITUNG [[');
t = t.replace(/\[\[\s*[KC]ategor[iy]e?\s*:\s*/gi, '[[Kategorie:');
t = t.replace(/(\[\[Datei:[^\[\]\n]*\S)\s*\|+\s*(?:(?:rechts|right)\s*\|+\s*)?(?:miniatur|thumb)\s*\|+\s*((rechts|right)\s*\|+\s*)?/gi,
'$1|miniatur|');
t = t.replace(/(\[\[Datei:[^\[\]\n]*\S)\s*\|+\s*(?:rechts|right)\s*\|+\s*/gi, '$1|rechts|');
t = t.replace(/(\[\[Datei:[^\[\]\n]*\S)\s*\|+\s*(?:links|left)\s*\|+\s*/gi, '$1|links|');
t = t.replace(/(\[\[Datei:[^\[\]\n]*\S)\s*\|+\s*(?:hochkant|upright)\s*(?:(=)\s*([\d.]*)\s*)?\|+\s*/gi,
'$1|hochkant$2$3|');
}
t = this.cleanTemplates(t);
t = t.replace(/\bclass\s*=\s*("( *[\w-]+ )*)? *\bprettytable\b/g, 'class=$1wikitable');
t = this.cleanReferences(t);
t = this.cleanDefaultSort(t);
t = this.cleanInternalLinks(t);
t = this.cleanExternalLinks(t);
/* Jede Kategorie in einer eigenen Zeile */
t = t.replace(/(\[\[Kategorie:[^\[\]\n]*\]\]) *(?!\n|$)/gi, '$1\n');
t = t.replace(/([^\n])(\[\[Kategorie:[^\[\]\n]*\]\])/gi, '$1\n$2');
t = t.replace(/(\[\[Kategorie:[^\[\]\n]*\]\]\n)(?!\[\[Kategorie:|\n|$)/gi, '$1\n');
/* Versehentliche Links um ISO-Daten entfernen */
t = t.replace(/\[+([12]\d{3}-[01]\d-[0-3]\d)\]+/g, '$1');
t = this.cleanDuplicateLinks(t);
/* Unnötige Leerzeichen bei HTML-Attributen, wichtig vor den Anführungszeichen */
while (/[<|][^<>|]*\b *= +"/i.test(t))
t = t.replace(/([<|][^<>|]*)\b *= +"/gi, '$1="');
/* Anführungszeichen */
t = t.replace(/([\s!#'(*+\/:;>[|-])"([^\s"“”„][^\n"“”„]*)"(?=[\s!'),.\/:;<?\]}-])/g, '$1„$2“');
/* Auslassungspunkte */
t = t.replace(/([([])\.\.\.(?=[)\]])/g, '$1…');
/* Bis-Striche bei Jahreszahlenbereichen */
t = t.replace(/([\s(|][12]\d{3}) *[–-] *(?=[12]\d{3}[^\d\]–-])/g, '$1–');
/* ISSNs aber ohne Bis-Striche */
t = t.replace(/(IS\wN\W*\d+)–(?=\d)/g, '$1-');
t = this.cleanISBNs(t);
t = this.cleanDates(t);
var re = /([\w'\)\]\xC0-\xFF]) +- +(?=[\w'\(\[\xC0-\xFF])/gi;
var i = t.search(/^\[\[ *[a-z-]{2,12} *:/m); //12 wegen [[zh-classical:…]]
if (i >= 0) t = t.slice(0, i).replace(re, '$1 – ') + t.slice(i);
else t = t.replace(re, '$1 – ');
t = this.cleanUnits(t);
/* Paragraf, Abs. und Satz mit geschützten Leerzeichen */
t = t.replace(/§(?: *| )(\d\w* +A[bsatz.]+)(?: *| )(\d+ +S[atz.]+) *(?=\d)/gi,
'§ $1 $2 ');
t = t.replace(/§(?: *| )(\d\w* +A[bsatz.]+) *(?=\d)/gi,
'§ $1 ');
t = t.replace(/§ *(?=\d)/gi,
'§ ');
t = this.executeUserReplacements(t);
t = this.cleanRedundantTemplateParameters(t);
t = this.cleanInfoboxVerwaltungseinheitInDeutschland(t);
t = this.restoreFilenames(t);
t = this.restoreNowikis(t);
return this.isChanged(oldValue, t) ? t : false;
},
cleanHeadlines: function(t)
{
/* Keine geschützten Leerzeichen in Überschriften */
while (/^=.* .*=$/im.test(t))
t = t.replace(/^(=.*) (?=.*=$)/gim, '$1 ');
/* Fettung zumindest kompletter Überschriften ist unerwünscht */
t = t.replace(/^(=+) *'''([^']+)''' *(?==+$)/gm, '$1 $2 ');
/* Repariert kaputte Überschriften, entfernt Doppelpunkte, setzt Leerzeichen */
t = t.replace(/^(=+) *(.*[^\s=:]):? *\1$/gm, '$1 $2 $1');
return t;
},
cleanInternalLinks: function(t)
{
var lang = mw.config.get('wgContentLanguage');
var ns = mw.config.get('wgFormattedNamespaces')[-1];
/* Permanente Weblinks in Spezialseiten-Syntax umwandeln */
var permaLinkReplace = function($0, $1, $2, $3)
{
return '[[:' + $1 + ':' +
($1 == lang ? ns : 'Special') + ':' +
($1 == 'de' ? 'Permanenter Link' : 'PermanentLink') + '/' + $2 +
(typeof $3 == 'string' ? '|' + $3 : '') + ']]';
}
/* Weblinks auf Sprachversionen (auch auf die eigene) in Wikilinks umwandeln */
var interWikiReplace = function($0, $1, $2, $3)
{
return '[[:' + $1 + ':' + $2.replace(/_/g, ' ') +
(typeof $3 == 'string' ? '|' + $3 : '') + ']]';
}
/* Schreibweise [[Weblink#Anker mit Leerzeichen|Beschriftung]] reparieren */
t = t.replace(/\[+ *(?:https?:)?\/\/([a-z-]+)\.wikipedia\.org\/w\/[\w.]*\?(?:title=[^&|\[\]\s]*&)?oldid=([^?|\[\]\n]+?) *\|+ *([^|\[\]\n]+?) *\]+/gi, permaLinkReplace);
t = t.replace(/\[+ *(?:https?:)?\/\/([a-z-]+)\.wikipedia\.org\/wiki\/([^?|\[\]\n]*?) *\|+ *([^|\[\]\n]+?) *\]+/gi, interWikiReplace);
/* Schreibweise [Weblink#Anker Beschriftung] umwandeln */
t = t.replace(/\[+ *(?:https?:)?\/\/([a-z-]+)\.wikipedia\.org\/w\/[\w.]*\?(?:title=[^&|\[\]\s]*&)?oldid=([^?|\[\]\s]+) +([^|\[\]\n]+?) *\]+/gi, permaLinkReplace);
t = t.replace(/\[+ *(?:https?:)?\/\/([a-z-]+)\.wikipedia\.org\/wiki\/([^?|\[\]\s]*) +([^|\[\]\n]+?) *\]+/gi, interWikiReplace);
/* Schreibweise [Weblink#Anker] umwandeln */
t = t.replace(/\[+ *(?:https?:)?\/\/([a-z-]+)\.wikipedia\.org\/w\/[\w.]*\?(?:title=[^&|\[\]\s]*&)?oldid=([^?|\[\]\s]+) *\]+/gi, permaLinkReplace);
t = t.replace(/\[+ *(?:https?:)?\/\/([a-z-]+)\.wikipedia\.org\/wiki\/([^?|\[\]\s]*) *\]+/gi, interWikiReplace);
/* Verbliebene projektinterne Weblinks ohne eckige Klammern ebenfalls umwandeln */
t = t.replace(/(?:https?:)?\/\/([a-z-]+)\.wikipedia\.org\/w\/[\w.]*\?(?:title=[^&|\[\]\s]*&)?oldid=([^?|\[\]\s]*[^?|\[\]\s!"),.:;<>?\\])/gi, permaLinkReplace);
t = t.replace(/(?:https?:)?\/\/([a-z-]+)\.wikipedia\.org\/wiki\/([^?|\[\]\s]*[^?|\[\]\s!"),.:;<>?\\])/gi, interWikiReplace);
/* Wikilinks mit unnötigem Präfix „:w:de:“, „w:de:“ oder „:de:“ vereinfachen */
t = t.replace(new RegExp('\\[\\[ *(?::? *w *)?: *' + lang + ' *: *((Kategorie *:|Category *:)?[^\\[\\]\\n]*\\S) *\\]\\]', 'gi'), function($0, $1, $2)
{
return '[[' + ($2 ? ':' : '') + $1 + ']]';
});
/* Anker in internen Links dekodieren */
t = t.replace(/(\[\[[^\n#\[\]{|]*#)([^\n\[\]|]+)(?=\|?[^\n\[\]|]*\]\])/gi, function($0, $1, $2)
{
try
{
$2 = decodeURIComponent($2.replace(/\.([289A-E][\dA-F]|[357][B-F]|40|60)/g, '%$1')).
replace(/[%\[\]{|}]/g, function($0)
{
return '%' + $0.charCodeAt(0).toString(16).toUpperCase();
});
}
catch (ex) { /* keine Veränderung im Fehlerfall */ }
return ($1 + $2).replace(/_/g, ' ');
});
/* [[Link|Die]]s wird zu [[Link|Dies]] weil besser lesbar */
t = t.replace(/(\[\[[^|\[\]\n]+\|[^|\[\]\n]+)\]\]([a-zäöüß]+)/gi, '$1$2]]');
/* [[Link|Link]]s werden zu [[Link]]s weil kürzer und besser lesbar */
t = t.replace(/\[\[([^|\[\]\n]+)\|\1([^|\[\]\n]*)\]\]/g, '[[$1]]$2');
return t;
},
cleanExternalLinks: function(t)
{
/* Doppelte eckige Klammern um Weblinks vereinfachen */
t = t.replace(/\[+ *(https?:\/\/[^\n[\]]*?) *\]+/gi, '[$1]');
/* Weblinks mit senkrechtem Strich reparieren */
t = t.replace(/(\[https?:\/\/[^\s[\]|]*?) *\| *(?=[^\n=[\]|]+\])/gi, '$1 ');
/* Schrägstriche am Ende einfacher Domains ergänzen */
t = t.replace(/(\[https?:\/\/(?:[\w-]+\.)+\w+) +/gi, '$1/ ');
/* Domains klein schreiben, egal ob beschriftet oder nicht */
t = t.replace(/\bhttps?:\/\/[a-z.-]*[A-Z][\w.-]*/g, function($0) { return $0.toLowerCase(); });
/* Verbliebene projektinterne Weblinks protokollrelativ machen */
t = t.replace(/\[ *https?:\/+(?=[a-z-]+\.wikipedia\.org\b)/gi, '[//');
return t;
},
cleanDuplicateLinks: function(t)
{
/* Daten und Jahreszahlen in Personendaten nie verlinken */
var re = /\{\{\s*[Pp]ersonendaten\s*\|\s*([^{}]*?)\s*\|\s*((?:GEBURTS|STERBE)DATUM\s*=[^\n=[\]{|}]*)\[+([^\n=[\]{|}]+)\]+/g;
while (re.test(t)) t = t.replace(re, '{{Personendaten\n|$1\n|$2$3');
/* Doppelte Jahreszahlen entlinken */
var re = /\[\[([12]\d{3})\]\]/g, found = [], a = [];
/* Jeweils ersten Fund eines Jahres merken, danach entlinken */
while (m = re.exec(t)) found[m[1]] ? a.push(m) : found[m[1]] = true;
for (var i = a.length; i--; )
t = t.slice(0, a[i].index) + a[i][1] + t.slice(a[i].index + a[i][0].length);
return t;
},
cleanDates: function(t)
{
var months = ['Januar', 'Februar', 'März', 'April', 'Mai', 'Juni',
'Juli', 'August', 'September', 'Oktober', 'November', 'Dezember'];
/* Missverständliches deutsches Datumsformat durch Langform ersetzen */
t = t.replace(/([\s(])(3[01]|[12]\d|0?[1-9])\. *(1[012]|0?[1-9])\. *(?=[12]\d{3}[^\d.–-])/g,
function($0, $1, $2, $3, $4)
{
return $1 + 1 * $2 + '. ' + months[$3 - 1] + ' ';
});
/* In „1850–14. Januar“ immer „bis“ einsetzen */
t = t.replace(/(\s[12]\d{3}) *[–-] *(3[01]|[12]\d|0?[1-9])\. *([A-Z][a-zä]+)/g,
function($0, $1, $2, $3)
{
for (var i = months.length; i--; )
if ($3 == months[i]) return $1 + ' bis ' + $2 + '. ' + $3;
return $0;
});
/* „1980–90“ immer als „1980–1990“ */
t = t.replace(/([\s(|][12]\d{3}) *- *(?=(?:1[3-9]|[2-9]\d)\b[^.=–-])/g, '$1–'); /* Bindestrich gefunden */
t = t.replace(/\b(([12]\d)\d\d) *– *(?=\d\d\b[^=–-])/g, '$1–$2'); /* Bis-Strich gefunden */
return t;
},
cleanUnits: function(t)
{
/* Maßeinheiten immer mit Leerzeichen */
t = t.replace(/(\b\d+) ?(?=(?:k[Bgm]|Ki?B|[MGT](i?B|Hz)|[mc]m|k?Hz|EUR|Euro|CHF|US[D$]|JPY|[gm€¥])[\s!'),.\/:;?²³“])/g,
'$1 ');
/* Prozentwerte erhalten seit Mitte 2007 automatisch ein geschütztes Leerzeichen */
t = t.replace(/([\s'(*+,\/:;>–„−-]\d+)(?: )?(?=%[\s!'),.\/:<?\]|“])/gi, '$1 ');
return t;
},
cleanISBNs: function(t)
{
/* ISBNs mit Bindestrichen gliedern, Gruppennummern 0 und 1 für englischsprachige Bücher */
t = t.replace(/\bISBN(?:-?1[03])?:?\s*((?:9-?7-?[89]-?)?0)([\d-]{8,}[\dX]\b)/gi, function($0, $1, $2) {
return 'ISBN ' + $1.replace(/^9\D*7\D*([89])\D*0/, '97$1-0') + '-' + $2.replace(/[^\dX]/gi, '').
replace(/^([01]\d)(\d{6})/, '$1-$2-').
replace(/^([2-6]\d\d)(\d{5})/, '$1-$2-').
replace(/^(7\d{3}|8[0-4]\d\d)(\d{4})/, '$1-$2-').
replace(/^(8[5-9]\d{3})(\d{3})/, '$1-$2-').
replace(/^(9[0-4]\d{4})(\d\d)/, '$1-$2-').
replace(/^(9[5-9]\d{5})(\d)/, '$1-$2-');
});
t = t.replace(/\bISBN(?:-?1[03])?:?\s*((?:9-?7-?[89]-?)?1)([\d-]{8,}[\dX]\b)/gi, function($0, $1, $2) {
return 'ISBN ' + $1.replace(/^9\D*7\D*([89])\D*1/, '97$1-1') + '-' + $2.replace(/[^\dX]/gi, '').
replace(/^(0\d)(\d{6})/, '$1-$2-').
replace(/^([1-3]\d\d)(\d{5})/, '$1-$2-').
replace(/^(4\d{3}|5[0-4]\d\d)(\d{4})/, '$1-$2-').
replace(/^(5[5-9]\d{3}|[67]\d{4}|8[0-5]\d{3}|86[0-8]\d\d|869[0-7]\d)(\d{3})/, '$1-$2-').
replace(/^(869[89]\d\d|8[7-9]\d{4}|9[0-8]\d{4}|99[0-8]\d{3})(\d\d)/, '$1-$2-').
replace(/^(999\d{4})(\d)/, '$1-$2-');
});
/* Gruppennummer 3 für deutschsprachige Bücher */
t = t.replace(/\bISBN(?:-?1[03])?:?\s*((?:9-?7-?[89]-?)?3)([\d-]{8,}[\dX]\b)/gi, function($0, $1, $2) {
return 'ISBN ' + $1.replace(/^9\D*7\D*([89])\D*3/, '97$1-3') + '-' + $2.replace(/[^\dX]/gi, '').
replace(/^(0[0-24-9]|1\d)(\d{6})/, '$1-$2-').
replace(/^(03[0-3]|[2-6]\d\d)(\d{5})/, '$1-$2-').
replace(/^(03[4-6]\d|7\d{3}|8[0-4]\d\d)(\d{4})/, '$1-$2-').
replace(/^(03[7-9]\d\d|8[5-9]\d{3}|95[4-9]\d\d|9[69]\d{3})(\d{3})/, '$1-$2-').
replace(/^(9[0-4]\d{4})(\d\d)/, '$1-$2-').
replace(/^(95[0-3]\d{4}|9[78]\d{5})(\d)/, '$1-$2-');
});
return t;
},
cleanReferences: function(t)
{
t = t.replace(/<\s*references\s*(\s\b[^<\/>]*?)?\s*\/\s*>/gi, '<references$1 />');
t = t.replace(/<\s*ref\s*(\s\b[^<\/>]*?)\s*\/\s*>/gi, '<ref$1 />');
t = t.replace(/<\s*ref\s*(\s\b[^<\/>]*?)?\s*>\s*/gi, '<ref$1>');
t = t.replace(/\s*<\s*\/\s*ref\s*>/gi, '</ref>');
/* Lücken zwischen Satzende und <ref> oder zwei <ref> entfernen */
t = t.replace(/([!,.;?]|<ref\b[^<\/>]*(?:\/|>[^<>]*<\/ref)>) *[ \n](?=<ref[ >])/gi, '$1');
/* Zwei gleiche Satzzeichen vor und nach einem <ref> auf eins kürzen */
t = t.replace(/([!,.;?])(<ref\b[^<\/>]*(?:\/|>[^<>]*<\/ref)>)\1/gi, '$1$2');
return t;
},
cleanDefaultSort: function(t)
{
if (this.locale) t = t.replace(/\{\{\s*(SORTIERUNG|DEFAULTSORT):\s*/gi, '{{SORTIERUNG:');
/* Sortierung soll nie Umlaute enthalten */
t = t.replace(/(\{\{(?:SORTIERUNG|DEFAULTSORT):[^{}\n]*?)ß/, '$1ss');
var sortSrc = '¢£¥©ª®²³¹ºÀÁÂÃÄÅÇÈÉÊËÌÍÎÏÐÑÒÓÔÕÖØÙÚÛÜÝàáâãäåçèéêëìíîïðñòóôõöøùúûüýÿ';
var sortDst = 'cLYCaR231oAAAAAACEEEEIIIIDNOOOOOOUUUUYaaaaaaceeeeiiiidnoooooouuuuyy';
var re = new RegExp('(\\{\\{(?:SORTIERUNG|DEFAULTSORT):[^{}\n]*?)([' + sortSrc + '])', 'g');
while (re.test(t))
t = t.replace(re, function($0, $1, $2) { return $1 + sortDst.charAt(sortSrc.indexOf($2)); });
/* Sortierung soll einheitliche Groß/Kleinschreibung haben */
re = /(\{\{(?:SORTIERUNG|DEFAULTSORT):[^{}\n]*?)(\b[a-z])/g;
while (re.test(t)) t = t.replace(re, function($0, $1, $2) { return $1 + $2.toUpperCase(); });
re = /(\{\{(?:SORTIERUNG|DEFAULTSORT):[^{}\n]*?)(\B[A-Z]+)/g;
while (re.test(t)) t = t.replace(re, function($0, $1, $2) { return $1 + $2.toLowerCase(); });
return t;
},
cleanTemplates: function(t)
{
t = t.replace(/\{\{\s*(?:Vorlage|Template)\s*:\s*/gi, '{{');
/* Unterstriche aus allen Vorlagennamen entfernen */
while (/(?:^|[^{])\{\{[ 0-9a-z\xC0-\xFF-]+_/i.test(t))
t = t.replace(/((?:^|[^{])\{\{[ 0-9a-z\xC0-\xFF-]+)_/gi, '$1 ');
/* {{Commons|Kategorie:…}} durch {{Commonscat|…}} austauschen */
t = t.replace(/\{\{\s*Commons(?:cat)?\s*\|\s*[KC]ategor[iy]e?\s*:\s*/gi, '{{Commonscat|');
/* Projektweit einheitliche Schreibweisen für häufig verwendete Vorlagen */
t = t.replace(/\{\{\s*Belege fehlen[ |]*(?=\s*(?:\||\}\}))/gi, '{{Belege fehlen');
t = t.replace(/\{\{\s*Commons(cat)?[\s|]*\}\}/gi, '{{Commons$1}}');
t = t.replace(/\{\{\s*Commons(cat)?\s*\|\s*/gi, '{{Commons$1|');
t = t.replace(/\{\{\s*Dieser Artikel\s*\|\s*/gi, '{{Dieser Artikel|');
t = t.replace(/\{\{\s*Dts(x?)\s*\|\s*/gi, '{{dts$1|');
t = t.replace(/\{\{\s*(?:Hauptartikel|Hauptseite|Main)\s*\|\s*/gi, '{{Hauptartikel|');
t = t.replace(/\{\{\s*(?:Internetquelle|Weblink)(?=\s*\|)/gi, '{{Internetquelle');
t = t.replace(/\{\{\s*Lang\s*\|\s*/gi, '{{lang|');
t = t.replace(/\{\{\s*Nts\s*\|\s*/gi, '{{nts|');
t = t.replace(/\{\{\s*Rotten ?Tomatoes\s*\|\s*/gi, '{{Rotten Tomatoes|');
t = t.replace(/\{\{\s*S(?:iehe auch|ee also)\s*\|\s*/gi, '{{Siehe auch|');
t = t.replace(/\{\{\s*Sort(Date|Key|KeyName)\s*\|\s*/gi, '{{Sort$1|');
t = t.replace(/\{\{\s*Wik(ibooks|inews|iquote|isource|tionary)[\s|]*(?=\||\}\})/gi, '{{Wik$1');
t = t.replace(/\{\{\s*Normdaten\s*\|\s*PND\s*=\s*/g, '{{Normdaten|TYP=p|GND=');
t = t.replace(/\{\{\s*WBA\s*\|\s*/gi, '{{Waybackarchiv|');
t = t.replace(/\[ *\{\{\s*Wayback\w*\s*\|\s*url\s*=\s*[^\s\d{|}]*(\d{1,14})\/(\w+:[^\s{|}]*)[^{}]*\}\}\s+([^[\]|]*)\]/gi,
'{{Webarchiv | url=$2 | wayback=$1 | text=$3}}');
return t;
},
cleanRedundantTemplateParameters: function(t)
{
var parameters = window.redundantTemplateParameters || [
'Infobox Arcade|Titel',
'Infobox Band|Name',
'Infobox Berg|NAME',
'Infobox Burg|Name',
'Infobox Chemikalie|Name',
'Infobox Computer- und Videospiel|Titel',
'Infobox Eishockeyspieler|Name',
'Infobox Flughafen|Name',
'Infobox Flugzeug|Name',
'Infobox Fluss|NAME',
'Infobox Fußballspieler|kurzname',
'Infobox Gemeinde in Österreich|Name',
'Infobox Gemeinde in Deutschland|Name',
'Infobox Gemeinde in Italien|nomeComune',
'Infobox Gemeindeverband in Deutschland|Name',
'Infobox Insel|NAME',
'Infobox Landkreis|Name',
'Infobox Musikalbum|Titel',
'Infobox Ort in den Vereinigten Staaten|Name',
'Infobox Ort in der Schweiz|NAME_ORT',
'Infobox Ort in Polen|Ort',
'Infobox Ort in Tschechien|Ort',
'Infobox Ortsteil einer Gemeinde|Ortsteil',
'Infobox Ortsteil einer Gemeinde in Deutschland|Ortsteil',
'Infobox PKW-Modell|Modell',
'Infobox Publikation|titel',
'Infobox Schiff|Name',
'Infobox See|NAME',
'Infobox Software|Name',
'Infobox Stadion|Name',
'Infobox Unternehmen|Name'];
var title = '\\s*=\\s*(?:' +
mw.config.get('wgTitle').replace(/([.\\+*?\[\](){}|])/g, '\\$1').replace(/\s+/g, '\\s+') +
'|\\{+\\w*\\}+)?\\s*(?=[}|])';
for (var i = parameters.length; i--; )
{
t = t.replace(new RegExp('(\\{\\{\\s*' +
parameters[i].replace(/[\s_]+/g, '[\\s_]+').replace('|', '\\s*(?:\\|[^{}]*)?)\\|\\s*') +
title, 'g'), '$1');
}
return t;
},
cleanInfoboxVerwaltungseinheitInDeutschland: function(t)
{
t = t.replace(/\{\{\s*Infobox Ort in Deutschland/gi, '{{Infobox Gemeinde in Deutschland');
if (!/\{\{\s*Infobox Gemeinde in Deutschland/i.test(t)) return t;
t = t.replace(/\|\s*lat_deg\s*=\s*([^\s{|}]*)(?:\s*\|\s*lat_min\s*=\s*([^\s{|}]*)(?:\s*\|\s*lat_sec\s*=\s*([^\s{|}]*))?)?/ig,
'|Breitengrad = $1/$2/$3');
t = t.replace(/\|\s*lon_deg\s*=\s*([^\s{|}]*)(?:\s*\|\s*lon_min\s*=\s*([^\s{|}]*)(?:\s*\|\s*lon_sec\s*=\s*([^\s{|}]*))?)?/ig,
'|Längengrad = $1/$2/$3');
t = t.replace(/(\|\s*(?:Breit|Läng)engrad\s*=\s*[\d.\/]*?)(?:\/0*)+(?=[\n|}])/g, '$1');
if (/Art\s*=\s*(?:Amt|(?:Samt|Verbands)gemeinde|Verwaltungsgemeinschaft|(?:Gemeinde)?verwaltungsverband)/i.test(t))
{
t = t.replace(/\{\{\s*Infobox Gemeinde in Deutschland/gi, '{{Infobox Gemeindeverband in Deutschland');
t = t.replace(/\|\s*(?:PLZ(?:-alt)?|Vorwahl|Kfz)\s*=[^{|}]*/ig, '');
t = t.replace(/\|\s*Bürgermeister( {0,5}|titel)?\s*=\s*/ig, '|Organwalter$1 = ');
if (!/\|\s*Gliederung\s*=/.test(t))
t = t.replace(/(\|\s*Adresse\s*=)/i, '|Gliederung = \n$1');
t = t.replace(/\|\s*Gliederung\s*=\s*(\d*).*/i, '|Gliederung = $1');
}
if (!/\|\s*Partei\s*=/.test(t))
t = t.replace(/((?:\|\s*(?:Bürgermeister|Organwalter)[^|}]*)+)/i, '$1|Partei =\n');
t = t.replace(/\n+\s*\|\s*([ \wäöüß-]{17})\s*=/ig, '\n|$1 =');
return t;
},
executeUserReplacements: function(t)
{
var from, replacements = window.autoFormatReplacements || {};
for (from in replacements)
{
var to = replacements[from];
/* Wenn die Ersetzungen kein assoziatives Objekt sondern ein 2-dimensionales Array sind */
if (typeof to == 'object' && to.length > 1) from = to[0], to = to[1];
/* Wenn das Suchmuster schon ein regulärer Ausdruck ist */
if (typeof from == 'object')
{
t = t.replace(from, to);
continue;
}
/* Leere Suchmuster sicherheitshalber nicht zulassen */
if (/^\s*$/.test(from) || typeof to != 'string') continue;
/* Die meisten Regex-Zeichen maskieren, außer Zeichenklassen */
from = from.replace(/(?=[.+*?(){}<>|])/g, '\\');
/* Wortgrenzen beachten */
from = from.replace(/^(?=\w|\\d)/, '\\b').replace(/(\w)$/, '$1\\b');
var a = [];
for (var m, re = /\\[dw]/g, i = 1; m = re.exec(from); a.push(m))
to = to.replace(m[0], '$' + i++);
for (var i = a.length; i--; )
from = from.slice(0, a[i].index) + (a[i][0] == '\\d' ? '(\\d+)' :
'([A-Za-z\xB5\xC0-\xD6\xD8-\xF6\xF8-\u02AF]+)') +
from.slice(a[i].index + 2);
/* Look-ahead verwenden, wenn ein Platzhalter in Suchmuster und Ersatz am Ende steht */
if (/\+\)\\b$/.test(from) && new RegExp('\\$' + a.length + '$').test(to))
{
from = from.replace(/([^()]+)\)\\b$/, '?=$1\\b)');
to = to.replace(/\$\d+$/, '');
}
/* Punkte im Suchmuster um optionale Leerzeichen erweitern */
from = from.replace(/\.(?=[\w\xC0-\xFF(])/g, '.(?: | )*');
t = t.replace(new RegExp(from, 'g'), to);
}
return t;
},
backupNowikis: function(t)
{
this.nowikis = [];
var m, re = /<(includeonly|nowiki|pre|syntaxhighlight|source|math|code)\b[\s\S]*?<\/\1\s*>/gi;
while (m = re.exec(t))
{
delete m.input;
this.nowikis.push(m);
}
for (var i = this.nowikis.length; i--; )
{
var placeholder = '<nowiki>' + i + '</nowiki>';
t = t.slice(0, this.nowikis[i].index) + placeholder +
t.slice(this.nowikis[i].index + this.nowikis[i][0].length);
if (this.nowikis[i][1] == 'source')
this.nowikis[i][0] = this.nowikis[i][0].replace(/^(<)\w+|\w+\s*(>)$/g, '$1syntaxhighlight$2');
this.nowikis[i][1] = placeholder;
delete this.nowikis[i].index;
}
return t;
},
restoreNowikis: function(t)
{
for (var i = 0, len = this.nowikis.length, index = 0; i < len; i++)
{
index = t.indexOf(this.nowikis[i][1], index);
if (index >= 0)
t = t.slice(0, index) + this.nowikis[i][0] +
t.slice(index + this.nowikis[i][1].length);
}
delete this.nowikis;
return t;
},
backupFilenames: function(t)
{
/* Dateinamen retten incl. Vereinheitlichung als „Datei:“ */
this.files = [];
var m;
/* Match <gallery> lines, [[File:Thumbnails]] and {{Template|Parameters.jpg}} */
var re = /(\n|\[\[:?)\s*(Bild|Datei|File|Image) *: *([^\n[\]|]*?) *([\n\]|])|(\|(?:[^=[\]{|}]*=)?\s*)([^\n[\]{|}]*\.(?:gif|jpe?g|ogg|png|svg))(\s*[|\}])/gi;
while (m = re.exec(t))
{
/* Einheitliche Schreibweise und Leerzeichenausgleich */
m[1] = m[5] || m[1] + (this.locale ? 'Datei' : m[2]) + ':';
/* Keine geschützten Leerzeichen/Unterstriche in Dateinamen */
m[2] = (m[6] || m[3]).replace(/(_|%20| )/gi, ' ');
m[3] = m[7] || m[4];
this.files.push(m);
}
for (var i = this.files.length; i--; )
t = t.slice(0, this.files[i].index) + this.files[i][1] + 'FILE' + i +
this.files[i][3] + t.slice(this.files[i].index + this.files[i][0].length);
return t;
},
restoreFilenames: function(t)
{
/* Gerettete Dateinamen wieder einsetzen */
for (var i = this.files.length; i--; )
{
t = t.replace(this.files[i][1] + 'FILE' + i + this.files[i][3],
this.files[i][1] + this.files[i][2] + this.files[i][3]);
}
delete this.files;
return t;
}
};
if (typeof $ == 'function' && typeof mw == 'object' && mw.user.options.get('usebetatoolbar'))
{
mw.loader.using('ext.wikiEditor.toolbar', function()
{
$(document).ready(function()
{
$('#wpTextbox1').wikiEditor('addToToolbar', {
'section': 'main',
'group': 'format',
'tools': {
'autoFormatter': {
'label': 'Auto-Format',
'type': 'button',
'icon': '//upload.wikimedia.org/wikipedia/commons/thumb/2/2c/Broom_icon.svg/22px-Broom_icon.svg.png',
'action': {
'type': 'callback',
'execute': function() { return autoFormatter.click(this); }
}
}
}
});
});
});
}
else if (typeof $ == 'function' && typeof mw == 'object' && mw.user.options.get('showtoolbar'))
{
mw.loader.using('mediawiki.action.edit', function()
{
mw.toolbar.addButton('//upload.wikimedia.org/wikipedia/commons/2/2e/Button_broom.png',
'Auto-Format', '', '', '', 'mw-customeditbutton-autoFormatter');
$(function()
{
$('#mw-customeditbutton-autoFormatter').click(function() { return autoFormatter.click(this); });
});
});
}
else if (typeof hookEvent == 'function')
{
hookEvent('load', function()
{
/* Notfalls als Link unter dem Bearbeitungsfenster */
var e = document.getElementById('editform');
if (!e) return;
var e = e.getElementsByTagName('SPAN');
for (var i = e.length; i--; )
if (e[i].className == 'editHelp')
{
var a = document.createElement('A');
a.href = '#';
a.onclick = function() { return autoFormatter.click(this); }
a.appendChild(document.createTextNode('Auto-Format'));
e[i].appendChild(document.createTextNode(' | '));
e[i].appendChild(a);
break;
}
});
}
})(jQuery, mediaWiki);