2007-01-04 13:15:04 +13:00
|
|
|
<?php
|
2012-01-05 09:29:55 +08:00
|
|
|
// 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/>.
|
|
|
|
|
|
|
|
|
2007-01-04 13:15:04 +13:00
|
|
|
/**
|
2007-08-15 23:51:07 +00:00
|
|
|
* Extra library for groups and groupings.
|
2007-01-04 13:15:04 +13:00
|
|
|
*
|
2012-01-05 09:29:55 +08:00
|
|
|
* @copyright 2006 The Open University, J.White AT open.ac.uk, Petr Skoda (skodak)
|
|
|
|
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
|
|
|
|
* @package core_group
|
2007-01-04 13:15:04 +13:00
|
|
|
*/
|
|
|
|
|
2007-08-15 23:51:07 +00:00
|
|
|
/*
|
|
|
|
* INTERNAL FUNCTIONS - to be used by moodle core only
|
|
|
|
* require_once $CFG->dirroot.'/group/lib.php' must be used
|
|
|
|
*/
|
|
|
|
|
2007-08-16 21:14:03 +00:00
|
|
|
/**
|
|
|
|
* Adds a specified user to a group
|
2012-01-05 09:29:55 +08:00
|
|
|
*
|
|
|
|
* @param mixed $grouporid The group id or group object
|
|
|
|
* @param mixed $userorid The user id or user object
|
2012-03-20 17:41:33 +00:00
|
|
|
* @param string $component Optional component name e.g. 'enrol_imsenterprise'
|
|
|
|
* @param int $itemid Optional itemid associated with component
|
2012-01-05 09:29:55 +08:00
|
|
|
* @return bool True if user added successfully or the user is already a
|
2007-11-19 20:31:57 +00:00
|
|
|
* member of the group, false otherwise.
|
2007-08-16 21:14:03 +00:00
|
|
|
*/
|
2012-03-20 17:41:33 +00:00
|
|
|
function groups_add_member($grouporid, $userorid, $component=null, $itemid=0) {
|
2008-06-01 13:09:04 +00:00
|
|
|
global $DB;
|
|
|
|
|
2009-09-14 23:52:08 +00:00
|
|
|
if (is_object($userorid)) {
|
|
|
|
$userid = $userorid->id;
|
|
|
|
$user = $userorid;
|
2012-09-26 17:13:42 +02:00
|
|
|
if (!isset($user->deleted)) {
|
|
|
|
$user = $DB->get_record('user', array('id'=>$userid), '*', MUST_EXIST);
|
|
|
|
}
|
2009-09-14 23:52:08 +00:00
|
|
|
} else {
|
|
|
|
$userid = $userorid;
|
2009-09-16 10:08:37 +00:00
|
|
|
$user = $DB->get_record('user', array('id'=>$userid), '*', MUST_EXIST);
|
2009-03-09 02:19:38 +00:00
|
|
|
}
|
|
|
|
|
2012-09-26 17:13:42 +02:00
|
|
|
if ($user->deleted) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2009-09-14 23:52:08 +00:00
|
|
|
if (is_object($grouporid)) {
|
|
|
|
$groupid = $grouporid->id;
|
|
|
|
$group = $grouporid;
|
|
|
|
} else {
|
|
|
|
$groupid = $grouporid;
|
|
|
|
$group = $DB->get_record('groups', array('id'=>$groupid), '*', MUST_EXIST);
|
2007-08-16 21:14:03 +00:00
|
|
|
}
|
|
|
|
|
2013-08-19 16:17:49 +08:00
|
|
|
// Check if the user a participant of the group course.
|
|
|
|
$context = context_course::instance($group->courseid);
|
|
|
|
if (!is_enrolled($context, $userid)) {
|
2009-09-14 23:52:08 +00:00
|
|
|
return false;
|
2009-03-13 06:51:30 +00:00
|
|
|
}
|
|
|
|
|
2007-08-16 21:14:03 +00:00
|
|
|
if (groups_is_member($groupid, $userid)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2010-09-21 08:16:49 +00:00
|
|
|
$member = new stdClass();
|
2007-08-16 21:14:03 +00:00
|
|
|
$member->groupid = $groupid;
|
|
|
|
$member->userid = $userid;
|
|
|
|
$member->timeadded = time();
|
2012-03-20 17:41:33 +00:00
|
|
|
$member->component = '';
|
|
|
|
$member->itemid = 0;
|
|
|
|
|
|
|
|
// Check the component exists if specified
|
|
|
|
if (!empty($component)) {
|
2013-07-16 22:42:37 +02:00
|
|
|
$dir = core_component::get_component_directory($component);
|
2012-03-20 17:41:33 +00:00
|
|
|
if ($dir && is_dir($dir)) {
|
|
|
|
// Component exists and can be used
|
|
|
|
$member->component = $component;
|
|
|
|
$member->itemid = $itemid;
|
|
|
|
} else {
|
|
|
|
throw new coding_exception('Invalid call to groups_add_member(). An invalid component was specified');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($itemid !== 0 && empty($member->component)) {
|
|
|
|
// An itemid can only be specified if a valid component was found
|
|
|
|
throw new coding_exception('Invalid call to groups_add_member(). A component must be specified if an itemid is given');
|
|
|
|
}
|
2007-08-16 21:14:03 +00:00
|
|
|
|
2009-02-17 16:44:48 +00:00
|
|
|
$DB->insert_record('groups_members', $member);
|
2007-08-16 21:14:03 +00:00
|
|
|
|
2013-08-19 16:17:49 +08:00
|
|
|
// Update group info, and group object.
|
2008-06-01 13:09:04 +00:00
|
|
|
$DB->set_field('groups', 'timemodified', $member->timeadded, array('id'=>$groupid));
|
2013-08-19 16:17:49 +08:00
|
|
|
$group->timemodified = $member->timeadded;
|
|
|
|
|
2017-11-23 23:15:27 +08:00
|
|
|
// Invalidate the group and grouping cache for users.
|
|
|
|
cache_helper::invalidate_by_definition('core', 'user_group_groupings', array(), array($userid));
|
|
|
|
|
2018-10-26 10:30:52 +08:00
|
|
|
// Group conversation messaging.
|
|
|
|
if ($conversation = \core_message\api::get_conversation_by_area('core_group', 'groups', $groupid, $context->id)) {
|
|
|
|
\core_message\api::add_members_to_conversation([$userid], $conversation->id);
|
|
|
|
}
|
|
|
|
|
2013-08-19 16:17:49 +08:00
|
|
|
// Trigger group event.
|
|
|
|
$params = array(
|
|
|
|
'context' => $context,
|
|
|
|
'objectid' => $groupid,
|
|
|
|
'relateduserid' => $userid,
|
|
|
|
'other' => array(
|
|
|
|
'component' => $member->component,
|
|
|
|
'itemid' => $member->itemid
|
|
|
|
)
|
|
|
|
);
|
|
|
|
$event = \core\event\group_member_added::create($params);
|
|
|
|
$event->add_record_snapshot('groups', $group);
|
|
|
|
$event->trigger();
|
2007-08-16 21:14:03 +00:00
|
|
|
|
2024-01-04 12:46:55 +11:00
|
|
|
// Dispatch the hook for a user added to the group.
|
|
|
|
$hook = new \core_group\hook\after_group_membership_added(
|
|
|
|
groupinstance: $group,
|
|
|
|
userids: [$userid],
|
|
|
|
);
|
|
|
|
\core\di::get(\core\hook\manager::class)->dispatch($hook);
|
|
|
|
|
2007-08-16 21:14:03 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2012-03-20 17:41:33 +00:00
|
|
|
/**
|
|
|
|
* Checks whether the current user is permitted (using the normal UI) to
|
|
|
|
* remove a specific group member, assuming that they have access to remove
|
|
|
|
* group members in general.
|
|
|
|
*
|
|
|
|
* For automatically-created group member entries, this checks with the
|
|
|
|
* relevant plugin to see whether it is permitted. The default, if the plugin
|
|
|
|
* doesn't provide a function, is true.
|
|
|
|
*
|
|
|
|
* For other entries (and any which have already been deleted/don't exist) it
|
|
|
|
* just returns true.
|
|
|
|
*
|
|
|
|
* @param mixed $grouporid The group id or group object
|
|
|
|
* @param mixed $userorid The user id or user object
|
|
|
|
* @return bool True if permitted, false otherwise
|
|
|
|
*/
|
|
|
|
function groups_remove_member_allowed($grouporid, $userorid) {
|
|
|
|
global $DB;
|
|
|
|
|
|
|
|
if (is_object($userorid)) {
|
|
|
|
$userid = $userorid->id;
|
|
|
|
} else {
|
|
|
|
$userid = $userorid;
|
|
|
|
}
|
|
|
|
if (is_object($grouporid)) {
|
|
|
|
$groupid = $grouporid->id;
|
|
|
|
} else {
|
|
|
|
$groupid = $grouporid;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get entry
|
|
|
|
if (!($entry = $DB->get_record('groups_members',
|
|
|
|
array('groupid' => $groupid, 'userid' => $userid), '*', IGNORE_MISSING))) {
|
|
|
|
// If the entry does not exist, they are allowed to remove it (this
|
|
|
|
// is consistent with groups_remove_member below).
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the entry does not have a component value, they can remove it
|
|
|
|
if (empty($entry->component)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// It has a component value, so we need to call a plugin function (if it
|
|
|
|
// exists); the default is to allow removal
|
|
|
|
return component_callback($entry->component, 'allow_group_member_remove',
|
|
|
|
array($entry->itemid, $entry->groupid, $entry->userid), true);
|
|
|
|
}
|
|
|
|
|
2007-08-16 21:14:03 +00:00
|
|
|
/**
|
|
|
|
* Deletes the link between the specified user and group.
|
2012-01-05 09:29:55 +08:00
|
|
|
*
|
|
|
|
* @param mixed $grouporid The group id or group object
|
|
|
|
* @param mixed $userorid The user id or user object
|
|
|
|
* @return bool True if deletion was successful, false otherwise
|
2007-08-16 21:14:03 +00:00
|
|
|
*/
|
2009-09-14 23:52:08 +00:00
|
|
|
function groups_remove_member($grouporid, $userorid) {
|
2008-06-01 13:09:04 +00:00
|
|
|
global $DB;
|
|
|
|
|
2009-09-14 23:52:08 +00:00
|
|
|
if (is_object($userorid)) {
|
|
|
|
$userid = $userorid->id;
|
|
|
|
} else {
|
|
|
|
$userid = $userorid;
|
2009-03-09 02:19:38 +00:00
|
|
|
}
|
|
|
|
|
2009-09-14 23:52:08 +00:00
|
|
|
if (is_object($grouporid)) {
|
|
|
|
$groupid = $grouporid->id;
|
|
|
|
$group = $grouporid;
|
|
|
|
} else {
|
|
|
|
$groupid = $grouporid;
|
|
|
|
$group = $DB->get_record('groups', array('id'=>$groupid), '*', MUST_EXIST);
|
2007-08-16 21:14:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!groups_is_member($groupid, $userid)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2009-02-17 16:44:48 +00:00
|
|
|
$DB->delete_records('groups_members', array('groupid'=>$groupid, 'userid'=>$userid));
|
|
|
|
|
2013-08-19 16:17:49 +08:00
|
|
|
// Update group info.
|
|
|
|
$time = time();
|
|
|
|
$DB->set_field('groups', 'timemodified', $time, array('id' => $groupid));
|
|
|
|
$group->timemodified = $time;
|
|
|
|
|
2017-11-23 23:15:27 +08:00
|
|
|
// Invalidate the group and grouping cache for users.
|
|
|
|
cache_helper::invalidate_by_definition('core', 'user_group_groupings', array(), array($userid));
|
|
|
|
|
2018-10-26 10:30:52 +08:00
|
|
|
// Group conversation messaging.
|
|
|
|
$context = context_course::instance($group->courseid);
|
|
|
|
if ($conversation = \core_message\api::get_conversation_by_area('core_group', 'groups', $groupid, $context->id)) {
|
|
|
|
\core_message\api::remove_members_from_conversation([$userid], $conversation->id);
|
|
|
|
}
|
|
|
|
|
2013-08-19 16:17:49 +08:00
|
|
|
// Trigger group event.
|
|
|
|
$params = array(
|
|
|
|
'context' => context_course::instance($group->courseid),
|
|
|
|
'objectid' => $groupid,
|
|
|
|
'relateduserid' => $userid
|
|
|
|
);
|
|
|
|
$event = \core\event\group_member_removed::create($params);
|
|
|
|
$event->add_record_snapshot('groups', $group);
|
|
|
|
$event->trigger();
|
2008-07-06 17:57:06 +00:00
|
|
|
|
2024-01-04 12:46:55 +11:00
|
|
|
// Dispatch the hook for a user removed from the group.
|
|
|
|
$hook = new \core_group\hook\after_group_membership_removed(
|
|
|
|
groupinstance: $group,
|
|
|
|
userids: [$userid],
|
|
|
|
);
|
|
|
|
\core\di::get(\core\hook\manager::class)->dispatch($hook);
|
|
|
|
|
2007-08-16 21:14:03 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2007-08-15 23:51:07 +00:00
|
|
|
/**
|
|
|
|
* Add a new group
|
2012-01-05 09:29:55 +08:00
|
|
|
*
|
|
|
|
* @param stdClass $data group properties
|
|
|
|
* @param stdClass $editform
|
|
|
|
* @param array $editoroptions
|
2023-02-28 13:13:42 +00:00
|
|
|
* @return int id of group or throws an exception on error
|
|
|
|
* @throws moodle_exception
|
2007-08-15 23:51:07 +00:00
|
|
|
*/
|
2010-07-11 13:07:24 +00:00
|
|
|
function groups_create_group($data, $editform = false, $editoroptions = false) {
|
2018-10-22 22:40:40 +02:00
|
|
|
global $CFG, $DB, $USER;
|
2009-11-01 16:48:45 +00:00
|
|
|
|
2009-03-09 02:19:38 +00:00
|
|
|
//check that courseid exists
|
2013-08-21 13:42:30 +08:00
|
|
|
$course = $DB->get_record('course', array('id' => $data->courseid), '*', MUST_EXIST);
|
2012-08-02 11:20:48 +08:00
|
|
|
$context = context_course::instance($course->id);
|
2007-08-15 23:51:07 +00:00
|
|
|
|
2008-06-01 13:09:04 +00:00
|
|
|
$data->timecreated = time();
|
2007-08-15 23:51:07 +00:00
|
|
|
$data->timemodified = $data->timecreated;
|
2008-06-01 13:09:04 +00:00
|
|
|
$data->name = trim($data->name);
|
2012-03-12 16:34:45 +00:00
|
|
|
if (isset($data->idnumber)) {
|
|
|
|
$data->idnumber = trim($data->idnumber);
|
|
|
|
if (groups_get_group_by_idnumber($course->id, $data->idnumber)) {
|
|
|
|
throw new moodle_exception('idnumbertaken');
|
|
|
|
}
|
|
|
|
}
|
2009-11-04 06:14:06 +00:00
|
|
|
|
MDL-68093 groups: Add visibility and participation settings
These new settings are designed to enchance user privacy surrounding
groups. They allow groups to be configured so that users outside the
group cannot see the group, so that users in the group cannot see each
other, or so that users cannot see the group at all, even if they are in
it. This avoids issues where a group may be assigned based on sensitive
personal information (such as a person requiring special arrangements
due to a disability).
By default, groups are visible to all and available for participation in
activities, which maintains the current behaviour.
For performance, a new cache has been added to track the number of
groups on a course that are not visible to non-members. This allows us
to revert to the existing behaviour if the new features are not being
used at all on a course, and only apply the new visibility conditions if
they are.
Users who have the moodle/course:viewhiddengroups capability should be
concious of exposing hidden groups when showing their screen to other
users. The "Switch role to..." feature can be used to show a course page
on screen without exposing private availability conditions, for example.
The changes cover several specific areas:
* grouplib functions, which most code should use to get lists of groups
and members (this includes the participants page).
* Activities supporting group overrides will not allow overrides for
groups that are hidden from all users.
* Activities supporting separate/visible groups modes will only allow
groups with the new "participation" flag enabled to be selected.
* Group messaging will be disabled for groups where members cannot see
each other, or cannot see the group at all.
2022-09-06 09:14:24 +01:00
|
|
|
$data->visibility ??= GROUPS_VISIBILITY_ALL;
|
|
|
|
|
|
|
|
if (!in_array($data->visibility, [GROUPS_VISIBILITY_ALL, GROUPS_VISIBILITY_MEMBERS])) {
|
|
|
|
$data->participation = false;
|
|
|
|
$data->enablemessaging = false;
|
|
|
|
}
|
|
|
|
|
2010-07-11 13:07:24 +00:00
|
|
|
if ($editform and $editoroptions) {
|
2009-11-04 06:14:06 +00:00
|
|
|
$data->description = $data->description_editor['text'];
|
|
|
|
$data->descriptionformat = $data->description_editor['format'];
|
|
|
|
}
|
|
|
|
|
2010-07-11 13:07:24 +00:00
|
|
|
$data->id = $DB->insert_record('groups', $data);
|
|
|
|
|
2023-08-09 11:02:49 +10:00
|
|
|
$handler = \core_group\customfield\group_handler::create();
|
|
|
|
$handler->instance_form_save($data, true);
|
|
|
|
|
2010-07-11 13:07:24 +00:00
|
|
|
if ($editform and $editoroptions) {
|
|
|
|
// Update description from editor with fixed files
|
|
|
|
$data = file_postupdate_standard_editor($data, 'description', $editoroptions, $context, 'group', 'description', $data->id);
|
2010-09-21 08:16:49 +00:00
|
|
|
$upd = new stdClass();
|
2010-07-11 13:07:24 +00:00
|
|
|
$upd->id = $data->id;
|
|
|
|
$upd->description = $data->description;
|
|
|
|
$upd->descriptionformat = $data->descriptionformat;
|
|
|
|
$DB->update_record('groups', $upd);
|
|
|
|
}
|
|
|
|
|
|
|
|
$group = $DB->get_record('groups', array('id'=>$data->id));
|
2007-08-15 23:51:07 +00:00
|
|
|
|
2009-02-17 16:44:48 +00:00
|
|
|
if ($editform) {
|
2010-07-11 13:07:24 +00:00
|
|
|
groups_update_group_icon($group, $data, $editform);
|
2007-08-15 23:51:07 +00:00
|
|
|
}
|
|
|
|
|
2012-12-21 09:43:31 +13:00
|
|
|
// Invalidate the grouping cache for the course
|
|
|
|
cache_helper::invalidate_by_definition('core', 'groupdata', array(), array($course->id));
|
MDL-68093 groups: Add visibility and participation settings
These new settings are designed to enchance user privacy surrounding
groups. They allow groups to be configured so that users outside the
group cannot see the group, so that users in the group cannot see each
other, or so that users cannot see the group at all, even if they are in
it. This avoids issues where a group may be assigned based on sensitive
personal information (such as a person requiring special arrangements
due to a disability).
By default, groups are visible to all and available for participation in
activities, which maintains the current behaviour.
For performance, a new cache has been added to track the number of
groups on a course that are not visible to non-members. This allows us
to revert to the existing behaviour if the new features are not being
used at all on a course, and only apply the new visibility conditions if
they are.
Users who have the moodle/course:viewhiddengroups capability should be
concious of exposing hidden groups when showing their screen to other
users. The "Switch role to..." feature can be used to show a course page
on screen without exposing private availability conditions, for example.
The changes cover several specific areas:
* grouplib functions, which most code should use to get lists of groups
and members (this includes the participants page).
* Activities supporting group overrides will not allow overrides for
groups that are hidden from all users.
* Activities supporting separate/visible groups modes will only allow
groups with the new "participation" flag enabled to be selected.
* Group messaging will be disabled for groups where members cannot see
each other, or cannot see the group at all.
2022-09-06 09:14:24 +01:00
|
|
|
// Rebuild the coursehiddengroups cache for the course.
|
|
|
|
\core_group\visibility::update_hiddengroups_cache($course->id);
|
2012-12-21 09:43:31 +13:00
|
|
|
|
2018-10-22 22:40:40 +02:00
|
|
|
// Group conversation messaging.
|
|
|
|
if (\core_message\api::can_create_group_conversation($USER->id, $context)) {
|
|
|
|
if (!empty($data->enablemessaging)) {
|
2018-10-25 12:10:29 +08:00
|
|
|
\core_message\api::create_conversation(
|
|
|
|
\core_message\api::MESSAGE_CONVERSATION_TYPE_GROUP,
|
|
|
|
[],
|
|
|
|
$group->name,
|
|
|
|
\core_message\api::MESSAGE_CONVERSATION_ENABLED,
|
|
|
|
'core_group',
|
|
|
|
'groups',
|
|
|
|
$group->id,
|
|
|
|
$context->id);
|
2018-10-22 22:40:40 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-19 16:17:49 +08:00
|
|
|
// Trigger group event.
|
|
|
|
$params = array(
|
|
|
|
'context' => $context,
|
|
|
|
'objectid' => $group->id
|
|
|
|
);
|
|
|
|
$event = \core\event\group_created::create($params);
|
|
|
|
$event->add_record_snapshot('groups', $group);
|
|
|
|
$event->trigger();
|
2009-02-17 16:44:48 +00:00
|
|
|
|
2024-01-04 12:46:55 +11:00
|
|
|
// Dispatch the hook for post group creation actions.
|
|
|
|
$hook = new \core_group\hook\after_group_created(
|
|
|
|
groupinstance: $group,
|
|
|
|
);
|
|
|
|
\core\di::get(\core\hook\manager::class)->dispatch($hook);
|
|
|
|
|
2010-07-11 13:07:24 +00:00
|
|
|
return $group->id;
|
2007-08-15 23:51:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2007-11-19 20:31:57 +00:00
|
|
|
* Add a new grouping
|
2012-01-05 09:29:55 +08:00
|
|
|
*
|
|
|
|
* @param stdClass $data grouping properties
|
|
|
|
* @param array $editoroptions
|
2023-02-28 13:13:42 +00:00
|
|
|
* @return int id of grouping or throws an exception on error
|
|
|
|
* @throws moodle_exception
|
2007-11-19 20:31:57 +00:00
|
|
|
*/
|
2009-11-04 06:14:06 +00:00
|
|
|
function groups_create_grouping($data, $editoroptions=null) {
|
2008-06-01 13:09:04 +00:00
|
|
|
global $DB;
|
2007-11-19 20:31:57 +00:00
|
|
|
|
2008-06-01 13:09:04 +00:00
|
|
|
$data->timecreated = time();
|
2007-11-19 20:31:57 +00:00
|
|
|
$data->timemodified = $data->timecreated;
|
2008-06-01 13:09:04 +00:00
|
|
|
$data->name = trim($data->name);
|
2012-03-12 16:34:45 +00:00
|
|
|
if (isset($data->idnumber)) {
|
|
|
|
$data->idnumber = trim($data->idnumber);
|
|
|
|
if (groups_get_grouping_by_idnumber($data->courseid, $data->idnumber)) {
|
|
|
|
throw new moodle_exception('idnumbertaken');
|
|
|
|
}
|
|
|
|
}
|
2009-11-04 06:14:06 +00:00
|
|
|
|
|
|
|
if ($editoroptions !== null) {
|
|
|
|
$data->description = $data->description_editor['text'];
|
|
|
|
$data->descriptionformat = $data->description_editor['format'];
|
|
|
|
}
|
|
|
|
|
2008-07-06 17:57:06 +00:00
|
|
|
$id = $DB->insert_record('groupings', $data);
|
2009-02-17 16:44:48 +00:00
|
|
|
$data->id = $id;
|
2009-11-04 06:14:06 +00:00
|
|
|
|
2023-08-09 11:02:49 +10:00
|
|
|
$handler = \core_group\customfield\grouping_handler::create();
|
|
|
|
$handler->instance_form_save($data, true);
|
|
|
|
|
2009-11-04 06:14:06 +00:00
|
|
|
if ($editoroptions !== null) {
|
|
|
|
$description = new stdClass;
|
|
|
|
$description->id = $data->id;
|
|
|
|
$description->description_editor = $data->description_editor;
|
2010-07-03 13:37:13 +00:00
|
|
|
$description = file_postupdate_standard_editor($description, 'description', $editoroptions, $editoroptions['context'], 'grouping', 'description', $description->id);
|
2009-11-04 06:14:06 +00:00
|
|
|
$DB->update_record('groupings', $description);
|
|
|
|
}
|
|
|
|
|
2012-12-21 09:43:31 +13:00
|
|
|
// Invalidate the grouping cache for the course
|
|
|
|
cache_helper::invalidate_by_definition('core', 'groupdata', array(), array($data->courseid));
|
|
|
|
|
2013-08-19 16:17:49 +08:00
|
|
|
// Trigger group event.
|
|
|
|
$params = array(
|
|
|
|
'context' => context_course::instance($data->courseid),
|
|
|
|
'objectid' => $id
|
|
|
|
);
|
|
|
|
$event = \core\event\grouping_created::create($params);
|
|
|
|
$event->trigger();
|
2008-07-06 17:57:06 +00:00
|
|
|
|
|
|
|
return $id;
|
2007-11-19 20:31:57 +00:00
|
|
|
}
|
|
|
|
|
2010-07-11 13:07:24 +00:00
|
|
|
/**
|
|
|
|
* Update the group icon from form data
|
2012-01-05 09:29:55 +08:00
|
|
|
*
|
|
|
|
* @param stdClass $group group information
|
|
|
|
* @param stdClass $data
|
|
|
|
* @param stdClass $editform
|
2010-07-11 13:07:24 +00:00
|
|
|
*/
|
|
|
|
function groups_update_group_icon($group, $data, $editform) {
|
|
|
|
global $CFG, $DB;
|
|
|
|
require_once("$CFG->libdir/gdlib.php");
|
|
|
|
|
|
|
|
$fs = get_file_storage();
|
2012-08-02 11:20:48 +08:00
|
|
|
$context = context_course::instance($group->courseid, MUST_EXIST);
|
2014-11-06 15:39:12 +08:00
|
|
|
$newpicture = $group->picture;
|
2010-07-11 13:07:24 +00:00
|
|
|
|
2014-11-06 15:39:12 +08:00
|
|
|
if (!empty($data->deletepicture)) {
|
|
|
|
$fs->delete_area_files($context->id, 'group', 'icon', $group->id);
|
|
|
|
$newpicture = 0;
|
|
|
|
} else if ($iconfile = $editform->save_temp_file('imagefile')) {
|
2014-11-06 15:23:58 +08:00
|
|
|
if ($rev = process_new_icon($context, 'group', 'icon', $group->id, $iconfile)) {
|
2014-11-06 15:39:12 +08:00
|
|
|
$newpicture = $rev;
|
2013-03-22 16:51:18 +01:00
|
|
|
} else {
|
|
|
|
$fs->delete_area_files($context->id, 'group', 'icon', $group->id);
|
2014-11-06 15:39:12 +08:00
|
|
|
$newpicture = 0;
|
2010-07-11 13:07:24 +00:00
|
|
|
}
|
2013-03-22 16:51:18 +01:00
|
|
|
@unlink($iconfile);
|
2014-11-06 15:39:12 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if ($newpicture != $group->picture) {
|
|
|
|
$DB->set_field('groups', 'picture', $newpicture, array('id' => $group->id));
|
|
|
|
$group->picture = $newpicture;
|
|
|
|
|
2013-03-22 16:51:18 +01:00
|
|
|
// Invalidate the group data as we've updated the group record.
|
|
|
|
cache_helper::invalidate_by_definition('core', 'groupdata', array(), array($group->courseid));
|
2010-07-11 13:07:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-11-19 20:31:57 +00:00
|
|
|
/**
|
|
|
|
* Update group
|
2012-01-05 09:29:55 +08:00
|
|
|
*
|
|
|
|
* @param stdClass $data group properties (with magic quotes)
|
|
|
|
* @param stdClass $editform
|
2010-07-11 13:07:24 +00:00
|
|
|
* @param array $editoroptions
|
2012-01-05 09:29:55 +08:00
|
|
|
* @return bool true or exception
|
2007-08-15 23:51:07 +00:00
|
|
|
*/
|
2010-07-11 13:07:24 +00:00
|
|
|
function groups_update_group($data, $editform = false, $editoroptions = false) {
|
2018-10-22 22:40:40 +02:00
|
|
|
global $CFG, $DB, $USER;
|
2010-07-11 13:07:24 +00:00
|
|
|
|
2012-08-02 11:20:48 +08:00
|
|
|
$context = context_course::instance($data->courseid);
|
2007-08-15 23:51:07 +00:00
|
|
|
|
|
|
|
$data->timemodified = time();
|
2014-08-12 17:25:38 +01:00
|
|
|
if (isset($data->name)) {
|
|
|
|
$data->name = trim($data->name);
|
|
|
|
}
|
2012-03-12 16:34:45 +00:00
|
|
|
if (isset($data->idnumber)) {
|
|
|
|
$data->idnumber = trim($data->idnumber);
|
|
|
|
if (($existing = groups_get_group_by_idnumber($data->courseid, $data->idnumber)) && $existing->id != $data->id) {
|
|
|
|
throw new moodle_exception('idnumbertaken');
|
|
|
|
}
|
|
|
|
}
|
MDL-68093 groups: Add visibility and participation settings
These new settings are designed to enchance user privacy surrounding
groups. They allow groups to be configured so that users outside the
group cannot see the group, so that users in the group cannot see each
other, or so that users cannot see the group at all, even if they are in
it. This avoids issues where a group may be assigned based on sensitive
personal information (such as a person requiring special arrangements
due to a disability).
By default, groups are visible to all and available for participation in
activities, which maintains the current behaviour.
For performance, a new cache has been added to track the number of
groups on a course that are not visible to non-members. This allows us
to revert to the existing behaviour if the new features are not being
used at all on a course, and only apply the new visibility conditions if
they are.
Users who have the moodle/course:viewhiddengroups capability should be
concious of exposing hidden groups when showing their screen to other
users. The "Switch role to..." feature can be used to show a course page
on screen without exposing private availability conditions, for example.
The changes cover several specific areas:
* grouplib functions, which most code should use to get lists of groups
and members (this includes the participants page).
* Activities supporting group overrides will not allow overrides for
groups that are hidden from all users.
* Activities supporting separate/visible groups modes will only allow
groups with the new "participation" flag enabled to be selected.
* Group messaging will be disabled for groups where members cannot see
each other, or cannot see the group at all.
2022-09-06 09:14:24 +01:00
|
|
|
if (isset($data->visibility) && !in_array($data->visibility, [GROUPS_VISIBILITY_ALL, GROUPS_VISIBILITY_MEMBERS])) {
|
|
|
|
$data->participation = false;
|
|
|
|
$data->enablemessaging = false;
|
|
|
|
}
|
2009-11-04 06:14:06 +00:00
|
|
|
|
2010-07-11 13:07:24 +00:00
|
|
|
if ($editform and $editoroptions) {
|
|
|
|
$data = file_postupdate_standard_editor($data, 'description', $editoroptions, $context, 'group', 'description', $data->id);
|
2009-11-04 06:14:06 +00:00
|
|
|
}
|
|
|
|
|
2009-02-17 16:44:48 +00:00
|
|
|
$DB->update_record('groups', $data);
|
2008-07-06 17:57:06 +00:00
|
|
|
|
2023-08-09 11:02:49 +10:00
|
|
|
$handler = \core_group\customfield\group_handler::create();
|
|
|
|
$handler->instance_form_save($data);
|
|
|
|
|
2012-12-21 09:43:31 +13:00
|
|
|
// Invalidate the group data.
|
|
|
|
cache_helper::invalidate_by_definition('core', 'groupdata', array(), array($data->courseid));
|
MDL-68093 groups: Add visibility and participation settings
These new settings are designed to enchance user privacy surrounding
groups. They allow groups to be configured so that users outside the
group cannot see the group, so that users in the group cannot see each
other, or so that users cannot see the group at all, even if they are in
it. This avoids issues where a group may be assigned based on sensitive
personal information (such as a person requiring special arrangements
due to a disability).
By default, groups are visible to all and available for participation in
activities, which maintains the current behaviour.
For performance, a new cache has been added to track the number of
groups on a course that are not visible to non-members. This allows us
to revert to the existing behaviour if the new features are not being
used at all on a course, and only apply the new visibility conditions if
they are.
Users who have the moodle/course:viewhiddengroups capability should be
concious of exposing hidden groups when showing their screen to other
users. The "Switch role to..." feature can be used to show a course page
on screen without exposing private availability conditions, for example.
The changes cover several specific areas:
* grouplib functions, which most code should use to get lists of groups
and members (this includes the participants page).
* Activities supporting group overrides will not allow overrides for
groups that are hidden from all users.
* Activities supporting separate/visible groups modes will only allow
groups with the new "participation" flag enabled to be selected.
* Group messaging will be disabled for groups where members cannot see
each other, or cannot see the group at all.
2022-09-06 09:14:24 +01:00
|
|
|
// Rebuild the coursehiddengroups cache for the course.
|
|
|
|
\core_group\visibility::update_hiddengroups_cache($data->courseid);
|
2012-12-21 09:43:31 +13:00
|
|
|
|
2010-07-11 13:07:24 +00:00
|
|
|
$group = $DB->get_record('groups', array('id'=>$data->id));
|
|
|
|
|
2009-02-17 16:44:48 +00:00
|
|
|
if ($editform) {
|
2010-07-11 13:07:24 +00:00
|
|
|
groups_update_group_icon($group, $data, $editform);
|
2007-08-15 23:51:07 +00:00
|
|
|
}
|
|
|
|
|
2018-10-22 22:40:40 +02:00
|
|
|
// Group conversation messaging.
|
|
|
|
if (\core_message\api::can_create_group_conversation($USER->id, $context)) {
|
2018-10-25 12:10:29 +08:00
|
|
|
if ($conversation = \core_message\api::get_conversation_by_area('core_group', 'groups', $group->id, $context->id)) {
|
|
|
|
if ($data->enablemessaging && $data->enablemessaging != $conversation->enabled) {
|
|
|
|
\core_message\api::enable_conversation($conversation->id);
|
2018-10-22 22:40:40 +02:00
|
|
|
}
|
2018-10-25 12:10:29 +08:00
|
|
|
if (!$data->enablemessaging && $data->enablemessaging != $conversation->enabled) {
|
|
|
|
\core_message\api::disable_conversation($conversation->id);
|
2018-10-22 22:40:40 +02:00
|
|
|
}
|
2018-10-25 12:10:29 +08:00
|
|
|
\core_message\api::update_conversation_name($conversation->id, $group->name);
|
2018-10-22 22:40:40 +02:00
|
|
|
} else {
|
|
|
|
if (!empty($data->enablemessaging)) {
|
2018-10-26 10:30:52 +08:00
|
|
|
$conversation = \core_message\api::create_conversation(
|
2018-10-25 12:10:29 +08:00
|
|
|
\core_message\api::MESSAGE_CONVERSATION_TYPE_GROUP,
|
|
|
|
[],
|
|
|
|
$group->name,
|
|
|
|
\core_message\api::MESSAGE_CONVERSATION_ENABLED,
|
|
|
|
'core_group',
|
|
|
|
'groups',
|
|
|
|
$group->id,
|
|
|
|
$context->id
|
|
|
|
);
|
2018-10-26 10:30:52 +08:00
|
|
|
|
|
|
|
// Add members to conversation if they exists in the group.
|
|
|
|
if ($groupmemberroles = groups_get_members_by_role($group->id, $group->courseid, 'u.id')) {
|
|
|
|
$users = [];
|
|
|
|
foreach ($groupmemberroles as $roleid => $roledata) {
|
|
|
|
foreach ($roledata->users as $member) {
|
|
|
|
$users[] = $member->id;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
\core_message\api::add_members_to_conversation($users, $conversation->id);
|
|
|
|
}
|
2018-10-22 22:40:40 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-19 16:17:49 +08:00
|
|
|
// Trigger group event.
|
|
|
|
$params = array(
|
|
|
|
'context' => $context,
|
|
|
|
'objectid' => $group->id
|
|
|
|
);
|
|
|
|
$event = \core\event\group_updated::create($params);
|
|
|
|
$event->add_record_snapshot('groups', $group);
|
|
|
|
$event->trigger();
|
2009-02-17 16:44:48 +00:00
|
|
|
|
2024-01-04 12:46:55 +11:00
|
|
|
// Dispatch the hook for post group update actions.
|
|
|
|
$hook = new \core_group\hook\after_group_updated(
|
|
|
|
groupinstance: $group,
|
|
|
|
);
|
|
|
|
\core\di::get(\core\hook\manager::class)->dispatch($hook);
|
|
|
|
|
2009-02-17 16:44:48 +00:00
|
|
|
return true;
|
2007-08-15 23:51:07 +00:00
|
|
|
}
|
|
|
|
|
2007-11-19 20:31:57 +00:00
|
|
|
/**
|
|
|
|
* Update grouping
|
2012-01-05 09:29:55 +08:00
|
|
|
*
|
|
|
|
* @param stdClass $data grouping properties (with magic quotes)
|
|
|
|
* @param array $editoroptions
|
|
|
|
* @return bool true or exception
|
2007-11-19 20:31:57 +00:00
|
|
|
*/
|
2009-11-04 06:14:06 +00:00
|
|
|
function groups_update_grouping($data, $editoroptions=null) {
|
2008-06-01 13:09:04 +00:00
|
|
|
global $DB;
|
2007-11-19 20:31:57 +00:00
|
|
|
$data->timemodified = time();
|
2014-08-12 17:25:38 +01:00
|
|
|
if (isset($data->name)) {
|
|
|
|
$data->name = trim($data->name);
|
|
|
|
}
|
2012-03-12 16:34:45 +00:00
|
|
|
if (isset($data->idnumber)) {
|
|
|
|
$data->idnumber = trim($data->idnumber);
|
|
|
|
if (($existing = groups_get_grouping_by_idnumber($data->courseid, $data->idnumber)) && $existing->id != $data->id) {
|
|
|
|
throw new moodle_exception('idnumbertaken');
|
|
|
|
}
|
|
|
|
}
|
2009-11-04 06:14:06 +00:00
|
|
|
if ($editoroptions !== null) {
|
2010-07-03 13:37:13 +00:00
|
|
|
$data = file_postupdate_standard_editor($data, 'description', $editoroptions, $editoroptions['context'], 'grouping', 'description', $data->id);
|
2009-11-04 06:14:06 +00:00
|
|
|
}
|
2009-02-17 16:44:48 +00:00
|
|
|
$DB->update_record('groupings', $data);
|
2012-12-21 09:43:31 +13:00
|
|
|
|
2023-08-09 11:02:49 +10:00
|
|
|
$handler = \core_group\customfield\grouping_handler::create();
|
|
|
|
$handler->instance_form_save($data);
|
|
|
|
|
2012-12-21 09:43:31 +13:00
|
|
|
// Invalidate the group data.
|
|
|
|
cache_helper::invalidate_by_definition('core', 'groupdata', array(), array($data->courseid));
|
|
|
|
|
2013-08-19 16:17:49 +08:00
|
|
|
// Trigger group event.
|
|
|
|
$params = array(
|
|
|
|
'context' => context_course::instance($data->courseid),
|
|
|
|
'objectid' => $data->id
|
|
|
|
);
|
|
|
|
$event = \core\event\grouping_updated::create($params);
|
|
|
|
$event->trigger();
|
2009-02-17 16:44:48 +00:00
|
|
|
|
|
|
|
return true;
|
2007-11-19 20:31:57 +00:00
|
|
|
}
|
|
|
|
|
2007-08-15 23:51:07 +00:00
|
|
|
/**
|
|
|
|
* Delete a group best effort, first removing members and links with courses and groupings.
|
|
|
|
* Removes group avatar too.
|
2012-01-05 09:29:55 +08:00
|
|
|
*
|
2008-07-06 17:57:06 +00:00
|
|
|
* @param mixed $grouporid The id of group to delete or full group object
|
2012-01-05 09:29:55 +08:00
|
|
|
* @return bool True if deletion was successful, false otherwise
|
2007-08-15 23:51:07 +00:00
|
|
|
*/
|
2008-07-06 17:57:06 +00:00
|
|
|
function groups_delete_group($grouporid) {
|
2009-09-16 10:06:05 +00:00
|
|
|
global $CFG, $DB;
|
|
|
|
require_once("$CFG->libdir/gdlib.php");
|
2007-08-15 23:51:07 +00:00
|
|
|
|
2008-07-06 17:57:06 +00:00
|
|
|
if (is_object($grouporid)) {
|
|
|
|
$groupid = $grouporid->id;
|
|
|
|
$group = $grouporid;
|
|
|
|
} else {
|
|
|
|
$groupid = $grouporid;
|
|
|
|
if (!$group = $DB->get_record('groups', array('id'=>$groupid))) {
|
2009-09-14 23:52:08 +00:00
|
|
|
//silently ignore attempts to delete missing already deleted groups ;-)
|
|
|
|
return true;
|
2008-07-06 17:57:06 +00:00
|
|
|
}
|
2007-08-15 23:51:07 +00:00
|
|
|
}
|
|
|
|
|
2019-02-28 14:41:43 +08:00
|
|
|
$context = context_course::instance($group->courseid);
|
|
|
|
|
2007-11-29 14:43:04 +00:00
|
|
|
// delete group calendar events
|
2008-06-01 13:09:04 +00:00
|
|
|
$DB->delete_records('event', array('groupid'=>$groupid));
|
2007-08-15 23:51:07 +00:00
|
|
|
//first delete usage in groupings_groups
|
2008-06-01 13:09:04 +00:00
|
|
|
$DB->delete_records('groupings_groups', array('groupid'=>$groupid));
|
2007-08-15 23:51:07 +00:00
|
|
|
//delete members
|
2008-06-01 13:09:04 +00:00
|
|
|
$DB->delete_records('groups_members', array('groupid'=>$groupid));
|
2019-02-28 14:41:43 +08:00
|
|
|
|
|
|
|
// Delete any members in a conversation related to this group.
|
|
|
|
if ($conversation = \core_message\api::get_conversation_by_area('core_group', 'groups', $groupid, $context->id)) {
|
|
|
|
\core_message\api::delete_all_conversation_data($conversation->id);
|
|
|
|
}
|
|
|
|
|
2007-08-15 23:51:07 +00:00
|
|
|
//group itself last
|
2009-02-17 16:44:48 +00:00
|
|
|
$DB->delete_records('groups', array('id'=>$groupid));
|
2009-11-04 06:14:06 +00:00
|
|
|
|
|
|
|
// Delete all files associated with this group
|
2012-08-02 11:20:48 +08:00
|
|
|
$context = context_course::instance($group->courseid);
|
2009-11-04 06:14:06 +00:00
|
|
|
$fs = get_file_storage();
|
2010-07-11 13:07:24 +00:00
|
|
|
$fs->delete_area_files($context->id, 'group', 'description', $groupid);
|
|
|
|
$fs->delete_area_files($context->id, 'group', 'icon', $groupid);
|
2009-11-04 06:14:06 +00:00
|
|
|
|
2012-12-21 09:43:31 +13:00
|
|
|
// Invalidate the grouping cache for the course
|
|
|
|
cache_helper::invalidate_by_definition('core', 'groupdata', array(), array($group->courseid));
|
2017-11-23 23:15:27 +08:00
|
|
|
// Purge the group and grouping cache for users.
|
|
|
|
cache_helper::purge_by_definition('core', 'user_group_groupings');
|
MDL-68093 groups: Add visibility and participation settings
These new settings are designed to enchance user privacy surrounding
groups. They allow groups to be configured so that users outside the
group cannot see the group, so that users in the group cannot see each
other, or so that users cannot see the group at all, even if they are in
it. This avoids issues where a group may be assigned based on sensitive
personal information (such as a person requiring special arrangements
due to a disability).
By default, groups are visible to all and available for participation in
activities, which maintains the current behaviour.
For performance, a new cache has been added to track the number of
groups on a course that are not visible to non-members. This allows us
to revert to the existing behaviour if the new features are not being
used at all on a course, and only apply the new visibility conditions if
they are.
Users who have the moodle/course:viewhiddengroups capability should be
concious of exposing hidden groups when showing their screen to other
users. The "Switch role to..." feature can be used to show a course page
on screen without exposing private availability conditions, for example.
The changes cover several specific areas:
* grouplib functions, which most code should use to get lists of groups
and members (this includes the participants page).
* Activities supporting group overrides will not allow overrides for
groups that are hidden from all users.
* Activities supporting separate/visible groups modes will only allow
groups with the new "participation" flag enabled to be selected.
* Group messaging will be disabled for groups where members cannot see
each other, or cannot see the group at all.
2022-09-06 09:14:24 +01:00
|
|
|
// Rebuild the coursehiddengroups cache for the course.
|
|
|
|
\core_group\visibility::update_hiddengroups_cache($group->courseid);
|
2012-12-21 09:43:31 +13:00
|
|
|
|
2013-08-19 16:17:49 +08:00
|
|
|
// Trigger group event.
|
|
|
|
$params = array(
|
|
|
|
'context' => $context,
|
|
|
|
'objectid' => $groupid
|
|
|
|
);
|
|
|
|
$event = \core\event\group_deleted::create($params);
|
|
|
|
$event->add_record_snapshot('groups', $group);
|
|
|
|
$event->trigger();
|
2008-07-06 17:57:06 +00:00
|
|
|
|
2024-01-04 12:46:55 +11:00
|
|
|
// Dispatch the hook for post group delete actions.
|
|
|
|
$hook = new \core_group\hook\after_group_deleted(
|
|
|
|
groupinstance: $group,
|
|
|
|
);
|
|
|
|
\core\di::get(\core\hook\manager::class)->dispatch($hook);
|
|
|
|
|
2009-02-17 16:44:48 +00:00
|
|
|
return true;
|
2007-08-15 23:51:07 +00:00
|
|
|
}
|
|
|
|
|
2007-11-19 20:31:57 +00:00
|
|
|
/**
|
|
|
|
* Delete grouping
|
2012-01-05 09:29:55 +08:00
|
|
|
*
|
2011-11-09 10:43:10 +01:00
|
|
|
* @param int $groupingorid
|
2007-11-19 20:31:57 +00:00
|
|
|
* @return bool success
|
|
|
|
*/
|
2008-07-06 17:57:06 +00:00
|
|
|
function groups_delete_grouping($groupingorid) {
|
2008-06-01 13:09:04 +00:00
|
|
|
global $DB;
|
|
|
|
|
2008-07-06 17:57:06 +00:00
|
|
|
if (is_object($groupingorid)) {
|
|
|
|
$groupingid = $groupingorid->id;
|
|
|
|
$grouping = $groupingorid;
|
|
|
|
} else {
|
|
|
|
$groupingid = $groupingorid;
|
|
|
|
if (!$grouping = $DB->get_record('groupings', array('id'=>$groupingorid))) {
|
2009-09-14 23:52:08 +00:00
|
|
|
//silently ignore attempts to delete missing already deleted groupings ;-)
|
|
|
|
return true;
|
2008-07-06 17:57:06 +00:00
|
|
|
}
|
2007-08-15 23:51:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//first delete usage in groupings_groups
|
2008-06-01 13:09:04 +00:00
|
|
|
$DB->delete_records('groupings_groups', array('groupingid'=>$groupingid));
|
2007-08-15 23:51:07 +00:00
|
|
|
// remove the default groupingid from course
|
2008-06-01 13:09:04 +00:00
|
|
|
$DB->set_field('course', 'defaultgroupingid', 0, array('defaultgroupingid'=>$groupingid));
|
2007-08-15 23:51:07 +00:00
|
|
|
// remove the groupingid from all course modules
|
2008-06-01 13:09:04 +00:00
|
|
|
$DB->set_field('course_modules', 'groupingid', 0, array('groupingid'=>$groupingid));
|
2007-08-15 23:51:07 +00:00
|
|
|
//group itself last
|
2009-02-17 16:44:48 +00:00
|
|
|
$DB->delete_records('groupings', array('id'=>$groupingid));
|
2009-11-04 06:14:06 +00:00
|
|
|
|
2012-08-02 11:20:48 +08:00
|
|
|
$context = context_course::instance($grouping->courseid);
|
2009-11-04 06:14:06 +00:00
|
|
|
$fs = get_file_storage();
|
2010-07-03 13:37:13 +00:00
|
|
|
$files = $fs->get_area_files($context->id, 'grouping', 'description', $groupingid);
|
2009-11-04 06:14:06 +00:00
|
|
|
foreach ($files as $file) {
|
|
|
|
$file->delete();
|
|
|
|
}
|
|
|
|
|
2012-12-21 09:43:31 +13:00
|
|
|
// Invalidate the grouping cache for the course
|
|
|
|
cache_helper::invalidate_by_definition('core', 'groupdata', array(), array($grouping->courseid));
|
2017-11-23 23:15:27 +08:00
|
|
|
// Purge the group and grouping cache for users.
|
|
|
|
cache_helper::purge_by_definition('core', 'user_group_groupings');
|
2012-12-21 09:43:31 +13:00
|
|
|
|
2013-08-19 16:17:49 +08:00
|
|
|
// Trigger group event.
|
|
|
|
$params = array(
|
|
|
|
'context' => $context,
|
|
|
|
'objectid' => $groupingid
|
|
|
|
);
|
|
|
|
$event = \core\event\grouping_deleted::create($params);
|
|
|
|
$event->add_record_snapshot('groupings', $grouping);
|
|
|
|
$event->trigger();
|
2008-07-06 17:57:06 +00:00
|
|
|
|
2009-02-17 16:44:48 +00:00
|
|
|
return true;
|
2007-08-15 23:51:07 +00:00
|
|
|
}
|
|
|
|
|
2007-11-19 20:31:57 +00:00
|
|
|
/**
|
2008-07-06 22:54:46 +00:00
|
|
|
* Remove all users (or one user) from all groups in course
|
2012-01-05 09:29:55 +08:00
|
|
|
*
|
2007-11-19 20:31:57 +00:00
|
|
|
* @param int $courseid
|
2008-07-06 22:54:46 +00:00
|
|
|
* @param int $userid 0 means all users
|
2015-12-30 09:05:43 +00:00
|
|
|
* @param bool $unused - formerly $showfeedback, is no longer used.
|
2007-11-19 20:31:57 +00:00
|
|
|
* @return bool success
|
|
|
|
*/
|
2015-12-30 09:05:43 +00:00
|
|
|
function groups_delete_group_members($courseid, $userid=0, $unused=false) {
|
2009-08-18 05:00:29 +00:00
|
|
|
global $DB, $OUTPUT;
|
2007-08-15 23:51:07 +00:00
|
|
|
|
2015-12-21 17:05:50 +08:00
|
|
|
// Get the users in the course which are in a group.
|
|
|
|
$sql = "SELECT gm.id as gmid, gm.userid, g.*
|
|
|
|
FROM {groups_members} gm
|
|
|
|
INNER JOIN {groups} g
|
|
|
|
ON gm.groupid = g.id
|
|
|
|
WHERE g.courseid = :courseid";
|
|
|
|
$params = array();
|
|
|
|
$params['courseid'] = $courseid;
|
|
|
|
// Check if we want to delete a specific user.
|
|
|
|
if ($userid) {
|
|
|
|
$sql .= " AND gm.userid = :userid";
|
|
|
|
$params['userid'] = $userid;
|
|
|
|
}
|
|
|
|
$rs = $DB->get_recordset_sql($sql, $params);
|
|
|
|
foreach ($rs as $usergroup) {
|
|
|
|
groups_remove_member($usergroup, $usergroup->userid);
|
|
|
|
}
|
|
|
|
$rs->close();
|
2008-07-06 22:54:46 +00:00
|
|
|
|
2007-08-15 23:51:07 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2007-11-29 14:43:04 +00:00
|
|
|
/**
|
|
|
|
* Remove all groups from all groupings in course
|
2012-01-05 09:29:55 +08:00
|
|
|
*
|
2007-11-29 14:43:04 +00:00
|
|
|
* @param int $courseid
|
|
|
|
* @param bool $showfeedback
|
|
|
|
* @return bool success
|
|
|
|
*/
|
|
|
|
function groups_delete_groupings_groups($courseid, $showfeedback=false) {
|
2009-08-18 05:00:29 +00:00
|
|
|
global $DB, $OUTPUT;
|
2007-11-29 14:43:04 +00:00
|
|
|
|
2008-06-01 13:09:04 +00:00
|
|
|
$groupssql = "SELECT id FROM {groups} g WHERE g.courseid = ?";
|
2013-08-19 16:17:49 +08:00
|
|
|
$results = $DB->get_recordset_select('groupings_groups', "groupid IN ($groupssql)",
|
|
|
|
array($courseid), '', 'groupid, groupingid');
|
|
|
|
|
|
|
|
foreach ($results as $result) {
|
|
|
|
groups_unassign_grouping($result->groupingid, $result->groupid, false);
|
|
|
|
}
|
2017-10-11 12:16:12 +01:00
|
|
|
$results->close();
|
2007-11-29 14:43:04 +00:00
|
|
|
|
2012-12-21 09:43:31 +13:00
|
|
|
// Invalidate the grouping cache for the course
|
|
|
|
cache_helper::invalidate_by_definition('core', 'groupdata', array(), array($courseid));
|
2017-11-23 23:15:27 +08:00
|
|
|
// Purge the group and grouping cache for users.
|
|
|
|
cache_helper::purge_by_definition('core', 'user_group_groupings');
|
2012-12-21 09:43:31 +13:00
|
|
|
|
2011-11-09 10:43:10 +01:00
|
|
|
// no need to show any feedback here - we delete usually first groupings and then groups
|
2007-11-29 14:43:04 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2007-11-19 20:31:57 +00:00
|
|
|
/**
|
|
|
|
* Delete all groups from course
|
2012-01-05 09:29:55 +08:00
|
|
|
*
|
2007-11-19 20:31:57 +00:00
|
|
|
* @param int $courseid
|
|
|
|
* @param bool $showfeedback
|
|
|
|
* @return bool success
|
|
|
|
*/
|
2007-08-15 23:51:07 +00:00
|
|
|
function groups_delete_groups($courseid, $showfeedback=false) {
|
2009-08-18 05:00:29 +00:00
|
|
|
global $CFG, $DB, $OUTPUT;
|
2007-08-15 23:51:07 +00:00
|
|
|
|
2013-08-19 16:17:49 +08:00
|
|
|
$groups = $DB->get_recordset('groups', array('courseid' => $courseid));
|
|
|
|
foreach ($groups as $group) {
|
|
|
|
groups_delete_group($group);
|
|
|
|
}
|
2017-10-11 12:16:12 +01:00
|
|
|
$groups->close();
|
2008-07-06 17:57:06 +00:00
|
|
|
|
2012-12-21 09:43:31 +13:00
|
|
|
// Invalidate the grouping cache for the course
|
|
|
|
cache_helper::invalidate_by_definition('core', 'groupdata', array(), array($courseid));
|
2017-11-23 23:15:27 +08:00
|
|
|
// Purge the group and grouping cache for users.
|
|
|
|
cache_helper::purge_by_definition('core', 'user_group_groupings');
|
MDL-68093 groups: Add visibility and participation settings
These new settings are designed to enchance user privacy surrounding
groups. They allow groups to be configured so that users outside the
group cannot see the group, so that users in the group cannot see each
other, or so that users cannot see the group at all, even if they are in
it. This avoids issues where a group may be assigned based on sensitive
personal information (such as a person requiring special arrangements
due to a disability).
By default, groups are visible to all and available for participation in
activities, which maintains the current behaviour.
For performance, a new cache has been added to track the number of
groups on a course that are not visible to non-members. This allows us
to revert to the existing behaviour if the new features are not being
used at all on a course, and only apply the new visibility conditions if
they are.
Users who have the moodle/course:viewhiddengroups capability should be
concious of exposing hidden groups when showing their screen to other
users. The "Switch role to..." feature can be used to show a course page
on screen without exposing private availability conditions, for example.
The changes cover several specific areas:
* grouplib functions, which most code should use to get lists of groups
and members (this includes the participants page).
* Activities supporting group overrides will not allow overrides for
groups that are hidden from all users.
* Activities supporting separate/visible groups modes will only allow
groups with the new "participation" flag enabled to be selected.
* Group messaging will be disabled for groups where members cannot see
each other, or cannot see the group at all.
2022-09-06 09:14:24 +01:00
|
|
|
// Rebuild the coursehiddengroups cache for the course.
|
|
|
|
\core_group\visibility::update_hiddengroups_cache($courseid);
|
2012-12-21 09:43:31 +13:00
|
|
|
|
2007-08-15 23:51:07 +00:00
|
|
|
if ($showfeedback) {
|
2011-11-09 10:43:10 +01:00
|
|
|
echo $OUTPUT->notification(get_string('deleted').' - '.get_string('groups', 'group'), 'notifysuccess');
|
2007-08-15 23:51:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2007-11-19 20:31:57 +00:00
|
|
|
/**
|
|
|
|
* Delete all groupings from course
|
2012-01-05 09:29:55 +08:00
|
|
|
*
|
2007-11-19 20:31:57 +00:00
|
|
|
* @param int $courseid
|
|
|
|
* @param bool $showfeedback
|
|
|
|
* @return bool success
|
|
|
|
*/
|
2007-08-15 23:51:07 +00:00
|
|
|
function groups_delete_groupings($courseid, $showfeedback=false) {
|
2009-08-18 05:00:29 +00:00
|
|
|
global $DB, $OUTPUT;
|
2007-08-15 23:51:07 +00:00
|
|
|
|
2013-08-19 16:17:49 +08:00
|
|
|
$groupings = $DB->get_recordset_select('groupings', 'courseid = ?', array($courseid));
|
|
|
|
foreach ($groupings as $grouping) {
|
|
|
|
groups_delete_grouping($grouping);
|
|
|
|
}
|
2017-10-11 12:16:12 +01:00
|
|
|
$groupings->close();
|
2009-11-04 06:14:06 +00:00
|
|
|
|
2013-08-19 16:17:49 +08:00
|
|
|
// Invalidate the grouping cache for the course.
|
2012-12-21 09:43:31 +13:00
|
|
|
cache_helper::invalidate_by_definition('core', 'groupdata', array(), array($courseid));
|
2017-11-23 23:15:27 +08:00
|
|
|
// Purge the group and grouping cache for users.
|
|
|
|
cache_helper::purge_by_definition('core', 'user_group_groupings');
|
2012-12-21 09:43:31 +13:00
|
|
|
|
2007-08-15 23:51:07 +00:00
|
|
|
if ($showfeedback) {
|
2011-11-09 10:43:10 +01:00
|
|
|
echo $OUTPUT->notification(get_string('deleted').' - '.get_string('groupings', 'group'), 'notifysuccess');
|
2007-08-15 23:51:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* =================================== */
|
|
|
|
/* various functions used by groups UI */
|
|
|
|
/* =================================== */
|
|
|
|
|
2008-01-17 11:28:54 +00:00
|
|
|
/**
|
|
|
|
* Obtains a list of the possible roles that group members might come from,
|
MDL-21782 reworked enrolment framework, the core infrastructure is in place, the basic plugins are all implemented; see the tracker issue for list of unfinished bits, expect more changes and improvements during the next week
AMOS START
MOV [sendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage,enrol_self]
MOV [configsendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage_desc,enrol_self]
MOV [enrolstartdate,core],[enrolstartdate,enrol_self]
MOV [enrolenddate,core],[enrolenddate,enrol_self]
CPY [welcometocourse,core],[welcometocourse,enrol_self]
CPY [welcometocoursetext,core],[welcometocoursetext,enrol_self]
MOV [notenrollable,core],[notenrollable,core_enrol]
MOV [enrolenddaterror,core],[enrolenddaterror,enrol_self]
MOV [enrolmentkeyhint,core],[passwordinvalidhint,enrol_self]
MOV [coursemanager,core_admin],[coursecontact,core_admin]
MOV [configcoursemanager,core_admin],[coursecontact_desc,core_admin]
MOV [enrolledincourserole,core],[enrolledincourserole,enrol_manual]
MOV [enrolme,core],[enrolme,core_enrol]
MOV [unenrol,core],[unenrol,core_enrol]
MOV [unenrolme,core],[unenrolme,core_enrol]
MOV [enrolmentnew,core],[enrolmentnew,core_enrol]
MOV [enrolmentnewuser,core],[enrolmentnewuser,core_enrol]
MOV [enrolments,core],[enrolments,core_enrol]
MOV [enrolperiod,core],[enrolperiod,core_enrol]
MOV [unenrolroleusers,core],[unenrolroleusers,core_enrol]
AMOS END
2010-06-21 15:30:49 +00:00
|
|
|
* on a course. Generally this includes only profile roles.
|
2012-01-05 09:29:55 +08:00
|
|
|
*
|
|
|
|
* @param context $context Context of course
|
2008-01-17 11:28:54 +00:00
|
|
|
* @return Array of role ID integers, or false if error/none.
|
|
|
|
*/
|
|
|
|
function groups_get_possible_roles($context) {
|
MDL-21782 reworked enrolment framework, the core infrastructure is in place, the basic plugins are all implemented; see the tracker issue for list of unfinished bits, expect more changes and improvements during the next week
AMOS START
MOV [sendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage,enrol_self]
MOV [configsendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage_desc,enrol_self]
MOV [enrolstartdate,core],[enrolstartdate,enrol_self]
MOV [enrolenddate,core],[enrolenddate,enrol_self]
CPY [welcometocourse,core],[welcometocourse,enrol_self]
CPY [welcometocoursetext,core],[welcometocoursetext,enrol_self]
MOV [notenrollable,core],[notenrollable,core_enrol]
MOV [enrolenddaterror,core],[enrolenddaterror,enrol_self]
MOV [enrolmentkeyhint,core],[passwordinvalidhint,enrol_self]
MOV [coursemanager,core_admin],[coursecontact,core_admin]
MOV [configcoursemanager,core_admin],[coursecontact_desc,core_admin]
MOV [enrolledincourserole,core],[enrolledincourserole,enrol_manual]
MOV [enrolme,core],[enrolme,core_enrol]
MOV [unenrol,core],[unenrol,core_enrol]
MOV [unenrolme,core],[unenrolme,core_enrol]
MOV [enrolmentnew,core],[enrolmentnew,core_enrol]
MOV [enrolmentnewuser,core],[enrolmentnewuser,core_enrol]
MOV [enrolments,core],[enrolments,core_enrol]
MOV [enrolperiod,core],[enrolperiod,core_enrol]
MOV [unenrolroleusers,core],[unenrolroleusers,core_enrol]
AMOS END
2010-06-21 15:30:49 +00:00
|
|
|
$roles = get_profile_roles($context);
|
|
|
|
return array_keys($roles);
|
2007-09-24 21:55:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets potential group members for grouping
|
2012-01-05 09:29:55 +08:00
|
|
|
*
|
2007-09-24 21:55:15 +00:00
|
|
|
* @param int $courseid The id of the course
|
|
|
|
* @param int $roleid The role to select users from
|
2014-08-06 10:50:14 +09:30
|
|
|
* @param mixed $source restrict to cohort, grouping or group id
|
2010-05-21 19:25:13 +00:00
|
|
|
* @param string $orderby The column to sort users by
|
2014-07-10 15:44:21 -04:00
|
|
|
* @param int $notingroup restrict to users not in existing groups
|
2016-02-11 10:23:22 +08:00
|
|
|
* @param bool $onlyactiveenrolments restrict to users who have an active enrolment in the course
|
2019-11-29 07:45:55 +00:00
|
|
|
* @param array $extrafields Extra user fields to return
|
2007-09-24 21:55:15 +00:00
|
|
|
* @return array An array of the users
|
|
|
|
*/
|
2014-08-06 10:50:14 +09:30
|
|
|
function groups_get_potential_members($courseid, $roleid = null, $source = null,
|
2014-07-10 15:44:21 -04:00
|
|
|
$orderby = 'lastname ASC, firstname ASC',
|
2019-11-29 07:45:55 +00:00
|
|
|
$notingroup = null, $onlyactiveenrolments = false, $extrafields = []) {
|
2009-12-16 22:14:17 +00:00
|
|
|
global $DB;
|
2007-11-19 20:31:57 +00:00
|
|
|
|
2012-08-02 11:20:48 +08:00
|
|
|
$context = context_course::instance($courseid);
|
2007-09-24 21:55:15 +00:00
|
|
|
|
2016-02-11 10:23:22 +08:00
|
|
|
list($esql, $params) = get_enrolled_sql($context, '', 0, $onlyactiveenrolments);
|
2010-07-03 13:37:13 +00:00
|
|
|
|
2014-07-10 15:44:21 -04:00
|
|
|
$notingroupsql = "";
|
|
|
|
if ($notingroup) {
|
|
|
|
// We want to eliminate users that are already associated with a course group.
|
|
|
|
$notingroupsql = "u.id NOT IN (SELECT userid
|
|
|
|
FROM {groups_members}
|
|
|
|
WHERE groupid IN (SELECT id
|
|
|
|
FROM {groups}
|
|
|
|
WHERE courseid = :courseid))";
|
|
|
|
$params['courseid'] = $courseid;
|
|
|
|
}
|
|
|
|
|
2007-09-24 21:55:15 +00:00
|
|
|
if ($roleid) {
|
2013-07-09 13:34:39 +08:00
|
|
|
// We are looking for all users with this role assigned in this context or higher.
|
2014-07-10 15:44:21 -04:00
|
|
|
list($relatedctxsql, $relatedctxparams) = $DB->get_in_or_equal($context->get_parent_context_ids(true),
|
|
|
|
SQL_PARAMS_NAMED,
|
|
|
|
'relatedctx');
|
2013-07-09 13:34:39 +08:00
|
|
|
|
|
|
|
$params = array_merge($params, $relatedctxparams, array('roleid' => $roleid));
|
2010-03-31 07:41:31 +00:00
|
|
|
$where = "WHERE u.id IN (SELECT userid
|
|
|
|
FROM {role_assignments}
|
2013-07-09 13:34:39 +08:00
|
|
|
WHERE roleid = :roleid AND contextid $relatedctxsql)";
|
2014-07-10 15:44:21 -04:00
|
|
|
$where .= $notingroup ? "AND $notingroupsql" : "";
|
|
|
|
} else if ($notingroup) {
|
|
|
|
$where = "WHERE $notingroupsql";
|
2007-09-24 21:55:15 +00:00
|
|
|
} else {
|
2010-03-31 07:41:31 +00:00
|
|
|
$where = "";
|
2007-09-24 21:55:15 +00:00
|
|
|
}
|
2007-11-19 20:31:57 +00:00
|
|
|
|
2014-08-06 10:50:14 +09:30
|
|
|
$sourcejoin = "";
|
|
|
|
if (is_int($source)) {
|
|
|
|
$sourcejoin .= "JOIN {cohort_members} cm ON (cm.userid = u.id AND cm.cohortid = :cohortid) ";
|
|
|
|
$params['cohortid'] = $source;
|
2010-04-23 09:15:55 +00:00
|
|
|
} else {
|
2014-08-06 10:50:14 +09:30
|
|
|
// Auto-create groups from an existing cohort membership.
|
|
|
|
if (isset($source['cohortid'])) {
|
|
|
|
$sourcejoin .= "JOIN {cohort_members} cm ON (cm.userid = u.id AND cm.cohortid = :cohortid) ";
|
|
|
|
$params['cohortid'] = $source['cohortid'];
|
|
|
|
}
|
|
|
|
// Auto-create groups from an existing group membership.
|
|
|
|
if (isset($source['groupid'])) {
|
|
|
|
$sourcejoin .= "JOIN {groups_members} gp ON (gp.userid = u.id AND gp.groupid = :groupid) ";
|
|
|
|
$params['groupid'] = $source['groupid'];
|
|
|
|
}
|
|
|
|
// Auto-create groups from an existing grouping membership.
|
|
|
|
if (isset($source['groupingid'])) {
|
|
|
|
$sourcejoin .= "JOIN {groupings_groups} gg ON gg.groupingid = :groupingid ";
|
|
|
|
$sourcejoin .= "JOIN {groups_members} gm ON (gm.userid = u.id AND gm.groupid = gg.groupid) ";
|
|
|
|
$params['groupingid'] = $source['groupingid'];
|
|
|
|
}
|
2010-04-23 09:15:55 +00:00
|
|
|
}
|
|
|
|
|
2021-03-15 15:36:32 +00:00
|
|
|
$userfieldsapi = \core_user\fields::for_userpic()->including(...$extrafields);
|
2020-10-12 17:51:20 +01:00
|
|
|
$allusernamefields = $userfieldsapi->get_sql('u', false, '', '', false)->selects;
|
2014-08-06 10:50:14 +09:30
|
|
|
$sql = "SELECT DISTINCT u.id, u.username, $allusernamefields, u.idnumber
|
2008-06-01 13:09:04 +00:00
|
|
|
FROM {user} u
|
2010-03-31 07:41:31 +00:00
|
|
|
JOIN ($esql) e ON e.id = u.id
|
2014-08-06 10:50:14 +09:30
|
|
|
$sourcejoin
|
2010-03-31 07:41:31 +00:00
|
|
|
$where
|
2008-06-01 13:09:04 +00:00
|
|
|
ORDER BY $orderby";
|
2007-11-19 20:31:57 +00:00
|
|
|
|
2008-06-01 13:09:04 +00:00
|
|
|
return $DB->get_records_sql($sql, $params);
|
2007-11-19 20:31:57 +00:00
|
|
|
|
2007-09-24 21:55:15 +00:00
|
|
|
}
|
2007-01-04 13:15:04 +13:00
|
|
|
|
2007-09-24 21:55:15 +00:00
|
|
|
/**
|
|
|
|
* Parse a group name for characters to replace
|
2012-01-05 09:29:55 +08:00
|
|
|
*
|
2007-09-24 21:55:15 +00:00
|
|
|
* @param string $format The format a group name will follow
|
|
|
|
* @param int $groupnumber The number of the group to be used in the parsed format string
|
|
|
|
* @return string the parsed format string
|
|
|
|
*/
|
|
|
|
function groups_parse_name($format, $groupnumber) {
|
|
|
|
if (strstr($format, '@') !== false) { // Convert $groupnumber to a character series
|
2007-11-19 20:31:57 +00:00
|
|
|
$letter = 'A';
|
|
|
|
for($i=0; $i<$groupnumber; $i++) {
|
|
|
|
$letter++;
|
2007-09-24 21:55:15 +00:00
|
|
|
}
|
2007-11-19 20:31:57 +00:00
|
|
|
$str = str_replace('@', $letter, $format);
|
2007-09-24 21:55:15 +00:00
|
|
|
} else {
|
2009-12-16 22:14:17 +00:00
|
|
|
$str = str_replace('#', $groupnumber+1, $format);
|
2007-09-24 21:55:15 +00:00
|
|
|
}
|
|
|
|
return($str);
|
2007-08-15 23:51:07 +00:00
|
|
|
}
|
2007-01-04 13:15:04 +13:00
|
|
|
|
2007-11-19 20:31:57 +00:00
|
|
|
/**
|
|
|
|
* Assigns group into grouping
|
2012-01-05 09:29:55 +08:00
|
|
|
*
|
2007-11-19 20:31:57 +00:00
|
|
|
* @param int groupingid
|
|
|
|
* @param int groupid
|
2013-01-08 13:17:24 +08:00
|
|
|
* @param int $timeadded The time the group was added to the grouping.
|
2017-11-23 23:15:27 +08:00
|
|
|
* @param bool $invalidatecache If set to true the course group cache and the user group cache will be invalidated as well.
|
2009-02-17 16:44:48 +00:00
|
|
|
* @return bool true or exception
|
2007-11-19 20:31:57 +00:00
|
|
|
*/
|
2012-12-21 09:43:31 +13:00
|
|
|
function groups_assign_grouping($groupingid, $groupid, $timeadded = null, $invalidatecache = true) {
|
2008-06-01 13:09:04 +00:00
|
|
|
global $DB;
|
|
|
|
|
|
|
|
if ($DB->record_exists('groupings_groups', array('groupingid'=>$groupingid, 'groupid'=>$groupid))) {
|
2007-11-19 20:31:57 +00:00
|
|
|
return true;
|
|
|
|
}
|
2010-09-21 08:16:49 +00:00
|
|
|
$assign = new stdClass();
|
2007-11-19 20:31:57 +00:00
|
|
|
$assign->groupingid = $groupingid;
|
2008-06-01 13:09:04 +00:00
|
|
|
$assign->groupid = $groupid;
|
2013-01-08 13:17:24 +08:00
|
|
|
if ($timeadded != null) {
|
|
|
|
$assign->timeadded = (integer)$timeadded;
|
|
|
|
} else {
|
|
|
|
$assign->timeadded = time();
|
|
|
|
}
|
2009-02-17 16:44:48 +00:00
|
|
|
$DB->insert_record('groupings_groups', $assign);
|
|
|
|
|
2016-02-28 23:43:47 +07:00
|
|
|
$courseid = $DB->get_field('groupings', 'courseid', array('id' => $groupingid));
|
2012-12-21 09:43:31 +13:00
|
|
|
if ($invalidatecache) {
|
|
|
|
// Invalidate the grouping cache for the course
|
|
|
|
cache_helper::invalidate_by_definition('core', 'groupdata', array(), array($courseid));
|
2017-11-23 23:15:27 +08:00
|
|
|
// Purge the group and grouping cache for users.
|
|
|
|
cache_helper::purge_by_definition('core', 'user_group_groupings');
|
2012-12-21 09:43:31 +13:00
|
|
|
}
|
|
|
|
|
2016-02-28 23:43:47 +07:00
|
|
|
// Trigger event.
|
|
|
|
$params = array(
|
|
|
|
'context' => context_course::instance($courseid),
|
|
|
|
'objectid' => $groupingid,
|
|
|
|
'other' => array('groupid' => $groupid)
|
|
|
|
);
|
|
|
|
$event = \core\event\grouping_group_assigned::create($params);
|
|
|
|
$event->trigger();
|
|
|
|
|
2009-02-17 16:44:48 +00:00
|
|
|
return true;
|
2007-11-19 20:31:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2012-12-21 09:43:31 +13:00
|
|
|
* Unassigns group from grouping
|
2012-01-05 09:29:55 +08:00
|
|
|
*
|
2007-11-19 20:31:57 +00:00
|
|
|
* @param int groupingid
|
|
|
|
* @param int groupid
|
2017-11-23 23:15:27 +08:00
|
|
|
* @param bool $invalidatecache If set to true the course group cache and the user group cache will be invalidated as well.
|
2007-11-19 20:31:57 +00:00
|
|
|
* @return bool success
|
|
|
|
*/
|
2012-12-21 09:43:31 +13:00
|
|
|
function groups_unassign_grouping($groupingid, $groupid, $invalidatecache = true) {
|
2008-06-01 13:09:04 +00:00
|
|
|
global $DB;
|
2009-02-17 16:44:48 +00:00
|
|
|
$DB->delete_records('groupings_groups', array('groupingid'=>$groupingid, 'groupid'=>$groupid));
|
2008-06-01 13:09:04 +00:00
|
|
|
|
2016-02-28 23:43:47 +07:00
|
|
|
$courseid = $DB->get_field('groupings', 'courseid', array('id' => $groupingid));
|
2012-12-21 09:43:31 +13:00
|
|
|
if ($invalidatecache) {
|
|
|
|
// Invalidate the grouping cache for the course
|
|
|
|
cache_helper::invalidate_by_definition('core', 'groupdata', array(), array($courseid));
|
2017-11-23 23:15:27 +08:00
|
|
|
// Purge the group and grouping cache for users.
|
|
|
|
cache_helper::purge_by_definition('core', 'user_group_groupings');
|
2012-12-21 09:43:31 +13:00
|
|
|
}
|
|
|
|
|
2016-02-28 23:43:47 +07:00
|
|
|
// Trigger event.
|
|
|
|
$params = array(
|
|
|
|
'context' => context_course::instance($courseid),
|
|
|
|
'objectid' => $groupingid,
|
|
|
|
'other' => array('groupid' => $groupid)
|
|
|
|
);
|
|
|
|
$event = \core\event\grouping_group_unassigned::create($params);
|
|
|
|
$event->trigger();
|
|
|
|
|
2009-02-17 16:44:48 +00:00
|
|
|
return true;
|
2007-11-19 20:31:57 +00:00
|
|
|
}
|
|
|
|
|
2008-01-17 11:28:54 +00:00
|
|
|
/**
|
|
|
|
* Lists users in a group based on their role on the course.
|
2009-11-01 16:48:45 +00:00
|
|
|
* Returns false if there's an error or there are no users in the group.
|
2008-01-17 11:28:54 +00:00
|
|
|
* Otherwise returns an array of role ID => role data, where role data includes:
|
|
|
|
* (role) $id, $shortname, $name
|
|
|
|
* $users: array of objects for each user which include the specified fields
|
|
|
|
* Users who do not have a role are stored in the returned array with key '-'
|
|
|
|
* and pseudo-role details (including a name, 'No role'). Users with multiple
|
|
|
|
* roles, same deal with key '*' and name 'Multiple roles'. You can find out
|
|
|
|
* which roles each has by looking in the $roles array of the user object.
|
2012-01-05 09:29:55 +08:00
|
|
|
*
|
2008-01-17 11:28:54 +00:00
|
|
|
* @param int $groupid
|
|
|
|
* @param int $courseid Course ID (should match the group's course)
|
2021-03-29 09:38:50 +01:00
|
|
|
* @param string $fields List of fields from user table (prefixed with u) and joined tables, default 'u.*'
|
|
|
|
* @param string|null $sort SQL ORDER BY clause, default (when null passed) is what comes from users_order_by_sql.
|
2008-10-29 08:18:24 +00:00
|
|
|
* @param string $extrawheretest extra SQL conditions ANDed with the existing where clause.
|
2021-03-29 09:38:50 +01:00
|
|
|
* @param array $whereorsortparams any parameters required by $extrawheretest or $joins (named parameters).
|
|
|
|
* @param string $joins any joins required to get the specified fields.
|
2008-01-17 11:28:54 +00:00
|
|
|
* @return array Complex array as described above
|
|
|
|
*/
|
2021-03-29 09:38:50 +01:00
|
|
|
function groups_get_members_by_role(int $groupid, int $courseid, string $fields = 'u.*',
|
|
|
|
?string $sort = null, string $extrawheretest = '', array $whereorsortparams = [], string $joins = '') {
|
2013-07-09 13:34:39 +08:00
|
|
|
global $DB;
|
2008-01-17 11:28:54 +00:00
|
|
|
|
|
|
|
// Retrieve information about all users and their roles on the course or
|
2009-11-01 16:48:45 +00:00
|
|
|
// parent ('related') contexts
|
2012-08-02 11:20:48 +08:00
|
|
|
$context = context_course::instance($courseid);
|
2008-06-01 13:09:04 +00:00
|
|
|
|
2013-07-09 13:34:39 +08:00
|
|
|
// We are looking for all users with this role assigned in this context or higher.
|
|
|
|
list($relatedctxsql, $relatedctxparams) = $DB->get_in_or_equal($context->get_parent_context_ids(true), SQL_PARAMS_NAMED, 'relatedctx');
|
|
|
|
|
2008-10-29 08:18:24 +00:00
|
|
|
if ($extrawheretest) {
|
|
|
|
$extrawheretest = ' AND ' . $extrawheretest;
|
|
|
|
}
|
|
|
|
|
2012-08-08 13:56:12 +01:00
|
|
|
if (is_null($sort)) {
|
|
|
|
list($sort, $sortparams) = users_order_by_sql('u');
|
|
|
|
$whereorsortparams = array_merge($whereorsortparams, $sortparams);
|
|
|
|
}
|
|
|
|
|
2012-05-18 11:40:35 +02:00
|
|
|
$sql = "SELECT r.id AS roleid, u.id AS userid, $fields
|
2008-06-01 13:09:04 +00:00
|
|
|
FROM {groups_members} gm
|
|
|
|
JOIN {user} u ON u.id = gm.userid
|
2013-07-09 13:34:39 +08:00
|
|
|
LEFT JOIN {role_assignments} ra ON (ra.userid = u.id AND ra.contextid $relatedctxsql)
|
2010-10-26 13:38:05 +00:00
|
|
|
LEFT JOIN {role} r ON r.id = ra.roleid
|
2021-03-29 09:38:50 +01:00
|
|
|
$joins
|
2010-03-31 07:41:31 +00:00
|
|
|
WHERE gm.groupid=:mgroupid
|
2010-10-26 13:38:05 +00:00
|
|
|
".$extrawheretest."
|
2008-06-01 13:09:04 +00:00
|
|
|
ORDER BY r.sortorder, $sort";
|
2013-07-09 13:34:39 +08:00
|
|
|
$whereorsortparams = array_merge($whereorsortparams, $relatedctxparams, array('mgroupid' => $groupid));
|
2012-08-08 13:56:12 +01:00
|
|
|
$rs = $DB->get_recordset_sql($sql, $whereorsortparams);
|
2008-06-01 13:09:04 +00:00
|
|
|
|
|
|
|
return groups_calculate_role_people($rs, $context);
|
2008-01-17 11:28:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Internal function used by groups_get_members_by_role to handle the
|
|
|
|
* results of a database query that includes a list of users and possible
|
|
|
|
* roles on a course.
|
|
|
|
*
|
2012-01-05 09:29:55 +08:00
|
|
|
* @param moodle_recordset $rs The record set (may be false)
|
|
|
|
* @param int $context ID of course context
|
2009-11-01 16:48:45 +00:00
|
|
|
* @return array As described in groups_get_members_by_role
|
2008-01-17 11:28:54 +00:00
|
|
|
*/
|
2008-06-01 13:09:04 +00:00
|
|
|
function groups_calculate_role_people($rs, $context) {
|
|
|
|
global $CFG, $DB;
|
|
|
|
|
|
|
|
if (!$rs) {
|
|
|
|
return array();
|
2008-01-18 11:03:00 +00:00
|
|
|
}
|
2008-01-17 11:28:54 +00:00
|
|
|
|
2012-05-18 11:40:35 +02:00
|
|
|
$allroles = role_fix_names(get_all_roles($context), $context);
|
2017-11-15 15:08:10 +00:00
|
|
|
$visibleroles = get_viewable_roles($context);
|
2008-06-01 13:09:04 +00:00
|
|
|
|
2008-01-17 11:28:54 +00:00
|
|
|
// Array of all involved roles
|
2008-06-01 13:09:04 +00:00
|
|
|
$roles = array();
|
2008-01-17 11:28:54 +00:00
|
|
|
// Array of all retrieved users
|
2008-06-01 13:09:04 +00:00
|
|
|
$users = array();
|
2008-01-17 11:28:54 +00:00
|
|
|
// Fill arrays
|
2008-06-01 13:09:04 +00:00
|
|
|
foreach ($rs as $rec) {
|
2008-01-17 11:28:54 +00:00
|
|
|
// Create information about user if this is a new one
|
2008-06-01 13:09:04 +00:00
|
|
|
if (!array_key_exists($rec->userid, $users)) {
|
2008-01-17 11:28:54 +00:00
|
|
|
// User data includes all the optional fields, but not any of the
|
|
|
|
// stuff we added to get the role details
|
2010-10-26 13:38:05 +00:00
|
|
|
$userdata = clone($rec);
|
2008-01-17 11:28:54 +00:00
|
|
|
unset($userdata->roleid);
|
|
|
|
unset($userdata->roleshortname);
|
|
|
|
unset($userdata->rolename);
|
|
|
|
unset($userdata->userid);
|
2008-06-01 13:09:04 +00:00
|
|
|
$userdata->id = $rec->userid;
|
2008-01-17 11:28:54 +00:00
|
|
|
|
|
|
|
// Make an array to hold the list of roles for this user
|
2008-06-01 13:09:04 +00:00
|
|
|
$userdata->roles = array();
|
|
|
|
$users[$rec->userid] = $userdata;
|
2008-01-17 11:28:54 +00:00
|
|
|
}
|
|
|
|
// If user has a role...
|
2008-06-01 13:09:04 +00:00
|
|
|
if (!is_null($rec->roleid)) {
|
2008-01-17 11:28:54 +00:00
|
|
|
// Create information about role if this is a new one
|
2012-05-18 11:40:35 +02:00
|
|
|
if (!array_key_exists($rec->roleid, $roles)) {
|
|
|
|
$role = $allroles[$rec->roleid];
|
2010-09-21 08:16:49 +00:00
|
|
|
$roledata = new stdClass();
|
2012-05-18 11:40:35 +02:00
|
|
|
$roledata->id = $role->id;
|
|
|
|
$roledata->shortname = $role->shortname;
|
|
|
|
$roledata->name = $role->localname;
|
2008-06-01 13:09:04 +00:00
|
|
|
$roledata->users = array();
|
|
|
|
$roles[$roledata->id] = $roledata;
|
2008-01-17 11:28:54 +00:00
|
|
|
}
|
|
|
|
// Record that user has role
|
2013-01-11 11:12:33 +00:00
|
|
|
$users[$rec->userid]->roles[$rec->roleid] = $roles[$rec->roleid];
|
2008-01-17 11:28:54 +00:00
|
|
|
}
|
|
|
|
}
|
2008-06-01 13:09:04 +00:00
|
|
|
$rs->close();
|
2008-01-17 11:28:54 +00:00
|
|
|
|
|
|
|
// Return false if there weren't any users
|
2010-10-26 13:38:05 +00:00
|
|
|
if (count($users) == 0) {
|
2008-01-17 11:28:54 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add pseudo-role for multiple roles
|
2010-09-21 08:16:49 +00:00
|
|
|
$roledata = new stdClass();
|
2008-06-01 13:09:04 +00:00
|
|
|
$roledata->name = get_string('multipleroles','role');
|
|
|
|
$roledata->users = array();
|
|
|
|
$roles['*'] = $roledata;
|
2008-01-17 11:28:54 +00:00
|
|
|
|
2010-10-26 13:38:05 +00:00
|
|
|
$roledata = new stdClass();
|
|
|
|
$roledata->name = get_string('noroles','role');
|
|
|
|
$roledata->users = array();
|
|
|
|
$roles[0] = $roledata;
|
|
|
|
|
2008-01-17 11:28:54 +00:00
|
|
|
// Now we rearrange the data to store users by role
|
2008-06-01 13:09:04 +00:00
|
|
|
foreach ($users as $userid=>$userdata) {
|
2017-11-15 15:08:10 +00:00
|
|
|
$visibleuserroles = array_intersect_key($userdata->roles, $visibleroles);
|
|
|
|
$rolecount = count($visibleuserroles);
|
2010-10-26 13:38:05 +00:00
|
|
|
if ($rolecount == 0) {
|
|
|
|
// does not have any roles
|
|
|
|
$roleid = 0;
|
|
|
|
} else if($rolecount > 1) {
|
2008-06-01 13:09:04 +00:00
|
|
|
$roleid = '*';
|
2008-01-17 11:28:54 +00:00
|
|
|
} else {
|
2017-11-15 15:08:10 +00:00
|
|
|
$userrole = reset($visibleuserroles);
|
2013-01-11 11:12:33 +00:00
|
|
|
$roleid = $userrole->id;
|
2008-01-17 11:28:54 +00:00
|
|
|
}
|
2008-06-01 13:09:04 +00:00
|
|
|
$roles[$roleid]->users[$userid] = $userdata;
|
2008-01-17 11:28:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Delete roles not used
|
2008-06-01 13:09:04 +00:00
|
|
|
foreach ($roles as $key=>$roledata) {
|
|
|
|
if (count($roledata->users)===0) {
|
2008-01-17 11:28:54 +00:00
|
|
|
unset($roles[$key]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return list of roles containing their users
|
|
|
|
return $roles;
|
|
|
|
}
|
2015-04-09 12:28:51 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Synchronises enrolments with the group membership
|
|
|
|
*
|
|
|
|
* Designed for enrolment methods provide automatic synchronisation between enrolled users
|
|
|
|
* and group membership, such as enrol_cohort and enrol_meta .
|
|
|
|
*
|
|
|
|
* @param string $enrolname name of enrolment method without prefix
|
|
|
|
* @param int $courseid course id where sync needs to be performed (0 for all courses)
|
|
|
|
* @param string $gidfield name of the field in 'enrol' table that stores group id
|
|
|
|
* @return array Returns the list of removed and added users. Each record contains fields:
|
|
|
|
* userid, enrolid, courseid, groupid, groupname
|
|
|
|
*/
|
|
|
|
function groups_sync_with_enrolment($enrolname, $courseid = 0, $gidfield = 'customint2') {
|
|
|
|
global $DB;
|
|
|
|
$onecourse = $courseid ? "AND e.courseid = :courseid" : "";
|
|
|
|
$params = array(
|
|
|
|
'enrolname' => $enrolname,
|
|
|
|
'component' => 'enrol_'.$enrolname,
|
|
|
|
'courseid' => $courseid
|
|
|
|
);
|
|
|
|
|
|
|
|
$affectedusers = array(
|
|
|
|
'removed' => array(),
|
|
|
|
'added' => array()
|
|
|
|
);
|
|
|
|
|
|
|
|
// Remove invalid.
|
|
|
|
$sql = "SELECT ue.userid, ue.enrolid, e.courseid, g.id AS groupid, g.name AS groupname
|
|
|
|
FROM {groups_members} gm
|
|
|
|
JOIN {groups} g ON (g.id = gm.groupid)
|
|
|
|
JOIN {enrol} e ON (e.enrol = :enrolname AND e.courseid = g.courseid $onecourse)
|
|
|
|
JOIN {user_enrolments} ue ON (ue.userid = gm.userid AND ue.enrolid = e.id)
|
|
|
|
WHERE gm.component=:component AND gm.itemid = e.id AND g.id <> e.{$gidfield}";
|
|
|
|
|
|
|
|
$rs = $DB->get_recordset_sql($sql, $params);
|
|
|
|
foreach ($rs as $gm) {
|
|
|
|
groups_remove_member($gm->groupid, $gm->userid);
|
|
|
|
$affectedusers['removed'][] = $gm;
|
|
|
|
}
|
|
|
|
$rs->close();
|
|
|
|
|
|
|
|
// Add missing.
|
|
|
|
$sql = "SELECT ue.userid, ue.enrolid, e.courseid, g.id AS groupid, g.name AS groupname
|
|
|
|
FROM {user_enrolments} ue
|
|
|
|
JOIN {enrol} e ON (e.id = ue.enrolid AND e.enrol = :enrolname $onecourse)
|
|
|
|
JOIN {groups} g ON (g.courseid = e.courseid AND g.id = e.{$gidfield})
|
|
|
|
JOIN {user} u ON (u.id = ue.userid AND u.deleted = 0)
|
|
|
|
LEFT JOIN {groups_members} gm ON (gm.groupid = g.id AND gm.userid = ue.userid)
|
|
|
|
WHERE gm.id IS NULL";
|
|
|
|
|
|
|
|
$rs = $DB->get_recordset_sql($sql, $params);
|
|
|
|
foreach ($rs as $ue) {
|
|
|
|
groups_add_member($ue->groupid, $ue->userid, 'enrol_'.$enrolname, $ue->enrolid);
|
|
|
|
$affectedusers['added'][] = $ue;
|
|
|
|
}
|
|
|
|
$rs->close();
|
|
|
|
|
|
|
|
return $affectedusers;
|
2015-12-30 09:05:43 +00:00
|
|
|
}
|
2017-06-29 10:30:04 +08:00
|
|
|
|
2017-07-07 10:08:18 +08:00
|
|
|
/**
|
|
|
|
* Callback for inplace editable API.
|
|
|
|
*
|
|
|
|
* @param string $itemtype - Only user_groups is supported.
|
|
|
|
* @param string $itemid - Userid and groupid separated by a :
|
|
|
|
* @param string $newvalue - json encoded list of groupids.
|
|
|
|
* @return \core\output\inplace_editable
|
|
|
|
*/
|
2017-06-29 10:30:04 +08:00
|
|
|
function core_group_inplace_editable($itemtype, $itemid, $newvalue) {
|
|
|
|
if ($itemtype === 'user_groups') {
|
|
|
|
return \core_group\output\user_groups_editable::update($itemid, $newvalue);
|
|
|
|
}
|
|
|
|
}
|
2022-12-15 10:59:05 +10:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Updates group messaging to enable/disable in bulk.
|
|
|
|
*
|
|
|
|
* @param array $groupids array of group id numbers.
|
|
|
|
* @param bool $enabled if true, enables messaging else disables messaging
|
|
|
|
*/
|
|
|
|
function set_groups_messaging(array $groupids, bool $enabled): void {
|
|
|
|
foreach ($groupids as $groupid) {
|
|
|
|
$data = groups_get_group($groupid, '*', MUST_EXIST);
|
|
|
|
$data->enablemessaging = $enabled;
|
|
|
|
groups_update_group($data);
|
|
|
|
}
|
|
|
|
}
|
2023-08-09 11:02:49 +10:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns custom fields data for provided groups.
|
|
|
|
*
|
|
|
|
* @param array $groupids a list of group IDs to provide data for.
|
|
|
|
* @return \core_customfield\data_controller[]
|
|
|
|
*/
|
|
|
|
function get_group_custom_fields_data(array $groupids): array {
|
|
|
|
$result = [];
|
|
|
|
|
|
|
|
if (!empty($groupids)) {
|
|
|
|
$handler = \core_group\customfield\group_handler::create();
|
|
|
|
$customfieldsdata = $handler->get_instances_data($groupids, true);
|
|
|
|
|
|
|
|
foreach ($customfieldsdata as $groupid => $fieldcontrollers) {
|
|
|
|
foreach ($fieldcontrollers as $fieldcontroller) {
|
|
|
|
$result[$groupid][] = [
|
|
|
|
'type' => $fieldcontroller->get_field()->get('type'),
|
|
|
|
'value' => $fieldcontroller->export_value(),
|
|
|
|
'valueraw' => $fieldcontroller->get_value(),
|
|
|
|
'name' => $fieldcontroller->get_field()->get('name'),
|
|
|
|
'shortname' => $fieldcontroller->get_field()->get('shortname'),
|
|
|
|
];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $result;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns custom fields data for provided groupings.
|
|
|
|
*
|
|
|
|
* @param array $groupingids a list of group IDs to provide data for.
|
|
|
|
* @return \core_customfield\data_controller[]
|
|
|
|
*/
|
|
|
|
function get_grouping_custom_fields_data(array $groupingids): array {
|
|
|
|
$result = [];
|
|
|
|
|
|
|
|
if (!empty($groupingids)) {
|
|
|
|
$handler = \core_group\customfield\grouping_handler::create();
|
|
|
|
$customfieldsdata = $handler->get_instances_data($groupingids, true);
|
|
|
|
|
|
|
|
foreach ($customfieldsdata as $groupingid => $fieldcontrollers) {
|
|
|
|
foreach ($fieldcontrollers as $fieldcontroller) {
|
|
|
|
$result[$groupingid][] = [
|
|
|
|
'type' => $fieldcontroller->get_field()->get('type'),
|
|
|
|
'value' => $fieldcontroller->export_value(),
|
|
|
|
'valueraw' => $fieldcontroller->get_value(),
|
|
|
|
'name' => $fieldcontroller->get_field()->get('name'),
|
|
|
|
'shortname' => $fieldcontroller->get_field()->get('shortname'),
|
|
|
|
];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $result;
|
|
|
|
}
|