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

/**
 * Forum subscription manager.
 *
 * @package    mod_forum
 * @copyright  2014 Andrew Nicols <andrew@nicols.co.uk>
 * @license    http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
 */

namespace mod_forum;

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

/**
 * Forum subscription manager.
 *
 * @copyright  2014 Andrew Nicols <andrew@nicols.co.uk>
 * @license    http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
 */
class subscriptions {

    /**
     * The status value for an unsubscribed discussion.
     *
     * @var int
     */
    const FORUM_DISCUSSION_UNSUBSCRIBED = -1;

    /**
     * The subscription cache for forums.
     *
     * The first level key is the user ID
     * The second level is the forum ID
     * The Value then is bool for subscribed of not.
     *
     * @var array[] An array of arrays.
     */
    protected static $forumcache = array();

    /**
     * The list of forums which have been wholly retrieved for the forum subscription cache.
     *
     * This allows for prior caching of an entire forum to reduce the
     * number of DB queries in a subscription check loop.
     *
     * @var bool[]
     */
    protected static $fetchedforums = array();

    /**
     * The subscription cache for forum discussions.
     *
     * The first level key is the user ID
     * The second level is the forum ID
     * The third level key is the discussion ID
     * The value is then the users preference (int)
     *
     * @var array[]
     */
    protected static $forumdiscussioncache = array();

    /**
     * The list of forums which have been wholly retrieved for the forum discussion subscription cache.
     *
     * This allows for prior caching of an entire forum to reduce the
     * number of DB queries in a subscription check loop.
     *
     * @var bool[]
     */
    protected static $discussionfetchedforums = array();

    /**
     * Whether a user is subscribed to this forum, or a discussion within
     * the forum.
     *
     * If a discussion is specified, then report whether the user is
     * subscribed to posts to this particular discussion, taking into
     * account the forum preference.
     *
     * If it is not specified then only the forum preference is considered.
     *
     * @param int $userid The user ID
     * @param \stdClass $forum The record of the forum to test
     * @param int $discussionid The ID of the discussion to check
     * @param $cm The coursemodule record. If not supplied, this will be calculated using get_fast_modinfo instead.
     * @return boolean
     */
    public static function is_subscribed($userid, $forum, $discussionid = null, $cm = null) {
        // If forum is force subscribed and has allowforcesubscribe, then user is subscribed.
        if (self::is_forcesubscribed($forum)) {
            if (!$cm) {
                $cm = get_fast_modinfo($forum->course)->instances['forum'][$forum->id];
            }
            if (has_capability('mod/forum:allowforcesubscribe', \context_module::instance($cm->id), $userid)) {
                return true;
            }
        }

        if ($discussionid === null) {
            return self::is_subscribed_to_forum($userid, $forum);
        }

        $subscriptions = self::fetch_discussion_subscription($forum->id, $userid);

        // Check whether there is a record for this discussion subscription.
        if (isset($subscriptions[$discussionid])) {
            return ($subscriptions[$discussionid] != self::FORUM_DISCUSSION_UNSUBSCRIBED);
        }

        return self::is_subscribed_to_forum($userid, $forum);
    }

    /**
     * Whether a user is subscribed to this forum.
     *
     * @param int $userid The user ID
     * @param \stdClass $forum The record of the forum to test
     * @return boolean
     */
    protected static function is_subscribed_to_forum($userid, $forum) {
        return self::fetch_subscription_cache($forum->id, $userid);
    }

    /**
     * Helper to determine whether a forum has it's subscription mode set
     * to forced subscription.
     *
     * @param \stdClass $forum The record of the forum to test
     * @return bool
     */
    public static function is_forcesubscribed($forum) {
        return ($forum->forcesubscribe == FORUM_FORCESUBSCRIBE);
    }

    /**
     * Helper to determine whether a forum has it's subscription mode set to disabled.
     *
     * @param \stdClass $forum The record of the forum to test
     * @return bool
     */
    public static function subscription_disabled($forum) {
        return ($forum->forcesubscribe == FORUM_DISALLOWSUBSCRIBE);
    }

