МедияУики:Gadget-EditToolbar-core.js
Забележка: За да се видят промените, необходимо е след публикуване на страницата, кешът на браузъра да бъде изтрит.
- Firefox / Safari: Задържа се клавиш Shift и се щраква върху Презареждане (Reload) или чрез клавишната комбинация Ctrl-F5 or Ctrl-R (⌘-R за Mac);
- Google Chrome: клавишна комбинация Ctrl-Shift-R (⌘-Shift-R за Mac)
- Internet Explorer / Edge: Задържа се клавиш Ctrl и се щраква върху Refresh или чрез клавишната комбинация Ctrl-F5;
- Opera: Press Ctrl-F5.
/**
* Core of the EditToolbar
* a modified version of https://bg.wikipedia.org/wiki/МедияУики:Gadget-EditToolbar-core.js
*
* Some functions and arrays are still bound to window for backwards compatibility
*/
window.setCustomInsButton = function(code, left, middle, right, shownText, title) {
customInsButtons[code] = [left, middle, right, shownText, title];
};
window.setCustomMiscButton = function(code, codeToRun, shownText, title) {
customMiscButtons[code] = [codeToRun, shownText, title];
};
window.rmCustomInsButtons = function(rmButtons) {
rmCustomButtons(customInsButtons, rmButtons);
};
window.rmCustomMiscButtons = function(rmButtons) {
rmCustomButtons(customMiscButtons, rmButtons);
};
window.rmCustomButtons = function(allButtons, rmButtons) {
for (var i = rmButtons.length - 1; i >= 0; i--) {
delete( allButtons[ rmButtons[i] ] );
}
};
/**
insert an edit toolbar before the textarea.
useful for testing user script pages while previewing
use it with:
putToolbar();
put in your script page, e.g. User:Your_Name/monobook.js
*/
window.putToolbar = function(rightNow) {
var toolbar = $('<div>', {'class': 'custom-toolbar'});
var putIt = function() {
$("#editform").before(toolbar);
};
if ( window.rightNow ) {
putIt();
} else {
$(putIt);
}
return toolbar;
};
mw.libs.EditToolbar = {};
/* * * Extra buttons for text insertion * * */
// от тези данни ще се генерират допълнителни бутони с insertTags()
window.customInsButtons = {
// "CODE" : ["LEFT", "MIDDLE", "RIGHT", "SHOWN TEXT", "TITLE"],
"b2" : ["<code>", "моля, въведете програмен код", "</code>", "<tt>код</tt>", "Текст в равноширок шрифт — обикновено код"],
"b5" : ["\u00a0", "", "", "nbsp", "+несекаем интервал"],
"b6" : ["„", "текст в кавички", "“", "„“", "+български кавички"],
//"b7" : ["<del>", "зачертан текст", "</del>", "<del>del</del>", "Отбелязване на текст като изтрит"],
"b11" : ["–", "", "", " – ", "+средна чертица — ndash"],
//"b12" : ["mw.libs.EditToolbar.graveAccent()", "удар.", "+ударение за гласна буква или замяна на предходно „и“ или „й“ с „ѝ“"],
"b13" : ["<!-- ", "моля, въведете коментар", " -->", "<!--", "+коментар"],
"b21" : ["<ref>", "", "</ref>", "<ref>", "+източник / бележка под линия"],
"b21n" : ['<ref name="">', "", "</ref>", "<ref name>", "+многократна употреба на източник / бележка под линия"],
"b8" : ["{"+"{", "", "}}", "{{}}", "+скоби за шаблон"],
"b9" : ["|", "", "", " | ", "+отвесна черта — |"],
"b21o" : ["[", "", "]", "[ ]", "+външна препратка"],
"b22" : ["["+"[", "", "]]", "[[...]]", "+препратка без разделител"],
"b23l" : ["["+"[", "", "|]]", "[[...|]]", "+препратка с разделител (курсорът е отляво на разделителя)"],
"b23r" : ["["+"[|", "", "]]", "[[|...]]", "+препратка с разделител (курсорът е отдясно на разделителя)"],
"b24" : ["["+"[:", "", "]]", "[[:...]]", "+текстова препратка"],
"linkWithAddrAndLabel" : ["mw.libs.EditToolbar.linkWithAddrAndLabel()", "[[.|.]]",
"Копиране на маркирания текст и като адрес и като надпис в уикипрепратка"]
};
window.atpl1 = window.atpl1 || {
// "ПОКАЗВАН ТЕКСТ" : "ИМЕ НА СТРАНИЦАТА",
"Част на речта…" : "-",
"Български език" : {
"Съществ. нариц. име": "Шаблон:Съществително нарицателно име/празен",
"Прилагателно име": "Шаблон:Прилагателно име/празен",
"Местоимение": "Шаблон:Местоимение/празен",
"Числително име": "Шаблон:Числително име/празен",
"Глагол": "Шаблон:Глагол/празен",
"Наречие": "Шаблон:Наречие/празен",
"Съюз": "Шаблон:Съюз/празен",
"Междуметие": "Шаблон:Междуметие/празен",
"Частица": "Шаблон:Частица/празен",
"Предлог": "Шаблон:Предлог/празен",
"Съществ. собств. име": "Шаблон:Съществително собствено име/празен",
"Представка": "Шаблон:Представка/празен",
"Наставка": "Шаблон:Наставка/празен",
"Словоформа": "Шаблон:Форма/празен"
},
"Други езици": {
"Съществително име": "Шаблон:Noun/празен",
"Прилагателно име": "Шаблон:Adjective/празен",
"Глагол": "Шаблон:Verb/празен",
"Наречие": "Шаблон:Adverb/празен"
}
};
mw.messages.set({
"et-addchar": "Вмъкване на знака „$1“",
"et-addpref": "Вмъкване на $1",
"et-ddmenutitle": "Оттук можете да вмъкнете празен шаблон",
"et-ajaxerror": "Шаблонът [[$1]] не можа да бъде зареден."
});
var atplb = "МедияУики:common.js/Edit tools data/";
/*
window.atpl1 = window.atpl1 || {
// "SHOWN TEXT" : "PAGE NAME",
"Тест…" : "-",
"Съществително" : atplb + "Съществително",
"Глагол" : atplb + "Глагол"
};
*/
/* * * Extra buttons for miscelaneous functions * * */
// името на елемента за допълнителните знаци
var charsElemId = "extraChars";
// данни за още бутони с код по желание
window.customMiscButtons = {
// "CODE" : ["CODE TO RUN", "SHOWN TEXT", "TITLE"],
"mkBulletList" : ["mw.libs.EditToolbar.mkList(false)", "*..", "Добавяне на * в началото на всеки ред"],
"mkNumList" : ["mw.libs.EditToolbar.mkList(true)", "#..", "Добавяне на # в началото на всеки ред"],
// допълнителните знаци
"ch" : ["mw.libs.EditToolbar.toggleChars()", "Още…", "Виртуална клавиатура"]
};
/* * * Drop down menus for template insertion * * */
/* по идея на [[:he:MediaWiki:Summary|еврейската Уикипедия]] */
/*
window.tpl1 = window.tpl1 || {
// "SHOWN TEXT" : "TEMPLATE CONTENT",
"Елементи от статията…" : "-",
"Източници" : "\n== Източници ==\n<references />\n>>|<<",
"Бележки" : "\n== Бележки ==\n<references />\n>>|<<",
"Вижте също" : "\n== Вижте също ==\n* [[>>|<<]]\n",
};*/
var tplVarBaseName = "tpl";
var atplVarBaseName = "atpl";
var chars = [
['А', 'Б', 'В', 'Г', 'Д', 'Е', 'Ж', 'З', 'И', 'Й', 'К', 'Л', 'М', 'Н', 'О', 'П', 'Р', 'С',
'Т', 'У', 'Ф', 'Х', 'Ц', 'Ч', 'Ш', 'Щ', 'Ъ', 'Ь', 'Ю', 'Я', 'Ы', 'Э', 'а', 'б', 'в', 'г',
'д', 'е', 'ж', 'з', 'и', 'й', 'к', 'л', 'м', 'н', 'о', 'п', 'р', 'с', 'т', 'у', 'ф', 'х',
'ц', 'ч', 'ш', 'щ', 'ъ', 'ь', 'ю', 'я', 'ы', 'э'],
['ѣ', 'ѫ', 'ѭ', 'ѧ', 'ѩ', 'Ї', 'Ҁ', 'Ѹ', 'Ѡ', 'Ѻ', 'Ъ', 'І', 'Ҍ', 'Ѩ', 'Ѭ', 'Ѯ', 'Ѵ',
'Ѥ', 'Ѿ'],
['α', 'β', 'γ', 'δ', 'ε', 'ζ', 'η', 'θ', 'ι', 'κ', 'λ', 'μ', 'ν', 'ξ', 'ο', 'π', 'ρ', 'σ',
'ς', 'τ', 'υ', 'φ', 'χ', 'ψ', 'ω', 'Γ', 'Δ', 'Θ', 'Λ', 'Ξ', 'Π', 'Σ', 'Φ', 'Ψ', 'Ω'],
['½', '¼', '¾', '∫', '∑', '∏', '√', '—', '−', '±', '∞', '≈', '~', '∝', '≡', '≠', '≤', '≥', '×',
'·', '÷', '∂', '′', '″', '∇', '∮', '⊥', '‰', '∴', 'ℵ', 'ℋ', '℧', '^', '¹', '²', '³', '∈',
'∉', '∩', '∪', '⊂', '⊃', '⊆', '⊇', '∧', '∨', 'ø', '¬', '∃', '∀', '⇒', '⇐','⇓', '⇑',
'⇔', '→', '←', '↓', '↑', '↔', '⇄', '⇆', '⇋', '⇌', 'ℕ', 'ℤ', 'ℚ', 'ℝ', 'ℂ', '∅', '⋮',
'⋯'],
['*', '~', '|', '[', ']', '°', '№', '™', '©', '®', '¢', '€', '¥', '£', '¤', '¿', '¡',
'«', '»', '§', '¶', '†', '‡', '•', '♀', '♂', '…', '¨'],
['Á', 'á', 'É', 'é', 'Í', 'í', 'Ó', 'ó', 'Ú', 'ú', 'À', 'à', 'È', 'è', 'Ì', 'ì', 'Ò', 'ò',
'Ù', 'ù', 'Â', 'â', 'Ê', 'ê', 'Î', 'î', 'Ô', 'ô', 'Û', 'û', 'Ä', 'ä', 'Ë', 'ë', 'Ï', 'ï',
'Ö', 'ö', 'Ü', 'ü', 'ß', 'Ã', 'ã', 'Ñ', 'ñ', 'Õ', 'õ', 'Ç', 'ç', 'Ģ', 'ģ', 'Ķ', 'ķ', 'Ļ',
'ļ', 'Ņ', 'ņ', 'Ŗ', 'ŗ', 'Ş', 'ş', 'Ţ', 'ţ', 'Ć', 'ć', 'Ĺ', 'ĺ', 'Ń', 'ń', 'Ŕ', 'ŕ', 'Ś',
'ś', 'Ý', 'ý', 'Ź', 'ź', 'Đ', 'đ', 'Ů', 'ů', 'Č', 'č', 'Ď', 'ď', 'Ľ', 'ľ', 'Ň', 'ň', 'Ř',
'ř', 'Š', 'š', 'Ť', 'ť', 'Ž', 'ž', 'Ǎ', 'ǎ', 'Ě', 'ě', 'Ǐ', 'ǐ', 'Ǒ', 'ǒ', 'Ǔ', 'ǔ', 'Ā',
'ā', 'Ē', 'ē', 'Ī', 'ī', 'Ō', 'ō', 'Ū', 'ū', 'ǖ', 'ǘ', 'ǚ', 'ǜ', 'Ĉ', 'ĉ', 'Ĝ', 'ĝ', 'Ĥ',
'ĥ', 'Ĵ', 'ĵ', 'Ŝ', 'ŝ', 'Ŵ', 'ŵ', 'Ŷ', 'ŷ', 'Ă', 'ă', 'Ğ', 'ğ', 'Ŭ', 'ŭ', 'Ċ', 'ċ', 'Ė',
'ė', 'Ġ', 'ġ', 'İ', 'ı', 'Ż', 'ż', 'Ą', 'ą', 'Ę', 'ę', 'Į', 'į', 'Ų', 'ų', 'Ł', 'ł', 'Ő',
'ő', 'Ű', 'ű', 'Ŀ', 'ŀ', 'Ħ', 'ħ', 'Ð', 'ð', 'Þ', 'þ', 'Œ', 'œ', 'Æ', 'æ', 'Ø', 'ø', 'Å',
'å']
];
window.showMenus = true;
window.showButtons = true;
/** generate and add extra chars in the element $charsElemId */
function addChars() {
$chars = $('#'+charsElemId);
if (!$chars.is(':empty')) {
// if there are extra chars already, do nothing
return $chars;
}
var cont = '';
var len = chars.length;
for (var i in chars) {
for (var j in chars[i]) {
cont += "<a href=\"javascript:mw.toolbar.insertTags('"+chars[i][j]+"', '', '')\" "+
'title="' + mw.msg("et-addchar", chars[i][j]) +'">'+chars[i][j]+'</a> ';
}
if (i != len-1) {
cont += '<br/>';
}
}
return $chars.html(cont);
}
mw.libs.EditToolbar.toggleChars = function() {
addChars().toggle();
};
/* * * Extra buttons for text insertion * * */
/** add some buttons and drop-down menus */
function setupCustomEditTools() {
if ( !$("#editform").length ) {
return;
}
mw.loader.using("mediawiki.toolbar", function(){
var toolbar = putToolbar(true);
toolbar.addClass("buttonlinks");
if ( showMenus ) {
// drop-down menus inserting text put direct in the javascript
appendDropDownMenus(toolbar, tplVarBaseName, insertIntoWikiText);
// drop-down menus inserting content from wiki pages
appendDropDownMenus(toolbar, atplVarBaseName, loadPage);
}
if ( showButtons ) {
appendCustomButtons(toolbar);
appendExtraChars(toolbar);
}
});
}
$(function() {
mw.loader.using("user", function(){
setupCustomEditTools();
});
});
function appendCustomButtons(parent) {
var buts = $('<div>', { id: "custombuttons" });
for (var i in customInsButtons) {
var el = customInsButtons[i];
var title, href, html;
if (el.length > 3) { // an Insert button
href = "javascript:mw.toolbar.insertTags('"+el[0] +"','"+el[2]+"','"+ el[1]+"')";
title = el[4];
html = el[3];
}
else { // a Misc button
href = "javascript:"+el[0];
title = el[2];
html = el[1];
}
if ( title.charAt(0) == "+" ) {
title = mw.msg("et-addpref", [title.substr(1)]);
}
appendCustomButton(buts, {
href: href,
title: title,
html: html
});
}
for (var i in customMiscButtons) {
var el = customMiscButtons[i];
appendCustomButton(buts, {
href: "javascript:"+el[0],
title: el[2],
html: el[1]
});
}
parent.append(buts);
}
function appendCustomButton(box, item) {
box.append($('<a>', item), ' ');
}
function appendExtraChars(parent) {
$('<div>', { id: charsElemId, style: 'display: none' }).appendTo(parent);
}
function appendDropDownMenus(parent, tplVarBaseName, callback) {
var tplVar = null;
for ( var i = 1; tplVar = tplVarBaseName + i,
eval("var tpl = typeof("+ tplVar +") == 'object' ? "+ tplVar +" : null"),
tpl != null; i++ ) {
appendDropDownMenu(parent, tpl, callback, "ddmenu_" + tplVar);
}
}
/** generates a drop-down menu */
function appendDropDownMenu(parent, content, callback, id) {
var box = $('<select>', {
id: id,
title: mw.msg("et-ddmenutitle")
}).on('change', function() {
if (this.value != "-") {
callback(this.value);
this.selectedIndex = 0;
}
return;
});
if ( appendOptions(box, content) > 1 ) {
parent.append(box);
}
}
function appendOptions(box, opts) {
var count = 0;
for (var i in opts) {
if (opts[i] == "") {
continue; // skip emtpy entries
}
if (typeof opts[i] == "object") {
var g = $('<optgroup>', {label: i});
for (var k in opts[i]) {
$('<option>', { value: opts[i][k], text: k }).appendTo(g);
}
box.append(g);
} else {
$('<option>', { value: opts[i], text: i }).appendTo(box);
}
count++;
}
return count;
}
/* * * * * * * * * * Ajax functions * * * * * * * * * */
function loadPage(page) {
showLoadIndicator();
var pageUrl = mw.util.getUrl(page, { action: 'raw', templates: 'expand' });
$.get(pageUrl)
.done(insertIntoWikiText)
.fail(function() {
alert(mw.msg('et-ajaxerror', [page]));
}).always(function() {
hideLoadIndicator();
});
}
function insertIntoWikiText(content) {
// delete text marked for no inclusion + <pre> and <nowiki>
var re = /<!--noinclude-->.*<!--\/noinclude-->|<\/?pre>|<\/?nowiki>/g;
content = content.replace(re, "");
// replace escaped tags
var specials = ['pre', 'nowiki'];
for (var i in specials) {
re = new RegExp("\\[(\/?)"+ specials[i] +"\\]", "g");
content = content.replace(re, "<$1"+ specials[i] +">");
}
// we can have >>|sample text|<< or >>|<< or just simple text
var parts = null;
var left, right, def = '';
content = escapeNl(content);
if ( ( parts = content.match(/(.*)>>\|(.*)\|<<(.*)/) ) ) {
left = parts[1];
def = parts[2];
right = parts[3];
} else { // no sample text: split at caret’s position
parts = content.split('>>|<<');
left = parts[0];
delete(parts[0]);
right = parts.join('');
}
mw.toolbar.insertTags(unescapeNl(left), unescapeNl(right), unescapeNl(def));
}
function escapeNl(s) {
return s.replace(/\n/g, "\x01");
}
function unescapeNl(s) {
return s.replace(/\x01/g, "\n");
}
var $loadIndicator;
function showLoadIndicator() {
if (!$loadIndicator) {
$loadIndicator = $('<div class="mw-ajax-loader"/>')
.css({
position: 'absolute',
top: $('#wpTextbox1').offset().top + 10,
left: '50%'
})
.hide()
.appendTo('body');
}
$loadIndicator.fadeIn();
}
function hideLoadIndicator() {
if ($loadIndicator) {
$loadIndicator.hide();
}
}
/* * * * * * * * * * Misc buttons * * * * * * * * * */
// if the cursor is within a wikilink or immediately after it,
// or if text is selected within the link, or including only one link, this function
// returns the index of the first [ and the index of the following ]] + 2,
// as well as the whole link text starting with [[ and ending with the closing ]]
mw.libs.EditToolbar.findFocusedLink = function (text, selStart, selEnd) {
var start = text.lastIndexOf('[[', selEnd - 1);
if (start == -1 || text.slice(start + 2).match(/^(File|Image|Файл|Картинка)/i)) return null;
var end = text.indexOf(']]', selStart - (selStart == selEnd ? 2 : 1)) + 2;
if (end == -1 || start > end) return null;
if (text.slice(start+2, end-2).match(/\[\[|\]\]/)) return null;
return {start: start, end: end, sel: text.slice(start, end)};
};
mw.libs.EditToolbar.linkWithAddrAndLabel = function () {
// en:Wikipedia:Page_name#Invalid_page_names :
var wikiLinkIllegalChars = /[\[\]{}<>|\u0000-\u001f\u007f\ufffd]/;
var indexOfWithDefault = function (str, searchStr, defaultVal, from) {
var index = str.indexOf(searchStr, (from || 0));
return (index > -1 ? index : defaultVal);
};
var $ta = $('#wpTextbox1');
var caretPos = $ta.textSelection( 'getCaretPosition', {startAndEnd: true} );
var text = $ta.val().replace(/\r/g, ''); // old IE returns new lines as \r\n
var sel, start, end;
var linkObj = mw.libs.EditToolbar.findFocusedLink(text, caretPos[0], caretPos[1]);
if (linkObj) {
start = linkObj.start;
end = linkObj.end;
sel = linkObj.sel;
}
else {
start = caretPos[0];
end = caretPos[1];
sel = text.slice(caretPos[0], caretPos[1]);
}
if (linkObj) { // if cursor before 1st space in address -> move before 1st space in link label
var indexOfBar = sel.indexOf('|');
if (indexOfBar > -1) {
var indexOf1stSpace = indexOfWithDefault(sel, ' ', indexOfBar);
if (caretPos[0] == caretPos[1] && caretPos[0] == start + indexOf1stSpace) {
var firstSpaceInLabel = indexOfWithDefault(sel, ' ', sel.length - 2, indexOfBar);
var newPos = start + firstSpaceInLabel;
$ta.focus().textSelection('setSelection', {start: newPos, end: newPos});
return;
}
}
}
var selTrimmed = sel.replace(/^ *(\[\[)? *| *(\]\])? *$/g, '').replace(/_+| +/g, ' ');
var linkAddr = selTrimmed.replace(/\u00a0| /g, ' '); // not necessary
if (wikiLinkIllegalChars.test(linkAddr)) { $ta.focus(); return; }
var spacesLeft = (sel.match(/^ +[^ ]/) || [' '])[0].slice(0, -1); // none if only spaces in string
var spacesRight = (sel.match(/ +$/) || [''])[0];
var firstSpaceIndex = indexOfWithDefault(linkAddr, ' ', linkAddr.length);
var newCaretPos = start + spacesLeft.length + 2 + firstSpaceIndex;
var wScrollTop = $(window).scrollTop(); // Chromium scrolls the page and textarea
var taScrollTop = $ta.scrollTop();
$ta.val( text.slice(0, start)
+ spacesLeft + '[[' + linkAddr + '|' + selTrimmed + ']]' + spacesRight
+ text.slice(end)
).focus().textSelection('setSelection', {start: newCaretPos, end: newCaretPos}).scrollTop(taScrollTop);
$(window).scrollTop(wScrollTop);
};
mw.libs.EditToolbar.graveAccent = function () {
var $ta = $('#wpTextbox1');
var val = $ta.val().replace(/\r/g, ''); // old IE returns new lines as \r\n
var caretPos = $ta.textSelection( 'getCaretPosition', {startAndEnd: true} );
var prevCharPos = caretPos[1] - 1;
if (prevCharPos < 0) {$ta.focus(); return;}
var prevChar = val[prevCharPos];
if (prevChar == 'и' || prevChar == 'й') result = 'ѝ';
else result = prevChar + '̀'
val = val.slice(0, prevCharPos) + result + val.slice(prevCharPos + 1);
var newCarretPos = prevCharPos + result.length;
var wScrollTop = $(window).scrollTop(); // Chromium scrolls the page and textarea
var taScrollTop = $ta.scrollTop();
$ta.val(val).focus().textSelection('setSelection', {start: newCarretPos, end: newCarretPos}).scrollTop(taScrollTop);
$(window).scrollTop(wScrollTop);
}
mw.libs.EditToolbar.mkList = function (numbered) {
function linesMatchRE(ls, re) {
return ls[0].match(re) !== null && ls[1].match(re) !== null && re;
}
var $ta = $('#wpTextbox1');
var sel = $ta.focus().textSelection('getSelection');
var lines = sel.replace(/<br\/?>/gi, '\n').replace(/^[ \t]+|[ \t]+$/mg, '')
.replace(/(\r?\n)+/g, '\n').split('\n');
var numLineRE = /^\d+\. */;
var numSignLineRE = /^# */;
var asteriskLineRE = /^\* */;
var removeRE;
var rmFirstLn = lines.length > 1 && lines[0] === '';
var rmLastLn = lines.length > 1 && lines[lines.length - 1] === '';
if (rmFirstLn) lines.shift(); // don't add * or # to first line if it's blank
if (rmLastLn) lines.pop(); // same for last line
if (lines.length > 1) // which regex to apply to rm chars from start of each line:
removeRE = linesMatchRE(lines, numLineRE)
|| (numbered && linesMatchRE(lines, asteriskLineRE)) // if adding # and ln start w/ *, rm *
|| (!numbered && linesMatchRE(lines, numSignLineRE))
if (!removeRE) removeRE = /^ +/;
for (var l, noSpace, i = lines.length - 1; i >= 0; i--) {
l = lines[i];
l = l.replace(removeRE, '');
noSpace = l.match(/^[*#]/) || sel === '';
lines[i] = (numbered ? '#' : '*') + (noSpace ? '' : ' ') + l;
}
if (rmFirstLn) lines.unshift('');
if (rmLastLn) lines.push('');
$ta.focus().textSelection('encapsulateSelection', {pre: lines.join('\n'), peri: '', post: '', replace: true});
}