Просмотр файла sys/plugins/classes/bbcode.class.php

Размер файла: 44.35Kb
<?php

/**
 * Обработчик BBCODE
 */
class bbcode {
    /*
      Описания тегов. Каждое описание - масив свойств:
      'handler'  - название функции - обработчика тегов.
      'is_close' - true, если тег всегда считается закрытым (например [hr]).
      'lbr'       - число переводов строк, которые следует игнорировать перед
      элементом.
      'rbr'      - число переводов строк, которые следует игнорировать после
      элемента.
      'ends'     - список тегов, начало которых обязательно закрывает данный.
      'permission_top_level' - true, если тегу разрешено находиться в корне
      дерева элементов.
      'children' - список тегов, которым разрешено быть вложенными в данный.
     */

    var $info_about_tags = array(
        'nobb' => array(
            'handler' => 'nobb_2html',
            'is_close' => false,
            'lbr' => 0,
            'rbr' => 0,
            'ends' => array(),
            'permission_top_level' => true,
            'children' => array()
        ),
        'u' => array(
            'handler' => 'u_2html',
            'is_close' => false,
            'lbr' => 0,
            'rbr' => 0,
            'ends' => array(),
            'permission_top_level' => true,
            'children' => array('big', 'u', 'i', 'b', 'small', 'url', 'vremja', 'user', 'no', 'smile', 'color', 'red', 'green', 'blue', 'yellow', 'gradient')
        ),
        'no' => array(
            'handler' => 'no_2html',
            'is_close' => false,
            'lbr' => 0,
            'rbr' => 0,
            'ends' => array(),
            'permission_top_level' => true,
            'children' => array('big', 'u', 'i', 'b', 'small', 'url', 'vremja', 'user', 'smile', 'color', 'red', 'green', 'blue', 'yellow', 'gradient')
        ),
        'i' => array(
            'handler' => 'i_2html',
            'is_close' => false,
            'lbr' => 0,
            'rbr' => 0,
            'ends' => array(),
            'permission_top_level' => true,
            'children' => array('big', 'u', 'i', 'b', 'small', 'url', 'vremja', 'user', 'no', 'smile', 'color', 'red', 'green', 'blue', 'yellow', 'gradient')
        ),
        'b' => array(
            'handler' => 'b_2html',
            'is_close' => false,
            'lbr' => 0,
            'rbr' => 0,
            'ends' => array(),
            'permission_top_level' => true,
            'children' => array('big', 'u', 'i', 'b', 'small', 'url', 'vremja', 'user', 'no', 'smile', 'color', 'red', 'green', 'blue', 'yellow', 'gradient')
        ),
        'big' => array(
            'handler' => 'big_2html',
            'is_close' => false,
            'lbr' => 0,
            'rbr' => 0,
            'ends' => array(),
            'permission_top_level' => true,
            'children' => array('big', 'u', 'i', 'b', 'small', 'url', 'vremja', 'user', 'no', 'smile', 'color', 'red', 'green', 'blue', 'yellow', 'gradient')
        ),
        'mark' => array(
            'handler' => 'mark_2html',
            'is_close' => false,
            'lbr' => 0,
            'rbr' => 0,
            'ends' => array(),
            'permission_top_level' => true,
            'children' => array('big', 'u', 'i', 'b', 'small', 'url', 'vremja', 'user', 'no', 'smile')
        ),
        'small' => array(
            'handler' => 'small_2html',
            'is_close' => false,
            'lbr' => 0,
            'rbr' => 0,
            'ends' => array(),
            'permission_top_level' => true,
            'children' => array('big', 'u', 'i', 'b', 'small', 'url', 'vremja', 'user', 'no', 'smile', 'color', 'red', 'green', 'blue', 'yellow', 'gradient')
        ),
        'user' => array(
            'handler' => 'user_2html',
            'is_close' => false,
            'lbr' => 0,
            'rbr' => 0,
            'ends' => array(),
            'permission_top_level' => true,
            'children' => array()
        ),
        'smile' => array(
            'handler' => 'smile_2html',
            'is_close' => false,
            'lbr' => 0,
            'rbr' => 0,
            'ends' => array(),
            'permission_top_level' => true,
            'children' => array()
        ),
        'spoiler' => array(
            'handler' => 'spoiler_2html',
            'is_close' => false,
            'lbr' => 0,
            'rbr' => 0,
            'ends' => array(),
            'permission_top_level' => true,
            'children' => array('big', 'u', 'i', 'b', 'small', 'user', 'smile', 'url', 'vremja', 'no', 'php', 'hide', 'spoiler', 'color', 'red', 'green', 'blue', 'yellow', 'localimg', 'gradient')
        ),
        'localimg' => array(
            'handler' => 'img_2html',
            'is_close' => false,
            'lbr' => 0,
            'rbr' => 0,
            'ends' => array(),
            'permission_top_level' => true,
            'children' => array()
        ),
        'url' => array(
            'handler' => 'url_2html',
            'is_close' => false,
            'lbr' => 0,
            'rbr' => 0,
            'ends' => array(),
            'permission_top_level' => true,
            'children' => array('big', 'u', 'i', 'b', 'small', 'smile', 'color', 'red', 'green', 'blue', 'yellow', 'gradient')
        ),
        'php' => array(
            'handler' => 'php_2html',
            'is_close' => false,
            'lbr' => 0,
            'rbr' => 1,
            'ends' => array(),
            'permission_top_level' => true,
            'children' => array()
        ),
        'quote' => array(
            'handler' => 'quote_2html',
            'is_close' => false,
            'lbr' => 0,
            'rbr' => 0,
            'ends' => array(),
            'permission_top_level' => true,
            'children' => array('big', 'u', 'i', 'b', 'small', 'user', 'smile', 'url', 'quote', 'vremja', 'no', 'php', 'gradient', 'localimg', 'spoiler')
        ),
        'color' => array(
            'handler' => 'color_2html',
            'is_close' => false,
            'lbr' => 0,
            'rbr' => 0,
            'ends' => array('user', 'smile', 'quote'),
            'permission_top_level' => true,
            'children' => array('big', 'u', 'i', 'b', 'small', 'vremja', 'no', 'url')
        ),
        'gradient' => array(
            'handler' => 'gradient_2html',
            'is_close' => false,
            'lbr' => 0,
            'rbr' => 0,
            'ends' => array('user', 'smile', 'url', 'quote'),
            'permission_top_level' => true,
            'children' => array('big', 'u', 'i', 'b', 'small', 'vremja', 'no')
        ),
        'red' => array(
            'handler' => 'color_2html',
            'is_close' => false,
            'lbr' => 0,
            'rbr' => 0,
            'ends' => array('user', 'smile', 'quote'),
            'permission_top_level' => true,
            'children' => array('big', 'u', 'i', 'b', 'small', 'vremja', 'no', 'url')
        ),
        'green' => array(
            'handler' => 'color_2html',
            'is_close' => false,
            'lbr' => 0,
            'rbr' => 0,
            'ends' => array('user', 'smile', 'quote'),
            'permission_top_level' => true,
            'children' => array('big', 'u', 'i', 'b', 'small', 'vremja', 'no', 'url')
        ),
        'blue' => array(
            'handler' => 'color_2html',
            'is_close' => false,
            'lbr' => 0,
            'rbr' => 0,
            'ends' => array('user', 'smile', 'quote'),
            'permission_top_level' => true,
            'children' => array('big', 'u', 'i', 'b', 'small', 'vremja', 'no', 'url')
        ),
        'yellow' => array(
            'handler' => 'color_2html',
            'is_close' => false,
            'lbr' => 0,
            'rbr' => 0,
            'ends' => array('user', 'smile', 'quote'),
            'permission_top_level' => true,
            'children' => array('big', 'u', 'i', 'b', 'small', 'vremja', 'no', 'url')
        ),
        'vremja' => array(
            'handler' => 'vremja_2html',
            'is_close' => false,
            'lbr' => 0,
            'rbr' => 0,
            'ends' => array(),
            'permission_top_level' => true,
            'children' => array()
        ),
    );
    var $mnemonics = array();
    var $syntax = array();

