mirror of
https://github.com/discourse/discourse.git
synced 2025-01-25 08:20:45 +08:00
315 lines
9.0 KiB
Ruby
315 lines
9.0 KiB
Ruby
require_dependency 'topic_view'
|
|
require_dependency 'promotion'
|
|
|
|
class TopicsController < ApplicationController
|
|
|
|
# Avatar is an image request, not XHR
|
|
before_filter :ensure_logged_in, only: [:timings,
|
|
:destroy_timings,
|
|
:update,
|
|
:star,
|
|
:destroy,
|
|
:status,
|
|
:invite,
|
|
:mute,
|
|
:unmute,
|
|
:set_notifications,
|
|
:move_posts,
|
|
:merge_topic,
|
|
:clear_pin,
|
|
:autoclose]
|
|
|
|
before_filter :consider_user_for_promotion, only: :show
|
|
|
|
skip_before_filter :check_xhr, only: [:avatar, :show, :feed]
|
|
caches_action :avatar, cache_path: Proc.new {|c| "#{c.params[:post_number]}-#{c.params[:topic_id]}" }
|
|
|
|
def show
|
|
|
|
# We'd like to migrate the wordpress feed to another url. This keeps up backwards compatibility with
|
|
# existing installs.
|
|
return wordpress if params[:best].present?
|
|
|
|
opts = params.slice(:username_filters, :best_of, :page, :post_number, :posts_before, :posts_after)
|
|
begin
|
|
@topic_view = TopicView.new(params[:id] || params[:topic_id], current_user, opts)
|
|
rescue Discourse::NotFound
|
|
topic = Topic.where(slug: params[:id]).first if params[:id]
|
|
raise Discourse::NotFound unless topic
|
|
return redirect_to(topic.relative_url)
|
|
end
|
|
|
|
anonymous_etag(@topic_view.topic) do
|
|
redirect_to_correct_topic && return if slugs_do_not_match
|
|
View.create_for(@topic_view.topic, request.remote_ip, current_user)
|
|
track_visit_to_topic
|
|
perform_show_response
|
|
end
|
|
|
|
canonical_url @topic_view.canonical_path
|
|
end
|
|
|
|
def wordpress
|
|
params.require(:best)
|
|
params.require(:topic_id)
|
|
params.permit(:min_trust_level, :min_score, :min_replies)
|
|
|
|
@topic_view = TopicView.new(
|
|
params[:topic_id],
|
|
current_user,
|
|
best: params[:best].to_i,
|
|
min_trust_level: params[:min_trust_level].nil? ? 1 : params[:min_trust_level].to_i,
|
|
min_score: params[:min_score].to_i,
|
|
min_replies: params[:min_replies].to_i
|
|
)
|
|
|
|
anonymous_etag(@topic_view.topic) do
|
|
wordpress_serializer = TopicViewWordpressSerializer.new(@topic_view, scope: guardian, root: false)
|
|
render_json_dump(wordpress_serializer)
|
|
end
|
|
|
|
end
|
|
|
|
def destroy_timings
|
|
PostTiming.destroy_for(current_user.id, params[:topic_id].to_i)
|
|
render nothing: true
|
|
end
|
|
|
|
def update
|
|
topic = Topic.where(id: params[:topic_id]).first
|
|
guardian.ensure_can_edit!(topic)
|
|
topic.title = params[:title] if params[:title].present?
|
|
|
|
# TODO: we may need smarter rules about converting archetypes
|
|
if current_user.admin?
|
|
topic.archetype = "regular" if params[:archetype] == 'regular'
|
|
end
|
|
|
|
success = false
|
|
Topic.transaction do
|
|
success = topic.save
|
|
topic.change_category(params[:category]) if success
|
|
end
|
|
|
|
# this is used to return the title to the client as it may have been
|
|
# changed by "TextCleaner"
|
|
if success
|
|
render_serialized(topic, BasicTopicSerializer)
|
|
else
|
|
render_json_error(topic)
|
|
end
|
|
end
|
|
|
|
def similar_to
|
|
params.require(:title)
|
|
params.require(:raw)
|
|
title, raw = params[:title], params[:raw]
|
|
|
|
raise Discourse::InvalidParameters.new(:title) if title.length < SiteSetting.min_title_similar_length
|
|
raise Discourse::InvalidParameters.new(:raw) if raw.length < SiteSetting.min_body_similar_length
|
|
|
|
# Only suggest similar topics if the site has a minimmum amount of topics present.
|
|
if Topic.count > SiteSetting.minimum_topics_similar
|
|
topics = Topic.similar_to(title, raw, current_user)
|
|
end
|
|
|
|
render_serialized(topics, BasicTopicSerializer)
|
|
end
|
|
|
|
def status
|
|
params.require(:status)
|
|
params.require(:enabled)
|
|
|
|
raise Discourse::InvalidParameters.new(:status) unless %w(visible closed pinned archived).include?(params[:status])
|
|
@topic = Topic.where(id: params[:topic_id].to_i).first
|
|
guardian.ensure_can_moderate!(@topic)
|
|
@topic.update_status(params[:status], (params[:enabled] == 'true'), current_user)
|
|
render nothing: true
|
|
end
|
|
|
|
def star
|
|
@topic = Topic.where(id: params[:topic_id].to_i).first
|
|
guardian.ensure_can_see!(@topic)
|
|
|
|
@topic.toggle_star(current_user, params[:starred] == 'true')
|
|
render nothing: true
|
|
end
|
|
|
|
def mute
|
|
toggle_mute
|
|
end
|
|
|
|
def unmute
|
|
toggle_mute
|
|
end
|
|
|
|
def autoclose
|
|
raise Discourse::InvalidParameters.new(:auto_close_days) unless params.has_key?(:auto_close_days)
|
|
@topic = Topic.where(id: params[:topic_id].to_i).first
|
|
guardian.ensure_can_moderate!(@topic)
|
|
@topic.set_auto_close(params[:auto_close_days], current_user)
|
|
@topic.save
|
|
render nothing: true
|
|
end
|
|
|
|
def destroy
|
|
topic = Topic.where(id: params[:id]).first
|
|
guardian.ensure_can_delete!(topic)
|
|
topic.trash!
|
|
render nothing: true
|
|
end
|
|
|
|
def excerpt
|
|
render nothing: true
|
|
end
|
|
|
|
def remove_allowed_user
|
|
params.require(:username)
|
|
topic = Topic.where(id: params[:topic_id]).first
|
|
guardian.ensure_can_remove_allowed_users!(topic)
|
|
|
|
if topic.remove_allowed_user(params[:username])
|
|
render json: success_json
|
|
else
|
|
render json: failed_json, status: 422
|
|
end
|
|
end
|
|
|
|
def invite
|
|
params.require(:user)
|
|
topic = Topic.where(id: params[:topic_id]).first
|
|
guardian.ensure_can_invite_to!(topic)
|
|
|
|
if topic.invite(current_user, params[:user])
|
|
user = User.find_by_username_or_email(params[:user])
|
|
if user
|
|
render_json_dump BasicUserSerializer.new(user, scope: guardian, root: 'user')
|
|
else
|
|
render json: success_json
|
|
end
|
|
else
|
|
render json: failed_json, status: 422
|
|
end
|
|
end
|
|
|
|
def set_notifications
|
|
topic = Topic.find(params[:topic_id].to_i)
|
|
TopicUser.change(current_user, topic.id, notification_level: params[:notification_level].to_i)
|
|
render json: success_json
|
|
end
|
|
|
|
def merge_topic
|
|
params.require(:destination_topic_id)
|
|
|
|
topic = Topic.where(id: params[:topic_id]).first
|
|
guardian.ensure_can_move_posts!(topic)
|
|
|
|
dest_topic = topic.move_posts(current_user, topic.posts.pluck(:id), destination_topic_id: params[:destination_topic_id].to_i)
|
|
render_topic_changes(dest_topic)
|
|
end
|
|
|
|
def move_posts
|
|
params.require(:post_ids)
|
|
|
|
topic = Topic.where(id: params[:topic_id]).first
|
|
guardian.ensure_can_move_posts!(topic)
|
|
|
|
dest_topic = move_post_to_destination(topic)
|
|
render_topic_changes(dest_topic)
|
|
end
|
|
|
|
def clear_pin
|
|
topic = Topic.where(id: params[:topic_id].to_i).first
|
|
guardian.ensure_can_see!(topic)
|
|
topic.clear_pin_for(current_user)
|
|
render nothing: true
|
|
end
|
|
|
|
def timings
|
|
PostTiming.process_timings(
|
|
current_user,
|
|
params[:topic_id].to_i,
|
|
params[:topic_time].to_i,
|
|
(params[:timings] || []).map{|post_number, t| [post_number.to_i, t.to_i]}
|
|
)
|
|
render nothing: true
|
|
end
|
|
|
|
def feed
|
|
@topic_view = TopicView.new(params[:topic_id])
|
|
anonymous_etag(@topic_view.topic) do
|
|
render 'topics/show', formats: [:rss]
|
|
end
|
|
end
|
|
|
|
private
|
|
|
|
def toggle_mute
|
|
@topic = Topic.where(id: params[:topic_id].to_i).first
|
|
guardian.ensure_can_see!(@topic)
|
|
|
|
@topic.toggle_mute(current_user)
|
|
render nothing: true
|
|
end
|
|
|
|
def consider_user_for_promotion
|
|
Promotion.new(current_user).review if current_user.present?
|
|
end
|
|
|
|
def slugs_do_not_match
|
|
params[:slug] && @topic_view.topic.slug != params[:slug]
|
|
end
|
|
|
|
def redirect_to_correct_topic
|
|
fullpath = request.fullpath
|
|
|
|
split = fullpath.split('/')
|
|
split[2] = @topic_view.topic.slug
|
|
|
|
redirect_to split.join('/'), status: 301
|
|
end
|
|
|
|
def track_visit_to_topic
|
|
return unless should_track_visit_to_topic?
|
|
TopicUser.track_visit! @topic_view.topic, current_user
|
|
@topic_view.draft = Draft.get(current_user, @topic_view.draft_key, @topic_view.draft_sequence)
|
|
end
|
|
|
|
def should_track_visit_to_topic?
|
|
(!request.xhr? || params[:track_visit]) && current_user
|
|
end
|
|
|
|
def perform_show_response
|
|
topic_view_serializer = TopicViewSerializer.new(@topic_view, scope: guardian, root: false)
|
|
|
|
respond_to do |format|
|
|
format.html do
|
|
store_preloaded("topic_#{@topic_view.topic.id}", MultiJson.dump(topic_view_serializer))
|
|
end
|
|
|
|
format.json do
|
|
render_json_dump(topic_view_serializer)
|
|
end
|
|
end
|
|
end
|
|
|
|
def render_topic_changes(dest_topic)
|
|
if dest_topic.present?
|
|
render json: {success: true, url: dest_topic.relative_url}
|
|
else
|
|
render json: {success: false}
|
|
end
|
|
end
|
|
|
|
private
|
|
|
|
def move_post_to_destination(topic)
|
|
args = {}
|
|
args[:title] = params[:title] if params[:title].present?
|
|
args[:destination_topic_id] = params[:destination_topic_id].to_i if params[:destination_topic_id].present?
|
|
|
|
topic.move_posts(current_user, params[:post_ids].map {|p| p.to_i}, args)
|
|
end
|
|
|
|
end
|