| | |
| | | +-----------------------------------------------------------------------+ |
| | | | program/include/rcube_template.php | |
| | | | | |
| | | | This file is part of the RoundCube Webmail client | |
| | | | Copyright (C) 2006-2010, RoundCube Dev. - Switzerland | |
| | | | This file is part of the Roundcube Webmail client | |
| | | | Copyright (C) 2006-2010, Roundcube Dev. - Switzerland | |
| | | | Licensed under the GNU GPL | |
| | | | | |
| | | | PURPOSE: | |
| | |
| | | */ |
| | | class rcube_template extends rcube_html_page |
| | | { |
| | | var $app; |
| | | var $config; |
| | | var $framed = false; |
| | | var $pagetitle = ''; |
| | | var $message = null; |
| | | var $env = array(); |
| | | var $js_env = array(); |
| | | var $js_commands = array(); |
| | | var $object_handlers = array(); |
| | | private $app; |
| | | private $config; |
| | | private $pagetitle = ''; |
| | | private $message = null; |
| | | private $js_env = array(); |
| | | private $js_commands = array(); |
| | | private $object_handlers = array(); |
| | | private $plugin_skin_path; |
| | | |
| | | public $browser; |
| | | public $framed = false; |
| | | public $env = array(); |
| | | public $type = 'html'; |
| | | public $ajax_call = false; |
| | | |
| | | // deprecated names of templates used before 0.5 |
| | | private $deprecated_templates = array( |
| | | 'contact' => 'showcontact', |
| | | 'contactadd' => 'addcontact', |
| | | 'contactedit' => 'editcontact', |
| | | 'identityedit' => 'editidentity', |
| | | 'messageprint' => 'printmessage', |
| | | ); |
| | | |
| | | /** |
| | | * Constructor |
| | |
| | | $this->app = rcmail::get_instance(); |
| | | $this->config = $this->app->config->all(); |
| | | $this->browser = new rcube_browser(); |
| | | |
| | | |
| | | //$this->framed = $framed; |
| | | $this->set_env('task', $task); |
| | | $this->set_env('request_token', $this->app->get_request_token()); |
| | |
| | | else { |
| | | $title = ucfirst($this->env['task']); |
| | | } |
| | | |
| | | |
| | | return $title; |
| | | } |
| | | |
| | |
| | | */ |
| | | public function set_skin($skin) |
| | | { |
| | | if (!empty($skin) && is_dir('skins/'.$skin) && is_readable('skins/'.$skin)) |
| | | $valid = false; |
| | | |
| | | if (!empty($skin) && is_dir('skins/'.$skin) && is_readable('skins/'.$skin)) { |
| | | $skin_path = 'skins/'.$skin; |
| | | else |
| | | $valid = true; |
| | | } |
| | | else { |
| | | $skin_path = $this->config['skin_path'] ? $this->config['skin_path'] : 'skins/default'; |
| | | $valid = !$skin; |
| | | } |
| | | |
| | | $this->app->config->set('skin_path', $skin_path); |
| | | $this->config['skin_path'] = $skin_path; |
| | | |
| | | return $valid; |
| | | } |
| | | |
| | | /** |
| | |
| | | public function template_exists($name) |
| | | { |
| | | $filename = $this->config['skin_path'] . '/templates/' . $name . '.html'; |
| | | |
| | | return (is_file($filename) && is_readable($filename)); |
| | | return (is_file($filename) && is_readable($filename)) || ($this->deprecated_templates[$name] && $this->template_exists($this->deprecated_templates[$name])); |
| | | } |
| | | |
| | | /** |
| | |
| | | $args = func_get_args(); |
| | | if (count($args) == 1 && is_array($args[0])) |
| | | $args = $args[0]; |
| | | |
| | | |
| | | foreach ($args as $name) { |
| | | $this->command('add_label', $name, rcube_label($name)); |
| | | } |
| | |
| | | // set output asap |
| | | ob_flush(); |
| | | flush(); |
| | | |
| | | |
| | | if ($exit) { |
| | | exit; |
| | | } |
| | |
| | | public function write($template = '') |
| | | { |
| | | // unlock interface after iframe load |
| | | $unlock = preg_replace('/[^a-z0-9]/i', '', $_GET['_unlock']); |
| | | if ($this->framed) { |
| | | array_unshift($this->js_commands, array('set_busy', false)); |
| | | array_unshift($this->js_commands, array('set_busy', false, null, $unlock)); |
| | | } |
| | | else if ($unlock) { |
| | | array_unshift($this->js_commands, array('hide_message', $unlock)); |
| | | } |
| | | // write all env variables to client |
| | | $js = $this->framed ? "if(window.parent) {\n" : ''; |
| | | $js .= $this->get_js_commands() . ($this->framed ? ' }' : ''); |
| | | $this->add_script($js, 'head_top'); |
| | | |
| | | |
| | | // make sure all <form> tags have a valid request token |
| | | $template = preg_replace_callback('/<form\s+([^>]+)>/Ui', array($this, 'alter_form_tag'), $template); |
| | | $this->footer = preg_replace_callback('/<form\s+([^>]+)>/Ui', array($this, 'alter_form_tag'), $this->footer); |
| | |
| | | { |
| | | $skin_path = $this->config['skin_path']; |
| | | $plugin = false; |
| | | |
| | | $this->plugin_skin_path = null; |
| | | |
| | | $temp = explode(".", $name, 2); |
| | | if (count($temp) > 1) { |
| | | $plugin = $temp[0]; |
| | | $name = $temp[1]; |
| | | $skin_dir = $plugin . '/skins/' . $this->config['skin']; |
| | | $skin_path = $this->app->plugins->dir . $skin_dir; |
| | | $skin_path = $this->plugin_skin_path = $this->app->plugins->dir . $skin_dir; |
| | | if (!is_dir($skin_path)) { // fallback to default skin |
| | | $skin_dir = $plugin . '/skins/default'; |
| | | $skin_path = $this->app->plugins->dir . $skin_dir; |
| | | $skin_path = $this->plugin_skin_path = $this->app->plugins->dir . $skin_dir; |
| | | } |
| | | } |
| | | |
| | | |
| | | $path = "$skin_path/templates/$name.html"; |
| | | |
| | | if (!is_readable($path) && $this->deprecated_templates[$name]) { |
| | | $path = "$skin_path/templates/".$this->deprecated_templates[$name].".html"; |
| | | if (is_readable($path)) |
| | | raise_error(array('code' => 502, 'type' => 'php', |
| | | 'file' => __FILE__, 'line' => __LINE__, |
| | | 'message' => "Using deprecated template '".$this->deprecated_templates[$name]."' in ".$this->config['skin_path']."/templates. Please rename to '".$name."'"), |
| | | true, false); |
| | | } |
| | | |
| | | // read template file |
| | | if (($templ = @file_get_contents($path)) === false) { |
| | |
| | | ), true, true); |
| | | return false; |
| | | } |
| | | |
| | | |
| | | // replace all path references to plugins/... with the configured plugins dir |
| | | // and /this/ to the current plugin skin directory |
| | | if ($plugin) { |
| | |
| | | // parse for specialtags |
| | | $output = $this->parse_conditions($templ); |
| | | $output = $this->parse_xml($output); |
| | | |
| | | |
| | | // trigger generic hook where plugins can put additional content to the page |
| | | $hook = $this->app->plugins->exec_hook("render_page", array('template' => $name, 'content' => $output)); |
| | | |
| | |
| | | <textarea name="console" id="dbgconsole" rows="20" cols="40" wrap="off" style="display:none;width:400px;border:none;font-size:10px" spellcheck="false"></textarea></div>' |
| | | ); |
| | | } |
| | | |
| | | |
| | | $output = $this->parse_with_globals($hook['content']); |
| | | $this->write(trim($output)); |
| | | if ($exit) { |
| | |
| | | implode(',', $args) |
| | | ); |
| | | } |
| | | |
| | | |
| | | return $out; |
| | | } |
| | | |
| | |
| | | */ |
| | | public function abs_url($str) |
| | | { |
| | | return preg_replace('/^\//', $this->config['skin_path'].'/', $str); |
| | | if ($str[0] == '/') |
| | | return $this->config['skin_path'] . $str; |
| | | else |
| | | return $str; |
| | | } |
| | | |
| | | |
| | |
| | | { |
| | | return eval("return (".$this->parse_expression($condition).");"); |
| | | } |
| | | |
| | | |
| | | |
| | | |
| | | /** |
| | | * Inserts hidden field with CSRF-prevention-token into POST forms |
| | | */ |
| | |
| | | { |
| | | $out = $matches[0]; |
| | | $attrib = parse_attrib_string($matches[1]); |
| | | |
| | | |
| | | if (strtolower($attrib['method']) == 'post') { |
| | | $hidden = new html_hiddenfield(array('name' => '_token', 'value' => $this->app->get_request_token())); |
| | | $out .= "\n" . $hidden->show(); |
| | | } |
| | | |
| | | |
| | | return $out; |
| | | } |
| | | |
| | |
| | | |
| | | // include a file |
| | | case 'include': |
| | | $path = realpath($this->config['skin_path'].$attrib['file']); |
| | | if (!$this->plugin_skin_path || !is_file($path = realpath($this->plugin_skin_path . $attrib['file']))) |
| | | $path = realpath(($attrib['skin_path'] ? $attrib['skin_path'] : $this->config['skin_path']).$attrib['file']); |
| | | |
| | | if (is_readable($path)) { |
| | | if ($this->config['skin_include_php']) { |
| | | $incl = $this->include_php($path); |
| | |
| | | $hook = $this->app->plugins->exec_hook("template_plugin_include", $attrib); |
| | | return $hook['content']; |
| | | break; |
| | | |
| | | |
| | | // define a container block |
| | | case 'container': |
| | | if ($attrib['name'] && $attrib['id']) { |
| | |
| | | $content = call_user_func($handler, $attrib); |
| | | } |
| | | else if ($object == 'productname') { |
| | | $name = !empty($this->config['product_name']) ? $this->config['product_name'] : 'RoundCube Webmail'; |
| | | $name = !empty($this->config['product_name']) ? $this->config['product_name'] : 'Roundcube Webmail'; |
| | | $content = Q($name); |
| | | } |
| | | else if ($object == 'version') { |
| | |
| | | $title .= $this->get_pagetitle(); |
| | | $content = Q($title); |
| | | } |
| | | |
| | | |
| | | // exec plugin hooks for this template object |
| | | $hook = $this->app->plugins->exec_hook("template_object_$object", $attrib + array('content' => $content)); |
| | | return $hook['content']; |
| | |
| | | case 'exp': |
| | | $value = $this->parse_expression($attrib['expression']); |
| | | return eval("return Q($value);"); |
| | | |
| | | |
| | | // return variable |
| | | case 'var': |
| | | $var = explode(':', $attrib['name']); |
| | |
| | | */ |
| | | public function button($attrib) |
| | | { |
| | | static $sa_buttons = array(); |
| | | static $s_button_count = 100; |
| | | |
| | | // these commands can be called directly via url |
| | |
| | | else { |
| | | $attrib['type'] = ($attrib['image'] || $attrib['imagepas'] || $attrib['imageact']) ? 'image' : 'link'; |
| | | } |
| | | |
| | | $command = $attrib['command']; |
| | | |
| | | // take the button from the stack |
| | | if ($attrib['name'] && $sa_buttons[$attrib['name']]) { |
| | | $attrib = $sa_buttons[$attrib['name']]; |
| | | } |
| | | else if($attrib['image'] || $attrib['imageact'] || $attrib['imagepas'] || $attrib['class']) { |
| | | // add button to button stack |
| | | if (!$attrib['name']) { |
| | | $attrib['name'] = $command; |
| | | } |
| | | if (!$attrib['image']) { |
| | | $attrib['image'] = $attrib['imagepas'] ? $attrib['imagepas'] : $attrib['imageact']; |
| | | } |
| | | $sa_buttons[$attrib['name']] = $attrib; |
| | | } |
| | | else if ($command && $sa_buttons[$command]) { |
| | | // get saved button for this command/name |
| | | $attrib = $sa_buttons[$command]; |
| | | if ($attrib['task']) |
| | | $command = $attrib['task'] . '.' . $command; |
| | | |
| | | if (!$attrib['image']) { |
| | | $attrib['image'] = $attrib['imagepas'] ? $attrib['imagepas'] : $attrib['imageact']; |
| | | } |
| | | |
| | | if (!$attrib['id']) { |
| | |
| | | if (in_array($attrib['command'], rcmail::$main_tasks)) { |
| | | $attrib['href'] = rcmail_url(null, null, $attrib['command']); |
| | | } |
| | | else if ($attrib['task'] && in_array($attrib['task'], rcmail::$main_tasks)) { |
| | | $attrib['href'] = rcmail_url($attrib['command'], null, $attrib['task']); |
| | | } |
| | | else if (in_array($attrib['command'], $a_static_commands)) { |
| | | $attrib['href'] = rcmail_url($attrib['command']); |
| | | } |
| | |
| | | if (!$attrib['href']) { |
| | | $attrib['href'] = '#'; |
| | | } |
| | | if ($command && !$attrib['onclick']) { |
| | | if ($attrib['task']) { |
| | | if ($attrib['classact']) |
| | | $attrib['class'] = $attrib['classact']; |
| | | } |
| | | else if ($command && !$attrib['onclick']) { |
| | | $attrib['onclick'] = sprintf( |
| | | "return %s.command('%s','%s',this)", |
| | | JS_OBJECT_NAME, |
| | |
| | | $hiddenfield = new html_hiddenfield(array('name' => '_framed', 'value' => '1')); |
| | | $hidden = $hiddenfield->show(); |
| | | } |
| | | |
| | | |
| | | if (!$content) |
| | | $attrib['noclose'] = true; |
| | | |
| | | |
| | | return html::tag('form', |
| | | $attrib + array('action' => "./", 'method' => "get"), |
| | | $hidden . $content, |
| | | array('id','class','style','name','method','action','enctype','onsubmit')); |
| | | } |
| | | |
| | | |
| | | |
| | | |
| | | /** |
| | | * Build a form tag with a unique request token |
| | | * |
| | |
| | | if ($attrib['action']) { |
| | | $hidden->add(array('name' => '_action', 'value' => $attrib['action'])); |
| | | } |
| | | |
| | | |
| | | unset($attrib['task'], $attrib['request']); |
| | | $attrib['action'] = './'; |
| | | |
| | | |
| | | // we already have a <form> tag |
| | | if ($attrib['form']) |
| | | return $hidden->show() . $content; |
| | |
| | | return $username; |
| | | } |
| | | |
| | | // Current username is an e-mail address |
| | | if (strpos($_SESSION['username'], '@')) { |
| | | $username = $_SESSION['username']; |
| | | } |
| | | // get e-mail address from default identity |
| | | if ($sql_arr = $this->app->user->get_identity()) { |
| | | else if ($sql_arr = $this->app->user->get_identity()) { |
| | | $username = $sql_arr['email']; |
| | | } |
| | | else { |
| | | $username = $this->app->user->get_username(); |
| | | } |
| | | |
| | | return $username; |
| | | return idn_to_utf8($username); |
| | | } |
| | | |
| | | |
| | |
| | | private function login_form($attrib) |
| | | { |
| | | $default_host = $this->config['default_host']; |
| | | $autocomplete = (int) $this->config['login_autocomplete']; |
| | | |
| | | $_SESSION['temp'] = true; |
| | | |
| | | |
| | | // save original url |
| | | $url = get_input_value('_url', RCUBE_INPUT_POST); |
| | | if (empty($url) && !preg_match('/_(task|action)=logout/', $_SERVER['QUERY_STRING'])) |
| | | $url = $_SERVER['QUERY_STRING']; |
| | | |
| | | $input_user = new html_inputfield(array('name' => '_user', 'id' => 'rcmloginuser') + $attrib); |
| | | $input_pass = new html_passwordfield(array('name' => '_pass', 'id' => 'rcmloginpwd') + $attrib); |
| | | // set atocomplete attribute |
| | | $user_attrib = $autocomplete > 0 ? array() : array('autocomplete' => 'off'); |
| | | $host_attrib = $autocomplete > 0 ? array() : array('autocomplete' => 'off'); |
| | | $pass_attrib = $autocomplete > 1 ? array() : array('autocomplete' => 'off'); |
| | | |
| | | $input_action = new html_hiddenfield(array('name' => '_action', 'value' => 'login')); |
| | | $input_tzone = new html_hiddenfield(array('name' => '_timezone', 'id' => 'rcmlogintz', 'value' => '_default_')); |
| | | $input_url = new html_hiddenfield(array('name' => '_url', 'id' => 'rcmloginurl', 'value' => $url)); |
| | | $input_user = new html_inputfield(array('name' => '_user', 'id' => 'rcmloginuser') |
| | | + $attrib + $user_attrib); |
| | | $input_pass = new html_passwordfield(array('name' => '_pass', 'id' => 'rcmloginpwd') |
| | | + $attrib + $pass_attrib); |
| | | $input_host = null; |
| | | |
| | | if (is_array($default_host) && count($default_host) > 1) { |
| | |
| | | 'name' => '_host', 'id' => 'rcmloginhost', 'value' => $host) + $attrib); |
| | | } |
| | | else if (empty($default_host)) { |
| | | $input_host = new html_inputfield(array('name' => '_host', 'id' => 'rcmloginhost') + $attrib); |
| | | $input_host = new html_inputfield(array('name' => '_host', 'id' => 'rcmloginhost') |
| | | + $attrib + $host_attrib); |
| | | } |
| | | |
| | | $form_name = !empty($attrib['form']) ? $attrib['form'] : 'form'; |
| | |
| | | $table = new html_table(array('cols' => 2)); |
| | | |
| | | $table->add('title', html::label('rcmloginuser', Q(rcube_label('username')))); |
| | | $table->add(null, $input_user->show(get_input_value('_user', RCUBE_INPUT_POST))); |
| | | $table->add(null, $input_user->show(get_input_value('_user', RCUBE_INPUT_GPC))); |
| | | |
| | | $table->add('title', html::label('rcmloginpwd', Q(rcube_label('password')))); |
| | | $table->add(null, $input_pass->show()); |
| | |
| | | // add host selection row |
| | | if (is_object($input_host) && !$hide_host) { |
| | | $table->add('title', html::label('rcmloginhost', Q(rcube_label('server')))); |
| | | $table->add(null, $input_host->show(get_input_value('_host', RCUBE_INPUT_POST))); |
| | | $table->add(null, $input_host->show(get_input_value('_host', RCUBE_INPUT_GPC))); |
| | | } |
| | | |
| | | $out = $input_action->show(); |
| | | $out .= $input_tzone->show(); |
| | | $out .= $input_url->show(); |
| | | $out .= $table->show(); |
| | | |
| | | |
| | | if ($hide_host) { |
| | | $out .= $input_host->show(); |
| | | } |
| | |
| | | if ($attrib['type'] == 'search' && !$this->browser->khtml) { |
| | | unset($attrib['type'], $attrib['results']); |
| | | } |
| | | |
| | | |
| | | $input_q = new html_inputfield($attrib); |
| | | $out = $input_q->show(); |
| | | |