To check out this repository please hg clone the following URL, or open the URL using EasyMercurial or your preferred Mercurial client.

Statistics Download as Zip
| Branch: | Tag: | Revision:

root / app / helpers / issues_helper.rb @ 1535:e2c122809c5c

History | View | Annotate | Download (15.1 KB)

1
# encoding: utf-8
2
#
3
# Redmine - project management software
4
# Copyright (C) 2006-2014  Jean-Philippe Lang
5
#
6
# This program is free software; you can redistribute it and/or
7
# modify it under the terms of the GNU General Public License
8
# as published by the Free Software Foundation; either version 2
9
# of the License, or (at your option) any later version.
10
#
11
# This program is distributed in the hope that it will be useful,
12
# but WITHOUT ANY WARRANTY; without even the implied warranty of
13
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14
# GNU General Public License for more details.
15
#
16
# You should have received a copy of the GNU General Public License
17
# along with this program; if not, write to the Free Software
18
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.
19

    
20
module IssuesHelper
21
  include ApplicationHelper
22

    
23
  def issue_list(issues, &block)
24
    ancestors = []
25
    issues.each do |issue|
26
      while (ancestors.any? && !issue.is_descendant_of?(ancestors.last))
27
        ancestors.pop
28
      end
29
      yield issue, ancestors.size
30
      ancestors << issue unless issue.leaf?
31
    end
32
  end
33

    
34
  # Renders a HTML/CSS tooltip
35
  #
36
  # To use, a trigger div is needed.  This is a div with the class of "tooltip"
37
  # that contains this method wrapped in a span with the class of "tip"
38
  #
39
  #    <div class="tooltip"><%= link_to_issue(issue) %>
40
  #      <span class="tip"><%= render_issue_tooltip(issue) %></span>
41
  #    </div>
42
  #
43
  def render_issue_tooltip(issue)
44
    @cached_label_status ||= l(:field_status)
45
    @cached_label_start_date ||= l(:field_start_date)
46
    @cached_label_due_date ||= l(:field_due_date)
47
    @cached_label_assigned_to ||= l(:field_assigned_to)
48
    @cached_label_priority ||= l(:field_priority)
49
    @cached_label_project ||= l(:field_project)
50

    
51
    link_to_issue(issue) + "<br /><br />".html_safe +
52
      "<strong>#{@cached_label_project}</strong>: #{link_to_project(issue.project)}<br />".html_safe +
53
      "<strong>#{@cached_label_status}</strong>: #{h(issue.status.name)}<br />".html_safe +
54
      "<strong>#{@cached_label_start_date}</strong>: #{format_date(issue.start_date)}<br />".html_safe +
55
      "<strong>#{@cached_label_due_date}</strong>: #{format_date(issue.due_date)}<br />".html_safe +
56
      "<strong>#{@cached_label_assigned_to}</strong>: #{h(issue.assigned_to)}<br />".html_safe +
57
      "<strong>#{@cached_label_priority}</strong>: #{h(issue.priority.name)}".html_safe
58
  end
59

    
60
  def issue_heading(issue)
61
    h("#{issue.tracker} ##{issue.id}")
62
  end
63

    
64
  def render_issue_subject_with_tree(issue)
65
    s = ''
66
    ancestors = issue.root? ? [] : issue.ancestors.visible.all
67
    ancestors.each do |ancestor|
68
      s << '<div>' + content_tag('p', link_to_issue(ancestor, :project => (issue.project_id != ancestor.project_id)))
69
    end
70
    s << '<div>'
71
    subject = h(issue.subject)
72
    if issue.is_private?
73
      subject = content_tag('span', l(:field_is_private), :class => 'private') + ' ' + subject
74
    end
75
    s << content_tag('h3', subject)
76
    s << '</div>' * (ancestors.size + 1)
77
    s.html_safe
78
  end
79

    
80
  def render_descendants_tree(issue)
81
    s = '<form><table class="list issues">'
82
    issue_list(issue.descendants.visible.sort_by(&:lft)) do |child, level|
83
      css = "issue issue-#{child.id} hascontextmenu"
84
      css << " idnt idnt-#{level}" if level > 0
85
      s << content_tag('tr',
86
             content_tag('td', check_box_tag("ids[]", child.id, false, :id => nil), :class => 'checkbox') +
87
             content_tag('td', link_to_issue(child, :truncate => 60, :project => (issue.project_id != child.project_id)), :class => 'subject') +
88
             content_tag('td', h(child.status)) +
89
             content_tag('td', link_to_user(child.assigned_to)) +
90
             content_tag('td', progress_bar(child.done_ratio, :width => '80px')),
91
             :class => css)
