annotate app/helpers/projects_helper.rb @ 133:1890a968b404 cannam-pre-20110113-merge

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