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

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();
}
}
});
} );
}
}


$(initClockBar);
$( function () {
ensureToolbarStyles();
});
initClockBar();
} );
} );