跳转到内容

MediaWiki:Gadget-fixlinkstyle.js

维基百科,自由的百科全书

这是本页的一个历史版本,由Antigng留言 | 贡献2016年10月30日 (日) 04:20 (oldid=41962776)编辑。这可能和当前版本存在着巨大的差异。

(差异) ←上一修订 | 最后版本 (差异) | 下一修订→ (差异)
注意:保存之后,你必须清除浏览器缓存才能看到做出的更改。Google ChromeFirefoxMicrosoft EdgeSafari:按住⇧ Shift键并单击工具栏的“刷新”按钮。参阅Help:绕过浏览器缓存以获取更多帮助。
/**
 * 維基百科{{link style}}修復工具 ver: 0.3
 *
 * Author: [[User:逆襲的天邪鬼]]
 *
 * 該工具會在{{link style}}模板下面顯示個按鈕,或者在「編輯」模式頁面最底下添一個「修復link style」按鈕。
 *
 * 注意事項:
 * 1. 清理的時候一定要檢查一下上下文。
 *
 * 已知bug:
 * 1. == 標題 ==
 *    === [[摘不掉|摘不掉]] ===
 *
 *
 * 更新日誌:
 * 0.1(2016-10-14):修復部分連結有中文但未找到的bug;加入一些細節,例如查看差異、突出顯示文內的跨語言連結。
 * 0.2(2016-10-15):能夠直接修改上下文。
 * 0.3(2016-10-16):修復bug
 *
 * 待添加功能:
 * 1. 奇葩標題檢測:
 *     ==這也是標題<ref>
 *          aaa
 *          </ref>==
 *     ==這還是標題<nowiki>
 *     ==這不是標題==
 *     </nowiki>==
 */

