跳转到内容

MediaWiki:Gadget-charinsert-core.js

勤求古训,博采众方
鹿野耕云留言 | 贡献2025年10月27日 (一) 15:13的版本
(差异) ←上一版本 | 最后版本 (差异) | 下一版本→ (差异)

注意:在发布之后,您可能需要清除浏览器缓存才能看到所作出的更改的影响。

  • Firefox或Safari:按住Shift的同时单击刷新,或按Ctrl-F5Ctrl-R(Mac为⌘-R
  • Google Chrome:Ctrl-Shift-R(Mac为⌘-Shift-R
  • Edge:按住Ctrl的同时单击刷新,或按Ctrl-F5
/** 
 * <https://en.wikipedia.org/wiki/MediaWiki:Gadget-charinsert-core.js> 
 * 
 * 最初基于 [[mw:User:Alex Smotrov/edittools.js]] 开发,为英文维基百科使用进行了修改。 
 * 配置(需在 [[Special:MyPage/common.js]] 中设置): 
 * window.charinsertCustom – 对象类型。会合并到默认的字符插入列表中。例如,将其设置为 
 * { Symbols: '‽' },会在“符号”(Symbols)分类的末尾添加疑问惊叹号。 
 * window.editToolsRecall – 布尔类型。设为 true 可创建“恢复”(recall)开关。 
 * window.charinsertDontMove – 布尔类型。设为 true 可让字符插入框保持默认位置,不 
 * 移动到编辑摘要上方。 
 * window.updateEditTools() – 函数。更新 window.charinsertCustom 后调用此函数,可重新生成 
 * 编辑工具(EditTools)窗口。 
 */ 
/* 全局变量声明:$, mw, charinsertCustom */ 
window.updateEditTools = function () { }; 

$( function () { 	
	var $currentFocused, 		// 存储当前获取焦点的输入元素(文本框/编辑器)
		editTools; 				// 编辑工具核心对象,封装所有功能和配置

	/**
	 * 获取已选中的字符分类
	 * 优先从本地存储读取,本地存储无数据时从会话存储读取
	 * @returns {string|undefined} 存储的分类索引,无存储数据则返回 undefined
	 */
	function getSelectedSection() { 		
		var selectedSection = mw.storage.get( editTools.storageKey ) 
			|| mw.storage.session.get( editTools.storageKey ); 		
		return selectedSection; 
	} 

	/**
	 * 保存当前选中的字符分类索引
	 * 优先存入本地存储,本地存储失败时存入会话存储
	 * @param {string} newIndex - 新的分类索引(对应下拉选择框的选项序号)
	 */
	function saveSelectedSection( newIndex ) { 		
		mw.storage.set( editTools.storageKey, newIndex ) 
			|| mw.storage.session.set( editTools.storageKey, newIndex ); 
	} 

    editTools = {
        // 以 ␥(U+2425,删除格式符号二)开头的条目,不会在主命名空间(命名空间 0)显示。
        // 如需修改,请同时编辑 [[MediaWiki:Edittools]];但在该页面中,无需使用 ␥ 符号,改用 {{#ifeq:{{NAMESPACE}}|{{ns:0}}| | }} 控制主命名空间显示。
        charinsert: {
            '插入': '– — ′ ″ ‘+’ “+” 《+》 · § [\[+]] {\{+}} <sub>+</sub> 在讨论页签名: ~~\~~  引用来源: <ref>+</ref>',
            '维基标签': '<ref>+</ref> ~~\~~ {\{+}} {\{\{+}}} | [+] [\[+]] [\[Category:+]] #REDIRECT.[\[+]] &nbsp; <u>+</u> <s>+</s> <sup>+</sup> <sub>+</sub> <code>+</code> <pre>+</pre> <blockquote>+</blockquote> <ref.name="+"_/> {\{Reflist}} <references./> <includeonly>+</includeonly> <onlyinclude>+</onlyinclude> <noinclude>+</noinclude> <nowiki>+</nowiki> <code><nowiki>+</nowiki></code> <!--.+_--> [\[:File:+]] <syntaxhighlight.lang="php">+</syntaxhighlight>',
            '常用模板': ' {{subst:+}} {{SERVER}} {{localurl:+}} {{fullurl:+}} {{FULLPAGENAME}} {{PAGENAME}} {{DEFAULTSORT:+}} {{#expr:+}} {{#if:+}} {{#ifeq:+}} {{#ifexpr:+}} {{#switch:+}} {{#ifexist:+}} {{#time:Y年Fj日&#32;H:i|+}}',
            '常用符号': '“+” ‘+’ 「+」 『+』 (+) «+» ‹+› „+“ ‚+‘ - —— …… 《+》 〈+〉 【+】 〖+〗 〔+〕 ~ | ¡¿†‡↔→↑←↓↖↗↘↙•▪¶#∞ ‹+› «+» − °℃℉‰ ¤₳฿₵¢₡₢$₫₯€₠₣ƒ₴₭₤ℳ₥₦№₧₰£៛₨₪৳₮₩¥¥ ♠♣♥♦ 𝄫♭♮♯𝄪 ©®™ ◌ ☉☾☿♀🜨♂♃♄⛢♆',
            '拼音符号': '拼音: Á á À à Ǎ ǎ Ā ā É é È è Ě ě Ē ē Í í Ì ì Ǐ ǐ Ī ī Ó ó Ò ò Ǒ ǒ Ō ō Ú ú Ù ù Ü ü Ǔ ǔ Ū ū Ǘ ǘ Ǜ ǜ Ǚ ǚ Ǖ ǖ ê ê+̄ ế ê+̌ ề  注音: ㄅ ㄆ ㄇ ㄈ ㄉ ㄊ ㄋ ㄌ ㄍ ㄎ ㄏ ㄐ ㄑ ㄒ ㄓ ㄔ ㄕ ㄖ ㄗ ㄘ ㄙ ㄧ ㄨ ㄩ ㄚ ㄛ ㄜ ㄝ ㄞ ㄟ ㄠ ㄡ ㄢ ㄣ ㄤ ㄥ ㄦ ˉ ˊ ˇ ˋ ˙',
            '常用序号': '⓪ ① ② ③ ④ ⑤ ⑥ ⑦ ⑧ ⑨ ⑩ ⑪ ⑫ ⑬ ⑭ ⑮ ⑯ ⑰ ⑱ ⑲ ⑳ ㉑ ㉒ ㉓ ㉔ ㉕ ㉖ ㉗ ㉘ ㉙ ㉚ ㉛ ㉜ ㉝ ㉞ ㉟ ㊱ ㊲ ㊳ ㊴ ㊵ ㊶ ㊷ ㊸ ㊹ ㊺ ㊻ ㊼ ㊽ ㊾ ㊿ ㊀ ㊁ ㊂ ㊃ ㊄ ㊅ ㊆ ㊇ ㊈ ㊉ ㈠ ㈡ ㈢ ㈣ ㈤ ㈥ ㈦ ㈧ ㈨ ㈩ ⒈ ⒉ ⒊ ⒋ ⒌ ⒍ ⒎ ⒏ ⒐ ⒑ ⒒ ⒓ ⒔ ⒕ ⒖ ⒗ ⒘ ⒙ ⒚ ⒛ ⑴ ⑵ ⑶ ⑷ ⑸ ⑹ ⑺ ⑻ ⑼ ⑽ ⑾ ⑿ ⒀ ⒁ ⒂ ⒃ ⒄ ⒅ ⒆ ⒇ ❶ ❷ ❸ ❹ ❺ ❻ ❼ ❽ ❾ ❿ ⓫ ⓬ ⓭ ⓮ ⓯ ⓰ ⓱ ⓲ ⓳ ⓴ ⒜ ⒝ ⒞ ⒟ ⒠ ⒡ ⒢ ⒣ ⒤ ⒥ ⒦ ⒧ ⒨ ⒩ ⒪ ⒫ ⒬ ⒭ ⒮ ⒯ ⒰ ⒱ ⒲ ⒳ ⒴ ⒵ Ⓐ Ⓑ Ⓒ Ⓓ Ⓔ Ⓕ Ⓖ Ⓗ Ⓘ Ⓙ Ⓚ Ⓛ Ⓜ Ⓝ Ⓞ Ⓟ Ⓠ Ⓡ Ⓢ Ⓣ Ⓤ Ⓥ Ⓦ Ⓧ Ⓨ Ⓩ ⓐ ⓑ ⓒ ⓓ ⓔ ⓕ ⓖ ⓗ ⓘ ⓙ ⓚ ⓛ ⓜ ⓝ ⓞ ⓟ ⓠ ⓡ ⓢ ⓣ ⓤ ⓥ ⓦ ⓧ ⓨ ⓩ Ⅰ Ⅱ Ⅲ Ⅳ Ⅴ Ⅵ Ⅶ Ⅷ Ⅸ Ⅹ Ⅺ Ⅻ Ⅼ Ⅽ Ⅾ Ⅿ 〇 ⅰ ⅱ ⅲ ⅳ ⅴ ⅵ ⅶ ⅷ ⅸ ⅹ ⅺ ⅻ ⅼ ⅽ ⅾ ⅿ 〡 〢 〣 〤 〥 〦 〧 〨 〩 ㋀ ㋁ ㋂ ㋃ ㋄ ㋅ ㋆ ㋇ ㋈ ㋉ ㋊ ㋋ ㏠ ㏡ ㏢ ㏣ ㏤ ㏥ ㏦ ㏧ ㏨ ㏩ ㏪ ㏫ ㏬ ㏭ ㏮ ㏯ ㏰ ㏱ ㏲ ㏳ ㏴ ㏵ ㏶ ㏷ ㏸ ㏹ ㏺ ㏻ ㏼ ㏽ ㏾ ㍘ ㍙ ㍚ ㍛ ㍜ ㍝ ㍞ ㍟ ㍠ ㍡ ㍢ ㍣ ㍤ ㍥ ㍦ ㍧ ㍨ ㍩ ㍪ ㍫ ㍬ ㍭ ㍮ ㍯ ㍰ ㈰ ㈪ ㈫ ㈬ ㈭ ㈮ ㈯ ㊤ ㊥ ㊦ ㊧ ㊨',
            '数学与逻辑': '+ − × ÷ ⋅ … ¼ ½ ¾ ¹ ² ³ ° ′ ″ ∗ ∘ ± ∓ ≤ ≥ < > ≮ ≯ ≠ ≡ ∥ ∽ ≅ ≜ ≝ ≐ ≃ ≈ ⊕ ⊗ ⇐ ⇔ ⇒ ∞ ← ↔ → ≪ ≫ ∝ √ ∤ ≀ ◅ ▻ ⋉ ⋊ ⋈ ∴ ∵ ∷ ↦ ¬ ∧ ∨ ⊻ ∀ ∃ ∈ ∉ ∋ ⊆ ⊈ ⊊ ⊂ ⊄ ⊇ ⊉ ⊋ ⊃ ⊅ ⌒ ⊙ ∪ ∩ ∑ ∏ ∐ ∫ ∬ ∭ ∮ ∇ ∂ ∆ ∅ ℂ ℍ ℕ ℙ ℚ ℝ ℤ ℵ ‖ ⌊ ⌋ ⌈ ⌉ ⊤ ⊥ ⊢ ⊣ ⊧ □ ∠ ⟨ ⟩ <math>+</math> {\{math|+}} {\{mvar|+}} {\{frac|+|}} {\{sfrac|+|}}',
            '日语': '平假名: あ い う え お か き く け こ さ し す せ そ た ち つ て と な に ぬ ね の は ひ ふ へ ほ ま み む め も や ゆ よ ら り る れ ろ わ ゐ ゑ を ん 片假名: ア イ ウ エ オ カ キ ク ケ コ サ シ ス セ ソ タ チ ツ テ ト ナ ニ ヌ ネ ノ ハ ヒ フ ヘ ホ マ ミ ム メ モ ヤ ユ ヨ ラ リ ル レ ロ ワ ヰ ヱ ヲ ン 日语罗马字: Ā ā Ē ē Ī ī Ō ō Ū ū',
            '拉丁语': 'A a Á á À à  â Ä ä Ǎ ǎ Ă ă Ā ā à ã Å å Ą ą Æ æ Ǣ ǣ  B b  C c Ć ć Ċ ċ Ĉ ĉ Č č Ç ç  D d Ď ď Đ đ Ḍ ḍ Ð ð  E e É é È è Ė ė Ê ê Ë ë Ě ě Ĕ ĕ Ē ē Ẽ ẽ Ę ę Ẹ ẹ Ɛ ɛ Ǝ ǝ Ə ə  F f  G g Ġ ġ Ĝ ĝ Ğ ğ Ģ ģ  H h Ĥ ĥ Ħ ħ Ḥ ḥ  I i İ ı Í í Ì ì Î î Ï ï Ǐ ǐ Ĭ ĭ Ī ī Ĩ ĩ Į į Ị ị  J j Ĵ ĵ  K k Ķ ķ  L l Ĺ ĺ Ŀ ŀ Ľ ľ Ļ ļ Ł ł Ḷ ḷ Ḹ ḹ  M m Ṃ ṃ  N n Ń ń Ň ň Ñ ñ Ņ ņ Ṇ ṇ Ŋ ŋ  O o Ó ó Ò ò Ô ô Ö ö Ǒ ǒ Ŏ ŏ Ō ō Õ õ Ǫ ǫ Ọ ọ Ő ő Ø ø Œ œ  Ɔ ɔ  P p  Q q  R r Ŕ ŕ Ř ř Ŗ ŗ Ṛ ṛ Ṝ ṝ  S s Ś ś Ŝ ŝ Š š Ş ş Ș ș Ṣ ṣ ß  T t Ť ť Ţ ţ Ț ț Ṭ ṭ Þ þ  U u Ú ú Ù ù Û û Ü ü Ǔ ǔ Ŭ ŭ Ū ū Ũ ũ Ů ů Ų ų Ụ ụ Ű ű Ǘ ǘ Ǜ ǜ Ǚ ǚ Ǖ ǖ  V v  W w Ŵ ŵ  X x  Y y Ý ý Ŷ ŷ Ÿ ÿ Ỹ ỹ Ȳ ȳ  Z z Ź ź Ż ż Ž ž  ß Ð ð Þ þ Ŋ ŋ Ə ə Ɂ ɂ  Ꞌ ꞌ  ʻ  ʼ  ʽ  ꞉ ꞏ',	
            '希腊语': 'ΆάΈέΉήΊίΌόΎύΏώ  ΑαΒβΓγΔδ  ΕεΖζΗηΘθ  ΙιΚκΛλΜμ  ΝνΞξΟοΠπ  ΡρΣσςΤτΥυ  ΦφΧχΨψΩω Ϝϝυ̯ι̯  ᾼᾳᾴᾺὰᾲᾶᾷἈἀᾈᾀἉἁᾉᾁἌἄᾌᾄἊἂᾊᾂἎἆᾎᾆἍἅᾍᾅἋἃᾋᾃἏἇᾏᾇ  ῈὲἘἐἙἑἜἔἚἒἝἕἛἓ  ῌῃῄῊὴῂῆῇἨἠᾘᾐἩἡᾙᾑἬἤᾜᾔἪἢᾚᾒἮἦᾞᾖἭἥᾝᾕἫἣᾛᾓἯἧᾟᾗ  ῚὶῖἸἰἹἱἼἴἺἲἾἶἽἵἻἳἿἷΪϊΐῒῗ  ῸὸὈὀὉὁὌὄὊὂὍὅὋὃ  ῤῬῥ  ῪὺῦὐὙὑὔὒὖὝὕὛὓὟὗΫϋΰῢῧ  ῼῳῴῺὼῲῶῷὨὠᾨᾠὩὡᾩᾡὬὤᾬᾤὪὢᾪᾢὮὦᾮᾦὭὥᾭᾥὫὣᾫᾣὯὧᾯᾧ ᾹᾱᾸᾰῙῑῘῐῩῡῨῠ',
            '西里尔语': 'АаБбВвГг  ҐґЃѓДдЂђ  ЕеЁёЄєЖж  ЗзЅѕИиІі  ЇїЙйЈјКк  ЌќЛлЉљМм  НнЊњОоПп  РрСсТтЋћ  УуЎўФфХх  ЦцЧчЏџШш  ЩщЪъЫыЬь  ЭэЮюЯя ӘәӨөҒғҖҗ ҚқҜҝҢңҮү ҰұҲҳҸҹҺһ  ҔҕӢӣӮӯҘҙ  ҠҡҤҥҪҫӐӑ  ӒӓӔӕӖӗӰӱ  ӲӳӸӹӀ  ҞҟҦҧҨҩҬҭ  ҴҵҶҷҼҽҾҿ  ӁӂӃӄӇӈӋӌ  ӚӛӜӝӞӟӠӡ  ӤӥӦӧӪӫӴӵ  ́',
            '阿拉伯语': '  音译: ʾ  ā ī ū ṯ ḥ ḫ ẖ ḏ š ṣ ḍ ṭ ẓ ʿ ġ ẗ á ا ﺁ ب ت ث ج ح خ د ذ ر ز س ش ص ض ط ظ ع غ ف ق ك ل م ن ه ة و ي ى ء أ إ ؤ ئ',
            '国际音标 (英语)': 'ˈ ˌ  ŋ ɡ tʃ dʒ ʃ ʒ θ ð ʔ  ɑː ɒ æ aɪ aʊ ɛ ɛər+ eɪ ɪ ɪər+ iː ɔː ɔɪ oʊ ʊ ʊər+ uː ʌ ɜːr+  ə ər  ɒ̃ æ̃',
            '国际音标': '辅音: ɱɳɲŋɴ : t̪ d̪ ʈɖɟɡɢʡʔ : ɸβθð  ʃʒʂʐɕʑ  çʝɣχʁ  ħʕʜʢɦɧ : ʋɹɻɥɰʍ : ʙⱱɾɽʀ  ɺ  ɫɬɮɭʎʟ : ɓɗᶑʄɠʛ  ʘǀǃǂǁ  元音: ɪʏɨʉɯʊ : øɘɵɤ  ə ɚ  ɛœɜɝɞʌɔ : æɶɐɑɒ  间隔变音符号: ˈˌːˑʼˀˤᵝᵊᶢˠʰʱʲˡⁿᵑʷᶣ˞‿˕˔  组合变音符号: ̚ ̪ ̺ ̻ ̼ ̬  ̊ ̥ ̞ ̝ ̘ ̙ ̽ ̟ ̠  ̈ ̤ ̹ ̜ ̍ ̩  ̆ ̯  ̃ ̰ ͡ ͜  声调:  ̋  ́  ̄  ̀  ̏  ̌  ̂ ᷄ ᷅ ᷇ ᷆ ᷈ ᷉  ˥˦˧˨˩ꜛꜜ : ↗↘‖  扩展国际音标: ͈ ͉ ͎ ̣ ̫ ͊ ᷽ ͇ : ˭ᵻᵿ',
            '古代英语': 'Ā ā Æ æ Ǣ ǣ Ǽ ǽ Ċ ċ Ð ð Ē ē Ġ ġ Ī ī Ō ō Ū ū Ƿ ƿ Ȳ ȳ Þ þ Ȝ ȝ',
            '希伯来语': 'אבגדהוזחטיכךלמםנןסעפףצץקרשת  ׳ ״  װױײ', 
            '意第绪语': ' א אַ אָ ב בֿ ג ד ה ו וּ װ ױ ז זש ח ט י יִ ײ ײַ כ ך כּ ל ל+ מ ם נ ן ס ע ע+ פ פּ פֿ ף צ ץ ק ר ש שׂ תּ ת ׳ ״ ־',
            '天成文书(梵文)': 'ँ ं ः अ आ इ ई उ ऊ ऋ ऌ ऍ ऎ ए ऐ ऑ ऒ ओ औ क क़ ख ख़ ग ग़ घ ङ च छ ज ज़ झ ञ ट ठ ड ड़ द ढ ढ़ ण त थ ध न ऩ प फ फ़ ब भ म य य़ र ऱ ल ळ ऴ व श ष स ह ़ ऽ ा ि ॊ ो ौ ् ी ु ू ृ ॄ ॅ ॆ े ै ॉ ॐ ॑ ॒ ॓ ॔ ॠ ॡ ॢ ॣ । ॥ ॰',
            '德语': 'Ä ä Ö ö Ü ü ß ẞ',
            '法语': 'À à  â Ç ç É é È è Ê ê Ë ë Î î Ï ï Ô ô Œ œ Ù ù Û û Ü ü Ÿ ÿ',
            '冰岛语': 'Á á Ð ð É é Í í Ó ó Ú ú Ý ý Þ þ Æ æ Ö ö',
            '波兰语': 'ą Ą ć Ć ę Ę ł Ł ń Ń ó Ó ś Ś ź Ź ż Ż',
            '捷克语': 'Á á Č č Ď ď É é Ě ě Í í Ň ň Ó ó Ř ř Š š Ť ť Ú ú Ů ů Ý ý Ž ž',
            '世界语': 'Ĉ ĉ Ĝ ĝ Ĥ ĥ Ĵ ĵ Ŝ ŝ Ŭ ŭ',
            '越南语': 'À à Ả ả Á á Ạ ạ Ã ã Ă ă Ằ ằ Ẳ ẳ Ẵ ẵ Ắ ắ Ặ ặ Â â Ầ ầ Ẩ ẩ Ẫ ẫ Ấ ấ Ậ ậ Đ đ È è Ẻ ẻ Ẽ ẽ É é Ẹ ẹ Ê ê Ề ề Ể ể Ễ ễ Ế ế Ệ ệ Ỉ ỉ Ĩ ĩ Í í Ị ị Ì ì Ỏ ỏ Ó ó Ọ ọ Ò ò Õ õ Ô ô Ồ ồ Ổ ổ Ỗ ỗ Ố ố Ộ ộ Ơ ơ Ờ ờ Ở ở Ỡ ỡ Ớ ớ Ợ ợ Ù ù Ủ ủ Ũ ũ Ú ú Ụ ụ Ư ư Ừ ừ Ử ử Ữ ữ Ứ ứ Ự ự Ỳ ỳ Ỷ ỷ Ỹ ỹ Ỵ ỵ Ý ý',
            '立陶宛语': 'Ą ą Č č Ę ę Ė ė Į į Š š Ų ų Ū ū Ž ž',
            '马耳他语': 'Ċ ċ Ġ ġ Ħ ħ Ż ż',
            '葡萄牙语': 'Á á À à Â â Ã ã Ç ç É é Ê ê Í í Ó ó Ô ô Õ õ Ú ú Ü ü',
            '土耳其语': 'Ç ç Ğ ğ İ ı Ö ö Ş ş Ü ü Â â Î î Û û',
            '夏威夷语': 'Ā ā Ē ē Ī ī Ō ō Ū ū ʻ',
            '匈牙利语': 'Ő ő Ű ű',
            '西班牙语': 'Á á É é Í í Ñ ñ Ó ó Ú ú Ü ü ¡ ¿',
            '意大利语': 'Á á À à É é È è Í í Ì ì Ó ó Ò ò Ú ú Ù ù',
            '威尔士语': 'Á á À à Â â Ä ä É é È è Ê ê Ë ë Ì ì Î î Ï ï Ó ó Ò ò Ô ô Ö ö Ù ù Û û Ẁ ẁ Ŵ ŵ Ẅ ẅ Ý ý Ỳ ỳ Ŷ ŷ Ÿ ÿ',
            '爱沙尼亚语': 'Č č Š š Ž ž Õ õ Ä ä Ö ö Ü ü',
           
            '拉脱维亚语': 'Ā ā Č č Ē ē Ģ ģ Ī ī Ķ ķ Ļ ļ Ņ ņ Š š Ū ū Ž ž',
            '罗马尼亚语': 'Ă ă Â â Î î Ş ş Ţ ţ',
            '塞尔维亚语': 'А а Б б В в Г г Д д Ђ ђ Е е Ж ж З з И и Ј ј К к Л л Љ љ М м Н н Њ њ О о П п Р р С с Т т Ћ ћ У у Ф ф Х х Ц ц Ч ч Џ џ Ш ш',
            '加泰罗尼亚语': 'Á á À à Ç ç É é È è Ë ë Í í Ï ï Ó ó Ò ò Ö ö Ú ú Ù ù',
            '斯堪的纳维亚语': 'À à É é Å å Æ æ Ä ä Ø ø Ö ö'
        },
        
        // 字符分隔符:使用不间断空格(\240 对应 Unicode 中的 nbsp)
charinsertDivider: "\240",
// 用于本地存储的键名:存储字符子集的选择状态
storageKey: 'edittoolscharsubset',

/**
 * 创建编辑工具(特殊字符选择面板)
 * @param {HTMLElement} placeholder - 占位元素,将被工具面板替换
 */
createEditTools: function (placeholder) {
    var sel, id;
    // 创建工具面板的外层容器
    var box = document.createElement('div');
    // 记录上一个选中的字符子集索引和当前选中的索引
    var prevSubset = 0, curSubset = 0;

    // 设置工具面板的基础属性
    box.id = 'editpage-specialchars';
    box.className = "nopopups";
    box.title = '点击字符或标签,将其插入到编辑窗口中';

    // 追加用户自定义的字符集(如果存在)
    if (window.charinsertCustom) {
        for (id in charinsertCustom) {
            // 若编辑工具中没有该自定义字符集,则初始化为空字符串
            if (!editTools.charinsert[id]) {
                editTools.charinsert[id] = '';
            }
        }
    }

    // 创建下拉选择框(用于切换不同字符子集)
    sel = document.createElement('select');
    // 为每个字符子集添加下拉选项
    for (id in editTools.charinsert) {
        sel.options[sel.options.length] = new Option(id, id);
    }
    // 默认选中第一个选项
    sel.selectedIndex = 0;
    // 设置下拉框样式:右侧间距
    sel.style.marginRight = '.3em';
    sel.title = '选择字符子集';
    // 绑定下拉框的切换事件(change 和 keyup 事件均触发子集切换)
    sel.onchange = sel.onkeyup = selectSubset;
    // 将下拉框添加到工具面板
    box.appendChild(sel);

    // 创建“召回”开关(用于快速切换回上一个选中的子集,若开启该功能)
    if (window.editToolsRecall) {
        var recall = document.createElement('span');
        // 添加召回按钮的图标(上下箭头)
        recall.appendChild(document.createTextNode('↕')); // ↔ 为备选图标
        // 点击召回按钮时,切换回上一个选中的子集
        recall.onclick = function () {
            sel.selectedIndex = prevSubset;
            selectSubset();
        };
        // 设置召回按钮样式:左浮动、右侧间距、鼠标指针样式
        recall.style.cssFloat = 'left';
        recall.style.marginRight = '5px';
        recall.style.cursor = 'pointer';
        // 将召回按钮添加到工具面板
        box.appendChild(recall);
    }

    // 如果有已选中的字符子集(从存储中读取),则设置下拉框的选中状态
    if (getSelectedSection()) {
        sel.selectedIndex = getSelectedSection();
    }

    // 用工具面板替换占位元素
    placeholder.parentNode.replaceChild(box, placeholder);
    // 初始化时执行一次子集切换,显示默认选中的字符子集
    selectSubset();
    return;

    /**
     * 切换字符子集的核心函数
     * 功能:隐藏其他子集、显示/创建当前选中的子集、保存选中状态到本地存储
     */
    function selectSubset() {
        // 记录上一个选中的子集索引(用于召回功能)
        prevSubset = curSubset;
        // 更新当前选中的子集索引
        curSubset = sel.selectedIndex;
        // 将当前选中的子集索引保存到本地存储(实现状态持久化)
        saveSelectedSection(curSubset);

        // 隐藏所有已存在的字符子集面板(p 标签)
        var pp = box.getElementsByTagName('p');
        for (var i = 0; i < pp.length; i++) {
            pp[i].style.display = 'none';
        }

        // 获取当前选中的子集 ID
        var id = sel.options[curSubset].value;
        // 查找当前子集对应的面板(p 标签)
        var p = document.getElementById(id);

        // 如果面板不存在,则创建新面板
        if (!p) {
            p = document.createElement('p');
            p.className = 'nowraplinks'; // 防止链接换行的样式类
            p.id = id;

            // 针对阿拉伯语和希伯来语子集:设置特殊样式(字体大小、文字方向从右到左)
            if (id == '阿拉伯语' || id == '希伯来语') {
                p.style.fontSize = '120%';
                p.dir = 'rtl';
            }

            // 获取当前子集的字符/标签数据
            var tokens = editTools.charinsert[id];
            // 如果存在该子集的自定义数据,则追加到原有数据中
            if (window.charinsertCustom && charinsertCustom[id]) {
                if (tokens.length > 0) {
                    tokens += ' ';
                }
                tokens += charinsertCustom[id];
            }

            // 创建当前子集的字符/标签元素(调用 createTokens 方法)
            editTools.createTokens(p, tokens);
            // 将新面板添加到工具面板
            box.appendChild(p);
        }

        // 显示当前选中的子集面板
        p.style.display = 'inline';
    }
},

/**
 * 创建字符/标签元素(将字符子集数据解析为可点击的页面元素)
 * @param {HTMLElement} paragraph - 承载字符/标签的容器(p 标签)
 * @param {string} str - 字符子集的原始数据(空格分隔的 token 字符串)
 */
createTokens: function (paragraph, str) {
    // 将原始数据按空格分割为单个 token
    var tokens = str.split(' '), token, i, n;

    // 遍历每个 token,解析并创建对应的页面元素
    for (i = 0; i < tokens.length; i++) {
        token = tokens[i];
        // 查找 token 中的 '+' 符号(用于识别标签对,如 <tag>+</tag>)
        n = token.indexOf('+');

        // 处理特殊前缀 '␥'(仅在主命名空间下生效,用于过滤特定 token)
        if (token.charAt(0) === '␥') {
            // 如果 token 长度大于 1 且当前是主命名空间(wgNamespaceNumber === 0),则跳过该 token
            if (token.length > 1 && mw.config.get('wgNamespaceNumber') === 0) {
                continue;
            } else {
                // 否则,移除前缀 '␥',保留剩余部分作为 token
                token = token.substring(1);
            }
        }

        // 根据 token 的不同类型,创建对应的元素
        if (token === '' || token === '_') {
            // 空 token 或 '_':添加字符分隔符(配合空格)
            addText(editTools.charinsertDivider + ' ');
        } else if (token === '\n') {
            // '\n':添加换行标签(br)
            paragraph.appendChild(document.createElement('br'));
        } else if (token === '___') {
            // '___':添加水平分隔线(hr)
            paragraph.appendChild(document.createElement('hr'));
        } else if (token.charAt(token.length - 1) === ':') {
            // 以 ':' 结尾的 token:创建粗体文本(仅用于显示,不可点击插入)
            addBold(token);
        } else if (n === 0) {
            // 以 '+' 开头的 token:解析为 <tag>+</tag> 格式(如 +b 对应 <b>+</b>)
            addLink(token.substring(1), '</' + token.substring(2), token.substring(1));
        } else if (n > 0) {
            // 包含 '+' 且不在开头的 token:解析为标签对(如 b+ 对应 <b></b>)
            addLink(token.substring(0, n), token.substring(n + 1));
        } else {
            // 普通 token:判断是单个字符还是多个字符的组合
            var chars = Array.from(token);
            // 如果是长度大于 2 的非 ASCII 字符(编码 > 127),则拆分为单个字符分别创建元素
            if (chars.length > 2 && token.charCodeAt(0) > 127) {
                for (var j = 0; j < chars.length; j++) {
                    addLink(chars[j], '');
                }
            } else {
                // 其他情况:直接将 token 作为可插入元素
                addLink(token, '');
            }
        }
    }
    return;

    /**
     * 创建可点击的链接元素(用于插入字符或标签)
     * @param {string} tagOpen - 起始标签/字符(如 '<b>' 或 'α')
     * @param {string} tagClose - 结束标签(如 '</b>',字符类型为空)
     * @param {string} name - 链接显示的文本(默认使用 tagOpen + tagClose)
     */
    function addLink(tagOpen, tagClose, name) {
        var handler;
        // 查找 tagOpen 中的特殊字符 '\x10'(用于识别自定义点击事件)
        var dle = tagOpen.indexOf('\x10');
        var a = document.createElement('a');

        // 如果存在 '\x10',则解析为自定义事件(格式:前缀\x10对象.方法)
        if (dle > 0) {
            var path = tagOpen.substring(dle + 1).split('.');
            tagOpen = tagOpen.substring(0, dle);
            // 从 window 中获取对应的方法(如 path 为 ['foo','bar'] 则对应 window.foo.bar)
            handler = window;
            for (var i = 0; i < path.length; i++) {
                handler = handler[path[i]];
            }
            // 为链接绑定自定义点击事件
            $(a).on('click', handler);
        } else {
            // 普通情况:处理标签中的特殊字符(点号替换为空格,下划线替换为空格)
            tagOpen = tagOpen.replace(/\./g, ' ');
            tagClose = tagClose ? tagClose.replace(/_/g, ' ') : '';
            // 为链接绑定默认的插入标签事件(传递标签前后缀数据)
            $(a).on('click', {
                tagOpen: tagOpen,
                sampleText: '',
                tagClose: tagClose
            }, insertTags);
        }

        // 设置链接显示的文本(默认使用标签组合,移除换行符)
        name = name || tagOpen + tagClose;
        name = name.replace(/\\n/g, '');
        a.appendChild(document.createTextNode(name));
        // 清除链接的默认跳转行为(href 设为空)
        a.href = '';
        // 将链接添加到面板容器
        paragraph.appendChild(a);
        // 在链接后添加空格(用于分隔元素)
        addText(' ');
    }

    /**
     * 创建粗体文本元素(用于显示分类标题,不可点击)
     * @param {string} text - 粗体文本内容(将下划线替换为空格)
     */
    function addBold(text) {
        var b = document.createElement('b');
        b.appendChild(document.createTextNode(text.replace(/_/g, ' ')));
        paragraph.appendChild(b);
        // 在粗体文本后添加空格(用于分隔元素)
        addText(' ');
    }

    /**
     * 向容器中添加普通文本(用于分隔符或空格)
     * @param {string} txt - 要添加的文本内容
     */
    function addText(txt) {
        paragraph.appendChild(document.createTextNode(txt));
    }

    /**
     * 插入标签/字符到编辑框的核心函数
     * @param {Event} e - 点击事件对象(包含标签前后缀数据)
     */
    function insertTags(e) {
        // 阻止链接的默认跳转行为
        e.preventDefault();

        // 检查当前是否有聚焦的可编辑元素(且非只读)
        if ($currentFocused && $currentFocused.length && !$currentFocused.prop('readonly')) {
            // 使用 textSelection 插件,将标签插入到编辑框(包裹选中内容或直接插入)
            $currentFocused.textSelection(
                'encapsulateSelection', {
                    pre: e.data.tagOpen,    // 插入到选中内容前的标签/字符
                    peri: e.data.sampleText,// 示例文本(此处为空)
                    post: e.data.tagClose   // 插入到选中内容后的标签
                }
            );
        }
    }
},

/**
 * 初始化编辑工具(入口函数)
 * 功能:创建工具面板容器、绑定焦点事件、调用 createEditTools 生成面板
 */
setup: function () {
    var placeholder;

    // 查找已存在的工具面板容器(若有则复用)
    if ($('#editpage-specialchars').length) {
        placeholder = $('#editpage-specialchars')[0];
    } else {
        // 若不存在,则创建新的占位容器,并添加到编辑工具区域(.mw-editTools)
        placeholder = $('<div id="editpage-specialchars"> </div>').prependTo('.mw-editTools')[0];
    }

    // 如果占位容器不存在,则终止初始化
    if (!placeholder) {
        return;
    }

    // 如果没有设置“不移动”标记,则将占位容器移动到编辑选项(.editOptions)之前
    if (!window.charinsertDontMove) {
        $('.editOptions').before(placeholder);
    }

    // 初始化当前聚焦的编辑框(默认是主编辑框 #wpTextbox1)
    $currentFocused = $('#wpTextbox1');

    // 为动态创建的编辑框(如 textarea、input:text、CodeMirror)绑定焦点事件
    $(document).on('focus', 'textarea, input:text, .CodeMirror', function () {
        if ($(this).is('.CodeMirror')) {
            // CodeMirror 编辑器:其文本操作依赖主编辑框 #wpTextbox1,故聚焦主编辑框
            $currentFocused = $('#wpTextbox1');
        } else {
            // 普通编辑框:直接将当前聚焦元素设为目标
            $currentFocused = $(this);
        }
    });

    // 调用 createEditTools 生成工具面板(传入占位容器)
    editTools.createEditTools(placeholder);

    // 定义全局更新函数(用于动态更新工具面板)
    window.updateEditTools = function () {
        editTools.createEditTools($('#editpage-specialchars')[0]);
    };
}
}; // 结束 editTools 对象定义

// 调用初始化函数,启动编辑工具
editTools.setup();
} );