(mixed) custom id attribute value
 *  if numeric value is passed it'll be just appended to the name e.g. {filed-name}-{value}
 *  if false is passed id will be not created
 *  if empty string is passed (or no 'id' option is found)
 *  in all other cases the value will be used as field id
 * 	default: empty string
 *
 *  - class => (string) field class(es)
 * 	Example: 'tbox select class1 class2 class3'
 * 	NOTE: this will override core classes, so you have to explicit include them!
 * 	default: empty string
 *
 *  - size => (int) size attribute value (used when needed)
 *	default: 40
 *
 *  - title (string) title attribute
 *  default: empty string (omitted)
 *
 *  - readonly => (bool) readonly attribute
 * 	default: false
 *
 *  - selected => (bool) selected attribute (used when needed)
 * 	default: false
 *
 *  checked => (bool) checked attribute (used when needed)
 *  default: false
 *  - disabled => (bool) disabled attribute
 *  default: false
 *
 *  - tabindex => (int) tabindex attribute value
 *	default: inner tabindex counter
 *
 *  - other => (string) additional data
 *  Example: 'attribute1="value1" attribute2="value2"'
 *  default: empty string
 */
class e_form
{
	protected   $_tabindex_counter = 0;
	protected   $_tabindex_enabled = true;
	protected   $_cached_attributes = array();
	protected   $_field_warnings = array();
	private     $_inline_token;
	public      $_snippets = false; // use snippets or not. - experimental, and may be removed -  use at own risk.
	private     $_fontawesome = false;
	private     $_helptip = 1;
	/**
	 * @var user_class
	 */
	protected $_uc;
	protected $_required_string;
	public function __construct($enable_tabindex = false)
	{
		e107::loadAdminIcons(); // required below.
		e107_include_once(e_LANGUAGEDIR.e_LANGUAGE. '/lan_form_handler.php');
		$this->_tabindex_enabled = $enable_tabindex;
		$this->_uc = e107::getUserClass();
		$this->setRequiredString(' * ');
		if(defset('THEME_VERSION') === 2.3)
		{
			$this->_snippets = true;
		}
		if(deftrue('FONTAWESOME'))
		{
			$this->_fontawesome = true;
		}
		$this->_helptip = (int) e107::getPref('admin_helptip', 1);
	}
	/**
	 * @param $tmp
	 * @return array
	 * @see https://github.com/e107inc/e107/issues/3533
	 */
	private static function sort_get_files_output($tmp)
	{
		usort($tmp, static function ($left, $right) {
			$left_full_path = $left['path'] . $left['fname'];
			$right_full_path = $right['path'] . $right['fname'];
			return strcmp($left_full_path, $right_full_path);
		});
		return $tmp;
	}
	public function addWarning($field)
	{
		$this->_field_warnings[] = $field;
	}
	/**
	 * Open a new form
	 * @param string name
	 * @param $method - post|get  default is post
	 * @param string target - e_REQUEST_URI by default
	 * @param array|string $options
	 * @return string
	 */
	public function open($name, $method=null, $target=null, $options=null)
	{
		if($target == null)
		{
			$target = e_REQUEST_URI;	
		}
		
		if($method == null)
		{
			$method = 'post';
		}
		$autoComplete 	= '';
		if(is_string($options))
		{
			parse_str($options, $options);
		}
	
		if(!empty($options['class']))
		{
			$class = "class='".$options['class']."'";
		}
		else  // default 
		{
			$class= "class='form-horizontal'"; 
		}
		
		if(isset($options['autocomplete'])) // leave as isset()
		{
			$autoComplete = " autocomplete='".($options['autocomplete'] ? 'on' : 'off')."'";	
		}
		
		if($method === 'get' && strpos($target,'='))
		{
			list($url,$qry) = explode('?',$target);
			$text = "\n
';
		
	}
	/**
	 * Render a country drop-down list.
	 * @param string $name
	 * @param string $value
	 * @param array $options
	 * @return string
	 */
	public function country($name, $value, $options=array())
	{
		$arr = $this->getCountry();
		$placeholder = isset($options['placeholder']) ? $options['placeholder'] : ' ';
		return $this->select($name, $arr, $value, $options, $placeholder);
	}
	/**
	 * Get a list of countries.
	 * @param null|string $iso ISO code.
	 * @return array|mixed|string
	 */
	public function getCountry($iso=null)  // move to parser?
	{
		$c = array();
		 $c['af'] = 'Afghanistan';
		 $c['al'] = 'Albania';
		 $c['dz'] = 'Algeria';
		 $c['as'] = 'American Samoa';
		 $c['ad'] = 'Andorra';
		 $c['ao'] = 'Angola';
		 $c['ai'] = 'Anguilla';
		 $c['aq'] = 'Antarctica';
		 $c['ag'] = 'Antigua and Barbuda';
		 $c['ar'] = 'Argentina';
		 $c['am'] = 'Armenia';
		 $c['aw'] = 'Aruba';
		 $c['au'] = 'Australia';
		 $c['at'] = 'Austria';
		 $c['az'] = 'Azerbaijan';
		 $c['bs'] = 'Bahamas';
		 $c['bh'] = 'Bahrain';
		 $c['bd'] = 'Bangladesh';
		 $c['bb'] = 'Barbados';
		 $c['by'] = 'Belarus';
		 $c['be'] = 'Belgium';
		 $c['bz'] = 'Belize';
		 $c['bj'] = 'Benin';
		 $c['bm'] = 'Bermuda';
		 $c['bt'] = 'Bhutan';
		 $c['bo'] = 'Bolivia';
		 $c['ba'] = 'Bosnia-Herzegovina';
		 $c['bw'] = 'Botswana';
		 $c['bv'] = 'Bouvet Island';
		 $c['br'] = 'Brazil';
		 $c['io'] = 'British Indian Ocean Territory';
		 $c['bn'] = 'Brunei Darussalam';
		 $c['bg'] = 'Bulgaria';
		 $c['bf'] = 'Burkina Faso';
		 $c['bi'] = 'Burundi';
		 $c['kh'] = 'Cambodia';
		 $c['cm'] = 'Cameroon';
		 $c['ca'] = 'Canada';
		 $c['cv'] = 'Cape Verde';
		 $c['ky'] = 'Cayman Islands';
		 $c['cf'] = 'Central African Republic';
		 $c['td'] = 'Chad';
		 $c['cl'] = 'Chile';
		 $c['cn'] = 'China';
		 $c['cx'] = 'Christmas Island';
		 $c['cc'] = 'Cocos (Keeling) Islands';
		 $c['co'] = 'Colombia';
		 $c['km'] = 'Comoros';
		 $c['cg'] = 'Congo';
		 $c['cd'] = 'Congo (Dem.Rep)';
		 $c['ck'] = 'Cook Islands';
		 $c['cr'] = 'Costa Rica';
		 $c['hr'] = 'Croatia';
		 $c['cu'] = 'Cuba';
		 $c['cy'] = 'Cyprus';
		 $c['cz'] = 'Czech Republic';
		 $c['dk'] = 'Denmark';
		 $c['dj'] = 'Djibouti';
		 $c['dm'] = 'Dominica';
		 $c['do'] = 'Dominican Republic';
		 $c['tp'] = 'East Timor';
		 $c['ec'] = 'Ecuador';
		 $c['eg'] = 'Egypt';
		 $c['sv'] = 'El Salvador';
		 $c['gq'] = 'Equatorial Guinea';
		 $c['er'] = 'Eritrea';
		 $c['ee'] = 'Estonia';
		 $c['et'] = 'Ethiopia';
		 $c['fk'] = 'Falkland Islands';
		 $c['fo'] = 'Faroe Islands';
		 $c['fj'] = 'Fiji';
		 $c['fi'] = 'Finland';
		// $c['cs'] = "Former Czechoslovakia";
		// $c['su'] = "Former USSR";
		 $c['fr'] = 'France';
		// $c['fx'] = "France (European Territory)";
		 $c['gf'] = 'French Guyana';
		 $c['tf'] = 'French Southern Territories';
		 $c['ga'] = 'Gabon';
		 $c['gm'] = 'Gambia';
		 $c['ge'] = 'Georgia';
		 $c['de'] = 'Germany';
		 $c['gh'] = 'Ghana';
		 $c['gi'] = 'Gibraltar';
		 $c['gr'] = 'Greece';
		 $c['gl'] = 'Greenland';
		 $c['gd'] = 'Grenada';
		 $c['gp'] = 'Guadeloupe (French)';
		 $c['gu'] = 'Guam (USA)';
		 $c['gt'] = 'Guatemala';
		 $c['gn'] = 'Guinea';
		 $c['gw'] = 'Guinea Bissau';
		 $c['gy'] = 'Guyana';
		 $c['ht'] = 'Haiti';
		 $c['hm'] = 'Heard and McDonald Islands';
		 $c['hn'] = 'Honduras';
		 $c['hk'] = 'Hong Kong';
		 $c['hu'] = 'Hungary';
		 $c['is'] = 'Iceland';
		 $c['in'] = 'India';
		 $c['id'] = 'Indonesia';
		 $c['ir'] = 'Iran';
		 $c['iq'] = 'Iraq';
		 $c['ie'] = 'Ireland';
		 $c['il'] = 'Israel';
		 $c['it'] = 'Italy';
		 $c['ci'] = "Ivory Coast (Cote D'Ivoire)";
		 $c['jm'] = 'Jamaica';
		 $c['jp'] = 'Japan';
		 $c['jo'] = 'Jordan';
		 $c['kz'] = 'Kazakhstan';
		 $c['ke'] = 'Kenya';
		 $c['ki'] = 'Kiribati';
		 $c['kp'] = 'Korea (North)';
		 $c['kr'] = 'Korea (South)';
		 $c['kw'] = 'Kuwait';
		 $c['kg'] = 'Kyrgyzstan';
		 $c['la'] = 'Laos';
		 $c['lv'] = 'Latvia';
		 $c['lb'] = 'Lebanon';
		 $c['ls'] = 'Lesotho';
		 $c['lr'] = 'Liberia';
		 $c['ly'] = 'Libya';
		 $c['li'] = 'Liechtenstein';
		 $c['lt'] = 'Lithuania';
		 $c['lu'] = 'Luxembourg';
		 $c['mo'] = 'Macau';
		 $c['mk'] = 'Macedonia';
		 $c['mg'] = 'Madagascar';
		 $c['mw'] = 'Malawi';
		 $c['my'] = 'Malaysia';
		 $c['mv'] = 'Maldives';
		 $c['ml'] = 'Mali';
		 $c['mt'] = 'Malta';
		 $c['mh'] = 'Marshall Islands';
		 $c['mq'] = 'Martinique (French)';
		 $c['mr'] = 'Mauritania';
		 $c['mu'] = 'Mauritius';
		 $c['yt'] = 'Mayotte';
		 $c['mx'] = 'Mexico';
		 $c['fm'] = 'Micronesia';
		 $c['md'] = 'Moldavia';
		 $c['mc'] = 'Monaco';
		 $c['mn'] = 'Mongolia';
		 $c['me'] = 'Montenegro';
		 $c['ms'] = 'Montserrat';
		 $c['ma'] = 'Morocco';
		 $c['mz'] = 'Mozambique';
		 $c['mm'] = 'Myanmar';
		 $c['na'] = 'Namibia';
		 $c['nr'] = 'Nauru';
		 $c['np'] = 'Nepal';
		 $c['nl'] = 'Netherlands';
		 $c['an'] = 'Netherlands Antilles';
		 // $c['net'] = "Network";
		 $c['nc'] = 'New Caledonia (French)';
		 $c['nz'] = 'New Zealand';
		 $c['ni'] = 'Nicaragua';
		 $c['ne'] = 'Niger';
		 $c['ng'] = 'Nigeria';
		 $c['nu'] = 'Niue';
		 $c['nf'] = 'Norfolk Island';
		 $c['mp'] = 'Northern Mariana Islands';
		 $c['no'] = 'Norway';
		//  $c['arpa'] = "Old style Arpanet";
		 $c['om'] = 'Oman';
		 $c['pk'] = 'Pakistan';
		 $c['pw'] = 'Palau';
		 $c['pa'] = 'Panama';
		 $c['pg'] = 'Papua New Guinea';
		 $c['py'] = 'Paraguay';
		 $c['pe'] = 'Peru';
		 $c['ph'] = 'Philippines';
		 $c['pn'] = 'Pitcairn Island';
		 $c['pl'] = 'Poland';
		 $c['pf'] = 'Polynesia (French)';
		 $c['pt'] = 'Portugal';
		 $c['pr'] = 'Puerto Rico';
		 $c['ps'] = 'Palestine';
		 $c['qa'] = 'Qatar';
		 $c['re'] = 'Reunion (French)';
		 $c['ro'] = 'Romania';
		 $c['ru'] = 'Russia';
		 $c['rw'] = 'Rwanda';
		 $c['gs'] = 'S. Georgia & S. Sandwich Isls.';
		 $c['sh'] = 'Saint Helena';
		 $c['kn'] = 'Saint Kitts & Nevis';
		 $c['lc'] = 'Saint Lucia';
		 $c['pm'] = 'Saint Pierre and Miquelon';
		 $c['st'] = 'Saint Tome (Sao Tome) and Principe';
		 $c['vc'] = 'Saint Vincent & Grenadines';
		 $c['ws'] = 'Samoa';
		 $c['sm'] = 'San Marino';
		 $c['sa'] = 'Saudi Arabia';
		 $c['sn'] = 'Senegal';
		 $c['rs'] = 'Serbia';
		 $c['sc'] = 'Seychelles';
		 $c['sl'] = 'Sierra Leone';
		 $c['sg'] = 'Singapore';
		 $c['sk'] = 'Slovak Republic';
		 $c['si'] = 'Slovenia';
		 $c['sb'] = 'Solomon Islands';
		 $c['so'] = 'Somalia';
		 $c['za'] = 'South Africa';
		 $c['es'] = 'Spain';
		 $c['lk'] = 'Sri Lanka';
		 $c['sd'] = 'Sudan';
		 $c['sr'] = 'Suriname';
		 $c['sj'] = 'Svalbard and Jan Mayen Islands';
		 $c['sz'] = 'Swaziland';
		 $c['se'] = 'Sweden';
		 $c['ch'] = 'Switzerland';
		 $c['sy'] = 'Syria';
		 $c['tj'] = 'Tadjikistan';
		 $c['tw'] = 'Taiwan';
		 $c['tz'] = 'Tanzania';
		 $c['th'] = 'Thailand';
		 $c['ti'] = 'Tibet';
		 $c['tg'] = 'Togo';
		 $c['tk'] = 'Tokelau';
		 $c['to'] = 'Tonga';
		 $c['tt'] = 'Trinidad and Tobago';
		 $c['tn'] = 'Tunisia';
		 $c['tr'] = 'Turkey';
		 $c['tm'] = 'Turkmenistan';
		 $c['tc'] = 'Turks and Caicos Islands';
		 $c['tv'] = 'Tuvalu';
		 $c['ug'] = 'Uganda';
		 $c['ua'] = 'Ukraine';
		 $c['ae'] = 'United Arab Emirates';
		 $c['gb'] = 'United Kingdom';
		 $c['us'] = 'United States';
		 $c['uy'] = 'Uruguay';
		 $c['um'] = 'US Minor Outlying Islands';
		 $c['uz'] = 'Uzbekistan';
		 $c['vu'] = 'Vanuatu';
		 $c['va'] = 'Vatican City State';
		 $c['ve'] = 'Venezuela';
		 $c['vn'] = 'Vietnam';
		 $c['vg'] = 'Virgin Islands (British)';
		 $c['vi'] = 'Virgin Islands (USA)';
		 $c['wf'] = 'Wallis and Futuna Islands';
		 $c['eh'] = 'Western Sahara';
		 $c['ye'] = 'Yemen';
		// $c['zr'] = "(deprecated) Zaire";
		 $c['zm'] = 'Zambia';
		 $c['zw'] = 'Zimbabwe';
        if(!empty($iso) && !empty($c[$iso]))
        {
            return $c[$iso];
        }
		return ($iso === null) ? $c : '';
	}
	/**
	 * Get required field markup string
	 * @return string
	 */
	public function getRequiredString()
	{
		return $this->_required_string;
	}
	/**
	 * Set required field markup string
	 * @param string $string
	 * @return e_form
	 */
	public function setRequiredString($string)
	{
		$this->_required_string = $string;
		return $this;
	}
	
	// For Comma separated keyword tags.
	/**
	 * @param $name
	 * @param $value
	 * @param int $maxlength
	 * @param string|array $options
	 * @return string
	 */
	public function tags($name, $value, $maxlength = 200, $options = null)
	{
	  if(is_string($options))
	  {
		  parse_str($options, $options);
	  }
	  $defaults['selectize'] = array(
		'create'   => true,
		'maxItems' => vartrue($options['maxItems'], 7),
		'mode'     => 'multi',
		'plugins'  => array('remove_button'),
	  );
	  $options = array_replace_recursive($defaults, $options);
	  return $this->text($name, $value, $maxlength, $options);
	}
	/**
	 * Render Bootstrap Tabs
	 *
	 * @param $array
	 * @param $options
	 * @return string
	 * @example
	 *        $array = array(
	 *        'home' => array('caption' => 'Home', 'text' => 'some tab content' ),
	 *        'other' => array('caption' => 'Other', 'text' => 'second tab content' )
	 *        );
	 */
	public function tabs($array, $options = array())
	{
		$initTab = varset($options['active'],false);
		$id = !empty($options['id']) ? 'id="'.$options['id'].'"' : '';
		$text  ='
		
			';
		$c = 0;
		foreach($array as $key=>$tab)
		{
			if(is_numeric($key))
			{
				$key = 'tab-'.$key;
			}
			if($c == 0 & $initTab == false)
			{
				$initTab = $key;
			}
			
			$active = ($key ==$initTab) ? ' class="nav-item active"' : ' class="nav-item"';
			$text .= ''.$tab['caption'].'  ';
		$initTab = varset($options['active'],false);
		$tabClass = varset($options['class'],null);
		$text .= '
		
		';
		
		$c=0;
		foreach($array as $key=>$tab)
		{
			if(is_numeric($key))
			{
				$key = 'tab-'.$key;
			}
			if($c == 0 & $initTab == false)
			{
				$initTab = $key;
			}
			
			$active = ($key == $initTab) ? ' active' : '';
			$text .= '
'.$tab['text'].'
';
			$c++;
		}
		
		$text .= '
		
';
		if($indicate && (count($array) > 1))
		{
			$indicators = '
	        
	        
			';
			$c = 0;
			foreach($array as $key=>$tab)
			{
				$active = ($c == $act) ? ' class="active"' : '';
				$indicators .=  ' ';
		}
		$inner = '
		
		';
		
		$c=0;
		foreach($array as $key=>$tab)
		{
			$active = ($c == $act) ? ' active' : '';
			$label = !empty($tab['label']) ? ' data-label="'.$tab['label'].'"' : '';
			$inner .= '
';
			$inner .= $tab['text'];
			
			if(!empty($tab['caption']))
			{
				$inner .= '
'.$tab['caption'].'
';
			}
			
			$inner .= '
';
			$c++;
		}
		
		$inner .= '
		
';
		if($navigation && (count($array) > 1))
		{
			$controls = '
			
	         
			
			 ';
		}
		$end = '
".$label. ' '; // using bootstrap.
		if(!e107::getRegistry('core/form/mediaurl'))
		{
			e107::setRegistry('core/form/mediaurl', true);
		}
		return $ret;
	}
	/**
	 * Avatar Picker
	 * @param string $name - form element name ie. value to be posted.
	 * @param string $curVal - current avatar value. ie. the image-file name or URL.
	 * @param array $options
	 * @todo add a pref for allowing external or internal avatars or both.
	 * @return string
	 */
	public function avatarpicker($name, $curVal='', $options=array())
	{
		
		$tp 		= e107::getParser();
		$pref 		= e107::getPref();
		
		$attr 		= 'aw=' .$pref['im_width']. '&ah=' .$pref['im_height'];
		$tp->setThumbSize($pref['im_width'],$pref['im_height']);
		
		$blankImg 	= $tp->thumbUrl(e_IMAGE. 'generic/blank_avatar.jpg',$attr);
		$localonly 	= true;
		$idinput 	= $this->name2id($name);
		$previnput	= $idinput. '-preview';
		$optioni 	= $idinput. '-options';
		
		
		$path = (strpos($curVal,'-upload-') === 0) ? '{e_AVATAR}upload/' : '{e_AVATAR}default/';
		$newVal = str_replace('-upload-','',$curVal);
	
		$img = (strpos($curVal, '://')!==false) ? $curVal : $tp->thumbUrl($path.$newVal);
				
		if(!$curVal)
		{
			$img = $blankImg;	
		}
		
		$parm = $options;
		$classlocal = (!empty($parm['class'])) ? "class='".$parm['class']." e-expandit  e-tip avatar'" : " class='img-rounded rounded e-expandit e-tip avatar ";
		$class = (!empty($parm['class'])) ? "class='".$parm['class']." e-expandit '" : " class='img-rounded rounded btn btn-default btn-secondary button e-expandit ";
	
		if($localonly == true)
		{
			$text = "\n"; //TODO unique id. 
		$count = 0;
		if (!empty($pref['avatar_upload']) && FILE_UPLOADS && !empty($options['upload']))
		{
				$diz = LAN_USET_32.($pref['im_width'] || $pref['im_height'] ? "\n".str_replace(array('[x]-','[y]'), array($pref['im_width'], $pref['im_height']), LAN_USER_86) : '');
	
				$text .= "
".LAN_USET_26."
				
";
				
				if(count($avFiles) > 0)
				{
					$text .= "
".LAN_EFORM_003. ' 
';
					$count = 1;
				}
		}
		
		foreach($avFiles as $fi)
		{
			$img_path = $tp->thumbUrl(e_AVATAR_DEFAULT.$fi['fname']);	
			$text .= "\n
 ";
			$count++;
			//TODO javascript CSS selector
		}
		
		if($count == 0)
		{
			$text .= "
";
			$text .= "
".LAN_EFORM_005. '
';
			if(ADMIN)
			{
				$EAVATAR = e_AVATAR_DEFAULT;
				$text .= "
";
				$text .= e107::getParser()->lanVars(e107::getParser()->toHTML(LAN_EFORM_006, true), array('x'=>$EAVATAR));
				$text .= '
';
			}
			$text .= '
';
		}
		
		
		$text .= '
		
".LAN_SIGNUP_25."  ".LAN_SIGNUP_34."
";
			}
		
			if (false && $pref['photo_upload'] && FILE_UPLOADS)
			{
				$text .= "".LAN_SIGNUP_26."  ".LAN_SIGNUP_34."
";
			}  */
	}
	/**
	 * Image Picker
	 *
	 * @param string $name          input name
	 * @param string $default       default value
	 * @param string $previewURL
	 * @param string $sc_parameters shortcode parameters
	 *                              --- SC Parameter list ---
	 *                              - media: if present - load from media category table
	 *                              - w: preview width in pixels
	 *                              - h: preview height in pixels
	 *                              - help: tooltip
	 *                              - video: when set to true, will enable the Youtube  (video) tab.
	 * @return string html output
	 * @example $frm->imagepicker('banner_image', $_POST['banner_image'], '', 'banner'); // all images from category 'banner_image' + common images.
	 * @example $frm->imagepicker('banner_image', $_POST['banner_image'], '', 'media=banner&w=600');
	 */
	public function imagepicker($name, $default, $previewURL = '', $sc_parameters = '')
	{
	//	$tp = e107::getParser();
	//	$name_id = $this->name2id($name);
	//	$meta_id = $name_id."-meta";
		
		if(is_string($sc_parameters))
		{
			if(strpos($sc_parameters, '=') === false)
			{
				$sc_parameters = 'media=' . $sc_parameters;
			}
			parse_str($sc_parameters, $sc_parameters);
		}
		elseif(empty($sc_parameters))
		{
			$sc_parameters = array();
		}
	//	$cat = $tp->toDB(vartrue($sc_parameters['media']));
		// v2.2.0
		unset($previewURL );
		$sc_parameters['image'] = 1;
		$sc_parameters['dropzone'] = 1;
		if(!empty($sc_parameters['video'])) // bc fix
		{
			$sc_parameters['youtube'] = 1;
		}
		return $this->mediapicker($name, $default, $sc_parameters);
	}
/**
	 * Media Picker
 *
	 * @param string $name input name
	 * @param string $default default value
	 * @param string $parms shortcode parameters
	 *  --- $parms list ---
	 * - media: if present - load from media category table
	 * - w: preview width in pixels
	 * - h: preview height in pixels
	 * - help: tooltip
	 * - youtube=1 (Enables the Youtube tab)
     * - image=1 (Enable the Images tab)
	 * - video=1 (Enable the Video tab)
	 * - audio=1  (Enable the Audio tab)
     * - glyph=1 (Enable the Glyphs tab).
     * - path=plugin (store in media/plugins/{current-plugin])
     * - edit=false (disable media-manager popup button)
	 * - rename (string) rename file to this value after upload.  (don't forget the extension)
	 * - resize array with numberic x values. (array 'w'=>x, 'h'=>x)  - resize the uploaded image before importing during upload.
     * - convert=jpg (override pref and convert uploaded image to jpeg format. )
	 * @return string html output
	 *@example $frm->imagepicker('banner_image', $_POST['banner_image'], '', 'media=banner&w=600');
	 */
	public function mediapicker($name, $default, $parms = '')
	{
		$tp = e107::getParser();
		$name_id = $this->name2id($name);
		$meta_id = $name_id. '-meta';
		if(is_string($parms))
		{
			if(strpos($parms, '=') === false)
			{
				$parms = 'media=' . $parms;
			}
			parse_str($parms, $parms);
		}
		elseif(empty($parms))
		{
			$parms = array();
		}
		if(empty($parms['media']))
		{
			$parms['media'] = '_common';
		}
		$title = !empty($parms['help']) ? "title='".$parms['help']."'" : '';
		if(!isset($parms['w']))
		{
			$parms['w'] = 206;
		}
		if(!isset($parms['h']))
		{
			$parms['h'] = 190; // 178
		}
	//	$width = vartrue($parms['w'], 220);
	//	$height = vartrue($parms['h'], 190);
	// e107::getDebug()->log($parms);
		// Test Files...
	//	$default = '{e_MEDIA_VIDEO}2018-07/samplevideo_720x480_2mb.mp4';
	//	$default = '{e_MEDIA_FILE}2016-03/Colony_Harry_Gregson_Williams.mp3';
	//	$default = '{e_PLUGIN}gallery/images/butterfly.jpg';
	//	$default = 'NuIAYHVeFYs.youtube';
	//	$default = ''; // empty
	//	$default = '{e_MEDIA_IMAGE}2018-07/Jellyfish.jpg';
		$class = '';
		if(!empty($parms['icon']))
		{
			$class = 'icon-preview mediaselector-container-icon';
			$parms['type'] = 'icon';
		}
		$preview = e107::getMedia()->previewTag($default,$parms);
		$cat = $tp->toDB(vartrue($parms['media']));
		$ret = "\n";
		$ret .=	"".basename($default_label). ' ';
			
		$sc_parameters['mode'] 		= 'main';
		$sc_parameters['action'] 	= 'dialog';	
			
		
	//	$ret .= $this->mediaUrl($cat, $label,$name_id,"mode=dialog&action=list");
		$ret .= $this->mediaUrl($cat, $label,$name_id,$sc_parameters);
	
		
	
		
		return $ret;
	
		
	}
	/**
	 *	Date field with popup calendar // NEW in 0.8/2.0
	 * on Submit returns unix timestamp or string value.
	 * @param string $name the name of the field
	 * @param int|bool $datestamp UNIX timestamp - default value of the field
	 * @param array|string {
	 *      @type string mode date or datetime
	 *      @type string format strftime format eg. '%Y-%m-%d'
	 *      @type string timezone eg. 'America/Los_Angeles' - intended timezone of the date/time entered. (offsets UTC value)
	 *      }
	 * @example $frm->datepicker('my_field',time(),'mode=date');
	 * @example $frm->datepicker('my_field',time(),'mode=datetime&inline=1');
	 * @example $frm->datepicker('my_field',time(),'mode=date&format=yyyy-mm-dd');
	 * @example $frm->datepicker('my_field',time(),'mode=datetime&format=MM, dd, yyyy hh:ii');
	 * @example $frm->datepicker('my_field',time(),'mode=datetime&return=string');
	 * 
	 * @url http://trentrichardson.com/examples/timepicker/
	 * @return string
	 */
	public function datepicker($name, $datestamp = false, $options = null)
	{
		if(!empty($options) && is_string($options))
		{
			parse_str($options,$options);
		}
		$mode = !empty($options['mode']) ? trim($options['mode']) : 'date'; // OR  'datetime'
		if(!empty($options['type'])) /** BC Fix. 'type' is @deprecated */
		{
			$mode = trim($options['type']);
		}
		$dateFormat  = !empty($options['format']) ? trim($options['format']) :e107::getPref('inputdate', '%Y-%m-%d');
		$ampm		 = (preg_match('/%l|%I|%p|%P/',$dateFormat)) ? 'true' : 'false';
		$value		 = null;
		$hiddenValue = null;
		$useUnix     = (isset($options['return']) && ($options['return'] === 'string')) ? 'false' : 'true';
		$id          = !empty($options['id']) ? $options['id'] : $this->name2id($name);
		$classes     = array('date' => 'tbox e-date', 'datetime' => 'tbox e-datetime');
		if($mode === 'datetime' && !varset($options['format']))
		{
			$dateFormat .= ' ' .e107::getPref('inputtime', '%H:%M:%S');
		}
		$dformat = e107::getDate()->toMask($dateFormat);
		// If default value is set.
		if ($datestamp && $datestamp !='0000-00-00') // date-field support.
		{
			if(!is_numeric($datestamp))
			{
				$datestamp = strtotime($datestamp);
			}
			// Convert date to proper (selected) format.
			$value = e107::getDate()->convert_date($datestamp, $dformat);
			$hiddenValue = $value;
			if ($useUnix === 'true')
			{
				$hiddenValue = $datestamp;
			}
		}
		$class 		= (isset($classes[$mode])) ? $classes[$mode] : 'tbox e-date';
		$size 		= !empty($options['size']) ? (int) $options['size'] : 40;
		$required 	= !empty($options['required']) ? 'required' : '';
		$firstDay	= isset($options['firstDay']) ? $options['firstDay'] : 0;
		$xsize		= (!empty($options['size']) && !is_numeric($options['size'])) ? $options['size'] : 'xlarge';
		$disabled 	= !empty($options['disabled']) ? 'disabled' : '';
		$placeholder = !empty($options['placeholder']) ? 'placeholder="'.$options['placeholder'].'"' : '';
		$timezone    = '';
		if(!empty($options['timezone'])) // since datetimepicker does not support timezones and assumes the browser timezone is the intended timezone.
		{
			date_default_timezone_set($options['timezone']);
			$targetOffset = date('Z');
			date_default_timezone_set(USERTIMEZONE);
			$timezone = "data-date-timezone-offset='".$targetOffset."'";
		}
		$text = '';
		if(!empty($options['inline']))
		{
			$text .= "
";
			$text .= "'.LAN_GENERATE.'  ';
			
			if(empty($options['nomask']))
			{
				$gen .= ''.LAN_SHOW.'  
";
		}
		
		$options['pattern'] = vartrue($options['pattern'],'[\S].{2,}[\S]');
		$options['required'] = varset($options['required'], 1);
		$options['class'] = vartrue($options['class'],'e-password tbox');
		e107::js('core', 	'password/jquery.pwdMeter.js', 'jquery', 2);
		e107::js('footer-inline', '
			$(".e-password").pwdMeter({
	            minLength: 6,
	            displayGeneratePassword: true,
	            generatePassText: "Generate",
	            randomPassLength: 12
	        });
	    ');
		if(deftrue('BOOTSTRAP'))
		{
			$options['class'] .= ' form-control';
		}
		
		if(!empty($options['size']) && !is_numeric($options['size']))
		{
			$options['class'] .= ' input-' .$options['size'];
			unset($options['size']); // don't include in html 'size='. 	
		}
		
		$type = empty($options['nomask']) ? 'password' : 'text';
		
		$options = $this->format_options('text', $name, $options);
	
		//never allow id in format name-value for text fields
		$text = "
get_attributes($options, $name). ' />';
		if(empty($gen) && empty($addon))
		{
			return $text;	
		}
		return "
".$text.$gen. ' ' .vartrue($addon);
	}
	/**
	 * Render Pagination using 'nextprev' shortcode.
	 * @param string $url eg. e_REQUEST_SELF.'?from=[FROM]'
	 * @param int $total total records
	 * @param int $from value to replace [FROM] with in the URL
	 * @param int $perPage number of items per page
	 * @param array $options template, type, glyphs
	 * @return string
	 */
	public function pagination($url='', $total=0, $from=0, $perPage=10, $options=array())
	{
		if(empty($total) || empty($perPage))
		{
			return '';
		}
		
		if(defined('BOOTSTRAP') && BOOTSTRAP === 4)
		{
			return '';
		}
		if(!is_numeric($total))
		{
			return '';
		}
		require_once(e_CORE. 'shortcodes/single/nextprev.php');
		$nextprev = array(
			'tmpl_prefix'	=> varset($options['template'],'default'),
			'total'			=> (int) $total,
			'amount'		=> (int) $perPage,
			'current'		=> (int) $from,
			'url'			=> urldecode($url),
			'type'          => varset($options['type'],'record'), // page|record
			'glyphs'        => vartrue($options['glyphs'],false) // 1|0
		);
	//	e107::getDebug()->log($nextprev);
		return nextprev_shortcode($nextprev);
	}
	/**
	 * Render a bootStrap ProgressBar.
	 *
	 * @param string        $name
	 * @param number|string $value
	 * @param array         $options
	 * @return string
	 * @example  Use
	 */
	public function progressBar($name,$value,$options=array())
	{
		if(!deftrue('BOOTSTRAP')) // Legacy ProgressBar.
		{
			$barl = (file_exists(THEME.'images/barl.png') ? THEME_ABS.'images/barl.png' : e_PLUGIN_ABS.'poll/images/barl.png');
			$barr = (file_exists(THEME.'images/barr.png') ? THEME_ABS.'images/barr.png' : e_PLUGIN_ABS.'poll/images/barr.png');
			$bar = (file_exists(THEME.'images/bar.png') ? THEME_ABS.'images/bar.png' : e_PLUGIN_ABS.'poll/images/bar.png');
			return "
			
			
";
		}
			
		$class = vartrue($options['class']);
		$target = $this->name2id($name);
		
		$striped = (vartrue($options['btn-label'])) ? ' progress-striped active' : '';	
		if(strpos($value,'/')!==false)
		{
			$label = $value;
			list($score,$denom) = explode('/',$value);
		//	$multiplier = 100 / (int) $denom;
			$value = ((int) $score / (int) $denom) * 100;
		//	$value = (int) $score * (int) $multiplier;
			$percVal = round((float) $value).'%';
		}
		else
		{
			$percVal = round((float) $value).'%';
			$label = $percVal;
		}
		if(!empty($options['label']))
		{
			$label = $options['label'];
		}
		$id = !empty($options['id']) ? "id='".$options['id']."'" : '';
		$text =	"
   		 	";
   		$text .= $label;
   		 	$text .= '
    	 ';
		
		$loading = vartrue($options['loading'], defset('LAN_LOADING', 'Loading'));
		
		$buttonId = $target.'-start';
		
		
		
		if(!empty($options['btn-label']))
		{
			$interval = vartrue($options['interval'],1000);
			$text .= '
'.$options['btn-label'].' ';
			$text .= ' 
'.LAN_CANCEL.' ';
		}
		
		
		return $text;
		
	}
	/**
	 * Textarea Element
	 * @param string $name
	 * @param string $value
	 * @param int $rows
	 * @param int $cols
	 * @param array|string $options
	 * @param int|bool $counter
	 * @return string
	 */
	public function textarea($name, $value, $rows = 10, $cols = 80, $options = null, $counter = false)
	{
		if(is_string($options))
		{
			 parse_str($options, $options);
		}
		else
		{
			$options = (array) $options;
		}
		// auto-height support
		if(empty($options['class']))
		{
			$options['class'] = '';
		}
		if(!empty($options['size']) && !is_numeric($options['size']))
		{
			$options['class'] .= ' form-control input-' .$options['size'];
			unset($options['size']); // don't include in html 'size='. 	
		}
		elseif(empty($options['noresize']))
		{
			$options['class'] = (isset($options['class']) && $options['class']) ? $options['class'].' e-autoheight' : 'tbox col-md-7 span7 e-autoheight form-control';
		}
		$options = $this->format_options('textarea', $name, $options);
		
//		print_a($options);
		//never allow id in format name-value for text fields
		return "
".($counter !== false ? $this->hidden('__'.$name.'autoheight_opt', $counter) : '');
	}
	/**
	 * Bbcode Area. Name, value, template, media-Cat, size, options array eg. counter
	 * IMPORTANT: $$mediaCat is also used is the media-manager category identifier
	 *
	 * @param string $name
	 * @param mixed  $value
	 * @param string $template
	 * @param string $mediaCat _common
	 * @param string $size     : small | medium | large
	 * @param array  $options  {
	 *  @type bool wysiwyg  when set to false will disable wysiwyg if active.
	 *  @type string class override class.
	 *                         }
	 * @return string
	 */
	public function bbarea($name, $value='', $template = '', $mediaCat='_common', $size = 'large', $options = array())
	{
		if(is_string($options))
		{
			parse_str($options, $options);
		}
		//size - large|medium|small
		//width should be explicit set by current admin theme
	//	$size = 'input-large';
		$height = '';
		$cols = 70;
		
		switch($size)
		{
			case 'tiny':
				$rows = '3';
				$cols = 50;
			//	$height = "style='height:250px'"; // inline required for wysiwyg
			break;
			
			
			case 'small':
				$rows = '7';
				$height = "style='height:230px'"; // inline required for wysiwyg
				$size = 'input-block-level';
			break;
						
			case 'medium':
				$rows = '10';
               
				$height = "style='height:375px'"; // inline required for wysiwyg
				$size = 'input-block-level';
			break;
			case 'large':
			default:
				$rows = '20';
				$size = 'large input-block-level';
			//	$height = "style='height:500px;width:1025px'"; // inline required for wysiwyg
			break;
		}
		// auto-height support
/*
		$bbbar 				= '';
		$wysiwyg = null;
		$wysiwygClass = ' e-wysiwyg';
		if(isset($options['wysiwyg']))
		{
			$wysiwyg = $options['wysiwyg'];
		}
		if($wysiwyg === false)
		{
			$wysiwygClass = '';
		}
		$options['class'] 	= 'tbox bbarea '.($size ? ' '.$size : '').$wysiwygClass.' e-autoheight form-control';
*/
		$options['class'] 	= 'tbox bbarea '.($size ? ' '.$size : '').' e-wysiwyg e-autoheight form-control';
		if (isset($options['id']) && !empty($options['id']))
		{
			$help_tagid 		= $this->name2id($options['id']). '--preview';
		}
		else
		{
			$help_tagid 		= $this->name2id($name). '--preview';
		}
		if (!isset($options['wysiwyg']))
		{
			$options['wysiwyg'] = true;
		}
		//if(e107::wysiwyg(true) === false || $wysiwyg === false) // bbarea loaded, so activate wysiwyg (if enabled in preferences)
		if(e107::wysiwyg($options['wysiwyg'],true) === 'bbcode') // bbarea loaded, so activate wysiwyg (if enabled in preferences)
		{
			$options['other'] 	= "onselect='storeCaret(this);' onclick='storeCaret(this);' onkeyup='storeCaret(this);' {$height}";
		}
		else
		{
			$options['other'] 	= ' ' .$height;
		}
		$counter 			= vartrue($options['counter'],false); 
		
		$ret = "
		
\n";
		if(e107::wysiwyg() === true) // && $wysiwyg !== false)
		{
			$eParseList = e107::getConfig()->get('e_parse_list');
			if(!empty($eParseList))
			{
				$opts = array(
					'field' => $name,
				);
				foreach($eParseList as $plugin)
				{
					$hookObj = e107::getAddon($plugin, 'e_parse');
					if($tmp = e107::callMethod($hookObj, 'toWYSIWYG', $value, $opts))
					{
						$value = $tmp;
					}
				}
			}
		}
		$ret .=	e107::getBB()->renderButtons($template,$help_tagid);
		$ret .=	$this->textarea($name, $value, $rows, $cols, $options, $counter); // higher thank 70 will break some layouts.
			
		$ret .= "
 \n";
		
		$_SESSION['media_category'] = $mediaCat; // used by TinyMce. 
	
		
		return $ret;
		
		// Quick fix - hide TinyMCE links if not installed, dups are handled by JS handler
		/*
		
				e107::getJs()->footerInline("
						if(typeof tinyMCE === 'undefined')
						{
							\$$('a.e-wysiwyg-switch').invoke('hide');
						}
				");
		*/
		
		
	}
	/**
	 * Checks for a theme snippet and returns it
	 * @param string $type
	 * @return string|false
	 */
	private function getSnippet($type)
	{
		if($this->_snippets === false || deftrue('e_ADMIN_AREA'))
		{
			return false;
		}
		$regId = 'core/form/snippet/'.$type;
		if($snippet = e107::getRegistry($regId))
		{
			return $snippet;
		}
		$snippetPath = THEME. 'snippets/form_' .$type. '.html';
		if(!file_exists($snippetPath))
		{
			return false;
		}
		$content =  file_get_contents($snippetPath, false, null, 0, 1024);
		e107::setRegistry($regId, $content);
		return $content;
	}
	private function renderSnippet($snippet, $options, $name, $value)
	{
		$snip  = (array) $options;
		if(!empty($options['class']))
		{
			$snip['class'] = trim($options['class']);
			unset($options['class']);
		}
		if(!empty($options['label']))
		{
			$snip['label'] = trim($options['label']);
			unset($options['label']);
		}
		$snip['id'] = $this->_format_id(varset($options['id']), $name, $value, null);
		unset($options['id']);
		$snip['attributes'] = "name='".$name."' value='".$value."' ".$this->get_attributes($options, $name, $value);
		foreach($snip as $k=>$v)
		{
			$search[] = '{'.$k.'}';
		}
		return str_replace($search, array_values($snip), $snippet);
	}
	/**
	* Render a checkbox 
	* @param string $name
	* @param mixed $value
	* @param boolean $checked
	* @param mixed $options query-string or array or string for a label. eg. label=Hello&foo=bar or array('label'=>Hello') or 'Hello'
	* @return string
	*/
	public function checkbox($name, $value, $checked = false, $options = array())
	{
		if(!is_array($options))
		{
			if(strpos($options, '=')!==false)
			{
			 	parse_str($options, $options);
			}
			elseif(is_string($options))
			{
				$options = array('label'=>$options);
			}
		}
		$labelClass = (!empty($options['inline'])) ? 'checkbox-inline' : 'checkbox form-check';
		$labelTitle = '';
		$options = $this->format_options('checkbox', $name, $options);
		
		$options['checked'] = $checked; //comes as separate argument just for convenience
		
		$text = '';
		$active = ($checked === true) ? ' active' : ''; // allow for styling if needed.
		if(!empty($options['label'])) // add attributes to 
		{
			if(!empty($options['title']))
			{
				$labelTitle = ' title="' .$options['title']. '"';
				unset($options['title']);
			}
			if(!empty($options['class']))
			{
				$labelClass .= ' ' .$options['class'];
				unset($options['class']);
			}
		}
		if(!isset($options['class']))
        {
            $options['class'] = '';
        }
        $options['class'] .= ' form-check-input';
		if($snippet = $this->getSnippet('checkbox'))
		{
			return $this->renderSnippet($snippet, $options, $name, $value);
		}
		$pre = (!empty($options['label'])) ? "" : ''; // Bootstrap compatible markup
		$post = (!empty($options['label'])) ? '' .$options['label']. '  ' : '';
		unset($options['label']); // not to be used as attribute;
		
		return $pre. "".implode('',$text). '
';
		}
		return $text;
		
	}
	
	public function checkbox_label($label_title, $name, $value, $checked = false, $options = array())
	{
		return $this->checkbox($name, $value, $checked, $options).$this->label($label_title, $name, $value);
	}
	public function checkbox_switch($name, $value, $checked = false, $label = '')
	{
		return $this->checkbox($name, $value, $checked).$this->label($label ?: LAN_ENABLED, $name, $value);
	}
	public function checkbox_toggle($name, $selector = 'multitoggle', $id = false, $label='') //TODO Fixme - labels will break this. Don't use checkbox, use html.
	{
		$selector = 'jstarget:'.$selector;
		if($id)
		{
			$id = $this->name2id($id);
		}
		return $this->checkbox($name, $selector, false, array('id' => $id,'class' => 'checkbox checkbox-inline toggle-all','label'=>$label));
	}
	public function uc_checkbox($name, $current_value, $uc_options, $field_options = array())
	{
		if(!is_array($field_options))
		{
			parse_str($field_options, $field_options);
		}
		return '
			
				'.$this->_uc->vetted_tree($name, array($this, '_uc_checkbox_cb'), $current_value, $uc_options, $field_options).'
			
		';
	}
	/**
	 *	Callback function used with $this->uc_checkbox
	 *
	 *	@see user_class->select() for parameters
	 */
	public function _uc_checkbox_cb($treename, $classnum, $current_value, $nest_level, $field_options)
	{
		if($classnum == e_UC_BLANK)
		{
			return '';
		}
		if (!is_array($current_value))
		{
			$tmp = explode(',', $current_value);
		}
		$classIndex = abs($classnum);			// Handle negative class values
		$classSign = (strpos($classnum, '-') === 0) ? '-' : '';
		$style = '';
		$class = $style;
		if($nest_level == 0)
		{
			$class = ' strong';
		}
		else
		{
			$style = " style='text-indent:" . (1.2 * $nest_level) . "em'";
		}
		$descr = varset($field_options['description']) ? ' ('.$this->_uc->getDescription($classnum).') ' : '';
		return "".$this->checkbox($treename.'[]', $classnum, in_array($classnum, $tmp), $field_options).$this->label($this->_uc->getName($classIndex).$descr, $treename.'[]', $classnum)."
\n";
	}
	public function uc_label($classnum)
	{
		return $this->_uc->getName($classnum);
	}
	/**
	 * A Radio Button Form Element
	 * @param $name
	 * @param @value array pair-values|string - auto-detected. 
	 * @param $checked boolean
	 * @param $options 
	 */
	public function radio($name, $value, $checked = false, $options = null)
	{
		if(!is_array($options))
		{
			parse_str($options, $options);
		}
		
		if(is_array($value))
		{
			return $this->radio_multi($name, $value, $checked, $options);
		}
		
		$labelFound = vartrue($options['label']);
		unset($options['label']); // label attribute not valid in html5
		$options = $this->format_options('radio', $name, $options);
		$options['checked'] = $checked; //comes as separate argument just for convenience
		if(empty($options['id']))
		{
			unset($options['id']);
		}
		if($snippet = $this->getSnippet('radio'))
		{
			$options['label'] = $labelFound;
			return $this->renderSnippet($snippet, $options, $name, $value);
		}
		// $options['class'] = 'inline';	
		$text = '';
	//	return print_a($options,true);
		if($labelFound) // Bootstrap compatible markup
		{
			$defaultClass = (deftrue('BOOTSTRAP')) ? 'radio-inline form-check-inline' : 'radio inline';
			$dis = (!empty($options['disabled'])) ? ' disabled' : '';
			$text .= "";
			
		}
		
		
		$text .= "".$options['help']. '
';
		}
		
		if($labelFound)
		{
			$text .= ' ' .$labelFound. '  ';
		}
		
		return $text;
	}
	/**
	 * Boolean Radio Buttons / Checkbox (with Bootstrap Switch).
	 *
	 * @param string $name
	 *  Form element name.
	 * @param bool $checked_enabled
	 *  Use the checked attribute or not.
	 * @param string $label_enabled
	 *  Default is LAN_ENABLED
	 * @param string $label_disabled
	 *  Default is LAN_DISABLED
	 * @param array|string $options
	 *  - 'inverse' => 1 (invert values)
	 *  - 'reverse' => 1 (switch display order)
	 *  - 'switch'  => 'normal' (size for Bootstrap Switch... mini, small, normal, large)
	 *
	 * @return string $text
	 */
	public function radio_switch($name, $checked_enabled = false, $label_enabled = '', $label_disabled = '', $options = null)
	{
		if(is_string($options))
		{
			parse_str($options, $options);
		}
		$options_on = varset($options['enabled'], array());
		$options_off = varset($options['disabled'], array());
		unset($options['enabled'], $options['disabled']);
		$options_on = array_merge($options_on, $options);
		$options_off = array_merge($options_off, $options);
		if(!empty($options['expandit']) || vartrue($options['class']) === 'e-expandit' ) // See admin->prefs 'Single Login' for an example.
		{
			$options_on = array_merge($options, array('class' => 'e-expandit-on'));
			$options_off = array_merge($options, array('class' => 'e-expandit-off'));
		}
		if(deftrue('e_ADMIN_AREA'))
		{
			$options['switch'] = 'small';
			$label_enabled = ($label_enabled) ? strtoupper($label_enabled) : strtoupper(LAN_ON);
			$label_disabled = ($label_disabled) ?  strtoupper($label_disabled): strtoupper(LAN_OFF);
		}
		$options_on['label'] = $label_enabled ? defset($label_enabled, $label_enabled) : LAN_ENABLED;
		$options_off['label'] = $label_disabled ? defset($label_disabled, $label_disabled) : LAN_DISABLED;
		if (!empty($options['switch']))
		{
			return $this->flipswitch($name,$checked_enabled, array('on'=>$options_on['label'],'off'=>$options_off['label']),$options);
		}
		if(!empty($options['inverse']))
		{
			$text = $this->radio($name, 0, !$checked_enabled, $options_on) . ' 	' . $this->radio($name, 1, $checked_enabled, $options_off);
		}
		elseif(!empty($options['reverse'])) // reverse display order.
		{
			$text = $this->radio($name, 0, !$checked_enabled, $options_off) . ' ' . $this->radio($name, 1, $checked_enabled, $options_on);
		}
		else
		{
			$text = $this->radio($name, 1, $checked_enabled, $options_on) . ' 	' . $this->radio($name, 0, !$checked_enabled, $options_off);
		}
		return $text;
	}
	/**
	 * @param string $name
	 * @param bool|false $checked_enabled
	 * @param array $labels on & off
	 * @param array $options
	 * @return string
	 */
	public function flipswitch($name, $checked_enabled = false, $labels=null, $options = array())
	{
		if(empty($labels))
		{
			$labels = array('on' =>strtoupper(LAN_ON), 'off' =>strtoupper(LAN_OFF));
		}
		$value = $checked_enabled;
		if(!empty($options['inverse']))
		{
			$checked_enabled = !$checked_enabled;
		}
		if(!empty($options['reverse']))
		{
			$on = $labels['on'];
			$options_on['label'] = $labels['off'];
			$options_off['label'] = $on;
			unset($on);
		}
		if(empty($options['switch']))
		{
			$options['switch'] = 'small';
		}
		$switchName = $this->name2id($name) . '__switch'; // fixes array names.
		$switchAttributes = array(
			'data-type'    => 'switch',
			'data-name'    => $name,
			'data-size'    => $options['switch'],
			'data-on'      => $labels['on'],
			'data-off'     => $labels['off'],
			'data-inverse' => (int) !empty($options['inverse']),
		);
		$options += $switchAttributes;
		if(deftrue('e_ADMIN_AREA'))
		{
			$options['data-wrapper'] = 'wrapper form-control';
		}
		e107::library('load', 'bootstrap.switch');
		e107::js('footer', '{e_WEB}js/bootstrap.switch.init.js', 'jquery', 5);
		$text = $this->hidden($name, (int) $value);
		$text .= $this->checkbox($switchName, (int) $checked_enabled, $checked_enabled, $options);
		return $text;
	}
	/**
	 * XXX INTERNAL ONLY - Use radio() instead. array will automatically trigger this internal method.  
	 * @param string $name 
	 * @param array|string $elements = arrays value => label
	 * @param string/integer $checked = current value
	 * @param boolean $multi_line
	 * @param mixed $help array of field help items or string of field-help (to show on all)
	 */
	private function radio_multi($name, $elements, $checked, $options=array(), $help = null)
	{
		
		
		
		/* // Bootstrap Test. 
		 return'    
     
     
    
     
    
     ';
		*/
		
		
		$text = array();
				
		if(is_string($elements))
		{
			parse_str($elements, $elements);
		}
		if(!is_array($options))
		{
			parse_str($options, $options);
		}
		if(!empty($options['help']))
		{
			$help = "".$options['help']. '
';
			unset($options['help']);
		}
		
		foreach ($elements as $value => $label)
		{
			$label = defset($label, $label);
			
			$helpLabel = (is_array($help)) ? vartrue($help[$value]) : $help;
		
		// Bootstrap Style Code - for use later. 	
			$options['label'] = $label;
			$options['help'] = $helpLabel;
			$text[] = $this->radio($name, $value, (string) $checked === (string) $value, $options);
	
		//	$text[] = $this->radio($name, $value, (string) $checked === (string) $value)."".$this->label($label, $name, $value).(isset($helpLabel) ? "".$helpLabel."
" : '');
		}
		
	//	if($multi_line === false)
	//	{
		//	return implode("  ", $text);
	//	}
		
		// support of UI owned 'newline' parameter
		if(!varset($options['sep']) && vartrue($options['newline']))
		{
			$options['sep'] = '".implode("
", $text)."
";
	}
	/**
	 * Just for BC - use the $options['label'] instead. 
	 */
	public function label($text, $name = '', $value = '')
	{
	//	$backtrack = debug_backtrace(DEBUG_BACKTRACE_IGNORE_ARGS,2); 
	//	e107::getMessage()->addDebug("Deprecated \$frm->label() used in: ".print_a($backtrack,true));
		$for_id = $this->_format_id('', $name, $value, 'for');
		return "{$text} ";
	}
	
	public function help($text)
	{
		if(empty($text) || $this->_helptip === 0)
		{
			return null;
		}
		$ret = '';
		$ret .= ''.defset($text,$text).'
'; // display:none to prevent visibility during page load. 
		return $ret;
	}
	public function select_open($name, $options = array())
	{
		if(!is_array($options))
		{
			parse_str($options, $options);
		}
		if(!empty($options['size']) && !is_numeric($options['size']))
		{
			if(!empty($options['class']))
			{
				$options['class'] .= ' form-control input-' .$options['size'];
			}
			else
			{
				$options['class'] = 'form-control input-' .$options['size'];
			}
			unset($options['size']); // don't include in html 'size='. 	
		}
        if(!empty($options['title']) && is_array($options['title']))
        {
            unset($options['title']);
        }
		$options = $this->format_options('select', $name, $options);
	
		return "get_attributes($options, $name). '>';
	}
	/**
	 * @deprecated - use select() instead.
	 */
	public function selectbox($name, $option_array, $selected = false, $options = array(), $defaultBlank= false)
	{
		trigger_error(''.__METHOD__.' is deprecated.  Use select() instead.', E_USER_DEPRECATED); // NO LAN
		return $this->select($name, $option_array, $selected, $options, $defaultBlank);	
	}
	/**
	 *
	 * @param string        $name
	 * @param array|string  $option_array
	 * @param boolean       $selected [optional]
	 * @param string|array  $options [optional]
	 * @param bool          $options['useValues']   when true uses array values as the key.
	 * @param array         $options['disabled'] list of $option_array keys which should be disabled. eg. array('key_1', 'key_2');
	 * @param bool|string   $defaultBlank [optional] set to TRUE if the first entry should be blank, or to a string to use it for the blank description.
	 * @return string       HTML text for display
	 */
	public function select($name, $option_array, $selected = false, $options = array(), $defaultBlank= false)
	{
		if(!is_array($options))
		{
			parse_str($options, $options);
		}
		if($option_array === 'yesno')
		{
			$option_array = array(1 => LAN_YES, 0 => LAN_NO);
		}
		if(!empty($options['multiple']))
		{
			$name = (strpos($name, '[') === false) ? $name.'[]' : $name;
			if(!is_array($selected))
			{
				$selected = explode(',', $selected);
			}
		}
		$text = $this->select_open($name, $options)."\n";
		if(isset($options['default']))
		{
			if($options['default'] === 'blank')
			{
				$options['default'] = ' ';			
			}
			$text .= $this->option($options['default'], varset($options['defaultValue']));
		}
		elseif($defaultBlank)
		{
			$diz = is_string($defaultBlank) ? $defaultBlank : ' ';
			$text .= $this->option($diz, '');
		}
		
		if(!empty($options['useValues'])) // use values as keys.
		{
			$new = array();
			foreach($option_array as $v)
			{
				$new[$v] = (string) $v;
			}	
			$option_array = $new;	
		}
		$text .= $this->option_multi($option_array, $selected, $options)."\n".$this->select_close();
		return $text;
	}
	
	
	
	
	
	/**
	 * Universal Userclass selector - checkboxes, dropdown, everything. 
	 * @param string $name - form element name
	 * @param int $curval - current userclass value(s) as array or comma separated.
	 * @param string $type - checkbox|dropdown  default is dropdown.
	 * @param string|array $options - classlist or query string or key=value pair.
	 * @param string $options['options'] comma-separated list of display options. 'options=admin,mainadmin,classes&vetted=1&exclusions=0' etc.
	 *
	 * @example $frm->userclass('name', 0, 'dropdown', 'classes'); // display all userclasses
	 * @example $frm->userclass('name', 0, 'dropdown', 'classes,matchclass'); // display only classes to which the user belongs.
	 * @return string form element(s)
	 */
	public function userclass($name, $curval=255, $type=null, $options=null)
	{
		if(!empty($options))
		{
			if(is_array($options))
			{
				$opt = $options;
			}
			elseif(strpos($options,'=')!==false)
			{
				parse_str($options,$opt);
			}
			else
			{
				$opt = array('options'=>$options);
			}
		}
		else
		{
			$opt = array();
		}
		$optlist = vartrue($opt['options'],null);
		switch ($type)
		{
			case 'checkbox':
				return e107::getUserClass()->uc_checkboxes($name, $curval, $optlist, null,false);
			break;
			case 'dropdown':
			default:
				return e107::getUserClass()->uc_dropdown($name, $curval, $optlist, $opt);
			break;
		}
	}
	
	
	/**
	 * Renders a generic search box. If $filter has values, a filter box will be included with the options provided. 
	 * 
	 */
	public function search($name, $searchVal, $submitName, $filterName='', $filterArray=false, $filterVal=false)
	{
		$tp = e107::getParser();
		
		$text = '
    		'.$this->text($name, $searchVal,20,'class=search-query&placeholder='.LAN_SEARCH.'…').'
   			 '.$tp->toGlyph('fa-search').'  ';
		
		
		
		if(is_array($filterArray))
		{
			$text .= $this->select($filterName, $filterArray, $filterVal);
		}
		
	//	$text .= $this->admin_button($submitName,LAN_SEARCH,'search');
		
		return $text;
		
		/*
		$text .= 
		
						
							Display All 
							Clear Filter 
								
General 
Misc 
Test 3 
	 
						 Display All  
						
						
						
			Filter 
							 	
		
		*/
	}
	/**
	 * @param       $name
	 * @param null  $current_value
	 * @param null  $uc_options
	 * @param array $select_options multiple, default
	 * @param array $opt_options
	 * @return string
	 */
	public function uc_select($name, $current_value=null, $uc_options=null, $select_options = array(), $opt_options = array())
	{
/*	var_dump($name);
	var_dump($current_value);
var_dump($uc_options);
var_dump($select_options);*/
		if(!empty($select_options['multiple']) && substr($name,-1) !== ']')
		{
			$name .= '[]';
		}
		if(($current_value === null || $current_value === '') && !empty($uc_options)) // make the first in the opt list the default value.
		{
			$tmp = explode(',', $uc_options);
			$current_value =  e107::getUserClass()->getClassFromKey($tmp[0]);
			if(isset($select_options['default']))
			{
				$current_value = (int) $select_options['default'];
			}
		}
		if(!empty($current_value) && !is_numeric($current_value)) // convert name to id.
		{
			//$current_value = $this->_uc->getID($current_value);
			// issue #3249 Accept also comma separated values
			if (!is_array($current_value))
			{
				$current_value = explode(',', $current_value);
			}
			$tmp = array();
			foreach($current_value as $val)
			{
				if (!empty($val))
				{
					$tmp[] = !is_numeric($val) ? $this->_uc->getID(trim($val)) : (int) $val;
				}
			}
			$current_value = implode(',', $tmp);
			unset($tmp);
		}
		$text = $this->select_open($name, $select_options)."\n";
		$text .= $this->_uc->vetted_tree($name, array($this, '_uc_select_cb'), $current_value, $uc_options, $opt_options)."\n";
		$text .= $this->select_close();
		return $text;
	}
	// Callback for vetted_tree - Creates the option list for a selection box
	public function _uc_select_cb($treename, $classnum, $current_value, $nest_level)
	{
		$classIndex = abs($classnum);			// Handle negative class values
		$classSign = (strpos($classnum, '-') === 0) ? '-' : '';
		
		if($classnum == e_UC_BLANK)
		{
			return $this->option(' ', '');
		}
		$tmp = explode(',', $current_value);
		if($nest_level == 0)
		{
			$prefix = '';
			$style = 'font-weight:bold; font-style: italic;';
		}
		elseif($nest_level == 1)
		{
			$prefix = '  ';
			$style = 'font-weight:bold';
		}
		else
		{
			$prefix = '  '.str_repeat('--', $nest_level - 1).'>';
			$style = '';
		}
		return $this->option($prefix.$this->_uc->getName($classnum), $classSign.$classIndex, ($current_value !== '' && in_array($classnum, $tmp)), array('style' => ($style)))."\n";
	}
	public function optgroup_open($label, $disabled = false, $options = null)
	{
		return "\n";
	}
    /**
     *  tag generation. 
     * @param $option_title 
     * @param $value
     * @param $selected
     * @param $options (eg. disabled=1)
     */
	public function option($option_title, $value, $selected = false, $options = '')
	{
	    if(is_string($options))
	    {
		    parse_str($options, $options);
	    }
		if($value === false)
		{
			$value = '';
		}
		$options = $this->format_options('option', '', $options);
		$options['selected'] = $selected; //comes as separate argument just for convenience
		return " get_attributes($options). '>' .defset($option_title, $option_title). ' ';
	}
    /**
    * Use selectbox() instead.
    */
	public function option_multi($option_array, $selected = false, $options = array())
	{
		if(is_string($option_array))
		{
			parse_str($option_array, $option_array);
		}
		$text = '';
		if(empty($option_array))
		{
			return $this->option('','');
		}
		$opts = $options;
		foreach ((array) $option_array as $value => $label)
		{
			if(is_array($label))
			{
				$text .= $this->optgroup($value, $label, $selected, $options, 0);
			}
			else
			{
				$sel = is_array($selected) ? in_array($value, $selected) : ($value == $selected); // comparison as int/string currently required for admin-ui to function correctly.
				if(!empty($options['optDisabled']) && is_array($options['optDisabled']))
				{
					$opts['disabled'] = in_array($value, $options['optDisabled']);
				}
				if(!empty($options['title'][$value]))
				{
					$opts['data-title'] = $options['title'][$value];
				}
				elseif(isset($opts['data-title']))
				{
					unset($opts['data-title']);
				}
				$text .= $this->option($label, $value, $sel, $opts)."\n";
			}
		}
		return $text;
	}
	/**
	 * No compliant, but it works.
	 * @param $value
	 * @param $label
	 * @param $selected
	 * @param $options
	 * @param int $level
	 * @return string
	 */
	private function optgroup($value, $label, $selected, $options, $level=1)
	{
		$level++;
		$text = $this->optgroup_open($value, null, array('class'=>'level-'.$level));
		$opts = $options;
		foreach($label as $val => $lab)
		{
			if(is_array($lab))
			{
				$text .= $this->optgroup($val,$lab,$selected,$options,$level);
			}
			else
			{
				if(!empty($options['optDisabled']) && is_array($options['optDisabled']))
				{
					$opts['disabled'] = in_array($val, $options['optDisabled']);
				}
				$text .= $this->option($lab, $val, (is_array($selected) ? in_array($val, $selected) : $selected == $val), $opts)."\n";
			}
		}
		$text .= $this->optgroup_close();
		return $text;
	}
	public function optgroup_close()
	{
		return " \n";
	}
	public function select_close()
	{
		return ' ';
	}
	public function hidden($name, $value, $options = array())
	{
		$options = $this->format_options('hidden', $name, $options);
		return "get_attributes($options, $name, $value). '  >' .$icon. ' ';
	
	}
	/**
	 * Alias of admin_button, adds the etrigger_ prefix required for UI triggers
	 * @see e_form::admin_button()
	 */
	public function admin_trigger($name, $value, $action = 'submit', $label = '', $options = array())
	{
		return $this->admin_button('etrigger_'.$name, $value, $action, $label, $options);
	}
	/**
	 * Generic Button Element. 
	 * @param string $name
	 * @param string|array $value
	 * @param string $action [optional] default is submit - use 'dropdown' for a bootstrap dropdown button. 
	 * @param string $label [optional]
	 * @param string|array $options [optional]
	 * @return string
	 */
	public function button($name, $value, $action = 'submit', $label = '', $options = array())
	{
		if($action === 'dropdown' && deftrue('BOOTSTRAP') && is_array($value))
		{
		//	$options = $this->format_options('admin_button', $name, $options);
			$options['class'] = vartrue($options['class']);
			
			$align = vartrue($options['align'],'left');
					
			$text = '';
			
			return $text;	
		}			
				
		
		return $this->admin_button($name, $value, $action, $label, $options);
		
	}
	/**
	 * Render a Breadcrumb in Bootstrap format. 
	 * @param array $array
	 * @param $array[url]
	 * @param $array[text]
	 * @param bool $force - used internally to prevent duplicate {--BREADCUMB---} and template breadcrumbs from both displaying at once.
	 */
	public function breadcrumb($array, $force = false)
	{
		if($force === false && defset('THEME_VERSION') === 2.3) // ignore template breadcrumb.
		{
			return null;
		}
		if(deftrue('e_FRONTPAGE'))
		{
			return null;
		}
		if(!is_array($array)){ return; }
		
		$opt = array();
		
		$homeicon = ($this->_fontawesome) ? 'fa-home' : 'icon-home.glyph';
		$homeIcon = e107::getParser()->toGlyph($homeicon,false);
		
		
		$opt[] = "".$homeIcon. ' '; // Add Site-Pref to disable?
		
		$text = "\n\n";
		$text .= '';
		foreach($array as $val)
		{
			if($val['url'] === e_REQUEST_URI) // automatic link removal for current page.
			{
				$val['url']= null;
			}
			$ret = '';
			$ret .= !empty($val['url']) ? "" : '';
			$ret .= vartrue($val['text']);
			$ret .= !empty($val['url']) ? ' ' : '';
			
			if($ret != '')
			{
				$opt[] = $ret;
			}	
		}
	
		$sep = (deftrue('BOOTSTRAP')) ? '' : "/ ";
	
		$text .= implode($sep." \n",$opt);
	
		$text .= " \n ";
		
	//	return print_a($opt,true);
	
		return $text;	
	}
	/**
	 * Render a direct link admin-edit button on the frontend.
	 * @param $url
	 * @param string $perms
	 * @return string
	 */
	public function instantEditButton($url, $perms='0')
	{
		if(deftrue('BOOTSTRAP') && getperms($perms))
		{
			return "".e107::getParser()->toGlyph('fa-edit'). ' get_attributes($options, $name) . ">{$label}  ";
	}
	/**
	 * Helper function to check if a (CSS) class already contains a button class?
	 *
	 * @param string $class
	 *  The class we want to check.
	 *
	 * @return bool
	 *  True if $class already contains a button class. Otherwise false.
	 */
	private function defaultButtonClassExists($class = '')
	{
		// Bootstrap button classes.
		// @see http://getbootstrap.com/css/#buttons-options
		$btnClasses = array(
			'btn-default',
			'btn-primary',
			'btn-success',
			'btn-info',
			'btn-warning',
			'btn-danger',
			'btn-link',
		);
		foreach($btnClasses as $btnClass)
		{
			if(strpos($class, $btnClass) !== false)
			{
				return true;
			}
		}
		return false;
	}
	/**
	 * Helper function to get default button class by action.
	 *
	 * @param string $action
	 *  Action for a button. See button().
	 *
	 * @return string $class
	 *  Default button class.
	 */
	private function getDefaultButtonClassByAction($action)
	{
		switch($action)
		{
			case 'update':
			case 'create':
			case 'import':
			case 'submit':
			case 'execute':
			case 'success':
				$class = 'btn-success';
				break;
			case 'delete':
			case 'danger':
				$class = 'btn-danger';
				break;
			case 'other':
			case 'login':
			case 'batch e-hide-if-js':
			case 'filter e-hide-if-js':
			case 'batch':
			case 'filter':
			case 'primary':
				$class = 'btn-primary';
				break;
			case 'warning':
			case 'confirm':
				$class = 'btn-warning';
				break;
			case 'default':
			case 'checkall':
			case 'cancel':
			default:
				$class = 'btn-default';
				break;
		}
		return $class;
	}
	public function getNext()
	{
		if(!$this->_tabindex_enabled)
		{
			return 0;
		}
		++$this->_tabindex_counter;
		return $this->_tabindex_counter;
	}
	public function getCurrent()
	{
		if(!$this->_tabindex_enabled)
		{
			return 0;
		}
		return $this->_tabindex_counter;
	}
	public function resetTabindex($reset = 0)
	{
		$this->_tabindex_counter = $reset;
	}
	public function get_attributes($options, $name = '', $value = '')
	{
		$ret = '';
		//
		foreach ($options as $option => $optval)
		{
			$optval = trim($optval);
			switch ($option)
			{
				case 'id':
					$ret .= $this->_format_id($optval, $name, $value);
					break;
				case 'class':
					if(!empty($optval))
					{
						$ret .= " class='{$optval}'";
					}
					break;
				case 'size':
					if($optval)
					{
						$ret .= " size='{$optval}'";
					}
					break;
				case 'title':
					if($optval)
					{
						$ret .= " title='{$optval}'";
					}
					break;
				case 'label':
					if($optval)
					{
						$ret .= " label='{$optval}'";
					}
					break;
				case 'tabindex':
					if($optval)
					{
						$ret .= " tabindex='{$optval}'";
					}
					elseif($optval === false || !$this->_tabindex_enabled)
					{
						break;
					}
					else
					{
						++$this->_tabindex_counter;
						$ret .= " tabindex='".$this->_tabindex_counter."'";
					}
					break;
				case 'readonly':
					if($optval)
					{
						$ret .= " readonly='readonly'";
					}
					break;
				case 'multiple':
					if($optval)
					{
						$ret .= " multiple='multiple'";
					}
					break;
				case 'selected':
					if($optval)
					{
						$ret .= " selected='selected'";
					}
					break;
				case 'maxlength':
					if($optval)
					{
						$ret .= " maxlength='{$optval}'";
					}
					break;
				case 'checked':
					if($optval)
					{
						$ret .= " checked='checked'";
					}
					break;
				case 'disabled':
					if($optval)
					{
						$ret .= " disabled='disabled'";
					}
					break;
					
				case 'required':
					if($optval)
					{
						$ret .= " required='required'";
					}
					break;
					
				case 'autofocus':
					if($optval)
					{
						$ret .= " autofocus='autofocus'";
					}
					break;
					
				case 'placeholder':
					if($optval) {  
					  $optval = deftrue($optval, $optval);  
					  $ret .= " placeholder='{$optval}'";
					}
					break;
				case 'wrap':
					if($optval)
					{
						$ret .= " wrap='{$optval}'";
					}
					break;
					
				case 'autocomplete':
					if($optval)
					{
						$ret .= " autocomplete='{$optval}'";
					}
					break;
					
				case 'pattern':
					if($optval)
					{
						$ret .= " pattern='{$optval}'";
					}
					break;
				case 'other':
					if($optval)
					{
						$ret .= " $optval";
					}
					break;
				default:
					if(strpos($option,'data-') === 0)
					{
						$ret .= ' ' .$option."='{$optval}'";
					}
				break;
			}
				
		}
		return $ret;
	}
	/**
	 * Auto-build field attribute id
	 *
	 * @param string $id_value value for attribute id passed with the option array
	 * @param string $name the name attribute passed to that field
	 * @param mixed $value the value attribute passed to that field
	 * @return string formatted id attribute
	 */
	protected function _format_id($id_value, $name, $value = null, $return_attribute = 'id')
	{
		if($id_value === false)
		{
			 return '';
		}
		if(is_array($value))
		{
			$value = null;
		}
		//format data first
		$name = trim($this->name2id($name), '-');
		$value = trim(preg_replace('#[^a-zA-Z0-9\-]#', '-', $value), '-');
		//$value = trim(preg_replace('#[^a-z0-9\-]#/i','-', $value), '-');		// This should work - but didn't for me!
		$value = trim(str_replace('/', '-', $value), '-');                    // Why?
		if (!$id_value && is_numeric($value))
		{
			$id_value = $value;
		}
		// clean - do it better, this could lead to dups
		$id_value = trim($id_value, '-');
		if($return_attribute === null) // return only the value.
		{
			if (empty($id_value))
			{
				$ret = ($name) . ($value ? "-{$value}" : '');
			}
			elseif (is_numeric($id_value) && $name) // also useful when name is e.g. name='my_name[some_id]'
			{
				$ret = "{$name}-{$id_value}";
			}
			else // also useful when name is e.g. name='my_name[]'
			{
				$ret = (string) ($id_value);
			}
			return $ret;
		}
		if (empty($id_value))
		{
			$ret = " {$return_attribute}='{$name}" . ($value ? "-{$value}" : '') . "'";
		}
		elseif (is_numeric($id_value) && $name) // also useful when name is e.g. name='my_name[some_id]'
		{
			$ret = " {$return_attribute}='{$name}-{$id_value}'";
		}
		else // also useful when name is e.g. name='my_name[]'
		{
			$ret = " {$return_attribute}='{$id_value}'";
		}
		return $ret;
	}
	public function name2id($name)
	{
		$name = strtolower($name);
		return rtrim(str_replace(array('[]', '[', ']', '_', '/', ' ','.', '(', ')', '::', ':', '?','='), array('-', '-', '', '-', '-', '-', '-','','','-','','-','-'), $name), '-');
	}
	/**
	 * Format options based on the field type,
	 * merge with default
	 *
	 * @param string $type
	 * @param string $name form name attribute value
	 * @param array|string $user_options
	 * @return array merged options
	 */
	public function format_options($type, $name, $user_options=null)
	{
		if(is_string($user_options))
		{
			parse_str($user_options, $user_options); 
		}
		$def_options = $this->_default_options($type);
	
		if(is_array($user_options))
		{
			$user_options['name'] = $name; //required for some of the automated tasks
			
			foreach (array_keys($user_options) as $key)
			{
				if(!isset($def_options[$key]) && strpos($key,'data-') !== 0)
				{
					unset($user_options[$key]); // data-xxxx exempt //remove it?
				}
			}	
		}
		else 
		{
			$user_options = array('name' => $name); //required for some of the automated tasks	
		}
		
		return array_merge($def_options, $user_options);
	}
	/**
	 * Get default options array based on the field type
	 *
	 * @param string $type
	 * @return array default options
	 */
	protected function _default_options($type)
	{
		if(isset($this->_cached_attributes[$type]))
		{
			return $this->_cached_attributes[$type];
		}
		$def_options = array(
			'id' 			=> '',
			'class' 		=> '',
			'title' 		=> '',
			'size' 			=> '',
			'readonly' 		=> false,
			'selected' 		=> false,
			'checked' 		=> false,
			'disabled' 		=> false,
			'required' 		=> false,	
			'autofocus'		=> false,	
			'tabindex' 		=> 0,
			'label' 		=> '',
			'placeholder' 	=> '',
			'pattern'		=> '',
			'other' 		=> '',
			'autocomplete' 	=> '',
			'maxlength'		=> '',
			'wrap'          => '',
			'multiple'      => '',
			//	'multiple' => false, - see case 'select'
		);
		$form_control = (THEME_LEGACY !== true) ? ' form-control' : '';
		switch ($type) {
			case 'hidden':
				$def_options = array('id' => false, 'disabled' => false, 'other' => '');
				break;
			case 'text':
				$def_options['class'] = 'tbox input-text'.$form_control;
				unset($def_options['selected'], $def_options['checked']);
				break;
			case 'number':
				$def_options['class'] = 'tbox '.$form_control;
				break;
			case 'file':
				$def_options['class'] = 'tbox file';
				unset($def_options['selected'], $def_options['checked']);
				break;
			case 'textarea':
				$def_options['class'] = 'tbox textarea'.$form_control;
				unset($def_options['selected'], $def_options['checked'], $def_options['size']);
				break;
			case 'select':
				$def_options['class'] = 'tbox select'.$form_control;
				$def_options['multiple'] = false;
				unset($def_options['checked']);
				break;
			case 'option':
				$def_options = array('class' => '', 'selected' => false, 'other' => '', 'disabled' => false, 'label' => '');
				break;
			case 'radio':
				//$def_options['class'] = ' ';
				$def_options = array('class' => '', 'id'=>'');
				unset($def_options['size'], $def_options['selected']);
				break;
			case 'checkbox':
				unset($def_options['size'],  $def_options['selected']);
				break;
			case 'submit':
				$def_options['class'] = 'button btn btn-primary';
				unset($def_options['checked'], $def_options['selected'], $def_options['readonly']);
				break;
			case 'submit_image':
				$def_options['class'] = 'action';
				unset($def_options['checked'], $def_options['selected'], $def_options['readonly']);
				break;
			case 'admin_button':
				unset($def_options['checked'],  $def_options['selected'], $def_options['readonly']);
				break;
		}
		$this->_cached_attributes[$type] = $def_options;
		return $def_options;
	}
	public function columnSelector($columnsArray, $columnsDefault = array(), $id = 'column_options')
	{
		$columnsArray = array_filter($columnsArray);
		
	// navbar-header nav-header
	// navbar-header nav-header
		$text = '';
			
	//	$text .= "  ";
		$text .= '';
	
	
	/*
	$text = '';
	*/	
		return $text;
	}
	
	
	
	
	
	
	
	
	
	
	public function colGroup($fieldarray, $columnPref = '')
	{
        $text = '';
		foreach($fieldarray as $key=>$val)
		{
			if (($key === 'options' || in_array($key, $columnPref) ||  !empty($val['forced'])) && empty($val['nolist']))
			{
				$class = vartrue($val['class']) ? 'class="'.$val['class'].'"' : '';
				$width = vartrue($val['width']) ? ' style="width:'.$val['width'].'"' : '';
				$text .= '
				'.$text.'
			 
		';
	}
	public function thead($fieldarray, $columnPref = array(), $querypattern = '', $requeststr = '')
	{
        $text = '';
        $querypattern = filter_var($querypattern, FILTER_SANITIZE_STRING);
        if(!$requeststr)
        {
	        $requeststr = rawurldecode(e_QUERY);
        }
        $requeststr = filter_var($requeststr, FILTER_SANITIZE_STRING);
		// Recommended pattern: mode=list&field=[FIELD]&asc=[ASC]&from=[FROM]
		if(strpos($querypattern,'&')!==FALSE)
		{
			// we can assume it's always $_GET since that's what it will generate
			// more flexible (e.g. pass default values for order/field when they can't be found in e_QUERY) & secure
			$tmp = str_replace('&', '&', $requeststr ?: e_QUERY);
			parse_str($tmp, $tmp);
			$etmp = array();
			parse_str(str_replace('&', '&', $querypattern), $etmp);
		}
		else // Legacy Queries. eg. main.[FIELD].[ASC].[FROM]
		{
			$tmp = explode('.', ($requeststr ?: e_QUERY));
			$etmp = explode('.', $querypattern);
		}
		foreach($etmp as $key => $val)    // I'm sure there's a more efficient way to do this, but too tired to see it right now!.
		{
        	if($val === '[FIELD]')
			{
            	$field = varset($tmp[$key]);
			}
			if($val === '[ASC]')
			{
            	$ascdesc = varset($tmp[$key]);
			}
			if($val === '[FROM]')
			{
            	$fromval = varset($tmp[$key]);
			}
		}
		if(!varset($fromval)){ $fromval = 0; }
		$sorted = varset($ascdesc);
        $ascdesc = ($sorted === 'desc') ? 'asc' : 'desc';
		foreach($fieldarray as $key=>$val)
		{
     		if ((in_array($key, $columnPref) || ($key === 'options' && isset($val['title'])) || (vartrue($val['forced']))) && !vartrue($val['nolist']))
			{
				$cl = (vartrue($val['thclass'])) ? " class='".$val['thclass']."'" : '';
				$aClass = ($key === $field) ? "class='sorted-".$sorted."'" : '';
				$text .= "
					
				";
                if($querypattern!= '' && $key !== 'options' && $key !== 'checkboxes' && !vartrue($val['nosort']))
				{
					$from = ($key == $field) ? $fromval : 0;
					$srch = array('[FIELD]', '[ASC]', '[FROM]');
					$repl = array($key,$ascdesc,$from);
                	$val['url'] = e_SELF. '?' .str_replace($srch,$repl,$querypattern);
				}
				$text .= (vartrue($val['url'])) ? '" : '';  // Really this column-sorting link should be auto-generated, or be autocreated via unobtrusive js.
	            $text .= !empty($val['title']) ? defset($val['title'], $val['title']) : '';
				$text .= ($val['url']) ? ' ' : '';
	            $text .= ($key === 'options' && !vartrue($val['noselector'])) ? $this->columnSelector($fieldarray, $columnPref) : '';
				$text .= ($key === 'checkboxes') ? $this->checkbox_toggle('e-column-toggle', vartrue($val['toggle'], 'multiselect')) : '';
	
	 			$text .= '
					 
				';
			}
		}
		return '
		
	  		' .$text. ' 
		 
		';
	}
	/**
	 * Render Table cells from hooks.
	 * @param array $data 
	 * @return string
	 */
	public function renderHooks($data)
	{
		$hooks = e107::getEvent()->triggerHook($data);
				
		$text = '';
		
		if(!empty($hooks))
		{
			foreach($hooks as $plugin => $hk)
			{
				$text .= "\n\n\n";
				
				if(!empty($hk))
				{
					foreach($hk as $hook)
					{
						$text .= "\t\t\t\n";
						$text .= "\t\t\t".$hook['caption']." \n";
						$text .= "\t\t\t".$hook['html']. '';
						$text .= (varset($hook['help'])) ? "\n".$hook['help']. ' ' : '';
						$text .= " \n\t\t\t \n";
					}
					
				}
			}
		}
		
		return $text;			
	}
			
	/**
	 * Render Related Items for the current page/news-item etc. 
	 * @param string $type : comma separated list. ie. plugin folder names. 
	 * @param string $tags : comma separated list of keywords to return related items of.
	 * @param array $curVal. eg. array('page'=> current-page-id-value); 
	 */
	public function renderRelated($parm, $tags, $curVal, $template=null) //XXX TODO Cache!
	{
		
		if(empty($tags))
		{
			return;	
		}
		$tags = str_replace(', ',',', $tags); //BC Fix, all tags should be comma separated without spaces ie. one,two NOT one, two
		e107::setRegistry('core/form/related',$tags); // TODO Move to elsewhere so it works without rendering? e107::related() set and get by plugins?
		if(!varset($parm['limit']))
		{
			$parm = array('limit' => 5);
		}
		
		if(!varset($parm['types']))
		{
			$parm['types'] = 'news';	
		}
		if(empty($template))
		{
			$TEMPLATE['start'] = '' .defset('LAN_RELATED', 'Related')." ';
		}
		else
		{
			$TEMPLATE = $template;
		}
		
			
		$tp = e107::getParser();
			
		$types = explode(',',$parm['types']);
		$list = array();
		
		$head = $tp->parseTemplate($TEMPLATE['start']);
		foreach($types as $plug)
		{
		
			if(!$obj = e107::getAddon($plug,'e_related'))
			{
				continue;
			}
			
			$parm['current'] = (int) varset($curVal[$plug]);
		
			$tmp = $obj->compile($tags,$parm);	
		
			if(is_array($tmp) && count($tmp))
			{
				foreach($tmp as $val)
				{
					$row = array(
						'RELATED_URL'       => $tp->replaceConstants($val['url'],'full'),
						'RELATED_TITLE'     => $val['title'],
						'RELATED_IMAGE'     => $tp->toImage($val['image']),
						'RELATED_SUMMARY'   => $tp->toHTML($val['summary'],true,'BODY'),
						'RELATED_DATE'		=> $val['date'],	
					);
					$list[] = $tp->simpleParse($TEMPLATE['item'], $row);
				}
			}		
		}
		
		if(count($list))
		{
			$text = "".$head.implode("\n",$list).$tp->parseTemplate($TEMPLATE['end']). '
';
			$caption = $tp->parseTemplate(varset($TEMPLATE['caption']));
			return e107::getRender()->tablerender($caption, $text, 'related', true);
		}
		
	}		
	/**
	 * Render Table cells from field listing.
	 * @param array $fieldarray - eg. $this->fields
	 * @param array $currentlist - eg $this->fieldpref
	 * @param array $fieldvalues - eg. $row
	 * @param string $pid - eg. table_id
	 * @return string
	 */
	public function renderTableCells($fieldarray, $currentlist, $fieldvalues, $pid)
	{
		$cnt = 0;
		$text = '';
		foreach ($fieldarray as $field => $data)
		{
			if(!isset($data['readParms']) || $data['readParms'] === '' )
			{
				$data['readParms'] = array();
			}
			elseif(is_string($data['readParms'])) // fix for readParms = '';
			{
				parse_str($data['readParms'],$data['readParms']);
			}
			// shouldn't happen... test with Admin->Users with user_xup visible and NULL values in user_xup table column before re-enabling this code.
			/*
			if(!isset($fieldvalues[$field]) && vartrue($data['alias']))
			{
				$fieldvalues[$data['alias']] = $fieldvalues[$data['field']];
				$field = $data['alias'];
			}
			*/
			//Not found
			if(!empty($data['nolist']) || (empty($data['forced']) && !in_array($field, $currentlist)))
			{
				continue;
			}
			if(vartrue($data['type']) !== 'method' && empty($data['forced']) && !isset($fieldvalues[$field]) && $fieldvalues[$field] !== null)
			{
				$text .= "
					
						Not Found! ($field)
					 
				";
				continue;
			}
			$tdclass = vartrue($data['class']);
            if($field === 'checkboxes')
            {
	            $tdclass = $tdclass ? $tdclass . ' autocheck e-pointer' : 'autocheck e-pointer';
            }
			if($field === 'options')
			{
				$tdclass = $tdclass ? $tdclass . ' options' : 'options';
			}
			// there is no other way for now - prepare user data
			if(vartrue($data['type']) === 'user' /* && isset($data['readParms']['idField'])*/)
			{
				if(varset($data['readParms']) && is_string($data['readParms']))
				{
					parse_str($data['readParms'], $data['readParms']);
				}
				if(isset($data['readParms']['idField']))
				{
					$data['readParms']['__idval'] = $fieldvalues[$data['readParms']['idField']];
				}
				elseif(isset($fieldvalues['user_id'])) // Default
				{
					$data['readParms']['__idval'] = $fieldvalues['user_id'];
				}
				if(isset($data['readParms']['nameField']))
				{
					$data['readParms']['__nameval'] = $fieldvalues[$data['readParms']['nameField']];
				}
				elseif(isset($fieldvalues['user_name'])) // Default
				{
					$data['readParms']['__nameval'] = $fieldvalues['user_name'];
				}
			}
			$value = $this->renderValue($field, varset($fieldvalues[$field]), $data, varset($fieldvalues[$pid]));
			if($tdclass)
			{
				$tdclass = ' class="'.$tdclass.'"';
			}
			$text .= '
				
					'.$value.'
				 
			';
			$cnt++;
		}
		if($cnt)
		{
			return $text;
		}
		return null;
	}
	/**
	 * Render Table row and cells from field listing.
	 *
	 * @param array  $fieldArray  - eg. $this->fields
	 * @param array  $fieldPref   - eg $this->fieldpref
	 * @param array  $fieldValues - eg. $row
	 * @param string $pid         - eg. table_id
	 * @return string
	 */
	public function renderTableRow($fieldArray, $fieldPref, $fieldValues, $pid)
	{
		if(!$ret = $this->renderTableCells($fieldArray, $fieldPref, $fieldValues, $pid))
		{
			return '';
		}
		$trclass = '';
		//	$trclass = vartrue($fieldvalues['__trclass']) ?  ' class="'.$trclass.'"' : '';
		unset($fieldValues['__trclass']);
		return '
				
					'.$ret.'
				 
			';
	}
	/**
	 * Inline Token
	 * @return string
	 */
	private function inlineToken()
	{
		$this->_inline_token = $this->_inline_token ?:
			password_hash(session_id(), PASSWORD_DEFAULT, ['cost' => 04]);
		return $this->_inline_token;
	}
	/**
	 * Create an Inline Edit link. 
	 * @param string $dbField : field being edited
	 * @param int $pid : primary ID of the row being edited.
	 * @param string $fieldName - Description of the field name (caption)
	 * @param mixed $curVal : existing value of in the field
	 * @param mixed $linkText : existing value displayed
	 * @param string $type text|textarea|select|date|checklist
	 * @param array $array : array data used in dropdowns etc.
	 */
	public function renderInline($dbField, $pid, $fieldName, $curVal, $linkText, $type='text', $array=null, $options=array())
	{
		$jsonArray = array();
				
		if(!empty($array))
		{
			foreach($array as $k=>$v)
			{
				$jsonArray[$k] = str_replace("'", '`', $v);
			}
		}
		$source = e107::getParser()->toJSON($jsonArray, true);
		
		$mode = preg_replace('/[\W]/', '', vartrue($_GET['mode']));
		if(!isset($options['url']))
		{
			$options['url'] = e_SELF."?mode={$mode}&action=inline&id={$pid}&ajax_used=1";
		}
		if(!empty($pid))
		{
			$options['pk'] = $pid;
		}
		$title = varset($options['title'] , (LAN_EDIT. ' ' .$fieldName));
		$class = varset($options['class']);
		unset( $options['title']);
		$text = "inlineToken();
		if(!empty($options))
		{
			foreach($options as $k=>$opt)
			{
				if(!empty($opt))
				{
					$text .= ' data-' .$k."='".$opt."'";
				}
			}
		}
		$text .= '>' .$linkText. ' ';
		
		return $text;	
	}
	/**
	 * Check if a value should be linked and wrap in  tag if required.
	 * @todo global pref for the target option?
	 * @param mixed $value
	 * @param array $parms
	 * @param $id
	 * @example $frm->renderLink('label', array('link'=>'{e_PLUGIN}myplugin/myurl.php','target'=>'blank')
	 * @example $frm->renderLink('label', array('link'=>'{e_PLUGIN}myplugin/myurl.php?id=[id]','target'=>'blank')
	 * @example $frm->renderLink('label', array('link'=>'{e_PLUGIN}myplugin/myurl.php?id=[field-name]','target'=>'blank')
	 * @example $frm->renderLink('label', array('link'=>'db-field-name','target'=>'blank')
	 * @example $frm->renderLink('label', array('url'=>'e_url.php key','title'=>'click here');
	 * @return string
	 */
	public function renderLink($value, $parms, $id=null)
	{
		if(empty($parms['link']) && empty($parms['url']))
		{
			return $value;
		}
		/** @var e_admin_model $model */
		if(!$model = e107::getRegistry('core/adminUI/currentListModel')) // Try list model
		{
			$model = e107::getRegistry('core/adminUI/currentModel'); // try create/edit model.
		}
		$dialog     = vartrue($parms['target']) === 'dialog' ? ' e-modal' : ''; // iframe
		$ext        = vartrue($parms['target']) === 'blank' ? " rel='external' " : ''; // new window
		$modal      = vartrue($parms['target']) === 'modal' ? " data-toggle='modal' data-bs-toggle='modal' data-cache='false' data-target='#uiModal' " : '';
		$link = null;
		if(!empty($parms['url']) && !empty($model)) // ie. use e_url.php
		{
			//$plugin = $this->getController()->getPluginName();
			if($plugin = e107::getRegistry('core/adminUI/currentPlugin'))
			{
				$data = $model->getData();
				$link = e107::url($plugin,$parms['url'],$data);
			}
		}
		elseif(!empty($model)) // old way.
		{
			$tp = e107::getParser();
			$data = $model->getData();
			$link       = str_replace('[id]',$id,$parms['link']);
			$link       = $tp->replaceConstants($link); // SEF URL is not important since we're in admin.
			if($parms['link'] === 'sef' )
			{
				if(!$model->getUrl())
				{
					/** @var e_admin_controller_ui $controller */
					$controller = $this->getController();
					$model->setUrl($controller->getUrl());
				}
				// assemble the url
				$link = $model->url(null);
			}
			elseif(!empty($data[$parms['link']])) // support for a field-name as the link. eg. link_url.
			{
				$link = $tp->replaceConstants(vartrue($data[$parms['link']]));
			}
			elseif(strpos($link,'[')!==false && preg_match('/\[(\w+)\]/',$link, $match)) // field-name within [ ] brackets.
			{
				$field = $match[1];
				$link = str_replace($match[0], $data[$field],$link);
			}
		}
					// in case something goes wrong...
		if($link)
		{
			return " ".$value. ' ';
		}
		return $value;
	}
	private function renderOptions($parms, $id, $attributes)
	{
		$tp = e107::getParser();
		$cls = false;
		$editIconDefault = deftrue('ADMIN_EDIT_ICON', $tp->toGlyph('fa-edit'));
		$deleteIconDefault = deftrue('ADMIN_DELETE_ICON', $tp->toGlyph('fa-trash'));
/*
		if($attributes['grid'])
		{
			$editIconDefault = $tp->toGlyph('fa-edit');
			$deleteIconDefault = $tp->toGlyph('fa-trash');
		}
*/
		/** @var e_admin_controller_ui $controller */
		$controller = $this->getController();
		$sf = $controller->getSortField();
		if(!isset($parms['sort']) && !empty($sf))
		{
			$parms['sort'] = true;
		}
		$text = "";
		if(!empty($parms['sort']) && empty($attributes['grid']))
		{
			$mode = preg_replace('/[\W]/', '', vartrue($_GET['mode']));
			$from = (int) vartrue($_GET['from'], 0);
			$text .= "
".ADMIN_SORT_ICON. '  ';
		}
		if(varset($parms['editClass']))
		{
			$cls = (deftrue($parms['editClass'])) ? constant($parms['editClass']) : $parms['editClass'];
		}
		if(($cls === false || check_class($cls)) && varset($parms['edit'],1) == 1)
		{
			parse_str(str_replace('&', '&', e_QUERY), $query); //FIXME - FIX THIS
				// keep other vars in tact
			$query['action'] = 'edit';
			$query['id'] = $id;
			if(!empty($parms['target']) && $parms['target'] === 'modal')
			{
				$eModal = ' e-modal ';
				$eModalCap = !empty($parms['modalCaption']) ? "data-modal-caption='".$parms['modalCaption']."'" : "data-modal-caption='#".$id."'";
				$query['iframe'] = 1;
			}
			else
			{
				$eModal = '';
				$eModalCap = '';
			}
			if(!empty($parms['modalSubmit']))
			{
				$eModalCap .= " data-modal-submit='true'";
			}
			$query = http_build_query($query, null, '&');
			$text .= "
				".$editIconDefault. ' ';
		}
		$delcls = !empty($attributes['noConfirm']) ? ' no-confirm' : '';
		if(varset($parms['deleteClass']) && varset($parms['delete'],1) == 1)
		{
			$cls = (deftrue($parms['deleteClass'])) ? constant($parms['deleteClass']) : $parms['deleteClass'];
			if(check_class($cls))
			{
				$parms['class'] =  'action delete btn btn-default'.$delcls;
				unset($parms['deleteClass']);
				$parms['icon'] = $deleteIconDefault;
				$text .= $this->submit_image('etrigger_delete['.$id.']', $id, 'delete', LAN_DELETE.' [ ID: '.$id.' ]', $parms);
			}
		}
		else
		{
			$parms['class'] =  'action delete btn btn-default'.$delcls;
			$parms['icon'] = $deleteIconDefault;
			$text .= $this->submit_image('etrigger_delete['.$id.']', $id, 'delete', LAN_DELETE.' [ ID: '.$id.' ]', $parms);
		}
				//$attributes['type'] = 'text';
		$text .= '
".e107::getIPHandler()->ipDecode($value).' ';
				// else same
			break;
			case 'templates':
			case 'layouts':
				if(!empty($attributes['writeParms']) && is_string($attributes['writeParms']))
				{
					parse_str($attributes['writeParms'], $attributes['writeParms']);
				}
				if(empty($attributes['noedit']) && !empty($parms['editable']) && empty($parms['link'])) // avoid bad markup, better solution coming up
				{
					$wparms     = $attributes['writeParms'];
					$location   = vartrue($wparms['plugin']); // empty - core
					$ilocation  = vartrue($wparms['id'], $location); // omit if same as plugin name
					$where      = vartrue($wparms['area'], 'front'); //default is 'front'
					$filter     = varset($wparms['filter']);
					$merge      = isset($wparms['merge']) ? (bool) $wparms['merge'] : true;
					$layouts    = e107::getLayouts($location, $ilocation, $where, $filter, $merge, false);
					$label   = varset($layouts[$value], $value);
					$value = $this->renderInline($field, $id, $attributes['title'], $value, $label, 'select', $layouts);
				}
				$value = vartrue($parms['pre']) . $value . vartrue($parms['post']);
			break;
			case 'checkboxes':
			case 'comma':
			case 'dropdown':
				// XXX - should we use readParams at all here? see writeParms check below
			//	if($parms && is_array($parms)) // FIXME - add support for multi-level arrays (option groups)
			//	{
					//FIXME return no value at all when 'editable=1' is a readParm. See FAQs templates. 
				//	$value = vartrue($parms['pre']).vartrue($parms[$value]).vartrue($parms['post']);
				//	break; 
			//	}
				
				// NEW - multiple (array values) support
				// FIXME - add support for multi-level arrays (option groups)
				if(!is_array($attributes['writeParms']))
				{
					parse_str($attributes['writeParms'], $attributes['writeParms']);
				}
				$wparms = $attributes['writeParms'];
				
				if(!is_array(varset($wparms['__options'])))
				{
					parse_str($wparms['__options'], $wparms['__options']);
				}
				if(!empty($wparms['optArray']))
				{
					$fopts = $wparms;
					$wparms = $fopts['optArray'];
					unset($fopts['optArray']);
					$wparms['__options'] = $fopts;
				}
				$opts = $wparms['__options'];
				unset($wparms['__options']);
				$_value = $value;
				
				if($attributes['type'] === 'checkboxes' || $attributes['type'] === 'comma')
				{
					$opts['multiple'] = true;	
				}
			
				if(!empty($opts['multiple']))
				{
					$ret = array();
					$value = is_array($value) ? $value : explode(',', $value);
					foreach ($value as $v)
					{
						if(isset($wparms[$v]))
						{
							$ret[] = $wparms[$v];
						}
					}
					$value = implode(', ', $ret);
				}
				else
				{
					$ret = '';
					if(isset($wparms[$value]))
					{
						$ret = $wparms[$value];
					}
					$value = $ret;
				}
			
				$value = ($value ? vartrue($parms['pre']).defset($value, $value).vartrue($parms['post']) : '');
				
				if(empty($attributes['noedit']) && !empty($parms['editable']) && empty($parms['link'])) // avoid bad markup, better solution coming up
				{				
					$xtype = ($attributes['type'] === 'dropdown') ? 'select' : 'checklist';
					$value = $this->renderInline($field, $id, $attributes['title'], $_value, $value, $xtype, $wparms);
				}
								
				// return ;
			break;
			case 'radio':
				if($parms && isset($parms[$value])) // FIXME - add support for multi-level arrays (option groups)
				{
					$value = vartrue($parms['pre']).vartrue($parms[$value]).vartrue($parms['post']);
					break;
				}
				if(!is_array($attributes['writeParms']))
				{
					parse_str($attributes['writeParms'], $attributes['writeParms']);
				}
				if(!empty($attributes['writeParms']['optArray']))
				{
					$radioValue = $attributes['writeParms']['optArray'][$value];
					if(empty($attributes['noedit']) && !empty($parms['editable']) && empty($parms['link'])) // avoid bad markup, better solution coming up
					{
						$radioValue = $this->renderInline($field, $id, $attributes['title'], $value, $radioValue, 'select', $attributes['writeParms']['optArray']);
					}
				}
				else
				{
					$radioValue = vartrue($attributes['writeParms'][$value]);
				}
				$value = vartrue($attributes['writeParms']['__options']['pre']).$radioValue.vartrue($attributes['writeParms']['__options']['post']);
			break;
			case 'tags':
				if(!empty($parms['constant']))
				{
					$value = defset($value, $value);
				}
				if(!empty($parms['truncate']))
				{
					$value = $tp->text_truncate($value, $parms['truncate'], '...');
				}
				elseif(!empty($parms['htmltruncate']))
				{
					$value = $tp->html_truncate($value, $parms['htmltruncate']);
				}
				if(!empty($parms['wrap']))
				{
					$value = $tp->htmlwrap($value, (int) $parms['wrap'], varset($parms['wrapChar'], ' '));
				}
				$value = $this->renderLink($value,$parms,$id);
				if(empty($value))
				{
					$value = '-';
					$setValue = "data-value=''";
				}
				else
				{
					$setValue = '';
					if($attributes['type'] === 'tags' && !empty($value))
					{
						$setValue = "data-value='" . $value . "'";
						$value = str_replace(',', ', ', $value); // add spaces so it wraps, but don't change the actual values.
					}
				}
				if(!vartrue($attributes['noedit']) && vartrue($parms['editable']) && !vartrue($parms['link'])) // avoid bad markup, better solution coming up
				{
					$options['selectize'] = array(
						'create'     => true,
						'maxItems'   => vartrue($parms['maxItems'], 7),
						'mode'       => 'multi',
						'e_editable' => $field . '_' . $id,
					);
					$tpl = $this->text($field, $value, 80, $options);
					$mode = preg_replace('/[\W]/', '', vartrue($_GET['mode']));
					$value = "" . $value . ' ';
				}
				$value = vartrue($parms['pre']) . $value . vartrue($parms['post']);
				break;
			case 'text':
				if(!empty($parms['constant']))
				{
					$value = defset($value,$value);
				}
				if(is_array($value) && ($attributes['data'] === 'json'))
				{
					$value = e107::serialize($value, 'json');
				}
				if(!empty($parms['truncate']))
				{
					$value = $tp->text_truncate($value, $parms['truncate'], '...');
				}
				elseif(!empty($parms['htmltruncate']))
				{
					$value = $tp->html_truncate($value, $parms['htmltruncate']);
				}
				if(!empty($parms['wrap']))
				{
					$value = $tp->htmlwrap($value, (int)$parms['wrap'], varset($parms['wrapChar'], ' '));
				}
				$value = $this->renderLink($value,$parms,$id);
				if(empty($value))
				{
					$value = '-';
					$setValue = "data-value=''";
				}
				else
				{
					$setValue = '';
					if($attributes['type'] === 'tags' && !empty($value))
					{
						$setValue = "data-value='".$value."'";
						$value = str_replace(',', ', ', $value); // add spaces so it wraps, but don't change the actual values.
					}
				}
					
				if(empty($attributes['noedit']) && !empty($parms['editable']) && empty($parms['link'])) // avoid bad markup, better solution coming up
				{
					$value = $this->renderInline($field,$id,$attributes['title'],$value, $value);
				}
				$value = vartrue($parms['pre']).$value.vartrue($parms['post']);
			break;
            
            
			case 'bbarea':
			case 'textarea':
				
				
				if($attributes['type'] === 'textarea' && !vartrue($attributes['noedit']) && vartrue($parms['editable']) && !vartrue($parms['link'])) // avoid bad markup, better solution coming up
				{
					return $this->renderInline($field,$id,$attributes['title'],$value,substr($value,0,50). '...','textarea'); //FIXME.
				}
				$expand = '... ';
				$toexpand = false;
				if($attributes['type'] === 'bbarea' && !isset($parms['bb']))
				{
					$parms['bb'] = true;
				} //force bb parsing for bbareas
				$elid = trim(str_replace('_', '-', $field)).'-'.$id;
				if(!vartrue($parms['noparse']))
				{
					$value = $tp->toHTML($value, (vartrue($parms['bb']) ? true : false), vartrue($parms['parse']));
				}
				if(!empty($parms['expand']) || !empty($parms['truncate']) || !empty($parms['htmltruncate']))
				{
					$ttl = vartrue($parms['expand']);
					if($ttl == 1)
					{
						$dataAttr = "data-text-more='" . LAN_MORE . "' data-text-less='" . LAN_LESS . "'";
						$ttl = $expand."" . LAN_MORE . ' ';
					}
					
					$expands = ''.defset($ttl, $ttl). ' ';
				}
				$oldval = $value;
				if(!empty($parms['truncate']))
				{
					$oldval = strip_tags($value);
					$value = $oldval;
					$value = $tp->text_truncate($value, $parms['truncate'], '');
					$toexpand = $value != $oldval;
				}
				elseif(!empty($parms['htmltruncate']))
				{
					$value = $tp->html_truncate($value, $parms['htmltruncate'], '');
					$toexpand = $value != $oldval;
				}
				if($toexpand)
				{
					// force hide! TODO - core style .expand-c (expand container)
					$value .= ''.str_replace($value,'',$oldval).' ".$tp->toIcon($value,$parms). ' ';
				
			break;
			
			case 'file':
				if(!empty($parms['base']))
				{
					$url = $parms['base'].$value;
				}
				else
				{
					$url = e107::getParser()->replaceConstants($value, 'full');
				}
				$name = basename($value);
				$value = ''.$name.' ';
			break;
			case 'image': //js tooltip...
				$thparms = array();
				$createLink = true;
						// Support readParms example: thumb=1&w=200&h=300
						// Support readParms example: thumb=1&aw=80&ah=30
				if(isset($parms['h']))		{ 	$thparms['h'] 	= (int) $parms['h']; 		}
				if(isset($parms['ah']))		{ 	$thparms['ah'] 	= (int) $parms['ah']; 	}
				if(isset($parms['w']))		{ 	$thparms['w'] 	= (int) $parms['w']; 		}
				if(isset($parms['aw']))		{ 	$thparms['aw'] 	= (int) $parms['aw']; 	}
				if(isset($parms['crop']))	{ 	$thparms['crop'] = $parms['crop']; 	}
				if($value)
				{
					
					if(strpos($value, ',')!==false)
					{
						$tmp = explode(',',$value);
						$value = $tmp[0];
						unset($tmp);	
					}		
					if(empty($parms['thumb_aw']) && !empty($parms['thumb']) && strpos($parms['thumb'],'x')!==false)
					{
						list($parms['thumb_aw'],$parms['thumb_ah']) = explode('x',$parms['thumb']);
					}
					$vparm = array('thumb'=>'tag','w'=> vartrue($parms['thumb_aw'],'80'));
					
					if($video = e107::getParser()->toVideo($value,$vparm))
					{
						return $video;
					}
					$fileOnly = basename($value);
					// Not an image but a file.  (media manager)  
					if(!preg_match("/\.(png|jpg|jpeg|gif|webp|PNG|JPG|JPEG|GIF|WEBP)/", $fileOnly) && strpos($fileOnly,'.') !== false)
					{
						$icon = '{e_IMAGE}filemanager/zip_32.png';
						$src = $tp->replaceConstants(vartrue($parms['pre']).$icon, 'abs');
					//	return $value;
						return e107::getParser()->toGlyph('fa-file','size=2x');
				//		return ''.$ttl.' ';
					}
					else
					{
						$src = $tp->replaceConstants(vartrue($parms['pre']).$value, 'abs');
						$alt = $src; //basename($value);
						$ttl = vartrue($parms['title'], 'LAN_PREVIEW');
						$value = ''.defset($ttl, $ttl).' ';
					}
				}
				elseif(!empty($parms['fallback']))
				{
					$value = $parms['fallback'];
					$thparms['class'] = 'thumbnail e-thumb fallback';
					return $tp->toImage($value, $thparms);
				}
			break;
			case 'media':
			case 'images':
				$firstItem = !empty($value[0]['path']) ? $value[0]['path'] : null; // display first item.
				return e107::getMedia()->previewTag($firstItem, $parms);
			break;
			
			case 'files':
				if(!empty($value))
				{
					if(!is_array($value))
					{
						return "Type 'files' must have a data type of 'array' or 'json'";
					}
					$ret = '';
					for ($i=0; $i < 5; $i++)
					{
						$ival 	= $value[$i]['path'];
						if(empty($ival))
						{
							continue;
						}
						$ret .=  ''.$ival.' ';
					}
					$ret .= ' ';
					$value = $ret;
				}
			break; 
			
			case 'datestamp':
				$value = $value ? e107::getDate()->convert_date($value, vartrue($parms['mask'], 'short')) : '';
			break;
			
			case 'date':
				if(empty($attributes['noedit']) && !empty($parms['editable']) && empty($parms['link'])) // avoid bad markup, better solution coming up
				{
					$value = $this->renderInline($field,$id,$attributes['title'],$value, $value);
				}
				// just show original value
			break;
			case 'userclass':
				$dispvalue = $this->_uc->getName($value);
					// Inline Editing.  
				if(empty($attributes['noedit']) && !empty($parms['editable']) && empty($parms['link'])) // avoid bad markup, better solution coming up
				{
					// $mode = preg_replace('/[^\w]/', '', vartrue($_GET['mode'], ''));
					$uc_options = vartrue($parms['classlist'], 'public,guest,nobody,member,admin,main,classes'); // defaults to 'public,guest,nobody,member,classes' (userclass handler)
					unset($parms['classlist']);
					$array = e107::getUserClass()->uc_required_class_list($uc_options); //XXX Ugly looking (non-standard) function naming - TODO discuss name change.
					$value = $this->renderInline($field, $id, $attributes['title'], $value, $dispvalue, 'select', $array, array('placement'=>'left'));
				}
				else 
				{
					$value = $dispvalue;	
				}
			break;
			case 'userclasses':
			//	return $value;
				$classes = explode(',', $value);
				$uv = array();
				foreach ($classes as $cid)
				{
					if(!empty($parms['defaultLabel']) && $cid === '')
					{
						$uv[] = $parms['defaultLabel'];
						continue;
					}
					$uv[] = $this->_uc->getName($cid);
				}
				$dispvalue = implode(vartrue($parms['separator'], '".$dispvalue. ' ';
				}
				else 
				{
					$value = $dispvalue;	
				}
				unset($parms['classlist']);
				
			break;
			/*case 'user_name':
			case 'user_loginname':
			case 'user_login':
			case 'user_customtitle':
			case 'user_email':*/
			case 'user':
				
				/*if(is_numeric($value))
				{
					$value = e107::user($value);
					if($value)
					{
						$value = $value[$attributes['type']] ? $value[$attributes['type']] : $value['user_name'];
					}
					else
					{
						$value = 'not found';
					}
				}*/
				$row_id = $id;
				// Dirty, but the only way for now
				$id = 0;
				$ttl = LAN_ANONYMOUS;
				//Defaults to user_id and user_name (when present) and when idField and nameField are not present.
				// previously set - real parameters are idField && nameField
				$id = vartrue($parms['__idval']);
				if($value && !is_numeric($value))
				{
					$id = vartrue($parms['__idval']);
					$ttl = $value;
				}
				elseif($value && is_numeric($value))
				{
					$id = $value;
					if (vartrue($parms['__nameval']))
					{
						$ttl = $parms['__nameval'];
					}
					else
					{
						$user = e107::user($value);
						if (vartrue($user['user_name']))
						{
							$ttl = $user['user_name'];
						}
					}
				}
				if(!empty($parms['link']) && $id && $ttl && is_numeric($id))
				{
					// Stay in admin area.
					$link = e_ADMIN. 'users.php?mode=main&action=edit&id=' .$id. '&readonly=1&iframe=1'; // e107::getUrl()->create('user/profile/view', array('id' => $id, 'name' => $ttl))
					$value = ''.$ttl.' ';
				}
				else
				{
					$value = $ttl;
				}
				// Inline Editing.
				if(!vartrue($attributes['noedit']) && vartrue($parms['editable']) && !vartrue($parms['link'])) // avoid bad markup, better solution coming up
				{
					// Need a Unique Field ID to store field settings using e107::js('settings').
					$fieldID = $this->name2id($field . '_' . microtime(true));
					// Unique ID for each rows.
					$eEditableID = $this->name2id($fieldID . '_' . $row_id);
				//	$tpl = $this->userpicker($field, '', $ttl, $id, array('id' => $fieldID, 'selectize' => array('e_editable' => $eEditableID)));
					$tpl = $this->userpicker($fieldID, array('user_id'=>$id, 'user_name'=>$ttl),  array('id' => $fieldID, 'inline' => $eEditableID));
					$mode = preg_replace('/[\W]/', '', vartrue($_GET['mode']));
					$value = "" . $ttl . ' ';
				}
				
			break;
			/**
			 * $parms['true']  - label to use for true
			 * $parms['false'] - label to use for false
			 * $parms['enabled'] - alias of $parms['true']
			 * $parms['disabled'] - alias of $parms['false']
			 * $parms['reverse'] - use 0 for true and 1 for false.
			 */
			case 'bool':
			case 'boolean':
				$false = vartrue($parms['trueonly']) ? '' : ADMIN_FALSE_ICON;
				if(!empty($parms['enabled']))
				{
					$parms['true'] = $parms['enabled'];
				}
				if(!empty($parms['disabled']))
				{
					$parms['false'] = $parms['disabled'];
				}
				if(!vartrue($attributes['noedit']) && vartrue($parms['editable']) && !vartrue($parms['link'])) // avoid bad markup, better solution coming up
				{
					if(isset($parms['false'])) // custom representation for 'false'. (supports font-awesome when set by css)
					{
						$false = $parms['false'];	
					}
					else
					{
						// https://stackoverflow.com/questions/2965971/how-to-add-images-in-select-list
						$false = ($value === '') ? '□' : '⨯'; // "✗";
					}
					
					$true = varset($parms['true'], '✔' /*'✓'*/); // custom representation for 'true'. (supports font-awesome when set by css)
			//		$true = '';
			//		$false = '\f00d';
					$value = (int) $value;
							
					$wparms = (vartrue($parms['reverse'])) ? array(0=>$true, 1=>$false) : array(0=>$false, 1=>$true);
					$dispValue = $wparms[$value];
					$styleClass = '';
                    if($true ==='✔')
                    {
					    $styleClass = ($value === 1) ? 'admin-true-icon' : 'admin-false-icon';
                    }
					return $this->renderInline($field, $id, $attributes['title'], $value, $dispValue, 'select', $wparms, array('class'=>'e-editable-boolean '.$styleClass));
				}
				
				if(!empty($parms['reverse']))
				{
					$value = ($value) ? $false : ADMIN_TRUE_ICON;	
				}
				else
				{
					$value = $value ? ADMIN_TRUE_ICON : $false;	
				}	
							
			break;
			case 'url':
				if(!$value)
				{
					break;
				}
				$ttl = $value;
				if(!empty($parms['href']))
				{
					return $tp->replaceConstants(vartrue($parms['pre']).$value, varset($parms['replace_mod'],'abs'));
				}
				if(!empty($parms['truncate']))
				{
					$ttl = $tp->text_truncate($value, $parms['truncate'], '...');
				}
				$target = (!empty($parms['target'])) ? " target='".$parms['target']."' " : '';
				$class = (!empty($parms['class'])) ? " class='".$parms['class']."' " : '';
				$value = 'replaceConstants(vartrue($parms['pre']).$value, 'abs')."' title='{$value}'>".$ttl. ' ';
			break;
			case 'email':
				if(!$value)
				{
					break;
				}
				$ttl = $value;
				if(!empty($parms['truncate']))
				{
					$ttl = $tp->text_truncate($value, $parms['truncate'], '...');
				}
				$target = (!empty($parms['target'])) ? " target='".$parms['target']."' " : '';
				$class = (!empty($parms['class'])) ? " class='".$parms['class']."' " : '';
				$value = '".$ttl. ' ';
			break;
			case 'method': // Custom Function			
				$method = varset($attributes['field']); // prevents table alias in method names. ie. u.my_method.
				$_value = $value;
				$meth = (!empty($attributes['method'])) ? $attributes['method'] : $method;
				if(strpos($meth,'::')!==false)
				{
					list($className,$meth) = explode('::', $meth);
					$cls = new $className();
				}
				else
				{
					$cls = $this;
				}
				if(method_exists($cls,$meth))
				{
					$parms['field'] = $field;
					$mode = (!empty($attributes['mode'])) ? $attributes['mode'] :'read';
					$value = call_user_func_array(array($cls, $meth), array($value, $mode, $parms));
				}
				else
				{
					$className = get_class($cls);
					e107::getDebug()->log('Missing Method: ' .$className. '::' .$meth. ' ' .print_a($attributes,true));
					return "Missing Method ";
				}
			//	 print_a($attributes);
					// Inline Editing.  
				if(empty($attributes['noedit']) && !empty($parms['editable'])) // avoid bad markup, better solution coming up
				{
					
					$mode = preg_replace('/[\W]/', '', vartrue($_GET['mode']));
					$methodParms = call_user_func_array(array($this, $meth), array($_value, 'inline', $parms));
					$inlineParms = (!empty($methodParms['inlineParms'])) ? $methodParms['inlineParms'] : null;
					if(!empty($methodParms['inlineType']))
					{
						$attributes['inline'] = $methodParms['inlineType'];
						$methodParms = (!empty($methodParms['inlineData'])) ? $methodParms['inlineData'] : null;
					}
					if(is_string($attributes['inline'])) // text, textarea, select, checklist. 
					{
						switch ($attributes['inline']) 
						{
					
							case 'checklist':
								$xtype = 'checklist';		
							break;
							
							case 'select':
							case 'dropdown':
								$xtype = 'select';		
							break;
							
							case 'textarea':
								$xtype = 'textarea';		
							break;
							
							
							default:
								$xtype = 'text';
								 $methodParms = null;
							break;
						}
					}
					if(!empty($xtype))
					{
						$value = varset($inlineParms['pre']).$this->renderInline($field, $id, $attributes['title'], $_value, $value, $xtype, $methodParms,$inlineParms).varset($inlineParms['post']);
					}
				}
							
			break;
			case 'hidden':
				if(!empty($parms['show']))
				{
					return ($value ?: vartrue($parms['empty']));
				}
				return '';
			break;
			
			case 'language': // All Known Languages. 
					
				if(!empty($value))
				{
					$_value = $value;
					if(strlen($value) === 2)
					{
						$value = e107::getLanguage()->convert($value);
					}
				}
				
				if(!vartrue($attributes['noedit']) && vartrue($parms['editable'])) 
				{
					$wparms = e107::getLanguage()->getList();
					return $this->renderInline($field, $id, $attributes['title'], $_value, $value, 'select', $wparms);	
				}	
				
				return $value;
				
			break;
			case 'lanlist': // installed languages. 
				$options = e107::getLanguage()->getLanSelectArray();
				if($options) // FIXME - add support for multi-level arrays (option groups)
				{
					if(!is_array($attributes['writeParms']))
					{
						parse_str($attributes['writeParms'], $attributes['writeParms']);
					}
					$wparms = $attributes['writeParms'];
					if(!is_array(varset($wparms['__options'])))
					{
						parse_str($wparms['__options'], $wparms['__options']);
					}
					$opts = $wparms['__options'];
					if($opts['multiple'])
					{
						$ret = array();
						$value = is_array($value) ? $value : explode(',', $value);
						foreach ($value as $v)
						{
							if(isset($options[$v]))
							{
								$ret[] = $options[$v];
							}
						}
						$value = implode(', ', $ret);
					}
					else
					{
						$ret = '';
						if(isset($options[$value]))
						{
							$ret = $options[$value];
						}
						$value = $ret;
					}
					$value = ($value ? vartrue($parms['pre']).$value.vartrue($parms['post']) : '');
				}
				else
				{
					$value = '';
				}
			break;
			//TODO - order
			default:
				$value = $this->renderLink($value,$parms,$id);
				//unknown type
			break;
		}
		return $value;
	}
	/**
	 * Auto-render Form Element
	 * @param string $key
	 * @param mixed $value
	 * @param array $attributes field attributes including render parameters, element options - see e_admin_ui::$fields for required format
	 * #param array (under construction) $required_data required array as defined in e_model/validator
	 * @param mixed $attributes['writeParms']['default'] default value when empty (or default option when type='dropdown')
	 * @param mixed $attributes['writeParms']['defaultValue'] default option value when type='dropdown'
	 * @param mixed $attributes['writeParms']['empty'] default value when value is empty (dropdown and hidden only right now)
	 * @return string
	 */
	public function renderElement($key, $value, $attributes, $required_data = array(), $id = 0)
	{
		if(!empty($value) && !empty($attributes['data']) && ($attributes['data'] === 'array' || $attributes['data'] === 'json'))
		{
			$value = e107::unserialize($value);
		}
		$tp = e107::getParser();
		$ret = '';
		$parms = vartrue($attributes['writeParms'], array());
		if($tmpOpt = $tp->isJSON($parms))
		{
			$parms = $tmpOpt;
			unset($tmpOpt);
		}
		if(is_string($parms))
		{
			parse_str($parms, $parms);
		}
		$ajaxParms = array();
		if(!empty($parms['ajax']))
		{
			$ajaxParms['data-src'] = varset($parms['ajax']['src']);
			$ajaxParms['data-target'] = varset($parms['ajax']['target']);
			$ajaxParms['data-method'] = varset($parms['ajax']['method'], 'html');
			$ajaxParms['data-loading'] = varset($parms['ajax']['loading'], 'fa-spinner'); //$tp->toGlyph('fa-spinner', array('spin'=>1))
			unset($attributes['writeParms']['ajax']);
		//	e107::getDebug()->log($parms['ajax']);
		}
		if(!empty($attributes['multilan']))
		{
			$value = is_array($value) ? varset($value[e_LANGUAGE]) : $value;
			$parms['post'] = "".$tp->toGlyph('fa-language'). ' ' .varset($parms['post']);
			$key .= '[' . e_LANGUAGE . ']';
		}
		if(empty($value) && !empty($parms['default']) && $attributes['type'] !== 'dropdown') // Allow writeParms to set default value.
		{
			$value = $parms['default'];
		}
		// Two modes of read-only. 1 = read-only, but only when there is a value, 2 = read-only regardless.
		if(!empty($attributes['readonly']) && (!empty($value) || vartrue($attributes['readonly'])===2)) // quick fix (maybe 'noedit'=>'readonly'?)
		{
			if(!empty($attributes['writeParms'])) // eg. different size thumbnail on the edit page.
			{
				$attributes['readParms'] = $attributes['writeParms'];
			}
			return $this->renderValue($key, $value, $attributes).$this->hidden($key, $value); //
		}
		
		// FIXME standard - writeParams['__options'] is introduced for list elements, bundle adding to writeParms is non reliable way
		$writeParamsOptionable =  array('dropdown', 'comma', 'radio', 'lanlist', 'language', 'user');
		$writeParamsDisabled =  array('layouts', 'templates', 'userclass', 'userclasses');
		// FIXME it breaks all list like elements - dropdowns, radio, etc
		if(!empty($required_data[0]) || !empty($attributes['required'])) // HTML5 'required' attribute
		{
			// FIXME - another approach, raise standards, remove checks
			if(in_array($attributes['type'], $writeParamsOptionable))
			{
				$parms['__options']['required'] = 1;	
			}
			elseif(!in_array($attributes['type'], $writeParamsDisabled))
			{
				$parms['required'] = 1;	
			}
		}
		
		// FIXME it breaks all list like elements - dropdowns, radio, etc
		if(!empty($required_data[3]) || !empty($attributes['pattern'])) // HTML5 'pattern' attribute
		{
			// FIXME - another approach, raise standards, remove checks
			if(in_array($attributes['type'], $writeParamsOptionable))
			{
				$parms['__options']['pattern'] = vartrue($attributes['pattern'], $required_data[3]);
			}
			elseif(!in_array($attributes['type'], $writeParamsDisabled))
			{
				$parms['pattern'] = vartrue($attributes['pattern'], $required_data[3]);	
			}
		}
		// XXX Fixes For the above.  - use optArray variable. eg. $field['key']['writeParms']['optArray'] = array('one','two','three');
		if(($attributes['type'] === 'dropdown' || $attributes['type'] === 'radio' || $attributes['type'] === 'checkboxes') && isset($parms['optArray']))
		{
			$fopts = $parms;
			$parms = $fopts['optArray'];
			unset($fopts['optArray']);
			$parms['__options'] = $fopts;
		}
		$this->renderElementTrigger($key, $value, $parms, $required_data, $id);
		
		switch($attributes['type'])
		{
			case 'number':
				$maxlength = vartrue($parms['maxlength'], 255);
				unset($parms['maxlength']);
				if(!vartrue($parms['size']))
				{
					$parms['size'] = 'small';
				}
				if(!vartrue($parms['class']))
				{
					$parms['class'] = 'tbox number e-spinner ';
				}
				if(!$value)
				{
					$value = '0';
				}
				$ret =  vartrue($parms['pre']).$this->number($key, $value, $maxlength, $parms).vartrue($parms['post']);
			break;
			case 'country':
				$ret = vartrue($parms['pre']).$this->country($key, $value, $parms).vartrue($parms['post']);
			break;
			case 'ip':
				$ret = vartrue($parms['pre']).$this->text($key, e107::getIPHandler()->ipDecode($value), 32, $parms).vartrue($parms['post']);
			break;
			case 'email':
				$maxlength = vartrue($parms['maxlength'], 255);
				unset($parms['maxlength']);
				$ret =  vartrue($parms['pre']).$this->email($key, $value, $maxlength, $parms).vartrue($parms['post']); // vartrue($parms['__options']) is limited. See 'required'=>true
			break;
			case 'url':
				$maxlength = vartrue($parms['maxlength'], 255);
				unset($parms['maxlength']);
				$ret =  vartrue($parms['pre']).$this->url($key, $value, $maxlength, $parms).vartrue($parms['post']); // vartrue($parms['__options']) is limited. See 'required'=>true
		
			break; 
		//	case 'email':
		
			case 'password': // encrypts to md5 when saved. 
				$maxlength = vartrue($parms['maxlength'], 255);
				unset($parms['maxlength']);
				if(!isset($parms['required']))
				{
					$parms['required'] = false;
				}
				$ret =  vartrue($parms['pre']).$this->password($key, $value, $maxlength, $parms).vartrue($parms['post']); // vartrue($parms['__options']) is limited. See 'required'=>true
			
			break; 
			case 'text':
			case 'progressbar':
				$maxlength = vartrue($parms['maxlength'], 255);
				unset($parms['maxlength']);
				if(!empty($parms['sef']) && e_LANGUAGE !== 'Japanese' && e_LANGUAGE !== 'Korean' && e_LANGUAGE !== 'Hebrew') // unsupported languages.(FIXME there are more)
				{
					$sefSource = $this->name2id($parms['sef']);
					$sefTarget = $this->name2id($key);
					if(!empty($parms['tdClassRight']))
					{
						$parms['tdClassRight'] .= 'input-group';
					}
					else
					{
						$parms['tdClassRight'] = 'input-group';
					}
					$parms['post'] = "' .LAN_GENERATE. ' ".$ret. ' ';
				}
				
			break;
			
			case 'tags':
				$maxlength = vartrue($parms['maxlength'], 255);
				$ret =  vartrue($parms['pre']).$this->tags($key, $value, $maxlength, $parms).vartrue($parms['post']); // vartrue($parms['__options']) is limited. See 'required'=>true
			break;
			case 'textarea':
				$text = '';
				if(!empty($parms['append']) && !empty($value)) // similar to comments - TODO TBD. a 'comment' field type may be better.
				{
					$attributes['readParms'] = 'bb=1';
					
					$text = $this->renderValue($key, $value, $attributes);					
					$text .= '";
				for ($i=0; $i < $max; $i++)
				{				
					$k 		= $key.'['.$i.'][path]';
					$ival 	= $value[$i]['path'];
					
					$ret .=  $this->imagepicker($k, $ival, defset($label, $label), $parms);		
				}
				$ret .= '
';
			break;
			/** Generic Media Pick for combinations of images, audio, video, glyphs, files, etc. Field Type = json */
			case 'media':
				$max = varset($parms['max'],1);
				$ret = "";
				for ($i=0; $i < $max; $i++)
				{
					$k 		= $key.'['.$i.'][path]';
					$ival 	= isset($value[$i]) ? $value[$i]['path'] : '';
					$ret .=  $this->mediapicker($k, $ival, $parms);
				}
				$ret .= '
';
				return $ret;
			break;
			
			case 'files':
				$label = varset($parms['label'], 'LAN_EDIT');
				if(!empty($attributes['data']) && ($attributes['data'] === 'array' || $attributes['data'] === 'json'))
				{
					$parms['data'] = 'array';	
				}
				$ret = '';
				for ($i=0; $i < 5; $i++) 
				{				
				//	$k 		= $key.'['.$i.'][path]';
				//	$ival 	= $value[$i]['path'];
					$k 		= $key.'['.$i.']';
					$ival 	= $value[$i];
					$ret .=  ''.$this->filepicker($k, $ival, defset($label, $label), $parms).' ';		
				}
				$ret .= ' ';
			break;
			
			case 'file': //TODO - thumb, image list shortcode, js tooltip...
				$label = varset($parms['label'], 'LAN_EDIT');
				unset($parms['label']);
				$ret =  $this->filepicker($key, $value, defset($label, $label), $parms);
			break;
			case 'icon':
				$label = varset($parms['label'], 'LAN_EDIT');
				$ajax = varset($parms['ajax'], true) ? true : false;
				unset($parms['label'], $parms['ajax']);
				$ret =  $this->iconpicker($key, $value, defset($label, $label), $parms, $ajax);
			break;
			case 'date': // date will show the datepicker but won't convert the value to unix. ie. string value will be saved. (or may be processed manually with beforeCreate() etc. Format may be determined by $parm. 
			case 'datestamp':
				// If hidden, value is updated regardless. eg. a 'last updated' field.
				// If not hidden, and there is a value, it is retained. eg. during the update of an existing record.
				// otherwise it is added. eg. during the creation of a new record.
				if(!empty($parms['auto']) && (($value == null) || !empty($parms['hidden'])))
				{
					$value = time();
				}
				
				if(!empty($parms['readonly'])) // different to 'attribute-readonly' since the value may be auto-generated.
				{
					$ret =  $this->renderValue($key, $value, $attributes).$this->hidden($key, $value);
				}
				elseif(!empty($parms['hidden']))
				{
					$ret =  $this->hidden($key, $value);
				}
				else
				{
					$ret =  $this->datepicker($key, $value, $parms);	
				}				
			break;
			case 'layouts': //to do - exclude param (exact match)
				$location   = varset($parms['plugin']); // empty - core
				$ilocation  = vartrue($parms['id'], $location); // omit if same as plugin name
				$where      = vartrue($parms['area'], 'front'); //default is 'front'
				$filter     = varset($parms['filter']);
				$merge      = isset($parms['merge']) ? (bool) $parms['merge'] : true;
				$layouts = e107::getLayouts($location, $ilocation, $where, $filter, $merge, false);
				return vartrue($parms['pre']).$this->select($key, $layouts,$value,$parms).vartrue($parms['post']);
			/*	if($tmp = e107::getTemplateInfo($location,$ilocation, null,true,$merge)) // read xxxx_INFO array from template file.
				{
					$opt = array();
					foreach($tmp as $k=>$inf)
					{
						$opt[$k] = $inf['title'];
					}
					return vartrue($parms['pre'],'').$this->select($key,$opt,$value,$parms).vartrue($parms['post'],'');
				}*/
/*
				if(varset($parms['default']) && !isset($layouts[0]['default']))
				{
					$layouts[0] = array('default' => $parms['default']) + $layouts[0];
				}
				$info = array();
				if($layouts[1])
				{
					foreach ($layouts[1] as $k => $info_array)
					{
						if(isset($info_array['description']))
						$info[$k] = defset($info_array['description'], $info_array['description']);
					}
				}
				*/
				//$this->selectbox($key, $layouts, $value)
			//	$ret =  (vartrue($parms['raw']) ? $layouts[0] : $this->radio_multi($key, $layouts[0], $value,array('sep'=>"Method ".$meth."  not found in ".get_class($cls)." 
";
				}
			break;
			case 'upload': //TODO - from method
				// TODO uploadfile SC is now processing uploads as well (add it to admin UI), write/readParms have to be added (see uploadfile.php parms)
				$disbut = varset($parms['disable_button'], '0');
				$ret =  $tp->parseTemplate('{UPLOADFILE=' .(vartrue($parms['path']) ? e107::getParser()->replaceConstants($parms['path']) : e_UPLOAD)."|nowarn&trigger=etrigger_uploadfiles&disable_button={$disbut}}");
			break;
			case 'hidden':
				$value = (isset($parms['value'])) ? $parms['value'] : $value;
				if(!empty($parms['show']))
				{
					$ret = ($value ?: varset($parms['empty'], $value));
				}
				else
				{
					$ret = '';
				}
				if(is_array($value) && ($attributes['data'] === 'json'))
				{
					$value = e107::serialize($value, 'json');
				}
				$ret .=  $this->hidden($key, $value);
			break;
			case 'lanlist': // installed languages
			case 'language': // all languages
				
				$options = ($attributes['type'] === 'language') ? e107::getLanguage()->getList() : e107::getLanguage()->getLanSelectArray();
				$eloptions  = vartrue($parms['__options'], array());
				if(!is_array($eloptions))
				{
					parse_str($eloptions, $eloptions);
				}
				unset($parms['__options']);
				if(vartrue($eloptions['multiple']) && !is_array($value))
				{
					$value = explode(',', $value);
				}
				$ret =  vartrue($eloptions['pre']).$this->select($key, $options, $value, $eloptions).vartrue($eloptions['post']);
			break;
			case null:
			//	Possibly used in db but should not be submitted in form. @see news_extended.
			break;
			default:// No LAN necessary, debug only. 
				$ret =  (ADMIN) ? "".LAN_ERROR." Unknown 'type' : ".$attributes['type'] . ' ' : $value;
			break;
		}
		if(!empty($parms['expand']))
		{
			$k = 'exp-' .$this->name2id($key);
			$text = "".$parms['expand']. ' ';
			$text .= vartrue($parms['help']) ? ''.$parms['help'].'
' : '';
			$text .= "".$ret. '
';
			return $text;	
		}
		else
		{
			/** @deprecated usage @see renderCreateFieldset() should be attributes['help'] */
			$ret .= vartrue($parms['help']) ? ''.$tp->toHTML($parms['help'],false,'defs').'
' : '';	
		}
		return $ret;
	}
	/**
	 * @param $name
	 * @param $value
	 * @param $parms
	 * @return string
	 */
	public function radioImage($name,$value,$parms)
	{
		if(!empty($parms['path']))
		{
			$parms['legacy'] = $parms['path'];
		}
		$text = '';
		$class = varset($parms['block-class'],'col-md-2');
		foreach($parms['optArray'] as $key=>$val)
		{
			$thumbnail    = e107::getParser()->toImage($val['thumbnail'],$parms);
			$selected = ($key === $value) ? " checked='checked'" : '';
			$active = ($key === $value) ? ' active' : '';
			$text .= "
							".$thumbnail. "
						";
			$text .= isset($val['label']) ? "
".$val['label']."
" : '';
				$text .= "
			
 ";
		}
		$text .= '
';
		foreach($parms['optArray'] as $key=>$val)
		{
			$thumbnail    = e107::getParser()->toImage($val,$parms);
			$selected = ($val == $value) ? ' checked' : '';
				$text .= "
									
										".$thumbnail. "
										";
				$text .= isset($parms['labels'][$key]) ? "
".$parms['labels'][$key]."
" : '';
				$text .= "
							
 ";
		}
		$text .= '
	 *  'myplugin', // unique string used for building element ids, REQUIRED
	 * 		'pid' => 'primary_id', // primary field name, REQUIRED
	 * 		'url' => '{e_PLUGIN}myplug/admin_config.php', // if not set, e_SELF is used
	 * 		'query' => 'mode=main&action=list', // or e_QUERY if not set
	 * 		'head_query' => 'mode=main&action=list', // without field, asc and from vars, REQUIRED
	 * 		'np_query' => 'mode=main&action=list', // without from var, REQUIRED for next/prev functionality
	 * 		'legend' => 'Fieldset Legend', // hidden by default
	 * 		'form_pre' => '', // markup to be added before opening form element (e.g. Filter form)
	 * 		'form_post' => '', // markup to be added after closing form element
	 * 		'fields' => array(...), // see e_admin_ui::$fields
	 * 		'fieldpref' => array(...), // see e_admin_ui::$fieldpref
	 * 		'table_pre' => '', // markup to be added before opening table element
	 * 		'table_post' => '', // markup to be added after closing table element (e.g. Batch actions)
	 * 		'fieldset_pre' => '', // markup to be added before opening fieldset element
	 * 		'fieldset_post' => '', // markup to be added after closing fieldset element
	 * 		'perPage' => 15, // if 0 - no next/prev navigation
	 * 		'from' => 0, // current page, default 0
	 * 		'field' => 'field_name', //current order field name, default - primary field
	 * 		'asc' => 'desc', //current 'order by' rule, default 'asc'
	 * );
	 * $tree_models['myplugin'] = new e_admin_tree_model($data);
	 * 
	 * TODO - move fieldset & table generation in separate methods, needed for ajax calls
	 * @param array $form_options
	 * @param e_admin_tree_model $tree_model
	 * @param boolean $nocontainer don't enclose form in div container
	 * @return string
	 */
	public function renderListForm($form_options, $tree_models, $nocontainer = false)
	{
		$tp = e107::getParser();
		$text = '';
		$formPre = '';
		$formPost = '';
		foreach ($form_options as $fid => $options)
		{
			list($type,$plugin) = explode('-',$fid,2);
			$plugin = str_replace('-','_',$plugin);
			e107::setRegistry('core/adminUI/currentPlugin', $plugin);
			/** @var e_tree_model $tree_model */
			$tree_model = $tree_models[$fid];
			$tree = $tree_model->getTree();
			$total = $tree_model->getTotal();
		
			$amount = $options['perPage'];
			$from = vartrue($options['from'], 0);
			$field = vartrue($options['field'], $options['pid']);
			$asc = strtoupper(vartrue($options['asc'], 'asc'));
			$elid = $fid;//$options['id'];
			$query = vartrue($options['query'],e_QUERY); //  ? $options['query'] :  ;
			if(vartrue($_GET['action']) === 'list')
			{
				$query = e_QUERY; //XXX Quick fix for loss of pagination after 'delete'. 	
			}
			$url = (isset($options['url']) ? $tp->replaceConstants($options['url'], 'abs') : e_SELF);
			$formurl = $url.($query ? '?'.$query : '');
			$fields = $options['fields'];
			$current_fields = varset($options['fieldpref']) ? $options['fieldpref'] : array_keys($options['fields']);
			$legend_class = vartrue($options['legend_class'], 'e-hideme');
			
	        $text .= "
				
			';
			e107::setRegistry('core/adminUI/currentPlugin');
			$formPre = vartrue($options['form_pre']);
			$formPost = vartrue($options['form_post']);
		}
		if(!$nocontainer)
		{
			$class = deftrue('e_IFRAME') ? 'e-container e-container-modal' : 'e-container';
			$text = ''.$text.'
';
		}
		return $formPre . $text . $formPost;
	}
	/**
	 * Used with 'carousel' generates slides with X number of cells/blocks per slide.
	 * @param $cells
	 * @param int $perPage
	 * @return array
	 */
	private function slides($cells, $perPage=12)
	{
		$tmp = '';
		$s = 0;
		$slides = array();
		foreach($cells as $cell)
		{
			$tmp .= $cell;
			$s++;
			if($s == $perPage)
			{
				$slides[] = array('text'=>$tmp);
				$tmp = '';
				$s = 0;
			}
		}
		if($s != $perPage && $s != 0)
		{
			$slides[] = array('text'=>$tmp);
		}
		return $slides;
	}
	/**
	 * Render Grid-list layout.  used internally by admin UI
	 * @param $form_options
	 * @param $tree_models
	 * @param bool|false $nocontainer
	 * @return string
	 */
	public function renderGridForm($form_options, $tree_models, $nocontainer = false)
	{
		$tp = e107::getParser();
		$text = '';
		// print_a($form_options);
		foreach ($form_options as $fid => $options)
		{
			$tree_model = $tree_models[$fid];
			$tree = $tree_model->getTree();
			$total = $tree_model->getTotal();
			$amount = $options['perPage'];
			$from = vartrue($options['from'], 0);
			$field = vartrue($options['field'], $options['pid']);
			$asc = strtoupper(vartrue($options['asc'], 'asc'));
			$elid = $fid;//$options['id'];
			$query = vartrue($options['query'],e_QUERY); //  ? $options['query'] :  ;
			if(vartrue($_GET['action']) === 'list')
			{
				$query = e_QUERY; //XXX Quick fix for loss of pagination after 'delete'.
			}
			$url = (isset($options['url']) ? $tp->replaceConstants($options['url'], 'abs') : e_SELF);
			$formurl = $url.($query ? '?'.$query : '');
			$fields = $options['fields'];
			$current_fields = varset($options['fieldpref']) ? $options['fieldpref'] : array_keys($options['fields']);
			$legend_class = vartrue($options['legend_class'], 'e-hideme');
	        $text .= "
				
			';
		}
		if(!$nocontainer)
		{
			$class = deftrue('e_IFRAME') ? 'e-container e-container-modal' : 'e-container';
			$text = ''.$text.'
';
		}
		return (vartrue($options['form_pre']).$text.vartrue($options['form_post']));
	}
	/**
	 * Generic DB Record Management Form.
	 * TODO - lans
	 * TODO - move fieldset & table generation in separate methods, needed for ajax calls
	 * Expected arrays format:
	 * 
	 *  'myplugin',
	 * 		'url' => '{e_PLUGIN}myplug/admin_config.php', //if not set, e_SELF is used
	 * 		'query' => 'mode=main&action=edit&id=1', //or e_QUERY if not set
	 * 		'tabs' => true, 	 *
	 *      'fieldsets' => array(
	 * 			'general' => array(
	 * 				'legend' => 'Fieldset Legend',
	 * 				'fields' => array(...), //see e_admin_ui::$fields
	 * 				'after_submit_options' => array('action' => 'Label'[,...]), // or true for default redirect options
	 * 				'after_submit_default' => 'action_name',
	 * 				'triggers' => 'auto', // standard create/update-cancel triggers
	 * 				//or custom trigger array in format array('sibmit' => array('Title', 'create', '1'), 'cancel') - trigger name - title, action, optional hidden value (in this case named sibmit_value)
	 * 			),
	 *
	 * 			'advanced' => array(
	 * 				'legend' => 'Fieldset Legend',
	 * 				'fields' => array(...), //see e_admin_ui::$fields
	 * 				'after_submit_options' => array('__default' => 'action_name' 'action' => 'Label'[,...]), // or true for default redirect options
	 * 				'triggers' => 'auto', // standard create/update-cancel triggers
	 * 				//or custom trigger array in format array('submit' => array('Title', 'create', '1'), 'cancel' => array('cancel', 'cancel')) - trigger name - title, action, optional hidden value (in this case named sibmit_value)
	 * 			)
	 * 		)
	 * );
	 * $models[0] = new e_admin_model($data);
	 * $models[0]->setFieldIdName('primary_id'); // you need to do it if you don't use your own admin model extension
	 * 
	 * @param array $forms numerical array
	 * @param array $models numerical array with values instance of e_admin_model
	 * @param boolean $nocontainer don't enclose in div container
	 * @return string
	 */
	public function renderCreateForm($forms, $models, $nocontainer = false)
	{
		$text = '';
		foreach ($forms as $fid => $form)
		{
			$model = $models[$fid];
			e107::setRegistry('core/adminUI/currentModel', $model);
			if(!is_object($model))
			{
				e107::getDebug()->log('No model object found with key ' .$fid);
			}
			$query = isset($form['query']) ? $form['query'] : e_QUERY ;
			$url = (isset($form['url']) ? e107::getParser()->replaceConstants($form['url'], 'abs') : e_SELF).($query ? '?'.$query : '');
			$curTab = (string) varset($_GET['tab'], '0');
			$text .= "
				
			';
			
			// e107::js('footer-inline',"Form.focusFirstElement('{$form['id']}-form');",'prototype');
			// e107::getJs()->footerInline("Form.focusFirstElement('{$form['id']}-form');");
		}
		if(!$nocontainer)
		{
			$class = deftrue('e_IFRAME') ? 'e-container e-container-modal' : 'e-container';
			$text = ''.$text.'
';
		}
		return $text;
	}
	/**
	 * Create form fieldset, called internal by {@link renderCreateForm())
	 *
	 * @param string $id field id
	 * @param array $fdata fieldset data
	 * @param object $model
	 * @return string | false
	 */
	public function renderCreateFieldset($id, $fdata, $model, $tab=0)
	{
		$start = vartrue($fdata['fieldset_pre'])."
			
				".vartrue($fdata['legend']). ' 
				' .vartrue($fdata['table_pre'])."
				';
			$text .= vartrue($fdata['table_post']);
			$text .= implode("\n", $hidden_fields);
			$text .= ' ';
			$text .= vartrue($fdata['fieldset_post']);
			return $text;
		}
		
		return false;
		
	}
	/**
	 * Render Create/Edit Fieldset Row.
	 * @param string $label
	 * @param string $control
	 * @param array $att
	 * @return string
	 */
	public function renderCreateFieldRow($label, $control, $att = array())
	{
		$writeParms = $att['writeParms'];
		if(vartrue($att['type']) === 'bbarea' || !empty($writeParms['nolabel']))
		{
			$text = "
			
			";
			$text .= (isset($writeParms['nolabel']) && $writeParms['nolabel'] == 2) ? '' : "".$label. '
';
			$text .= $control. '
			 			
			 
			';
			return $text;
		}
		$leftCellClass  = (!empty($writeParms['tdClassLeft'])) ? " class='".$writeParms['tdClassLeft']."'" : '';
		$rightCellClass = (!empty($writeParms['tdClassRight'])) ? " class='".$writeParms['tdClassRight']."'" : '';
		$trClass        = (!empty($writeParms['trClass'])) ? " class='".$writeParms['trClass']."'" : '';
		$text = "
				
					
						".$label."
					 
					
						".$control. '
					 
				 
				';
		return $text;
	}
	/**
	 * Render the submit buttons in the Create/Edit Form.
	 * @param array $fdata - admin-ui data such as $fields, $tabs, $after_submit_options etc.
	 * @param int $id Primary ID of the record being edited (only in edit-mode)
	 * @return string
	 */
	public function renderCreateButtonsBar($fdata, $id=null) // XXX Note model and $tab removed as of v2.3
	{
		$text = "
				
	
		';
		return $text;
	}
	/**
	 * Generic renderForm solution
	 * @param @forms
	 * @param @nocontainer
	 * @return string
	 */
	public function renderForm($forms, $nocontainer = false)
	{
		$text = '';
		foreach ($forms as $fid => $form)
		{
			$query = isset($form['query']) ? $form['query'] : e_QUERY ;
			$url = (isset($form['url']) ? e107::getParser()->replaceConstants($form['url'], 'abs') : e_SELF).($query ? '?'.$query : '');
			$text .= '
				' .vartrue($form['form_pre'])."
				
			' .vartrue($form['form_post']). '
			';
		}
		if(!$nocontainer)
		{
			$class = deftrue('e_IFRAME') ? 'e-container e-container-modal' : 'e-container';
			$text = ''.$text.'
';
		}
		return $text;
	}
  
    /**
     * Generic renderFieldset solution, will be split to renderTable, renderCol/Row/Box etc - Still in use. 
     */
	public function renderFieldset($id, $fdata)
	{
		$colgroup = '';
		if(vartrue($fdata['table_colgroup']))
		{
			$colgroup = "
				
			";
			foreach ($fdata['table_colgroup'] as $i => $colgr)
			{
				$colgroup .= ' 
			';
		}
		$text = vartrue($fdata['fieldset_pre'])."
			
				".vartrue($fdata['legend']). ' 
				' .vartrue($fdata['table_pre']). '
		';
		if(!empty($fdata['table_rows']) || !empty($fdata['table_body']))
		{
			$text .= "
				
					' .$note. '
					' .vartrue($fdata['table_post']). '
			';
		}
		$triggers = vartrue($fdata['triggers'], array());
		if($triggers)
		{
			$text .= "
				".vartrue($fdata['pre_triggers']). '
			';
			foreach ($triggers as $trigger => $tdata)
			{
				if(is_string($tdata))
				{
					$text .= $tdata;
					continue;
				}
				$text .= $this->admin_button('etrigger_'.$trigger, $tdata[0], $tdata[1]);
				if(isset($tdata[2]))
				{
					$text .= $this->hidden($trigger.'_value', $tdata[2]);
				}
			}
			$text .= '
';
		}
		$text .= '
			 
			' .vartrue($fdata['fieldset_post']). '
		';
		return $text;
	}
	
	/**
	 * Render Value Trigger - override to modify field/value/parameters
	 * @param string $field field name
	 * @param mixed $value field value
	 * @param array $params 'writeParams' key (see $controller->fields array)
	 * @param int $id record ID
	 */
	public function renderValueTrigger(&$field, &$value, &$params, $id)
	{
		
	}
	
	/**
	 * Render Element Trigger - override to modify field/value/parameters/validation data
	 * @param string $field field name
	 * @param mixed $value field value
	 * @param array $params 'writeParams' key (see $controller->fields array)
	 * @param array $required_data validation data
	 * @param int $id record ID
	 */
	public function renderElementTrigger(&$field, &$value, &$params, &$required_data, $id)
	{
		
	}
}
// DEPRECATED - use above methods instead ($frm)
class form 
{
	public function form_open($form_method, $form_action, $form_name = '', $form_target = '', $form_enctype = '', $form_js = '')
	{
		$method = ($form_method ? "method='".$form_method."'" : '');
		$target = ($form_target ? " target='".$form_target."'" : '');
		$name = ($form_name ? " id='".$form_name."' " : " id='myform'");
		return "\n";
	}
}
/*
Usage
echo $rs->form_open("post", e_SELF, "_blank");
echo $rs->form_text("testname", 100, "this is the value", 100, 0, "tooltip");
echo $rs->form_button("submit", "testsubmit", "SUBMIT!", "", "Click to submit");
echo $rs->form_button("reset", "testreset", "RESET!", "", "Click to reset");
echo $rs->form_textarea("textareaname", 10, 10, "Value", "overflow:hidden");
echo $rs->form_checkbox("testcheckbox", 1, 1);
echo $rs->form_checkbox("testcheckbox2", 2);
echo $rs->form_hidden("hiddenname", "hiddenvalue");
echo $rs->form_radio("testcheckbox", 1, 1);
echo $rs->form_radio("testcheckbox", 1);
echo $rs->form_file("testfile", "20");
echo $rs->form_select_open("testselect");
echo $rs->form_option("Option 1");
echo $rs->form_option("Option 2");
echo $rs->form_option("Option 3", 1, "defaultvalue");
echo $rs->form_option("Option 4");
echo $rs->form_select_close();
echo $rs->form_close();
*/