search
HomeBackend DevelopmentPHP TutorialCI framework source code reading---------Router.php_PHP tutorial

[php]  

/** 
 * CodeIgniter 
 * 
 * An open source application development framework for PHP 5.1.6 or newer 
 * 
 * @package     CodeIgniter 
 * @author      ExpressionEngine Dev Team 
 * @copyright   Copyright (c) 2008 - 2011, EllisLab, Inc. 
 * @license     http://codeigniter.com/user_guide/license.html 
 * @link        http://codeigniter.com 
 * @since       Version 1.0 
 * @filesource 
 */  
  
// ------------------------------------------------------------------------  
  
/** 
 * Router Class 
 * 
 * Parses URIs and determines routing 
 * 
 * @package     CodeIgniter 
 * @subpackage  Libraries 
 * @author      ExpressionEngine Dev Team 
 * @category    Libraries 
 * @link        http://codeigniter.com/user_guide/general/routing.html 
 */  
class CI_Router {  
  
    /** 
     * Config class 
     * 配置 
     * @var object 
     * @access public 
     */  
    var $config;  
    /**
* List of routes
* Route list, value comes from APPPATH/config/route.php
* @var array
* @access public
*/  
    var $routes         = array();  
    /**
* List of error routes
* Error routing list
* @var array
* @access public
*/  
    var $error_routes   = array();  
    /** 
     * Current class name 
     * URI中的Controller 
     * @var string 
     * @access public 
     */  
    var $class          = '';  
    /**
* Current method name
* The function called is displayed in the URI, the default is index()
* @var string
* @access public
*/  
    var $method         = 'index';  
    /**
* Sub-directory that contains the requested controller class
* The actual directory information in the URI
* @var string
* @access public
*/  
    var $directory      = '';  
    /**
* Default controller (and method if specific)
* Default controller
* @var string
* @access public
*/  
    var $default_controller;  
  
