Site Notice

hello, world

Difference between revisions of "MediaWiki:Js-InPageEdit-canary.js"

From Project-EPB Commons
([InPageEdit] 没有编辑摘要)
([InPageEdit] 没有编辑摘要)
Line 4: Line 4:
 
  * GNU GENERAL PUBLIC LICENSE 3.0
 
  * GNU GENERAL PUBLIC LICENSE 3.0
 
  *
 
  *
  * MediaWiki JS Plugin: In Page Edit Canary Edition
+
  * MediaWiki JS Plugin: In Page Edit
  * Version: InPageEdit-canary
+
  * Version: See version-info file
 
  * Author: 机智的小鱼君
 
  * Author: 机智的小鱼君
 
  * Url:
 
  * Url:
Line 16: Line 16:
 
(function () {
 
(function () {
 
   'use strict';
 
   'use strict';
   if (typeof InPageEdit !== 'undefined') throw '[InPageEdit] 已经有一个IPE插件在执行了';
+
  // 防止多次载入
 +
   if (typeof InPageEdit !== 'undefined') throw '[InPageEdit] 已经有一个IPE插件在执行了。';
 +
  // 由于兼容性问题,阻止低版本平台
 +
  if (mw.config.get('wgVersion').split('.')[1] < 21) throw '[InPageEdit] 警告:InPageEdit暂不支持您所在的平台';
  
 
   // 创建全局函数
 
   // 创建全局函数
Line 26: Line 29:
  
 
   /** 获取版本信息 **/
 
   /** 获取版本信息 **/
   mw.loader.load('https://common.wjghj.cn/js/InPageEdit-v2.js/version-info-canary');
+
   mw.loader.load('https://common.wjghj.cn/js/InPageEdit-v2.js/version-info');
  
 
   /** 样式表 **/
 
   /** 样式表 **/
Line 36: Line 39:
 
   /*** BOT FLAG ***/
 
   /*** BOT FLAG ***/
 
   /** InPageEdit主框架 **/
 
   /** InPageEdit主框架 **/
 +
  // i18n
 +
  var i18n = {
 +
    'en': {
 +
      'foo': 'en'
 +
    },
 +
    'zh-hans': {
 +
      'foo': 'zh-hans'
 +
    }
 +
  }
 +
  function msg(i) {
 +
    var lang = mw.config.get('wgUserLanguage');
 +
    if (i18n.lang && i18n.lang.hasOwnProperty(i)) {
 +
      return i18n[lang][i];
 +
    } else if (i18n.en.hasOwnProperty(i)) {
 +
      return i18n.en[i];
 +
    } else {
 +
      return '&lt;' + i + '&gt;';
 +
    }
 +
  }
 
   /** 快速编辑模块 **/
 
   /** 快速编辑模块 **/
 
   InPageEdit.edit = function (option) {
 
   InPageEdit.edit = function (option) {
Line 81: Line 103:
 
       ssi_modal.notify('warning', {
 
       ssi_modal.notify('warning', {
 
         className: 'in-page-edit',
 
         className: 'in-page-edit',
         content: '您正在编辑页面的历史版本。',
+
         content: msg('notify-editing-history'),
         title: '提示'
+
         title: msg('notify-title-info')
 
       });
 
       });
 
       delete jsonGet.page;
 
       delete jsonGet.page;
 
       jsonGet.oldid = editRevision;
 
       jsonGet.oldid = editRevision;
       titleRevision = '<span style="font-size:small;">(历史版本:' + editRevision + ')</span>';
+
       titleRevision = '<span style="font-size:small;">(' + msg('editor-title-editRevision') + ':' + editRevision + ')</span>';
       summaryRevision = '(编辑自[[Special:Diff/' + editRevision + ']])';
+
       summaryRevision = '(' + msg('editor-summary-rivision') + '[[Special:Diff/' + editRevision + ']])';
 
     } else {
 
     } else {
 
       if (editSection !== undefined && editSection !== '' && editSection !== null) {
 
       if (editSection !== undefined && editSection !== '' && editSection !== null) {
 
         jsonGet.section = editSection;
 
         jsonGet.section = editSection;
         titleSection = '(' + editSection + '部分)';
+
         titleSection = msg('editor-title-editSection').replace('$1', editSection);
 
       }
 
       }
 
     }
 
     }
Line 119: Line 141:
 
     // 显示主窗口
 
     // 显示主窗口
 
     ssi_modal.show({
 
     ssi_modal.show({
       title: '正在编辑:<u class="editPage">' + editPage + '</u>' + titleSection + titleRevision,
+
       title: msg('editor-title-editing') + ': <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>',
+
       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">' + msg('editSummary') + ':</label><br/><input id="editSummary" class="editSummary" placeholder="Edit via InPageEdit~"/><br/><input id="editMinor" class="editMinor" type="checkbox" style=""/><label for="editMinor">' + msg('markAsSummary') + '</label></div></section>',
 
       outSideClose: outSideClose,
 
       outSideClose: outSideClose,
 
       className: 'in-page-edit ipe-editor timestamp-' + timestamp,
 
       className: 'in-page-edit ipe-editor timestamp-' + timestamp,
Line 133: Line 155:
 
             className: 'in-page-edit',
 
             className: 'in-page-edit',
 
             center: true,
 
             center: true,
             content: '是否保存',
+
             content: msg('editor-confirm-save'),
 
             okBtn: {
 
             okBtn: {
 
               className: 'btn btn-primary',
 
               className: 'btn btn-primary',
               label: '确定'
+
               label: msg('confirm')
 
             },
 
             },
 
             cancelBtn: {
 
             cancelBtn: {
 
               className: 'btn btn-secondary',
 
               className: 'btn btn-secondary',
               label: '取消'
+
               label: msg('cancel')
 
             },
 
             },
 
           },
 
           },
Line 160: Line 182:
 
         }
 
         }
 
       }, {
 
       }, {
         label: '显示预览',
+
         label: msg('editor-button-preview'),
 
         className: 'btn btn-secondary leftBtn editForm',
 
         className: 'btn btn-secondary leftBtn editForm',
 
         method: function () {
 
         method: function () {
Line 175: Line 197:
 
         }
 
         }
 
       }, {
 
       }, {
         label: '比较差异',
+
         label: msg('editor-button-diff'),
 
         className: 'btn btn-secondary leftBtn editForm',
 
         className: 'btn btn-secondary leftBtn editForm',
 
         method: function () {
 
         method: function () {
Line 189: Line 211:
 
         }
 
         }
 
       }, {
 
       }, {
         label: '查找替换',
+
         label: msg('editor-button-findAndReplace'),
 
         className: 'btn btn-secondary leftBtn editForm',
 
         className: 'btn btn-secondary leftBtn editForm',
 
         method: function () {
 
         method: function () {
Line 196: Line 218:
 
         }
 
         }
 
       }, {
 
       }, {
         label: '取消',
+
         label: msg('cancel'),
 
         className: 'btn btn-danger',
 
         className: 'btn btn-danger',
 
         method: function (e, modal) {
 
         method: function (e, modal) {
Line 208: Line 230:
 
         // 绑定事件,在尝试离开页面时提示
 
         // 绑定事件,在尝试离开页面时提示
 
         $(window).bind('beforeunload', function () {
 
         $(window).bind('beforeunload', function () {
           return '您输入的内容尚未保存,确定离开此页面吗?';
+
           return msg('window-leave-confirm');
 
         });
 
         });
 
         // 设置样式
 
         // 设置样式
Line 240: Line 262:
 
                       className: 'in-page-edit',
 
                       className: 'in-page-edit',
 
                       position: 'center bottom',
 
                       position: 'center bottom',
                       title: '权限不足',
+
                       title: msg('notify-no-right'),
                       content: '您没有编辑此页面的权限,请前往本页面的讨论页申请编辑。',
+
                       content: msg('editor-no-right'),
 
                       okBtn: {
 
                       okBtn: {
                         label: '好的',
+
                         label: msg('ok'),
 
                         className: 'btn btn-primary',
 
                         className: 'btn btn-primary',
 
                         method: function (e, modal) {
 
                         method: function (e, modal) {
Line 312: Line 334:
 
                     className: 'in-page-edit',
 
                     className: 'in-page-edit',
 
                     center: true,
 
                     center: true,
                     title: '编辑提示',
+
                     title: msg('editor-title-editNotice'),
 
                     content: editNotice
 
                     content: editNotice
 
                   });
 
                   });
 
                 })
 
                 })
                 .html('<i class="material-icons">info</i> 当前页面有编辑提示,点击查看')
+
                 .html('<i class="material-icons">info</i> ' + msg('editor-has-editNotice'))
 
             );
 
             );
 
           });
 
           });
