Find this useful? Enter your email to receive occasional updates for securing PHP code.

Signing you up...

Thank you for signing up!

PHP Decode

<?php /** * Browser detection class file. * This file contains everything required ..

Decoded Output download

<?php 
 
/** 
 * Browser detection class file. 
 * This file contains everything required to use the BrowserDetection class. 
 * 
 * This program is free software; you can redistribute it and/or modify it under the terms of the GNU Lesser General 
 * Public License as published by the Free Software Foundation; either version 3 of the License, or (at your option) any 
 * later version (if any). 
 * 
 * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied 
 * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more 
 * details at: http://www.gnu.org/licenses/lgpl.html 
 * 
 * @package Browser_Detection 
 * @version 2.3.0 
 * @last-modified February 11, 2016 
 * @author Alexandre Valiquette 
 * @copyright Copyright (c) 2016, Wolfcast 
 * @link http://wolfcast.com/ 
 */ 
 
 
/** 
 * The BrowserDetection class facilitates the identification of the user's environment such as Web browser, version, 
 * platform or if it's a mobile device. 
 * 
 * Typical usage: 
 * 
 * $browser = new BrowserDetection(); 
 * if ($browser->getName() == BrowserDetection::BROWSER_FIREFOX && 
 *     $browser->compareVersions($browser->getVersion(), '5.0') >= 0) { 
 *     echo 'You are using FireFox version 5 or greater.'; 
 * } 
 * 
 * The class is an updated version of Chris Schuld's Browser class version 1.9 which is unmaintained since August 20th, 
 * 2010. Chris' class was based on the original work from Gary White. 
 * 
 * Updates: 
 * 
 * 2016-02-11: Version 2.3.0 
 *  + WARNING! Breaking change: public method getBrowser() is renamed to getName(). 
 *  + WARNING! Breaking change: changed the compareVersions() return values to be more in line with other libraries. 
 *  + You can now get the exact platform version (name or version numbers) on which the browser is run on with 
 *    getPlatformVersion(). Only working with Windows operating systems at the moment. 
 *  + You can now determine if the browser is executed from a 64-bit platform with is64bitPlatform(). 
 *  + Better detection of mobile platform for Googlebot. 
 * 
 * 2016-01-04: Version 2.2.0 
 *  + Added support for Microsoft Edge. 
 * 
 * 2014-12-30: Version 2.1.2 
 *  + Better detection of Opera. 
 * 
 * 2014-07-11: Version 2.1.1 
 *  + Better detection of mobile devices and platforms. 
 * 
 * 2014-06-04: Version 2.1 
 *  + Added support for IE 11+. 
 * 
 * 2013-05-27: Version 2.0 which is (almost) a complete rewrite based on Chris Schuld's Browser class version 1.9 plus 
 * changes below. 
 *  + Added support for Opera Mobile 
 *  + Added support for the Windows Phone (formerly Windows Mobile) platform 
 *  + Added support for BlackBerry Tablet OS and BlackBerry 10 
 *  + Added support for the Symbian platform 
 *  + Added support for Bingbot 
 *  + Added support for the Yahoo! Multimedia crawler 
 *  + Removed iPhone/iPad/iPod browsers since there are not browsers but platforms - test them with getPlatform() 
 *  + Removed support for Shiretoko (Firefox 3.5 alpha/beta) and MSN Browser 
 *  + Merged Nokia and Nokia S60 
 *  + Updated some deprecated browser names 
 *  + Many public methods are now protected 
 *  + Documentation updated 
 * 
 * 2010-07-04: 
 *  + Added detection of IE compatibility view - test with getIECompatibilityView() 
 *  + Added support for all (deprecated) Netscape versions 
 *  + Added support for Safari < 3.0 
 *  + Better Firefox version parsing 
 *  + Better Opera version parsing 
 *  + Better Mozilla detection 
 * 
 * @package Browser_Detection 
 * @version 2.3.0 
 * @last-modified February 11, 2016 
 * @author Alexandre Valiquette, Chris Schuld, Gary White 
 * @copyright Copyright (c) 2016, Wolfcast 
 * @license http://www.gnu.org/licenses/lgpl.html 
 * @link http://wolfcast.com/ 
 * @link http://wolfcast.com/open-source/browser-detection/tutorial.php 
 * @link http://chrisschuld.com/ 
 * @link http://www.apptools.com/phptools/browser/ 
 */ 
class BrowserDetection 
{ 
 
    /**#@+ 
     * Constant for the name of the Web browser. 
     */ 
    const BROWSER_AMAYA = 'Amaya'; 
    const BROWSER_ANDROID = 'Android'; 
    const BROWSER_BINGBOT = 'Bingbot'; 
    const BROWSER_BLACKBERRY = 'BlackBerry'; 
    const BROWSER_CHROME = 'Chrome'; 
    const BROWSER_EDGE = 'Edge'; 
    const BROWSER_FIREBIRD = 'Firebird'; 
    const BROWSER_FIREFOX = 'Firefox'; 
    const BROWSER_GALEON = 'Galeon'; 
    const BROWSER_GOOGLEBOT = 'Googlebot'; 
    const BROWSER_ICAB = 'iCab'; 
    const BROWSER_ICECAT = 'GNU IceCat'; 
    const BROWSER_ICEWEASEL = 'GNU IceWeasel'; 
    const BROWSER_IE = 'Internet Explorer'; 
    const BROWSER_IE_MOBILE = 'Internet Explorer Mobile'; 
    const BROWSER_KONQUEROR = 'Konqueror'; 
    const BROWSER_LYNX = 'Lynx'; 
    const BROWSER_MOZILLA = 'Mozilla'; 
    const BROWSER_MSNBOT = 'MSNBot'; 
    const BROWSER_MSNTV = 'MSN TV'; 
    const BROWSER_NETPOSITIVE = 'NetPositive'; 
    const BROWSER_NETSCAPE = 'Netscape'; 
    const BROWSER_NOKIA = 'Nokia Browser'; 
    const BROWSER_OMNIWEB = 'OmniWeb'; 
    const BROWSER_OPERA = 'Opera'; 
    const BROWSER_OPERA_MINI = 'Opera Mini'; 
    const BROWSER_OPERA_MOBILE = 'Opera Mobile'; 
    const BROWSER_PHOENIX = 'Phoenix'; 
    const BROWSER_SAFARI = 'Safari'; 
    const BROWSER_SLURP = 'Yahoo! Slurp'; 
    const BROWSER_TABLET_OS = 'BlackBerry Tablet OS'; 
    const BROWSER_UNKNOWN = 'unknown'; 
    const BROWSER_W3CVALIDATOR = 'W3C Validator'; 
    const BROWSER_YAHOO_MM = 'Yahoo! Multimedia'; 
    /**#@-*/ 
 
    /**#@+ 
     * Constant for the name of the platform of the Web browser. 
     */ 
    const PLATFORM_ANDROID = 'Android'; 
    const PLATFORM_BEOS = 'BeOS'; 
    const PLATFORM_BLACKBERRY = 'BlackBerry'; 
    const PLATFORM_FREEBSD = 'FreeBSD'; 
    const PLATFORM_IPAD = 'iPad'; 
    const PLATFORM_IPHONE = 'iPhone'; 
    const PLATFORM_IPOD = 'iPod'; 
    const PLATFORM_LINUX = 'Linux'; 
    const PLATFORM_MACINTOSH = 'Macintosh'; 
    const PLATFORM_NETBSD = 'NetBSD'; 
    const PLATFORM_NOKIA = 'Nokia'; 
    const PLATFORM_OPENBSD = 'OpenBSD'; 
    const PLATFORM_OPENSOLARIS = 'OpenSolaris'; 
    const PLATFORM_OS2 = 'OS/2'; 
    const PLATFORM_SUNOS = 'SunOS'; 
    const PLATFORM_SYMBIAN = 'Symbian'; 
    const PLATFORM_UNKNOWN = 'unknown'; 
    const PLATFORM_VERSION_UNKNOWN = 'unknown'; 
    const PLATFORM_WINDOWS = 'Windows'; 
    const PLATFORM_WINDOWS_CE = 'Windows CE'; 
    const PLATFORM_WINDOWS_PHONE = 'Windows Phone'; 
    /**#@-*/ 
 
    /** 
     * Version unknown constant. 
     */ 
    const VERSION_UNKNOWN = 'unknown'; 
 
 
    /** 
     * @var string 
     * @access private 
     */ 
    private $_agent = ''; 
 
    /** 
     * @var string 
     * @access private 
     */ 
    private $_aolVersion = ''; 
 
    /** 
     * @var string 
     * @access private 
     */ 
    private $_browserName = ''; 
 
    /** 
     * @var string 
     * @access private 
     */ 
    private $_compatibilityViewName = ''; 
 
    /** 
     * @var string 
     * @access private 
     */ 
    private $_compatibilityViewVer = ''; 
 
    /** 
     * @var boolean 
     * @access private 
     */ 
    private $_is64bit = false; 
 
    /** 
     * @var boolean 
     * @access private 
     */ 
    private $_isAol = false; 
 
    /** 
     * @var boolean 
     * @access private 
     */ 
    private $_isMobile = false; 
 
    /** 
     * @var boolean 
     * @access private 
     */ 
    private $_isRobot = false; 
 
    /** 
     * @var string 
     * @access private 
     */ 
    private $_platform = ''; 
 
    /** 
     * @var string 
     * @access private 
     */ 
    private $_platformVersion = ''; 
 
    /** 
     * @var string 
     * @access private 
     */ 
    private $_version = ''; 
 
 
    //--- MAGIC METHODS ------------------------------------------------------------------------------------------------ 
 
 
    /** 
     * BrowserDetection class constructor. 
     * @param string $useragent The user agent to work with. Leave empty for the current user agent (contained in 
     * $_SERVER['HTTP_USER_AGENT']). 
     */ 
    public function __construct($useragent = '') 
    { 
        $this->setUserAgent($useragent); 
    } 
 
    /** 
     * Determine how the class will react when it is treated like a string. 
     * @return string Returns an HTML formatted string with a summary of the browser informations. 
     */ 
    public function __toString() 
    { 
        $result = ''; 
 
        $values = array(); 
        $values[] = array('label' => 'User agent', 'value' => $this->getUserAgent()); 
        $values[] = array('label' => 'Browser name', 'value' => $this->getName()); 
        $values[] = array('label' => 'Browser version', 'value' => $this->getVersion()); 
        $values[] = array('label' => 'Platform family', 'value' => $this->getPlatform()); 
        $values[] = array('label' => 'Platform version', 'value' => $this->getPlatformVersion(true)); 
        $values[] = array('label' => 'Platform version name', 'value' => $this->getPlatformVersion()); 
        $values[] = array('label' => 'Platform is 64-bit', 'value' => $this->is64bitPlatform() ? 'true' : 'false'); 
        $values[] = array('label' => 'Is mobile', 'value' => $this->isMobile() ? 'true' : 'false'); 
        $values[] = array('label' => 'Is robot', 'value' => $this->isRobot() ? 'true' : 'false'); 
        $values[] = array('label' => 'IE is in compatibility view', 'value' => $this->isInIECompatibilityView() ? 'true' : 'false'); 
        $values[] = array('label' => 'Emulated IE version', 'value' => $this->isInIECompatibilityView() ? $this->getIECompatibilityView() : 'Not applicable'); 
        $values[] = array('label' => 'Is Chrome Frame', 'value' => $this->isChromeFrame() ? 'true' : 'false'); 
        $values[] = array('label' => 'Is AOL optimized', 'value' => $this->isAol() ? 'true' : 'false'); 
        $values[] = array('label' => 'AOL version', 'value' => $this->isAol() ? $this->getAolVersion() : 'Not applicable'); 
 
        foreach ($values as $currVal) { 
            $result .= '<strong>' . htmlspecialchars($currVal['label'], ENT_NOQUOTES) . ':</strong> ' . $currVal['value'] . '<br />' . PHP_EOL; 
        } 
 
        return $result; 
    } 
 
 
    //--- PUBLIC MEMBERS ----------------------------------------------------------------------------------------------- 
 
 
    /** 
     * Compare two version number strings. 
     * @param string $sourceVer The source version number. 
     * @param string $compareVer The version number to compare with the source version number. 
     * @return int Returns -1 if $sourceVer < $compareVer, 0 if $sourceVer == $compareVer or 1 if $sourceVer > 
     * $compareVer. 
     */ 
    public function compareVersions($sourceVer, $compareVer) 
    { 
        $sourceVer = explode('.', $sourceVer); 
        foreach ($sourceVer as $k => $v) { 
            $sourceVer[$k] = $this->parseInt($v); 
        } 
 
        $compareVer = explode('.', $compareVer); 
        foreach ($compareVer as $k => $v) { 
            $compareVer[$k] = $this->parseInt($v); 
        } 
 
        if (count($sourceVer) != count($compareVer)) { 
            if (count($sourceVer) > count($compareVer)) { 
                for ($i = count($compareVer); $i < count($sourceVer); $i++) { 
                    $compareVer[$i] = 0; 
                } 
            } else { 
                for ($i = count($sourceVer); $i < count($compareVer); $i++) { 
                    $sourceVer[$i] = 0; 
                } 
            } 
        } 
 
        foreach ($sourceVer as $i => $srcVerPart) { 
            if ($srcVerPart > $compareVer[$i]) { 
                return 1; 
            } else { 
                if ($srcVerPart < $compareVer[$i]) { 
                    return -1; 
                } 
            } 
        } 
 
        return 0; 
    } 
 
    /** 
     * Get the version of AOL (if any). AOL releases "optimized" Internet Explorer and Firefox versions. In the making 
     * they add their version number in the user agent string of these browsers. 
     * @return string Returns the version of AOL or an empty string if no AOL version was found. 
     */ 
    public function getAolVersion() 
    { 
        return $this->_aolVersion; 
    } 
 
    /** 
     * Get the name of the browser. All of the return values are class constants. You can compare them like this: 
     * $myBrowserInstance->getName() == BrowserDetection::BROWSER_FIREFOX. 
     * @return string Returns the name of the browser. 
     */ 
    public function getName() 
    { 
        return $this->_browserName; 
    } 
 
    /** 
     * Get the name and version of the browser emulated in the compatibility view mode (if any). Since Internet 
     * Explorer 8, IE can be put in compatibility mode to make websites that were created for older browsers, especially 
     * IE 6 and 7, look better in IE 8+ which renders web pages closer to the standards and thus differently from those 
     * older versions of IE. 
     * @param boolean $asArray Determines if the return value must be an array (true) or a string (false). 
     * @return mixed If a string was requested, the function returns the name and version of the browser emulated in the 
     * compatibility view mode or an empty string if the browser is not in compatibility view mode. If an array was 
     * requested, an array with the keys 'browser' and 'version' is returned. 
     */ 
    public function getIECompatibilityView($asArray = false) 
    { 
        if ($asArray) { 
            return array('browser' => $this->_compatibilityViewName, 'version' => $this->_compatibilityViewVer); 
        } else { 
            return trim($this->_compatibilityViewName . ' ' . $this->_compatibilityViewVer); 
        } 
    } 
 
    /** 
     * Get the name of the platform family on which the browser is run on (such as Windows, Apple, iPhone, etc.). All of 
     * the return values are class constants. You can compare them like this: 
     * $myBrowserInstance->getPlatform() == BrowserDetection::PLATFORM_ANDROID. 
     * @return string Returns the name of the platform or BrowserDetection::PLATFORM_UNKNOWN if unknown. 
     */ 
    public function getPlatform() 
    { 
        return $this->_platform; 
    } 
 
    /** 
     * Get the platform version on which the browser is run on. It can be returned as a string number like 'NT 6.3' or 
     * as a name like 'Windows 8.1'. When returning version string numbers for Windows NT OS families the number is 
     * prefixed by 'NT ' to differentiate from older Windows 3.x & 9x release. At the moment only the Windows operating 
     * systems is supported. 
     * @param boolean $returnVersionNumbers Determines if the return value must be versions numbers as a string (true) 
     * or the version name (false). 
     * @param boolean $returnServerFlavor Since some Windows NT versions have the same values, this flag determines if 
     * the Server flavor is returned or not. For instance Windows 8.1 and Windows Server 2012 R2 both use version 6.3. 
     * @return string Returns the version name/version numbers of the platform or the constant PLATFORM_VERSION_UNKNOWN 
     * if unknown. 
     */ 
    public function getPlatformVersion($returnVersionNumbers = false, $returnServerFlavor = false) 
    { 
        if ($this->_platformVersion == self::PLATFORM_VERSION_UNKNOWN || $this->_platformVersion == '') { 
            return self::PLATFORM_VERSION_UNKNOWN; 
        } 
 
        if ($returnVersionNumbers) { 
            return $this->_platformVersion; 
        } else { 
            switch ($this->getPlatform()) { 
                case self::PLATFORM_WINDOWS: 
                    if (substr($this->_platformVersion, 0, 3) == 'NT ') { 
                        return $this->windowsNTVerToStr(substr($this->_platformVersion, 3), $returnServerFlavor); 
                    } else { 
                        return $this->windowsVerToStr($this->_platformVersion); 
                    } 
                break; 
 
                default: return self::PLATFORM_VERSION_UNKNOWN; 
            } 
        } 
    } 
 
    /** 
     * Get the user agent value used by the class to determine the browser details. 
     * @return string The user agent string. 
     */ 
    public function getUserAgent() 
    { 
        return $this->_agent; 
    } 
 