    /**
* Constructor
*
* Runs the route mapping function.
* Load and instantiate the config class and URI class
*/  
    function __construct()  
    {  
        $this->config =& load_class('Config', 'core');  
$this->uri =& load_class('URI', 'core');
log_message('debug', "Router Class Initialized");
}
//------------------------------------------------ -----------------------
/**
* Set the route mapping
*
* This function determines what should be served based on the URI request,
* as well as any "routes" that have been set in the routing config file.
* Set the default routing information. If there is no controller information, load the default controller according to the settings of routes.php,
*
* @access private
* @return void
*/
function _set_routing()
{
// Are query strings enabled in the config file? Normally CI doesn't utilize query strings
// since URI segments are more search-engine friendly, but they can optionally be used.
// If this feature is enabled, we will gather the directory/class/method a little differently
// If the project allows the query_strings format and requests the controller via $_GET, it will be routed in the query_string format
// Why do we need to judge whether the controller is specified by get method above?
// Actually it’s because if query_string request routing is allowed, but it is not configured under APPPATH/config/config.php
// Controller_trigger, function_trigger, directory_trigger cannot be used in query_strings form
// At this time, we will still use the "segment" form.
$segments = array();
if ($this->config->item('enable_query_strings') === TRUE AND isset($_GET[$this->config->item('controller_trigger')]))
{
                                        //Get the variable names passed by the directory name, control name and method name. These three items are defined in config/config.php.
if (isset($_GET[$this->config->item('directory_trigger')]))
                                                                 
$this->set_directory(trim($this->uri->_filter_uri($_GET[$this->config->item('directory_trigger')])));
$segments[] = $this->fetch_directory();
      }  
if (isset($_GET[$this->config->item('controller_trigger')]))
                                                                 
                     $this->set_class(trim($this->uri->_filter_uri($_GET[$this->config->item('controller_trigger')])));
$segments[] = $this->fetch_class();
      }  
if (isset($_GET[$this->config->item('function_trigger')]))
                                                                 
$this->set_method(trim($this->uri->_filter_uri($_GET[$this->config->item('function_trigger')])));
$segments[] = $this->fetch_method();
      }  
} }
// Load the routes.php file.
// Load routes.php under APPPATH according to the current environment
if (defined('ENVIRONMENT') AND is_file(APPPATH.'config/'.ENVIRONMENT.'/routes.php'))
{
include(APPPATH.'config/'.ENVIRONMENT.'/routes.php');
} }
elseif (is_file(APPPATH.'config/routes.php'))
{
include(APPPATH.'config/routes.php');
} }
// The following $route variable is defined in routes.php to set the default controller and default 404 page
$this->routes = ( ! isset($route) OR ! is_array($route)) ? array() : $route;
Unset ($ Route); // Drain it after use, dismantle the bridge across the river, and no mercy.
// Set the default controller so we can display it in the event
// the URI doesn't correlated to a valid controller.
// Set the default controller based on the configuration information just now. If not, it will be FLASE.
$this->default_controller = ( ! isset($this->routes['default_controller']) OR $this->routes['default_controller'] == '') ? FALSE : strtolower($this ->routes['default_controller']);
// Were there any query string segments? If so, we'll validate them and bail out since we're done.
// Verify whether any of the directory name, control name and method name is obtained through query string.
// If you get it, directly determine the route and return
if (count($segments) > 0)
{
// Determine and set the route.
Return $ this- & gt; _Validate_request ($ segments);
} }
// Fetch the complete URI string
// The purpose of this function is to detect and process the uri and get the information we need to determine the routing (such as index.php/index/welcome/1)
// Put the string index/welcome/1) into $this->uri->uri_string.
$this->uri->_fetch_uri_string();
// Is there a URI string? If not, the default controller specified in the "routes" file will be shown.
// If uri_string is empty, then set the route to the default.
/*if ($this->uri->uri_string == '')
{
return $this->_set_default_controller();
}*/
// Do we need to remove the URL suffix?
// Remove the URI suffix, because CI allows adding a suffix after the uri, but it is actually redundant for us to find routes and will have an impact, so remove it first.
$this->uri->_remove_url_suffix();
// Compile compile the segments into an array
// Split the uri into regular segments, filter each segment, and store it in $this->segments[]
$this->uri->_explode_segments();
// Parse any custom routing that may exist
// Process routing, set APPPATH/config/routes.php according to routing
$this->_parse_routes();
// Re-index the segment array so that it starts with 1 rather than 0
// Set the uri segment index to start from 1
$this->uri->_reindex_segments();
}
//------------------------------------------------ -----------------------
/**
* Set the default controller
* Set default controller
* @access private
* @return void
*/
function _set_default_controller()
{
// Read the default controller name from the configuration file in the Router::_set_routing() function, if not there will be FALSE
// Line 158 of this file
if ($this->default_controller === FALSE)
{
show_error("Unable to determine what should be displayed. A default route has not been specified in the routing file.");
} }
// Is the method being specified?
// Determine whether there is a specified method by judging whether $this->default_controller has /.
// If not set as the default method index
if (strpos($this->default_controller, '/') !== FALSE)
{
$x = explode('/', $this->default_controller);
$this->set_class($x[0]);
$this->set_method($x[1]);
$this->_set_request($x);
} }
else
{
$this->set_class($this->default_controller);
$this->set_method('index');
$this->_set_request(array($this->default_controller, 'index'));
} }
// re-index the routed segments array so it starts with 1 rather than 0
                                     // Re-index the segment so that the outgoing segments are saved starting with the subscript 1.
