Bikarhêner:Şêr/autoFormatter.js

Ji Wîkîpediya, ensîklopediya azad.

Zanibe: Piştî weşandinê, ji bo dîtina guhartinan dibe ku hewce be "cache"ya geroka xwe paqij bikî.

  • Firefox / Safari: Pê li Shift û Reload bike an jî Ctrl-F5 an Ctrl-R bike (ji bo Mac: ⌘-R)
  • Google Chrome: Pê li Ctrl-Shift-R (ji bo Mac: ⌘-Shift-R) bike
  • Internet Explorer / Edge: Pê li Ctrl û Refresh bike, an jî Ctrl-F5 bike
  • Opera: Pê li Ctrl-F5 bike.
/**
 * Blendet eine „Auto-Format“-Funktion unter dem Bearbeitungsfenster ein, die
 * viele typische Wikifizierungs-Fehler automatisch korrigiert.
 * Eine ausführliche Beschreibung ist auf der Diskussionsseite zu finden.
 */
function doAutoFormat(a)
{
    var i;
    var e = document.forms['editform'].elements;
    var b = e['wpTextbox1'];
    var t = b.value;

    /* Überflüssige Leerräume entfernen, aber das allein rechtfertigt keinen Edit */
    t = t.replace(/\s+$/, '');
    t = t.replace(/[ \t\r]+\n/g, '\n');
    b.value = 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 "&nbsp;", vereinfacht viele folgende Suchmuster */
    t = t.replace(/&#(x0*A|0*16)0 *;/gi, '&nbsp;');
    /* Die verschiedensten Formen der Zeilenumbrüche durch korrekte ersetzen */
    t = t.replace(/<[\s\/]*br\s*(\s\w[^>]*?)?[\s\/]*>/gi, '<br$1 />');

    /* Dateinamen retten incl. Vereinheitlichung als "Datei:" */
    var regex = /(^|\[\[)\s*(Bild|Datei|File|Image)\s*:\s*([^|\]\n]*?)\s*([|\]\n])/gim;
    var files = [], match;
    while (match = regex.exec(t)) files[files.length] = match;
    for (i = files.length; i--; )
    {
        if (!files[i][1]) files[i][0] = (files[i][1] = '\n') + files[i][0];
        /* Keine geschützten Leerzeichen/Unterstriche in Dateinamen */
        files[i][3] = files[i][3].replace(/(_|%20|&nbsp;)/gi, ' ');
        /* Einheitliche Schreibweise und Leerzeichenausgleich */
        t = t.replace(files[i][0], files[i][1] + 'Datei:' + i + files[i][4]);
    }

    /* == Überschriften == immer mit Leerzeichen, repariert auch kaputte Überschriften */
    t = t.replace(/^(=+) *(.*[^ =]) *\1$/gm, '$1 $2 $1');

    /* Weblinks immer als Weblinks */
    t = t.replace(/^== *(External|Externer?)? *(Links?|Weblinks?|Webseiten?|Websites?) *==/gim,
        '== Weblinks ==');

    /* Einheitliche Schreibweisen für Schlüsselwörter incl. Leerzeichenausgleich */
    t = t.replace(/\[\[\s*(Category|Kategorie)\s*:\s*/gi, '[[Kategorie:');
    t = t.replace(/(\[\[Datei:[^\]]*?)\s*\|+\s*((rechts|right)\s*\|+\s*)?(miniatur|thumb)\s*\|+\s*((rechts|right)\s*\|+\s*)?/gi, '$1|miniatur|');
    t = t.replace(/(\[\[Datei:[^\]]*?)\s*\|+\s*(rechts|right)\s*\|+\s*/gi, '$1|rechts|');
    t = t.replace(/(\[\[Datei:[^\]]*?)\s*\|+\s*(links|left)\s*\|+\s*/gi, '$1|links|');
    t = t.replace(/(\[\[Datei:[^\]]*?)\s*\|+\s*(hochkant|upright)\s*((=)\s*([\d.]*))?\s*\|+\s*/gi, '$1|hochkant$4$5|');
    t = t.replace(/\{\{\s*(Template|Vorlage)\s*:\s*/gi, '{{');
    t = t.replace(/\{\{\s*Commons(cat\s*\|(\s*(Category|Kategorie)\s*:)?|\s*\|\s*(Category|Kategorie)\s*:)\s*/gi, '{{Commonscat|');
    t = t.replace(/\{\{\s*Commons\s*\|\s*/gi, '{{Commons|');
    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');

    /* Sortierung soll nie Umlaute enthalten */
    t = t.replace(/(\{\{SORTIERUNG:[^}]*?)ß/, '$1ss');
    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(); });

    /* Weblinks auf Wikipedia-Sprachversionen in Wikilinks umwandeln */
    var interWikiReplace = function($0, $1, $2, $3)
    {
        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]');

    /* Anker in Links dekodieren */
    t = t.replace(/(\[\[[^#|\]]*#)([^|\]]+)(\|?[^|\]]*\]\])/gi, function($0, $1, $2, $3)
    {
        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;
    });

    /* [[Link|Die]]s wird zu [[Link|Dies]] weil besser lesbar */
    t = t.replace(/(\[\[[^\|\]]+\|[^\|\]]+)\]\]([a-zäöüß]+)/gi, '$1$2]]');
    /* [[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');

    /* Versehentliche Links um ISO-Daten entfernen */
    t = t.replace(/\[+([12]\d{3}-[01]\d-[0-3]\d)\]+/g, '$1');
    /* Doppelte Jahreszahlen entlinken */
    var m = t.match(/\[\[[12]\d{3}\]\]/g);
    if (m && m.length > 1)
    {
        var p, pMax = t.lastIndexOf('{{Personendaten');
        for (i = m.length; i--; )
        {
            p = t.lastIndexOf(m[i], pMax > 0 ? pMax : t.length);
            /* Je ein Duplikat entfernen, wenn vorhanden */
            if (p > t.indexOf(m[i]))
                t = t.substr(0, p) + m[i].replace(/[\[\]]+/g, '') +
                    t.substring(p + m[i].length, t.length);
        }
    }

    /* Unnötige Leerzeichen bei HTML-Attributen, wichtig vor den Anführungszeichen */
    while (t.match(/[<|][^|>]*\b *= +"/im))
        t = t.replace(/([<|][^|>]*)\b *= +"/gim, '$1="');
    /* Anführungszeichen */
    t = t.replace(/([^\w=])["“]([^\n"“”]+)["”]([^\w])/gm, '$1„$2“$3');
    /* Bis-Striche bei Jahreszahlen */
    t = t.replace(/([\s(|][12]\d{3}) *[–-]{1,2} *([12]\d{3}[^\d–-])/g, '$1–$2');
    /* ISSNs aber ohne Bis-Striche */
    t = t.replace(/(IS\wN\W*\d+)–(\d)/g, '$1-$2');
    /* ISBNs mit Bindestrichen gliedern */
    t = t.replace(/\bISBN(-?1[03])?:?\s*((9-?7-?[89]-?)?3)([\d-]{8,}[\dX]\b)/gi, function($0, $1, $2, $3, $4) {
        return 'ISBN ' + $2.replace(/^9\D*7\D*([89])\D*3/, '97$1-3') + '-' + $4.replace(/[^\dX]/gi, '').
            replace(/^(0[5-9]|1\d)(\d{6})/, '$1-$2-').
            replace(/^([2-6]\d\d)(\d{5})/, '$1-$2-'). /* einschl. Einzel-ISBNs Österreich*/
            replace(/^(7[3-9]\d\d|8[0-4]\d\d)(\d{4})/, '$1-$2-').
            replace(/^(8[5-9]\d{3})(\d{3})/, '$1-$2-').
            replace(/^(9[1-4]\d{4})(\d\d)/, '$1-$2-').
            replace(/^(9[78]\d{5})(\d)/, '$1-$2-');
    });

    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] + ' ' + $4;
        });
    /* In "1850–14. Januar" immer "bis" einsetzen */
    t = t.replace(/(\s[12]\d{3}) *[–-]{1,2} *(3[01]|[12]\d|0?[1-9])\. *([A-Z][a-zä]+)/g,
        function($0, $1, $2, $3)
        {
            for (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)[^\d.=–-])/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');

    /* Maßeinheiten immer mit Leerzeichen */
    t = t.replace(/\b(\d+) ?([mck]?m|kg|[KMG]iB|[kMG](B|Hz)|KB)\b/g, '$1&nbsp;$2');
    /* Prozentwerte erhalten seit Mitte 2007 automatisch ein geschütztes Leerzeichen */
    t = t.replace(/\b(\d+)(&nbsp;)?(%[^\w"%;])/gi, '$1 $3');
    /* Paragraf, Abs. und Satz mit geschützten Leerzeichen */
    t = t.replace(/§( |&nbsp;)+(\d+\w* +Abs\.)( |&nbsp;)+(\d+ +Satz) +(\d+)/gi,
        '§&nbsp;$2&nbsp;$4&nbsp;$5');
    t = t.replace(/§( |&nbsp;)+(\d+\w* +Abs\.) +(\d+)/gi,
        '§&nbsp;$2&nbsp;$3');
    t = t.replace(/§ +(\d+)/gi,
        '§&nbsp;$1');

    var autoFormatReplacements = window.autoFormatReplacements || [], from;
    for (from in autoFormatReplacements)
    {
        var to = autoFormatReplacements[from];
        from = from.replace(/([.+*?(){}<>|])/g, '\\$1'); //Regex-Zeichen maskieren
        from = from.replace(/^(\w)/, '\\b$1').replace(/(\w)$/, '$1\\b'); //Wortgrenzen beachten
        for (i = 1; from.match(/\.\\d/); i++)
        {
            from = from.replace(/\.\\d/, '.(\\d+)'); //Platzhalter für Zahlen
            to = to.replace(/\\d/, '$' + i);
        }
        from = from.replace(/\.([\w\xC0-\xFF(])/g, '.(?: |&nbsp;)*$1'); //Leerzeichen nach Punkt
        t = t.replace(new RegExp(from, 'g'), to);
    }

    /* Keine geschützten Leerzeichen in Überschriften */
    while (t.match(/^=.*&nbsp;.*=$/m))
        t = t.replace(/^(=.*)&nbsp;(.*=)$/gm, '$1 $2');

    var redundantTemplateParameters = window.redundantTemplateParameters || [
        'Infobox Arcade|Titel',
        'Infobox Band|Name',
        'Infobox Berg|NAME',
        'Infobox Computer- und Videospiel|Titel',
        'Infobox Eishockeyspieler|Name',
        'Infobox Fluss|NAME',
        'Infobox Fußballspieler|kurzname',
        'Infobox Gemeinde in Deutschland|Name',
        'Infobox Gemeinde in Italien|nomeComune',
        'Infobox Gemeindeverband in Deutschland|Name',
        'Infobox Landkreis|Name',
        'Infobox Musikalbum|Titel',
        'Infobox Ort in Tschechien|Ort',
        'Infobox Ortsteil einer Gemeinde|Ortsteil',
        '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*') +
            parameter, 'g'), '$1$3');
    }

    /* Spezielle Infobox-Ersetzungen */
    if (t.match(/Infobox[ _](Gemeinde|Ort)[ _]in[ _]Deutschland/i))
    {
        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,
            '|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))
        {
            t = t.replace(/[ _]Gemeinde[ _]in[ _]Deutschland/i, ' Gemeindeverband in Deutschland');
            t = t.replace(/\|\s*(PLZ(-alt)?|Vorwahl|Kfz)\s*=.*[\r\n]+/ig, '');
            t = t.replace(/\|\s*Bürgermeister( {0,5}|titel)?\s*=\s*/ig, '|Organwalter$1  = ');
            if (!t.match(/\|\s*Gliederung\s*=/))
                t = t.replace(/(\|\s*Adresse\s*=)/i, '|Gliederung        = \n$1');
            t = t.replace(/\|\s*Gliederung\s*=\s*(\d*).*/i, '|Gliederung        = $1');
            e['wpSummary'].value += 'Einsatz der neuen [[Vorlage:Infobox Gemeindeverband in Deutschland]]';
            e['wpMinoredit'].checked = true;
        }
        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 =');
    }

    /* Gerettete Dateinamen wieder einsetzen */
    for (i = files.length; i--; )
    {
        t = t.replace(files[i][1] + 'Datei:' + i + files[i][4], files[i][1] + 'Datei:' + files[i][3] + files[i][4]);
    }

    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 */
    {
        var e = $(a && a.nodeType ? a : 'img[rel=autoFormatter]');
        e.css('backgroundColor', changed ? '#DEF740' : '');
        e.css('borderRadius', changed ? '3px' : '');
        e.css('opacity', changed ? '' : '0.4');
    }
    else if (a) a.style.color = changed ? 'green' : 'silver';
    return false;
}

if (typeof $ === 'function' && typeof mw === 'object' && mw.user.options.get('usebetatoolbar')) /* Vector */
{
    mw.loader.using('ext.wikiEditor.toolbar', function() {
        $('#wpTextbox1').wikiEditor('addToToolbar', {
            'section': 'main', /* oder advanced */
            '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 doAutoFormat(this); }
                    }
                }
            }
        });
    });
}
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);
    });