    /** 
     * Get the version of the browser. 
     * @return string Returns the version of the browser or BrowserDetection::VERSION_UNKNOWN if unknown. 
     */ 
    public function getVersion() 
    { 
        return $this->_version; 
    } 
 
    /** 
     * Determine if the browser is executed from a 64-bit platform. Keep in mind that not all platforms/browsers report 
     * this and the result may not always be accurate. 
     * @return boolean Returns true if the browser is executed from a 64-bit platform. 
     */ 
    public function is64bitPlatform() 
    { 
        return $this->_is64bit; 
    } 
 
    /** 
     * Determine if the browser is from AOL. AOL releases "optimized" Internet Explorer and Firefox versions. In the 
     * making they add their details in the user agent string of these browsers. 
     * @return boolean Returns true if the browser is from AOL, false otherwise. 
     */ 
    public function isAol() 
    { 
        return $this->_isAol; 
    } 
 
    /** 
     * Determine if the browser runs Google Chrome Frame (it's a plug-in designed for Internet Explorer 6+ based on the 
     * open-source Chromium project - it's like a Chrome browser within IE). 
     * @return boolean Returns true if the browser is using Google Chrome Frame, false otherwise. 
     */ 
    public function isChromeFrame() 
    { 
        return stripos($this->_agent, 'chromeframe') !== false; 
    } 
 
    /** 
     * Determine if the browser is in compatibility view or not. Since Internet Explorer 8, IE can be put in 
     * compatibility mode to make websites that were created for older browsers, especially IE 6 and 7, look better in 
     * IE 8+ which renders web pages closer to the standards and thus differently from those older versions of IE. 
     * @return boolean Returns true if the browser is in compatibility view, false otherwise. 
     */ 
    public function isInIECompatibilityView() 
    { 
        return ($this->_compatibilityViewName != '') || ($this->_compatibilityViewVer != ''); 
    } 
 
    /** 
     * Determine if the browser is from a mobile device or not. 
     * @return boolean Returns true if the browser is from a mobile device, false otherwise. 
     */ 
    public function isMobile() 
    { 
        return $this->_isMobile; 
    } 
 
    /** 
     * Determine if the browser is a robot (Googlebot, Bingbot, Yahoo! Slurp...) or not. 
     * @return boolean Returns true if the browser is a robot, false otherwise. 
     */ 
    public function isRobot() 
    { 
        return $this->_isRobot; 
    } 
 
    /** 
     * Set the user agent to use with the class. 
     * @param string $agentString The value of the user agent. If an empty string is sent (default), 
     * $_SERVER['HTTP_USER_AGENT'] will be used. 
     */ 
    public function setUserAgent($agentString = '') 
    { 
        if (!is_string($agentString) || trim($agentString) == '') { 
            if (array_key_exists('HTTP_USER_AGENT', $_SERVER) && is_string($_SERVER['HTTP_USER_AGENT'])) { 
                $agentString = $_SERVER['HTTP_USER_AGENT']; 
            } else { 
                $agentString = ''; 
            } 
        } 
 
        $this->reset(); 
        $this->_agent = $agentString; 
        $this->detect(); 
    } 
 
 
    //--- PROTECTED MEMBERS -------------------------------------------------------------------------------------------- 
 
 
    /** 
     * Determine if the browser is the Amaya Web editor or not. 
     * @access protected 
     * @link http://www.w3.org/Amaya/ 
     * @return boolean Returns true if the browser is Amaya, false otherwise. 
     */ 
    protected function checkBrowserAmaya() 
    { 
        return $this->checkSimpleBrowserUA('amaya', $this->_agent, self::BROWSER_AMAYA); 
    } 
 
    /** 
     * Determine if the browser is the Android browser (based on the WebKit layout engine and coupled with Chrome's 
     * JavaScript engine) or not. 
     * @access protected 
     * @return boolean Returns true if the browser is the Android browser, false otherwise. 
     */ 
    protected function checkBrowserAndroid() 
    { 
        //Android don't use the standard "Android/1.0", it uses "Android 1.0;" instead 
        return $this->checkSimpleBrowserUA('Android', $this->_agent, self::BROWSER_ANDROID, true); 
    } 
 
    /** 
     * Determine if the browser is the Bingbot crawler or not. 
     * @access protected 
     * @link http://www.bing.com/webmaster/help/which-crawlers-does-bing-use-8c184ec0 
     * @return boolean Returns true if the browser is Bingbot, false otherwise. 
     */ 
    protected function checkBrowserBingbot() 
    { 
        return $this->checkSimpleBrowserUA('bingbot', $this->_agent, self::BROWSER_BINGBOT, false, true); 
    } 
 
    /** 
     * Determine if the browser is the BlackBerry browser or not. 
     * @access protected 
     * @link http://supportforums.blackberry.com/t5/Web-and-WebWorks-Development/How-to-detect-the-BlackBerry-Browser/ta-p/559862 
     * @return boolean Returns true if the browser is the BlackBerry browser, false otherwise. 
     */ 
    protected function checkBrowserBlackBerry() 
    { 
        $found = false; 
 
        //Tablet OS check 
        if ($this->checkSimpleBrowserUA('RIM Tablet OS', $this->_agent, self::BROWSER_TABLET_OS, true)) { 
            return true; 
        } 
 
        //Version 6, 7 & 10 check (versions 8 & 9 does not exists) 
        if ($this->checkBrowserUAWithVersion(array('BlackBerry', 'BB10'), $this->_agent, self::BROWSER_BLACKBERRY, true)) { 
            if ($this->getVersion() == self::VERSION_UNKNOWN) { 
                $found = true; 
            } else { 
                return true; 
            } 
        } 
 
        //Version 4.2 to 5.0 check 
        if ($this->checkSimpleBrowserUA('BlackBerry', $this->_agent, self::BROWSER_BLACKBERRY, true)) { 
            if ($this->getVersion() == self::VERSION_UNKNOWN) { 
                $found = true; 
            } else { 
                return true; 
            } 
        } 
 
        return $found; 
    } 
 
    /** 
     * Determine if the browser is Chrome or not. 
     * @access protected 
     * @link http://www.google.com/chrome/ 
     * @return boolean Returns true if the browser is Chrome, false otherwise. 
     */ 
    protected function checkBrowserChrome() 
    { 
        return $this->checkSimpleBrowserUA('Chrome', $this->_agent, self::BROWSER_CHROME); 
    } 
 
    /** 
     * Determine if the browser is Edge or not. 
     * @access protected 
     * @return boolean Returns true if the browser is Edge, false otherwise. 
     */ 
    protected function checkBrowserEdge() 
    { 
        return $this->checkSimpleBrowserUA('Edge', $this->_agent, self::BROWSER_EDGE); 
    } 
 
    /** 
     * Determine if the browser is Firebird or not. Firebird was the name of Firefox from version 0.6 to 0.7.1. 
     * @access protected 
     * @return boolean Returns true if the browser is Firebird, false otherwise. 
     */ 
    protected function checkBrowserFirebird() 
    { 
        return $this->checkSimpleBrowserUA('Firebird', $this->_agent, self::BROWSER_FIREBIRD); 
    } 
 
    /** 
     * Determine if the browser is Firefox or not. 
     * @access protected 
     * @link http://www.mozilla.org/en-US/firefox/new/ 
     * @return boolean Returns true if the browser is Firefox, false otherwise. 
     */ 
    protected function checkBrowserFirefox() 
    { 
        //Safari heavily matches with Firefox, ensure that Safari is filtered out... 
        if (preg_match('/.*Firefox[ (\/]*([a-z0-9.-]*)/i', $this->_agent, $matches) && 
                stripos($this->_agent, 'Safari') === false) { 
            $this->setBrowser(self::BROWSER_FIREFOX); 
            $this->setVersion($matches[1]); 
            $this->setMobile(false); 
            $this->setRobot(false); 
 
            return true; 
        } 
 
        return false; 
    } 
 
    /** 
     * Determine if the browser is Galeon or not. The browser was discontinued on September 27, 2008. 
     * @access protected 
     * @link http://en.wikipedia.org/wiki/Galeon 
     * @return boolean Returns true if the browser is Galeon, false otherwise. 
     */ 
    protected function checkBrowserGaleon() 
    { 
        return $this->checkSimpleBrowserUA('Galeon', $this->_agent, self::BROWSER_GALEON); 
    } 
 
    /** 
     * Determine if the browser is the Googlebot crawler or not. 
     * @access protected 
     * @return boolean Returns true if the browser is Googlebot, false otherwise. 
     */ 
    protected function checkBrowserGooglebot() 
    { 
        if ($this->checkSimpleBrowserUA('Googlebot', $this->_agent, self::BROWSER_GOOGLEBOT, false, true)) { 
            if (strpos(strtolower($this->_agent), 'googlebot-mobile') !== false) { 
                $this->setMobile(true); 
            } 
 
            return true; 
        } 
 
        return false; 
    } 
 
    /** 
     * Determine if the browser is iCab or not. 
     * @access protected 
     * @link http://www.icab.de/ 
     * @return boolean Returns true if the browser is iCab, false otherwise. 
     */ 
    protected function checkBrowserIcab() 
    { 
        //Some (early) iCab versions don't use the standard "iCab/1.0", they uses "iCab 1.0;" instead 
        return $this->checkSimpleBrowserUA('iCab', $this->_agent, self::BROWSER_ICAB); 
    } 
 
    /** 
     * Determine if the browser is GNU IceCat (formerly known as GNU IceWeasel) or not. 
     * @access protected 
     * @link http://www.gnu.org/software/gnuzilla/ 
     * @return boolean Returns true if the browser is GNU IceCat, false otherwise. 
     */ 
    protected function checkBrowserIceCat() 
    { 
        return $this->checkSimpleBrowserUA('IceCat', $this->_agent, self::BROWSER_ICECAT); 
    } 
 
    /** 
     * Determine if the browser is GNU IceWeasel (now know as GNU IceCat) or not. 
     * @access protected 
     * @see checkBrowserIceCat() 
     * @return boolean Returns true if the browser is GNU IceWeasel, false otherwise. 
     */ 
    protected function checkBrowserIceWeasel() 
    { 
        return $this->checkSimpleBrowserUA('Iceweasel', $this->_agent, self::BROWSER_ICEWEASEL); 
    } 
 
    /** 
     * Determine if the browser is Internet Explorer or not. 
     * @access protected 
     * @link http://www.microsoft.com/ie/ 
     * @link http://en.wikipedia.org/wiki/Internet_Explorer_Mobile 
     * @return boolean Returns true if the browser is Internet Explorer, false otherwise. 
     */ 
    protected function checkBrowserInternetExplorer() 
    { 
        //Test for Internet Explorer Mobile (formerly Pocket Internet Explorer) 
        if ($this->checkSimpleBrowserUA(array('IEMobile', 'MSPIE'), $this->_agent, self::BROWSER_IE_MOBILE, true)) { 
            return true; 
        } 
 
        //Several browsers uses IE compatibility UAs filter these browsers out (but after testing for IE Mobile) 
        if (stripos($this->_agent, 'Opera') !== false || 
                stripos($this->_agent, 'BlackBerry') !== false || 
                stripos($this->_agent, 'Nokia') !== false) { 
            return false; 
        } 
 
        //Test for Internet Explorer 1 
        if ($this->checkSimpleBrowserUA('Microsoft Internet Explorer', $this->_agent, self::BROWSER_IE)) { 
            if ($this->getVersion() == self::VERSION_UNKNOWN) { 
                if (preg_match('/308|425|426|474|0b1/i', $this->_agent)) { 
                    $this->setVersion('1.5'); 
                } else { 
                    $this->setVersion('1.0'); 
                } 
            } 
            return true; 
        } 
 
        //Test for Internet Explorer 2+ 
        if (stripos($this->_agent, 'MSIE') !== false || stripos($this->_agent, 'Trident') !== false) { 
            $version = ''; 
 
            if (stripos($this->_agent, 'Trident') !== false) { 
                //Test for Internet Explorer 11+ (check the rv: string) 
                if (stripos($this->_agent, 'rv:') !== false) { 
                    if ($this->checkSimpleBrowserUA('Trident', $this->_agent, self::BROWSER_IE, false, false, 'rv:')) { 
                        return true; 
                    } 
                } else { 
                    //Test for Internet Explorer 8, 9 & 10 (check the Trident string) 
                    if (preg_match('/Trident\/([\d]+)/i', $this->_agent, $foundVersion)) { 
                        //Trident started with version 4.0 on IE 8 
                        $verFromTrident = $this->parseInt($foundVersion[1]) + 4; 
                        if ($verFromTrident >= 8) { 
                            $version = $verFromTrident . '.0'; 
                        } 
                    } 
                } 
 
                //If we have the IE version from Trident, we can check for the compatibility view mode 
                if ($version != '') { 
                    $emulatedVer = ''; 
                    preg_match_all('/MSIE\s*([^\s;$]+)/i', $this->_agent, $foundVersions); 
                    foreach ($foundVersions[1] as $currVer) { 
                        //Keep the lowest MSIE version for the emulated version (in compatibility view mode) 
                        if ($emulatedVer == '' || $this->compareVersions($emulatedVer, $currVer) == 1) { 
                            $emulatedVer = $currVer; 
                        } 
                    } 
                    //Set the compatibility view mode if $version != $emulatedVer 
                    if ($this->compareVersions($version, $emulatedVer) != 0) { 
                        $this->_compatibilityViewName = self::BROWSER_IE; 
                        $this->_compatibilityViewVer = $this->cleanVersion($emulatedVer); 
                    } 
                } 
            } 
 
            //Test for Internet Explorer 2-7 versions if needed 
            if ($version == '') { 
                preg_match_all('/MSIE\s+([^\s;$]+)/i', $this->_agent, $foundVersions); 
                foreach ($foundVersions[1] as $currVer) { 
                    //Keep the highest MSIE version 
                    if ($version == '' || $this->compareVersions($version, $currVer) == -1) { 
                        $version = $currVer; 
                    } 
                } 
            } 
 
            $this->setBrowser(self::BROWSER_IE); 
            $this->setVersion($version); 
            $this->setMobile(false); 
            $this->setRobot(false); 
 
            return true; 
        } 
 
        return false; 
    } 
 
    /** 
     * Determine if the browser is Konqueror or not. 
     * @access protected 
     * @link http://www.konqueror.org/ 
     * @return boolean Returns true if the browser is Konqueror, false otherwise. 
     */ 
    protected function checkBrowserKonqueror() 
    { 
        return $this->checkSimpleBrowserUA('Konqueror', $this->_agent, self::BROWSER_KONQUEROR); 
    } 
 
    /** 
     * Determine if the browser is Lynx or not. It is the oldest web browser currently in general use and development. 
     * It is a text-based only Web browser. 
     * @access protected 
     * @link http://en.wikipedia.org/wiki/Lynx 
     * @return boolean Returns true if the browser is Lynx, false otherwise. 
     */ 
    protected function checkBrowserLynx() 
    { 
        return $this->checkSimpleBrowserUA('Lynx', $this->_agent, self::BROWSER_LYNX); 
    } 
 
    /** 
     * Determine if the browser is Mozilla or not. 
     * @access protected 
     * @return boolean Returns true if the browser is Mozilla, false otherwise. 
     */ 
    protected function checkBrowserMozilla() 
    { 
        return $this->checkSimpleBrowserUA('Mozilla', $this->_agent, self::BROWSER_MOZILLA, false, false, 'rv:'); 
    } 
 
    /** 
     * Determine if the browser is the MSNBot crawler or not. In October 2010 it was replaced by the Bingbot robot. 
     * @access protected 
     * @see checkBrowserBingbot() 
     * @return boolean Returns true if the browser is MSNBot, false otherwise. 
     */ 
    protected function checkBrowserMsnBot() 
    { 
        return $this->checkSimpleBrowserUA('msnbot', $this->_agent, self::BROWSER_MSNBOT, false, true); 
    } 
 
    /** 
     * Determine if the browser is MSN TV (formerly WebTV) or not. 
     * @access protected 
     * @link http://en.wikipedia.org/wiki/MSN_TV 
     * @return boolean Returns true if the browser is WebTv, false otherwise. 
     */ 
    protected function checkBrowserMsnTv() 
    { 
        return $this->checkSimpleBrowserUA('webtv', $this->_agent, self::BROWSER_MSNTV); 
    } 
 
    /** 
     * Determine if the browser is NetPositive or not. The browser is discontinued since November 2001. 
     * @access protected 
     * @link http://en.wikipedia.org/wiki/NetPositive 
     * @return boolean Returns true if the browser is NetPositive, false otherwise. 
     */ 
    protected function checkBrowserNetPositive() 
    { 
        return $this->checkSimpleBrowserUA('NetPositive', $this->_agent, self::BROWSER_NETPOSITIVE); 
    } 
 