    function get_array_of_tokens($code) {
        $length = strlen($code);
        $tokens = array();
        $token_key = -1;
        $type_of_char = null;
        for ($i = 0; $i < $length; ++$i) {
            $previous_type = $type_of_char;
            switch ($code{$i}) {
                case '[':
                    $type_of_char = 0;
                    break;
                case ']':
                    $type_of_char = 1;
                    break;
                case '"':
                    $type_of_char = 2;
                    break;
                case "'":
                    $type_of_char = 3;
                    break;
                case "=":
                    $type_of_char = 4;
                    break;
                case '/':
                    $type_of_char = 5;
                    break;
                case ' ':
                    $type_of_char = 6;
                    break;
                case "\t":
                    $type_of_char = 6;
                    break;
                case "\n":
                    $type_of_char = 6;
                    break;
                case "\r":
                    $type_of_char = 6;
                    break;
                case "\0":
                    $type_of_char = 6;
                    break;
                case "\x0B":
                    $type_of_char = 6;
                    break;
                default:
                    $type_of_char = 7;
            }
            if (7 == $previous_type && $type_of_char != $previous_type) {
                $word = strtolower($tokens[$token_key][1]);
                if (isset($this->info_about_tags[$word])) {
                    $tokens[$token_key][0] = 8;
                }
            }
            switch ($type_of_char) {
                case 6:
                    if (6 == $previous_type) {
                        $tokens[$token_key][1] .= $code{$i};
                    } else {
                        $tokens[++$token_key] = array(6, $code{$i});
                    }
                    break;
                case 7:
                    if (7 == $previous_type) {
                        $tokens[$token_key][1] .= $code{$i};
                    } else {
                        $tokens[++$token_key] = array(7, $code{$i});
                    }
                    break;
                default:
                    $tokens[++$token_key] = array($type_of_char, $code{$i});
            }
        }
        return $tokens;
    }

