. /** * Renderers for the mymobile theme * * @package theme_mymobile * @copyright John Stabinger * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ /** * A custom renderer for the mymobile theme to produce snippets of content. * * @package theme_mymobile * @copyright John Stabinger * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ class theme_mymobile_renderer extends plugin_renderer_base { /** * Produces the settings tree * * @param settings_navigation $navigation * @return string */ public function settings_tree(settings_navigation $navigation) { $content = $this->navigation_node($navigation, array('class' => 'settings')); if (has_capability('moodle/site:config', context_system::instance())) { // TODO: Work out whether something is missing from here. } return $content; } /** * Produces the navigation tree * * @param global_navigation $navigation * @return string */ public function navigation_tree(global_navigation $navigation) { return $this->navigation_node($navigation, array()); } /** * Protected method to render a navigaiton node * * @param navigation_node $node * @param array $attrs * @return type */ protected function navigation_node(navigation_node $node, $attrs = array()) { $items = $node->children; // exit if empty, we don't want an empty ul element if ($items->count() == 0) { return ''; } // array of nested li elements $lis = array(); foreach ($items as $item) { if (!$item->display) { continue; } $isbranch = ($item->children->count() > 0 || $item->nodetype == navigation_node::NODETYPE_BRANCH); $item->hideicon = true; $content = $this->output->render($item); $content .= $this->navigation_node($item); if ($isbranch && !(is_string($item->action) || empty($item->action))) { $content = html_writer::tag('li', $content, array('data-role' => 'list-divider', 'class' => (string)$item->key)); } else if($isbranch) { $content = html_writer::tag('li', $content, array('data-role' => 'list-divider')); } else { $content = html_writer::tag('li', $content, array('class' => (string)$item->text)); } $lis[] = $content; } if (!count($lis)) { return ''; } return implode("\n", $lis); } } /** * Overridden core renderer for the mymobile theme * * @package theme_mymobile * @copyright John Stabinger * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ class theme_mymobile_core_renderer extends core_renderer { /** * Returns the dtheme to use for the selected swatch * * @return string */ protected function theme_swatch() { $showswatch = 'light'; if (!empty($this->page->theme->settings->colourswatch)) { $showswatch = $this->page->theme->settings->colourswatch; } if ($showswatch == 'light') { $dtheme = 'b'; } else { $dtheme = 'd'; } return $dtheme; } /** * Produces a heading * * @param string $text * @param int $level * @param string $classes * @param string $id * @return string */ public function heading($text, $level = 2, $classes = 'main', $id = null) { if ($classes == 'helpheading') { // Keeps wrap from help headings in dialog. $content = parent::heading($text, $level, $classes, $id); } else { $content = html_writer::start_tag('div', array('class' => 'headingwrap ui-bar-'.$this->theme_swatch() .' ui-footer')); $content .= parent::heading($text, $level, $classes.' ui-title', $id); $content .= html_writer::end_tag('div'); } return $content; } /** * Renders a block * * @param block_contents $bc * @param string $region * @return string */ public function block(block_contents $bc, $region) { // Avoid messing up the object passed in. $bc = clone($bc); // The mymobile theme does not support collapsible blocks. $bc->collapsible = block_contents::NOT_HIDEABLE; // There are no controls that are usable within the $bc->controls = array(); // TODO: Do we still need to support accessibility here? Surely screen // readers don't present themselves as mobile devices too often. $skiptitle = strip_tags($bc->title); if (empty($skiptitle)) { $output = ''; $skipdest = ''; } else { $output = html_writer::tag('a', get_string('skipa', 'access', $skiptitle), array('href' => '#sb-' . $bc->skipid, 'class' => 'skip-block')); $skipdest = html_writer::tag('span', '', array('id' => 'sb-' . $bc->skipid, 'class' => 'skip-block-to')); } $testb = $bc->attributes['class']; $testc = $bc->attributes['id']; // TODO: Find a better solution to this hardcoded block checks. if ($testb == "block_calendar_month2 block") { $output = html_writer::start_tag('span'); } else if ($testb == "block_course_overview block") { $output = html_writer::start_tag('div'); } else { if (!empty($this->page->theme->settings->colourswatch)) { $showswatch = $this->page->theme->settings->colourswatch; } else { $showswatch = ''; } if ($showswatch == 'light') { $dtheme = 'd'; } else { $dtheme = 'c'; } if ($testc == "mod_quiz_navblock") { $collap = 'false'; } else { $collap = 'true'; } $output = html_writer::start_tag('div', array('data-role' => 'collapsible', 'data-collapsed' => $collap, 'data-content-theme' => $dtheme)); } $output .= html_writer::tag('h1', $this->block_header($bc)); $output .= html_writer::start_tag('div', $bc->attributes); $output .= $this->block_content($bc); $output .= html_writer::end_tag('div'); $output .= html_writer::end_tag('div'); $output .= $this->block_annotation($bc); $output .= $skipdest; return $output; } /** * Produces a blocks header * * @param block_contents $bc * @return string */ protected function block_header(block_contents $bc) { $title = ''; if (!$bc->title) { return ' '; } $output = html_writer::start_tag('div', array('class' => 'header')); $output .= html_writer::tag('div', html_writer::tag('div', '', array('class'=>'block_action')). $bc->title, array('class' => 'title')); $output .= html_writer::end_tag('div'); return $output; } /** * An evil function we don't want to execute * * @param block_contents $bc */ protected function init_block_hider_js(block_contents $bc) { // The mymobile theme in no shape or form supports the hiding of blocks // this function has been defined and left empty intentionally so that // the block hider JS is not even included. } /** * Produces the navigation bar for the mymobile theme * * @return string */ public function navbar() { $items = $this->page->navbar->get_items(); $htmlblocks = array(html_writer::tag('option', get_string('navigation'), array('data-placeholder' => 'true', 'value' => '-1'))); // Iterate the navarray and display each node $itemcount = count($items); $separator = ""; for ($i = 0; $i < $itemcount; $i++) { $item = $items[$i]; $item->hideicon = true; if ($i === 0) { $content = html_writer::tag('option', $this->render($item), array('value' => (string)$item->action)); } else if (!empty($item->action)) { $content = html_writer::tag('option', $this->render($item), array('value' => (string)$item->action)); } else { $content = ''; } $htmlblocks[] = $content; } $navbarcontent = html_writer::start_tag('form', array('id' => 'navselectform')); $navbarcontent .= html_writer::start_tag('select', array('id' => 'navselect', 'data-theme' => 'c', 'data-inline' => 'false', 'data-icon' => 'false')); $navbarcontent .= join('', $htmlblocks); $navbarcontent .= html_writer::end_tag('select'); $navbarcontent .= html_writer::end_tag('form'); // XHTML return $navbarcontent; } /** * Renders a navigation node * * This function has been overridden to remove tabindexs * * @param navigation_node $item * @return string */ protected function render_navigation_node(navigation_node $item) { // Generate the content normally $content = parent::render_navigation_node($item); // Strip out any tabindex's $content = str_replace(' tabindex="0"', '', $content); $content = str_replace(' tabindex=\'0\'', '', $content); // Return the cleaned content return $content; } /** * Displays login info * * @return string */ public function login_info($withlinks = null) { global $USER, $CFG, $DB, $SESSION; if (during_initial_install()) { return ''; } $course = $this->page->course; if (session_is_loggedinas()) { $realuser = session_get_realuser(); $fullname = fullname($realuser, true); $realuserinfo = " [wwwroot/course/loginas.php?id=$course->id&sesskey=".sesskey()."\">$fullname] "; } else { $realuserinfo = ''; } $loginurl = get_login_url(); if (empty($course->id)) { // $course->id is not defined during installation return ''; } else if (isloggedin()) { $context = context_course::instance($course->id); $fullname = fullname($USER, true); // Since Moodle 2.0 this link always goes to the public profile page (not the course profile page) // TODO: Test what happens when someone is using this via mnet [for this as well as login_info_footer] // TODO: Test what happens when you use the loginas feature [for this as well as login_info_footer] $username = ""; if (is_mnet_remote_user($USER) and $idprovider = $DB->get_record('mnet_host', array('id'=>$USER->mnethostid))) { $username .= " from wwwroot}\">{$idprovider->name}"; } if (isguestuser()) { $loggedinas = $realuserinfo.get_string('loggedinasguest')." (".get_string('login').')'; } else if (is_role_switched($course->id)) { // Has switched roles $rolename = ''; if ($role = $DB->get_record('role', array('id'=>$USER->access['rsw'][$context->path]))) { $rolename = ': '.format_string($role->name); } } else { $loggedinas = $realuserinfo.$username.' '.get_string('logout').''; } } else { $loggedinas = ''.get_string('login').''; } // TODO: Enable $CFG->displayloginfailures and test as admin what happens after you succesfully // log in after a failed log in attempt. [for this as well as login_info_footer] // This is probably totally unneeded if (isset($SESSION->justloggedin)) { unset($SESSION->justloggedin); if (!empty($CFG->displayloginfailures)) { if (!isguestuser()) { if ($count = count_login_failures($CFG->displayloginfailures, $USER->username, $USER->lastlogin)) { $loggedinas .= '
Error output, so disabling automatic redirect.
'; } $output .= $this->footer(); return $output; } /** * Renders a help icon * * @param help_icon $helpicon * @return string */ protected function render_help_icon(help_icon $helpicon) { global $CFG; // first get the help image icon $src = $this->pix_url('help'); $title = get_string($helpicon->identifier, $helpicon->component); if (empty($helpicon->linktext)) { $alt = $title; } else { $alt = get_string('helpwiththis'); } $attributes = array('src'=>$src, 'alt'=>$alt, 'class'=>'iconhelp', 'data-role'=>'button', 'data-inline'=>'true'); $output = html_writer::empty_tag('img', $attributes); // add the link text if given if (!empty($helpicon->linktext)) { // the spacing has to be done through CSS $output .= $helpicon->linktext; } // now create the link around it // TODO: Do we need to specify the theme in the help.php link? $url = new moodle_url('/help.php', array('component' => $helpicon->component, 'identifier' => $helpicon->identifier, 'lang'=>current_language(), 'theme'=>'mymobile')); // note: this title is displayed only if JS is disabled, otherwise the link will have the new ajax tooltip $title = get_string('helpprefix2', '', trim($title, ". \t")); $attributes = array('href'=>$url, 'title'=>$title); $id = html_writer::random_id('helpicon'); $attributes['id'] = $id; $attributes['rel'] = 'notexternal'; $attributes['data-rel'] = 'dialog'; $attributes['data-transition'] = 'flow'; $output = html_writer::tag('a', $output, $attributes); // and finally span return html_writer::tag('span', $output, array('class' => 'helplink2')); } /** * Renders a single button * * @param single_button $button * @return string */ protected function render_single_button(single_button $button) { $attributes = array( 'type' => 'submit', 'value' => $button->label, 'disabled' => $button->disabled ? 'disabled' : null, 'title' => $button->tooltip ); if ($button->actions) { $id = html_writer::random_id('single_button'); $attributes['id'] = $id; foreach ($button->actions as $action) { $this->add_action_handler($action, $id); } } // first the input element $output = html_writer::empty_tag('input', $attributes); // then hidden fields $params = $button->url->params(); if ($button->method === 'post') { $params['sesskey'] = sesskey(); } foreach ($params as $var => $val) { $output .= html_writer::empty_tag('input', array('type' => 'hidden', 'name' => $var, 'value' => $val)); } // then div wrapper for xhtml strictness $output = html_writer::tag('div', $output, array('rel' => $button->url->out_omit_querystring())); // TODO: Test a single_button that has an anchor and is set to use post // now the form itself around it $url = $button->url->out_omit_querystring(); // url without params if ($url === '') { $url = '#'; // there has to be always some action } // TODO: This is surely a bug that needs fixing.. all of a sudden we've switched // to the pages URL. // Test an single button with an external URL as its url // If the url has http, cool, if not we need to add it, JOHN $urlcheck = substr($url, 0, 4); if ($urlcheck != 'http') { $url = $this->page->url->out_omit_querystring(); } $attributes = array( 'method' => $button->method, 'action' => $url, 'id' => $button->formid ); $output = html_writer::tag('form', $output, $attributes); // and finally one more wrapper with class return html_writer::tag('div', $output, array('class' => $button->class)); } /** * Renders the header for the page * * @return string */ public function header() { global $USER, $CFG; if (session_is_loggedinas()) { $this->page->add_body_class('userloggedinas'); } // Give themes a chance to init/alter the page object. $this->page->theme->init_page($this->page); $this->page->set_state(moodle_page::STATE_PRINTING_HEADER); // Find the appropriate page layout file, based on $this->page->pagelayout. $layoutfile = $this->page->theme->layout_file($this->page->pagelayout); // Render the layout using the layout file. $rendered = $this->render_page_layout($layoutfile); // Slice the rendered output into header and footer. $cutpos = strpos($rendered, $this->unique_main_content_token); if ($cutpos === false) { $cutpos = strpos($rendered, self::MAIN_CONTENT_TOKEN); $token = self::MAIN_CONTENT_TOKEN; } else { $token = $this->unique_main_content_token; } if ($cutpos === false) { // TODO: Search for a better solution to this... check this is even needed? // The following code will lead to header containing nothing, and // footer containing all of the content for the template. // turned off error by john for ajax load of blocks without main content. // throw new coding_exception('page layout file ' . $layoutfile . // ' does not contain the string "' . self::MAIN_CONTENT_TOKEN . '".'); } $header = substr($rendered, 0, $cutpos); $footer = substr($rendered, $cutpos + strlen($token)); if (empty($this->contenttype)) { debugging('The page layout file did not call $OUTPUT->doctype()'); $header = $this->doctype() . $header; } send_headers($this->contenttype, $this->page->cacheable); $this->opencontainers->push('header/footer', $footer); $this->page->set_state(moodle_page::STATE_IN_BODY); return $header . $this->skip_link_target('maincontent'); } /** * Renders a notification * * @param string $message * @param string $classes * @return string */ public function notification($message, $classes = 'notifyproblem') { return html_writer::tag('div', clean_text($message), array('data-role'=>'none', 'data-icon'=>'alert', 'data-theme'=>'d', 'class' => renderer_base::prepare_classes($classes))); } /** * Renders the blocks for a block region in the page * * @param type $region * @return string */ public function blocks_for_region($region) { $blockcontents = $this->page->blocks->get_content_for_region($region, $this); $blocks = $this->page->blocks->get_blocks_for_region($region); $lastblock = null; $zones = array(); foreach ($blocks as $block) { $zones[] = $block->title; } $output = ''; foreach ($blockcontents as $bc) { if ($bc instanceof block_contents) { $lastblock = $bc->title; // We don't want to print navigation and settings blocks here. if ($bc->attributes['class'] != 'block_settings block' && $bc->attributes['class'] != 'block_navigation block') { $output .= $this->block($bc, $region); } } else if ($bc instanceof block_move_target) { $output .= $this->block_move_target($bc, $zones, $lastblock); } else { throw new coding_exception('Unexpected type of thing (' . get_class($bc) . ') found in list of block contents.'); } } return $output; } /** * Renders a single select instance * * @param single_select $select * @return string */ protected function render_single_select(single_select $select) { $select = clone($select); if (empty($select->formid)) { $select->formid = html_writer::random_id('single_select_f'); } $output = ''; $params = $select->url->params(); if ($select->method === 'post') { $params['sesskey'] = sesskey(); } foreach ($params as $name=>$value) { $output .= html_writer::empty_tag('input', array('type'=>'hidden', 'name'=>$name, 'value'=>$value)); } if (empty($select->attributes['id'])) { $select->attributes['id'] = html_writer::random_id('single_select'); //$select->attributes['data-native-menu'] = 'false'; //above by john for select elements to use native style and help performance? } if ($select->disabled) { $select->attributes['disabled'] = 'disabled'; } if ($select->tooltip) { $select->attributes['title'] = $select->tooltip; } $select->attributes['class'] = 'autosubmit'; if ($select->class) { $select->attributes['class'] .= ' ' . $select->class; } if ($select->label) { $output .= html_writer::label($select->label, $select->attributes['id']); } if ($select->helpicon instanceof help_icon) { $output .= $this->render($select->helpicon); } $output .= html_writer::select($select->options, $select->name, $select->selected, $select->nothing, $select->attributes); //by john show go button to fix selects $go = ''; $output .= html_writer::empty_tag('input data-inline="true"', array('type' => 'submit','value' => get_string('go'))); $output .= html_writer::tag('noscript', html_writer::tag('div', $go), array('style' => 'inline')); $nothing = empty($select->nothing) ? false : key($select->nothing); $this->page->requires->yui_module('moodle-core-formautosubmit', 'M.core.init_formautosubmit', array(array('selectid' => $select->attributes['id'], 'nothing' => $nothing)) ); // then div wrapper for xhtml strictness $output = html_writer::tag('div', $output); // now the form itself around it $formattributes = array( 'method' => $select->method, 'action' => $select->url->out_omit_querystring(), 'id' => $select->formid ); $output = html_writer::tag('form', $output, $formattributes); // and finally one more wrapper with class return html_writer::tag('div', $output, array('class' => $select->class)); } } /** * Overridden choice module renderer for the mymobile theme * * @package theme_mymobile * @copyright John Stabinger * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ $choice = get_plugin_directory('mod', 'choice'); if (file_exists($choice . '/renderer.php')) { require_once($CFG->dirroot . '/theme/mymobile/renderers/mod_choice_renderer.php'); }