FEATURE: Bulk Unlisting of topics

This commit is contained in:
Robin Ward 2015-10-27 16:57:40 -04:00
parent 46ca66771b
commit 6b236d3c83
5 changed files with 77 additions and 46 deletions

View File

@ -13,12 +13,13 @@ addBulkButton('closeTopics', 'close_topics');
addBulkButton('archiveTopics', 'archive_topics'); addBulkButton('archiveTopics', 'archive_topics');
addBulkButton('showNotificationLevel', 'notification_level'); addBulkButton('showNotificationLevel', 'notification_level');
addBulkButton('resetRead', 'reset_read'); addBulkButton('resetRead', 'reset_read');
addBulkButton('unlistTopics', 'unlist_topics');
// Modal for performing bulk actions on topics // Modal for performing bulk actions on topics
export default Ember.ArrayController.extend(ModalFunctionality, { export default Ember.ArrayController.extend(ModalFunctionality, {
buttonRows: null, buttonRows: null,
onShow: function() { onShow() {
this.set('controllers.modal.modalClass', 'topic-bulk-actions-modal small'); this.set('controllers.modal.modalClass', 'topic-bulk-actions-modal small');
const buttonRows = []; const buttonRows = [];
@ -36,87 +37,80 @@ export default Ember.ArrayController.extend(ModalFunctionality, {
this.send('changeBulkTemplate', 'modal/bulk_actions_buttons'); this.send('changeBulkTemplate', 'modal/bulk_actions_buttons');
}, },
perform: function(operation) { perform(operation) {
this.set('loading', true); this.set('loading', true);
var self = this, const topics = this.get('model');
topics = this.get('model'); return Discourse.Topic.bulkOperation(this.get('model'), operation).then(result => {
return Discourse.Topic.bulkOperation(this.get('model'), operation).then(function(result) { this.set('loading', false);
self.set('loading', false);
if (result && result.topic_ids) { if (result && result.topic_ids) {
return result.topic_ids.map(function (t) { return result.topic_ids.map(t => topics.findBy('id', t));
return topics.findBy('id', t);
});
} }
return result; return result;
}).catch(function() { }).catch(() => {
bootbox.alert(I18n.t('generic_error')); bootbox.alert(I18n.t('generic_error'));
self.set('loading', false); this.set('loading', false);
}); });
}, },
forEachPerformed: function(operation, cb) { forEachPerformed(operation, cb) {
var self = this; this.perform(operation).then(topics => {
this.perform(operation).then(function (topics) {
if (topics) { if (topics) {
topics.forEach(cb); topics.forEach(cb);
const refreshTarget = self.get('refreshTarget'); const refreshTarget = this.get('refreshTarget');
if (refreshTarget) { refreshTarget.send('refresh'); } if (refreshTarget) { refreshTarget.send('refresh'); }
self.send('closeModal'); this.send('closeModal');
} }
}); });
}, },
performAndRefresh: function(operation) { performAndRefresh(operation) {
const self = this; return this.perform(operation).then(() => {
return this.perform(operation).then(function() { const refreshTarget = this.get('refreshTarget');
const refreshTarget = self.get('refreshTarget');
if (refreshTarget) { refreshTarget.send('refresh'); } if (refreshTarget) { refreshTarget.send('refresh'); }
self.send('closeModal'); this.send('closeModal');
}); });
}, },
actions: { actions: {
showChangeCategory: function() { showChangeCategory() {
this.send('changeBulkTemplate', 'modal/bulk_change_category'); this.send('changeBulkTemplate', 'modal/bulk_change_category');
this.set('controllers.modal.modalClass', 'topic-bulk-actions-modal full'); this.set('controllers.modal.modalClass', 'topic-bulk-actions-modal full');
}, },
showNotificationLevel: function() { showNotificationLevel() {
this.send('changeBulkTemplate', 'modal/bulk_notification_level'); this.send('changeBulkTemplate', 'modal/bulk_notification_level');
}, },
deleteTopics: function() { deleteTopics() {
this.performAndRefresh({type: 'delete'}); this.performAndRefresh({type: 'delete'});
}, },
closeTopics: function() { closeTopics() {
this.forEachPerformed({type: 'close'}, function(t) { this.forEachPerformed({type: 'close'}, t => t.set('closed', true));
t.set('closed', true);
});
}, },
archiveTopics: function() { archiveTopics() {
this.forEachPerformed({type: 'archive'}, function(t) { this.forEachPerformed({type: 'archive'}, t => t.set('archived', true));
t.set('archived', true);
});
}, },
changeCategory: function() { unlistTopics() {
var categoryId = parseInt(this.get('newCategoryId'), 10) || 0, this.forEachPerformed({type: 'unlist'}, t => t.set('visible', false));
category = Discourse.Category.findById(categoryId), },
self = this;
this.perform({type: 'change_category', category_id: categoryId}).then(function(topics) { changeCategory() {
topics.forEach(function(t) { const categoryId = parseInt(this.get('newCategoryId'), 10) || 0;
t.set('category', category); const category = Discourse.Category.findById(categoryId);
});
const refreshTarget = self.get('refreshTarget'); this.perform({type: 'change_category', category_id: categoryId}).then(topics => {
topics.forEach(t => t.set('category', category));
const refreshTarget = this.get('refreshTarget');
if (refreshTarget) { refreshTarget.send('refresh'); } if (refreshTarget) { refreshTarget.send('refresh'); }
self.send('closeModal'); this.send('closeModal');
}); });
}, },
resetRead: function() { resetRead() {
this.performAndRefresh({ type: 'reset_read' }); this.performAndRefresh({ type: 'reset_read' });
} }
} }

View File

@ -1,6 +1,6 @@
{{#each row in buttonRows}} {{#each buttonRows as |row|}}
<p> <p>
{{#each button in row}} {{#each row as |button|}}
{{d-button action=button.action label=button.label}} {{d-button action=button.action label=button.label}}
{{/each}} {{/each}}
</p> </p>

View File

@ -1008,6 +1008,7 @@ en:
topics: topics:
bulk: bulk:
unlist_topics: "Unlist Topics"
reset_read: "Reset Read" reset_read: "Reset Read"
delete: "Delete Topics" delete: "Delete Topics"
dismiss: "Dismiss" dismiss: "Dismiss"

View File

@ -8,7 +8,7 @@ class TopicsBulkAction
end end
def self.operations def self.operations
@operations ||= %w(change_category close archive change_notification_level reset_read dismiss_posts delete) @operations ||= %w(change_category close archive change_notification_level reset_read dismiss_posts delete unlist)
end end
def self.register_operation(name, &block) def self.register_operation(name, &block)
@ -66,6 +66,15 @@ class TopicsBulkAction
end end
end end
def unlist
topics.each do |t|
if guardian.can_moderate?(t)
t.update_status('visible', false, @user)
@changed_ids << t.id
end
end
end
def archive def archive
topics.each do |t| topics.each do |t|
if guardian.can_moderate?(t) if guardian.can_moderate?(t)

View File

@ -153,4 +153,31 @@ describe TopicsBulkAction do
end end
end end
end end
describe "unlist" do
let(:topic) { Fabricate(:topic) }
context "when the user can moderate the topic" do
it "unlists the topic and returns the topic_id" do
Guardian.any_instance.expects(:can_moderate?).returns(true)
Guardian.any_instance.expects(:can_create?).returns(true)
tba = TopicsBulkAction.new(topic.user, [topic.id], type: 'unlist')
topic_ids = tba.perform!
expect(topic_ids).to eq([topic.id])
topic.reload
expect(topic).not_to be_visible
end
end
context "when the user can't edit the topic" do
it "doesn't unlist the topic" do
Guardian.any_instance.expects(:can_moderate?).returns(false)
tba = TopicsBulkAction.new(topic.user, [topic.id], type: 'unlist')
topic_ids = tba.perform!
expect(topic_ids).to be_blank
topic.reload
expect(topic).to be_visible
end
end
end
end end