$this->uri->_reindex_segments();
log_message('debug', "No URI present. Default controller set.");
}
//------------------------------------------------ -----------------------
/**
* Set the Route
* Set routing
* This function takes take, prepare action an array of URI segments as
* input, and sets the current class/method
*
* @access private
* @param array
* @param bool
* @return void
*/
function _set_request($segments = array())
{
// The function of Router::_validate_request() is to detect and find a correctly existing route and determine it
$segments = $this->_validate_request($segments);
// This function is called by _set_default_controller. If you look at lines 216-230, you will find that $segments will definitely not be empty.
// Then the following two sentences may be to prevent this method from being called in other places and calling _set_default_controller() when the parameter is empty
// If you call it back, $segments will not be empty, because it is impossible to set routing when $segments is empty.
if (count($segments) == 0)
{
Return $ this- & gt; _set_default_Controller ();
} }
// Set the class name which is also the directory name
$this->set_class($segments[0]);
// If the method name exists, set it. If it does not exist, set it to index
if (isset($segments[1]))
{
// A standard request method
$this->set_method($segments[1]);
} }
else
{
                          // This lets the "routed" segment array identify that the default  
// index method is being used.
$segments[1] = 'index';
} }
// Update our "routed" segment array to contain the segments.
// Note: If there is no custom routing, this array will be
// identical to $this->uri->segments
// Update the segment array of the route. If there is no custom route here, it will be the same as $this->uri->segments
$this->uri->rsegments = $segments;
}
//------------------------------------------------ -----------------------
/**
* Validates validates the supplied segments. Attempts attempt to determine the path to
* the controller.
* Validates the provided segment and attempts to determine the path to the controller
* @access private
* @param array
* @return array
*/
function _validate_request($segments)
{
// ? ? ? ? ? ? ? ? ? ? ? ? ? ?
if (count($segments) == 0)
{
return $segments;
} }
// Does the requested controller exist in the root folder?
// Determine whether $segments[0] exists in the php file under the APPPATH/controllers/ folder
if (file_exists(APPPATH.'controllers/'.$segments[0].'.php'))
{
return $segments;
} }
// Is the controller in a sub-folder?
// Whether $segments[0] is a subdirectory under APPPATH/controllers
if (is_dir(APPPATH.'controllers/'.$segments[0]))
{
// Set the directory and remove it from the segment array
                                                                                                    // If it is indeed a directory, then the directory part of the route can be determined. Set directory
$this->set_directory($segments[0]);
                      // Remove the directory part. Further route search.
$segments = array_slice($segments, 1);
                                                                     
// If there are other paragraphs in addition to the directory in the URI request, it means that there is a specified controller.
if (count($segments) > 0)
                                                                 
                                                                                                                                                                                                 
// Determine the request's $ segments [0] whether the app file in the Appppath/Controller/subdirectory
                    if ( ! file_exists(APPPATH.'controllers/'.$this->fetch_directory().$segments[0].'.php'))
                                                                         
// There are two ways to report an error, one is default, and the other is self -determined.
                                                                                                                                                                                                                        // The following 404_override is a route defined in config/routes.php that cannot be found
// The default processing controller at that time, if we are defined, we call it.
                  if ( ! emptyempty($this->routes['404_override']))  
                                                                                 
                                                         
                                                                                                                                                                                                                        $x = explode('/', $this->routes['404_override']);
// Remove the directory part of the route that I just set, because the route is now the 404 route we defined.
$this->set_directory('');
// It can be seen here that the 404 routing we define is not allowed to be placed in a certain directory, and we can only be placed directly in controllers/below
                                                                                                                                                                 
                                                                                        $this->set_method(isset($x[1]) ? $x[1] : 'index');
                        return $x;                                          
                                                                       
              else  
                                                                                 
                                                                                                                                                                                                                                                                                                 
show_404($this->fetch_directory().$segments[0]);
                                                                       
        }  
      }  
        else  
                                                                 
// If we only have a directory in the URI request, we will come to this
                       // Is the method being specified in the route?
// The judgment below is just judging the $ this-& gt; default_controller. Is there any specified method?
              if (strpos($this->default_controller, '/') !== FALSE)
                                                                         
                                                                                                                                                                             
