xref: /dokuwiki/inc/Ui/PageDiff.php (revision 295564cdcae151b907a029a6535147f59c9e5ae1)
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{
15edb50e6aSSatoshi Sahara    /* @var string */
16edb50e6aSSatoshi Sahara    protected $text = '';
17defc7576SSatoshi Sahara
18defc7576SSatoshi Sahara    /**
19defc7576SSatoshi Sahara     * PageDiff Ui constructor
20defc7576SSatoshi Sahara     *
21edb50e6aSSatoshi Sahara     * @param string $id  page id
22defc7576SSatoshi Sahara     */
23e4c881bdSSatoshi Sahara    public function __construct($id = null)
24defc7576SSatoshi Sahara    {
25defc7576SSatoshi Sahara        global $INFO;
26edb50e6aSSatoshi Sahara        $this->id = isset($id) ? $id : $INFO['id'];
27edb50e6aSSatoshi Sahara
28*295564cdSSatoshi Sahara        // init preference
29edb50e6aSSatoshi Sahara        $this->preference['showIntro'] = true;
3091e70b5fSSatoshi Sahara        $this->preference['difftype'] = 'sidebyside'; // diff view type: inline or sidebyside
318068440fSSatoshi Sahara
328068440fSSatoshi Sahara        $this->setChangeLog();
338068440fSSatoshi Sahara    }
348068440fSSatoshi Sahara
358068440fSSatoshi Sahara    /** @inheritdoc */
368068440fSSatoshi Sahara    protected function setChangeLog()
378068440fSSatoshi Sahara    {
388068440fSSatoshi Sahara        $this->changelog = new PageChangeLog($this->id);
39defc7576SSatoshi Sahara    }
40defc7576SSatoshi Sahara
41e4c881bdSSatoshi Sahara    /**
42e4c881bdSSatoshi Sahara     * Set text to be compared with most current version
43e4c881bdSSatoshi Sahara     * exclusively use of the compare($old, $new) method
44e4c881bdSSatoshi Sahara     *
45e4c881bdSSatoshi Sahara     * @param string $text
46e4c881bdSSatoshi Sahara     * @return $this
47e4c881bdSSatoshi Sahara     */
48e4c881bdSSatoshi Sahara    public function compareWith($text = null)
49e4c881bdSSatoshi Sahara    {
50e4c881bdSSatoshi Sahara        if (isset($text)) {
51e4c881bdSSatoshi Sahara            $this->text = $text;
52e4c881bdSSatoshi Sahara            $this->old_rev = '';
53e4c881bdSSatoshi Sahara        }
54e4c881bdSSatoshi Sahara        return $this;
55e4c881bdSSatoshi Sahara    }
56e4c881bdSSatoshi Sahara
57b4b4c5c6SSatoshi Sahara    /** @inheritdoc */
58b4b4c5c6SSatoshi Sahara    protected function preProcess()
59b4b4c5c6SSatoshi Sahara    {
60b4b4c5c6SSatoshi Sahara        parent::preProcess();
61b4b4c5c6SSatoshi Sahara        if (!isset($this->old_rev, $this->new_rev)) {
62b4b4c5c6SSatoshi Sahara            // no revision was given, compare previous to current
638068440fSSatoshi Sahara            $revs = $this->changelog->getRevisions(0, 1);
64b4b4c5c6SSatoshi Sahara            $this->old_rev = $revs[0];
65b4b4c5c6SSatoshi Sahara            $this->new_rev = '';
66b4b4c5c6SSatoshi Sahara
67b4b4c5c6SSatoshi Sahara            global $REV;
68b4b4c5c6SSatoshi Sahara            $REV = $this->old_rev; // store revision back in $REV
69b4b4c5c6SSatoshi Sahara        }
70b4b4c5c6SSatoshi Sahara    }
71b4b4c5c6SSatoshi Sahara
72defc7576SSatoshi Sahara    /**
73edb50e6aSSatoshi Sahara     * Show diff
74edb50e6aSSatoshi Sahara     * between current page version and provided $text
75edb50e6aSSatoshi Sahara     * or between the revisions provided via GET or POST
76edb50e6aSSatoshi Sahara     *
77edb50e6aSSatoshi Sahara     * @author Andreas Gohr <andi@splitbrain.org>
78edb50e6aSSatoshi Sahara     *
79edb50e6aSSatoshi Sahara     * @return void
80defc7576SSatoshi Sahara     */
81309aaee5SSatoshi Sahara    public function show()
82edb50e6aSSatoshi Sahara    {
83675f74fbSSatoshi Sahara       // determine left and right revision
84b4b4c5c6SSatoshi Sahara        $this->preProcess();
85b4b4c5c6SSatoshi Sahara        [$l_rev, $r_rev] = [$this->old_rev, $this->new_rev];
86edb50e6aSSatoshi Sahara
87309aaee5SSatoshi Sahara       // build html diff view components
88edb50e6aSSatoshi Sahara        list(
89edb50e6aSSatoshi Sahara            $l_minor, $r_minor,
90edb50e6aSSatoshi Sahara            $l_head,  $r_head,
91edb50e6aSSatoshi Sahara            $l_text,  $r_text,
92edb50e6aSSatoshi Sahara            $l_nav,   $r_nav,
93ec019cbfSSatoshi Sahara        ) = $this->buildDiffViewComponents($l_rev, $r_rev);
94edb50e6aSSatoshi Sahara
95309aaee5SSatoshi Sahara        // create difference engine object
96309aaee5SSatoshi Sahara        $Difference = new \Diff(explode("\n", $l_text), explode("\n", $r_text));
97defc7576SSatoshi Sahara
98675f74fbSSatoshi Sahara        // display intro
99675f74fbSSatoshi Sahara        if ($this->preference['showIntro']) echo p_locale_xhtml('diff');
100defc7576SSatoshi Sahara
101675f74fbSSatoshi Sahara        // print form to choose diff view type, and exact url reference to the view
102defc7576SSatoshi Sahara        if (!$this->text) {
10391e70b5fSSatoshi Sahara            $this->showDiffViewSelector();
104defc7576SSatoshi Sahara        }
105defc7576SSatoshi Sahara
106309aaee5SSatoshi Sahara        // display diff view table
107*295564cdSSatoshi Sahara        echo '<div class="table">';
108*295564cdSSatoshi Sahara        echo '<table class="diff diff_'.$this->preference['difftype'] .'">';
109defc7576SSatoshi Sahara
110defc7576SSatoshi Sahara        //navigation and header
111309aaee5SSatoshi Sahara        switch ($this->preference['difftype']) {
112309aaee5SSatoshi Sahara            case 'inline':
113defc7576SSatoshi Sahara                if (!$this->text) {
114*295564cdSSatoshi Sahara                    echo '<tr>'
115defc7576SSatoshi Sahara                        .'<td class="diff-lineheader">-</td>'
116defc7576SSatoshi Sahara                        .'<td class="diffnav">'. $l_nav .'</td>'
117defc7576SSatoshi Sahara                        .'</tr>';
118*295564cdSSatoshi Sahara                    echo '<tr>'
119defc7576SSatoshi Sahara                        .'<th class="diff-lineheader">-</th>'
120defc7576SSatoshi Sahara                        .'<th '. $l_minor .'>'. $l_head .'</th>'
121defc7576SSatoshi Sahara                        .'</tr>';
122defc7576SSatoshi Sahara                }
123*295564cdSSatoshi Sahara                echo '<tr>'
124defc7576SSatoshi Sahara                    .'<td class="diff-lineheader">+</td>'
125defc7576SSatoshi Sahara                    .'<td class="diffnav">'. $r_nav .'</td>'
126defc7576SSatoshi Sahara                    .'</tr>';
127*295564cdSSatoshi Sahara                echo '<tr>'
128defc7576SSatoshi Sahara                    .'<th class="diff-lineheader">+</th>'
129defc7576SSatoshi Sahara                    .'<th '. $r_minor .'>'. $r_head .'</th>'
130defc7576SSatoshi Sahara                    .'</tr>';
131309aaee5SSatoshi Sahara                // create formatter object
132309aaee5SSatoshi Sahara                $DiffFormatter = new \InlineDiffFormatter();
133309aaee5SSatoshi Sahara                break;
134309aaee5SSatoshi Sahara
135309aaee5SSatoshi Sahara            case 'sidebyside':
136309aaee5SSatoshi Sahara            default:
137defc7576SSatoshi Sahara                if (!$this->text) {
138*295564cdSSatoshi Sahara                    echo '<tr>'
139defc7576SSatoshi Sahara                        .'<td colspan="2" class="diffnav">'. $l_nav .'</td>'
140defc7576SSatoshi Sahara                        .'<td colspan="2" class="diffnav">'. $r_nav .'</td>'
141defc7576SSatoshi Sahara                        .'</tr>';
142defc7576SSatoshi Sahara                }
143*295564cdSSatoshi Sahara                echo '<tr>'
144defc7576SSatoshi Sahara                    .'<th colspan="2" '. $l_minor .'>'. $l_head .'</th>'
145defc7576SSatoshi Sahara                    .'<th colspan="2" '. $r_minor .'>'. $r_head .'</th>'
146defc7576SSatoshi Sahara                    .'</tr>';
147309aaee5SSatoshi Sahara                // create formatter object
148675f74fbSSatoshi Sahara                $DiffFormatter = new \TableDiffFormatter();
149309aaee5SSatoshi Sahara                break;
150675f74fbSSatoshi Sahara        }
151675f74fbSSatoshi Sahara
152309aaee5SSatoshi Sahara        // output formatted difference
153*295564cdSSatoshi Sahara        echo $this->insertSoftbreaks($DiffFormatter->format($Difference));
154defc7576SSatoshi Sahara
155*295564cdSSatoshi Sahara        echo '</table>';
156*295564cdSSatoshi Sahara        echo '</div>';
157defc7576SSatoshi Sahara    }
158defc7576SSatoshi Sahara
159675f74fbSSatoshi Sahara    /**
16091e70b5fSSatoshi Sahara     * Print form to choose diff view type, and exact url reference to the view
16191e70b5fSSatoshi Sahara     */
16291e70b5fSSatoshi Sahara    protected function showDiffViewSelector()
16391e70b5fSSatoshi Sahara    {
16491e70b5fSSatoshi Sahara        global $INFO, $lang;
16591e70b5fSSatoshi Sahara
16691e70b5fSSatoshi Sahara        echo '<div class="diffoptions group">';
16791e70b5fSSatoshi Sahara
16891e70b5fSSatoshi Sahara        // create the form to select difftype
16991e70b5fSSatoshi Sahara        $form = new Form(['action' => wl()]);
17091e70b5fSSatoshi Sahara        $form->setHiddenField('id', $this->id);
17191e70b5fSSatoshi Sahara        $form->setHiddenField('rev2[0]', $this->old_rev ?: 'current');
17291e70b5fSSatoshi Sahara        $form->setHiddenField('rev2[1]', $this->new_rev ?: 'current');
17391e70b5fSSatoshi Sahara        $form->setHiddenField('do', 'diff');
17491e70b5fSSatoshi Sahara        $options = array(
17591e70b5fSSatoshi Sahara                     'sidebyside' => $lang['diff_side'],
17691e70b5fSSatoshi Sahara                     'inline' => $lang['diff_inline']
17791e70b5fSSatoshi Sahara        );
17891e70b5fSSatoshi Sahara        $input = $form->addDropdown('difftype', $options, $lang['diff_type'])
17991e70b5fSSatoshi Sahara            ->val($this->preference['difftype'])
18091e70b5fSSatoshi Sahara            ->addClass('quickselect');
18191e70b5fSSatoshi Sahara        $input->useInput(false); // inhibit prefillInput() during toHTML() process
18291e70b5fSSatoshi Sahara        $form->addButton('do[diff]', 'Go')->attr('type','submit');
18391e70b5fSSatoshi Sahara        echo $form->toHTML();
18491e70b5fSSatoshi Sahara
18591e70b5fSSatoshi Sahara        echo '<p>';
18691e70b5fSSatoshi Sahara        // link to exactly this view FS#2835
18791e70b5fSSatoshi Sahara        echo $this->diffViewlink('difflink', $l_rev, ($r_rev ?: $INFO['currentrev']));
18891e70b5fSSatoshi Sahara        echo '</p>';
18991e70b5fSSatoshi Sahara
19091e70b5fSSatoshi Sahara        echo '</div>'; // .diffoptions
19191e70b5fSSatoshi Sahara    }
19291e70b5fSSatoshi Sahara
19391e70b5fSSatoshi Sahara    /**
194309aaee5SSatoshi Sahara     * Build html diff view components
195675f74fbSSatoshi Sahara     *
196675f74fbSSatoshi Sahara     * @param int $l_rev  revision timestamp of left side
197675f74fbSSatoshi Sahara     * @param int $r_rev  revision timestamp of right side
198675f74fbSSatoshi Sahara     * @return array
199675f74fbSSatoshi Sahara     *       $l_minor, $r_minor,  // string  class attributes
200675f74fbSSatoshi Sahara     *       $l_head,  $r_head,   // string  html snippet
201675f74fbSSatoshi Sahara     *       $l_text,  $r_text,   // string  raw wiki text
202675f74fbSSatoshi Sahara     *       $l_nav,   $r_nav,    // string  html snippet
203675f74fbSSatoshi Sahara     */
204ec019cbfSSatoshi Sahara    protected function buildDiffViewComponents($l_rev, $r_rev)
205675f74fbSSatoshi Sahara    {
206675f74fbSSatoshi Sahara        global $lang;
207675f74fbSSatoshi Sahara
208675f74fbSSatoshi Sahara        if ($this->text) { // compare text to the most current revision
209675f74fbSSatoshi Sahara            $r_minor = '';
210675f74fbSSatoshi Sahara            $l_head = '<a class="wikilink1" href="'. wl($this->id) .'">'
211675f74fbSSatoshi Sahara                . $this->id .' '. dformat((int) @filemtime(wikiFN($this->id))) .'</a> '
212675f74fbSSatoshi Sahara                . $lang['current'];
21391e70b5fSSatoshi Sahara            $l_text = rawWiki($this->id, '');
214675f74fbSSatoshi Sahara
215675f74fbSSatoshi Sahara            $l_minor = '';
216675f74fbSSatoshi Sahara            $r_head = $lang['yours'];
21791e70b5fSSatoshi Sahara            $r_text = cleanText($this->text);
218675f74fbSSatoshi Sahara
219675f74fbSSatoshi Sahara        } else {
220675f74fbSSatoshi Sahara            // when both revisions are empty then the page was created just now
221675f74fbSSatoshi Sahara            if (!$l_rev && !$r_rev) {
222675f74fbSSatoshi Sahara                $l_text = '';
223675f74fbSSatoshi Sahara            } else {
224675f74fbSSatoshi Sahara                $l_text = rawWiki($this->id, $l_rev);
225675f74fbSSatoshi Sahara            }
226675f74fbSSatoshi Sahara            $r_text = rawWiki($this->id, $r_rev);
227675f74fbSSatoshi Sahara
228675f74fbSSatoshi Sahara            // get header of diff HTML
22991e70b5fSSatoshi Sahara            list(
23091e70b5fSSatoshi Sahara                $l_head,  $r_head,
23191e70b5fSSatoshi Sahara                $l_minor, $r_minor,
23291e70b5fSSatoshi Sahara            ) = $this->buildDiffHead($l_rev, $r_rev);
233675f74fbSSatoshi Sahara        }
234675f74fbSSatoshi Sahara        // build navigation
235675f74fbSSatoshi Sahara        $l_nav = '';
236675f74fbSSatoshi Sahara        $r_nav = '';
237675f74fbSSatoshi Sahara        if (!$this->text) {
2388068440fSSatoshi Sahara            list($l_nav, $r_nav) = $this->buildDiffNavigation($l_rev, $r_rev);
239675f74fbSSatoshi Sahara        }
240675f74fbSSatoshi Sahara
241675f74fbSSatoshi Sahara        return array(
242675f74fbSSatoshi Sahara            $l_minor, $r_minor,
243675f74fbSSatoshi Sahara            $l_head,  $r_head,
244675f74fbSSatoshi Sahara            $l_text,  $r_text,
245675f74fbSSatoshi Sahara            $l_nav,   $r_nav,
246675f74fbSSatoshi Sahara        );
247675f74fbSSatoshi Sahara    }
248675f74fbSSatoshi Sahara
249675f74fbSSatoshi Sahara    /**
250defc7576SSatoshi Sahara     * Create html for revision navigation
251defc7576SSatoshi Sahara     *
252defc7576SSatoshi Sahara     * @param int           $l_rev   left revision timestamp
253defc7576SSatoshi Sahara     * @param int           $r_rev   right revision timestamp
254defc7576SSatoshi Sahara     * @return string[] html of left and right navigation elements
255defc7576SSatoshi Sahara     */
2568068440fSSatoshi Sahara    protected function buildDiffNavigation($l_rev, $r_rev)
257defc7576SSatoshi Sahara    {
258edb50e6aSSatoshi Sahara        global $INFO;
259defc7576SSatoshi Sahara
260defc7576SSatoshi Sahara        // last timestamp is not in changelog, retrieve timestamp from metadata
261defc7576SSatoshi Sahara        // note: when page is removed, the metadata timestamp is zero
262defc7576SSatoshi Sahara        if (!$r_rev) {
263defc7576SSatoshi Sahara            if (isset($INFO['meta']['last_change']['date'])) {
264defc7576SSatoshi Sahara                $r_rev = $INFO['meta']['last_change']['date'];
265defc7576SSatoshi Sahara            } else {
266defc7576SSatoshi Sahara                $r_rev = 0;
267defc7576SSatoshi Sahara            }
268defc7576SSatoshi Sahara        }
269defc7576SSatoshi Sahara
270defc7576SSatoshi Sahara        //retrieve revisions with additional info
2718068440fSSatoshi Sahara        list($l_revs, $r_revs) = $this->changelog->getRevisionsAround($l_rev, $r_rev);
272*295564cdSSatoshi Sahara
273defc7576SSatoshi Sahara        $l_revisions = array();
274defc7576SSatoshi Sahara        if (!$l_rev) {
275defc7576SSatoshi Sahara            //no left revision given, add dummy
276defc7576SSatoshi Sahara            $l_revisions[0]= array('label' => '', 'attrs' => []);
277defc7576SSatoshi Sahara        }
278defc7576SSatoshi Sahara        foreach ($l_revs as $rev) {
2798068440fSSatoshi Sahara            $info = $this->changelog->getRevisionInfo($rev);
280defc7576SSatoshi Sahara            $l_revisions[$rev] = array(
281*295564cdSSatoshi Sahara                'label' => implode(' ', array(
282*295564cdSSatoshi Sahara                            dformat($info['date']),
283*295564cdSSatoshi Sahara                            editorinfo($info['user'], true),
284*295564cdSSatoshi Sahara                            $info['sum'],
285*295564cdSSatoshi Sahara                           )),
286defc7576SSatoshi Sahara                'attrs' => ['title' => $rev],
287defc7576SSatoshi Sahara            );
288*295564cdSSatoshi Sahara            if ($r_rev ? $rev >= $r_rev : false)
289*295564cdSSatoshi Sahara                $l_revisions[$rev]['attrs']['disabled'] = 'disabled';
290defc7576SSatoshi Sahara        }
291*295564cdSSatoshi Sahara
292defc7576SSatoshi Sahara        $r_revisions = array();
293defc7576SSatoshi Sahara        if (!$r_rev) {
294defc7576SSatoshi Sahara            //no right revision given, add dummy
295defc7576SSatoshi Sahara            $r_revisions[0] = array('label' => '', 'attrs' => []);
296defc7576SSatoshi Sahara        }
297defc7576SSatoshi Sahara        foreach ($r_revs as $rev) {
2988068440fSSatoshi Sahara            $info = $this->changelog->getRevisionInfo($rev);
299defc7576SSatoshi Sahara            $r_revisions[$rev] = array(
300*295564cdSSatoshi Sahara                'label' => implode(' ', array(
301*295564cdSSatoshi Sahara                            dformat($info['date']),
302*295564cdSSatoshi Sahara                            editorinfo($info['user'], true),
303*295564cdSSatoshi Sahara                            $info['sum'],
304*295564cdSSatoshi Sahara                           )),
305defc7576SSatoshi Sahara                'attrs' => ['title' => $rev],
306defc7576SSatoshi Sahara            );
307*295564cdSSatoshi Sahara            if ($rev <= $l_rev)
308*295564cdSSatoshi Sahara                $r_revisions[$rev]['attrs']['disabled'] = 'disabled';
309defc7576SSatoshi Sahara        }
310defc7576SSatoshi Sahara
311defc7576SSatoshi Sahara        //determine previous/next revisions
312defc7576SSatoshi Sahara        $l_index = array_search($l_rev, $l_revs);
313defc7576SSatoshi Sahara        $l_prev = $l_revs[$l_index + 1];
314defc7576SSatoshi Sahara        $l_next = $l_revs[$l_index - 1];
315defc7576SSatoshi Sahara        if ($r_rev) {
316defc7576SSatoshi Sahara            $r_index = array_search($r_rev, $r_revs);
317defc7576SSatoshi Sahara            $r_prev = $r_revs[$r_index + 1];
318defc7576SSatoshi Sahara            $r_next = $r_revs[$r_index - 1];
319defc7576SSatoshi Sahara        } else {
320defc7576SSatoshi Sahara            //removed page
321*295564cdSSatoshi Sahara            $r_prev = ($l_next) ? $r_revs[0] : null;
322defc7576SSatoshi Sahara            $r_next = null;
323defc7576SSatoshi Sahara        }
324defc7576SSatoshi Sahara
325defc7576SSatoshi Sahara        /*
326defc7576SSatoshi Sahara         * Left side:
327defc7576SSatoshi Sahara         */
328defc7576SSatoshi Sahara        $l_nav = '';
329defc7576SSatoshi Sahara        //move back
330defc7576SSatoshi Sahara        if ($l_prev) {
331defc7576SSatoshi Sahara            $l_nav .= $this->diffViewlink('diffbothprevrev', $l_prev, $r_prev);
332defc7576SSatoshi Sahara            $l_nav .= $this->diffViewlink('diffprevrev', $l_prev, $r_rev);
333defc7576SSatoshi Sahara        }
334defc7576SSatoshi Sahara        //dropdown
335defc7576SSatoshi Sahara        $form = new Form(['action' => wl()]);
336edb50e6aSSatoshi Sahara        $form->setHiddenField('id', $this->id);
337309aaee5SSatoshi Sahara        $form->setHiddenField('difftype', $this->preference['difftype']);
33858c5cb81SSatoshi Sahara        $form->setHiddenField('rev2[1]', $r_rev ?: 'current');
339defc7576SSatoshi Sahara        $form->setHiddenField('do', 'diff');
340*295564cdSSatoshi Sahara        $input = $form->addDropdown('rev2[0]', $l_revisions)
341*295564cdSSatoshi Sahara            ->val($l_rev ?: 'current')->addClass('quickselect');
342defc7576SSatoshi Sahara        $input->useInput(false); // inhibit prefillInput() during toHTML() process
343defc7576SSatoshi Sahara        $form->addButton('do[diff]', 'Go')->attr('type','submit');
344defc7576SSatoshi Sahara        $l_nav .= $form->toHTML();
345defc7576SSatoshi Sahara        //move forward
346defc7576SSatoshi Sahara        if ($l_next && ($l_next < $r_rev || !$r_rev)) {
347defc7576SSatoshi Sahara            $l_nav .= $this->diffViewlink('diffnextrev', $l_next, $r_rev);
348defc7576SSatoshi Sahara        }
349defc7576SSatoshi Sahara
350defc7576SSatoshi Sahara        /*
351defc7576SSatoshi Sahara         * Right side:
352defc7576SSatoshi Sahara         */
353defc7576SSatoshi Sahara        $r_nav = '';
354defc7576SSatoshi Sahara        //move back
355defc7576SSatoshi Sahara        if ($l_rev < $r_prev) {
356defc7576SSatoshi Sahara            $r_nav .= $this->diffViewlink('diffprevrev', $l_rev, $r_prev);
357defc7576SSatoshi Sahara        }
358defc7576SSatoshi Sahara        //dropdown
359defc7576SSatoshi Sahara        $form = new Form(['action' => wl()]);
360edb50e6aSSatoshi Sahara        $form->setHiddenField('id', $this->id);
36158c5cb81SSatoshi Sahara        $form->setHiddenField('rev2[0]', $l_rev ?: 'current');
362309aaee5SSatoshi Sahara        $form->setHiddenField('difftype', $this->preference['difftype']);
363defc7576SSatoshi Sahara        $form->setHiddenField('do', 'diff');
364*295564cdSSatoshi Sahara        $input = $form->addDropdown('rev2[1]', $r_revisions)
365*295564cdSSatoshi Sahara            ->val($r_rev ?: 'current')->addClass('quickselect');
366defc7576SSatoshi Sahara        $input->useInput(false); // inhibit prefillInput() during toHTML() process
367defc7576SSatoshi Sahara        $form->addButton('do[diff]', 'Go')->attr('type','submit');
368defc7576SSatoshi Sahara        $r_nav .= $form->toHTML();
369defc7576SSatoshi Sahara        //move forward
370defc7576SSatoshi Sahara        if ($r_next) {
3718068440fSSatoshi Sahara            if ($this->changelog->isCurrentRevision($r_next)) {
372defc7576SSatoshi Sahara                //last revision is diff with current page
373defc7576SSatoshi Sahara                $r_nav .= $this->diffViewlink('difflastrev', $l_rev);
374defc7576SSatoshi Sahara            } else {
375defc7576SSatoshi Sahara                $r_nav .= $this->diffViewlink('diffnextrev', $l_rev, $r_next);
376defc7576SSatoshi Sahara            }
377defc7576SSatoshi Sahara            $r_nav .= $this->diffViewlink('diffbothnextrev', $l_next, $r_next);
378defc7576SSatoshi Sahara        }
379defc7576SSatoshi Sahara        return array($l_nav, $r_nav);
380defc7576SSatoshi Sahara    }
381defc7576SSatoshi Sahara
382defc7576SSatoshi Sahara    /**
383defc7576SSatoshi Sahara     * Create html link to a diff view defined by two revisions
384defc7576SSatoshi Sahara     *
385defc7576SSatoshi Sahara     * @param string $linktype
386defc7576SSatoshi Sahara     * @param int $lrev oldest revision
387defc7576SSatoshi Sahara     * @param int $rrev newest revision or null for diff with current revision
388defc7576SSatoshi Sahara     * @return string html of link to a diff view
389defc7576SSatoshi Sahara     */
390defc7576SSatoshi Sahara    protected function diffViewlink($linktype, $lrev, $rrev = null)
391defc7576SSatoshi Sahara    {
392edb50e6aSSatoshi Sahara        global $lang;
393defc7576SSatoshi Sahara        if ($rrev === null) {
394defc7576SSatoshi Sahara            $urlparam = array(
395defc7576SSatoshi Sahara                'do' => 'diff',
396defc7576SSatoshi Sahara                'rev' => $lrev,
397edb50e6aSSatoshi Sahara                'difftype' => $this->preference['difftype'],
398defc7576SSatoshi Sahara            );
399defc7576SSatoshi Sahara        } else {
400defc7576SSatoshi Sahara            $urlparam = array(
401defc7576SSatoshi Sahara                'do' => 'diff',
402defc7576SSatoshi Sahara                'rev2[0]' => $lrev,
403defc7576SSatoshi Sahara                'rev2[1]' => $rrev,
404edb50e6aSSatoshi Sahara                'difftype' => $this->preference['difftype'],
405defc7576SSatoshi Sahara            );
406defc7576SSatoshi Sahara        }
407*295564cdSSatoshi Sahara        $attr = array(
408*295564cdSSatoshi Sahara            'class' => $linktype,
409*295564cdSSatoshi Sahara            'href'  => wl($this->id, $urlparam),
410*295564cdSSatoshi Sahara            'title' => $lang[$linktype],
411*295564cdSSatoshi Sahara        );
412*295564cdSSatoshi Sahara        return '<a '. buildAttributes($attr) .'><span>'. $lang[$linktype] .'</span></a>';
413defc7576SSatoshi Sahara    }
414defc7576SSatoshi Sahara
415defc7576SSatoshi Sahara
416defc7576SSatoshi Sahara    /**
417defc7576SSatoshi Sahara     * Insert soft breaks in diff html
418defc7576SSatoshi Sahara     *
419defc7576SSatoshi Sahara     * @param string $diffhtml
420defc7576SSatoshi Sahara     * @return string
421defc7576SSatoshi Sahara     */
422defc7576SSatoshi Sahara    public function insertSoftbreaks($diffhtml)
423defc7576SSatoshi Sahara    {
424defc7576SSatoshi Sahara        // search the diff html string for both:
425defc7576SSatoshi Sahara        // - html tags, so these can be ignored
426defc7576SSatoshi Sahara        // - long strings of characters without breaking characters
427defc7576SSatoshi Sahara        return preg_replace_callback('/<[^>]*>|[^<> ]{12,}/', function ($match) {
428defc7576SSatoshi Sahara            // if match is an html tag, return it intact
429defc7576SSatoshi Sahara            if ($match[0][0] == '<') return $match[0];
430defc7576SSatoshi Sahara            // its a long string without a breaking character,
431defc7576SSatoshi Sahara            // make certain characters into breaking characters by inserting a
432defc7576SSatoshi Sahara            // word break opportunity (<wbr> tag) in front of them.
433defc7576SSatoshi Sahara            $regex = <<< REGEX
434defc7576SSatoshi Sahara(?(?=              # start a conditional expression with a positive look ahead ...
435defc7576SSatoshi Sahara&\#?\\w{1,6};)     # ... for html entities - we don't want to split them (ok to catch some invalid combinations)
436defc7576SSatoshi Sahara&\#?\\w{1,6};      # yes pattern - a quicker match for the html entity, since we know we have one
437defc7576SSatoshi Sahara|
438defc7576SSatoshi Sahara[?/,&\#;:]         # no pattern - any other group of 'special' characters to insert a breaking character after
439defc7576SSatoshi Sahara)+                 # end conditional expression
440defc7576SSatoshi SaharaREGEX;
441defc7576SSatoshi Sahara            return preg_replace('<'.$regex.'>xu', '\0<wbr>', $match[0]);
442defc7576SSatoshi Sahara        }, $diffhtml);
443defc7576SSatoshi Sahara    }
444defc7576SSatoshi Sahara
445defc7576SSatoshi Sahara}
446