View file fs-phpfox340b2/upload/include/library/phpfox/parse/input.class.php

File size: 29.91Kb
<?php
/**
 * [PHPFOX_HEADER]
 */

defined('PHPFOX') or exit('NO DICE!');

/**
 * Input Parser
 * Class is used to parse all incoming data sent by end users via HTML forms.
 * Goal is to remove any "evil" data and convert it into safe HTML. This allows
 * us to store the original data and a safe version.
 *
 * @copyright		[PHPFOX_COPYRIGHT]
 * @author			Raymond Benc
 * @package 		Phpfox
 * @version 		$Id: input.class.php 4736 2012-09-24 08:36:44Z Raymond_Benc $
 */
class Phpfox_Parse_Input
{
	/**
	 * Invalid events we need to remove.
	 *
	 * @var array
	 */
	private $_aEvilEvents = array(
		'onActivate',
        'onAfterPrint',
		'onBeforePrint',
        'onAfterUpdate',
        'onBeforeUpdate',
        'onErrorUpdate',
        'onAbort',
        'onBeforeDeactivate',
        'onDeactivate',
        'onBeforeCopy',
        'onBeforeCut',
        'onBeforeEditFocus',
        'onBeforePaste',
        'onBeforeUnload',
        'onBlur',
        'onBounce',
        'onChange',
        'onClick',
        'onControlSelect',
        'onCopy',
        'onCut',
        'onDblClick',
        'onDrag',
        'onDragEnter',
        'onDragLeave',
        'onDragOver',
        'onDragStart',
        'onDrop',
        'onFilterChange',
        'onDragDrop',
        'onError',
        'onFilterChange',
        'onFinish',
        'onFocus',
        'onHelp',
        'onKeyDown',
        'onKeyPress',
        'onKeyUp',
        'onLoad',
        'OnLoseCapture',
        'onMouseDown',
        'onMouseEnter',
        'onMouseLeave',
        'onMouseMove',
        'onMouseOut',
        'onMouseOver',
        'onMouseUp',
        'onMove',
        'onPaste',
        'onPropertyChange',
        'onReadyStateChange',
        'onReset',
        'onResize',
        'onResizeEnd',
        'onResizeStart',
        'onScroll',
        'onSelectStart',
        'onSelect',
        'onSelectionChange',
        'onStart',
        'onStop',
        'onSubmit',
        'onUnload',
        'class',
        'style'
	);

	/**
	 * Invalid CSS properties we need to remove.
	 *
	 * @var array
	 */
	private $_aEvilCss = array(
		'position',
		'top',
		'left',
		'background',
		'background-image',
		'background-color',
		'width',
		'height',
		'behaviour'
	);

	/**
	 * REGEX used to get params from within {}
	 *
	 * @var string
	 */
	private $_sDbQstrRegexp = '"[^"\\\\]*(?:\\\\.[^"\\\\]*)*"';

	/**
	 * REGEX used to get params from within {}
	 *
	 * @var string
	 */	
	private $_sSiQstrRegexp = '\'[^\'\\\\]*(?:\\\\.[^\'\\\\]*)*\'';
	
	/**
	 * Store text we are parsing temp.
	 *
	 * @var string
	 */
	private $_sText = '';
	
	/**
	 * HTML tags we allow.
	 *
	 * @var array
	 */
	private $_aAllowedTags = array();

	/**
	 * Class constructor. Prepare regex for usgae a little later on in the script.
	 *
	 */
	public function __construct()
	{
		(($sPlugin = Phpfox_Plugin::get('parse_input_construct')) ? eval($sPlugin) : null);
		
		$this->_sQstrRegexp = '(?:' . $this->_sDbQstrRegexp . '|' . $this->_sSiQstrRegexp . ')';
	}

	/**
	 * Parse and clean a string. We mainly use this for a title of an item, which
	 * does not allow any HTML. It can also be used to shorten a string bassed on 
	 * the numerical value passed by the 2nd argument.
	 *
	 * @param string $sTxt Text to parse.
	 * @param int $iShorten (Optional) Define how short you want the string to be.
	 * @return string Returns the new parsed string.
	 */
	public function clean($sTxt, $iShorten = null)
	{
		$sTxt = Phpfox::getLib('parse.output')->htmlspecialchars($sTxt);

		// Parse for language package
		$sTxt = $this->_utf8ToUnicode($sTxt);
		$sTxt = str_replace('\\', '&#92;', $sTxt);
		
		if ($iShorten !== null)
		{			
			$sTxt = $this->_shorten($sTxt, $iShorten);
		}	
		
		return $sTxt;
	}

