]> CyberLeo.Net >> Repos - SourceForge/phpwiki.git/blob - lib/CachedMarkup.php
include => require
[SourceForge/phpwiki.git] / lib / CachedMarkup.php
1 <?php 
2 rcs_id('$Id: CachedMarkup.php,v 1.58 2007-07-14 12:30:53 rurban Exp $');
3 /* Copyright (C) 2002 Geoffrey T. Dairiki <dairiki@dairiki.org>
4  * Copyright (C) 2004,2005,2006,2007 $ThePhpWikiProgrammingTeam
5  *
6  * This file is part of PhpWiki.
7  * 
8  * PhpWiki is free software; you can redistribute it and/or modify
9  * it under the terms of the GNU General Public License as published by
10  * the Free Software Foundation; either version 2 of the License, or
11  * (at your option) any later version.
12  * 
13  * PhpWiki is distributed in the hope that it will be useful,
14  * but WITHOUT ANY WARRANTY; without even the implied warranty of
15  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16  * GNU General Public License for more details.
17  * 
18  * You should have received a copy of the GNU General Public License
19  * along with PhpWiki; if not, write to the Free Software
20  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
21  */
22
23 require_once("lib/Units.php");
24
25 class CacheableMarkup extends XmlContent {
26
27     function CacheableMarkup($content, $basepage) {
28         $this->_basepage = $basepage;
29         $this->_buf = '';
30         $this->_content = array();
31         $this->_append($content);
32         if ($this->_buf != '')
33             $this->_content[] = $this->_buf;
34         unset($this->_buf);
35     }
36
37     function pack() {
38         if (function_exists('gzcompress'))
39             return gzcompress(serialize($this), 9);
40         return serialize($this);
41
42         // FIXME: probably should implement some sort of "compression"
43         //   when no gzcompress is available.
44     }
45
46     function unpack($packed) {
47         if (!$packed)
48             return false;
49
50         // ZLIB format has a five bit checksum in it's header.
51         // Lets check for sanity.
52         if (((ord($packed[0]) * 256 + ord($packed[1])) % 31 == 0)
53              and (substr($packed,0,2) == "\037\213") 
54                   or (substr($packed,0,2) == "x\332"))   // 120, 218
55         {
56             if (function_exists('gzuncompress')) {
57                 // Looks like ZLIB.
58                 $data = gzuncompress($packed);
59                 return unserialize($data);
60             } else {
61                 // user our php lib. TESTME
62                 include_once("ziplib.php");
63                 $zip = new ZipReader($packed);
64                 list(,$data,$attrib) = $zip->readFile();
65                 return unserialize($data);
66             }
67         }
68         if (substr($packed,0,2) == "O:") {
69             // Looks like a serialized object
70             return unserialize($packed);
71         }
72         if (preg_match("/^\w+$/", $packed))
73             return $packed;
74         // happened with _BackendInfo problem also.
75         trigger_error("Can't unpack bad cached markup. Probably php_zlib extension not loaded.", 
76                       E_USER_WARNING);
77         return false;
78     }
79     
80     /** Get names of wikipages linked to.
81      *
82      * @return array of hashes { linkto=>pagename, relation=>pagename }
83      */
84     function getWikiPageLinks() {
85         $links = array();
86         foreach ($this->_content as $item) {
87             if (!isa($item, 'Cached_DynamicContent'))
88                 continue;
89             if (!($item_links = $item->getWikiPageLinks($this->_basepage)))
90                 continue;
91             $links = array_merge($links, $item_links);
92         }
93         // array_unique has a bug with hashes! 
94         // set_links checks for duplicates, array_merge does not
95         //return array_unique($links);
96         return $links;
97     }
98
99     /** Get link info.
100      *
101      * This is here to support the XML-RPC listLinks() method.
102      *
103      * @return array
104      * Returns an array of hashes.
105      */
106     function getLinkInfo() {
107         $link = array();
108         foreach ($this->_content as $link) {
109             if (! isa($link, 'Cached_Link'))
110                 continue;
111             $info = $link->getLinkInfo($this->_basepage);
112             $links[$info->href] = $info;
113         }
114         return array_values($links);
115     }
116
117     function _append($item) {
118         if (is_array($item)) {
119             foreach ($item as $subitem)
120                 $this->_append($subitem);
121         }
122         elseif (!is_object($item)) {
123             $this->_buf .= $this->_quote((string) $item);
124         }
125         elseif (isa($item, 'Cached_DynamicContent')) {
126             if ($this->_buf) {
127                 $this->_content[] = $this->_buf;
128                 $this->_buf = '';
129             }
130             $this->_content[] = $item;
131         }
132         elseif (isa($item, 'XmlElement')) {
133             if ($item->isEmpty()) {
134                 $this->_buf .= $item->emptyTag();
135             }
136             else {
137                 $this->_buf .= $item->startTag();
138                 foreach ($item->getContent() as $subitem)
139                     $this->_append($subitem);
140                 $this->_buf .= "</$item->_tag>";
141
142                 if (!isset($this->_description) and $item->getTag() == 'p')
143                     $this->_glean_description($item->asString());
144             }
145             if (!$item->isInlineElement())
146                 $this->_buf .= "\n";
147         }
148         elseif (isa($item, 'XmlContent')) {
149             foreach ($item->getContent() as $item)
150                 $this->_append($item);
151         }
152         elseif (method_exists($item, 'asXML')) {
153             $this->_buf .= $item->asXML();
154         }
155         elseif (method_exists($item, 'asString')) {
156             $this->_buf .= $this->_quote($item->asString());
157         }
158         else {
159             $this->_buf .= sprintf("==Object(%s)==", get_class($item));
160         }
161     }
162
163     function _glean_description($text) {
164         static $two_sentences;
165         if (!$two_sentences) {
166             $two_sentences = pcre_fix_posix_classes("[.?!][\")]*\s+[\"(]*[[:upper:])]"
167                                                     . ".*"
168                                                     . "[.?!][\")]*\s*[\"(]*([[:upper:])]|$)");
169         }
170         
171         if (!isset($this->_description) and preg_match("/$two_sentences/sx", $text))
172             $this->_description = preg_replace("/\s*\n\s*/", " ", trim($text));
173     }
174
175     /**
176      * Guess a short description of the page.
177      *
178      * Algorithm:
179      *
180      * This algorithm was suggested on MeatballWiki by
181      * Alex Schroeder <kensanata@yahoo.com>.
182      *
183      * Use the first paragraph in the page which contains at least two
184      * sentences.
185      *
186      * @see http://www.usemod.com/cgi-bin/mb.pl?MeatballWikiSuggestions
187      *
188      * @return string
189      */
190     function getDescription () {
191         return isset($this->_description) ? $this->_description : '';
192     }
193     
194     function asXML () {
195         $xml = '';
196         $basepage = $this->_basepage;
197         
198         foreach ($this->_content as $item) {
199             if (is_string($item)) {
200                 $xml .= $item;
201             }
202             elseif (is_subclass_of($item, 
203                                    check_php_version(5) 
204                                      ? 'Cached_DynamicContent' 
205                                      : 'cached_dynamiccontent'))
206             {
207                 $val = $item->expand($basepage, $this);
208                 $xml .= $val->asXML();
209             }
210             else {
211                 $xml .= $item->asXML();
212             }
213         }
214         return $xml;
215     }
216
217     function printXML () {
218         $basepage = $this->_basepage;
219         // _content might be changed from a plugin (CreateToc)
220         for ($i=0; $i < count($this->_content); $i++) {
221             $item = $this->_content[$i];
222             if (is_string($item)) {
223                 print $item;
224             }
225             elseif (is_subclass_of($item, 
226                                    check_php_version(5) 
227                                      ? 'Cached_DynamicContent' 
228                                      : 'cached_dynamiccontent')) 
229             {   // give the content the chance to know about itself or even 
230                 // to change itself
231                 $val = $item->expand($basepage, $this);
232                 $val->printXML();
233             }
234             else {
235                 $item->printXML();
236             }
237         }
238     }
239 }       
240
241 /**
242  * The base class for all dynamic content.
243  *
244  * Dynamic content is anything that can change even when the original
245  * wiki-text from which it was parsed is unchanged.
246  */
247 class Cached_DynamicContent {
248
249     function cache(&$cache) {
250         $cache[] = $this;
251     }
252
253     function expand($basepage, &$obj) {
254         trigger_error("Pure virtual", E_USER_ERROR);
255     }
256
257     function getWikiPageLinks($basepage) {
258         return false;
259     }
260 }
261
262 class XmlRpc_LinkInfo {
263     function XmlRpc_LinkInfo($page, $type, $href, $relation = '') {
264         $this->page = $page;
265         $this->type = $type;
266         $this->href = $href;
267         $this->relation = $relation;
268         //$this->pageref = str_replace("/RPC2.php", "/index.php", $href);
269     }
270 }
271
272 class Cached_Link extends Cached_DynamicContent {
273
274     function isInlineElement() {
275         return true;
276     }
277
278     /** Get link info (for XML-RPC support)
279      *
280      * This is here to support the XML-RPC listLinks method.
281      * (See http://www.ecyrd.com/JSPWiki/Wiki.jsp?page=WikiRPCInterface)
282      */
283     function getLinkInfo($basepage) {
284         return new XmlRpc_LinkInfo($this->_getName($basepage),
285                                    $this->_getType(),
286                                    $this->_getURL($basepage),
287                                    $this->_getRelation($basepage));
288     }
289     
290     function _getURL($basepage) {
291         return $this->_url;
292     }
293     function __getRelation($basepage) {
294         return $this->_relation;
295     }
296 }
297
298 class Cached_WikiLink extends Cached_Link {
299
300     function Cached_WikiLink ($page, $label = false, $anchor = false) {
301         $this->_page = $page;
302         /* ":DontStoreLink" */
303         if (substr($this->_page,0,1) == ':') {
304             $this->_page = substr($this->_page,1);
305             $this->_nolink = true;
306         }    
307         if ($anchor)
308             $this->_anchor = $anchor;
309         if ($label and $label != $page)
310             $this->_label = $label;
311         $this->_basepage = false;    
312     }
313
314     function _getType() {
315         return 'internal';
316     }
317     
318     function getPagename($basepage) {
319         $page = new WikiPageName($this->_page, $basepage);
320         if ($page->isValid()) return $page->name;
321         else return false;
322     }
323
324     function getWikiPageLinks($basepage) {
325         if ($basepage == '') return false;
326         if (isset($this->_nolink)) return false;
327         if ($link = $this->getPagename($basepage)) 
328             return array(array('linkto' => $link, 'relation' => 0));
329         else 
330             return false;
331     }
332
333     function _getName($basepage) {
334         return $this->getPagename($basepage);
335     }
336
337     function _getURL($basepage) {
338         return WikiURL($this->getPagename($basepage));
339         //return WikiURL($this->getPagename($basepage), false, 'abs_url');
340     }
341
342     function expand($basepage, &$markup) {
343         $this->_basepage = $basepage;
344         $label = isset($this->_label) ? $this->_label : false;
345         $anchor = isset($this->_anchor) ? (string)$this->_anchor : '';
346         $page = new WikiPageName($this->_page, $basepage, $anchor);
347         if ($page->isValid()) return WikiLink($page, 'auto', $label);
348         else return HTML($label);
349     }
350
351     function asXML() {
352         $label = isset($this->_label) ? $this->_label : false;
353         $anchor = isset($this->_anchor) ? (string)$this->_anchor : '';
354         //TODO: need basepage for subpages like /Remove (within CreateTOC)
355         $page = new WikiPageName($this->_page, $this->_basepage, $anchor);
356         $link = WikiLink($page, 'auto', $label);
357         return $link->asXML();
358     }
359
360     function asString() {
361         if (isset($this->_label))
362             return $this->_label;
363         return $this->_page;
364     }
365 }
366
367 class Cached_WikiLinkIfKnown extends Cached_WikiLink
368 {
369     function Cached_WikiLinkIfKnown ($moniker) {
370         $this->_page = $moniker;
371     }
372
373     function expand($basepage, &$markup) {
374         return WikiLink($this->_page, 'if_known');
375     }
376 }    
377
378 class Cached_SpellCheck extends Cached_WikiLink
379 {
380     function Cached_SpellCheck ($word, $suggs) {
381         $this->_page = $word;
382         $this->suggestions = $suggs;
383     }
384
385     function expand($basepage, &$markup) {
386         $link = HTML::a(array('class' => 'spell-wrong', 
387                               'title' => 'SpellCheck: '.join(', ', $this->suggestions),
388                               'name' => $this->_page), 
389                         $this->_page);
390         return $link;
391     }
392 }    
393     
394 class Cached_PhpwikiURL extends Cached_DynamicContent
395 {
396     function Cached_PhpwikiURL ($url, $label) {
397         $this->_url = $url;
398         if ($label)
399             $this->_label = $label;
400     }
401
402     function isInlineElement() {
403         return true;
404     }
405
406     function expand($basepage, &$markup) {
407         $label = isset($this->_label) ? $this->_label : false;
408         return LinkPhpwikiURL($this->_url, $label, $basepage);
409     }
410
411     function asXML() {
412         $label = isset($this->_label) ? $this->_label : false;
413         $link = LinkPhpwikiURL($this->_url, $label);
414         return $link->asXML();
415     }
416
417     function asString() {
418         if (isset($this->_label))
419             return $this->_label;
420         return $this->_url;
421     }
422 }    
423
424 /*
425  * Relations (::) are named links to pages.
426  * Attributes (:=) are named metadata per page, "named links to numbers with units". 
427  * We don't want to exhaust the linktable with numbers,
428  * since this would create empty pages per each value, 
429  * so we don't store the attributes as full relationlink. 
430  * But we do store the attribute name as relation with an empty pagename 
431  * to denote that this is an attribute, 
432  * and to enable a fast listRelations mode=attributes
433  */
434 class Cached_SemanticLink extends Cached_WikiLink {
435
436     function Cached_SemanticLink ($url, $label=false) {
437         $this->_url = $url;
438         if ($label && $label != $url)
439             $this->_label = $label;
440         $this->_expandurl($this->_url);
441     }
442
443     function isInlineElement() {
444         return true;
445     }
446
447     function getPagename($basepage) {
448         if (!isset($this->_page)) return false;
449         $page = new WikiPageName($this->_page, $basepage);
450         if ($page->isValid()) return $page->name;
451         else return false;
452     }
453
454     /* Add relation to the link table.
455      * attributes have the _relation, but not the _page set.
456      */
457     function getWikiPageLinks($basepage) {
458         if ($basepage == '') return false;
459         if (!isset($this->_page) and isset($this->_attribute)) {
460             // An attribute: we store it in the basepage now, to fill the cache for page->save
461             // TODO: side-effect free query
462             $page = $GLOBALS['request']->getPage($basepage); 
463             $page->setAttribute($this->_relation, $this->_attribute);
464             $this->_page = $basepage;
465             return array(array('linkto' => '', 'relation' => $this->_relation));
466         }
467         if ($link = $this->getPagename($basepage)) 
468             return array(array('linkto' => $link, 'relation' => $this->_relation));
469         else
470             return false;
471     }
472
473     function _expandurl($url) {
474         $m = array();
475         if (!preg_match('/^ ([^:]+) (:[:=]) (.+) $/x', $url, $m)) {
476             return HTML::strong(array('class' => 'rawurl'),
477                                 HTML::u(array('class' => 'baduri'),
478                                         _("BAD semantic relation link")));
479         }
480         $this->_relation = urldecode($m[1]);
481         $is_attribute = ($m[2] == ':=');
482         if ($is_attribute) {
483             $this->_attribute = urldecode($m[3]);
484             // since this stored in the markup cache, we are extra sensible 
485             // not to store false empty stuff.
486             $units = new Units();
487             if (!DISABLE_UNITS and !$units->errcode) 
488             {
489                 $this->_attribute_base = $units->Definition($this->_attribute);
490                 $this->_unit = $units->baseunit($this->_attribute);
491             }
492         } else {
493             $this->_page = urldecode($m[3]);
494         }
495         return $m;
496     }
497
498     function _expand($url, $label = false) {
499         $m = $this->_expandurl($url);
500         $class = 'wiki';
501         // do not link to the attribute value, but to the attribute
502         $is_attribute = ($m[2] == ':=');
503         if ($is_attribute)
504             $title = isset($this->_attribute_base)
505                 ? sprintf(_("Attribute %s, base value: %s"), $this->_relation, $this->_attribute_base)
506                 : sprintf(_("Attribute %s, value: %s"), $this->_relation, $this->_attribute);
507         if ($label) {
508             return HTML::span
509                 (
510                  HTML::a(array('href'  => WikiURL($is_attribute ? $this->_relation : $this->_page),
511                                'class' => "wiki ".($is_attribute ? "attribute" : "relation"),
512                                'title' => $is_attribute 
513                                ? $title 
514                                : sprintf(_("Relation %s to page %s"), $this->_relation, $this->_page)),
515                          $label)
516                  );
517         } elseif ($is_attribute) {
518             return HTML::span
519                 (
520                  HTML::a(array('href'  => WikiURL($this->_relation),
521                                'class' => "wiki attribute",
522                                'title' => $title),
523                          $url)
524                  );
525         } else {
526             return HTML::span
527                 (
528                  HTML::a(array('href'  => WikiURL($this->_relation),
529                                'class' => "wiki relation"),
530                          $this->_relation),
531                  HTML::span(array('class'=>'relation-symbol'), $m[2]),
532                  HTML::a(array('href'  => WikiURL($this->_page),
533                                'class' => "wiki"),
534                          $this->_page)
535                  );
536         }
537     }
538
539     function expand($basepage, &$markup) {
540         $label = isset($this->_label) ? $this->_label : false;
541         return $this->_expand($this->_url, $label);
542     }
543
544     function asXML() {
545         $label = isset($this->_label) ? $this->_label : false;
546         $link = $this->_expand($this->_url, $label);
547         return $link->asXML();
548     }
549
550     function asString() {
551         if (isset($this->_label))
552             return $this->_label;
553         return $this->_url;
554     }
555 }
556
557 /** 
558  * Highlight found search engine terms
559  */
560 class Cached_SearchHighlight extends Cached_DynamicContent
561 {
562     function Cached_SearchHighlight ($word, $engine) {
563         $this->_word = $word;
564         $this->engine = $engine;
565     }
566
567     function expand($basepage, &$markup) {
568         return HTML::span(array('class' => 'search-term',
569                                 'title' => _("Found by ") . $this->engine),
570                           $this->_word);
571     }
572 }    
573     
574 class Cached_ExternalLink extends Cached_Link {
575
576     function Cached_ExternalLink($url, $label=false) {
577         $this->_url = $url;
578         if ($label && $label != $url)
579             $this->_label = $label;
580     }
581
582     function _getType() {
583         return 'external';
584     }
585     
586     function _getName($basepage) {
587         $label = isset($this->_label) ? $this->_label : false;
588         return ($label and is_string($label)) ? $label : $this->_url;
589     }
590
591     function expand($basepage, &$markup) {
592         global $request;
593
594         $label = isset($this->_label) ? $this->_label : false;
595         $link = LinkURL($this->_url, $label);
596
597         if (GOOGLE_LINKS_NOFOLLOW) {
598             // Ignores nofollow when the user who saved the page was authenticated. 
599             $page = $request->getPage($basepage);
600             $current = $page->getCurrentRevision(false);
601             if (!$current->get('author_id'))
602                 $link->setAttr('rel', 'nofollow');
603         }
604         return $link;
605     }
606
607     function asString() {
608         if (isset($this->_label))
609             return $this->_label;
610         return $this->_url;
611     }
612 }
613
614 class Cached_InterwikiLink extends Cached_ExternalLink {
615     
616     function Cached_InterwikiLink($link, $label=false) {
617         $this->_link = $link;
618         if ($label)
619             $this->_label = $label;
620     }
621
622     function getPagename($basepage) {
623         list ($moniker, $page) = split (":", $this->_link, 2);
624         $page = new WikiPageName($page, $basepage);
625         if ($page->isValid()) return $page->name;
626         else return false;
627     }
628
629     function getWikiPageLinks($basepage) {
630         if ($basepage == '') return false;
631         /* ":DontStoreLink" */
632         if (substr($this->_link,0,1) == ':') return false;
633         /* store only links to valid pagenames */
634         if ($link = $this->getPagename($basepage)) 
635             return array(array('linkto' => $link, 'relation' => 0));
636         else return false; // dont store external links
637     }
638
639     function _getName($basepage) {
640         $label = isset($this->_label) ? $this->_label : false;
641         return ($label and is_string($label)) ? $label : $this->_link;
642     }
643     
644     /* there may be internal interwiki links also */
645     function _getType() {
646         return $this->getPagename(false) ? 'internal' : 'external';
647     }
648
649     function _getURL($basepage) {
650         $link = $this->expand($basepage, $this);
651         return $link->getAttr('href');
652     }
653
654     function expand($basepage, &$markup) {
655         $intermap = getInterwikiMap();
656         $label = isset($this->_label) ? $this->_label : false;
657         return $intermap->link($this->_link, $label);
658     }
659
660     function asString() {
661         if (isset($this->_label))
662             return $this->_label;
663         return $this->_link;
664     }
665 }
666
667 // Needed to put UserPages to backlinks. Special method to markup userpages with icons
668 // Thanks to PhpWiki:DanFr for finding this bug. 
669 // Fixed since 1.3.8, prev. versions had no userpages in backlinks
670 class Cached_UserLink extends Cached_WikiLink {
671     function expand($basepage, &$markup) {
672         $label = isset($this->_label) ? $this->_label : false;
673         $anchor = isset($this->_anchor) ? (string)$this->_anchor : '';
674         $page = new WikiPageName($this->_page, $basepage, $anchor);
675         $link = WikiLink($page, 'auto', $label);
676         // $link = HTML::a(array('href' => $PageName));
677         $link->setContent(PossiblyGlueIconToText('wikiuser', $this->_page));
678         $link->setAttr('class', 'wikiuser');
679         return $link;
680     }
681 }
682
683 /**
684  * 1.3.13: Previously stored was only _pi. 
685  * A fresh generated cache has now ->name and ->args also.
686  * main::isActionPage only checks the raw content.
687  */
688 class Cached_PluginInvocation extends Cached_DynamicContent {
689
690     function Cached_PluginInvocation ($pi) {
691         $this->_pi = $pi;
692         $loader = $this->_getLoader();
693         if (is_array($plugin_cmdline = $loader->parsePI($pi)) and $plugin_cmdline[1]) {
694             $this->pi_name = $plugin_cmdline[0]; // plugin, plugin-form, plugin-list, plugin-link
695             $this->name = $plugin_cmdline[1]->getName();
696             $this->args = $plugin_cmdline[2];
697         }
698     }
699
700     function setTightness($top, $bottom) {
701         $this->_tightenable = 0;
702         if ($top) $this->_tightenable |= 1;
703         if ($bottom) $this->_tightenable |= 2;
704     }
705     
706     function isInlineElement() {
707         return false;
708     }
709
710     function expand($basepage, &$markup) {
711         $loader = $this->_getLoader();
712
713         $xml = $loader->expandPI($this->_pi, $GLOBALS['request'], $markup, $basepage);
714         $div = HTML::div(array('class' => 'plugin'));
715         if (isset($this->name))
716             $id = GenerateId($this->name . 'Plugin');
717    
718         if (isset($this->_tightenable)) {
719             if ($this->_tightenable == 3) {
720                 $span = HTML::span(array('class' => 'plugin'), $xml);
721                 if (!empty($id))
722                     $span->setAttr('id', $id);
723                 return $span;
724             }
725             $div->setInClass('tightenable');
726             $div->setInClass('top', ($this->_tightenable & 1) != 0);
727             $div->setInClass('bottom', ($this->_tightenable & 2) != 0);
728         }
729         if (!empty($id))
730             $div->setAttr('id', $id);
731         $div->pushContent($xml);
732         return $div;
733     }
734
735     function asString() {
736         return $this->_pi;
737     }
738
739
740     function getWikiPageLinks($basepage) {
741         $loader = $this->_getLoader();
742
743         return $loader->getWikiPageLinks($this->_pi, $basepage);
744     }
745
746     function & _getLoader() {
747         static $loader = false;
748
749         if (!$loader) {
750             include_once('lib/WikiPlugin.php');
751             $loader = new WikiPluginLoader;
752         }
753         return $loader;
754     }
755 }
756
757 // $Log: not supported by cvs2svn $
758 // Revision 1.57  2007/05/28 20:13:46  rurban
759 // Overwrite all attributes at once at page->save to delete dangling meta
760 //
761 // Revision 1.56  2007/04/08 16:39:40  rurban
762 // fix when DISABLE_UNITS = true (thanks to Walter Rafelsberger)
763 // simplify title calculation
764 //
765 // Revision 1.55  2007/03/18 17:35:14  rurban
766 // Fix :DontStoreLink
767 //
768 // Revision 1.54  2007/01/25 07:41:41  rurban
769 // Print attribute in title. Use CSS formatting for ::=
770 //
771 // Revision 1.53  2007/01/21 23:26:52  rurban
772 // Translate Found by
773 //
774 // Revision 1.52  2007/01/20 15:53:51  rurban
775 // Rewrite of SearchHighlight: through ActionPage and InlineParser
776 //
777 // Revision 1.51  2007/01/20 11:24:53  rurban
778 // add SpellCheck support
779 //
780 // Revision 1.50  2007/01/07 18:41:51  rurban
781 // Fix fallback ZipReader syntax error. Use label=false. Add parsed plugin names to the stored tree.
782 //
783 // Revision 1.49  2007/01/04 16:40:35  rurban
784 // Remove units object from CachedMarkup links, Store parsed linkinfo only: basevalue, baseunit.
785 //
786 // Revision 1.48  2007/01/03 21:22:08  rurban
787 // Use Units for attributes. Store the unified base value as Cached_SemanticLink->_attribute_base in the wikimarkup and display it as title.
788 //
789 // Revision 1.47  2007/01/02 13:17:57  rurban
790 // fix semantic page links and attributes, esp. attributes. they get stored as link to empty page also. tighten semantic url expander regex, omit want_content if not necessary
791 //
792 // Revision 1.46  2006/12/22 00:11:38  rurban
793 // add seperate expandurl method, to simplify pagename parsing
794 //
795 // Revision 1.45  2006/10/12 06:33:50  rurban
796 // decide later with which class to render this link (fixes interwiki link layout)
797
798 // (c-file-style: "gnu")
799 // Local Variables:
800 // mode: php
801 // tab-width: 8
802 // c-basic-offset: 4
803 // c-hanging-comment-ender-p: nil
804 // indent-tabs-mode: nil
805 // End:   
806 ?>