4 * Copyright (C) 2003 Sameer D. Sahasrabuddhe
5 * Copyright (C) 2005 $ThePhpWikiProgrammingTeam
6 * Copyright (C) 2008-2010 Marc-Etienne Vargenau, Alcatel-Lucent
8 * This file is part of PhpWiki.
10 * PhpWiki is free software; you can redistribute it and/or modify
11 * it under the terms of the GNU General Public License as published by
12 * the Free Software Foundation; either version 2 of the License, or
13 * (at your option) any later version.
15 * PhpWiki is distributed in the hope that it will be useful,
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 * GNU General Public License for more details.
20 * You should have received a copy of the GNU General Public License
21 * along with PhpWiki; if not, write to the Free Software
22 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
26 * Standard Alcatel-Lucent disclaimer for contributing to open source
28 * "The MediawikiTablePlugin ("Contribution") has not been tested and/or
29 * validated for release as or in products, combinations with products or
30 * other commercial use. Any use of the Contribution is entirely made at
31 * the user's own responsibility and the user can not rely on any features,
32 * functionalities or performances Alcatel-Lucent has attributed to the
35 * THE CONTRIBUTION BY ALCATEL-LUCENT IS PROVIDED AS IS, WITHOUT WARRANTY
36 * OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
37 * WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, COMPLIANCE,
38 * NON-INTERFERENCE AND/OR INTERWORKING WITH THE SOFTWARE TO WHICH THE
39 * CONTRIBUTION HAS BEEN MADE, TITLE AND NON-INFRINGEMENT. IN NO EVENT SHALL
40 * ALCATEL-LUCENT BE LIABLE FOR ANY DAMAGES OR OTHER LIABLITY, WHETHER IN
41 * CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
42 * CONTRIBUTION OR THE USE OR OTHER DEALINGS IN THE CONTRIBUTION, WHETHER
43 * TOGETHER WITH THE SOFTWARE TO WHICH THE CONTRIBUTION RELATES OR ON A STAND
48 * MediawikiTablePlugin
49 * A PhpWiki plugin that allows insertion of tables using a Mediawiki-like
52 class WikiPlugin_MediawikiTable
56 return _("MediawikiTable");
59 function getDescription() {
60 return _("Layout tables using a Mediawiki-like markup style.");
63 function getDefaultArguments() {
67 function run($dbi, $argstr, &$request, $basepage) {
68 include_once("lib/BlockParser.php");
69 // MediawikiTablePlugin markup is new.
72 // We allow the compact Mediawiki syntax with:
73 // - multiple cells on the same line (separated by "||"),
74 // - multiple header cells on the same line (separated by "!!").
75 $argstr = str_replace("||", "\n| ", $argstr);
76 $argstr = str_replace("!!", "\n! ", $argstr);
78 $lines = explode("\n", $argstr);
79 $table = HTML::table();
81 // We always generate an Id for the table.
82 // This is convenient for tables of class "sortable".
83 // If user provides an Id, the generated Id will be overwritten below.
84 $table->setAttr("id", GenerateId("MediawikiTable"));
86 if (substr($lines[0],0,2) == "{|") {
88 $lines[0] = substr($lines[0],2);
90 if (($lines[0][0] != '|') and ($lines[0][0] != '!')) {
91 $line = array_shift($lines);
92 $attrs = parse_attributes($line);
93 foreach ($attrs as $key => $value) {
94 if (in_array ($key, array("id", "class", "title", "style",
95 "bgcolor", "frame", "rules", "border",
96 "cellspacing", "cellpadding",
97 "summary", "align", "width"))) {
98 $table->setAttr($key, $value);
103 if (count($lines) == 1) { // empty table, we only have closing "|}" line
104 return HTML::raw('');
107 foreach ($lines as $line){
108 if (substr($line,0,2) == "|}") {
112 if (substr($line,0,2) == "|-") {
115 if (isset($content)) {
116 if (is_numeric(trim($content))) {
117 $cell->pushContent(HTML::p(array('style' => "text-align:right"), trim($content)));
119 $cell->pushContent(TransformText(trim($content), $markup, $basepage));
123 $row->pushContent($cell);
126 if (!empty($row->_content)) {
128 $thead->pushContent($row);
129 $table->pushContent($thead);
131 $tbody = HTML::tbody();
133 $tbody->pushContent($row);
138 $attrs = parse_attributes(substr($line,2));
139 foreach ($attrs as $key => $value) {
140 if (in_array ($key, array("id", "class", "title", "style",
141 "bgcolor", "align", "valign"))) {
142 $row->setAttr($key, $value);
149 if (substr($line,0,2) == "|=") {
150 $line = substr($line,2);
151 $table->setAttr("summary", trim($line));
155 if (substr($line,0,2) == "|+") {
157 $caption = HTML::caption();
158 $line = substr($line,2);
159 $pospipe = strpos($line, "|");
160 $posbracket = strpos($line, "[");
161 if (($pospipe !== false) && (($posbracket === false) || ($posbracket > $pospipe))) {
162 $attrs = parse_attributes(substr($line, 0, $pospipe));
163 foreach ($attrs as $key => $value) {
164 if (in_array ($key, array("id", "class", "title", "style",
166 $caption->setAttr($key, $value);
169 $line=substr($line, $pospipe+1);
172 $caption->pushContent(trim($line));
173 $table->pushContent($caption);
176 if (((substr($line,0,1) == "|") or (substr($line,0,1) == "!")) and isset($row)) {
178 if (isset ($content)) {
179 if (is_numeric(trim($content))) {
180 $cell->pushContent(HTML::p(array('style' => "text-align:right"), trim($content)));
182 $cell->pushContent(TransformText(trim($content), $markup, $basepage));
186 $row->pushContent($cell);
188 if (substr($line,0,1) == "!") {
189 $cell = HTML::th(); // Header
190 $thead = HTML::thead();
193 if (!isset($tbody)) $tbody = HTML::tbody();
195 $line = substr($line, 1);
197 // If there is a "|" in the line, the start of line
198 // (before the "|") is made of attributes.
199 // The end of the line (after the "|") is the cell content
200 // This is not true if the pipe is inside [], {{}} or {{{}}}
202 // The following cases must work:
205 // | class="xxx" | foo
206 // | class="xxx" | [foo|bar]
207 // | {{tmpl|arg=val}}
208 // | {{image.png|alt}}
209 // | {{{ xxx | yyy }}}
210 $pospipe = strpos($line, "|");
211 $posbracket = strpos($line, "[");
212 $poscurly = strpos($line, "{");
213 if (($pospipe !== false) && (($posbracket === false) || ($posbracket > $pospipe)) && (($poscurly === false) || ($poscurly > $pospipe))) {
214 $attrs = parse_attributes(substr($line, 0, $pospipe));
215 foreach ($attrs as $key => $value) {
216 if (in_array ($key, array("id", "class", "title", "style", "scope",
217 "colspan", "rowspan", "width", "height",
218 "bgcolor", "align", "valign"))) {
219 $cell->setAttr($key, $value);
222 $line=substr($line, $pospipe+1);
223 if (is_numeric(trim($line))) {
224 $cell->pushContent(HTML::p(array('style' => "text-align:right"), trim($line)));
226 $cell->pushContent(TransformText(trim($line), $markup, $basepage));
231 if (isset($row) and isset($cell)) {
232 $line = str_replace("?\>", "?>", $line);
233 $line = str_replace("\~", "~", $line);
234 if (empty($content)) $content = '';
235 $content .= $line . "\n";
240 if (isset($content)) {
241 if (is_numeric(trim($content))) {
242 $cell->pushContent(HTML::p(array('style' => "text-align:right"), trim($content)));
244 $cell->pushContent(TransformText(trim($content), $markup, $basepage));
248 $row->pushContent($cell);
250 // If user put and extra "|-" without cells just before "|}"
251 // we ignore it to get valid XHTML code
252 if (!empty($row->_content)) {
253 $tbody->pushContent($row);
255 if (isset($tbody) && !empty($tbody->_content)) {
256 $table->pushContent($tbody);
259 if (isset($table) && !empty($table->_content)) {
262 return HTML::raw('');
271 // c-hanging-comment-ender-p: nil
272 // indent-tabs-mode: nil