$this->set_class($x[0]);
                    $this->set_method($x[1]);  
                }  
                else  
                {  
                    $this->set_class($this->default_controller);  
                    $this->set_method('index');  
                }  
  
                // Does the default controller exist in the sub-folder?  
                // 判断APPPATH/controllers/目录/下面是否存在默认的方法  
                if ( ! file_exists(APPPATH.'controllers/'.$this->fetch_directory().$this->default_controller.'.php'))  
                {  
                    $this->directory = '';  
                    return array();  
                }  
  
            }  
  
            return $segments;  
        }  
  
  
        // If we've gotten 达到 this far 遥远的 it means that the URI does not correlate 使相关联 to a valid  
        // controller class.  We will now see if there is an override  
        // 来到这里,就说明了,即找不到controllers/下相应的控制器,也找不到这样的目录。那就报错咯。  
        if ( ! emptyempty($this->routes['404_override']))  
        {  
            $x = explode('/', $this->routes['404_override']);  
  
            $this->set_class($x[0]);  
            $this->set_method(isset($x[1]) ? $x[1] : 'index');  
  
            return $x;  
        }  
  
  
        // Nothing else to do at this point but show a 404  
        // 展示一个404页面  
        show_404($segments[0]);  
    }  
  
    // --------------------------------------------------------------------  
  
    /**
* Parse Routes
* Parse route
* This function matches any routes that may exist in
* the config/routes.php file against, against the URI to
* determine decision, determination if the class/method need to be remapped. Redraw the map
*
* @access private
* @return void
*/  
    function _parse_routes()  
    {  
        // Turn the segment array into a URI string  
        // 将segments数组转为uri字符串的形式  
        $uri = implode('/', $this->uri->segments);  
  
        // Is there a literal 文字的,字面的 match?  If so we're done  
        // 如果这个uri是routes.php中定义的那么。。。。。  
        if (isset($this->routes[$uri]))  
        {  
Return $ this-& gt; _Set_request (explode ('/', $ this-& gt; routes [$ uri]));
} }
// Loop through the route array looking for wild-cards
foreach ($this->routes as $key => $val)
{
// Convert Wild-CARDS To Regex Using the pass to make a regular conversion
// If the $ key contains: Any convert to.+,: Num convert to [0-9]+
$key = str_replace(':any', '.+', str_replace(':num', '[0-9]+', $key));
// Does the RegEx match? Perform regular matching
                                                                                                                                                                                                                        
                                                // To match. For example
                                                                                                                                                                                                    $route['show:any:num'] = 'anynum';                                      
// With these two configurations, we can match a uri like showaa123 to its corresponding value which is anynum
if (preg_match('#^'.$key.'$#', $uri))
                                                                 
                                                       
                                                                                                                                                                                                                                                             // Do we have a back-reference? If there is $ in $val and there is (
in $key
                                     // I don’t understand the function of this if. . . Waiting for rescue by experts
                                                                                                                                                                                                                                                                            to
                if (strpos($val, '$') !== FALSE AND strpos($key, '(') !== FALSE)
                                                                         
// Replace the characters in the URI to the $ value
$val = preg_replace('#^'.$key.'$#', $val, $uri);
        }  
Return $ this- & gt; _Set_request (explode ('/', $ value));
      }  
} }
// If we got this far it means we didn't encounter a
// matching route so we'll set the site default route
// If we get this far, it means we didn’t encounter a matching route
// So we will set the website default route
$this->_set_request($this->uri->segments);
}
//------------------------------------------------ -----------------------
/** 
     * Set the class name 
     * 
     * @access  public 
     * @param   string 
     * @return  void 
     */
function set_class($class)
{
$this->class = str_replace(array('/', '.'), '', $class);
}
//------------------------------------------------ -----------------------
/** 
     * Fetch the current class 
     * 
     * @access  public 
     * @return  string 
     */
function fetch_class()
{
return $this->class;
    }  
  
    // --------------------------------------------------------------------  
  
    /** 
     *  Set the method name 
     * 
     * @access  public 
     * @param   string 
     * @return  void 
     */  
    function set_method($method)  
    {  
        $this->method = $method;  
    }  
  
    // --------------------------------------------------------------------  
  
    /** 
     *  Fetch the current method 
     * 
     * @access  public 
     * @return  string 
     */  
    function fetch_method()  
    {  
        if ($this->method == $this->fetch_class())  
        {  
            return 'index';  
        }  
  
        return $this->method;  
    }  
  
    // --------------------------------------------------------------------  
  
    /** 
     *  Set the directory name 
     * 
     * @access  public 
     * @param   string 
     * @return  void 
     */  
    function set_directory($dir)  
    {  
        $this->directory = str_replace(array('/', '.'), '', $dir).'/';  
    }  
  
    // --------------------------------------------------------------------  
  
    /** 
     *  Fetch the sub-directory (if any) that contains the requested controller class 
     * 
     * @access  public 
     * @return  string 
     */  
    function fetch_directory()  
    {  
        return $this->directory;  
    }  
  
    // --------------------------------------------------------------------  
  
    /**
* Set the controller overrides
* Controller overrides
* This function can cover directories, controllers, and methods again.
* @access public
* @param array
* @return null
*/  
    function _set_overrides($routing)  
    {  
        if ( ! is_array($routing))  
        {  
            return;  
        }  
  
        if (isset($routing['directory']))  
        {  
            $this->set_directory($routing['directory']);  
        }  
  
        if (isset($routing['controller']) AND $routing['controller'] != '')  
        {  www.2cto.com
            $this->set_class($routing['controller']);  
        }  
  
        if (isset($routing['function']))  
        {  www.2cto.com
            $routing['function'] = ($routing['function'] == '') ? 'index' : $routing['function'];  
            $this->set_method($routing['function']);  
        }  
    }  
  
  
}  
// END Router Class  
  