    function __construct($code) {

        $finite_automaton = array(
            // Предыдущие |   Состояния для текущих событий (лексем)   |
            //  состояния |  0 |  1 |  2 |  3 |  4 |  5 |  6 |  7 |  8 |
            0 => array(1, 0, 0, 0, 0, 0, 0, 0, 0)
            , 1 => array(2, 3, 3, 3, 3, 4, 3, 3, 5)
            , 2 => array(2, 3, 3, 3, 3, 4, 3, 3, 5)
            , 3 => array(1, 0, 0, 0, 0, 0, 0, 0, 0)
            , 4 => array(2, 6, 3, 3, 3, 3, 3, 3, 7)
            , 5 => array(2, 6, 3, 3, 8, 9, 10, 3, 3)
            , 6 => array(1, 0, 0, 0, 0, 0, 0, 0, 0)
            , 7 => array(2, 6, 3, 3, 3, 3, 3, 3, 3)
            , 8 => array(13, 13, 11, 12, 13, 13, 14, 13, 13)
            , 9 => array(2, 6, 3, 3, 3, 3, 3, 3, 3)
            , 10 => array(2, 6, 3, 3, 8, 9, 3, 15, 15)
            , 11 => array(16, 16, 17, 16, 16, 16, 16, 16, 16)
            , 12 => array(18, 18, 18, 17, 18, 18, 18, 18, 18)
            , 13 => array(19, 6, 19, 19, 19, 19, 17, 19, 19)
            , 14 => array(2, 3, 11, 12, 13, 13, 3, 13, 13)
            , 15 => array(2, 6, 3, 3, 8, 9, 10, 3, 3)
            , 16 => array(16, 16, 17, 16, 16, 16, 16, 16, 16)
            , 17 => array(2, 6, 3, 3, 3, 9, 20, 15, 15)
            , 18 => array(18, 18, 18, 17, 18, 18, 18, 18, 18)
            , 19 => array(19, 6, 19, 19, 19, 19, 20, 19, 19)
            , 20 => array(2, 6, 3, 3, 3, 9, 3, 15, 15)
        );
        // Получаем массив лексем:
        $array_of_tokens = $this->get_array_of_tokens($code);
        // Сканируем его с помощью построенного автомата:
        $mode = 0;
        $result = array();
        $tag_decomposition = array();
        $token_key = -1;
        foreach ($array_of_tokens as $token) {
            $previous_mode = $mode;
            $mode = $finite_automaton[$previous_mode][$token[0]];
            switch ($mode) {
                case 0:
                    if (-1 < $token_key && 'text' == $result[$token_key]['type']) {
                        $result[$token_key]['str'] .= $token[1];
                    } else {
                        $result[++$token_key] = array(
                            'type' => 'text',
                            'str' => $token[1]
                        );
                    }
                    break;
                case 1:
                    $tag_decomposition['name'] = '';
                    $tag_decomposition['type'] = '';
                    $tag_decomposition['str'] = '[';
                    $tag_decomposition['layout'][] = array(0, '[');
                    break;
                case 2:
                    if (-1 < $token_key && 'text' == $result[$token_key]['type']) {
                        $result[$token_key]['str'] .= $tag_decomposition['str'];
                    } else {
                        $result[++$token_key] = array(
                            'type' => 'text',
                            'str' => $tag_decomposition['str']
                        );
                    }
                    $tag_decomposition = array();
                    $tag_decomposition['name'] = '';
                    $tag_decomposition['type'] = '';
                    $tag_decomposition['str'] = '[';
                    $tag_decomposition['layout'][] = array(0, '[');
                    break;
                case 3:
                    if (-1 < $token_key && 'text' == $result[$token_key]['type']) {
                        $result[$token_key]['str'] .= $tag_decomposition['str'];
                        $result[$token_key]['str'] .= $token[1];
                    } else {
                        $result[++$token_key] = array(
                            'type' => 'text',
                            'str' => $tag_decomposition['str'] . $token[1]
                        );
                    }
                    $tag_decomposition = array();
                    break;
                case 4:
                    $tag_decomposition['type'] = 'close';
                    $tag_decomposition['str'] .= '/';
                    $tag_decomposition['layout'][] = array(1, '/');
                    break;
                case 5:
                    $tag_decomposition['type'] = 'open';
                    $name = strtolower($token[1]);
                    $tag_decomposition['name'] = $name;
                    $tag_decomposition['str'] .= $token[1];
                    $tag_decomposition['layout'][] = array(2, $token[1]);
                    $tag_decomposition['attrib'][$name] = '';
                    break;
                case 6:
                    if (!isset($tag_decomposition['name'])) {
                        $tag_decomposition['name'] = '';
                    }
                    if (13 == $previous_mode || 19 == $previous_mode) {
                        $tag_decomposition['layout'][] = array(7, $value);
                    }
                    $tag_decomposition['str'] .= ']';
                    $tag_decomposition['layout'][] = array(0, ']');
                    $result[++$token_key] = $tag_decomposition;
                    $tag_decomposition = array();
                    break;
                case 7:
                    $tag_decomposition['name'] = strtolower($token[1]);
                    $tag_decomposition['str'] .= $token[1];
                    $tag_decomposition['layout'][] = array(2, $token[1]);
                    break;
                case 8:
                    $tag_decomposition['str'] .= '=';
                    $tag_decomposition['layout'][] = array(3, '=');
                    break;
                case 9:
                    $tag_decomposition['type'] = 'open/close';
                    $tag_decomposition['str'] .= '/';
                    $tag_decomposition['layout'][] = array(1, '/');
                    break;
                case 10:
                    $tag_decomposition['str'] .= $token[1];
                    $tag_decomposition['layout'][] = array(4, $token[1]);
                    break;
                case 11:
                    $tag_decomposition['str'] .= '"';
                    $tag_decomposition['layout'][] = array(5, '"');
                    break;
                case 12:
                    $tag_decomposition['str'] .= "'";
                    $tag_decomposition['layout'][] = array(5, "'");
                    break;
                case 13:
                    $tag_decomposition['attrib'][$name] = $token[1];
                    $value = $token[1];
                    $tag_decomposition['str'] .= $token[1];
                    break;
                case 14:
                    $tag_decomposition['str'] .= $token[1];
                    $tag_decomposition['layout'][] = array(4, $token[1]);
                    break;
                case 15:
                    $name = strtolower($token[1]);
                    $tag_decomposition['str'] .= $token[1];
                    $tag_decomposition['layout'][] = array(6, $token[1]);
                    $tag_decomposition['attrib'][$name] = '';
                    break;
                case 16:
                    $tag_decomposition['str'] .= $token[1];
                    $tag_decomposition['attrib'][$name] .= $token[1];
                    @$value .= $token[1];
                    break;
                case 17:
                    $tag_decomposition['str'] .= $token[1];
                    $tag_decomposition['layout'][] = array(7, @$value);
                    @$value = '';
                    $tag_decomposition['layout'][] = array(5, $token[1]);
                    break;
                case 18:
                    $tag_decomposition['str'] .= $token[1];
                    $tag_decomposition['attrib'][$name] .= $token[1];
                    @$value .= $token[1];
                    break;
                case 19:
                    $tag_decomposition['str'] .= $token[1];
                    $tag_decomposition['attrib'][$name] .= $token[1];
                    @$value .= $token[1];
                    break;
                case 20:
                    $tag_decomposition['str'] .= $token[1];
                    if (13 == $previous_mode || 19 == $previous_mode) {
                        $tag_decomposition['layout'][] = array(7, $value);
                    }
                    $value = '';
                    $tag_decomposition['layout'][] = array(4, $token[1]);
                    break;
            }
        }
        if (count($tag_decomposition)) {
            if (-1 < $token_key && 'text' == $result[$token_key]['type']) {
                $result[$token_key]['str'] .= $tag_decomposition['str'];
            } else {
                $result[++$token_key] = array(
                    'type' => 'text',
                    'str' => $tag_decomposition['str']
                );
            }
        }
        $this->syntax = $result;
    }

