1
0
mirror of https://github.com/LaCasemate/fab-manager.git synced 2024-12-11 22:24:21 +01:00
fab-manager/app/models/event.rb
2022-07-20 15:54:04 +02:00

166 lines
5.1 KiB
Ruby

# frozen_string_literal: true
# Event is an happening organized by the Fablab about a general topic, which does not involve Machines or trainings member's skills.
class Event < ApplicationRecord
include NotifyWith::NotificationAttachedObject
include ApplicationHelper
has_one :event_image, as: :viewable, dependent: :destroy
accepts_nested_attributes_for :event_image, allow_destroy: true
has_many :event_files, as: :viewable, dependent: :destroy
accepts_nested_attributes_for :event_files, allow_destroy: true, reject_if: :all_blank
belongs_to :category
validates :category, presence: true
has_many :reservations, as: :reservable, dependent: :destroy
has_and_belongs_to_many :event_themes, join_table: 'events_event_themes', dependent: :destroy
has_many :event_price_categories, dependent: :destroy
has_many :price_categories, through: :event_price_categories
accepts_nested_attributes_for :event_price_categories, allow_destroy: true
belongs_to :age_range
belongs_to :availability, dependent: :destroy
accepts_nested_attributes_for :availability
attr_accessor :recurrence, :recurrence_end_at
before_save :update_nb_free_places
after_create :event_recurrence
# update event updated_at for index cache
after_save -> { touch }
def name
title
end
def themes
event_themes
end
def recurrence_events
Event.includes(:availability)
.where('events.recurrence_id = ? AND events.id != ? AND availabilities.start_at >= ?', recurrence_id, id, DateTime.current)
.references(:availabilities)
end
def safe_destroy
reservations = Reservation.where(reservable_type: 'Event', reservable_id: id)
if reservations.size.zero?
destroy
else
false
end
end
##
# @deprecated
# <b>DEPRECATED:</b> Please use <tt>event_price_categories</tt> instead.
# This method is for backward compatibility only, do not use in new code
def reduced_amount
if ActiveRecord::Base.connection.column_exists?(:events, :reduced_amount)
read_attribute(:reduced_amount)
else
pc = PriceCategory.find_by(name: I18n.t('price_category.reduced_fare'))
reduced_fare = event_price_categories.where(price_category: pc).first
if reduced_fare.nil?
nil
else
reduced_fare.amount
end
end
end
# def reservations
# Reservation.where(reservable: self)
# end
def update_nb_free_places
if nb_total_places.nil?
self.nb_free_places = nil
else
reserved_places = reservations.joins(:slots_reservations)
.where('slots_reservations.canceled_at': nil)
.map(&:total_booked_seats)
.inject(0) { |sum, t| sum + t }
self.nb_free_places = (nb_total_places - reserved_places)
end
end
def all_day?
availability.start_at.hour.zero?
end
private
def event_recurrence
return unless recurrence.present? && recurrence != 'none'
on = case recurrence
when 'day'
nil
when 'week'
availability.start_at.wday
when 'month'
availability.start_at.day
when 'year'
[availability.start_at.month, availability.start_at.day]
else
nil
end
r = Recurrence.new(every: recurrence, on: on, starts: availability.start_at + 1.day, until: recurrence_end_at)
service = Availabilities::CreateAvailabilitiesService.new
r.events.each do |date|
days_diff = availability.end_at.day - availability.start_at.day
start_at = DateTime.new(
date.year,
date.month,
date.day,
availability.start_at.hour,
availability.start_at.min,
availability.start_at.sec,
availability.start_at.zone
)
start_at = dst_correction(availability.start_at, start_at)
end_date = date + days_diff.days
end_at = DateTime.new(
end_date.year,
end_date.month,
end_date.day,
availability.end_at.hour,
availability.end_at.min,
availability.end_at.sec,
availability.end_at.zone
)
end_at = dst_correction(availability.start_at, end_at)
ei = EventImage.new(attachment: event_image.attachment) if event_image
efs = event_files.map do |f|
EventFile.new(attachment: f.attachment)
end
event_price_cats = []
event_price_categories.each do |epc|
event_price_cats.push(EventPriceCategory.new(price_category_id: epc.price_category_id, amount: epc.amount))
end
event = Event.new(
recurrence: 'none',
title: title,
description: description,
event_image: ei,
event_files: efs,
availability: Availability.new(start_at: start_at, end_at: end_at, available_type: 'event'),
availability_id: nil,
category_id: category_id,
age_range_id: age_range_id,
event_themes: event_themes,
amount: amount,
event_price_categories: event_price_cats,
nb_total_places: nb_total_places,
recurrence_id: id
)
event.save
service.create_slots(event.availability)
end
update_columns(recurrence_id: id)
end
end