mirror of
https://github.com/codeninjasllc/discourse.git
synced 2024-11-27 09:36:19 -05:00
FEATURE: Allow manual assignment of related post to badge
PERF: clean up performance of user badges admin when large number of badges exist
This commit is contained in:
parent
ff842758e1
commit
fe578f9944
9 changed files with 146 additions and 24 deletions
|
@ -13,6 +13,42 @@ export default Ember.ArrayController.extend({
|
||||||
sortProperties: ['granted_at'],
|
sortProperties: ['granted_at'],
|
||||||
sortAscending: false,
|
sortAscending: false,
|
||||||
|
|
||||||
|
groupedBadges: function(){
|
||||||
|
const badges = this.get('model');
|
||||||
|
|
||||||
|
var grouped = _.groupBy(badges, badge => badge.badge_id);
|
||||||
|
|
||||||
|
var expanded = [];
|
||||||
|
const expandedBadges = badges.get('expandedBadges');
|
||||||
|
|
||||||
|
_(grouped).each(function(badges){
|
||||||
|
var lastGranted = badges[0].granted_at;
|
||||||
|
|
||||||
|
_.each(badges, function(badge) {
|
||||||
|
lastGranted = lastGranted < badge.granted_at ? badge.granted_at : lastGranted;
|
||||||
|
});
|
||||||
|
|
||||||
|
if(badges.length===1 || _.include(expandedBadges, badges[0].badge.id)){
|
||||||
|
_.each(badges, badge => expanded.push(badge));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var result = {
|
||||||
|
badge: badges[0].badge,
|
||||||
|
granted_at: lastGranted,
|
||||||
|
badges: badges,
|
||||||
|
count: badges.length,
|
||||||
|
grouped: true
|
||||||
|
};
|
||||||
|
|
||||||
|
expanded.push(result);
|
||||||
|
});
|
||||||
|
|
||||||
|
return _(expanded).sortBy(group => group.granted_at).reverse().value();
|
||||||
|
|
||||||
|
|
||||||
|
}.property('model', 'model.@each', 'model.expandedBadges.@each'),
|
||||||
|
|
||||||
/**
|
/**
|
||||||
Array of badges that have not been granted to this user.
|
Array of badges that have not been granted to this user.
|
||||||
|
|
||||||
|
@ -45,6 +81,12 @@ export default Ember.ArrayController.extend({
|
||||||
|
|
||||||
actions: {
|
actions: {
|
||||||
|
|
||||||
|
expandGroup: function(userBadge){
|
||||||
|
const model = this.get('model');
|
||||||
|
model.set('expandedBadges', model.get('expandedBadges') || []);
|
||||||
|
model.get('expandedBadges').pushObject(userBadge.badge.id);
|
||||||
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
Grant the selected badge to the user.
|
Grant the selected badge to the user.
|
||||||
|
|
||||||
|
@ -53,7 +95,8 @@ export default Ember.ArrayController.extend({
|
||||||
**/
|
**/
|
||||||
grantBadge: function(badgeId) {
|
grantBadge: function(badgeId) {
|
||||||
var self = this;
|
var self = this;
|
||||||
Discourse.UserBadge.grant(badgeId, this.get('user.username')).then(function(userBadge) {
|
Discourse.UserBadge.grant(badgeId, this.get('user.username'), this.get('badgeReason')).then(function(userBadge) {
|
||||||
|
self.set('badgeReason', '');
|
||||||
self.pushObject(userBadge);
|
self.pushObject(userBadge);
|
||||||
Ember.run.next(function() {
|
Ember.run.next(function() {
|
||||||
// Update the selected badge ID after the combobox has re-rendered.
|
// Update the selected badge ID after the combobox has re-rendered.
|
||||||
|
|
|
@ -9,40 +9,53 @@
|
||||||
{{#loading-spinner condition=loading}}
|
{{#loading-spinner condition=loading}}
|
||||||
<div class='admin-container user-badges'>
|
<div class='admin-container user-badges'>
|
||||||
<h2>{{i18n 'admin.badges.grant_badge'}}</h2>
|
<h2>{{i18n 'admin.badges.grant_badge'}}</h2>
|
||||||
|
<br>
|
||||||
{{#if noBadges}}
|
{{#if noBadges}}
|
||||||
<p>{{i18n 'admin.badges.no_badges'}}</p>
|
<p>{{i18n 'admin.badges.no_badges'}}</p>
|
||||||
{{else}}
|
{{else}}
|
||||||
<br>
|
<form class="form-horizontal">
|
||||||
|
<div>
|
||||||
|
<label>{{i18n 'admin.badges.badge'}}</label>
|
||||||
{{combo-box valueAttribute="id" value=controller.selectedBadgeId content=controller.grantableBadges}}
|
{{combo-box valueAttribute="id" value=controller.selectedBadgeId content=controller.grantableBadges}}
|
||||||
|
</div>
|
||||||
|
<label>
|
||||||
|
<label>{{i18n 'admin.badges.reason'}}</label>
|
||||||
|
{{input type="text" value=badgeReason}}<br><small>{{i18n 'admin.badges.reason_help'}}</small>
|
||||||
|
</label>
|
||||||
<button class='btn btn-primary' {{action "grantBadge" controller.selectedBadgeId}}>{{i18n 'admin.badges.grant'}}</button>
|
<button class='btn btn-primary' {{action "grantBadge" controller.selectedBadgeId}}>{{i18n 'admin.badges.grant'}}</button>
|
||||||
|
</form>
|
||||||
{{/if}}
|
{{/if}}
|
||||||
|
|
||||||
<br>
|
<table id='user-badges'>
|
||||||
<br>
|
|
||||||
|
|
||||||
<h2>{{i18n 'admin.badges.granted_badges'}}</h2>
|
|
||||||
<br>
|
|
||||||
|
|
||||||
<table>
|
|
||||||
<tr>
|
<tr>
|
||||||
<th>{{i18n 'admin.badges.badge'}}</th>
|
<th>{{i18n 'admin.badges.badge'}}</th>
|
||||||
<th>{{i18n 'admin.badges.granted_by'}}</th>
|
<th>{{i18n 'admin.badges.granted_by'}}</th>
|
||||||
|
<th class='reason'>{{i18n 'admin.badges.reason'}}</th>
|
||||||
<th>{{i18n 'admin.badges.granted_at'}}</th>
|
<th>{{i18n 'admin.badges.granted_at'}}</th>
|
||||||
<th></th>
|
<th></th>
|
||||||
</tr>
|
</tr>
|
||||||
|
|
||||||
{{#each}}
|
{{#each userBadge in groupedBadges}}
|
||||||
<tr>
|
<tr>
|
||||||
<td>{{user-badge badge=badge}}</td>
|
<td>{{user-badge badge=userBadge.badge count=userBadge.count}}</td>
|
||||||
<td>
|
<td>
|
||||||
{{#link-to 'adminUser' badge.granted_by}}
|
{{#link-to 'adminUser' userBadge.badge.granted_by}}
|
||||||
{{avatar granted_by imageSize="tiny"}}
|
{{avatar userBadge.granted_by imageSize="tiny"}}
|
||||||
{{granted_by.username}}
|
{{userBadge.granted_by.username}}
|
||||||
{{/link-to}}
|
{{/link-to}}
|
||||||
</td>
|
</td>
|
||||||
<td>{{age-with-tooltip granted_at}}</td>
|
<td class='reason'>
|
||||||
|
{{#if userBadge.postUrl}}
|
||||||
|
<a href="{{unbound userBadge.postUrl}}">{{userBadge.topic_title}}</a>
|
||||||
|
{{/if}}
|
||||||
|
</td>
|
||||||
|
<td>{{age-with-tooltip userBadge.granted_at}}</td>
|
||||||
<td>
|
<td>
|
||||||
<button class='btn' {{action "revokeBadge" this}}>{{i18n 'admin.badges.revoke'}}</button>
|
{{#if userBadge.grouped}}
|
||||||
|
<button class='btn' {{action "expandGroup" userBadge}}>{{{i18n 'admin.badges.expand'}}}</button>
|
||||||
|
{{else}}
|
||||||
|
<button class='btn btn-danger' {{action "revokeBadge" userBadge}}>{{i18n 'admin.badges.revoke'}}</button>
|
||||||
|
{{/if}}
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
{{else}}
|
{{else}}
|
||||||
|
|
|
@ -7,6 +7,11 @@
|
||||||
@module Discourse
|
@module Discourse
|
||||||
**/
|
**/
|
||||||
Discourse.UserBadge = Discourse.Model.extend({
|
Discourse.UserBadge = Discourse.Model.extend({
|
||||||
|
postUrl: function() {
|
||||||
|
if(this.get('topic_title')) {
|
||||||
|
return "/t/-/" + this.get('topic_id') + "/" + this.get('post_number');
|
||||||
|
}
|
||||||
|
}.property(), // avoid the extra bindings for now
|
||||||
/**
|
/**
|
||||||
Revoke this badge.
|
Revoke this badge.
|
||||||
|
|
||||||
|
@ -93,7 +98,7 @@ Discourse.UserBadge.reopenClass({
|
||||||
@returns {Promise} a promise that resolves to an array of `Discourse.UserBadge`.
|
@returns {Promise} a promise that resolves to an array of `Discourse.UserBadge`.
|
||||||
**/
|
**/
|
||||||
findByUsername: function(username, options) {
|
findByUsername: function(username, options) {
|
||||||
var url = "/users/" + username + "/badges_json.json";
|
var url = "/user-badges/" + username + ".json";
|
||||||
if (options && options.grouped) {
|
if (options && options.grouped) {
|
||||||
url += "?grouped=true";
|
url += "?grouped=true";
|
||||||
}
|
}
|
||||||
|
@ -128,12 +133,13 @@ Discourse.UserBadge.reopenClass({
|
||||||
@param {String} username username of the user to be granted the badge.
|
@param {String} username username of the user to be granted the badge.
|
||||||
@returns {Promise} a promise that resolves to an instance of `Discourse.UserBadge`.
|
@returns {Promise} a promise that resolves to an instance of `Discourse.UserBadge`.
|
||||||
**/
|
**/
|
||||||
grant: function(badgeId, username) {
|
grant: function(badgeId, username, reason) {
|
||||||
return Discourse.ajax("/user_badges", {
|
return Discourse.ajax("/user_badges", {
|
||||||
type: "POST",
|
type: "POST",
|
||||||
data: {
|
data: {
|
||||||
username: username,
|
username: username,
|
||||||
badge_id: badgeId
|
badge_id: badgeId,
|
||||||
|
reason: reason
|
||||||
}
|
}
|
||||||
}).then(function(json) {
|
}).then(function(json) {
|
||||||
return Discourse.UserBadge.createFromJson(json);
|
return Discourse.UserBadge.createFromJson(json);
|
||||||
|
|
|
@ -1440,3 +1440,9 @@ tr.not-activated {
|
||||||
.preview {
|
.preview {
|
||||||
margin-top: 5px;
|
margin-top: 5px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
table#user-badges {
|
||||||
|
.reason {
|
||||||
|
max-width: 200px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -25,8 +25,10 @@ class UserBadgesController < ApplicationController
|
||||||
end
|
end
|
||||||
|
|
||||||
user_badges = user_badges.includes(badge: [:badge_grouping, :badge_type])
|
user_badges = user_badges.includes(badge: [:badge_grouping, :badge_type])
|
||||||
|
.includes(post: :topic)
|
||||||
|
.includes(:granted_by)
|
||||||
|
|
||||||
render_serialized(user_badges, BasicUserBadgeSerializer, root: "user_badges")
|
render_serialized(user_badges, DetailedUserBadgeSerializer, root: "user_badges")
|
||||||
end
|
end
|
||||||
|
|
||||||
def create
|
def create
|
||||||
|
@ -39,9 +41,22 @@ class UserBadgesController < ApplicationController
|
||||||
end
|
end
|
||||||
|
|
||||||
badge = fetch_badge_from_params
|
badge = fetch_badge_from_params
|
||||||
user_badge = BadgeGranter.grant(badge, user, granted_by: current_user)
|
post_id = nil
|
||||||
|
|
||||||
render_serialized(user_badge, UserBadgeSerializer, root: "user_badge")
|
if params[:reason].present?
|
||||||
|
path = URI.parse(params[:reason]).path rescue nil
|
||||||
|
route = Rails.application.routes.recognize_path(path) if path
|
||||||
|
if route
|
||||||
|
topic_id = route[:topic_id].to_i
|
||||||
|
post_number = route[:post_number] || 1
|
||||||
|
|
||||||
|
post_id = Post.find_by(topic_id: topic_id, post_number: post_number).try(:id) if topic_id > 0
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
user_badge = BadgeGranter.grant(badge, user, granted_by: current_user, post_id: post_id)
|
||||||
|
|
||||||
|
render_serialized(user_badge, DetailedUserBadgeSerializer, root: "user_badge")
|
||||||
end
|
end
|
||||||
|
|
||||||
def destroy
|
def destroy
|
||||||
|
|
26
app/serializers/detailed_user_badge_serializer.rb
Normal file
26
app/serializers/detailed_user_badge_serializer.rb
Normal file
|
@ -0,0 +1,26 @@
|
||||||
|
class DetailedUserBadgeSerializer < BasicUserBadgeSerializer
|
||||||
|
has_one :granted_by
|
||||||
|
|
||||||
|
attributes :post_number, :topic_id, :topic_title
|
||||||
|
|
||||||
|
def include_post_number?
|
||||||
|
object.post
|
||||||
|
end
|
||||||
|
|
||||||
|
alias :include_topic_id? :include_post_number?
|
||||||
|
alias :include_topic_title? :include_post_number?
|
||||||
|
|
||||||
|
|
||||||
|
def post_number
|
||||||
|
object.post.post_number if object.post
|
||||||
|
end
|
||||||
|
|
||||||
|
def topic_id
|
||||||
|
object.post.topic_id if object.post
|
||||||
|
end
|
||||||
|
|
||||||
|
def topic_title
|
||||||
|
object.post.topic.title if object.post && object.post.topic
|
||||||
|
end
|
||||||
|
|
||||||
|
end
|
|
@ -2194,10 +2194,13 @@ en:
|
||||||
modal_title: Badge Groupings
|
modal_title: Badge Groupings
|
||||||
granted_by: Granted By
|
granted_by: Granted By
|
||||||
granted_at: Granted At
|
granted_at: Granted At
|
||||||
|
reason_help: (A link to a post or topic)
|
||||||
save: Save
|
save: Save
|
||||||
delete: Delete
|
delete: Delete
|
||||||
delete_confirm: Are you sure you want to delete this badge?
|
delete_confirm: Are you sure you want to delete this badge?
|
||||||
revoke: Revoke
|
revoke: Revoke
|
||||||
|
reason: Reason
|
||||||
|
expand: Expand …
|
||||||
revoke_confirm: Are you sure you want to revoke this badge?
|
revoke_confirm: Are you sure you want to revoke this badge?
|
||||||
edit_badges: Edit Badges
|
edit_badges: Edit Badges
|
||||||
grant_badge: Grant Badge
|
grant_badge: Grant Badge
|
||||||
|
|
|
@ -261,7 +261,7 @@ Discourse::Application.routes.draw do
|
||||||
get "users/by-external/:external_id" => "users#show"
|
get "users/by-external/:external_id" => "users#show"
|
||||||
get "users/:username/flagged-posts" => "users#show", constraints: {username: USERNAME_ROUTE_FORMAT}
|
get "users/:username/flagged-posts" => "users#show", constraints: {username: USERNAME_ROUTE_FORMAT}
|
||||||
get "users/:username/deleted-posts" => "users#show", constraints: {username: USERNAME_ROUTE_FORMAT}
|
get "users/:username/deleted-posts" => "users#show", constraints: {username: USERNAME_ROUTE_FORMAT}
|
||||||
get "users/:username/badges_json" => "user_badges#username"
|
get "user-badges/:username" => "user_badges#username"
|
||||||
|
|
||||||
post "user_avatar/:username/refresh_gravatar" => "user_avatars#refresh_gravatar"
|
post "user_avatar/:username/refresh_gravatar" => "user_avatars#refresh_gravatar"
|
||||||
get "letter_avatar/:username/:size/:version.png" => "user_avatars#show_letter", format: false, constraints: { hostname: /[\w\.-]+/ }
|
get "letter_avatar/:username/:size/:version.png" => "user_avatars#show_letter", format: false, constraints: { hostname: /[\w\.-]+/ }
|
||||||
|
|
|
@ -63,13 +63,23 @@ describe UserBadgesController do
|
||||||
|
|
||||||
it 'grants badges from staff' do
|
it 'grants badges from staff' do
|
||||||
admin = Fabricate(:admin)
|
admin = Fabricate(:admin)
|
||||||
|
post = create_post
|
||||||
|
|
||||||
log_in_user admin
|
log_in_user admin
|
||||||
|
|
||||||
StaffActionLogger.any_instance.expects(:log_badge_grant).once
|
StaffActionLogger.any_instance.expects(:log_badge_grant).once
|
||||||
xhr :post, :create, badge_id: badge.id, username: user.username
|
|
||||||
|
xhr :post, :create, badge_id: badge.id,
|
||||||
|
username: user.username,
|
||||||
|
reason: Discourse.base_url + post.url
|
||||||
|
|
||||||
expect(response.status).to eq(200)
|
expect(response.status).to eq(200)
|
||||||
|
|
||||||
user_badge = UserBadge.find_by(user: user, badge: badge)
|
user_badge = UserBadge.find_by(user: user, badge: badge)
|
||||||
|
|
||||||
expect(user_badge).to be_present
|
expect(user_badge).to be_present
|
||||||
expect(user_badge.granted_by).to eq(admin)
|
expect(user_badge.granted_by).to eq(admin)
|
||||||
|
expect(user_badge.post_id).to eq(post.id)
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'does not grant badges from regular api calls' do
|
it 'does not grant badges from regular api calls' do
|
||||||
|
|
Loading…
Reference in a new issue