xref: /dokuwiki/inc/Ui/Revisions.php (revision 90c7493e5431e6bb1956245e233ab58734b59b05)
16c9fde82SSatoshi Sahara<?php
26c9fde82SSatoshi Sahara
36c9fde82SSatoshi Saharanamespace dokuwiki\Ui;
46c9fde82SSatoshi Sahara
5*90c7493eSSatoshi Saharause dokuwiki\ChangeLog\ChangeLog;
6*90c7493eSSatoshi Sahara
76c9fde82SSatoshi Sahara/**
89e316641SSatoshi Sahara * DokuWiki Revisions Interface
9ae5d2354SSatoshi Sahara * parent class of PageRevisions and MediaRevisions
106c9fde82SSatoshi Sahara *
116c9fde82SSatoshi Sahara * @package dokuwiki\Ui
126c9fde82SSatoshi Sahara */
13ae5d2354SSatoshi Saharaabstract class Revisions extends Ui
146c9fde82SSatoshi Sahara{
15ae5d2354SSatoshi Sahara    /* @var string */
16e71e09a6SSatoshi Sahara    protected $id;   // page id or media id
17e71e09a6SSatoshi Sahara    protected $item; // page or media
18e71e09a6SSatoshi Sahara
19e71e09a6SSatoshi Sahara    /* @var ChangeLog */
20e71e09a6SSatoshi Sahara    protected $changelog; // PageChangeLog or MediaChangeLog object
219e316641SSatoshi Sahara
229e316641SSatoshi Sahara    /**
239e316641SSatoshi Sahara     * Revisions Ui constructor
249e316641SSatoshi Sahara     *
25e71e09a6SSatoshi Sahara     * @param string $id  page id or media id
269e316641SSatoshi Sahara     */
27ae5d2354SSatoshi Sahara    public function __construct($id)
289e316641SSatoshi Sahara    {
29ae5d2354SSatoshi Sahara        $this->id = $id;
30e71e09a6SSatoshi Sahara        $this->setChangeLog();
31e71e09a6SSatoshi Sahara    }
32e71e09a6SSatoshi Sahara
33e71e09a6SSatoshi Sahara    /**
34e71e09a6SSatoshi Sahara     * set class property changelog
35e71e09a6SSatoshi Sahara     */
36e71e09a6SSatoshi Sahara    abstract protected function setChangeLog();
37e71e09a6SSatoshi Sahara
38e71e09a6SSatoshi Sahara    /**
39d9c75b22SSatoshi Sahara     * item filename resolver
40e71e09a6SSatoshi Sahara     *
41e71e09a6SSatoshi Sahara     * @param string $id  page id or media id
42d9c75b22SSatoshi Sahara     * @param int|string $rev revision timestamp, or empty string for current one
43e71e09a6SSatoshi Sahara     * @return string full path
44e71e09a6SSatoshi Sahara     */
45d9c75b22SSatoshi Sahara    abstract protected function itemFN($id, $rev = '');
469e316641SSatoshi Sahara
476c9fde82SSatoshi Sahara    /**
4867ef3e88SSatoshi Sahara     * Get revisions, and set correct pagenation parameters (first, hasNext)
4967ef3e88SSatoshi Sahara     *
5067ef3e88SSatoshi Sahara     * @param int  $first
5167ef3e88SSatoshi Sahara     * @param bool $hasNext
5267ef3e88SSatoshi Sahara     * @return array  revisions to be shown in a pagenated list
535525369fSSatoshi Sahara     * @see also https://www.dokuwiki.org/devel:changelog
5467ef3e88SSatoshi Sahara     */
55e71e09a6SSatoshi Sahara    protected function getRevisions(&$first, &$hasNext)
56e71e09a6SSatoshi Sahara    {
57e71e09a6SSatoshi Sahara        global $conf;
58e71e09a6SSatoshi Sahara
59e71e09a6SSatoshi Sahara        $changelog =& $this->changelog;
60*90c7493eSSatoshi Sahara        $revisions = [];
61e71e09a6SSatoshi Sahara
62*90c7493eSSatoshi Sahara        $extEditRevInfo = $changelog->getExternalEditRevInfo();
63e71e09a6SSatoshi Sahara
64e71e09a6SSatoshi Sahara        /* we need to get one additional log entry to be able to
65e71e09a6SSatoshi Sahara         * decide if this is the last page or is there another one.
66*90c7493eSSatoshi Sahara         * @see Ui\Recent::getRecents() as well
67e71e09a6SSatoshi Sahara         */
68*90c7493eSSatoshi Sahara        $num = $conf['recent'];
69*90c7493eSSatoshi Sahara        if ($first == 0) {
70*90c7493eSSatoshi Sahara            $num = $extEditRevInfo ? $num - 1 : $num;
71*90c7493eSSatoshi Sahara        }
72*90c7493eSSatoshi Sahara        $revlist = $changelog->getRevisions($first-1, $num +1 );
73e71e09a6SSatoshi Sahara        if (count($revlist) == 0 && $first != 0) {
74e71e09a6SSatoshi Sahara            $first = 0;
75*90c7493eSSatoshi Sahara            $num = $extEditRevInfo ? $num - 1 : $num;
76*90c7493eSSatoshi Sahara            $revlist = $changelog->getRevisions(-1, $num + 1);
77e71e09a6SSatoshi Sahara        }
78e71e09a6SSatoshi Sahara
79*90c7493eSSatoshi Sahara        if ($first == 0 && $extEditRevInfo) {
80*90c7493eSSatoshi Sahara            $revisions[] = $extEditRevInfo + [
81e71e09a6SSatoshi Sahara                    'item' => $this->item,
82e71e09a6SSatoshi Sahara                    'current' => true,
83*90c7493eSSatoshi Sahara            ];
84e71e09a6SSatoshi Sahara        }
85e71e09a6SSatoshi Sahara
86e71e09a6SSatoshi Sahara        // decide if this is the last page or is there another one
87e71e09a6SSatoshi Sahara        $hasNext = false;
88*90c7493eSSatoshi Sahara        if (count($revlist) > $num) {
89e71e09a6SSatoshi Sahara            $hasNext = true;
90e71e09a6SSatoshi Sahara            array_pop($revlist); // remove one additional log entry
91e71e09a6SSatoshi Sahara        }
92e71e09a6SSatoshi Sahara
93e71e09a6SSatoshi Sahara        // append each revison info array to the revisions
94*90c7493eSSatoshi Sahara        $fileLastMod = $this->itemFN($this->id);
95*90c7493eSSatoshi Sahara        $lastMod     = @filemtime($fileLastMod); // from wiki page, suppresses warning in case the file not exists
96e71e09a6SSatoshi Sahara        foreach ($revlist as $rev) {
97*90c7493eSSatoshi Sahara            $more = ['item' => $this->item];
98*90c7493eSSatoshi Sahara            if ($rev == $lastMod) {
99*90c7493eSSatoshi Sahara                $more['current'] = true;
100*90c7493eSSatoshi Sahara            }
101*90c7493eSSatoshi Sahara            $revisions[] = $changelog->getRevisionInfo($rev) + $more;
102e71e09a6SSatoshi Sahara        }
103e71e09a6SSatoshi Sahara        return $revisions;
104e71e09a6SSatoshi Sahara    }
10567ef3e88SSatoshi Sahara
10667ef3e88SSatoshi Sahara    /**
10767ef3e88SSatoshi Sahara     * Navigation buttons for Pagenation (prev/next)
10867ef3e88SSatoshi Sahara     *
10967ef3e88SSatoshi Sahara     * @param int  $first
11067ef3e88SSatoshi Sahara     * @param bool $hasNext
111ae5d2354SSatoshi Sahara     * @param callable $callback returns array of hidden fields for the form button
112*90c7493eSSatoshi Sahara     * @return string html
11367ef3e88SSatoshi Sahara     */
114ae5d2354SSatoshi Sahara    protected function navigation($first, $hasNext, $callback)
11567ef3e88SSatoshi Sahara    {
11667ef3e88SSatoshi Sahara        global $conf;
11767ef3e88SSatoshi Sahara
11867ef3e88SSatoshi Sahara        $html = '<div class="pagenav">';
1196c9fde82SSatoshi Sahara        $last = $first + $conf['recent'];
1206c9fde82SSatoshi Sahara        if ($first > 0) {
12167ef3e88SSatoshi Sahara            $first = max($first - $conf['recent'], 0);
12267ef3e88SSatoshi Sahara            $html.= '<div class="pagenav-prev">';
123ae5d2354SSatoshi Sahara            $html.= html_btn('newer', $this->id, "p", $callback($first));
12467ef3e88SSatoshi Sahara            $html.= '</div>';
1256c9fde82SSatoshi Sahara        }
1266c9fde82SSatoshi Sahara        if ($hasNext) {
12767ef3e88SSatoshi Sahara            $html.= '<div class="pagenav-next">';
128ae5d2354SSatoshi Sahara            $html.= html_btn('older', $this->id, "n", $callback($last));
12967ef3e88SSatoshi Sahara            $html.= '</div>';
1306c9fde82SSatoshi Sahara        }
13167ef3e88SSatoshi Sahara        $html.= '</div>';
13267ef3e88SSatoshi Sahara        return $html;
13367ef3e88SSatoshi Sahara    }
1346c9fde82SSatoshi Sahara
13567ef3e88SSatoshi Sahara    /**
13667ef3e88SSatoshi Sahara     * Returns instance of objRevInfo
13767ef3e88SSatoshi Sahara     *
13867ef3e88SSatoshi Sahara     * @param array $info  Revision info structure of a page or media file
13967ef3e88SSatoshi Sahara     * @return objRevInfo object (anonymous class)
14067ef3e88SSatoshi Sahara     */
1410bb448f0SSatoshi Sahara    public function getObjRevInfo(array $info)
14267ef3e88SSatoshi Sahara    {
14367ef3e88SSatoshi Sahara        return new class ($info) // anonymous class (objRevInfo)
14467ef3e88SSatoshi Sahara        {
14567ef3e88SSatoshi Sahara            protected $info;
14667ef3e88SSatoshi Sahara
14767ef3e88SSatoshi Sahara            public function __construct(array $info)
14867ef3e88SSatoshi Sahara            {
149c63f6084SDamien Regad                if (!isset($info['current'])) {
150c63f6084SDamien Regad                    $info['current'] = false;
151c63f6084SDamien Regad                }
15267ef3e88SSatoshi Sahara                $this->info = $info;
15367ef3e88SSatoshi Sahara            }
15467ef3e88SSatoshi Sahara
15567ef3e88SSatoshi Sahara            // current indicator
15667ef3e88SSatoshi Sahara            public function currentIndicator()
15767ef3e88SSatoshi Sahara            {
15867ef3e88SSatoshi Sahara                global $lang;
15967ef3e88SSatoshi Sahara                return ($this->info['current']) ? '('.$lang['current'].')' : '';
16067ef3e88SSatoshi Sahara            }
16167ef3e88SSatoshi Sahara
16267ef3e88SSatoshi Sahara            // edit date and time of the page or media file
16367ef3e88SSatoshi Sahara            public function editDate()
16467ef3e88SSatoshi Sahara            {
165*90c7493eSSatoshi Sahara                global $lang;
166*90c7493eSSatoshi Sahara                if ($this->info['date'] === 9999999999) {
167*90c7493eSSatoshi Sahara                    $date = $lang['unknowndate']; //unknown when files are deleted externally
168*90c7493eSSatoshi Sahara                } else {
169*90c7493eSSatoshi Sahara                    $date = dformat($this->info['date']);
170*90c7493eSSatoshi Sahara                }
171*90c7493eSSatoshi Sahara                return '<span class="date">'. $date .'</span>';
17267ef3e88SSatoshi Sahara            }
17367ef3e88SSatoshi Sahara
17467ef3e88SSatoshi Sahara            // edit summary
17567ef3e88SSatoshi Sahara            public function editSummary()
17667ef3e88SSatoshi Sahara            {
17767ef3e88SSatoshi Sahara                return '<span class="sum">'.' – '. hsc($this->info['sum']).'</span>';
17867ef3e88SSatoshi Sahara            }
17967ef3e88SSatoshi Sahara
18067ef3e88SSatoshi Sahara            // editor of the page or media file
18167ef3e88SSatoshi Sahara            public function editor()
18267ef3e88SSatoshi Sahara            {
18367ef3e88SSatoshi Sahara                // slightly different with display of Ui\Recent, i.e. external edit
18467ef3e88SSatoshi Sahara                global $lang;
18567ef3e88SSatoshi Sahara                $html = '<span class="user">';
18667ef3e88SSatoshi Sahara                if (!$this->info['user'] && !$this->info['ip']) {
18767ef3e88SSatoshi Sahara                    $html.= '('.$lang['external_edit'].')';
18867ef3e88SSatoshi Sahara                } elseif ($this->info['user']) {
18967ef3e88SSatoshi Sahara                    $html.= '<bdi>'. editorinfo($this->info['user']) .'</bdi>';
19067ef3e88SSatoshi Sahara                    if (auth_ismanager()) $html.= ' <bdo dir="ltr">('. $this->info['ip'] .')</bdo>';
19167ef3e88SSatoshi Sahara                } else {
19267ef3e88SSatoshi Sahara                    $html.= '<bdo dir="ltr">'. $this->info['ip'] .'</bdo>';
19367ef3e88SSatoshi Sahara                }
19467ef3e88SSatoshi Sahara                $html.= '</span>';
19567ef3e88SSatoshi Sahara                return $html;
19667ef3e88SSatoshi Sahara            }
19767ef3e88SSatoshi Sahara
19867ef3e88SSatoshi Sahara            // name of the page or media file
19967ef3e88SSatoshi Sahara            public function itemName()
20067ef3e88SSatoshi Sahara            {
20167ef3e88SSatoshi Sahara                // slightly different with display of Ui\Recent, i.e. revison may not exists
20267ef3e88SSatoshi Sahara                $id = $this->info['id'];
20367ef3e88SSatoshi Sahara                $rev = $this->info['date'];
20467ef3e88SSatoshi Sahara
205e71e09a6SSatoshi Sahara                switch ($this->info['item']) {
206e71e09a6SSatoshi Sahara                    case 'media': // media file revision
20767ef3e88SSatoshi Sahara                        if (isset($this->info['current'])) {
20867ef3e88SSatoshi Sahara                            $href = media_managerURL(['image'=> $id, 'tab_details'=> 'view'], '&');
20967ef3e88SSatoshi Sahara                            $html = '<a href="'.$href.'" class="wikilink1">'.$id.'</a>';
21067ef3e88SSatoshi Sahara                        } elseif (file_exists(mediaFN($id, $rev))) {
21167ef3e88SSatoshi Sahara                            $href = media_managerURL(['image'=> $id, 'tab_details'=> 'view', 'rev'=> $rev], '&');
21267ef3e88SSatoshi Sahara                            $html = '<a href="'.$href.'" class="wikilink1">'.$id.'</a>';
21367ef3e88SSatoshi Sahara                        } else {
21467ef3e88SSatoshi Sahara                            $html = $id;
21567ef3e88SSatoshi Sahara                        }
21667ef3e88SSatoshi Sahara                        return $html;
217e71e09a6SSatoshi Sahara                    case 'page': // page revision
21867ef3e88SSatoshi Sahara                        $display_name = useHeading('navigation') ? hsc(p_get_first_heading($id)) : $id;
21967ef3e88SSatoshi Sahara                        if (!$display_name) $display_name = $id;
22067ef3e88SSatoshi Sahara                        if ($this->info['current'] || page_exists($id, $rev)) {
22167ef3e88SSatoshi Sahara                            $href = wl($id, "rev=$rev", false, '&');
22267ef3e88SSatoshi Sahara                            $html = '<a href="'.$href.'" class="wikilink1">'.$display_name.'</a>';
22367ef3e88SSatoshi Sahara                        } else {
22467ef3e88SSatoshi Sahara                            $html = $display_name;
22567ef3e88SSatoshi Sahara                        }
22667ef3e88SSatoshi Sahara                        return $html;
22767ef3e88SSatoshi Sahara                }
228e71e09a6SSatoshi Sahara                return '';
22967ef3e88SSatoshi Sahara            }
23067ef3e88SSatoshi Sahara
23167ef3e88SSatoshi Sahara            // icon difflink
23267ef3e88SSatoshi Sahara            public function difflink()
23367ef3e88SSatoshi Sahara            {
23467ef3e88SSatoshi Sahara                global $lang;
23567ef3e88SSatoshi Sahara                $id = $this->info['id'];
23667ef3e88SSatoshi Sahara                $rev = $this->info['date'];
23767ef3e88SSatoshi Sahara
238e71e09a6SSatoshi Sahara                switch ($this->info['item']) {
239e71e09a6SSatoshi Sahara                    case 'media': // media file revision
24067ef3e88SSatoshi Sahara                        if (isset($this->info['current']) || !file_exists(mediaFN($id, $rev))) {
24167ef3e88SSatoshi Sahara                            $html = '<img src="'.DOKU_BASE.'lib/images/blank.gif" width="15" height="11" alt="" />';
24267ef3e88SSatoshi Sahara                        } else {
24367ef3e88SSatoshi Sahara                            $href = media_managerURL(['image'=> $id, 'rev'=> $rev, 'mediado'=>'diff'], '&');
24467ef3e88SSatoshi Sahara                            $html = '<a href="'.$href.'" class="diff_link">'
24567ef3e88SSatoshi Sahara                                  . '<img src="'.DOKU_BASE.'lib/images/diff.png" width="15" height="11"'
24667ef3e88SSatoshi Sahara                                  . ' title="'. $lang['diff'] .'" alt="'.$lang['diff'] .'" />'
24767ef3e88SSatoshi Sahara                                  . '</a> ';
24867ef3e88SSatoshi Sahara                        }
24967ef3e88SSatoshi Sahara                        return $html;
250e71e09a6SSatoshi Sahara                    case 'page': // page revision
25167ef3e88SSatoshi Sahara                        if ($this->info['current'] || !page_exists($id, $rev)) {
25267ef3e88SSatoshi Sahara                            $html = '<img src="'.DOKU_BASE.'lib/images/blank.gif" width="15" height="11" alt="" />';
25367ef3e88SSatoshi Sahara                        } else {
25467ef3e88SSatoshi Sahara                            $href = wl($id, "rev=$rev,do=diff", false, '&');
25567ef3e88SSatoshi Sahara                            $html = '<a href="'.$href.'" class="diff_link">'
25667ef3e88SSatoshi Sahara                                  . '<img src="'.DOKU_BASE.'lib/images/diff.png" width="15" height="11"'
25767ef3e88SSatoshi Sahara                                  . ' title="'.$lang['diff'].'" alt="'.$lang['diff'].'" />'
25867ef3e88SSatoshi Sahara                                  . '</a>';
25967ef3e88SSatoshi Sahara                        }
26067ef3e88SSatoshi Sahara                        return $html;
26167ef3e88SSatoshi Sahara                }
262e71e09a6SSatoshi Sahara                return '';
26367ef3e88SSatoshi Sahara            }
26467ef3e88SSatoshi Sahara
26567ef3e88SSatoshi Sahara            // size change
26667ef3e88SSatoshi Sahara            public function sizeChange()
26767ef3e88SSatoshi Sahara            {
26867ef3e88SSatoshi Sahara                $class = 'sizechange';
26967ef3e88SSatoshi Sahara                $value = filesize_h(abs($this->info['sizechange']));
27067ef3e88SSatoshi Sahara                if ($this->info['sizechange'] > 0) {
27167ef3e88SSatoshi Sahara                    $class .= ' positive';
27267ef3e88SSatoshi Sahara                    $value = '+' . $value;
27367ef3e88SSatoshi Sahara                } elseif ($this->info['sizechange'] < 0) {
27467ef3e88SSatoshi Sahara                    $class .= ' negative';
27567ef3e88SSatoshi Sahara                    $value = '-' . $value;
27667ef3e88SSatoshi Sahara                } else {
27767ef3e88SSatoshi Sahara                    $value = '±' . $value;
27867ef3e88SSatoshi Sahara                }
27967ef3e88SSatoshi Sahara                return '<span class="'.$class.'">'.$value.'</span>';
28067ef3e88SSatoshi Sahara            }
28167ef3e88SSatoshi Sahara        }; // end of anonymous class (objRevInfo)
2826c9fde82SSatoshi Sahara    }
2836c9fde82SSatoshi Sahara
2846c9fde82SSatoshi Sahara}
285