    /** 
     * Determine if the browser is Netscape or not. Official support for this browser ended on March 1st, 2008. 
     * @access protected 
     * @link http://en.wikipedia.org/wiki/Netscape 
     * @return boolean Returns true if the browser is Netscape, false otherwise. 
     */ 
    protected function checkBrowserNetscape() 
    { 
        //BlackBerry & Nokia UAs can conflict with Netscape UAs 
        if (stripos($this->_agent, 'BlackBerry') !== false || stripos($this->_agent, 'Nokia') !== false) { 
            return false; 
        } 
 
        //Netscape v6 to v9 check 
        if ($this->checkSimpleBrowserUA(array('Netscape', 'Navigator', 'Netscape6'), $this->_agent, self::BROWSER_NETSCAPE)) { 
            return true; 
        } 
 
        //Netscape v1-4 (v5 don't exists) 
        $found = false; 
        if (stripos($this->_agent, 'Mozilla') !== false && stripos($this->_agent, 'rv:') === false) { 
            $version = ''; 
            $verParts = explode('/', stristr($this->_agent, 'Mozilla')); 
            if (count($verParts) > 1) { 
                $verParts = explode(' ', $verParts[1]); 
                $verParts = explode('.', $verParts[0]); 
 
                $majorVer = $this->parseInt($verParts[0]); 
                if ($majorVer > 0 && $majorVer < 5) { 
                    $version = implode('.', $verParts); 
                    $found = true; 
 
                    if (strtolower(substr($version, -4)) == '-sgi') { 
                        $version = substr($version, 0, -4); 
                    } else { 
                        if (strtolower(substr($version, -4)) == 'gold') { 
                            $version = substr($version, 0, -4) . ' Gold'; //Doubles spaces (if any) will be normalized by setVersion() 
                        } 
                    } 
                } 
            } 
        } 
 
        if ($found) { 
            $this->setBrowser(self::BROWSER_NETSCAPE); 
            $this->setVersion($version); 
            $this->setMobile(false); 
            $this->setRobot(false); 
        } 
 
        return $found; 
    } 
 
    /** 
     * Determine if the browser is a Nokia browser or not. 
     * @access protected 
     * @link http://www.developer.nokia.com/Community/Wiki/User-Agent_headers_for_Nokia_devices 
     * @return boolean Returns true if the browser is a Nokia browser, false otherwise. 
     */ 
    protected function checkBrowserNokia() 
    { 
        if (stripos($this->_agent, 'Nokia5800') !== false || stripos($this->_agent, 'Nokia5530') !== false || stripos($this->_agent, 'Nokia5230') !== false) { 
            $this->setBrowser(self::BROWSER_NOKIA); 
            $this->setVersion('7.0'); 
            $this->setMobile(true); 
            $this->setRobot(false); 
 
            return true; 
        } 
 
        if ($this->checkSimpleBrowserUA(array('NokiaBrowser', 'BrowserNG', 'Series60', 'S60', 'S40OviBrowser'), $this->_agent, self::BROWSER_NOKIA, true)) { 
            return true; 
        } 
 
        return false; 
    } 
 
    /** 
     * Determine if the browser is OmniWeb or not. 
     * @access protected 
     * @link http://www.omnigroup.com/products/omniweb/ 
     * @return boolean Returns true if the browser is OmniWeb, false otherwise. 
     */ 
    protected function checkBrowserOmniWeb() 
    { 
        if ($this->checkSimpleBrowserUA('OmniWeb', $this->_agent, self::BROWSER_OMNIWEB)) { 
            //Some versions of OmniWeb prefix the version number with "v" 
            if ($this->getVersion() != self::VERSION_UNKNOWN && strtolower(substr($this->getVersion(), 0, 1)) == 'v') { 
                $this->setVersion(substr($this->getVersion(), 1)); 
            } 
            return true; 
        } 
 
        return false; 
    } 
 
    /** 
     * Determine if the browser is Opera or not. 
     * @access protected 
     * @link http://www.opera.com/ 
     * @link http://www.opera.com/mini/ 
     * @link http://www.opera.com/mobile/ 
     * @link http://my.opera.com/community/openweb/idopera/ 
     * @return boolean Returns true if the browser is Opera, false otherwise. 
     */ 
    protected function checkBrowserOpera() 
    { 
        if ($this->checkBrowserUAWithVersion('Opera Mobi', $this->_agent, self::BROWSER_OPERA_MOBILE, true)) { 
            return true; 
        } 
 
        if ($this->checkSimpleBrowserUA('Opera Mini', $this->_agent, self::BROWSER_OPERA_MINI, true)) { 
            return true; 
        } 
 
        $version = ''; 
        $found = $this->checkBrowserUAWithVersion('Opera', $this->_agent, self::BROWSER_OPERA); 
        if ($found && $this->getVersion() != self::VERSION_UNKNOWN) { 
            $version = $this->getVersion(); 
        } 
 
        if (!$found || $version == '') { 
            if ($this->checkSimpleBrowserUA('Opera', $this->_agent, self::BROWSER_OPERA)) { 
                return true; 
            } 
        } 
 
        if (!$found && $this->checkSimpleBrowserUA('Chrome', $this->_agent, self::BROWSER_CHROME) ) { 
            if ($this->checkSimpleBrowserUA('OPR/', $this->_agent, self::BROWSER_OPERA)) { 
                return true; 
            } 
        } 
 
        return $found; 
    } 
 
    /** 
     * Determine if the browser is Phoenix or not. Phoenix was the name of Firefox from version 0.1 to 0.5. 
     * @access protected 
     * @return boolean Returns true if the browser is Phoenix, false otherwise. 
     */ 
    protected function checkBrowserPhoenix() 
    { 
        return $this->checkSimpleBrowserUA('Phoenix', $this->_agent, self::BROWSER_PHOENIX); 
    } 
 
    /** 
     * Determine what is the browser used by the user. 
     * @access protected 
     * @return boolean Returns true if the browser has been identified, false otherwise. 
     */ 
    protected function checkBrowsers() 
    { 
        //Changing the check order can break the class detection results! 
        return 
               /* Major browsers and browsers that need to be detected in a special order */ 
               $this->checkBrowserMsnTv() ||            /* MSN TV is based on IE so we must check for MSN TV before IE */ 
               $this->checkBrowserInternetExplorer() || 
               $this->checkBrowserOpera() ||            /* Opera must be checked before Firefox, Netscape and Chrome to avoid conflicts */ 
               $this->checkBrowserEdge() ||             /* Edge must be checked before Firefox, Safari and Chrome to avoid conflicts */ 
               $this->checkBrowserChrome() ||           /* Chrome must be checked before Netscaoe and Mozilla to avoid conflicts */ 
               $this->checkBrowserOmniWeb() ||          /* OmniWeb must be checked before Safari (on which it's based on) and Netscape (since it have Mozilla UAs) */ 
               $this->checkBrowserIcab() ||             /* Check iCab before Netscape since iCab have Mozilla UAs */ 
               $this->checkBrowserNetPositive() ||      /* Check NetPositive before Netscape since NetPositive have Mozilla UAs */ 
               $this->checkBrowserNetscape() ||         /* Must be checked before Firefox since Netscape 8-9 are based on Firefox */ 
               $this->checkBrowserIceCat() ||           /* Check IceCat and IceWeasel before Firefox since they are GNU builds of Firefox */ 
               $this->checkBrowserIceWeasel() || 
               $this->checkBrowserGaleon() ||           /* Galeon is based on Firefox and needs to be tested before Firefox is tested */ 
               $this->checkBrowserFirefox() || 
               /* Current browsers that don't need to be detected in any special order */ 
               $this->checkBrowserKonqueror() || 
               $this->checkBrowserLynx() || 
               $this->checkBrowserAmaya() || 
               /* Mobile */ 
               $this->checkBrowserAndroid() || 
               $this->checkBrowserBlackBerry() || 
               $this->checkBrowserNokia() || 
               /* Bots */ 
               $this->checkBrowserGooglebot() || 
               $this->checkBrowserBingbot() || 
               $this->checkBrowserMsnBot() || 
               $this->checkBrowserSlurp() || 
               $this->checkBrowserYahooMultimedia() || 
               $this->checkBrowserW3CValidator() || 
               /* WebKit base check (after most other checks) */ 
               $this->checkBrowserSafari() || 
               /* Deprecated browsers that don't need to be detected in any special order */ 
               $this->checkBrowserFirebird() || 
               $this->checkBrowserPhoenix() || 
               /* Mozilla is such an open standard that it must be checked last */ 
               $this->checkBrowserMozilla(); 
    } 
 
    /** 
     * Determine if the browser is Safari or not. 
     * @access protected 
     * @link http://www.apple.com/safari/ 
     * @link http://web.archive.org/web/20080514173941/http://developer.apple.com/internet/safari/uamatrix.html 
     * @link http://en.wikipedia.org/wiki/Safari_version_history#Release_history 
     * @return boolean Returns true if the browser is Safari, false otherwise. 
     */ 
    protected function checkBrowserSafari() 
    { 
        $version = ''; 
 
        //Check for current versions of Safari 
        $found = $this->checkBrowserUAWithVersion(array('Safari', 'AppleWebKit'), $this->_agent, self::BROWSER_SAFARI); 
        if ($found && $this->getVersion() != self::VERSION_UNKNOWN) { 
            $version = $this->getVersion(); 
        } 
 
        //Safari 1-2 didn't had a "Version" string in the UA, only a WebKit build and/or Safari build, extract version from these... 
        if (!$found || $version == '') { 
            if (preg_match('/.*Safari[ (\/]*([a-z0-9.-]*)/i', $this->_agent, $matches)) { 
                $version = $this->safariBuildToSafariVer($matches[1]); 
                $found = true; 
            } 
        } 
        if (!$found || $version == '') { 
            if (preg_match('/.*AppleWebKit[ (\/]*([a-z0-9.-]*)/i', $this->_agent, $matches)) { 
                $version = $this->webKitBuildToSafariVer($matches[1]); 
                $found = true; 
            } 
        } 
 
        if ($found) { 
            $this->setBrowser(self::BROWSER_SAFARI); 
            $this->setVersion($version); 
            $this->setMobile(false); 
            $this->setRobot(false); 
        } 
 
        return $found; 
    } 
 
    /** 
     * Determine if the browser is the Yahoo! Slurp crawler or not. 
     * @access protected 
     * @return boolean Returns true if the browser is Yahoo! Slurp, false otherwise. 
     */ 
    protected function checkBrowserSlurp() 
    { 
        return $this->checkSimpleBrowserUA('Yahoo! Slurp', $this->_agent, self::BROWSER_SLURP, false, true); 
    } 
 
    /** 
     * Test the user agent for a specific browser that use a "Version" string (like Safari and Opera). The user agent 
     * should look like: "Version/1.0 Browser name/123.456" or "Browser name/123.456 Version/1.0". 
     * @access protected 
     * @param mixed $uaNameToLookFor The string (or array of strings) representing the browser name to find in the user 
     * agent. 
     * @param string $userAgent The user agent string to work with. 
     * @param string $browserName The literal browser name. Always use a class constant! 
     * @param boolean $isMobile Determines if the browser is from a mobile device. 
     * @param boolean $isRobot Determines if the browser is a robot or not. 
     * @return boolean Returns true if we found the browser we were looking for, false otherwise. 
     */ 
    protected function checkBrowserUAWithVersion($uaNameToLookFor, $userAgent, $browserName, $isMobile = false, $isRobot = false) 
    { 
        if (!is_array($uaNameToLookFor)) { 
            $uaNameToLookFor = array($uaNameToLookFor); 
        } 
 
        foreach ($uaNameToLookFor as $currUANameToLookFor) { 
            if (stripos($userAgent, $currUANameToLookFor) !== false) { 
                $version = ''; 
                $verParts = explode('/', stristr($this->_agent, 'Version')); 
                if (count($verParts) > 1) { 
                    $verParts = explode(' ', $verParts[1]); 
                    $version = $verParts[0]; 
                } 
 
                $this->setBrowser($browserName); 
                $this->setVersion($version); 
 
                $this->setMobile($isMobile); 
                $this->setRobot($isRobot); 
 
                return true; 
            } 
        } 
 
        return false; 
    } 
 
    /** 
     * Determine if the browser is the W3C Validator or not. 
     * @access protected 
     * @link http://validator.w3.org/ 
     * @return boolean Returns true if the browser is the W3C Validator, false otherwise. 
     */ 
    protected function checkBrowserW3CValidator() 
    { 
        //Since the W3C validates pages with different robots we will prefix our versions with the part validated on the page... 
 
        //W3C Link Checker (prefixed with "Link-") 
        if ($this->checkSimpleBrowserUA('W3C-checklink', $this->_agent, self::BROWSER_W3CVALIDATOR, false, true)) { 
            if ($this->getVersion() != self::VERSION_UNKNOWN) { 
                $this->setVersion('Link-' . $this->getVersion()); 
            } 
            return true; 
        } 
 
        //W3C CSS Validation Service (prefixed with "CSS-") 
        if ($this->checkSimpleBrowserUA('Jigsaw', $this->_agent, self::BROWSER_W3CVALIDATOR, false, true)) { 
            if ($this->getVersion() != self::VERSION_UNKNOWN) { 
                $this->setVersion('CSS-' . $this->getVersion()); 
            } 
            return true; 
        } 
 
        //W3C mobileOK Checker (prefixed with "mobileOK-") 
        if ($this->checkSimpleBrowserUA('W3C-mobileOK', $this->_agent, self::BROWSER_W3CVALIDATOR, false, true)) { 
            if ($this->getVersion() != self::VERSION_UNKNOWN) { 
                $this->setVersion('mobileOK-' . $this->getVersion()); 
            } 
            return true; 
        } 
 
        //W3C Markup Validation Service (no prefix) 
        return $this->checkSimpleBrowserUA('W3C_Validator', $this->_agent, self::BROWSER_W3CVALIDATOR, false, true); 
    } 
 
    /** 
     * Determine if the browser is the Yahoo! multimedia crawler or not. 
     * @access protected 
     * @return boolean Returns true if the browser is the Yahoo! multimedia crawler, false otherwise. 
     */ 
    protected function checkBrowserYahooMultimedia() 
    { 
        return $this->checkSimpleBrowserUA('Yahoo-MMCrawler', $this->_agent, self::BROWSER_YAHOO_MM, false, true); 
    } 
 
    /** 
     * Determine if the user is using an AOL "optimized" browser or not. 
     * @access protected 
     * @return boolean Returns true if the browser is AOL optimized, false otherwise. 
     */ 
    protected function checkForAol() 
    { 
        //AOL UAs don't use the "AOL/1.0" format, they uses "AOL 1.0; AOLBuild 100.00;" 
        if (stripos($this->_agent, 'AOL ') !== false) { 
            $version = ''; 
            $verParts = explode('AOL ', stristr($this->_agent, 'AOL ')); 
            if (count($verParts) > 1) { 
                $verParts = explode(' ', $verParts[1]); 
                $version = $verParts[0]; 
            } 
 
            $this->setAol(true); 
            $this->setAolVersion($version); 
 
            return true; 
        } else { 
            $this->setAol(false); 
            $this->setAolVersion(''); 
 
            return false; 
        } 
    } 
 
    /** 
     * Determine the user's platform. 
     * @access protected 
     */ 
    protected function checkPlatform() 
    { 
        /* Mobile platforms */ 
        if (stripos($this->_agent, 'Windows Phone') !== false ||     /* Check Windows Phone (formerly Windows Mobile) before Windows */ 
                stripos($this->_agent, 'IEMobile') !== false) { 
            $this->setPlatform(self::PLATFORM_WINDOWS_PHONE); 
            $this->setMobile(true); 
        } else if (stripos($this->_agent, 'Windows CE') !== false) { /* Check Windows CE before Windows */ 
            $this->setPlatform(self::PLATFORM_WINDOWS_CE); 
            $this->setMobile(true); 
        } else if (stripos($this->_agent, 'iPhone') !== false) {     /* Check iPad/iPod/iPhone before Macintosh */ 
            $this->setPlatform(self::PLATFORM_IPHONE); 
            $this->setMobile(true); 
        } else if (stripos($this->_agent, 'iPad') !== false) { 
            $this->setPlatform(self::PLATFORM_IPAD); 
            $this->setMobile(true); 
        } else if (stripos($this->_agent, 'iPod') !== false) { 
            $this->setPlatform(self::PLATFORM_IPOD); 
            $this->setMobile(true); 
        } else if (stripos($this->_agent, 'Android') !== false) { 
            $this->setPlatform(self::PLATFORM_ANDROID); 
            $this->setMobile(true); 
        } else if (stripos($this->_agent, 'Symbian') !== false) { 
            $this->setPlatform(self::PLATFORM_SYMBIAN); 
            $this->setMobile(true); 
        } else if (stripos($this->_agent, 'BlackBerry') !== false || 
                stripos($this->_agent, 'BB10') !== false || 
                stripos($this->_agent, 'RIM Tablet OS') !== false) { 
            $this->setPlatform(self::PLATFORM_BLACKBERRY); 
            $this->setMobile(true); 
        } else if (stripos($this->_agent, 'Nokia') !== false) { 
            $this->setPlatform(self::PLATFORM_NOKIA); 
            $this->setMobile(true); 
 
        /* Desktop platforms */ 
        } else if (stripos($this->_agent, 'Windows') !== false) { 
            $this->setPlatform(self::PLATFORM_WINDOWS); 
        } else if (stripos($this->_agent, 'Macintosh') !== false) { 
            $this->setPlatform(self::PLATFORM_MACINTOSH); 
        } else if (stripos($this->_agent, 'Linux') !== false) { 
            $this->setPlatform(self::PLATFORM_LINUX); 
        } else if (stripos($this->_agent, 'FreeBSD') !== false) { 
            $this->setPlatform(self::PLATFORM_FREEBSD); 
        } else if (stripos($this->_agent, 'OpenBSD') !== false) { 
            $this->setPlatform(self::PLATFORM_OPENBSD); 
        } else if (stripos($this->_agent, 'NetBSD') !== false) { 
            $this->setPlatform(self::PLATFORM_NETBSD); 
 
        /* Discontinued */ 
        } else if (stripos($this->_agent, 'OpenSolaris') !== false) { 
            $this->setPlatform(self::PLATFORM_OPENSOLARIS); 
        } else if (stripos($this->_agent, 'OS/2') !== false) { 
            $this->setPlatform(self::PLATFORM_OS2); 
        } else if (stripos($this->_agent, 'BeOS') !== false) { 
            $this->setPlatform(self::PLATFORM_BEOS); 
        } else if (stripos($this->_agent, 'SunOS') !== false) { 
            $this->setPlatform(self::PLATFORM_SUNOS); 
 
        /* Generic */ 
        } else if (stripos($this->_agent, 'Win') !== false) { 
            $this->setPlatform(self::PLATFORM_WINDOWS); 
        } else if (stripos($this->_agent, 'Mac') !== false) { 
            $this->setPlatform(self::PLATFORM_MACINTOSH); 
        } 
 
        //Check if it's a 64-bit platform 
        if (stripos($this->_agent, 'WOW64') !== false || stripos($this->_agent, 'Win64') !== false) { 
            $this->set64bit(true); 
        } 
 
        $this->checkPlatformVersion(); 
    } 
 
