2017-07-12 15:12:40 +00:00
|
|
|
( function ( M ) {
|
2019-02-07 16:34:18 +00:00
|
|
|
var
|
|
|
|
mobile = M.require( 'mobile.startup' ),
|
2019-02-08 17:04:26 +00:00
|
|
|
mfExtend = mobile.mfExtend,
|
2019-02-07 16:34:18 +00:00
|
|
|
View = mobile.View,
|
|
|
|
util = mobile.util,
|
|
|
|
Icon = mobile.Icon,
|
2018-01-05 18:52:17 +00:00
|
|
|
notificationIcon = new Icon( {
|
|
|
|
name: 'notifications',
|
|
|
|
glyphPrefix: 'minerva'
|
|
|
|
} ),
|
2019-02-07 16:34:18 +00:00
|
|
|
icons = mobile.icons;
|
2017-07-12 15:12:40 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* A notification button for communicating with an NotificationOverlay
|
|
|
|
* @class NotificationButton
|
|
|
|
* @extends View
|
2018-07-10 22:47:31 +00:00
|
|
|
* @param {Object} options Configuration options
|
2017-07-12 15:12:40 +00:00
|
|
|
*/
|
|
|
|
function NotificationBadge( options ) {
|
|
|
|
var $el,
|
2017-11-28 19:46:55 +00:00
|
|
|
count = options.notificationCountRaw || 0,
|
2017-07-12 15:12:40 +00:00
|
|
|
el = options.el;
|
|
|
|
|
|
|
|
if ( el ) {
|
|
|
|
$el = $( el );
|
|
|
|
options.hasUnseenNotifications = $el.find( '.notification-unseen' ).length;
|
|
|
|
options.hasNotifications = options.hasUnseenNotifications;
|
|
|
|
options.title = $el.find( 'a' ).attr( 'title' );
|
|
|
|
options.url = $el.find( 'a' ).attr( 'href' );
|
2017-10-18 18:28:37 +00:00
|
|
|
count = Number( $el.find( 'span' ).data( 'notification-count' ) );
|
2018-03-08 19:22:55 +00:00
|
|
|
options.onError = function () {
|
|
|
|
// FIXME: Blocked on T189173. Ideally we'd use the router here.
|
|
|
|
window.location.href = this.getNotificationURL();
|
|
|
|
}.bind( this );
|
2017-07-12 15:12:40 +00:00
|
|
|
}
|
2018-11-22 19:10:40 +00:00
|
|
|
View.call( this,
|
|
|
|
util.extend( options, {
|
|
|
|
isBorderBox: false
|
|
|
|
} )
|
|
|
|
);
|
2017-11-28 19:46:55 +00:00
|
|
|
this.url = options.url;
|
2017-07-12 15:12:40 +00:00
|
|
|
this._bindOverlayManager();
|
2017-10-18 18:28:37 +00:00
|
|
|
this.setCount( count );
|
2017-07-12 15:12:40 +00:00
|
|
|
}
|
|
|
|
|
2019-02-08 17:04:26 +00:00
|
|
|
mfExtend( NotificationBadge, View, {
|
2017-07-12 15:12:40 +00:00
|
|
|
/**
|
2018-07-03 14:50:09 +00:00
|
|
|
* @cfg {object} defaults Default options hash.
|
|
|
|
* @cfg {string} defaults.notificationIconClass e.g. mw-ui-icon for icon
|
|
|
|
* @cfg {string} defaults.loadingIconHtml for spinner
|
|
|
|
* @cfg {boolean} defaults.hasUnseenNotifications whether the user has unseen notifications
|
|
|
|
* @cfg {number} defaults.notificationCountRaw number of unread notifications
|
2018-08-20 23:40:40 +00:00
|
|
|
* @memberof NotificationBadge
|
|
|
|
* @instance
|
|
|
|
*/
|
2017-07-12 15:12:40 +00:00
|
|
|
defaults: {
|
|
|
|
notificationIconClass: notificationIcon.getClassName(),
|
|
|
|
loadingIconHtml: icons.spinner().toHtmlString(),
|
|
|
|
hasNotifications: false,
|
|
|
|
hasUnseenNotifications: false,
|
2017-10-18 18:28:37 +00:00
|
|
|
notificationCountRaw: 0
|
2017-07-12 15:12:40 +00:00
|
|
|
},
|
|
|
|
/**
|
|
|
|
* Loads a ResourceLoader module script. Shows ajax loader whilst loading.
|
|
|
|
* @method
|
|
|
|
* @private
|
2018-08-20 23:40:40 +00:00
|
|
|
* @memberof NotificationBadge
|
|
|
|
* @instance
|
2017-07-12 15:12:40 +00:00
|
|
|
* @param {string} moduleName Name of a module to fetch
|
2018-08-22 00:26:46 +00:00
|
|
|
* @return {JQuery.Promise}
|
2017-07-12 15:12:40 +00:00
|
|
|
*/
|
|
|
|
_loadModuleScript: function ( moduleName ) {
|
|
|
|
var self = this;
|
|
|
|
|
|
|
|
this.$el.html( this.options.loadingIconHtml );
|
2018-08-22 00:26:46 +00:00
|
|
|
return mw.loader.using( moduleName ).then( function () {
|
2017-07-12 15:12:40 +00:00
|
|
|
// trigger a re-render once one to remove loading icon
|
|
|
|
self.render();
|
|
|
|
} );
|
|
|
|
},
|
|
|
|
/**
|
|
|
|
* Load the notification overlay.
|
|
|
|
* @method
|
|
|
|
* @private
|
2018-08-20 23:40:40 +00:00
|
|
|
* @memberof NotificationBadge
|
|
|
|
* @instance
|
2017-07-12 15:12:40 +00:00
|
|
|
* @uses NotificationsOverlay
|
2018-07-03 14:50:09 +00:00
|
|
|
* @return {JQuery.Deferred} with an instance of NotificationsOverlay
|
2017-07-12 15:12:40 +00:00
|
|
|
*/
|
|
|
|
_loadNotificationOverlay: function () {
|
|
|
|
var self = this;
|
|
|
|
|
|
|
|
return this._loadModuleScript( 'mobile.notifications.overlay' ).then( function () {
|
|
|
|
var NotificationsOverlay =
|
|
|
|
M.require( 'mobile.notifications.overlay/NotificationsOverlay' ); // resource-modules-disable-line
|
|
|
|
return new NotificationsOverlay( {
|
|
|
|
badge: self
|
|
|
|
} );
|
|
|
|
} );
|
|
|
|
},
|
|
|
|
/**
|
|
|
|
* Sets up routes in overlay manager and click behaviour for NotificationBadge
|
|
|
|
* This is not unit tested as it's behaviour is covered by browser tests.
|
2018-08-20 23:40:40 +00:00
|
|
|
* @memberof NotificationBadge
|
|
|
|
* @instance
|
2017-07-12 15:12:40 +00:00
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
_bindOverlayManager: function () {
|
|
|
|
var self = this,
|
|
|
|
mainMenu = this.options.mainMenu;
|
|
|
|
|
2018-11-27 12:20:16 +00:00
|
|
|
this.$el.on( 'click', this.onClickBadge.bind( this ) );
|
2017-07-12 15:12:40 +00:00
|
|
|
this.options.overlayManager.add( /^\/notifications$/, function () {
|
2018-08-22 00:26:46 +00:00
|
|
|
return self._loadNotificationOverlay().then( function ( overlay ) {
|
2019-01-09 13:57:26 +00:00
|
|
|
// eslint-disable-next-line jquery/no-global-selector
|
|
|
|
var $pageCenter = $( '#mw-mf-page-center' );
|
|
|
|
|
2017-07-12 15:12:40 +00:00
|
|
|
mainMenu.openNavigationDrawer( 'secondary' );
|
|
|
|
overlay.on( 'hide', function () {
|
|
|
|
mainMenu.closeNavigationDrawers();
|
2019-01-09 13:57:26 +00:00
|
|
|
$pageCenter.off( '.secondary' );
|
2017-07-12 15:12:40 +00:00
|
|
|
} );
|
|
|
|
|
2019-01-09 13:57:26 +00:00
|
|
|
$pageCenter.one( 'click.secondary', function () {
|
2017-07-12 15:12:40 +00:00
|
|
|
self.options.router.back();
|
|
|
|
} );
|
2018-08-22 00:26:46 +00:00
|
|
|
return overlay;
|
2017-07-12 15:12:40 +00:00
|
|
|
} );
|
|
|
|
} );
|
|
|
|
},
|
|
|
|
template: mw.template.get( 'skins.minerva.notifications.badge', 'badge.hogan' ),
|
|
|
|
/**
|
|
|
|
* Click handler for clicking on the badge
|
2018-08-20 23:40:40 +00:00
|
|
|
* @memberof NotificationBadge
|
|
|
|
* @instance
|
2018-07-03 14:50:09 +00:00
|
|
|
* @return {boolean}
|
2017-07-12 15:12:40 +00:00
|
|
|
*/
|
|
|
|
onClickBadge: function () {
|
|
|
|
this.options.router.navigate( '#/notifications' );
|
2018-09-13 15:33:20 +00:00
|
|
|
// Important that we also prevent propagation to avoid interference with events that may
|
|
|
|
// be binded on #mw-mf-page-center that close overlay
|
2017-07-12 15:12:40 +00:00
|
|
|
return false;
|
|
|
|
},
|
|
|
|
/**
|
|
|
|
* Return the URL for the full non-overlay notification view
|
2018-08-20 23:40:40 +00:00
|
|
|
* @memberof NotificationBadge
|
|
|
|
* @instance
|
2018-07-03 14:50:09 +00:00
|
|
|
* @return {string} url
|
2017-07-12 15:12:40 +00:00
|
|
|
*/
|
|
|
|
getNotificationURL: function () {
|
|
|
|
return this.options.url;
|
|
|
|
},
|
|
|
|
/**
|
|
|
|
* Update the notification count
|
2018-08-20 23:40:40 +00:00
|
|
|
* @memberof NotificationBadge
|
|
|
|
* @instance
|
2018-07-03 14:50:09 +00:00
|
|
|
* @param {number} count
|
2017-07-12 15:12:40 +00:00
|
|
|
*/
|
|
|
|
setCount: function ( count ) {
|
2017-10-18 18:28:37 +00:00
|
|
|
if ( count > 100 ) {
|
|
|
|
count = 100;
|
|
|
|
}
|
|
|
|
this.options.notificationCountRaw = count;
|
|
|
|
this.options.notificationCountString = mw.message( 'echo-badge-count',
|
|
|
|
mw.language.convertNumber( count )
|
|
|
|
).text();
|
2017-07-12 15:12:40 +00:00
|
|
|
this.options.isNotificationCountZero = count === 0;
|
|
|
|
this.render();
|
|
|
|
},
|
|
|
|
/**
|
|
|
|
* Marks all notifications as seen
|
2018-08-20 23:40:40 +00:00
|
|
|
*
|
|
|
|
* @memberof NotificationBadge
|
|
|
|
* @instance
|
2017-07-12 15:12:40 +00:00
|
|
|
*/
|
|
|
|
markAsSeen: function () {
|
|
|
|
this.options.hasUnseenNotifications = false;
|
|
|
|
this.render();
|
|
|
|
}
|
|
|
|
} );
|
|
|
|
|
|
|
|
M.define( 'skins.minerva.notifications/NotificationBadge', NotificationBadge );
|
|
|
|
}( mw.mobileFrontend ) );
|