getArticleId(); if ( !isset( self::$propCache[ $id ] ) ) { self::$propCache[ $id ] = []; } if ( !isset( self::$propCache[ $id ][ $prop ] ) ) { $services = MediaWikiServices::getInstance(); $props = $services->getPageProps()->getProperties( $title, $prop ); self::$propCache[ $id ][ $prop ] = isset( $props[ $id ] ); } return self::$propCache[ $id ][ $prop ]; } /** * Check if a DiscussionTools feature is available to this user * * @param UserIdentity $user * @param string|null $feature Feature to check for (one of static::FEATURES) * Null will check for any DT feature. * @return bool */ public static function isFeatureAvailableToUser( UserIdentity $user, ?string $feature = null ): bool { $services = MediaWikiServices::getInstance(); $dtConfig = $services->getConfigFactory()->makeConfig( 'discussiontools' ); if ( !$dtConfig->get( 'DiscussionToolsEnable' ) ) { return false; } if ( ( $feature === self::TOPICSUBSCRIPTION || $feature === self::AUTOTOPICSUB ) && !$user->isRegistered() ) { // Users must be logged in to use topic subscription return false; } $optionsLookup = $services->getUserOptionsLookup(); if ( $feature ) { // Feature-specific override if ( $dtConfig->get( 'DiscussionTools_' . $feature ) !== 'default' ) { // Feature setting can be 'available' or 'unavailable', overriding any BetaFeatures settings return $dtConfig->get( 'DiscussionTools_' . $feature ) === 'available'; } } else { // Non-feature-specific override, check for any feature foreach ( static::FEATURES as $feat ) { if ( $dtConfig->get( 'DiscussionTools_' . $feat ) === 'available' ) { return true; } } } // Being in the "test" group for this feature means it's enabled. This // overrules the wiki's beta feature setting. (However, a user who's // in the control group can still bypass this and enable the feature // normally.) $abtest = static::determineUserABTestBucket( $user, $feature ); if ( $abtest === 'test' ) { return true; } // No feature-specific override found. if ( $dtConfig->get( 'DiscussionToolsBeta' ) ) { $betaenabled = $optionsLookup->getOption( $user, 'discussiontools-betaenable', 0 ); return (bool)$betaenabled; } // Assume that if BetaFeature is turned off, or user has it enabled, that // some features are available. // If this isn't the case, then DiscussionToolsEnable should have been set to false. return true; } /** * Check if a DiscussionTools feature is enabled by this user * * @param UserIdentity $user * @param string|null $feature Feature to check for (one of static::FEATURES) * Null will check for any DT feature. * @return bool */ public static function isFeatureEnabledForUser( UserIdentity $user, ?string $feature = null ): bool { if ( !static::isFeatureAvailableToUser( $user, $feature ) ) { return false; } $services = MediaWikiServices::getInstance(); $optionsLookup = $services->getUserOptionsLookup(); if ( $feature ) { // Check for a specific feature return $optionsLookup->getOption( $user, 'discussiontools-' . $feature ); } else { // Check for any feature foreach ( static::FEATURES as $feat ) { if ( $optionsLookup->getOption( $user, 'discussiontools-' . $feat ) ) { return true; } } return false; } } /** * Work out the A/B test bucket for the current user * * Checks whether the user has been enrolled in the last A/B test, if any was enabled. * * If the A/B test is enabled, and the user is eligible and not enrolled, it will enroll them. * * @param UserIdentity $user * @param string|null $feature Feature to check for (one of static::FEATURES) * Null will check for any DT feature. * @return string 'test' if in the test group, 'control' if in the control group, or '' if they've * never been in the test */ private static function determineUserABTestBucket( UserIdentity $user, ?string $feature = null ): string { $services = MediaWikiServices::getInstance(); $optionsManager = $services->getUserOptionsManager(); $dtConfig = $services->getConfigFactory()->makeConfig( 'discussiontools' ); $abtest = $dtConfig->get( 'DiscussionToolsABTest' ); $abstate = $optionsManager->getOption( $user, 'discussiontools-abtest2' ); if ( $user->isRegistered() && $feature && $abtest == $feature ) { // The A/B test is enabled, and the user is qualified to be in the // test by being logged in. if ( !$abstate && !$optionsManager->getOption( $user, 'discussiontools-newtopictool-opened' ) ) { // Assign the user to a group. This is only being done to // users who have never used the tool before, for which we're // using the absence of discussiontools-newtopictool-opened. $abstate = $user->getId() % 2 == 0 ? 'test' : 'control'; $optionsManager->setOption( $user, 'discussiontools-abtest2', $abstate ); $optionsManager->saveOptions( $user ); } return $abstate; } return ''; } /** * Check if the tools are available for a given title * * @param Title $title * @return bool */ public static function isAvailableForTitle( Title $title ): bool { // Only wikitext pages (e.g. not Flow boards, special pages) if ( $title->getContentModel() !== CONTENT_MODEL_WIKITEXT ) { return false; } if ( !$title->canExist() ) { return false; } $hasNewSectionLink = self::hasPagePropCached( $title, 'newsectionlink' ); $services = MediaWikiServices::getInstance(); // Check that the page supports discussions. // Treat pages with __NEWSECTIONLINK__ as talk pages (T245890) return $hasNewSectionLink || // `wantSignatures` includes talk pages $services->getNamespaceInfo()->wantSignatures( $title->getNamespace() ); // TODO: Consider not loading if forceHideNewSectionLink is true. } /** * Check if the tool is available on a given page * * @param OutputPage $output * @param string|null $feature Feature to check for (one of static::FEATURES) * Null will check for any DT feature. * @return bool */ public static function isFeatureEnabledForOutput( OutputPage $output, ?string $feature = null ): bool { // Don't show on edit pages, history, etc. if ( $feature !== self::NEWTOPICTOOL && $output->getActionName() !== 'view' ) { return false; } $title = $output->getTitle(); // Don't show on pages without a Title if ( !$title ) { return false; } // Topic subscription is not available on your own talk page, as you will // get 'edit-user-talk' notifications already. (T276996) if ( ( $feature === self::TOPICSUBSCRIPTION || $feature === self::AUTOTOPICSUB ) && $title->equals( $output->getUser()->getTalkPage() ) ) { return false; } // New topic tool is not available if __NONEWSECTIONLINK__ is set // We may need to move this check to the client when we support // launching the tool from other pages. if ( $feature === self::NEWTOPICTOOL ) { if ( self::hasPagePropCached( $title, 'nonewsectionlink' ) ) { return false; } } // ?dtenable=1 overrides all user and title checks $queryEnable = $output->getRequest()->getRawVal( 'dtenable' ) ?: // Extra hack for parses from API, where this parameter isn't passed to derivative requests RequestContext::getMain()->getRequest()->getRawVal( 'dtenable' ); if ( $queryEnable ) { return true; } if ( $queryEnable === '0' ) { // ?dtenable=0 forcibly disables the feature regardless of any other checks (T285578) return false; } if ( !static::isAvailableForTitle( $title ) ) { return false; } $isMobile = false; if ( ExtensionRegistry::getInstance()->isLoaded( 'MobileFrontend' ) ) { $mobFrontContext = MediaWikiServices::getInstance()->getService( 'MobileFrontend.Context' ); $isMobile = $mobFrontContext->shouldDisplayMobileView(); } $dtConfig = MediaWikiServices::getInstance()->getConfigFactory()->makeConfig( 'discussiontools' ); if ( $isMobile ) { // Enabling mobile removes MobileFrontend's reply and new topic tools, so always // enable these tools as a replacement. // Topic subscription is not yet available on mobile, awaiting UI implementation. return $dtConfig->get( 'DiscussionToolsEnableMobile' ) && ( $feature === null || $feature === self::REPLYTOOL || $feature === self::NEWTOPICTOOL || $feature === self::SOURCEMODETOOLBAR ); } return static::isFeatureEnabledForUser( $output->getUser(), $feature ); } /** * Check if this page view should open the new topic tool on page load. * * @param IContextSource $context * @return bool */ public static function shouldOpenNewTopicTool( IContextSource $context ): bool { $req = $context->getRequest(); $out = $context->getOutput(); return ( // ?title=...&action=edit§ion=new // ?title=...&veaction=editsource§ion=new ( $req->getVal( 'action' ) === 'edit' || $req->getVal( 'veaction' ) === 'editsource' ) && $req->getVal( 'section' ) === 'new' && // Adding a new topic with preloaded text is not supported yet (T269310) !( $req->getVal( 'editintro' ) || $req->getVal( 'preload' ) || $req->getVal( 'preloadparams' ) || $req->getVal( 'preloadtitle' ) ) && // User has new topic tool enabled (and not using &dtenable=0) self::isFeatureEnabledForOutput( $out, self::NEWTOPICTOOL ) ); } /** * Check if this page view should display the empty state for talk pages that don't exist. * * @param IContextSource $context * @return bool */ public static function shouldDisplayEmptyState( IContextSource $context ): bool { $req = $context->getRequest(); $out = $context->getOutput(); $title = $context->getTitle(); return ( ( // When following a red link from another page (but not when clicking the 'Edit' tab) ( $req->getVal( 'action' ) === 'edit' && $req->getVal( 'redlink' ) === '1' ) || // When the new topic tool will be opened (usually when clicking the 'Add topic' tab) self::shouldOpenNewTopicTool( $context ) || // In read mode (accessible for non-existent pages by clicking 'Cancel' in editor) $req->getVal( 'action', 'view' ) === 'view' ) && // Duh !$title->exists() && // Only in talk namespaces, not including other namespaces that isAvailableForTitle() allows $title->isTalkPage() && // The default display will probably be more useful for... // ...Permanent links to revisions of pages which have been deleted $req->getIntOrNull( 'oldid' ) === null && // ...Non-existent pages with default content, e.g. in 'MediaWiki:' namespace !$title->hasSourceText() && // User has new topic tool enabled (and not using &dtenable=0) self::isFeatureEnabledForOutput( $out, self::NEWTOPICTOOL ) ); } /** * Check if we should be adding automatic topic subscriptions for this user on this page. * * @param UserIdentity $user * @param Title $title * @return bool */ public static function shouldAddAutoSubscription( UserIdentity $user, Title $title ): bool { // This duplicates the logic from isFeatureEnabledForOutput(), // because we don't have access to the request or the output here. // Topic subscription is not available on your own talk page, as you will // get 'edit-user-talk' notifications already. (T276996) // (can't use User::getTalkPage() to check because this is a UserIdentity) if ( $title->inNamespace( NS_USER_TALK ) && $title->getText() === $user->getName() ) { return false; } // Users flagged as bots shouldn't be autosubscribed. They can // manually subscribe if it becomes relevant. (T301933) $user = MediaWikiServices::getInstance() ->getUserFactory() ->newFromUserIdentity( $user ); if ( $user->isBot() ) { return false; } // Check if the user has automatic subscriptions enabled, and the tools are enabled on the page. return static::isAvailableForTitle( $title ) && static::isFeatureEnabledForUser( $user, self::AUTOTOPICSUB ); } }