mirror of
https://gerrit.wikimedia.org/r/mediawiki/skins/Vector.git
synced 2024-11-12 09:21:11 +00:00
a274c9bcde
- Also renames 'header' to 'stickyHeaderElement' to be more clear Bug: T317661 Change-Id: Iad519036d4622ff28115abb7de544f69fe8eb597
355 lines
12 KiB
JavaScript
355 lines
12 KiB
JavaScript
// Enable Vector features limited to ES6 browse
|
|
const
|
|
limitedWidthToggle = require( './limitedWidthToggle.js' ),
|
|
searchToggle = require( './searchToggle.js' ),
|
|
stickyHeader = require( './stickyHeader.js' ),
|
|
scrollObserver = require( './scrollObserver.js' ),
|
|
initExperiment = require( './AB.js' ),
|
|
initSectionObserver = require( './sectionObserver.js' ),
|
|
initTableOfContents = require( './tableOfContents.js' ),
|
|
pinnableElement = require( './pinnableElement.js' ),
|
|
features = require( './features.js' ),
|
|
deferUntilFrame = require( './deferUntilFrame.js' ),
|
|
ABTestConfig = require( /** @type {string} */ ( './config.json' ) ).wgVectorWebABTestEnrollment || {},
|
|
STICKY_HEADER_VISIBLE_CLASS = 'vector-sticky-header-visible',
|
|
TOC_ID = 'vector-toc',
|
|
BODY_CONTENT_ID = 'bodyContent',
|
|
HEADLINE_SELECTOR = '.mw-headline',
|
|
TOC_SECTION_ID_PREFIX = 'toc-',
|
|
PAGE_TITLE_INTERSECTION_CLASS = 'vector-below-page-title';
|
|
|
|
const belowDesktopMedia = window.matchMedia( '(max-width: 999px)' );
|
|
|
|
/**
|
|
* @callback OnIntersection
|
|
* @param {HTMLElement} element The section that triggered the new intersection change.
|
|
*/
|
|
|
|
/**
|
|
* @ignore
|
|
* @param {Function} changeActiveSection
|
|
* @return {OnIntersection}
|
|
*/
|
|
const getHeadingIntersectionHandler = ( changeActiveSection ) => {
|
|
/**
|
|
* @param {HTMLElement} section
|
|
*/
|
|
return ( section ) => {
|
|
const headline = section.classList.contains( 'mw-body-content' ) ?
|
|
section :
|
|
section.querySelector( HEADLINE_SELECTOR );
|
|
if ( headline ) {
|
|
changeActiveSection( `${TOC_SECTION_ID_PREFIX}${headline.id}` );
|
|
}
|
|
};
|
|
};
|
|
|
|
/**
|
|
* Initialize sticky header AB tests and determine whether to show the sticky header
|
|
* based on which buckets the user is in.
|
|
*
|
|
* @typedef {Object} InitStickyHeaderABTests
|
|
* @property {boolean} disableEditIcons - Should the sticky header have an edit icon
|
|
* @property {boolean} showStickyHeader - Should the sticky header be shown
|
|
* @param {ABTestConfig} abConfig
|
|
* @param {boolean} isStickyHeaderFeatureAllowed and the user is logged in
|
|
* @param {function(ABTestConfig): initExperiment.WebABTest} getEnabledExperiment
|
|
* @return {InitStickyHeaderABTests}
|
|
*/
|
|
function initStickyHeaderABTests( abConfig, isStickyHeaderFeatureAllowed, getEnabledExperiment ) {
|
|
let showStickyHeader = isStickyHeaderFeatureAllowed,
|
|
stickyHeaderExperiment,
|
|
disableEditIcons = true;
|
|
|
|
// One of the sticky header AB tests is specified in the config
|
|
const abTestName = abConfig.name,
|
|
isStickyHeaderExperiment = abTestName === stickyHeader.STICKY_HEADER_EXPERIMENT_NAME ||
|
|
abTestName === stickyHeader.STICKY_HEADER_EDIT_EXPERIMENT_NAME;
|
|
|
|
// Determine if user is eligible for sticky header AB test
|
|
if (
|
|
isStickyHeaderFeatureAllowed && // The sticky header can be shown on the page
|
|
abConfig.enabled && // An AB test config is enabled
|
|
isStickyHeaderExperiment // The AB test is one of the sticky header experiments
|
|
) {
|
|
// If eligible, initialize the AB test
|
|
stickyHeaderExperiment = getEnabledExperiment( abConfig );
|
|
disableEditIcons = true;
|
|
|
|
// If running initial or edit AB test, show sticky header to treatment groups
|
|
// only. Unsampled and control buckets do not see sticky header.
|
|
if ( abTestName === stickyHeader.STICKY_HEADER_EXPERIMENT_NAME ||
|
|
abTestName === stickyHeader.STICKY_HEADER_EDIT_EXPERIMENT_NAME
|
|
) {
|
|
showStickyHeader = stickyHeaderExperiment.isInTreatmentBucket();
|
|
}
|
|
|
|
// If running edit-button AB test, the edit buttons in sticky header are shown
|
|
// to second treatment group only.
|
|
if ( abTestName === stickyHeader.STICKY_HEADER_EDIT_EXPERIMENT_NAME ) {
|
|
if ( stickyHeaderExperiment.isInTreatmentBucket( '1' ) ) {
|
|
disableEditIcons = true;
|
|
}
|
|
if ( stickyHeaderExperiment.isInTreatmentBucket( '2' ) ) {
|
|
disableEditIcons = false;
|
|
}
|
|
}
|
|
}
|
|
if ( !abConfig.enabled ) {
|
|
disableEditIcons = false;
|
|
}
|
|
|
|
return {
|
|
showStickyHeader,
|
|
disableEditIcons
|
|
};
|
|
}
|
|
|
|
/*
|
|
* Updates TOC's location in the DOM (in sidebar or sticky header)
|
|
* depending on if the TOC is collapsed and if the sticky header is visible
|
|
*
|
|
* @return {void}
|
|
*/
|
|
const updateTocLocation = () => {
|
|
const isPinned = features.isEnabled( 'toc-pinned' );
|
|
const isStickyHeaderVisible = document.body.classList.contains( STICKY_HEADER_VISIBLE_CLASS );
|
|
const isBelowDesktop = belowDesktopMedia.matches;
|
|
|
|
const pinnedContainerId = 'vector-toc-pinned-container';
|
|
const stickyHeaderUnpinnedContainerId = 'vector-sticky-header-toc-unpinned-container';
|
|
const pageTitlebarUnpinnedContainerId = 'vector-page-titlebar-toc-unpinned-container';
|
|
|
|
let newContainerId = '';
|
|
if ( isPinned ) {
|
|
if ( isBelowDesktop ) {
|
|
// Automatically move the ToC into the page titlebar when pinned on smaller resolutions
|
|
newContainerId = pageTitlebarUnpinnedContainerId;
|
|
} else {
|
|
newContainerId = pinnedContainerId;
|
|
}
|
|
} else {
|
|
if ( isStickyHeaderVisible && !isBelowDesktop ) {
|
|
newContainerId = stickyHeaderUnpinnedContainerId;
|
|
} else {
|
|
newContainerId = pageTitlebarUnpinnedContainerId;
|
|
}
|
|
}
|
|
|
|
pinnableElement.movePinnableElement( TOC_ID, newContainerId );
|
|
};
|
|
|
|
/**
|
|
* Return the computed value of the scroll intersection threshold,
|
|
* which is based off of the `scroll-margin-top` CSS property of the document element.
|
|
*
|
|
* T317661: Computed value results in a threshold 100px from the
|
|
* top of the viewport or bottom of the sticky header
|
|
*
|
|
* @return {number} Value of scroll-margin-top OR zero if falsy.
|
|
*/
|
|
function getScrollIntersectionThreshold() {
|
|
const documentStyles = getComputedStyle( document.documentElement );
|
|
const scrollPaddingTopString = documentStyles.getPropertyValue( 'scroll-padding-top' );
|
|
return ( parseInt( scrollPaddingTopString, 10 ) || 50 ) + 50;
|
|
}
|
|
|
|
/**
|
|
* @param {HTMLElement|null} tocElement
|
|
* @param {HTMLElement|null} bodyContent
|
|
* @param {initSectionObserver} initSectionObserverFn
|
|
* @return {tableOfContents|null}
|
|
*/
|
|
const setupTableOfContents = ( tocElement, bodyContent, initSectionObserverFn ) => {
|
|
if ( !(
|
|
tocElement &&
|
|
bodyContent
|
|
) ) {
|
|
return null;
|
|
}
|
|
|
|
const tableOfContents = initTableOfContents( {
|
|
container: tocElement,
|
|
onHeadingClick: ( id ) => {
|
|
|
|
// eslint-disable-next-line no-use-before-define
|
|
sectionObserver.pause();
|
|
|
|
tableOfContents.expandSection( id );
|
|
tableOfContents.changeActiveSection( id );
|
|
|
|
// T297614: We want the link that the user has clicked inside the TOC to
|
|
// be "active" (e.g. bolded) regardless of whether the browser's scroll
|
|
// position corresponds to that section. Therefore, we need to temporarily
|
|
// ignore section observer until the browser has finished scrolling to the
|
|
// section (if needed).
|
|
//
|
|
// However, because the scroll event happens asyncronously after the user
|
|
// clicks on a link and may not even happen at all (e.g. the user has
|
|
// scrolled all the way to the bottom and clicks a section that is already
|
|
// in the viewport), determining when we should resume section observer is
|
|
// a bit tricky.
|
|
//
|
|
// Because a scroll event may not even be triggered after clicking the
|
|
// link, we instead allow the browser to perform a maximum number of
|
|
// repaints before resuming sectionObserver. Per T297614#7687656, Firefox
|
|
// 97.0 wasn't consistently activating the table of contents section that
|
|
// the user clicked even after waiting 2 frames. After further
|
|
// investigation, it sometimes waits up to 3 frames before painting the
|
|
// new scroll position so we have that as the limit.
|
|
//
|
|
// eslint-disable-next-line no-use-before-define
|
|
deferUntilFrame( () => sectionObserver.resume(), 3 );
|
|
},
|
|
onToggleClick: ( id ) => {
|
|
tableOfContents.toggleExpandSection( id );
|
|
},
|
|
onTogglePinned: () => {
|
|
updateTocLocation();
|
|
pinnableElement.setFocusAfterToggle( TOC_ID );
|
|
}
|
|
} );
|
|
const headingSelector = [
|
|
'h1', 'h2', 'h3', 'h4', 'h5', 'h6'
|
|
].map( ( tag ) => `.mw-parser-output ${tag}` ).join( ',' );
|
|
const elements = () => bodyContent.querySelectorAll( `${headingSelector}, .mw-body-content` );
|
|
|
|
const sectionObserver = initSectionObserverFn( {
|
|
elements: elements(),
|
|
topMargin: getScrollIntersectionThreshold(),
|
|
onIntersection: getHeadingIntersectionHandler( tableOfContents.changeActiveSection )
|
|
} );
|
|
const updateElements = () => {
|
|
sectionObserver.resume();
|
|
sectionObserver.setElements( elements() );
|
|
};
|
|
mw.hook( 've.activationStart' ).add( () => {
|
|
sectionObserver.pause();
|
|
} );
|
|
// @ts-ignore
|
|
mw.hook( 'wikipage.tableOfContents' ).add( function ( sections ) {
|
|
tableOfContents.reloadTableOfContents( sections ).then( function () {
|
|
mw.hook( 'wikipage.tableOfContents.vector' ).fire( sections );
|
|
updateElements();
|
|
} );
|
|
} );
|
|
mw.hook( 've.deactivationComplete' ).add( updateElements );
|
|
return tableOfContents;
|
|
};
|
|
|
|
/**
|
|
* @return {void}
|
|
*/
|
|
const main = () => {
|
|
const isIntersectionObserverSupported = 'IntersectionObserver' in window;
|
|
|
|
limitedWidthToggle();
|
|
// Initialize the search toggle for the main header only. The sticky header
|
|
// toggle is initialized after Codex search loads.
|
|
const searchToggleElement = document.querySelector( '.mw-header .search-toggle' );
|
|
if ( searchToggleElement ) {
|
|
searchToggle( searchToggleElement );
|
|
}
|
|
|
|
//
|
|
// Pinnable elements
|
|
//
|
|
pinnableElement.initPinnableElement();
|
|
|
|
//
|
|
// Table of contents
|
|
//
|
|
const tocElement = document.getElementById( TOC_ID );
|
|
const bodyContent = document.getElementById( BODY_CONTENT_ID );
|
|
|
|
const isToCUpdatingAllowed = isIntersectionObserverSupported &&
|
|
window.requestAnimationFrame;
|
|
const tableOfContents = isToCUpdatingAllowed ?
|
|
setupTableOfContents( tocElement, bodyContent, initSectionObserver ) : null;
|
|
|
|
//
|
|
// Sticky header
|
|
//
|
|
const
|
|
stickyHeaderElement = document.getElementById( stickyHeader.STICKY_HEADER_ID ),
|
|
stickyIntersection = document.getElementById( stickyHeader.FIRST_HEADING_ID ),
|
|
userLinksDropdown = document.getElementById( stickyHeader.USER_LINKS_DROPDOWN_ID ),
|
|
allowedNamespace = stickyHeader.isAllowedNamespace( mw.config.get( 'wgNamespaceNumber' ) ),
|
|
allowedAction = stickyHeader.isAllowedAction( mw.config.get( 'wgAction' ) );
|
|
|
|
const isStickyHeaderAllowed =
|
|
!!stickyHeaderElement &&
|
|
!!stickyIntersection &&
|
|
!!userLinksDropdown &&
|
|
allowedNamespace &&
|
|
allowedAction &&
|
|
isIntersectionObserverSupported;
|
|
|
|
const { showStickyHeader, disableEditIcons } = initStickyHeaderABTests(
|
|
ABTestConfig,
|
|
isStickyHeaderAllowed && !mw.user.isAnon(),
|
|
( config ) => initExperiment(
|
|
Object.assign( {}, config, { token: mw.user.getId() } )
|
|
)
|
|
);
|
|
|
|
// Set up intersection observer for page title
|
|
// Used to show/hide sticky header and add class used by collapsible TOC (T307900)
|
|
const observer = scrollObserver.initScrollObserver(
|
|
() => {
|
|
if ( isStickyHeaderAllowed && showStickyHeader ) {
|
|
stickyHeader.show();
|
|
updateTocLocation();
|
|
}
|
|
document.body.classList.add( PAGE_TITLE_INTERSECTION_CLASS );
|
|
if ( tableOfContents ) {
|
|
tableOfContents.updateTocToggleStyles( true );
|
|
}
|
|
scrollObserver.firePageTitleScrollHook( 'down' );
|
|
},
|
|
() => {
|
|
if ( isStickyHeaderAllowed && showStickyHeader ) {
|
|
stickyHeader.hide();
|
|
updateTocLocation();
|
|
}
|
|
document.body.classList.remove( PAGE_TITLE_INTERSECTION_CLASS );
|
|
if ( tableOfContents ) {
|
|
tableOfContents.updateTocToggleStyles( false );
|
|
}
|
|
scrollObserver.firePageTitleScrollHook( 'up' );
|
|
}
|
|
);
|
|
|
|
// Handle toc location when sticky header is hidden on lower viewports
|
|
belowDesktopMedia.onchange = () => {
|
|
updateTocLocation();
|
|
};
|
|
|
|
updateTocLocation();
|
|
|
|
if ( !showStickyHeader ) {
|
|
stickyHeader.hide();
|
|
}
|
|
|
|
if ( isStickyHeaderAllowed && showStickyHeader ) {
|
|
stickyHeader.initStickyHeader( {
|
|
header: stickyHeaderElement,
|
|
userLinksDropdown,
|
|
observer,
|
|
stickyIntersection,
|
|
disableEditIcons
|
|
} );
|
|
} else if ( stickyIntersection ) {
|
|
observer.observe( stickyIntersection );
|
|
}
|
|
};
|
|
|
|
module.exports = {
|
|
main,
|
|
test: {
|
|
setupTableOfContents,
|
|
initStickyHeaderABTests,
|
|
getHeadingIntersectionHandler
|
|
}
|
|
};
|