/* End of file Router.php */  
/* Location: ./system/core/Router.php */  

www.bkjia.comtruehttp://www.bkjia.com/PHPjc/477681.htmlTechArticle[php] ?php if ( ! defined(BASEPATH)) exit(No direct script access allowed); /** * CodeIgniter * * An open source application development framework for PHP 5.1.6 or newer * * @packag...
Statement
The content of this article is voluntarily contributed by netizens, and the copyright belongs to the original author. This site does not assume corresponding legal responsibility. If you find any content suspected of plagiarism or infringement, please contact admin@php.cn
解读CRISP-ML(Q):机器学习生命周期流程解读CRISP-ML(Q):机器学习生命周期流程Apr 08, 2023 pm 01:21 PM

译者 | 布加迪审校 | 孙淑娟目前,没有用于构建和管理机器学习(ML)应用程序的标准实践。机器学习项目组织得不好,缺乏可重复性,而且从长远来看容易彻底失败。因此,我们需要一套流程来帮助自己在整个机器学习生命周期中保持质量、可持续性、稳健性和成本管理。图1. 机器学习开发生命周期流程使用质量保证方法开发机器学习应用程序的跨行业标准流程(CRISP-ML(Q))是CRISP-DM的升级版,以确保机器学习产品的质量。CRISP-ML(Q)有六个单独的阶段:1. 业务和数据理解2. 数据准备3. 模型

thinkphp是不是国产框架thinkphp是不是国产框架Sep 26, 2022 pm 05:11 PM

thinkphp是国产框架。ThinkPHP是一个快速、兼容而且简单的轻量级国产PHP开发框架,是为了简化企业级应用开发和敏捷WEB应用开发而诞生的。ThinkPHP从诞生以来一直秉承简洁实用的设计原则,在保持出色的性能和至简的代码的同时,也注重易用性。

Python 强大的任务调度框架 Celery!Python 强大的任务调度框架 Celery!Apr 12, 2023 pm 09:55 PM

什么是 celery这次我们来介绍一下 Python 的一个第三方模块 celery,那么 celery 是什么呢? celery 是一个灵活且可靠的,处理大量消息的分布式系统,可以在多个节点之间处理某个任务; celery 是一个专注于实时处理的任务队列,支持任务调度; celery 是开源的,有很多的使用者; celery 完全基于 Python 语言编写;所以 celery 本质上就是一个任务调度框架,类似于 Apache 的 airflow,当然 airflow 也是基于 Python

6个推荐的Python框架,用于构建可解释的人工智能系统(XAI)6个推荐的Python框架,用于构建可解释的人工智能系统(XAI)Apr 26, 2023 am 10:49 AM

