( function ( mw, Redux, ReduxThunk ) { var BLACKLISTED_LINKS = [ '.extiw', '.image', '.new', '.internal', '.external', '.oo-ui-buttonedElement-button', '.cancelLink a' ]; /** * Creates a gateway with sensible values for the dependencies. * * See `mw.popups.createGateway`. * * @return {ext.popups.Gateway} */ function createGateway() { return mw.popups.createGateway( new mw.Api() ); } /** * Subscribes the registered change listeners to the * [store](http://redux.js.org/docs/api/Store.html#store). * * Change listeners are registered by setting a property on * `mw.popups.changeListeners`. * * @param {Redux.Store} store * @param {Object} actions * @param {mw.eventLog.Schema} schema * @param {ext.popups.UserSettings} userSettings */ function registerChangeListeners( store, actions, schema, userSettings ) { // Sugar. var changeListeners = mw.popups.changeListeners, registerChangeListener = mw.popups.registerChangeListener; registerChangeListener( store, changeListeners.footerLink( actions ) ); registerChangeListener( store, changeListeners.linkTitle() ); registerChangeListener( store, changeListeners.render( actions ) ); registerChangeListener( store, changeListeners.eventLogging( actions, schema ) ); registerChangeListener( store, changeListeners.previewCount( userSettings ) ); } /** * Binds the actions (or "action creators") to the * [store](http://redux.js.org/docs/api/Store.html#store). * * @param {Redux.Store} store * @return {Object} */ function createBoundActions( store ) { return Redux.bindActionCreators( mw.popups.actions, store.dispatch ); } /** * Root reducer for all actions * * @param {Object} global state before action * @param {Object} action Redux action that modified state. * Must have `type` property. * @return {Object} global state after action */ mw.popups.reducers.rootReducer = Redux.combineReducers( mw.popups.reducers ); /* * Initialize the application by: * 1. Creating the state store * 2. Binding the actions to such store * 3. Trigger the boot action to bootstrap the system * 4. When the page content is ready: * - Process the eligible links for page previews * - Initialize the renderer * - Bind hover and click events to the eligible links to trigger actions */ mw.requestIdleCallback( function () { var compose = Redux.compose, store, actions, // So-called "services". generateToken = mw.user.generateRandomSessionId, gateway = createGateway(), userSettings, isUserInCondition, schema; userSettings = mw.popups.createUserSettings( mw.storage, mw.user ); isUserInCondition = mw.popups.createExperiment( mw.config, mw.user, userSettings ); schema = mw.popups.createSchema( mw.config, window ); // If debug mode is enabled, then enable Redux DevTools. if ( mw.config.get( 'debug' ) === true ) { compose = window.__REDUX_DEVTOOLS_EXTENSION_COMPOSE__ || compose; } store = Redux.createStore( mw.popups.reducers.rootReducer, compose( Redux.applyMiddleware( ReduxThunk.default ) ) ); actions = createBoundActions( store ); registerChangeListeners( store, actions, schema, userSettings ); actions.boot( isUserInCondition, mw.user, userSettings, generateToken, mw.config ); mw.hook( 'wikipage.content' ).add( function ( $container ) { var previewLinks = mw.popups.processLinks( $container, BLACKLISTED_LINKS, mw.config ); mw.popups.renderer.init(); previewLinks .on( 'mouseover focus', function ( event ) { actions.linkDwell( this, event, gateway, generateToken ); } ) .on( 'mouseout blur', function () { actions.linkAbandon( this ); } ) .on( 'click', function () { actions.linkClick( this ); } ); } ); } ); }( mediaWiki, Redux, ReduxThunk ) );