92
    end
93
    s << '</table></form>'
94
    s.html_safe
95
  end
96

    
97
  # Returns an array of error messages for bulk edited issues
98
  def bulk_edit_error_messages(issues)
99
    messages = {}
100
    issues.each do |issue|
101
      issue.errors.full_messages.each do |message|
102
        messages[message] ||= []
103
        messages[message] << issue
104
      end
105
    end
106
    messages.map { |message, issues|
107
      "#{message}: " + issues.map {|i| "##{i.id}"}.join(', ')
108
    }
109
 end
110

    
111
  # Returns a link for adding a new subtask to the given issue
112
  def link_to_new_subtask(issue)
113
    attrs = {
114
      :tracker_id => issue.tracker,
115
      :parent_issue_id => issue
116
    }
117
    link_to(l(:button_add), new_project_issue_path(issue.project, :issue => attrs))
118
  end
119

    
120
  class IssueFieldsRows
121
    include ActionView::Helpers::TagHelper
122

    
123
    def initialize
124
      @left = []
125
      @right = []
126
    end
127

    
128
    def left(*args)
129
      args.any? ? @left << cells(*args) : @left
130
    end
131

    
132
    def right(*args)
133
      args.any? ? @right << cells(*args) : @right
134
    end
135

    
136
    def size
137
      @left.size > @right.size ? @left.size : @right.size
138
    end
139

    
140
    def to_html
141
      html = ''.html_safe
142
      blank = content_tag('th', '') + content_tag('td', '')
143
      size.times do |i|
144
        left = @left[i] || blank
145
        right = @right[i] || blank
146
        html << content_tag('tr', left + right)
147
      end
148
      html
149
    end
150

    
151
    def cells(label, text, options={})
152
      content_tag('th', "#{label}:", options) + content_tag('td', text, options)
153
    end
154
  end
155

    
156
  def issue_fields_rows
157
    r = IssueFieldsRows.new
158
    yield r
159
    r.to_html
160
  end
161

    
162
  def render_custom_fields_rows(issue)
163
    values = issue.visible_custom_field_values
164
    return if values.empty?
165
    ordered_values = []
166
    half = (values.size / 2.0).ceil
167
    half.times do |i|
168
      ordered_values << values[i]
169
      ordered_values << values[i + half]
170
    end
171
    s = "<tr>\n"
172
    n = 0
173
    ordered_values.compact.each do |value|
174
      css = "cf_#{value.custom_field.id}"
175
      s << "</tr>\n<tr>\n" if n > 0 && (n % 2) == 0
176
      s << "\t<th class=\"#{css}\">#{ h(value.custom_field.name) }:</th><td class=\"#{css}\">#{ h(show_value(value)) }</td>\n"
177
      n += 1
178
    end
179
    s << "</tr>\n"
180
    s.html_safe
181
  end
182

    
183
  def issues_destroy_confirmation_message(issues)
184
    issues = [issues] unless issues.is_a?(Array)
185
    message = l(:text_issues_destroy_confirmation)
186
    descendant_count = issues.inject(0) {|memo, i| memo += (i.right - i.left - 1)/2}
187
    if descendant_count > 0
188
      issues.each do |issue|
189
        next if issue.root?
190
        issues.each do |other_issue|
191
          descendant_count -= 1 if issue.is_descendant_of?(other_issue)
192
        end
193
      end
194
      if descendant_count > 0
195
        message << "\n" + l(:text_issues_destroy_descendants_confirmation, :count => descendant_count)
196
      end
197
    end
198
    message
199
  end
200

    
201
  def sidebar_queries
202
    unless @sidebar_queries
203
      @sidebar_queries = IssueQuery.visible.
204
        order("#{Query.table_name}.name ASC").
205
        # Project specific queries and global queries
206
        where(@project.nil? ? ["project_id IS NULL"] : ["project_id IS NULL OR project_id = ?", @project.id]).
207
        all
208
    end
209
    @sidebar_queries
210
  end
211

    
212
  def query_links(title, queries)
213
    return '' if queries.empty?
214
    # links to #index on issues/show
215
    url_params = controller_name == 'issues' ? {:controller => 'issues', :action => 'index', :project_id => @project} : params
