discourse/app/controllers/list_controller.rb

165 lines
5.3 KiB
Ruby
Raw Normal View History

2013-02-06 03:16:51 +08:00
class ListController < ApplicationController
before_filter :ensure_logged_in, except: [:latest, :hot, :category, :category_feed, :latest_feed, :hot_feed, :topics_by]
before_filter :set_category, only: [:category, :category_feed]
2013-02-06 03:16:51 +08:00
skip_before_filter :check_xhr
# Create our filters
[:latest, :hot, :favorited, :read, :posted, :unread, :new].each do |filter|
2013-02-07 23:45:24 +08:00
define_method(filter) do
list_opts = build_topic_list_options
user = list_target_user
list = TopicQuery.new(user, list_opts).public_send("list_#{filter}")
list.more_topics_url = construct_url_with(filter, list_opts)
2013-10-12 00:35:12 +08:00
if [:latest, :hot].include?(filter)
@description = SiteSetting.site_description
@rss = filter
end
2013-02-07 23:45:24 +08:00
2013-02-06 03:16:51 +08:00
respond(list)
2013-02-07 23:45:24 +08:00
end
2013-02-06 03:16:51 +08:00
end
2013-07-06 04:49:06 +08:00
[:latest, :hot].each do |filter|
define_method("#{filter}_feed") do
discourse_expires_in 1.minute
@title = "#{filter.capitalize} Topics"
@link = "#{Discourse.base_url}/#{filter}"
@description = I18n.t("rss_description.#{filter}")
@atom_link = "#{Discourse.base_url}/#{filter}.rss"
@topic_list = TopicQuery.new.public_send("list_#{filter}")
render 'list', formats: [:rss]
2013-07-06 04:49:06 +08:00
end
end
[:topics_by, :private_messages, :private_messages_sent, :private_messages_unread].each do |action|
define_method("#{action}") do
list_opts = build_topic_list_options
target_user = fetch_user_from_params
guardian.ensure_can_see_private_messages!(target_user.id) unless action == :topics_by
list = generate_list_for(action.to_s, target_user, list_opts)
url_prefix = "topics" unless action == :topics_by
url = construct_url_with(action, list_opts, url_prefix)
list.more_topics_url = url_for(url)
respond(list)
end
end
2013-02-06 03:16:51 +08:00
def category
query = TopicQuery.new(current_user, page: params[:page])
if !@category
raise Discourse::NotFound
return
2013-02-06 03:16:51 +08:00
end
guardian.ensure_can_see!(@category)
list = query.list_category(@category)
@description = @category.description
2013-02-06 03:16:51 +08:00
2013-10-24 02:40:39 +08:00
if params[:parent_category].present?
list.more_topics_url = url_for(category_list_parent_path(params[:parent_category], params[:category], page: next_page, format: "json"))
else
list.more_topics_url = url_for(category_list_path(params[:category], page: next_page, format: "json"))
end
2013-02-06 03:16:51 +08:00
respond(list)
end
def category_feed
guardian.ensure_can_see!(@category)
discourse_expires_in 1.minute
@title = @category.name
@link = "#{Discourse.base_url}/category/#{@category.slug}"
@description = "#{I18n.t('topics_in_category', category: @category.name)} #{@category.description}"
@atom_link = "#{Discourse.base_url}/category/#{@category.slug}.rss"
@topic_list = TopicQuery.new.list_new_in_category(@category)
render 'list', formats: [:rss]
end
def popular_redirect
# We've renamed popular to latest. Use a redirect until we're sure we can
# safely remove this.
redirect_to latest_path, :status => 301
end
2013-02-06 03:16:51 +08:00
protected
def respond(list)
list.draft_key = Draft::NEW_TOPIC
list.draft_sequence = DraftSequence.current(current_user, Draft::NEW_TOPIC)
2013-02-07 23:45:24 +08:00
2013-02-06 03:16:51 +08:00
draft = Draft.get(current_user, list.draft_key, list.draft_sequence) if current_user
list.draft = draft
discourse_expires_in 1.minute
2013-02-06 03:16:51 +08:00
respond_to do |format|
format.html do
2013-02-07 23:45:24 +08:00
@list = list
2013-02-06 03:16:51 +08:00
store_preloaded('topic_list', MultiJson.dump(TopicListSerializer.new(list, scope: guardian)))
render 'list'
end
format.json do
render_serialized(list, TopicListSerializer)
end
end
end
2013-02-07 23:45:24 +08:00
2013-02-06 03:16:51 +08:00
def next_page
params[:page].to_i + 1
end
private
def set_category
slug = params.fetch(:category)
2013-10-24 02:40:39 +08:00
parent_slug = params[:parent_category]
parent_category_id = nil
if parent_slug.present?
parent_category_id = Category.where(slug: parent_slug).pluck(:id).first ||
Category.where(id: parent_slug.to_i).pluck(:id).first
raise Discourse::NotFound.new if parent_category_id.blank?
end
@category = Category.where(slug: slug, parent_category_id: parent_category_id).includes(:featured_users).first ||
Category.where(id: slug.to_i, parent_category_id: parent_category_id).includes(:featured_users).first
end
def build_topic_list_options
# html format means we need to parse exclude category (aka filter) from the site options top menu
menu_items = SiteSetting.top_menu_items
menu_item = menu_items.select { |item| item.query_should_exclude_category?(action_name, params[:format]) }.first
# exclude_category = 1. from params / 2. parsed from top menu / 3. nil
return {
page: params[:page],
topic_ids: param_to_integer_list(:topic_ids),
exclude_category: (params[:exclude_category] || menu_item.try(:filter)),
category: params[:category]
}
end
def list_target_user
if params[:user_id] && guardian.is_staff?
User.find(params[:user_id].to_i)
else
current_user
end
end
def generate_list_for(action, target_user, opts)
list = TopicQuery.new(current_user, opts)
list = list.send("list_#{action}", target_user)
end
def construct_url_with(action, opts, url_prefix=nil)
method = url_prefix.blank? ? "#{action}_path" : "#{url_prefix}_#{action}_path"
public_send(method, opts.merge(format: 'json', page: next_page))
end
2013-02-06 03:16:51 +08:00
end