MediaWiki:Common.js: различия между версиями

Нет описания правки
Нет описания правки
 
(не показано 14 промежуточных версий этого же участника)
Строка 627: Строка 627:
});
});
}
}
// Для "Шаблон:Ajax"
function fetchParsedData(wikiText) {
function initAjaxLoader() {
var apiUrl = "https://station14.ru/api.php?action=parse&format=json&prop=text&text=" + encodeURIComponent(wikiText) + "&origin=*";
var ajaxContainers = document.querySelectorAll('.ajax-load, .ajax-load-link');
return $.ajax({
var BATCH_SIZE = 1000;
url: apiUrl,
var queue = [];
method: "GET",
var processing = false;
dataType: "json"
});
}
 
function enhanceParsedContent(root) {
if (!root) return;
 
var scripts = Array.prototype.slice.call(root.querySelectorAll("script"));
scripts.forEach(function (scr) {
var parent = scr.parentNode;
var scriptNode = document.createElement("script");
 
if (scr.src) {
scriptNode.src = scr.src;
if (scr.defer) scriptNode.defer = true;
if (scr.async) scriptNode.async = true;
parent.replaceChild(scriptNode, scr);
} else {
try {
$.globalEval(scr.textContent || scr.innerText || "");
} catch (e) {}
parent.removeChild(scr);
}
});
 
mw.loader.using(['jquery.tablesorter', 'jquery.makeCollapsible'], function () {
$(root).find('table.sortable').tablesorter();
$(root).find('.mw-collapsible').makeCollapsible();
});
 
mw.hook('wikipage.content').fire(root);
 
initAjaxLoader(root);
}
 
function injectParsedHTML(parsedHTML, options) {
options = options || {};
 
var container = document.createElement("span");
container.innerHTML = parsedHTML;


var loadingImgHTML = '<img alt="Файл:Silva-loading.gif" src="/images/4/45/Silva-loading.gif?20260316201152" decoding="async" width="16" height="16" data-file-width="16" data-file-height="16" style="vertical-align:middle;margin-right:6px">';
if (options.keepNode) {
container.insertBefore(options.keepNode, container.firstChild);
}


function fetchParsedData(wikiText) {
if (options.placeholder && options.placeholder.parentNode) {
var apiUrl = "https://station14.ru/api.php?action=parse&format=json&prop=text&text=" + encodeURIComponent(wikiText) + "&origin=*";
options.placeholder.replaceWith(container);
return $.ajax({
} else if (options.targetRoot) {
url: apiUrl,
options.targetRoot.innerHTML = "";
method: "GET",
while (container.firstChild) {
dataType: "json"
options.targetRoot.appendChild(container.firstChild);
});
}
container = options.targetRoot;
}
}


function applyParsedHTML(parsedHTML, placeholder) {
enhanceParsedContent(container);
var newContainer = document.createElement("span");
newContainer.innerHTML = parsedHTML;


if (placeholder && placeholder.parentNode) {
return container;
placeholder.replaceWith(newContainer);
}
}


var scripts = Array.prototype.slice.call(newContainer.querySelectorAll("script"));
function parseAndInjectWikiText(wikiText, options) {
scripts.forEach(function (scr) {
return fetchParsedData(wikiText)
var parent = scr.parentNode;
.done(function (data) {
var scriptNode = document.createElement("script");
if (data && data.parse && data.parse.text) {
if (scr.src) {
var parsedHTML = data.parse.text["*"] || "";
scriptNode.src = scr.src;
injectParsedHTML(parsedHTML, options);
if (scr.defer) scriptNode.defer = true;
} else if (options && options.placeholder) {
if (scr.async) scriptNode.async = true;
options.placeholder.textContent = "API не вернул ожидаемых данных.";
parent.replaceChild(scriptNode, scr);
}
} else {
})
try { $.globalEval(scr.textContent || scr.innerText || ""); } catch (e) {}
.fail(function () {
parent.removeChild(scr);
if (options && options.placeholder) {
options.placeholder.textContent = "Ошибка при выполнении запроса к API.";
}
}
});
});
}


mw.loader.using(['jquery.tablesorter', 'jquery.makeCollapsible'], function () {
function createPlaceholderWithSpinner(text) {
$(newContainer).find('table.sortable').tablesorter();
if (!text) {
$(newContainer).find('.mw-collapsible').makeCollapsible();
loadingImgHTML = '<img alt="Файл:Silva-loading.gif" src="/images/4/45/Silva-loading.gif" decoding="async" width="16" height="16" style="vertical-align:middle;margin-right:6px">';
});
text = loadingImgHTML + '<span class="ajax-load-text">Пожалуйста, подождите, содержимое загружается...</span>';
}
 
var placeholder = document.createElement("span");
placeholder.className = "ajax-load-placeholder";
placeholder.innerHTML = text;
 
return placeholder;
}
 
// Для "Шаблон:Ajax"
function initAjaxLoader(root) {
root = root || document;


mw.hook('wikipage.content').fire(newContainer);
var ajaxContainers = root.querySelectorAll('.ajax-load, .ajax-load-link');
mw.loader.load('//station14.ru/w/index.php?title=MediaWiki:Common.js&action=raw&ctype=text/javascript');
var BATCH_SIZE = 1000;
}
var queue = [];
var processing = false;


function processQueue() {
function processQueue() {
Строка 691: Строка 745:


batch.forEach(function (job) {
batch.forEach(function (job) {
fetchParsedData(job.wikiText)
parseAndInjectWikiText(job.wikiText, { placeholder: job.placeholder })
.done(function (data) {
if (data.parse && data.parse.text) {
var parsedHTML = data.parse.text["*"] || "";
applyParsedHTML(parsedHTML, job.placeholder);
} else {
if (job.placeholder) job.placeholder.textContent = "API не вернул ожидаемых данных.";
}
})
.fail(function () {
if (job.placeholder) job.placeholder.textContent = "Ошибка при выполнении запроса к API.";
})
.always(function () {
.always(function () {
remaining--;
remaining--;
Строка 722: Строка 765:
if (!wikiText) return;
if (!wikiText) return;


var loadingText = (loadingEl && (loadingEl.textContent || loadingEl.innerText || "").trim()) || "Загрузка...";
var loadingHTML = loadingEl ? (loadingEl.innerHTML || "").trim() : "";


if (loadingEl && loadingEl.parentNode) {
if (loadingEl && loadingEl.parentNode) {
loadingEl.parentNode.removeChild(loadingEl);
loadingEl.parentNode.removeChild(loadingEl);
loadingEl = null;
}
function createPlaceholderWithSpinner(text) {
var placeholder = document.createElement("span");
placeholder.className = "ajax-load-placeholder";
placeholder.innerHTML = loadingImgHTML + '<span class="ajax-load-text">' + (text || '') + '</span>';
return placeholder;
}
}


Строка 743: Строка 778:
container.dataset.ajaxLoaded = "1";
container.dataset.ajaxLoaded = "1";


var placeholder = createPlaceholderWithSpinner(loadingText);
var placeholder = createPlaceholderWithSpinner(loadingHTML);


if (container && container.parentNode) {
if (container && container.parentNode) {
container.parentNode.replaceChild(placeholder, container);
container.parentNode.replaceChild(placeholder, container);
} else {
} else if (contentEl && contentEl.parentNode) {
if (contentEl && contentEl.parentNode) {
contentEl.parentNode.replaceChild(placeholder, contentEl);
contentEl.parentNode.replaceChild(placeholder, contentEl);
}
}
}


Строка 757: Строка 790:
}, { once: true });
}, { once: true });
} else {
} else {
var placeholder = createPlaceholderWithSpinner(loadingText);
var placeholder = createPlaceholderWithSpinner(loadingHTML);


if (contentEl && contentEl.parentNode) {
if (contentEl && contentEl.parentNode) {
Строка 988: Строка 1021:
}
}
}
}
// Для "Шаблон:ProjectSelectionGenerator"
// Для "Шаблон:Переключатель_проекта"
function initProjectSelectionGenerator() {
function initProjectSelectionGenerator() {
var containers = document.getElementsByClassName('js-project-selection-generator');
var containers = document.getElementsByClassName('js-project-selection-generator');
Строка 1003: Строка 1036:
}
}


function getCurrentPageWikitext() {
function getPageWikitext() {
var title = mw.config.get('wgPageName');
var title = mw.config.get('wgPageName');
var apiUrl = mw.util.wikiScript('api') +
var apiUrl = mw.util.wikiScript('api') + '?action=query&prop=revisions&rvslots=main&rvprop=content&titles=' + encodeURIComponent(title) + '&format=json&formatversion=2&origin=*';
'?action=query' +
'&prop=revisions' +
'&rvslots=main' +
'&rvprop=content' +
'&titles=' + encodeURIComponent(title) +
'&format=json' +
'&formatversion=2' +
'&origin=*';


return $.ajax({
return $.ajax({
Строка 1043: Строка 1068:


function stripProjectSelectionTemplate(text) {
function stripProjectSelectionTemplate(text) {
if (!text) return '';
return String(text || '').replace(/\{\{\s*переключатель проекта\b[\s\S]*?\}\}/ig, '');
 
return text.replace(/\{\{\s*projectSelectionGenerator\b[\s\S]*?\}\}/ig, '');
}
}


function buildWikitextForProject(text, projectName) {
function buildWikitext(text, projectName) {
var cleanedText = stripProjectSelectionTemplate(text);
return '{{#vardefine:JsonPath|' + projectName + '}}\n' + stripProjectSelectionTemplate(text);
 
return '{{#vardefine:Path|' + projectName + '}}\n' + cleanedText;
}
}


function parseWikiText(wikiText) {
function refreshForProject(projectName) {
var apiUrl = mw.util.wikiScript('api') +
var root = getContentRoot();
'?action=parse' +
if (!root) return;
'&format=json' +
'&formatversion=2' +
'&prop=text' +
'&text=' + encodeURIComponent(wikiText) +
'&disablelimitreport=1' +
'&disableeditsection=1' +
'&disabletoc=1' +
'&origin=*';


return $.ajax({
var placeholder = createPlaceholderWithSpinner();
url: apiUrl,
method: 'GET',
dataType: 'json'
});
}


function applyParsedHTML(parsedHTML, preservedMenus) {
root.innerHTML = '';
var contentRoot = getContentRoot();
root.appendChild(placeholder);
if (!contentRoot) return;


contentRoot.innerHTML = parsedHTML;
getPageWikitext()
.then(function (sourceText) {
var wikitext = buildWikitext(sourceText, projectName);


if (preservedMenus && preservedMenus.length > 0) {
return parseAndInjectWikiText(wikitext, {
for (var i = 0; i < preservedMenus.length; i++) {
placeholder: placeholder,
contentRoot.insertBefore(preservedMenus[i], contentRoot.firstChild);
targetRoot: root
}
});
}
 
mw.hook('wikipage.content').fire(contentRoot);
}
 
function refreshPageForProject(projectName, preservedMenus) {
getCurrentPageWikitext()
.then(function (sourceText) {
var newText = buildWikitextForProject(sourceText, projectName);
return parseWikiText(newText);
})
.then(function (data) {
if (data && data.parse && data.parse.text) {
var parsedHTML = data.parse.text["*"] || data.parse.text || "";
applyParsedHTML(parsedHTML, preservedMenus);
}
})
})
.fail(function () {
.fail(function () {
console.error('Не удалось перепарсить страницу для проекта:', projectName);
placeholder.textContent = 'Ошибка загрузки страницы.';
});
});
}
}
Строка 1107: Строка 1100:
for (var ci = 0; ci < containers.length; ci++) {
for (var ci = 0; ci < containers.length; ci++) {
var container = containers[ci];
var container = containers[ci];
if (container.getAttribute('data-project-selection-initialized') === '1') {
continue;
}


var rawText = container.textContent || '';
var rawText = container.textContent || '';
Строка 1155: Строка 1144:
}
}


container.textContent = '';
var portletList = document.querySelector('#mw-content .mw-portlet-body ul');
container.appendChild(form);
if (portletList) {
portletList.appendChild(form);
}


(function (menuContainer) {
(function (menuNode) {
var radios = menuContainer.querySelectorAll('input[type="radio"][name="js-project-selection"]');
menuNode.addEventListener('change', function (event) {
var target = event.target;
if (!target || target.tagName !== 'INPUT') return;
if (target.type !== 'radio') return;
if (target.name !== 'js-project-selection') return;
if (!target.checked) return;


for (var r = 0; r < radios.length; r++) {
refreshForProject(target.value);
radios[r].addEventListener('change', function () {
});
if (!this.checked) return;
})(form);
 
var projectName = this.value;
 
var preservedMenus = [];
var menuNode = menuContainer;
if (menuNode && menuNode.parentNode) {
preservedMenus.push(menuNode);
}
 
refreshPageForProject(projectName, preservedMenus);
});
}
})(container);
 
container.setAttribute('data-project-selection-initialized', '1');
}
}
}
}