Search moodle.org's
Developer Documentation

See Release Notes

  • Bug fixes for general core bugs in 4.2.x will end 22 April 2024 (12 months).
  • Bug fixes for security issues in 4.2.x will end 7 October 2024 (18 months).
  • PHP version: minimum PHP 8.0.0 Note: minimum PHP version has increased since Moodle 4.1. PHP 8.1.x is supported too.
<?php
// This file is part of Moodle - http://moodle.org/
//
// Moodle is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// Moodle is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with Moodle.  If not, see <http://www.gnu.org/licenses/>.

/**
 * Contains the default section course format output class.
 *
 * @package   core_courseformat
 * @copyright 2020 Ferran Recio <ferran@moodle.com>
 * @license   http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
 */

namespace core_courseformat\output;

defined('MOODLE_INTERNAL') || die();

global $CFG;
require_once($CFG->dirroot . '/course/renderer.php');

use action_menu;
use action_menu_link_secondary;
use cm_info;
use coding_exception;
use context_course;
use core_course_renderer;
use core_courseformat\base as course_format;
use html_writer;
use moodle_page;
use moodle_url;
use pix_icon;
use renderable;
use section_info;
use templatable;
use url_select;

/**
 * Base class to render a course add section buttons.
 *
 * @package   core_courseformat
 * @copyright 2020 Ferran Recio <ferran@moodle.com>
 * @license   http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
 */
abstract class section_renderer extends core_course_renderer {

    /**
     * @var core_course_renderer contains an instance of core course renderer
     * @deprecated since 4.0 - use $this to access course renderer methods
     */
    protected $courserenderer;

    /**
     * Constructor method, calls the parent constructor.
     *
     * @deprecated since 4.0
     *
     * Note: this method exists only for compatibilitiy with legacy course formats. Legacy formats
     * depends on $this->courserenderer to access the course renderer methods. Since Moodle 4.0
     * core_courseformat\output\section_renderer extends core_course_renderer and all metdhos can be used directly from $this.
     *
     * @param moodle_page $page
     * @param string $target one of rendering target constants
     */
    public function __construct(moodle_page $page, $target) {
        parent::__construct($page, $target);
        $this->courserenderer = $this->page->get_renderer('core', 'course');
    }

    /**
     * Renders the provided widget and returns the HTML to display it.
     *
     * Course format templates uses a similar subfolder structure to the renderable classes.
     * This method find out the specific template for a course widget. That's the reason why
     * this render method is different from the normal plugin renderer one.
     *
     * course format templatables can be rendered using the core_course/local/* templates.
     * Format plugins are free to override the default template location using render_xxx methods as usual.
     *
     * @param renderable $widget instance with renderable interface
     * @return string the widget HTML
     */
    public function render(renderable $widget) {
        global $CFG;
        $fullpath = str_replace('\\', '/', get_class($widget));
        $classparts = explode('/', $fullpath);
        // Strip namespaces.
        $classname = array_pop($classparts);
        // Remove _renderable suffixes.
        $classname = preg_replace('/_renderable$/', '', $classname);

        $rendermethod = 'render_' . $classname;
        if (method_exists($this, $rendermethod)) {
            return $this->$rendermethod($widget);
        }

        // If nothing works, let the parent class decide.
        return parent::render($widget);
    }

    /**
     * Generate the section title, wraps it in a link to the section page if page is to be displayed on a separate page
     *
     * @param stdClass $section The course_section entry from DB
     * @param stdClass $course The course entry from DB
     * @return string HTML to output.
     */
    public function section_title($section, $course) {
        $title = get_section_name($course, $section);
        $url = course_get_url($course, $section->section, array('navigation' => true));
        if ($url) {
            $title = html_writer::link($url, $title);
        }
        return $title;
    }

    /**
     * Generate the section title to be displayed on the section page, without a link
     *
     * @param stdClass $section The course_section entry from DB
     * @param stdClass $course The course entry from DB
     * @return string HTML to output.
     */
    public function section_title_without_link($section, $course) {
        return get_section_name($course, $section);
    }

    /**
     * Get the updated rendered version of a cm list item.
     *
     * This method is used when an activity is duplicated or copied in on the client side without refreshing the page.
     * It replaces the course renderer course_section_cm_list_item method but it's scope is different.
     * Note that the previous method is used every time an activity is rendered, independent of it is the initial page
     * loading or an Ajax update. In this case, course_section_updated_cm_item will only be used when the course editor
     * requires to get an updated cm item HTML to perform partial page refresh. It will be used for suporting the course
     * editor webservices.
     *
     * By default, the template used for update a cm_item is the same as when it renders initially, but format plugins are
     * free to override this methos to provide extra affects or so.
     *
     * @param course_format $format the course format
     * @param section_info $section the section info
     * @param cm_info $cm the course module ionfo
     * @param array $displayoptions optional extra display options
     * @return string the rendered element
     */
    public function course_section_updated_cm_item(
        course_format $format,
        section_info $section,
        cm_info $cm,
        array $displayoptions = []
    ) {

        $cmitemclass = $format->get_output_classname('content\\section\\cmitem');
        $cmitem = new $cmitemclass($format, $section, $cm, $displayoptions);
        return $this->render($cmitem);
    }

