* @link http://www.pradosoft.com/
* @copyright Copyright © 2005 PradoSoft
* @license http://www.pradosoft.com/license/
* @version $Revision: $ $Date: $
* @package System.Web.UI.WebControls
*/
/**
* TTextBox class
*
* TTextBox displays a text box on the Web page for user input.
* The text displayed in the TTextBox control is determined by the
* {@link setText Text} property. You can create a SingleLine,
* a MultiLine, or a Password text box by setting
* the {@link setTextMode TextMode} property. If the TTextBox control
* is a multiline text box, the number of rows it displays is determined
* by the {@link setRows Rows} property, and the {@link setWrap Wrap} property
* can be used to determine whether to wrap the text in the component.
*
* To specify the display width of the text box, in characters, set
* the {@link setColumns Columns} property. To prevent the text displayed
* in the component from being modified, set the {@link setReadOnly ReadOnly}
* property to true. If you want to limit the user input to a specified number
* of characters, set the {@link setMaxLength MaxLength} property.
* To use AutoComplete feature, set the {@link setAutoCompleteType AutoCompleteType} property.
*
* If {@link setAutoPostBack AutoPostBack} is set true, updating the text box
* and then changing the focus out of it will cause postback action.
* And if {@link setCausesValidation CausesValidation} is true, validation will
* also be processed, which can be further restricted within
* a {@link setValidationGroup ValidationGroup}.
*
* WARNING: Be careful if you want to display the text collected via TTextBox.
* Malicious cross-site script may be injected in. You may use {@link getSafeText SafeText}
* to prevent this problem.
*
* NOTE: If you set {@link setWrap Wrap} to false or use {@link setAutoCompleteType AutoCompleteType},
* the generated HTML output for the textbox will not be XHTML-compatible.
* Currently, no alternatives are available.
*
* @author Qiang Xue
* @version $Revision: $ $Date: $
* @package System.Web.UI.WebControls
* @since 3.0
*/
class TTextBox extends TWebControl implements IPostBackDataHandler, IValidatable
{
/**
* Default number of rows (for MultiLine text box)
*/
const DEFAULT_ROWS=4;
/**
* Default number of columns (for MultiLine text box)
*/
const DEFAULT_COLUMNS=20;
/**
* @var array list of auto complete types
*/
private static $_autoCompleteTypes=array('BusinessCity','BusinessCountryRegion','BusinessFax','BusinessPhone','BusinessState','BusinessStreetAddress','BusinessUrl','BusinessZipCode','Cellular','Company','Department','Disabled','DisplayName','Email','FirstName','Gender','HomeCity','HomeCountryRegion','HomeFax','Homepage','HomePhone','HomeState','HomeStreetAddress','HomeZipCode','JobTitle','LastName','MiddleName','None','Notes','Office','Pager','Search');
/**
* @var mixed safe text parser
*/
private static $_safeTextParser=null;
/**
* @var string safe textbox content with javascript stripped off
*/
private $_safeText;
/**
* @return string tag name of the textbox
*/
protected function getTagName()
{
return ($this->getTextMode()==='MultiLine')?'textarea':'input';
}
/**
* Adds attribute name-value pairs to renderer.
* This method overrides the parent implementation with additional textbox specific attributes.
* @param THtmlWriter the writer used for the rendering purpose
*/
protected function addAttributesToRender($writer)
{
$page=$this->getPage();
$page->ensureRenderInForm($this);
if(($uid=$this->getUniqueID())!=='')
$writer->addAttribute('name',$uid);
if(($textMode=$this->getTextMode())==='MultiLine')
{
if(($rows=$this->getRows())<=0)
$rows=self::DEFAULT_ROWS;
if(($cols=$this->getColumns())<=0)
$cols=self::DEFAULT_COLUMNS;
$writer->addAttribute('rows',"$rows");
$writer->addAttribute('cols',"$cols");
if(!$this->getWrap())
$writer->addAttribute('wrap','off');
}
else
{
if($textMode==='SingleLine')
{
$writer->addAttribute('type','text');
if(($text=$this->getText())!=='')
$writer->addAttribute('value',$text);
if(($act=$this->getAutoCompleteType())!=='None')
{
if($act==='Disabled')
$writer->addAttribute('autocomplete','off');
else if($act==='Search')
$writer->addAttribute('vcard_name','search');
else if($act==='HomeCountryRegion')
$writer->addAttribute('vcard_name','HomeCountry');
else if($act==='BusinessCountryRegion')
$writer->addAttribute('vcard_name','BusinessCountry');
else
{
if(($pos=strpos($act,'Business'))===0)
$act='Business'.'.'.substr($act,8);
else if(($pos=strpos($act,'Home'))===0)
$act='Home'.'.'.substr($act,4);
$writer->addAttribute('vcard_name','vCard.'.$act);
}
}
}
else
{
$writer->addAttribute('type','password');
}
if(($cols=$this->getColumns())>0)
$writer->addAttribute('size',"$cols");
if(($maxLength=$this->getMaxLength())>0)
$writer->addAttribute('maxlength',"$maxLength");
}
if($this->getReadOnly())
$writer->addAttribute('readonly','readonly');
if(!$this->getEnabled(true) && $this->getEnabled()) // in this case parent will not render 'disabled'
$writer->addAttribute('disabled','disabled');
if($this->getEnabled(true) && $this->getAutoPostBack() && $page->getClientSupportsJavaScript())
{
$writer->addAttribute('id',$this->getClientID());
$this->getPage()->getClientScript()->registerPostBackControl($this);
}
parent::addAttributesToRender($writer);
}
/**
* Sets the post back options for this textbox.
* @return TPostBackOptions
*/
public function getPostBackOptions()
{
$options['EventTarget'] = $this->getUniqueID();
$options['CausesValidation'] = $this->getCausesValidation();
$options['ValidationGroup'] = $this->getValidationGroup();
$options['TextMode'] = $this->getTextMode();
return $options;
}
/**
* Loads user input data.
* This method is primarly used by framework developers.
* @param string the key that can be used to retrieve data from the input data collection
* @param array the input data collection
* @return boolean whether the data of the component has been changed
*/
public function loadPostData($key,$values)
{
$value=$values[$key];
if(!$this->getReadOnly() && $this->getText()!==$value)
{
$this->setText($value);
return true;
}
else
return false;
}
/**
* Returns the value to be validated.
* This methid is required by IValidatable interface.
* @return mixed the value of the property to be validated.
*/
public function getValidationPropertyValue()
{
return $this->getText();
}
/**
* Raises OnTextChanged event.
* This method is invoked when the value of the {@link getText Text}
* property changes on postback.
* If you override this method, be sure to call the parent implementation to ensure
* the invocation of the attached event handlers.
* @param TEventParameter event parameter to be passed to the event handlers
*/
public function onTextChanged($param)
{
$this->raiseEvent('OnTextChanged',$this,$param);
}
/**
* Raises postdata changed event.
* This method is required by {@link IPostBackDataHandler} interface.
* It is invoked by the framework when {@link getText Text} property
* is changed on postback.
* This method is primarly used by framework developers.
*/
public function raisePostDataChangedEvent()
{
if($this->getAutoPostBack() && $this->getCausesValidation())
$this->getPage()->validate($this->getValidationGroup());
$this->onTextChanged(null);
}
/**
* Renders the body content of the textbox when it is in MultiLine text mode.
* @param THtmlWriter the writer for rendering
*/
protected function renderContents($writer)
{
if($this->getTextMode()==='MultiLine')
$writer->write(THttpUtility::htmlEncode($this->getText()));
}
/**
* @return string the AutoComplete type of the textbox
*/
public function getAutoCompleteType()
{
return $this->getViewState('AutoCompleteType','None');
}
/**
* @param string the AutoComplete type of the textbox, default value is 'None'.
* Valid values include:
* 'BusinessCity','BusinessCountryRegion','BusinessFax','BusinessPhone',
* 'BusinessState','BusinessStreetAddress','BusinessUrl','BusinessZipCode',
* 'Cellular','Company','Department','Disabled','DisplayName','Email',
* 'FirstName','Gender','HomeCity','HomeCountryRegion','HomeFax','Homepage',
* 'HomePhone','HomeState','HomeStreetAddress','HomeZipCode','JobTitle',
* 'LastName','MiddleName','None','Notes','Office','Pager','Search'
* @throws TInvalidDataValueException if the input parameter is not a valid AutoComplete type
*/
public function setAutoCompleteType($value)
{
$this->setViewState('AutoCompleteType',TPropertyValue::ensureEnum($value,self::$_autoCompleteTypes),'None');
}
/**
* @return boolean a value indicating whether an automatic postback to the server
* will occur whenever the user modifies the text in the TTextBox control and
* then tabs out of the component. Defaults to false.
*/
public function getAutoPostBack()
{
return $this->getViewState('AutoPostBack',false);
}
/**
* Sets the value indicating if postback automatically.
* An automatic postback to the server will occur whenever the user
* modifies the text in the TTextBox control and then tabs out of the component.
* @param boolean the value indicating if postback automatically
*/
public function setAutoPostBack($value)
{
$this->setViewState('AutoPostBack',TPropertyValue::ensureBoolean($value),false);
}
/**
* @return boolean whether postback event trigger by this text box will cause input validation, default is true.
*/
public function getCausesValidation()
{
return $this->getViewState('CausesValidation',true);
}
/**
* @param boolean whether postback event trigger by this text box will cause input validation.
*/
public function setCausesValidation($value)
{
$this->setViewState('CausesValidation',TPropertyValue::ensureBoolean($value),true);
}
/**
* @return integer the display width of the text box in characters, default is 0 meaning not set.
*/
public function getColumns()
{
return $this->getViewState('Columns',0);
}
/**
* Sets the display width of the text box in characters.
* @param integer the display width, set it 0 to clear the setting
*/
public function setColumns($value)
{
$this->setViewState('Columns',TPropertyValue::ensureInteger($value),0);
}
/**
* @return integer the maximum number of characters allowed in the text box, default is 0 meaning not set.
*/
public function getMaxLength()
{
return $this->getViewState('MaxLength',0);
}
/**
* Sets the maximum number of characters allowed in the text box.
* @param integer the maximum length, set it 0 to clear the setting
*/
public function setMaxLength($value)
{
$this->setViewState('MaxLength',TPropertyValue::ensureInteger($value),0);
}
/**
* @return boolean whether the textbox is read only, default is false.
*/
public function getReadOnly()
{
return $this->getViewState('ReadOnly',false);
}
/**
* @param boolean whether the textbox is read only
*/
public function setReadOnly($value)
{
$this->setViewState('ReadOnly',TPropertyValue::ensureBoolean($value),false);
}
/**
* @return integer the number of rows displayed in a multiline text box, default is 4
*/
public function getRows()
{
return $this->getViewState('Rows',self::DEFAULT_ROWS);
}
/**
* Sets the number of rows displayed in a multiline text box.
* @param integer the number of rows
*/
public function setRows($value)
{
$this->setViewState('Rows',TPropertyValue::ensureInteger($value),self::DEFAULT_ROWS);
}
/**
* @return string the text content of the TTextBox control.
*/
public function getText()
{
return $this->getViewState('Text','');
}
/**
* Sets the text content of the TTextBox control.
* @param string the text content
*/
public function setText($value)
{
$this->setViewState('Text',$value,'');
$this->_safeText = null;
}
/**
* @return string safe text content with javascript stripped off
*/
public function getSafeText()
{
if($this->_safeText===null)
$this->_safeText=$this->getSafeTextParser()->parse($this->getText());
return $this->_safeText;
}
/**
* @return mixed safe text parser
*/
protected function getSafeTextParser()
{
if(!self::$_safeTextParser)
self::$_safeTextParser=Prado::createComponent('System.3rdParty.SafeHtml.TSafeHtmlParser');
return self::$_safeTextParser;
}
/**
* @return string the behavior mode (SingleLine, MultiLine, or Password) of the TTextBox component. Defaults to SingleLine.
*/
public function getTextMode()
{
return $this->getViewState('TextMode','SingleLine');
}
/**
* Sets the behavior mode (SingleLine, MultiLine, or Password) of the TTextBox component.
* @param string the text mode
* @throws TInvalidDataValueException if the input value is not a valid text mode.
*/
public function setTextMode($value)
{
$this->setViewState('TextMode',TPropertyValue::ensureEnum($value,array('SingleLine','MultiLine','Password')),'SingleLine');
}
/**
* @return string the group of validators which the text box causes validation upon postback
*/
public function getValidationGroup()
{
return $this->getViewState('ValidationGroup','');
}
/**
* @param string the group of validators which the text box causes validation upon postback
*/
public function setValidationGroup($value)
{
$this->setViewState('ValidationGroup',$value,'');
}
/**
* @return boolean whether the text content wraps within a multiline text box. Defaults to true.
*/
public function getWrap()
{
return $this->getViewState('Wrap',true);
}
/**
* Sets the value indicating whether the text content wraps within a multiline text box.
* @param boolean whether the text content wraps within a multiline text box.
*/
public function setWrap($value)
{
$this->setViewState('Wrap',TPropertyValue::ensureBoolean($value),true);
}
}
?>