]> CyberLeo.Net >> Repos - SourceForge/phpwiki.git/blob - lib/interwiki.php
Read interwiki map from wiki page (InterWikiMap). The page must be
[SourceForge/phpwiki.git] / lib / interwiki.php
1 <?php rcs_id('$Id: interwiki.php,v 1.15 2002-01-31 05:10:28 dairiki Exp $');
2
3 class InterWikiMap {
4     function InterWikiMap (&$request) {
5         $dbi = $request->getDbh();
6
7         $intermap = $this->_getMapFromWikiPage($dbi->getPage(_("InterWikiMap")));
8
9         if (!$intermap && defined('INTERWIKI_MAP_FILE'))
10             $intermap = $this->_getMapFromFile(INTERWIKI_MAP_FILE);
11
12         $this->_map = $this->_parseMap($intermap);
13         $this->_regexp = $this->_getRegexp();
14     }
15
16     function GetMap (&$request) {
17         static $map;
18         if (empty($map))
19             $map = new InterWikiMap($request);
20         return $map;
21     }
22     
23     function getRegexp() {
24         return $this->_regexp;
25     }
26
27     function link ($link, $linktext = false) {
28
29         list ($moniker, $page) = split (":", $link, 2);
30         
31         if (!isset($this->_map[$moniker])) {
32             return HTML::span(array('class' => 'bad-interwiki'),
33                               $linktext ? $linktext : $link);
34         }
35
36         $url = $this->_map[$moniker];
37         
38         // Urlencode page only if it's a query arg.
39         // FIXME: this is a somewhat broken heuristic.
40         $page_enc = strstr($url, '?') ? rawurlencode($page) : $page;
41
42         if (strstr($url, '%s'))
43             $url = sprintf($url, $page_enc);
44         else
45             $url .= $page_enc;
46         
47         $link = HTML::a(array('href' => $url),
48                         IconForLink('interwiki'));
49
50         if (!$linktext) {
51             $link->pushContent("$moniker:",
52                                HTML::span(array('class' => 'wikipage'), $page));
53             $link->setAttr('class', 'interwiki');
54         }
55         else {
56             $link->pushContent($linktext);
57             $link->setAttr('class', 'named-interwiki');
58         }
59         
60         return $link;
61     }
62
63
64     function _parseMap ($text) {
65         global $AllowedProtocols;
66         if (!preg_match_all("/^\s*(\S+)\s+((?:$AllowedProtocols):[^\s<>\"']+)/m",
67                             $text, $matches, PREG_SET_ORDER))
68             return false;
69         foreach ($matches as $m)
70             $map[$m[1]] = $m[2];
71         return $map;
72     }
73         
74     function _getMapFromWikiPage ($page) {
75         if (! $page->get('locked'))
76             return false;
77         
78         $current = $page->getCurrentRevision();
79         
80         if (preg_match('|^<verbatim>\n(.*)^</verbatim>|ms',
81                        $current->getPackedContent(), $m)) {
82             return $m[1];
83         }
84         return false;
85     }
86
87     function _getMapFromFile ($filename) {
88         
89         @$fd = fopen ($filename, "rb");
90         @$data = fread ($fd, filesize($filename));
91         @fclose ($fd);
92         return $data;
93     }
94
95     function _getRegexp () {
96         if (!$this->_map)
97             return '(?:(?!a)a)'; //  Never matches.
98         
99         foreach (array_keys($this->_map) as $moniker)
100             $qkeys[] = preg_quote($moniker, '/');
101         return "(?:" . join("|", $qkeys) . ")";
102     }
103 }
104             
105
106 // For emacs users
107 // Local Variables:
108 // mode: php
109 // tab-width: 8
110 // c-basic-offset: 4
111 // c-hanging-comment-ender-p: nil
112 // indent-tabs-mode: nil
113 // End:
114 ?>