xref: /dokuwiki/inc/Ui/Revisions.php (revision e71e09a62e4986d588e0afb8128f2eb7cfa41a12)
16c9fde82SSatoshi Sahara<?php
26c9fde82SSatoshi Sahara
36c9fde82SSatoshi Saharanamespace dokuwiki\Ui;
46c9fde82SSatoshi Sahara
56c9fde82SSatoshi Sahara/**
69e316641SSatoshi Sahara * DokuWiki Revisions Interface
7ae5d2354SSatoshi Sahara * parent class of PageRevisions and MediaRevisions
86c9fde82SSatoshi Sahara *
96c9fde82SSatoshi Sahara * @package dokuwiki\Ui
106c9fde82SSatoshi Sahara */
11ae5d2354SSatoshi Saharaabstract class Revisions extends Ui
126c9fde82SSatoshi Sahara{
13ae5d2354SSatoshi Sahara    /* @var string */
14*e71e09a6SSatoshi Sahara    protected $id;   // page id or media id
15*e71e09a6SSatoshi Sahara    protected $item; // page or media
16*e71e09a6SSatoshi Sahara
17*e71e09a6SSatoshi Sahara    /* @var ChangeLog */
18*e71e09a6SSatoshi Sahara    protected $changelog; // PageChangeLog or MediaChangeLog object
199e316641SSatoshi Sahara
209e316641SSatoshi Sahara    /**
219e316641SSatoshi Sahara     * Revisions Ui constructor
229e316641SSatoshi Sahara     *
23*e71e09a6SSatoshi Sahara     * @param string $id  page id or media id
249e316641SSatoshi Sahara     */
25ae5d2354SSatoshi Sahara    public function __construct($id)
269e316641SSatoshi Sahara    {
27ae5d2354SSatoshi Sahara        $this->id = $id;
28*e71e09a6SSatoshi Sahara        $this->setChangeLog();
29*e71e09a6SSatoshi Sahara    }
30*e71e09a6SSatoshi Sahara
31*e71e09a6SSatoshi Sahara    /**
32*e71e09a6SSatoshi Sahara     * set class property changelog
33*e71e09a6SSatoshi Sahara     */
34*e71e09a6SSatoshi Sahara    abstract protected function setChangeLog();
35*e71e09a6SSatoshi Sahara
36*e71e09a6SSatoshi Sahara    /**
37*e71e09a6SSatoshi Sahara     * item source file resolver
38*e71e09a6SSatoshi Sahara     *
39*e71e09a6SSatoshi Sahara     * @param string $id  page id or media id
40*e71e09a6SSatoshi Sahara     * @return string full path
41*e71e09a6SSatoshi Sahara     */
42*e71e09a6SSatoshi Sahara    protected function itemFN($id)
43*e71e09a6SSatoshi Sahara    {
44*e71e09a6SSatoshi Sahara        switch ($this->item) {
45*e71e09a6SSatoshi Sahara            case 'page':  return wikiFN($id);
46*e71e09a6SSatoshi Sahara            case 'media': return mediaFN($id);
47*e71e09a6SSatoshi Sahara        }
489e316641SSatoshi Sahara    }
499e316641SSatoshi Sahara
506c9fde82SSatoshi Sahara    /**
5167ef3e88SSatoshi Sahara     * Get revisions, and set correct pagenation parameters (first, hasNext)
5267ef3e88SSatoshi Sahara     *
5367ef3e88SSatoshi Sahara     * @param int  $first
5467ef3e88SSatoshi Sahara     * @param bool $hasNext
5567ef3e88SSatoshi Sahara     * @return array  revisions to be shown in a pagenated list
565525369fSSatoshi Sahara     * @see also https://www.dokuwiki.org/devel:changelog
5767ef3e88SSatoshi Sahara     */
58*e71e09a6SSatoshi Sahara    protected function getRevisions(&$first, &$hasNext)
59*e71e09a6SSatoshi Sahara    {
60*e71e09a6SSatoshi Sahara        global $conf;
61*e71e09a6SSatoshi Sahara
62*e71e09a6SSatoshi Sahara        $changelog =& $this->changelog;
63*e71e09a6SSatoshi Sahara
64*e71e09a6SSatoshi Sahara        $revisions = array();
65*e71e09a6SSatoshi Sahara
66*e71e09a6SSatoshi Sahara        /* we need to get one additional log entry to be able to
67*e71e09a6SSatoshi Sahara         * decide if this is the last page or is there another one.
68*e71e09a6SSatoshi Sahara         * see also Ui\Recent::getRecents()
69*e71e09a6SSatoshi Sahara         */
70*e71e09a6SSatoshi Sahara        $revlist = $changelog->getRevisions($first, $conf['recent'] +1);
71*e71e09a6SSatoshi Sahara        if (count($revlist) == 0 && $first != 0) {
72*e71e09a6SSatoshi Sahara            $first = 0;
73*e71e09a6SSatoshi Sahara            $revlist = $changelog->getRevisions($first, $conf['recent'] +1);
74*e71e09a6SSatoshi Sahara        }
75*e71e09a6SSatoshi Sahara
76*e71e09a6SSatoshi Sahara        // add current page or media as revision[0] when necessary
77*e71e09a6SSatoshi Sahara        if ($first === 0 && file_exists($this->itemFN($this->id))) {
78*e71e09a6SSatoshi Sahara            $rev = filemtime(fullpath($this->itemFN($this->id)));
79*e71e09a6SSatoshi Sahara            $changelog->setChunkSize(1024); //FIXME why does chunksize change wanted?
80*e71e09a6SSatoshi Sahara            $revinfo = $changelog->getRevisionInfo($rev) ?: array(
81*e71e09a6SSatoshi Sahara                    'date' => $rev,
82*e71e09a6SSatoshi Sahara                    'ip'   => null,
83*e71e09a6SSatoshi Sahara                    'type' => null,
84*e71e09a6SSatoshi Sahara                    'id'   => $this->id,
85*e71e09a6SSatoshi Sahara                    'user' => null,
86*e71e09a6SSatoshi Sahara                    'sum'  => null,
87*e71e09a6SSatoshi Sahara                    'extra' => null,
88*e71e09a6SSatoshi Sahara                    'sizechange' => null,
89*e71e09a6SSatoshi Sahara            );
90*e71e09a6SSatoshi Sahara            $revisions[] = $revinfo + array(
91*e71e09a6SSatoshi Sahara                    'item' => $this->item,
92*e71e09a6SSatoshi Sahara                    'current' => true,
93*e71e09a6SSatoshi Sahara            );
94*e71e09a6SSatoshi Sahara        }
95*e71e09a6SSatoshi Sahara
96*e71e09a6SSatoshi Sahara        // decide if this is the last page or is there another one
97*e71e09a6SSatoshi Sahara        $hasNext = false;
98*e71e09a6SSatoshi Sahara        if (count($revlist) > $conf['recent']) {
99*e71e09a6SSatoshi Sahara            $hasNext = true;
100*e71e09a6SSatoshi Sahara            array_pop($revlist); // remove one additional log entry
101*e71e09a6SSatoshi Sahara        }
102*e71e09a6SSatoshi Sahara
103*e71e09a6SSatoshi Sahara        // append each revison info array to the revisions
104*e71e09a6SSatoshi Sahara        foreach ($revlist as $rev) {
105*e71e09a6SSatoshi Sahara            $revisions[] = $changelog->getRevisionInfo($rev) + array('item' => $this->item);
106*e71e09a6SSatoshi Sahara        }
107*e71e09a6SSatoshi Sahara        return $revisions;
108*e71e09a6SSatoshi Sahara    }
10967ef3e88SSatoshi Sahara
11067ef3e88SSatoshi Sahara    /**
11167ef3e88SSatoshi Sahara     * Navigation buttons for Pagenation (prev/next)
11267ef3e88SSatoshi Sahara     *
11367ef3e88SSatoshi Sahara     * @param int  $first
11467ef3e88SSatoshi Sahara     * @param bool $hasNext
115ae5d2354SSatoshi Sahara     * @param callable $callback returns array of hidden fields for the form button
11667ef3e88SSatoshi Sahara     * @return array  html
11767ef3e88SSatoshi Sahara     */
118ae5d2354SSatoshi Sahara    protected function navigation($first, $hasNext, $callback)
11967ef3e88SSatoshi Sahara    {
12067ef3e88SSatoshi Sahara        global $conf;
12167ef3e88SSatoshi Sahara
12267ef3e88SSatoshi Sahara        $html = '<div class="pagenav">';
1236c9fde82SSatoshi Sahara        $last = $first + $conf['recent'];
1246c9fde82SSatoshi Sahara        if ($first > 0) {
12567ef3e88SSatoshi Sahara            $first = max($first - $conf['recent'], 0);
12667ef3e88SSatoshi Sahara            $html.= '<div class="pagenav-prev">';
127ae5d2354SSatoshi Sahara            $html.= html_btn('newer', $this->id, "p", $callback($first));
12867ef3e88SSatoshi Sahara            $html.= '</div>';
1296c9fde82SSatoshi Sahara        }
1306c9fde82SSatoshi Sahara        if ($hasNext) {
13167ef3e88SSatoshi Sahara            $html.= '<div class="pagenav-next">';
132ae5d2354SSatoshi Sahara            $html.= html_btn('older', $this->id, "n", $callback($last));
13367ef3e88SSatoshi Sahara            $html.= '</div>';
1346c9fde82SSatoshi Sahara        }
13567ef3e88SSatoshi Sahara        $html.= '</div>';
13667ef3e88SSatoshi Sahara        return $html;
13767ef3e88SSatoshi Sahara    }
1386c9fde82SSatoshi Sahara
13967ef3e88SSatoshi Sahara    /**
14067ef3e88SSatoshi Sahara     * Returns instance of objRevInfo
14167ef3e88SSatoshi Sahara     *
14267ef3e88SSatoshi Sahara     * @param array $info  Revision info structure of a page or media file
14367ef3e88SSatoshi Sahara     * @return objRevInfo object (anonymous class)
14467ef3e88SSatoshi Sahara     */
1450bb448f0SSatoshi Sahara    public function getObjRevInfo(array $info)
14667ef3e88SSatoshi Sahara    {
14767ef3e88SSatoshi Sahara        return new class ($info) // anonymous class (objRevInfo)
14867ef3e88SSatoshi Sahara        {
14967ef3e88SSatoshi Sahara            protected $info;
15067ef3e88SSatoshi Sahara
15167ef3e88SSatoshi Sahara            public function __construct(array $info)
15267ef3e88SSatoshi Sahara            {
15367ef3e88SSatoshi Sahara                $this->info = $info;
15467ef3e88SSatoshi Sahara            }
15567ef3e88SSatoshi Sahara
15667ef3e88SSatoshi Sahara            // current indicator
15767ef3e88SSatoshi Sahara            public function currentIndicator()
15867ef3e88SSatoshi Sahara            {
15967ef3e88SSatoshi Sahara                global $lang;
16067ef3e88SSatoshi Sahara                return ($this->info['current']) ? '('.$lang['current'].')' : '';
16167ef3e88SSatoshi Sahara            }
16267ef3e88SSatoshi Sahara
16367ef3e88SSatoshi Sahara            // edit date and time of the page or media file
16467ef3e88SSatoshi Sahara            public function editDate()
16567ef3e88SSatoshi Sahara            {
16667ef3e88SSatoshi Sahara                return '<span class="date">'. dformat($this->info['date']) .'</span>';
16767ef3e88SSatoshi Sahara            }
16867ef3e88SSatoshi Sahara
16967ef3e88SSatoshi Sahara            // edit summary
17067ef3e88SSatoshi Sahara            public function editSummary()
17167ef3e88SSatoshi Sahara            {
17267ef3e88SSatoshi Sahara                return '<span class="sum">'.' – '. hsc($this->info['sum']).'</span>';
17367ef3e88SSatoshi Sahara            }
17467ef3e88SSatoshi Sahara
17567ef3e88SSatoshi Sahara            // editor of the page or media file
17667ef3e88SSatoshi Sahara            public function editor()
17767ef3e88SSatoshi Sahara            {
17867ef3e88SSatoshi Sahara                // slightly different with display of Ui\Recent, i.e. external edit
17967ef3e88SSatoshi Sahara                global $lang;
18067ef3e88SSatoshi Sahara                $html = '<span class="user">';
18167ef3e88SSatoshi Sahara                if (!$this->info['user'] && !$this->info['ip']) {
18267ef3e88SSatoshi Sahara                    $html.= '('.$lang['external_edit'].')';
18367ef3e88SSatoshi Sahara                } elseif ($this->info['user']) {
18467ef3e88SSatoshi Sahara                    $html.= '<bdi>'. editorinfo($this->info['user']) .'</bdi>';
18567ef3e88SSatoshi Sahara                    if (auth_ismanager()) $html.= ' <bdo dir="ltr">('. $this->info['ip'] .')</bdo>';
18667ef3e88SSatoshi Sahara                } else {
18767ef3e88SSatoshi Sahara                    $html.= '<bdo dir="ltr">'. $this->info['ip'] .'</bdo>';
18867ef3e88SSatoshi Sahara                }
18967ef3e88SSatoshi Sahara                $html.= '</span>';
19067ef3e88SSatoshi Sahara                return $html;
19167ef3e88SSatoshi Sahara            }
19267ef3e88SSatoshi Sahara
19367ef3e88SSatoshi Sahara            // name of the page or media file
19467ef3e88SSatoshi Sahara            public function itemName()
19567ef3e88SSatoshi Sahara            {
19667ef3e88SSatoshi Sahara                // slightly different with display of Ui\Recent, i.e. revison may not exists
19767ef3e88SSatoshi Sahara                $id = $this->info['id'];
19867ef3e88SSatoshi Sahara                $rev = $this->info['date'];
19967ef3e88SSatoshi Sahara
200*e71e09a6SSatoshi Sahara                switch ($this->info['item']) {
201*e71e09a6SSatoshi Sahara                    case 'media': // media file revision
20267ef3e88SSatoshi Sahara                        if (isset($this->info['current'])) {
20367ef3e88SSatoshi Sahara                            $href = media_managerURL(['image'=> $id, 'tab_details'=> 'view'], '&');
20467ef3e88SSatoshi Sahara                            $html = '<a href="'.$href.'" class="wikilink1">'.$id.'</a>';
20567ef3e88SSatoshi Sahara                        } elseif (file_exists(mediaFN($id, $rev))) {
20667ef3e88SSatoshi Sahara                            $href = media_managerURL(['image'=> $id, 'tab_details'=> 'view', 'rev'=> $rev], '&');
20767ef3e88SSatoshi Sahara                            $html = '<a href="'.$href.'" class="wikilink1">'.$id.'</a>';
20867ef3e88SSatoshi Sahara                        } else {
20967ef3e88SSatoshi Sahara                            $html = $id;
21067ef3e88SSatoshi Sahara                        }
21167ef3e88SSatoshi Sahara                        return $html;
212*e71e09a6SSatoshi Sahara                    case 'page': // page revision
21367ef3e88SSatoshi Sahara                        $display_name = useHeading('navigation') ? hsc(p_get_first_heading($id)) : $id;
21467ef3e88SSatoshi Sahara                        if (!$display_name) $display_name = $id;
21567ef3e88SSatoshi Sahara                        if ($this->info['current'] || page_exists($id, $rev)) {
21667ef3e88SSatoshi Sahara                            $href = wl($id, "rev=$rev", false, '&');
21767ef3e88SSatoshi Sahara                            $html = '<a href="'.$href.'" class="wikilink1">'.$display_name.'</a>';
21867ef3e88SSatoshi Sahara                        } else {
21967ef3e88SSatoshi Sahara                            $html = $display_name;
22067ef3e88SSatoshi Sahara                        }
22167ef3e88SSatoshi Sahara                        return $html;
22267ef3e88SSatoshi Sahara                }
223*e71e09a6SSatoshi Sahara                return '';
22467ef3e88SSatoshi Sahara            }
22567ef3e88SSatoshi Sahara
22667ef3e88SSatoshi Sahara            // icon difflink
22767ef3e88SSatoshi Sahara            public function difflink()
22867ef3e88SSatoshi Sahara            {
22967ef3e88SSatoshi Sahara                global $lang;
23067ef3e88SSatoshi Sahara                $id = $this->info['id'];
23167ef3e88SSatoshi Sahara                $rev = $this->info['date'];
23267ef3e88SSatoshi Sahara
233*e71e09a6SSatoshi Sahara                switch ($this->info['item']) {
234*e71e09a6SSatoshi Sahara                    case 'media': // media file revision
23567ef3e88SSatoshi Sahara                        if (isset($this->info['current']) || !file_exists(mediaFN($id, $rev))) {
23667ef3e88SSatoshi Sahara                            $html = '<img src="'.DOKU_BASE.'lib/images/blank.gif" width="15" height="11" alt="" />';
23767ef3e88SSatoshi Sahara                        } else {
23867ef3e88SSatoshi Sahara                            $href = media_managerURL(['image'=> $id, 'rev'=> $rev, 'mediado'=>'diff'], '&');
23967ef3e88SSatoshi Sahara                            $html = '<a href="'.$href.'" class="diff_link">'
24067ef3e88SSatoshi Sahara                                  . '<img src="'.DOKU_BASE.'lib/images/diff.png" width="15" height="11"'
24167ef3e88SSatoshi Sahara                                  . ' title="'. $lang['diff'] .'" alt="'.$lang['diff'] .'" />'
24267ef3e88SSatoshi Sahara                                  . '</a> ';
24367ef3e88SSatoshi Sahara                        }
24467ef3e88SSatoshi Sahara                        return $html;
245*e71e09a6SSatoshi Sahara                    case 'page': // page revision
24667ef3e88SSatoshi Sahara                        if ($this->info['current'] || !page_exists($id, $rev)) {
24767ef3e88SSatoshi Sahara                            $html = '<img src="'.DOKU_BASE.'lib/images/blank.gif" width="15" height="11" alt="" />';
24867ef3e88SSatoshi Sahara                        } else {
24967ef3e88SSatoshi Sahara                            $href = wl($id, "rev=$rev,do=diff", false, '&');
25067ef3e88SSatoshi Sahara                            $html = '<a href="'.$href.'" class="diff_link">'
25167ef3e88SSatoshi Sahara                                  . '<img src="'.DOKU_BASE.'lib/images/diff.png" width="15" height="11"'
25267ef3e88SSatoshi Sahara                                  . ' title="'.$lang['diff'].'" alt="'.$lang['diff'].'" />'
25367ef3e88SSatoshi Sahara                                  . '</a>';
25467ef3e88SSatoshi Sahara                        }
25567ef3e88SSatoshi Sahara                        return $html;
25667ef3e88SSatoshi Sahara                }
257*e71e09a6SSatoshi Sahara                return '';
25867ef3e88SSatoshi Sahara            }
25967ef3e88SSatoshi Sahara
26067ef3e88SSatoshi Sahara            // size change
26167ef3e88SSatoshi Sahara            public function sizeChange()
26267ef3e88SSatoshi Sahara            {
26367ef3e88SSatoshi Sahara                $class = 'sizechange';
26467ef3e88SSatoshi Sahara                $value = filesize_h(abs($this->info['sizechange']));
26567ef3e88SSatoshi Sahara                if ($this->info['sizechange'] > 0) {
26667ef3e88SSatoshi Sahara                    $class .= ' positive';
26767ef3e88SSatoshi Sahara                    $value = '+' . $value;
26867ef3e88SSatoshi Sahara                } elseif ($this->info['sizechange'] < 0) {
26967ef3e88SSatoshi Sahara                    $class .= ' negative';
27067ef3e88SSatoshi Sahara                    $value = '-' . $value;
27167ef3e88SSatoshi Sahara                } else {
27267ef3e88SSatoshi Sahara                    $value = '±' . $value;
27367ef3e88SSatoshi Sahara                }
27467ef3e88SSatoshi Sahara                return '<span class="'.$class.'">'.$value.'</span>';
27567ef3e88SSatoshi Sahara            }
27667ef3e88SSatoshi Sahara        }; // end of anonymous class (objRevInfo)
2776c9fde82SSatoshi Sahara    }
2786c9fde82SSatoshi Sahara
2796c9fde82SSatoshi Sahara}
280