Users (Basic) Time Sheets (Advanced) Issues (API)

Developer Project Company Report hours
Wiley Wuckert odio velit repudiandae
Apple
64.0
Elise Waelchi quibusdam sed sit
Apple
64.0
Gonzalo Blanda doloribus hic corporis
Microsoft
64.0
Crissy Brekke animi aperiam tenetur
Google
64.0
Desmond Murray doloribus hic corporis
Microsoft
64.0
Rossana Schneider suscipit consectetur beatae
Google
64.0
Terisa Ruecker doloribus hic corporis
Microsoft
64.0
Alex Legros repudiandae officiis qui
Google
64.0
Shannon Botsford animi aperiam tenetur
Google
64.0
Desmond Murray animi aperiam tenetur
Google
64.0
Jewell Swaniawski suscipit consectetur beatae
Google
64.0
Gwendolyn Satterfield perspiciatis totam perferendis
Apple
64.0
Raquel Runolfsson animi aperiam tenetur
Google
64.0
Blake Hoppe doloribus hic corporis
Microsoft
64.0
Elise Waelchi animi aperiam tenetur
Google
63.0
Jamie Stehr animi aperiam tenetur
Google
63.0
Almeta Adams animi aperiam tenetur
Google
63.0
Wiley Wuckert suscipit consectetur beatae
Google
63.0
Gonzalo Blanda repudiandae officiis qui
Google
63.0
Gonzalo Blanda perspiciatis totam perferendis
Apple
63.0
Raquel Runolfsson eum amet quia
Google
63.0
Jamison Hagenes eum amet quia
Google
63.0
Elisa Wintheiser repudiandae officiis qui
Google
63.0
Almeta Adams repudiandae officiis qui
Google
63.0
Johnny Spinka eum amet quia
Google
63.0

Grid:

class TimeEntriesGrid < ApplicationGrid

  #
  # Scope
  #

  scope do
    User.select(
      "users.name",
      "projects.name as project_name",
      "accounts.name as account_name",
      "sum(time_entries.hours) as report_hours"
    ).joins(
      time_entries: {project: :account}
    ).group("projects.name", "users.name", "accounts.name").
    order("users.name")
  end


  #
  # Filters
  #

  filter(
    :project_id, :enum,
    :select => :project_id_select,
    :multiple => true,
    :include_blank => false
  ) do |value|
    self.where(:time_entries => {:project_id => value})
  end


  filter(
    :year, :enum,
    :select => :year_select,
    :include_blank => false,
    :default => lambda {Date.today.year}
  ) do |value|
    self.where([
      "extract(year from time_entries.date) = ?",
      value.to_i
    ])
  end

  filter(
    :month, :enum,
    :select => :month_select,
    :include_blank => false,
    :default => lambda {Date.today.month}
  ) do |value|

    self.where([
      "extract(month from time_entries.date) = ?",
      value.to_i
    ])
  end


  #
  # Columns
  #

  column(:user_name, :header => "Developer", :order => "users.name") do
    self.name
  end
  column(:project_name, :header => "Project", :order => "projects.name")
  column(
    :account_name,
    :header => "Company",
    :order => "accounts.name",
  ) do |model|
    format(model.account_name) do
      render :partial => "time_entries/company", :locals => {:model => model}
    end
  end

  column(:report_hours, order: "sum(time_entries.hours)")

  protected

  def year_select
    TimeEntry.all.any? ? (TimeEntry.minimum(:date).year..TimeEntry.maximum(:date).year) : []
  end
  def month_select
    Date::MONTHNAMES[1..12].enum_for(:each_with_index).collect {|name, index| [name, index + 1]}
  end

  def project_id_select
    Project.all.map {|p| [p.name, p.id]}
  end
end

Controller:

class TimeEntriesController < ApplicationController

  def index
    @time_entries_grid = TimeEntriesGrid.new(params[:g]) do |scope|
      scope.page(params[:page] )
    end
  end
end

View:

.grid-index-grid
  %div{class: 'overflow-x-auto p-4'}
    = datagrid_form_with model: @time_entries_grid, url: time_entries_path
    %br/
    = datagrid_table(@time_entries_grid)
    = paginate @time_entries_grid.assets, window: 2
  %div
    = render :partial => "shared/source", :object => @time_entries_grid