feat(groups): user_groups lookup, admin assignment UI, public directory with DataTables
This commit is contained in:
15
application/_installation/04-create-table-user-groups.sql
Normal file
15
application/_installation/04-create-table-user-groups.sql
Normal file
@@ -0,0 +1,15 @@
|
||||
CREATE TABLE IF NOT EXISTS `huge`.`user_groups` (
|
||||
`group_id` TINYINT(1) NOT NULL COMMENT 'numeric user group id, matches users.user_account_type',
|
||||
`group_name` VARCHAR(64) COLLATE utf8_unicode_ci NOT NULL COMMENT 'human readable group name',
|
||||
PRIMARY KEY (`group_id`)
|
||||
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci COMMENT='user groups lookup';
|
||||
|
||||
INSERT INTO `huge`.`user_groups` (`group_id`, `group_name`) VALUES
|
||||
(1, 'Gast'),
|
||||
(2, 'Benutzer'),
|
||||
(3, 'Gruppe 3'),
|
||||
(4, 'Gruppe 4'),
|
||||
(5, 'Gruppe 5'),
|
||||
(6, 'Gruppe 6'),
|
||||
(7, 'Admin')
|
||||
ON DUPLICATE KEY UPDATE `group_name` = VALUES(`group_name`);
|
||||
@@ -20,7 +20,9 @@ class AdminController extends Controller
|
||||
public function index()
|
||||
{
|
||||
$this->View->render('admin/index', array(
|
||||
'users' => UserModel::getPublicProfilesOfAllUsers())
|
||||
'users' => UserModel::getPublicProfilesOfAllUsers(),
|
||||
'groups' => GroupModel::getAllGroups()
|
||||
)
|
||||
);
|
||||
}
|
||||
|
||||
@@ -32,4 +34,10 @@ class AdminController extends Controller
|
||||
|
||||
Redirect::to("admin");
|
||||
}
|
||||
|
||||
public function changeUserGroup()
|
||||
{
|
||||
GroupModel::setUserGroup(Request::post('user_id'), Request::post('group_id'));
|
||||
Redirect::to("admin");
|
||||
}
|
||||
}
|
||||
|
||||
16
application/controller/DirectoryController.php
Normal file
16
application/controller/DirectoryController.php
Normal file
@@ -0,0 +1,16 @@
|
||||
<?php
|
||||
|
||||
class DirectoryController extends Controller
|
||||
{
|
||||
public function __construct()
|
||||
{
|
||||
parent::__construct();
|
||||
}
|
||||
|
||||
public function index()
|
||||
{
|
||||
$this->View->render('directory/index', [
|
||||
'users' => UserModel::getUsersWithGroups()
|
||||
]);
|
||||
}
|
||||
}
|
||||
54
application/model/GroupModel.php
Normal file
54
application/model/GroupModel.php
Normal file
@@ -0,0 +1,54 @@
|
||||
<?php
|
||||
|
||||
class GroupModel
|
||||
{
|
||||
public static function getAllGroups()
|
||||
{
|
||||
$database = DatabaseFactory::getFactory()->getConnection();
|
||||
$sql = "SELECT group_id, group_name FROM user_groups ORDER BY group_id";
|
||||
$query = $database->prepare($sql);
|
||||
$query->execute();
|
||||
return $query->fetchAll();
|
||||
}
|
||||
|
||||
public static function getGroupNameById($group_id)
|
||||
{
|
||||
$database = DatabaseFactory::getFactory()->getConnection();
|
||||
$sql = "SELECT group_name FROM user_groups WHERE group_id = :gid LIMIT 1";
|
||||
$query = $database->prepare($sql);
|
||||
$query->execute(array(':gid' => $group_id));
|
||||
$row = $query->fetch();
|
||||
return $row ? $row->group_name : null;
|
||||
}
|
||||
|
||||
public static function setUserGroup($userId, $groupId)
|
||||
{
|
||||
if (!is_numeric($userId) || !is_numeric($groupId)) {
|
||||
return false;
|
||||
}
|
||||
|
||||
// Do not allow changing own group via admin UI to prevent lockout
|
||||
if ((int)$userId === (int)Session::get('user_id')) {
|
||||
Session::add('feedback_negative', Text::get('FEEDBACK_ACCOUNT_CANT_DELETE_SUSPEND_OWN'));
|
||||
return false;
|
||||
}
|
||||
|
||||
// Only allow groups that exist in lookup
|
||||
$database = DatabaseFactory::getFactory()->getConnection();
|
||||
$check = $database->prepare("SELECT 1 FROM user_groups WHERE group_id = :gid LIMIT 1");
|
||||
$check->execute([':gid' => $groupId]);
|
||||
if ($check->rowCount() !== 1) {
|
||||
return false;
|
||||
}
|
||||
|
||||
$query = $database->prepare("UPDATE users SET user_account_type = :gid WHERE user_id = :uid LIMIT 1");
|
||||
$query->execute([':gid' => $groupId, ':uid' => $userId]);
|
||||
|
||||
if ($query->rowCount() === 1) {
|
||||
Session::add('feedback_positive', 'Benutzergruppe aktualisiert.');
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
}
|
||||
@@ -19,7 +19,7 @@ class UserModel
|
||||
{
|
||||
$database = DatabaseFactory::getFactory()->getConnection();
|
||||
|
||||
$sql = "SELECT user_id, user_name, user_email, user_active, user_has_avatar, user_deleted FROM users";
|
||||
$sql = "SELECT user_id, user_name, user_email, user_active, user_has_avatar, user_deleted, user_account_type FROM users";
|
||||
$query = $database->prepare($sql);
|
||||
$query->execute();
|
||||
|
||||
@@ -39,11 +39,46 @@ class UserModel
|
||||
$all_users_profiles[$user->user_id]->user_active = $user->user_active;
|
||||
$all_users_profiles[$user->user_id]->user_deleted = $user->user_deleted;
|
||||
$all_users_profiles[$user->user_id]->user_avatar_link = (Config::get('USE_GRAVATAR') ? AvatarModel::getGravatarLinkByEmail($user->user_email) : AvatarModel::getPublicAvatarFilePathOfUser($user->user_has_avatar, $user->user_id));
|
||||
$all_users_profiles[$user->user_id]->user_account_type = $user->user_account_type;
|
||||
}
|
||||
|
||||
return $all_users_profiles;
|
||||
}
|
||||
|
||||
/**
|
||||
* Gets list of users including their group name via user_groups lookup.
|
||||
* @return array
|
||||
*/
|
||||
public static function getUsersWithGroups()
|
||||
{
|
||||
$database = DatabaseFactory::getFactory()->getConnection();
|
||||
|
||||
$sql = "SELECT u.user_id, u.user_name, u.user_email, u.user_active, u.user_has_avatar, u.user_deleted, u.user_account_type,
|
||||
g.group_name
|
||||
FROM users u
|
||||
LEFT JOIN user_groups g ON g.group_id = u.user_account_type";
|
||||
$query = $database->prepare($sql);
|
||||
$query->execute();
|
||||
|
||||
$result = [];
|
||||
foreach ($query->fetchAll() as $user) {
|
||||
array_walk_recursive($user, 'Filter::XSSFilter');
|
||||
|
||||
$obj = new stdClass();
|
||||
$obj->user_id = $user->user_id;
|
||||
$obj->user_name = $user->user_name;
|
||||
$obj->user_email = $user->user_email;
|
||||
$obj->user_active = $user->user_active;
|
||||
$obj->user_deleted = $user->user_deleted;
|
||||
$obj->user_account_type = $user->user_account_type;
|
||||
$obj->group_name = $user->group_name;
|
||||
$obj->user_avatar_link = (Config::get('USE_GRAVATAR') ? AvatarModel::getGravatarLinkByEmail($user->user_email) : AvatarModel::getPublicAvatarFilePathOfUser($user->user_has_avatar, $user->user_id));
|
||||
|
||||
$result[] = $obj;
|
||||
}
|
||||
return $result;
|
||||
}
|
||||
|
||||
/**
|
||||
* Gets a user's profile data, according to the given $user_id
|
||||
* @param int $user_id The user's id
|
||||
|
||||
@@ -24,6 +24,9 @@
|
||||
<li <?php if (View::checkForActiveController($filename, "profile")) { echo ' class="active" '; } ?> >
|
||||
<a href="<?php echo Config::get('URL'); ?>profile/index">Profiles</a>
|
||||
</li>
|
||||
<li <?php if (View::checkForActiveController($filename, "directory")) { echo ' class="active" '; } ?> >
|
||||
<a href="<?php echo Config::get('URL'); ?>directory/index">Benutzer</a>
|
||||
</li>
|
||||
<?php if (Session::userIsLoggedIn()) { ?>
|
||||
<li <?php if (View::checkForActiveController($filename, "dashboard")) { echo ' class="active" '; } ?> >
|
||||
<a href="<?php echo Config::get('URL'); ?>dashboard/index">Dashboard</a>
|
||||
|
||||
@@ -22,6 +22,7 @@
|
||||
<td>User's email</td>
|
||||
<td>Activated ?</td>
|
||||
<td>Link to user's profile</td>
|
||||
<td>Group</td>
|
||||
<td>suspension Time in days</td>
|
||||
<td>Soft delete</td>
|
||||
<td>Submit</td>
|
||||
@@ -41,6 +42,19 @@
|
||||
<td>
|
||||
<a href="<?= Config::get('URL') . 'profile/showProfile/' . $user->user_id; ?>">Profile</a>
|
||||
</td>
|
||||
<td>
|
||||
<form action="<?= Config::get('URL'); ?>admin/changeUserGroup" method="post">
|
||||
<input type="hidden" name="user_id" value="<?= $user->user_id; ?>" />
|
||||
<select name="group_id">
|
||||
<?php foreach ($this->groups as $group) { ?>
|
||||
<option value="<?= $group->group_id; ?>" <?= (isset($user->user_account_type) && (int)$user->user_account_type === (int)$group->group_id ? 'selected' : '') ?>>
|
||||
<?= (int)$group->group_id; ?> - <?= htmlspecialchars($group->group_name, ENT_QUOTES, 'UTF-8'); ?>
|
||||
</option>
|
||||
<?php } ?>
|
||||
</select>
|
||||
<input type="submit" value="Save" />
|
||||
</form>
|
||||
</td>
|
||||
<form action="<?= config::get("URL"); ?>admin/actionAccountSettings" method="post">
|
||||
<td><input type="number" name="suspension" /></td>
|
||||
<td><input type="checkbox" name="softDelete" <?php if ($user->user_deleted) { ?> checked <?php } ?> /></td>
|
||||
|
||||
54
application/view/directory/index.php
Normal file
54
application/view/directory/index.php
Normal file
@@ -0,0 +1,54 @@
|
||||
<div class="container">
|
||||
<h1>Benutzerverzeichnis</h1>
|
||||
|
||||
<div class="box">
|
||||
<?php $this->renderFeedbackMessages(); ?>
|
||||
|
||||
<table id="users-table" class="overview-table" style="width:100%">
|
||||
<thead>
|
||||
<tr>
|
||||
<td>Id</td>
|
||||
<td>Avatar</td>
|
||||
<td>Benutzername</td>
|
||||
<td>Email</td>
|
||||
<td>Aktiv?</td>
|
||||
<td>Gruppe</td>
|
||||
<td>Profil</td>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody>
|
||||
<?php foreach ($this->users as $user) { ?>
|
||||
<tr class="<?= ($user->user_active == 0 ? 'inactive' : 'active'); ?>">
|
||||
<td><?= $user->user_id; ?></td>
|
||||
<td class="avatar">
|
||||
<?php if (isset($user->user_avatar_link)) { ?>
|
||||
<img src="<?= $user->user_avatar_link; ?>"/>
|
||||
<?php } ?>
|
||||
</td>
|
||||
<td><?= $user->user_name; ?></td>
|
||||
<td><?= $user->user_email; ?></td>
|
||||
<td><?= ($user->user_active == 0 ? 'Nein' : 'Ja'); ?></td>
|
||||
<td><?= htmlspecialchars($user->group_name ?: $user->user_account_type, ENT_QUOTES, 'UTF-8'); ?></td>
|
||||
<td><a href="<?= Config::get('URL') . 'profile/showProfile/' . $user->user_id; ?>">Profil</a></td>
|
||||
</tr>
|
||||
<?php } ?>
|
||||
</tbody>
|
||||
</table>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- jQuery & DataTables CDN -->
|
||||
<link rel="stylesheet" href="https://cdn.datatables.net/1.13.8/css/jquery.dataTables.min.css">
|
||||
<script src="https://code.jquery.com/jquery-3.7.1.min.js"></script>
|
||||
<script src="https://cdn.datatables.net/1.13.8/js/jquery.dataTables.min.js"></script>
|
||||
<script>
|
||||
$(document).ready(function(){
|
||||
$('#users-table').DataTable({
|
||||
pageLength: 10,
|
||||
order: [[ 0, 'asc' ]],
|
||||
language: {
|
||||
url: 'https://cdn.datatables.net/plug-ins/1.13.8/i18n/de-DE.json'
|
||||
}
|
||||
});
|
||||
});
|
||||
</script>
|
||||
Reference in New Issue
Block a user