##// END OF EJS Templates
Improved Redmine links:...
Jean-Philippe Lang -
r703:fdf842a4c458
parent child
Show More
@@ -1,292 +1,308
1 1 # redMine - project management software
2 2 # Copyright (C) 2006-2007 Jean-Philippe Lang
3 3 #
4 4 # This program is free software; you can redistribute it and/or
5 5 # modify it under the terms of the GNU General Public License
6 6 # as published by the Free Software Foundation; either version 2
7 7 # of the License, or (at your option) any later version.
8 8 #
9 9 # This program is distributed in the hope that it will be useful,
10 10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
11 11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 12 # GNU General Public License for more details.
13 13 #
14 14 # You should have received a copy of the GNU General Public License
15 15 # along with this program; if not, write to the Free Software
16 16 # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
17 17
18 18 module ApplicationHelper
19 19
20 20 def current_role
21 21 @current_role ||= User.current.role_for_project(@project)
22 22 end
23 23
24 24 # Return true if user is authorized for controller/action, otherwise false
25 25 def authorize_for(controller, action)
26 26 User.current.allowed_to?({:controller => controller, :action => action}, @project)
27 27 end
28 28
29 29 # Display a link if user is authorized
30 30 def link_to_if_authorized(name, options = {}, html_options = nil, *parameters_for_method_reference)
31 31 link_to(name, options, html_options, *parameters_for_method_reference) if authorize_for(options[:controller] || params[:controller], options[:action])
32 32 end
33 33
34 34 # Display a link to user's account page
35 35 def link_to_user(user)
36 36 link_to user.name, :controller => 'account', :action => 'show', :id => user
37 37 end
38 38
39 39 def link_to_issue(issue)
40 40 link_to "#{issue.tracker.name} ##{issue.id}", :controller => "issues", :action => "show", :id => issue
41 41 end
42 42
43 43 def toggle_link(name, id, options={})
44 44 onclick = "Element.toggle('#{id}'); "
45 45 onclick << (options[:focus] ? "Form.Element.focus('#{options[:focus]}'); " : "this.blur(); ")
46 46 onclick << "return false;"
47 47 link_to(name, "#", :onclick => onclick)
48 48 end
49 49
50 50 def image_to_function(name, function, html_options = {})
51 51 html_options.symbolize_keys!
52 52 tag(:input, html_options.merge({
53 53 :type => "image", :src => image_path(name),
54 54 :onclick => (html_options[:onclick] ? "#{html_options[:onclick]}; " : "") + "#{function};"
55 55 }))
56 56 end
57 57
58 58 def prompt_to_remote(name, text, param, url, html_options = {})
59 59 html_options[:onclick] = "promptToRemote('#{text}', '#{param}', '#{url_for(url)}'); return false;"
60 60 link_to name, {}, html_options
61 61 end
62 62
63 63 def format_date(date)
64 64 return nil unless date
65 65 @date_format_setting ||= Setting.date_format.to_i
66 66 @date_format_setting == 0 ? l_date(date) : date.strftime("%Y-%m-%d")
67 67 end
68 68
69 69 def format_time(time)
70 70 return nil unless time
71 71 @date_format_setting ||= Setting.date_format.to_i
72 72 time = time.to_time if time.is_a?(String)
73 73 @date_format_setting == 0 ? l_datetime(time) : (time.strftime("%Y-%m-%d") + ' ' + l_time(time))
74 74 end
75 75
76 76 def day_name(day)
77 77 l(:general_day_names).split(',')[day-1]
78 78 end
79 79
80 80 def month_name(month)
81 81 l(:actionview_datehelper_select_month_names).split(',')[month-1]
82 82 end
83 83
84 84 def pagination_links_full(paginator, options={}, html_options={})
85 85 page_param = options.delete(:page_param) || :page
86 86
87 87 html = ''
88 88 html << link_to_remote(('&#171; ' + l(:label_previous)),
89 89 {:update => "content", :url => options.merge(page_param => paginator.current.previous)},
90 90 {:href => url_for(:params => options.merge(page_param => paginator.current.previous))}) + ' ' if paginator.current.previous
91 91
92 92 html << (pagination_links_each(paginator, options) do |n|
93 93 link_to_remote(n.to_s,
94 94 {:url => {:params => options.merge(page_param => n)}, :update => 'content'},
95 95 {:href => url_for(:params => options.merge(page_param => n))})
96 96 end || '')
97 97
98 98 html << ' ' + link_to_remote((l(:label_next) + ' &#187;'),
99 99 {:update => "content", :url => options.merge(page_param => paginator.current.next)},
100 100 {:href => url_for(:params => options.merge(page_param => paginator.current.next))}) if paginator.current.next
101 101 html
102 102 end
103 103
104 104 # format text according to system settings
105 105 def textilizable(text, options = {})
106 106 return "" if text.blank?
107 107
108 108 # when using an image link, try to use an attachment, if possible
109 109 attachments = options[:attachments]
110 110 if attachments
111 111 text = text.gsub(/!([<>=]*)(\S+\.(gif|jpg|jpeg|png))!/) do |m|
112 112 align = $1
113 113 filename = $2
114 114 rf = Regexp.new(filename, Regexp::IGNORECASE)
115 115 # search for the picture in attachments
116 116 if found = attachments.detect { |att| att.filename =~ rf }
117 117 image_url = url_for :controller => 'attachments', :action => 'download', :id => found.id
118 118 "!#{align}#{image_url}!"
119 119 else
120 120 "!#{align}#{filename}!"
121 121 end
122 122 end
123 123 end
124 124
125 125 text = (Setting.text_formatting == 'textile') ?
126 126 Redmine::WikiFormatting.to_html(text) : simple_format(auto_link(h(text)))
127 127
128 128 # different methods for formatting wiki links
129 129 case options[:wiki_links]
130 130 when :local
131 131 # used for local links to html files
132 132 format_wiki_link = Proc.new {|project, title| "#{title}.html" }
133 133 when :anchor
134 134 # used for single-file wiki export
135 135 format_wiki_link = Proc.new {|project, title| "##{title}" }
136 136 else
137 137 format_wiki_link = Proc.new {|project, title| url_for :controller => 'wiki', :action => 'index', :id => project, :page => title }
138 138 end
139 139
140 140 project = options[:project] || @project
141 141
142 142 # turn wiki links into html links
143 143 # example:
144 144 # [[mypage]]
145 145 # [[mypage|mytext]]
146 146 # wiki links can refer other project wikis, using project name or identifier:
147 147 # [[project:]] -> wiki starting page
148 148 # [[project:|mytext]]
149 149 # [[project:mypage]]
150 150 # [[project:mypage|mytext]]
151 151 text = text.gsub(/\[\[([^\]\|]+)(\|([^\]\|]+))?\]\]/) do |m|
152 152 link_project = project
153 153 page = $1
154 154 title = $3
155 155 if page =~ /^([^\:]+)\:(.*)$/
156 156 link_project = Project.find_by_name($1) || Project.find_by_identifier($1)
157 157 page = title || $2
158 158 title = $1 if page.blank?
159 159 end
160 link_to((title || page), format_wiki_link.call(link_project, Wiki.titleize(page)), :class => 'wiki-page')
160
161 if link_project && link_project.wiki
162 # check if page exists
163 wiki_page = link_project.wiki.find_page(page)
164 link_to((title || page), format_wiki_link.call(link_project, Wiki.titleize(page)),
165 :class => ('wiki-page' + (wiki_page ? '' : ' new')))
166 else
167 # project or wiki doesn't exist
168 title || page
169 end
161 170 end
162 171
163 172 # turn issue and revision ids into links
164 173 # example:
165 174 # #52 -> <a href="/issues/show/52">#52</a>
166 175 # r52 -> <a href="/repositories/revision/6?rev=52">r52</a> (project.id is 6)
167 176 text = text.gsub(%r{([\s,-^])(#|r)(\d+)(?=[[:punct:]]|\s|<|$)}) do |m|
168 177 leading, otype, oid = $1, $2, $3
169 178 link = nil
170 179 if otype == 'r'
171 link = link_to("r#{oid}", {:controller => 'repositories', :action => 'revision', :id => project.id, :rev => oid}, :class => 'changeset') if project
180 if project && (changeset = project.changesets.find_by_revision(oid))
181 link = link_to("r#{oid}", {:controller => 'repositories', :action => 'revision', :id => project.id, :rev => oid}, :class => 'changeset',
182 :title => truncate(changeset.comments, 100))
183 end
172 184 else
173 link = link_to("##{oid}", {:controller => 'issues', :action => 'show', :id => oid}, :class => 'issue')
185 if issue = Issue.find_by_id(oid.to_i, :include => [:project, :status], :conditions => Project.visible_by(User.current))
186 link = link_to("##{oid}", {:controller => 'issues', :action => 'show', :id => oid}, :class => 'issue',
187 :title => "#{truncate(issue.subject, 100)} (#{issue.status.name})")
188 link = content_tag('del', link) if issue.closed?
189 end
174 190 end
175 191 leading + (link || "#{otype}#{oid}")
176 192 end
177 193
178 194 text
179 195 end
180 196
181 197 # Same as Rails' simple_format helper without using paragraphs
182 198 def simple_format_without_paragraph(text)
183 199 text.to_s.
184 200 gsub(/\r\n?/, "\n"). # \r\n and \r -> \n
185 201 gsub(/\n\n+/, "<br /><br />"). # 2+ newline -> 2 br
186 202 gsub(/([^\n]\n)(?=[^\n])/, '\1<br />') # 1 newline -> br
187 203 end
188 204
189 205 def error_messages_for(object_name, options = {})
190 206 options = options.symbolize_keys
191 207 object = instance_variable_get("@#{object_name}")
192 208 if object && !object.errors.empty?
193 209 # build full_messages here with controller current language
194 210 full_messages = []
195 211 object.errors.each do |attr, msg|
196 212 next if msg.nil?
197 213 msg = msg.first if msg.is_a? Array
198 214 if attr == "base"
199 215 full_messages << l(msg)
200 216 else
201 217 full_messages << "&#171; " + (l_has_string?("field_" + attr) ? l("field_" + attr) : object.class.human_attribute_name(attr)) + " &#187; " + l(msg) unless attr == "custom_values"
202 218 end
203 219 end
204 220 # retrieve custom values error messages
205 221 if object.errors[:custom_values]
206 222 object.custom_values.each do |v|
207 223 v.errors.each do |attr, msg|
208 224 next if msg.nil?
209 225 msg = msg.first if msg.is_a? Array
210 226 full_messages << "&#171; " + v.custom_field.name + " &#187; " + l(msg)
211 227 end
212 228 end
213 229 end
214 230 content_tag("div",
215 231 content_tag(
216 232 options[:header_tag] || "span", lwr(:gui_validation_error, full_messages.length) + ":"
217 233 ) +
218 234 content_tag("ul", full_messages.collect { |msg| content_tag("li", msg) }),
219 235 "id" => options[:id] || "errorExplanation", "class" => options[:class] || "errorExplanation"
220 236 )
221 237 else
222 238 ""
223 239 end
224 240 end
225 241
226 242 def lang_options_for_select(blank=true)
227 243 (blank ? [["(auto)", ""]] : []) +
228 244 GLoc.valid_languages.collect{|lang| [ ll(lang.to_s, :general_lang_name), lang.to_s]}.sort{|x,y| x.first <=> y.first }
229 245 end
230 246
231 247 def label_tag_for(name, option_tags = nil, options = {})
232 248 label_text = l(("field_"+field.to_s.gsub(/\_id$/, "")).to_sym) + (options.delete(:required) ? @template.content_tag("span", " *", :class => "required"): "")
233 249 content_tag("label", label_text)
234 250 end
235 251
236 252 def labelled_tabular_form_for(name, object, options, &proc)
237 253 options[:html] ||= {}
238 254 options[:html].store :class, "tabular"
239 255 form_for(name, object, options.merge({ :builder => TabularFormBuilder, :lang => current_language}), &proc)
240 256 end
241 257
242 258 def check_all_links(form_name)
243 259 link_to_function(l(:button_check_all), "checkAll('#{form_name}', true)") +
244 260 " | " +
245 261 link_to_function(l(:button_uncheck_all), "checkAll('#{form_name}', false)")
246 262 end
247 263
248 264 def calendar_for(field_id)
249 265 image_tag("calendar.png", {:id => "#{field_id}_trigger",:class => "calendar-trigger"}) +
250 266 javascript_tag("Calendar.setup({inputField : '#{field_id}', ifFormat : '%Y-%m-%d', button : '#{field_id}_trigger' });")
251 267 end
252 268
253 269 def wikitoolbar_for(field_id)
254 270 return '' unless Setting.text_formatting == 'textile'
255 271 javascript_include_tag('jstoolbar') + javascript_tag("var toolbar = new jsToolBar($('#{field_id}')); toolbar.draw();")
256 272 end
257 273 end
258 274
259 275 class TabularFormBuilder < ActionView::Helpers::FormBuilder
260 276 include GLoc
261 277
262 278 def initialize(object_name, object, template, options, proc)
263 279 set_language_if_valid options.delete(:lang)
264 280 @object_name, @object, @template, @options, @proc = object_name, object, template, options, proc
265 281 end
266 282
267 283 (field_helpers - %w(radio_button hidden_field) + %w(date_select)).each do |selector|
268 284 src = <<-END_SRC
269 285 def #{selector}(field, options = {})
270 286 return super if options.delete :no_label
271 287 label_text = l(options[:label]) if options[:label]
272 288 label_text ||= l(("field_"+field.to_s.gsub(/\_id$/, "")).to_sym)
273 289 label_text << @template.content_tag("span", " *", :class => "required") if options.delete(:required)
274 290 label = @template.content_tag("label", label_text,
275 291 :class => (@object && @object.errors[field] ? "error" : nil),
276 292 :for => (@object_name.to_s + "_" + field.to_s))
277 293 label + super
278 294 end
279 295 END_SRC
280 296 class_eval src, __FILE__, __LINE__
281 297 end
282 298
283 299 def select(field, choices, options = {}, html_options = {})
284 300 label_text = l(("field_"+field.to_s.gsub(/\_id$/, "")).to_sym) + (options.delete(:required) ? @template.content_tag("span", " *", :class => "required"): "")
285 301 label = @template.content_tag("label", label_text,
286 302 :class => (@object && @object.errors[field] ? "error" : nil),
287 303 :for => (@object_name.to_s + "_" + field.to_s))
288 304 label + super
289 305 end
290 306
291 307 end
292 308
@@ -1,129 +1,130
1 1 # redMine - project management software
2 2 # Copyright (C) 2006 Jean-Philippe Lang
3 3 #
4 4 # This program is free software; you can redistribute it and/or
5 5 # modify it under the terms of the GNU General Public License
6 6 # as published by the Free Software Foundation; either version 2
7 7 # of the License, or (at your option) any later version.
8 8 #
9 9 # This program is distributed in the hope that it will be useful,
10 10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
11 11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 12 # GNU General Public License for more details.
13 13 #
14 14 # You should have received a copy of the GNU General Public License
15 15 # along with this program; if not, write to the Free Software
16 16 # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
17 17
18 18 class Project < ActiveRecord::Base
19 19 # Project statuses
20 20 STATUS_ACTIVE = 1
21 21 STATUS_ARCHIVED = 9
22 22
23 23 has_many :members, :dependent => :delete_all, :include => :user, :conditions => "#{User.table_name}.status=#{User::STATUS_ACTIVE}"
24 24 has_many :users, :through => :members
25 25 has_many :custom_values, :dependent => :delete_all, :as => :customized
26 26 has_many :issues, :dependent => :destroy, :order => "#{Issue.table_name}.created_on DESC", :include => [:status, :tracker]
27 27 has_many :issue_changes, :through => :issues, :source => :journals
28 28 has_many :versions, :dependent => :destroy, :order => "#{Version.table_name}.effective_date DESC, #{Version.table_name}.name DESC"
29 29 has_many :time_entries, :dependent => :delete_all
30 30 has_many :queries, :dependent => :delete_all
31 31 has_many :documents, :dependent => :destroy
32 32 has_many :news, :dependent => :delete_all, :include => :author
33 33 has_many :issue_categories, :dependent => :delete_all, :order => "#{IssueCategory.table_name}.name"
34 34 has_many :boards, :order => "position ASC"
35 35 has_one :repository, :dependent => :destroy
36 has_many :changesets, :through => :repository
36 37 has_one :wiki, :dependent => :destroy
37 38 has_and_belongs_to_many :custom_fields, :class_name => 'IssueCustomField', :join_table => "#{table_name_prefix}custom_fields_projects#{table_name_suffix}", :association_foreign_key => 'custom_field_id'
38 39 acts_as_tree :order => "name", :counter_cache => true
39 40
40 41 attr_protected :status
41 42
42 43 validates_presence_of :name, :description, :identifier
43 44 validates_uniqueness_of :name, :identifier
44 45 validates_associated :custom_values, :on => :update
45 46 validates_associated :repository, :wiki
46 47 validates_length_of :name, :maximum => 30
47 48 validates_format_of :name, :with => /^[\w\s\'\-]*$/i
48 49 validates_length_of :description, :maximum => 255
49 50 validates_length_of :homepage, :maximum => 30
50 51 validates_length_of :identifier, :in => 3..12
51 52 validates_format_of :identifier, :with => /^[a-z0-9\-]*$/
52 53
53 54 def identifier=(identifier)
54 55 super unless identifier_frozen?
55 56 end
56 57
57 58 def identifier_frozen?
58 59 errors[:identifier].nil? && !(new_record? || identifier.blank?)
59 60 end
60 61
61 62 def issues_with_subprojects(include_subprojects=false)
62 63 conditions = nil
63 64 if include_subprojects && !active_children.empty?
64 65 ids = [id] + active_children.collect {|c| c.id}
65 66 conditions = ["#{Issue.table_name}.project_id IN (#{ids.join(',')})"]
66 67 end
67 68 conditions ||= ["#{Issue.table_name}.project_id = ?", id]
68 69 Issue.with_scope :find => { :conditions => conditions } do
69 70 yield
70 71 end
71 72 end
72 73
73 74 # returns latest created projects
74 75 # non public projects will be returned only if user is a member of those
75 76 def self.latest(user=nil, count=5)
76 77 find(:all, :limit => count, :conditions => visible_by(user), :order => "created_on DESC")
77 78 end
78 79
79 80 def self.visible_by(user=nil)
80 81 if user && user.admin?
81 82 return "#{Project.table_name}.status=#{Project::STATUS_ACTIVE}"
82 83 elsif user && user.memberships.any?
83 84 return "#{Project.table_name}.status=#{Project::STATUS_ACTIVE} AND (#{Project.table_name}.is_public = #{connection.quoted_true} or #{Project.table_name}.id IN (#{user.memberships.collect{|m| m.project_id}.join(',')}))"
84 85 else
85 86 return "#{Project.table_name}.status=#{Project::STATUS_ACTIVE} AND #{Project.table_name}.is_public = #{connection.quoted_true}"
86 87 end
87 88 end
88 89
89 90 def active?
90 91 self.status == STATUS_ACTIVE
91 92 end
92 93
93 94 def archive
94 95 # Archive subprojects if any
95 96 children.each do |subproject|
96 97 subproject.archive
97 98 end
98 99 update_attribute :status, STATUS_ARCHIVED
99 100 end
100 101
101 102 def unarchive
102 103 return false if parent && !parent.active?
103 104 update_attribute :status, STATUS_ACTIVE
104 105 end
105 106
106 107 def active_children
107 108 children.select {|child| child.active?}
108 109 end
109 110
110 111 # Returns an array of all custom fields enabled for project issues
111 112 # (explictly associated custom fields and custom fields enabled for all projects)
112 113 def custom_fields_for_issues(tracker)
113 114 all_custom_fields.select {|c| tracker.custom_fields.include? c }
114 115 end
115 116
116 117 def all_custom_fields
117 118 @all_custom_fields ||= (IssueCustomField.for_all + custom_fields).uniq
118 119 end
119 120
120 121 def <=>(project)
121 122 name <=> project.name
122 123 end
123 124
124 125 protected
125 126 def validate
126 127 errors.add(parent_id, " must be a root project") if parent and parent.parent
127 128 errors.add_to_base("A project with subprojects can't be a subproject") if parent and children.size > 0
128 129 end
129 130 end
@@ -1,62 +1,62
1 1 <%= error_messages_for 'project' %>
2 2
3 3 <div class="box">
4 4 <!--[form:project]-->
5 5 <p><%= f.text_field :name, :required => true %><br /><em><%= l(:text_caracters_maximum, 30) %></em></p>
6 6
7 7 <% if User.current.admin? and !@root_projects.empty? %>
8 8 <p><%= f.select :parent_id, (@root_projects.collect {|p| [p.name, p.id]}), { :include_blank => true } %></p>
9 9 <% end %>
10 10
11 11 <p><%= f.text_area :description, :required => true, :cols => 60, :rows => 5 %><em><%= l(:text_caracters_maximum, 255) %></em></p>
12 12 <p><%= f.text_field :identifier, :required => true, :size => 15, :disabled => @project.identifier_frozen? %><br /><em><%= l(:text_length_between, 3, 12) %> <%= l(:text_project_identifier_info) unless @project.identifier_frozen? %></em></p>
13 13 <p><%= f.text_field :homepage, :size => 40 %></p>
14 14 <p><%= f.check_box :is_public %></p>
15 15 <%= wikitoolbar_for 'project_description' %>
16 16
17 17 <% for @custom_value in @custom_values %>
18 18 <p><%= custom_field_tag_with_label @custom_value %></p>
19 19 <% end %>
20 20
21 21 <% unless @custom_fields.empty? %>
22 22 <p><label><%=l(:label_custom_field_plural)%></label>
23 23 <% for custom_field in @custom_fields %>
24 24 <%= check_box_tag "custom_field_ids[]", custom_field.id, ((@project.custom_fields.include? custom_field) or custom_field.is_for_all?), (custom_field.is_for_all? ? {:disabled => "disabled"} : {}) %>
25 25 <%= custom_field.name %>
26 26 <% end %></p>
27 27 <% end %>
28 28 <!--[eoform:project]-->
29 29 </div>
30 30
31 31 <div class="box">
32 32 <h3><%= check_box_tag "repository_enabled", 1, !@project.repository.nil?, :onclick => "Element.toggle('repository');" %> <%= l(:label_repository) %></h3>
33 33 <%= hidden_field_tag "repository_enabled", 0 %>
34 34 <div id="repository">
35 35 <p class="tabular"><label>SCM</label><%= scm_select_tag %></p>
36 36 <div id="repository_fields">
37 37 <%= render :partial => 'projects/repository', :locals => {:repository => @project.repository} if @project.repository %>
38 38 </div>
39 39 </div>
40 40 </div>
41 41 <%= javascript_tag "Element.hide('repository');" if @project.repository.nil? %>
42 42
43 43 <div class="box">
44 44 <h3><%= check_box_tag "wiki_enabled", 1, !@project.wiki.nil?, :onclick => "Element.toggle('wiki');" %> <%= l(:label_wiki) %></h3>
45 45 <%= hidden_field_tag "wiki_enabled", 0 %>
46 46 <div id="wiki">
47 47 <% fields_for :wiki, @project.wiki, { :builder => TabularFormBuilder, :lang => current_language} do |wiki| %>
48 <p><%= wiki.text_field :start_page, :size => 60, :required => true %><br /><em><%= l(:text_unallowed_characters) %>: , . / ? ; |</em></p>
48 <p><%= wiki.text_field :start_page, :size => 60, :required => true %><br /><em><%= l(:text_unallowed_characters) %>: , . / ? ; : |</em></p>
49 49 <% # content_tag("div", "", :id => "wiki_start_page_auto_complete", :class => "auto_complete") +
50 50 # auto_complete_field("wiki_start_page", { :url => { :controller => 'wiki', :action => 'auto_complete_for_wiki_page', :id => @project } })
51 51 %>
52 52 <% end %>
53 53 </div>
54 54 <%= javascript_tag "Element.hide('wiki');" if @project.wiki.nil? %>
55 55 </div>
56 56
57 57 <% content_for :header_tags do %>
58 58 <%= javascript_include_tag 'calendar/calendar' %>
59 59 <%= javascript_include_tag "calendar/lang/calendar-#{current_language}.js" %>
60 60 <%= javascript_include_tag 'calendar/calendar-setup' %>
61 61 <%= stylesheet_link_tag 'calendar' %>
62 62 <% end %>
@@ -1,5 +1,3
1 1 <div class="wiki">
2 <% cache "wiki/show/#{content.page.id}/#{content.version}" do %>
3 2 <%= textilizable content.text, :attachments => content.page.attachments %>
4 <% end %>
5 3 </div>
1 NO CONTENT: modified file, binary diff hidden
@@ -1,702 +1,706
1 1 /* andreas08 - an open source xhtml/css website layout by Andreas Viklund - http://andreasviklund.com . Free to use in any way and for any purpose as long as the proper credits are given to the original designer. Version: 1.0, November 28, 2005 */
2 2 /* Edited by Jean-Philippe Lang *>
3 3 /**************** Body and tag styles ****************/
4 4
5 5 #header * {margin:0; padding:0;}
6 6 p, ul, ol, li {margin:0; padding:0;}
7 7
8 8 body{
9 9 font:76% Verdana,Tahoma,Arial,sans-serif;
10 10 line-height:1.4em;
11 11 text-align:center;
12 12 color:#303030;
13 13 background:#e8eaec;
14 14 margin:0;
15 15 }
16 16
17 17 a{color:#467aa7;font-weight:bold;text-decoration:none;background-color:inherit;}
18 18 a:hover{color:#2a5a8a; text-decoration:none; background-color:inherit;}
19 19 a img{border:none;}
20 20
21 21 p{margin:0 0 1em 0;}
22 22 p form{margin-top:0; margin-bottom:20px;}
23 23
24 24 img.left,img.center,img.right{padding:4px; border:1px solid #a0a0a0;}
25 25 img.left{float:left; margin:0 12px 5px 0;}
26 26 img.center{display:block; margin:0 auto 5px auto;}
27 27 img.right{float:right; margin:0 0 5px 12px;}
28 28
29 29 /**************** Header and navigation styles ****************/
30 30
31 31 #container{
32 32 width:100%;
33 33 min-width: 800px;
34 34 margin:0;
35 35 padding:0;
36 36 text-align:left;
37 37 background:#ffffff;
38 38 color:#303030;
39 39 }
40 40
41 41 #header{
42 42 height:4.5em;
43 43 margin:0;
44 44 background:#467aa7;
45 45 color:#ffffff;
46 46 margin-bottom:1px;
47 47 }
48 48
49 49 #header h1{
50 50 padding:10px 0 0 20px;
51 51 font-size:2em;
52 52 background-color:inherit;
53 53 color:#fff;
54 54 letter-spacing:-1px;
55 55 font-weight:bold;
56 56 font-family: Trebuchet MS,Georgia,"Times New Roman",serif;
57 57 }
58 58
59 59 #header h2{
60 60 margin:3px 0 0 40px;
61 61 font-size:1.5em;
62 62 background-color:inherit;
63 63 color:#f0f2f4;
64 64 letter-spacing:-1px;
65 65 font-weight:normal;
66 66 font-family: Trebuchet MS,Georgia,"Times New Roman",serif;
67 67 }
68 68
69 69 #header a {color:#fff;}
70 70
71 71 #navigation{
72 72 height:2.2em;
73 73 line-height:2.2em;
74 74 margin:0;
75 75 background:#578bb8;
76 76 color:#ffffff;
77 77 }
78 78
79 79 #navigation li{
80 80 float:left;
81 81 list-style-type:none;
82 82 border-right:1px solid #ffffff;
83 83 white-space:nowrap;
84 84 }
85 85
86 86 #navigation li.right {
87 87 float:right;
88 88 list-style-type:none;
89 89 border-right:0;
90 90 border-left:1px solid #ffffff;
91 91 white-space:nowrap;
92 92 }
93 93
94 94 #navigation li a{
95 95 display:block;
96 96 padding:0px 10px 0px 22px;
97 97 font-size:0.8em;
98 98 font-weight:normal;
99 99 text-decoration:none;
100 100 background-color:inherit;
101 101 color: #ffffff;
102 102 }
103 103
104 104 #navigation li.submenu {background:url(../images/arrow_down.png) 96% 80% no-repeat;}
105 105 #navigation li.submenu a {padding:0px 16px 0px 22px;}
106 106 * html #navigation a {width:1%;}
107 107
108 108 #navigation .selected,#navigation a:hover{
109 109 color:#ffffff;
110 110 text-decoration:none;
111 111 background-color: #80b0da;
112 112 }
113 113
114 114 /**************** Icons *******************/
115 115 .icon {
116 116 background-position: 0% 40%;
117 117 background-repeat: no-repeat;
118 118 padding-left: 20px;
119 119 padding-top: 2px;
120 120 padding-bottom: 3px;
121 121 vertical-align: middle;
122 122 }
123 123
124 124 #navigation .icon {
125 125 background-position: 4px 50%;
126 126 }
127 127
128 128 .icon22 {
129 129 background-position: 0% 40%;
130 130 background-repeat: no-repeat;
131 131 padding-left: 26px;
132 132 line-height: 22px;
133 133 vertical-align: middle;
134 134 }
135 135
136 136 .icon-add { background-image: url(../images/add.png); }
137 137 .icon-edit { background-image: url(../images/edit.png); }
138 138 .icon-del { background-image: url(../images/delete.png); }
139 139 .icon-move { background-image: url(../images/move.png); }
140 140 .icon-save { background-image: url(../images/save.png); }
141 141 .icon-cancel { background-image: url(../images/cancel.png); }
142 142 .icon-pdf { background-image: url(../images/pdf.png); }
143 143 .icon-csv { background-image: url(../images/csv.png); }
144 144 .icon-html { background-image: url(../images/html.png); }
145 145 .icon-image { background-image: url(../images/image.png); }
146 146 .icon-txt { background-image: url(../images/txt.png); }
147 147 .icon-file { background-image: url(../images/file.png); }
148 148 .icon-folder { background-image: url(../images/folder.png); }
149 149 .icon-package { background-image: url(../images/package.png); }
150 150 .icon-home { background-image: url(../images/home.png); }
151 151 .icon-user { background-image: url(../images/user.png); }
152 152 .icon-mypage { background-image: url(../images/user_page.png); }
153 153 .icon-admin { background-image: url(../images/admin.png); }
154 154 .icon-projects { background-image: url(../images/projects.png); }
155 155 .icon-logout { background-image: url(../images/logout.png); }
156 156 .icon-help { background-image: url(../images/help.png); }
157 157 .icon-attachment { background-image: url(../images/attachment.png); }
158 158 .icon-index { background-image: url(../images/index.png); }
159 159 .icon-history { background-image: url(../images/history.png); }
160 160 .icon-feed { background-image: url(../images/feed.png); }
161 161 .icon-time { background-image: url(../images/time.png); }
162 162 .icon-stats { background-image: url(../images/stats.png); }
163 163 .icon-warning { background-image: url(../images/warning.png); }
164 164 .icon-fav { background-image: url(../images/fav.png); }
165 165 .icon-fav-off { background-image: url(../images/fav_off.png); }
166 166 .icon-reload { background-image: url(../images/reload.png); }
167 167 .icon-lock { background-image: url(../images/locked.png); }
168 168 .icon-unlock { background-image: url(../images/unlock.png); }
169 169
170 170 .icon22-projects { background-image: url(../images/22x22/projects.png); }
171 171 .icon22-users { background-image: url(../images/22x22/users.png); }
172 172 .icon22-tracker { background-image: url(../images/22x22/tracker.png); }
173 173 .icon22-role { background-image: url(../images/22x22/role.png); }
174 174 .icon22-workflow { background-image: url(../images/22x22/workflow.png); }
175 175 .icon22-options { background-image: url(../images/22x22/options.png); }
176 176 .icon22-notifications { background-image: url(../images/22x22/notifications.png); }
177 177 .icon22-authent { background-image: url(../images/22x22/authent.png); }
178 178 .icon22-info { background-image: url(../images/22x22/info.png); }
179 179 .icon22-comment { background-image: url(../images/22x22/comment.png); }
180 180 .icon22-package { background-image: url(../images/22x22/package.png); }
181 181 .icon22-settings { background-image: url(../images/22x22/settings.png); }
182 182
183 183 /**************** Content styles ****************/
184 184
185 185 html>body #content {
186 186 height: auto;
187 187 min-height: 500px;
188 188 }
189 189
190 190 #content{
191 191 width: auto;
192 192 height:500px;
193 193 font-size:0.9em;
194 194 padding:20px 10px 10px 20px;
195 195 margin-left: 120px;
196 196 border-left: 1px dashed #c0c0c0;
197 197
198 198 }
199 199
200 200 #content h2, #content div.wiki h1 {
201 201 display:block;
202 202 margin:0 0 16px 0;
203 203 font-size:1.7em;
204 204 font-weight:normal;
205 205 letter-spacing:-1px;
206 206 color:#606060;
207 207 background-color:inherit;
208 208 font-family: Trebuchet MS,Georgia,"Times New Roman",serif;
209 209 }
210 210
211 211 #content h2 a{font-weight:normal;}
212 212 #content h3{margin:0 0 12px 0; font-size:1.4em;color:#707070;font-family: Trebuchet MS,Georgia,"Times New Roman",serif;}
213 213 #content h4{font-size: 1em; margin-bottom: 12px; margin-top: 20px; font-weight: normal; border-bottom: dotted 1px #c0c0c0;}
214 214 #content a:hover,#subcontent a:hover{text-decoration:underline;}
215 215 #content ul,#content ol{margin:0 5px 16px 35px;}
216 216 #content dl{margin:0 5px 10px 25px;}
217 217 #content dt{font-weight:bold; margin-bottom:5px;}
218 218 #content dd{margin:0 0 10px 15px;}
219 219
220 220 #content .tabs{height: 2.6em;}
221 221 #content .tabs ul{margin:0;}
222 222 #content .tabs ul li{
223 223 float:left;
224 224 list-style-type:none;
225 225 white-space:nowrap;
226 226 margin-right:8px;
227 227 background:#fff;
228 228 }
229 229 #content .tabs ul li a{
230 230 display:block;
231 231 font-size: 0.9em;
232 232 text-decoration:none;
233 233 line-height:1em;
234 234 padding:4px;
235 235 border: 1px solid #c0c0c0;
236 236 }
237 237
238 238 #content .tabs ul li a.selected, #content .tabs ul li a:hover{
239 239 background-color: #80b0da;
240 240 border: 1px solid #80b0da;
241 241 color: #fff;
242 242 text-decoration:none;
243 243 }
244 244
245 245 /***********************************************/
246 246
247 247 form {display: inline;}
248 248 blockquote {padding-left: 6px; border-left: 2px solid #ccc;}
249 249 input, select {vertical-align: middle; margin-top: 1px; margin-bottom: 1px;}
250 250
251 251 input.button-small {font-size: 0.8em;}
252 252 textarea.wiki-edit { width: 99.5%; }
253 253 .select-small {font-size: 0.8em;}
254 254 label {font-weight: bold; font-size: 1em; color: #505050;}
255 255 fieldset {border:1px solid #c0c0c0; padding: 6px;}
256 256 legend {color: #505050;}
257 257 .required {color: #bb0000;}
258 258 .odd {background-color:#f6f7f8;}
259 259 .even {background-color: #fff;}
260 260 hr { border:0; border-top: dotted 1px #fff; border-bottom: dotted 1px #c0c0c0; }
261 261 table p {margin:0; padding:0;}
262 262
263 263 .highlight { background-color: #FCFD8D;}
264 264
265 265 div.square {
266 266 border: 1px solid #999;
267 267 float: left;
268 268 margin: .4em .5em 0 0;
269 269 overflow: hidden;
270 270 width: .6em; height: .6em;
271 271 }
272 272
273 273 ul.documents {
274 274 list-style-type: none;
275 275 padding: 0;
276 276 margin: 0;
277 277 }
278 278
279 279 ul.documents li {
280 280 background-image: url(../images/32x32/file.png);
281 281 background-repeat: no-repeat;
282 282 background-position: 0 1px;
283 283 padding-left: 36px;
284 284 margin-bottom: 10px;
285 285 margin-left: -37px;
286 286 }
287 287
288 288 /********** Table used to display lists of things ***********/
289 289
290 290 table.list {
291 291 width:100%;
292 292 border-collapse: collapse;
293 293 border: 1px dotted #d0d0d0;
294 294 margin-bottom: 6px;
295 295 }
296 296
297 297 table.with-cells td {
298 298 border: 1px solid #d7d7d7;
299 299 }
300 300
301 301 table.list td {
302 302 padding:2px;
303 303 }
304 304
305 305 table.list thead th {
306 306 text-align: center;
307 307 background: #eee;
308 308 border: 1px solid #d7d7d7;
309 309 color: #777;
310 310 }
311 311
312 312 table.list tbody th {
313 313 font-weight: bold;
314 314 background: #eed;
315 315 border: 1px solid #d7d7d7;
316 316 color: #777;
317 317 }
318 318
319 319 /*========== Drop down menu ==============*/
320 320 div.menu {
321 321 background-color: #FFFFFF;
322 322 border-style: solid;
323 323 border-width: 1px;
324 324 border-color: #7F9DB9;
325 325 position: absolute;
326 326 top: 0px;
327 327 left: 0px;
328 328 padding: 0;
329 329 visibility: hidden;
330 330 z-index: 101;
331 331 }
332 332
333 333 div.menu a.menuItem {
334 334 font-size: 10px;
335 335 font-weight: normal;
336 336 line-height: 2em;
337 337 color: #000000;
338 338 background-color: #FFFFFF;
339 339 cursor: default;
340 340 display: block;
341 341 padding: 0 1em;
342 342 margin: 0;
343 343 border: 0;
344 344 text-decoration: none;
345 345 white-space: nowrap;
346 346 }
347 347
348 348 div.menu a.menuItem:hover, div.menu a.menuItemHighlight {
349 349 background-color: #80b0da;
350 350 color: #ffffff;
351 351 }
352 352
353 353 div.menu a.menuItem span.menuItemText {}
354 354
355 355 div.menu a.menuItem span.menuItemArrow {
356 356 margin-right: -.75em;
357 357 }
358 358
359 359 /**************** Sidebar styles ****************/
360 360
361 361 #subcontent{
362 362 position: absolute;
363 363 left: 0px;
364 364 width:95px;
365 365 padding:20px 20px 10px 5px;
366 366 overflow: hidden;
367 367 }
368 368
369 369 #subcontent h2{
370 370 display:block;
371 371 margin:0 0 5px 0;
372 372 font-size:1.0em;
373 373 font-weight:bold;
374 374 text-align:left;
375 375 color:#606060;
376 376 background-color:inherit;
377 377 font-family: Trebuchet MS,Georgia,"Times New Roman",serif;
378 378 }
379 379
380 380 #subcontent p{margin:0 0 16px 0; font-size:0.9em;}
381 381
382 382 /**************** Menublock styles ****************/
383 383
384 384 .menublock{margin:0 0 20px 8px; font-size:0.8em;}
385 385 .menublock li{list-style:none; display:block; padding:1px; margin-bottom:0px;}
386 386 .menublock li a{font-weight:bold; text-decoration:none;}
387 387 .menublock li a:hover{text-decoration:none;}
388 388 .menublock li ul{margin:0; font-size:1em; font-weight:normal;}
389 389 .menublock li ul li{margin-bottom:0;}
390 390 .menublock li ul a{font-weight:normal;}
391 391
392 392 /**************** Footer styles ****************/
393 393
394 394 #footer{
395 395 clear:both;
396 396 padding:5px 0;
397 397 margin:0;
398 398 font-size:0.9em;
399 399 color:#f0f0f0;
400 400 background:#467aa7;
401 401 }
402 402
403 403 #footer p{padding:0; margin:0; text-align:center;}
404 404 #footer a{color:#f0f0f0; background-color:inherit; font-weight:bold;}
405 405 #footer a:hover{color:#ffffff; background-color:inherit; text-decoration: underline;}
406 406
407 407 /**************** Misc classes and styles ****************/
408 408
409 409 .splitcontentleft{float:left; width:49%;}
410 410 .splitcontentright{float:right; width:49%;}
411 411 .clear{clear:both;}
412 412 .small{font-size:0.8em;line-height:1.4em;padding:0 0 0 0;}
413 413 .hide{display:none;}
414 414 .textcenter{text-align:center;}
415 415 .textright{text-align:right;}
416 416 .important{color:#f02025; background-color:inherit; font-weight:bold;}
417 417
418 418 .box{
419 419 margin:0 0 20px 0;
420 420 padding:10px;
421 421 border:1px solid #c0c0c0;
422 422 background-color:#fafbfc;
423 423 color:#505050;
424 424 line-height:1.5em;
425 425 }
426 426
427 427 a.close-icon {
428 428 display:block;
429 429 margin-top:3px;
430 430 overflow:hidden;
431 431 width:12px;
432 432 height:12px;
433 433 background-repeat: no-repeat;
434 434 cursor:pointer;
435 435 background-image:url('../images/close.png');
436 436 }
437 437
438 438 a.close-icon:hover {
439 439 background-image:url('../images/close_hl.png');
440 440 }
441 441
442 442 .rightbox{
443 443 background: #fafbfc;
444 444 border: 1px solid #c0c0c0;
445 445 float: right;
446 446 padding: 8px;
447 447 position: relative;
448 448 margin: 0 5px 5px;
449 449 }
450 450
451 451 div.attachments {padding-left: 6px; border-left: 2px solid #ccc; margin-bottom: 8px;}
452 452 div.attachments p {margin-bottom:2px;}
453 453
454 454 .overlay{
455 455 position: absolute;
456 456 margin-left:0;
457 457 z-index: 50;
458 458 }
459 459
460 460 .layout-active {
461 461 background: #ECF3E1;
462 462 }
463 463
464 464 .block-receiver {
465 465 border:1px dashed #c0c0c0;
466 466 margin-bottom: 20px;
467 467 padding: 15px 0 15px 0;
468 468 }
469 469
470 470 .mypage-box {
471 471 margin:0 0 20px 0;
472 472 color:#505050;
473 473 line-height:1.5em;
474 474 }
475 475
476 476 .handle {
477 477 cursor: move;
478 478 }
479 479
480 480 .login {
481 481 width: 50%;
482 482 text-align: left;
483 483 }
484 484
485 485 img.calendar-trigger {
486 486 cursor: pointer;
487 487 vertical-align: middle;
488 488 margin-left: 4px;
489 489 }
490 490
491 491 #history p {
492 492 margin-left: 34px;
493 493 }
494 494
495 495 .progress {
496 496 border: 1px solid #D7D7D7;
497 497 border-collapse: collapse;
498 498 border-spacing: 0pt;
499 499 empty-cells: show;
500 500 padding: 3px;
501 501 width: 40em;
502 502 text-align: center;
503 503 }
504 504
505 505 .progress td { height: 1em; }
506 506 .progress .closed { background: #BAE0BA none repeat scroll 0%; }
507 507 .progress .open { background: #FFF none repeat scroll 0%; }
508 508
509 509 /***** Contextual links div *****/
510 510 .contextual {
511 511 float: right;
512 512 font-size: 0.8em;
513 513 line-height: 16px;
514 514 padding: 2px;
515 515 }
516 516
517 517 .contextual select, .contextual input {
518 518 font-size: 1em;
519 519 }
520 520
521 521 /***** Gantt chart *****/
522 522 .gantt_hdr {
523 523 position:absolute;
524 524 top:0;
525 525 height:16px;
526 526 border-top: 1px solid #c0c0c0;
527 527 border-bottom: 1px solid #c0c0c0;
528 528 border-right: 1px solid #c0c0c0;
529 529 text-align: center;
530 530 overflow: hidden;
531 531 }
532 532
533 533 .task {
534 534 position: absolute;
535 535 height:8px;
536 536 font-size:0.8em;
537 537 color:#888;
538 538 padding:0;
539 539 margin:0;
540 540 line-height:0.8em;
541 541 }
542 542
543 543 .task_late { background:#f66 url(../images/task_late.png); border: 1px solid #f66; }
544 544 .task_done { background:#66f url(../images/task_done.png); border: 1px solid #66f; }
545 545 .task_todo { background:#aaa url(../images/task_todo.png); border: 1px solid #aaa; }
546 546 .milestone { background-image:url(../images/milestone.png); background-repeat: no-repeat; border: 0; }
547 547
548 548 /***** Tooltips ******/
549 549 .tooltip{position:relative;z-index:24;}
550 550 .tooltip:hover{z-index:25;color:#000;}
551 551 .tooltip span.tip{display: none; text-align:left;}
552 552
553 553 div.tooltip:hover span.tip{
554 554 display:block;
555 555 position:absolute;
556 556 top:12px; left:24px; width:270px;
557 557 border:1px solid #555;
558 558 background-color:#fff;
559 559 padding: 4px;
560 560 font-size: 0.8em;
561 561 color:#505050;
562 562 }
563 563
564 564 /***** CSS FORM ******/
565 565 .tabular p{
566 566 margin: 0;
567 567 padding: 5px 0 8px 0;
568 568 padding-left: 180px; /*width of left column containing the label elements*/
569 569 height: 1%;
570 570 clear:both;
571 571 }
572 572
573 573 .tabular label{
574 574 font-weight: bold;
575 575 float: left;
576 576 margin-left: -180px; /*width of left column*/
577 577 margin-bottom: 10px;
578 578 width: 175px; /*width of labels. Should be smaller than left column to create some right
579 579 margin*/
580 580 }
581 581
582 582 .error {
583 583 color: #cc0000;
584 584 }
585 585
586 586 #settings .tabular p{ padding-left: 300px; }
587 587 #settings .tabular label{ margin-left: -300px; width: 295px; }
588 588
589 589 /*.threepxfix class below:
590 590 Targets IE6- ONLY. Adds 3 pixel indent for multi-line form contents.
591 591 to account for 3 pixel bug: http://www.positioniseverything.net/explorer/threepxtest.html
592 592 */
593 593
594 594 * html .threepxfix{
595 595 margin-left: 3px;
596 596 }
597 597
598 598 /***** Wiki sections ****/
599 599 #content div.wiki { font-size: 110%}
600 600
601 601 #content div.wiki h2, div.wiki h3 { font-family: Trebuchet MS,Georgia,"Times New Roman",serif; color:#606060; }
602 602 #content div.wiki h2 { font-size: 1.4em;}
603 603 #content div.wiki h3 { font-size: 1.2em;}
604 604
605 605 div.wiki table {
606 606 border: 1px solid #505050;
607 607 border-collapse: collapse;
608 608 }
609 609
610 610 div.wiki table, div.wiki td, div.wiki th {
611 611 border: 1px solid #bbb;
612 612 padding: 4px;
613 613 }
614 614
615 615 div.wiki a {
616 616 background-position: 0% 60%;
617 617 background-repeat: no-repeat;
618 padding-left: 14px;
618 padding-left: 12px;
619 619 background-image: url(../images/external.png);
620 620 }
621 621
622 622 div.wiki a.wiki-page, div.wiki a.issue, div.wiki a.changeset, div.wiki a.email {
623 623 padding-left: 0;
624 624 background-image: none;
625 625 }
626 626
627 div.wiki a.new {
628 color: #b73535;
629 }
630
627 631 div.wiki code {
628 632 font-size: 1.2em;
629 633 }
630 634
631 635 div.wiki img {
632 636 margin: 6px;
633 637 }
634 638
635 639 div.wiki pre {
636 640 margin: 1em 1em 1em 1.6em;
637 641 padding: 2px;
638 642 background-color: #fafafa;
639 643 border: 1px solid #dadada;
640 644 }
641 645
642 646 .diff_out{
643 647 background: #fcc;
644 648 }
645 649
646 650 .diff_in{
647 651 background: #cfc;
648 652 }
649 653
650 654 #preview .preview { background: #fafbfc url(../images/draft.png); }
651 655
652 656 #ajax-indicator {
653 657 position: absolute; /* fixed not supported by IE */
654 658 background-color:#eee;
655 659 border: 1px solid #bbb;
656 660 top:35%;
657 661 left:40%;
658 662 width:20%;
659 663 font-weight:bold;
660 664 text-align:center;
661 665 padding:0.6em;
662 666 z-index:100;
663 667 filter:alpha(opacity=50);
664 668 -moz-opacity:0.5;
665 669 opacity: 0.5;
666 670 -khtml-opacity: 0.5;
667 671 }
668 672
669 673 html>body #ajax-indicator { position: fixed; }
670 674
671 675 #ajax-indicator span {
672 676 background-position: 0% 40%;
673 677 background-repeat: no-repeat;
674 678 background-image: url(../images/loading.gif);
675 679 padding-left: 26px;
676 680 vertical-align: bottom;
677 681 }
678 682
679 683 /***** Flash & error messages ****/
680 684 #flash div, #errorExplanation {
681 685 padding: 4px 4px 4px 30px;
682 686 margin-bottom: 16px;
683 687 font-size: 1.1em;
684 688 border: 2px solid;
685 689 }
686 690
687 691 #flash div.error, #errorExplanation {
688 692 background: url(../images/false.png) 8px 5px no-repeat;
689 693 background-color: #ffe3e3;
690 694 border-color: #dd0000;
691 695 color: #550000;
692 696 }
693 697
694 698 #flash div.notice {
695 699 background: url(../images/true.png) 8px 5px no-repeat;
696 700 background-color: #dfffdf;
697 701 border-color: #9fcf9f;
698 702 color: #005f00;
699 703 }
700 704
701 705 #errorExplanation ul { margin-bottom: 0px; }
702 706 #errorExplanation ul li { list-style: none; margin-left: -16px;}
@@ -1,66 +1,69
1 1 # redMine - project management software
2 2 # Copyright (C) 2006-2007 Jean-Philippe Lang
3 3 #
4 4 # This program is free software; you can redistribute it and/or
5 5 # modify it under the terms of the GNU General Public License
6 6 # as published by the Free Software Foundation; either version 2
7 7 # of the License, or (at your option) any later version.
8 8 #
9 9 # This program is distributed in the hope that it will be useful,
10 10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
11 11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 12 # GNU General Public License for more details.
13 13 #
14 14 # You should have received a copy of the GNU General Public License
15 15 # along with this program; if not, write to the Free Software
16 16 # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
17 17
18 18 require File.dirname(__FILE__) + '/../../test_helper'
19 19
20 20 class ApplicationHelperTest < HelperTestCase
21 21 include ApplicationHelper
22 fixtures :projects
22 include ActionView::Helpers::TextHelper
23 fixtures :projects, :repositories, :changesets, :trackers, :issue_statuses, :issues
23 24
24 25 def setup
25 26 super
26 27 end
27 28
28 29 def test_auto_links
29 30 to_test = {
30 31 'http://foo.bar' => '<a href="http://foo.bar">http://foo.bar</a>',
31 32 'www.foo.bar' => '<a href="http://www.foo.bar">www.foo.bar</a>',
32 33 'http://foo.bar/page?p=1&t=z&s=' => '<a href="http://foo.bar/page?p=1&#38;t=z&#38;s=">http://foo.bar/page?p=1&#38;t=z&#38;s=</a>',
33 34 'http://foo.bar/page#125' => '<a href="http://foo.bar/page#125">http://foo.bar/page#125</a>'
34 35 }
35 36 to_test.each { |text, result| assert_equal "<p>#{result}</p>", textilizable(text) }
36 37 end
37 38
38 39 def test_auto_mailto
39 40 assert_equal '<p><a href="mailto:test@foo.bar" class="email">test@foo.bar</a></p>',
40 41 textilizable('test@foo.bar')
41 42 end
42 43
43 44 def test_textile_tags
44 45 to_test = {
45 46 # inline images
46 47 '!http://foo.bar/image.jpg!' => '<img src="http://foo.bar/image.jpg" alt="" />',
47 48 'floating !>http://foo.bar/image.jpg!' => 'floating <div style="float:right"><img src="http://foo.bar/image.jpg" alt="" /></div>',
48 49 # textile links
49 50 'This is a "link":http://foo.bar' => 'This is a <a href="http://foo.bar">link</a>',
50 51 '"link (Link title)":http://foo.bar' => '<a href="http://foo.bar" title="Link title">link</a>'
51 52 }
52 53 to_test.each { |text, result| assert_equal "<p>#{result}</p>", textilizable(text) }
53 54 end
54 55
55 56 def test_redmine_links
56 issue_link = link_to('#52', {:controller => 'issues', :action => 'show', :id => 52}, :class => 'issue')
57 changeset_link = link_to('r19', {:controller => 'repositories', :action => 'revision', :id => 1, :rev => 19}, :class => 'changeset')
57 issue_link = link_to('#3', {:controller => 'issues', :action => 'show', :id => 3},
58 :class => 'issue', :title => 'Error 281 when updating a recipe (New)')
59 changeset_link = link_to('r1', {:controller => 'repositories', :action => 'revision', :id => 1, :rev => 1},
60 :class => 'changeset', :title => 'My very first commit')
58 61
59 62 to_test = {
60 '#52, #52 and #52.' => "#{issue_link}, #{issue_link} and #{issue_link}.",
61 'r19' => changeset_link
63 '#3, #3 and #3.' => "#{issue_link}, #{issue_link} and #{issue_link}.",
64 'r1' => changeset_link
62 65 }
63 66 @project = Project.find(1)
64 67 to_test.each { |text, result| assert_equal "<p>#{result}</p>", textilizable(text) }
65 68 end
66 69 end
General Comments 0
You need to be logged in to leave comments. Login now