216

    
217
    content_tag('h3', title) + "\n" +
218
      content_tag('ul',
219
        queries.collect {|query|
220
            css = 'query'
221
            css << ' selected' if query == @query
222
            content_tag('li', link_to(query.name, url_params.merge(:query_id => query), :class => css))
223
          }.join("\n").html_safe,
224
        :class => 'queries'
225
      ) + "\n"
226
  end
227

    
228
  def render_sidebar_queries
229
    out = ''.html_safe
230
    out << query_links(l(:label_my_queries), sidebar_queries.select(&:is_private?))
231
    out << query_links(l(:label_query_plural), sidebar_queries.reject(&:is_private?))
232
    out
233
  end
234

    
235
  def email_issue_attributes(issue, user)
236
    items = []
237
    %w(author status priority assigned_to category fixed_version).each do |attribute|
238
      unless issue.disabled_core_fields.include?(attribute+"_id")
239
        items << "#{l("field_#{attribute}")}: #{issue.send attribute}"
240
      end
241
    end
242
    issue.visible_custom_field_values(user).each do |value|
243
      items << "#{value.custom_field.name}: #{show_value(value, false)}"
244
    end
245
    items
246
  end
247

    
248
  def render_email_issue_attributes(issue, user, html=false)
249
    items = email_issue_attributes(issue, user)
250
    if html
251
      content_tag('ul', items.map{|s| content_tag('li', s)}.join("\n").html_safe)
252
    else
253
      items.map{|s| "* #{s}"}.join("\n")
254
    end
255
  end
256

    
257
  # Returns the textual representation of a journal details
258
  # as an array of strings
259
  def details_to_strings(details, no_html=false, options={})
260
    options[:only_path] = (options[:only_path] == false ? false : true)
261
    strings = []
262
    values_by_field = {}
263
    details.each do |detail|
264
      if detail.property == 'cf'
265
        field = detail.custom_field
266
        if field && field.multiple?
267
          values_by_field[field] ||= {:added => [], :deleted => []}
268
          if detail.old_value
269
            values_by_field[field][:deleted] << detail.old_value
270
          end
271
          if detail.value
272
            values_by_field[field][:added] << detail.value
273
          end
274
          next
275
        end
276
      end
277
      strings << show_detail(detail, no_html, options)
278
    end
279
    values_by_field.each do |field, changes|
280
      detail = JournalDetail.new(:property => 'cf', :prop_key => field.id.to_s)
281
      detail.instance_variable_set "@custom_field", field
282
      if changes[:added].any?
283
        detail.value = changes[:added]
284
        strings << show_detail(detail, no_html, options)
285
      elsif changes[:deleted].any?
286
        detail.old_value = changes[:deleted]
287
        strings << show_detail(detail, no_html, options)
288
      end
289
    end
290
    strings
291
  end
292

    
293
  # Returns the textual representation of a single journal detail
294
  def show_detail(detail, no_html=false, options={})
295
    multiple = false
296
    case detail.property
297
    when 'attr'
298
      field = detail.prop_key.to_s.gsub(/\_id$/, "")
299
      label = l(("field_" + field).to_sym)
300
      case detail.prop_key
301
      when 'due_date', 'start_date'
302
        value = format_date(detail.value.to_date) if detail.value
303
        old_value = format_date(detail.old_value.to_date) if detail.old_value
304

    
305
      when 'project_id', 'status_id', 'tracker_id', 'assigned_to_id',
306
            'priority_id', 'category_id', 'fixed_version_id'
307
        value = find_name_by_reflection(field, detail.value)
308
        old_value = find_name_by_reflection(field, detail.old_value)
309

    
310
      when 'estimated_hours'
311
        value = "%0.02f" % detail.value.to_f unless detail.value.blank?
312
        old_value = "%0.02f" % detail.old_value.to_f unless detail.old_value.blank?
313

    
314
      when 'parent_id'
315
        label = l(:field_parent_issue)
316
        value = "##{detail.value}" unless detail.value.blank?
317
        old_value = "##{detail.old_value}" unless detail.old_value.blank?
318

    
319
      when 'is_private'
320
        value = l(detail.value == "0" ? :general_text_No : :general_text_Yes) unless detail.value.blank?
321
        old_value = l(detail.old_value == "0" ? :general_text_No : :general_text_Yes) unless detail.old_value.blank?
322
      end
323
    when 'cf'
