class Debate < ApplicationRecord include Flaggable include Taggable include Conflictable include Measurable include Sanitizable include Searchable include Filterable include HasPublicAuthor include Graphqlable include Relationable include Notifiable include Randomizable include SDG::Relatable acts_as_votable acts_as_paranoid column: :hidden_at include ActsAsParanoidAliases translates :title, touch: true translates :description, touch: true include Globalizable belongs_to :author, -> { with_hidden }, class_name: "User", inverse_of: :debates belongs_to :geozone has_many :comments, as: :commentable, inverse_of: :commentable validates_translation :title, presence: true, length: { in: 4..Debate.title_max_length } validates_translation :description, presence: true validate :description_length validates :author, presence: true validates :terms_of_service, acceptance: { allow_nil: false }, on: :create before_save :calculate_hot_score, :calculate_confidence_score scope :for_render, -> { includes(:tags) } scope :sort_by_hot_score, -> { reorder(hot_score: :desc) } scope :sort_by_confidence_score, -> { reorder(confidence_score: :desc) } scope :sort_by_created_at, -> { reorder(created_at: :desc) } scope :sort_by_most_commented, -> { reorder(comments_count: :desc) } scope :sort_by_relevance, -> { all } scope :sort_by_flags, -> { order(flags_count: :desc, updated_at: :desc) } scope :sort_by_recommendations, -> { order(cached_votes_total: :desc) } scope :last_week, -> { where("created_at >= ?", 7.days.ago) } scope :featured, -> { where.not(featured_at: nil) } scope :public_for_api, -> { all } visitable class_name: "Visit" attr_accessor :link_required def self.recommendations(user) tagged_with(user.interests, any: true).where.not(author_id: user.id) end def searchable_translations_definitions { title => "A", description => "D" } end def searchable_values { author.username => "B", tag_list.join(" ") => "B", geozone&.name => "B" }.merge!(searchable_globalized_values) end def self.search(terms) pg_search(terms) end def to_param "#{id}-#{title}".parameterize end def likes cached_votes_up end def dislikes cached_votes_down end def total_votes cached_votes_total end def votes_score cached_votes_score end def total_anonymous_votes cached_anonymous_votes_total end def editable? total_votes <= Setting["max_votes_for_debate_edit"].to_i end def editable_by?(user) editable? && author == user end def register_vote(user, vote_value) if votable_by?(user) transaction do if user.unverified? && !user.voted_for?(self) Debate.increment_counter(:cached_anonymous_votes_total, id) end vote_by(voter: user, vote: vote_value) end end end def votable_by?(user) return false unless user total_votes <= 100 || !user.unverified? || Setting["max_ratio_anon_votes_on_debates"].to_i == 100 || anonymous_votes_ratio < Setting["max_ratio_anon_votes_on_debates"].to_i || user.voted_for?(self) end def anonymous_votes_ratio return 0 if cached_votes_total == 0 (cached_anonymous_votes_total.to_f / cached_votes_total) * 100 end def after_commented save # update cache when it has a new comment end def calculate_hot_score self.hot_score = ScoreCalculator.hot_score(self) end def calculate_confidence_score self.confidence_score = ScoreCalculator.confidence_score(cached_votes_total, cached_votes_up) end def after_hide tags.each { |t| t.decrement_custom_counter_for("Debate") } end def after_restore tags.each { |t| t.increment_custom_counter_for("Debate") } end def featured? featured_at.present? end def self.debates_orders(user) orders = %w[hot_score confidence_score created_at relevance] if Setting["feature.user.recommendations_on_debates"] && user&.recommended_debates orders << "recommendations" end orders end def description_length real_description_length = ActionView::Base.full_sanitizer.sanitize(description.to_s).squish.length if real_description_length < Debate.description_min_length errors.add(:description, :too_short, count: Debate.description_min_length) translation.errors.add(:description, :too_short, count: Debate.description_min_length) end if real_description_length > Debate.description_max_length errors.add(:description, :too_long, count: Debate.description_max_length) translation.errors.add(:description, :too_long, count: Debate.description_max_length) end end end