MediaWiki:Mobile.js

H萌娘,万物皆可H的百科全书!
imported>=海豚=2020年3月11日 (三) 13:32的版本
跳到导航 跳到搜索

注意:这类代码页面在保存之后,您可能需要清除浏览器缓存才能看到所作出的变更的影响。

  • 按住CTRL+SHIFT+DEL 或 ⌘-Shift-R来清除缓存!
  • 或尝试在地址栏的地址最后添加代码?_=1来访问最新页面。
    添加代码后的本页地址如下:-{R|https://hmoegirl.com/MediaWiki:Mobile.js?_=1}-
  • 你还可以在设置中勾选小工具在页面右上角添加清除缓存按钮!
1. 桌面版CSS MediaWiki:Common.css
2. 桌面版JS MediaWiki:Common.js
3. 移动端CSS MediaWiki:Mobile.css
4. 移动端JS MediaWiki:Mobile.js
5. 全局共用CSS MediaWiki:Gadget-site-styles.css
6. 全局共用JS MediaWiki:Gadget-site-javascript.js
7. 管理员CSS MediaWiki:Group-sysop.css
8. 管理员JS MediaWiki:Group-sysop.js
9. 编辑时载入 MediaWiki:Common.js/edit.js
引用api时用mw.loader.using("mw.Api"),引用uri时用mw.loader.using("mediawiki.Uri")
// <pre>
/* 这里的任何JavaScript将只在移动端加载 
 * 请尊重萌娘百科版权,以下代码复制需要注明原自萌娘百科,并且附上URL地址http://zh.moegirl.org/MediaWiki:Common.js
 * 版权协定:知识共享 署名-非商业性使用-相同方式共享 3.0
 */
/* 维护者请注意,由于可视化编辑器已不起作用,电脑端可以直接跳转到源代码编辑,可移动端无法跳转,失去了编辑的功能,所以默认在移动端启用了InPageEdit小工具,具体使用方法见帮助。----2020年3月11日留。 */
(function ($, mw) { //使用匿名函数以避免污染顶级变量
    /* 函数定义体 */
    //Tabs
    function tabs() {
        var defaultStyle = {
            purple: {
                labelColor: " ", //anti check
                labelBackgroundColor: "#9070c0",
                labelBorderColor: "#b090e0 #7050a0 #9070c0 #b090e0",
                labelPadding: ".2em .3em .2em .3em",
                textBorderColor: "#9070c0",
                textBackgroundColor: "#f0edf5",
                textPadding: "1em",
            },
            green: {
                labelColor: " ",
                labelBackgroundColor: "#75c045",
                labelBorderColor: "#90d060 #60b030 #75c045 #90d060",
                labelPadding: ".2em .3em .2em .3em",
                textBorderColor: "#75c045 #60b030 #60b030 #75c045",
                textBackgroundColor: "#f5fffa",
                textPadding: "1em",
            },
            red: {
                labelColor: " ",
                labelBackgroundColor: "#FF0000",
                labelBorderColor: "#FF8888 #CC0000 #FF0000 #FF8888",
                labelPadding: ".2em .3em .2em .3em",
                textBorderColor: "#FF0000 #CC0000 #CC0000 #FF0000",
                textBackgroundColor: "#fffafa",
                textPadding: "1em",
            },
            blue: {
                labelColor: " ",
                labelBackgroundColor: "#5b8dd6",
                labelBorderColor: "#88abde #3379de #5b8dd6 #88abde",
                labelPadding: ".2em .3em .2em .3em",
                textBackgroundColor: "#f0f8ff",
                textBorderColor: "#5b8dd6 #3379de #3379de #5b8dd6",
                textPadding: "1em",
            },
            yellow: {
                labelColor: " ",
                labelBackgroundColor: "#ffe147",
                labelBorderColor: "#ffe977 #ffd813 #ffe147 #ffe977",
                labelPadding: ".2em .3em .2em .3em",
                textBackgroundColor: "#fffce8",
                textBorderColor: "#ffe147 #ffd813 #ffd813 #ffe147",
                textPadding: "1em",
            },
            orange: {
                labelColor: " ",
                labelBackgroundColor: "#ff9d42",
                labelBorderColor: "#ffac5d #ff820e #ff9d42 #ffac5d",
                labelPadding: ".2em .3em .2em .3em",
                textBackgroundColor: "#ffeedd",
                textBorderColor: "#ff9d42 #ff820e #ff820e #ff9d42",
                textPadding: "1em",
            },
            black: {
                labelColor: " ",
                labelBackgroundColor: "#7f7f7f",
                labelBorderColor: "#999999 #4c4c4c #7f7f7f #999999",
                labelPadding: ".2em .3em .2em .3em",
                textBackgroundColor: "#e5e5e5",
                textBorderColor: "#7f7f7f #4c4c4c #4c4c4c #7f7f7f",
                textPadding: "1em",
            },
        };
        $("body").addClass("tab");
        // A Class
        function StyleSheet() {}
        StyleSheet.prototype.getOwnPropertyNamesLength = function getOwnPropertyNamesLength() {
            return Object.getOwnPropertyNames(this).length;
        };
        String.prototype.toLowerFirstCase = function toLowerFirstCase() {
            return this[0].toLowerCase() + this.substring(1);
        };
        $(".Tabs").each(function() {
            if ($(this).children(".TabLabel")[0]) { return true; }
            var self = $(this),
                classList = Array.from(this.classList).filter(function(n) { return n in defaultStyle; }),
                data = $.extend({
                    labelPadding: null,
                    labelBorderColor: null,
                    labelColor: null,
                    labelBackgroundColor: $("#content").css("background-color"),
                    textPadding: null,
                    textBorderColor: null,
                    textBackgroundColor: null,
                    defaultTab: 1,
                }, classList[0] ? defaultStyle[classList[0]] || {} : {}, this.dataset || {}),
                tabLabel = self.append('<div class="TabLabel"></div>').children(".TabLabel"),
                tabContent = self.append('<div class="TabContent"></div>').children(".TabContent"),
                labelPadding = data.labelPadding,
                labelColor = data.labelColor,
                styleSheet = {
                    label: new StyleSheet(),
                    text: new StyleSheet(),
                },
                defaultTab = parseInt(data.defaultTab);
            self.children(".Tab").each(function() {
                if ($(this).children(".TabLabelText").text().replace(/\s/g, "").length || $(this).children(".TabLabelText").children().length) {
                    $(this).children(".TabLabelText").appendTo(tabLabel);
                    $(this).children(".TabContentText").appendTo(self.children(".TabContent"));
                }
                $(this).remove();
            });
            if (isNaN(defaultTab) || defaultTab <= 0 || defaultTab > tabLabel.children(".TabLabelText").length) { defaultTab = 1; }
            tabLabel.children(".TabLabelText").on("click", function() {
                var label = $(this);
                label.addClass("selected").siblings().removeClass("selected").css({
                    "border-color": "transparent",
                    "background-color": "inherit",
                });
                tabContent.children(".TabContentText").eq(tabLabel.children(".TabLabelText").index(label)).addClass("selected").siblings().removeClass("selected").removeAttr("style");
                if (styleSheet.label.getOwnPropertyNamesLength()) { label.css(styleSheet.label); }
            }).eq(defaultTab - 1).click();
            if (labelPadding) { tabLabel.children(".TabLabelText").css("padding", labelPadding); }
            ["labelBorderColor", "labelBackgroundColor", "textPadding", "textBorderColor", "textBackgroundColor"].forEach(function(n) {
                var target = /^label/.test(n) ? "label" : "text",
                    key = n.replace(target, "").toLowerFirstCase();
                styleSheet[target][key] = data[n];
            });
            if (labelColor) {
                styleSheet.label.borderTopColor = labelColor;
 
            } else if (styleSheet.label.borderColor) {
                styleSheet.label.borderTopColor = "green";
            }
            tabLabel.find(".selected").click();
            if (styleSheet.text.getOwnPropertyNamesLength()) { tabContent.css(styleSheet.text); }
            if (data.autoWidth === "yes") { self.css("display", "inline-block"); }
        });
    }
    //用户资料页相关
    function isUserProfile() {
        return mw.config.get('wgCanonicalSpecialPageName') === "UserProfile" && mw.config.get('wgArticleId') === "0";
    }
    //页顶提示模板相关
    function commonBoxs() {
        if (window.mw && !!mw.config.get('wgNamespaceNumber')) return;
        var contentParent = $('#mw-content-text')[0] ? $('#mw-content-text') : $('#content'),
            commonBoxes = contentParent.find('.common-box');
        if (!commonBoxes[0]) return;
        var commonBoxContainer = $('<div id="commonBoxContainer"><div id="commonBoxInfo"></div></div>').prependTo(contentParent),
            commonBoxList = $('<div id="commonBoxList"></div>').appendTo('#commonBoxInfo');
        commonBoxes.each(function () {
            var commonBoxButton = $('<div class="commonBoxButton"></div>').appendTo(commonBoxList),
                commonBox = $(this);
            commonBoxButton.data('element', commonBox).css({
                'border-color': commonBox.css('border-left-color'),
                'background-image': 'url(' + commonBox.find('tbody > tr > td:first-child img').prop('src') + ')',
            }).on('click', function () {
                if (commonBox.is(':visible')[0]) {
                    commonBoxes.hide();
                    $(this).add($(this).siblings()).removeClass('current');
                    commonBoxList.removeClass('open');
                } else {
                    commonBoxes.filter(':visible').not(commonBox).hide();
                    commonBox.show();
                    $(this).toggleClass('current', commonBox.is(':visible')).siblings().removeClass('current');
                    commonBoxList.toggleClass('open', commonBox.is(':visible'));
                }
            });
        }).appendTo(commonBoxContainer).hide();
    }
        //页顶提示模板相关
        commonBoxs();
        //用户资料页
        if (isUserProfile()) {
            var cardContainer = $('.card-container'),
                containerImage = cardContainer.find('.card'),
                containerImageFile = containerImage.find('a.image'),
                containerImageCaption = containerImage.find('.caption');
            containerImageCaption.css("padding", "0 8px");
            containerImageFile.before('<div id="#containerImage" class="listThumb list-thumb-placeholder" style="text-align: center;"><img src="http://static.mengniang.org/common/a/a4/Placeholder-upload.png" style="height:32px" /></div>');
        }
        //黑幕
        $('.heimu a').on("click", function () {
            if (!$(this).closest('.heimu').is(':active, :focus')) return false;
        });
        //Template:hide
        if ($('.mw-collapsible')[0]) mw.loader.using('jquery.makeCollapsible').then(function () {
            //console.debug('jquery.makeCollapsible is loaded.');
            $('.mw-collapsible').makeCollapsible();
        });
        //链接提示
        linkConfirm();

        //防滥用即将删除
        if (!mw.config.get('wgUserGroups').includes('sysop') && !mw.config.get('wgUserGroups').includes('patroller')) {
            setInterval(function () {
                if ($('#wikitext-editor')[0]) {
                    var target = $('#wikitext-editor');
                    if (target.data('isTrusted') !== true) {
                        target.data('isTrusted', true);
                        $('.editor-overlay').css('z-index', 3);
                        antiAbuse(target[0], $('.editor-overlay .header .header-action button.continue')[0], undefined);
                    }
                }
            }, 1000);
        }
    });
})(jQuery, mediaWiki); //立即执行匿名函数并传递原始变量
//移动端替代编辑器
/**
 *『Wjghj Project Static』
 * This _JavaScript_ code is from https://common.wjghj.cn
 * GNU GENERAL PUBLIC LICENSE 3.0
 *
 * MediaWiki JS Plugin: In Page Edit
 * Version: See version-info file
 * Author: 机智的小鱼君
 * Url:
 ** https://github.com/Dragon-Fish/wjghj-wiki/edit/master/Gadgets/in-page-edit
 ** https://common.wjghj.cn/wiki/InPageEdit-v2
 * Logs:
 ** https://github.com/Dragon-Fish/wjghj-wiki/blob/master/Gadgets/in-page-edit/version-info.md
 ** https://common.wjghj.cn/wiki/InPageEdit-v2/version-info
 **/