    // Функция возвращает нормализует и возвращает дерево элементов
    function get_tree_of_elems() {
        /* Первый этап нормализации: превращаем $this -> syntax в правильную
          скобочную структуру */
        $structure = array();
        $structure_key = -1;
        $level = 0;
        $open_tags = array();
        foreach ($this->syntax as $syntax_key => $val) {
            unset($val['layout']);
            switch ($val['type']) {
                case 'text':
                    $type = (-1 < $structure_key) ? $structure[$structure_key]['type'] : false;
                    if ('text' == $type) {
                        $structure[$structure_key]['str'] .= $val['str'];
                    } else {
                        $structure[++$structure_key] = $val;
                        $structure[$structure_key]['level'] = $level;
                    }
                    break;
                case 'open/close':
                    foreach (array_reverse($open_tags, true) as $ult_key => $ultimate) {
                        $ends = $this->info_about_tags[$ultimate]['ends'];
                        if (in_array($val['name'], $ends)) {
                            $structure[++$structure_key] = array(
                                'type' => 'close',
                                'name' => $ultimate,
                                'str' => '',
                                'level' => --$level
                            );
                            unset($open_tags[$ult_key]);
                        } else {
                            break;
                        }
                    }
                    $structure[++$structure_key] = $val;
                    $structure[$structure_key]['level'] = $level;
                    break;
                case 'open':
                    foreach (array_reverse($open_tags, true) as $ult_key => $ultimate) {
                        $ends = $this->info_about_tags[$ultimate]['ends'];
                        if (in_array($val['name'], $ends)) {
                            $structure[++$structure_key] = array(
                                'type' => 'close',
                                'name' => $ultimate,
                                'str' => '',
                                'level' => --$level
                            );
                            unset($open_tags[$ult_key]);
                        } else {
                            break;
                        }
                    }
                    if ($this->info_about_tags[$val['name']]['is_close']) {
                        $val['type'] = 'open/close';
                        $structure[++$structure_key] = $val;
                        $structure[$structure_key]['level'] = $level;
                    } else {
                        $structure[++$structure_key] = $val;
                        $structure[$structure_key]['level'] = $level++;
                        $open_tags[] = $val['name'];
                    }
                    break;
                case 'close':
                    if (!count($open_tags)) {
                        $type = (-1 < $structure_key) ? $structure[$structure_key]['type'] : false;
                        if ('text' == $type) {
                            $structure[$structure_key]['str'] .= $val['str'];
                        } else {
                            $structure[++$structure_key] = array(
                                'type' => 'text',
                                'str' => $val['str'],
                                'level' => 0
                            );
                        }
                        break;
                    }
                    if (!$val['name']) {
                        end($open_tags);
                        list($ult_key, $ultimate) = each($open_tags);
                        $val['name'] = $ultimate;
                        $structure[++$structure_key] = $val;
                        $structure[$structure_key]['level'] = --$level;
                        unset($open_tags[$ult_key]);
                        break;
                    }
                    if (!in_array($val['name'], $open_tags)) {
                        $type = (-1 < $structure_key) ? $structure[$structure_key]['type'] : false;
                        if ('text' == $type) {
                            $structure[$structure_key]['str'] .= $val['str'];
                        } else {
                            $structure[++$structure_key] = array(
                                'type' => 'text',
                                'str' => $val['str'],
                                'level' => $level
                            );
                        }
                        break;
                    }
                    foreach (array_reverse($open_tags, true) as $ult_key => $ultimate) {
                        if ($ultimate != $val['name']) {
                            $structure[++$structure_key] = array(
                                'type' => 'close',
                                'name' => $ultimate,
                                'str' => '',
                                'level' => --$level
                            );
                            unset($open_tags[$ult_key]);
                        } else {
                            break;
                        }
                    }
                    $structure[++$structure_key] = $val;
                    $structure[$structure_key]['level'] = --$level;
                    unset($open_tags[$ult_key]);
            }
        }
        foreach (array_reverse($open_tags, true) as $ult_key => $ultimate) {
            $structure[++$structure_key] = array(
                'type' => 'close',
                'name' => $ultimate,
                'str' => '',
                'level' => --$level
            );
            unset($open_tags[$ult_key]);
        }
        /* Второй этап нормализации: Отслеживаем, имеют ли элементы
          неразрешенные подэлементы. Соответственно этому исправляем
          $structure. */
        $normalized = array();
        $normal_key = -1;
        $level = 0;
        $open_tags = array();
        $not_tags = array();
        foreach ($structure as $structure_key => $val) {
            switch ($val['type']) {
                case 'text':
                    $type = (-1 < $normal_key) ? $normalized[$normal_key]['type'] : false;
                    if ('text' == $type) {
                        $normalized[$normal_key]['str'] .= $val['str'];
                    } else {
                        $normalized[++$normal_key] = $val;
                        $normalized[$normal_key]['level'] = $level;
                    }
                    break;
                case 'open/close':
                    $is_open = count($open_tags);
                    end($open_tags);
                    $info = $this->info_about_tags[$val['name']];
                    $children = $is_open ? $this->info_about_tags[current($open_tags)]['children'] : array();
                    $not_normal = !$level && !$info['permission_top_level']
                            || $is_open && !in_array($val['name'], $children);
                    if ($not_normal) {
                        $type = (-1 < $normal_key) ? $normalized[$normal_key]['type'] : false;
                        if ('text' == $type) {
                            $normalized[$normal_key]['str'] .= $val['str'];
                        } else {
                            $normalized[++$normal_key] = array(
                                'type' => 'text',
                                'str' => $val['str'],
                                'level' => $level
                            );
                        }
                        break;
                    }
                    $normalized[++$normal_key] = $val;
                    $normalized[$normal_key]['level'] = $level;
                    break;
                case 'open':
                    $is_open = count($open_tags);
                    end($open_tags);
                    $info = $this->info_about_tags[$val['name']];
                    $children = $is_open ? $this->info_about_tags[current($open_tags)]['children'] : array();
                    $not_normal = !$level && !$info['permission_top_level']
                            || $is_open && !in_array($val['name'], $children);
                    if ($not_normal) {
                        $not_tags[$val['level']] = $val['name'];
                        $type = (-1 < $normal_key) ? $normalized[$normal_key]['type'] : false;
                        if ('text' == $type) {
                            $normalized[$normal_key]['str'] .= $val['str'];
                        } else {
                            $normalized[++$normal_key] = array(
                                'type' => 'text',
                                'str' => $val['str'],
                                'level' => $level
                            );
                        }
                        break;
                    }
                    $normalized[++$normal_key] = $val;
                    $normalized[$normal_key]['level'] = $level++;
                    $ult_key = count($open_tags);
                    $open_tags[$ult_key] = $val['name'];
                    break;
                case 'close':
                    $not_normal = isset($not_tags[$val['level']])
                            && $not_tags[$val['level']] = $val['name'];
                    if ($not_normal) {
                        unset($not_tags[$val['level']]);
                        $type = (-1 < $normal_key) ? $normalized[$normal_key]['type'] : false;
                        if ('text' == $type) {
                            $normalized[$normal_key]['str'] .= $val['str'];
                        } else {
                            $normalized[++$normal_key] = array(
                                'type' => 'text',
                                'str' => $val['str'],
                                'level' => $level
                            );
                        }
                        break;
                    }
                    $normalized[++$normal_key] = $val;
                    $normalized[$normal_key]['level'] = --$level;
                    $ult_key = count($open_tags) - 1;
                    unset($open_tags[$ult_key]);
                    break;
            }
        }
        // Формируем дерево элементов
        $result = array();
        $result_key = -1;
        $open_tags = array();
        $val_key = -1;
        foreach ($normalized as $normal_key => $val) {
            switch ($val['type']) {
                case 'text':
                    if (!$val['level']) {
                        $result[++$result_key] = array(
                            'type' => 'text',
                            'str' => $val['str']
                        );
                        break;
                    }
                    $open_tags[$val['level'] - 1]['val'][] = array(
                        'type' => 'text',
                        'str' => $val['str']
                    );
                    break;
                case 'open/close':
                    if (!$val['level']) {
                        $result[++$result_key] = array(
                            'type' => 'item',
                            'name' => $val['name'],
                            'attrib' => $val['attrib'],
                            'val' => array()
                        );
                        break;
                    }
                    $open_tags[$val['level'] - 1]['val'][] = array(
                        'type' => 'item',
                        'name' => $val['name'],
                        'attrib' => $val['attrib'],
                        'val' => array()
                    );
                    break;
                case 'open':
                    $open_tags[$val['level']] = array(
                        'type' => 'item',
                        'name' => $val['name'],
                        'attrib' => $val['attrib'],
                        'val' => array()
                    );
                    break;
                case 'close':
                    if (!$val['level']) {
                        $result[++$result_key] = $open_tags[0];
                        unset($open_tags[0]);
                        break;
                    }
                    $open_tags[$val['level'] - 1]['val'][] = $open_tags[$val['level']];
                    unset($open_tags[$val['level']]);
                    break;
            }
        }
        return $result;
    }

