2011-08-21 13:55:56 +02:00
|
|
|
<?php
|
|
|
|
/**
|
|
|
|
* EGroupware - eTemplate serverside grid widget
|
|
|
|
*
|
|
|
|
* @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License
|
2016-03-19 14:06:07 +01:00
|
|
|
* @package api
|
|
|
|
* @subpackage etemplate
|
2011-08-21 13:55:56 +02:00
|
|
|
* @link http://www.egroupware.org
|
|
|
|
* @author Ralf Becker <RalfBecker@outdoor-training.de>
|
2016-03-19 14:06:07 +01:00
|
|
|
* @copyright 2002-16 by RalfBecker@outdoor-training.de
|
2011-08-21 13:55:56 +02:00
|
|
|
* @version $Id$
|
|
|
|
*/
|
|
|
|
|
2016-03-19 14:06:07 +01:00
|
|
|
namespace EGroupware\Api\Etemplate\Widget;
|
|
|
|
|
|
|
|
use EGroupware\Api\Etemplate;
|
|
|
|
|
2011-08-21 13:55:56 +02:00
|
|
|
/**
|
2011-08-21 14:43:02 +02:00
|
|
|
* eTemplate grid widget incl. row(s) and column(s)
|
2012-05-03 16:17:47 +02:00
|
|
|
*
|
|
|
|
* Example of a grid containing 3 columns and 2 rows (last one autorepeated)
|
|
|
|
*
|
|
|
|
* <grid>
|
|
|
|
* <columns>
|
|
|
|
* <column>
|
|
|
|
* <column width="75%">
|
|
|
|
* <column disabled="@no_actions">
|
|
|
|
* </columns>
|
|
|
|
* <rows>
|
|
|
|
* <row>
|
|
|
|
* <description value="ID" />
|
|
|
|
* <description value="Text" />
|
|
|
|
* <description value="Action" />
|
|
|
|
* <row>
|
|
|
|
* <row>
|
|
|
|
* <description id="${row}[id]" />
|
|
|
|
* <textbox id="${row}[text]" />
|
|
|
|
* <button id="delete[$row_cont[id]]" value="Delete" />
|
|
|
|
* </row>
|
|
|
|
* </rows>
|
|
|
|
* </grid>
|
2011-08-21 13:55:56 +02:00
|
|
|
*/
|
2016-03-19 14:06:07 +01:00
|
|
|
class Grid extends Box
|
2011-08-21 13:55:56 +02:00
|
|
|
{
|
|
|
|
/**
|
|
|
|
* (Array of) comma-separated list of legacy options to automatically replace when parsing with set_attrs
|
|
|
|
*
|
2016-03-19 14:06:07 +01:00
|
|
|
* Not used for grid, just need to be set to unset array of extended Box
|
2011-08-21 13:55:56 +02:00
|
|
|
*
|
|
|
|
* @var string|array
|
|
|
|
*/
|
|
|
|
protected $legacy_options = null;
|
2011-08-21 14:43:02 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Run a given method on all children
|
|
|
|
*
|
|
|
|
* Reimplemented to implement column based disabling:
|
2012-05-03 16:17:47 +02:00
|
|
|
* - added 5th var-parameter to hold key of disabled column
|
2011-08-21 14:43:02 +02:00
|
|
|
* - if a column is disabled run returns false and key get added by columns run to $columns_disabled
|
|
|
|
* - row run method checks now for each child (arbitrary widget) if it the column's key is included in $columns_disabled
|
|
|
|
* - as a grid can contain other grid's as direct child, we have to backup and initialise $columns_disabled in grid run!
|
|
|
|
*
|
2019-03-19 15:34:44 +01:00
|
|
|
* @param string|callable $method_name or function($cname, $expand, $widget)
|
2016-03-19 14:06:07 +01:00
|
|
|
* @param array $params =array('') parameter(s) first parameter has to be the cname, second $expand!
|
|
|
|
* @param boolean $respect_disabled =false false (default): ignore disabled, true: method is NOT run for disabled widgets AND their children
|
2011-08-21 14:43:02 +02:00
|
|
|
* @param array $columns_disabled=array() disabled columns
|
|
|
|
*/
|
|
|
|
public function run($method_name, $params=array(''), $respect_disabled=false, &$columns_disabled=array())
|
|
|
|
{
|
2012-05-03 16:17:47 +02:00
|
|
|
// maintain $expand array name-expansion
|
|
|
|
$cname =& $params[0];
|
|
|
|
$expand =& $params[1];
|
|
|
|
$old_cname = $params[0];
|
2013-11-19 00:11:27 +01:00
|
|
|
$old_expand = $params[1];
|
2014-06-09 21:24:48 +02:00
|
|
|
|
2011-08-21 14:43:02 +02:00
|
|
|
// as a grid can contain other grid's as direct child, we have to backup and initialise $columns_disabled
|
|
|
|
if ($this->type == 'grid')
|
|
|
|
{
|
|
|
|
$backup_columns_disabled = $columns_disabled;
|
|
|
|
$columns_disabled = array();
|
|
|
|
}
|
2012-05-03 16:17:47 +02:00
|
|
|
|
2021-10-05 14:11:11 +02:00
|
|
|
if ($respect_disabled && isset($this->attrs['disabled']) && self::check_disabled($this->attrs['disabled'], $expand))
|
2011-08-21 14:43:02 +02:00
|
|
|
{
|
2013-10-24 19:05:22 +02:00
|
|
|
//error_log(__METHOD__."('$method_name', ".array2string($params).', '.array2string($respect_disabled).") $this disabled='{$this->attrs['disabled']}'=".array2string($disabled).": NOT running");
|
2013-11-19 00:11:27 +01:00
|
|
|
$params[0] = $old_cname;
|
|
|
|
$params[1] = $old_expand;
|
2013-08-23 15:00:13 +02:00
|
|
|
return false; // return
|
2011-08-21 14:43:02 +02:00
|
|
|
}
|
2016-03-19 14:06:07 +01:00
|
|
|
|
2021-10-13 23:08:16 +02:00
|
|
|
if($this->id && $this->type !== 'row')
|
|
|
|
{
|
|
|
|
$cname = self::form_name($cname, $this->id, $expand);
|
|
|
|
}
|
2021-10-21 10:39:30 +02:00
|
|
|
if ($cname && (empty($expand['cname']) || $expand['cname'] !== $cname))
|
2014-06-09 21:24:48 +02:00
|
|
|
{
|
|
|
|
$expand['cont'] =& self::get_array(self::$request->content, $cname);
|
|
|
|
$expand['cname'] = $cname;
|
|
|
|
}
|
|
|
|
|
2019-07-11 19:25:30 +02:00
|
|
|
if (is_string($method_name) && method_exists($this, $method_name))
|
2011-08-21 14:43:02 +02:00
|
|
|
{
|
|
|
|
call_user_func_array(array($this, $method_name), $params);
|
|
|
|
}
|
2019-03-19 15:34:44 +01:00
|
|
|
// allow calling with a function or closure --> call it with widget as first param
|
|
|
|
elseif (is_callable($method_name))
|
|
|
|
{
|
|
|
|
$params[2] = $this;
|
|
|
|
call_user_func_array($method_name, $params);
|
|
|
|
}
|
2012-05-03 16:17:47 +02:00
|
|
|
//foreach($this->children as $n => $child)
|
2013-07-17 11:38:37 +02:00
|
|
|
$repeat_child = null;
|
2012-05-03 16:17:47 +02:00
|
|
|
for($n = 0; ; ++$n)
|
2011-08-21 14:43:02 +02:00
|
|
|
{
|
2012-05-03 16:17:47 +02:00
|
|
|
// maintain $expand array name-expansion
|
|
|
|
switch($this->type)
|
|
|
|
{
|
|
|
|
case 'rows':
|
|
|
|
$expand['row'] = $n;
|
|
|
|
break;
|
|
|
|
case 'columns':
|
|
|
|
$expand['c'] = $n;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (isset($this->children[$n]))
|
|
|
|
{
|
|
|
|
$child = $this->children[$n];
|
2013-07-17 11:38:37 +02:00
|
|
|
if($this->type == 'rows' || $this->type == 'columns')
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* We store a clone of the repeated child, because at the end
|
|
|
|
* of this loop the function $method_name is run on $child.
|
|
|
|
* We want to run the function again each repeat, on an unmodified
|
|
|
|
* row / column.
|
|
|
|
*/
|
|
|
|
$repeat_child = clone $child;
|
|
|
|
}
|
2012-05-03 16:17:47 +02:00
|
|
|
}
|
|
|
|
// check if we need to autorepeat last row ($child)
|
|
|
|
elseif (isset($child) && ($this->type == 'rows' || $this->type == 'columns') && $child->need_autorepeat($cname, $expand))
|
|
|
|
{
|
|
|
|
// not breaking repeats last row/column ($child)
|
2013-07-17 11:38:37 +02:00
|
|
|
// Clone the repeating child, to avoid modifying it
|
|
|
|
$child = clone $repeat_child;
|
2012-05-03 16:17:47 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
2011-08-21 16:02:31 +02:00
|
|
|
if ($this->type == 'row')
|
2011-08-21 14:43:02 +02:00
|
|
|
{
|
|
|
|
if (in_array($n, $columns_disabled))
|
|
|
|
{
|
2013-10-24 19:05:22 +02:00
|
|
|
//error_log(__METHOD__."('$method_name', ".array2string($params).', '.array2string($respect_disabled).") $this column $n is disabled: NOT running");
|
2011-08-21 14:43:02 +02:00
|
|
|
continue; // do NOT run $method_name on disabled columns
|
|
|
|
}
|
|
|
|
}
|
2013-02-04 13:35:30 +01:00
|
|
|
//error_log('Running ' . $method_name . ' on child ' . $n . '(' . $child . ') ['.$expand['row'] . ','.$expand['c'] . ']');
|
2011-08-21 14:43:02 +02:00
|
|
|
$disabled = $child->run($method_name, $params, $respect_disabled, $columns_disabled) === false;
|
|
|
|
|
|
|
|
if ($this->type == 'columns' && $disabled)
|
|
|
|
{
|
|
|
|
$columns_disabled[] = $n; // mark column as disabled
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ($this->type == 'grid')
|
|
|
|
{
|
|
|
|
$columns_disabled = $backup_columns_disabled;
|
|
|
|
}
|
2012-05-03 16:17:47 +02:00
|
|
|
$params[0] = $old_cname;
|
2013-11-19 00:11:27 +01:00
|
|
|
$params[1] = $old_expand;
|
2012-05-03 16:17:47 +02:00
|
|
|
|
2011-08-21 14:43:02 +02:00
|
|
|
return true;
|
|
|
|
}
|
2012-05-03 16:17:47 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Check if a row or column needs autorepeating, because still content left
|
|
|
|
*
|
|
|
|
* We only check direct children and grandchildren of first child (eg. box in first column)!
|
|
|
|
*
|
|
|
|
* @param string $cname
|
|
|
|
* @param array $expand
|
|
|
|
*/
|
|
|
|
private function need_autorepeat($cname, array $expand)
|
|
|
|
{
|
|
|
|
// check id's of children
|
|
|
|
foreach($this->children as $n => $direct_child)
|
|
|
|
{
|
|
|
|
foreach(array_merge(array($direct_child), $n ? array() : $direct_child->children) as $child)
|
|
|
|
{
|
|
|
|
$pat = $child->id;
|
2023-06-07 21:50:08 +02:00
|
|
|
while($pat && ($patstr = strstr($pat, '$')))
|
2012-05-03 16:17:47 +02:00
|
|
|
{
|
2023-06-07 21:50:08 +02:00
|
|
|
$pat = substr($patstr, $patstr[1] == '{' ? 2 : 1);
|
2012-05-03 16:17:47 +02:00
|
|
|
|
|
|
|
switch ($this->type)
|
|
|
|
{
|
|
|
|
case 'column':
|
|
|
|
$Ok = $pat[0] == 'c' && !(substr($pat,0,4) == 'cont' || substr($pat,0,2) == 'c_' ||
|
|
|
|
substr($pat,0,4) == 'col_');
|
|
|
|
break;
|
|
|
|
case 'row':
|
2014-03-28 18:32:32 +01:00
|
|
|
$Ok = $pat[0] == 'r' && !(substr($pat,0,2) == 'r_' ||
|
|
|
|
substr($pat,0,4) == 'row_' && substr($pat,0,8) != 'row_cont');
|
|
|
|
//error_log(__METHOD__."() pat='$pat' --> Ok=".array2string($Ok));
|
2012-05-03 16:17:47 +02:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
2023-02-03 19:31:39 +01:00
|
|
|
if($Ok && ($fname = self::form_name($cname, $child->id, $expand)) &&
|
2014-03-28 18:32:32 +01:00
|
|
|
// need to break if fname ends in [] as get_array() will ignore it and returns whole array
|
|
|
|
// for an id like "run[$row_cont[appname]]"
|
|
|
|
substr($fname, -2) != '[]' &&
|
2023-02-03 19:31:39 +01:00
|
|
|
($value = self::get_array(self::$request->content, $fname, false, true)) !== null) // null = not found (can be false!)
|
2012-05-03 16:17:47 +02:00
|
|
|
{
|
2014-03-28 18:32:32 +01:00
|
|
|
//error_log(__METHOD__."('$cname', ) $this autorepeating row $expand[row] because of $child->id = '$fname' is ".array2string($value));
|
2016-03-19 14:06:07 +01:00
|
|
|
unset($value);
|
2012-05-03 16:17:47 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-03-28 18:32:32 +01:00
|
|
|
//error_log(__METHOD__."('$cname', ) $this NOT autorepeating row $expand[row]");
|
2012-05-03 16:17:47 +02:00
|
|
|
return false;
|
|
|
|
}
|
2011-08-21 13:55:56 +02:00
|
|
|
}
|
2016-03-19 14:06:07 +01:00
|
|
|
Etemplate\Widget::registerWidget(__NAMESPACE__.'\\Grid', array('grid', 'rows', 'row', 'columns', 'column'));
|