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 / projects_helper.rb @ 442:753f1380d6bc

History | View | Annotate | Download (9.54 KB)

1 441:cbce1fd3b1b7 Chris
# Redmine - project management software
2
# Copyright (C) 2006-2011  Jean-Philippe Lang
3 0:513646585e45 Chris
#
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 441:cbce1fd3b1b7 Chris
#
9 0:513646585e45 Chris
# 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 441:cbce1fd3b1b7 Chris
#
14 0:513646585e45 Chris
# 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
module ProjectsHelper
19
  def link_to_version(version, options = {})
20
    return '' unless version && version.is_a?(Version)
21
    link_to_if version.visible?, format_version_name(version), { :controller => 'versions', :action => 'show', :id => version }, options
22
  end
23 441:cbce1fd3b1b7 Chris
24 0:513646585e45 Chris
  def project_settings_tabs
25
    tabs = [{:name => 'info', :action => :edit_project, :partial => 'projects/edit', :label => :label_information_plural},
26
            {:name => 'modules', :action => :select_project_modules, :partial => 'projects/settings/modules', :label => :label_module_plural},
27
            {:name => 'members', :action => :manage_members, :partial => 'projects/settings/members', :label => :label_member_plural},
28
            {:name => 'versions', :action => :manage_versions, :partial => 'projects/settings/versions', :label => :label_version_plural},
29
            {:name => 'categories', :action => :manage_categories, :partial => 'projects/settings/issue_categories', :label => :label_issue_category_plural},
30
            {:name => 'wiki', :action => :manage_wiki, :partial => 'projects/settings/wiki', :label => :label_wiki},
31
            {:name => 'repository', :action => :manage_repository, :partial => 'projects/settings/repository', :label => :label_repository},
32
            {:name => 'boards', :action => :manage_boards, :partial => 'projects/settings/boards', :label => :label_board_plural},
33
            {:name => 'activities', :action => :manage_project_activities, :partial => 'projects/settings/activities', :label => :enumeration_activities}
34
            ]
35 441:cbce1fd3b1b7 Chris
    tabs.select {|tab| User.current.allowed_to?(tab[:action], @project)}
36 0:513646585e45 Chris
  end
37 441:cbce1fd3b1b7 Chris
38 0:513646585e45 Chris
  def parent_project_select_tag(project)
39
    selected = project.parent
40
    # retrieve the requested parent project
41
    parent_id = (params[:project] && params[:project][:parent_id]) || params[:parent_id]
42
    if parent_id
43
      selected = (parent_id.blank? ? nil : Project.find(parent_id))
44
    end
45 441:cbce1fd3b1b7 Chris
46 0:513646585e45 Chris
    options = ''
47
    options << "<option value=''></option>" if project.allowed_parents.include?(nil)
48
    options << project_tree_options_for_select(project.allowed_parents.compact, :selected => selected)
49
    content_tag('select', options, :name => 'project[parent_id]', :id => 'project_parent_id')
50
  end
51 441:cbce1fd3b1b7 Chris
52 0:513646585e45 Chris
  # Renders a tree of projects as a nested set of unordered lists
53
  # The given collection may be a subset of the whole project tree
54
  # (eg. some intermediate nodes are private and can not be seen)
55
  def render_project_hierarchy(projects)
56
    s = ''
57
    if projects.any?
58
      ancestors = []
59
      original_project = @project
60
      projects.each do |project|
61
        # set the project environment to please macros.
62
        @project = project
63
        if (ancestors.empty? || project.is_descendant_of?(ancestors.last))
64
          s << "<ul class='projects #{ ancestors.empty? ? 'root' : nil}'>\n"
65
        else
66
          ancestors.pop
67
          s << "</li>"
68 441:cbce1fd3b1b7 Chris
          while (ancestors.any? && !project.is_descendant_of?(ancestors.last))
69 0:513646585e45 Chris
            ancestors.pop