    /** 
     * Determine the user's platform version. 
     * @access protected 
     */ 
    protected function checkPlatformVersion() 
    { 
        //https://support.microsoft.com/en-us/kb/158238 
 
        $result = ''; 
 
        switch ($this->getPlatform()) { 
            case self::PLATFORM_WINDOWS: 
                if (preg_match('/Windows NT\s*([^\s;\)$]+)/i', $this->_agent, $foundVersion)) { 
                    //Windows NT family 
                    $result = 'NT ' . $foundVersion[1]; 
                } else { 
                    //Windows 3.x / 9x family 
                    if (stripos($this->_agent, 'Windows ME') !== false) { 
                        $result = '4.90.3000'; //Windows Me version range from 4.90.3000 to 4.90.3000A 
                    } else if (stripos($this->_agent, 'Windows 98') !== false) { 
                        $result = '4.10'; //Windows 98 version range from 4.10.1998 to 4.10.2222B 
                    } else if (stripos($this->_agent, 'Windows 95') !== false) { 
                        $result = '4.00'; //Windows 95 version range from 4.00.950 to 4.03.1214 
                    } else if (preg_match('/Windows 3\.([^\s;\)$]+)/i', $this->_agent, $foundVersion)) { 
                        $result = '3.' . $foundVersion[1]; 
                    } 
                } 
 
            break; 
        } 
 
        if (trim($result) == '') { 
            $result = self::PLATFORM_VERSION_UNKNOWN; 
        } 
        $this->setPlatformVersion($result); 
    } 
 
    /** 
     * Test the user agent for a specific browser where the browser name is immediately followed by the version number. 
     * The user agent should look like: "Browser name/1.0" or "Browser 1.0;". 
     * @access protected 
     * @param mixed $uaNameToLookFor The string (or array of strings) representing the browser name to find in the user 
     * agent. 
     * @param string $userAgent The user agent string to work with. 
     * @param string $browserName The literal browser name. Always use a class constant! 
     * @param boolean $isMobile Determines if the browser is from a mobile device. 
     * @param boolean $isRobot Determines if the browser is a robot or not. 
     * @param string $separator The separator string used to split the browser name and the version number in the user 
     * agent. 
     * @return boolean Returns true if we found the browser we were looking for, false otherwise. 
     */ 
    protected function checkSimpleBrowserUA($uaNameToLookFor, $userAgent, $browserName, $isMobile = false, $isRobot = false, $separator = '/') 
    { 
        if (!is_array($uaNameToLookFor)) { 
            $uaNameToLookFor = array($uaNameToLookFor); 
        } 
 
        foreach ($uaNameToLookFor as $currUANameToLookFor) { 
            if (stripos($userAgent, $currUANameToLookFor) !== false) { 
                //Many browsers don't use the standard "Browser/1.0" format, they uses "Browser 1.0;" instead 
                if (stripos($userAgent, $currUANameToLookFor . $separator) === false) { 
                    $userAgent = str_ireplace($currUANameToLookFor . ' ', $currUANameToLookFor . $separator, $this->_agent); 
                } 
 
                $version = ''; 
                $verParts = explode($separator, stristr($userAgent, $currUANameToLookFor)); 
                if (count($verParts) > 1) { 
                    $verParts = explode(' ', $verParts[1]); 
                    $version = $verParts[0]; 
                } 
 
                $this->setBrowser($browserName); 
                $this->setVersion($version); 
 
                $this->setMobile($isMobile); 
                $this->setRobot($isRobot); 
 
                return true; 
            } 
        } 
 
        return false; 
    } 
 
    /** 
     * Detect the user environment from the details in the user agent string. 
     * @access protected 
     */ 
    protected function detect() 
    { 
        $this->checkBrowsers(); 
        $this->checkPlatform(); //Check the platform after the browser since some platforms can change the mobile value 
        $this->checkForAol(); 
    } 
 
    /** 
     * Clean a version string from unwanted characters. 
     * @access protected 
     * @param string $version The version string to clean. 
     * @return string Returns the cleaned version number string. 
     */ 
    protected function cleanVersion($version) 
    { 
        //Clear anything that is in parentheses (and the parentheses themselves) - will clear started but unclosed ones too 
        $cleanVer = preg_replace('/\([^)]+\)?/', '', $version); 
        //Replace with a space any character which is NOT an alphanumeric, dot (.), hyphen (-), underscore (_) or space 
        $cleanVer = preg_replace('/[^0-9.a-zA-Z_ -]/', ' ', $cleanVer); 
        //Remove trailing and leading spaces 
        $cleanVer = trim($cleanVer); 
        //Remove double spaces if any 
        while (strpos($cleanVer, '  ') !== false) { 
            $cleanVer = str_replace('  ', ' ', $cleanVer); 
        } 
 
        return $cleanVer; 
    } 
 
    /** 
     * Get the integer value of a string variable. 
     * @access protected 
     * @param string $intStr The scalar value being converted to an integer. 
     * @return int The integer value of $intStr on success, or 0 on failure. 
     */ 
    protected function parseInt($intStr) 
    { 
        return intval($intStr, 10); 
    } 
 
    /** 
     * Reset all the properties of the class. 
     * @access protected 
     */ 
    protected function reset() 
    { 
        $this->_agent = ''; 
        $this->_aolVersion = ''; 
        $this->_browserName = self::BROWSER_UNKNOWN; 
        $this->_compatibilityViewName = ''; 
        $this->_compatibilityViewVer = ''; 
        $this->_is64bit = false; 
        $this->_isAol = false; 
        $this->_isMobile = false; 
        $this->_isRobot = false; 
        $this->_platform = self::PLATFORM_UNKNOWN; 
        $this->_platformVersion = self::PLATFORM_VERSION_UNKNOWN; 
        $this->_version = self::VERSION_UNKNOWN; 
    } 
 
    /** 
     * Convert a Safari build number to a Safari version number. 
     * @access protected 
     * @param string $version A string representing the version number. 
     * @link http://web.archive.org/web/20080514173941/http://developer.apple.com/internet/safari/uamatrix.html 
     * @return string Returns the Safari version string. If the version can't be determined, an empty string is 
     * returned. 
     */ 
    protected function safariBuildToSafariVer($version) 
    { 
        $verParts = explode('.', $version); 
 
        //We need a 3 parts version (version 2 will becomes 2.0.0) 
        while (count($verParts) < 3) { 
            $verParts[] = 0; 
        } 
        foreach ($verParts as $i => $currPart) { 
            $verParts[$i] = $this->parseInt($currPart); 
        } 
 
        switch ($verParts[0]) { 
            case 419: $result = '2.0.4'; 
                break; 
            case 417: $result = '2.0.3'; 
                break; 
            case 416: $result = '2.0.2'; 
                break; 
 
            case 412: 
                if ($verParts[1] >= 5) { 
                    $result = '2.0.1'; 
                } else { 
                    $result = '2.0'; 
                } 
                break; 
 
            case 312: 
                if ($verParts[1] >= 5) { 
                    $result = '1.3.2'; 
                } else { 
                    if ($verParts[1] >= 3) { 
                        $result = '1.3.1'; 
                    } else { 
                        $result = '1.3'; 
                    } 
                } 
                break; 
 
            case 125: 
                if ($verParts[1] >= 11) { 
                    $result = '1.2.4'; 
                } else { 
                    if ($verParts[1] >= 9) { 
                        $result = '1.2.3'; 
                    } else { 
                        if ($verParts[1] >= 7) { 
                            $result = '1.2.2'; 
                        } else { 
                            $result = '1.2'; 
                        } 
                    } 
                } 
                break; 
 
            case 100: 
                if ($verParts[1] >= 1) { 
                    $result = '1.1.1'; 
                } else { 
                    $result = '1.1'; 
                } 
                break; 
 
            case 85: 
                if ($verParts[1] >= 8) { 
                    $result = '1.0.3'; 
                } else { 
                    if ($verParts[1] >= 7) { 
                        $result = '1.0.2'; 
                    } else { 
                        $result = '1.0'; 
                    } 
                } 
                break; 
 
            case 73: $result = '0.9'; 
                break; 
            case 51: $result = '0.8.1'; 
                break; 
            case 48: $result = '0.8'; 
                break; 
 
            default: $result = ''; 
        } 
 
        return $result; 
    } 
 
    /** 
     * Set if the browser is executed from a 64-bit platform. 
     * @access protected 
     * @param boolean $is64bit Value that tells if the browser is executed from a 64-bit platform. 
     */ 
    protected function set64bit($is64bit) 
    { 
        $this->_is64bit = $is64bit == true; 
    } 
 
    /** 
     * Set the browser to be from AOL or not. 
     * @access protected 
     * @param boolean $isAol Value that tells if the browser is AOL or not. 
     */ 
    protected function setAol($isAol) 
    { 
        $this->_isAol = $isAol == true; 
    } 
 
    /** 
     * Set the version of AOL. 
     * @access protected 
     * @param string $version The version of AOL (will be cleaned). 
     */ 
    protected function setAolVersion($version) 
    { 
        $cleanVer = $this->cleanVersion($version); 
 
        $this->_aolVersion = $cleanVer; 
    } 
 
    /** 
     * Set the name of the browser. 
     * @access protected 
     * @param string $browserName The name of the browser. 
     */ 
    protected function setBrowser($browserName) 
    { 
        $this->_browserName = $browserName; 
    } 
 
    /** 
     * Set the browser to be from a mobile device or not. 
     * @access protected 
     * @param boolean $isMobile Value that tells if the browser is on a mobile device or not. 
     */ 
    protected function setMobile($isMobile = true) 
    { 
        $this->_isMobile = $isMobile == true; 
    } 
 
    /** 
     * Set the platform on which the browser is on. 
     * @access protected 
     * @param string $platform The name of the platform. 
     */ 
    protected function setPlatform($platform) 
    { 
        $this->_platform = $platform; 
    } 
 
    /** 
     * Set the platform version on which the browser is on. 
     * @access protected 
     * @param string $platformVer The version numbers of the platform. 
     */ 
    protected function setPlatformVersion($platformVer) 
    { 
        $this->_platformVersion = $platformVer; 
    } 
 
    /** 
     * Set the browser to be a robot (crawler) or not. 
     * @access protected 
     * @param boolean $isRobot Value that tells if the browser is a robot or not. 
     */ 
    protected function setRobot($isRobot = true) 
    { 
        $this->_isRobot = $isRobot == true; 
    } 
 
    /** 
     * Set the version of the browser. 
     * @access protected 
     * @param string $version The version of the browser. 
     */ 
    protected function setVersion($version) 
    { 
        $cleanVer = $this->cleanVersion($version); 
 
        if ($cleanVer == '') { 
            $this->_version = self::VERSION_UNKNOWN; 
        } else { 
            $this->_version = $cleanVer; 
        } 
    } 
 
    /** 
     * Convert a WebKit build number to a Safari version number. 
     * @access protected 
     * @param string $version A string representing the version number. 
     * @link http://web.archive.org/web/20080514173941/http://developer.apple.com/internet/safari/uamatrix.html 
     * @return string Returns the Safari version string. If the version can't be determined, an empty string is 
     * returned. 
     */ 
    protected function webKitBuildToSafariVer($version) 
    { 
        $verParts = explode('.', $version); 
 
        //We need a 3 parts version (version 2 will becomes 2.0.0) 
        while (count($verParts) < 3) { 
            $verParts[] = 0; 
        } 
        foreach ($verParts as $i => $currPart) { 
            $verParts[$i] = $this->parseInt($currPart); 
        } 
 
        switch ($verParts[0]) { 
            case 419: $result = '2.0.4'; 
                break; 
 
            case 418: 
                if ($verParts[1] >= 8) { 
                    $result = '2.0.4'; 
                } else { 
                    $result = '2.0.3'; 
                } 
                break; 
 
            case 417: $result = '2.0.3'; 
                break; 
 
            case 416: $result = '2.0.2'; 
                break; 
 
            case 412: 
                if ($verParts[1] >= 7) { 
                    $result = '2.0.1'; 
                } else { 
                    $result = '2.0'; 
                } 
                break; 
 
            case 312: 
                if ($verParts[1] >= 8) { 
                    $result = '1.3.2'; 
                } else { 
                    if ($verParts[1] >= 5) { 
                        $result = '1.3.1'; 
                    } else { 
                        $result = '1.3'; 
                    } 
                } 
                break; 
 
            case 125: 
                if ($this->compareVersions('5.4', $verParts[1] . '.' . $verParts[2]) == -1) { 
                    $result = '1.2.4'; //125.5.5+ 
                } else { 
                    if ($verParts[1] >= 4) { 
                        $result = '1.2.3'; 
                    } else { 
                        if ($verParts[1] >= 2) { 
                            $result = '1.2.2'; 
                        } else { 
                            $result = '1.2'; 
                        } 
                    } 
                } 
                break; 
 
            //WebKit 100 can be either Safari 1.1 (Safari build 100) or 1.1.1 (Safari build 100.1) 
            //for this reason, check the Safari build before the WebKit build. 
            case 100: $result = '1.1.1'; 
                break; 
 
            case 85: 
                if ($verParts[1] >= 8) { 
                    $result = '1.0.3'; 
                } else { 
                    if ($verParts[1] >= 7) { 
                        //WebKit 85.7 can be either Safari 1.0 (Safari build 85.5) or 1.0.2 (Safari build 85.7) 
                        //for this reason, check the Safari build before the WebKit build. 
                        $result = '1.0.2'; 
                    } else { 
                        $result = '1.0'; 
                    } 
                } 
                break; 
 
            case 73: $result = '0.9'; 
                break; 
            case 51: $result = '0.8.1'; 
                break; 
            case 48: $result = '0.8'; 
                break; 
 
            default: $result = ''; 
        } 
 
        return $result; 
    } 
 
    /** 
     * Convert the Windows NT family version numbers to the operating system name. For instance '5.1' returns 
     * 'Windows XP'. 
     * @access protected 
     * @param string $winVer The Windows NT family version numbers as a string. 
     * @param boolean $returnServerFlavor Since some Windows NT versions have the same values, this flag determines if 
     * the Server flavor is returned or not. For instance Windows 8.1 and Windows Server 2012 R2 both use version 6.3. 
     * @return string The operating system name or the constant PLATFORM_VERSION_UNKNOWN if nothing match the version 
     * numbers. 
     */ 
    protected function windowsNTVerToStr($winVer, $returnServerFlavor = false) 
    { 
        //https://en.wikipedia.org/wiki/List_of_Microsoft_Windows_versions 
 
        $cleanWinVer = explode('.', $winVer); 
        while (count($cleanWinVer) > 2) { 
            array_pop($cleanWinVer); 
        } 
        $cleanWinVer = implode('.', $cleanWinVer); 
 
        if ($this->compareVersions($cleanWinVer, '11') >= 0) { 
            //Future versions of Windows 
            return self::PLATFORM_WINDOWS . ' ' . $winVer; 
        } else if ($this->compareVersions($cleanWinVer, '10') >= 0) { 
            //Current version of Windows 
            return $returnServerFlavor ? (self::PLATFORM_WINDOWS . ' Server 2016') : (self::PLATFORM_WINDOWS . ' 10'); 
        } else if ($this->compareVersions($cleanWinVer, '7') < 0) { 
            if ($this->compareVersions($cleanWinVer, '6.3') == 0) { 
                return $returnServerFlavor ? (self::PLATFORM_WINDOWS . ' Server 2012 R2') : (self::PLATFORM_WINDOWS . ' 8.1'); 
            } else if ($this->compareVersions($cleanWinVer, '6.2') == 0) { 
                return $returnServerFlavor ? (self::PLATFORM_WINDOWS . ' Server 2012') : (self::PLATFORM_WINDOWS . ' 8'); 
            } else if ($this->compareVersions($cleanWinVer, '6.1') == 0) { 
                return $returnServerFlavor ? (self::PLATFORM_WINDOWS . ' Server 2008 R2') : (self::PLATFORM_WINDOWS . ' 7'); 
            } else if ($this->compareVersions($cleanWinVer, '6') == 0) { 
                return $returnServerFlavor ? (self::PLATFORM_WINDOWS . ' Server 2008') : (self::PLATFORM_WINDOWS . ' Vista'); 
            } else if ($this->compareVersions($cleanWinVer, '5.2') == 0) { 
                return $returnServerFlavor ? (self::PLATFORM_WINDOWS . ' Server 2003 / ' . self::PLATFORM_WINDOWS . ' Server 2003 R2') : (self::PLATFORM_WINDOWS . ' XP x64 Edition'); 
            } else if ($this->compareVersions($cleanWinVer, '5.1') == 0) { 
                return self::PLATFORM_WINDOWS . ' XP'; 
            } else if ($this->compareVersions($cleanWinVer, '5') == 0) { 
                return self::PLATFORM_WINDOWS . ' 2000'; 
            } else if ($this->compareVersions($cleanWinVer, '5') < 0 && $this->compareVersions($cleanWinVer, '3') >= 0) { 
                return self::PLATFORM_WINDOWS . ' NT ' . $winVer; 
            } 
        } 
 
        return self::PLATFORM_VERSION_UNKNOWN; //Invalid Windows NT version 
    } 
 
