Site Notice

hello, world

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

From Project-EPB Commons
([InPageEdit] 没有编辑摘要)
([InPageEdit] 没有编辑摘要)
Line 1: Line 1:
 
/**
 
/**
  * GNU GENERAL PUBLIC LICENSE 3.0
+
  * @license GPL-3.0 GNU GENERAL PUBLIC LICENSE 3.0
 
  *
 
  *
  * MediaWiki JS Plugin: In Page Edit
+
  * @name InPageEdit-v2
  * Author: 机智的小鱼君
+
* @description A JavaScript-based MediaWiki Plugin:  
  * Url: https://github.com/Dragon-Fish/InPageEdit-v2
+
  * @author 机智的小鱼君
  **/
+
  * @url https://github.com/Dragon-Fish/InPageEdit-v2
 +
  */
 
!(function () {
 
!(function () {
 
   'use strict';
 
   'use strict';
   // 创建全局函数
+
   /**
 +
  * @description 检查插件是否已经运行,创建全局函数
 +
  * @global InPageEdit {Object}
 +
  */
 
   if (typeof (InPageEdit) !== 'undefined' && typeof (InPageEdit.version) !== 'undefined') throw '[InPageEdit] 已经有一个IPE插件在执行了';
 
   if (typeof (InPageEdit) !== 'undefined' && typeof (InPageEdit.version) !== 'undefined') throw '[InPageEdit] 已经有一个IPE插件在执行了';
 
   window.InPageEdit = window.InPageEdit || {};
 
   window.InPageEdit = window.InPageEdit || {};
   InPageEdit.isCanary = true;
+
   InPageEdit.isCanary = false;
 
   InPageEdit.api = {
 
   InPageEdit.api = {
 
     analysis: 'https://doc.wjghj.cn/inpageedit-v2/analysis/api/index.php',
 
     analysis: 'https://doc.wjghj.cn/inpageedit-v2/analysis/api/index.php',
Line 17: Line 21:
 
     updatelogsJson: 'https://cdn.jsdelivr.net/gh/dragon-fish/inpageedit-v2@master/docs/update-logs.json'
 
     updatelogsJson: 'https://cdn.jsdelivr.net/gh/dragon-fish/inpageedit-v2@master/docs/update-logs.json'
 
   }
 
   }
   /*=version*/InPageEdit.version = '2.13.3';/*version=*/
+
   InPageEdit.version = '2.13.4';
 +
  Object.freeze(InPageEdit.api); // 冻结重要全局变量
  
   // 常用函数
+
   // 缓存常用函数
 
   var config = mw.config.get();
 
   var config = mw.config.get();
  
   /** 导入模态框插件 **/
+
   /** 导入依赖 **/
 +
  // 模态框
 
   mw.loader.load('https://cdn.jsdelivr.net/gh/dragon-fish/inpageedit-v2@master/src/ssi_modal/script.min.js');
 
   mw.loader.load('https://cdn.jsdelivr.net/gh/dragon-fish/inpageedit-v2@master/src/ssi_modal/script.min.js');
 
   /** 样式表 **/
 
   /** 样式表 **/
Line 28: Line 34:
 
     // 模态框
 
     // 模态框
 
     $('<link>', { rel: 'stylesheet', href: 'https://cdn.jsdelivr.net/gh/dragon-fish/inpageedit-v2@master/src/ssi_modal/style.min.css' }),
 
     $('<link>', { rel: 'stylesheet', href: 'https://cdn.jsdelivr.net/gh/dragon-fish/inpageedit-v2@master/src/ssi_modal/style.min.css' }),
 +
    // FontAwesome
 +
    $('<link>', { rel: 'stylesheet', href: 'https://cdn.jsdelivr.net/npm/[email protected]/css/font-awesome.min.css' }),
 
     // 覆写
 
     // 覆写
 
     $('<link>', { rel: 'stylesheet', href: 'https://cdn.jsdelivr.net/gh/dragon-fish/inpageedit-v2@master/src/override.min.css' })
 
     $('<link>', { rel: 'stylesheet', href: 'https://cdn.jsdelivr.net/gh/dragon-fish/inpageedit-v2@master/src/override.min.css' })
 
   );
 
   );
  
  /*** BOT FLAG ***/
 
 
   /** 导入 i18n 组件 **/
 
   /** 导入 i18n 组件 **/
 
   mw.loader.load('https://cdn.jsdelivr.net/gh/dragon-fish/i18n-js@master/script.min.js');
 
   mw.loader.load('https://cdn.jsdelivr.net/gh/dragon-fish/i18n-js@master/script.min.js');
Line 39: Line 46:
 
   });
 
   });
  
   /** InPageEdit主框架 **/
