From 4b3aa8d05e50f3db2ac21b9e50236b4339975b19 Mon Sep 17 00:00:00 2001 From: John Hsu Date: Wed, 8 Nov 2023 14:39:51 -0800 Subject: [PATCH] HACK Vector skin hack to hide redundant login btn The user links menu ("..." on the upper right) has a redundant login button inside it. There's a login button in the overflow menu next to it that's more visible. I've set this login button to be uncollapsible, so it should always be visible even on small screens. So we just need to hide the user links menu for unsigned in users. --- Dockerfile | 1 + skins/Vector/includes/Hooks.php | 6 +- skins/Vector/includes/SkinVector.php | 942 +++++++++++++++++++++++++++ 3 files changed, 948 insertions(+), 1 deletion(-) create mode 100644 skins/Vector/includes/SkinVector.php diff --git a/Dockerfile b/Dockerfile index e1e5119..1aec6ec 100644 --- a/Dockerfile +++ b/Dockerfile @@ -115,6 +115,7 @@ COPY ./extensions/DynamicPageList/includes/Query.php /var/www/html/extensions/Dy # TODO: Remove if >REL1_40, as this is a backport from Vector REL1_40 # Add login button next to "..." menu in top-right corner COPY skins/Vector/includes/Hooks.php /var/www/html/skins/Vector/includes/Hooks.php +COPY skins/Vector/includes/SkinVector.php /var/www/html/skins/Vector/includes/SkinVector.php # composer.local.json merges in composer.json from caliper extension, so we # need to run composer update after getting the extensions. diff --git a/skins/Vector/includes/Hooks.php b/skins/Vector/includes/Hooks.php index 299e72c..a0a1cfb 100644 --- a/skins/Vector/includes/Hooks.php +++ b/skins/Vector/includes/Hooks.php @@ -322,12 +322,16 @@ private static function updateUserLinksOverflowItems( $sk, &$content_navigation 'icon' => '', ] ); } + # TODO: remove customization after 1.40+ + # ubc custom: backport from 1.40, add login button to the overflow menu + # we're hiding the user links button, so the login button cannot be + # allowed to collapse if ( isset( $content_navigation['user-menu']['login'] ) ) { $content_navigation[$overflow]['login'] = array_merge( $content_navigation['user-menu']['login'], [ 'id' => 'pt-login-2', 'button' => true, - 'collapsible' => true, + 'collapsible' => false, // Remove icon //'icon' => '', ] ); diff --git a/skins/Vector/includes/SkinVector.php b/skins/Vector/includes/SkinVector.php new file mode 100644 index 0000000..6823a33 --- /dev/null +++ b/skins/Vector/includes/SkinVector.php @@ -0,0 +1,942 @@ + '#', + 'id' => 'ca-talk-sticky-header', + 'event' => 'talk-sticky-header', + 'icon' => 'wikimedia-speechBubbles', + 'is-quiet' => true, + 'tabindex' => '-1', + 'class' => 'sticky-header-icon' + ]; + private const SUBJECT_ICON = [ + 'href' => '#', + 'id' => 'ca-subject-sticky-header', + 'event' => 'subject-sticky-header', + 'icon' => 'wikimedia-article', + 'is-quiet' => true, + 'tabindex' => '-1', + 'class' => 'sticky-header-icon' + ]; + private const HISTORY_ICON = [ + 'href' => '#', + 'id' => 'ca-history-sticky-header', + 'event' => 'history-sticky-header', + 'icon' => 'wikimedia-history', + 'is-quiet' => true, + 'tabindex' => '-1', + 'class' => 'sticky-header-icon' + ]; + // Event and icon will be updated depending on watchstar state + private const WATCHSTAR_ICON = [ + 'href' => '#', + 'id' => 'ca-watchstar-sticky-header', + 'event' => 'watch-sticky-header', + 'icon' => 'wikimedia-star', + 'is-quiet' => true, + 'tabindex' => '-1', + 'class' => 'sticky-header-icon mw-watchlink' + ]; + private const EDIT_VE_ICON = [ + 'href' => '#', + 'id' => 'ca-ve-edit-sticky-header', + 'event' => 've-edit-sticky-header', + 'icon' => 'wikimedia-edit', + 'is-quiet' => true, + 'tabindex' => '-1', + 'class' => 'sticky-header-icon' + ]; + private const EDIT_WIKITEXT_ICON = [ + 'href' => '#', + 'id' => 'ca-edit-sticky-header', + 'event' => 'wikitext-edit-sticky-header', + 'icon' => 'wikimedia-wikiText', + 'is-quiet' => true, + 'tabindex' => '-1', + 'class' => 'sticky-header-icon' + ]; + private const EDIT_PROTECTED_ICON = [ + 'href' => '#', + 'id' => 'ca-viewsource-sticky-header', + 'event' => 've-edit-protected-sticky-header', + 'icon' => 'wikimedia-editLock', + 'is-quiet' => true, + 'tabindex' => '-1', + 'class' => 'sticky-header-icon' + ]; + private const SEARCH_SHOW_THUMBNAIL_CLASS = 'vector-search-box-show-thumbnail'; + private const SEARCH_AUTO_EXPAND_WIDTH_CLASS = 'vector-search-box-auto-expand-width'; + private const CLASS_PROGRESSIVE = 'mw-ui-progressive'; + + /** + * T243281: Code used to track clicks to opt-out link. + * + * The "vct" substring is used to describe the newest "Vector" (non-legacy) + * feature. The "w" describes the web platform. The "1" describes the version + * of the feature. + * + * @see https://wikitech.wikimedia.org/wiki/Provenance + * @var string + */ + private const OPT_OUT_LINK_TRACKING_CODE = 'vctw1'; + + abstract protected function isLegacy(): bool; + + /** + * Calls getLanguages with caching. + * @return array + */ + protected function getLanguagesCached(): array { + if ( $this->languages === null ) { + $this->languages = $this->getLanguages(); + } + return $this->languages; + } + + /** + * This should be upstreamed to the Skin class in core once the logic is finalized. + * Returns false if the page is a special page without any languages, or if an action + * other than view is being used. + * @return bool + */ + private function canHaveLanguages(): bool { + if ( $this->getContext()->getActionName() !== 'view' ) { + return false; + } + $title = $this->getTitle(); + // Defensive programming - if a special page has added languages explicitly, best to show it. + if ( $title && $title->isSpecialPage() && empty( $this->getLanguagesCached() ) ) { + return false; + } + return true; + } + + /** + * @param string $location Either 'top' or 'bottom' is accepted. + * @return bool + */ + protected function isLanguagesInContentAt( $location ) { + if ( !$this->canHaveLanguages() ) { + return false; + } + $featureManager = VectorServices::getFeatureManager(); + $inContent = $featureManager->isFeatureEnabled( + Constants::FEATURE_LANGUAGE_IN_HEADER + ); + $isMainPage = $this->getTitle() ? $this->getTitle()->isMainPage() : false; + + switch ( $location ) { + case 'top': + return $isMainPage ? $inContent && $featureManager->isFeatureEnabled( + Constants::FEATURE_LANGUAGE_IN_MAIN_PAGE_HEADER + ) : $inContent; + case 'bottom': + return $inContent && $isMainPage && !$featureManager->isFeatureEnabled( + Constants::FEATURE_LANGUAGE_IN_MAIN_PAGE_HEADER + ); + default: + throw new RuntimeException( 'unknown language button location' ); + } + } + + /** + * Whether or not the languages are out of the sidebar and in the content either at + * the top or the bottom. + * @return bool + */ + private function isLanguagesInContent() { + return $this->isLanguagesInContentAt( 'top' ) || $this->isLanguagesInContentAt( 'bottom' ); + } + + /** + * Show the ULS button if it's modern Vector, languages in header is enabled, + * and the ULS extension is enabled. Hide it otherwise. + * There is no point in showing the language button if ULS extension is unavailable + * as there is no ways to add languages without it. + * @return bool + */ + protected function shouldHideLanguages() { + return $this->isLegacy() || !$this->isLanguagesInContent() || !$this->isULSExtensionEnabled(); + } + + /** + * Returns HTML for the create account link inside the anon user links + * @param string[] $returnto array of query strings used to build the login link + * @param bool $isDropdownItem Set true for create account link inside the user menu dropdown + * which includes icon classes and is not styled like a button + * @return string + */ + private function getCreateAccountHTML( $returnto, $isDropdownItem ) { + $createAccountData = $this->buildCreateAccountData( $returnto ); + $createAccountData = array_merge( $createAccountData, [ + 'class' => $isDropdownItem ? [ + 'vector-menu-content-item', + ] : '', + 'collapsible' => true, + 'icon' => $isDropdownItem ? $createAccountData['icon'] : null, + 'button' => !$isDropdownItem, + ] ); + $createAccountData = Hooks::updateLinkData( $createAccountData ); + return $this->makeLink( 'create-account', $createAccountData ); + } + + /** + * Returns HTML for the create account button, login button and learn more link inside the anon user menu + * @param string[] $returnto array of query strings used to build the login link + * @param bool $useCombinedLoginLink if a combined login/signup link will be used + * @param bool $isTempUser + * @param bool $includeLearnMoreLink Pass `true` to include the learn more + * link in the menu for anon users. This param will be inert for temp users. + * @return string + */ + private function getAnonMenuBeforePortletHTML( + $returnto, + $useCombinedLoginLink, + $isTempUser, + $includeLearnMoreLink + ) { + $templateParser = $this->getTemplateParser(); + $loginLinkData = array_merge( $this->buildLoginData( $returnto, $useCombinedLoginLink ), [ + 'class' => [ 'vector-menu-content-item', 'vector-menu-content-item-login' ], + ] ); + $loginLinkData = Hooks::updateLinkData( $loginLinkData ); + $templateData = [ + 'htmlCreateAccount' => $this->getCreateAccountHTML( $returnto, true ), + 'htmlLogin' => $this->makeLink( 'login', $loginLinkData ), + 'data-anon-editor' => [] + ]; + + $templateName = $isTempUser ? 'UserLinks__templogin' : 'UserLinks__login'; + + if ( !$isTempUser && $includeLearnMoreLink ) { + try { + $learnMoreLinkData = [ + 'text' => $this->msg( 'vector-anon-user-menu-pages-learn' )->text(), + 'href' => Title::newFromText( $this->msg( 'vector-intro-page' )->text() )->getLocalURL(), + 'aria-label' => $this->msg( 'vector-anon-user-menu-pages-label' )->text(), + ]; + + $templateData['data-anon-editor'] = [ + 'htmlLearnMoreLink' => $this->makeLink( '', $learnMoreLinkData ), + 'msgLearnMore' => $this->msg( 'vector-anon-user-menu-pages' ) + ]; + } catch ( MalformedTitleException $e ) { + // ignore (T340220) + } + } + + return $templateParser->processTemplate( $templateName, $templateData ); + } + + /** + * Returns HTML for the logout button that should be placed in the user (personal) menu + * after the menu itself. + * @return string + */ + private function getLogoutHTML() { + $logoutLinkData = array_merge( $this->buildLogoutLinkData(), [ + 'class' => [ 'vector-menu-content-item', 'vector-menu-content-item-logout' ], + ] ); + $logoutLinkData = Hooks::updateLinkData( $logoutLinkData ); + + $templateParser = $this->getTemplateParser(); + return $templateParser->processTemplate( 'UserLinks__logout', [ + 'htmlLogout' => $this->makeLink( 'logout', $logoutLinkData ) + ] ); + } + + /** + * Returns template data for UserLinks.mustache + * @param array $menuData existing menu template data to be transformed and copied for UserLinks + * @param User $user the context user + * @return array + */ + private function getUserLinksTemplateData( $menuData, $user ): array { + $isAnon = !$user->isRegistered(); + $isTempUser = $user->isTemp(); + $returnto = $this->getReturnToParam(); + $useCombinedLoginLink = $this->useCombinedLoginLink(); + $userMenuOverflowData = $menuData[ 'data-vector-user-menu-overflow' ]; + $userMenuData = $menuData[ 'data-user-menu' ]; + if ( $isAnon || $isTempUser ) { + $userMenuData[ 'html-before-portal' ] .= $this->getAnonMenuBeforePortletHTML( + $returnto, + $useCombinedLoginLink, + $isTempUser, + // 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. + !$userMenuData['is-empty'] + ); + # TODO: remove customization not after 1.40+ + # ubc custom: remove user links menu for unsigned in users the + # login button is in overflow and not allowed to collapse, so the + # login button insider user links is not needed + $userMenuData = []; + } else { + // Appending as to not override data potentially set by the onSkinAfterPortlet hook. + $userMenuData[ 'html-after-portal' ] .= $this->getLogoutHTML(); + } + + $moreItems = substr_count( $userMenuOverflowData['html-items'], ' $moreItems > 3, + 'data-user-menu-overflow' => $menuData[ 'data-vector-user-menu-overflow' ], + 'data-user-menu' => $userMenuData + ]; + } + + /** + * @inheritDoc + */ + protected function runOnSkinTemplateNavigationHooks( SkinTemplate $skin, &$content_navigation ) { + parent::runOnSkinTemplateNavigationHooks( $skin, $content_navigation ); + Hooks::onSkinTemplateNavigation( $skin, $content_navigation ); + } + + /** + * Check whether ULS is enabled + * + * @return bool + */ + private function isULSExtensionEnabled(): bool { + return ExtensionRegistry::getInstance()->isLoaded( 'UniversalLanguageSelector' ); + } + + /** + * Generate data needed to generate the sticky header. + * @param array $searchBoxData + * @param bool $includeEditIcons + * @return array + */ + final protected function getStickyHeaderData( $searchBoxData, $includeEditIcons ): array { + $btns = [ + self::TALK_ICON, + self::SUBJECT_ICON, + self::HISTORY_ICON, + self::WATCHSTAR_ICON, + ]; + if ( $includeEditIcons ) { + $btns[] = self::EDIT_WIKITEXT_ICON; + $btns[] = self::EDIT_PROTECTED_ICON; + $btns[] = self::EDIT_VE_ICON; + } + $btns[] = $this->getAddSectionButtonData(); + + $tocPortletData = $this->decoratePortletData( 'data-sticky-header-toc', [ + 'id' => 'p-sticky-header-toc', + 'class' => 'mw-portlet mw-portlet-sticky-header-toc vector-sticky-header-toc', + 'html-items' => '', + 'html-vector-menu-checkbox-attributes' => 'tabindex="-1"', + 'html-vector-menu-heading-attributes' => 'tabindex="-1"', + 'button' => true, + 'text-hidden' => true, + 'icon' => 'listBullet' + ] ); + + // Show sticky ULS if the ULS extension is enabled and the ULS in header is not hidden + $showStickyULS = $this->isULSExtensionEnabled() && !$this->shouldHideLanguages(); + return [ + 'data-sticky-header-toc' => $tocPortletData, + 'data-primary-action' => $showStickyULS ? + $this->getULSButtonData() : null, + 'data-button-start' => [ + 'label' => $this->msg( 'search' ), + 'icon' => 'wikimedia-search', + 'is-quiet' => true, + 'tabindex' => '-1', + 'class' => 'vector-sticky-header-search-toggle', + 'event' => 'ui.' . $searchBoxData['form-id'] . '.icon' + ], + 'data-search' => $searchBoxData, + 'data-buttons' => $btns, + ]; + } + + /** + * Generate data needed to create SidebarAction item. + * @param array $htmlData data to make a link or raw html + * @param array $headingOptions optional heading for the html + * @return array keyed data for the SidebarAction template + */ + private function makeSidebarActionData( array $htmlData = [], array $headingOptions = [] ): array { + $htmlContent = ''; + // Populates the sidebar as a standalone link or custom html. + if ( array_key_exists( 'link', $htmlData ) ) { + $htmlContent = $this->makeLink( 'link', $htmlData['link'] ); + } elseif ( array_key_exists( 'html-content', $htmlData ) ) { + $htmlContent = $htmlData['html-content']; + } + + return $headingOptions + [ + 'html-content' => $htmlContent, + ]; + } + + /** + * Determines if the language switching alert box should be in the sidebar. + * + * @return bool + */ + private function shouldLanguageAlertBeInSidebar(): bool { + $featureManager = VectorServices::getFeatureManager(); + $isMainPage = $this->getTitle() ? $this->getTitle()->isMainPage() : false; + $shouldShowOnMainPage = $isMainPage && !empty( $this->getLanguagesCached() ) && + $featureManager->isFeatureEnabled( Constants::FEATURE_LANGUAGE_IN_MAIN_PAGE_HEADER ); + return ( $this->isLanguagesInContentAt( 'top' ) && !$isMainPage && !$this->shouldHideLanguages() && + $featureManager->isFeatureEnabled( Constants::FEATURE_LANGUAGE_ALERT_IN_SIDEBAR ) ) || + $shouldShowOnMainPage; + } + + /** + * @inheritDoc + */ + public function getTemplateData(): array { + $skin = $this; + + $parentData = $this->decoratePortletsData( parent::getTemplateData() ); + + // SkinVector sometimes serves new Vector as part of removing the + // skin version user preference. TCho avoid T302461 we need to unset it here. + // This shouldn't be run on SkinVector22. + if ( $this->getSkinName() === 'vector' ) { + unset( $parentData['data-toc'] ); + } + + // + // Naming conventions for Mustache parameters. + // + // Value type (first segment): + // - Prefix "is" or "has" for boolean values. + // - Prefix "msg-" for interface message text. + // - Prefix "html-" for raw HTML. + // - Prefix "data-" for an array of template parameters that should be passed directly + // to a template partial. + // - Prefix "array-" for lists of any values. + // + // Source of value (first or second segment) + // - Segment "page-" for data relating to the current page (e.g. Title, WikiPage, or OutputPage). + // - Segment "hook-" for any thing generated from a hook. + // It should be followed by the name of the hook in hyphenated lowercase. + // + // Conditionally used values must use null to indicate absence (not false or ''). + $commonSkinData = array_merge( $parentData, [ + 'is-legacy' => $this->isLegacy(), + 'input-location' => $this->getSearchBoxInputLocation(), + 'sidebar-visible' => $this->isSidebarVisible(), + 'is-language-in-content' => $this->isLanguagesInContent(), + 'is-language-in-content-top' => $this->isLanguagesInContentAt( 'top' ), + 'is-language-in-content-bottom' => $this->isLanguagesInContentAt( 'bottom' ), + 'data-search-box' => $this->getSearchData( + $parentData['data-search-box'], + !$this->isLegacy(), + // is primary mode of search + true, + 'searchform', + true + ) + ] ); + + $user = $skin->getUser(); + if ( $user->isRegistered() ) { + // Note: This data is also passed to legacy template where it is unused. + $optOutUrl = [ + 'text' => $this->msg( 'vector-opt-out' )->text(), + 'href' => SpecialPage::getTitleFor( + 'Preferences', + false, + 'mw-prefsection-rendering-skin' + )->getLinkURL( 'useskin=vector&wprov=' . self::OPT_OUT_LINK_TRACKING_CODE ), + 'title' => $this->msg( 'vector-opt-out-tooltip' )->text(), + 'active' => false, + ]; + $htmlData = [ + 'link' => $optOutUrl, + ]; + $commonSkinData['data-emphasized-sidebar-action'][] = $this->makeSidebarActionData( + $htmlData, + [] + ); + } + + if ( !$this->isLegacy() ) { + $commonSkinData['data-vector-user-links'] = $this->getUserLinksTemplateData( + $commonSkinData['data-portlets'], + $user + ); + + // T295555 Add language switch alert message temporarily (to be removed). + if ( $this->shouldLanguageAlertBeInSidebar() ) { + $languageSwitchAlert = [ + 'html-content' => Html::noticeBox( + $this->msg( 'vector-language-redirect-to-top' )->parse(), + 'vector-language-sidebar-alert' + ), + ]; + $headingOptions = [ + 'heading' => $this->msg( 'vector-languages' )->plain(), + ]; + $commonSkinData['data-vector-language-switch-alert'][] = $this->makeSidebarActionData( + $languageSwitchAlert, + $headingOptions + ); + } + } + + return $commonSkinData; + } + + /** + * Annotates search box with Vector-specific information + * + * @param array $searchBoxData + * @param bool $isCollapsible + * @param bool $isPrimary + * @param string $formId + * @param bool $autoExpandWidth + * @return array modified version of $searchBoxData + */ + final protected function getSearchData( + array $searchBoxData, + bool $isCollapsible, + bool $isPrimary, + string $formId, + bool $autoExpandWidth + ) { + $searchClass = 'vector-search-box-vue '; + + if ( $isCollapsible ) { + $searchClass .= ' vector-search-box-collapses '; + } + + if ( $this->doesSearchHaveThumbnails() ) { + $searchClass .= ' ' . self::SEARCH_SHOW_THUMBNAIL_CLASS . + ( $autoExpandWidth ? ' ' . self::SEARCH_AUTO_EXPAND_WIDTH_CLASS : '' ); + } + + // Annotate search box with a component class. + $searchBoxData['class'] = trim( $searchClass ); + $searchBoxData['is-collapsible'] = $isCollapsible; + $searchBoxData['is-primary'] = $isPrimary; + $searchBoxData['form-id'] = $formId; + + // At lower resolutions the search input is hidden search and only the submit button is shown. + // It should behave like a form submit link (e.g. submit the form with no input value). + // We'll wire this up in a later task T284242. + $collapseIconAttrs = Linker::tooltipAndAccesskeyAttribs( 'search' ); + $searchBoxData['data-collapse-icon'] = array_merge( [ + 'href' => Title::newFromText( $searchBoxData['page-title'] )->getLocalUrl(), + 'label' => $this->msg( 'search' ), + 'icon' => 'wikimedia-search', + 'is-quiet' => true, + 'class' => 'search-toggle', + ], $collapseIconAttrs ); + + return $searchBoxData; + } + + /** + * Gets the value of the "input-location" parameter for the SearchBox Mustache template. + * + * @return string Either `Constants::SEARCH_BOX_INPUT_LOCATION_DEFAULT` or + * `Constants::SEARCH_BOX_INPUT_LOCATION_MOVED` + */ + private function getSearchBoxInputLocation(): string { + if ( $this->isLegacy() ) { + return Constants::SEARCH_BOX_INPUT_LOCATION_DEFAULT; + } + + return Constants::SEARCH_BOX_INPUT_LOCATION_MOVED; + } + + /** + * @inheritDoc + */ + public function isResponsive() { + // Check it's enabled by user preference and configuration + $responsive = parent::isResponsive() && $this->getConfig()->get( 'VectorResponsive' ); + // For historic reasons, the viewport is added when Vector is loaded on the mobile + // domain. This is only possible for 3rd parties or by useskin parameter as there is + // no preference for changing mobile skin. Only need to check if $responsive is falsey. + if ( !$responsive && ExtensionRegistry::getInstance()->isLoaded( 'MobileFrontend' ) ) { + $mobFrontContext = MediaWikiServices::getInstance()->getService( 'MobileFrontend.Context' ); + if ( $mobFrontContext->shouldDisplayMobileView() ) { + return true; + } + } + return $responsive; + } + + /** + * Returns `true` if Vue search is enabled to show thumbnails and `false` otherwise. + * Note this is only relevant for Vue search experience (not legacy search). + * + * @return bool + */ + private function doesSearchHaveThumbnails(): bool { + return $this->getConfig()->get( 'VectorWvuiSearchOptions' )['showThumbnail']; + } + + /** + * Determines wheather the initial state of sidebar is visible on not + * + * @return bool + */ + private function isSidebarVisible() { + $skin = $this->getSkin(); + if ( $skin->getUser()->isRegistered() ) { + $userOptionsLookup = MediaWikiServices::getInstance()->getUserOptionsLookup(); + $userPrefSidebarState = $userOptionsLookup->getOption( + $skin->getUser(), + Constants::PREF_KEY_SIDEBAR_VISIBLE + ); + + $defaultLoggedinSidebarState = $this->getConfig()->get( + Constants::CONFIG_KEY_DEFAULT_SIDEBAR_VISIBLE_FOR_AUTHORISED_USER + ); + + // If the sidebar user preference has been set, return that value, + // if not, then the default sidebar state for logged-in users. + return ( $userPrefSidebarState !== null ) + ? (bool)$userPrefSidebarState + : $defaultLoggedinSidebarState; + } + return $this->getConfig()->get( + Constants::CONFIG_KEY_DEFAULT_SIDEBAR_VISIBLE_FOR_ANONYMOUS_USER + ); + } + + /** + * Get the ULS button label, accounting for the number of available + * languages. + * + * @return array + */ + private function getULSLabels(): array { + $numLanguages = count( $this->getLanguagesCached() ); + + if ( $numLanguages === 0 ) { + return [ + 'label' => $this->msg( 'vector-no-language-button-label' )->text(), + 'aria-label' => $this->msg( 'vector-no-language-button-aria-label' )->text() + ]; + } else { + return [ + 'label' => $this->msg( 'vector-language-button-label' )->numParams( $numLanguages )->escaped(), + 'aria-label' => $this->msg( 'vector-language-button-aria-label' )->numParams( $numLanguages )->escaped() + ]; + } + } + + /** + * Creates button data for the "Add section" button in the sticky header + * + * @return array + */ + private function getAddSectionButtonData() { + return [ + 'href' => '#', + 'id' => 'ca-addsection-sticky-header', + 'event' => 'addsection-sticky-header', + 'html-vector-button-icon' => Hooks::makeIcon( 'wikimedia-speechBubbleAdd-progressive' ), + 'label' => $this->msg( [ 'vector-2022-action-addsection', 'skin-action-addsection' ] ), + 'is-quiet' => true, + 'tabindex' => '-1', + 'class' => 'sticky-header-icon mw-ui-primary mw-ui-progressive' + ]; + } + + /** + * Creates button data for the ULS button in the sticky header + * + * @return array + */ + private function getULSButtonData() { + $numLanguages = count( $this->getLanguagesCached() ); + + return [ + 'id' => 'p-lang-btn-sticky-header', + 'class' => 'mw-interlanguage-selector', + 'is-quiet' => true, + 'tabindex' => '-1', + 'label' => $this->getULSLabels()[ 'label' ], + 'html-vector-button-icon' => Hooks::makeIcon( 'wikimedia-language' ), + 'event' => 'ui.dropdown-p-lang-btn-sticky-header' + ]; + } + + /** + * Creates portlet data for the ULS button in the header + * + * @return array + */ + private function getULSPortletData() { + $numLanguages = count( $this->getLanguagesCached() ); + + $languageButtonData = [ + 'id' => 'p-lang-btn', + 'label' => $this->getULSLabels()['label'], + 'aria-label' => $this->getULSLabels()['aria-label'], + // ext.uls.interface attaches click handler to this selector. + 'checkbox-class' => ' mw-interlanguage-selector ', + 'icon' => 'language-progressive', + 'button' => true, + 'heading-class' => self::CLASS_PROGRESSIVE . ' mw-portlet-lang-heading-' . strval( $numLanguages ), + ]; + + // Adds class to hide language button + // Temporary solution to T287206, can be removed when ULS dialog includes interwiki links + if ( $this->shouldHideLanguages() ) { + $languageButtonData['class'] = ' mw-portlet-empty'; + } + + return $languageButtonData; + } + + /** + * Creates portlet data for the user menu dropdown + * + * @param array $portletData + * @return array + */ + private function getUserMenuPortletData( $portletData ) { + // T317789: Core can undesirably add an 'emptyPortlet' class that hides the + // user menu. This is a result of us manually removing items from the menu + // in Hooks::updateUserLinksDropdownItems which can make + // SkinTemplate::getPortletData apply the `emptyPortlet` class if there are + // no menu items. Since we subsequently add menu items in + // SkinVector::getUserLinksTemplateData, the `emptyPortlet` class is + // innaccurate. This is why we add the desired classes, `mw-portlet` and + // `mw-portlet-personal` here instead. This can potentially be removed upon + // completion of T319356. + // + // Also, add target class to apply different icon to personal menu dropdown for logged in users. + $portletData['class'] = 'mw-portlet mw-portlet-personal vector-user-menu'; + $portletData['class'] .= $this->loggedin ? + ' vector-user-menu-logged-in' : + ' vector-user-menu-logged-out'; + if ( $this->getUser()->isTemp() ) { + $icon = 'userAnonymous'; + } elseif ( $this->loggedin ) { + $icon = 'userAvatar'; + } else { + $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. + $portletData['html-tooltip'] = Linker::tooltip( 'vector-anon-user-menu-title' ); + } + $portletData['icon'] = $icon; + $portletData['button'] = true; + $portletData['text-hidden'] = true; + return $portletData; + } + + /** + * Helper for applying Vector menu classes to portlets + * + * @param array $portletData returned by SkinMustache to decorate + * @param int $type representing one of the menu types (see MENU_TYPE_* constants) + * @return array modified version of portletData input + */ + private function updatePortletClasses( + array $portletData, + int $type = self::MENU_TYPE_DEFAULT + ) { + $extraClasses = [ + self::MENU_TYPE_DROPDOWN => 'vector-menu-dropdown', + self::MENU_TYPE_TABS => 'vector-menu-tabs', + self::MENU_TYPE_PORTAL => 'vector-menu-portal portal', + self::MENU_TYPE_DEFAULT => '', + ]; + if ( $this->isLegacy() ) { + $extraClasses[self::MENU_TYPE_TABS] .= ' vector-menu-tabs-legacy'; + } + $portletData['class'] .= ' ' . $extraClasses[$type]; + + if ( !isset( $portletData['heading-class'] ) ) { + $portletData['heading-class'] = ''; + } + if ( $type === self::MENU_TYPE_DROPDOWN ) { + $portletData = Hooks::updateDropdownMenuData( $portletData ); + } + + $portletData['class'] = trim( $portletData['class'] ); + $portletData['heading-class'] = trim( $portletData['heading-class'] ); + return $portletData; + } + + /** + * Performs updates to all portlets. + * + * @param array $data + * @return array + */ + private function decoratePortletsData( array $data ) { + foreach ( $data['data-portlets'] as $key => $pData ) { + $data['data-portlets'][$key] = $this->decoratePortletData( + $key, + $pData + ); + } + $sidebar = $data['data-portlets-sidebar']; + $sidebar['data-portlets-first'] = $this->decoratePortletData( + 'navigation', $sidebar['data-portlets-first'] + ); + $rest = $sidebar['array-portlets-rest']; + foreach ( $rest as $key => $pData ) { + $rest[$key] = $this->decoratePortletData( + $pData['id'], $pData + ); + } + $sidebar['array-portlets-rest'] = $rest; + $data['data-portlets-sidebar'] = $sidebar; + return $data; + } + + /** + * Performs the following updates to portlet data: + * - Adds concept of menu types + * - Marks the selected variant in the variant portlet + * - modifies tooltips of personal and user-menu portlets + * @param string $key + * @param array $portletData + * @return array + */ + private function decoratePortletData( + string $key, + array $portletData + ): array { + switch ( $key ) { + case 'data-user-menu': + case 'data-actions': + case 'data-variants': + case 'data-sticky-header-toc': + $type = self::MENU_TYPE_DROPDOWN; + break; + case 'data-views': + case 'data-associated-pages': + case 'data-namespaces': + $type = self::MENU_TYPE_TABS; + break; + case 'data-notifications': + case 'data-personal': + case 'data-user-page': + case 'data-vector-user-menu-overflow': + $type = self::MENU_TYPE_DEFAULT; + break; + case 'data-languages': + $type = $this->isLanguagesInContent() ? + self::MENU_TYPE_DROPDOWN : self::MENU_TYPE_PORTAL; + break; + default: + $type = self::MENU_TYPE_PORTAL; + break; + } + + if ( $key === 'data-languages' && $this->isLanguagesInContent() ) { + $portletData = array_merge( $portletData, $this->getULSPortletData() ); + } + + if ( $key === 'data-user-menu' && !$this->isLegacy() ) { + $portletData = $this->getUserMenuPortletData( $portletData ); + } + + if ( $key === 'data-vector-user-menu-overflow' ) { + $portletData['class'] .= ' vector-user-menu-overflow'; + } + + if ( $key === 'data-personal' && $this->isLegacy() ) { + // Set tooltip to empty string for the personal menu for both logged-in and logged-out users + // to avoid showing the tooltip for legacy version. + $portletData['html-tooltip'] = ''; + $portletData['class'] .= ' vector-user-menu-legacy'; + } + + // Special casing for Variant to change label to selected. + // Hopefully we can revisit and possibly remove this code when the language switcher is moved. + if ( $key === 'data-variants' ) { + $languageConverterFactory = MediaWikiServices::getInstance()->getLanguageConverterFactory(); + $pageLang = $this->getTitle()->getPageLanguage(); + $converter = $languageConverterFactory->getLanguageConverter( $pageLang ); + $portletData['label'] = $pageLang->getVariantname( + $converter->getPreferredVariant() + ); + // T289523 Add aria-label data to the language variant switcher. + $portletData['aria-label'] = $this->msg( 'vector-language-variant-switcher-label' ); + } + + $portletData = $this->updatePortletClasses( + $portletData, + $type + ); + + return $portletData + [ + 'is-dropdown' => $type === self::MENU_TYPE_DROPDOWN, + 'is-portal' => $type === self::MENU_TYPE_PORTAL, + ]; + } +}