notifications = $notifications; $this->categories = $categories; $this->defaultNotifyTypeAvailability = $defaultNotifyTypeAvailability; $this->notifyTypeAvailabilityByCategory = $notifyTypeAvailabilityByCategory; $this->dismissabilityByCategory = null; $this->notifiers = $notifiers; } /** * Create an instance from global variables * @return EchoAttributeManager */ public static function newFromGlobalVars() { global $wgEchoNotifications, $wgEchoNotificationCategories, $wgDefaultNotifyTypeAvailability, $wgNotifyTypeAvailabilityByCategory, $wgEchoNotifiers; // Unit test may alter the global data for test purpose if ( defined( 'MW_PHPUNIT_TEST' ) ) { return new self( $wgEchoNotifications, $wgEchoNotificationCategories, $wgDefaultNotifyTypeAvailability, $wgNotifyTypeAvailabilityByCategory, $wgEchoNotifiers ); } if ( self::$globalVarInstance === null ) { self::$globalVarInstance = new self( $wgEchoNotifications, $wgEchoNotificationCategories, $wgDefaultNotifyTypeAvailability, $wgNotifyTypeAvailabilityByCategory, $wgEchoNotifiers ); } return self::$globalVarInstance; } /** * Get the user-locators|user-filters related to the provided event type * * @param string $type * @param string $locator Either self::ATTR_LOCATORS or self::ATTR_FILTERS * @return array */ public function getUserCallable( $type, $locator = self::ATTR_LOCATORS ) { if ( isset( $this->notifications[$type][$locator] ) ) { return (array)$this->notifications[$type][$locator]; } else { return []; } } /** * Get the enabled events for a user, which excludes user-dismissed events * from the general enabled events * @param User $user * @param string $notifyType Either "web" or "email". * @return string[] */ public function getUserEnabledEvents( User $user, $notifyType ) { $eventTypesToLoad = $this->notifications; foreach ( $eventTypesToLoad as $eventType => $eventData ) { $category = $this->getNotificationCategory( $eventType ); // Make sure the user is eligible to receive this type of notification if ( !$this->getCategoryEligibility( $user, $category ) ) { unset( $eventTypesToLoad[$eventType] ); } if ( !$user->getOption( 'echo-subscriptions-' . $notifyType . '-' . $category ) ) { unset( $eventTypesToLoad[$eventType] ); } } $eventTypes = array_keys( $eventTypesToLoad ); return $eventTypes; } /** * Get the user enabled events for the specified sections * @param User $user * @param string $notifyType Either "web" or "email". * @param string[] $sections * @return string[] */ public function getUserEnabledEventsbySections( User $user, $notifyType, array $sections ) { $events = []; foreach ( $sections as $section ) { $events = array_merge( $events, $this->getEventsForSection( $section ) ); } return array_intersect( $this->getUserEnabledEvents( $user, $notifyType ), $events ); } /** * Gets events (notification types) for a given section * * @param string $section Internal section name, one of the values from self::$sections * * @return array Array of notification types in this section */ public function getEventsForSection( $section ) { $events = []; $isDefault = ( $section === self::$DEFAULT_SECTION ); foreach ( $this->notifications as $event => $attribs ) { if ( ( isset( $attribs['section'] ) && $attribs['section'] === $section ) || ( $isDefault && ( !isset( $attribs['section'] ) || // Invalid section !in_array( $attribs['section'], self::$sections ) ) ) ) { $events[] = $event; } } return $events; } /** * Gets array of internal category names * * @return All internal names */ public function getInternalCategoryNames() { return array_keys( $this->categories ); } /** * See if a user is eligible to receive a certain type of notification * (based on user groups, not user preferences) * * @param User * @param string A notification category defined in $wgEchoNotificationCategories * @return bool */ public function getCategoryEligibility( $user, $category ) { $usersGroups = $user->getGroups(); if ( isset( $this->categories[$category]['usergroups'] ) ) { $allowedGroups = $this->categories[$category]['usergroups']; if ( !array_intersect( $usersGroups, $allowedGroups ) ) { return false; } } return true; } /** * Get the priority for a specific notification type * * @param string A notification type defined in $wgEchoNotifications * @return int From 1 to 10 (10 is default) */ public function getNotificationPriority( $notificationType ) { $category = $this->getNotificationCategory( $notificationType ); return $this->getCategoryPriority( $category ); } /** * Get the priority for a notification category * * @param string A notification category defined in $wgEchoNotificationCategories * @return int From 1 to 10 (10 is default) */ public function getCategoryPriority( $category ) { if ( isset( $this->categories[$category]['priority'] ) ) { $priority = $this->categories[$category]['priority']; if ( $priority >= 1 && $priority <= 10 ) { return $priority; } } return 10; } /** * Get the notification category for a notification type * * @param string A notification type defined in $wgEchoNotifications * @return string The name of the notification category or 'other' if no * category is explicitly assigned. */ public function getNotificationCategory( $notificationType ) { if ( isset( $this->notifications[$notificationType]['category'] ) ) { $category = $this->notifications[$notificationType]['category']; if ( isset( $this->categories[$category] ) ) { return $category; } } return 'other'; } /** * Gets an associative array mapping categories to the notification types in * the category * * @return array Associative array with category as key */ public function getEventsByCategory() { $eventsByCategory = []; foreach ( $this->categories as $category => $categoryDetails ) { $eventsByCategory[$category] = []; } foreach ( $this->notifications as $notificationType => $notificationDetails ) { $category = $notificationDetails['category']; if ( isset( $eventsByCategory[$category] ) ) { // Only real categories. Currently, this excludes the 'foreign' // psuedo-category. $eventsByCategory[$category][] = $notificationType; } } return $eventsByCategory; } /** * Checks whether the specified notify type is available for the specified * category. * * This means whether users *can* turn notifications for this category and format * on, regardless of the default or a particular user's preferences. * * @param string $category Category name * @param string $notifyType notify type, e.g. email/web. */ public function isNotifyTypeAvailableForCategory( $category, $notifyType ) { if ( isset( $this->notifyTypeAvailabilityByCategory[$category][$notifyType] ) ) { return $this->notifyTypeAvailabilityByCategory[$category][$notifyType]; } else { return $this->defaultNotifyTypeAvailability[$notifyType]; } } /** * Checks whether category is displayed in preferences * * @param string $category Category name */ public function isCategoryDisplayedInPreferences( $category ) { return !( isset( $this->categories[$category]['no-dismiss'] ) && in_array( 'all', $this->categories[$category]['no-dismiss'] ) ); } /** * Checks whether the specified notify type is dismissable for the specified * category. * * This means whether the user is allowed to opt out of receiving notifications * for this category and format. * * @param string $category Name of category * @param string $notifyType notify type, e.g. email/web. */ public function isNotifyTypeDismissableForCategory( $category, $notifyType ) { return !( isset( $this->categories[$category]['no-dismiss'] ) && ( in_array( 'all', $this->categories[$category]['no-dismiss'] ) || in_array( $notifyType, $this->categories[$category]['no-dismiss'] ) ) ); } /** * Get notification section for a notification type * @todo add a unit test case * @param string $notificationType * @return string */ public function getNotificationSection( $notificationType ) { if ( isset( $this->notifications[$notificationType]['section'] ) ) { return $this->notifications[$notificationType]['section']; } return 'alert'; } }