You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

1096 lines
47 KiB

<?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/>.
/**
* This file contains main class for the course format Tiles
*
* @since Moodle 2.7
* @package format_tiles
* @copyright 2016 David Watson {@link http://evolutioncode.uk}
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
*/
defined('MOODLE_INTERNAL') || die();
define('FORMAT_TILES_FILTERBAR_NONE', 0);
define('FORMAT_TILES_FILTERBAR_NUMBERS', 1);
define('FORMAT_TILES_FILTERBAR_OUTCOMES', 2);
define('FORMAT_TILES_FILTERBAR_BOTH', 3);
require_once($CFG->dirroot . '/course/format/lib.php');
/**
* Main class for the course format Tiles
*
* @since Moodle 2.7
* @package format_tiles
* @copyright 2016 David Watson {@link http://evolutioncode.uk}
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
*/
class format_tiles extends format_base {
/**
* We want to treat label and plugins that behave like labels as labels.
* E.g. we don't render them as subtiles but show their content directly on page.
* And we don't count them for completiontracking.
* This includes plugins like mod_customlabel and mod_unilabel, as defined here.
* @var []
*/
public $labellikecoursemods = ['label', 'customlabel', 'unilabel'];
/**
* Creates a new instance of class
*
* Please use {@link course_get_format($courseorid)} to get an instance of the format class
*
* @param string $format
* @param int $courseid
*/
protected function __construct($format, $courseid) {
if ($courseid === 0) {
global $COURSE;
$courseid = $COURSE->id; // Save lots of global $COURSE as we will never be the site course.
}
parent::__construct($format, $courseid);
}
/**
* Returns true if this course format uses sections
*
* @return bool
*/
public function uses_sections() {
return true;
}
/**
* Returns the display name of the given section that the course prefers.
*
* Use section name is specified by user. Otherwise use default ("Topic #")
*
* @param int|stdClass $section Section object from database or just field section.section
* @return string Display name that the course format prefers, e.g. "Topic 2"
* @throws moodle_exception
*/
public function get_section_name($section) {
$section = $this->get_section($section);
if ((string)$section->name !== '') {
return format_string($section->name, true,
array('context' => context_course::instance($this->courseid)));
} else if ($section->section == 0) {
return get_string('section0name', 'format_tiles');
} else {
return get_string('sectionname', 'format_tiles') . ' ' . $section->section;
}
}
/**
* Returns the default section name for the topics course format.
*
* If the section number is 0, it will use the string with key = section0name from the course format's lang file.
* If the section number is not 0, the base implementation of format_base::get_default_section_name which uses
* the string with the key = 'sectionname' from the course format's lang file + the section number will be used.
*
* @param stdClass $section Section object from database or just field course_sections section
* @return string The default value for the section name.
* @throws coding_exception
*/
public function get_default_section_name($section) {
if ($section->section == 0) {
// Return the general section.
return get_string('section0name', 'format_tiles');
} else {
// Use format_base::get_default_section_name implementation which will display the section name in "Topic n" format.
return parent::get_default_section_name($section);
}
}
/**
* Indicates whether the course format supports the creation of a news forum.
* Required in Moodle 3.2 onwards
*
* @return bool
*/
public function supports_news() {
return true;
}
/**
* The URL to use for the specified course (with section)
*
* @param int|stdClass $section Section object from database or just field course_sections.section
* if omitted the course view page is returned
* @param array $options options for view URL. At the moment core uses:
* 'navigation' (bool) if true and section has no separate page, the function returns null
* 'sr' (int) used by multipage formats to specify to which section to return
* @return null|moodle_url
* @throws moodle_exception
*/
public function get_view_url($section, $options = array()) {
$course = $this->get_course();
$url = new moodle_url('/course/view.php', array('id' => $course->id));
$sr = null;
if (array_key_exists('sr', $options)) {
$sr = $options['sr'];
}
if (is_object($section)) {
$sectionno = $section->section;
} else {
$sectionno = $section;
}
if ($sectionno !== null) {
if ($sr !== null) {
$sectionno = $sr;
}
if ($sectionno != 0) {
$url->param('section', $sectionno);
} else {
if (!empty($options['navigation'])) {
return null;
}
$url->set_anchor('section-' . $sectionno);
}
}
return $url;
}
/**
* Returns the information about the ajax support in the given source format
*
* The returned object's property (boolean)capable indicates that
* the course format supports Moodle course ajax features.
*
* @return stdClass
*/
public function supports_ajax() {
$ajaxsupport = new stdClass();
$ajaxsupport->capable = true;
return $ajaxsupport;
}
/**
* Override if you need to perform some extra validation of the format options
*
* @param array $data array of ("fieldname"=>value) of submitted data
* @param array $files array of uploaded files "element_name"=>tmp_file_path
* @param array $errors errors already discovered in edit form validation
* @return array of "element_name"=>"error_description" if there are errors,
* or an empty array if everything is OK.
* Do not repeat errors from $errors param here
* @throws coding_exception
* @throws moodle_exception
*/
public function edit_form_validation($data, $files, $errors) {
$courseid = $data['id'];
$reterrors = array();
if (!$data['enablecompletion'] && $data['courseshowtileprogress']) {
$reterrors['courseshowtileprogress'] = get_string('courseshowtileprogress_error', 'format_tiles');
}
if (($data['displayfilterbar'] == FORMAT_TILES_FILTERBAR_OUTCOMES
|| $data['displayfilterbar'] == FORMAT_TILES_FILTERBAR_BOTH)
&& empty($this->format_tiles_get_course_outcomes($courseid))) {
$outcomeslink = html_writer::link(
new moodle_url('/grade/edit/outcome/course.php', array('id' => $courseid)),
new lang_string('outcomes', 'format_tiles')
);
$reterrors['displayfilterbar'] = get_string('displayfilterbar_error', 'format_tiles') . ' ' . $outcomeslink;
}
return $reterrors;
}
/**
* Loads all of the course sections into the navigation
*
* @param global_navigation $navigation
* @param navigation_node $node The course node within the navigation
* @return void
* @throws coding_exception
* @throws moodle_exception
*/
public function extend_course_navigation($navigation, navigation_node $node) {
global $PAGE;
// If section is specified in course/view.php, make sure it is expanded in navigation.
if ($navigation->includesectionnum === false) {
$selectedsection = optional_param('section', null, PARAM_INT);
if ($selectedsection !== null && (!defined('AJAX_SCRIPT') || AJAX_SCRIPT == '0') &&
$PAGE->url->compare(new moodle_url('/course/view.php'), URL_MATCH_BASE)) {
$navigation->includesectionnum = $selectedsection;
}
}
// Check if there are callbacks to extend course navigation.
parent::extend_course_navigation($navigation, $node);
// We want to remove the general section if it is empty.
$course = $this->get_course();
$modinfo = get_fast_modinfo($course);
$sections = $modinfo->get_sections();
if (!isset($sections[0])) {
// The general section is empty to find the navigation node for it we need to get its ID.
$section = $modinfo->get_section_info(0);
$generalsection = $node->get($section->id, navigation_node::TYPE_SECTION);
if ($generalsection) {
// We found the node - now remove it.
$generalsection->remove();
}
}
if (get_config('format_tiles', 'usejavascriptnav') && !(\core_useragent::is_ie())) {
if (!get_user_preferences('format_tiles_stopjsnav', 0)) {
$url = new moodle_url('/course/view.php', array('id' => $course->id, 'stopjsnav' => 1));
$settingnode = $node->add(
get_string('jsdeactivate', 'format_tiles'),
$url->out(),
navigation_node::TYPE_SETTING,
null,
null,
new pix_icon(
'toggle-on',
get_string('jsdeactivate', 'format_tiles'),
'format_tiles'
)
);
$settingnode->nodetype = navigation_node::NODETYPE_LEAF;
// Can't add classes or ids here if using boost (works in clean).
$settingnode->id = 'tiles_stopjsnav';
$settingnode->add_class('tiles_coursenav hidden');
// Now the Data Preference menu item.
if (!get_config('format_tiles', 'assumedatastoreconsent')) {
$url = new moodle_url('/course/view.php', array('id' => $course->id, 'datapref' => 1));
$settingnode = $node->add(
get_string('datapref', 'format_tiles'),
$url->out(),
navigation_node::TYPE_SETTING,
null,
null,
new pix_icon(
'i/db',
get_string('datapref', 'format_tiles')
)
);
$settingnode->nodetype = navigation_node::NODETYPE_LEAF;
// Can't add classes or ids here if using boost (works in clean).
$settingnode->id = 'tiles_datapref';
$settingnode->add_class('tiles_coursenav hidden');
}
} else {
$settingnode = $node->add(
get_string('jsactivate', 'format_tiles'),
new moodle_url('/course/view.php', array('id' => $course->id, 'stopjsnav' => 1)),
navigation_node::TYPE_SETTING,
null,
null,
new pix_icon(
'toggle-off',
get_string('jsactivate', 'format_tiles'),
'format_tiles'
)
);
$settingnode->nodetype = navigation_node::NODETYPE_LEAF;
// Can't add classes or ids here if using boost (works in clean).
$settingnode->id = 'tiles_stopjsnav';
$settingnode->add_class('tiles_coursenav hidden');
}
}
}
/**
* Custom action after section has been moved in AJAX mode
*
* Used in course/rest.php
*
* @return array This will be passed in ajax response
* @throws moodle_exception
*/
public function ajax_section_move() {
global $PAGE;
$titles = array();
$course = $this->get_course();
$modinfo = get_fast_modinfo($course);
$renderer = $this->get_renderer($PAGE);
if ($renderer && ($sections = $modinfo->get_section_info_all())) {
foreach ($sections as $number => $section) {
$titles[$number] = $renderer->section_title($section, $course);
}
}
return array('sectiontitles' => $titles, 'action' => 'move');
}
/**
* Returns the list of blocks to be automatically added for the newly created course
*
* @return array of default blocks, must contain two keys BLOCK_POS_LEFT and BLOCK_POS_RIGHT
* each of values is an array of block names (for left and right side columns)
*/
public function get_default_blocks() {
return array(
BLOCK_POS_LEFT => array(),
BLOCK_POS_RIGHT => array()
);
}
/**
* Iterates through all the colours entered by the administrator under the plugin settings page
* @return array list of all the colours and their names for use in the settings forms
* @throws dml_exception
*/
private function format_tiles_get_tiles_palette() {
$palette = array();
for ($i = 1; $i <= 10; $i++) {
$colourname = get_config('format_tiles', 'colourname' . $i);
$tilecolour = get_config('format_tiles', 'tilecolour' . $i);
if ($tilecolour != '' and $tilecolour != '#000') {
$palette[$tilecolour] = $colourname;
}
}
return $palette;
}
/**
* Whether this format allows to delete sections (Moodle 3.1+)
* If format supports deleting sections it is also recommended to define language string
* 'deletesection' inside the format.
* Do not call this function directly, instead use {@link course_can_delete_section()}
*
* @param int|stdClass|section_info $section
* @return bool
*/
public function can_delete_section($section) {
return true;
}
/**
* Definitions of the additional options that this course format uses for course
*
* @param bool $foreditform
* @return array of options
* @throws coding_exception
* @throws dml_exception
* @throws moodle_exception
*/
public function course_format_options($foreditform = false) {
static $courseformatoptions = false;
if ($courseformatoptions === false) {
$courseformatoptions = array(
'hiddensections' => array(
'default' => 1,
'type' => PARAM_INT,
),
'coursedisplay' => array(
'default' => 1,
'type' => PARAM_INT,
),
'defaulttileicon' => array(
'default' => 'pie-chart',
'type' => PARAM_TEXT,
),
'basecolour' => array(
'default' => get_config('format_tiles', 'tilecolour1'),
'type' => PARAM_TEXT,
),
'courseusesubtiles' => array(
'default' => 0,
'type' => PARAM_INT,
),
'courseshowtileprogress' => array(
'default' => 0,
'type' => PARAM_INT,
),
'displayfilterbar' => array(
'default' => 0,
'type' => PARAM_INT,
),
'usesubtilesseczero' => array(
'default' => 0,
'type' => PARAM_INT
),
'courseusebarforheadings' => array(
'default' => 1,
'type' => PARAM_INT,
)
);
if ((get_config('format_tiles', 'followthemecolour'))) {
unset($courseformatoptions['basecolour']);
}
if (!get_config('format_tiles', 'allowsubtilesview')) {
unset($courseformatoptions['courseusesubtiles']);
}
}
if ($foreditform && !isset($courseformatoptions['coursedisplay']['label'])) {
$tilespalette = $this->format_tiles_get_tiles_palette();
$tileicons = (new \format_tiles\icon_set)->available_tile_icons($this->get_courseid());
$courseconfig = get_config('moodlecourse');
$max = $courseconfig->maxsections;
if (!isset($max) || !is_numeric($max)) {
$max = 52;
}
$sectionmenu = array();
for ($i = 0; $i <= $max; $i++) {
$sectionmenu[$i] = "$i";
}
$courseformatoptionsedit = array(
'hiddensections' => array(
'label' => new lang_string('hiddensections'),
'element_type' => 'hidden',
'element_attributes' => array(
array(1 => new lang_string('hiddensectionsinvisible'))
),
),
'coursedisplay' => array(
'label' => new lang_string('coursedisplay'),
'element_type' => 'hidden',
'element_attributes' => array(
array(
COURSE_DISPLAY_MULTIPAGE => new lang_string('coursedisplay_multi')
)
),
),
);
$label = get_string('defaulttileicon', 'format_tiles');
$courseformatoptionsedit['defaulttileicon'] = array(
'label' => $label,
'element_type' => 'select',
'element_attributes' => array($tileicons),
'help' => 'defaulttileicon',
'help_component' => 'format_tiles',
);
if (!(get_config('format_tiles', 'followthemecolour'))) {
$courseformatoptionsedit['basecolour'] = array(
'label' => new lang_string('basecolour', 'format_tiles'),
'element_type' => 'select',
'element_attributes' => array($tilespalette),
'help' => 'basecolour',
'help_component' => 'format_tiles',
);
}
$attributes = array(
FORMAT_TILES_FILTERBAR_NONE => new lang_string('hide', 'format_tiles'),
FORMAT_TILES_FILTERBAR_NUMBERS => new lang_string('filternumbers', 'format_tiles'),
);
$outcomeslink = '(' . new lang_string('outcomesunavailable', 'format_tiles') . ')';
global $CFG;
if (!empty($CFG->enableoutcomes)) {
$outcomeslink = html_writer::link(
new moodle_url('/grade/edit/outcome/course.php',
array('id' => $this->get_courseid())),
'(' . new lang_string('outcomes', 'format_tiles') . ')'
);
$attributes[FORMAT_TILES_FILTERBAR_OUTCOMES] = new lang_string('filteroutcomes', 'format_tiles');
$attributes[FORMAT_TILES_FILTERBAR_BOTH] = new lang_string('filterboth', 'format_tiles');
}
$courseformatoptionsedit['displayfilterbar'] = array(
'label' => new lang_string('displayfilterbar', 'format_tiles') . ' ' . $outcomeslink,
'element_type' => 'select',
'element_attributes' => array($attributes),
'help' => 'displayfilterbar',
'help_component' => 'format_tiles',
);
$courseformatoptionsedit['courseshowtileprogress'] = array(
'label' => new lang_string('courseshowtileprogress', 'format_tiles'),
'element_type' => 'select',
'element_attributes' => array(
array(
0 => new lang_string('hide', 'format_tiles'),
1 => new lang_string('asfraction', 'format_tiles'),
2 => new lang_string('aspercentagedial', 'format_tiles'),
),
),
'help' => 'courseshowtileprogress',
'help_component' => 'format_tiles'
);
if (get_config('format_tiles', 'allowsubtilesview')) {
$courseformatoptionsedit['courseusesubtiles'] = array(
'label' => new lang_string('courseusesubtiles', 'format_tiles'),
'element_type' => 'advcheckbox',
'element_attributes' => array(get_string('yes')),
'help' => 'courseusesubtiles',
'help_component' => 'format_tiles',
);
}
$courseformatoptionsedit['courseusebarforheadings'] = array(
'label' => new lang_string(
'courseusebarforheadings', 'format_tiles'
),
'element_type' => 'advcheckbox',
'element_attributes' => array(get_string('yes')),
'help' => 'courseusebarforheadings',
'help_component' => 'format_tiles',
);
$courseformatoptionsedit['usesubtilesseczero'] = array(
'label' => new lang_string('usesubtilesseczero', 'format_tiles'),
'element_type' => 'advcheckbox',
'element_attributes' => array(get_string('notrecommended', 'format_tiles')),
'help' => 'usesubtilesseczero',
'help_component' => 'format_tiles',
);
$courseformatoptions = array_merge_recursive($courseformatoptions, $courseformatoptionsedit);
}
return $courseformatoptions;
}
/**
* Definitions of the additional options that this course format uses for section
*
* See {@link format_base::course_format_options()} for return array definition.
*
* Additionally section format options may have property 'cache' set to true
* if this option needs to be cached in {@link get_fast_modinfo()}. The 'cache' property
* is recommended to be set only for fields used in {@link format_base::get_section_name()},
* {@link format_base::extend_course_navigation()} and {@link format_base::get_view_url()}
*
* For better performance cached options are recommended to have 'cachedefault' property
* Unlike 'default', 'cachedefault' should be static and not access get_config().
*
* Regardless of value of 'cache' all options are accessed in the code as
* $sectioninfo->OPTIONNAME
* where $sectioninfo is instance of section_info, returned by
* get_fast_modinfo($course)->get_section_info($sectionnum)
* or get_fast_modinfo($course)->get_section_info_all()
*
* All format options for particular section are returned by calling:
* $this->get_format_options($section);
*
* @param bool $foreditform
* @return array
* @throws coding_exception
* @throws moodle_exception
*/
public function section_format_options($foreditform = false) {
global $DB;
$course = $this->get_course();
$sectionformatoptions = array(
'tileicon' => array(
'default' => '',
'type' => PARAM_TEXT,
),
);
if ($course->displayfilterbar == FORMAT_TILES_FILTERBAR_OUTCOMES
|| $course->displayfilterbar == FORMAT_TILES_FILTERBAR_BOTH) {
$sectionformatoptions['tileoutcomeid'] = array(
'default' => 0,
'type' => PARAM_INT,
);
}
if (get_config('format_tiles', 'allowphototiles')) {
$sectionformatoptions['tilephoto'] = array(
'default' => '',
'type' => PARAM_TEXT
);
}
if ($foreditform) {
$defaultcoursetile = $course->defaulttileicon;
$defaulticonarray = array(
'' => get_string('defaultthiscourse', 'format_tiles') . ' (' . $defaultcoursetile . ')'
);
$tileicons = (new \format_tiles\icon_set)->available_tile_icons($course->id);
$tileicons = array_merge($defaulticonarray, $tileicons);
$sectionformatoptionsedit = array();
$label = get_string('tileicon', 'format_tiles');
$sectionformatoptionsedit['tileicon'] = array(
'label' => $label,
'element_type' => 'select',
'element_attributes' => array($tileicons),
'help' => 'tileicon',
);
if ($course->displayfilterbar == FORMAT_TILES_FILTERBAR_OUTCOMES
|| $course->displayfilterbar == FORMAT_TILES_FILTERBAR_BOTH) {
$outcomeslink = html_writer::link(
new moodle_url('/grade/edit/outcome/course.php', array('id' => $course->id)),
'(' . new lang_string('outcomes', 'format_tiles') . ')'
);
$label = get_string('tileoutcome', 'format_tiles') . ' ' . $outcomeslink;
$outcomes = $this->format_tiles_get_course_outcomes($course->id);
if (!empty($outcomes)) {
$outcomes[0] = get_string('none', 'format_tiles');
}
$sectionformatoptionsedit['tileoutcomeid'] = array(
'label' => $label,
'element_type' => 'select',
'element_attributes' => array($outcomes),
'help' => 'tileoutcome',
);
}
if (get_config('format_tiles', 'allowphototiles')) {
$sectionformatoptionsedit['tilephoto'] = array(
'label' => get_string('uploadnewphoto', 'format_tiles'),
'element_type' => 'hidden',
'element_attributes' => array('' => '')
);
}
$sectionformatoptions = array_merge_recursive($sectionformatoptions, $sectionformatoptionsedit);
}
return $sectionformatoptions;
}
/**
* Adds format options elements to the course/section edit form.
*
* This function is called from {@link course_edit_form::definition_after_data()}.
*
* @param MoodleQuickForm $mform form the elements are added to.
* @param bool $forsection 'true' if this is a section edit form, 'false' if this is course edit form.
* @return array array of references to the added form elements.
* @throws HTML_QuickForm_Error
* @throws coding_exception
* @throws dml_exception
*/
public function create_edit_form_elements(&$mform, $forsection = false) {
global $COURSE, $PAGE, $DB, $USER;
$elements = parent::create_edit_form_elements($mform, $forsection);
// Call the JS edit_form_helper.js, which in turn will call edit_icon_picker.js.
if ($forsection) {
$sectionid = optional_param('id', 0, PARAM_INT);
$section = $DB->get_field('course_sections', 'section', array('id' => $sectionid));
} else {
// We are on the course setting page so can ignore section.
$section = 0;
$sectionid = 0;
}
$jsparams = array(
'pageType' => $PAGE->pagetype,
'courseDefaultIcon' => $this->get_format_options()['defaulttileicon'],
'courseId' => $COURSE->id,
'sectionId' => $sectionid,
'section' => $section,
'userId' => $USER->id,
get_config('format_tiles', 'allowphototiles') && $section !== 0, // No photos on course page.
get_config('format_tiles', 'documentationurl')
);
$PAGE->requires->js_call_amd('format_tiles/edit_form_helper', 'init', $jsparams);
if (!$forsection && (empty($COURSE->id) || $COURSE->id == SITEID)) {
// Add "numsections" to create course form - will force the course pre-populated with empty sections.
// The "Number of sections" option is no longer available when editing course.
// Instead teachers should delete and add sections when needed.
$courseconfig = get_config('moodlecourse');
$max = (int)$courseconfig->maxsections;
$element = $mform->addElement('select', 'numsections', get_string('numberweeks'), range(0, $max ?: 52));
$mform->setType('numsections', PARAM_INT);
if (is_null($mform->getElementValue('numsections'))) {
$mform->setDefault('numsections', $courseconfig->numsections);
}
array_unshift($elements, $element);
}
return $elements;
}
/**
* Updates format options for a course
*
* If course format was changed to 'tiles', we try to copy options
* from the previous format. We do not copy 'coursedisplay',
* and 'hiddensections' as a defaut value of one makes sense for these for tiles format,
* regardless of what they were.
*
* @param stdClass|array $data return value from {@link moodleform::get_data()} or array with data
* @param stdClass $oldcourse if this function is called from {@link update_course()}
* this object contains information about the course before update
* @return bool whether there were any changes to the options values
* @throws coding_exception
* @throws dml_exception
* @throws moodle_exception
*/
public function update_course_format_options($data, $oldcourse = null) {
global $DB, $USER;
$data = (array)$data;
if ($oldcourse !== null) {
$oldcourse = (array)$oldcourse;
$options = $this->course_format_options();
foreach ($options as $key => $unused) {
if (!array_key_exists($key, $data)) {
if (array_key_exists($key, $oldcourse)) {
$data[$key] = $oldcourse[$key];
}
}
}
}
if (isset($data['id']) && $data['id']) {
$courseid = $data['id'];
$coursecontext = context_course::instance($courseid);
if (has_capability('moodle/course:update', $coursecontext)) {
if ($oldcourse['format'] !== 'tiles' && $oldcourse['format'] !== 'tiles') {
// We are switching in to tiles from something else.
// Double check we don't have any old tiles images in the {files} table.
format_tiles\tile_photo::delete_all_tile_photos_course($courseid);
}
// If we are changing from Grid format, we iterate through each of the grid images and set it up for this format.
if ($oldcourse['format'] == 'grid') {
$gridformaticons = $DB->get_records('format_grid_icon', array('courseid' => $courseid), 'sectionid');
$fs = get_file_storage();
foreach ($gridformaticons as $gridformaticon) {
if (!$gridformaticon->image) {
continue;
}
$tilephoto = new \format_tiles\tile_photo($courseid, $gridformaticon->sectionid);
$gridfile = $fs->get_file(
$coursecontext->id,
'course',
'section',
$gridformaticon->sectionid,
'/gridimage/',
$gridformaticon->displayedimageindex . '_' . $gridformaticon->image
);
if ($gridfile) {
// We copy the grid image file into Tiles format, so it is included in backups etc.
$fs = get_file_storage();
$newfilerecord = \format_tiles\tile_photo::file_api_params();
$newfilerecord['contextid'] = $coursecontext->id;
$newfilerecord['itemid'] = $gridformaticon->sectionid;
$newfilerecord['userid'] = $USER->id;
$newfilerecord['filename'] = str_replace('_goi_', '_', $gridfile->get_filename());
$fs->delete_area_files(
$coursecontext->id,
$newfilerecord['component'],
$newfilerecord['filearea'],
$newfilerecord['itemid']
);
$newfile = $fs->create_file_from_storedfile($newfilerecord, $gridfile);
if ($newfile) {
$tilephoto->set_file($newfile);
// We *could* delete grid format files here, but we don't as they don't belong to us.
// If we don't, they will be included in export course archives.
}
} else {
debugging(
'Grid format image not found '
. $gridformaticon->displayedimageindex . '_' . $gridformaticon->image,
DEBUG_DEVELOPER
);
}
}
}
// While we are changing the format options, set section zero to visible if it is hidden.
// Should never be hidden but rarely it happens, for reasons which are not clear esp with onetopic format.
// See https://moodle.org/mod/forum/discuss.php?d=356850 and MDL-37256).
if ($section = $DB->get_record("course_sections", array('course' => $courseid, 'section' => 0))) {
if (!$section->visible) {
set_section_visible($section->course, 0, 1);
}
}
}
}
if (isset($data['courseusesubtiles']) && $data['courseusesubtiles'] == 0) {
// We are deactivating sub tiles at course level so do it at sec zero level too.
$data['usesubtilesseczero'] = 0;
}
return $this->update_format_options($data);
}
/**
* Updates format options for a section
* Includes a check to strip out default values for tile icon or outcome id
* as it would be wasteful to store large volumes of these on a per section basis
*
* Section id is expected in $data->id (or $data['id'])
* If $data does not contain property with the option name, the option will not be updated
*
* @param stdClass|array $data return value from {@link moodleform::get_data()} or array with data
* @return bool whether there were any changes to the options values
* @throws dml_exception
*/
public function update_section_format_options($data) {
global $DB;
$data = (array)$data;
$oldvalues = array(
'iconthistile' => $DB->get_field(
'course_format_options', 'value',
['format' => 'tiles', 'sectionid' => $data['id'], 'name' => 'tileicon']
),
'outcomethistile' => $DB->get_record(
'course_format_options',
['format' => 'tiles', 'sectionid' => $data['id'], 'name' => 'tileoutcomeid']
),
'photothistile' => \format_tiles\tile_photo::get_course_format_option_value($data['id'])
);
// If the edit is taking place from format_tiles_inplace_editable(),
// the data array may not contain the tile icon and outcome id at all.
// So add these items in if missing.
if (!isset($data['tileicon']) && $oldvalues['iconthistile']) {
$data['tileicon'] = $oldvalues['iconthistile'];
}
if (!isset($data['tileoutcomeid']) && $oldvalues['outcomethistile']) {
$data['tileoutcomeid'] = $oldvalues['outcomethistile'];
}
if (!isset($data['tilephoto']) && $oldvalues['photothistile']) {
$data['tilephoto'] = $oldvalues['photothistile'];
}
// Unset the new values if null, before we send to update.
// This is so that we don't get a false positive as to whether it has changed or not.
if (isset($data['tileicon']) && $data['tileicon'] == '') {
unset($data['tileicon']);
}
if (isset($data['tileoutcomeid']) && $data['tileoutcomeid'] == '0') {
unset($data['tileoutcomeid']);
}
if (isset($data['tilephoto']) && $data['tilephoto'] == '') {
unset($data['tilephoto']);
}
// Now send the update.
$result = $this->update_format_options($data, $data['id']);
// Now remove any default values such as '' or '0' which the update stored in the database as they are redundant.
$keystoremove = ['tileicon', 'tileoutcomeid', 'tilephoto'];
foreach ($keystoremove as $key) {
if (!isset($data[$key])) {
$DB->delete_records('course_format_options', ['format' => 'tiles', 'sectionid' => $data['id'], 'name' => $key]);
if (isset($oldvalues[$key]) && $oldvalues[$key]) {
// Used to have a value so return true to indicate it changed.
$result = true;
}
}
}
return $result;
}
/**
* Prepares the templateable object to display section name
*
* @param \section_info|\stdClass $section
* @param bool $linkifneeded
* @param bool $editable
* @param null|lang_string|string $edithint
* @param null|lang_string|string $editlabel
* @return \core\output\inplace_editable
* @throws coding_exception
*/
public function inplace_editable_render_section_name($section, $linkifneeded = true,
$editable = null, $edithint = null, $editlabel = null) {
global $USER;
if (empty($edithint)) {
$edithint = new lang_string('editsectionname', 'format_tiles');
}
if (empty($editlabel)) {
$title = get_section_name($section->course, $section);
$editlabel = new lang_string('newsectionname', 'format_tiles', $title);
}
if ($editable === null) {
$editable = !empty($USER->editing) && has_capability('moodle/course:update',
context_course::instance($section->course));
}
$displayvalue = $title = get_section_name($section->course, $section);
if ($linkifneeded) {
// Display link under the section name if the course format setting is to display one section per page.
$url = new moodle_url(
'/course/view.php',
array('id' => $section->course, 'section' => $section->section, 'singlesec' => $section->section)
);
if ($url) {
$displayvalue = html_writer::link($url, $title);
}
$itemtype = 'sectionname';
} else {
// If $linkifneeded==false, we never display the link (this is used when rendering the section header).
// Itemtype 'sectionnamenl' (nl=no link) will tell the callback that link should not be rendered -
// there is no other way callback can know where we display the section name.
$itemtype = 'sectionnamenl';
}
if (empty($edithint)) {
$edithint = new lang_string('editsectionname');
}
if (empty($editlabel)) {
$editlabel = new lang_string('newsectionname', '', $title);
}
return new \core\output\inplace_editable('format_' . $this->format, $itemtype, $section->id, $editable,
$displayvalue, $section->name, $edithint, $editlabel);
}
/**
* Get an array of all the Outcomes set for this course by the teacher, so that they can
* be attached to individual Tiles, and then used to filter tiles by Outcome
* @see get_filter_outcome_buttons()
* @see course_format_options() and the displayfilterbar option
* @param int $courseid
* @return array|null
*/
public function format_tiles_get_course_outcomes($courseid) {
global $CFG;
if (!empty($CFG->enableoutcomes)) {
require_once($CFG->libdir . '/gradelib.php');
$outcomes = [];
$outcomesfull = grade_outcome::fetch_all_available($courseid);
foreach ($outcomesfull as $outcome) {
$outcomes[$outcome->id] = $outcome->fullname;
}
asort($outcomes);
return $outcomes;
} else {
return null;
}
}
/**
* Returns whether this course format allows the activity to
* have "triple visibility state" - visible always, hidden on course page but available, hidden.
* Copied from format_topics
*
* @param stdClass|cm_info $cm course module (may be null if we are displaying a form for adding a module)
* @param stdClass|section_info $section section where this module is located or will be added to
* @return bool
*/
public function allow_stealth_module_visibility($cm, $section) {
// Allow the third visibility state inside visible sections or in section 0.
return !$section->section || $section->visible;
}
/**
* Callback used in WS core_course_edit_section when teacher performs an AJAX action on a section (show/hide)
*
* Access to the course is already validated in the WS but the callback has to make sure
* that particular action is allowed by checking capabilities
*
* Course formats should register
*
* @param stdClass|section_info $section
* @param string $action
* @param int $sr
* @return null|array|stdClass any data for the Javascript post-processor (must be json-encodeable)
* @throws moodle_exception
* @throws required_capability_exception
*/
public function section_action($section, $action, $sr) {
global $PAGE;
if ($section->section && ($action === 'setmarker' || $action === 'removemarker')) {
// Format 'tiles' allows to set and remove markers in addition to common section actions.
require_capability('moodle/course:setcurrentsection', context_course::instance($this->courseid));
course_set_marker($this->courseid, ($action === 'setmarker') ? $section->section : 0);
return null;
}
// For show/hide actions call the parent method and return the new content for .section_availability element.
$rv = parent::section_action($section, $action, $sr);
$renderer = $PAGE->get_renderer('format_tiles');
$rv['section_availability'] = $renderer->section_availability($this->get_section($section));
return $rv;
}
/**
* Allows course format to execute code on moodle_page::set_course()
* Used here to ensure that, before starting to load the page,
* we establish if the user is changing their pref for using JS nav
* and change the setting if so
*
* @param moodle_page $page instance of page calling set_course
* @throws coding_exception
* @throws dml_exception
* @throws moodle_exception
*/
public function page_set_course(moodle_page $page) {
if (get_config('format_tiles', 'usejavascriptnav')) {
if (optional_param('stopjsnav', 0, PARAM_INT) == 1) {
// User is toggling JS nav setting.
$existingstoppref = get_user_preferences('format_tiles_stopjsnav', 0);
if (!$existingstoppref) {
// Did not already have it disabled.
set_user_preference('format_tiles_stopjsnav', 1);
} else {
// User previously disabled it, but now is re-enabling.
unset_user_preference('format_tiles_stopjsnav');
\core\notification::success(get_string('jsreactivated', 'format_tiles'));
}
if ($page->course->id) {
redirect(new moodle_url('/course/view.php', array('id' => $page->course->id)));
}
}
}
}
}
/**
* Implements callback inplace_editable() allowing to edit values in-place
*
* @param string $itemtype
* @param int $itemid
* @param mixed $newvalue
* @return \core\output\inplace_editable | null
* @throws dml_exception
*/
function format_tiles_inplace_editable($itemtype, $itemid, $newvalue) {
global $DB, $CFG;
require_once($CFG->dirroot . '/course/lib.php');
if ($itemtype === 'sectionname' || $itemtype === 'sectionnamenl') {
$section = $DB->get_record_sql(
'SELECT s.* FROM {course_sections} s JOIN {course} c ON s.course = c.id WHERE s.id = ? AND c.format = ?',
array($itemid, 'tiles'), MUST_EXIST);
return course_get_format($section->course)->inplace_editable_update_section_name($section, $itemtype, $newvalue);
}
}
/**
* Get icon mapping for font-awesome.
* @return array the icons for which theme should use font awesome.
*/
function format_tiles_get_fontawesome_icon_map() {
$iconset = new format_tiles\icon_set();
return $iconset->get_font_awesome_icon_map();
}
/**
* Serves any files associated with the plugin (e.g. tile photos).
* For explanation see https://docs.moodle.org/dev/File_API
*
* @param stdClass $course
* @param stdClass $cm
* @param context $context
* @param string $filearea
* @param array $args
* @param bool $forcedownload
* @param array $options
* @return bool
*/
function format_tiles_pluginfile($course, $cm, $context, $filearea, $args, $forcedownload, array $options = array()) {
if ($context->contextlevel != CONTEXT_COURSE && $context->contextlevel != CONTEXT_SYSTEM) {
send_file_not_found();
}
if ($filearea !== 'tilephoto') {
debugging('Invalid file area ' . $filearea);
send_file_not_found();
}
// Make sure the user is logged in and has access to the course.
require_login($course);
$fileapiparams = \format_tiles\tile_photo::file_api_params();
$fs = get_file_storage();
$sectionid = (int)$args[0];
$filepath = '/' . $args[1] .'/';
$filename = $args[2];
$file = $fs->get_file($context->id, $fileapiparams['component'], $filearea, $sectionid, $filepath, $filename);
send_stored_file($file, 86400, 0, $forcedownload, $options);
}