Вход Регистрация
Файл: sys/plugins/classes/bbcode.class.php
Строк: 877
<?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',
                
'when',
                
'user',
                
'no',
                
'smile',
                
'color',
                
'red',
                
'green',
                
'blue',
                
'yellow',
                
'gradient',
                
'font',
                
'mark'
            
)
        ),
        
'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',
                
'when',
                
'user',
                
'smile',
                
'color',
                
'red',
                
'green',
                
'blue',
                
'yellow',
                
'gradient',
                
'font',
                
'mark'
            
)
        ),
        
'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',
                
'when',
                
'user',
                
'no',
                
'smile',
                
'color',
                
'red',
                
'green',
                
'blue',
                
'yellow',
                
'gradient',
                
'font',
                
'mark'
            
)
        ),
        
'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',
                
'when',
                
'user',
                
'no',
                
'smile',
                
'color',
                
'red',
                
'green',
                
'blue',
                
'yellow',
                
'gradient',
                
'font',
                
'mark'
            
)
        ),
        
'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',
                
'when',
                
'user',
                
'no',
                
'smile',
                
'color',
                
'red',
                
'green',
                
'blue',
                
'yellow',
                
'gradient',
                
'font'
            
)
        ),
        
'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',
                
'when',
                
'user',
                
'no',
                
'smile',
                
'font',
                
'left',
                
'center',
                
'right'
            
)
        ),
        
'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',
                
'when',
                
'user',
                
'no',
                
'smile',
                
'color',
                
'red',
                
'green',
                
'blue',
                
'yellow',
                
'gradient',
                
'font',
                
'mark'
            
)
        ),
        
'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' => 1,
            
'rbr' => 1,
            
'ends' => array(),
            
'permission_top_level' => true,
            
'children' => array(
                
'big',
                
'u',
                
'i',
                
'b',
                
'quote',
                
'small',
                
'user',
                
'smile',
                
'url',
                
'when',
                
'no',
                
'php',
                
'hide',
                
'spoiler',
                
'color',
                
'red',
                
'green',
                
'blue',
                
'yellow',
                
'localimg',
                
'gradient',
                
'youtube',
                
'font',
                
'mark',
                
'left',
                
'center',
                
'right'
            
)
        ),
        
'youtube' => array(
            
'handler' => 'youtube_2html',
            
'is_close' => false,
            
'lbr' => 1,
            
'rbr' => 1,
            
'ends' => array(),
            
'permission_top_level' => true,
            
'children' => array()
        ),
        
'vk_video' => array(
            
'handler' => 'vk_video_2html',
            
'is_close' => true,
            
'lbr' => 1,
            
'rbr' => 1,
            
'ends' => array(),
            
'permission_top_level' => true,
            
'children' => array()
        ),
        
'localimg' => array(
            
'handler' => 'img_2html',
            
'is_close' => false,
            
'lbr' => 1,
            
'rbr' => 1,
            
'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',
                
'localimg',
                
'font',
                
'mark'
            
)
        ),
        
'php' => array(
            
'handler' => 'php_2html',
            
'is_close' => false,
            
'lbr' => 1,
            
'rbr' => 1,
            
'ends' => array(),
            
'permission_top_level' => true,
            
'children' => array()
        ),
        
'quote' => array(
            
'handler' => 'quote_2html',
            
'is_close' => false,
            
'lbr' => 1,
            
'rbr' => 1,
            
'ends' => array(),
            
'permission_top_level' => true,
            
'children' => array(
                
'big',
                
'u',
                
'i',
                
'b',
                
'small',
                
'user',
                
'smile',
                
'url',
                
'quote',
                
'when',
                
'no',
                
'php',
                
'gradient',
                
'localimg',
                
'spoiler',
                
'color',
                
'font',
                
'mark',
                
'nobb',
                
'red',
                
'green',
                
'blue',
                
'yellow',
                
'left',
                
'center',
                
'right'
            
)
        ),
        
'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''when''no''url''font')
        ),
        
'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''when''no''font''mark')
        ),
        
'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''when''no''url''font')
        ),
        
'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''when''no''url''font')
        ),
        
'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''when''no''url''font')
        ),
        
'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''when''no''url''font')
        ),
        
