comparison .svn/pristine/3d/3d39eaf1a13ab4e59f2f962003d5f13e35d876e3.svn-base @ 1295:622f24f53b42 redmine-2.3

Update to Redmine SVN revision 11972 on 2.3-stable branch
author Chris Cannam
date Fri, 14 Jun 2013 09:02:21 +0100
parents
children
comparison
equal deleted inserted replaced
1294:3e4c3460b6ca 1295:622f24f53b42
1 # Redmine - project management software
2 # Copyright (C) 2006-2012 Jean-Philippe Lang
3 #
4 # This program is free software; you can redistribute it and/or
5 # modify it under the terms of the GNU General Public License
6 # as published by the Free Software Foundation; either version 2
7 # of the License, or (at your option) any later version.
8 #
9 # This program is distributed in the hope that it will be useful,
10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 # GNU General Public License for more details.
13 #
14 # You should have received a copy of the GNU General Public License
15 # along with this program; if not, write to the Free Software
16 # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
17
18 class TimelogController < ApplicationController
19 menu_item :issues
20
21 before_filter :find_project_for_new_time_entry, :only => [:create]
22 before_filter :find_time_entry, :only => [:show, :edit, :update]
23 before_filter :find_time_entries, :only => [:bulk_edit, :bulk_update, :destroy]
24 before_filter :authorize, :except => [:new, :index, :report]
25
26 before_filter :find_optional_project, :only => [:index, :report]
27 before_filter :find_optional_project_for_new_time_entry, :only => [:new]
28 before_filter :authorize_global, :only => [:new, :index, :report]
29
30 accept_rss_auth :index
31 accept_api_auth :index, :show, :create, :update, :destroy
32
33 helper :sort
34 include SortHelper
35 helper :issues
36 include TimelogHelper
37 helper :custom_fields
38 include CustomFieldsHelper
39
40 def index
41 sort_init 'spent_on', 'desc'
42 sort_update 'spent_on' => ['spent_on', "#{TimeEntry.table_name}.created_on"],
43 'user' => 'user_id',
44 'activity' => 'activity_id',
45 'project' => "#{Project.table_name}.name",
46 'issue' => 'issue_id',
47 'hours' => 'hours'
48
49 retrieve_date_range
50
51 scope = TimeEntry.visible.spent_between(@from, @to)
52 if @issue
53 scope = scope.on_issue(@issue)
54 elsif @project
55 scope = scope.on_project(@project, Setting.display_subprojects_issues?)
56 end
57
58 respond_to do |format|
59 format.html {
60 # Paginate results
61 @entry_count = scope.count
62 @entry_pages = Paginator.new self, @entry_count, per_page_option, params['page']
63 @entries = scope.all(
64 :include => [:project, :activity, :user, {:issue => :tracker}],
65 :order => sort_clause,
66 :limit => @entry_pages.items_per_page,
67 :offset => @entry_pages.current.offset
68 )
69 @total_hours = scope.sum(:hours).to_f
70
71 render :layout => !request.xhr?
72 }
73 format.api {
74 @entry_count = scope.count
75 @offset, @limit = api_offset_and_limit
76 @entries = scope.all(
77 :include => [:project, :activity, :user, {:issue => :tracker}],
78 :order => sort_clause,
79 :limit => @limit,
80 :offset => @offset
81 )
82 }
83 format.atom {
84 entries = scope.all(
85 :include => [:project, :activity, :user, {:issue => :tracker}],
86 :order => "#{TimeEntry.table_name}.created_on DESC",
87 :limit => Setting.feeds_limit.to_i
88 )
89 render_feed(entries, :title => l(:label_spent_time))
90 }
91 format.csv {
92 # Export all entries
93 @entries = scope.all(
94 :include => [:project, :activity, :user, {:issue => [:tracker, :assigned_to, :priority]}],
95 :order => sort_clause
96 )
97 send_data(entries_to_csv(@entries), :type => 'text/csv; header=present', :filename => 'timelog.csv')
98 }
99 end
100 end
101
102 def report
103 retrieve_date_range
104 @report = Redmine::Helpers::TimeReport.new(@project, @issue, params[:criteria], params[:columns], @from, @to)
105
106 respond_to do |format|
107 format.html { render :layout => !request.xhr? }
108 format.csv { send_data(report_to_csv(@report), :type => 'text/csv; header=present', :filename => 'timelog.csv') }
109 end
110 end
111
112 def show
113 respond_to do |format|
114 # TODO: Implement html response
115 format.html { render :nothing => true, :status => 406 }
116 format.api
117 end
118 end
119
120 def new
121 @time_entry ||= TimeEntry.new(:project => @project, :issue => @issue, :user => User.current, :spent_on => User.current.today)
122 @time_entry.safe_attributes = params[:time_entry]
123 end
124
125 def create
126 @time_entry ||= TimeEntry.new(:project => @project, :issue => @issue, :user => User.current, :spent_on => User.current.today)
127 @time_entry.safe_attributes = params[:time_entry]
128
129 call_hook(:controller_timelog_edit_before_save, { :params => params, :time_entry => @time_entry })
130
131 if @time_entry.save
132 respond_to do |format|
133 format.html {
134 flash[:notice] = l(:notice_successful_create)
135 if params[:continue]
136 if params[:project_id]
137 redirect_to :action => 'new', :project_id => @time_entry.project, :issue_id => @time_entry.issue,
138 :time_entry => {:issue_id => @time_entry.issue_id, :activity_id => @time_entry.activity_id},
139 :back_url => params[:back_url]
140 else
141 redirect_to :action => 'new',
142 :time_entry => {:project_id => @time_entry.project_id, :issue_id => @time_entry.issue_id, :activity_id => @time_entry.activity_id},
143 :back_url => params[:back_url]
144 end
145 else
146 redirect_back_or_default :action => 'index', :project_id => @time_entry.project
147 end
148 }
149 format.api { render :action => 'show', :status => :created, :location => time_entry_url(@time_entry) }
150 end
151 else
152 respond_to do |format|
153 format.html { render :action => 'new' }
154 format.api { render_validation_errors(@time_entry) }
155 end
156 end
157 end
158
159 def edit
160 @time_entry.safe_attributes = params[:time_entry]
161 end
162
163 def update
164 @time_entry.safe_attributes = params[:time_entry]
165
166 call_hook(:controller_timelog_edit_before_save, { :params => params, :time_entry => @time_entry })
167
168 if @time_entry.save
169 respond_to do |format|
170 format.html {
171 flash[:notice] = l(:notice_successful_update)
172 redirect_back_or_default :action => 'index', :project_id => @time_entry.project
173 }
174 format.api { render_api_ok }
175 end
176 else
177 respond_to do |format|
178 format.html { render :action => 'edit' }
179 format.api { render_validation_errors(@time_entry) }
180 end
181 end
182 end
183
184 def bulk_edit
185 @available_activities = TimeEntryActivity.shared.active
186 @custom_fields = TimeEntry.first.available_custom_fields
187 end
188
189 def bulk_update
190 attributes = parse_params_for_bulk_time_entry_attributes(params)
191
192 unsaved_time_entry_ids = []
193 @time_entries.each do |time_entry|
194 time_entry.reload
195 time_entry.safe_attributes = attributes
196 call_hook(:controller_time_entries_bulk_edit_before_save, { :params => params, :time_entry => time_entry })
197 unless time_entry.save
198 # Keep unsaved time_entry ids to display them in flash error
199 unsaved_time_entry_ids << time_entry.id
200 end
201 end
202 set_flash_from_bulk_time_entry_save(@time_entries, unsaved_time_entry_ids)
203 redirect_back_or_default({:controller => 'timelog', :action => 'index', :project_id => @projects.first})
204 end
205
206 def destroy
207 destroyed = TimeEntry.transaction do
208 @time_entries.each do |t|
209 unless t.destroy && t.destroyed?
210 raise ActiveRecord::Rollback
211 end
212 end
213 end
214
215 respond_to do |format|
216 format.html {
217 if destroyed
218 flash[:notice] = l(:notice_successful_delete)
219 else
220 flash[:error] = l(:notice_unable_delete_time_entry)
221 end
222 redirect_back_or_default(:action => 'index', :project_id => @projects.first)
223 }
224 format.api {
225 if destroyed
226 render_api_ok
227 else
228 render_validation_errors(@time_entries)
229 end
230 }
231 end
232 end
233
234 private
235 def find_time_entry
236 @time_entry = TimeEntry.find(params[:id])
237 unless @time_entry.editable_by?(User.current)
238 render_403
239 return false
240 end
241 @project = @time_entry.project
242 rescue ActiveRecord::RecordNotFound
243 render_404
244 end
245
246 def find_time_entries
247 @time_entries = TimeEntry.find_all_by_id(params[:id] || params[:ids])
248 raise ActiveRecord::RecordNotFound if @time_entries.empty?
249 @projects = @time_entries.collect(&:project).compact.uniq
250 @project = @projects.first if @projects.size == 1
251 rescue ActiveRecord::RecordNotFound
252 render_404
253 end
254
255 def set_flash_from_bulk_time_entry_save(time_entries, unsaved_time_entry_ids)
256 if unsaved_time_entry_ids.empty?
257 flash[:notice] = l(:notice_successful_update) unless time_entries.empty?
258 else
259 flash[:error] = l(:notice_failed_to_save_time_entries,
260 :count => unsaved_time_entry_ids.size,
261 :total => time_entries.size,
262 :ids => '#' + unsaved_time_entry_ids.join(', #'))
263 end
264 end
265
266 def find_optional_project_for_new_time_entry
267 if (project_id = (params[:project_id] || params[:time_entry] && params[:time_entry][:project_id])).present?
268 @project = Project.find(project_id)
269 end
270 if (issue_id = (params[:issue_id] || params[:time_entry] && params[:time_entry][:issue_id])).present?
271 @issue = Issue.find(issue_id)
272 @project ||= @issue.project
273 end
274 rescue ActiveRecord::RecordNotFound
275 render_404
276 end
277
278 def find_project_for_new_time_entry
279 find_optional_project_for_new_time_entry
280 if @project.nil?
281 render_404
282 end
283 end
284
285 def find_optional_project
286 if !params[:issue_id].blank?
287 @issue = Issue.find(params[:issue_id])
288 @project = @issue.project
289 elsif !params[:project_id].blank?
290 @project = Project.find(params[:project_id])
291 end
292 end
293
294 # Retrieves the date range based on predefined ranges or specific from/to param dates
295 def retrieve_date_range
296 @free_period = false
297 @from, @to = nil, nil
298
299 if params[:period_type] == '1' || (params[:period_type].nil? && !params[:period].nil?)
300 case params[:period].to_s
301 when 'today'
302 @from = @to = Date.today
303 when 'yesterday'
304 @from = @to = Date.today - 1
305 when 'current_week'
306 @from = Date.today - (Date.today.cwday - 1)%7
307 @to = @from + 6
308 when 'last_week'
309 @from = Date.today - 7 - (Date.today.cwday - 1)%7
310 @to = @from + 6
311 when 'last_2_weeks'
312 @from = Date.today - 14 - (Date.today.cwday - 1)%7
313 @to = @from + 13
314 when '7_days'
315 @from = Date.today - 7
316 @to = Date.today
317 when 'current_month'
318 @from = Date.civil(Date.today.year, Date.today.month, 1)
319 @to = (@from >> 1) - 1
320 when 'last_month'
321 @from = Date.civil(Date.today.year, Date.today.month, 1) << 1
322 @to = (@from >> 1) - 1
323 when '30_days'
324 @from = Date.today - 30
325 @to = Date.today
326 when 'current_year'
327 @from = Date.civil(Date.today.year, 1, 1)
328 @to = Date.civil(Date.today.year, 12, 31)
329 end
330 elsif params[:period_type] == '2' || (params[:period_type].nil? && (!params[:from].nil? || !params[:to].nil?))
331 begin; @from = params[:from].to_s.to_date unless params[:from].blank?; rescue; end
332 begin; @to = params[:to].to_s.to_date unless params[:to].blank?; rescue; end
333 @free_period = true
334 else
335 # default
336 end
337
338 @from, @to = @to, @from if @from && @to && @from > @to
339 end
340
341 def parse_params_for_bulk_time_entry_attributes(params)
342 attributes = (params[:time_entry] || {}).reject {|k,v| v.blank?}
343 attributes.keys.each {|k| attributes[k] = '' if attributes[k] == 'none'}
344 attributes[:custom_field_values].reject! {|k,v| v.blank?} if attributes[:custom_field_values]
345 attributes
346 end
347 end