324
      custom_field = detail.custom_field
325
      if custom_field
326
        multiple = custom_field.multiple?
327
        label = custom_field.name
328
        value = format_value(detail.value, custom_field) if detail.value
329
        old_value = format_value(detail.old_value, custom_field) if detail.old_value
330
      end
331
    when 'attachment'
332
      label = l(:label_attachment)
333
    when 'relation'
334
      if detail.value && !detail.old_value
335
        rel_issue = Issue.visible.find_by_id(detail.value)
336
        value = rel_issue.nil? ? "#{l(:label_issue)} ##{detail.value}" :
337
                  (no_html ? rel_issue : link_to_issue(rel_issue, :only_path => options[:only_path]))
338
      elsif detail.old_value && !detail.value
339
        rel_issue = Issue.visible.find_by_id(detail.old_value)
340
        old_value = rel_issue.nil? ? "#{l(:label_issue)} ##{detail.old_value}" :
341
                          (no_html ? rel_issue : link_to_issue(rel_issue, :only_path => options[:only_path]))
342
      end
343
      relation_type = IssueRelation::TYPES[detail.prop_key]
344
      label = l(relation_type[:name]) if relation_type
345
    end
346
    call_hook(:helper_issues_show_detail_after_setting,
347
              {:detail => detail, :label => label, :value => value, :old_value => old_value })
348

    
349
    label ||= detail.prop_key
350
    value ||= detail.value
351
    old_value ||= detail.old_value
352

    
353
    unless no_html
354
      label = content_tag('strong', label)
355
      old_value = content_tag("i", h(old_value)) if detail.old_value
356
      if detail.old_value && detail.value.blank? && detail.property != 'relation'
357
        old_value = content_tag("del", old_value)
358
      end
359
      if detail.property == 'attachment' && !value.blank? && atta = Attachment.find_by_id(detail.prop_key)
360
        # Link to the attachment if it has not been removed
361
        value = link_to_attachment(atta, :download => true, :only_path => options[:only_path])
362
        if options[:only_path] != false && atta.is_text?
363
          value += link_to(
364
                       image_tag('magnifier.png'),
365
                       :controller => 'attachments', :action => 'show',
366
                       :id => atta, :filename => atta.filename
367
                     )
368
        end
369
      else
370
        value = content_tag("i", h(value)) if value
371
      end
372
    end
373

    
374
    if detail.property == 'attr' && detail.prop_key == 'description'
375
      s = l(:text_journal_changed_no_detail, :label => label)
376
      unless no_html
377
        diff_link = link_to 'diff',
378
          {:controller => 'journals', :action => 'diff', :id => detail.journal_id,
379
           :detail_id => detail.id, :only_path => options[:only_path]},
380
          :title => l(:label_view_diff)
381
        s << " (#{ diff_link })"
382
      end
383
      s.html_safe
384
    elsif detail.value.present?
385
      case detail.property
386
      when 'attr', 'cf'
387
        if detail.old_value.present?
388
          l(:text_journal_changed, :label => label, :old => old_value, :new => value).html_safe
389
        elsif multiple
390
          l(:text_journal_added, :label => label, :value => value).html_safe
391
        else
392
          l(:text_journal_set_to, :label => label, :value => value).html_safe
393
        end
394
      when 'attachment', 'relation'
395
        l(:text_journal_added, :label => label, :value => value).html_safe
396
      end
397
    else
398
      l(:text_journal_deleted, :label => label, :old => old_value).html_safe
399
    end
400
  end
401

    
402
  # Find the name of an associated record stored in the field attribute
403
  def find_name_by_reflection(field, id)
404
    unless id.present?
405
      return nil
406
    end
407
    association = Issue.reflect_on_association(field.to_sym)
408
    if association
409
      record = association.class_name.constantize.find_by_id(id)
410
      if record
411
        record.name.force_encoding('UTF-8') if record.name.respond_to?(:force_encoding)
412
        return record.name
413
      end
414
    end
415
  end
416

    
417
  # Renders issue children recursively
418
  def render_api_issue_children(issue, api)
419
    return if issue.leaf?
420
    api.array :children do
421
      issue.children.each do |child|
422
        api.issue(:id => child.id) do
423
          api.tracker(:id => child.tracker_id, :name => child.tracker.name) unless child.tracker.nil?
424
          api.subject child.subject
425
          render_api_issue_children(child, api)
426
        end
427
      end
428
    end
429
  end
430
end