123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297 |
- # Redmine - project management software
- # Copyright (C) 2006-2014 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.
-
- class TimelogController < ApplicationController
- menu_item :issues
-
- before_filter :find_project_for_new_time_entry, :only => [:create]
- before_filter :find_time_entry, :only => [:show, :edit, :update]
- before_filter :find_time_entries, :only => [:bulk_edit, :bulk_update, :destroy]
- before_filter :authorize, :except => [:new, :index, :report]
-
- before_filter :find_optional_project, :only => [:index, :report]
- before_filter :find_optional_project_for_new_time_entry, :only => [:new]
- before_filter :authorize_global, :only => [:new, :index, :report]
-
- accept_rss_auth :index
- accept_api_auth :index, :show, :create, :update, :destroy
-
- rescue_from Query::StatementInvalid, :with => :query_statement_invalid
-
- helper :sort
- include SortHelper
- helper :issues
- include TimelogHelper
- helper :custom_fields
- include CustomFieldsHelper
- helper :queries
- include QueriesHelper
-
- def index
- @query = TimeEntryQuery.build_from_params(params, :project => @project, :name => '_')
-
- sort_init(@query.sort_criteria.empty? ? [['spent_on', 'desc']] : @query.sort_criteria)
- sort_update(@query.sortable_columns)
- scope = time_entry_scope(:order => sort_clause).
- includes(:project, :user, :issue).
- preload(:issue => [:project, :tracker, :status, :assigned_to, :priority])
-
- respond_to do |format|
- format.html {
- @entry_count = scope.count
- @entry_pages = Paginator.new @entry_count, per_page_option, params['page']
- @entries = scope.offset(@entry_pages.offset).limit(@entry_pages.per_page).all
- @total_hours = scope.sum(:hours).to_f
-
- render :layout => !request.xhr?
- }
- format.api {
- @entry_count = scope.count
- @offset, @limit = api_offset_and_limit
- @entries = scope.offset(@offset).limit(@limit).preload(:custom_values => :custom_field).all
- }
- format.atom {
- entries = scope.limit(Setting.feeds_limit.to_i).reorder("#{TimeEntry.table_name}.created_on DESC").all
- render_feed(entries, :title => l(:label_spent_time))
- }
- format.csv {
- # Export all entries
- @entries = scope.all
- send_data(query_to_csv(@entries, @query, params), :type => 'text/csv; header=present', :filename => 'timelog.csv')
- }
- end
- end
-
- def report
- @query = TimeEntryQuery.build_from_params(params, :project => @project, :name => '_')
- scope = time_entry_scope
-
- @report = Redmine::Helpers::TimeReport.new(@project, @issue, params[:criteria], params[:columns], scope)
-
- respond_to do |format|
- format.html { render :layout => !request.xhr? }
- format.csv { send_data(report_to_csv(@report), :type => 'text/csv; header=present', :filename => 'timelog.csv') }
- end
- end
-
- def show
- respond_to do |format|
- # TODO: Implement html response
- format.html { render :nothing => true, :status => 406 }
- format.api
- end
- end
-
- def new
- @time_entry ||= TimeEntry.new(:project => @project, :issue => @issue, :user => User.current, :spent_on => User.current.today)
- @time_entry.safe_attributes = params[:time_entry]
- end
-
- def create
- @time_entry ||= TimeEntry.new(:project => @project, :issue => @issue, :user => User.current, :spent_on => User.current.today)
- @time_entry.safe_attributes = params[:time_entry]
-
- call_hook(:controller_timelog_edit_before_save, { :params => params, :time_entry => @time_entry })
-
- if @time_entry.save
- respond_to do |format|
- format.html {
- flash[:notice] = l(:notice_successful_create)
- if params[:continue]
- if params[:project_id]
- options = {
- :time_entry => {:issue_id => @time_entry.issue_id, :activity_id => @time_entry.activity_id},
- :back_url => params[:back_url]
- }
- if @time_entry.issue
- redirect_to new_project_issue_time_entry_path(@time_entry.project, @time_entry.issue, options)
- else
- redirect_to new_project_time_entry_path(@time_entry.project, options)
- end
- else
- options = {
- :time_entry => {:project_id => @time_entry.project_id, :issue_id => @time_entry.issue_id, :activity_id => @time_entry.activity_id},
- :back_url => params[:back_url]
- }
- redirect_to new_time_entry_path(options)
- end
- else
- redirect_back_or_default project_time_entries_path(@time_entry.project)
- end
- }
- format.api { render :action => 'show', :status => :created, :location => time_entry_url(@time_entry) }
- end
- else
- respond_to do |format|
- format.html { render :action => 'new' }
- format.api { render_validation_errors(@time_entry) }
- end
- end
- end
-
- def edit
- @time_entry.safe_attributes = params[:time_entry]
- end
-
- def update
- @time_entry.safe_attributes = params[:time_entry]
-
- call_hook(:controller_timelog_edit_before_save, { :params => params, :time_entry => @time_entry })
-
- if @time_entry.save
- respond_to do |format|
- format.html {
- flash[:notice] = l(:notice_successful_update)
- redirect_back_or_default project_time_entries_path(@time_entry.project)
- }
- format.api { render_api_ok }
- end
- else
- respond_to do |format|
- format.html { render :action => 'edit' }
- format.api { render_validation_errors(@time_entry) }
- end
- end
- end
-
- def bulk_edit
- @available_activities = TimeEntryActivity.shared.active
- @custom_fields = TimeEntry.first.available_custom_fields
- end
-
- def bulk_update
- attributes = parse_params_for_bulk_time_entry_attributes(params)
-
- unsaved_time_entry_ids = []
- @time_entries.each do |time_entry|
- time_entry.reload
- time_entry.safe_attributes = attributes
- call_hook(:controller_time_entries_bulk_edit_before_save, { :params => params, :time_entry => time_entry })
- unless time_entry.save
- logger.info "time entry could not be updated: #{time_entry.errors.full_messages}" if logger && logger.info
- # Keep unsaved time_entry ids to display them in flash error
- unsaved_time_entry_ids << time_entry.id
- end
- end
- set_flash_from_bulk_time_entry_save(@time_entries, unsaved_time_entry_ids)
- redirect_back_or_default project_time_entries_path(@projects.first)
- end
-
- def destroy
- destroyed = TimeEntry.transaction do
- @time_entries.each do |t|
- unless t.destroy && t.destroyed?
- raise ActiveRecord::Rollback
- end
- end
- end
-
- respond_to do |format|
- format.html {
- if destroyed
- flash[:notice] = l(:notice_successful_delete)
- else
- flash[:error] = l(:notice_unable_delete_time_entry)
- end
- redirect_back_or_default project_time_entries_path(@projects.first)
- }
- format.api {
- if destroyed
- render_api_ok
- else
- render_validation_errors(@time_entries)
- end
- }
- end
- end
-
- private
- def find_time_entry
- @time_entry = TimeEntry.find(params[:id])
- unless @time_entry.editable_by?(User.current)
- render_403
- return false
- end
- @project = @time_entry.project
- rescue ActiveRecord::RecordNotFound
- render_404
- end
-
- def find_time_entries
- @time_entries = TimeEntry.find_all_by_id(params[:id] || params[:ids])
- raise ActiveRecord::RecordNotFound if @time_entries.empty?
- @projects = @time_entries.collect(&:project).compact.uniq
- @project = @projects.first if @projects.size == 1
- rescue ActiveRecord::RecordNotFound
- render_404
- end
-
- def set_flash_from_bulk_time_entry_save(time_entries, unsaved_time_entry_ids)
- if unsaved_time_entry_ids.empty?
- flash[:notice] = l(:notice_successful_update) unless time_entries.empty?
- else
- flash[:error] = l(:notice_failed_to_save_time_entries,
- :count => unsaved_time_entry_ids.size,
- :total => time_entries.size,
- :ids => '#' + unsaved_time_entry_ids.join(', #'))
- end
- end
-
- def find_optional_project_for_new_time_entry
- if (project_id = (params[:project_id] || params[:time_entry] && params[:time_entry][:project_id])).present?
- @project = Project.find(project_id)
- end
- if (issue_id = (params[:issue_id] || params[:time_entry] && params[:time_entry][:issue_id])).present?
- @issue = Issue.find(issue_id)
- @project ||= @issue.project
- end
- rescue ActiveRecord::RecordNotFound
- render_404
- end
-
- def find_project_for_new_time_entry
- find_optional_project_for_new_time_entry
- if @project.nil?
- render_404
- end
- end
-
- 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
- end
-
- # Returns the TimeEntry scope for index and report actions
- def time_entry_scope(options={})
- scope = @query.results_scope(options)
- if @issue
- scope = scope.on_issue(@issue)
- end
- scope
- end
-
- def parse_params_for_bulk_time_entry_attributes(params)
- attributes = (params[:time_entry] || {}).reject {|k,v| v.blank?}
- attributes.keys.each {|k| attributes[k] = '' if attributes[k] == 'none'}
- attributes[:custom_field_values].reject! {|k,v| v.blank?} if attributes[:custom_field_values]
- attributes
- end
- end
|