1<?php 2 3namespace dokuwiki\Ui; 4 5use dokuwiki\ChangeLog\PageChangeLog; 6use dokuwiki\ChangeLog\MediaChangeLog; 7use dokuwiki\Extension\Event; 8use dokuwiki\Form\Form; 9 10/** 11 * DokuWiki Revisions Interface 12 * 13 * @package dokuwiki\Ui 14 */ 15class Revisions extends Ui 16{ 17 protected $first; 18 protected $media_id; 19 20 /** 21 * Revisions Ui constructor 22 * 23 * @param int $first skip the first n changelog lines 24 * @param bool|string $media_id id of media, or false for current page 25 */ 26 public function __construct($first = 0, $media_id = false) 27 { 28 $this->first = $first; 29 $this->media_id = $media_id; 30 } 31 32 /** 33 * Display list of old revisions 34 * 35 * @author Andreas Gohr <andi@splitbrain.org> 36 * @author Ben Coburn <btcoburn@silicodon.net> 37 * @author Kate Arzamastseva <pshns@ukr.net> 38 * 39 * @triggers HTML_REVISIONSFORM_OUTPUT 40 * @return void 41 */ 42 public function show() 43 { 44 global $ID; 45 46 if ($this->media_id) { 47 return $this->showMediaRevisions($this->media_id); 48 } else { 49 return $this->showPageRevisions($ID); 50 } 51 } 52 53 /** 54 * Display a list of Media Revisions in the MediaManager 55 * 56 * @param string $id media id 57 * @return void 58 */ 59 protected function showMediaRevisions($id) 60 { 61 global $lang; 62 63 // get revisions, and set correct pagenation parameters (first, hasNext) 64 $first = $this->first; 65 $hasNext = false; 66 $revisions = $this->getRevisions($first, $hasNext); 67 68 // create the form 69 $form = new Form([ 70 'id' => 'page__revisions', // must not be "media__revisions" 71 'action' => media_managerURL(['image' => $id], '&'), 72 'class' => 'changes', 73 ]); 74 $form->setHiddenField('mediado', 'diff'); // required for media revisions 75 $form->addTagOpen('div')->addClass('no'); 76 77 // start listing 78 $form->addTagOpen('ul'); 79 foreach ($revisions as $info) { 80 $rev = $info['date']; 81 $class = ($info['type'] === DOKU_CHANGE_TYPE_MINOR_EDIT) ? 'minor' : ''; 82 $form->addTagOpen('li')->addClass($class); 83 $form->addTagOpen('div')->addClass('li'); 84 85 if (isset($info['current'])) { 86 $form->addCheckbox('rev2[]')->val('current'); 87 } elseif (file_exists(mediaFN($id, $rev))) { 88 $form->addCheckbox('rev2[]')->val($rev); 89 } else { 90 $form->addCheckbox('')->val($rev)->attr('disabled','disabled'); 91 } 92 $form->addHTML(' '); 93 94 $objRevInfo = $this->getObjRevInfo($info); 95 $html = implode(' ', [ 96 $objRevInfo->editDate(), // edit date and time 97 $objRevInfo->difflink(), // link to diffview icon 98 $objRevInfo->itemName(), // name of page or media 99 '<div>', 100 $objRevInfo->editSummary(), // edit summary 101 $objRevInfo->editor(), // editor info 102 html_sizechange($info['sizechange']), // size change indicator 103 $objRevInfo->currentIndicator(), // current indicator (only when k=1) 104 '</div>', 105 ]); 106 $form->addHTML($html); 107 108 $form->addTagClose('div'); 109 $form->addTagClose('li'); 110 } 111 $form->addTagClose('ul'); // end of revision list 112 113 // show button for diff view 114 $form->addButton('do[diff]', $lang['diff2'])->attr('type', 'submit'); 115 116 $form->addTagClose('div'); // close div class=no 117 118 // emit HTML_REVISIONSFORM_OUTPUT event 119 Event::createAndTrigger('HTML_REVISIONSFORM_OUTPUT', $form, null, false); 120 print $form->toHTML(); 121 122 // provide navigation for pagenated revision list (of pages and/or media files) 123 print $this->htmlNavigation($id, $first, $hasNext); 124 } 125 126 /** 127 * Display a list of Page Revisions 128 * 129 * @return void 130 */ 131 protected function showPageRevisions($id) 132 { 133 global $lang; 134 135 // get revisions, and set correct pagenation parameters (first, hasNext) 136 $first = $this->first; 137 $hasNext = false; 138 $revisions = $this->getRevisions($first, $hasNext); 139 140 // print intro 141 print p_locale_xhtml('revisions'); 142 143 // create the form 144 $form = new Form([ 145 'id' => 'page__revisions', 146 'class' => 'changes', 147 ]); 148 $form->addTagOpen('div')->addClass('no'); 149 150 // start listing 151 $form->addTagOpen('ul'); 152 foreach ($revisions as $info) { 153 $rev = $info['date']; 154 $class = ($info['type'] === DOKU_CHANGE_TYPE_MINOR_EDIT) ? 'minor' : ''; 155 $form->addTagOpen('li')->addClass($class); 156 $form->addTagOpen('div')->addClass('li'); 157 158 if (page_exists($id, $rev)) { 159 $form->addCheckbox('rev2[]')->val($rev); 160 } else { 161 $form->addCheckbox('')->val($rev)->attr('disabled','disabled'); 162 } 163 $form->addHTML(' '); 164 165 $objRevInfo = $this->getObjRevInfo($info); 166 $html = implode(' ', [ 167 $objRevInfo->editDate(), // edit date and time 168 $objRevInfo->difflink(), // link to diffview icon 169 $objRevInfo->itemName(), // name of page or media 170 $objRevInfo->editSummary(), // edit summary 171 $objRevInfo->editor(), // editor info 172 html_sizechange($info['sizechange']), // size change indicator 173 $objRevInfo->currentIndicator(), // current indicator (only when k=1) 174 ]); 175 $form->addHTML($html); 176 $form->addTagClose('div'); 177 $form->addTagClose('li'); 178 } 179 $form->addTagClose('ul'); // end of revision list 180 181 // show button for diff view 182 $form->addButton('do[diff]', $lang['diff2'])->attr('type', 'submit'); 183 184 $form->addTagClose('div'); // close div class=no 185 186 // emit HTML_REVISIONSFORM_OUTPUT event 187 Event::createAndTrigger('HTML_REVISIONSFORM_OUTPUT', $form, null, false); 188 print $form->toHTML(); 189 190 // provide navigation for pagenated revision list (of pages and/or media files) 191 print $this->htmlNavigation($id, $first, $hasNext); 192 } 193 194 195 /** 196 * Get revisions, and set correct pagenation parameters (first, hasNext) 197 * 198 * @param int $first 199 * @param bool $hasNext 200 * @return array revisions to be shown in a pagenated list 201 */ 202 protected function getRevisions(&$first, &$hasNext) 203 { 204 global $INFO, $conf; 205 206 if ($this->media_id) { 207 $changelog = new MediaChangeLog($this->media_id); 208 } else { 209 $changelog = new PageChangeLog($INFO['id']); 210 } 211 212 $revisions = []; 213 214 /* we need to get one additional log entry to be able to 215 * decide if this is the last page or is there another one. 216 * see also Ui\Recent::getRecents() 217 */ 218 $revlist = $changelog->getRevisions($first, $conf['recent'] +1); 219 if (count($revlist) == 0 && $first != 0) { 220 $first = 0; 221 $revlist = $changelog->getRevisions($first, $conf['recent'] +1); 222 } 223 $exists = ($this->media_id) ? file_exists(mediaFN($this->media_id)) : $INFO['exists']; 224 if ($first === 0 && $exists) { 225 // add current page or media as revision[0] 226 if ($this->media_id) { 227 $rev = filemtime(fullpath(mediaFN($this->media_id))); 228 $revisions[] = $changelog->getRevisionInfo($rev) + array( 229 'media' => true, 230 'current' => true, 231 ); 232 } else { 233 $revisions[] = array( 234 'date' => $INFO['lastmod'], 235 'ip' => null, 236 'type' => $INFO['meta']['last_change']['type'], 237 'id' => $INFO['id'], 238 'user' => $INFO['editor'], 239 'sum' => $INFO['sum'], 240 'extra' => null, 241 'sizechange' => $INFO['meta']['last_change']['sizechange'], 242 'current' => true, 243 ); 244 } 245 } 246 247 // decide if this is the last page or is there another one 248 $hasNext = false; 249 if (count($revlist) > $conf['recent']) { 250 $hasNext = true; 251 array_pop($revlist); // remove one additional log entry 252 } 253 254 // append each revison info array to the revisions 255 foreach ($revlist as $rev) { 256 if ($this->media_id) { 257 $revisions[] = $changelog->getRevisionInfo($rev) + array('media' => true); 258 } else { 259 $revisions[] = $changelog->getRevisionInfo($rev); 260 } 261 } 262 return $revisions; 263 } 264 265 /** 266 * Navigation buttons for Pagenation (prev/next) 267 * 268 * @param string $id page id or media id 269 * @param int $first 270 * @param bool $hasNext 271 * @return array html 272 */ 273 protected function htmlNavigation($id, $first, $hasNext) 274 { 275 global $conf; 276 277 $html = '<div class="pagenav">'; 278 $last = $first + $conf['recent']; 279 if ($first > 0) { 280 $first = max($first - $conf['recent'], 0); 281 $html.= '<div class="pagenav-prev">'; 282 if ($this->media_id) { 283 $html.= html_btn('newer', $id, "p", media_managerURL(['first' => $first], '&', false, true)); 284 } else { 285 $html.= html_btn('newer', $id, "p" ,['do' => 'revisions', 'first' => $first]); 286 } 287 $html.= '</div>'; 288 } 289 if ($hasNext) { 290 $html.= '<div class="pagenav-next">'; 291 if ($this->media_id) { 292 $html.= html_btn('older', $id, "n", media_managerURL(['first' => $last], '&', false, true)); 293 } else { 294 $html.= html_btn('older', $id, "n", ['do' => 'revisions', 'first' => $last]); 295 } 296 $html.= '</div>'; 297 } 298 $html.= '</div>'; 299 return $html; 300 } 301 302/* ---------------------------------------------------------------------------------------- 303 $html = implode(' ', [ // Ui\Recent display // Ui\Revisions display 304 $objRevInfo->itemIcon(), // filetype icon 305 $objRevInfo->editDate(), // edit date and time // edit date and time 306 $objRevInfo->difflink(), // link to diffview icon // link to diffview icon 307 $objRevInfo->revisionlink(), // linkto revisions icon 308 $objRevInfo->itemName(), // name of page or media // name of page 309 $objRevInfo->editSummary(), // edit summary // edit summary 310 $objRevInfo->editor(), // editor info // editor info 311 html_sizechange // size change indicator // size change indicator 312 // current indicator (only when k=1) 313 ]); 314 * ---------------------------------------------------------------------------------------- */ 315 316 /** 317 * Returns instance of objRevInfo 318 * 319 * @param array $info Revision info structure of a page or media file 320 * @return objRevInfo object (anonymous class) 321 */ 322 protected function getObjRevInfo(array $info) 323 { 324 return new class ($info) // anonymous class (objRevInfo) 325 { 326 protected $info; 327 328 public function __construct(array $info) 329 { 330 $this->info = $info; 331 } 332 333 // current indicator 334 public function currentIndicator() 335 { 336 global $lang; 337 return ($this->info['current']) ? '('.$lang['current'].')' : ''; 338 } 339 340 // edit date and time of the page or media file 341 public function editDate() 342 { 343 return '<span class="date">'. dformat($this->info['date']) .'</span>'; 344 } 345 346 // edit summary 347 public function editSummary() 348 { 349 return '<span class="sum">'.' – '. hsc($this->info['sum']).'</span>'; 350 } 351 352 // editor of the page or media file 353 public function editor() 354 { 355 // slightly different with display of Ui\Recent, i.e. external edit 356 global $lang; 357 $html = '<span class="user">'; 358 if (!$this->info['user'] && !$this->info['ip']) { 359 $html.= '('.$lang['external_edit'].')'; 360 } elseif ($this->info['user']) { 361 $html.= '<bdi>'. editorinfo($this->info['user']) .'</bdi>'; 362 if (auth_ismanager()) $html.= ' <bdo dir="ltr">('. $this->info['ip'] .')</bdo>'; 363 } else { 364 $html.= '<bdo dir="ltr">'. $this->info['ip'] .'</bdo>'; 365 } 366 $html.= '</span>'; 367 return $html; 368 } 369 370 // name of the page or media file 371 public function itemName() 372 { 373 // slightly different with display of Ui\Recent, i.e. revison may not exists 374 $id = $this->info['id']; 375 $rev = $this->info['date']; 376 377 if (isset($this->info['media'])) { 378 // media file revision 379 if (isset($this->info['current'])) { 380 $href = media_managerURL(['image'=> $id, 'tab_details'=> 'view'], '&'); 381 $html = '<a href="'.$href.'" class="wikilink1">'.$id.'</a>'; 382 } elseif (file_exists(mediaFN($id, $rev))) { 383 $href = media_managerURL(['image'=> $id, 'tab_details'=> 'view', 'rev'=> $rev], '&'); 384 $html = '<a href="'.$href.'" class="wikilink1">'.$id.'</a>'; 385 } else { 386 $html = $id; 387 } 388 return $html; 389 } else { 390 // page revision 391 $display_name = useHeading('navigation') ? hsc(p_get_first_heading($id)) : $id; 392 if (!$display_name) $display_name = $id; 393 if ($this->info['current'] || page_exists($id, $rev)) { 394 $href = wl($id, "rev=$rev", false, '&'); 395 $html = '<a href="'.$href.'" class="wikilink1">'.$display_name.'</a>'; 396 } else { 397 $html = $display_name; 398 } 399 return $html; 400 } 401 } 402 403 // icon difflink 404 public function difflink() 405 { 406 global $lang; 407 $id = $this->info['id']; 408 $rev = $this->info['date']; 409 410 if (isset($this->info['media'])) { 411 // media file revision 412 if (isset($this->info['current']) || !file_exists(mediaFN($id, $rev))) { 413 $html = '<img src="'.DOKU_BASE.'lib/images/blank.gif" width="15" height="11" alt="" />'; 414 } else { 415 $href = media_managerURL(['image'=> $id, 'rev'=> $rev, 'mediado'=>'diff'], '&'); 416 $html = '<a href="'.$href.'" class="diff_link">' 417 . '<img src="'.DOKU_BASE.'lib/images/diff.png" width="15" height="11"' 418 . ' title="'. $lang['diff'] .'" alt="'.$lang['diff'] .'" />' 419 . '</a> '; 420 } 421 return $html; 422 } else { 423 // page revision 424 if ($this->info['current'] || !page_exists($id, $rev)) { 425 $html = '<img src="'.DOKU_BASE.'lib/images/blank.gif" width="15" height="11" alt="" />'; 426 } else { 427 $href = wl($id, "rev=$rev,do=diff", false, '&'); 428 $html = '<a href="'.$href.'" class="diff_link">' 429 . '<img src="'.DOKU_BASE.'lib/images/diff.png" width="15" height="11"' 430 . ' title="'.$lang['diff'].'" alt="'.$lang['diff'].'" />' 431 . '</a>'; 432 } 433 return $html; 434 } 435 } 436 437 // size change 438 public function sizeChange() 439 { 440 $class = 'sizechange'; 441 $value = filesize_h(abs($this->info['sizechange'])); 442 if ($this->info['sizechange'] > 0) { 443 $class .= ' positive'; 444 $value = '+' . $value; 445 } elseif ($this->info['sizechange'] < 0) { 446 $class .= ' negative'; 447 $value = '-' . $value; 448 } else { 449 $value = '±' . $value; 450 } 451 return '<span class="'.$class.'">'.$value.'</span>'; 452 } 453 }; // end of anonymous class (objRevInfo) 454 } 455 456} 457