rewrite in 12/2006 * @author Pim Snel author of the idots template set * @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License * @package api * @subpackage framework * @access public * @version $Id$ */ /** * eGW idots template * * The idots_framework class draws the default idots template. It's a phplib template based template-set. * * Other phplib template based template-sets should extend (not copy!) this class and reimplement methods they which to change. */ class idots_framework extends egw_framework { /** * HTML of the sidebox menu, get's collected here by calls to $this->sidebox * * @var string */ var $sidebox_content = ''; /** * Instance of the phplib Template class for the API's template dir (EGW_TEMPLATE_DIR) * * @var Template */ var $tpl; /** * Instance of the Savant template class * * @var tplsavant2 */ var $tplsav2; /** * Contains array with linked icons in the topmenu * * @var mixed * @access public */ var $topmenu_icon_arr = array(); /** * Contains array of information for additional topmenu items added * by hooks */ private static $hook_items = array(); /** * Constructor * * @param string $template='idots' name of the template * @return idots_framework */ function __construct($template='idots') { parent::__construct($template); // call the constructor of the extended class // js stuff is not needed by login page and gives errors $GLOBALS['egw_info']['flags']['js_link_registry'] = $GLOBALS['egw_info']['flags']['currentapp'] !== 'login'; $this->tplsav2 = new tplsavant2(); $this->tplsav2->set_tpl_path(EGW_SERVER_ROOT.SEP.'phpgwapi'.SEP.'templates'.SEP.'idots'); } /** * @deprecated use __construct() */ function idots_framework($template='idots') { self::__construct($template); } /** * Returns the html-header incl. the opening body tag * * @param array $extra=array() extra attributes passed as data-attribute to egw.js * @return string with html */ function header(array $extra=array()) { // make sure header is output only once if (self::$header_done) return ''; self::$header_done = true; self::_send_headers(); // catch error echo'ed before the header, ob_start'ed in the header.inc.php $content = ob_get_contents(); ob_end_clean(); // the instanciation of the template has to be here and not in the constructor, // as the old Template class has problems if restored from the session (php-restore) if (!is_object($this->tpl)) ; $this->tpl = new Template(EGW_TEMPLATE_DIR,'keep'); $this->tpl->set_file(array('_head' => 'head.tpl')); $this->tpl->set_block('_head','head'); if (html::$ua_mobile) { self::$css_include_files[] = '/phpgwapi/templates/idots/mobile.css'; // hide location bar egw_framework::set_onload('window.setTimeout(function(){window.scrollTo(0, 1);}, 100);'); } // load idots specific javascript files, if we are not in login or logout if (!in_array($GLOBALS['egw_info']['flags']['currentapp'], array('login', 'logout'))) { // include regular include slidereffects.js if (!$GLOBALS['egw_info']['user']['preferences']['common']['disable_slider_effects']) { self::validate_file('/phpgwapi/templates/idots/js/slidereffects.js'); } else { self::validate_file('/phpgwapi/templates/idots/js/simple_show_hide.js'); } self::validate_file('/phpgwapi/templates/idots/js/idots.js'); } if ($GLOBALS['egw_info']['user']['preferences']['common']['click_or_onmouseover'] == 'onmouseover' && !html::$ua_mobile) { $show_menu_event = 'mouseover'; } else { $show_menu_event = 'click'; } $extra['slide-out'] = $this->slide_out_menus($show_menu_event); $this->tpl->set_var($this->_get_header($extra)); $content .= $this->tpl->fp('out','head'); $this->sidebox_content = ''; // need to be emptied here, as the object get's stored in the session return $content; } /** * Return slide-out-menu config for idots.js * * @param string $show_menu_event='click' * @return array */ protected function slide_out_menus($show_menu_event='click') { return array( array( 'id' => 'menu1', 'dir' => 'down', 'left' => 10, 'top' => html::$ua_mobile ? 0 : 114, 'width' => 180, 'height' => 200, 'pos' => 'right', 'bind' => array( '#extra_icons_show' => array('event' => $show_menu_event, 'method' => 'showMenu'), '#menu1close' => array('event' => $show_menu_event, 'method' => 'hide'), ), ), array( 'id' => 'menu2', 'dir' => 'right', 'left' => 0, 'top' => html::$ua_mobile ? 0 : 105, 'width' => 100, 'height' => 200, 'bind' => array( '#menu2show' => array('event' => $show_menu_event, 'method' => 'showMenu'), '#menu2close' => array('event' => 'click', 'method' => 'hide'), ), ), ); } /** * Returns the html from the body-tag til the main application area (incl. opening div tag) * * @return string with html */ function navbar() { if (self::$navbar_done) return ''; if (!empty($_GET['nonavbar']) || $GLOBALS['egw_info']['flags']['currentapp'] == 'admin' && empty($_GET['ajax'])) { if (!self::$header_done) return $this->header(); return ''; } self::$navbar_done = true; // the navbar if (!is_object($this->tpl)) $this->tpl = new Template(EGW_TEMPLATE_DIR,'keep'); $this->tpl->set_file(array('navbar' => 'navbar.tpl')); $this->tpl->set_block('navbar','extra_blocks_header','extra_block_header'); $this->tpl->set_block('navbar','extra_block_row','extra_block_row'); $this->tpl->set_block('navbar','extra_block_row_raw','extra_block_row_raw'); $this->tpl->set_block('navbar','extra_block_row_no_link','extra_block_row_no_link'); $this->tpl->set_block('navbar','extra_block_spacer','extra_block_spacer'); $this->tpl->set_block('navbar','extra_blocks_footer','extra_blocks_footer'); $this->tpl->set_block('navbar','sidebox_hide_header','sidebox_hide_header'); $this->tpl->set_block('navbar','sidebox_hide_footer','sidebox_hide_footer'); $this->tpl->set_block('navbar','appbox','appbox'); $this->tpl->set_block('navbar','navbar_footer','navbar_footer'); $this->tpl->set_block('navbar','upper_tab_block','upper_tabs'); $this->tpl->set_block('navbar','app_icon_block','app_icons'); $this->tpl->set_block('navbar','app_title_block','app_titles'); $this->tpl->set_block('navbar','app_extra_block','app_extra_icons'); $this->tpl->set_block('navbar','app_extra_icons_div'); $this->tpl->set_block('navbar','app_extra_icons_icon'); if (html::$ua_mobile) // replace whole navbar with just the extra apps icon { $this->tpl->set_block('navbar','navbar','mobil_not_needed'); $this->tpl->set_block('app_extra_icons_icon','extra_icons_show'); $this->tpl->set_var('mobil_not_needed',$this->tpl->get_var('extra_icons_show')); } $this->tpl->set_block('navbar','navbar_header','navbar_header'); $apps = $this->_get_navbar_apps(); $vars = $this->_get_navbar($apps); // add link registry to non-popup windows if (!isset($GLOBALS['egw_info']['flags']['js_link_registry'])) { self::validate_file('/phpgwapi/config.php'); self::validate_file('/phpgwapi/images.php',array('template' => $GLOBALS['egw_info']['user']['preferences']['common']['template_set'])); $content .= ''."\n"; } if($GLOBALS['egw_info']['user']['preferences']['common']['show_general_menu'] != 'sidebox' && !html::$ua_mobile) { $content .= $this->topmenu($vars,$apps); $vars['current_users'] = $vars['quick_add'] = $vars['user_info']=''; } $this->tpl->set_var($vars); $content .= $this->tpl->fp('out','navbar_header'); // general (app-unspecific) sidebox menu, instead of topmenu if($GLOBALS['egw_info']['user']['preferences']['common']['show_general_menu'] == 'sidebox') { $menu_title = lang('General Menu'); $this->topmenu($vars,$apps); $file = $this->tplsav2->menuitems; $this->sidebox('',$menu_title,$file); } // allow other apps to hook into sidebox menu of an app, hook-name: sidebox_$app $GLOBALS['egw']->hooks->process('sidebox_'.$GLOBALS['egw_info']['flags']['currentapp'], array($GLOBALS['egw_info']['flags']['currentapp']),true); // true = call independent of app-permissions // calling the old hook $GLOBALS['egw']->hooks->single('sidebox_menu',$GLOBALS['egw_info']['flags']['currentapp']); // allow other apps to hook into sidebox menu of every app: sidebox_all $GLOBALS['egw']->hooks->process('sidebox_all',array($GLOBALS['egw_info']['flags']['currentapp']),true); if($this->sidebox_content) { if($GLOBALS['egw_info']['user']['preferences']['common']['auto_hide_sidebox'] || html::$ua_mobile) { $this->tpl->set_var('lang_show_menu',lang('show menu')); $content .= $this->tpl->parse('out','sidebox_hide_header'); $content .= $this->sidebox_content; // content from calls to $this->sidebox $content .= $this->tpl->parse('out','sidebox_hide_footer'); $var['sideboxcolstart'] = ''; $this->tpl->set_var($var); $content .= $this->tpl->parse('out','appbox'); $var['remove_padding'] = 'style="padding-left:0px;"'; $var['sideboxcolend'] = ''; } else { $prefs = array(); if (isset($GLOBALS['egw_info']['user']['preferences'][$GLOBALS['egw_info']['flags']['currentapp']]['idotssideboxwidth'])) { $sideboxwidth = $GLOBALS['egw_info']['user']['preferences'][$GLOBALS['egw_info']['flags']['currentapp']]['idotssideboxwidth']; } if((int)$sideboxwidth < 1) { $sideboxwidth = 203; } $var['menu_link'] = ''; $var['sideboxcolstart'] = '
'; $var['sideboxcolstart'] .= '
'; $var['remove_padding'] = ''; $this->tpl->set_var($var); $content .= $this->tpl->parse('out','appbox'); $content .= $this->sidebox_content; $var['sideboxcolend'] = '
'; $this->tplsav2->assign('sideboxwidth', $sideboxwidth); } } else { $var['sideboxcolend']=''; } $this->tpl->set_var($var); $content .= $this->tpl->parse('out','navbar_footer'); // depricated (!) application header, if not disabled // ToDo: check if it can be removed if(!@$GLOBALS['egw_info']['flags']['noappheader'] && @isset($_GET['menuaction'])) { list($app,$class,$method) = explode('.',$_GET['menuaction']); if(is_array($GLOBALS[$class]->public_functions) && $GLOBALS[$class]->public_functions['header']) { ob_start(); $GLOBALS[$class]->header(); $content .= ob_get_contents(); ob_end_clean(); } } // hook after navbar $content .= $this->_get_after_navbar(); // make sure header is output (not explicitly calling header, allows to put validate calls eg. in sidebox) if (!self::$header_done) $content = $this->header() . $content; return $content; } /** * Return true if we are rendering the top-level EGroupware window * * A top-level EGroupware window has a navbar: eg. no popup and for a framed template (jdots) only frameset itself * * @return boolean $consider_navbar_not_yet_called_as_true=true * @return boolean */ public function isTop($consider_navbar_not_yet_called_as_true=true) { return self::$navbar_done || $consider_navbar_not_yet_called_as_true || isset($GLOBALS['egw_info']['flags']['nonavbar']) && !$GLOBALS['egw_info']['flags']['nonavbar']; } /** * Get navbar as array to eg. set as vars for a template (from idots' navbar.inc.php) * * Reimplemented so set the vars for the navbar itself (uses $this->tpl and the blocks a and b) * * @internal PHP5 protected * @param array $apps navbar apps from _get_navbar_apps * @return array */ function _get_navbar($apps) { $var = parent::_get_navbar($apps); if($GLOBALS['egw_info']['user']['userid'] == 'anonymous') { $config_reg = config::read('registration'); $this->tpl->set_var(array( 'url' => $GLOBALS['egw']->link('/logout.php'), 'title' => lang('Login'), )); $this->tpl->fp('upper_tabs','upper_tab_block'); if ($config_reg[enable_registration]=='True' && $config_reg[register_link]=='True') { $this->tpl->set_var(array( 'url' => $GLOBALS['egw']->link('/registration/index.php'), 'title' => lang('Register'), )); } } else { $this->tpl->set_var('upper_tabs',''); } if (html::$ua_mobile) { $max_icons = 0; $this->tpl->set_var('app_icons',''); } elseif (!($max_icons=$GLOBALS['egw_info']['user']['preferences']['common']['max_icons'])) { $max_icons = 30; } if($GLOBALS['egw_info']['user']['preferences']['common']['start_and_logout_icons'] == 'no' && !html::$ua_mobile) { $tdwidth = 100 / $max_icons; } else { $tdwidth = 100 / ($max_icons+1); // +1 for logout } $this->tpl->set_var('tdwidth',round($tdwidth)); // not shown in the navbar foreach($apps as $app => $app_data) { if ($app != 'preferences' && $app != 'about' && $app != 'logout' && $app != 'manual' && ($app != 'home' || $GLOBALS['egw_info']['user']['preferences']['common']['start_and_logout_icons'] != 'no') || html::$ua_mobile && in_array($app,array('preferences','logout','home'))) { $this->tpl->set_var($app_data); if($i < $max_icons) { $this->tpl->set_var($app_data); if($GLOBALS['egw_info']['user']['preferences']['common']['navbar_format'] != 'text') { $this->tpl->fp('app_icons','app_icon_block',true); } if($GLOBALS['egw_info']['user']['preferences']['common']['navbar_format'] != 'icons') { $this->tpl->fp('app_titles','app_title_block',true); } } else // generate extra icon layer shows icons and/or text { $this->tpl->fp('app_extra_icons','app_extra_block',true); } $i++; } } // settings for the extra icons dif if ($i <= $max_icons) // no extra icon div { $this->tpl->set_var('app_extra_icons_div',''); $this->tpl->set_var('app_extra_icons_icon',''); } else { $var['lang_close'] = lang('Close'); $var['lang_show_more_apps'] = lang('show_more_apps'); } if ($GLOBALS['egw_info']['user']['preferences']['common']['start_and_logout_icons'] != 'no' && $GLOBALS['egw_info']['user']['userid'] != 'anonymous') { $this->tpl->set_var($apps['logout']); if($GLOBALS['egw_info']['user']['preferences']['common']['navbar_format'] != 'text') { $this->tpl->fp('app_icons','app_icon_block',true); } if($GLOBALS['egw_info']['user']['preferences']['common']['navbar_format'] != 'icons') { $this->tpl->fp('app_titles','app_title_block',true); } } if($GLOBALS['egw_info']['user']['preferences']['common']['navbar_format'] == 'icons') { $var['app_titles'] = ' '; } return $var; } /** * Add menu items to the topmenu template class to be displayed * * @param array $app application data * @param mixed $alt_label string with alternative menu item label default value = null * @param string $urlextra string with alternate additional code inside -tag * @access protected * @return void */ function _add_topmenu_item(array $app_data,$alt_label=null) { $_item['link'] = $_item['url'] = htmlspecialchars($app_data['url']); $_item['target'] = $_item['urlextra'] = $app_data['target']; $_item['text'] = $_item['label'] = $alt_label ? $alt_label : $app_data['title']; $this->tplsav2->menuitems[] = $_item; $this->tplsav2->icon_or_star = common::image('phpgwapi','bullet'); } /** * Add info items to the topmenu template class to be displayed * * @param string $content html of item * @param string $id=null * @access protected * @return void */ function _add_topmenu_info_item($content, $id=null) { $this->tplsav2->menuinfoitems[] = $content; } /** * Display the string with html of the topmenu if its enabled * * @param array $vars * @param array $apps * @return string */ function topmenu(array $vars,array $apps) { $this->tplsav2->menuitems = array(); $this->tplsav2->menuinfoitems = array(); parent::topmenu($vars,$apps); $this->tplsav2->assign('info_icons',$this->topmenu_icon_arr); return $this->tplsav2->fetch('topmenu.tpl.php'); } /** * called by hooks to add an icon in the topmenu info location * * @param string $id unique element id * @param string $icon_src src of the icon image. Make sure this nog height then 18pixels * @param string $iconlink where the icon links to * @param booleon $blink set true to make the icon blink * @param mixed $tooltip string containing the tooltip html, or null of no tooltip * @access public * @return void */ function topmenu_info_icon($id,$icon_src,$iconlink,$blink=false,$tooltip=null) { $icon_arr['id'] = $id; $icon_arr['blink'] = $blink; $icon_arr['link'] = $iconlink; $icon_arr['image'] = $icon_src; if(!is_null($tooltip)) { $icon_arr['tooltip'] = html::tooltip($tooltip); } $this->topmenu_icon_arr[]=$icon_arr; } /** * Returns the html from the closing div of the main application area to the closing html-tag * * @return string html or null if no footer needed/wanted */ function footer() { static $footer_done; if ($footer_done++) return; // prevent multiple footers, not sure we still need this (RalfBecker) if (!isset($GLOBALS['egw_info']['flags']['nofooter']) || !$GLOBALS['egw_info']['flags']['nofooter']) { // get the (depricated) application footer $content = $this->_get_app_footer(); // run the hook navbar_end // ToDo: change to return the content ob_start(); $GLOBALS['egw']->hooks->process('navbar_end'); $content .= ob_get_contents(); ob_end_clean(); // eg. javascript, which need to be at the end of the page if ($GLOBALS['egw_info']['flags']['need_footer']) { $content .= $GLOBALS['egw_info']['flags']['need_footer']; } // do the template sets footer, former parse_navbar_end function // this closes the application area AND renders the closing body- and html-tag if (self::$navbar_done) { if (!is_a($this->tpl,'Template')) $this->tpl = new Template(EGW_TEMPLATE_DIR); $this->tpl->set_file(array('footer' => 'footer.tpl')); $this->tpl->set_var($this->_get_footer()); $content .= $this->tpl->fp('out','footer'); } elseif (!isset($GLOBALS['egw_info']['flags']['noheader']) || !$GLOBALS['egw_info']['flags']['noheader'] || self::$header_done || !empty($_GET['nonavbar']) || $GLOBALS['egw_info']['flags']['currentapp'] == 'admin' && empty($_GET['ajax'])) { $content .= "\n\n"; // close body and html tag, eg. for popups } return $content; } } /** * Parses one sidebox menu and add's the html to $this->sidebox_content for later use by $this->navbar * * @param string $appname * @param string $menu_title * @param array $file * @param string $type=null 'admin', 'preferences', 'favorites', ... */ function sidebox($appname,$menu_title,$file,$type=null) { if((!$appname || ($appname==$GLOBALS['egw_info']['flags']['currentapp'] && $file)) && is_object($this->tpl)) { // fix app admin menus to use admin.admin_ui.index loader if (($type == 'admin' || $menu_title == lang('Admin')) && $appname != 'admin') { $file = preg_replace("/^(.*)menuaction=([^&]+)(.*)$/", '$1menuaction=admin.admin_ui.index&load=$2$3&ajax=true', $file); } $this->tpl->set_var('lang_title',$menu_title); $this->sidebox_content .= $this->tpl->fp('out','extra_blocks_header'); foreach($file as $text => $url) { $this->sidebox_content .= $this->_sidebox_menu_item($url,$text); } $this->sidebox_content .= $this->tpl->parse('out','extra_blocks_footer'); } } /** * Return a sidebox menu item * * @internal PHP5 protected * @param string $item_link * @param string $item_text * @return string */ function _sidebox_menu_item($item_link='',$item_text='') { if($item_text === '_NewLine_' || $item_link === '_NewLine_') { return $this->tpl->parse('out','extra_block_spacer'); } if (strtolower($item_text) == 'grant access' && $GLOBALS['egw_info']['server']['deny_user_grants_access']) { return; } $var['icon_or_star']='ball'; $var['target'] = ''; if(is_array($item_link)) { if(isset($item_link['icon'])) { $app = isset($item_link['app']) ? $item_link['app'] : $GLOBALS['egw_info']['flags']['currentapp']; $var['icon_or_star'] = $item_link['icon'] ? '' : False; } $var['lang_item'] = isset($item_link['no_lang']) && $item_link['no_lang'] ? $item_link['text'] : lang($item_link['text']); $var['item_link'] = $item_link['link']; if ($item_link['target']) { if (strpos($item_link['target'], 'target=') !== false) { $var['target'] = $item_link['target']; } else { $var['target'] = ' target="' . $item_link['target'] . '"'; } } } else { $var['lang_item'] = lang($item_text); $var['item_link'] = $item_link; } $this->tpl->set_var($var); $block = 'extra_block_row'; if ($var['item_link'] === False) { $block .= $var['icon_or_star'] === False ? '_raw' : '_no_link'; } return $this->tpl->parse('out',$block); } /** * Return javascript (eg. for onClick) to open manual with given url * * @param string $url * @return string */ function open_manual_js($url) { return "egw_openWindowCentered2('$url','manual',800,600,'yes')"; } }