<?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/>. /** * Semester customfield data controller * * Semesters are encoded as YYYYS, where YYYY is the year when the semester begins and * S is the identifier of the term. * * The identifier of the term is depending on the admin setting customfield_semester/internaltermrepresentation. * By default, it is 0 = summer term and 1 = winter term. So YYYYS = 20191 stands for the winter term 2019/20. * * In addition to that, there are term-independent terms which are represented by YYYYS = 1. * * @package customfield_semester * @copyright 2020 Justus Dieckmann WWU * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ namespace customfield_semester; use DateTime; defined('MOODLE_INTERNAL') || die; /** * Semester customfield data controller * * @package customfield_semester * @copyright 2020 Justus Dieckmann WWU * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ class data_controller extends \core_customfield\data_controller { /** * Return the name of the field where the information is stored * * @return string */ public function datafield(): string { return 'intvalue'; } /** * Returns the default value as it would be stored in the database (not in human-readable format). * * @return mixed */ public function get_default_value() { $defaultmonthsintofuture = $this->get_field()->get_configdata_property('defaultmonthsintofuture'); return self::get_semester_for_datetime(new DateTime("+$defaultmonthsintofuture months")); } /** * Add fields for editing a textarea field. * * @param \MoodleQuickForm $mform */ public function instance_form_definition(\MoodleQuickForm $mform) { global $CFG; // Require local library. require_once($CFG->dirroot.'/customfield/field/semester/locallib.php'); // Get config from DB. $config = get_config('customfield_semester'); // Compose the field values. $field = $this->get_field(); $formattedoptions = array( self::get_termindependent_representation() => get_string('semesterindependent', 'customfield_semester') ); $showmonthsintofuture = $this->get_field()->get_configdata_property('showmonthsintofuture'); $endtime = new DateTime("+$showmonthsintofuture months"); $endkey = self::get_semester_for_datetime($endtime); $endyear = intdiv($endkey, 10); $endsemester = $endkey % 10; $beginofsemesters = $this->get_field()->get_configdata_property('beginofsemesters'); for ($year = $beginofsemesters; $year <= $endyear; $year++) { $formattedoptions[$year * 10 + self::get_summerterm_representation()] = get_string('summersemester', 'customfield_semester', $year); if ($year == $endyear && $endsemester == self::get_summerterm_representation()) { break; } $formattedoptions[$year * 10 + self::get_winterterm_representation()] = get_string('wintersemester', 'customfield_semester', $year . '/' . substr($year + 1, 2, 2)); } // The values were composed in CUSTOMFIELD_SEMESTER_PRESENTATION_ASC order here. // If the admin wants to present them in CUSTOMFIELD_SEMESTER_PRESENTATION_DESC order, we need to reverse the array now. if ($config->termpresentationorder == CUSTOMFIELD_SEMESTER_PRESENTATION_DESC) { $formattedoptions = array_reverse($formattedoptions, true); } // Build the field widget. $elementname = $this->get_form_element_name(); $mform->addElement('select', $elementname, $this->get_field()->get_formatted_name(), $formattedoptions); $mform->setDefault($elementname, $this->get_default_value()); // Add the required flag if necessary. if ($field->get_configdata_property('required')) { $mform->addRule($elementname, null, 'required', null, 'client'); } } /** * Validates data for this field. * * @param array $data * @param array $files * @return array */ public function instance_form_validation(array $data, array $files): array { $errors = parent::instance_form_validation($data, $files); if ($this->get_field()->get_configdata_property('required')) { // Standard required rule does not work on select element. $elementname = $this->get_form_element_name(); if (empty($data[$elementname])) { $errors[$elementname] = get_string('err_required', 'form'); } } return $errors; } /** * Returns value in a human-readable format * * @return mixed|null value or null if empty */ public function export_value() { $value = $this->get_value(); return self::get_name_for_semester($value); } /** * Returns the human readable Semester name for a semesterid. * * @param int $value the semesterid (YYYYS as described at the top of the file). * @return string|null The human readable semester name */ public static function get_name_for_semester(int $value) { if ($value === self::get_termindependent_representation()) { return get_string('semesterindependent', 'customfield_semester'); } else if ($value == null) { return null; } else { $year = intdiv($value, 10); $semester = $value % 10; if ($semester === self::get_summerterm_representation()) { return get_string('summersemester', 'customfield_semester', $year); } else if ($semester === self::get_winterterm_representation()) { return get_string('wintersemester', 'customfield_semester', $year . '/' . substr($year + 1, 2, 2)); } else { return null; } } } /** * returns a semesterid, given a datetime. * @param DateTime $datetime the datetime * @return int the corresponding semesterid */ public static function get_semester_for_datetime(DateTime $datetime): int { $year = (int) $datetime->format('Y'); $month = (int) $datetime->format('m'); $summertermstartmonth = self::get_summerterm_startmonth(); $wintertermstartmonth = self::get_winterterm_startmonth(); if ($month < $summertermstartmonth) { $year--; $semester = self::get_winterterm_representation(); } else if ($month < $wintertermstartmonth) { $semester = self::get_summerterm_representation(); } else { $semester = self::get_winterterm_representation(); } return $year * 10 + $semester; } /** * Returns the configured start month of the summer term from the plugin settings. * * @return int */ public static function get_summerterm_startmonth(): int { global $CFG; // Require local library. require_once($CFG->dirroot.'/customfield/field/semester/locallib.php'); // Get config from DB. $config = get_config('customfield_semester'); // Double-check that the value is within the acceptable range. If not, return the default value. if (is_numeric($config->summertermstartmonth) == false || $config->summertermstartmonth < 1 || $config->summertermstartmonth > 12 || $config->summertermstartmonth > $config->wintertermstartmonth) { return CUSTOMFIELD_SEMESTER_SUMMERTERMSTART; } return $config->summertermstartmonth; } /** * Returns the configured start month of the winter term from the plugin settings. * * @return int */ public static function get_winterterm_startmonth(): int { global $CFG; // Require local library. require_once($CFG->dirroot.'/customfield/field/semester/locallib.php'); // Get config from DB. $config = get_config('customfield_semester'); // Double-check that the value is within the acceptable range. If not, return the default value. if (is_numeric($config->wintertermstartmonth) == false || $config->wintertermstartmonth < 1 || $config->wintertermstartmonth > 12 || $config->wintertermstartmonth < $config->summertermstartmonth) { return CUSTOMFIELD_SEMESTER_WINTERTERMSTART; } return $config->wintertermstartmonth; } /** * Returns the configured internal representation for the summer term. * * @return int */ public static function get_summerterm_representation(): int { global $CFG; // Static variable to remember the return value for subsequent calls of this function. static $returnvalue = null; // If not already done in a previous call, calculate the return value. if ($returnvalue === null) { // Require local library. require_once($CFG->dirroot.'/customfield/field/semester/locallib.php'); // Get config from DB. $config = get_config('customfield_semester'); // If the setting was changed to something different from the default. if ($config->internaltermrepresentation == CUSTOMFIELD_SEMESTER_INTERNAL_ST1WT2) { $returnvalue = CUSTOMFIELD_SEMESTER_INTERNAL_ST1WT2_ST; // Otherwise, return the default representation. } else { $returnvalue = CUSTOMFIELD_SEMESTER_INTERNAL_ST0WT1_ST; } } // Return value. return $returnvalue; } /** * Returns the configured internal representation for the winter term. * * @return int */ public static function get_winterterm_representation(): int { global $CFG; // Static variable to remember the return value for subsequent calls of this function. static $returnvalue = null; // If not already done in a previous call, calculate the return value. if ($returnvalue === null) { // Require local library. require_once($CFG->dirroot.'/customfield/field/semester/locallib.php'); // Get config from DB. $config = get_config('customfield_semester'); // If the setting was changed to something different from the default. if ($config->internaltermrepresentation == CUSTOMFIELD_SEMESTER_INTERNAL_ST1WT2) { $returnvalue = CUSTOMFIELD_SEMESTER_INTERNAL_ST1WT2_WT; // Otherwise, return the default representation. } else { $returnvalue = CUSTOMFIELD_SEMESTER_INTERNAL_ST0WT1_WT; } } // Return value. return $returnvalue; } /** * Returns the configured internal representation for the term-independent term. * * @return int */ public static function get_termindependent_representation(): int { global $CFG; // Static variable to remember the return value for subsequent calls of this function. static $returnvalue = null; // If not already done in a previous call, calculate the return value. if ($returnvalue === null) { // Require local library. require_once($CFG->dirroot.'/customfield/field/semester/locallib.php'); // Return the default representation. $returnvalue = CUSTOMFIELD_SEMESTER_INTERNAL_TERMINDEPENDENT; } // Return value. return $returnvalue; } }