    /** 
     * Convert the Windows 3.x & 9x family version numbers to the operating system name. For instance '4.10.1998' 
     * returns 'Windows 98'. 
     * @access protected 
     * @param string $winVer The Windows 3.x or 9x family version numbers as a string. 
     * @return string The operating system name or the constant PLATFORM_VERSION_UNKNOWN if nothing match the version 
     * numbers. 
     */ 
    protected function windowsVerToStr($winVer) 
    { 
        //https://support.microsoft.com/en-us/kb/158238 
 
        if ($this->compareVersions($winVer, '4.90') >= 0 && $this->compareVersions($winVer, '4.91') < 0) { 
            return self::PLATFORM_WINDOWS . ' Me'; //Normally range from 4.90.3000 to 4.90.3000A 
        } else if ($this->compareVersions($winVer, '4.10') >= 0 && $this->compareVersions($winVer, '4.11') < 0) { 
            return self::PLATFORM_WINDOWS . ' 98'; //Normally range from 4.10.1998 to 4.10.2222B 
        } else if ($this->compareVersions($winVer, '4') >= 0 && $this->compareVersions($winVer, '4.04') < 0) { 
            return self::PLATFORM_WINDOWS . ' 95'; //Normally range from 4.00.950 to 4.03.1214 
        } else if ($this->compareVersions($winVer, '3.1') == 0 || $this->compareVersions($winVer, '3.11') == 0) { 
            return self::PLATFORM_WINDOWS . ' ' . $winVer; 
        } else { 
            return self::PLATFORM_VERSION_UNKNOWN; //Invalid Windows version 
        } 
    } 
} ?>

Did this file decode correctly?

Original Code

<?php

/**
 * Browser detection class file.
 * This file contains everything required to use the BrowserDetection class.
 *
 * This program is free software; you can redistribute it and/or modify it under the terms of the GNU Lesser General
 * Public License as published by the Free Software Foundation; either version 3 of the License, or (at your option) any
 * later version (if any).
 *
 * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied
 * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more
 * details at: http://www.gnu.org/licenses/lgpl.html
 *
 * @package Browser_Detection
 * @version 2.3.0
 * @last-modified February 11, 2016
 * @author Alexandre Valiquette
 * @copyright Copyright (c) 2016, Wolfcast
 * @link http://wolfcast.com/
 */


/**
 * The BrowserDetection class facilitates the identification of the user's environment such as Web browser, version,
 * platform or if it's a mobile device.
 *
 * Typical usage:
 *
 * $browser = new BrowserDetection();
 * if ($browser->getName() == BrowserDetection::BROWSER_FIREFOX &&
 *     $browser->compareVersions($browser->getVersion(), '5.0') >= 0) {
 *     echo 'You are using FireFox version 5 or greater.';
 * }
 *
 * The class is an updated version of Chris Schuld's Browser class version 1.9 which is unmaintained since August 20th,
 * 2010. Chris' class was based on the original work from Gary White.
 *
 * Updates:
 *
 * 2016-02-11: Version 2.3.0
 *  + WARNING! Breaking change: public method getBrowser() is renamed to getName().
 *  + WARNING! Breaking change: changed the compareVersions() return values to be more in line with other libraries.
 *  + You can now get the exact platform version (name or version numbers) on which the browser is run on with
 *    getPlatformVersion(). Only working with Windows operating systems at the moment.
 *  + You can now determine if the browser is executed from a 64-bit platform with is64bitPlatform().
 *  + Better detection of mobile platform for Googlebot.
 *
 * 2016-01-04: Version 2.2.0
 *  + Added support for Microsoft Edge.
 *
 * 2014-12-30: Version 2.1.2
 *  + Better detection of Opera.
 *
 * 2014-07-11: Version 2.1.1
 *  + Better detection of mobile devices and platforms.
 *
 * 2014-06-04: Version 2.1
 *  + Added support for IE 11+.
 *
 * 2013-05-27: Version 2.0 which is (almost) a complete rewrite based on Chris Schuld's Browser class version 1.9 plus
 * changes below.
 *  + Added support for Opera Mobile
 *  + Added support for the Windows Phone (formerly Windows Mobile) platform
 *  + Added support for BlackBerry Tablet OS and BlackBerry 10
 *  + Added support for the Symbian platform
 *  + Added support for Bingbot
 *  + Added support for the Yahoo! Multimedia crawler
 *  + Removed iPhone/iPad/iPod browsers since there are not browsers but platforms - test them with getPlatform()
 *  + Removed support for Shiretoko (Firefox 3.5 alpha/beta) and MSN Browser
 *  + Merged Nokia and Nokia S60
 *  + Updated some deprecated browser names
 *  + Many public methods are now protected
 *  + Documentation updated
 *
 * 2010-07-04:
 *  + Added detection of IE compatibility view - test with getIECompatibilityView()
 *  + Added support for all (deprecated) Netscape versions
 *  + Added support for Safari < 3.0
 *  + Better Firefox version parsing
 *  + Better Opera version parsing
 *  + Better Mozilla detection
 *
 * @package Browser_Detection
 * @version 2.3.0
 * @last-modified February 11, 2016
 * @author Alexandre Valiquette, Chris Schuld, Gary White
 * @copyright Copyright (c) 2016, Wolfcast
 * @license http://www.gnu.org/licenses/lgpl.html
 * @link http://wolfcast.com/
 * @link http://wolfcast.com/open-source/browser-detection/tutorial.php
 * @link http://chrisschuld.com/
 * @link http://www.apptools.com/phptools/browser/
 */
class BrowserDetection
{

    /**#@+
     * Constant for the name of the Web browser.
     */
    const BROWSER_AMAYA = 'Amaya';
    const BROWSER_ANDROID = 'Android';
    const BROWSER_BINGBOT = 'Bingbot';
    const BROWSER_BLACKBERRY = 'BlackBerry';
    const BROWSER_CHROME = 'Chrome';
    const BROWSER_EDGE = 'Edge';
    const BROWSER_FIREBIRD = 'Firebird';
    const BROWSER_FIREFOX = 'Firefox';
    const BROWSER_GALEON = 'Galeon';
    const BROWSER_GOOGLEBOT = 'Googlebot';
    const BROWSER_ICAB = 'iCab';
    const BROWSER_ICECAT = 'GNU IceCat';
    const BROWSER_ICEWEASEL = 'GNU IceWeasel';
    const BROWSER_IE = 'Internet Explorer';
    const BROWSER_IE_MOBILE = 'Internet Explorer Mobile';
    const BROWSER_KONQUEROR = 'Konqueror';
    const BROWSER_LYNX = 'Lynx';
    const BROWSER_MOZILLA = 'Mozilla';
    const BROWSER_MSNBOT = 'MSNBot';
    const BROWSER_MSNTV = 'MSN TV';
    const BROWSER_NETPOSITIVE = 'NetPositive';
    const BROWSER_NETSCAPE = 'Netscape';
    const BROWSER_NOKIA = 'Nokia Browser';
    const BROWSER_OMNIWEB = 'OmniWeb';
    const BROWSER_OPERA = 'Opera';
    const BROWSER_OPERA_MINI = 'Opera Mini';
    const BROWSER_OPERA_MOBILE = 'Opera Mobile';
    const BROWSER_PHOENIX = 'Phoenix';
    const BROWSER_SAFARI = 'Safari';
    const BROWSER_SLURP = 'Yahoo! Slurp';
    const BROWSER_TABLET_OS = 'BlackBerry Tablet OS';
    const BROWSER_UNKNOWN = 'unknown';
    const BROWSER_W3CVALIDATOR = 'W3C Validator';
    const BROWSER_YAHOO_MM = 'Yahoo! Multimedia';
    /**#@-*/

    /**#@+
     * Constant for the name of the platform of the Web browser.
     */
    const PLATFORM_ANDROID = 'Android';
    const PLATFORM_BEOS = 'BeOS';
    const PLATFORM_BLACKBERRY = 'BlackBerry';
    const PLATFORM_FREEBSD = 'FreeBSD';
    const PLATFORM_IPAD = 'iPad';
    const PLATFORM_IPHONE = 'iPhone';
    const PLATFORM_IPOD = 'iPod';
    const PLATFORM_LINUX = 'Linux';
    const PLATFORM_MACINTOSH = 'Macintosh';
    const PLATFORM_NETBSD = 'NetBSD';
    const PLATFORM_NOKIA = 'Nokia';
    const PLATFORM_OPENBSD = 'OpenBSD';
    const PLATFORM_OPENSOLARIS = 'OpenSolaris';
    const PLATFORM_OS2 = 'OS/2';
    const PLATFORM_SUNOS = 'SunOS';
    const PLATFORM_SYMBIAN = 'Symbian';
    const PLATFORM_UNKNOWN = 'unknown';
    const PLATFORM_VERSION_UNKNOWN = 'unknown';
    const PLATFORM_WINDOWS = 'Windows';
    const PLATFORM_WINDOWS_CE = 'Windows CE';
    const PLATFORM_WINDOWS_PHONE = 'Windows Phone';
    /**#@-*/

    /**
     * Version unknown constant.
     */
    const VERSION_UNKNOWN = 'unknown';


    /**
     * @var string
     * @access private
     */
    private $_agent = '';

    /**
     * @var string
     * @access private
     */
    private $_aolVersion = '';

    /**
     * @var string
     * @access private
     */
    private $_browserName = '';

    /**
     * @var string
     * @access private
     */
    private $_compatibilityViewName = '';

    /**
     * @var string
     * @access private
     */
    private $_compatibilityViewVer = '';

    /**
     * @var boolean
     * @access private
     */
    private $_is64bit = false;

    /**
     * @var boolean
     * @access private
     */
    private $_isAol = false;

    /**
     * @var boolean
     * @access private
     */
    private $_isMobile = false;

    /**
     * @var boolean
     * @access private
     */
    private $_isRobot = false;

    /**
     * @var string
     * @access private
     */
    private $_platform = '';

    /**
     * @var string
     * @access private
     */
    private $_platformVersion = '';

    /**
     * @var string
     * @access private
     */
    private $_version = '';


    //--- MAGIC METHODS ------------------------------------------------------------------------------------------------


    /**
     * BrowserDetection class constructor.
     * @param string $useragent The user agent to work with. Leave empty for the current user agent (contained in
     * $_SERVER['HTTP_USER_AGENT']).
     */
    public function __construct($useragent = '')
    {
        $this->setUserAgent($useragent);
    }

    /**
     * Determine how the class will react when it is treated like a string.
     * @return string Returns an HTML formatted string with a summary of the browser informations.
     */
    public function __toString()
    {
        $result = '';

        $values = array();
        $values[] = array('label' => 'User agent', 'value' => $this->getUserAgent());
        $values[] = array('label' => 'Browser name', 'value' => $this->getName());
        $values[] = array('label' => 'Browser version', 'value' => $this->getVersion());
        $values[] = array('label' => 'Platform family', 'value' => $this->getPlatform());
        $values[] = array('label' => 'Platform version', 'value' => $this->getPlatformVersion(true));
        $values[] = array('label' => 'Platform version name', 'value' => $this->getPlatformVersion());
        $values[] = array('label' => 'Platform is 64-bit', 'value' => $this->is64bitPlatform() ? 'true' : 'false');
        $values[] = array('label' => 'Is mobile', 'value' => $this->isMobile() ? 'true' : 'false');
        $values[] = array('label' => 'Is robot', 'value' => $this->isRobot() ? 'true' : 'false');
        $values[] = array('label' => 'IE is in compatibility view', 'value' => $this->isInIECompatibilityView() ? 'true' : 'false');
        $values[] = array('label' => 'Emulated IE version', 'value' => $this->isInIECompatibilityView() ? $this->getIECompatibilityView() : 'Not applicable');
        $values[] = array('label' => 'Is Chrome Frame', 'value' => $this->isChromeFrame() ? 'true' : 'false');
        $values[] = array('label' => 'Is AOL optimized', 'value' => $this->isAol() ? 'true' : 'false');
        $values[] = array('label' => 'AOL version', 'value' => $this->isAol() ? $this->getAolVersion() : 'Not applicable');

        foreach ($values as $currVal) {
            $result .= '<strong>' . htmlspecialchars($currVal['label'], ENT_NOQUOTES) . ':</strong> ' . $currVal['value'] . '<br />' . PHP_EOL;
        }

        return $result;
    }


    //--- PUBLIC MEMBERS -----------------------------------------------------------------------------------------------


    /**
     * Compare two version number strings.
     * @param string $sourceVer The source version number.
     * @param string $compareVer The version number to compare with the source version number.
     * @return int Returns -1 if $sourceVer < $compareVer, 0 if $sourceVer == $compareVer or 1 if $sourceVer >
     * $compareVer.
     */
    public function compareVersions($sourceVer, $compareVer)
    {
        $sourceVer = explode('.', $sourceVer);
        foreach ($sourceVer as $k => $v) {
            $sourceVer[$k] = $this->parseInt($v);
        }

        $compareVer = explode('.', $compareVer);
        foreach ($compareVer as $k => $v) {
            $compareVer[$k] = $this->parseInt($v);
        }

        if (count($sourceVer) != count($compareVer)) {
            if (count($sourceVer) > count($compareVer)) {
                for ($i = count($compareVer); $i < count($sourceVer); $i++) {
                    $compareVer[$i] = 0;
                }
            } else {
                for ($i = count($sourceVer); $i < count($compareVer); $i++) {
                    $sourceVer[$i] = 0;
                }
            }
        }

        foreach ($sourceVer as $i => $srcVerPart) {
            if ($srcVerPart > $compareVer[$i]) {
                return 1;
            } else {
                if ($srcVerPart < $compareVer[$i]) {
                    return -1;
                }
            }
        }

        return 0;
    }

    /**
     * Get the version of AOL (if any). AOL releases "optimized" Internet Explorer and Firefox versions. In the making
     * they add their version number in the user agent string of these browsers.
     * @return string Returns the version of AOL or an empty string if no AOL version was found.
     */
    public function getAolVersion()
    {
        return $this->_aolVersion;
    }

    /**
     * Get the name of the browser. All of the return values are class constants. You can compare them like this:
     * $myBrowserInstance->getName() == BrowserDetection::BROWSER_FIREFOX.
     * @return string Returns the name of the browser.
     */
    public function getName()
    {
        return $this->_browserName;
    }

    /**
     * Get the name and version of the browser emulated in the compatibility view mode (if any). Since Internet
     * Explorer 8, IE can be put in compatibility mode to make websites that were created for older browsers, especially
     * IE 6 and 7, look better in IE 8+ which renders web pages closer to the standards and thus differently from those
     * older versions of IE.
     * @param boolean $asArray Determines if the return value must be an array (true) or a string (false).
     * @return mixed If a string was requested, the function returns the name and version of the browser emulated in the
     * compatibility view mode or an empty string if the browser is not in compatibility view mode. If an array was
     * requested, an array with the keys 'browser' and 'version' is returned.
     */
    public function getIECompatibilityView($asArray = false)
    {
        if ($asArray) {
            return array('browser' => $this->_compatibilityViewName, 'version' => $this->_compatibilityViewVer);
        } else {
            return trim($this->_compatibilityViewName . ' ' . $this->_compatibilityViewVer);
        }
    }

    /**
     * Get the name of the platform family on which the browser is run on (such as Windows, Apple, iPhone, etc.). All of
     * the return values are class constants. You can compare them like this:
     * $myBrowserInstance->getPlatform() == BrowserDetection::PLATFORM_ANDROID.
     * @return string Returns the name of the platform or BrowserDetection::PLATFORM_UNKNOWN if unknown.
     */
    public function getPlatform()
    {
        return $this->_platform;
    }