	/**
	 * Shortens a string respecting non english characters
	 * @param string $sTxt string to shorten
	 * @param int $iLetters how many characters must the resulting string have
	 * @return string shortened string
	 */
	private function _shorten($sTxt, $iLetters)
	{
		if (!preg_match('/(&#[0-9]+;)/', $sTxt))
		{
			return substr($sTxt, 0, $iLetters);
		}
		$sOut = '';
		$iOutLen = 0;
		$iPos = 0; 
		$iTxtLen = strlen($sTxt);
		for ($iPos; $iPos < $iTxtLen && $iOutLen <= $iLetters; $iPos++)
		{
			if ($sTxt[$iPos] == '&')
			{
				$iEnd = strpos($sTxt, ';', $iPos) + 1;
				$sTemp = substr($sTxt, $iPos, $iEnd - $iPos);
				if (preg_match('/(&#[0-9]+;)/', $sTemp))
				{
					$sTmp = $sOut;
					$sOut .= $sTemp; // add the entity altogether
					if (strlen($sOut) > $iLetters)
					{
						return $sTmp;
					}
					$iOutLen++; // increment the length of the returning string
					$iPos = $iEnd-1; // move the pointer to skip the entity in the next run
					continue;
				}
			}
			$sOut .= $sTxt[$iPos];
			$iOutLen++;
		}
		return $sOut;
	}
	
	/**
	 * Parse and clean a title of an item and convert it into a URL title string.
	 * Example if you had:
	 * <code>
	 *  this is a TEST string!!!
	 * </code>
	 * It would convert it to:
	 * <code>
	 * this-is-a-test-string
	 * </code>
	 * Which, we would then use in a URL:
	 * <code>
	 * http://www.yoursite.com/this-is-a-test-string/
	 * </code>
	 *
	 * @param string $sUrls String to convert into a URL.
	 * @return string Converted URL.
	 */
	public function cleanTitle($sUrls)
	{
		$sUrls = trim(strip_tags($sUrls));
		$sUrls = $this->_utf8ToUnicode($sUrls, true);		
		$sUrls = preg_replace("/ +/", "-", $sUrls);		
		$sUrls = rawurlencode($sUrls);		
		$sUrls = str_replace(array('"', "'"), '', $sUrls);
		$sUrls = str_replace(' ', '-', $sUrls);
		$sUrls = str_replace(array('-----', '----', '---', '--'), '-', $sUrls);
		$sUrls = rtrim($sUrls, '-');
		$sUrls = ltrim($sUrls, '-');
		
		if (empty($sUrls))
		{
			$sUrls = PHPFOX_TIME;
		}
		
		$sUrls = strtolower($sUrls);

		return $sUrls;
	}
	
	/**
	 * Cleans a file name and removes any non-latin characters.
	 *
	 * @param string $sName Name of the file.
	 * @return string Clean file name.
	 */
	public function cleanFileName($sName)
	{
		$sName = preg_replace( '/ +/', '_',preg_replace('/[^0-9a-zA-Z]+/', '_', $sName));
		$sName = strtolower($sName);
		$sName = rtrim($sName, '_');		
		
		return $sName;	
	}

	/**
	 * Checks if a title of the item can be used in the sites root. This is mainly used
	 * to check a persons vanity URL name. Since if a user would use the name "friend"
	 * it would cause problems when trying to visit anything related to the "friend" module
	 * as it would instead load this persons profile. With this check it makes sure that the 
	 * name or title being used is not a module, a folder in the sites root directory, rewrite rule, page
	 * or the user name isn't already in use.
	 *
	 * @param string $sTitle Title to check.
	 * @param string $sErrorMessage Error message you want to return in case there is an error.
	 * @return bool TRUE if the title is allowed otherwise FALSE.
	 */
	public function allowTitle($sTitle, $sErrorMessage)
	{
		$bIsOk = true;		

		$hDir = opendir(PHPFOX_DIR_MODULE);
		while ($sFile = readdir($hDir))
		{
			if ($sFile == '.' || $sFile == '..')
			{
				continue;
			}

			if (strtolower($sFile) === strtolower($sTitle))
			{
				$bIsOk = false;
					
				break;
			}
		}
		closedir($hDir);
		
		$aRewrites = Phpfox::getLib('url')->getRewrite();		
		foreach ($aRewrites as $sUrl => $aRow)
		{		    
		    if (strtolower($sUrl) == strtolower($sTitle) || strtolower($aRow['module']) == strtolower($sTitle))
		    {
				$bIsOk = false;
				break;
		    }
		}

		if (Phpfox::getService('user')->isUser($sTitle))
		{
			$bIsOk = false;
		}	
		
		if (Phpfox::isModule('pages') && Phpfox::getService('pages')->isPage($sTitle))
		{
			$bIsOk = false;
		}			
		
		$hDir = opendir(PHPFOX_DIR);
		while ($sFile = readdir($hDir))
		{
			if ($sFile == '.' || $sFile == '..')
			{
				continue;
			}

			if (strtolower($sFile) === strtolower($sTitle))
			{
				$bIsOk = false;
					
				break;
			}
		}
		closedir($hDir);		

		return ($bIsOk ? true : Phpfox_Error::set($sErrorMessage));
	}