Line 327: Line 349:
 
           className: 'in-page-edit',
 
           className: 'in-page-edit',
 
           center: true,
 
           center: true,
           content: '您输入的内容尚未保存,确定关闭窗口吗?',
+
           content: msg('editor-leave-confirm'),
 
           okBtn: {
 
           okBtn: {
 
             className: 'btn btn-danger',
 
             className: 'btn btn-danger',
             label: '确定'
+
             label: msg('confirm')
 
           },
 
           },
 
           cancelBtn: {
 
           cancelBtn: {
 
             className: 'btn btn-secondary',
 
             className: 'btn btn-secondary',
             label: '取消'
+
             label: msg('cancel')
 
           }
 
           }
 
         },
 
         },
Line 346: Line 368:
 
                 className: 'in-page-edit',
 
                 className: 'in-page-edit',
 
                 position: 'right top',
 
                 position: 'right top',
                 title: '操作取消',
+
                 title: msg('cancel'),
                 content: '没有进行任何改动。'
+
                 content: msg('notify-no-change')
 
               })
 
               })
 
             }
 
             }
Line 358: Line 380:
 
     function postArticle(pValue) {
 
     function postArticle(pValue) {
 
       InPageEdit.analysis({ type: 'functionCount', function: '保存编辑' });
 
       InPageEdit.analysis({ type: 'functionCount', function: '保存编辑' });
       InPageEdit.progress('正在发布编辑...');
+
       InPageEdit.progress(msg('editor-title-saving'));
 
       jsonPost = {
 
       jsonPost = {
 
         action: 'edit',
 
         action: 'edit',
Line 382: Line 404:
 
           className: 'in-page-edit',
 
           className: 'in-page-edit',
 
           position: 'right top',
 
           position: 'right top',
           title: '成功',
+
           title: msg('notify-success'),
           content: '成功,正在刷新页面。'
+
           content: msg('notify-save-success')
 
         });
 
         });
 
         setTimeout(function () {
 
         setTimeout(function () {
Line 400: Line 422:
 
             time: 15
 
             time: 15
 
           },
 
           },
           title: '警告',
+
           title: msg('notify-error'),
           content: '发布编辑时发生错误:<br/><span style="font-size:amall">' + errorThrown.errors[0]['*'] + '(<code>' + errorThrown.errors[0]['code'] + '</code>)</span>'
+
           content: msg('editor-save-error') + ':<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]['*']);
 
         console.error('[InPageEdit] Submit failed: \nCode: ' + errorThrown.errors[0]['code'] + '\nDescription: ' + errorThrown.errors[0]['*']);
        InPageEdit.error(errorThrown.errors[0]['code']);
 
 
       });
 
       });
 
     }
 
     }
Line 410: Line 431:
 
   }
 
   }
  
   /** 查找替换 **/
+
   /** 快速重定向模块 **/
 +
  InPageEdit.redirect = function (type) {
 +
    var json = {
 +
      action: 'edit',
 +
      minor: JSON.parse(localStorage.getItem('InPageEditPreference')).editMinor,
 +
      token: mw.user.tokens.get('editToken'),
 +
      errorformat: 'plaintext'
 +
    },
 +
      summary = msg('redirect-summary') + ' → [[:$1]]',
 +
      text = '#REDIRECT [[:$1]]',
 +
      question,
 +
      target;
 +
    switch (type) {
 +
      case 'to':
 +
        json.title = mw.config.get('wgPageName');
 +
        question = msg('redirect-question-to').replace('$1', '<b>' + mw.config.get('wgPageName') + '</b>');
 +
        break;
 +
      case 'from':
 +
        question = msg('redirect-question-from').replace('$1', '<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: msg('redirect-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: msg('confirm'),
 +
        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: msg('notify-redirect-success'),
 +
                title: msg('notify-success')
 +
              });
 +
              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: msg('notify-redirect-error'),
 +
                title: msg('notify-error')
 +
              });
 +
            });
 +
          }
 +
        }
 +
      }
 +
      ]
 +
    });
 +
  }
 +
 
 +
  /**
 +
  * 查找替换模块
 +
  * 部分代码借鉴 https://dev.fandom.com/wiki/MediaWiki:FindAndReplace/code.js
 +
  **/
 
   InPageEdit.findAndReplace = function (contengut) {
 
   InPageEdit.findAndReplace = function (contengut) {
 
     if (contengut === this.undefined) contengut = $('.in-page-edit.ipe-editor .editArea');
 
     if (contengut === this.undefined) contengut = $('.in-page-edit.ipe-editor .editArea');
Line 486: Line 595:
 
           }
 
           }
 
         }
 
         }
      ]
 
    });
 
  }
 
 
  /** 快速重定向模块 **/
 
  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: '失败'
 
              });
 
            });
 
          }
 
        }
 
      }
 
 
       ]
 
       ]
 
     });
 
     });
Line 579: Line 603:
 
     InPageEdit.analysis({ type: 'functionCount', function: '快速删除' });
 
     InPageEdit.analysis({ type: 'functionCount', function: '快速删除' });
 
     var reasonType,
 
     var reasonType,
       reason = '不再需要的页面';
+
       reason = msg('delete-reason-default');
 
     if (page === this.undefined) page = mw.config.get('wgPageName');
 
     if (page === this.undefined) page = mw.config.get('wgPageName');
  
Line 587: Line 611:
 
       center: true,
 
       center: true,
 
       sizeClass: 'dialog',
 
       sizeClass: 'dialog',
       title: '删除页面',
