moodle/group/lib.php

744 lines
23 KiB
PHP
Raw Normal View History

<?php
/**
* Extra library for groups and groupings.
*
* @copyright &copy; 2006 The Open University
* @author J.White AT open.ac.uk, Petr Skoda (skodak)
* @license http://www.gnu.org/copyleft/gpl.html GNU Public License
* @package groups
*/
/*
* INTERNAL FUNCTIONS - to be used by moodle core only
* require_once $CFG->dirroot.'/group/lib.php' must be used
*/
/**
* Adds a specified user to a group
* @param int $userid The user id
* @param int $groupid The group id
* @return boolean True if user added successfully or the user is already a
* member of the group, false otherwise.
*/
function groups_add_member($groupid, $userid) {
2008-06-01 13:09:04 +00:00
global $DB;
if (!groups_group_exists($groupid)) {
return false;
}
if (groups_is_member($groupid, $userid)) {
return true;
}
$member = new object();
$member->groupid = $groupid;
$member->userid = $userid;
$member->timeadded = time();
2008-06-01 13:09:04 +00:00
if (!$DB->insert_record('groups_members', $member)) {
return false;
}
//update group info
2008-06-01 13:09:04 +00:00
$DB->set_field('groups', 'timemodified', $member->timeadded, array('id'=>$groupid));
//trigger groups events
$eventdata = new object();
$eventdata->groupid = $groupid;
$eventdata->userid = $userid;
events_trigger('groups_member_added', $eventdata);
return true;
}
/**
* Deletes the link between the specified user and group.
* @param int $groupid The group to delete the user from
* @param int $userid The user to delete
* @return boolean True if deletion was successful, false otherwise
*/
function groups_remove_member($groupid, $userid) {
2008-06-01 13:09:04 +00:00
global $DB;
if (!groups_group_exists($groupid)) {
return false;
}
if (!groups_is_member($groupid, $userid)) {
return true;
}
2008-06-01 13:09:04 +00:00
if (!$DB->delete_records('groups_members', array('groupid'=>$groupid, 'userid'=>$userid))) {
return false;
}
//update group info
2008-06-01 13:09:04 +00:00
$DB->set_field('groups', 'timemodified', time(), array('id'=>$groupid));
//trigger groups events
$eventdata = new object();
$eventdata->groupid = $groupid;
$eventdata->userid = $userid;
events_trigger('groups_member_removed', $eventdata);
return true;
}
/**
* Add a new group
* @param object $data group properties (with magic quotes)
* @param object $um upload manager with group picture
* @return id of group or false if error
*/
function groups_create_group($data, $editform=false) {
2008-06-01 13:09:04 +00:00
global $CFG, $DB;
require_once("$CFG->libdir/gdlib.php");
2008-06-01 13:09:04 +00:00
$data->timecreated = time();
$data->timemodified = $data->timecreated;
2008-06-01 13:09:04 +00:00
$data->name = trim($data->name);
$id = $DB->insert_record('groups', $data);
if ($id) {
$data->id = $id;
if ($editform) {
//update image
if (save_profile_image($id, $editform, 'groups')) {
$DB->set_field('groups', 'picture', 1, array('id'=>$id));
}
$data->picture = 1;
}
//trigger groups events
events_trigger('groups_group_created', $data);
}
return $id;
}
/**
* Add a new grouping
* @param object $data grouping properties (with magic quotes)
* @return id of grouping or false if error
*/
function groups_create_grouping($data) {
2008-06-01 13:09:04 +00:00
global $DB;
2008-06-01 13:09:04 +00:00
$data->timecreated = time();
$data->timemodified = $data->timecreated;
2008-06-01 13:09:04 +00:00
$data->name = trim($data->name);
$id = $DB->insert_record('groupings', $data);
if ($id) {
//trigger groups events
$data->id = $id;
events_trigger('groups_grouping_created', $data);
}
return $id;
}
/**
* Update group
* @param object $data group properties (with magic quotes)
* @param object $um upload manager with group picture
* @return boolean success
*/
function groups_update_group($data, $editform=false) {
2008-06-01 13:09:04 +00:00
global $CFG, $DB;
require_once("$CFG->libdir/gdlib.php");
$data->timemodified = time();
2008-06-01 13:09:04 +00:00
$data->name = trim($data->name);
$result = $DB->update_record('groups', $data);
if ($result) {
if ($editform) {
//update image
if (save_profile_image($data->id, $editform, 'groups')) {
2008-06-01 13:09:04 +00:00
$DB->set_field('groups', 'picture', 1, array('id'=>$data->id));
$data->picture = 1;
}
}
//trigger groups events
events_trigger('groups_group_updated', $data);
}
return $result;
}
/**
* Update grouping
* @param object $data grouping properties (with magic quotes)
* @return boolean success
*/
function groups_update_grouping($data) {
2008-06-01 13:09:04 +00:00
global $DB;
$data->timemodified = time();
2008-06-01 13:09:04 +00:00
$data->name = trim($data->name);
$result = $DB->update_record('groupings', $data);
if ($result) {
//trigger groups events
events_trigger('groups_grouping_updated', $data);
}
return $result;
}
/**
* Delete a group best effort, first removing members and links with courses and groupings.
* Removes group avatar too.
* @param mixed $grouporid The id of group to delete or full group object
* @return boolean True if deletion was successful, false otherwise
*/
function groups_delete_group($grouporid) {
2008-06-01 13:09:04 +00:00
global $CFG, $DB;
require_once($CFG->libdir.'/gdlib.php');
if (is_object($grouporid)) {
$groupid = $grouporid->id;
$group = $grouporid;
} else {
$groupid = $grouporid;
if (!$group = $DB->get_record('groups', array('id'=>$groupid))) {
return false;
}
}
// delete group calendar events
2008-06-01 13:09:04 +00:00
$DB->delete_records('event', array('groupid'=>$groupid));
//first delete usage in groupings_groups
2008-06-01 13:09:04 +00:00
$DB->delete_records('groupings_groups', array('groupid'=>$groupid));
//delete members
2008-06-01 13:09:04 +00:00
$DB->delete_records('groups_members', array('groupid'=>$groupid));
//then imge
delete_profile_image($groupid, 'groups');
//group itself last
$result = $DB->delete_records('groups', array('id'=>$groupid));
if ($result) {
//trigger groups events
events_trigger('groups_group_deleted', $group);
}
return $result;
}
/**
* Delete grouping
* @param int $groupingid
* @return bool success
*/
function groups_delete_grouping($groupingorid) {
2008-06-01 13:09:04 +00:00
global $DB;
if (is_object($groupingorid)) {
$groupingid = $groupingorid->id;
$grouping = $groupingorid;
} else {
$groupingid = $groupingorid;
if (!$grouping = $DB->get_record('groupings', array('id'=>$groupingorid))) {
return false;
}
}
//first delete usage in groupings_groups
2008-06-01 13:09:04 +00:00
$DB->delete_records('groupings_groups', array('groupingid'=>$groupingid));
// remove the default groupingid from course
2008-06-01 13:09:04 +00:00
$DB->set_field('course', 'defaultgroupingid', 0, array('defaultgroupingid'=>$groupingid));
// 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));
//group itself last
$result = $DB->delete_records('groupings', array('id'=>$groupingid));
if ($result) {
//trigger groups events
events_trigger('groups_grouping_deleted', $grouping);
}
return $result;
}
/**
* Remove all users (or one user) from all groups in course
* @param int $courseid
* @param int $userid 0 means all users
* @param bool $showfeedback
* @return bool success
*/
function groups_delete_group_members($courseid, $userid=0, $showfeedback=false) {
2008-06-01 13:09:04 +00:00
global $DB;
if (is_bool($userid)) {
debugging('Incorrect userid function parameter');
return false;
}
$params = array('courseid'=>$courseid);
if ($userid) {
$usersql = "AND userid = :userid";
$params['userid'] = $userid;
} else {
$usersql = "";
}
$groupssql = "SELECT id FROM {groups} g WHERE g.courseid = :courseid";
$DB->delete_records_select('groups_members', "groupid IN ($groupssql) $usersql", $params);
//trigger groups events
$eventdata = new object();
$eventdata->courseid = $courseid;
$eventdata->userid = $userid;
events_trigger('groups_members_removed', $eventdata);
if ($showfeedback) {
notify(get_string('deleted').' groups_members');
}
return true;
}
/**
* Remove all groups from all groupings in course
* @param int $courseid
* @param bool $showfeedback
* @return bool success
*/
function groups_delete_groupings_groups($courseid, $showfeedback=false) {
2008-06-01 13:09:04 +00:00
global $DB;
2008-06-01 13:09:04 +00:00
$groupssql = "SELECT id FROM {groups} g WHERE g.courseid = ?";
$DB->delete_records_select('groupings_groups', "groupid IN ($groupssql)", array($courseid));
//trigger groups events
events_trigger('groups_groupings_groups_removed', $courseid);
if ($showfeedback) {
notify(get_string('deleted').' groupings_groups');
}
return true;
}
/**
* Delete all groups from course
* @param int $courseid
* @param bool $showfeedback
* @return bool success
*/
function groups_delete_groups($courseid, $showfeedback=false) {
2008-06-01 13:09:04 +00:00
global $CFG, $DB;
require_once($CFG->libdir.'/gdlib.php');
// delete any uses of groups
groups_delete_groupings_groups($courseid, $showfeedback);
groups_delete_group_members($courseid, 0, $showfeedback);
// delete group pictures
2008-06-01 13:09:04 +00:00
if ($groups = $DB->get_records('groups', array('courseid'=>$courseid))) {
foreach($groups as $group) {
delete_profile_image($group->id, 'groups');
}
}
// delete group calendar events
2008-06-01 13:09:04 +00:00
$groupssql = "SELECT id FROM {groups} g WHERE g.courseid = ?";
$DB->delete_records_select('event', "groupid IN ($groupssql)", array($courseid));
2008-06-01 13:09:04 +00:00
$DB->delete_records('groups', array('courseid'=>$courseid));
//trigger groups events
events_trigger('groups_groups_deleted', $courseid);
if ($showfeedback) {
notify(get_string('deleted').' groups');
}
return true;
}
/**
* Delete all groupings from course
* @param int $courseid
* @param bool $showfeedback
* @return bool success
*/
function groups_delete_groupings($courseid, $showfeedback=false) {
2008-06-01 13:09:04 +00:00
global $DB;
// delete any uses of groupings
2008-06-01 13:09:04 +00:00
$sql = "DELETE FROM {groupings_groups}
WHERE groupingid in (SELECT id FROM {groupings} g WHERE g.courseid = ?)";
$DB->execute($sql, array($courseid));
// remove the default groupingid from course
2008-06-01 13:09:04 +00:00
$DB->set_field('course', 'defaultgroupingid', 0, array('id'=>$courseid));
// remove the groupingid from all course modules
2008-06-01 13:09:04 +00:00
$DB->set_field('course_modules', 'groupingid', 0, array('course'=>$courseid));
2008-06-01 13:09:04 +00:00
$DB->delete_records('groupings', array('courseid'=>$courseid));
//trigger groups events
events_trigger('groups_groupings_deleted', $courseid);
if ($showfeedback) {
notify(get_string('deleted').' groupings');
}
return true;
}
/* =================================== */
/* various functions used by groups UI */
/* =================================== */
/**
* Gets the users for a course who are not in a specified group, and returns
* them in an array organised by role. For the array format, see
* groups_get_members_by_role.
* @param int $groupid The id of the group
* @param string searchtext similar to searchtext in role assign, search
* @return array An array of role id or '*' => information about that role
* including a list of users
*/
function groups_get_users_not_in_group_by_role($courseid, $groupid, $searchtext='', $sort = 'u.lastname ASC') {
2008-06-01 13:09:04 +00:00
global $CFG, $DB;
$context = get_context_instance(CONTEXT_COURSE, $courseid);
2008-06-01 13:09:04 +00:00
/// Get list of allowed roles
if (!$validroleids = groups_get_possible_roles($context)) {
return array();
}
list($roleids, $params) = $DB->get_in_or_equal($validroleids, SQL_PARAMS_NAMED, $start='r0');
if ($searchtext !== '') { // Search for a subset of remaining users
2008-06-01 13:09:04 +00:00
$LIKE = $DB->sql_ilike();
$FULLNAME = $DB->sql_fullname();
$wheresearch = " AND u.id IN (SELECT id FROM {user} WHERE $FULLNAME $LIKE :search1 OR email $LIKE :search2)";
$params['search1'] = "%$searchtext%";
$params['search2'] = "%$searchtext%";
} else {
$wheresearch = '';
}
/// Construct the main SQL
2008-06-01 13:09:04 +00:00
$sql = " SELECT r.id AS roleid,r.shortname AS roleshortname,r.name AS rolename,
u.id AS userid, u.firstname, u.lastname
FROM {user} u
JOIN {role_assignments} ra ON ra.userid = u.id
JOIN {role} r ON r.id = ra.roleid
WHERE ra.contextid ".get_related_contexts_string($context)."
AND u.deleted = 0
AND ra.roleid $roleids
AND u.id NOT IN (SELECT userid
FROM {groups_members}
WHERE groupid = :groupid)
$wheresearch
ORDER BY $sort";
$params['groupid'] = $groupid;
$rs = $DB->get_recordset_sql($sql, $params);
return groups_calculate_role_people($rs, $context);
}
/**
* Obtains a list of the possible roles that group members might come from,
* on a course. Generally this includes all the roles who would have
* course:view on that course, except the doanything roles.
* @param object $context Context of course
* @return Array of role ID integers, or false if error/none.
*/
function groups_get_possible_roles($context) {
$capability = 'moodle/course:view';
$doanything = false;
// find all possible "student" roles
if ($possibleroles = get_roles_with_capability($capability, CAP_ALLOW, $context)) {
if (!$doanything) {
if (!$sitecontext = get_context_instance(CONTEXT_SYSTEM)) {
return false; // Something is seriously wrong
}
$doanythingroles = get_roles_with_capability('moodle/site:doanything', CAP_ALLOW, $sitecontext);
}
$validroleids = array();
foreach ($possibleroles as $possiblerole) {
if (!$doanything) {
if (isset($doanythingroles[$possiblerole->id])) { // We don't want these included
continue;
}
}
if ($caps = role_context_capabilities($possiblerole->id, $context, $capability)) { // resolved list
if (isset($caps[$capability]) && $caps[$capability] > 0) { // resolved capability > 0
$validroleids[] = $possiblerole->id;
}
}
}
if (empty($validroleids)) {
return false;
}
return $validroleids;
} else {
return false; // No need to continue, since no roles have this capability set
}
}
/**
* Gets potential group members for grouping
* @param int $courseid The id of the course
* @param int $roleid The role to select users from
* @param string $orderby The colum to sort users by
* @return array An array of the users
*/
function groups_get_potential_members($courseid, $roleid = null, $orderby = 'lastname,firstname') {
2008-06-01 13:09:04 +00:00
global $DB;
$context = get_context_instance(CONTEXT_COURSE, $courseid);
$sitecontext = get_context_instance(CONTEXT_SYSTEM);
$rolenames = array();
$avoidroles = array();
if ($roles = get_roles_used_in_context($context, true)) {
$canviewroles = get_roles_with_capability('moodle/course:view', CAP_ALLOW, $context);
$doanythingroles = get_roles_with_capability('moodle/site:doanything', CAP_ALLOW, $sitecontext);
foreach ($roles as $role) {
if (!isset($canviewroles[$role->id])) { // Avoid this role (eg course creator)
$avoidroles[] = $role->id;
unset($roles[$role->id]);
continue;
}
if (isset($doanythingroles[$role->id])) { // Avoid this role (ie admin)
$avoidroles[] = $role->id;
unset($roles[$role->id]);
continue;
}
$rolenames[$role->id] = strip_tags(role_get_name($role, $context)); // Used in menus etc later on
}
}
if ($avoidroles) {
2008-06-01 13:09:04 +00:00
list($adminroles, $params) = $DB->get_in_or_equal($avoidroles, SQL_PARAMS_NAMED, 'ar0', false);
$adminroles = "AND r.roleid $adminroles";
} else {
2008-06-01 13:09:04 +00:00
$adminroles = "";
$params = array();
}
// we are looking for all users with this role assigned in this context or higher
if ($usercontexts = get_parent_contexts($context)) {
2008-06-01 13:09:04 +00:00
$listofcontexts = 'IN ('.implode(',', $usercontexts).')';
} else {
2008-06-01 13:09:04 +00:00
$listofcontexts = '='.$sitecontext->id.')'; // must be site
}
if ($roleid) {
2008-06-01 13:09:04 +00:00
$selectrole = "AND r.roleid = :roleid";
$params['roleid'] = $roleid;
} else {
2008-06-01 13:09:04 +00:00
$selectrole = "";
}
2008-06-01 13:09:04 +00:00
$sql = "SELECT u.id, u.username, u.firstname, u.lastname, u.idnumber
FROM {user} u
JOIN {role_assignments} r on u.id=r.userid
WHERE (r.contextid = :contextid OR r.contextid $listofcontexts)
AND u.deleted = 0 AND u.username != 'guest'
$selectrole $adminroles
ORDER BY $orderby";
$params['contextid'] = $context->id;
2008-06-01 13:09:04 +00:00
return $DB->get_records_sql($sql, $params);
}
/**
* Parse a group name for characters to replace
* @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
$letter = 'A';
for($i=0; $i<$groupnumber; $i++) {
$letter++;
}
$str = str_replace('@', $letter, $format);
} else {
$str = str_replace('#', $groupnumber+1, $format);
}
return($str);
}
/**
* Assigns group into grouping
* @param int groupingid
* @param int groupid
* @return bool success
*/
function groups_assign_grouping($groupingid, $groupid) {
2008-06-01 13:09:04 +00:00
global $DB;
if ($DB->record_exists('groupings_groups', array('groupingid'=>$groupingid, 'groupid'=>$groupid))) {
return true;
}
$assign = new object();
$assign->groupingid = $groupingid;
2008-06-01 13:09:04 +00:00
$assign->groupid = $groupid;
$assign->timeadded = time();
return (bool)$DB->insert_record('groupings_groups', $assign);
}
/**
* Unassigns group grom grouping
* @param int groupingid
* @param int groupid
* @return bool success
*/
function groups_unassign_grouping($groupingid, $groupid) {
2008-06-01 13:09:04 +00:00
global $DB;
return $DB->delete_records('groupings_groups', array('groupingid'=>$groupingid, 'groupid'=>$groupid));
}
/**
* Lists users in a group based on their role on the course.
* Returns false if there's an error or there are no users in the group.
* 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.
* @param int $groupid
* @param int $courseid Course ID (should match the group's course)
* @param string $fields List of fields from user table prefixed with u, default 'u.*'
* @param string $sort SQL ORDER BY clause, default 'u.lastname ASC'
* @return array Complex array as described above
*/
function groups_get_members_by_role($groupid, $courseid, $fields='u.*', $sort='u.lastname ASC') {
2008-06-01 13:09:04 +00:00
global $CFG, $DB;
// Retrieve information about all users and their roles on the course or
// parent ('related') contexts
2008-06-01 13:09:04 +00:00
$context = get_context_instance(CONTEXT_COURSE, $courseid);
$sql = "SELECT r.id AS roleid, r.shortname AS roleshortname, r.name AS rolename,
u.id AS userid, $fields
FROM {groups_members} gm
JOIN {user} u ON u.id = gm.userid
JOIN {role_assignments} ra ON ra.userid = u.id
JOIN {role} r ON r.id = ra.roleid
WHERE gm.groupid=?
AND ra.contextid ".get_related_contexts_string($context)."
ORDER BY r.sortorder, $sort";
$params = array($groupid);
$rs = $DB->get_recordset_sql($sql, $params);
return groups_calculate_role_people($rs, $context);
}
/**
* 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.
*
* @param object $rs The record set (may be false)
* @param int $contextid ID of course context
* @return array As described in groups_get_members_by_role
*/
2008-06-01 13:09:04 +00:00
function groups_calculate_role_people($rs, $context) {
global $CFG, $DB;
if (!$rs) {
return array();
}
2008-06-01 13:09:04 +00:00
$roles = $DB->get_records_menu('role', null, 'name', 'id, name');
$aliasnames = role_fix_names($roles, $context);
// Array of all involved roles
2008-06-01 13:09:04 +00:00
$roles = array();
// Array of all retrieved users
2008-06-01 13:09:04 +00:00
$users = array();
// Fill arrays
2008-06-01 13:09:04 +00:00
foreach ($rs as $rec) {
// 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)) {
// User data includes all the optional fields, but not any of the
// stuff we added to get the role details
$userdata=clone($rec);
unset($userdata->roleid);
unset($userdata->roleshortname);
unset($userdata->rolename);
unset($userdata->userid);
2008-06-01 13:09:04 +00:00
$userdata->id = $rec->userid;
// 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;
}
// If user has a role...
2008-06-01 13:09:04 +00:00
if (!is_null($rec->roleid)) {
// Create information about role if this is a new one
2008-06-01 13:09:04 +00:00
if (!array_key_exists($rec->roleid,$roles)) {
$roledata = new object();
$roledata->id = $rec->roleid;
$roledata->shortname = $rec->roleshortname;
if (array_key_exists($rec->roleid, $aliasnames)) {
$roledata->name = $aliasnames[$rec->roleid];
} else {
2008-06-01 13:09:04 +00:00
$roledata->name = $rec->rolename;
}
2008-06-01 13:09:04 +00:00
$roledata->users = array();
$roles[$roledata->id] = $roledata;
}
// Record that user has role
$users[$rec->userid]->roles[] = $roles[$rec->roleid];
}
}
2008-06-01 13:09:04 +00:00
$rs->close();
// Return false if there weren't any users
2008-06-01 13:09:04 +00:00
if (count($users)==0) {
return false;
}
// Add pseudo-role for multiple roles
2008-06-01 13:09:04 +00:00
$roledata = new object();
$roledata->name = get_string('multipleroles','role');
$roledata->users = array();
$roles['*'] = $roledata;
// Now we rearrange the data to store users by role
2008-06-01 13:09:04 +00:00
foreach ($users as $userid=>$userdata) {
$rolecount = count($userdata->roles);
if ($rolecount==0) {
debugging("Unexpected: user $userid is missing roles");
} else if($rolecount>1) {
2008-06-01 13:09:04 +00:00
$roleid = '*';
} else {
2008-06-01 13:09:04 +00:00
$roleid = $userdata->roles[0]->id;
}
2008-06-01 13:09:04 +00:00
$roles[$roleid]->users[$userid] = $userdata;
}
// Delete roles not used
2008-06-01 13:09:04 +00:00
foreach ($roles as $key=>$roledata) {
if (count($roledata->users)===0) {
unset($roles[$key]);
}
}
// Return list of roles containing their users
return $roles;
}
?>