    /**
     * Get the platform version on which the browser is run on. It can be returned as a string number like 'NT 6.3' or
     * as a name like 'Windows 8.1'. When returning version string numbers for Windows NT OS families the number is
     * prefixed by 'NT ' to differentiate from older Windows 3.x & 9x release. At the moment only the Windows operating
     * systems is supported.
     * @param boolean $returnVersionNumbers Determines if the return value must be versions numbers as a string (true)
     * or the version name (false).
     * @param boolean $returnServerFlavor Since some Windows NT versions have the same values, this flag determines if
     * the Server flavor is returned or not. For instance Windows 8.1 and Windows Server 2012 R2 both use version 6.3.
     * @return string Returns the version name/version numbers of the platform or the constant PLATFORM_VERSION_UNKNOWN
     * if unknown.
     */
    public function getPlatformVersion($returnVersionNumbers = false, $returnServerFlavor = false)
    {
        if ($this->_platformVersion == self::PLATFORM_VERSION_UNKNOWN || $this->_platformVersion == '') {
            return self::PLATFORM_VERSION_UNKNOWN;
        }

        if ($returnVersionNumbers) {
            return $this->_platformVersion;
        } else {
            switch ($this->getPlatform()) {
                case self::PLATFORM_WINDOWS:
                    if (substr($this->_platformVersion, 0, 3) == 'NT ') {
                        return $this->windowsNTVerToStr(substr($this->_platformVersion, 3), $returnServerFlavor);
                    } else {
                        return $this->windowsVerToStr($this->_platformVersion);
                    }
                break;

                default: return self::PLATFORM_VERSION_UNKNOWN;
            }
        }
    }

    /**
     * Get the user agent value used by the class to determine the browser details.
     * @return string The user agent string.
     */
    public function getUserAgent()
    {
        return $this->_agent;
    }

    /**
     * Get the version of the browser.
     * @return string Returns the version of the browser or BrowserDetection::VERSION_UNKNOWN if unknown.
     */
    public function getVersion()
    {
        return $this->_version;
    }

    /**
     * Determine if the browser is executed from a 64-bit platform. Keep in mind that not all platforms/browsers report
     * this and the result may not always be accurate.
     * @return boolean Returns true if the browser is executed from a 64-bit platform.
     */
    public function is64bitPlatform()
    {
        return $this->_is64bit;
    }

    /**
     * Determine if the browser is from AOL. AOL releases "optimized" Internet Explorer and Firefox versions. In the
     * making they add their details in the user agent string of these browsers.
     * @return boolean Returns true if the browser is from AOL, false otherwise.
     */
    public function isAol()
    {
        return $this->_isAol;
    }

    /**
     * Determine if the browser runs Google Chrome Frame (it's a plug-in designed for Internet Explorer 6+ based on the
     * open-source Chromium project - it's like a Chrome browser within IE).
     * @return boolean Returns true if the browser is using Google Chrome Frame, false otherwise.
     */
    public function isChromeFrame()
    {
        return stripos($this->_agent, 'chromeframe') !== false;
    }

    /**
     * Determine if the browser is in compatibility view or not. Since Internet Explorer 8, IE can be put in
     * compatibility mode to make websites that were created for older browsers, especially IE 6 and 7, look better in
     * IE 8+ which renders web pages closer to the standards and thus differently from those older versions of IE.
     * @return boolean Returns true if the browser is in compatibility view, false otherwise.
     */
    public function isInIECompatibilityView()
    {
        return ($this->_compatibilityViewName != '') || ($this->_compatibilityViewVer != '');
    }

    /**
     * Determine if the browser is from a mobile device or not.
     * @return boolean Returns true if the browser is from a mobile device, false otherwise.
     */
    public function isMobile()
    {
        return $this->_isMobile;
    }

    /**
     * Determine if the browser is a robot (Googlebot, Bingbot, Yahoo! Slurp...) or not.
     * @return boolean Returns true if the browser is a robot, false otherwise.
     */
    public function isRobot()
    {
        return $this->_isRobot;
    }

    /**
     * Set the user agent to use with the class.
     * @param string $agentString The value of the user agent. If an empty string is sent (default),
     * $_SERVER['HTTP_USER_AGENT'] will be used.
     */
    public function setUserAgent($agentString = '')
    {
        if (!is_string($agentString) || trim($agentString) == '') {
            if (array_key_exists('HTTP_USER_AGENT', $_SERVER) && is_string($_SERVER['HTTP_USER_AGENT'])) {
                $agentString = $_SERVER['HTTP_USER_AGENT'];
            } else {
                $agentString = '';
            }
        }

        $this->reset();
        $this->_agent = $agentString;
        $this->detect();
    }


    //--- PROTECTED MEMBERS --------------------------------------------------------------------------------------------


    /**
     * Determine if the browser is the Amaya Web editor or not.
     * @access protected
     * @link http://www.w3.org/Amaya/
     * @return boolean Returns true if the browser is Amaya, false otherwise.
     */
    protected function checkBrowserAmaya()
    {
        return $this->checkSimpleBrowserUA('amaya', $this->_agent, self::BROWSER_AMAYA);
    }

    /**
     * Determine if the browser is the Android browser (based on the WebKit layout engine and coupled with Chrome's
     * JavaScript engine) or not.
     * @access protected
     * @return boolean Returns true if the browser is the Android browser, false otherwise.
     */
    protected function checkBrowserAndroid()
    {
        //Android don't use the standard "Android/1.0", it uses "Android 1.0;" instead
        return $this->checkSimpleBrowserUA('Android', $this->_agent, self::BROWSER_ANDROID, true);
    }

    /**
     * Determine if the browser is the Bingbot crawler or not.
     * @access protected
     * @link http://www.bing.com/webmaster/help/which-crawlers-does-bing-use-8c184ec0
     * @return boolean Returns true if the browser is Bingbot, false otherwise.
     */
    protected function checkBrowserBingbot()
    {
        return $this->checkSimpleBrowserUA('bingbot', $this->_agent, self::BROWSER_BINGBOT, false, true);
    }

    /**
     * Determine if the browser is the BlackBerry browser or not.
     * @access protected
     * @link http://supportforums.blackberry.com/t5/Web-and-WebWorks-Development/How-to-detect-the-BlackBerry-Browser/ta-p/559862
     * @return boolean Returns true if the browser is the BlackBerry browser, false otherwise.
     */
    protected function checkBrowserBlackBerry()
    {
        $found = false;

        //Tablet OS check
        if ($this->checkSimpleBrowserUA('RIM Tablet OS', $this->_agent, self::BROWSER_TABLET_OS, true)) {
            return true;
        }

        //Version 6, 7 & 10 check (versions 8 & 9 does not exists)
        if ($this->checkBrowserUAWithVersion(array('BlackBerry', 'BB10'), $this->_agent, self::BROWSER_BLACKBERRY, true)) {
            if ($this->getVersion() == self::VERSION_UNKNOWN) {
                $found = true;
            } else {
                return true;
            }
        }

        //Version 4.2 to 5.0 check
        if ($this->checkSimpleBrowserUA('BlackBerry', $this->_agent, self::BROWSER_BLACKBERRY, true)) {
            if ($this->getVersion() == self::VERSION_UNKNOWN) {
                $found = true;
            } else {
                return true;
            }
        }

        return $found;
    }

    /**
     * Determine if the browser is Chrome or not.
     * @access protected
     * @link http://www.google.com/chrome/
     * @return boolean Returns true if the browser is Chrome, false otherwise.
     */
    protected function checkBrowserChrome()
    {
        return $this->checkSimpleBrowserUA('Chrome', $this->_agent, self::BROWSER_CHROME);
    }

    /**
     * Determine if the browser is Edge or not.
     * @access protected
     * @return boolean Returns true if the browser is Edge, false otherwise.
     */
    protected function checkBrowserEdge()
    {
        return $this->checkSimpleBrowserUA('Edge', $this->_agent, self::BROWSER_EDGE);
    }

    /**
     * Determine if the browser is Firebird or not. Firebird was the name of Firefox from version 0.6 to 0.7.1.
     * @access protected
     * @return boolean Returns true if the browser is Firebird, false otherwise.
     */
    protected function checkBrowserFirebird()
    {
        return $this->checkSimpleBrowserUA('Firebird', $this->_agent, self::BROWSER_FIREBIRD);
    }

    /**
     * Determine if the browser is Firefox or not.
     * @access protected
     * @link http://www.mozilla.org/en-US/firefox/new/
     * @return boolean Returns true if the browser is Firefox, false otherwise.
     */
    protected function checkBrowserFirefox()
    {
        //Safari heavily matches with Firefox, ensure that Safari is filtered out...
        if (preg_match('/.*Firefox[ (\/]*([a-z0-9.-]*)/i', $this->_agent, $matches) &&
                stripos($this->_agent, 'Safari') === false) {
            $this->setBrowser(self::BROWSER_FIREFOX);
            $this->setVersion($matches[1]);
            $this->setMobile(false);
            $this->setRobot(false);

            return true;
        }

        return false;
    }

    /**
     * Determine if the browser is Galeon or not. The browser was discontinued on September 27, 2008.
     * @access protected
     * @link http://en.wikipedia.org/wiki/Galeon
     * @return boolean Returns true if the browser is Galeon, false otherwise.
     */
    protected function checkBrowserGaleon()
    {
        return $this->checkSimpleBrowserUA('Galeon', $this->_agent, self::BROWSER_GALEON);
    }

    /**
     * Determine if the browser is the Googlebot crawler or not.
     * @access protected
     * @return boolean Returns true if the browser is Googlebot, false otherwise.
     */
    protected function checkBrowserGooglebot()
    {
        if ($this->checkSimpleBrowserUA('Googlebot', $this->_agent, self::BROWSER_GOOGLEBOT, false, true)) {
            if (strpos(strtolower($this->_agent), 'googlebot-mobile') !== false) {
                $this->setMobile(true);
            }

            return true;
        }

        return false;
    }

    /**
     * Determine if the browser is iCab or not.
     * @access protected
     * @link http://www.icab.de/
     * @return boolean Returns true if the browser is iCab, false otherwise.
     */
    protected function checkBrowserIcab()
    {
        //Some (early) iCab versions don't use the standard "iCab/1.0", they uses "iCab 1.0;" instead
        return $this->checkSimpleBrowserUA('iCab', $this->_agent, self::BROWSER_ICAB);
    }

    /**
     * Determine if the browser is GNU IceCat (formerly known as GNU IceWeasel) or not.
     * @access protected
     * @link http://www.gnu.org/software/gnuzilla/
     * @return boolean Returns true if the browser is GNU IceCat, false otherwise.
     */
    protected function checkBrowserIceCat()
    {
        return $this->checkSimpleBrowserUA('IceCat', $this->_agent, self::BROWSER_ICECAT);
    }

    /**
     * Determine if the browser is GNU IceWeasel (now know as GNU IceCat) or not.
     * @access protected
     * @see checkBrowserIceCat()
     * @return boolean Returns true if the browser is GNU IceWeasel, false otherwise.
     */
    protected function checkBrowserIceWeasel()
    {
        return $this->checkSimpleBrowserUA('Iceweasel', $this->_agent, self::BROWSER_ICEWEASEL);
    }

    /**
     * Determine if the browser is Internet Explorer or not.
     * @access protected
     * @link http://www.microsoft.com/ie/
     * @link http://en.wikipedia.org/wiki/Internet_Explorer_Mobile
     * @return boolean Returns true if the browser is Internet Explorer, false otherwise.
     */
    protected function checkBrowserInternetExplorer()
    {
        //Test for Internet Explorer Mobile (formerly Pocket Internet Explorer)
        if ($this->checkSimpleBrowserUA(array('IEMobile', 'MSPIE'), $this->_agent, self::BROWSER_IE_MOBILE, true)) {
            return true;
        }

        //Several browsers uses IE compatibility UAs filter these browsers out (but after testing for IE Mobile)
        if (stripos($this->_agent, 'Opera') !== false ||
                stripos($this->_agent, 'BlackBerry') !== false ||
                stripos($this->_agent, 'Nokia') !== false) {
            return false;
        }

        //Test for Internet Explorer 1
        if ($this->checkSimpleBrowserUA('Microsoft Internet Explorer', $this->_agent, self::BROWSER_IE)) {
            if ($this->getVersion() == self::VERSION_UNKNOWN) {
                if (preg_match('/308|425|426|474|0b1/i', $this->_agent)) {
                    $this->setVersion('1.5');
                } else {
                    $this->setVersion('1.0');
                }
            }
            return true;
        }

        //Test for Internet Explorer 2+
        if (stripos($this->_agent, 'MSIE') !== false || stripos($this->_agent, 'Trident') !== false) {
            $version = '';

            if (stripos($this->_agent, 'Trident') !== false) {
                //Test for Internet Explorer 11+ (check the rv: string)
                if (stripos($this->_agent, 'rv:') !== false) {
                    if ($this->checkSimpleBrowserUA('Trident', $this->_agent, self::BROWSER_IE, false, false, 'rv:')) {
                        return true;
                    }
                } else {
                    //Test for Internet Explorer 8, 9 & 10 (check the Trident string)
                    if (preg_match('/Trident\/([\d]+)/i', $this->_agent, $foundVersion)) {
                        //Trident started with version 4.0 on IE 8
                        $verFromTrident = $this->parseInt($foundVersion[1]) + 4;
                        if ($verFromTrident >= 8) {
                            $version = $verFromTrident . '.0';
                        }
                    }
                }

                //If we have the IE version from Trident, we can check for the compatibility view mode
                if ($version != '') {
                    $emulatedVer = '';
                    preg_match_all('/MSIE\s*([^\s;$]+)/i', $this->_agent, $foundVersions);
                    foreach ($foundVersions[1] as $currVer) {
                        //Keep the lowest MSIE version for the emulated version (in compatibility view mode)
                        if ($emulatedVer == '' || $this->compareVersions($emulatedVer, $currVer) == 1) {
                            $emulatedVer = $currVer;
                        }
                    }
                    //Set the compatibility view mode if $version != $emulatedVer
                    if ($this->compareVersions($version, $emulatedVer) != 0) {
                        $this->_compatibilityViewName = self::BROWSER_IE;
                        $this->_compatibilityViewVer = $this->cleanVersion($emulatedVer);
                    }
                }
            }

            //Test for Internet Explorer 2-7 versions if needed
            if ($version == '') {
                preg_match_all('/MSIE\s+([^\s;$]+)/i', $this->_agent, $foundVersions);
                foreach ($foundVersions[1] as $currVer) {
                    //Keep the highest MSIE version
                    if ($version == '' || $this->compareVersions($version, $currVer) == -1) {
                        $version = $currVer;
                    }
                }
            }

            $this->setBrowser(self::BROWSER_IE);
            $this->setVersion($version);
            $this->setMobile(false);
            $this->setRobot(false);

            return true;
        }

        return false;
    }

    /**
     * Determine if the browser is Konqueror or not.
     * @access protected
     * @link http://www.konqueror.org/
     * @return boolean Returns true if the browser is Konqueror, false otherwise.
     */
    protected function checkBrowserKonqueror()
    {
        return $this->checkSimpleBrowserUA('Konqueror', $this->_agent, self::BROWSER_KONQUEROR);
    }

    /**
     * Determine if the browser is Lynx or not. It is the oldest web browser currently in general use and development.
     * It is a text-based only Web browser.
     * @access protected
     * @link http://en.wikipedia.org/wiki/Lynx
     * @return boolean Returns true if the browser is Lynx, false otherwise.
     */
    protected function checkBrowserLynx()
    {
        return $this->checkSimpleBrowserUA('Lynx', $this->_agent, self::BROWSER_LYNX);
    }

    /**
     * Determine if the browser is Mozilla or not.
     * @access protected
     * @return boolean Returns true if the browser is Mozilla, false otherwise.
     */
    protected function checkBrowserMozilla()
    {
        return $this->checkSimpleBrowserUA('Mozilla', $this->_agent, self::BROWSER_MOZILLA, false, false, 'rv:');
    }

    /**
     * Determine if the browser is the MSNBot crawler or not. In October 2010 it was replaced by the Bingbot robot.
     * @access protected
     * @see checkBrowserBingbot()
     * @return boolean Returns true if the browser is MSNBot, false otherwise.
     */
    protected function checkBrowserMsnBot()
    {
        return $this->checkSimpleBrowserUA('msnbot', $this->_agent, self::BROWSER_MSNBOT, false, true);
    }

    /**
     * Determine if the browser is MSN TV (formerly WebTV) or not.
     * @access protected
     * @link http://en.wikipedia.org/wiki/MSN_TV
     * @return boolean Returns true if the browser is WebTv, false otherwise.
     */
    protected function checkBrowserMsnTv()
    {
        return $this->checkSimpleBrowserUA('webtv', $this->_agent, self::BROWSER_MSNTV);
    }

    /**
     * Determine if the browser is NetPositive or not. The browser is discontinued since November 2001.
     * @access protected
     * @link http://en.wikipedia.org/wiki/NetPositive
     * @return boolean Returns true if the browser is NetPositive, false otherwise.
     */
    protected function checkBrowserNetPositive()
    {
        return $this->checkSimpleBrowserUA('NetPositive', $this->_agent, self::BROWSER_NETPOSITIVE);
    }

