wikimedia/mediawiki-extensions-Translate

View on GitHub
messagegroups/WikiPageMessageGroup.php

Summary

Maintainability
A
25 mins
Test Coverage
<?php
/**
 * This file contains an unmanaged message group implementation.
 *
 * @file
 * @author Niklas Laxström
 * @author Siebrand Mazeland
 * @license GPL-2.0-or-later
 */

use MediaWiki\Context\IContextSource;
use MediaWiki\Extension\Translate\PageTranslation\Hooks;
use MediaWiki\Extension\Translate\PageTranslation\TranslatablePageInsertablesSuggester;
use MediaWiki\Extension\Translate\PageTranslation\TranslationUnit;
use MediaWiki\Extension\Translate\Utilities\Utilities;
use MediaWiki\Extension\Translate\Validation\ValidationRunner;
use MediaWiki\Linker\LinkTarget;
use MediaWiki\MediaWikiServices;
use MediaWiki\Revision\SlotRecord;
use MediaWiki\Title\Title;

/**
 * Wraps the translatable page sections into a message group.
 * @ingroup PageTranslation MessageGroup
 */
class WikiPageMessageGroup extends MessageGroupOld {
    /** @var Title|string */
    protected $title;
    /** @var int */
    protected $namespace = NS_TRANSLATIONS;

    /**
     * @param string $id
     * @param Title|string $title
     */
    public function __construct( $id, $title ) {
        $this->id = $id;
        $this->title = $title;
    }

    public function getSourceLanguage() {
        return $this->getTitle()->getPageLanguage()->getCode();
    }

    public function getTitle(): Title {
        if ( is_string( $this->title ) ) {
            $this->title = Title::newFromText( $this->title );
        }

        return $this->title;
    }

    /**
     * Only used for caching to avoid repeating database queries
     * for example during message index rebuild.
     * @var array
     */
    protected $definitions;

    /** @return string[] */
    public function getDefinitions() {
        if ( is_array( $this->definitions ) ) {
            return $this->definitions;
        }

        $title = $this->getTitle();

        $dbr = Utilities::getSafeReadDB();
        $res = $dbr->newSelectQueryBuilder()
            ->select( [ 'trs_key', 'trs_text' ] )
            ->from( 'page' )
            ->join( 'translate_sections', null, 'page_id = trs_page' )
            ->where( [
                'page_namespace' => $title->getNamespace(),
                'page_title' => $title->getDBkey(),
            ] )
            ->caller( __METHOD__ )
            ->orderBy( 'trs_order' )
            ->fetchResultSet();

        $defs = [];

        foreach ( $res as $r ) {
            $section = new TranslationUnit( $r->trs_text );
            $defs[$r->trs_key] = $section->getTextWithVariables();
        }

        $groupKeys = $this->makeGroupKeys( array_keys( $defs ) );
        $this->definitions = array_combine( $groupKeys, array_values( $defs ) );

        return $this->definitions;
    }

    /**
     * @param string[] $keys
     * @return string[]
     */
    public function makeGroupKeys( array $keys ): array {
        $prefix = $this->getTitle()->getPrefixedDBkey() . '/';
        return array_map( static function ( string $key ) use ( $prefix ): string {
            return $prefix . str_replace( ' ', '_', $key );
        }, $keys );
    }

    /**
     * Overriding the getLabel method and deriving the label from the title.
     * Mainly to reduce the amount of data stored in the cache.
     *
     * @param IContextSource|null $context
     * @return string
     */
    public function getLabel( IContextSource $context = null ) {
        return $this->getTitle()->getPrefixedText();
    }

    /**
     * Clear caches to avoid stale data.
     *
     * For example JobQueue can run for a longer time, and stale definitions would
     * cause the total number of messages to be incorrect.
     *
     * @since 2016.04
     */
    public function clearCaches() {
        $this->definitions = null;
    }

    public function load( $code ) {
        if ( $this->isSourceLanguage( $code ) ) {
            return $this->getDefinitions();
        }

        return [];
    }

    /**
     * Returns of stored translation of message specified by the $key in language
     * code $code.
     *
     * @param string $key Message key
     * @param string $code Language code
     * @param int $flags READ_* class constant bitfield
     * @return string|null Stored translation or null.
     */
    public function getMessage( $key, $code, $flags = IDBAccessObject::READ_LATEST ) {
        if ( $this->isSourceLanguage( $code ) ) {
            $stuff = $this->load( $code );

            $title = Title::newFromText( $key );
            if ( $title ) {
                $key = $title->getPrefixedDBkey();
            }

            return $stuff[$key] ?? null;
        }

        $title = Title::makeTitleSafe( $this->getNamespace(), "$key/$code" );
        if ( Hooks::$renderingContext ) {
            $revFlags = IDBAccessObject::READ_NORMAL; // bug T95753
        } else {
            $revFlags = ( $flags & IDBAccessObject::READ_LATEST ) == IDBAccessObject::READ_LATEST
                ? IDBAccessObject::READ_LATEST
                : IDBAccessObject::READ_NORMAL;
        }
        $rev = MediaWikiServices::getInstance()
            ->getRevisionLookup()
            ->getRevisionByTitle( $title, 0, $revFlags );

        if ( !$rev ) {
            return null;
        }

        $content = $rev->getContent( SlotRecord::MAIN );
        return ( $content instanceof TextContent ) ? $content->getText() : null;
    }

    /** @return ValidationRunner */
    public function getValidator() {
        $validator = new ValidationRunner( $this->getId() );
        $validator->setValidators( [
            [ 'id' => 'MediaWikiPlural' ],
            [ 'id' => 'BraceBalance' ]
        ] );

        return $validator;
    }

    public function getInsertablesSuggester() {
        return new TranslatablePageInsertablesSuggester();
    }

    public function getDescription( IContextSource $context = null ) {
        $title = $this->getTitle()->getPrefixedText();
        $target = ":$title";
        $pageLanguageCode = $this->getSourceLanguage();
        $inLanguageCode = $context ? $context->getLanguage()->getCode() : null;
        $languageName = MediaWikiServices::getInstance()->getLanguageNameUtils()
            ->getLanguageName( $pageLanguageCode, $inLanguageCode );

        // Allow for adding a custom group description by using
        // "MediaWiki:Tp-custom-<group ID>".
        $customText = '';
        $msg = wfMessage( 'tp-custom-' . $this->id );
        self::addContext( $msg, $context );
        if ( $msg->exists() ) {
            $customText = $msg->plain();
        }

        $msg = wfMessage( 'translate-tag-page-desc', $title, $target, $languageName, $pageLanguageCode );
        self::addContext( $msg, $context );

        return $msg->plain() . $customText;
    }

    /** @inheritDoc */
    public function getRelatedPage(): ?LinkTarget {
        return $this->getTitle();
    }
}