diff options
-rw-r--r-- | lib/redmine/helpers/gantt.rb | 274 |
1 files changed, 127 insertions, 147 deletions
diff --git a/lib/redmine/helpers/gantt.rb b/lib/redmine/helpers/gantt.rb index 64d301fa5..66ab3148a 100644 --- a/lib/redmine/helpers/gantt.rb +++ b/lib/redmine/helpers/gantt.rb @@ -283,149 +283,54 @@ module Redmine end def subject_for_project(project, options) - case options[:format] - when :html - html_class = "" - html_class << 'icon icon-projects ' - html_class << (project.overdue? ? 'project-overdue' : '') - s = view.link_to_project(project).html_safe - subject = view.content_tag(:span, s, - :class => html_class).html_safe - html_subject(options, subject, :css => "project-name") - when :image - image_subject(options, project.name) - when :pdf - pdf_new_page?(options) - pdf_subject(options, project.name) - end + subject(project.name, options, project) end def line_for_project(project, options) # Skip versions that don't have a start_date or due date if project.is_a?(Project) && project.start_date && project.due_date - options[:zoom] ||= 1 - options[:g_width] ||= (self.date_to - self.date_from + 1) * options[:zoom] - coords = coordinates(project.start_date, project.due_date, nil, options[:zoom]) - label = h(project) - case options[:format] - when :html - html_task(options, coords, :css => "project task", :label => label, :markers => true) - when :image - image_task(options, coords, :label => label, :markers => true, :height => 3) - when :pdf - pdf_task(options, coords, :label => label, :markers => true, :height => 0.8) - end - else - '' + label = project.name + line(project.start_date, project.due_date, nil, true, label, options, project) end end def subject_for_version(version, options) - case options[:format] - when :html - html_class = "" - html_class << 'icon icon-package ' - html_class << (version.behind_schedule? ? 'version-behind-schedule' : '') << " " - html_class << (version.overdue? ? 'version-overdue' : '') - html_class << ' version-closed' unless version.open? - if version.start_date && version.due_date && version.completed_percent - progress_date = calc_progress_date(version.start_date, - version.due_date, version.completed_percent) - html_class << ' behind-start-date' if progress_date < self.date_from - html_class << ' over-end-date' if progress_date > self.date_to - end - s = view.link_to_version(version).html_safe - subject = view.content_tag(:span, s, - :class => html_class).html_safe - html_subject(options, subject, :css => "version-name", - :id => "version-#{version.id}") - when :image - image_subject(options, version.to_s_with_project) - when :pdf - pdf_new_page?(options) - pdf_subject(options, version.to_s_with_project) - end + subject(version.to_s_with_project, options, version) end def line_for_version(version, options) # Skip versions that don't have a start_date if version.is_a?(Version) && version.due_date && version.start_date - options[:zoom] ||= 1 - options[:g_width] ||= (self.date_to - self.date_from + 1) * options[:zoom] - coords = coordinates(version.start_date, - version.due_date, version.completed_percent, - options[:zoom]) label = "#{h(version)} #{h(version.completed_percent.to_f.round)}%" label = h("#{version.project} -") + label unless @project && @project == version.project - case options[:format] - when :html - html_task(options, coords, :css => "version task", - :label => label, :markers => true, :version => version) - when :image - image_task(options, coords, :label => label, :markers => true, :height => 3) - when :pdf - pdf_task(options, coords, :label => label, :markers => true, :height => 0.8) - end - else - '' + line(version.start_date, version.due_date, version.completed_percent, true, label, options, version) end end def subject_for_issue(issue, options) - case options[:format] - when :html - css_classes = '' - css_classes << ' issue-overdue' if issue.overdue? - css_classes << ' issue-behind-schedule' if issue.behind_schedule? - css_classes << ' icon icon-issue' unless Setting.gravatar_enabled? && issue.assigned_to - css_classes << ' issue-closed' if issue.closed? - if issue.start_date && issue.due_before && issue.done_ratio - progress_date = calc_progress_date(issue.start_date, - issue.due_before, issue.done_ratio) - css_classes << ' behind-start-date' if progress_date < self.date_from - css_classes << ' over-end-date' if progress_date > self.date_to - end - s = "".html_safe - if issue.assigned_to.present? - assigned_string = l(:field_assigned_to) + ": " + issue.assigned_to.name - s << view.avatar(issue.assigned_to, - :class => 'gravatar icon-gravatar', - :size => 10, - :title => assigned_string).to_s.html_safe - end - s << view.link_to_issue(issue).html_safe - subject = view.content_tag(:span, s, :class => css_classes).html_safe - html_subject(options, subject, :css => "issue-subject", - :title => issue.subject, :id => "issue-#{issue.id}") + "\n" - when :image - image_subject(options, issue.subject) - when :pdf - pdf_new_page?(options) - pdf_subject(options, issue.subject) - end + subject(issue.subject, options, issue) end def line_for_issue(issue, options) # Skip issues that don't have a due_before (due_date or version's due_date) if issue.is_a?(Issue) && issue.due_before - coords = coordinates(issue.start_date, issue.due_before, issue.done_ratio, options[:zoom]) label = "#{issue.status.name} #{issue.done_ratio}%" - case options[:format] - when :html - html_task(options, coords, - :css => "task " + (issue.leaf? ? 'leaf' : 'parent'), - :label => label, :issue => issue, - :markers => !issue.leaf?) - when :image - image_task(options, coords, :label => label) - when :pdf - pdf_task(options, coords, :label => label) - end - else - '' + markers = !issue.leaf? + line(issue.start_date, issue.due_before, issue.done_ratio, markers, label, options, issue) end end + def subject(label, options, object) + send "#{options[:format]}_subject", options, label, object + end + + def line(start_date, end_date, done_ratio, markers, label, options, object=nil) + options[:zoom] ||= 1 + options[:g_width] ||= (self.date_to - self.date_from + 1) * options[:zoom] + coords = coordinates(start_date, end_date, done_ratio, options[:zoom]) + send "#{options[:format]}_task", options, coords, markers, label, object + end + # Generates a gantt image # Only defined if RMagick is avalaible def to_image(format='PNG') @@ -713,18 +618,79 @@ module Redmine end end - def html_subject(params, subject, options={}) + def html_subject_content(object) + case object + when Issue + issue = object + css_classes = '' + css_classes << ' issue-overdue' if issue.overdue? + css_classes << ' issue-behind-schedule' if issue.behind_schedule? + css_classes << ' icon icon-issue' unless Setting.gravatar_enabled? && issue.assigned_to + css_classes << ' issue-closed' if issue.closed? + if issue.start_date && issue.due_before && issue.done_ratio + progress_date = calc_progress_date(issue.start_date, + issue.due_before, issue.done_ratio) + css_classes << ' behind-start-date' if progress_date < self.date_from + css_classes << ' over-end-date' if progress_date > self.date_to + end + s = "".html_safe + if issue.assigned_to.present? + assigned_string = l(:field_assigned_to) + ": " + issue.assigned_to.name + s << view.avatar(issue.assigned_to, + :class => 'gravatar icon-gravatar', + :size => 10, + :title => assigned_string).to_s.html_safe + end + s << view.link_to_issue(issue).html_safe + view.content_tag(:span, s, :class => css_classes).html_safe + when Version + version = object + html_class = "" + html_class << 'icon icon-package ' + html_class << (version.behind_schedule? ? 'version-behind-schedule' : '') << " " + html_class << (version.overdue? ? 'version-overdue' : '') + html_class << ' version-closed' unless version.open? + if version.start_date && version.due_date && version.completed_percent + progress_date = calc_progress_date(version.start_date, + version.due_date, version.completed_percent) + html_class << ' behind-start-date' if progress_date < self.date_from + html_class << ' over-end-date' if progress_date > self.date_to + end + s = view.link_to_version(version).html_safe + view.content_tag(:span, s, :class => html_class).html_safe + when Project + project = object + html_class = "" + html_class << 'icon icon-projects ' + html_class << (project.overdue? ? 'project-overdue' : '') + s = view.link_to_project(project).html_safe + view.content_tag(:span, s, :class => html_class).html_safe + end + end + + def html_subject(params, subject, object) style = "position: absolute;top:#{params[:top]}px;left:#{params[:indent]}px;" style << "width:#{params[:subject_width] - params[:indent]}px;" if params[:subject_width] - output = view.content_tag(:div, subject, - :class => options[:css], :style => style, - :title => options[:title], - :id => options[:id]) + content = html_subject_content(object) || subject + tag_options = {:style => style} + case object + when Issue + tag_options[:id] = "issue-#{object.id}" + tag_options[:class] = "issue-subject" + tag_options[:title] = object.subject + when Version + tag_options[:id] = "version-#{object.id}" + tag_options[:class] = "version-name" + when Project + tag_options[:class] = "project-name" + end + output = view.content_tag(:div, content, tag_options) @subjects << output output end def pdf_subject(params, subject, options={}) + pdf_new_page?(params) params[:pdf].SetY(params[:top]) params[:pdf].SetX(15) char_limit = PDF::MaxCharactorsForSubject - params[:indent] @@ -754,8 +720,20 @@ module Redmine rels end - def html_task(params, coords, options={}) + def html_task(params, coords, markers, label, object) output = '' + + css = "task " + case object + when Project + "project" + when Version + "version" + when Issue + object.leaf? ? 'leaf' : 'parent' + else + "" + end + # Renders the task bar, with progress and late if coords[:bar_start] && coords[:bar_end] width = coords[:bar_end] - coords[:bar_start] - 2 @@ -763,13 +741,13 @@ module Redmine style << "top:#{params[:top]}px;" style << "left:#{coords[:bar_start]}px;" style << "width:#{width}px;" - html_id = "task-todo-issue-#{options[:issue].id}" if options[:issue] - html_id = "task-todo-version-#{options[:version].id}" if options[:version] + html_id = "task-todo-issue-#{object.id}" if object.is_a?(Issue) + html_id = "task-todo-version-#{object.id}" if object.is_a?(Version) content_opt = {:style => style, - :class => "#{options[:css]} task_todo", + :class => "#{css} task_todo", :id => html_id} - if options[:issue] - rels = issue_relations(options[:issue]) + if object.is_a?(Issue) + rels = issue_relations(object) if rels.present? content_opt[:data] = {"rels" => rels.to_json} end @@ -783,7 +761,7 @@ module Redmine style << "width:#{width}px;" output << view.content_tag(:div, ' '.html_safe, :style => style, - :class => "#{options[:css]} task_late") + :class => "#{css} task_late") end if coords[:bar_progress_end] width = coords[:bar_progress_end] - coords[:bar_start] - 2 @@ -791,16 +769,16 @@ module Redmine style << "top:#{params[:top]}px;" style << "left:#{coords[:bar_start]}px;" style << "width:#{width}px;" - html_id = "task-done-issue-#{options[:issue].id}" if options[:issue] - html_id = "task-done-version-#{options[:version].id}" if options[:version] + html_id = "task-done-issue-#{object.id}" if object.is_a?(Issue) + html_id = "task-done-version-#{object.id}" if object.is_a?(Version) output << view.content_tag(:div, ' '.html_safe, :style => style, - :class => "#{options[:css]} task_done", + :class => "#{css} task_done", :id => html_id) end end # Renders the markers - if options[:markers] + if true if coords[:start] style = "" style << "top:#{params[:top]}px;" @@ -808,7 +786,7 @@ module Redmine style << "width:15px;" output << view.content_tag(:div, ' '.html_safe, :style => style, - :class => "#{options[:css]} marker starting") + :class => "#{css} marker starting") end if coords[:end] style = "" @@ -817,23 +795,23 @@ module Redmine style << "width:15px;" output << view.content_tag(:div, ' '.html_safe, :style => style, - :class => "#{options[:css]} marker ending") + :class => "#{css} marker ending") end end # Renders the label on the right - if options[:label] + if label style = "" style << "top:#{params[:top]}px;" style << "left:#{(coords[:bar_end] || 0) + 8}px;" style << "width:15px;" - output << view.content_tag(:div, options[:label], + output << view.content_tag(:div, label, :style => style, - :class => "#{options[:css]} label") + :class => "#{css} label") end # Renders the tooltip - if options[:issue] && coords[:bar_start] && coords[:bar_end] + if object.is_a?(Issue) && coords[:bar_start] && coords[:bar_end] s = view.content_tag(:span, - view.render_issue_tooltip(options[:issue]).html_safe, + view.render_issue_tooltip(object).html_safe, :class => "tip") style = "" style << "position: absolute;" @@ -849,11 +827,12 @@ module Redmine output end - def pdf_task(params, coords, options={}) + def pdf_task(params, coords, markers, label, object) cell_height_ratio = params[:pdf].get_cell_height_ratio() params[:pdf].set_cell_height_ratio(0.1) - height = options[:height] || 2 + height = 2 + height /= 2 if markers # Renders the task bar, with progress and late if coords[:bar_start] && coords[:bar_end] params[:pdf].SetY(params[:top] + 1.5) @@ -874,7 +853,7 @@ module Redmine end end # Renders the markers - if options[:markers] + if markers if coords[:start] params[:pdf].SetY(params[:top] + 1) params[:pdf].SetX(params[:subject_width] + coords[:start] - 1) @@ -889,16 +868,17 @@ module Redmine end end # Renders the label on the right - if options[:label] + if label params[:pdf].SetX(params[:subject_width] + (coords[:bar_end] || 0) + 5) - params[:pdf].RDMCell(30, 2, options[:label]) + params[:pdf].RDMCell(30, 2, label) end params[:pdf].set_cell_height_ratio(cell_height_ratio) end - def image_task(params, coords, options={}) - height = options[:height] || 6 + def image_task(params, coords, markers, label, object) + height = 6 + height /= 2 if markers # Renders the task bar, with progress and late if coords[:bar_start] && coords[:bar_end] params[:image].fill('#aaa') @@ -922,7 +902,7 @@ module Redmine end end # Renders the markers - if options[:markers] + if markers if coords[:start] x = params[:subject_width] + coords[:start] y = params[:top] - height / 2 @@ -937,11 +917,11 @@ module Redmine end end # Renders the label on the right - if options[:label] + if label params[:image].fill('black') params[:image].text(params[:subject_width] + (coords[:bar_end] || 0) + 5, params[:top] + 1, - options[:label]) + label) end end end |