AI就像一个黑匣子,能自己做出决定,但是人们并不清楚其中缘由。建立一个AI模型,输入数据,然后再输出结果,但有一个问题就是我们不能解释AI为何会得出这样的结论。需要了解AI如何得出某个结论背后的原因,而不是仅仅接受一个在没有上下文或解释的情况下输出的结果。可解释性旨在帮助人们理解:如何学习的?学到了什么?针对一个特定输入为什么会做出如此决策?决策是否可靠?在本文中,我将介绍6个用于可解释性的Python框架。SHAPSHapleyAdditiveexplanation(SHapleyAdditi

如何在PHP中使用AOP框架如何在PHP中使用AOP框架May 19, 2023 pm 01:21 PM

AOP(面向切面编程)是一种编程思想,用于解耦业务逻辑和横切关注点(如日志、权限等)。在PHP中,使用AOP框架可以简化编码,提高代码可维护性和可扩展性。本文将介绍在PHP中使用AOP框架的基本原理和实现方法。一、AOP的概念和原理面向切面编程,指的是将程序的业务逻辑和横切关注点分离开来,通过AOP框架来实现统一管理。横切关注点指的是在程序中需要重复出现并且

Microsoft .NET Framework 4.5.2、4.6 和 4.6.1 将于 2022 年 4 月终止支持Microsoft .NET Framework 4.5.2、4.6 和 4.6.1 将于 2022 年 4 月终止支持Apr 17, 2023 pm 02:25 PM

已安装Microsoft.NET版本4.5.2、4.6或4.6.1的MicrosoftWindows用户如果希望Microsoft将来通过产品更新支持该框架,则必须安装较新版本的Microsoft框架。据微软称,这三个框架都将在2022年4月26日停止支持。支持日期结束后,产品将不会收到“安全修复或技术支持”。大多数家庭设备通过Windows更新保持最新。这些设备已经安装了较新版本的框架,例如.NETFramework4.8。未自动更新的设备可能

KB5013943 2022 年 5 月更新使 Windows 11 上的应用程序崩溃KB5013943 2022 年 5 月更新使 Windows 11 上的应用程序崩溃Apr 16, 2023 pm 10:52 PM

如果你在Windows11上安装了2022年5月累积更新,你可能已经注意到你一直使用的许多应用程序都不像以前那样工作了。强制性安全更新KB5013943正在使某些使用.NET框架的应用程序崩溃。在某些情况下,用户会收到错误代码:0xc0000135。可选更新中报告了类似的问题,但并不普遍。随着2022年5月的更新,该错误似乎已进入生产渠道,这次有更多用户受到影响。崩溃在使用.NETFramework的应用程序中很常见,Discord或MicrosoftTeams等

朱军团队在清华开源了首个基于Transformer的多模态扩散大型模型,经过文本和图像改写全部完成。朱军团队在清华开源了首个基于Transformer的多模态扩散大型模型,经过文本和图像改写全部完成。May 08, 2023 pm 08:34 PM

据悉GPT-4将于本周发布,多模态将成为其一大亮点。当前的大语言模型正在成为理解各种模态的通用接口,能够根据不同模态信息来给出回复文本,但大语言模型生成的内容也仅仅局限于文本。另一方面,当前的扩散模型DALL・E2、Imagen、StableDiffusion等在视觉创作上掀起一场革命,但这些模型仅仅支持文到图的单一跨模态功能,离通用式生成模型还有一定距离。而多模态大模型将能够打通各种模态能力,实现任意模态之间转化,被认为是通用式生成模型的未来发展方向。清华大学计算机系朱军教授带领的TSAI

See all articles

Hot AI Tools

Undresser.AI Undress

Undresser.AI Undress

AI-powered app for creating realistic nude photos

AI Clothes Remover

AI Clothes Remover

Online AI tool for removing clothes from photos.

Undress AI Tool

Undress AI Tool

Undress images for free

Clothoff.io

Clothoff.io

AI clothes remover

AI Hentai Generator

AI Hentai Generator

Generate AI Hentai for free.

Hot Article

Repo: How To Revive Teammates
1 months agoBy尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. Energy Crystals Explained and What They Do (Yellow Crystal)
2 weeks agoBy尊渡假赌尊渡假赌尊渡假赌
Hello Kitty Island Adventure: How To Get Giant Seeds
1 months agoBy尊渡假赌尊渡假赌尊渡假赌

Hot Tools

Safe Exam Browser

Safe Exam Browser

Safe Exam Browser is a secure browser environment for taking online exams securely. This software turns any computer into a secure workstation. It controls access to any utility and prevents students from using unauthorized resources.

PhpStorm Mac version

PhpStorm Mac version

The latest (2018.2.1) professional PHP integrated development tool

MinGW - Minimalist GNU for Windows

MinGW - Minimalist GNU for Windows

This project is in the process of being migrated to osdn.net/projects/mingw, you can continue to follow us there. MinGW: A native Windows port of the GNU Compiler Collection (GCC), freely distributable import libraries and header files for building native Windows applications; includes extensions to the MSVC runtime to support C99 functionality. All MinGW software can run on 64-bit Windows platforms.

WebStorm Mac version

WebStorm Mac version

Useful JavaScript development tools

mPDF

mPDF

mPDF is a PHP library that can generate PDF files from UTF-8 encoded HTML. The original author, Ian Back, wrote mPDF to output PDF files "on the fly" from his website and handle different languages. It is slower than original scripts like HTML2FPDF and produces larger files when using Unicode fonts, but supports CSS styles etc. and has a lot of enhancements. Supports almost all languages, including RTL (Arabic and Hebrew) and CJK (Chinese, Japanese and Korean). Supports nested block-level elements (such as P, DIV),