View Issue Details
ID | Project | Category | View Status | Date Submitted | Last Update |
---|---|---|---|---|---|
923 | Composr | wiki | public | 2012-12-02 02:50 | 2012-12-02 13:39 |
Reporter | Guest | Assigned To | Chris Graham | ||
Priority | normal | Severity | trivial | ||
Status | resolved | Resolution | fixed | ||
Summary | 923: Incorrect string formatting after deleting wiki+ menu from sidebar | ||||
Description | After creating a new site, I decided to delete the wiki+ menu from the sidebar. Once I deleted the menu, the other menus in the sidebar were incorrectly formatted. It would have {$*} instead of the string being formatted into a correct title. The HTML of the sidebar was messy and unreadable so I couldn't find the source of the problem. | ||||
Steps To Reproduce | 1.Set up a new site. 2.Make sure wiki+ is installed 3.Edit left sidebar 4.Click on "Edit" 5.Click on the block that is the Wiki+ menu 6.Check Delete 7.Save 8.Save the sidebar 9.Examine the incorrectly formatted titles of the remaining menus. | ||||
Additional Information | Browser:Firefox 16 | ||||
Tags | No tags attached. | ||||
Attach Tags | |||||
Attached Files | |||||
Time estimation (hours) | |||||
Sponsorship open | |||||
|
Fixed sources/comcode_renderer.php attached. comcode_renderer.php (96,933 bytes)
<?php /* ocPortal Copyright (c) ocProducts, 2004-2012 See text/EN/licence.txt for full licencing information. NOTE TO PROGRAMMERS: Do not edit this file. If you need to make changes, save your changed file to the appropriate *_custom folder **** If you ignore this advice, then your website upgrades (e.g. for bug fixes) will likely kill your changes **** */ /** * @license http://opensource.org/licenses/cpal_1.0 Common Public Attribution License * @copyright ocProducts Ltd * @package core_rich_media */ /** * Standard code module initialisation function. */ function init__comcode_renderer() { require_code('comcode_text'); global $STRUCTURE_LIST; $STRUCTURE_LIST=array(); global $DONT_CARE_MISSING_PAGES; $DONT_CARE_MISSING_PAGES=array('topics','chat','forumview','topicview','search'); } /** * Check the specified URL for potentially malicious JavaScript/etc. If any is found, the hack attack is logged if in an active post request by the submitting member otherwise filtered out. * * @param MEMBER The member who submitted the URL * @param URLPATH The URL to check * @param boolean Whether to check as arbitrary admin * @return URLPATH Filtered input URL. */ function check_naughty_javascript_url($source_member,$url,$as_admin) { global $POTENTIAL_JS_NAUGHTY_ARRAY; if ((!$as_admin) && (!has_specific_permission($source_member,'use_very_dangerous_comcode'))) { $url2=strtolower($url); $matches=array(); $bad=preg_match_all('#&\#(\d+)#',preg_replace('#\s#','',$url),$matches)!=0; if ($bad) { for ($i=0;$i<count($matches[0]);$i++) { $matched_entity=intval($matches[1][$i]); if (($matched_entity<127) && (array_key_exists(chr($matched_entity),$POTENTIAL_JS_NAUGHTY_ARRAY))) { if ((count($_POST)!=0) && (get_member()==$source_member)) log_hack_attack_and_exit('ASCII_ENTITY_URL_HACK',$url); return ''; } } } $bad=preg_match_all('#&\#x([\dA-Za-z][\dA-Za-z]+)#',preg_replace('#\s#','',$url),$matches)!=0; if ($bad) { for ($i=0;$i<count($matches[0]);$i++) { $matched_entity=intval(base_convert($matches[1][$i],16,10)); if (($matched_entity<127) && (array_key_exists(chr($matched_entity),$POTENTIAL_JS_NAUGHTY_ARRAY))) { if ((count($_POST)!=0) && (get_member()==$source_member)) log_hack_attack_and_exit('ASCII_ENTITY_URL_HACK',$url); return ''; } } } $bad=(strpos($url2,'script:')!==false) || (strpos($url2,'data:')!==false); if ($bad) { if ((count($_POST)!=0) && (get_member()==$source_member)) log_hack_attack_and_exit('SCRIPT_URL_HACK',$url2); return ''; } } return $url; } /** * Load up custom comcode tags so that we may parse them. * * @param object The database connection to use */ function _custom_comcode_import($connection) { global $IN_MINIKERNEL_VERSION; global $DANGEROUS_TAGS,$VALID_COMCODE_TAGS,$BLOCK_TAGS,$TEXTUAL_TAGS,$IMPORTED_CUSTOM_COMCODE,$REPLACE_TARGETS; if ($IN_MINIKERNEL_VERSION==0) { // From forum driver if (method_exists($GLOBALS['FORUM_DRIVER'],'get_custom_bbcode')) { $custom_bbcode=$GLOBALS['FORUM_DRIVER']->get_custom_bbcode(); foreach ($custom_bbcode as $code) { $VALID_COMCODE_TAGS[$code['tag']]=1; if ($code['block_tag']==1) $BLOCK_TAGS[$code['tag']]=1; if ($code['textual_tag']==1) $TEXTUAL_TAGS[$code['tag']]=1; if ($code['dangerous_tag']==1) $DANGEROUS_TAGS[$code['tag']]=1; //if (is_object($code['replace'])) $code['replace']=$code['replace']->evaluate(); Never tempcode $REPLACE_TARGETS[$code['tag']]=array('replace'=>$code['replace'],'parameters'=>$code['parameters']); } } // From Custom Comcode $tags=array(); if (addon_installed('custom_comcode')) { if (($GLOBALS['FORUM_DB']!=$connection) && (!is_null($GLOBALS['FORUM_DB'])) && (get_forum_type()=='ocf')) $tags=array_merge($tags,$GLOBALS['FORUM_DB']->query_select('custom_comcode',array('tag_parameters','tag_replace','tag_tag','tag_dangerous_tag','tag_block_tag','tag_textual_tag'),array('tag_enabled'=>1))); if ($connection->connection_write!=$GLOBALS['SITE_DB']->connection_write) $tags=array_merge($tags,$GLOBALS['SITE_DB']->query_select('custom_comcode',array('tag_parameters','tag_replace','tag_tag','tag_dangerous_tag','tag_block_tag','tag_textual_tag'),array('tag_enabled'=>1))); $tags=array_merge($tags,$connection->query_select('custom_comcode',array('tag_parameters','tag_replace','tag_tag','tag_dangerous_tag','tag_block_tag','tag_textual_tag'),array('tag_enabled'=>1))); } foreach ($tags as $tag) { $VALID_COMCODE_TAGS[$tag['tag_tag']]=1; if ($tag['tag_block_tag']==1) $BLOCK_TAGS[$tag['tag_tag']]=1; if ($tag['tag_textual_tag']==1) $TEXTUAL_TAGS[$tag['tag_tag']]=1; if ($tag['tag_dangerous_tag']==1) $DANGEROUS_TAGS[$tag['tag_tag']]=1; //if (is_object($tag['tag_replace'])) $tag['tag_replace']=$tag['tag_replace']->evaluate(); Never tempcode $REPLACE_TARGETS[$tag['tag_tag']]=array('replace'=>$tag['tag_replace'],'parameters'=>$tag['tag_parameters']); } // From Comcode hooks $hooks=find_all_hooks('systems','comcode'); foreach (array_keys($hooks) as $hook) { require_code('hooks/systems/comcode/'.filter_naughty_harsh($hook)); $object=object_factory('Hook_comcode_'.filter_naughty_harsh($hook),true); $tag=$object->get_tag(); $VALID_COMCODE_TAGS[$tag['tag_tag']]=1; if ($tag['tag_block_tag']==1) $BLOCK_TAGS[$tag['tag_tag']]=1; if ($tag['tag_textual_tag']==1) $TEXTUAL_TAGS[$tag['tag_tag']]=1; if ($tag['tag_dangerous_tag']==1) $DANGEROUS_TAGS[$tag['tag_tag']]=1; $REPLACE_TARGETS[$tag['tag_tag']]=array('replace'=>$tag['tag_replace'],'parameters'=>$tag['tag_parameters']); } } $IMPORTED_CUSTOM_COMCODE=true; } /** * Convert the specified comcode (unknown format) into a tempcode tree. You shouldn't output the tempcode tree to the browser, as it looks really horrible. If you are in a rare case where you need to output directly (not through templates), you should call the evaluate method on the tempcode object, to convert it into a string. * * @param LONG_TEXT The comcode to convert * @param ?MEMBER The member the evaluation is running as. This is a security issue, and you should only run as an administrator if you have considered where the comcode came from carefully (NULL: current member) * @param boolean Whether to explicitly execute this with admin rights. There are a few rare situations where this should be done, for data you know didn't come from a member, but is being evaluated by one. * @param ?integer The position to conduct wordwrapping at (NULL: do not conduct word-wrapping) * @param ?string A special identifier that can identify this resource in a sea of our resources of this class; usually this can be ignored, but may be used to provide a binding between Javascript in evaluated comcode, and the surrounding environment (NULL: no explicit binding) * @param ?object The database connection to use (NULL: standard site connection) * @param boolean Whether to parse so as to create something that would fit inside a semihtml tag. It means we generate HTML, with Comcode written into it where the tag could never be reverse-converted (e.g. a block). * @param boolean Whether this is being pre-parsed, to pick up errors before row insertion. * @param boolean Whether to treat this whole thing as being wrapped in semihtml, but apply normal security otherwise. * @param boolean Whether we are only doing this parse to find the title structure * @param boolean Whether to only check the Comcode. It's best to use the check_comcode function which will in turn use this parameter. * @param ?array A list of words to highlight (NULL: none) * @param ?MEMBER The member we are running on behalf of, with respect to how attachments are handled; we may use this members attachments that are already within this post, and our new attachments will be handed to this member (NULL: member evaluating) * @return tempcode The tempcode generated */ function _comcode_to_tempcode($comcode,$source_member=NULL,$as_admin=false,$wrap_pos=60,$pass_id=NULL,$connection=NULL,$semiparse_mode=false,$preparse_mode=false,$is_all_semihtml=false,$structure_sweep=false,$check_only=false,$highlight_bits=NULL,$on_behalf_of_member=NULL) { /*if (get_option('anti_leech',true)==='1') No, better to see it, even if custom settings are lost { unset($REVERSABLE_TAGS['attachment_safe']); }*/ if (is_null($connection)) $connection=$GLOBALS['SITE_DB']; if (is_null($source_member)) { $source_member=(function_exists('get_member'))?get_member():0; } if (!$structure_sweep) $comcode=unixify_line_format($comcode); // Done already if this is a structure sweep // Ensures the 'title' tags are incremental with their anchors global $STRUCTURE_LIST; $old_structure_list=$STRUCTURE_LIST; $STRUCTURE_LIST=array(); if (substr($comcode,0,8)=='<comcode') { require_code('comcode_xml'); if (!$as_admin) check_specific_permission('comcode_dangerous',NULL,$source_member); $parser=new comcode_xml_to_tempcode($comcode,$source_member,$wrap_pos,$pass_id,$connection,$semiparse_mode,$preparse_mode,$is_all_semihtml,$structure_sweep,$check_only,$on_behalf_of_member); $ret=$parser->tempcode; } else { require_code('comcode_text'); $ret=comcode_text_to_tempcode($comcode,$source_member,$as_admin,$wrap_pos,$pass_id,$connection,$semiparse_mode,$preparse_mode,$is_all_semihtml,$structure_sweep,$check_only,$highlight_bits,$on_behalf_of_member); } $STRUCTURE_LIST=$old_structure_list; // Restore, so that Comcode pages being loaded up in search results don't get skewed TOC's return $ret; } /** * Show a comcode parser error. * * @param boolean Whether this is being pre-parsed, to pick up errors before row insertion. * @param array Error message details to pass to do_lang, or if the first in the list is NULL, use directly * @param integer The position during parsing that the error occurred at * @param LONG_TEXT The comcode the parser error occurred in * @param boolean Whether to only check the Comcode. * @return tempcode An error message to put in the output stream (shown in certain situations, where in other situations we bomb out). */ function comcode_parse_error($preparse_mode,$_message,$pos,$comcode,$check_only=false) { //echo $comcode; require_lang('comcode'); if (is_null($_message[0])) { $message=$_message[1]; } else { if (strpos($_message[0],':')===false) $_message[0]='comcode:'.$_message[0]; $message=call_user_func_array('do_lang_tempcode',array_map('escape_html',$_message)); } $posted=false; foreach ($_POST+$_GET as $name=>$val) { if (is_array($val)) continue; if ((post_param($name,'')==$comcode) || (substr($name,-7)=='_parsed')) $posted=true; } if (!$check_only) { if (((get_page_name()=='admin_import') || (count($_POST)==0) || /*(strpos($comcode,']new_')!==false) || */(!$posted)) && (!$preparse_mode)) { $line=substr_count(substr($comcode,0,$pos),chr(10))+1; $out=do_template('COMCODE_CRITICAL_PARSE_ERROR',array('LINE'=>integer_format($line),'MESSAGE'=>$message,'SOURCE'=>$comcode)); // Won't parse, but we can't help it, so we will skip on return $out; } } $len=strlen($comcode); $lines=new ocp_tempcode(); $number=1; $sofar=''; $line=NULL; for ($i=0;$i<$len;$i++) { $char=$comcode[$i]; if ($i==$pos) { $tmp_tpl=do_template('COMCODE_MISTAKE_ERROR'); $sofar.=$tmp_tpl->evaluate(); $line=$number; } if ($char==chr(10)) { $lines->attach(do_template('COMCODE_MISTAKE_LINE',array('_GUID'=>'2022be3de10590d525f333b6ac0da37b','NUMBER'=>integer_format($number),'LINE'=>make_string_tempcode($sofar)))); $sofar=''; $number++; } $sofar.=escape_html($char); } if ($i==$pos) { $tmp_tpl=do_template('COMCODE_MISTAKE_ERROR'); $sofar.=$tmp_tpl->evaluate(); } $lines->attach(do_template('COMCODE_MISTAKE_LINE',array('_GUID'=>'eebfe1342f3129d4e31fc9fc1963af2b','NUMBER'=>integer_format($number),'LINE'=>make_string_tempcode($sofar)))); if (is_null($line)) $line=$number; // Now, using some kind of miracle, we need to find out what parameter name blew-up. Let's look through the parameters and see what // is equal to $comcode. I'd rather not do this in a hackerish way - but the architecture was not designed for this. $name=NULL; foreach ($_POST as $key=>$val) { if (!is_string($val)) continue; if (post_param($key)==$comcode) // We have to use post_param, because it might be unix-converted / word-filtered { $name=$key; break; } } if (is_null($name)) { if ($check_only) // Maybe it has been appended with something else, so search deeper (we suspect this as we have been explictly asked to check the Comcode) { foreach ($_POST as $key=>$val) { if (!is_string($val)) continue; $val=post_param($key); if ((strlen($val)>10) && ((strpos($comcode,$val)===0) || (strpos($comcode,$val)==strlen($comcode)-strlen($val)))) { $name=$key; break; } } } if (is_null($name)) warn_exit(do_lang_tempcode('COMCODE_ERROR',$message,integer_format($line))); } if (!running_script('comcode_convert')) // Don't want it running in background { $GLOBALS['HTTP_STATUS_CODE']='400'; if (!headers_sent()) { if (/*(!browser_matches('ie')) && */(strpos(ocp_srv('SERVER_SOFTWARE'),'IIS')===false)) header('HTTP/1.0 400 Bad Request'); } } // Output our error / correction form @ob_end_clean(); $hidden=build_keep_post_fields(array($name)); require_code('form_templates'); $fields=form_input_text_comcode(do_lang_tempcode('NEW'),do_lang_tempcode('COMCODE_REPLACEMENT'),$name,$comcode,true,NULL,true); $post_url=get_self_url(); $form=do_template('FORM',array('_GUID'=>'207bad1252add775029b34ba36e02856','URL'=>$post_url,'TEXT'=>'','HIDDEN'=>$hidden,'FIELDS'=>$fields,'SUBMIT_NAME'=>do_lang_tempcode('PROCEED'))); $output=do_template('COMCODE_MISTAKE_SCREEN',array('_GUID'=>'0010230e6612b0775566d07ddf54305a','EDITABLE'=>!running_script('preview'),'FORM'=>$form,'TITLE'=>get_screen_title('ERROR_OCCURRED'),'LINE'=>integer_format($line),'MESSAGE'=>$message,'LINES'=>$lines)); $echo=new ocp_tempcode(); if (!running_script('preview')) { $echo=globalise($output,NULL,'',true); $echo->handle_symbol_preprocessing(); } else { $echo->attach(do_template('STANDALONE_HTML_WRAP',array('TITLE'=>do_lang_tempcode('PREVIEW'),'FRAME'=>running_script('preview'),'TARGET'=>'_top','CONTENT'=>$output))); } $echo->evaluate_echo(); exit(); return new ocp_tempcode(); // to trick code checker } /** * Build some Comcode-syntax attribute parameters from our in-memory parameters. * * @param array In-memory array. * @return string Reconstructed syntax. */ function reinsert_parameters($attributes) { $out=''; foreach ($attributes as $key=>$val) { $out.=' '.$key.'="'.comcode_escape($val).'"'; } return $out; } /** * Test a URL as a broken link. * * @param URLPATH URL to test. * @param string Comcode tag type, to which the URL is associated. * @param string URL actually provided. * @param MEMBER The member who is responsible for this Comcode * @return tempcode Error message, or blank if no error. */ function test_url($url_full,$tag_type,$given_url,$source_member) { if (get_option('check_broken_urls')=='0') return new ocp_tempcode(); if (strpos($url_full,'{$')!==false) return new ocp_tempcode(); $temp_tpl=new ocp_tempcode(); if (!handle_has_checked_recently($url_full)) { $GLOBALS['COMCODE_PARSE_URLS_CHECKED']++; $test=($GLOBALS['COMCODE_PARSE_URLS_CHECKED']>=MAX_URLS_TO_READ)?'':http_download_file($url_full,0,false); if ((is_null($test)) && (in_array($GLOBALS['HTTP_MESSAGE'],array('404','could not connect to host')))) { $temp_tpl=do_template('WARNING_BOX',array('FOR_GUESTS'=>false,'WARNING'=>do_lang_tempcode('MISSING_URL_COMCODE',$tag_type,escape_html($url_full)))); if (array_key_exists('COMCODE_BROKEN_URLS',$GLOBALS)) { $GLOBALS['COMCODE_BROKEN_URLS'][]=array($url_full,NULL); } elseif ((!in_array(get_page_name(),$GLOBALS['DONT_CARE_MISSING_PAGES'])) && (!running_script('iframe')) && (!running_script('comcode_convert')) && (!running_script('preview'))) { $found_in_post=false; // We don't want to send email if someone's just posting it right now, because they'll see the error on their screen, and we don't want staff spammed by member mistakes foreach ($_POST as $val) { if (is_array($_POST)) continue; if (get_magic_quotes_gpc()) $val=stripslashes($val); if ((is_string($val)) && (strpos($val,$given_url)!==false)) $found_in_post=true; } if (!$found_in_post) { require_code('failure'); relay_error_notification(do_lang('MISSING_URL_COMCODE',$tag_type,$url_full),false,$GLOBALS['FORUM_DRIVER']->is_staff($source_member)?'error_occurred_missing_reference_important':'error_occurred_missing_reference'); } } } } return $temp_tpl; } /** * Render a code box. * * @param string The data type (e.g. file extension) we are rendering. * @param tempcode Contents (code) to render. * @param boolean Whether to show line numbers. * @param boolean Whether what we have came from inside a semihtml tag * @param boolean Whether what we have came from semihtml mode * @return array A pair: The tempcode for the code box, and the title of the box */ function do_code_box($type,$embed,$numbers=true,$in_semihtml=false,$is_all_semihtml=false) { $_embed=mixed(); $title=mixed(); if ((file_exists(get_file_base().'/sources/geshi/'.filter_naughty(($type=='HTML')?'html4strict':strtolower($type)).'.php')) || (file_exists(get_file_base().'/sources_custom/geshi/'.filter_naughty(($type=='HTML')?'html4strict':strtolower($type)).'.php'))) { $evaluated=$embed->evaluate(); if (($in_semihtml) || ($is_all_semihtml)) { require_code('comcode_from_html'); $evaluated=semihtml_to_comcode($evaluated); } require_code('geshi'); if (class_exists('GeSHi')) { require_code('developer_tools'); destrictify(false); $geshi=new GeSHi($evaluated,($type=='HTML')?'html4strict':strtolower($type)); $geshi->set_header_type(GESHI_HEADER_DIV); if ($numbers) $geshi->enable_line_numbers(GESHI_NORMAL_LINE_NUMBERS); require_lang('comcode'); $title=do_lang_tempcode('comcode:CODE_IN_LANGUAGE',escape_html($type)); require_code('xhtml'); $_embed=xhtmlise_html($geshi->parse_code()); restrictify(); } } else { switch (strtolower($type)) { case 'php': if (!function_exists('highlight_string')) break; $evaluated=$embed->evaluate(); if (($in_semihtml) || ($is_all_semihtml)) { require_code('comcode_from_html'); $evaluated=semihtml_to_comcode($evaluated); } if (strpos($evaluated,'<'.'?php')===false) { $strip=true; $evaluated="<"."?php\n".$evaluated."\n?".">"; } else $strip=false; require_code('xhtml'); if (defined('HIPHOP_PHP')) { $h_result=nl2br(escape_html($evaluated)); } else { ob_start(); highlight_string($evaluated); $h_result=ob_get_contents(); ob_end_clean(); } $_embed=xhtmlise_html($h_result); if ($strip) { $_embed=str_replace('<?php<br />','',$_embed); $_embed=str_replace('?>','',$_embed); } $title=do_lang_tempcode('PHP_CODE'); break; } } return array($_embed,$title); } /** * Get tempcode for a Comcode tag. This function should always return (errors should be placed in the Comcode output stream), for stability reasons (i.e. if you're submitting something, you can't have the whole submit process die half way through in an unstructured fashion). * * @param string The tag being converted * @param array A map of the attributes (name=>val) for the tag. Val is usually a string, although in select places, the XML parser may pass tempcode. * @param mixed Tempcode of the inside of the tag ([between]THIS[/between]); the XML parser may pass in special stuff here, which is interpreted only for select tags * @param boolean Whether we are allowed to proceed even if this tag is marked as 'dangerous' * @param string A special identifier to mark where the resultant tempcode is going to end up (e.g. the ID of a post) * @param integer The position this tag occurred at in the Comcode * @param MEMBER The member who is responsible for this Comcode * @param boolean Whether to check as arbitrary admin * @param object The database connection to use * @param string The whole chunk of comcode * @param boolean Whether this is for WML output (no longer supported) * @param boolean Whether this is only a structure sweep * @param boolean Whether we are in semi-parse-mode (some tags might convert differently) * @param ?array A list of words to highlight (NULL: none) * @param ?MEMBER The member we are running on behalf of, with respect to how attachments are handled; we may use this members attachments that are already within this post, and our new attachments will be handed to this member (NULL: member evaluating) * @param boolean Whether what we have came from inside a semihtml tag * @param boolean Whether what we have came from semihtml mode * @return tempcode The tempcode for the Comcode */ function _do_tags_comcode($tag,$attributes,$embed,$comcode_dangerous,$pass_id,$marker,$source_member,$as_admin,$connection,&$comcode,$wml,$structure_sweep,$semiparse_mode,$highlight_bits=NULL,$on_behalf_of_member=NULL,$in_semihtml=false,$is_all_semihtml=false) { if (($structure_sweep) && ($tag!='title')) return new ocp_tempcode(); $param_given=isset($attributes['param']); if ((!isset($attributes['param'])) && ($tag!='block')) $attributes['param']=''; global $DANGEROUS_TAGS,$STRUCTURE_LIST,$COMCODE_PARSE_TITLE; if ((isset($DANGEROUS_TAGS[$tag])) && (!$comcode_dangerous)) { //if (($source_member==get_member()) && (strpos(serialize($_POST),'['.$tag)!==false)) { $username=$GLOBALS['FORUM_DRIVER']->get_username($source_member); if ($semiparse_mode) // Can't load through error for this, so just show it as a tag { $params=''; foreach ($attributes as $key=>$val) { $params.=' '.$key.'="'.str_replace('"','\"',$val).'"'; } return make_string_tempcode('<input class="ocp_keep_ui_controlled" size="45" title="['.$tag.''.(escape_html($params)).']'.((($in_semihtml) || ($is_all_semihtml))?$embed->evaluate():(escape_html($embed->evaluate()))).'[/'.$tag.']" type="text" value="'.($tag=='block'?do_lang('COMCODE_EDITABLE_BLOCK',escape_html($embed->evaluate())):do_lang('COMCODE_EDITABLE_TAG',escape_html($tag))).'" />'); } return do_template('WARNING_BOX',array('WARNING'=>do_lang_tempcode('comcode:NO_ACCESS_FOR_TAG',escape_html($tag),escape_html($username)))); } //return new ocp_tempcode(); } // These are just bbcode compatibility tags.. we will remap to our proper comcode if ($tag=='php') { $attributes['param']='php'; $tag='code'; } elseif ($tag=='sql') { $attributes['param']='sql'; $tag='code'; } elseif ($tag=='codebox') { $attributes['scroll']=1; $tag='code'; } elseif ($tag=='left') { $attributes['param']='left'; $tag='align'; } elseif ($tag=='center') { $attributes['param']='center'; $tag='align'; } elseif ($tag=='right') { $attributes['param']='right'; $tag='align'; } elseif ($tag=='thread') { $tag='topic'; } elseif (($tag=='internal_table') || ($tag=='external_table')) { $tag='box'; if (array_key_exists('class',$attributes)) $attributes['type']=$attributes['class']; } if ($semiparse_mode) // We have got to this point because we want to provide a special 'button' editing representation for these tags { $non_text_tags=array('attachment','section_controller','big_tab_controller','currency','block','contents','concepts','flash','menu','email','reference','upload','page','exp_thumb','exp_ref','thumb','snapback','post','thread','topic','include','random','jumping','shocker'); // Also in JAVASCRIPT_EDITING.tpl if ($tag=='attachment_safe') { if (preg_match('#^new\_\d+$#',$embed->evaluate())!=0) $non_text_tags[]='attachment_safe'; } if (in_array($tag,$non_text_tags)) { $params=''; foreach ($attributes as $key=>$val) { $params.=' '.$key.'="'.str_replace('"','\"',$val).'"'; } return make_string_tempcode('<input class="ocp_keep_ui_controlled" size="45" title="['.$tag.''.(escape_html($params)).']'.((($in_semihtml) || ($is_all_semihtml))?$embed->evaluate():(escape_html($embed->evaluate()))).'[/'.$tag.']" type="text" value="'.($tag=='block'?do_lang('COMCODE_EDITABLE_BLOCK',escape_html($embed->evaluate())):do_lang('COMCODE_EDITABLE_TAG',escape_html($tag))).'" />'); } } $temp_tpl=new ocp_tempcode(); switch ($tag) { case 'no_parse': $temp_tpl->attach($embed); break; case 'currency': if (addon_installed('ecommerce')) { $bracket=(array_key_exists('bracket',$attributes) && ($attributes['bracket']=='1')); if ($attributes['param']=='') $attributes['param']=get_option('currency'); $temp_tpl=do_template('COMCODE_CURRENCY',array('_GUID'=>'ee1fcdae082af6397ff3bad89006e012','AMOUNT'=>$embed,'FROM_CURRENCY'=>$attributes['param'],'BRACKET'=>$bracket)); } break; case 'overlay': $x=strval(array_key_exists('x',$attributes)?intval($attributes['x']):100); $y=strval(array_key_exists('y',$attributes)?intval($attributes['y']):100); $width=strval(array_key_exists('width',$attributes)?intval($attributes['width']):300); $height=strval(array_key_exists('height',$attributes)?intval($attributes['height']):300); $timein=strval(array_key_exists('timein',$attributes)?intval($attributes['timein']):0); $timeout=strval(array_key_exists('timeout',$attributes)?intval($attributes['timeout']):-1); $temp_tpl=do_template('COMCODE_OVERLAY',array('_GUID'=>'dfd0f7a72cc2bf6b613b28f8165a0034','UNIQ_ID'=>'a'.uniqid(''),'EMBED'=>$embed,'ID'=>($attributes['param']!='')?$attributes['param']:('rand'.uniqid('')),'X'=>$x,'Y'=>$y,'WIDTH'=>$width,'HEIGHT'=>$height,'TIMEIN'=>$timein,'TIMEOUT'=>$timeout)); break; case 'code': list($_embed,$title)=do_code_box($attributes['param'],$embed,(array_key_exists('numbers',$attributes)) && ($attributes['numbers']==1),$in_semihtml,$is_all_semihtml); if (!is_null($_embed)) { $tpl=(array_key_exists('scroll',$attributes) && ($attributes['scroll']==1))?'COMCODE_CODE_SCROLL':'COMCODE_CODE'; if (($tpl=='COMCODE_CODE_SCROLL') && (substr_count($_embed,chr(10))<10)) { $style='height: auto'; } else $style=''; $temp_tpl=do_template($tpl,array('_GUID'=>'c5d46d0927272fcacbbabcfab0ef6b0c','STYLE'=>$style,'TYPE'=>$attributes['param'],'CONTENT'=>$_embed,'TITLE'=>$title)); } else $_embed=''; if ($temp_tpl->is_empty()) { if (($in_semihtml) || ($is_all_semihtml)) // Yuck. We've double converted. Ideally we would have parsed a direct stream of HTML. But we could not do that for security reasons. { require_code('comcode_from_html'); $back_to_comcode=semihtml_to_comcode($embed->evaluate()); // Undo what's happened already //$back_to_comcode=html_entity_decode($back_to_comcode,ENT_QUOTES,get_charset()); // Remove the escaping entities that were inside the code tag $embed=comcode_to_tempcode($back_to_comcode,$source_member,$as_admin,80,$pass_id,$connection,true); // Re-parse (with full security) } $_embed=$embed->evaluate(); if ((!array_key_exists('scroll',$attributes)) && (strlen($_embed)>1000)) $attributes['scroll']=1; $tpl=(array_key_exists('scroll',$attributes) && ($attributes['scroll']==1))?'COMCODE_CODE_SCROLL':'COMCODE_CODE'; $title=do_lang_tempcode('CODE'); if (($tpl=='COMCODE_CODE_SCROLL') && (substr_count($_embed,chr(10))<10)) { $style='height: auto'; } else $style=''; $temp_tpl=do_template($tpl,array('CONTENT'=>$_embed,'TITLE'=>$title,'STYLE'=>$style,'TYPE'=>$attributes['param'])); } break; case 'list': if (is_array($embed)) { $parts=$embed; } else { $_embed=trim($embed->evaluate()); $_embed=str_replace('[/*]','',$_embed); $parts=explode('[*]',$_embed); } if (isset($temp_tpl->preprocessable_bits)) $temp_tpl->preprocessable_bits=array_merge($temp_tpl->preprocessable_bits,$embed->preprocessable_bits); $type=$attributes['param']; if ($type!='') { if ($type=='1') $type='decimal'; elseif ($type=='a') $type='lower-alpha'; elseif ($type=='i') $type='lower-roman'; elseif ($type=='x') $type='none'; elseif (!in_array($type,array('circle','disc','square','armenian','decimal','decimal-leading-zero','georgian','lower-alpha','lower-greek','lower-latin','lower-roman','upper-alpha','upper-latin','upper-roman'))) $type='disc'; $tag=in_array($type,array('circle','disc','square'))?'ul':'ol'; $temp_tpl->attach('<'.$tag.' style="list-style-type: '.$type.'">'); foreach ($parts as $i=>$part) { if (($i==0) && (str_replace(array(' ','<br />',' '),array('','',''),trim($part))=='')) continue; $temp_tpl->attach('<li>'.preg_replace('#\<br /\>(\ |\s)*$#D','',preg_replace('#^\<br /\>(\ |\s)*#D','',$part)).'</li>'); } $temp_tpl->attach('</'.$tag.'>'); } else { $temp_tpl->attach('<ul>'); foreach ($parts as $i=>$part) { if (($i==0) && (str_replace(array(' ','<br />',' '),array('','',''),trim($part))=='')) continue; $temp_tpl->attach('<li>'.preg_replace('#\<br /\>(\ |\s)*$#D','',preg_replace('#^\<br /\>(\ |\s)*#D','',$part)).'</li>'); } $temp_tpl->attach('</ul>'); } break; case 'snapback': require_lang('ocf'); $post_id=intval($embed->evaluate()); $s_title=($attributes['param']=='')?do_lang_tempcode('FORUM_POST_NUMBERED',integer_format($post_id)):make_string_tempcode($attributes['param']); $forum=array_key_exists('forum',$attributes)?$attributes['forum']:''; $temp_tpl=do_template('COMCODE_SNAPBACK',array('URL'=>$GLOBALS['FORUM_DRIVER']->post_url($post_id,$forum),'TITLE'=>$s_title)); break; case 'post': require_lang('ocf'); $post_id=intval($embed->evaluate()); $s_title=($attributes['param']=='')?do_lang_tempcode('FORUM_POST_NUMBERED',integer_format($post_id)):make_string_tempcode($attributes['param']); $forum=array_key_exists('forum',$attributes)?$attributes['forum']:''; $temp_tpl->attach(hyperlink($GLOBALS['FORUM_DRIVER']->post_url($post_id,$forum),$s_title)); break; case 'topic': require_lang('ocf'); $topic_id=intval($embed->evaluate()); $s_title=($attributes['param']=='')?do_lang_tempcode('FORUM_TOPIC_NUMBERED',integer_format($topic_id)):make_string_tempcode($attributes['param']); $forum=array_key_exists('forum',$attributes)?$attributes['forum']:''; $temp_tpl->attach(hyperlink($GLOBALS['FORUM_DRIVER']->topic_url($topic_id,$forum),$s_title)); break; case 'staff_note': $temp_tpl=new ocp_tempcode(); return $temp_tpl; case 'section': $name=(array_key_exists('param',$attributes))?$attributes['param']:'section'.strval(mt_rand(0,100)); $default=(array_key_exists('default',$attributes))?$attributes['default']:'0'; $temp_tpl=do_template('COMCODE_SECTION',array('_GUID'=>'a902962ccdc80046c999d6fed907d105','PASS_ID'=>'x'.$pass_id,'DEFAULT'=>$default=='1','NAME'=>$name,'CONTENT'=>$embed)); break; case 'section_controller': $sections=explode(',',$embed->evaluate()); $temp_tpl=do_template('COMCODE_SECTION_CONTROLLER',array('_GUID'=>'133bf24892e9e3ec2a01146d6ec418fe','SECTIONS'=>$sections,'PASS_ID'=>'x'.$pass_id)); break; case 'big_tab': $name=(array_key_exists('param',$attributes))?$attributes['param']:'big_tab'.strval(mt_rand(0,100)); $default=(array_key_exists('default',$attributes))?$attributes['default']:'0'; $temp_tpl=do_template('COMCODE_BIG_TABS_TAB',array('_GUID'=>'f6219b1acd6999acae770da20b95fb99','PASS_ID'=>'x'.$pass_id,'DEFAULT'=>$default=='1','NAME'=>$name,'CONTENT'=>$embed)); break; case 'big_tab_controller': $tabs=explode(',',$embed->evaluate()); if (!array_key_exists('switch_time',$attributes)) $attributes['switch_time']='6000'; $temp_tpl=do_template('COMCODE_BIG_TABS_CONTROLLER',array('SWITCH_TIME'=>($attributes['switch_time']=='')?NULL:strval(intval($attributes['switch_time'])),'TABS'=>$tabs,'PASS_ID'=>'x'.$pass_id)); break; case 'tab': $default=(array_key_exists('default',$attributes))?$attributes['default']:'0'; $temp_tpl=do_template('COMCODE_TAB_BODY',array('DEFAULT'=>$default=='1','TITLE'=>trim($attributes['param']),'CONTENT'=>$embed)); break; case 'tabs': $heads=new ocp_tempcode(); $tabs=explode(',',$attributes['param']); foreach ($tabs as $i=>$tab) { $heads->attach(do_template('COMCODE_TAB_HEAD',array('TITLE'=>trim($tab),'FIRST'=>$i==0,'LAST'=>!array_key_exists($i+1,$tabs)))); } $temp_tpl=do_template('COMCODE_TAB_CONTROLLER',array('_GUID'=>'0e56cf180973c57f3633aae54dd9cddc','HEADS'=>$heads,'CONTENT'=>$embed)); break; case 'carousel': if ($attributes['param']=='') $attributes['param']='40'; $temp_tpl=do_template('COMCODE_CAROUSEL',array('_GUID'=>'2d0a327a6cb60e3168a5022eb0cfba9a','CONTENT'=>$embed,'SCROLL_AMOUNT'=>$attributes['param'])); break; case 'menu': $name=(array_key_exists('param',$attributes))?$attributes['param']:'mnu'.strval(mt_rand(0,100)); $type=(array_key_exists('type',$attributes))?$attributes['type']:'tree'; require_code('menus'); require_code('menus_comcode'); $temp_tpl=build_comcode_menu($embed->evaluate(),$name,$source_member,$type); break; case 'if_in_group': $groups=''; $_groups=explode(',',$attributes['param']); $all_groups=$GLOBALS['FORUM_DRIVER']->get_usergroup_list(); foreach ($_groups as $group) { $find=array_search($group,$all_groups); if ($find===false) { if ($groups!='') $groups.=','; $groups.=$group; } else { if ($groups!='') $groups.=','; $groups.=strval($find); } } $temp_tpl=do_template('COMCODE_IF_IN_GROUP',array('_GUID'=>'761a7cc07f7b4b68508d68ce19b87d2c','TYPE'=>array_key_exists('type',$attributes)?$attributes['type']:'','CONTENT'=>$embed,'GROUPS'=>$groups)); break; case 'acronym': case 'abbr': $temp_tpl=do_template('COMCODE_ABBR',array('_GUID'=>'acbc4f991dsf03f81b61919b74ac24c91','CONTENT'=>$embed,'TITLE'=>$attributes['param'])); break; case 'address': $temp_tpl=do_template('COMCODE_ADDRESS',array('_GUID'=>'acbcsdf9910703f81b61919b74ac24c91','CONTENT'=>$embed)); break; case 'dfn': $temp_tpl=do_template('COMCODE_DFN',array('_GUID'=>'acbc4f9910703f81b61sf19b74ac24c91','CONTENT'=>$embed)); break; case 'pulse': $min_color=array_key_exists('min',$attributes)?$attributes['min']:'0000FF'; $max_color=array_key_exists('max',$attributes)?$attributes['max']:'FF0044'; if (substr($min_color,0,1)=='#') $min_color=substr($min_color,1); if (substr($max_color,0,1)=='#') $max_color=substr($max_color,1); $speed=($attributes['param']=='')?100:intval($attributes['param']); $temp_tpl=do_template('COMCODE_PULSE',array('_GUID'=>'adsd4f9910sfd03f81b61919b74ac24c91','CONTENT'=>$embed,'MIN_COLOR'=>$min_color,'MAX_COLOR'=>$max_color,'SPEED'=>strval($speed))); break; case 'del': $cite=array_key_exists('cite',$attributes)?$attributes['cite']:NULL; if (!is_null($cite)) { $temp_tpl=test_url($cite,'del',$cite,$source_member); } $datetime=array_key_exists('datetime',$attributes)?$attributes['datetime']:NULL; $temp_tpl->attach(do_template('COMCODE_DEL',array('_GUID'=>'acsd4f9910sfd03f81b61919b74ac24c91','CONTENT'=>$embed,'CITE'=>$cite,'DATETIME'=>$datetime))); break; case 'ins': $cite=array_key_exists('cite',$attributes)?$attributes['cite']:NULL; if (!is_null($cite)) { $temp_tpl=test_url($cite,'ins',$cite,$source_member); if (!$temp_tpl->is_empty()) break; } $datetime=array_key_exists('datetime',$attributes)?$attributes['datetime']:NULL; $temp_tpl->attach(do_template('COMCODE_INS',array('_GUID'=>'asss4f9910703f81b61919bsfc24c91','CONTENT'=>$embed,'CITE'=>$cite,'DATETIME'=>$datetime))); break; case 'cite': $temp_tpl=do_template('COMCODE_CITE',array('_GUID'=>'acbcsf910703f81b61919b74ac24c91','CONTENT'=>$embed)); break; case 'b': if ($semiparse_mode) { $temp_tpl=make_string_tempcode('<b>'.$embed->evaluate().'</b>'); break; } $temp_tpl=do_template('COMCODE_BOLD',array('_GUID'=>'acbc4fds910703f81b619sf74ac24c91','CONTENT'=>$embed)); break; case 'align': $align=array_key_exists('param',$attributes)?$attributes['param']:'left'; $temp_tpl=do_template('COMCODE_ALIGN',array('_GUID'=>'950b4d9db12cac6bf536860bedd96a36','ALIGN'=>$align,'CONTENT'=>$embed)); break; case 'indent': $indent=array_key_exists('param',$attributes)?$attributes['param']:'10'; if (!is_numeric($indent)) $indent='10'; $temp_tpl=do_template('COMCODE_INDENT',array('_GUID'=>'d8e69fa17eebd5312e3ad5788e3a1343','INDENT'=>$indent,'CONTENT'=>$embed)); break; case 'surround': if (($semiparse_mode) && ($embed->evaluate()=='')) // This is probably some signal like a break, so show it in Comcode form { $temp_tpl=make_string_tempcode('<kbd class="ocp_keep" title="no_parse">[surround="'.comcode_escape(array_key_exists('param',$attributes)?$attributes['param']:'float_surrounder').'"]'.$embed->evaluate().'[/surround]</kbd>'); break; } $class=(array_key_exists('param',$attributes) && ($attributes['param']!=''))?$attributes['param']:'float_surrounder'; $temp_tpl=do_template('COMCODE_SURROUND',array('_GUID'=>'e8e69fa17eebd5312e3ad5788e3a1343','CLASS'=>$class,'CONTENT'=>$embed)); break; case 'i': if ($semiparse_mode) { $temp_tpl=make_string_tempcode('<i>'.$embed->evaluate().'</i>'); break; } $temp_tpl=do_template('COMCODE_ITALICS',array('_GUID'=>'4321a1fe3825418e57a29410183c0c60','CONTENT'=>$embed)); break; case 'u': if ($semiparse_mode) { $temp_tpl=make_string_tempcode('<u>'.$embed->evaluate().'</u>'); break; } $temp_tpl=do_template('COMCODE_UNDERLINE',array('_GUID'=>'69cc8e73b17f9e6a35eb1af2bd1dc6ab','CONTENT'=>$embed)); break; case 's': if ($semiparse_mode) { $temp_tpl=make_string_tempcode('<strike>'.$embed->evaluate().'</strike>'); break; } $temp_tpl=do_template('COMCODE_STRIKE',array('_GUID'=>'ed242591cefd365497cc0c63abbb11a9','CONTENT'=>$embed)); break; case 'tooltip': $param=is_object($attributes['param'])?$attributes['param']:comcode_to_tempcode($attributes['param'],$source_member,$as_admin,60,NULL,$connection,false,false,false,false,false,$highlight_bits,$on_behalf_of_member); $temp_tpl=do_template('COMCODE_TOOLTIP',array('_GUID'=>'c9f4793dc0c1a92cd7d08ae1b87c2308','URL'=>array_key_exists('url',$attributes)?$attributes['url']:'','TOOLTIP'=>$param,'CONTENT'=>$embed)); break; case 'sup': $temp_tpl=do_template('COMCODE_SUP',array('_GUID'=>'74d2ecfe193dacb6d922bc288828196a','CONTENT'=>$embed)); break; case 'sub': $temp_tpl=do_template('COMCODE_SUB',array('_GUID'=>'515e310e00a6d7c30f7dca0a5956ebcf','CONTENT'=>$embed)); break; case 'title': if (($semiparse_mode) && (strpos($comcode,'[contents')!==false)) { $temp_tpl=make_string_tempcode('[title'.reinsert_parameters($attributes).']'.$embed->evaluate().'[/title]'); break; } $level=($attributes['param']!='')?intval($attributes['param']):1; if ($level==0) $level=1; // Stop crazy Comcode causing stack errors with the toc $uniq_id=strval(count($STRUCTURE_LIST)); $STRUCTURE_LIST[]=array($level,$embed,$uniq_id); if ($level==1) $template='SCREEN_TITLE'; elseif ($level==2) $template='COMCODE_SECTION_TITLE'; elseif ($level==3) $template='COMCODE_MINOR_TITLE'; elseif ($level==4) $template='COMCODE_VERY_MINOR_TITLE'; else $template='COMCODE_VERY_MINOR_TITLE'; if ($level==1) { if (is_null($COMCODE_PARSE_TITLE)) { $COMCODE_PARSE_TITLE=$embed->evaluate(); if (is_object($COMCODE_PARSE_TITLE)) $COMCODE_PARSE_TITLE=$COMCODE_PARSE_TITLE->evaluate(); } } $base=array_key_exists('base',$attributes)?intval($attributes['base']):2; if ((array_key_exists('number',$attributes)) && ($level>=$base)) { $list_types=($attributes['number']=='')?array():explode(',',$attributes['number']); $list_types+=array('decimal','lower-alpha','lower-roman','upper-alpha','upper-roman','disc'); $numerals=array('i','ii','iii','iv','v','vi','viii','ix','x','xi','xii','xiii','xiv','xv','xvi','xvii','xviii','xix','xx'); $symbol_lookup=array('decimal'=>range(1,100),'lower-alpha'=>range('a','z'),'lower-roman'=>$numerals,'upper-alpha'=>range('A','Z'),'upper-roman'=>str_replace('i','I',str_replace('v','V',str_replace('x','X',$numerals)))); $level_text=''; $list_pos=count($STRUCTURE_LIST)-2; for ($j=$level;$j>=$base;$j--) { $num_before=0; for ($i=$list_pos;$i>=0;$i--) { $list_pos--; if ($STRUCTURE_LIST[$i][0]==$j-1) { break; } if ($STRUCTURE_LIST[$i][0]==$j) $num_before++; } $level_number=strval($symbol_lookup[$list_types[$j-$base]][$num_before]); $level_text=$level_number.(($level_text!='')?'.':'').$level_text; } $old_embed=$embed; $embed=make_string_tempcode($level_text.' – '); $embed->attach($old_embed); } if ($semiparse_mode) { $temp_tpl=make_string_tempcode('<h'.strval($level).(($level==1)?' class="screen_title"':'').'><span class="inner">'.$embed->evaluate().'</span></h'.strval($level).'>'); break; } $tpl_map=array('ID'=>(substr($pass_id,0,5)=='panel')?NULL:$uniq_id,'TITLE'=>$embed,'HELP_URL'=>'','HELP_TERM'=>''); if (array_key_exists('sub',$attributes)) $tpl_map['SUB']=protect_from_escaping(comcode_to_tempcode($attributes['sub'],$source_member,$as_admin,60,NULL,$connection,false,false,false,false,false,$highlight_bits,$on_behalf_of_member)); $temp_tpl=do_template($template,$tpl_map); break; case 'attachment': case 'attachment_safe': require_code('attachments'); if (is_null($on_behalf_of_member)) $on_behalf_of_member=$source_member; $id=$embed->evaluate(); global $COMCODE_ATTACHMENTS; if ((!is_numeric($id)) && (!$as_admin) && (!has_specific_permission($source_member,'exceed_filesize_limit'))) { // We work all this out before we do any downloads, to make sure orphaned files aren't dumped on the file system (possible hack method) if (get_forum_type()=='ocf') { require_lang('ocf'); require_code('ocf_groups'); $daily_quota=ocf_get_member_best_group_property($source_member,'max_daily_upload_mb'); } else { $daily_quota=5; // 5 is a hard coded default for non-OCF forums } if (!is_null($daily_quota)) { $_size_uploaded_today=$connection->query('SELECT SUM(a_file_size) AS the_answer FROM '.$connection->get_table_prefix().'attachments WHERE a_member_id='.strval((integer)$source_member).' AND a_add_time>'.strval(time()-60*60*24)); if (is_null($_size_uploaded_today[0]['the_answer'])) $_size_uploaded_today[0]['the_answer']=0; $size_uploaded_today=ceil(((float)$_size_uploaded_today[0]['the_answer'])/1024.0/1024.0); $attach_size=0; require_code('uploads'); is_swf_upload(true); foreach ($_FILES as $_file) $attach_size+=floatval($_file['size'])/1024.0/1024.0; if (($size_uploaded_today+$attach_size)>floatval($daily_quota)) { $temp_tpl=do_template('WARNING_BOX',array('WARNING'=>do_lang_tempcode('OVER_DAILY_QUOTA',integer_format($daily_quota),float_format($size_uploaded_today)))); break; } } } $thumb_url=array_key_exists('thumb_url',$attributes)?$attributes['thumb_url']:''; // Embedded attachments if ((!is_numeric($id)) && (substr($id,0,4)!='new_') && (substr($id,0,4)!='url_')) { $file=base64_decode(str_replace(chr(10),'',$id)); if ($file===false) { $temp_tpl=do_template('WARNING_BOX',array('WARNING'=>do_lang_tempcode('comcode:CORRUPT_ATTACHMENT'))); break; } $md5=md5(substr($file,0,30)); $original_filename=array_key_exists('filename',$attributes)?$attributes['filename']:($md5.'.dat'); if (get_file_extension($original_filename)!='dat') { require_code('files2'); check_extension($original_filename,true); $new_filename=$md5.'.'.get_file_extension($original_filename).'.dat'; } else { $new_filename=$md5.'.'.get_file_extension($original_filename); } $path=get_custom_file_base().'/uploads/attachments/'.$new_filename; $myfile=@fopen($path,'wb'); if ($myfile===false) { $temp_tpl=do_template('WARNING_BOX',array('WARNING'=>intelligent_write_error_inline($path))); break; } if (fwrite($myfile,$file)<strlen($file)) warn_exit(do_lang_tempcode('COULD_NOT_SAVE_FILE')); fclose($myfile); fix_permissions($path); sync_file($path); $_size=strlen($file); $url='uploads/attachments/'.$new_filename; if ($connection->connection_write!=$GLOBALS['SITE_DB']->connection_write) $url=get_custom_base_url().'/'.$url; // Thumbnail if ($thumb_url=='') { require_code('images'); if (is_image($original_filename)) { $gd=((get_option('is_on_gd')=='1') && (function_exists('imagetypes'))); if ($gd) { require_code('images'); if (!is_saveable_image($url)) $ext='.png'; else $ext='.'.get_file_extension($original_filename); $thumb_url='uploads/attachments_thumbs/'.$md5.$ext; convert_image(get_custom_base_url().'/'.$url,get_custom_file_base().'/'.$thumb_url,-1,-1,intval(get_option('thumb_width')),true,NULL,false,true); if ($connection->connection_write!=$GLOBALS['SITE_DB']->connection_write) $thumb_url=get_custom_base_url().'/'.$thumb_url; } else $thumb_url=$url; } } if (addon_installed('galleries')) { require_code('images'); if ((is_video($url)) && ($connection->connection_read==$GLOBALS['SITE_DB']->connection_read)) { require_code('transcoding'); $url=transcode_video($url,'attachments','a_url','a_original_filename',NULL,NULL); } } $attachment=array( 'a_member_id'=>$on_behalf_of_member, 'a_file_size'=>$_size, 'a_url'=>$url, 'a_thumb_url'=>$thumb_url, 'a_original_filename'=>$original_filename, 'a_num_downloads'=>0, 'a_last_downloaded_time'=>NULL, 'a_add_time'=>time()); $attachment['a_description']=array_key_exists('description',$attributes)?(is_object($attributes['description'])?('[html]'.$attributes['description']->evaluate().'[/html]'):$attributes['description']):''; $attach_id=$connection->query_insert('attachments',$attachment,true); $attachment['id']=$attach_id; // Create and document attachment if (!array_key_exists('type',$attributes)) $attributes['type']='auto'; $COMCODE_ATTACHMENTS[$pass_id][]=array('tag_type'=>$tag,'type'=>'new','attachmenttype'=>$attributes['type'],'description'=>$attachment['a_description'],'id'=>intval($attach_id),'marker'=>$marker,'comcode'=>$comcode); // Marker will allow us to search back and replace this with the added id } // New attachments elseif (!is_numeric($id)) { require_code('uploads'); if (substr($id,0,4)=='new_') { $_id=substr($id,4); if (!is_numeric($_id)) { $temp_tpl=do_template('WARNING_BOX',array('WARNING'=>do_lang_tempcode('comcode:INVALID_ATTACHMENT'))); break; } $attributes['type']=post_param('attachmenttype'.$_id,array_key_exists('type',$attributes)?$attributes['type']:'auto'); if (substr($attributes['type'],-8)=='_extract') $attributes['type']=substr($attributes['type'],0,strlen($attributes['type'])-8); $urls=get_url('','file'.$_id,'uploads/attachments',2,OCP_UPLOAD_ANYTHING,((!array_key_exists('thumb',$attributes)) || ($attributes['thumb']!='0')) && ($thumb_url==''),'','',true,true,true,true); if ($urls[0]=='') return new ocp_tempcode();//warn_exit(do_lang_tempcode('ERROR_UPLOADING')); Can't do this, because this might not be post-calculated if something went wrong once is_swf_upload(true); $_size=$_FILES['file'.$_id]['size']; $original_filename=$_FILES['file'.$_id]['name']; if (get_magic_quotes_gpc()) $original_filename=stripslashes($original_filename); } elseif (substr($id,0,4)=='url_') { if ((!has_specific_permission($source_member,'draw_to_server')) && (!$as_admin)) break; $_id='!'; $attributes['type']=post_param('attachmenttype'.$_id,array_key_exists('type',$attributes)?$attributes['type']:'auto'); $url=remove_url_mistakes(substr($id,4)); $_POST['_specify_url']=$url; // Little hack, as we need to read it from a POST if (get_magic_quotes_gpc()) $_POST['_specify_url']=addslashes($_POST['_specify_url']); $urls=get_url('_specify_url','','uploads/filedump',1,OCP_UPLOAD_ANYTHING,((!array_key_exists('thumb',$attributes)) || ($attributes['thumb']!='0')) && ($thumb_url==''),'','',true); $original_filename=rawurldecode(substr($url,strrpos($url,'/')+1)); if (url_is_local($urls[0])) { $_size=@filesize(get_custom_file_base().'/'.rawurldecode($urls[0])); if ($_size===false) $_size=filesize(get_file_base().'/'.rawurldecode($urls[0])); } else $_size=0; } else { $temp_tpl=do_template('WARNING_BOX',array('WARNING'=>do_lang_tempcode('comcode:INVALID_ATTACHMENT'))); break; } if ($urls[0]=='') { require_code('images'); require_code('files2'); $temp_tpl=do_template('WARNING_BOX',array('WARNING'=>do_lang_tempcode('ATTACHMENT_WOULD_NOT_UPLOAD',float_format(get_max_file_size()/1024/1024),float_format(get_max_image_size()/1024/1024)))); break; } $url=$urls[0]; if ($connection->connection_write!=$GLOBALS['SITE_DB']->connection_write) $url=get_custom_base_url().'/'.$url; if ($thumb_url=='') $thumb_url=array_key_exists(1,$urls)?$urls[1]:''; if (($thumb_url!='') && ($connection!=$GLOBALS['SITE_DB'])) $thumb_url=get_custom_base_url().'/'.$thumb_url; $num_downloads=0; $last_downloaded_time=NULL; $add_time=time(); $member_id=$on_behalf_of_member; if (addon_installed('galleries')) { require_code('images'); if ((is_video($url)) && ($connection->connection_read==$GLOBALS['SITE_DB']->connection_read)) { require_code('transcoding'); $url=transcode_video($url,'attachments','a_url','a_original_filename',NULL,NULL); } } $attachment=array( 'a_member_id'=>$member_id, 'a_file_size'=>$_size, 'a_url'=>$url, 'a_thumb_url'=>$thumb_url, 'a_original_filename'=>$original_filename, 'a_num_downloads'=>$num_downloads, 'a_last_downloaded_time'=>$last_downloaded_time, 'a_add_time'=>$add_time); $attachment['a_description']=post_param('caption'.$_id,array_key_exists('description',$attributes)?(is_object($attributes['description'])?('[html]'.$attributes['description']->evaluate().'[/html]'):$attributes['description']):''); $attach_id=$connection->query_insert('attachments',$attachment,true); $attachment['id']=$attach_id; if (($tag=='attachment_safe') || (substr($id,0,4)=='url_')) // Lock it if we are starting with this tag { $connection->query_delete('attachment_refs',array('r_referer_type'=>'null','r_referer_id'=>'','a_id'=>$attachment['id']),'',1); $connection->query_insert('attachment_refs',array('r_referer_type'=>'null','r_referer_id'=>'','a_id'=>$attachment['id'])); } // Create and document attachment $COMCODE_ATTACHMENTS[$pass_id][]=array('tag_type'=>$tag,'time'=>time(),'type'=>(substr($id,0,4)=='new_')?'new':'url','attachmenttype'=>$attributes['type'],'description'=>$attachment['a_description'],'id'=>intval($attach_id),'marker'=>$marker,'comcode'=>$comcode); // Marker will allow us to search back and replace this with the added id // Existing attachments } else { $__id=intval($id); // Check we have permission to re-use this $owner=$connection->query_value_null_ok('attachments','a_member_id',array('id'=>$__id)); if (is_null($owner)) // Missing attachment! { $temp_tpl=do_template('WARNING_BOX',array('WARNING'=>do_lang_tempcode('MISSING_RESOURCE_COMCODE','attachment',escape_html(strval($__id))))); if ((!in_array(get_page_name(),$GLOBALS['DONT_CARE_MISSING_PAGES'])) && (!running_script('iframe'))) { require_code('failure'); relay_error_notification(do_lang('MISSING_RESOURCE_COMCODE','attachment',strval($__id)),false,$GLOBALS['FORUM_DRIVER']->is_staff($source_member)?'error_occurred_missing_reference_important':'error_occurred_missing_reference'); } break; } $_attachment=$connection->query_select('attachments',array('*'),array('id'=>$__id),'',1); $attachment=$_attachment[0]; $already_referenced=array_key_exists($__id,$GLOBALS['ATTACHMENTS_ALREADY_REFERENCED']); if (($already_referenced) || ($as_admin) || (/*(!is_guest($source_member)) && */($source_member===$owner)) || (((has_specific_permission($source_member,'reuse_others_attachments')) || ($owner==$source_member)) && (has_attachment_access($source_member,$__id)))) { if (!array_key_exists('type',$attributes)) $attributes['type']='auto'; $COMCODE_ATTACHMENTS[$pass_id][]=array('tag_type'=>$tag,'time'=>$attachment['a_add_time'],'type'=>'existing','id'=>$__id,'attachmenttype'=>$attributes['type'],'marker'=>$marker,'comcode'=>$comcode); } else { require_lang('permissions'); $temp_tpl=do_template('WARNING_BOX',array('WARNING'=>do_lang_tempcode('permissions:ACCESS_DENIED__REUSE_ATTACHMENT',$GLOBALS['FORUM_DRIVER']->get_username($source_member)))); break; //access_denied('REUSE_ATTACHMENT'); } if ($connection->connection_write!=$GLOBALS['SITE_DB']->connection_write) { if (url_is_local($attachment['a_url'])) $attachment['a_url']=get_custom_base_url().'/'.$attachment['a_url']; if (url_is_local($attachment['a_url'])) $attachment['a_thumb_url']=get_custom_base_url().'/'.$attachment['a_thumb_url']; } $attachment['a_description']=array_key_exists('description',$attributes)?(is_object($attributes['description'])?('[html]'.$attributes['description']->evaluate().'[/html]'):$attributes['description']):$attachment['a_description']; } // Now, render it // ============== $temp_tpl=render_attachment($tag,$attributes,$attachment,$pass_id,$source_member,$as_admin,$connection,$highlight_bits,$on_behalf_of_member,$semiparse_mode); if (array_key_exists('float',$attributes)) $temp_tpl=do_template('FLOATER',array('_GUID'=>'802fe29019be80993296de7cc8b5cc5e','FLOAT'=>$attributes['float'],'CONTENT'=>$temp_tpl)); break; case 'include': $codename=$embed->evaluate(); $zone=$attributes['param']; if ($zone=='_SEARCH') $zone=get_comcode_zone($codename); if ($zone=='_SELF') $zone=get_zone_name(); $temp_comcode_parse_title=$COMCODE_PARSE_TITLE; $temp=request_page($codename,false,$zone,NULL,true); $COMCODE_PARSE_TITLE=$temp_comcode_parse_title; if ($temp->is_empty()) { $temp_tpl=do_template('WARNING_BOX',array('WARNING'=>do_lang_tempcode('MISSING_RESOURCE_COMCODE','include',hyperlink(build_url(array('page'=>'cms_comcode_pages','type'=>'_ed','page_link'=>$zone.':'.$codename),get_module_zone('cms_comcode_pages')),$zone.':'.$codename,false,true)))); if ((!in_array(get_page_name(),$GLOBALS['DONT_CARE_MISSING_PAGES'])) && (!running_script('iframe'))) { require_code('failure'); relay_error_notification(do_lang('MISSING_RESOURCE_COMCODE','include',$zone.':'.$codename),false,$GLOBALS['FORUM_DRIVER']->is_staff($source_member)?'error_occurred_missing_reference_important':'error_occurred_missing_reference'); } } else { $temp_tpl=symbol_tempcode('LOAD_PAGE',array($codename,$zone)); } break; case 'random': unset($attributes['param']); $max=($embed->evaluate()=='')?intval($embed->evaluate()):0; foreach ($attributes as $num=>$val) { $_temp=is_object($val)?$val:comcode_to_tempcode($val,$source_member,$as_admin,60,NULL,$connection,false,false,false,false,false,$highlight_bits,$on_behalf_of_member); $attributes[$num]=$_temp->evaluate(); if (intval($num)>$max) $max=intval($num); } $_parts=new ocp_tempcode(); krsort($attributes); foreach ($attributes as $num=>$val) { $_parts->attach(do_template('COMCODE_RANDOM_PART',array('_GUID'=>'5fa49a916304f9caa0ddedeb01531142','NUM'=>strval($num),'VAL'=>$val))); } $temp_tpl=do_template('COMCODE_RANDOM',array('_GUID'=>'9b77aaf593b12c763fb0c367fab415b6','UNIQID'=>uniqid(''),'FULL'=>$embed,'MAX'=>strval($max),'PARTS'=>$_parts)); break; case 'jumping': unset($attributes['param']); $_parts=new ocp_tempcode(); foreach ($attributes as $val) { $_temp=is_object($val)?$val:comcode_to_tempcode($val,$source_member,$as_admin,60,NULL,$connection,false,false,false,false,false,$highlight_bits,$on_behalf_of_member); $_parts->attach(do_template('COMCODE_JUMPING_PART',array('_GUID'=>'d163bd11920f39f0cb8ff2f6ba48bc80','PART'=>$_temp->evaluate()))); } $embed=$embed->evaluate(); $temp_tpl=do_template('COMCODE_JUMPING',array('_GUID'=>'85e9f83ed134868436a7db7692f56047','UNIQID'=>uniqid(''),'FULL'=>implode(', ',$attributes),'TIME'=>strval((integer)$embed),'PARTS'=>$_parts)); break; case 'shocker': $_parts=new ocp_tempcode(); foreach ($attributes as $key=>$val) { if (substr($key,0,5)=='left_') { $left=$val; $right=array_key_exists('right_'.substr($key,5),$attributes)?$attributes['right_'.substr($key,5)]:''; $left=is_object($left)?$left:comcode_to_tempcode($left,$source_member,$as_admin,60,NULL,$connection,false,false,false,false,false,$highlight_bits,$on_behalf_of_member); $right=is_object($right)?$right:comcode_to_tempcode($right,$source_member,$as_admin,60,NULL,$connection,false,false,false,false,false,$highlight_bits,$on_behalf_of_member); $_parts->attach(do_template('COMCODE_SHOCKER_PART',array('_GUID'=>'512b1cfef8fe56597ae440e924bf38a7','LEFT'=>$left,'RIGHT'=>$right))); } } $min_color=array_key_exists('min',$attributes)?$attributes['min']:'0000FF'; $max_color=array_key_exists('max',$attributes)?$attributes['max']:'FF0044'; if (substr($min_color,0,1)=='#') $min_color=substr($min_color,1); if (substr($max_color,0,1)=='#') $max_color=substr($max_color,1); $embed=$embed->evaluate(); $temp_tpl=do_template('COMCODE_SHOCKER',array('UNIQID'=>uniqid(''),'MIN_COLOR'=>$min_color,'MAX_COLOR'=>$max_color,'FULL'=>implode(', ',$attributes),'TIME'=>strval((integer)$embed),'PARTS'=>$_parts)); break; case 'ticker': $width=$attributes['param']; if (!is_numeric($width)) $width='300'; $fspeed=array_key_exists('speed',$attributes)?float_to_raw_string(floatval($attributes['speed'])):'1'; $temp_tpl=do_template('COMCODE_TICKER',array('_GUID'=>'e48893cda61995261577f0556443c537','UNIQID'=>uniqid(''),'SPEED'=>$fspeed,'WIDTH'=>$width,'TEXT'=>$embed)); break; case 'highlight': $temp_tpl=do_template('COMCODE_HIGHLIGHT',array('_GUID'=>'695d041b6605f06ec2aeee1e82f87185','CONTENT'=>$embed)); break; case 'size': $size=array_key_exists('param',$attributes)?($attributes['param']):'1'; if (is_numeric($size)) { $size='font-size: '.$size.'em;'; } elseif (substr($size,0,1)=='+') { $size='font-size: '.substr($size,1).'em'; } elseif (substr($size,-1)=='%') { $size='font-size: '.float_to_raw_string(floatval(substr($size,0,strlen($size)-1))/100.0).'em'; } elseif (substr($size,-2)=='of') { $new_size='1em'; switch ($size) { case '1of': $new_size='8pt'; break; case '2of': $new_size='10pt'; break; case '3of': $new_size='12pt'; break; case '4of': $new_size='14pt'; break; case '5of': $new_size='18pt'; break; case '6of': $new_size='24pt'; break; case '7of': $new_size='36pt'; break; } $size='font-size: '.$new_size; } else { $size='font-size: '.$size; } $size_len=strlen($size); filter_html($as_admin,$source_member,0,$size_len,$size,false,false); $temp_tpl=do_template('COMCODE_FONT',array('_GUID'=>'fb23fdcb45aabdfeca9f37ed8098948e','CONTENT'=>$embed,'SIZE'=>$size,'COLOR'=>'','FACE'=>'')); break; case 'color': $color=array_key_exists('param',$attributes)?('color: '.$attributes['param'].';'):''; $temp_tpl=do_template('COMCODE_FONT',array('_GUID'=>'bd146414c9239ba2076f4b683df437d7','CONTENT'=>$embed,'SIZE'=>'','COLOR'=>$color,'FACE'=>'')); $color_len=strlen($color); filter_html($as_admin,$source_member,0,$color_len,$color,false,false); break; case 'tt': $temp_tpl=do_template('COMCODE_TELETYPE',array('_GUID'=>'422a4785fc9bb0d1a26a09a59184f107','CONTENT'=>$embed)); break; case 'samp': $temp_tpl=do_template('COMCODE_SAMP',array('_GUID'=>'386eddbd74f45a8596f2f21680df99f8','CONTENT'=>$embed)); break; case 'q': $temp_tpl=do_template('COMCODE_Q',array('_GUID'=>'ab5dc7cddf0ec01be969605cde87356c','CONTENT'=>$embed)); break; case 'var': $temp_tpl=do_template('COMCODE_VAR',array('_GUID'=>'75097f9f0de04bfd92507fdc07547237','CONTENT'=>$embed)); break; case 'font': $face=$attributes['param']; if (($face=='') && (array_key_exists('face',$attributes))) $face=$attributes['face']; $color=array_key_exists('color',$attributes)?$attributes['color']:''; $size=array_key_exists('size',$attributes)?$attributes['size']:''; if ($face=='/') $face=''; if ($color=='/') $color=''; if ($size=='/') $size=''; if ($color!='') $color='color: '.$color.';'; if ($size!='') { if (is_numeric($size)) { $size='font-size: '.$size.'em;'; } elseif (substr($size,0,1)=='+') { $size='font-size: '.substr($size,1).'em'; } elseif (substr($size,-1)=='%') { $size='font-size: '.float_to_raw_string(floatval(substr($size,0,strlen($size)-1))/100.0).'em'; } elseif (substr($size,-2)=='of') { $new_size='1em'; switch ($size) { case '1of': $new_size='8pt'; break; case '2of': $new_size='10pt'; break; case '3of': $new_size='12pt'; break; case '4of': $new_size='14pt'; break; case '5of': $new_size='18pt'; break; case '6of': $new_size='24pt'; break; case '7of': $new_size='36pt'; break; } $size='font-size: '.$new_size; } else { $size='font-size: '.$size; } } if ($face!='') $face='font-family: '.str_replace('\'','',$face).';'; $size_len=strlen($size); filter_html($as_admin,$source_member,0,$size_len,$size,false,false); $color_len=strlen($color); filter_html($as_admin,$source_member,0,$color_len,$color,false,false); $face_len=strlen($face); filter_html($as_admin,$source_member,0,$face_len,$face,false,false); $temp_tpl=do_template('COMCODE_FONT',array('_GUID'=>'f5fcafe737b8fdf466a6a51773e09c9b','CONTENT'=>$embed,'SIZE'=>$size,'COLOR'=>$color,'FACE'=>$face)); break; case 'box': $width=array_key_exists('width',$attributes)?comcode_to_tempcode($attributes['width'],$source_member,$as_admin,60,NULL,$connection,false,false,false,false,false,$highlight_bits,$on_behalf_of_member):make_string_tempcode('auto'); $type=array_key_exists('type',$attributes)?$attributes['type']:''; if ($type=='light' || $type=='med' || $type=='classic' || $type=='curved') $type='default'; // TODO: Remove, legacy $options=array_key_exists('options',$attributes)?$attributes['options']:''; $meta=($comcode_dangerous&&array_key_exists('meta',$attributes))?$attributes['meta']:''; //Insecure, unneeded here $links=($comcode_dangerous&&array_key_exists('links',$attributes))?$attributes['links']:''; //Insecure, unneeded here $converted_title=is_object($attributes['param'])?$attributes['param']:comcode_to_tempcode($attributes['param'],$source_member,$as_admin,60,NULL,$connection,false,false,false,false,false,$highlight_bits,$on_behalf_of_member); $temp_tpl=directive_tempcode('BOX',$embed,array($converted_title,make_string_tempcode($type),$width,make_string_tempcode($options),make_string_tempcode($meta),make_string_tempcode($links))); if (array_key_exists('float',$attributes)) $temp_tpl=do_template('FLOATER',array('_GUID'=>'54e8fc9ec1e16cfc5c8824e22f1e8745','FLOAT'=>$attributes['float'],'CONTENT'=>$temp_tpl)); break; case 'concept': if ((!array_key_exists('param',$attributes)) || ($attributes['param']=='')) { $text=$embed->evaluate(); $page=get_tutorial_link('concept__'.preg_replace('#[^\w_]#','_',$text)); if (!is_null($page)) { $zone=get_comcode_zone($page,false); } if ((is_null($page)) || (is_null($zone))) { $temp_tpl=make_string_tempcode($text); } else { $_url=build_url(array('page'=>$page),$zone); $_url->attach('#concept__'.preg_replace('#[^\w_]#','_',$text)); $temp_tpl=do_template('COMCODE_CONCEPT',array('_GUID'=>'ee0cd05f87329923f05145180004d8a8','TEXT'=>$text,'URL'=>$_url)); } } else { $temp_tpl=do_template('COMCODE_CONCEPT_INLINE',array('_GUID'=>'381a59de4d6f8967446c12bf4641a9ce','TEXT'=>$embed,'FULL'=>$attributes['param'])); } break; case 'concepts': $title=$embed->evaluate(); $concepts=new ocp_tempcode(); foreach ($attributes as $_key=>$_value) { if (substr($_key,-4)=='_key') { $key=$_value; $cid=substr($_key,0,strlen($_key)-4); $to_parse=array_key_exists($cid.'_value',$attributes)?$attributes[$cid.'_value']:new ocp_tempcode(); $value=is_object($to_parse)?$to_parse:comcode_to_tempcode($to_parse,$source_member,$as_admin,60,NULL,$connection,false,false,false,false,false,$highlight_bits,$on_behalf_of_member); $concepts->attach(do_template('COMCODE_CONCEPTS_CONCEPT',array('_GUID'=>'4baf6dabc32146c594c7fd922791b6b2','A'=>'concept__'.preg_replace('#[^\w]#','_',$_value),'KEY'=>$key,'VALUE'=>$value))); if ((get_param('type','')=='') && (get_param('id','',true)=='')) set_tutorial_link('concept__'.preg_replace('#[^\w]#','_',$key),get_page_name()); } } $temp_tpl=do_template('COMCODE_CONCEPTS',array('_GUID'=>'4c7a1d70753dc1d209b9951aa10f361a','TITLE'=>$title,'CONCEPTS'=>$concepts)); break; case 'exp_ref': $_embed=$embed->evaluate(); if (strpos($_embed,'.')!==false) break; $stub=get_file_base().'/data_custom/images/'.get_zone_name().'/'; $stub2=get_base_url().'/data_custom/images/'.get_zone_name().'/'; if (!file_exists($stub)) { $stub=get_file_base().'/data/images/'.get_zone_name().'/'; $stub2=get_base_url().'/data/images/'.get_zone_name().'/'; } if (!file_exists($stub)) { $stub=get_file_base().'/data_custom/images/'; $stub2=get_base_url().'/data_custom/images/'; } if (!file_exists($stub)) { $stub=get_file_base().'/data/images/'; $stub2=get_base_url().'/data/images/'; } if (substr($_embed,0,1)=='/') $_embed=substr($_embed,1); if (file_exists($stub.$_embed.'.png')) $url=$stub2.$_embed.'.png'; elseif (file_exists($stub.$_embed.'.gif')) $url=$stub2.$_embed.'.gif'; elseif (file_exists($stub.$_embed.'.jpg')) $url=$stub2.$_embed.'.jpg'; elseif (file_exists($stub.$_embed.'.jpeg')) $url=$stub2.$_embed.'.jpeg'; else { $stub=get_file_base().'/data_custom/images/docs/'; $stub2=get_base_url().'/data_custom/images/docs/'; if (substr($_embed,0,1)=='/') $_embed=substr($_embed,1); if (file_exists($stub.$_embed.'.png')) $url=$stub2.$_embed.'.png'; elseif (file_exists($stub.$_embed.'.gif')) $url=$stub2.$_embed.'.gif'; elseif (file_exists($stub.$_embed.'.jpg')) $url=$stub2.$_embed.'.jpg'; elseif (file_exists($stub.$_embed.'.jpeg')) $url=$stub2.$_embed.'.jpeg'; else { $temp_tpl=do_template('WARNING_BOX',array('WARNING'=>do_lang_tempcode('MISSING_RESOURCE_COMCODE','exp_ref',escape_html($_embed)))); if (array_key_exists('COMCODE_BROKEN_URLS',$GLOBALS)) { $GLOBALS['COMCODE_BROKEN_URLS'][]=array($_embed,NULL); } elseif ((!in_array(get_page_name(),$GLOBALS['DONT_CARE_MISSING_PAGES'])) && (!running_script('iframe'))) { require_code('failure'); relay_error_notification(do_lang('MISSING_RESOURCE_COMCODE','exp_ref',$_embed),false,$GLOBALS['FORUM_DRIVER']->is_staff($source_member)?'error_occurred_missing_reference_important':'error_occurred_missing_reference'); } break; } } $text=make_string_tempcode($attributes['param']); if ($text->is_empty()) $text=do_lang_tempcode('EXAMPLE'); $temp_tpl=do_template('COMCODE_EXP_REF',array('_GUID'=>'89e7f528e72096e3458d6acb70734d0b','TEXT'=>$text,'URL'=>$url)); break; case 'exp_thumb': $_embed=$embed->evaluate(); if (strpos($_embed,'.')!==false) break; $stub=get_file_base().'/data/images/'.get_zone_name().'/'; $stub2=get_base_url().'/data/images/'.get_zone_name().'/'; if (substr($_embed,0,1)=='/') $_embed=substr($_embed,1); if (file_exists($stub.$_embed.'.png')) $url_full=$stub2.$_embed.'.png'; elseif (file_exists($stub.$_embed.'.gif')) $url_full=$stub2.$_embed.'.gif'; elseif (file_exists($stub.$_embed.'.jpg')) $url_full=$stub2.$_embed.'.jpg'; elseif (file_exists($stub.$_embed.'.jpeg')) $url_full=$stub2.$_embed.'.jpeg'; else { $stub=get_file_base().'/data/images/docs/'; $stub2=get_base_url().'/data/images/docs/'; if (substr($_embed,0,1)=='/') $_embed=substr($_embed,1); if (file_exists($stub.$_embed.'.png')) $url_full=$stub2.$_embed.'.png'; elseif (file_exists($stub.$_embed.'.gif')) $url_full=$stub2.$_embed.'.gif'; elseif (file_exists($stub.$_embed.'.jpg')) $url_full=$stub2.$_embed.'.jpg'; elseif (file_exists($stub.$_embed.'.jpeg')) $url_full=$stub2.$_embed.'.jpeg'; else { $temp_tpl=do_template('WARNING_BOX',array('WARNING'=>do_lang_tempcode('MISSING_RESOURCE_COMCODE','exp_thumb',escape_html($_embed)))); if (array_key_exists('COMCODE_BROKEN_URLS',$GLOBALS)) { $GLOBALS['COMCODE_BROKEN_URLS'][]=$_embed; } elseif ((!in_array(get_page_name(),$GLOBALS['DONT_CARE_MISSING_PAGES'])) && (!running_script('iframe'))) { require_code('failure'); relay_error_notification(do_lang('MISSING_RESOURCE_COMCODE','exp_thumb',$_embed),false,$GLOBALS['FORUM_DRIVER']->is_staff($source_member)?'error_occurred_missing_reference_important':'error_occurred_missing_reference'); } break; } } $float=array_key_exists('float',$attributes)?$attributes['float']:'right'; $text=$attributes['param']; if ((get_option('is_on_gd')=='0') || (!function_exists('imagetypes'))) { $url_thumb=$url_full; } else { $new_name=$_embed.'_thumb.png'; $file_thumb=$stub.$new_name; if (file_exists($file_thumb)) { $url_thumb=$stub2.rawurlencode($new_name); } else { $new_name=$_embed.'.png'; $file_thumb=get_custom_file_base().'/uploads/auto_thumbs/'.$new_name; if (!file_exists($file_thumb)) { require_code('images'); convert_image($url_full,$file_thumb,-1,-1,150,false); } $url_thumb=get_custom_base_url().'/uploads/auto_thumbs/'.rawurlencode($new_name); } } $temp_tpl=do_template('COMCODE_EXP_THUMB',array('_GUID'=>'ce7f8a7fa29c2335f381a0beb3da9406','FLOAT'=>$float,'TEXT'=>$text,'URL_THUMB'=>$url_thumb,'URL_FULL'=>$url_full)); break; case 'thumb': $_embed=$embed->evaluate(); $_embed=remove_url_mistakes($_embed); $_embed=check_naughty_javascript_url($source_member,$_embed,$as_admin); if (substr($_embed,0,1)=='/') $_embed=substr($_embed,1); if (url_is_local($_embed)) { if ((file_exists(get_file_base().'/'.$_embed)) && (!file_exists(get_custom_file_base().'/'.$_embed))) { $url_full=get_base_url().'/'.$_embed; } else { $url_full=get_custom_base_url().'/'.$_embed; } } else { $url_full=$_embed; } $align=array_key_exists('align',$attributes)?$attributes['align']:'bottom'; if ((get_option('is_on_gd')=='0') || (!function_exists('imagetypes')) || ((!has_specific_permission($source_member,'draw_to_server')) && (!$as_admin))) { $url_thumb=$url_full; } else { if ($attributes['param']!='') { $url_thumb=url_is_local($attributes['param'])?get_custom_base_url().'/'.$attributes['param']:$attributes['param']; } if (($attributes['param']=='') || ((url_is_local($attributes['param'])) && (!file_exists(get_custom_file_base().'/'.rawurldecode($attributes['param']))))) { $new_name=url_to_filename($url_full); require_code('images'); if (!is_saveable_image($new_name)) $new_name.='.png'; if (is_null($new_name)) { $temp_tpl=do_template('WARNING_BOX',array('WARNING'=>do_lang_tempcode('URL_THUMB_TOO_LONG'))); break; } $file_thumb=get_custom_file_base().'/uploads/auto_thumbs/'.$new_name; if ((!file_exists($file_thumb)) && (strpos($file_thumb,'{$')===false)) { convert_image($url_full,$file_thumb,-1,-1,intval(get_option('thumb_width')),false); } $url_thumb=get_custom_base_url().'/uploads/auto_thumbs/'.rawurlencode($new_name); } } $caption=array_key_exists('caption',$attributes)?$attributes['caption']:''; $temp_tpl=do_template('COMCODE_THUMB',array('_GUID'=>'1b0d25f72ef5f816091269e29c586d60','CAPTION'=>$caption,'ALIGN'=>$align,'PASS_ID'=>(intval($pass_id)<0)?strval(mt_rand(0,10000)):$pass_id,'URL_THUMB'=>$url_thumb,'URL_FULL'=>$url_full)); if (array_key_exists('float',$attributes)) $temp_tpl=do_template('FLOATER',array('_GUID'=>'cbc56770714a44f56676f43da282cc7a','FLOAT'=>$attributes['float'],'CONTENT'=>$temp_tpl)); break; case 'img': if (($semiparse_mode) && (array_key_exists('rollover',$attributes))) { $temp_tpl=make_string_tempcode('[img'.reinsert_parameters($attributes).']'.$embed->evaluate().'[/img]'); break; } $_embed=$embed->evaluate(); $given_url=$_embed; $_embed=remove_url_mistakes($_embed); if (substr($_embed,0,1)=='/') $_embed=substr($_embed,1); $_embed=check_naughty_javascript_url($source_member,$_embed,$as_admin); if (url_is_local($_embed)) { if ((file_exists(get_file_base().'/'.$_embed)) && (!file_exists(get_custom_file_base().'/'.$_embed))) { $url_full=get_base_url().'/'.$_embed; } else { $url_full=get_custom_base_url().'/'.$_embed; } } else { $url_full=$_embed; } $temp_tpl=test_url($url_full,'img',@html_entity_decode($given_url,ENT_QUOTES,get_charset()),$source_member); $align=array_key_exists('align',$attributes)?$attributes['align']:''; $caption=is_object($attributes['param'])?$attributes['param']:comcode_to_tempcode($attributes['param'],$source_member,$as_admin,60,NULL,$connection,false,false,false,false,false,$highlight_bits,$on_behalf_of_member); if (array_key_exists('title',$attributes)) { $tooltip=is_object($attributes['title'])?$attributes['title']:comcode_to_tempcode($attributes['title'],$source_member,$as_admin,60,NULL,$connection,false,false,false,false,false,$highlight_bits,$on_behalf_of_member); } else { $tooltip=$caption; } $rollover=array_key_exists('rollover',$attributes)?$attributes['rollover']:NULL; if ((!is_null($rollover)) && (url_is_local($rollover))) { if ((file_exists(get_file_base().'/'.$rollover)) && (!file_exists(get_custom_file_base().'/'.$rollover))) { $rollover=get_base_url().'/'.$rollover; } else { $rollover=get_custom_base_url().'/'.$rollover; } } $refresh_time=array_key_exists('refresh_time',$attributes)?strval(intval($attributes['refresh_time'])):'0'; $temp_tpl->attach(do_template('COMCODE_IMG',array('_GUID'=>'70166d8dbb0aff064b99c0dd30ed77a8','REFRESH_TIME'=>$refresh_time,'ROLLOVER'=>$rollover,'ALIGN'=>$align,'URL'=>$url_full,'TOOLTIP'=>$tooltip,'CAPTION'=>$caption))); if (array_key_exists('float',$attributes)) $temp_tpl=do_template('FLOATER',array('_GUID'=>'918162250c80e10212efd9a051545b9b','FLOAT'=>$attributes['float'],'CONTENT'=>$temp_tpl)); break; case 'flash': $_embed=$embed->evaluate(); $given_url=$_embed; $_embed=remove_url_mistakes($_embed); if (substr($_embed,0,1)=='/') $_embed=substr($_embed,1); $_embed=check_naughty_javascript_url($source_member,$_embed,$as_admin); $url_full=url_is_local($_embed)?(get_custom_base_url().'/'.$_embed):$_embed; $temp_tpl=test_url($url_full,'flash',@html_entity_decode($given_url,ENT_QUOTES,get_charset()),$source_member); if (($attributes['param']=='') || (strpos($attributes['param'],'x')===false)) { if (!array_key_exists('width',$attributes)) $attributes['width']='300'; if (!array_key_exists('height',$attributes)) $attributes['height']='300'; $attributes['param']=$attributes['width'].'x'.$attributes['height']; } list($width,$height)=explode('x',$attributes['param'],2); if ((addon_installed('jwplayer')) && ((substr($url_full,-4)=='.flv') || (substr($url_full,-4)=='.mp4') || (substr($url_full,-4)=='.mp3') || (substr($url_full,-4)=='.webm'))) { $temp_tpl->attach(do_template('COMCODE_FLV',array('_GUID'=>'4746684d9e098709cc6671e1b00ce47e','URL'=>$url_full,'WIDTH'=>$width,'HEIGHT'=>$height))); } else { $temp_tpl->attach(do_template('COMCODE_SWF',array('_GUID'=>'8bc61ad75977a5a85eff96454af31fe8','URL'=>$url_full,'WIDTH'=>$width,'HEIGHT'=>$height))); } break; case 'url': // Make them both HTML strings $url=$embed->evaluate(); if (is_object($attributes['param'])) { $switch_over=true; // We know if must be Comcode XML $attributes['param']=$attributes['param']->evaluate(); } else { $switch_over=((!looks_like_url($url)) && (looks_like_url($attributes['param'],true))); if ((strpos($attributes['param'],'[')!==false) || (strpos($attributes['param'],'{')!==false)) // Extra Comcode parsing wanted? { $param_temp=comcode_to_tempcode(escape_html($attributes['param']),$source_member,$as_admin,60,NULL,$connection,false,false,true,false,false,$highlight_bits,$on_behalf_of_member); global $ADVERTISING_BANNERS; $temp_ab=$ADVERTISING_BANNERS; $ADVERTISING_BANNERS=array(); $caption=$param_temp; $ADVERTISING_BANNERS=$temp_ab; } else { $caption=make_string_tempcode(escape_html($attributes['param'])); // Consistency of escaping } } // Do we need to switch around? if ($switch_over) { $url=$attributes['param']; if ((strpos($url,'[')!==false) || (strpos($url,'{')!==false)) // Extra Comcode parsing wanted? { $url=static_evaluate_tempcode(comcode_to_tempcode($url,$source_member,$as_admin,60,NULL,$connection,false,false,true,false,false,$highlight_bits,$on_behalf_of_member)); } $caption=$embed; } // If we weren't given a caption, use the URL, but crop if necessary if ($caption->evaluate()=='') { $_caption=$url; // Shorten the URL if it is too long $max_link_length=50; if (strlen($_caption)>$max_link_length) { $_caption=escape_html(substr(@html_entity_decode($_caption,ENT_QUOTES,get_charset()),0,intval($max_link_length/2-3))).'…'.escape_html(substr(@html_entity_decode($_caption,ENT_QUOTES,get_charset()),intval(-$max_link_length/2))); } $caption=make_string_tempcode($_caption); } // Tidy up the URL now $url=@html_entity_decode($url,ENT_QUOTES,get_charset()); $url=fixup_protocolless_urls($url); // Integrity and security $url=check_naughty_javascript_url($source_member,$url,$as_admin); // More URL tidying $local=(url_is_local($url)) || (strpos($url,get_domain())!==false); $given_url=$url; if (($url!='') && ($url[0]!='#')) { /*if (substr($url,0,1)=='/') { $url_full=preg_replace('#/.*#','',get_base_url()).'/'.$url; } else*/ { if (substr($url,0,1)=='/') $url=substr($url,1); $url_full=url_is_local($url)?(get_base_url().'/'.$url):$url; if ($GLOBALS['XSS_DETECT']) ocp_mark_as_escaped($url_full); } } else $url_full=$url; $striped_base_url=str_replace('www.','',str_replace('http://','',get_base_url())); if (($striped_base_url!='') && (substr($url,0,1)!='%') && (strpos($url_full,$striped_base_url)===false)) // We don't want to hammer our own server when we have Comcode pages full of links to our own site (much less risk of hammering other people's servers, as we won't tend to have loads of links to them). Would also create bugs in emails sent out - e.g. auto-running validateip.php links hence voiding the intent of the feature. { $temp_tpl=test_url($url_full,'url',$given_url,$source_member); } // Render if (!array_key_exists('target',$attributes)) $attributes['target']=$local?'_top':'_blank'; if ($attributes['target']=='blank') $attributes['target']='_blank'; $rel=(($as_admin) || has_specific_permission($source_member,'search_engine_links'))?'':'nofollow'; if ($attributes['target']=='_blank') { $title=do_lang_tempcode('LINK_NEW_WINDOW'); } else $title=''; $temp_tpl->attach(do_template('COMCODE_URL',array('_GUID'=>'d1657530e6d3d57e6a4791fb3bfa0dd7','TITLE'=>$title,'REL'=>$rel,'TARGET'=>$attributes['target'],'URL'=>$url_full,'CAPTION'=>$caption))); break; case 'email': $_embed=$embed->evaluate(); require_code('type_validation'); require_code('obfuscate'); // If we need to switch if ((is_object($attributes['param'])) || ((!is_valid_email_address($_embed)) && (is_valid_email_address($attributes['param'])))) { $temp=$embed; // Is tempcode $_embed=$attributes['param']; $attributes['param']=$temp; } else { $attributes['param']=comcode_to_tempcode($attributes['param'],$source_member,$as_admin,60,NULL,$connection,false,false,false,false,false,$highlight_bits,$on_behalf_of_member); // Becomes tempcode } if ($attributes['param']->is_empty()) $attributes['param']=obfuscate_email_address($_embed); $subject=array_key_exists('subject',$attributes)?$attributes['subject']:''; $body=array_key_exists('body',$attributes)?$attributes['body']:''; $title=''; if (array_key_exists('title',$attributes)) $title=$attributes['title']; $temp_tpl=do_template('COMCODE_EMAIL',array('_GUID'=>'5f6ade8fe07701b6858575153d78f4e9','TITLE'=>$title,'ADDRESS'=>obfuscate_email_address($_embed),'SUBJECT'=>$subject,'BODY'=>$body,'CAPTION'=>$attributes['param'])); break; case 'reference': if ((array_key_exists('type',$attributes)) && ($attributes['type']=='url')) { $_embed=$embed->evaluate(); $_embed=check_naughty_javascript_url($source_member,$_embed,$as_admin); if (!array_key_exists('title',$attributes)) $attributes['title']=$attributes['param']; if ((is_object($attributes['title'])) || ($attributes['title']!='')) { $_title=is_object($attributes['title'])?$attributes['title']:comcode_to_tempcode($attributes['title'],$source_member,$as_admin,60,NULL,$connection,false,false,false,false,false,$highlight_bits,$on_behalf_of_member); $title=$_title->evaluate(); } else $title=$_embed; $embed=hyperlink($_embed,$title,true); } $temp_tpl=do_template('COMCODE_REFERENCE',array_merge($attributes,array('SOURCE'=>$embed))); break; case 'upload': // This points to a file path, not a URL $_embed=$embed->evaluate(); $type=(array_key_exists('type',$attributes))?$attributes['type']:'downloads'; if ((is_object($attributes['param'])) || ($attributes['param']!='')) { $_caption=is_object($attributes['param'])?$attributes['param']:comcode_to_tempcode($attributes['param'],$source_member,$as_admin,60,NULL,$connection,false,false,false,false,false,$highlight_bits,$on_behalf_of_member); $__caption=$_caption->evaluate(); } else $__caption=$_embed; $url=get_custom_base_url().'/'.$type.'/'.rawurlencode($_embed); $url=check_naughty_javascript_url($source_member,$url,$as_admin); $temp_tpl=test_url($url,'upload',$_embed,$source_member); $temp_tpl->attach(hyperlink($url,$__caption)); break; case 'page': $ignore_if_hidden=(array_key_exists('ignore_if_hidden',$attributes)) && ($attributes['ignore_if_hidden']=='1'); unset($attributes['ignore_if_hidden']); $hash=''; $caption=$embed; global $OVERRIDE_SELF_ZONE; list($zone,$attributes,$hash)=page_link_decode($attributes['param']); if (!array_key_exists('page',$attributes)) $attributes['page']=''; if (($zone=='_SELF') && (!is_null($OVERRIDE_SELF_ZONE))) $zone=$OVERRIDE_SELF_ZONE; unset($attributes['param']); foreach ($attributes as $key=>$val) { if (is_object($val)) $attributes[$key]=$val->evaluate(); } if ($zone=='_SEARCH') { $zone=get_page_zone($attributes['page'],false); if (is_null($zone)) $zone=''; } $pl_url=build_url($attributes,$zone,NULL,false,false,false,$hash); $temp_tpl=hyperlink($pl_url,$caption); $page=$attributes['page']; if ($page!='') { if ($zone=='_SELF') $zone=get_zone_name(); if ($zone=='_SEARCH') { $zone=get_page_zone($page,false); if (is_null($zone)) $zone=''; // Oh dear, well it will be correctly identified as not found anyway } $ptest=_request_page($page,$zone); if ($ptest!==false) { if (($page=='topicview') && (array_key_exists('id',$attributes))) { if (!is_numeric($attributes['id'])) $attributes['id']=$GLOBALS['SITE_DB']->query_value_null_ok('url_id_monikers','m_resource_id',array('m_resource_page'=>$page,'m_moniker'=>$attributes['id'])); if (!is_null($attributes['id'])) { $test=$GLOBALS['FORUM_DB']->query_value_null_ok('f_topics','id',array('id'=>$attributes['id'])); if (is_null($test)) $ptest=false; } else $ptest=false; } } if ($ptest===false) { //$temp_tpl->attach(' ['.do_lang('MISSING_RESOURCE').']'); // Don't want this as we might be making the page immediately if ((!in_array(get_page_name(),$GLOBALS['DONT_CARE_MISSING_PAGES'])) && (!running_script('iframe'))) { if ($ignore_if_hidden) { $temp_tpl=do_template('COMCODE_DEL',array('_GUID'=>'df638c61bc17ca975e95cf5f749836f5','CONTENT'=>$caption)); } else { require_code('failure'); relay_error_notification(do_lang('MISSING_RESOURCE_COMCODE','page',$zone.':'.$page),false,$GLOBALS['FORUM_DRIVER']->is_staff($source_member)?'error_occurred_missing_reference_important':'error_occurred_missing_reference'); } } } } break; case 'hide': if (array_key_exists('param',$attributes)) { $text=is_object($attributes['param'])?$attributes['param']:comcode_to_tempcode($attributes['param'],$source_member,$as_admin,60,NULL,$connection,false,false,false,false,false,$highlight_bits,$on_behalf_of_member); } else { $text=do_lang_tempcode('EXPAND'); } $temp_tpl=do_template('COMCODE_HIDE',array('_GUID'=>'a591a0d1e6bb3dde0f22cebb9c7ab93e','TEXT'=>$text,'CONTENT'=>$embed)); break; case 'quote': $cite=array_key_exists('cite',$attributes)?$attributes['cite']:NULL; if (!is_null($cite)) { $temp_tpl=test_url($cite,'del',$cite,$source_member); } if ($attributes['param']!='') { if (is_numeric($attributes['param'])) { $attributes['param']=$GLOBALS['FORUM_DRIVER']->get_username($attributes['param']); if (is_null($attributes['param'])) $attributes['param']=do_lang('UNKNOWN'); } else { $attributes['param']=protect_from_escaping(comcode_to_tempcode($attributes['param'],$source_member,$as_admin,60,NULL,$connection,false,false,false,false,false,$highlight_bits,$on_behalf_of_member)); } $temp_tpl->attach(do_template('COMCODE_QUOTE_BY',array('_GUID'=>'18f55a548892ad08b0b50b3b586b5b95','CITE'=>$cite,'CONTENT'=>$embed,'BY'=>$attributes['param'],'SAIDLESS'=>array_key_exists('saidless',$attributes)?$attributes['saidless']:'0'))); } else { $temp_tpl->attach(do_template('COMCODE_QUOTE',array('_GUID'=>'fa275de59433c17da19b22814c17fdc5','CITE'=>$cite,'CONTENT'=>$embed))); } break; case 'html': $temp_tpl=$embed; // Plain HTML. But it's been filtered already break; case 'semihtml': $temp_tpl=$embed; // Hybrid HTML. But it's been filtered already break; case 'block': $attributes['block']=trim($embed->evaluate()); if (preg_match('#^[\w\-]*$#',$attributes['block'])==0) { $temp_tpl=do_template('WARNING_BOX',array('WARNING'=>do_lang_tempcode('MISSING_BLOCK_FILE',escape_html($attributes['block'])))); break; // Avoids a suspected hack attempt by just filtering early } $_attributes=array(); foreach ($attributes as $key=>$val) { $_attributes[]=$key.'='.$val; } $temp_tpl=symbol_tempcode('BLOCK',$_attributes); break; case 'contents': // Do structure sweep $urls_for=array(); $old_structure_list=$STRUCTURE_LIST; $STRUCTURE_LIST=array(); // reset for e.g. comcode_text_to_tempcode calls (which don't itself reset it, although _comcode_to_tempcode does for top level parses) if ((array_key_exists('files',$attributes)) && ($comcode_dangerous)) { $s_zone=array_key_exists('zone',$attributes)?$attributes['zone']:get_zone_name(); $pages=find_all_pages($s_zone,'comcode_custom/'.get_site_default_lang(),'txt')+find_all_pages($s_zone,'comcode/'.get_site_default_lang(),'txt'); $prefix=$attributes['files']; foreach ($pages as $pg_name=>$pg_type) { if (substr($pg_name,0,strlen($prefix))==$prefix) { $i=count($STRUCTURE_LIST); comcode_to_tempcode(file_get_contents(zone_black_magic_filterer(get_file_base().'/'.$s_zone.'/pages/'.$pg_type.'/'.$pg_name.'.txt')),$source_member,$as_admin,60,NULL,$connection,false,false,false,true,false,NULL,$on_behalf_of_member); $page_url=build_url(array('page'=>$pg_name),$s_zone); while (array_key_exists($i,$STRUCTURE_LIST)) { $urls_for[]=$page_url; $i++; } } } $base=array_key_exists('base',$attributes)?intval($attributes['base']):1; } else { if (substr($comcode,0,8)=='<comcode') { require_code('comcode_xml'); if (!$as_admin) check_specific_permission('comcode_dangerous',NULL,$source_member); $_=new comcode_xml_to_tempcode($comcode,$source_member,60,NULL,$connection,false,false,false,true,false,$on_behalf_of_member); } else { require_code('comcode_text'); comcode_text_to_tempcode($comcode,$source_member,$as_admin,60,NULL,$connection,false,false,false,true,false,NULL,$on_behalf_of_member); } $base=array_key_exists('base',$attributes)?intval($attributes['base']):2; } $_embed=$embed->evaluate(); if (preg_match('#^test\d+$#',$_embed)!=0) // Little bit of inbuilt test code, for a particularly dangerously wrong tree { if ($_embed=='test1') { $test_data=array( 2, 3, 2, 3, 4, 4, 4, 2, 3, 2, 1, 1, ); } elseif ($_embed=='test2') { $test_data=array( 1, 2, 3, 1, 2, 3, ); } elseif ($_embed=='test3') { $test_data=array( 1, 4, 6, 1, 4, 6, ); } else { $test_data=array( 6, 4, 1, 6, 4, 1, ); } $STRUCTURE_LIST=array(); foreach ($test_data as $t) { $STRUCTURE_LIST[]=array($t,make_string_tempcode(strval($t)),uniqid('')); } $list_types=array(); } else { $list_types=($embed->evaluate()=='')?array():explode(',',$_embed); } $list_types+=array('decimal','lower-alpha','lower-roman','upper-alpha','upper-roman','disc'); $levels_allowed=array_key_exists('levels',$attributes)?intval($attributes['levels']):NULL; // Convert the list structure into a tree structure $past_level_stack=array(); $subtree_stack=array(array('','','',array())); // Children will be gathered into a 4th entry in the tuple by the end of the stack unravelling process -- our result $actual_past_stack_levels=0; foreach ($STRUCTURE_LIST as $i=>$struct) // Really complex stack of trees algorithm { $level=$struct[0]; $title=$struct[1]; $_title=$title->evaluate(); $uniq_id=$struct[2]; $url=array_key_exists($i,$urls_for)?$urls_for[$i]:''; if ((!is_null($levels_allowed)) && ($level>$levels_allowed)) continue; // Going back up the tree, destroying levels that must have now closed off while (($actual_past_stack_levels>0) && ($level<=$past_level_stack[$actual_past_stack_levels-1])) { array_pop($past_level_stack); // Value useless now, as the $actual_past_stack_levels is the true indicator and the $past_level_stack is just used as a navigation reference point for stack control $subtree=array_pop($subtree_stack); $actual_past_stack_levels--; // Alter the last of the next level on stack so it is actually taking the closed off level as children $subtree_stack[count($subtree_stack)-1][3][]=$subtree; } // Going down the tree array_push($past_level_stack,$level); array_push($subtree_stack,array($uniq_id,$_title,$url,array())); $actual_past_stack_levels++; } // Close off all levels still open while ($actual_past_stack_levels>0) // Pretty much the same as the while loop above { array_pop($past_level_stack); // Value useless now, as the $actual_past_stack_levels is the true indicator and the $past_level_stack is just used as a navigation reference point for stack control $subtree=array_pop($subtree_stack); $actual_past_stack_levels--; // Alter the last of the next level on stack so it is actually taking the closed off level as children $subtree_stack[count($subtree_stack)-1][3][]=$subtree; } // Now we have the structure to display $levels_t=_do_contents_level($subtree_stack[0][3],$list_types,$base-1); $temp_tpl=do_template('COMCODE_CONTENTS',array('_GUID'=>'ca2f5320fa930e2257a2e74e4f98e5a0','LEVELS'=>$levels_t)); $STRUCTURE_LIST=$old_structure_list; // Restore, so subsequent 'title' tags have correct numbering break; } // Last ditch effort: custom tags if ($temp_tpl->is_definitely_empty()) { global $REPLACE_TARGETS; if (array_key_exists($tag,$REPLACE_TARGETS)) { $replace=$REPLACE_TARGETS[$tag]['replace']; $parameters=explode(',',$REPLACE_TARGETS[$tag]['parameters']); $binding=array('CONTENT'=>$embed); foreach ($parameters as $parameter) { $parameter=trim($parameter); $parts=explode('=',$parameter); if (count($parts)==1) $parts[]=''; if (count($parts)!=2) continue; list($parameter,$default)=$parts; if ((!array_key_exists($parameter,$attributes)) || ($attributes[$parameter]=='')) $attributes[$parameter]=$default; $binding[strtoupper($parameter)]=$attributes[$parameter]; $replace=str_replace('{'.$parameter.'}','{'.strtoupper($parameter).'*}',$replace); } $replace=str_replace('{content}',array_key_exists($tag,$GLOBALS['TEXTUAL_TAGS'])?'{CONTENT}':'{CONTENT*}',$replace); require_code('tempcode_compiler'); $temp_tpl=template_to_tempcode($replace); $temp_tpl=$temp_tpl->bind($binding,'(custom comcode: '.$tag.')'); } } return $temp_tpl; } /** * Recursive algorithm to make table of contents. * * @param array The TOC (sub)tree * @param array The list types to use for each level * @param integer The level to start from * @param integer The level we are at in the recursion * @return tempcode The TOC node. */ function _do_contents_level($tree_structure,$list_types,$base,$the_level=0) { $lines=new ocp_tempcode(); foreach ($tree_structure as $level) { $_line=do_template('COMCODE_CONTENTS_LINE_FINAL',array('_GUID'=>'a3dd1bf2e16080993cf72edccb7f3608','ID'=>$level[0],'LINE'=>$level[1],'URL'=>$level[2])); if (array_key_exists(3,$level)) { $under=_do_contents_level($level[3],$list_types,$base,$the_level+1); if ($the_level+1==$base-1) return $under; // Top level not assembled because it has top level title, above contents $_line->attach($under); } $lines->attach(do_template('COMCODE_CONTENTS_LINE',array('_GUID'=>'f6891cb85d93facbc37f7fd3ef403950','LINE'=>$_line))); } return do_template('COMCODE_CONTENTS_LEVEL',array('_GUID'=>'cd2811bf69387ca05bf9612319db956b','TYPE'=>$list_types[max($the_level-$base,0)],'LINES'=>$lines)); } /** * Turn keys of a map to upper case, and return modified map. * * @param array Input map * @return array Adjusted map */ function map_keys_to_upper($array) { $out=array(); foreach ($array as $key=>$val) { $out[strtoupper($key)]=$val; } return $out; } |