508 lines
		
	
	
	
		
			20 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			508 lines
		
	
	
	
		
			20 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| // This file is part of the Studyplan plugin for Moodle
 | |
| //
 | |
| // 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 <https://www.gnu.org/licenses/>.
 | |
| 
 | |
| /**
 | |
|  * Moodle hook functions and some internally used functions
 | |
|  * @package    local_treestudyplan
 | |
|  * @copyright  2023 P.M. Kuipers
 | |
|  * @license    https://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
 | |
|  */
 | |
| 
 | |
|  defined('MOODLE_INTERNAL') || die();
 | |
|  require_once($CFG->dirroot.'/course/modlib.php');
 | |
| 
 | |
| use local_treestudyplan\local\helpers\webservicehelper;
 | |
| use local_treestudyplan\studyplan;
 | |
| use local_treestudyplan\studyplanpage;
 | |
| 
 | |
| /**
 | |
|  * Describe editor options
 | |
|  * @param context $context Context for options
 | |
|  * @return array Editor options
 | |
|  */
 | |
| function local_treestudyplan_unit_get_editor_options(context $context) {
 | |
|     global $CFG;
 | |
|     return ['subdirs' => 1,
 | |
|             'maxbytes' => $CFG->maxbytes,
 | |
|             'maxfiles' => -1,
 | |
|             'changeformat' => 1,
 | |
|             'context' => $context,
 | |
|             'noclean' => 1,
 | |
|             'trusttext' => 0];
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Create primary navigation links for studyplan if needed
 | |
|  */
 | |
| function local_treestudyplan_autofill_customusermenuitems() {
 | |
|     if (get_config("local_treestudyplan", "primary_nav_autofill")) {
 | |
|         $lang = current_language();
 | |
|         $navitems = [
 | |
|             "/local/treestudyplan/myreport.php" => ["included" => false, "strkey" => "link_myreport"],
 | |
|             "/local/treestudyplan/view-plan.php" => ["included" => false, "strkey" => "link_viewplan"],
 | |
|             "/local/treestudyplan/edit-plan.php" => ["included" => false, "strkey" => "link_editplan"],
 | |
|         ];
 | |
|         if (\get_config("local_treestudyplan", "enablecoach")) {
 | |
|             // Also include the coach role if enabled.
 | |
|             $navitems["/local/treestudyplan/coach.php"] = ["included" => false, "strkey" => "link_coach"];
 | |
|         }
 | |
| 
 | |
|         // Load the custom menu items from config.
 | |
|         $custommenuitems = get_config("core", "custommenuitems");
 | |
| 
 | |
|         // Scan through all the lines to see if it is a link to one of our nav items in the current language.
 | |
|         $lines = explode("\n", $custommenuitems);
 | |
| 
 | |
|         $links = array_keys($navitems);
 | |
|         foreach ($lines as $line) {
 | |
|             $parms = explode('|', $line);
 | |
|             if (count($parms) > 3) {
 | |
|                 $link = trim($parms[1]);
 | |
|                 if (trim($parms[3]) == $lang && in_array($link, $links)) {
 | |
|                     // Register the link as already included if it is found.
 | |
|                     $navitems[$link]["included"] = true;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         // List through all the links to see if we need to add one or more.
 | |
|         foreach ($navitems as $link => $details) {
 | |
|             if (!$details["included"]) {
 | |
|                 $line = implode("|", [
 | |
|                     get_string($details["strkey"], "local_treestudyplan"), // Menu text.
 | |
|                     $link, // Link.
 | |
|                     '', // Tooltip.
 | |
|                     $lang, // Language code.
 | |
|                     " #Automatically added by studyplan plugin. See setting 'primary_nav_autofill' to disable this",
 | |
|                 ]);
 | |
|                 $custommenuitems = trim($custommenuitems)."\n".$line;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         // Store the modified custom menu items.
 | |
|         set_config("custommenuitems", $custommenuitems);
 | |
|     }
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Hook to extend navigation
 | |
|  * @param global_navigation $navigation Navigation object
 | |
|  */
 | |
| function local_treestudyplan_extend_navigation(global_navigation $navigation) {
 | |
|     global $CFG, $PAGE, $COURSE, $USER, $DB;
 | |
| 
 | |
|     $systemcontext = context_system::instance();
 | |
| 
 | |
|     /* Moodle 4.0-4.2 do not yet support customizing the primary navigation bar (it is a planned feature though).
 | |
|        For now, go to theme settings and add the following into "Custom menu items".
 | |
|           [your name for my studyplan]|/local/treestudyplan/myreport.php.
 | |
|           [your name for studyplan viewing]|/local/treestudyplan/view-plan.php.
 | |
|           [your name for studyplan managing]|/local/treestudyplan/edit-plan.php.
 | |
|        For example:.
 | |
|           Mijn studieplan|/local/treestudyplan/myreport.php.
 | |
|           Studieplannen|/local/treestudyplan/view-plan.php.
 | |
|           Studieplannen beheren|/local/treestudyplan/edit-plan.php.
 | |
| 
 | |
|        Using some javascript magic we'll hide the links that are not accessible.
 | |
|        (Since the Output API does not easily support inline style tags, adding one through Javascript is easier, .
 | |
|         and not much more complex than loading a separate stylesheet for each link we want to hide).
 | |
|        We will add all the hrefs that should be hidden to this variable below.
 | |
|     */
 | |
| 
 | |
|     /*
 | |
|         In addition, the function local_treestudyplan_autofill_customusermenuitems() called below will
 | |
|         automatically generate the required lines if they are missing...
 | |
|     */
 | |
|     local_treestudyplan_autofill_customusermenuitems();
 | |
| 
 | |
|     $hideprimaryhrefs = [];
 | |
| 
 | |
|     if ($USER->id > 1) {
 | |
|         // Don't show if user is not logged in (id == 0) or is guest user (id == 1).
 | |
| 
 | |
|         $userstudyplans = studyplan::find_for_user($USER->id);
 | |
|         if (!empty($userstudyplans)) {
 | |
| 
 | |
|             // Create studyplan node.
 | |
|             $node = navigation_node::create(
 | |
|                 get_string("link_myreport", "local_treestudyplan"),
 | |
|                 new moodle_url($CFG->wwwroot . "/local/treestudyplan/myreport.php", []),
 | |
|                 global_navigation::TYPE_SYSTEM,
 | |
|                 null,
 | |
|                 "local_treestudyplan_myreport",
 | |
|                 new pix_icon("myreport", '', 'local_treestudyplan')
 | |
|                 );
 | |
|             $node->showinflatnavigation = true;
 | |
|             $node->showinsecondarynavigation = true;
 | |
| 
 | |
|             // Create invitenode node.
 | |
|             $invitenode = navigation_node::create(
 | |
|                 get_string("manage_invites", "local_treestudyplan"),
 | |
|                 new moodle_url($CFG->wwwroot . "/local/treestudyplan/invitations.php", []),
 | |
|                 global_navigation::TYPE_CUSTOM ,
 | |
|                 null,
 | |
|                 "local_treestudyplan_invitemgmt",
 | |
|                 new pix_icon("invitemgmt", '', 'local_treestudyplan')
 | |
|                 );
 | |
|             $invitenode->showinflatnavigation = false;
 | |
|             $node->add_node($invitenode);
 | |
| 
 | |
|             $navigation->add_node($node, 'mycourses');
 | |
|         } else {
 | |
|             $hideprimaryhrefs[] = "/local/treestudyplan/myreport.php";
 | |
|         }
 | |
|         if (has_capability('local/treestudyplan:viewuserreports', context_system::instance())
 | |
|             || webservicehelper::has_capability_in_any_category('local/treestudyplan:viewuserreports')) {
 | |
|             $node = navigation_node::create(
 | |
|                 get_string("link_viewplan", "local_treestudyplan"),
 | |
|                 new moodle_url($CFG->wwwroot . "/local/treestudyplan/view-plan.php", []),
 | |
|                 global_navigation::TYPE_SYSTEM ,
 | |
|                 null,
 | |
|                 "local_treestudyplan_viewplan",
 | |
|                 new pix_icon("viewplans", '', 'local_treestudyplan')
 | |
|                 );
 | |
|             $node->showinflatnavigation = true;
 | |
|             $node->showinsecondarynavigation = true;
 | |
|             $navigation->add_node($node, 'mycourses');
 | |
|         } else {
 | |
|             $hideprimaryhrefs[] = "/local/treestudyplan/view-plan.php";
 | |
|         }
 | |
|         if (has_capability('local/treestudyplan:editstudyplan', context_system::instance())
 | |
|               || webservicehelper::has_capability_in_any_category('local/treestudyplan:editstudyplan')
 | |
|            ) {
 | |
|             $node = navigation_node::create(
 | |
|                 get_string("link_editplan", "local_treestudyplan"),
 | |
|                 new moodle_url($CFG->wwwroot . "/local/treestudyplan/edit-plan.php", []),
 | |
|                 global_navigation::TYPE_SYSTEM ,
 | |
|                 null,
 | |
|                 "local_treestudyplan_editplan",
 | |
|                 new pix_icon("viewplans", '', 'local_treestudyplan')
 | |
|                 );
 | |
|             $node->showinflatnavigation = true;
 | |
|             $node->showinsecondarynavigation = true;
 | |
|             $navigation->add_node($node, 'mycourses');
 | |
|         } else {
 | |
|             $hideprimaryhrefs[] = "/local/treestudyplan/edit-plan.php";
 | |
|         }
 | |
| 
 | |
|         $coachsql = "SELECT COUNT('id') FROM {local_treestudyplan_coach} c
 | |
|                      INNER JOIN {local_treestudyplan} t ON c.studyplan_id = t.id
 | |
|                      WHERE c.user_id = :user_id";
 | |
| 
 | |
|         if (\get_config("local_treestudyplan", "enablecoach") &&
 | |
|              (has_capability('local/treestudyplan:coach', context_system::instance())
 | |
|              || webservicehelper::has_capability_in_any_category('local/treestudyplan:coach')
 | |
|              ) && $DB->count_records_sql($coachsql, ["user_id" => $USER->id]) > 0
 | |
|             ) {
 | |
|             $node = navigation_node::create(
 | |
|                 get_string("link_coach", "local_treestudyplan"),
 | |
|                 new moodle_url($CFG->wwwroot . "/local/treestudyplan/coach.php", []),
 | |
|                 global_navigation::TYPE_SYSTEM ,
 | |
|                 null,
 | |
|                 "local_treestudyplan_coach",
 | |
|                 new pix_icon("viewplans", '', 'local_treestudyplan')
 | |
|                 );
 | |
|             $node->showinflatnavigation = true;
 | |
|             $node->showinsecondarynavigation = true;
 | |
|             $navigation->add_node($node, 'mycourses');
 | |
|         } else {
 | |
|             $hideprimaryhrefs[] = "/local/treestudyplan/coach.php";
 | |
|         }
 | |
|     } else {
 | |
|         $hideprimaryhrefs[] = "/local/treestudyplan/myreport.php";
 | |
|         $hideprimaryhrefs[] = "/local/treestudyplan/edit-plan.php";
 | |
|         $hideprimaryhrefs[] = "/local/treestudyplan/view-plan.php";
 | |
|         $hideprimaryhrefs[] = "/local/treestudyplan/coach.php";
 | |
|     }
 | |
|     // Create invitenode node.
 | |
|     $invitenode = navigation_node::create(
 | |
|         get_string("nav_invited", "local_treestudyplan"),
 | |
|         new moodle_url($CFG->wwwroot . "/local/treestudyplan/invited.php", []),
 | |
|         global_navigation::TYPE_USER ,
 | |
|         null,
 | |
|         "local_treestudyplan_invitemgmt",
 | |
|         new pix_icon("nav_invited", '', 'local_treestudyplan')
 | |
|         );
 | |
|     $invitenode->showinflatnavigation = false;
 | |
|     $navigation->add_node($invitenode, 'mycourses');
 | |
| 
 | |
|     // Now using some javascript magic, we'll hide the links that are not accessible.
 | |
|     $PAGE->requires->js_call_amd('local_treestudyplan/primary-nav-tools', 'hidePrimary', [$hideprimaryhrefs]);
 | |
| 
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Hook to extend navigation in category view
 | |
|  * @param mixed $navigation
 | |
|  * @param context_coursecat $coursecategorycontext
 | |
|  */
 | |
| function local_treestudyplan_extend_navigation_category_settings($navigation, context_coursecat $coursecategorycontext) {
 | |
|     global $CFG, $PAGE;
 | |
|     $categoryid = $coursecategorycontext->instanceid;
 | |
|     if (has_capability('local/treestudyplan:editstudyplan', $coursecategorycontext)) {
 | |
|         $node = $navigation->add(
 | |
|             get_string('treestudyplan:editstudyplan', "local_treestudyplan"),
 | |
|             new moodle_url($CFG->wwwroot . "/local/treestudyplan/edit-plan.php", ["categoryid" => $categoryid]),
 | |
|             global_navigation::TYPE_CATEGORY,
 | |
|             null,
 | |
|             "local_treestudyplan_editplan",
 | |
|             new pix_icon("editplans", '', 'local_treestudyplan')
 | |
|         );
 | |
| 
 | |
|     }
 | |
|     if (has_capability('local/treestudyplan:viewuserreports', $coursecategorycontext)) {
 | |
|         $node = $navigation->add(
 | |
|             get_string('link_viewplan', "local_treestudyplan"),
 | |
|             new moodle_url($CFG->wwwroot . "/local/treestudyplan/view-plan.php", ["categoryid" => $categoryid]),
 | |
|             global_navigation::TYPE_CATEGORY,
 | |
|             null,
 | |
|             "local_treestudyplan_viewplan",
 | |
|             new pix_icon("viewplans", '', 'local_treestudyplan')
 | |
|         );
 | |
| 
 | |
|     }
 | |
| 
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Map fontawesome icons for use in flat navigation
 | |
|  * @return array Icon mapping
 | |
|  *
 | |
|  */
 | |
| function local_treestudyplan_get_fontawesome_icon_map() {
 | |
| 
 | |
|     // Create the icon map with the icons which are used in any case.
 | |
|     $iconmapping = [
 | |
|             'local_treestudyplan:myreport' => 'fa-vcard',
 | |
|             'local_treestudyplan:editplans' => 'fa-share-alt',
 | |
|             'local_treestudyplan:viewplans' => 'fa-share-alt',
 | |
|     ];
 | |
|     return $iconmapping;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Helper function to reset the icon system used as updatecallback function when saving some of the plugin's settings.
 | |
|  */
 | |
| function local_treestudyplan_reset_fontawesome_icon_map() {
 | |
|     // Reset the icon system cache.
 | |
|     // There is the function \core\output\icon_system::reset_caches() which does seem to be only usable in unit tests.
 | |
|     // Thus, we clear the icon system cache brutally.
 | |
|     $cache = \cache::make('core', 'fontawesomeiconmapping');
 | |
|     $cache->delete('mapping');
 | |
|     // And rebuild it brutally.
 | |
|     $instance = \core\output\icon_system::instance(\core\output\icon_system::FONTAWESOME);
 | |
|     $instance->get_icon_name_map();
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Send invitation to invited person
 | |
|  * @param mixed $inviteid Database id of the invitation
 | |
|  *
 | |
|  */
 | |
| function local_treestudyplan_send_invite($inviteid) {
 | |
|     global $DB, $USER, $CFG;
 | |
|     $invite = $DB->get_record("local_treestudyplan_invit", ['id' => $inviteid]);
 | |
| 
 | |
|     $noreply = 'noreply@' . get_host_from_url($CFG->wwwroot);
 | |
|     $mailer = get_mailer();
 | |
|     if ($mailer != null ) {
 | |
|         $mailer->setFrom($noreply, "{$USER->firstname} {$USER->lastname}");
 | |
|         $mailer->addAddress($invite->email, $invite->name);
 | |
|         $mailer->addReplyTo($USER->email, "{$USER->firstname} {$USER->lastname}");
 | |
| 
 | |
|         $invitehref = $CFG->wwwroot."/local/treestudyplan/invited.php?key={$invite->invitekey}";
 | |
| 
 | |
|         $data = [    'permissions' => '',
 | |
|                     'invitee' => $invite->name,
 | |
|                     'sender' => "{$USER->firstname} {$USER->lastname}",
 | |
|                     'link' => $invitehref];
 | |
| 
 | |
|         if ($invite->allow_details || $invite->allow_calendar || $invite->allow_badges) {
 | |
|             $data['permissions'] = get_string('invite_mail_permissions', 'local_treestudyplan');
 | |
|             $data['permissions'] .= "<ul>\n";
 | |
|             if ($invite->allow_details) {
 | |
|                 $data['permissions'] .= "<li>".get_string('invite_allow_details', 'local_treestudyplan')."</li>\n";
 | |
|             }
 | |
|             if ($invite->allow_calendar) {
 | |
|                 $data['permissions'] .= "<li>".get_string('invite_allow_calendar', 'local_treestudyplan')."</li>\n";
 | |
|             }
 | |
|             if ($invite->allow_badges) {
 | |
|                 $data['permissions'] .= "<li>".get_string('invite_allow_badges', 'local_treestudyplan')."</li>\n";
 | |
|             }
 | |
| 
 | |
|             $data['permissions'] .= "</ul></p>\n";
 | |
|         }
 | |
| 
 | |
|         $body = get_string('invite_mail_text', 'local_treestudyplan', $data);
 | |
|         $subject = get_string('invite_mail_subject', 'local_treestudyplan', $data);
 | |
| 
 | |
|         $html = "
 | |
|         <!DOCTYPE html PUBLIC '-//W3C//DTD XHTML 1.0 Transitional//EN' 'http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd'>.
 | |
|         <html xmlns='http://www.w3.org/1999/xhtml'>.
 | |
|         <head>
 | |
|         <meta http-equiv='Content-Type' content='text/html; charset=UTF-8' />
 | |
|         <title>{$subject}</title>
 | |
|         <meta name='viewport' content='width=device-width, initial-scale=1.0'/>
 | |
|         </head>
 | |
|         <body>
 | |
|         {$body}
 | |
|         </body>
 | |
|         </html>";
 | |
| 
 | |
|         $mailer->isHTML(true);
 | |
|         $mailer->Subject = $subject;
 | |
|         $mailer->Body = $html;
 | |
|         $mailer->AltBody = strip_tags($body);
 | |
| 
 | |
|         $mailer->send();
 | |
|     }
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Hook to display fragment of activity/mod settings editor. Used in feature to edit name and description of activity
 | |
|  * @param mixed $args
 | |
|  * @return string Rendered form output HTML
 | |
|  */
 | |
| function local_treestudyplan_output_fragment_mod_edit_form($args) {
 | |
|     global $CFG;
 | |
|     global $DB;
 | |
|     $args = (object)$args;
 | |
|     $context = $args->context;
 | |
| 
 | |
|     if (empty($args->cmid)) {
 | |
|         return "RANDOM!";
 | |
|     }
 | |
| 
 | |
|     // Check the course module exists.
 | |
|     $cm = \get_coursemodule_from_id('', $args->cmid, 0, false, MUST_EXIST);
 | |
| 
 | |
|     // Check the course exists.
 | |
|     $course = \get_course($cm->course);
 | |
| 
 | |
|     // Require_login.
 | |
|     require_login($course, false, $cm); // Needed to setup proper $COURSE.
 | |
| 
 | |
|     list($cm, $context, $module, $data, $cw) = \get_moduleinfo_data($cm, $course);
 | |
| 
 | |
|     $modmoodleform = "$CFG->dirroot/mod/$module->name/mod_form.php";
 | |
|     if (file_exists($modmoodleform)) {
 | |
|         require_once($modmoodleform);
 | |
|     } else {
 | |
|         throw new \moodle_exception('noformdesc', 'local_treestudyplan');;
 | |
|     }
 | |
| 
 | |
|     $mformclassname = 'mod_'.$module->name.'_mod_form';
 | |
|     $mform = new $mformclassname($data, $cw->section, $cm, $course);
 | |
|     $mform->set_data($data);
 | |
| 
 | |
|     return $mform->render();
 | |
| 
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Serve the files from the myplugin file areas.
 | |
|  *
 | |
|  * @param stdClass $course the course object
 | |
|  * @param stdClass $cm the course module object
 | |
|  * @param stdClass $context the context
 | |
|  * @param string $filearea the name of the file area
 | |
|  * @param array $args extra arguments (itemid, path)
 | |
|  * @param bool $forcedownload whether or not force download
 | |
|  * @param array $options additional options affecting the file serving
 | |
|  * @return bool false if the file not found, just send the file otherwise and do not return anything
 | |
|  */
 | |
| function local_treestudyplan_pluginfile(
 | |
|     $course,
 | |
|     $cm,
 | |
|     $context,
 | |
|     string $filearea,
 | |
|     array $args,
 | |
|     bool $forcedownload,
 | |
|     array $options = []
 | |
| ): bool {
 | |
|     global $DB, $USER;
 | |
| 
 | |
|     $studyplanfilecaps = ["local/treestudyplan:editstudyplan", "local/treestudyplan:viewuserreports"];
 | |
| 
 | |
|     // Check the contextlevel is as expected - the studyplan plugin only uses system context for storing files.
 | |
|     // This avoids headaches when moving studyplans between contexts, while the security impact is minimal...
 | |
|     if ($context->contextlevel != CONTEXT_SYSTEM) {
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     // Make sure the filearea is one of those used by the plugin.
 | |
|     if (in_array($filearea, ["studyplan", "icon", "studyplanpage"])) {
 | |
|         // The args is an array containing [itemid, path].
 | |
|         // Fetch the itemid from the path.
 | |
|         $itemid = array_shift($args);
 | |
| 
 | |
|         // Studyplan icons and description images are not secret, so don't overdo it on access control...
 | |
|         if (true) {
 | |
|             // Extract the filename / filepath from the $args array.
 | |
|             $filename = array_pop($args); // The last item in the $args array.
 | |
|             if (empty($args)) {
 | |
|                 // Var $args is empty => the path is '/'.
 | |
|                 $filepath = '/';
 | |
|             } else {
 | |
|                 // Var $args contains the remaining elements of the filepath.
 | |
|                 $filepath = '/' . implode('/', $args) . '/';
 | |
|             }
 | |
| 
 | |
|             // Retrieve the file from the Files API.
 | |
|             $fs = get_file_storage();
 | |
|             $file = $fs->get_file(\context_system::instance()->id, 'local_treestudyplan', $filearea, $itemid, $filepath, $filename);
 | |
|             if (!$file) {
 | |
|                 // The file does not exist.
 | |
|                 return false;
 | |
|             }
 | |
|             // We can now send the file back to the browser - in this case with a cache lifetime of 1 day and no filtering.
 | |
|             send_stored_file($file, 24 * 60 * 60, 0, $forcedownload, $options);
 | |
|             return true;
 | |
|         } else {
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|     } else if (in_array($filearea, ['defaulticon'])) {
 | |
|         // The args is an array containing [itemid, path].
 | |
|         // Fetch the itemid from the path.
 | |
|         $itemid = array_shift($args);
 | |
| 
 | |
|         // Extract the filename / filepath from the $args array.
 | |
|         $filename = array_pop($args); // The last item in the $args array.
 | |
|         if (empty($args)) {
 | |
|             // Var $args is empty => the path is '/'.
 | |
|             $filepath = '/';
 | |
|         } else {
 | |
|             // Var $args contains the remaining elements of the filepath.
 | |
|             $filepath = '/' . implode('/', $args) . '/';
 | |
|         }
 | |
| 
 | |
|         // Retrieve the file from the Files API.
 | |
|         $fs = get_file_storage();
 | |
|         $file = $fs->get_file(\context_system::instance()->id, 'local_treestudyplan', $filearea, $itemid, $filepath, $filename);
 | |
|         if (!$file) {
 | |
|             // The file does not exist.
 | |
|             return false;
 | |
|         }
 | |
|         // We can now send the file back to the browser - in this case with a cache lifetime of 1 day and no filtering.
 | |
|         send_stored_file($file, 24 * 60 * 60, 0, $forcedownload, $options);
 | |
|         return true;
 | |
|     } else {
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
| }
 | |
| 
 | 
