MediaWiki:Gadget-RightToolbar.js:修订间差异
MediaWiki界面页面
更多操作
删除的内容 添加的内容
小 修改bug |
小 切换优化 |
||
| 第1行: | 第1行: | ||
/* ================================================================ |
/* ================================================================ |
||
MediaWiki Gadget: |
MediaWiki Gadget: Localized Clock Bar (增强版) |
||
功能: |
功能: |
||
- 显示浏览器本地时间和 UTC 时间 |
|||
- 回到顶部 / 回到底部按钮 |
|||
- 使用 MediaWiki 用户语言格式 |
|||
- 本地化日期和时间显示(浏览器时区 + UTC) |
|||
- 动态响应语言切换,无需刷新 |
|||
- 深色模式自动适配 |
|||
- |
- 深色模式适配 + 滑入动画 |
||
作者:你自己 |
作者:你自己 |
||
================================================================= */ |
================================================================= */ |
||
mw.loader.using(['mediawiki.util |
mw.loader.using(['mediawiki.util'], function () { |
||
// 工具栏主函数 |
|||
function updateClock() { |
|||
function initClockBar() { |
|||
// 浏览器本地时间 |
|||
var local = new Date(); |
|||
// UTC 时间,从浏览器系统获取即可 |
|||
var utc = new Date(local.getTime() + local.getTimezoneOffset() * 60000); |
|||
// MediaWiki 提供的语言设置 |
|||
var userLang = mw.config.get('wgUserLanguage') || 'en'; |
|||
// 本地化格式化函数 |
|||
// dateStyle/timeStyle 采用 Intl.DateTimeFormat 可选参数,也可以用短/长格式 |
|||
var localString = new Intl.DateTimeFormat(userLang, { |
|||
year: 'numeric', month: 'long', day: 'numeric', |
|||
hour: '2-digit', minute: '2-digit', second: '2-digit', |
|||
hour12: false |
|||
}).format(local); |
|||
var utcString = new Intl.DateTimeFormat(userLang, { |
|||
year: 'numeric', month: 'long', day: 'numeric', |
|||
hour: '2-digit', minute: '2-digit', second: '2-digit', |
|||
hour12: false, timeZone: 'UTC' |
|||
}).format(utc); |
|||
$('#current-local-time').text(localString); |
|||
$('#current-utc-time').text(utcString); |
|||
} |
|||
$(function () { |
|||
var isMobile = window.innerWidth <= 768; |
var isMobile = window.innerWidth <= 768; |
||
/ |
// ---------- 深色模式检测 ---------- |
||
const isDark = window.matchMedia('(prefers-color-scheme: dark)').matches; |
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 bg = isDark ? 'rgba(40,40,40,0.85)' : 'rgba(255,255,255,0.85)'; |
||
| 第47行: | 第21行: | ||
const color = isDark ? '#eee' : '#333'; |
const color = isDark ? '#eee' : '#333'; |
||
/ |
// ---------- 创建工具栏容器 ---------- |
||
var $container = $(' |
var $container = $('#mw-custom-tools'); |
||
if (!$container.length) { |
|||
position: 'fixed', |
|||
$container = $('<div id="mw-custom-tools"></div>').css({ |
|||
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 $('<div></div>', { id: id, title: tooltip, text: text }).css({ |
||
| 第82行: | 第59行: | ||
} |
} |
||
/ |
// ---------- 创建按钮 ---------- |
||
var $topButton = |
var $topButton = $('#mw-scroll-top'); |
||
if (!$topButton.length) { |
|||
$('html, body').animate({ scrollTop: 0 }, { duration: 500, easing: 'swing' }); |
|||
$topButton = createButton('mw-scroll-top', '▲', '回到顶部', function () { |
|||
}); |
|||
$('html, body').animate({ scrollTop: 0 }, { duration: 500, easing: 'swing' }); |
|||
}); |
|||
$container.append($topButton); |
|||
} |
|||
var $bottomButton = |
var $bottomButton = $('#mw-scroll-bottom'); |
||
if (!$bottomButton.length) { |
|||
$('html, body').animate({ scrollTop: $(document).height() }, { duration: 500, easing: 'swing' }); |
|||
$bottomButton = createButton('mw-scroll-bottom', '▼', '回到底部', function () { |
|||
}); |
|||
$('html, body').animate({ scrollTop: $(document).height() }, { duration: 500, easing: 'swing' }); |
|||
}); |
|||
$container.append($bottomButton); |
|||
} |
|||
/ |
// ---------- 时钟容器(桌面端) ---------- |
||
var $clockDiv = $('#current-clock'); |
|||
if (!isMobile && !$clockDiv.length) { |
|||
$clockDiv = $('<div id="current-clock"></div>').css({ |
|||
padding: '6px 12px', |
padding: '6px 12px', |
||
fontSize: '12px', |
fontSize: '12px', |
||
| 第105行: | 第91行: | ||
backdropFilter: 'blur(4px)' |
backdropFilter: 'blur(4px)' |
||
}).append( |
}).append( |
||
$('<div id=" |
$('<div id="local-date">加载中...</div>'), |
||
$('<div id=" |
$('<div id="local-time"></div>'), |
||
$('<div id="utc-time"></div>') |
|||
); |
); |
||
$container.append($clockDiv); |
$container.append($clockDiv); |
||
} |
} |
||
/ |
// ---------- 滚动动态显示 ---------- |
||
$topButton.hide(); // 初始隐藏 |
|||
$container.append($topButton, $bottomButton); |
|||
$(window).on('scroll', function () { |
|||
var scrollTop = $(this).scrollTop(); |
|||
if (scrollTop > 200) $topButton.fadeIn(); else $topButton.fadeOut(); |
|||
}); |
|||
/ |
// ---------- 初始动画 ---------- |
||
$container.animate({ opacity: 1, transform: 'translateY(0)' }, 600); |
$container.animate({ opacity: 1, transform: 'translateY(0)' }, 600); |
||
/ |
// ---------- 渲染时钟函数 ---------- |
||
function renderClock() { |
|||
var now = new Date(); |
|||
setInterval(updateClock, 1000); |
|||
// 获取 MediaWiki 用户语言(wgUserLanguage) |
|||
var userLang = mw.config.get('wgUserLanguage') || 'en'; |
|||
// 本地时间 |
|||
var localFormatter = new Intl.DateTimeFormat(userLang, { |
|||
year: 'numeric', month: 'long', day: 'numeric', |
|||
hour: '2-digit', minute: '2-digit', second: '2-digit', |
|||
hour12: false, |
|||
timeZone: Intl.DateTimeFormat().resolvedOptions().timeZone |
|||
}); |
|||
var localParts = localFormatter.formatToParts(now); |
|||
var localDateStr = localParts.filter(p => p.type === 'year' || p.type === 'month' || p.type === 'day') |
|||
.map(p => p.value).join(' '); |
|||
var localTimeStr = localParts.filter(p => p.type === 'hour' || p.type === 'minute' || p.type === 'second') |
|||
.map(p => p.value).join(':'); |
|||
$('#local-date').text('本地日期: ' + localDateStr); |
|||
$('#local-time').text('本地时间: ' + localTimeStr); |
|||
// UTC 时间 |
|||
var utcFormatter = new Intl.DateTimeFormat(userLang, { |
|||
year: 'numeric', month: 'long', day: 'numeric', |
|||
hour: '2-digit', minute: '2-digit', second: '2-digit', |
|||
hour12: false, |
|||
timeZone: 'UTC' |
|||
}); |
|||
var utcParts = utcFormatter.formatToParts(now); |
|||
var utcDateStr = utcParts.filter(p => p.type === 'year' || p.type === 'month' || p.type === 'day') |
|||
.map(p => p.value).join(' '); |
|||
var utcTimeStr = utcParts.filter(p => p.type === 'hour' || p.type === 'minute' || p.type === 'second') |
|||
.map(p => p.value).join(':'); |
|||
$('#utc-time').text('UTC 时间: ' + utcDateStr + ' ' + utcTimeStr); |
|||
} |
} |
||
/ |
// ---------- 定时刷新 ---------- |
||
setInterval(renderClock, 1000); |
|||
$topButton.hide(); // 初始隐藏 |
|||
renderClock(); // 启动立即渲染 |
|||
$(window).on('scroll', function () { |
|||
var scrollTop = $(this).scrollTop(); |
|||
// ---------- 语言切换动态刷新 ---------- |
|||
if (scrollTop > 200) { |
|||
// MediaWiki 用户语言修改后触发 user.languageChange hook |
|||
$topButton.fadeIn(); |
|||
mw.hook('user.languageChange').add(renderClock); |
|||
} else { |
|||
} |
|||
$topButton.fadeOut(); |
|||
} |
|||
// 初始化工具条 |
|||
}); |
|||
$(initClockBar); |
|||
}); |
}); |
||