changes to manage/users

v1.18.x
barisusakli 8 years ago
parent e4d2c54742
commit 9a55498fc1

@ -2,12 +2,10 @@
/* global config, socket, define, templates, bootbox, app, ajaxify */ /* global config, socket, define, templates, bootbox, app, ajaxify */
define('admin/manage/users', ['admin/modules/selectable', 'translator'], function(selectable, translator) { define('admin/manage/users', ['translator'], function(translator) {
var Users = {}; var Users = {};
Users.init = function() { Users.init = function() {
selectable.enable('#users-container', '.users-box');
var navPills = $('.nav-pills li'); var navPills = $('.nav-pills li');
var pathname = window.location.pathname; var pathname = window.location.pathname;
if (!navPills.find('a[href="' + pathname + '"]').length) { if (!navPills.find('a[href="' + pathname + '"]').length) {
@ -17,25 +15,29 @@ define('admin/manage/users', ['admin/modules/selectable', 'translator'], functio
function getSelectedUids() { function getSelectedUids() {
var uids = []; var uids = [];
$('#users-container .users-box.ui-selected').each(function() {
uids.push(this.getAttribute('data-uid')); $('.users-table [component="user/select/single"]').each(function() {
if ($(this).is(':checked')) {
uids.push($(this).attr('data-uid'));
}
}); });
return uids; return uids;
} }
function update(className, state) { function update(className, state) {
$('#users-container .users-box.ui-selected .labels').find(className).each(function() { $('.users-table [component="user/select/single"]:checked').parents('.user-row').find(className).each(function() {
$(this).toggleClass('hide', !state); $(this).toggleClass('hidden', !state);
}); });
} }
function unselectAll() { function unselectAll() {
$('#users-container .users-box.ui-selected').removeClass('ui-selected'); $('.users-table [component="user/select/single"]').prop('checked', false);
$('.users-table [component="user/select/all"]').prop('checked', false);
} }
function removeSelected() { function removeSelected() {
$('#users-container .users-box.ui-selected').remove(); $('.users-table [component="user/select/single"]:checked').parents('.user-row').remove();
} }
function done(successMessage, className, flag) { function done(successMessage, className, flag) {
@ -51,6 +53,14 @@ define('admin/manage/users', ['admin/modules/selectable', 'translator'], functio
}; };
} }
$('[component="user/select/all"]').on('click', function() {
if ($(this).is(':checked')) {
$('.users-table [component="user/select/single"]').prop('checked', true);
} else {
$('.users-table [component="user/select/single"]').prop('checked', false);
}
});
$('.ban-user').on('click', function() { $('.ban-user').on('click', function() {
var uids = getSelectedUids(); var uids = getSelectedUids();
if (!uids.length) { if (!uids.length) {
@ -164,9 +174,18 @@ define('admin/manage/users', ['admin/modules/selectable', 'translator'], functio
} }
bootbox.confirm('Do you want to validate email(s) of these user(s)?', function(confirm) { bootbox.confirm('Do you want to validate email(s) of these user(s)?', function(confirm) {
if (confirm) { if (!confirm) {
socket.emit('admin.user.validateEmail', uids, done('Emails validated', '.notvalidated', false)); return;
} }
socket.emit('admin.user.validateEmail', uids, function(err) {
if (err) {
return app.alertError(err.message);
}
app.alertSuccess('Emails validated');
update('.notvalidated', false);
update('.validated', true);
unselectAll();
});
}); });
}); });
@ -299,8 +318,6 @@ define('admin/manage/users', ['admin/modules/selectable', 'translator'], functio
var timeoutId = 0; var timeoutId = 0;
$('#search-user-name, #search-user-email, #search-user-ip').on('keyup', function() { $('#search-user-name, #search-user-email, #search-user-ip').on('keyup', function() {
if (timeoutId !== 0) { if (timeoutId !== 0) {
clearTimeout(timeoutId); clearTimeout(timeoutId);
@ -319,23 +336,25 @@ define('admin/manage/users', ['admin/modules/selectable', 'translator'], functio
} }
templates.parse('admin/manage/users', 'users', data, function(html) { templates.parse('admin/manage/users', 'users', data, function(html) {
$('#users-container').html(html).find('.timeago').timeago(); html = $(html);
$('.users-table tr').not(':first').remove();
$('.users-table tr').first().after(html);
html.find('.timeago').timeago();
$('.fa-spinner').addClass('hidden'); $('.fa-spinner').addClass('hidden');
if (data && data.users.length === 0) { if (data && data.users.length === 0) {
$('#user-notfound-notify').html('User not found!') $('#user-notfound-notify').html('User not found!')
.show() .removeClass('hide')
.addClass('label-danger') .addClass('label-danger')
.removeClass('label-success'); .removeClass('label-success');
} else { } else {
$('#user-notfound-notify').html(data.users.length + ' user' + (data.users.length > 1 ? 's' : '') + ' found! Search took ' + data.timing + ' ms.') $('#user-notfound-notify').html(data.users.length + ' user' + (data.users.length > 1 ? 's' : '') + ' found! Search took ' + data.timing + ' ms.')
.show() .removeClass('hide')
.addClass('label-success') .addClass('label-success')
.removeClass('label-danger'); .removeClass('label-danger');
} }
selectable.enable('#users-container', '.users-box');
}); });
}); });
}, 250); }, 250);

@ -1,6 +1,8 @@
"use strict"; "use strict";
var async = require('async'); var async = require('async');
var validator = require('validator');
var user = require('../../user'); var user = require('../../user');
var meta = require('../../meta'); var meta = require('../../meta');
var db = require('../../database'); var db = require('../../database');
@ -10,6 +12,9 @@ var plugins = require('../../plugins');
var usersController = {}; var usersController = {};
var userFields = ['uid', 'username', 'userslug', 'email', 'postcount', 'joindate', 'banned',
'reputation', 'picture', 'flags', 'lastonline', 'email:confirmed'];
usersController.search = function(req, res, next) { usersController.search = function(req, res, next) {
res.render('admin/manage/users', { res.render('admin/manage/users', {
search_display: '', search_display: '',
@ -18,67 +23,29 @@ usersController.search = function(req, res, next) {
}; };
usersController.sortByJoinDate = function(req, res, next) { usersController.sortByJoinDate = function(req, res, next) {
getUsers('users:joindate', 'latest', req, res, next); getUsers('users:joindate', 'latest', undefined, undefined, req, res, next);
}; };
usersController.notValidated = function(req, res, next) { usersController.notValidated = function(req, res, next) {
getUsers('users:notvalidated', 'notvalidated', req, res, next); getUsers('users:notvalidated', 'notvalidated', undefined, undefined, req, res, next);
}; };
usersController.noPosts = function(req, res, next) { usersController.noPosts = function(req, res, next) {
getUsersByScore('users:postcount', 'noposts', 0, 0, req, res, next); getUsers('users:postcount', 'noposts', '-inf', 0, req, res, next);
}; };
usersController.flagged = function(req, res, next) { usersController.flagged = function(req, res, next) {
getUsersByScore('users:flags', 'mostflags', 1, '+inf', req, res, next); getUsers('users:flags', 'mostflags', 1, '+inf', req, res, next);
}; };
usersController.inactive = function(req, res, next) { usersController.inactive = function(req, res, next) {
var timeRange = 1000 * 60 * 60 * 24 * 30 * (parseInt(req.query.months, 10) || 3); var timeRange = 1000 * 60 * 60 * 24 * 30 * (parseInt(req.query.months, 10) || 3);
var cutoff = Date.now() - timeRange; var cutoff = Date.now() - timeRange;
getUsersByScore('users:online', 'inactive', '-inf', cutoff, req, res, next); getUsers('users:online', 'inactive', '-inf', cutoff, req, res, next);
}; };
function getUsersByScore(set, section, min, max, req, res, callback) {
var page = parseInt(req.query.page, 10) || 1;
var resultsPerPage = 25;
var start = Math.max(0, page - 1) * resultsPerPage;
var count = 0;
async.waterfall([
function (next) {
async.parallel({
count: function (next) {
db.sortedSetCount(set, min, max, next);
},
uids: function (next) {
db.getSortedSetRevRangeByScore(set, start, resultsPerPage, max, min, next);
}
}, next);
},
function (results, next) {
count = results.count;
user.getUsers(results.uids, req.uid, next);
}
], function(err, users) {
if (err) {
return callback(err);
}
users = users.filter(function(user) {
return user && parseInt(user.uid, 10);
});
var data = {
users: users,
page: page,
pageCount: Math.ceil(count / resultsPerPage)
};
data[section] = true;
render(req, res, data);
});
}
usersController.banned = function(req, res, next) { usersController.banned = function(req, res, next) {
getUsers('users:banned', 'banned', req, res, next); getUsers('users:banned', 'banned', undefined, undefined, req, res, next);
}; };
usersController.registrationQueue = function(req, res, next) { usersController.registrationQueue = function(req, res, next) {
@ -141,18 +108,34 @@ usersController.registrationQueue = function(req, res, next) {
}); });
}; };
function getUsers(set, section, req, res, next) { function getUsers(set, section, min, max, req, res, next) {
var page = parseInt(req.query.page, 10) || 1; var page = parseInt(req.query.page, 10) || 1;
var resultsPerPage = 25; var resultsPerPage = 50;
var start = Math.max(0, page - 1) * resultsPerPage; var start = Math.max(0, page - 1) * resultsPerPage;
var stop = start + resultsPerPage - 1; var stop = start + resultsPerPage - 1;
var byScore = min !== undefined && max !== undefined;
async.parallel({ async.parallel({
count: function(next) { count: function(next) {
db.sortedSetCard(set, next); if (byScore) {
db.sortedSetCount(set, min, max, next);
} else {
db.sortedSetCard(set, next);
}
}, },
users: function(next) { users: function(next) {
user.getUsersFromSet(set, req.uid, start, stop, next); async.waterfall([
function(next) {
if (byScore) {
db.getSortedSetRevRangeByScore(set, start, resultsPerPage, max, min, next);
} else {
user.getUidsFromSet(set, start, stop, next);
}
},
function(uids, next) {
user.getUsersWithFields(uids, userFields, req.uid, next);
}
], next);
} }
}, function(err, results) { }, function(err, results) {
if (err) { if (err) {
@ -160,6 +143,7 @@ function getUsers(set, section, req, res, next) {
} }
results.users = results.users.filter(function(user) { results.users = results.users.filter(function(user) {
user.email = validator.escape(String(user.email || ''));
return user && parseInt(user.uid, 10); return user && parseInt(user.uid, 10);
}); });
var data = { var data = {

@ -198,7 +198,7 @@ User.search = function(socket, data, callback) {
return user && user.uid; return user && user.uid;
}); });
user.getUsersFields(uids, ['email', 'flags'], function(err, userInfo) { user.getUsersFields(uids, ['email', 'flags', 'lastonline', 'joindate'], function(err, userInfo) {
if (err) { if (err) {
return callback(err); return callback(err);
} }
@ -207,6 +207,8 @@ User.search = function(socket, data, callback) {
if (user && userInfo[index]) { if (user && userInfo[index]) {
user.email = validator.escape(String(userInfo[index].email || '')); user.email = validator.escape(String(userInfo[index].email || ''));
user.flags = userInfo[index].flags || 0; user.flags = userInfo[index].flags || 0;
user.lastonlineISO = userInfo[index].lastonlineISO;
user.joindateISO = userInfo[index].joindateISO;
} }
}); });

@ -8,7 +8,6 @@ var db = require('./database');
var topics = require('./topics'); var topics = require('./topics');
var privileges = require('./privileges'); var privileges = require('./privileges');
var meta = require('./meta'); var meta = require('./meta');
var utils = require('../public/src/utils');
(function(User) { (function(User) {
@ -90,10 +89,7 @@ var utils = require('../public/src/utils');
], callback); ], callback);
}; };
User.getUsers = function(uids, uid, callback) { User.getUsersWithFields = function(uids, fields, uid, callback) {
var fields = ['uid', 'username', 'userslug', 'picture', 'status', 'flags',
'banned', 'banned:expire', 'joindate', 'postcount', 'reputation', 'email:confirmed', 'lastonline'];
async.waterfall([ async.waterfall([
function (next) { function (next) {
plugins.fireHook('filter:users.addFields', {fields: fields}, next); plugins.fireHook('filter:users.addFields', {fields: fields}, next);
@ -116,13 +112,11 @@ var utils = require('../public/src/utils');
results.userData.forEach(function(user, index) { results.userData.forEach(function(user, index) {
if (user) { if (user) {
user.status = User.getStatus(user); user.status = User.getStatus(user);
user.joindateISO = utils.toISOString(user.joindate);
user.administrator = results.isAdmin[index]; user.administrator = results.isAdmin[index];
user.banned = parseInt(user.banned, 10) === 1; user.banned = parseInt(user.banned, 10) === 1;
user.banned_until = parseInt(user['banned:expire'], 10) || 0; user.banned_until = parseInt(user['banned:expire'], 10) || 0;
user.banned_until_readable = user.banned_until ? new Date(user.banned_until).toString() : 'Not Banned'; user.banned_until_readable = user.banned_until ? new Date(user.banned_until).toString() : 'Not Banned';
user['email:confirmed'] = parseInt(user['email:confirmed'], 10) === 1; user['email:confirmed'] = parseInt(user['email:confirmed'], 10) === 1;
user.lastonlineISO = utils.toISOString(user.lastonline) || user.joindateISO;
} }
}); });
plugins.fireHook('filter:userlist.get', {users: results.userData, uid: uid}, next); plugins.fireHook('filter:userlist.get', {users: results.userData, uid: uid}, next);
@ -133,6 +127,13 @@ var utils = require('../public/src/utils');
], callback); ], callback);
}; };
User.getUsers = function(uids, uid, callback) {
var fields = ['uid', 'username', 'userslug', 'picture', 'status', 'flags',
'banned', 'banned:expire', 'joindate', 'postcount', 'reputation', 'email:confirmed', 'lastonline'];
User.getUsersWithFields(uids, fields, uid, callback);
};
User.getStatus = function(userData) { User.getStatus = function(userData) {
var isOnline = (Date.now() - parseInt(userData.lastonline, 10)) < 300000; var isOnline = (Date.now() - parseInt(userData.lastonline, 10)) < 300000;
return isOnline ? (userData.status || 'online') : 'offline'; return isOnline ? (userData.status || 'online') : 'offline';

@ -6,6 +6,7 @@ var winston = require('winston');
var db = require('../database'); var db = require('../database');
var plugins = require('../plugins'); var plugins = require('../plugins');
var utils = require('../../public/src/utils');
module.exports = function(User) { module.exports = function(User) {
@ -136,6 +137,14 @@ module.exports = function(User) {
return cur + next.charCodeAt(); return cur + next.charCodeAt();
}, 0) % iconBackgrounds.length]; }, 0) % iconBackgrounds.length];
} }
if (user.hasOwnProperty('joindate')) {
user.joindateISO = utils.toISOString(user.joindate);
}
if (user.hasOwnProperty('lastonline')) {
user.lastonlineISO = utils.toISOString(user.lastonline) || user.joindateISO;
}
}); });
plugins.fireHook('filter:users.get', users, callback); plugins.fireHook('filter:users.get', users, callback);

@ -1,5 +1,6 @@
<div class="manage-users"> <div class="manage-users">
<div class="col-lg-9">
<div class="col-lg-12">
<div class="panel panel-default"> <div class="panel panel-default">
<div class="panel-heading"><i class="fa fa-user"></i> Users</div> <div class="panel-heading"><i class="fa fa-user"></i> Users</div>
<div class="panel-body"> <div class="panel-body">
@ -12,7 +13,6 @@
<li><a href='{config.relative_path}/admin/manage/users/banned'>Banned</a></li> <li><a href='{config.relative_path}/admin/manage/users/banned'>Banned</a></li>
<li><a href='{config.relative_path}/admin/manage/users/search'>User Search</a></li> <li><a href='{config.relative_path}/admin/manage/users/search'>User Search</a></li>
<div class="btn-group pull-right"> <div class="btn-group pull-right">
<button class="btn btn-default dropdown-toggle" data-toggle="dropdown" type="button">Edit <span class="caret"></span></button> <button class="btn btn-default dropdown-toggle" data-toggle="dropdown" type="button">Edit <span class="caret"></span></button>
<ul class="dropdown-menu"> <ul class="dropdown-menu">
@ -33,6 +33,16 @@
<li><a href="#" class="delete-user-and-content"><i class="fa fa-fw fa-trash-o"></i> Delete User(s) and Content</a></li> <li><a href="#" class="delete-user-and-content"><i class="fa fa-fw fa-trash-o"></i> Delete User(s) and Content</a></li>
</ul> </ul>
</div> </div>
<a target="_blank" href="{config.relative_path}/api/admin/users/csv" class="btn btn-primary pull-right">Download CSV</a>
<!-- IF inviteOnly -->
<!-- IF loggedIn -->
<button component="user/invite" class="btn btn-success form-control"><i class="fa fa-users"></i> Invite</button>
<!-- ENDIF loggedIn -->
<!-- ENDIF inviteOnly -->
<button id="createUser" class="btn btn-primary pull-right">New User</button>
</ul> </ul>
<br /> <br />
@ -50,68 +60,51 @@
<i class="fa fa-spinner fa-spin hidden"></i> <i class="fa fa-spinner fa-spin hidden"></i>
<span id="user-notfound-notify" class="label label-danger hide">User not found!</span><br/> <span id="user-notfound-notify" class="label label-danger hide">User not found!</span><br/>
</div> </div>
<!-- IF inactive --> <!-- IF inactive -->
<a href="{config.relative_path}/admin/manage/users/inactive?months=3" class="btn btn-default">3 months</a> <a href="{config.relative_path}/admin/manage/users/inactive?months=3" class="btn btn-default">3 months</a>
<a href="{config.relative_path}/admin/manage/users/inactive?months=6" class="btn btn-default">6 months</a> <a href="{config.relative_path}/admin/manage/users/inactive?months=6" class="btn btn-default">6 months</a>
<a href="{config.relative_path}/admin/manage/users/inactive?months=12" class="btn btn-default">12 months</a> <a href="{config.relative_path}/admin/manage/users/inactive?months=12" class="btn btn-default">12 months</a>
<!-- ENDIF inactive --> <!-- ENDIF inactive -->
<table class="table table-striped users-table">
<ul id="users-container"> <tr>
<th><input component="user/select/all" type="checkbox"/></th>
<th>uid</th>
<th>username</th>
<th>admin</th>
<th>email</th>
<th class="text-right">postcount</th>
<th class="text-right">reputation</th>
<th class="text-right">flags</th>
<th>joined</th>
<th>last online</th>
<th>banned</th>
</tr>
<!-- BEGIN users --> <!-- BEGIN users -->
<div class="users-box" data-uid="{users.uid}" data-username="{users.username}"> <tr class="user-row">
<div class="user-image"> <th><input component="user/select/single" data-uid="{users.uid}" type="checkbox"/></th>
<!-- IF users.picture --> <td class="text-right">{users.uid}</td>
<img src="{users.picture}" class="img-thumbnail"/> <td><a href="{config.relative_path}/user/{users.userslug}">{users.username}</a></td>
<!-- ELSE --> <td class="text-center"><i class="administrator fa fa-shield text-success<!-- IF !users.administrator --> hidden<!-- ENDIF !users.administrator -->"></td>
<div class="user-icon" style="background-color: {users.icon:bgColor};">{users.icon:text}</div> <td>
<!-- ENDIF users.picture --> <!-- IF config.requireEmailConfirmation -->
<div class="labels">
<!-- IF config.requireEmailConfirmation --> <i class="validated fa fa-check text-success<!-- IF !users.email:confirmed --> hidden<!-- ENDIF !users.email:confirmed -->" title="validated"></i>
<!-- IF !users.email:confirmed --> <i class="notvalidated fa fa-times text-danger<!-- IF users.email:confirmed --> hidden<!-- ENDIF users.email:confirmed -->" title="not validated"></i>
<span class="notvalidated label label-danger">Not Validated</span> <!-- ENDIF config.requireEmailConfirmation --> {users.email}</td>
<!-- ENDIF !users.email:confirmed --> <td class="text-right">{users.postcount}</td>
<!-- ENDIF config.requireEmailConfirmation --> <td class="text-right">{users.reputation}</td>
<span class="administrator label label-primary <!-- IF !users.administrator -->hide<!-- ENDIF !users.administrator -->">Admin</span> <td class="text-right"><!-- IF users.flags -->{users.flags}<!-- ELSE -->0<!-- ENDIF users.flags --></td>
<span class="ban label label-danger <!-- IF !users.banned -->hide<!-- ENDIF !users.banned -->">Banned<!-- IF users.banned_until --> <i class="fa fa-clock-o" title="Banned until {../banned_until_readable}"></i><!-- ENDIF users.banned_until --></span> <td><span class="timeago" title="{users.joindateISO}"></span></td>
</div> <td><span class="timeago" title="{users.lastonlineISO}"></span></td>
</div> <td class="text-center"><i class="ban fa fa-gavel text-danger<!-- IF !users.banned --> hidden<!-- ENDIF !users.banned -->"></i></td>
</tr>
<a href="{config.relative_path}/user/{users.userslug}" target="_blank">{users.username} ({users.uid})</a><br/>
<!-- IF users.email -->
<small><span title="{users.email}">{users.email}</span></small><br/>
<!-- ENDIF users.email -->
joined <span class="timeago" title="{users.joindateISO}"></span><br/>
login <span class="timeago" title="{users.lastonlineISO}"></span><br/>
posts {users.postcount}
<!-- IF users.flags -->
<div><small><span><i class="fa fa-flag"></i> <a href="{config.relative_path}/admin/manage/flags?byUsername={users.username}">{users.flags}</a></span></small></div>
<!-- ENDIF users.flags -->
</div>
<!-- END users --> <!-- END users -->
</ul> </table>
<!-- IMPORT partials/paginator.tpl --> <!-- IMPORT partials/paginator.tpl -->
</div> </div>
</div> </div>
</div> </div>
<div class="col-lg-3 acp-sidebar">
<div class="panel panel-default">
<div class="panel-heading">Users Control Panel</div>
<div class="panel-body">
<button id="createUser" class="btn btn-primary form-control">New User</button>
<a target="_blank" href="{config.relative_path}/api/admin/users/csv" class="btn btn-primary form-control">Download CSV</a>
<!-- IF inviteOnly -->
<!-- IF loggedIn -->
<button component="user/invite" class="btn btn-success form-control"><i class="fa fa-users"></i> Invite</button>
<!-- ENDIF loggedIn -->
<!-- ENDIF inviteOnly -->
</div>
</div>
</div>
</div> </div>

Loading…
Cancel
Save