	/**
	 * Remove evil attributes from JavaScript.
	 *
	 * @param string $sTxt JavaScript code to parse.
	 * @return string Parsed JavaScript code.
	 */
	public function jsClean($sTxt)
	{
		return $this->_removeEvilAttributes($sTxt, true);
	}
	
	/**
	 * Reverse prepares strings based on what we converted with the method prepare().
	 *
	 * @see self::prepare()
	 * @param string $sTxt String to parse.
	 * @return string Parsed string.
	 */
	public function reversePrepare($sTxt)
	{
		$sTxt = html_entity_decode($sTxt);
		
		return $this->prepare($sTxt);
	}

	/**
	 * Prepare text strings. Used to prepare all data that can contain HTML. Not only does
	 * it protect against harmful HTML and CSS, it also has support for emoticon and BBCode conversion.
	 *
	 * @param string $sTxt Text to parse.
	 * @return string Parsed string.
	 */
	public function prepare($sTxt, $bNoClean = false)
	{
		// Parse Emoticons
		if (Phpfox::isModule('emoticon'))
		{		
			$sTxt = Phpfox::getService('emoticon')->parse($sTxt);
		}
		
		$sTxt = str_replace('\\', '&#92;', $sTxt);
		$sTxt = str_replace(array('&lt;', '&gt;'), array('<', '>'), $sTxt);
		$sTxt = str_replace('[*]', '<li>', $sTxt);
		
		$oFilterBbcode = Phpfox::getLib('parse.bbcode');		

		$sTxt = $oFilterBbcode->preParse($sTxt);		

		// Parse for language package
		$sTxt = $this->_utf8ToUnicode($sTxt);
		$sTxt = str_replace('\\', '&#92;', $sTxt);
		
		// Clean out the HTML
		if (!$bNoClean)
		{
			$sTxt = $this->_cleanHtml($sTxt);
		}

		// Parse BBCode
		$sTxt = $oFilterBbcode->parse($sTxt);		
		
		$sTxt = str_replace('<br /><li>', '<li>', $sTxt);
		$sTxt = str_replace('<br /></ul>', '</ul>', $sTxt);	
		$sTxt = str_replace('<br /><tr>', '<tr>', $sTxt);
		$sTxt = str_replace('<br /><td>', '<td>', $sTxt);
		$sTxt = str_replace('<br /></tr>', '</tr>', $sTxt);
		$sTxt = str_replace('<br /></table>', '</table>', $sTxt);		

		return $sTxt;
	}

	/**
	 * Converts a string that contains non-latin characters into UNICODE.
	 *
	 * @see self::_utf8ToUnicode()
	 * @param string $sTxt Text to convert to UNICODE.
	 * @return string Converted text.
	 */
	public function convert($sTxt)
	{
		return $this->_utf8ToUnicode($sTxt);
	}

	/**
	 * Fixes any odd HTML, mainly dealing with HTML output from TinyMCE (WYSIWYG Editor).
	 *
	 * @param string $sTxt Text to parse.
	 * @return string Parsed text.
	 */
	public function fixHtml($sTxt)
	{
		// Replacements done becaues of TinyMCE (WYSIWYG Editor)
		$sTxt = preg_replace("/\&lt;(.*?)\&gt;/si","<\\1>",$sTxt);
		$sTxt = str_replace("&quot;",'"',$sTxt);
		$sTxt = preg_replace("/ href=\"(.*?)\" mce_href=\"(.*?)\"/i", " href=\"\\2\"", $sTxt);
		$sTxt = preg_replace("/ src=\"(.*?)\" mce_src=\"(.*?)\"/i", " src=\"\\2\"", $sTxt);
		
		return $sTxt;
	}

	/**
	 * Add line breaks. Unlike the PHP version of this function it looks into not adding
	 * line breaks within HTML and PHP code.
	 *
	 * @param string $sTxt String we need to parse.
	 * @return string Parsed string with newly added breaks.
	 */
	public function addBreak($sTxt)
	{		
		$sTxt = str_replace("\n", "[pf_break]", $sTxt);		
		$sTxt = preg_replace_callback('/\{php\}(.*?)\{\/php\}/is', array(&$this, '_addBreak'), $sTxt);
		$sTxt = preg_replace_callback('/<(.*?)>/is', array(&$this, '_addBreak'), $sTxt);
		
		$sTxt = str_replace('[pf_break]', "<br class=\"pf_break\" />", $sTxt);		
		$sTxt = stripslashes($sTxt);

		return $sTxt;
	}	
	
