You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
889 lines
22 KiB
889 lines
22 KiB
<?php
|
|
/**
|
|
* Part of the Fuel framework.
|
|
*
|
|
* @package Fuel
|
|
* @version 1.8
|
|
* @author Fuel Development Team
|
|
* @license MIT License
|
|
* @copyright 2010 - 2016 Fuel Development Team
|
|
* @link http://fuelphp.com
|
|
*/
|
|
|
|
namespace Fuel\Core;
|
|
|
|
// ------------------------------------------------------------------------
|
|
|
|
/**
|
|
* Fieldset Class
|
|
*
|
|
* Define a set of fields that can be used to generate a form or to validate input.
|
|
*
|
|
* @package Fuel
|
|
* @category Core
|
|
*/
|
|
class Fieldset
|
|
{
|
|
/**
|
|
* @var Fieldset
|
|
*/
|
|
protected static $_instance;
|
|
|
|
/**
|
|
* @var array contains references to all instantiations of Fieldset
|
|
*/
|
|
protected static $_instances = array();
|
|
|
|
/**
|
|
* Create Fieldset object
|
|
*
|
|
* @param string $name Identifier for this fieldset
|
|
* @param array $config Configuration array
|
|
* @return Fieldset
|
|
*/
|
|
public static function forge($name = 'default', array $config = array())
|
|
{
|
|
if ($exists = static::instance($name))
|
|
{
|
|
\Errorhandler::notice('Fieldset with this name exists already, cannot be overwritten.');
|
|
return $exists;
|
|
}
|
|
|
|
static::$_instances[$name] = new static($name, $config);
|
|
|
|
if ($name == 'default')
|
|
{
|
|
static::$_instance = static::$_instances[$name];
|
|
}
|
|
|
|
return static::$_instances[$name];
|
|
}
|
|
|
|
/**
|
|
* Return a specific instance, or the default instance (is created if necessary)
|
|
*
|
|
* @param Fieldset $instance
|
|
* @return Fieldset
|
|
*/
|
|
public static function instance($instance = null)
|
|
{
|
|
if ($instance !== null)
|
|
{
|
|
if ( ! array_key_exists($instance, static::$_instances))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
return static::$_instances[$instance];
|
|
}
|
|
|
|
if (static::$_instance === null)
|
|
{
|
|
static::$_instance = static::forge();
|
|
}
|
|
|
|
return static::$_instance;
|
|
}
|
|
|
|
/**
|
|
* @var string instance id
|
|
*/
|
|
protected $name;
|
|
|
|
/**
|
|
* @var string tag used to wrap this instance
|
|
*/
|
|
protected $fieldset_tag = null;
|
|
|
|
/**
|
|
* @var Fieldset instance to which this instance belongs
|
|
*/
|
|
protected $fieldset_parent = null;
|
|
|
|
/**
|
|
* @var array instances that belong to this one
|
|
*/
|
|
protected $fieldset_children = array();
|
|
|
|
/**
|
|
* @var array array of Fieldset_Field objects
|
|
*/
|
|
protected $fields = array();
|
|
|
|
/**
|
|
* @var Validation instance of validation
|
|
*/
|
|
protected $validation;
|
|
|
|
/**
|
|
* @var Form instance of form
|
|
*/
|
|
protected $form;
|
|
|
|
/**
|
|
* @var array configuration array
|
|
*/
|
|
protected $config = array();
|
|
|
|
/**
|
|
* @var array disabled fields array
|
|
*/
|
|
protected $disabled = array();
|
|
|
|
/**
|
|
* @var string name of class providing the tabular form
|
|
*/
|
|
protected $tabular_form_model = null;
|
|
|
|
/**
|
|
* @var string name of the relation of the parent object this tabular form is modeled on
|
|
*/
|
|
protected $tabular_form_relation = null;
|
|
|
|
/**
|
|
* Object constructor
|
|
*
|
|
* @param string
|
|
* @param array
|
|
*/
|
|
public function __construct($name = '', array $config = array())
|
|
{
|
|
// support new Fieldset($config) syntax
|
|
if (is_array($name))
|
|
{
|
|
$config = $name;
|
|
$name = '';
|
|
}
|
|
|
|
if (isset($config['validation_instance']))
|
|
{
|
|
$this->validation($config['validation_instance']);
|
|
unset($config['validation_instance']);
|
|
}
|
|
if (isset($config['form_instance']))
|
|
{
|
|
$this->form($config['form_instance']);
|
|
unset($config['form_instance']);
|
|
}
|
|
|
|
$this->name = (string) $name;
|
|
$this->config = $config;
|
|
}
|
|
|
|
/**
|
|
* Get related Validation instance or create it
|
|
*
|
|
* @param bool|Validation $instance
|
|
* @return Validation
|
|
*/
|
|
public function validation($instance = true)
|
|
{
|
|
if ($instance instanceof Validation)
|
|
{
|
|
$this->validation = $instance;
|
|
return $instance;
|
|
}
|
|
|
|
if (empty($this->validation) and $instance === true)
|
|
{
|
|
$this->validation = \Validation::forge($this);
|
|
}
|
|
|
|
return $this->validation;
|
|
}
|
|
|
|
/**
|
|
* Get related Form instance or create it
|
|
*
|
|
* @param bool|Form $instance
|
|
* @return Form
|
|
*/
|
|
public function form($instance = true)
|
|
{
|
|
if ($instance instanceof Form)
|
|
{
|
|
$this->form = $instance;
|
|
return $instance;
|
|
}
|
|
|
|
if (empty($this->form) and $instance === true)
|
|
{
|
|
$this->form = \Form::forge($this);
|
|
}
|
|
|
|
return $this->form;
|
|
}
|
|
|
|
/**
|
|
* Set the tag to be used for this fieldset
|
|
*
|
|
* @param string $tag
|
|
* @return Fieldset this, to allow chaining
|
|
*/
|
|
public function set_fieldset_tag($tag)
|
|
{
|
|
$this->fieldset_tag = $tag;
|
|
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* Set the parent Fieldset instance
|
|
*
|
|
* @param Fieldset $fieldset parent fieldset to which this belongs
|
|
* @return Fieldset
|
|
*/
|
|
public function set_parent(Fieldset $fieldset)
|
|
{
|
|
if ( ! empty($this->fieldset_parent))
|
|
{
|
|
throw new \RuntimeException('Fieldset already has a parent, belongs to "'.$this->parent()->name.'".');
|
|
}
|
|
|
|
$children = $fieldset->children();
|
|
while ($child = array_shift($children))
|
|
{
|
|
if ($child === $this)
|
|
{
|
|
throw new \RuntimeException('Circular reference detected, adding a Fieldset that\'s already a child as a parent.');
|
|
}
|
|
$children = array_merge($child->children(), $children);
|
|
}
|
|
|
|
$this->fieldset_parent = $fieldset;
|
|
$fieldset->add_child($this);
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* Add a child Fieldset instance
|
|
*
|
|
* @param Fieldset $fieldset
|
|
* @return Fieldset
|
|
*/
|
|
protected function add_child(Fieldset $fieldset)
|
|
{
|
|
if (is_null($fieldset->fieldset_tag))
|
|
{
|
|
$fieldset->fieldset_tag = 'fieldset';
|
|
}
|
|
|
|
$this->fieldset_children[$fieldset->name] = $fieldset;
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* Factory for Fieldset_Field objects
|
|
*
|
|
* @param string
|
|
* @param string
|
|
* @param array
|
|
* @param array
|
|
* @return Fieldset_Field
|
|
*/
|
|
public function add($name, $label = '', array $attributes = array(), array $rules = array())
|
|
{
|
|
if ($name instanceof Fieldset_Field)
|
|
{
|
|
if ($name->name == '' or $this->field($name->name) !== false)
|
|
{
|
|
throw new \RuntimeException('Fieldname empty or already exists in this Fieldset: "'.$name->name.'".');
|
|
}
|
|
|
|
$name->set_fieldset($this);
|
|
$this->fields[$name->name] = $name;
|
|
return $name;
|
|
}
|
|
elseif ($name instanceof Fieldset)
|
|
{
|
|
if (empty($name->name) or $this->field($name->name) !== false)
|
|
{
|
|
throw new \RuntimeException('Fieldset name empty or already exists in this Fieldset: "'.$name->name.'".');
|
|
}
|
|
|
|
$name->set_parent($this);
|
|
$this->fields[$name->name] = $name;
|
|
return $name;
|
|
}
|
|
|
|
if (empty($name) || (is_array($name) and empty($name['name'])))
|
|
{
|
|
throw new \InvalidArgumentException('Cannot create field without name.');
|
|
}
|
|
|
|
// Allow passing the whole config in an array, will overwrite other values if that's the case
|
|
if (is_array($name))
|
|
{
|
|
$attributes = $name;
|
|
$label = isset($name['label']) ? $name['label'] : '';
|
|
$rules = isset($name['rules']) ? $name['rules'] : array();
|
|
$name = $name['name'];
|
|
}
|
|
|
|
// Check if it exists already, if so: return and give notice
|
|
if ($field = $this->field($name))
|
|
{
|
|
\Errorhandler::notice('Field with this name exists already in this fieldset: "'.$name.'".');
|
|
return $field;
|
|
}
|
|
|
|
$this->fields[$name] = new \Fieldset_Field($name, $label, $attributes, $rules, $this);
|
|
|
|
return $this->fields[$name];
|
|
}
|
|
|
|
/**
|
|
* Add a new Fieldset_Field before an existing field in a Fieldset
|
|
*
|
|
* @param string $name
|
|
* @param string $label
|
|
* @param array $attributes
|
|
* @param array $rules
|
|
* @param string $fieldname fieldname before which the new field is inserted in the fieldset
|
|
* @return Fieldset_Field
|
|
*/
|
|
public function add_before($name, $label = '', array $attributes = array(), array $rules = array(), $fieldname = null)
|
|
{
|
|
$field = $this->add($name, $label, $attributes, $rules);
|
|
|
|
// Remove from tail and reinsert at correct location
|
|
unset($this->fields[$field->name]);
|
|
|
|
if ( ! \Arr::insert_before_key($this->fields, array($field->name => $field), $fieldname, true))
|
|
{
|
|
throw new \RuntimeException('Field "'.$fieldname.'" does not exist in this Fieldset. Field "'.$name.'" can not be added.');
|
|
}
|
|
|
|
return $field;
|
|
}
|
|
|
|
/**
|
|
* Add a new Fieldset_Field after an existing field in a Fieldset
|
|
*
|
|
* @param string $name
|
|
* @param string $label
|
|
* @param array $attributes
|
|
* @param array $rules
|
|
* @param string $fieldname fieldname after which the new field is inserted in the fieldset
|
|
* @return Fieldset_Field
|
|
*/
|
|
public function add_after($name, $label = '', array $attributes = array(), array $rules = array(), $fieldname = null)
|
|
{
|
|
$field = $this->add($name, $label, $attributes, $rules);
|
|
|
|
// Remove from tail and reinsert at correct location
|
|
unset($this->fields[$field->name]);
|
|
if ( ! \Arr::insert_after_key($this->fields, array($field->name => $field), $fieldname, true))
|
|
{
|
|
throw new \RuntimeException('Field "'.$fieldname.'" does not exist in this Fieldset. Field "'.$name.'" can not be added.');
|
|
}
|
|
|
|
return $field;
|
|
}
|
|
|
|
/**
|
|
* Delete a field instance
|
|
*
|
|
* @param string field name or null to fetch an array of all
|
|
* @return Fieldset this fieldset, for chaining
|
|
*/
|
|
public function delete($name)
|
|
{
|
|
if (isset($this->fields[$name]))
|
|
{
|
|
unset($this->fields[$name]);
|
|
}
|
|
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* Get Field instance
|
|
*
|
|
* @param string|null $name field name or null to fetch an array of all
|
|
* @param bool $flatten whether to get the fields array or flattened array
|
|
* @param bool $tabular_form whether to include tabular form fields in the flattened array
|
|
* @return Fieldset_Field|false returns false when field wasn't found
|
|
*/
|
|
public function field($name = null, $flatten = false, $tabular_form = true)
|
|
{
|
|
if ($name === null)
|
|
{
|
|
$fields = $this->fields;
|
|
|
|
if ($flatten)
|
|
{
|
|
foreach ($this->fieldset_children as $fs_name => $fieldset)
|
|
{
|
|
if ($tabular_form or ! $fieldset->get_tabular_form())
|
|
{
|
|
\Arr::insert_after_key($fields, $fieldset->field(null, true), $fs_name);
|
|
}
|
|
unset($fields[$fs_name]);
|
|
}
|
|
}
|
|
return $fields;
|
|
}
|
|
|
|
if ( ! array_key_exists($name, $this->fields))
|
|
{
|
|
if ($flatten)
|
|
{
|
|
foreach ($this->fieldset_children as $fieldset)
|
|
{
|
|
if (($field = $fieldset->field($name)) !== false)
|
|
{
|
|
return $field;
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
return $this->fields[$name];
|
|
}
|
|
|
|
/**
|
|
* Add a model's fields
|
|
* The model must have a method "set_form_fields" that takes this Fieldset instance
|
|
* and adds fields to it.
|
|
*
|
|
* @param string|Object $class either a full classname (including full namespace) or object instance
|
|
* @param array|Object $instance array or object that has the exactly same named properties to populate the fields
|
|
* @param string $method method name to call on model for field fetching
|
|
* @return Fieldset this, to allow chaining
|
|
*/
|
|
public function add_model($class, $instance = null, $method = 'set_form_fields')
|
|
{
|
|
// Add model to validation callables for validation rules
|
|
$this->validation()->add_callable($class);
|
|
|
|
if ((is_string($class) and is_callable($callback = array('\\'.$class, $method)))
|
|
|| is_callable($callback = array($class, $method)))
|
|
{
|
|
$instance ? call_user_func($callback, $this, $instance) : call_user_func($callback, $this);
|
|
}
|
|
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* Sets a config value on the fieldset
|
|
*
|
|
* @param string $config
|
|
* @param mixed $value
|
|
* @return Fieldset this, to allow chaining
|
|
*/
|
|
public function set_config($config, $value = null)
|
|
{
|
|
$config = is_array($config) ? $config : array($config => $value);
|
|
foreach ($config as $key => $value)
|
|
{
|
|
if (strpos($key, '.') === false)
|
|
{
|
|
$this->config[$key] = $value;
|
|
}
|
|
else
|
|
{
|
|
\Arr::set($this->config, $key, $value);
|
|
}
|
|
}
|
|
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* Get a single or multiple config values by key
|
|
*
|
|
* @param string|array $key a single key or multiple in an array, empty to fetch all
|
|
* @param mixed $default default output when config wasn't set
|
|
* @return mixed|array a single config value or multiple in an array when $key input was an array
|
|
*/
|
|
public function get_config($key = null, $default = null)
|
|
{
|
|
if ($key === null)
|
|
{
|
|
return $this->config;
|
|
}
|
|
|
|
if (is_array($key))
|
|
{
|
|
$output = array();
|
|
foreach ($key as $k)
|
|
{
|
|
$output[$k] = $this->get_config($k, $default);
|
|
}
|
|
return $output;
|
|
}
|
|
|
|
if (strpos($key, '.') === false)
|
|
{
|
|
return array_key_exists($key, $this->config) ? $this->config[$key] : $default;
|
|
}
|
|
else
|
|
{
|
|
return \Arr::get($this->config, $key, $default);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Populate the form's values using an input array or object
|
|
*
|
|
* @param array|object $input
|
|
* @param bool $repopulate
|
|
* @return Fieldset this, to allow chaining
|
|
*/
|
|
public function populate($input, $repopulate = false)
|
|
{
|
|
$fields = $this->field(null, true, false);
|
|
foreach ($fields as $f)
|
|
{
|
|
if (is_array($input) or $input instanceof \ArrayAccess)
|
|
{
|
|
// convert form field array's to Fuel dotted notation
|
|
$name = str_replace(array('[', ']'), array('.', ''), $f->name);
|
|
|
|
// fetch the value for this field, and set it if found
|
|
$value = \Arr::get($input, $name, null);
|
|
$value === null and $value = \Arr::get($input, $f->basename, null);
|
|
$value !== null and $f->set_value($value, true);
|
|
}
|
|
elseif (is_object($input) and property_exists($input, $f->basename))
|
|
{
|
|
$f->set_value($input->{$f->basename}, true);
|
|
}
|
|
}
|
|
|
|
// Optionally overwrite values using post/get
|
|
if ($repopulate)
|
|
{
|
|
$this->repopulate();
|
|
}
|
|
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* Set all fields to the input from get or post (depends on the form method attribute)
|
|
*
|
|
* @return Fieldset this, to allow chaining
|
|
*/
|
|
public function repopulate()
|
|
{
|
|
$fields = $this->field(null, true);
|
|
foreach ($fields as $f)
|
|
{
|
|
// Don't repopulate the CSRF field
|
|
if ($f->name === \Config::get('security.csrf_token_key', 'fuel_csrf_token'))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
if (($value = $f->input()) !== null)
|
|
{
|
|
$f->set_value($value, true);
|
|
}
|
|
}
|
|
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* Build the fieldset HTML
|
|
*
|
|
* @param mixed $action
|
|
* @return string
|
|
*/
|
|
public function build($action = null)
|
|
{
|
|
$attributes = $this->get_config('form_attributes');
|
|
if ($action and ($this->fieldset_tag == 'form' or empty($this->fieldset_tag)))
|
|
{
|
|
$attributes['action'] = $action;
|
|
}
|
|
|
|
$open = ($this->fieldset_tag == 'form' or empty($this->fieldset_tag))
|
|
? $this->form()->open($attributes).PHP_EOL
|
|
: $this->form()->{$this->fieldset_tag.'_open'}($attributes);
|
|
|
|
$fields_output = '';
|
|
|
|
// construct the tabular form table header
|
|
if ($this->tabular_form_relation)
|
|
{
|
|
$properties = call_user_func($this->tabular_form_model.'::properties');
|
|
$primary_keys = call_user_func($this->tabular_form_model.'::primary_key');
|
|
$fields_output .= '<thead><tr>'.PHP_EOL;
|
|
foreach ($properties as $field => $settings)
|
|
{
|
|
if ((isset($settings['skip']) and $settings['skip']) or in_array($field, $primary_keys))
|
|
{
|
|
continue;
|
|
}
|
|
if (isset($settings['form']['type']) and ($settings['form']['type'] === false or $settings['form']['type'] === 'hidden'))
|
|
{
|
|
continue;
|
|
}
|
|
$fields_output .= "\t".'<th class="'.$this->tabular_form_relation.'_col_'.$field.'">'.(isset($settings['label']) ? \Lang::get($settings['label'], array(), $settings['label']) : '').'</th>'.PHP_EOL;
|
|
}
|
|
$fields_output .= "\t".'<th>'.\Config::get('form.tabular_delete_label', 'Delete?').'</th>'.PHP_EOL;
|
|
|
|
$fields_output .= '</tr></thead>'.PHP_EOL;
|
|
}
|
|
|
|
foreach ($this->field() as $f)
|
|
{
|
|
in_array($f->name, $this->disabled) or $fields_output .= $f->build().PHP_EOL;
|
|
}
|
|
|
|
$close = ($this->fieldset_tag == 'form' or empty($this->fieldset_tag))
|
|
? $this->form()->close($attributes).PHP_EOL
|
|
: $this->form()->{$this->fieldset_tag.'_close'}($attributes);
|
|
|
|
$template = $this->form()->get_config((empty($this->fieldset_tag) ? 'form' : $this->fieldset_tag).'_template',
|
|
"\n\t\t{open}\n\t\t<table>\n{fields}\n\t\t</table>\n\t\t{close}\n");
|
|
|
|
$template = str_replace(array('{form_open}', '{open}', '{fields}', '{form_close}', '{close}'),
|
|
array($open, $open, $fields_output, $close, $close),
|
|
$template);
|
|
|
|
return $template;
|
|
}
|
|
|
|
/**
|
|
* Enable a disabled field from being build
|
|
*
|
|
* @param mixed $name
|
|
* @return Fieldset this, to allow chaining
|
|
*/
|
|
public function enable($name = null)
|
|
{
|
|
// Check if it exists. if not, bail out
|
|
if ( ! $this->field($name))
|
|
{
|
|
throw new \RuntimeException('Field "'.$name.'" does not exist in this Fieldset.');
|
|
}
|
|
|
|
if (isset($this->disabled[$name]))
|
|
{
|
|
unset($this->disabled[$name]);
|
|
}
|
|
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* Disable a field from being build
|
|
*
|
|
* @param mixed $name
|
|
* @return Fieldset this, to allow chaining
|
|
*/
|
|
public function disable($name = null)
|
|
{
|
|
// Check if it exists. if not, bail out
|
|
if ( ! $this->field($name))
|
|
{
|
|
throw new \RuntimeException('Field "'.$name.'" does not exist in this Fieldset.');
|
|
}
|
|
|
|
isset($this->disabled[$name]) or $this->disabled[$name] = $name;
|
|
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* Magic method toString that will build this as a form
|
|
*
|
|
* @return string
|
|
*/
|
|
public function __toString()
|
|
{
|
|
return $this->build();
|
|
}
|
|
|
|
/**
|
|
* Return parent Fieldset
|
|
*
|
|
* @return Fieldset
|
|
*/
|
|
public function parent()
|
|
{
|
|
return $this->fieldset_parent;
|
|
}
|
|
|
|
/**
|
|
* Return the child fieldset instances
|
|
*
|
|
* @return array
|
|
*/
|
|
public function children()
|
|
{
|
|
return $this->fieldset_children;
|
|
}
|
|
|
|
/**
|
|
* Alias for $this->validation()->input()
|
|
*
|
|
* @param string $field
|
|
* @return mixed
|
|
*/
|
|
public function input($field = null)
|
|
{
|
|
return $this->validation()->input($field);
|
|
}
|
|
|
|
/**
|
|
* Alias for $this->validation()->validated()
|
|
*
|
|
* @param string $field
|
|
* @return mixed
|
|
*/
|
|
public function validated($field = null)
|
|
{
|
|
return $this->validation()->validated($field);
|
|
}
|
|
|
|
/**
|
|
* Alias for $this->validation()->error()
|
|
*
|
|
* @param string $field
|
|
* @return Validation_Error|array
|
|
*/
|
|
public function error($field = null)
|
|
{
|
|
return $this->validation()->error($field);
|
|
}
|
|
|
|
/**
|
|
* Alias for $this->validation()->show_errors()
|
|
*
|
|
* @param array $config
|
|
* @return string
|
|
*/
|
|
public function show_errors(array $config = array())
|
|
{
|
|
return $this->validation()->show_errors($config);
|
|
}
|
|
|
|
/**
|
|
* Get the fieldset name
|
|
*
|
|
* @return string
|
|
*/
|
|
public function get_name()
|
|
{
|
|
return $this->name;
|
|
}
|
|
|
|
/**
|
|
* Enable or disable the tabular form feature of this fieldset
|
|
*
|
|
* @param string $model Model on which to define the tabular form
|
|
* @param string $relation Relation of the Model on the tabular form is modeled
|
|
* @param array $parent Collection of Model objects from a many relation
|
|
* @param int $blanks Number of empty rows to generate
|
|
*
|
|
* @return Fieldset this, to allow chaining
|
|
*/
|
|
public function set_tabular_form($model, $relation, $parent, $blanks = 1)
|
|
{
|
|
// make sure our parent is an ORM model instance
|
|
if ( ! $parent instanceOf \Orm\Model)
|
|
{
|
|
throw new \RuntimeException('Parent passed to set_tabular_form() is not an ORM model object.');
|
|
}
|
|
|
|
// validate the model and relation
|
|
// fetch the relations of the parent model
|
|
$relations = call_user_func(array($parent, 'relations'));
|
|
if ( ! array_key_exists($relation, $relations))
|
|
{
|
|
throw new \RuntimeException('Relation passed to set_tabular_form() is not a valid relation of the ORM parent model object.');
|
|
}
|
|
|
|
// check for compound primary keys
|
|
try
|
|
{
|
|
// fetch the relations of the parent model
|
|
$primary_key = call_user_func($model.'::primary_key');
|
|
|
|
// we don't support compound primary keys
|
|
if (count($primary_key) !== 1)
|
|
{
|
|
throw new \RuntimeException('set_tabular_form() does not supports models with compound primary keys.');
|
|
}
|
|
|
|
// store the primary key name, we need that later
|
|
$primary_key = reset($primary_key);
|
|
}
|
|
catch (\Exception $e)
|
|
{
|
|
throw new \RuntimeException('Unable to fetch the models primary key information.');
|
|
}
|
|
|
|
// store the tabular form class name
|
|
$this->tabular_form_model = $model;
|
|
|
|
// and the relation on which we model the rows
|
|
$this->tabular_form_relation = $relation;
|
|
|
|
// load the form config if not loaded yet
|
|
\Config::load('form', true);
|
|
|
|
// load the config for embedded forms
|
|
$this->set_config(array(
|
|
'form_template' => \Config::get('form.tabular_form_template', "<table>{fields}</table>\n"),
|
|
'field_template' => \Config::get('form.tabular_field_template', "{field}"),
|
|
));
|
|
|
|
// add the rows to the tabular form fieldset
|
|
foreach ($parent->{$relation} as $row)
|
|
{
|
|
// add the row fieldset to the tabular form fieldset
|
|
$this->add($fieldset = \Fieldset::forge($this->tabular_form_relation.'_row_'.$row->{$primary_key}));
|
|
|
|
// and add the model fields to the row fielset
|
|
$fieldset->add_model($model, $row)->set_fieldset_tag(false);
|
|
$fieldset->set_config(array(
|
|
'form_template' => \Config::get('form.tabular_row_template', "<table>{fields}</table>\n"),
|
|
'field_template' => \Config::get('form.tabular_row_field_template', "{field}"),
|
|
));
|
|
$fieldset->add($this->tabular_form_relation.'['.$row->{$primary_key}.'][_delete]', '', array('type' => 'checkbox', 'value' => 1));
|
|
}
|
|
|
|
// and finish with zero or more empty rows so we can add new data
|
|
if ( ! is_numeric($blanks) or $blanks < 0)
|
|
{
|
|
$blanks = 1;
|
|
}
|
|
for ($i = 0; $i < $blanks; $i++)
|
|
{
|
|
$this->add($fieldset = \Fieldset::forge($this->tabular_form_relation.'_new_'.$i));
|
|
$fieldset->add_model($model)->set_fieldset_tag(false);
|
|
$fieldset->set_config(array(
|
|
'form_template' => \Config::get('form.tabular_row_template', "<tr>{fields}</tr>"),
|
|
'field_template' => \Config::get('form.tabular_row_field_template', "{field}"),
|
|
));
|
|
$fieldset->add($this->tabular_form_relation.'_new['.$i.'][_delete]', '', array('type' => 'checkbox', 'value' => 0, 'disabled' => 'disabled'));
|
|
|
|
// no required rules on this row
|
|
foreach ($fieldset->field() as $f)
|
|
{
|
|
$f->delete_rule('required', false)->delete_rule('required_with', false);
|
|
}
|
|
}
|
|
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* return the tabular form relation of this fieldset
|
|
*
|
|
* @return bool
|
|
*/
|
|
public function get_tabular_form()
|
|
{
|
|
return $this->tabular_form_relation;
|
|
}
|
|
}
|