2022-12-07 19:31:14 +00:00
|
|
|
<?php
|
|
|
|
namespace MediaWiki\Skins\Vector\Components;
|
|
|
|
|
2024-01-02 19:54:08 +00:00
|
|
|
use MediaWiki\Linker\Linker;
|
2023-02-01 23:38:44 +00:00
|
|
|
use MediaWiki\Skin\SkinComponentLink;
|
2024-01-02 19:54:08 +00:00
|
|
|
use MediaWiki\Title\MalformedTitleException;
|
2023-08-19 04:23:07 +00:00
|
|
|
use MediaWiki\Title\Title;
|
2024-03-13 20:59:40 +00:00
|
|
|
use MediaWiki\User\UserIdentity;
|
2023-01-06 01:45:38 +00:00
|
|
|
use Message;
|
|
|
|
use MessageLocalizer;
|
|
|
|
|
2022-12-07 19:31:14 +00:00
|
|
|
/**
|
|
|
|
* VectorComponentUserLinks component
|
|
|
|
*/
|
|
|
|
class VectorComponentUserLinks implements VectorComponent {
|
2023-11-08 01:01:59 +00:00
|
|
|
|
|
|
|
private const BUTTON_CLASSES = 'cdx-button cdx-button--fake-button '
|
|
|
|
. 'cdx-button--fake-button--enabled cdx-button--weight-quiet';
|
|
|
|
private const ICON_ONLY_BUTTON_CLASS = 'cdx-button--icon-only';
|
|
|
|
|
2023-01-06 01:45:38 +00:00
|
|
|
/** @var MessageLocalizer */
|
|
|
|
private $localizer;
|
2024-03-13 20:59:40 +00:00
|
|
|
/** @var UserIdentity */
|
2023-01-06 01:45:38 +00:00
|
|
|
private $user;
|
2023-02-01 23:38:44 +00:00
|
|
|
/** @var array */
|
|
|
|
private $portletData;
|
|
|
|
/** @var array */
|
|
|
|
private $linkOptions;
|
2023-02-03 00:35:07 +00:00
|
|
|
/** @var string */
|
|
|
|
private $userIcon;
|
2023-01-06 01:45:38 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @param MessageLocalizer $localizer
|
2024-03-13 20:59:40 +00:00
|
|
|
* @param UserIdentity $user
|
2023-02-01 23:38:44 +00:00
|
|
|
* @param array $portletData
|
|
|
|
* @param array $linkOptions
|
2023-02-03 00:35:07 +00:00
|
|
|
* @param string $userIcon that represents the current type of user
|
2023-01-06 01:45:38 +00:00
|
|
|
*/
|
|
|
|
public function __construct(
|
|
|
|
MessageLocalizer $localizer,
|
2024-03-13 20:59:40 +00:00
|
|
|
UserIdentity $user,
|
2023-02-01 23:38:44 +00:00
|
|
|
array $portletData,
|
2023-02-03 00:35:07 +00:00
|
|
|
array $linkOptions,
|
2023-11-21 17:01:04 +00:00
|
|
|
string $userIcon = 'userAvatar'
|
2023-01-06 01:45:38 +00:00
|
|
|
) {
|
|
|
|
$this->localizer = $localizer;
|
|
|
|
$this->user = $user;
|
2023-02-01 23:38:44 +00:00
|
|
|
$this->portletData = $portletData;
|
|
|
|
$this->linkOptions = $linkOptions;
|
2023-02-03 00:35:07 +00:00
|
|
|
$this->userIcon = $userIcon;
|
2023-01-06 01:45:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param string $key
|
|
|
|
* @return Message
|
|
|
|
*/
|
|
|
|
private function msg( $key ): Message {
|
|
|
|
return $this->localizer->msg( $key );
|
|
|
|
}
|
2022-12-07 19:31:14 +00:00
|
|
|
|
|
|
|
/**
|
2023-02-01 23:38:44 +00:00
|
|
|
* @param bool $isDefaultAnonUserLinks
|
|
|
|
* @param bool $isAnonEditorLinksEnabled
|
|
|
|
* @return VectorComponentDropdown
|
2022-12-07 19:31:14 +00:00
|
|
|
*/
|
2023-02-01 23:38:44 +00:00
|
|
|
private function getDropdown( $isDefaultAnonUserLinks, $isAnonEditorLinksEnabled ) {
|
2023-01-06 01:45:38 +00:00
|
|
|
$user = $this->user;
|
|
|
|
$isAnon = !$user->isRegistered();
|
2023-02-01 23:38:44 +00:00
|
|
|
|
|
|
|
$class = 'vector-user-menu';
|
2023-04-27 20:39:40 +00:00
|
|
|
$class .= ' vector-button-flush-right';
|
2023-02-01 23:38:44 +00:00
|
|
|
$class .= !$isAnon ?
|
2023-01-06 01:45:38 +00:00
|
|
|
' vector-user-menu-logged-in' :
|
|
|
|
' vector-user-menu-logged-out';
|
|
|
|
|
2023-02-01 23:38:44 +00:00
|
|
|
// Hide entire user links dropdown on larger viewports if it only contains
|
|
|
|
// create account & login link, which are only shown on smaller viewports
|
|
|
|
if ( $isAnon && $isDefaultAnonUserLinks && !$isAnonEditorLinksEnabled ) {
|
|
|
|
$class .= ' user-links-collapsible-item';
|
|
|
|
}
|
|
|
|
|
2023-01-06 01:45:38 +00:00
|
|
|
$tooltip = '';
|
2023-02-03 00:35:07 +00:00
|
|
|
$icon = $this->userIcon;
|
|
|
|
if ( $icon === '' ) {
|
2023-01-06 01:45:38 +00:00
|
|
|
$icon = 'ellipsis';
|
|
|
|
// T287494 We use tooltip messages to provide title attributes on hover over certain menu icons.
|
|
|
|
// For modern Vector, the "tooltip-p-personal" key is set to "User menu" which is appropriate for
|
|
|
|
// the user icon (dropdown indicator for user links menu) for logged-in users.
|
|
|
|
// This overrides the tooltip for the user links menu icon which is an ellipsis for anonymous users.
|
|
|
|
$tooltip = Linker::tooltip( 'vector-anon-user-menu-title' ) ?? '';
|
|
|
|
}
|
2023-02-01 23:38:44 +00:00
|
|
|
|
|
|
|
return new VectorComponentDropdown(
|
2023-02-16 17:06:22 +00:00
|
|
|
'vector-user-links-dropdown', $this->msg( 'personaltools' )->text(), $class, $icon, $tooltip
|
2023-01-06 01:45:38 +00:00
|
|
|
);
|
2023-02-01 23:38:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param bool $isDefaultAnonUserLinks
|
|
|
|
* @param bool $isAnonEditorLinksEnabled
|
|
|
|
* @return array
|
|
|
|
*/
|
2023-06-30 21:49:17 +00:00
|
|
|
private function getMenus( $isDefaultAnonUserLinks, $isAnonEditorLinksEnabled ) {
|
2023-02-01 23:38:44 +00:00
|
|
|
$user = $this->user;
|
|
|
|
$isAnon = !$user->isRegistered();
|
|
|
|
$portletData = $this->portletData;
|
|
|
|
|
|
|
|
// Hide default user menu on larger viewports if it only contains
|
|
|
|
// create account & login link, which are only shown on smaller viewports
|
|
|
|
// FIXME: Replace array_merge with an add class helper function
|
|
|
|
$userMenuClass = $portletData[ 'data-user-menu' ][ 'class' ];
|
|
|
|
$userMenuClass = $isAnon && $isDefaultAnonUserLinks ?
|
|
|
|
$userMenuClass . ' user-links-collapsible-item' : $userMenuClass;
|
|
|
|
$dropdownMenus = [
|
|
|
|
new VectorComponentMenu( [
|
|
|
|
'label' => null,
|
|
|
|
'class' => $userMenuClass
|
|
|
|
] + $portletData[ 'data-user-menu' ] )
|
|
|
|
];
|
|
|
|
|
|
|
|
if ( $isAnon ) {
|
|
|
|
// T317789: The `anontalk` and `anoncontribs` links will not be added to
|
|
|
|
// the menu if `$wgGroupPermissions['*']['edit']` === false which can
|
|
|
|
// leave the menu empty due to our removal of other user menu items in
|
|
|
|
// `Hooks::updateUserLinksDropdownItems`. In this case, we do not want
|
|
|
|
// to render the anon "learn more" link.
|
|
|
|
if ( $isAnonEditorLinksEnabled ) {
|
2023-06-28 14:20:06 +00:00
|
|
|
$anonUserMenuData = $portletData[ 'data-user-menu-anon-editor' ];
|
|
|
|
try {
|
|
|
|
$anonEditorLabelLinkData = [
|
|
|
|
'text' => $this->msg( 'vector-anon-user-menu-pages-learn' )->text(),
|
|
|
|
'href' => Title::newFromTextThrow( $this->msg( 'vector-intro-page' )->text() )->getLocalURL(),
|
|
|
|
'aria-label' => $this->msg( 'vector-anon-user-menu-pages-label' )->text(),
|
|
|
|
];
|
|
|
|
$anonEditorLabelLink = new SkinComponentLink(
|
|
|
|
'', $anonEditorLabelLinkData, $this->localizer, $this->linkOptions
|
|
|
|
);
|
|
|
|
$anonEditorLabelLinkHtml = $anonEditorLabelLink->getTemplateData()[ 'html' ];
|
|
|
|
$anonUserMenuData['html-label'] = $this->msg( 'vector-anon-user-menu-pages' )->escaped() .
|
|
|
|
" " . $anonEditorLabelLinkHtml;
|
|
|
|
$anonUserMenuData['label'] = null;
|
|
|
|
} catch ( MalformedTitleException $e ) {
|
|
|
|
// ignore (T340220)
|
|
|
|
}
|
|
|
|
$dropdownMenus[] = new VectorComponentMenu( $anonUserMenuData );
|
2023-02-01 23:38:44 +00:00
|
|
|
}
|
|
|
|
} else {
|
2023-06-23 17:06:43 +00:00
|
|
|
// Logout isn't enabled for temp users, who are considered still considered registered
|
2023-02-03 00:35:07 +00:00
|
|
|
$isLogoutLinkEnabled = isset( $portletData[ 'data-user-menu-logout' ][ 'is-empty' ] ) &&
|
|
|
|
!$portletData[ 'data-user-menu-logout'][ 'is-empty' ];
|
|
|
|
if ( $isLogoutLinkEnabled ) {
|
2023-02-01 23:38:44 +00:00
|
|
|
$dropdownMenus[] = new VectorComponentMenu( [
|
|
|
|
'label' => null
|
|
|
|
] + $portletData[ 'data-user-menu-logout' ] );
|
|
|
|
}
|
2023-01-06 01:45:38 +00:00
|
|
|
}
|
|
|
|
|
2023-02-01 23:38:44 +00:00
|
|
|
return $dropdownMenus;
|
|
|
|
}
|
|
|
|
|
2023-11-08 01:01:59 +00:00
|
|
|
/**
|
|
|
|
* Strips icons from the menu.
|
|
|
|
*
|
|
|
|
* @param array $arrayListItems
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
private static function stripIcons( array $arrayListItems ) {
|
|
|
|
return array_map( static function ( $item ) {
|
|
|
|
$item['array-links'] = array_map( static function ( $link ) {
|
|
|
|
$link['icon'] = null;
|
|
|
|
return $link;
|
|
|
|
}, $item['array-links'] );
|
|
|
|
return $item;
|
|
|
|
}, $arrayListItems );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Converts links to button icons
|
|
|
|
*
|
|
|
|
* @param array $arrayListItems
|
|
|
|
* @param bool $iconOnlyButton whether label should be visible.
|
2023-11-13 19:16:23 +00:00
|
|
|
* @param array $exceptions list of names of items that should not be converted.
|
2023-11-08 01:01:59 +00:00
|
|
|
* @return array
|
|
|
|
*/
|
2023-11-13 19:16:23 +00:00
|
|
|
private static function makeLinksButtons( $arrayListItems, $iconOnlyButton = true, $exceptions = [] ) {
|
|
|
|
return array_map( static function ( $item ) use ( $iconOnlyButton, $exceptions ) {
|
|
|
|
if ( in_array( $item[ 'name'], $exceptions ) ) {
|
|
|
|
return $item;
|
|
|
|
}
|
2023-11-08 01:01:59 +00:00
|
|
|
$item['array-links'] = array_map( static function ( $link ) use ( $iconOnlyButton ) {
|
|
|
|
$link['array-attributes'] = array_map( static function ( $attribute ) use ( $iconOnlyButton ) {
|
|
|
|
if ( $attribute['key'] === 'class' ) {
|
|
|
|
$newClass = $attribute['value'] . ' ' . self::BUTTON_CLASSES;
|
|
|
|
if ( $iconOnlyButton ) {
|
|
|
|
$newClass .= ' ' . self::ICON_ONLY_BUTTON_CLASS;
|
|
|
|
}
|
|
|
|
$attribute['value'] = $newClass;
|
|
|
|
}
|
|
|
|
return $attribute;
|
|
|
|
}, $link['array-attributes'] );
|
|
|
|
return $link;
|
|
|
|
}, $item['array-links'] );
|
|
|
|
return $item;
|
|
|
|
}, $arrayListItems );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Makes all menu items collapsible at lower resolutions.
|
|
|
|
*
|
|
|
|
* @param array $arrayListItems
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
private static function makeItemsCollapsible( $arrayListItems ) {
|
|
|
|
return array_map( static function ( $item ) {
|
|
|
|
$item['class'] .= ' user-links-collapsible-item';
|
|
|
|
return $item;
|
|
|
|
}, $arrayListItems );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* What class should the overflow menu have?
|
|
|
|
*
|
|
|
|
* @param array $arrayListItems
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
private static function getOverflowMenuClass( $arrayListItems ) {
|
|
|
|
$overflowMenuClass = 'mw-portlet';
|
|
|
|
if ( count( $arrayListItems ) === 0 ) {
|
|
|
|
$overflowMenuClass .= ' emptyPortlet';
|
|
|
|
}
|
|
|
|
return $overflowMenuClass;
|
|
|
|
}
|
|
|
|
|
2023-02-01 23:38:44 +00:00
|
|
|
/**
|
|
|
|
* @inheritDoc
|
|
|
|
*/
|
|
|
|
public function getTemplateData(): array {
|
|
|
|
$portletData = $this->portletData;
|
|
|
|
|
|
|
|
$isDefaultAnonUserLinks = count( $portletData['data-user-menu']['array-items'] ) === 2;
|
|
|
|
$isAnonEditorLinksEnabled = isset( $portletData['data-user-menu-anon-editor']['is-empty'] )
|
|
|
|
&& !$portletData['data-user-menu-anon-editor']['is-empty'];
|
|
|
|
|
2023-11-08 01:01:59 +00:00
|
|
|
$userInterfacePreferences = $this->makeLinksButtons(
|
|
|
|
$this->makeItemsCollapsible(
|
|
|
|
$portletData[ 'data-user-interface-preferences' ]['array-items'] ?? []
|
|
|
|
),
|
|
|
|
false
|
|
|
|
);
|
|
|
|
$userPage = $this->makeItemsCollapsible(
|
|
|
|
$this->stripIcons( $portletData[ 'data-user-page' ]['array-items'] ?? [] )
|
|
|
|
);
|
|
|
|
$notifications = $this->makeLinksButtons(
|
2023-11-13 19:16:23 +00:00
|
|
|
$portletData[ 'data-notifications' ]['array-items'] ?? [],
|
|
|
|
true,
|
|
|
|
[ 'talk-alert' ]
|
2023-11-08 01:01:59 +00:00
|
|
|
);
|
|
|
|
|
|
|
|
$overflow = $this->makeItemsCollapsible(
|
|
|
|
array_map(
|
|
|
|
static function ( $item ) {
|
|
|
|
// Since we're creating duplicate icons
|
|
|
|
$item['id'] .= '-2';
|
|
|
|
// Restore icon removed in hooks.
|
|
|
|
if ( $item['name'] === 'watchlist' ) {
|
|
|
|
$item['icon'] = 'watchlist';
|
|
|
|
}
|
|
|
|
return $item;
|
|
|
|
},
|
|
|
|
// array_filter preserves keys so use array_values to restore array.
|
|
|
|
array_values(
|
|
|
|
array_filter(
|
|
|
|
$portletData['data-user-menu']['array-items'] ?? [],
|
|
|
|
static function ( $item ) {
|
|
|
|
// Only certain items get promoted to the overflow menu:
|
|
|
|
// * watchlist
|
|
|
|
// * login
|
|
|
|
// * create account
|
|
|
|
$name = $item['name'];
|
|
|
|
return in_array( $name, [ 'watchlist', 'createaccount', 'login', 'login-private' ] );
|
|
|
|
}
|
|
|
|
)
|
|
|
|
)
|
|
|
|
)
|
|
|
|
);
|
|
|
|
// Convert to buttons for logged in users.
|
|
|
|
// For anons these will remain as links.
|
|
|
|
// Note: This list is empty for temporary users currently.
|
2024-03-13 20:59:40 +00:00
|
|
|
if ( $this->user->isRegistered() ) {
|
2023-11-08 01:01:59 +00:00
|
|
|
$overflow = $this->makeLinksButtons( $overflow );
|
|
|
|
}
|
|
|
|
|
|
|
|
$preferencesMenu = new VectorComponentMenu( [
|
|
|
|
'id' => 'p-vector-user-menu-preferences',
|
|
|
|
'class' => self::getOverflowMenuClass( $userInterfacePreferences ),
|
|
|
|
'label' => null,
|
|
|
|
'html-items' => null,
|
|
|
|
'array-list-items' => $userInterfacePreferences,
|
|
|
|
] );
|
|
|
|
$userPageMenu = new VectorComponentMenu( [
|
|
|
|
'id' => 'p-vector-user-menu-userpage',
|
|
|
|
'class' => self::getOverflowMenuClass( $userPage ),
|
|
|
|
'label' => null,
|
|
|
|
'html-items' => null,
|
|
|
|
'array-list-items' => $userPage,
|
|
|
|
] );
|
|
|
|
$notificationsMenu = new VectorComponentMenu( [
|
|
|
|
'id' => 'p-vector-user-menu-notifications',
|
|
|
|
'class' => self::getOverflowMenuClass( $notifications ),
|
|
|
|
'label' => null,
|
|
|
|
'html-items' => null,
|
|
|
|
'array-list-items' => $notifications,
|
|
|
|
] );
|
2023-02-01 23:38:44 +00:00
|
|
|
$overflowMenu = new VectorComponentMenu( [
|
2023-11-08 01:01:59 +00:00
|
|
|
'id' => 'p-vector-user-menu-overflow',
|
|
|
|
'class' => self::getOverflowMenuClass( $overflow ),
|
2023-02-01 23:38:44 +00:00
|
|
|
'label' => null,
|
2023-11-08 01:01:59 +00:00
|
|
|
'html-items' => null,
|
|
|
|
'array-list-items' => $overflow,
|
|
|
|
] );
|
2023-02-01 23:38:44 +00:00
|
|
|
|
|
|
|
return [
|
2024-01-12 00:25:40 +00:00
|
|
|
'is-wide' => array_filter(
|
|
|
|
[ $overflow, $notifications, $userPage, $userInterfacePreferences ]
|
|
|
|
) !== [],
|
2023-11-08 01:01:59 +00:00
|
|
|
'data-user-links-notifications' => $notificationsMenu->getTemplateData(),
|
|
|
|
'data-user-links-overflow' => $overflowMenu->getTemplateData(),
|
|
|
|
'data-user-links-preferences' => $preferencesMenu->getTemplateData(),
|
|
|
|
'data-user-links-user-page' => $userPageMenu->getTemplateData(),
|
2023-02-01 23:38:44 +00:00
|
|
|
'data-user-links-dropdown' => $this->getDropdown( $isDefaultAnonUserLinks, $isAnonEditorLinksEnabled )
|
|
|
|
->getTemplateData(),
|
2023-06-30 21:49:17 +00:00
|
|
|
'data-user-links-menus' => array_map( static function ( $menu ) {
|
2023-02-01 23:38:44 +00:00
|
|
|
return $menu->getTemplateData();
|
2023-06-30 21:49:17 +00:00
|
|
|
}, $this->getMenus( $isDefaultAnonUserLinks, $isAnonEditorLinksEnabled ) ),
|
2023-01-06 01:45:38 +00:00
|
|
|
];
|
2022-12-07 19:31:14 +00:00
|
|
|
}
|
|
|
|
}
|