	/**
	 * Get params within SMARTY {} tags.
	 *
	 * @param string $sTxt String we need to parse.
	 * @return array ARRAY matches are returned.
	 */
	public function getParams($sTxt)
	{
        $aResult = array();
		if (preg_match_all('/(?:' . $this->_sQstrRegexp . ' | (?>[^"\'=\s]+))+|[=]/x', $sTxt, $aMatches))
		{
			$iState = 0;
			foreach($aMatches[0] as $mValue)
			{				
				switch($iState)
				{
					case 0:
						if (is_string($mValue))
						{		
							$sName = $mValue;							
							$iState = 1;
						}
						else
						{
						//	Phpfox_Error::trigger("Invalid Attribute Name", E_USER_ERROR);
						}
						break;
					case 1:
						if ($mValue == '=')
						{
							$iState = 2;
						}
						else
						{
						//	 Phpfox_Error::trigger("Expecting '=' After '{$sLastValue}'", E_USER_ERROR);
						}
						break;
					case 2:
						if ($mValue != '=')
						{
							$iState = 0;
							$aResult[strtolower($sName)] = $mValue;
						}
						else
						{
						//	Phpfox_Error::trigger("'=' cannot be an attribute value", E_USER_ERROR);
						}
						break;
				}
				$sLastValue = $mValue;
			}		
		}
		
		return $aResult;
	}
	
	/**
	 * Removes evit attributes within HTML.
	 *
	 * @see self::_removeEvilAttributes()
	 * @param string $sTxt String to parse.
	 * @return string Parsed string.
	 */
	public function removeEvilAttributes($sTxt)
	{
		return $this->_removeEvilAttributes($sTxt);
	}
	
	/**
	 * Preparing a URL title. Will be used to replace a title "this is a TITLE" to
	 * "this-is-a-title".
	 * 
	 * Example:
	 * <code>
	 * Phpfox::getLib('parse.input')->prepareTitle('photo', $aVals['title'], 'name_url', Phpfox::getUserId(), Phpfox::getT('photo_album'));
	 * </code>
	 *
	 * @param string $sModule Module ID.
	 * @param string $sTitle Title to parse and fix.
	 * @param string $sField Database field to check if such titles already exist.
	 * @param int $iUserId User ID to check 
	 * @param string $sTable Name of the database table.
	 * @param mixed $mCondition Database WHERE condition.
	 * @param boolean $bCleanOnly Return true if you want to return the clean title without running the existing title check.
	 * @param array $bCache FALSE will force a new check, while default TRUE will cache previous checks.
	 * @return string New fixed title.
	 */
	public function prepareTitle($sModule, $sTitle, $sField, $iUserId = null, $sTable, $mCondition = null, $bCleanOnly = false, $bCache = true)
	{
		static $aTitle = array();
		static $iCacheCount = 0;
		
		if (defined('PHPFOX_INSTALLER'))
		{
			$bCache = false;
		}
		
		if ($bCache && isset($aTitle[$sTitle]))
		{
			return $aTitle[$sTitle];
		}		

		$sNewTitle = $this->cleanTitle($sTitle);

		if ($bCleanOnly)
		{
			return $sNewTitle;
		}

		if (!defined('PHPFOX_INSTALLER'))
		{
		    $sNewTitle = substr($sNewTitle, 0, Phpfox::getParam('core.crop_seo_url'));
		}

		$oDb = Phpfox::getLib('database');
		$aOlds = $oDb->select($sField . ' AS title_url')
			->from($sTable)
			->where(($mCondition === null ? $sField . ' LIKE \'%' . $oDb->escape($sNewTitle) . '%\'' : $mCondition))
			->execute('getRows');	
					
		$iTotal = 0;
		$aNumbers = array();
		$aIntNumbers = array();
		foreach ($aOlds as $aOld)
		{			
			if (preg_match("/(.*)-([0-9]+)/i", $aOld['title_url'], $aMatches))
			{
				$aIntNumbers[] = $aMatches[2];
			}
			
			if ($aOld['title_url'] === $sNewTitle)
			{
				$aNumbers[] = $sNewTitle;
			}
		}
				
		// Is this a valid module?
		if (Phpfox::isModule($sModule))
		{
			// Open the modules controller directory
			$hDir = opendir(PHPFOX_DIR_MODULE . $sModule . PHPFOX_DS . PHPFOX_DIR_MODULE_COMPONENT . PHPFOX_DS . 'controller' . PHPFOX_DS);
			while ($sFile = readdir($hDir))
			{
				if ($sFile == '.' || $sFile == '..' || $sFile == '.svn')
				{
					continue;
				}
				
				// Put the directory name within an array
				if (is_dir(PHPFOX_DIR_MODULE . $sModule . PHPFOX_DS . PHPFOX_DIR_MODULE_COMPONENT . PHPFOX_DS . 'controller' . PHPFOX_DS . $sFile) && $sNewTitle === $sFile)
				{
					$aNumbers[] = $sFile;
				}
				
				// Put the file name within an array
				if (preg_match("/(.*?)\.class\.php/i", $sFile, $aMatches) && $sNewTitle === $aMatches[1])
				{
					$aNumbers[] = $aMatches[1];					
				}
			}
		}
		
		if (count($aIntNumbers))
		{
			$iTotal = max($aIntNumbers) + 1;
		}
		else
		{
			if (count($aNumbers))
			{
				arsort($aNumbers);
				$iTotal = (count($aNumbers) + 1);
			}
		}
		if (!$bCache && isset($aTitle[$sTitle]))
		{	
			$iCacheCount++;

			return $aTitle[$sTitle] . '-' . $iCacheCount;	
		}		
	
		// Do we have any titles that match and if we do add a new count after it.
		$aTitle[$sTitle] =  $sNewTitle . ($iTotal > 0 ? '-' . $iTotal : '');		
		
		return $aTitle[$sTitle];
	}	
	