70
            s << "</ul></li>\n"
71
          end
72
        end
73
        classes = (ancestors.empty? ? 'root' : 'child')
74
        s << "<li class='#{classes}'><div class='#{classes}'>" +
75 14:1d32c0a0efbf Chris
               link_to_project(project, {}, :class => "project #{User.current.member_of?(project) ? 'my-project' : nil}")
76 0:513646585e45 Chris
        s << "<div class='wiki description'>#{textilizable(project.short_description, :project => project)}</div>" unless project.description.blank?
77
        s << "</div>\n"
78
        ancestors << project
79
      end
80
      s << ("</li></ul>\n" * ancestors.size)
81
      @project = original_project
82
    end
83
    s
84
  end
85
86 68:60c0a4e08e09 luisf
87
  # Renders a tree of projects where the current user belongs
88
  # as a nested set of unordered lists
89
  # The given collection may be a subset of the whole project tree
90
  # (eg. some intermediate nodes are private and can not be seen)
91
  def render_my_project_hierarchy(projects)
92
    s = ''
93 69:dc22c3eb3c81 luisf
94
    a = ''
95
96
    # Flag to tell if user has any projects
97
    t = FALSE
98 68:60c0a4e08e09 luisf
99
    if projects.any?
100
      ancestors = []
101
      original_project = @project
102
      projects.each do |project|
103
        # set the project environment to please macros.
104
105
        @project = project
106
107
        if User.current.member_of?(project):
108
109 69:dc22c3eb3c81 luisf
          t = TRUE
110
111 68:60c0a4e08e09 luisf
          if (ancestors.empty? || project.is_descendant_of?(ancestors.last))
112
            s << "<ul class='projects #{ ancestors.empty? ? 'root' : nil}'>\n"
113
          else
114
            ancestors.pop
115
            s << "</li>"
116
            while (ancestors.any? && !project.is_descendant_of?(ancestors.last))
117
              ancestors.pop
118
              s << "</ul></li>\n"
119
            end
120
          end
121
122
          classes = (ancestors.empty? ? 'root' : 'child')
123
          s << "<li class='#{classes}'><div class='#{classes}'>" +
124 125:ca48d8b6f55d chris
                 link_to_project(project, {}, :class => "project my-project")
125
          if project.is_public?
126
            s << " <span class='public'>" << l("field_is_public") << "</span>"
127
          else
128
            s << " <span class='private'>" << l("field_is_private") << "</span>"
129
          end
130 68:60c0a4e08e09 luisf
          s << "<div class='wiki description'>#{textilizable(project.short_description, :project => project)}</div>" unless project.description.blank?
131
          s << "</div>\n"
132
          ancestors << project
133
        end
134
       end
135
        s << ("</li></ul>\n" * ancestors.size)
136
        @project = original_project
137
    end
138 69:dc22c3eb3c81 luisf
139
    if t == TRUE
140
      a << "<h2>"
141
      a <<  l("label_my_project_plural")
142
      a << "</h2>"
143
      a << s
144
    else
145
      a = s
146
    end
147
148
    a
149 68:60c0a4e08e09 luisf
  end
150
151 124:bc91f2025d05 chris
  # Renders a tree of projects that the current user does not belong
152
  # to, or of all projects if the current user is not logged in.  The
153
  # given collection may be a subset of the whole project tree
154
  # (eg. some intermediate nodes are private and can not be seen).  We
155
  # are potentially interested in various things: the project name,
156
  # description, manager(s), creation date, last activity date,
157
  # general activity level, whether there is anything actually hosted
158
  # here for the project, etc.
159
  def render_project_table(projects)
160 68:60c0a4e08e09 luisf
161 124:bc91f2025d05 chris
    s = ""
162
    s << "<div class='autoscroll'>"
163
    s << "<table class='list projects'>"
164
    s << "<thead><tr>"
165
166 205:05f9a2a9c753 chris
    s << sort_header_tag('name', :caption => l("field_name"))