    /*
      Функция мнемонизирует HTML-код, вставляет в текст разрывы <br />, смайлики и
      "автоматические ссылки".
     */

    function insert_smiles($text) {


        $text = htmlspecialchars($text, ENT_QUOTES, 'UTF-8');


        $text = nl2br($text);
        //$text = str_replace('  ', '&nbsp;&nbsp;', $text);

        foreach ($this->mnemonics as $mnemonic => $value) {
            $text = str_replace($mnemonic, $value, $text);
        }
        return $text;
    }

    // Функция конвертит дерево элементов BBCode в HTML и возвращает результат
    function get_html($tree_of_elems = false) {
        if (!is_array($tree_of_elems)) {
            $tree_of_elems = $this->get_tree_of_elems();
        }
        $result = '';
        $lbr = 0;
        $rbr = 0;
        foreach ($tree_of_elems as $elem) {
            if ('text' == $elem['type']) {
                $elem['str'] = $this->insert_smiles($elem['str']);
                for ($i = 0; $i < $rbr; ++$i) {
                    $elem['str'] = ltrim($elem['str']);
                    if ('<br />' == substr($elem['str'], 0, 6)) {
                        $elem['str'] = substr_replace($elem['str'], '', 0, 6);
                    }
                }
                $result .= $elem['str'];
            } else {
                $lbr = $this->info_about_tags[$elem['name']]['lbr'];
                $rbr = $this->info_about_tags[$elem['name']]['rbr'];
                for ($i = 0; $i < $lbr; ++$i) {
                    $result = rtrim($result);
                    if ('<br />' == substr($result, -6)) {
                        $result = substr_replace($result, '', -6, 6);
                    }
                }
                $func_name = $this->info_about_tags[$elem['name']]['handler'];
                $result .= call_user_func(array(&$this, $func_name), $elem);
            }
        }
        return $result;
    }

