[ Index ]

PHP Cross Reference of DokuWiki

title

Body

[close]

/inc/Ui/ -> MediaDiff.php (source)

   1  <?php
   2  
   3  namespace dokuwiki\Ui;
   4  
   5  use dokuwiki\ChangeLog\MediaChangeLog;
   6  use dokuwiki\ChangeLog\RevisionInfo;
   7  use dokuwiki\Form\Form;
   8  use InvalidArgumentException;
   9  use JpegMeta;
  10  
  11  /**
  12   * DokuWiki MediaDiff Interface
  13   *
  14   * @package dokuwiki\Ui
  15   */
  16  class MediaDiff extends Diff
  17  {
  18      /* @var MediaChangeLog */
  19      protected $changelog;
  20  
  21      /* @var RevisionInfo older revision */
  22      protected $RevInfo1;
  23      /* @var RevisionInfo newer revision */
  24      protected $RevInfo2;
  25  
  26      /* @var bool */
  27      protected $is_img;
  28  
  29      /**
  30       * MediaDiff Ui constructor
  31       *
  32       * @param string $id media id
  33       */
  34      public function __construct($id)
  35      {
  36          if (!isset($id)) {
  37              throw new InvalidArgumentException('media id should not be empty!');
  38          }
  39  
  40          // init preference
  41          $this->preference['fromAjax'] = false;  // see dokuwiki\Ajax::callMediadiff()
  42          $this->preference['showIntro'] = false;
  43          $this->preference['difftype'] = 'both'; // diff view type: both, opacity or portions
  44  
  45          parent::__construct($id);
  46      }
  47  
  48      /** @inheritdoc */
  49      protected function setChangeLog()
  50      {
  51          $this->changelog = new MediaChangeLog($this->id);
  52      }
  53  
  54      /**
  55       * Handle requested revision(s) and diff view preferences
  56       *
  57       * @return void
  58       */
  59      protected function handle()
  60      {
  61          global $INPUT;
  62  
  63          // retrieve requested rev or rev2
  64          parent::handle();
  65  
  66          // requested diff view type
  67          if ($INPUT->has('difftype')) {
  68              $this->preference['difftype'] = $INPUT->str('difftype');
  69          }
  70      }
  71  
  72      /**
  73       * Prepare revision info of comparison pair
  74       */
  75      protected function preProcess()
  76      {
  77          $changelog =& $this->changelog;
  78  
  79          // create revision info object for older and newer sides
  80          // RevInfo1 : older, left side
  81          // RevInfo2 : newer, right side
  82  
  83          $changelogRev1 = $changelog->getRevisionInfo($this->rev1);
  84          $changelogRev2 = $changelog->getRevisionInfo($this->rev2);
  85  
  86          $this->RevInfo1 = new RevisionInfo($changelogRev1);
  87          $this->RevInfo2 = new RevisionInfo($changelogRev2);
  88  
  89          $this->is_img = preg_match('/\.(jpe?g|gif|png)$/', $this->id);
  90  
  91          foreach ([$this->RevInfo1, $this->RevInfo2] as $RevInfo) {
  92              $isCurrent = $changelog->isCurrentRevision($RevInfo->val('date'));
  93              $RevInfo->isCurrent($isCurrent);
  94  
  95              if ($this->is_img) {
  96                  $rev = $isCurrent ? '' : $RevInfo->val('date');
  97                  $meta = new JpegMeta(mediaFN($this->id, $rev));
  98                  // get image width and height for the media manager preview panel
  99                  $RevInfo->append([
 100                      'previewSize' => media_image_preview_size($this->id, $rev, $meta)
 101                  ]);
 102              }
 103          }
 104  
 105          // re-check image, ensure minimum image width for showImageDiff()
 106          $this->is_img = ($this->is_img
 107              && ($this->RevInfo1->val('previewSize')[0] ?? 0) >= 30
 108              && ($this->RevInfo2->val('previewSize')[0] ?? 0) >= 30
 109          );
 110          // adjust requested diff view type
 111          if (!$this->is_img) {
 112              $this->preference['difftype'] = 'both';
 113          }
 114      }
 115  
 116  
 117      /**
 118       * Shows difference between two revisions of media
 119       *
 120       * @author Kate Arzamastseva <pshns@ukr.net>
 121       */
 122      public function show()
 123      {
 124          global $conf;
 125  
 126          $ns = getNS($this->id);
 127          $auth = auth_quickaclcheck("$ns:*");
 128  
 129          if ($auth < AUTH_READ || !$this->id || !$conf['mediarevisions']) return;
 130  
 131          // retrieve form parameters: rev, rev2, difftype
 132          $this->handle();
 133          // prepare revision info of comparison pair
 134          $this->preProcess();
 135  
 136          // display intro
 137          if ($this->preference['showIntro']) echo p_locale_xhtml('diff');
 138  
 139          // print form to choose diff view type
 140          if ($this->is_img && !$this->preference['fromAjax']) {
 141              $this->showDiffViewSelector();
 142              echo '<div id="mediamanager__diff" >';
 143          }
 144  
 145          switch ($this->preference['difftype']) {
 146              case 'opacity':
 147              case 'portions':
 148                  $this->showImageDiff();
 149                  break;
 150              case 'both':
 151              default:
 152                  $this->showFileDiff();
 153                  break;
 154          }
 155  
 156          if ($this->is_img && !$this->preference['fromAjax']) {
 157              echo '</div>';
 158          }
 159      }
 160  
 161      /**
 162       * Print form to choose diff view type
 163       * the dropdown is to be added through JavaScript, see lib/scripts/media.js
 164       */
 165      protected function showDiffViewSelector()
 166      {
 167          // use timestamp for current revision, date may be false when revisions < 2
 168          [$rev1, $rev2] = [(int)$this->RevInfo1->val('date'), (int)$this->RevInfo2->val('date')];
 169  
 170          echo '<div class="diffoptions group">';
 171  
 172          $form = new Form([
 173              'id' => 'mediamanager__form_diffview',
 174              'action' => media_managerURL([], '&'),
 175              'method' => 'get',
 176              'class' => 'diffView',
 177          ]);
 178          $form->addTagOpen('div')->addClass('no');
 179          $form->setHiddenField('sectok', null);
 180          $form->setHiddenField('mediado', 'diff');
 181          $form->setHiddenField('rev2[0]', $rev1);
 182          $form->setHiddenField('rev2[1]', $rev2);
 183          $form->addTagClose('div');
 184          echo $form->toHTML();
 185  
 186          echo '</div>'; // .diffoptions
 187      }
 188  
 189      /**
 190       * Prints two images side by side
 191       * and slider
 192       *
 193       * @author Kate Arzamastseva <pshns@ukr.net>
 194       */
 195      protected function showImageDiff()
 196      {
 197          $rev1 = $this->RevInfo1->isCurrent() ? '' : $this->RevInfo1->val('date');
 198          $rev2 = $this->RevInfo2->isCurrent() ? '' : $this->RevInfo2->val('date');
 199  
 200          // diff view type: opacity or portions
 201          $type = $this->preference['difftype'];
 202  
 203          // adjust image width, right side (newer) has priority
 204          $rev1Size = $this->RevInfo1->val('previewSize');
 205          $rev2Size = $this->RevInfo2->val('previewSize');
 206          if ($rev1Size != $rev2Size) {
 207              if ($rev2Size[0] > $rev1Size[0]) {
 208                  $rev1Size = $rev2Size;
 209              }
 210          }
 211  
 212          $rev1Src = ml($this->id, ['rev' => $rev1, 'h' => $rev1Size[1], 'w' => $rev1Size[0]]);
 213          $rev2Src = ml($this->id, ['rev' => $rev2, 'h' => $rev1Size[1], 'w' => $rev1Size[0]]);
 214  
 215          // slider
 216          echo '<div class="slider" style="max-width: ' . ($rev1Size[0] - 20) . 'px;" ></div>';
 217  
 218          // two images in divs
 219          echo '<div class="imageDiff ' . $type . '">';
 220          echo '<div class="image1" style="max-width: ' . $rev1Size[0] . 'px;">';
 221          echo '<img src="' . $rev1Src . '" alt="" />';
 222          echo '</div>';
 223          echo '<div class="image2" style="max-width: ' . $rev1Size[0] . 'px;">';
 224          echo '<img src="' . $rev2Src . '" alt="" />';
 225          echo '</div>';
 226          echo '</div>';
 227      }
 228  
 229      /**
 230       * Shows difference between two revisions of media file
 231       *
 232       * @author Kate Arzamastseva <pshns@ukr.net>
 233       */
 234      protected function showFileDiff()
 235      {
 236          global $lang;
 237  
 238          $ns = getNS($this->id);
 239          $auth = auth_quickaclcheck("$ns:*");
 240  
 241          $rev1 = $this->RevInfo1->isCurrent() ? '' : (int)$this->RevInfo1->val('date');
 242          $rev2 = $this->RevInfo2->isCurrent() ? '' : (int)$this->RevInfo2->val('date');
 243  
 244          // revision title
 245          $rev1Title = trim($this->RevInfo1->showRevisionTitle() . ' ' . $this->RevInfo1->showCurrentIndicator());
 246          $rev1Summary = ($this->RevInfo1->val('date'))
 247              ? $this->RevInfo1->showEditSummary() . ' ' . $this->RevInfo1->showEditor()
 248              : '';
 249          $rev2Title = trim($this->RevInfo2->showRevisionTitle() . ' ' . $this->RevInfo2->showCurrentIndicator());
 250          $rev2Summary = ($this->RevInfo2->val('date'))
 251              ? $this->RevInfo2->showEditSummary() . ' ' . $this->RevInfo2->showEditor()
 252              : '';
 253  
 254          $rev1Meta = new JpegMeta(mediaFN($this->id, $rev1));
 255          $rev2Meta = new JpegMeta(mediaFN($this->id, $rev2));
 256  
 257          // display diff view table
 258          echo '<div class="table">';
 259          echo '<table>';
 260          echo '<tr>';
 261          echo '<th>' . $rev1Title . ' ' . $rev1Summary . '</th>';
 262          echo '<th>' . $rev2Title . ' ' . $rev2Summary . '</th>';
 263          echo '</tr>';
 264  
 265          echo '<tr class="image">';
 266          echo '<td>';
 267          media_preview($this->id, $auth, $rev1, $rev1Meta); // $auth not used in media_preview()?
 268          echo '</td>';
 269  
 270          echo '<td>';
 271          media_preview($this->id, $auth, $rev2, $rev2Meta);
 272          echo '</td>';
 273          echo '</tr>';
 274  
 275          echo '<tr class="actions">';
 276          echo '<td>';
 277          media_preview_buttons($this->id, $auth, $rev1); // $auth used in media_preview_buttons()
 278          echo '</td>';
 279  
 280          echo '<td>';
 281          media_preview_buttons($this->id, $auth, $rev2);
 282          echo '</td>';
 283          echo '</tr>';
 284  
 285          $rev1Tags = media_file_tags($rev1Meta);
 286          $rev2Tags = media_file_tags($rev2Meta);
 287          // FIXME rev2Tags-only stuff ignored
 288          foreach ($rev1Tags as $key => $tag) {
 289              if ($tag['value'] != $rev2Tags[$key]['value']) {
 290                  $rev2Tags[$key]['highlighted'] = true;
 291                  $rev1Tags[$key]['highlighted'] = true;
 292              } elseif (!$tag['value'] || !$rev2Tags[$key]['value']) {
 293                  unset($rev2Tags[$key]);
 294                  unset($rev1Tags[$key]);
 295              }
 296          }
 297  
 298          echo '<tr>';
 299          foreach ([$rev1Tags, $rev2Tags] as $tags) {
 300              echo '<td>';
 301  
 302              echo '<dl class="img_tags">';
 303              foreach ($tags as $tag) {
 304                  $value = cleanText($tag['value']);
 305                  if (!$value) $value = '-';
 306                  echo '<dt>' . $lang[$tag['tag'][1]] . '</dt>';
 307                  echo '<dd>';
 308                  if (!empty($tag['highlighted'])) echo '<strong>';
 309                  if ($tag['tag'][2] == 'date') {
 310                      echo dformat($value);
 311                  } else {
 312                      echo hsc($value);
 313                  }
 314                  if (!empty($tag['highlighted'])) echo '</strong>';
 315                  echo '</dd>';
 316              }
 317              echo '</dl>';
 318  
 319              echo '</td>';
 320          }
 321          echo '</tr>';
 322  
 323          echo '</table>';
 324          echo '</div>';
 325      }
 326  }