diff options
author | natec <natec@users.sourceforge.net> | 2001-03-02 05:15:59 +0000 |
---|---|---|
committer | natec <natec@users.sourceforge.net> | 2001-03-02 05:15:59 +0000 |
commit | adb77ffa23bd7ae158b95a363235421255592980 (patch) | |
tree | f6f89954bbe503be3233ccd78655b691ea015be2 /phpBB/functions/bbcode.php | |
parent | 864262c2ab49981990074f2d289c80365196bddf (diff) | |
download | forums-adb77ffa23bd7ae158b95a363235421255592980.tar forums-adb77ffa23bd7ae158b95a363235421255592980.tar.gz forums-adb77ffa23bd7ae158b95a363235421255592980.tar.bz2 forums-adb77ffa23bd7ae158b95a363235421255592980.tar.xz forums-adb77ffa23bd7ae158b95a363235421255592980.zip |
Major bbcode refactoring. Also removed all non-relevant bbcode functions.
git-svn-id: file:///svn/phpbb/trunk@72 89ea8834-ac86-4346-8a33-228a782c2dd0
Diffstat (limited to 'phpBB/functions/bbcode.php')
-rw-r--r-- | phpBB/functions/bbcode.php | 983 |
1 files changed, 203 insertions, 780 deletions
diff --git a/phpBB/functions/bbcode.php b/phpBB/functions/bbcode.php index 112c8763c6..dcc64b9afa 100644 --- a/phpBB/functions/bbcode.php +++ b/phpBB/functions/bbcode.php @@ -21,21 +21,22 @@ * ***************************************************************************/ - //include('phptimer.php'); - define("BBCODE_UID_LEN", 10); +/** + * Does second-pass bbencoding. This should be used before displaying the message in + * a thread. Assumes the message is already first-pass encoded, and has the required + * "[uid:...]" tag as the very first thing in the text. + */ function bbencode_second_pass(&$text) { $uid_tag_length = strpos($text, ']') + 1; $uid = substr($text, 5, BBCODE_UID_LEN); - $max_code_nesting = substr($text, BBCODE_UID_LEN + 6, ($uid_tag_length - BBCODE_UID_LEN - 7)); $text = substr($text, $uid_tag_length); - // pad it with a space so we can distinguish between FALSE and matching the 1st char (index 0). // This is important; bbencode_quote(), bbencode_list(), and bbencode_code() all depend on it. $text = " " . $text; @@ -49,7 +50,7 @@ function bbencode_second_pass(&$text) } // [CODE] and [/CODE] for posting code (HTML, PHP, C etc etc) in your posts. - bbencode_second_pass_code($text, $uid, $max_code_nesting); + $text = bbencode_second_pass_code($text, $uid); // [list] and [list=x] for (un)ordered lists. // unordered lists @@ -108,7 +109,8 @@ function bbencode_second_pass(&$text) $text = substr($text, 1); return TRUE; -} + +} // bbencode_second_pass() @@ -118,20 +120,30 @@ function bbencode_first_pass($text) $uid = md5(uniqid(rand())); $uid = substr($uid, 0, BBCODE_UID_LEN); - echo "UID LENGTH: " . strlen($uid) . "<br>"; + //echo "UID LENGTH: " . strlen($uid) . "<br>"; // pad it with a space so we can distinguish between FALSE and matching the 1st char (index 0). // This is important; bbencode_quote(), bbencode_list(), and bbencode_code() all depend on it. $text = " " . $text; // [CODE] and [/CODE] for posting code (HTML, PHP, C etc etc) in your posts. - $max_code_nesting = bbencode_first_pass_code($text, $uid); + $text = bbencode_first_pass_pda($text, $uid, '[code]', '[/code]', '', true, ''); // [QUOTE] and [/QUOTE] for posting replies with quote, or just for quoting stuff. - bbencode_first_pass_quote($text, $uid); + $text = bbencode_first_pass_pda($text, $uid, '[quote]', '[/quote]', '', false, ''); // [list] and [list=x] for (un)ordered lists. - bbencode_first_pass_list($text, $uid); + $open_tag = array(); + $open_tag[0] = "[list]"; + + // unordered.. + $text = bbencode_first_pass_pda($text, $uid, $open_tag, "[/list]", "[/list:u]", false, 'replace_listitems'); + + $open_tag[0] = "[list=1]"; + $open_tag[1] = "[list=a]"; + + // ordered. + $text = bbencode_first_pass_pda($text, $uid, $open_tag, "[/list]", "[/list:o]", false, 'replace_listitems'); // [b] and [/b] for bolding text. $text = preg_replace("#\[b\](.*?)\[/b\]#si", "[b:$uid]\\1[/b:$uid]", $text); @@ -146,278 +158,80 @@ function bbencode_first_pass($text) $text = substr($text, 1); // Add the uid tag to the start of the string.. - $text = '[uid=' . $uid . ':' . $max_code_nesting . ']' . $text; + $text = '[uid=' . $uid . ']' . $text; return $text; } // bbencode_first_pass() -/** - * Nathan Codding - Feb. 14, 2001. - * Performs [quote][/quote] bbencoding on the given string. - * Any unmatched "[quote]" or "[/quote]" token will just be left alone. - * This works fine with both having more than one quote in a message, and with nested quotes. - * Since that is not a regular language, this is actually a PDA and uses a stack. Great fun. +/** + * $text - The text to operate on. + * $uid - The UID to add to matching tags. + * $open_tag - The opening tag to match. Can be an array of opening tags. + * $close_tag - The closing tag to match. + * $close_tag_new - The closing tag to replace with. + * $mark_lowest_level - boolean - should we specially mark the tags that occur + * at the lowest level of nesting? (useful for [code], because + * we need to match these tags first and transform HTML tags + * in their contents.. + * $func - This variable should contain a string that is the name of a function. + * That function will be called when a match is found, and passed 2 + * parameters: ($text, $uid). The function should return a string. + * This is used when some transformation needs to be applied to the + * text INSIDE a pair of matching tags. If this variable is FALSE or the + * empty string, it will not be executed. + * If open_tag is an array, then the pda will try to match pairs consisting of + * any element of open_tag followed by close_tag. This allows us to match things + * like [list=A]...[/list] and [list=1]...[/list] in one pass of the PDA. * - * Note: This function assumes the first character of $text is a space, which is added by - * bbencode(). + * NOTES: - this function assumes the first character of $text is a space. + * - every opening tag and closing tag must be of the [...] format. */ -function bbencode_first_pass_quote(&$text, $uid) +function bbencode_first_pass_pda($text, $uid, $open_tag, $close_tag, $close_tag_new, $mark_lowest_level, $func) { - // First things first: If there aren't any "[quote]" strings in the message, we don't - // need to process it at all. + $open_tag_count = 0; + $open_tag_length = array(); - if (!strpos(strtolower($text), "[quote]")) + if (!$close_tag_new || ($close_tag_new == '')) { - return TRUE; + $close_tag_new = $close_tag; } - $stack = Array(); - $curr_pos = 1; - while ($curr_pos && ($curr_pos < strlen($text))) - { - $curr_pos = strpos($text, "[", $curr_pos); + $close_tag_length = strlen($close_tag); + $close_tag_new_length = strlen($close_tag_new); + $uid_length = strlen($uid); - // If not found, $curr_pos will be 0, and the loop will end. - if ($curr_pos) - { - // We found a [. It starts at $curr_pos. - // check if it's a starting or ending quote tag. - $possible_start = substr($text, $curr_pos, 7); - $possible_end = substr($text, $curr_pos, 8); - if (strcasecmp("[quote]", $possible_start) == 0) - { - // We have a starting quote tag. - // Push its position on to the stack, and then keep going to the right. - bbcode_array_push($stack, $curr_pos); - ++$curr_pos; - } - else if (strcasecmp("[/quote]", $possible_end) == 0) - { - // We have an ending quote tag. - // Check if we've already found a matching starting tag. - if (sizeof($stack) > 0) - { - // There exists a starting tag. - // We need to do 2 replacements now. - $start_index = bbcode_array_pop($stack); - - // everything before the [quote] tag. - $before_start_tag = substr($text, 0, $start_index); - - // everything after the [quote] tag, but before the [/quote] tag. - $between_tags = substr($text, $start_index + 7, $curr_pos - $start_index - 7); - - // everything after the [/quote] tag. - $after_end_tag = substr($text, $curr_pos + 8); - - $text = $before_start_tag . "[quote:$uid]"; - $text .= $between_tags . "[/quote:$uid]"; - $text .= $after_end_tag; - - // Now.. we've screwed up the indices by changing the length of the string. - // So, if there's anything in the stack, we want to resume searching just after it. - // otherwise, we go back to the start. - if (sizeof($stack) > 0) - { - $curr_pos = array_pop($stack); - bbcode_array_push($stack, $curr_pos); - ++$curr_pos; - } - else - { - $curr_pos = 1; - } - } - else - { - // No matching start tag found. Increment pos, keep going. - ++$curr_pos; - } - } - else - { - // No starting tag or ending tag.. Increment pos, keep looping., - ++$curr_pos; - } - } - } // while + $use_function_pointer = ($func && ($func != '')); - return TRUE; + $stack = array(); -} // bbencode_first_pass_quote() - - -/** - * Nathan Codding - Feb. 14, 2001. - * Performs [code][/code] bbencoding on the given string. - * Any unmatched "[code]" or "[/code]" token will just be left alone. - * This works fine with both having more than one code block in a message, and with nested code blocks. - * Since that is not a regular language, this is actually a PDA and uses a stack. Great fun. - * - * Note: This function assumes the first character of $message is a space, which is added by - * bbencode(). - */ -function bbencode_first_pass_code(&$text, $uid) -{ - // First things first: If there aren't any "[code]" strings in the message, we don't - // need to process it at all. - if (!strpos(strtolower($text), "[code]")) + if (is_array($open_tag)) { - return 0; - } - - // Second things second: we have to watch out for stuff like [1code] or [/code1] in the - // input.. So escape them to [#1code] or [/code#1] for now: - $temp_uid = md5(uniqid(rand())); - - $text = preg_replace("#\[([0-9]+?)code\]#si", "[$temp_uid:\\1code]", $text); - $text = preg_replace("#\[/code([0-9]+?)\]#si", "[/code$temp_uid:\\1]", $text); - - $stack = Array(); - $curr_pos = 1; - $max_nesting_depth = 0; - while ($curr_pos && ($curr_pos < strlen($text))) - { - $curr_pos = strpos($text, "[", $curr_pos); - - // If not found, $curr_pos will be 0, and the loop will end. - if ($curr_pos) + if (0 == count($open_tag)) { - // We found a [. It starts at $curr_pos. - // check if it's a starting or ending code tag. - $possible_start = substr($text, $curr_pos, 6); - $possible_end = substr($text, $curr_pos, 7); - if (strcasecmp("[code]", $possible_start) == 0) - { - // We have a starting code tag. - // Push its position on to the stack, and then keep going to the right. - bbcode_array_push($stack, $curr_pos); - ++$curr_pos; - } - else if (strcasecmp("[/code]", $possible_end) == 0) - { - // We have an ending code tag. - // Check if we've already found a matching starting tag. - if (sizeof($stack) > 0) - { - // There exists a starting tag. - $curr_nesting_depth = sizeof($stack); - $max_nesting_depth = ($curr_nesting_depth > $max_nesting_depth) ? $curr_nesting_depth : $max_nesting_depth; - - // We need to do 2 replacements now. - $start_index = bbcode_array_pop($stack); - - // everything before the [code] tag. - $before_start_tag = substr($text, 0, $start_index); - - // everything after the [code] tag, but before the [/code] tag. - $between_tags = substr($text, $start_index + 6, $curr_pos - $start_index - 6); - - // everything after the [/code] tag. - $after_end_tag = substr($text, $curr_pos + 7); - - $text = $before_start_tag . '[code:' . $curr_nesting_depth . ':' . $uid . ']'; - $text .= $between_tags . '[/code:' . $curr_nesting_depth . ':' . $uid . ']'; - $text .= $after_end_tag; - - // Now.. we've screwed up the indices by changing the length of the string. - // So, if there's anything in the stack, we want to resume searching just after it. - // otherwise, we go back to the start. - if (sizeof($stack) > 0) - { - $curr_pos = bbcode_array_pop($stack); - bbcode_array_push($stack, $curr_pos); - ++$curr_pos; - } - else - { - $curr_pos = 1; - } - } - else - { - // No matching start tag found. Increment pos, keep going. - ++$curr_pos; - } - } - else - { - // No starting tag or ending tag.. Increment pos, keep looping., - ++$curr_pos; - } + // No opening tags to match, so return. + return $text; } - } // while - - // Undo our escaping from "second things second" above.. - $text = preg_replace("#\[$temp_uid:([0-9]+?)code\]#si", "[\\1code]", $text); - $text = preg_replace("#\[/code$temp_uid:([0-9]+?)\]#si", "[/code\\1]", $text); - - return $max_nesting_depth; - -} // bbencode_first_pass_code() - - - -function bbencode_second_pass_code(&$text, $uid, $max_nesting_depth) -{ - for ($i = 1; $i <= $max_nesting_depth; ++$i) - { - $match_count = preg_match_all("#\[code:$i:$uid\](.*?)\[/code:$i:$uid\]#si", $text, $matches); - - for ($j = 0; $j < $match_count; $j++) + + for ($i = 0; $i < count($open_tag); $i++) { - $before_replace = $matches[1][$j]; - $after_replace = $matches[1][$j]; - - if ($i < 2) - { - // don't escape special chars when we're nested, 'cause it was already done - // at the lower level.. - $after_replace = htmlspecialchars($after_replace); - } - - $str_to_match = "[code:$i:$uid]" . $before_replace . "[/code:$i:$uid]"; - - $replacement = '<TABLE BORDER="0" ALIGN="CENTER" WIDTH="85%"><TR><TD><font size="-1">Code:</font><HR></TD></TR><TR><TD><FONT SIZE="-1"><PRE>'; - $replacement .= $after_replace; - $replacement .= '</PRE></FONT></TD></TR><TR><TD><HR></TD></TR></TABLE>'; - - $text = str_replace($str_to_match, $replacement, $text); - + ++$open_tag_count; + $open_tag_length[$i] = strlen($open_tag[$i]); } } - -} // bbencode_second_pass_code() - - - -/** - * Nathan Codding - Jan. 12, 2001. - * Performs [list][/list] and [list=?][/list] bbencoding on the given string, and returns the results. - * Any unmatched "[list]" or "[/list]" token will just be left alone. - * This works fine with both having more than one list in a message, and with nested lists. - * Since that is not a regular language, this is actually a PDA and uses a stack. Great fun. - * - * Note: This function assumes the first character of $message is a space, which is added by - * bbencode(). - */ -function bbencode_first_pass_list(&$text, $uid) -{ - $start_length = Array(); - $start_length['ordered'] = 8; - $start_length['unordered'] = 6; - - // First things first: If there aren't any "[list" strings in the message, we don't - // need to process it at all. - - if (!strpos(strtolower($text), "[list")) + else { - return TRUE; + // only one opening tag. make it into a 1-element array. + $open_tag_temp = $open_tag; + $open_tag = array(); + $open_tag[0] = $open_tag_temp; + $open_tag_length[0] = strlen($open_tag[0]); + $open_tag_count = 1; } - $stack = Array(); + + // Start at the 2nd char of the string, looking for opening tags. $curr_pos = 1; while ($curr_pos && ($curr_pos < strlen($text))) { @@ -427,568 +241,175 @@ function bbencode_first_pass_list(&$text, $uid) if ($curr_pos) { // We found a [. It starts at $curr_pos. - // check if it's a starting or ending list tag. - $possible_ordered_start = substr($text, $curr_pos, $start_length[ordered]); - $possible_unordered_start = substr($text, $curr_pos, $start_length[unordered]); - $possible_end = substr($text, $curr_pos, 7); - if (strcasecmp("[list]", $possible_unordered_start) == 0) + // check if it's a starting or ending tag. + $found_start = false; + $which_start_tag = -1; + for ($i = 0; $i < $open_tag_count; $i++) { - // We have a starting unordered list tag. - // Push its position on to the stack, and then keep going to the right. - bbcode_array_push($stack, array($curr_pos, "")); - ++$curr_pos; + $possible_start = substr($text, $curr_pos, $open_tag_length[$i]); + if (0 == strcasecmp($open_tag[$i], $possible_start)) + { + $found_start = true; + $which_start_tag = $i; + break; + } } - else if (preg_match("/\[list=([a1])\]/si", $possible_ordered_start, $matches)) + + if ($found_start) { - // We have a starting ordered list tag. - // Push its position on to the stack, and the starting char onto the start - // char stack, the keep going to the right. - bbcode_array_push($stack, array($curr_pos, $matches[1])); + // We have an opening tag. + // Push its position and length on to the stack, and then keep going to the right. + $match = array("pos" => $curr_pos, "tag" => $which_start_tag); + bbcode_array_push($stack, $match); ++$curr_pos; } - else if (strcasecmp("[/list]", $possible_end) == 0) + else { - // We have an ending list tag. - // Check if we've already found a matching starting tag. - if (sizeof($stack) > 0) + // check for a closing tag.. + $possible_end = substr($text, $curr_pos, $close_tag_length); + if (0 == strcasecmp($close_tag, $possible_end)) { - // There exists a starting tag. - // We need to do 2 replacements now. - $start = bbcode_array_pop($stack); - $start_index = $start[0]; - $start_char = $start[1]; - $is_ordered = ($start_char != ""); - $start_tag_length = ($is_ordered) ? $start_length[ordered] : $start_length[unordered]; - - // everything before the [list] tag. - $before_start_tag = substr($text, 0, $start_index); - - // everything after the [list] tag, but before the [/list] tag. - $between_tags = substr($text, $start_index + $start_tag_length, $curr_pos - $start_index - $start_tag_length); - // Need to replace [*] with <LI> inside the list. - $between_tags = str_replace('[*]', "[*:$uid]", $between_tags); - - // everything after the [/list] tag. - $after_end_tag = substr($text, $curr_pos + 7); - - if ($is_ordered) - { - $text = $before_start_tag . '[list=' . $start_char . ':' . $uid . ']'; - $text .= $between_tags . '[/list:o:' . $uid . ']'; - } - else - { - $text = $before_start_tag . '[list:' . $uid . ']'; - $text .= $between_tags . '[/list:u:' . $uid . ']'; - } - - $text .= $after_end_tag; - - // Now.. we've screwed up the indices by changing the length of the string. - // So, if there's anything in the stack, we want to resume searching just after it. - // otherwise, we go back to the start. + // We have an ending tag. + // Check if we've already found a matching starting tag. if (sizeof($stack) > 0) { - $a = bbcode_array_pop($stack); - $curr_pos = $a[0]; - bbcode_array_push($stack, $a); - ++$curr_pos; + // There exists a starting tag. + $curr_nesting_depth = sizeof($stack); + // We need to do 2 replacements now. + $match = bbcode_array_pop($stack); + $start_index = $match['pos']; + $which_start_tag = $match['tag']; + $start_length = $open_tag_length[$which_start_tag]; + $start_tag = $open_tag[$which_start_tag]; + + // everything before the opening tag. + $before_start_tag = substr($text, 0, $start_index); + + // everything after the opening tag, but before the closing tag. + $between_tags = substr($text, $start_index + $start_length, $curr_pos - $start_index - $start_length); + + // Run the given function on the text between the tags.. + if ($use_function_pointer) + { + $between_tags = $func($between_tags, $uid); + } + + // everything after the closing tag. + $after_end_tag = substr($text, $curr_pos + $close_tag_length); + + // Mark the lowest nesting level if needed. + if ($mark_lowest_level && ($curr_nesting_depth == 1)) + { + $text = $before_start_tag . substr($start_tag, 0, $start_length - 1) . ":$curr_nesting_depth:$uid]"; + $text .= $between_tags . substr($close_tag_new, 0, $close_tag_new_length - 1) . ":$curr_nesting_depth:$uid]"; + } + else + { + $text = $before_start_tag . substr($start_tag, 0, $start_length - 1) . ":$uid]"; + $text .= $between_tags . substr($close_tag_new, 0, $close_tag_new_length - 1) . ":$uid]"; + } + + $text .= $after_end_tag; + + // Now.. we've screwed up the indices by changing the length of the string. + // So, if there's anything in the stack, we want to resume searching just after it. + // otherwise, we go back to the start. + if (sizeof($stack) > 0) + { + $match = bbcode_array_pop($stack); + $curr_pos = $match['pos']; + bbcode_array_push($stack, $match); + ++$curr_pos; + } + else + { + $curr_pos = 1; + } } else { - $curr_pos = 1; + // No matching start tag found. Increment pos, keep going. + ++$curr_pos; } } else { - // No matching start tag found. Increment pos, keep going. + // No starting tag or ending tag.. Increment pos, keep looping., ++$curr_pos; } } - else - { - // No starting tag or ending tag.. Increment pos, keep looping., - ++$curr_pos; - } } } // while - - return TRUE; - -} // bbencode_first_pass_list() - - - -// END new 2-pass functions. + return $text; + +} // bbencode_first_pass_pda() /** - * bbdecode/bbencode functions: - * Rewritten - Nathan Codding - Aug 24, 2000 - * quote, code, and list rewritten again in Jan. 2001. - * All BBCode tags now implemented. Nesting and multiple occurances should be - * handled fine for all of them. Using str_replace() instead of regexps often - * for efficiency. quote, list, and code are not regular, so they are - * implemented as PDAs - probably not all that efficient, but that's the way it is. - * - * Note: all BBCode tags are case-insensitive. + * Does second-pass bbencoding of the [code] tags. This includes + * running htmlspecialchars() over the text contained between + * any pair of [code] tags that are at the first level of + * nesting. Tags at the first level of nesting are indicated + * by this format: [code:1:$uid] ... [/code:1:$uid] + * Other tags are in this format: [code:$uid] ... [/code:$uid] */ -function bbdecode($message) { - - // Undo [code] - $code_start_html = "<!-- BBCode Start --><TABLE BORDER=0 ALIGN=CENTER WIDTH=85%><TR><TD><font size=-1>Code:</font><HR></TD></TR><TR><TD><FONT SIZE=-1><PRE>"; - $code_end_html = "</PRE></FONT></TD></TR><TR><TD><HR></TD></TR></TABLE><!-- BBCode End -->"; - $message = str_replace($code_start_html, "[code]", $message); - $message = str_replace($code_end_html, "[/code]", $message); +function bbencode_second_pass_code(&$text, $uid) +{ + + $code_start_html = '<TABLE BORDER="0" ALIGN="CENTER" WIDTH="85%"><TR><TD><font size="-1">Code:</font><HR></TD></TR><TR><TD><FONT SIZE="-1"><PRE>'; + $code_end_html = '</PRE></FONT></TD></TR><TR><TD><HR></TD></TR></TABLE>'; + + // First, do all the 1st-level matches. These need an htmlspecialchars() run, + // so they have to be handled differently. + $match_count = preg_match_all("#\[code:1:$uid\](.*?)\[/code:1:$uid\]#si", $text, $matches); - // Undo [quote] - $quote_start_html = "<!-- BBCode Quote Start --><TABLE BORDER=0 ALIGN=CENTER WIDTH=85%><TR><TD><font size=-1>Quote:</font><HR></TD></TR><TR><TD><FONT SIZE=-1><BLOCKQUOTE>"; - $quote_end_html = "</BLOCKQUOTE></FONT></TD></TR><TR><TD><HR></TD></TR></TABLE><!-- BBCode Quote End -->"; - $message = str_replace($quote_start_html, "[quote]", $message); - $message = str_replace($quote_end_html, "[/quote]", $message); - - // Undo [b] and [i] - $message = preg_replace("#<!-- BBCode Start --><B>(.*?)</B><!-- BBCode End -->#s", "[b]\\1[/b]", $message); - $message = preg_replace("#<!-- BBCode Start --><I>(.*?)</I><!-- BBCode End -->#s", "[i]\\1[/i]", $message); - - // Undo [url] (both forms) - $message = preg_replace("#<!-- BBCode Start --><A HREF=\"([a-z]+?://)(.*?)\" TARGET=\"_blank\">(.*?)</A><!-- BBCode End -->#s", "[url=\\1\\2]\\3[/url]", $message); - - // Undo [email] - $message = preg_replace("#<!-- BBCode Start --><A HREF=\"mailto:(.*?)\">(.*?)</A><!-- BBCode End -->#s", "[email]\\1[/email]", $message); - - // Undo [img] - $message = preg_replace("#<!-- BBCode Start --><IMG SRC=\"(.*?)\" BORDER=\"0\"><!-- BBCode End -->#s", "[img]\\1[/img]", $message); + for ($i = 0; $i < $match_count; $i++) + { + $before_replace = $matches[1][$i]; + $after_replace = $matches[1][$i]; - // Undo lists (unordered/ordered) - - // <li> tags: - $message = str_replace("<!-- BBCode --><LI>", "[*]", $message); + $after_replace = htmlspecialchars($after_replace); - // [list] tags: - $message = str_replace("<!-- BBCode ulist Start --><UL>", "[list]", $message); + $str_to_match = "[code:1:$uid]" . $before_replace . "[/code:1:$uid]"; - // [list=x] tags: - $message = preg_replace("#<!-- BBCode olist Start --><OL TYPE=([A1])>#si", "[list=\\1]", $message); + $replacement = $code_start_html; + $replacement .= $after_replace; + $replacement .= $code_end_html; - // [/list] tags: - $message = str_replace("</UL><!-- BBCode ulist End -->", "[/list]", $message); - $message = str_replace("</OL><!-- BBCode olist End -->", "[/list]", $message); - - return($message); -} - -/** - * Nathan Codding - Jan. 12, 2001. - * Performs [quote][/quote] bbencoding on the given string, and returns the results. - * Any unmatched "[quote]" or "[/quote]" token will just be left alone. - * This works fine with both having more than one quote in a message, and with nested quotes. - * Since that is not a regular language, this is actually a PDA and uses a stack. Great fun. - * - * Note: This function assumes the first character of $message is a space, which is added by - * bbencode(). - */ -function bbencode_quote($message) -{ - // First things first: If there aren't any "[quote]" strings in the message, we don't - // need to process it at all. - - if (!strpos(strtolower($message), "[quote]")) - { - return $message; + $text = str_replace($str_to_match, $replacement, $text); } - $stack = Array(); - $curr_pos = 1; - while ($curr_pos && ($curr_pos < strlen($message))) - { - $curr_pos = strpos($message, "[", $curr_pos); - - // If not found, $curr_pos will be 0, and the loop will end. - if ($curr_pos) - { - // We found a [. It starts at $curr_pos. - // check if it's a starting or ending quote tag. - $possible_start = substr($message, $curr_pos, 7); - $possible_end = substr($message, $curr_pos, 8); - if (strcasecmp("[quote]", $possible_start) == 0) - { - // We have a starting quote tag. - // Push its position on to the stack, and then keep going to the right. - bbcode_array_push($stack, $curr_pos); - ++$curr_pos; - } - else if (strcasecmp("[/quote]", $possible_end) == 0) - { - // We have an ending quote tag. - // Check if we've already found a matching starting tag. - if (sizeof($stack) > 0) - { - // There exists a starting tag. - // We need to do 2 replacements now. - $start_index = bbcode_array_pop($stack); - - // everything before the [quote] tag. - $before_start_tag = substr($message, 0, $start_index); - - // everything after the [quote] tag, but before the [/quote] tag. - $between_tags = substr($message, $start_index + 7, $curr_pos - $start_index - 7); + // Now, do all the non-first-level matches. These are simple. + $text = str_replace("[code:$uid]", $code_start_html, $text); + $text = str_replace("[/code:$uid]", $code_end_html, $text); - // everything after the [/quote] tag. - $after_end_tag = substr($message, $curr_pos + 8); - - $message = $before_start_tag . "<!-- BBCode Quote Start --><TABLE BORDER=0 ALIGN=CENTER WIDTH=85%><TR><TD><font size=-1>Quote:</font><HR></TD></TR><TR><TD><FONT SIZE=-1><BLOCKQUOTE>"; - $message .= $between_tags . "</BLOCKQUOTE></FONT></TD></TR><TR><TD><HR></TD></TR></TABLE><!-- BBCode Quote End -->"; - $message .= $after_end_tag; - - // Now.. we've screwed up the indices by changing the length of the string. - // So, if there's anything in the stack, we want to resume searching just after it. - // otherwise, we go back to the start. - if (sizeof($stack) > 0) - { - $curr_pos = array_pop($stack); - bbcode_array_push($stack, $curr_pos); - ++$curr_pos; - } - else - { - $curr_pos = 1; - } - } - else - { - // No matching start tag found. Increment pos, keep going. - ++$curr_pos; - } - } - else - { - // No starting tag or ending tag.. Increment pos, keep looping., - ++$curr_pos; - } - } - } // while - - return $message; + return $text; -} // bbencode_quote() - +} // bbencode_second_pass_code() -/** - * Nathan Codding - Jan. 12, 2001. - * Performs [code][/code] bbencoding on the given string, and returns the results. - * Any unmatched "[code]" or "[/code]" token will just be left alone. - * This works fine with both having more than one code block in a message, and with nested code blocks. - * Since that is not a regular language, this is actually a PDA and uses a stack. Great fun. - * - * Note: This function assumes the first character of $message is a space, which is added by - * bbencode(). - */ -function bbencode_code($message) -{ - // First things first: If there aren't any "[code]" strings in the message, we don't - // need to process it at all. - if (!strpos(strtolower($message), "[code]")) - { - return $message; - } - - // Second things second: we have to watch out for stuff like [1code] or [/code1] in the - // input.. So escape them to [#1code] or [/code#1] for now: - $message = preg_replace("/\[([0-9]+?)code\]/si", "[#\\1code]", $message); - $message = preg_replace("/\[\/code([0-9]+?)\]/si", "[/code#\\1]", $message); - - $stack = Array(); - $curr_pos = 1; - $max_nesting_depth = 0; - while ($curr_pos && ($curr_pos < strlen($message))) - { - $curr_pos = strpos($message, "[", $curr_pos); - - // If not found, $curr_pos will be 0, and the loop will end. - if ($curr_pos) - { - // We found a [. It starts at $curr_pos. - // check if it's a starting or ending code tag. - $possible_start = substr($message, $curr_pos, 6); - $possible_end = substr($message, $curr_pos, 7); - if (strcasecmp("[code]", $possible_start) == 0) - { - // We have a starting code tag. - // Push its position on to the stack, and then keep going to the right. - bbcode_array_push($stack, $curr_pos); - ++$curr_pos; - } - else if (strcasecmp("[/code]", $possible_end) == 0) - { - // We have an ending code tag. - // Check if we've already found a matching starting tag. - if (sizeof($stack) > 0) - { - // There exists a starting tag. - $curr_nesting_depth = sizeof($stack); - $max_nesting_depth = ($curr_nesting_depth > $max_nesting_depth) ? $curr_nesting_depth : $max_nesting_depth; - - // We need to do 2 replacements now. - $start_index = bbcode_array_pop($stack); - // everything before the [code] tag. - $before_start_tag = substr($message, 0, $start_index); - // everything after the [code] tag, but before the [/code] tag. - $between_tags = substr($message, $start_index + 6, $curr_pos - $start_index - 6); - // everything after the [/code] tag. - $after_end_tag = substr($message, $curr_pos + 7); - - $message = $before_start_tag . "[" . $curr_nesting_depth . "code]"; - $message .= $between_tags . "[/code" . $curr_nesting_depth . "]"; - $message .= $after_end_tag; - - // Now.. we've screwed up the indices by changing the length of the string. - // So, if there's anything in the stack, we want to resume searching just after it. - // otherwise, we go back to the start. - if (sizeof($stack) > 0) - { - $curr_pos = bbcode_array_pop($stack); - bbcode_array_push($stack, $curr_pos); - ++$curr_pos; - } - else - { - $curr_pos = 1; - } - } - else - { - // No matching start tag found. Increment pos, keep going. - ++$curr_pos; - } - } - else - { - // No starting tag or ending tag.. Increment pos, keep looping., - ++$curr_pos; - } - } - } // while - - if ($max_nesting_depth > 0) - { - for ($i = 1; $i <= $max_nesting_depth; ++$i) - { - $start_tag = escape_slashes(preg_quote("[" . $i . "code]")); - $end_tag = escape_slashes(preg_quote("[/code" . $i . "]")); - - $match_count = preg_match_all("/$start_tag(.*?)$end_tag/si", $message, $matches); - - for ($j = 0; $j < $match_count; $j++) - { - $before_replace = escape_slashes(preg_quote($matches[1][$j])); - $after_replace = $matches[1][$j]; - - if ($i < 2) - { - // don't escape special chars when we're nested, 'cause it was already done - // at the lower level.. - $after_replace = htmlspecialchars($after_replace); - } - - $str_to_match = $start_tag . $before_replace . $end_tag; - - $message = preg_replace("/$str_to_match/si", "<!-- BBCode Start --><TABLE BORDER=0 ALIGN=CENTER WIDTH=85%><TR><TD><font size=-1>Code:</font><HR></TD></TR><TR><TD><FONT SIZE=-1><PRE>$after_replace</PRE></FONT></TD></TR><TR><TD><HR></TD></TR></TABLE><!-- BBCode End -->", $message); - } - } - } - - // Undo our escaping from "second things second" above.. - $message = preg_replace("/\[#([0-9]+?)code\]/si", "[\\1code]", $message); - $message = preg_replace("/\[\/code#([0-9]+?)\]/si", "[/code\\1]", $message); - - return $message; - -} // bbencode_code() /** - * Nathan Codding - Jan. 12, 2001. - * Performs [list][/list] and [list=?][/list] bbencoding on the given string, and returns the results. - * Any unmatched "[list]" or "[/list]" token will just be left alone. - * This works fine with both having more than one list in a message, and with nested lists. - * Since that is not a regular language, this is actually a PDA and uses a stack. Great fun. - * - * Note: This function assumes the first character of $message is a space, which is added by - * bbencode(). + * This is used to change a [*] tag into a [*:$uid] tag as part + * of the first-pass bbencoding of [list] tags. It fits the + * standard required in order to be passed as a variable + * function into bbencode_first_pass_pda(). */ -function bbencode_list($message) -{ - $start_length = Array(); - $start_length[ordered] = 8; - $start_length[unordered] = 6; - - // First things first: If there aren't any "[list" strings in the message, we don't - // need to process it at all. - - if (!strpos(strtolower($message), "[list")) - { - return $message; - } - - $stack = Array(); - $curr_pos = 1; - while ($curr_pos && ($curr_pos < strlen($message))) - { - $curr_pos = strpos($message, "[", $curr_pos); - - // If not found, $curr_pos will be 0, and the loop will end. - if ($curr_pos) - { - // We found a [. It starts at $curr_pos. - // check if it's a starting or ending list tag. - $possible_ordered_start = substr($message, $curr_pos, $start_length[ordered]); - $possible_unordered_start = substr($message, $curr_pos, $start_length[unordered]); - $possible_end = substr($message, $curr_pos, 7); - if (strcasecmp("[list]", $possible_unordered_start) == 0) - { - // We have a starting unordered list tag. - // Push its position on to the stack, and then keep going to the right. - bbcode_array_push($stack, array($curr_pos, "")); - ++$curr_pos; - } - else if (preg_match("/\[list=([a1])\]/si", $possible_ordered_start, $matches)) - { - // We have a starting ordered list tag. - // Push its position on to the stack, and the starting char onto the start - // char stack, the keep going to the right. - bbcode_array_push($stack, array($curr_pos, $matches[1])); - ++$curr_pos; - } - else if (strcasecmp("[/list]", $possible_end) == 0) - { - // We have an ending list tag. - // Check if we've already found a matching starting tag. - if (sizeof($stack) > 0) - { - // There exists a starting tag. - // We need to do 2 replacements now. - $start = bbcode_array_pop($stack); - $start_index = $start[0]; - $start_char = $start[1]; - $is_ordered = ($start_char != ""); - $start_tag_length = ($is_ordered) ? $start_length[ordered] : $start_length[unordered]; - - // everything before the [list] tag. - $before_start_tag = substr($message, 0, $start_index); - - // everything after the [list] tag, but before the [/list] tag. - $between_tags = substr($message, $start_index + $start_tag_length, $curr_pos - $start_index - $start_tag_length); - // Need to replace [*] with <LI> inside the list. - $between_tags = str_replace("[*]", "<!-- BBCode --><LI>", $between_tags); - - // everything after the [/list] tag. - $after_end_tag = substr($message, $curr_pos + 7); - - if ($is_ordered) - { - $message = $before_start_tag . "<!-- BBCode olist Start --><OL TYPE=" . $start_char . ">"; - $message .= $between_tags . "</OL><!-- BBCode olist End -->"; - } - else - { - $message = $before_start_tag . "<!-- BBCode ulist Start --><UL>"; - $message .= $between_tags . "</UL><!-- BBCode ulist End -->"; - } - - $message .= $after_end_tag; - - // Now.. we've screwed up the indices by changing the length of the string. - // So, if there's anything in the stack, we want to resume searching just after it. - // otherwise, we go back to the start. - if (sizeof($stack) > 0) - { - $a = bbcode_array_pop($stack); - $curr_pos = $a[0]; - bbcode_array_push($stack, $a); - ++$curr_pos; - } - else - { - $curr_pos = 1; - } - } - else - { - // No matching start tag found. Increment pos, keep going. - ++$curr_pos; - } - } - else - { - // No starting tag or ending tag.. Increment pos, keep looping., - ++$curr_pos; - } - } - } // while - - return $message; - -} // bbencode_list() - - -function bbencode($message) { - - // pad it with a space so we can distinguish between FALSE and matching the 1st char (index 0). - // This is important; bbencode_quote(), bbencode_list(), and bbencode_code() all depend on it. - $message = " " . $message; - - // First: If there isn't a "[" and a "]" in the message, don't bother. - if (! (strpos($message, "[") && strpos($message, "]")) ) - { - // Remove padding, return. - $message = substr($message, 1); - return $message; - } - - // [CODE] and [/CODE] for posting code (HTML, PHP, C etc etc) in your posts. - $message = bbencode_code($message); - - // [QUOTE] and [/QUOTE] for posting replies with quote, or just for quoting stuff. - $message = bbencode_quote($message); - - // [list] and [list=x] for (un)ordered lists. - $message = bbencode_list($message); - - // [b] and [/b] for bolding text. - $message = preg_replace("/\[b\](.*?)\[\/b\]/si", "<!-- BBCode Start --><B>\\1</B><!-- BBCode End -->", $message); - - // [i] and [/i] for italicizing text. - $message = preg_replace("/\[i\](.*?)\[\/i\]/si", "<!-- BBCode Start --><I>\\1</I><!-- BBCode End -->", $message); - - // [url]www.phpbb.com[/url] code.. - $message = preg_replace("/\[url\](http:\/\/)?(.*?)\[\/url\]/si", "<!-- BBCode Start --><A HREF=\"http://\\2\" TARGET=\"_blank\">\\2</A><!-- BBCode End -->", $message); - - // [url=xxxx://www.phpbb.com]phpBB[/url] code.. - $message = preg_replace("#\[url=([a-z]+?://)?(.*?)\](.*?)\[/url\]#si", "<!-- BBCode Start --><A HREF=\"\\1\\2\" TARGET=\"_blank\">\\3</A><!-- BBCode End -->", $message); - - // [email]user@domain.tld[/email] code.. - $message = preg_replace("/\[email\](.*?)\[\/email\]/si", "<!-- BBCode Start --><A HREF=\"mailto:\\1\">\\1</A><!-- BBCode End -->", $message); - - // [img]image_url_here[/img] code.. - $message = preg_replace("/\[img\](.*?)\[\/img\]/si", "<!-- BBCode Start --><IMG SRC=\"\\1\" BORDER=\"0\"><!-- BBCode End -->", $message); - - // Remove our padding from the string.. - $message = substr($message, 1); - return $message; +function replace_listitems($text, $uid) +{ + $text = str_replace("[*]", "[*:$uid]", $text); -} // bbencode() + return $text; +} /** - * Nathan Codding - Oct. 30, 2000 - * * Escapes the "/" character with "\/". This is useful when you need * to stick a runtime string into a PREG regexp that is being delimited * with slashes. @@ -1000,11 +421,9 @@ function escape_slashes($input) } - /** - * James Atkinson - Feb 5, 2001 * This function does exactly what the PHP4 function array_push() does - * however, to keep phpBB compatable with PHP 3 we had to come up with out own + * however, to keep phpBB compatable with PHP 3 we had to come up with our own * method of doing it. */ function bbcode_array_push(&$stack, $value) { @@ -1013,26 +432,30 @@ function bbcode_array_push(&$stack, $value) { } /** - * James Atkinson - Feb 5, 2001 * This function does exactly what the PHP4 function array_pop() does - * however, to keep phpBB compatable with PHP 3 we had to come up with out own + * however, to keep phpBB compatable with PHP 3 we had to come up with our own * method of doing it. */ function bbcode_array_pop(&$stack) { $arrSize = count($stack); $x = 1; - while(list($key, $val) = each($stack)) { - if($x < count($stack)) { - $tmpArr[] = $val; + while(list($key, $val) = each($stack)) + { + if($x < count($stack)) + { + $tmpArr[] = $val; } - else { - $return_val = $val; + else + { + $return_val = $val; } $x++; } $stack = $tmpArr; + return($return_val); } + ?> |