    /**
     * Get the updated rendered version of a section.
     *
     * This method will only be used when the course editor requires to get an updated cm item HTML
     * to perform partial page refresh. It will be used for supporting the course editor webservices.
     *
     * By default, the template used for update a section is the same as when it renders initially,
     * but format plugins are free to override this method to provide extra effects or so.
     *
     * @param course_format $format the course format
     * @param section_info $section the section info
     * @return string the rendered element
     */
    public function course_section_updated(
        course_format $format,
        section_info $section
    ): string {
        $sectionclass = $format->get_output_classname('content\\section');
        $output = new $sectionclass($format, $section);
        return $this->render($output);
    }

    /**
     * Get the course index drawer with placeholder.
     *
     * The default course index is loaded after the page is ready. Format plugins can override
     * this method to provide an alternative course index.
     *
     * If the format is not compatible with the course index, this method will return an empty string.
     *
     * @param course_format $format the course format
     * @return String the course index HTML.
     */
    public function course_index_drawer(course_format $format): ?String {
        if ($format->uses_course_index()) {
            include_course_editor($format);
            return $this->render_from_template('core_courseformat/local/courseindex/drawer', []);
        }
        return '';
    }

    /**
> * Render the enable bulk editing button. * Generate the edit control action menu > * @param course_format $format the course format * > * @return string|null the enable bulk button HTML (or null if no bulk available). * @deprecated since 4.0 MDL-72656 - use core_course output components instead. > */ * > public function bulk_editing_button(course_format $format): ?string { * The section edit controls are now part of the main core_courseformat\output\local\content\section output > if (!$format->show_editor() || !$format->supports_components()) { * and does not use renderer methods anymore. > return null; * > } * @param array $controls The edit control items from section_edit_control_items > $widgetclass = $format->get_output_classname('content\\bulkedittoggler'); * @param stdClass $course The course entry from DB (not used) > $widget = new $widgetclass($format); * @param stdClass $section The course_section entry from DB > return $this->render($widget); * @return string HTML to output. > } */ > protected function section_edit_control_menu($controls, $course, $section) { > /**
debugging('section_edit_control_menu() can not be used anymore. Please use ' . 'core_courseformat\\output\\local\\content\\section to render a section. In case you need to modify those controls ' . 'override core_courseformat\\output\\local\\content\\section\\controlmenu in your format plugin.', DEBUG_DEVELOPER); $o = ""; if (!empty($controls)) { $menu = new action_menu(); $menu->set_menu_trigger(get_string('edit')); $menu->attributes['class'] .= ' section-actions'; foreach ($controls as $value) { $url = empty($value['url']) ? '' : $value['url']; $icon = empty($value['icon']) ? '' : $value['icon']; $name = empty($value['name']) ? '' : $value['name']; $attr = empty($value['attr']) ? array() : $value['attr']; $class = empty($value['pixattr']['class']) ? '' : $value['pixattr']['class']; $al = new action_menu_link_secondary( new moodle_url($url), new pix_icon($icon, '', null, array('class' => "smallicon " . $class)), $name, $attr ); $menu->add($al); } $o .= html_writer::div( $this->render($menu), 'section_action_menu', array('data-sectionid' => $section->id) ); } return $o; } /** * Generate the content to displayed on the right part of a section * before course modules are included * * @deprecated since 4.0 MDL-72656 - use core_course output components instead. * * Spatial references like "left" or "right" are limiting the way formats and themes can * extend courses. The elements from this method are now included in the * core_courseformat\output\local\content\section output components. * * @param stdClass $section The course_section entry from DB * @param stdClass $course The course entry from DB * @param bool $onsectionpage true if being printed on a section page * @return string HTML to output. */ protected function section_right_content($section, $course, $onsectionpage) { debugging('section_right_content() can not be used anymore. Please use ' . 'core_courseformat\\output\\local\\content\\section to render a section.', DEBUG_DEVELOPER); $o = $this->output->spacer(); $controls = $this->section_edit_control_items($course, $section, $onsectionpage); $o .= $this->section_edit_control_menu($controls, $course, $section); return $o; } /** * Generate the content to displayed on the left part of a section * before course modules are included * * @deprecated since 4.0 MDL-72656 - use core_course output components instead. * * Spatial references like "left" or "right" are limiting the way formats and themes can * extend courses. The elements from this method are now included in the * core_courseformat\output\local\content\section output components. * * @param stdClass $section The course_section entry from DB * @param stdClass $course The course entry from DB * @param bool $onsectionpage true if being printed on a section page * @return string HTML to output. */ protected function section_left_content($section, $course, $onsectionpage) { debugging('section_left_content() can not be used anymore. Please use ' . 'core_courseformat\\output\\local\\content\\section to render a section.', DEBUG_DEVELOPER); $o = ''; if ($section->section != 0) { // Only in the non-general sections. if (course_get_format($course)->is_section_current($section)) { $o = get_accesshide(get_string('currentsection', 'format_' . $course->format)); } } return $o; } /** * Generate the display of the header part of a section before * course modules are included * * @deprecated since 4.0 MDL-72656 - use core_course output components instead. * * This element is now a core_courseformat\output\content\section output component and it is displayed using * mustache templates instead of a renderer method. * * @param stdClass $section The course_section entry from DB * @param stdClass $course The course entry from DB * @param bool $onsectionpage true if being printed on a single-section page * @param int $sectionreturn The section to return to after an action * @return string HTML to output. */ protected function section_header($section, $course, $onsectionpage, $sectionreturn = null) { debugging('section_header() is deprecated. Please use ' . 'core_courseformat\\output\\local\\content\\section to render a section ' . 'or core_courseformat\output\\local\\content\\section\\header ' . 'to print only the header.', DEBUG_DEVELOPER); $o = ''; $sectionstyle = ''; if ($section->section != 0) { // Only in the non-general sections. if (!$section->visible) { $sectionstyle = ' hidden'; } if (course_get_format($course)->is_section_current($section)) { $sectionstyle = ' current'; } } $o .= html_writer::start_tag('li', [ 'id' => 'section-' . $section->section, 'class' => 'section main clearfix' . $sectionstyle, 'role' => 'region', 'aria-labelledby' => "sectionid-{$section->id}-title", 'data-sectionid' => $section->section, 'data-sectionreturnid' => $sectionreturn ]); $leftcontent = $this->section_left_content($section, $course, $onsectionpage); $o .= html_writer::tag('div', $leftcontent, array('class' => 'left side')); $rightcontent = $this->section_right_content($section, $course, $onsectionpage); $o .= html_writer::tag('div', $rightcontent, array('class' => 'right side')); $o .= html_writer::start_tag('div', array('class' => 'content')); // When not on a section page, we display the section titles except the general section if null. $hasnamenotsecpg = (!$onsectionpage && ($section->section != 0 || !is_null($section->name))); // When on a section page, we only display the general section title, if title is not the default one. $hasnamesecpg = ($onsectionpage && ($section->section == 0 && !is_null($section->name))); $classes = ' accesshide'; if ($hasnamenotsecpg || $hasnamesecpg) { $classes = ''; } $sectionname = html_writer::tag('span', $this->section_title($section, $course)); $o .= $this->output->heading($sectionname, 3, 'sectionname' . $classes, "sectionid-{$section->id}-title"); $o .= $this->section_availability($section); $o .= html_writer::start_tag('div', array('class' => 'summary')); if ($section->uservisible || $section->visible) { // Show summary if section is available or has availability restriction information. // Do not show summary if section is hidden but we still display it because of course setting // "Hidden sections are shown as not available". $o .= $this->format_summary_text($section); } $o .= html_writer::end_tag('div'); return $o; } /** * Generate the display of the footer part of a section. * * @deprecated since 4.0 MDL-72656 - use core_course output components instead. * * This element is integrated into core_courseformat\output\local\content\section output component and it is * displayed using mustache templates instead of a renderer method. * * @return string HTML to output. */ protected function section_footer() { debugging('section_footer() is deprecated. Please use ' . 'core_courseformat\\output\\local\\content\\section to render individual sections or .' . 'core_courseformat\\output\\local\\content to render the full course', DEBUG_DEVELOPER); $o = html_writer::end_tag('div'); $o .= html_writer::end_tag('li'); return $o; } /** * Generate the starting container html for a list of sections. * * @deprecated since 4.0 MDL-72656 - use core_course output components instead. * * @return string HTML to output. */ protected function start_section_list() { debugging('start_section_list() is deprecated. Please use ' . 'core_courseformat\\output\\local\\content\\section to render individual sections or .' . 'core_courseformat\\output\\local\\content to render the full course', DEBUG_DEVELOPER); return html_writer::start_tag('ul', ['class' => 'sections']); } /** * Generate the closing container html for a list of sections. * * @deprecated since 4.0 MDL-72656 - use core_course output components instead.y * * @return string HTML to output. */ protected function end_section_list() { debugging('end_section_list() is deprecated. Please use ' . 'core_courseformat\\output\\local\\content\\section to render individual sections or .' . 'core_courseformat\\output\\local\\content to render the full course', DEBUG_DEVELOPER); return html_writer::end_tag('ul'); } /** * Old method to print section edit controls. Do not use it! * * @deprecated since Moodle 3.0 MDL-48947 - Use core_courseformat\output\section_renderer::section_edit_control_items() instead */ protected function section_edit_controls() { throw new coding_exception('section_edit_controls() can not be used anymore. Please use ' . 'section_edit_control_items() instead.'); } /** * Generate the edit control items of a section * * @deprecated since 4.0 MDL-72656 - use core_course output components instead. * * This element is now a core_courseformat\output\content\section output component and it is displayed using * mustache templates instead of a renderer method. * * @param stdClass $course The course entry from DB * @param stdClass $section The course_section entry from DB * @param bool $onsectionpage true if being printed on a section page * @return array of edit control items */ protected function section_edit_control_items($course, $section, $onsectionpage = false) { debugging('section_edit_control_items() is deprecated, please use or extend' . 'core_courseformat\output\\local\\content\\section\\controlmenu instead (like topics format does).', DEBUG_DEVELOPER); $format = course_get_format($course); $modinfo = $format->get_modinfo(); if ($onsectionpage) { $format->set_section_number($section->section); } // We need a section_info object, not a record. $section = $modinfo->get_section_info($section->section); $widgetclass = $format->get_output_classname('content\\section\\controlmenu'); $widget = new $widgetclass($format, $section); return $widget->section_control_items(); } /** * Generate a summary of a section for display on the 'course index page' * * @deprecated since 4.0 MDL-72656 - use core_course output components instead. * * This element is now a core_courseformat\output\content\section output component and it is displayed using * mustache templates instead of a renderer method. * * @param stdClass $section The course_section entry from DB * @param stdClass $course The course entry from DB * @param array $mods (argument not used) * @return string HTML to output. */ protected function section_summary($section, $course, $mods) { debugging('section_summary() is deprecated. Please use ' . 'core_courseformat\output\\local\\content\\section to render sections. If you need to modify those summary, extend ' . 'core_courseformat\output\\local\\content\\section\\summary in your format plugin.', DEBUG_DEVELOPER); $classattr = 'section main section-summary clearfix'; $linkclasses = ''; // If section is hidden then display grey section link. if (!$section->visible) { $classattr .= ' hidden'; $linkclasses .= ' dimmed_text'; } else if (course_get_format($course)->is_section_current($section)) { $classattr .= ' current'; } $title = get_section_name($course, $section); $o = ''; $o .= html_writer::start_tag('li', [ 'id' => 'section-' . $section->section, 'class' => $classattr, 'role' => 'region', 'aria-label' => $title, 'data-sectionid' => $section->section ]); $o .= html_writer::tag('div', '', array('class' => 'left side')); $o .= html_writer::tag('div', '', array('class' => 'right side')); $o .= html_writer::start_tag('div', array('class' => 'content')); if ($section->uservisible) { $title = html_writer::tag( 'a', $title, array('href' => course_get_url($course, $section->section), 'class' => $linkclasses) ); } $o .= $this->output->heading($title, 3, 'section-title'); $o .= $this->section_availability($section); $o .= html_writer::start_tag('div', array('class' => 'summarytext')); if ($section->uservisible || $section->visible) { // Show summary if section is available or has availability restriction information. // Do not show summary if section is hidden but we still display it because of course setting // "Hidden sections are shown as not available". $o .= $this->format_summary_text($section); } $o .= html_writer::end_tag('div'); $o .= $this->section_activity_summary($section, $course, null); $o .= html_writer::end_tag('div'); $o .= html_writer::end_tag('li'); return $o; } /** * Generate a summary of the activites in a section * * @deprecated since 4.0 MDL-72656 - use core_course output components instead. * * This element is now a core_courseformat\output\content\section output component and it is displayed using * mustache templates instead of a renderer method. * * @param stdClass $section The course_section entry from DB * @param stdClass $course the course record from DB * @param array $mods (argument not used) * @return string HTML to output. */ protected function section_activity_summary($section, $course, $mods) { debugging('section_activity_summary() is deprecated. Please use ' . 'core_courseformat\output\\local\\content\\section to render sections. ' . 'If you need to modify those information, extend ' . 'core_courseformat\output\\local\\content\\section\\cmsummary in your format plugin.', DEBUG_DEVELOPER); $format = course_get_format($course); $widgetclass = $format->get_output_classname('content\\section\\cmsummary'); $widget = new $widgetclass($format, $section); $this->render($widget); } /** * If section is not visible, display the message about that ('Not available * until...', that sort of thing). Otherwise, returns blank. * * For users with the ability to view hidden sections, it shows the * information even though you can view the section and also may include * slightly fuller information (so that teachers can tell when sections * are going to be unavailable etc). This logic is the same as for * activities. * * @deprecated since 4.0 MDL-72656 - use core_course output components instead. * * This element is now a core_courseformat\output\content\section output component and it is displayed using * mustache templates instead of a renderer method. * * @param section_info $section The course_section entry from DB * @param bool $canviewhidden True if user can view hidden sections * @return string HTML to output */ protected function section_availability_message($section, $canviewhidden) { global $CFG; debugging('section_availability_message() is deprecated. Please use ' . 'core_courseformat\output\\local\\content\\section to render sections. If you need to modify this element, extend ' . 'core_courseformat\output\\local\\content\\section\\availability in your format plugin.', DEBUG_DEVELOPER); $course = $section->course; $format = course_get_format($course); $widgetclass = $format->get_output_classname('content\\section\\availability'); $widget = new $widgetclass($format, $section); $this->render($widget); } /** * Displays availability information for the section (hidden, not available unles, etc.) * * @deprecated since 4.0 MDL-72656 - use core_course output components instead. * * This element is now a core_courseformat\output\content\section output component and it is displayed using * mustache templates instead of a renderer method. * * @param section_info $section * @return string */ public function section_availability($section) { debugging('section_availability() is deprecated. Please use ' . 'core_courseformat\output\\local\\content\\section to render sections. If you need to modify this element, extend ' . 'core_courseformat\output\\local\\content\\section\\availability in your format plugin.', DEBUG_DEVELOPER); $context = context_course::instance($section->course); $canviewhidden = has_capability('moodle/course:viewhiddensections', $context); return html_writer::div($this->section_availability_message($section, $canviewhidden), 'section_availability'); } /** * Show if something is on on the course clipboard (moving around) * * @deprecated since 4.0 MDL-72656 - use core_course output components instead. * * While the non ajax course eidtion is still supported, the old clipboard will be * emulated by core_courseformat\output\local\content\section\cmlist. * * @param stdClass $course The course entry from DB * @param int $sectionno The section number in the course which is being displayed * @return string HTML to output. */ protected function course_activity_clipboard($course, $sectionno = null) { global $USER; debugging('Non ajax course edition using course_activity_clipboard is not supported anymore.', DEBUG_DEVELOPER); $o = ''; // If currently moving a file then show the current clipboard. if (ismoving($course->id)) { $url = new moodle_url( '/course/mod.php', array( 'sesskey' => sesskey(), 'cancelcopy' => true, 'sr' => $sectionno, ) ); $o .= html_writer::start_tag('div', array('class' => 'clipboard')); $o .= strip_tags(get_string('activityclipboard', '', $USER->activitycopyname)); $o .= ' (' . html_writer::link($url, get_string('cancel')) . ')'; $o .= html_writer::end_tag('div'); } return $o; } /** * Generate next/previous section links for naviation. * * @deprecated since 4.0 MDL-72656 - use core_course output components instead. * * This element is now a core_courseformat\output\content\section output component and it is displayed using * mustache templates instead of a renderer method. * * @param stdClass $course The course entry from DB * @param array $sections The course_sections entries from the DB * @param int $sectionno The section number in the course which is being displayed * @return array associative array with previous and next section link */ protected function get_nav_links($course, $sections, $sectionno) { debugging('get_nav_links() is deprecated. Please use ' . 'core_courseformat\\output\\local\\content to render a course. If you need to modify this element, extend ' . 'core_courseformat\\output\\local\\content\\sectionnavigation in your format plugin.', DEBUG_DEVELOPER); // FIXME: This is really evil and should by using the navigation API. $course = course_get_format($course)->get_course(); $canviewhidden = has_capability('moodle/course:viewhiddensections', context_course::instance($course->id)) or !$course->hiddensections; $links = array('previous' => '', 'next' => ''); $back = $sectionno - 1; while ($back > 0 and empty($links['previous'])) { if ($canviewhidden || $sections[$back]->uservisible) { $params = array(); if (!$sections[$back]->visible) { $params = array('class' => 'dimmed_text'); } $previouslink = html_writer::tag('span', $this->output->larrow(), array('class' => 'larrow')); $previouslink .= get_section_name($course, $sections[$back]); $links['previous'] = html_writer::link(course_get_url($course, $back), $previouslink, $params); } $back--; } $forward = $sectionno + 1; $numsections = course_get_format($course)->get_last_section_number(); while ($forward <= $numsections and empty($links['next'])) { if ($canviewhidden || $sections[$forward]->uservisible) { $params = array(); if (!$sections[$forward]->visible) { $params = array('class' => 'dimmed_text'); } $nextlink = get_section_name($course, $sections[$forward]); $nextlink .= html_writer::tag('span', $this->output->rarrow(), array('class' => 'rarrow')); $links['next'] = html_writer::link(course_get_url($course, $forward), $nextlink, $params); } $forward++; } return $links; } /** * Generate the header html of a stealth section * * @deprecated since 4.0 MDL-72656 - use core_course output components instead. * * This element is now a core_courseformat\output\content\section output component and it is displayed using * mustache templates instead of a renderer method. * * @param int $sectionno The section number in the course which is being displayed * @return string HTML to output. */ protected function stealth_section_header($sectionno) { debugging('stealth_section_header() is deprecated. Please use ' . 'core_courseformat\output\\local\\content\\section to render sections.', DEBUG_DEVELOPER); $o = ''; $o .= html_writer::start_tag('li', [ 'id' => 'section-' . $sectionno, 'class' => 'section main clearfix orphaned hidden', 'data-sectionid' => $sectionno ]); $o .= html_writer::tag('div', '', array('class' => 'left side')); $course = course_get_format($this->page->course)->get_course(); $section = course_get_format($this->page->course)->get_section($sectionno); $rightcontent = $this->section_right_content($section, $course, false); $o .= html_writer::tag('div', $rightcontent, array('class' => 'right side')); $o .= html_writer::start_tag('div', array('class' => 'content')); $o .= $this->output->heading( get_string('orphanedactivitiesinsectionno', '', $sectionno), 3, 'sectionname' ); return $o; } /** * Generate footer html of a stealth section * * @deprecated since 4.0 MDL-72656 - use core_course output components instead. * * This element is now a core_courseformat\output\content\section output component and it is displayed using * mustache templates instead of a renderer method. * * @return string HTML to output. */ protected function stealth_section_footer() { debugging('stealth_section_footer() is deprecated. Please use ' . 'core_courseformat\output\\local\\content\\section to render sections.', DEBUG_DEVELOPER); $o = html_writer::end_tag('div'); $o .= html_writer::end_tag('li'); return $o; } /** * Generate the html for a hidden section * * @param int $sectionno The section number in the course which is being displayed * @param int|stdClass $courseorid The course to get the section name for (object or just course id) * @return string HTML to output. */ protected function section_hidden($sectionno, $courseorid = null) { if ($courseorid) { $sectionname = get_section_name($courseorid, $sectionno); $strnotavailable = get_string('notavailablecourse', '', $sectionname); } else { $strnotavailable = get_string('notavailable'); } $o = ''; $o .= html_writer::start_tag('li', [ 'id' => 'section-' . $sectionno, 'class' => 'section main clearfix hidden', 'data-sectionid' => $sectionno ]); $o .= html_writer::tag('div', '', array('class' => 'left side')); $o .= html_writer::tag('div', '', array('class' => 'right side')); $o .= html_writer::start_tag('div', array('class' => 'content')); $o .= html_writer::tag('div', $strnotavailable); $o .= html_writer::end_tag('div'); $o .= html_writer::end_tag('li'); return $o; } /** * Generate the html for the 'Jump to' menu on a single section page. * * @deprecated since 4.0 MDL-72656 - use core_course output components instead. * * This element is now a core_courseformat\output\content\section output component and it is displayed using * mustache templates instead of a renderer method. * * @param stdClass $course The course entry from DB * @param array $sections The course_sections entries from the DB * @param int $displaysection the current displayed section number. * * @return string HTML to output. */ protected function section_nav_selection($course, $sections, $displaysection) { debugging('section_nav_selection() can not be used anymore. Please use ' . 'core_courseformat\\output\\local\\content to render a course. If you need to modify this element, extend ' . 'core_courseformat\\output\\local\\content\\sectionnavigation or ' . 'core_courseformat\\output\\local\\content\\sectionselector in your format plugin.', DEBUG_DEVELOPER); $o = ''; $sectionmenu = array(); $sectionmenu[course_get_url($course)->out(false)] = get_string('maincoursepage'); $modinfo = get_fast_modinfo($course); $section = 1; $numsections = course_get_format($course)->get_last_section_number(); while ($section <= $numsections) { $thissection = $modinfo->get_section_info($section); if (($thissection->uservisible) && ($section != $displaysection) && ($url = course_get_url($course, $section))) { $sectionmenu[$url->out(false)] = get_section_name($course, $section); } $section++; } $select = new url_select($sectionmenu, '', array('' => get_string('jumpto'))); $select->class = 'jumpmenu'; $select->formid = 'sectionmenu'; $o .= $this->output->render($select); return $o; } /** * Output the html for a single section page. * * @deprecated since 4.0 * * This is a deprecated method and it is mantain only for compatibilitiy with legacy course formats. * Please, to render a single section page use: * * $format = course_get_format($course); * // Set the section to display. * $format->set_section_number($displaysection); * $outputclass = $format->get_output_classname('content'); * $widget = new $outputclass($format); * echo $this->render($widget); * * @param stdClass $course The course entry from DB * @param array $sections (argument not used) * @param array $mods (argument not used) * @param array $modnames (argument not used) * @param array $modnamesused (argument not used) * @param int $displaysection The section number in the course which is being displayed */ public function print_single_section_page($course, $sections, $mods, $modnames, $modnamesused, $displaysection) { debugging('Method print_single_section_page is deprecated, please use' . 'core_courseformat\\output\\local\\content instead ' . 'or override render_content method to use a different template', DEBUG_DEVELOPER); // Some abstract methods are not needed anymore. We simulate them in case they are not present. if (method_exists($this, 'start_section_list')) { $startlist = $this->start_section_list(); } else { $startlist = html_writer::start_tag('ul', ['class' => '']); } if (method_exists($this, 'end_section_list')) { $endlist = $this->end_section_list(); } else { $endlist = html_writer::end_tag('ul'); } $format = course_get_format($course); // Set the section to display. $format->set_section_number($displaysection); $modinfo = $format->get_modinfo(); $course = $format->get_course(); // Can we view the section in question? if (!($sectioninfo = $modinfo->get_section_info($displaysection)) || !$sectioninfo->uservisible) { // This section doesn't exist or is not available for the user. // We actually already check this in course/view.php but just in case exit from this function as well. throw new \moodle_exception( 'unknowncoursesection', 'error', course_get_url($course), format_string($course->fullname) ); } // Copy activity clipboard.. echo $this->course_activity_clipboard($course, $displaysection); $thissection = $modinfo->get_section_info(0); if ($thissection->summary or !empty($modinfo->sections[0]) or $format->show_editor()) { echo $startlist; echo $this->section_header($thissection, $course, true, $displaysection); echo $this->course_section_cm_list($course, $thissection, $displaysection); echo $this->course_section_add_cm_control($course, 0, $displaysection); echo $this->section_footer(); echo $endlist; } // Start single-section div. echo html_writer::start_tag('div', array('class' => 'single-section')); // The requested section page. $thissection = $modinfo->get_section_info($displaysection); // Title with section navigation links. $sectionnavlinks = $this->get_nav_links($course, $modinfo->get_section_info_all(), $displaysection); $sectiontitle = ''; $sectiontitle .= html_writer::start_tag('div', array('class' => 'section-navigation navigationtitle')); $sectiontitle .= html_writer::tag('span', $sectionnavlinks['previous'], array('class' => 'mdl-left')); $sectiontitle .= html_writer::tag('span', $sectionnavlinks['next'], array('class' => 'mdl-right')); // Title attributes. $classes = 'sectionname'; if (!$thissection->visible) { $classes .= ' dimmed_text'; } $sectionname = html_writer::tag('span', $this->section_title_without_link($thissection, $course)); $sectiontitle .= $this->output->heading($sectionname, 3, $classes); $sectiontitle .= html_writer::end_tag('div'); echo $sectiontitle; // Now the list of sections. echo $startlist; echo $this->section_header($thissection, $course, true, $displaysection); echo $this->course_section_cm_list($course, $thissection, $displaysection); echo $this->course_section_add_cm_control($course, $displaysection, $displaysection); echo $this->section_footer(); echo $endlist; // Display section bottom navigation. $sectionbottomnav = ''; $sectionbottomnav .= html_writer::start_tag('div', array('class' => 'section-navigation mdl-bottom')); $sectionbottomnav .= html_writer::tag('span', $sectionnavlinks['previous'], array('class' => 'mdl-left')); $sectionbottomnav .= html_writer::tag('span', $sectionnavlinks['next'], array('class' => 'mdl-right')); $sectionbottomnav .= html_writer::tag( 'div', $this->section_nav_selection($course, $sections, $displaysection), array('class' => 'mdl-align') ); $sectionbottomnav .= html_writer::end_tag('div'); echo $sectionbottomnav; // Close single-section div. echo html_writer::end_tag('div'); } /** * Output the html for a multiple section page * * @deprecated since 4.0 * * This is a deprecated method and it is mantain only for compatibilitiy with legacy course formats. * Please, to render a single section page use: * * $format = course_get_format($course); * $outputclass = $format->get_output_classname('content'); * $widget = new $outputclass($format); * echo $this->render($widget); * * @param stdClass $course The course entry from DB * @param array $sections (argument not used) * @param array $mods (argument not used) * @param array $modnames (argument not used) * @param array $modnamesused (argument not used) */ public function print_multiple_section_page($course, $sections, $mods, $modnames, $modnamesused) { debugging('Method print_multiple_section_page is deprecated, please use' . 'core_courseformat\\output\\local\\content instead ' . 'or override render_content method to use a diferent template', DEBUG_DEVELOPER); // Some abstract methods are not needed anymore. We simulate them in case they are not present. if (method_exists($this, 'start_section_list')) { $startlist = $this->start_section_list(); } else { $startlist = html_writer::start_tag('ul', ['class' => '']); } if (method_exists($this, 'end_section_list')) { $endlist = $this->end_section_list(); } else { $endlist = html_writer::end_tag('ul'); } if (method_exists($this, 'page_title')) { $pagetitle = $this->page_title(); } else { $pagetitle = ''; } $format = course_get_format($course); $modinfo = $format->get_modinfo(); $course = $format->get_course(); $context = context_course::instance($course->id); echo $this->output->heading($pagetitle, 2, 'accesshide'); // Copy activity clipboard.. echo $this->course_activity_clipboard($course, 0); // Now the list of sections.. echo $startlist; $numsections = course_get_format($course)->get_last_section_number(); foreach ($modinfo->get_section_info_all() as $section => $thissection) { if ($section == 0) { // 0-section is displayed a little different then the others if ($thissection->summary or !empty($modinfo->sections[0]) or $format->show_editor()) { echo $this->section_header($thissection, $course, false, 0); echo $this->course_section_cm_list($course, $thissection, 0); echo $this->course_section_add_cm_control($course, 0, 0); echo $this->section_footer(); } continue; } if ($section > $numsections) { // Activities inside this section are 'orphaned', this section will be printed as 'stealth' below. continue; } if (!$format->is_section_visible($thissection)) { continue; } if (!$format->show_editor() && $format->get_course_display() == COURSE_DISPLAY_MULTIPAGE) { // Display section summary only. echo $this->section_summary($thissection, $course, null); } else { echo $this->section_header($thissection, $course, false, 0); if ($thissection->uservisible) { echo $this->course_section_cm_list($course, $thissection, 0); echo $this->course_section_add_cm_control($course, $section, 0); } echo $this->section_footer(); } } if ($format->show_editor()) { // Print stealth sections if present. foreach ($modinfo->get_section_info_all() as $section => $thissection) { if ($section <= $numsections or empty($modinfo->sections[$section])) { // This is not stealth section or it is empty. continue; } echo $this->stealth_section_header($section); echo $this->course_section_cm_list($course, $thissection, 0); echo $this->stealth_section_footer(); } echo $endlist; echo $this->change_number_sections($course, 0); } else { echo $endlist; } } /** * Returns controls in the bottom of the page to increase/decrease number of sections * * @deprecated since 4.0 MDL-72656 - use core_course output components instead. * * @param stdClass $course * @param int|null $sectionreturn */ protected function change_number_sections($course, $sectionreturn = null) { debugging('Method change_number_sections is deprecated, please use' . 'core_courseformat\\output\\local\\content\\addsection instead', DEBUG_DEVELOPER); $format = course_get_format($course); if ($sectionreturn) { $format->set_section_number($sectionreturn); } $outputclass = $format->get_output_classname('content\\addsection'); $widget = new $outputclass($format); echo $this->render($widget); } /** * Generate html for a section summary text * * @deprecated since 4.0 MDL-72656 - use core_course output components instead. * * @param stdClass $section The course_section entry from DB * @return string HTML to output. */ protected function format_summary_text($section) { debugging('Method format_summary_text is deprecated, please use' . 'core_courseformat\output\\local\\content\\section\\summary::format_summary_text instead', DEBUG_DEVELOPER); $format = course_get_format($section->course); if (!($section instanceof section_info)) { $modinfo = $format->get_modinfo(); $section = $modinfo->get_section_info($section->section); } $summaryclass = $format->get_output_classname('content\\section\\summary'); $summary = new $summaryclass($format, $section); return $summary->format_summary_text(); } }