You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

205 lines
6.2 KiB

class EntriesController < ApplicationController
before_action :authenticate_user!
before_action :set_entry, only: %i[edit update destroy stop_timer]
def index
@entries = current_user.entries.order(date: :desc)
@running_entry = current_user.entries.find_by(end_time: nil, beschreibung: "Timer")
# Gesamtzeit in Minuten
@total_minutes = @entries.sum { |e| e.hours.to_i * 60 + e.minutes.to_i }
@total_minutes_praktikum_prop = @entries.where(praktikums_typ: Entry::PRAKTIKUMSTYPEN[0], entry_art: Entry::ENTRY_ARTEN[0]).sum { |e| e.hours.to_i * 60 + e.minutes.to_i }
@total_minutes_praktikum_fach = @entries.where(praktikums_typ: Entry::PRAKTIKUMSTYPEN[1], entry_art: Entry::ENTRY_ARTEN[0]).sum { |e| e.hours.to_i * 60 + e.minutes.to_i }
# Gesamtbetrag der Kilometerpauschale
@total_kilometer_pauschale = @entries.sum(&:kilometer_pauschale)
# Zeitverbrauch je Kombination (typ + art)
@time_by_typ_art = @entries.group_by(&:praktikums_typ).transform_values do |group|
group.group_by(&:entry_art).transform_values do |entries|
entries.sum { |e| e.hours.to_i * 60 + e.minutes.to_i }
end
end
# Verbleibende Minuten je Kombination
@remaining_minutes_matrix = {}
@time_by_typ_art.each do |typ, arts|
@remaining_minutes_matrix[typ] ||= {}
arts.each do |art, spent_minutes|
target = current_user.required_hours_matrix.dig(typ, art).to_i * 60
remaining = [target - spent_minutes, 0].max
@remaining_minutes_matrix[typ][art] = remaining
end
end
@completion_percent_by_typ_art = {}
User::PRAKTIKUMSTYPEN.product(User::ENTRY_ARTEN).each do |typ, art|
total_required = current_user.required_hours_matrix.to_h.dig(typ, art).to_f
remaining_minutes = @remaining_minutes_matrix.dig(typ, art).to_i
required_minutes = (total_required * 60).to_i
entries_exist = current_user.entries.where(praktikums_typ: typ, entry_art: art).exists?
if total_required > 0 && entries_exist
done_minutes = required_minutes - remaining_minutes
percent = (done_minutes / required_minutes.to_f * 100).round
percent = 100 if percent > 100
@completion_percent_by_typ_art[[typ, art]] = percent
else
@completion_percent_by_typ_art[[typ, art]] = 0
end
end
@total_kilometer_costs_by_year = Entry.total_kilometer_cost_by_year(current_user)
@fortbildungskosten_by_year = Entry.total_fortbildungskosten_by_year(current_user)
@selbstsupervision_by_year = Entry.total_supervision_by_year(current_user)
@selbsterfahrungskosten_by_year = Entry.total_selbsterfahrungskosten_by_year(current_user)
@allekosten_by_year = Entry.total_semesterkosten_by_year(current_user)
# Voraussichtliches Ende je Kombination basierend auf weekly_target_matrix
#
@estimated_end_by_typ_art = {}
@remaining_minutes_matrix.each do |typ, arts|
@estimated_end_by_typ_art[typ] ||= {}
arts.each do |art, remaining_minutes|
hours_remaining = remaining_minutes / 60.0
weekly_hours = current_user.weekly_target_matrix.dig(typ, art).to_f
if weekly_hours > 0
weeks_remaining = (hours_remaining / weekly_hours).ceil
@estimated_end_by_typ_art[typ][art] = Date.today + weeks_remaining.weeks
else
@estimated_end_by_typ_art[typ][art] = nil
end
end
end
end
def new
@entry = current_user.entries.build
end
def start_timer
typ = params[:typ]
art = params[:art]
@entry = current_user.entries.create!(start_time: Time.current, beschreibung: "Timer", praktikums_typ: typ, entry_art: art)
redirect_to entries_path, notice: "Timer gestartet"
end
# neue Aktion: Timer stoppen
def stop_timer
#@entry = current_user.entries.where(end_time: nil).order(start_time: :desc).first
if @entry
@entry.end_time = Time.current
@entry.lunch_break_minutes = 30 if ActiveModel::Type::Boolean.new.cast(params[:lunch_break])
total_minutes = @entry.total_minutes_including_break
@entry.hours = total_minutes / 60
@entry.minutes = total_minutes % 60
@entry.save!
notice = "Eintrag gestoppt – Gesamtzeit: #{@entry.hours} h #{@entry.minutes} min"
else
notice = "Kein laufender Eintrag gefunden"
end
redirect_to entries_path, notice: notice
end
def create
@entry = current_user.entries.new(entry_params)
if current_user.praepedeutikum_abgeschlossen? && @entry.praktikums_typ == 'propädeutikum'
redirect_to entries_path, alert: "Propädeutikum ist bereits abgeschlossen – Neuer Eintrag dieses Typs ist nicht erlaubt."
return
end
if @entry.save
redirect_to entries_path, notice: "Eintrag gespeichert"
else
render :new
end
end
def edit
@entry
end
def update
if @entry.update(entry_params)
redirect_to entries_path, notice: "Eintrag aktualisiert"
else
render :edit
end
end
def destroy
@entry.destroy
redirect_to entries_path, notice: "Eintrag gelöscht"
end
def export_csv
@entries = current_user.entries.order(date: :desc)
respond_to do |format|
format.csv { send_data @entries.to_csv, filename: "eintraege-#{Date.today}.csv" }
end
end
def monthly_report
# Nur die wirklich benötigten Spalten abfragen
rows = current_user.entries
.select(
Arel.sql("DATE_TRUNC('month', date) AS month"),
:praktikums_typ,
:entry_art,
Arel.sql("SUM(hours * 60 + minutes) AS total_minutes")
)
.group(
Arel.sql("DATE_TRUNC('month', date)"),
:praktikums_typ,
:entry_art
)
.order(Arel.sql("DATE_TRUNC('month', date) DESC"))
# In Hash-Struktur für die View umwandeln
@report = {}
rows.each do |r|
month = r.attributes["month"].to_date
total = r.attributes["total_minutes"].to_i
typ_art = [r.praktikums_typ, r.entry_art]
@report[month] ||= {}
@report[month][typ_art] = {
total_minutes: total,
hours: total / 60,
minutes: total % 60
}
end
end
private
def set_entry
@entry = current_user.entries.find(params[:id])
end
def entry_params
params.require(:entry).permit(
:date,
:hours,
:minutes,
:praktikums_typ,
:entry_art,
:distance_km,
:beschreibung,
:kosten,
:zaehlt_als_fortbildung
)
end
end