add make/remove banner topic actions

This commit is contained in:
Régis Hanol 2014-06-16 18:28:07 +02:00
parent 3153a84da2
commit 5238a95efb
8 changed files with 130 additions and 44 deletions

View File

@ -302,6 +302,14 @@ Discourse.TopicController = Discourse.ObjectController.extend(Discourse.Selected
this.get('content').toggleStatus('closed'); this.get('content').toggleStatus('closed');
}, },
makeBanner: function() {
this.get('content').makeBanner();
},
removeBanner: function() {
this.get('content').removeBanner();
},
togglePinned: function() { togglePinned: function() {
// Note that this is different than clearPin // Note that this is different than clearPin
this.get('content').setStatus('pinned', this.get('pinned_at') ? false : true); this.get('content').setStatus('pinned', this.get('pinned_at') ? false : true);
@ -316,10 +324,6 @@ Discourse.TopicController = Discourse.ObjectController.extend(Discourse.Selected
this.get('content').toggleStatus('archived'); this.get('content').toggleStatus('archived');
}, },
convertToRegular: function() {
this.get('content').convertArchetype('regular');
},
// Toggle the star on the topic // Toggle the star on the topic
toggleStar: function() { toggleStar: function() {
this.get('content').toggleStar(); this.get('content').toggleStar();

View File

@ -19,22 +19,6 @@ Discourse.Topic = Discourse.Model.extend({
invisible: Em.computed.not('visible'), invisible: Em.computed.not('visible'),
deleted: Em.computed.notEmpty('deleted_at'), deleted: Em.computed.notEmpty('deleted_at'),
canConvertToRegular: function() {
var a = this.get('archetype');
return a !== 'regular' && a !== 'private_message';
}.property('archetype'),
convertArchetype: function() {
var a = this.get('archetype');
if (a !== 'regular' && a !== 'private_message') {
this.set('archetype', 'regular');
return Discourse.ajax(this.get('url'), {
type: 'PUT',
data: {archetype: 'regular'}
});
}
},
searchContext: function() { searchContext: function() {
return ({ type: 'topic', id: this.get('id') }); return ({ type: 'topic', id: this.get('id') });
}.property('id'), }.property('id'),
@ -131,6 +115,7 @@ Discourse.Topic = Discourse.Model.extend({
}.property('archetype'), }.property('archetype'),
isPrivateMessage: Em.computed.equal('archetype', 'private_message'), isPrivateMessage: Em.computed.equal('archetype', 'private_message'),
isBanner: Em.computed.equal('archetype', 'banner'),
toggleStatus: function(property) { toggleStatus: function(property) {
this.toggleProperty(property); this.toggleProperty(property);
@ -155,6 +140,18 @@ Discourse.Topic = Discourse.Model.extend({
}); });
}, },
makeBanner: function() {
var self = this;
return Discourse.ajax('/t/' + this.get('id') + '/make-banner', { type: 'PUT' })
.then(function () { self.set('archetype', 'banner'); })
},
removeBanner: function() {
var self = this;
return Discourse.ajax('/t/' + this.get('id') + '/remove-banner', { type: 'PUT' })
.then(function () { self.set('archetype', 'regular'); })
},
starTooltipKey: function() { starTooltipKey: function() {
return this.get('starred') ? 'starred.help.unstar' : 'starred.help.star'; return this.get('starred') ? 'starred.help.unstar' : 'starred.help.star';
}.property('starred'), }.property('starred'),

View File

@ -28,6 +28,15 @@
{{/if}} {{/if}}
</li> </li>
{{#unless isPrivateMessage}}
<li>
{{#if isBanner}}
<button {{action removeBanner}} class='btn btn-admin'><i class='fa fa-bullhorn'></i> {{i18n topic.actions.remove_banner}}</button>
{{else}}
<button {{action makeBanner}} class='btn btn-admin'><i class='fa fa-bullhorn'></i> {{i18n topic.actions.make_banner}}</button>
{{/if}}
</li>
<li> <li>
{{#if pinned_at}} {{#if pinned_at}}
<button {{action togglePinned}} class='btn btn-admin'><i class='fa fa-thumb-tack'></i> {{i18n topic.actions.unpin}}</button> <button {{action togglePinned}} class='btn btn-admin'><i class='fa fa-thumb-tack'></i> {{i18n topic.actions.unpin}}</button>
@ -36,6 +45,7 @@
<button {{action togglePinnedGlobally}} class='btn btn-admin'><i class='fa fa-thumb-tack'></i> {{i18n topic.actions.pin_globally}}</button> <button {{action togglePinnedGlobally}} class='btn btn-admin'><i class='fa fa-thumb-tack'></i> {{i18n topic.actions.pin_globally}}</button>
{{/if}} {{/if}}
</li> </li>
{{/unless}}
<li> <li>
{{#if archived}} {{#if archived}}
@ -53,12 +63,6 @@
{{/if}} {{/if}}
</li> </li>
{{#if canConvertToRegular}}
<li>
<button {{action convertToRegular}} class='btn btn-admin'><i class='fa fa-minus'></i> {{i18n topic.actions.convert_to_topic}}</button>
</li>
{{/if}}
{{#if currentUser.staff}} {{#if currentUser.staff}}
<li> <li>
<button class='btn btn-admin' {{action resetRead}}><i class='fa fa-times'></i> {{i18n topic.actions.reset_read}}</button> <button class='btn btn-admin' {{action resetRead}}><i class='fa fa-times'></i> {{i18n topic.actions.reset_read}}</button>

View File

@ -75,6 +75,7 @@ class TopicsController < ApplicationController
params.require(:best) params.require(:best)
params.require(:topic_id) params.require(:topic_id)
params.permit(:min_trust_level, :min_score, :min_replies, :bypass_trust_level_score, :only_moderator_liked) params.permit(:min_trust_level, :min_score, :min_replies, :bypass_trust_level_score, :only_moderator_liked)
opts = { best: params[:best].to_i, opts = { best: params[:best].to_i,
min_trust_level: params[:min_trust_level] ? 1 : params[:min_trust_level].to_i, min_trust_level: params[:min_trust_level] ? 1 : params[:min_trust_level].to_i,
min_score: params[:min_score].to_i, min_score: params[:min_score].to_i,
@ -105,13 +106,9 @@ class TopicsController < ApplicationController
def update def update
topic = Topic.find_by(id: params[:topic_id]) topic = Topic.find_by(id: params[:topic_id])
title, archetype = params[:title], params[:archetype]
guardian.ensure_can_edit!(topic) guardian.ensure_can_edit!(topic)
topic.title = params[:title] if title.present? topic.title = params[:title] if params[:title].present?
# TODO: we may need smarter rules about converting archetypes
topic.archetype = "regular" if current_user.admin? && archetype == 'regular'
topic.acting_user = current_user topic.acting_user = current_user
success = false success = false
@ -176,6 +173,32 @@ class TopicsController < ApplicationController
end end
end end
def make_banner
topic = Topic.find_by(id: params[:topic_id].to_i)
guardian.ensure_can_moderate!(topic)
# TODO: only one banner at the same time
topic.archetype = Archetype.banner
topic.add_moderator_post(current_user, I18n.t("archetypes.banner.message.make"))
topic.save
render nothing: true
end
def remove_banner
topic = Topic.find_by(id: params[:topic_id].to_i)
guardian.ensure_can_moderate!(topic)
topic.archetype = Archetype.default
topic.add_moderator_post(current_user, I18n.t("archetypes.banner.message.remove"))
topic.save
render nothing: true
end
def destroy def destroy
topic = Topic.find_by(id: params[:id]) topic = Topic.find_by(id: params[:id])
guardian.ensure_can_delete!(topic) guardian.ensure_can_delete!(topic)

View File

@ -837,6 +837,8 @@ en:
open: "Open Topic" open: "Open Topic"
close: "Close Topic" close: "Close Topic"
auto_close: "Auto Close" auto_close: "Auto Close"
make_banner: "Banner Topic"
remove_banner: "Remove Banner Topic"
unpin: "Un-Pin Topic" unpin: "Un-Pin Topic"
pin: "Pin Topic" pin: "Pin Topic"
pin_globally: "Pin Topic Globally" pin_globally: "Pin Topic Globally"
@ -846,7 +848,6 @@ en:
visible: "Make Visible" visible: "Make Visible"
reset_read: "Reset Read Data" reset_read: "Reset Read Data"
multi_select: "Select Posts" multi_select: "Select Posts"
convert_to_topic: "Convert to Regular Topic"
reply: reply:
title: 'Reply' title: 'Reply'

View File

@ -423,6 +423,10 @@ en:
archetypes: archetypes:
regular: regular:
title: "Regular Topic" title: "Regular Topic"
banner:
message:
make: "This topic is now a banner topic. It will appear at the top of every pages until it is dismissed by the users."
remove: "This topic is not a banner topic anymore. It will no longer appear at the top of every pages."
unsubscribed: unsubscribed:
title: 'Unsubscribed' title: 'Unsubscribed'
@ -989,7 +993,6 @@ en:
autoclosed_disabled: "This topic is now opened. New replies are allowed." autoclosed_disabled: "This topic is now opened. New replies are allowed."
pinned_enabled: "This topic is now pinned. It will appear at the top of its category until it is unpinned by staff for everyone, or by individual users for themselves." pinned_enabled: "This topic is now pinned. It will appear at the top of its category until it is unpinned by staff for everyone, or by individual users for themselves."
pinned_disabled: "This topic is now unpinned. It will no longer appear at the top of its category." pinned_disabled: "This topic is now unpinned. It will no longer appear at the top of its category."
pinned_globally_enabled: "This topic is now pinned globally. It will appear at the top of its category and all topic lists until it is unpinned by staff for everyone, or by individual users for themselves." pinned_globally_enabled: "This topic is now pinned globally. It will appear at the top of its category and all topic lists until it is unpinned by staff for everyone, or by individual users for themselves."
pinned_globally_disabled: "This topic is now unpinned. It will no longer appear at the top of its category." pinned_globally_disabled: "This topic is now unpinned. It will no longer appear at the top of its category."
visible_enabled: "This topic is now visible. It will be displayed in topic lists." visible_enabled: "This topic is now visible. It will be displayed in topic lists."

View File

@ -340,6 +340,8 @@ Discourse::Application.routes.draw do
put "t/:topic_id/mute" => "topics#mute", constraints: {topic_id: /\d+/} put "t/:topic_id/mute" => "topics#mute", constraints: {topic_id: /\d+/}
put "t/:topic_id/unmute" => "topics#unmute", constraints: {topic_id: /\d+/} put "t/:topic_id/unmute" => "topics#unmute", constraints: {topic_id: /\d+/}
put "t/:topic_id/autoclose" => "topics#autoclose", constraints: {topic_id: /\d+/} put "t/:topic_id/autoclose" => "topics#autoclose", constraints: {topic_id: /\d+/}
put "t/:topic_id/make-banner" => "topics#make_banner", constraints: {topic_id: /\d+/}
put "t/:topic_id/remove-banner" => "topics#remove_banner", constraints: {topic_id: /\d+/}
put "t/:topic_id/remove-allowed-user" => "topics#remove_allowed_user", constraints: {topic_id: /\d+/} put "t/:topic_id/remove-allowed-user" => "topics#remove_allowed_user", constraints: {topic_id: /\d+/}
put "t/:topic_id/recover" => "topics#recover", constraints: {topic_id: /\d+/} put "t/:topic_id/recover" => "topics#recover", constraints: {topic_id: /\d+/}
get "t/:topic_id/:post_number" => "topics#show", constraints: {topic_id: /\d+/, post_number: /\d+/} get "t/:topic_id/:post_number" => "topics#show", constraints: {topic_id: /\d+/, post_number: /\d+/}

View File

@ -847,6 +847,58 @@ describe TopicsController do
end end
describe 'make_banner' do
it 'needs you to be a staff member' do
log_in
xhr :put, :make_banner, topic_id: 99
response.should be_forbidden
end
describe 'when logged in' do
before do
@admin = log_in(:admin)
@topic = Fabricate(:topic, user: @admin)
end
it "changes the topic archetype to 'banner'" do
Topic.any_instance.expects(:archetype=).with(Archetype.banner)
Topic.any_instance.expects(:add_moderator_post)
xhr :put, :make_banner, topic_id: @topic.id
response.should be_success
end
end
end
describe 'remove_banner' do
it 'needs you to be a staff member' do
log_in
xhr :put, :remove_banner, topic_id: 99
response.should be_forbidden
end
describe 'when logged in' do
before do
@admin = log_in(:admin)
@topic = Fabricate(:topic, user: @admin)
end
it "resets the topic archetype" do
Topic.any_instance.expects(:archetype=).with(Archetype.default)
Topic.any_instance.expects(:add_moderator_post)
xhr :put, :remove_banner, topic_id: @topic.id
response.should be_success
end
end
end
describe "bulk" do describe "bulk" do
it 'needs you to be logged in' do it 'needs you to be logged in' do
lambda { xhr :put, :bulk }.should raise_error(Discourse::NotLoggedIn) lambda { xhr :put, :bulk }.should raise_error(Discourse::NotLoggedIn)