167 124:bc91f2025d05 chris
    s << "<th class='managers'>" << l("label_managers") << "</th>"
168
    s << sort_header_tag('created_on', :default_order => 'desc')
169
    s << sort_header_tag('updated_on', :default_order => 'desc')
170 100:1412841d48a3 Chris
171 124:bc91f2025d05 chris
    s << "</tr></thead><tbody>"
172 68:60c0a4e08e09 luisf
173 124:bc91f2025d05 chris
    original_project = @project
174 68:60c0a4e08e09 luisf
175 124:bc91f2025d05 chris
    projects.each do |project|
176 205:05f9a2a9c753 chris
      s << render_project_in_table(project, cycle('odd', 'even'), 0)
177 68:60c0a4e08e09 luisf
    end
178 69:dc22c3eb3c81 luisf
179 124:bc91f2025d05 chris
    s << "</table>"
180 69:dc22c3eb3c81 luisf
181 124:bc91f2025d05 chris
    @project = original_project
182 248:3bcfbf971c40 Chris
183 124:bc91f2025d05 chris
    s
184 68:60c0a4e08e09 luisf
  end
185
186
187 205:05f9a2a9c753 chris
  def render_project_in_table(project, oddeven, level)
188
189
    # set the project environment to please macros.
190
    @project = project
191
192
    classes = (level == 0 ? 'root' : 'child')
193
194
    s = ""
195
196
    s << "<tr class='#{oddeven} #{classes} level#{level}'>"
197
    s << "<td class='firstcol' align=top><div class='name hosted_here"
198
    s << " no_description" if project.description.blank?
199
    s << "'>" << link_to_project(project, {}, :class => "project #{User.current.member_of?(project) ? 'my-project' : nil}");
200
    s << "</div>"
201
    unless project.description.blank?
202
      s << "<div class='wiki description'>"
203
      s << textilizable(project.short_description, :project => project)
204
      s << "</div>"
205
    end
206
207
    s << "<td class='managers' align=top>"
208
209
    u = project.users_by_role
210
    if u
211
      u.keys.each do |r|
212
        if r.allowed_to?(:edit_project)
213
          mgrs = []
214
          u[r].sort.each do |m|
215
            mgrs << link_to_user(m)
216
          end
217
          if mgrs.size < 3
218
            s << '<nobr>' << mgrs.join(', ') << '</nobr>'
219
          else
220
            s << mgrs.join(', ')
221
          end
222
        end
223
      end
224
    end
225
226
    s << "</td>"
227
    s << "<td class='created_on' align=top>" << format_date(project.created_on) << "</td>"
228
    s << "<td class='updated_on' align=top>" << format_date(project.updated_on) << "</td>"
229
230
    s << "</tr>"
231
232
    project.children.each do |child|
233
      s << render_project_in_table(child, oddeven, level + 1)
234
    end
235
236
    s
237
  end
238
239 68:60c0a4e08e09 luisf
240 0:513646585e45 Chris
  # Returns a set of options for a select field, grouped by project.
241
  def version_options_for_select(versions, selected=nil)
242
    grouped = Hash.new {|h,k| h[k] = []}
243
    versions.each do |version|
244
      grouped[version.project.name] << [version.name, version.id]
245
    end
246
    # Add in the selected
247
    if selected && !versions.include?(selected)
248
      grouped[selected.project.name] << [selected.name, selected.id]
249
    end
250 441:cbce1fd3b1b7 Chris
251 0:513646585e45 Chris
    if grouped.keys.size > 1
252
      grouped_options_for_select(grouped, selected && selected.id)
253
    else
254
      options_for_select((grouped.values.first || []), selected && selected.id)
255
    end
256
  end
257
258
  def format_version_sharing(sharing)
259
    sharing = 'none' unless Version::VERSION_SHARINGS.include?(sharing)
260
    l("label_version_sharing_#{sharing}")
261
  end
262
end