MediaWiki:Timeless.js:修订间差异
来自「荏苒之境」
| 无编辑摘要 | 无编辑摘要 | ||
| (未显示同一用户的6个中间版本) | |||
| 第105行: | 第105行: | ||
| let floatingToC; | let floatingToC; | ||
| let toc = document.getElementById('toc'); | let toc = document.getElementById('toc'); | ||
| let tocInitialParent = toc.parentNode; | |||
| const userAgent = navigator.userAgent; | const userAgent = navigator.userAgent; | ||
| 第119行: | 第120行: | ||
| }); | }); | ||
| const  | const updateToCParent = () => { | ||
| 	let width = window.innerWidth; | 	let width = window.innerWidth; | ||
| 	if (isSafari && width > 1354 || width >  | 	if (isSafari && width > 1354 || !isSafari && width > 1339) { | ||
| 		if (toc.parentNode != siteNavigation) { | 		if (toc.parentNode != siteNavigation) { | ||
| 			toc.parentNode.removeChild(toc); | 			toc.parentNode.removeChild(toc); | ||
| 第128行: | 第129行: | ||
| 	} | 	} | ||
| 	else { | 	else { | ||
| 		if (toc.parentNode == siteNavigation) { | 		if (toc.parentNode == tocInitialParent || toc.parentNode == siteNavigation) { | ||
| 			toc.parentNode.removeChild(toc); | 			toc.parentNode.removeChild(toc); | ||
| 			siteNavigation.parentNode.lastChild.before(toc); | 			siteNavigation.parentNode.lastChild.before(toc); | ||
| 第139行: | 第140行: | ||
| 		return; | 		return; | ||
| 	} | 	} | ||
| 	updateToCParent(); | |||
| }); | }); | ||
| document.addEventListener('pageupdate', () => { | document.addEventListener('pageupdate', () => { | ||
| 	let  | 	let newToC = document.querySelectorAll('.mw-parser-output > #toc')[0]; | ||
| 	if ( | 	if (newToC == null) { | ||
| 		return; | 		return; | ||
| 	} | 	} | ||
| 第150行: | 第151行: | ||
| 		floatingToC.remove(); | 		floatingToC.remove(); | ||
| 	} | 	} | ||
| 	toc =  | 	toc = newToC; | ||
| 	floatingToC = new FloatingToC(toc); | |||
| 	updateToCParent(); | |||
| }); | }); | ||
| if (toc) { | if (toc) { | ||
|      floatingToC = new FloatingToC(toc); |      floatingToC = new FloatingToC(toc); | ||
|      updateToCParent(); | |||
| } | } | ||
2025年8月27日 (三) 17:17的最新版本
/* 将为Timeless皮肤的用户加载此处的所有JavaScript  */
class HeadingTracker {
    constructor(rootElement) {
    	this.rootElement = rootElement
        this.headings = [];
        this.cacheExpired = true;
        this.init();
    }
    init() {
        this.collectHeadings();
        window.addEventListener('resize', () => this.cacheExpired = true);
        window.addEventListener('DOMContentLoaded', () => this.cacheExpired = true);
    }
    collectHeadings() {
        const rawHeadings = Array.from(this.rootElement.querySelectorAll('h1, h2, h3, h4, h5, h6'));
        this.headings = rawHeadings
            .map(heading => ({
                element: heading,
                top: heading.getBoundingClientRect().top + window.scrollY
            }))
            .sort((a, b) => a.top - b.top);
        this.cacheExpired = false;
    }
    getNearestHeadingAboveViewport() {
        if (this.cacheExpired) this.collectHeadings();
        const scrollY = window.scrollY + 60;
        let left = 0, right = this.headings.length - 1;
        let result = null;
        while (left <= right) {
            const mid = Math.floor((left + right) / 2);
            if (this.headings[mid].top <= scrollY) {
                result = this.headings[mid].element;
                left = mid + 1;
            } else {
                right = mid - 1;
            }
        }
        return result;
    }
}
class FloatingToC {
	constructor(toc) {
		this.element = toc;
		this.ignoreScrollUpdate = false;
	    
	    let tocTextList = Array.from(toc.querySelectorAll(".toctext"));
	    this.tocTexts = new Map(tocTextList.map(t => {
	    	let textContent = t.textContent
	    	t.parentElement.onclick = () => {
	    		this.ignoreScrollUpdate = true;
	    		this.setCurrentHeading(textContent);
	    	};
	    	return [textContent, t];
	    }));
		this.tracker = new HeadingTracker(
			document.getElementById('mw-content-text'));
		
		this.scrollHandler = this.onScroll.bind(this);
		window.addEventListener('scroll', this.scrollHandler);
	}
	
	remove() {
		window.removeEventListener('scroll', this.scrollHandler);
		this.element.remove();
	}
	
    setCurrentHeading(heading) {
    	const lastHeading = this.lastHeading;
    	if (heading == lastHeading) {
    		return;
    	}
    	if (lastHeading) {
    		let text = this.tocTexts.get(lastHeading)
    		if (text) text.classList.remove('current-heading');
    	}
    	this.lastHeading = heading;
    	if (heading) {
    		let text = this.tocTexts.get(heading)
    		if (text) text.classList.add('current-heading');
    	}
    };
	
	onScroll() {
		if (window.innerWidth < 1100) {
			return;
		}
		if (this.ignoreScrollUpdate) {
			this.ignoreScrollUpdate = false;
			return;
		}
	    let heading = this.tracker.getNearestHeadingAboveViewport();
	    this.setCurrentHeading(heading ? heading.textContent : null);
	}
}
let siteNavigation = document.getElementById('mw-site-navigation');
let floatingToC;
let toc = document.getElementById('toc');
let tocInitialParent = toc.parentNode;
const userAgent = navigator.userAgent;
const isSafari = userAgent.includes("Safari");
mw.hook('ve.newTarget').add(target => {
	if (target.constructor.static.name !== 'article') {
		return;
	}
	target.on('teardown', () => {
		var event = new CustomEvent("pageupdate");
		document.dispatchEvent(event);
	});
});
const updateToCParent = () => {
	let width = window.innerWidth;
	if (isSafari && width > 1354 || !isSafari && width > 1339) {
		if (toc.parentNode != siteNavigation) {
			toc.parentNode.removeChild(toc);
			siteNavigation.appendChild(toc);
		}
	}
	else {
		if (toc.parentNode == tocInitialParent || toc.parentNode == siteNavigation) {
			toc.parentNode.removeChild(toc);
			siteNavigation.parentNode.lastChild.before(toc);
		}
	}
}
window.addEventListener("resize", () => {
	if (floatingToC == null) {
		return;
	}
	updateToCParent();
});
document.addEventListener('pageupdate', () => {
	let newToC = document.querySelectorAll('.mw-parser-output > #toc')[0];
	if (newToC == null) {
		return;
	}
	if (floatingToC != null) {
		floatingToC.remove();
	}
	toc = newToC;
	floatingToC = new FloatingToC(toc);
	updateToCParent();
});
if (toc) {
    floatingToC = new FloatingToC(toc);
    updateToCParent();
}