    /**
     * Determine if the browser is Netscape or not. Official support for this browser ended on March 1st, 2008.
     * @access protected
     * @link http://en.wikipedia.org/wiki/Netscape
     * @return boolean Returns true if the browser is Netscape, false otherwise.
     */
    protected function checkBrowserNetscape()
    {
        //BlackBerry & Nokia UAs can conflict with Netscape UAs
        if (stripos($this->_agent, 'BlackBerry') !== false || stripos($this->_agent, 'Nokia') !== false) {
            return false;
        }

        //Netscape v6 to v9 check
        if ($this->checkSimpleBrowserUA(array('Netscape', 'Navigator', 'Netscape6'), $this->_agent, self::BROWSER_NETSCAPE)) {
            return true;
        }

        //Netscape v1-4 (v5 don't exists)
        $found = false;
        if (stripos($this->_agent, 'Mozilla') !== false && stripos($this->_agent, 'rv:') === false) {
            $version = '';
            $verParts = explode('/', stristr($this->_agent, 'Mozilla'));
            if (count($verParts) > 1) {
                $verParts = explode(' ', $verParts[1]);
                $verParts = explode('.', $verParts[0]);

                $majorVer = $this->parseInt($verParts[0]);
                if ($majorVer > 0 && $majorVer < 5) {
                    $version = implode('.', $verParts);
                    $found = true;

                    if (strtolower(substr($version, -4)) == '-sgi') {
                        $version = substr($version, 0, -4);
                    } else {
                        if (strtolower(substr($version, -4)) == 'gold') {
                            $version = substr($version, 0, -4) . ' Gold'; //Doubles spaces (if any) will be normalized by setVersion()
                        }
                    }
                }
            }
        }

        if ($found) {
            $this->setBrowser(self::BROWSER_NETSCAPE);
            $this->setVersion($version);
            $this->setMobile(false);
            $this->setRobot(false);
        }

        return $found;
    }

    /**
     * Determine if the browser is a Nokia browser or not.
     * @access protected
     * @link http://www.developer.nokia.com/Community/Wiki/User-Agent_headers_for_Nokia_devices
     * @return boolean Returns true if the browser is a Nokia browser, false otherwise.
     */
    protected function checkBrowserNokia()
    {
        if (stripos($this->_agent, 'Nokia5800') !== false || stripos($this->_agent, 'Nokia5530') !== false || stripos($this->_agent, 'Nokia5230') !== false) {
            $this->setBrowser(self::BROWSER_NOKIA);
            $this->setVersion('7.0');
            $this->setMobile(true);
            $this->setRobot(false);

            return true;
        }

        if ($this->checkSimpleBrowserUA(array('NokiaBrowser', 'BrowserNG', 'Series60', 'S60', 'S40OviBrowser'), $this->_agent, self::BROWSER_NOKIA, true)) {
            return true;
        }

        return false;
    }

    /**
     * Determine if the browser is OmniWeb or not.
     * @access protected
     * @link http://www.omnigroup.com/products/omniweb/
     * @return boolean Returns true if the browser is OmniWeb, false otherwise.
     */
    protected function checkBrowserOmniWeb()
    {
        if ($this->checkSimpleBrowserUA('OmniWeb', $this->_agent, self::BROWSER_OMNIWEB)) {
            //Some versions of OmniWeb prefix the version number with "v"
            if ($this->getVersion() != self::VERSION_UNKNOWN && strtolower(substr($this->getVersion(), 0, 1)) == 'v') {
                $this->setVersion(substr($this->getVersion(), 1));
            }
            return true;
        }

        return false;
    }

    /**
     * Determine if the browser is Opera or not.
     * @access protected
     * @link http://www.opera.com/
     * @link http://www.opera.com/mini/
     * @link http://www.opera.com/mobile/
     * @link http://my.opera.com/community/openweb/idopera/
     * @return boolean Returns true if the browser is Opera, false otherwise.
     */
    protected function checkBrowserOpera()
    {
        if ($this->checkBrowserUAWithVersion('Opera Mobi', $this->_agent, self::BROWSER_OPERA_MOBILE, true)) {
            return true;
        }

        if ($this->checkSimpleBrowserUA('Opera Mini', $this->_agent, self::BROWSER_OPERA_MINI, true)) {
            return true;
        }

        $version = '';
        $found = $this->checkBrowserUAWithVersion('Opera', $this->_agent, self::BROWSER_OPERA);
        if ($found && $this->getVersion() != self::VERSION_UNKNOWN) {
            $version = $this->getVersion();
        }

        if (!$found || $version == '') {
            if ($this->checkSimpleBrowserUA('Opera', $this->_agent, self::BROWSER_OPERA)) {
                return true;
            }
        }

        if (!$found && $this->checkSimpleBrowserUA('Chrome', $this->_agent, self::BROWSER_CHROME) ) {
            if ($this->checkSimpleBrowserUA('OPR/', $this->_agent, self::BROWSER_OPERA)) {
                return true;
            }
        }

        return $found;
    }

    /**
     * Determine if the browser is Phoenix or not. Phoenix was the name of Firefox from version 0.1 to 0.5.
     * @access protected
     * @return boolean Returns true if the browser is Phoenix, false otherwise.
     */
    protected function checkBrowserPhoenix()
    {
        return $this->checkSimpleBrowserUA('Phoenix', $this->_agent, self::BROWSER_PHOENIX);
    }

    /**
     * Determine what is the browser used by the user.
     * @access protected
     * @return boolean Returns true if the browser has been identified, false otherwise.
     */
    protected function checkBrowsers()
    {
        //Changing the check order can break the class detection results!
        return
               /* Major browsers and browsers that need to be detected in a special order */
               $this->checkBrowserMsnTv() ||            /* MSN TV is based on IE so we must check for MSN TV before IE */
               $this->checkBrowserInternetExplorer() ||
               $this->checkBrowserOpera() ||            /* Opera must be checked before Firefox, Netscape and Chrome to avoid conflicts */
               $this->checkBrowserEdge() ||             /* Edge must be checked before Firefox, Safari and Chrome to avoid conflicts */
               $this->checkBrowserChrome() ||           /* Chrome must be checked before Netscaoe and Mozilla to avoid conflicts */
               $this->checkBrowserOmniWeb() ||          /* OmniWeb must be checked before Safari (on which it's based on) and Netscape (since it have Mozilla UAs) */
               $this->checkBrowserIcab() ||             /* Check iCab before Netscape since iCab have Mozilla UAs */
               $this->checkBrowserNetPositive() ||      /* Check NetPositive before Netscape since NetPositive have Mozilla UAs */
               $this->checkBrowserNetscape() ||         /* Must be checked before Firefox since Netscape 8-9 are based on Firefox */
               $this->checkBrowserIceCat() ||           /* Check IceCat and IceWeasel before Firefox since they are GNU builds of Firefox */
               $this->checkBrowserIceWeasel() ||
               $this->checkBrowserGaleon() ||           /* Galeon is based on Firefox and needs to be tested before Firefox is tested */
               $this->checkBrowserFirefox() ||
               /* Current browsers that don't need to be detected in any special order */
               $this->checkBrowserKonqueror() ||
               $this->checkBrowserLynx() ||
               $this->checkBrowserAmaya() ||
               /* Mobile */
               $this->checkBrowserAndroid() ||
               $this->checkBrowserBlackBerry() ||
               $this->checkBrowserNokia() ||
               /* Bots */
               $this->checkBrowserGooglebot() ||
               $this->checkBrowserBingbot() ||
               $this->checkBrowserMsnBot() ||
               $this->checkBrowserSlurp() ||
               $this->checkBrowserYahooMultimedia() ||
               $this->checkBrowserW3CValidator() ||
               /* WebKit base check (after most other checks) */
               $this->checkBrowserSafari() ||
               /* Deprecated browsers that don't need to be detected in any special order */
               $this->checkBrowserFirebird() ||
               $this->checkBrowserPhoenix() ||
               /* Mozilla is such an open standard that it must be checked last */
               $this->checkBrowserMozilla();
    }

    /**
     * Determine if the browser is Safari or not.
     * @access protected
     * @link http://www.apple.com/safari/
     * @link http://web.archive.org/web/20080514173941/http://developer.apple.com/internet/safari/uamatrix.html
     * @link http://en.wikipedia.org/wiki/Safari_version_history#Release_history
     * @return boolean Returns true if the browser is Safari, false otherwise.
     */
    protected function checkBrowserSafari()
    {
        $version = '';

        //Check for current versions of Safari
        $found = $this->checkBrowserUAWithVersion(array('Safari', 'AppleWebKit'), $this->_agent, self::BROWSER_SAFARI);
        if ($found && $this->getVersion() != self::VERSION_UNKNOWN) {
            $version = $this->getVersion();
        }

        //Safari 1-2 didn't had a "Version" string in the UA, only a WebKit build and/or Safari build, extract version from these...
        if (!$found || $version == '') {
            if (preg_match('/.*Safari[ (\/]*([a-z0-9.-]*)/i', $this->_agent, $matches)) {
                $version = $this->safariBuildToSafariVer($matches[1]);
                $found = true;
            }
        }
        if (!$found || $version == '') {
            if (preg_match('/.*AppleWebKit[ (\/]*([a-z0-9.-]*)/i', $this->_agent, $matches)) {
                $version = $this->webKitBuildToSafariVer($matches[1]);
                $found = true;
            }
        }

        if ($found) {
            $this->setBrowser(self::BROWSER_SAFARI);
            $this->setVersion($version);
            $this->setMobile(false);
            $this->setRobot(false);
        }

        return $found;
    }

    /**
     * Determine if the browser is the Yahoo! Slurp crawler or not.
     * @access protected
     * @return boolean Returns true if the browser is Yahoo! Slurp, false otherwise.
     */
    protected function checkBrowserSlurp()
    {
        return $this->checkSimpleBrowserUA('Yahoo! Slurp', $this->_agent, self::BROWSER_SLURP, false, true);
    }

    /**
     * Test the user agent for a specific browser that use a "Version" string (like Safari and Opera). The user agent
     * should look like: "Version/1.0 Browser name/123.456" or "Browser name/123.456 Version/1.0".
     * @access protected
     * @param mixed $uaNameToLookFor The string (or array of strings) representing the browser name to find in the user
     * agent.
     * @param string $userAgent The user agent string to work with.
     * @param string $browserName The literal browser name. Always use a class constant!
     * @param boolean $isMobile Determines if the browser is from a mobile device.
     * @param boolean $isRobot Determines if the browser is a robot or not.
     * @return boolean Returns true if we found the browser we were looking for, false otherwise.
     */
    protected function checkBrowserUAWithVersion($uaNameToLookFor, $userAgent, $browserName, $isMobile = false, $isRobot = false)
    {
        if (!is_array($uaNameToLookFor)) {
            $uaNameToLookFor = array($uaNameToLookFor);
        }

        foreach ($uaNameToLookFor as $currUANameToLookFor) {
            if (stripos($userAgent, $currUANameToLookFor) !== false) {
                $version = '';
                $verParts = explode('/', stristr($this->_agent, 'Version'));
                if (count($verParts) > 1) {
                    $verParts = explode(' ', $verParts[1]);
                    $version = $verParts[0];
                }

                $this->setBrowser($browserName);
                $this->setVersion($version);

                $this->setMobile($isMobile);
                $this->setRobot($isRobot);

                return true;
            }
        }

        return false;
    }

    /**
     * Determine if the browser is the W3C Validator or not.
     * @access protected
     * @link http://validator.w3.org/
     * @return boolean Returns true if the browser is the W3C Validator, false otherwise.
     */
    protected function checkBrowserW3CValidator()
    {
        //Since the W3C validates pages with different robots we will prefix our versions with the part validated on the page...

        //W3C Link Checker (prefixed with "Link-")
        if ($this->checkSimpleBrowserUA('W3C-checklink', $this->_agent, self::BROWSER_W3CVALIDATOR, false, true)) {
            if ($this->getVersion() != self::VERSION_UNKNOWN) {
                $this->setVersion('Link-' . $this->getVersion());
            }
            return true;
        }

        //W3C CSS Validation Service (prefixed with "CSS-")
        if ($this->checkSimpleBrowserUA('Jigsaw', $this->_agent, self::BROWSER_W3CVALIDATOR, false, true)) {
            if ($this->getVersion() != self::VERSION_UNKNOWN) {
                $this->setVersion('CSS-' . $this->getVersion());
            }
            return true;
        }

        //W3C mobileOK Checker (prefixed with "mobileOK-")
        if ($this->checkSimpleBrowserUA('W3C-mobileOK', $this->_agent, self::BROWSER_W3CVALIDATOR, false, true)) {
            if ($this->getVersion() != self::VERSION_UNKNOWN) {
                $this->setVersion('mobileOK-' . $this->getVersion());
            }
            return true;
        }

        //W3C Markup Validation Service (no prefix)
        return $this->checkSimpleBrowserUA('W3C_Validator', $this->_agent, self::BROWSER_W3CVALIDATOR, false, true);
    }

    /**
     * Determine if the browser is the Yahoo! multimedia crawler or not.
     * @access protected
     * @return boolean Returns true if the browser is the Yahoo! multimedia crawler, false otherwise.
     */
    protected function checkBrowserYahooMultimedia()
    {
        return $this->checkSimpleBrowserUA('Yahoo-MMCrawler', $this->_agent, self::BROWSER_YAHOO_MM, false, true);
    }

    /**
     * Determine if the user is using an AOL "optimized" browser or not.
     * @access protected
     * @return boolean Returns true if the browser is AOL optimized, false otherwise.
     */
    protected function checkForAol()
    {
        //AOL UAs don't use the "AOL/1.0" format, they uses "AOL 1.0; AOLBuild 100.00;"
        if (stripos($this->_agent, 'AOL ') !== false) {
            $version = '';
            $verParts = explode('AOL ', stristr($this->_agent, 'AOL '));
            if (count($verParts) > 1) {
                $verParts = explode(' ', $verParts[1]);
                $version = $verParts[0];
            }

            $this->setAol(true);
            $this->setAolVersion($version);

            return true;
        } else {
            $this->setAol(false);
            $this->setAolVersion('');

            return false;
        }
    }

    /**
     * Determine the user's platform.
     * @access protected
     */
    protected function checkPlatform()
    {
        /* Mobile platforms */
        if (stripos($this->_agent, 'Windows Phone') !== false ||     /* Check Windows Phone (formerly Windows Mobile) before Windows */
                stripos($this->_agent, 'IEMobile') !== false) {
            $this->setPlatform(self::PLATFORM_WINDOWS_PHONE);
            $this->setMobile(true);
        } else if (stripos($this->_agent, 'Windows CE') !== false) { /* Check Windows CE before Windows */
            $this->setPlatform(self::PLATFORM_WINDOWS_CE);
            $this->setMobile(true);
        } else if (stripos($this->_agent, 'iPhone') !== false) {     /* Check iPad/iPod/iPhone before Macintosh */
            $this->setPlatform(self::PLATFORM_IPHONE);
            $this->setMobile(true);
        } else if (stripos($this->_agent, 'iPad') !== false) {
            $this->setPlatform(self::PLATFORM_IPAD);
            $this->setMobile(true);
        } else if (stripos($this->_agent, 'iPod') !== false) {
            $this->setPlatform(self::PLATFORM_IPOD);
            $this->setMobile(true);
        } else if (stripos($this->_agent, 'Android') !== false) {
            $this->setPlatform(self::PLATFORM_ANDROID);
            $this->setMobile(true);
        } else if (stripos($this->_agent, 'Symbian') !== false) {
            $this->setPlatform(self::PLATFORM_SYMBIAN);
            $this->setMobile(true);
        } else if (stripos($this->_agent, 'BlackBerry') !== false ||
                stripos($this->_agent, 'BB10') !== false ||
                stripos($this->_agent, 'RIM Tablet OS') !== false) {
            $this->setPlatform(self::PLATFORM_BLACKBERRY);
            $this->setMobile(true);
        } else if (stripos($this->_agent, 'Nokia') !== false) {
            $this->setPlatform(self::PLATFORM_NOKIA);
            $this->setMobile(true);

        /* Desktop platforms */
        } else if (stripos($this->_agent, 'Windows') !== false) {
            $this->setPlatform(self::PLATFORM_WINDOWS);
        } else if (stripos($this->_agent, 'Macintosh') !== false) {
            $this->setPlatform(self::PLATFORM_MACINTOSH);
        } else if (stripos($this->_agent, 'Linux') !== false) {
            $this->setPlatform(self::PLATFORM_LINUX);
        } else if (stripos($this->_agent, 'FreeBSD') !== false) {
            $this->setPlatform(self::PLATFORM_FREEBSD);
        } else if (stripos($this->_agent, 'OpenBSD') !== false) {
            $this->setPlatform(self::PLATFORM_OPENBSD);
        } else if (stripos($this->_agent, 'NetBSD') !== false) {
            $this->setPlatform(self::PLATFORM_NETBSD);

        /* Discontinued */
        } else if (stripos($this->_agent, 'OpenSolaris') !== false) {
            $this->setPlatform(self::PLATFORM_OPENSOLARIS);
        } else if (stripos($this->_agent, 'OS/2') !== false) {
            $this->setPlatform(self::PLATFORM_OS2);
        } else if (stripos($this->_agent, 'BeOS') !== false) {
            $this->setPlatform(self::PLATFORM_BEOS);
        } else if (stripos($this->_agent, 'SunOS') !== false) {
            $this->setPlatform(self::PLATFORM_SUNOS);

        /* Generic */
        } else if (stripos($this->_agent, 'Win') !== false) {
            $this->setPlatform(self::PLATFORM_WINDOWS);
        } else if (stripos($this->_agent, 'Mac') !== false) {
            $this->setPlatform(self::PLATFORM_MACINTOSH);
        }

        //Check if it's a 64-bit platform
        if (stripos($this->_agent, 'WOW64') !== false || stripos($this->_agent, 'Win64') !== false) {
            $this->set64bit(true);
        }

        $this->checkPlatformVersion();
    }

