hacken-in/hacken-in

View on GitHub
app/models/single_event.rb

Summary

Maintainability
A
3 hrs
Test Coverage
class SingleEvent < ActiveRecord::Base
  include TwitterHashTagFixer

  after_destroy :update_event

  belongs_to :category
  belongs_to :venue
  belongs_to :event
  belongs_to :picture

  delegate :title, :description, to: :event, prefix: true

  has_and_belongs_to_many :users

  has_many :external_users, :class_name => 'SingleEventExternalUser', :dependent => :destroy

  belongs_to :region

  validates_presence_of :event_id
  validates :duration, :numericality => { :greater_than => 0 }

  scope :in_future, -> { where("occurrence >= ?", Time.now) }
  scope :today_or_in_future, -> { where("occurrence >= ?", Time.now.beginning_of_day)}
  scope :most_current_for_event, -> { order("occurrence DESC") }
  scope :recent, ->(limit=3) { today_or_in_future.limit(limit) }
  scope :rule_based_in_future, -> { in_future.where(based_on_rule: true) }
  scope :in_next, ->(delta) { where(occurrence: (Time.now.to_date)..((Time.now + delta).to_date)) }
  scope :in_next_from, ->(delta, start_date) { where(occurrence: (start_date)..((start_date + delta).to_date)) }
  scope :recent_to_soon, ->(delta) { where(occurrence: (Time.now.to_date - delta)..((Time.now + delta).to_date + 1.day)) }
  scope :only_tagged_with, ->(tag) { tagged_with(tag) | joins(:event).where('events.id in (?)', Event.tagged_with(tag).map(&:id)) }
  scope :in_categories, ->(categories) { categories.blank? ? scoped : scoped.joins(:event).where('single_events.category_id IN (?) OR (single_events.category_id IS NULL AND events.category_id IN (?))', categories, categories) }
  scope :this_week, -> { where(occurrence: (Date.today.beginning_of_week)..(Date.today.end_of_week)) }
  scope :group_by_category, -> { joins(:event).group("events.category_id") }
  # Search for region, but also check for region_id = 1, that is the global region
  scope :in_region, ->(region) { joins(:event).where('(single_events.region_id is not null and (single_events.region_id = ? or single_events.region_id = 1)) or (single_events.region_id is null and (events.region_id = ? or events.region_id = 1))', region, region)}

  scope :name_or_description_like, ->(search) { where(arel_table[:name].matches(search).or(arel_table[:description].matches(search))) }

  default_scope -> { includes(:event).order([:occurrence, 'single_events.name ASC', 'events.name ASC']) }

  acts_as_taggable

  def self.search(search)
    search.strip!
    # Name + Description in Single Event
    sevents = today_or_in_future.name_or_description_like("%#{search}%")

    Event.search(search).each do |e|
      sevents.concat(e.single_events.today_or_in_future)
    end

    Tag.where("name like ?", "%#{search}%").each do |t|
      sevents.concat(SingleEvent.tagged_with(t.name).today_or_in_future)
      Event.tagged_with(t.name).each do |e|
        sevents.concat(e.single_events.today_or_in_future)
      end
    end

    Venue.where("location like ?", "%#{search}%").each do |v|
      sevents.concat(v.single_events.today_or_in_future)
      v.events.each do |e|
        sevents.concat(e.single_events.today_or_in_future)
      end
    end

    sevents.to_a.uniq.sort
  end

  def self.search_in_region(search_parameters, region)
    self.in_region(region).search(search_parameters)
  end

  def self.list_all(opts)
    SingleEvent.in_next_from(opts[:in_next], opts[:from]).in_region(opts[:for_region])
  end

  def date
    occurrence.to_date
  end

  # Return a hash mapping dates to the events occurring on that date in a specific date range
  # Includes days without events
  def self.events_per_day_in(date_range)
    result = where(occurrence: date_range).group('DATE(occurrence)').reorder("DATE(occurrence) ASC").count
    days_with_events = result.keys
    days_without_events = date_range.to_a - days_with_events

    days_without_events.each do |day|
      result[day] = 0
    end

    result
  end

  # This is a little workaround. Rails optimizes .exists? queries and
  # removes includes from the current used scope. Sadly we have the
  # event table in the default scope and need the include. The workaround
  # is to use an unscoped query to check if an entry exists. This
  # can have sideeffects. Be warned!
  def self.exists?(id)
    unscoped.exists?(id)
  end

  def full_name
    self.name.blank? ? self.event.name : "#{self.event.name} (#{self.name})"
  end

  alias :to_s :full_name
  alias :title :full_name

  def name_with_date
    if self.full_day
      "#{self.full_name} am #{self.occurrence.strftime("%d.%m.%Y")}"
    else
      "#{self.full_name} am #{self.occurrence.strftime("%d.%m.%Y um %H:%M")}"
    end
  end

  def <=>(other)
    if (self.occurrence.year != other.occurrence.year) || (self.occurrence.month != other.occurrence.month) || (self.occurrence.day != other.occurrence.day)
      # not on same day
      return self.occurrence <=> other.occurrence
    elsif self.full_day
      if other.full_day
        # both are all day, sort via topic
        return self.full_name.downcase <=> other.full_name.downcase
      else
        # self is all day, other is not
        return -1
      end
    elsif other.full_day
      # other is all day, self is not
      return 1
    else
      # both are not all day, sort via time
      time_comparison = (self.occurrence <=> other.occurrence)

      if time_comparison == 0
        # they are at the same time, sort via topic
        return self.full_name.downcase <=> other.full_name.downcase
      else
        return time_comparison
      end
    end
  end

  def short_description
    return event.short_description if self.description.blank?
    ActionController::Base.helpers.strip_tags(self.description).truncate 80
  end

  def to_opengraph
    ogdata = event.to_opengraph.merge({
      "og:title"       => name_with_date,
      "og:description" => short_description
    })
    if picture.present? && picture.box_image.url.present?
      ogdata["og:image"] = "http://hacken.in#{picture.box_image.url}"
    end
    ogdata = ogdata.merge venue.to_opengraph unless venue.nil?
    ogdata = ogdata.reject { |key, value| value.blank? }
    ogdata
  end

  alias :self_category :category
  def category
    self.self_category || (self.event && self.event.category)
  end

  alias :self_venue :venue
  def venue
    self.self_venue || self.event.venue
  end

  alias :self_picture :picture
  def picture
    self.self_picture || self.event.picture
  end

  def venue_info
    venue_info = self.read_attribute(:venue_info)
    if use_venue_info_of_event && venue_info.blank?
      venue_info = self.event.venue_info
    end
    venue_info
  end

  # Get the attribute from the Event model unless they exist here
  [:url, :twitter, :twitter_hashtag, :duration, :full_day, :category_id].each do |item|
    define_method item.to_s do
      value = self.read_attribute(item)
      if !value.nil? && !(value.class.to_s == "String" && value.blank?)
        value
      elsif !self.event.blank?
        self.event.send(item)
      end
    end
    alias_method :"#{item}?", item
  end

  def update_event
    if based_on_rule
      event.schedule.add_exception_time occurrence
      if event.schedule.rtimes.include? occurrence
        event.schedule.remove_recurrence_time occurrence
      end
      event.save
    end
  end

  def is_for_user?(user)
    return true unless user
    # Let us be a little more verbose than the old code
    # !((self.event.tag_list & user.hate_list).length > 0 && self.users.exclude?(user))
    hated_tags_event = (self.event.tag_list & user.hate_list)
    hated_tags_self  = (self.tag_list & user.hate_list)

    loved_tags_event = (self.event.tag_list & user.like_list)
    loved_tags_self  = (self.tag_list & user.like_list)

    # Wenn der Event oder der Single Event einen Tag haben, den der Benutzer hasst...
    if hated_tags_event.size + hated_tags_self.size > 0
      # ... muss der Benutzer mindestens einen der anderen Tags des Events lieben
      if loved_tags_event.size + loved_tags_self.size > 0
        # ... damit er angezeigt wird
        true
      # ... oder er muss an dem Event teilnehmen
      elsif self.users.to_a.include? user
        true
      else
        false
      end
    else
      true
    end
  end

  def attended_by_user?(user)
    users.include? user
  end

  def attended_by_external_user?(session_uuid)
    session_uuid.present? && external_users.find_by_session_token(session_uuid).present?
  end
end