	/**
	 * Removes HTML found within HTML.
	 *
	 * @see self::_stripInnerHtml()
	 * @param string $sText Text to parse.
	 * @return string Returns parsed text.
	 */
	public function stripInnerHtml($sText)
	{
		$this->_sText = $sText;
		
		preg_replace('/<(.*?)>/ise', "'<'.\$this->_stripInnerHtml('\\1').'>'", $sText);
		
		$this->_sText = strip_tags($this->_sText);
		
		return $this->_sText;
	}
	
	/**
	 * Removes HTML found within HTML.
	 *
	 * @see self::stripInnerHtml()
	 * @param string $sText Text to parse.
	 * @return string Returns parsed text.
	 */	
	private function _stripInnerHtml($sTag)
	{	
		if (substr($sTag, 0, 1) != '/')
		{
			$aPart = explode(' ', $sTag);
			$sTag = $aPart[0];
		}
		$iLength = strlen($sTag);	
		
		$sLowerText = strtolower($this->_sText);
		$aStartPos = array();
		$iCurPos = 0;
		do
		{
			$sPos = strpos($sLowerText, '<' . $sTag . '', $iCurPos);		
			if ($sPos !== false)
			{
				$aStartPos[$sPos] = 'start';
			}
	
			$iCurPos = ($sPos + $iLength + 1);		
		}
		while ($sPos !== false);
	
		if (sizeof($aStartPos) == 0)
		{
			return false;
		}
	
		$aEndPos = array();
		$iCurPos = 0;
		do
		{
			$sPos = strpos($sLowerText, '</' . $sTag . '>', $iCurPos);
			if ($sPos !== false)
			{
				$aEndPos[$sPos] = 'end';
				$iCurPos = ($sPos + $iLength + 3);
			}
		}
		while ($sPos !== false);
	
		if (sizeof($aEndPos) == 0)
		{
			return false;
		}
	
		$aPosList = $aStartPos + $aEndPos;
		ksort($aPosList);
	
		do
		{
			$aStack = array();
			$sNewText = '';
			$iSubstrPos = 0;
			foreach ($aPosList AS $sPos => $sType)
			{
				$aStacksize = sizeof($aStack);
				if ($sType == 'start')
				{
					if ($aStacksize == 0)
					{
						$sNewText .= substr($this->_sText, $iSubstrPos, $sPos - $iSubstrPos);
					}
					array_push($aStack, $sPos);
				}
				else
				{
					if ($aStacksize)
					{
						array_pop($aStack);
						$iSubstrPos = ($sPos + $iLength + 3);
					}
				}
			}
	
			$sNewText .= substr($this->_sText, $iSubstrPos);
	
			if ($aStack)
			{
				foreach ($aStack AS $sPos)
				{
					unset($aPosList[$sPos]);
				}
			}
		}
		while ($aStack);
	
		$this->_sText = $sNewText;
	}	
	
