MediaWiki:Gadget-RightToolbar.js:修订间差异
MediaWiki界面页面
更多操作
删除的内容 添加的内容
Maintenance script(留言 | 贡献) 拆分样式并优化性能 |
Maintenance script(留言 | 贡献) Inline fallback styles for RightToolbar |
||
| 第7行: | 第7行: | ||
================================================================= */ |
================================================================= */ |
||
mw.loader.using(['mediawiki.util']).then(function () { |
mw.loader.using( [ 'mediawiki.util' ] ).then( function () { |
||
var STYLE_ID = 'mw-right-toolbar-style'; |
|||
function ensureToolbarStyles() { |
|||
if ( document.getElementById( STYLE_ID ) ) { |
|||
return; |
|||
} |
|||
var css = [ |
|||
'#mw-right-toolbar, .mw-right-toolbar {', |
|||
' position: fixed;', |
|||
' bottom: 96px;', |
|||
' right: 24px;', |
|||
' display: flex;', |
|||
' flex-direction: column;', |
|||
' align-items: flex-end;', |
|||
' gap: 6px;', |
|||
' z-index: 2147483647;', |
|||
' opacity: 0;', |
|||
' transform: translateY(12px);', |
|||
' transition: opacity 0.4s ease, transform 0.4s ease;', |
|||
'}', |
|||
'', |
|||
'.mw-right-toolbar.is-visible {', |
|||
' opacity: 1;', |
|||
' transform: translateY(0);', |
|||
'}', |
|||
'', |
|||
'.mw-right-toolbar .mw-rt-button,', |
|||
'.mw-right-toolbar .mw-rt-clock {', |
|||
' background: var(--color-surface-0, rgba(255, 255, 255, 0.85));', |
|||
' border: 1px solid var(--border-color-base, #aaa);', |
|||
' color: var(--color-base, #333);', |
|||
' border-radius: 4px;', |
|||
' box-shadow: 0 1px 3px rgba(0, 0, 0, 0.2);', |
|||
' font-size: 12px;', |
|||
' line-height: 1.4;', |
|||
' padding: 6px 12px;', |
|||
' backdrop-filter: blur(4px);', |
|||
'}', |
|||
'', |
|||
'.mw-right-toolbar .mw-rt-button {', |
|||
' cursor: pointer;', |
|||
' opacity: 0.8;', |
|||
' transition: background 0.2s ease, opacity 0.2s ease, transform 0.2s ease;', |
|||
'}', |
|||
'', |
|||
'.mw-right-toolbar .mw-rt-button:hover {', |
|||
' opacity: 1;', |
|||
' background: var(--background-color-progressive-subtle, #ddd);', |
|||
'}', |
|||
'', |
|||
'.mw-right-toolbar .mw-rt-button:active {', |
|||
' transform: translateY(1px);', |
|||
'}', |
|||
'', |
|||
'.skin-theme-clientpref-night .mw-right-toolbar .mw-rt-button,', |
|||
'.skin-theme-clientpref-night .mw-right-toolbar .mw-rt-clock {', |
|||
' background: rgba(30, 30, 30, 0.85);', |
|||
' border-color: #555;', |
|||
' color: #eee;', |
|||
'}', |
|||
'', |
|||
'@media (max-width: 768px) {', |
|||
' #mw-right-toolbar, .mw-right-toolbar {', |
|||
' bottom: 72px;', |
|||
' right: 16px;', |
|||
' }', |
|||
'}' |
|||
].join( '\n' ); |
|||
var styleNode = mw.loader.addStyleTag( css ); |
|||
styleNode.id = STYLE_ID; |
|||
} |
|||
function initClockBar() { |
function initClockBar() { |
||
ensureToolbarStyles(); |
|||
var isMobile = window.innerWidth <= 768; |
var isMobile = window.innerWidth <= 768; |
||
var $container = $('#mw-right-toolbar'); |
var $container = $( '#mw-right-toolbar' ); |
||
if (!$container.length) { |
if ( !$container.length ) { |
||
$container = $('<div id="mw-right-toolbar" class="mw-right-toolbar"></div>').appendTo('body'); |
$container = $( '<div id="mw-right-toolbar" class="mw-right-toolbar"></div>' ).appendTo( 'body' ); |
||
} |
} |
||
function createButton(id, text, tooltip, clickHandler) { |
function createButton( id, text, tooltip, clickHandler ) { |
||
return $('<button></button>', { |
return $( '<button></button>', { |
||
id: id, |
id: id, |
||
class: 'mw-rt-button', |
class: 'mw-rt-button', |
||
| 第23行: | 第99行: | ||
text: text, |
text: text, |
||
type: 'button' |
type: 'button' |
||
}).on('click', clickHandler); |
} ).on( 'click', clickHandler ); |
||
} |
} |
||
var $topButton = $('#mw-scroll-top'); |
var $topButton = $( '#mw-scroll-top' ); |
||
if (!$topButton.length) { |
if ( !$topButton.length ) { |
||
$topButton = createButton('mw-scroll-top', '▲', '回到顶部', function () { |
$topButton = createButton( 'mw-scroll-top', '▲', '回到顶部', function () { |
||
$('html, body').animate({ scrollTop: 0 }, { duration: 500, easing: 'swing' }); |
$( 'html, body' ).animate( { scrollTop: 0 }, { duration: 500, easing: 'swing' } ); |
||
}); |
} ); |
||
$container.append($topButton); |
$container.append( $topButton ); |
||
} |
} |
||
var $bottomButton = $('#mw-scroll-bottom'); |
var $bottomButton = $( '#mw-scroll-bottom' ); |
||
if (!$bottomButton.length) { |
if ( !$bottomButton.length ) { |
||
$bottomButton = createButton('mw-scroll-bottom', '▼', '回到底部', function () { |
$bottomButton = createButton( 'mw-scroll-bottom', '▼', '回到底部', function () { |
||
$('html, body').animate({ scrollTop: $(document).height() }, { duration: 500, easing: 'swing' }); |
$( 'html, body' ).animate( { scrollTop: $( document ).height() }, { duration: 500, easing: 'swing' } ); |
||
}); |
} ); |
||
$container.append($bottomButton); |
$container.append( $bottomButton ); |
||
} |
} |
||
var $clockDiv = $('#mw-right-toolbar-clock'); |
var $clockDiv = $( '#mw-right-toolbar-clock' ); |
||
if (!isMobile && !$clockDiv.length) { |
if ( !isMobile && !$clockDiv.length ) { |
||
$clockDiv = $('<div id="mw-right-toolbar-clock" class="mw-rt-clock"></div>').append( |
$clockDiv = $( '<div id="mw-right-toolbar-clock" class="mw-rt-clock"></div>' ).append( |
||
$('<div id="mw-local-date">加载中...</div>'), |
$( '<div id="mw-local-date">加载中...</div>' ), |
||
$('<div id="mw-local-time"></div>'), |
$( '<div id="mw-local-time"></div>' ), |
||
$('<div id="mw-utc-time"></div>') |
$( '<div id="mw-utc-time"></div>' ) |
||
); |
); |
||
$container.append($clockDiv); |
$container.append( $clockDiv ); |
||
} |
} |
||
$topButton.hide(); |
$topButton.hide(); |
||
var ticking = false; |
var ticking = false; |
||
$(window).on('scroll', function () { |
$( window ).on( 'scroll', function () { |
||
if (ticking) { |
if ( ticking ) { |
||
return; |
return; |
||
} |
} |
||
ticking = true; |
ticking = true; |
||
window.requestAnimationFrame(function () { |
window.requestAnimationFrame( function () { |
||
var scrollTop = $(window).scrollTop(); |
var scrollTop = $( window ).scrollTop(); |
||
if (scrollTop > 200) { |
if ( scrollTop > 200 ) { |
||
$topButton.fadeIn(); |
$topButton.fadeIn(); |
||
} else { |
} else { |
||
| 第67行: | 第143行: | ||
} |
} |
||
ticking = false; |
ticking = false; |
||
}); |
} ); |
||
}); |
} ); |
||
window.requestAnimationFrame(function () { |
window.requestAnimationFrame( function () { |
||
$container.addClass('is-visible'); |
$container.addClass( 'is-visible' ); |
||
}); |
} ); |
||
function renderClock() { |
function renderClock() { |
||
var now = new Date(); |
var now = new Date(); |
||
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', |
year: 'numeric', |
||
month: 'long', |
month: 'long', |
||
| 第87行: | 第163行: | ||
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(function (p) { |
var localDateStr = localParts.filter( function ( p ) { |
||
return p.type === 'year' || p.type === 'month' || p.type === 'day'; |
return p.type === 'year' || p.type === 'month' || p.type === 'day'; |
||
}).map(function (p) { |
} ).map( function ( p ) { |
||
return p.value; |
return p.value; |
||
}).join(' '); |
} ).join( ' ' ); |
||
var localTimeStr = localParts.filter(function (p) { |
var localTimeStr = localParts.filter( function ( p ) { |
||
return p.type === 'hour' || p.type === 'minute' || p.type === 'second'; |
return p.type === 'hour' || p.type === 'minute' || p.type === 'second'; |
||
}).map(function (p) { |
} ).map( function ( p ) { |
||
return p.value; |
return p.value; |
||
}).join(':'); |
} ).join( ':' ); |
||
$('#mw-local-date').text(localDateStr); |
$( '#mw-local-date' ).text( localDateStr ); |
||
$('#mw-local-time').text(localTimeStr); |
$( '#mw-local-time' ).text( localTimeStr ); |
||
var utcFormatter = new Intl.DateTimeFormat(userLang, { |
var utcFormatter = new Intl.DateTimeFormat( userLang, { |
||
year: 'numeric', |
year: 'numeric', |
||
month: 'long', |
month: 'long', |
||
| 第114行: | 第190行: | ||
hour12: false, |
hour12: false, |
||
timeZone: 'UTC' |
timeZone: 'UTC' |
||
}); |
} ); |
||
var utcParts = utcFormatter.formatToParts(now); |
var utcParts = utcFormatter.formatToParts( now ); |
||
var utcDateStr = utcParts.filter(function (p) { |
var utcDateStr = utcParts.filter( function ( p ) { |
||
return p.type === 'year' || p.type === 'month' || p.type === 'day'; |
return p.type === 'year' || p.type === 'month' || p.type === 'day'; |
||
}).map(function (p) { |
} ).map( function ( p ) { |
||
return p.value; |
return p.value; |
||
}).join(' '); |
} ).join( ' ' ); |
||
var utcTimeStr = utcParts.filter(function (p) { |
var utcTimeStr = utcParts.filter( function ( p ) { |
||
return p.type === 'hour' || p.type === 'minute' || p.type === 'second'; |
return p.type === 'hour' || p.type === 'minute' || p.type === 'second'; |
||
}).map(function (p) { |
} ).map( function ( p ) { |
||
return p.value; |
return p.value; |
||
}).join(':'); |
} ).join( ':' ); |
||
$('#mw-utc-time').text('UTC: ' + utcDateStr + ' ' + utcTimeStr); |
$( '#mw-utc-time' ).text( 'UTC: ' + utcDateStr + ' ' + utcTimeStr ); |
||
} |
} |
||
var clockTimer = window.setInterval(renderClock, 1000); |
var clockTimer = window.setInterval( renderClock, 1000 ); |
||
renderClock(); |
renderClock(); |
||
mw.hook('user.languageChange').add(renderClock); |
mw.hook( 'user.languageChange' ).add( renderClock ); |
||
document.addEventListener('visibilitychange', function () { |
document.addEventListener( 'visibilitychange', function () { |
||
if (document.hidden) { |
if ( document.hidden ) { |
||
window.clearInterval(clockTimer); |
window.clearInterval( clockTimer ); |
||
clockTimer = null; |
clockTimer = null; |
||
return; |
return; |
||
} |
} |
||
if (!clockTimer) { |
if ( !clockTimer ) { |
||
clockTimer = window.setInterval(renderClock, 1000); |
clockTimer = window.setInterval( renderClock, 1000 ); |
||
renderClock(); |
renderClock(); |
||
} |
} |
||
}); |
} ); |
||
} |
} |
||
$( |
$( function () { |
||
ensureToolbarStyles(); |
|||
| ⚫ | |||
initClockBar(); |
|||
} ); |
|||
| ⚫ | |||