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