# # TWiki WikiClone ($wikiversion has version info) # # Copyright (C) 2002 Peter Thoeny, Peter@Thoeny.com # # This program is free software; you can redistribute it and/or # modify it under the terms of the GNU General Public License # as published by the Free Software Foundation; either version 2 # of the License, or (at your option) any later version. # # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details, published at # http://www.gnu.org/copyleft/gpl.html # # ========================= # # This is the EditTablePlugin used to edit tables in place. # # Each plugin is a package that contains the subs: # # initPlugin ( $topic, $web, $user, $installWeb ) # commonTagsHandler ( $text, $topic, $web ) # startRenderingHandler( $text, $web ) # outsidePREHandler ( $text ) # insidePREHandler ( $text ) # endRenderingHandler ( $text ) # # initPlugin is required, all other are optional. # For increased performance, DISABLE handlers you don't need. # # NOTE: To interact with TWiki use the official TWiki functions # in the &TWiki::Func module. Do not reference any functions or # variables elsewhere in TWiki!! # FIXME: The following TWiki calls used will likely break in a # future TWiki release: # TWiki::Store::readTopic(), TWiki::Store::saveTopic() # TWiki::Store::lockTopic(), TWiki::Store::topicIsLockedBy() # ========================= package TWiki::Plugins::EditTablePlugin; # ========================= use vars qw( $web $topic $user $installWeb $VERSION $debug $query $renderingWeb $preSp $header $footer @format @formatExpansed $changeRows $helpTopic $nrCols $encodeStart $encodeEnd $table ); $VERSION = '1.001'; $encodeStart = "--EditTableEncodeStart--"; $encodeEnd = "--EditTableEncodeEnd--"; undef $table; # ========================= sub initPlugin { ( $topic, $web, $user, $installWeb ) = @_; # check for Plugins.pm versions if( $TWiki::Plugins::VERSION < 1 ) { &TWiki::Func::writeWarning( "Version mismatch between EditTablePlugin and Plugins.pm" ); return 0; } $query = &TWiki::Func::getCgiQuery(); if( ! $query ) { return 0; } # Get plugin preferences # $doEnable = &TWiki::Func::getPreferencesFlag( "EDITTABLEPLUGIN_ENABLE" ) || ""; # Get plugin debug flag $debug = &TWiki::Func::getPreferencesFlag( "EDITTABLEPLUGIN_DEBUG" ); $renderingWeb = $web; # Plugin correctly initialized &TWiki::Func::writeDebug( "- TWiki::Plugins::EditTablePlugin::initPlugin( $web.$topic ) is OK" ) if $debug; # Initialize $table such that the code will correctly detect when to # read in a topic. undef $table; return 1; } # ========================= sub extractParameters { my( $theArgs, $theHeader, $theFooter, $theFormat, $theChangeRows, $theHelpTopic ) = @_; my $tmp = &TWiki::Func::extractNameValuePair( $theArgs, "header" ); $theHeader = $tmp if( $tmp ); $tmp = &TWiki::Func::extractNameValuePair( $theArgs, "footer" ); $theFooter = $tmp if( $tmp ); $tmp = &TWiki::Func::extractNameValuePair( $theArgs, "format" ); $tmp =~ s/^\s*\|*\s*//o; $tmp =~ s/\s*\|*\s*$//o; $theFormat = $tmp if( $tmp ); $tmp = &TWiki::Func::extractNameValuePair( $theArgs, "changerows" ); $theChangeRows = $tmp if( $tmp ); $tmp = &TWiki::Func::extractNameValuePair( $theArgs, "helptopic" ); $theHelpTopic = $tmp if( $tmp ); return ( $theHeader, $theFooter, $theFormat, $theChangeRows, $theHelpTopic ); } # ========================= sub commonTagsHandler { ### my ( $text, $topic, $web ) = @_; # do not uncomment, use $_[0], $_[1]... instead &TWiki::Func::writeDebug( "- EditTablePlugin::commonTagsHandler( $_[2].$_[1] )" ) if $debug; return unless $_[0] =~ /%EDITTABLE{(.*)}%/os; my $theWeb = $_[2]; my $theTopic = $_[1]; my $result = ""; my $tableNr = 0; my $rowNr = 0; my $enableForm = 0; my $insideTable = 0; my $doEdit = 0; my $cgiRows = -1; foreach( split( /\n/, $_[0] ) ) { if( s/(\s*)%EDITTABLE{(.*)}%/&handleEditTableTag( $theWeb, $1, $2 )/geo ) { $enableForm = 1; $tableNr += 1; my $cgiTableNr = $query->param( 'ettablenr' ) || 0; $cgiRows = $query->param( 'etrows' ) || -1; if( $cgiTableNr == $tableNr ) { if( $query->param( 'etsave' ) ) { # [Save table] button pressed doSaveTable( $theWeb, $theTopic, $tableNr, "" ); # never return return; # in case browser does not redirect } elsif( $query->param( 'etqsave' ) ) { # [Quietsave] button pressed doSaveTable( $theWeb, $theTopic, $tableNr, "on" ); # never return return; # in case browser does not redirect } elsif( $query->param( 'etcancel' ) ) { # [Cancel] button pressed doCancelEdit( $theWeb, $theTopic ); # never return return; # in case browser does not redirect } elsif( $query->param( 'etaddrow' ) ) { # [Add row] button pressed $cgiRows++ if( $cgiRows >= 0 ); $doEdit = doEnableEdit( $theWeb, $theTopic, 0 ); return unless( $doEdit ); } elsif( $query->param( 'etdelrow' ) ) { # [Delete row] button pressed $cgiRows-- if( $cgiRows > 1 ); $doEdit = doEnableEdit( $theWeb, $theTopic, 0 ); return unless( $doEdit ); } elsif( $query->param( 'etedit' ) ) { # [Edit table] button pressed $doEdit = doEnableEdit( $theWeb, $theTopic, 1 ); # never return if locked or no permission return unless( $doEdit ); $cgiRows = -1; # make sure to get the actual number of rows } } } if( $enableForm ) { if( /^(\s*)\|.*\|\s*$/ ) { # found table row $result .= handleTableStart( $theWeb, $theTopic, $tableNr, $doEdit ) unless $insideTable; $insideTable = 1; $rowNr++; if( ( $doEdit ) && ( $cgiRows >= 0 ) && ( $rowNr > $cgiRows ) ) { # deleted row $rowNr--; next; } s/^(\s*)\|(.*)/&handleTableRow( $1, $2, $tableNr, $cgiRows, $rowNr, $doEdit, 0 )/eo; } elsif( $insideTable ) { # end of table $insideTable = 0; if( ( $doEdit ) && ( $cgiRows >= 0 ) && ( $rowNr < $cgiRows ) ) { while( $rowNr < $cgiRows ) { $rowNr++; $result .= handleTableRow( $theSp, "", $tableNr, $cgiRows, $rowNr, $doEdit, 0 ) . "\n"; } } $result .= handleTableEnd( $theWeb, $rowNr, $doEdit ); $enableForm = 0; $doEdit = 0; $rowNr = 0; } if( /^\s*$/ ) { # empty line if( $enableForm ) { # empty %EDITTABLE%, so create a default table $result .= handleTableStart( $theWeb, $theTopic, $tableNr, $doEdit ); $rowNr = 0; if( $doEdit ) { if( $header ) { $rowNr++; $result .= handleTableRow( $preSp, "", $tableNr, $cgiRows, $rowNr, $doEdit, 0 ) . "\n"; } do { $rowNr++; $result .= handleTableRow( $preSp, "", $tableNr, $cgiRows, $rowNr, $doEdit, 0 ) . "\n"; } while( $rowNr < $cgiRows ); } $result .= handleTableEnd( $theWeb, $rowNr, $doEdit ); $enableForm = 0; } $doEdit = 0; $rowNr = 0; } } $result .= "$_\n"; } $_[0] = $result; } # ========================= sub DISABLE_startRenderingHandler { ### my ( $text, $web ) = @_; # do not uncomment, use $_[0], $_[1] instead &TWiki::Func::writeDebug( "- EditTablePlugin::startRenderingHandler( $_[1] )" ) if $debug; # This handler is called by getRenderedVersion just before the line loop $renderingWeb = $_[1]; } # ========================= sub DISABLE_outsidePREHandler { ### my ( $text ) = @_; # do not uncomment, use $_[0] instead &TWiki::Func::writeDebug( "- EditTablePlugin::outsidePREHandler( $renderingWeb.$topic )" ) if $debug; # This handler is called by getRenderedVersion, in loop outside of
tag
# This is the place to define customized rendering rules
# do custom extension rule, like for example:
# $_[0] =~ s/old/new/go;
}
# =========================
sub DISABLE_insidePREHandler
{
### my ( $text ) = @_; # do not uncomment, use $_[0] instead
&TWiki::Func::writeDebug( "- EditTablePlugin::insidePREHandler( $web.$topic )" ) if $debug;
# This handler is called by getRenderedVersion, in loop inside of tag
# This is the place to define customized rendering rules
# do custom extension rule, like for example:
# $_[0] =~ s/old/new/go;
}
# =========================
sub endRenderingHandler
{
### my ( $text ) = @_; # do not uncomment, use $_[0] instead
&TWiki::Func::writeDebug( "- EditTablePlugin::endRenderingHandler( $web.$topic )" ) if $debug;
# This handler is called by getRenderedVersion just after the line loop
return unless $_[0] =~ /$encodeStart/os;
$_[0] =~ s/$encodeStart(.*?)$encodeEnd/&decodeValue($1)/geos;
}
# =========================
sub encodeValue
{
my( $theText ) = @_;
# WindRiver specific hack to remove SprPlugin rendering
$theText =~ s///gos;
$tFormat = &TWiki::Func::expandCommonVariables( $tFormat, $theTopic, $theWeb );
@formatExpansed = split( /\s*\|\s*/, $tFormat );
$formatExpansed[0] = "text,16" unless @formatExpansed;
# FIXME: No handling yet of footer
return "$preSp";
}
# =========================
sub handleTableStart
{
my( $theWeb, $theTopic, $theTableNr, $doEdit ) = @_;
my $viewUrl = &TWiki::Func::getScriptUrl( $theWeb, $theTopic, "viewauth" ) . "\#edittable$theTableNr";
my $text = "";
$text .= "$preSp\n" if $doEdit;
$text .= "$preSp\n";
$text .= "$preSp\n";
$text .= "$preSp \n" if $doEdit;
return $text;
}
# =========================
sub inputElement
{
my ( $theTableNr, $theRowNr, $theCol, $theName, $theValue ) = @_;
$theValue = "" if( $theValue eq " " );
my $text = "";
my $i = @format - 1;
$i = $theCol if( $theCol < $i );
my @bits = split( /,\s*/, $format[$i] );
my @bitsExpansed = split( /,\s*/, $formatExpansed[$i] );
my $type = "text";
$type = $bits[0] if @bits > 0;
my $size = 0;
$size = $bits[1] if @bits > 1;
my $val = "";
my $valExpansed = "";
my $sel = "";
my $style = " style='background:#e8e8e8'" if ($theRowNr % 2);
if( $type eq "select" ) {
$size = 1 if $size < 1;
$text = "";
} elsif( $type eq "row" ) {
$size = $size + $theRowNr;
$text = "$size";
} elsif( $type eq "label" ) {
# show label text as is, and add a hidden field with value
$text = $theValue;
# To optimize things, only in the case where a read-only column is
# being processed (inside of this unless() statement) do we actually
# go out and read the original topic. Thus the reason for the
# following unless() so we only read the topic the first time through.
unless( defined $table ) {
# To deal with the situation where TWiki variables, like
# %CALC%, have already been processed and end up getting saved
# in the table that way (processed), we need to read in the
# topic page in raw format
my( $meta, $topicContents ) = TWiki::Func::readTopic( $web, $topic );
$table = TWiki::Plugins::Table->new( $topicContents );
}
my $cell = $table->getCell( $theTableNr, $theRowNr - 1, $theCol );
$theValue = $cell if( defined $cell ); # original value from file
$theValue = $encodeStart . encodeValue( $theValue ) . $encodeEnd if $theValue;
$text .= "";
} elsif( $type eq "textarea" ) {
my ($rows, $cols) = split( /x/, $size );
$rows = 3 if $rows < 1;
$cols = 30 if $cols < 1;
$theValue = $encodeStart . encodeValue( $theValue ) . $encodeEnd if $theValue;
$text .= "";
} else { # if( $type eq "text")
$size = 16 if $size < 1;
$theValue = $encodeStart . encodeValue( $theValue ) . $encodeEnd if $theValue;
$text = "";
}
return $text;
}
# =========================
sub handleTableRow
{
my ( $thePre, $theRow, $theTableNr, $theRowMax, $theRowNr, $doEdit, $doSave ) = @_;
my $text = "$thePre\|";
if( $doEdit ) {
$theRow =~ s/\|\s*$//o;
@cells = split( /\|/, $theRow );
my $tmp = @cells;
$nrCols = $tmp if( $tmp > $nrCols ); # expand number of cols
my $val = "";
my $cell = "";
my $cellDefined = 0;
my $col = 0;
while( $col < $nrCols ) {
$col += 1;
$cellDefined = 0;
$val = $query->param( "etcell${theRowNr}x$col" );
if( defined $val ) {
# change any new line character sequences to
$val =~ s/(\n\r?)|(\r\n?)+/
/gos;
$cellDefined = 1;
$cell = $val;
} elsif( $col <= @cells ) {
$cell = $cells[$col-1];
$cellDefined = 1 if( length( $cell ) > 0 );
$cell =~ s/^\s//o;
$cell =~ s/\s$//o;
} else {
$cell = "";
}
if( ( $theRowNr <= 1 ) && ( $header ) ) {
unless( $cell ) {
if( $header =~ /^on$/i ) {
if( ( @format >= $col ) && ( $format[$col-1] =~ /(.*?)\,/ ) ) {
$cell = $1;
}
$cell = "text" unless $cell;
$cell = "*$cell*";
} else {
my @hCells = split( /\|/, $header );
$cell = $hCells[$col-1] if( @hCells >= $col );
$cell = "*text*" unless $cell;
}
}
$text .= "$cell\|";
} elsif( $doSave ) {
$text .= " $cell \|";
} else {
if( ( ! $cellDefined ) && ( @format >= $col )
&& ( $format[$col-1] =~ /^\s*(.*?)\,\s*(.*?)\,\s*(.*?)\s*$/ ) ) {
# default value of "| text, 20, a, b,c |" cell is "a, b, c"
# default value of "| select, 1, a, b, c |" cell is "a"
$val = $1; # type
$cell = $3 || "";
$cell =~ s/\,.*$//o if( $val eq "select" );
}
$text .= inputElement( $theTableNr, $theRowNr, $col-1, "etcell${theRowNr}x$col", $cell ) . " \|";
}
}
} else {
$text .= "$theRow";
}
return $text;
}
# =========================
sub doSaveTable
{
my ( $theWeb, $theTopic, $theTableNr, $quiet ) = @_;
&TWiki::Func::writeDebug( "- EditTablePlugin::doSaveTable( $theWeb, $theTopic, $theTableNr, $quiet )" ) if $debug;
my( $meta, $text ) = &TWiki::Store::readTopic( $theWeb, $theTopic );
my $cgiRows = $query->param( 'etrows' ) || 1;
my $tableNr = 0;
my $rowNr = 0;
my $insideTable = 0;
my $doSave = 0;
my $result = "";
foreach( split( /\n/, $text ) ) {
if( /%EDITTABLE{(.*)}%/o ) {
$tableNr += 1;
if( $tableNr == $theTableNr ) {
$doSave = 1;
}
}
if( $doSave ) {
if( /^(\s*)\|.*\|\s*$/ ) {
$insideTable = 1;
$rowNr++;
if( $rowNr > $cgiRows ) {
# deleted row
$rowNr--;
next;
}
s/^(\s*)\|(.*)/&handleTableRow( $1, $2, $tableNr, $cgiRows, $rowNr, 1, 1 )/eo;
} elsif( $insideTable ) {
$insideTable = 0;
if( $rowNr < $cgiRows ) {
while( $rowNr < $cgiRows ) {
$rowNr++;
$result .= handleTableRow( $preSp, "", $tableNr, $cgiRows, $rowNr, 1, 1 ) . "\n";
}
}
$doSave = 0;
$rowNr = 0;
}
if( /^\s*$/ ) { # empty line
if( $doSave ) {
# empty %EDITTABLE%, so create a default table
$rowNr = 0;
if( $header ) {
$rowNr++;
$result .= handleTableRow( $preSp, "", $tableNr, $cgiRows, $rowNr,1 , 1 ) . "\n";
}
while( $rowNr < $cgiRows ) {
$rowNr++;
$result .= handleTableRow( $preSp, "", $tableNr, $cgiRows, $rowNr, 1, 1 ) . "\n";
}
}
$doSave = 0;
$rowNr = 0;
}
}
$result .= "$_\n";
}
my $error = &TWiki::Store::saveTopic( $theWeb, $theTopic, $result, $meta, "", "on", $quiet );
&TWiki::Store::lockTopic( $theTopic, "on" );
my $url = &TWiki::Func::getViewUrl( $theWeb, $theTopic );
if( $error ) {
$url = &TWiki::Func::getOopsUrl( $theWeb, $theTopic, "oopssaveerr", $error );
}
&TWiki::Func::redirectCgiQuery( $query, $url );
}
# =========================
sub doCancelEdit
{
my ( $theWeb, $theTopic ) = @_;
&TWiki::Func::writeDebug( "- EditTablePlugin::doCancelEdit( $theWeb, $theTopic )" ) if $debug;
&TWiki::Store::lockTopic( $theTopic, "on" );
&TWiki::Func::redirectCgiQuery( $query, &TWiki::Func::getViewUrl( $theWeb, $theTopic ) );
}
# =========================
sub doEnableEdit
{
my ( $theWeb, $theTopic, $doCheckIfLocked ) = @_;
&TWiki::Func::writeDebug( "- EditTablePlugin::doEnableEdit( $theWeb, $theTopic )" ) if $debug;
my $wikiUserName = &TWiki::Func::getWikiUserName();
if( ! &TWiki::Func::checkAccessPermission( "change", $wikiUserName, "", $theTopic, $theWeb ) ) {
# user has not permission to change the topic
my $url = &TWiki::Func::getOopsUrl( $theWeb, $theTopic, "oopsaccesschange" );
&TWiki::Func::redirectCgiQuery( $query, $url );
return 0;
}
my( $lockUser, $lockTime ) = &TWiki::Store::topicIsLockedBy( $theWeb, $theTopic );
if( ( $doCheckIfLocked ) && ( $lockUser ) ) {
# warn user that other person is editing this topic
$lockUser = &TWiki::Func::userToWikiName( $lockUser );
use integer;
$lockTime = ( $lockTime / 60 ) + 1; # convert to minutes
my $editLock = $TWiki::editLockTime / 60;
my $url = &TWiki::Func::getOopsUrl( $theWeb, $theTopic, "oopslocked",
$lockUser, $editLock, $lockTime );
&TWiki::Func::redirectCgiQuery( $query, $url );
return 0;
}
&TWiki::Store::lockTopic( $theTopic );
return 1;
}
# =========================
# The following code is copied from the ChartPlugin Table object.
package TWiki::Plugins::Table;
sub new
{
my ($class, $topicContents) = @_;
my $this = {};
bless $this, $class;
$this->_parseOutTables($topicContents);
return $this;
}
# The guts of this routine was initially copied from SpreadSheetPlugin.pm
# and were used in the ChartPlugin Table object which this was copied from,
# but this has been modified to support the functionality needed by the
# EditTablePlugin. One major change is to only count and save tables
# following an %EDITTABLE{.*}% tag.
#
# This routine basically returns an array of hashes where each hash
# contains the information for a single table. Thus the first hash in the
# array represents the first table found on the topic page, the second hash
# in the array represents the second table found on the topic page, etc.
sub _parseOutTables
{
my ($this, $topic) = @_;
my $tableNum = 1; # Table number (only count tables with EDITTABLE tag)
my @tableMatrix; # Currently parsed table.
my $inEditTable = 0; # Flag to keep track if in an EDITTABLE table
my $result = "";
my $insidePRE = 0;
my $insideTABLE = 0;
my $line = "";
my @row = ();
$topic =~ s/\r//go;
$topic =~ s/\\\n//go; # Join lines ending in "\"
foreach( split( /\n/, $topic ) ) {
# change state:
m||i && ( $insidePRE = 1 );
m||i && ( $insidePRE = 1 );
m| |i && ( $insidePRE = 0 );
m||i && ( $insidePRE = 0 );
if( ! $insidePRE ) {
$inEditTable = 1 if (/%EDITTABLE{(.*)}%/);
if ($inEditTable) {
if( /^\s*\|.*\|\s*$/ ) {
# inside | table |
$insideTABLE = 1;
$line = $_;
$line =~ s/^(\s*\|)(.*)\|\s*$/$2/o; # Remove starting '|'
@row = split( /\|/o, $line, -1 );
_trim(\@row);
push (@tableMatrix, [ @row ]);
} else {
# outside | table |
if( $insideTABLE ) {
# We were inside a table and are now outside of it so
# save the table info into the Table object.
$insideTABLE = 0;
$inEditTable = 0;
if (@tableMatrix != 0) {
# Save the table via its table number
$$this{"TABLE_$tableNum"} = [@tableMatrix];
$tableNum++;
}
undef @tableMatrix; # reset table matrix
}
}
}
}
$result .= "$_\n";
}
$$this{NUM_TABLES} = $tableNum;
}
# Trim any leading and trailing white space and/or '*'.
sub _trim
{
my ($totrim) = @_;
for my $element (@$totrim) {
$element =~ s/^[\s\*]+//; # Strip of leading white/*
$element =~ s/[\s\*]+$//; # Strip of trailing white/*
}
}
# Return the contents of the specified cell
sub getCell
{
my ( $this, $tableNum, $row, $column ) = @_;
my @selectedTable = $this->getTable( $tableNum );
my $value = $selectedTable[$row][$column];
return $value;
}
sub getTable
{
my ($this, $tableNumber) = @_;
my $table = $$this{"TABLE_$tableNumber"};
return @$table if defined( $table );
return ();
}
1;