(function() {
    'use strict';

    mw.loader.using('jquery.ui.dialog');

    var UI = {
        ACTION: '處理',
        AND: '和',
        ARTICLENAME: '條目名',
        BUGREPORT: '<a href="https://zh.wikipedia.org/wiki/User_talk:逆襲的天邪鬼" target="_blank" style="display:inline-block;float:right;"><small>提出意見/報告Bug</small></a>',
        CHECKINGTITLE: '正在檢查是否存在同名條目...',
        CHINESE: '中文',
        CHINESEARTICLE: '對應中文條目:',
        CHINESETEST: '檢索條目:',
        CHINESETEST2: '注意同名條目:',
        CLEARBUTTON2: '清理link style',
        CLEARBUTTON: '清理',
        DISPLAYTEXT: '顯示文本',
        DONTMODIFY: '不修改',
        FOUND: '找到$1個跨語言連結',
        GOOGLEIT: 'Google搜索中文名',
        GOOGLELANG: 'zh-TW',        // Google翻譯所用語言,zh-CN/zh-TW
        GOOGLETRANSLATE: 'Google翻譯',
        LANGUAGE: '語言',
        LINK: '連結',
        LOADING: '正在加載...',
        MISSING: '外文維基百科無此條目',
        NETWORKERROR: '網路錯誤',
        NONE: '沒有發現跨語言連結。這通常可能是:<br>① 跨語言連結已被清理,但沒有去掉{{link style}}模板;<br>② 條目中沒有跨語言連結,但條目中所用的模板有。可以通過檢查文章中被高亮顯示的部分來確認;<br><br>',
        ORIGINALNAME: '原名',
        PAGENOTCHANGED: '無法更改頁面',
        PAGENOTFOUND: '頁面未找到',
        REMOVELINK: '去除連結',
        REPLACETO: '替換成',
        REPLACECONTEXT: '檢視/修改上下文',
        SAVING: '正在保存...',
        SHOWDIFF: '查看差異',
        SUBMIT: '直接提交',
        SUCCESS: '成功',
        SUMMARY: '[[User:逆襲的天邪鬼/fixlinkstyle.js|摘掉]]Link Style模板',
        SUMMARY2: '[[User:逆襲的天邪鬼/fixlinkstyle.js|清理]]部分跨語言連結',
        TITLE: '維基百科{{link style}}修復工具',
        TITLELINKS: '$1個標題連結',
        TITLELINKSCLEAN: '<br><br>本文在章節標題處有$1個連結,它們將被無條件清除。',
        TRANSLATING: '正在嘗試尋找對應中文條目...',
        WELCOME: '本工具用於處理跨語言連結和章節標題上的連結,摘掉條目的{{link style}}模板。<span style="color:red;">在清理過程中請認真檢查上下文!</span>$TITLELINKS<br><br>請選擇各跨語言連結的處理方法:',
    };

    // 各種語言
    var LANGUAGES = {
'aa':'aa','ab':'ab','ace':'ace','ady':'ady','af':'af','ak':'ak','als':'als','am':'am','an':'an','ang':'ang','ar':'ar','arc':'arc','arz':'arz','as':'as','ast':'ast','av':'av','ay':'ay','az':'az','azb':'azb','ba':'ba','bar':'bar','bat-smg':'bat-smg','bcl':'bcl','be':'be','be-tarask':'be-x-old','be-x-old':'be-x-old','bg':'bg','bh':'bh','bi':'bi','bjn':'bjn','bm':'bm','bn':'bn','bo':'bo','bpy':'bpy','br':'br','bs':'bs','bug':'bug','bxr':'bxr','ca':'ca','cbk-zam':'cbk-zam','cdo':'cdo','ce':'ce','ceb':'ceb','ch':'ch','cho':'cho','chr':'chr','chy':'chy','ckb':'ckb','co':'co','cr':'cr','crh':'crh','cs':'cs','csb':'csb','cu':'cu','cv':'cv','cy':'cy','da':'da','de':'de','diq':'diq','dsb':'dsb','dv':'dv','dz':'dz','ee':'ee','egl':'eml','el':'el','eml':'eml','en':'en','eo':'eo','es':'es','et':'et','eu':'eu','ext':'ext','fa':'fa','ff':'ff','fi':'fi','fiu-vro':'fiu-vro','fj':'fj','fo':'fo','fr':'fr','frp':'frp','frr':'frr','fur':'fur','fy':'fy','ga':'ga','gag':'gag','gan':'gan','gd':'gd','gl':'gl','glk':'glk','gn':'gn','gom':'gom','got':'got','gsw':'als','gu':'gu','gv':'gv','ha':'ha','hak':'hak','haw':'haw','he':'he','hi':'hi','hif':'hif','ho':'ho','hr':'hr','hsb':'hsb','ht':'ht','hu':'hu','hy':'hy','hz':'hz','ia':'ia','id':'id','ie':'ie','ig':'ig','ii':'ii','ik':'ik','ilo':'ilo','io':'io','is':'is','it':'it','iu':'iu','ja':'ja','jam':'jam','jbo':'jbo','jv':'jv','ka':'ka','kaa':'kaa','kab':'kab','kbd':'kbd','kg':'kg','ki':'ki','kj':'kj','kk':'kk','kl':'kl','km':'km','kn':'kn','ko':'ko','koi':'koi','kr':'kr','krc':'krc','ks':'ks','ksh':'ksh','ku':'ku','kv':'kv','kw':'kw','ky':'ky','la':'la','lad':'lad','lb':'lb','lbe':'lbe','lez':'lez','lg':'lg','li':'li','lij':'lij','lmo':'lmo','ln':'ln','lo':'lo','lrc':'lrc','lt':'lt','ltg':'ltg','lv':'lv','lzh':'zh-classical','mai':'mai','map-bms':'map-bms','mdf':'mdf','mg':'mg','mh':'mh','mhr':'mhr','mi':'mi','min':'min','mk':'mk','ml':'ml','mn':'mn','mo':'mo','mr':'mr','mrj':'mrj','ms':'ms','mt':'mt','mus':'mus','mwl':'mwl','my':'my','myv':'myv','mzn':'mzn','na':'na','nah':'nah','nan':'zh-min-nan','nap':'nap','nb':'no','nds':'nds','nds-nl':'nds-nl','ne':'ne','new':'new','ng':'ng','nl':'nl','nn':'nn','no':'no','nov':'nov','nrm':'nrm','nso':'nso','nv':'nv','ny':'ny','oc':'oc','olo':'olo','om':'om','or':'or','os':'os','pa':'pa','pag':'pag','pam':'pam','pap':'pap','pcd':'pcd','pdc':'pdc','pfl':'pfl','pi':'pi','pih':'pih','pl':'pl','pms':'pms','pnb':'pnb','pnt':'pnt','ps':'ps','pt':'pt','qu':'qu','rm':'rm','rmy':'rmy','rn':'rn','ro':'ro','roa-rup':'roa-rup','roa-tara':'roa-tara','ru':'ru','rue':'rue','rup':'roa-rup','rw':'rw','sa':'sa','sah':'sah','sc':'sc','scn':'scn','sco':'sco','sd':'sd','se':'se','sg':'sg','sgs':'bat-smg','sh':'sh','si':'si','simple':'simple','sk':'sk','sl':'sl','sm':'sm','sn':'sn','so':'so','sq':'sq','sr':'sr','srn':'srn','ss':'ss','st':'st','stq':'stq','su':'su','sv':'sv','sw':'sw','szl':'szl','ta':'ta','tcy':'tcy','te':'te','tet':'tet','tg':'tg','th':'th','ti':'ti','tk':'tk','tl':'tl','tn':'tn','to':'to','tpi':'tpi','tr':'tr','ts':'ts','tt':'tt','tum':'tum','tw':'tw','ty':'ty','tyv':'tyv','udm':'udm','ug':'ug','uk':'uk','ur':'ur','uz':'uz','ve':'ve','vec':'vec','vep':'vep','vi':'vi','vls':'vls','vo':'vo','vro':'fiu-vro','wa':'wa','war':'war','wo':'wo','wuu':'wuu','xal':'xal','xh':'xh','xmf':'xmf','yi':'yi','yo':'yo','yue':'zh-yue','za':'za','zea':'zea','zh':'zh','zh-classical':'zh-classical','zh-cn':'zh','zh-min-nan':'zh-min-nan','zh-tw':'zh','zh-yue':'zh-yue','zu':'zu'
    };

    var wiki = {
        loadPage: function (pageName, success, failure) {
            // 注意:該函數只加載和處理一個頁面……
            $.ajax({
                url: mw.util.wikiScript('api'),
                data: {
                    action: 'query',
                    prop: 'revisions',
                    rvprop: 'content',
                    titles: pageName,
                    redirects: true,
                    format: 'json'
                },
                dataType: 'json',
                type: 'POST',
                success: function (data) {
                    if (data.query) {
                        // var r = {};
                        var pages = data.query.pages;
                        for (var pageid in pages) {
                            var page = pages[pageid];
                            if (!('missing' in page)) {
                                // r[page.title] = page.revisions[0]['*'];
                                if (typeof success === 'function') {
                                    success(page.revisions[0]['*']);
                                    return;
                                }
                            }
                        }
                        /*
                        if (typeof success === 'function') {
                            success(r);
                        }
                        */
                    }

                    if (typeof failure === 'function') {
                        failure(UI.PAGENOTFOUND);
                    }
                },
                error: function (xhr) {
                    if (typeof failure === 'function') {
                        failure(UI.NETWORKERROR);
                    }
                }
            });
        },
        savePage: function (title, content, summary, success, failure) {
            // 保存頁面
            // 該編輯會被標記為小編輯
            $.ajax({
                url: mw.util.wikiScript('api'),
                data: {
                    format: 'json',
                    action: 'edit',
                    title: title,
                    summary: summary,
                    minor: true,
                    text: content,
                    token: mw.user.tokens.get('editToken')
                },
                dataType: 'json',
                type: 'POST',
                success: function (data) {
                    if (data && data.edit && data.edit.result === 'Success') {
                        if (typeof success === 'function') {
                            success();
                        }
                    } else {
                        if (typeof failure === 'function') {
                            failure(UI.PAGENOTCHANGED);
                        }
                    }
                },
                error: function (xhr) {
                    if (typeof failure === 'function') {
                        failure(UI.NETWORKERROR);
                    }
                }
            });
        },
        translateTitle: function (titles, fromLang, toLang, transTable, success, failure) {
            $.ajax({
                url: 'https://' + LANGUAGES[fromLang] + '.wikipedia.org/w/api.php',
                data: {
                    action: 'query',
                    prop: 'langlinks',
                    lllang: toLang,
                    format: 'json',
                    redirects: true,
                    lllimit: 50,
                    titles: titles
                },
                dataType: 'jsonp',
                type: 'POST',
                success: function (data) {
                    if (data.query) {
                        // 獲取原來的標題
                        var orinTitle = {};
                        var convert = function (arr) {
                            if (arr) {
                                for (var i=0; i<arr.length; i++) {
                                    if (orinTitle[arr[i].from]) {
                                        orinTitle[arr[i].to] = orinTitle[arr[i].from];
                                    } else {
                                        orinTitle[arr[i].to] = arr[i].from;
                                    }
                                }
                            }
                        };
                        convert(data.query.normalized);
                        convert(data.query.redirects);

                        // 標明翻譯情況
                        var pages = data.query.pages;
                        for (var pageid in pages) {
                            var page = pages[pageid];
                            var title = page.title;
                            if (orinTitle[title]) {
                                title = orinTitle[title];
                            }

                            if (page.missing !== undefined) {
                                transTable[fromLang + ':' + title] = null;
                            } else if (page.langlinks) {
                                transTable[fromLang + ':' + title] = page.langlinks[0]['*'];
                            }
                        }
                    }
                    if (typeof success === 'function') {
                        success();
                    }
                },
                error: function (xhr) {
                    if (typeof failure === 'function') {
                        failure(UI.NETWORKERROR);
                    }
                }
            });
        },
        checkTitle: function (titles, checkTable, success, failure) {
            $.ajax({
                url: mw.util.wikiScript('api'),
                data: {
                    action: 'query',
                    format: 'json',
                    redirects: true,
                    titles: titles
                },
                dataType: 'json',
                type: 'POST',
                success: function (data) {
                    if (data.query) {
                        // 獲取原來的標題
                        var orinTitle = {};
                        var convert = function (arr) {
                            if (arr) {
                                for (var i=0; i<arr.length; i++) {
                                    if (orinTitle[arr[i].from]) {
                                        orinTitle[arr[i].to] = orinTitle[arr[i].from];
                                    } else {
                                        orinTitle[arr[i].to] = arr[i].from;
                                    }
                                }
                            }
                        };
                        convert(data.query.normalized);
                        convert(data.query.redirects);

                        // 標明翻譯情況
                        var pages = data.query.pages;
                        for (var pageid in pages) {
                            var page = pages[pageid];
                            var title = page.title;
                            if (orinTitle[title]) {
                                title = orinTitle[title];
                            }

                            if (page.pageid) {
                                checkTable[title] = true;
                            }
                        }
                    }
                    if (typeof success === 'function') {
                        success();
                    }
                },
                error: function (xhr) {
                    if (typeof failure === 'function') {
                        failure(UI.NETWORKERROR);
                    }
                }
            });
        }
    };

    var article = {
        title: '',
        content: '',
        interlinks: [],
        _segments: [],
        scaninterlinks: function() {
            this._segments = [];
            this._linkpos = [];
            this.interlinks = [];

            var segments = this._segments;
            var linkpos = this._linkpos;
            var interlinks = this.interlinks;

            var splitLines = function (str) {
                var seq = str.split('\n');

                for (var i=0; i<seq.length-1; i++) {
                    segments.push(seq[i]);
                    segments.push('\n');
                }
                segments.push(seq[seq.length-1]);
            };

            var re = /\[\[(.*?)\]\]/g;
            var extract = /:(.*?):(.*)/;
            var match;
            var text = this.content;
            var lastPos = 0;
            while ((match = re.exec(text)) !== null) {
                // 將outertext以\n劃塊
                splitLines(text.substring(lastPos, re.lastIndex-match[0].length));

                segments.push(match[0]);
                lastPos = re.lastIndex;

                // 分析
                var match2 = extract.exec(match[1]);
                if (match2 !== null && LANGUAGES[match2[1]]) {
                    var link = {
                        type: 'link',
                        lang: match2[1],
                        original: match[0],
                        index: this._segments.length - 1
                    };
                    var sepPos = match2[2].indexOf('|');

                    if (sepPos === -1) {
                        link.target = match2[2];
                        link.text = '';
                    } else {
                        link.target = match2[2].substring(0, sepPos);
                        link.text = match2[2].substring(sepPos+1);
                    }

                    interlinks.push(link);
                }
            }

            splitLines(text.substring(lastPos));

            // 找到各連結的上下文
            for (var i=0; i<interlinks.length; i++) {
                var ilink = interlinks[i];

                // 不能碰到上個link的to
                var pos1 = 0;
                if (i>0) {
                    pos1 = interlinks[i-1].context.to + 1;
                }

                var p = interlinks[i].index;
                while (p >= 0 && segments[p] !== '\n') {
                    p--;
                }
                p++;

                var from = pos1>p ? pos1 : p;

                // 不能碰到下個link
                pos1 = segments.length;
                if (i < interlinks.length-1) {
                    pos1 = interlinks[i+1].index - 1;
                }

                p = interlinks[i].index;
                while (p < segments.length && segments[p] !== '\n') {
                    p++;
                }
                p--;

                var to = pos1<p ? pos1 : p;

                ilink.context = {from: from, to: to};
            }
        },
        getContext: function (linkid) {
            var pos = article.interlinks[linkid].context;
            return article._segments.slice(pos.from, pos.to+1).join('');
        },
        setContext: function (linkid, text) {
            var link = article.interlinks[linkid];
            var pos = link.context;
            link.type = 'plain';
            link.text = text;
            article._segments.fill('', pos.from, pos.to+1);
        },
        toString: function () {
            var segments = this._segments;
            var links = this.interlinks;

            for (var i=0; i<links.length; i++) {
                var r;
                var link = links[i];

                switch (link.type) {
                    case 'link':
                        r = ['[['];
                        if (link.lang && link.lang.length > 0) {
                            r.push(':', link.lang, ':');
                        }
                        r.push(link.target);
                        if (link.text && link.text.length > 0) {
                            if (link.target && link.target.length > 0) {
                                r.push('|');
                            }
                            r.push(link.text);
                        }
                        r.push(']]');
                        break;
                    case 'tsl':
                        r = ['{{tsl|', link.lang, '|', link.source, '|', link.target];
                        if (link.text && link.text.length > 0) {
                            r.push('|', link.text);
                        }
                        r.push('}}');
                        break;
                    case 'ilh':
                        r = ['{{link-', link.lang, '|', link.target, '|', link.source];
                        if (link.text && link.text.length > 0) {
                            r.push('|', link.text);
                        }
                        r.push('}}');
                        break;
                    case 'plain':
                        r = [link.text];
                        break;
                }

                segments[link.index] = r.join('');
            }

            return segments.join('');
        },
        update: function () {
            this.content = this.toString();
        },
        translateTitle: function (callback) {
            var transTable = {};
            var counter = 0;
            var queue = {};
            var queue2 = [];
            var i;
            var links = article.interlinks;

            // 根據語言整理隊列
            for (i = 0; i<links.length; i++) {
                var link = links[i];
                if (!queue[link.lang]) {
                    queue[link.lang] = [];
                }
                queue[link.lang].push(link.target);
            }
            // 以50為一組進行分割
            // 計算需要執行的次數,次數為0之後整理結果,並執行callback
            for (var lang in queue) {
                var len = queue[lang].length;
                var l2 = parseInt((len+50)/50);
                counter += l2;

                for (var j = 0; j<l2; j++) {
                    queue2.push({
                        lang: lang,
                        titles: queue[lang].slice(50*j, 50*(j+1)).join('|')
                    });
                }
            }

            var stop = false;
            var tide = function () {
                for (i=0; i<links.length; i++) {
                    var link = links[i];
                    var key = link.lang + ':' + link.target;
                    if (transTable[key] === null) {
                        link.missing = true;
                    } else if (transTable[key]) {
                        link.translation = transTable[key];
                    }
                }
            };
            var proc = function () {
                counter--;
                if (counter <= 0) {
                    tide();

                    if (typeof callback === 'function') {
                        callback();
                    }
                }
            };
            var error = function () {
                stop = true;
                tide();
                if (typeof callback === 'function') {
                    callback();
                }
            };

            if (queue2.length > 0) {
                for (i = 0; i<queue2.length; i++) {
                    if (stop) {
                        break;
                    }
                    wiki.translateTitle(queue2[i].titles, queue2[i].lang, 'zh', transTable, proc, error);
                }
            } else {
                proc();
            }
        },
        titleCheckTable: {},
        checkTitle: function (callback) {
            article.titleCheckTable = {};
            var counter = 0;
            var queue = {};
            var queue2 = [];
            var i;
            var links = article.interlinks;

            // 整理隊列
            for (i = 0; i<links.length; i++) {
                var link = links[i];
                queue[link.target] = true;
                queue[link.text] = true;
            }
            queue[''] = undefined;

            for (var title in queue) {
                queue2.push(title);
            }

            queue = [];
            counter = parseInt((queue2.length+50)/50);

            for (i=0; i<counter; i++) {
                queue.push(queue2.slice(50*i, 50*(i+1)).join('|'));
            }

            var stop = false;
            var proc = function () {
                counter--;
                if (counter <= 0) {
                    if (typeof callback === 'function') {
                        callback();
                    }
                }
            };
            var error = function () {
                stop = true;
                if (typeof callback === 'function') {
                    callback();
                }
            };

            if (queue.length > 0) {
                for (i = 0; i<queue.length; i++) {
                    if (stop) {
                        break;
                    }
                    wiki.checkTitle(queue[i], article.titleCheckTable, proc, error);
                }
            } else {
                if (typeof callback === 'function') {
                    callback();
                }
            }
        },
        titlelinks: 0,
        removeTitleLinks: function () {
            article.titlelinks = 0;
            article.content = article.content.replace(/\n *={2,5} *.*? *={2,5} *\n/g, function (str) {
                return str.replace(/\[\[.*?\]\]/g, function (link) {
                    article.titlelinks++;
                    return link.replace(/\[\[(.*?)\]\]/, '$1').replace(/:.*?:(.*)/, '$1').replace(/.*\|(.*)/, '$1');
                });
            });
        }
    };

    var dialog = {
        createRadio: function (name, value, checked) {
            var $e = $('<input>')
                .attr('id', name + '_' + value)
                .attr('name', name)
                .attr('value', value)
                .attr('type', 'radio');
            if (checked) {
                $e.attr('checked', 'checked');
            }
            return $e;
        },
        createTextBox: function (name, value, placeholder, size) {
            return $('<input>')
                .attr('id', name)
                .attr('name', name)
                .attr('value', value)
                .attr('size', size ? size : '15')
                .attr('placeholder', placeholder ? placeholder : '');
        },
        createLink: function (text, href) {
            return $('<a>')
                .attr('href', href)
                .attr('target', '_blank')
                .html(text);
        },
        createRow: function (link, index) {
            var $row = $('<tr>');
            var $tmp;
            var translation = '';
            var name = 'ilf_' + index;
            $row.append('<td>'+index+'</td>');

            $tmp = $('<td>').append(
                dialog.createLink(link.original, 'https://' + LANGUAGES[link.lang] + '.wikipedia.org/wiki/' + link.target)
            );
            if (link.missing) {
                $tmp.append('<br><br><small style="color:red;">' + UI.MISSING + '</small>');
            }
            if (link.translation) {
                translation = link.translation;
                $tmp.append('<br><br>', $('<small style="color:green;">').append(
                                UI.CHINESEARTICLE,
                                dialog.createLink(link.translation, 'https://zh.wikipedia.org/wiki/' + link.translation)
                            ));
            } else {
                if (article.titleCheckTable[link.target]) {
                    $tmp.append('<br><br>', $('<small style="color:red;">').append(
                                    UI.CHINESETEST2,
                                    dialog.createLink(link.target, 'https://zh.wikipedia.org/wiki/' + link.target)
                                ));
                } else {
                    $tmp.append('<br><br>', $('<small>').append(
                                    UI.CHINESETEST,
                                    dialog.createLink(link.target, 'https://zh.wikipedia.org/w/index.php?search=' + link.target)
                                ));
                }

                if (link.text && link.text !== link.target) {
                    if (article.titleCheckTable[link.text]) {
                        $tmp.append('<br>', $('<small style="color:red;">').append(
                                        UI.CHINESETEST2,
                                        dialog.createLink(link.text, 'https://zh.wikipedia.org/wiki/' + link.text)
                                    ));
                    } else {
                        $tmp.append('<br>', $('<small>').append(
                                        UI.CHINESETEST,
                                        dialog.createLink(link.text, 'https://zh.wikipedia.org/w/index.php?search=' + link.text)
                                    ));
                    }
                }
            }
            $tmp.append('<br><br>',
                        $('<small>').append(dialog.createLink(UI.GOOGLEIT, 'https://www.google.com/search?q=' + encodeURIComponent(link.target + ' ' + UI.CHINESE))),
                        '<br>',
                        $('<small>').append(dialog.createLink(UI.GOOGLETRANSLATE, 'https://translate.google.com/#auto/' + UI.GOOGLELANG + '/' + encodeURIComponent(link.target)))
                       );
            $row.append($tmp);

            $tmp = $('<td>').append(
                dialog.createRadio(name, '0', true),
                    '<label for="' + name + '_0">' + UI.DONTMODIFY + '</label>',
                    '<br>',
                dialog.createRadio(name, '1'),
                    '<label for="' + name + '_1">' + UI.REMOVELINK + '</label>',
                    '<br>',
                dialog.createRadio(name, '2'),
                    '[[',
                    dialog.createTextBox(name+'_21', translation, UI.ARTICLENAME),
                    '|',
                    dialog.createTextBox(name+'_22', link.text === translation ? '' : link.text, UI.DISPLAYTEXT),
                    ']]',
                    '<br>',
                dialog.createRadio(name, '3'),
                    '{{tsl|',
                    dialog.createTextBox(name+'_31', link.lang, UI.LANGUAGE, '5'),
                    '|',
                    dialog.createTextBox(name+'_32', link.target, UI.ORIGINALNAME),
                    '|',
                    dialog.createTextBox(name+'_33', translation.length>0 ? translation : link.text, UI.ARTICLENAME),
                    '|',
                    dialog.createTextBox(name+'_34', (translation.length===0 || link.text===translation) ? '' : link.text, UI.DISPLAYTEXT),
                    '}}',
                    '<br>',
                dialog.createRadio(name, '4'),
                    '{{link-',
                    dialog.createTextBox(name+'_41', link.lang, UI.LANGUAGE, '5'),
                    '|',
                    dialog.createTextBox(name+'_42', translation.length>0 ? translation : link.text, UI.ARTICLENAME),
                    '|',
                    dialog.createTextBox(name+'_43', link.target, UI.ORIGINALNAME),
                    '|',
                    dialog.createTextBox(name+'_44', (translation.length===0 || link.text===translation) ? '' : link.text, UI.DISPLAYTEXT),
                    '}}',
                    '<br>',
                dialog.createRadio(name, '5'),
                    UI.REPLACETO + ': ',
                    dialog.createTextBox(name+'_51', ''),
                    '<br>',
                dialog.createRadio(name, '6'),
                    '<label for="' + name + '_6">' + UI.REPLACECONTEXT + '</label>'
            );

            $row.append($tmp);

            return $row;
        },
        createRow2: function (link, index) {
            var name = 'ilf_' + index;
            var $row = $('<tr>')
                            .attr('id', name + '_6_row')
                            .css('display', 'none');
            $row.append('<td>');

            var $tmp = $('<td>').attr('colspan', '2');
            $tmp.append($('<textarea>')
                            .attr('id', name + '_6_t')
                            .attr('rows', '3')
                            .val(article.getContext(index))
                       );

            $row.append($tmp);

            return $row;
        },
        handlers: {
            api: {
                name: 'api',
                load: function (complete, error) {
                    wiki.loadPage(mw.config.get('wgPageName'), complete, error);
                },
                save: function (summary, complete, error) {
                    wiki.savePage(article.title, article.content, summary, complete, error);
                },
                diff: function (summary) {
                    $('<form method="post">')
                        .attr('action', mw.util.getUrl(mw.config.get('wgPageName'), {action: 'submit'}))
                        .attr('enctype', 'multipart/form-data')
                        .append($('<textarea name="wpTextbox1">').val(article.content),
                                $('<input name="wpSummary">').val(summary),
                                '<input name="wpMinoredit" type="checkbox" value="1" checked="checked">',
                                '<input name="format" value="text/x-wiki">',
                                '<input name="model" value="wikitext">',
                                '<input name="wpEditToken" value="' + mw.user.tokens.get('editToken') + '">',
                                '<input name="wpUltimateParam" value="1">',
                                '<input name="wpDiff" value="aa">',
                                '<input name="mode" value="text">')
                        .submit();
                },
                end: function () {
                    setTimeout(function () {
                        history.go(0);
                    }, 500);
                }
            },
            text: {
                name: 'text',
                load: function (complete) {
                    if (typeof complete === 'function') {
                        complete($('#wpTextbox1').val());
                    }
                },
                save: function (summary, complete) {
                    $('#wpTextbox1').val(article.content);
                    $('#wpSummary').val(summary);
                    $('#wpMinoredit').attr('checked', 'checked');

                    if (typeof complete === 'function') {
                        complete();
                    }
                },
                diff: function (summary, complete) {
                    $('#wpTextbox1').val(article.content);
                    $('#wpSummary').val(summary);
                    $('#wpMinoredit').attr('checked', 'checked');

                    $('#wpDiff').click();

                    if (typeof complete === 'function') {
                        complete();
                    }
                },
                end: function () {
                    setTimeout(function () {
                        $('#interlinkfix_dialog').dialog('close');
                        $('#wpTextbox1').focus();
                    }, 500);
                }
            }
        },
        handler: {},
        links: {
            list: null,
            border: false,
            blink: function () {
                var i;
                var links = dialog.links.list;
                dialog.links.border = !dialog.links.border;
                if (dialog.links.border) {
                    for (i=0; i<links.length; i++) {
                        links[i].style.border = '1px solid red';
                    }
                } else {
                    for (i=0; i<links.length; i++) {
                        links[i].style.border = 'none';
                    }
                }
            },
            get: function () {
                dialog.links.list = $('.extiw', '#mw-content-text').filter(function () {
                    return this.parentNode.className.indexOf('ilh-link') === -1 && this.href && this.href.indexOf('.wikipedia.org') > -1;
                });
            },
            highlight: function (value) {
                if (value && dialog.links.timerId === null) {
                    dialog.links.timerId = setInterval(dialog.links.blink, 1000);
                } else if (!value) {
                    dialog.links.border = true;
                    dialog.links.blink();
                    clearInterval(dialog.links.timerId);
                    dialog.links.timerId = null;
                }
            },
            timerId: null
        },
        init: function () {
            if (document.getElementById('interlinkfix_dialog') === null) {
                $('body').append('<div id="interlinkfix_dialog" style="display:none;" title="' + UI.TITLE + '"></div>');
            }

            dialog.links.get();
            dialog.links.highlight(true);
            $('#interlinkfix_dialog').html(UI.LOADING);

            $('#interlinkfix_dialog').dialog({
                modal: false,
                close: function() {
                    dialog.links.highlight(false);
                },
                draggable: true,
                width: 700,
                height: 500
            });

            article.title = mw.config.get('wgPageName');
            dialog.handler.load(function (content) {
                article.content = content;

                article.removeTitleLinks();

                article.scaninterlinks();
                $('#interlinkfix_dialog').append('<br>', UI.FOUND.replace('$1', article.interlinks.length), '<br>', UI.TRANSLATING);

                article.translateTitle(function () {
                    $('#interlinkfix_dialog').append('<br>', UI.CHECKINGTITLE);
                    article.checkTitle(function () {
                        dialog.putlinks();
                    });
                });
            }, function (e) {
                $('#interlinkfix_dialog').append('<br><span style="color:red;">' + e + '</span>');
            });
        },
        putlinks: function () {
            var $table = $('<table class="wikitable">').append('<tr><th>#</th><th>'+UI.LINK+'</th><th>'+UI.ACTION+'</th></tr>');

            for (var i=0; i<article.interlinks.length; i++) {
                $table.append(dialog.createRow(article.interlinks[i], i));
                $table.append(dialog.createRow2(article.interlinks[i], i));
            }

            var titlelinkstext = (article.titlelinks > 0) ? UI.TITLELINKSCLEAN.replace('$1', article.titlelinks) : '';
            $('#interlinkfix_dialog').html(UI.WELCOME.replace('$TITLELINKS', titlelinkstext));

            if (article.interlinks.length === 0) {
                $('#interlinkfix_dialog').append('<br><br>', UI.NONE);
            } else {
                $('#interlinkfix_dialog').append($('<form id="ilf_frm">').append($table));
            }

            $('#interlinkfix_dialog').append(
                '<br>',
                '<a class="mw-ui-button mw-ui-progressive" href="#" id="interlinkfix_diff">' + UI.SHOWDIFF + '</a>&nbsp;',
                '<a class="mw-ui-button" href="#" id="interlinkfix_submit">' + UI.SUBMIT + '</a>',
                UI.BUGREPORT
            );

            $('input:text', '#ilf_frm').click(function () {
                var name = this.name;
                var i = name.lastIndexOf('_');
                var radioname = name.slice(0, i);
                var key = name.charAt(i+1);
                document.forms.ilf_frm[radioname].value = key;
                $('#' + radioname + '_6_row').hide();
            });

            $('input:radio', '#ilf_frm').click(function () {
                var name = this.name;
                var value = this.value;
                if (value === '6') {
                    $('#' + name + '_6_row').show();
                    $('#' + name + '_6_t').focus();
                } else {
                    $('#' + name + '_6_row').hide();
                }
            });

            $('#interlinkfix_diff').unbind().click(function () {
                dialog.check(dialog.diff);
            });

            $('#interlinkfix_submit').unbind().click(function () {
                dialog.check(dialog.submit);
            });
        },
        check: function (complete) {
            var frm = document.forms.ilf_frm;
            var links = article.interlinks;

            for (var i=0; i<links.length; i++) {
                var link = links[i];
                switch (frm['ilf_' + i].value) {
                    case '0':
                        break;
                    case '1':
                        link.type = 'plain';
                        if ((!link.text) || (link.text === '')) {
                            link.text = link.target;
                        }
                        break;
                    case '2':
                        link.type = 'link';
                        link.lang = '';
                        link.target = frm['ilf_' + i + '_21'].value;
                        link.text = frm['ilf_' + i + '_22'].value;
                        break;
                    case '3':
                        link.type = 'tsl';
                        link.lang = frm['ilf_' + i + '_31'].value;
                        link.source = frm['ilf_' + i + '_32'].value;
                        link.target = frm['ilf_' + i + '_33'].value;
                        link.text = frm['ilf_' + i + '_34'].value;
                        break;
                    case '4':
                        link.type = 'ilh';
                        link.lang = frm['ilf_' + i + '_41'].value;
                        link.target = frm['ilf_' + i + '_42'].value;
                        link.source = frm['ilf_' + i + '_43'].value;
                        link.text = frm['ilf_' + i + '_44'].value;
                        break;
                    case '5':
                        link.type = 'plain';
                        link.text = frm['ilf_' + i + '_51'].value;
                        break;
                    case '6':
                        article.setContext(i, $('#ilf_' + i + '_6_t').val());
                        break;
                }
            }

            article.update();

            // 檢查是否仍有跨語言連結,如果沒有,摘掉link style模板
            var summary = UI.SUMMARY2;
            var linkstyleTemplate = /{{\s*link style.*?}}\s*/i;

            article.scaninterlinks();
            if (article.interlinks.length === 0 && linkstyleTemplate.test(article.content)) {
                article.content = article.content.replace(linkstyleTemplate, '');
                summary = UI.SUMMARY;
            }

            if (article.titlelinks > 0) {
                summary = summary + UI.AND + UI.TITLELINKS.replace('$1', article.titlelinks);
            }

            if (typeof complete === 'function') {
                complete(summary);
            }
        },
        submit: function (summary) {
            $('#interlinkfix_dialog').html(UI.SAVING);
            dialog.handler.save(summary, function () {
                $('#interlinkfix_dialog').append('<br><span style="color:green;">'+UI.SUCCESS+'</span>');
                dialog.handler.end();
            }, function (e) {
                $('#interlinkfix_dialog').append('<br><span style="color:red;">'+e+'</span>');
            });
        },
        diff: function (summary) {
            $('#interlinkfix_dialog').html(UI.SAVING);
            dialog.handler.diff(summary, function () {
                $('#interlinkfix_dialog').dialog('close');
            });
        }
    };

    var wgaction = mw.config.get('wgAction');

    var addfixbutton = function () {
        var $tmp = $('.mbox-text-span', '.ambox-link-style');
        switch (wgaction) {
            case 'view':
                $('#fixinterlink').remove();

                dialog.handler = dialog.handlers.api;

                if ($tmp.length > 0) {
                    $tmp.append('<center><a id="fixinterlink" href="#"><span class="mw-ui-button mw-ui-progressive">' + UI.CLEARBUTTON + '</span></a></center>');
                } else {
                    $('#firstHeading').append('<span style="font-size:small;margin-left:1em;" id="fixinterlink">[<a href="#">' + UI.CLEARBUTTON2 + '</a>]</scan>');
                }

                $('#fixinterlink').click(function () {
                    dialog.init();
                });
                break;

            case 'submit':
            case 'edit':
                $('#fixinterlink').remove();
                $('#fixinterlink_button').remove();

                dialog.handler = dialog.handlers.text;

                $('.mbox-text-span', '.ambox-link-style').append('<center><a id="fixinterlink" href="#"><span class="mw-ui-button mw-ui-progressive">' + UI.CLEARBUTTON + '</span></a></center>');
                $('#fixinterlink').click(function () {
                    dialog.init();
                });

                $('#wpDiff').after('<input id="fixinterlink_button" value="' + UI.CLEARBUTTON2 + '" type="button">');
                $('#fixinterlink_button').click(function () {
                    dialog.init();
                });
            break;
        }
    };

    if (mw.config.get('wgCanonicalNamespace') === '') {
        switch (wgaction) {
            case 'view':
                var linklist = $('.extiw', '#mw-content-text').filter(function () {
                    return this.parentNode.className.indexOf('ilh-link') === -1 && this.href && this.href.indexOf('.wikipedia.org') > -1;
                });

                if (linklist.length > 0) {
                    addfixbutton();
                }
                break;

            case 'submit':
            case 'edit':
                article.content = $('#wpTextbox1').val();
                article.scaninterlinks();
                if (article.interlinks.length > 0) {
                    addfixbutton();
                }
                break;
        }
    }

    window.fixlinkstyle = addfixbutton;
})();