timelog_controller.rb
290 lines
| 12.6 KiB
| text/x-ruby
|
RubyLexer
|
r569 | # redMine - project management software | ||
# Copyright (C) 2006-2007 Jean-Philippe Lang | ||||
# | ||||
# This program is free software; you can redistribute it and/or | ||||
# modify it under the terms of the GNU General Public License | ||||
# as published by the Free Software Foundation; either version 2 | ||||
# of the License, or (at your option) any later version. | ||||
# | ||||
# This program is distributed in the hope that it will be useful, | ||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||||
# GNU General Public License for more details. | ||||
# | ||||
# You should have received a copy of the GNU General Public License | ||||
# along with this program; if not, write to the Free Software | ||||
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | ||||
|
r365 | class TimelogController < ApplicationController | ||
|
r1062 | menu_item :issues | ||
|
r1777 | before_filter :find_project, :authorize, :only => [:edit, :destroy] | ||
before_filter :find_optional_project, :only => [:report, :details] | ||||
|
r365 | |||
|
r1235 | verify :method => :post, :only => :destroy, :redirect_to => { :action => :details } | ||
|
r365 | helper :sort | ||
include SortHelper | ||||
|
r783 | helper :issues | ||
|
r1159 | include TimelogHelper | ||
|
r1325 | helper :custom_fields | ||
include CustomFieldsHelper | ||||
|
r365 | |||
|
r569 | def report | ||
|
r1162 | @available_criterias = { 'project' => {:sql => "#{TimeEntry.table_name}.project_id", | ||
:klass => Project, | ||||
:label => :label_project}, | ||||
'version' => {:sql => "#{Issue.table_name}.fixed_version_id", | ||||
:klass => Version, | ||||
|
r569 | :label => :label_version}, | ||
'category' => {:sql => "#{Issue.table_name}.category_id", | ||||
|
r1162 | :klass => IssueCategory, | ||
|
r569 | :label => :field_category}, | ||
'member' => {:sql => "#{TimeEntry.table_name}.user_id", | ||||
|
r1162 | :klass => User, | ||
|
r569 | :label => :label_member}, | ||
'tracker' => {:sql => "#{Issue.table_name}.tracker_id", | ||||
|
r1162 | :klass => Tracker, | ||
|
r569 | :label => :label_tracker}, | ||
'activity' => {:sql => "#{TimeEntry.table_name}.activity_id", | ||||
|
r1162 | :klass => Enumeration, | ||
|
r1304 | :label => :label_activity}, | ||
'issue' => {:sql => "#{TimeEntry.table_name}.issue_id", | ||||
:klass => Issue, | ||||
:label => :label_issue} | ||||
|
r569 | } | ||
|
r1325 | # Add list and boolean custom fields as available criterias | ||
|
r1777 | custom_fields = (@project.nil? ? IssueCustomField.for_all : @project.all_issue_custom_fields) | ||
custom_fields.select {|cf| %w(list bool).include? cf.field_format }.each do |cf| | ||||
|
r1675 | @available_criterias["cf_#{cf.id}"] = {:sql => "(SELECT c.value FROM #{CustomValue.table_name} c WHERE c.custom_field_id = #{cf.id} AND c.customized_type = 'Issue' AND c.customized_id = #{Issue.table_name}.id)", | ||
|
r1325 | :format => cf.field_format, | ||
:label => cf.name} | ||||
|
r1777 | end if @project | ||
|
r1325 | |||
|
r1674 | # Add list and boolean time entry custom fields | ||
TimeEntryCustomField.find(:all).select {|cf| %w(list bool).include? cf.field_format }.each do |cf| | ||||
|
r1675 | @available_criterias["cf_#{cf.id}"] = {:sql => "(SELECT c.value FROM #{CustomValue.table_name} c WHERE c.custom_field_id = #{cf.id} AND c.customized_type = 'TimeEntry' AND c.customized_id = #{TimeEntry.table_name}.id)", | ||
|
r1674 | :format => cf.field_format, | ||
:label => cf.name} | ||||
end | ||||
|
r569 | @criterias = params[:criterias] || [] | ||
@criterias = @criterias.select{|criteria| @available_criterias.has_key? criteria} | ||||
@criterias.uniq! | ||||
|
r1162 | @criterias = @criterias[0,3] | ||
|
r569 | |||
|
r1311 | @columns = (params[:columns] && %w(year month week day).include?(params[:columns])) ? params[:columns] : 'month' | ||
|
r569 | |||
|
r1303 | retrieve_date_range | ||
|
r569 | |||
unless @criterias.empty? | ||||
sql_select = @criterias.collect{|criteria| @available_criterias[criteria][:sql] + " AS " + criteria}.join(', ') | ||||
sql_group_by = @criterias.collect{|criteria| @available_criterias[criteria][:sql]}.join(', ') | ||||
|
r1311 | sql = "SELECT #{sql_select}, tyear, tmonth, tweek, spent_on, SUM(hours) AS hours" | ||
|
r1162 | sql << " FROM #{TimeEntry.table_name}" | ||
sql << " LEFT JOIN #{Issue.table_name} ON #{TimeEntry.table_name}.issue_id = #{Issue.table_name}.id" | ||||
sql << " LEFT JOIN #{Project.table_name} ON #{TimeEntry.table_name}.project_id = #{Project.table_name}.id" | ||||
|
r1777 | sql << " WHERE" | ||
sql << " (%s) AND" % @project.project_condition(Setting.display_subprojects_issues?) if @project | ||||
sql << " (%s) AND" % Project.allowed_to_condition(User.current, :view_time_entries) | ||||
sql << " (spent_on BETWEEN '%s' AND '%s')" % [ActiveRecord::Base.connection.quoted_date(@from.to_time), ActiveRecord::Base.connection.quoted_date(@to.to_time)] | ||||
|
r1311 | sql << " GROUP BY #{sql_group_by}, tyear, tmonth, tweek, spent_on" | ||
|
r569 | |||
@hours = ActiveRecord::Base.connection.select_all(sql) | ||||
@hours.each do |row| | ||||
case @columns | ||||
when 'year' | ||||
row['year'] = row['tyear'] | ||||
when 'month' | ||||
row['month'] = "#{row['tyear']}-#{row['tmonth']}" | ||||
when 'week' | ||||
row['week'] = "#{row['tyear']}-#{row['tweek']}" | ||||
|
r1311 | when 'day' | ||
row['day'] = "#{row['spent_on']}" | ||||
|
r569 | end | ||
end | ||||
|
r1162 | |||
@total_hours = @hours.inject(0) {|s,k| s = s + k['hours'].to_f} | ||||
|
r1303 | |||
@periods = [] | ||||
# Date#at_beginning_of_ not supported in Rails 1.2.x | ||||
date_from = @from.to_time | ||||
# 100 columns max | ||||
while date_from <= @to.to_time && @periods.length < 100 | ||||
case @columns | ||||
when 'year' | ||||
@periods << "#{date_from.year}" | ||||
date_from = (date_from + 1.year).at_beginning_of_year | ||||
when 'month' | ||||
@periods << "#{date_from.year}-#{date_from.month}" | ||||
date_from = (date_from + 1.month).at_beginning_of_month | ||||
when 'week' | ||||
@periods << "#{date_from.year}-#{date_from.to_date.cweek}" | ||||
date_from = (date_from + 7.day).at_beginning_of_week | ||||
|
r1311 | when 'day' | ||
@periods << "#{date_from.to_date}" | ||||
date_from = date_from + 1.day | ||||
|
r1303 | end | ||
|
r569 | end | ||
end | ||||
|
r1323 | respond_to do |format| | ||
format.html { render :layout => !request.xhr? } | ||||
format.csv { send_data(report_to_csv(@criterias, @periods, @hours).read, :type => 'text/csv; header=present', :filename => 'timelog.csv') } | ||||
end | ||||
|
r569 | end | ||
|
r365 | def details | ||
sort_init 'spent_on', 'desc' | ||||
|
r2169 | sort_update 'spent_on' => 'spent_on', | ||
'user' => 'user_id', | ||||
'activity' => 'activity_id', | ||||
'project' => "#{Project.table_name}.name", | ||||
'issue' => 'issue_id', | ||||
'hours' => 'hours' | ||||
|
r365 | |||
|
r1162 | cond = ARCondition.new | ||
|
r1777 | if @project.nil? | ||
cond << Project.allowed_to_condition(User.current, :view_time_entries) | ||||
elsif @issue.nil? | ||||
cond << @project.project_condition(Setting.display_subprojects_issues?) | ||||
else | ||||
cond << ["#{TimeEntry.table_name}.issue_id = ?", @issue.id] | ||||
end | ||||
|
r1162 | |||
|
r1303 | retrieve_date_range | ||
|
r1311 | cond << ['spent_on BETWEEN ? AND ?', @from, @to] | ||
|
r1159 | |||
|
r1162 | TimeEntry.visible_by(User.current) do | ||
respond_to do |format| | ||||
format.html { | ||||
# Paginate results | ||||
@entry_count = TimeEntry.count(:include => :project, :conditions => cond.conditions) | ||||
@entry_pages = Paginator.new self, @entry_count, per_page_option, params['page'] | ||||
@entries = TimeEntry.find(:all, | ||||
:include => [:project, :activity, :user, {:issue => :tracker}], | ||||
:conditions => cond.conditions, | ||||
:order => sort_clause, | ||||
:limit => @entry_pages.items_per_page, | ||||
:offset => @entry_pages.current.offset) | ||||
@total_hours = TimeEntry.sum(:hours, :include => :project, :conditions => cond.conditions).to_f | ||||
|
r1311 | |||
|
r1162 | render :layout => !request.xhr? | ||
} | ||||
|
r1546 | format.atom { | ||
entries = TimeEntry.find(:all, | ||||
:include => [:project, :activity, :user, {:issue => :tracker}], | ||||
:conditions => cond.conditions, | ||||
:order => "#{TimeEntry.table_name}.created_on DESC", | ||||
:limit => Setting.feeds_limit.to_i) | ||||
render_feed(entries, :title => l(:label_spent_time)) | ||||
} | ||||
|
r1162 | format.csv { | ||
# Export all entries | ||||
@entries = TimeEntry.find(:all, | ||||
:include => [:project, :activity, :user, {:issue => [:tracker, :assigned_to, :priority]}], | ||||
:conditions => cond.conditions, | ||||
:order => sort_clause) | ||||
send_data(entries_to_csv(@entries).read, :type => 'text/csv; header=present', :filename => 'timelog.csv') | ||||
} | ||||
end | ||||
|
r1159 | end | ||
|
r365 | end | ||
def edit | ||||
|
r1235 | render_403 and return if @time_entry && !@time_entry.editable_by?(User.current) | ||
|
r906 | @time_entry ||= TimeEntry.new(:project => @project, :issue => @issue, :user => User.current, :spent_on => Date.today) | ||
|
r365 | @time_entry.attributes = params[:time_entry] | ||
if request.post? and @time_entry.save | ||||
flash[:notice] = l(:notice_successful_update) | ||||
|
r1891 | redirect_back_or_default :action => 'details', :project_id => @time_entry.project | ||
|
r365 | return | ||
end | ||||
end | ||||
|
r1235 | |||
def destroy | ||||
render_404 and return unless @time_entry | ||||
render_403 and return unless @time_entry.editable_by?(User.current) | ||||
@time_entry.destroy | ||||
flash[:notice] = l(:notice_successful_delete) | ||||
|
r1245 | redirect_to :back | ||
|
r1777 | rescue ::ActionController::RedirectBackError | ||
|
r1235 | redirect_to :action => 'details', :project_id => @time_entry.project | ||
end | ||||
|
r365 | |||
private | ||||
def find_project | ||||
if params[:id] | ||||
@time_entry = TimeEntry.find(params[:id]) | ||||
@project = @time_entry.project | ||||
elsif params[:issue_id] | ||||
@issue = Issue.find(params[:issue_id]) | ||||
@project = @issue.project | ||||
elsif params[:project_id] | ||||
@project = Project.find(params[:project_id]) | ||||
else | ||||
render_404 | ||||
return false | ||||
end | ||||
|
r1235 | rescue ActiveRecord::RecordNotFound | ||
render_404 | ||||
|
r365 | end | ||
|
r1303 | |||
|
r1777 | def find_optional_project | ||
if !params[:issue_id].blank? | ||||
@issue = Issue.find(params[:issue_id]) | ||||
@project = @issue.project | ||||
elsif !params[:project_id].blank? | ||||
@project = Project.find(params[:project_id]) | ||||
end | ||||
deny_access unless User.current.allowed_to?(:view_time_entries, @project, :global => true) | ||||
end | ||||
|
r1304 | # Retrieves the date range based on predefined ranges or specific from/to param dates | ||
|
r1303 | def retrieve_date_range | ||
@free_period = false | ||||
@from, @to = nil, nil | ||||
if params[:period_type] == '1' || (params[:period_type].nil? && !params[:period].nil?) | ||||
case params[:period].to_s | ||||
when 'today' | ||||
@from = @to = Date.today | ||||
when 'yesterday' | ||||
@from = @to = Date.today - 1 | ||||
when 'current_week' | ||||
@from = Date.today - (Date.today.cwday - 1)%7 | ||||
@to = @from + 6 | ||||
when 'last_week' | ||||
@from = Date.today - 7 - (Date.today.cwday - 1)%7 | ||||
@to = @from + 6 | ||||
when '7_days' | ||||
@from = Date.today - 7 | ||||
@to = Date.today | ||||
when 'current_month' | ||||
@from = Date.civil(Date.today.year, Date.today.month, 1) | ||||
@to = (@from >> 1) - 1 | ||||
when 'last_month' | ||||
@from = Date.civil(Date.today.year, Date.today.month, 1) << 1 | ||||
@to = (@from >> 1) - 1 | ||||
when '30_days' | ||||
@from = Date.today - 30 | ||||
@to = Date.today | ||||
when 'current_year' | ||||
@from = Date.civil(Date.today.year, 1, 1) | ||||
@to = Date.civil(Date.today.year, 12, 31) | ||||
end | ||||
elsif params[:period_type] == '2' || (params[:period_type].nil? && (!params[:from].nil? || !params[:to].nil?)) | ||||
begin; @from = params[:from].to_s.to_date unless params[:from].blank?; rescue; end | ||||
begin; @to = params[:to].to_s.to_date unless params[:to].blank?; rescue; end | ||||
@free_period = true | ||||
else | ||||
# default | ||||
end | ||||
@from, @to = @to, @from if @from && @to && @from > @to | ||||
|
r1777 | @from ||= (TimeEntry.minimum(:spent_on, :include => :project, :conditions => Project.allowed_to_condition(User.current, :view_time_entries)) || Date.today) - 1 | ||
@to ||= (TimeEntry.maximum(:spent_on, :include => :project, :conditions => Project.allowed_to_condition(User.current, :view_time_entries)) || Date.today) | ||||
|
r1303 | end | ||
|
r365 | end | ||