    function nobb_2html($elem) {
        return $this->get_html($elem['val']);
    }

    function u_2html($elem) {
        return '<span style="text-decoration:underline">' . $this->get_html($elem['val']) . '</span>';
    }

    function i_2html($elem) {
        return '<span style="font-style:italic">' . $this->get_html($elem['val']) . '</span>';
    }

    function b_2html($elem) {
        return '<span style="font-weight:bolder">' . $this->get_html($elem['val']) . '</span>';
    }

    function no_2html($elem) {
        return '<span style="text-decoration:line-through">' . $this->get_html($elem['val']) . '</span>';
    }

    function mark_2html($elem) {
        return '<span class="mark">' . $this->get_html($elem['val']) . '</span>';
    }

    function big_2html($elem) {
        return '<span style="font-size:larger">' . $this->get_html($elem['val']) . '</span>';
    }

    function small_2html($elem) {
        return '<span style="font-size:smaller">' . $this->get_html($elem['val']) . '</span>';
    }

    function vremja_2html($elem) {
        return vremja($elem['val'][0]['str']);
        //return '<span style="font-size:smaller">' . $this->get_html($elem['val']) . '</span>';
    }

    function user_2html($elem) {

        //return '<pre>'.print_r($elem,1).'</pre>';

        $ank = new user((int) $elem['val'][0]['str']);
        return '<a href="/profile.view.php?id=' . $ank->id . '">' . $ank->nick() . '</a>';
    }