(function () {
  'use strict';
  if (typeof InPageEdit !== 'undefined') throw '[InPageEdit] 已经有一个IPE插件在执行了';

  // 创建全局函数
  window.InPageEdit = {};

  /** 导入模态框插件 **/
  mw.loader.load('https://cdn.bootcss.com/ssi-modal/1.0.27/js/ssi-modal.min.js');
  $('title').after('<link id="ssi-modal-style" rel="stylesheet" href="https://cdn.bootcss.com/ssi-modal/1.0.27/styles/ssi-modal.min.css"/>');

  /** 获取版本信息 **/
  mw.loader.load('https://common.wjghj.cn/js/InPageEdit-v2.js/version-info');

  /** 样式表 **/
  // 皮肤
  $('link#ssi-modal-style').after('<link rel="stylesheet" href="https://common.wjghj.cn/css/InPageEdit-v2"/>');
  // Material icons
  mw.loader.load('https://cdn.bootcss.com/material-design-icons/3.0.1/iconfont/material-icons.min.css', 'text/css');

  /*** BOT FLAG ***/
  /** InPageEdit主框架 **/
  /** 快速编辑模块 **/
  InPageEdit.edit = function (option) {
    // 变量
    if (option === undefined)
      option = {};
    var preference = JSON.parse(localStorage.getItem('InPageEditPreference'));
    var editPage = decodeURIComponent(option.page),
      editSection = option.section,
      titleSection = '',
      editRevision = option.revision,
      titleRevision = '',
      summaryRevision = '',
      editText,
      editSummary = preference.editSummary,
      editMinor = preference.editMinor,
      editNotice = '',
      outSideClose = preference.outSideClose,
      jsonGet = {
        action: 'parse',
        page: editPage,
        prop: 'wikitext',
        format: 'json'
      },
      jsonGetInfo = {
        action: 'query',
        titles: editPage,
        prop: 'revisions|info',
        inprop: 'timestamp|protection',
        format: 'json'
      },
      jsonPost = {},
      protection = '',
      basetimestamp,
      date = new Date(),
      timestamp = date.getTime(),
      now = date.toUTCString(); // 缓存时间戳

    InPageEdit.analysis({ type: 'functionCount', function: '快速编辑' });
    InPageEdit.analysis({ type: 'siteCount' });
    InPageEdit.analysis({ type: 'dateCount' });

    if (editPage === undefined) editPage = mw.config.get('wgPageName');
    if (editRevision !== undefined && editRevision !== '' && editRevision !== mw.config.get('wgCurRevisionId')) {
      ssi_modal.notify('warning', {
        className: 'in-page-edit',
        content: '您正在编辑页面的历史版本。',
        title: '提示'
      });
      delete jsonGet.page;
      jsonGet.oldid = editRevision;
      titleRevision = '<span style="font-size:small;">(历史版本:' + editRevision + ')</span>';
      summaryRevision = '(编辑自[[Special:Diff/' + editRevision + ']])';
    } else {
      if (editSection !== undefined && editSection !== '' && editSection !== null) {
        jsonGet.section = editSection;
        titleSection = '(第' + editSection + '部分)';
      }
    }
    if (typeof (MyInPageEditPreference) !== 'undefined') {
      if (typeof (MyInPageEditPreference.editSummary) === 'string')
        editSummary = MyInPageEditPreference.editSummary;
      if (typeof (MyInPageEditPreference.editMinor) === 'boolean')
        editMinor = MyInPageEditPreference.editMinor;
      if (typeof (MyInPageEditPreference.outSideClose) === 'boolean')
        outSideClose = MyInPageEditPreference.outSideClose;
    }

    // Debug
    console.time('[InPageEdit] 获取页面源代码');
    console.info('%c[InPageEdit] Edit function running with params: ', 'color:#fe20d1');
    console.table({
      'editPage': editPage,
      'editSection': editSection,
      'titleSection': titleSection,
      'editRevision': editRevision,
      'titleRevision': titleRevision,
      'editSummary': editSummary,
      'editMinor': editMinor,
      'now': now
    });

    // 显示主窗口
    ssi_modal.show({
      title: '正在编辑:<u class="editPage">' + editPage + '</u>' + titleSection + titleRevision,
      content: '<div class="ipe-progress" style="width:100%"><div class="ipe-progress-bar"></div></div><section class="editForm"><textarea class="editArea"></textarea><div class="editOptionsLabel editForm"><label for="editSummary">摘要:</label><br/><input id="editSummary" class="editSummary" placeholder="Edit via InPageEdit~"/><br/><input id="editMinor" class="editMinor" type="checkbox" style=""/><label for="editMinor">标记为小编辑</label></div></section>',
      outSideClose: outSideClose,
      className: 'in-page-edit ipe-editor timestamp-' + timestamp,
      sizeClass: 'large',

      /* 按钮 */
      buttons: [{
        label: '保存更改',
        className: 'btn btn-primary leftBtn editForm',
        method: function () {
          ssi_modal.confirm({
            className: 'in-page-edit centerbox',
            content: '是否保存',
            okBtn: {
              className: 'btn btn-primary',
              label: '确定'
            },
            cancelBtn: {
              className: 'btn btn-secondary',
              label: '取消'
            },
          },
            function (result) {
              if (result) {
                var text = $('.ipe-editor.timestamp-' + timestamp + ' .editArea').val(),
                  minor = $('.ipe-editor.timestamp-' + timestamp + ' .editMinor').prop('checked'),
                  section = option.section,
                  summary = $('.ipe-editor.timestamp-' + timestamp + ' .editSummary').val();
                postArticle({
                  text: text,
                  page: editPage,
                  minor: minor,
                  section: section,
                  summary: summary
                });
              }
            })
        }
      }, {
        label: '显示预览',
        className: 'btn btn-secondary leftBtn editForm',
        method: function () {
          InPageEdit.analysis({ type: 'functionCount', function: 'previewEdit' });
          var text = $('.ipe-editor.timestamp-' + timestamp + ' .editArea').val();
          InPageEdit.quickPreview({
            action: 'parse',
            title: editPage,
            text: text,
            prop: 'text',
            preview: true,
            format: 'json'
          });
        }
      }, {
        label: '比较差异',
        className: 'btn btn-secondary leftBtn editForm',
        method: function () {
          InPageEdit.analysis({ type: 'functionCount', function: '快速差异Edit' });
          var text = $('.editArea').val();
          var diffJson = {};
          diffJson.fromtext = editText;
          diffJson.totext = text;
          diffJson.hideBtn = true;
          diffJson.pageName = editPage;
          diffJson.isPreview = true;
          InPageEdit.quickDiff(diffJson);
        }
      }, {
        label: '取消',
        className: 'btn btn-danger',
        method: function (e, modal) {
          modal.close();
        }
      }
      ],

      /* 开始执行后续程序 */
      onShow: function (modal) {
        // 绑定事件,在尝试离开页面时提示
        $(window).bind('beforeunload', function () {
          return '您输入的内容尚未保存,确定离开此页面吗?';
        });
        // 设置样式
        $('.ipe-editor.timestamp-' + timestamp + ' .editForm').hide();
        $('.ipe-editor.timestamp-' + timestamp + ' .ipe-progress').css('margin', Number($(window).height() / 3 - 50) + 'px 0');
        $('.ipe-editor.timestamp-' + timestamp + ' .editArea').css('height', $(window).height() / 3 * 2 - 100);
        $('.ipe-editor.timestamp-' + timestamp + ' .editOptionsLabel').prependTo('.ipe-editor.timestamp-' + timestamp + ' .ssi-buttons');
        $('.leftBtn').appendTo('.ssi-leftButtons');
        // 获取页面保护等级+最后编辑时间戳
        console.time('[InPageEdit] 获取页面基础信息');
        new mw.Api().get(jsonGetInfo).then(function (data) {
          if (data && data.query && data.query.pages) {
            var info = data.query.pages;
            for (var key in info) {
              if (key !== '-1') {
                console.timeEnd('[InPageEdit] 获取页面基础信息');
                console.info('[InPageEdit] 获取页面基础信息成功');
                if (info[key].touched) {
                  $('.ipe-editor.timestamp-' + timestamp).attr('data-basetimestamp', info[key].touched);
                } else {
                  $('.ipe-editor.timestamp-' + timestamp).attr('data-basetimestamp', now);
                }
                if (info[key].protection) {
                  if (typeof (info[key].protection[0].level) !== 'undefined') {
                    if (info[key].protection[0].type === 'edit') {
                      protection = info[key].protection[0].level;
                    }
                  }
                  if ((protection === 'autoconfirmed' && !InPageEdit.hasRight('autoconfirmed')) || (protection === 'sysop' && !InPageEdit.hasRight('editprotected')) || (mw.config.get('wgNamespaceNumber') === 8 && !InPageEdit.hasRight('editinterface'))) {
                    ssi_modal.notify('dialog', {
                      position: 'center bottom',
                      title: '权限不足',
                      content: '您没有编辑此页面的权限,请前往本页面的讨论页申请编辑。',
                      okBtn: {
                        label: '好的',
                        className: 'btn btn-primary',
                        method: function (e, modal) {
                          modal.close();
                        }
                      }
                    });
                    $('.ipe-editor.timestamp-' + timestamp + ' .editArea').attr('readonly', 'readonly');
                    $('.ipe-editor.timestamp-' + timestamp + ' button.editForm').attr('disabled', 'disabled');
                  }
                }
              } else {
                console.timeEnd('[InPageEdit] 获取页面基础信息');
                console.warn('[InPageEdit] 获取页面基础信息失败');
              }
            }
          }
        }).fail(function () {
          console.timeEnd('[InPageEdit] 获取页面基础信息');
          console.warn('[InPageEdit] 获取页面基础信息失败');
        });
        // 获取页面代码
        new mw.Api().get(jsonGet).then(function (data) {
          console.timeEnd('[InPageEdit] 获取页面源代码');
          if (data.error === undefined) {
            editText = data.parse.wikitext['*']
          } else {
            console.timeEnd('[InPageEdit] 获取页面源代码');
            editText = '<!-- 警告:无法获取页面内容 -->';
            console.error('[InPageEdit]警告:无法获取页面内容');
          }
          $('.ipe-editor.timestamp-' + timestamp + ' .ipe-progress').hide();
          $('.ipe-editor.timestamp-' + timestamp + ' .editForm').fadeIn(500);
          $('.ipe-editor.timestamp-' + timestamp + ' .editArea').val(editText + '\n');
        }).fail(function (data) {
          console.timeEnd('[InPageEdit] 获取页面源代码');
          editText = '<!-- 警告:无法获取页面内容 -->';
          console.error('[InPageEdit]警告:无法获取页面内容');
          $('.ipe-editor.timestamp-' + timestamp + ' .ipe-progress').hide();
          $('.ipe-editor.timestamp-' + timestamp + ' .editForm').fadeIn(500);
          $('.ipe-editor.timestamp-' + timestamp + ' .editArea').val(editText + '\n');
        });
        // 设定状态
        if (editMinor) {
          $('.ipe-editor.timestamp-' + timestamp + ' .editMinor').attr('checked', 'checked');
        }
        $('.ipe-editor.timestamp-' + timestamp + ' .editSummary').val(editSummary.replace(/\$section/ig, $($.parseHTML(titleSection)).text()).replace(/\$oldid/ig, summaryRevision));
        // 获取编辑提示
        new mw.Api().get({
          action: 'parse',
          page: 'MediaWiki:Editnotice-' + mw.config.get('wgNamespaceNumber'),
          prop: 'wikitext',
          format: 'json'
        }).then(function (data) {
          editNotice = data.parse.wikitext['*'];
          new mw.Api().post({
            action: 'parse',
            title: editPage.replace(/\.(js|css|json)/g, '@ipePreviewDot@$1'),
            preview: true,
            text: editNotice
          }).then(function (data) {
            editNotice = '<section class="editNotice" style="display:none">' + data.parse.text['*'].replace(/\@ipePreviewDot\@/g, '.') + '</section>';
            $('.ipe-editor.timestamp-' + timestamp + ' .ssi-modalContent').prepend(editNotice);
            $('.ipe-editor.timestamp-' + timestamp + ' .editNotice').fadeIn(500);
          });
        });
      },

      /* 确认是否取消 */
      beforeClose: function (modal) {
        ssi_modal.confirm({
          className: 'in-page-edit centerbox',
          position: 'top center',
          content: '您输入的内容尚未保存,确定关闭窗口吗?',
          okBtn: {
            className: 'btn btn-danger',
            label: '确定'
          },
          cancelBtn: {
            className: 'btn btn-secondary',
            label: '取消'
          }
        },
          function (result) {
            if (result === true) {
              $(window).unbind('beforeunload');
              modal.options.keepContent = false;
              modal.options.beforeClose = '';
              modal.close();
              ssi_modal.notify('error', {
                className: 'in-page-edit',
                position: 'right top',
                title: '取消',
                content: "编辑被取消"
              })
            }
          });
        return false;
      }
    });

    // 发布编辑模块
    function postArticle(pValue) {
      InPageEdit.analysis({ type: 'functionCount', function: '保存编辑' });
      InPageEdit.progress('正在发布编辑...');
      jsonPost = {
        action: 'edit',
        basetimestamp: $('.ipe-editor.timestamp-' + timestamp).attr('data-basetimestamp'),
        starttimestamp: now,
        text: pValue.text,
        title: pValue.page,
        minor: pValue.minor,
        summary: pValue.summary,
        errorformat: 'plaintext'
      }
      if (pValue.section !== undefined && pValue.section !== '' && pValue.section !== null) {
        jsonPost.section = pValue.section;
        delete jsonPost.basetimestamp;
      }
      // Debug
      console.info('%c[InPageEdit] Submitting with params: ', 'color:#fe20d1');
      console.table(jsonPost);
      new mw.Api().postWithToken('csrf', jsonPost).then(function (data) {
        InPageEdit.progress(true);
        $(window).unbind('beforeunload');
        ssi_modal.notify('success', {
          className: 'in-page-edit',
          position: 'right top',
          title: '成功',
          content: '成功,正在刷新页面。'
        });
        setTimeout(function () {
          if (pValue.page === mw.config.get('wgPageName')) {
            window.location = mw.config.get('wgArticlePath').replace('$1', pValue.page);
          } else {
            window.location.reload();
          }
        }, 500);
      }).fail(function (errorCode, feedback, errorThrown) {
        InPageEdit.progress(false);
        ssi_modal.notify('error', {
          className: 'in-page-edit',
          position: 'right top',
          closeAfter: {
            time: 15
          },
          title: '警告',
          content: '发布编辑时发生错误:<br/><span style="font-size:amall">' + errorThrown.errors[0]['*'] + '(<code>' + errorThrown.errors[0]['code'] + '</code>)</span>'
        });
        console.error('[InPageEdit] Submit failed: \nCode: ' + errorThrown.errors[0]['code'] + '\nDescription: ' + errorThrown.errors[0]['*']);
        InPageEdit.error(errorThrown.errors[0]['code']);
      });
    }

  }

  /** 快速重定向模块 **/
  InPageEdit.redirect = function (type) {
    var json = {
      action: 'edit',
      minor: JSON.parse(localStorage.getItem('InPageEditPreference')).editMinor,
      token: mw.user.tokens.get('editToken'),
      errorformat: 'plaintext'
    },
      summary = '快速重定向 → [[:$1]] //使用API进行的操作,请核查本次编辑',
      text = '#REDIRECT [[:$1]]',
      question,
      target;
    switch (type) {
      case 'to':
        json.title = mw.config.get('wgPageName');
        question = '您要将 <b>' + mw.config.get('wgPageName') + '</b> 重定向到哪个页面?';
        break;
      case 'from':
        question = '您要将哪个页面重定向到 <b>' + mw.config.get('wgPageName') + '</b>?';
        json.text = text.replace('$1', mw.config.get('wgPageName'));
        json.summary = summary.replace('$1', mw.config.get('wgPageName'));
        break;
    }
    ssi_modal.show({
      outSideClose: false,
      className: 'in-page-edit quick-redirect',
      center: true,
      sizeClass: 'dialog',
      title: '快速重定向',
      content: '<section>' + question + '<br/><input id="redirect-page" style="width:80%;margin: 0 10%;" onclick="$(this).css(\'box-shadow\',\'\')"/></section><div class="ipe-progress" style="width:100%;display:none"><div class="ipe-progress-bar"></div></div>',
      buttons: [{
        label: '确定',
        className: 'btn btn-primary btn-single okBtn',
        method: function (a, modal) {
          InPageEdit.analysis({ type: 'functionCount', function: '快速重定向' });
          InPageEdit.analysis({ type: 'dateCount' });
          InPageEdit.analysis({ type: 'siteCount' });
          var input = $('#redirect-page').val();
          if (input === '' || input === mw.config.get('wgPageName')) {
            $('#redirect-page').css('box-shadow', '0 0 4px red');
          } else {
            $('.in-page-edit.quick-redirect .ipe-progress').show();
            $('.in-page-edit.quick-redirect section').hide();
            $('.in-page-edit.quick-redirect .okBtn').attr('disabled', 'disabled');
            switch (type) {
              case 'to':
                json.summary = summary.replace('$1', input);
                json.text = text.replace('$1', input);
                break;
              case 'from':
                json.title = input;
                break;
            }

            new mw.Api().post(json).done(function () {
              $('.in-page-edit.quick-redirect .ipe-progress').addClass('done');
              ssi_modal.notify('success', {
                className: 'in-page-edit',
                content: '已创建重定向!',
                title: '成功'
              });
              if (type === 'to') {
                window.location.reload();
              } else {
                $('.in-page-edit.quick-redirect .ipe-progress').addClass('done');
                setTimeout(function () { modal.close() }, 2000);
              }
            }).fail(function () {
              $('.in-page-edit.quick-redirect .ipe-progress').hide();
              $('.in-page-edit.quick-redirect section').show();
              $('.in-page-edit.quick-redirect .okBtn').attr('disabled', false);
              $('.in-page-edit.quick-redirect .ipe-progress').addClass('done');
              ssi_modal.notify('error', {
                className: 'in-page-edit',
                content: '创建重定向时遇到未知问题!',
                title: '失败'
              });
            });
          }
        }
      }
      ]
    });
  }

  /** 删除页面模块 **/
  InPageEdit.deletepage = function () {
    InPageEdit.analysis({ type: 'functionCount', function: '快速删除' });
    var page = mw.config.get('wgPageName'),
      reasonType,
      reason = '不再需要的页面';

    ssi_modal.show({
      outSideClose: false,
      className: 'in-page-edit quick-delete',
      center: true,
      sizeClass: 'dialog',
      title: '删除页面',
      content: '<b style="color:#b00">本功能暂未实装</b><section id="InPageEditDeletepage">您要将 <b>' + page + '</b> 删除的理由?<br/><select id="reasonType" style="width: 23%;padding: 2px;margin: 2px;"><option value="其他">其他</option></select><input id="reason" style="width: 73%;padding: 2px;margin: 2px;" value="' + reason + '"><br/><input id="confirm" type="checkbox"/><label for="confirm">确定删除</label></section>',
      beforeShow: function () {
        if (!InPageEdit.hasRight('delete')) {
          ssi_modal.dialog({
            title: '权限不足',
            content: '抱歉,您没有删除页面(<code>delete</code>)的权限。',
            className: 'in-page-edit quick-deletepage',
            center: true,
            okBtn: {
              className: 'btn btn-primary btn-single'
            }
          });
          return false;
        }
      }
    });
  }

  /** 重命名模块 **/
  InPageEdit.renamepage = function () {
    var from = mw.config.get('wgPageName'),
      to,
      reason = '重命名页面 → [[:$1]] //使用API进行的操作,请核查本次编辑',
      movetalk,
      movesubpages,
      noredirect,
      ignorewarnings;

    ssi_modal.show({
      outSideClose: false,
      className: 'in-page-edit quick-rename',
      center: true,
      sizeClass: 'dialog',
      title: '重命名页面',
      content: '<section id="InPageEditRename"><label for="move-to">您要将 <b>' + from + '</b> 重命名为?</label><br/><input style="width:90%" id="move-to"><br/><input type="checkbox" id="movetalk" checked="checked" disabled="disabled"/><label for="movetalk">同时移动讨论页(若存在,暂时预设必选)</label><br/><input id="movesubpages" type="checkbox" checked="checked"/><label for="movesubpages">同时移动子页面(若存在)</label><br/><input id="noredirect" type="checkbox"/><label for="noredirect">不在本页面创建到新页面的重定向(需要权限)</label><br/><label for="move-reason">编辑摘要(<code>$1</code>将替换为新页面名)<br/><input style="width:90%" value="' + reason + '" id="move-reason"/></label></section>',
      buttons: [{
        label: '取消',
        className: 'btn btn-secondary',
        method: function (a, modal) {
          modal.close();
        }
      }, {
        label: '确定',
        className: 'btn btn-primary',
        method: function () {
          InPageEdit.analysis({ type: 'functionCount', function: '快速重命名' });
          InPageEdit.analysis({ type: 'dateCount' });
          InPageEdit.analysis({ type: 'siteCount' });
          InPageEdit.progress('正在和土豆交涉……');
          movetalk = $('#movetalk').prop('checked');
          movesubpages = $('#movesubpages').prop('checked');
          noredirect = $('#noredirect').prop('checked');
          to = $('#move-to').val();
          reason = $('#move-reason').val().replace('$1', to);
          new mw.Api().postWithToken('csrf', {
            action: 'move',
            from: from,
            to: to,
            reason: reason,
            movetalk: movetalk,
            movesubpages: movesubpages,
            noredirect: noredirect
          }).done(function () {
            InPageEdit.progress(true);
            ssi_modal.notify('success', {
              className: 'in-page-edit',
              content: '正在刷新页面……',
              title: '成功'
            });
            location.href = mw.config.get('wgArticlePath').replace('$1', to);
          }).fail(function (errorCode, feedback, errorThrown) {
            InPageEdit.progress(false);
            ssi_modal.notify('error', {
              className: 'in-page-edit',
              content: '重命名时遇到问题:<br/>' + errorThrown.error.info + '<code>' + errorThrown.error.code + '</code>',
              title: '失败'
            });
            if (errorThrown.error.code === 'articleexists') {
              ssi_modal.dialog({
                className: 'in-page-edit centerbox',
                title: '目标页面已存在',
                content: '强制移动页面到已存在的标题需要利用删除功能,这个选项会在“快速删除”模块完成时添加,请暂时使用MediaWiki自带的移动并选择强制删除目标页面。',
                okBtn: {
                  label: '好的',
                  className: 'btn btn-primary only-btn'
                }
              });
            }
          });
        }
      }],
      beforeShow: function () {
        if (!InPageEdit.hasRight('move')) {
          ssi_modal.dialog({
            title: '权限不足',
            content: '抱歉,您没有移动页面(<code>move</code>)的权限。',
            className: 'in-page-edit quick-deletepage',
            center: true,
            okBtn: {
              className: 'btn btn-primary btn-single'
            }
          });
          return false;
        }
      }
    });
  }

  /** 个人设置模块 **/
  InPageEdit.preference = function () {
    InPageEdit.analysis({ type: 'functionCount', function: '插件设置' });
    var settings = JSON.parse(localStorage.getItem('InPageEditPreference')),
      minor = settings.editMinor,
      summary = settings.editSummary,
      outSideClose = settings.outSideClose;

    ssi_modal.show({
      outSideClose: false,
      title: 'InPageEdit偏好设定 - ' + InPageEdit.version,
      content: '<section id="InPageEditSettingBox"><b>InPageEdit编辑器</b><br/><input id="ipeSetoutSideClose" type="checkbox"/> <label for="ipeSetoutSideClose">点击编辑窗口外侧关闭编辑窗口</label><br/><input id="ipeSetMinor" type="checkbox"/> <label for="ipeSetMinor">默认标记所有编辑为小编辑</label><br/><b>摘要</b><br/><label>默认编辑摘要<br/><span style="font-size:10px"><code>$section</code> - 若编辑的是段落,将替换为<code>/* 段落标题 */</code><br/><code>$oldid</code> - 若编辑的是历史版本,将替换为<code>(编辑自[[Special:Diff/revid]])</code>)</span><input id="ipeSetSummary" value="' + summary + '" style="width:100%"/></label><br/><b>InPageEdit数据收集</b><br/><input id="ipeDoNotUseMyInfo" type="checkbox" disabled=""/> <label for="ipeDoNotUseMyInfo">不要收集我使用InPageEdit的信息</label><br/><span style="font-size:10px">关于数据收集,可以访问<a href="https://doc.wjghj.cn/InPageEditAnalysis/" target="_blank">https://doc.wjghj.cn/InPageEditAnalysis/</a>查看</span><br/><b>关于InPageEdit-v2</b><br/><button class="btn btn-secondary" onclick="mw.loader.load(\'https://common.wjghj.cn/js/InPageEdit-v2.js/about\')">关于&帮助</button>&nbsp;<button class="btn btn-secondary" onclick="InPageEdit.versionInfo()">更新日志</button>&nbsp;<button class="btn btn-danger" id="ipeUninstall">卸载</button><hr><span style="font-size:10px;line-height:5px">您可以在这里保存InPageEdit-v2的个人偏好。<br/><b>注意</b>:这些设置保存在您的浏览器本地,这意味着你必须在不同的设备上分别保存设置。<a href="javascript:;" id="ipeSaveLocalShow">永久保存</a>(不推荐)</span></section>',
      sizeClass: 'dialog',
      className: 'in-page-edit ipe-preference',
      center: true,
      buttons: [{
        label: '重置',
        className: 'btn btn-danger',
        method: function () {
          $('#InPageEditSettingBox #ipeSetoutSideClose').prop('checked', true);
          $('#InPageEditSettingBox #ipeSetMinor').prop('checked', false);
          $('#InPageEditSettingBox #ipeSetSummary').val('[InPageEdit] 没有编辑摘要 $section$oldid');
        }
      }, {
        label: '保存',
        className: 'btn btn-primary',
        method: function (a, modal) {
          localStorage.setItem('InPageEditPreference', JSON.stringify({
            outSideClose: $('#InPageEditSettingBox #ipeSetoutSideClose').prop('checked'),
            editMinor: $('#InPageEditSettingBox #ipeSetMinor').prop('checked'),
            editSummary: $('#InPageEditSettingBox #ipeSetSummary').val()
          }));

          modal.close();
        }
      }
      ]
    });
    $('#ipeSaveLocalShow').click(function () {
      ssi_modal.dialog({
        className: 'in-page-edit',
        center: true,
        content: '<section id="ipeSaveLocal"><b>不推荐这种做法</b><br/>在您个人js页调用本插件的代码的上方添加以下代码:<br/><input style="width:100%" readonly="readonly" onclick="$(this).select()"/><br/>注意:虽然这样做免去了更换设备时需要重新进行设置的麻烦,但我们没有参数纠错功能,您必须自行确认json设定是否正确。</section>',
        okBtn: {
          className: 'btn btn-primary btn-single'
        }
      });
      $('#ipeSaveLocal input').val('var MyInPageEditPreference = ' + JSON.stringify({
        outSideClose: $('#InPageEditSettingBox #ipeSetoutSideClose').prop('checked'),
        editMinor: $('#InPageEditSettingBox #ipeSetMinor').prop('checked'),
        editSummary: $('#InPageEditSettingBox #ipeSetSummary').val()
      }) + ';');
    });
    $('#ipeUninstall').click(function () {
      ssi_modal.show({
        className: 'in-page-edit',
        center: true,
        sizeClass: 'dialog',
        title: '卸载InPageEdit-v2',
        content: '插件目前处于不断开发的阶段,很多功能并不完善,可能没有达到您的预期,肥肠抱歉!由衷希望您可以等待插件继续被开发、完善,并希望您可以为我提供宝贵的建议!<br/>插件在安装时会同时保存一些localStorage,如果您使用的是手机那么将非常难以删除,因此我提供了这个可以清理残余项的卸载功能。<br/><span style="font-size:small"><b>注意</b>:本功能尚处于测试阶段,使用正则表达式简单粗暴地从您的个人js页识别并删除插件相关的代码,有可能会卸载失败,建议按照插件说明页手动删除插件。</span>',
        buttons: [{
          label: '废话少说,残忍卸载',
          className: 'btn btn-danger',
          enableAfter: 5,
          method: function () {
            InPageEdit.analysis({ type: 'functionCount', function: '卸载插件' });
            mw.loader.load('https://common.wjghj.cn/js/InPageEdit-v2.js/uninstall');
          }
        }, {
          label: '算了,暂时不卸载了',
          className: 'btn btn-primary',
          method: function (event, modal) {
            modal.close();
          }
        }
        ]
      });
    });
    if (outSideClose) {
      $('#InPageEditSettingBox #ipeSetoutSideClose').prop('checked', true);
    }
    if (minor) {
      $('#InPageEditSettingBox #ipeSetMinor').prop('checked', true);
    }

    if (typeof (MyInPageEditPreference) !== 'undefined') {
      $('.ipe-preference .ssi-buttons .ssi-modalBtn').attr('disabled', '');
      if (typeof (MyInPageEditPreference.editMinor) === 'boolean') {
        $('#InPageEditSettingBox #ipeSetMinor').prop('checked', MyInPageEditPreference.editMinor).attr('disabled', '');
      }
      if (typeof (MyInPageEditPreference.outSideClose) === 'boolean') {
        $('#InPageEditSettingBox #ipeSetoutSideClose').prop('checked', MyInPageEditPreference.outSideClose).attr('disabled', '');
      }
      if (typeof (MyInPageEditPreference.editSummary) === 'string') {
        $('#InPageEditSettingBox #ipeSetSummary').attr('disabled', '').val(MyInPageEditPreference.editSummary);
      }
      ssi_modal.dialog({
        content: '您使用自己的个人js保存了设定参数,请浏览<a href="' + mw.config.get('wgArticlePath').replace('$1', 'Special:Mypage/common.js') + '">您的个人js页</a>修改。<br/>我们不推荐保存静态的设定,虽然这样做免去了更换设备时需要重新进行设置的麻烦,但因为暂时没有参数纠错功能,您必须自行确认json设定是否正确。',
        className: 'in-page-edit',
        center: true,
        okBtn: {
          className: 'btn btn-primary btn-single'
        }
      });
    }
  }

  /** 快速页面差异模块 **/
  InPageEdit.quickDiff = function (param) {
    InPageEdit.analysis({ type: 'dateCount' });
    InPageEdit.analysis({ type: 'siteCount' });
    // InPageEdit.analysis({ type: 'functionCount', function: '快速差异' });
    if ($('[href*="mediawiki.diff.styles"]').length < 1) {
      mw.loader.load(mw.config.get('wgScriptPath') + '/load.php?modules=mediawiki.legacy.shared|mediawiki.diff.styles&only=styles', 'text/css');
    }
    if ($('.quick-diff').length > 0) {
      console.info('[InPageEdit] Quick diff 正在加载新内容');
      $('.in-page-edit.quick-diff .diffArea').hide().html('加载中');
      if (param.isPreview) {
        $('.quick-diff').appendTo('body');
      }
    } else {
      ssi_modal.show({
        className: 'in-page-edit quick-diff',
        sizeClass: 'large',
        fixedHeight: true,
        fitScreen: true,
        title: '<span class="pageName">正在加载差异……</span>',
        content: '<div class="ipe-progress" style="width:100%"><div class="ipe-progress-bar"></div></div><div class="diffArea"></div>',
        buttons: [{
          label: '转到原版比较页面',
          className: 'btn btn-secondary toDiffPage',
          method: function () {
            // Placeholder
          }
        }]
      });
    }
    $('.in-page-edit.quick-diff .ipe-progress').show().css('margin-top', $('.in-page-edit.quick-diff .ipe-progress').parent().height() / 2);
    $('.quick-diff button.toDiffPage').unbind();
    param.action = 'compare';
    param.prop = 'diff|diffsize|rel|ids|title|user|comment|parsedcomment|size';
    param.format = 'json';
    new mw.Api().post(param).then(function (data) {
      var diffTable = data.compare['*'];
      $('.in-page-edit.quick-diff .ipe-progress').hide();
      if (param.pageName === undefined) {
        var toTitle = data.compare.totitle;
      } else {
        var toTitle = param.pageName;
      }
      var userlink = function (user) {
        return '<a href="' + mw.util.getUrl('User:' + user) + '">' + user + '</a> ( <a href="' + mw.util.getUrl('User_talk:' + user) + '">讨论</a> | <a href="' + mw.util.getUrl('Special:Contributions/' + user) + '">贡献</a> | <a href="' + mw.util.getUrl('Special:Block/' + user) + '">封禁</a> )';
      }
      $('.quick-diff .pageName').html('比较差异:<u>' + toTitle + '</u>');
      $('.quick-diff .diffArea').show().html(
        '<table class="diff diffTable">' +
        '<colgroup>' +
        '<col class="diff-marker">' +
        '<col class="diff-content">' +
        '<col class="diff-marker">' +
        '<col class="diff-content">' +
        '</colgroup>' +
        '<tbody>' +
        '<tr class="diff-title">' +
        '<td colspan="2" class="diff-otitle">' +
        '<a class="" href="' + mw.config.get('wgScript') + '?oldid=' + data.compare.fromrevid + '">' + data.compare.fromtitle + '</a> (版本' + data.compare.fromrevid + ') (<a class="editLink" href="' + mw.config.get('wgScript') + '?action=edit&title=' + data.compare.fromtitle + '&oldid=' + data.compare.fromrevid + '">编辑</a>)<br/>' + userlink(data.compare.fromuser) + '<br/>(' + data.compare.fromparsedcomment + ')<br/><a class="prevVersion" href="javascript:void(0);" onclick="InPageEdit.quickDiff({fromrev:' + data.compare.fromrevid + ',torelative:\'prev\'});InPageEdit.analysis({type:\'functionCount\',function:\'快速差异History\'});">←上一版本</a>' +
        '</td>' +
        '<td colspan="2" class="diff-ntitle">' +
        '<a class="" href="' + mw.config.get('wgScript') + '?oldid=' + data.compare.torevid + '">' + data.compare.totitle + '</a> (版本' + data.compare.torevid + ') (<a class="editLink" href="' + mw.config.get('wgScript') + '?action=edit&title=' + data.compare.totitle + '&oldid=' + data.compare.torevid + '">编辑</a>)<br/>' + userlink(data.compare.touser) + '<br/>(' + data.compare.toparsedcomment + ')<br/><a class="nextVersion" href="javascript:void(0);" onclick="InPageEdit.quickDiff({fromrev:' + data.compare.torevid + ',torelative:\'next\'});InPageEdit.analysis({type:\'functionCount\',function:\'快速差异History\'});">下一版本→</a>' +
        '</td>' +
        '</tr>' +
        diffTable +
        '<tr class="diffSize" style="text-align: center;"><td colspan="2">' + data.compare.fromsize + '字节</td><td colspan="2">' + data.compare.tosize + '字节</td></tr>' +
        '</tbody>' +
        '</table>'
      );
      $('.quick-diff button.toDiffPage').click(function () {
        location.href = mw.config.get('wgScript') + '?oldid=' + data.compare.fromrevid + '&diff=' + data.compare.torevid;
      });
      InPageEdit.articleLink($('.quick-diff .editLink'));
      if (param.isPreview === true) {
        $('.quick-diff button.toDiffPage').hide();
        $('.quick-diff .diff-otitle').html('<b>原始内容</b>');
        $('.quick-diff .diff-ntitle').html('<b>您的编辑</b>');
      }
      if (data.compare.fromsize === undefined || data.compare.tosize === undefined) {
        $('.quick-diff .diffSize').hide();
      }
      if (data.compare.fromrevid === undefined && param.isPreview !== true) {
        $('.quick-diff .diff-otitle').html('<span class="noNextVerson"><b>没有更多了哟~</b><br/><span style="font-size:10px">没有之前的版本了!之前的版本即使是盘古也没有见过呀!</span></span>');
      } else if (data.compare.torevid === undefined && param.isPreview !== true) {
        $('.quick-diff .diff-ntitle').html('<span class="noNextVerson"><b>没有更多了哟~</b><br/><span style="font-size:10px">没有之后的版本了!一点也没有了!!真的没有了!!!</span></span>');
      }
    }).fail(function (a, b, c) {
      $('.in-page-edit.quick-diff .ipe-progress').hide();
      $('.diffArea').html('比较差异时出现错误: ' + b);
    });
  }
  // 加载预设的快速最近更改模块
  InPageEdit.loadQuickDiff = function () {
    // 最近更改
    function addLink(origin) {
      $('.mw-changeslist-groupdiff, .mw-changeslist-diff, .mw-changeslist-diff-cur, .mw-history-histlinks a').unbind('click', ipeDiffLink);
      var ipeDiffLink = $('.mw-changeslist-groupdiff, .mw-changeslist-diff, .mw-changeslist-diff-cur, .mw-history-histlinks a').click(function (e) {
        e.preventDefault();
        InPageEdit.analysis({ type: 'functionCount', function: '快速差异RC' });
        var $this = $(this),
          href = $this.attr('href'),
          diff = mw.util.getParamValue('diff', href),
          curid = mw.util.getParamValue('curid', href),
          oldid = mw.util.getParamValue('oldid', href);
        if (diff === '0') {
          InPageEdit.quickDiff({ fromrev: oldid, toid: curid });
        } else if (diff === 'prev' || diff === 'next' || diff === 'cur') {
          InPageEdit.quickDiff({ fromrev: oldid, torelative: diff });
        } else {
          InPageEdit.quickDiff({ fromrev: oldid, torev: diff });
        }
      });
    }
    if ($('.mw-rcfilters-enabled').length > 0) {
      setInterval(addLink, 500);
      $('.mw-rcfilters-enabled').addClass('ipe-continuous-active');
    } else {
      addLink();
    }
    // 查看历史页面的比较按钮与快速编辑
    if (mw.config.get('wgAction') === 'history') {
      $('.historysubmit.mw-history-compareselectedversions-button').after(
        $('<button>').text('快速对比差异').click(function (e) {
          e.preventDefault();
          InPageEdit.analysis({ type: 'functionCount', function: '快速差异History' });
          var before = $('.selected.before').attr('data-mw-revid'),
            after = $('.selected.after').attr('data-mw-revid');
          InPageEdit.quickDiff({ fromrev: after, torev: before });
        })
      );
      $('[data-mw-revid]').each(function () {
        var $this = $(this),
          oldid = $this.attr('data-mw-revid');
        $this.find('.mw-history-undo').after(
          $('<span>').html(' | <a class="in-page-edit-article-link" href="javascript:void(0);" onclick="InPageEdit.edit({page:mw.config.get(\'wgPageName\'),revision:' + oldid + '});">快速编辑</a>')
        );
      });
    }
  }

  /** 获取段落编辑以及编辑链接 **/
  InPageEdit.articleLink = function (element) {
    if (element === undefined)
      element = $('#mw-content-text a:not(.new)');
    element.each(function (i) {
      if ($(this).attr('href') === undefined)
        return;
      var url = $(this).attr('href'),
        title = mw.util.getParamValue('title', url),
        section = mw.util.getParamValue('section', url),
        revision = mw.util.getParamValue('oldid', url);

      // 不是本地编辑链接
      if (url.split('/')['2'] !== location.href.split('/')['2'] && url.substr(0, 1) !== '/')
        return;

      // 不是 index.php?title=FOO 形式的url
      if (title === null) {
        var splitStr = mw.config.get('wgArticlePath').replace('$1', '');
        if (splitStr === '/') {
          splitStr = mw.config.get('wgServer').substring(mw.config.get('wgServer').length - 4) + '/';
        }
        title = url.split(splitStr).pop().split('?')['0'];
      }

      if (mw.util.getParamValue('action', url) === 'edit' && title !== undefined && section !== 'new') {
        $(this).after(
          $('<span>', {
            'class': 'in-page-edit-article-link-group'
          }).append(
            $('<a>', {
              'href': 'javascript:void(0)',
              'class': 'in-page-edit-article-link'
            })
              .text('快速编辑')
              .click(function () {
                if (revision !== null) {
                  InPageEdit.edit({
                    page: title,
                    revision: revision
                  });
                } else {
                  InPageEdit.edit({
                    page: title,
                    section: section
                  });
                }
              })));
      }
    });
  }

  /** 快速预览文章页 **/
  InPageEdit.quickPreview = function (params) {
    var timestamp = new Date().getTime();
    console.time('[InPageEdit] Request preview');
    ssi_modal.show({
      sizeClass: 'large',
      className: 'in-page-edit previewbox',
      content: '<div class="ipe-progress" style="width:100%"><div class="ipe-progress-bar"></div></div><section id="InPageEditPreview" data-timestamp="' + timestamp + '" style="display:none">正在读取预览……</section>',
      title: '预览',
      fixedHeight: true,
      fitScreen: true,
      buttons: [{ label: '', className: 'hideThisBtn' }],
      onShow: function (modal) {
        $('.previewbox .ipe-progress').css('margin-top', $('.previewbox .ipe-progress').parent().height() / 2);
        $('.previewbox .hideThisBtn').hide();
        new mw.Api().post(params).then(function (data) {
          console.timeEnd('[InPageEdit] Request preview');
          var content = data.parse.text['*'];
          $('.previewbox .ipe-progress').hide(150);
          $('#InPageEditPreview[data-timestamp="' + timestamp + '"]').fadeIn(500).html(content);
        }).fail(function () {
          console.timeEnd('[InPageEdit] Request preview');
          console.warn('[InPageEdit] 预览失败');
          $('.previewbox .ipe-progress').hide(150);
          $('#InPageEditPreview[data-timestamp="' + timestamp + '"]').fadeIn(500).html('获取预览时发生错误!');
        });
      }
    });
  }

  /** 载入中模块 **/
  InPageEdit.progress = function (title) {
    if (title === true) {
      $('.in-page-edit.loadingbox .ssi-modalTitle').html('完成');
      $('.in-page-edit.loadingbox .ipe-progress').addClass('done');
    } else if (title === false) {
      if ($('.in-page-edit.loadingbox').length > 0) {
        $('.in-page-edit.loadingbox').appendTo('body');
        ssi_modal.close();
      }
    } else {
      if ($('.in-page-edit.loadingbox').length > 0) return;
      if (typeof (title) === 'undefined') {
        title = '加载中...'
      }
      ssi_modal.show({
        title: title,
        content: '<div class="ipe-progress" style="width:100%"><div class="ipe-progress-bar"></div></div>',
        className: 'in-page-edit loadingbox',
        center: true,
        sizeClass: 'dialog',
        closeIcon: false,
        outSideClose: false
      });
    }
  }

  /** 常见错误集 **/
  InPageEdit.error = function (code) {
    var errorTitle = '发生未知错误',
      errorContent = '错误代码为:<code>' + code + '</code><br height:"0.5em"/>常见的错误代码请查看<a href="https://www.mediawiki.org/wiki/API:Edit#Possible_errors" target="_blank">这里</a>。大多数情况下刷新页面可以解决问题。';
    switch (code) {
      case 'editconflict':
        errorTitle = '检测到编辑冲突';
        errorContent = '在您打开编辑窗口到保存的期间,有用户修改了页面的内容,建议您现在复制您的编辑内容,重新打开编辑窗口检查并再次提交您的内容';
        break;
      case 'badtoken':
        errorTitle = '编辑令牌错误';
        errorContent = '系统获取并提交的token被系统拒绝,暂时只能通过刷新页面来重新获取,刷新前请备份您的编辑!';
        break;
      case 'pagedeleted':
        errorTitle = '页面已被删除';
        errorContent = '在您保存页面时该页面已被删除,请备份您的编辑内容并检查删除日志确认具体情况。';
        break;
      case 'protectedpage':
      case 'protectednamespace-interface':
      case 'permissiondenied':
        errorTitle = '权限不足';
        errorContent = '您没有编辑此页面的权限,可能是因为该页面受到人为保护或系统保护以防止改动,请在讨论页提交编辑申请或联系wiki管理员。';
        break;
      case 'ratelimited':
        errorTitle = '触发频率保护';
        errorContent = '您的提交频率触发了保护性上限,请稍候再试。';
        break;
    }
    errorContent = errorContent + '<hr/><span style="font-size:small;line-height: 4px;">若此问题多次出现,请复制编辑内容并使用MediaWiki原生编辑页面发布您的编辑。若您愿意,请提供控制台的错误代码片段或者截图给作者,谢谢!</span>';

    ssi_modal.show({
      outSideClose: false,
      sizeClass: 'dialog',
      className: 'in-page-edit',
      center: true,
      title: errorTitle,
      content: errorContent,
      buttons: [{
        label: '*口吐芬芳*',
        className: 'btn btn-danger',
        method: function (a, modal) { window.open('https://blog.wjghj.cn/index.php/contact') }
      }, {
        label: '好的',
        className: 'btn btn-primary',
        method: function (a, modal) { modal.close() }
      }]
    });
  }
  /** 提交统计信息模块 **/
  InPageEdit.analysis = function (params) {
    switch (params.type) {
      case 'siteCount':
        $.ajax({
          url: 'https://doc.wjghj.cn/InPageEditAnalysis/site/log.php',
          data: {
            'sitename': mw.config.get('wgSiteName')
          },
          dataType: 'json'
        });
        break;
      case 'dateCount':
        var now = new Date(),
          y = now.getFullYear(), m = Number(now.getMonth() + 1), d = Number(now.getDate());
        if (m < 10) m = '0' + m;
        if (d < 10) d = '0' + d;
        var time = y + '-' + m + '-' + d;
        $.ajax({
          url: 'https://doc.wjghj.cn/InPageEditAnalysis/date/log.php',
          data: {
            'date': time
          },
          dataType: 'json'
        });
        break;
      case 'functionCount':
        $.ajax({
          url: 'https://doc.wjghj.cn/InPageEditAnalysis/function/log.php',
          data: {
            'function': params.function
          },
          dataType: 'json'
        });
        break;
    }
  }

  /** 获取用户权限信息 **/
  $(function () {
    var wgUserRights,
      user = mw.config.get('wgUserName');
    if (user === null) {
      console.warn('[InPageEdit] 警告:用户未登录');
      mw.config.set('wgUserRights', '');
      return;
    }
    new mw.Api().get({
      action: 'query',
      list: 'users',
      usprop: 'rights',
      ususers: user
    }).done(function (data) {
      console.info('[InPageEdit] 成功获取用户权限信息');
      mw.config.set('wgUserRights', data.query.users[0]['rights']);
    }).fail(function () {
      console.warn('[InPageEdit] 警告:无法获取用户权限信息');
      mw.config.set('wgUserRights', '');
    });
  });
  InPageEdit.hasRight = function (right) {
    if (mw.config.get('wgUserRights').indexOf(right) > -1) {
      return true;
    } else {
      return false;
    }
  }
  /** 初始化 **/
  $(function () {
    // 由于兼容性问题,阻止Fandom平台
    if (location.href.split('.')[1] === 'fandom') {
      new BannerNotification('注意,当前版本InPageEdit扩展在Fandom平台有严重兼容性问题。<br/>目前不建议在Fandom使用,后续的支持计划请暂定于Fandom社区统一计划(UCP)完成后。', 'warn').show();
      $('.ssi-stack').remove();
      console.error('%c[InPageEdit] 警告:暂不支持的平台。', 'color:darkred;font-size:1.2em;font-weight:bold');
      return;
    }
    /** 额外的模块 **/
    // 快速页面差异模块
    InPageEdit.loadQuickDiff();

    /** Toolbox模块 **/
    // 检测是否为文章页
    if (mw.config.get('wgIsArticle') === false) {
      console.warn('%c[InPageEdit] 不是文章页面,未载入工具盒。', 'color:orange;font-size:1.2em;font-weight:bold');
      return;
    }

    // 读取设定
    if (localStorage.getItem('InPageEditPreference') === null) {
      // 没有保存任何设置
      var ipePreference = {};
      ipePreference.outSideClose = true;
      ipePreference.editMinor = false;
      ipePreference.editSummary = '[InPageEdit] 没有编辑摘要 $section$oldid';
      localStorage.setItem('InPageEditPreference', JSON.stringify(ipePreference));
    }

    // ipe工具盒
    $('body').append(
      '<div id="ipe-edit-toolbox">' +

      // group1 上方的一列按钮
      '<ul class="btn-group group1">' +
      '<li class="btn-tip-group"><div class="btn-tip">快速编辑</div><button id="edit-btn" class="ipe-toolbox-btn material-icons">edit</button></li>' +
      '<li class="btn-tip-group"><div class="btn-tip">重定向至此</div><button id="redirectfrom-btn" class="ipe-toolbox-btn material-icons">flight_land</button></li>' +
      '<li class="btn-tip-group"><div class="btn-tip">重定向到</div><button id="redirectto-btn" class="ipe-toolbox-btn material-icons">flight_takeoff</button></li>' +
      '</ul>' +

      // group2 左边的一排按钮
      '<ul class="btn-group group2"><div style="display: flex;">' +
      '<li class="btn-tip-group"><div class="btn-tip">删除本页面</div><button id="deletepage-btn" class="ipe-toolbox-btn material-icons">delete_forever</button></li>' +
      '<li class="btn-tip-group"><div class="btn-tip">重命名页面</div><button id="renamepage-btn" class="ipe-toolbox-btn material-icons">format_italic</button></li>' +
      '<li class="btn-tip-group"><div class="btn-tip">自定义设置</div><button id="preference-btn" class="ipe-toolbox-btn material-icons">settings</button></li>' +
      '</div></ul>' +

      '<button class="ipe-toolbox-btn material-icons" id="toolbox-toggle">add</button>' +
      '</div>');
    $('#ipe-edit-toolbox #toolbox-toggle').click(function () {
      $('#ipe-edit-toolbox #toolbox-toggle, #ipe-edit-toolbox .btn-group').toggleClass('opened');
    });
    $('body > *:not(#ipe-edit-toolbox)').click(function () {
      $('#ipe-edit-toolbox #toolbox-toggle, #ipe-edit-toolbox .btn-group').removeClass('opened');
    });
    $('#ipe-edit-toolbox .btn-group .ipe-toolbox-btn').click(function () {
      InPageEdit.analysis({ type: 'functionCount', function: '工具盒' });
      switch ($(this).attr('id')) {
        case 'edit-btn':
          InPageEdit.edit({
            page: mw.config.get('wgPageName'),
            revision: mw.config.get('wgRevisionId')
          });
          break;
        case 'redirectfrom-btn':
          InPageEdit.redirect('from');
          break;
        case 'redirectto-btn':
          InPageEdit.redirect('to');
          break;
        case 'preference-btn':
          InPageEdit.preference();
          break;
        case 'deletepage-btn':
          InPageEdit.deletepage();
          break;
        case 'renamepage-btn':
          InPageEdit.renamepage();
          break;
      }
    });
    // 加载段落编辑模块
    InPageEdit.articleLink();
    // 花里胡哨的加载提示
    console.info('    ____      ____                   ______    ___ __              _    _____ \n   /  _/___  / __ \\____ _____ ____  / ____/___/ (_) /_            | |  / /__ \\\n   / // __ \\/ /_/ / __ `/ __ `/ _ \\/ __/ / __  / / __/  ______    | | / /__/ /\n _/ // / / / ____/ /_/ / /_/ /  __/ /___/ /_/ / / /_   /_____/    | |/ // __/ \n/___/_/ /_/_/    \\__,_/\\__, /\\___/_____/\\__,_/_/\\__/              |___//____/ \n                      /____/');
  });
}());

// </pre>