( function ( $, mw ) { /** * Check whether the Navigation Popups gadget is enabled * * @return {boolean} */ function isNavigationPopupsGadgetEnabled() { return window.pg !== undefined; } /** * `mouseleave` and `blur` events handler for links that are eligible for * popups, but when Popups are disabled. * * @param {Object} event */ function onLinkAbandon( event ) { var $this = $( this ), data = event.data || {}; $this.off( 'mouseleave blur', onLinkAbandon ); if ( data.dwellStartTime && data.linkInteractionToken && mw.now() - data.dwellStartTime >= 250 ) { mw.track( 'ext.popups.schemaPopups', { pageTitleHover: $this.attr( 'title' ), action: 'dwelledButAbandoned', totalInteractionTime: Math.round( mw.now() - data.dwellStartTime ), linkInteractionToken: data.linkInteractionToken } ); } } /** * `click` event handler for links that are eligible for popups, but when * Popups are disabled. * * @param {Object} event */ function onLinkClick( event ) { var $this = $( this ), action = mw.popups.getAction( event ), href = $this.attr( 'href' ), data = event.data || {}; $this.off( 'click', onLinkClick ); mw.track( 'ext.popups.schemaPopups', { pageTitleHover: $this.attr( 'title' ), action: action, totalInteractionTime: Math.round( mw.now() - data.dwellStartTime ), linkInteractionToken: data.linkInteractionToken } ); if ( action === 'opened in same tab' ) { event.preventDefault(); window.location.href = href; } } mw.popups.enabled = mw.popups.getEnabledState(); /** * Creates the SVG mask used to create the * the triangle pointer on popups with images * * @method createSVGMask */ mw.popups.createSVGMask = function () { $( '
' ) .attr( 'id', 'mwe-popups-svg' ) .appendTo( document.body ) .html( '' + '' + '' + '' + '' + '' + '' + '' + '' + '' + '' + '' + '' + '' + '' + '' ); return true; }; /** * Create the element that holds the popups * * @method createPopupElement */ mw.popups.createPopupElement = function () { mw.popups.$popup = $( '
' ) .attr( { 'class': 'mwe-popups', role: 'tooltip', 'aria-hidden': 'true' } ) .appendTo( document.body ); }; /** * Checks if the user is scrolling, sets to false on mousemove * * @method checkScroll */ mw.popups.checkScroll = function () { $( window ).on( 'scroll', function () { mw.popups.scrolled = true; } ); $( window ).on( 'mousemove', function () { mw.popups.scrolled = false; } ); }; mw.hook( 'wikipage.content' ).add( function ( $content ) { var $elements, dwellStartTime, linkInteractionToken; mw.popups.$content = $content; $elements = mw.popups.selectPopupElements(); $elements.on( 'mouseenter focus', function ( event ) { var $link = $( this ); // Only enable Popups when the Navigation popups gadget is not enabled if ( !isNavigationPopupsGadgetEnabled() && mw.popups.enabled ) { if ( mw.popups.scrolled ) { return; } mw.popups.removeTooltips( $link ); mw.popups.render.render( $link, event, mw.now(), mw.popups.getRandomToken() ); } else { // Cache the hover start time and link interaction token for a later use dwellStartTime = mw.now(); linkInteractionToken = mw.popups.getRandomToken(); $link // We are passing the same data, rather than a shared object, into two different functions. // The reason is that we don't want one function to change the data and // have a side-effect on the other function's data. .on( 'mouseleave blur', { dwellStartTime: dwellStartTime, linkInteractionToken: linkInteractionToken }, onLinkAbandon ) .on( 'click', { dwellStartTime: dwellStartTime, linkInteractionToken: linkInteractionToken }, onLinkClick ); } } ); } ); function initPopups() { mw.popups.checkScroll(); mw.popups.createSVGMask(); mw.popups.createPopupElement(); } $( function () { var jStorage = mw.storage.get( 'jStorage' ); // FIXME: Can be removed a month from jStorage removal. Not perfect but should work in most cases. if ( jStorage && jStorage.indexOf( 'popups-enabled' ) > 0 ) { // slower loading popups when in transition mode mw.loader.using( 'jquery.jStorage' ).done( function () { var val = $.jStorage.get( 'mwe-popups-enabled' ); // delete jStorage key $.jStorage.deleteKey( 'mwe-popups-enabled' ); if ( val === 'false' ) { // copy over mw.storage.set( 'mwe-popups-enabled', '0' ); } else { if ( val ) { mw.storage.set( 'mwe-popups-enabled', '1' ); } initPopups(); } } ); } else if ( mw.popups.enabled ) { initPopups(); } mw.track( 'ext.popups.schemaPopups', { action: 'pageLoaded' } ); } ); } )( jQuery, mediaWiki );