'when' => array(
            
'handler' => 'vremja_2html',
            
'is_close' => false,
            
'lbr' => 0,
            
'rbr' => 0,
            
'ends' => array(),
            
'permission_top_level' => true,
            
'children' => array()
        ),
        
'font' => array(
            
'handler' => 'font_2html',
            
'is_close' => false,
            
'lbr' => 0,
            
'rbr' => 0,
            
'ends' => array(),
            
'permission_top_level' => true,
            
'children' => array()
        ),
        
'hr' => array(
            
'handler' => 'hr_2html',
            
'is_close' => true,
            
'lbr' => 0,
            
'rbr' => 0,
            
'ends' => array(),
            
'permission_top_level' => true,
            
'children' => array()
        ),
        
'left' => array(
            
'handler' => 'left_2html',
            
'is_close' => false,
            
'lbr' => 1,
            
'rbr' => 1,
            
'ends' => array(),
            
'permission_top_level' => true,
            
'children' => array(
                
'big',
                
'u',
                
'i',
                
'b',
                
'small',
                
'user',
                
'smile',
                
'url',
                
'quote',
                
'when',
                
'no',
                
'php',
                
'gradient',
                
'localimg',
                
'spoiler',
                
'color',
                
'font',
                
'mark'
            
)
        ),
        
'center' => array(
            
'handler' => 'center_2html',
            
'is_close' => false,
            
'lbr' => 1,
            
'rbr' => 1,
            
'ends' => array(),
            
'permission_top_level' => true,
            
'children' => array(
                
'big',
                
'u',
                
'i',
                
'b',
                
'small',
                
'user',
                
'smile',
                
'url',
                
'quote',
                
'when',
                
'no',
                
'php',
                
'gradient',
                
'localimg',
                
'spoiler',
                
'color',
                
'font',
                
'mark'
            
)
        ),
        
'right' => array(
            
'handler' => 'right_2html',
            
'is_close' => false,
            
'lbr' => 1,
            
'rbr' => 1,
            
'ends' => array(),
            
'permission_top_level' => true,
            
'children' => array(
                
'big',
                
'u',
                
'i',
                
'b',
                
'small',
                
'user',
                
'smile',
                
'url',
                
'quote',
                
'when',
                
'no',
                
'php',
                
'gradient',
                
'localimg',
                
'spoiler',
                
'color',
                
'font',
                
'mark'
            
)
        ),
        
'indent' => array(
            
'handler' => 'indent_2html',
            
'is_close' => false,
            
'lbr' => 1,
            
'rbr' => 1,
            
'ends' => array(),
            
'permission_top_level' => true,
            
'children' => array(
                
'big',
                
'u',
                
'i',
                
'b',
                
'small',
                
'user',
                
'smile',
                
'url',
                
'quote',
                
'when',
                
'no',
                
'php',
                
'gradient',
                
'localimg',
                
'spoiler',
                
'color',
                
'font',
                
'mark'
            
)
        ),
    );
    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 
"":
                    
$type_of_char 6;
                    break;
                case 
"x0B":
                    
$type_of_char 6;
                    break;
                default:
                    
