-
Notifications
You must be signed in to change notification settings - Fork 0
Form Library
Category:Libraries::Data Presentation
The Form library is a complete replacement for the Form helper. It reads an XML document and converts it to an array (via the Xml_Library).
[b]Updates:[/b]
2006-11-06:
- Changed several bits, parse by fieldsets
- Changed the format of form templates
2006-11-04:
- Made validate() set the form values, no need to set them manually
- Tweaked the valid types and attributes
[b]Features:[/b]
- Built in validation using CI Validation
- 1 call data retrieval
- XHTML 1.0 output code
- written for PHP5 ([b]not[/b] PHP4 compatible)
[b]Example Usage:[/b] [code]$this->load->library('form'); $this->form->load('login'); // Relative to APPPATH/data/forms/, ".xml" appended if ($data = $this->form->post_data()) { print_r($data); } else { print $this->form->build(); }[/code]
[b]Library:[/b] [code]<?php if (!defined('BASEPATH')) exit('No direct script access allowed');
/***
- Form library for CodeIgniter
- author: Woody Gilk
- copyright: (c) 2006
- license: http://creativecommons.org/licenses/by-sa/2.5/
-
file: libraries/Form.php
*/
class Form {
public function Form () { /*** * @constructor / $obj =& get_instance(); $obj->load->library('xml'); $this->ci =& $obj; } /** END ***/
/*** Public variables ***/ public $error = ''; public $set_error = ''; public $output = '';
/*** Internal variables ***/ private $ci; private $rules; private $data; private $set_data; private $elements; private $action;
public function load ($name) { /*** * @public * Load a form definition for parsing */ if (! $this->ci->xml->load ("data/forms/$name")) { $this->error = "Failed to load form: $name"; return false; }
$data = $this->ci->xml->parse ();
if (! is_array($data)) {
$this->error = "No form data found in /data/forms/$name.xml";
return false;
}
else {
$data = $data['form'][0];
$this->data = $data;
$this->action = $data['__attrs']['action'];
$this->elements = $this->get_fields($data);
}
return true;
} /*** END load ***/
private function get_fields ($array) { /*** * @private * Extract the rules and field names from the data */ $return = array();
foreach ($array as $key => $val) {
if ($key == 'fieldset' && is_array ($val)) {
$return = array_merge ($return, $this->get_fields ($val));
}
else {
if (! is_numeric ($key) && $key != '__attrs' && $key != 'fieldset') {
$return[] = $key;
foreach ($val as $_key => $_val) {
if (isset ($_val['__attrs']) && isset ($_val['__attrs']['rules'])) {
$this->rules[$key] = $_val['__attrs']['rules'];
}
}
}
}
}
return array_unique ($return);
} /*** END get_fields ***/
public function set_action ($location) { /*** * @public * Set the form action */
$this->action = $location;
return true;
} /** END action ***/
public function set ($element, $key, $value = false) { /*** * @public * Set an attribute of an element, or a new elemenet */ if (is_array ($key)) { foreach($key as $_key => $_value) { $this->set($element, $_key, $_value); }
return true;
}
$this->set_data[$element][$key] = $value;
return true;
} /*** END set ***/
public function post_data ($validate = true) { /*** * @public * Return all the post data from the loaded form */ if ($validate == true && !$this->validate ()) { return false; } if (! is_array($this->elements)) { return false; }
$data = array ();
foreach ($this->elements as $elem) {
$data[$elem] = $this->ci->input->post($elem);
}
return $data;
} /*** END post_data ***/
public function validate ($name = false) { /*** * @public * Validates a form based on the rules found in the definition */ if ($name != false && ! $this->load ($name)) { return false; } elseif (! is_array ($this->rules)) { return false; }
$this->ci->load->library ('validation');
$this->ci->validation->set_rules ($this->rules);
if ($this->ci->validation->run()) {
return true;
}
else {
foreach ($this->post_data (false) as $key => $val) {
if ($val != false) { // Set default values
$this->set($key, 'value', $val);
}
}
$this->set_error = '';
foreach ($this->ci->validation->_error_array as $error) {
$alt = (isset ($alt) && $alt == '')
? ' alt'
: '';
$this->set_error .= "\t<p class=\"error$alt\">$error</p>\n";
}
}
return false;
} /*** END validate ***/
private function build_group ($group, $depth = 0) { static $first_run;
// Set the valid attributes and type values
$valid_attr = array(
'type', 'maxlength', 'value',
'selected', 'rows', 'cols',
'onclick', 'onmouseover', 'onmouseout', 'onchange'
);
$valid_type = array(
'text', 'textarea', 'password',
'dropdown', 'radio', 'checkbox',
'submit', 'button'
);
$tabs = repeater("\t", $depth);
$html = sprintf ("$tabs".'<fieldset><legend>%s</legend>'."\n", $group['__attrs']['name']);
if ($first_run !== false) {
$html .= "$tabs". preg_replace("|\n\t+|", "\n$tabs\t", $this->set_error) ."\n";
$first_run = false;
}
$html .= "$tabs\t".'<ol class="form">'."\n";
foreach ($group as $name => $val) {
if ($name == '__attrs') {
continue;
}
elseif ($name == 'fieldset') {
foreach ($val as $_group) {
$html .= "$tabs\t<li>\n". $this->build_group ($_group, $depth+2) ."$tabs\t</li>\n";
}
}
else {
foreach ($val as $index => $def) {
foreach ($def as $key => $val) {
if ($key == '__attrs') {
unset ($def[$key]);
continue;
}
$def[$key] = $val[0];
}
// Skips defs that have no type attribute
if (! isset($def['type']) || ! in_array ($def['type'], $valid_type)) {
continue;
}
// Externally set data is present, merge with stored efinition
if (isset ($this->set_data[$name]) && is_array ($this->set_data[$name])) {
$def = array_merge($def, $this->set_data[$name]);
}
// Choose a label
$label = isset($def['label'])
? ucwords($def['label'])
: ucwords($name);
$label = strip_tags($label);
// Create the id and name attributes
$idname = sprintf('name="%s" id="%s"', $name, $name);
// Add "*" on required items
$label = isset ($def['required']) && $def['required'] == true
? "$label <em>*</em>"
: $label;
$row = "";
$row .= "$tabs\t<li>\n";
$row .= $def['type'] != 'submit'
? "$tabs\t\t<label>$label</label>\n"
: "";
// Handle non-input elements
switch ($def['type']) {
case 'textarea':
$input = "$tabs\t\t<textarea $idname %s>". $def['value'] ."</textarea>\n";
unset ($def['type'], $def['value']);
break;
case 'dropdown':
$options = "";
foreach ($def['options'] as $_key => $_val) {
$options = "$tabs\t\t\t<option value=\"$_val\">$_key</option>\n";
}
$input = "$tabs\t\t<select $idname %s >\n$options</select>\n";
unset ($def['type'], $def['options']);
break;
default:
$input = "$tabs\t\t<input $idname %s />\n";
}
// Parse attributes
$attributes = '';
foreach ($def as $attr => $val) {
if (in_array ($attr, $valid_attr)) {
$attributes .= " $attr=\"$val\" ";
}
}
$row .= sprintf($input, $attributes);
$row .= "$tabs\t</li>\n";
$html .= "$row";
}
}
}
$html .= "$tabs\t".'</ol>'."\n";
$html .= "$tabs".'</fieldset>'."\n";
return $html;
}
public function build ($name = false) { /*** * @public * Convert a form definition into an XHTML form */ if ($name != false && ! $this->load ($name)) { return false; } elseif (! is_array ($this->data)) { return false; } $this->ci->load->helper('string');
$out =& $this->output;
$out = '';
$out .= sprintf("".'<form action="%s" id="%s" method="post">'."\n",
site_url($this->action),
strtolower(preg_replace('|\W|', '_', $this->data['__attrs']['name']))
);
foreach ($this->data['fieldset'] as $group) {
$out .= $this->build_group($group);
}
$out .= "</form>\n";
$this->output = $out;
return $out;
} /*** END build ***/
}
?>[/code]
[b]Sample Form:[/b] [code]<?xml version="1.0" encoding="UTF-8" ?> <form action="user/login" name="user_login">
text 32 true password 32 true submit Login </form>[/code]