3 * This program is free software; you can redistribute it and/or modify
4 * it under the terms of the GNU General Public License as published by
5 * the Free Software Foundation; either version 2 of the License, or
6 * (at your option) any later version.
8 * This program is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 * GNU General Public License for more details.
13 * You should have received a copy of the GNU General Public License along
14 * with this program; if not, write to the Free Software Foundation, Inc.,
15 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
16 * http://www.gnu.org/copyleft/gpl.html
21 use MediaWiki\Context\ContextSource
;
22 use MediaWiki\HookContainer\ProtectedHookAccessorTrait
;
23 use MediaWiki\Html\Html
;
24 use MediaWiki\Linker\Linker
;
25 use MediaWiki\MainConfigNames
;
26 use MediaWiki\MediaWikiServices
;
27 use MediaWiki\SpecialPage\SpecialPage
;
28 use MediaWiki\Title\Title
;
31 * Builds the image revision log shown on image pages
35 class ImageHistoryList
extends ContextSource
{
36 use ProtectedHookAccessorTrait
;
38 protected Title
$title;
40 protected ImagePage
$imagePage;
41 protected File
$current;
43 protected bool $showThumb;
45 protected $preventClickjacking = false;
48 * @param ImagePage $imagePage
50 public function __construct( $imagePage ) {
51 $context = $imagePage->getContext();
52 $this->current
= $imagePage->getPage()->getFile();
53 $this->img
= $imagePage->getDisplayedFile();
54 $this->title
= $imagePage->getTitle();
55 $this->imagePage
= $imagePage;
56 $this->showThumb
= $context->getConfig()->get( MainConfigNames
::ShowArchiveThumbnails
) &&
57 $this->img
->canRender();
58 $this->setContext( $context );
64 public function getImagePage() {
65 return $this->imagePage
;
71 public function getFile() {
78 public function beginImageHistoryList() {
79 // Styles for class=history-deleted
80 $this->getOutput()->addModuleStyles( 'mediawiki.interface.helpers.styles' );
83 $canDelete = $this->current
->isLocal() &&
84 $this->getAuthority()->isAllowedAny( 'delete', 'deletedhistory' );
88 $canDelete ?
'' : null,
90 $this->showThumb ?
'filehist-thumb' : null,
91 'filehist-dimensions',
95 if ( $key !== null ) {
96 $html .= Html
::element( 'th', [], $key ?
$this->msg( $key )->text() : '' );
100 return Html
::openElement( 'table', [ 'class' => 'wikitable filehistory' ] ) . "\n"
101 . Html
::rawElement( 'tr', [], $html ) . "\n";
107 public function endImageHistoryList() {
108 return Html
::closeElement( 'table' ) . "\n";
115 * @param string $formattedComment
118 public function imageHistoryLine( $iscur, $file, $formattedComment ) {
119 $user = $this->getUser();
120 $lang = $this->getLanguage();
121 $linkRenderer = MediaWikiServices
::getInstance()->getLinkRenderer();
122 $timestamp = wfTimestamp( TS_MW
, $file->getTimestamp() );
123 // @phan-suppress-next-line PhanUndeclaredMethod
124 $img = $iscur ?
$file->getName() : $file->getArchiveName();
125 $uploader = $file->getUploader( File
::FOR_THIS_USER
, $user );
127 $local = $this->current
->isLocal();
131 if ( $local && ( $this->getAuthority()->isAllowedAny( 'delete', 'deletedhistory' ) ) ) {
132 $row .= Html
::openElement( 'td' );
133 # Link to hide content. Don't show useless link to people who cannot hide revisions.
134 if ( !$iscur && $this->getAuthority()->isAllowed( 'deleterevision' ) ) {
135 // If file is top revision, is missing or locked from this user, don't link
136 if ( !$file->userCan( File
::DELETED_RESTRICTED
, $user ) ||
!$file->exists() ) {
137 $row .= Html
::check( 'deleterevisions', false, [ 'disabled' => 'disabled' ] );
139 $row .= Html
::check( 'ids[' . explode( '!', $img, 2 )[0] . ']', false );
141 if ( $this->getAuthority()->isAllowed( 'delete' ) ) {
145 # Link to remove from history
146 if ( $this->getAuthority()->isAllowed( 'delete' ) ) {
147 if ( $file->exists() ) {
148 $row .= $linkRenderer->makeKnownLink(
150 $this->msg( $iscur ?
'filehist-deleteall' : 'filehist-deleteone' )->text(),
152 [ 'action' => 'delete', 'oldimage' => $iscur ?
null : $img ]
155 // T244567: Non-existing file can not be deleted.
156 $row .= $this->msg( 'filehist-missing' )->escaped();
160 $row .= Html
::closeElement( 'td' );
163 // Reversion link/current indicator
164 $row .= Html
::openElement( 'td' );
166 $row .= $this->msg( 'filehist-current' )->escaped();
167 } elseif ( $local && $this->getAuthority()->probablyCan( 'edit', $this->title
)
168 && $this->getAuthority()->probablyCan( 'upload', $this->title
)
170 if ( $file->isDeleted( File
::DELETED_FILE
) ) {
171 $row .= $this->msg( 'filehist-revert' )->escaped();
172 } elseif ( !$file->exists() ) {
173 // T328112: Lost file, in this case there's no version to revert back to.
174 $row .= $this->msg( 'filehist-missing' )->escaped();
176 $row .= $linkRenderer->makeKnownLink(
178 $this->msg( 'filehist-revert' )->text(),
181 'action' => 'revert',
187 $row .= Html
::closeElement( 'td' );
189 // Date/time and image link
190 $selected = $file->getTimestamp() === $this->img
->getTimestamp();
191 $row .= Html
::openElement( 'td', [
192 'class' => $selected ?
'filehistory-selected' : null,
193 'style' => 'white-space: nowrap;'
195 if ( !$file->userCan( File
::DELETED_FILE
, $user ) ) {
196 # Don't link to unviewable files
197 $row .= Html
::element( 'span', [ 'class' => 'history-deleted' ],
198 $lang->userTimeAndDate( $timestamp, $user )
200 } elseif ( $file->isDeleted( File
::DELETED_FILE
) ) {
201 $timeAndDate = $lang->userTimeAndDate( $timestamp, $user );
203 $this->setPreventClickjacking( true );
204 # Make a link to review the image
205 $url = $linkRenderer->makeKnownLink(
206 SpecialPage
::getTitleFor( 'Revisiondelete' ),
210 'target' => $this->title
->getPrefixedText(),
212 'token' => $user->getEditToken( $img )
216 $url = htmlspecialchars( $timeAndDate );
218 $row .= Html
::rawElement( 'span', [ 'class' => 'history-deleted' ], $url );
219 } elseif ( !$file->exists() ) {
220 $row .= Html
::element( 'span', [ 'class' => 'mw-file-missing' ],
221 $lang->userTimeAndDate( $timestamp, $user )
224 $url = $iscur ?
$this->current
->getUrl() : $this->current
->getArchiveUrl( $img );
225 $row .= Html
::element( 'a', [ 'href' => $url ],
226 $lang->userTimeAndDate( $timestamp, $user )
229 $row .= Html
::closeElement( 'td' );
232 if ( $this->showThumb
) {
233 $row .= Html
::rawElement( 'td', [],
234 $this->getThumbForLine( $file, $iscur ) ??
$this->msg( 'filehist-nothumb' )->escaped()
238 // Image dimensions + size
239 $row .= Html
::openElement( 'td' );
240 $row .= htmlspecialchars( $file->getDimensionsString() );
241 $row .= $this->msg( 'word-separator' )->escaped();
242 $row .= Html
::element( 'span', [ 'style' => 'white-space: nowrap;' ],
243 $this->msg( 'parentheses' )->sizeParams( $file->getSize() )->text()
245 $row .= Html
::closeElement( 'td' );
248 $row .= Html
::openElement( 'td' );
249 // Hide deleted usernames
251 $row .= Linker
::userLink( $uploader->getId(), $uploader->getName() );
253 $row .= Html
::rawElement( 'span', [ 'style' => 'white-space: nowrap;' ],
254 Linker
::userToolLinks( $uploader->getId(), $uploader->getName() )
258 $row .= Html
::element( 'span', [ 'class' => 'history-deleted' ],
259 $this->msg( 'rev-deleted-user' )->text()
262 $row .= Html
::closeElement( 'td' );
264 // Don't show deleted descriptions
265 if ( $file->isDeleted( File
::DELETED_COMMENT
) ) {
266 $row .= Html
::rawElement( 'td', [],
267 Html
::element( 'span', [ 'class' => 'history-deleted' ],
268 $this->msg( 'rev-deleted-comment' )->text()
272 $contLang = MediaWikiServices
::getInstance()->getContentLanguage();
273 $row .= Html
::rawElement( 'td', [ 'dir' => $contLang->getDir() ], $formattedComment );
277 $this->getHookRunner()->onImagePageFileHistoryLine( $this, $file, $row, $rowClass );
279 return Html
::rawElement( 'tr', [ 'class' => $rowClass ], $row ) . "\n";
285 * @return string|null
287 protected function getThumbForLine( $file, $iscur ) {
288 $user = $this->getUser();
289 if ( !$file->allowInlineDisplay() ||
290 $file->isDeleted( File
::DELETED_FILE
) ||
291 !$file->userCan( File
::DELETED_FILE
, $user )
296 $thumbnail = $file->transform(
300 'isFilePageThumb' => $iscur // old revisions are already versioned
307 $lang = $this->getLanguage();
308 $timestamp = wfTimestamp( TS_MW
, $file->getTimestamp() );
310 'filehist-thumbtext',
311 $lang->userTimeAndDate( $timestamp, $user ),
312 $lang->userDate( $timestamp, $user ),
313 $lang->userTime( $timestamp, $user )
315 return $thumbnail->toHtml( [ 'alt' => $alt, 'file-link' => true, 'loading' => 'lazy' ] );
319 * @param bool $enable
320 * @deprecated since 1.38, use ::setPreventClickjacking() instead
322 protected function preventClickjacking( $enable = true ) {
323 $this->preventClickjacking
= $enable;
327 * @param bool $enable
330 protected function setPreventClickjacking( bool $enable ) {
331 $this->preventClickjacking
= $enable;
337 public function getPreventClickjacking() {
338 return $this->preventClickjacking
;