    /**
     * Determine the user's platform version.
     * @access protected
     */
    protected function checkPlatformVersion()
    {
        //https://support.microsoft.com/en-us/kb/158238

        $result = '';

        switch ($this->getPlatform()) {
            case self::PLATFORM_WINDOWS:
                if (preg_match('/Windows NT\s*([^\s;\)$]+)/i', $this->_agent, $foundVersion)) {
                    //Windows NT family
                    $result = 'NT ' . $foundVersion[1];
                } else {
                    //Windows 3.x / 9x family
                    if (stripos($this->_agent, 'Windows ME') !== false) {
                        $result = '4.90.3000'; //Windows Me version range from 4.90.3000 to 4.90.3000A
                    } else if (stripos($this->_agent, 'Windows 98') !== false) {
                        $result = '4.10'; //Windows 98 version range from 4.10.1998 to 4.10.2222B
                    } else if (stripos($this->_agent, 'Windows 95') !== false) {
                        $result = '4.00'; //Windows 95 version range from 4.00.950 to 4.03.1214
                    } else if (preg_match('/Windows 3\.([^\s;\)$]+)/i', $this->_agent, $foundVersion)) {
                        $result = '3.' . $foundVersion[1];
                    }
                }

            break;
        }

        if (trim($result) == '') {
            $result = self::PLATFORM_VERSION_UNKNOWN;
        }
        $this->setPlatformVersion($result);
    }

    /**
     * Test the user agent for a specific browser where the browser name is immediately followed by the version number.
     * The user agent should look like: "Browser name/1.0" or "Browser 1.0;".
     * @access protected
     * @param mixed $uaNameToLookFor The string (or array of strings) representing the browser name to find in the user
     * agent.
     * @param string $userAgent The user agent string to work with.
     * @param string $browserName The literal browser name. Always use a class constant!
     * @param boolean $isMobile Determines if the browser is from a mobile device.
     * @param boolean $isRobot Determines if the browser is a robot or not.
     * @param string $separator The separator string used to split the browser name and the version number in the user
     * agent.
     * @return boolean Returns true if we found the browser we were looking for, false otherwise.
     */
    protected function checkSimpleBrowserUA($uaNameToLookFor, $userAgent, $browserName, $isMobile = false, $isRobot = false, $separator = '/')
    {
        if (!is_array($uaNameToLookFor)) {
            $uaNameToLookFor = array($uaNameToLookFor);
        }

        foreach ($uaNameToLookFor as $currUANameToLookFor) {
            if (stripos($userAgent, $currUANameToLookFor) !== false) {
                //Many browsers don't use the standard "Browser/1.0" format, they uses "Browser 1.0;" instead
                if (stripos($userAgent, $currUANameToLookFor . $separator) === false) {
                    $userAgent = str_ireplace($currUANameToLookFor . ' ', $currUANameToLookFor . $separator, $this->_agent);
                }

                $version = '';
                $verParts = explode($separator, stristr($userAgent, $currUANameToLookFor));
                if (count($verParts) > 1) {
                    $verParts = explode(' ', $verParts[1]);
                    $version = $verParts[0];
                }

                $this->setBrowser($browserName);
                $this->setVersion($version);

                $this->setMobile($isMobile);
                $this->setRobot($isRobot);

                return true;
            }
        }

        return false;
    }

    /**
     * Detect the user environment from the details in the user agent string.
     * @access protected
     */
    protected function detect()
    {
        $this->checkBrowsers();
        $this->checkPlatform(); //Check the platform after the browser since some platforms can change the mobile value
        $this->checkForAol();
    }

    /**
     * Clean a version string from unwanted characters.
     * @access protected
     * @param string $version The version string to clean.
     * @return string Returns the cleaned version number string.
     */
    protected function cleanVersion($version)
    {
        //Clear anything that is in parentheses (and the parentheses themselves) - will clear started but unclosed ones too
        $cleanVer = preg_replace('/\([^)]+\)?/', '', $version);
        //Replace with a space any character which is NOT an alphanumeric, dot (.), hyphen (-), underscore (_) or space
        $cleanVer = preg_replace('/[^0-9.a-zA-Z_ -]/', ' ', $cleanVer);
        //Remove trailing and leading spaces
        $cleanVer = trim($cleanVer);
        //Remove double spaces if any
        while (strpos($cleanVer, '  ') !== false) {
            $cleanVer = str_replace('  ', ' ', $cleanVer);
        }

        return $cleanVer;
    }

    /**
     * Get the integer value of a string variable.
     * @access protected
     * @param string $intStr The scalar value being converted to an integer.
     * @return int The integer value of $intStr on success, or 0 on failure.
     */
    protected function parseInt($intStr)
    {
        return intval($intStr, 10);
    }

    /**
     * Reset all the properties of the class.
     * @access protected
     */
    protected function reset()
    {
        $this->_agent = '';
        $this->_aolVersion = '';
        $this->_browserName = self::BROWSER_UNKNOWN;
        $this->_compatibilityViewName = '';
        $this->_compatibilityViewVer = '';
        $this->_is64bit = false;
        $this->_isAol = false;
        $this->_isMobile = false;
        $this->_isRobot = false;
        $this->_platform = self::PLATFORM_UNKNOWN;
        $this->_platformVersion = self::PLATFORM_VERSION_UNKNOWN;
        $this->_version = self::VERSION_UNKNOWN;
    }

    /**
     * Convert a Safari build number to a Safari version number.
     * @access protected
     * @param string $version A string representing the version number.
     * @link http://web.archive.org/web/20080514173941/http://developer.apple.com/internet/safari/uamatrix.html
     * @return string Returns the Safari version string. If the version can't be determined, an empty string is
     * returned.
     */
    protected function safariBuildToSafariVer($version)
    {
        $verParts = explode('.', $version);

        //We need a 3 parts version (version 2 will becomes 2.0.0)
        while (count($verParts) < 3) {
            $verParts[] = 0;
        }
        foreach ($verParts as $i => $currPart) {
            $verParts[$i] = $this->parseInt($currPart);
        }

        switch ($verParts[0]) {
            case 419: $result = '2.0.4';
                break;
            case 417: $result = '2.0.3';
                break;
            case 416: $result = '2.0.2';
                break;

            case 412:
                if ($verParts[1] >= 5) {
                    $result = '2.0.1';
                } else {
                    $result = '2.0';
                }
                break;

            case 312:
                if ($verParts[1] >= 5) {
                    $result = '1.3.2';
                } else {
                    if ($verParts[1] >= 3) {
                        $result = '1.3.1';
                    } else {
                        $result = '1.3';
                    }
                }
                break;

            case 125:
                if ($verParts[1] >= 11) {
                    $result = '1.2.4';
                } else {
                    if ($verParts[1] >= 9) {
                        $result = '1.2.3';
                    } else {
                        if ($verParts[1] >= 7) {
                            $result = '1.2.2';
                        } else {
                            $result = '1.2';
                        }
                    }
                }
                break;

            case 100:
                if ($verParts[1] >= 1) {
                    $result = '1.1.1';
                } else {
                    $result = '1.1';
                }
                break;

            case 85:
                if ($verParts[1] >= 8) {
                    $result = '1.0.3';
                } else {
                    if ($verParts[1] >= 7) {
                        $result = '1.0.2';
                    } else {
                        $result = '1.0';
                    }
                }
                break;

            case 73: $result = '0.9';
                break;
            case 51: $result = '0.8.1';
                break;
            case 48: $result = '0.8';
                break;

            default: $result = '';
        }

        return $result;
    }

    /**
     * Set if the browser is executed from a 64-bit platform.
     * @access protected
     * @param boolean $is64bit Value that tells if the browser is executed from a 64-bit platform.
     */
    protected function set64bit($is64bit)
    {
        $this->_is64bit = $is64bit == true;
    }

    /**
     * Set the browser to be from AOL or not.
     * @access protected
     * @param boolean $isAol Value that tells if the browser is AOL or not.
     */
    protected function setAol($isAol)
    {
        $this->_isAol = $isAol == true;
    }

    /**
     * Set the version of AOL.
     * @access protected
     * @param string $version The version of AOL (will be cleaned).
     */
    protected function setAolVersion($version)
    {
        $cleanVer = $this->cleanVersion($version);

        $this->_aolVersion = $cleanVer;
    }

    /**
     * Set the name of the browser.
     * @access protected
     * @param string $browserName The name of the browser.
     */
    protected function setBrowser($browserName)
    {
        $this->_browserName = $browserName;
    }

    /**
     * Set the browser to be from a mobile device or not.
     * @access protected
     * @param boolean $isMobile Value that tells if the browser is on a mobile device or not.
     */
    protected function setMobile($isMobile = true)
    {
        $this->_isMobile = $isMobile == true;
    }

    /**
     * Set the platform on which the browser is on.
     * @access protected
     * @param string $platform The name of the platform.
     */
    protected function setPlatform($platform)
    {
        $this->_platform = $platform;
    }

    /**
     * Set the platform version on which the browser is on.
     * @access protected
     * @param string $platformVer The version numbers of the platform.
     */
    protected function setPlatformVersion($platformVer)
    {
        $this->_platformVersion = $platformVer;
    }

    /**
     * Set the browser to be a robot (crawler) or not.
     * @access protected
     * @param boolean $isRobot Value that tells if the browser is a robot or not.
     */
    protected function setRobot($isRobot = true)
    {
        $this->_isRobot = $isRobot == true;
    }

    /**
     * Set the version of the browser.
     * @access protected
     * @param string $version The version of the browser.
     */
    protected function setVersion($version)
    {
        $cleanVer = $this->cleanVersion($version);

        if ($cleanVer == '') {
            $this->_version = self::VERSION_UNKNOWN;
        } else {
            $this->_version = $cleanVer;
        }
    }

    /**
     * Convert a WebKit build number to a Safari version number.
     * @access protected
     * @param string $version A string representing the version number.
     * @link http://web.archive.org/web/20080514173941/http://developer.apple.com/internet/safari/uamatrix.html
     * @return string Returns the Safari version string. If the version can't be determined, an empty string is
     * returned.
     */
    protected function webKitBuildToSafariVer($version)
    {
        $verParts = explode('.', $version);

        //We need a 3 parts version (version 2 will becomes 2.0.0)
        while (count($verParts) < 3) {
            $verParts[] = 0;
        }
        foreach ($verParts as $i => $currPart) {
            $verParts[$i] = $this->parseInt($currPart);
        }

        switch ($verParts[0]) {
            case 419: $result = '2.0.4';
                break;

            case 418:
                if ($verParts[1] >= 8) {
                    $result = '2.0.4';
                } else {
                    $result = '2.0.3';
                }
                break;

            case 417: $result = '2.0.3';
                break;

            case 416: $result = '2.0.2';
                break;

            case 412:
                if ($verParts[1] >= 7) {
                    $result = '2.0.1';
                } else {
                    $result = '2.0';
                }
                break;

            case 312:
                if ($verParts[1] >= 8) {
                    $result = '1.3.2';
                } else {
                    if ($verParts[1] >= 5) {
                        $result = '1.3.1';
                    } else {
                        $result = '1.3';
                    }
                }
                break;

            case 125:
                if ($this->compareVersions('5.4', $verParts[1] . '.' . $verParts[2]) == -1) {
                    $result = '1.2.4'; //125.5.5+
                } else {
                    if ($verParts[1] >= 4) {
                        $result = '1.2.3';
                    } else {
                        if ($verParts[1] >= 2) {
                            $result = '1.2.2';
                        } else {
                            $result = '1.2';
                        }
                    }
                }
                break;

            //WebKit 100 can be either Safari 1.1 (Safari build 100) or 1.1.1 (Safari build 100.1)
            //for this reason, check the Safari build before the WebKit build.
            case 100: $result = '1.1.1';
                break;

            case 85:
                if ($verParts[1] >= 8) {
                    $result = '1.0.3';
                } else {
                    if ($verParts[1] >= 7) {
                        //WebKit 85.7 can be either Safari 1.0 (Safari build 85.5) or 1.0.2 (Safari build 85.7)
                        //for this reason, check the Safari build before the WebKit build.
                        $result = '1.0.2';
                    } else {
                        $result = '1.0';
                    }
                }
                break;

            case 73: $result = '0.9';
                break;
            case 51: $result = '0.8.1';
                break;
            case 48: $result = '0.8';
                break;

            default: $result = '';
        }

        return $result;
    }

    /**
     * Convert the Windows NT family version numbers to the operating system name. For instance '5.1' returns
     * 'Windows XP'.
     * @access protected
     * @param string $winVer The Windows NT family version numbers as a string.
     * @param boolean $returnServerFlavor Since some Windows NT versions have the same values, this flag determines if
     * the Server flavor is returned or not. For instance Windows 8.1 and Windows Server 2012 R2 both use version 6.3.
     * @return string The operating system name or the constant PLATFORM_VERSION_UNKNOWN if nothing match the version
     * numbers.
     */
    protected function windowsNTVerToStr($winVer, $returnServerFlavor = false)
    {
        //https://en.wikipedia.org/wiki/List_of_Microsoft_Windows_versions

        $cleanWinVer = explode('.', $winVer);
        while (count($cleanWinVer) > 2) {
            array_pop($cleanWinVer);
        }
        $cleanWinVer = implode('.', $cleanWinVer);

        if ($this->compareVersions($cleanWinVer, '11') >= 0) {
            //Future versions of Windows
            return self::PLATFORM_WINDOWS . ' ' . $winVer;
        } else if ($this->compareVersions($cleanWinVer, '10') >= 0) {
            //Current version of Windows
            return $returnServerFlavor ? (self::PLATFORM_WINDOWS . ' Server 2016') : (self::PLATFORM_WINDOWS . ' 10');
        } else if ($this->compareVersions($cleanWinVer, '7') < 0) {
            if ($this->compareVersions($cleanWinVer, '6.3') == 0) {
                return $returnServerFlavor ? (self::PLATFORM_WINDOWS . ' Server 2012 R2') : (self::PLATFORM_WINDOWS . ' 8.1');
            } else if ($this->compareVersions($cleanWinVer, '6.2') == 0) {
                return $returnServerFlavor ? (self::PLATFORM_WINDOWS . ' Server 2012') : (self::PLATFORM_WINDOWS . ' 8');
            } else if ($this->compareVersions($cleanWinVer, '6.1') == 0) {
                return $returnServerFlavor ? (self::PLATFORM_WINDOWS . ' Server 2008 R2') : (self::PLATFORM_WINDOWS . ' 7');
            } else if ($this->compareVersions($cleanWinVer, '6') == 0) {
                return $returnServerFlavor ? (self::PLATFORM_WINDOWS . ' Server 2008') : (self::PLATFORM_WINDOWS . ' Vista');
            } else if ($this->compareVersions($cleanWinVer, '5.2') == 0) {
                return $returnServerFlavor ? (self::PLATFORM_WINDOWS . ' Server 2003 / ' . self::PLATFORM_WINDOWS . ' Server 2003 R2') : (self::PLATFORM_WINDOWS . ' XP x64 Edition');
            } else if ($this->compareVersions($cleanWinVer, '5.1') == 0) {
                return self::PLATFORM_WINDOWS . ' XP';
            } else if ($this->compareVersions($cleanWinVer, '5') == 0) {
                return self::PLATFORM_WINDOWS . ' 2000';
            } else if ($this->compareVersions($cleanWinVer, '5') < 0 && $this->compareVersions($cleanWinVer, '3') >= 0) {
                return self::PLATFORM_WINDOWS . ' NT ' . $winVer;
            }
        }

        return self::PLATFORM_VERSION_UNKNOWN; //Invalid Windows NT version
    }

    /**
     * Convert the Windows 3.x & 9x family version numbers to the operating system name. For instance '4.10.1998'
     * returns 'Windows 98'.
     * @access protected
     * @param string $winVer The Windows 3.x or 9x family version numbers as a string.
     * @return string The operating system name or the constant PLATFORM_VERSION_UNKNOWN if nothing match the version
     * numbers.
     */
    protected function windowsVerToStr($winVer)
    {
        //https://support.microsoft.com/en-us/kb/158238

        if ($this->compareVersions($winVer, '4.90') >= 0 && $this->compareVersions($winVer, '4.91') < 0) {
            return self::PLATFORM_WINDOWS . ' Me'; //Normally range from 4.90.3000 to 4.90.3000A
        } else if ($this->compareVersions($winVer, '4.10') >= 0 && $this->compareVersions($winVer, '4.11') < 0) {
            return self::PLATFORM_WINDOWS . ' 98'; //Normally range from 4.10.1998 to 4.10.2222B
        } else if ($this->compareVersions($winVer, '4') >= 0 && $this->compareVersions($winVer, '4.04') < 0) {
            return self::PLATFORM_WINDOWS . ' 95'; //Normally range from 4.00.950 to 4.03.1214
        } else if ($this->compareVersions($winVer, '3.1') == 0 || $this->compareVersions($winVer, '3.11') == 0) {
            return self::PLATFORM_WINDOWS . ' ' . $winVer;
        } else {
            return self::PLATFORM_VERSION_UNKNOWN; //Invalid Windows version
        }
    }
}

Function Calls

None

Variables

None

Stats

MD5 440a4353652f2e06128b350226eac912
Eval Count 0
Decode Time 172 ms