2017-05-26 17:35:07 +00:00
|
|
|
/**
|
|
|
|
* @module popups
|
|
|
|
*/
|
|
|
|
|
2017-07-28 17:32:46 +00:00
|
|
|
import * as Redux from 'redux';
|
|
|
|
import * as ReduxThunk from 'redux-thunk';
|
|
|
|
|
2019-01-31 11:05:57 +00:00
|
|
|
import createPagePreviewGateway from './gateway/page';
|
2018-12-12 18:37:19 +00:00
|
|
|
import createReferenceGateway from './gateway/reference';
|
2017-07-28 17:32:46 +00:00
|
|
|
import createUserSettings from './userSettings';
|
|
|
|
import createPreviewBehavior from './previewBehavior';
|
2018-03-08 20:57:22 +00:00
|
|
|
import createSettingsDialogRenderer from './ui/settingsDialogRenderer';
|
2017-07-28 17:32:46 +00:00
|
|
|
import registerChangeListener from './changeListener';
|
2021-03-31 08:51:54 +00:00
|
|
|
import createIsPagePreviewsEnabled from './isPagePreviewsEnabled';
|
2017-07-28 17:32:46 +00:00
|
|
|
import { fromElement as titleFromElement } from './title';
|
2022-07-12 22:48:46 +00:00
|
|
|
import { init as rendererInit, registerPreviewUI, createPagePreview,
|
|
|
|
createDisambiguationPreview,
|
|
|
|
createReferencePreview
|
|
|
|
} from './ui/renderer';
|
2017-07-28 17:32:46 +00:00
|
|
|
import createExperiments from './experiments';
|
|
|
|
import { isEnabled as isStatsvEnabled } from './instrumentation/statsv';
|
|
|
|
import changeListeners from './changeListeners';
|
|
|
|
import * as actions from './actions';
|
|
|
|
import reducers from './reducers';
|
2017-08-09 09:14:52 +00:00
|
|
|
import createMediaWikiPopupsObject from './integrations/mwpopups';
|
2022-07-12 22:48:46 +00:00
|
|
|
import { previewTypes, getPreviewType,
|
|
|
|
registerModel,
|
2022-12-12 20:08:14 +00:00
|
|
|
isAnythingEligible, findNearestEligibleTarget } from './preview/model';
|
2020-12-02 15:48:33 +00:00
|
|
|
import isReferencePreviewsEnabled from './isReferencePreviewsEnabled';
|
2021-03-09 17:21:48 +00:00
|
|
|
import setUserConfigFlags from './setUserConfigFlags';
|
2022-07-12 22:48:46 +00:00
|
|
|
import { registerGatewayForPreviewType, getGatewayForPreviewType } from './gateway';
|
2023-05-09 15:11:44 +00:00
|
|
|
import { initReferencePreviewsInstrumentation } from './instrumentation/referencePreviews';
|
2022-07-12 22:48:46 +00:00
|
|
|
|
2020-06-09 06:52:42 +00:00
|
|
|
const EXCLUDED_LINK_SELECTORS = [
|
2019-10-17 08:51:49 +00:00
|
|
|
'.extiw',
|
2022-07-12 22:48:46 +00:00
|
|
|
// ignore links that point to the same article
|
|
|
|
'.mw-selflink',
|
2019-10-17 08:51:49 +00:00
|
|
|
'.image',
|
|
|
|
'.new',
|
|
|
|
'.internal',
|
|
|
|
'.external',
|
|
|
|
'.mw-cite-backlink a',
|
|
|
|
'.oo-ui-buttonedElement-button',
|
2020-08-07 14:32:18 +00:00
|
|
|
'.ve-ce-surface a', // T259889
|
2023-01-11 16:36:51 +00:00
|
|
|
'.cancelLink a',
|
|
|
|
// T198652: lists to hash fragments are ignored.
|
|
|
|
// Note links that include the path will still trigger a hover,
|
|
|
|
// e.g. <a href="Foo#foo"> will trigger a preview but <a href="#foo"> will not.
|
|
|
|
// This is intentional behaviour that will not be handled by page previews, to avoid
|
|
|
|
// introducing complex behaviour. If a link must include the path it should make use of
|
|
|
|
// the .mw-selflink-fragment class.
|
|
|
|
'.mw-selflink-fragment',
|
|
|
|
'[href^="#"]'
|
2019-10-17 08:51:49 +00:00
|
|
|
];
|
2017-02-14 20:19:12 +00:00
|
|
|
|
2017-04-27 12:53:15 +00:00
|
|
|
/**
|
2017-06-14 11:00:01 +00:00
|
|
|
* @typedef {Function} EventTracker
|
2017-04-27 12:53:15 +00:00
|
|
|
*
|
2017-06-14 11:00:01 +00:00
|
|
|
* An analytics event tracker, i.e. `mw.track`.
|
|
|
|
*
|
2018-07-13 13:07:35 +00:00
|
|
|
* @param {string} topic
|
2017-06-14 11:00:01 +00:00
|
|
|
* @param {Object} data
|
|
|
|
*
|
|
|
|
* @global
|
2017-04-27 12:53:15 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the appropriate analytics event tracker for logging metrics to StatsD
|
2017-06-14 11:00:01 +00:00
|
|
|
* via [the "StatsD timers and counters" analytics event protocol][0].
|
2017-04-27 12:53:15 +00:00
|
|
|
*
|
2017-06-14 11:00:01 +00:00
|
|
|
* If logging metrics to StatsD is enabled for the duration of the user's
|
|
|
|
* session, then the appriopriate function is `mw.track`; otherwise it's
|
2018-10-29 01:48:38 +00:00
|
|
|
* `() => {}`.
|
2017-04-27 12:53:15 +00:00
|
|
|
*
|
2017-06-14 11:00:01 +00:00
|
|
|
* [0]: https://github.com/wikimedia/mediawiki-extensions-WikimediaEvents/blob/29c864a0/modules/ext.wikimediaEvents.statsd.js
|
2017-04-27 12:53:15 +00:00
|
|
|
*
|
|
|
|
* @param {Object} user
|
|
|
|
* @param {Object} config
|
2017-06-14 11:00:01 +00:00
|
|
|
* @param {Experiments} experiments
|
|
|
|
* @return {EventTracker}
|
2017-04-27 12:53:15 +00:00
|
|
|
*/
|
|
|
|
function getStatsvTracker( user, config, experiments ) {
|
2018-10-29 01:48:38 +00:00
|
|
|
return isStatsvEnabled( user, config, experiments ) ? mw.track : () => {};
|
2017-04-27 12:53:15 +00:00
|
|
|
}
|
|
|
|
|
2021-04-28 14:28:11 +00:00
|
|
|
/**
|
|
|
|
* Gets the appropriate analytics event tracker for logging virtual pageviews.
|
|
|
|
*
|
|
|
|
* @param {Object} config
|
|
|
|
* @return {EventTracker}
|
|
|
|
*/
|
|
|
|
function getPageviewTracker( config ) {
|
|
|
|
return config.get( 'wgPopupsVirtualPageViews' ) ? mw.track : () => {
|
|
|
|
// NOP
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2017-02-14 20:19:12 +00:00
|
|
|
/**
|
|
|
|
* Subscribes the registered change listeners to the
|
|
|
|
* [store](http://redux.js.org/docs/api/Store.html#store).
|
|
|
|
*
|
|
|
|
* @param {Redux.Store} store
|
2020-09-18 18:15:57 +00:00
|
|
|
* @param {Object} registerActions
|
2017-06-14 11:00:01 +00:00
|
|
|
* @param {UserSettings} userSettings
|
2017-02-14 20:19:12 +00:00
|
|
|
* @param {Function} settingsDialog
|
2017-06-14 11:00:01 +00:00
|
|
|
* @param {PreviewBehavior} previewBehavior
|
|
|
|
* @param {EventTracker} statsvTracker
|
2018-02-21 18:28:56 +00:00
|
|
|
* @param {EventTracker} pageviewTracker
|
2018-07-09 16:56:45 +00:00
|
|
|
* @return {void}
|
2017-02-14 20:19:12 +00:00
|
|
|
*/
|
2018-01-18 18:48:16 +00:00
|
|
|
function registerChangeListeners(
|
2020-09-18 18:15:57 +00:00
|
|
|
store, registerActions, userSettings, settingsDialog, previewBehavior,
|
2021-03-02 17:08:45 +00:00
|
|
|
statsvTracker, pageviewTracker
|
2018-01-18 18:48:16 +00:00
|
|
|
) {
|
2021-01-05 20:12:50 +00:00
|
|
|
registerChangeListener( store, changeListeners.footerLink( registerActions ) );
|
2020-12-11 16:30:03 +00:00
|
|
|
registerChangeListener( store, changeListeners.linkTitle() );
|
2017-02-14 20:19:12 +00:00
|
|
|
registerChangeListener( store, changeListeners.render( previewBehavior ) );
|
2018-01-18 18:48:16 +00:00
|
|
|
registerChangeListener(
|
2020-09-18 18:15:57 +00:00
|
|
|
store, changeListeners.statsv( registerActions, statsvTracker ) );
|
2018-01-18 18:48:16 +00:00
|
|
|
registerChangeListener(
|
|
|
|
store, changeListeners.syncUserSettings( userSettings ) );
|
|
|
|
registerChangeListener(
|
2020-09-18 18:15:57 +00:00
|
|
|
store, changeListeners.settings( registerActions, settingsDialog ) );
|
2018-02-08 22:11:44 +00:00
|
|
|
registerChangeListener( store,
|
2020-09-18 18:15:57 +00:00
|
|
|
changeListeners.pageviews( registerActions, pageviewTracker )
|
2018-02-08 22:11:44 +00:00
|
|
|
);
|
2017-02-14 20:19:12 +00:00
|
|
|
}
|
|
|
|
|
2022-07-12 22:48:46 +00:00
|
|
|
/**
|
|
|
|
* Creates an event handler that only executes if the current target
|
|
|
|
* is eligible for page previews and a title can be associated with the element.
|
|
|
|
*
|
|
|
|
* @param {Function} handler
|
|
|
|
* @return {Function}
|
|
|
|
*/
|
|
|
|
function handleDOMEventIfEligible( handler ) {
|
|
|
|
return function ( event ) {
|
2023-07-17 21:53:49 +00:00
|
|
|
let target = event && event.target;
|
|
|
|
if ( !target ) {
|
|
|
|
return;
|
|
|
|
}
|
2023-07-10 15:57:38 +00:00
|
|
|
// if the element is a text node, as events can be triggered on text nodes
|
|
|
|
// it won't have a closest method, so we get its parent element (T340081)
|
2023-07-17 21:53:49 +00:00
|
|
|
if ( target.nodeType === 3 ) {
|
|
|
|
target = target.parentNode;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the closest method is not defined, let's return early and
|
|
|
|
// understand this better by logging an error. (T340081)
|
|
|
|
if ( target && !target.closest ) {
|
|
|
|
const err = new Error( `T340081: Unexpected DOM element ${target.tagName} with nodeType ${target.nodeType}` );
|
|
|
|
mw.errorLogger.logError( err, 'error.web-team' );
|
|
|
|
return;
|
2023-07-10 15:57:38 +00:00
|
|
|
}
|
2023-07-17 21:53:49 +00:00
|
|
|
|
2023-05-15 14:26:37 +00:00
|
|
|
// If the event bubbles up all the way,
|
|
|
|
// document does not have closest method, so exit early (T336650).
|
2023-07-17 21:53:49 +00:00
|
|
|
if ( target === document ) {
|
2023-05-15 14:26:37 +00:00
|
|
|
return;
|
|
|
|
}
|
2023-07-17 21:53:49 +00:00
|
|
|
target = findNearestEligibleTarget( target );
|
2022-12-12 20:08:14 +00:00
|
|
|
if ( target === null ) {
|
2022-07-12 22:48:46 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
const mwTitle = titleFromElement( target, mw.config );
|
|
|
|
if ( mwTitle ) {
|
|
|
|
handler( target, mwTitle, event );
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
2017-02-14 20:19:12 +00:00
|
|
|
/*
|
|
|
|
* Initialize the application by:
|
2018-03-23 10:09:36 +00:00
|
|
|
* 1. Initializing side-effects and "services"
|
|
|
|
* 2. Creating the state store
|
|
|
|
* 3. Binding the actions to such store
|
|
|
|
* 4. Registering change listeners
|
|
|
|
* 5. Triggering the boot action to bootstrap the system
|
|
|
|
* 6. When the page content is ready:
|
|
|
|
* - Initializing the renderer
|
|
|
|
* - Binding hover and click events to the eligible links to trigger actions
|
2017-02-14 20:19:12 +00:00
|
|
|
*/
|
2018-04-12 18:10:05 +00:00
|
|
|
( function init() {
|
2021-03-09 17:21:48 +00:00
|
|
|
setUserConfigFlags( mw.config );
|
|
|
|
|
2018-03-19 19:39:41 +00:00
|
|
|
let compose = Redux.compose;
|
|
|
|
const
|
2017-02-14 20:19:12 +00:00
|
|
|
// So-called "services".
|
|
|
|
generateToken = mw.user.generateRandomSessionId,
|
2019-01-17 11:32:22 +00:00
|
|
|
pagePreviewGateway = createPagePreviewGateway( mw.config ),
|
2018-12-12 18:37:19 +00:00
|
|
|
referenceGateway = createReferenceGateway(),
|
2018-03-23 10:09:36 +00:00
|
|
|
userSettings = createUserSettings( mw.storage ),
|
2021-04-28 13:32:46 +00:00
|
|
|
referencePreviewsState = isReferencePreviewsEnabled( mw.user, userSettings, mw.config ),
|
|
|
|
settingsDialog = createSettingsDialogRenderer( referencePreviewsState !== null ),
|
2018-03-23 10:09:36 +00:00
|
|
|
experiments = createExperiments( mw.experiments ),
|
|
|
|
statsvTracker = getStatsvTracker( mw.user, mw.config, experiments ),
|
2021-04-28 14:28:11 +00:00
|
|
|
pageviewTracker = getPageviewTracker( mw.config ),
|
2021-04-16 11:06:13 +00:00
|
|
|
initiallyEnabled = {
|
|
|
|
[ previewTypes.TYPE_PAGE ]:
|
|
|
|
createIsPagePreviewsEnabled( mw.user, userSettings, mw.config ),
|
2021-04-28 13:32:46 +00:00
|
|
|
[ previewTypes.TYPE_REFERENCE ]: referencePreviewsState
|
2021-04-16 11:06:13 +00:00
|
|
|
};
|
2017-02-14 20:19:12 +00:00
|
|
|
|
|
|
|
// If debug mode is enabled, then enable Redux DevTools.
|
2021-04-09 09:27:44 +00:00
|
|
|
if ( mw.config.get( 'debug' ) ||
|
2019-03-18 11:48:58 +00:00
|
|
|
/* global process */
|
|
|
|
process.env.NODE_ENV !== 'production' ) {
|
2017-02-14 20:19:12 +00:00
|
|
|
// eslint-disable-next-line no-underscore-dangle
|
|
|
|
compose = window.__REDUX_DEVTOOLS_EXTENSION_COMPOSE__ || compose;
|
2016-11-14 19:37:11 +00:00
|
|
|
}
|
|
|
|
|
2018-03-19 19:39:41 +00:00
|
|
|
const store = Redux.createStore(
|
2017-03-14 11:33:15 +00:00
|
|
|
Redux.combineReducers( reducers ),
|
2017-02-14 20:19:12 +00:00
|
|
|
compose( Redux.applyMiddleware(
|
|
|
|
ReduxThunk.default
|
|
|
|
) )
|
|
|
|
);
|
2018-03-19 19:39:41 +00:00
|
|
|
const boundActions = Redux.bindActionCreators( actions, store.dispatch );
|
2018-06-27 22:44:04 +00:00
|
|
|
const previewBehavior = createPreviewBehavior( mw.user, boundActions );
|
2017-02-14 20:19:12 +00:00
|
|
|
|
2017-03-07 00:27:38 +00:00
|
|
|
registerChangeListeners(
|
2017-03-16 21:03:21 +00:00
|
|
|
store, boundActions, userSettings, settingsDialog,
|
2021-03-02 17:08:45 +00:00
|
|
|
previewBehavior, statsvTracker, pageviewTracker
|
2017-03-07 00:27:38 +00:00
|
|
|
);
|
2017-02-14 20:19:12 +00:00
|
|
|
|
2017-03-14 11:28:58 +00:00
|
|
|
boundActions.boot(
|
2021-04-16 11:06:13 +00:00
|
|
|
initiallyEnabled,
|
2017-02-14 20:19:12 +00:00
|
|
|
mw.user,
|
|
|
|
userSettings,
|
2018-02-15 20:15:23 +00:00
|
|
|
mw.config,
|
|
|
|
window.location.href
|
2017-02-14 20:19:12 +00:00
|
|
|
);
|
|
|
|
|
2017-08-09 09:14:52 +00:00
|
|
|
/*
|
|
|
|
* Register external interface exposing popups internals so that other
|
|
|
|
* extensions can query it (T171287)
|
|
|
|
*/
|
2022-07-12 22:48:46 +00:00
|
|
|
mw.popups = createMediaWikiPopupsObject(
|
|
|
|
store, registerModel, registerPreviewUI, registerGatewayForPreviewType
|
|
|
|
);
|
2017-08-09 09:14:52 +00:00
|
|
|
|
2021-04-16 11:06:13 +00:00
|
|
|
if ( initiallyEnabled[ previewTypes.TYPE_PAGE ] !== null ) {
|
2020-06-09 06:52:42 +00:00
|
|
|
const excludedLinksSelector = EXCLUDED_LINK_SELECTORS.join( ', ' );
|
2022-07-12 22:48:46 +00:00
|
|
|
// Register default preview type
|
|
|
|
mw.popups.register( {
|
|
|
|
type: previewTypes.TYPE_PAGE,
|
|
|
|
selector: `#mw-content-text a[href][title]:not(${excludedLinksSelector})`,
|
|
|
|
gateway: pagePreviewGateway,
|
|
|
|
renderFn: createPagePreview,
|
|
|
|
subTypes: [
|
|
|
|
{
|
|
|
|
type: previewTypes.TYPE_DISAMBIGUATION,
|
|
|
|
renderFn: createDisambiguationPreview
|
|
|
|
}
|
|
|
|
]
|
|
|
|
} );
|
2019-10-10 14:24:51 +00:00
|
|
|
}
|
2021-04-16 11:06:13 +00:00
|
|
|
if ( initiallyEnabled[ previewTypes.TYPE_REFERENCE ] !== null ) {
|
2022-07-12 22:48:46 +00:00
|
|
|
// Register the reference preview type
|
|
|
|
mw.popups.register( {
|
|
|
|
type: previewTypes.TYPE_REFERENCE,
|
|
|
|
selector: '#mw-content-text .reference a[ href*="#" ]',
|
|
|
|
gateway: referenceGateway,
|
2023-05-09 15:11:44 +00:00
|
|
|
renderFn: createReferencePreview,
|
|
|
|
init: () => {
|
|
|
|
initReferencePreviewsInstrumentation();
|
|
|
|
}
|
2022-07-12 22:48:46 +00:00
|
|
|
} );
|
2019-10-10 14:24:51 +00:00
|
|
|
}
|
2022-07-12 22:48:46 +00:00
|
|
|
if ( !isAnythingEligible() ) {
|
2021-05-06 08:05:24 +00:00
|
|
|
mw.log.warn( 'ext.popups was loaded but everything is disabled' );
|
2019-10-10 14:24:51 +00:00
|
|
|
return;
|
2018-12-12 18:37:19 +00:00
|
|
|
}
|
2017-02-14 20:19:12 +00:00
|
|
|
|
2018-04-11 17:22:50 +00:00
|
|
|
rendererInit();
|
2017-02-14 20:19:12 +00:00
|
|
|
|
2018-04-11 17:22:50 +00:00
|
|
|
/*
|
|
|
|
* Binding hover and click events to the eligible links to trigger actions
|
|
|
|
*/
|
2023-04-14 17:48:42 +00:00
|
|
|
function setupEventListeners() {
|
|
|
|
const onHover = handleDOMEventIfEligible( function ( target, mwTitle, event ) {
|
|
|
|
const type = getPreviewType( target );
|
|
|
|
const gateway = getGatewayForPreviewType( type );
|
|
|
|
if ( !gateway ) {
|
|
|
|
return;
|
|
|
|
}
|
2022-07-12 22:48:46 +00:00
|
|
|
|
2023-04-14 17:48:42 +00:00
|
|
|
const scrollTop = window.scrollY;
|
|
|
|
const bbox = target.getBoundingClientRect();
|
|
|
|
const offset = {
|
|
|
|
top: scrollTop + bbox.y,
|
|
|
|
left: window.scrollX + bbox.x
|
|
|
|
};
|
|
|
|
const measures = {
|
|
|
|
pageX: event.pageX,
|
|
|
|
pageY: event.pageY,
|
|
|
|
clientY: event.clientY,
|
|
|
|
width: target.offsetWidth,
|
|
|
|
height: target.offsetHeight,
|
|
|
|
offset,
|
|
|
|
clientRects: target.getClientRects(),
|
|
|
|
windowWidth: window.innerWidth,
|
|
|
|
windowHeight: window.innerHeight,
|
|
|
|
scrollTop
|
|
|
|
};
|
2022-07-12 22:48:46 +00:00
|
|
|
|
2023-04-14 17:48:42 +00:00
|
|
|
boundActions.linkDwell( mwTitle, target, measures, gateway, generateToken, type );
|
|
|
|
} );
|
|
|
|
const onHoverOut = handleDOMEventIfEligible( function () {
|
|
|
|
boundActions.abandon();
|
|
|
|
} );
|
|
|
|
const onClick = handleDOMEventIfEligible( function ( target ) {
|
|
|
|
if ( previewTypes.TYPE_PAGE === getPreviewType( target ) ) {
|
|
|
|
boundActions.linkClick( target );
|
|
|
|
}
|
|
|
|
} );
|
|
|
|
document.addEventListener( 'mouseover', onHover );
|
|
|
|
document.addEventListener( 'keyup', onHover );
|
|
|
|
document.addEventListener( 'mouseout', onHoverOut );
|
|
|
|
document.addEventListener( 'blur', onHoverOut );
|
|
|
|
document.addEventListener( 'click', onClick );
|
|
|
|
}
|
|
|
|
setupEventListeners();
|
2018-04-12 18:10:05 +00:00
|
|
|
}() );
|
2016-11-08 10:05:40 +00:00
|
|
|
|
2017-02-14 20:19:12 +00:00
|
|
|
window.Redux = Redux;
|
|
|
|
window.ReduxThunk = ReduxThunk;
|