2024-05-24 12:31:30 +00:00
|
|
|
let newTopicButton, ledeSectionDialog;
|
|
|
|
let viewportScrollContainer = null;
|
|
|
|
let wasKeyboardOpen = null;
|
|
|
|
let initialClientHeight = null;
|
2022-11-02 13:47:35 +00:00
|
|
|
// Copied from ve.init.Platform.static.isIos
|
2024-05-24 12:31:30 +00:00
|
|
|
const isIos = /ipad|iphone|ipod/i.test( navigator.userAgent );
|
2022-11-02 13:47:35 +00:00
|
|
|
|
|
|
|
$( document.body ).toggleClass( 'ext-discussiontools-init-ios', isIos );
|
|
|
|
|
|
|
|
function onViewportChange() {
|
2024-05-24 12:20:50 +00:00
|
|
|
let isKeyboardOpen;
|
2022-11-02 13:47:35 +00:00
|
|
|
|
|
|
|
if ( isIos ) {
|
|
|
|
isKeyboardOpen = visualViewport.height < viewportScrollContainer.clientHeight;
|
|
|
|
} else {
|
|
|
|
// TODO: Support orientation changes?
|
|
|
|
isKeyboardOpen = viewportScrollContainer.clientHeight < initialClientHeight;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( isKeyboardOpen !== wasKeyboardOpen ) {
|
|
|
|
$( document.body ).toggleClass( 'ext-discussiontools-init-virtual-keyboard-open', isKeyboardOpen );
|
|
|
|
}
|
|
|
|
|
|
|
|
wasKeyboardOpen = isKeyboardOpen;
|
|
|
|
}
|
2022-10-11 12:30:37 +00:00
|
|
|
|
2023-10-11 12:04:39 +00:00
|
|
|
function init( $container ) {
|
2023-04-03 19:41:42 +00:00
|
|
|
// For compatibility with MobileWebUIActionsTracking logging (T295490)
|
2022-10-11 12:30:37 +00:00
|
|
|
$container.find( '.section-heading' ).attr( 'data-event-name', 'talkpage.section' );
|
|
|
|
|
2022-11-02 13:47:35 +00:00
|
|
|
// Keyboard body classes
|
|
|
|
if ( !viewportScrollContainer && window.visualViewport ) {
|
|
|
|
viewportScrollContainer = OO.ui.Element.static.getClosestScrollableContainer( document.body );
|
|
|
|
initialClientHeight = viewportScrollContainer.clientHeight;
|
2024-05-24 12:20:50 +00:00
|
|
|
const onViewportChangeThrottled = OO.ui.throttle( onViewportChange, 100 );
|
2022-11-02 13:47:35 +00:00
|
|
|
$( visualViewport ).on( 'resize', onViewportChangeThrottled );
|
|
|
|
}
|
|
|
|
|
2022-10-11 12:30:37 +00:00
|
|
|
// Mobile overflow menu
|
2022-09-27 14:48:25 +00:00
|
|
|
|
2024-05-24 12:20:50 +00:00
|
|
|
const $ledeContent = $container.find( '.mf-section-0' ).children( ':not( .ext-discussiontools-emptystate )' )
|
2022-11-29 17:34:54 +00:00
|
|
|
// On non-existent pages MobileFrontend wrapping isn't there
|
|
|
|
.add( $container.find( '.mw-talkpageheader' ) );
|
2024-05-24 12:20:50 +00:00
|
|
|
const $ledeButton = $container.find( '.ext-discussiontools-init-lede-button' );
|
2022-09-27 14:48:25 +00:00
|
|
|
if ( $ledeButton.length ) {
|
2024-05-24 12:20:50 +00:00
|
|
|
const windowManager = OO.ui.getWindowManager();
|
2022-09-27 14:48:25 +00:00
|
|
|
if ( !ledeSectionDialog ) {
|
2024-05-24 12:20:50 +00:00
|
|
|
const LedeSectionDialog = require( './LedeSectionDialog.js' );
|
2022-09-27 14:48:25 +00:00
|
|
|
ledeSectionDialog = new LedeSectionDialog();
|
|
|
|
windowManager.addWindows( [ ledeSectionDialog ] );
|
|
|
|
}
|
|
|
|
|
|
|
|
// Lede section popup
|
2024-04-18 18:37:58 +00:00
|
|
|
OO.ui.infuse( $ledeButton ).on( 'click', () => {
|
|
|
|
mw.loader.using( 'oojs-ui-windows' ).then( () => {
|
2022-09-27 14:48:25 +00:00
|
|
|
windowManager.openWindow( 'ledeSection', { $content: $ledeContent } );
|
2022-12-15 17:27:26 +00:00
|
|
|
mw.track( 'webuiactions_log.click', 'lede-button' );
|
2022-09-27 14:48:25 +00:00
|
|
|
} );
|
|
|
|
} );
|
2023-05-04 16:25:40 +00:00
|
|
|
mw.track( 'webuiactions_log.show', 'lede-button' );
|
2022-09-27 14:48:25 +00:00
|
|
|
}
|
2022-11-01 13:55:45 +00:00
|
|
|
|
2023-09-17 20:24:40 +00:00
|
|
|
// eslint-disable-next-line no-jquery/no-global-selector
|
2024-05-24 12:20:50 +00:00
|
|
|
const $newTopicWrapper = $( '.ext-discussiontools-init-new-topic' );
|
2023-09-17 20:24:40 +00:00
|
|
|
|
2022-11-01 13:55:45 +00:00
|
|
|
if (
|
|
|
|
!newTopicButton &&
|
|
|
|
// eslint-disable-next-line no-jquery/no-global-selector
|
|
|
|
$( '.ext-discussiontools-init-new-topic-button' ).length
|
|
|
|
) {
|
|
|
|
// eslint-disable-next-line no-jquery/no-global-selector
|
|
|
|
newTopicButton = OO.ui.infuse( $( '.ext-discussiontools-init-new-topic-button' ) );
|
2023-04-03 19:41:42 +00:00
|
|
|
// For compatibility with MobileWebUIActionsTracking logging (T295490)
|
2022-11-01 13:55:45 +00:00
|
|
|
newTopicButton.$element.attr( 'data-event-name', 'talkpage.add-topic' );
|
2024-05-24 12:20:50 +00:00
|
|
|
const $scrollContainer = $( OO.ui.Element.static.getClosestScrollableContainer( document.body ) );
|
|
|
|
const $scrollListener = $scrollContainer.is( 'html, body' ) ? $( OO.ui.Element.static.getWindow( $scrollContainer[ 0 ] ) ) : $scrollContainer;
|
|
|
|
let lastScrollTop = $scrollContainer.scrollTop();
|
|
|
|
let wasScrollDown = null;
|
|
|
|
const $body = $( document.body );
|
2023-08-09 15:43:39 +00:00
|
|
|
// This block of code is only run once, so we don't need to remove this listener ever
|
2024-04-18 18:37:58 +00:00
|
|
|
$scrollListener[ 0 ].addEventListener( 'scroll', OO.ui.throttle( () => {
|
2022-11-29 14:29:50 +00:00
|
|
|
// Round negative values up to 0 to ignore iOS scroll bouncing (T323400)
|
2024-05-24 12:20:50 +00:00
|
|
|
const scrollTop = Math.max( $scrollContainer.scrollTop(), 0 );
|
|
|
|
const isScrollDown = scrollTop > lastScrollTop;
|
2022-11-01 13:55:45 +00:00
|
|
|
if ( isScrollDown !== wasScrollDown ) {
|
|
|
|
if ( !isScrollDown ) {
|
2023-02-01 18:45:34 +00:00
|
|
|
$newTopicWrapper.css( 'transition', 'none' );
|
2022-11-01 13:55:45 +00:00
|
|
|
}
|
|
|
|
$body.removeClass( [ 'ext-discussiontools-init-new-topic-closed', 'ext-discussiontools-init-new-topic-opened' ] );
|
2024-04-18 18:37:58 +00:00
|
|
|
requestAnimationFrame( () => {
|
2023-02-01 18:45:34 +00:00
|
|
|
$newTopicWrapper.css( 'transition', '' );
|
2022-11-01 13:55:45 +00:00
|
|
|
$body.addClass( isScrollDown ? 'ext-discussiontools-init-new-topic-close' : 'ext-discussiontools-init-new-topic-open' );
|
2024-04-18 18:37:58 +00:00
|
|
|
setTimeout( () => {
|
2022-11-01 13:55:45 +00:00
|
|
|
$body.removeClass( [ 'ext-discussiontools-init-new-topic-close', 'ext-discussiontools-init-new-topic-open' ] );
|
|
|
|
$body.addClass( isScrollDown ? 'ext-discussiontools-init-new-topic-closed' : 'ext-discussiontools-init-new-topic-opened' );
|
|
|
|
}, 250 );
|
|
|
|
} );
|
|
|
|
}
|
|
|
|
|
2024-05-24 12:20:50 +00:00
|
|
|
const observer = new IntersectionObserver(
|
2024-04-18 18:37:58 +00:00
|
|
|
( ( entries ) => {
|
2023-02-01 18:45:34 +00:00
|
|
|
$newTopicWrapper.toggleClass( 'ext-discussiontools-init-new-topic-pinned', entries[ 0 ].intersectionRatio === 1 );
|
2024-04-18 18:37:58 +00:00
|
|
|
} ),
|
2022-11-01 13:55:45 +00:00
|
|
|
{ threshold: [ 1 ] }
|
|
|
|
);
|
|
|
|
|
2023-02-01 18:45:34 +00:00
|
|
|
observer.observe( $newTopicWrapper[ 0 ] );
|
2022-11-01 13:55:45 +00:00
|
|
|
|
|
|
|
lastScrollTop = scrollTop;
|
|
|
|
wasScrollDown = isScrollDown;
|
2023-08-09 15:43:39 +00:00
|
|
|
}, 200 ), { passive: true } );
|
2022-11-01 13:55:45 +00:00
|
|
|
}
|
2023-02-01 16:46:39 +00:00
|
|
|
|
|
|
|
// Tweak to prevent our footer buttons from overlapping Minerva skin elements (T328452).
|
|
|
|
// TODO: It would be more elegant to do this in just CSS somehow.
|
|
|
|
// BEWARE: I have wasted 4 hours here trying to make that happen. The elements are not nested in a
|
|
|
|
// helpful way, and moving them around tends to break the stickiness of the "Add topic" button.
|
2024-01-04 18:33:42 +00:00
|
|
|
/* eslint-disable no-jquery/no-global-selector */
|
2023-02-01 16:46:39 +00:00
|
|
|
if (
|
2023-02-07 01:56:05 +00:00
|
|
|
$( '.catlinks' ).filter( '[data-mw="interface"]' ).length ||
|
|
|
|
$( '#page-secondary-actions' ).children().length ||
|
2023-02-01 16:46:39 +00:00
|
|
|
$( '.return-link' ).length
|
|
|
|
) {
|
2023-09-17 20:24:40 +00:00
|
|
|
$newTopicWrapper.addClass( 'ext-discussiontools-init-button-notFlush' );
|
2023-02-01 16:46:39 +00:00
|
|
|
}
|
|
|
|
/* eslint-enable no-jquery/no-global-selector */
|
2022-10-11 12:30:37 +00:00
|
|
|
}
|
|
|
|
|
2023-09-28 11:00:36 +00:00
|
|
|
// Handler for "edit" link in overflow menu, only setup once as the hook is global
|
2024-04-18 18:37:58 +00:00
|
|
|
mw.hook( 'discussionToolsOverflowMenuOnChoose' ).add( ( id, menuItem, threadItem ) => {
|
2023-09-28 11:00:36 +00:00
|
|
|
if ( id === 'edit' ) {
|
|
|
|
// Click the hidden section-edit link
|
2023-12-15 16:22:52 +00:00
|
|
|
$( threadItem.getRange().commonAncestorContainer )
|
2023-10-05 17:51:29 +00:00
|
|
|
.closest( '.ext-discussiontools-init-section' ).find( '.mw-editsection > a' ).trigger( 'click' );
|
2023-09-28 11:00:36 +00:00
|
|
|
}
|
|
|
|
} );
|
|
|
|
|
2023-04-13 23:27:17 +00:00
|
|
|
/**
|
|
|
|
* Close the lede section dialog if it is open.
|
|
|
|
*
|
|
|
|
* @return {jQuery.Promise} Promise resolved when the dialog is closed (or if it wasn't open)
|
|
|
|
*/
|
|
|
|
function closeLedeSectionDialog() {
|
|
|
|
if ( ledeSectionDialog && ledeSectionDialog.isOpened() ) {
|
|
|
|
return ledeSectionDialog.close().closed;
|
|
|
|
}
|
|
|
|
return $.Deferred().resolve();
|
|
|
|
}
|
|
|
|
|
2022-10-11 12:30:37 +00:00
|
|
|
module.exports = {
|
2023-04-13 23:27:17 +00:00
|
|
|
init: init,
|
|
|
|
closeLedeSectionDialog: closeLedeSectionDialog
|
2022-10-11 12:30:37 +00:00
|
|
|
};
|