打开/关闭菜单
打开/关闭外观设置菜单
打开/关闭个人菜单
未登录
未登录用户的IP地址会在进行任意编辑后公开展示。

MediaWiki:Gadget-RightToolbar.js:修订间差异

MediaWiki界面页面
删除的内容 添加的内容
XP-jia留言 | 贡献
简化提示
Maintenance script留言 | 贡献
拆分样式并优化性能
第1行: 第1行:
/* ================================================================
/* ================================================================
MediaWiki Gadget: Localized Clock Bar (增强版)
MediaWiki Gadget: Localized Clock Bar
功能:
功能:
- 显示浏览器本地时间和 UTC 时间
- 显示浏览器本地时间和 UTC 时间
- 使用 MediaWiki 用户语言格式
- 使用 MediaWiki 用户语言格式
- 动态响应语言切换,无需刷新
- 深色模式适配 + 滑入动画
- 深色模式适配 + 滑入动画
作者:你自己
================================================================= */
================================================================= */


mw.loader.using(['mediawiki.util'], function () {
mw.loader.using(['mediawiki.util']).then(function () {

// 工具栏主函数
function initClockBar() {
function initClockBar() {
var isMobile = window.innerWidth <= 768;
var isMobile = window.innerWidth <= 768;
var $container = $('#mw-right-toolbar');


// ---------- 深色模式检测 ----------
const isDark = window.matchMedia('(prefers-color-scheme: dark)').matches;
const bg = isDark ? 'rgba(40,40,40,0.85)' : 'rgba(255,255,255,0.85)';
const border = isDark ? '#555' : '#aaa';
const color = isDark ? '#eee' : '#333';

// ---------- 创建工具栏容器 ----------
var $container = $('#mw-custom-tools');
if (!$container.length) {
if (!$container.length) {
$container = $('<div id="mw-custom-tools"></div>').css({
$container = $('<div id="mw-right-toolbar" class="mw-right-toolbar"></div>').appendTo('body');
position: 'fixed',
bottom: '120px',
right: '50px',
display: 'flex',
flexDirection: 'column',
alignItems: 'flex-end',
gap: '5px',
zIndex: 9999,
opacity: 0,
transform: 'translateY(20px)', // 初始位置
}).appendTo('body');
}
}


// ---------- 按钮生成函数 ----------
function createButton(id, text, tooltip, clickHandler) {
function createButton(id, text, tooltip, clickHandler) {
return $('<div></div>', { id: id, title: tooltip, text: text }).css({
return $('<button></button>', {
padding: '5px 10px',
id: id,
fontSize: '12px',
class: 'mw-rt-button',
background: bg,
title: tooltip,
border: '1px solid ' + border,
text: text,
borderRadius: '4px',
type: 'button'
boxShadow: '0 1px 3px rgba(0,0,0,0.2)',
}).on('click', clickHandler);
color: color,
cursor: 'pointer',
textAlign: 'center',
opacity: 0.7,
transition: 'background 0.3s, opacity 0.3s, transform 0.3s',
backdropFilter: 'blur(4px)'
}).hover(
function () { $(this).css({ background: isDark ? '#555' : '#ddd', opacity: 1 }); },
function () { $(this).css({ background: bg, opacity: 0.7 }); }
).click(clickHandler);
}
}


// ---------- 创建按钮 ----------
var $topButton = $('#mw-scroll-top');
var $topButton = $('#mw-scroll-top');
if (!$topButton.length) {
if (!$topButton.length) {
第76行: 第42行:
}
}


var $clockDiv = $('#mw-right-toolbar-clock');
// ---------- 时钟容器(桌面端) ----------
var $clockDiv = $('#current-clock');
if (!isMobile && !$clockDiv.length) {
if (!isMobile && !$clockDiv.length) {
$clockDiv = $('<div id="current-clock"></div>').css({
$clockDiv = $('<div id="mw-right-toolbar-clock" class="mw-rt-clock"></div>').append(
padding: '6px 12px',
$('<div id="mw-local-date">加载中...</div>'),
fontSize: '12px',
$('<div id="mw-local-time"></div>'),
background: bg,
$('<div id="mw-utc-time"></div>')
border: '1px solid ' + border,
borderRadius: '4px',
boxShadow: '0 1px 3px rgba(0,0,0,0.2)',
color: color,
textAlign: 'center',
lineHeight: '1.4',
backdropFilter: 'blur(4px)'
}).append(
$('<div id="local-date">加载中...</div>'),
$('<div id="local-time"></div>'),
$('<div id="utc-time"></div>')
);
);
$container.append($clockDiv);
$container.append($clockDiv);
}
}


$topButton.hide();
// ---------- 滚动动态显示 ----------
$topButton.hide(); // 初始隐藏
var ticking = false;
$(window).on('scroll', function () {
$(window).on('scroll', function () {
var scrollTop = $(this).scrollTop();
if (ticking) {
if (scrollTop > 200) $topButton.fadeIn(); else $topButton.fadeOut();
return;
}
ticking = true;
window.requestAnimationFrame(function () {
var scrollTop = $(window).scrollTop();
if (scrollTop > 200) {
$topButton.fadeIn();
} else {
$topButton.fadeOut();
}
ticking = false;
});
});
});


window.requestAnimationFrame(function () {
// ---------- 初始动画 ----------
$container.addClass('is-visible');
$container.animate({ opacity: 1, transform: 'translateY(0)' }, 600);
});


// ---------- 渲染时钟函数 ----------
function renderClock() {
function renderClock() {
var now = new Date();
var now = new Date();

// 获取 MediaWiki 用户语言(wgUserLanguage)
var userLang = mw.config.get('wgUserLanguage') || 'en';
var userLang = mw.config.get('wgUserLanguage') || 'en';


// 本地时间
var localFormatter = new Intl.DateTimeFormat(userLang, {
var localFormatter = new Intl.DateTimeFormat(userLang, {
year: 'numeric', month: 'long', day: 'numeric',
year: 'numeric',
hour: '2-digit', minute: '2-digit', second: '2-digit',
month: 'long',
day: 'numeric',
hour: '2-digit',
minute: '2-digit',
second: '2-digit',
hour12: false,
hour12: false,
timeZone: Intl.DateTimeFormat().resolvedOptions().timeZone
timeZone: Intl.DateTimeFormat().resolvedOptions().timeZone
});
});

var localParts = localFormatter.formatToParts(now);
var localParts = localFormatter.formatToParts(now);
var localDateStr = localParts.filter(p => p.type === 'year' || p.type === 'month' || p.type === 'day')
var localDateStr = localParts.filter(function (p) {
.map(p => p.value).join(' ');
return p.type === 'year' || p.type === 'month' || p.type === 'day';
}).map(function (p) {
var localTimeStr = localParts.filter(p => p.type === 'hour' || p.type === 'minute' || p.type === 'second')
.map(p => p.value).join(':');
return p.value;
$('#local-date').text(localDateStr);
}).join(' ');

$('#local-time').text(localTimeStr);
var localTimeStr = localParts.filter(function (p) {
return p.type === 'hour' || p.type === 'minute' || p.type === 'second';
}).map(function (p) {
return p.value;
}).join(':');

$('#mw-local-date').text(localDateStr);
$('#mw-local-time').text(localTimeStr);


// UTC 时间
var utcFormatter = new Intl.DateTimeFormat(userLang, {
var utcFormatter = new Intl.DateTimeFormat(userLang, {
year: 'numeric', month: 'long', day: 'numeric',
year: 'numeric',
hour: '2-digit', minute: '2-digit', second: '2-digit',
month: 'long',
day: 'numeric',
hour: '2-digit',
minute: '2-digit',
second: '2-digit',
hour12: false,
hour12: false,
timeZone: 'UTC'
timeZone: 'UTC'
});
});

var utcParts = utcFormatter.formatToParts(now);
var utcParts = utcFormatter.formatToParts(now);
var utcDateStr = utcParts.filter(p => p.type === 'year' || p.type === 'month' || p.type === 'day')
var utcDateStr = utcParts.filter(function (p) {
.map(p => p.value).join(' ');
return p.type === 'year' || p.type === 'month' || p.type === 'day';
}).map(function (p) {
var utcTimeStr = utcParts.filter(p => p.type === 'hour' || p.type === 'minute' || p.type === 'second')
.map(p => p.value).join(':');
return p.value;
$('#utc-time').text('UTC: ' + utcDateStr + ' ' + utcTimeStr);
}).join(' ');

var utcTimeStr = utcParts.filter(function (p) {
return p.type === 'hour' || p.type === 'minute' || p.type === 'second';
}).map(function (p) {
return p.value;
}).join(':');

$('#mw-utc-time').text('UTC: ' + utcDateStr + ' ' + utcTimeStr);
}
}


var clockTimer = window.setInterval(renderClock, 1000);
// ---------- 定时刷新 ----------
setInterval(renderClock, 1000);
renderClock();
renderClock(); // 启动立即渲染


// ---------- 语言切换动态刷新 ----------
// MediaWiki 用户语言修改后触发 user.languageChange hook
mw.hook('user.languageChange').add(renderClock);
mw.hook('user.languageChange').add(renderClock);

document.addEventListener('visibilitychange', function () {
if (document.hidden) {
window.clearInterval(clockTimer);
clockTimer = null;
return;
}
if (!clockTimer) {
clockTimer = window.setInterval(renderClock, 1000);
renderClock();
}
});
}
}


// 初始化工具条
$(initClockBar);
$(initClockBar);
});
});