+
   /**
 +
  * @function InPageEdit
 +
  */
 
   function init(i18n) {
 
   function init(i18n) {
 
     // i18n
 
     // i18n
 
     function msg(i) {
 
     function msg(i) {
 
       return i18n.msg(i).parse();
 
       return i18n.msg(i).parse();
     };
+
     }
 
     /** HTML 组件 **/
 
     /** HTML 组件 **/
 
     const $br = $('<br/>'),
 
     const $br = $('<br/>'),
Line 51: Line 60:
 
       $progress = $('<div>', { class: 'ipe-progress', style: 'width: 100%' }).append($('<div>', { class: 'ipe-progress-bar' }));
 
       $progress = $('<div>', { class: 'ipe-progress', style: 'width: 100%' }).append($('<div>', { class: 'ipe-progress-bar' }));
  
     /** 快速编辑模块 **/
+
     /**
     InPageEdit.edit = InPageEdit.quickEdit = function (option) {
+
    * @module 快速编辑模块
 +
    *
 +
    * @property {Object} options
 +
    * @property {String} options.page edit page (default: wgPageName)
 +
    * @property {Number} options.revision page rev ID
 +
    * @property {Number} options.section edit section
 +
    * @property {Boolean} options.reload if reload page after save successful (default: true)
 +
    */
 +
     InPageEdit.edit = InPageEdit.quickEdit = function (options) {
 
       mw.hook('InPageEdit.quickEdit').fire();
 
       mw.hook('InPageEdit.quickEdit').fire();
       // 变量
+
       /** 获取设定信息,设置缺省值 **/
       var option = option || {};
+
       var options = options || {};
       var preference = localStorage.InPageEditPreference || '{}';
+
       var defaultOptions = {
      preference = JSON.parse(preference);
+
        page: config.wgPageName,
      var editPage = option.page || config.wgPageName;
+
         pageId: 0,
      editPage = decodeURIComponent(editPage);
+
         revision: null,
      var
+
         summaryRevision: '',
         editSection = option.section || false,
+
        section: null,
         editRevision = option.revision || false,
+
         editText: '',
         summaryRevision = '',
+
        editMinor: false,
         editText,
+
         editSummary: msg('preference-summary-default'),
         editSummary = preference.editSummary || msg('preference-summary-default'),
+
         editNotice: '',
        editMinor = preference.editMinor || false,
+
         outSideClose: true,
         editNotice = '',
+
        jsonGet: {
         outSideClose = preference.outSideClose;
+
          action: 'parse',
      if (outSideClose === undefined) outSideClose = true;
+
          page: options.page,
      var jsonGet = {
+
          prop: 'wikitext|langlinks|categories|templates|images|sections',
        action: 'parse',
+
          format: 'json'
        page: editPage,
+
        },
        prop: 'wikitext|langlinks|categories|templates|images|sections',
+
         jsonPost: {},
        format: 'json'
+
         pageDetail: {},
      },
+
         jumpTo: '',
         pageId,
+
         reload: true
         jsonPost = {},
+
      }
         protection = '',
+
 
         pageDetail,
+
      /** 获取用户设置 **/
        basetimestamp,
+
      var preference = InPageEdit.preference.get();
        jumpTo = '#';
+
 
       var
+
      /** 缓存时间戳 **/
        date = new Date(),
+
       var date = new Date(),
 
         timestamp = date.getTime(),
 
         timestamp = date.getTime(),
         now = date.toUTCString(); // 缓存时间戳
+
         now = date.toUTCString();
       var reloadPage = option.reload;
+
 
       if (reloadPage === undefined) reloadPage = true;
+
      /** 将选项合并并标准化 **/
 +
       options = $.extend(defaultOptions, options, preference);
 +
       options.page = decodeURIComponent(options.page); // 解码网址 Unicode
  
 
       _analysis('quick_edit');
 
       _analysis('quick_edit');
  
       if (editRevision !== false && editRevision !== '' && editRevision !== config.wgCurRevisionId) {
+
       if (options.revision !== null && options.revision !== '' && options.revision !== config.wgCurRevisionId) {
 
         ssi_modal.notify('warning', {
 
         ssi_modal.notify('warning', {
 
           className: 'in-page-edit',
 
           className: 'in-page-edit',
Line 98: Line 117:
 
         });
 
         });
 
         delete jsonGet.page;
 
         delete jsonGet.page;
         jsonGet.oldid = editRevision;
+
         jsonGet.oldid = options.revision;
         summaryRevision = '(' + msg('editor-summary-rivision') + '[[Special:Diff/' + editRevision + ']])';
+
         options.summaryRevision = '(' + msg('editor-summary-rivision') + '[[Special:Diff/' + options.revision + ']])';
 
       } else {
 
       } else {
         if (editSection !== false && editSection !== '') {
+
         if (options.section !== null && editSection !== '') {
 
           jsonGet.section = editSection;
 
           jsonGet.section = editSection;
 
         }
 
         }
      }
 
      if (typeof (InPageEdit.myPreference) !== 'undefined') {
 
        if (typeof (InPageEdit.myPreference.editSummary) === 'string')
 
          editSummary = InPageEdit.myPreference.editSummary;
 
        if (typeof (InPageEdit.myPreference.editMinor) === 'boolean')
 
          editMinor = InPageEdit.myPreference.editMinor;
 
        if (typeof (InPageEdit.myPreference.outSideClose) === 'boolean')
 
          outSideClose = InPageEdit.myPreference.outSideClose;
 
 
       }
 
       }
  
Line 119: Line 130:
 
       // 显示主窗口
 
       // 显示主窗口
 
       ssi_modal.show({
 
       ssi_modal.show({
         title: msg('editor-title-editing') + ': <u class="editPage">' + editPage.replace(/\_/g, ' ') + '</u>',
+
         title: msg('editor-title-editing') + ': <u class="editPage">' + options.page.replace(/\_/g, ' ') + '</u>',
 
         content:
 
         content:
 
           $('<div>').append(
 
           $('<div>').append(
Line 174: Line 185:
 
                 $('<label>', { for: 'editSummary', text: msg('editSummary') }),
 
                 $('<label>', { for: 'editSummary', text: msg('editSummary') }),
 
                 $('<br>'),
 
                 $('<br>'),
                 $('<input>', { class: 'editSummary', id: 'editSummary', placeholder: 'Edit via InPageEdit~', value: editSummary.replace(/\$oldid/ig, summaryRevision) }),
+
                 $('<input>', { class: 'editSummary', id: 'editSummary', placeholder: 'Edit via InPageEdit~', value: options.editSummary.replace(/\$oldid/ig, options.summaryRevision) }),
 
                 $('<br>'),
 
                 $('<br>'),
 
                 $('<label>').append(
 
                 $('<label>').append(
Line 210: Line 221:
 
                   var text = $('.ipe-editor.timestamp-' + timestamp + ' .editArea').val(),
 
                   var text = $('.ipe-editor.timestamp-' + timestamp + ' .editArea').val(),
 
                     minor = $('.ipe-editor.timestamp-' + timestamp + ' .editMinor').prop('checked'),
 
                     minor = $('.ipe-editor.timestamp-' + timestamp + ' .editMinor').prop('checked'),
                     section = option.section,
+
                     section = options.section,
 
                     summary = $('.ipe-editor.timestamp-' + timestamp + ' .editSummary').val();
 
                     summary = $('.ipe-editor.timestamp-' + timestamp + ' .editSummary').val();
 
                   postArticle({
 
                   postArticle({
 
                     text: text,
 
                     text: text,
                     page: editPage,
+
                     page: options.page,
 
                     minor: minor,
 
                     minor: minor,
 
                     section: section,
 
                     section: section,
Line 230: Line 241:
 
             InPageEdit.quickPreview({
 
             InPageEdit.quickPreview({
 
               action: 'parse',
 
               action: 'parse',
               title: editPage,
+
               title: options.page,
 
               text: text,
 
               text: text,
 
               prop: 'text',
 
               prop: 'text',
Line 244: Line 255:
 
             var text = $('.ipe-editor.timestamp-' + timestamp + ' .editArea').val();
 
             var text = $('.ipe-editor.timestamp-' + timestamp + ' .editArea').val();
 
             var diffJson = {};
 
             var diffJson = {};
             diffJson.fromtext = editText;
+
             diffJson.fromtext = options.editText;
 
             diffJson.totext = text;
 
             diffJson.totext = text;
 
             diffJson.hideBtn = true;
 
             diffJson.hideBtn = true;
             diffJson.pageName = editPage;
+
             diffJson.pageName = options.page;
 
             diffJson.isPreview = true;
 
             diffJson.isPreview = true;
 
             InPageEdit.quickDiff(diffJson);
 
             InPageEdit.quickDiff(diffJson);
Line 290: Line 301:
 
             var btns = InPageEdit.buttons;
 
             var btns = InPageEdit.buttons;
 
             $('.ipe-editor.timestamp-' + timestamp + ' .btnGroup.extra').show();
 
             $('.ipe-editor.timestamp-' + timestamp + ' .btnGroup.extra').show();
             function addBtn(open, middle, close, text) {
+
             function addBtn(open, middle, close, icon) {
 
               var open = open || '',
 
               var open = open || '',
 
                 middle = middle || '',
 
                 middle = middle || '',
 
                 close = close || '',
 
                 close = close || '',
                 text = text || '<span class="material-icons">all_inclusive</span>';
+
                 icon = 'fa-' + icon || 'fa-wrench';
 
               $('.ipe-editor.timestamp-' + timestamp + ' .btnGroup.extra').append(
 
               $('.ipe-editor.timestamp-' + timestamp + ' .btnGroup.extra').append(
                 $('<button>', { class: 'editToolBtn btn', 'data-open': open, 'data-middle': middle, 'data-close': close, html: text })
+
                 $('<button>', { class: 'editToolBtn btn', 'data-open': open, 'data-middle': middle, 'data-close': close, html: `<i class="fa ${icon}"></i>` })
 
               );
 
               );
 
             }
 
             }
Line 330: Line 341:
 
           });
 
           });
 
         },
 
         },
         /* 模态框显示后 */
+
         /**
 +
        * @event onShow
 +
        * @description 模态框显示后
 +
        */
 
         onShow: function (a, modal) {
 
         onShow: function (a, modal) {
 
           mw.hook('InPageEdit.quickEdit.modal').fire();
 
           mw.hook('InPageEdit.quickEdit.modal').fire();
Line 341: Line 355:
 
           });
 
           });
 
           // 获取权限
 
           // 获取权限
           if (InPageEdit.hasRight('edit') === false) {
+
           if (_hasRight('edit') === false) {
 
             ssi_modal.notify('dialog', {
 
             ssi_modal.notify('dialog', {
 
               className: 'in-page-edit',
 
               className: 'in-page-edit',
Line 359: Line 373:
  
 
           // 解析页面内容
 
           // 解析页面内容
           new mw.Api().get(jsonGet).done(function (data) {
+
           new mw.Api().get(options.jsonGet).done(function (data) {
 
             var data = data;
 
             var data = data;
 
             console.timeEnd('[InPageEdit] 获取页面源代码');
 
             console.timeEnd('[InPageEdit] 获取页面源代码');
Line 373: Line 387:
 
           function contentDone(data) {
 
           function contentDone(data) {
 
             var data = data;
 
             var data = data;
             pageDetail = data;
+
             options.pageDetail = data;
  
 
             if (data.hasOwnProperty('error')) {
 
             if (data.hasOwnProperty('error')) {
 
               console.warn('[InPageEdit]警告:无法获取页面内容');
 
               console.warn('[InPageEdit]警告:无法获取页面内容');
               editText = '<!-- ' + data.error.info + ' -->';
+
               options.editText = '<!-- ' + data.error.info + ' -->';
               pageId = -1;
+
               options.pageId = 0;
 
               $('.ipe-editor.timestamp-' + timestamp + ' .detailArea').hide();
 
               $('.ipe-editor.timestamp-' + timestamp + ' .detailArea').hide();
 
             } else {
 
             } else {
               editText = data.parse.wikitext['*'];
+
               options.editText = data.parse.wikitext['*'];
               pageId = data.parse.pageid;
+
               options.pageId = data.parse.pageid;
 
             }
 
             }
 
             // 设定一堆子样式
 
             // 设定一堆子样式
Line 389: Line 403:
 
             $('.ipe-editor.timestamp-' + timestamp + ' .editArea').val(editText + '\n');
 
             $('.ipe-editor.timestamp-' + timestamp + ' .editArea').val(editText + '\n');
  
             if (editSection !== false) {
+
             if (options.editSection !== null) {
 
               var val = $('.ipe-editor.timestamp-' + timestamp + ' .editSummary').val();
 
               var val = $('.ipe-editor.timestamp-' + timestamp + ' .editSummary').val();
 
               val = val.replace(/\$section/ig, '/* ' + data.parse.sections[0].line + ' */');
 
               val = val.replace(/\$section/ig, '/* ' + data.parse.sections[0].line + ' */');
Line 399: Line 413:
 
               val = val.replace(/\$section/ig, '');
 
               val = val.replace(/\$section/ig, '');
 
               $('.ipe-editor.timestamp-' + timestamp + ' .editSummary').val(val);
 
               $('.ipe-editor.timestamp-' + timestamp + ' .editSummary').val(val);
               jumpTo = '#';
+
               jumpTo = '';
 
             }
 
             }
             if (editRevision !== false && editRevision !== '' && editRevision !== config.wgCurRevisionId) {
+
             if (options.revision !== null && options.revision !== '' && options.revision !== config.wgCurRevisionId) {
               $('.ipe-editor.timestamp-' + timestamp + ' .editPage').after('<span class="editRevision">(' + msg('editor-title-editRevision') + ':' + editRevision + ')</span>');
+
               $('.ipe-editor.timestamp-' + timestamp + ' .editPage').after('<span class="editRevision">(' + msg('editor-title-editRevision') + ':' + options.revision + ')</span>');
 
             }
 
             }
  
Line 414: Line 428:
 
             }
 
             }
 
             if (pageId !== -1) {
 
             if (pageId !== -1) {
               queryJson.pageids = pageId;
+
               queryJson.pageids = options.pageId;
 
             } else {
 
             } else {
               queryJson.titles = editPage;
+
               queryJson.titles = options.editPage;
 
             }
 
             }
 
             new mw.Api().get(queryJson).done(function (data) {
 
             new mw.Api().get(queryJson).done(function (data) {
Line 435: Line 449:
 
             function queryDone(data) {
 
             function queryDone(data) {
 
               var data = data;
 
               var data = data;
               if (data.query.pages[pageId]['protection'].length > 0) {
+
               if (data.query.pages[options.pageId]['protection'].length > 0) {
                 var protection = data.query.pages[pageId]['protection'];
+
                 var protection = data.query.pages[options.pageId]['protection'];
 
                 for (var i = 0; i < protection.length; i++) {
 
                 for (var i = 0; i < protection.length; i++) {
 
                   if (protection[i].type === 'edit') {
 
                   if (protection[i].type === 'edit') {
 
                     if (
 
                     if (
                       (protection[i].level === 'autoconfirmed' && !InPageEdit.hasRight('autoconfirmed')) ||
+
                       (protection[i].level === 'autoconfirmed' && !_hasRight('autoconfirmed')) ||
                       (protection[i].level === 'sysop' && !InPageEdit.hasRight('editprotected')) ||
+
                       (protection[i].level === 'sysop' && !_hasRight('editprotected')) ||
                       (config.wgNamespaceNumber === 8 && !InPageEdit.hasRight('editinterface'))
+
                       (config.wgNamespaceNumber === 8 && !_hasRight('editinterface'))
 
                     ) {
 
                     ) {
 
                       ssi_modal.notify('dialog', {
 
                       ssi_modal.notify('dialog', {
Line 465: Line 479:
 
               $.get(config.wgScript, {
 
               $.get(config.wgScript, {
 
                 action: 'raw',
 
                 action: 'raw',
                 title: 'MediaWiki:Editnotice-' + data.query.pages[pageId].ns
+
                 title: 'MediaWiki:Editnotice-' + data.query.pages[options.pageId].ns
 
               }, function (data) {
 
               }, function (data) {
 
                 var data = data;
 
                 var data = data;
 
                 new mw.Api().post({
 
                 new mw.Api().post({
 
                   action: 'parse',
 
                   action: 'parse',
                   title: editPage,
+
                   title: options.editPage,
 
                   contentmodel: 'wikitext',
 
                   contentmodel: 'wikitext',
 
                   preview: true,
 
                   preview: true,
Line 552: Line 566:
 
         switch (id) {
 
         switch (id) {
 
           case 'showTemplates':
 
           case 'showTemplates':
             var templates = pageDetail.parse.templates,
+
             var templates = options.pageDetail.parse.templates,
 
               templateName;
 
               templateName;
 
             for (var i = 0; i < templates.length; i++) {
 
             for (var i = 0; i < templates.length; i++) {
Line 559: Line 573:
 
                 $('<a>', { href: mw.util.getUrl(templateName), target: '_blank', text: templateName }),
 
                 $('<a>', { href: mw.util.getUrl(templateName), target: '_blank', text: templateName }),
 
                 $('<span> (</span>'),
 
                 $('<span> (</span>'),
                 $('<a>', { href: 'javascript:;', text: msg('quick-edit') }).click(function () {
+
                 $('<a>', { href: 'javascript:;', text: msg('quick-edit'), onclick: InPageEdit.quickEdit({ page: templateName, reload: false }) }),
                  InPageEdit.quickEdit({ page: templateName, reload: false });
 
                }),
 
 
                 $('<span>)</span>')
 
                 $('<span>)</span>')
 
               ).appendTo(content);
 
               ).appendTo(content);
Line 573: Line 585:
 
             break;
 
             break;
 
           case 'showImages':
 
           case 'showImages':
             var images = pageDetail.parse.images,
+
             var images = options.pageDetail.parse.images,
 
               imageName;
 
               imageName;
 
             for (var i = 0; i < images.length; i++) {
 
             for (var i = 0; i < images.length; i++) {
 
               imageName = images[i];
 
               imageName = images[i];
               $('<li>').append(
+
               $('li').append(
 
                 $('<a>', { href: mw.util.getUrl('File:' + imageName), target: '_balnk' }),
 
                 $('<a>', { href: mw.util.getUrl('File:' + imageName), target: '_balnk' }),
 
                 $('<span> (</span>'),
 
                 $('<span> (</span>'),
                 $('<a>', { href: 'javascript:;', class: 'quickViewImage', text: msg('editor-detail-images-quickview') }).click(function () {
+
                 $('<a>', { href: 'javascript:;', class: 'quickViewImage', text: msg('editor-detail-images-quickview') }).bind('click', function () {
 +
                  var imageName = imageName;
 
                   ssi_modal.show({
 
                   ssi_modal.show({
 
                     className: 'in-page-edit quick-view-image',
 
                     className: 'in-page-edit quick-view-image',
Line 638: Line 651:
 
           InPageEdit.progress(true);
 
           InPageEdit.progress(true);
 
           // GitHub@issue#40
 
           // GitHub@issue#40
           if (data.edit.result !== 'Success') {
+
           if (data.edit.result !== 'Success' || data.errors !== undefined) {
 
             // 出事了
 
             // 出事了
 
             InPageEdit.progress(false);
 
             InPageEdit.progress(false);
 +
            var errorCode,
 +
              errorInfo,
 +
              errorMore = '';
 +
            if (data.errors !== undefined) {
 +
              errorCode = data.errors.code;
 +
              errorInfo = data.errors['*'];
 +
              errorMore = '';
 +
            } else if (data.edit.result !== 'Success') {
 +
              errorCode = data.edit.code || 'Unknown';
 +
              errorInfo = data.edit.info || 'Reason unknown.';
 +
              errorMore = data.edit.warning || '';
 +
            } else {
 +
              errorCode = 'unknown';
 +
              errorInfo = 'Reason unknown.';
 +
              errorMore = 'Please contact plug-in author or try again.'
 +
            }
 
             ssi_modal.show({
 
             ssi_modal.show({
 
               className: 'in-page-edit',
 
               className: 'in-page-edit',
 
               sizeClass: 'dialog',
 
               sizeClass: 'dialog',
 
               title: msg('editor-save-error'),
 
               title: msg('editor-save-error'),
               content: data.edit.info + '<hr style="clear: both" />' + data.edit.warning
+
               content: errorInfo + '<hr style="clear: both" />' + errorMore
 
             });
 
             });
 
             ssi_modal.notify('error', {
 
             ssi_modal.notify('error', {
Line 654: Line 683:
 
               },
 
               },
 
               title: msg('notify-error'),
 
               title: msg('notify-error'),
               content: msg('editor-save-error') + ':<code>' + data.edit.code + '</code>'
+
               content: msg('editor-save-error') + ':<code>' + errorCode + '</code>'
 
             });
 
             });
  
             console.error('[InPageEdit] Submit failed: \nCode: ' + data.edit.code);
+
             console.error('[InPageEdit] Submit failed: \nCode: ' + errorCode);
 
             return;
 
             return;
 
           } else {
 
           } else {
Line 694: Line 723:
 
     }
 
     }
  
     /** 查找替换模块 **/
+
     /**
 +
    * @module 查找替换模块
 +
    * @param contengut <Element> Textarea
 +
    */
 
     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 923: Line 955:
 
         ),
 
         ),
 
         beforeShow: function () {
 
         beforeShow: function () {
           if (!InPageEdit.hasRight('delete')) {
+
           if (!_hasRight('delete')) {
 
             ssi_modal.dialog({
 
             ssi_modal.dialog({
 
               title: msg('notify-no-right'),
 
               title: msg('notify-no-right'),
Line 1,110: Line 1,142:
 
         }],
 
         }],
 
         beforeShow: function () {
 
         beforeShow: function () {
           if (!InPageEdit.hasRight('move')) {
+
           if (!_hasRight('move')) {
 
             ssi_modal.dialog({
 
             ssi_modal.dialog({
 
               title: msg('notify-no-right'),
 
               title: msg('notify-no-right'),
Line 1,126: Line 1,158:
 
     }
 
     }
  
     /** 个人设置模块 **/
+
     /**
     InPageEdit.preference = function () {
+
    * @module 个人设置模块
      mw.hook('InPageEdit.preference').fire();
+
    */
       _analysis('plugin_setting');
+
     InPageEdit.preference = {
      var settings = JSON.parse(localStorage.getItem('InPageEditPreference')),
+
      /* 预设值 */
         minor = settings.editMinor,
+
      "default": {
         summary = settings.editSummary,
+
        doNotCollectMyInfo: false,
         outSideClose = settings.outSideClose;
+
        doNotShowLocalWarn: false,
 +
        editMinor: false,
 +
        editSummary: msg('preference-summary-default'),
 +
        outSideClose: true,
 +
        watchList: mw.user.options.get().watchdefault
 +
      },
 +
      /* 获取设置 */
 +
      get: function () {
 +
        var local = localStorage.getItem('InPageEditPreference') || '{}',
 +
          local = JSON.parse(local);
 +
        if (typeof InPageEdit.myPreference === 'object') {
 +
          local = $.extend(local, InPageEdit.myPreference);
 +
        }
 +
        return $.extend(InPageEdit.preference.default, local);
 +
       },
 +
      /**
 +
      * @description 保存设置
 +
      * @param {Object} options
 +
      */
 +
      set: function (options) {
 +
        var options = options || {};
 +
        if (typeof options !== 'object') return;
 +
        options = $.extend(InPageEdit.preference.get(), options);
 +
        options = JSON.stringify(options);
 +
        localStorage.setItem('InPageEditPreference', options);
 +
      },
 +
      /**
 +
      * @description 用户图形界面
 +
      */
 +
      modal: function () {
 +
         mw.hook('InPageEdit.preference').fire();
 +
         InPageEdit.preference.set();
 +
         var local = InPageEdit.preference.get();
 +
        _analysis('plugin_setting');
  
      ssi_modal.show({
+
        ssi_modal.show({
        outSideClose: false,
+
          outSideClose: false,
        title: msg('preference-title') + ' - ' + InPageEdit.version,
+
          title: msg('preference-title') + ' - ' + InPageEdit.version,
        content:
+
          content:
          $('<section>', { id: 'InPageEditSettingBox', class: 'ipe-preference' }).append(
+
            $('<section>', { id: 'InPageEditSettingBox', class: 'ipe-preference' }).append(
            $('<h4>', { text: msg('preference-editor-label') }),
+
              $('<h4>', { text: msg('preference-editor-label') }),
            $('<label>').append(
+
              $('<label>').append(
              $('<input>', { id: 'ipeSetoutSideClose', type: 'checkbox' }),
+
                $('<input>', { id: 'outSideClose', type: 'checkbox' }),
              $('<span>', { text: msg('preference-outSideClose') })
+
                $('<span>', { text: msg('preference-outSideClose') })
            ),
+
              ),
            $('<br>'),
+
              $('<br>'),
            $('<label>').append(
+
              $('<label>').append(
              $('<input>', { id: 'ipeSetMinor', type: 'checkbox' }),
+
                $('<input>', { id: 'editMinor', type: 'checkbox' }),
               $('<span>', { text: msg('preference-setMinor') })
+
                $('<span>', { text: msg('preference-setMinor') })
 +
              ),
 +
              $('<br>'),
 +
              $('<h4>', { text: msg('preference-summary-label') }),
 +
              $('<label>', { for: 'editSummary', style: 'padding-left: 0; font-size: small', html: msg('preference-editSummary').replace('%br%', '<br/>').replace('$1', '<code>$oldid</code>').replace('$2', '<code>' + msg('editor-summary-rivision') + ' [[Special:Diff/oldid]]</code>').replace('$3', '<code>$section</code>').replace('$4', '<code>/* SECTION TITLE */</code>') }),
 +
              $('<br>'),
 +
              $('<input>', { id: 'editSummary', style: 'width: 96%', value: local.editSummary, placeholder: 'Edit via InPageEdit, yeah~' }),
 +
              $('<h4>', { text: msg('preference-analysis-label') }),
 +
              $('<span>', { style: 'font-size: small; line-height: 0.9em', html: msg('preference-analysis-view').replace('$1', '<a href="https://doc.wjghj.cn/inpageedit-v2/analysis/" target="_blank">https://doc.wjghj.cn/inpageedit-v2/analysis/</a>') }),
 +
              $('<h4>', { text: msg('preference-about-label') }),
 +
              $('<button>', { class: 'btn btn-secondary', onclick: "InPageEdit.about()", text: msg('preference-aboutAndHelp') }),
 +
              $('<button>', { class: 'btn btn-secondary', style: 'margin-left: 1em;', onclick: "InPageEdit.versionInfo()", text: msg('preference-updatelog') }),
 +
              $hr,
 +
              $('<strong>', { style: 'font-size: small; line-height: 0.9em', text: msg('preference-savelocal-label') }),
 +
              $('<br>'),
 +
               $('<span>', { style: 'font-size: small; line-height: 0.9em', text: msg('preference-savelocal') }).append(
 +
                $('<a>', { href: 'javascript:;', id: 'ipeSaveLocalShow', text: msg('preference-savelocal-btn') }).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/><textarea style="font-size: small;" readonly></textarea><br/>' + msg('preference-savelocal-popup-notice') + '</section>',
 +
                    okBtn: {
 +
                      className: 'btn btn-primary btn-single',
 +
                      label: msg('ok')
 +
                    }
 +
                  });
 +
                  $('#ipeSaveLocal textarea').val('window.InPageEdit = window.InPageEdit || {}; // Keep this line\nInPageEdit.myPreference = ' + JSON.stringify({
 +
                    outSideClose: $('#InPageEditSettingBox #ipeSetoutSideClose').prop('checked'),
 +
                    editMinor: $('#InPageEditSettingBox #ipeSetMinor').prop('checked'),
 +
                    editSummary: $('#InPageEditSettingBox #ipeSetSummary').val()
 +
                  }) + '; // InPageEdit preference');
 +
                })
 +
              )
 
             ),
 
             ),
            $('<br>'),
+
          sizeClass: 'dialog',
            $('<h4>', { text: msg('preference-summary-label') }),
+
          className: 'in-page-edit ipe-preference',
            $('<label>', { for: 'ipeSetSummary', style: 'padding-left: 0; font-size: small', html: msg('preference-editSummary').replace('%br%', '<br/>').replace('$1', '<code>$oldid</code>').replace('$2', '<code>' + msg('editor-summary-rivision') + ' [[Special:Diff/oldid]]</code>').replace('$3', '<code>$section</code>').replace('$4', '<code>/* SECTION TITLE */</code>') }),
+
          center: true,
            $('<br>'),
+
          buttons: [{
            $('<input>', { id: 'ipeSetSummary', style: 'width: 96%', value: summary, placeholder: 'Edit via InPageEdit, yeah~' }),
+
             label: msg('preference-reset'),
            $('<h4>', { text: msg('preference-analysis-label') }),
+
             className: 'btn btn-danger',
            $('<span>', { style: 'font-size: small; line-height: 0.9em', html: msg('preference-analysis-view').replace('$1', '<a href="https://doc.wjghj.cn/InPageEditAnalysis/" to="_blank">https://doc.wjghj.cn/InPageEditAnalysis/</a>') }),
+
            method: function (a, modal) {
             $('<h4>', { text: msg('preference-about-label') }),
+
              $('#InPageEditSettingBox #ipeSetoutSideClose').prop('checked', true);
             $('<button>', { class: 'btn btn-secondary', onclick: "InPageEdit.about()", text: msg('preference-aboutAndHelp') }),
+
              $('#InPageEditSettingBox #ipeSetMinor').prop('checked', false);
            $('<button>', { class: 'btn btn-secondary', style: 'margin-left: 1em;', onclick: "InPageEdit.versionInfo()", text: msg('preference-updatelog') }),
+
              $('#InPageEditSettingBox #ipeSetSummary').val(msg('preference-summary-default'));
            $hr,
+
             }
            $('<strong>', { style: 'font-size: small; line-height: 0.9em', text: msg('preference-savelocal-label') }),
+
           }, {
            $('<br>'),
+
            label: msg('preference-save'),
            $('<span>', { style: 'font-size: small; line-height: 0.9em', text: msg('preference-savelocal') }).append(
+
            className: 'btn btn-primary',
              $('<a>', { href: 'javascript:;', id: 'ipeSaveLocalShow', text: msg('preference-savelocal-btn') })
+
            method: function (a, modal) {
             )
+
              InPageEdit.preference.set({
           ),
+
                outSideClose: $('#InPageEditSettingBox #ipeSetoutSideClose').prop('checked'),
        sizeClass: 'dialog',
+
                editMinor: $('#InPageEditSettingBox #ipeSetMinor').prop('checked'),
        className: 'in-page-edit ipe-preference',
+
                editSummary: $('#InPageEditSettingBox #ipeSetSummary').val()
        center: true,
+
              });
        buttons: [{
+
              modal.close();
          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'),
+
           onShow: function () {
           className: 'btn btn-primary',
+
             $('.in-page-edit.ipe-preference').data()
          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/><pre style="font-size: small;word-break: break-all;font-family: monospace;"></pre><br/>' + msg('preference-savelocal-popup-notice') + '</section>',
 
          okBtn: {
 
            className: 'btn btn-primary btn-single',
 
            label: msg('ok')
 
 
           }
 
           }
 
         });
 
         });
         $('#ipeSaveLocal pre').text('window.InPageEdit = window.InPageEdit || {}; // Keep this line\nInPageEdit.myPreference = ' + JSON.stringify({
+
         if (local.outSideClose) {
          outSideClose: $('#InPageEditSettingBox #ipeSetoutSideClose').prop('checked'),
+
           $('#InPageEditSettingBox #ipeSetoutSideClose').prop('checked', true);
           editMinor: $('#InPageEditSettingBox #ipeSetMinor').prop('checked'),
 
          editSummary: $('#InPageEditSettingBox #ipeSetSummary').val()
 
        }) + '; // InPageEdit preference');
 
      });
 
      if (outSideClose) {
 
        $('#InPageEditSettingBox #ipeSetoutSideClose').prop('checked', true);
 
      }
 
      if (minor) {
 
        $('#InPageEditSettingBox #ipeSetMinor').prop('checked', true);
 
      }
 
 
 
      if (typeof (InPageEdit.myPreference) !== 'undefined') {
 
        $('.ipe-preference .ssi-buttons .ssi-modalBtn').attr('disabled', '');
 
        if (typeof (InPageEdit.myPreference.editMinor) === 'boolean') {
 
          $('#InPageEditSettingBox #ipeSetMinor').prop('checked', InPageEdit.myPreference.editMinor).attr('disabled', '');
 
 
         }
 
         }
         if (typeof (InPageEdit.myPreference.outSideClose) === 'boolean') {
+
         if (local.editMinor) {
           $('#InPageEditSettingBox #ipeSetoutSideClose').prop('checked', InPageEdit.myPreference.outSideClose).attr('disabled', '');
+
           $('#InPageEditSettingBox #ipeSetMinor').prop('checked', true);
 
         }
 
         }
         if (typeof (InPageEdit.myPreference.editSummary) === 'string') {
+
 
           $('#InPageEditSettingBox #ipeSetSummary').attr('disabled', '').val(InPageEdit.myPreference.editSummary);
+
         if (typeof (InPageEdit.myPreference) !== 'undefined') {
        }
+
           $('.ipe-preference .ssi-buttons .ssi-modalBtn').attr('disabled', '');
        ssi_modal.dialog({
+
          if (typeof (InPageEdit.myPreference.editMinor) === 'boolean') {
          content: msg('preference-savelocal-popup-haslocal').replace('$1', '<a href="' + mw.util.getUrl('Special:Mypage/common.js') + '">' + msg('preference-savelocal-popup-yourjspage') + '</a>'),
+
            $('#InPageEditSettingBox #ipeSetMinor').prop('checked', InPageEdit.myPreference.editMinor).attr('disabled', '');
          className: 'in-page-edit',
+
          }
           center: true,
+
          if (typeof (InPageEdit.myPreference.outSideClose) === 'boolean') {
           okBtn: {
+
            $('#InPageEditSettingBox #ipeSetoutSideClose').prop('checked', InPageEdit.myPreference.outSideClose).attr('disabled', '');
             className: 'btn btn-primary btn-single'
+
           }
 +
           if (typeof (InPageEdit.myPreference.editSummary) === 'string') {
 +
             $('#InPageEditSettingBox #ipeSetSummary').attr('disabled', '').val(InPageEdit.myPreference.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'
 +
            }
 +
          });
 +
        }
 
       }
 
       }
 
     }
 
     }
 +
    Object.freeze(InPageEdit.preference.default); // 冻结重要变量
  
     /** 快速页面差异模块 **/
+
     /**
 +
    * @module 快速页面差异模块
 +
    * @param {Object} param standard MediaWiki API params
 +
    */
 
     InPageEdit.quickDiff = function (param) {
 
     InPageEdit.quickDiff = function (param) {
 
       mw.hook('InPageEdit.quickDiff').fire();
 
       mw.hook('InPageEdit.quickDiff').fire();
Line 1,286: Line 1,357:
 
         }
 
         }
 
         $('.quick-diff .pageName').html(msg('diff-title') + ': <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">
          '<colgroup>' +
+
            <colgroup>
          '<col class="diff-marker">' +
+
              <col class="diff-marker">
          '<col class="diff-content">' +
+
              <col class="diff-content">
          '<col class="diff-marker">' +
+
              <col class="diff-marker">
          '<col class="diff-content">' +
+
              <col class="diff-content">
          '</colgroup>' +
+
            </colgroup>
          '<tbody>' +
+
            <tbody>
          '<tr class="diff-title">' +
+
              <tr class="diff-title">
          '<td colspan="2" class="diff-otitle">' +
+
                <td colspan="2" class="diff-otitle">
          '<a class="" href="' + config.wgScript + '?oldid=' + data.compare.fromrevid + '">' + data.compare.fromtitle + '</a> (<span class="diff-version">' + msg('diff-version') + data.compare.fromrevid + '</span>) (<a class="editLink" href="' + config.wgScript + '?action=edit&title=' + data.compare.fromtitle + '&oldid=' + data.compare.fromrevid + '">' + msg('diff-edit') + '</a>)<br/>' + userlink(data.compare.fromuser) + '<br/>(<span class="diff-comment">' + data.compare.fromparsedcomment + '</span>)<br/><a class="prevVersion ipe-analysis-quick_diff_modalclick" href="javascript:void(0);" onclick="InPageEdit.quickDiff({fromrev:' + data.compare.fromrevid + ',torelative:\'prev\'});">←' + msg('diff-prev') + '</a>' +
+
                  <a class="" href="${config.wgScript}?oldid=${data.compare.fromrevid}">${data.compare.fromtitle}</a> (<span class="diff-version">${msg('diff-version')}${data.compare.fromrevid}</span>) (<a class="editLink" href="${config.wgScript}?action=edit&title=${data.compare.fromtitle}&oldid=${data.compare.fromrevid}">${msg('diff-edit')}</a>)
          '</td>' +
+
                  <br/>
          '<td colspan="2" class="diff-ntitle">' +
+
                  ${userlink(data.compare.fromuser)}
          '<a class="" href="' + config.wgScript + '?oldid=' + data.compare.torevid + '">' + data.compare.totitle + '</a> (<span class="diff-version">' + msg('diff-version') + data.compare.torevid + '</span>) (<a class="editLink" href="' + config.wgScript + '?action=edit&title=' + data.compare.totitle + '&oldid=' + data.compare.torevid + '">' + msg('diff-edit') + '</a>)<br/>' + userlink(data.compare.touser) + '<br/>(<span class="diff-comment">' + data.compare.toparsedcomment + '</span>)<br/><a class="nextVersion ipe-analysis-quick_diff_modalclick" href="javascript:void(0);" onclick="InPageEdit.quickDiff({fromrev:' + data.compare.torevid + ',torelative:\'next\'});">' + msg('diff-nextv') + '→</a>' +
+
                  <br/>
          '</td>' +
+
                  (<span class="diff-comment">${data.compare.fromparsedcomment}</span>)
          '</tr>' +
+
                  <br/>
          diffTable +
+
                  <a class="prevVersion ipe-analysis-quick_diff_modalclick" href="javascript:void(0);" onclick="InPageEdit.quickDiff({fromrev:${data.compare.fromrevid},torelative:'prev'});">←${msg('diff-prev')}</a>
          '<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>' +
+
                </td>
          '</tbody>' +
+
                <td colspan="2" class="diff-otitle">
           '</table>'
+
                  <a class="" href="${config.wgScript}?oldid=${data.compare.torevid}">${data.compare.totitle}</a> (<span class="diff-version">${msg('diff-version')}${data.compare.torevid}</span>) (<a class="editLink" href="${config.wgScript}?action=edit&title=${data.compare.totitle}&oldid=${data.compare.torevid}">${msg('diff-edit')}</a>)
         );
+
                  <br/>
 +
                  ${userlink(data.compare.touser)}
 +
                  <br/>
 +
                  (<span class="diff-comment">${data.compare.toparsedcomment}</span>)
 +
                  <br/>
 +
                  <a class="prevVersion ipe-analysis-quick_diff_modalclick" href="javascript:void(0);" onclick="InPageEdit.quickDiff({fromrev:${data.compare.torevid},torelative:'prev'});">${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>
 +
         `);
 
         $('.ipe-analysis-quick_diff_modalclick').click(function () {
 
         $('.ipe-analysis-quick_diff_modalclick').click(function () {
 
           _analysis('quick_diff_modalclick');
 
           _analysis('quick_diff_modalclick');
Line 1,406: Line 1,492:
 
     }
 
     }
  
     /** 获取段落编辑以及编辑链接 **/
+
     /**
 +
    * @module 获取段落编辑以及编辑链接
 +
    * @param {Element} element parent element to find edit links
 +
    */
 
     InPageEdit.articleLink = function (element) {
 
     InPageEdit.articleLink = function (element) {
 
       if (element === undefined)
 
       if (element === undefined)
Line 1,438: Line 1,527:
 
               $('<a>', {
 
               $('<a>', {
 
                 'href': 'javascript:void(0)',
 
                 'href': 'javascript:void(0)',
                 'class': 'in-page-edit-article-link'
+
                 'class': 'in-page-edit-article-link',
 +
                text: msg('quick-edit')
 +
              }).click(function () {
 +
                if (revision !== null) {
 +
                  InPageEdit.quickEdit({
 +
                    page: title,
 +
                    revision: revision
 +
                  });
 +
                } else {
 +
                  InPageEdit.quickEdit({
 +
                    page: title,
 +
                    section: section
 +
                  });
 +
                }
 
               })
 
               })
                .text(msg('quick-edit'))
+
            )
                .click(function () {
+
          );
                  if (revision !== null) {
 
                    InPageEdit.quickEdit({
 
                      page: title,
 
                      revision: revision
 
                    });
 
                  } else {
 
                    InPageEdit.quickEdit({
 
                      page: title,
 
                      section: section
 
                    });
 
                  }
 
                })));
 
 
         }
 
         }
 
       });
 
       });
 
     }
 
     }
  
     /** 快速预览文章页 **/
+
     /**
 +
    * @module 快速预览文章页
 +
    * @param params {Object}
 +
    */
 
     InPageEdit.quickPreview = function (params) {
 
     InPageEdit.quickPreview = function (params) {
 
       mw.hook('InPageEdit.quickPreview').fire();
 
       mw.hook('InPageEdit.quickPreview').fire();
Line 1,492: Line 1,585:
 
     }
 
     }
  
     /** 载入中模块 **/
+
     /**
 +
    * @module 载入中模块
 +
    * @param title
 +
    *  - √Boolean× true: Mark top progress box as done; false: Close top progress box
 +
    * - "String" Show new progress box with title @default 'Loading...'
 +
    */
 
     InPageEdit.progress = function (title) {
 
     InPageEdit.progress = function (title) {
 
       if (title === true) {
 
       if (title === true) {
Line 1,522: Line 1,620:
 
     }
 
     }
  
     /** 提交统计信息模块 **/
+
     /**
 +
    * @module 提交统计信息模块
 +
    * @description Internal module
 +
    */
 
     const _analysis = function (functionID) {
 
     const _analysis = function (functionID) {
 
       if (InPageEdit.doNotCollectMyInfo === true) {
 
       if (InPageEdit.doNotCollectMyInfo === true) {
Line 1,546: Line 1,647:
 
     }
 
     }
  
     /** 版本信息模块 **/
+
     /**
 +
    * @module 版本信息模块
 +
    * @description Show Update Logs Modal box
 +
    */
 
     InPageEdit.versionInfo = function () {
 
     InPageEdit.versionInfo = function () {
 
       // 显示模态框
 
       // 显示模态框
Line 1,577: Line 1,681:
 
     };
 
     };
  
     /** 关于插件模块 **/
+
     /**
 +
    * @module 关于插件模块
 +
    * @description Show "What is" modal box of IPE2
 +
    */
 
     InPageEdit.about = function () {
 
     InPageEdit.about = function () {
 
       ssi_modal.show({
 
       ssi_modal.show({
Line 1,589: Line 1,696:
 
     }
 
     }
  
     /** 获取版本更新提示 **/
+
     /**
 +
    * @module 获取版本更新提示
 +
    */
 
     $(window).load(function () {
 
     $(window).load(function () {
 
       var version = InPageEdit.version;
 
       var version = InPageEdit.version;
Line 1,625: Line 1,734:
 
       }
 
       }
 
       // 特殊提示
 
       // 特殊提示
       if (localStorage.InPageEditNoticeId !== InPageEdit.specialNotice.id || InPageEdit.isCanary) {
+
       if (localStorage.InPageEditNoticeId !== InPageEdit.specialNotice.id) {
 
         ssi_modal.notify('dialog', {
 
         ssi_modal.notify('dialog', {
 
           className: 'in-page-edit ipe-special-notice',
 
           className: 'in-page-edit ipe-special-notice',
Line 1,656: Line 1,765:
 
     }
 
     }
  
     /** 获取用户权限信息 **/
+
     /**
 +
    * @description 获取用户权限信息
 +
    */
 
     !(function () {
 
     !(function () {
 
       mw.user.getRights().then(function (rights) {
 
       mw.user.getRights().then(function (rights) {
Line 1,681: Line 1,792:
 
     }());
 
     }());
  
     InPageEdit.hasRight = function (right) {
+
     /**
 +
    * @module 是否拥有权限
 +
    * @param {String} right
 +
    * @return {Boolean}
 +
    */
 +
    const _hasRight = function (right) {
 
       if (config.wgUserIsBlocked === true) {
 
       if (config.wgUserIsBlocked === true) {
 
         return false;
 
         return false;
Line 1,692: Line 1,808:
 
     };
 
     };
  
     /** 页面载入完成,自动加载某些模块 **/
+
     /**
 +
    * @description 页面载入完成,自动加载某些模块
 +
    */
 
     !(function () {
 
     !(function () {
       /** 额外的模块 **/
+
       // 初始化设定
 +
      _preference.set();
 
       // 快速页面差异模块
 
       // 快速页面差异模块
 
       InPageEdit.loadQuickDiff();
 
       InPageEdit.loadQuickDiff();
      /** 读取设定 **/
 
      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));
 
      }
 
 
       // 加载段落编辑模块
 
       // 加载段落编辑模块
 
       InPageEdit.articleLink();
 
       InPageEdit.articleLink();
 
     }());
 
     }());
  
     /** Toolbox模块 **/
+
     /**
 +
    * @module Toolbox模块
 +
    */
 
     mw.hook('InPageEdit').add(function () {
 
     mw.hook('InPageEdit').add(function () {
 
       // 检测是否为文章页
 
       // 检测是否为文章页
Line 1,723: Line 1,835:
 
           $('<li>', { class: 'btn-tip-group' }).append(
 
           $('<li>', { class: 'btn-tip-group' }).append(
 
             $('<div>', { class: 'btn-tip', text: msg('quick-edit') }),
 
             $('<div>', { class: 'btn-tip', text: msg('quick-edit') }),
             $('<button>', { id: 'edit-btn', class: 'ipe-toolbox-btn material-icons', text: 'edit' })
+
             $('<button>', { id: 'edit-btn', class: 'ipe-toolbox-btn', html: '<i class="fa fa-pencil"></i>' }).click(function () {
 +
              InPageEdit.quickEdit({
 +
                page: config.wgPageName,
 +
                revision: config.wgRevisionId
 +
              });
 +
            })
 
           ),
 
           ),
 
           $('<li>', { class: 'btn-tip-group' }).append(
 
           $('<li>', { class: 'btn-tip-group' }).append(
 
             $('<div>', { class: 'btn-tip', text: msg('redirect-from') }),
 
             $('<div>', { class: 'btn-tip', text: msg('redirect-from') }),
             $('<button>', { id: 'redirectfrom-btn', class: 'ipe-toolbox-btn material-icons', text: 'flight_land' })
+
             $('<button>', { id: 'redirectfrom-btn', class: 'ipe-toolbox-btn', html: '<i class="fa fa-chevron-down"></i>' }).click(function () {
 +
              InPageEdit.quickRedirect('from');
 +
            })
 
           ),
 
           ),
 
           $('<li>', { class: 'btn-tip-group' }).append(
 
           $('<li>', { class: 'btn-tip-group' }).append(
             $('<div>', { class: 'btn-tip', text: msg('redirect-to') }),
+
             $('<div>', { class: 'btn-tip', text: msg('redirect-to') }),
             $('<button>', { id: 'redirectto-btn', class: 'ipe-toolbox-btn material-icons', text: 'flight_takeoff' })
+
             $('<button>', { id: 'redirectto-btn', class: 'ipe-toolbox-btn', html: '<i class="fa fa-chevron-up"></i>' }).click(function () {
           )
+
              InPageEdit.quickRedirect('to');
         ),
+
            })
         $('<ul>', { class: 'btn-group group2' }).append(
+
           )
           $('<div>', { style: 'display: flex;' }).append(
+
         ),
             $('<li>', { class: 'btn-tip-group' }).append(
+
         $('<ul>', { class: 'btn-group group2' }).append(
               $('<div>', { class: 'btn-tip', text: msg('quick-delete') }),
+
           $('<div>', { style: 'display: flex;' }).append(
               $('<button>', { id: 'deletepage-btn', class: 'ipe-toolbox-btn material-icons', text: 'delete_forever' })
+
             $('<li>', { class: 'btn-tip-group' }).append(
             ),
+
               $('<div>', { class: 'btn-tip', text: msg('quick-delete') }),
             $('<li>', { class: 'btn-tip-group' }).append(
+
               $('<button>', { id: 'deletepage-btn', class: 'ipe-toolbox-btn', html: '<i class="fa fa-trash"></i>' }).click(function () {
               $('<div>', { class: 'btn-tip', text: msg('quick-rename') }),
+
                InPageEdit.quickDelete();
               $('<button>', { id: 'renamepage-btn', class: 'ipe-toolbox-btn material-icons', text: 'format_italic' })
+
              })
             ),
+
             ),
             $('<li>', { class: 'btn-tip-group' }).append(
+
             $('<li>', { class: 'btn-tip-group' }).append(
               $('<div>', { class: 'btn-tip', text: msg('ipe-preference') }),
+
               $('<div>', { class: 'btn-tip', text: msg('quick-rename') }),
               $('<button>', { id: 'preference-btn', class: 'ipe-toolbox-btn material-icons', text: 'settings' })
+
               $('<button>', { id: 'renamepage-btn', class: 'ipe-toolbox-btn', html: '<i class="fa fa-italic"></i>' }).click(function () {
             )
+
                InPageEdit.quickRename();
           )
+
              })
         ),
+
             ),
         $('<button>', { class: 'ipe-toolbox-btn material-icons', id: 'toolbox-toggle', text: 'add' })
+
             $('<li>', { class: 'btn-tip-group' }).append(
       ).appendTo('body');
+
               $('<div>', { class: 'btn-tip', text: msg('ipe-preference') }),
 +
               $('<button>', { id: 'preference-btn', class: 'ipe-toolbox-btn', html: '<i class="fa fa-gear"></i>' }).click(function () {
 +
                InPageEdit.preference.modal();
 +
              })
 +
             )
 +
           )
 +
         ),
 +
         $('<button>', { class: 'ipe-toolbox-btn', id: 'toolbox-toggle', html: '<i class="fa fa-plus"></i>' })
 +
       ).appendTo('body');
 +
 
 +
      $('#ipe-edit-toolbox .btn-group button').click(function () {
 +
        _analysis('tool_box')
 +
      });
  
 
       // 设置开关等
 
       // 设置开关等
Line 1,770: Line 1,901:
 
         if (!$('#ipe-edit-toolbox #toolbox-toggle').hasClass('click')) {
 
         if (!$('#ipe-edit-toolbox #toolbox-toggle').hasClass('click')) {
 
           toolBoxInner.removeClass('opened');
 
           toolBoxInner.removeClass('opened');
        }
 
      });
 
 
      // 按钮功能
 
      $('#ipe-edit-toolbox .btn-group .ipe-toolbox-btn').click(function () {
 
        _analysis('tool_box');
 
        switch ($(this).attr('id')) {
 
          case 'edit-btn':
 
            InPageEdit.quickEdit({
 
              page: config.wgPageName,
 
              revision: config.wgRevisionId
 
            });
 
            break;
 
          case 'redirectfrom-btn':
 
            InPageEdit.quickRedirect('from');
 
            break;
 
          case 'redirectto-btn':
 
            InPageEdit.quickRedirect('to');
 
            break;
 
          case 'preference-btn':
 
            InPageEdit.preference();
 
            break;
 
          case 'deletepage-btn':
 
            InPageEdit.quickDelete();
 
            break;
 
          case 'renamepage-btn':
 
            InPageEdit.quickRename();
 
            break;
 
 
         }
 
         }
 
       });
 
       });

Revision as of 23:33, 7 May 2020

/**
 * @license GPL-3.0 GNU GENERAL PUBLIC LICENSE 3.0
 *
 * @name InPageEdit-v2
 * @description A JavaScript-based MediaWiki Plugin: 
 * @author 机智的小鱼君
 * @url https://github.com/Dragon-Fish/InPageEdit-v2
 */
!(function () {
  'use strict';
  /**
   * @description 检查插件是否已经运行,创建全局函数
   * @global InPageEdit {Object}
   */
  if (typeof (InPageEdit) !== 'undefined' && typeof (InPageEdit.version) !== 'undefined') throw '[InPageEdit] 已经有一个IPE插件在执行了';
  window.InPageEdit = window.InPageEdit || {};
  InPageEdit.isCanary = false;
  InPageEdit.api = {
    analysis: 'https://doc.wjghj.cn/inpageedit-v2/analysis/api/index.php',
    aboutUrl: 'https://dragon-fish.github.io/inpageedit-v2/',
    updatelogsJson: 'https://cdn.jsdelivr.net/gh/dragon-fish/inpageedit-v2@master/docs/update-logs.json'
  }
  InPageEdit.version = '2.13.4';
  Object.freeze(InPageEdit.api); // 冻结重要全局变量

  // 缓存常用函数
  var config = mw.config.get();

  /** 导入依赖 **/
  // 模态框
  mw.loader.load('https://cdn.jsdelivr.net/gh/dragon-fish/inpageedit-v2@master/src/ssi_modal/script.min.js');
  /** 样式表 **/
  $('title').after(
    // 模态框
    $('<link>', { rel: 'stylesheet', href: 'https://cdn.jsdelivr.net/gh/dragon-fish/inpageedit-v2@master/src/ssi_modal/style.min.css' }),
    // FontAwesome
    $('<link>', { rel: 'stylesheet', href: 'https://cdn.jsdelivr.net/npm/[email protected]/css/font-awesome.min.css' }),
    // 覆写
    $('<link>', { rel: 'stylesheet', href: 'https://cdn.jsdelivr.net/gh/dragon-fish/inpageedit-v2@master/src/override.min.css' })
  );

  /** 导入 i18n 组件 **/
  mw.loader.load('https://cdn.jsdelivr.net/gh/dragon-fish/i18n-js@master/script.min.js');
  mw.hook('dfgh.i18n').add(function (i18no) {
    i18no.loadMessages('InPageEdit-v2').then(init);
  });

  /**
   * @function InPageEdit
   */
  function init(i18n) {
    // i18n
    function msg(i) {
      return i18n.msg(i).parse();
    }
    /** HTML 组件 **/
    const $br = $('<br/>'),
      $clear = $('<div>', { style: 'clear:both' }),
      $hr = $('<hr/>'),
      $progress = $('<div>', { class: 'ipe-progress', style: 'width: 100%' }).append($('<div>', { class: 'ipe-progress-bar' }));

    /**
     * @module 快速编辑模块
     * 
     * @property {Object} options
     * @property {String} options.page edit page (default: wgPageName)
     * @property {Number} options.revision page rev ID
     * @property {Number} options.section edit section
     * @property {Boolean} options.reload if reload page after save successful (default: true)
     */
    InPageEdit.edit = InPageEdit.quickEdit = function (options) {
      mw.hook('InPageEdit.quickEdit').fire();
      /** 获取设定信息,设置缺省值 **/
      var options = options || {};
      var defaultOptions = {
        page: config.wgPageName,
        pageId: 0,
        revision: null,
        summaryRevision: '',
        section: null,
        editText: '',
        editMinor: false,
        editSummary: msg('preference-summary-default'),
        editNotice: '',
        outSideClose: true,
        jsonGet: {
          action: 'parse',
          page: options.page,
          prop: 'wikitext|langlinks|categories|templates|images|sections',
          format: 'json'
        },
        jsonPost: {},
        pageDetail: {},
        jumpTo: '',
        reload: true
      }

      /** 获取用户设置 **/
      var preference = InPageEdit.preference.get();

      /** 缓存时间戳 **/
      var date = new Date(),
        timestamp = date.getTime(),
        now = date.toUTCString();

      /** 将选项合并并标准化 **/
      options = $.extend(defaultOptions, options, preference);
      options.page = decodeURIComponent(options.page); // 解码网址 Unicode

      _analysis('quick_edit');

      if (options.revision !== null && options.revision !== '' && options.revision !== config.wgCurRevisionId) {
        ssi_modal.notify('warning', {
          className: 'in-page-edit',
          content: msg('notify-editing-history'),
          title: msg('notify-info')
        });
        delete jsonGet.page;
        jsonGet.oldid = options.revision;
        options.summaryRevision = '(' + msg('editor-summary-rivision') + '[[Special:Diff/' + options.revision + ']])';
      } else {
        if (options.section !== null && editSection !== '') {
          jsonGet.section = editSection;
        }
      }

      // Debug
      console.time('[InPageEdit] 获取页面源代码');

      // 显示主窗口
      ssi_modal.show({
        title: msg('editor-title-editing') + ': <u class="editPage">' + options.page.replace(/\_/g, ' ') + '</u>',
        content:
          $('<div>').append(
            $progress,
            $('<section>', { class: 'editForm' }).append(
              // 编辑工具条
              $('<div>', { class: 'editTools' }).append(
                $('<div>', { class: 'btnGroup' }).append(
                  $('<select>', { class: 'formatSelect' }).append(
                    $('<option>', { value: 'default', 'data-open': '', 'data-middle': '', 'data-close': '', text: msg('editor-edittool-header') }),
                    $('<option>', { value: 'h2', 'data-open': '\n== ', 'data-middle': msg('editor-edittool-header-text'), 'data-close': ' ==\n', text: 'H2' }),
                    $('<option>', { value: 'h3', 'data-open': '\n=== ', 'data-middle': msg('editor-edittool-header-text'), 'data-close': ' ===\n', text: 'H3' }),
                    $('<option>', { value: 'h4', 'data-open': '\n==== ', 'data-middle': msg('editor-edittool-header-text'), 'data-close': ' ====\n', text: 'H4' }),
                    $('<option>', { value: 'h5', 'data-open': '\n===== ', 'data-middle': msg('editor-edittool-header-text'), 'data-close': ' =====\n', text: 'H5' })
                  )
                ),
                $('<div>', { class: 'btnGroup' }).append(
                  $('<span>', { class: 'label', text: '格式' }),
                  $('<button>', { class: 'editToolBtn material-icons btn', 'data-open': "'''", 'data-middle': msg('editor-edittool-bold'), 'data-close': "'''", text: 'format_bold' }),
                  $('<button>', { class: 'editToolBtn material-icons btn', 'data-open': "''", 'data-middle': msg('editor-edittool-italic'), 'data-close': "''", text: 'format_italic' }),
                  $('<button>', { class: 'editToolBtn material-icons btn', 'data-open': '\n* ', 'data-middle': msg('editor-edittool-list-bulleted'), 'data-close': '\n', text: 'format_list_bulleted' }),
                  $('<button>', { class: 'editToolBtn material-icons btn', 'data-open': '\n# ', 'data-middle': msg('editor-edittool-list-numbered'), 'data-close': '\n', text: 'format_list_numbered' }),
                  $('<button>', { class: 'editToolBtn material-icons btn', 'data-open': '<' + 'nowiki>', 'data-middle': msg('editor-edittool-nowiki'), 'data-close': '</nowiki>', text: 'format_clear' }),
                  $('<button>', { class: 'editToolBtn material-icons btn', 'data-open': '<br>\n', 'data-middle': '', 'data-close': '', text: 'keyboard_return' })
                ),
                $('<div>', { class: 'btnGroup' }).append(
                  $('<span>', { class: 'label', text: '插入' }),
                  $('<button>', { class: 'editToolBtn material-icons btn', 'data-open': '[' + '[', 'data-middle': msg('editor-edittool-internal-link'), 'data-close': ']]', text: 'insert_link' }),
                  $('<button>', { class: 'editToolBtn material-icons btn', 'data-open': '[' + '[File:', 'data-middle': 'Example.png', 'data-close': '|thumb]]', text: 'image' }),
                  $('<button>', { class: 'editToolBtn material-icons btn', 'data-open': '\n<' + 'gallery>\n', 'data-middle': 'Example1.jpg|Description\nExample2.png|Description', 'data-close': '\n</gallery>\n', text: 'collections' })
                ),
                $('<div>', { class: 'btnGroup extra', style: 'display: none' }).append(
                  $('<span>', { class: 'label', text: '自定义' })
                ),
                $('<div>', { class: 'btnGroup special-tools', style: 'float: right' }).append(
                  $('<button>', { class: 'material-icons btn', text: 'find_in_page' }).click(function () {
                    InPageEdit.findAndReplace($('.ipe-editor.timestamp-' + timestamp + ' .editArea'))
                  })
                )
              ),
              // 编辑框
              $('<textarea>', { class: 'editArea', style: 'margin-top: 0;' }),
              // 页面分析
              $('<div>', { class: 'editOptionsLabel editForm' }).append(
                $('<aside>', { class: 'detailArea' }).append(
                  $('<label>', { class: 'detailToggle', text: msg('editor-detail-button-toggle') }),
                  $('<div>', { class: 'detailBtnGroup' }).append(
                    $('<a>', { href: 'javascript:;', class: 'detailBtn', id: 'showTemplates', text: msg('editor-detail-button-templates') }),
                    $('<span>', { text: ' | ' }),
                    $('<a>', { href: 'javascript:;', class: 'detailBtn', id: 'showImages', text: msg('editor-detail-button-images') })
                  )
                ),
                // 摘要&小编辑
                $('<label>', { for: 'editSummary', text: msg('editSummary') }),
                $('<br>'),
                $('<input>', { class: 'editSummary', id: 'editSummary', placeholder: 'Edit via InPageEdit~', value: options.editSummary.replace(/\$oldid/ig, options.summaryRevision) }),
                $('<br>'),
                $('<label>').append(
                  $('<input>', { type: 'checkbox', class: 'editMinor', id: 'editMinor', checked: editMinor }),
                  $('<span>', { text: msg('markAsMinor') })
                )
              )
            )
          ),
        outSideClose: outSideClose,
        className: 'in-page-edit ipe-editor timestamp-' + timestamp,
        sizeClass: 'large',

        /* 按钮 */
        buttons: [{
          label: msg('editor-button-save'),
          className: 'btn btn-primary leftBtn editForm save-btn',
          method: function (e, modal) {
            var modal = modal;
            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 = options.section,
                    summary = $('.ipe-editor.timestamp-' + timestamp + ' .editSummary').val();
                  postArticle({
                    text: text,
                    page: options.page,
                    minor: minor,
                    section: section,
                    summary: summary
                  }, modal);
                }
              });
          }
        }, {
          label: msg('editor-button-preview'),
          className: 'btn btn-secondary leftBtn editForm',
          method: function () {
            _analysis('preview_edit');
            var text = $('.ipe-editor.timestamp-' + timestamp + ' .editArea').val();
            InPageEdit.quickPreview({
              action: 'parse',
              title: options.page,
              text: text,
              prop: 'text',
              preview: true,
              format: 'json'
            });
          }
        }, {
          label: msg('editor-button-diff'),
          className: 'btn btn-secondary leftBtn editForm',
          method: function () {
            _analysis('quick_diff_edit');
            var text = $('.ipe-editor.timestamp-' + timestamp + ' .editArea').val();
            var diffJson = {};
            diffJson.fromtext = options.editText;
            diffJson.totext = text;
            diffJson.hideBtn = true;
            diffJson.pageName = options.page;
            diffJson.isPreview = true;
            InPageEdit.quickDiff(diffJson);
          }
        }, {
          label: msg('cancel'),
          className: 'btn btn-danger',
          method: function (e, modal) {
            modal.close();
          }
        }
        ],

        /* 预加载 */
        beforeShow: function () {
          // 设置样式
          $('.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');
          $('.ipe-editor.timestamp-' + timestamp + ' .leftBtn').appendTo('.ipe-editor.timestamp-' + timestamp + ' .ssi-leftButtons');

          /** Edit-Tool 扩展 **/
          function insertText(strings, obj) {
            var strings = strings,
              textarea = obj || $('.in-page-edit.ipe-editor .editArea')[0],
              start = textarea.selectionStart,
              stop = textarea.selectionEnd,
              selectedText = textarea.value.slice(start, stop);
            textarea.value =
              textarea.value.slice(0, start) +
              (strings.open || '') +
              (selectedText || strings.middle || '') +
              (strings.close || '') +
              textarea.value.slice(stop);
            var selectStart = start + (strings.open.length || 0);
            textarea.setSelectionRange(selectStart, selectStart + (selectedText.length || strings.middle.length || 0));
            textarea.focus();
          }
          // 用户自定义按钮
          if (InPageEdit.buttons) {
            var btns = InPageEdit.buttons;
            $('.ipe-editor.timestamp-' + timestamp + ' .btnGroup.extra').show();
            function addBtn(open, middle, close, icon) {
              var open = open || '',
                middle = middle || '',
                close = close || '',
                icon = 'fa-' + icon || 'fa-wrench';
              $('.ipe-editor.timestamp-' + timestamp + ' .btnGroup.extra').append(
                $('<button>', { class: 'editToolBtn btn', 'data-open': open, 'data-middle': middle, 'data-close': close, html: `<i class="fa ${icon}"></i>` })
              );
            }
            for (var i = 0; i < btns.length; i++) {
              var btn = btns[i];
              addBtn(btn.open, btn.middle, btn.close, btn.text);
            }
          }
          $('.ipe-editor.timestamp-' + timestamp + ' .editToolBtn').click(function (e) {
            e.preventDefault();
            var $this = $(this),
              $open = $this.attr('data-open') || '',
              $middle = $this.attr('data-middle') || '',
              $close = $this.attr('data-close') || '';
            insertText({
              open: $open,
              middle: $middle,
              close: $close
            }, $('.ipe-editor.timestamp-' + timestamp + ' .editArea')[0]);
          });
          $('.ipe-editor.timestamp-' + timestamp + ' .formatSelect').change(function () {
            var $this = $(this),
              val = $this.val(),
              $open = $this.find('[value="' + val + '"]').attr('data-open') || '',
              $middle = $this.find('[value="' + val + '"]').attr('data-middle') || '',
              $close = $this.find('[value="' + val + '"]').attr('data-close') || '';
            insertText({
              open: $open,
              middle: $middle,
              close: $close
            }, $('.ipe-editor.timestamp-' + timestamp + ' .editArea')[0]);
            $this.val('default');
          });
        },
        /**
         * @event onShow
         * @description 模态框显示后
         */
        onShow: function (a, modal) {
          mw.hook('InPageEdit.quickEdit.modal').fire();
          // 绑定事件,在尝试离开页面时提示
          $('.ipe-editor.timestamp-' + timestamp + ' .editArea').change(function () {
            $(this).attr('data-modifiled', 'true');
            $(window).bind('beforeunload', function () {
              return msg('window-leave-confirm');
            });
          });
          // 获取权限
          if (_hasRight('edit') === false) {
            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 + ' .save-btn').addClass('btn-danger');
          }

          // 解析页面内容
          new mw.Api().get(options.jsonGet).done(function (data) {
            var data = data;
            console.timeEnd('[InPageEdit] 获取页面源代码');
            contentDone(data);
          }).fail(function (a, b, errorThrown) {
            console.timeEnd('[InPageEdit] 获取页面源代码');
            console.warn('[InPageEdit]警告:无法获取页面内容');
            var data = errorThrown;
            contentDone(data);
          });

          // 页面内容获取完毕,后续工作
          function contentDone(data) {
            var data = data;
            options.pageDetail = data;

            if (data.hasOwnProperty('error')) {
              console.warn('[InPageEdit]警告:无法获取页面内容');
              options.editText = '<!-- ' + data.error.info + ' -->';
              options.pageId = 0;
              $('.ipe-editor.timestamp-' + timestamp + ' .detailArea').hide();
            } else {
              options.editText = data.parse.wikitext['*'];
              options.pageId = data.parse.pageid;
            }
            // 设定一堆子样式
            $('.ipe-editor.timestamp-' + timestamp + ' .ipe-progress').hide();
            $('.ipe-editor.timestamp-' + timestamp + ' .editForm').fadeIn(500);
            $('.ipe-editor.timestamp-' + timestamp + ' .editArea').val(editText + '\n');

            if (options.editSection !== null) {
              var val = $('.ipe-editor.timestamp-' + timestamp + ' .editSummary').val();
              val = val.replace(/\$section/ig, '/* ' + data.parse.sections[0].line + ' */');
              $('.ipe-editor.timestamp-' + timestamp + ' .editSummary').val(val);
              $('.ipe-editor.timestamp-' + timestamp + ' .editPage').after('<span class="editSection">→' + data.parse.sections[0].line + '</span>');
              jumpTo = '#' + data.parse.sections[0].anchor;
            } else {
              var val = $('.ipe-editor.timestamp-' + timestamp + ' .editSummary').val();
              val = val.replace(/\$section/ig, '');
              $('.ipe-editor.timestamp-' + timestamp + ' .editSummary').val(val);
              jumpTo = '';
            }
            if (options.revision !== null && options.revision !== '' && options.revision !== config.wgCurRevisionId) {
              $('.ipe-editor.timestamp-' + timestamp + ' .editPage').after('<span class="editRevision">(' + msg('editor-title-editRevision') + ':' + options.revision + ')</span>');
            }

            // 获取页面基础信息
            console.time('[InPageEdit] 获取页面基础信息');
            var queryJson = {
              action: 'query',
              prop: 'revisions|info',
              inprop: 'protection',
              format: 'json'
            }
            if (pageId !== -1) {
              queryJson.pageids = options.pageId;
            } else {
              queryJson.titles = options.editPage;
            }
            new mw.Api().get(queryJson).done(function (data) {
              var data = data;
              console.info('[InPageEdit] 获取页面基础信息成功');
              console.timeEnd('[InPageEdit] 获取页面基础信息');
              $('.ipe-editor.timestamp-' + timestamp).attr('data-basetimestamp', data.query.pages[pageId].touched);
              queryDone(data);
            }).fail(function (a, b, errorThrown) {
              var data = errorThrown;
              console.timeEnd('[InPageEdit] 获取页面基础信息');
              console.warn('[InPageEdit] 获取页面基础信息失败');
              $('.ipe-editor.timestamp-' + timestamp).attr('data-basetimestamp', now);
              queryDone(data);
            });

            // 页面保护等级和编辑提示等
            function queryDone(data) {
              var data = data;
              if (data.query.pages[options.pageId]['protection'].length > 0) {
                var protection = data.query.pages[options.pageId]['protection'];
                for (var i = 0; i < protection.length; i++) {
                  if (protection[i].type === 'edit') {
                    if (
                      (protection[i].level === 'autoconfirmed' && !_hasRight('autoconfirmed')) ||
                      (protection[i].level === 'sysop' && !_hasRight('editprotected')) ||
                      (config.wgNamespaceNumber === 8 && !_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 + ' .save-btn').addClass('btn-danger');
                    }
                  }
                }
              }
              // 获取编辑提示
              $.get(config.wgScript, {
                action: 'raw',
                title: 'MediaWiki:Editnotice-' + data.query.pages[options.pageId].ns
              }, function (data) {
                var data = data;
                new mw.Api().post({
                  action: 'parse',
                  title: options.editPage,
                  contentmodel: 'wikitext',
                  preview: true,
                  text: data
                }).done(function (data) {
                  var data = data;
                  editNotice = data.parse.text['*'];
                  $('.ipe-editor.timestamp-' + timestamp + ' .ssi-modalTitle').append(
                    $('<a>', {
                      id: 'showEditNotice',
                      href: 'javascript:;',
                      html: '<i class="material-icons">info</i> ' + msg('editor-has-editNotice')
                    }).click(function () {
                      ssi_modal.show({
                        className: 'in-page-edit',
                        center: true,
                        title: msg('editor-title-editNotice'),
                        content: '<section class="editNotice">' + editNotice + '</section>'
                      });
                    })
                  );
                });
              });
            }
          }
        },

        /* 确认是否取消 */
        beforeClose: function (modal) {
          if ($('.ipe-editor.timestamp-' + timestamp + ' .editArea').attr('data-modifiled') !== 'true') {
            close();
            return;
          } else if ($('.ipe-editor.timestamp-' + timestamp + ' .editArea').attr('data-confirmclose') === 'true') {
            closeNoReload();
            return;
          }
          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) {
                close();
              }
            });
          function close() {
            $(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')
            });
          }
          function closeNoReload() {
            modal.options.keepContent = false;
            modal.options.beforeClose = '';
            modal.close();
          }
          return false;
        }
      });

      // 页面详情模块
      $('.ipe-editor.timestamp-' + timestamp + ' .detailBtnGroup .detailBtn').click(function () {
        var $this = $(this),
          id = $this.attr('id'),
          content = $('<ul>');
        switch (id) {
          case 'showTemplates':
            var templates = options.pageDetail.parse.templates,
              templateName;
            for (var i = 0; i < templates.length; i++) {
              templateName = templates[i]['*'];
              $('<li>').append(
                $('<a>', { href: mw.util.getUrl(templateName), target: '_blank', text: templateName }),
                $('<span> (</span>'),
                $('<a>', { href: 'javascript:;', text: msg('quick-edit'), onclick: InPageEdit.quickEdit({ page: templateName, reload: false }) }),
                $('<span>)</span>')
              ).appendTo(content);
            }
            ssi_modal.show({
              className: 'in-page-edit quick-edit-detail',
              sizeClass: 'dialog',
              title: msg('editor-detail-title-templates'),
              content: content
            });
            break;
          case 'showImages':
            var images = options.pageDetail.parse.images,
              imageName;
            for (var i = 0; i < images.length; i++) {
              imageName = images[i];
              $('li').append(
                $('<a>', { href: mw.util.getUrl('File:' + imageName), target: '_balnk' }),
                $('<span> (</span>'),
                $('<a>', { href: 'javascript:;', class: 'quickViewImage', text: msg('editor-detail-images-quickview') }).bind('click', function () {
                  var imageName = imageName;
                  ssi_modal.show({
                    className: 'in-page-edit quick-view-image',
                    center: true,
                    title: imageName,
                    content: $('<center>').append(
                      $('<image>', { src: config.wgScript + '?title=Special:Filepath/' + imageName, style: 'max-width: 80%; max-height: 60vh' })
                    ),
                    buttons: [{
                      label: msg('editor-detail-images-upload'),
                      className: 'btn btn-primary',
                      method: function (a, modal) { window.open(config.wgScript + '?title=Special:Upload&wpDestFile=' + imageName + '&wpForReUpload=1') }
                    }, {
                      label: msg('close'),
                      className: 'btn btn-secondary',
                      method: function (a, modal) { modal.close() }
                    }]
                  });
                }),
                $('<span> | </span>'),
                $('<a>', { href: config.wgScript + '?title=Special:Upload&wpDestFile=' + imageName + '&wpForReUpload=1', target: '_balnk', text: msg('editor-detail-images-upload') }),
                $('<span>)</span>')
              ).appendTo(content);
            }
            ssi_modal.show({
              className: 'in-page-edit quick-edit-detail',
              sizeClass: 'dialog',
              title: msg('editor-detail-title-images'),
              content: content
            });
            break;
        }
      });

      // 发布编辑模块
      function postArticle(pValue, modal) {
        var pValue = pValue;
        var modal = modal;
        _analysis('quick_edit_save');
        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;
        }

        new mw.Api().postWithToken('csrf', jsonPost).always(function (data) {
          InPageEdit.progress(true);
          // GitHub@issue#40
          if (data.edit.result !== 'Success' || data.errors !== undefined) {
            // 出事了
            InPageEdit.progress(false);
            var errorCode,
              errorInfo,
              errorMore = '';
            if (data.errors !== undefined) {
              errorCode = data.errors.code;
              errorInfo = data.errors['*'];
              errorMore = '';
            } else if (data.edit.result !== 'Success') {
              errorCode = data.edit.code || 'Unknown';
              errorInfo = data.edit.info || 'Reason unknown.';
              errorMore = data.edit.warning || '';
            } else {
              errorCode = 'unknown';
              errorInfo = 'Reason unknown.';
              errorMore = 'Please contact plug-in author or try again.'
            }
            ssi_modal.show({
              className: 'in-page-edit',
              sizeClass: 'dialog',
              title: msg('editor-save-error'),
              content: errorInfo + '<hr style="clear: both" />' + errorMore
            });
            ssi_modal.notify('error', {
              className: 'in-page-edit',
              position: 'right top',
              closeAfter: {
                time: 15
              },
              title: msg('notify-error'),
              content: msg('editor-save-error') + ':<code>' + errorCode + '</code>'
            });

            console.error('[InPageEdit] Submit failed: \nCode: ' + errorCode);
            return;
          } else {
            // 保存正常
            if (reloadPage) {
              var content;
              $(window).unbind('beforeunload');
              content = msg('notify-save-success');
              setTimeout(function () {
                if (pValue.page === config.wgPageName) {
                  window.location = mw.util.getUrl(pValue.page) + jumpTo;
                  window.location.reload();
                } else {
                  window.location.reload();
                }
              }, 500);
            } else {
              console.info('[InPageEdit] 将不会重载页面!');
              content = msg('notify-save-success-noreload');
              setTimeout(function () {
                InPageEdit.progress(false);
                $('.ipe-editor.timestamp-' + timestamp + ' .editArea').attr('data-confirmclose', 'true');
                modal.close();
              }, 1500);
            }

            ssi_modal.notify('success', {
              className: 'in-page-edit',
              position: 'right top',
              title: msg('notify-success'),
              content: content
            });
          }
        });
      }
    }

    /**
     * @module 查找替换模块
     * @param contengut <Element> Textarea
     */
    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',
        sizeClass: 'dialog',
        center: true,
        outSideClose: false,
        // position: 'right bottom',
        title: msg('fAndR-title'),
        content:
          $('<div>', { class: 'module far-module' }).append(
            $('<div>', { class: 'module_content', id: 'findfielddiv' }).append(
              $('<section>').append(
                $('<h4>', { text: msg('fAndR-find-text') }),
                $('<textarea>', { id: 'find_this', style: 'margin: 0', rows: 4 }),
                $('<h4>', { text: msg('fAndR-replace-text') }),
                $('<textarea>', { id: 'replace_with', style: 'margin: 0', rows: 4 })
              ),
              $('<section>', { style: 'padding: 7px 0' }).append(
                $('<label>').append(
                  $('<input>', { type: 'checkbox', id: 'globl', checked: '' }),
                  $('<span>', { text: msg('fAndR-globl') })
                ),
                $('<br>'),
                $('<label>').append(
                  $('<input>', { type: 'checkbox', id: 'case_sen' }),
                  $('<span>', { text: msg('fAndR-case-sen') })
                ),
                $('<br>'),
                $('<label>').append(
                  $('<input>', { type: 'checkbox', id: 'regex_search' }),
                  $('<span>', { text: msg('fAndR-enable-regex') })
                ),
              )
            )
          ),
        buttons: [
          {
            label: msg('fAndR-button-undo'),
            className: 'btn btn-danger',
            method: function (e, modal) {
              contengut.val(origin);
              ssi_modal.notify('info', {
                className: 'in-page-edit',
                title: msg('notify-info'),
                content: msg('notify-fAndR-undo')
              });
              // modal.close();
            }
          },
          {
            className: 'btn btn-primary',
            label: msg('fAndR-button-replace'),
            method: function (a, modal) {
              /**
               * 查找替换主函数
               * 借鉴:https://dev.fandom.com/wiki/MediaWiki:FindAndReplace/code.js
               **/
              if ($('#find_this').val() === '') 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('success', {
                className: 'in-page-edit',
                title: msg('notify-success'),
                content: msg('notify-fAndR-done').replace('$1', rcount)
              });
              // modal.close();
            }
          }
        ]
      });
    }

    /** 快速重定向模块 **/
    InPageEdit.redirect = InPageEdit.quickRedirect = function (type) {
      mw.hook('InPageEdit.quickRedirect').fire();
      _analysis('find_replace');
      var text = '#REDIRECT [[:$1]]',
        question,
        target,
        json = {
          action: 'edit',
          minor: JSON.parse(localStorage.getItem('InPageEditPreference')).editMinor,
          token: mw.user.tokens.get('editToken'),
          errorformat: 'plaintext'
        },
        summary;

      switch (type) {
        case 'to':
          json.title = config.wgPageName;
          question = msg('redirect-question-to').replace('$1', '<b>' + config.wgPageName.replace(/\_/g, ' ') + '</b>');
          break;
        case 'from':
          question = msg('redirect-question-from').replace('$1', '<b>' + config.wgPageName.replace(/\_/g, ' ') + '</b>');
          json.text = text.replace('$1', config.wgPageName);
          break;
      }
      ssi_modal.show({
        outSideClose: false,
        className: 'in-page-edit quick-redirect',
        center: true,
        sizeClass: 'dialog',
        title: msg('redirect-title'),
        content: $('<div>').append(
          $('<section>').append(
            $('<span>', { html: question }),
            $('<br>'),
            $('<input>', { id: 'redirect-page', style: 'width:96%' }).click(function () { $(this).css('box-shadow', '') }),
            $('<br>'),
            $('<label>', { for: 'redirect-reason', text: msg('editSummary') }),
            $('<input>', { id: 'redirect-reason', style: 'width:96%' })
          ),
          $progress.css('display', 'none')
        ),
        buttons: [{
          label: msg('confirm'),
          className: 'btn btn-primary btn-single okBtn',
          method: function (a, modal) {
            target = $('.in-page-edit.quick-redirect #redirect-page').val();
            if (target === '' || target === config.wgPageName || target === config.wgPageName.replace(/\_/g, ' ')) {
              $('.in-page-edit.quick-redirect #redirect-page').css('box-shadow', '0 0 4px #f00');
              return;
            }

            _analysis('quick_redirect');

            summary = msg('redirect-summary') + ' → [[:' + target + ']]';
            if ($('.in-page-edit.quick-redirect #redirect-reason').val() !== '') {
              summary = summary + ' (' + $('.in-page-edit.quick-redirect #redirect-reason').val() + ')';
            }
            json.summary = summary;

            $('.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.text = text.replace('$1', target);
                break;
              case 'from':
                json.title = target;
                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')
              });
            });
          }
        }
        ]
      });
    }

    /** 删除页面模块 **/
    InPageEdit.deletepage = InPageEdit.quickDelete = function (page) {
      mw.hook('InPageEdit.quickDelete').fire();
      var reason,
        page = page || config.wgPageName;

      ssi_modal.show({
        outSideClose: false,
        className: 'in-page-edit quick-delete',
        center: true,
        sizeClass: 'dialog',
        title: msg('delete-title'),
        content: $('<div>').append(
          $('<section>', { id: 'InPageEditDeletepage' }).append(
            $('<span>', { html: msg('delete-reason').replace('$1', '<b>' + page.replace(/\_/g, ' ') + '</b>') }),
            $('<br>'),
            $('<label>', { for: 'delete-reason', text: msg('editSummary') }),
            $('<input>', { id: 'delete-reason', style: 'width:96%', onclick: "$(this).css('box-shadow', '')" })
          )
        ),
        beforeShow: function () {
          if (!_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) {
              reason = $('#InPageEditDeletepage #delete-reason').val();
              if (reason === '') {
                $('#InPageEditDeletepage #delete-reason').css('box-shadow', '0 0 4px #f00');
                return;
              }
              _analysis('quick_delete');

              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) {
                  reason = msg('delete-title') + ' (' + reason + ')';
                  new mw.Api().postWithToken('csrf', {
                    action: 'delete',
                    title: page,
                    reason: reason,
                    format: 'json'
                  }).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.error['*'] + '(<code>' + errorThrown.error['code'] + '</code>)</span>'
                    });
                  });
                  modal.close();
                } else {
                  return false;
                }
              });
            }
          }
        ]
      });
    }

    /** 重命名模块 **/
    InPageEdit.renamepage = InPageEdit.quickRename = function (from, to) {
      mw.hook('InPageEdit.quickRename').fire();
      var from = from || config.wgPageName,
        to = 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>').append(
            $('<label>', { for: 'move-to', html: msg('rename-moveTo').replace('$1', '<b>' + from.replace(/\_/g, ' ') + '</b>') }),
            $('<br>'),
            $('<input>', { id: 'move-to', style: 'width:96%', onclick: "$(this).css('box-shadow','')" }),
            $('<br>'),
            $('<label>', { for: 'move-reason', text: msg('editSummary') }),
            $('<br>'),
            $('<input>', { id: 'move-reason', style: 'width:96%' }),
            $('<br>'),
            $('<label>').append(
              $('<input>', { type: 'checkbox', id: 'movetalk', checked: 'checked' }),
              $('<span>', { text: msg('rename-movetalk') })
            ),
            $('<br>'),
            $('<label>').append(
              $('<input>', { type: 'checkbox', id: 'movesubpages', checked: 'checked' }),
              $('<span>', { text: msg('rename-movesubpages') })
            ),
            $('<br>'),

            $('<label>').append(
              $('<input>', { type: 'checkbox', id: 'noredirect' }),
              $('<span>', { text: msg('rename-noredirect') })
            ),
          ),
        buttons: [{
          label: msg('cancel'),
          className: 'btn btn-secondary',
          method: function (a, modal) {
            modal.close();
          }
        }, {
          label: msg('confirm'),
          className: 'btn btn-primary',
          method: function () {
            to = $('.in-page-edit.quick-rename #move-to').val();
            if (to === '' || to === config.wgPageName || to === config.wgPageName.replace(/\_/g, ' ')) {
              $('.in-page-edit.quick-rename #move-to').css('box-shadow', '0 0 4px #f00');
              return;
            }

            _analysis('quick_move');

            InPageEdit.progress(msg('editor-title-saving'));
            movetalk = $('.in-page-edit.quick-rename #movetalk').prop('checked');
            movesubpages = $('.in-page-edit.quick-rename #movesubpages').prop('checked');
            noredirect = $('.in-page-edit.quick-rename #noredirect').prop('checked');
            reason = $('.in-page-edit.quick-rename #move-reason').val();

            if (reason === '') {
              reason = msg('rename-summary') + ' → [[:' + to + ']]';
            } else {
              reason = msg('rename-summary') + ' → [[:' + to + ']] (' + reason + ')';
            }
            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 = config.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 (!_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;
          }
        }
      });
    }

    /**
     * @module 个人设置模块
     */
    InPageEdit.preference = {
      /* 预设值 */
      "default": {
        doNotCollectMyInfo: false,
        doNotShowLocalWarn: false,
        editMinor: false,
        editSummary: msg('preference-summary-default'),
        outSideClose: true,
        watchList: mw.user.options.get().watchdefault
      },
      /* 获取设置 */
      get: function () {
        var local = localStorage.getItem('InPageEditPreference') || '{}',
          local = JSON.parse(local);
        if (typeof InPageEdit.myPreference === 'object') {
          local = $.extend(local, InPageEdit.myPreference);
        }
        return $.extend(InPageEdit.preference.default, local);
      },
      /**
       * @description 保存设置
       * @param {Object} options 
       */
      set: function (options) {
        var options = options || {};
        if (typeof options !== 'object') return;
        options = $.extend(InPageEdit.preference.get(), options);
        options = JSON.stringify(options);
        localStorage.setItem('InPageEditPreference', options);
      },
      /**
       * @description 用户图形界面
       */
      modal: function () {
        mw.hook('InPageEdit.preference').fire();
        InPageEdit.preference.set();
        var local = InPageEdit.preference.get();
        _analysis('plugin_setting');

        ssi_modal.show({
          outSideClose: false,
          title: msg('preference-title') + ' - ' + InPageEdit.version,
          content:
            $('<section>', { id: 'InPageEditSettingBox', class: 'ipe-preference' }).append(
              $('<h4>', { text: msg('preference-editor-label') }),
              $('<label>').append(
                $('<input>', { id: 'outSideClose', type: 'checkbox' }),
                $('<span>', { text: msg('preference-outSideClose') })
              ),
              $('<br>'),
              $('<label>').append(
                $('<input>', { id: 'editMinor', type: 'checkbox' }),
                $('<span>', { text: msg('preference-setMinor') })
              ),
              $('<br>'),
              $('<h4>', { text: msg('preference-summary-label') }),
              $('<label>', { for: 'editSummary', style: 'padding-left: 0; font-size: small', html: msg('preference-editSummary').replace('%br%', '<br/>').replace('$1', '<code>$oldid</code>').replace('$2', '<code>' + msg('editor-summary-rivision') + ' [[Special:Diff/oldid]]</code>').replace('$3', '<code>$section</code>').replace('$4', '<code>/* SECTION TITLE */</code>') }),
              $('<br>'),
              $('<input>', { id: 'editSummary', style: 'width: 96%', value: local.editSummary, placeholder: 'Edit via InPageEdit, yeah~' }),
              $('<h4>', { text: msg('preference-analysis-label') }),
              $('<span>', { style: 'font-size: small; line-height: 0.9em', html: msg('preference-analysis-view').replace('$1', '<a href="https://doc.wjghj.cn/inpageedit-v2/analysis/" target="_blank">https://doc.wjghj.cn/inpageedit-v2/analysis/</a>') }),
              $('<h4>', { text: msg('preference-about-label') }),
              $('<button>', { class: 'btn btn-secondary', onclick: "InPageEdit.about()", text: msg('preference-aboutAndHelp') }),
              $('<button>', { class: 'btn btn-secondary', style: 'margin-left: 1em;', onclick: "InPageEdit.versionInfo()", text: msg('preference-updatelog') }),
              $hr,
              $('<strong>', { style: 'font-size: small; line-height: 0.9em', text: msg('preference-savelocal-label') }),
              $('<br>'),
              $('<span>', { style: 'font-size: small; line-height: 0.9em', text: msg('preference-savelocal') }).append(
                $('<a>', { href: 'javascript:;', id: 'ipeSaveLocalShow', text: msg('preference-savelocal-btn') }).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/><textarea style="font-size: small;" readonly></textarea><br/>' + msg('preference-savelocal-popup-notice') + '</section>',
                    okBtn: {
                      className: 'btn btn-primary btn-single',
                      label: msg('ok')
                    }
                  });
                  $('#ipeSaveLocal textarea').val('window.InPageEdit = window.InPageEdit || {}; // Keep this line\nInPageEdit.myPreference = ' + JSON.stringify({
                    outSideClose: $('#InPageEditSettingBox #ipeSetoutSideClose').prop('checked'),
                    editMinor: $('#InPageEditSettingBox #ipeSetMinor').prop('checked'),
                    editSummary: $('#InPageEditSettingBox #ipeSetSummary').val()
                  }) + '; // InPageEdit preference');
                })
              )
            ),
          sizeClass: 'dialog',
          className: 'in-page-edit ipe-preference',
          center: true,
          buttons: [{
            label: msg('preference-reset'),
            className: 'btn btn-danger',
            method: function (a, modal) {
              $('#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) {
              InPageEdit.preference.set({
                outSideClose: $('#InPageEditSettingBox #ipeSetoutSideClose').prop('checked'),
                editMinor: $('#InPageEditSettingBox #ipeSetMinor').prop('checked'),
                editSummary: $('#InPageEditSettingBox #ipeSetSummary').val()
              });
              modal.close();
            }
          }
          ],
          onShow: function () {
            $('.in-page-edit.ipe-preference').data()
          }
        });
        if (local.outSideClose) {
          $('#InPageEditSettingBox #ipeSetoutSideClose').prop('checked', true);
        }
        if (local.editMinor) {
          $('#InPageEditSettingBox #ipeSetMinor').prop('checked', true);
        }

        if (typeof (InPageEdit.myPreference) !== 'undefined') {
          $('.ipe-preference .ssi-buttons .ssi-modalBtn').attr('disabled', '');
          if (typeof (InPageEdit.myPreference.editMinor) === 'boolean') {
            $('#InPageEditSettingBox #ipeSetMinor').prop('checked', InPageEdit.myPreference.editMinor).attr('disabled', '');
          }
          if (typeof (InPageEdit.myPreference.outSideClose) === 'boolean') {
            $('#InPageEditSettingBox #ipeSetoutSideClose').prop('checked', InPageEdit.myPreference.outSideClose).attr('disabled', '');
          }
          if (typeof (InPageEdit.myPreference.editSummary) === 'string') {
            $('#InPageEditSettingBox #ipeSetSummary').attr('disabled', '').val(InPageEdit.myPreference.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'
            }
          });
        }
      }
    }
    Object.freeze(InPageEdit.preference.default); // 冻结重要变量

    /**
     * @module 快速页面差异模块
     * @param {Object} param standard MediaWiki API params
     */
    InPageEdit.quickDiff = function (param) {
      mw.hook('InPageEdit.quickDiff').fire();
      _analysis('quick_diff');
      if ($('[href*="mediawiki.diff.styles"]').length < 1) {
        mw.loader.load(mw.util.wikiScript('load') + '?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 class="diff-user" 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="${config.wgScript}?oldid=${data.compare.fromrevid}">${data.compare.fromtitle}</a> (<span class="diff-version">${msg('diff-version')}${data.compare.fromrevid}</span>) (<a class="editLink" href="${config.wgScript}?action=edit&title=${data.compare.fromtitle}&oldid=${data.compare.fromrevid}">${msg('diff-edit')}</a>)
                  <br/>
                  ${userlink(data.compare.fromuser)}
                  <br/>
                  (<span class="diff-comment">${data.compare.fromparsedcomment}</span>)
                  <br/>
                  <a class="prevVersion ipe-analysis-quick_diff_modalclick" href="javascript:void(0);" onclick="InPageEdit.quickDiff({fromrev:${data.compare.fromrevid},torelative:'prev'});">←${msg('diff-prev')}</a>
                </td>
                <td colspan="2" class="diff-otitle">
                  <a class="" href="${config.wgScript}?oldid=${data.compare.torevid}">${data.compare.totitle}</a> (<span class="diff-version">${msg('diff-version')}${data.compare.torevid}</span>) (<a class="editLink" href="${config.wgScript}?action=edit&title=${data.compare.totitle}&oldid=${data.compare.torevid}">${msg('diff-edit')}</a>)
                  <br/>
                  ${userlink(data.compare.touser)}
                  <br/>
                  (<span class="diff-comment">${data.compare.toparsedcomment}</span>)
                  <br/>
                  <a class="prevVersion ipe-analysis-quick_diff_modalclick" href="javascript:void(0);" onclick="InPageEdit.quickDiff({fromrev:${data.compare.torevid},torelative:'prev'});">${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>
        `);
        $('.ipe-analysis-quick_diff_modalclick').click(function () {
          _analysis('quick_diff_modalclick');
        });
        $('.quick-diff button.toDiffPage').click(function () {
          location.href = config.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">' + data.warnings.compare['*'] + '</span>');
        } else if (data.compare.torevid === undefined && param.isPreview !== true) {
          $('.quick-diff .diff-ntitle').html('<span class="noNextVerson">' + data.warnings.compare['*'] + '</span>');
        }
        // GitHub@issue#5 修复被隐藏版本的问题
        if (data.compare.fromtexthidden !== undefined) {
          $('.quick-diff .diff-otitle .diff-version').addClass('diff-hidden-history');
        }
        if (data.compare.totexthidden !== undefined) {
          $('.quick-diff .diff-ntitle .diff-version').addClass('diff-hidden-history');
        }
        if (data.compare.fromuserhidden !== undefined) {
          $('.quick-diff .diff-otitle .diff-user').addClass('diff-hidden-history');
        }
        if (data.compare.touserhidden !== undefined) {
          $('.quick-diff .diff-ntitle .diff-user').addClass('diff-hidden-history');
        }
        if (data.compare.fromcommenthidden !== undefined) {
          $('.quick-diff .diff-otitle .diff-comment').addClass('diff-hidden-history');
        }
        if (data.compare.tocommenthidden !== undefined) {
          $('.quick-diff .diff-ntitle .diff-comment').addClass('diff-hidden-history');
        }
        if (data.hasOwnProperty('error')) {
          console.warn('[InPageEdit] 快速差异获取时系统告知出现问题');
          $('.diffArea').html(msg('diff-error') + ': ' + data.error.info + '(<code>' + data.error.code + '</code>)');
        }
      }).fail(function (errorCode, feedback, errorThrown) {
        console.warn('[InPageEdit] 快速差异获取失败');
        $('.in-page-edit.quick-diff .ipe-progress').hide();
        $('.diffArea').show().html(msg('diff-error') + ': ' + errorThrown.error['info'] + '(<code>' + errorThrown.error['code'] + '</code>)');
      });
    }
    // 加载预设的快速最近更改模块
    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();
          _analysis('quick_diff_recentchanges');
          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 (config.wgAction === 'history') {
        $('.historysubmit.mw-history-compareselectedversions-button').after(
          $('<button>').text(msg('quick-diff')).click(function (e) {
            e.preventDefault();
            _analysis('quick_diff_history_page');
            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.quickEdit({page:mw.config.get(\'wgPageName\'),revision:' + oldid + '});">' + msg('quick-edit') + '</a>')
          );
        });
      }
    }

    /**
     * @module 获取段落编辑以及编辑链接
     * @param {Element} element parent element to find edit links
     */
    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 = config.wgArticlePath.replace('$1', '');
          if (splitStr === '/') {
            splitStr = config.wgServer.substring(config.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.quickEdit({
                    page: title,
                    revision: revision
                  });
                } else {
                  InPageEdit.quickEdit({
                    page: title,
                    section: section
                  });
                }
              })
            )
          );
        }
      });
    }

    /**
     * @module 快速预览文章页
     * @param params {Object} 
     */
    InPageEdit.quickPreview = function (params) {
      mw.hook('InPageEdit.quickPreview').fire();
      var timestamp = new Date().getTime();
      console.time('[InPageEdit] Request preview');
      ssi_modal.show({
        sizeClass: 'large',
        className: 'in-page-edit previewbox',
        title: msg('preview-title'),
        content: $('<section>').append(
          $progress,
          $('<div>', { id: 'InPageEditPreview', 'data-timestamp': timestamp, style: 'display:none', text: msg('preview-placeholder') })
        ),
        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'));
          });
        }
      });
    }

    /**
     * @module 载入中模块
     * @param title
     *  - √Boolean× true: Mark top progress box as done; false: Close top progress box
     *  - "String" Show new progress box with title @default 'Loading...'
     */
    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%' }).append(
              $('<div>', { class: 'ipe-progress-bar' })
            ),
          className: 'in-page-edit loadingbox',
          center: true,
          sizeClass: 'dialog',
          closeIcon: false,
          outSideClose: false
        });
      }
    }

    /**
     * @module 提交统计信息模块
     * @description Internal module
     */
    const _analysis = function (functionID) {
      if (InPageEdit.doNotCollectMyInfo === true) {
        // console.info('[InPageEdit] 我们已不再收集您使用插件的信息。');
        // return;
      }
      var functionID = functionID;
      var submitdata = {
        'action': 'submit',
        'url': config.wgServer + config.wgArticlePath.replace('$1', ''),
        'sitename': config.wgSiteName,
        'username': config.wgUserName,
        'function': functionID
      }
      $.ajax({
        url: InPageEdit.api.analysis,
        data: submitdata,
        type: 'get',
        dataType: 'json'
      }).done(function (data) {
        console.log('[InPageEdit] Analysis response\nStatus: ' + data.status + '\nMessage: ' + data.msg);
      });
    }

    /**
     * @module 版本信息模块
     * @description Show Update Logs Modal box
     */
    InPageEdit.versionInfo = function () {
      // 显示模态框
      ssi_modal.show({
        className: 'in-page-edit update-logs-modal',
        title: msg('updatelog-title') + ' - <span id="yourVersion">' + InPageEdit.version + '</span>',
        content: $('<section>').append(
          $('<iframe>', { style: 'margin: 0;padding: 0;width: 100%;height: 80vh;border: 0;', src: 'https://dragon-fish.github.io/inpageedit-v2/update-logs/?iframe=1' })
        ),
        buttons: [{
          label: 'GitHub',
          className: 'btn btn-secondary',
          method: function () {
            window.open('https://github.com/Dragon-Fish/InPageEdit-v2');
          }
        }, {
          label: msg('updatelog-about'),
          className: 'btn btn-secondary',
          method: function () {
            window.open('https://dragon-fish.github.io/inpageedit-v2/');
          }
        }, {
          label: msg('close'),
          className: 'btn btn-primary',
          method: function (a, modal) {
            modal.close();
          }
        }]
      });
    };

    /**
     * @module 关于插件模块
     * @description Show "What is" modal box of IPE2
     */
    InPageEdit.about = function () {
      ssi_modal.show({
        title: '关于InPageEdit',
        className: 'in-page-edit in-page-edit-about',
        // sizeClass: 'dialog',
        content: $('<section>').append(
          $('<iframe>', { style: 'margin: 0;padding: 0;width: 100%;height: 80vh;border: 0;', src: 'https://dragon-fish.github.io/inpageedit-v2/about/?iframe=1' })
        )
      });
    }

    /**
     * @module 获取版本更新提示
     */
    $(window).load(function () {
      var version = InPageEdit.version;
      // 版本更新
      if (localStorage.InPageEditVersion === null || localStorage.InPageEditVersion !== version) {
        if (typeof (ssi_modal) === undefined) return;
        ssi_modal.notify('', {
          title: msg('updatelog-update-success-title'),
          content: msg('updatelog-update-success').replace('$1', version),
          className: 'in-page-edit',
          buttons: [{
            className: 'btn btn-primary',
            label: msg('updatelog-button-versioninfo'),
            method: function (a, modal) {
              localStorage.InPageEditVersion = version;
              InPageEdit.versionInfo();
              modal.close();
            }
          }],
          closeAfter: {
            time: 30,
            resetOnHover: true
          },
          onClose: function () {
            ssi_modal.notify('', {
              className: 'in-page-edit',
              content: msg('updatelog-after-close').replace('$1', '<a href="' + msg('updatelog-url') + '" to="_blank">' + msg('updatelog-url') + '</a>').replace('$2', '<a href="https://github.com/Dragon-Fish/InPageEdit-v2">' + msg('updatelog-file-issue') + '</a>'),
              closeAfter: {
                time: 10
              }
            });
            localStorage.InPageEditVersion = version;
          }
        });
      }
      // 特殊提示
      if (localStorage.InPageEditNoticeId !== InPageEdit.specialNotice.id) {
        ssi_modal.notify('dialog', {
          className: 'in-page-edit ipe-special-notice',
          title: InPageEdit.specialNotice.title,
          content: InPageEdit.specialNotice.content,
          okBtn: {
            label: msg('updatelog-dismiss'),
            className: 'btn btn-primary'
          }
        }, function (e, modal) {
          localStorage.InPageEditNoticeId = InPageEdit.specialNotice.id;
          modal.close();
        });
      }
    });

    // 特别通知
    if (InPageEdit.isCanary) {
      InPageEdit.specialNotice = {
        id: msg('noticeid-canary'),
        title: msg('version-notice-canary-title'),
        content: msg('version-notice-canary')
      };
    } else {
      InPageEdit.specialNotice = {
        id: msg('noticeid'),
        title: msg('version-notice-title'),
        content: msg('version-notice')
      };
    }

    /**
     * @description 获取用户权限信息
     */
    !(function () {
      mw.user.getRights().then(function (rights) {
        console.info('[InPageEdit] 成功获取用户权限信息');
        mw.config.set('wgUserRights', rights);
      }).fail(function () {
        console.warn('[InPageEdit] 警告:无法获取用户权限信息');
        mw.config.set('wgUserRights', '');
      });
      if (mw.user.getName() !== null) {
        new mw.Api().get({
          action: 'query',
          list: 'users',
          usprop: 'blockinfo',
          ususers: mw.user.getName()
        }).then(function (data) {
          if (data.query.users[0].hasOwnProperty('blockid')) {
            mw.config.set('wgUserIsBlocked', true);
          } else {
            mw.config.set('wgUserIsBlocked', false);
          }
        });
      }
    }());

    /** 
     * @module 是否拥有权限
     * @param {String} right
     * @return {Boolean}
     */
    const _hasRight = function (right) {
      if (config.wgUserIsBlocked === true) {
        return false;
      }
      if (mw.config.get('wgUserRights').indexOf(right) > -1) {
        return true;
      } else {
        return false;
      }
    };

    /**
     * @description 页面载入完成,自动加载某些模块
     */
    !(function () {
      // 初始化设定
      _preference.set();
      // 快速页面差异模块
      InPageEdit.loadQuickDiff();
      // 加载段落编辑模块
      InPageEdit.articleLink();
    }());

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

      /** IPE工具盒 **/
      $('<div>', { id: 'ipe-edit-toolbox' }).append(
        $('<ul>', { class: 'btn-group group1' }).append(
          $('<li>', { class: 'btn-tip-group' }).append(
            $('<div>', { class: 'btn-tip', text: msg('quick-edit') }),
            $('<button>', { id: 'edit-btn', class: 'ipe-toolbox-btn', html: '<i class="fa fa-pencil"></i>' }).click(function () {
              InPageEdit.quickEdit({
                page: config.wgPageName,
                revision: config.wgRevisionId
              });
            })
          ),
          $('<li>', { class: 'btn-tip-group' }).append(
            $('<div>', { class: 'btn-tip', text: msg('redirect-from') }),
            $('<button>', { id: 'redirectfrom-btn', class: 'ipe-toolbox-btn', html: '<i class="fa fa-chevron-down"></i>' }).click(function () {
              InPageEdit.quickRedirect('from');
            })
          ),
          $('<li>', { class: 'btn-tip-group' }).append(
            $('<div>', { class: 'btn-tip', text: msg('redirect-to') }),
            $('<button>', { id: 'redirectto-btn', class: 'ipe-toolbox-btn', html: '<i class="fa fa-chevron-up"></i>' }).click(function () {
              InPageEdit.quickRedirect('to');
            })
          )
        ),
        $('<ul>', { class: 'btn-group group2' }).append(
          $('<div>', { style: 'display: flex;' }).append(
            $('<li>', { class: 'btn-tip-group' }).append(
              $('<div>', { class: 'btn-tip', text: msg('quick-delete') }),
              $('<button>', { id: 'deletepage-btn', class: 'ipe-toolbox-btn', html: '<i class="fa fa-trash"></i>' }).click(function () {
                InPageEdit.quickDelete();
              })
            ),
            $('<li>', { class: 'btn-tip-group' }).append(
              $('<div>', { class: 'btn-tip', text: msg('quick-rename') }),
              $('<button>', { id: 'renamepage-btn', class: 'ipe-toolbox-btn', html: '<i class="fa fa-italic"></i>' }).click(function () {
                InPageEdit.quickRename();
              })
            ),
            $('<li>', { class: 'btn-tip-group' }).append(
              $('<div>', { class: 'btn-tip', text: msg('ipe-preference') }),
              $('<button>', { id: 'preference-btn', class: 'ipe-toolbox-btn', html: '<i class="fa fa-gear"></i>' }).click(function () {
                InPageEdit.preference.modal();
              })
            )
          )
        ),
        $('<button>', { class: 'ipe-toolbox-btn', id: 'toolbox-toggle', html: '<i class="fa fa-plus"></i>' })
      ).appendTo('body');

      $('#ipe-edit-toolbox .btn-group button').click(function () {
        _analysis('tool_box')
      });

      // 设置开关等
      var toolBoxInner = $('#ipe-edit-toolbox #toolbox-toggle, #ipe-edit-toolbox .btn-group');
      $('#ipe-edit-toolbox #toolbox-toggle').click(function () {
        if ($(this).hasClass('opened') && !$(this).hasClass('click')) {
          toolBoxInner.addClass('click');
        } else if ($(this).hasClass('click')) {
          toolBoxInner.removeClass('click opened');
        } else {
          toolBoxInner.addClass('click opened');
        }
      });
      $('#ipe-edit-toolbox').hover(function () {
        toolBoxInner.addClass('hover opened');
      }, function () {
        toolBoxInner.removeClass('hover');
        if (!$('#ipe-edit-toolbox #toolbox-toggle').hasClass('click')) {
          toolBoxInner.removeClass('opened');
        }
      });
      mw.hook('InPageEdit.toolbox').fire();
    });

    // Init End
  }

  // 花里胡哨的加载提示
  mw.hook('InPageEdit').fire();
  console.info('    ____      ____                   ______    ___ __              _    _____ \n   /  _/___  / __ \\____ _____ ____  / ____/___/ (_) /_            | |  / /__ \\\n   / // __ \\/ /_/ / __ `/ __ `/ _ \\/ __/ / __  / / __/  ______    | | / /__/ /\n _/ // / / / ____/ /_/ / /_/ /  __/ /___/ /_/ / / /_   /_____/    | |/ // __/ \n/___/_/ /_/_/    \\__,_/\\__, /\\___/_____/\\__,_/_/\\__/              |___//____/ \n                      /____/');
}());