+
       title: msg('delete-title'),
 
       content: '<b style="color:#b00">本功能处于测试阶段,谨慎使用</b><section id="InPageEditDeletepage">您要将 <b>' + page + '</b> 删除的理由?<br/><i>自动填充</i><!-- <select id="reasonType" style="width: 23%;padding: 2px;margin: 2px;"><option value="破坏" selected="selected">这是一个破坏</option><option value="被标记">被标记为删除</option><option value="其他">其他</option></select><input id="reason" style="width: 73%;padding: 2px;margin: 2px;" value="' + reason + '"><br/> --></section>',
 
       content: '<b style="color:#b00">本功能处于测试阶段,谨慎使用</b><section id="InPageEditDeletepage">您要将 <b>' + page + '</b> 删除的理由?<br/><i>自动填充</i><!-- <select id="reasonType" style="width: 23%;padding: 2px;margin: 2px;"><option value="破坏" selected="selected">这是一个破坏</option><option value="被标记">被标记为删除</option><option value="其他">其他</option></select><input id="reason" style="width: 73%;padding: 2px;margin: 2px;" value="' + reason + '"><br/> --></section>',
 
       beforeShow: function () {
 
       beforeShow: function () {
 
         if (!InPageEdit.hasRight('delete')) {
 
         if (!InPageEdit.hasRight('delete')) {
 
           ssi_modal.dialog({
 
           ssi_modal.dialog({
             title: '权限不足',
+
             title: msg('notify-no-right'),
             content: '抱歉,您没有删除页面(<code>delete</code>)的权限。',
+
             content: msg('delete-no-right'),
 
             className: 'in-page-edit quick-deletepage',
 
             className: 'in-page-edit quick-deletepage',
 
             center: true,
 
             center: true,
Line 605: Line 629:
 
       buttons: [
 
       buttons: [
 
         {
 
         {
           label: '取消',
+
           label: msg('cancel'),
 
           className: 'btn btn-primary',
 
           className: 'btn btn-primary',
 
           method: function (e, modal) {
 
           method: function (e, modal) {
Line 611: Line 635:
 
           }
 
           }
 
         }, {
 
         }, {
           label: '确定',
+
           label: msg('confirm'),
 
           className: 'btn btn-danger',
 
           className: 'btn btn-danger',
 
           method: function (e, modal) {
 
           method: function (e, modal) {
Line 617: Line 641:
 
               center: true,
 
               center: true,
 
               className: 'in-page-edit',
 
               className: 'in-page-edit',
               title: '确认删除',
+
               title: msg('delete-confirm-title'),
               content: '请确认是否执行删除操作。',
+
               content: msg('delete-confirm-content'),
 
               okBtn: {
 
               okBtn: {
                 label: '删除',
+
                 label: msg('confirm'),
 
                 className: 'btn btn-danger'
 
                 className: 'btn btn-danger'
 
               },
 
               },
 
               cancelBtn: {
 
               cancelBtn: {
                 label: '取消',
+
                 label: msg('cancel'),
 
                 className: 'btn'
 
                 className: 'btn'
 
               }
 
               }
Line 636: Line 660:
 
                   ssi_modal.notify('success', {
 
                   ssi_modal.notify('success', {
 
                     className: 'in-page-edit',
 
                     className: 'in-page-edit',
                     title: '已删除',
+
                     title: msg('notify-success'),
                     content: '页面' + page + '已被删除。'
+
                     content: msg('notify-delete-success').replace('$1', page)
 
                   });
 
                   });
 
                 }).fail(function (errorCode, feedback, errorThrown) {
 
                 }).fail(function (errorCode, feedback, errorThrown) {
 
                   ssi_modal.notify('error', {
 
                   ssi_modal.notify('error', {
 
                     className: 'in-page-edit',
 
                     className: 'in-page-edit',
                     title: '未完成删除',
+
                     title: msg('notify-error'),
                     content: '提交删除时发生错误:<br/><span style="font-size:amall">' + errorThrown.errors[0]['*'] + '(<code>' + errorThrown.errors[0]['code'] + '</code>)</span>'
+
                     content: msg('notify-delete-error') + ': <br/><span style="font-size:amall">' + errorThrown.errors[0]['*'] + '(<code>' + errorThrown.errors[0]['code'] + '</code>)</span>'
 
                   });
 
                   });
 
                 });
 
                 });
Line 661: Line 685:
 
     var from = mw.config.get('wgPageName'),
 
     var from = mw.config.get('wgPageName'),
 
       to,
 
       to,
       reason = '重命名页面 → [[:$1]] //使用API进行的操作,请核查本次编辑',
+
       reason,
 
       movetalk,
 
       movetalk,
 
       movesubpages,
 
       movesubpages,
Line 672: Line 696:
 
       center: true,
 
       center: true,
 
       sizeClass: 'dialog',
 
       sizeClass: 'dialog',
       title: '重命名页面',
+
       title: msg('rename-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>',
+
       content: '<section id="InPageEditRename"><label for="move-to">' + msg('rename-moveTo').replace('$1', '<b>' + from + '</b>') + '</label><br/><input style="width:90%" id="move-to"><br/><input type="checkbox" id="movetalk" checked="checked"/><label for="movetalk">' + msg('rename-movetalk') + '</label><br/><input id="movesubpages" type="checkbox" checked="checked"/><label for="movesubpages">' + msg('rename-movesubpages') + '</label><br/><input id="noredirect" type="checkbox"/><label for="noredirect">' + msg('rename-noredirect') + '</label><br/><label for="move-reason">' + msg('editSummary') + '<br/><input style="width:90%" id="move-reason"/></label></section>',
 
       buttons: [{
 
       buttons: [{
 
         label: '取消',
 
         label: '取消',
Line 692: Line 716:
 
           noredirect = $('#noredirect').prop('checked');
 
           noredirect = $('#noredirect').prop('checked');
 
           to = $('#move-to').val();
 
           to = $('#move-to').val();
           reason = $('#move-reason').val().replace('$1', to);
+
           if ($('#move-reason').val() === '') {
 +
            reason = msg('rename-summary') + ' → [[:' + to + ']]';
 +
          } else {
 +
            reason = msg('rename-summary') + ' → [[:' + to + ']] (Reason: ' + $('#move-reason').val() + ')';
 +
          }
 
           new mw.Api().postWithToken('csrf', {
 
           new mw.Api().postWithToken('csrf', {
 
             action: 'move',
 
             action: 'move',
Line 705: Line 733:
 
             ssi_modal.notify('success', {
 
             ssi_modal.notify('success', {
 
               className: 'in-page-edit',
 
               className: 'in-page-edit',
               content: '正在刷新页面……',
+
               content: msg('notify-rename-success'),
               title: '成功'
+
               title: msg('notify-success')
 
             });
 
             });
 
             location.href = mw.config.get('wgArticlePath').replace('$1', to);
 
             location.href = mw.config.get('wgArticlePath').replace('$1', to);
Line 713: Line 741:
 
             ssi_modal.notify('error', {
 
             ssi_modal.notify('error', {
 
               className: 'in-page-edit',
 
               className: 'in-page-edit',
               content: '重命名时遇到问题:<br/>' + errorThrown.error.info + '<code>' + errorThrown.error.code + '</code>',
+
               content: msg('notify-rename-error') + ': ' + errorThrown.error.info + '<code>' + errorThrown.error.code + '</code>',
               title: '失败'
+
               title: msg('notify-error')
 
             });
 
             });
 
             if (errorThrown.error.code === 'articleexists') {
 
             if (errorThrown.error.code === 'articleexists') {
 
               ssi_modal.dialog({
 
               ssi_modal.dialog({
 
                 className: 'in-page-edit',
 
                 className: 'in-page-edit',
                 title: '目标页面已存在',
+
                 title: msg('rename-articleexists-title'),
 
                 center: true,
 
                 center: true,
                 content: '强制移动页面到已存在的标题需要利用删除功能,这个选项会在“快速删除”模块完成时添加,请暂时使用MediaWiki自带的移动并选择强制删除目标页面。',
+
                 content: msg('rename-articleexists'),
 
                 okBtn: {
 
                 okBtn: {
                   label: '好的',
+
                   label: msg('ok'),
 
                   className: 'btn btn-primary only-btn'
 
                   className: 'btn btn-primary only-btn'
 
                 }
 
                 }
Line 734: Line 762:
 
         if (!InPageEdit.hasRight('move')) {
 
         if (!InPageEdit.hasRight('move')) {
 
           ssi_modal.dialog({
 
           ssi_modal.dialog({
             title: '权限不足',
+
             title: msg('notify-no-right'),
             content: '抱歉,您没有移动页面(<code>move</code>)的权限。',
+
             content: msg('rename-no-right'),
 
             className: 'in-page-edit quick-deletepage',
 
             className: 'in-page-edit quick-deletepage',
 
             center: true,
 
             center: true,
Line 758: Line 786:
 
     ssi_modal.show({
 
     ssi_modal.show({
 
       outSideClose: false,
 
       outSideClose: false,
       title: 'InPageEdit偏好设定 - ' + InPageEdit.version,
+
       title: msg('preference-title') + ' - ' + 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>',
+
       content: '<section id="InPageEditSettingBox"><b>InPageEdit编辑器</b><br/><input id="ipeSetoutSideClose" type="checkbox"/> <label for="ipeSetoutSideClose">' + msg('preference-outSideClose') + '</label><br/><input id="ipeSetMinor" type="checkbox"/> <label for="ipeSetMinor">' + msg('preference-setMinor') + '</label><br/><b>' + msg('preference-setMinor') + '</b><br/><label>' + msg('preference-editSummary') + '<br/><span style="font-size:10px"></span><input id="ipeSetSummary" value="' + summary + '" style="width:100%"/></label><br/><b>' + msg('preference-analysis') + '</b><br/><span style="font-size:10px">' + msg('preference-analysis-view').replace('$1', '<a href="https://doc.wjghj.cn/InPageEditAnalysis/" target="_blank">https://doc.wjghj.cn/InPageEditAnalysis/</a>') + '</span><br/><b>' + msg('preference-about') + '</b><br/><button class="btn btn-secondary" onclick="mw.loader.load(\'https://common.wjghj.cn/js/InPageEdit-v2.js/about\')">' + msg('preference-aboutAndHelp') + '</button>&nbsp;<button class="btn btn-secondary" onclick="InPageEdit.versionInfo()">' + msg('preference-updatelog') + '</button><hr><span style="font-size:10px;line-height:5px">' + msg('preference-savelocal-label') + '<br/>' + msg('preference-savelocal') + '<a href="javascript:;" id="ipeSaveLocalShow">' + msg('preference-savelocal-btn') + '</a></span></section>',
 
       sizeClass: 'dialog',
 
       sizeClass: 'dialog',
 
       className: 'in-page-edit ipe-preference',
 
       className: 'in-page-edit ipe-preference',
 
       center: true,
 
       center: true,
 
       buttons: [{
 
       buttons: [{
         label: '重置',
+
         label: msg('preference-reset'),
 
         className: 'btn btn-danger',
 
         className: 'btn btn-danger',
 
         method: function () {
 
         method: function () {
 
           $('#InPageEditSettingBox #ipeSetoutSideClose').prop('checked', true);
 
           $('#InPageEditSettingBox #ipeSetoutSideClose').prop('checked', true);
 
           $('#InPageEditSettingBox #ipeSetMinor').prop('checked', false);
 
           $('#InPageEditSettingBox #ipeSetMinor').prop('checked', false);
           $('#InPageEditSettingBox #ipeSetSummary').val('[InPageEdit] 没有编辑摘要 $section$oldid');
+
           $('#InPageEditSettingBox #ipeSetSummary').val(msg('preference-summary-default'));
 
         }
 
         }
 
       }, {
 
       }, {
         label: '保存',
+
         label: msg('preference-save'),
 
         className: 'btn btn-primary',
 
         className: 'btn btn-primary',
 
         method: function (a, modal) {
 
         method: function (a, modal) {
Line 790: Line 818:
 
         className: 'in-page-edit',
 
         className: 'in-page-edit',
 
         center: true,
 
         center: true,
         content: '<section id="ipeSaveLocal"><b>不推荐这种做法</b><br/>在您个人js页调用本插件的代码的上方添加以下代码:<br/><input style="width:100%" readonly="readonly" onclick="$(this).select()"/><br/>注意:虽然这样做免去了更换设备时需要重新进行设置的麻烦,但我们没有参数纠错功能,您必须自行确认json设定是否正确。</section>',
+
        title: msg('preference-savelocal-popup-title'),
 +
         content: '<section id="ipeSaveLocal"><b>' + msg('preference-savelocal-popup-notrecommended') + '</b><br/>' + msg('preference-savelocal-popup') + '<br/><input style="width:100%" readonly="readonly" onclick="$(this).select()"/><br/>' + msg('preference-savelocal-popup-notice') + '</section>',
 
         okBtn: {
 
         okBtn: {
 
           className: 'btn btn-primary btn-single'
 
           className: 'btn btn-primary btn-single'
Line 800: Line 829:
 
         editSummary: $('#InPageEditSettingBox #ipeSetSummary').val()
 
         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) {
 
     if (outSideClose) {
Line 845: Line 849:
 
       }
 
       }
 
       ssi_modal.dialog({
 
       ssi_modal.dialog({
         content: '您使用自己的个人js保存了设定参数,请浏览<a href="' + mw.config.get('wgArticlePath').replace('$1', 'Special:Mypage/common.js') + '">您的个人js页</a>修改。<br/>我们不推荐保存静态的设定,虽然这样做免去了更换设备时需要重新进行设置的麻烦,但因为暂时没有参数纠错功能,您必须自行确认json设定是否正确。',
+
         content: msg('preference-savelocal-popup-haslocal').replace('$1', '<a href="' + mw.util.getUrl('Special:Mypage/common.js') + '">' + msg('preference-savelocal-popup-yourjspage') + '</a>'),
 
         className: 'in-page-edit',
 
         className: 'in-page-edit',
 
         center: true,
 
         center: true,
Line 865: Line 869:
 
     if ($('.quick-diff').length > 0) {
 
     if ($('.quick-diff').length > 0) {
 
       console.info('[InPageEdit] Quick diff 正在加载新内容');
 
       console.info('[InPageEdit] Quick diff 正在加载新内容');
       $('.in-page-edit.quick-diff .diffArea').hide().html('加载中');
+
       $('.in-page-edit.quick-diff .diffArea').hide().html(msg('diff-loading'));
 
       if (param.isPreview) {
 
       if (param.isPreview) {
 
         $('.quick-diff').appendTo('body');
 
         $('.quick-diff').appendTo('body');
Line 875: Line 879:
 
         fixedHeight: true,
 
         fixedHeight: true,
 
         fitScreen: true,
 
         fitScreen: true,
         title: '<span class="pageName">正在加载差异……</span>',
+
         title: '<span class="pageName">' + msg('diff-loading') + '</span>',
 
         content: '<div class="ipe-progress" style="width:100%"><div class="ipe-progress-bar"></div></div><div class="diffArea"></div>',
 
         content: '<div class="ipe-progress" style="width:100%"><div class="ipe-progress-bar"></div></div><div class="diffArea"></div>',
 
         buttons: [{
 
         buttons: [{
           label: '转到原版比较页面',
+
           label: msg('diff-button-todiffpage'),
 
           className: 'btn btn-secondary toDiffPage',
 
           className: 'btn btn-secondary toDiffPage',
 
           method: function () {
 
           method: function () {
Line 900: Line 904:
 
       }
 
       }
 
       var userlink = function (user) {
 
       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> )';
+
         return '<a href="' + mw.util.getUrl('User:' + user) + '">' + user + '</a> ( <a href="' + mw.util.getUrl('User_talk:' + user) + '">' + msg('diff-usertalk') + '</a> | <a href="' + mw.util.getUrl('Special:Contributions/' + user) + '">' + msg('diff-usercontrib') + '</a> | <a href="' + mw.util.getUrl('Special:Block/' + user) + '">' + msg('diff-userblock') + '</a> )';
 
       }
 
       }
       $('.quick-diff .pageName').html('比较差异:<u>' + toTitle + '</u>');
+
       $('.quick-diff .pageName').html(msg('diff-title') + ': <u>' + toTitle + '</u>');
 
       $('.quick-diff .diffArea').show().html(
 
       $('.quick-diff .diffArea').show().html(
 
         '<table class="diff diffTable">' +
 
         '<table class="diff diffTable">' +
Line 914: Line 918:
 
         '<tr class="diff-title">' +
 
         '<tr class="diff-title">' +
 
         '<td colspan="2" class="diff-otitle">' +
 
         '<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>' +
+
         '<a class="" href="' + mw.config.get('wgScript') + '?oldid=' + data.compare.fromrevid + '">' + data.compare.fromtitle + '</a> (' + msg('diff-version') + data.compare.fromrevid + ') (<a class="editLink" href="' + mw.config.get('wgScript') + '?action=edit&title=' + data.compare.fromtitle + '&oldid=' + data.compare.fromrevid + '">' + msg('diff-edit') + '</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\'});">←' + msg('diff-prev') + '</a>' +
 
         '</td>' +
 
         '</td>' +
 
         '<td colspan="2" class="diff-ntitle">' +
 
         '<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>' +
+
         '<a class="" href="' + mw.config.get('wgScript') + '?oldid=' + data.compare.torevid + '">' + data.compare.totitle + '</a> (' + msg('diff-version') + data.compare.torevid + ') (<a class="editLink" href="' + mw.config.get('wgScript') + '?action=edit&title=' + data.compare.totitle + '&oldid=' + data.compare.torevid + '">' + msg('diff-edit') + '</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\'});">' + msg('diff-nextv') + '→</a>' +
 
         '</td>' +
 
         '</td>' +
 
         '</tr>' +
 
         '</tr>' +
 
         diffTable +
 
         diffTable +
         '<tr class="diffSize" style="text-align: center;"><td colspan="2">' + data.compare.fromsize + '字节</td><td colspan="2">' + data.compare.tosize + '字节</td></tr>' +
+
         '<tr class="diffSize" style="text-align: center;"><td colspan="2">' + data.compare.fromsize + msg('diff-bytes') + '</td><td colspan="2">' + data.compare.tosize + msg('diff-bytes') + '</td></tr>' +
 
         '</tbody>' +
 
         '</tbody>' +
 
         '</table>'
 
         '</table>'
Line 931: Line 935:
 
       if (param.isPreview === true) {
 
       if (param.isPreview === true) {
 
         $('.quick-diff button.toDiffPage').hide();
 
         $('.quick-diff button.toDiffPage').hide();
         $('.quick-diff .diff-otitle').html('<b>原始内容</b>');
+
         $('.quick-diff .diff-otitle').html('<b>' + msg('diff-title-original-content') + '</b>');
         $('.quick-diff .diff-ntitle').html('<b>您的编辑</b>');
+
         $('.quick-diff .diff-ntitle').html('<b>' + msg('diff-title-your-content') + '</b>');
 
       }
 
       }
 
       if (data.compare.fromsize === undefined || data.compare.tosize === undefined) {
 
       if (data.compare.fromsize === undefined || data.compare.tosize === undefined) {
Line 944: Line 948:
 
     }).fail(function (a, b, c) {
 
     }).fail(function (a, b, c) {
 
       $('.in-page-edit.quick-diff .ipe-progress').hide();
 
       $('.in-page-edit.quick-diff .ipe-progress').hide();
       $('.diffArea').html('比较差异时出现错误: ' + b);
+
       $('.diffArea').html(msg('diff-error') + ': ' + b);
 
     });
 
     });
 
   }
 
   }
Line 978: Line 982:
 
     if (mw.config.get('wgAction') === 'history') {
 
     if (mw.config.get('wgAction') === 'history') {
 
       $('.historysubmit.mw-history-compareselectedversions-button').after(
 
       $('.historysubmit.mw-history-compareselectedversions-button').after(
         $('<button>').text('快速对比差异').click(function (e) {
+
         $('<button>').text(msg('quick-diff')).click(function (e) {
 
           e.preventDefault();
 
           e.preventDefault();
 
           InPageEdit.analysis({ type: 'functionCount', function: '快速差异History' });
 
           InPageEdit.analysis({ type: 'functionCount', function: '快速差异History' });
Line 990: Line 994:
 
           oldid = $this.attr('data-mw-revid');
 
           oldid = $this.attr('data-mw-revid');
 
         $this.find('.mw-history-undo').after(
 
         $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>')
+
           $('<span>').html(' | <a class="in-page-edit-article-link" href="javascript:void(0);" onclick="InPageEdit.edit({page:mw.config.get(\'wgPageName\'),revision:' + oldid + '});">' + msg('quick-edit') + '</a>')
 
         );
 
         );
 
       });
 
       });
Line 1,030: Line 1,034:
 
               'class': 'in-page-edit-article-link'
 
               'class': 'in-page-edit-article-link'
 
             })
 
             })
               .text('快速编辑')
+
               .text(msg('quick-edit'))
 
               .click(function () {
 
               .click(function () {
 
                 if (revision !== null) {
 
                 if (revision !== null) {
Line 1,055: Line 1,059:
 
       sizeClass: 'large',
 
       sizeClass: 'large',
 
       className: 'in-page-edit previewbox',
 
       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>',
+
       content: '<div class="ipe-progress" style="width:100%"><div class="ipe-progress-bar"></div></div><section id="InPageEditPreview" data-timestamp="' + timestamp + '" style="display:none">' + msg('preview-placeholder') + '</section>',
       title: '预览',
+
       title: msg('preview-title'),
 
       fixedHeight: true,
 
       fixedHeight: true,
 
       fitScreen: true,
 
       fitScreen: true,
Line 1,072: Line 1,076:
 
           console.warn('[InPageEdit] 预览失败');
 
           console.warn('[InPageEdit] 预览失败');
 
           $('.previewbox .ipe-progress').hide(150);
 
           $('.previewbox .ipe-progress').hide(150);
           $('#InPageEditPreview[data-timestamp="' + timestamp + '"]').fadeIn(500).html('获取预览时发生错误!');
+
           $('#InPageEditPreview[data-timestamp="' + timestamp + '"]').fadeIn(500).html(msg('preview-error'));
 
         });
 
         });
 
       }
 
       }
Line 1,081: Line 1,085:
 
   InPageEdit.progress = function (title) {
 
   InPageEdit.progress = function (title) {
 
     if (title === true) {
 
     if (title === true) {
       $('.in-page-edit.loadingbox .ssi-modalTitle').html('完成');
+
       $('.in-page-edit.loadingbox .ssi-modalTitle').html(msg('done'));
 
       $('.in-page-edit.loadingbox .ipe-progress').addClass('done');
 
       $('.in-page-edit.loadingbox .ipe-progress').addClass('done');
 
     } else if (title === false) {
 
     } else if (title === false) {
Line 1,091: Line 1,095:
 
       if ($('.in-page-edit.loadingbox').length > 0) return;
 
       if ($('.in-page-edit.loadingbox').length > 0) return;
 
       if (typeof (title) === 'undefined') {
 
       if (typeof (title) === 'undefined') {
         title = '加载中...'
+
         title = 'Loading...'
 
       }
 
       }
 
       ssi_modal.show({
 
       ssi_modal.show({
Line 1,105: Line 1,109:
 
   }
 
   }
  
  /** 常见错误集 **/
 
  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) {
 
   InPageEdit.analysis = function (params) {
Line 1,189: Line 1,145:
 
         break;
 
         break;
 
     }
 
     }
   };
+
   }
  
  /** 获取用户权限信息 **/
+
    /** 获取用户权限信息 **/
  (function () {
+
    (function () {
    var user = mw.config.get('wgUserName');
+
      var user = mw.config.get('wgUserName');
    if (user === null) {
+
      if (user === null) {
      console.warn('[InPageEdit] 警告:用户未登录');
+
        console.warn('[InPageEdit] 警告:用户未登录');
      mw.config.set('wgUserRights', '');
+
        mw.config.set('wgUserRights', '');
      return;
+
        return;
    }
+
      }
    new mw.Api().get({
+
      new mw.Api().get({
      action: 'query',
+
        action: 'query',
      list: 'users',
+
        list: 'users',
      usprop: 'rights',
+
        usprop: 'rights',
      ususers: user
+
        ususers: user
    }).done(function (data) {
+
      }).done(function (data) {
      console.info('[InPageEdit] 成功获取用户权限信息');
+
        console.info('[InPageEdit] 成功获取用户权限信息');
      mw.config.set('wgUserRights', data.query.users[0]['rights']);
+
        mw.config.set('wgUserRights', data.query.users[0]['rights']);
    }).fail(function () {
+
      }).fail(function () {
      console.warn('[InPageEdit] 警告:无法获取用户权限信息');
+
        console.warn('[InPageEdit] 警告:无法获取用户权限信息');
      mw.config.set('wgUserRights', '');
+
        mw.config.set('wgUserRights', '');
    });
+
      });
  })();
+
    }());
 
   InPageEdit.hasRight = function (right) {
 
   InPageEdit.hasRight = function (right) {
 
     if (mw.config.get('wgUserRights').indexOf(right) > -1) {
 
     if (mw.config.get('wgUserRights').indexOf(right) > -1) {
Line 1,218: Line 1,174:
 
       return false;
 
       return false;
 
     }
 
     }
   }
+
   };
 
   /** 初始化 **/
 
   /** 初始化 **/
   $(function () {
+
   (function () {
    // 由于兼容性问题,阻止低版本平台
 
    if (mw.config.get('wgVersion').split('.')[1] < 21) {
 
      $('.ssi-stack').remove();
 
      console.error('%c[InPageEdit] 警告:InPageEdit暂不支持您所在的平台。', 'color:darkred;font-size:1.2em;font-weight:bold');
 
      return;
 
    }
 
 
     /** 额外的模块 **/
 
     /** 额外的模块 **/
 
     // 快速页面差异模块
 
     // 快速页面差异模块
Line 1,244: Line 1,194:
 
       ipePreference.outSideClose = true;
 
       ipePreference.outSideClose = true;
 
       ipePreference.editMinor = false;
 
       ipePreference.editMinor = false;
       ipePreference.editSummary = '[InPageEdit] 没有编辑摘要 $section$oldid';
+
       ipePreference.editSummary = msg('preference-summary-default');
 
       localStorage.setItem('InPageEditPreference', JSON.stringify(ipePreference));
 
       localStorage.setItem('InPageEditPreference', JSON.stringify(ipePreference));
 
     }
 
     }
Line 1,254: Line 1,204:
 
       // group1 上方的一列按钮
 
       // group1 上方的一列按钮
 
       '<ul class="btn-group 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">' + msg('quick-edit') + '</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">' + msg('redirect-from') + '</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>' +
+
       '<li class="btn-tip-group"><div class="btn-tip">' + msg('redirect-to') + '</div><button id="redirectto-btn" class="ipe-toolbox-btn material-icons">flight_takeoff</button></li>' +
 
       '</ul>' +
 
       '</ul>' +
  
 
       // group2 左边的一排按钮
 
       // group2 左边的一排按钮
 
       '<ul class="btn-group group2"><div style="display: flex;">' +
 
       '<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">' + msg('quick-delete') + '</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">' + msg('quick-rename') + '</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>' +
+
       '<li class="btn-tip-group"><div class="btn-tip">' + msg('ipe-preference') + '</div><button id="preference-btn" class="ipe-toolbox-btn material-icons">settings</button></li>' +
 
       '</div></ul>' +
 
       '</div></ul>' +
  
Line 1,304: Line 1,254:
 
     // 花里胡哨的加载提示
 
     // 花里胡哨的加载提示
 
     console.info('    ____      ____                  ______    ___ __              _    _____ \n  /  _/___  / __ \\____ _____ ____  / ____/___/ (_) /_            | |  / /__ \\\n  / // __ \\/ /_/ / __ `/ __ `/ _ \\/ __/ / __  / / __/  ______    | | / /__/ /\n _/ // / / / ____/ /_/ / /_/ /  __/ /___/ /_/ / / /_  /_____/    | |/ // __/ \n/___/_/ /_/_/    \\__,_/\\__, /\\___/_____/\\__,_/_/\\__/              |___//____/ \n                      /____/');
 
     console.info('    ____      ____                  ______    ___ __              _    _____ \n  /  _/___  / __ \\____ _____ ____  / ____/___/ (_) /_            | |  / /__ \\\n  / // __ \\/ /_/ / __ `/ __ `/ _ \\/ __/ / __  / / __/  ______    | | / /__/ /\n _/ // / / / ____/ /_/ / /_/ /  __/ /___/ /_/ / / /_  /_____/    | |/ // __/ \n/___/_/ /_/_/    \\__,_/\\__, /\\___/_____/\\__,_/_/\\__/              |___//____/ \n                      /____/');
   });
+
   }());
 
}());
 
}());

Revision as of 20:42, 25 March 2020

/**
 *『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/InPageEdit-v2
 ** https://common.wjghj.cn/wiki/InPageEdit-v2
 * Logs:
 ** https://common.wjghj.cn/wiki/InPageEdit-v2/version-info
 **/

(function () {
  'use strict';
  // 防止多次载入
  if (typeof InPageEdit !== 'undefined') throw '[InPageEdit] 已经有一个IPE插件在执行了。';
  // 由于兼容性问题,阻止低版本平台
  if (mw.config.get('wgVersion').split('.')[1] < 21) throw '[InPageEdit] 警告:InPageEdit暂不支持您所在的平台';

  // 创建全局函数
  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主框架 **/
  // i18n
  var i18n = {
    'en': {
      'foo': 'en'
    },
    'zh-hans': {
      'foo': 'zh-hans'
    }
  }
  function msg(i) {
    var lang = mw.config.get('wgUserLanguage');
    if (i18n.lang && i18n.lang.hasOwnProperty(i)) {
      return i18n[lang][i];
    } else if (i18n.en.hasOwnProperty(i)) {
      return i18n.en[i];
    } else {
      return '&lt;' + i + '&gt;';
    }
  }
  /** 快速编辑模块 **/
  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: msg('notify-editing-history'),
        title: msg('notify-title-info')
      });
      delete jsonGet.page;
      jsonGet.oldid = editRevision;
      titleRevision = '<span style="font-size:small;">(' + msg('editor-title-editRevision') + ':' + editRevision + ')</span>';
      summaryRevision = '(' + msg('editor-summary-rivision') + '[[Special:Diff/' + editRevision + ']])';
    } else {
      if (editSection !== undefined && editSection !== '' && editSection !== null) {
        jsonGet.section = editSection;
        titleSection = msg('editor-title-editSection').replace('$1', 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: msg('editor-title-editing') + ': <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">' + msg('editSummary') + ':</label><br/><input id="editSummary" class="editSummary" placeholder="Edit via InPageEdit~"/><br/><input id="editMinor" class="editMinor" type="checkbox" style=""/><label for="editMinor">' + msg('markAsSummary') + '</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',
            center: true,
            content: msg('editor-confirm-save'),
            okBtn: {
              className: 'btn btn-primary',
              label: msg('confirm')
            },
            cancelBtn: {
              className: 'btn btn-secondary',
              label: msg('cancel')
            },
          },
            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: msg('editor-button-preview'),
        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: msg('editor-button-diff'),
        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: msg('editor-button-findAndReplace'),
        className: 'btn btn-secondary leftBtn editForm',
        method: function () {
          InPageEdit.analysis({ type: 'functionCount', function: '查找替换' });
          InPageEdit.findAndReplace($('.ipe-editor.timestamp-' + timestamp + ' .editArea'));
        }
      }, {
        label: msg('cancel'),
        className: 'btn btn-danger',
        method: function (e, modal) {
          modal.close();
        }
      }
      ],

      /* 开始执行后续程序 */
      onShow: function (modal) {
        // 绑定事件,在尝试离开页面时提示
        $(window).bind('beforeunload', function () {
          return msg('window-leave-confirm');
        });
        // 设置样式
        $('.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', {
                      className: 'in-page-edit',
                      position: 'center bottom',
                      title: msg('notify-no-right'),
                      content: msg('editor-no-right'),
                      okBtn: {
                        label: msg('ok'),
                        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));
        // 获取编辑提示
        $.get(mw.config.get('wgScript'), {
          action: 'raw',
          title: 'MediaWiki:Editnotice-' + mw.config.get('wgNamespaceNumber')
        }, function (data) {
          new mw.Api().post({
            action: 'parse',
            title: editPage.replace(/\.(js|css|json)/g, '@Dot@$1'),
            preview: true,
            text: data
          }).then(function (data) {
            editNotice = '<section class="editNotice">' + data.parse.text['*'].replace(/\@Dot\@/g, '.') + '</section>';
            $('.ipe-editor.timestamp-' + timestamp + ' .ssi-modalTitle').append(
              $('<a>')
                .attr({
                  id: 'showEditNotice',
                  href: 'javascript:;'
                })
                .click(function () {
                  ssi_modal.show({
                    className: 'in-page-edit',
                    center: true,
                    title: msg('editor-title-editNotice'),
                    content: editNotice
                  });
                })
                .html('<i class="material-icons">info</i> ' + msg('editor-has-editNotice'))
            );
          });
        });
      },

      /* 确认是否取消 */
      beforeClose: function (modal) {
        ssi_modal.confirm({
          className: 'in-page-edit',
          center: true,
          content: msg('editor-leave-confirm'),
          okBtn: {
            className: 'btn btn-danger',
            label: msg('confirm')
          },
          cancelBtn: {
            className: 'btn btn-secondary',
            label: msg('cancel')
          }
        },
          function (result) {
            if (result === true) {
              $(window).unbind('beforeunload');
              modal.options.keepContent = false;
              modal.options.beforeClose = '';
              modal.close();
              ssi_modal.notify('info', {
                className: 'in-page-edit',
                position: 'right top',
                title: msg('cancel'),
                content: msg('notify-no-change')
              })
            }
          });
        return false;
      }
    });

    // 发布编辑模块
    function postArticle(pValue) {
      InPageEdit.analysis({ type: 'functionCount', function: '保存编辑' });
      InPageEdit.progress(msg('editor-title-saving'));
      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: msg('notify-success'),
          content: msg('notify-save-success')
        });
        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: msg('notify-error'),
          content: msg('editor-save-error') + ':<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.redirect = function (type) {
    var json = {
      action: 'edit',
      minor: JSON.parse(localStorage.getItem('InPageEditPreference')).editMinor,
      token: mw.user.tokens.get('editToken'),
      errorformat: 'plaintext'
    },
      summary = msg('redirect-summary') + ' → [[:$1]]',
      text = '#REDIRECT [[:$1]]',
      question,
      target;
    switch (type) {
      case 'to':
        json.title = mw.config.get('wgPageName');
        question = msg('redirect-question-to').replace('$1', '<b>' + mw.config.get('wgPageName') + '</b>');
        break;
      case 'from':
        question = msg('redirect-question-from').replace('$1', '<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: msg('redirect-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: msg('confirm'),
        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: msg('notify-redirect-success'),
                title: msg('notify-success')
              });
              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: msg('notify-redirect-error'),
                title: msg('notify-error')
              });
            });
          }
        }
      }
      ]
    });
  }

  /** 
   * 查找替换模块
   * 部分代码借鉴 https://dev.fandom.com/wiki/MediaWiki:FindAndReplace/code.js
   **/
  InPageEdit.findAndReplace = function (contengut) {
    if (contengut === this.undefined) contengut = $('.in-page-edit.ipe-editor .editArea');
    var origin = contengut.val();

    ssi_modal.show({
      className: 'in-page-edit',
      title: '查找和替换',
      content: '<div class="module far-module"><div class="module_content" id="findfielddiv"><div>查找文本<div style="padding-top:3px;"><textarea id="find_this" style="margin: 0; width:100%;" rows="4" wrap="off"></textarea></div></div><div style="padding-top:8px;">替换文本</div><div style="padding-top:3px;"><textarea id="replace_with" style="margin: 0; width:100%;" rows="4" wrap="off" placeholder=""></textarea></div><div style="padding:7px 0px 7px 0px;"><table><tr><td><input type="checkbox" id="globl" checked disabled><label for="globl">全局匹配(暂时只能全局替换)</label></td></tr><tr><td><input type="checkbox" id="case_sen"><label for="case_sen">区分大小写</label></td></tr><tr><td><input type="checkbox" id="regex_search"><label for="regex_search">启用正则表达式</label></td></tr><tr><td><span id="far-found"></span></td></tr></table></div></div></div>',
      buttons: [
        {
          label: '还原',
          className: 'btn btn-danger',
          method: function (e, modal) {
            contengut.val(origin);
            ssi_modal.notify('info', {
              title: '还原替换的文字',
              content: '已将编辑框的内容还原为本次开启窗口前的内容。'
            });
            // modal.close();
          }
        },
        {
          className: 'btn btn-primary',
          label: '查找并替换',
          method: function (nothing, modal) {
            /**
             * 查找替换主函数
             * 借鉴:https://dev.fandom.com/wiki/MediaWiki:FindAndReplace/code.js
             **/
            if ($('#find_this').val() === '') {
              ssi_modal.notify('info', {
                title: '查找替换失败',
                content: '请输入需要查找的内容。'
              });
              return;
            }
            var searchfor = '',
              searchexp,
              $textarea = contengut,
              replacewith = $('#replace_with').val().replace(/\r/gi, ''),
              text = $textarea.val().replace(/\r/gi, ''),
              flagg = 'g',
              flagi = 'i',
              enableregex = 0;

            if ($('#globl').prop('checked') === false) {
              flagg = '';
            }
            if ($('#case_sen').prop('checked') === true) {
              flagi = '';
            }
            if ($('#regex_search').prop('checked') === true) {
              enableregex = 1;
            }
            var flags = flagg + flagi + 'm';
            if (enableregex === 1) {
              searchfor = $('#find_this').val();
            } else {
              searchfor = $('#find_this').val().replace(/\r/gi, '').replace(/([.*+?^=!:${}()|\[\]\/\\])/g, '\\$1');
            }
            searchexp = new RegExp(searchfor, flags);
            var rcount = 0;
            var matched = text.match(searchexp);
            if (matched !== null) {
              rcount = matched.length;
            }
            text = text.replace(searchexp, replacewith);
            $textarea.val(text);
            ssi_modal.notify('info', {
              title: '查找并替换',
              content: '已替换' + rcount + '处文本。'
            });
            // modal.close();
          }
        }
      ]
    });
  }

  /** 删除页面模块 **/
  InPageEdit.deletepage = function (page) {
    InPageEdit.analysis({ type: 'functionCount', function: '快速删除' });
    var reasonType,
      reason = msg('delete-reason-default');
    if (page === this.undefined) page = mw.config.get('wgPageName');

    ssi_modal.show({
      outSideClose: false,
      className: 'in-page-edit quick-delete',
      center: true,
      sizeClass: 'dialog',
      title: msg('delete-title'),
      content: '<b style="color:#b00">本功能处于测试阶段,谨慎使用</b><section id="InPageEditDeletepage">您要将 <b>' + page + '</b> 删除的理由?<br/><i>自动填充</i><!-- <select id="reasonType" style="width: 23%;padding: 2px;margin: 2px;"><option value="破坏" selected="selected">这是一个破坏</option><option value="被标记">被标记为删除</option><option value="其他">其他</option></select><input id="reason" style="width: 73%;padding: 2px;margin: 2px;" value="' + reason + '"><br/> --></section>',
      beforeShow: function () {
        if (!InPageEdit.hasRight('delete')) {
          ssi_modal.dialog({
            title: msg('notify-no-right'),
            content: msg('delete-no-right'),
            className: 'in-page-edit quick-deletepage',
            center: true,
            okBtn: {
              className: 'btn btn-primary btn-single'
            }
          });
          return false;
        }
      },
      buttons: [
        {
          label: msg('cancel'),
          className: 'btn btn-primary',
          method: function (e, modal) {
            modal.close();
          }
        }, {
          label: msg('confirm'),
          className: 'btn btn-danger',
          method: function (e, modal) {
            ssi_modal.confirm({
              center: true,
              className: 'in-page-edit',
              title: msg('delete-confirm-title'),
              content: msg('delete-confirm-content'),
              okBtn: {
                label: msg('confirm'),
                className: 'btn btn-danger'
              },
              cancelBtn: {
                label: msg('cancel'),
                className: 'btn'
              }
            }, function (result) {
              if (result) {
                var fiNalReason = '[' + $('#InPageEditDeletepage #reasonType').val() + '] ' + $('#InPageEditDeletepage #reason').val();
                new mw.Api().postWithToken('csrf', {
                  action: 'delete',
                  title: page
                }).then(function (data) {
                  ssi_modal.notify('success', {
                    className: 'in-page-edit',
                    title: msg('notify-success'),
                    content: msg('notify-delete-success').replace('$1', page)
                  });
                }).fail(function (errorCode, feedback, errorThrown) {
                  ssi_modal.notify('error', {
                    className: 'in-page-edit',
                    title: msg('notify-error'),
                    content: msg('notify-delete-error') + ': <br/><span style="font-size:amall">' + errorThrown.errors[0]['*'] + '(<code>' + errorThrown.errors[0]['code'] + '</code>)</span>'
                  });
                });
                modal.close();
              } else {
                return false;
              }
            });
          }
        }
      ]
    });
  }

  /** 重命名模块 **/
  InPageEdit.renamepage = function () {
    var from = mw.config.get('wgPageName'),
      to,
      reason,
      movetalk,
      movesubpages,
      noredirect,
      ignorewarnings;

    ssi_modal.show({
      outSideClose: false,
      className: 'in-page-edit quick-rename',
      center: true,
      sizeClass: 'dialog',
      title: msg('rename-title'),
      content: '<section id="InPageEditRename"><label for="move-to">' + msg('rename-moveTo').replace('$1', '<b>' + from + '</b>') + '</label><br/><input style="width:90%" id="move-to"><br/><input type="checkbox" id="movetalk" checked="checked"/><label for="movetalk">' + msg('rename-movetalk') + '</label><br/><input id="movesubpages" type="checkbox" checked="checked"/><label for="movesubpages">' + msg('rename-movesubpages') + '</label><br/><input id="noredirect" type="checkbox"/><label for="noredirect">' + msg('rename-noredirect') + '</label><br/><label for="move-reason">' + msg('editSummary') + '<br/><input style="width:90%" 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();
          if ($('#move-reason').val() === '') {
            reason = msg('rename-summary') + ' → [[:' + to + ']]';
          } else {
            reason = msg('rename-summary') + ' → [[:' + to + ']] (Reason: ' + $('#move-reason').val() + ')';
          }
          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: msg('notify-rename-success'),
              title: msg('notify-success')
            });
            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: msg('notify-rename-error') + ': ' + errorThrown.error.info + '<code>' + errorThrown.error.code + '</code>',
              title: msg('notify-error')
            });
            if (errorThrown.error.code === 'articleexists') {
              ssi_modal.dialog({
                className: 'in-page-edit',
                title: msg('rename-articleexists-title'),
                center: true,
                content: msg('rename-articleexists'),
                okBtn: {
                  label: msg('ok'),
                  className: 'btn btn-primary only-btn'
                }
              });
            }
          });
        }
      }],
      beforeShow: function () {
        if (!InPageEdit.hasRight('move')) {
          ssi_modal.dialog({
            title: msg('notify-no-right'),
            content: msg('rename-no-right'),
            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: msg('preference-title') + ' - ' + InPageEdit.version,
      content: '<section id="InPageEditSettingBox"><b>InPageEdit编辑器</b><br/><input id="ipeSetoutSideClose" type="checkbox"/> <label for="ipeSetoutSideClose">' + msg('preference-outSideClose') + '</label><br/><input id="ipeSetMinor" type="checkbox"/> <label for="ipeSetMinor">' + msg('preference-setMinor') + '</label><br/><b>' + msg('preference-setMinor') + '</b><br/><label>' + msg('preference-editSummary') + '<br/><span style="font-size:10px"></span><input id="ipeSetSummary" value="' + summary + '" style="width:100%"/></label><br/><b>' + msg('preference-analysis') + '</b><br/><span style="font-size:10px">' + msg('preference-analysis-view').replace('$1', '<a href="https://doc.wjghj.cn/InPageEditAnalysis/" target="_blank">https://doc.wjghj.cn/InPageEditAnalysis/</a>') + '</span><br/><b>' + msg('preference-about') + '</b><br/><button class="btn btn-secondary" onclick="mw.loader.load(\'https://common.wjghj.cn/js/InPageEdit-v2.js/about\')">' + msg('preference-aboutAndHelp') + '</button>&nbsp;<button class="btn btn-secondary" onclick="InPageEdit.versionInfo()">' + msg('preference-updatelog') + '</button><hr><span style="font-size:10px;line-height:5px">' + msg('preference-savelocal-label') + '<br/>' + msg('preference-savelocal') + '<a href="javascript:;" id="ipeSaveLocalShow">' + msg('preference-savelocal-btn') + '</a></span></section>',
      sizeClass: 'dialog',
      className: 'in-page-edit ipe-preference',
      center: true,
      buttons: [{
        label: msg('preference-reset'),
        className: 'btn btn-danger',
        method: function () {
          $('#InPageEditSettingBox #ipeSetoutSideClose').prop('checked', true);
          $('#InPageEditSettingBox #ipeSetMinor').prop('checked', false);
          $('#InPageEditSettingBox #ipeSetSummary').val(msg('preference-summary-default'));
        }
      }, {
        label: msg('preference-save'),
        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,
        title: msg('preference-savelocal-popup-title'),
        content: '<section id="ipeSaveLocal"><b>' + msg('preference-savelocal-popup-notrecommended') + '</b><br/>' + msg('preference-savelocal-popup') + '<br/><input style="width:100%" readonly="readonly" onclick="$(this).select()"/><br/>' + msg('preference-savelocal-popup-notice') + '</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()
      }) + ';');
    });
    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: msg('preference-savelocal-popup-haslocal').replace('$1', '<a href="' + mw.util.getUrl('Special:Mypage/common.js') + '">' + msg('preference-savelocal-popup-yourjspage') + '</a>'),
        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(msg('diff-loading'));
      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">' + msg('diff-loading') + '</span>',
        content: '<div class="ipe-progress" style="width:100%"><div class="ipe-progress-bar"></div></div><div class="diffArea"></div>',
        buttons: [{
          label: msg('diff-button-todiffpage'),
          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) + '">' + msg('diff-usertalk') + '</a> | <a href="' + mw.util.getUrl('Special:Contributions/' + user) + '">' + msg('diff-usercontrib') + '</a> | <a href="' + mw.util.getUrl('Special:Block/' + user) + '">' + msg('diff-userblock') + '</a> )';
      }
      $('.quick-diff .pageName').html(msg('diff-title') + ': <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> (' + msg('diff-version') + data.compare.fromrevid + ') (<a class="editLink" href="' + mw.config.get('wgScript') + '?action=edit&title=' + data.compare.fromtitle + '&oldid=' + data.compare.fromrevid + '">' + msg('diff-edit') + '</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\'});">←' + msg('diff-prev') + '</a>' +
        '</td>' +
        '<td colspan="2" class="diff-ntitle">' +
        '<a class="" href="' + mw.config.get('wgScript') + '?oldid=' + data.compare.torevid + '">' + data.compare.totitle + '</a> (' + msg('diff-version') + data.compare.torevid + ') (<a class="editLink" href="' + mw.config.get('wgScript') + '?action=edit&title=' + data.compare.totitle + '&oldid=' + data.compare.torevid + '">' + msg('diff-edit') + '</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\'});">' + msg('diff-nextv') + '→</a>' +
        '</td>' +
        '</tr>' +
        diffTable +
        '<tr class="diffSize" style="text-align: center;"><td colspan="2">' + data.compare.fromsize + msg('diff-bytes') + '</td><td colspan="2">' + data.compare.tosize + msg('diff-bytes') + '</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>' + msg('diff-title-original-content') + '</b>');
        $('.quick-diff .diff-ntitle').html('<b>' + msg('diff-title-your-content') + '</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(msg('diff-error') + ': ' + 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(msg('quick-diff')).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 + '});">' + msg('quick-edit') + '</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(msg('quick-edit'))
              .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">' + msg('preview-placeholder') + '</section>',
      title: msg('preview-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(msg('preview-error'));
        });
      }
    });
  }

  /** 载入中模块 **/
  InPageEdit.progress = function (title) {
    if (title === true) {
      $('.in-page-edit.loadingbox .ssi-modalTitle').html(msg('done'));
      $('.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 = 'Loading...'
      }
      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.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 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 () {
    /** 额外的模块 **/
    // 快速页面差异模块
    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 = msg('preference-summary-default');
      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">' + msg('quick-edit') + '</div><button id="edit-btn" class="ipe-toolbox-btn material-icons">edit</button></li>' +
      '<li class="btn-tip-group"><div class="btn-tip">' + msg('redirect-from') + '</div><button id="redirectfrom-btn" class="ipe-toolbox-btn material-icons">flight_land</button></li>' +
      '<li class="btn-tip-group"><div class="btn-tip">' + msg('redirect-to') + '</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">' + msg('quick-delete') + '</div><button id="deletepage-btn" class="ipe-toolbox-btn material-icons">delete_forever</button></li>' +
      '<li class="btn-tip-group"><div class="btn-tip">' + msg('quick-rename') + '</div><button id="renamepage-btn" class="ipe-toolbox-btn material-icons">format_italic</button></li>' +
      '<li class="btn-tip-group"><div class="btn-tip">' + msg('ipe-preference') + '</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                      /____/');
  }());
}());