##// END OF EJS Templates
Changes how relative date filters work and adds specific filters for filtering dates in past/next n days (#11426)....
Jean-Philippe Lang -
r10546:d62ef6b9b1dc
parent child
Show More
@@ -1,1068 +1,1084
1 1 # Redmine - project management software
2 2 # Copyright (C) 2006-2012 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 QueryColumn
19 19 attr_accessor :name, :sortable, :groupable, :default_order
20 20 include Redmine::I18n
21 21
22 22 def initialize(name, options={})
23 23 self.name = name
24 24 self.sortable = options[:sortable]
25 25 self.groupable = options[:groupable] || false
26 26 if groupable == true
27 27 self.groupable = name.to_s
28 28 end
29 29 self.default_order = options[:default_order]
30 30 @caption_key = options[:caption] || "field_#{name}"
31 31 end
32 32
33 33 def caption
34 34 l(@caption_key)
35 35 end
36 36
37 37 # Returns true if the column is sortable, otherwise false
38 38 def sortable?
39 39 !@sortable.nil?
40 40 end
41 41
42 42 def sortable
43 43 @sortable.is_a?(Proc) ? @sortable.call : @sortable
44 44 end
45 45
46 46 def value(issue)
47 47 issue.send name
48 48 end
49 49
50 50 def css_classes
51 51 name
52 52 end
53 53 end
54 54
55 55 class QueryCustomFieldColumn < QueryColumn
56 56
57 57 def initialize(custom_field)
58 58 self.name = "cf_#{custom_field.id}".to_sym
59 59 self.sortable = custom_field.order_statement || false
60 60 self.groupable = custom_field.group_statement || false
61 61 @cf = custom_field
62 62 end
63 63
64 64 def caption
65 65 @cf.name
66 66 end
67 67
68 68 def custom_field
69 69 @cf
70 70 end
71 71
72 72 def value(issue)
73 73 cv = issue.custom_values.select {|v| v.custom_field_id == @cf.id}.collect {|v| @cf.cast_value(v.value)}
74 74 cv.size > 1 ? cv.sort {|a,b| a.to_s <=> b.to_s} : cv.first
75 75 end
76 76
77 77 def css_classes
78 78 @css_classes ||= "#{name} #{@cf.field_format}"
79 79 end
80 80 end
81 81
82 82 class Query < ActiveRecord::Base
83 83 class StatementInvalid < ::ActiveRecord::StatementInvalid
84 84 end
85 85
86 86 belongs_to :project
87 87 belongs_to :user
88 88 serialize :filters
89 89 serialize :column_names
90 90 serialize :sort_criteria, Array
91 91
92 92 attr_protected :project_id, :user_id
93 93
94 94 validates_presence_of :name
95 95 validates_length_of :name, :maximum => 255
96 96 validate :validate_query_filters
97 97
98 98 @@operators = { "=" => :label_equals,
99 99 "!" => :label_not_equals,
100 100 "o" => :label_open_issues,
101 101 "c" => :label_closed_issues,
102 102 "!*" => :label_none,
103 103 "*" => :label_any,
104 104 ">=" => :label_greater_or_equal,
105 105 "<=" => :label_less_or_equal,
106 106 "><" => :label_between,
107 107 "<t+" => :label_in_less_than,
108 108 ">t+" => :label_in_more_than,
109 "><t+"=> :label_in_the_next_days,
109 110 "t+" => :label_in,
110 111 "t" => :label_today,
111 112 "w" => :label_this_week,
112 113 ">t-" => :label_less_than_ago,
113 114 "<t-" => :label_more_than_ago,
115 "><t-"=> :label_in_the_past_days,
114 116 "t-" => :label_ago,
115 117 "~" => :label_contains,
116 118 "!~" => :label_not_contains,
117 119 "=p" => :label_any_issues_in_project,
118 120 "=!p" => :label_any_issues_not_in_project,
119 121 "!p" => :label_no_issues_in_project}
120 122
121 123 cattr_reader :operators
122 124
123 125 @@operators_by_filter_type = { :list => [ "=", "!" ],
124 126 :list_status => [ "o", "=", "!", "c", "*" ],
125 127 :list_optional => [ "=", "!", "!*", "*" ],
126 128 :list_subprojects => [ "*", "!*", "=" ],
127 :date => [ "=", ">=", "<=", "><", "<t+", ">t+", "t+", "t", "w", ">t-", "<t-", "t-", "!*", "*" ],
128 :date_past => [ "=", ">=", "<=", "><", ">t-", "<t-", "t-", "t", "w", "!*", "*" ],
129 :date => [ "=", ">=", "<=", "><", "<t+", ">t+", "><t+", "t+", "t", "w", ">t-", "<t-", "><t-", "t-", "!*", "*" ],
130 :date_past => [ "=", ">=", "<=", "><", ">t-", "<t-", "><t-", "t-", "t", "w", "!*", "*" ],
129 131 :string => [ "=", "~", "!", "!~", "!*", "*" ],
130 132 :text => [ "~", "!~", "!*", "*" ],
131 133 :integer => [ "=", ">=", "<=", "><", "!*", "*" ],
132 134 :float => [ "=", ">=", "<=", "><", "!*", "*" ],
133 135 :relation => ["=", "=p", "=!p", "!p", "!*", "*"]}
134 136
135 137 cattr_reader :operators_by_filter_type
136 138
137 139 @@available_columns = [
138 140 QueryColumn.new(:project, :sortable => "#{Project.table_name}.name", :groupable => true),
139 141 QueryColumn.new(:tracker, :sortable => "#{Tracker.table_name}.position", :groupable => true),
140 142 QueryColumn.new(:parent, :sortable => ["#{Issue.table_name}.root_id", "#{Issue.table_name}.lft ASC"], :default_order => 'desc', :caption => :field_parent_issue),
141 143 QueryColumn.new(:status, :sortable => "#{IssueStatus.table_name}.position", :groupable => true),
142 144 QueryColumn.new(:priority, :sortable => "#{IssuePriority.table_name}.position", :default_order => 'desc', :groupable => true),
143 145 QueryColumn.new(:subject, :sortable => "#{Issue.table_name}.subject"),
144 146 QueryColumn.new(:author, :sortable => lambda {User.fields_for_order_statement("authors")}, :groupable => true),
145 147 QueryColumn.new(:assigned_to, :sortable => lambda {User.fields_for_order_statement}, :groupable => true),
146 148 QueryColumn.new(:updated_on, :sortable => "#{Issue.table_name}.updated_on", :default_order => 'desc'),
147 149 QueryColumn.new(:category, :sortable => "#{IssueCategory.table_name}.name", :groupable => true),
148 150 QueryColumn.new(:fixed_version, :sortable => lambda {Version.fields_for_order_statement}, :groupable => true),
149 151 QueryColumn.new(:start_date, :sortable => "#{Issue.table_name}.start_date"),
150 152 QueryColumn.new(:due_date, :sortable => "#{Issue.table_name}.due_date"),
151 153 QueryColumn.new(:estimated_hours, :sortable => "#{Issue.table_name}.estimated_hours"),
152 154 QueryColumn.new(:done_ratio, :sortable => "#{Issue.table_name}.done_ratio", :groupable => true),
153 155 QueryColumn.new(:created_on, :sortable => "#{Issue.table_name}.created_on", :default_order => 'desc'),
154 156 QueryColumn.new(:relations, :caption => :label_related_issues)
155 157 ]
156 158 cattr_reader :available_columns
157 159
158 160 scope :visible, lambda {|*args|
159 161 user = args.shift || User.current
160 162 base = Project.allowed_to_condition(user, :view_issues, *args)
161 163 user_id = user.logged? ? user.id : 0
162 164 {
163 165 :conditions => ["(#{table_name}.project_id IS NULL OR (#{base})) AND (#{table_name}.is_public = ? OR #{table_name}.user_id = ?)", true, user_id],
164 166 :include => :project
165 167 }
166 168 }
167 169
168 170 def initialize(attributes=nil, *args)
169 171 super attributes
170 172 self.filters ||= { 'status_id' => {:operator => "o", :values => [""]} }
171 173 @is_for_all = project.nil?
172 174 end
173 175
174 176 def validate_query_filters
175 177 filters.each_key do |field|
176 178 if values_for(field)
177 179 case type_for(field)
178 180 when :integer
179 181 add_filter_error(field, :invalid) if values_for(field).detect {|v| v.present? && !v.match(/^[+-]?\d+$/) }
180 182 when :float
181 183 add_filter_error(field, :invalid) if values_for(field).detect {|v| v.present? && !v.match(/^[+-]?\d+(\.\d*)?$/) }
182 184 when :date, :date_past
183 185 case operator_for(field)
184 186 when "=", ">=", "<=", "><"
185 187 add_filter_error(field, :invalid) if values_for(field).detect {|v| v.present? && (!v.match(/^\d{4}-\d{2}-\d{2}$/) || (Date.parse(v) rescue nil).nil?) }
186 when ">t-", "<t-", "t-", ">t+", "<t+", "t+"
188 when ">t-", "<t-", "t-", ">t+", "<t+", "t+", "><t+", "><t-"
187 189 add_filter_error(field, :invalid) if values_for(field).detect {|v| v.present? && !v.match(/^\d+$/) }
188 190 end
189 191 end
190 192 end
191 193
192 194 add_filter_error(field, :blank) unless
193 195 # filter requires one or more values
194 196 (values_for(field) and !values_for(field).first.blank?) or
195 197 # filter doesn't require any value
196 198 ["o", "c", "!*", "*", "t", "w"].include? operator_for(field)
197 199 end if filters
198 200 end
199 201
200 202 def add_filter_error(field, message)
201 203 m = label_for(field) + " " + l(message, :scope => 'activerecord.errors.messages')
202 204 errors.add(:base, m)
203 205 end
204 206
205 207 # Returns true if the query is visible to +user+ or the current user.
206 208 def visible?(user=User.current)
207 209 (project.nil? || user.allowed_to?(:view_issues, project)) && (self.is_public? || self.user_id == user.id)
208 210 end
209 211
210 212 def editable_by?(user)
211 213 return false unless user
212 214 # Admin can edit them all and regular users can edit their private queries
213 215 return true if user.admin? || (!is_public && self.user_id == user.id)
214 216 # Members can not edit public queries that are for all project (only admin is allowed to)
215 217 is_public && !@is_for_all && user.allowed_to?(:manage_public_queries, project)
216 218 end
217 219
218 220 def trackers
219 221 @trackers ||= project.nil? ? Tracker.find(:all, :order => 'position') : project.rolled_up_trackers
220 222 end
221 223
222 224 # Returns a hash of localized labels for all filter operators
223 225 def self.operators_labels
224 226 operators.inject({}) {|h, operator| h[operator.first] = l(operator.last); h}
225 227 end
226 228
227 229 def available_filters
228 230 return @available_filters if @available_filters
229 231 @available_filters = {
230 232 "status_id" => {
231 233 :type => :list_status, :order => 0,
232 234 :values => IssueStatus.find(:all, :order => 'position').collect{|s| [s.name, s.id.to_s] }
233 235 },
234 236 "tracker_id" => {
235 237 :type => :list, :order => 2, :values => trackers.collect{|s| [s.name, s.id.to_s] }
236 238 },
237 239 "priority_id" => {
238 240 :type => :list, :order => 3, :values => IssuePriority.all.collect{|s| [s.name, s.id.to_s] }
239 241 },
240 242 "subject" => { :type => :text, :order => 8 },
241 243 "created_on" => { :type => :date_past, :order => 9 },
242 244 "updated_on" => { :type => :date_past, :order => 10 },
243 245 "start_date" => { :type => :date, :order => 11 },
244 246 "due_date" => { :type => :date, :order => 12 },
245 247 "estimated_hours" => { :type => :float, :order => 13 },
246 248 "done_ratio" => { :type => :integer, :order => 14 }
247 249 }
248 250 IssueRelation::TYPES.each do |relation_type, options|
249 251 @available_filters[relation_type] = {
250 252 :type => :relation, :order => @available_filters.size + 100,
251 253 :label => options[:name]
252 254 }
253 255 end
254 256 principals = []
255 257 if project
256 258 principals += project.principals.sort
257 259 unless project.leaf?
258 260 subprojects = project.descendants.visible.all
259 261 if subprojects.any?
260 262 @available_filters["subproject_id"] = {
261 263 :type => :list_subprojects, :order => 13,
262 264 :values => subprojects.collect{|s| [s.name, s.id.to_s] }
263 265 }
264 266 principals += Principal.member_of(subprojects)
265 267 end
266 268 end
267 269 else
268 270 if all_projects.any?
269 271 # members of visible projects
270 272 principals += Principal.member_of(all_projects)
271 273 # project filter
272 274 project_values = []
273 275 if User.current.logged? && User.current.memberships.any?
274 276 project_values << ["<< #{l(:label_my_projects).downcase} >>", "mine"]
275 277 end
276 278 project_values += all_projects_values
277 279 @available_filters["project_id"] = {
278 280 :type => :list, :order => 1, :values => project_values
279 281 } unless project_values.empty?
280 282 end
281 283 end
282 284 principals.uniq!
283 285 principals.sort!
284 286 users = principals.select {|p| p.is_a?(User)}
285 287
286 288 assigned_to_values = []
287 289 assigned_to_values << ["<< #{l(:label_me)} >>", "me"] if User.current.logged?
288 290 assigned_to_values += (Setting.issue_group_assignment? ?
289 291 principals : users).collect{|s| [s.name, s.id.to_s] }
290 292 @available_filters["assigned_to_id"] = {
291 293 :type => :list_optional, :order => 4, :values => assigned_to_values
292 294 } unless assigned_to_values.empty?
293 295
294 296 author_values = []
295 297 author_values << ["<< #{l(:label_me)} >>", "me"] if User.current.logged?
296 298 author_values += users.collect{|s| [s.name, s.id.to_s] }
297 299 @available_filters["author_id"] = {
298 300 :type => :list, :order => 5, :values => author_values
299 301 } unless author_values.empty?
300 302
301 303 group_values = Group.all.collect {|g| [g.name, g.id.to_s] }
302 304 @available_filters["member_of_group"] = {
303 305 :type => :list_optional, :order => 6, :values => group_values
304 306 } unless group_values.empty?
305 307
306 308 role_values = Role.givable.collect {|r| [r.name, r.id.to_s] }
307 309 @available_filters["assigned_to_role"] = {
308 310 :type => :list_optional, :order => 7, :values => role_values
309 311 } unless role_values.empty?
310 312
311 313 if User.current.logged?
312 314 @available_filters["watcher_id"] = {
313 315 :type => :list, :order => 15, :values => [["<< #{l(:label_me)} >>", "me"]]
314 316 }
315 317 end
316 318
317 319 if project
318 320 # project specific filters
319 321 categories = project.issue_categories.all
320 322 unless categories.empty?
321 323 @available_filters["category_id"] = {
322 324 :type => :list_optional, :order => 6,
323 325 :values => categories.collect{|s| [s.name, s.id.to_s] }
324 326 }
325 327 end
326 328 versions = project.shared_versions.all
327 329 unless versions.empty?
328 330 @available_filters["fixed_version_id"] = {
329 331 :type => :list_optional, :order => 7,
330 332 :values => versions.sort.collect{|s| ["#{s.project.name} - #{s.name}", s.id.to_s] }
331 333 }
332 334 end
333 335 add_custom_fields_filters(project.all_issue_custom_fields)
334 336 else
335 337 # global filters for cross project issue list
336 338 system_shared_versions = Version.visible.find_all_by_sharing('system')
337 339 unless system_shared_versions.empty?
338 340 @available_filters["fixed_version_id"] = {
339 341 :type => :list_optional, :order => 7,
340 342 :values => system_shared_versions.sort.collect{|s|
341 343 ["#{s.project.name} - #{s.name}", s.id.to_s]
342 344 }
343 345 }
344 346 end
345 347 add_custom_fields_filters(
346 348 IssueCustomField.find(:all,
347 349 :conditions => {
348 350 :is_filter => true,
349 351 :is_for_all => true
350 352 }))
351 353 end
352 354 add_associations_custom_fields_filters :project, :author, :assigned_to, :fixed_version
353 355 if User.current.allowed_to?(:set_issues_private, nil, :global => true) ||
354 356 User.current.allowed_to?(:set_own_issues_private, nil, :global => true)
355 357 @available_filters["is_private"] = {
356 358 :type => :list, :order => 16,
357 359 :values => [[l(:general_text_yes), "1"], [l(:general_text_no), "0"]]
358 360 }
359 361 end
360 362 Tracker.disabled_core_fields(trackers).each {|field|
361 363 @available_filters.delete field
362 364 }
363 365 @available_filters.each do |field, options|
364 366 options[:name] ||= l(options[:label] || "field_#{field}".gsub(/_id$/, ''))
365 367 end
366 368 @available_filters
367 369 end
368 370
369 371 # Returns a representation of the available filters for JSON serialization
370 372 def available_filters_as_json
371 373 json = {}
372 374 available_filters.each do |field, options|
373 375 json[field] = options.slice(:type, :name, :values).stringify_keys
374 376 end
375 377 json
376 378 end
377 379
378 380 def all_projects
379 381 @all_projects ||= Project.visible.all
380 382 end
381 383
382 384 def all_projects_values
383 385 return @all_projects_values if @all_projects_values
384 386
385 387 values = []
386 388 Project.project_tree(all_projects) do |p, level|
387 389 prefix = (level > 0 ? ('--' * level + ' ') : '')
388 390 values << ["#{prefix}#{p.name}", p.id.to_s]
389 391 end
390 392 @all_projects_values = values
391 393 end
392 394
393 395 def add_filter(field, operator, values)
394 396 # values must be an array
395 397 return unless values.nil? || values.is_a?(Array)
396 398 # check if field is defined as an available filter
397 399 if available_filters.has_key? field
398 400 filter_options = available_filters[field]
399 401 # check if operator is allowed for that filter
400 402 #if @@operators_by_filter_type[filter_options[:type]].include? operator
401 403 # allowed_values = values & ([""] + (filter_options[:values] || []).collect {|val| val[1]})
402 404 # filters[field] = {:operator => operator, :values => allowed_values } if (allowed_values.first and !allowed_values.first.empty?) or ["o", "c", "!*", "*", "t"].include? operator
403 405 #end
404 406 filters[field] = {:operator => operator, :values => (values || [''])}
405 407 end
406 408 end
407 409
408 410 def add_short_filter(field, expression)
409 411 return unless expression && available_filters.has_key?(field)
410 412 field_type = available_filters[field][:type]
411 413 @@operators_by_filter_type[field_type].sort.reverse.detect do |operator|
412 414 next unless expression =~ /^#{Regexp.escape(operator)}(.*)$/
413 415 add_filter field, operator, $1.present? ? $1.split('|') : ['']
414 416 end || add_filter(field, '=', expression.split('|'))
415 417 end
416 418
417 419 # Add multiple filters using +add_filter+
418 420 def add_filters(fields, operators, values)
419 421 if fields.is_a?(Array) && operators.is_a?(Hash) && (values.nil? || values.is_a?(Hash))
420 422 fields.each do |field|
421 423 add_filter(field, operators[field], values && values[field])
422 424 end
423 425 end
424 426 end
425 427
426 428 def has_filter?(field)
427 429 filters and filters[field]
428 430 end
429 431
430 432 def type_for(field)
431 433 available_filters[field][:type] if available_filters.has_key?(field)
432 434 end
433 435
434 436 def operator_for(field)
435 437 has_filter?(field) ? filters[field][:operator] : nil
436 438 end
437 439
438 440 def values_for(field)
439 441 has_filter?(field) ? filters[field][:values] : nil
440 442 end
441 443
442 444 def value_for(field, index=0)
443 445 (values_for(field) || [])[index]
444 446 end
445 447
446 448 def label_for(field)
447 449 label = available_filters[field][:name] if available_filters.has_key?(field)
448 450 label ||= l("field_#{field.to_s.gsub(/_id$/, '')}", :default => field)
449 451 end
450 452
451 453 def available_columns
452 454 return @available_columns if @available_columns
453 455 @available_columns = ::Query.available_columns.dup
454 456 @available_columns += (project ?
455 457 project.all_issue_custom_fields :
456 458 IssueCustomField.find(:all)
457 459 ).collect {|cf| QueryCustomFieldColumn.new(cf) }
458 460
459 461 if User.current.allowed_to?(:view_time_entries, project, :global => true)
460 462 index = nil
461 463 @available_columns.each_with_index {|column, i| index = i if column.name == :estimated_hours}
462 464 index = (index ? index + 1 : -1)
463 465 # insert the column after estimated_hours or at the end
464 466 @available_columns.insert index, QueryColumn.new(:spent_hours,
465 467 :sortable => "(SELECT COALESCE(SUM(hours), 0) FROM #{TimeEntry.table_name} WHERE #{TimeEntry.table_name}.issue_id = #{Issue.table_name}.id)",
466 468 :default_order => 'desc',
467 469 :caption => :label_spent_time
468 470 )
469 471 end
470 472
471 473 if User.current.allowed_to?(:set_issues_private, nil, :global => true) ||
472 474 User.current.allowed_to?(:set_own_issues_private, nil, :global => true)
473 475 @available_columns << QueryColumn.new(:is_private, :sortable => "#{Issue.table_name}.is_private")
474 476 end
475 477
476 478 disabled_fields = Tracker.disabled_core_fields(trackers).map {|field| field.sub(/_id$/, '')}
477 479 @available_columns.reject! {|column|
478 480 disabled_fields.include?(column.name.to_s)
479 481 }
480 482
481 483 @available_columns
482 484 end
483 485
484 486 def self.available_columns=(v)
485 487 self.available_columns = (v)
486 488 end
487 489
488 490 def self.add_available_column(column)
489 491 self.available_columns << (column) if column.is_a?(QueryColumn)
490 492 end
491 493
492 494 # Returns an array of columns that can be used to group the results
493 495 def groupable_columns
494 496 available_columns.select {|c| c.groupable}
495 497 end
496 498
497 499 # Returns a Hash of columns and the key for sorting
498 500 def sortable_columns
499 501 {'id' => "#{Issue.table_name}.id"}.merge(available_columns.inject({}) {|h, column|
500 502 h[column.name.to_s] = column.sortable
501 503 h
502 504 })
503 505 end
504 506
505 507 def columns
506 508 # preserve the column_names order
507 509 (has_default_columns? ? default_columns_names : column_names).collect do |name|
508 510 available_columns.find { |col| col.name == name }
509 511 end.compact
510 512 end
511 513
512 514 def default_columns_names
513 515 @default_columns_names ||= begin
514 516 default_columns = Setting.issue_list_default_columns.map(&:to_sym)
515 517
516 518 project.present? ? default_columns : [:project] | default_columns
517 519 end
518 520 end
519 521
520 522 def column_names=(names)
521 523 if names
522 524 names = names.select {|n| n.is_a?(Symbol) || !n.blank? }
523 525 names = names.collect {|n| n.is_a?(Symbol) ? n : n.to_sym }
524 526 # Set column_names to nil if default columns
525 527 if names == default_columns_names
526 528 names = nil
527 529 end
528 530 end
529 531 write_attribute(:column_names, names)
530 532 end
531 533
532 534 def has_column?(column)
533 535 column_names && column_names.include?(column.is_a?(QueryColumn) ? column.name : column)
534 536 end
535 537
536 538 def has_default_columns?
537 539 column_names.nil? || column_names.empty?
538 540 end
539 541
540 542 def sort_criteria=(arg)
541 543 c = []
542 544 if arg.is_a?(Hash)
543 545 arg = arg.keys.sort.collect {|k| arg[k]}
544 546 end
545 547 c = arg.select {|k,o| !k.to_s.blank?}.slice(0,3).collect {|k,o| [k.to_s, (o == 'desc' || o == false) ? 'desc' : 'asc']}
546 548 write_attribute(:sort_criteria, c)
547 549 end
548 550
549 551 def sort_criteria
550 552 read_attribute(:sort_criteria) || []
551 553 end
552 554
553 555 def sort_criteria_key(arg)
554 556 sort_criteria && sort_criteria[arg] && sort_criteria[arg].first
555 557 end
556 558
557 559 def sort_criteria_order(arg)
558 560 sort_criteria && sort_criteria[arg] && sort_criteria[arg].last
559 561 end
560 562
561 563 def sort_criteria_order_for(key)
562 564 sort_criteria.detect {|k, order| key.to_s == k}.try(:last)
563 565 end
564 566
565 567 # Returns the SQL sort order that should be prepended for grouping
566 568 def group_by_sort_order
567 569 if grouped? && (column = group_by_column)
568 570 order = sort_criteria_order_for(column.name) || column.default_order
569 571 column.sortable.is_a?(Array) ?
570 572 column.sortable.collect {|s| "#{s} #{order}"}.join(',') :
571 573 "#{column.sortable} #{order}"
572 574 end
573 575 end
574 576
575 577 # Returns true if the query is a grouped query
576 578 def grouped?
577 579 !group_by_column.nil?
578 580 end
579 581
580 582 def group_by_column
581 583 groupable_columns.detect {|c| c.groupable && c.name.to_s == group_by}
582 584 end
583 585
584 586 def group_by_statement
585 587 group_by_column.try(:groupable)
586 588 end
587 589
588 590 def project_statement
589 591 project_clauses = []
590 592 if project && !project.descendants.active.empty?
591 593 ids = [project.id]
592 594 if has_filter?("subproject_id")
593 595 case operator_for("subproject_id")
594 596 when '='
595 597 # include the selected subprojects
596 598 ids += values_for("subproject_id").each(&:to_i)
597 599 when '!*'
598 600 # main project only
599 601 else
600 602 # all subprojects
601 603 ids += project.descendants.collect(&:id)
602 604 end
603 605 elsif Setting.display_subprojects_issues?
604 606 ids += project.descendants.collect(&:id)
605 607 end
606 608 project_clauses << "#{Project.table_name}.id IN (%s)" % ids.join(',')
607 609 elsif project
608 610 project_clauses << "#{Project.table_name}.id = %d" % project.id
609 611 end
610 612 project_clauses.any? ? project_clauses.join(' AND ') : nil
611 613 end
612 614
613 615 def statement
614 616 # filters clauses
615 617 filters_clauses = []
616 618 filters.each_key do |field|
617 619 next if field == "subproject_id"
618 620 v = values_for(field).clone
619 621 next unless v and !v.empty?
620 622 operator = operator_for(field)
621 623
622 624 # "me" value subsitution
623 625 if %w(assigned_to_id author_id watcher_id).include?(field)
624 626 if v.delete("me")
625 627 if User.current.logged?
626 628 v.push(User.current.id.to_s)
627 629 v += User.current.group_ids.map(&:to_s) if field == 'assigned_to_id'
628 630 else
629 631 v.push("0")
630 632 end
631 633 end
632 634 end
633 635
634 636 if field == 'project_id'
635 637 if v.delete('mine')
636 638 v += User.current.memberships.map(&:project_id).map(&:to_s)
637 639 end
638 640 end
639 641
640 642 if field =~ /cf_(\d+)$/
641 643 # custom field
642 644 filters_clauses << sql_for_custom_field(field, operator, v, $1)
643 645 elsif respond_to?("sql_for_#{field}_field")
644 646 # specific statement
645 647 filters_clauses << send("sql_for_#{field}_field", field, operator, v)
646 648 else
647 649 # regular field
648 650 filters_clauses << '(' + sql_for_field(field, operator, v, Issue.table_name, field) + ')'
649 651 end
650 652 end if filters and valid?
651 653
652 654 filters_clauses << project_statement
653 655 filters_clauses.reject!(&:blank?)
654 656
655 657 filters_clauses.any? ? filters_clauses.join(' AND ') : nil
656 658 end
657 659
658 660 # Returns the issue count
659 661 def issue_count
660 662 Issue.visible.count(:include => [:status, :project], :conditions => statement)
661 663 rescue ::ActiveRecord::StatementInvalid => e
662 664 raise StatementInvalid.new(e.message)
663 665 end
664 666
665 667 # Returns the issue count by group or nil if query is not grouped
666 668 def issue_count_by_group
667 669 r = nil
668 670 if grouped?
669 671 begin
670 672 # Rails3 will raise an (unexpected) RecordNotFound if there's only a nil group value
671 673 r = Issue.visible.count(:group => group_by_statement, :include => [:status, :project], :conditions => statement)
672 674 rescue ActiveRecord::RecordNotFound
673 675 r = {nil => issue_count}
674 676 end
675 677 c = group_by_column
676 678 if c.is_a?(QueryCustomFieldColumn)
677 679 r = r.keys.inject({}) {|h, k| h[c.custom_field.cast_value(k)] = r[k]; h}
678 680 end
679 681 end
680 682 r
681 683 rescue ::ActiveRecord::StatementInvalid => e
682 684 raise StatementInvalid.new(e.message)
683 685 end
684 686
685 687 # Returns the issues
686 688 # Valid options are :order, :offset, :limit, :include, :conditions
687 689 def issues(options={})
688 690 order_option = [group_by_sort_order, options[:order]].reject {|s| s.blank?}.join(',')
689 691 order_option = nil if order_option.blank?
690 692
691 693 issues = Issue.visible.scoped(:conditions => options[:conditions]).find :all, :include => ([:status, :project] + (options[:include] || [])).uniq,
692 694 :conditions => statement,
693 695 :order => order_option,
694 696 :joins => joins_for_order_statement(order_option),
695 697 :limit => options[:limit],
696 698 :offset => options[:offset]
697 699
698 700 if has_column?(:spent_hours)
699 701 Issue.load_visible_spent_hours(issues)
700 702 end
701 703 if has_column?(:relations)
702 704 Issue.load_visible_relations(issues)
703 705 end
704 706 issues
705 707 rescue ::ActiveRecord::StatementInvalid => e
706 708 raise StatementInvalid.new(e.message)
707 709 end
708 710
709 711 # Returns the issues ids
710 712 def issue_ids(options={})
711 713 order_option = [group_by_sort_order, options[:order]].reject {|s| s.blank?}.join(',')
712 714 order_option = nil if order_option.blank?
713 715
714 716 Issue.visible.scoped(:conditions => options[:conditions]).scoped(:include => ([:status, :project] + (options[:include] || [])).uniq,
715 717 :conditions => statement,
716 718 :order => order_option,
717 719 :joins => joins_for_order_statement(order_option),
718 720 :limit => options[:limit],
719 721 :offset => options[:offset]).find_ids
720 722 rescue ::ActiveRecord::StatementInvalid => e
721 723 raise StatementInvalid.new(e.message)
722 724 end
723 725
724 726 # Returns the journals
725 727 # Valid options are :order, :offset, :limit
726 728 def journals(options={})
727 729 Journal.visible.find :all, :include => [:details, :user, {:issue => [:project, :author, :tracker, :status]}],
728 730 :conditions => statement,
729 731 :order => options[:order],
730 732 :limit => options[:limit],
731 733 :offset => options[:offset]
732 734 rescue ::ActiveRecord::StatementInvalid => e
733 735 raise StatementInvalid.new(e.message)
734 736 end
735 737
736 738 # Returns the versions
737 739 # Valid options are :conditions
738 740 def versions(options={})
739 741 Version.visible.scoped(:conditions => options[:conditions]).find :all, :include => :project, :conditions => project_statement
740 742 rescue ::ActiveRecord::StatementInvalid => e
741 743 raise StatementInvalid.new(e.message)
742 744 end
743 745
744 746 def sql_for_watcher_id_field(field, operator, value)
745 747 db_table = Watcher.table_name
746 748 "#{Issue.table_name}.id #{ operator == '=' ? 'IN' : 'NOT IN' } (SELECT #{db_table}.watchable_id FROM #{db_table} WHERE #{db_table}.watchable_type='Issue' AND " +
747 749 sql_for_field(field, '=', value, db_table, 'user_id') + ')'
748 750 end
749 751
750 752 def sql_for_member_of_group_field(field, operator, value)
751 753 if operator == '*' # Any group
752 754 groups = Group.all
753 755 operator = '=' # Override the operator since we want to find by assigned_to
754 756 elsif operator == "!*"
755 757 groups = Group.all
756 758 operator = '!' # Override the operator since we want to find by assigned_to
757 759 else
758 760 groups = Group.find_all_by_id(value)
759 761 end
760 762 groups ||= []
761 763
762 764 members_of_groups = groups.inject([]) {|user_ids, group|
763 765 if group && group.user_ids.present?
764 766 user_ids << group.user_ids
765 767 end
766 768 user_ids.flatten.uniq.compact
767 769 }.sort.collect(&:to_s)
768 770
769 771 '(' + sql_for_field("assigned_to_id", operator, members_of_groups, Issue.table_name, "assigned_to_id", false) + ')'
770 772 end
771 773
772 774 def sql_for_assigned_to_role_field(field, operator, value)
773 775 case operator
774 776 when "*", "!*" # Member / Not member
775 777 sw = operator == "!*" ? 'NOT' : ''
776 778 nl = operator == "!*" ? "#{Issue.table_name}.assigned_to_id IS NULL OR" : ''
777 779 "(#{nl} #{Issue.table_name}.assigned_to_id #{sw} IN (SELECT DISTINCT #{Member.table_name}.user_id FROM #{Member.table_name}" +
778 780 " WHERE #{Member.table_name}.project_id = #{Issue.table_name}.project_id))"
779 781 when "=", "!"
780 782 role_cond = value.any? ?
781 783 "#{MemberRole.table_name}.role_id IN (" + value.collect{|val| "'#{connection.quote_string(val)}'"}.join(",") + ")" :
782 784 "1=0"
783 785
784 786 sw = operator == "!" ? 'NOT' : ''
785 787 nl = operator == "!" ? "#{Issue.table_name}.assigned_to_id IS NULL OR" : ''
786 788 "(#{nl} #{Issue.table_name}.assigned_to_id #{sw} IN (SELECT DISTINCT #{Member.table_name}.user_id FROM #{Member.table_name}, #{MemberRole.table_name}" +
787 789 " WHERE #{Member.table_name}.project_id = #{Issue.table_name}.project_id AND #{Member.table_name}.id = #{MemberRole.table_name}.member_id AND #{role_cond}))"
788 790 end
789 791 end
790 792
791 793 def sql_for_is_private_field(field, operator, value)
792 794 op = (operator == "=" ? 'IN' : 'NOT IN')
793 795 va = value.map {|v| v == '0' ? connection.quoted_false : connection.quoted_true}.uniq.join(',')
794 796
795 797 "#{Issue.table_name}.is_private #{op} (#{va})"
796 798 end
797 799
798 800 def sql_for_relations(field, operator, value, options={})
799 801 relation_options = IssueRelation::TYPES[field]
800 802 return relation_options unless relation_options
801 803
802 804 relation_type = field
803 805 join_column, target_join_column = "issue_from_id", "issue_to_id"
804 806 if relation_options[:reverse] || options[:reverse]
805 807 relation_type = relation_options[:reverse] || relation_type
806 808 join_column, target_join_column = target_join_column, join_column
807 809 end
808 810
809 811 sql = case operator
810 812 when "*", "!*"
811 813 op = (operator == "*" ? 'IN' : 'NOT IN')
812 814 "#{Issue.table_name}.id #{op} (SELECT DISTINCT #{IssueRelation.table_name}.#{join_column} FROM #{IssueRelation.table_name} WHERE #{IssueRelation.table_name}.relation_type = '#{connection.quote_string(relation_type)}')"
813 815 when "=", "!"
814 816 op = (operator == "=" ? 'IN' : 'NOT IN')
815 817 "#{Issue.table_name}.id #{op} (SELECT DISTINCT #{IssueRelation.table_name}.#{join_column} FROM #{IssueRelation.table_name} WHERE #{IssueRelation.table_name}.relation_type = '#{connection.quote_string(relation_type)}' AND #{IssueRelation.table_name}.#{target_join_column} = #{value.first.to_i})"
816 818 when "=p", "=!p", "!p"
817 819 op = (operator == "!p" ? 'NOT IN' : 'IN')
818 820 comp = (operator == "=!p" ? '<>' : '=')
819 821 "#{Issue.table_name}.id #{op} (SELECT DISTINCT #{IssueRelation.table_name}.#{join_column} FROM #{IssueRelation.table_name}, #{Issue.table_name} relissues WHERE #{IssueRelation.table_name}.relation_type = '#{connection.quote_string(relation_type)}' AND #{IssueRelation.table_name}.#{target_join_column} = relissues.id AND relissues.project_id #{comp} #{value.first.to_i})"
820 822 end
821 823
822 824 if relation_options[:sym] == field && !options[:reverse]
823 825 sqls = [sql, sql_for_relations(field, operator, value, :reverse => true)]
824 826 sqls.join(["!", "!*", "!p"].include?(operator) ? " AND " : " OR ")
825 827 else
826 828 sql
827 829 end
828 830 end
829 831
830 832 IssueRelation::TYPES.keys.each do |relation_type|
831 833 alias_method "sql_for_#{relation_type}_field".to_sym, :sql_for_relations
832 834 end
833 835
834 836 private
835 837
836 838 def sql_for_custom_field(field, operator, value, custom_field_id)
837 839 db_table = CustomValue.table_name
838 840 db_field = 'value'
839 841 filter = @available_filters[field]
840 842 return nil unless filter
841 843 if filter[:format] == 'user'
842 844 if value.delete('me')
843 845 value.push User.current.id.to_s
844 846 end
845 847 end
846 848 not_in = nil
847 849 if operator == '!'
848 850 # Makes ! operator work for custom fields with multiple values
849 851 operator = '='
850 852 not_in = 'NOT'
851 853 end
852 854 customized_key = "id"
853 855 customized_class = Issue
854 856 if field =~ /^(.+)\.cf_/
855 857 assoc = $1
856 858 customized_key = "#{assoc}_id"
857 859 customized_class = Issue.reflect_on_association(assoc.to_sym).klass.base_class rescue nil
858 860 raise "Unknown Issue association #{assoc}" unless customized_class
859 861 end
860 862 "#{Issue.table_name}.#{customized_key} #{not_in} IN (SELECT #{customized_class.table_name}.id FROM #{customized_class.table_name} LEFT OUTER JOIN #{db_table} ON #{db_table}.customized_type='#{customized_class}' AND #{db_table}.customized_id=#{customized_class.table_name}.id AND #{db_table}.custom_field_id=#{custom_field_id} WHERE " +
861 863 sql_for_field(field, operator, value, db_table, db_field, true) + ')'
862 864 end
863 865
864 866 # Helper method to generate the WHERE sql for a +field+, +operator+ and a +value+
865 867 def sql_for_field(field, operator, value, db_table, db_field, is_custom_filter=false)
866 868 sql = ''
867 869 case operator
868 870 when "="
869 871 if value.any?
870 872 case type_for(field)
871 873 when :date, :date_past
872 874 sql = date_clause(db_table, db_field, (Date.parse(value.first) rescue nil), (Date.parse(value.first) rescue nil))
873 875 when :integer
874 876 if is_custom_filter
875 877 sql = "(#{db_table}.#{db_field} <> '' AND CAST(#{db_table}.#{db_field} AS decimal(60,3)) = #{value.first.to_i})"
876 878 else
877 879 sql = "#{db_table}.#{db_field} = #{value.first.to_i}"
878 880 end
879 881 when :float
880 882 if is_custom_filter
881 883 sql = "(#{db_table}.#{db_field} <> '' AND CAST(#{db_table}.#{db_field} AS decimal(60,3)) BETWEEN #{value.first.to_f - 1e-5} AND #{value.first.to_f + 1e-5})"
882 884 else
883 885 sql = "#{db_table}.#{db_field} BETWEEN #{value.first.to_f - 1e-5} AND #{value.first.to_f + 1e-5}"
884 886 end
885 887 else
886 888 sql = "#{db_table}.#{db_field} IN (" + value.collect{|val| "'#{connection.quote_string(val)}'"}.join(",") + ")"
887 889 end
888 890 else
889 891 # IN an empty set
890 892 sql = "1=0"
891 893 end
892 894 when "!"
893 895 if value.any?
894 896 sql = "(#{db_table}.#{db_field} IS NULL OR #{db_table}.#{db_field} NOT IN (" + value.collect{|val| "'#{connection.quote_string(val)}'"}.join(",") + "))"
895 897 else
896 898 # NOT IN an empty set
897 899 sql = "1=1"
898 900 end
899 901 when "!*"
900 902 sql = "#{db_table}.#{db_field} IS NULL"
901 903 sql << " OR #{db_table}.#{db_field} = ''" if is_custom_filter
902 904 when "*"
903 905 sql = "#{db_table}.#{db_field} IS NOT NULL"
904 906 sql << " AND #{db_table}.#{db_field} <> ''" if is_custom_filter
905 907 when ">="
906 908 if [:date, :date_past].include?(type_for(field))
907 909 sql = date_clause(db_table, db_field, (Date.parse(value.first) rescue nil), nil)
908 910 else
909 911 if is_custom_filter
910 912 sql = "(#{db_table}.#{db_field} <> '' AND CAST(#{db_table}.#{db_field} AS decimal(60,3)) >= #{value.first.to_f})"
911 913 else
912 914 sql = "#{db_table}.#{db_field} >= #{value.first.to_f}"
913 915 end
914 916 end
915 917 when "<="
916 918 if [:date, :date_past].include?(type_for(field))
917 919 sql = date_clause(db_table, db_field, nil, (Date.parse(value.first) rescue nil))
918 920 else
919 921 if is_custom_filter
920 922 sql = "(#{db_table}.#{db_field} <> '' AND CAST(#{db_table}.#{db_field} AS decimal(60,3)) <= #{value.first.to_f})"
921 923 else
922 924 sql = "#{db_table}.#{db_field} <= #{value.first.to_f}"
923 925 end
924 926 end
925 927 when "><"
926 928 if [:date, :date_past].include?(type_for(field))
927 929 sql = date_clause(db_table, db_field, (Date.parse(value[0]) rescue nil), (Date.parse(value[1]) rescue nil))
928 930 else
929 931 if is_custom_filter
930 932 sql = "(#{db_table}.#{db_field} <> '' AND CAST(#{db_table}.#{db_field} AS decimal(60,3)) BETWEEN #{value[0].to_f} AND #{value[1].to_f})"
931 933 else
932 934 sql = "#{db_table}.#{db_field} BETWEEN #{value[0].to_f} AND #{value[1].to_f}"
933 935 end
934 936 end
935 937 when "o"
936 938 sql = "#{Issue.table_name}.status_id IN (SELECT id FROM #{IssueStatus.table_name} WHERE is_closed=#{connection.quoted_false})" if field == "status_id"
937 939 when "c"
938 940 sql = "#{Issue.table_name}.status_id IN (SELECT id FROM #{IssueStatus.table_name} WHERE is_closed=#{connection.quoted_true})" if field == "status_id"
939 when ">t-"
941 when "><t-"
942 # between today - n days and today
940 943 sql = relative_date_clause(db_table, db_field, - value.first.to_i, 0)
944 when ">t-"
945 # >= today - n days
946 sql = relative_date_clause(db_table, db_field, - value.first.to_i, nil)
941 947 when "<t-"
948 # <= today - n days
942 949 sql = relative_date_clause(db_table, db_field, nil, - value.first.to_i)
943 950 when "t-"
951 # = n days in past
944 952 sql = relative_date_clause(db_table, db_field, - value.first.to_i, - value.first.to_i)
953 when "><t+"
954 # between today and today + n days
955 sql = relative_date_clause(db_table, db_field, 0, value.first.to_i)
945 956 when ">t+"
957 # >= today + n days
946 958 sql = relative_date_clause(db_table, db_field, value.first.to_i, nil)
947 959 when "<t+"
948 sql = relative_date_clause(db_table, db_field, 0, value.first.to_i)
960 # <= today + n days
961 sql = relative_date_clause(db_table, db_field, nil, value.first.to_i)
949 962 when "t+"
963 # = today + n days
950 964 sql = relative_date_clause(db_table, db_field, value.first.to_i, value.first.to_i)
951 965 when "t"
966 # = today
952 967 sql = relative_date_clause(db_table, db_field, 0, 0)
953 968 when "w"
969 # = this week
954 970 first_day_of_week = l(:general_first_day_of_week).to_i
955 971 day_of_week = Date.today.cwday
956 972 days_ago = (day_of_week >= first_day_of_week ? day_of_week - first_day_of_week : day_of_week + 7 - first_day_of_week)
957 973 sql = relative_date_clause(db_table, db_field, - days_ago, - days_ago + 6)
958 974 when "~"
959 975 sql = "LOWER(#{db_table}.#{db_field}) LIKE '%#{connection.quote_string(value.first.to_s.downcase)}%'"
960 976 when "!~"
961 977 sql = "LOWER(#{db_table}.#{db_field}) NOT LIKE '%#{connection.quote_string(value.first.to_s.downcase)}%'"
962 978 else
963 979 raise "Unknown query operator #{operator}"
964 980 end
965 981
966 982 return sql
967 983 end
968 984
969 985 def add_custom_fields_filters(custom_fields, assoc=nil)
970 986 return unless custom_fields.present?
971 987 @available_filters ||= {}
972 988
973 989 custom_fields.select(&:is_filter?).each do |field|
974 990 case field.field_format
975 991 when "text"
976 992 options = { :type => :text, :order => 20 }
977 993 when "list"
978 994 options = { :type => :list_optional, :values => field.possible_values, :order => 20}
979 995 when "date"
980 996 options = { :type => :date, :order => 20 }
981 997 when "bool"
982 998 options = { :type => :list, :values => [[l(:general_text_yes), "1"], [l(:general_text_no), "0"]], :order => 20 }
983 999 when "int"
984 1000 options = { :type => :integer, :order => 20 }
985 1001 when "float"
986 1002 options = { :type => :float, :order => 20 }
987 1003 when "user", "version"
988 1004 next unless project
989 1005 values = field.possible_values_options(project)
990 1006 if User.current.logged? && field.field_format == 'user'
991 1007 values.unshift ["<< #{l(:label_me)} >>", "me"]
992 1008 end
993 1009 options = { :type => :list_optional, :values => values, :order => 20}
994 1010 else
995 1011 options = { :type => :string, :order => 20 }
996 1012 end
997 1013 filter_id = "cf_#{field.id}"
998 1014 filter_name = field.name
999 1015 if assoc.present?
1000 1016 filter_id = "#{assoc}.#{filter_id}"
1001 1017 filter_name = l("label_attribute_of_#{assoc}", :name => filter_name)
1002 1018 end
1003 1019 @available_filters[filter_id] = options.merge({
1004 1020 :name => filter_name,
1005 1021 :format => field.field_format,
1006 1022 :field => field
1007 1023 })
1008 1024 end
1009 1025 end
1010 1026
1011 1027 def add_associations_custom_fields_filters(*associations)
1012 1028 fields_by_class = CustomField.where(:is_filter => true).group_by(&:class)
1013 1029 associations.each do |assoc|
1014 1030 association_klass = Issue.reflect_on_association(assoc).klass
1015 1031 fields_by_class.each do |field_class, fields|
1016 1032 if field_class.customized_class <= association_klass
1017 1033 add_custom_fields_filters(fields, assoc)
1018 1034 end
1019 1035 end
1020 1036 end
1021 1037 end
1022 1038
1023 1039 # Returns a SQL clause for a date or datetime field.
1024 1040 def date_clause(table, field, from, to)
1025 1041 s = []
1026 1042 if from
1027 1043 from_yesterday = from - 1
1028 1044 from_yesterday_time = Time.local(from_yesterday.year, from_yesterday.month, from_yesterday.day)
1029 1045 if self.class.default_timezone == :utc
1030 1046 from_yesterday_time = from_yesterday_time.utc
1031 1047 end
1032 1048 s << ("#{table}.#{field} > '%s'" % [connection.quoted_date(from_yesterday_time.end_of_day)])
1033 1049 end
1034 1050 if to
1035 1051 to_time = Time.local(to.year, to.month, to.day)
1036 1052 if self.class.default_timezone == :utc
1037 1053 to_time = to_time.utc
1038 1054 end
1039 1055 s << ("#{table}.#{field} <= '%s'" % [connection.quoted_date(to_time.end_of_day)])
1040 1056 end
1041 1057 s.join(' AND ')
1042 1058 end
1043 1059
1044 1060 # Returns a SQL clause for a date or datetime field using relative dates.
1045 1061 def relative_date_clause(table, field, days_from, days_to)
1046 1062 date_clause(table, field, (days_from ? Date.today + days_from : nil), (days_to ? Date.today + days_to : nil))
1047 1063 end
1048 1064
1049 1065 # Additional joins required for the given sort options
1050 1066 def joins_for_order_statement(order_options)
1051 1067 joins = []
1052 1068
1053 1069 if order_options
1054 1070 if order_options.include?('authors')
1055 1071 joins << "LEFT OUTER JOIN #{User.table_name} authors ON authors.id = #{Issue.table_name}.author_id"
1056 1072 end
1057 1073 order_options.scan(/cf_\d+/).uniq.each do |name|
1058 1074 column = available_columns.detect {|c| c.name.to_s == name}
1059 1075 join = column && column.custom_field.join_for_order_statement
1060 1076 if join
1061 1077 joins << join
1062 1078 end
1063 1079 end
1064 1080 end
1065 1081
1066 1082 joins.any? ? joins.join(' ') : nil
1067 1083 end
1068 1084 end
@@ -1,1079 +1,1081
1 1 en:
2 2 # Text direction: Left-to-Right (ltr) or Right-to-Left (rtl)
3 3 direction: ltr
4 4 date:
5 5 formats:
6 6 # Use the strftime parameters for formats.
7 7 # When no format has been given, it uses default.
8 8 # You can provide other formats here if you like!
9 9 default: "%m/%d/%Y"
10 10 short: "%b %d"
11 11 long: "%B %d, %Y"
12 12
13 13 day_names: [Sunday, Monday, Tuesday, Wednesday, Thursday, Friday, Saturday]
14 14 abbr_day_names: [Sun, Mon, Tue, Wed, Thu, Fri, Sat]
15 15
16 16 # Don't forget the nil at the beginning; there's no such thing as a 0th month
17 17 month_names: [~, January, February, March, April, May, June, July, August, September, October, November, December]
18 18 abbr_month_names: [~, Jan, Feb, Mar, Apr, May, Jun, Jul, Aug, Sep, Oct, Nov, Dec]
19 19 # Used in date_select and datime_select.
20 20 order:
21 21 - :year
22 22 - :month
23 23 - :day
24 24
25 25 time:
26 26 formats:
27 27 default: "%m/%d/%Y %I:%M %p"
28 28 time: "%I:%M %p"
29 29 short: "%d %b %H:%M"
30 30 long: "%B %d, %Y %H:%M"
31 31 am: "am"
32 32 pm: "pm"
33 33
34 34 datetime:
35 35 distance_in_words:
36 36 half_a_minute: "half a minute"
37 37 less_than_x_seconds:
38 38 one: "less than 1 second"
39 39 other: "less than %{count} seconds"
40 40 x_seconds:
41 41 one: "1 second"
42 42 other: "%{count} seconds"
43 43 less_than_x_minutes:
44 44 one: "less than a minute"
45 45 other: "less than %{count} minutes"
46 46 x_minutes:
47 47 one: "1 minute"
48 48 other: "%{count} minutes"
49 49 about_x_hours:
50 50 one: "about 1 hour"
51 51 other: "about %{count} hours"
52 52 x_hours:
53 53 one: "1 hour"
54 54 other: "%{count} hours"
55 55 x_days:
56 56 one: "1 day"
57 57 other: "%{count} days"
58 58 about_x_months:
59 59 one: "about 1 month"
60 60 other: "about %{count} months"
61 61 x_months:
62 62 one: "1 month"
63 63 other: "%{count} months"
64 64 about_x_years:
65 65 one: "about 1 year"
66 66 other: "about %{count} years"
67 67 over_x_years:
68 68 one: "over 1 year"
69 69 other: "over %{count} years"
70 70 almost_x_years:
71 71 one: "almost 1 year"
72 72 other: "almost %{count} years"
73 73
74 74 number:
75 75 format:
76 76 separator: "."
77 77 delimiter: ""
78 78 precision: 3
79 79
80 80 human:
81 81 format:
82 82 delimiter: ""
83 83 precision: 3
84 84 storage_units:
85 85 format: "%n %u"
86 86 units:
87 87 byte:
88 88 one: "Byte"
89 89 other: "Bytes"
90 90 kb: "KB"
91 91 mb: "MB"
92 92 gb: "GB"
93 93 tb: "TB"
94 94
95 95 # Used in array.to_sentence.
96 96 support:
97 97 array:
98 98 sentence_connector: "and"
99 99 skip_last_comma: false
100 100
101 101 activerecord:
102 102 errors:
103 103 template:
104 104 header:
105 105 one: "1 error prohibited this %{model} from being saved"
106 106 other: "%{count} errors prohibited this %{model} from being saved"
107 107 messages:
108 108 inclusion: "is not included in the list"
109 109 exclusion: "is reserved"
110 110 invalid: "is invalid"
111 111 confirmation: "doesn't match confirmation"
112 112 accepted: "must be accepted"
113 113 empty: "can't be empty"
114 114 blank: "can't be blank"
115 115 too_long: "is too long (maximum is %{count} characters)"
116 116 too_short: "is too short (minimum is %{count} characters)"
117 117 wrong_length: "is the wrong length (should be %{count} characters)"
118 118 taken: "has already been taken"
119 119 not_a_number: "is not a number"
120 120 not_a_date: "is not a valid date"
121 121 greater_than: "must be greater than %{count}"
122 122 greater_than_or_equal_to: "must be greater than or equal to %{count}"
123 123 equal_to: "must be equal to %{count}"
124 124 less_than: "must be less than %{count}"
125 125 less_than_or_equal_to: "must be less than or equal to %{count}"
126 126 odd: "must be odd"
127 127 even: "must be even"
128 128 greater_than_start_date: "must be greater than start date"
129 129 not_same_project: "doesn't belong to the same project"
130 130 circular_dependency: "This relation would create a circular dependency"
131 131 cant_link_an_issue_with_a_descendant: "An issue cannot be linked to one of its subtasks"
132 132
133 133 actionview_instancetag_blank_option: Please select
134 134
135 135 general_text_No: 'No'
136 136 general_text_Yes: 'Yes'
137 137 general_text_no: 'no'
138 138 general_text_yes: 'yes'
139 139 general_lang_name: 'English'
140 140 general_csv_separator: ','
141 141 general_csv_decimal_separator: '.'
142 142 general_csv_encoding: ISO-8859-1
143 143 general_pdf_encoding: UTF-8
144 144 general_first_day_of_week: '7'
145 145
146 146 notice_account_updated: Account was successfully updated.
147 147 notice_account_invalid_creditentials: Invalid user or password
148 148 notice_account_password_updated: Password was successfully updated.
149 149 notice_account_wrong_password: Wrong password
150 150 notice_account_register_done: Account was successfully created. To activate your account, click on the link that was emailed to you.
151 151 notice_account_unknown_email: Unknown user.
152 152 notice_can_t_change_password: This account uses an external authentication source. Impossible to change the password.
153 153 notice_account_lost_email_sent: An email with instructions to choose a new password has been sent to you.
154 154 notice_account_activated: Your account has been activated. You can now log in.
155 155 notice_successful_create: Successful creation.
156 156 notice_successful_update: Successful update.
157 157 notice_successful_delete: Successful deletion.
158 158 notice_successful_connection: Successful connection.
159 159 notice_file_not_found: The page you were trying to access doesn't exist or has been removed.
160 160 notice_locking_conflict: Data has been updated by another user.
161 161 notice_not_authorized: You are not authorized to access this page.
162 162 notice_not_authorized_archived_project: The project you're trying to access has been archived.
163 163 notice_email_sent: "An email was sent to %{value}"
164 164 notice_email_error: "An error occurred while sending mail (%{value})"
165 165 notice_feeds_access_key_reseted: Your RSS access key was reset.
166 166 notice_api_access_key_reseted: Your API access key was reset.
167 167 notice_failed_to_save_issues: "Failed to save %{count} issue(s) on %{total} selected: %{ids}."
168 168 notice_failed_to_save_time_entries: "Failed to save %{count} time entrie(s) on %{total} selected: %{ids}."
169 169 notice_failed_to_save_members: "Failed to save member(s): %{errors}."
170 170 notice_no_issue_selected: "No issue is selected! Please, check the issues you want to edit."
171 171 notice_account_pending: "Your account was created and is now pending administrator approval."
172 172 notice_default_data_loaded: Default configuration successfully loaded.
173 173 notice_unable_delete_version: Unable to delete version.
174 174 notice_unable_delete_time_entry: Unable to delete time log entry.
175 175 notice_issue_done_ratios_updated: Issue done ratios updated.
176 176 notice_gantt_chart_truncated: "The chart was truncated because it exceeds the maximum number of items that can be displayed (%{max})"
177 177 notice_issue_successful_create: "Issue %{id} created."
178 178 notice_issue_update_conflict: "The issue has been updated by an other user while you were editing it."
179 179 notice_account_deleted: "Your account has been permanently deleted."
180 180 notice_user_successful_create: "User %{id} created."
181 181
182 182 error_can_t_load_default_data: "Default configuration could not be loaded: %{value}"
183 183 error_scm_not_found: "The entry or revision was not found in the repository."
184 184 error_scm_command_failed: "An error occurred when trying to access the repository: %{value}"
185 185 error_scm_annotate: "The entry does not exist or cannot be annotated."
186 186 error_scm_annotate_big_text_file: "The entry cannot be annotated, as it exceeds the maximum text file size."
187 187 error_issue_not_found_in_project: 'The issue was not found or does not belong to this project'
188 188 error_no_tracker_in_project: 'No tracker is associated to this project. Please check the Project settings.'
189 189 error_no_default_issue_status: 'No default issue status is defined. Please check your configuration (Go to "Administration -> Issue statuses").'
190 190 error_can_not_delete_custom_field: Unable to delete custom field
191 191 error_can_not_delete_tracker: "This tracker contains issues and cannot be deleted."
192 192 error_can_not_remove_role: "This role is in use and cannot be deleted."
193 193 error_can_not_reopen_issue_on_closed_version: 'An issue assigned to a closed version cannot be reopened'
194 194 error_can_not_archive_project: This project cannot be archived
195 195 error_issue_done_ratios_not_updated: "Issue done ratios not updated."
196 196 error_workflow_copy_source: 'Please select a source tracker or role'
197 197 error_workflow_copy_target: 'Please select target tracker(s) and role(s)'
198 198 error_unable_delete_issue_status: 'Unable to delete issue status'
199 199 error_unable_to_connect: "Unable to connect (%{value})"
200 200 error_attachment_too_big: "This file cannot be uploaded because it exceeds the maximum allowed file size (%{max_size})"
201 201 error_session_expired: "Your session has expired. Please login again."
202 202 warning_attachments_not_saved: "%{count} file(s) could not be saved."
203 203
204 204 mail_subject_lost_password: "Your %{value} password"
205 205 mail_body_lost_password: 'To change your password, click on the following link:'
206 206 mail_subject_register: "Your %{value} account activation"
207 207 mail_body_register: 'To activate your account, click on the following link:'
208 208 mail_body_account_information_external: "You can use your %{value} account to log in."
209 209 mail_body_account_information: Your account information
210 210 mail_subject_account_activation_request: "%{value} account activation request"
211 211 mail_body_account_activation_request: "A new user (%{value}) has registered. The account is pending your approval:"
212 212 mail_subject_reminder: "%{count} issue(s) due in the next %{days} days"
213 213 mail_body_reminder: "%{count} issue(s) that are assigned to you are due in the next %{days} days:"
214 214 mail_subject_wiki_content_added: "'%{id}' wiki page has been added"
215 215 mail_body_wiki_content_added: "The '%{id}' wiki page has been added by %{author}."
216 216 mail_subject_wiki_content_updated: "'%{id}' wiki page has been updated"
217 217 mail_body_wiki_content_updated: "The '%{id}' wiki page has been updated by %{author}."
218 218
219 219 gui_validation_error: 1 error
220 220 gui_validation_error_plural: "%{count} errors"
221 221
222 222 field_name: Name
223 223 field_description: Description
224 224 field_summary: Summary
225 225 field_is_required: Required
226 226 field_firstname: First name
227 227 field_lastname: Last name
228 228 field_mail: Email
229 229 field_filename: File
230 230 field_filesize: Size
231 231 field_downloads: Downloads
232 232 field_author: Author
233 233 field_created_on: Created
234 234 field_updated_on: Updated
235 235 field_field_format: Format
236 236 field_is_for_all: For all projects
237 237 field_possible_values: Possible values
238 238 field_regexp: Regular expression
239 239 field_min_length: Minimum length
240 240 field_max_length: Maximum length
241 241 field_value: Value
242 242 field_category: Category
243 243 field_title: Title
244 244 field_project: Project
245 245 field_issue: Issue
246 246 field_status: Status
247 247 field_notes: Notes
248 248 field_is_closed: Issue closed
249 249 field_is_default: Default value
250 250 field_tracker: Tracker
251 251 field_subject: Subject
252 252 field_due_date: Due date
253 253 field_assigned_to: Assignee
254 254 field_priority: Priority
255 255 field_fixed_version: Target version
256 256 field_user: User
257 257 field_principal: Principal
258 258 field_role: Role
259 259 field_homepage: Homepage
260 260 field_is_public: Public
261 261 field_parent: Subproject of
262 262 field_is_in_roadmap: Issues displayed in roadmap
263 263 field_login: Login
264 264 field_mail_notification: Email notifications
265 265 field_admin: Administrator
266 266 field_last_login_on: Last connection
267 267 field_language: Language
268 268 field_effective_date: Date
269 269 field_password: Password
270 270 field_new_password: New password
271 271 field_password_confirmation: Confirmation
272 272 field_version: Version
273 273 field_type: Type
274 274 field_host: Host
275 275 field_port: Port
276 276 field_account: Account
277 277 field_base_dn: Base DN
278 278 field_attr_login: Login attribute
279 279 field_attr_firstname: Firstname attribute
280 280 field_attr_lastname: Lastname attribute
281 281 field_attr_mail: Email attribute
282 282 field_onthefly: On-the-fly user creation
283 283 field_start_date: Start date
284 284 field_done_ratio: "% Done"
285 285 field_auth_source: Authentication mode
286 286 field_hide_mail: Hide my email address
287 287 field_comments: Comment
288 288 field_url: URL
289 289 field_start_page: Start page
290 290 field_subproject: Subproject
291 291 field_hours: Hours
292 292 field_activity: Activity
293 293 field_spent_on: Date
294 294 field_identifier: Identifier
295 295 field_is_filter: Used as a filter
296 296 field_issue_to: Related issue
297 297 field_delay: Delay
298 298 field_assignable: Issues can be assigned to this role
299 299 field_redirect_existing_links: Redirect existing links
300 300 field_estimated_hours: Estimated time
301 301 field_column_names: Columns
302 302 field_time_entries: Log time
303 303 field_time_zone: Time zone
304 304 field_searchable: Searchable
305 305 field_default_value: Default value
306 306 field_comments_sorting: Display comments
307 307 field_parent_title: Parent page
308 308 field_editable: Editable
309 309 field_watcher: Watcher
310 310 field_identity_url: OpenID URL
311 311 field_content: Content
312 312 field_group_by: Group results by
313 313 field_sharing: Sharing
314 314 field_parent_issue: Parent task
315 315 field_member_of_group: "Assignee's group"
316 316 field_assigned_to_role: "Assignee's role"
317 317 field_text: Text field
318 318 field_visible: Visible
319 319 field_warn_on_leaving_unsaved: "Warn me when leaving a page with unsaved text"
320 320 field_issues_visibility: Issues visibility
321 321 field_is_private: Private
322 322 field_commit_logs_encoding: Commit messages encoding
323 323 field_scm_path_encoding: Path encoding
324 324 field_path_to_repository: Path to repository
325 325 field_root_directory: Root directory
326 326 field_cvsroot: CVSROOT
327 327 field_cvs_module: Module
328 328 field_repository_is_default: Main repository
329 329 field_multiple: Multiple values
330 330 field_auth_source_ldap_filter: LDAP filter
331 331 field_core_fields: Standard fields
332 332 field_timeout: "Timeout (in seconds)"
333 333 field_board_parent: Parent forum
334 334 field_private_notes: Private notes
335 335
336 336 setting_app_title: Application title
337 337 setting_app_subtitle: Application subtitle
338 338 setting_welcome_text: Welcome text
339 339 setting_default_language: Default language
340 340 setting_login_required: Authentication required
341 341 setting_self_registration: Self-registration
342 342 setting_attachment_max_size: Maximum attachment size
343 343 setting_issues_export_limit: Issues export limit
344 344 setting_mail_from: Emission email address
345 345 setting_bcc_recipients: Blind carbon copy recipients (bcc)
346 346 setting_plain_text_mail: Plain text mail (no HTML)
347 347 setting_host_name: Host name and path
348 348 setting_text_formatting: Text formatting
349 349 setting_wiki_compression: Wiki history compression
350 350 setting_feeds_limit: Maximum number of items in Atom feeds
351 351 setting_default_projects_public: New projects are public by default
352 352 setting_autofetch_changesets: Fetch commits automatically
353 353 setting_sys_api_enabled: Enable WS for repository management
354 354 setting_commit_ref_keywords: Referencing keywords
355 355 setting_commit_fix_keywords: Fixing keywords
356 356 setting_autologin: Autologin
357 357 setting_date_format: Date format
358 358 setting_time_format: Time format
359 359 setting_cross_project_issue_relations: Allow cross-project issue relations
360 360 setting_cross_project_subtasks: Allow cross-project subtasks
361 361 setting_issue_list_default_columns: Default columns displayed on the issue list
362 362 setting_repositories_encodings: Attachments and repositories encodings
363 363 setting_emails_header: Emails header
364 364 setting_emails_footer: Emails footer
365 365 setting_protocol: Protocol
366 366 setting_per_page_options: Objects per page options
367 367 setting_user_format: Users display format
368 368 setting_activity_days_default: Days displayed on project activity
369 369 setting_display_subprojects_issues: Display subprojects issues on main projects by default
370 370 setting_enabled_scm: Enabled SCM
371 371 setting_mail_handler_body_delimiters: "Truncate emails after one of these lines"
372 372 setting_mail_handler_api_enabled: Enable WS for incoming emails
373 373 setting_mail_handler_api_key: API key
374 374 setting_sequential_project_identifiers: Generate sequential project identifiers
375 375 setting_gravatar_enabled: Use Gravatar user icons
376 376 setting_gravatar_default: Default Gravatar image
377 377 setting_diff_max_lines_displayed: Maximum number of diff lines displayed
378 378 setting_file_max_size_displayed: Maximum size of text files displayed inline
379 379 setting_repository_log_display_limit: Maximum number of revisions displayed on file log
380 380 setting_openid: Allow OpenID login and registration
381 381 setting_password_min_length: Minimum password length
382 382 setting_new_project_user_role_id: Role given to a non-admin user who creates a project
383 383 setting_default_projects_modules: Default enabled modules for new projects
384 384 setting_issue_done_ratio: Calculate the issue done ratio with
385 385 setting_issue_done_ratio_issue_field: Use the issue field
386 386 setting_issue_done_ratio_issue_status: Use the issue status
387 387 setting_start_of_week: Start calendars on
388 388 setting_rest_api_enabled: Enable REST web service
389 389 setting_cache_formatted_text: Cache formatted text
390 390 setting_default_notification_option: Default notification option
391 391 setting_commit_logtime_enabled: Enable time logging
392 392 setting_commit_logtime_activity_id: Activity for logged time
393 393 setting_gantt_items_limit: Maximum number of items displayed on the gantt chart
394 394 setting_issue_group_assignment: Allow issue assignment to groups
395 395 setting_default_issue_start_date_to_creation_date: Use current date as start date for new issues
396 396 setting_commit_cross_project_ref: Allow issues of all the other projects to be referenced and fixed
397 397 setting_unsubscribe: Allow users to delete their own account
398 398 setting_session_lifetime: Session maximum lifetime
399 399 setting_session_timeout: Session inactivity timeout
400 400 setting_thumbnails_enabled: Display attachment thumbnails
401 401 setting_thumbnails_size: Thumbnails size (in pixels)
402 402 setting_non_working_week_days: Non-working days
403 403
404 404 permission_add_project: Create project
405 405 permission_add_subprojects: Create subprojects
406 406 permission_edit_project: Edit project
407 407 permission_close_project: Close / reopen the project
408 408 permission_select_project_modules: Select project modules
409 409 permission_manage_members: Manage members
410 410 permission_manage_project_activities: Manage project activities
411 411 permission_manage_versions: Manage versions
412 412 permission_manage_categories: Manage issue categories
413 413 permission_view_issues: View Issues
414 414 permission_add_issues: Add issues
415 415 permission_edit_issues: Edit issues
416 416 permission_manage_issue_relations: Manage issue relations
417 417 permission_set_issues_private: Set issues public or private
418 418 permission_set_own_issues_private: Set own issues public or private
419 419 permission_add_issue_notes: Add notes
420 420 permission_edit_issue_notes: Edit notes
421 421 permission_edit_own_issue_notes: Edit own notes
422 422 permission_view_private_notes: View private notes
423 423 permission_set_notes_private: Set notes as private
424 424 permission_move_issues: Move issues
425 425 permission_delete_issues: Delete issues
426 426 permission_manage_public_queries: Manage public queries
427 427 permission_save_queries: Save queries
428 428 permission_view_gantt: View gantt chart
429 429 permission_view_calendar: View calendar
430 430 permission_view_issue_watchers: View watchers list
431 431 permission_add_issue_watchers: Add watchers
432 432 permission_delete_issue_watchers: Delete watchers
433 433 permission_log_time: Log spent time
434 434 permission_view_time_entries: View spent time
435 435 permission_edit_time_entries: Edit time logs
436 436 permission_edit_own_time_entries: Edit own time logs
437 437 permission_manage_news: Manage news
438 438 permission_comment_news: Comment news
439 439 permission_manage_documents: Manage documents
440 440 permission_view_documents: View documents
441 441 permission_manage_files: Manage files
442 442 permission_view_files: View files
443 443 permission_manage_wiki: Manage wiki
444 444 permission_rename_wiki_pages: Rename wiki pages
445 445 permission_delete_wiki_pages: Delete wiki pages
446 446 permission_view_wiki_pages: View wiki
447 447 permission_view_wiki_edits: View wiki history
448 448 permission_edit_wiki_pages: Edit wiki pages
449 449 permission_delete_wiki_pages_attachments: Delete attachments
450 450 permission_protect_wiki_pages: Protect wiki pages
451 451 permission_manage_repository: Manage repository
452 452 permission_browse_repository: Browse repository
453 453 permission_view_changesets: View changesets
454 454 permission_commit_access: Commit access
455 455 permission_manage_boards: Manage forums
456 456 permission_view_messages: View messages
457 457 permission_add_messages: Post messages
458 458 permission_edit_messages: Edit messages
459 459 permission_edit_own_messages: Edit own messages
460 460 permission_delete_messages: Delete messages
461 461 permission_delete_own_messages: Delete own messages
462 462 permission_export_wiki_pages: Export wiki pages
463 463 permission_manage_subtasks: Manage subtasks
464 464 permission_manage_related_issues: Manage related issues
465 465
466 466 project_module_issue_tracking: Issue tracking
467 467 project_module_time_tracking: Time tracking
468 468 project_module_news: News
469 469 project_module_documents: Documents
470 470 project_module_files: Files
471 471 project_module_wiki: Wiki
472 472 project_module_repository: Repository
473 473 project_module_boards: Forums
474 474 project_module_calendar: Calendar
475 475 project_module_gantt: Gantt
476 476
477 477 label_user: User
478 478 label_user_plural: Users
479 479 label_user_new: New user
480 480 label_user_anonymous: Anonymous
481 481 label_project: Project
482 482 label_project_new: New project
483 483 label_project_plural: Projects
484 484 label_x_projects:
485 485 zero: no projects
486 486 one: 1 project
487 487 other: "%{count} projects"
488 488 label_project_all: All Projects
489 489 label_project_latest: Latest projects
490 490 label_issue: Issue
491 491 label_issue_new: New issue
492 492 label_issue_plural: Issues
493 493 label_issue_view_all: View all issues
494 494 label_issues_by: "Issues by %{value}"
495 495 label_issue_added: Issue added
496 496 label_issue_updated: Issue updated
497 497 label_issue_note_added: Note added
498 498 label_issue_status_updated: Status updated
499 499 label_issue_priority_updated: Priority updated
500 500 label_document: Document
501 501 label_document_new: New document
502 502 label_document_plural: Documents
503 503 label_document_added: Document added
504 504 label_role: Role
505 505 label_role_plural: Roles
506 506 label_role_new: New role
507 507 label_role_and_permissions: Roles and permissions
508 508 label_role_anonymous: Anonymous
509 509 label_role_non_member: Non member
510 510 label_member: Member
511 511 label_member_new: New member
512 512 label_member_plural: Members
513 513 label_tracker: Tracker
514 514 label_tracker_plural: Trackers
515 515 label_tracker_new: New tracker
516 516 label_workflow: Workflow
517 517 label_issue_status: Issue status
518 518 label_issue_status_plural: Issue statuses
519 519 label_issue_status_new: New status
520 520 label_issue_category: Issue category
521 521 label_issue_category_plural: Issue categories
522 522 label_issue_category_new: New category
523 523 label_custom_field: Custom field
524 524 label_custom_field_plural: Custom fields
525 525 label_custom_field_new: New custom field
526 526 label_enumerations: Enumerations
527 527 label_enumeration_new: New value
528 528 label_information: Information
529 529 label_information_plural: Information
530 530 label_please_login: Please log in
531 531 label_register: Register
532 532 label_login_with_open_id_option: or login with OpenID
533 533 label_password_lost: Lost password
534 534 label_home: Home
535 535 label_my_page: My page
536 536 label_my_account: My account
537 537 label_my_projects: My projects
538 538 label_my_page_block: My page block
539 539 label_administration: Administration
540 540 label_login: Sign in
541 541 label_logout: Sign out
542 542 label_help: Help
543 543 label_reported_issues: Reported issues
544 544 label_assigned_to_me_issues: Issues assigned to me
545 545 label_last_login: Last connection
546 546 label_registered_on: Registered on
547 547 label_activity: Activity
548 548 label_overall_activity: Overall activity
549 549 label_user_activity: "%{value}'s activity"
550 550 label_new: New
551 551 label_logged_as: Logged in as
552 552 label_environment: Environment
553 553 label_authentication: Authentication
554 554 label_auth_source: Authentication mode
555 555 label_auth_source_new: New authentication mode
556 556 label_auth_source_plural: Authentication modes
557 557 label_subproject_plural: Subprojects
558 558 label_subproject_new: New subproject
559 559 label_and_its_subprojects: "%{value} and its subprojects"
560 560 label_min_max_length: Min - Max length
561 561 label_list: List
562 562 label_date: Date
563 563 label_integer: Integer
564 564 label_float: Float
565 565 label_boolean: Boolean
566 566 label_string: Text
567 567 label_text: Long text
568 568 label_attribute: Attribute
569 569 label_attribute_plural: Attributes
570 570 label_download: "%{count} Download"
571 571 label_download_plural: "%{count} Downloads"
572 572 label_no_data: No data to display
573 573 label_change_status: Change status
574 574 label_history: History
575 575 label_attachment: File
576 576 label_attachment_new: New file
577 577 label_attachment_delete: Delete file
578 578 label_attachment_plural: Files
579 579 label_file_added: File added
580 580 label_report: Report
581 581 label_report_plural: Reports
582 582 label_news: News
583 583 label_news_new: Add news
584 584 label_news_plural: News
585 585 label_news_latest: Latest news
586 586 label_news_view_all: View all news
587 587 label_news_added: News added
588 588 label_news_comment_added: Comment added to a news
589 589 label_settings: Settings
590 590 label_overview: Overview
591 591 label_version: Version
592 592 label_version_new: New version
593 593 label_version_plural: Versions
594 594 label_close_versions: Close completed versions
595 595 label_confirmation: Confirmation
596 596 label_export_to: 'Also available in:'
597 597 label_read: Read...
598 598 label_public_projects: Public projects
599 599 label_open_issues: open
600 600 label_open_issues_plural: open
601 601 label_closed_issues: closed
602 602 label_closed_issues_plural: closed
603 603 label_x_open_issues_abbr_on_total:
604 604 zero: 0 open / %{total}
605 605 one: 1 open / %{total}
606 606 other: "%{count} open / %{total}"
607 607 label_x_open_issues_abbr:
608 608 zero: 0 open
609 609 one: 1 open
610 610 other: "%{count} open"
611 611 label_x_closed_issues_abbr:
612 612 zero: 0 closed
613 613 one: 1 closed
614 614 other: "%{count} closed"
615 615 label_x_issues:
616 616 zero: 0 issues
617 617 one: 1 issue
618 618 other: "%{count} issues"
619 619 label_total: Total
620 620 label_permissions: Permissions
621 621 label_current_status: Current status
622 622 label_new_statuses_allowed: New statuses allowed
623 623 label_all: all
624 624 label_any: any
625 625 label_none: none
626 626 label_nobody: nobody
627 627 label_next: Next
628 628 label_previous: Previous
629 629 label_used_by: Used by
630 630 label_details: Details
631 631 label_add_note: Add a note
632 632 label_per_page: Per page
633 633 label_calendar: Calendar
634 634 label_months_from: months from
635 635 label_gantt: Gantt
636 636 label_internal: Internal
637 637 label_last_changes: "last %{count} changes"
638 638 label_change_view_all: View all changes
639 639 label_personalize_page: Personalize this page
640 640 label_comment: Comment
641 641 label_comment_plural: Comments
642 642 label_x_comments:
643 643 zero: no comments
644 644 one: 1 comment
645 645 other: "%{count} comments"
646 646 label_comment_add: Add a comment
647 647 label_comment_added: Comment added
648 648 label_comment_delete: Delete comments
649 649 label_query: Custom query
650 650 label_query_plural: Custom queries
651 651 label_query_new: New query
652 652 label_my_queries: My custom queries
653 653 label_filter_add: Add filter
654 654 label_filter_plural: Filters
655 655 label_equals: is
656 656 label_not_equals: is not
657 657 label_in_less_than: in less than
658 658 label_in_more_than: in more than
659 label_in_the_next_days: in the next
660 label_in_the_past_days: in the past
659 661 label_greater_or_equal: '>='
660 662 label_less_or_equal: '<='
661 663 label_between: between
662 664 label_in: in
663 665 label_today: today
664 666 label_all_time: all time
665 667 label_yesterday: yesterday
666 668 label_this_week: this week
667 669 label_last_week: last week
668 670 label_last_n_weeks: "last %{count} weeks"
669 671 label_last_n_days: "last %{count} days"
670 672 label_this_month: this month
671 673 label_last_month: last month
672 674 label_this_year: this year
673 675 label_date_range: Date range
674 676 label_less_than_ago: less than days ago
675 677 label_more_than_ago: more than days ago
676 678 label_ago: days ago
677 679 label_contains: contains
678 680 label_not_contains: doesn't contain
679 681 label_any_issues_in_project: any issues in project
680 682 label_any_issues_not_in_project: any issues not in project
681 683 label_no_issues_in_project: no issues in project
682 684 label_day_plural: days
683 685 label_repository: Repository
684 686 label_repository_new: New repository
685 687 label_repository_plural: Repositories
686 688 label_browse: Browse
687 689 label_modification: "%{count} change"
688 690 label_modification_plural: "%{count} changes"
689 691 label_branch: Branch
690 692 label_tag: Tag
691 693 label_revision: Revision
692 694 label_revision_plural: Revisions
693 695 label_revision_id: "Revision %{value}"
694 696 label_associated_revisions: Associated revisions
695 697 label_added: added
696 698 label_modified: modified
697 699 label_copied: copied
698 700 label_renamed: renamed
699 701 label_deleted: deleted
700 702 label_latest_revision: Latest revision
701 703 label_latest_revision_plural: Latest revisions
702 704 label_view_revisions: View revisions
703 705 label_view_all_revisions: View all revisions
704 706 label_max_size: Maximum size
705 707 label_sort_highest: Move to top
706 708 label_sort_higher: Move up
707 709 label_sort_lower: Move down
708 710 label_sort_lowest: Move to bottom
709 711 label_roadmap: Roadmap
710 712 label_roadmap_due_in: "Due in %{value}"
711 713 label_roadmap_overdue: "%{value} late"
712 714 label_roadmap_no_issues: No issues for this version
713 715 label_search: Search
714 716 label_result_plural: Results
715 717 label_all_words: All words
716 718 label_wiki: Wiki
717 719 label_wiki_edit: Wiki edit
718 720 label_wiki_edit_plural: Wiki edits
719 721 label_wiki_page: Wiki page
720 722 label_wiki_page_plural: Wiki pages
721 723 label_index_by_title: Index by title
722 724 label_index_by_date: Index by date
723 725 label_current_version: Current version
724 726 label_preview: Preview
725 727 label_feed_plural: Feeds
726 728 label_changes_details: Details of all changes
727 729 label_issue_tracking: Issue tracking
728 730 label_spent_time: Spent time
729 731 label_overall_spent_time: Overall spent time
730 732 label_f_hour: "%{value} hour"
731 733 label_f_hour_plural: "%{value} hours"
732 734 label_time_tracking: Time tracking
733 735 label_change_plural: Changes
734 736 label_statistics: Statistics
735 737 label_commits_per_month: Commits per month
736 738 label_commits_per_author: Commits per author
737 739 label_diff: diff
738 740 label_view_diff: View differences
739 741 label_diff_inline: inline
740 742 label_diff_side_by_side: side by side
741 743 label_options: Options
742 744 label_copy_workflow_from: Copy workflow from
743 745 label_permissions_report: Permissions report
744 746 label_watched_issues: Watched issues
745 747 label_related_issues: Related issues
746 748 label_applied_status: Applied status
747 749 label_loading: Loading...
748 750 label_relation_new: New relation
749 751 label_relation_delete: Delete relation
750 752 label_relates_to: Related to
751 753 label_duplicates: Duplicates
752 754 label_duplicated_by: Duplicated by
753 755 label_blocks: Blocks
754 756 label_blocked_by: Blocked by
755 757 label_precedes: Precedes
756 758 label_follows: Follows
757 759 label_copied_to: Copied to
758 760 label_copied_from: Copied from
759 761 label_end_to_start: end to start
760 762 label_end_to_end: end to end
761 763 label_start_to_start: start to start
762 764 label_start_to_end: start to end
763 765 label_stay_logged_in: Stay logged in
764 766 label_disabled: disabled
765 767 label_show_completed_versions: Show completed versions
766 768 label_me: me
767 769 label_board: Forum
768 770 label_board_new: New forum
769 771 label_board_plural: Forums
770 772 label_board_locked: Locked
771 773 label_board_sticky: Sticky
772 774 label_topic_plural: Topics
773 775 label_message_plural: Messages
774 776 label_message_last: Last message
775 777 label_message_new: New message
776 778 label_message_posted: Message added
777 779 label_reply_plural: Replies
778 780 label_send_information: Send account information to the user
779 781 label_year: Year
780 782 label_month: Month
781 783 label_week: Week
782 784 label_date_from: From
783 785 label_date_to: To
784 786 label_language_based: Based on user's language
785 787 label_sort_by: "Sort by %{value}"
786 788 label_send_test_email: Send a test email
787 789 label_feeds_access_key: RSS access key
788 790 label_missing_feeds_access_key: Missing a RSS access key
789 791 label_feeds_access_key_created_on: "RSS access key created %{value} ago"
790 792 label_module_plural: Modules
791 793 label_added_time_by: "Added by %{author} %{age} ago"
792 794 label_updated_time_by: "Updated by %{author} %{age} ago"
793 795 label_updated_time: "Updated %{value} ago"
794 796 label_jump_to_a_project: Jump to a project...
795 797 label_file_plural: Files
796 798 label_changeset_plural: Changesets
797 799 label_default_columns: Default columns
798 800 label_no_change_option: (No change)
799 801 label_bulk_edit_selected_issues: Bulk edit selected issues
800 802 label_bulk_edit_selected_time_entries: Bulk edit selected time entries
801 803 label_theme: Theme
802 804 label_default: Default
803 805 label_search_titles_only: Search titles only
804 806 label_user_mail_option_all: "For any event on all my projects"
805 807 label_user_mail_option_selected: "For any event on the selected projects only..."
806 808 label_user_mail_option_none: "No events"
807 809 label_user_mail_option_only_my_events: "Only for things I watch or I'm involved in"
808 810 label_user_mail_option_only_assigned: "Only for things I am assigned to"
809 811 label_user_mail_option_only_owner: "Only for things I am the owner of"
810 812 label_user_mail_no_self_notified: "I don't want to be notified of changes that I make myself"
811 813 label_registration_activation_by_email: account activation by email
812 814 label_registration_manual_activation: manual account activation
813 815 label_registration_automatic_activation: automatic account activation
814 816 label_display_per_page: "Per page: %{value}"
815 817 label_age: Age
816 818 label_change_properties: Change properties
817 819 label_general: General
818 820 label_more: More
819 821 label_scm: SCM
820 822 label_plugins: Plugins
821 823 label_ldap_authentication: LDAP authentication
822 824 label_downloads_abbr: D/L
823 825 label_optional_description: Optional description
824 826 label_add_another_file: Add another file
825 827 label_preferences: Preferences
826 828 label_chronological_order: In chronological order
827 829 label_reverse_chronological_order: In reverse chronological order
828 830 label_planning: Planning
829 831 label_incoming_emails: Incoming emails
830 832 label_generate_key: Generate a key
831 833 label_issue_watchers: Watchers
832 834 label_example: Example
833 835 label_display: Display
834 836 label_sort: Sort
835 837 label_ascending: Ascending
836 838 label_descending: Descending
837 839 label_date_from_to: From %{start} to %{end}
838 840 label_wiki_content_added: Wiki page added
839 841 label_wiki_content_updated: Wiki page updated
840 842 label_group: Group
841 843 label_group_plural: Groups
842 844 label_group_new: New group
843 845 label_time_entry_plural: Spent time
844 846 label_version_sharing_none: Not shared
845 847 label_version_sharing_descendants: With subprojects
846 848 label_version_sharing_hierarchy: With project hierarchy
847 849 label_version_sharing_tree: With project tree
848 850 label_version_sharing_system: With all projects
849 851 label_update_issue_done_ratios: Update issue done ratios
850 852 label_copy_source: Source
851 853 label_copy_target: Target
852 854 label_copy_same_as_target: Same as target
853 855 label_display_used_statuses_only: Only display statuses that are used by this tracker
854 856 label_api_access_key: API access key
855 857 label_missing_api_access_key: Missing an API access key
856 858 label_api_access_key_created_on: "API access key created %{value} ago"
857 859 label_profile: Profile
858 860 label_subtask_plural: Subtasks
859 861 label_project_copy_notifications: Send email notifications during the project copy
860 862 label_principal_search: "Search for user or group:"
861 863 label_user_search: "Search for user:"
862 864 label_additional_workflow_transitions_for_author: Additional transitions allowed when the user is the author
863 865 label_additional_workflow_transitions_for_assignee: Additional transitions allowed when the user is the assignee
864 866 label_issues_visibility_all: All issues
865 867 label_issues_visibility_public: All non private issues
866 868 label_issues_visibility_own: Issues created by or assigned to the user
867 869 label_git_report_last_commit: Report last commit for files and directories
868 870 label_parent_revision: Parent
869 871 label_child_revision: Child
870 872 label_export_options: "%{export_format} export options"
871 873 label_copy_attachments: Copy attachments
872 874 label_copy_subtasks: Copy subtasks
873 875 label_item_position: "%{position} of %{count}"
874 876 label_completed_versions: Completed versions
875 877 label_search_for_watchers: Search for watchers to add
876 878 label_session_expiration: Session expiration
877 879 label_show_closed_projects: View closed projects
878 880 label_status_transitions: Status transitions
879 881 label_fields_permissions: Fields permissions
880 882 label_readonly: Read-only
881 883 label_required: Required
882 884 label_attribute_of_project: "Project's %{name}"
883 885 label_attribute_of_author: "Author's %{name}"
884 886 label_attribute_of_assigned_to: "Assignee's %{name}"
885 887 label_attribute_of_fixed_version: "Target version's %{name}"
886 888 label_cross_project_descendants: With subprojects
887 889 label_cross_project_tree: With project tree
888 890 label_cross_project_hierarchy: With project hierarchy
889 891 label_cross_project_system: With all projects
890 892
891 893 button_login: Login
892 894 button_submit: Submit
893 895 button_save: Save
894 896 button_check_all: Check all
895 897 button_uncheck_all: Uncheck all
896 898 button_collapse_all: Collapse all
897 899 button_expand_all: Expand all
898 900 button_delete: Delete
899 901 button_create: Create
900 902 button_create_and_continue: Create and continue
901 903 button_test: Test
902 904 button_edit: Edit
903 905 button_edit_associated_wikipage: "Edit associated Wiki page: %{page_title}"
904 906 button_add: Add
905 907 button_change: Change
906 908 button_apply: Apply
907 909 button_clear: Clear
908 910 button_lock: Lock
909 911 button_unlock: Unlock
910 912 button_download: Download
911 913 button_list: List
912 914 button_view: View
913 915 button_move: Move
914 916 button_move_and_follow: Move and follow
915 917 button_back: Back
916 918 button_cancel: Cancel
917 919 button_activate: Activate
918 920 button_sort: Sort
919 921 button_log_time: Log time
920 922 button_rollback: Rollback to this version
921 923 button_watch: Watch
922 924 button_unwatch: Unwatch
923 925 button_reply: Reply
924 926 button_archive: Archive
925 927 button_unarchive: Unarchive
926 928 button_reset: Reset
927 929 button_rename: Rename
928 930 button_change_password: Change password
929 931 button_copy: Copy
930 932 button_copy_and_follow: Copy and follow
931 933 button_annotate: Annotate
932 934 button_update: Update
933 935 button_configure: Configure
934 936 button_quote: Quote
935 937 button_duplicate: Duplicate
936 938 button_show: Show
937 939 button_hide: Hide
938 940 button_edit_section: Edit this section
939 941 button_export: Export
940 942 button_delete_my_account: Delete my account
941 943 button_close: Close
942 944 button_reopen: Reopen
943 945
944 946 status_active: active
945 947 status_registered: registered
946 948 status_locked: locked
947 949
948 950 project_status_active: active
949 951 project_status_closed: closed
950 952 project_status_archived: archived
951 953
952 954 version_status_open: open
953 955 version_status_locked: locked
954 956 version_status_closed: closed
955 957
956 958 field_active: Active
957 959
958 960 text_select_mail_notifications: Select actions for which email notifications should be sent.
959 961 text_regexp_info: eg. ^[A-Z0-9]+$
960 962 text_min_max_length_info: 0 means no restriction
961 963 text_project_destroy_confirmation: Are you sure you want to delete this project and related data?
962 964 text_subprojects_destroy_warning: "Its subproject(s): %{value} will be also deleted."
963 965 text_workflow_edit: Select a role and a tracker to edit the workflow
964 966 text_are_you_sure: Are you sure?
965 967 text_are_you_sure_with_children: "Delete issue and all child issues?"
966 968 text_journal_changed: "%{label} changed from %{old} to %{new}"
967 969 text_journal_changed_no_detail: "%{label} updated"
968 970 text_journal_set_to: "%{label} set to %{value}"
969 971 text_journal_deleted: "%{label} deleted (%{old})"
970 972 text_journal_added: "%{label} %{value} added"
971 973 text_tip_issue_begin_day: issue beginning this day
972 974 text_tip_issue_end_day: issue ending this day
973 975 text_tip_issue_begin_end_day: issue beginning and ending this day
974 976 text_project_identifier_info: 'Only lower case letters (a-z), numbers, dashes and underscores are allowed.<br />Once saved, the identifier cannot be changed.'
975 977 text_caracters_maximum: "%{count} characters maximum."
976 978 text_caracters_minimum: "Must be at least %{count} characters long."
977 979 text_length_between: "Length between %{min} and %{max} characters."
978 980 text_tracker_no_workflow: No workflow defined for this tracker
979 981 text_unallowed_characters: Unallowed characters
980 982 text_comma_separated: Multiple values allowed (comma separated).
981 983 text_line_separated: Multiple values allowed (one line for each value).
982 984 text_issues_ref_in_commit_messages: Referencing and fixing issues in commit messages
983 985 text_issue_added: "Issue %{id} has been reported by %{author}."
984 986 text_issue_updated: "Issue %{id} has been updated by %{author}."
985 987 text_wiki_destroy_confirmation: Are you sure you want to delete this wiki and all its content?
986 988 text_issue_category_destroy_question: "Some issues (%{count}) are assigned to this category. What do you want to do?"
987 989 text_issue_category_destroy_assignments: Remove category assignments
988 990 text_issue_category_reassign_to: Reassign issues to this category
989 991 text_user_mail_option: "For unselected projects, you will only receive notifications about things you watch or you're involved in (eg. issues you're the author or assignee)."
990 992 text_no_configuration_data: "Roles, trackers, issue statuses and workflow have not been configured yet.\nIt is highly recommended to load the default configuration. You will be able to modify it once loaded."
991 993 text_load_default_configuration: Load the default configuration
992 994 text_status_changed_by_changeset: "Applied in changeset %{value}."
993 995 text_time_logged_by_changeset: "Applied in changeset %{value}."
994 996 text_issues_destroy_confirmation: 'Are you sure you want to delete the selected issue(s)?'
995 997 text_issues_destroy_descendants_confirmation: "This will also delete %{count} subtask(s)."
996 998 text_time_entries_destroy_confirmation: 'Are you sure you want to delete the selected time entr(y/ies)?'
997 999 text_select_project_modules: 'Select modules to enable for this project:'
998 1000 text_default_administrator_account_changed: Default administrator account changed
999 1001 text_file_repository_writable: Attachments directory writable
1000 1002 text_plugin_assets_writable: Plugin assets directory writable
1001 1003 text_rmagick_available: RMagick available (optional)
1002 1004 text_destroy_time_entries_question: "%{hours} hours were reported on the issues you are about to delete. What do you want to do?"
1003 1005 text_destroy_time_entries: Delete reported hours
1004 1006 text_assign_time_entries_to_project: Assign reported hours to the project
1005 1007 text_reassign_time_entries: 'Reassign reported hours to this issue:'
1006 1008 text_user_wrote: "%{value} wrote:"
1007 1009 text_enumeration_destroy_question: "%{count} objects are assigned to this value."
1008 1010 text_enumeration_category_reassign_to: 'Reassign them to this value:'
1009 1011 text_email_delivery_not_configured: "Email delivery is not configured, and notifications are disabled.\nConfigure your SMTP server in config/configuration.yml and restart the application to enable them."
1010 1012 text_repository_usernames_mapping: "Select or update the Redmine user mapped to each username found in the repository log.\nUsers with the same Redmine and repository username or email are automatically mapped."
1011 1013 text_diff_truncated: '... This diff was truncated because it exceeds the maximum size that can be displayed.'
1012 1014 text_custom_field_possible_values_info: 'One line for each value'
1013 1015 text_wiki_page_destroy_question: "This page has %{descendants} child page(s) and descendant(s). What do you want to do?"
1014 1016 text_wiki_page_nullify_children: "Keep child pages as root pages"
1015 1017 text_wiki_page_destroy_children: "Delete child pages and all their descendants"
1016 1018 text_wiki_page_reassign_children: "Reassign child pages to this parent page"
1017 1019 text_own_membership_delete_confirmation: "You are about to remove some or all of your permissions and may no longer be able to edit this project after that.\nAre you sure you want to continue?"
1018 1020 text_zoom_in: Zoom in
1019 1021 text_zoom_out: Zoom out
1020 1022 text_warn_on_leaving_unsaved: "The current page contains unsaved text that will be lost if you leave this page."
1021 1023 text_scm_path_encoding_note: "Default: UTF-8"
1022 1024 text_git_repository_note: Repository is bare and local (e.g. /gitrepo, c:\gitrepo)
1023 1025 text_mercurial_repository_note: Local repository (e.g. /hgrepo, c:\hgrepo)
1024 1026 text_scm_command: Command
1025 1027 text_scm_command_version: Version
1026 1028 text_scm_config: You can configure your scm commands in config/configuration.yml. Please restart the application after editing it.
1027 1029 text_scm_command_not_available: Scm command is not available. Please check settings on the administration panel.
1028 1030 text_issue_conflict_resolution_overwrite: "Apply my changes anyway (previous notes will be kept but some changes may be overwritten)"
1029 1031 text_issue_conflict_resolution_add_notes: "Add my notes and discard my other changes"
1030 1032 text_issue_conflict_resolution_cancel: "Discard all my changes and redisplay %{link}"
1031 1033 text_account_destroy_confirmation: "Are you sure you want to proceed?\nYour account will be permanently deleted, with no way to reactivate it."
1032 1034 text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
1033 1035 text_project_closed: This project is closed and read-only.
1034 1036
1035 1037 default_role_manager: Manager
1036 1038 default_role_developer: Developer
1037 1039 default_role_reporter: Reporter
1038 1040 default_tracker_bug: Bug
1039 1041 default_tracker_feature: Feature
1040 1042 default_tracker_support: Support
1041 1043 default_issue_status_new: New
1042 1044 default_issue_status_in_progress: In Progress
1043 1045 default_issue_status_resolved: Resolved
1044 1046 default_issue_status_feedback: Feedback
1045 1047 default_issue_status_closed: Closed
1046 1048 default_issue_status_rejected: Rejected
1047 1049 default_doc_category_user: User documentation
1048 1050 default_doc_category_tech: Technical documentation
1049 1051 default_priority_low: Low
1050 1052 default_priority_normal: Normal
1051 1053 default_priority_high: High
1052 1054 default_priority_urgent: Urgent
1053 1055 default_priority_immediate: Immediate
1054 1056 default_activity_design: Design
1055 1057 default_activity_development: Development
1056 1058
1057 1059 enumeration_issue_priorities: Issue priorities
1058 1060 enumeration_doc_categories: Document categories
1059 1061 enumeration_activities: Activities (time tracking)
1060 1062 enumeration_system_activity: System Activity
1061 1063 description_filter: Filter
1062 1064 description_search: Searchfield
1063 1065 description_choose_project: Projects
1064 1066 description_project_scope: Search scope
1065 1067 description_notes: Notes
1066 1068 description_message_content: Message content
1067 1069 description_query_sort_criteria_attribute: Sort attribute
1068 1070 description_query_sort_criteria_direction: Sort direction
1069 1071 description_user_mail_notification: Mail notification settings
1070 1072 description_available_columns: Available Columns
1071 1073 description_selected_columns: Selected Columns
1072 1074 description_all_columns: All Columns
1073 1075 description_issue_category_reassign: Choose issue category
1074 1076 description_wiki_subpages_reassign: Choose new parent page
1075 1077 description_date_range_list: Choose range from list
1076 1078 description_date_range_interval: Choose range by selecting start and end date
1077 1079 description_date_from: Enter start date
1078 1080 description_date_to: Enter end date
1079 1081 text_repository_identifier_info: 'Only lower case letters (a-z), numbers, dashes and underscores are allowed.<br />Once saved, the identifier cannot be changed.'
@@ -1,1096 +1,1098
1 1 # French translations for Ruby on Rails
2 2 # by Christian Lescuyer (christian@flyingcoders.com)
3 3 # contributor: Sebastien Grosjean - ZenCocoon.com
4 4 # contributor: Thibaut Cuvelier - Developpez.com
5 5
6 6 fr:
7 7 direction: ltr
8 8 date:
9 9 formats:
10 10 default: "%d/%m/%Y"
11 11 short: "%e %b"
12 12 long: "%e %B %Y"
13 13 long_ordinal: "%e %B %Y"
14 14 only_day: "%e"
15 15
16 16 day_names: [dimanche, lundi, mardi, mercredi, jeudi, vendredi, samedi]
17 17 abbr_day_names: [dim, lun, mar, mer, jeu, ven, sam]
18 18 month_names: [~, janvier, fΓ©vrier, mars, avril, mai, juin, juillet, aoΓ»t, septembre, octobre, novembre, dΓ©cembre]
19 19 abbr_month_names: [~, jan., fΓ©v., mar., avr., mai, juin, juil., aoΓ»t, sept., oct., nov., dΓ©c.]
20 20 order:
21 21 - :day
22 22 - :month
23 23 - :year
24 24
25 25 time:
26 26 formats:
27 27 default: "%d/%m/%Y %H:%M"
28 28 time: "%H:%M"
29 29 short: "%d %b %H:%M"
30 30 long: "%A %d %B %Y %H:%M:%S %Z"
31 31 long_ordinal: "%A %d %B %Y %H:%M:%S %Z"
32 32 only_second: "%S"
33 33 am: 'am'
34 34 pm: 'pm'
35 35
36 36 datetime:
37 37 distance_in_words:
38 38 half_a_minute: "30 secondes"
39 39 less_than_x_seconds:
40 40 zero: "moins d'une seconde"
41 41 one: "moins d'uneΒ seconde"
42 42 other: "moins de %{count}Β secondes"
43 43 x_seconds:
44 44 one: "1Β seconde"
45 45 other: "%{count}Β secondes"
46 46 less_than_x_minutes:
47 47 zero: "moins d'une minute"
48 48 one: "moins d'uneΒ minute"
49 49 other: "moins de %{count}Β minutes"
50 50 x_minutes:
51 51 one: "1Β minute"
52 52 other: "%{count}Β minutes"
53 53 about_x_hours:
54 54 one: "environ une heure"
55 55 other: "environ %{count}Β heures"
56 56 x_hours:
57 57 one: "une heure"
58 58 other: "%{count}Β heures"
59 59 x_days:
60 60 one: "unΒ jour"
61 61 other: "%{count}Β jours"
62 62 about_x_months:
63 63 one: "environ un mois"
64 64 other: "environ %{count}Β mois"
65 65 x_months:
66 66 one: "unΒ mois"
67 67 other: "%{count}Β mois"
68 68 about_x_years:
69 69 one: "environ un an"
70 70 other: "environ %{count}Β ans"
71 71 over_x_years:
72 72 one: "plus d'un an"
73 73 other: "plus de %{count}Β ans"
74 74 almost_x_years:
75 75 one: "presqu'un an"
76 76 other: "presque %{count} ans"
77 77 prompts:
78 78 year: "AnnΓ©e"
79 79 month: "Mois"
80 80 day: "Jour"
81 81 hour: "Heure"
82 82 minute: "Minute"
83 83 second: "Seconde"
84 84
85 85 number:
86 86 format:
87 87 precision: 3
88 88 separator: ','
89 89 delimiter: 'Β '
90 90 currency:
91 91 format:
92 92 unit: '€'
93 93 precision: 2
94 94 format: '%nΒ %u'
95 95 human:
96 96 format:
97 97 precision: 3
98 98 storage_units:
99 99 format: "%n %u"
100 100 units:
101 101 byte:
102 102 one: "octet"
103 103 other: "octet"
104 104 kb: "ko"
105 105 mb: "Mo"
106 106 gb: "Go"
107 107 tb: "To"
108 108
109 109 support:
110 110 array:
111 111 sentence_connector: 'et'
112 112 skip_last_comma: true
113 113 word_connector: ", "
114 114 two_words_connector: " et "
115 115 last_word_connector: " et "
116 116
117 117 activerecord:
118 118 errors:
119 119 template:
120 120 header:
121 121 one: "Impossible d'enregistrer %{model} : une erreur"
122 122 other: "Impossible d'enregistrer %{model} : %{count} erreurs."
123 123 body: "Veuillez vΓ©rifier les champs suivantsΒ :"
124 124 messages:
125 125 inclusion: "n'est pas inclus(e) dans la liste"
126 126 exclusion: "n'est pas disponible"
127 127 invalid: "n'est pas valide"
128 128 confirmation: "ne concorde pas avec la confirmation"
129 129 accepted: "doit Γͺtre acceptΓ©(e)"
130 130 empty: "doit Γͺtre renseignΓ©(e)"
131 131 blank: "doit Γͺtre renseignΓ©(e)"
132 132 too_long: "est trop long (pas plus de %{count} caractères)"
133 133 too_short: "est trop court (au moins %{count} caractères)"
134 134 wrong_length: "ne fait pas la bonne longueur (doit comporter %{count} caractères)"
135 135 taken: "est dΓ©jΓ  utilisΓ©"
136 136 not_a_number: "n'est pas un nombre"
137 137 not_a_date: "n'est pas une date valide"
138 138 greater_than: "doit Γͺtre supΓ©rieur Γ  %{count}"
139 139 greater_than_or_equal_to: "doit Γͺtre supΓ©rieur ou Γ©gal Γ  %{count}"
140 140 equal_to: "doit Γͺtre Γ©gal Γ  %{count}"
141 141 less_than: "doit Γͺtre infΓ©rieur Γ  %{count}"
142 142 less_than_or_equal_to: "doit Γͺtre infΓ©rieur ou Γ©gal Γ  %{count}"
143 143 odd: "doit Γͺtre impair"
144 144 even: "doit Γͺtre pair"
145 145 greater_than_start_date: "doit Γͺtre postΓ©rieure Γ  la date de dΓ©but"
146 146 not_same_project: "n'appartient pas au mΓͺme projet"
147 147 circular_dependency: "Cette relation crΓ©erait une dΓ©pendance circulaire"
148 148 cant_link_an_issue_with_a_descendant: "Une demande ne peut pas Γͺtre liΓ©e Γ  l'une de ses sous-tΓ’ches"
149 149
150 150 actionview_instancetag_blank_option: Choisir
151 151
152 152 general_text_No: 'Non'
153 153 general_text_Yes: 'Oui'
154 154 general_text_no: 'non'
155 155 general_text_yes: 'oui'
156 156 general_lang_name: 'FranΓ§ais'
157 157 general_csv_separator: ';'
158 158 general_csv_decimal_separator: ','
159 159 general_csv_encoding: ISO-8859-1
160 160 general_pdf_encoding: UTF-8
161 161 general_first_day_of_week: '1'
162 162
163 163 notice_account_updated: Le compte a été mis à jour avec succès.
164 164 notice_account_invalid_creditentials: Identifiant ou mot de passe invalide.
165 165 notice_account_password_updated: Mot de passe mis à jour avec succès.
166 166 notice_account_wrong_password: Mot de passe incorrect
167 167 notice_account_register_done: Un message contenant les instructions pour activer votre compte vous a Γ©tΓ© envoyΓ©.
168 168 notice_account_unknown_email: Aucun compte ne correspond Γ  cette adresse.
169 169 notice_can_t_change_password: Ce compte utilise une authentification externe. Impossible de changer le mot de passe.
170 170 notice_account_lost_email_sent: Un message contenant les instructions pour choisir un nouveau mot de passe vous a Γ©tΓ© envoyΓ©.
171 171 notice_account_activated: Votre compte a Γ©tΓ© activΓ©. Vous pouvez Γ  prΓ©sent vous connecter.
172 172 notice_successful_create: Création effectuée avec succès.
173 173 notice_successful_update: Mise à jour effectuée avec succès.
174 174 notice_successful_delete: Suppression effectuée avec succès.
175 175 notice_successful_connection: Connexion rΓ©ussie.
176 176 notice_file_not_found: "La page Γ  laquelle vous souhaitez accΓ©der n'existe pas ou a Γ©tΓ© supprimΓ©e."
177 177 notice_locking_conflict: Les donnΓ©es ont Γ©tΓ© mises Γ  jour par un autre utilisateur. Mise Γ  jour impossible.
178 178 notice_not_authorized: "Vous n'Γͺtes pas autorisΓ© Γ  accΓ©der Γ  cette page."
179 179 notice_not_authorized_archived_project: Le projet auquel vous tentez d'accΓ©der a Γ©tΓ© archivΓ©.
180 180 notice_email_sent: "Un email a Γ©tΓ© envoyΓ© Γ  %{value}"
181 181 notice_email_error: "Erreur lors de l'envoi de l'email (%{value})"
182 182 notice_feeds_access_key_reseted: "Votre clé d'accès aux flux RSS a été réinitialisée."
183 183 notice_failed_to_save_issues: "%{count} demande(s) sur les %{total} sΓ©lectionnΓ©es n'ont pas pu Γͺtre mise(s) Γ  jour : %{ids}."
184 184 notice_failed_to_save_time_entries: "%{count} temps passΓ©(s) sur les %{total} sΓ©lectionnΓ©s n'ont pas pu Γͺtre mis Γ  jour: %{ids}."
185 185 notice_no_issue_selected: "Aucune demande sΓ©lectionnΓ©e ! Cochez les demandes que vous voulez mettre Γ  jour."
186 186 notice_account_pending: "Votre compte a été créé et attend l'approbation de l'administrateur."
187 187 notice_default_data_loaded: Paramétrage par défaut chargé avec succès.
188 188 notice_unable_delete_version: Impossible de supprimer cette version.
189 189 notice_issue_done_ratios_updated: L'avancement des demandes a Γ©tΓ© mis Γ  jour.
190 190 notice_api_access_key_reseted: Votre clé d'accès API a été réinitialisée.
191 191 notice_gantt_chart_truncated: "Le diagramme a Γ©tΓ© tronquΓ© car il excΓ¨de le nombre maximal d'Γ©lΓ©ments pouvant Γͺtre affichΓ©s (%{max})"
192 192 notice_issue_successful_create: "Demande %{id} créée."
193 193 notice_issue_update_conflict: "La demande a Γ©tΓ© mise Γ  jour par un autre utilisateur pendant que vous la modifiez."
194 194 notice_account_deleted: "Votre compte a Γ©tΓ© dΓ©finitivement supprimΓ©."
195 195 notice_user_successful_create: "Utilisateur %{id} créé."
196 196
197 197 error_can_t_load_default_data: "Une erreur s'est produite lors du chargement du paramΓ©trage : %{value}"
198 198 error_scm_not_found: "L'entrΓ©e et/ou la rΓ©vision demandΓ©e n'existe pas dans le dΓ©pΓ΄t."
199 199 error_scm_command_failed: "Une erreur s'est produite lors de l'accès au dépôt : %{value}"
200 200 error_scm_annotate: "L'entrΓ©e n'existe pas ou ne peut pas Γͺtre annotΓ©e."
201 201 error_issue_not_found_in_project: "La demande n'existe pas ou n'appartient pas Γ  ce projet"
202 202 error_can_not_reopen_issue_on_closed_version: 'Une demande assignΓ©e Γ  une version fermΓ©e ne peut pas Γͺtre rΓ©ouverte'
203 203 error_can_not_archive_project: "Ce projet ne peut pas Γͺtre archivΓ©"
204 204 error_workflow_copy_source: 'Veuillez sΓ©lectionner un tracker et/ou un rΓ΄le source'
205 205 error_workflow_copy_target: 'Veuillez sΓ©lectionner les trackers et rΓ΄les cibles'
206 206 error_issue_done_ratios_not_updated: L'avancement des demandes n'a pas pu Γͺtre mis Γ  jour.
207 207 error_attachment_too_big: Ce fichier ne peut pas Γͺtre attachΓ© car il excΓ¨de la taille maximale autorisΓ©e (%{max_size})
208 208 error_session_expired: "Votre session a expirΓ©. Veuillez vous reconnecter."
209 209
210 210 warning_attachments_not_saved: "%{count} fichier(s) n'ont pas pu Γͺtre sauvegardΓ©s."
211 211
212 212 mail_subject_lost_password: "Votre mot de passe %{value}"
213 213 mail_body_lost_password: 'Pour changer votre mot de passe, cliquez sur le lien suivant :'
214 214 mail_subject_register: "Activation de votre compte %{value}"
215 215 mail_body_register: 'Pour activer votre compte, cliquez sur le lien suivant :'
216 216 mail_body_account_information_external: "Vous pouvez utiliser votre compte %{value} pour vous connecter."
217 217 mail_body_account_information: Paramètres de connexion de votre compte
218 218 mail_subject_account_activation_request: "Demande d'activation d'un compte %{value}"
219 219 mail_body_account_activation_request: "Un nouvel utilisateur (%{value}) s'est inscrit. Son compte nΓ©cessite votre approbation :"
220 220 mail_subject_reminder: "%{count} demande(s) arrivent Γ  Γ©chΓ©ance (%{days})"
221 221 mail_body_reminder: "%{count} demande(s) qui vous sont assignΓ©es arrivent Γ  Γ©chΓ©ance dans les %{days} prochains jours :"
222 222 mail_subject_wiki_content_added: "Page wiki '%{id}' ajoutΓ©e"
223 223 mail_body_wiki_content_added: "La page wiki '%{id}' a Γ©tΓ© ajoutΓ©e par %{author}."
224 224 mail_subject_wiki_content_updated: "Page wiki '%{id}' mise Γ  jour"
225 225 mail_body_wiki_content_updated: "La page wiki '%{id}' a Γ©tΓ© mise Γ  jour par %{author}."
226 226
227 227 gui_validation_error: 1 erreur
228 228 gui_validation_error_plural: "%{count} erreurs"
229 229
230 230 field_name: Nom
231 231 field_description: Description
232 232 field_summary: RΓ©sumΓ©
233 233 field_is_required: Obligatoire
234 234 field_firstname: PrΓ©nom
235 235 field_lastname: Nom
236 236 field_mail: "Email "
237 237 field_filename: Fichier
238 238 field_filesize: Taille
239 239 field_downloads: TΓ©lΓ©chargements
240 240 field_author: Auteur
241 241 field_created_on: "Créé "
242 242 field_updated_on: "Mis-Γ -jour "
243 243 field_field_format: Format
244 244 field_is_for_all: Pour tous les projets
245 245 field_possible_values: Valeurs possibles
246 246 field_regexp: Expression régulière
247 247 field_min_length: Longueur minimum
248 248 field_max_length: Longueur maximum
249 249 field_value: Valeur
250 250 field_category: CatΓ©gorie
251 251 field_title: Titre
252 252 field_project: Projet
253 253 field_issue: Demande
254 254 field_status: Statut
255 255 field_notes: Notes
256 256 field_is_closed: Demande fermΓ©e
257 257 field_is_default: Valeur par dΓ©faut
258 258 field_tracker: Tracker
259 259 field_subject: Sujet
260 260 field_due_date: EchΓ©ance
261 261 field_assigned_to: AssignΓ© Γ 
262 262 field_priority: PrioritΓ©
263 263 field_fixed_version: Version cible
264 264 field_user: Utilisateur
265 265 field_role: RΓ΄le
266 266 field_homepage: "Site web "
267 267 field_is_public: Public
268 268 field_parent: Sous-projet de
269 269 field_is_in_roadmap: Demandes affichΓ©es dans la roadmap
270 270 field_login: "Identifiant "
271 271 field_mail_notification: Notifications par mail
272 272 field_admin: Administrateur
273 273 field_last_login_on: "Dernière connexion "
274 274 field_language: Langue
275 275 field_effective_date: Date
276 276 field_password: Mot de passe
277 277 field_new_password: Nouveau mot de passe
278 278 field_password_confirmation: Confirmation
279 279 field_version: Version
280 280 field_type: Type
281 281 field_host: HΓ΄te
282 282 field_port: Port
283 283 field_account: Compte
284 284 field_base_dn: Base DN
285 285 field_attr_login: Attribut Identifiant
286 286 field_attr_firstname: Attribut PrΓ©nom
287 287 field_attr_lastname: Attribut Nom
288 288 field_attr_mail: Attribut Email
289 289 field_onthefly: CrΓ©ation des utilisateurs Γ  la volΓ©e
290 290 field_start_date: DΓ©but
291 291 field_done_ratio: "% rΓ©alisΓ©"
292 292 field_auth_source: Mode d'authentification
293 293 field_hide_mail: Cacher mon adresse mail
294 294 field_comments: Commentaire
295 295 field_url: URL
296 296 field_start_page: Page de dΓ©marrage
297 297 field_subproject: Sous-projet
298 298 field_hours: Heures
299 299 field_activity: ActivitΓ©
300 300 field_spent_on: Date
301 301 field_identifier: Identifiant
302 302 field_is_filter: UtilisΓ© comme filtre
303 303 field_issue_to: Demande liΓ©e
304 304 field_delay: Retard
305 305 field_assignable: Demandes assignables Γ  ce rΓ΄le
306 306 field_redirect_existing_links: Rediriger les liens existants
307 307 field_estimated_hours: Temps estimΓ©
308 308 field_column_names: Colonnes
309 309 field_time_zone: Fuseau horaire
310 310 field_searchable: UtilisΓ© pour les recherches
311 311 field_default_value: Valeur par dΓ©faut
312 312 field_comments_sorting: Afficher les commentaires
313 313 field_parent_title: Page parent
314 314 field_editable: Modifiable
315 315 field_watcher: Observateur
316 316 field_identity_url: URL OpenID
317 317 field_content: Contenu
318 318 field_group_by: Grouper par
319 319 field_sharing: Partage
320 320 field_active: Actif
321 321 field_parent_issue: TΓ’che parente
322 322 field_visible: Visible
323 323 field_warn_on_leaving_unsaved: "M'avertir lorsque je quitte une page contenant du texte non sauvegardΓ©"
324 324 field_issues_visibility: VisibilitΓ© des demandes
325 325 field_is_private: PrivΓ©e
326 326 field_commit_logs_encoding: Encodage des messages de commit
327 327 field_repository_is_default: DΓ©pΓ΄t principal
328 328 field_multiple: Valeurs multiples
329 329 field_auth_source_ldap_filter: Filtre LDAP
330 330 field_core_fields: Champs standards
331 331 field_timeout: "Timeout (en secondes)"
332 332 field_board_parent: Forum parent
333 333 field_private_notes: Notes privΓ©es
334 334
335 335 setting_app_title: Titre de l'application
336 336 setting_app_subtitle: Sous-titre de l'application
337 337 setting_welcome_text: Texte d'accueil
338 338 setting_default_language: Langue par dΓ©faut
339 339 setting_login_required: Authentification obligatoire
340 340 setting_self_registration: Inscription des nouveaux utilisateurs
341 341 setting_attachment_max_size: Taille maximale des fichiers
342 342 setting_issues_export_limit: Limite d'exportation des demandes
343 343 setting_mail_from: Adresse d'Γ©mission
344 344 setting_bcc_recipients: Destinataires en copie cachΓ©e (cci)
345 345 setting_plain_text_mail: Mail en texte brut (non HTML)
346 346 setting_host_name: Nom d'hΓ΄te et chemin
347 347 setting_text_formatting: Formatage du texte
348 348 setting_wiki_compression: Compression de l'historique des pages wiki
349 349 setting_feeds_limit: Nombre maximal d'Γ©lΓ©ments dans les flux Atom
350 350 setting_default_projects_public: DΓ©finir les nouveaux projets comme publics par dΓ©faut
351 351 setting_autofetch_changesets: RΓ©cupΓ©ration automatique des commits
352 352 setting_sys_api_enabled: Activer les WS pour la gestion des dΓ©pΓ΄ts
353 353 setting_commit_ref_keywords: Mots-clΓ©s de rΓ©fΓ©rencement
354 354 setting_commit_fix_keywords: Mots-clΓ©s de rΓ©solution
355 355 setting_autologin: DurΓ©e maximale de connexion automatique
356 356 setting_date_format: Format de date
357 357 setting_time_format: Format d'heure
358 358 setting_cross_project_issue_relations: Autoriser les relations entre demandes de diffΓ©rents projets
359 359 setting_cross_project_subtasks: Autoriser les sous-tΓ’ches dans des projets diffΓ©rents
360 360 setting_issue_list_default_columns: Colonnes affichΓ©es par dΓ©faut sur la liste des demandes
361 361 setting_emails_footer: Pied-de-page des emails
362 362 setting_protocol: Protocole
363 363 setting_per_page_options: Options d'objets affichΓ©s par page
364 364 setting_user_format: Format d'affichage des utilisateurs
365 365 setting_activity_days_default: Nombre de jours affichΓ©s sur l'activitΓ© des projets
366 366 setting_display_subprojects_issues: Afficher par dΓ©faut les demandes des sous-projets sur les projets principaux
367 367 setting_enabled_scm: SCM activΓ©s
368 368 setting_mail_handler_body_delimiters: "Tronquer les emails après l'une de ces lignes"
369 369 setting_mail_handler_api_enabled: "Activer le WS pour la rΓ©ception d'emails"
370 370 setting_mail_handler_api_key: ClΓ© de protection de l'API
371 371 setting_sequential_project_identifiers: GΓ©nΓ©rer des identifiants de projet sΓ©quentiels
372 372 setting_gravatar_enabled: Afficher les Gravatar des utilisateurs
373 373 setting_diff_max_lines_displayed: Nombre maximum de lignes de diff affichΓ©es
374 374 setting_file_max_size_displayed: Taille maximum des fichiers texte affichΓ©s en ligne
375 375 setting_repository_log_display_limit: "Nombre maximum de rΓ©visions affichΓ©es sur l'historique d'un fichier"
376 376 setting_openid: "Autoriser l'authentification et l'enregistrement OpenID"
377 377 setting_password_min_length: Longueur minimum des mots de passe
378 378 setting_new_project_user_role_id: RΓ΄le donnΓ© Γ  un utilisateur non-administrateur qui crΓ©e un projet
379 379 setting_default_projects_modules: Modules activΓ©s par dΓ©faut pour les nouveaux projets
380 380 setting_issue_done_ratio: Calcul de l'avancement des demandes
381 381 setting_issue_done_ratio_issue_status: Utiliser le statut
382 382 setting_issue_done_ratio_issue_field: 'Utiliser le champ % effectuΓ©'
383 383 setting_rest_api_enabled: Activer l'API REST
384 384 setting_gravatar_default: Image Gravatar par dΓ©faut
385 385 setting_start_of_week: Jour de dΓ©but des calendriers
386 386 setting_cache_formatted_text: Mettre en cache le texte formatΓ©
387 387 setting_commit_logtime_enabled: Permettre la saisie de temps
388 388 setting_commit_logtime_activity_id: ActivitΓ© pour le temps saisi
389 389 setting_gantt_items_limit: Nombre maximum d'Γ©lΓ©ments affichΓ©s sur le gantt
390 390 setting_issue_group_assignment: Permettre l'assignement des demandes aux groupes
391 391 setting_default_issue_start_date_to_creation_date: Donner Γ  la date de dΓ©but d'une nouvelle demande la valeur de la date du jour
392 392 setting_commit_cross_project_ref: Permettre le rΓ©fΓ©rencement et la rΓ©solution des demandes de tous les autres projets
393 393 setting_unsubscribe: Permettre aux utilisateurs de supprimer leur propre compte
394 394 setting_session_lifetime: DurΓ©e de vie maximale des sessions
395 395 setting_session_timeout: DurΓ©e maximale d'inactivitΓ©
396 396 setting_thumbnails_enabled: Afficher les vignettes des images
397 397 setting_thumbnails_size: Taille des vignettes (en pixels)
398 398 setting_non_working_week_days: Jours non travaillΓ©s
399 399
400 400 permission_add_project: CrΓ©er un projet
401 401 permission_add_subprojects: CrΓ©er des sous-projets
402 402 permission_edit_project: Modifier le projet
403 403 permission_close_project: Fermer / rΓ©ouvrir le projet
404 404 permission_select_project_modules: Choisir les modules
405 405 permission_manage_members: GΓ©rer les membres
406 406 permission_manage_versions: GΓ©rer les versions
407 407 permission_manage_categories: GΓ©rer les catΓ©gories de demandes
408 408 permission_view_issues: Voir les demandes
409 409 permission_add_issues: CrΓ©er des demandes
410 410 permission_edit_issues: Modifier les demandes
411 411 permission_manage_issue_relations: GΓ©rer les relations
412 412 permission_set_issues_private: Rendre les demandes publiques ou privΓ©es
413 413 permission_set_own_issues_private: Rendre ses propres demandes publiques ou privΓ©es
414 414 permission_add_issue_notes: Ajouter des notes
415 415 permission_edit_issue_notes: Modifier les notes
416 416 permission_edit_own_issue_notes: Modifier ses propres notes
417 417 permission_view_private_notes: Voir les notes privΓ©es
418 418 permission_set_notes_private: Rendre les notes privΓ©es
419 419 permission_move_issues: DΓ©placer les demandes
420 420 permission_delete_issues: Supprimer les demandes
421 421 permission_manage_public_queries: GΓ©rer les requΓͺtes publiques
422 422 permission_save_queries: Sauvegarder les requΓͺtes
423 423 permission_view_gantt: Voir le gantt
424 424 permission_view_calendar: Voir le calendrier
425 425 permission_view_issue_watchers: Voir la liste des observateurs
426 426 permission_add_issue_watchers: Ajouter des observateurs
427 427 permission_delete_issue_watchers: Supprimer des observateurs
428 428 permission_log_time: Saisir le temps passΓ©
429 429 permission_view_time_entries: Voir le temps passΓ©
430 430 permission_edit_time_entries: Modifier les temps passΓ©s
431 431 permission_edit_own_time_entries: Modifier son propre temps passΓ©
432 432 permission_manage_news: GΓ©rer les annonces
433 433 permission_comment_news: Commenter les annonces
434 434 permission_manage_documents: GΓ©rer les documents
435 435 permission_view_documents: Voir les documents
436 436 permission_manage_files: GΓ©rer les fichiers
437 437 permission_view_files: Voir les fichiers
438 438 permission_manage_wiki: GΓ©rer le wiki
439 439 permission_rename_wiki_pages: Renommer les pages
440 440 permission_delete_wiki_pages: Supprimer les pages
441 441 permission_view_wiki_pages: Voir le wiki
442 442 permission_view_wiki_edits: "Voir l'historique des modifications"
443 443 permission_edit_wiki_pages: Modifier les pages
444 444 permission_delete_wiki_pages_attachments: Supprimer les fichiers joints
445 445 permission_protect_wiki_pages: ProtΓ©ger les pages
446 446 permission_manage_repository: GΓ©rer le dΓ©pΓ΄t de sources
447 447 permission_browse_repository: Parcourir les sources
448 448 permission_view_changesets: Voir les rΓ©visions
449 449 permission_commit_access: Droit de commit
450 450 permission_manage_boards: GΓ©rer les forums
451 451 permission_view_messages: Voir les messages
452 452 permission_add_messages: Poster un message
453 453 permission_edit_messages: Modifier les messages
454 454 permission_edit_own_messages: Modifier ses propres messages
455 455 permission_delete_messages: Supprimer les messages
456 456 permission_delete_own_messages: Supprimer ses propres messages
457 457 permission_export_wiki_pages: Exporter les pages
458 458 permission_manage_project_activities: GΓ©rer les activitΓ©s
459 459 permission_manage_subtasks: GΓ©rer les sous-tΓ’ches
460 460 permission_manage_related_issues: GΓ©rer les demandes associΓ©es
461 461
462 462 project_module_issue_tracking: Suivi des demandes
463 463 project_module_time_tracking: Suivi du temps passΓ©
464 464 project_module_news: Publication d'annonces
465 465 project_module_documents: Publication de documents
466 466 project_module_files: Publication de fichiers
467 467 project_module_wiki: Wiki
468 468 project_module_repository: DΓ©pΓ΄t de sources
469 469 project_module_boards: Forums de discussion
470 470
471 471 label_user: Utilisateur
472 472 label_user_plural: Utilisateurs
473 473 label_user_new: Nouvel utilisateur
474 474 label_user_anonymous: Anonyme
475 475 label_project: Projet
476 476 label_project_new: Nouveau projet
477 477 label_project_plural: Projets
478 478 label_x_projects:
479 479 zero: aucun projet
480 480 one: un projet
481 481 other: "%{count} projets"
482 482 label_project_all: Tous les projets
483 483 label_project_latest: Derniers projets
484 484 label_issue: Demande
485 485 label_issue_new: Nouvelle demande
486 486 label_issue_plural: Demandes
487 487 label_issue_view_all: Voir toutes les demandes
488 488 label_issue_added: Demande ajoutΓ©e
489 489 label_issue_updated: Demande mise Γ  jour
490 490 label_issue_note_added: Note ajoutΓ©e
491 491 label_issue_status_updated: Statut changΓ©
492 492 label_issue_priority_updated: PrioritΓ© changΓ©e
493 493 label_issues_by: "Demandes par %{value}"
494 494 label_document: Document
495 495 label_document_new: Nouveau document
496 496 label_document_plural: Documents
497 497 label_document_added: Document ajoutΓ©
498 498 label_role: RΓ΄le
499 499 label_role_plural: RΓ΄les
500 500 label_role_new: Nouveau rΓ΄le
501 501 label_role_and_permissions: RΓ΄les et permissions
502 502 label_role_anonymous: Anonyme
503 503 label_role_non_member: Non membre
504 504 label_member: Membre
505 505 label_member_new: Nouveau membre
506 506 label_member_plural: Membres
507 507 label_tracker: Tracker
508 508 label_tracker_plural: Trackers
509 509 label_tracker_new: Nouveau tracker
510 510 label_workflow: Workflow
511 511 label_issue_status: Statut de demandes
512 512 label_issue_status_plural: Statuts de demandes
513 513 label_issue_status_new: Nouveau statut
514 514 label_issue_category: CatΓ©gorie de demandes
515 515 label_issue_category_plural: CatΓ©gories de demandes
516 516 label_issue_category_new: Nouvelle catΓ©gorie
517 517 label_custom_field: Champ personnalisΓ©
518 518 label_custom_field_plural: Champs personnalisΓ©s
519 519 label_custom_field_new: Nouveau champ personnalisΓ©
520 520 label_enumerations: Listes de valeurs
521 521 label_enumeration_new: Nouvelle valeur
522 522 label_information: Information
523 523 label_information_plural: Informations
524 524 label_please_login: Identification
525 525 label_register: S'enregistrer
526 526 label_login_with_open_id_option: S'authentifier avec OpenID
527 527 label_password_lost: Mot de passe perdu
528 528 label_home: Accueil
529 529 label_my_page: Ma page
530 530 label_my_account: Mon compte
531 531 label_my_projects: Mes projets
532 532 label_my_page_block: Blocs disponibles
533 533 label_administration: Administration
534 534 label_login: Connexion
535 535 label_logout: DΓ©connexion
536 536 label_help: Aide
537 537 label_reported_issues: "Demandes soumises "
538 538 label_assigned_to_me_issues: Demandes qui me sont assignΓ©es
539 539 label_last_login: "Dernière connexion "
540 540 label_registered_on: "Inscrit le "
541 541 label_activity: ActivitΓ©
542 542 label_overall_activity: ActivitΓ© globale
543 543 label_user_activity: "ActivitΓ© de %{value}"
544 544 label_new: Nouveau
545 545 label_logged_as: ConnectΓ© en tant que
546 546 label_environment: Environnement
547 547 label_authentication: Authentification
548 548 label_auth_source: Mode d'authentification
549 549 label_auth_source_new: Nouveau mode d'authentification
550 550 label_auth_source_plural: Modes d'authentification
551 551 label_subproject_plural: Sous-projets
552 552 label_subproject_new: Nouveau sous-projet
553 553 label_and_its_subprojects: "%{value} et ses sous-projets"
554 554 label_min_max_length: Longueurs mini - maxi
555 555 label_list: Liste
556 556 label_date: Date
557 557 label_integer: Entier
558 558 label_float: Nombre dΓ©cimal
559 559 label_boolean: BoolΓ©en
560 560 label_string: Texte
561 561 label_text: Texte long
562 562 label_attribute: Attribut
563 563 label_attribute_plural: Attributs
564 564 label_download: "%{count} tΓ©lΓ©chargement"
565 565 label_download_plural: "%{count} tΓ©lΓ©chargements"
566 566 label_no_data: Aucune donnΓ©e Γ  afficher
567 567 label_change_status: Changer le statut
568 568 label_history: Historique
569 569 label_attachment: Fichier
570 570 label_attachment_new: Nouveau fichier
571 571 label_attachment_delete: Supprimer le fichier
572 572 label_attachment_plural: Fichiers
573 573 label_file_added: Fichier ajoutΓ©
574 574 label_report: Rapport
575 575 label_report_plural: Rapports
576 576 label_news: Annonce
577 577 label_news_new: Nouvelle annonce
578 578 label_news_plural: Annonces
579 579 label_news_latest: Dernières annonces
580 580 label_news_view_all: Voir toutes les annonces
581 581 label_news_added: Annonce ajoutΓ©e
582 582 label_news_comment_added: Commentaire ajoutΓ© Γ  une annonce
583 583 label_settings: Configuration
584 584 label_overview: AperΓ§u
585 585 label_version: Version
586 586 label_version_new: Nouvelle version
587 587 label_version_plural: Versions
588 588 label_confirmation: Confirmation
589 589 label_export_to: 'Formats disponibles :'
590 590 label_read: Lire...
591 591 label_public_projects: Projets publics
592 592 label_open_issues: ouvert
593 593 label_open_issues_plural: ouverts
594 594 label_closed_issues: fermΓ©
595 595 label_closed_issues_plural: fermΓ©s
596 596 label_x_open_issues_abbr_on_total:
597 597 zero: 0 ouverte sur %{total}
598 598 one: 1 ouverte sur %{total}
599 599 other: "%{count} ouvertes sur %{total}"
600 600 label_x_open_issues_abbr:
601 601 zero: 0 ouverte
602 602 one: 1 ouverte
603 603 other: "%{count} ouvertes"
604 604 label_x_closed_issues_abbr:
605 605 zero: 0 fermΓ©e
606 606 one: 1 fermΓ©e
607 607 other: "%{count} fermΓ©es"
608 608 label_x_issues:
609 609 zero: 0 demande
610 610 one: 1 demande
611 611 other: "%{count} demandes"
612 612 label_total: Total
613 613 label_permissions: Permissions
614 614 label_current_status: Statut actuel
615 615 label_new_statuses_allowed: Nouveaux statuts autorisΓ©s
616 616 label_all: tous
617 617 label_any: tous
618 618 label_none: aucun
619 619 label_nobody: personne
620 620 label_next: Suivant
621 621 label_previous: PrΓ©cΓ©dent
622 622 label_used_by: UtilisΓ© par
623 623 label_details: DΓ©tails
624 624 label_add_note: Ajouter une note
625 625 label_per_page: Par page
626 626 label_calendar: Calendrier
627 627 label_months_from: mois depuis
628 628 label_gantt: Gantt
629 629 label_internal: Interne
630 630 label_last_changes: "%{count} derniers changements"
631 631 label_change_view_all: Voir tous les changements
632 632 label_personalize_page: Personnaliser cette page
633 633 label_comment: Commentaire
634 634 label_comment_plural: Commentaires
635 635 label_x_comments:
636 636 zero: aucun commentaire
637 637 one: un commentaire
638 638 other: "%{count} commentaires"
639 639 label_comment_add: Ajouter un commentaire
640 640 label_comment_added: Commentaire ajoutΓ©
641 641 label_comment_delete: Supprimer les commentaires
642 642 label_query: Rapport personnalisΓ©
643 643 label_query_plural: Rapports personnalisΓ©s
644 644 label_query_new: Nouveau rapport
645 645 label_my_queries: Mes rapports personnalisΓ©s
646 646 label_filter_add: "Ajouter le filtre "
647 647 label_filter_plural: Filtres
648 648 label_equals: Γ©gal
649 649 label_not_equals: diffΓ©rent
650 650 label_in_less_than: dans moins de
651 651 label_in_more_than: dans plus de
652 label_in_the_next_days: dans les prochains jours
653 label_in_the_past_days: dans les derniers jours
652 654 label_in: dans
653 655 label_today: aujourd'hui
654 656 label_all_time: toute la pΓ©riode
655 657 label_yesterday: hier
656 658 label_this_week: cette semaine
657 659 label_last_week: la semaine dernière
658 660 label_last_n_weeks: "les %{count} dernières semaines"
659 661 label_last_n_days: "les %{count} derniers jours"
660 662 label_this_month: ce mois-ci
661 663 label_last_month: le mois dernier
662 664 label_this_year: cette annΓ©e
663 665 label_date_range: PΓ©riode
664 666 label_less_than_ago: il y a moins de
665 667 label_more_than_ago: il y a plus de
666 668 label_ago: il y a
667 669 label_contains: contient
668 670 label_not_contains: ne contient pas
669 671 label_any_issues_in_project: une demande du projet
670 672 label_any_issues_not_in_project: une demande hors du projet
671 673 label_no_issues_in_project: aucune demande du projet
672 674 label_day_plural: jours
673 675 label_repository: DΓ©pΓ΄t
674 676 label_repository_new: Nouveau dΓ©pΓ΄t
675 677 label_repository_plural: DΓ©pΓ΄ts
676 678 label_browse: Parcourir
677 679 label_modification: "%{count} modification"
678 680 label_modification_plural: "%{count} modifications"
679 681 label_revision: "RΓ©vision "
680 682 label_revision_plural: RΓ©visions
681 683 label_associated_revisions: RΓ©visions associΓ©es
682 684 label_added: ajoutΓ©
683 685 label_modified: modifiΓ©
684 686 label_copied: copiΓ©
685 687 label_renamed: renommΓ©
686 688 label_deleted: supprimΓ©
687 689 label_latest_revision: Dernière révision
688 690 label_latest_revision_plural: Dernières révisions
689 691 label_view_revisions: Voir les rΓ©visions
690 692 label_max_size: Taille maximale
691 693 label_sort_highest: Remonter en premier
692 694 label_sort_higher: Remonter
693 695 label_sort_lower: Descendre
694 696 label_sort_lowest: Descendre en dernier
695 697 label_roadmap: Roadmap
696 698 label_roadmap_due_in: "Γ‰chΓ©ance dans %{value}"
697 699 label_roadmap_overdue: "En retard de %{value}"
698 700 label_roadmap_no_issues: Aucune demande pour cette version
699 701 label_search: "Recherche "
700 702 label_result_plural: RΓ©sultats
701 703 label_all_words: Tous les mots
702 704 label_wiki: Wiki
703 705 label_wiki_edit: RΓ©vision wiki
704 706 label_wiki_edit_plural: RΓ©visions wiki
705 707 label_wiki_page: Page wiki
706 708 label_wiki_page_plural: Pages wiki
707 709 label_index_by_title: Index par titre
708 710 label_index_by_date: Index par date
709 711 label_current_version: Version actuelle
710 712 label_preview: PrΓ©visualisation
711 713 label_feed_plural: Flux RSS
712 714 label_changes_details: DΓ©tails de tous les changements
713 715 label_issue_tracking: Suivi des demandes
714 716 label_spent_time: Temps passΓ©
715 717 label_f_hour: "%{value} heure"
716 718 label_f_hour_plural: "%{value} heures"
717 719 label_time_tracking: Suivi du temps
718 720 label_change_plural: Changements
719 721 label_statistics: Statistiques
720 722 label_commits_per_month: Commits par mois
721 723 label_commits_per_author: Commits par auteur
722 724 label_view_diff: Voir les diffΓ©rences
723 725 label_diff_inline: en ligne
724 726 label_diff_side_by_side: cΓ΄te Γ  cΓ΄te
725 727 label_options: Options
726 728 label_copy_workflow_from: Copier le workflow de
727 729 label_permissions_report: Synthèse des permissions
728 730 label_watched_issues: Demandes surveillΓ©es
729 731 label_related_issues: Demandes liΓ©es
730 732 label_applied_status: Statut appliquΓ©
731 733 label_loading: Chargement...
732 734 label_relation_new: Nouvelle relation
733 735 label_relation_delete: Supprimer la relation
734 736 label_relates_to: LiΓ© Γ 
735 737 label_duplicates: Duplique
736 738 label_duplicated_by: DupliquΓ© par
737 739 label_blocks: Bloque
738 740 label_blocked_by: BloquΓ© par
739 741 label_precedes: Précède
740 742 label_follows: Suit
741 743 label_copied_to: CopiΓ© vers
742 744 label_copied_from: CopiΓ© depuis
743 745 label_end_to_start: fin Γ  dΓ©but
744 746 label_end_to_end: fin Γ  fin
745 747 label_start_to_start: dΓ©but Γ  dΓ©but
746 748 label_start_to_end: dΓ©but Γ  fin
747 749 label_stay_logged_in: Rester connectΓ©
748 750 label_disabled: dΓ©sactivΓ©
749 751 label_show_completed_versions: Voir les versions passΓ©es
750 752 label_me: moi
751 753 label_board: Forum
752 754 label_board_new: Nouveau forum
753 755 label_board_plural: Forums
754 756 label_topic_plural: Discussions
755 757 label_message_plural: Messages
756 758 label_message_last: Dernier message
757 759 label_message_new: Nouveau message
758 760 label_message_posted: Message ajoutΓ©
759 761 label_reply_plural: RΓ©ponses
760 762 label_send_information: Envoyer les informations Γ  l'utilisateur
761 763 label_year: AnnΓ©e
762 764 label_month: Mois
763 765 label_week: Semaine
764 766 label_date_from: Du
765 767 label_date_to: Au
766 768 label_language_based: BasΓ© sur la langue de l'utilisateur
767 769 label_sort_by: "Trier par %{value}"
768 770 label_send_test_email: Envoyer un email de test
769 771 label_feeds_access_key_created_on: "Clé d'accès RSS créée il y a %{value}"
770 772 label_module_plural: Modules
771 773 label_added_time_by: "AjoutΓ© par %{author} il y a %{age}"
772 774 label_updated_time_by: "Mis Γ  jour par %{author} il y a %{age}"
773 775 label_updated_time: "Mis Γ  jour il y a %{value}"
774 776 label_jump_to_a_project: Aller Γ  un projet...
775 777 label_file_plural: Fichiers
776 778 label_changeset_plural: RΓ©visions
777 779 label_default_columns: Colonnes par dΓ©faut
778 780 label_no_change_option: (Pas de changement)
779 781 label_bulk_edit_selected_issues: Modifier les demandes sΓ©lectionnΓ©es
780 782 label_theme: Thème
781 783 label_default: DΓ©faut
782 784 label_search_titles_only: Uniquement dans les titres
783 785 label_user_mail_option_all: "Pour tous les Γ©vΓ©nements de tous mes projets"
784 786 label_user_mail_option_selected: "Pour tous les Γ©vΓ©nements des projets sΓ©lectionnΓ©s..."
785 787 label_user_mail_no_self_notified: "Je ne veux pas Γͺtre notifiΓ© des changements que j'effectue"
786 788 label_registration_activation_by_email: activation du compte par email
787 789 label_registration_manual_activation: activation manuelle du compte
788 790 label_registration_automatic_activation: activation automatique du compte
789 791 label_display_per_page: "Par page : %{value}"
790 792 label_age: Γ‚ge
791 793 label_change_properties: Changer les propriΓ©tΓ©s
792 794 label_general: GΓ©nΓ©ral
793 795 label_more: Plus
794 796 label_scm: SCM
795 797 label_plugins: Plugins
796 798 label_ldap_authentication: Authentification LDAP
797 799 label_downloads_abbr: D/L
798 800 label_optional_description: Description facultative
799 801 label_add_another_file: Ajouter un autre fichier
800 802 label_preferences: PrΓ©fΓ©rences
801 803 label_chronological_order: Dans l'ordre chronologique
802 804 label_reverse_chronological_order: Dans l'ordre chronologique inverse
803 805 label_planning: Planning
804 806 label_incoming_emails: Emails entrants
805 807 label_generate_key: GΓ©nΓ©rer une clΓ©
806 808 label_issue_watchers: Observateurs
807 809 label_example: Exemple
808 810 label_display: Affichage
809 811 label_sort: Tri
810 812 label_ascending: Croissant
811 813 label_descending: DΓ©croissant
812 814 label_date_from_to: Du %{start} au %{end}
813 815 label_wiki_content_added: Page wiki ajoutΓ©e
814 816 label_wiki_content_updated: Page wiki mise Γ  jour
815 817 label_group_plural: Groupes
816 818 label_group: Groupe
817 819 label_group_new: Nouveau groupe
818 820 label_time_entry_plural: Temps passΓ©
819 821 label_version_sharing_none: Non partagΓ©
820 822 label_version_sharing_descendants: Avec les sous-projets
821 823 label_version_sharing_hierarchy: Avec toute la hiΓ©rarchie
822 824 label_version_sharing_tree: Avec tout l'arbre
823 825 label_version_sharing_system: Avec tous les projets
824 826 label_copy_source: Source
825 827 label_copy_target: Cible
826 828 label_copy_same_as_target: Comme la cible
827 829 label_update_issue_done_ratios: Mettre Γ  jour l'avancement des demandes
828 830 label_display_used_statuses_only: N'afficher que les statuts utilisΓ©s dans ce tracker
829 831 label_api_access_key: Clé d'accès API
830 832 label_api_access_key_created_on: Clé d'accès API créée il y a %{value}
831 833 label_feeds_access_key: Clé d'accès RSS
832 834 label_missing_api_access_key: Clé d'accès API manquante
833 835 label_missing_feeds_access_key: Clé d'accès RSS manquante
834 836 label_close_versions: Fermer les versions terminΓ©es
835 837 label_revision_id: RΓ©vision %{value}
836 838 label_profile: Profil
837 839 label_subtask_plural: Sous-tΓ’ches
838 840 label_project_copy_notifications: Envoyer les notifications durant la copie du projet
839 841 label_principal_search: "Rechercher un utilisateur ou un groupe :"
840 842 label_user_search: "Rechercher un utilisateur :"
841 843 label_additional_workflow_transitions_for_author: Autorisations supplémentaires lorsque l'utilisateur a créé la demande
842 844 label_additional_workflow_transitions_for_assignee: Autorisations supplΓ©mentaires lorsque la demande est assignΓ©e Γ  l'utilisateur
843 845 label_issues_visibility_all: Toutes les demandes
844 846 label_issues_visibility_public: Toutes les demandes non privΓ©es
845 847 label_issues_visibility_own: Demandes créées par ou assignées à l'utilisateur
846 848 label_export_options: Options d'exportation %{export_format}
847 849 label_copy_attachments: Copier les fichiers
848 850 label_copy_subtasks: Copier les sous-tΓ’ches
849 851 label_item_position: "%{position} sur %{count}"
850 852 label_completed_versions: Versions passΓ©es
851 853 label_session_expiration: Expiration des sessions
852 854 label_show_closed_projects: Voir les projets fermΓ©s
853 855 label_status_transitions: Changements de statut
854 856 label_fields_permissions: Permissions sur les champs
855 857 label_readonly: Lecture
856 858 label_required: Obligatoire
857 859 label_attribute_of_project: "%{name} du projet"
858 860 label_attribute_of_author: "%{name} de l'auteur"
859 861 label_attribute_of_assigned_to: "%{name} de l'assignΓ©"
860 862 label_attribute_of_fixed_version: "%{name} de la version cible"
861 863 label_cross_project_descendants: Avec les sous-projets
862 864 label_cross_project_tree: Avec tout l'arbre
863 865 label_cross_project_hierarchy: Avec toute la hiΓ©rarchie
864 866 label_cross_project_system: Avec tous les projets
865 867
866 868 button_login: Connexion
867 869 button_submit: Soumettre
868 870 button_save: Sauvegarder
869 871 button_check_all: Tout cocher
870 872 button_uncheck_all: Tout dΓ©cocher
871 873 button_collapse_all: Plier tout
872 874 button_expand_all: DΓ©plier tout
873 875 button_delete: Supprimer
874 876 button_create: CrΓ©er
875 877 button_create_and_continue: CrΓ©er et continuer
876 878 button_test: Tester
877 879 button_edit: Modifier
878 880 button_add: Ajouter
879 881 button_change: Changer
880 882 button_apply: Appliquer
881 883 button_clear: Effacer
882 884 button_lock: Verrouiller
883 885 button_unlock: DΓ©verrouiller
884 886 button_download: TΓ©lΓ©charger
885 887 button_list: Lister
886 888 button_view: Voir
887 889 button_move: DΓ©placer
888 890 button_move_and_follow: DΓ©placer et suivre
889 891 button_back: Retour
890 892 button_cancel: Annuler
891 893 button_activate: Activer
892 894 button_sort: Trier
893 895 button_log_time: Saisir temps
894 896 button_rollback: Revenir Γ  cette version
895 897 button_watch: Surveiller
896 898 button_unwatch: Ne plus surveiller
897 899 button_reply: RΓ©pondre
898 900 button_archive: Archiver
899 901 button_unarchive: DΓ©sarchiver
900 902 button_reset: RΓ©initialiser
901 903 button_rename: Renommer
902 904 button_change_password: Changer de mot de passe
903 905 button_copy: Copier
904 906 button_copy_and_follow: Copier et suivre
905 907 button_annotate: Annoter
906 908 button_update: Mettre Γ  jour
907 909 button_configure: Configurer
908 910 button_quote: Citer
909 911 button_duplicate: Dupliquer
910 912 button_show: Afficher
911 913 button_hide: Cacher
912 914 button_edit_section: Modifier cette section
913 915 button_export: Exporter
914 916 button_delete_my_account: Supprimer mon compte
915 917 button_close: Fermer
916 918 button_reopen: RΓ©ouvrir
917 919
918 920 status_active: actif
919 921 status_registered: enregistrΓ©
920 922 status_locked: verrouillΓ©
921 923
922 924 project_status_active: actif
923 925 project_status_closed: fermΓ©
924 926 project_status_archived: archivΓ©
925 927
926 928 version_status_open: ouvert
927 929 version_status_locked: verrouillΓ©
928 930 version_status_closed: fermΓ©
929 931
930 932 text_select_mail_notifications: Actions pour lesquelles une notification par e-mail est envoyΓ©e
931 933 text_regexp_info: ex. ^[A-Z0-9]+$
932 934 text_min_max_length_info: 0 pour aucune restriction
933 935 text_project_destroy_confirmation: Êtes-vous sûr de vouloir supprimer ce projet et toutes ses données ?
934 936 text_subprojects_destroy_warning: "Ses sous-projets : %{value} seront Γ©galement supprimΓ©s."
935 937 text_workflow_edit: SΓ©lectionner un tracker et un rΓ΄le pour Γ©diter le workflow
936 938 text_are_you_sure: Êtes-vous sûr ?
937 939 text_tip_issue_begin_day: tΓ’che commenΓ§ant ce jour
938 940 text_tip_issue_end_day: tΓ’che finissant ce jour
939 941 text_tip_issue_begin_end_day: tΓ’che commenΓ§ant et finissant ce jour
940 942 text_project_identifier_info: 'Seuls les lettres minuscules (a-z), chiffres, tirets et underscore sont autorisΓ©s.<br />Un fois sauvegardΓ©, l''identifiant ne pourra plus Γͺtre modifiΓ©.'
941 943 text_caracters_maximum: "%{count} caractères maximum."
942 944 text_caracters_minimum: "%{count} caractères minimum."
943 945 text_length_between: "Longueur comprise entre %{min} et %{max} caractères."
944 946 text_tracker_no_workflow: Aucun worflow n'est dΓ©fini pour ce tracker
945 947 text_unallowed_characters: Caractères non autorisés
946 948 text_comma_separated: Plusieurs valeurs possibles (sΓ©parΓ©es par des virgules).
947 949 text_line_separated: Plusieurs valeurs possibles (une valeur par ligne).
948 950 text_issues_ref_in_commit_messages: RΓ©fΓ©rencement et rΓ©solution des demandes dans les commentaires de commits
949 951 text_issue_added: "La demande %{id} a Γ©tΓ© soumise par %{author}."
950 952 text_issue_updated: "La demande %{id} a Γ©tΓ© mise Γ  jour par %{author}."
951 953 text_wiki_destroy_confirmation: Etes-vous sΓ»r de vouloir supprimer ce wiki et tout son contenu ?
952 954 text_issue_category_destroy_question: "%{count} demandes sont affectΓ©es Γ  cette catΓ©gorie. Que voulez-vous faire ?"
953 955 text_issue_category_destroy_assignments: N'affecter les demandes Γ  aucune autre catΓ©gorie
954 956 text_issue_category_reassign_to: RΓ©affecter les demandes Γ  cette catΓ©gorie
955 957 text_user_mail_option: "Pour les projets non sΓ©lectionnΓ©s, vous recevrez seulement des notifications pour ce que vous surveillez ou Γ  quoi vous participez (exemple: demandes dont vous Γͺtes l'auteur ou la personne assignΓ©e)."
956 958 text_no_configuration_data: "Les rΓ΄les, trackers, statuts et le workflow ne sont pas encore paramΓ©trΓ©s.\nIl est vivement recommandΓ© de charger le paramΓ©trage par defaut. Vous pourrez le modifier une fois chargΓ©."
957 959 text_load_default_configuration: Charger le paramΓ©trage par dΓ©faut
958 960 text_status_changed_by_changeset: "AppliquΓ© par commit %{value}."
959 961 text_time_logged_by_changeset: "AppliquΓ© par commit %{value}"
960 962 text_issues_destroy_confirmation: 'Êtes-vous sûr de vouloir supprimer la ou les demandes(s) selectionnée(s) ?'
961 963 text_issues_destroy_descendants_confirmation: "Cela entrainera Γ©galement la suppression de %{count} sous-tΓ’che(s)."
962 964 text_select_project_modules: 'SΓ©lectionner les modules Γ  activer pour ce projet :'
963 965 text_default_administrator_account_changed: Compte administrateur par dΓ©faut changΓ©
964 966 text_file_repository_writable: RΓ©pertoire de stockage des fichiers accessible en Γ©criture
965 967 text_plugin_assets_writable: RΓ©pertoire public des plugins accessible en Γ©criture
966 968 text_rmagick_available: Bibliothèque RMagick présente (optionnelle)
967 969 text_destroy_time_entries_question: "%{hours} heures ont Γ©tΓ© enregistrΓ©es sur les demandes Γ  supprimer. Que voulez-vous faire ?"
968 970 text_destroy_time_entries: Supprimer les heures
969 971 text_assign_time_entries_to_project: Reporter les heures sur le projet
970 972 text_reassign_time_entries: 'Reporter les heures sur cette demande:'
971 973 text_user_wrote: "%{value} a Γ©crit :"
972 974 text_enumeration_destroy_question: "Cette valeur est affectΓ©e Γ  %{count} objets."
973 975 text_enumeration_category_reassign_to: 'RΓ©affecter les objets Γ  cette valeur:'
974 976 text_email_delivery_not_configured: "L'envoi de mail n'est pas configurΓ©, les notifications sont dΓ©sactivΓ©es.\nConfigurez votre serveur SMTP dans config/configuration.yml et redΓ©marrez l'application pour les activer."
975 977 text_repository_usernames_mapping: "Vous pouvez sΓ©lectionner ou modifier l'utilisateur Redmine associΓ© Γ  chaque nom d'utilisateur figurant dans l'historique du dΓ©pΓ΄t.\nLes utilisateurs avec le mΓͺme identifiant ou la mΓͺme adresse mail seront automatiquement associΓ©s."
976 978 text_diff_truncated: '... Ce diffΓ©rentiel a Γ©tΓ© tronquΓ© car il excΓ¨de la taille maximale pouvant Γͺtre affichΓ©e.'
977 979 text_custom_field_possible_values_info: 'Une ligne par valeur'
978 980 text_wiki_page_destroy_question: "Cette page possède %{descendants} sous-page(s) et descendante(s). Que voulez-vous faire ?"
979 981 text_wiki_page_nullify_children: "Conserver les sous-pages en tant que pages racines"
980 982 text_wiki_page_destroy_children: "Supprimer les sous-pages et toutes leurs descedantes"
981 983 text_wiki_page_reassign_children: "RΓ©affecter les sous-pages Γ  cette page"
982 984 text_own_membership_delete_confirmation: "Vous allez supprimer tout ou partie de vos permissions sur ce projet et ne serez peut-Γͺtre plus autorisΓ© Γ  modifier ce projet.\nEtes-vous sΓ»r de vouloir continuer ?"
983 985 text_warn_on_leaving_unsaved: "Cette page contient du texte non sauvegardΓ© qui sera perdu si vous quittez la page."
984 986 text_issue_conflict_resolution_overwrite: "Appliquer quand mΓͺme ma mise Γ  jour (les notes prΓ©cΓ©dentes seront conservΓ©es mais des changements pourront Γͺtre Γ©crasΓ©s)"
985 987 text_issue_conflict_resolution_add_notes: "Ajouter mes notes et ignorer mes autres changements"
986 988 text_issue_conflict_resolution_cancel: "Annuler ma mise Γ  jour et rΓ©afficher %{link}"
987 989 text_account_destroy_confirmation: "Êtes-vous sûr de vouloir continuer ?\nVotre compte sera définitivement supprimé, sans aucune possibilité de le réactiver."
988 990 text_session_expiration_settings: "Attention : le changement de ces paramètres peut entrainer l'expiration des sessions utilisateurs en cours, y compris la vôtre."
989 991 text_project_closed: Ce projet est fermΓ© et accessible en lecture seule.
990 992
991 993 default_role_manager: "Manager "
992 994 default_role_developer: "DΓ©veloppeur "
993 995 default_role_reporter: "Rapporteur "
994 996 default_tracker_bug: Anomalie
995 997 default_tracker_feature: Evolution
996 998 default_tracker_support: Assistance
997 999 default_issue_status_new: Nouveau
998 1000 default_issue_status_in_progress: En cours
999 1001 default_issue_status_resolved: RΓ©solu
1000 1002 default_issue_status_feedback: Commentaire
1001 1003 default_issue_status_closed: FermΓ©
1002 1004 default_issue_status_rejected: RejetΓ©
1003 1005 default_doc_category_user: Documentation utilisateur
1004 1006 default_doc_category_tech: Documentation technique
1005 1007 default_priority_low: Bas
1006 1008 default_priority_normal: Normal
1007 1009 default_priority_high: Haut
1008 1010 default_priority_urgent: Urgent
1009 1011 default_priority_immediate: ImmΓ©diat
1010 1012 default_activity_design: Conception
1011 1013 default_activity_development: DΓ©veloppement
1012 1014
1013 1015 enumeration_issue_priorities: PrioritΓ©s des demandes
1014 1016 enumeration_doc_categories: CatΓ©gories des documents
1015 1017 enumeration_activities: ActivitΓ©s (suivi du temps)
1016 1018 label_greater_or_equal: ">="
1017 1019 label_less_or_equal: "<="
1018 1020 label_between: entre
1019 1021 label_view_all_revisions: Voir toutes les rΓ©visions
1020 1022 label_tag: Tag
1021 1023 label_branch: Branche
1022 1024 error_no_tracker_in_project: "Aucun tracker n'est associΓ© Γ  ce projet. VΓ©rifier la configuration du projet."
1023 1025 error_no_default_issue_status: "Aucun statut de demande n'est dΓ©fini par dΓ©faut. VΓ©rifier votre configuration (Administration -> Statuts de demandes)."
1024 1026 text_journal_changed: "%{label} changΓ© de %{old} Γ  %{new}"
1025 1027 text_journal_changed_no_detail: "%{label} mis Γ  jour"
1026 1028 text_journal_set_to: "%{label} mis Γ  %{value}"
1027 1029 text_journal_deleted: "%{label} %{old} supprimΓ©"
1028 1030 text_journal_added: "%{label} %{value} ajoutΓ©"
1029 1031 enumeration_system_activity: Activité système
1030 1032 label_board_sticky: Sticky
1031 1033 label_board_locked: VerrouillΓ©
1032 1034 error_unable_delete_issue_status: Impossible de supprimer le statut de demande
1033 1035 error_can_not_delete_custom_field: Impossible de supprimer le champ personnalisΓ©
1034 1036 error_unable_to_connect: Connexion impossible (%{value})
1035 1037 error_can_not_remove_role: Ce rΓ΄le est utilisΓ© et ne peut pas Γͺtre supprimΓ©.
1036 1038 error_can_not_delete_tracker: Ce tracker contient des demandes et ne peut pas Γͺtre supprimΓ©.
1037 1039 field_principal: Principal
1038 1040 notice_failed_to_save_members: "Erreur lors de la sauvegarde des membres: %{errors}."
1039 1041 text_zoom_out: Zoom arrière
1040 1042 text_zoom_in: Zoom avant
1041 1043 notice_unable_delete_time_entry: Impossible de supprimer le temps passΓ©.
1042 1044 label_overall_spent_time: Temps passΓ© global
1043 1045 field_time_entries: Temps passΓ©
1044 1046 project_module_gantt: Gantt
1045 1047 project_module_calendar: Calendrier
1046 1048 button_edit_associated_wikipage: "Modifier la page wiki associΓ©e: %{page_title}"
1047 1049 text_are_you_sure_with_children: Supprimer la demande et toutes ses sous-demandes ?
1048 1050 field_text: Champ texte
1049 1051 label_user_mail_option_only_owner: Seulement pour ce que j'ai créé
1050 1052 setting_default_notification_option: Option de notification par dΓ©faut
1051 1053 label_user_mail_option_only_my_events: Seulement pour ce que je surveille
1052 1054 label_user_mail_option_only_assigned: Seulement pour ce qui m'est assignΓ©
1053 1055 label_user_mail_option_none: Aucune notification
1054 1056 field_member_of_group: Groupe de l'assignΓ©
1055 1057 field_assigned_to_role: RΓ΄le de l'assignΓ©
1056 1058 setting_emails_header: En-tΓͺte des emails
1057 1059 label_bulk_edit_selected_time_entries: Modifier les temps passΓ©s sΓ©lectionnΓ©s
1058 1060 text_time_entries_destroy_confirmation: "Etes-vous sΓ»r de vouloir supprimer les temps passΓ©s sΓ©lectionnΓ©s ?"
1059 1061 field_scm_path_encoding: Encodage des chemins
1060 1062 text_scm_path_encoding_note: "DΓ©faut : UTF-8"
1061 1063 field_path_to_repository: Chemin du dΓ©pΓ΄t
1062 1064 field_root_directory: RΓ©pertoire racine
1063 1065 field_cvs_module: Module
1064 1066 field_cvsroot: CVSROOT
1065 1067 text_mercurial_repository_note: "DΓ©pΓ΄t local (exemples : /hgrepo, c:\\hgrepo)"
1066 1068 text_scm_command: Commande
1067 1069 text_scm_command_version: Version
1068 1070 label_git_report_last_commit: Afficher le dernier commit des fichiers et rΓ©pertoires
1069 1071 text_scm_config: Vous pouvez configurer les commandes des SCM dans config/configuration.yml. Redémarrer l'application après modification.
1070 1072 text_scm_command_not_available: Ce SCM n'est pas disponible. Vérifier les paramètres dans la section administration.
1071 1073 label_diff: diff
1072 1074 text_git_repository_note: Repository is bare and local (e.g. /gitrepo, c:\gitrepo)
1073 1075 description_query_sort_criteria_direction: Ordre de tri
1074 1076 description_project_scope: Périmètre de recherche
1075 1077 description_filter: Filtre
1076 1078 description_user_mail_notification: Option de notification
1077 1079 description_date_from: Date de dΓ©but
1078 1080 description_message_content: Contenu du message
1079 1081 description_available_columns: Colonnes disponibles
1080 1082 description_all_columns: Toutes les colonnes
1081 1083 description_date_range_interval: Choisir une pΓ©riode
1082 1084 description_issue_category_reassign: Choisir une catΓ©gorie
1083 1085 description_search: Champ de recherche
1084 1086 description_notes: Notes
1085 1087 description_date_range_list: Choisir une pΓ©riode prΓ©dΓ©finie
1086 1088 description_choose_project: Projets
1087 1089 description_date_to: Date de fin
1088 1090 description_query_sort_criteria_attribute: Critère de tri
1089 1091 description_wiki_subpages_reassign: Choisir une nouvelle page parent
1090 1092 description_selected_columns: Colonnes sΓ©lectionnΓ©es
1091 1093 label_parent_revision: Parent
1092 1094 label_child_revision: Enfant
1093 1095 error_scm_annotate_big_text_file: Cette entrΓ©e ne peut pas Γͺtre annotΓ©e car elle excΓ¨de la taille maximale.
1094 1096 setting_repositories_encodings: Encodages des fichiers et des dΓ©pΓ΄ts
1095 1097 label_search_for_watchers: Rechercher des observateurs
1096 1098 text_repository_identifier_info: 'Seuls les lettres minuscules (a-z), chiffres, tirets et underscore sont autorisΓ©s.<br />Un fois sauvegardΓ©, l''identifiant ne pourra plus Γͺtre modifiΓ©.'
@@ -1,601 +1,603
1 1 /* Redmine - project management software
2 2 Copyright (C) 2006-2012 Jean-Philippe Lang */
3 3
4 4 function checkAll(id, checked) {
5 5 if (checked) {
6 6 $('#'+id).find('input[type=checkbox]').attr('checked', true);
7 7 } else {
8 8 $('#'+id).find('input[type=checkbox]').removeAttr('checked');
9 9 }
10 10 }
11 11
12 12 function toggleCheckboxesBySelector(selector) {
13 13 var all_checked = true;
14 14 $(selector).each(function(index) {
15 15 if (!$(this).is(':checked')) { all_checked = false; }
16 16 });
17 17 $(selector).attr('checked', !all_checked)
18 18 }
19 19
20 20 function showAndScrollTo(id, focus) {
21 21 $('#'+id).show();
22 22 if (focus!=null) {
23 23 $('#'+focus).focus();
24 24 }
25 25 $('html, body').animate({scrollTop: $('#'+id).offset().top}, 100);
26 26 }
27 27
28 28 function toggleRowGroup(el) {
29 29 var tr = $(el).parents('tr').first();
30 30 var n = tr.next();
31 31 tr.toggleClass('open');
32 32 while (n.length && !n.hasClass('group')) {
33 33 n.toggle();
34 34 n = n.next('tr');
35 35 }
36 36 }
37 37
38 38 function collapseAllRowGroups(el) {
39 39 var tbody = $(el).parents('tbody').first();
40 40 tbody.children('tr').each(function(index) {
41 41 if ($(this).hasClass('group')) {
42 42 $(this).removeClass('open');
43 43 } else {
44 44 $(this).hide();
45 45 }
46 46 });
47 47 }
48 48
49 49 function expandAllRowGroups(el) {
50 50 var tbody = $(el).parents('tbody').first();
51 51 tbody.children('tr').each(function(index) {
52 52 if ($(this).hasClass('group')) {
53 53 $(this).addClass('open');
54 54 } else {
55 55 $(this).show();
56 56 }
57 57 });
58 58 }
59 59
60 60 function toggleAllRowGroups(el) {
61 61 var tr = $(el).parents('tr').first();
62 62 if (tr.hasClass('open')) {
63 63 collapseAllRowGroups(el);
64 64 } else {
65 65 expandAllRowGroups(el);
66 66 }
67 67 }
68 68
69 69 function toggleFieldset(el) {
70 70 var fieldset = $(el).parents('fieldset').first();
71 71 fieldset.toggleClass('collapsed');
72 72 fieldset.children('div').toggle();
73 73 }
74 74
75 75 function hideFieldset(el) {
76 76 var fieldset = $(el).parents('fieldset').first();
77 77 fieldset.toggleClass('collapsed');
78 78 fieldset.children('div').hide();
79 79 }
80 80
81 81 function initFilters(){
82 82 $('#add_filter_select').change(function(){
83 83 addFilter($(this).val(), '', []);
84 84 });
85 85 $('#filters-table td.field input[type=checkbox]').each(function(){
86 86 toggleFilter($(this).val());
87 87 });
88 88 $('#filters-table td.field input[type=checkbox]').live('click',function(){
89 89 toggleFilter($(this).val());
90 90 });
91 91 $('#filters-table .toggle-multiselect').live('click',function(){
92 92 toggleMultiSelect($(this).siblings('select'));
93 93 });
94 94 $('#filters-table input[type=text]').live('keypress', function(e){
95 95 if (e.keyCode == 13) submit_query_form("query_form");
96 96 });
97 97 }
98 98
99 99 function addFilter(field, operator, values) {
100 100 var fieldId = field.replace('.', '_');
101 101 var tr = $('#tr_'+fieldId);
102 102 if (tr.length > 0) {
103 103 tr.show();
104 104 } else {
105 105 buildFilterRow(field, operator, values);
106 106 }
107 107 $('#cb_'+fieldId).attr('checked', true);
108 108 toggleFilter(field);
109 109 $('#add_filter_select').val('').children('option').each(function(){
110 110 if ($(this).attr('value') == field) {
111 111 $(this).attr('disabled', true);
112 112 }
113 113 });
114 114 }
115 115
116 116 function buildFilterRow(field, operator, values) {
117 117 var fieldId = field.replace('.', '_');
118 118 var filterTable = $("#filters-table");
119 119 var filterOptions = availableFilters[field];
120 120 var operators = operatorByType[filterOptions['type']];
121 121 var filterValues = filterOptions['values'];
122 122 var i, select;
123 123
124 124 var tr = $('<tr class="filter">').attr('id', 'tr_'+fieldId).html(
125 125 '<td class="field"><input checked="checked" id="cb_'+fieldId+'" name="f[]" value="'+field+'" type="checkbox"><label for="cb_'+fieldId+'"> '+filterOptions['name']+'</label></td>' +
126 126 '<td class="operator"><select id="operators_'+fieldId+'" name="op['+field+']"></td>' +
127 127 '<td class="values"></td>'
128 128 );
129 129 filterTable.append(tr);
130 130
131 131 select = tr.find('td.operator select');
132 132 for (i=0;i<operators.length;i++){
133 133 var option = $('<option>').val(operators[i]).text(operatorLabels[operators[i]]);
134 134 if (operators[i] == operator) {option.attr('selected', true)};
135 135 select.append(option);
136 136 }
137 137 select.change(function(){toggleOperator(field)});
138 138
139 139 switch (filterOptions['type']){
140 140 case "list":
141 141 case "list_optional":
142 142 case "list_status":
143 143 case "list_subprojects":
144 144 tr.find('td.values').append(
145 145 '<span style="display:none;"><select class="value" id="values_'+fieldId+'_1" name="v['+field+'][]"></select>' +
146 146 ' <span class="toggle-multiselect">&nbsp;</span></span>'
147 147 );
148 148 select = tr.find('td.values select');
149 149 if (values.length > 1) {select.attr('multiple', true)};
150 150 for (i=0;i<filterValues.length;i++){
151 151 var filterValue = filterValues[i];
152 152 var option = $('<option>');
153 153 if ($.isArray(filterValue)) {
154 154 option.val(filterValue[1]).text(filterValue[0]);
155 155 if ($.inArray(filterValue[1], values) > -1) {option.attr('selected', true);}
156 156 } else {
157 157 option.val(filterValue).text(filterValue);
158 158 if ($.inArray(filterValue, values) > -1) {option.attr('selected', true);}
159 159 }
160 160 select.append(option);
161 161 }
162 162 break;
163 163 case "date":
164 164 case "date_past":
165 165 tr.find('td.values').append(
166 166 '<span style="display:none;"><input type="text" name="v['+field+'][]" id="values_'+fieldId+'_1" size="10" class="value date_value" /></span>' +
167 167 ' <span style="display:none;"><input type="text" name="v['+field+'][]" id="values_'+fieldId+'_2" size="10" class="value date_value" /></span>' +
168 168 ' <span style="display:none;"><input type="text" name="v['+field+'][]" id="values_'+fieldId+'" size="3" class="value" /> '+labelDayPlural+'</span>'
169 169 );
170 170 $('#values_'+fieldId+'_1').val(values[0]).datepicker(datepickerOptions);
171 171 $('#values_'+fieldId+'_2').val(values[1]).datepicker(datepickerOptions);
172 172 $('#values_'+fieldId).val(values[0]);
173 173 break;
174 174 case "string":
175 175 case "text":
176 176 tr.find('td.values').append(
177 177 '<span style="display:none;"><input type="text" name="v['+field+'][]" id="values_'+fieldId+'" size="30" class="value" /></span>'
178 178 );
179 179 $('#values_'+fieldId).val(values[0]);
180 180 break;
181 181 case "relation":
182 182 tr.find('td.values').append(
183 183 '<span style="display:none;"><input type="text" name="v['+field+'][]" id="values_'+fieldId+'" size="6" class="value" /></span>' +
184 184 '<span style="display:none;"><select class="value" name="v['+field+'][]" id="values_'+fieldId+'_1"></select></span>'
185 185 );
186 186 $('#values_'+fieldId).val(values[0]);
187 187 select = tr.find('td.values select');
188 188 for (i=0;i<allProjects.length;i++){
189 189 var filterValue = allProjects[i];
190 190 var option = $('<option>');
191 191 option.val(filterValue[1]).text(filterValue[0]);
192 192 if (values[0] == filterValue[1]) {option.attr('selected', true)};
193 193 select.append(option);
194 194 }
195 195 case "integer":
196 196 case "float":
197 197 tr.find('td.values').append(
198 198 '<span style="display:none;"><input type="text" name="v['+field+'][]" id="values_'+fieldId+'_1" size="6" class="value" /></span>' +
199 199 ' <span style="display:none;"><input type="text" name="v['+field+'][]" id="values_'+fieldId+'_2" size="6" class="value" /></span>'
200 200 );
201 201 $('#values_'+fieldId+'_1').val(values[0]);
202 202 $('#values_'+fieldId+'_2').val(values[1]);
203 203 break;
204 204 }
205 205 }
206 206
207 207 function toggleFilter(field) {
208 208 var fieldId = field.replace('.', '_');
209 209 if ($('#cb_' + fieldId).is(':checked')) {
210 210 $("#operators_" + fieldId).show().removeAttr('disabled');
211 211 toggleOperator(field);
212 212 } else {
213 213 $("#operators_" + fieldId).hide().attr('disabled', true);
214 214 enableValues(field, []);
215 215 }
216 216 }
217 217
218 218 function enableValues(field, indexes) {
219 219 var fieldId = field.replace('.', '_');
220 220 $('#tr_'+fieldId+' td.values .value').each(function(index) {
221 221 if ($.inArray(index, indexes) >= 0) {
222 222 $(this).removeAttr('disabled');
223 223 $(this).parents('span').first().show();
224 224 } else {
225 225 $(this).val('');
226 226 $(this).attr('disabled', true);
227 227 $(this).parents('span').first().hide();
228 228 }
229 229
230 230 if ($(this).hasClass('group')) {
231 231 $(this).addClass('open');
232 232 } else {
233 233 $(this).show();
234 234 }
235 235 });
236 236 }
237 237
238 238 function toggleOperator(field) {
239 239 var fieldId = field.replace('.', '_');
240 240 var operator = $("#operators_" + fieldId);
241 241 switch (operator.val()) {
242 242 case "!*":
243 243 case "*":
244 244 case "t":
245 245 case "w":
246 246 case "o":
247 247 case "c":
248 248 enableValues(field, []);
249 249 break;
250 250 case "><":
251 251 enableValues(field, [0,1]);
252 252 break;
253 253 case "<t+":
254 254 case ">t+":
255 case "><t+":
255 256 case "t+":
256 257 case ">t-":
257 258 case "<t-":
259 case "><t-":
258 260 case "t-":
259 261 enableValues(field, [2]);
260 262 break;
261 263 case "=p":
262 264 case "=!p":
263 265 case "!p":
264 266 enableValues(field, [1]);
265 267 break;
266 268 default:
267 269 enableValues(field, [0]);
268 270 break;
269 271 }
270 272 }
271 273
272 274 function toggleMultiSelect(el) {
273 275 if (el.attr('multiple')) {
274 276 el.removeAttr('multiple');
275 277 } else {
276 278 el.attr('multiple', true);
277 279 }
278 280 }
279 281
280 282 function submit_query_form(id) {
281 283 selectAllOptions("selected_columns");
282 284 $('#'+id).submit();
283 285 }
284 286
285 287 var fileFieldCount = 1;
286 288 function addFileField() {
287 289 var fields = $('#attachments_fields');
288 290 if (fields.children().length >= 10) return false;
289 291 fileFieldCount++;
290 292 var s = fields.children('span').first().clone();
291 293 s.children('input.file').attr('name', "attachments[" + fileFieldCount + "][file]").val('');
292 294 s.children('input.description').attr('name', "attachments[" + fileFieldCount + "][description]").val('');
293 295 fields.append(s);
294 296 }
295 297
296 298 function removeFileField(el) {
297 299 var fields = $('#attachments_fields');
298 300 var s = $(el).parents('span').first();
299 301 if (fields.children().length > 1) {
300 302 s.remove();
301 303 } else {
302 304 s.children('input.file').val('');
303 305 s.children('input.description').val('');
304 306 }
305 307 }
306 308
307 309 function checkFileSize(el, maxSize, message) {
308 310 var files = el.files;
309 311 if (files) {
310 312 for (var i=0; i<files.length; i++) {
311 313 if (files[i].size > maxSize) {
312 314 alert(message);
313 315 el.value = "";
314 316 }
315 317 }
316 318 }
317 319 }
318 320
319 321 function showTab(name) {
320 322 $('div#content .tab-content').hide();
321 323 $('div.tabs a').removeClass('selected');
322 324 $('#tab-content-' + name).show();
323 325 $('#tab-' + name).addClass('selected');
324 326 return false;
325 327 }
326 328
327 329 function moveTabRight(el) {
328 330 var lis = $(el).parents('div.tabs').first().find('ul').children();
329 331 var tabsWidth = 0;
330 332 var i = 0;
331 333 lis.each(function(){
332 334 if ($(this).is(':visible')) {
333 335 tabsWidth += $(this).width() + 6;
334 336 }
335 337 });
336 338 if (tabsWidth < $(el).parents('div.tabs').first().width() - 60) { return; }
337 339 while (i<lis.length && !lis.eq(i).is(':visible')) { i++; }
338 340 lis.eq(i).hide();
339 341 }
340 342
341 343 function moveTabLeft(el) {
342 344 var lis = $(el).parents('div.tabs').first().find('ul').children();
343 345 var i = 0;
344 346 while (i<lis.length && !lis.eq(i).is(':visible')) { i++; }
345 347 if (i>0) {
346 348 lis.eq(i-1).show();
347 349 }
348 350 }
349 351
350 352 function displayTabsButtons() {
351 353 var lis;
352 354 var tabsWidth = 0;
353 355 var el;
354 356 $('div.tabs').each(function() {
355 357 el = $(this);
356 358 lis = el.find('ul').children();
357 359 lis.each(function(){
358 360 if ($(this).is(':visible')) {
359 361 tabsWidth += $(this).width() + 6;
360 362 }
361 363 });
362 364 if ((tabsWidth < el.width() - 60) && (lis.first().is(':visible'))) {
363 365 el.find('div.tabs-buttons').hide();
364 366 } else {
365 367 el.find('div.tabs-buttons').show();
366 368 }
367 369 });
368 370 }
369 371
370 372 function setPredecessorFieldsVisibility() {
371 373 var relationType = $('#relation_relation_type');
372 374 if (relationType.val() == "precedes" || relationType.val() == "follows") {
373 375 $('#predecessor_fields').show();
374 376 } else {
375 377 $('#predecessor_fields').hide();
376 378 }
377 379 }
378 380
379 381 function showModal(id, width) {
380 382 var el = $('#'+id).first();
381 383 if (el.length == 0 || el.is(':visible')) {return;}
382 384 var title = el.find('h3.title').text();
383 385 el.dialog({
384 386 width: width,
385 387 modal: true,
386 388 resizable: false,
387 389 dialogClass: 'modal',
388 390 title: title
389 391 });
390 392 el.find("input[type=text], input[type=submit]").first().focus();
391 393 }
392 394
393 395 function hideModal(el) {
394 396 var modal;
395 397 if (el) {
396 398 modal = $(el).parents('.ui-dialog-content');
397 399 } else {
398 400 modal = $('#ajax-modal');
399 401 }
400 402 modal.dialog("close");
401 403 }
402 404
403 405 function submitPreview(url, form, target) {
404 406 $.ajax({
405 407 url: url,
406 408 type: 'post',
407 409 data: $('#'+form).serialize(),
408 410 success: function(data){
409 411 $('#'+target).html(data);
410 412 }
411 413 });
412 414 }
413 415
414 416 function collapseScmEntry(id) {
415 417 $('.'+id).each(function() {
416 418 if ($(this).hasClass('open')) {
417 419 collapseScmEntry($(this).attr('id'));
418 420 }
419 421 $(this).hide();
420 422 });
421 423 $('#'+id).removeClass('open');
422 424 }
423 425
424 426 function expandScmEntry(id) {
425 427 $('.'+id).each(function() {
426 428 $(this).show();
427 429 if ($(this).hasClass('loaded') && !$(this).hasClass('collapsed')) {
428 430 expandScmEntry($(this).attr('id'));
429 431 }
430 432 });
431 433 $('#'+id).addClass('open');
432 434 }
433 435
434 436 function scmEntryClick(id, url) {
435 437 el = $('#'+id);
436 438 if (el.hasClass('open')) {
437 439 collapseScmEntry(id);
438 440 el.addClass('collapsed');
439 441 return false;
440 442 } else if (el.hasClass('loaded')) {
441 443 expandScmEntry(id);
442 444 el.removeClass('collapsed');
443 445 return false;
444 446 }
445 447 if (el.hasClass('loading')) {
446 448 return false;
447 449 }
448 450 el.addClass('loading');
449 451 $.ajax({
450 452 url: url,
451 453 success: function(data){
452 454 el.after(data);
453 455 el.addClass('open').addClass('loaded').removeClass('loading');
454 456 }
455 457 });
456 458 return true;
457 459 }
458 460
459 461 function randomKey(size) {
460 462 var chars = new Array('0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z', 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z');
461 463 var key = '';
462 464 for (i = 0; i < size; i++) {
463 465 key += chars[Math.floor(Math.random() * chars.length)];
464 466 }
465 467 return key;
466 468 }
467 469
468 470 // Can't use Rails' remote select because we need the form data
469 471 function updateIssueFrom(url) {
470 472 $.ajax({
471 473 url: url,
472 474 type: 'post',
473 475 data: $('#issue-form').serialize()
474 476 });
475 477 }
476 478
477 479 function updateBulkEditFrom(url) {
478 480 $.ajax({
479 481 url: url,
480 482 type: 'post',
481 483 data: $('#bulk_edit_form').serialize()
482 484 });
483 485 }
484 486
485 487 function observeAutocompleteField(fieldId, url) {
486 488 $('#'+fieldId).autocomplete({
487 489 source: url,
488 490 minLength: 2
489 491 });
490 492 }
491 493
492 494 function observeSearchfield(fieldId, targetId, url) {
493 495 $('#'+fieldId).each(function() {
494 496 var $this = $(this);
495 497 $this.attr('data-value-was', $this.val());
496 498 var check = function() {
497 499 var val = $this.val();
498 500 if ($this.attr('data-value-was') != val){
499 501 $this.attr('data-value-was', val);
500 502 $.ajax({
501 503 url: url,
502 504 type: 'get',
503 505 data: {q: $this.val()},
504 506 success: function(data){ $('#'+targetId).html(data); },
505 507 beforeSend: function(){ $this.addClass('ajax-loading'); },
506 508 complete: function(){ $this.removeClass('ajax-loading'); }
507 509 });
508 510 }
509 511 };
510 512 var reset = function() {
511 513 if (timer) {
512 514 clearInterval(timer);
513 515 timer = setInterval(check, 300);
514 516 }
515 517 };
516 518 var timer = setInterval(check, 300);
517 519 $this.bind('keyup click mousemove', reset);
518 520 });
519 521 }
520 522
521 523 function observeProjectModules() {
522 524 var f = function() {
523 525 /* Hides trackers and issues custom fields on the new project form when issue_tracking module is disabled */
524 526 if ($('#project_enabled_module_names_issue_tracking').attr('checked')) {
525 527 $('#project_trackers').show();
526 528 }else{
527 529 $('#project_trackers').hide();
528 530 }
529 531 };
530 532
531 533 $(window).load(f);
532 534 $('#project_enabled_module_names_issue_tracking').change(f);
533 535 }
534 536
535 537 function initMyPageSortable(list, url) {
536 538 $('#list-'+list).sortable({
537 539 connectWith: '.block-receiver',
538 540 tolerance: 'pointer',
539 541 update: function(){
540 542 $.ajax({
541 543 url: url,
542 544 type: 'post',
543 545 data: {'blocks': $.map($('#list-'+list).children(), function(el){return $(el).attr('id');})}
544 546 });
545 547 }
546 548 });
547 549 $("#list-top, #list-left, #list-right").disableSelection();
548 550 }
549 551
550 552 var warnLeavingUnsavedMessage;
551 553 function warnLeavingUnsaved(message) {
552 554 warnLeavingUnsavedMessage = message;
553 555
554 556 $('form').submit(function(){
555 557 $('textarea').removeData('changed');
556 558 });
557 559 $('textarea').change(function(){
558 560 $(this).data('changed', 'changed');
559 561 });
560 562 window.onbeforeunload = function(){
561 563 var warn = false;
562 564 $('textarea').blur().each(function(){
563 565 if ($(this).data('changed')) {
564 566 warn = true;
565 567 }
566 568 });
567 569 if (warn) {return warnLeavingUnsavedMessage;}
568 570 };
569 571 };
570 572
571 573 $(document).ready(function(){
572 574 $('#ajax-indicator').bind('ajaxSend', function(){
573 575 if ($('.ajax-loading').length == 0) {
574 576 $('#ajax-indicator').show();
575 577 }
576 578 });
577 579 $('#ajax-indicator').bind('ajaxStop', function(){
578 580 $('#ajax-indicator').hide();
579 581 });
580 582 });
581 583
582 584 function hideOnLoad() {
583 585 $('.hol').hide();
584 586 }
585 587
586 588 function addFormObserversForDoubleSubmit() {
587 589 $('form[method=post]').each(function() {
588 590 if (!$(this).hasClass('multiple-submit')) {
589 591 $(this).submit(function(form_submission) {
590 592 if ($(form_submission.target).attr('data-submitted')) {
591 593 form_submission.preventDefault();
592 594 } else {
593 595 $(form_submission.target).attr('data-submitted', true);
594 596 }
595 597 });
596 598 }
597 599 });
598 600 }
599 601
600 602 $(document).ready(hideOnLoad);
601 603 $(document).ready(addFormObserversForDoubleSubmit);
@@ -1,1215 +1,1232
1 1 # Redmine - project management software
2 2 # Copyright (C) 2006-2012 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.expand_path('../../test_helper', __FILE__)
19 19
20 20 class QueryTest < ActiveSupport::TestCase
21 21 include Redmine::I18n
22 22
23 23 fixtures :projects, :enabled_modules, :users, :members,
24 24 :member_roles, :roles, :trackers, :issue_statuses,
25 25 :issue_categories, :enumerations, :issues,
26 26 :watchers, :custom_fields, :custom_values, :versions,
27 27 :queries,
28 28 :projects_trackers,
29 29 :custom_fields_trackers
30 30
31 31 def test_custom_fields_for_all_projects_should_be_available_in_global_queries
32 32 query = Query.new(:project => nil, :name => '_')
33 33 assert query.available_filters.has_key?('cf_1')
34 34 assert !query.available_filters.has_key?('cf_3')
35 35 end
36 36
37 37 def test_system_shared_versions_should_be_available_in_global_queries
38 38 Version.find(2).update_attribute :sharing, 'system'
39 39 query = Query.new(:project => nil, :name => '_')
40 40 assert query.available_filters.has_key?('fixed_version_id')
41 41 assert query.available_filters['fixed_version_id'][:values].detect {|v| v.last == '2'}
42 42 end
43 43
44 44 def test_project_filter_in_global_queries
45 45 query = Query.new(:project => nil, :name => '_')
46 46 project_filter = query.available_filters["project_id"]
47 47 assert_not_nil project_filter
48 48 project_ids = project_filter[:values].map{|p| p[1]}
49 49 assert project_ids.include?("1") #public project
50 50 assert !project_ids.include?("2") #private project user cannot see
51 51 end
52 52
53 53 def find_issues_with_query(query)
54 54 Issue.includes([:assigned_to, :status, :tracker, :project, :priority]).where(
55 55 query.statement
56 56 ).all
57 57 end
58 58
59 59 def assert_find_issues_with_query_is_successful(query)
60 60 assert_nothing_raised do
61 61 find_issues_with_query(query)
62 62 end
63 63 end
64 64
65 65 def assert_query_statement_includes(query, condition)
66 66 assert query.statement.include?(condition), "Query statement condition not found in: #{query.statement}"
67 67 end
68 68
69 69 def assert_query_result(expected, query)
70 70 assert_nothing_raised do
71 71 assert_equal expected.map(&:id).sort, query.issues.map(&:id).sort
72 72 assert_equal expected.size, query.issue_count
73 73 end
74 74 end
75 75
76 76 def test_query_should_allow_shared_versions_for_a_project_query
77 77 subproject_version = Version.find(4)
78 78 query = Query.new(:project => Project.find(1), :name => '_')
79 79 query.add_filter('fixed_version_id', '=', [subproject_version.id.to_s])
80 80
81 81 assert query.statement.include?("#{Issue.table_name}.fixed_version_id IN ('4')")
82 82 end
83 83
84 84 def test_query_with_multiple_custom_fields
85 85 query = Query.find(1)
86 86 assert query.valid?
87 87 assert query.statement.include?("#{CustomValue.table_name}.value IN ('MySQL')")
88 88 issues = find_issues_with_query(query)
89 89 assert_equal 1, issues.length
90 90 assert_equal Issue.find(3), issues.first
91 91 end
92 92
93 93 def test_operator_none
94 94 query = Query.new(:project => Project.find(1), :name => '_')
95 95 query.add_filter('fixed_version_id', '!*', [''])
96 96 query.add_filter('cf_1', '!*', [''])
97 97 assert query.statement.include?("#{Issue.table_name}.fixed_version_id IS NULL")
98 98 assert query.statement.include?("#{CustomValue.table_name}.value IS NULL OR #{CustomValue.table_name}.value = ''")
99 99 find_issues_with_query(query)
100 100 end
101 101
102 102 def test_operator_none_for_integer
103 103 query = Query.new(:project => Project.find(1), :name => '_')
104 104 query.add_filter('estimated_hours', '!*', [''])
105 105 issues = find_issues_with_query(query)
106 106 assert !issues.empty?
107 107 assert issues.all? {|i| !i.estimated_hours}
108 108 end
109 109
110 110 def test_operator_none_for_date
111 111 query = Query.new(:project => Project.find(1), :name => '_')
112 112 query.add_filter('start_date', '!*', [''])
113 113 issues = find_issues_with_query(query)
114 114 assert !issues.empty?
115 115 assert issues.all? {|i| i.start_date.nil?}
116 116 end
117 117
118 118 def test_operator_none_for_string_custom_field
119 119 query = Query.new(:project => Project.find(1), :name => '_')
120 120 query.add_filter('cf_2', '!*', [''])
121 121 assert query.has_filter?('cf_2')
122 122 issues = find_issues_with_query(query)
123 123 assert !issues.empty?
124 124 assert issues.all? {|i| i.custom_field_value(2).blank?}
125 125 end
126 126
127 127 def test_operator_all
128 128 query = Query.new(:project => Project.find(1), :name => '_')
129 129 query.add_filter('fixed_version_id', '*', [''])
130 130 query.add_filter('cf_1', '*', [''])
131 131 assert query.statement.include?("#{Issue.table_name}.fixed_version_id IS NOT NULL")
132 132 assert query.statement.include?("#{CustomValue.table_name}.value IS NOT NULL AND #{CustomValue.table_name}.value <> ''")
133 133 find_issues_with_query(query)
134 134 end
135 135
136 136 def test_operator_all_for_date
137 137 query = Query.new(:project => Project.find(1), :name => '_')
138 138 query.add_filter('start_date', '*', [''])
139 139 issues = find_issues_with_query(query)
140 140 assert !issues.empty?
141 141 assert issues.all? {|i| i.start_date.present?}
142 142 end
143 143
144 144 def test_operator_all_for_string_custom_field
145 145 query = Query.new(:project => Project.find(1), :name => '_')
146 146 query.add_filter('cf_2', '*', [''])
147 147 assert query.has_filter?('cf_2')
148 148 issues = find_issues_with_query(query)
149 149 assert !issues.empty?
150 150 assert issues.all? {|i| i.custom_field_value(2).present?}
151 151 end
152 152
153 153 def test_numeric_filter_should_not_accept_non_numeric_values
154 154 query = Query.new(:name => '_')
155 155 query.add_filter('estimated_hours', '=', ['a'])
156 156
157 157 assert query.has_filter?('estimated_hours')
158 158 assert !query.valid?
159 159 end
160 160
161 161 def test_operator_is_on_float
162 162 Issue.update_all("estimated_hours = 171.2", "id=2")
163 163
164 164 query = Query.new(:name => '_')
165 165 query.add_filter('estimated_hours', '=', ['171.20'])
166 166 issues = find_issues_with_query(query)
167 167 assert_equal 1, issues.size
168 168 assert_equal 2, issues.first.id
169 169 end
170 170
171 171 def test_operator_is_on_integer_custom_field
172 172 f = IssueCustomField.create!(:name => 'filter', :field_format => 'int', :is_for_all => true, :is_filter => true)
173 173 CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => '7')
174 174 CustomValue.create!(:custom_field => f, :customized => Issue.find(2), :value => '12')
175 175 CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => '')
176 176
177 177 query = Query.new(:name => '_')
178 178 query.add_filter("cf_#{f.id}", '=', ['12'])
179 179 issues = find_issues_with_query(query)
180 180 assert_equal 1, issues.size
181 181 assert_equal 2, issues.first.id
182 182 end
183 183
184 184 def test_operator_is_on_integer_custom_field_should_accept_negative_value
185 185 f = IssueCustomField.create!(:name => 'filter', :field_format => 'int', :is_for_all => true, :is_filter => true)
186 186 CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => '7')
187 187 CustomValue.create!(:custom_field => f, :customized => Issue.find(2), :value => '-12')
188 188 CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => '')
189 189
190 190 query = Query.new(:name => '_')
191 191 query.add_filter("cf_#{f.id}", '=', ['-12'])
192 192 assert query.valid?
193 193 issues = find_issues_with_query(query)
194 194 assert_equal 1, issues.size
195 195 assert_equal 2, issues.first.id
196 196 end
197 197
198 198 def test_operator_is_on_float_custom_field
199 199 f = IssueCustomField.create!(:name => 'filter', :field_format => 'float', :is_filter => true, :is_for_all => true)
200 200 CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => '7.3')
201 201 CustomValue.create!(:custom_field => f, :customized => Issue.find(2), :value => '12.7')
202 202 CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => '')
203 203
204 204 query = Query.new(:name => '_')
205 205 query.add_filter("cf_#{f.id}", '=', ['12.7'])
206 206 issues = find_issues_with_query(query)
207 207 assert_equal 1, issues.size
208 208 assert_equal 2, issues.first.id
209 209 end
210 210
211 211 def test_operator_is_on_float_custom_field_should_accept_negative_value
212 212 f = IssueCustomField.create!(:name => 'filter', :field_format => 'float', :is_filter => true, :is_for_all => true)
213 213 CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => '7.3')
214 214 CustomValue.create!(:custom_field => f, :customized => Issue.find(2), :value => '-12.7')
215 215 CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => '')
216 216
217 217 query = Query.new(:name => '_')
218 218 query.add_filter("cf_#{f.id}", '=', ['-12.7'])
219 219 assert query.valid?
220 220 issues = find_issues_with_query(query)
221 221 assert_equal 1, issues.size
222 222 assert_equal 2, issues.first.id
223 223 end
224 224
225 225 def test_operator_is_on_multi_list_custom_field
226 226 f = IssueCustomField.create!(:name => 'filter', :field_format => 'list', :is_filter => true, :is_for_all => true,
227 227 :possible_values => ['value1', 'value2', 'value3'], :multiple => true)
228 228 CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => 'value1')
229 229 CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => 'value2')
230 230 CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => 'value1')
231 231
232 232 query = Query.new(:name => '_')
233 233 query.add_filter("cf_#{f.id}", '=', ['value1'])
234 234 issues = find_issues_with_query(query)
235 235 assert_equal [1, 3], issues.map(&:id).sort
236 236
237 237 query = Query.new(:name => '_')
238 238 query.add_filter("cf_#{f.id}", '=', ['value2'])
239 239 issues = find_issues_with_query(query)
240 240 assert_equal [1], issues.map(&:id).sort
241 241 end
242 242
243 243 def test_operator_is_not_on_multi_list_custom_field
244 244 f = IssueCustomField.create!(:name => 'filter', :field_format => 'list', :is_filter => true, :is_for_all => true,
245 245 :possible_values => ['value1', 'value2', 'value3'], :multiple => true)
246 246 CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => 'value1')
247 247 CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => 'value2')
248 248 CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => 'value1')
249 249
250 250 query = Query.new(:name => '_')
251 251 query.add_filter("cf_#{f.id}", '!', ['value1'])
252 252 issues = find_issues_with_query(query)
253 253 assert !issues.map(&:id).include?(1)
254 254 assert !issues.map(&:id).include?(3)
255 255
256 256 query = Query.new(:name => '_')
257 257 query.add_filter("cf_#{f.id}", '!', ['value2'])
258 258 issues = find_issues_with_query(query)
259 259 assert !issues.map(&:id).include?(1)
260 260 assert issues.map(&:id).include?(3)
261 261 end
262 262
263 263 def test_operator_is_on_is_private_field
264 264 # is_private filter only available for those who can set issues private
265 265 User.current = User.find(2)
266 266
267 267 query = Query.new(:name => '_')
268 268 assert query.available_filters.key?('is_private')
269 269
270 270 query.add_filter("is_private", '=', ['1'])
271 271 issues = find_issues_with_query(query)
272 272 assert issues.any?
273 273 assert_nil issues.detect {|issue| !issue.is_private?}
274 274 ensure
275 275 User.current = nil
276 276 end
277 277
278 278 def test_operator_is_not_on_is_private_field
279 279 # is_private filter only available for those who can set issues private
280 280 User.current = User.find(2)
281 281
282 282 query = Query.new(:name => '_')
283 283 assert query.available_filters.key?('is_private')
284 284
285 285 query.add_filter("is_private", '!', ['1'])
286 286 issues = find_issues_with_query(query)
287 287 assert issues.any?
288 288 assert_nil issues.detect {|issue| issue.is_private?}
289 289 ensure
290 290 User.current = nil
291 291 end
292 292
293 293 def test_operator_greater_than
294 294 query = Query.new(:project => Project.find(1), :name => '_')
295 295 query.add_filter('done_ratio', '>=', ['40'])
296 296 assert query.statement.include?("#{Issue.table_name}.done_ratio >= 40.0")
297 297 find_issues_with_query(query)
298 298 end
299 299
300 300 def test_operator_greater_than_a_float
301 301 query = Query.new(:project => Project.find(1), :name => '_')
302 302 query.add_filter('estimated_hours', '>=', ['40.5'])
303 303 assert query.statement.include?("#{Issue.table_name}.estimated_hours >= 40.5")
304 304 find_issues_with_query(query)
305 305 end
306 306
307 307 def test_operator_greater_than_on_int_custom_field
308 308 f = IssueCustomField.create!(:name => 'filter', :field_format => 'int', :is_filter => true, :is_for_all => true)
309 309 CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => '7')
310 310 CustomValue.create!(:custom_field => f, :customized => Issue.find(2), :value => '12')
311 311 CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => '')
312 312
313 313 query = Query.new(:project => Project.find(1), :name => '_')
314 314 query.add_filter("cf_#{f.id}", '>=', ['8'])
315 315 issues = find_issues_with_query(query)
316 316 assert_equal 1, issues.size
317 317 assert_equal 2, issues.first.id
318 318 end
319 319
320 320 def test_operator_lesser_than
321 321 query = Query.new(:project => Project.find(1), :name => '_')
322 322 query.add_filter('done_ratio', '<=', ['30'])
323 323 assert query.statement.include?("#{Issue.table_name}.done_ratio <= 30.0")
324 324 find_issues_with_query(query)
325 325 end
326 326
327 327 def test_operator_lesser_than_on_custom_field
328 328 f = IssueCustomField.create!(:name => 'filter', :field_format => 'int', :is_filter => true, :is_for_all => true)
329 329 query = Query.new(:project => Project.find(1), :name => '_')
330 330 query.add_filter("cf_#{f.id}", '<=', ['30'])
331 331 assert query.statement.include?("CAST(custom_values.value AS decimal(60,3)) <= 30.0")
332 332 find_issues_with_query(query)
333 333 end
334 334
335 335 def test_operator_between
336 336 query = Query.new(:project => Project.find(1), :name => '_')
337 337 query.add_filter('done_ratio', '><', ['30', '40'])
338 338 assert_include "#{Issue.table_name}.done_ratio BETWEEN 30.0 AND 40.0", query.statement
339 339 find_issues_with_query(query)
340 340 end
341 341
342 342 def test_operator_between_on_custom_field
343 343 f = IssueCustomField.create!(:name => 'filter', :field_format => 'int', :is_filter => true, :is_for_all => true)
344 344 query = Query.new(:project => Project.find(1), :name => '_')
345 345 query.add_filter("cf_#{f.id}", '><', ['30', '40'])
346 346 assert_include "CAST(custom_values.value AS decimal(60,3)) BETWEEN 30.0 AND 40.0", query.statement
347 347 find_issues_with_query(query)
348 348 end
349 349
350 350 def test_date_filter_should_not_accept_non_date_values
351 351 query = Query.new(:name => '_')
352 352 query.add_filter('created_on', '=', ['a'])
353 353
354 354 assert query.has_filter?('created_on')
355 355 assert !query.valid?
356 356 end
357 357
358 358 def test_date_filter_should_not_accept_invalid_date_values
359 359 query = Query.new(:name => '_')
360 360 query.add_filter('created_on', '=', ['2011-01-34'])
361 361
362 362 assert query.has_filter?('created_on')
363 363 assert !query.valid?
364 364 end
365 365
366 366 def test_relative_date_filter_should_not_accept_non_integer_values
367 367 query = Query.new(:name => '_')
368 368 query.add_filter('created_on', '>t-', ['a'])
369 369
370 370 assert query.has_filter?('created_on')
371 371 assert !query.valid?
372 372 end
373 373
374 374 def test_operator_date_equals
375 375 query = Query.new(:name => '_')
376 376 query.add_filter('due_date', '=', ['2011-07-10'])
377 377 assert_match /issues\.due_date > '2011-07-09 23:59:59(\.9+)?' AND issues\.due_date <= '2011-07-10 23:59:59(\.9+)?/, query.statement
378 378 find_issues_with_query(query)
379 379 end
380 380
381 381 def test_operator_date_lesser_than
382 382 query = Query.new(:name => '_')
383 383 query.add_filter('due_date', '<=', ['2011-07-10'])
384 384 assert_match /issues\.due_date <= '2011-07-10 23:59:59(\.9+)?/, query.statement
385 385 find_issues_with_query(query)
386 386 end
387 387
388 388 def test_operator_date_greater_than
389 389 query = Query.new(:name => '_')
390 390 query.add_filter('due_date', '>=', ['2011-07-10'])
391 391 assert_match /issues\.due_date > '2011-07-09 23:59:59(\.9+)?'/, query.statement
392 392 find_issues_with_query(query)
393 393 end
394 394
395 395 def test_operator_date_between
396 396 query = Query.new(:name => '_')
397 397 query.add_filter('due_date', '><', ['2011-06-23', '2011-07-10'])
398 398 assert_match /issues\.due_date > '2011-06-22 23:59:59(\.9+)?' AND issues\.due_date <= '2011-07-10 23:59:59(\.9+)?/, query.statement
399 399 find_issues_with_query(query)
400 400 end
401 401
402 402 def test_operator_in_more_than
403 403 Issue.find(7).update_attribute(:due_date, (Date.today + 15))
404 404 query = Query.new(:project => Project.find(1), :name => '_')
405 405 query.add_filter('due_date', '>t+', ['15'])
406 406 issues = find_issues_with_query(query)
407 407 assert !issues.empty?
408 408 issues.each {|issue| assert(issue.due_date >= (Date.today + 15))}
409 409 end
410 410
411 411 def test_operator_in_less_than
412 412 query = Query.new(:project => Project.find(1), :name => '_')
413 413 query.add_filter('due_date', '<t+', ['15'])
414 414 issues = find_issues_with_query(query)
415 415 assert !issues.empty?
416 issues.each {|issue| assert(issue.due_date <= (Date.today + 15))}
417 end
418
419 def test_operator_in_the_next_days
420 query = Query.new(:project => Project.find(1), :name => '_')
421 query.add_filter('due_date', '><t+', ['15'])
422 issues = find_issues_with_query(query)
423 assert !issues.empty?
416 424 issues.each {|issue| assert(issue.due_date >= Date.today && issue.due_date <= (Date.today + 15))}
417 425 end
418 426
419 427 def test_operator_less_than_ago
420 428 Issue.find(7).update_attribute(:due_date, (Date.today - 3))
421 429 query = Query.new(:project => Project.find(1), :name => '_')
422 430 query.add_filter('due_date', '>t-', ['3'])
423 431 issues = find_issues_with_query(query)
424 432 assert !issues.empty?
433 issues.each {|issue| assert(issue.due_date >= (Date.today - 3))}
434 end
435
436 def test_operator_in_the_past_days
437 Issue.find(7).update_attribute(:due_date, (Date.today - 3))
438 query = Query.new(:project => Project.find(1), :name => '_')
439 query.add_filter('due_date', '><t-', ['3'])
440 issues = find_issues_with_query(query)
441 assert !issues.empty?
425 442 issues.each {|issue| assert(issue.due_date >= (Date.today - 3) && issue.due_date <= Date.today)}
426 443 end
427 444
428 445 def test_operator_more_than_ago
429 446 Issue.find(7).update_attribute(:due_date, (Date.today - 10))
430 447 query = Query.new(:project => Project.find(1), :name => '_')
431 448 query.add_filter('due_date', '<t-', ['10'])
432 449 assert query.statement.include?("#{Issue.table_name}.due_date <=")
433 450 issues = find_issues_with_query(query)
434 451 assert !issues.empty?
435 452 issues.each {|issue| assert(issue.due_date <= (Date.today - 10))}
436 453 end
437 454
438 455 def test_operator_in
439 456 Issue.find(7).update_attribute(:due_date, (Date.today + 2))
440 457 query = Query.new(:project => Project.find(1), :name => '_')
441 458 query.add_filter('due_date', 't+', ['2'])
442 459 issues = find_issues_with_query(query)
443 460 assert !issues.empty?
444 461 issues.each {|issue| assert_equal((Date.today + 2), issue.due_date)}
445 462 end
446 463
447 464 def test_operator_ago
448 465 Issue.find(7).update_attribute(:due_date, (Date.today - 3))
449 466 query = Query.new(:project => Project.find(1), :name => '_')
450 467 query.add_filter('due_date', 't-', ['3'])
451 468 issues = find_issues_with_query(query)
452 469 assert !issues.empty?
453 470 issues.each {|issue| assert_equal((Date.today - 3), issue.due_date)}
454 471 end
455 472
456 473 def test_operator_today
457 474 query = Query.new(:project => Project.find(1), :name => '_')
458 475 query.add_filter('due_date', 't', [''])
459 476 issues = find_issues_with_query(query)
460 477 assert !issues.empty?
461 478 issues.each {|issue| assert_equal Date.today, issue.due_date}
462 479 end
463 480
464 481 def test_operator_this_week_on_date
465 482 query = Query.new(:project => Project.find(1), :name => '_')
466 483 query.add_filter('due_date', 'w', [''])
467 484 find_issues_with_query(query)
468 485 end
469 486
470 487 def test_operator_this_week_on_datetime
471 488 query = Query.new(:project => Project.find(1), :name => '_')
472 489 query.add_filter('created_on', 'w', [''])
473 490 find_issues_with_query(query)
474 491 end
475 492
476 493 def test_operator_contains
477 494 query = Query.new(:project => Project.find(1), :name => '_')
478 495 query.add_filter('subject', '~', ['uNable'])
479 496 assert query.statement.include?("LOWER(#{Issue.table_name}.subject) LIKE '%unable%'")
480 497 result = find_issues_with_query(query)
481 498 assert result.empty?
482 499 result.each {|issue| assert issue.subject.downcase.include?('unable') }
483 500 end
484 501
485 502 def test_range_for_this_week_with_week_starting_on_monday
486 503 I18n.locale = :fr
487 504 assert_equal '1', I18n.t(:general_first_day_of_week)
488 505
489 506 Date.stubs(:today).returns(Date.parse('2011-04-29'))
490 507
491 508 query = Query.new(:project => Project.find(1), :name => '_')
492 509 query.add_filter('due_date', 'w', [''])
493 510 assert query.statement.match(/issues\.due_date > '2011-04-24 23:59:59(\.9+)?' AND issues\.due_date <= '2011-05-01 23:59:59(\.9+)?/), "range not found in #{query.statement}"
494 511 I18n.locale = :en
495 512 end
496 513
497 514 def test_range_for_this_week_with_week_starting_on_sunday
498 515 I18n.locale = :en
499 516 assert_equal '7', I18n.t(:general_first_day_of_week)
500 517
501 518 Date.stubs(:today).returns(Date.parse('2011-04-29'))
502 519
503 520 query = Query.new(:project => Project.find(1), :name => '_')
504 521 query.add_filter('due_date', 'w', [''])
505 522 assert query.statement.match(/issues\.due_date > '2011-04-23 23:59:59(\.9+)?' AND issues\.due_date <= '2011-04-30 23:59:59(\.9+)?/), "range not found in #{query.statement}"
506 523 end
507 524
508 525 def test_operator_does_not_contains
509 526 query = Query.new(:project => Project.find(1), :name => '_')
510 527 query.add_filter('subject', '!~', ['uNable'])
511 528 assert query.statement.include?("LOWER(#{Issue.table_name}.subject) NOT LIKE '%unable%'")
512 529 find_issues_with_query(query)
513 530 end
514 531
515 532 def test_filter_assigned_to_me
516 533 user = User.find(2)
517 534 group = Group.find(10)
518 535 User.current = user
519 536 i1 = Issue.generate!(:project_id => 1, :tracker_id => 1, :assigned_to => user)
520 537 i2 = Issue.generate!(:project_id => 1, :tracker_id => 1, :assigned_to => group)
521 538 i3 = Issue.generate!(:project_id => 1, :tracker_id => 1, :assigned_to => Group.find(11))
522 539 group.users << user
523 540
524 541 query = Query.new(:name => '_', :filters => { 'assigned_to_id' => {:operator => '=', :values => ['me']}})
525 542 result = query.issues
526 543 assert_equal Issue.visible.all(:conditions => {:assigned_to_id => ([2] + user.reload.group_ids)}).sort_by(&:id), result.sort_by(&:id)
527 544
528 545 assert result.include?(i1)
529 546 assert result.include?(i2)
530 547 assert !result.include?(i3)
531 548 end
532 549
533 550 def test_user_custom_field_filtered_on_me
534 551 User.current = User.find(2)
535 552 cf = IssueCustomField.create!(:field_format => 'user', :is_for_all => true, :is_filter => true, :name => 'User custom field', :tracker_ids => [1])
536 553 issue1 = Issue.create!(:project_id => 1, :tracker_id => 1, :custom_field_values => {cf.id.to_s => '2'}, :subject => 'Test', :author_id => 1)
537 554 issue2 = Issue.generate!(:project_id => 1, :tracker_id => 1, :custom_field_values => {cf.id.to_s => '3'})
538 555
539 556 query = Query.new(:name => '_', :project => Project.find(1))
540 557 filter = query.available_filters["cf_#{cf.id}"]
541 558 assert_not_nil filter
542 559 assert_include 'me', filter[:values].map{|v| v[1]}
543 560
544 561 query.filters = { "cf_#{cf.id}" => {:operator => '=', :values => ['me']}}
545 562 result = query.issues
546 563 assert_equal 1, result.size
547 564 assert_equal issue1, result.first
548 565 end
549 566
550 567 def test_filter_my_projects
551 568 User.current = User.find(2)
552 569 query = Query.new(:name => '_')
553 570 filter = query.available_filters['project_id']
554 571 assert_not_nil filter
555 572 assert_include 'mine', filter[:values].map{|v| v[1]}
556 573
557 574 query.filters = { 'project_id' => {:operator => '=', :values => ['mine']}}
558 575 result = query.issues
559 576 assert_nil result.detect {|issue| !User.current.member_of?(issue.project)}
560 577 end
561 578
562 579 def test_filter_watched_issues
563 580 User.current = User.find(1)
564 581 query = Query.new(:name => '_', :filters => { 'watcher_id' => {:operator => '=', :values => ['me']}})
565 582 result = find_issues_with_query(query)
566 583 assert_not_nil result
567 584 assert !result.empty?
568 585 assert_equal Issue.visible.watched_by(User.current).sort_by(&:id), result.sort_by(&:id)
569 586 User.current = nil
570 587 end
571 588
572 589 def test_filter_unwatched_issues
573 590 User.current = User.find(1)
574 591 query = Query.new(:name => '_', :filters => { 'watcher_id' => {:operator => '!', :values => ['me']}})
575 592 result = find_issues_with_query(query)
576 593 assert_not_nil result
577 594 assert !result.empty?
578 595 assert_equal((Issue.visible - Issue.watched_by(User.current)).sort_by(&:id).size, result.sort_by(&:id).size)
579 596 User.current = nil
580 597 end
581 598
582 599 def test_filter_on_project_custom_field
583 600 field = ProjectCustomField.create!(:name => 'Client', :is_filter => true, :field_format => 'string')
584 601 CustomValue.create!(:custom_field => field, :customized => Project.find(3), :value => 'Foo')
585 602 CustomValue.create!(:custom_field => field, :customized => Project.find(5), :value => 'Foo')
586 603
587 604 query = Query.new(:name => '_')
588 605 filter_name = "project.cf_#{field.id}"
589 606 assert_include filter_name, query.available_filters.keys
590 607 query.filters = {filter_name => {:operator => '=', :values => ['Foo']}}
591 608 assert_equal [3, 5], find_issues_with_query(query).map(&:project_id).uniq.sort
592 609 end
593 610
594 611 def test_filter_on_author_custom_field
595 612 field = UserCustomField.create!(:name => 'Client', :is_filter => true, :field_format => 'string')
596 613 CustomValue.create!(:custom_field => field, :customized => User.find(3), :value => 'Foo')
597 614
598 615 query = Query.new(:name => '_')
599 616 filter_name = "author.cf_#{field.id}"
600 617 assert_include filter_name, query.available_filters.keys
601 618 query.filters = {filter_name => {:operator => '=', :values => ['Foo']}}
602 619 assert_equal [3], find_issues_with_query(query).map(&:author_id).uniq.sort
603 620 end
604 621
605 622 def test_filter_on_assigned_to_custom_field
606 623 field = UserCustomField.create!(:name => 'Client', :is_filter => true, :field_format => 'string')
607 624 CustomValue.create!(:custom_field => field, :customized => User.find(3), :value => 'Foo')
608 625
609 626 query = Query.new(:name => '_')
610 627 filter_name = "assigned_to.cf_#{field.id}"
611 628 assert_include filter_name, query.available_filters.keys
612 629 query.filters = {filter_name => {:operator => '=', :values => ['Foo']}}
613 630 assert_equal [3], find_issues_with_query(query).map(&:assigned_to_id).uniq.sort
614 631 end
615 632
616 633 def test_filter_on_fixed_version_custom_field
617 634 field = VersionCustomField.create!(:name => 'Client', :is_filter => true, :field_format => 'string')
618 635 CustomValue.create!(:custom_field => field, :customized => Version.find(2), :value => 'Foo')
619 636
620 637 query = Query.new(:name => '_')
621 638 filter_name = "fixed_version.cf_#{field.id}"
622 639 assert_include filter_name, query.available_filters.keys
623 640 query.filters = {filter_name => {:operator => '=', :values => ['Foo']}}
624 641 assert_equal [2], find_issues_with_query(query).map(&:fixed_version_id).uniq.sort
625 642 end
626 643
627 644 def test_filter_on_relations_with_a_specific_issue
628 645 IssueRelation.delete_all
629 646 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Issue.find(2))
630 647 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(3), :issue_to => Issue.find(1))
631 648
632 649 query = Query.new(:name => '_')
633 650 query.filters = {"relates" => {:operator => '=', :values => ['1']}}
634 651 assert_equal [2, 3], find_issues_with_query(query).map(&:id).sort
635 652
636 653 query = Query.new(:name => '_')
637 654 query.filters = {"relates" => {:operator => '=', :values => ['2']}}
638 655 assert_equal [1], find_issues_with_query(query).map(&:id).sort
639 656 end
640 657
641 658 def test_filter_on_relations_with_any_issues_in_a_project
642 659 IssueRelation.delete_all
643 660 with_settings :cross_project_issue_relations => '1' do
644 661 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Project.find(2).issues.first)
645 662 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(2), :issue_to => Project.find(2).issues.first)
646 663 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Project.find(3).issues.first)
647 664 end
648 665
649 666 query = Query.new(:name => '_')
650 667 query.filters = {"relates" => {:operator => '=p', :values => ['2']}}
651 668 assert_equal [1, 2], find_issues_with_query(query).map(&:id).sort
652 669
653 670 query = Query.new(:name => '_')
654 671 query.filters = {"relates" => {:operator => '=p', :values => ['3']}}
655 672 assert_equal [1], find_issues_with_query(query).map(&:id).sort
656 673
657 674 query = Query.new(:name => '_')
658 675 query.filters = {"relates" => {:operator => '=p', :values => ['4']}}
659 676 assert_equal [], find_issues_with_query(query).map(&:id).sort
660 677 end
661 678
662 679 def test_filter_on_relations_with_any_issues_not_in_a_project
663 680 IssueRelation.delete_all
664 681 with_settings :cross_project_issue_relations => '1' do
665 682 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Project.find(2).issues.first)
666 683 #IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(2), :issue_to => Project.find(1).issues.first)
667 684 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Project.find(3).issues.first)
668 685 end
669 686
670 687 query = Query.new(:name => '_')
671 688 query.filters = {"relates" => {:operator => '=!p', :values => ['1']}}
672 689 assert_equal [1], find_issues_with_query(query).map(&:id).sort
673 690 end
674 691
675 692 def test_filter_on_relations_with_no_issues_in_a_project
676 693 IssueRelation.delete_all
677 694 with_settings :cross_project_issue_relations => '1' do
678 695 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Project.find(2).issues.first)
679 696 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(2), :issue_to => Project.find(3).issues.first)
680 697 IssueRelation.create!(:relation_type => "relates", :issue_to => Project.find(2).issues.first, :issue_from => Issue.find(3))
681 698 end
682 699
683 700 query = Query.new(:name => '_')
684 701 query.filters = {"relates" => {:operator => '!p', :values => ['2']}}
685 702 ids = find_issues_with_query(query).map(&:id).sort
686 703 assert_include 2, ids
687 704 assert_not_include 1, ids
688 705 assert_not_include 3, ids
689 706 end
690 707
691 708 def test_filter_on_relations_with_no_issues
692 709 IssueRelation.delete_all
693 710 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Issue.find(2))
694 711 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(3), :issue_to => Issue.find(1))
695 712
696 713 query = Query.new(:name => '_')
697 714 query.filters = {"relates" => {:operator => '!*', :values => ['']}}
698 715 ids = find_issues_with_query(query).map(&:id)
699 716 assert_equal [], ids & [1, 2, 3]
700 717 assert_include 4, ids
701 718 end
702 719
703 720 def test_filter_on_relations_with_any_issues
704 721 IssueRelation.delete_all
705 722 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Issue.find(2))
706 723 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(3), :issue_to => Issue.find(1))
707 724
708 725 query = Query.new(:name => '_')
709 726 query.filters = {"relates" => {:operator => '*', :values => ['']}}
710 727 assert_equal [1, 2, 3], find_issues_with_query(query).map(&:id).sort
711 728 end
712 729
713 730 def test_statement_should_be_nil_with_no_filters
714 731 q = Query.new(:name => '_')
715 732 q.filters = {}
716 733
717 734 assert q.valid?
718 735 assert_nil q.statement
719 736 end
720 737
721 738 def test_default_columns
722 739 q = Query.new
723 740 assert !q.columns.empty?
724 741 end
725 742
726 743 def test_set_column_names
727 744 q = Query.new
728 745 q.column_names = ['tracker', :subject, '', 'unknonw_column']
729 746 assert_equal [:tracker, :subject], q.columns.collect {|c| c.name}
730 747 c = q.columns.first
731 748 assert q.has_column?(c)
732 749 end
733 750
734 751 def test_query_should_preload_spent_hours
735 752 q = Query.new(:name => '_', :column_names => [:subject, :spent_hours])
736 753 assert q.has_column?(:spent_hours)
737 754 issues = q.issues
738 755 assert_not_nil issues.first.instance_variable_get("@spent_hours")
739 756 end
740 757
741 758 def test_groupable_columns_should_include_custom_fields
742 759 q = Query.new
743 760 column = q.groupable_columns.detect {|c| c.name == :cf_1}
744 761 assert_not_nil column
745 762 assert_kind_of QueryCustomFieldColumn, column
746 763 end
747 764
748 765 def test_groupable_columns_should_not_include_multi_custom_fields
749 766 field = CustomField.find(1)
750 767 field.update_attribute :multiple, true
751 768
752 769 q = Query.new
753 770 column = q.groupable_columns.detect {|c| c.name == :cf_1}
754 771 assert_nil column
755 772 end
756 773
757 774 def test_groupable_columns_should_include_user_custom_fields
758 775 cf = IssueCustomField.create!(:name => 'User', :is_for_all => true, :tracker_ids => [1], :field_format => 'user')
759 776
760 777 q = Query.new
761 778 assert q.groupable_columns.detect {|c| c.name == "cf_#{cf.id}".to_sym}
762 779 end
763 780
764 781 def test_groupable_columns_should_include_version_custom_fields
765 782 cf = IssueCustomField.create!(:name => 'User', :is_for_all => true, :tracker_ids => [1], :field_format => 'version')
766 783
767 784 q = Query.new
768 785 assert q.groupable_columns.detect {|c| c.name == "cf_#{cf.id}".to_sym}
769 786 end
770 787
771 788 def test_grouped_with_valid_column
772 789 q = Query.new(:group_by => 'status')
773 790 assert q.grouped?
774 791 assert_not_nil q.group_by_column
775 792 assert_equal :status, q.group_by_column.name
776 793 assert_not_nil q.group_by_statement
777 794 assert_equal 'status', q.group_by_statement
778 795 end
779 796
780 797 def test_grouped_with_invalid_column
781 798 q = Query.new(:group_by => 'foo')
782 799 assert !q.grouped?
783 800 assert_nil q.group_by_column
784 801 assert_nil q.group_by_statement
785 802 end
786 803
787 804 def test_sortable_columns_should_sort_assignees_according_to_user_format_setting
788 805 with_settings :user_format => 'lastname_coma_firstname' do
789 806 q = Query.new
790 807 assert q.sortable_columns.has_key?('assigned_to')
791 808 assert_equal %w(users.lastname users.firstname users.id), q.sortable_columns['assigned_to']
792 809 end
793 810 end
794 811
795 812 def test_sortable_columns_should_sort_authors_according_to_user_format_setting
796 813 with_settings :user_format => 'lastname_coma_firstname' do
797 814 q = Query.new
798 815 assert q.sortable_columns.has_key?('author')
799 816 assert_equal %w(authors.lastname authors.firstname authors.id), q.sortable_columns['author']
800 817 end
801 818 end
802 819
803 820 def test_sortable_columns_should_include_custom_field
804 821 q = Query.new
805 822 assert q.sortable_columns['cf_1']
806 823 end
807 824
808 825 def test_sortable_columns_should_not_include_multi_custom_field
809 826 field = CustomField.find(1)
810 827 field.update_attribute :multiple, true
811 828
812 829 q = Query.new
813 830 assert !q.sortable_columns['cf_1']
814 831 end
815 832
816 833 def test_default_sort
817 834 q = Query.new
818 835 assert_equal [], q.sort_criteria
819 836 end
820 837
821 838 def test_set_sort_criteria_with_hash
822 839 q = Query.new
823 840 q.sort_criteria = {'0' => ['priority', 'desc'], '2' => ['tracker']}
824 841 assert_equal [['priority', 'desc'], ['tracker', 'asc']], q.sort_criteria
825 842 end
826 843
827 844 def test_set_sort_criteria_with_array
828 845 q = Query.new
829 846 q.sort_criteria = [['priority', 'desc'], 'tracker']
830 847 assert_equal [['priority', 'desc'], ['tracker', 'asc']], q.sort_criteria
831 848 end
832 849
833 850 def test_create_query_with_sort
834 851 q = Query.new(:name => 'Sorted')
835 852 q.sort_criteria = [['priority', 'desc'], 'tracker']
836 853 assert q.save
837 854 q.reload
838 855 assert_equal [['priority', 'desc'], ['tracker', 'asc']], q.sort_criteria
839 856 end
840 857
841 858 def test_sort_by_string_custom_field_asc
842 859 q = Query.new
843 860 c = q.available_columns.find {|col| col.is_a?(QueryCustomFieldColumn) && col.custom_field.field_format == 'string' }
844 861 assert c
845 862 assert c.sortable
846 863 issues = Issue.includes([:assigned_to, :status, :tracker, :project, :priority]).where(
847 864 q.statement
848 865 ).order("#{c.sortable} ASC").all
849 866 values = issues.collect {|i| i.custom_value_for(c.custom_field).to_s}
850 867 assert !values.empty?
851 868 assert_equal values.sort, values
852 869 end
853 870
854 871 def test_sort_by_string_custom_field_desc
855 872 q = Query.new
856 873 c = q.available_columns.find {|col| col.is_a?(QueryCustomFieldColumn) && col.custom_field.field_format == 'string' }
857 874 assert c
858 875 assert c.sortable
859 876 issues = Issue.includes([:assigned_to, :status, :tracker, :project, :priority]).where(
860 877 q.statement
861 878 ).order("#{c.sortable} DESC").all
862 879 values = issues.collect {|i| i.custom_value_for(c.custom_field).to_s}
863 880 assert !values.empty?
864 881 assert_equal values.sort.reverse, values
865 882 end
866 883
867 884 def test_sort_by_float_custom_field_asc
868 885 q = Query.new
869 886 c = q.available_columns.find {|col| col.is_a?(QueryCustomFieldColumn) && col.custom_field.field_format == 'float' }
870 887 assert c
871 888 assert c.sortable
872 889 issues = Issue.includes([:assigned_to, :status, :tracker, :project, :priority]).where(
873 890 q.statement
874 891 ).order("#{c.sortable} ASC").all
875 892 values = issues.collect {|i| begin; Kernel.Float(i.custom_value_for(c.custom_field).to_s); rescue; nil; end}.compact
876 893 assert !values.empty?
877 894 assert_equal values.sort, values
878 895 end
879 896
880 897 def test_invalid_query_should_raise_query_statement_invalid_error
881 898 q = Query.new
882 899 assert_raise Query::StatementInvalid do
883 900 q.issues(:conditions => "foo = 1")
884 901 end
885 902 end
886 903
887 904 def test_issue_count
888 905 q = Query.new(:name => '_')
889 906 issue_count = q.issue_count
890 907 assert_equal q.issues.size, issue_count
891 908 end
892 909
893 910 def test_issue_count_with_archived_issues
894 911 p = Project.generate! do |project|
895 912 project.status = Project::STATUS_ARCHIVED
896 913 end
897 914 i = Issue.generate!( :project => p, :tracker => p.trackers.first )
898 915 assert !i.visible?
899 916
900 917 test_issue_count
901 918 end
902 919
903 920 def test_issue_count_by_association_group
904 921 q = Query.new(:name => '_', :group_by => 'assigned_to')
905 922 count_by_group = q.issue_count_by_group
906 923 assert_kind_of Hash, count_by_group
907 924 assert_equal %w(NilClass User), count_by_group.keys.collect {|k| k.class.name}.uniq.sort
908 925 assert_equal %w(Fixnum), count_by_group.values.collect {|k| k.class.name}.uniq
909 926 assert count_by_group.has_key?(User.find(3))
910 927 end
911 928
912 929 def test_issue_count_by_list_custom_field_group
913 930 q = Query.new(:name => '_', :group_by => 'cf_1')
914 931 count_by_group = q.issue_count_by_group
915 932 assert_kind_of Hash, count_by_group
916 933 assert_equal %w(NilClass String), count_by_group.keys.collect {|k| k.class.name}.uniq.sort
917 934 assert_equal %w(Fixnum), count_by_group.values.collect {|k| k.class.name}.uniq
918 935 assert count_by_group.has_key?('MySQL')
919 936 end
920 937
921 938 def test_issue_count_by_date_custom_field_group
922 939 q = Query.new(:name => '_', :group_by => 'cf_8')
923 940 count_by_group = q.issue_count_by_group
924 941 assert_kind_of Hash, count_by_group
925 942 assert_equal %w(Date NilClass), count_by_group.keys.collect {|k| k.class.name}.uniq.sort
926 943 assert_equal %w(Fixnum), count_by_group.values.collect {|k| k.class.name}.uniq
927 944 end
928 945
929 946 def test_issue_count_with_nil_group_only
930 947 Issue.update_all("assigned_to_id = NULL")
931 948
932 949 q = Query.new(:name => '_', :group_by => 'assigned_to')
933 950 count_by_group = q.issue_count_by_group
934 951 assert_kind_of Hash, count_by_group
935 952 assert_equal 1, count_by_group.keys.size
936 953 assert_nil count_by_group.keys.first
937 954 end
938 955
939 956 def test_issue_ids
940 957 q = Query.new(:name => '_')
941 958 order = "issues.subject, issues.id"
942 959 issues = q.issues(:order => order)
943 960 assert_equal issues.map(&:id), q.issue_ids(:order => order)
944 961 end
945 962
946 963 def test_label_for
947 964 set_language_if_valid 'en'
948 965 q = Query.new
949 966 assert_equal 'Assignee', q.label_for('assigned_to_id')
950 967 end
951 968
952 969 def test_label_for_fr
953 970 set_language_if_valid 'fr'
954 971 q = Query.new
955 972 s = "Assign\xc3\xa9 \xc3\xa0"
956 973 s.force_encoding('UTF-8') if s.respond_to?(:force_encoding)
957 974 assert_equal s, q.label_for('assigned_to_id')
958 975 end
959 976
960 977 def test_editable_by
961 978 admin = User.find(1)
962 979 manager = User.find(2)
963 980 developer = User.find(3)
964 981
965 982 # Public query on project 1
966 983 q = Query.find(1)
967 984 assert q.editable_by?(admin)
968 985 assert q.editable_by?(manager)
969 986 assert !q.editable_by?(developer)
970 987
971 988 # Private query on project 1
972 989 q = Query.find(2)
973 990 assert q.editable_by?(admin)
974 991 assert !q.editable_by?(manager)
975 992 assert q.editable_by?(developer)
976 993
977 994 # Private query for all projects
978 995 q = Query.find(3)
979 996 assert q.editable_by?(admin)
980 997 assert !q.editable_by?(manager)
981 998 assert q.editable_by?(developer)
982 999
983 1000 # Public query for all projects
984 1001 q = Query.find(4)
985 1002 assert q.editable_by?(admin)
986 1003 assert !q.editable_by?(manager)
987 1004 assert !q.editable_by?(developer)
988 1005 end
989 1006
990 1007 def test_visible_scope
991 1008 query_ids = Query.visible(User.anonymous).map(&:id)
992 1009
993 1010 assert query_ids.include?(1), 'public query on public project was not visible'
994 1011 assert query_ids.include?(4), 'public query for all projects was not visible'
995 1012 assert !query_ids.include?(2), 'private query on public project was visible'
996 1013 assert !query_ids.include?(3), 'private query for all projects was visible'
997 1014 assert !query_ids.include?(7), 'public query on private project was visible'
998 1015 end
999 1016
1000 1017 context "#available_filters" do
1001 1018 setup do
1002 1019 @query = Query.new(:name => "_")
1003 1020 end
1004 1021
1005 1022 should "include users of visible projects in cross-project view" do
1006 1023 users = @query.available_filters["assigned_to_id"]
1007 1024 assert_not_nil users
1008 1025 assert users[:values].map{|u|u[1]}.include?("3")
1009 1026 end
1010 1027
1011 1028 should "include users of subprojects" do
1012 1029 user1 = User.generate!
1013 1030 user2 = User.generate!
1014 1031 project = Project.find(1)
1015 1032 Member.create!(:principal => user1, :project => project.children.visible.first, :role_ids => [1])
1016 1033 @query.project = project
1017 1034
1018 1035 users = @query.available_filters["assigned_to_id"]
1019 1036 assert_not_nil users
1020 1037 assert users[:values].map{|u|u[1]}.include?(user1.id.to_s)
1021 1038 assert !users[:values].map{|u|u[1]}.include?(user2.id.to_s)
1022 1039 end
1023 1040
1024 1041 should "include visible projects in cross-project view" do
1025 1042 projects = @query.available_filters["project_id"]
1026 1043 assert_not_nil projects
1027 1044 assert projects[:values].map{|u|u[1]}.include?("1")
1028 1045 end
1029 1046
1030 1047 context "'member_of_group' filter" do
1031 1048 should "be present" do
1032 1049 assert @query.available_filters.keys.include?("member_of_group")
1033 1050 end
1034 1051
1035 1052 should "be an optional list" do
1036 1053 assert_equal :list_optional, @query.available_filters["member_of_group"][:type]
1037 1054 end
1038 1055
1039 1056 should "have a list of the groups as values" do
1040 1057 Group.destroy_all # No fixtures
1041 1058 group1 = Group.generate!.reload
1042 1059 group2 = Group.generate!.reload
1043 1060
1044 1061 expected_group_list = [
1045 1062 [group1.name, group1.id.to_s],
1046 1063 [group2.name, group2.id.to_s]
1047 1064 ]
1048 1065 assert_equal expected_group_list.sort, @query.available_filters["member_of_group"][:values].sort
1049 1066 end
1050 1067
1051 1068 end
1052 1069
1053 1070 context "'assigned_to_role' filter" do
1054 1071 should "be present" do
1055 1072 assert @query.available_filters.keys.include?("assigned_to_role")
1056 1073 end
1057 1074
1058 1075 should "be an optional list" do
1059 1076 assert_equal :list_optional, @query.available_filters["assigned_to_role"][:type]
1060 1077 end
1061 1078
1062 1079 should "have a list of the Roles as values" do
1063 1080 assert @query.available_filters["assigned_to_role"][:values].include?(['Manager','1'])
1064 1081 assert @query.available_filters["assigned_to_role"][:values].include?(['Developer','2'])
1065 1082 assert @query.available_filters["assigned_to_role"][:values].include?(['Reporter','3'])
1066 1083 end
1067 1084
1068 1085 should "not include the built in Roles as values" do
1069 1086 assert ! @query.available_filters["assigned_to_role"][:values].include?(['Non member','4'])
1070 1087 assert ! @query.available_filters["assigned_to_role"][:values].include?(['Anonymous','5'])
1071 1088 end
1072 1089
1073 1090 end
1074 1091
1075 1092 end
1076 1093
1077 1094 context "#statement" do
1078 1095 context "with 'member_of_group' filter" do
1079 1096 setup do
1080 1097 Group.destroy_all # No fixtures
1081 1098 @user_in_group = User.generate!
1082 1099 @second_user_in_group = User.generate!
1083 1100 @user_in_group2 = User.generate!
1084 1101 @user_not_in_group = User.generate!
1085 1102
1086 1103 @group = Group.generate!.reload
1087 1104 @group.users << @user_in_group
1088 1105 @group.users << @second_user_in_group
1089 1106
1090 1107 @group2 = Group.generate!.reload
1091 1108 @group2.users << @user_in_group2
1092 1109
1093 1110 end
1094 1111
1095 1112 should "search assigned to for users in the group" do
1096 1113 @query = Query.new(:name => '_')
1097 1114 @query.add_filter('member_of_group', '=', [@group.id.to_s])
1098 1115
1099 1116 assert_query_statement_includes @query, "#{Issue.table_name}.assigned_to_id IN ('#{@user_in_group.id}','#{@second_user_in_group.id}')"
1100 1117 assert_find_issues_with_query_is_successful @query
1101 1118 end
1102 1119
1103 1120 should "search not assigned to any group member (none)" do
1104 1121 @query = Query.new(:name => '_')
1105 1122 @query.add_filter('member_of_group', '!*', [''])
1106 1123
1107 1124 # Users not in a group
1108 1125 assert_query_statement_includes @query, "#{Issue.table_name}.assigned_to_id IS NULL OR #{Issue.table_name}.assigned_to_id NOT IN ('#{@user_in_group.id}','#{@second_user_in_group.id}','#{@user_in_group2.id}')"
1109 1126 assert_find_issues_with_query_is_successful @query
1110 1127 end
1111 1128
1112 1129 should "search assigned to any group member (all)" do
1113 1130 @query = Query.new(:name => '_')
1114 1131 @query.add_filter('member_of_group', '*', [''])
1115 1132
1116 1133 # Only users in a group
1117 1134 assert_query_statement_includes @query, "#{Issue.table_name}.assigned_to_id IN ('#{@user_in_group.id}','#{@second_user_in_group.id}','#{@user_in_group2.id}')"
1118 1135 assert_find_issues_with_query_is_successful @query
1119 1136 end
1120 1137
1121 1138 should "return an empty set with = empty group" do
1122 1139 @empty_group = Group.generate!
1123 1140 @query = Query.new(:name => '_')
1124 1141 @query.add_filter('member_of_group', '=', [@empty_group.id.to_s])
1125 1142
1126 1143 assert_equal [], find_issues_with_query(@query)
1127 1144 end
1128 1145
1129 1146 should "return issues with ! empty group" do
1130 1147 @empty_group = Group.generate!
1131 1148 @query = Query.new(:name => '_')
1132 1149 @query.add_filter('member_of_group', '!', [@empty_group.id.to_s])
1133 1150
1134 1151 assert_find_issues_with_query_is_successful @query
1135 1152 end
1136 1153 end
1137 1154
1138 1155 context "with 'assigned_to_role' filter" do
1139 1156 setup do
1140 1157 @manager_role = Role.find_by_name('Manager')
1141 1158 @developer_role = Role.find_by_name('Developer')
1142 1159
1143 1160 @project = Project.generate!
1144 1161 @manager = User.generate!
1145 1162 @developer = User.generate!
1146 1163 @boss = User.generate!
1147 1164 @guest = User.generate!
1148 1165 User.add_to_project(@manager, @project, @manager_role)
1149 1166 User.add_to_project(@developer, @project, @developer_role)
1150 1167 User.add_to_project(@boss, @project, [@manager_role, @developer_role])
1151 1168
1152 1169 @issue1 = Issue.generate!(:project => @project, :assigned_to_id => @manager.id)
1153 1170 @issue2 = Issue.generate!(:project => @project, :assigned_to_id => @developer.id)
1154 1171 @issue3 = Issue.generate!(:project => @project, :assigned_to_id => @boss.id)
1155 1172 @issue4 = Issue.generate!(:project => @project, :assigned_to_id => @guest.id)
1156 1173 @issue5 = Issue.generate!(:project => @project)
1157 1174 end
1158 1175
1159 1176 should "search assigned to for users with the Role" do
1160 1177 @query = Query.new(:name => '_', :project => @project)
1161 1178 @query.add_filter('assigned_to_role', '=', [@manager_role.id.to_s])
1162 1179
1163 1180 assert_query_result [@issue1, @issue3], @query
1164 1181 end
1165 1182
1166 1183 should "search assigned to for users with the Role on the issue project" do
1167 1184 other_project = Project.generate!
1168 1185 User.add_to_project(@developer, other_project, @manager_role)
1169 1186
1170 1187 @query = Query.new(:name => '_', :project => @project)
1171 1188 @query.add_filter('assigned_to_role', '=', [@manager_role.id.to_s])
1172 1189
1173 1190 assert_query_result [@issue1, @issue3], @query
1174 1191 end
1175 1192
1176 1193 should "return an empty set with empty role" do
1177 1194 @empty_role = Role.generate!
1178 1195 @query = Query.new(:name => '_', :project => @project)
1179 1196 @query.add_filter('assigned_to_role', '=', [@empty_role.id.to_s])
1180 1197
1181 1198 assert_query_result [], @query
1182 1199 end
1183 1200
1184 1201 should "search assigned to for users without the Role" do
1185 1202 @query = Query.new(:name => '_', :project => @project)
1186 1203 @query.add_filter('assigned_to_role', '!', [@manager_role.id.to_s])
1187 1204
1188 1205 assert_query_result [@issue2, @issue4, @issue5], @query
1189 1206 end
1190 1207
1191 1208 should "search assigned to for users not assigned to any Role (none)" do
1192 1209 @query = Query.new(:name => '_', :project => @project)
1193 1210 @query.add_filter('assigned_to_role', '!*', [''])
1194 1211
1195 1212 assert_query_result [@issue4, @issue5], @query
1196 1213 end
1197 1214
1198 1215 should "search assigned to for users assigned to any Role (all)" do
1199 1216 @query = Query.new(:name => '_', :project => @project)
1200 1217 @query.add_filter('assigned_to_role', '*', [''])
1201 1218
1202 1219 assert_query_result [@issue1, @issue2, @issue3], @query
1203 1220 end
1204 1221
1205 1222 should "return issues with ! empty role" do
1206 1223 @empty_role = Role.generate!
1207 1224 @query = Query.new(:name => '_', :project => @project)
1208 1225 @query.add_filter('assigned_to_role', '!', [@empty_role.id.to_s])
1209 1226
1210 1227 assert_query_result [@issue1, @issue2, @issue3, @issue4, @issue5], @query
1211 1228 end
1212 1229 end
1213 1230 end
1214 1231
1215 1232 end
General Comments 0
You need to be logged in to leave comments. Login now