Wednesday, 14 February 2007

PHP: Form with smarty + captcha

Form composition with smarty + captcha


/*
Usage:
$ff=array(
array('text', 'field_name', 'label', array('class'=>'area'), array('postlabel'=>'ddd', 'validate'=>'empty')),
array('separator', 'td_class', 'text here'),
array('options', 'field_name2', 'label', array('options'=>array(1=>2)), array('postlabel'=>'ddd', 'check'=>'empty')),
array('submit', '', '', array('value'=>'Sign up')),
array('hidden', 'hid', '', array('value'=>'blah'))
);
sf_init($ff);

$smarty->assign(array('f2'=>$ff, 'in'=>$_REQUEST));
$sm->register_function('smarty_form', 'smarty_form');
{smarty_form fields=$f2 in=$in table_attr="class=tabt cellpadding=0 cellspacing=0" form_attr="method=post" colspan_class="separator"}

*/

function sf_init(&$ff) {
$ft=array('type', 'name', 'label', 'fp', 'gp'); // make readable array
foreach ($ff as $fd) {
$fdh=array(); foreach ($ft as $i=>$t) {if (isset($fd[$i])) $fdh[$t]=$fd[$i];}
$f2[]=$fdh;
}
$ff=$f2;
}
function iname($f2, $fd) {
foreach ($f2 as $i=>$f) {if ($f['name']==$fd) $fi=$i;}
return (isset($fi) ? $fi : -1);
}
function vfields($f2) {
$fr=array();
foreach ($f2 as $f) {
if (!in_array($f['type'], array('separator', 'submit')) && $f['name']!='captcha') $fr[]=$f['name'];
}
return $fr;
}

function smarty_form($p, &$smarty) {
$hidden=array();
foreach ($p['fields'] as $i=>$_fd) {$fd=&$p['fields'][$i];
// start values
$fd['fp']['name']=$fd['name']; // for form tags
$fd['fp']['type']=$fd['type'];
if (isset($p['in'][$fd['name']])) { // in-value
$par=(in_array($fd['type'], array('options', 'radios', 'checkboxes')) ? 'selected' : 'value');
$fd['fp'][$par]=$p['in'][$fd['name']];
}
// generate fields
if (in_array($fd['type'], array('options', 'radios', 'checkboxes'))) {
require_once(SMARTY_DIR."plugins/function.html_$fd[type].php");
if (isset($p['in'][$fd['name']])) $fd['fp']['selected']=$p['in'][$fd['name']];
$fields_t[$fd['name']]=call_user_func("smarty_function_html_$fd[type]", $fd['fp'], $smarty);
} elseif (!in_array($fd['type'], array('separator', 'hidden'))) {
$fields_t[$fd['name']]=_field_gen($fd['fp'], $fd['type']);
}
// data for table, hidden fields
if ($fd['type']=='hidden') $hidden[]=_field_gen($fd['fp'], $fd['type']);
elseif ($fd['type']=='separator') $data[][]=$fd['label'];
else $data[]=array(
$fd['label'].(isset($fd['gp']['check']) ? ' *' : ''),
$fields_t[$fd['name']].(isset($fd['gp']['plabel']) ? $fd['gp']['plabel'] : '')
);
}

$js=_jstext($p['fields']); // javascript-check
return $js.'<form '.$p['form_attr'].($js ? ' onsubmit="return validate_form(this)"' : '').'>'.
"\n".implode("\n", $hidden)."\n"._tablefm($data, &$p).'</form>';
}
function _tablefm($data, $p) {
$res[]='<table '.$p['table_attr'].'>';
$cols=0; foreach ($data as $line) {$c=count($line); if ($c>$cols) $cols=$c;}
foreach ($data as $line) {
$res[]='<tr>';
if (count($line)==1) $res[]="<td class=\"$p[colspan_class]\" colspan=$cols>$line[0]</td>";
else {foreach ($line as $y=>$v) $res[]="<td>$v</td>";}
$res[]='</tr>';
}
$res[]='</table>';
return implode("\n", $res);
}
function _jstext($fields_h) {
foreach ($fields_h as $fd) {
if (!isset($fd['gp']['check'])) continue;
$rs['name'][]=$fd['name'];
$rs['label'][]=$fd['label'];
$rs['check'][]=$fd['gp']['check'];
}
if (!$rs) return '';
return '
<script>
v_names = new Array("'.implode('", "', $rs['name']).'");
v_checks = new Array("'.implode('", "', $rs['check']).'");
v_labels = new Array("'.implode('", "', $rs['label']).'");

function validate_form(obj) {
errors="";
for (i=0; i<obj.elements.length; i++) {
for (y=0; y<v_names.length; y++) {
if (obj.elements[i].name==v_names[y]) errors+=_validate(v_checks[y], obj.elements[i], v_labels[y]);
}
}
if (errors) {alert(errors); return false;}
else return true;
}
function _validate(type, obj, label) {
res="";
if (type=="email") {
if (obj.value.indexOf(".")>2 && obj.value.indexOf("@") > 0) res="";
else res="E-mail address is not valid\n";
} else if (type=="empty") {
if (obj.value=="" || obj.value==null) res="Field \""+label+"\" is empty\n";
}
return res;
}
</script>
';
}
function _field_gen($params, $type='input') {
if ($type=='textarea') {
foreach ($params as $k=>$v) {if ($k!='value') $pp="$k=\"$v\"";}
return '<textarea '.implode(' ', $pp).'>'.(isset($params['value']) ? $params['value'] : '').'</textarea>';
} else {
foreach ($params as $k=>$v) {$pp[]="$k=\"$v\"";}
return '<input '.implode(' ', $pp).'>';
}
}


/*
Usage:
image.php:
$captcha=array(140, 34, 'britanic.ttf', 25, 0, 0, 0, array(255,255,255), array(0,0,50));
captcha();
form.php:
<form method="POST"><img src="img.php"><input type="text" name="captcha"><input type="submit"></form>
if ($_POST && check_captcha()) echo "ok";

*/

function captcha() {
global $captcha;
list($box_w, $box_h, $font, $font_size, $font_angle, $margin_left, $margin_top, $color_bg, $color_text)=$captcha;
$pass=''; $chm=rand(0,1);
$ch=array(str_split('aeiouy'), str_split('bcdfjhgkmnprstvxz'));
while (strlen($pass)<6) {$pass.=$ch[$chm][array_rand($ch[$chm])]; $chm=($chm ? 0:1);}
session_start();
$_SESSION['captcha'] = $pass;
$image = ImageCreatetruecolor($box_w,$box_h); // create the image resource
$color_bg = ImageColorAllocate($image, $color_bg[0], $color_bg[1], $color_bg[2]); // set colors
$color_text = ImageColorAllocate($image, $color_text[0], $color_text[1], $color_text[2]);
imagefill($image, 0, 0, $color_bg); // set background
imagettftext($image, $font_size, $font_angle, $margin_left, $font_size + $margin_top, $color_text, $font, $pass); // set text
header("Content-Type: image/jpg"); // or image/jpg
imagejpeg($image); // created image
imagedestroy($image);
}
function check_captcha() {
session_start();
if (isset($_SESSION['captcha']) && isset($_REQUEST['captcha']) && $_SESSION['captcha']==$_REQUEST['captcha']) {
unset($_SESSION['captcha']);
return 1;
}
return 0;
}