Difference between revisions of "Extension:Treeview4.php"
m |
(4.0.1 - made more efficient rendering by using single call to wiki-parser) |
||
Line 3: | Line 3: | ||
# - See http://www.mediawiki.org/wiki/Extension:Tree_view for installation and usage details | # - See http://www.mediawiki.org/wiki/Extension:Tree_view for installation and usage details | ||
# - Licenced under LGPL (http://www.gnu.org/copyleft/lesser.html) | # - Licenced under LGPL (http://www.gnu.org/copyleft/lesser.html) | ||
− | # - Author: | + | # - Author: http://www.organicdesign.co.nz/nad |
− | + | ||
− | |||
if (!defined('MEDIAWIKI')) die('Not an entry point.'); | if (!defined('MEDIAWIKI')) die('Not an entry point.'); | ||
− | + | ||
− | define('TREEVIEW4_VERSION','4.0. | + | define('TREEVIEW4_VERSION','4.0.1, 2007-09-07'); |
− | + | ||
# Set any unset images to default titles | # Set any unset images to default titles | ||
if (!is_array($wgTreeViewImages)) $wgTreeViewImages = array(); | if (!is_array($wgTreeViewImages)) $wgTreeViewImages = array(); | ||
Line 18: | Line 17: | ||
if (!isset($wgTreeViewImages['doc'])) $wgTreeViewImages['doc'] = 'Doc-icon.gif'; | if (!isset($wgTreeViewImages['doc'])) $wgTreeViewImages['doc'] = 'Doc-icon.gif'; | ||
if (!isset($wgTreeViewImages['spacer'])) $wgTreeViewImages['spacer'] = 'Blank.gif'; | if (!isset($wgTreeViewImages['spacer'])) $wgTreeViewImages['spacer'] = 'Blank.gif'; | ||
− | + | ||
# Keep track of JavaScript added to page to avoid doubleups | # Keep track of JavaScript added to page to avoid doubleups | ||
if (!isset($wgJS)) $wgJS = array(); | if (!isset($wgJS)) $wgJS = array(); | ||
− | + | ||
$wgTreeView4Magic = "tree"; # the parser-function name for trees | $wgTreeView4Magic = "tree"; # the parser-function name for trees | ||
$wgExtensionFunctions[] = 'wfSetupTreeView4'; | $wgExtensionFunctions[] = 'wfSetupTreeView4'; | ||
$wgHooks['LanguageGetMagic'][] = 'wfTreeView4LanguageGetMagic'; | $wgHooks['LanguageGetMagic'][] = 'wfTreeView4LanguageGetMagic'; | ||
− | + | ||
$wgExtensionCredits['parserhook'][] = array( | $wgExtensionCredits['parserhook'][] = array( | ||
'name' => 'Treeview4', | 'name' => 'Treeview4', | ||
Line 33: | Line 32: | ||
'version' => TREEVIEW4_VERSION | 'version' => TREEVIEW4_VERSION | ||
); | ); | ||
− | + | ||
class TreeView4 { | class TreeView4 { | ||
− | + | ||
var $version = TREEVIEW4_VERSION; | var $version = TREEVIEW4_VERSION; | ||
var $width = 16; | var $width = 16; | ||
var $uniq = array(); | var $uniq = array(); | ||
var $js = 0; | var $js = 0; | ||
− | + | ||
# Constructor | # Constructor | ||
function TreeView4($magic) { | function TreeView4($magic) { | ||
Line 58: | Line 57: | ||
} | } | ||
} | } | ||
− | + | ||
# Restructure recursive trees into a single bullet list surrounded by internal treeview4 tags | # Restructure recursive trees into a single bullet list surrounded by internal treeview4 tags | ||
function Tree(&$parser) { | function Tree(&$parser) { | ||
Line 65: | Line 64: | ||
$args = "id='$id'"; | $args = "id='$id'"; | ||
foreach (func_get_args() as $arg) if (!is_object($arg)) { | foreach (func_get_args() as $arg) if (!is_object($arg)) { | ||
− | if (preg_match('/^(.+? | + | if (preg_match('/^(.+?=.+)$/',$arg,$m)) $args .= " $arg"; else $text = $arg; |
} | } | ||
− | $text = preg_replace( | + | |
+ | # Back magic | ||
+ | $text = $this->uniq[$id] = preg_replace( | ||
'/^\\*(\\**)\\s*(.UNIQ.+?-treeview4(.+?)-.+?-QINU)/me', | '/^\\*(\\**)\\s*(.UNIQ.+?-treeview4(.+?)-.+?-QINU)/me', | ||
'$this->uniq["$3"] ? preg_replace("/^(\\*+)/m","$1\$1",$this->uniq["$3"]) : "$1$2"', | '$this->uniq["$3"] ? preg_replace("/^(\\*+)/m","$1\$1",$this->uniq["$3"]) : "$1$2"', | ||
$text | $text | ||
); | ); | ||
− | + | ||
+ | # And a little more voodoo here | ||
$wgParser->setHook("treeview4$id",array($this,'treeview')); | $wgParser->setHook("treeview4$id",array($this,'treeview')); | ||
return "<treeview4$id $args>$text</treeview4$id>"; | return "<treeview4$id $args>$text</treeview4$id>"; | ||
Line 80: | Line 82: | ||
function treeview($text,$argv,&$parser) { | function treeview($text,$argv,&$parser) { | ||
global $wgTreeViewImages; | global $wgTreeViewImages; | ||
− | + | ||
$id = $argv['id']; | $id = $argv['id']; | ||
$ver = $this->version; | $ver = $this->version; | ||
Line 90: | Line 92: | ||
$spacer = $wgTreeViewImages['spacer']; | $spacer = $wgTreeViewImages['spacer']; | ||
$doc = $wgTreeViewImages['doc']; | $doc = $wgTreeViewImages['doc']; | ||
− | $default = isset($ | + | $default = isset($argv['openlevels']) ? $argv['openlevels']+1 : 1; |
$rows = array(); | $rows = array(); | ||
$tree = ''; | $tree = ''; | ||
− | + | ||
− | # | + | # Protect the asterisk structure and wiki-parse the bullet tree |
− | + | $text = preg_replace_callback( | |
− | + | "/^(\\*+)\\s*(.+)\\s*$/m", | |
− | + | create_function('$m','return "@@@".strlen($m[1])."@$m[2]@@@";'), | |
− | + | $text | |
+ | ); | ||
+ | $out = $parser->parse($text,$parser->mTitle,$parser->mOptions,false,false); | ||
+ | $text = $out->getText(); | ||
+ | |||
+ | # Convert the processed text into a tree table | ||
+ | if (preg_match_all('/@@@([0-9]+)@(.+?)@@@/',$text,$rows,PREG_SET_ORDER)) { | ||
+ | foreach ($rows as $i => $row) { | ||
+ | list(,$depth,$item) = $row; | ||
+ | $next = $i < count($rows) ? $rows[$i+1][1] : 0; | ||
+ | $show = $depth > $default ? ' style="display:none"' : ''; | ||
+ | if ($depth >= $default) { $open = $plus; $icon = $closed; } else { $open = $minus; $icon = $opened; } | ||
+ | if ($depth >= $next) { $open = "<img src='$spacer' width='$width'/>"; $icon = $doc; } | ||
+ | else $open = "<a href='javascript:toggleTreeviewItem(\"$id\",$i)'><img id='tvi$id$i' src='$open'/></a>"; | ||
+ | $tree .= "<tr$show class='tree-row' depth='$depth' id='tvr$id$i'>" | ||
+ | . "<td valign='bottom'><img src='$spacer' width='".($depth*$width)."' height='1'/>$open" | ||
+ | . " <img id='tvf$id$i' src='$icon'/> $item</td></tr>\n"; | ||
} | } | ||
− | |||
− | |||
− | |||
− | |||
− | |||
− | |||
− | |||
− | |||
− | |||
− | |||
− | |||
− | |||
− | |||
} | } | ||
$tree = "<table class='tree-view' id='tv$id' title='Extension:Treeview (ver $ver)'>\n$tree</table>"; | $tree = "<table class='tree-view' id='tv$id' title='Extension:Treeview (ver $ver)'>\n$tree</table>"; | ||
Line 118: | Line 123: | ||
return $tree; | return $tree; | ||
} | } | ||
− | + | ||
# Add the javascript to the output object if not added yet and there is at least one tree | # Add the javascript to the output object if not added yet and there is at least one tree | ||
function addJS() { | function addJS() { | ||
Line 157: | Line 162: | ||
} | } | ||
} | } | ||
− | + | ||
# Called from $wgExtensionFunctions array when initialising extensions | # Called from $wgExtensionFunctions array when initialising extensions | ||
function wfSetupTreeView4() { | function wfSetupTreeView4() { | ||
Line 164: | Line 169: | ||
$wgTreeView4->addJS(); # Make code unconditional for now due to parser caching | $wgTreeView4->addJS(); # Make code unconditional for now due to parser caching | ||
} | } | ||
− | + | ||
# Needed in MediaWiki >1.8.0 for magic word hooks to work properly | # Needed in MediaWiki >1.8.0 for magic word hooks to work properly | ||
function wfTreeView4LanguageGetMagic(&$magicWords,$langCode = 0) { | function wfTreeView4LanguageGetMagic(&$magicWords,$langCode = 0) { |
Revision as of 06:07, 7 September 2007
<?php
- MediaWiki Treeview ExtensionTemplate:Php
- - See http://www.mediawiki.org/wiki/Extension:Tree_view for installation and usage details
- - Licenced under LGPL (http://www.gnu.org/copyleft/lesser.html)
- - Author: http://www.organicdesign.co.nz/nad
if (!defined('MEDIAWIKI')) die('Not an entry point.');
define('TREEVIEW4_VERSION','4.0.1, 2007-09-07');
- Set any unset images to default titles
if (!is_array($wgTreeViewImages)) $wgTreeViewImages = array(); if (!isset($wgTreeViewImages['plus'])) $wgTreeViewImages['plus'] = 'Plus.gif'; if (!isset($wgTreeViewImages['minus'])) $wgTreeViewImages['minus'] = 'Minus.gif'; if (!isset($wgTreeViewImages['opened'])) $wgTreeViewImages['opened'] = 'Folder_opn_sml_yel.gif'; if (!isset($wgTreeViewImages['closed'])) $wgTreeViewImages['closed'] = 'Folder_sml_yel.gif'; if (!isset($wgTreeViewImages['doc'])) $wgTreeViewImages['doc'] = 'Doc-icon.gif'; if (!isset($wgTreeViewImages['spacer'])) $wgTreeViewImages['spacer'] = 'Blank.gif';
- Keep track of JavaScript added to page to avoid doubleups
if (!isset($wgJS)) $wgJS = array();
$wgTreeView4Magic = "tree"; # the parser-function name for trees $wgExtensionFunctions[] = 'wfSetupTreeView4'; $wgHooks['LanguageGetMagic'][] = 'wfTreeView4LanguageGetMagic';
$wgExtensionCredits['parserhook'][] = array( 'name' => 'Treeview4', 'author' => 'User:Nad', 'url' => 'http://www.mediawiki.org/wiki/Extension:Treeview', 'description' => 'Allows dynamic tree-views to be made with bullet-list syntax', 'version' => TREEVIEW4_VERSION );
class TreeView4 {
var $version = TREEVIEW4_VERSION; var $width = 16; var $uniq = array(); var $js = 0;
# Constructor function TreeView4($magic) { global $wgParser,$wgHooks,$wgTreeViewImages;
# Set hooks $wgParser->setFunctionHook($magic,array($this,'Tree'));
# Convert image titles to file paths and obtain pixel width of items foreach ($wgTreeViewImages as $k => $v) { $title = Title::newFromText($v,NS_IMAGE); $image = Image::newFromTitle($title); if ($image && $image->exists()) { $wgTreeViewImages[$k] = $image->getURL(); if ($k == 'plus') $this->width = $image->getWidth(); } } }
# Restructure recursive trees into a single bullet list surrounded by internal treeview4 tags function Tree(&$parser) { global $wgParser; $id = uniqid(); $args = "id='$id'"; foreach (func_get_args() as $arg) if (!is_object($arg)) { if (preg_match('/^(.+?=.+)$/',$arg,$m)) $args .= " $arg"; else $text = $arg; }
# Back magic $text = $this->uniq[$id] = preg_replace( '/^\\*(\\**)\\s*(.UNIQ.+?-treeview4(.+?)-.+?-QINU)/me', '$this->uniq["$3"] ? preg_replace("/^(\\*+)/m","$1\$1",$this->uniq["$3"]) : "$1$2"', $text );
# And a little more voodoo here $wgParser->setHook("treeview4$id",array($this,'treeview')); return "<treeview4$id $args>$text</treeview4$id>"; }
# Convert a bullet list into a treeview function treeview($text,$argv,&$parser) { global $wgTreeViewImages;
$id = $argv['id']; $ver = $this->version; $width = $this->width; $plus = $wgTreeViewImages['plus']; $minus = $wgTreeViewImages['minus']; $opened = $wgTreeViewImages['opened']; $closed = $wgTreeViewImages['closed']; $spacer = $wgTreeViewImages['spacer']; $doc = $wgTreeViewImages['doc']; $default = isset($argv['openlevels']) ? $argv['openlevels']+1 : 1; $rows = array();
$tree = ;
# Protect the asterisk structure and wiki-parse the bullet tree $text = preg_replace_callback( "/^(\\*+)\\s*(.+)\\s*$/m", create_function('$m','return "@@@".strlen($m[1])."@$m[2]@@@";'), $text ); $out = $parser->parse($text,$parser->mTitle,$parser->mOptions,false,false); $text = $out->getText();
# Convert the processed text into a tree table if (preg_match_all('/@@@([0-9]+)@(.+?)@@@/',$text,$rows,PREG_SET_ORDER)) { foreach ($rows as $i => $row) { list(,$depth,$item) = $row; $next = $i < count($rows) ? $rows[$i+1][1] : 0; $show = $depth > $default ? ' style="display:none"' : ; if ($depth >= $default) { $open = $plus; $icon = $closed; } else { $open = $minus; $icon = $opened; } if ($depth >= $next) { $open = "<img src='$spacer' width='$width'/>"; $icon = $doc; } else $open = "<a href='javascript:toggleTreeviewItem(\"$id\",$i)'><img id='tvi$id$i' src='$open'/></a>"; $tree .= "<tr$show class='tree-row' depth='$depth' id='tvr$id$i'>"
. "<img src='$spacer' width='".($depth*$width)."' height='1'/>$open" . " <img id='tvf$id$i' src='$icon'/> $item\n"; } } $tree = "
\n$tree";
return $tree; }
# Add the javascript to the output object if not added yet and there is at least one tree function addJS() { global $wgOut,$wgTreeViewImages,$wgJS,$wgJsMimeType; if (isset($wgJS['TreeView4'])) return; $wgJS['TreeView4'] = true; $plus = $wgTreeViewImages['plus']; $minus = $wgTreeViewImages['minus']; $opened = $wgTreeViewImages['opened']; $closed = $wgTreeViewImages['closed']; $spacer = $wgTreeViewImages['spacer']; $doc = $wgTreeViewImages['doc']; $wgOut->addScript('<script type="'.$wgJsMimeType.'"> function toggleTreeviewItem(id,row) { var plus = "'.$plus.'"; var minus = "'.$minus.'"; var opened = "'.$opened.'"; var closed = "'.$closed.'"; var doc = "'.$doc.'"; var item = document.getElementById("tvr"+id+row); var next = document.getElementById("tvr"+id+(row+1)); var depth = 0+item.getAttribute("depth"); var close = next.style.display != "none"; var img = document.getElementById("tvi"+id+row); var fld = document.getElementById("tvf"+id+row); fld.setAttribute("src",close ? closed : opened); img.setAttribute("src",close ? plus : minus); while ((item = document.getElementById("tvr"+id+(++row))) && (0+item.getAttribute("depth") > depth)) { if (close) item.style.display = "none"; else if (depth == item.getAttribute("depth")-1) { item.style.display = ""; if (img = document.getElementById("tvi"+id+row)) img.setAttribute("src",plus); if (fld = document.getElementById("tvf"+id+row)) if (fld.getAttribute("src") == opened) fld.setAttribute("src",closed); } } }</script>'); } }
- Called from $wgExtensionFunctions array when initialising extensions
function wfSetupTreeView4() { global $wgTreeView4,$wgTreeView4Magic; $wgTreeView4 = new TreeView4($wgTreeView4Magic); $wgTreeView4->addJS(); # Make code unconditional for now due to parser caching }
- Needed in MediaWiki >1.8.0 for magic word hooks to work properly
function wfTreeView4LanguageGetMagic(&$magicWords,$langCode = 0) { global $wgTreeView4Magic; $magicWords[$wgTreeView4Magic] = array(0,$wgTreeView4Magic); return true; } ?>