mirror of
https://gerrit.wikimedia.org/r/mediawiki/extensions/Thanks
synced 2024-11-30 17:44:18 +00:00
4bf74e6415
Hooks::canReceiveThanks is still static, because it is also used in MobileFrontendHandler. Change-Id: Idff34ebce914ad37bcaea8de04b3ef5e01d7d98d
545 lines
17 KiB
PHP
545 lines
17 KiB
PHP
<?php
|
|
|
|
// phpcs:disable MediaWiki.NamingConventions.LowerCamelFunctionsName.FunctionName
|
|
|
|
namespace MediaWiki\Extension\Thanks;
|
|
|
|
use ApiModuleManager;
|
|
use Article;
|
|
use DatabaseLogEntry;
|
|
use DifferenceEngine;
|
|
use EchoAttributeManager;
|
|
use EchoUserLocator;
|
|
use ExtensionRegistry;
|
|
use GenderCache;
|
|
use IContextSource;
|
|
use LogEventsList;
|
|
use LogPage;
|
|
use MediaWiki\Api\Hook\ApiMain__moduleManagerHook;
|
|
use MediaWiki\Auth\Hook\LocalUserCreatedHook;
|
|
use MediaWiki\Block\Hook\GetAllBlockActionsHook;
|
|
use MediaWiki\Config\Config;
|
|
use MediaWiki\Config\ConfigException;
|
|
use MediaWiki\Diff\Hook\DifferenceEngineViewHeaderHook;
|
|
use MediaWiki\Diff\Hook\DiffToolsHook;
|
|
use MediaWiki\Extension\Notifications\Hooks\BeforeCreateEchoEventHook;
|
|
use MediaWiki\Extension\Notifications\Hooks\EchoGetBundleRulesHook;
|
|
use MediaWiki\Extension\Notifications\Model\Event;
|
|
use MediaWiki\Extension\Thanks\Api\ApiFlowThank;
|
|
use MediaWiki\Hook\BeforePageDisplayHook;
|
|
use MediaWiki\Hook\GetLogTypesOnUserHook;
|
|
use MediaWiki\Hook\HistoryToolsHook;
|
|
use MediaWiki\Hook\LogEventsListLineEndingHook;
|
|
use MediaWiki\Hook\PageHistoryBeforeListHook;
|
|
use MediaWiki\Html\Html;
|
|
use MediaWiki\Linker\LinkTarget;
|
|
use MediaWiki\Output\OutputPage;
|
|
use MediaWiki\Permissions\PermissionManager;
|
|
use MediaWiki\Revision\RevisionLookup;
|
|
use MediaWiki\Revision\RevisionRecord;
|
|
use MediaWiki\SpecialPage\SpecialPage;
|
|
use MediaWiki\Title\Title;
|
|
use MediaWiki\User\Options\UserOptionsManager;
|
|
use MediaWiki\User\User;
|
|
use MediaWiki\User\UserFactory;
|
|
use MediaWiki\User\UserIdentity;
|
|
use RequestContext;
|
|
use Skin;
|
|
|
|
/**
|
|
* Hooks for Thanks extension
|
|
*
|
|
* @file
|
|
* @ingroup Extensions
|
|
*/
|
|
class Hooks implements
|
|
ApiMain__moduleManagerHook,
|
|
BeforeCreateEchoEventHook,
|
|
BeforePageDisplayHook,
|
|
DiffToolsHook,
|
|
DifferenceEngineViewHeaderHook,
|
|
EchoGetBundleRulesHook,
|
|
GetAllBlockActionsHook,
|
|
GetLogTypesOnUserHook,
|
|
HistoryToolsHook,
|
|
LocalUserCreatedHook,
|
|
LogEventsListLineEndingHook,
|
|
PageHistoryBeforeListHook
|
|
{
|
|
private Config $config;
|
|
private GenderCache $genderCache;
|
|
private PermissionManager $permissionManager;
|
|
private RevisionLookup $revisionLookup;
|
|
private UserFactory $userFactory;
|
|
private UserOptionsManager $userOptionsManager;
|
|
|
|
public function __construct(
|
|
Config $config,
|
|
GenderCache $genderCache,
|
|
PermissionManager $permissionManager,
|
|
RevisionLookup $revisionLookup,
|
|
UserFactory $userFactory,
|
|
UserOptionsManager $userOptionsManager
|
|
) {
|
|
$this->config = $config;
|
|
$this->genderCache = $genderCache;
|
|
$this->permissionManager = $permissionManager;
|
|
$this->revisionLookup = $revisionLookup;
|
|
$this->userFactory = $userFactory;
|
|
$this->userOptionsManager = $userOptionsManager;
|
|
}
|
|
|
|
/**
|
|
* Handler for the HistoryTools hook
|
|
*
|
|
* @param RevisionRecord $revisionRecord
|
|
* @param array &$links
|
|
* @param RevisionRecord|null $oldRevisionRecord
|
|
* @param UserIdentity $userIdentity
|
|
*/
|
|
public function onHistoryTools(
|
|
$revisionRecord,
|
|
&$links,
|
|
$oldRevisionRecord,
|
|
$userIdentity
|
|
) {
|
|
$this->insertThankLink( $revisionRecord,
|
|
$links, $userIdentity );
|
|
}
|
|
|
|
/**
|
|
* Handler for the DiffTools hook
|
|
*
|
|
* @param RevisionRecord $revisionRecord
|
|
* @param array &$links
|
|
* @param RevisionRecord|null $oldRevisionRecord
|
|
* @param UserIdentity $userIdentity
|
|
*/
|
|
public function onDiffTools(
|
|
$revisionRecord,
|
|
&$links,
|
|
$oldRevisionRecord,
|
|
$userIdentity
|
|
) {
|
|
// Don't allow thanking for a diff that includes multiple revisions
|
|
// This does a query that is too expensive for history rows (T284274)
|
|
$previous = $this->revisionLookup->getPreviousRevision( $revisionRecord );
|
|
if ( $oldRevisionRecord && $previous &&
|
|
$previous->getId() !== $oldRevisionRecord->getId()
|
|
) {
|
|
return;
|
|
}
|
|
|
|
$this->insertThankLink( $revisionRecord,
|
|
$links, $userIdentity, true );
|
|
}
|
|
|
|
/**
|
|
* Insert a 'thank' link into revision interface, if the user is allowed to thank.
|
|
*
|
|
* @param RevisionRecord $revisionRecord RevisionRecord object to add the thank link for
|
|
* @param array &$links Links to add to the revision interface
|
|
* @param UserIdentity $userIdentity The user performing the thanks.
|
|
* @param bool $isPrimaryButton whether the link/button should be progressive
|
|
*/
|
|
private function insertThankLink(
|
|
RevisionRecord $revisionRecord,
|
|
array &$links,
|
|
UserIdentity $userIdentity,
|
|
bool $isPrimaryButton = false
|
|
) {
|
|
$recipient = $revisionRecord->getUser();
|
|
if ( $recipient === null ) {
|
|
// Cannot see the user
|
|
return;
|
|
}
|
|
|
|
$user = $this->userFactory->newFromUserIdentity( $userIdentity );
|
|
|
|
// Don't let users thank themselves.
|
|
// Exclude anonymous users.
|
|
// Exclude temp users (T345679)
|
|
// Exclude users who are blocked.
|
|
// Check whether bots are allowed to receive thanks.
|
|
// Don't allow thanking for a diff that includes multiple revisions
|
|
// Check whether we have a revision id to link to
|
|
if ( $user->isNamed()
|
|
&& !$userIdentity->equals( $recipient )
|
|
&& !$this->isUserBlockedFromTitle( $user, $revisionRecord->getPageAsLinkTarget() )
|
|
&& !self::isUserBlockedFromThanks( $user )
|
|
&& self::canReceiveThanks( $this->config, $this->userFactory, $recipient )
|
|
&& !$revisionRecord->isDeleted( RevisionRecord::DELETED_TEXT )
|
|
&& $revisionRecord->getId() !== 0
|
|
) {
|
|
$links[] = $this->generateThankElement(
|
|
$revisionRecord->getId(),
|
|
$user,
|
|
$recipient,
|
|
'revision',
|
|
$isPrimaryButton
|
|
);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Check whether the user is blocked from the title associated with the revision.
|
|
*
|
|
* This queries the replicas for a block; if 'no block' is incorrectly reported, it
|
|
* will be caught by ApiThank::dieOnUserBlockedFromTitle when the user attempts to thank.
|
|
*
|
|
* @param User $user
|
|
* @param LinkTarget $title
|
|
* @return bool
|
|
*/
|
|
private function isUserBlockedFromTitle( User $user, LinkTarget $title ) {
|
|
return $this->permissionManager->isBlockedFrom( $user, $title, true );
|
|
}
|
|
|
|
/**
|
|
* Check whether the user is blocked from giving thanks.
|
|
*
|
|
* @param User $user
|
|
* @return bool
|
|
*/
|
|
private static function isUserBlockedFromThanks( User $user ) {
|
|
$block = $user->getBlock();
|
|
return $block && ( $block->isSitewide() || $block->appliesToRight( 'thanks' ) );
|
|
}
|
|
|
|
/**
|
|
* Check whether a user is allowed to receive thanks or not
|
|
*
|
|
* @param Config $config
|
|
* @param UserFactory $userFactory
|
|
* @param UserIdentity $user Recipient
|
|
* @return bool true if allowed, false if not
|
|
*/
|
|
public static function canReceiveThanks(
|
|
Config $config,
|
|
UserFactory $userFactory,
|
|
UserIdentity $user
|
|
) {
|
|
$legacyUser = $userFactory->newFromUserIdentity( $user );
|
|
if ( !$user->isRegistered() || $legacyUser->isSystemUser() ) {
|
|
return false;
|
|
}
|
|
|
|
if ( !$config->get( 'ThanksSendToBots' ) &&
|
|
$legacyUser->isBot()
|
|
) {
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Helper for self::insertThankLink
|
|
* Creates either a thank link or thanked span based on users session
|
|
* @param int $id Revision or log ID to generate the thank element for.
|
|
* @param User $sender User who sends thanks notification.
|
|
* @param UserIdentity $recipient User who receives thanks notification.
|
|
* @param string $type Either 'revision' or 'log'.
|
|
* @param bool $isPrimaryButton whether the link/button should be progressive
|
|
* @return string
|
|
*/
|
|
protected function generateThankElement(
|
|
$id, User $sender, UserIdentity $recipient, $type = 'revision',
|
|
bool $isPrimaryButton = false
|
|
) {
|
|
$useCodex = RequestContext::getMain()->getSkin()->getSkinName() === 'minerva';
|
|
// Check if the user has already thanked for this revision or log entry.
|
|
// Session keys are backwards-compatible, and are also used in the ApiCoreThank class.
|
|
$sessionKey = ( $type === 'revision' ) ? $id : $type . $id;
|
|
$class = $useCodex ? 'cdx-button cdx-button--fake-button cdx-button--fake-button--enabled' : '';
|
|
if ( $isPrimaryButton && $useCodex ) {
|
|
$class .= ' cdx-button--weight-primary cdx-button--action-progressive';
|
|
}
|
|
if ( $sender->getRequest()->getSessionData( "thanks-thanked-$sessionKey" ) ) {
|
|
$class .= ' mw-thanks-thanked';
|
|
|
|
return Html::element(
|
|
'span',
|
|
[ 'class' => $class ],
|
|
wfMessage( 'thanks-thanked', $sender->getName(), $recipient->getName() )->text()
|
|
);
|
|
}
|
|
|
|
// Add 'thank' link
|
|
$tooltip = wfMessage( 'thanks-thank-tooltip' )
|
|
->params( $sender->getName(), $recipient->getName() )
|
|
->text();
|
|
|
|
$class .= ' mw-thanks-thank-link';
|
|
$subpage = ( $type === 'revision' ) ? '' : 'Log/';
|
|
return Html::element(
|
|
'a',
|
|
[
|
|
'class' => $class,
|
|
'href' => SpecialPage::getTitleFor( 'Thanks', $subpage . $id )->getFullURL(),
|
|
'title' => $tooltip,
|
|
'data-' . $type . '-id' => $id,
|
|
'data-recipient-gender' => $this->genderCache->getGenderOf( $recipient->getName(), __METHOD__ ),
|
|
],
|
|
wfMessage( 'thanks-thank', $sender->getName(), $recipient->getName() )->text()
|
|
);
|
|
}
|
|
|
|
/**
|
|
* @param OutputPage $outputPage The OutputPage to add the module to.
|
|
*/
|
|
protected function addThanksModule( OutputPage $outputPage ) {
|
|
$confirmationRequired = $this->config->get( 'ThanksConfirmationRequired' );
|
|
$outputPage->addModules( [ 'ext.thanks.corethank' ] );
|
|
$outputPage->addJsConfigVars( 'thanks-confirmation-required', $confirmationRequired );
|
|
}
|
|
|
|
/**
|
|
* Handler for PageHistoryBeforeList hook.
|
|
* @see https://www.mediawiki.org/wiki/Manual:Hooks/PageHistoryBeforeList
|
|
*
|
|
* @param Article $page Not used
|
|
* @param IContextSource $context RequestContext object
|
|
*/
|
|
public function onPageHistoryBeforeList( $page, $context ) {
|
|
if ( $context->getUser()->isRegistered() ) {
|
|
$this->addThanksModule( $context->getOutput() );
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Handler for DifferenceEngineViewHeader hook.
|
|
* @see https://www.mediawiki.org/wiki/Manual:Hooks/DifferenceEngineViewHeader
|
|
* @param DifferenceEngine $diff DifferenceEngine object that's calling.
|
|
*/
|
|
public function onDifferenceEngineViewHeader( $diff ) {
|
|
if ( $diff->getUser()->isRegistered() ) {
|
|
$this->addThanksModule( $diff->getOutput() );
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Add Thanks events to Echo
|
|
*
|
|
* @param array &$notifications array of Echo notifications
|
|
* @param array &$notificationCategories array of Echo notification categories
|
|
* @param array &$icons array of icon details
|
|
*/
|
|
public function onBeforeCreateEchoEvent(
|
|
array &$notifications, array &$notificationCategories, array &$icons
|
|
) {
|
|
$notificationCategories['edit-thank'] = [
|
|
'priority' => 3,
|
|
'tooltip' => 'echo-pref-tooltip-edit-thank',
|
|
];
|
|
|
|
$notifications['edit-thank'] = [
|
|
// The following message is generated by the category name:
|
|
// * echo-category-title-edit-thank
|
|
'category' => 'edit-thank',
|
|
'group' => 'positive',
|
|
'section' => 'message',
|
|
'presentation-model' => EchoCoreThanksPresentationModel::class,
|
|
'bundle' => [
|
|
'web' => true,
|
|
'expandable' => true,
|
|
],
|
|
EchoAttributeManager::ATTR_LOCATORS => [
|
|
[
|
|
[ EchoUserLocator::class, 'locateFromEventExtra' ],
|
|
[ 'thanked-user-id' ]
|
|
],
|
|
],
|
|
];
|
|
|
|
if ( ExtensionRegistry::getInstance()->isLoaded( 'Flow' ) ) {
|
|
$notifications['flow-thank'] = [
|
|
'category' => 'edit-thank',
|
|
'group' => 'positive',
|
|
'section' => 'message',
|
|
'presentation-model' => EchoFlowThanksPresentationModel::class,
|
|
'bundle' => [
|
|
'web' => true,
|
|
'expandable' => true,
|
|
],
|
|
EchoAttributeManager::ATTR_LOCATORS => [
|
|
[
|
|
[ EchoUserLocator::class, 'locateFromEventExtra' ],
|
|
[ 'thanked-user-id' ]
|
|
],
|
|
],
|
|
];
|
|
}
|
|
|
|
$icons['thanks'] = [
|
|
'path' => [
|
|
'ltr' => 'Thanks/modules/userTalk-constructive-ltr.svg',
|
|
'rtl' => 'Thanks/modules/userTalk-constructive-rtl.svg'
|
|
]
|
|
];
|
|
}
|
|
|
|
/**
|
|
* Handler for LocalUserCreated hook
|
|
* @see https://www.mediawiki.org/wiki/Manual:Hooks/LocalUserCreated
|
|
* @param User $user User object that was created.
|
|
* @param bool $autocreated True when account was auto-created
|
|
*/
|
|
public function onLocalUserCreated( $user, $autocreated ) {
|
|
// New users get echo preferences set that are not the default settings for existing users.
|
|
// Specifically, new users are opted into email notifications for thanks.
|
|
if ( !$user->isTemp() && !$autocreated ) {
|
|
$this->userOptionsManager->setOption( $user, 'echo-subscriptions-email-edit-thank', true );
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Handler for GetLogTypesOnUser.
|
|
* So users can just type in a username for target and it'll work.
|
|
* @link https://www.mediawiki.org/wiki/Manual:Hooks/GetLogTypesOnUser
|
|
* @param string[] &$types The list of log types, to add to.
|
|
*/
|
|
public function onGetLogTypesOnUser( &$types ) {
|
|
$types[] = 'thanks';
|
|
}
|
|
|
|
public function onGetAllBlockActions( &$actions ) {
|
|
$actions[ 'thanks' ] = 100;
|
|
}
|
|
|
|
/**
|
|
* Handler for BeforePageDisplay. Inserts javascript to enhance thank
|
|
* links from static urls to in-page dialogs along with reloading
|
|
* the previously thanked state.
|
|
* @link https://www.mediawiki.org/wiki/Manual:Hooks/BeforePageDisplay
|
|
* @param OutputPage $out OutputPage object
|
|
* @param Skin $skin The skin in use.
|
|
*/
|
|
public function onBeforePageDisplay( $out, $skin ): void {
|
|
$title = $out->getTitle();
|
|
// Add to Flow boards.
|
|
if ( $title instanceof Title && $title->hasContentModel( 'flow-board' ) ) {
|
|
$out->addModules( 'ext.thanks.flowthank' );
|
|
}
|
|
// Add to special pages where thank links appear
|
|
if (
|
|
$title->isSpecial( 'Log' ) ||
|
|
$title->isSpecial( 'Contributions' ) ||
|
|
$title->isSpecial( 'DeletedContributions' ) ||
|
|
$title->isSpecial( 'Recentchanges' ) ||
|
|
$title->isSpecial( 'Recentchangeslinked' ) ||
|
|
$title->isSpecial( 'Watchlist' )
|
|
) {
|
|
$this->addThanksModule( $out );
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Conditionally load API module 'flowthank' depending on whether or not
|
|
* Flow is installed.
|
|
*
|
|
* @param ApiModuleManager $moduleManager Module manager instance
|
|
*/
|
|
public function onApiMain__moduleManager( $moduleManager ) {
|
|
if ( ExtensionRegistry::getInstance()->isLoaded( 'Flow' ) ) {
|
|
$moduleManager->addModule(
|
|
'flowthank',
|
|
'action',
|
|
[
|
|
"class" => ApiFlowThank::class,
|
|
"services" => [
|
|
"PermissionManager",
|
|
"ThanksLogStore"
|
|
]
|
|
]
|
|
);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Handler for EchoGetBundleRule hook, which defines the bundle rules for each notification.
|
|
*
|
|
* @param Event $event The event being notified.
|
|
* @param string &$bundleString Determines how the notification should be bundled.
|
|
*/
|
|
public function onEchoGetBundleRules( Event $event, string &$bundleString ) {
|
|
switch ( $event->getType() ) {
|
|
case 'edit-thank':
|
|
$bundleString = 'edit-thank';
|
|
// Try to get either the revid or logid parameter.
|
|
$revOrLogId = $event->getExtraParam( 'logid' );
|
|
if ( $revOrLogId ) {
|
|
// avoid collision with revision ids
|
|
$revOrLogId = 'log' . $revOrLogId;
|
|
} else {
|
|
$revOrLogId = $event->getExtraParam( 'revid' );
|
|
}
|
|
if ( $revOrLogId ) {
|
|
$bundleString .= $revOrLogId;
|
|
}
|
|
break;
|
|
case 'flow-thank':
|
|
$bundleString = 'flow-thank';
|
|
$postId = $event->getExtraParam( 'post-id' );
|
|
if ( $postId ) {
|
|
$bundleString .= $postId;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Insert a 'thank' link into the log interface, if the user is allowed to thank.
|
|
*
|
|
* @link https://www.mediawiki.org/wiki/Manual:Hooks/LogEventsListLineEnding
|
|
* @param LogEventsList $page The log events list.
|
|
* @param string &$ret The line ending HTML, to modify.
|
|
* @param DatabaseLogEntry $entry The log entry.
|
|
* @param string[] &$classes CSS classes to add to the line.
|
|
* @param string[] &$attribs HTML attributes to add to the line.
|
|
* @throws ConfigException
|
|
*/
|
|
public function onLogEventsListLineEnding(
|
|
$page, &$ret, $entry, &$classes, &$attribs
|
|
) {
|
|
$user = $page->getUser();
|
|
|
|
// Don't thank if anonymous or blocked or if user is deleted from the log entry
|
|
if (
|
|
$user->isAnon()
|
|
|| $entry->isDeleted( LogPage::DELETED_USER )
|
|
|| $this->isUserBlockedFromTitle( $user, $entry->getTarget() )
|
|
|| self::isUserBlockedFromThanks( $user )
|
|
) {
|
|
return;
|
|
}
|
|
|
|
// Make sure this log type is allowed.
|
|
$allowedLogTypes = $this->config->get( 'ThanksAllowedLogTypes' );
|
|
if ( !in_array( $entry->getType(), $allowedLogTypes )
|
|
&& !in_array( $entry->getType() . '/' . $entry->getSubtype(), $allowedLogTypes ) ) {
|
|
return;
|
|
}
|
|
|
|
// Don't thank if no recipient,
|
|
// or if recipient is the current user or unable to receive thanks.
|
|
// Don't check for deleted revision (this avoids extraneous queries from Special:Log).
|
|
|
|
$recipient = $entry->getPerformerIdentity();
|
|
if ( $recipient->getId() === $user->getId() ||
|
|
!self::canReceiveThanks( $this->config, $this->userFactory, $recipient )
|
|
) {
|
|
return;
|
|
}
|
|
|
|
// Create thank link either for the revision (if there is an associated revision ID)
|
|
// or the log entry.
|
|
$type = $entry->getAssociatedRevId() ? 'revision' : 'log';
|
|
$id = $entry->getAssociatedRevId() ?: $entry->getId();
|
|
$thankLink = $this->generateThankElement( $id, $user, $recipient, $type );
|
|
|
|
// Add parentheses to match what's done with Thanks in revision lists and diff displays.
|
|
$ret .= ' ' . wfMessage( 'parentheses' )->rawParams( $thankLink )->escaped();
|
|
}
|
|
}
|