	/**
	 * Common routine we run when using the prepare() method to clean HTML to make sure it is safe.
	 *
	 * @param string $sTxt Text to parse.
	 * @return string Parsed text.
	 */
	private function _cleanHtml($sTxt)
	{		
		$this->fixHtml($sTxt);		
		
		if (!defined('PHPFOX_DO_NOT_PARSE_BREAK'))
		{
			$sTxt = str_replace("\n", '[br]', $sTxt);	
		}
		
		preg_match_all('/<\/?\w+((\s+\w+(\s*=\s*(?:".*?"|\'.*?\'|[^\'">\s]+))?)+\s*|\s*)\/?>/i', $sTxt, $aMatched);
		
		$iUnid1 = ' ' .uniqid();
		$iUnid2 = uniqid() . ' ';
		$aAllowed = Phpfox::getParam('core.allowed_html');
		$sAllowed = '';
		foreach ($aAllowed as $sVal => $iNum)
		{
			$sAllowed .= '<' . $sVal.'> </'.$sVal .'> <'.$sVal .'/> <'.$sVal .' /> ';
		}
		
		foreach ($aMatched[0] as $iKey => $sMatch)
		{
			$sTemp = str_replace($aMatched[1][$iKey], '', $sMatch);
			$sTemp = strtolower($sTemp);

			if (strpos($sAllowed, $sTemp) !== false)
			{				
				$sMatch2 = str_replace($aMatched[1][$iKey], $this->_removeEvilAttributes($aMatched[1][$iKey], true), $sMatch);
				$sNew = str_replace(array('<', '>'),array($iUnid1, $iUnid2), $sMatch2);
				$sTxt = str_replace($sMatch, $sNew, $sTxt);
				continue;
			}
			
		}
		
		$sTxt = str_replace(array('<', '>'), array('&lt;', '&gt;'), $sTxt);
		$sTxt = str_replace(array($iUnid1, $iUnid2), array('<', '>'), $sTxt);		
		
		if (!defined('PHPFOX_DO_NOT_PARSE_BREAK'))
		{
			$sTxt = str_replace('[br]', '<br />', $sTxt);
		}
		/* Allow iframe from youtube*/
		if ( (Phpfox::isModule('video') && Phpfox::getParam('video.allow_youtube_iframe')) || defined('PHPFOX_FORCE_IFRAME'))
		{
			/* get all the iframes in text*/			
			if (preg_match_all('/(&lt;iframe [a-z0-9=_\-\.:&\/\?; \"]*&gt;&lt;\/iframe&gt;)/i', $sTxt, $aIframes))
			{
				/* for each iframe get the params */
				foreach ($aIframes as $iKey => $aSubFrames)
				{
					foreach ($aSubFrames as $aFrame)
					{					
						$sReplace = '<iframe ';					
						// $aFrame = reset($aFrame);
						if (preg_match_all('/(title|width|height|src|frameborder)="([^"]*)/i',$aFrame,$aMatch,PREG_SET_ORDER))
						{
							foreach ($aMatch as $aSub)
							{
								switch($aSub[1])
								{
									case 'width':
									case 'height':
									case 'frameborder':
										$sReplace .= $aSub[1] .'="' . ((int)$aSub[2]) .'" ';
										break;
									case 'title':
										$sReplace .= 'title="' . $this->clean($aSub[2]).'" ';
										break;
									case 'src':
										if (defined('PHPFOX_FORCE_IFRAME'))
										{
											$sReplace .= 'src="' . $aSub[2] . '" ';
										}
										else
										{
											$sLink = preg_match('/http:\/\/www\.youtube.com\/embed\/([a-z0-9\-_]*)/i',$aSub[2],$aVal);
											if (isset($aVal[1]))
											{
												$sReplace .= 'src="http://www.youtube.com/embed/' . $this->clean($aVal[1]) .'?wmode=transparent" wmode="Opaque"';
											}
											else
											{
												$sReplace = '';
											}
										}
								}
							}
							if (strpos($aFrame,'allowfullscreen'))
							{
								$sReplace .='allowfullscreen ';
							}
						}
						$sReplace = rtrim($sReplace);
						$sReplace .='></iframe>';
						$sTxt = str_replace($aFrame,$sReplace,$sTxt);
					}
				}				
			}
		}
		return $sTxt;
		
	}	
	
	/**
	 * Our method of PHP strip_tags().
	 *
	 * @deprecated 2.0.0rc1
	 * @see strip_tags()
	 * @param string $sTxt String to parse.
	 * @return string Parsed string.
	 */
	private function _stripTags($sTxt)
	{
		foreach ($this->_aAllowedTags as $sAllowTag)
		{
			$sAllowTag = preg_replace("/^<(.*?)>$/i", "\\1", $sAllowTag);
			$sAllowTag = str_replace(' ', '', $sAllowTag);
			if (substr($sAllowTag, -1) == '/')
			{
				$sTxt = preg_replace("/&lt;" . substr_replace($sAllowTag, '', -1) . "(.*)&gt;/ise", "''.\$this->_cacheHtml('\\1', null, '$sAllowTag', true).''", $sTxt);
			}
			else 
			{
				$sAllowTag = str_replace('/', '\/', $sAllowTag);				
				$sTxt = preg_replace("/&lt;{$sAllowTag}(.*?)&gt;(.*)&lt;\/{$sAllowTag}&gt;/ise", "''.\$this->_cacheHtml('\\1', '\\2', '$sAllowTag').''", $sTxt);
			}
		}
		
		return $sTxt;	
	}
	
