mediawiki-skins-Vector/resources/skins.vector.js/limitedWidthToggle.js
Jon Robson e5bf8adad7 Limited width uses new client preferences system
* Update classes to use clientpref-1 and clientpref-0 suffix for limited width
I've limited this to the only client preference for now to reduce
risk.
* For cached HTML retain existing CSS rules, and continue saving
a cookie
* Migrate cookie if found for newly generated pages. This will be
to ensure the old cookie and new cookie are in sync (this should be
a one time operation)

Depends-On: I1e635f843ac9b2f248b1f7618134598e80291b38
Bug: T341641
Change-Id: I120f8f7114b33d2cfbd1c3c57ebf41f8b2d7fec4
2023-08-04 21:31:21 +00:00

104 lines
3.6 KiB
JavaScript

const features = require( './features.js' );
const popupNotification = require( './popupNotification.js' );
const LIMITED_WIDTH_FEATURE_NAME = 'limited-width';
const TOGGLE_ID = 'toggleWidth';
/**
* Sets data attribute for click tracking purposes.
*
* @param {HTMLElement} toggleBtn
*/
function setDataAttribute( toggleBtn ) {
toggleBtn.dataset.eventName = features.isEnabled( LIMITED_WIDTH_FEATURE_NAME ) ?
'limited-width-toggle-off' : 'limited-width-toggle-on';
}
/**
* Gets appropriate popup text based off the limited width feature flag
*
* @return {string}
*/
function getPopupText() {
const label = features.isEnabled( LIMITED_WIDTH_FEATURE_NAME ) ?
'vector-limited-width-toggle-off-popup' : 'vector-limited-width-toggle-on-popup';
// possible messages:
// * vector-limited-width-toggle-off-popup
// * vector-limited-width-toggle-on-popup
return mw.msg( label );
}
/**
* adds a toggle button
*/
function init() {
const settings = /** @type {HTMLElement|null} */ ( document.querySelector( '.vector-settings' ) );
const toggle = /** @type {HTMLElement|null} */ ( document.querySelector( '.vector-limited-width-toggle' ) );
const toggleIcon = /** @type {HTMLElement|null} */ ( document.querySelector( '.vector-limited-width-toggle .vector-icon' ) );
const cookie = mw.cookie.get( 'mwclientprefs' );
// @ts-ignore https://github.com/wikimedia/typescript-types/pull/42
const existingValue = mw.user.clientPrefs.get( 'vector-feature-limited-width' );
// On a cached page existingValue will return false.
// If the cookie is set then we need to add the appropriate class to the page
// so that the toggle works correctly.
if ( !existingValue ) {
document.documentElement.classList.add(
`vector-feature-limited-width-clientpref-${cookie ? '0' : '1'}`
);
document.documentElement.classList.remove(
'vector-feature-limited-width-disabled',
'vector-feature-limited-width-enabled'
);
}
// Sync cookie value with clientPrefs if the two are misaligned.
if ( cookie && existingValue === '1' ) {
// @ts-ignore https://github.com/wikimedia/typescript-types/pull/42
mw.user.clientPrefs.set( 'vector-feature-limited-width', '0' );
// Update the display.
document.documentElement.classList.remove(
'vector-feature-limited-width-clientpref-1',
'vector-feature-limited-width-enabled'
);
document.documentElement.classList.add(
'vector-feature-limited-width-clientpref-0'
);
}
if ( !settings || !toggle || !toggleIcon ) {
return;
}
setDataAttribute( toggle );
/**
* @param {string} id this allows us to group notifications making sure only one is visible
* at any given time. All existing popups associated with ID will be removed.
* @param {number|false} timeout
*/
const showPopup = ( id, timeout = 4000 ) => {
popupNotification.add( settings, getPopupText(), id, [], timeout )
.then( ( popupWidget ) => {
if ( popupWidget ) {
popupNotification.show( popupWidget, timeout );
}
} );
};
toggle.addEventListener( 'click', function () {
const isLimitedWidth = features.isEnabled( LIMITED_WIDTH_FEATURE_NAME );
const oldIcon = isLimitedWidth ? 'fullScreen' : 'exitFullscreen';
const newIcon = isLimitedWidth ? 'exitFullscreen' : 'fullScreen';
features.toggle( LIMITED_WIDTH_FEATURE_NAME );
setDataAttribute( toggle );
toggleIcon.classList.remove( `mw-ui-icon-wikimedia-${oldIcon}` );
toggleIcon.classList.add( `mw-ui-icon-wikimedia-${newIcon}` );
window.dispatchEvent( new Event( 'resize' ) );
if ( isLimitedWidth ) {
// Now is full width, show notification
showPopup( TOGGLE_ID );
}
} );
}
module.exports = init;