    function smile_2html($elem) {
        return smiles::bbcode($elem['val'][0]['str']);
        // return $arr[0];
    }

    function spoiler_2html($elem) {
        //return '<pre>'.print_r($elem,1).'</pre>';

        if (!empty($elem['attrib']['title'])) {
            $title = $elem['attrib']['title'];
        } elseif (!empty($elem['attrib']['spoiler'])) {
            $title = $elem['attrib']['spoiler'];
        } else {
            $title = __('Скрытый текст');
        }
        return '<div class="DCMS_spoiler"><span class="DCMS_spoiler_title">' . for_value($title) . '</span><div class="DCMS_spoiler_content">' . $this->get_html($elem['val']) . '</div></div>';


        return smiles::bbcode($elem['val'][0]['str']);
        // return $arr[0];
    }

    function url_2html($elem) {

        if (empty($elem['attrib']['url']) || empty($elem['val'][0]['str'])) {
            return false;
        }
        $aturl = $elem['attrib']['url'];
        $text = @$elem['val'][0]['str'];

        if (!$text) {
            return false;
        }

        $text = text::substr($text, 40);


        global $dcms;
        $aturl = str_replace(array("\n", "\r", "\t"), '', $aturl);

        if (preg_match('#^ *(javascript|data)#', $aturl))
            return '!!! Javascript запрещен !!!';

        if (preg_match('#://#', $aturl)) {
            // внешняя ссылка



            $url = 'http://' . $_SERVER ['HTTP_HOST'] . '/link.ext.php?url=' . urlencode($aturl);
            $new_window = @$dcms->browser_type == 'web' ? ' target="_blank"' : '';
            if ($parse_url = @parse_url($aturl)) {
                if (!empty($parse_url['host']) && @$dcms->subdomain_main && strpos($parse_url['host'], '.' . $dcms->subdomain_main) !== false) {

                    if (@$dcms->subdomain_replace_url) {
                        // вырезаем поддомен из локальных ссылок
                        $aturl = str_replace($parse_url['host'], $dcms->subdomain_main, $aturl);
                        $url = 'http://' . $_SERVER ['HTTP_HOST'] . '/link.ext.php?url=' . urlencode($aturl);
                    }

                    $new_window = '';
                }
            }


            return '<a' . $new_window . ' href="' . $url . '">' . for_value($text) . '</a>';
        } else {
            // внутренняя 
            $url = preg_replace('#^http://' . preg_quote($_SERVER ['HTTP_HOST']) . '(/|$)#ui', '/', $aturl);
            return '<a href="' . for_value($url) . '">' . for_value($text) . '</a>';
        }
    }