	/**
	 * Part of the method _stripTags() to remove unwanted HTML tags.
	 *
	 * @deprecated 2.0.0rc1
	 * @see self::_stripTags()
	 * @param string $sAttr HTML attributes.
	 * @param string $sTxt HTML inner text.
	 * @param string $sTag HTML tag.
	 * @param bool $bClose TRUE will close the tag, FALSE will leave it open.
	 * @return string Returns fixed HTML tags with clean attributes and inner content.
	 */
	private function _cacheHtml($sAttr, $sTxt = null, $sTag, $bClose = false)
	{		
		if ($bClose)
		{
			return '<' . substr_replace($sTag, '', -1) . ' ' . stripslashes($sAttr) . '>';
		}
		else 
		{
			return '<' . $sTag . stripslashes($sAttr) . '>' . $this->_stripTags(stripslashes($sTxt)) . '</' . $sTag . '>';		
		}
	}	
	
	/**
	 * Removes all evil attributes in a string to make sure the data returned is safe to output on the site.
	 *
	 * @param string $sTxt Text we need to parse.
	 * @param bool $bCleanOnly TRUE to just clean the text, FALSE to clean and make sure HTML tags are valid.
	 * @return string Parsed string with all evil attributes removed.
	 */
    private function _removeEvilAttributes($sTxt, $bCleanOnly = false)
    {
    	$sTxt = stripslashes($sTxt);
    	$sTxt = str_replace("xhref","href",$sTxt);
    	$sTxt = str_replace("xsrc","src",$sTxt);
        $sTxt = strip_tags($sTxt);
        $sTxt = str_replace('[br]', '', $sTxt);
        
        $aParts = explode(' ', $sTxt);
        
        foreach ($this->_aEvilEvents as $sRemove)
        {
            $sTxt = preg_replace('#'.$sRemove.'=#i', 'title=', $sTxt);
        }

        $sTxt = preg_replace('#javascript:#i', '', $sTxt);
		
        foreach($this->_aEvilCss as $sCss)
        {
			$sTxt = preg_replace('/'. $sCss .':(.*?)/i', 'replaced:', $sTxt);
       	}      
		
		$aParams = $this->getParams(substr_replace($sTxt, '', 0, strlen($aParts[0]))); 
		foreach ($aParams as $sKey => $mValue)
        {
			if ($sKey == 'src')
			{
				if (!preg_match('/(http|https):\/\//is', $mValue))
				{
					$sTxt = preg_replace('/'. $sKey .'=' . preg_quote($mValue, '/')  . '/is', 'replaced=""', $sTxt);
				}
			}			
		}
       	
       	if ($bCleanOnly === true)
       	{
       		return $sTxt;
       	}       	

       	$bFailed = false;       	
        switch (strtolower($aParts[0]))
        {
        	case 'object':
        	case 'embed':
        	case 'param':
        	case 'iframe':
  
        		break;   
        	case 'font':
        		$aParams = $this->getParams(substr_replace($sTxt, '', 0, strlen($aParts[0]))); 		
        		$sTxt = 'font';
        		foreach ($aParams as $sKey => $mValue)
        		{ 			
        			if (!in_array($sKey, array(
        						'color',
        						'size',
        						'face'
        					)
        				)
        			)
        			{
        				continue;
        			}
        			$sTxt .= ' ' . $sKey . '=' . $mValue . '';
        		}
        		break;
        	case 'img':		
				$aParams = $this->getParams(substr_replace($sTxt, '', 0, strlen($aParts[0])));	
				$sTempSource = '';
				if (isset($aParams['src']))
				{
					$sTempSource = str_replace(array('"', "'"), '', $aParams['src']);			
				}
        		if (!empty($sTempSource) && preg_match('/http:\/\//i', $sTempSource))
        		{
        			$sTxt = 'img src=' . $aParams['src'];    
        			if (isset($aParams['alt']))
        			{ 
        				$sTxt .= ' alt=' . $aParams['alt'];
        			}
        			$sTxt .= ' class="parsed_image" /';        			
        		}
        		else 
        		{
        			$bFailed = true;
        		}
        		break;
        	case 'a':        		
        		$aParams = $this->getParams(substr_replace($sTxt, '', 0, strlen($aParts[0])));         	
        		if (isset($aParams['href']))
        		{
        			$sTxt = 'a href=' . $aParams['href'];
        		}
        		else 
        		{
        			$bFailed = true;
        		}
        		break;
        	default:        	

        		break;
        }
        
        $aHtmlTags = Phpfox::getParam('core.allowed_html');  
        
        (($sPlugin = Phpfox_Plugin::get('parse_input__removeevilattributes')) ? eval($sPlugin) : null);     
		
		if ((isset($aHtmlTags[strtolower($aParts[0])]) || isset($aHtmlTags[substr_replace(strtolower($aParts[0]), '', 0, 1)])) && $bFailed === false)
		{			
			return '<' . $sTxt . '>';
		}
		else 
		{			
			return '&lt;' . $sTxt . '&gt;';
		}        
    }