    /**
     * Helper to determine whether the specified forum can be subscribed to.
     *
     * @param \stdClass $forum The record of the forum to test
     * @return bool
     */
    public static function is_subscribable($forum) {
        return (isloggedin() && !isguestuser() &&
                !\mod_forum\subscriptions::is_forcesubscribed($forum) &&
                !\mod_forum\subscriptions::subscription_disabled($forum));
    }

    /**
     * Set the forum subscription mode.
     *
     * By default when called without options, this is set to FORUM_FORCESUBSCRIBE.
     *
     * @param \stdClass $forum The record of the forum to set
     * @param int $status The new subscription state
< * @return bool
> * @return bool true > * @throws dml_exception A DML specific exception is thrown for any errors.
*/
< public static function set_subscription_mode($forumid, $status = 1) {
> public static function set_subscription_mode($forum, $status = FORUM_FORCESUBSCRIBE): bool {
global $DB;
< return $DB->set_field("forum", "forcesubscribe", $status, array("id" => $forumid));
> > if (is_numeric($forum)) { > debugging(__METHOD__.': Argument #1 ($forum) must be a stdClass record of a forum', DEBUG_DEVELOPER); > > $forum = $DB->get_record("forum", ["id" => $forum], '*', MUST_EXIST); > } > > $DB->set_field("forum", "forcesubscribe", $status, ["id" => $forum->id]); > > if ($forum->forcesubscribe != $status) { > // Trigger event if subscription mode has been changed. > $event = \mod_forum\event\subscription_mode_updated::create([ > "context" => forum_get_context($forum->id), > "objectid" => $forum->id, > "other" => ["oldvalue" => $forum->forcesubscribe, "newvalue" => $status], > ]); > $event->add_record_snapshot("forum", $forum); > $event->trigger(); > } > > return true;
} /** * Returns the current subscription mode for the forum. * * @param \stdClass $forum The record of the forum to set * @return int The forum subscription mode */ public static function get_subscription_mode($forum) { return $forum->forcesubscribe; } /** * Returns an array of forums that the current user is subscribed to and is allowed to unsubscribe from * * @return array An array of unsubscribable forums */ public static function get_unsubscribable_forums() { global $USER, $DB; // Get courses that $USER is enrolled in and can see. $courses = enrol_get_my_courses(); if (empty($courses)) { return array(); } $courseids = array(); foreach($courses as $course) { $courseids[] = $course->id; } list($coursesql, $courseparams) = $DB->get_in_or_equal($courseids, SQL_PARAMS_NAMED, 'c'); // Get all forums from the user's courses that they are subscribed to and which are not set to forced. // It is possible for users to be subscribed to a forum in subscription disallowed mode so they must be listed // here so that that can be unsubscribed from. $sql = "SELECT f.id, cm.id as cm, cm.visible, f.course FROM {forum} f JOIN {course_modules} cm ON cm.instance = f.id JOIN {modules} m ON m.name = :modulename AND m.id = cm.module LEFT JOIN {forum_subscriptions} fs ON (fs.forum = f.id AND fs.userid = :userid) WHERE f.forcesubscribe <> :forcesubscribe AND fs.id IS NOT NULL AND cm.course $coursesql"; $params = array_merge($courseparams, array( 'modulename'=>'forum', 'userid' => $USER->id, 'forcesubscribe' => FORUM_FORCESUBSCRIBE, )); $forums = $DB->get_recordset_sql($sql, $params); $unsubscribableforums = array(); foreach($forums as $forum) { if (empty($forum->visible)) { // The forum is hidden - check if the user can view the forum. $context = \context_module::instance($forum->cm); if (!has_capability('moodle/course:viewhiddenactivities', $context)) { // The user can't see the hidden forum to cannot unsubscribe. continue; } } $unsubscribableforums[] = $forum; } $forums->close(); return $unsubscribableforums; } /** * Get the list of potential subscribers to a forum. * * @param context_module $context the forum context. * @param integer $groupid the id of a group, or 0 for all groups. * @param string $fields the list of fields to return for each user. As for get_users_by_capability. * @param string $sort sort order. As for get_users_by_capability. * @return array list of users. */ public static function get_potential_subscribers($context, $groupid, $fields, $sort = '') { global $DB; // Only active enrolled users or everybody on the frontpage. list($esql, $params) = get_enrolled_sql($context, 'mod/forum:allowforcesubscribe', $groupid, true); if (!$sort) { list($sort, $sortparams) = users_order_by_sql('u'); $params = array_merge($params, $sortparams); } $sql = "SELECT $fields FROM {user} u JOIN ($esql) je ON je.id = u.id WHERE u.auth <> 'nologin' AND u.suspended = 0 AND u.confirmed = 1 ORDER BY $sort"; return $DB->get_records_sql($sql, $params); } /** * Fetch the forum subscription data for the specified userid and forum. * * @param int $forumid The forum to retrieve a cache for * @param int $userid The user ID * @return boolean */ public static function fetch_subscription_cache($forumid, $userid) { if (isset(self::$forumcache[$userid]) && isset(self::$forumcache[$userid][$forumid])) { return self::$forumcache[$userid][$forumid]; } self::fill_subscription_cache($forumid, $userid); if (!isset(self::$forumcache[$userid]) || !isset(self::$forumcache[$userid][$forumid])) { return false; } return self::$forumcache[$userid][$forumid]; } /** * Fill the forum subscription data for the specified userid and forum. * * If the userid is not specified, then all subscription data for that forum is fetched in a single query and used * for subsequent lookups without requiring further database queries. * * @param int $forumid The forum to retrieve a cache for * @param int $userid The user ID * @return void */ public static function fill_subscription_cache($forumid, $userid = null) { global $DB; if (!isset(self::$fetchedforums[$forumid])) { // This forum has not been fetched as a whole. if (isset($userid)) { if (!isset(self::$forumcache[$userid])) { self::$forumcache[$userid] = array(); } if (!isset(self::$forumcache[$userid][$forumid])) { if ($DB->record_exists('forum_subscriptions', array( 'userid' => $userid, 'forum' => $forumid, ))) { self::$forumcache[$userid][$forumid] = true; } else { self::$forumcache[$userid][$forumid] = false; } } } else { $subscriptions = $DB->get_recordset('forum_subscriptions', array( 'forum' => $forumid, ), '', 'id, userid'); foreach ($subscriptions as $id => $data) { if (!isset(self::$forumcache[$data->userid])) { self::$forumcache[$data->userid] = array(); } self::$forumcache[$data->userid][$forumid] = true; } self::$fetchedforums[$forumid] = true; $subscriptions->close(); } } } /** * Fill the forum subscription data for all forums that the specified userid can subscribe to in the specified course. * * @param int $courseid The course to retrieve a cache for * @param int $userid The user ID * @return void */ public static function fill_subscription_cache_for_course($courseid, $userid) { global $DB; if (!isset(self::$forumcache[$userid])) { self::$forumcache[$userid] = array(); } $sql = "SELECT f.id AS forumid, s.id AS subscriptionid FROM {forum} f LEFT JOIN {forum_subscriptions} s ON (s.forum = f.id AND s.userid = :userid) WHERE f.course = :course AND f.forcesubscribe <> :subscriptionforced"; $subscriptions = $DB->get_recordset_sql($sql, array( 'course' => $courseid, 'userid' => $userid, 'subscriptionforced' => FORUM_FORCESUBSCRIBE, )); foreach ($subscriptions as $id => $data) { self::$forumcache[$userid][$id] = !empty($data->subscriptionid); } $subscriptions->close(); } /** * Returns a list of user objects who are subscribed to this forum. * * @param stdClass $forum The forum record. * @param int $groupid The group id if restricting subscriptions to a group of users, or 0 for all. * @param context_module $context the forum context, to save re-fetching it where possible. * @param string $fields requested user fields (with "u." table prefix). * @param boolean $includediscussionsubscriptions Whether to take discussion subscriptions and unsubscriptions into consideration. * @return array list of users. */ public static function fetch_subscribed_users($forum, $groupid = 0, $context = null, $fields = null, $includediscussionsubscriptions = false) { global $CFG, $DB; if (empty($fields)) { $userfieldsapi = \core_user\fields::for_name(); $allnames = $userfieldsapi->get_sql('u', false, '', '', false)->selects; $fields ="u.id, u.username, $allnames, u.maildisplay, u.mailformat, u.maildigest, u.imagealt, u.email, u.emailstop, u.city, u.country, u.lastaccess, u.lastlogin, u.picture, u.timezone, u.theme, u.lang, u.trackforums, u.mnethostid"; } // Retrieve the forum context if it wasn't specified. $context = forum_get_context($forum->id, $context); if (self::is_forcesubscribed($forum)) { $results = self::get_potential_subscribers($context, $groupid, $fields); } else { // Only active enrolled users or everybody on the frontpage. list($esql, $params) = get_enrolled_sql($context, '', $groupid, true); $params['forumid'] = $forum->id; list($sort, $sortparams) = users_order_by_sql('u'); $params = array_merge($params, $sortparams); if ($includediscussionsubscriptions) { $params['sforumid'] = $forum->id; $params['dsforumid'] = $forum->id; $params['unsubscribed'] = self::FORUM_DISCUSSION_UNSUBSCRIBED; $sql = "SELECT $fields FROM ( SELECT userid FROM {forum_subscriptions} s WHERE s.forum = :sforumid UNION SELECT userid FROM {forum_discussion_subs} ds WHERE ds.forum = :dsforumid AND ds.preference <> :unsubscribed ) subscriptions JOIN {user} u ON u.id = subscriptions.userid JOIN ($esql) je ON je.id = u.id WHERE u.auth <> 'nologin' AND u.suspended = 0 AND u.confirmed = 1 ORDER BY $sort"; } else { $sql = "SELECT $fields FROM {user} u JOIN ($esql) je ON je.id = u.id JOIN {forum_subscriptions} s ON s.userid = u.id WHERE s.forum = :forumid AND u.auth <> 'nologin' AND u.suspended = 0 AND u.confirmed = 1 ORDER BY $sort"; } $results = $DB->get_records_sql($sql, $params); } // Guest user should never be subscribed to a forum. unset($results[$CFG->siteguest]); // Apply the activity module availability resetrictions. $cm = get_coursemodule_from_instance('forum', $forum->id, $forum->course); $modinfo = get_fast_modinfo($forum->course); $info = new \core_availability\info_module($modinfo->get_cm($cm->id)); $results = $info->filter_user_list($results); return $results; } /** * Retrieve the discussion subscription data for the specified userid and forum. * * This is returned as an array of discussions for that forum which contain the preference in a stdClass. * * @param int $forumid The forum to retrieve a cache for * @param int $userid The user ID * @return array of stdClass objects with one per discussion in the forum. */ public static function fetch_discussion_subscription($forumid, $userid = null) { self::fill_discussion_subscription_cache($forumid, $userid); if (!isset(self::$forumdiscussioncache[$userid]) || !isset(self::$forumdiscussioncache[$userid][$forumid])) { return array(); } return self::$forumdiscussioncache[$userid][$forumid]; } /** * Fill the discussion subscription data for the specified userid and forum. * * If the userid is not specified, then all discussion subscription data for that forum is fetched in a single query * and used for subsequent lookups without requiring further database queries. * * @param int $forumid The forum to retrieve a cache for * @param int $userid The user ID * @return void */ public static function fill_discussion_subscription_cache($forumid, $userid = null) { global $DB; if (!isset(self::$discussionfetchedforums[$forumid])) { // This forum hasn't been fetched as a whole yet. if (isset($userid)) { if (!isset(self::$forumdiscussioncache[$userid])) { self::$forumdiscussioncache[$userid] = array(); } if (!isset(self::$forumdiscussioncache[$userid][$forumid])) { $subscriptions = $DB->get_recordset('forum_discussion_subs', array( 'userid' => $userid, 'forum' => $forumid, ), null, 'id, discussion, preference'); self::$forumdiscussioncache[$userid][$forumid] = array(); foreach ($subscriptions as $id => $data) { self::add_to_discussion_cache($forumid, $userid, $data->discussion, $data->preference); } $subscriptions->close(); } } else { $subscriptions = $DB->get_recordset('forum_discussion_subs', array( 'forum' => $forumid, ), null, 'id, userid, discussion, preference'); foreach ($subscriptions as $id => $data) { self::add_to_discussion_cache($forumid, $data->userid, $data->discussion, $data->preference); } self::$discussionfetchedforums[$forumid] = true; $subscriptions->close(); } } } /** * Add the specified discussion and user preference to the discussion * subscription cache. * * @param int $forumid The ID of the forum that this preference belongs to * @param int $userid The ID of the user that this preference belongs to * @param int $discussion The ID of the discussion that this preference relates to * @param int $preference The preference to store */ protected static function add_to_discussion_cache($forumid, $userid, $discussion, $preference) { if (!isset(self::$forumdiscussioncache[$userid])) { self::$forumdiscussioncache[$userid] = array(); } if (!isset(self::$forumdiscussioncache[$userid][$forumid])) { self::$forumdiscussioncache[$userid][$forumid] = array(); } self::$forumdiscussioncache[$userid][$forumid][$discussion] = $preference; } /** * Reset the discussion cache. * * This cache is used to reduce the number of database queries when * checking forum discussion subscription states. */ public static function reset_discussion_cache() { self::$forumdiscussioncache = array(); self::$discussionfetchedforums = array(); } /** * Reset the forum cache. * * This cache is used to reduce the number of database queries when * checking forum subscription states. */ public static function reset_forum_cache() { self::$forumcache = array(); self::$fetchedforums = array(); } /** * Adds user to the subscriber list. * * @param int $userid The ID of the user to subscribe * @param \stdClass $forum The forum record for this forum. * @param \context_module|null $context Module context, may be omitted if not known or if called for the current * module set in page. * @param boolean $userrequest Whether the user requested this change themselves. This has an effect on whether * discussion subscriptions are removed too. * @return bool|int Returns true if the user is already subscribed, or the forum_subscriptions ID if the user was * successfully subscribed. */ public static function subscribe_user($userid, $forum, $context = null, $userrequest = false) { global $DB; if (self::is_subscribed($userid, $forum)) { return true; } $sub = new \stdClass(); $sub->userid = $userid; $sub->forum = $forum->id; $result = $DB->insert_record("forum_subscriptions", $sub); if ($userrequest) { $discussionsubscriptions = $DB->get_recordset('forum_discussion_subs', array('userid' => $userid, 'forum' => $forum->id)); $DB->delete_records_select('forum_discussion_subs', 'userid = :userid AND forum = :forumid AND preference <> :preference', array( 'userid' => $userid, 'forumid' => $forum->id, 'preference' => self::FORUM_DISCUSSION_UNSUBSCRIBED, )); // Reset the subscription caches for this forum. // We know that the there were previously entries and there aren't any more. if (isset(self::$forumdiscussioncache[$userid]) && isset(self::$forumdiscussioncache[$userid][$forum->id])) { foreach (self::$forumdiscussioncache[$userid][$forum->id] as $discussionid => $preference) { if ($preference != self::FORUM_DISCUSSION_UNSUBSCRIBED) { unset(self::$forumdiscussioncache[$userid][$forum->id][$discussionid]); } } } } // Reset the cache for this forum. self::$forumcache[$userid][$forum->id] = true; $context = forum_get_context($forum->id, $context); $params = array( 'context' => $context, 'objectid' => $result, 'relateduserid' => $userid, 'other' => array('forumid' => $forum->id), ); $event = event\subscription_created::create($params); if ($userrequest && $discussionsubscriptions) { foreach ($discussionsubscriptions as $subscription) { $event->add_record_snapshot('forum_discussion_subs', $subscription); } $discussionsubscriptions->close(); } $event->trigger(); return $result; } /** * Removes user from the subscriber list * * @param int $userid The ID of the user to unsubscribe * @param \stdClass $forum The forum record for this forum. * @param \context_module|null $context Module context, may be omitted if not known or if called for the current * module set in page. * @param boolean $userrequest Whether the user requested this change themselves. This has an effect on whether * discussion subscriptions are removed too. * @return boolean Always returns true. */ public static function unsubscribe_user($userid, $forum, $context = null, $userrequest = false) { global $DB; $sqlparams = array( 'userid' => $userid, 'forum' => $forum->id, ); $DB->delete_records('forum_digests', $sqlparams); if ($forumsubscription = $DB->get_record('forum_subscriptions', $sqlparams)) { $DB->delete_records('forum_subscriptions', array('id' => $forumsubscription->id)); if ($userrequest) { $discussionsubscriptions = $DB->get_recordset('forum_discussion_subs', $sqlparams); $DB->delete_records('forum_discussion_subs', array('userid' => $userid, 'forum' => $forum->id, 'preference' => self::FORUM_DISCUSSION_UNSUBSCRIBED)); // We know that the there were previously entries and there aren't any more. if (isset(self::$forumdiscussioncache[$userid]) && isset(self::$forumdiscussioncache[$userid][$forum->id])) { self::$forumdiscussioncache[$userid][$forum->id] = array(); } } // Reset the cache for this forum. self::$forumcache[$userid][$forum->id] = false; $context = forum_get_context($forum->id, $context); $params = array( 'context' => $context, 'objectid' => $forumsubscription->id, 'relateduserid' => $userid, 'other' => array('forumid' => $forum->id), ); $event = event\subscription_deleted::create($params); $event->add_record_snapshot('forum_subscriptions', $forumsubscription); if ($userrequest && $discussionsubscriptions) { foreach ($discussionsubscriptions as $subscription) { $event->add_record_snapshot('forum_discussion_subs', $subscription); } $discussionsubscriptions->close(); } $event->trigger(); } return true; } /** * Subscribes the user to the specified discussion. * * @param int $userid The userid of the user being subscribed * @param \stdClass $discussion The discussion to subscribe to * @param \context_module|null $context Module context, may be omitted if not known or if called for the current * module set in page. * @return boolean Whether a change was made */ public static function subscribe_user_to_discussion($userid, $discussion, $context = null) { global $DB; // First check whether the user is subscribed to the discussion already. $subscription = $DB->get_record('forum_discussion_subs', array('userid' => $userid, 'discussion' => $discussion->id)); if ($subscription) { if ($subscription->preference != self::FORUM_DISCUSSION_UNSUBSCRIBED) { // The user is already subscribed to the discussion. Ignore. return false; } } // No discussion-level subscription. Check for a forum level subscription. if ($DB->record_exists('forum_subscriptions', array('userid' => $userid, 'forum' => $discussion->forum))) { if ($subscription && $subscription->preference == self::FORUM_DISCUSSION_UNSUBSCRIBED) { // The user is subscribed to the forum, but unsubscribed from the discussion, delete the discussion preference. $DB->delete_records('forum_discussion_subs', array('id' => $subscription->id)); unset(self::$forumdiscussioncache[$userid][$discussion->forum][$discussion->id]); } else { // The user is already subscribed to the forum. Ignore. return false; } } else { if ($subscription) { $subscription->preference = time(); $DB->update_record('forum_discussion_subs', $subscription); } else { $subscription = new \stdClass(); $subscription->userid = $userid; $subscription->forum = $discussion->forum; $subscription->discussion = $discussion->id; $subscription->preference = time(); $subscription->id = $DB->insert_record('forum_discussion_subs', $subscription); self::$forumdiscussioncache[$userid][$discussion->forum][$discussion->id] = $subscription->preference; } } $context = forum_get_context($discussion->forum, $context); $params = array( 'context' => $context, 'objectid' => $subscription->id, 'relateduserid' => $userid, 'other' => array( 'forumid' => $discussion->forum, 'discussion' => $discussion->id, ), ); $event = event\discussion_subscription_created::create($params); $event->trigger(); return true; } /** * Unsubscribes the user from the specified discussion. * * @param int $userid The userid of the user being unsubscribed * @param \stdClass $discussion The discussion to unsubscribe from * @param \context_module|null $context Module context, may be omitted if not known or if called for the current * module set in page. * @return boolean Whether a change was made */ public static function unsubscribe_user_from_discussion($userid, $discussion, $context = null) { global $DB; // First check whether the user's subscription preference for this discussion. $subscription = $DB->get_record('forum_discussion_subs', array('userid' => $userid, 'discussion' => $discussion->id)); if ($subscription) { if ($subscription->preference == self::FORUM_DISCUSSION_UNSUBSCRIBED) { // The user is already unsubscribed from the discussion. Ignore. return false; } } // No discussion-level preference. Check for a forum level subscription. if (!$DB->record_exists('forum_subscriptions', array('userid' => $userid, 'forum' => $discussion->forum))) { if ($subscription && $subscription->preference != self::FORUM_DISCUSSION_UNSUBSCRIBED) { // The user is not subscribed to the forum, but subscribed from the discussion, delete the discussion subscription. $DB->delete_records('forum_discussion_subs', array('id' => $subscription->id)); unset(self::$forumdiscussioncache[$userid][$discussion->forum][$discussion->id]); } else { // The user is not subscribed from the forum. Ignore. return false; } } else { if ($subscription) { $subscription->preference = self::FORUM_DISCUSSION_UNSUBSCRIBED; $DB->update_record('forum_discussion_subs', $subscription); } else { $subscription = new \stdClass(); $subscription->userid = $userid; $subscription->forum = $discussion->forum; $subscription->discussion = $discussion->id; $subscription->preference = self::FORUM_DISCUSSION_UNSUBSCRIBED; $subscription->id = $DB->insert_record('forum_discussion_subs', $subscription); } self::$forumdiscussioncache[$userid][$discussion->forum][$discussion->id] = $subscription->preference; } $context = forum_get_context($discussion->forum, $context); $params = array( 'context' => $context, 'objectid' => $subscription->id, 'relateduserid' => $userid, 'other' => array( 'forumid' => $discussion->forum, 'discussion' => $discussion->id, ), ); $event = event\discussion_subscription_deleted::create($params); $event->trigger(); return true; } /** * Gets the default subscription value for the logged in user. * * @param \stdClass $forum The forum record * @param \context $context The course context * @param \cm_info $cm cm_info * @param int|null $discussionid The discussion we are checking against * @return bool Default subscription * @throws coding_exception */ public static function get_user_default_subscription($forum, $context, $cm, ?int $discussionid) { global $USER; $manageactivities = has_capability('moodle/course:manageactivities', $context); if (\mod_forum\subscriptions::subscription_disabled($forum) && !$manageactivities) { // User does not have permission to subscribe to this discussion at all. $discussionsubscribe = false; } else if (\mod_forum\subscriptions::is_forcesubscribed($forum)) { // User does not have permission to unsubscribe from this discussion at all. $discussionsubscribe = true; } else { if (isset($discussionid) && self::is_subscribed($USER->id, $forum, $discussionid, $cm)) { // User is subscribed to the discussion - continue the subscription. $discussionsubscribe = true; } else if (!isset($discussionid) && \mod_forum\subscriptions::is_subscribed($USER->id, $forum, null, $cm)) { // Starting a new discussion, and the user is subscribed to the forum - subscribe to the discussion. $discussionsubscribe = true; } else { // User is not subscribed to either forum or discussion. Follow user preference. $discussionsubscribe = $USER->autosubscribe ?? false; } } return $discussionsubscribe; } }