    function quote_2html($elem) {
        if (!empty($elem['attrib']['quote'])) {
            if (preg_match('#^([0-9]+):([0-9]+):(.+)$#ui', $elem['attrib']['quote'], $log)) {
                $time = (int) $log [1];
                $ank = new user((int) $log[2]);
            } else {
                return '<div class="DCMS_quote">' . $this->get_html($elem['val']) . '</div>';
            }
        } elseif (!empty($elem['attrib']['time']) && !empty($elem['attrib']['id_user'])) {
            $time = (int) $elem['attrib']['time'];
            $ank = new user((int) $elem['attrib']['id_user']);
        } else {
            return '<div class="DCMS_quote">' . $this->get_html($elem['val']) . '</div>';
        }


        if ($time && $ank->id) {
            $title = "<span class='DCMS_quote_title'><a href='/profile.view.php?id=$ank->id'>" . $ank->nick() . "</a> (" . vremja($time) . ")</span>:";
        } else {
            $title = '';
        }
        return '<div class="DCMS_quote">' . $title . $this->get_html($elem['val']) . '</div>';
    }

    function img_2html($elem) {
        static $design = false;
        if ($design === false)
            $design = new design ();

        if (empty($elem['attrib']['file'])) {
            return false;
        }
        if (empty($elem['val'][0]['str'])) {
            return false;
        }
        $file = basename($elem['attrib']['file'], '.jpg');

        if (!@file_exists(H . '/sys/files/.bbcode/' . $file . '.jpg')) {
            return false;
        }

        $file = new files_file(FILES . '/.bbcode', $file . '.jpg');

        if ($screen = $file->getScreen($design->img_max_width())) {
            return '<img src="' . $screen . '" alt="' . for_value($elem['val'][0]['str']) . '" />';
        }
    }

    function php_2html($elem) {
        $code = "<?php\n" . trim(preg_replace('#^\<\?(php)?|\?\>$#i', '', @$elem['val'][0]['str'])) . "\n?>";
        $code = highlight_string($code, true);
        $code = preg_replace('#<code>(.*?)</code>#si', '<div class="DCMS_phpcode">\\1</div>', $code);
        $code = preg_replace("#[\n\r\t]+#", '', $code);
        return $code;
    }

    function color_2html($elem) {

        // название тега является цветом (так как сама по себе данная функция вызваться не может, дополнительную проверку делать не будем.)
        if ($elem['name'] !== 'color') {
            return '<span style="color:' . $elem['name'] . '">' . $this->get_html($elem['val']) . '</span>';
        }

        // нет аттрибута с цветом
        if (empty($elem['attrib']['color'])) {
            return '[color="' . __('Цвет не указан') . '"]' . $this->get_html($elem['val']) . '[/color]';
        }

        // шестнадцатеричное указание цвета
        if (preg_match('/^#([0-9a-f]{6}|[0-9a-f]{3})$/ui', $elem['attrib']['color'])) {
            return '<span style="color:' . $elem['attrib']['color'] . '">' . $this->get_html($elem['val']) . '</span>';
        }

        // не корректный цвет
        return '[color="' . for_value($elem['attrib']['color']) . '"]' . $this->get_html($elem['val']) . '[/color]';
    }

    function gradient_2html($elem) {

        if (empty($elem['val'][0]['str'])) {
            return false;
        }
        $str = $elem['val'][0]['str'];
        $str_len = text::strlen($str);

        if (empty($elem['attrib']['from']) || empty($elem['attrib']['to'])) {
            return '[gradient="' . __('Не указан один из цветов') . '"]' . $this->get_html($elem['val']) . '[/gradient]';
        }
        $from = $elem['attrib']['from'];
        $to = $elem['attrib']['to'];

        if (!preg_match('/^#([0-9a-f]{6}|[0-9a-f]{3})$/ui', $from) || !preg_match('/^#([0-9a-f]{6}|[0-9a-f]{3})$/ui', $to)) {
            return '[gradient="' . __('Один из цветов указан не корректно') . '"]' . $this->get_html($elem['val']) . '[/gradient]';
        }

        $from = str_replace('#', '', $from);
        $to = str_replace('#', '', $to);

        $from_col_len = text::strlen($from) / 3;
        $to_col_len = text::strlen($to) / 3;


        $from2 = array();
        $to2 = array();
        for ($i = 0; $i < 3; $i++) {
            $from2[] = hexdec(str_repeat(text::substr($from, $from_col_len, $from_col_len * $i, ''), 2 / $from_col_len));
            $to2[] = hexdec(str_repeat(text::substr($to, $to_col_len, $to_col_len * $i, ''), 2 / $to_col_len));
        }

        $colors = array();
        for ($i = 0; $i < 3; $i++) {
            $iteration_col = ($to2[$i] - $from2[$i]) / $str_len;
            for ($s = 0; $s < $str_len; $s++) {
                $hex = dechex(round($from2[$i] + $iteration_col * $s));
                //$colors[$s][$i] = $iteration_col;
                $colors[$s][$i] = text::strlen($hex) == 1 ? '0' . $hex : $hex;
            }
        }

        $return = '';
        for ($i = 0; $i < $str_len; $i++) {
            $return .= '<span style="color:#' . implode('', $colors[$i]) . '">' . text::substr($str, 1, $i, '') . '</span>';
        }
        return $return;
    }

}

?>