xref: /dokuwiki/inc/Ui/PageDiff.php (revision edb50e6a038ec10ad5f1879b2753ed4f4eaf0b27)
1defc7576SSatoshi Sahara<?php
2defc7576SSatoshi Sahara
3defc7576SSatoshi Saharanamespace dokuwiki\Ui;
4defc7576SSatoshi Sahara
5defc7576SSatoshi Saharause dokuwiki\ChangeLog\PageChangeLog;
6defc7576SSatoshi Saharause dokuwiki\Form\Form;
7defc7576SSatoshi Sahara
8defc7576SSatoshi Sahara/**
9defc7576SSatoshi Sahara * DokuWiki PageDiff Interface
10defc7576SSatoshi Sahara *
11defc7576SSatoshi Sahara * @package dokuwiki\Ui
12defc7576SSatoshi Sahara */
1363ab9afeSSatoshi Saharaclass PageDiff extends Diff
14defc7576SSatoshi Sahara{
15*edb50e6aSSatoshi Sahara    /* @var string */
16*edb50e6aSSatoshi Sahara    protected $id;
17*edb50e6aSSatoshi Sahara
18*edb50e6aSSatoshi Sahara    /* @var string */
19*edb50e6aSSatoshi Sahara    protected $text = '';
20defc7576SSatoshi Sahara
21defc7576SSatoshi Sahara    /**
22defc7576SSatoshi Sahara     * PageDiff Ui constructor
23defc7576SSatoshi Sahara     *
24*edb50e6aSSatoshi Sahara     * @param string $id  page id
25defc7576SSatoshi Sahara     * @param string $text  when non-empty: compare with this text with most current version
26defc7576SSatoshi Sahara     */
27*edb50e6aSSatoshi Sahara    public function __construct($id = null, $text = null)
28defc7576SSatoshi Sahara    {
29defc7576SSatoshi Sahara        global $INFO;
30*edb50e6aSSatoshi Sahara        $this->id = isset($id) ? $id : $INFO['id'];
31*edb50e6aSSatoshi Sahara        if (isset($text)) $this->text = $text;
32*edb50e6aSSatoshi Sahara
33*edb50e6aSSatoshi Sahara        $this->preference['showIntro'] = true;
34*edb50e6aSSatoshi Sahara        $this->preference['difftype']  = null; // inline or sidebyside
35defc7576SSatoshi Sahara    }
36defc7576SSatoshi Sahara
37defc7576SSatoshi Sahara    /**
38*edb50e6aSSatoshi Sahara     * Determine requested diff view type
39defc7576SSatoshi Sahara     *
40*edb50e6aSSatoshi Sahara     * @param string $mode  diff view type (inline or sidebyside)
41defc7576SSatoshi Sahara     */
42*edb50e6aSSatoshi Sahara    public function getDiffType($mode = null)
43defc7576SSatoshi Sahara    {
44defc7576SSatoshi Sahara        global $INPUT;
45defc7576SSatoshi Sahara        global $INFO;
46*edb50e6aSSatoshi Sahara        $difftype =& $this->preference['difftype'];
47defc7576SSatoshi Sahara
48*edb50e6aSSatoshi Sahara        if (!isset($mode)) {
49*edb50e6aSSatoshi Sahara            // retrieve requested $difftype or read preference from DokuWiki cookie
50*edb50e6aSSatoshi Sahara            $difftype = $INPUT->str('difftype') ?: get_doku_pref('difftype', $mode);
51*edb50e6aSSatoshi Sahara            if (empty($difftype)) {
52*edb50e6aSSatoshi Sahara                $difftype = $INFO['ismobile'] ? 'inline' : 'sidebyside';
53*edb50e6aSSatoshi Sahara            }
54*edb50e6aSSatoshi Sahara        } elseif (in_array($mode, ['inline', 'sidebyside'])) {
55*edb50e6aSSatoshi Sahara            $difftype = $mode;
56*edb50e6aSSatoshi Sahara        } else {
57*edb50e6aSSatoshi Sahara            $difftype = 'sidebyside';
58*edb50e6aSSatoshi Sahara        }
59*edb50e6aSSatoshi Sahara        return $this->preference['difftype'];
60*edb50e6aSSatoshi Sahara    }
61*edb50e6aSSatoshi Sahara
62*edb50e6aSSatoshi Sahara    /**
63defc7576SSatoshi Sahara     * Determine requested revision(s)
64defc7576SSatoshi Sahara     */
65*edb50e6aSSatoshi Sahara    protected function getRevisions()
66*edb50e6aSSatoshi Sahara    {
67*edb50e6aSSatoshi Sahara        global $REV;
68*edb50e6aSSatoshi Sahara        global $INPUT;
69*edb50e6aSSatoshi Sahara
70defc7576SSatoshi Sahara        // we're trying to be clever here, revisions to compare can be either
71defc7576SSatoshi Sahara        // given as rev and rev2 parameters, with rev2 being optional. Or in an
72defc7576SSatoshi Sahara        // array in rev2.
73defc7576SSatoshi Sahara        $rev1 = $REV;
74defc7576SSatoshi Sahara
75defc7576SSatoshi Sahara        $rev2 = $INPUT->ref('rev2');
76defc7576SSatoshi Sahara        if (is_array($rev2)) {
77defc7576SSatoshi Sahara            $rev1 = (int) $rev2[0];
78defc7576SSatoshi Sahara            $rev2 = (int) $rev2[1];
79defc7576SSatoshi Sahara
80defc7576SSatoshi Sahara            if (!$rev1) {
81defc7576SSatoshi Sahara                $rev1 = $rev2;
82*edb50e6aSSatoshi Sahara                $rev2 = null;
83defc7576SSatoshi Sahara            }
84defc7576SSatoshi Sahara        } else {
85defc7576SSatoshi Sahara            $rev2 = $INPUT->int('rev2');
86defc7576SSatoshi Sahara        }
87*edb50e6aSSatoshi Sahara        return array($rev1, $rev2);
88*edb50e6aSSatoshi Sahara    }
89defc7576SSatoshi Sahara
90*edb50e6aSSatoshi Sahara    /**
91defc7576SSatoshi Sahara     * Determine left and right revision, its texts and the header
92*edb50e6aSSatoshi Sahara     *
93*edb50e6aSSatoshi Sahara     * @return array
94*edb50e6aSSatoshi Sahara     *       $l_rev,   $r_rev,    // int     left and right revisions
95*edb50e6aSSatoshi Sahara     *       $l_minor, $r_minor,  // string  class attributes
96*edb50e6aSSatoshi Sahara     *       $l_head,  $r_head,   // string  html snippet
97*edb50e6aSSatoshi Sahara     *       $l_text,  $r_text,   // string  raw wiki text
98*edb50e6aSSatoshi Sahara     *       $l_nav,   $r_nav,    // string  html snippet
99defc7576SSatoshi Sahara     */
100*edb50e6aSSatoshi Sahara    protected function getHtmlParts()
101*edb50e6aSSatoshi Sahara    {
102*edb50e6aSSatoshi Sahara        global $REV;
103*edb50e6aSSatoshi Sahara        global $lang;
104*edb50e6aSSatoshi Sahara
105*edb50e6aSSatoshi Sahara        // determine requested revision(s)
106*edb50e6aSSatoshi Sahara        list($rev1, $rev2) = $this->getRevisions();
107*edb50e6aSSatoshi Sahara        if ($rev2 === null) unset($rev2);
108*edb50e6aSSatoshi Sahara
109*edb50e6aSSatoshi Sahara        $pagelog = new PageChangeLog($this->id);
110defc7576SSatoshi Sahara
111defc7576SSatoshi Sahara        if ($this->text) { // compare text to the most current revision
112defc7576SSatoshi Sahara            $l_rev = '';
113*edb50e6aSSatoshi Sahara            $r_minor = '';
114*edb50e6aSSatoshi Sahara            $l_text = rawWiki($this->id, '');
115*edb50e6aSSatoshi Sahara            $l_head = '<a class="wikilink1" href="'. wl($this->id) .'">'
116*edb50e6aSSatoshi Sahara                . $this->id .' '. dformat((int) @filemtime(wikiFN($this->id))) .'</a> '
117defc7576SSatoshi Sahara                . $lang['current'];
118defc7576SSatoshi Sahara
119defc7576SSatoshi Sahara            $r_rev = '';
120*edb50e6aSSatoshi Sahara            $l_minor = '';
121defc7576SSatoshi Sahara            $r_text = cleanText($this->text);
122defc7576SSatoshi Sahara            $r_head = $lang['yours'];
123defc7576SSatoshi Sahara        } else {
124defc7576SSatoshi Sahara            if ($rev1 && isset($rev2) && $rev2) { // two specific revisions wanted
125defc7576SSatoshi Sahara                // make sure order is correct (older on the left)
126defc7576SSatoshi Sahara                if ($rev1 < $rev2) {
127defc7576SSatoshi Sahara                    $l_rev = $rev1;
128defc7576SSatoshi Sahara                    $r_rev = $rev2;
129defc7576SSatoshi Sahara                } else {
130defc7576SSatoshi Sahara                    $l_rev = $rev2;
131defc7576SSatoshi Sahara                    $r_rev = $rev1;
132defc7576SSatoshi Sahara                }
133defc7576SSatoshi Sahara            } elseif ($rev1) { // single revision given, compare to current
134defc7576SSatoshi Sahara                $r_rev = '';
135defc7576SSatoshi Sahara                $l_rev = $rev1;
136defc7576SSatoshi Sahara            } else { // no revision was given, compare previous to current
137defc7576SSatoshi Sahara                $r_rev = '';
138defc7576SSatoshi Sahara                $revs = $pagelog->getRevisions(0, 1);
139defc7576SSatoshi Sahara                $l_rev = $revs[0];
140defc7576SSatoshi Sahara                $REV = $l_rev; // store revision back in $REV
141defc7576SSatoshi Sahara            }
142defc7576SSatoshi Sahara
143defc7576SSatoshi Sahara            // when both revisions are empty then the page was created just now
144defc7576SSatoshi Sahara            if (!$l_rev && !$r_rev) {
145defc7576SSatoshi Sahara                $l_text = '';
146defc7576SSatoshi Sahara            } else {
147*edb50e6aSSatoshi Sahara                $l_text = rawWiki($this->id, $l_rev);
148defc7576SSatoshi Sahara            }
149*edb50e6aSSatoshi Sahara            $r_text = rawWiki($this->id, $r_rev);
150defc7576SSatoshi Sahara
15163ab9afeSSatoshi Sahara            // get header of diff HTML
152*edb50e6aSSatoshi Sahara            list($l_head, $r_head, $l_minor, $r_minor) = $this->diffHead($pagelog, $l_rev, $r_rev);
153defc7576SSatoshi Sahara        }
154defc7576SSatoshi Sahara
155*edb50e6aSSatoshi Sahara        // build navigation
156defc7576SSatoshi Sahara        $l_nav = '';
157defc7576SSatoshi Sahara        $r_nav = '';
158defc7576SSatoshi Sahara        if (!$this->text) {
159defc7576SSatoshi Sahara            list($l_nav, $r_nav) = $this->diffNavigation($pagelog, $l_rev, $r_rev);
160defc7576SSatoshi Sahara        }
161*edb50e6aSSatoshi Sahara
162*edb50e6aSSatoshi Sahara        return array(
163*edb50e6aSSatoshi Sahara            $l_rev,   $r_rev,
164*edb50e6aSSatoshi Sahara            $l_minor, $r_minor,
165*edb50e6aSSatoshi Sahara            $l_head,  $r_head,
166*edb50e6aSSatoshi Sahara            $l_text,  $r_text,
167*edb50e6aSSatoshi Sahara            $l_nav,   $r_nav,
168*edb50e6aSSatoshi Sahara        );
169*edb50e6aSSatoshi Sahara    }
170*edb50e6aSSatoshi Sahara
171*edb50e6aSSatoshi Sahara    /**
172*edb50e6aSSatoshi Sahara     * Show diff
173*edb50e6aSSatoshi Sahara     * between current page version and provided $text
174*edb50e6aSSatoshi Sahara     * or between the revisions provided via GET or POST
175*edb50e6aSSatoshi Sahara     *
176*edb50e6aSSatoshi Sahara     * @author Andreas Gohr <andi@splitbrain.org>
177*edb50e6aSSatoshi Sahara     *
178*edb50e6aSSatoshi Sahara     * @return void
179defc7576SSatoshi Sahara     */
180*edb50e6aSSatoshi Sahara    public function show($difftype = null)
181*edb50e6aSSatoshi Sahara    {
182*edb50e6aSSatoshi Sahara        global $lang;
183*edb50e6aSSatoshi Sahara        global $INFO;
184*edb50e6aSSatoshi Sahara
185*edb50e6aSSatoshi Sahara        $difftype = $this->getDiffType($difftype);
186*edb50e6aSSatoshi Sahara
187*edb50e6aSSatoshi Sahara        // determine left and right revision, its texts and the header
188*edb50e6aSSatoshi Sahara        list(
189*edb50e6aSSatoshi Sahara            $l_rev,   $r_rev,
190*edb50e6aSSatoshi Sahara            $l_minor, $r_minor,
191*edb50e6aSSatoshi Sahara            $l_head,  $r_head,
192*edb50e6aSSatoshi Sahara            $l_text,  $r_text,
193*edb50e6aSSatoshi Sahara            $l_nav,   $r_nav,
194*edb50e6aSSatoshi Sahara        ) = $this->getHtmlParts();
195*edb50e6aSSatoshi Sahara
196*edb50e6aSSatoshi Sahara        // Create diff object and the formatter
197defc7576SSatoshi Sahara        $diff = new \Diff(explode("\n", $l_text), explode("\n", $r_text));
198defc7576SSatoshi Sahara
199*edb50e6aSSatoshi Sahara        if ($difftype == 'inline') {
200defc7576SSatoshi Sahara            $diffformatter = new \InlineDiffFormatter();
201defc7576SSatoshi Sahara        } else {
202defc7576SSatoshi Sahara            $diffformatter = new \TableDiffFormatter();
203defc7576SSatoshi Sahara        }
204defc7576SSatoshi Sahara
205*edb50e6aSSatoshi Sahara        // Display intro
206*edb50e6aSSatoshi Sahara        if ($this->preference['showIntro']) print p_locale_xhtml('diff');
207*edb50e6aSSatoshi Sahara
208*edb50e6aSSatoshi Sahara        // Display type and exact reference
209defc7576SSatoshi Sahara        if (!$this->text) {
210defc7576SSatoshi Sahara            print '<div class="diffoptions group">';
211defc7576SSatoshi Sahara
212defc7576SSatoshi Sahara            // create the form to select difftype
213defc7576SSatoshi Sahara            $form = new Form(['action' => wl()]);
214*edb50e6aSSatoshi Sahara            $form->setHiddenField('id', $this->id);
215defc7576SSatoshi Sahara            $form->setHiddenField('rev2[0]', $l_rev);
216defc7576SSatoshi Sahara            $form->setHiddenField('rev2[1]', $r_rev);
217defc7576SSatoshi Sahara            $form->setHiddenField('do', 'diff');
218defc7576SSatoshi Sahara            $options = array(
219defc7576SSatoshi Sahara                         'sidebyside' => $lang['diff_side'],
220defc7576SSatoshi Sahara                         'inline' => $lang['diff_inline']
221defc7576SSatoshi Sahara            );
222defc7576SSatoshi Sahara            $input = $form->addDropdown('difftype', $options, $lang['diff_type'])
223*edb50e6aSSatoshi Sahara                ->val($difftype)->addClass('quickselect');
224defc7576SSatoshi Sahara            $input->useInput(false); // inhibit prefillInput() during toHTML() process
225defc7576SSatoshi Sahara            $form->addButton('do[diff]', 'Go')->attr('type','submit');
226defc7576SSatoshi Sahara            print $form->toHTML();
227defc7576SSatoshi Sahara
228defc7576SSatoshi Sahara            print '<p>';
229defc7576SSatoshi Sahara            // link to exactly this view FS#2835
230defc7576SSatoshi Sahara            print $this->diffViewlink('difflink', $l_rev, ($r_rev ?: $INFO['currentrev']));
231defc7576SSatoshi Sahara            print '</p>';
232defc7576SSatoshi Sahara
233defc7576SSatoshi Sahara            print '</div>'; // .diffoptions
234defc7576SSatoshi Sahara        }
235defc7576SSatoshi Sahara
236defc7576SSatoshi Sahara        /*
237defc7576SSatoshi Sahara         * Display diff view table
238defc7576SSatoshi Sahara         */
239defc7576SSatoshi Sahara        print '<div class="table">';
240*edb50e6aSSatoshi Sahara        print '<table class="diff diff_'. $difftype .'">';
241defc7576SSatoshi Sahara
242defc7576SSatoshi Sahara        //navigation and header
243*edb50e6aSSatoshi Sahara        if ($difftype == 'inline') {
244defc7576SSatoshi Sahara            if (!$this->text) {
245defc7576SSatoshi Sahara                print '<tr>'
246defc7576SSatoshi Sahara                    . '<td class="diff-lineheader">-</td>'
247defc7576SSatoshi Sahara                    . '<td class="diffnav">'. $l_nav .'</td>'
248defc7576SSatoshi Sahara                    . '</tr>';
249defc7576SSatoshi Sahara                print '<tr>'
250defc7576SSatoshi Sahara                    . '<th class="diff-lineheader">-</th>'
251defc7576SSatoshi Sahara                    . '<th '. $l_minor .'>'. $l_head .'</th>'
252defc7576SSatoshi Sahara                    .'</tr>';
253defc7576SSatoshi Sahara            }
254defc7576SSatoshi Sahara            print '<tr>'
255defc7576SSatoshi Sahara                . '<td class="diff-lineheader">+</td>'
256defc7576SSatoshi Sahara                . '<td class="diffnav">'. $r_nav .'</td>'
257defc7576SSatoshi Sahara                .'</tr>';
258defc7576SSatoshi Sahara            print '<tr>'
259defc7576SSatoshi Sahara                . '<th class="diff-lineheader">+</th>'
260defc7576SSatoshi Sahara                . '<th '. $r_minor .'>'. $r_head .'</th>'
261defc7576SSatoshi Sahara                . '</tr>';
262defc7576SSatoshi Sahara        } else {
263defc7576SSatoshi Sahara            if (!$this->text) {
264defc7576SSatoshi Sahara                print '<tr>'
265defc7576SSatoshi Sahara                    . '<td colspan="2" class="diffnav">'. $l_nav .'</td>'
266defc7576SSatoshi Sahara                    . '<td colspan="2" class="diffnav">'. $r_nav .'</td>'
267defc7576SSatoshi Sahara                    . '</tr>';
268defc7576SSatoshi Sahara            }
269defc7576SSatoshi Sahara            print '<tr>'
270defc7576SSatoshi Sahara                . '<th colspan="2" '. $l_minor .'>'. $l_head .'</th>'
271defc7576SSatoshi Sahara                . '<th colspan="2" '. $r_minor .'>'. $r_head .'</th>'
272defc7576SSatoshi Sahara                . '</tr>';
273defc7576SSatoshi Sahara        }
274defc7576SSatoshi Sahara
275defc7576SSatoshi Sahara        //diff view
276defc7576SSatoshi Sahara        print $this->insertSoftbreaks($diffformatter->format($diff));
277defc7576SSatoshi Sahara
278defc7576SSatoshi Sahara        print '</table>';
279defc7576SSatoshi Sahara        print '</div>';
280defc7576SSatoshi Sahara    }
281defc7576SSatoshi Sahara
282defc7576SSatoshi Sahara
283defc7576SSatoshi Sahara    /**
284defc7576SSatoshi Sahara     * Create html for revision navigation
285defc7576SSatoshi Sahara     *
286defc7576SSatoshi Sahara     * @param PageChangeLog $pagelog changelog object of current page
287defc7576SSatoshi Sahara     * @param int           $l_rev   left revision timestamp
288defc7576SSatoshi Sahara     * @param int           $r_rev   right revision timestamp
289defc7576SSatoshi Sahara     * @return string[] html of left and right navigation elements
290defc7576SSatoshi Sahara     */
291defc7576SSatoshi Sahara    protected function diffNavigation($pagelog, $l_rev, $r_rev)
292defc7576SSatoshi Sahara    {
293*edb50e6aSSatoshi Sahara        global $INFO;
294defc7576SSatoshi Sahara
295defc7576SSatoshi Sahara        // last timestamp is not in changelog, retrieve timestamp from metadata
296defc7576SSatoshi Sahara        // note: when page is removed, the metadata timestamp is zero
297defc7576SSatoshi Sahara        if (!$r_rev) {
298defc7576SSatoshi Sahara            if (isset($INFO['meta']['last_change']['date'])) {
299defc7576SSatoshi Sahara                $r_rev = $INFO['meta']['last_change']['date'];
300defc7576SSatoshi Sahara            } else {
301defc7576SSatoshi Sahara                $r_rev = 0;
302defc7576SSatoshi Sahara            }
303defc7576SSatoshi Sahara        }
304defc7576SSatoshi Sahara
305defc7576SSatoshi Sahara        //retrieve revisions with additional info
306defc7576SSatoshi Sahara        list($l_revs, $r_revs) = $pagelog->getRevisionsAround($l_rev, $r_rev);
307defc7576SSatoshi Sahara        $l_revisions = array();
308defc7576SSatoshi Sahara        if (!$l_rev) {
309defc7576SSatoshi Sahara            //no left revision given, add dummy
310defc7576SSatoshi Sahara            $l_revisions[0]= array('label' => '', 'attrs' => []);
311defc7576SSatoshi Sahara        }
312defc7576SSatoshi Sahara        foreach ($l_revs as $rev) {
313defc7576SSatoshi Sahara            $info = $pagelog->getRevisionInfo($rev);
314defc7576SSatoshi Sahara            $l_revisions[$rev] = array(
315defc7576SSatoshi Sahara                'label' => dformat($info['date']) .' '. editorinfo($info['user'], true) .' '. $info['sum'],
316defc7576SSatoshi Sahara                'attrs' => ['title' => $rev],
317defc7576SSatoshi Sahara            );
318defc7576SSatoshi Sahara            if ($r_rev ? $rev >= $r_rev : false) $l_revisions[$rev]['attrs']['disabled'] = 'disabled';
319defc7576SSatoshi Sahara        }
320defc7576SSatoshi Sahara        $r_revisions = array();
321defc7576SSatoshi Sahara        if (!$r_rev) {
322defc7576SSatoshi Sahara            //no right revision given, add dummy
323defc7576SSatoshi Sahara            $r_revisions[0] = array('label' => '', 'attrs' => []);
324defc7576SSatoshi Sahara        }
325defc7576SSatoshi Sahara        foreach ($r_revs as $rev) {
326defc7576SSatoshi Sahara            $info = $pagelog->getRevisionInfo($rev);
327defc7576SSatoshi Sahara            $r_revisions[$rev] = array(
328defc7576SSatoshi Sahara                'label' => dformat($info['date']) .' '. editorinfo($info['user'], true) .' '. $info['sum'],
329defc7576SSatoshi Sahara                'attrs' => ['title' => $rev],
330defc7576SSatoshi Sahara            );
331defc7576SSatoshi Sahara            if ($rev <= $l_rev) $r_revisions[$rev]['attrs']['disabled'] = 'disabled';
332defc7576SSatoshi Sahara        }
333defc7576SSatoshi Sahara
334defc7576SSatoshi Sahara        //determine previous/next revisions
335defc7576SSatoshi Sahara        $l_index = array_search($l_rev, $l_revs);
336defc7576SSatoshi Sahara        $l_prev = $l_revs[$l_index + 1];
337defc7576SSatoshi Sahara        $l_next = $l_revs[$l_index - 1];
338defc7576SSatoshi Sahara        if ($r_rev) {
339defc7576SSatoshi Sahara            $r_index = array_search($r_rev, $r_revs);
340defc7576SSatoshi Sahara            $r_prev = $r_revs[$r_index + 1];
341defc7576SSatoshi Sahara            $r_next = $r_revs[$r_index - 1];
342defc7576SSatoshi Sahara        } else {
343defc7576SSatoshi Sahara            //removed page
344defc7576SSatoshi Sahara            if ($l_next) {
345defc7576SSatoshi Sahara                $r_prev = $r_revs[0];
346defc7576SSatoshi Sahara            } else {
347defc7576SSatoshi Sahara                $r_prev = null;
348defc7576SSatoshi Sahara            }
349defc7576SSatoshi Sahara            $r_next = null;
350defc7576SSatoshi Sahara        }
351defc7576SSatoshi Sahara
352defc7576SSatoshi Sahara        /*
353defc7576SSatoshi Sahara         * Left side:
354defc7576SSatoshi Sahara         */
355defc7576SSatoshi Sahara        $l_nav = '';
356defc7576SSatoshi Sahara        //move back
357defc7576SSatoshi Sahara        if ($l_prev) {
358defc7576SSatoshi Sahara            $l_nav .= $this->diffViewlink('diffbothprevrev', $l_prev, $r_prev);
359defc7576SSatoshi Sahara            $l_nav .= $this->diffViewlink('diffprevrev', $l_prev, $r_rev);
360defc7576SSatoshi Sahara        }
361defc7576SSatoshi Sahara        //dropdown
362defc7576SSatoshi Sahara        $form = new Form(['action' => wl()]);
363*edb50e6aSSatoshi Sahara        $form->setHiddenField('id', $this->id);
364defc7576SSatoshi Sahara        $form->setHiddenField('difftype', $this->difftype);
365defc7576SSatoshi Sahara        $form->setHiddenField('rev2[1]', $r_rev);
366defc7576SSatoshi Sahara        $form->setHiddenField('do', 'diff');
367defc7576SSatoshi Sahara        $input = $form->addDropdown('rev2[0]', $l_revisions)->val($l_rev)->addClass('quickselect');
368defc7576SSatoshi Sahara        $input->useInput(false); // inhibit prefillInput() during toHTML() process
369defc7576SSatoshi Sahara        $form->addButton('do[diff]', 'Go')->attr('type','submit');
370defc7576SSatoshi Sahara        $l_nav .= $form->toHTML();
371defc7576SSatoshi Sahara        //move forward
372defc7576SSatoshi Sahara        if ($l_next && ($l_next < $r_rev || !$r_rev)) {
373defc7576SSatoshi Sahara            $l_nav .= $this->diffViewlink('diffnextrev', $l_next, $r_rev);
374defc7576SSatoshi Sahara        }
375defc7576SSatoshi Sahara
376defc7576SSatoshi Sahara        /*
377defc7576SSatoshi Sahara         * Right side:
378defc7576SSatoshi Sahara         */
379defc7576SSatoshi Sahara        $r_nav = '';
380defc7576SSatoshi Sahara        //move back
381defc7576SSatoshi Sahara        if ($l_rev < $r_prev) {
382defc7576SSatoshi Sahara            $r_nav .= $this->diffViewlink('diffprevrev', $l_rev, $r_prev);
383defc7576SSatoshi Sahara        }
384defc7576SSatoshi Sahara        //dropdown
385defc7576SSatoshi Sahara        $form = new Form(['action' => wl()]);
386*edb50e6aSSatoshi Sahara        $form->setHiddenField('id', $this->id);
387defc7576SSatoshi Sahara        $form->setHiddenField('rev2[0]', $l_rev);
388defc7576SSatoshi Sahara        $form->setHiddenField('difftype', $this->difftype);
389defc7576SSatoshi Sahara        $form->setHiddenField('do', 'diff');
390defc7576SSatoshi Sahara        $input = $form->addDropdown('rev2[1]', $r_revisions)->val($r_rev)->addClass('quickselect');
391defc7576SSatoshi Sahara        $input->useInput(false); // inhibit prefillInput() during toHTML() process
392defc7576SSatoshi Sahara        $form->addButton('do[diff]', 'Go')->attr('type','submit');
393defc7576SSatoshi Sahara        $r_nav .= $form->toHTML();
394defc7576SSatoshi Sahara        //move forward
395defc7576SSatoshi Sahara        if ($r_next) {
396defc7576SSatoshi Sahara            if ($pagelog->isCurrentRevision($r_next)) {
397defc7576SSatoshi Sahara                //last revision is diff with current page
398defc7576SSatoshi Sahara                $r_nav .= $this->diffViewlink('difflastrev', $l_rev);
399defc7576SSatoshi Sahara            } else {
400defc7576SSatoshi Sahara                $r_nav .= $this->diffViewlink('diffnextrev', $l_rev, $r_next);
401defc7576SSatoshi Sahara            }
402defc7576SSatoshi Sahara        } else {
403defc7576SSatoshi Sahara            $r_nav .= $this->diffViewlink('diffbothnextrev', $l_next, $r_next);
404defc7576SSatoshi Sahara        }
405defc7576SSatoshi Sahara        return array($l_nav, $r_nav);
406defc7576SSatoshi Sahara    }
407defc7576SSatoshi Sahara
408defc7576SSatoshi Sahara    /**
409defc7576SSatoshi Sahara     * Create html link to a diff view defined by two revisions
410defc7576SSatoshi Sahara     *
411defc7576SSatoshi Sahara     * @param string $linktype
412defc7576SSatoshi Sahara     * @param int $lrev oldest revision
413defc7576SSatoshi Sahara     * @param int $rrev newest revision or null for diff with current revision
414defc7576SSatoshi Sahara     * @return string html of link to a diff view
415defc7576SSatoshi Sahara     */
416defc7576SSatoshi Sahara    protected function diffViewlink($linktype, $lrev, $rrev = null)
417defc7576SSatoshi Sahara    {
418*edb50e6aSSatoshi Sahara        global $lang;
419defc7576SSatoshi Sahara        if ($rrev === null) {
420defc7576SSatoshi Sahara            $urlparam = array(
421defc7576SSatoshi Sahara                'do' => 'diff',
422defc7576SSatoshi Sahara                'rev' => $lrev,
423*edb50e6aSSatoshi Sahara                'difftype' => $this->preference['difftype'],
424defc7576SSatoshi Sahara            );
425defc7576SSatoshi Sahara        } else {
426defc7576SSatoshi Sahara            $urlparam = array(
427defc7576SSatoshi Sahara                'do' => 'diff',
428defc7576SSatoshi Sahara                'rev2[0]' => $lrev,
429defc7576SSatoshi Sahara                'rev2[1]' => $rrev,
430*edb50e6aSSatoshi Sahara                'difftype' => $this->preference['difftype'],
431defc7576SSatoshi Sahara            );
432defc7576SSatoshi Sahara        }
433*edb50e6aSSatoshi Sahara        return  '<a class="'. $linktype .'" href="'. wl($this->id, $urlparam) .'" title="'. $lang[$linktype] .'">'
434defc7576SSatoshi Sahara              . '<span>'. $lang[$linktype] .'</span>'
435defc7576SSatoshi Sahara              . '</a>';
436defc7576SSatoshi Sahara    }
437defc7576SSatoshi Sahara
438defc7576SSatoshi Sahara
439defc7576SSatoshi Sahara    /**
440defc7576SSatoshi Sahara     * Insert soft breaks in diff html
441defc7576SSatoshi Sahara     *
442defc7576SSatoshi Sahara     * @param string $diffhtml
443defc7576SSatoshi Sahara     * @return string
444defc7576SSatoshi Sahara     */
445defc7576SSatoshi Sahara    public function insertSoftbreaks($diffhtml)
446defc7576SSatoshi Sahara    {
447defc7576SSatoshi Sahara        // search the diff html string for both:
448defc7576SSatoshi Sahara        // - html tags, so these can be ignored
449defc7576SSatoshi Sahara        // - long strings of characters without breaking characters
450defc7576SSatoshi Sahara        return preg_replace_callback('/<[^>]*>|[^<> ]{12,}/', function ($match) {
451defc7576SSatoshi Sahara            // if match is an html tag, return it intact
452defc7576SSatoshi Sahara            if ($match[0][0] == '<') return $match[0];
453defc7576SSatoshi Sahara            // its a long string without a breaking character,
454defc7576SSatoshi Sahara            // make certain characters into breaking characters by inserting a
455defc7576SSatoshi Sahara            // word break opportunity (<wbr> tag) in front of them.
456defc7576SSatoshi Sahara            $regex = <<< REGEX
457defc7576SSatoshi Sahara(?(?=              # start a conditional expression with a positive look ahead ...
458defc7576SSatoshi Sahara&\#?\\w{1,6};)     # ... for html entities - we don't want to split them (ok to catch some invalid combinations)
459defc7576SSatoshi Sahara&\#?\\w{1,6};      # yes pattern - a quicker match for the html entity, since we know we have one
460defc7576SSatoshi Sahara|
461defc7576SSatoshi Sahara[?/,&\#;:]         # no pattern - any other group of 'special' characters to insert a breaking character after
462defc7576SSatoshi Sahara)+                 # end conditional expression
463defc7576SSatoshi SaharaREGEX;
464defc7576SSatoshi Sahara            return preg_replace('<'.$regex.'>xu', '\0<wbr>', $match[0]);
465defc7576SSatoshi Sahara        }, $diffhtml);
466defc7576SSatoshi Sahara    }
467defc7576SSatoshi Sahara
468defc7576SSatoshi Sahara}
469