Proyectos de Subversion Moodle

Rev

Autoría | Ultima modificación | Ver Log |

<?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/>.

namespace core_courseformat;

use action_menu;
use cm_info;
use core_courseformat\base as course_format;
use core_courseformat\formatactions;
use core_courseformat\output\local\content\section\controlmenu;
use core_courseformat\stateupdates;
use renderer_base;
use section_info;
use stdClass;

/**
 * Class sectiondelegatemodule
 *
 * @package    core_courseformat
 * @copyright  2024 Mikel Martín <mikel@moodle.com>
 * @license    http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
 */
abstract class sectiondelegatemodule extends sectiondelegate {
    /** @var section_info $sectioninfo The section_info object of the delegated section module */

    /** @var cm_info|null $cm The cm_info object of the delegated section module */
    private $cm = null;

    /** @var stdClass|null $course The course object of the delegated section module */
    private $course = null;

    /**
     * Constructor.
     * @param section_info $sectioninfo
     */
    public function __construct(
        protected section_info $sectioninfo
    ) {
        parent::__construct($sectioninfo);
        /** @var \course_modinfo $modinfo */
        $modinfo = $sectioninfo->modinfo;
        try {
            // Disabled or missing plugins can throw exceptions.
            $this->cm = $modinfo->get_instance_of(
                $this->get_module_name(),
                $this->sectioninfo->itemid,
            );
            $this->course = $modinfo->get_course();
        } catch (\Exception $e) {
            $this->cm = null;
            $this->course = null;
        }
    }

    /**
     * Check if the delegated component is enabled.
     *
     * @return bool
     */
    public function is_enabled(): bool {
        return $this->cm !== null;
    }

    /**
     * Get the delegated section id controlled by a specific cm.
     *
     * This method is used when reverse search is needed bu we cannot access the database.
     * This happens mostly on backup and restore. Do NOT use for normal operations.
     *
     * @param stdClass|cm_info $cm a course module compatible data structure.
     * @return int the section id.
     */
    public static function delegated_section_id(stdClass|cm_info $cm): int {
        global $DB;
        return $DB->get_field(
            'course_sections',
            'id',
            [
                'course' => $cm->course,
                'component' => explode('\\', static::class)[0],
                'itemid' => $cm->instance,
            ],
            MUST_EXIST
        );
    }

    /**
     * Get the parent section of the current delegated section.
     *
     * @return section_info|null
     */
    public function get_parent_section(): ?section_info {
        return $this->cm->get_section_info();
    }

    /**
     * Get the course object.
     *
     * @return cm_info
     */
    public function get_cm(): cm_info {
        return $this->cm;
    }

    /**
     * Get the course object.
     *
     * @return stdClass
     */
    public function get_course(): stdClass {
        return $this->course;
    }

    /**
     * Get the module name from the section component frankenstyle name.
     *
     * @return string
     */
    private function get_module_name(): string {
        return \core_component::normalize_component($this->sectioninfo->component)[1];
    }

    /**
     * Sync the section renaming with the activity name.
     *
     * @param section_info $section
     * @param string|null $newname
     * @return string|null
     */
    public function preprocess_section_name(section_info $section, ?string $newname): ?string {
        $cm = get_coursemodule_from_instance($this->get_module_name(), $section->itemid);
        if (!$cm) {
            return $newname;
        }
        if (empty($newname) || $newname === $cm->name) {
            return $cm->name;
        }
        formatactions::cm($section->course)->rename($cm->id, $newname);
        return $newname;
    }

    /**
     * Allow delegate plugin to modify the available section menu.
     *
     * @param course_format $format The course format instance.
     * @param controlmenu $controlmenu The control menu instance.
     * @param renderer_base $output The renderer instance.
     * @return action_menu|null The new action menu with the list of edit control items or null if no action menu is available.
     */
    public function get_section_action_menu(
        course_format $format,
        controlmenu $controlmenu,
        renderer_base $output,
    ): ?action_menu {
        $controlmenuclass = $format->get_output_classname('content\\cm\\delegatedcontrolmenu');
        $controlmenu = new $controlmenuclass(
            $format,
            $this->sectioninfo,
            $this->cm,
        );
        return $controlmenu->get_action_menu($output);
    }

    /**
     * Add extra state updates when put or create a section.
     *
     * @param section_info $section the affected section.
     * @param stateupdates $updates the state updates object to notify the UI.
     */
    public function put_section_state_extra_updates(section_info $section, stateupdates $updates): void {
        $cm = get_coursemodule_from_instance($this->get_module_name(), $section->itemid);
        $updates->add_cm_put($cm->id);
    }

    public function section_updated(stdClass $sectionrecord): void {
        global $DB;

        $cmrecord = [];
        if (isset($sectionrecord->availability) && $sectionrecord->availability !== $this->cm->availability) {
            $cmrecord['availability'] = $sectionrecord->availability;
        }

        if (isset($sectionrecord->visible) && $sectionrecord->visible !== $this->cm->visible) {
            $cmrecord['visible'] = $sectionrecord->visible;
            $cmrecord['visibleold'] = $sectionrecord->visible;
        }

        if (empty($cmrecord)) {
            return;
        }

        $cmrecord['id'] = $this->cm->id;
        $DB->update_record('course_modules', (object)$cmrecord);
    }
}