    /**
     * Converts a string with non-latin characters into UNICODE. We convert all strings
     * before we enter them into the database so clients do not have to worry about database
     * collations and website encoding as all common browsers have no problems displaying UNICODE.
     *
     * @see self::_unicodeToEntitiesPreservingAscii()
     * @param string $str String we need to parse.
     * @param bool $bForUrl TRUE for URL strings, FALSE for general usage.
     * @return string Returns string that has been converted.
     */
    private function _utf8ToUnicode($str, $bForUrl = false)
    {
        $unicode = array();
        $values = array();
        $lookingFor = 1;

        for ($i = 0; $i < strlen( $str ); $i++ )
        {
            $thisValue = ord( $str[ $i ] );

            if ( $thisValue < 128 )
            {
            	$unicode[] = $thisValue;
            }
            else
            {
                if ( count( $values ) == 0 ) $lookingFor = ( $thisValue < 224 ) ? 2 : 3;

                $values[] = $thisValue;

                if ( count( $values ) == $lookingFor ) 
                {
                    $number = ( $lookingFor == 3 ) ?
                        ( ( $values[0] % 16 ) * 4096 ) + ( ( $values[1] % 64 ) * 64 ) + ( $values[2] % 64 ):
                    	( ( $values[0] % 32 ) * 64 ) + ( $values[1] % 64 );

                    $unicode[] = $number;
                    $values = array();
                    $lookingFor = 1;
                }
            }
        }

        return $this->_unicodeToEntitiesPreservingAscii($unicode, $bForUrl);
    }

    /**
     * Converts a string with non-latin characters into UNICODE. This method is used with the method _utf8ToUnicode().
     *
     * @see self::_utf8ToUnicode()
     * @param array $unicode ARRAY of unicode values.
     * @param bool $bForUrl TRUE for URL strings, FALSE for general usage.
     * @return string Returns string that has been converted.
     */
    private function _unicodeToEntitiesPreservingAscii($unicode, $bForUrl = false)
    {
        $entities = '';
        foreach( $unicode as $value )
        {
        	if ($bForUrl === true)
        	{
        		if ($value == 42 || $value > 127)
        		{
        			$sCacheValue = Phpfox::getLib('locale')->parse('&#' . $value . ';', false);       
        		
        			$entities .= (preg_match('/[^a-zA-Z]+/', $sCacheValue) ? '-' . $value : $sCacheValue);   			
        		}
        		else 
        		{
        			$entities .= (preg_match('/[^0-9a-zA-Z]+/', chr($value)) ? ' ' : chr($value));
        		}        		
        	}
        	else 
        	{
        		$entities .= ($value == 42 ? '&#' . $value . ';' : ( $value > 127 ) ? '&#' . $value . ';' : chr($value));
        	}
        }
		$entities = str_replace("'", '&#039;', $entities);
        return $entities;
    }

	/**
	 * Cleans HTML object to make it XHTML valid
	 *
	 * @param string $sObject is the object data
	 * @param string $sEmbed is all the <embed> tags found withing a <object>
	 * @return string New <object>
	 */
	private function _cleanObject($sObject, $sEmbed)
	{
		$sObject = stripslashes($sObject);
		$sObject = str_replace("'", "\'", $sObject);

		$sEmbed = stripslashes($sEmbed);
		$sEmbed = str_replace("/", '\/', $sEmbed);
		$sEmbed = str_replace("'", "\'", $sEmbed);

		$sTxt = '<script type="text/javascript">document.write(\'<object' . $sObject . '>' . $sEmbed . '<\/object>\');</script>';

		return $sTxt;
	}
	
	/**
	 * Add a line break.
	 *
	 * @see self::addBreak()
	 * @param array $aMatches Matches from regex found in public method addBreak().
	 * @return string Return the string match of the text with removing the BBCode break we placed in earlier.
	 */
	private function _addBreak($aMatches)
	{
		$aMatches[0] = str_replace('[pf_break]', '', $aMatches[0]);
		
		return $aMatches[0];
	}	
}

?>