Only store currently-existing categories in the categories table
[mediawiki.git] / includes / page / ImageHistoryList.php
blob607357fcd14b98a4d9ceec01e1f66b0ac14decd4
1 <?php
2 /**
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
18 * @file
21 /**
22 * Builds the image revision log shown on image pages
24 * @ingroup Media
26 class ImageHistoryList extends ContextSource {
28 /**
29 * @var Title
31 protected $title;
33 /**
34 * @var File
36 protected $img;
38 /**
39 * @var ImagePage
41 protected $imagePage;
43 /**
44 * @var File
46 protected $current;
48 protected $repo, $showThumb;
49 protected $preventClickjacking = false;
51 /**
52 * @param ImagePage $imagePage
54 public function __construct( $imagePage ) {
55 global $wgShowArchiveThumbnails;
56 $this->current = $imagePage->getPage()->getFile();
57 $this->img = $imagePage->getDisplayedFile();
58 $this->title = $imagePage->getTitle();
59 $this->imagePage = $imagePage;
60 $this->showThumb = $wgShowArchiveThumbnails && $this->img->canRender();
61 $this->setContext( $imagePage->getContext() );
64 /**
65 * @return ImagePage
67 public function getImagePage() {
68 return $this->imagePage;
71 /**
72 * @return File
74 public function getFile() {
75 return $this->img;
78 /**
79 * @param string $navLinks
80 * @return string
82 public function beginImageHistoryList( $navLinks = '' ) {
83 return Xml::element( 'h2', [ 'id' => 'filehistory' ], $this->msg( 'filehist' )->text() )
84 . "\n"
85 . "<div id=\"mw-imagepage-section-filehistory\">\n"
86 . $this->msg( 'filehist-help' )->parseAsBlock()
87 . $navLinks . "\n"
88 . Xml::openElement( 'table', [ 'class' => 'wikitable filehistory' ] ) . "\n"
89 . '<tr><th></th>'
90 . ( $this->current->isLocal()
91 && ( $this->getUser()->isAllowedAny( 'delete', 'deletedhistory' ) ) ? '<th></th>' : '' )
92 . '<th>' . $this->msg( 'filehist-datetime' )->escaped() . '</th>'
93 . ( $this->showThumb ? '<th>' . $this->msg( 'filehist-thumb' )->escaped() . '</th>' : '' )
94 . '<th>' . $this->msg( 'filehist-dimensions' )->escaped() . '</th>'
95 . '<th>' . $this->msg( 'filehist-user' )->escaped() . '</th>'
96 . '<th>' . $this->msg( 'filehist-comment' )->escaped() . '</th>'
97 . "</tr>\n";
101 * @param string $navLinks
102 * @return string
104 public function endImageHistoryList( $navLinks = '' ) {
105 return "</table>\n$navLinks\n</div>\n";
109 * @param bool $iscur
110 * @param File $file
111 * @return string
113 public function imageHistoryLine( $iscur, $file ) {
114 global $wgContLang;
116 $user = $this->getUser();
117 $lang = $this->getLanguage();
118 $timestamp = wfTimestamp( TS_MW, $file->getTimestamp() );
119 $img = $iscur ? $file->getName() : $file->getArchiveName();
120 $userId = $file->getUser( 'id' );
121 $userText = $file->getUser( 'text' );
122 $description = $file->getDescription( File::FOR_THIS_USER, $user );
124 $local = $this->current->isLocal();
125 $row = $selected = '';
127 // Deletion link
128 if ( $local && ( $user->isAllowedAny( 'delete', 'deletedhistory' ) ) ) {
129 $row .= '<td>';
130 # Link to remove from history
131 if ( $user->isAllowed( 'delete' ) ) {
132 $q = [ 'action' => 'delete' ];
133 if ( !$iscur ) {
134 $q['oldimage'] = $img;
136 $row .= Linker::linkKnown(
137 $this->title,
138 $this->msg( $iscur ? 'filehist-deleteall' : 'filehist-deleteone' )->escaped(),
139 [], $q
142 # Link to hide content. Don't show useless link to people who cannot hide revisions.
143 $canHide = $user->isAllowed( 'deleterevision' );
144 if ( $canHide || ( $user->isAllowed( 'deletedhistory' ) && $file->getVisibility() ) ) {
145 if ( $user->isAllowed( 'delete' ) ) {
146 $row .= '<br />';
148 // If file is top revision or locked from this user, don't link
149 if ( $iscur || !$file->userCan( File::DELETED_RESTRICTED, $user ) ) {
150 $del = Linker::revDeleteLinkDisabled( $canHide );
151 } else {
152 list( $ts, ) = explode( '!', $img, 2 );
153 $query = [
154 'type' => 'oldimage',
155 'target' => $this->title->getPrefixedText(),
156 'ids' => $ts,
158 $del = Linker::revDeleteLink( $query,
159 $file->isDeleted( File::DELETED_RESTRICTED ), $canHide );
161 $row .= $del;
163 $row .= '</td>';
166 // Reversion link/current indicator
167 $row .= '<td>';
168 if ( $iscur ) {
169 $row .= $this->msg( 'filehist-current' )->escaped();
170 } elseif ( $local && $this->title->quickUserCan( 'edit', $user )
171 && $this->title->quickUserCan( 'upload', $user )
173 if ( $file->isDeleted( File::DELETED_FILE ) ) {
174 $row .= $this->msg( 'filehist-revert' )->escaped();
175 } else {
176 $row .= Linker::linkKnown(
177 $this->title,
178 $this->msg( 'filehist-revert' )->escaped(),
181 'action' => 'revert',
182 'oldimage' => $img,
183 'wpEditToken' => $user->getEditToken( $img )
188 $row .= '</td>';
190 // Date/time and image link
191 if ( $file->getTimestamp() === $this->img->getTimestamp() ) {
192 $selected = "class='filehistory-selected'";
194 $row .= "<td $selected style='white-space: nowrap;'>";
195 if ( !$file->userCan( File::DELETED_FILE, $user ) ) {
196 # Don't link to unviewable files
197 $row .= '<span class="history-deleted">'
198 . $lang->userTimeAndDate( $timestamp, $user ) . '</span>';
199 } elseif ( $file->isDeleted( File::DELETED_FILE ) ) {
200 if ( $local ) {
201 $this->preventClickjacking();
202 $revdel = SpecialPage::getTitleFor( 'Revisiondelete' );
203 # Make a link to review the image
204 $url = Linker::linkKnown(
205 $revdel,
206 $lang->userTimeAndDate( $timestamp, $user ),
209 'target' => $this->title->getPrefixedText(),
210 'file' => $img,
211 'token' => $user->getEditToken( $img )
214 } else {
215 $url = $lang->userTimeAndDate( $timestamp, $user );
217 $row .= '<span class="history-deleted">' . $url . '</span>';
218 } elseif ( !$file->exists() ) {
219 $row .= '<span class="mw-file-missing">'
220 . $lang->userTimeAndDate( $timestamp, $user ) . '</span>';
221 } else {
222 $url = $iscur ? $this->current->getUrl() : $this->current->getArchiveUrl( $img );
223 $row .= Xml::element(
224 'a',
225 [ 'href' => $url ],
226 $lang->userTimeAndDate( $timestamp, $user )
229 $row .= "</td>";
231 // Thumbnail
232 if ( $this->showThumb ) {
233 $row .= '<td>' . $this->getThumbForLine( $file ) . '</td>';
236 // Image dimensions + size
237 $row .= '<td>';
238 $row .= htmlspecialchars( $file->getDimensionsString() );
239 $row .= $this->msg( 'word-separator' )->escaped();
240 $row .= '<span style="white-space: nowrap;">';
241 $row .= $this->msg( 'parentheses' )->sizeParams( $file->getSize() )->escaped();
242 $row .= '</span>';
243 $row .= '</td>';
245 // Uploading user
246 $row .= '<td>';
247 // Hide deleted usernames
248 if ( $file->isDeleted( File::DELETED_USER ) ) {
249 $row .= '<span class="history-deleted">'
250 . $this->msg( 'rev-deleted-user' )->escaped() . '</span>';
251 } else {
252 if ( $local ) {
253 $row .= Linker::userLink( $userId, $userText );
254 $row .= '<span style="white-space: nowrap;">';
255 $row .= Linker::userToolLinks( $userId, $userText );
256 $row .= '</span>';
257 } else {
258 $row .= htmlspecialchars( $userText );
261 $row .= '</td>';
263 // Don't show deleted descriptions
264 if ( $file->isDeleted( File::DELETED_COMMENT ) ) {
265 $row .= '<td><span class="history-deleted">' .
266 $this->msg( 'rev-deleted-comment' )->escaped() . '</span></td>';
267 } else {
268 $row .= '<td dir="' . $wgContLang->getDir() . '">' .
269 Linker::formatComment( $description, $this->title ) . '</td>';
272 $rowClass = null;
273 Hooks::run( 'ImagePageFileHistoryLine', [ $this, $file, &$row, &$rowClass ] );
274 $classAttr = $rowClass ? " class='$rowClass'" : '';
276 return "<tr{$classAttr}>{$row}</tr>\n";
280 * @param File $file
281 * @return string
283 protected function getThumbForLine( $file ) {
284 $lang = $this->getLanguage();
285 $user = $this->getUser();
286 if ( $file->allowInlineDisplay() && $file->userCan( File::DELETED_FILE, $user )
287 && !$file->isDeleted( File::DELETED_FILE )
289 $params = [
290 'width' => '120',
291 'height' => '120',
293 $timestamp = wfTimestamp( TS_MW, $file->getTimestamp() );
295 $thumbnail = $file->transform( $params );
296 $options = [
297 'alt' => $this->msg( 'filehist-thumbtext',
298 $lang->userTimeAndDate( $timestamp, $user ),
299 $lang->userDate( $timestamp, $user ),
300 $lang->userTime( $timestamp, $user ) )->text(),
301 'file-link' => true,
304 if ( !$thumbnail ) {
305 return $this->msg( 'filehist-nothumb' )->escaped();
308 return $thumbnail->toHtml( $options );
309 } else {
310 return $this->msg( 'filehist-nothumb' )->escaped();
315 * @param bool $enable
317 protected function preventClickjacking( $enable = true ) {
318 $this->preventClickjacking = $enable;
322 * @return bool
324 public function getPreventClickjacking() {
325 return $this->preventClickjacking;