$type_of_char 7;
            }
            if (
== $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 (
== $previous_type) {
                        
$tokens[$token_key][1] .= $code{$i};
                    } else {
                        
$tokens[++$token_key] = array(6$code{$i});
                    }
                    break;
                case 
7:
                    if (
== $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 |
            
=> array(100000000)
        ,
            
=> array(233334335)
        ,
            
=> array(233334335)
        ,
            
=> array(100000000)
        ,
            
=> array(263333337)
        ,
            
=> array(2633891033)
        ,
            
=> array(100000000)
        ,
            
=> array(263333333)
        ,
            
=> array(131311121313141313)
        ,
            
=> array(263333333)
        ,
            
10 => array(26338931515)
        ,
            
11 => array(161617161616161616)
        ,
            
12 => array(181818171818181818)
        ,
            
13 => array(19619191919171919)
        ,
            
14 => array(231112131331313)
        ,
            
15 => array(2633891033)
        ,
            
16 => array(161617161616161616)
        ,
            
17 => array(263339201515)
        ,
            
18 => array(181818171818181818)
        ,
            
19 => array(19619191919201919)
        ,
            
20 => array(26333931515)
        );
        
// Получаем массив лексем:
        
$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 (-
$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 (-
$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 (-
$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 (-
$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 = (-$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_tagstrue) 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_tagstrue) 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 = (-$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 = (-$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_tagstrue) 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_tagstrue) 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 = (-$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 = (-$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 = (-$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 = (-$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($textENT_QUOTES'UTF-8'false);
        
$text nl2br($text);
        
//$text = str_replace('  ', '  ', $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'], 06)) {
                        
$elem['str'] = substr_replace($elem['str'], ''06);
                    }
                }
                
$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'', -66);
                    }
                }
                
$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 
youtube_2html($elem)
    {
        return 
'<div class="DCMS_youtube"><iframe src="//www.youtube.com/embed/' text::toValue($elem['val'][0]['str']) . '" frameborder="0" allowfullscreen></iframe></div>';
    }

    function 
vk_video_2html($elem)
    {
        
$allow_params = array('oid''id''hash''hd');
        
$params = array();
        foreach (
$elem['attrib'] AS $key => $value) {
            if (!
in_array($key$allow_params)) {
                continue;
            }
            
$params[] = $key '=' $value;
        }
        return 
'<div class="DCMS_vk_video"><iframe src="//vk.com/video_ext.php?' text::toValue(join('&',
                
$params)) . '"  frameborder="0"></iframe></div>';
    }

    function 
font_2html($elem)
    {

        if (!empty(
$elem['attrib']['name'])) {
            
$name $elem['attrib']['name'];
        } elseif (!empty(
$elem['attrib']['font'])) {
            
$name $elem['attrib']['font'];
        } else {
            
$name __('Arial');
        }
        return 
'<span style="font-family:' text::toValue($name) . ';">' $this->get_html($elem['val']) . '</span>';
    }

    function 
left_2html($elem)
    {
        return 
'<div align="left">' $this->get_html($elem['val']) . '</div>';
    }

    function 
hr_2html($elem)
    {
        return 
'<div class="desc"></div>';
    }

    function 
center_2html($elem)
    {
        return 
'<div align="center">' $this->get_html($elem['val']) . '</div>';
    }

    function 
right_2html($elem)
    {
        return 
'<div align="right">' $this->get_html($elem['val']) . '</div>';
    }

    function 
indent_2html($elem)
    {
        return 
'<div style="margin:1em;">' $this->get_html($elem['val']) . '</div>';
    }

    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="DCMS_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 
misc::when($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">' text::toValue($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($text40);


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

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

        if (
preg_match('#://#'$aturl)) {
            
// внешняя ссылка
            
$url '//' $_SERVER ['HTTP_HOST'] . '/link.ext.php?url=' urlencode($aturl);
            
$new_window = @$dcms->browser_type == 'full' ' 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 'https?://' $_SERVER ['HTTP_HOST'] . '/link.ext.php?url=' urlencode($aturl);
                    }

                    
$new_window '';
                }
            }

            return 
'<a' $new_window ' href="' $url '">' text::toValue($text) . '</a>';
        } else {
            
// внутренняя 
            
$url preg_replace('#^https?://' preg_quote($_SERVER ['HTTP_HOST']) . '(/|$)#ui''/'$aturl);
            return 
'<a href="' text::toValue($url) . '">' text::toValue($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> (" misc::when($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('/sys/files/.bbcode/' $file '.jpg')) {
            return 
'<div class="error" style="color:gray">[' __('тут была картинка') . ']</div>';
        }

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

        if (
$screen $file->getScreen($design->img_max_width())) {
            return 
'<img class="DCMS_bb_image" src="' $screen '" alt="' text::toValue($elem['val'][0]['str']) . '" data-origin="' text::toValue(@$elem['attrib']['origin']) . '" />';
        }
    }

    function 
php_2html($elem)
    {
        
$code "<?phpn" trim(preg_replace('#^<?(php)?|?>$#i''', @$elem['val'][0]['str'])) . "n?>";
        
$code highlight_string($codetrue);
        
$code preg_replace('#<code>(.*?)</code>#si''<div class="DCMS_phpcode">\1</div>'$code);
        
$code preg_replace("#[nrt]+#"''$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="' text::toValue($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''),
                
$from_col_len));
            
$to2[] = hexdec(str_repeat(text::substr($to$to_col_len$to_col_len $i''), $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) == '0' $hex $hex;
            }
        }

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

}
Онлайн: 3
Реклама