##// END OF EJS Templates
Filters on chained custom fields and custom field attributes (#21249)....
Jean-Philippe Lang -
r15809:f1678e4f778c
parent child
Show More
@@ -1,355 +1,357
1 # encoding: utf-8
1 # encoding: utf-8
2 #
2 #
3 # Redmine - project management software
3 # Redmine - project management software
4 # Copyright (C) 2006-2016 Jean-Philippe Lang
4 # Copyright (C) 2006-2016 Jean-Philippe Lang
5 #
5 #
6 # This program is free software; you can redistribute it and/or
6 # This program is free software; you can redistribute it and/or
7 # modify it under the terms of the GNU General Public License
7 # modify it under the terms of the GNU General Public License
8 # as published by the Free Software Foundation; either version 2
8 # as published by the Free Software Foundation; either version 2
9 # of the License, or (at your option) any later version.
9 # of the License, or (at your option) any later version.
10 #
10 #
11 # This program is distributed in the hope that it will be useful,
11 # This program is distributed in the hope that it will be useful,
12 # but WITHOUT ANY WARRANTY; without even the implied warranty of
12 # but WITHOUT ANY WARRANTY; without even the implied warranty of
13 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 # GNU General Public License for more details.
14 # GNU General Public License for more details.
15 #
15 #
16 # You should have received a copy of the GNU General Public License
16 # You should have received a copy of the GNU General Public License
17 # along with this program; if not, write to the Free Software
17 # along with this program; if not, write to the Free Software
18 # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
18 # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
19
19
20 module QueriesHelper
20 module QueriesHelper
21 include ApplicationHelper
21 include ApplicationHelper
22
22
23 def filters_options_for_select(query)
23 def filters_options_for_select(query)
24 ungrouped = []
24 ungrouped = []
25 grouped = {}
25 grouped = {}
26 query.available_filters.map do |field, field_options|
26 query.available_filters.map do |field, field_options|
27 if field_options[:type] == :relation
27 if field_options[:type] == :relation
28 group = :label_relations
28 group = :label_relations
29 elsif field_options[:type] == :tree
29 elsif field_options[:type] == :tree
30 group = query.is_a?(IssueQuery) ? :label_relations : nil
30 group = query.is_a?(IssueQuery) ? :label_relations : nil
31 elsif field =~ /^cf_\d+\./
32 group = (field_options[:through] || field_options[:field]).try(:name)
31 elsif field =~ /^(.+)\./
33 elsif field =~ /^(.+)\./
32 # association filters
34 # association filters
33 group = "field_#{$1}".to_sym
35 group = "field_#{$1}".to_sym
34 elsif %w(member_of_group assigned_to_role).include?(field)
36 elsif %w(member_of_group assigned_to_role).include?(field)
35 group = :field_assigned_to
37 group = :field_assigned_to
36 elsif field_options[:type] == :date_past || field_options[:type] == :date
38 elsif field_options[:type] == :date_past || field_options[:type] == :date
37 group = :label_date
39 group = :label_date
38 end
40 end
39 if group
41 if group
40 (grouped[group] ||= []) << [field_options[:name], field]
42 (grouped[group] ||= []) << [field_options[:name], field]
41 else
43 else
42 ungrouped << [field_options[:name], field]
44 ungrouped << [field_options[:name], field]
43 end
45 end
44 end
46 end
45 # Don't group dates if there's only one (eg. time entries filters)
47 # Don't group dates if there's only one (eg. time entries filters)
46 if grouped[:label_date].try(:size) == 1
48 if grouped[:label_date].try(:size) == 1
47 ungrouped << grouped.delete(:label_date).first
49 ungrouped << grouped.delete(:label_date).first
48 end
50 end
49 s = options_for_select([[]] + ungrouped)
51 s = options_for_select([[]] + ungrouped)
50 if grouped.present?
52 if grouped.present?
51 localized_grouped = grouped.map {|k,v| [l(k), v]}
53 localized_grouped = grouped.map {|k,v| [k.is_a?(Symbol) ? l(k) : k.to_s, v]}
52 s << grouped_options_for_select(localized_grouped)
54 s << grouped_options_for_select(localized_grouped)
53 end
55 end
54 s
56 s
55 end
57 end
56
58
57 def query_filters_hidden_tags(query)
59 def query_filters_hidden_tags(query)
58 tags = ''.html_safe
60 tags = ''.html_safe
59 query.filters.each do |field, options|
61 query.filters.each do |field, options|
60 tags << hidden_field_tag("f[]", field, :id => nil)
62 tags << hidden_field_tag("f[]", field, :id => nil)
61 tags << hidden_field_tag("op[#{field}]", options[:operator], :id => nil)
63 tags << hidden_field_tag("op[#{field}]", options[:operator], :id => nil)
62 options[:values].each do |value|
64 options[:values].each do |value|
63 tags << hidden_field_tag("v[#{field}][]", value, :id => nil)
65 tags << hidden_field_tag("v[#{field}][]", value, :id => nil)
64 end
66 end
65 end
67 end
66 tags
68 tags
67 end
69 end
68
70
69 def query_columns_hidden_tags(query)
71 def query_columns_hidden_tags(query)
70 tags = ''.html_safe
72 tags = ''.html_safe
71 query.columns.each do |column|
73 query.columns.each do |column|
72 tags << hidden_field_tag("c[]", column.name, :id => nil)
74 tags << hidden_field_tag("c[]", column.name, :id => nil)
73 end
75 end
74 tags
76 tags
75 end
77 end
76
78
77 def query_hidden_tags(query)
79 def query_hidden_tags(query)
78 query_filters_hidden_tags(query) + query_columns_hidden_tags(query)
80 query_filters_hidden_tags(query) + query_columns_hidden_tags(query)
79 end
81 end
80
82
81 def group_by_column_select_tag(query)
83 def group_by_column_select_tag(query)
82 options = [[]] + query.groupable_columns.collect {|c| [c.caption, c.name.to_s]}
84 options = [[]] + query.groupable_columns.collect {|c| [c.caption, c.name.to_s]}
83 select_tag('group_by', options_for_select(options, @query.group_by))
85 select_tag('group_by', options_for_select(options, @query.group_by))
84 end
86 end
85
87
86 def available_block_columns_tags(query)
88 def available_block_columns_tags(query)
87 tags = ''.html_safe
89 tags = ''.html_safe
88 query.available_block_columns.each do |column|
90 query.available_block_columns.each do |column|
89 tags << content_tag('label', check_box_tag('c[]', column.name.to_s, query.has_column?(column), :id => nil) + " #{column.caption}", :class => 'inline')
91 tags << content_tag('label', check_box_tag('c[]', column.name.to_s, query.has_column?(column), :id => nil) + " #{column.caption}", :class => 'inline')
90 end
92 end
91 tags
93 tags
92 end
94 end
93
95
94 def available_totalable_columns_tags(query)
96 def available_totalable_columns_tags(query)
95 tags = ''.html_safe
97 tags = ''.html_safe
96 query.available_totalable_columns.each do |column|
98 query.available_totalable_columns.each do |column|
97 tags << content_tag('label', check_box_tag('t[]', column.name.to_s, query.totalable_columns.include?(column), :id => nil) + " #{column.caption}", :class => 'inline')
99 tags << content_tag('label', check_box_tag('t[]', column.name.to_s, query.totalable_columns.include?(column), :id => nil) + " #{column.caption}", :class => 'inline')
98 end
100 end
99 tags << hidden_field_tag('t[]', '')
101 tags << hidden_field_tag('t[]', '')
100 tags
102 tags
101 end
103 end
102
104
103 def query_available_inline_columns_options(query)
105 def query_available_inline_columns_options(query)
104 (query.available_inline_columns - query.columns).reject(&:frozen?).collect {|column| [column.caption, column.name]}
106 (query.available_inline_columns - query.columns).reject(&:frozen?).collect {|column| [column.caption, column.name]}
105 end
107 end
106
108
107 def query_selected_inline_columns_options(query)
109 def query_selected_inline_columns_options(query)
108 (query.inline_columns & query.available_inline_columns).reject(&:frozen?).collect {|column| [column.caption, column.name]}
110 (query.inline_columns & query.available_inline_columns).reject(&:frozen?).collect {|column| [column.caption, column.name]}
109 end
111 end
110
112
111 def render_query_columns_selection(query, options={})
113 def render_query_columns_selection(query, options={})
112 tag_name = (options[:name] || 'c') + '[]'
114 tag_name = (options[:name] || 'c') + '[]'
113 render :partial => 'queries/columns', :locals => {:query => query, :tag_name => tag_name}
115 render :partial => 'queries/columns', :locals => {:query => query, :tag_name => tag_name}
114 end
116 end
115
117
116 def grouped_query_results(items, query, item_count_by_group, &block)
118 def grouped_query_results(items, query, item_count_by_group, &block)
117 previous_group, first = false, true
119 previous_group, first = false, true
118 totals_by_group = query.totalable_columns.inject({}) do |h, column|
120 totals_by_group = query.totalable_columns.inject({}) do |h, column|
119 h[column] = query.total_by_group_for(column)
121 h[column] = query.total_by_group_for(column)
120 h
122 h
121 end
123 end
122 items.each do |item|
124 items.each do |item|
123 group_name = group_count = nil
125 group_name = group_count = nil
124 if query.grouped?
126 if query.grouped?
125 group = query.group_by_column.value(item)
127 group = query.group_by_column.value(item)
126 if first || group != previous_group
128 if first || group != previous_group
127 if group.blank? && group != false
129 if group.blank? && group != false
128 group_name = "(#{l(:label_blank_value)})"
130 group_name = "(#{l(:label_blank_value)})"
129 else
131 else
130 group_name = format_object(group)
132 group_name = format_object(group)
131 end
133 end
132 group_name ||= ""
134 group_name ||= ""
133 group_count = item_count_by_group ? item_count_by_group[group] : nil
135 group_count = item_count_by_group ? item_count_by_group[group] : nil
134 group_totals = totals_by_group.map {|column, t| total_tag(column, t[group] || 0)}.join(" ").html_safe
136 group_totals = totals_by_group.map {|column, t| total_tag(column, t[group] || 0)}.join(" ").html_safe
135 end
137 end
136 end
138 end
137 yield item, group_name, group_count, group_totals
139 yield item, group_name, group_count, group_totals
138 previous_group, first = group, false
140 previous_group, first = group, false
139 end
141 end
140 end
142 end
141
143
142 def render_query_totals(query)
144 def render_query_totals(query)
143 return unless query.totalable_columns.present?
145 return unless query.totalable_columns.present?
144 totals = query.totalable_columns.map do |column|
146 totals = query.totalable_columns.map do |column|
145 total_tag(column, query.total_for(column))
147 total_tag(column, query.total_for(column))
146 end
148 end
147 content_tag('p', totals.join(" ").html_safe, :class => "query-totals")
149 content_tag('p', totals.join(" ").html_safe, :class => "query-totals")
148 end
150 end
149
151
150 def total_tag(column, value)
152 def total_tag(column, value)
151 label = content_tag('span', "#{column.caption}:")
153 label = content_tag('span', "#{column.caption}:")
152 value = if [:hours, :spent_hours, :total_spent_hours, :estimated_hours].include? column.name
154 value = if [:hours, :spent_hours, :total_spent_hours, :estimated_hours].include? column.name
153 format_hours(value)
155 format_hours(value)
154 else
156 else
155 format_object(value)
157 format_object(value)
156 end
158 end
157 value = content_tag('span', value, :class => 'value')
159 value = content_tag('span', value, :class => 'value')
158 content_tag('span', label + " " + value, :class => "total-for-#{column.name.to_s.dasherize}")
160 content_tag('span', label + " " + value, :class => "total-for-#{column.name.to_s.dasherize}")
159 end
161 end
160
162
161 def column_header(column)
163 def column_header(column)
162 column.sortable ? sort_header_tag(column.name.to_s, :caption => column.caption,
164 column.sortable ? sort_header_tag(column.name.to_s, :caption => column.caption,
163 :default_order => column.default_order) :
165 :default_order => column.default_order) :
164 content_tag('th', h(column.caption))
166 content_tag('th', h(column.caption))
165 end
167 end
166
168
167 def column_content(column, item)
169 def column_content(column, item)
168 value = column.value_object(item)
170 value = column.value_object(item)
169 if value.is_a?(Array)
171 if value.is_a?(Array)
170 value.collect {|v| column_value(column, item, v)}.compact.join(', ').html_safe
172 value.collect {|v| column_value(column, item, v)}.compact.join(', ').html_safe
171 else
173 else
172 column_value(column, item, value)
174 column_value(column, item, value)
173 end
175 end
174 end
176 end
175
177
176 def column_value(column, item, value)
178 def column_value(column, item, value)
177 case column.name
179 case column.name
178 when :id
180 when :id
179 link_to value, issue_path(item)
181 link_to value, issue_path(item)
180 when :subject
182 when :subject
181 link_to value, issue_path(item)
183 link_to value, issue_path(item)
182 when :parent
184 when :parent
183 value ? (value.visible? ? link_to_issue(value, :subject => false) : "##{value.id}") : ''
185 value ? (value.visible? ? link_to_issue(value, :subject => false) : "##{value.id}") : ''
184 when :description
186 when :description
185 item.description? ? content_tag('div', textilizable(item, :description), :class => "wiki") : ''
187 item.description? ? content_tag('div', textilizable(item, :description), :class => "wiki") : ''
186 when :done_ratio
188 when :done_ratio
187 progress_bar(value)
189 progress_bar(value)
188 when :relations
190 when :relations
189 content_tag('span',
191 content_tag('span',
190 value.to_s(item) {|other| link_to_issue(other, :subject => false, :tracker => false)}.html_safe,
192 value.to_s(item) {|other| link_to_issue(other, :subject => false, :tracker => false)}.html_safe,
191 :class => value.css_classes_for(item))
193 :class => value.css_classes_for(item))
192 when :hours, :spent_hours, :total_spent_hours, :estimated_hours
194 when :hours, :spent_hours, :total_spent_hours, :estimated_hours
193 format_hours(value)
195 format_hours(value)
194 else
196 else
195 format_object(value)
197 format_object(value)
196 end
198 end
197 end
199 end
198
200
199 def csv_content(column, item)
201 def csv_content(column, item)
200 value = column.value_object(item)
202 value = column.value_object(item)
201 if value.is_a?(Array)
203 if value.is_a?(Array)
202 value.collect {|v| csv_value(column, item, v)}.compact.join(', ')
204 value.collect {|v| csv_value(column, item, v)}.compact.join(', ')
203 else
205 else
204 csv_value(column, item, value)
206 csv_value(column, item, value)
205 end
207 end
206 end
208 end
207
209
208 def csv_value(column, object, value)
210 def csv_value(column, object, value)
209 format_object(value, false) do |value|
211 format_object(value, false) do |value|
210 case value.class.name
212 case value.class.name
211 when 'Float'
213 when 'Float'
212 sprintf("%.2f", value).gsub('.', l(:general_csv_decimal_separator))
214 sprintf("%.2f", value).gsub('.', l(:general_csv_decimal_separator))
213 when 'IssueRelation'
215 when 'IssueRelation'
214 value.to_s(object)
216 value.to_s(object)
215 when 'Issue'
217 when 'Issue'
216 if object.is_a?(TimeEntry)
218 if object.is_a?(TimeEntry)
217 "#{value.tracker} ##{value.id}: #{value.subject}"
219 "#{value.tracker} ##{value.id}: #{value.subject}"
218 else
220 else
219 value.id
221 value.id
220 end
222 end
221 else
223 else
222 value
224 value
223 end
225 end
224 end
226 end
225 end
227 end
226
228
227 def query_to_csv(items, query, options={})
229 def query_to_csv(items, query, options={})
228 options ||= {}
230 options ||= {}
229 columns = (options[:columns] == 'all' ? query.available_inline_columns : query.inline_columns)
231 columns = (options[:columns] == 'all' ? query.available_inline_columns : query.inline_columns)
230 query.available_block_columns.each do |column|
232 query.available_block_columns.each do |column|
231 if options[column.name].present?
233 if options[column.name].present?
232 columns << column
234 columns << column
233 end
235 end
234 end
236 end
235
237
236 Redmine::Export::CSV.generate do |csv|
238 Redmine::Export::CSV.generate do |csv|
237 # csv header fields
239 # csv header fields
238 csv << columns.map {|c| c.caption.to_s}
240 csv << columns.map {|c| c.caption.to_s}
239 # csv lines
241 # csv lines
240 items.each do |item|
242 items.each do |item|
241 csv << columns.map {|c| csv_content(c, item)}
243 csv << columns.map {|c| csv_content(c, item)}
242 end
244 end
243 end
245 end
244 end
246 end
245
247
246 # Retrieve query from session or build a new query
248 # Retrieve query from session or build a new query
247 def retrieve_query(klass=IssueQuery, use_session=true)
249 def retrieve_query(klass=IssueQuery, use_session=true)
248 session_key = klass.name.underscore.to_sym
250 session_key = klass.name.underscore.to_sym
249
251
250 if params[:query_id].present?
252 if params[:query_id].present?
251 cond = "project_id IS NULL"
253 cond = "project_id IS NULL"
252 cond << " OR project_id = #{@project.id}" if @project
254 cond << " OR project_id = #{@project.id}" if @project
253 @query = klass.where(cond).find(params[:query_id])
255 @query = klass.where(cond).find(params[:query_id])
254 raise ::Unauthorized unless @query.visible?
256 raise ::Unauthorized unless @query.visible?
255 @query.project = @project
257 @query.project = @project
256 session[session_key] = {:id => @query.id, :project_id => @query.project_id} if use_session
258 session[session_key] = {:id => @query.id, :project_id => @query.project_id} if use_session
257 sort_clear
259 sort_clear
258 elsif api_request? || params[:set_filter] || !use_session || session[session_key].nil? || session[session_key][:project_id] != (@project ? @project.id : nil)
260 elsif api_request? || params[:set_filter] || !use_session || session[session_key].nil? || session[session_key][:project_id] != (@project ? @project.id : nil)
259 # Give it a name, required to be valid
261 # Give it a name, required to be valid
260 @query = klass.new(:name => "_", :project => @project)
262 @query = klass.new(:name => "_", :project => @project)
261 @query.build_from_params(params)
263 @query.build_from_params(params)
262 session[session_key] = {:project_id => @query.project_id, :filters => @query.filters, :group_by => @query.group_by, :column_names => @query.column_names, :totalable_names => @query.totalable_names} if use_session
264 session[session_key] = {:project_id => @query.project_id, :filters => @query.filters, :group_by => @query.group_by, :column_names => @query.column_names, :totalable_names => @query.totalable_names} if use_session
263 else
265 else
264 # retrieve from session
266 # retrieve from session
265 @query = nil
267 @query = nil
266 @query = klass.find_by_id(session[session_key][:id]) if session[session_key][:id]
268 @query = klass.find_by_id(session[session_key][:id]) if session[session_key][:id]
267 @query ||= klass.new(:name => "_", :filters => session[session_key][:filters], :group_by => session[session_key][:group_by], :column_names => session[session_key][:column_names], :totalable_names => session[session_key][:totalable_names])
269 @query ||= klass.new(:name => "_", :filters => session[session_key][:filters], :group_by => session[session_key][:group_by], :column_names => session[session_key][:column_names], :totalable_names => session[session_key][:totalable_names])
268 @query.project = @project
270 @query.project = @project
269 end
271 end
270 end
272 end
271
273
272 def retrieve_query_from_session(klass=IssueQuery)
274 def retrieve_query_from_session(klass=IssueQuery)
273 session_key = klass.name.underscore.to_sym
275 session_key = klass.name.underscore.to_sym
274 session_data = session[session_key]
276 session_data = session[session_key]
275
277
276 if session_data
278 if session_data
277 if session_data[:id]
279 if session_data[:id]
278 @query = IssueQuery.find_by_id(session_data[:id])
280 @query = IssueQuery.find_by_id(session_data[:id])
279 return unless @query
281 return unless @query
280 else
282 else
281 @query = IssueQuery.new(:name => "_", :filters => session_data[:filters], :group_by => session_data[:group_by], :column_names => session_data[:column_names], :totalable_names => session_data[:totalable_names])
283 @query = IssueQuery.new(:name => "_", :filters => session_data[:filters], :group_by => session_data[:group_by], :column_names => session_data[:column_names], :totalable_names => session_data[:totalable_names])
282 end
284 end
283 if session_data.has_key?(:project_id)
285 if session_data.has_key?(:project_id)
284 @query.project_id = session_data[:project_id]
286 @query.project_id = session_data[:project_id]
285 else
287 else
286 @query.project = @project
288 @query.project = @project
287 end
289 end
288 @query
290 @query
289 end
291 end
290 end
292 end
291
293
292 # Returns the query definition as hidden field tags
294 # Returns the query definition as hidden field tags
293 def query_as_hidden_field_tags(query)
295 def query_as_hidden_field_tags(query)
294 tags = hidden_field_tag("set_filter", "1", :id => nil)
296 tags = hidden_field_tag("set_filter", "1", :id => nil)
295
297
296 if query.filters.present?
298 if query.filters.present?
297 query.filters.each do |field, filter|
299 query.filters.each do |field, filter|
298 tags << hidden_field_tag("f[]", field, :id => nil)
300 tags << hidden_field_tag("f[]", field, :id => nil)
299 tags << hidden_field_tag("op[#{field}]", filter[:operator], :id => nil)
301 tags << hidden_field_tag("op[#{field}]", filter[:operator], :id => nil)
300 filter[:values].each do |value|
302 filter[:values].each do |value|
301 tags << hidden_field_tag("v[#{field}][]", value, :id => nil)
303 tags << hidden_field_tag("v[#{field}][]", value, :id => nil)
302 end
304 end
303 end
305 end
304 else
306 else
305 tags << hidden_field_tag("f[]", "", :id => nil)
307 tags << hidden_field_tag("f[]", "", :id => nil)
306 end
308 end
307 if query.column_names.present?
309 if query.column_names.present?
308 query.column_names.each do |name|
310 query.column_names.each do |name|
309 tags << hidden_field_tag("c[]", name, :id => nil)
311 tags << hidden_field_tag("c[]", name, :id => nil)
310 end
312 end
311 end
313 end
312 if query.totalable_names.present?
314 if query.totalable_names.present?
313 query.totalable_names.each do |name|
315 query.totalable_names.each do |name|
314 tags << hidden_field_tag("t[]", name, :id => nil)
316 tags << hidden_field_tag("t[]", name, :id => nil)
315 end
317 end
316 end
318 end
317 if query.group_by.present?
319 if query.group_by.present?
318 tags << hidden_field_tag("group_by", query.group_by, :id => nil)
320 tags << hidden_field_tag("group_by", query.group_by, :id => nil)
319 end
321 end
320
322
321 tags
323 tags
322 end
324 end
323
325
324 # Returns the queries that are rendered in the sidebar
326 # Returns the queries that are rendered in the sidebar
325 def sidebar_queries(klass, project)
327 def sidebar_queries(klass, project)
326 klass.visible.global_or_on_project(@project).sorted.to_a
328 klass.visible.global_or_on_project(@project).sorted.to_a
327 end
329 end
328
330
329 # Renders a group of queries
331 # Renders a group of queries
330 def query_links(title, queries)
332 def query_links(title, queries)
331 return '' if queries.empty?
333 return '' if queries.empty?
332 # links to #index on issues/show
334 # links to #index on issues/show
333 url_params = controller_name == 'issues' ? {:controller => 'issues', :action => 'index', :project_id => @project} : {}
335 url_params = controller_name == 'issues' ? {:controller => 'issues', :action => 'index', :project_id => @project} : {}
334
336
335 content_tag('h3', title) + "\n" +
337 content_tag('h3', title) + "\n" +
336 content_tag('ul',
338 content_tag('ul',
337 queries.collect {|query|
339 queries.collect {|query|
338 css = 'query'
340 css = 'query'
339 css << ' selected' if query == @query
341 css << ' selected' if query == @query
340 content_tag('li', link_to(query.name, url_params.merge(:query_id => query), :class => css))
342 content_tag('li', link_to(query.name, url_params.merge(:query_id => query), :class => css))
341 }.join("\n").html_safe,
343 }.join("\n").html_safe,
342 :class => 'queries'
344 :class => 'queries'
343 ) + "\n"
345 ) + "\n"
344 end
346 end
345
347
346 # Renders the list of queries for the sidebar
348 # Renders the list of queries for the sidebar
347 def render_sidebar_queries(klass, project)
349 def render_sidebar_queries(klass, project)
348 queries = sidebar_queries(klass, project)
350 queries = sidebar_queries(klass, project)
349
351
350 out = ''.html_safe
352 out = ''.html_safe
351 out << query_links(l(:label_my_queries), queries.select(&:is_private?))
353 out << query_links(l(:label_my_queries), queries.select(&:is_private?))
352 out << query_links(l(:label_query_plural), queries.reject(&:is_private?))
354 out << query_links(l(:label_query_plural), queries.reject(&:is_private?))
353 out
355 out
354 end
356 end
355 end
357 end
@@ -1,1226 +1,1307
1 # Redmine - project management software
1 # Redmine - project management software
2 # Copyright (C) 2006-2016 Jean-Philippe Lang
2 # Copyright (C) 2006-2016 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, :totalable, :default_order
19 attr_accessor :name, :sortable, :groupable, :totalable, :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.totalable = options[:totalable] || false
29 self.totalable = options[:totalable] || false
30 self.default_order = options[:default_order]
30 self.default_order = options[:default_order]
31 @inline = options.key?(:inline) ? options[:inline] : true
31 @inline = options.key?(:inline) ? options[:inline] : true
32 @caption_key = options[:caption] || "field_#{name}".to_sym
32 @caption_key = options[:caption] || "field_#{name}".to_sym
33 @frozen = options[:frozen]
33 @frozen = options[:frozen]
34 end
34 end
35
35
36 def caption
36 def caption
37 case @caption_key
37 case @caption_key
38 when Symbol
38 when Symbol
39 l(@caption_key)
39 l(@caption_key)
40 when Proc
40 when Proc
41 @caption_key.call
41 @caption_key.call
42 else
42 else
43 @caption_key
43 @caption_key
44 end
44 end
45 end
45 end
46
46
47 # Returns true if the column is sortable, otherwise false
47 # Returns true if the column is sortable, otherwise false
48 def sortable?
48 def sortable?
49 !@sortable.nil?
49 !@sortable.nil?
50 end
50 end
51
51
52 def sortable
52 def sortable
53 @sortable.is_a?(Proc) ? @sortable.call : @sortable
53 @sortable.is_a?(Proc) ? @sortable.call : @sortable
54 end
54 end
55
55
56 def inline?
56 def inline?
57 @inline
57 @inline
58 end
58 end
59
59
60 def frozen?
60 def frozen?
61 @frozen
61 @frozen
62 end
62 end
63
63
64 def value(object)
64 def value(object)
65 object.send name
65 object.send name
66 end
66 end
67
67
68 def value_object(object)
68 def value_object(object)
69 object.send name
69 object.send name
70 end
70 end
71
71
72 def css_classes
72 def css_classes
73 name
73 name
74 end
74 end
75 end
75 end
76
76
77 class QueryAssociationColumn < QueryColumn
77 class QueryAssociationColumn < QueryColumn
78
78
79 def initialize(association, attribute, options={})
79 def initialize(association, attribute, options={})
80 @association = association
80 @association = association
81 @attribute = attribute
81 @attribute = attribute
82 name_with_assoc = "#{association}.#{attribute}".to_sym
82 name_with_assoc = "#{association}.#{attribute}".to_sym
83 super(name_with_assoc, options)
83 super(name_with_assoc, options)
84 end
84 end
85
85
86 def value_object(object)
86 def value_object(object)
87 if assoc = object.send(@association)
87 if assoc = object.send(@association)
88 assoc.send @attribute
88 assoc.send @attribute
89 end
89 end
90 end
90 end
91
91
92 def css_classes
92 def css_classes
93 @css_classes ||= "#{@association}-#{@attribute}"
93 @css_classes ||= "#{@association}-#{@attribute}"
94 end
94 end
95 end
95 end
96
96
97 class QueryCustomFieldColumn < QueryColumn
97 class QueryCustomFieldColumn < QueryColumn
98
98
99 def initialize(custom_field, options={})
99 def initialize(custom_field, options={})
100 self.name = "cf_#{custom_field.id}".to_sym
100 self.name = "cf_#{custom_field.id}".to_sym
101 self.sortable = custom_field.order_statement || false
101 self.sortable = custom_field.order_statement || false
102 self.groupable = custom_field.group_statement || false
102 self.groupable = custom_field.group_statement || false
103 self.totalable = options.key?(:totalable) ? !!options[:totalable] : custom_field.totalable?
103 self.totalable = options.key?(:totalable) ? !!options[:totalable] : custom_field.totalable?
104 @inline = true
104 @inline = true
105 @cf = custom_field
105 @cf = custom_field
106 end
106 end
107
107
108 def caption
108 def caption
109 @cf.name
109 @cf.name
110 end
110 end
111
111
112 def custom_field
112 def custom_field
113 @cf
113 @cf
114 end
114 end
115
115
116 def value_object(object)
116 def value_object(object)
117 if custom_field.visible_by?(object.project, User.current)
117 if custom_field.visible_by?(object.project, User.current)
118 cv = object.custom_values.select {|v| v.custom_field_id == @cf.id}
118 cv = object.custom_values.select {|v| v.custom_field_id == @cf.id}
119 cv.size > 1 ? cv.sort {|a,b| a.value.to_s <=> b.value.to_s} : cv.first
119 cv.size > 1 ? cv.sort {|a,b| a.value.to_s <=> b.value.to_s} : cv.first
120 else
120 else
121 nil
121 nil
122 end
122 end
123 end
123 end
124
124
125 def value(object)
125 def value(object)
126 raw = value_object(object)
126 raw = value_object(object)
127 if raw.is_a?(Array)
127 if raw.is_a?(Array)
128 raw.map {|r| @cf.cast_value(r.value)}
128 raw.map {|r| @cf.cast_value(r.value)}
129 elsif raw
129 elsif raw
130 @cf.cast_value(raw.value)
130 @cf.cast_value(raw.value)
131 else
131 else
132 nil
132 nil
133 end
133 end
134 end
134 end
135
135
136 def css_classes
136 def css_classes
137 @css_classes ||= "#{name} #{@cf.field_format}"
137 @css_classes ||= "#{name} #{@cf.field_format}"
138 end
138 end
139 end
139 end
140
140
141 class QueryAssociationCustomFieldColumn < QueryCustomFieldColumn
141 class QueryAssociationCustomFieldColumn < QueryCustomFieldColumn
142
142
143 def initialize(association, custom_field, options={})
143 def initialize(association, custom_field, options={})
144 super(custom_field, options)
144 super(custom_field, options)
145 self.name = "#{association}.cf_#{custom_field.id}".to_sym
145 self.name = "#{association}.cf_#{custom_field.id}".to_sym
146 # TODO: support sorting/grouping by association custom field
146 # TODO: support sorting/grouping by association custom field
147 self.sortable = false
147 self.sortable = false
148 self.groupable = false
148 self.groupable = false
149 @association = association
149 @association = association
150 end
150 end
151
151
152 def value_object(object)
152 def value_object(object)
153 if assoc = object.send(@association)
153 if assoc = object.send(@association)
154 super(assoc)
154 super(assoc)
155 end
155 end
156 end
156 end
157
157
158 def css_classes
158 def css_classes
159 @css_classes ||= "#{@association}_cf_#{@cf.id} #{@cf.field_format}"
159 @css_classes ||= "#{@association}_cf_#{@cf.id} #{@cf.field_format}"
160 end
160 end
161 end
161 end
162
162
163 class QueryFilter
163 class QueryFilter
164 include Redmine::I18n
164 include Redmine::I18n
165
165
166 def initialize(field, options)
166 def initialize(field, options)
167 @field = field.to_s
167 @field = field.to_s
168 @options = options
168 @options = options
169 @options[:name] ||= l(options[:label] || "field_#{field}".gsub(/_id$/, ''))
169 @options[:name] ||= l(options[:label] || "field_#{field}".gsub(/_id$/, ''))
170 # Consider filters with a Proc for values as remote by default
170 # Consider filters with a Proc for values as remote by default
171 @remote = options.key?(:remote) ? options[:remote] : options[:values].is_a?(Proc)
171 @remote = options.key?(:remote) ? options[:remote] : options[:values].is_a?(Proc)
172 end
172 end
173
173
174 def [](arg)
174 def [](arg)
175 if arg == :values
175 if arg == :values
176 values
176 values
177 else
177 else
178 @options[arg]
178 @options[arg]
179 end
179 end
180 end
180 end
181
181
182 def values
182 def values
183 @values ||= begin
183 @values ||= begin
184 values = @options[:values]
184 values = @options[:values]
185 if values.is_a?(Proc)
185 if values.is_a?(Proc)
186 values = values.call
186 values = values.call
187 end
187 end
188 values
188 values
189 end
189 end
190 end
190 end
191
191
192 def remote
192 def remote
193 @remote
193 @remote
194 end
194 end
195 end
195 end
196
196
197 class Query < ActiveRecord::Base
197 class Query < ActiveRecord::Base
198 class StatementInvalid < ::ActiveRecord::StatementInvalid
198 class StatementInvalid < ::ActiveRecord::StatementInvalid
199 end
199 end
200
200
201 include Redmine::SubclassFactory
201 include Redmine::SubclassFactory
202
202
203 VISIBILITY_PRIVATE = 0
203 VISIBILITY_PRIVATE = 0
204 VISIBILITY_ROLES = 1
204 VISIBILITY_ROLES = 1
205 VISIBILITY_PUBLIC = 2
205 VISIBILITY_PUBLIC = 2
206
206
207 belongs_to :project
207 belongs_to :project
208 belongs_to :user
208 belongs_to :user
209 has_and_belongs_to_many :roles, :join_table => "#{table_name_prefix}queries_roles#{table_name_suffix}", :foreign_key => "query_id"
209 has_and_belongs_to_many :roles, :join_table => "#{table_name_prefix}queries_roles#{table_name_suffix}", :foreign_key => "query_id"
210 serialize :filters
210 serialize :filters
211 serialize :column_names
211 serialize :column_names
212 serialize :sort_criteria, Array
212 serialize :sort_criteria, Array
213 serialize :options, Hash
213 serialize :options, Hash
214
214
215 attr_protected :project_id, :user_id
215 attr_protected :project_id, :user_id
216
216
217 validates_presence_of :name
217 validates_presence_of :name
218 validates_length_of :name, :maximum => 255
218 validates_length_of :name, :maximum => 255
219 validates :visibility, :inclusion => { :in => [VISIBILITY_PUBLIC, VISIBILITY_ROLES, VISIBILITY_PRIVATE] }
219 validates :visibility, :inclusion => { :in => [VISIBILITY_PUBLIC, VISIBILITY_ROLES, VISIBILITY_PRIVATE] }
220 validate :validate_query_filters
220 validate :validate_query_filters
221 validate do |query|
221 validate do |query|
222 errors.add(:base, l(:label_role_plural) + ' ' + l('activerecord.errors.messages.blank')) if query.visibility == VISIBILITY_ROLES && roles.blank?
222 errors.add(:base, l(:label_role_plural) + ' ' + l('activerecord.errors.messages.blank')) if query.visibility == VISIBILITY_ROLES && roles.blank?
223 end
223 end
224
224
225 after_save do |query|
225 after_save do |query|
226 if query.visibility_changed? && query.visibility != VISIBILITY_ROLES
226 if query.visibility_changed? && query.visibility != VISIBILITY_ROLES
227 query.roles.clear
227 query.roles.clear
228 end
228 end
229 end
229 end
230
230
231 class_attribute :operators
231 class_attribute :operators
232 self.operators = {
232 self.operators = {
233 "=" => :label_equals,
233 "=" => :label_equals,
234 "!" => :label_not_equals,
234 "!" => :label_not_equals,
235 "o" => :label_open_issues,
235 "o" => :label_open_issues,
236 "c" => :label_closed_issues,
236 "c" => :label_closed_issues,
237 "!*" => :label_none,
237 "!*" => :label_none,
238 "*" => :label_any,
238 "*" => :label_any,
239 ">=" => :label_greater_or_equal,
239 ">=" => :label_greater_or_equal,
240 "<=" => :label_less_or_equal,
240 "<=" => :label_less_or_equal,
241 "><" => :label_between,
241 "><" => :label_between,
242 "<t+" => :label_in_less_than,
242 "<t+" => :label_in_less_than,
243 ">t+" => :label_in_more_than,
243 ">t+" => :label_in_more_than,
244 "><t+"=> :label_in_the_next_days,
244 "><t+"=> :label_in_the_next_days,
245 "t+" => :label_in,
245 "t+" => :label_in,
246 "t" => :label_today,
246 "t" => :label_today,
247 "ld" => :label_yesterday,
247 "ld" => :label_yesterday,
248 "w" => :label_this_week,
248 "w" => :label_this_week,
249 "lw" => :label_last_week,
249 "lw" => :label_last_week,
250 "l2w" => [:label_last_n_weeks, {:count => 2}],
250 "l2w" => [:label_last_n_weeks, {:count => 2}],
251 "m" => :label_this_month,
251 "m" => :label_this_month,
252 "lm" => :label_last_month,
252 "lm" => :label_last_month,
253 "y" => :label_this_year,
253 "y" => :label_this_year,
254 ">t-" => :label_less_than_ago,
254 ">t-" => :label_less_than_ago,
255 "<t-" => :label_more_than_ago,
255 "<t-" => :label_more_than_ago,
256 "><t-"=> :label_in_the_past_days,
256 "><t-"=> :label_in_the_past_days,
257 "t-" => :label_ago,
257 "t-" => :label_ago,
258 "~" => :label_contains,
258 "~" => :label_contains,
259 "!~" => :label_not_contains,
259 "!~" => :label_not_contains,
260 "=p" => :label_any_issues_in_project,
260 "=p" => :label_any_issues_in_project,
261 "=!p" => :label_any_issues_not_in_project,
261 "=!p" => :label_any_issues_not_in_project,
262 "!p" => :label_no_issues_in_project,
262 "!p" => :label_no_issues_in_project,
263 "*o" => :label_any_open_issues,
263 "*o" => :label_any_open_issues,
264 "!o" => :label_no_open_issues
264 "!o" => :label_no_open_issues
265 }
265 }
266
266
267 class_attribute :operators_by_filter_type
267 class_attribute :operators_by_filter_type
268 self.operators_by_filter_type = {
268 self.operators_by_filter_type = {
269 :list => [ "=", "!" ],
269 :list => [ "=", "!" ],
270 :list_status => [ "o", "=", "!", "c", "*" ],
270 :list_status => [ "o", "=", "!", "c", "*" ],
271 :list_optional => [ "=", "!", "!*", "*" ],
271 :list_optional => [ "=", "!", "!*", "*" ],
272 :list_subprojects => [ "*", "!*", "=" ],
272 :list_subprojects => [ "*", "!*", "=" ],
273 :date => [ "=", ">=", "<=", "><", "<t+", ">t+", "><t+", "t+", "t", "ld", "w", "lw", "l2w", "m", "lm", "y", ">t-", "<t-", "><t-", "t-", "!*", "*" ],
273 :date => [ "=", ">=", "<=", "><", "<t+", ">t+", "><t+", "t+", "t", "ld", "w", "lw", "l2w", "m", "lm", "y", ">t-", "<t-", "><t-", "t-", "!*", "*" ],
274 :date_past => [ "=", ">=", "<=", "><", ">t-", "<t-", "><t-", "t-", "t", "ld", "w", "lw", "l2w", "m", "lm", "y", "!*", "*" ],
274 :date_past => [ "=", ">=", "<=", "><", ">t-", "<t-", "><t-", "t-", "t", "ld", "w", "lw", "l2w", "m", "lm", "y", "!*", "*" ],
275 :string => [ "=", "~", "!", "!~", "!*", "*" ],
275 :string => [ "=", "~", "!", "!~", "!*", "*" ],
276 :text => [ "~", "!~", "!*", "*" ],
276 :text => [ "~", "!~", "!*", "*" ],
277 :integer => [ "=", ">=", "<=", "><", "!*", "*" ],
277 :integer => [ "=", ">=", "<=", "><", "!*", "*" ],
278 :float => [ "=", ">=", "<=", "><", "!*", "*" ],
278 :float => [ "=", ">=", "<=", "><", "!*", "*" ],
279 :relation => ["=", "=p", "=!p", "!p", "*o", "!o", "!*", "*"],
279 :relation => ["=", "=p", "=!p", "!p", "*o", "!o", "!*", "*"],
280 :tree => ["=", "~", "!*", "*"]
280 :tree => ["=", "~", "!*", "*"]
281 }
281 }
282
282
283 class_attribute :available_columns
283 class_attribute :available_columns
284 self.available_columns = []
284 self.available_columns = []
285
285
286 class_attribute :queried_class
286 class_attribute :queried_class
287
287
288 # Permission required to view the queries, set on subclasses.
288 # Permission required to view the queries, set on subclasses.
289 class_attribute :view_permission
289 class_attribute :view_permission
290
290
291 # Scope of queries that are global or on the given project
291 # Scope of queries that are global or on the given project
292 scope :global_or_on_project, lambda {|project|
292 scope :global_or_on_project, lambda {|project|
293 where(:project_id => (project.nil? ? nil : [nil, project.id]))
293 where(:project_id => (project.nil? ? nil : [nil, project.id]))
294 }
294 }
295
295
296 scope :sorted, lambda {order(:name, :id)}
296 scope :sorted, lambda {order(:name, :id)}
297
297
298 # Scope of visible queries, can be used from subclasses only.
298 # Scope of visible queries, can be used from subclasses only.
299 # Unlike other visible scopes, a class methods is used as it
299 # Unlike other visible scopes, a class methods is used as it
300 # let handle inheritance more nicely than scope DSL.
300 # let handle inheritance more nicely than scope DSL.
301 def self.visible(*args)
301 def self.visible(*args)
302 if self == ::Query
302 if self == ::Query
303 # Visibility depends on permissions for each subclass,
303 # Visibility depends on permissions for each subclass,
304 # raise an error if the scope is called from Query (eg. Query.visible)
304 # raise an error if the scope is called from Query (eg. Query.visible)
305 raise Exception.new("Cannot call .visible scope from the base Query class, but from subclasses only.")
305 raise Exception.new("Cannot call .visible scope from the base Query class, but from subclasses only.")
306 end
306 end
307
307
308 user = args.shift || User.current
308 user = args.shift || User.current
309 base = Project.allowed_to_condition(user, view_permission, *args)
309 base = Project.allowed_to_condition(user, view_permission, *args)
310 scope = joins("LEFT OUTER JOIN #{Project.table_name} ON #{table_name}.project_id = #{Project.table_name}.id").
310 scope = joins("LEFT OUTER JOIN #{Project.table_name} ON #{table_name}.project_id = #{Project.table_name}.id").
311 where("#{table_name}.project_id IS NULL OR (#{base})")
311 where("#{table_name}.project_id IS NULL OR (#{base})")
312
312
313 if user.admin?
313 if user.admin?
314 scope.where("#{table_name}.visibility <> ? OR #{table_name}.user_id = ?", VISIBILITY_PRIVATE, user.id)
314 scope.where("#{table_name}.visibility <> ? OR #{table_name}.user_id = ?", VISIBILITY_PRIVATE, user.id)
315 elsif user.memberships.any?
315 elsif user.memberships.any?
316 scope.where("#{table_name}.visibility = ?" +
316 scope.where("#{table_name}.visibility = ?" +
317 " OR (#{table_name}.visibility = ? AND #{table_name}.id IN (" +
317 " OR (#{table_name}.visibility = ? AND #{table_name}.id IN (" +
318 "SELECT DISTINCT q.id FROM #{table_name} q" +
318 "SELECT DISTINCT q.id FROM #{table_name} q" +
319 " INNER JOIN #{table_name_prefix}queries_roles#{table_name_suffix} qr on qr.query_id = q.id" +
319 " INNER JOIN #{table_name_prefix}queries_roles#{table_name_suffix} qr on qr.query_id = q.id" +
320 " INNER JOIN #{MemberRole.table_name} mr ON mr.role_id = qr.role_id" +
320 " INNER JOIN #{MemberRole.table_name} mr ON mr.role_id = qr.role_id" +
321 " INNER JOIN #{Member.table_name} m ON m.id = mr.member_id AND m.user_id = ?" +
321 " INNER JOIN #{Member.table_name} m ON m.id = mr.member_id AND m.user_id = ?" +
322 " WHERE q.project_id IS NULL OR q.project_id = m.project_id))" +
322 " WHERE q.project_id IS NULL OR q.project_id = m.project_id))" +
323 " OR #{table_name}.user_id = ?",
323 " OR #{table_name}.user_id = ?",
324 VISIBILITY_PUBLIC, VISIBILITY_ROLES, user.id, user.id)
324 VISIBILITY_PUBLIC, VISIBILITY_ROLES, user.id, user.id)
325 elsif user.logged?
325 elsif user.logged?
326 scope.where("#{table_name}.visibility = ? OR #{table_name}.user_id = ?", VISIBILITY_PUBLIC, user.id)
326 scope.where("#{table_name}.visibility = ? OR #{table_name}.user_id = ?", VISIBILITY_PUBLIC, user.id)
327 else
327 else
328 scope.where("#{table_name}.visibility = ?", VISIBILITY_PUBLIC)
328 scope.where("#{table_name}.visibility = ?", VISIBILITY_PUBLIC)
329 end
329 end
330 end
330 end
331
331
332 # Returns true if the query is visible to +user+ or the current user.
332 # Returns true if the query is visible to +user+ or the current user.
333 def visible?(user=User.current)
333 def visible?(user=User.current)
334 return true if user.admin?
334 return true if user.admin?
335 return false unless project.nil? || user.allowed_to?(self.class.view_permission, project)
335 return false unless project.nil? || user.allowed_to?(self.class.view_permission, project)
336 case visibility
336 case visibility
337 when VISIBILITY_PUBLIC
337 when VISIBILITY_PUBLIC
338 true
338 true
339 when VISIBILITY_ROLES
339 when VISIBILITY_ROLES
340 if project
340 if project
341 (user.roles_for_project(project) & roles).any?
341 (user.roles_for_project(project) & roles).any?
342 else
342 else
343 Member.where(:user_id => user.id).joins(:roles).where(:member_roles => {:role_id => roles.map(&:id)}).any?
343 Member.where(:user_id => user.id).joins(:roles).where(:member_roles => {:role_id => roles.map(&:id)}).any?
344 end
344 end
345 else
345 else
346 user == self.user
346 user == self.user
347 end
347 end
348 end
348 end
349
349
350 def is_private?
350 def is_private?
351 visibility == VISIBILITY_PRIVATE
351 visibility == VISIBILITY_PRIVATE
352 end
352 end
353
353
354 def is_public?
354 def is_public?
355 !is_private?
355 !is_private?
356 end
356 end
357
357
358 def queried_table_name
358 def queried_table_name
359 @queried_table_name ||= self.class.queried_class.table_name
359 @queried_table_name ||= self.class.queried_class.table_name
360 end
360 end
361
361
362 def initialize(attributes=nil, *args)
362 def initialize(attributes=nil, *args)
363 super attributes
363 super attributes
364 @is_for_all = project.nil?
364 @is_for_all = project.nil?
365 end
365 end
366
366
367 # Builds the query from the given params
367 # Builds the query from the given params
368 def build_from_params(params)
368 def build_from_params(params)
369 if params[:fields] || params[:f]
369 if params[:fields] || params[:f]
370 self.filters = {}
370 self.filters = {}
371 add_filters(params[:fields] || params[:f], params[:operators] || params[:op], params[:values] || params[:v])
371 add_filters(params[:fields] || params[:f], params[:operators] || params[:op], params[:values] || params[:v])
372 else
372 else
373 available_filters.keys.each do |field|
373 available_filters.keys.each do |field|
374 add_short_filter(field, params[field]) if params[field]
374 add_short_filter(field, params[field]) if params[field]
375 end
375 end
376 end
376 end
377 self.group_by = params[:group_by] || (params[:query] && params[:query][:group_by])
377 self.group_by = params[:group_by] || (params[:query] && params[:query][:group_by])
378 self.column_names = params[:c] || (params[:query] && params[:query][:column_names])
378 self.column_names = params[:c] || (params[:query] && params[:query][:column_names])
379 self.totalable_names = params[:t] || (params[:query] && params[:query][:totalable_names])
379 self.totalable_names = params[:t] || (params[:query] && params[:query][:totalable_names])
380 self
380 self
381 end
381 end
382
382
383 # Builds a new query from the given params and attributes
383 # Builds a new query from the given params and attributes
384 def self.build_from_params(params, attributes={})
384 def self.build_from_params(params, attributes={})
385 new(attributes).build_from_params(params)
385 new(attributes).build_from_params(params)
386 end
386 end
387
387
388 def validate_query_filters
388 def validate_query_filters
389 filters.each_key do |field|
389 filters.each_key do |field|
390 if values_for(field)
390 if values_for(field)
391 case type_for(field)
391 case type_for(field)
392 when :integer
392 when :integer
393 add_filter_error(field, :invalid) if values_for(field).detect {|v| v.present? && !v.match(/\A[+-]?\d+(,[+-]?\d+)*\z/) }
393 add_filter_error(field, :invalid) if values_for(field).detect {|v| v.present? && !v.match(/\A[+-]?\d+(,[+-]?\d+)*\z/) }
394 when :float
394 when :float
395 add_filter_error(field, :invalid) if values_for(field).detect {|v| v.present? && !v.match(/\A[+-]?\d+(\.\d*)?\z/) }
395 add_filter_error(field, :invalid) if values_for(field).detect {|v| v.present? && !v.match(/\A[+-]?\d+(\.\d*)?\z/) }
396 when :date, :date_past
396 when :date, :date_past
397 case operator_for(field)
397 case operator_for(field)
398 when "=", ">=", "<=", "><"
398 when "=", ">=", "<=", "><"
399 add_filter_error(field, :invalid) if values_for(field).detect {|v|
399 add_filter_error(field, :invalid) if values_for(field).detect {|v|
400 v.present? && (!v.match(/\A\d{4}-\d{2}-\d{2}(T\d{2}((:)?\d{2}){0,2}(Z|\d{2}:?\d{2})?)?\z/) || parse_date(v).nil?)
400 v.present? && (!v.match(/\A\d{4}-\d{2}-\d{2}(T\d{2}((:)?\d{2}){0,2}(Z|\d{2}:?\d{2})?)?\z/) || parse_date(v).nil?)
401 }
401 }
402 when ">t-", "<t-", "t-", ">t+", "<t+", "t+", "><t+", "><t-"
402 when ">t-", "<t-", "t-", ">t+", "<t+", "t+", "><t+", "><t-"
403 add_filter_error(field, :invalid) if values_for(field).detect {|v| v.present? && !v.match(/^\d+$/) }
403 add_filter_error(field, :invalid) if values_for(field).detect {|v| v.present? && !v.match(/^\d+$/) }
404 end
404 end
405 end
405 end
406 end
406 end
407
407
408 add_filter_error(field, :blank) unless
408 add_filter_error(field, :blank) unless
409 # filter requires one or more values
409 # filter requires one or more values
410 (values_for(field) and !values_for(field).first.blank?) or
410 (values_for(field) and !values_for(field).first.blank?) or
411 # filter doesn't require any value
411 # filter doesn't require any value
412 ["o", "c", "!*", "*", "t", "ld", "w", "lw", "l2w", "m", "lm", "y", "*o", "!o"].include? operator_for(field)
412 ["o", "c", "!*", "*", "t", "ld", "w", "lw", "l2w", "m", "lm", "y", "*o", "!o"].include? operator_for(field)
413 end if filters
413 end if filters
414 end
414 end
415
415
416 def add_filter_error(field, message)
416 def add_filter_error(field, message)
417 m = label_for(field) + " " + l(message, :scope => 'activerecord.errors.messages')
417 m = label_for(field) + " " + l(message, :scope => 'activerecord.errors.messages')
418 errors.add(:base, m)
418 errors.add(:base, m)
419 end
419 end
420
420
421 def editable_by?(user)
421 def editable_by?(user)
422 return false unless user
422 return false unless user
423 # Admin can edit them all and regular users can edit their private queries
423 # Admin can edit them all and regular users can edit their private queries
424 return true if user.admin? || (is_private? && self.user_id == user.id)
424 return true if user.admin? || (is_private? && self.user_id == user.id)
425 # Members can not edit public queries that are for all project (only admin is allowed to)
425 # Members can not edit public queries that are for all project (only admin is allowed to)
426 is_public? && !@is_for_all && user.allowed_to?(:manage_public_queries, project)
426 is_public? && !@is_for_all && user.allowed_to?(:manage_public_queries, project)
427 end
427 end
428
428
429 def trackers
429 def trackers
430 @trackers ||= (project.nil? ? Tracker.all : project.rolled_up_trackers).visible.sorted
430 @trackers ||= (project.nil? ? Tracker.all : project.rolled_up_trackers).visible.sorted
431 end
431 end
432
432
433 # Returns a hash of localized labels for all filter operators
433 # Returns a hash of localized labels for all filter operators
434 def self.operators_labels
434 def self.operators_labels
435 operators.inject({}) {|h, operator| h[operator.first] = l(*operator.last); h}
435 operators.inject({}) {|h, operator| h[operator.first] = l(*operator.last); h}
436 end
436 end
437
437
438 # Returns a representation of the available filters for JSON serialization
438 # Returns a representation of the available filters for JSON serialization
439 def available_filters_as_json
439 def available_filters_as_json
440 json = {}
440 json = {}
441 available_filters.each do |field, filter|
441 available_filters.each do |field, filter|
442 options = {:type => filter[:type], :name => filter[:name]}
442 options = {:type => filter[:type], :name => filter[:name]}
443 options[:remote] = true if filter.remote
443 options[:remote] = true if filter.remote
444
444
445 if has_filter?(field) || !filter.remote
445 if has_filter?(field) || !filter.remote
446 options[:values] = filter.values
446 options[:values] = filter.values
447 if options[:values] && values_for(field)
447 if options[:values] && values_for(field)
448 missing = Array(values_for(field)).select(&:present?) - options[:values].map(&:last)
448 missing = Array(values_for(field)).select(&:present?) - options[:values].map(&:last)
449 if missing.any? && respond_to?(method = "find_#{field}_filter_values")
449 if missing.any? && respond_to?(method = "find_#{field}_filter_values")
450 options[:values] += send(method, missing)
450 options[:values] += send(method, missing)
451 end
451 end
452 end
452 end
453 end
453 end
454 json[field] = options.stringify_keys
454 json[field] = options.stringify_keys
455 end
455 end
456 json
456 json
457 end
457 end
458
458
459 def all_projects
459 def all_projects
460 @all_projects ||= Project.visible.to_a
460 @all_projects ||= Project.visible.to_a
461 end
461 end
462
462
463 def all_projects_values
463 def all_projects_values
464 return @all_projects_values if @all_projects_values
464 return @all_projects_values if @all_projects_values
465
465
466 values = []
466 values = []
467 Project.project_tree(all_projects) do |p, level|
467 Project.project_tree(all_projects) do |p, level|
468 prefix = (level > 0 ? ('--' * level + ' ') : '')
468 prefix = (level > 0 ? ('--' * level + ' ') : '')
469 values << ["#{prefix}#{p.name}", p.id.to_s]
469 values << ["#{prefix}#{p.name}", p.id.to_s]
470 end
470 end
471 @all_projects_values = values
471 @all_projects_values = values
472 end
472 end
473
473
474 def project_values
474 def project_values
475 project_values = []
475 project_values = []
476 if User.current.logged? && User.current.memberships.any?
476 if User.current.logged? && User.current.memberships.any?
477 project_values << ["<< #{l(:label_my_projects).downcase} >>", "mine"]
477 project_values << ["<< #{l(:label_my_projects).downcase} >>", "mine"]
478 end
478 end
479 project_values += all_projects_values
479 project_values += all_projects_values
480 project_values
480 project_values
481 end
481 end
482
482
483 def subproject_values
483 def subproject_values
484 project.descendants.visible.collect{|s| [s.name, s.id.to_s] }
484 project.descendants.visible.collect{|s| [s.name, s.id.to_s] }
485 end
485 end
486
486
487 def principals
487 def principals
488 @principal ||= begin
488 @principal ||= begin
489 principals = []
489 principals = []
490 if project
490 if project
491 principals += project.principals.visible
491 principals += project.principals.visible
492 unless project.leaf?
492 unless project.leaf?
493 principals += Principal.member_of(project.descendants.visible).visible
493 principals += Principal.member_of(project.descendants.visible).visible
494 end
494 end
495 else
495 else
496 principals += Principal.member_of(all_projects).visible
496 principals += Principal.member_of(all_projects).visible
497 end
497 end
498 principals.uniq!
498 principals.uniq!
499 principals.sort!
499 principals.sort!
500 principals.reject! {|p| p.is_a?(GroupBuiltin)}
500 principals.reject! {|p| p.is_a?(GroupBuiltin)}
501 principals
501 principals
502 end
502 end
503 end
503 end
504
504
505 def users
505 def users
506 principals.select {|p| p.is_a?(User)}
506 principals.select {|p| p.is_a?(User)}
507 end
507 end
508
508
509 def author_values
509 def author_values
510 author_values = []
510 author_values = []
511 author_values << ["<< #{l(:label_me)} >>", "me"] if User.current.logged?
511 author_values << ["<< #{l(:label_me)} >>", "me"] if User.current.logged?
512 author_values += users.collect{|s| [s.name, s.id.to_s] }
512 author_values += users.collect{|s| [s.name, s.id.to_s] }
513 author_values
513 author_values
514 end
514 end
515
515
516 def assigned_to_values
516 def assigned_to_values
517 assigned_to_values = []
517 assigned_to_values = []
518 assigned_to_values << ["<< #{l(:label_me)} >>", "me"] if User.current.logged?
518 assigned_to_values << ["<< #{l(:label_me)} >>", "me"] if User.current.logged?
519 assigned_to_values += (Setting.issue_group_assignment? ? principals : users).collect{|s| [s.name, s.id.to_s] }
519 assigned_to_values += (Setting.issue_group_assignment? ? principals : users).collect{|s| [s.name, s.id.to_s] }
520 assigned_to_values
520 assigned_to_values
521 end
521 end
522
522
523 def fixed_version_values
523 def fixed_version_values
524 versions = []
524 versions = []
525 if project
525 if project
526 versions = project.shared_versions.to_a
526 versions = project.shared_versions.to_a
527 else
527 else
528 versions = Version.visible.where(:sharing => 'system').to_a
528 versions = Version.visible.where(:sharing => 'system').to_a
529 end
529 end
530 Version.sort_by_status(versions).collect{|s| ["#{s.project.name} - #{s.name}", s.id.to_s, l("version_status_#{s.status}")] }
530 Version.sort_by_status(versions).collect{|s| ["#{s.project.name} - #{s.name}", s.id.to_s, l("version_status_#{s.status}")] }
531 end
531 end
532
532
533 # Adds available filters
533 # Adds available filters
534 def initialize_available_filters
534 def initialize_available_filters
535 # implemented by sub-classes
535 # implemented by sub-classes
536 end
536 end
537 protected :initialize_available_filters
537 protected :initialize_available_filters
538
538
539 # Adds an available filter
539 # Adds an available filter
540 def add_available_filter(field, options)
540 def add_available_filter(field, options)
541 @available_filters ||= ActiveSupport::OrderedHash.new
541 @available_filters ||= ActiveSupport::OrderedHash.new
542 @available_filters[field] = QueryFilter.new(field, options)
542 @available_filters[field] = QueryFilter.new(field, options)
543 @available_filters
543 @available_filters
544 end
544 end
545
545
546 # Removes an available filter
546 # Removes an available filter
547 def delete_available_filter(field)
547 def delete_available_filter(field)
548 if @available_filters
548 if @available_filters
549 @available_filters.delete(field)
549 @available_filters.delete(field)
550 end
550 end
551 end
551 end
552
552
553 # Return a hash of available filters
553 # Return a hash of available filters
554 def available_filters
554 def available_filters
555 unless @available_filters
555 unless @available_filters
556 initialize_available_filters
556 initialize_available_filters
557 @available_filters ||= {}
557 @available_filters ||= {}
558 end
558 end
559 @available_filters
559 @available_filters
560 end
560 end
561
561
562 def add_filter(field, operator, values=nil)
562 def add_filter(field, operator, values=nil)
563 # values must be an array
563 # values must be an array
564 return unless values.nil? || values.is_a?(Array)
564 return unless values.nil? || values.is_a?(Array)
565 # check if field is defined as an available filter
565 # check if field is defined as an available filter
566 if available_filters.has_key? field
566 if available_filters.has_key? field
567 filter_options = available_filters[field]
567 filter_options = available_filters[field]
568 filters[field] = {:operator => operator, :values => (values || [''])}
568 filters[field] = {:operator => operator, :values => (values || [''])}
569 end
569 end
570 end
570 end
571
571
572 def add_short_filter(field, expression)
572 def add_short_filter(field, expression)
573 return unless expression && available_filters.has_key?(field)
573 return unless expression && available_filters.has_key?(field)
574 field_type = available_filters[field][:type]
574 field_type = available_filters[field][:type]
575 operators_by_filter_type[field_type].sort.reverse.detect do |operator|
575 operators_by_filter_type[field_type].sort.reverse.detect do |operator|
576 next unless expression =~ /^#{Regexp.escape(operator)}(.*)$/
576 next unless expression =~ /^#{Regexp.escape(operator)}(.*)$/
577 values = $1
577 values = $1
578 add_filter field, operator, values.present? ? values.split('|') : ['']
578 add_filter field, operator, values.present? ? values.split('|') : ['']
579 end || add_filter(field, '=', expression.to_s.split('|'))
579 end || add_filter(field, '=', expression.to_s.split('|'))
580 end
580 end
581
581
582 # Add multiple filters using +add_filter+
582 # Add multiple filters using +add_filter+
583 def add_filters(fields, operators, values)
583 def add_filters(fields, operators, values)
584 if fields.is_a?(Array) && operators.is_a?(Hash) && (values.nil? || values.is_a?(Hash))
584 if fields.is_a?(Array) && operators.is_a?(Hash) && (values.nil? || values.is_a?(Hash))
585 fields.each do |field|
585 fields.each do |field|
586 add_filter(field, operators[field], values && values[field])
586 add_filter(field, operators[field], values && values[field])
587 end
587 end
588 end
588 end
589 end
589 end
590
590
591 def has_filter?(field)
591 def has_filter?(field)
592 filters and filters[field]
592 filters and filters[field]
593 end
593 end
594
594
595 def type_for(field)
595 def type_for(field)
596 available_filters[field][:type] if available_filters.has_key?(field)
596 available_filters[field][:type] if available_filters.has_key?(field)
597 end
597 end
598
598
599 def operator_for(field)
599 def operator_for(field)
600 has_filter?(field) ? filters[field][:operator] : nil
600 has_filter?(field) ? filters[field][:operator] : nil
601 end
601 end
602
602
603 def values_for(field)
603 def values_for(field)
604 has_filter?(field) ? filters[field][:values] : nil
604 has_filter?(field) ? filters[field][:values] : nil
605 end
605 end
606
606
607 def value_for(field, index=0)
607 def value_for(field, index=0)
608 (values_for(field) || [])[index]
608 (values_for(field) || [])[index]
609 end
609 end
610
610
611 def label_for(field)
611 def label_for(field)
612 label = available_filters[field][:name] if available_filters.has_key?(field)
612 label = available_filters[field][:name] if available_filters.has_key?(field)
613 label ||= queried_class.human_attribute_name(field, :default => field)
613 label ||= queried_class.human_attribute_name(field, :default => field)
614 end
614 end
615
615
616 def self.add_available_column(column)
616 def self.add_available_column(column)
617 self.available_columns << (column) if column.is_a?(QueryColumn)
617 self.available_columns << (column) if column.is_a?(QueryColumn)
618 end
618 end
619
619
620 # Returns an array of columns that can be used to group the results
620 # Returns an array of columns that can be used to group the results
621 def groupable_columns
621 def groupable_columns
622 available_columns.select {|c| c.groupable}
622 available_columns.select {|c| c.groupable}
623 end
623 end
624
624
625 # Returns a Hash of columns and the key for sorting
625 # Returns a Hash of columns and the key for sorting
626 def sortable_columns
626 def sortable_columns
627 available_columns.inject({}) {|h, column|
627 available_columns.inject({}) {|h, column|
628 h[column.name.to_s] = column.sortable
628 h[column.name.to_s] = column.sortable
629 h
629 h
630 }
630 }
631 end
631 end
632
632
633 def columns
633 def columns
634 # preserve the column_names order
634 # preserve the column_names order
635 cols = (has_default_columns? ? default_columns_names : column_names).collect do |name|
635 cols = (has_default_columns? ? default_columns_names : column_names).collect do |name|
636 available_columns.find { |col| col.name == name }
636 available_columns.find { |col| col.name == name }
637 end.compact
637 end.compact
638 available_columns.select(&:frozen?) | cols
638 available_columns.select(&:frozen?) | cols
639 end
639 end
640
640
641 def inline_columns
641 def inline_columns
642 columns.select(&:inline?)
642 columns.select(&:inline?)
643 end
643 end
644
644
645 def block_columns
645 def block_columns
646 columns.reject(&:inline?)
646 columns.reject(&:inline?)
647 end
647 end
648
648
649 def available_inline_columns
649 def available_inline_columns
650 available_columns.select(&:inline?)
650 available_columns.select(&:inline?)
651 end
651 end
652
652
653 def available_block_columns
653 def available_block_columns
654 available_columns.reject(&:inline?)
654 available_columns.reject(&:inline?)
655 end
655 end
656
656
657 def available_totalable_columns
657 def available_totalable_columns
658 available_columns.select(&:totalable)
658 available_columns.select(&:totalable)
659 end
659 end
660
660
661 def default_columns_names
661 def default_columns_names
662 []
662 []
663 end
663 end
664
664
665 def default_totalable_names
665 def default_totalable_names
666 []
666 []
667 end
667 end
668
668
669 def column_names=(names)
669 def column_names=(names)
670 if names
670 if names
671 names = names.select {|n| n.is_a?(Symbol) || !n.blank? }
671 names = names.select {|n| n.is_a?(Symbol) || !n.blank? }
672 names = names.collect {|n| n.is_a?(Symbol) ? n : n.to_sym }
672 names = names.collect {|n| n.is_a?(Symbol) ? n : n.to_sym }
673 # Set column_names to nil if default columns
673 # Set column_names to nil if default columns
674 if names == default_columns_names
674 if names == default_columns_names
675 names = nil
675 names = nil
676 end
676 end
677 end
677 end
678 write_attribute(:column_names, names)
678 write_attribute(:column_names, names)
679 end
679 end
680
680
681 def has_column?(column)
681 def has_column?(column)
682 column_names && column_names.include?(column.is_a?(QueryColumn) ? column.name : column)
682 column_names && column_names.include?(column.is_a?(QueryColumn) ? column.name : column)
683 end
683 end
684
684
685 def has_custom_field_column?
685 def has_custom_field_column?
686 columns.any? {|column| column.is_a? QueryCustomFieldColumn}
686 columns.any? {|column| column.is_a? QueryCustomFieldColumn}
687 end
687 end
688
688
689 def has_default_columns?
689 def has_default_columns?
690 column_names.nil? || column_names.empty?
690 column_names.nil? || column_names.empty?
691 end
691 end
692
692
693 def totalable_columns
693 def totalable_columns
694 names = totalable_names
694 names = totalable_names
695 available_totalable_columns.select {|column| names.include?(column.name)}
695 available_totalable_columns.select {|column| names.include?(column.name)}
696 end
696 end
697
697
698 def totalable_names=(names)
698 def totalable_names=(names)
699 if names
699 if names
700 names = names.select(&:present?).map {|n| n.is_a?(Symbol) ? n : n.to_sym}
700 names = names.select(&:present?).map {|n| n.is_a?(Symbol) ? n : n.to_sym}
701 end
701 end
702 options[:totalable_names] = names
702 options[:totalable_names] = names
703 end
703 end
704
704
705 def totalable_names
705 def totalable_names
706 options[:totalable_names] || default_totalable_names || []
706 options[:totalable_names] || default_totalable_names || []
707 end
707 end
708
708
709 def sort_criteria=(arg)
709 def sort_criteria=(arg)
710 c = []
710 c = []
711 if arg.is_a?(Hash)
711 if arg.is_a?(Hash)
712 arg = arg.keys.sort.collect {|k| arg[k]}
712 arg = arg.keys.sort.collect {|k| arg[k]}
713 end
713 end
714 if arg
714 if arg
715 c = arg.select {|k,o| !k.to_s.blank?}.slice(0,3).collect {|k,o| [k.to_s, (o == 'desc' || o == false) ? 'desc' : 'asc']}
715 c = arg.select {|k,o| !k.to_s.blank?}.slice(0,3).collect {|k,o| [k.to_s, (o == 'desc' || o == false) ? 'desc' : 'asc']}
716 end
716 end
717 write_attribute(:sort_criteria, c)
717 write_attribute(:sort_criteria, c)
718 end
718 end
719
719
720 def sort_criteria
720 def sort_criteria
721 read_attribute(:sort_criteria) || []
721 read_attribute(:sort_criteria) || []
722 end
722 end
723
723
724 def sort_criteria_key(arg)
724 def sort_criteria_key(arg)
725 sort_criteria && sort_criteria[arg] && sort_criteria[arg].first
725 sort_criteria && sort_criteria[arg] && sort_criteria[arg].first
726 end
726 end
727
727
728 def sort_criteria_order(arg)
728 def sort_criteria_order(arg)
729 sort_criteria && sort_criteria[arg] && sort_criteria[arg].last
729 sort_criteria && sort_criteria[arg] && sort_criteria[arg].last
730 end
730 end
731
731
732 def sort_criteria_order_for(key)
732 def sort_criteria_order_for(key)
733 sort_criteria.detect {|k, order| key.to_s == k}.try(:last)
733 sort_criteria.detect {|k, order| key.to_s == k}.try(:last)
734 end
734 end
735
735
736 # Returns the SQL sort order that should be prepended for grouping
736 # Returns the SQL sort order that should be prepended for grouping
737 def group_by_sort_order
737 def group_by_sort_order
738 if column = group_by_column
738 if column = group_by_column
739 order = (sort_criteria_order_for(column.name) || column.default_order || 'asc').try(:upcase)
739 order = (sort_criteria_order_for(column.name) || column.default_order || 'asc').try(:upcase)
740 Array(column.sortable).map {|s| "#{s} #{order}"}
740 Array(column.sortable).map {|s| "#{s} #{order}"}
741 end
741 end
742 end
742 end
743
743
744 # Returns true if the query is a grouped query
744 # Returns true if the query is a grouped query
745 def grouped?
745 def grouped?
746 !group_by_column.nil?
746 !group_by_column.nil?
747 end
747 end
748
748
749 def group_by_column
749 def group_by_column
750 groupable_columns.detect {|c| c.groupable && c.name.to_s == group_by}
750 groupable_columns.detect {|c| c.groupable && c.name.to_s == group_by}
751 end
751 end
752
752
753 def group_by_statement
753 def group_by_statement
754 group_by_column.try(:groupable)
754 group_by_column.try(:groupable)
755 end
755 end
756
756
757 def project_statement
757 def project_statement
758 project_clauses = []
758 project_clauses = []
759 if project && !project.descendants.active.empty?
759 if project && !project.descendants.active.empty?
760 if has_filter?("subproject_id")
760 if has_filter?("subproject_id")
761 case operator_for("subproject_id")
761 case operator_for("subproject_id")
762 when '='
762 when '='
763 # include the selected subprojects
763 # include the selected subprojects
764 ids = [project.id] + values_for("subproject_id").each(&:to_i)
764 ids = [project.id] + values_for("subproject_id").each(&:to_i)
765 project_clauses << "#{Project.table_name}.id IN (%s)" % ids.join(',')
765 project_clauses << "#{Project.table_name}.id IN (%s)" % ids.join(',')
766 when '!*'
766 when '!*'
767 # main project only
767 # main project only
768 project_clauses << "#{Project.table_name}.id = %d" % project.id
768 project_clauses << "#{Project.table_name}.id = %d" % project.id
769 else
769 else
770 # all subprojects
770 # all subprojects
771 project_clauses << "#{Project.table_name}.lft >= #{project.lft} AND #{Project.table_name}.rgt <= #{project.rgt}"
771 project_clauses << "#{Project.table_name}.lft >= #{project.lft} AND #{Project.table_name}.rgt <= #{project.rgt}"
772 end
772 end
773 elsif Setting.display_subprojects_issues?
773 elsif Setting.display_subprojects_issues?
774 project_clauses << "#{Project.table_name}.lft >= #{project.lft} AND #{Project.table_name}.rgt <= #{project.rgt}"
774 project_clauses << "#{Project.table_name}.lft >= #{project.lft} AND #{Project.table_name}.rgt <= #{project.rgt}"
775 else
775 else
776 project_clauses << "#{Project.table_name}.id = %d" % project.id
776 project_clauses << "#{Project.table_name}.id = %d" % project.id
777 end
777 end
778 elsif project
778 elsif project
779 project_clauses << "#{Project.table_name}.id = %d" % project.id
779 project_clauses << "#{Project.table_name}.id = %d" % project.id
780 end
780 end
781 project_clauses.any? ? project_clauses.join(' AND ') : nil
781 project_clauses.any? ? project_clauses.join(' AND ') : nil
782 end
782 end
783
783
784 def statement
784 def statement
785 # filters clauses
785 # filters clauses
786 filters_clauses = []
786 filters_clauses = []
787 filters.each_key do |field|
787 filters.each_key do |field|
788 next if field == "subproject_id"
788 next if field == "subproject_id"
789 v = values_for(field).clone
789 v = values_for(field).clone
790 next unless v and !v.empty?
790 next unless v and !v.empty?
791 operator = operator_for(field)
791 operator = operator_for(field)
792
792
793 # "me" value substitution
793 # "me" value substitution
794 if %w(assigned_to_id author_id user_id watcher_id).include?(field)
794 if %w(assigned_to_id author_id user_id watcher_id).include?(field)
795 if v.delete("me")
795 if v.delete("me")
796 if User.current.logged?
796 if User.current.logged?
797 v.push(User.current.id.to_s)
797 v.push(User.current.id.to_s)
798 v += User.current.group_ids.map(&:to_s) if field == 'assigned_to_id'
798 v += User.current.group_ids.map(&:to_s) if field == 'assigned_to_id'
799 else
799 else
800 v.push("0")
800 v.push("0")
801 end
801 end
802 end
802 end
803 end
803 end
804
804
805 if field == 'project_id'
805 if field == 'project_id'
806 if v.delete('mine')
806 if v.delete('mine')
807 v += User.current.memberships.map(&:project_id).map(&:to_s)
807 v += User.current.memberships.map(&:project_id).map(&:to_s)
808 end
808 end
809 end
809 end
810
810
811 if field =~ /cf_(\d+)$/
811 if field =~ /^cf_(\d+)\.cf_(\d+)$/
812 filters_clauses << sql_for_chained_custom_field(field, operator, v, $1, $2)
813 elsif field =~ /cf_(\d+)$/
812 # custom field
814 # custom field
813 filters_clauses << sql_for_custom_field(field, operator, v, $1)
815 filters_clauses << sql_for_custom_field(field, operator, v, $1)
816 elsif field =~ /^cf_(\d+)\.(.+)$/
817 filters_clauses << sql_for_custom_field_attribute(field, operator, v, $1, $2)
814 elsif respond_to?(method = "sql_for_#{field.gsub('.','_')}_field")
818 elsif respond_to?(method = "sql_for_#{field.gsub('.','_')}_field")
815 # specific statement
819 # specific statement
816 filters_clauses << send(method, field, operator, v)
820 filters_clauses << send(method, field, operator, v)
817 else
821 else
818 # regular field
822 # regular field
819 filters_clauses << '(' + sql_for_field(field, operator, v, queried_table_name, field) + ')'
823 filters_clauses << '(' + sql_for_field(field, operator, v, queried_table_name, field) + ')'
820 end
824 end
821 end if filters and valid?
825 end if filters and valid?
822
826
823 if (c = group_by_column) && c.is_a?(QueryCustomFieldColumn)
827 if (c = group_by_column) && c.is_a?(QueryCustomFieldColumn)
824 # Excludes results for which the grouped custom field is not visible
828 # Excludes results for which the grouped custom field is not visible
825 filters_clauses << c.custom_field.visibility_by_project_condition
829 filters_clauses << c.custom_field.visibility_by_project_condition
826 end
830 end
827
831
828 filters_clauses << project_statement
832 filters_clauses << project_statement
829 filters_clauses.reject!(&:blank?)
833 filters_clauses.reject!(&:blank?)
830
834
831 filters_clauses.any? ? filters_clauses.join(' AND ') : nil
835 filters_clauses.any? ? filters_clauses.join(' AND ') : nil
832 end
836 end
833
837
834 # Returns the sum of values for the given column
838 # Returns the sum of values for the given column
835 def total_for(column)
839 def total_for(column)
836 total_with_scope(column, base_scope)
840 total_with_scope(column, base_scope)
837 end
841 end
838
842
839 # Returns a hash of the sum of the given column for each group,
843 # Returns a hash of the sum of the given column for each group,
840 # or nil if the query is not grouped
844 # or nil if the query is not grouped
841 def total_by_group_for(column)
845 def total_by_group_for(column)
842 grouped_query do |scope|
846 grouped_query do |scope|
843 total_with_scope(column, scope)
847 total_with_scope(column, scope)
844 end
848 end
845 end
849 end
846
850
847 def totals
851 def totals
848 totals = totalable_columns.map {|column| [column, total_for(column)]}
852 totals = totalable_columns.map {|column| [column, total_for(column)]}
849 yield totals if block_given?
853 yield totals if block_given?
850 totals
854 totals
851 end
855 end
852
856
853 def totals_by_group
857 def totals_by_group
854 totals = totalable_columns.map {|column| [column, total_by_group_for(column)]}
858 totals = totalable_columns.map {|column| [column, total_by_group_for(column)]}
855 yield totals if block_given?
859 yield totals if block_given?
856 totals
860 totals
857 end
861 end
858
862
859 private
863 private
860
864
861 def grouped_query(&block)
865 def grouped_query(&block)
862 r = nil
866 r = nil
863 if grouped?
867 if grouped?
864 begin
868 begin
865 # Rails3 will raise an (unexpected) RecordNotFound if there's only a nil group value
869 # Rails3 will raise an (unexpected) RecordNotFound if there's only a nil group value
866 r = yield base_group_scope
870 r = yield base_group_scope
867 rescue ActiveRecord::RecordNotFound
871 rescue ActiveRecord::RecordNotFound
868 r = {nil => yield(base_scope)}
872 r = {nil => yield(base_scope)}
869 end
873 end
870 c = group_by_column
874 c = group_by_column
871 if c.is_a?(QueryCustomFieldColumn)
875 if c.is_a?(QueryCustomFieldColumn)
872 r = r.keys.inject({}) {|h, k| h[c.custom_field.cast_value(k)] = r[k]; h}
876 r = r.keys.inject({}) {|h, k| h[c.custom_field.cast_value(k)] = r[k]; h}
873 end
877 end
874 end
878 end
875 r
879 r
876 rescue ::ActiveRecord::StatementInvalid => e
880 rescue ::ActiveRecord::StatementInvalid => e
877 raise StatementInvalid.new(e.message)
881 raise StatementInvalid.new(e.message)
878 end
882 end
879
883
880 def total_with_scope(column, scope)
884 def total_with_scope(column, scope)
881 unless column.is_a?(QueryColumn)
885 unless column.is_a?(QueryColumn)
882 column = column.to_sym
886 column = column.to_sym
883 column = available_totalable_columns.detect {|c| c.name == column}
887 column = available_totalable_columns.detect {|c| c.name == column}
884 end
888 end
885 if column.is_a?(QueryCustomFieldColumn)
889 if column.is_a?(QueryCustomFieldColumn)
886 custom_field = column.custom_field
890 custom_field = column.custom_field
887 send "total_for_custom_field", custom_field, scope
891 send "total_for_custom_field", custom_field, scope
888 else
892 else
889 send "total_for_#{column.name}", scope
893 send "total_for_#{column.name}", scope
890 end
894 end
891 rescue ::ActiveRecord::StatementInvalid => e
895 rescue ::ActiveRecord::StatementInvalid => e
892 raise StatementInvalid.new(e.message)
896 raise StatementInvalid.new(e.message)
893 end
897 end
894
898
895 def base_scope
899 def base_scope
896 raise "unimplemented"
900 raise "unimplemented"
897 end
901 end
898
902
899 def base_group_scope
903 def base_group_scope
900 base_scope.
904 base_scope.
901 joins(joins_for_order_statement(group_by_statement)).
905 joins(joins_for_order_statement(group_by_statement)).
902 group(group_by_statement)
906 group(group_by_statement)
903 end
907 end
904
908
905 def total_for_custom_field(custom_field, scope, &block)
909 def total_for_custom_field(custom_field, scope, &block)
906 total = custom_field.format.total_for_scope(custom_field, scope)
910 total = custom_field.format.total_for_scope(custom_field, scope)
907 total = map_total(total) {|t| custom_field.format.cast_total_value(custom_field, t)}
911 total = map_total(total) {|t| custom_field.format.cast_total_value(custom_field, t)}
908 total
912 total
909 end
913 end
910
914
911 def map_total(total, &block)
915 def map_total(total, &block)
912 if total.is_a?(Hash)
916 if total.is_a?(Hash)
913 total.keys.each {|k| total[k] = yield total[k]}
917 total.keys.each {|k| total[k] = yield total[k]}
914 else
918 else
915 total = yield total
919 total = yield total
916 end
920 end
917 total
921 total
918 end
922 end
919
923
920 def sql_for_custom_field(field, operator, value, custom_field_id)
924 def sql_for_custom_field(field, operator, value, custom_field_id)
921 db_table = CustomValue.table_name
925 db_table = CustomValue.table_name
922 db_field = 'value'
926 db_field = 'value'
923 filter = @available_filters[field]
927 filter = @available_filters[field]
924 return nil unless filter
928 return nil unless filter
925 if filter[:field].format.target_class && filter[:field].format.target_class <= User
929 if filter[:field].format.target_class && filter[:field].format.target_class <= User
926 if value.delete('me')
930 if value.delete('me')
927 value.push User.current.id.to_s
931 value.push User.current.id.to_s
928 end
932 end
929 end
933 end
930 not_in = nil
934 not_in = nil
931 if operator == '!'
935 if operator == '!'
932 # Makes ! operator work for custom fields with multiple values
936 # Makes ! operator work for custom fields with multiple values
933 operator = '='
937 operator = '='
934 not_in = 'NOT'
938 not_in = 'NOT'
935 end
939 end
936 customized_key = "id"
940 customized_key = "id"
937 customized_class = queried_class
941 customized_class = queried_class
938 if field =~ /^(.+)\.cf_/
942 if field =~ /^(.+)\.cf_/
939 assoc = $1
943 assoc = $1
940 customized_key = "#{assoc}_id"
944 customized_key = "#{assoc}_id"
941 customized_class = queried_class.reflect_on_association(assoc.to_sym).klass.base_class rescue nil
945 customized_class = queried_class.reflect_on_association(assoc.to_sym).klass.base_class rescue nil
942 raise "Unknown #{queried_class.name} association #{assoc}" unless customized_class
946 raise "Unknown #{queried_class.name} association #{assoc}" unless customized_class
943 end
947 end
944 where = sql_for_field(field, operator, value, db_table, db_field, true)
948 where = sql_for_field(field, operator, value, db_table, db_field, true)
945 if operator =~ /[<>]/
949 if operator =~ /[<>]/
946 where = "(#{where}) AND #{db_table}.#{db_field} <> ''"
950 where = "(#{where}) AND #{db_table}.#{db_field} <> ''"
947 end
951 end
948 "#{queried_table_name}.#{customized_key} #{not_in} IN (" +
952 "#{queried_table_name}.#{customized_key} #{not_in} IN (" +
949 "SELECT #{customized_class.table_name}.id FROM #{customized_class.table_name}" +
953 "SELECT #{customized_class.table_name}.id FROM #{customized_class.table_name}" +
950 " 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}" +
954 " 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}" +
951 " WHERE (#{where}) AND (#{filter[:field].visibility_by_project_condition}))"
955 " WHERE (#{where}) AND (#{filter[:field].visibility_by_project_condition}))"
952 end
956 end
953
957
958 def sql_for_chained_custom_field(field, operator, value, custom_field_id, chained_custom_field_id)
959 not_in = nil
960 if operator == '!'
961 # Makes ! operator work for custom fields with multiple values
962 operator = '='
963 not_in = 'NOT'
964 end
965
966 filter = available_filters[field]
967 target_class = filter[:through].format.target_class
968
969 "#{queried_table_name}.id #{not_in} IN (" +
970 "SELECT customized_id FROM #{CustomValue.table_name}" +
971 " WHERE customized_type='#{queried_class}' AND custom_field_id=#{custom_field_id}" +
972 " AND value <> '' AND CAST(value AS integer) IN (" +
973 " SELECT customized_id FROM #{CustomValue.table_name}" +
974 " WHERE customized_type='#{target_class}' AND custom_field_id=#{chained_custom_field_id}" +
975 " AND #{sql_for_field(field, operator, value, CustomValue.table_name, 'value')}))"
976
977 end
978
979 def sql_for_custom_field_attribute(field, operator, value, custom_field_id, attribute)
980 attribute = 'effective_date' if attribute == 'due_date'
981 not_in = nil
982 if operator == '!'
983 # Makes ! operator work for custom fields with multiple values
984 operator = '='
985 not_in = 'NOT'
986 end
987
988 filter = available_filters[field]
989 target_table_name = filter[:field].format.target_class.table_name
990
991 "#{queried_table_name}.id #{not_in} IN (" +
992 "SELECT customized_id FROM #{CustomValue.table_name}" +
993 " WHERE customized_type='#{queried_class}' AND custom_field_id=#{custom_field_id}" +
994 " AND value <> '' AND CAST(value AS integer) IN (" +
995 " SELECT id FROM #{target_table_name} WHERE #{sql_for_field(field, operator, value, filter[:field].format.target_class.table_name, attribute)}))"
996 end
997
954 # Helper method to generate the WHERE sql for a +field+, +operator+ and a +value+
998 # Helper method to generate the WHERE sql for a +field+, +operator+ and a +value+
955 def sql_for_field(field, operator, value, db_table, db_field, is_custom_filter=false)
999 def sql_for_field(field, operator, value, db_table, db_field, is_custom_filter=false)
956 sql = ''
1000 sql = ''
957 case operator
1001 case operator
958 when "="
1002 when "="
959 if value.any?
1003 if value.any?
960 case type_for(field)
1004 case type_for(field)
961 when :date, :date_past
1005 when :date, :date_past
962 sql = date_clause(db_table, db_field, parse_date(value.first), parse_date(value.first), is_custom_filter)
1006 sql = date_clause(db_table, db_field, parse_date(value.first), parse_date(value.first), is_custom_filter)
963 when :integer
1007 when :integer
964 int_values = value.first.to_s.scan(/[+-]?\d+/).map(&:to_i).join(",")
1008 int_values = value.first.to_s.scan(/[+-]?\d+/).map(&:to_i).join(",")
965 if int_values.present?
1009 if int_values.present?
966 if is_custom_filter
1010 if is_custom_filter
967 sql = "(#{db_table}.#{db_field} <> '' AND CAST(CASE #{db_table}.#{db_field} WHEN '' THEN '0' ELSE #{db_table}.#{db_field} END AS decimal(30,3)) IN (#{int_values}))"
1011 sql = "(#{db_table}.#{db_field} <> '' AND CAST(CASE #{db_table}.#{db_field} WHEN '' THEN '0' ELSE #{db_table}.#{db_field} END AS decimal(30,3)) IN (#{int_values}))"
968 else
1012 else
969 sql = "#{db_table}.#{db_field} IN (#{int_values})"
1013 sql = "#{db_table}.#{db_field} IN (#{int_values})"
970 end
1014 end
971 else
1015 else
972 sql = "1=0"
1016 sql = "1=0"
973 end
1017 end
974 when :float
1018 when :float
975 if is_custom_filter
1019 if is_custom_filter
976 sql = "(#{db_table}.#{db_field} <> '' AND CAST(CASE #{db_table}.#{db_field} WHEN '' THEN '0' ELSE #{db_table}.#{db_field} END AS decimal(30,3)) BETWEEN #{value.first.to_f - 1e-5} AND #{value.first.to_f + 1e-5})"
1020 sql = "(#{db_table}.#{db_field} <> '' AND CAST(CASE #{db_table}.#{db_field} WHEN '' THEN '0' ELSE #{db_table}.#{db_field} END AS decimal(30,3)) BETWEEN #{value.first.to_f - 1e-5} AND #{value.first.to_f + 1e-5})"
977 else
1021 else
978 sql = "#{db_table}.#{db_field} BETWEEN #{value.first.to_f - 1e-5} AND #{value.first.to_f + 1e-5}"
1022 sql = "#{db_table}.#{db_field} BETWEEN #{value.first.to_f - 1e-5} AND #{value.first.to_f + 1e-5}"
979 end
1023 end
980 else
1024 else
981 sql = queried_class.send(:sanitize_sql_for_conditions, ["#{db_table}.#{db_field} IN (?)", value])
1025 sql = queried_class.send(:sanitize_sql_for_conditions, ["#{db_table}.#{db_field} IN (?)", value])
982 end
1026 end
983 else
1027 else
984 # IN an empty set
1028 # IN an empty set
985 sql = "1=0"
1029 sql = "1=0"
986 end
1030 end
987 when "!"
1031 when "!"
988 if value.any?
1032 if value.any?
989 sql = queried_class.send(:sanitize_sql_for_conditions, ["(#{db_table}.#{db_field} IS NULL OR #{db_table}.#{db_field} NOT IN (?))", value])
1033 sql = queried_class.send(:sanitize_sql_for_conditions, ["(#{db_table}.#{db_field} IS NULL OR #{db_table}.#{db_field} NOT IN (?))", value])
990 else
1034 else
991 # NOT IN an empty set
1035 # NOT IN an empty set
992 sql = "1=1"
1036 sql = "1=1"
993 end
1037 end
994 when "!*"
1038 when "!*"
995 sql = "#{db_table}.#{db_field} IS NULL"
1039 sql = "#{db_table}.#{db_field} IS NULL"
996 sql << " OR #{db_table}.#{db_field} = ''" if is_custom_filter
1040 sql << " OR #{db_table}.#{db_field} = ''" if is_custom_filter
997 when "*"
1041 when "*"
998 sql = "#{db_table}.#{db_field} IS NOT NULL"
1042 sql = "#{db_table}.#{db_field} IS NOT NULL"
999 sql << " AND #{db_table}.#{db_field} <> ''" if is_custom_filter
1043 sql << " AND #{db_table}.#{db_field} <> ''" if is_custom_filter
1000 when ">="
1044 when ">="
1001 if [:date, :date_past].include?(type_for(field))
1045 if [:date, :date_past].include?(type_for(field))
1002 sql = date_clause(db_table, db_field, parse_date(value.first), nil, is_custom_filter)
1046 sql = date_clause(db_table, db_field, parse_date(value.first), nil, is_custom_filter)
1003 else
1047 else
1004 if is_custom_filter
1048 if is_custom_filter
1005 sql = "(#{db_table}.#{db_field} <> '' AND CAST(CASE #{db_table}.#{db_field} WHEN '' THEN '0' ELSE #{db_table}.#{db_field} END AS decimal(30,3)) >= #{value.first.to_f})"
1049 sql = "(#{db_table}.#{db_field} <> '' AND CAST(CASE #{db_table}.#{db_field} WHEN '' THEN '0' ELSE #{db_table}.#{db_field} END AS decimal(30,3)) >= #{value.first.to_f})"
1006 else
1050 else
1007 sql = "#{db_table}.#{db_field} >= #{value.first.to_f}"
1051 sql = "#{db_table}.#{db_field} >= #{value.first.to_f}"
1008 end
1052 end
1009 end
1053 end
1010 when "<="
1054 when "<="
1011 if [:date, :date_past].include?(type_for(field))
1055 if [:date, :date_past].include?(type_for(field))
1012 sql = date_clause(db_table, db_field, nil, parse_date(value.first), is_custom_filter)
1056 sql = date_clause(db_table, db_field, nil, parse_date(value.first), is_custom_filter)
1013 else
1057 else
1014 if is_custom_filter
1058 if is_custom_filter
1015 sql = "(#{db_table}.#{db_field} <> '' AND CAST(CASE #{db_table}.#{db_field} WHEN '' THEN '0' ELSE #{db_table}.#{db_field} END AS decimal(30,3)) <= #{value.first.to_f})"
1059 sql = "(#{db_table}.#{db_field} <> '' AND CAST(CASE #{db_table}.#{db_field} WHEN '' THEN '0' ELSE #{db_table}.#{db_field} END AS decimal(30,3)) <= #{value.first.to_f})"
1016 else
1060 else
1017 sql = "#{db_table}.#{db_field} <= #{value.first.to_f}"
1061 sql = "#{db_table}.#{db_field} <= #{value.first.to_f}"
1018 end
1062 end
1019 end
1063 end
1020 when "><"
1064 when "><"
1021 if [:date, :date_past].include?(type_for(field))
1065 if [:date, :date_past].include?(type_for(field))
1022 sql = date_clause(db_table, db_field, parse_date(value[0]), parse_date(value[1]), is_custom_filter)
1066 sql = date_clause(db_table, db_field, parse_date(value[0]), parse_date(value[1]), is_custom_filter)
1023 else
1067 else
1024 if is_custom_filter
1068 if is_custom_filter
1025 sql = "(#{db_table}.#{db_field} <> '' AND CAST(CASE #{db_table}.#{db_field} WHEN '' THEN '0' ELSE #{db_table}.#{db_field} END AS decimal(30,3)) BETWEEN #{value[0].to_f} AND #{value[1].to_f})"
1069 sql = "(#{db_table}.#{db_field} <> '' AND CAST(CASE #{db_table}.#{db_field} WHEN '' THEN '0' ELSE #{db_table}.#{db_field} END AS decimal(30,3)) BETWEEN #{value[0].to_f} AND #{value[1].to_f})"
1026 else
1070 else
1027 sql = "#{db_table}.#{db_field} BETWEEN #{value[0].to_f} AND #{value[1].to_f}"
1071 sql = "#{db_table}.#{db_field} BETWEEN #{value[0].to_f} AND #{value[1].to_f}"
1028 end
1072 end
1029 end
1073 end
1030 when "o"
1074 when "o"
1031 sql = "#{queried_table_name}.status_id IN (SELECT id FROM #{IssueStatus.table_name} WHERE is_closed=#{self.class.connection.quoted_false})" if field == "status_id"
1075 sql = "#{queried_table_name}.status_id IN (SELECT id FROM #{IssueStatus.table_name} WHERE is_closed=#{self.class.connection.quoted_false})" if field == "status_id"
1032 when "c"
1076 when "c"
1033 sql = "#{queried_table_name}.status_id IN (SELECT id FROM #{IssueStatus.table_name} WHERE is_closed=#{self.class.connection.quoted_true})" if field == "status_id"
1077 sql = "#{queried_table_name}.status_id IN (SELECT id FROM #{IssueStatus.table_name} WHERE is_closed=#{self.class.connection.quoted_true})" if field == "status_id"
1034 when "><t-"
1078 when "><t-"
1035 # between today - n days and today
1079 # between today - n days and today
1036 sql = relative_date_clause(db_table, db_field, - value.first.to_i, 0, is_custom_filter)
1080 sql = relative_date_clause(db_table, db_field, - value.first.to_i, 0, is_custom_filter)
1037 when ">t-"
1081 when ">t-"
1038 # >= today - n days
1082 # >= today - n days
1039 sql = relative_date_clause(db_table, db_field, - value.first.to_i, nil, is_custom_filter)
1083 sql = relative_date_clause(db_table, db_field, - value.first.to_i, nil, is_custom_filter)
1040 when "<t-"
1084 when "<t-"
1041 # <= today - n days
1085 # <= today - n days
1042 sql = relative_date_clause(db_table, db_field, nil, - value.first.to_i, is_custom_filter)
1086 sql = relative_date_clause(db_table, db_field, nil, - value.first.to_i, is_custom_filter)
1043 when "t-"
1087 when "t-"
1044 # = n days in past
1088 # = n days in past
1045 sql = relative_date_clause(db_table, db_field, - value.first.to_i, - value.first.to_i, is_custom_filter)
1089 sql = relative_date_clause(db_table, db_field, - value.first.to_i, - value.first.to_i, is_custom_filter)
1046 when "><t+"
1090 when "><t+"
1047 # between today and today + n days
1091 # between today and today + n days
1048 sql = relative_date_clause(db_table, db_field, 0, value.first.to_i, is_custom_filter)
1092 sql = relative_date_clause(db_table, db_field, 0, value.first.to_i, is_custom_filter)
1049 when ">t+"
1093 when ">t+"
1050 # >= today + n days
1094 # >= today + n days
1051 sql = relative_date_clause(db_table, db_field, value.first.to_i, nil, is_custom_filter)
1095 sql = relative_date_clause(db_table, db_field, value.first.to_i, nil, is_custom_filter)
1052 when "<t+"
1096 when "<t+"
1053 # <= today + n days
1097 # <= today + n days
1054 sql = relative_date_clause(db_table, db_field, nil, value.first.to_i, is_custom_filter)
1098 sql = relative_date_clause(db_table, db_field, nil, value.first.to_i, is_custom_filter)
1055 when "t+"
1099 when "t+"
1056 # = today + n days
1100 # = today + n days
1057 sql = relative_date_clause(db_table, db_field, value.first.to_i, value.first.to_i, is_custom_filter)
1101 sql = relative_date_clause(db_table, db_field, value.first.to_i, value.first.to_i, is_custom_filter)
1058 when "t"
1102 when "t"
1059 # = today
1103 # = today
1060 sql = relative_date_clause(db_table, db_field, 0, 0, is_custom_filter)
1104 sql = relative_date_clause(db_table, db_field, 0, 0, is_custom_filter)
1061 when "ld"
1105 when "ld"
1062 # = yesterday
1106 # = yesterday
1063 sql = relative_date_clause(db_table, db_field, -1, -1, is_custom_filter)
1107 sql = relative_date_clause(db_table, db_field, -1, -1, is_custom_filter)
1064 when "w"
1108 when "w"
1065 # = this week
1109 # = this week
1066 first_day_of_week = l(:general_first_day_of_week).to_i
1110 first_day_of_week = l(:general_first_day_of_week).to_i
1067 day_of_week = User.current.today.cwday
1111 day_of_week = User.current.today.cwday
1068 days_ago = (day_of_week >= first_day_of_week ? day_of_week - first_day_of_week : day_of_week + 7 - first_day_of_week)
1112 days_ago = (day_of_week >= first_day_of_week ? day_of_week - first_day_of_week : day_of_week + 7 - first_day_of_week)
1069 sql = relative_date_clause(db_table, db_field, - days_ago, - days_ago + 6, is_custom_filter)
1113 sql = relative_date_clause(db_table, db_field, - days_ago, - days_ago + 6, is_custom_filter)
1070 when "lw"
1114 when "lw"
1071 # = last week
1115 # = last week
1072 first_day_of_week = l(:general_first_day_of_week).to_i
1116 first_day_of_week = l(:general_first_day_of_week).to_i
1073 day_of_week = User.current.today.cwday
1117 day_of_week = User.current.today.cwday
1074 days_ago = (day_of_week >= first_day_of_week ? day_of_week - first_day_of_week : day_of_week + 7 - first_day_of_week)
1118 days_ago = (day_of_week >= first_day_of_week ? day_of_week - first_day_of_week : day_of_week + 7 - first_day_of_week)
1075 sql = relative_date_clause(db_table, db_field, - days_ago - 7, - days_ago - 1, is_custom_filter)
1119 sql = relative_date_clause(db_table, db_field, - days_ago - 7, - days_ago - 1, is_custom_filter)
1076 when "l2w"
1120 when "l2w"
1077 # = last 2 weeks
1121 # = last 2 weeks
1078 first_day_of_week = l(:general_first_day_of_week).to_i
1122 first_day_of_week = l(:general_first_day_of_week).to_i
1079 day_of_week = User.current.today.cwday
1123 day_of_week = User.current.today.cwday
1080 days_ago = (day_of_week >= first_day_of_week ? day_of_week - first_day_of_week : day_of_week + 7 - first_day_of_week)
1124 days_ago = (day_of_week >= first_day_of_week ? day_of_week - first_day_of_week : day_of_week + 7 - first_day_of_week)
1081 sql = relative_date_clause(db_table, db_field, - days_ago - 14, - days_ago - 1, is_custom_filter)
1125 sql = relative_date_clause(db_table, db_field, - days_ago - 14, - days_ago - 1, is_custom_filter)
1082 when "m"
1126 when "m"
1083 # = this month
1127 # = this month
1084 date = User.current.today
1128 date = User.current.today
1085 sql = date_clause(db_table, db_field, date.beginning_of_month, date.end_of_month, is_custom_filter)
1129 sql = date_clause(db_table, db_field, date.beginning_of_month, date.end_of_month, is_custom_filter)
1086 when "lm"
1130 when "lm"
1087 # = last month
1131 # = last month
1088 date = User.current.today.prev_month
1132 date = User.current.today.prev_month
1089 sql = date_clause(db_table, db_field, date.beginning_of_month, date.end_of_month, is_custom_filter)
1133 sql = date_clause(db_table, db_field, date.beginning_of_month, date.end_of_month, is_custom_filter)
1090 when "y"
1134 when "y"
1091 # = this year
1135 # = this year
1092 date = User.current.today
1136 date = User.current.today
1093 sql = date_clause(db_table, db_field, date.beginning_of_year, date.end_of_year, is_custom_filter)
1137 sql = date_clause(db_table, db_field, date.beginning_of_year, date.end_of_year, is_custom_filter)
1094 when "~"
1138 when "~"
1095 sql = sql_contains("#{db_table}.#{db_field}", value.first)
1139 sql = sql_contains("#{db_table}.#{db_field}", value.first)
1096 when "!~"
1140 when "!~"
1097 sql = sql_contains("#{db_table}.#{db_field}", value.first, false)
1141 sql = sql_contains("#{db_table}.#{db_field}", value.first, false)
1098 else
1142 else
1099 raise "Unknown query operator #{operator}"
1143 raise "Unknown query operator #{operator}"
1100 end
1144 end
1101
1145
1102 return sql
1146 return sql
1103 end
1147 end
1104
1148
1105 # Returns a SQL LIKE statement with wildcards
1149 # Returns a SQL LIKE statement with wildcards
1106 def sql_contains(db_field, value, match=true)
1150 def sql_contains(db_field, value, match=true)
1107 queried_class.send :sanitize_sql_for_conditions,
1151 queried_class.send :sanitize_sql_for_conditions,
1108 [Redmine::Database.like(db_field, '?', :match => match), "%#{value}%"]
1152 [Redmine::Database.like(db_field, '?', :match => match), "%#{value}%"]
1109 end
1153 end
1110
1154
1111 # Adds a filter for the given custom field
1155 # Adds a filter for the given custom field
1112 def add_custom_field_filter(field, assoc=nil)
1156 def add_custom_field_filter(field, assoc=nil)
1113 options = field.query_filter_options(self)
1157 options = field.query_filter_options(self)
1114
1158
1115 filter_id = "cf_#{field.id}"
1159 filter_id = "cf_#{field.id}"
1116 filter_name = field.name
1160 filter_name = field.name
1117 if assoc.present?
1161 if assoc.present?
1118 filter_id = "#{assoc}.#{filter_id}"
1162 filter_id = "#{assoc}.#{filter_id}"
1119 filter_name = l("label_attribute_of_#{assoc}", :name => filter_name)
1163 filter_name = l("label_attribute_of_#{assoc}", :name => filter_name)
1120 end
1164 end
1121 add_available_filter filter_id, options.merge({
1165 add_available_filter filter_id, options.merge({
1122 :name => filter_name,
1166 :name => filter_name,
1123 :field => field
1167 :field => field
1124 })
1168 })
1125 end
1169 end
1126
1170
1171 # Adds filters for custom fields associated to the custom field target class
1172 # Eg. having a version custom field "Milestone" for issues and a date custom field "Release date"
1173 # for versions, it will add an issue filter on Milestone'e Release date.
1174 def add_chained_custom_field_filters(field)
1175 klass = field.format.target_class
1176 if klass
1177 CustomField.where(:is_filter => true, :type => "#{klass.name}CustomField").each do |chained|
1178 options = chained.query_filter_options(self)
1179
1180 filter_id = "cf_#{field.id}.cf_#{chained.id}"
1181 filter_name = chained.name
1182
1183 add_available_filter filter_id, options.merge({
1184 :name => l(:label_attribute_of_object, :name => chained.name, :object_name => field.name),
1185 :field => chained,
1186 :through => field
1187 })
1188 end
1189 end
1190 end
1191
1127 # Adds filters for the given custom fields scope
1192 # Adds filters for the given custom fields scope
1128 def add_custom_fields_filters(scope, assoc=nil)
1193 def add_custom_fields_filters(scope, assoc=nil)
1129 scope.visible.where(:is_filter => true).sorted.each do |field|
1194 scope.visible.where(:is_filter => true).sorted.each do |field|
1130 add_custom_field_filter(field, assoc)
1195 add_custom_field_filter(field, assoc)
1196 if assoc.nil?
1197 add_chained_custom_field_filters(field)
1198
1199 if field.format.target_class && field.format.target_class == Version
1200 add_available_filter "cf_#{field.id}.due_date",
1201 :type => :date,
1202 :field => field,
1203 :name => l(:label_attribute_of_object, :name => l(:field_effective_date), :object_name => field.name)
1204
1205 add_available_filter "cf_#{field.id}.status",
1206 :type => :list,
1207 :field => field,
1208 :name => l(:label_attribute_of_object, :name => l(:field_status), :object_name => field.name),
1209 :values => Version::VERSION_STATUSES.map{|s| [l("version_status_#{s}"), s] }
1210 end
1211 end
1131 end
1212 end
1132 end
1213 end
1133
1214
1134 # Adds filters for the given associations custom fields
1215 # Adds filters for the given associations custom fields
1135 def add_associations_custom_fields_filters(*associations)
1216 def add_associations_custom_fields_filters(*associations)
1136 fields_by_class = CustomField.visible.where(:is_filter => true).group_by(&:class)
1217 fields_by_class = CustomField.visible.where(:is_filter => true).group_by(&:class)
1137 associations.each do |assoc|
1218 associations.each do |assoc|
1138 association_klass = queried_class.reflect_on_association(assoc).klass
1219 association_klass = queried_class.reflect_on_association(assoc).klass
1139 fields_by_class.each do |field_class, fields|
1220 fields_by_class.each do |field_class, fields|
1140 if field_class.customized_class <= association_klass
1221 if field_class.customized_class <= association_klass
1141 fields.sort.each do |field|
1222 fields.sort.each do |field|
1142 add_custom_field_filter(field, assoc)
1223 add_custom_field_filter(field, assoc)
1143 end
1224 end
1144 end
1225 end
1145 end
1226 end
1146 end
1227 end
1147 end
1228 end
1148
1229
1149 def quoted_time(time, is_custom_filter)
1230 def quoted_time(time, is_custom_filter)
1150 if is_custom_filter
1231 if is_custom_filter
1151 # Custom field values are stored as strings in the DB
1232 # Custom field values are stored as strings in the DB
1152 # using this format that does not depend on DB date representation
1233 # using this format that does not depend on DB date representation
1153 time.strftime("%Y-%m-%d %H:%M:%S")
1234 time.strftime("%Y-%m-%d %H:%M:%S")
1154 else
1235 else
1155 self.class.connection.quoted_date(time)
1236 self.class.connection.quoted_date(time)
1156 end
1237 end
1157 end
1238 end
1158
1239
1159 def date_for_user_time_zone(y, m, d)
1240 def date_for_user_time_zone(y, m, d)
1160 if tz = User.current.time_zone
1241 if tz = User.current.time_zone
1161 tz.local y, m, d
1242 tz.local y, m, d
1162 else
1243 else
1163 Time.local y, m, d
1244 Time.local y, m, d
1164 end
1245 end
1165 end
1246 end
1166
1247
1167 # Returns a SQL clause for a date or datetime field.
1248 # Returns a SQL clause for a date or datetime field.
1168 def date_clause(table, field, from, to, is_custom_filter)
1249 def date_clause(table, field, from, to, is_custom_filter)
1169 s = []
1250 s = []
1170 if from
1251 if from
1171 if from.is_a?(Date)
1252 if from.is_a?(Date)
1172 from = date_for_user_time_zone(from.year, from.month, from.day).yesterday.end_of_day
1253 from = date_for_user_time_zone(from.year, from.month, from.day).yesterday.end_of_day
1173 else
1254 else
1174 from = from - 1 # second
1255 from = from - 1 # second
1175 end
1256 end
1176 if self.class.default_timezone == :utc
1257 if self.class.default_timezone == :utc
1177 from = from.utc
1258 from = from.utc
1178 end
1259 end
1179 s << ("#{table}.#{field} > '%s'" % [quoted_time(from, is_custom_filter)])
1260 s << ("#{table}.#{field} > '%s'" % [quoted_time(from, is_custom_filter)])
1180 end
1261 end
1181 if to
1262 if to
1182 if to.is_a?(Date)
1263 if to.is_a?(Date)
1183 to = date_for_user_time_zone(to.year, to.month, to.day).end_of_day
1264 to = date_for_user_time_zone(to.year, to.month, to.day).end_of_day
1184 end
1265 end
1185 if self.class.default_timezone == :utc
1266 if self.class.default_timezone == :utc
1186 to = to.utc
1267 to = to.utc
1187 end
1268 end
1188 s << ("#{table}.#{field} <= '%s'" % [quoted_time(to, is_custom_filter)])
1269 s << ("#{table}.#{field} <= '%s'" % [quoted_time(to, is_custom_filter)])
1189 end
1270 end
1190 s.join(' AND ')
1271 s.join(' AND ')
1191 end
1272 end
1192
1273
1193 # Returns a SQL clause for a date or datetime field using relative dates.
1274 # Returns a SQL clause for a date or datetime field using relative dates.
1194 def relative_date_clause(table, field, days_from, days_to, is_custom_filter)
1275 def relative_date_clause(table, field, days_from, days_to, is_custom_filter)
1195 date_clause(table, field, (days_from ? User.current.today + days_from : nil), (days_to ? User.current.today + days_to : nil), is_custom_filter)
1276 date_clause(table, field, (days_from ? User.current.today + days_from : nil), (days_to ? User.current.today + days_to : nil), is_custom_filter)
1196 end
1277 end
1197
1278
1198 # Returns a Date or Time from the given filter value
1279 # Returns a Date or Time from the given filter value
1199 def parse_date(arg)
1280 def parse_date(arg)
1200 if arg.to_s =~ /\A\d{4}-\d{2}-\d{2}T/
1281 if arg.to_s =~ /\A\d{4}-\d{2}-\d{2}T/
1201 Time.parse(arg) rescue nil
1282 Time.parse(arg) rescue nil
1202 else
1283 else
1203 Date.parse(arg) rescue nil
1284 Date.parse(arg) rescue nil
1204 end
1285 end
1205 end
1286 end
1206
1287
1207 # Additional joins required for the given sort options
1288 # Additional joins required for the given sort options
1208 def joins_for_order_statement(order_options)
1289 def joins_for_order_statement(order_options)
1209 joins = []
1290 joins = []
1210
1291
1211 if order_options
1292 if order_options
1212 if order_options.include?('authors')
1293 if order_options.include?('authors')
1213 joins << "LEFT OUTER JOIN #{User.table_name} authors ON authors.id = #{queried_table_name}.author_id"
1294 joins << "LEFT OUTER JOIN #{User.table_name} authors ON authors.id = #{queried_table_name}.author_id"
1214 end
1295 end
1215 order_options.scan(/cf_\d+/).uniq.each do |name|
1296 order_options.scan(/cf_\d+/).uniq.each do |name|
1216 column = available_columns.detect {|c| c.name.to_s == name}
1297 column = available_columns.detect {|c| c.name.to_s == name}
1217 join = column && column.custom_field.join_for_order_statement
1298 join = column && column.custom_field.join_for_order_statement
1218 if join
1299 if join
1219 joins << join
1300 joins << join
1220 end
1301 end
1221 end
1302 end
1222 end
1303 end
1223
1304
1224 joins.any? ? joins.join(' ') : nil
1305 joins.any? ? joins.join(' ') : nil
1225 end
1306 end
1226 end
1307 end
@@ -1,1207 +1,1208
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: "cannot be empty"
113 empty: "cannot be empty"
114 blank: "cannot be blank"
114 blank: "cannot 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 earlier_than_minimum_start_date: "cannot be earlier than %{date} because of preceding issues"
132 earlier_than_minimum_start_date: "cannot be earlier than %{date} because of preceding issues"
133 not_a_regexp: "is not a valid regular expression"
133 not_a_regexp: "is not a valid regular expression"
134 open_issue_with_closed_parent: "An open issue cannot be attached to a closed parent task"
134 open_issue_with_closed_parent: "An open issue cannot be attached to a closed parent task"
135
135
136 actionview_instancetag_blank_option: Please select
136 actionview_instancetag_blank_option: Please select
137
137
138 general_text_No: 'No'
138 general_text_No: 'No'
139 general_text_Yes: 'Yes'
139 general_text_Yes: 'Yes'
140 general_text_no: 'no'
140 general_text_no: 'no'
141 general_text_yes: 'yes'
141 general_text_yes: 'yes'
142 general_lang_name: 'English'
142 general_lang_name: 'English'
143 general_csv_separator: ','
143 general_csv_separator: ','
144 general_csv_decimal_separator: '.'
144 general_csv_decimal_separator: '.'
145 general_csv_encoding: ISO-8859-1
145 general_csv_encoding: ISO-8859-1
146 general_pdf_fontname: freesans
146 general_pdf_fontname: freesans
147 general_pdf_monospaced_fontname: freemono
147 general_pdf_monospaced_fontname: freemono
148 general_first_day_of_week: '7'
148 general_first_day_of_week: '7'
149
149
150 notice_account_updated: Account was successfully updated.
150 notice_account_updated: Account was successfully updated.
151 notice_account_invalid_credentials: Invalid user or password
151 notice_account_invalid_credentials: Invalid user or password
152 notice_account_password_updated: Password was successfully updated.
152 notice_account_password_updated: Password was successfully updated.
153 notice_account_wrong_password: Wrong password
153 notice_account_wrong_password: Wrong password
154 notice_account_register_done: Account was successfully created. An email containing the instructions to activate your account was sent to %{email}.
154 notice_account_register_done: Account was successfully created. An email containing the instructions to activate your account was sent to %{email}.
155 notice_account_unknown_email: Unknown user.
155 notice_account_unknown_email: Unknown user.
156 notice_account_not_activated_yet: You haven't activated your account yet. If you want to receive a new activation email, please <a href="%{url}">click this link</a>.
156 notice_account_not_activated_yet: You haven't activated your account yet. If you want to receive a new activation email, please <a href="%{url}">click this link</a>.
157 notice_account_locked: Your account is locked.
157 notice_account_locked: Your account is locked.
158 notice_can_t_change_password: This account uses an external authentication source. Impossible to change the password.
158 notice_can_t_change_password: This account uses an external authentication source. Impossible to change the password.
159 notice_account_lost_email_sent: An email with instructions to choose a new password has been sent to you.
159 notice_account_lost_email_sent: An email with instructions to choose a new password has been sent to you.
160 notice_account_activated: Your account has been activated. You can now log in.
160 notice_account_activated: Your account has been activated. You can now log in.
161 notice_successful_create: Successful creation.
161 notice_successful_create: Successful creation.
162 notice_successful_update: Successful update.
162 notice_successful_update: Successful update.
163 notice_successful_delete: Successful deletion.
163 notice_successful_delete: Successful deletion.
164 notice_successful_connection: Successful connection.
164 notice_successful_connection: Successful connection.
165 notice_file_not_found: The page you were trying to access doesn't exist or has been removed.
165 notice_file_not_found: The page you were trying to access doesn't exist or has been removed.
166 notice_locking_conflict: Data has been updated by another user.
166 notice_locking_conflict: Data has been updated by another user.
167 notice_not_authorized: You are not authorized to access this page.
167 notice_not_authorized: You are not authorized to access this page.
168 notice_not_authorized_archived_project: The project you're trying to access has been archived.
168 notice_not_authorized_archived_project: The project you're trying to access has been archived.
169 notice_email_sent: "An email was sent to %{value}"
169 notice_email_sent: "An email was sent to %{value}"
170 notice_email_error: "An error occurred while sending mail (%{value})"
170 notice_email_error: "An error occurred while sending mail (%{value})"
171 notice_feeds_access_key_reseted: Your Atom access key was reset.
171 notice_feeds_access_key_reseted: Your Atom access key was reset.
172 notice_api_access_key_reseted: Your API access key was reset.
172 notice_api_access_key_reseted: Your API access key was reset.
173 notice_failed_to_save_issues: "Failed to save %{count} issue(s) on %{total} selected: %{ids}."
173 notice_failed_to_save_issues: "Failed to save %{count} issue(s) on %{total} selected: %{ids}."
174 notice_failed_to_save_time_entries: "Failed to save %{count} time entrie(s) on %{total} selected: %{ids}."
174 notice_failed_to_save_time_entries: "Failed to save %{count} time entrie(s) on %{total} selected: %{ids}."
175 notice_failed_to_save_members: "Failed to save member(s): %{errors}."
175 notice_failed_to_save_members: "Failed to save member(s): %{errors}."
176 notice_no_issue_selected: "No issue is selected! Please, check the issues you want to edit."
176 notice_no_issue_selected: "No issue is selected! Please, check the issues you want to edit."
177 notice_account_pending: "Your account was created and is now pending administrator approval."
177 notice_account_pending: "Your account was created and is now pending administrator approval."
178 notice_default_data_loaded: Default configuration successfully loaded.
178 notice_default_data_loaded: Default configuration successfully loaded.
179 notice_unable_delete_version: Unable to delete version.
179 notice_unable_delete_version: Unable to delete version.
180 notice_unable_delete_time_entry: Unable to delete time log entry.
180 notice_unable_delete_time_entry: Unable to delete time log entry.
181 notice_issue_done_ratios_updated: Issue done ratios updated.
181 notice_issue_done_ratios_updated: Issue done ratios updated.
182 notice_gantt_chart_truncated: "The chart was truncated because it exceeds the maximum number of items that can be displayed (%{max})"
182 notice_gantt_chart_truncated: "The chart was truncated because it exceeds the maximum number of items that can be displayed (%{max})"
183 notice_issue_successful_create: "Issue %{id} created."
183 notice_issue_successful_create: "Issue %{id} created."
184 notice_issue_update_conflict: "The issue has been updated by an other user while you were editing it."
184 notice_issue_update_conflict: "The issue has been updated by an other user while you were editing it."
185 notice_account_deleted: "Your account has been permanently deleted."
185 notice_account_deleted: "Your account has been permanently deleted."
186 notice_user_successful_create: "User %{id} created."
186 notice_user_successful_create: "User %{id} created."
187 notice_new_password_must_be_different: The new password must be different from the current password
187 notice_new_password_must_be_different: The new password must be different from the current password
188 notice_import_finished: "%{count} items have been imported"
188 notice_import_finished: "%{count} items have been imported"
189 notice_import_finished_with_errors: "%{count} out of %{total} items could not be imported"
189 notice_import_finished_with_errors: "%{count} out of %{total} items could not be imported"
190
190
191 error_can_t_load_default_data: "Default configuration could not be loaded: %{value}"
191 error_can_t_load_default_data: "Default configuration could not be loaded: %{value}"
192 error_scm_not_found: "The entry or revision was not found in the repository."
192 error_scm_not_found: "The entry or revision was not found in the repository."
193 error_scm_command_failed: "An error occurred when trying to access the repository: %{value}"
193 error_scm_command_failed: "An error occurred when trying to access the repository: %{value}"
194 error_scm_annotate: "The entry does not exist or cannot be annotated."
194 error_scm_annotate: "The entry does not exist or cannot be annotated."
195 error_scm_annotate_big_text_file: "The entry cannot be annotated, as it exceeds the maximum text file size."
195 error_scm_annotate_big_text_file: "The entry cannot be annotated, as it exceeds the maximum text file size."
196 error_issue_not_found_in_project: 'The issue was not found or does not belong to this project'
196 error_issue_not_found_in_project: 'The issue was not found or does not belong to this project'
197 error_no_tracker_in_project: 'No tracker is associated to this project. Please check the Project settings.'
197 error_no_tracker_in_project: 'No tracker is associated to this project. Please check the Project settings.'
198 error_no_default_issue_status: 'No default issue status is defined. Please check your configuration (Go to "Administration -> Issue statuses").'
198 error_no_default_issue_status: 'No default issue status is defined. Please check your configuration (Go to "Administration -> Issue statuses").'
199 error_can_not_delete_custom_field: Unable to delete custom field
199 error_can_not_delete_custom_field: Unable to delete custom field
200 error_can_not_delete_tracker: "This tracker contains issues and cannot be deleted."
200 error_can_not_delete_tracker: "This tracker contains issues and cannot be deleted."
201 error_can_not_remove_role: "This role is in use and cannot be deleted."
201 error_can_not_remove_role: "This role is in use and cannot be deleted."
202 error_can_not_reopen_issue_on_closed_version: 'An issue assigned to a closed version cannot be reopened'
202 error_can_not_reopen_issue_on_closed_version: 'An issue assigned to a closed version cannot be reopened'
203 error_can_not_archive_project: This project cannot be archived
203 error_can_not_archive_project: This project cannot be archived
204 error_issue_done_ratios_not_updated: "Issue done ratios not updated."
204 error_issue_done_ratios_not_updated: "Issue done ratios not updated."
205 error_workflow_copy_source: 'Please select a source tracker or role'
205 error_workflow_copy_source: 'Please select a source tracker or role'
206 error_workflow_copy_target: 'Please select target tracker(s) and role(s)'
206 error_workflow_copy_target: 'Please select target tracker(s) and role(s)'
207 error_unable_delete_issue_status: 'Unable to delete issue status'
207 error_unable_delete_issue_status: 'Unable to delete issue status'
208 error_unable_to_connect: "Unable to connect (%{value})"
208 error_unable_to_connect: "Unable to connect (%{value})"
209 error_attachment_too_big: "This file cannot be uploaded because it exceeds the maximum allowed file size (%{max_size})"
209 error_attachment_too_big: "This file cannot be uploaded because it exceeds the maximum allowed file size (%{max_size})"
210 error_session_expired: "Your session has expired. Please login again."
210 error_session_expired: "Your session has expired. Please login again."
211 warning_attachments_not_saved: "%{count} file(s) could not be saved."
211 warning_attachments_not_saved: "%{count} file(s) could not be saved."
212 error_password_expired: "Your password has expired or the administrator requires you to change it."
212 error_password_expired: "Your password has expired or the administrator requires you to change it."
213 error_invalid_file_encoding: "The file is not a valid %{encoding} encoded file"
213 error_invalid_file_encoding: "The file is not a valid %{encoding} encoded file"
214 error_invalid_csv_file_or_settings: "The file is not a CSV file or does not match the settings below"
214 error_invalid_csv_file_or_settings: "The file is not a CSV file or does not match the settings below"
215 error_can_not_read_import_file: "An error occurred while reading the file to import"
215 error_can_not_read_import_file: "An error occurred while reading the file to import"
216 error_attachment_extension_not_allowed: "Attachment extension %{extension} is not allowed"
216 error_attachment_extension_not_allowed: "Attachment extension %{extension} is not allowed"
217 error_ldap_bind_credentials: "Invalid LDAP Account/Password"
217 error_ldap_bind_credentials: "Invalid LDAP Account/Password"
218 error_no_tracker_allowed_for_new_issue_in_project: "The project doesn't have any trackers for which you can create an issue"
218 error_no_tracker_allowed_for_new_issue_in_project: "The project doesn't have any trackers for which you can create an issue"
219 error_no_projects_with_tracker_allowed_for_new_issue: "There are no projects with trackers for which you can create an issue"
219 error_no_projects_with_tracker_allowed_for_new_issue: "There are no projects with trackers for which you can create an issue"
220 error_move_of_child_not_possible: "Subtask %{child} could not be moved to the new project: %{errors}"
220 error_move_of_child_not_possible: "Subtask %{child} could not be moved to the new project: %{errors}"
221 error_cannot_reassign_time_entries_to_an_issue_about_to_be_deleted: "Spent time cannot be reassigned to an issue that is about to be deleted"
221 error_cannot_reassign_time_entries_to_an_issue_about_to_be_deleted: "Spent time cannot be reassigned to an issue that is about to be deleted"
222
222
223 mail_subject_lost_password: "Your %{value} password"
223 mail_subject_lost_password: "Your %{value} password"
224 mail_body_lost_password: 'To change your password, click on the following link:'
224 mail_body_lost_password: 'To change your password, click on the following link:'
225 mail_subject_register: "Your %{value} account activation"
225 mail_subject_register: "Your %{value} account activation"
226 mail_body_register: 'To activate your account, click on the following link:'
226 mail_body_register: 'To activate your account, click on the following link:'
227 mail_body_account_information_external: "You can use your %{value} account to log in."
227 mail_body_account_information_external: "You can use your %{value} account to log in."
228 mail_body_account_information: Your account information
228 mail_body_account_information: Your account information
229 mail_subject_account_activation_request: "%{value} account activation request"
229 mail_subject_account_activation_request: "%{value} account activation request"
230 mail_body_account_activation_request: "A new user (%{value}) has registered. The account is pending your approval:"
230 mail_body_account_activation_request: "A new user (%{value}) has registered. The account is pending your approval:"
231 mail_subject_reminder: "%{count} issue(s) due in the next %{days} days"
231 mail_subject_reminder: "%{count} issue(s) due in the next %{days} days"
232 mail_body_reminder: "%{count} issue(s) that are assigned to you are due in the next %{days} days:"
232 mail_body_reminder: "%{count} issue(s) that are assigned to you are due in the next %{days} days:"
233 mail_subject_wiki_content_added: "'%{id}' wiki page has been added"
233 mail_subject_wiki_content_added: "'%{id}' wiki page has been added"
234 mail_body_wiki_content_added: "The '%{id}' wiki page has been added by %{author}."
234 mail_body_wiki_content_added: "The '%{id}' wiki page has been added by %{author}."
235 mail_subject_wiki_content_updated: "'%{id}' wiki page has been updated"
235 mail_subject_wiki_content_updated: "'%{id}' wiki page has been updated"
236 mail_body_wiki_content_updated: "The '%{id}' wiki page has been updated by %{author}."
236 mail_body_wiki_content_updated: "The '%{id}' wiki page has been updated by %{author}."
237 mail_subject_security_notification: "Security notification"
237 mail_subject_security_notification: "Security notification"
238 mail_body_security_notification_change: "%{field} was changed."
238 mail_body_security_notification_change: "%{field} was changed."
239 mail_body_security_notification_change_to: "%{field} was changed to %{value}."
239 mail_body_security_notification_change_to: "%{field} was changed to %{value}."
240 mail_body_security_notification_add: "%{field} %{value} was added."
240 mail_body_security_notification_add: "%{field} %{value} was added."
241 mail_body_security_notification_remove: "%{field} %{value} was removed."
241 mail_body_security_notification_remove: "%{field} %{value} was removed."
242 mail_body_security_notification_notify_enabled: "Email address %{value} now receives notifications."
242 mail_body_security_notification_notify_enabled: "Email address %{value} now receives notifications."
243 mail_body_security_notification_notify_disabled: "Email address %{value} no longer receives notifications."
243 mail_body_security_notification_notify_disabled: "Email address %{value} no longer receives notifications."
244 mail_body_settings_updated: "The following settings were changed:"
244 mail_body_settings_updated: "The following settings were changed:"
245 mail_body_password_updated: "Your password has been changed."
245 mail_body_password_updated: "Your password has been changed."
246
246
247 field_name: Name
247 field_name: Name
248 field_description: Description
248 field_description: Description
249 field_summary: Summary
249 field_summary: Summary
250 field_is_required: Required
250 field_is_required: Required
251 field_firstname: First name
251 field_firstname: First name
252 field_lastname: Last name
252 field_lastname: Last name
253 field_mail: Email
253 field_mail: Email
254 field_address: Email
254 field_address: Email
255 field_filename: File
255 field_filename: File
256 field_filesize: Size
256 field_filesize: Size
257 field_downloads: Downloads
257 field_downloads: Downloads
258 field_author: Author
258 field_author: Author
259 field_created_on: Created
259 field_created_on: Created
260 field_updated_on: Updated
260 field_updated_on: Updated
261 field_closed_on: Closed
261 field_closed_on: Closed
262 field_field_format: Format
262 field_field_format: Format
263 field_is_for_all: For all projects
263 field_is_for_all: For all projects
264 field_possible_values: Possible values
264 field_possible_values: Possible values
265 field_regexp: Regular expression
265 field_regexp: Regular expression
266 field_min_length: Minimum length
266 field_min_length: Minimum length
267 field_max_length: Maximum length
267 field_max_length: Maximum length
268 field_value: Value
268 field_value: Value
269 field_category: Category
269 field_category: Category
270 field_title: Title
270 field_title: Title
271 field_project: Project
271 field_project: Project
272 field_issue: Issue
272 field_issue: Issue
273 field_status: Status
273 field_status: Status
274 field_notes: Notes
274 field_notes: Notes
275 field_is_closed: Issue closed
275 field_is_closed: Issue closed
276 field_is_default: Default value
276 field_is_default: Default value
277 field_tracker: Tracker
277 field_tracker: Tracker
278 field_subject: Subject
278 field_subject: Subject
279 field_due_date: Due date
279 field_due_date: Due date
280 field_assigned_to: Assignee
280 field_assigned_to: Assignee
281 field_priority: Priority
281 field_priority: Priority
282 field_fixed_version: Target version
282 field_fixed_version: Target version
283 field_user: User
283 field_user: User
284 field_principal: Principal
284 field_principal: Principal
285 field_role: Role
285 field_role: Role
286 field_homepage: Homepage
286 field_homepage: Homepage
287 field_is_public: Public
287 field_is_public: Public
288 field_parent: Subproject of
288 field_parent: Subproject of
289 field_is_in_roadmap: Issues displayed in roadmap
289 field_is_in_roadmap: Issues displayed in roadmap
290 field_login: Login
290 field_login: Login
291 field_mail_notification: Email notifications
291 field_mail_notification: Email notifications
292 field_admin: Administrator
292 field_admin: Administrator
293 field_last_login_on: Last connection
293 field_last_login_on: Last connection
294 field_language: Language
294 field_language: Language
295 field_effective_date: Due date
295 field_effective_date: Due date
296 field_password: Password
296 field_password: Password
297 field_new_password: New password
297 field_new_password: New password
298 field_password_confirmation: Confirmation
298 field_password_confirmation: Confirmation
299 field_version: Version
299 field_version: Version
300 field_type: Type
300 field_type: Type
301 field_host: Host
301 field_host: Host
302 field_port: Port
302 field_port: Port
303 field_account: Account
303 field_account: Account
304 field_base_dn: Base DN
304 field_base_dn: Base DN
305 field_attr_login: Login attribute
305 field_attr_login: Login attribute
306 field_attr_firstname: Firstname attribute
306 field_attr_firstname: Firstname attribute
307 field_attr_lastname: Lastname attribute
307 field_attr_lastname: Lastname attribute
308 field_attr_mail: Email attribute
308 field_attr_mail: Email attribute
309 field_onthefly: On-the-fly user creation
309 field_onthefly: On-the-fly user creation
310 field_start_date: Start date
310 field_start_date: Start date
311 field_done_ratio: "% Done"
311 field_done_ratio: "% Done"
312 field_auth_source: Authentication mode
312 field_auth_source: Authentication mode
313 field_hide_mail: Hide my email address
313 field_hide_mail: Hide my email address
314 field_comments: Comment
314 field_comments: Comment
315 field_url: URL
315 field_url: URL
316 field_start_page: Start page
316 field_start_page: Start page
317 field_subproject: Subproject
317 field_subproject: Subproject
318 field_hours: Hours
318 field_hours: Hours
319 field_activity: Activity
319 field_activity: Activity
320 field_spent_on: Date
320 field_spent_on: Date
321 field_identifier: Identifier
321 field_identifier: Identifier
322 field_is_filter: Used as a filter
322 field_is_filter: Used as a filter
323 field_issue_to: Related issue
323 field_issue_to: Related issue
324 field_delay: Delay
324 field_delay: Delay
325 field_assignable: Issues can be assigned to this role
325 field_assignable: Issues can be assigned to this role
326 field_redirect_existing_links: Redirect existing links
326 field_redirect_existing_links: Redirect existing links
327 field_estimated_hours: Estimated time
327 field_estimated_hours: Estimated time
328 field_column_names: Columns
328 field_column_names: Columns
329 field_time_entries: Log time
329 field_time_entries: Log time
330 field_time_zone: Time zone
330 field_time_zone: Time zone
331 field_searchable: Searchable
331 field_searchable: Searchable
332 field_default_value: Default value
332 field_default_value: Default value
333 field_comments_sorting: Display comments
333 field_comments_sorting: Display comments
334 field_parent_title: Parent page
334 field_parent_title: Parent page
335 field_editable: Editable
335 field_editable: Editable
336 field_watcher: Watcher
336 field_watcher: Watcher
337 field_identity_url: OpenID URL
337 field_identity_url: OpenID URL
338 field_content: Content
338 field_content: Content
339 field_group_by: Group results by
339 field_group_by: Group results by
340 field_sharing: Sharing
340 field_sharing: Sharing
341 field_parent_issue: Parent task
341 field_parent_issue: Parent task
342 field_member_of_group: "Assignee's group"
342 field_member_of_group: "Assignee's group"
343 field_assigned_to_role: "Assignee's role"
343 field_assigned_to_role: "Assignee's role"
344 field_text: Text field
344 field_text: Text field
345 field_visible: Visible
345 field_visible: Visible
346 field_warn_on_leaving_unsaved: "Warn me when leaving a page with unsaved text"
346 field_warn_on_leaving_unsaved: "Warn me when leaving a page with unsaved text"
347 field_issues_visibility: Issues visibility
347 field_issues_visibility: Issues visibility
348 field_is_private: Private
348 field_is_private: Private
349 field_commit_logs_encoding: Commit messages encoding
349 field_commit_logs_encoding: Commit messages encoding
350 field_scm_path_encoding: Path encoding
350 field_scm_path_encoding: Path encoding
351 field_path_to_repository: Path to repository
351 field_path_to_repository: Path to repository
352 field_root_directory: Root directory
352 field_root_directory: Root directory
353 field_cvsroot: CVSROOT
353 field_cvsroot: CVSROOT
354 field_cvs_module: Module
354 field_cvs_module: Module
355 field_repository_is_default: Main repository
355 field_repository_is_default: Main repository
356 field_multiple: Multiple values
356 field_multiple: Multiple values
357 field_auth_source_ldap_filter: LDAP filter
357 field_auth_source_ldap_filter: LDAP filter
358 field_core_fields: Standard fields
358 field_core_fields: Standard fields
359 field_timeout: "Timeout (in seconds)"
359 field_timeout: "Timeout (in seconds)"
360 field_board_parent: Parent forum
360 field_board_parent: Parent forum
361 field_private_notes: Private notes
361 field_private_notes: Private notes
362 field_inherit_members: Inherit members
362 field_inherit_members: Inherit members
363 field_generate_password: Generate password
363 field_generate_password: Generate password
364 field_must_change_passwd: Must change password at next logon
364 field_must_change_passwd: Must change password at next logon
365 field_default_status: Default status
365 field_default_status: Default status
366 field_users_visibility: Users visibility
366 field_users_visibility: Users visibility
367 field_time_entries_visibility: Time logs visibility
367 field_time_entries_visibility: Time logs visibility
368 field_total_estimated_hours: Total estimated time
368 field_total_estimated_hours: Total estimated time
369 field_default_version: Default version
369 field_default_version: Default version
370 field_remote_ip: IP address
370 field_remote_ip: IP address
371 field_textarea_font: Font used for text areas
371 field_textarea_font: Font used for text areas
372
372
373 setting_app_title: Application title
373 setting_app_title: Application title
374 setting_app_subtitle: Application subtitle
374 setting_app_subtitle: Application subtitle
375 setting_welcome_text: Welcome text
375 setting_welcome_text: Welcome text
376 setting_default_language: Default language
376 setting_default_language: Default language
377 setting_login_required: Authentication required
377 setting_login_required: Authentication required
378 setting_self_registration: Self-registration
378 setting_self_registration: Self-registration
379 setting_attachment_max_size: Maximum attachment size
379 setting_attachment_max_size: Maximum attachment size
380 setting_issues_export_limit: Issues export limit
380 setting_issues_export_limit: Issues export limit
381 setting_mail_from: Emission email address
381 setting_mail_from: Emission email address
382 setting_bcc_recipients: Blind carbon copy recipients (bcc)
382 setting_bcc_recipients: Blind carbon copy recipients (bcc)
383 setting_plain_text_mail: Plain text mail (no HTML)
383 setting_plain_text_mail: Plain text mail (no HTML)
384 setting_host_name: Host name and path
384 setting_host_name: Host name and path
385 setting_text_formatting: Text formatting
385 setting_text_formatting: Text formatting
386 setting_wiki_compression: Wiki history compression
386 setting_wiki_compression: Wiki history compression
387 setting_feeds_limit: Maximum number of items in Atom feeds
387 setting_feeds_limit: Maximum number of items in Atom feeds
388 setting_default_projects_public: New projects are public by default
388 setting_default_projects_public: New projects are public by default
389 setting_autofetch_changesets: Fetch commits automatically
389 setting_autofetch_changesets: Fetch commits automatically
390 setting_sys_api_enabled: Enable WS for repository management
390 setting_sys_api_enabled: Enable WS for repository management
391 setting_commit_ref_keywords: Referencing keywords
391 setting_commit_ref_keywords: Referencing keywords
392 setting_commit_fix_keywords: Fixing keywords
392 setting_commit_fix_keywords: Fixing keywords
393 setting_autologin: Autologin
393 setting_autologin: Autologin
394 setting_date_format: Date format
394 setting_date_format: Date format
395 setting_time_format: Time format
395 setting_time_format: Time format
396 setting_timespan_format: Time span format
396 setting_timespan_format: Time span format
397 setting_cross_project_issue_relations: Allow cross-project issue relations
397 setting_cross_project_issue_relations: Allow cross-project issue relations
398 setting_cross_project_subtasks: Allow cross-project subtasks
398 setting_cross_project_subtasks: Allow cross-project subtasks
399 setting_issue_list_default_columns: Default columns displayed on the issue list
399 setting_issue_list_default_columns: Default columns displayed on the issue list
400 setting_repositories_encodings: Attachments and repositories encodings
400 setting_repositories_encodings: Attachments and repositories encodings
401 setting_emails_header: Email header
401 setting_emails_header: Email header
402 setting_emails_footer: Email footer
402 setting_emails_footer: Email footer
403 setting_protocol: Protocol
403 setting_protocol: Protocol
404 setting_per_page_options: Objects per page options
404 setting_per_page_options: Objects per page options
405 setting_user_format: Users display format
405 setting_user_format: Users display format
406 setting_activity_days_default: Days displayed on project activity
406 setting_activity_days_default: Days displayed on project activity
407 setting_display_subprojects_issues: Display subprojects issues on main projects by default
407 setting_display_subprojects_issues: Display subprojects issues on main projects by default
408 setting_enabled_scm: Enabled SCM
408 setting_enabled_scm: Enabled SCM
409 setting_mail_handler_body_delimiters: "Truncate emails after one of these lines"
409 setting_mail_handler_body_delimiters: "Truncate emails after one of these lines"
410 setting_mail_handler_enable_regex_delimiters: "Enable regular expressions"
410 setting_mail_handler_enable_regex_delimiters: "Enable regular expressions"
411 setting_mail_handler_api_enabled: Enable WS for incoming emails
411 setting_mail_handler_api_enabled: Enable WS for incoming emails
412 setting_mail_handler_api_key: Incoming email WS API key
412 setting_mail_handler_api_key: Incoming email WS API key
413 setting_sys_api_key: Repository management WS API key
413 setting_sys_api_key: Repository management WS API key
414 setting_sequential_project_identifiers: Generate sequential project identifiers
414 setting_sequential_project_identifiers: Generate sequential project identifiers
415 setting_gravatar_enabled: Use Gravatar user icons
415 setting_gravatar_enabled: Use Gravatar user icons
416 setting_gravatar_default: Default Gravatar image
416 setting_gravatar_default: Default Gravatar image
417 setting_diff_max_lines_displayed: Maximum number of diff lines displayed
417 setting_diff_max_lines_displayed: Maximum number of diff lines displayed
418 setting_file_max_size_displayed: Maximum size of text files displayed inline
418 setting_file_max_size_displayed: Maximum size of text files displayed inline
419 setting_repository_log_display_limit: Maximum number of revisions displayed on file log
419 setting_repository_log_display_limit: Maximum number of revisions displayed on file log
420 setting_openid: Allow OpenID login and registration
420 setting_openid: Allow OpenID login and registration
421 setting_password_max_age: Require password change after
421 setting_password_max_age: Require password change after
422 setting_password_min_length: Minimum password length
422 setting_password_min_length: Minimum password length
423 setting_lost_password: Allow password reset via email
423 setting_lost_password: Allow password reset via email
424 setting_new_project_user_role_id: Role given to a non-admin user who creates a project
424 setting_new_project_user_role_id: Role given to a non-admin user who creates a project
425 setting_default_projects_modules: Default enabled modules for new projects
425 setting_default_projects_modules: Default enabled modules for new projects
426 setting_issue_done_ratio: Calculate the issue done ratio with
426 setting_issue_done_ratio: Calculate the issue done ratio with
427 setting_issue_done_ratio_issue_field: Use the issue field
427 setting_issue_done_ratio_issue_field: Use the issue field
428 setting_issue_done_ratio_issue_status: Use the issue status
428 setting_issue_done_ratio_issue_status: Use the issue status
429 setting_start_of_week: Start calendars on
429 setting_start_of_week: Start calendars on
430 setting_rest_api_enabled: Enable REST web service
430 setting_rest_api_enabled: Enable REST web service
431 setting_cache_formatted_text: Cache formatted text
431 setting_cache_formatted_text: Cache formatted text
432 setting_default_notification_option: Default notification option
432 setting_default_notification_option: Default notification option
433 setting_commit_logtime_enabled: Enable time logging
433 setting_commit_logtime_enabled: Enable time logging
434 setting_commit_logtime_activity_id: Activity for logged time
434 setting_commit_logtime_activity_id: Activity for logged time
435 setting_gantt_items_limit: Maximum number of items displayed on the gantt chart
435 setting_gantt_items_limit: Maximum number of items displayed on the gantt chart
436 setting_issue_group_assignment: Allow issue assignment to groups
436 setting_issue_group_assignment: Allow issue assignment to groups
437 setting_default_issue_start_date_to_creation_date: Use current date as start date for new issues
437 setting_default_issue_start_date_to_creation_date: Use current date as start date for new issues
438 setting_commit_cross_project_ref: Allow issues of all the other projects to be referenced and fixed
438 setting_commit_cross_project_ref: Allow issues of all the other projects to be referenced and fixed
439 setting_unsubscribe: Allow users to delete their own account
439 setting_unsubscribe: Allow users to delete their own account
440 setting_session_lifetime: Session maximum lifetime
440 setting_session_lifetime: Session maximum lifetime
441 setting_session_timeout: Session inactivity timeout
441 setting_session_timeout: Session inactivity timeout
442 setting_thumbnails_enabled: Display attachment thumbnails
442 setting_thumbnails_enabled: Display attachment thumbnails
443 setting_thumbnails_size: Thumbnails size (in pixels)
443 setting_thumbnails_size: Thumbnails size (in pixels)
444 setting_non_working_week_days: Non-working days
444 setting_non_working_week_days: Non-working days
445 setting_jsonp_enabled: Enable JSONP support
445 setting_jsonp_enabled: Enable JSONP support
446 setting_default_projects_tracker_ids: Default trackers for new projects
446 setting_default_projects_tracker_ids: Default trackers for new projects
447 setting_mail_handler_excluded_filenames: Exclude attachments by name
447 setting_mail_handler_excluded_filenames: Exclude attachments by name
448 setting_force_default_language_for_anonymous: Force default language for anonymous users
448 setting_force_default_language_for_anonymous: Force default language for anonymous users
449 setting_force_default_language_for_loggedin: Force default language for logged-in users
449 setting_force_default_language_for_loggedin: Force default language for logged-in users
450 setting_link_copied_issue: Link issues on copy
450 setting_link_copied_issue: Link issues on copy
451 setting_max_additional_emails: Maximum number of additional email addresses
451 setting_max_additional_emails: Maximum number of additional email addresses
452 setting_search_results_per_page: Search results per page
452 setting_search_results_per_page: Search results per page
453 setting_attachment_extensions_allowed: Allowed extensions
453 setting_attachment_extensions_allowed: Allowed extensions
454 setting_attachment_extensions_denied: Disallowed extensions
454 setting_attachment_extensions_denied: Disallowed extensions
455 setting_new_item_menu_tab: Project menu tab for creating new objects
455 setting_new_item_menu_tab: Project menu tab for creating new objects
456 setting_commit_logs_formatting: Apply text formatting to commit messages
456 setting_commit_logs_formatting: Apply text formatting to commit messages
457 setting_timelog_required_fields: Required fields for time logs
457 setting_timelog_required_fields: Required fields for time logs
458
458
459 permission_add_project: Create project
459 permission_add_project: Create project
460 permission_add_subprojects: Create subprojects
460 permission_add_subprojects: Create subprojects
461 permission_edit_project: Edit project
461 permission_edit_project: Edit project
462 permission_close_project: Close / reopen the project
462 permission_close_project: Close / reopen the project
463 permission_select_project_modules: Select project modules
463 permission_select_project_modules: Select project modules
464 permission_manage_members: Manage members
464 permission_manage_members: Manage members
465 permission_manage_project_activities: Manage project activities
465 permission_manage_project_activities: Manage project activities
466 permission_manage_versions: Manage versions
466 permission_manage_versions: Manage versions
467 permission_manage_categories: Manage issue categories
467 permission_manage_categories: Manage issue categories
468 permission_view_issues: View Issues
468 permission_view_issues: View Issues
469 permission_add_issues: Add issues
469 permission_add_issues: Add issues
470 permission_edit_issues: Edit issues
470 permission_edit_issues: Edit issues
471 permission_copy_issues: Copy issues
471 permission_copy_issues: Copy issues
472 permission_manage_issue_relations: Manage issue relations
472 permission_manage_issue_relations: Manage issue relations
473 permission_set_issues_private: Set issues public or private
473 permission_set_issues_private: Set issues public or private
474 permission_set_own_issues_private: Set own issues public or private
474 permission_set_own_issues_private: Set own issues public or private
475 permission_add_issue_notes: Add notes
475 permission_add_issue_notes: Add notes
476 permission_edit_issue_notes: Edit notes
476 permission_edit_issue_notes: Edit notes
477 permission_edit_own_issue_notes: Edit own notes
477 permission_edit_own_issue_notes: Edit own notes
478 permission_view_private_notes: View private notes
478 permission_view_private_notes: View private notes
479 permission_set_notes_private: Set notes as private
479 permission_set_notes_private: Set notes as private
480 permission_move_issues: Move issues
480 permission_move_issues: Move issues
481 permission_delete_issues: Delete issues
481 permission_delete_issues: Delete issues
482 permission_manage_public_queries: Manage public queries
482 permission_manage_public_queries: Manage public queries
483 permission_save_queries: Save queries
483 permission_save_queries: Save queries
484 permission_view_gantt: View gantt chart
484 permission_view_gantt: View gantt chart
485 permission_view_calendar: View calendar
485 permission_view_calendar: View calendar
486 permission_view_issue_watchers: View watchers list
486 permission_view_issue_watchers: View watchers list
487 permission_add_issue_watchers: Add watchers
487 permission_add_issue_watchers: Add watchers
488 permission_delete_issue_watchers: Delete watchers
488 permission_delete_issue_watchers: Delete watchers
489 permission_log_time: Log spent time
489 permission_log_time: Log spent time
490 permission_view_time_entries: View spent time
490 permission_view_time_entries: View spent time
491 permission_edit_time_entries: Edit time logs
491 permission_edit_time_entries: Edit time logs
492 permission_edit_own_time_entries: Edit own time logs
492 permission_edit_own_time_entries: Edit own time logs
493 permission_manage_news: Manage news
493 permission_manage_news: Manage news
494 permission_comment_news: Comment news
494 permission_comment_news: Comment news
495 permission_view_documents: View documents
495 permission_view_documents: View documents
496 permission_add_documents: Add documents
496 permission_add_documents: Add documents
497 permission_edit_documents: Edit documents
497 permission_edit_documents: Edit documents
498 permission_delete_documents: Delete documents
498 permission_delete_documents: Delete documents
499 permission_manage_files: Manage files
499 permission_manage_files: Manage files
500 permission_view_files: View files
500 permission_view_files: View files
501 permission_manage_wiki: Manage wiki
501 permission_manage_wiki: Manage wiki
502 permission_rename_wiki_pages: Rename wiki pages
502 permission_rename_wiki_pages: Rename wiki pages
503 permission_delete_wiki_pages: Delete wiki pages
503 permission_delete_wiki_pages: Delete wiki pages
504 permission_view_wiki_pages: View wiki
504 permission_view_wiki_pages: View wiki
505 permission_view_wiki_edits: View wiki history
505 permission_view_wiki_edits: View wiki history
506 permission_edit_wiki_pages: Edit wiki pages
506 permission_edit_wiki_pages: Edit wiki pages
507 permission_delete_wiki_pages_attachments: Delete attachments
507 permission_delete_wiki_pages_attachments: Delete attachments
508 permission_protect_wiki_pages: Protect wiki pages
508 permission_protect_wiki_pages: Protect wiki pages
509 permission_manage_repository: Manage repository
509 permission_manage_repository: Manage repository
510 permission_browse_repository: Browse repository
510 permission_browse_repository: Browse repository
511 permission_view_changesets: View changesets
511 permission_view_changesets: View changesets
512 permission_commit_access: Commit access
512 permission_commit_access: Commit access
513 permission_manage_boards: Manage forums
513 permission_manage_boards: Manage forums
514 permission_view_messages: View messages
514 permission_view_messages: View messages
515 permission_add_messages: Post messages
515 permission_add_messages: Post messages
516 permission_edit_messages: Edit messages
516 permission_edit_messages: Edit messages
517 permission_edit_own_messages: Edit own messages
517 permission_edit_own_messages: Edit own messages
518 permission_delete_messages: Delete messages
518 permission_delete_messages: Delete messages
519 permission_delete_own_messages: Delete own messages
519 permission_delete_own_messages: Delete own messages
520 permission_export_wiki_pages: Export wiki pages
520 permission_export_wiki_pages: Export wiki pages
521 permission_manage_subtasks: Manage subtasks
521 permission_manage_subtasks: Manage subtasks
522 permission_manage_related_issues: Manage related issues
522 permission_manage_related_issues: Manage related issues
523 permission_import_issues: Import issues
523 permission_import_issues: Import issues
524
524
525 project_module_issue_tracking: Issue tracking
525 project_module_issue_tracking: Issue tracking
526 project_module_time_tracking: Time tracking
526 project_module_time_tracking: Time tracking
527 project_module_news: News
527 project_module_news: News
528 project_module_documents: Documents
528 project_module_documents: Documents
529 project_module_files: Files
529 project_module_files: Files
530 project_module_wiki: Wiki
530 project_module_wiki: Wiki
531 project_module_repository: Repository
531 project_module_repository: Repository
532 project_module_boards: Forums
532 project_module_boards: Forums
533 project_module_calendar: Calendar
533 project_module_calendar: Calendar
534 project_module_gantt: Gantt
534 project_module_gantt: Gantt
535
535
536 label_user: User
536 label_user: User
537 label_user_plural: Users
537 label_user_plural: Users
538 label_user_new: New user
538 label_user_new: New user
539 label_user_anonymous: Anonymous
539 label_user_anonymous: Anonymous
540 label_project: Project
540 label_project: Project
541 label_project_new: New project
541 label_project_new: New project
542 label_project_plural: Projects
542 label_project_plural: Projects
543 label_x_projects:
543 label_x_projects:
544 zero: no projects
544 zero: no projects
545 one: 1 project
545 one: 1 project
546 other: "%{count} projects"
546 other: "%{count} projects"
547 label_project_all: All Projects
547 label_project_all: All Projects
548 label_project_latest: Latest projects
548 label_project_latest: Latest projects
549 label_issue: Issue
549 label_issue: Issue
550 label_issue_new: New issue
550 label_issue_new: New issue
551 label_issue_plural: Issues
551 label_issue_plural: Issues
552 label_issue_view_all: View all issues
552 label_issue_view_all: View all issues
553 label_issues_by: "Issues by %{value}"
553 label_issues_by: "Issues by %{value}"
554 label_issue_added: Issue added
554 label_issue_added: Issue added
555 label_issue_updated: Issue updated
555 label_issue_updated: Issue updated
556 label_issue_note_added: Note added
556 label_issue_note_added: Note added
557 label_issue_status_updated: Status updated
557 label_issue_status_updated: Status updated
558 label_issue_assigned_to_updated: Assignee updated
558 label_issue_assigned_to_updated: Assignee updated
559 label_issue_priority_updated: Priority updated
559 label_issue_priority_updated: Priority updated
560 label_document: Document
560 label_document: Document
561 label_document_new: New document
561 label_document_new: New document
562 label_document_plural: Documents
562 label_document_plural: Documents
563 label_document_added: Document added
563 label_document_added: Document added
564 label_role: Role
564 label_role: Role
565 label_role_plural: Roles
565 label_role_plural: Roles
566 label_role_new: New role
566 label_role_new: New role
567 label_role_and_permissions: Roles and permissions
567 label_role_and_permissions: Roles and permissions
568 label_role_anonymous: Anonymous
568 label_role_anonymous: Anonymous
569 label_role_non_member: Non member
569 label_role_non_member: Non member
570 label_member: Member
570 label_member: Member
571 label_member_new: New member
571 label_member_new: New member
572 label_member_plural: Members
572 label_member_plural: Members
573 label_tracker: Tracker
573 label_tracker: Tracker
574 label_tracker_plural: Trackers
574 label_tracker_plural: Trackers
575 label_tracker_all: All trackers
575 label_tracker_all: All trackers
576 label_tracker_new: New tracker
576 label_tracker_new: New tracker
577 label_workflow: Workflow
577 label_workflow: Workflow
578 label_issue_status: Issue status
578 label_issue_status: Issue status
579 label_issue_status_plural: Issue statuses
579 label_issue_status_plural: Issue statuses
580 label_issue_status_new: New status
580 label_issue_status_new: New status
581 label_issue_category: Issue category
581 label_issue_category: Issue category
582 label_issue_category_plural: Issue categories
582 label_issue_category_plural: Issue categories
583 label_issue_category_new: New category
583 label_issue_category_new: New category
584 label_custom_field: Custom field
584 label_custom_field: Custom field
585 label_custom_field_plural: Custom fields
585 label_custom_field_plural: Custom fields
586 label_custom_field_new: New custom field
586 label_custom_field_new: New custom field
587 label_enumerations: Enumerations
587 label_enumerations: Enumerations
588 label_enumeration_new: New value
588 label_enumeration_new: New value
589 label_information: Information
589 label_information: Information
590 label_information_plural: Information
590 label_information_plural: Information
591 label_please_login: Please log in
591 label_please_login: Please log in
592 label_register: Register
592 label_register: Register
593 label_login_with_open_id_option: or login with OpenID
593 label_login_with_open_id_option: or login with OpenID
594 label_password_lost: Lost password
594 label_password_lost: Lost password
595 label_password_required: Confirm your password to continue
595 label_password_required: Confirm your password to continue
596 label_home: Home
596 label_home: Home
597 label_my_page: My page
597 label_my_page: My page
598 label_my_account: My account
598 label_my_account: My account
599 label_my_projects: My projects
599 label_my_projects: My projects
600 label_my_page_block: My page block
600 label_my_page_block: My page block
601 label_administration: Administration
601 label_administration: Administration
602 label_login: Sign in
602 label_login: Sign in
603 label_logout: Sign out
603 label_logout: Sign out
604 label_help: Help
604 label_help: Help
605 label_reported_issues: Reported issues
605 label_reported_issues: Reported issues
606 label_assigned_issues: Assigned issues
606 label_assigned_issues: Assigned issues
607 label_assigned_to_me_issues: Issues assigned to me
607 label_assigned_to_me_issues: Issues assigned to me
608 label_last_login: Last connection
608 label_last_login: Last connection
609 label_registered_on: Registered on
609 label_registered_on: Registered on
610 label_activity: Activity
610 label_activity: Activity
611 label_overall_activity: Overall activity
611 label_overall_activity: Overall activity
612 label_user_activity: "%{value}'s activity"
612 label_user_activity: "%{value}'s activity"
613 label_new: New
613 label_new: New
614 label_logged_as: Logged in as
614 label_logged_as: Logged in as
615 label_environment: Environment
615 label_environment: Environment
616 label_authentication: Authentication
616 label_authentication: Authentication
617 label_auth_source: Authentication mode
617 label_auth_source: Authentication mode
618 label_auth_source_new: New authentication mode
618 label_auth_source_new: New authentication mode
619 label_auth_source_plural: Authentication modes
619 label_auth_source_plural: Authentication modes
620 label_subproject_plural: Subprojects
620 label_subproject_plural: Subprojects
621 label_subproject_new: New subproject
621 label_subproject_new: New subproject
622 label_and_its_subprojects: "%{value} and its subprojects"
622 label_and_its_subprojects: "%{value} and its subprojects"
623 label_min_max_length: Min - Max length
623 label_min_max_length: Min - Max length
624 label_list: List
624 label_list: List
625 label_date: Date
625 label_date: Date
626 label_integer: Integer
626 label_integer: Integer
627 label_float: Float
627 label_float: Float
628 label_boolean: Boolean
628 label_boolean: Boolean
629 label_string: Text
629 label_string: Text
630 label_text: Long text
630 label_text: Long text
631 label_attribute: Attribute
631 label_attribute: Attribute
632 label_attribute_plural: Attributes
632 label_attribute_plural: Attributes
633 label_no_data: No data to display
633 label_no_data: No data to display
634 label_no_preview: No preview available
634 label_no_preview: No preview available
635 label_change_status: Change status
635 label_change_status: Change status
636 label_history: History
636 label_history: History
637 label_attachment: File
637 label_attachment: File
638 label_attachment_new: New file
638 label_attachment_new: New file
639 label_attachment_delete: Delete file
639 label_attachment_delete: Delete file
640 label_attachment_plural: Files
640 label_attachment_plural: Files
641 label_file_added: File added
641 label_file_added: File added
642 label_report: Report
642 label_report: Report
643 label_report_plural: Reports
643 label_report_plural: Reports
644 label_news: News
644 label_news: News
645 label_news_new: Add news
645 label_news_new: Add news
646 label_news_plural: News
646 label_news_plural: News
647 label_news_latest: Latest news
647 label_news_latest: Latest news
648 label_news_view_all: View all news
648 label_news_view_all: View all news
649 label_news_added: News added
649 label_news_added: News added
650 label_news_comment_added: Comment added to a news
650 label_news_comment_added: Comment added to a news
651 label_settings: Settings
651 label_settings: Settings
652 label_overview: Overview
652 label_overview: Overview
653 label_version: Version
653 label_version: Version
654 label_version_new: New version
654 label_version_new: New version
655 label_version_plural: Versions
655 label_version_plural: Versions
656 label_close_versions: Close completed versions
656 label_close_versions: Close completed versions
657 label_confirmation: Confirmation
657 label_confirmation: Confirmation
658 label_export_to: 'Also available in:'
658 label_export_to: 'Also available in:'
659 label_read: Read...
659 label_read: Read...
660 label_public_projects: Public projects
660 label_public_projects: Public projects
661 label_open_issues: open
661 label_open_issues: open
662 label_open_issues_plural: open
662 label_open_issues_plural: open
663 label_closed_issues: closed
663 label_closed_issues: closed
664 label_closed_issues_plural: closed
664 label_closed_issues_plural: closed
665 label_x_open_issues_abbr:
665 label_x_open_issues_abbr:
666 zero: 0 open
666 zero: 0 open
667 one: 1 open
667 one: 1 open
668 other: "%{count} open"
668 other: "%{count} open"
669 label_x_closed_issues_abbr:
669 label_x_closed_issues_abbr:
670 zero: 0 closed
670 zero: 0 closed
671 one: 1 closed
671 one: 1 closed
672 other: "%{count} closed"
672 other: "%{count} closed"
673 label_x_issues:
673 label_x_issues:
674 zero: 0 issues
674 zero: 0 issues
675 one: 1 issue
675 one: 1 issue
676 other: "%{count} issues"
676 other: "%{count} issues"
677 label_total: Total
677 label_total: Total
678 label_total_plural: Totals
678 label_total_plural: Totals
679 label_total_time: Total time
679 label_total_time: Total time
680 label_permissions: Permissions
680 label_permissions: Permissions
681 label_current_status: Current status
681 label_current_status: Current status
682 label_new_statuses_allowed: New statuses allowed
682 label_new_statuses_allowed: New statuses allowed
683 label_all: all
683 label_all: all
684 label_any: any
684 label_any: any
685 label_none: none
685 label_none: none
686 label_nobody: nobody
686 label_nobody: nobody
687 label_next: Next
687 label_next: Next
688 label_previous: Previous
688 label_previous: Previous
689 label_used_by: Used by
689 label_used_by: Used by
690 label_details: Details
690 label_details: Details
691 label_add_note: Add a note
691 label_add_note: Add a note
692 label_calendar: Calendar
692 label_calendar: Calendar
693 label_months_from: months from
693 label_months_from: months from
694 label_gantt: Gantt
694 label_gantt: Gantt
695 label_internal: Internal
695 label_internal: Internal
696 label_last_changes: "last %{count} changes"
696 label_last_changes: "last %{count} changes"
697 label_change_view_all: View all changes
697 label_change_view_all: View all changes
698 label_personalize_page: Personalize this page
698 label_personalize_page: Personalize this page
699 label_comment: Comment
699 label_comment: Comment
700 label_comment_plural: Comments
700 label_comment_plural: Comments
701 label_x_comments:
701 label_x_comments:
702 zero: no comments
702 zero: no comments
703 one: 1 comment
703 one: 1 comment
704 other: "%{count} comments"
704 other: "%{count} comments"
705 label_comment_add: Add a comment
705 label_comment_add: Add a comment
706 label_comment_added: Comment added
706 label_comment_added: Comment added
707 label_comment_delete: Delete comments
707 label_comment_delete: Delete comments
708 label_query: Custom query
708 label_query: Custom query
709 label_query_plural: Custom queries
709 label_query_plural: Custom queries
710 label_query_new: New query
710 label_query_new: New query
711 label_my_queries: My custom queries
711 label_my_queries: My custom queries
712 label_filter_add: Add filter
712 label_filter_add: Add filter
713 label_filter_plural: Filters
713 label_filter_plural: Filters
714 label_equals: is
714 label_equals: is
715 label_not_equals: is not
715 label_not_equals: is not
716 label_in_less_than: in less than
716 label_in_less_than: in less than
717 label_in_more_than: in more than
717 label_in_more_than: in more than
718 label_in_the_next_days: in the next
718 label_in_the_next_days: in the next
719 label_in_the_past_days: in the past
719 label_in_the_past_days: in the past
720 label_greater_or_equal: '>='
720 label_greater_or_equal: '>='
721 label_less_or_equal: '<='
721 label_less_or_equal: '<='
722 label_between: between
722 label_between: between
723 label_in: in
723 label_in: in
724 label_today: today
724 label_today: today
725 label_all_time: all time
725 label_all_time: all time
726 label_yesterday: yesterday
726 label_yesterday: yesterday
727 label_this_week: this week
727 label_this_week: this week
728 label_last_week: last week
728 label_last_week: last week
729 label_last_n_weeks: "last %{count} weeks"
729 label_last_n_weeks: "last %{count} weeks"
730 label_last_n_days: "last %{count} days"
730 label_last_n_days: "last %{count} days"
731 label_this_month: this month
731 label_this_month: this month
732 label_last_month: last month
732 label_last_month: last month
733 label_this_year: this year
733 label_this_year: this year
734 label_date_range: Date range
734 label_date_range: Date range
735 label_less_than_ago: less than days ago
735 label_less_than_ago: less than days ago
736 label_more_than_ago: more than days ago
736 label_more_than_ago: more than days ago
737 label_ago: days ago
737 label_ago: days ago
738 label_contains: contains
738 label_contains: contains
739 label_not_contains: doesn't contain
739 label_not_contains: doesn't contain
740 label_any_issues_in_project: any issues in project
740 label_any_issues_in_project: any issues in project
741 label_any_issues_not_in_project: any issues not in project
741 label_any_issues_not_in_project: any issues not in project
742 label_no_issues_in_project: no issues in project
742 label_no_issues_in_project: no issues in project
743 label_any_open_issues: any open issues
743 label_any_open_issues: any open issues
744 label_no_open_issues: no open issues
744 label_no_open_issues: no open issues
745 label_day_plural: days
745 label_day_plural: days
746 label_repository: Repository
746 label_repository: Repository
747 label_repository_new: New repository
747 label_repository_new: New repository
748 label_repository_plural: Repositories
748 label_repository_plural: Repositories
749 label_browse: Browse
749 label_browse: Browse
750 label_branch: Branch
750 label_branch: Branch
751 label_tag: Tag
751 label_tag: Tag
752 label_revision: Revision
752 label_revision: Revision
753 label_revision_plural: Revisions
753 label_revision_plural: Revisions
754 label_revision_id: "Revision %{value}"
754 label_revision_id: "Revision %{value}"
755 label_associated_revisions: Associated revisions
755 label_associated_revisions: Associated revisions
756 label_added: added
756 label_added: added
757 label_modified: modified
757 label_modified: modified
758 label_copied: copied
758 label_copied: copied
759 label_renamed: renamed
759 label_renamed: renamed
760 label_deleted: deleted
760 label_deleted: deleted
761 label_latest_revision: Latest revision
761 label_latest_revision: Latest revision
762 label_latest_revision_plural: Latest revisions
762 label_latest_revision_plural: Latest revisions
763 label_view_revisions: View revisions
763 label_view_revisions: View revisions
764 label_view_all_revisions: View all revisions
764 label_view_all_revisions: View all revisions
765 label_max_size: Maximum size
765 label_max_size: Maximum size
766 label_sort_highest: Move to top
766 label_sort_highest: Move to top
767 label_sort_higher: Move up
767 label_sort_higher: Move up
768 label_sort_lower: Move down
768 label_sort_lower: Move down
769 label_sort_lowest: Move to bottom
769 label_sort_lowest: Move to bottom
770 label_roadmap: Roadmap
770 label_roadmap: Roadmap
771 label_roadmap_due_in: "Due in %{value}"
771 label_roadmap_due_in: "Due in %{value}"
772 label_roadmap_overdue: "%{value} late"
772 label_roadmap_overdue: "%{value} late"
773 label_roadmap_no_issues: No issues for this version
773 label_roadmap_no_issues: No issues for this version
774 label_search: Search
774 label_search: Search
775 label_result_plural: Results
775 label_result_plural: Results
776 label_all_words: All words
776 label_all_words: All words
777 label_wiki: Wiki
777 label_wiki: Wiki
778 label_wiki_edit: Wiki edit
778 label_wiki_edit: Wiki edit
779 label_wiki_edit_plural: Wiki edits
779 label_wiki_edit_plural: Wiki edits
780 label_wiki_page: Wiki page
780 label_wiki_page: Wiki page
781 label_wiki_page_plural: Wiki pages
781 label_wiki_page_plural: Wiki pages
782 label_wiki_page_new: New wiki page
782 label_wiki_page_new: New wiki page
783 label_index_by_title: Index by title
783 label_index_by_title: Index by title
784 label_index_by_date: Index by date
784 label_index_by_date: Index by date
785 label_current_version: Current version
785 label_current_version: Current version
786 label_preview: Preview
786 label_preview: Preview
787 label_feed_plural: Feeds
787 label_feed_plural: Feeds
788 label_changes_details: Details of all changes
788 label_changes_details: Details of all changes
789 label_issue_tracking: Issue tracking
789 label_issue_tracking: Issue tracking
790 label_spent_time: Spent time
790 label_spent_time: Spent time
791 label_total_spent_time: Total spent time
791 label_total_spent_time: Total spent time
792 label_overall_spent_time: Overall spent time
792 label_overall_spent_time: Overall spent time
793 label_f_hour: "%{value} hour"
793 label_f_hour: "%{value} hour"
794 label_f_hour_plural: "%{value} hours"
794 label_f_hour_plural: "%{value} hours"
795 label_f_hour_short: "%{value} h"
795 label_f_hour_short: "%{value} h"
796 label_time_tracking: Time tracking
796 label_time_tracking: Time tracking
797 label_change_plural: Changes
797 label_change_plural: Changes
798 label_statistics: Statistics
798 label_statistics: Statistics
799 label_commits_per_month: Commits per month
799 label_commits_per_month: Commits per month
800 label_commits_per_author: Commits per author
800 label_commits_per_author: Commits per author
801 label_diff: diff
801 label_diff: diff
802 label_view_diff: View differences
802 label_view_diff: View differences
803 label_diff_inline: inline
803 label_diff_inline: inline
804 label_diff_side_by_side: side by side
804 label_diff_side_by_side: side by side
805 label_options: Options
805 label_options: Options
806 label_copy_workflow_from: Copy workflow from
806 label_copy_workflow_from: Copy workflow from
807 label_permissions_report: Permissions report
807 label_permissions_report: Permissions report
808 label_watched_issues: Watched issues
808 label_watched_issues: Watched issues
809 label_related_issues: Related issues
809 label_related_issues: Related issues
810 label_applied_status: Applied status
810 label_applied_status: Applied status
811 label_loading: Loading...
811 label_loading: Loading...
812 label_relation_new: New relation
812 label_relation_new: New relation
813 label_relation_delete: Delete relation
813 label_relation_delete: Delete relation
814 label_relates_to: Related to
814 label_relates_to: Related to
815 label_duplicates: Duplicates
815 label_duplicates: Duplicates
816 label_duplicated_by: Duplicated by
816 label_duplicated_by: Duplicated by
817 label_blocks: Blocks
817 label_blocks: Blocks
818 label_blocked_by: Blocked by
818 label_blocked_by: Blocked by
819 label_precedes: Precedes
819 label_precedes: Precedes
820 label_follows: Follows
820 label_follows: Follows
821 label_copied_to: Copied to
821 label_copied_to: Copied to
822 label_copied_from: Copied from
822 label_copied_from: Copied from
823 label_stay_logged_in: Stay logged in
823 label_stay_logged_in: Stay logged in
824 label_disabled: disabled
824 label_disabled: disabled
825 label_show_completed_versions: Show completed versions
825 label_show_completed_versions: Show completed versions
826 label_me: me
826 label_me: me
827 label_board: Forum
827 label_board: Forum
828 label_board_new: New forum
828 label_board_new: New forum
829 label_board_plural: Forums
829 label_board_plural: Forums
830 label_board_locked: Locked
830 label_board_locked: Locked
831 label_board_sticky: Sticky
831 label_board_sticky: Sticky
832 label_topic_plural: Topics
832 label_topic_plural: Topics
833 label_message_plural: Messages
833 label_message_plural: Messages
834 label_message_last: Last message
834 label_message_last: Last message
835 label_message_new: New message
835 label_message_new: New message
836 label_message_posted: Message added
836 label_message_posted: Message added
837 label_reply_plural: Replies
837 label_reply_plural: Replies
838 label_send_information: Send account information to the user
838 label_send_information: Send account information to the user
839 label_year: Year
839 label_year: Year
840 label_month: Month
840 label_month: Month
841 label_week: Week
841 label_week: Week
842 label_date_from: From
842 label_date_from: From
843 label_date_to: To
843 label_date_to: To
844 label_language_based: Based on user's language
844 label_language_based: Based on user's language
845 label_sort_by: "Sort by %{value}"
845 label_sort_by: "Sort by %{value}"
846 label_send_test_email: Send a test email
846 label_send_test_email: Send a test email
847 label_feeds_access_key: Atom access key
847 label_feeds_access_key: Atom access key
848 label_missing_feeds_access_key: Missing a Atom access key
848 label_missing_feeds_access_key: Missing a Atom access key
849 label_feeds_access_key_created_on: "Atom access key created %{value} ago"
849 label_feeds_access_key_created_on: "Atom access key created %{value} ago"
850 label_module_plural: Modules
850 label_module_plural: Modules
851 label_added_time_by: "Added by %{author} %{age} ago"
851 label_added_time_by: "Added by %{author} %{age} ago"
852 label_updated_time_by: "Updated by %{author} %{age} ago"
852 label_updated_time_by: "Updated by %{author} %{age} ago"
853 label_updated_time: "Updated %{value} ago"
853 label_updated_time: "Updated %{value} ago"
854 label_jump_to_a_project: Jump to a project...
854 label_jump_to_a_project: Jump to a project...
855 label_file_plural: Files
855 label_file_plural: Files
856 label_changeset_plural: Changesets
856 label_changeset_plural: Changesets
857 label_default_columns: Default columns
857 label_default_columns: Default columns
858 label_no_change_option: (No change)
858 label_no_change_option: (No change)
859 label_bulk_edit_selected_issues: Bulk edit selected issues
859 label_bulk_edit_selected_issues: Bulk edit selected issues
860 label_bulk_edit_selected_time_entries: Bulk edit selected time entries
860 label_bulk_edit_selected_time_entries: Bulk edit selected time entries
861 label_theme: Theme
861 label_theme: Theme
862 label_default: Default
862 label_default: Default
863 label_search_titles_only: Search titles only
863 label_search_titles_only: Search titles only
864 label_user_mail_option_all: "For any event on all my projects"
864 label_user_mail_option_all: "For any event on all my projects"
865 label_user_mail_option_selected: "For any event on the selected projects only..."
865 label_user_mail_option_selected: "For any event on the selected projects only..."
866 label_user_mail_option_none: "No events"
866 label_user_mail_option_none: "No events"
867 label_user_mail_option_only_my_events: "Only for things I watch or I'm involved in"
867 label_user_mail_option_only_my_events: "Only for things I watch or I'm involved in"
868 label_user_mail_option_only_assigned: "Only for things I watch or I am assigned to"
868 label_user_mail_option_only_assigned: "Only for things I watch or I am assigned to"
869 label_user_mail_option_only_owner: "Only for things I watch or I am the owner of"
869 label_user_mail_option_only_owner: "Only for things I watch or I am the owner of"
870 label_user_mail_no_self_notified: "I don't want to be notified of changes that I make myself"
870 label_user_mail_no_self_notified: "I don't want to be notified of changes that I make myself"
871 label_registration_activation_by_email: account activation by email
871 label_registration_activation_by_email: account activation by email
872 label_registration_manual_activation: manual account activation
872 label_registration_manual_activation: manual account activation
873 label_registration_automatic_activation: automatic account activation
873 label_registration_automatic_activation: automatic account activation
874 label_display_per_page: "Per page: %{value}"
874 label_display_per_page: "Per page: %{value}"
875 label_age: Age
875 label_age: Age
876 label_change_properties: Change properties
876 label_change_properties: Change properties
877 label_general: General
877 label_general: General
878 label_more: More
878 label_more: More
879 label_scm: SCM
879 label_scm: SCM
880 label_plugins: Plugins
880 label_plugins: Plugins
881 label_ldap_authentication: LDAP authentication
881 label_ldap_authentication: LDAP authentication
882 label_downloads_abbr: D/L
882 label_downloads_abbr: D/L
883 label_optional_description: Optional description
883 label_optional_description: Optional description
884 label_add_another_file: Add another file
884 label_add_another_file: Add another file
885 label_preferences: Preferences
885 label_preferences: Preferences
886 label_chronological_order: In chronological order
886 label_chronological_order: In chronological order
887 label_reverse_chronological_order: In reverse chronological order
887 label_reverse_chronological_order: In reverse chronological order
888 label_planning: Planning
888 label_planning: Planning
889 label_incoming_emails: Incoming emails
889 label_incoming_emails: Incoming emails
890 label_generate_key: Generate a key
890 label_generate_key: Generate a key
891 label_issue_watchers: Watchers
891 label_issue_watchers: Watchers
892 label_example: Example
892 label_example: Example
893 label_display: Display
893 label_display: Display
894 label_sort: Sort
894 label_sort: Sort
895 label_ascending: Ascending
895 label_ascending: Ascending
896 label_descending: Descending
896 label_descending: Descending
897 label_date_from_to: From %{start} to %{end}
897 label_date_from_to: From %{start} to %{end}
898 label_wiki_content_added: Wiki page added
898 label_wiki_content_added: Wiki page added
899 label_wiki_content_updated: Wiki page updated
899 label_wiki_content_updated: Wiki page updated
900 label_group: Group
900 label_group: Group
901 label_group_plural: Groups
901 label_group_plural: Groups
902 label_group_new: New group
902 label_group_new: New group
903 label_group_anonymous: Anonymous users
903 label_group_anonymous: Anonymous users
904 label_group_non_member: Non member users
904 label_group_non_member: Non member users
905 label_time_entry_plural: Spent time
905 label_time_entry_plural: Spent time
906 label_version_sharing_none: Not shared
906 label_version_sharing_none: Not shared
907 label_version_sharing_descendants: With subprojects
907 label_version_sharing_descendants: With subprojects
908 label_version_sharing_hierarchy: With project hierarchy
908 label_version_sharing_hierarchy: With project hierarchy
909 label_version_sharing_tree: With project tree
909 label_version_sharing_tree: With project tree
910 label_version_sharing_system: With all projects
910 label_version_sharing_system: With all projects
911 label_update_issue_done_ratios: Update issue done ratios
911 label_update_issue_done_ratios: Update issue done ratios
912 label_copy_source: Source
912 label_copy_source: Source
913 label_copy_target: Target
913 label_copy_target: Target
914 label_copy_same_as_target: Same as target
914 label_copy_same_as_target: Same as target
915 label_display_used_statuses_only: Only display statuses that are used by this tracker
915 label_display_used_statuses_only: Only display statuses that are used by this tracker
916 label_api_access_key: API access key
916 label_api_access_key: API access key
917 label_missing_api_access_key: Missing an API access key
917 label_missing_api_access_key: Missing an API access key
918 label_api_access_key_created_on: "API access key created %{value} ago"
918 label_api_access_key_created_on: "API access key created %{value} ago"
919 label_profile: Profile
919 label_profile: Profile
920 label_subtask_plural: Subtasks
920 label_subtask_plural: Subtasks
921 label_project_copy_notifications: Send email notifications during the project copy
921 label_project_copy_notifications: Send email notifications during the project copy
922 label_principal_search: "Search for user or group:"
922 label_principal_search: "Search for user or group:"
923 label_user_search: "Search for user:"
923 label_user_search: "Search for user:"
924 label_additional_workflow_transitions_for_author: Additional transitions allowed when the user is the author
924 label_additional_workflow_transitions_for_author: Additional transitions allowed when the user is the author
925 label_additional_workflow_transitions_for_assignee: Additional transitions allowed when the user is the assignee
925 label_additional_workflow_transitions_for_assignee: Additional transitions allowed when the user is the assignee
926 label_issues_visibility_all: All issues
926 label_issues_visibility_all: All issues
927 label_issues_visibility_public: All non private issues
927 label_issues_visibility_public: All non private issues
928 label_issues_visibility_own: Issues created by or assigned to the user
928 label_issues_visibility_own: Issues created by or assigned to the user
929 label_git_report_last_commit: Report last commit for files and directories
929 label_git_report_last_commit: Report last commit for files and directories
930 label_parent_revision: Parent
930 label_parent_revision: Parent
931 label_child_revision: Child
931 label_child_revision: Child
932 label_export_options: "%{export_format} export options"
932 label_export_options: "%{export_format} export options"
933 label_copy_attachments: Copy attachments
933 label_copy_attachments: Copy attachments
934 label_copy_subtasks: Copy subtasks
934 label_copy_subtasks: Copy subtasks
935 label_item_position: "%{position} of %{count}"
935 label_item_position: "%{position} of %{count}"
936 label_completed_versions: Completed versions
936 label_completed_versions: Completed versions
937 label_search_for_watchers: Search for watchers to add
937 label_search_for_watchers: Search for watchers to add
938 label_session_expiration: Session expiration
938 label_session_expiration: Session expiration
939 label_show_closed_projects: View closed projects
939 label_show_closed_projects: View closed projects
940 label_status_transitions: Status transitions
940 label_status_transitions: Status transitions
941 label_fields_permissions: Fields permissions
941 label_fields_permissions: Fields permissions
942 label_readonly: Read-only
942 label_readonly: Read-only
943 label_required: Required
943 label_required: Required
944 label_hidden: Hidden
944 label_hidden: Hidden
945 label_attribute_of_project: "Project's %{name}"
945 label_attribute_of_project: "Project's %{name}"
946 label_attribute_of_issue: "Issue's %{name}"
946 label_attribute_of_issue: "Issue's %{name}"
947 label_attribute_of_author: "Author's %{name}"
947 label_attribute_of_author: "Author's %{name}"
948 label_attribute_of_assigned_to: "Assignee's %{name}"
948 label_attribute_of_assigned_to: "Assignee's %{name}"
949 label_attribute_of_user: "User's %{name}"
949 label_attribute_of_user: "User's %{name}"
950 label_attribute_of_fixed_version: "Target version's %{name}"
950 label_attribute_of_fixed_version: "Target version's %{name}"
951 label_attribute_of_object: "%{object_name}'s %{name}"
951 label_cross_project_descendants: With subprojects
952 label_cross_project_descendants: With subprojects
952 label_cross_project_tree: With project tree
953 label_cross_project_tree: With project tree
953 label_cross_project_hierarchy: With project hierarchy
954 label_cross_project_hierarchy: With project hierarchy
954 label_cross_project_system: With all projects
955 label_cross_project_system: With all projects
955 label_gantt_progress_line: Progress line
956 label_gantt_progress_line: Progress line
956 label_visibility_private: to me only
957 label_visibility_private: to me only
957 label_visibility_roles: to these roles only
958 label_visibility_roles: to these roles only
958 label_visibility_public: to any users
959 label_visibility_public: to any users
959 label_link: Link
960 label_link: Link
960 label_only: only
961 label_only: only
961 label_drop_down_list: drop-down list
962 label_drop_down_list: drop-down list
962 label_checkboxes: checkboxes
963 label_checkboxes: checkboxes
963 label_radio_buttons: radio buttons
964 label_radio_buttons: radio buttons
964 label_link_values_to: Link values to URL
965 label_link_values_to: Link values to URL
965 label_custom_field_select_type: Select the type of object to which the custom field is to be attached
966 label_custom_field_select_type: Select the type of object to which the custom field is to be attached
966 label_check_for_updates: Check for updates
967 label_check_for_updates: Check for updates
967 label_latest_compatible_version: Latest compatible version
968 label_latest_compatible_version: Latest compatible version
968 label_unknown_plugin: Unknown plugin
969 label_unknown_plugin: Unknown plugin
969 label_add_projects: Add projects
970 label_add_projects: Add projects
970 label_users_visibility_all: All active users
971 label_users_visibility_all: All active users
971 label_users_visibility_members_of_visible_projects: Members of visible projects
972 label_users_visibility_members_of_visible_projects: Members of visible projects
972 label_edit_attachments: Edit attached files
973 label_edit_attachments: Edit attached files
973 label_link_copied_issue: Link copied issue
974 label_link_copied_issue: Link copied issue
974 label_ask: Ask
975 label_ask: Ask
975 label_search_attachments_yes: Search attachment filenames and descriptions
976 label_search_attachments_yes: Search attachment filenames and descriptions
976 label_search_attachments_no: Do not search attachments
977 label_search_attachments_no: Do not search attachments
977 label_search_attachments_only: Search attachments only
978 label_search_attachments_only: Search attachments only
978 label_search_open_issues_only: Open issues only
979 label_search_open_issues_only: Open issues only
979 label_email_address_plural: Emails
980 label_email_address_plural: Emails
980 label_email_address_add: Add email address
981 label_email_address_add: Add email address
981 label_enable_notifications: Enable notifications
982 label_enable_notifications: Enable notifications
982 label_disable_notifications: Disable notifications
983 label_disable_notifications: Disable notifications
983 label_blank_value: blank
984 label_blank_value: blank
984 label_parent_task_attributes: Parent tasks attributes
985 label_parent_task_attributes: Parent tasks attributes
985 label_parent_task_attributes_derived: Calculated from subtasks
986 label_parent_task_attributes_derived: Calculated from subtasks
986 label_parent_task_attributes_independent: Independent of subtasks
987 label_parent_task_attributes_independent: Independent of subtasks
987 label_time_entries_visibility_all: All time entries
988 label_time_entries_visibility_all: All time entries
988 label_time_entries_visibility_own: Time entries created by the user
989 label_time_entries_visibility_own: Time entries created by the user
989 label_member_management: Member management
990 label_member_management: Member management
990 label_member_management_all_roles: All roles
991 label_member_management_all_roles: All roles
991 label_member_management_selected_roles_only: Only these roles
992 label_member_management_selected_roles_only: Only these roles
992 label_import_issues: Import issues
993 label_import_issues: Import issues
993 label_select_file_to_import: Select the file to import
994 label_select_file_to_import: Select the file to import
994 label_fields_separator: Field separator
995 label_fields_separator: Field separator
995 label_fields_wrapper: Field wrapper
996 label_fields_wrapper: Field wrapper
996 label_encoding: Encoding
997 label_encoding: Encoding
997 label_comma_char: Comma
998 label_comma_char: Comma
998 label_semi_colon_char: Semicolon
999 label_semi_colon_char: Semicolon
999 label_quote_char: Quote
1000 label_quote_char: Quote
1000 label_double_quote_char: Double quote
1001 label_double_quote_char: Double quote
1001 label_fields_mapping: Fields mapping
1002 label_fields_mapping: Fields mapping
1002 label_file_content_preview: File content preview
1003 label_file_content_preview: File content preview
1003 label_create_missing_values: Create missing values
1004 label_create_missing_values: Create missing values
1004 label_api: API
1005 label_api: API
1005 label_field_format_enumeration: Key/value list
1006 label_field_format_enumeration: Key/value list
1006 label_default_values_for_new_users: Default values for new users
1007 label_default_values_for_new_users: Default values for new users
1007 label_relations: Relations
1008 label_relations: Relations
1008 label_new_project_issue_tab_enabled: Display the "New issue" tab
1009 label_new_project_issue_tab_enabled: Display the "New issue" tab
1009 label_new_object_tab_enabled: Display the "+" drop-down
1010 label_new_object_tab_enabled: Display the "+" drop-down
1010 label_table_of_contents: Table of contents
1011 label_table_of_contents: Table of contents
1011 label_font_default: Default font
1012 label_font_default: Default font
1012 label_font_monospace: Monospaced font
1013 label_font_monospace: Monospaced font
1013 label_font_proportional: Proportional font
1014 label_font_proportional: Proportional font
1014
1015
1015 button_login: Login
1016 button_login: Login
1016 button_submit: Submit
1017 button_submit: Submit
1017 button_save: Save
1018 button_save: Save
1018 button_check_all: Check all
1019 button_check_all: Check all
1019 button_uncheck_all: Uncheck all
1020 button_uncheck_all: Uncheck all
1020 button_collapse_all: Collapse all
1021 button_collapse_all: Collapse all
1021 button_expand_all: Expand all
1022 button_expand_all: Expand all
1022 button_delete: Delete
1023 button_delete: Delete
1023 button_create: Create
1024 button_create: Create
1024 button_create_and_continue: Create and continue
1025 button_create_and_continue: Create and continue
1025 button_test: Test
1026 button_test: Test
1026 button_edit: Edit
1027 button_edit: Edit
1027 button_edit_associated_wikipage: "Edit associated Wiki page: %{page_title}"
1028 button_edit_associated_wikipage: "Edit associated Wiki page: %{page_title}"
1028 button_add: Add
1029 button_add: Add
1029 button_change: Change
1030 button_change: Change
1030 button_apply: Apply
1031 button_apply: Apply
1031 button_clear: Clear
1032 button_clear: Clear
1032 button_lock: Lock
1033 button_lock: Lock
1033 button_unlock: Unlock
1034 button_unlock: Unlock
1034 button_download: Download
1035 button_download: Download
1035 button_list: List
1036 button_list: List
1036 button_view: View
1037 button_view: View
1037 button_move: Move
1038 button_move: Move
1038 button_move_and_follow: Move and follow
1039 button_move_and_follow: Move and follow
1039 button_back: Back
1040 button_back: Back
1040 button_cancel: Cancel
1041 button_cancel: Cancel
1041 button_activate: Activate
1042 button_activate: Activate
1042 button_sort: Sort
1043 button_sort: Sort
1043 button_log_time: Log time
1044 button_log_time: Log time
1044 button_rollback: Rollback to this version
1045 button_rollback: Rollback to this version
1045 button_watch: Watch
1046 button_watch: Watch
1046 button_unwatch: Unwatch
1047 button_unwatch: Unwatch
1047 button_reply: Reply
1048 button_reply: Reply
1048 button_archive: Archive
1049 button_archive: Archive
1049 button_unarchive: Unarchive
1050 button_unarchive: Unarchive
1050 button_reset: Reset
1051 button_reset: Reset
1051 button_rename: Rename
1052 button_rename: Rename
1052 button_change_password: Change password
1053 button_change_password: Change password
1053 button_copy: Copy
1054 button_copy: Copy
1054 button_copy_and_follow: Copy and follow
1055 button_copy_and_follow: Copy and follow
1055 button_annotate: Annotate
1056 button_annotate: Annotate
1056 button_update: Update
1057 button_update: Update
1057 button_configure: Configure
1058 button_configure: Configure
1058 button_quote: Quote
1059 button_quote: Quote
1059 button_duplicate: Duplicate
1060 button_duplicate: Duplicate
1060 button_show: Show
1061 button_show: Show
1061 button_hide: Hide
1062 button_hide: Hide
1062 button_edit_section: Edit this section
1063 button_edit_section: Edit this section
1063 button_export: Export
1064 button_export: Export
1064 button_delete_my_account: Delete my account
1065 button_delete_my_account: Delete my account
1065 button_close: Close
1066 button_close: Close
1066 button_reopen: Reopen
1067 button_reopen: Reopen
1067 button_import: Import
1068 button_import: Import
1068 button_filter: Filter
1069 button_filter: Filter
1069
1070
1070 status_active: active
1071 status_active: active
1071 status_registered: registered
1072 status_registered: registered
1072 status_locked: locked
1073 status_locked: locked
1073
1074
1074 project_status_active: active
1075 project_status_active: active
1075 project_status_closed: closed
1076 project_status_closed: closed
1076 project_status_archived: archived
1077 project_status_archived: archived
1077
1078
1078 version_status_open: open
1079 version_status_open: open
1079 version_status_locked: locked
1080 version_status_locked: locked
1080 version_status_closed: closed
1081 version_status_closed: closed
1081
1082
1082 field_active: Active
1083 field_active: Active
1083
1084
1084 text_select_mail_notifications: Select actions for which email notifications should be sent.
1085 text_select_mail_notifications: Select actions for which email notifications should be sent.
1085 text_regexp_info: eg. ^[A-Z0-9]+$
1086 text_regexp_info: eg. ^[A-Z0-9]+$
1086 text_min_max_length_info: 0 means no restriction
1087 text_min_max_length_info: 0 means no restriction
1087 text_project_destroy_confirmation: Are you sure you want to delete this project and related data?
1088 text_project_destroy_confirmation: Are you sure you want to delete this project and related data?
1088 text_subprojects_destroy_warning: "Its subproject(s): %{value} will be also deleted."
1089 text_subprojects_destroy_warning: "Its subproject(s): %{value} will be also deleted."
1089 text_workflow_edit: Select a role and a tracker to edit the workflow
1090 text_workflow_edit: Select a role and a tracker to edit the workflow
1090 text_are_you_sure: Are you sure?
1091 text_are_you_sure: Are you sure?
1091 text_journal_changed: "%{label} changed from %{old} to %{new}"
1092 text_journal_changed: "%{label} changed from %{old} to %{new}"
1092 text_journal_changed_no_detail: "%{label} updated"
1093 text_journal_changed_no_detail: "%{label} updated"
1093 text_journal_set_to: "%{label} set to %{value}"
1094 text_journal_set_to: "%{label} set to %{value}"
1094 text_journal_deleted: "%{label} deleted (%{old})"
1095 text_journal_deleted: "%{label} deleted (%{old})"
1095 text_journal_added: "%{label} %{value} added"
1096 text_journal_added: "%{label} %{value} added"
1096 text_tip_issue_begin_day: issue beginning this day
1097 text_tip_issue_begin_day: issue beginning this day
1097 text_tip_issue_end_day: issue ending this day
1098 text_tip_issue_end_day: issue ending this day
1098 text_tip_issue_begin_end_day: issue beginning and ending this day
1099 text_tip_issue_begin_end_day: issue beginning and ending this day
1099 text_project_identifier_info: 'Only lower case letters (a-z), numbers, dashes and underscores are allowed, must start with a lower case letter.<br />Once saved, the identifier cannot be changed.'
1100 text_project_identifier_info: 'Only lower case letters (a-z), numbers, dashes and underscores are allowed, must start with a lower case letter.<br />Once saved, the identifier cannot be changed.'
1100 text_caracters_maximum: "%{count} characters maximum."
1101 text_caracters_maximum: "%{count} characters maximum."
1101 text_caracters_minimum: "Must be at least %{count} characters long."
1102 text_caracters_minimum: "Must be at least %{count} characters long."
1102 text_length_between: "Length between %{min} and %{max} characters."
1103 text_length_between: "Length between %{min} and %{max} characters."
1103 text_tracker_no_workflow: No workflow defined for this tracker
1104 text_tracker_no_workflow: No workflow defined for this tracker
1104 text_unallowed_characters: Unallowed characters
1105 text_unallowed_characters: Unallowed characters
1105 text_comma_separated: Multiple values allowed (comma separated).
1106 text_comma_separated: Multiple values allowed (comma separated).
1106 text_line_separated: Multiple values allowed (one line for each value).
1107 text_line_separated: Multiple values allowed (one line for each value).
1107 text_issues_ref_in_commit_messages: Referencing and fixing issues in commit messages
1108 text_issues_ref_in_commit_messages: Referencing and fixing issues in commit messages
1108 text_issue_added: "Issue %{id} has been reported by %{author}."
1109 text_issue_added: "Issue %{id} has been reported by %{author}."
1109 text_issue_updated: "Issue %{id} has been updated by %{author}."
1110 text_issue_updated: "Issue %{id} has been updated by %{author}."
1110 text_wiki_destroy_confirmation: Are you sure you want to delete this wiki and all its content?
1111 text_wiki_destroy_confirmation: Are you sure you want to delete this wiki and all its content?
1111 text_issue_category_destroy_question: "Some issues (%{count}) are assigned to this category. What do you want to do?"
1112 text_issue_category_destroy_question: "Some issues (%{count}) are assigned to this category. What do you want to do?"
1112 text_issue_category_destroy_assignments: Remove category assignments
1113 text_issue_category_destroy_assignments: Remove category assignments
1113 text_issue_category_reassign_to: Reassign issues to this category
1114 text_issue_category_reassign_to: Reassign issues to this category
1114 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)."
1115 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)."
1115 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."
1116 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."
1116 text_load_default_configuration: Load the default configuration
1117 text_load_default_configuration: Load the default configuration
1117 text_status_changed_by_changeset: "Applied in changeset %{value}."
1118 text_status_changed_by_changeset: "Applied in changeset %{value}."
1118 text_time_logged_by_changeset: "Applied in changeset %{value}."
1119 text_time_logged_by_changeset: "Applied in changeset %{value}."
1119 text_issues_destroy_confirmation: 'Are you sure you want to delete the selected issue(s)?'
1120 text_issues_destroy_confirmation: 'Are you sure you want to delete the selected issue(s)?'
1120 text_issues_destroy_descendants_confirmation: "This will also delete %{count} subtask(s)."
1121 text_issues_destroy_descendants_confirmation: "This will also delete %{count} subtask(s)."
1121 text_time_entries_destroy_confirmation: 'Are you sure you want to delete the selected time entr(y/ies)?'
1122 text_time_entries_destroy_confirmation: 'Are you sure you want to delete the selected time entr(y/ies)?'
1122 text_select_project_modules: 'Select modules to enable for this project:'
1123 text_select_project_modules: 'Select modules to enable for this project:'
1123 text_default_administrator_account_changed: Default administrator account changed
1124 text_default_administrator_account_changed: Default administrator account changed
1124 text_file_repository_writable: Attachments directory writable
1125 text_file_repository_writable: Attachments directory writable
1125 text_plugin_assets_writable: Plugin assets directory writable
1126 text_plugin_assets_writable: Plugin assets directory writable
1126 text_rmagick_available: RMagick available (optional)
1127 text_rmagick_available: RMagick available (optional)
1127 text_convert_available: ImageMagick convert available (optional)
1128 text_convert_available: ImageMagick convert available (optional)
1128 text_destroy_time_entries_question: "%{hours} hours were reported on the issues you are about to delete. What do you want to do?"
1129 text_destroy_time_entries_question: "%{hours} hours were reported on the issues you are about to delete. What do you want to do?"
1129 text_destroy_time_entries: Delete reported hours
1130 text_destroy_time_entries: Delete reported hours
1130 text_assign_time_entries_to_project: Assign reported hours to the project
1131 text_assign_time_entries_to_project: Assign reported hours to the project
1131 text_reassign_time_entries: 'Reassign reported hours to this issue:'
1132 text_reassign_time_entries: 'Reassign reported hours to this issue:'
1132 text_user_wrote: "%{value} wrote:"
1133 text_user_wrote: "%{value} wrote:"
1133 text_enumeration_destroy_question: "%{count} objects are assigned to the value β€œ%{name}”."
1134 text_enumeration_destroy_question: "%{count} objects are assigned to the value β€œ%{name}”."
1134 text_enumeration_category_reassign_to: 'Reassign them to this value:'
1135 text_enumeration_category_reassign_to: 'Reassign them to this value:'
1135 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."
1136 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."
1136 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."
1137 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."
1137 text_diff_truncated: '... This diff was truncated because it exceeds the maximum size that can be displayed.'
1138 text_diff_truncated: '... This diff was truncated because it exceeds the maximum size that can be displayed.'
1138 text_custom_field_possible_values_info: 'One line for each value'
1139 text_custom_field_possible_values_info: 'One line for each value'
1139 text_wiki_page_destroy_question: "This page has %{descendants} child page(s) and descendant(s). What do you want to do?"
1140 text_wiki_page_destroy_question: "This page has %{descendants} child page(s) and descendant(s). What do you want to do?"
1140 text_wiki_page_nullify_children: "Keep child pages as root pages"
1141 text_wiki_page_nullify_children: "Keep child pages as root pages"
1141 text_wiki_page_destroy_children: "Delete child pages and all their descendants"
1142 text_wiki_page_destroy_children: "Delete child pages and all their descendants"
1142 text_wiki_page_reassign_children: "Reassign child pages to this parent page"
1143 text_wiki_page_reassign_children: "Reassign child pages to this parent page"
1143 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?"
1144 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?"
1144 text_zoom_in: Zoom in
1145 text_zoom_in: Zoom in
1145 text_zoom_out: Zoom out
1146 text_zoom_out: Zoom out
1146 text_warn_on_leaving_unsaved: "The current page contains unsaved text that will be lost if you leave this page."
1147 text_warn_on_leaving_unsaved: "The current page contains unsaved text that will be lost if you leave this page."
1147 text_scm_path_encoding_note: "Default: UTF-8"
1148 text_scm_path_encoding_note: "Default: UTF-8"
1148 text_subversion_repository_note: "Examples: file:///, http://, https://, svn://, svn+[tunnelscheme]://"
1149 text_subversion_repository_note: "Examples: file:///, http://, https://, svn://, svn+[tunnelscheme]://"
1149 text_git_repository_note: Repository is bare and local (e.g. /gitrepo, c:\gitrepo)
1150 text_git_repository_note: Repository is bare and local (e.g. /gitrepo, c:\gitrepo)
1150 text_mercurial_repository_note: Local repository (e.g. /hgrepo, c:\hgrepo)
1151 text_mercurial_repository_note: Local repository (e.g. /hgrepo, c:\hgrepo)
1151 text_scm_command: Command
1152 text_scm_command: Command
1152 text_scm_command_version: Version
1153 text_scm_command_version: Version
1153 text_scm_config: You can configure your SCM commands in config/configuration.yml. Please restart the application after editing it.
1154 text_scm_config: You can configure your SCM commands in config/configuration.yml. Please restart the application after editing it.
1154 text_scm_command_not_available: SCM command is not available. Please check settings on the administration panel.
1155 text_scm_command_not_available: SCM command is not available. Please check settings on the administration panel.
1155 text_issue_conflict_resolution_overwrite: "Apply my changes anyway (previous notes will be kept but some changes may be overwritten)"
1156 text_issue_conflict_resolution_overwrite: "Apply my changes anyway (previous notes will be kept but some changes may be overwritten)"
1156 text_issue_conflict_resolution_add_notes: "Add my notes and discard my other changes"
1157 text_issue_conflict_resolution_add_notes: "Add my notes and discard my other changes"
1157 text_issue_conflict_resolution_cancel: "Discard all my changes and redisplay %{link}"
1158 text_issue_conflict_resolution_cancel: "Discard all my changes and redisplay %{link}"
1158 text_account_destroy_confirmation: "Are you sure you want to proceed?\nYour account will be permanently deleted, with no way to reactivate it."
1159 text_account_destroy_confirmation: "Are you sure you want to proceed?\nYour account will be permanently deleted, with no way to reactivate it."
1159 text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
1160 text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
1160 text_project_closed: This project is closed and read-only.
1161 text_project_closed: This project is closed and read-only.
1161 text_turning_multiple_off: "If you disable multiple values, multiple values will be removed in order to preserve only one value per item."
1162 text_turning_multiple_off: "If you disable multiple values, multiple values will be removed in order to preserve only one value per item."
1162
1163
1163 default_role_manager: Manager
1164 default_role_manager: Manager
1164 default_role_developer: Developer
1165 default_role_developer: Developer
1165 default_role_reporter: Reporter
1166 default_role_reporter: Reporter
1166 default_tracker_bug: Bug
1167 default_tracker_bug: Bug
1167 default_tracker_feature: Feature
1168 default_tracker_feature: Feature
1168 default_tracker_support: Support
1169 default_tracker_support: Support
1169 default_issue_status_new: New
1170 default_issue_status_new: New
1170 default_issue_status_in_progress: In Progress
1171 default_issue_status_in_progress: In Progress
1171 default_issue_status_resolved: Resolved
1172 default_issue_status_resolved: Resolved
1172 default_issue_status_feedback: Feedback
1173 default_issue_status_feedback: Feedback
1173 default_issue_status_closed: Closed
1174 default_issue_status_closed: Closed
1174 default_issue_status_rejected: Rejected
1175 default_issue_status_rejected: Rejected
1175 default_doc_category_user: User documentation
1176 default_doc_category_user: User documentation
1176 default_doc_category_tech: Technical documentation
1177 default_doc_category_tech: Technical documentation
1177 default_priority_low: Low
1178 default_priority_low: Low
1178 default_priority_normal: Normal
1179 default_priority_normal: Normal
1179 default_priority_high: High
1180 default_priority_high: High
1180 default_priority_urgent: Urgent
1181 default_priority_urgent: Urgent
1181 default_priority_immediate: Immediate
1182 default_priority_immediate: Immediate
1182 default_activity_design: Design
1183 default_activity_design: Design
1183 default_activity_development: Development
1184 default_activity_development: Development
1184
1185
1185 enumeration_issue_priorities: Issue priorities
1186 enumeration_issue_priorities: Issue priorities
1186 enumeration_doc_categories: Document categories
1187 enumeration_doc_categories: Document categories
1187 enumeration_activities: Activities (time tracking)
1188 enumeration_activities: Activities (time tracking)
1188 enumeration_system_activity: System Activity
1189 enumeration_system_activity: System Activity
1189 description_filter: Filter
1190 description_filter: Filter
1190 description_search: Searchfield
1191 description_search: Searchfield
1191 description_choose_project: Projects
1192 description_choose_project: Projects
1192 description_project_scope: Search scope
1193 description_project_scope: Search scope
1193 description_notes: Notes
1194 description_notes: Notes
1194 description_message_content: Message content
1195 description_message_content: Message content
1195 description_query_sort_criteria_attribute: Sort attribute
1196 description_query_sort_criteria_attribute: Sort attribute
1196 description_query_sort_criteria_direction: Sort direction
1197 description_query_sort_criteria_direction: Sort direction
1197 description_user_mail_notification: Mail notification settings
1198 description_user_mail_notification: Mail notification settings
1198 description_available_columns: Available Columns
1199 description_available_columns: Available Columns
1199 description_selected_columns: Selected Columns
1200 description_selected_columns: Selected Columns
1200 description_all_columns: All Columns
1201 description_all_columns: All Columns
1201 description_issue_category_reassign: Choose issue category
1202 description_issue_category_reassign: Choose issue category
1202 description_wiki_subpages_reassign: Choose new parent page
1203 description_wiki_subpages_reassign: Choose new parent page
1203 description_date_range_list: Choose range from list
1204 description_date_range_list: Choose range from list
1204 description_date_range_interval: Choose range by selecting start and end date
1205 description_date_range_interval: Choose range by selecting start and end date
1205 description_date_from: Enter start date
1206 description_date_from: Enter start date
1206 description_date_to: Enter end date
1207 description_date_to: Enter end date
1207 text_repository_identifier_info: 'Only lower case letters (a-z), numbers, dashes and underscores are allowed.<br />Once saved, the identifier cannot be changed.'
1208 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,1227 +1,1228
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
18
19 # Don't forget the nil at the beginning; there's no such thing as a 0th month
19 # Don't forget the nil at the beginning; there's no such thing as a 0th month
20 month_names: [~, janvier, fΓ©vrier, mars, avril, mai, juin, juillet, aoΓ»t, septembre, octobre, novembre, dΓ©cembre]
20 month_names: [~, janvier, fΓ©vrier, mars, avril, mai, juin, juillet, aoΓ»t, septembre, octobre, novembre, dΓ©cembre]
21 abbr_month_names: [~, jan., fΓ©v., mar., avr., mai, juin, juil., aoΓ»t, sept., oct., nov., dΓ©c.]
21 abbr_month_names: [~, jan., fΓ©v., mar., avr., mai, juin, juil., aoΓ»t, sept., oct., nov., dΓ©c.]
22 # Used in date_select and datime_select.
22 # Used in date_select and datime_select.
23 order:
23 order:
24 - :day
24 - :day
25 - :month
25 - :month
26 - :year
26 - :year
27
27
28 time:
28 time:
29 formats:
29 formats:
30 default: "%d/%m/%Y %H:%M"
30 default: "%d/%m/%Y %H:%M"
31 time: "%H:%M"
31 time: "%H:%M"
32 short: "%d %b %H:%M"
32 short: "%d %b %H:%M"
33 long: "%A %d %B %Y %H:%M:%S %Z"
33 long: "%A %d %B %Y %H:%M:%S %Z"
34 long_ordinal: "%A %d %B %Y %H:%M:%S %Z"
34 long_ordinal: "%A %d %B %Y %H:%M:%S %Z"
35 only_second: "%S"
35 only_second: "%S"
36 am: 'am'
36 am: 'am'
37 pm: 'pm'
37 pm: 'pm'
38
38
39 datetime:
39 datetime:
40 distance_in_words:
40 distance_in_words:
41 half_a_minute: "30 secondes"
41 half_a_minute: "30 secondes"
42 less_than_x_seconds:
42 less_than_x_seconds:
43 zero: "moins d'une seconde"
43 zero: "moins d'une seconde"
44 one: "moins d'uneΒ seconde"
44 one: "moins d'uneΒ seconde"
45 other: "moins de %{count}Β secondes"
45 other: "moins de %{count}Β secondes"
46 x_seconds:
46 x_seconds:
47 one: "1Β seconde"
47 one: "1Β seconde"
48 other: "%{count}Β secondes"
48 other: "%{count}Β secondes"
49 less_than_x_minutes:
49 less_than_x_minutes:
50 zero: "moins d'une minute"
50 zero: "moins d'une minute"
51 one: "moins d'uneΒ minute"
51 one: "moins d'uneΒ minute"
52 other: "moins de %{count}Β minutes"
52 other: "moins de %{count}Β minutes"
53 x_minutes:
53 x_minutes:
54 one: "1Β minute"
54 one: "1Β minute"
55 other: "%{count}Β minutes"
55 other: "%{count}Β minutes"
56 about_x_hours:
56 about_x_hours:
57 one: "environ une heure"
57 one: "environ une heure"
58 other: "environ %{count}Β heures"
58 other: "environ %{count}Β heures"
59 x_hours:
59 x_hours:
60 one: "une heure"
60 one: "une heure"
61 other: "%{count}Β heures"
61 other: "%{count}Β heures"
62 x_days:
62 x_days:
63 one: "unΒ jour"
63 one: "unΒ jour"
64 other: "%{count}Β jours"
64 other: "%{count}Β jours"
65 about_x_months:
65 about_x_months:
66 one: "environ un mois"
66 one: "environ un mois"
67 other: "environ %{count}Β mois"
67 other: "environ %{count}Β mois"
68 x_months:
68 x_months:
69 one: "unΒ mois"
69 one: "unΒ mois"
70 other: "%{count}Β mois"
70 other: "%{count}Β mois"
71 about_x_years:
71 about_x_years:
72 one: "environ un an"
72 one: "environ un an"
73 other: "environ %{count}Β ans"
73 other: "environ %{count}Β ans"
74 over_x_years:
74 over_x_years:
75 one: "plus d'un an"
75 one: "plus d'un an"
76 other: "plus de %{count}Β ans"
76 other: "plus de %{count}Β ans"
77 almost_x_years:
77 almost_x_years:
78 one: "presqu'un an"
78 one: "presqu'un an"
79 other: "presque %{count} ans"
79 other: "presque %{count} ans"
80 prompts:
80 prompts:
81 year: "AnnΓ©e"
81 year: "AnnΓ©e"
82 month: "Mois"
82 month: "Mois"
83 day: "Jour"
83 day: "Jour"
84 hour: "Heure"
84 hour: "Heure"
85 minute: "Minute"
85 minute: "Minute"
86 second: "Seconde"
86 second: "Seconde"
87
87
88 number:
88 number:
89 format:
89 format:
90 precision: 3
90 precision: 3
91 separator: ','
91 separator: ','
92 delimiter: 'Β '
92 delimiter: 'Β '
93 currency:
93 currency:
94 format:
94 format:
95 unit: '€'
95 unit: '€'
96 precision: 2
96 precision: 2
97 format: '%nΒ %u'
97 format: '%nΒ %u'
98 human:
98 human:
99 format:
99 format:
100 precision: 3
100 precision: 3
101 storage_units:
101 storage_units:
102 format: "%n %u"
102 format: "%n %u"
103 units:
103 units:
104 byte:
104 byte:
105 one: "octet"
105 one: "octet"
106 other: "octets"
106 other: "octets"
107 kb: "ko"
107 kb: "ko"
108 mb: "Mo"
108 mb: "Mo"
109 gb: "Go"
109 gb: "Go"
110 tb: "To"
110 tb: "To"
111
111
112 support:
112 support:
113 array:
113 array:
114 sentence_connector: 'et'
114 sentence_connector: 'et'
115 skip_last_comma: true
115 skip_last_comma: true
116 word_connector: ", "
116 word_connector: ", "
117 two_words_connector: " et "
117 two_words_connector: " et "
118 last_word_connector: " et "
118 last_word_connector: " et "
119
119
120 activerecord:
120 activerecord:
121 errors:
121 errors:
122 template:
122 template:
123 header:
123 header:
124 one: "Impossible d'enregistrer %{model} : une erreur"
124 one: "Impossible d'enregistrer %{model} : une erreur"
125 other: "Impossible d'enregistrer %{model} : %{count} erreurs."
125 other: "Impossible d'enregistrer %{model} : %{count} erreurs."
126 body: "Veuillez vΓ©rifier les champs suivantsΒ :"
126 body: "Veuillez vΓ©rifier les champs suivantsΒ :"
127 messages:
127 messages:
128 inclusion: "n'est pas inclus(e) dans la liste"
128 inclusion: "n'est pas inclus(e) dans la liste"
129 exclusion: "n'est pas disponible"
129 exclusion: "n'est pas disponible"
130 invalid: "n'est pas valide"
130 invalid: "n'est pas valide"
131 confirmation: "ne concorde pas avec la confirmation"
131 confirmation: "ne concorde pas avec la confirmation"
132 accepted: "doit Γͺtre acceptΓ©(e)"
132 accepted: "doit Γͺtre acceptΓ©(e)"
133 empty: "doit Γͺtre renseignΓ©(e)"
133 empty: "doit Γͺtre renseignΓ©(e)"
134 blank: "doit Γͺtre renseignΓ©(e)"
134 blank: "doit Γͺtre renseignΓ©(e)"
135 too_long: "est trop long (pas plus de %{count} caractères)"
135 too_long: "est trop long (pas plus de %{count} caractères)"
136 too_short: "est trop court (au moins %{count} caractères)"
136 too_short: "est trop court (au moins %{count} caractères)"
137 wrong_length: "ne fait pas la bonne longueur (doit comporter %{count} caractères)"
137 wrong_length: "ne fait pas la bonne longueur (doit comporter %{count} caractères)"
138 taken: "est dΓ©jΓ  utilisΓ©"
138 taken: "est dΓ©jΓ  utilisΓ©"
139 not_a_number: "n'est pas un nombre"
139 not_a_number: "n'est pas un nombre"
140 not_a_date: "n'est pas une date valide"
140 not_a_date: "n'est pas une date valide"
141 greater_than: "doit Γͺtre supΓ©rieur Γ  %{count}"
141 greater_than: "doit Γͺtre supΓ©rieur Γ  %{count}"
142 greater_than_or_equal_to: "doit Γͺtre supΓ©rieur ou Γ©gal Γ  %{count}"
142 greater_than_or_equal_to: "doit Γͺtre supΓ©rieur ou Γ©gal Γ  %{count}"
143 equal_to: "doit Γͺtre Γ©gal Γ  %{count}"
143 equal_to: "doit Γͺtre Γ©gal Γ  %{count}"
144 less_than: "doit Γͺtre infΓ©rieur Γ  %{count}"
144 less_than: "doit Γͺtre infΓ©rieur Γ  %{count}"
145 less_than_or_equal_to: "doit Γͺtre infΓ©rieur ou Γ©gal Γ  %{count}"
145 less_than_or_equal_to: "doit Γͺtre infΓ©rieur ou Γ©gal Γ  %{count}"
146 odd: "doit Γͺtre impair"
146 odd: "doit Γͺtre impair"
147 even: "doit Γͺtre pair"
147 even: "doit Γͺtre pair"
148 greater_than_start_date: "doit Γͺtre postΓ©rieure Γ  la date de dΓ©but"
148 greater_than_start_date: "doit Γͺtre postΓ©rieure Γ  la date de dΓ©but"
149 not_same_project: "n'appartient pas au mΓͺme projet"
149 not_same_project: "n'appartient pas au mΓͺme projet"
150 circular_dependency: "Cette relation crΓ©erait une dΓ©pendance circulaire"
150 circular_dependency: "Cette relation crΓ©erait une dΓ©pendance circulaire"
151 cant_link_an_issue_with_a_descendant: "Une demande ne peut pas Γͺtre liΓ©e Γ  l'une de ses sous-tΓ’ches"
151 cant_link_an_issue_with_a_descendant: "Une demande ne peut pas Γͺtre liΓ©e Γ  l'une de ses sous-tΓ’ches"
152 earlier_than_minimum_start_date: "ne peut pas Γͺtre antΓ©rieure au %{date} Γ  cause des demandes qui prΓ©cΓ¨dent"
152 earlier_than_minimum_start_date: "ne peut pas Γͺtre antΓ©rieure au %{date} Γ  cause des demandes qui prΓ©cΓ¨dent"
153 not_a_regexp: "n'est pas une expression regulière valide"
153 not_a_regexp: "n'est pas une expression regulière valide"
154 open_issue_with_closed_parent: "Une demande ouverte ne peut pas Γͺtre rattachΓ©e Γ  une demande fermΓ©e"
154 open_issue_with_closed_parent: "Une demande ouverte ne peut pas Γͺtre rattachΓ©e Γ  une demande fermΓ©e"
155
155
156 actionview_instancetag_blank_option: Choisir
156 actionview_instancetag_blank_option: Choisir
157
157
158 general_text_No: 'Non'
158 general_text_No: 'Non'
159 general_text_Yes: 'Oui'
159 general_text_Yes: 'Oui'
160 general_text_no: 'non'
160 general_text_no: 'non'
161 general_text_yes: 'oui'
161 general_text_yes: 'oui'
162 general_lang_name: 'French (FranΓ§ais)'
162 general_lang_name: 'French (FranΓ§ais)'
163 general_csv_separator: ';'
163 general_csv_separator: ';'
164 general_csv_decimal_separator: ','
164 general_csv_decimal_separator: ','
165 general_csv_encoding: ISO-8859-1
165 general_csv_encoding: ISO-8859-1
166 general_pdf_fontname: freesans
166 general_pdf_fontname: freesans
167 general_pdf_monospaced_fontname: freemono
167 general_pdf_monospaced_fontname: freemono
168 general_first_day_of_week: '1'
168 general_first_day_of_week: '1'
169
169
170 notice_account_updated: Le compte a été mis à jour avec succès.
170 notice_account_updated: Le compte a été mis à jour avec succès.
171 notice_account_invalid_credentials: Identifiant ou mot de passe invalide.
171 notice_account_invalid_credentials: Identifiant ou mot de passe invalide.
172 notice_account_password_updated: Mot de passe mis à jour avec succès.
172 notice_account_password_updated: Mot de passe mis à jour avec succès.
173 notice_account_wrong_password: Mot de passe incorrect
173 notice_account_wrong_password: Mot de passe incorrect
174 notice_account_register_done: Un message contenant les instructions pour activer votre compte vous a Γ©tΓ© envoyΓ© Γ  l'adresse %{email}.
174 notice_account_register_done: Un message contenant les instructions pour activer votre compte vous a Γ©tΓ© envoyΓ© Γ  l'adresse %{email}.
175 notice_account_unknown_email: Aucun compte ne correspond Γ  cette adresse.
175 notice_account_unknown_email: Aucun compte ne correspond Γ  cette adresse.
176 notice_account_not_activated_yet: Vous n'avez pas encore activΓ© votre compte. Si vous voulez recevoir un nouveau message d'activation, veuillez <a href="%{url}">cliquer sur ce lien</a>.
176 notice_account_not_activated_yet: Vous n'avez pas encore activΓ© votre compte. Si vous voulez recevoir un nouveau message d'activation, veuillez <a href="%{url}">cliquer sur ce lien</a>.
177 notice_account_locked: Votre compte est verrouillΓ©.
177 notice_account_locked: Votre compte est verrouillΓ©.
178 notice_can_t_change_password: Ce compte utilise une authentification externe. Impossible de changer le mot de passe.
178 notice_can_t_change_password: Ce compte utilise une authentification externe. Impossible de changer le mot de passe.
179 notice_account_lost_email_sent: Un message contenant les instructions pour choisir un nouveau mot de passe vous a Γ©tΓ© envoyΓ©.
179 notice_account_lost_email_sent: Un message contenant les instructions pour choisir un nouveau mot de passe vous a Γ©tΓ© envoyΓ©.
180 notice_account_activated: Votre compte a Γ©tΓ© activΓ©. Vous pouvez Γ  prΓ©sent vous connecter.
180 notice_account_activated: Votre compte a Γ©tΓ© activΓ©. Vous pouvez Γ  prΓ©sent vous connecter.
181 notice_successful_create: Création effectuée avec succès.
181 notice_successful_create: Création effectuée avec succès.
182 notice_successful_update: Mise à jour effectuée avec succès.
182 notice_successful_update: Mise à jour effectuée avec succès.
183 notice_successful_delete: Suppression effectuée avec succès.
183 notice_successful_delete: Suppression effectuée avec succès.
184 notice_successful_connection: Connexion rΓ©ussie.
184 notice_successful_connection: Connexion rΓ©ussie.
185 notice_file_not_found: "La page Γ  laquelle vous souhaitez accΓ©der n'existe pas ou a Γ©tΓ© supprimΓ©e."
185 notice_file_not_found: "La page Γ  laquelle vous souhaitez accΓ©der n'existe pas ou a Γ©tΓ© supprimΓ©e."
186 notice_locking_conflict: Les donnΓ©es ont Γ©tΓ© mises Γ  jour par un autre utilisateur. Mise Γ  jour impossible.
186 notice_locking_conflict: Les donnΓ©es ont Γ©tΓ© mises Γ  jour par un autre utilisateur. Mise Γ  jour impossible.
187 notice_not_authorized: "Vous n'Γͺtes pas autorisΓ© Γ  accΓ©der Γ  cette page."
187 notice_not_authorized: "Vous n'Γͺtes pas autorisΓ© Γ  accΓ©der Γ  cette page."
188 notice_not_authorized_archived_project: Le projet auquel vous tentez d'accΓ©der a Γ©tΓ© archivΓ©.
188 notice_not_authorized_archived_project: Le projet auquel vous tentez d'accΓ©der a Γ©tΓ© archivΓ©.
189 notice_email_sent: "Un email a Γ©tΓ© envoyΓ© Γ  %{value}"
189 notice_email_sent: "Un email a Γ©tΓ© envoyΓ© Γ  %{value}"
190 notice_email_error: "Erreur lors de l'envoi de l'email (%{value})"
190 notice_email_error: "Erreur lors de l'envoi de l'email (%{value})"
191 notice_feeds_access_key_reseted: "Votre clé d'accès aux flux Atom a été réinitialisée."
191 notice_feeds_access_key_reseted: "Votre clé d'accès aux flux Atom a été réinitialisée."
192 notice_api_access_key_reseted: Votre clé d'accès API a été réinitialisée.
192 notice_api_access_key_reseted: Votre clé d'accès API a été réinitialisée.
193 notice_failed_to_save_issues: "%{count} demande(s) sur les %{total} sΓ©lectionnΓ©es n'ont pas pu Γͺtre mise(s) Γ  jour : %{ids}."
193 notice_failed_to_save_issues: "%{count} demande(s) sur les %{total} sΓ©lectionnΓ©es n'ont pas pu Γͺtre mise(s) Γ  jour : %{ids}."
194 notice_failed_to_save_time_entries: "%{count} temps passΓ©(s) sur les %{total} sΓ©lectionnΓ©s n'ont pas pu Γͺtre mis Γ  jour: %{ids}."
194 notice_failed_to_save_time_entries: "%{count} temps passΓ©(s) sur les %{total} sΓ©lectionnΓ©s n'ont pas pu Γͺtre mis Γ  jour: %{ids}."
195 notice_failed_to_save_members: "Erreur lors de la sauvegarde des membres: %{errors}."
195 notice_failed_to_save_members: "Erreur lors de la sauvegarde des membres: %{errors}."
196 notice_no_issue_selected: "Aucune demande sΓ©lectionnΓ©e ! Cochez les demandes que vous voulez mettre Γ  jour."
196 notice_no_issue_selected: "Aucune demande sΓ©lectionnΓ©e ! Cochez les demandes que vous voulez mettre Γ  jour."
197 notice_account_pending: "Votre compte a été créé et attend l'approbation de l'administrateur."
197 notice_account_pending: "Votre compte a été créé et attend l'approbation de l'administrateur."
198 notice_default_data_loaded: Paramétrage par défaut chargé avec succès.
198 notice_default_data_loaded: Paramétrage par défaut chargé avec succès.
199 notice_unable_delete_version: Impossible de supprimer cette version.
199 notice_unable_delete_version: Impossible de supprimer cette version.
200 notice_unable_delete_time_entry: Impossible de supprimer le temps passΓ©.
200 notice_unable_delete_time_entry: Impossible de supprimer le temps passΓ©.
201 notice_issue_done_ratios_updated: L'avancement des demandes a Γ©tΓ© mis Γ  jour.
201 notice_issue_done_ratios_updated: L'avancement des demandes a Γ©tΓ© mis Γ  jour.
202 notice_gantt_chart_truncated: "Le diagramme a Γ©tΓ© tronquΓ© car il excΓ¨de le nombre maximal d'Γ©lΓ©ments pouvant Γͺtre affichΓ©s (%{max})"
202 notice_gantt_chart_truncated: "Le diagramme a Γ©tΓ© tronquΓ© car il excΓ¨de le nombre maximal d'Γ©lΓ©ments pouvant Γͺtre affichΓ©s (%{max})"
203 notice_issue_successful_create: "Demande %{id} créée."
203 notice_issue_successful_create: "Demande %{id} créée."
204 notice_issue_update_conflict: "La demande a Γ©tΓ© mise Γ  jour par un autre utilisateur pendant que vous la modifiez."
204 notice_issue_update_conflict: "La demande a Γ©tΓ© mise Γ  jour par un autre utilisateur pendant que vous la modifiez."
205 notice_account_deleted: "Votre compte a Γ©tΓ© dΓ©finitivement supprimΓ©."
205 notice_account_deleted: "Votre compte a Γ©tΓ© dΓ©finitivement supprimΓ©."
206 notice_user_successful_create: "Utilisateur %{id} créé."
206 notice_user_successful_create: "Utilisateur %{id} créé."
207 notice_new_password_must_be_different: Votre nouveau mot de passe doit Γͺtre diffΓ©rent de votre mot de passe actuel
207 notice_new_password_must_be_different: Votre nouveau mot de passe doit Γͺtre diffΓ©rent de votre mot de passe actuel
208 notice_import_finished: "%{count} Γ©lΓ©ments ont Γ©tΓ© importΓ©(s)"
208 notice_import_finished: "%{count} Γ©lΓ©ments ont Γ©tΓ© importΓ©(s)"
209 notice_import_finished_with_errors: "%{count} Γ©lΓ©ment(s) sur %{total} n'ont pas pu Γͺtre importΓ©(s)"
209 notice_import_finished_with_errors: "%{count} Γ©lΓ©ment(s) sur %{total} n'ont pas pu Γͺtre importΓ©(s)"
210
210
211 error_can_t_load_default_data: "Une erreur s'est produite lors du chargement du paramΓ©trage : %{value}"
211 error_can_t_load_default_data: "Une erreur s'est produite lors du chargement du paramΓ©trage : %{value}"
212 error_scm_not_found: "L'entrΓ©e et/ou la rΓ©vision demandΓ©e n'existe pas dans le dΓ©pΓ΄t."
212 error_scm_not_found: "L'entrΓ©e et/ou la rΓ©vision demandΓ©e n'existe pas dans le dΓ©pΓ΄t."
213 error_scm_command_failed: "Une erreur s'est produite lors de l'accès au dépôt : %{value}"
213 error_scm_command_failed: "Une erreur s'est produite lors de l'accès au dépôt : %{value}"
214 error_scm_annotate: "L'entrΓ©e n'existe pas ou ne peut pas Γͺtre annotΓ©e."
214 error_scm_annotate: "L'entrΓ©e n'existe pas ou ne peut pas Γͺtre annotΓ©e."
215 error_scm_annotate_big_text_file: Cette entrΓ©e ne peut pas Γͺtre annotΓ©e car elle excΓ¨de la taille maximale.
215 error_scm_annotate_big_text_file: Cette entrΓ©e ne peut pas Γͺtre annotΓ©e car elle excΓ¨de la taille maximale.
216 error_issue_not_found_in_project: "La demande n'existe pas ou n'appartient pas Γ  ce projet"
216 error_issue_not_found_in_project: "La demande n'existe pas ou n'appartient pas Γ  ce projet"
217 error_no_tracker_in_project: "Aucun tracker n'est associΓ© Γ  ce projet. VΓ©rifier la configuration du projet."
217 error_no_tracker_in_project: "Aucun tracker n'est associΓ© Γ  ce projet. VΓ©rifier la configuration du projet."
218 error_no_default_issue_status: "Aucun statut de demande n'est dΓ©fini par dΓ©faut. VΓ©rifier votre configuration (Administration -> Statuts de demandes)."
218 error_no_default_issue_status: "Aucun statut de demande n'est dΓ©fini par dΓ©faut. VΓ©rifier votre configuration (Administration -> Statuts de demandes)."
219 error_can_not_delete_custom_field: Impossible de supprimer le champ personnalisΓ©
219 error_can_not_delete_custom_field: Impossible de supprimer le champ personnalisΓ©
220 error_can_not_delete_tracker: Ce tracker contient des demandes et ne peut pas Γͺtre supprimΓ©.
220 error_can_not_delete_tracker: Ce tracker contient des demandes et ne peut pas Γͺtre supprimΓ©.
221 error_can_not_remove_role: Ce rΓ΄le est utilisΓ© et ne peut pas Γͺtre supprimΓ©.
221 error_can_not_remove_role: Ce rΓ΄le est utilisΓ© et ne peut pas Γͺtre supprimΓ©.
222 error_can_not_reopen_issue_on_closed_version: 'Une demande assignΓ©e Γ  une version fermΓ©e ne peut pas Γͺtre rΓ©ouverte'
222 error_can_not_reopen_issue_on_closed_version: 'Une demande assignΓ©e Γ  une version fermΓ©e ne peut pas Γͺtre rΓ©ouverte'
223 error_can_not_archive_project: "Ce projet ne peut pas Γͺtre archivΓ©"
223 error_can_not_archive_project: "Ce projet ne peut pas Γͺtre archivΓ©"
224 error_issue_done_ratios_not_updated: L'avancement des demandes n'a pas pu Γͺtre mis Γ  jour.
224 error_issue_done_ratios_not_updated: L'avancement des demandes n'a pas pu Γͺtre mis Γ  jour.
225 error_workflow_copy_source: 'Veuillez sΓ©lectionner un tracker et/ou un rΓ΄le source'
225 error_workflow_copy_source: 'Veuillez sΓ©lectionner un tracker et/ou un rΓ΄le source'
226 error_workflow_copy_target: 'Veuillez sΓ©lectionner les trackers et rΓ΄les cibles'
226 error_workflow_copy_target: 'Veuillez sΓ©lectionner les trackers et rΓ΄les cibles'
227 error_unable_delete_issue_status: Impossible de supprimer le statut de demande
227 error_unable_delete_issue_status: Impossible de supprimer le statut de demande
228 error_unable_to_connect: Connexion impossible (%{value})
228 error_unable_to_connect: Connexion impossible (%{value})
229 error_attachment_too_big: Ce fichier ne peut pas Γͺtre attachΓ© car il excΓ¨de la taille maximale autorisΓ©e (%{max_size})
229 error_attachment_too_big: Ce fichier ne peut pas Γͺtre attachΓ© car il excΓ¨de la taille maximale autorisΓ©e (%{max_size})
230 error_session_expired: "Votre session a expirΓ©. Veuillez vous reconnecter."
230 error_session_expired: "Votre session a expirΓ©. Veuillez vous reconnecter."
231 warning_attachments_not_saved: "%{count} fichier(s) n'ont pas pu Γͺtre sauvegardΓ©s."
231 warning_attachments_not_saved: "%{count} fichier(s) n'ont pas pu Γͺtre sauvegardΓ©s."
232 error_password_expired: "Votre mot de passe a expirΓ© ou nΓ©cessite d'Γͺtre changΓ©."
232 error_password_expired: "Votre mot de passe a expirΓ© ou nΓ©cessite d'Γͺtre changΓ©."
233 error_invalid_file_encoding: "Le fichier n'est pas un fichier %{encoding} valide"
233 error_invalid_file_encoding: "Le fichier n'est pas un fichier %{encoding} valide"
234 error_invalid_csv_file_or_settings: "Le fichier n'est pas un fichier CSV ou n'est pas conforme aux paramètres sélectionnés"
234 error_invalid_csv_file_or_settings: "Le fichier n'est pas un fichier CSV ou n'est pas conforme aux paramètres sélectionnés"
235 error_can_not_read_import_file: "Une erreur est survenue lors de la lecture du fichier Γ  importer"
235 error_can_not_read_import_file: "Une erreur est survenue lors de la lecture du fichier Γ  importer"
236 error_attachment_extension_not_allowed: "L'extension %{extension} n'est pas autorisΓ©e"
236 error_attachment_extension_not_allowed: "L'extension %{extension} n'est pas autorisΓ©e"
237 error_ldap_bind_credentials: "Identifiant ou mot de passe LDAP incorrect"
237 error_ldap_bind_credentials: "Identifiant ou mot de passe LDAP incorrect"
238 error_no_tracker_allowed_for_new_issue_in_project: "Le projet ne dispose d'aucun tracker sur lequel vous pouvez crΓ©er une demande"
238 error_no_tracker_allowed_for_new_issue_in_project: "Le projet ne dispose d'aucun tracker sur lequel vous pouvez crΓ©er une demande"
239 error_no_projects_with_tracker_allowed_for_new_issue: "Aucun projet ne dispose d'un tracker sur lequel vous pouvez crΓ©er une demande"
239 error_no_projects_with_tracker_allowed_for_new_issue: "Aucun projet ne dispose d'un tracker sur lequel vous pouvez crΓ©er une demande"
240 error_move_of_child_not_possible: "La sous-tΓ’che %{child} n'a pas pu Γͺtre dΓ©placΓ©e dans le nouveau projet : %{errors}"
240 error_move_of_child_not_possible: "La sous-tΓ’che %{child} n'a pas pu Γͺtre dΓ©placΓ©e dans le nouveau projet : %{errors}"
241 error_cannot_reassign_time_entries_to_an_issue_about_to_be_deleted: "Le temps passΓ© ne peut pas Γͺtre rΓ©affectΓ© Γ  une demande qui va Γͺtre supprimΓ©e"
241 error_cannot_reassign_time_entries_to_an_issue_about_to_be_deleted: "Le temps passΓ© ne peut pas Γͺtre rΓ©affectΓ© Γ  une demande qui va Γͺtre supprimΓ©e"
242
242
243 mail_subject_lost_password: "Votre mot de passe %{value}"
243 mail_subject_lost_password: "Votre mot de passe %{value}"
244 mail_body_lost_password: 'Pour changer votre mot de passe, cliquez sur le lien suivant :'
244 mail_body_lost_password: 'Pour changer votre mot de passe, cliquez sur le lien suivant :'
245 mail_subject_register: "Activation de votre compte %{value}"
245 mail_subject_register: "Activation de votre compte %{value}"
246 mail_body_register: 'Pour activer votre compte, cliquez sur le lien suivant :'
246 mail_body_register: 'Pour activer votre compte, cliquez sur le lien suivant :'
247 mail_body_account_information_external: "Vous pouvez utiliser votre compte %{value} pour vous connecter."
247 mail_body_account_information_external: "Vous pouvez utiliser votre compte %{value} pour vous connecter."
248 mail_body_account_information: Paramètres de connexion de votre compte
248 mail_body_account_information: Paramètres de connexion de votre compte
249 mail_subject_account_activation_request: "Demande d'activation d'un compte %{value}"
249 mail_subject_account_activation_request: "Demande d'activation d'un compte %{value}"
250 mail_body_account_activation_request: "Un nouvel utilisateur (%{value}) s'est inscrit. Son compte nΓ©cessite votre approbation :"
250 mail_body_account_activation_request: "Un nouvel utilisateur (%{value}) s'est inscrit. Son compte nΓ©cessite votre approbation :"
251 mail_subject_reminder: "%{count} demande(s) arrivent Γ  Γ©chΓ©ance (%{days})"
251 mail_subject_reminder: "%{count} demande(s) arrivent Γ  Γ©chΓ©ance (%{days})"
252 mail_body_reminder: "%{count} demande(s) qui vous sont assignΓ©es arrivent Γ  Γ©chΓ©ance dans les %{days} prochains jours :"
252 mail_body_reminder: "%{count} demande(s) qui vous sont assignΓ©es arrivent Γ  Γ©chΓ©ance dans les %{days} prochains jours :"
253 mail_subject_wiki_content_added: "Page wiki '%{id}' ajoutΓ©e"
253 mail_subject_wiki_content_added: "Page wiki '%{id}' ajoutΓ©e"
254 mail_body_wiki_content_added: "La page wiki '%{id}' a Γ©tΓ© ajoutΓ©e par %{author}."
254 mail_body_wiki_content_added: "La page wiki '%{id}' a Γ©tΓ© ajoutΓ©e par %{author}."
255 mail_subject_wiki_content_updated: "Page wiki '%{id}' mise Γ  jour"
255 mail_subject_wiki_content_updated: "Page wiki '%{id}' mise Γ  jour"
256 mail_body_wiki_content_updated: "La page wiki '%{id}' a Γ©tΓ© mise Γ  jour par %{author}."
256 mail_body_wiki_content_updated: "La page wiki '%{id}' a Γ©tΓ© mise Γ  jour par %{author}."
257 mail_body_settings_updated: "Les paramètres suivants ont été modifiés :"
257 mail_body_settings_updated: "Les paramètres suivants ont été modifiés :"
258 mail_body_password_updated: "Votre mot de passe a Γ©tΓ© changΓ©."
258 mail_body_password_updated: "Votre mot de passe a Γ©tΓ© changΓ©."
259
259
260 field_name: Nom
260 field_name: Nom
261 field_description: Description
261 field_description: Description
262 field_summary: RΓ©sumΓ©
262 field_summary: RΓ©sumΓ©
263 field_is_required: Obligatoire
263 field_is_required: Obligatoire
264 field_firstname: PrΓ©nom
264 field_firstname: PrΓ©nom
265 field_lastname: Nom
265 field_lastname: Nom
266 field_mail: Email
266 field_mail: Email
267 field_address: Email
267 field_address: Email
268 field_filename: Fichier
268 field_filename: Fichier
269 field_filesize: Taille
269 field_filesize: Taille
270 field_downloads: TΓ©lΓ©chargements
270 field_downloads: TΓ©lΓ©chargements
271 field_author: Auteur
271 field_author: Auteur
272 field_created_on: Créé
272 field_created_on: Créé
273 field_updated_on: Mis-Γ -jour
273 field_updated_on: Mis-Γ -jour
274 field_closed_on: FermΓ©
274 field_closed_on: FermΓ©
275 field_field_format: Format
275 field_field_format: Format
276 field_is_for_all: Pour tous les projets
276 field_is_for_all: Pour tous les projets
277 field_possible_values: Valeurs possibles
277 field_possible_values: Valeurs possibles
278 field_regexp: Expression régulière
278 field_regexp: Expression régulière
279 field_min_length: Longueur minimum
279 field_min_length: Longueur minimum
280 field_max_length: Longueur maximum
280 field_max_length: Longueur maximum
281 field_value: Valeur
281 field_value: Valeur
282 field_category: CatΓ©gorie
282 field_category: CatΓ©gorie
283 field_title: Titre
283 field_title: Titre
284 field_project: Projet
284 field_project: Projet
285 field_issue: Demande
285 field_issue: Demande
286 field_status: Statut
286 field_status: Statut
287 field_notes: Notes
287 field_notes: Notes
288 field_is_closed: Demande fermΓ©e
288 field_is_closed: Demande fermΓ©e
289 field_is_default: Valeur par dΓ©faut
289 field_is_default: Valeur par dΓ©faut
290 field_tracker: Tracker
290 field_tracker: Tracker
291 field_subject: Sujet
291 field_subject: Sujet
292 field_due_date: EchΓ©ance
292 field_due_date: EchΓ©ance
293 field_assigned_to: AssignΓ© Γ 
293 field_assigned_to: AssignΓ© Γ 
294 field_priority: PrioritΓ©
294 field_priority: PrioritΓ©
295 field_fixed_version: Version cible
295 field_fixed_version: Version cible
296 field_user: Utilisateur
296 field_user: Utilisateur
297 field_principal: Principal
297 field_principal: Principal
298 field_role: RΓ΄le
298 field_role: RΓ΄le
299 field_homepage: Site web
299 field_homepage: Site web
300 field_is_public: Public
300 field_is_public: Public
301 field_parent: Sous-projet de
301 field_parent: Sous-projet de
302 field_is_in_roadmap: Demandes affichΓ©es dans la roadmap
302 field_is_in_roadmap: Demandes affichΓ©es dans la roadmap
303 field_login: Identifiant
303 field_login: Identifiant
304 field_mail_notification: Notifications par mail
304 field_mail_notification: Notifications par mail
305 field_admin: Administrateur
305 field_admin: Administrateur
306 field_last_login_on: Dernière connexion
306 field_last_login_on: Dernière connexion
307 field_language: Langue
307 field_language: Langue
308 field_effective_date: Date
308 field_effective_date: Date
309 field_password: Mot de passe
309 field_password: Mot de passe
310 field_new_password: Nouveau mot de passe
310 field_new_password: Nouveau mot de passe
311 field_password_confirmation: Confirmation
311 field_password_confirmation: Confirmation
312 field_version: Version
312 field_version: Version
313 field_type: Type
313 field_type: Type
314 field_host: HΓ΄te
314 field_host: HΓ΄te
315 field_port: Port
315 field_port: Port
316 field_account: Compte
316 field_account: Compte
317 field_base_dn: Base DN
317 field_base_dn: Base DN
318 field_attr_login: Attribut Identifiant
318 field_attr_login: Attribut Identifiant
319 field_attr_firstname: Attribut PrΓ©nom
319 field_attr_firstname: Attribut PrΓ©nom
320 field_attr_lastname: Attribut Nom
320 field_attr_lastname: Attribut Nom
321 field_attr_mail: Attribut Email
321 field_attr_mail: Attribut Email
322 field_onthefly: CrΓ©ation des utilisateurs Γ  la volΓ©e
322 field_onthefly: CrΓ©ation des utilisateurs Γ  la volΓ©e
323 field_start_date: DΓ©but
323 field_start_date: DΓ©but
324 field_done_ratio: "% rΓ©alisΓ©"
324 field_done_ratio: "% rΓ©alisΓ©"
325 field_auth_source: Mode d'authentification
325 field_auth_source: Mode d'authentification
326 field_hide_mail: Cacher mon adresse mail
326 field_hide_mail: Cacher mon adresse mail
327 field_comments: Commentaire
327 field_comments: Commentaire
328 field_url: URL
328 field_url: URL
329 field_start_page: Page de dΓ©marrage
329 field_start_page: Page de dΓ©marrage
330 field_subproject: Sous-projet
330 field_subproject: Sous-projet
331 field_hours: Heures
331 field_hours: Heures
332 field_activity: ActivitΓ©
332 field_activity: ActivitΓ©
333 field_spent_on: Date
333 field_spent_on: Date
334 field_identifier: Identifiant
334 field_identifier: Identifiant
335 field_is_filter: UtilisΓ© comme filtre
335 field_is_filter: UtilisΓ© comme filtre
336 field_issue_to: Demande liΓ©e
336 field_issue_to: Demande liΓ©e
337 field_delay: Retard
337 field_delay: Retard
338 field_assignable: Demandes assignables Γ  ce rΓ΄le
338 field_assignable: Demandes assignables Γ  ce rΓ΄le
339 field_redirect_existing_links: Rediriger les liens existants
339 field_redirect_existing_links: Rediriger les liens existants
340 field_estimated_hours: Temps estimΓ©
340 field_estimated_hours: Temps estimΓ©
341 field_column_names: Colonnes
341 field_column_names: Colonnes
342 field_time_entries: Temps passΓ©
342 field_time_entries: Temps passΓ©
343 field_time_zone: Fuseau horaire
343 field_time_zone: Fuseau horaire
344 field_searchable: UtilisΓ© pour les recherches
344 field_searchable: UtilisΓ© pour les recherches
345 field_default_value: Valeur par dΓ©faut
345 field_default_value: Valeur par dΓ©faut
346 field_comments_sorting: Afficher les commentaires
346 field_comments_sorting: Afficher les commentaires
347 field_parent_title: Page parent
347 field_parent_title: Page parent
348 field_editable: Modifiable
348 field_editable: Modifiable
349 field_watcher: Observateur
349 field_watcher: Observateur
350 field_identity_url: URL OpenID
350 field_identity_url: URL OpenID
351 field_content: Contenu
351 field_content: Contenu
352 field_group_by: Grouper par
352 field_group_by: Grouper par
353 field_sharing: Partage
353 field_sharing: Partage
354 field_parent_issue: TΓ’che parente
354 field_parent_issue: TΓ’che parente
355 field_member_of_group: Groupe de l'assignΓ©
355 field_member_of_group: Groupe de l'assignΓ©
356 field_assigned_to_role: RΓ΄le de l'assignΓ©
356 field_assigned_to_role: RΓ΄le de l'assignΓ©
357 field_text: Champ texte
357 field_text: Champ texte
358 field_visible: Visible
358 field_visible: Visible
359 field_warn_on_leaving_unsaved: "M'avertir lorsque je quitte une page contenant du texte non sauvegardΓ©"
359 field_warn_on_leaving_unsaved: "M'avertir lorsque je quitte une page contenant du texte non sauvegardΓ©"
360 field_issues_visibility: VisibilitΓ© des demandes
360 field_issues_visibility: VisibilitΓ© des demandes
361 field_is_private: PrivΓ©e
361 field_is_private: PrivΓ©e
362 field_commit_logs_encoding: Encodage des messages de commit
362 field_commit_logs_encoding: Encodage des messages de commit
363 field_scm_path_encoding: Encodage des chemins
363 field_scm_path_encoding: Encodage des chemins
364 field_path_to_repository: Chemin du dΓ©pΓ΄t
364 field_path_to_repository: Chemin du dΓ©pΓ΄t
365 field_root_directory: RΓ©pertoire racine
365 field_root_directory: RΓ©pertoire racine
366 field_cvsroot: CVSROOT
366 field_cvsroot: CVSROOT
367 field_cvs_module: Module
367 field_cvs_module: Module
368 field_repository_is_default: DΓ©pΓ΄t principal
368 field_repository_is_default: DΓ©pΓ΄t principal
369 field_multiple: Valeurs multiples
369 field_multiple: Valeurs multiples
370 field_auth_source_ldap_filter: Filtre LDAP
370 field_auth_source_ldap_filter: Filtre LDAP
371 field_core_fields: Champs standards
371 field_core_fields: Champs standards
372 field_timeout: "Timeout (en secondes)"
372 field_timeout: "Timeout (en secondes)"
373 field_board_parent: Forum parent
373 field_board_parent: Forum parent
374 field_private_notes: Notes privΓ©es
374 field_private_notes: Notes privΓ©es
375 field_inherit_members: HΓ©riter les membres
375 field_inherit_members: HΓ©riter les membres
376 field_generate_password: GΓ©nΓ©rer un mot de passe
376 field_generate_password: GΓ©nΓ©rer un mot de passe
377 field_must_change_passwd: Doit changer de mot de passe Γ  la prochaine connexion
377 field_must_change_passwd: Doit changer de mot de passe Γ  la prochaine connexion
378 field_default_status: Statut par dΓ©faut
378 field_default_status: Statut par dΓ©faut
379 field_users_visibility: VisibilitΓ© des utilisateurs
379 field_users_visibility: VisibilitΓ© des utilisateurs
380 field_time_entries_visibility: VisibilitΓ© du temps passΓ©
380 field_time_entries_visibility: VisibilitΓ© du temps passΓ©
381 field_total_estimated_hours: Temps estimΓ© total
381 field_total_estimated_hours: Temps estimΓ© total
382 field_default_version: Version par dΓ©faut
382 field_default_version: Version par dΓ©faut
383 field_textarea_font: Police utilisΓ©e pour les champs texte
383 field_textarea_font: Police utilisΓ©e pour les champs texte
384
384
385 setting_app_title: Titre de l'application
385 setting_app_title: Titre de l'application
386 setting_app_subtitle: Sous-titre de l'application
386 setting_app_subtitle: Sous-titre de l'application
387 setting_welcome_text: Texte d'accueil
387 setting_welcome_text: Texte d'accueil
388 setting_default_language: Langue par dΓ©faut
388 setting_default_language: Langue par dΓ©faut
389 setting_login_required: Authentification obligatoire
389 setting_login_required: Authentification obligatoire
390 setting_self_registration: Inscription des nouveaux utilisateurs
390 setting_self_registration: Inscription des nouveaux utilisateurs
391 setting_attachment_max_size: Taille maximale des fichiers
391 setting_attachment_max_size: Taille maximale des fichiers
392 setting_issues_export_limit: Limite d'exportation des demandes
392 setting_issues_export_limit: Limite d'exportation des demandes
393 setting_mail_from: Adresse d'Γ©mission
393 setting_mail_from: Adresse d'Γ©mission
394 setting_bcc_recipients: Destinataires en copie cachΓ©e (cci)
394 setting_bcc_recipients: Destinataires en copie cachΓ©e (cci)
395 setting_plain_text_mail: Mail en texte brut (non HTML)
395 setting_plain_text_mail: Mail en texte brut (non HTML)
396 setting_host_name: Nom d'hΓ΄te et chemin
396 setting_host_name: Nom d'hΓ΄te et chemin
397 setting_text_formatting: Formatage du texte
397 setting_text_formatting: Formatage du texte
398 setting_wiki_compression: Compression de l'historique des pages wiki
398 setting_wiki_compression: Compression de l'historique des pages wiki
399 setting_feeds_limit: Nombre maximal d'Γ©lΓ©ments dans les flux Atom
399 setting_feeds_limit: Nombre maximal d'Γ©lΓ©ments dans les flux Atom
400 setting_default_projects_public: DΓ©finir les nouveaux projets comme publics par dΓ©faut
400 setting_default_projects_public: DΓ©finir les nouveaux projets comme publics par dΓ©faut
401 setting_autofetch_changesets: RΓ©cupΓ©ration automatique des commits
401 setting_autofetch_changesets: RΓ©cupΓ©ration automatique des commits
402 setting_sys_api_enabled: Activer les WS pour la gestion des dΓ©pΓ΄ts
402 setting_sys_api_enabled: Activer les WS pour la gestion des dΓ©pΓ΄ts
403 setting_commit_ref_keywords: Mots-clΓ©s de rΓ©fΓ©rencement
403 setting_commit_ref_keywords: Mots-clΓ©s de rΓ©fΓ©rencement
404 setting_commit_fix_keywords: Mots-clΓ©s de rΓ©solution
404 setting_commit_fix_keywords: Mots-clΓ©s de rΓ©solution
405 setting_autologin: DurΓ©e maximale de connexion automatique
405 setting_autologin: DurΓ©e maximale de connexion automatique
406 setting_date_format: Format de date
406 setting_date_format: Format de date
407 setting_time_format: Format d'heure
407 setting_time_format: Format d'heure
408 setting_timespan_format: Format des temps en heures
408 setting_timespan_format: Format des temps en heures
409 setting_cross_project_issue_relations: Autoriser les relations entre demandes de diffΓ©rents projets
409 setting_cross_project_issue_relations: Autoriser les relations entre demandes de diffΓ©rents projets
410 setting_cross_project_subtasks: Autoriser les sous-tΓ’ches dans des projets diffΓ©rents
410 setting_cross_project_subtasks: Autoriser les sous-tΓ’ches dans des projets diffΓ©rents
411 setting_issue_list_default_columns: Colonnes affichΓ©es par dΓ©faut sur la liste des demandes
411 setting_issue_list_default_columns: Colonnes affichΓ©es par dΓ©faut sur la liste des demandes
412 setting_repositories_encodings: Encodages des fichiers et des dΓ©pΓ΄ts
412 setting_repositories_encodings: Encodages des fichiers et des dΓ©pΓ΄ts
413 setting_emails_header: En-tΓͺte des emails
413 setting_emails_header: En-tΓͺte des emails
414 setting_emails_footer: Pied-de-page des emails
414 setting_emails_footer: Pied-de-page des emails
415 setting_protocol: Protocole
415 setting_protocol: Protocole
416 setting_per_page_options: Options d'objets affichΓ©s par page
416 setting_per_page_options: Options d'objets affichΓ©s par page
417 setting_user_format: Format d'affichage des utilisateurs
417 setting_user_format: Format d'affichage des utilisateurs
418 setting_activity_days_default: Nombre de jours affichΓ©s sur l'activitΓ© des projets
418 setting_activity_days_default: Nombre de jours affichΓ©s sur l'activitΓ© des projets
419 setting_display_subprojects_issues: Afficher par dΓ©faut les demandes des sous-projets sur les projets principaux
419 setting_display_subprojects_issues: Afficher par dΓ©faut les demandes des sous-projets sur les projets principaux
420 setting_enabled_scm: SCM activΓ©s
420 setting_enabled_scm: SCM activΓ©s
421 setting_mail_handler_body_delimiters: "Tronquer les emails après l'une de ces lignes"
421 setting_mail_handler_body_delimiters: "Tronquer les emails après l'une de ces lignes"
422 setting_mail_handler_enable_regex_delimiters: "Utiliser les expressions regulières"
422 setting_mail_handler_enable_regex_delimiters: "Utiliser les expressions regulières"
423 setting_mail_handler_api_enabled: "Activer le WS pour la rΓ©ception d'emails"
423 setting_mail_handler_api_enabled: "Activer le WS pour la rΓ©ception d'emails"
424 setting_mail_handler_api_key: ClΓ© de protection de l'API
424 setting_mail_handler_api_key: ClΓ© de protection de l'API
425 setting_sequential_project_identifiers: GΓ©nΓ©rer des identifiants de projet sΓ©quentiels
425 setting_sequential_project_identifiers: GΓ©nΓ©rer des identifiants de projet sΓ©quentiels
426 setting_gravatar_enabled: Afficher les Gravatar des utilisateurs
426 setting_gravatar_enabled: Afficher les Gravatar des utilisateurs
427 setting_gravatar_default: Image Gravatar par dΓ©faut
427 setting_gravatar_default: Image Gravatar par dΓ©faut
428 setting_diff_max_lines_displayed: Nombre maximum de lignes de diff affichΓ©es
428 setting_diff_max_lines_displayed: Nombre maximum de lignes de diff affichΓ©es
429 setting_file_max_size_displayed: Taille maximum des fichiers texte affichΓ©s en ligne
429 setting_file_max_size_displayed: Taille maximum des fichiers texte affichΓ©s en ligne
430 setting_repository_log_display_limit: "Nombre maximum de rΓ©visions affichΓ©es sur l'historique d'un fichier"
430 setting_repository_log_display_limit: "Nombre maximum de rΓ©visions affichΓ©es sur l'historique d'un fichier"
431 setting_openid: "Autoriser l'authentification et l'enregistrement OpenID"
431 setting_openid: "Autoriser l'authentification et l'enregistrement OpenID"
432 setting_password_max_age: Expiration des mots de passe après
432 setting_password_max_age: Expiration des mots de passe après
433 setting_password_min_length: Longueur minimum des mots de passe
433 setting_password_min_length: Longueur minimum des mots de passe
434 setting_new_project_user_role_id: RΓ΄le donnΓ© Γ  un utilisateur non-administrateur qui crΓ©e un projet
434 setting_new_project_user_role_id: RΓ΄le donnΓ© Γ  un utilisateur non-administrateur qui crΓ©e un projet
435 setting_default_projects_modules: Modules activΓ©s par dΓ©faut pour les nouveaux projets
435 setting_default_projects_modules: Modules activΓ©s par dΓ©faut pour les nouveaux projets
436 setting_issue_done_ratio: Calcul de l'avancement des demandes
436 setting_issue_done_ratio: Calcul de l'avancement des demandes
437 setting_issue_done_ratio_issue_field: 'Utiliser le champ % effectuΓ©'
437 setting_issue_done_ratio_issue_field: 'Utiliser le champ % effectuΓ©'
438 setting_issue_done_ratio_issue_status: Utiliser le statut
438 setting_issue_done_ratio_issue_status: Utiliser le statut
439 setting_start_of_week: Jour de dΓ©but des calendriers
439 setting_start_of_week: Jour de dΓ©but des calendriers
440 setting_rest_api_enabled: Activer l'API REST
440 setting_rest_api_enabled: Activer l'API REST
441 setting_cache_formatted_text: Mettre en cache le texte formatΓ©
441 setting_cache_formatted_text: Mettre en cache le texte formatΓ©
442 setting_default_notification_option: Option de notification par dΓ©faut
442 setting_default_notification_option: Option de notification par dΓ©faut
443 setting_commit_logtime_enabled: Permettre la saisie de temps
443 setting_commit_logtime_enabled: Permettre la saisie de temps
444 setting_commit_logtime_activity_id: ActivitΓ© pour le temps saisi
444 setting_commit_logtime_activity_id: ActivitΓ© pour le temps saisi
445 setting_gantt_items_limit: Nombre maximum d'Γ©lΓ©ments affichΓ©s sur le gantt
445 setting_gantt_items_limit: Nombre maximum d'Γ©lΓ©ments affichΓ©s sur le gantt
446 setting_issue_group_assignment: Permettre l'assignation des demandes aux groupes
446 setting_issue_group_assignment: Permettre l'assignation des demandes aux groupes
447 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
447 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
448 setting_commit_cross_project_ref: Permettre le rΓ©fΓ©rencement et la rΓ©solution des demandes de tous les autres projets
448 setting_commit_cross_project_ref: Permettre le rΓ©fΓ©rencement et la rΓ©solution des demandes de tous les autres projets
449 setting_unsubscribe: Permettre aux utilisateurs de supprimer leur propre compte
449 setting_unsubscribe: Permettre aux utilisateurs de supprimer leur propre compte
450 setting_session_lifetime: DurΓ©e de vie maximale des sessions
450 setting_session_lifetime: DurΓ©e de vie maximale des sessions
451 setting_session_timeout: DurΓ©e maximale d'inactivitΓ©
451 setting_session_timeout: DurΓ©e maximale d'inactivitΓ©
452 setting_thumbnails_enabled: Afficher les vignettes des images
452 setting_thumbnails_enabled: Afficher les vignettes des images
453 setting_thumbnails_size: Taille des vignettes (en pixels)
453 setting_thumbnails_size: Taille des vignettes (en pixels)
454 setting_non_working_week_days: Jours non travaillΓ©s
454 setting_non_working_week_days: Jours non travaillΓ©s
455 setting_jsonp_enabled: Activer le support JSONP
455 setting_jsonp_enabled: Activer le support JSONP
456 setting_default_projects_tracker_ids: Trackers par dΓ©faut pour les nouveaux projets
456 setting_default_projects_tracker_ids: Trackers par dΓ©faut pour les nouveaux projets
457 setting_mail_handler_excluded_filenames: Exclure les fichiers attachΓ©s par leur nom
457 setting_mail_handler_excluded_filenames: Exclure les fichiers attachΓ©s par leur nom
458 setting_force_default_language_for_anonymous: Forcer la langue par dΓ©fault pour les utilisateurs anonymes
458 setting_force_default_language_for_anonymous: Forcer la langue par dΓ©fault pour les utilisateurs anonymes
459 setting_force_default_language_for_loggedin: Forcer la langue par dΓ©fault pour les utilisateurs identifiΓ©s
459 setting_force_default_language_for_loggedin: Forcer la langue par dΓ©fault pour les utilisateurs identifiΓ©s
460 setting_link_copied_issue: Lier les demandes lors de la copie
460 setting_link_copied_issue: Lier les demandes lors de la copie
461 setting_max_additional_emails: Nombre maximal d'adresses email additionnelles
461 setting_max_additional_emails: Nombre maximal d'adresses email additionnelles
462 setting_search_results_per_page: RΓ©sultats de recherche affichΓ©s par page
462 setting_search_results_per_page: RΓ©sultats de recherche affichΓ©s par page
463 setting_attachment_extensions_allowed: Extensions autorisΓ©es
463 setting_attachment_extensions_allowed: Extensions autorisΓ©es
464 setting_attachment_extensions_denied: Extensions non autorisΓ©es
464 setting_attachment_extensions_denied: Extensions non autorisΓ©es
465 setting_sys_api_key: ClΓ© de protection de l'API
465 setting_sys_api_key: ClΓ© de protection de l'API
466 setting_lost_password: Autoriser la rΓ©initialisation par email de mot de passe perdu
466 setting_lost_password: Autoriser la rΓ©initialisation par email de mot de passe perdu
467 setting_new_item_menu_tab: Onglet de crΓ©ation d'objets dans le menu du project
467 setting_new_item_menu_tab: Onglet de crΓ©ation d'objets dans le menu du project
468 setting_commit_logs_formatting: Appliquer le formattage de texte aux messages de commit
468 setting_commit_logs_formatting: Appliquer le formattage de texte aux messages de commit
469 setting_timelog_required_fields: Champs obligatoire pour les temps passΓ©s
469 setting_timelog_required_fields: Champs obligatoire pour les temps passΓ©s
470
470
471 permission_add_project: CrΓ©er un projet
471 permission_add_project: CrΓ©er un projet
472 permission_add_subprojects: CrΓ©er des sous-projets
472 permission_add_subprojects: CrΓ©er des sous-projets
473 permission_edit_project: Modifier le projet
473 permission_edit_project: Modifier le projet
474 permission_close_project: Fermer / rΓ©ouvrir le projet
474 permission_close_project: Fermer / rΓ©ouvrir le projet
475 permission_select_project_modules: Choisir les modules
475 permission_select_project_modules: Choisir les modules
476 permission_manage_members: GΓ©rer les membres
476 permission_manage_members: GΓ©rer les membres
477 permission_manage_project_activities: GΓ©rer les activitΓ©s
477 permission_manage_project_activities: GΓ©rer les activitΓ©s
478 permission_manage_versions: GΓ©rer les versions
478 permission_manage_versions: GΓ©rer les versions
479 permission_manage_categories: GΓ©rer les catΓ©gories de demandes
479 permission_manage_categories: GΓ©rer les catΓ©gories de demandes
480 permission_view_issues: Voir les demandes
480 permission_view_issues: Voir les demandes
481 permission_add_issues: CrΓ©er des demandes
481 permission_add_issues: CrΓ©er des demandes
482 permission_edit_issues: Modifier les demandes
482 permission_edit_issues: Modifier les demandes
483 permission_copy_issues: Copier les demandes
483 permission_copy_issues: Copier les demandes
484 permission_manage_issue_relations: GΓ©rer les relations
484 permission_manage_issue_relations: GΓ©rer les relations
485 permission_set_issues_private: Rendre les demandes publiques ou privΓ©es
485 permission_set_issues_private: Rendre les demandes publiques ou privΓ©es
486 permission_set_own_issues_private: Rendre ses propres demandes publiques ou privΓ©es
486 permission_set_own_issues_private: Rendre ses propres demandes publiques ou privΓ©es
487 permission_add_issue_notes: Ajouter des notes
487 permission_add_issue_notes: Ajouter des notes
488 permission_edit_issue_notes: Modifier les notes
488 permission_edit_issue_notes: Modifier les notes
489 permission_edit_own_issue_notes: Modifier ses propres notes
489 permission_edit_own_issue_notes: Modifier ses propres notes
490 permission_view_private_notes: Voir les notes privΓ©es
490 permission_view_private_notes: Voir les notes privΓ©es
491 permission_set_notes_private: Rendre les notes privΓ©es
491 permission_set_notes_private: Rendre les notes privΓ©es
492 permission_move_issues: DΓ©placer les demandes
492 permission_move_issues: DΓ©placer les demandes
493 permission_delete_issues: Supprimer les demandes
493 permission_delete_issues: Supprimer les demandes
494 permission_manage_public_queries: GΓ©rer les requΓͺtes publiques
494 permission_manage_public_queries: GΓ©rer les requΓͺtes publiques
495 permission_save_queries: Sauvegarder les requΓͺtes
495 permission_save_queries: Sauvegarder les requΓͺtes
496 permission_view_gantt: Voir le gantt
496 permission_view_gantt: Voir le gantt
497 permission_view_calendar: Voir le calendrier
497 permission_view_calendar: Voir le calendrier
498 permission_view_issue_watchers: Voir la liste des observateurs
498 permission_view_issue_watchers: Voir la liste des observateurs
499 permission_add_issue_watchers: Ajouter des observateurs
499 permission_add_issue_watchers: Ajouter des observateurs
500 permission_delete_issue_watchers: Supprimer des observateurs
500 permission_delete_issue_watchers: Supprimer des observateurs
501 permission_log_time: Saisir le temps passΓ©
501 permission_log_time: Saisir le temps passΓ©
502 permission_view_time_entries: Voir le temps passΓ©
502 permission_view_time_entries: Voir le temps passΓ©
503 permission_edit_time_entries: Modifier les temps passΓ©s
503 permission_edit_time_entries: Modifier les temps passΓ©s
504 permission_edit_own_time_entries: Modifier son propre temps passΓ©
504 permission_edit_own_time_entries: Modifier son propre temps passΓ©
505 permission_manage_news: GΓ©rer les annonces
505 permission_manage_news: GΓ©rer les annonces
506 permission_comment_news: Commenter les annonces
506 permission_comment_news: Commenter les annonces
507 permission_view_documents: Voir les documents
507 permission_view_documents: Voir les documents
508 permission_add_documents: Ajouter des documents
508 permission_add_documents: Ajouter des documents
509 permission_edit_documents: Modifier les documents
509 permission_edit_documents: Modifier les documents
510 permission_delete_documents: Supprimer les documents
510 permission_delete_documents: Supprimer les documents
511 permission_manage_files: GΓ©rer les fichiers
511 permission_manage_files: GΓ©rer les fichiers
512 permission_view_files: Voir les fichiers
512 permission_view_files: Voir les fichiers
513 permission_manage_wiki: GΓ©rer le wiki
513 permission_manage_wiki: GΓ©rer le wiki
514 permission_rename_wiki_pages: Renommer les pages
514 permission_rename_wiki_pages: Renommer les pages
515 permission_delete_wiki_pages: Supprimer les pages
515 permission_delete_wiki_pages: Supprimer les pages
516 permission_view_wiki_pages: Voir le wiki
516 permission_view_wiki_pages: Voir le wiki
517 permission_view_wiki_edits: "Voir l'historique des modifications"
517 permission_view_wiki_edits: "Voir l'historique des modifications"
518 permission_edit_wiki_pages: Modifier les pages
518 permission_edit_wiki_pages: Modifier les pages
519 permission_delete_wiki_pages_attachments: Supprimer les fichiers joints
519 permission_delete_wiki_pages_attachments: Supprimer les fichiers joints
520 permission_protect_wiki_pages: ProtΓ©ger les pages
520 permission_protect_wiki_pages: ProtΓ©ger les pages
521 permission_manage_repository: GΓ©rer le dΓ©pΓ΄t de sources
521 permission_manage_repository: GΓ©rer le dΓ©pΓ΄t de sources
522 permission_browse_repository: Parcourir les sources
522 permission_browse_repository: Parcourir les sources
523 permission_view_changesets: Voir les rΓ©visions
523 permission_view_changesets: Voir les rΓ©visions
524 permission_commit_access: Droit de commit
524 permission_commit_access: Droit de commit
525 permission_manage_boards: GΓ©rer les forums
525 permission_manage_boards: GΓ©rer les forums
526 permission_view_messages: Voir les messages
526 permission_view_messages: Voir les messages
527 permission_add_messages: Poster un message
527 permission_add_messages: Poster un message
528 permission_edit_messages: Modifier les messages
528 permission_edit_messages: Modifier les messages
529 permission_edit_own_messages: Modifier ses propres messages
529 permission_edit_own_messages: Modifier ses propres messages
530 permission_delete_messages: Supprimer les messages
530 permission_delete_messages: Supprimer les messages
531 permission_delete_own_messages: Supprimer ses propres messages
531 permission_delete_own_messages: Supprimer ses propres messages
532 permission_export_wiki_pages: Exporter les pages
532 permission_export_wiki_pages: Exporter les pages
533 permission_manage_subtasks: GΓ©rer les sous-tΓ’ches
533 permission_manage_subtasks: GΓ©rer les sous-tΓ’ches
534 permission_manage_related_issues: GΓ©rer les demandes associΓ©es
534 permission_manage_related_issues: GΓ©rer les demandes associΓ©es
535 permission_import_issues: Importer des demandes
535 permission_import_issues: Importer des demandes
536
536
537 project_module_issue_tracking: Suivi des demandes
537 project_module_issue_tracking: Suivi des demandes
538 project_module_time_tracking: Suivi du temps passΓ©
538 project_module_time_tracking: Suivi du temps passΓ©
539 project_module_news: Publication d'annonces
539 project_module_news: Publication d'annonces
540 project_module_documents: Publication de documents
540 project_module_documents: Publication de documents
541 project_module_files: Publication de fichiers
541 project_module_files: Publication de fichiers
542 project_module_wiki: Wiki
542 project_module_wiki: Wiki
543 project_module_repository: DΓ©pΓ΄t de sources
543 project_module_repository: DΓ©pΓ΄t de sources
544 project_module_boards: Forums de discussion
544 project_module_boards: Forums de discussion
545 project_module_calendar: Calendrier
545 project_module_calendar: Calendrier
546 project_module_gantt: Gantt
546 project_module_gantt: Gantt
547
547
548 label_user: Utilisateur
548 label_user: Utilisateur
549 label_user_plural: Utilisateurs
549 label_user_plural: Utilisateurs
550 label_user_new: Nouvel utilisateur
550 label_user_new: Nouvel utilisateur
551 label_user_anonymous: Anonyme
551 label_user_anonymous: Anonyme
552 label_project: Projet
552 label_project: Projet
553 label_project_new: Nouveau projet
553 label_project_new: Nouveau projet
554 label_project_plural: Projets
554 label_project_plural: Projets
555 label_x_projects:
555 label_x_projects:
556 zero: aucun projet
556 zero: aucun projet
557 one: un projet
557 one: un projet
558 other: "%{count} projets"
558 other: "%{count} projets"
559 label_project_all: Tous les projets
559 label_project_all: Tous les projets
560 label_project_latest: Derniers projets
560 label_project_latest: Derniers projets
561 label_issue: Demande
561 label_issue: Demande
562 label_issue_new: Nouvelle demande
562 label_issue_new: Nouvelle demande
563 label_issue_plural: Demandes
563 label_issue_plural: Demandes
564 label_issue_view_all: Voir toutes les demandes
564 label_issue_view_all: Voir toutes les demandes
565 label_issues_by: "Demandes par %{value}"
565 label_issues_by: "Demandes par %{value}"
566 label_issue_added: Demande ajoutΓ©e
566 label_issue_added: Demande ajoutΓ©e
567 label_issue_updated: Demande mise Γ  jour
567 label_issue_updated: Demande mise Γ  jour
568 label_issue_note_added: Note ajoutΓ©e
568 label_issue_note_added: Note ajoutΓ©e
569 label_issue_status_updated: Statut changΓ©
569 label_issue_status_updated: Statut changΓ©
570 label_issue_assigned_to_updated: AssignΓ© changΓ©
570 label_issue_assigned_to_updated: AssignΓ© changΓ©
571 label_issue_priority_updated: PrioritΓ© changΓ©e
571 label_issue_priority_updated: PrioritΓ© changΓ©e
572 label_document: Document
572 label_document: Document
573 label_document_new: Nouveau document
573 label_document_new: Nouveau document
574 label_document_plural: Documents
574 label_document_plural: Documents
575 label_document_added: Document ajoutΓ©
575 label_document_added: Document ajoutΓ©
576 label_role: RΓ΄le
576 label_role: RΓ΄le
577 label_role_plural: RΓ΄les
577 label_role_plural: RΓ΄les
578 label_role_new: Nouveau rΓ΄le
578 label_role_new: Nouveau rΓ΄le
579 label_role_and_permissions: RΓ΄les et permissions
579 label_role_and_permissions: RΓ΄les et permissions
580 label_role_anonymous: Anonyme
580 label_role_anonymous: Anonyme
581 label_role_non_member: Non membre
581 label_role_non_member: Non membre
582 label_member: Membre
582 label_member: Membre
583 label_member_new: Nouveau membre
583 label_member_new: Nouveau membre
584 label_member_plural: Membres
584 label_member_plural: Membres
585 label_tracker: Tracker
585 label_tracker: Tracker
586 label_tracker_plural: Trackers
586 label_tracker_plural: Trackers
587 label_tracker_all: Tous les trackers
587 label_tracker_all: Tous les trackers
588 label_tracker_new: Nouveau tracker
588 label_tracker_new: Nouveau tracker
589 label_workflow: Workflow
589 label_workflow: Workflow
590 label_issue_status: Statut de demandes
590 label_issue_status: Statut de demandes
591 label_issue_status_plural: Statuts de demandes
591 label_issue_status_plural: Statuts de demandes
592 label_issue_status_new: Nouveau statut
592 label_issue_status_new: Nouveau statut
593 label_issue_category: CatΓ©gorie de demandes
593 label_issue_category: CatΓ©gorie de demandes
594 label_issue_category_plural: CatΓ©gories de demandes
594 label_issue_category_plural: CatΓ©gories de demandes
595 label_issue_category_new: Nouvelle catΓ©gorie
595 label_issue_category_new: Nouvelle catΓ©gorie
596 label_custom_field: Champ personnalisΓ©
596 label_custom_field: Champ personnalisΓ©
597 label_custom_field_plural: Champs personnalisΓ©s
597 label_custom_field_plural: Champs personnalisΓ©s
598 label_custom_field_new: Nouveau champ personnalisΓ©
598 label_custom_field_new: Nouveau champ personnalisΓ©
599 label_enumerations: Listes de valeurs
599 label_enumerations: Listes de valeurs
600 label_enumeration_new: Nouvelle valeur
600 label_enumeration_new: Nouvelle valeur
601 label_information: Information
601 label_information: Information
602 label_information_plural: Informations
602 label_information_plural: Informations
603 label_please_login: Identification
603 label_please_login: Identification
604 label_register: S'enregistrer
604 label_register: S'enregistrer
605 label_login_with_open_id_option: S'authentifier avec OpenID
605 label_login_with_open_id_option: S'authentifier avec OpenID
606 label_password_lost: Mot de passe perdu
606 label_password_lost: Mot de passe perdu
607 label_password_required: Confirmez votre mot de passe pour continuer
607 label_password_required: Confirmez votre mot de passe pour continuer
608 label_home: Accueil
608 label_home: Accueil
609 label_my_page: Ma page
609 label_my_page: Ma page
610 label_my_account: Mon compte
610 label_my_account: Mon compte
611 label_my_projects: Mes projets
611 label_my_projects: Mes projets
612 label_my_page_block: Blocs disponibles
612 label_my_page_block: Blocs disponibles
613 label_administration: Administration
613 label_administration: Administration
614 label_login: Connexion
614 label_login: Connexion
615 label_logout: DΓ©connexion
615 label_logout: DΓ©connexion
616 label_help: Aide
616 label_help: Aide
617 label_reported_issues: Demandes soumises
617 label_reported_issues: Demandes soumises
618 label_assigned_issues: Demandes assignΓ©es
618 label_assigned_issues: Demandes assignΓ©es
619 label_assigned_to_me_issues: Demandes qui me sont assignΓ©es
619 label_assigned_to_me_issues: Demandes qui me sont assignΓ©es
620 label_last_login: Dernière connexion
620 label_last_login: Dernière connexion
621 label_registered_on: Inscrit le
621 label_registered_on: Inscrit le
622 label_activity: ActivitΓ©
622 label_activity: ActivitΓ©
623 label_overall_activity: ActivitΓ© globale
623 label_overall_activity: ActivitΓ© globale
624 label_user_activity: "ActivitΓ© de %{value}"
624 label_user_activity: "ActivitΓ© de %{value}"
625 label_new: Nouveau
625 label_new: Nouveau
626 label_logged_as: ConnectΓ© en tant que
626 label_logged_as: ConnectΓ© en tant que
627 label_environment: Environnement
627 label_environment: Environnement
628 label_authentication: Authentification
628 label_authentication: Authentification
629 label_auth_source: Mode d'authentification
629 label_auth_source: Mode d'authentification
630 label_auth_source_new: Nouveau mode d'authentification
630 label_auth_source_new: Nouveau mode d'authentification
631 label_auth_source_plural: Modes d'authentification
631 label_auth_source_plural: Modes d'authentification
632 label_subproject_plural: Sous-projets
632 label_subproject_plural: Sous-projets
633 label_subproject_new: Nouveau sous-projet
633 label_subproject_new: Nouveau sous-projet
634 label_and_its_subprojects: "%{value} et ses sous-projets"
634 label_and_its_subprojects: "%{value} et ses sous-projets"
635 label_min_max_length: Longueurs mini - maxi
635 label_min_max_length: Longueurs mini - maxi
636 label_list: Liste
636 label_list: Liste
637 label_date: Date
637 label_date: Date
638 label_integer: Entier
638 label_integer: Entier
639 label_float: Nombre dΓ©cimal
639 label_float: Nombre dΓ©cimal
640 label_boolean: BoolΓ©en
640 label_boolean: BoolΓ©en
641 label_string: Texte
641 label_string: Texte
642 label_text: Texte long
642 label_text: Texte long
643 label_attribute: Attribut
643 label_attribute: Attribut
644 label_attribute_plural: Attributs
644 label_attribute_plural: Attributs
645 label_no_data: Aucune donnΓ©e Γ  afficher
645 label_no_data: Aucune donnΓ©e Γ  afficher
646 label_change_status: Changer le statut
646 label_change_status: Changer le statut
647 label_history: Historique
647 label_history: Historique
648 label_attachment: Fichier
648 label_attachment: Fichier
649 label_attachment_new: Nouveau fichier
649 label_attachment_new: Nouveau fichier
650 label_attachment_delete: Supprimer le fichier
650 label_attachment_delete: Supprimer le fichier
651 label_attachment_plural: Fichiers
651 label_attachment_plural: Fichiers
652 label_file_added: Fichier ajoutΓ©
652 label_file_added: Fichier ajoutΓ©
653 label_report: Rapport
653 label_report: Rapport
654 label_report_plural: Rapports
654 label_report_plural: Rapports
655 label_news: Annonce
655 label_news: Annonce
656 label_news_new: Nouvelle annonce
656 label_news_new: Nouvelle annonce
657 label_news_plural: Annonces
657 label_news_plural: Annonces
658 label_news_latest: Dernières annonces
658 label_news_latest: Dernières annonces
659 label_news_view_all: Voir toutes les annonces
659 label_news_view_all: Voir toutes les annonces
660 label_news_added: Annonce ajoutΓ©e
660 label_news_added: Annonce ajoutΓ©e
661 label_news_comment_added: Commentaire ajoutΓ© Γ  une annonce
661 label_news_comment_added: Commentaire ajoutΓ© Γ  une annonce
662 label_settings: Configuration
662 label_settings: Configuration
663 label_overview: AperΓ§u
663 label_overview: AperΓ§u
664 label_version: Version
664 label_version: Version
665 label_version_new: Nouvelle version
665 label_version_new: Nouvelle version
666 label_version_plural: Versions
666 label_version_plural: Versions
667 label_close_versions: Fermer les versions terminΓ©es
667 label_close_versions: Fermer les versions terminΓ©es
668 label_confirmation: Confirmation
668 label_confirmation: Confirmation
669 label_export_to: 'Formats disponibles :'
669 label_export_to: 'Formats disponibles :'
670 label_read: Lire...
670 label_read: Lire...
671 label_public_projects: Projets publics
671 label_public_projects: Projets publics
672 label_open_issues: ouvert
672 label_open_issues: ouvert
673 label_open_issues_plural: ouverts
673 label_open_issues_plural: ouverts
674 label_closed_issues: fermΓ©
674 label_closed_issues: fermΓ©
675 label_closed_issues_plural: fermΓ©s
675 label_closed_issues_plural: fermΓ©s
676 label_x_open_issues_abbr:
676 label_x_open_issues_abbr:
677 zero: 0 ouverte
677 zero: 0 ouverte
678 one: 1 ouverte
678 one: 1 ouverte
679 other: "%{count} ouvertes"
679 other: "%{count} ouvertes"
680 label_x_closed_issues_abbr:
680 label_x_closed_issues_abbr:
681 zero: 0 fermΓ©e
681 zero: 0 fermΓ©e
682 one: 1 fermΓ©e
682 one: 1 fermΓ©e
683 other: "%{count} fermΓ©es"
683 other: "%{count} fermΓ©es"
684 label_x_issues:
684 label_x_issues:
685 zero: 0 demande
685 zero: 0 demande
686 one: 1 demande
686 one: 1 demande
687 other: "%{count} demandes"
687 other: "%{count} demandes"
688 label_total: Total
688 label_total: Total
689 label_total_plural: Totaux
689 label_total_plural: Totaux
690 label_total_time: Temps total
690 label_total_time: Temps total
691 label_permissions: Permissions
691 label_permissions: Permissions
692 label_current_status: Statut actuel
692 label_current_status: Statut actuel
693 label_new_statuses_allowed: Nouveaux statuts autorisΓ©s
693 label_new_statuses_allowed: Nouveaux statuts autorisΓ©s
694 label_all: tous
694 label_all: tous
695 label_any: tous
695 label_any: tous
696 label_none: aucun
696 label_none: aucun
697 label_nobody: personne
697 label_nobody: personne
698 label_next: Suivant
698 label_next: Suivant
699 label_previous: PrΓ©cΓ©dent
699 label_previous: PrΓ©cΓ©dent
700 label_used_by: UtilisΓ© par
700 label_used_by: UtilisΓ© par
701 label_details: DΓ©tails
701 label_details: DΓ©tails
702 label_add_note: Ajouter une note
702 label_add_note: Ajouter une note
703 label_calendar: Calendrier
703 label_calendar: Calendrier
704 label_months_from: mois depuis
704 label_months_from: mois depuis
705 label_gantt: Gantt
705 label_gantt: Gantt
706 label_internal: Interne
706 label_internal: Interne
707 label_last_changes: "%{count} derniers changements"
707 label_last_changes: "%{count} derniers changements"
708 label_change_view_all: Voir tous les changements
708 label_change_view_all: Voir tous les changements
709 label_personalize_page: Personnaliser cette page
709 label_personalize_page: Personnaliser cette page
710 label_comment: Commentaire
710 label_comment: Commentaire
711 label_comment_plural: Commentaires
711 label_comment_plural: Commentaires
712 label_x_comments:
712 label_x_comments:
713 zero: aucun commentaire
713 zero: aucun commentaire
714 one: un commentaire
714 one: un commentaire
715 other: "%{count} commentaires"
715 other: "%{count} commentaires"
716 label_comment_add: Ajouter un commentaire
716 label_comment_add: Ajouter un commentaire
717 label_comment_added: Commentaire ajoutΓ©
717 label_comment_added: Commentaire ajoutΓ©
718 label_comment_delete: Supprimer les commentaires
718 label_comment_delete: Supprimer les commentaires
719 label_query: Rapport personnalisΓ©
719 label_query: Rapport personnalisΓ©
720 label_query_plural: Rapports personnalisΓ©s
720 label_query_plural: Rapports personnalisΓ©s
721 label_query_new: Nouveau rapport
721 label_query_new: Nouveau rapport
722 label_my_queries: Mes rapports personnalisΓ©s
722 label_my_queries: Mes rapports personnalisΓ©s
723 label_filter_add: Ajouter le filtre
723 label_filter_add: Ajouter le filtre
724 label_filter_plural: Filtres
724 label_filter_plural: Filtres
725 label_equals: Γ©gal
725 label_equals: Γ©gal
726 label_not_equals: diffΓ©rent
726 label_not_equals: diffΓ©rent
727 label_in_less_than: dans moins de
727 label_in_less_than: dans moins de
728 label_in_more_than: dans plus de
728 label_in_more_than: dans plus de
729 label_in_the_next_days: dans les prochains jours
729 label_in_the_next_days: dans les prochains jours
730 label_in_the_past_days: dans les derniers jours
730 label_in_the_past_days: dans les derniers jours
731 label_greater_or_equal: '>='
731 label_greater_or_equal: '>='
732 label_less_or_equal: '<='
732 label_less_or_equal: '<='
733 label_between: entre
733 label_between: entre
734 label_in: dans
734 label_in: dans
735 label_today: aujourd'hui
735 label_today: aujourd'hui
736 label_all_time: toute la pΓ©riode
736 label_all_time: toute la pΓ©riode
737 label_yesterday: hier
737 label_yesterday: hier
738 label_this_week: cette semaine
738 label_this_week: cette semaine
739 label_last_week: la semaine dernière
739 label_last_week: la semaine dernière
740 label_last_n_weeks: "les %{count} dernières semaines"
740 label_last_n_weeks: "les %{count} dernières semaines"
741 label_last_n_days: "les %{count} derniers jours"
741 label_last_n_days: "les %{count} derniers jours"
742 label_this_month: ce mois-ci
742 label_this_month: ce mois-ci
743 label_last_month: le mois dernier
743 label_last_month: le mois dernier
744 label_this_year: cette annΓ©e
744 label_this_year: cette annΓ©e
745 label_date_range: PΓ©riode
745 label_date_range: PΓ©riode
746 label_less_than_ago: il y a moins de
746 label_less_than_ago: il y a moins de
747 label_more_than_ago: il y a plus de
747 label_more_than_ago: il y a plus de
748 label_ago: il y a
748 label_ago: il y a
749 label_contains: contient
749 label_contains: contient
750 label_not_contains: ne contient pas
750 label_not_contains: ne contient pas
751 label_any_issues_in_project: une demande du projet
751 label_any_issues_in_project: une demande du projet
752 label_any_issues_not_in_project: une demande hors du projet
752 label_any_issues_not_in_project: une demande hors du projet
753 label_no_issues_in_project: aucune demande du projet
753 label_no_issues_in_project: aucune demande du projet
754 label_any_open_issues: une demande ouverte
754 label_any_open_issues: une demande ouverte
755 label_no_open_issues: aucune demande ouverte
755 label_no_open_issues: aucune demande ouverte
756 label_day_plural: jours
756 label_day_plural: jours
757 label_repository: DΓ©pΓ΄t
757 label_repository: DΓ©pΓ΄t
758 label_repository_new: Nouveau dΓ©pΓ΄t
758 label_repository_new: Nouveau dΓ©pΓ΄t
759 label_repository_plural: DΓ©pΓ΄ts
759 label_repository_plural: DΓ©pΓ΄ts
760 label_browse: Parcourir
760 label_browse: Parcourir
761 label_branch: Branche
761 label_branch: Branche
762 label_tag: Tag
762 label_tag: Tag
763 label_revision: RΓ©vision
763 label_revision: RΓ©vision
764 label_revision_plural: RΓ©visions
764 label_revision_plural: RΓ©visions
765 label_revision_id: "RΓ©vision %{value}"
765 label_revision_id: "RΓ©vision %{value}"
766 label_associated_revisions: RΓ©visions associΓ©es
766 label_associated_revisions: RΓ©visions associΓ©es
767 label_added: ajoutΓ©
767 label_added: ajoutΓ©
768 label_modified: modifiΓ©
768 label_modified: modifiΓ©
769 label_copied: copiΓ©
769 label_copied: copiΓ©
770 label_renamed: renommΓ©
770 label_renamed: renommΓ©
771 label_deleted: supprimΓ©
771 label_deleted: supprimΓ©
772 label_latest_revision: Dernière révision
772 label_latest_revision: Dernière révision
773 label_latest_revision_plural: Dernières révisions
773 label_latest_revision_plural: Dernières révisions
774 label_view_revisions: Voir les rΓ©visions
774 label_view_revisions: Voir les rΓ©visions
775 label_view_all_revisions: Voir toutes les rΓ©visions
775 label_view_all_revisions: Voir toutes les rΓ©visions
776 label_max_size: Taille maximale
776 label_max_size: Taille maximale
777 label_sort_highest: Remonter en premier
777 label_sort_highest: Remonter en premier
778 label_sort_higher: Remonter
778 label_sort_higher: Remonter
779 label_sort_lower: Descendre
779 label_sort_lower: Descendre
780 label_sort_lowest: Descendre en dernier
780 label_sort_lowest: Descendre en dernier
781 label_roadmap: Roadmap
781 label_roadmap: Roadmap
782 label_roadmap_due_in: "Γ‰chΓ©ance dans %{value}"
782 label_roadmap_due_in: "Γ‰chΓ©ance dans %{value}"
783 label_roadmap_overdue: "En retard de %{value}"
783 label_roadmap_overdue: "En retard de %{value}"
784 label_roadmap_no_issues: Aucune demande pour cette version
784 label_roadmap_no_issues: Aucune demande pour cette version
785 label_search: Recherche
785 label_search: Recherche
786 label_result_plural: RΓ©sultats
786 label_result_plural: RΓ©sultats
787 label_all_words: Tous les mots
787 label_all_words: Tous les mots
788 label_wiki: Wiki
788 label_wiki: Wiki
789 label_wiki_edit: RΓ©vision wiki
789 label_wiki_edit: RΓ©vision wiki
790 label_wiki_edit_plural: RΓ©visions wiki
790 label_wiki_edit_plural: RΓ©visions wiki
791 label_wiki_page: Page wiki
791 label_wiki_page: Page wiki
792 label_wiki_page_plural: Pages wiki
792 label_wiki_page_plural: Pages wiki
793 label_wiki_page_new: Nouvelle page wiki
793 label_wiki_page_new: Nouvelle page wiki
794 label_index_by_title: Index par titre
794 label_index_by_title: Index par titre
795 label_index_by_date: Index par date
795 label_index_by_date: Index par date
796 label_current_version: Version actuelle
796 label_current_version: Version actuelle
797 label_preview: PrΓ©visualisation
797 label_preview: PrΓ©visualisation
798 label_feed_plural: Flux Atom
798 label_feed_plural: Flux Atom
799 label_changes_details: DΓ©tails de tous les changements
799 label_changes_details: DΓ©tails de tous les changements
800 label_issue_tracking: Suivi des demandes
800 label_issue_tracking: Suivi des demandes
801 label_spent_time: Temps passΓ©
801 label_spent_time: Temps passΓ©
802 label_total_spent_time: Temps passΓ© total
802 label_total_spent_time: Temps passΓ© total
803 label_overall_spent_time: Temps passΓ© global
803 label_overall_spent_time: Temps passΓ© global
804 label_f_hour: "%{value} heure"
804 label_f_hour: "%{value} heure"
805 label_f_hour_plural: "%{value} heures"
805 label_f_hour_plural: "%{value} heures"
806 label_f_hour_short: "%{value} h"
806 label_f_hour_short: "%{value} h"
807 label_time_tracking: Suivi du temps
807 label_time_tracking: Suivi du temps
808 label_change_plural: Changements
808 label_change_plural: Changements
809 label_statistics: Statistiques
809 label_statistics: Statistiques
810 label_commits_per_month: Commits par mois
810 label_commits_per_month: Commits par mois
811 label_commits_per_author: Commits par auteur
811 label_commits_per_author: Commits par auteur
812 label_diff: diff
812 label_diff: diff
813 label_view_diff: Voir les diffΓ©rences
813 label_view_diff: Voir les diffΓ©rences
814 label_diff_inline: en ligne
814 label_diff_inline: en ligne
815 label_diff_side_by_side: cΓ΄te Γ  cΓ΄te
815 label_diff_side_by_side: cΓ΄te Γ  cΓ΄te
816 label_options: Options
816 label_options: Options
817 label_copy_workflow_from: Copier le workflow de
817 label_copy_workflow_from: Copier le workflow de
818 label_permissions_report: Synthèse des permissions
818 label_permissions_report: Synthèse des permissions
819 label_watched_issues: Demandes surveillΓ©es
819 label_watched_issues: Demandes surveillΓ©es
820 label_related_issues: Demandes liΓ©es
820 label_related_issues: Demandes liΓ©es
821 label_applied_status: Statut appliquΓ©
821 label_applied_status: Statut appliquΓ©
822 label_loading: Chargement...
822 label_loading: Chargement...
823 label_relation_new: Nouvelle relation
823 label_relation_new: Nouvelle relation
824 label_relation_delete: Supprimer la relation
824 label_relation_delete: Supprimer la relation
825 label_relates_to: LiΓ© Γ 
825 label_relates_to: LiΓ© Γ 
826 label_duplicates: Duplique
826 label_duplicates: Duplique
827 label_duplicated_by: DupliquΓ© par
827 label_duplicated_by: DupliquΓ© par
828 label_blocks: Bloque
828 label_blocks: Bloque
829 label_blocked_by: BloquΓ© par
829 label_blocked_by: BloquΓ© par
830 label_precedes: Précède
830 label_precedes: Précède
831 label_follows: Suit
831 label_follows: Suit
832 label_copied_to: CopiΓ© vers
832 label_copied_to: CopiΓ© vers
833 label_copied_from: CopiΓ© depuis
833 label_copied_from: CopiΓ© depuis
834 label_stay_logged_in: Rester connectΓ©
834 label_stay_logged_in: Rester connectΓ©
835 label_disabled: dΓ©sactivΓ©
835 label_disabled: dΓ©sactivΓ©
836 label_show_completed_versions: Voir les versions passΓ©es
836 label_show_completed_versions: Voir les versions passΓ©es
837 label_me: moi
837 label_me: moi
838 label_board: Forum
838 label_board: Forum
839 label_board_new: Nouveau forum
839 label_board_new: Nouveau forum
840 label_board_plural: Forums
840 label_board_plural: Forums
841 label_board_locked: VerrouillΓ©
841 label_board_locked: VerrouillΓ©
842 label_board_sticky: Sticky
842 label_board_sticky: Sticky
843 label_topic_plural: Discussions
843 label_topic_plural: Discussions
844 label_message_plural: Messages
844 label_message_plural: Messages
845 label_message_last: Dernier message
845 label_message_last: Dernier message
846 label_message_new: Nouveau message
846 label_message_new: Nouveau message
847 label_message_posted: Message ajoutΓ©
847 label_message_posted: Message ajoutΓ©
848 label_reply_plural: RΓ©ponses
848 label_reply_plural: RΓ©ponses
849 label_send_information: Envoyer les informations Γ  l'utilisateur
849 label_send_information: Envoyer les informations Γ  l'utilisateur
850 label_year: AnnΓ©e
850 label_year: AnnΓ©e
851 label_month: Mois
851 label_month: Mois
852 label_week: Semaine
852 label_week: Semaine
853 label_date_from: Du
853 label_date_from: Du
854 label_date_to: Au
854 label_date_to: Au
855 label_language_based: BasΓ© sur la langue de l'utilisateur
855 label_language_based: BasΓ© sur la langue de l'utilisateur
856 label_sort_by: "Trier par %{value}"
856 label_sort_by: "Trier par %{value}"
857 label_send_test_email: Envoyer un email de test
857 label_send_test_email: Envoyer un email de test
858 label_feeds_access_key: Clé d'accès Atom
858 label_feeds_access_key: Clé d'accès Atom
859 label_missing_feeds_access_key: Clé d'accès Atom manquante
859 label_missing_feeds_access_key: Clé d'accès Atom manquante
860 label_feeds_access_key_created_on: "Clé d'accès Atom créée il y a %{value}"
860 label_feeds_access_key_created_on: "Clé d'accès Atom créée il y a %{value}"
861 label_module_plural: Modules
861 label_module_plural: Modules
862 label_added_time_by: "AjoutΓ© par %{author} il y a %{age}"
862 label_added_time_by: "AjoutΓ© par %{author} il y a %{age}"
863 label_updated_time_by: "Mis Γ  jour par %{author} il y a %{age}"
863 label_updated_time_by: "Mis Γ  jour par %{author} il y a %{age}"
864 label_updated_time: "Mis Γ  jour il y a %{value}"
864 label_updated_time: "Mis Γ  jour il y a %{value}"
865 label_jump_to_a_project: Aller Γ  un projet...
865 label_jump_to_a_project: Aller Γ  un projet...
866 label_file_plural: Fichiers
866 label_file_plural: Fichiers
867 label_changeset_plural: RΓ©visions
867 label_changeset_plural: RΓ©visions
868 label_default_columns: Colonnes par dΓ©faut
868 label_default_columns: Colonnes par dΓ©faut
869 label_no_change_option: (Pas de changement)
869 label_no_change_option: (Pas de changement)
870 label_bulk_edit_selected_issues: Modifier les demandes sΓ©lectionnΓ©es
870 label_bulk_edit_selected_issues: Modifier les demandes sΓ©lectionnΓ©es
871 label_bulk_edit_selected_time_entries: Modifier les temps passΓ©s sΓ©lectionnΓ©s
871 label_bulk_edit_selected_time_entries: Modifier les temps passΓ©s sΓ©lectionnΓ©s
872 label_theme: Thème
872 label_theme: Thème
873 label_default: DΓ©faut
873 label_default: DΓ©faut
874 label_search_titles_only: Uniquement dans les titres
874 label_search_titles_only: Uniquement dans les titres
875 label_user_mail_option_all: "Pour tous les Γ©vΓ©nements de tous mes projets"
875 label_user_mail_option_all: "Pour tous les Γ©vΓ©nements de tous mes projets"
876 label_user_mail_option_selected: "Pour tous les Γ©vΓ©nements des projets sΓ©lectionnΓ©s..."
876 label_user_mail_option_selected: "Pour tous les Γ©vΓ©nements des projets sΓ©lectionnΓ©s..."
877 label_user_mail_option_none: Aucune notification
877 label_user_mail_option_none: Aucune notification
878 label_user_mail_option_only_my_events: Seulement pour ce que je surveille
878 label_user_mail_option_only_my_events: Seulement pour ce que je surveille
879 label_user_mail_option_only_assigned: Seulement pour ce qui m'est assignΓ©
879 label_user_mail_option_only_assigned: Seulement pour ce qui m'est assignΓ©
880 label_user_mail_option_only_owner: Seulement pour ce que j'ai créé
880 label_user_mail_option_only_owner: Seulement pour ce que j'ai créé
881 label_user_mail_no_self_notified: "Je ne veux pas Γͺtre notifiΓ© des changements que j'effectue"
881 label_user_mail_no_self_notified: "Je ne veux pas Γͺtre notifiΓ© des changements que j'effectue"
882 label_registration_activation_by_email: activation du compte par email
882 label_registration_activation_by_email: activation du compte par email
883 label_registration_manual_activation: activation manuelle du compte
883 label_registration_manual_activation: activation manuelle du compte
884 label_registration_automatic_activation: activation automatique du compte
884 label_registration_automatic_activation: activation automatique du compte
885 label_display_per_page: "Par page : %{value}"
885 label_display_per_page: "Par page : %{value}"
886 label_age: Γ‚ge
886 label_age: Γ‚ge
887 label_change_properties: Changer les propriΓ©tΓ©s
887 label_change_properties: Changer les propriΓ©tΓ©s
888 label_general: GΓ©nΓ©ral
888 label_general: GΓ©nΓ©ral
889 label_more: Plus
889 label_more: Plus
890 label_scm: SCM
890 label_scm: SCM
891 label_plugins: Plugins
891 label_plugins: Plugins
892 label_ldap_authentication: Authentification LDAP
892 label_ldap_authentication: Authentification LDAP
893 label_downloads_abbr: D/L
893 label_downloads_abbr: D/L
894 label_optional_description: Description facultative
894 label_optional_description: Description facultative
895 label_add_another_file: Ajouter un autre fichier
895 label_add_another_file: Ajouter un autre fichier
896 label_preferences: PrΓ©fΓ©rences
896 label_preferences: PrΓ©fΓ©rences
897 label_chronological_order: Dans l'ordre chronologique
897 label_chronological_order: Dans l'ordre chronologique
898 label_reverse_chronological_order: Dans l'ordre chronologique inverse
898 label_reverse_chronological_order: Dans l'ordre chronologique inverse
899 label_planning: Planning
899 label_planning: Planning
900 label_incoming_emails: Emails entrants
900 label_incoming_emails: Emails entrants
901 label_generate_key: GΓ©nΓ©rer une clΓ©
901 label_generate_key: GΓ©nΓ©rer une clΓ©
902 label_issue_watchers: Observateurs
902 label_issue_watchers: Observateurs
903 label_example: Exemple
903 label_example: Exemple
904 label_display: Affichage
904 label_display: Affichage
905 label_sort: Tri
905 label_sort: Tri
906 label_ascending: Croissant
906 label_ascending: Croissant
907 label_descending: DΓ©croissant
907 label_descending: DΓ©croissant
908 label_date_from_to: Du %{start} au %{end}
908 label_date_from_to: Du %{start} au %{end}
909 label_wiki_content_added: Page wiki ajoutΓ©e
909 label_wiki_content_added: Page wiki ajoutΓ©e
910 label_wiki_content_updated: Page wiki mise Γ  jour
910 label_wiki_content_updated: Page wiki mise Γ  jour
911 label_group: Groupe
911 label_group: Groupe
912 label_group_plural: Groupes
912 label_group_plural: Groupes
913 label_group_new: Nouveau groupe
913 label_group_new: Nouveau groupe
914 label_group_anonymous: Utilisateurs anonymes
914 label_group_anonymous: Utilisateurs anonymes
915 label_group_non_member: Utilisateurs non membres
915 label_group_non_member: Utilisateurs non membres
916 label_time_entry_plural: Temps passΓ©
916 label_time_entry_plural: Temps passΓ©
917 label_version_sharing_none: Non partagΓ©
917 label_version_sharing_none: Non partagΓ©
918 label_version_sharing_descendants: Avec les sous-projets
918 label_version_sharing_descendants: Avec les sous-projets
919 label_version_sharing_hierarchy: Avec toute la hiΓ©rarchie
919 label_version_sharing_hierarchy: Avec toute la hiΓ©rarchie
920 label_version_sharing_tree: Avec tout l'arbre
920 label_version_sharing_tree: Avec tout l'arbre
921 label_version_sharing_system: Avec tous les projets
921 label_version_sharing_system: Avec tous les projets
922 label_update_issue_done_ratios: Mettre Γ  jour l'avancement des demandes
922 label_update_issue_done_ratios: Mettre Γ  jour l'avancement des demandes
923 label_copy_source: Source
923 label_copy_source: Source
924 label_copy_target: Cible
924 label_copy_target: Cible
925 label_copy_same_as_target: Comme la cible
925 label_copy_same_as_target: Comme la cible
926 label_display_used_statuses_only: N'afficher que les statuts utilisΓ©s dans ce tracker
926 label_display_used_statuses_only: N'afficher que les statuts utilisΓ©s dans ce tracker
927 label_api_access_key: Clé d'accès API
927 label_api_access_key: Clé d'accès API
928 label_missing_api_access_key: Clé d'accès API manquante
928 label_missing_api_access_key: Clé d'accès API manquante
929 label_api_access_key_created_on: Clé d'accès API créée il y a %{value}
929 label_api_access_key_created_on: Clé d'accès API créée il y a %{value}
930 label_profile: Profil
930 label_profile: Profil
931 label_subtask_plural: Sous-tΓ’ches
931 label_subtask_plural: Sous-tΓ’ches
932 label_project_copy_notifications: Envoyer les notifications durant la copie du projet
932 label_project_copy_notifications: Envoyer les notifications durant la copie du projet
933 label_principal_search: "Rechercher un utilisateur ou un groupe :"
933 label_principal_search: "Rechercher un utilisateur ou un groupe :"
934 label_user_search: "Rechercher un utilisateur :"
934 label_user_search: "Rechercher un utilisateur :"
935 label_additional_workflow_transitions_for_author: Autorisations supplémentaires lorsque l'utilisateur a créé la demande
935 label_additional_workflow_transitions_for_author: Autorisations supplémentaires lorsque l'utilisateur a créé la demande
936 label_additional_workflow_transitions_for_assignee: Autorisations supplΓ©mentaires lorsque la demande est assignΓ©e Γ  l'utilisateur
936 label_additional_workflow_transitions_for_assignee: Autorisations supplΓ©mentaires lorsque la demande est assignΓ©e Γ  l'utilisateur
937 label_issues_visibility_all: Toutes les demandes
937 label_issues_visibility_all: Toutes les demandes
938 label_issues_visibility_public: Toutes les demandes non privΓ©es
938 label_issues_visibility_public: Toutes les demandes non privΓ©es
939 label_issues_visibility_own: Demandes créées par ou assignées à l'utilisateur
939 label_issues_visibility_own: Demandes créées par ou assignées à l'utilisateur
940 label_git_report_last_commit: Afficher le dernier commit des fichiers et rΓ©pertoires
940 label_git_report_last_commit: Afficher le dernier commit des fichiers et rΓ©pertoires
941 label_parent_revision: Parent
941 label_parent_revision: Parent
942 label_child_revision: Enfant
942 label_child_revision: Enfant
943 label_export_options: Options d'exportation %{export_format}
943 label_export_options: Options d'exportation %{export_format}
944 label_copy_attachments: Copier les fichiers
944 label_copy_attachments: Copier les fichiers
945 label_copy_subtasks: Copier les sous-tΓ’ches
945 label_copy_subtasks: Copier les sous-tΓ’ches
946 label_item_position: "%{position} sur %{count}"
946 label_item_position: "%{position} sur %{count}"
947 label_completed_versions: Versions passΓ©es
947 label_completed_versions: Versions passΓ©es
948 label_search_for_watchers: Rechercher des observateurs
948 label_search_for_watchers: Rechercher des observateurs
949 label_session_expiration: Expiration des sessions
949 label_session_expiration: Expiration des sessions
950 label_show_closed_projects: Voir les projets fermΓ©s
950 label_show_closed_projects: Voir les projets fermΓ©s
951 label_status_transitions: Changements de statut
951 label_status_transitions: Changements de statut
952 label_fields_permissions: Permissions sur les champs
952 label_fields_permissions: Permissions sur les champs
953 label_readonly: Lecture
953 label_readonly: Lecture
954 label_required: Obligatoire
954 label_required: Obligatoire
955 label_hidden: CachΓ©
955 label_hidden: CachΓ©
956 label_attribute_of_project: "%{name} du projet"
956 label_attribute_of_project: "%{name} du projet"
957 label_attribute_of_issue: "%{name} de la demande"
957 label_attribute_of_issue: "%{name} de la demande"
958 label_attribute_of_author: "%{name} de l'auteur"
958 label_attribute_of_author: "%{name} de l'auteur"
959 label_attribute_of_assigned_to: "%{name} de l'assignΓ©"
959 label_attribute_of_assigned_to: "%{name} de l'assignΓ©"
960 label_attribute_of_user: "%{name} de l'utilisateur"
960 label_attribute_of_user: "%{name} de l'utilisateur"
961 label_attribute_of_fixed_version: "%{name} de la version cible"
961 label_attribute_of_fixed_version: "%{name} de la version cible"
962 label_attribute_of_object: "%{name} de \"%{object_name}\""
962 label_cross_project_descendants: Avec les sous-projets
963 label_cross_project_descendants: Avec les sous-projets
963 label_cross_project_tree: Avec tout l'arbre
964 label_cross_project_tree: Avec tout l'arbre
964 label_cross_project_hierarchy: Avec toute la hiΓ©rarchie
965 label_cross_project_hierarchy: Avec toute la hiΓ©rarchie
965 label_cross_project_system: Avec tous les projets
966 label_cross_project_system: Avec tous les projets
966 label_gantt_progress_line: Ligne de progression
967 label_gantt_progress_line: Ligne de progression
967 label_visibility_private: par moi uniquement
968 label_visibility_private: par moi uniquement
968 label_visibility_roles: par ces rΓ΄les uniquement
969 label_visibility_roles: par ces rΓ΄les uniquement
969 label_visibility_public: par tout le monde
970 label_visibility_public: par tout le monde
970 label_link: Lien
971 label_link: Lien
971 label_only: seulement
972 label_only: seulement
972 label_drop_down_list: liste dΓ©roulante
973 label_drop_down_list: liste dΓ©roulante
973 label_checkboxes: cases Γ  cocher
974 label_checkboxes: cases Γ  cocher
974 label_radio_buttons: boutons radio
975 label_radio_buttons: boutons radio
975 label_link_values_to: Lier les valeurs vers l'URL
976 label_link_values_to: Lier les valeurs vers l'URL
976 label_custom_field_select_type: Selectionner le type d'objet auquel attacher le champ personnalisΓ©
977 label_custom_field_select_type: Selectionner le type d'objet auquel attacher le champ personnalisΓ©
977 label_check_for_updates: VΓ©rifier les mises Γ  jour
978 label_check_for_updates: VΓ©rifier les mises Γ  jour
978 label_latest_compatible_version: Dernière version compatible
979 label_latest_compatible_version: Dernière version compatible
979 label_unknown_plugin: Plugin inconnu
980 label_unknown_plugin: Plugin inconnu
980 label_add_projects: Ajouter des projets
981 label_add_projects: Ajouter des projets
981 label_users_visibility_all: Tous les utilisateurs actifs
982 label_users_visibility_all: Tous les utilisateurs actifs
982 label_users_visibility_members_of_visible_projects: Membres des projets visibles
983 label_users_visibility_members_of_visible_projects: Membres des projets visibles
983 label_edit_attachments: Modifier les fichiers attachΓ©s
984 label_edit_attachments: Modifier les fichiers attachΓ©s
984 label_link_copied_issue: Lier la demande copiΓ©e
985 label_link_copied_issue: Lier la demande copiΓ©e
985 label_ask: Demander
986 label_ask: Demander
986 label_search_attachments_yes: Rechercher les noms et descriptions de fichiers
987 label_search_attachments_yes: Rechercher les noms et descriptions de fichiers
987 label_search_attachments_no: Ne pas rechercher les fichiers
988 label_search_attachments_no: Ne pas rechercher les fichiers
988 label_search_attachments_only: Rechercher les fichiers uniquement
989 label_search_attachments_only: Rechercher les fichiers uniquement
989 label_search_open_issues_only: Demandes ouvertes uniquement
990 label_search_open_issues_only: Demandes ouvertes uniquement
990 label_email_address_plural: Emails
991 label_email_address_plural: Emails
991 label_email_address_add: Ajouter une adresse email
992 label_email_address_add: Ajouter une adresse email
992 label_enable_notifications: Activer les notifications
993 label_enable_notifications: Activer les notifications
993 label_disable_notifications: DΓ©sactiver les notifications
994 label_disable_notifications: DΓ©sactiver les notifications
994 label_blank_value: non renseignΓ©
995 label_blank_value: non renseignΓ©
995 label_parent_task_attributes: Attributs des tΓ’ches parentes
996 label_parent_task_attributes: Attributs des tΓ’ches parentes
996 label_time_entries_visibility_all: Tous les temps passΓ©s
997 label_time_entries_visibility_all: Tous les temps passΓ©s
997 label_time_entries_visibility_own: Ses propres temps passΓ©s
998 label_time_entries_visibility_own: Ses propres temps passΓ©s
998 label_member_management: Gestion des membres
999 label_member_management: Gestion des membres
999 label_member_management_all_roles: Tous les rΓ΄les
1000 label_member_management_all_roles: Tous les rΓ΄les
1000 label_member_management_selected_roles_only: Ces rΓ΄les uniquement
1001 label_member_management_selected_roles_only: Ces rΓ΄les uniquement
1001 label_import_issues: Importer des demandes
1002 label_import_issues: Importer des demandes
1002 label_select_file_to_import: SΓ©lectionner le fichier Γ  importer
1003 label_select_file_to_import: SΓ©lectionner le fichier Γ  importer
1003 label_fields_separator: SΓ©parateur de champs
1004 label_fields_separator: SΓ©parateur de champs
1004 label_fields_wrapper: DΓ©limiteur de texte
1005 label_fields_wrapper: DΓ©limiteur de texte
1005 label_encoding: Encodage
1006 label_encoding: Encodage
1006 label_comma_char: Virgule
1007 label_comma_char: Virgule
1007 label_semi_colon_char: Point virgule
1008 label_semi_colon_char: Point virgule
1008 label_quote_char: Apostrophe
1009 label_quote_char: Apostrophe
1009 label_double_quote_char: Double apostrophe
1010 label_double_quote_char: Double apostrophe
1010 label_fields_mapping: Correspondance des champs
1011 label_fields_mapping: Correspondance des champs
1011 label_file_content_preview: AperΓ§u du contenu du fichier
1012 label_file_content_preview: AperΓ§u du contenu du fichier
1012 label_create_missing_values: CrΓ©er les valeurs manquantes
1013 label_create_missing_values: CrΓ©er les valeurs manquantes
1013 label_api: API
1014 label_api: API
1014 label_field_format_enumeration: Liste clΓ©/valeur
1015 label_field_format_enumeration: Liste clΓ©/valeur
1015 label_default_values_for_new_users: Valeurs par dΓ©faut pour les nouveaux utilisateurs
1016 label_default_values_for_new_users: Valeurs par dΓ©faut pour les nouveaux utilisateurs
1016 label_relations: Relations
1017 label_relations: Relations
1017 label_new_project_issue_tab_enabled: Afficher l'onglet "Nouvelle demande"
1018 label_new_project_issue_tab_enabled: Afficher l'onglet "Nouvelle demande"
1018 label_new_object_tab_enabled: Afficher le menu dΓ©roulant "+"
1019 label_new_object_tab_enabled: Afficher le menu dΓ©roulant "+"
1019 label_table_of_contents: Contenu
1020 label_table_of_contents: Contenu
1020 label_font_default: Police par dΓ©faut
1021 label_font_default: Police par dΓ©faut
1021 label_font_monospace: Police non proportionnelle
1022 label_font_monospace: Police non proportionnelle
1022 label_font_proportional: Police proportionnelle
1023 label_font_proportional: Police proportionnelle
1023
1024
1024 button_login: Connexion
1025 button_login: Connexion
1025 button_submit: Soumettre
1026 button_submit: Soumettre
1026 button_save: Sauvegarder
1027 button_save: Sauvegarder
1027 button_check_all: Tout cocher
1028 button_check_all: Tout cocher
1028 button_uncheck_all: Tout dΓ©cocher
1029 button_uncheck_all: Tout dΓ©cocher
1029 button_collapse_all: Plier tout
1030 button_collapse_all: Plier tout
1030 button_expand_all: DΓ©plier tout
1031 button_expand_all: DΓ©plier tout
1031 button_delete: Supprimer
1032 button_delete: Supprimer
1032 button_create: CrΓ©er
1033 button_create: CrΓ©er
1033 button_create_and_continue: CrΓ©er et continuer
1034 button_create_and_continue: CrΓ©er et continuer
1034 button_test: Tester
1035 button_test: Tester
1035 button_edit: Modifier
1036 button_edit: Modifier
1036 button_edit_associated_wikipage: "Modifier la page wiki associΓ©e: %{page_title}"
1037 button_edit_associated_wikipage: "Modifier la page wiki associΓ©e: %{page_title}"
1037 button_add: Ajouter
1038 button_add: Ajouter
1038 button_change: Changer
1039 button_change: Changer
1039 button_apply: Appliquer
1040 button_apply: Appliquer
1040 button_clear: Effacer
1041 button_clear: Effacer
1041 button_lock: Verrouiller
1042 button_lock: Verrouiller
1042 button_unlock: DΓ©verrouiller
1043 button_unlock: DΓ©verrouiller
1043 button_download: TΓ©lΓ©charger
1044 button_download: TΓ©lΓ©charger
1044 button_list: Lister
1045 button_list: Lister
1045 button_view: Voir
1046 button_view: Voir
1046 button_move: DΓ©placer
1047 button_move: DΓ©placer
1047 button_move_and_follow: DΓ©placer et suivre
1048 button_move_and_follow: DΓ©placer et suivre
1048 button_back: Retour
1049 button_back: Retour
1049 button_cancel: Annuler
1050 button_cancel: Annuler
1050 button_activate: Activer
1051 button_activate: Activer
1051 button_sort: Trier
1052 button_sort: Trier
1052 button_log_time: Saisir temps
1053 button_log_time: Saisir temps
1053 button_rollback: Revenir Γ  cette version
1054 button_rollback: Revenir Γ  cette version
1054 button_watch: Surveiller
1055 button_watch: Surveiller
1055 button_unwatch: Ne plus surveiller
1056 button_unwatch: Ne plus surveiller
1056 button_reply: RΓ©pondre
1057 button_reply: RΓ©pondre
1057 button_archive: Archiver
1058 button_archive: Archiver
1058 button_unarchive: DΓ©sarchiver
1059 button_unarchive: DΓ©sarchiver
1059 button_reset: RΓ©initialiser
1060 button_reset: RΓ©initialiser
1060 button_rename: Renommer
1061 button_rename: Renommer
1061 button_change_password: Changer de mot de passe
1062 button_change_password: Changer de mot de passe
1062 button_copy: Copier
1063 button_copy: Copier
1063 button_copy_and_follow: Copier et suivre
1064 button_copy_and_follow: Copier et suivre
1064 button_annotate: Annoter
1065 button_annotate: Annoter
1065 button_update: Mettre Γ  jour
1066 button_update: Mettre Γ  jour
1066 button_configure: Configurer
1067 button_configure: Configurer
1067 button_quote: Citer
1068 button_quote: Citer
1068 button_duplicate: Dupliquer
1069 button_duplicate: Dupliquer
1069 button_show: Afficher
1070 button_show: Afficher
1070 button_hide: Cacher
1071 button_hide: Cacher
1071 button_edit_section: Modifier cette section
1072 button_edit_section: Modifier cette section
1072 button_export: Exporter
1073 button_export: Exporter
1073 button_delete_my_account: Supprimer mon compte
1074 button_delete_my_account: Supprimer mon compte
1074 button_close: Fermer
1075 button_close: Fermer
1075 button_reopen: RΓ©ouvrir
1076 button_reopen: RΓ©ouvrir
1076 button_import: Importer
1077 button_import: Importer
1077 button_filter: Filtrer
1078 button_filter: Filtrer
1078
1079
1079 status_active: actif
1080 status_active: actif
1080 status_registered: enregistrΓ©
1081 status_registered: enregistrΓ©
1081 status_locked: verrouillΓ©
1082 status_locked: verrouillΓ©
1082
1083
1083 project_status_active: actif
1084 project_status_active: actif
1084 project_status_closed: fermΓ©
1085 project_status_closed: fermΓ©
1085 project_status_archived: archivΓ©
1086 project_status_archived: archivΓ©
1086
1087
1087 version_status_open: ouvert
1088 version_status_open: ouvert
1088 version_status_locked: verrouillΓ©
1089 version_status_locked: verrouillΓ©
1089 version_status_closed: fermΓ©
1090 version_status_closed: fermΓ©
1090
1091
1091 field_active: Actif
1092 field_active: Actif
1092
1093
1093 text_select_mail_notifications: Actions pour lesquelles une notification par e-mail est envoyΓ©e
1094 text_select_mail_notifications: Actions pour lesquelles une notification par e-mail est envoyΓ©e
1094 text_regexp_info: ex. ^[A-Z0-9]+$
1095 text_regexp_info: ex. ^[A-Z0-9]+$
1095 text_min_max_length_info: 0 pour aucune restriction
1096 text_min_max_length_info: 0 pour aucune restriction
1096 text_project_destroy_confirmation: Êtes-vous sûr de vouloir supprimer ce projet et toutes ses données ?
1097 text_project_destroy_confirmation: Êtes-vous sûr de vouloir supprimer ce projet et toutes ses données ?
1097 text_subprojects_destroy_warning: "Ses sous-projets : %{value} seront Γ©galement supprimΓ©s."
1098 text_subprojects_destroy_warning: "Ses sous-projets : %{value} seront Γ©galement supprimΓ©s."
1098 text_workflow_edit: SΓ©lectionner un tracker et un rΓ΄le pour Γ©diter le workflow
1099 text_workflow_edit: SΓ©lectionner un tracker et un rΓ΄le pour Γ©diter le workflow
1099 text_are_you_sure: Êtes-vous sûr ?
1100 text_are_you_sure: Êtes-vous sûr ?
1100 text_journal_changed: "%{label} changΓ© de %{old} Γ  %{new}"
1101 text_journal_changed: "%{label} changΓ© de %{old} Γ  %{new}"
1101 text_journal_changed_no_detail: "%{label} mis Γ  jour"
1102 text_journal_changed_no_detail: "%{label} mis Γ  jour"
1102 text_journal_set_to: "%{label} mis Γ  %{value}"
1103 text_journal_set_to: "%{label} mis Γ  %{value}"
1103 text_journal_deleted: "%{label} %{old} supprimΓ©"
1104 text_journal_deleted: "%{label} %{old} supprimΓ©"
1104 text_journal_added: "%{label} %{value} ajoutΓ©"
1105 text_journal_added: "%{label} %{value} ajoutΓ©"
1105 text_tip_issue_begin_day: tΓ’che commenΓ§ant ce jour
1106 text_tip_issue_begin_day: tΓ’che commenΓ§ant ce jour
1106 text_tip_issue_end_day: tΓ’che finissant ce jour
1107 text_tip_issue_end_day: tΓ’che finissant ce jour
1107 text_tip_issue_begin_end_day: tΓ’che commenΓ§ant et finissant ce jour
1108 text_tip_issue_begin_end_day: tΓ’che commenΓ§ant et finissant ce jour
1108 text_project_identifier_info: 'Seuls les lettres minuscules (a-z), chiffres, tirets et tirets bas sont autorisΓ©s, doit commencer par une minuscule.<br />Un fois sauvegardΓ©, l''identifiant ne pourra plus Γͺtre modifiΓ©.'
1109 text_project_identifier_info: 'Seuls les lettres minuscules (a-z), chiffres, tirets et tirets bas sont autorisΓ©s, doit commencer par une minuscule.<br />Un fois sauvegardΓ©, l''identifiant ne pourra plus Γͺtre modifiΓ©.'
1109 text_caracters_maximum: "%{count} caractères maximum."
1110 text_caracters_maximum: "%{count} caractères maximum."
1110 text_caracters_minimum: "%{count} caractères minimum."
1111 text_caracters_minimum: "%{count} caractères minimum."
1111 text_length_between: "Longueur comprise entre %{min} et %{max} caractères."
1112 text_length_between: "Longueur comprise entre %{min} et %{max} caractères."
1112 text_tracker_no_workflow: Aucun worflow n'est dΓ©fini pour ce tracker
1113 text_tracker_no_workflow: Aucun worflow n'est dΓ©fini pour ce tracker
1113 text_unallowed_characters: Caractères non autorisés
1114 text_unallowed_characters: Caractères non autorisés
1114 text_comma_separated: Plusieurs valeurs possibles (sΓ©parΓ©es par des virgules).
1115 text_comma_separated: Plusieurs valeurs possibles (sΓ©parΓ©es par des virgules).
1115 text_line_separated: Plusieurs valeurs possibles (une valeur par ligne).
1116 text_line_separated: Plusieurs valeurs possibles (une valeur par ligne).
1116 text_issues_ref_in_commit_messages: RΓ©fΓ©rencement et rΓ©solution des demandes dans les commentaires de commits
1117 text_issues_ref_in_commit_messages: RΓ©fΓ©rencement et rΓ©solution des demandes dans les commentaires de commits
1117 text_issue_added: "La demande %{id} a Γ©tΓ© soumise par %{author}."
1118 text_issue_added: "La demande %{id} a Γ©tΓ© soumise par %{author}."
1118 text_issue_updated: "La demande %{id} a Γ©tΓ© mise Γ  jour par %{author}."
1119 text_issue_updated: "La demande %{id} a Γ©tΓ© mise Γ  jour par %{author}."
1119 text_wiki_destroy_confirmation: Etes-vous sΓ»r de vouloir supprimer ce wiki et tout son contenu ?
1120 text_wiki_destroy_confirmation: Etes-vous sΓ»r de vouloir supprimer ce wiki et tout son contenu ?
1120 text_issue_category_destroy_question: "%{count} demandes sont affectΓ©es Γ  cette catΓ©gorie. Que voulez-vous faire ?"
1121 text_issue_category_destroy_question: "%{count} demandes sont affectΓ©es Γ  cette catΓ©gorie. Que voulez-vous faire ?"
1121 text_issue_category_destroy_assignments: N'affecter les demandes Γ  aucune autre catΓ©gorie
1122 text_issue_category_destroy_assignments: N'affecter les demandes Γ  aucune autre catΓ©gorie
1122 text_issue_category_reassign_to: RΓ©affecter les demandes Γ  cette catΓ©gorie
1123 text_issue_category_reassign_to: RΓ©affecter les demandes Γ  cette catΓ©gorie
1123 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)."
1124 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)."
1124 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Γ©."
1125 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Γ©."
1125 text_load_default_configuration: Charger le paramΓ©trage par dΓ©faut
1126 text_load_default_configuration: Charger le paramΓ©trage par dΓ©faut
1126 text_status_changed_by_changeset: "AppliquΓ© par commit %{value}."
1127 text_status_changed_by_changeset: "AppliquΓ© par commit %{value}."
1127 text_time_logged_by_changeset: "AppliquΓ© par commit %{value}"
1128 text_time_logged_by_changeset: "AppliquΓ© par commit %{value}"
1128 text_issues_destroy_confirmation: 'Êtes-vous sûr de vouloir supprimer la ou les demandes(s) selectionnée(s) ?'
1129 text_issues_destroy_confirmation: 'Êtes-vous sûr de vouloir supprimer la ou les demandes(s) selectionnée(s) ?'
1129 text_issues_destroy_descendants_confirmation: "Cela entrainera Γ©galement la suppression de %{count} sous-tΓ’che(s)."
1130 text_issues_destroy_descendants_confirmation: "Cela entrainera Γ©galement la suppression de %{count} sous-tΓ’che(s)."
1130 text_time_entries_destroy_confirmation: "Etes-vous sΓ»r de vouloir supprimer les temps passΓ©s sΓ©lectionnΓ©s ?"
1131 text_time_entries_destroy_confirmation: "Etes-vous sΓ»r de vouloir supprimer les temps passΓ©s sΓ©lectionnΓ©s ?"
1131 text_select_project_modules: 'SΓ©lectionner les modules Γ  activer pour ce projet :'
1132 text_select_project_modules: 'SΓ©lectionner les modules Γ  activer pour ce projet :'
1132 text_default_administrator_account_changed: Compte administrateur par dΓ©faut changΓ©
1133 text_default_administrator_account_changed: Compte administrateur par dΓ©faut changΓ©
1133 text_file_repository_writable: RΓ©pertoire de stockage des fichiers accessible en Γ©criture
1134 text_file_repository_writable: RΓ©pertoire de stockage des fichiers accessible en Γ©criture
1134 text_plugin_assets_writable: RΓ©pertoire public des plugins accessible en Γ©criture
1135 text_plugin_assets_writable: RΓ©pertoire public des plugins accessible en Γ©criture
1135 text_rmagick_available: Bibliothèque RMagick présente (optionnelle)
1136 text_rmagick_available: Bibliothèque RMagick présente (optionnelle)
1136 text_convert_available: Binaire convert de ImageMagick prΓ©sent (optionel)
1137 text_convert_available: Binaire convert de ImageMagick prΓ©sent (optionel)
1137 text_destroy_time_entries_question: "%{hours} heures ont Γ©tΓ© enregistrΓ©es sur les demandes Γ  supprimer. Que voulez-vous faire ?"
1138 text_destroy_time_entries_question: "%{hours} heures ont Γ©tΓ© enregistrΓ©es sur les demandes Γ  supprimer. Que voulez-vous faire ?"
1138 text_destroy_time_entries: Supprimer les heures
1139 text_destroy_time_entries: Supprimer les heures
1139 text_assign_time_entries_to_project: Reporter les heures sur le projet
1140 text_assign_time_entries_to_project: Reporter les heures sur le projet
1140 text_reassign_time_entries: 'Reporter les heures sur cette demande:'
1141 text_reassign_time_entries: 'Reporter les heures sur cette demande:'
1141 text_user_wrote: "%{value} a Γ©crit :"
1142 text_user_wrote: "%{value} a Γ©crit :"
1142 text_enumeration_destroy_question: "La valeur Β« %{name} Β» est affectΓ©e Γ  %{count} objet(s)."
1143 text_enumeration_destroy_question: "La valeur Β« %{name} Β» est affectΓ©e Γ  %{count} objet(s)."
1143 text_enumeration_category_reassign_to: 'RΓ©affecter les objets Γ  cette valeur:'
1144 text_enumeration_category_reassign_to: 'RΓ©affecter les objets Γ  cette valeur:'
1144 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."
1145 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."
1145 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."
1146 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."
1146 text_diff_truncated: '... Ce diffΓ©rentiel a Γ©tΓ© tronquΓ© car il excΓ¨de la taille maximale pouvant Γͺtre affichΓ©e.'
1147 text_diff_truncated: '... Ce diffΓ©rentiel a Γ©tΓ© tronquΓ© car il excΓ¨de la taille maximale pouvant Γͺtre affichΓ©e.'
1147 text_custom_field_possible_values_info: 'Une ligne par valeur'
1148 text_custom_field_possible_values_info: 'Une ligne par valeur'
1148 text_wiki_page_destroy_question: "Cette page possède %{descendants} sous-page(s) et descendante(s). Que voulez-vous faire ?"
1149 text_wiki_page_destroy_question: "Cette page possède %{descendants} sous-page(s) et descendante(s). Que voulez-vous faire ?"
1149 text_wiki_page_nullify_children: "Conserver les sous-pages en tant que pages racines"
1150 text_wiki_page_nullify_children: "Conserver les sous-pages en tant que pages racines"
1150 text_wiki_page_destroy_children: "Supprimer les sous-pages et toutes leurs descedantes"
1151 text_wiki_page_destroy_children: "Supprimer les sous-pages et toutes leurs descedantes"
1151 text_wiki_page_reassign_children: "RΓ©affecter les sous-pages Γ  cette page"
1152 text_wiki_page_reassign_children: "RΓ©affecter les sous-pages Γ  cette page"
1152 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 ?"
1153 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 ?"
1153 text_zoom_in: Zoom avant
1154 text_zoom_in: Zoom avant
1154 text_zoom_out: Zoom arrière
1155 text_zoom_out: Zoom arrière
1155 text_warn_on_leaving_unsaved: "Cette page contient du texte non sauvegardΓ© qui sera perdu si vous quittez la page."
1156 text_warn_on_leaving_unsaved: "Cette page contient du texte non sauvegardΓ© qui sera perdu si vous quittez la page."
1156 text_scm_path_encoding_note: "DΓ©faut : UTF-8"
1157 text_scm_path_encoding_note: "DΓ©faut : UTF-8"
1157 text_subversion_repository_note: "Exemples (en fonction des protocoles supportΓ©s) : file:///, http://, https://, svn://, svn+[tunnelscheme]://"
1158 text_subversion_repository_note: "Exemples (en fonction des protocoles supportΓ©s) : file:///, http://, https://, svn://, svn+[tunnelscheme]://"
1158 text_git_repository_note: "Chemin vers un dΓ©pΓ΄t vide et local (exemples : /gitrepo, c:\\gitrepo)"
1159 text_git_repository_note: "Chemin vers un dΓ©pΓ΄t vide et local (exemples : /gitrepo, c:\\gitrepo)"
1159 text_mercurial_repository_note: "Chemin vers un dΓ©pΓ΄t local (exemples : /hgrepo, c:\\hgrepo)"
1160 text_mercurial_repository_note: "Chemin vers un dΓ©pΓ΄t local (exemples : /hgrepo, c:\\hgrepo)"
1160 text_scm_command: Commande
1161 text_scm_command: Commande
1161 text_scm_command_version: Version
1162 text_scm_command_version: Version
1162 text_scm_config: Vous pouvez configurer les commandes des SCM dans config/configuration.yml. Redémarrer l'application après modification.
1163 text_scm_config: Vous pouvez configurer les commandes des SCM dans config/configuration.yml. Redémarrer l'application après modification.
1163 text_scm_command_not_available: Ce SCM n'est pas disponible. Vérifier les paramètres dans la section administration.
1164 text_scm_command_not_available: Ce SCM n'est pas disponible. Vérifier les paramètres dans la section administration.
1164 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)"
1165 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)"
1165 text_issue_conflict_resolution_add_notes: "Ajouter mes notes et ignorer mes autres changements"
1166 text_issue_conflict_resolution_add_notes: "Ajouter mes notes et ignorer mes autres changements"
1166 text_issue_conflict_resolution_cancel: "Annuler ma mise Γ  jour et rΓ©afficher %{link}"
1167 text_issue_conflict_resolution_cancel: "Annuler ma mise Γ  jour et rΓ©afficher %{link}"
1167 text_account_destroy_confirmation: "Êtes-vous sûr de vouloir continuer ?\nVotre compte sera définitivement supprimé, sans aucune possibilité de le réactiver."
1168 text_account_destroy_confirmation: "Êtes-vous sûr de vouloir continuer ?\nVotre compte sera définitivement supprimé, sans aucune possibilité de le réactiver."
1168 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."
1169 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."
1169 text_project_closed: Ce projet est fermΓ© et accessible en lecture seule.
1170 text_project_closed: Ce projet est fermΓ© et accessible en lecture seule.
1170 text_turning_multiple_off: "Si vous dΓ©sactivez les valeurs multiples, les valeurs multiples seront supprimΓ©es pour n'en conserver qu'une par objet."
1171 text_turning_multiple_off: "Si vous dΓ©sactivez les valeurs multiples, les valeurs multiples seront supprimΓ©es pour n'en conserver qu'une par objet."
1171
1172
1172 default_role_manager: Manager
1173 default_role_manager: Manager
1173 default_role_developer: DΓ©veloppeur
1174 default_role_developer: DΓ©veloppeur
1174 default_role_reporter: Rapporteur
1175 default_role_reporter: Rapporteur
1175 default_tracker_bug: Anomalie
1176 default_tracker_bug: Anomalie
1176 default_tracker_feature: Evolution
1177 default_tracker_feature: Evolution
1177 default_tracker_support: Assistance
1178 default_tracker_support: Assistance
1178 default_issue_status_new: Nouveau
1179 default_issue_status_new: Nouveau
1179 default_issue_status_in_progress: En cours
1180 default_issue_status_in_progress: En cours
1180 default_issue_status_resolved: RΓ©solu
1181 default_issue_status_resolved: RΓ©solu
1181 default_issue_status_feedback: Commentaire
1182 default_issue_status_feedback: Commentaire
1182 default_issue_status_closed: FermΓ©
1183 default_issue_status_closed: FermΓ©
1183 default_issue_status_rejected: RejetΓ©
1184 default_issue_status_rejected: RejetΓ©
1184 default_doc_category_user: Documentation utilisateur
1185 default_doc_category_user: Documentation utilisateur
1185 default_doc_category_tech: Documentation technique
1186 default_doc_category_tech: Documentation technique
1186 default_priority_low: Bas
1187 default_priority_low: Bas
1187 default_priority_normal: Normal
1188 default_priority_normal: Normal
1188 default_priority_high: Haut
1189 default_priority_high: Haut
1189 default_priority_urgent: Urgent
1190 default_priority_urgent: Urgent
1190 default_priority_immediate: ImmΓ©diat
1191 default_priority_immediate: ImmΓ©diat
1191 default_activity_design: Conception
1192 default_activity_design: Conception
1192 default_activity_development: DΓ©veloppement
1193 default_activity_development: DΓ©veloppement
1193
1194
1194 enumeration_issue_priorities: PrioritΓ©s des demandes
1195 enumeration_issue_priorities: PrioritΓ©s des demandes
1195 enumeration_doc_categories: CatΓ©gories des documents
1196 enumeration_doc_categories: CatΓ©gories des documents
1196 enumeration_activities: ActivitΓ©s (suivi du temps)
1197 enumeration_activities: ActivitΓ©s (suivi du temps)
1197 enumeration_system_activity: Activité système
1198 enumeration_system_activity: Activité système
1198 description_filter: Filtre
1199 description_filter: Filtre
1199 description_search: Champ de recherche
1200 description_search: Champ de recherche
1200 description_choose_project: Projets
1201 description_choose_project: Projets
1201 description_project_scope: Périmètre de recherche
1202 description_project_scope: Périmètre de recherche
1202 description_notes: Notes
1203 description_notes: Notes
1203 description_message_content: Contenu du message
1204 description_message_content: Contenu du message
1204 description_query_sort_criteria_attribute: Critère de tri
1205 description_query_sort_criteria_attribute: Critère de tri
1205 description_query_sort_criteria_direction: Ordre de tri
1206 description_query_sort_criteria_direction: Ordre de tri
1206 description_user_mail_notification: Option de notification
1207 description_user_mail_notification: Option de notification
1207 description_available_columns: Colonnes disponibles
1208 description_available_columns: Colonnes disponibles
1208 description_selected_columns: Colonnes sΓ©lectionnΓ©es
1209 description_selected_columns: Colonnes sΓ©lectionnΓ©es
1209 description_all_columns: Toutes les colonnes
1210 description_all_columns: Toutes les colonnes
1210 description_issue_category_reassign: Choisir une catΓ©gorie
1211 description_issue_category_reassign: Choisir une catΓ©gorie
1211 description_wiki_subpages_reassign: Choisir une nouvelle page parent
1212 description_wiki_subpages_reassign: Choisir une nouvelle page parent
1212 description_date_range_list: Choisir une pΓ©riode prΓ©dΓ©finie
1213 description_date_range_list: Choisir une pΓ©riode prΓ©dΓ©finie
1213 description_date_range_interval: Choisir une pΓ©riode
1214 description_date_range_interval: Choisir une pΓ©riode
1214 description_date_from: Date de dΓ©but
1215 description_date_from: Date de dΓ©but
1215 description_date_to: Date de fin
1216 description_date_to: Date de fin
1216 text_repository_identifier_info: 'Seuls les lettres minuscules (a-z), chiffres, tirets et tirets bas sont autorisΓ©s.<br />Un fois sauvegardΓ©, l''identifiant ne pourra plus Γͺtre modifiΓ©.'
1217 text_repository_identifier_info: 'Seuls les lettres minuscules (a-z), chiffres, tirets et tirets bas sont autorisΓ©s.<br />Un fois sauvegardΓ©, l''identifiant ne pourra plus Γͺtre modifiΓ©.'
1217 label_parent_task_attributes_derived: CalculΓ© Γ  partir des sous-tΓ’ches
1218 label_parent_task_attributes_derived: CalculΓ© Γ  partir des sous-tΓ’ches
1218 label_parent_task_attributes_independent: IndΓ©pendent des sous-tΓ’ches
1219 label_parent_task_attributes_independent: IndΓ©pendent des sous-tΓ’ches
1219 mail_subject_security_notification: Notification de sΓ©curitΓ©
1220 mail_subject_security_notification: Notification de sΓ©curitΓ©
1220 mail_body_security_notification_change: ! '%{field} modifiΓ©(e).'
1221 mail_body_security_notification_change: ! '%{field} modifiΓ©(e).'
1221 mail_body_security_notification_change_to: ! '%{field} changΓ©(e) en %{value}.'
1222 mail_body_security_notification_change_to: ! '%{field} changΓ©(e) en %{value}.'
1222 mail_body_security_notification_add: ! '%{field} %{value} ajoutΓ©(e).'
1223 mail_body_security_notification_add: ! '%{field} %{value} ajoutΓ©(e).'
1223 mail_body_security_notification_remove: ! '%{field} %{value} supprimΓ©(e).'
1224 mail_body_security_notification_remove: ! '%{field} %{value} supprimΓ©(e).'
1224 mail_body_security_notification_notify_enabled: Les notifications ont Γ©tΓ© activΓ©es pour l'adresse %{value}
1225 mail_body_security_notification_notify_enabled: Les notifications ont Γ©tΓ© activΓ©es pour l'adresse %{value}
1225 mail_body_security_notification_notify_disabled: Les notifications ont Γ©tΓ© dΓ©sactivΓ©es pour l'adresse %{value}
1226 mail_body_security_notification_notify_disabled: Les notifications ont Γ©tΓ© dΓ©sactivΓ©es pour l'adresse %{value}
1226 field_remote_ip: Adresse IP
1227 field_remote_ip: Adresse IP
1227 label_no_preview: No preview available
1228 label_no_preview: No preview available
@@ -1,1835 +1,1878
1 # encoding: utf-8
1 # encoding: utf-8
2 #
2 #
3 # Redmine - project management software
3 # Redmine - project management software
4 # Copyright (C) 2006-2016 Jean-Philippe Lang
4 # Copyright (C) 2006-2016 Jean-Philippe Lang
5 #
5 #
6 # This program is free software; you can redistribute it and/or
6 # This program is free software; you can redistribute it and/or
7 # modify it under the terms of the GNU General Public License
7 # modify it under the terms of the GNU General Public License
8 # as published by the Free Software Foundation; either version 2
8 # as published by the Free Software Foundation; either version 2
9 # of the License, or (at your option) any later version.
9 # of the License, or (at your option) any later version.
10 #
10 #
11 # This program is distributed in the hope that it will be useful,
11 # This program is distributed in the hope that it will be useful,
12 # but WITHOUT ANY WARRANTY; without even the implied warranty of
12 # but WITHOUT ANY WARRANTY; without even the implied warranty of
13 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 # GNU General Public License for more details.
14 # GNU General Public License for more details.
15 #
15 #
16 # You should have received a copy of the GNU General Public License
16 # You should have received a copy of the GNU General Public License
17 # along with this program; if not, write to the Free Software
17 # along with this program; if not, write to the Free Software
18 # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
18 # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
19
19
20 require File.expand_path('../../test_helper', __FILE__)
20 require File.expand_path('../../test_helper', __FILE__)
21
21
22 class QueryTest < ActiveSupport::TestCase
22 class QueryTest < ActiveSupport::TestCase
23 include Redmine::I18n
23 include Redmine::I18n
24
24
25 fixtures :projects, :enabled_modules, :users, :members,
25 fixtures :projects, :enabled_modules, :users, :members,
26 :member_roles, :roles, :trackers, :issue_statuses,
26 :member_roles, :roles, :trackers, :issue_statuses,
27 :issue_categories, :enumerations, :issues,
27 :issue_categories, :enumerations, :issues,
28 :watchers, :custom_fields, :custom_values, :versions,
28 :watchers, :custom_fields, :custom_values, :versions,
29 :queries,
29 :queries,
30 :projects_trackers,
30 :projects_trackers,
31 :custom_fields_trackers,
31 :custom_fields_trackers,
32 :workflows
32 :workflows
33
33
34 def setup
34 def setup
35 User.current = nil
35 User.current = nil
36 end
36 end
37
37
38 def test_query_with_roles_visibility_should_validate_roles
38 def test_query_with_roles_visibility_should_validate_roles
39 set_language_if_valid 'en'
39 set_language_if_valid 'en'
40 query = IssueQuery.new(:name => 'Query', :visibility => IssueQuery::VISIBILITY_ROLES)
40 query = IssueQuery.new(:name => 'Query', :visibility => IssueQuery::VISIBILITY_ROLES)
41 assert !query.save
41 assert !query.save
42 assert_include "Roles cannot be blank", query.errors.full_messages
42 assert_include "Roles cannot be blank", query.errors.full_messages
43 query.role_ids = [1, 2]
43 query.role_ids = [1, 2]
44 assert query.save
44 assert query.save
45 end
45 end
46
46
47 def test_changing_roles_visibility_should_clear_roles
47 def test_changing_roles_visibility_should_clear_roles
48 query = IssueQuery.create!(:name => 'Query', :visibility => IssueQuery::VISIBILITY_ROLES, :role_ids => [1, 2])
48 query = IssueQuery.create!(:name => 'Query', :visibility => IssueQuery::VISIBILITY_ROLES, :role_ids => [1, 2])
49 assert_equal 2, query.roles.count
49 assert_equal 2, query.roles.count
50
50
51 query.visibility = IssueQuery::VISIBILITY_PUBLIC
51 query.visibility = IssueQuery::VISIBILITY_PUBLIC
52 query.save!
52 query.save!
53 assert_equal 0, query.roles.count
53 assert_equal 0, query.roles.count
54 end
54 end
55
55
56 def test_available_filters_should_be_ordered
56 def test_available_filters_should_be_ordered
57 set_language_if_valid 'en'
57 set_language_if_valid 'en'
58 query = IssueQuery.new
58 query = IssueQuery.new
59 assert_equal 0, query.available_filters.keys.index('status_id')
59 assert_equal 0, query.available_filters.keys.index('status_id')
60 expected_order = [
60 expected_order = [
61 "Status",
61 "Status",
62 "Project",
62 "Project",
63 "Tracker",
63 "Tracker",
64 "Priority"
64 "Priority"
65 ]
65 ]
66 assert_equal expected_order,
66 assert_equal expected_order,
67 (query.available_filters.values.map{|v| v[:name]} & expected_order)
67 (query.available_filters.values.map{|v| v[:name]} & expected_order)
68 end
68 end
69
69
70 def test_available_filters_with_custom_fields_should_be_ordered
70 def test_available_filters_with_custom_fields_should_be_ordered
71 set_language_if_valid 'en'
71 set_language_if_valid 'en'
72 UserCustomField.create!(
72 UserCustomField.create!(
73 :name => 'order test', :field_format => 'string',
73 :name => 'order test', :field_format => 'string',
74 :is_for_all => true, :is_filter => true
74 :is_for_all => true, :is_filter => true
75 )
75 )
76 query = IssueQuery.new
76 query = IssueQuery.new
77 expected_order = [
77 expected_order = [
78 "Searchable field",
78 "Searchable field",
79 "Database",
79 "Database",
80 "Project's Development status",
80 "Project's Development status",
81 "Author's order test",
81 "Author's order test",
82 "Assignee's order test"
82 "Assignee's order test"
83 ]
83 ]
84 assert_equal expected_order,
84 assert_equal expected_order,
85 (query.available_filters.values.map{|v| v[:name]} & expected_order)
85 (query.available_filters.values.map{|v| v[:name]} & expected_order)
86 end
86 end
87
87
88 def test_custom_fields_for_all_projects_should_be_available_in_global_queries
88 def test_custom_fields_for_all_projects_should_be_available_in_global_queries
89 query = IssueQuery.new(:project => nil, :name => '_')
89 query = IssueQuery.new(:project => nil, :name => '_')
90 assert query.available_filters.has_key?('cf_1')
90 assert query.available_filters.has_key?('cf_1')
91 assert !query.available_filters.has_key?('cf_3')
91 assert !query.available_filters.has_key?('cf_3')
92 end
92 end
93
93
94 def test_system_shared_versions_should_be_available_in_global_queries
94 def test_system_shared_versions_should_be_available_in_global_queries
95 Version.find(2).update_attribute :sharing, 'system'
95 Version.find(2).update_attribute :sharing, 'system'
96 query = IssueQuery.new(:project => nil, :name => '_')
96 query = IssueQuery.new(:project => nil, :name => '_')
97 assert query.available_filters.has_key?('fixed_version_id')
97 assert query.available_filters.has_key?('fixed_version_id')
98 assert query.available_filters['fixed_version_id'][:values].detect {|v| v[1] == '2'}
98 assert query.available_filters['fixed_version_id'][:values].detect {|v| v[1] == '2'}
99 end
99 end
100
100
101 def test_project_filter_in_global_queries
101 def test_project_filter_in_global_queries
102 query = IssueQuery.new(:project => nil, :name => '_')
102 query = IssueQuery.new(:project => nil, :name => '_')
103 project_filter = query.available_filters["project_id"]
103 project_filter = query.available_filters["project_id"]
104 assert_not_nil project_filter
104 assert_not_nil project_filter
105 project_ids = project_filter[:values].map{|p| p[1]}
105 project_ids = project_filter[:values].map{|p| p[1]}
106 assert project_ids.include?("1") #public project
106 assert project_ids.include?("1") #public project
107 assert !project_ids.include?("2") #private project user cannot see
107 assert !project_ids.include?("2") #private project user cannot see
108 end
108 end
109
109
110 def test_available_filters_should_not_include_fields_disabled_on_all_trackers
110 def test_available_filters_should_not_include_fields_disabled_on_all_trackers
111 Tracker.all.each do |tracker|
111 Tracker.all.each do |tracker|
112 tracker.core_fields = Tracker::CORE_FIELDS - ['start_date']
112 tracker.core_fields = Tracker::CORE_FIELDS - ['start_date']
113 tracker.save!
113 tracker.save!
114 end
114 end
115
115
116 query = IssueQuery.new(:name => '_')
116 query = IssueQuery.new(:name => '_')
117 assert_include 'due_date', query.available_filters
117 assert_include 'due_date', query.available_filters
118 assert_not_include 'start_date', query.available_filters
118 assert_not_include 'start_date', query.available_filters
119 end
119 end
120
120
121 def test_filter_values_without_project_should_be_arrays
121 def test_filter_values_without_project_should_be_arrays
122 q = IssueQuery.new
122 q = IssueQuery.new
123 assert_nil q.project
123 assert_nil q.project
124
124
125 q.available_filters.each do |name, filter|
125 q.available_filters.each do |name, filter|
126 values = filter.values
126 values = filter.values
127 assert (values.nil? || values.is_a?(Array)),
127 assert (values.nil? || values.is_a?(Array)),
128 "#values for #{name} filter returned a #{values.class.name}"
128 "#values for #{name} filter returned a #{values.class.name}"
129 end
129 end
130 end
130 end
131
131
132 def test_filter_values_with_project_should_be_arrays
132 def test_filter_values_with_project_should_be_arrays
133 q = IssueQuery.new(:project => Project.find(1))
133 q = IssueQuery.new(:project => Project.find(1))
134 assert_not_nil q.project
134 assert_not_nil q.project
135
135
136 q.available_filters.each do |name, filter|
136 q.available_filters.each do |name, filter|
137 values = filter.values
137 values = filter.values
138 assert (values.nil? || values.is_a?(Array)),
138 assert (values.nil? || values.is_a?(Array)),
139 "#values for #{name} filter returned a #{values.class.name}"
139 "#values for #{name} filter returned a #{values.class.name}"
140 end
140 end
141 end
141 end
142
142
143 def find_issues_with_query(query)
143 def find_issues_with_query(query)
144 Issue.joins(:status, :tracker, :project, :priority).where(
144 Issue.joins(:status, :tracker, :project, :priority).where(
145 query.statement
145 query.statement
146 ).to_a
146 ).to_a
147 end
147 end
148
148
149 def assert_find_issues_with_query_is_successful(query)
149 def assert_find_issues_with_query_is_successful(query)
150 assert_nothing_raised do
150 assert_nothing_raised do
151 find_issues_with_query(query)
151 find_issues_with_query(query)
152 end
152 end
153 end
153 end
154
154
155 def assert_query_statement_includes(query, condition)
155 def assert_query_statement_includes(query, condition)
156 assert_include condition, query.statement
156 assert_include condition, query.statement
157 end
157 end
158
158
159 def assert_query_result(expected, query)
159 def assert_query_result(expected, query)
160 assert_nothing_raised do
160 assert_nothing_raised do
161 assert_equal expected.map(&:id).sort, query.issues.map(&:id).sort
161 assert_equal expected.map(&:id).sort, query.issues.map(&:id).sort
162 assert_equal expected.size, query.issue_count
162 assert_equal expected.size, query.issue_count
163 end
163 end
164 end
164 end
165
165
166 def test_query_should_allow_shared_versions_for_a_project_query
166 def test_query_should_allow_shared_versions_for_a_project_query
167 subproject_version = Version.find(4)
167 subproject_version = Version.find(4)
168 query = IssueQuery.new(:project => Project.find(1), :name => '_')
168 query = IssueQuery.new(:project => Project.find(1), :name => '_')
169 filter = query.available_filters["fixed_version_id"]
169 filter = query.available_filters["fixed_version_id"]
170 assert_not_nil filter
170 assert_not_nil filter
171 assert_include subproject_version.id.to_s, filter[:values].map(&:second)
171 assert_include subproject_version.id.to_s, filter[:values].map(&:second)
172 end
172 end
173
173
174 def test_query_with_multiple_custom_fields
174 def test_query_with_multiple_custom_fields
175 query = IssueQuery.find(1)
175 query = IssueQuery.find(1)
176 assert query.valid?
176 assert query.valid?
177 issues = find_issues_with_query(query)
177 issues = find_issues_with_query(query)
178 assert_equal 1, issues.length
178 assert_equal 1, issues.length
179 assert_equal Issue.find(3), issues.first
179 assert_equal Issue.find(3), issues.first
180 end
180 end
181
181
182 def test_operator_none
182 def test_operator_none
183 query = IssueQuery.new(:project => Project.find(1), :name => '_')
183 query = IssueQuery.new(:project => Project.find(1), :name => '_')
184 query.add_filter('fixed_version_id', '!*', [''])
184 query.add_filter('fixed_version_id', '!*', [''])
185 query.add_filter('cf_1', '!*', [''])
185 query.add_filter('cf_1', '!*', [''])
186 assert query.statement.include?("#{Issue.table_name}.fixed_version_id IS NULL")
186 assert query.statement.include?("#{Issue.table_name}.fixed_version_id IS NULL")
187 assert query.statement.include?("#{CustomValue.table_name}.value IS NULL OR #{CustomValue.table_name}.value = ''")
187 assert query.statement.include?("#{CustomValue.table_name}.value IS NULL OR #{CustomValue.table_name}.value = ''")
188 find_issues_with_query(query)
188 find_issues_with_query(query)
189 end
189 end
190
190
191 def test_operator_none_for_integer
191 def test_operator_none_for_integer
192 query = IssueQuery.new(:project => Project.find(1), :name => '_')
192 query = IssueQuery.new(:project => Project.find(1), :name => '_')
193 query.add_filter('estimated_hours', '!*', [''])
193 query.add_filter('estimated_hours', '!*', [''])
194 issues = find_issues_with_query(query)
194 issues = find_issues_with_query(query)
195 assert !issues.empty?
195 assert !issues.empty?
196 assert issues.all? {|i| !i.estimated_hours}
196 assert issues.all? {|i| !i.estimated_hours}
197 end
197 end
198
198
199 def test_operator_none_for_date
199 def test_operator_none_for_date
200 query = IssueQuery.new(:project => Project.find(1), :name => '_')
200 query = IssueQuery.new(:project => Project.find(1), :name => '_')
201 query.add_filter('start_date', '!*', [''])
201 query.add_filter('start_date', '!*', [''])
202 issues = find_issues_with_query(query)
202 issues = find_issues_with_query(query)
203 assert !issues.empty?
203 assert !issues.empty?
204 assert issues.all? {|i| i.start_date.nil?}
204 assert issues.all? {|i| i.start_date.nil?}
205 end
205 end
206
206
207 def test_operator_none_for_string_custom_field
207 def test_operator_none_for_string_custom_field
208 CustomField.find(2).update_attribute :default_value, ""
208 CustomField.find(2).update_attribute :default_value, ""
209 query = IssueQuery.new(:project => Project.find(1), :name => '_')
209 query = IssueQuery.new(:project => Project.find(1), :name => '_')
210 query.add_filter('cf_2', '!*', [''])
210 query.add_filter('cf_2', '!*', [''])
211 assert query.has_filter?('cf_2')
211 assert query.has_filter?('cf_2')
212 issues = find_issues_with_query(query)
212 issues = find_issues_with_query(query)
213 assert !issues.empty?
213 assert !issues.empty?
214 assert issues.all? {|i| i.custom_field_value(2).blank?}
214 assert issues.all? {|i| i.custom_field_value(2).blank?}
215 end
215 end
216
216
217 def test_operator_all
217 def test_operator_all
218 query = IssueQuery.new(:project => Project.find(1), :name => '_')
218 query = IssueQuery.new(:project => Project.find(1), :name => '_')
219 query.add_filter('fixed_version_id', '*', [''])
219 query.add_filter('fixed_version_id', '*', [''])
220 query.add_filter('cf_1', '*', [''])
220 query.add_filter('cf_1', '*', [''])
221 assert query.statement.include?("#{Issue.table_name}.fixed_version_id IS NOT NULL")
221 assert query.statement.include?("#{Issue.table_name}.fixed_version_id IS NOT NULL")
222 assert query.statement.include?("#{CustomValue.table_name}.value IS NOT NULL AND #{CustomValue.table_name}.value <> ''")
222 assert query.statement.include?("#{CustomValue.table_name}.value IS NOT NULL AND #{CustomValue.table_name}.value <> ''")
223 find_issues_with_query(query)
223 find_issues_with_query(query)
224 end
224 end
225
225
226 def test_operator_all_for_date
226 def test_operator_all_for_date
227 query = IssueQuery.new(:project => Project.find(1), :name => '_')
227 query = IssueQuery.new(:project => Project.find(1), :name => '_')
228 query.add_filter('start_date', '*', [''])
228 query.add_filter('start_date', '*', [''])
229 issues = find_issues_with_query(query)
229 issues = find_issues_with_query(query)
230 assert !issues.empty?
230 assert !issues.empty?
231 assert issues.all? {|i| i.start_date.present?}
231 assert issues.all? {|i| i.start_date.present?}
232 end
232 end
233
233
234 def test_operator_all_for_string_custom_field
234 def test_operator_all_for_string_custom_field
235 query = IssueQuery.new(:project => Project.find(1), :name => '_')
235 query = IssueQuery.new(:project => Project.find(1), :name => '_')
236 query.add_filter('cf_2', '*', [''])
236 query.add_filter('cf_2', '*', [''])
237 assert query.has_filter?('cf_2')
237 assert query.has_filter?('cf_2')
238 issues = find_issues_with_query(query)
238 issues = find_issues_with_query(query)
239 assert !issues.empty?
239 assert !issues.empty?
240 assert issues.all? {|i| i.custom_field_value(2).present?}
240 assert issues.all? {|i| i.custom_field_value(2).present?}
241 end
241 end
242
242
243 def test_numeric_filter_should_not_accept_non_numeric_values
243 def test_numeric_filter_should_not_accept_non_numeric_values
244 query = IssueQuery.new(:name => '_')
244 query = IssueQuery.new(:name => '_')
245 query.add_filter('estimated_hours', '=', ['a'])
245 query.add_filter('estimated_hours', '=', ['a'])
246
246
247 assert query.has_filter?('estimated_hours')
247 assert query.has_filter?('estimated_hours')
248 assert !query.valid?
248 assert !query.valid?
249 end
249 end
250
250
251 def test_operator_is_on_float
251 def test_operator_is_on_float
252 Issue.where(:id => 2).update_all("estimated_hours = 171.2")
252 Issue.where(:id => 2).update_all("estimated_hours = 171.2")
253 query = IssueQuery.new(:name => '_')
253 query = IssueQuery.new(:name => '_')
254 query.add_filter('estimated_hours', '=', ['171.20'])
254 query.add_filter('estimated_hours', '=', ['171.20'])
255 issues = find_issues_with_query(query)
255 issues = find_issues_with_query(query)
256 assert_equal 1, issues.size
256 assert_equal 1, issues.size
257 assert_equal 2, issues.first.id
257 assert_equal 2, issues.first.id
258 end
258 end
259
259
260 def test_operator_is_on_issue_id_should_accept_comma_separated_values
260 def test_operator_is_on_issue_id_should_accept_comma_separated_values
261 query = IssueQuery.new(:name => '_')
261 query = IssueQuery.new(:name => '_')
262 query.add_filter("issue_id", '=', ['1,3'])
262 query.add_filter("issue_id", '=', ['1,3'])
263 issues = find_issues_with_query(query)
263 issues = find_issues_with_query(query)
264 assert_equal 2, issues.size
264 assert_equal 2, issues.size
265 assert_equal [1,3], issues.map(&:id).sort
265 assert_equal [1,3], issues.map(&:id).sort
266 end
266 end
267
267
268 def test_operator_between_on_issue_id_should_return_range
268 def test_operator_between_on_issue_id_should_return_range
269 query = IssueQuery.new(:name => '_')
269 query = IssueQuery.new(:name => '_')
270 query.add_filter("issue_id", '><', ['2','3'])
270 query.add_filter("issue_id", '><', ['2','3'])
271 issues = find_issues_with_query(query)
271 issues = find_issues_with_query(query)
272 assert_equal 2, issues.size
272 assert_equal 2, issues.size
273 assert_equal [2,3], issues.map(&:id).sort
273 assert_equal [2,3], issues.map(&:id).sort
274 end
274 end
275
275
276 def test_operator_is_on_integer_custom_field
276 def test_operator_is_on_integer_custom_field
277 f = IssueCustomField.create!(:name => 'filter', :field_format => 'int', :is_for_all => true, :is_filter => true, :trackers => Tracker.all)
277 f = IssueCustomField.create!(:name => 'filter', :field_format => 'int', :is_for_all => true, :is_filter => true, :trackers => Tracker.all)
278 CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => '7')
278 CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => '7')
279 CustomValue.create!(:custom_field => f, :customized => Issue.find(2), :value => '12')
279 CustomValue.create!(:custom_field => f, :customized => Issue.find(2), :value => '12')
280 CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => '')
280 CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => '')
281
281
282 query = IssueQuery.new(:name => '_')
282 query = IssueQuery.new(:name => '_')
283 query.add_filter("cf_#{f.id}", '=', ['12'])
283 query.add_filter("cf_#{f.id}", '=', ['12'])
284 issues = find_issues_with_query(query)
284 issues = find_issues_with_query(query)
285 assert_equal 1, issues.size
285 assert_equal 1, issues.size
286 assert_equal 2, issues.first.id
286 assert_equal 2, issues.first.id
287 end
287 end
288
288
289 def test_operator_is_on_integer_custom_field_should_accept_negative_value
289 def test_operator_is_on_integer_custom_field_should_accept_negative_value
290 f = IssueCustomField.create!(:name => 'filter', :field_format => 'int', :is_for_all => true, :is_filter => true, :trackers => Tracker.all)
290 f = IssueCustomField.create!(:name => 'filter', :field_format => 'int', :is_for_all => true, :is_filter => true, :trackers => Tracker.all)
291 CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => '7')
291 CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => '7')
292 CustomValue.create!(:custom_field => f, :customized => Issue.find(2), :value => '-12')
292 CustomValue.create!(:custom_field => f, :customized => Issue.find(2), :value => '-12')
293 CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => '')
293 CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => '')
294
294
295 query = IssueQuery.new(:name => '_')
295 query = IssueQuery.new(:name => '_')
296 query.add_filter("cf_#{f.id}", '=', ['-12'])
296 query.add_filter("cf_#{f.id}", '=', ['-12'])
297 assert query.valid?
297 assert query.valid?
298 issues = find_issues_with_query(query)
298 issues = find_issues_with_query(query)
299 assert_equal 1, issues.size
299 assert_equal 1, issues.size
300 assert_equal 2, issues.first.id
300 assert_equal 2, issues.first.id
301 end
301 end
302
302
303 def test_operator_is_on_float_custom_field
303 def test_operator_is_on_float_custom_field
304 f = IssueCustomField.create!(:name => 'filter', :field_format => 'float', :is_filter => true, :is_for_all => true, :trackers => Tracker.all)
304 f = IssueCustomField.create!(:name => 'filter', :field_format => 'float', :is_filter => true, :is_for_all => true, :trackers => Tracker.all)
305 CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => '7.3')
305 CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => '7.3')
306 CustomValue.create!(:custom_field => f, :customized => Issue.find(2), :value => '12.7')
306 CustomValue.create!(:custom_field => f, :customized => Issue.find(2), :value => '12.7')
307 CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => '')
307 CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => '')
308
308
309 query = IssueQuery.new(:name => '_')
309 query = IssueQuery.new(:name => '_')
310 query.add_filter("cf_#{f.id}", '=', ['12.7'])
310 query.add_filter("cf_#{f.id}", '=', ['12.7'])
311 issues = find_issues_with_query(query)
311 issues = find_issues_with_query(query)
312 assert_equal 1, issues.size
312 assert_equal 1, issues.size
313 assert_equal 2, issues.first.id
313 assert_equal 2, issues.first.id
314 end
314 end
315
315
316 def test_operator_is_on_float_custom_field_should_accept_negative_value
316 def test_operator_is_on_float_custom_field_should_accept_negative_value
317 f = IssueCustomField.create!(:name => 'filter', :field_format => 'float', :is_filter => true, :is_for_all => true, :trackers => Tracker.all)
317 f = IssueCustomField.create!(:name => 'filter', :field_format => 'float', :is_filter => true, :is_for_all => true, :trackers => Tracker.all)
318 CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => '7.3')
318 CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => '7.3')
319 CustomValue.create!(:custom_field => f, :customized => Issue.find(2), :value => '-12.7')
319 CustomValue.create!(:custom_field => f, :customized => Issue.find(2), :value => '-12.7')
320 CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => '')
320 CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => '')
321
321
322 query = IssueQuery.new(:name => '_')
322 query = IssueQuery.new(:name => '_')
323 query.add_filter("cf_#{f.id}", '=', ['-12.7'])
323 query.add_filter("cf_#{f.id}", '=', ['-12.7'])
324 assert query.valid?
324 assert query.valid?
325 issues = find_issues_with_query(query)
325 issues = find_issues_with_query(query)
326 assert_equal 1, issues.size
326 assert_equal 1, issues.size
327 assert_equal 2, issues.first.id
327 assert_equal 2, issues.first.id
328 end
328 end
329
329
330 def test_operator_is_on_multi_list_custom_field
330 def test_operator_is_on_multi_list_custom_field
331 f = IssueCustomField.create!(:name => 'filter', :field_format => 'list', :is_filter => true, :is_for_all => true,
331 f = IssueCustomField.create!(:name => 'filter', :field_format => 'list', :is_filter => true, :is_for_all => true,
332 :possible_values => ['value1', 'value2', 'value3'], :multiple => true, :trackers => Tracker.all)
332 :possible_values => ['value1', 'value2', 'value3'], :multiple => true, :trackers => Tracker.all)
333 CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => 'value1')
333 CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => 'value1')
334 CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => 'value2')
334 CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => 'value2')
335 CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => 'value1')
335 CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => 'value1')
336
336
337 query = IssueQuery.new(:name => '_')
337 query = IssueQuery.new(:name => '_')
338 query.add_filter("cf_#{f.id}", '=', ['value1'])
338 query.add_filter("cf_#{f.id}", '=', ['value1'])
339 issues = find_issues_with_query(query)
339 issues = find_issues_with_query(query)
340 assert_equal [1, 3], issues.map(&:id).sort
340 assert_equal [1, 3], issues.map(&:id).sort
341
341
342 query = IssueQuery.new(:name => '_')
342 query = IssueQuery.new(:name => '_')
343 query.add_filter("cf_#{f.id}", '=', ['value2'])
343 query.add_filter("cf_#{f.id}", '=', ['value2'])
344 issues = find_issues_with_query(query)
344 issues = find_issues_with_query(query)
345 assert_equal [1], issues.map(&:id).sort
345 assert_equal [1], issues.map(&:id).sort
346 end
346 end
347
347
348 def test_operator_is_not_on_multi_list_custom_field
348 def test_operator_is_not_on_multi_list_custom_field
349 f = IssueCustomField.create!(:name => 'filter', :field_format => 'list', :is_filter => true, :is_for_all => true,
349 f = IssueCustomField.create!(:name => 'filter', :field_format => 'list', :is_filter => true, :is_for_all => true,
350 :possible_values => ['value1', 'value2', 'value3'], :multiple => true, :trackers => Tracker.all)
350 :possible_values => ['value1', 'value2', 'value3'], :multiple => true, :trackers => Tracker.all)
351 CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => 'value1')
351 CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => 'value1')
352 CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => 'value2')
352 CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => 'value2')
353 CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => 'value1')
353 CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => 'value1')
354
354
355 query = IssueQuery.new(:name => '_')
355 query = IssueQuery.new(:name => '_')
356 query.add_filter("cf_#{f.id}", '!', ['value1'])
356 query.add_filter("cf_#{f.id}", '!', ['value1'])
357 issues = find_issues_with_query(query)
357 issues = find_issues_with_query(query)
358 assert !issues.map(&:id).include?(1)
358 assert !issues.map(&:id).include?(1)
359 assert !issues.map(&:id).include?(3)
359 assert !issues.map(&:id).include?(3)
360
360
361 query = IssueQuery.new(:name => '_')
361 query = IssueQuery.new(:name => '_')
362 query.add_filter("cf_#{f.id}", '!', ['value2'])
362 query.add_filter("cf_#{f.id}", '!', ['value2'])
363 issues = find_issues_with_query(query)
363 issues = find_issues_with_query(query)
364 assert !issues.map(&:id).include?(1)
364 assert !issues.map(&:id).include?(1)
365 assert issues.map(&:id).include?(3)
365 assert issues.map(&:id).include?(3)
366 end
366 end
367
367
368 def test_operator_is_on_string_custom_field_with_utf8_value
368 def test_operator_is_on_string_custom_field_with_utf8_value
369 f = IssueCustomField.create!(:name => 'filter', :field_format => 'string', :is_filter => true, :is_for_all => true, :trackers => Tracker.all)
369 f = IssueCustomField.create!(:name => 'filter', :field_format => 'string', :is_filter => true, :is_for_all => true, :trackers => Tracker.all)
370 CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => 'KiÑ»ƒm')
370 CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => 'KiÑ»ƒm')
371
371
372 query = IssueQuery.new(:name => '_')
372 query = IssueQuery.new(:name => '_')
373 query.add_filter("cf_#{f.id}", '=', ['KiÑ»ƒm'])
373 query.add_filter("cf_#{f.id}", '=', ['KiÑ»ƒm'])
374 issues = find_issues_with_query(query)
374 issues = find_issues_with_query(query)
375 assert_equal [1], issues.map(&:id).sort
375 assert_equal [1], issues.map(&:id).sort
376 end
376 end
377
377
378 def test_operator_is_on_is_private_field
378 def test_operator_is_on_is_private_field
379 # is_private filter only available for those who can set issues private
379 # is_private filter only available for those who can set issues private
380 User.current = User.find(2)
380 User.current = User.find(2)
381
381
382 query = IssueQuery.new(:name => '_')
382 query = IssueQuery.new(:name => '_')
383 assert query.available_filters.key?('is_private')
383 assert query.available_filters.key?('is_private')
384
384
385 query.add_filter("is_private", '=', ['1'])
385 query.add_filter("is_private", '=', ['1'])
386 issues = find_issues_with_query(query)
386 issues = find_issues_with_query(query)
387 assert issues.any?
387 assert issues.any?
388 assert_nil issues.detect {|issue| !issue.is_private?}
388 assert_nil issues.detect {|issue| !issue.is_private?}
389 ensure
389 ensure
390 User.current = nil
390 User.current = nil
391 end
391 end
392
392
393 def test_operator_is_not_on_is_private_field
393 def test_operator_is_not_on_is_private_field
394 # is_private filter only available for those who can set issues private
394 # is_private filter only available for those who can set issues private
395 User.current = User.find(2)
395 User.current = User.find(2)
396
396
397 query = IssueQuery.new(:name => '_')
397 query = IssueQuery.new(:name => '_')
398 assert query.available_filters.key?('is_private')
398 assert query.available_filters.key?('is_private')
399
399
400 query.add_filter("is_private", '!', ['1'])
400 query.add_filter("is_private", '!', ['1'])
401 issues = find_issues_with_query(query)
401 issues = find_issues_with_query(query)
402 assert issues.any?
402 assert issues.any?
403 assert_nil issues.detect {|issue| issue.is_private?}
403 assert_nil issues.detect {|issue| issue.is_private?}
404 ensure
404 ensure
405 User.current = nil
405 User.current = nil
406 end
406 end
407
407
408 def test_operator_greater_than
408 def test_operator_greater_than
409 query = IssueQuery.new(:project => Project.find(1), :name => '_')
409 query = IssueQuery.new(:project => Project.find(1), :name => '_')
410 query.add_filter('done_ratio', '>=', ['40'])
410 query.add_filter('done_ratio', '>=', ['40'])
411 assert query.statement.include?("#{Issue.table_name}.done_ratio >= 40.0")
411 assert query.statement.include?("#{Issue.table_name}.done_ratio >= 40.0")
412 find_issues_with_query(query)
412 find_issues_with_query(query)
413 end
413 end
414
414
415 def test_operator_greater_than_a_float
415 def test_operator_greater_than_a_float
416 query = IssueQuery.new(:project => Project.find(1), :name => '_')
416 query = IssueQuery.new(:project => Project.find(1), :name => '_')
417 query.add_filter('estimated_hours', '>=', ['40.5'])
417 query.add_filter('estimated_hours', '>=', ['40.5'])
418 assert query.statement.include?("#{Issue.table_name}.estimated_hours >= 40.5")
418 assert query.statement.include?("#{Issue.table_name}.estimated_hours >= 40.5")
419 find_issues_with_query(query)
419 find_issues_with_query(query)
420 end
420 end
421
421
422 def test_operator_greater_than_on_int_custom_field
422 def test_operator_greater_than_on_int_custom_field
423 f = IssueCustomField.create!(:name => 'filter', :field_format => 'int', :is_filter => true, :is_for_all => true, :trackers => Tracker.all)
423 f = IssueCustomField.create!(:name => 'filter', :field_format => 'int', :is_filter => true, :is_for_all => true, :trackers => Tracker.all)
424 CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => '7')
424 CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => '7')
425 CustomValue.create!(:custom_field => f, :customized => Issue.find(2), :value => '12')
425 CustomValue.create!(:custom_field => f, :customized => Issue.find(2), :value => '12')
426 CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => '')
426 CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => '')
427
427
428 query = IssueQuery.new(:project => Project.find(1), :name => '_')
428 query = IssueQuery.new(:project => Project.find(1), :name => '_')
429 query.add_filter("cf_#{f.id}", '>=', ['8'])
429 query.add_filter("cf_#{f.id}", '>=', ['8'])
430 issues = find_issues_with_query(query)
430 issues = find_issues_with_query(query)
431 assert_equal 1, issues.size
431 assert_equal 1, issues.size
432 assert_equal 2, issues.first.id
432 assert_equal 2, issues.first.id
433 end
433 end
434
434
435 def test_operator_lesser_than
435 def test_operator_lesser_than
436 query = IssueQuery.new(:project => Project.find(1), :name => '_')
436 query = IssueQuery.new(:project => Project.find(1), :name => '_')
437 query.add_filter('done_ratio', '<=', ['30'])
437 query.add_filter('done_ratio', '<=', ['30'])
438 assert query.statement.include?("#{Issue.table_name}.done_ratio <= 30.0")
438 assert query.statement.include?("#{Issue.table_name}.done_ratio <= 30.0")
439 find_issues_with_query(query)
439 find_issues_with_query(query)
440 end
440 end
441
441
442 def test_operator_lesser_than_on_custom_field
442 def test_operator_lesser_than_on_custom_field
443 f = IssueCustomField.create!(:name => 'filter', :field_format => 'int', :is_filter => true, :is_for_all => true)
443 f = IssueCustomField.create!(:name => 'filter', :field_format => 'int', :is_filter => true, :is_for_all => true)
444 query = IssueQuery.new(:project => Project.find(1), :name => '_')
444 query = IssueQuery.new(:project => Project.find(1), :name => '_')
445 query.add_filter("cf_#{f.id}", '<=', ['30'])
445 query.add_filter("cf_#{f.id}", '<=', ['30'])
446 assert_match /CAST.+ <= 30\.0/, query.statement
446 assert_match /CAST.+ <= 30\.0/, query.statement
447 find_issues_with_query(query)
447 find_issues_with_query(query)
448 end
448 end
449
449
450 def test_operator_lesser_than_on_date_custom_field
450 def test_operator_lesser_than_on_date_custom_field
451 f = IssueCustomField.create!(:name => 'filter', :field_format => 'date', :is_filter => true, :is_for_all => true, :trackers => Tracker.all)
451 f = IssueCustomField.create!(:name => 'filter', :field_format => 'date', :is_filter => true, :is_for_all => true, :trackers => Tracker.all)
452 CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => '2013-04-11')
452 CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => '2013-04-11')
453 CustomValue.create!(:custom_field => f, :customized => Issue.find(2), :value => '2013-05-14')
453 CustomValue.create!(:custom_field => f, :customized => Issue.find(2), :value => '2013-05-14')
454 CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => '')
454 CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => '')
455
455
456 query = IssueQuery.new(:project => Project.find(1), :name => '_')
456 query = IssueQuery.new(:project => Project.find(1), :name => '_')
457 query.add_filter("cf_#{f.id}", '<=', ['2013-05-01'])
457 query.add_filter("cf_#{f.id}", '<=', ['2013-05-01'])
458 issue_ids = find_issues_with_query(query).map(&:id)
458 issue_ids = find_issues_with_query(query).map(&:id)
459 assert_include 1, issue_ids
459 assert_include 1, issue_ids
460 assert_not_include 2, issue_ids
460 assert_not_include 2, issue_ids
461 assert_not_include 3, issue_ids
461 assert_not_include 3, issue_ids
462 end
462 end
463
463
464 def test_operator_between
464 def test_operator_between
465 query = IssueQuery.new(:project => Project.find(1), :name => '_')
465 query = IssueQuery.new(:project => Project.find(1), :name => '_')
466 query.add_filter('done_ratio', '><', ['30', '40'])
466 query.add_filter('done_ratio', '><', ['30', '40'])
467 assert_include "#{Issue.table_name}.done_ratio BETWEEN 30.0 AND 40.0", query.statement
467 assert_include "#{Issue.table_name}.done_ratio BETWEEN 30.0 AND 40.0", query.statement
468 find_issues_with_query(query)
468 find_issues_with_query(query)
469 end
469 end
470
470
471 def test_operator_between_on_custom_field
471 def test_operator_between_on_custom_field
472 f = IssueCustomField.create!(:name => 'filter', :field_format => 'int', :is_filter => true, :is_for_all => true)
472 f = IssueCustomField.create!(:name => 'filter', :field_format => 'int', :is_filter => true, :is_for_all => true)
473 query = IssueQuery.new(:project => Project.find(1), :name => '_')
473 query = IssueQuery.new(:project => Project.find(1), :name => '_')
474 query.add_filter("cf_#{f.id}", '><', ['30', '40'])
474 query.add_filter("cf_#{f.id}", '><', ['30', '40'])
475 assert_match /CAST.+ BETWEEN 30.0 AND 40.0/, query.statement
475 assert_match /CAST.+ BETWEEN 30.0 AND 40.0/, query.statement
476 find_issues_with_query(query)
476 find_issues_with_query(query)
477 end
477 end
478
478
479 def test_date_filter_should_not_accept_non_date_values
479 def test_date_filter_should_not_accept_non_date_values
480 query = IssueQuery.new(:name => '_')
480 query = IssueQuery.new(:name => '_')
481 query.add_filter('created_on', '=', ['a'])
481 query.add_filter('created_on', '=', ['a'])
482
482
483 assert query.has_filter?('created_on')
483 assert query.has_filter?('created_on')
484 assert !query.valid?
484 assert !query.valid?
485 end
485 end
486
486
487 def test_date_filter_should_not_accept_invalid_date_values
487 def test_date_filter_should_not_accept_invalid_date_values
488 query = IssueQuery.new(:name => '_')
488 query = IssueQuery.new(:name => '_')
489 query.add_filter('created_on', '=', ['2011-01-34'])
489 query.add_filter('created_on', '=', ['2011-01-34'])
490
490
491 assert query.has_filter?('created_on')
491 assert query.has_filter?('created_on')
492 assert !query.valid?
492 assert !query.valid?
493 end
493 end
494
494
495 def test_relative_date_filter_should_not_accept_non_integer_values
495 def test_relative_date_filter_should_not_accept_non_integer_values
496 query = IssueQuery.new(:name => '_')
496 query = IssueQuery.new(:name => '_')
497 query.add_filter('created_on', '>t-', ['a'])
497 query.add_filter('created_on', '>t-', ['a'])
498
498
499 assert query.has_filter?('created_on')
499 assert query.has_filter?('created_on')
500 assert !query.valid?
500 assert !query.valid?
501 end
501 end
502
502
503 def test_operator_date_equals
503 def test_operator_date_equals
504 query = IssueQuery.new(:name => '_')
504 query = IssueQuery.new(:name => '_')
505 query.add_filter('due_date', '=', ['2011-07-10'])
505 query.add_filter('due_date', '=', ['2011-07-10'])
506 assert_match /issues\.due_date > '#{quoted_date "2011-07-09"} 23:59:59(\.\d+)?' AND issues\.due_date <= '#{quoted_date "2011-07-10"} 23:59:59(\.\d+)?/,
506 assert_match /issues\.due_date > '#{quoted_date "2011-07-09"} 23:59:59(\.\d+)?' AND issues\.due_date <= '#{quoted_date "2011-07-10"} 23:59:59(\.\d+)?/,
507 query.statement
507 query.statement
508 find_issues_with_query(query)
508 find_issues_with_query(query)
509 end
509 end
510
510
511 def test_operator_date_lesser_than
511 def test_operator_date_lesser_than
512 query = IssueQuery.new(:name => '_')
512 query = IssueQuery.new(:name => '_')
513 query.add_filter('due_date', '<=', ['2011-07-10'])
513 query.add_filter('due_date', '<=', ['2011-07-10'])
514 assert_match /issues\.due_date <= '#{quoted_date "2011-07-10"} 23:59:59(\.\d+)?/, query.statement
514 assert_match /issues\.due_date <= '#{quoted_date "2011-07-10"} 23:59:59(\.\d+)?/, query.statement
515 find_issues_with_query(query)
515 find_issues_with_query(query)
516 end
516 end
517
517
518 def test_operator_date_lesser_than_with_timestamp
518 def test_operator_date_lesser_than_with_timestamp
519 query = IssueQuery.new(:name => '_')
519 query = IssueQuery.new(:name => '_')
520 query.add_filter('updated_on', '<=', ['2011-07-10T19:13:52'])
520 query.add_filter('updated_on', '<=', ['2011-07-10T19:13:52'])
521 assert_match /issues\.updated_on <= '#{quoted_date "2011-07-10"} 19:13:52/, query.statement
521 assert_match /issues\.updated_on <= '#{quoted_date "2011-07-10"} 19:13:52/, query.statement
522 find_issues_with_query(query)
522 find_issues_with_query(query)
523 end
523 end
524
524
525 def test_operator_date_greater_than
525 def test_operator_date_greater_than
526 query = IssueQuery.new(:name => '_')
526 query = IssueQuery.new(:name => '_')
527 query.add_filter('due_date', '>=', ['2011-07-10'])
527 query.add_filter('due_date', '>=', ['2011-07-10'])
528 assert_match /issues\.due_date > '#{quoted_date "2011-07-09"} 23:59:59(\.\d+)?'/, query.statement
528 assert_match /issues\.due_date > '#{quoted_date "2011-07-09"} 23:59:59(\.\d+)?'/, query.statement
529 find_issues_with_query(query)
529 find_issues_with_query(query)
530 end
530 end
531
531
532 def test_operator_date_greater_than_with_timestamp
532 def test_operator_date_greater_than_with_timestamp
533 query = IssueQuery.new(:name => '_')
533 query = IssueQuery.new(:name => '_')
534 query.add_filter('updated_on', '>=', ['2011-07-10T19:13:52'])
534 query.add_filter('updated_on', '>=', ['2011-07-10T19:13:52'])
535 assert_match /issues\.updated_on > '#{quoted_date "2011-07-10"} 19:13:51(\.0+)?'/, query.statement
535 assert_match /issues\.updated_on > '#{quoted_date "2011-07-10"} 19:13:51(\.0+)?'/, query.statement
536 find_issues_with_query(query)
536 find_issues_with_query(query)
537 end
537 end
538
538
539 def test_operator_date_between
539 def test_operator_date_between
540 query = IssueQuery.new(:name => '_')
540 query = IssueQuery.new(:name => '_')
541 query.add_filter('due_date', '><', ['2011-06-23', '2011-07-10'])
541 query.add_filter('due_date', '><', ['2011-06-23', '2011-07-10'])
542 assert_match /issues\.due_date > '#{quoted_date "2011-06-22"} 23:59:59(\.\d+)?' AND issues\.due_date <= '#{quoted_date "2011-07-10"} 23:59:59(\.\d+)?'/,
542 assert_match /issues\.due_date > '#{quoted_date "2011-06-22"} 23:59:59(\.\d+)?' AND issues\.due_date <= '#{quoted_date "2011-07-10"} 23:59:59(\.\d+)?'/,
543 query.statement
543 query.statement
544 find_issues_with_query(query)
544 find_issues_with_query(query)
545 end
545 end
546
546
547 def test_operator_in_more_than
547 def test_operator_in_more_than
548 Issue.find(7).update_attribute(:due_date, (Date.today + 15))
548 Issue.find(7).update_attribute(:due_date, (Date.today + 15))
549 query = IssueQuery.new(:project => Project.find(1), :name => '_')
549 query = IssueQuery.new(:project => Project.find(1), :name => '_')
550 query.add_filter('due_date', '>t+', ['15'])
550 query.add_filter('due_date', '>t+', ['15'])
551 issues = find_issues_with_query(query)
551 issues = find_issues_with_query(query)
552 assert !issues.empty?
552 assert !issues.empty?
553 issues.each {|issue| assert(issue.due_date >= (Date.today + 15))}
553 issues.each {|issue| assert(issue.due_date >= (Date.today + 15))}
554 end
554 end
555
555
556 def test_operator_in_less_than
556 def test_operator_in_less_than
557 query = IssueQuery.new(:project => Project.find(1), :name => '_')
557 query = IssueQuery.new(:project => Project.find(1), :name => '_')
558 query.add_filter('due_date', '<t+', ['15'])
558 query.add_filter('due_date', '<t+', ['15'])
559 issues = find_issues_with_query(query)
559 issues = find_issues_with_query(query)
560 assert !issues.empty?
560 assert !issues.empty?
561 issues.each {|issue| assert(issue.due_date <= (Date.today + 15))}
561 issues.each {|issue| assert(issue.due_date <= (Date.today + 15))}
562 end
562 end
563
563
564 def test_operator_in_the_next_days
564 def test_operator_in_the_next_days
565 query = IssueQuery.new(:project => Project.find(1), :name => '_')
565 query = IssueQuery.new(:project => Project.find(1), :name => '_')
566 query.add_filter('due_date', '><t+', ['15'])
566 query.add_filter('due_date', '><t+', ['15'])
567 issues = find_issues_with_query(query)
567 issues = find_issues_with_query(query)
568 assert !issues.empty?
568 assert !issues.empty?
569 issues.each {|issue| assert(issue.due_date >= Date.today && issue.due_date <= (Date.today + 15))}
569 issues.each {|issue| assert(issue.due_date >= Date.today && issue.due_date <= (Date.today + 15))}
570 end
570 end
571
571
572 def test_operator_less_than_ago
572 def test_operator_less_than_ago
573 Issue.find(7).update_attribute(:due_date, (Date.today - 3))
573 Issue.find(7).update_attribute(:due_date, (Date.today - 3))
574 query = IssueQuery.new(:project => Project.find(1), :name => '_')
574 query = IssueQuery.new(:project => Project.find(1), :name => '_')
575 query.add_filter('due_date', '>t-', ['3'])
575 query.add_filter('due_date', '>t-', ['3'])
576 issues = find_issues_with_query(query)
576 issues = find_issues_with_query(query)
577 assert !issues.empty?
577 assert !issues.empty?
578 issues.each {|issue| assert(issue.due_date >= (Date.today - 3))}
578 issues.each {|issue| assert(issue.due_date >= (Date.today - 3))}
579 end
579 end
580
580
581 def test_operator_in_the_past_days
581 def test_operator_in_the_past_days
582 Issue.find(7).update_attribute(:due_date, (Date.today - 3))
582 Issue.find(7).update_attribute(:due_date, (Date.today - 3))
583 query = IssueQuery.new(:project => Project.find(1), :name => '_')
583 query = IssueQuery.new(:project => Project.find(1), :name => '_')
584 query.add_filter('due_date', '><t-', ['3'])
584 query.add_filter('due_date', '><t-', ['3'])
585 issues = find_issues_with_query(query)
585 issues = find_issues_with_query(query)
586 assert !issues.empty?
586 assert !issues.empty?
587 issues.each {|issue| assert(issue.due_date >= (Date.today - 3) && issue.due_date <= Date.today)}
587 issues.each {|issue| assert(issue.due_date >= (Date.today - 3) && issue.due_date <= Date.today)}
588 end
588 end
589
589
590 def test_operator_more_than_ago
590 def test_operator_more_than_ago
591 Issue.find(7).update_attribute(:due_date, (Date.today - 10))
591 Issue.find(7).update_attribute(:due_date, (Date.today - 10))
592 query = IssueQuery.new(:project => Project.find(1), :name => '_')
592 query = IssueQuery.new(:project => Project.find(1), :name => '_')
593 query.add_filter('due_date', '<t-', ['10'])
593 query.add_filter('due_date', '<t-', ['10'])
594 assert query.statement.include?("#{Issue.table_name}.due_date <=")
594 assert query.statement.include?("#{Issue.table_name}.due_date <=")
595 issues = find_issues_with_query(query)
595 issues = find_issues_with_query(query)
596 assert !issues.empty?
596 assert !issues.empty?
597 issues.each {|issue| assert(issue.due_date <= (Date.today - 10))}
597 issues.each {|issue| assert(issue.due_date <= (Date.today - 10))}
598 end
598 end
599
599
600 def test_operator_in
600 def test_operator_in
601 Issue.find(7).update_attribute(:due_date, (Date.today + 2))
601 Issue.find(7).update_attribute(:due_date, (Date.today + 2))
602 query = IssueQuery.new(:project => Project.find(1), :name => '_')
602 query = IssueQuery.new(:project => Project.find(1), :name => '_')
603 query.add_filter('due_date', 't+', ['2'])
603 query.add_filter('due_date', 't+', ['2'])
604 issues = find_issues_with_query(query)
604 issues = find_issues_with_query(query)
605 assert !issues.empty?
605 assert !issues.empty?
606 issues.each {|issue| assert_equal((Date.today + 2), issue.due_date)}
606 issues.each {|issue| assert_equal((Date.today + 2), issue.due_date)}
607 end
607 end
608
608
609 def test_operator_ago
609 def test_operator_ago
610 Issue.find(7).update_attribute(:due_date, (Date.today - 3))
610 Issue.find(7).update_attribute(:due_date, (Date.today - 3))
611 query = IssueQuery.new(:project => Project.find(1), :name => '_')
611 query = IssueQuery.new(:project => Project.find(1), :name => '_')
612 query.add_filter('due_date', 't-', ['3'])
612 query.add_filter('due_date', 't-', ['3'])
613 issues = find_issues_with_query(query)
613 issues = find_issues_with_query(query)
614 assert !issues.empty?
614 assert !issues.empty?
615 issues.each {|issue| assert_equal((Date.today - 3), issue.due_date)}
615 issues.each {|issue| assert_equal((Date.today - 3), issue.due_date)}
616 end
616 end
617
617
618 def test_operator_today
618 def test_operator_today
619 query = IssueQuery.new(:project => Project.find(1), :name => '_')
619 query = IssueQuery.new(:project => Project.find(1), :name => '_')
620 query.add_filter('due_date', 't', [''])
620 query.add_filter('due_date', 't', [''])
621 issues = find_issues_with_query(query)
621 issues = find_issues_with_query(query)
622 assert !issues.empty?
622 assert !issues.empty?
623 issues.each {|issue| assert_equal Date.today, issue.due_date}
623 issues.each {|issue| assert_equal Date.today, issue.due_date}
624 end
624 end
625
625
626 def test_operator_date_periods
626 def test_operator_date_periods
627 %w(t ld w lw l2w m lm y).each do |operator|
627 %w(t ld w lw l2w m lm y).each do |operator|
628 query = IssueQuery.new(:name => '_')
628 query = IssueQuery.new(:name => '_')
629 query.add_filter('due_date', operator, [''])
629 query.add_filter('due_date', operator, [''])
630 assert query.valid?
630 assert query.valid?
631 assert query.issues
631 assert query.issues
632 end
632 end
633 end
633 end
634
634
635 def test_operator_datetime_periods
635 def test_operator_datetime_periods
636 %w(t ld w lw l2w m lm y).each do |operator|
636 %w(t ld w lw l2w m lm y).each do |operator|
637 query = IssueQuery.new(:name => '_')
637 query = IssueQuery.new(:name => '_')
638 query.add_filter('created_on', operator, [''])
638 query.add_filter('created_on', operator, [''])
639 assert query.valid?
639 assert query.valid?
640 assert query.issues
640 assert query.issues
641 end
641 end
642 end
642 end
643
643
644 def test_operator_contains
644 def test_operator_contains
645 issue = Issue.generate!(:subject => 'AbCdEfG')
645 issue = Issue.generate!(:subject => 'AbCdEfG')
646
646
647 query = IssueQuery.new(:name => '_')
647 query = IssueQuery.new(:name => '_')
648 query.add_filter('subject', '~', ['cdeF'])
648 query.add_filter('subject', '~', ['cdeF'])
649 result = find_issues_with_query(query)
649 result = find_issues_with_query(query)
650 assert_include issue, result
650 assert_include issue, result
651 result.each {|issue| assert issue.subject.downcase.include?('cdef') }
651 result.each {|issue| assert issue.subject.downcase.include?('cdef') }
652 end
652 end
653
653
654 def test_operator_contains_with_utf8_string
654 def test_operator_contains_with_utf8_string
655 issue = Issue.generate!(:subject => 'Subject contains Kiểm')
655 issue = Issue.generate!(:subject => 'Subject contains Kiểm')
656
656
657 query = IssueQuery.new(:name => '_')
657 query = IssueQuery.new(:name => '_')
658 query.add_filter('subject', '~', ['Kiểm'])
658 query.add_filter('subject', '~', ['Kiểm'])
659 result = find_issues_with_query(query)
659 result = find_issues_with_query(query)
660 assert_include issue, result
660 assert_include issue, result
661 assert_equal 1, result.size
661 assert_equal 1, result.size
662 end
662 end
663
663
664 def test_operator_does_not_contain
664 def test_operator_does_not_contain
665 issue = Issue.generate!(:subject => 'AbCdEfG')
665 issue = Issue.generate!(:subject => 'AbCdEfG')
666
666
667 query = IssueQuery.new(:name => '_')
667 query = IssueQuery.new(:name => '_')
668 query.add_filter('subject', '!~', ['cdeF'])
668 query.add_filter('subject', '!~', ['cdeF'])
669 result = find_issues_with_query(query)
669 result = find_issues_with_query(query)
670 assert_not_include issue, result
670 assert_not_include issue, result
671 end
671 end
672
672
673 def test_range_for_this_week_with_week_starting_on_monday
673 def test_range_for_this_week_with_week_starting_on_monday
674 I18n.locale = :fr
674 I18n.locale = :fr
675 assert_equal '1', I18n.t(:general_first_day_of_week)
675 assert_equal '1', I18n.t(:general_first_day_of_week)
676
676
677 Date.stubs(:today).returns(Date.parse('2011-04-29'))
677 Date.stubs(:today).returns(Date.parse('2011-04-29'))
678
678
679 query = IssueQuery.new(:project => Project.find(1), :name => '_')
679 query = IssueQuery.new(:project => Project.find(1), :name => '_')
680 query.add_filter('due_date', 'w', [''])
680 query.add_filter('due_date', 'w', [''])
681 assert_match /issues\.due_date > '#{quoted_date "2011-04-24"} 23:59:59(\.\d+)?' AND issues\.due_date <= '#{quoted_date "2011-05-01"} 23:59:59(\.\d+)?/,
681 assert_match /issues\.due_date > '#{quoted_date "2011-04-24"} 23:59:59(\.\d+)?' AND issues\.due_date <= '#{quoted_date "2011-05-01"} 23:59:59(\.\d+)?/,
682 query.statement
682 query.statement
683 I18n.locale = :en
683 I18n.locale = :en
684 end
684 end
685
685
686 def test_range_for_this_week_with_week_starting_on_sunday
686 def test_range_for_this_week_with_week_starting_on_sunday
687 I18n.locale = :en
687 I18n.locale = :en
688 assert_equal '7', I18n.t(:general_first_day_of_week)
688 assert_equal '7', I18n.t(:general_first_day_of_week)
689
689
690 Date.stubs(:today).returns(Date.parse('2011-04-29'))
690 Date.stubs(:today).returns(Date.parse('2011-04-29'))
691
691
692 query = IssueQuery.new(:project => Project.find(1), :name => '_')
692 query = IssueQuery.new(:project => Project.find(1), :name => '_')
693 query.add_filter('due_date', 'w', [''])
693 query.add_filter('due_date', 'w', [''])
694 assert_match /issues\.due_date > '#{quoted_date "2011-04-23"} 23:59:59(\.\d+)?' AND issues\.due_date <= '#{quoted_date "2011-04-30"} 23:59:59(\.\d+)?/,
694 assert_match /issues\.due_date > '#{quoted_date "2011-04-23"} 23:59:59(\.\d+)?' AND issues\.due_date <= '#{quoted_date "2011-04-30"} 23:59:59(\.\d+)?/,
695 query.statement
695 query.statement
696 end
696 end
697
697
698 def test_filter_assigned_to_me
698 def test_filter_assigned_to_me
699 user = User.find(2)
699 user = User.find(2)
700 group = Group.find(10)
700 group = Group.find(10)
701 group.users << user
701 group.users << user
702 other_group = Group.find(11)
702 other_group = Group.find(11)
703 Member.create!(:project_id => 1, :principal => group, :role_ids => [1])
703 Member.create!(:project_id => 1, :principal => group, :role_ids => [1])
704 Member.create!(:project_id => 1, :principal => other_group, :role_ids => [1])
704 Member.create!(:project_id => 1, :principal => other_group, :role_ids => [1])
705 User.current = user
705 User.current = user
706
706
707 with_settings :issue_group_assignment => '1' do
707 with_settings :issue_group_assignment => '1' do
708 i1 = Issue.generate!(:project_id => 1, :tracker_id => 1, :assigned_to => user)
708 i1 = Issue.generate!(:project_id => 1, :tracker_id => 1, :assigned_to => user)
709 i2 = Issue.generate!(:project_id => 1, :tracker_id => 1, :assigned_to => group)
709 i2 = Issue.generate!(:project_id => 1, :tracker_id => 1, :assigned_to => group)
710 i3 = Issue.generate!(:project_id => 1, :tracker_id => 1, :assigned_to => other_group)
710 i3 = Issue.generate!(:project_id => 1, :tracker_id => 1, :assigned_to => other_group)
711
711
712 query = IssueQuery.new(:name => '_', :filters => { 'assigned_to_id' => {:operator => '=', :values => ['me']}})
712 query = IssueQuery.new(:name => '_', :filters => { 'assigned_to_id' => {:operator => '=', :values => ['me']}})
713 result = query.issues
713 result = query.issues
714 assert_equal Issue.visible.where(:assigned_to_id => ([2] + user.reload.group_ids)).sort_by(&:id), result.sort_by(&:id)
714 assert_equal Issue.visible.where(:assigned_to_id => ([2] + user.reload.group_ids)).sort_by(&:id), result.sort_by(&:id)
715
715
716 assert result.include?(i1)
716 assert result.include?(i1)
717 assert result.include?(i2)
717 assert result.include?(i2)
718 assert !result.include?(i3)
718 assert !result.include?(i3)
719 end
719 end
720 end
720 end
721
721
722 def test_user_custom_field_filtered_on_me
722 def test_user_custom_field_filtered_on_me
723 User.current = User.find(2)
723 User.current = User.find(2)
724 cf = IssueCustomField.create!(:field_format => 'user', :is_for_all => true, :is_filter => true, :name => 'User custom field', :tracker_ids => [1])
724 cf = IssueCustomField.create!(:field_format => 'user', :is_for_all => true, :is_filter => true, :name => 'User custom field', :tracker_ids => [1])
725 issue1 = Issue.create!(:project_id => 1, :tracker_id => 1, :custom_field_values => {cf.id.to_s => '2'}, :subject => 'Test', :author_id => 1)
725 issue1 = Issue.create!(:project_id => 1, :tracker_id => 1, :custom_field_values => {cf.id.to_s => '2'}, :subject => 'Test', :author_id => 1)
726 issue2 = Issue.generate!(:project_id => 1, :tracker_id => 1, :custom_field_values => {cf.id.to_s => '3'})
726 issue2 = Issue.generate!(:project_id => 1, :tracker_id => 1, :custom_field_values => {cf.id.to_s => '3'})
727
727
728 query = IssueQuery.new(:name => '_', :project => Project.find(1))
728 query = IssueQuery.new(:name => '_', :project => Project.find(1))
729 filter = query.available_filters["cf_#{cf.id}"]
729 filter = query.available_filters["cf_#{cf.id}"]
730 assert_not_nil filter
730 assert_not_nil filter
731 assert_include 'me', filter[:values].map{|v| v[1]}
731 assert_include 'me', filter[:values].map{|v| v[1]}
732
732
733 query.filters = { "cf_#{cf.id}" => {:operator => '=', :values => ['me']}}
733 query.filters = { "cf_#{cf.id}" => {:operator => '=', :values => ['me']}}
734 result = query.issues
734 result = query.issues
735 assert_equal 1, result.size
735 assert_equal 1, result.size
736 assert_equal issue1, result.first
736 assert_equal issue1, result.first
737 end
737 end
738
738
739 def test_filter_on_me_by_anonymous_user
739 def test_filter_on_me_by_anonymous_user
740 User.current = nil
740 User.current = nil
741 query = IssueQuery.new(:name => '_', :filters => { 'assigned_to_id' => {:operator => '=', :values => ['me']}})
741 query = IssueQuery.new(:name => '_', :filters => { 'assigned_to_id' => {:operator => '=', :values => ['me']}})
742 assert_equal [], query.issues
742 assert_equal [], query.issues
743 end
743 end
744
744
745 def test_filter_my_projects
745 def test_filter_my_projects
746 User.current = User.find(2)
746 User.current = User.find(2)
747 query = IssueQuery.new(:name => '_')
747 query = IssueQuery.new(:name => '_')
748 filter = query.available_filters['project_id']
748 filter = query.available_filters['project_id']
749 assert_not_nil filter
749 assert_not_nil filter
750 assert_include 'mine', filter[:values].map{|v| v[1]}
750 assert_include 'mine', filter[:values].map{|v| v[1]}
751
751
752 query.filters = { 'project_id' => {:operator => '=', :values => ['mine']}}
752 query.filters = { 'project_id' => {:operator => '=', :values => ['mine']}}
753 result = query.issues
753 result = query.issues
754 assert_nil result.detect {|issue| !User.current.member_of?(issue.project)}
754 assert_nil result.detect {|issue| !User.current.member_of?(issue.project)}
755 end
755 end
756
756
757 def test_filter_watched_issues
757 def test_filter_watched_issues
758 User.current = User.find(1)
758 User.current = User.find(1)
759 query = IssueQuery.new(:name => '_', :filters => { 'watcher_id' => {:operator => '=', :values => ['me']}})
759 query = IssueQuery.new(:name => '_', :filters => { 'watcher_id' => {:operator => '=', :values => ['me']}})
760 result = find_issues_with_query(query)
760 result = find_issues_with_query(query)
761 assert_not_nil result
761 assert_not_nil result
762 assert !result.empty?
762 assert !result.empty?
763 assert_equal Issue.visible.watched_by(User.current).sort_by(&:id), result.sort_by(&:id)
763 assert_equal Issue.visible.watched_by(User.current).sort_by(&:id), result.sort_by(&:id)
764 User.current = nil
764 User.current = nil
765 end
765 end
766
766
767 def test_filter_unwatched_issues
767 def test_filter_unwatched_issues
768 User.current = User.find(1)
768 User.current = User.find(1)
769 query = IssueQuery.new(:name => '_', :filters => { 'watcher_id' => {:operator => '!', :values => ['me']}})
769 query = IssueQuery.new(:name => '_', :filters => { 'watcher_id' => {:operator => '!', :values => ['me']}})
770 result = find_issues_with_query(query)
770 result = find_issues_with_query(query)
771 assert_not_nil result
771 assert_not_nil result
772 assert !result.empty?
772 assert !result.empty?
773 assert_equal((Issue.visible - Issue.watched_by(User.current)).sort_by(&:id).size, result.sort_by(&:id).size)
773 assert_equal((Issue.visible - Issue.watched_by(User.current)).sort_by(&:id).size, result.sort_by(&:id).size)
774 User.current = nil
774 User.current = nil
775 end
775 end
776
776
777 def test_filter_on_custom_field_should_ignore_projects_with_field_disabled
777 def test_filter_on_custom_field_should_ignore_projects_with_field_disabled
778 field = IssueCustomField.generate!(:trackers => Tracker.all, :project_ids => [1, 3, 4], :is_for_all => false, :is_filter => true)
778 field = IssueCustomField.generate!(:trackers => Tracker.all, :project_ids => [1, 3, 4], :is_for_all => false, :is_filter => true)
779 Issue.generate!(:project_id => 3, :tracker_id => 2, :custom_field_values => {field.id.to_s => 'Foo'})
779 Issue.generate!(:project_id => 3, :tracker_id => 2, :custom_field_values => {field.id.to_s => 'Foo'})
780 Issue.generate!(:project_id => 4, :tracker_id => 2, :custom_field_values => {field.id.to_s => 'Foo'})
780 Issue.generate!(:project_id => 4, :tracker_id => 2, :custom_field_values => {field.id.to_s => 'Foo'})
781
781
782 query = IssueQuery.new(:name => '_', :project => Project.find(1))
782 query = IssueQuery.new(:name => '_', :project => Project.find(1))
783 query.filters = {"cf_#{field.id}" => {:operator => '=', :values => ['Foo']}}
783 query.filters = {"cf_#{field.id}" => {:operator => '=', :values => ['Foo']}}
784 assert_equal 2, find_issues_with_query(query).size
784 assert_equal 2, find_issues_with_query(query).size
785
785
786 field.project_ids = [1, 3] # Disable the field for project 4
786 field.project_ids = [1, 3] # Disable the field for project 4
787 field.save!
787 field.save!
788 assert_equal 1, find_issues_with_query(query).size
788 assert_equal 1, find_issues_with_query(query).size
789 end
789 end
790
790
791 def test_filter_on_custom_field_should_ignore_trackers_with_field_disabled
791 def test_filter_on_custom_field_should_ignore_trackers_with_field_disabled
792 field = IssueCustomField.generate!(:tracker_ids => [1, 2], :is_for_all => true, :is_filter => true)
792 field = IssueCustomField.generate!(:tracker_ids => [1, 2], :is_for_all => true, :is_filter => true)
793 Issue.generate!(:project_id => 1, :tracker_id => 1, :custom_field_values => {field.id.to_s => 'Foo'})
793 Issue.generate!(:project_id => 1, :tracker_id => 1, :custom_field_values => {field.id.to_s => 'Foo'})
794 Issue.generate!(:project_id => 1, :tracker_id => 2, :custom_field_values => {field.id.to_s => 'Foo'})
794 Issue.generate!(:project_id => 1, :tracker_id => 2, :custom_field_values => {field.id.to_s => 'Foo'})
795
795
796 query = IssueQuery.new(:name => '_', :project => Project.find(1))
796 query = IssueQuery.new(:name => '_', :project => Project.find(1))
797 query.filters = {"cf_#{field.id}" => {:operator => '=', :values => ['Foo']}}
797 query.filters = {"cf_#{field.id}" => {:operator => '=', :values => ['Foo']}}
798 assert_equal 2, find_issues_with_query(query).size
798 assert_equal 2, find_issues_with_query(query).size
799
799
800 field.tracker_ids = [1] # Disable the field for tracker 2
800 field.tracker_ids = [1] # Disable the field for tracker 2
801 field.save!
801 field.save!
802 assert_equal 1, find_issues_with_query(query).size
802 assert_equal 1, find_issues_with_query(query).size
803 end
803 end
804
804
805 def test_filter_on_project_custom_field
805 def test_filter_on_project_custom_field
806 field = ProjectCustomField.create!(:name => 'Client', :is_filter => true, :field_format => 'string')
806 field = ProjectCustomField.create!(:name => 'Client', :is_filter => true, :field_format => 'string')
807 CustomValue.create!(:custom_field => field, :customized => Project.find(3), :value => 'Foo')
807 CustomValue.create!(:custom_field => field, :customized => Project.find(3), :value => 'Foo')
808 CustomValue.create!(:custom_field => field, :customized => Project.find(5), :value => 'Foo')
808 CustomValue.create!(:custom_field => field, :customized => Project.find(5), :value => 'Foo')
809
809
810 query = IssueQuery.new(:name => '_')
810 query = IssueQuery.new(:name => '_')
811 filter_name = "project.cf_#{field.id}"
811 filter_name = "project.cf_#{field.id}"
812 assert_include filter_name, query.available_filters.keys
812 assert_include filter_name, query.available_filters.keys
813 query.filters = {filter_name => {:operator => '=', :values => ['Foo']}}
813 query.filters = {filter_name => {:operator => '=', :values => ['Foo']}}
814 assert_equal [3, 5], find_issues_with_query(query).map(&:project_id).uniq.sort
814 assert_equal [3, 5], find_issues_with_query(query).map(&:project_id).uniq.sort
815 end
815 end
816
816
817 def test_filter_on_author_custom_field
817 def test_filter_on_author_custom_field
818 field = UserCustomField.create!(:name => 'Client', :is_filter => true, :field_format => 'string')
818 field = UserCustomField.create!(:name => 'Client', :is_filter => true, :field_format => 'string')
819 CustomValue.create!(:custom_field => field, :customized => User.find(3), :value => 'Foo')
819 CustomValue.create!(:custom_field => field, :customized => User.find(3), :value => 'Foo')
820
820
821 query = IssueQuery.new(:name => '_')
821 query = IssueQuery.new(:name => '_')
822 filter_name = "author.cf_#{field.id}"
822 filter_name = "author.cf_#{field.id}"
823 assert_include filter_name, query.available_filters.keys
823 assert_include filter_name, query.available_filters.keys
824 query.filters = {filter_name => {:operator => '=', :values => ['Foo']}}
824 query.filters = {filter_name => {:operator => '=', :values => ['Foo']}}
825 assert_equal [3], find_issues_with_query(query).map(&:author_id).uniq.sort
825 assert_equal [3], find_issues_with_query(query).map(&:author_id).uniq.sort
826 end
826 end
827
827
828 def test_filter_on_assigned_to_custom_field
828 def test_filter_on_assigned_to_custom_field
829 field = UserCustomField.create!(:name => 'Client', :is_filter => true, :field_format => 'string')
829 field = UserCustomField.create!(:name => 'Client', :is_filter => true, :field_format => 'string')
830 CustomValue.create!(:custom_field => field, :customized => User.find(3), :value => 'Foo')
830 CustomValue.create!(:custom_field => field, :customized => User.find(3), :value => 'Foo')
831
831
832 query = IssueQuery.new(:name => '_')
832 query = IssueQuery.new(:name => '_')
833 filter_name = "assigned_to.cf_#{field.id}"
833 filter_name = "assigned_to.cf_#{field.id}"
834 assert_include filter_name, query.available_filters.keys
834 assert_include filter_name, query.available_filters.keys
835 query.filters = {filter_name => {:operator => '=', :values => ['Foo']}}
835 query.filters = {filter_name => {:operator => '=', :values => ['Foo']}}
836 assert_equal [3], find_issues_with_query(query).map(&:assigned_to_id).uniq.sort
836 assert_equal [3], find_issues_with_query(query).map(&:assigned_to_id).uniq.sort
837 end
837 end
838
838
839 def test_filter_on_fixed_version_custom_field
839 def test_filter_on_fixed_version_custom_field
840 field = VersionCustomField.create!(:name => 'Client', :is_filter => true, :field_format => 'string')
840 field = VersionCustomField.create!(:name => 'Client', :is_filter => true, :field_format => 'string')
841 CustomValue.create!(:custom_field => field, :customized => Version.find(2), :value => 'Foo')
841 CustomValue.create!(:custom_field => field, :customized => Version.find(2), :value => 'Foo')
842
842
843 query = IssueQuery.new(:name => '_')
843 query = IssueQuery.new(:name => '_')
844 filter_name = "fixed_version.cf_#{field.id}"
844 filter_name = "fixed_version.cf_#{field.id}"
845 assert_include filter_name, query.available_filters.keys
845 assert_include filter_name, query.available_filters.keys
846 query.filters = {filter_name => {:operator => '=', :values => ['Foo']}}
846 query.filters = {filter_name => {:operator => '=', :values => ['Foo']}}
847 assert_equal [2], find_issues_with_query(query).map(&:fixed_version_id).uniq.sort
847 assert_equal [2], find_issues_with_query(query).map(&:fixed_version_id).uniq.sort
848 end
848 end
849
849
850 def test_filter_on_fixed_version_due_date
850 def test_filter_on_fixed_version_due_date
851 query = IssueQuery.new(:name => '_')
851 query = IssueQuery.new(:name => '_')
852 filter_name = "fixed_version.due_date"
852 filter_name = "fixed_version.due_date"
853 assert_include filter_name, query.available_filters.keys
853 assert_include filter_name, query.available_filters.keys
854 query.filters = {filter_name => {:operator => '=', :values => [20.day.from_now.to_date.to_s(:db)]}}
854 query.filters = {filter_name => {:operator => '=', :values => [20.day.from_now.to_date.to_s(:db)]}}
855 issues = find_issues_with_query(query)
855 issues = find_issues_with_query(query)
856 assert_equal [2], issues.map(&:fixed_version_id).uniq.sort
856 assert_equal [2], issues.map(&:fixed_version_id).uniq.sort
857 assert_equal [2, 12], issues.map(&:id).sort
857 assert_equal [2, 12], issues.map(&:id).sort
858
858
859 query = IssueQuery.new(:name => '_')
859 query = IssueQuery.new(:name => '_')
860 query.filters = {filter_name => {:operator => '>=', :values => [21.day.from_now.to_date.to_s(:db)]}}
860 query.filters = {filter_name => {:operator => '>=', :values => [21.day.from_now.to_date.to_s(:db)]}}
861 assert_equal 0, find_issues_with_query(query).size
861 assert_equal 0, find_issues_with_query(query).size
862 end
862 end
863
863
864 def test_filter_on_fixed_version_status
864 def test_filter_on_fixed_version_status
865 query = IssueQuery.new(:name => '_')
865 query = IssueQuery.new(:name => '_')
866 filter_name = "fixed_version.status"
866 filter_name = "fixed_version.status"
867 assert_include filter_name, query.available_filters.keys
867 assert_include filter_name, query.available_filters.keys
868 query.filters = {filter_name => {:operator => '=', :values => ['closed']}}
868 query.filters = {filter_name => {:operator => '=', :values => ['closed']}}
869 issues = find_issues_with_query(query)
869 issues = find_issues_with_query(query)
870
870
871 assert_equal [1], issues.map(&:fixed_version_id).sort
871 assert_equal [1], issues.map(&:fixed_version_id).sort
872 assert_equal [11], issues.map(&:id).sort
872 assert_equal [11], issues.map(&:id).sort
873
873
874 # "is not" operator should include issues without target version
874 # "is not" operator should include issues without target version
875 query = IssueQuery.new(:name => '_')
875 query = IssueQuery.new(:name => '_')
876 query.filters = {filter_name => {:operator => '!', :values => ['open', 'closed', 'locked']}, "project_id" => {:operator => '=', :values => [1]}}
876 query.filters = {filter_name => {:operator => '!', :values => ['open', 'closed', 'locked']}, "project_id" => {:operator => '=', :values => [1]}}
877 assert_equal [1, 3, 7, 8], find_issues_with_query(query).map(&:id).uniq.sort
877 assert_equal [1, 3, 7, 8], find_issues_with_query(query).map(&:id).uniq.sort
878 end
878 end
879
879
880 def test_filter_on_version_custom_field
881 field = IssueCustomField.generate!(:field_format => 'version', :is_filter => true)
882 issue = Issue.generate!(:project_id => 1, :tracker_id => 1, :custom_field_values => {field.id.to_s => '2'})
883
884 query = IssueQuery.new(:name => '_')
885 filter_name = "cf_#{field.id}"
886 assert_include filter_name, query.available_filters.keys
887
888 query.filters = {filter_name => {:operator => '=', :values => ['2']}}
889 issues = find_issues_with_query(query)
890 assert_equal [issue.id], issues.map(&:id).sort
891 end
892
893 def test_filter_on_attribute_of_version_custom_field
894 field = IssueCustomField.generate!(:field_format => 'version', :is_filter => true)
895 version = Version.generate!(:effective_date => '2017-01-14')
896 issue = Issue.generate!(:project_id => 1, :tracker_id => 1, :custom_field_values => {field.id.to_s => version.id.to_s})
897
898 query = IssueQuery.new(:name => '_')
899 filter_name = "cf_#{field.id}.due_date"
900 assert_include filter_name, query.available_filters.keys
901
902 query.filters = {filter_name => {:operator => '=', :values => ['2017-01-14']}}
903 issues = find_issues_with_query(query)
904 assert_equal [issue.id], issues.map(&:id).sort
905 end
906
907 def test_filter_on_custom_field_of_version_custom_field
908 field = IssueCustomField.generate!(:field_format => 'version', :is_filter => true)
909 attr = VersionCustomField.generate!(:field_format => 'string', :is_filter => true)
910
911 version = Version.generate!(:custom_field_values => {attr.id.to_s => 'ABC'})
912 issue = Issue.generate!(:project_id => 1, :tracker_id => 1, :custom_field_values => {field.id.to_s => version.id.to_s})
913
914 query = IssueQuery.new(:name => '_')
915 filter_name = "cf_#{field.id}.cf_#{attr.id}"
916 assert_include filter_name, query.available_filters.keys
917
918 query.filters = {filter_name => {:operator => '=', :values => ['ABC']}}
919 issues = find_issues_with_query(query)
920 assert_equal [issue.id], issues.map(&:id).sort
921 end
922
880 def test_filter_on_relations_with_a_specific_issue
923 def test_filter_on_relations_with_a_specific_issue
881 IssueRelation.delete_all
924 IssueRelation.delete_all
882 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Issue.find(2))
925 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Issue.find(2))
883 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(3), :issue_to => Issue.find(1))
926 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(3), :issue_to => Issue.find(1))
884
927
885 query = IssueQuery.new(:name => '_')
928 query = IssueQuery.new(:name => '_')
886 query.filters = {"relates" => {:operator => '=', :values => ['1']}}
929 query.filters = {"relates" => {:operator => '=', :values => ['1']}}
887 assert_equal [2, 3], find_issues_with_query(query).map(&:id).sort
930 assert_equal [2, 3], find_issues_with_query(query).map(&:id).sort
888
931
889 query = IssueQuery.new(:name => '_')
932 query = IssueQuery.new(:name => '_')
890 query.filters = {"relates" => {:operator => '=', :values => ['2']}}
933 query.filters = {"relates" => {:operator => '=', :values => ['2']}}
891 assert_equal [1], find_issues_with_query(query).map(&:id).sort
934 assert_equal [1], find_issues_with_query(query).map(&:id).sort
892 end
935 end
893
936
894 def test_filter_on_relations_with_any_issues_in_a_project
937 def test_filter_on_relations_with_any_issues_in_a_project
895 IssueRelation.delete_all
938 IssueRelation.delete_all
896 with_settings :cross_project_issue_relations => '1' do
939 with_settings :cross_project_issue_relations => '1' do
897 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Project.find(2).issues.first)
940 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Project.find(2).issues.first)
898 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(2), :issue_to => Project.find(2).issues.first)
941 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(2), :issue_to => Project.find(2).issues.first)
899 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Project.find(3).issues.first)
942 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Project.find(3).issues.first)
900 end
943 end
901
944
902 query = IssueQuery.new(:name => '_')
945 query = IssueQuery.new(:name => '_')
903 query.filters = {"relates" => {:operator => '=p', :values => ['2']}}
946 query.filters = {"relates" => {:operator => '=p', :values => ['2']}}
904 assert_equal [1, 2], find_issues_with_query(query).map(&:id).sort
947 assert_equal [1, 2], find_issues_with_query(query).map(&:id).sort
905
948
906 query = IssueQuery.new(:name => '_')
949 query = IssueQuery.new(:name => '_')
907 query.filters = {"relates" => {:operator => '=p', :values => ['3']}}
950 query.filters = {"relates" => {:operator => '=p', :values => ['3']}}
908 assert_equal [1], find_issues_with_query(query).map(&:id).sort
951 assert_equal [1], find_issues_with_query(query).map(&:id).sort
909
952
910 query = IssueQuery.new(:name => '_')
953 query = IssueQuery.new(:name => '_')
911 query.filters = {"relates" => {:operator => '=p', :values => ['4']}}
954 query.filters = {"relates" => {:operator => '=p', :values => ['4']}}
912 assert_equal [], find_issues_with_query(query).map(&:id).sort
955 assert_equal [], find_issues_with_query(query).map(&:id).sort
913 end
956 end
914
957
915 def test_filter_on_relations_with_any_issues_not_in_a_project
958 def test_filter_on_relations_with_any_issues_not_in_a_project
916 IssueRelation.delete_all
959 IssueRelation.delete_all
917 with_settings :cross_project_issue_relations => '1' do
960 with_settings :cross_project_issue_relations => '1' do
918 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Project.find(2).issues.first)
961 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Project.find(2).issues.first)
919 #IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(2), :issue_to => Project.find(1).issues.first)
962 #IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(2), :issue_to => Project.find(1).issues.first)
920 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Project.find(3).issues.first)
963 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Project.find(3).issues.first)
921 end
964 end
922
965
923 query = IssueQuery.new(:name => '_')
966 query = IssueQuery.new(:name => '_')
924 query.filters = {"relates" => {:operator => '=!p', :values => ['1']}}
967 query.filters = {"relates" => {:operator => '=!p', :values => ['1']}}
925 assert_equal [1], find_issues_with_query(query).map(&:id).sort
968 assert_equal [1], find_issues_with_query(query).map(&:id).sort
926 end
969 end
927
970
928 def test_filter_on_relations_with_no_issues_in_a_project
971 def test_filter_on_relations_with_no_issues_in_a_project
929 IssueRelation.delete_all
972 IssueRelation.delete_all
930 with_settings :cross_project_issue_relations => '1' do
973 with_settings :cross_project_issue_relations => '1' do
931 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Project.find(2).issues.first)
974 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Project.find(2).issues.first)
932 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(2), :issue_to => Project.find(3).issues.first)
975 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(2), :issue_to => Project.find(3).issues.first)
933 IssueRelation.create!(:relation_type => "relates", :issue_to => Project.find(2).issues.first, :issue_from => Issue.find(3))
976 IssueRelation.create!(:relation_type => "relates", :issue_to => Project.find(2).issues.first, :issue_from => Issue.find(3))
934 end
977 end
935
978
936 query = IssueQuery.new(:name => '_')
979 query = IssueQuery.new(:name => '_')
937 query.filters = {"relates" => {:operator => '!p', :values => ['2']}}
980 query.filters = {"relates" => {:operator => '!p', :values => ['2']}}
938 ids = find_issues_with_query(query).map(&:id).sort
981 ids = find_issues_with_query(query).map(&:id).sort
939 assert_include 2, ids
982 assert_include 2, ids
940 assert_not_include 1, ids
983 assert_not_include 1, ids
941 assert_not_include 3, ids
984 assert_not_include 3, ids
942 end
985 end
943
986
944 def test_filter_on_relations_with_any_open_issues
987 def test_filter_on_relations_with_any_open_issues
945 IssueRelation.delete_all
988 IssueRelation.delete_all
946 # Issue 1 is blocked by 8, which is closed
989 # Issue 1 is blocked by 8, which is closed
947 IssueRelation.create!(:relation_type => "blocked", :issue_from => Issue.find(1), :issue_to => Issue.find(8))
990 IssueRelation.create!(:relation_type => "blocked", :issue_from => Issue.find(1), :issue_to => Issue.find(8))
948 # Issue 2 is blocked by 3, which is open
991 # Issue 2 is blocked by 3, which is open
949 IssueRelation.create!(:relation_type => "blocked", :issue_from => Issue.find(2), :issue_to => Issue.find(3))
992 IssueRelation.create!(:relation_type => "blocked", :issue_from => Issue.find(2), :issue_to => Issue.find(3))
950
993
951 query = IssueQuery.new(:name => '_')
994 query = IssueQuery.new(:name => '_')
952 query.filters = {"blocked" => {:operator => "*o", :values => ['']}}
995 query.filters = {"blocked" => {:operator => "*o", :values => ['']}}
953 ids = find_issues_with_query(query).map(&:id)
996 ids = find_issues_with_query(query).map(&:id)
954 assert_equal [], ids & [1]
997 assert_equal [], ids & [1]
955 assert_include 2, ids
998 assert_include 2, ids
956 end
999 end
957
1000
958 def test_filter_on_relations_with_no_open_issues
1001 def test_filter_on_relations_with_no_open_issues
959 IssueRelation.delete_all
1002 IssueRelation.delete_all
960 # Issue 1 is blocked by 8, which is closed
1003 # Issue 1 is blocked by 8, which is closed
961 IssueRelation.create!(:relation_type => "blocked", :issue_from => Issue.find(1), :issue_to => Issue.find(8))
1004 IssueRelation.create!(:relation_type => "blocked", :issue_from => Issue.find(1), :issue_to => Issue.find(8))
962 # Issue 2 is blocked by 3, which is open
1005 # Issue 2 is blocked by 3, which is open
963 IssueRelation.create!(:relation_type => "blocked", :issue_from => Issue.find(2), :issue_to => Issue.find(3))
1006 IssueRelation.create!(:relation_type => "blocked", :issue_from => Issue.find(2), :issue_to => Issue.find(3))
964
1007
965 query = IssueQuery.new(:name => '_')
1008 query = IssueQuery.new(:name => '_')
966 query.filters = {"blocked" => {:operator => "!o", :values => ['']}}
1009 query.filters = {"blocked" => {:operator => "!o", :values => ['']}}
967 ids = find_issues_with_query(query).map(&:id)
1010 ids = find_issues_with_query(query).map(&:id)
968 assert_equal [], ids & [2]
1011 assert_equal [], ids & [2]
969 assert_include 1, ids
1012 assert_include 1, ids
970 end
1013 end
971
1014
972 def test_filter_on_relations_with_no_issues
1015 def test_filter_on_relations_with_no_issues
973 IssueRelation.delete_all
1016 IssueRelation.delete_all
974 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Issue.find(2))
1017 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Issue.find(2))
975 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(3), :issue_to => Issue.find(1))
1018 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(3), :issue_to => Issue.find(1))
976
1019
977 query = IssueQuery.new(:name => '_')
1020 query = IssueQuery.new(:name => '_')
978 query.filters = {"relates" => {:operator => '!*', :values => ['']}}
1021 query.filters = {"relates" => {:operator => '!*', :values => ['']}}
979 ids = find_issues_with_query(query).map(&:id)
1022 ids = find_issues_with_query(query).map(&:id)
980 assert_equal [], ids & [1, 2, 3]
1023 assert_equal [], ids & [1, 2, 3]
981 assert_include 4, ids
1024 assert_include 4, ids
982 end
1025 end
983
1026
984 def test_filter_on_relations_with_any_issues
1027 def test_filter_on_relations_with_any_issues
985 IssueRelation.delete_all
1028 IssueRelation.delete_all
986 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Issue.find(2))
1029 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Issue.find(2))
987 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(3), :issue_to => Issue.find(1))
1030 IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(3), :issue_to => Issue.find(1))
988
1031
989 query = IssueQuery.new(:name => '_')
1032 query = IssueQuery.new(:name => '_')
990 query.filters = {"relates" => {:operator => '*', :values => ['']}}
1033 query.filters = {"relates" => {:operator => '*', :values => ['']}}
991 assert_equal [1, 2, 3], find_issues_with_query(query).map(&:id).sort
1034 assert_equal [1, 2, 3], find_issues_with_query(query).map(&:id).sort
992 end
1035 end
993
1036
994 def test_filter_on_relations_should_not_ignore_other_filter
1037 def test_filter_on_relations_should_not_ignore_other_filter
995 issue = Issue.generate!
1038 issue = Issue.generate!
996 issue1 = Issue.generate!(:status_id => 1)
1039 issue1 = Issue.generate!(:status_id => 1)
997 issue2 = Issue.generate!(:status_id => 2)
1040 issue2 = Issue.generate!(:status_id => 2)
998 IssueRelation.create!(:relation_type => "relates", :issue_from => issue, :issue_to => issue1)
1041 IssueRelation.create!(:relation_type => "relates", :issue_from => issue, :issue_to => issue1)
999 IssueRelation.create!(:relation_type => "relates", :issue_from => issue, :issue_to => issue2)
1042 IssueRelation.create!(:relation_type => "relates", :issue_from => issue, :issue_to => issue2)
1000
1043
1001 query = IssueQuery.new(:name => '_')
1044 query = IssueQuery.new(:name => '_')
1002 query.filters = {
1045 query.filters = {
1003 "status_id" => {:operator => '=', :values => ['1']},
1046 "status_id" => {:operator => '=', :values => ['1']},
1004 "relates" => {:operator => '=', :values => [issue.id.to_s]}
1047 "relates" => {:operator => '=', :values => [issue.id.to_s]}
1005 }
1048 }
1006 assert_equal [issue1], find_issues_with_query(query)
1049 assert_equal [issue1], find_issues_with_query(query)
1007 end
1050 end
1008
1051
1009 def test_filter_on_parent
1052 def test_filter_on_parent
1010 Issue.delete_all
1053 Issue.delete_all
1011 parent = Issue.generate_with_descendants!
1054 parent = Issue.generate_with_descendants!
1012
1055
1013
1056
1014 query = IssueQuery.new(:name => '_')
1057 query = IssueQuery.new(:name => '_')
1015 query.filters = {"parent_id" => {:operator => '=', :values => [parent.id.to_s]}}
1058 query.filters = {"parent_id" => {:operator => '=', :values => [parent.id.to_s]}}
1016 assert_equal parent.children.map(&:id).sort, find_issues_with_query(query).map(&:id).sort
1059 assert_equal parent.children.map(&:id).sort, find_issues_with_query(query).map(&:id).sort
1017
1060
1018 query.filters = {"parent_id" => {:operator => '~', :values => [parent.id.to_s]}}
1061 query.filters = {"parent_id" => {:operator => '~', :values => [parent.id.to_s]}}
1019 assert_equal parent.descendants.map(&:id).sort, find_issues_with_query(query).map(&:id).sort
1062 assert_equal parent.descendants.map(&:id).sort, find_issues_with_query(query).map(&:id).sort
1020
1063
1021 query.filters = {"parent_id" => {:operator => '*', :values => ['']}}
1064 query.filters = {"parent_id" => {:operator => '*', :values => ['']}}
1022 assert_equal parent.descendants.map(&:id).sort, find_issues_with_query(query).map(&:id).sort
1065 assert_equal parent.descendants.map(&:id).sort, find_issues_with_query(query).map(&:id).sort
1023
1066
1024 query.filters = {"parent_id" => {:operator => '!*', :values => ['']}}
1067 query.filters = {"parent_id" => {:operator => '!*', :values => ['']}}
1025 assert_equal [parent.id], find_issues_with_query(query).map(&:id).sort
1068 assert_equal [parent.id], find_issues_with_query(query).map(&:id).sort
1026 end
1069 end
1027
1070
1028 def test_filter_on_invalid_parent_should_return_no_results
1071 def test_filter_on_invalid_parent_should_return_no_results
1029 query = IssueQuery.new(:name => '_')
1072 query = IssueQuery.new(:name => '_')
1030 query.filters = {"parent_id" => {:operator => '=', :values => '99999999999'}}
1073 query.filters = {"parent_id" => {:operator => '=', :values => '99999999999'}}
1031 assert_equal [], find_issues_with_query(query).map(&:id).sort
1074 assert_equal [], find_issues_with_query(query).map(&:id).sort
1032
1075
1033 query.filters = {"parent_id" => {:operator => '~', :values => '99999999999'}}
1076 query.filters = {"parent_id" => {:operator => '~', :values => '99999999999'}}
1034 assert_equal [], find_issues_with_query(query)
1077 assert_equal [], find_issues_with_query(query)
1035 end
1078 end
1036
1079
1037 def test_filter_on_child
1080 def test_filter_on_child
1038 Issue.delete_all
1081 Issue.delete_all
1039 parent = Issue.generate_with_descendants!
1082 parent = Issue.generate_with_descendants!
1040 child, leaf = parent.children.sort_by(&:id)
1083 child, leaf = parent.children.sort_by(&:id)
1041 grandchild = child.children.first
1084 grandchild = child.children.first
1042
1085
1043
1086
1044 query = IssueQuery.new(:name => '_')
1087 query = IssueQuery.new(:name => '_')
1045 query.filters = {"child_id" => {:operator => '=', :values => [grandchild.id.to_s]}}
1088 query.filters = {"child_id" => {:operator => '=', :values => [grandchild.id.to_s]}}
1046 assert_equal [child.id], find_issues_with_query(query).map(&:id).sort
1089 assert_equal [child.id], find_issues_with_query(query).map(&:id).sort
1047
1090
1048 query.filters = {"child_id" => {:operator => '~', :values => [grandchild.id.to_s]}}
1091 query.filters = {"child_id" => {:operator => '~', :values => [grandchild.id.to_s]}}
1049 assert_equal [parent, child].map(&:id).sort, find_issues_with_query(query).map(&:id).sort
1092 assert_equal [parent, child].map(&:id).sort, find_issues_with_query(query).map(&:id).sort
1050
1093
1051 query.filters = {"child_id" => {:operator => '*', :values => ['']}}
1094 query.filters = {"child_id" => {:operator => '*', :values => ['']}}
1052 assert_equal [parent, child].map(&:id).sort, find_issues_with_query(query).map(&:id).sort
1095 assert_equal [parent, child].map(&:id).sort, find_issues_with_query(query).map(&:id).sort
1053
1096
1054 query.filters = {"child_id" => {:operator => '!*', :values => ['']}}
1097 query.filters = {"child_id" => {:operator => '!*', :values => ['']}}
1055 assert_equal [grandchild, leaf].map(&:id).sort, find_issues_with_query(query).map(&:id).sort
1098 assert_equal [grandchild, leaf].map(&:id).sort, find_issues_with_query(query).map(&:id).sort
1056 end
1099 end
1057
1100
1058 def test_filter_on_invalid_child_should_return_no_results
1101 def test_filter_on_invalid_child_should_return_no_results
1059 query = IssueQuery.new(:name => '_')
1102 query = IssueQuery.new(:name => '_')
1060 query.filters = {"child_id" => {:operator => '=', :values => '99999999999'}}
1103 query.filters = {"child_id" => {:operator => '=', :values => '99999999999'}}
1061 assert_equal [], find_issues_with_query(query)
1104 assert_equal [], find_issues_with_query(query)
1062
1105
1063 query.filters = {"child_id" => {:operator => '~', :values => '99999999999'}}
1106 query.filters = {"child_id" => {:operator => '~', :values => '99999999999'}}
1064 assert_equal [].map(&:id).sort, find_issues_with_query(query)
1107 assert_equal [].map(&:id).sort, find_issues_with_query(query)
1065 end
1108 end
1066
1109
1067 def test_statement_should_be_nil_with_no_filters
1110 def test_statement_should_be_nil_with_no_filters
1068 q = IssueQuery.new(:name => '_')
1111 q = IssueQuery.new(:name => '_')
1069 q.filters = {}
1112 q.filters = {}
1070
1113
1071 assert q.valid?
1114 assert q.valid?
1072 assert_nil q.statement
1115 assert_nil q.statement
1073 end
1116 end
1074
1117
1075 def test_available_filters_as_json_should_include_missing_assigned_to_id_values
1118 def test_available_filters_as_json_should_include_missing_assigned_to_id_values
1076 user = User.generate!
1119 user = User.generate!
1077 with_current_user User.find(1) do
1120 with_current_user User.find(1) do
1078 q = IssueQuery.new
1121 q = IssueQuery.new
1079 q.filters = {"assigned_to_id" => {:operator => '=', :values => user.id.to_s}}
1122 q.filters = {"assigned_to_id" => {:operator => '=', :values => user.id.to_s}}
1080
1123
1081 filters = q.available_filters_as_json
1124 filters = q.available_filters_as_json
1082 assert_include [user.name, user.id.to_s], filters['assigned_to_id']['values']
1125 assert_include [user.name, user.id.to_s], filters['assigned_to_id']['values']
1083 end
1126 end
1084 end
1127 end
1085
1128
1086 def test_available_filters_as_json_should_include_missing_author_id_values
1129 def test_available_filters_as_json_should_include_missing_author_id_values
1087 user = User.generate!
1130 user = User.generate!
1088 with_current_user User.find(1) do
1131 with_current_user User.find(1) do
1089 q = IssueQuery.new
1132 q = IssueQuery.new
1090 q.filters = {"author_id" => {:operator => '=', :values => user.id.to_s}}
1133 q.filters = {"author_id" => {:operator => '=', :values => user.id.to_s}}
1091
1134
1092 filters = q.available_filters_as_json
1135 filters = q.available_filters_as_json
1093 assert_include [user.name, user.id.to_s], filters['author_id']['values']
1136 assert_include [user.name, user.id.to_s], filters['author_id']['values']
1094 end
1137 end
1095 end
1138 end
1096
1139
1097 def test_default_columns
1140 def test_default_columns
1098 q = IssueQuery.new
1141 q = IssueQuery.new
1099 assert q.columns.any?
1142 assert q.columns.any?
1100 assert q.inline_columns.any?
1143 assert q.inline_columns.any?
1101 assert q.block_columns.empty?
1144 assert q.block_columns.empty?
1102 end
1145 end
1103
1146
1104 def test_set_column_names
1147 def test_set_column_names
1105 q = IssueQuery.new
1148 q = IssueQuery.new
1106 q.column_names = ['tracker', :subject, '', 'unknonw_column']
1149 q.column_names = ['tracker', :subject, '', 'unknonw_column']
1107 assert_equal [:id, :tracker, :subject], q.columns.collect {|c| c.name}
1150 assert_equal [:id, :tracker, :subject], q.columns.collect {|c| c.name}
1108 end
1151 end
1109
1152
1110 def test_has_column_should_accept_a_column_name
1153 def test_has_column_should_accept_a_column_name
1111 q = IssueQuery.new
1154 q = IssueQuery.new
1112 q.column_names = ['tracker', :subject]
1155 q.column_names = ['tracker', :subject]
1113 assert q.has_column?(:tracker)
1156 assert q.has_column?(:tracker)
1114 assert !q.has_column?(:category)
1157 assert !q.has_column?(:category)
1115 end
1158 end
1116
1159
1117 def test_has_column_should_accept_a_column
1160 def test_has_column_should_accept_a_column
1118 q = IssueQuery.new
1161 q = IssueQuery.new
1119 q.column_names = ['tracker', :subject]
1162 q.column_names = ['tracker', :subject]
1120
1163
1121 tracker_column = q.available_columns.detect {|c| c.name==:tracker}
1164 tracker_column = q.available_columns.detect {|c| c.name==:tracker}
1122 assert_kind_of QueryColumn, tracker_column
1165 assert_kind_of QueryColumn, tracker_column
1123 category_column = q.available_columns.detect {|c| c.name==:category}
1166 category_column = q.available_columns.detect {|c| c.name==:category}
1124 assert_kind_of QueryColumn, category_column
1167 assert_kind_of QueryColumn, category_column
1125
1168
1126 assert q.has_column?(tracker_column)
1169 assert q.has_column?(tracker_column)
1127 assert !q.has_column?(category_column)
1170 assert !q.has_column?(category_column)
1128 end
1171 end
1129
1172
1130 def test_inline_and_block_columns
1173 def test_inline_and_block_columns
1131 q = IssueQuery.new
1174 q = IssueQuery.new
1132 q.column_names = ['subject', 'description', 'tracker']
1175 q.column_names = ['subject', 'description', 'tracker']
1133
1176
1134 assert_equal [:id, :subject, :tracker], q.inline_columns.map(&:name)
1177 assert_equal [:id, :subject, :tracker], q.inline_columns.map(&:name)
1135 assert_equal [:description], q.block_columns.map(&:name)
1178 assert_equal [:description], q.block_columns.map(&:name)
1136 end
1179 end
1137
1180
1138 def test_custom_field_columns_should_be_inline
1181 def test_custom_field_columns_should_be_inline
1139 q = IssueQuery.new
1182 q = IssueQuery.new
1140 columns = q.available_columns.select {|column| column.is_a? QueryCustomFieldColumn}
1183 columns = q.available_columns.select {|column| column.is_a? QueryCustomFieldColumn}
1141 assert columns.any?
1184 assert columns.any?
1142 assert_nil columns.detect {|column| !column.inline?}
1185 assert_nil columns.detect {|column| !column.inline?}
1143 end
1186 end
1144
1187
1145 def test_query_should_preload_spent_hours
1188 def test_query_should_preload_spent_hours
1146 q = IssueQuery.new(:name => '_', :column_names => [:subject, :spent_hours])
1189 q = IssueQuery.new(:name => '_', :column_names => [:subject, :spent_hours])
1147 assert q.has_column?(:spent_hours)
1190 assert q.has_column?(:spent_hours)
1148 issues = q.issues
1191 issues = q.issues
1149 assert_not_nil issues.first.instance_variable_get("@spent_hours")
1192 assert_not_nil issues.first.instance_variable_get("@spent_hours")
1150 end
1193 end
1151
1194
1152 def test_groupable_columns_should_include_custom_fields
1195 def test_groupable_columns_should_include_custom_fields
1153 q = IssueQuery.new
1196 q = IssueQuery.new
1154 column = q.groupable_columns.detect {|c| c.name == :cf_1}
1197 column = q.groupable_columns.detect {|c| c.name == :cf_1}
1155 assert_not_nil column
1198 assert_not_nil column
1156 assert_kind_of QueryCustomFieldColumn, column
1199 assert_kind_of QueryCustomFieldColumn, column
1157 end
1200 end
1158
1201
1159 def test_groupable_columns_should_not_include_multi_custom_fields
1202 def test_groupable_columns_should_not_include_multi_custom_fields
1160 field = CustomField.find(1)
1203 field = CustomField.find(1)
1161 field.update_attribute :multiple, true
1204 field.update_attribute :multiple, true
1162
1205
1163 q = IssueQuery.new
1206 q = IssueQuery.new
1164 column = q.groupable_columns.detect {|c| c.name == :cf_1}
1207 column = q.groupable_columns.detect {|c| c.name == :cf_1}
1165 assert_nil column
1208 assert_nil column
1166 end
1209 end
1167
1210
1168 def test_groupable_columns_should_include_user_custom_fields
1211 def test_groupable_columns_should_include_user_custom_fields
1169 cf = IssueCustomField.create!(:name => 'User', :is_for_all => true, :tracker_ids => [1], :field_format => 'user')
1212 cf = IssueCustomField.create!(:name => 'User', :is_for_all => true, :tracker_ids => [1], :field_format => 'user')
1170
1213
1171 q = IssueQuery.new
1214 q = IssueQuery.new
1172 assert q.groupable_columns.detect {|c| c.name == "cf_#{cf.id}".to_sym}
1215 assert q.groupable_columns.detect {|c| c.name == "cf_#{cf.id}".to_sym}
1173 end
1216 end
1174
1217
1175 def test_groupable_columns_should_include_version_custom_fields
1218 def test_groupable_columns_should_include_version_custom_fields
1176 cf = IssueCustomField.create!(:name => 'User', :is_for_all => true, :tracker_ids => [1], :field_format => 'version')
1219 cf = IssueCustomField.create!(:name => 'User', :is_for_all => true, :tracker_ids => [1], :field_format => 'version')
1177
1220
1178 q = IssueQuery.new
1221 q = IssueQuery.new
1179 assert q.groupable_columns.detect {|c| c.name == "cf_#{cf.id}".to_sym}
1222 assert q.groupable_columns.detect {|c| c.name == "cf_#{cf.id}".to_sym}
1180 end
1223 end
1181
1224
1182 def test_grouped_with_valid_column
1225 def test_grouped_with_valid_column
1183 q = IssueQuery.new(:group_by => 'status')
1226 q = IssueQuery.new(:group_by => 'status')
1184 assert q.grouped?
1227 assert q.grouped?
1185 assert_not_nil q.group_by_column
1228 assert_not_nil q.group_by_column
1186 assert_equal :status, q.group_by_column.name
1229 assert_equal :status, q.group_by_column.name
1187 assert_not_nil q.group_by_statement
1230 assert_not_nil q.group_by_statement
1188 assert_equal 'status', q.group_by_statement
1231 assert_equal 'status', q.group_by_statement
1189 end
1232 end
1190
1233
1191 def test_grouped_with_invalid_column
1234 def test_grouped_with_invalid_column
1192 q = IssueQuery.new(:group_by => 'foo')
1235 q = IssueQuery.new(:group_by => 'foo')
1193 assert !q.grouped?
1236 assert !q.grouped?
1194 assert_nil q.group_by_column
1237 assert_nil q.group_by_column
1195 assert_nil q.group_by_statement
1238 assert_nil q.group_by_statement
1196 end
1239 end
1197
1240
1198 def test_sortable_columns_should_sort_assignees_according_to_user_format_setting
1241 def test_sortable_columns_should_sort_assignees_according_to_user_format_setting
1199 with_settings :user_format => 'lastname_comma_firstname' do
1242 with_settings :user_format => 'lastname_comma_firstname' do
1200 q = IssueQuery.new
1243 q = IssueQuery.new
1201 assert q.sortable_columns.has_key?('assigned_to')
1244 assert q.sortable_columns.has_key?('assigned_to')
1202 assert_equal %w(users.lastname users.firstname users.id), q.sortable_columns['assigned_to']
1245 assert_equal %w(users.lastname users.firstname users.id), q.sortable_columns['assigned_to']
1203 end
1246 end
1204 end
1247 end
1205
1248
1206 def test_sortable_columns_should_sort_authors_according_to_user_format_setting
1249 def test_sortable_columns_should_sort_authors_according_to_user_format_setting
1207 with_settings :user_format => 'lastname_comma_firstname' do
1250 with_settings :user_format => 'lastname_comma_firstname' do
1208 q = IssueQuery.new
1251 q = IssueQuery.new
1209 assert q.sortable_columns.has_key?('author')
1252 assert q.sortable_columns.has_key?('author')
1210 assert_equal %w(authors.lastname authors.firstname authors.id), q.sortable_columns['author']
1253 assert_equal %w(authors.lastname authors.firstname authors.id), q.sortable_columns['author']
1211 end
1254 end
1212 end
1255 end
1213
1256
1214 def test_sortable_columns_should_include_custom_field
1257 def test_sortable_columns_should_include_custom_field
1215 q = IssueQuery.new
1258 q = IssueQuery.new
1216 assert q.sortable_columns['cf_1']
1259 assert q.sortable_columns['cf_1']
1217 end
1260 end
1218
1261
1219 def test_sortable_columns_should_not_include_multi_custom_field
1262 def test_sortable_columns_should_not_include_multi_custom_field
1220 field = CustomField.find(1)
1263 field = CustomField.find(1)
1221 field.update_attribute :multiple, true
1264 field.update_attribute :multiple, true
1222
1265
1223 q = IssueQuery.new
1266 q = IssueQuery.new
1224 assert !q.sortable_columns['cf_1']
1267 assert !q.sortable_columns['cf_1']
1225 end
1268 end
1226
1269
1227 def test_default_sort
1270 def test_default_sort
1228 q = IssueQuery.new
1271 q = IssueQuery.new
1229 assert_equal [], q.sort_criteria
1272 assert_equal [], q.sort_criteria
1230 end
1273 end
1231
1274
1232 def test_set_sort_criteria_with_hash
1275 def test_set_sort_criteria_with_hash
1233 q = IssueQuery.new
1276 q = IssueQuery.new
1234 q.sort_criteria = {'0' => ['priority', 'desc'], '2' => ['tracker']}
1277 q.sort_criteria = {'0' => ['priority', 'desc'], '2' => ['tracker']}
1235 assert_equal [['priority', 'desc'], ['tracker', 'asc']], q.sort_criteria
1278 assert_equal [['priority', 'desc'], ['tracker', 'asc']], q.sort_criteria
1236 end
1279 end
1237
1280
1238 def test_set_sort_criteria_with_array
1281 def test_set_sort_criteria_with_array
1239 q = IssueQuery.new
1282 q = IssueQuery.new
1240 q.sort_criteria = [['priority', 'desc'], 'tracker']
1283 q.sort_criteria = [['priority', 'desc'], 'tracker']
1241 assert_equal [['priority', 'desc'], ['tracker', 'asc']], q.sort_criteria
1284 assert_equal [['priority', 'desc'], ['tracker', 'asc']], q.sort_criteria
1242 end
1285 end
1243
1286
1244 def test_create_query_with_sort
1287 def test_create_query_with_sort
1245 q = IssueQuery.new(:name => 'Sorted')
1288 q = IssueQuery.new(:name => 'Sorted')
1246 q.sort_criteria = [['priority', 'desc'], 'tracker']
1289 q.sort_criteria = [['priority', 'desc'], 'tracker']
1247 assert q.save
1290 assert q.save
1248 q.reload
1291 q.reload
1249 assert_equal [['priority', 'desc'], ['tracker', 'asc']], q.sort_criteria
1292 assert_equal [['priority', 'desc'], ['tracker', 'asc']], q.sort_criteria
1250 end
1293 end
1251
1294
1252 def test_sort_by_string_custom_field_asc
1295 def test_sort_by_string_custom_field_asc
1253 q = IssueQuery.new
1296 q = IssueQuery.new
1254 c = q.available_columns.find {|col| col.is_a?(QueryCustomFieldColumn) && col.custom_field.field_format == 'string' }
1297 c = q.available_columns.find {|col| col.is_a?(QueryCustomFieldColumn) && col.custom_field.field_format == 'string' }
1255 assert c
1298 assert c
1256 assert c.sortable
1299 assert c.sortable
1257 issues = q.issues(:order => "#{c.sortable} ASC")
1300 issues = q.issues(:order => "#{c.sortable} ASC")
1258 values = issues.collect {|i| i.custom_value_for(c.custom_field).to_s}
1301 values = issues.collect {|i| i.custom_value_for(c.custom_field).to_s}
1259 assert !values.empty?
1302 assert !values.empty?
1260 assert_equal values.sort, values
1303 assert_equal values.sort, values
1261 end
1304 end
1262
1305
1263 def test_sort_by_string_custom_field_desc
1306 def test_sort_by_string_custom_field_desc
1264 q = IssueQuery.new
1307 q = IssueQuery.new
1265 c = q.available_columns.find {|col| col.is_a?(QueryCustomFieldColumn) && col.custom_field.field_format == 'string' }
1308 c = q.available_columns.find {|col| col.is_a?(QueryCustomFieldColumn) && col.custom_field.field_format == 'string' }
1266 assert c
1309 assert c
1267 assert c.sortable
1310 assert c.sortable
1268 issues = q.issues(:order => "#{c.sortable} DESC")
1311 issues = q.issues(:order => "#{c.sortable} DESC")
1269 values = issues.collect {|i| i.custom_value_for(c.custom_field).to_s}
1312 values = issues.collect {|i| i.custom_value_for(c.custom_field).to_s}
1270 assert !values.empty?
1313 assert !values.empty?
1271 assert_equal values.sort.reverse, values
1314 assert_equal values.sort.reverse, values
1272 end
1315 end
1273
1316
1274 def test_sort_by_float_custom_field_asc
1317 def test_sort_by_float_custom_field_asc
1275 q = IssueQuery.new
1318 q = IssueQuery.new
1276 c = q.available_columns.find {|col| col.is_a?(QueryCustomFieldColumn) && col.custom_field.field_format == 'float' }
1319 c = q.available_columns.find {|col| col.is_a?(QueryCustomFieldColumn) && col.custom_field.field_format == 'float' }
1277 assert c
1320 assert c
1278 assert c.sortable
1321 assert c.sortable
1279 issues = q.issues(:order => "#{c.sortable} ASC")
1322 issues = q.issues(:order => "#{c.sortable} ASC")
1280 values = issues.collect {|i| begin; Kernel.Float(i.custom_value_for(c.custom_field).to_s); rescue; nil; end}.compact
1323 values = issues.collect {|i| begin; Kernel.Float(i.custom_value_for(c.custom_field).to_s); rescue; nil; end}.compact
1281 assert !values.empty?
1324 assert !values.empty?
1282 assert_equal values.sort, values
1325 assert_equal values.sort, values
1283 end
1326 end
1284
1327
1285 def test_set_totalable_names
1328 def test_set_totalable_names
1286 q = IssueQuery.new
1329 q = IssueQuery.new
1287 q.totalable_names = ['estimated_hours', :spent_hours, '']
1330 q.totalable_names = ['estimated_hours', :spent_hours, '']
1288 assert_equal [:estimated_hours, :spent_hours], q.totalable_columns.map(&:name)
1331 assert_equal [:estimated_hours, :spent_hours], q.totalable_columns.map(&:name)
1289 end
1332 end
1290
1333
1291 def test_totalable_columns_should_default_to_settings
1334 def test_totalable_columns_should_default_to_settings
1292 with_settings :issue_list_default_totals => ['estimated_hours'] do
1335 with_settings :issue_list_default_totals => ['estimated_hours'] do
1293 q = IssueQuery.new
1336 q = IssueQuery.new
1294 assert_equal [:estimated_hours], q.totalable_columns.map(&:name)
1337 assert_equal [:estimated_hours], q.totalable_columns.map(&:name)
1295 end
1338 end
1296 end
1339 end
1297
1340
1298 def test_available_totalable_columns_should_include_estimated_hours
1341 def test_available_totalable_columns_should_include_estimated_hours
1299 q = IssueQuery.new
1342 q = IssueQuery.new
1300 assert_include :estimated_hours, q.available_totalable_columns.map(&:name)
1343 assert_include :estimated_hours, q.available_totalable_columns.map(&:name)
1301 end
1344 end
1302
1345
1303 def test_available_totalable_columns_should_include_spent_hours
1346 def test_available_totalable_columns_should_include_spent_hours
1304 User.current = User.find(1)
1347 User.current = User.find(1)
1305
1348
1306 q = IssueQuery.new
1349 q = IssueQuery.new
1307 assert_include :spent_hours, q.available_totalable_columns.map(&:name)
1350 assert_include :spent_hours, q.available_totalable_columns.map(&:name)
1308 end
1351 end
1309
1352
1310 def test_available_totalable_columns_should_include_int_custom_field
1353 def test_available_totalable_columns_should_include_int_custom_field
1311 field = IssueCustomField.generate!(:field_format => 'int', :is_for_all => true)
1354 field = IssueCustomField.generate!(:field_format => 'int', :is_for_all => true)
1312 q = IssueQuery.new
1355 q = IssueQuery.new
1313 assert_include "cf_#{field.id}".to_sym, q.available_totalable_columns.map(&:name)
1356 assert_include "cf_#{field.id}".to_sym, q.available_totalable_columns.map(&:name)
1314 end
1357 end
1315
1358
1316 def test_available_totalable_columns_should_include_float_custom_field
1359 def test_available_totalable_columns_should_include_float_custom_field
1317 field = IssueCustomField.generate!(:field_format => 'float', :is_for_all => true)
1360 field = IssueCustomField.generate!(:field_format => 'float', :is_for_all => true)
1318 q = IssueQuery.new
1361 q = IssueQuery.new
1319 assert_include "cf_#{field.id}".to_sym, q.available_totalable_columns.map(&:name)
1362 assert_include "cf_#{field.id}".to_sym, q.available_totalable_columns.map(&:name)
1320 end
1363 end
1321
1364
1322 def test_total_for_estimated_hours
1365 def test_total_for_estimated_hours
1323 Issue.delete_all
1366 Issue.delete_all
1324 Issue.generate!(:estimated_hours => 5.5)
1367 Issue.generate!(:estimated_hours => 5.5)
1325 Issue.generate!(:estimated_hours => 1.1)
1368 Issue.generate!(:estimated_hours => 1.1)
1326 Issue.generate!
1369 Issue.generate!
1327
1370
1328 q = IssueQuery.new
1371 q = IssueQuery.new
1329 assert_equal 6.6, q.total_for(:estimated_hours)
1372 assert_equal 6.6, q.total_for(:estimated_hours)
1330 end
1373 end
1331
1374
1332 def test_total_by_group_for_estimated_hours
1375 def test_total_by_group_for_estimated_hours
1333 Issue.delete_all
1376 Issue.delete_all
1334 Issue.generate!(:estimated_hours => 5.5, :assigned_to_id => 2)
1377 Issue.generate!(:estimated_hours => 5.5, :assigned_to_id => 2)
1335 Issue.generate!(:estimated_hours => 1.1, :assigned_to_id => 3)
1378 Issue.generate!(:estimated_hours => 1.1, :assigned_to_id => 3)
1336 Issue.generate!(:estimated_hours => 3.5)
1379 Issue.generate!(:estimated_hours => 3.5)
1337
1380
1338 q = IssueQuery.new(:group_by => 'assigned_to')
1381 q = IssueQuery.new(:group_by => 'assigned_to')
1339 assert_equal(
1382 assert_equal(
1340 {nil => 3.5, User.find(2) => 5.5, User.find(3) => 1.1},
1383 {nil => 3.5, User.find(2) => 5.5, User.find(3) => 1.1},
1341 q.total_by_group_for(:estimated_hours)
1384 q.total_by_group_for(:estimated_hours)
1342 )
1385 )
1343 end
1386 end
1344
1387
1345 def test_total_for_spent_hours
1388 def test_total_for_spent_hours
1346 TimeEntry.delete_all
1389 TimeEntry.delete_all
1347 TimeEntry.generate!(:hours => 5.5)
1390 TimeEntry.generate!(:hours => 5.5)
1348 TimeEntry.generate!(:hours => 1.1)
1391 TimeEntry.generate!(:hours => 1.1)
1349
1392
1350 q = IssueQuery.new
1393 q = IssueQuery.new
1351 assert_equal 6.6, q.total_for(:spent_hours)
1394 assert_equal 6.6, q.total_for(:spent_hours)
1352 end
1395 end
1353
1396
1354 def test_total_by_group_for_spent_hours
1397 def test_total_by_group_for_spent_hours
1355 TimeEntry.delete_all
1398 TimeEntry.delete_all
1356 TimeEntry.generate!(:hours => 5.5, :issue_id => 1)
1399 TimeEntry.generate!(:hours => 5.5, :issue_id => 1)
1357 TimeEntry.generate!(:hours => 1.1, :issue_id => 2)
1400 TimeEntry.generate!(:hours => 1.1, :issue_id => 2)
1358 Issue.where(:id => 1).update_all(:assigned_to_id => 2)
1401 Issue.where(:id => 1).update_all(:assigned_to_id => 2)
1359 Issue.where(:id => 2).update_all(:assigned_to_id => 3)
1402 Issue.where(:id => 2).update_all(:assigned_to_id => 3)
1360
1403
1361 q = IssueQuery.new(:group_by => 'assigned_to')
1404 q = IssueQuery.new(:group_by => 'assigned_to')
1362 assert_equal(
1405 assert_equal(
1363 {User.find(2) => 5.5, User.find(3) => 1.1},
1406 {User.find(2) => 5.5, User.find(3) => 1.1},
1364 q.total_by_group_for(:spent_hours)
1407 q.total_by_group_for(:spent_hours)
1365 )
1408 )
1366 end
1409 end
1367
1410
1368 def test_total_by_project_group_for_spent_hours
1411 def test_total_by_project_group_for_spent_hours
1369 TimeEntry.delete_all
1412 TimeEntry.delete_all
1370 TimeEntry.generate!(:hours => 5.5, :issue_id => 1)
1413 TimeEntry.generate!(:hours => 5.5, :issue_id => 1)
1371 TimeEntry.generate!(:hours => 1.1, :issue_id => 2)
1414 TimeEntry.generate!(:hours => 1.1, :issue_id => 2)
1372 Issue.where(:id => 1).update_all(:assigned_to_id => 2)
1415 Issue.where(:id => 1).update_all(:assigned_to_id => 2)
1373 Issue.where(:id => 2).update_all(:assigned_to_id => 3)
1416 Issue.where(:id => 2).update_all(:assigned_to_id => 3)
1374
1417
1375 q = IssueQuery.new(:group_by => 'project')
1418 q = IssueQuery.new(:group_by => 'project')
1376 assert_equal(
1419 assert_equal(
1377 {Project.find(1) => 6.6},
1420 {Project.find(1) => 6.6},
1378 q.total_by_group_for(:spent_hours)
1421 q.total_by_group_for(:spent_hours)
1379 )
1422 )
1380 end
1423 end
1381
1424
1382 def test_total_for_int_custom_field
1425 def test_total_for_int_custom_field
1383 field = IssueCustomField.generate!(:field_format => 'int', :is_for_all => true)
1426 field = IssueCustomField.generate!(:field_format => 'int', :is_for_all => true)
1384 CustomValue.create!(:customized => Issue.find(1), :custom_field => field, :value => '2')
1427 CustomValue.create!(:customized => Issue.find(1), :custom_field => field, :value => '2')
1385 CustomValue.create!(:customized => Issue.find(2), :custom_field => field, :value => '7')
1428 CustomValue.create!(:customized => Issue.find(2), :custom_field => field, :value => '7')
1386 CustomValue.create!(:customized => Issue.find(3), :custom_field => field, :value => '')
1429 CustomValue.create!(:customized => Issue.find(3), :custom_field => field, :value => '')
1387
1430
1388 q = IssueQuery.new
1431 q = IssueQuery.new
1389 assert_equal 9, q.total_for("cf_#{field.id}")
1432 assert_equal 9, q.total_for("cf_#{field.id}")
1390 end
1433 end
1391
1434
1392 def test_total_by_group_for_int_custom_field
1435 def test_total_by_group_for_int_custom_field
1393 field = IssueCustomField.generate!(:field_format => 'int', :is_for_all => true)
1436 field = IssueCustomField.generate!(:field_format => 'int', :is_for_all => true)
1394 CustomValue.create!(:customized => Issue.find(1), :custom_field => field, :value => '2')
1437 CustomValue.create!(:customized => Issue.find(1), :custom_field => field, :value => '2')
1395 CustomValue.create!(:customized => Issue.find(2), :custom_field => field, :value => '7')
1438 CustomValue.create!(:customized => Issue.find(2), :custom_field => field, :value => '7')
1396 Issue.where(:id => 1).update_all(:assigned_to_id => 2)
1439 Issue.where(:id => 1).update_all(:assigned_to_id => 2)
1397 Issue.where(:id => 2).update_all(:assigned_to_id => 3)
1440 Issue.where(:id => 2).update_all(:assigned_to_id => 3)
1398
1441
1399 q = IssueQuery.new(:group_by => 'assigned_to')
1442 q = IssueQuery.new(:group_by => 'assigned_to')
1400 assert_equal(
1443 assert_equal(
1401 {User.find(2) => 2, User.find(3) => 7},
1444 {User.find(2) => 2, User.find(3) => 7},
1402 q.total_by_group_for("cf_#{field.id}")
1445 q.total_by_group_for("cf_#{field.id}")
1403 )
1446 )
1404 end
1447 end
1405
1448
1406 def test_total_for_float_custom_field
1449 def test_total_for_float_custom_field
1407 field = IssueCustomField.generate!(:field_format => 'float', :is_for_all => true)
1450 field = IssueCustomField.generate!(:field_format => 'float', :is_for_all => true)
1408 CustomValue.create!(:customized => Issue.find(1), :custom_field => field, :value => '2.3')
1451 CustomValue.create!(:customized => Issue.find(1), :custom_field => field, :value => '2.3')
1409 CustomValue.create!(:customized => Issue.find(2), :custom_field => field, :value => '7')
1452 CustomValue.create!(:customized => Issue.find(2), :custom_field => field, :value => '7')
1410 CustomValue.create!(:customized => Issue.find(3), :custom_field => field, :value => '')
1453 CustomValue.create!(:customized => Issue.find(3), :custom_field => field, :value => '')
1411
1454
1412 q = IssueQuery.new
1455 q = IssueQuery.new
1413 assert_equal 9.3, q.total_for("cf_#{field.id}")
1456 assert_equal 9.3, q.total_for("cf_#{field.id}")
1414 end
1457 end
1415
1458
1416 def test_invalid_query_should_raise_query_statement_invalid_error
1459 def test_invalid_query_should_raise_query_statement_invalid_error
1417 q = IssueQuery.new
1460 q = IssueQuery.new
1418 assert_raise Query::StatementInvalid do
1461 assert_raise Query::StatementInvalid do
1419 q.issues(:conditions => "foo = 1")
1462 q.issues(:conditions => "foo = 1")
1420 end
1463 end
1421 end
1464 end
1422
1465
1423 def test_issue_count
1466 def test_issue_count
1424 q = IssueQuery.new(:name => '_')
1467 q = IssueQuery.new(:name => '_')
1425 issue_count = q.issue_count
1468 issue_count = q.issue_count
1426 assert_equal q.issues.size, issue_count
1469 assert_equal q.issues.size, issue_count
1427 end
1470 end
1428
1471
1429 def test_issue_count_with_archived_issues
1472 def test_issue_count_with_archived_issues
1430 p = Project.generate! do |project|
1473 p = Project.generate! do |project|
1431 project.status = Project::STATUS_ARCHIVED
1474 project.status = Project::STATUS_ARCHIVED
1432 end
1475 end
1433 i = Issue.generate!( :project => p, :tracker => p.trackers.first )
1476 i = Issue.generate!( :project => p, :tracker => p.trackers.first )
1434 assert !i.visible?
1477 assert !i.visible?
1435
1478
1436 test_issue_count
1479 test_issue_count
1437 end
1480 end
1438
1481
1439 def test_issue_count_by_association_group
1482 def test_issue_count_by_association_group
1440 q = IssueQuery.new(:name => '_', :group_by => 'assigned_to')
1483 q = IssueQuery.new(:name => '_', :group_by => 'assigned_to')
1441 count_by_group = q.issue_count_by_group
1484 count_by_group = q.issue_count_by_group
1442 assert_kind_of Hash, count_by_group
1485 assert_kind_of Hash, count_by_group
1443 assert_equal %w(NilClass User), count_by_group.keys.collect {|k| k.class.name}.uniq.sort
1486 assert_equal %w(NilClass User), count_by_group.keys.collect {|k| k.class.name}.uniq.sort
1444 assert_equal %w(Fixnum), count_by_group.values.collect {|k| k.class.name}.uniq
1487 assert_equal %w(Fixnum), count_by_group.values.collect {|k| k.class.name}.uniq
1445 assert count_by_group.has_key?(User.find(3))
1488 assert count_by_group.has_key?(User.find(3))
1446 end
1489 end
1447
1490
1448 def test_issue_count_by_list_custom_field_group
1491 def test_issue_count_by_list_custom_field_group
1449 q = IssueQuery.new(:name => '_', :group_by => 'cf_1')
1492 q = IssueQuery.new(:name => '_', :group_by => 'cf_1')
1450 count_by_group = q.issue_count_by_group
1493 count_by_group = q.issue_count_by_group
1451 assert_kind_of Hash, count_by_group
1494 assert_kind_of Hash, count_by_group
1452 assert_equal %w(NilClass String), count_by_group.keys.collect {|k| k.class.name}.uniq.sort
1495 assert_equal %w(NilClass String), count_by_group.keys.collect {|k| k.class.name}.uniq.sort
1453 assert_equal %w(Fixnum), count_by_group.values.collect {|k| k.class.name}.uniq
1496 assert_equal %w(Fixnum), count_by_group.values.collect {|k| k.class.name}.uniq
1454 assert count_by_group.has_key?('MySQL')
1497 assert count_by_group.has_key?('MySQL')
1455 end
1498 end
1456
1499
1457 def test_issue_count_by_date_custom_field_group
1500 def test_issue_count_by_date_custom_field_group
1458 q = IssueQuery.new(:name => '_', :group_by => 'cf_8')
1501 q = IssueQuery.new(:name => '_', :group_by => 'cf_8')
1459 count_by_group = q.issue_count_by_group
1502 count_by_group = q.issue_count_by_group
1460 assert_kind_of Hash, count_by_group
1503 assert_kind_of Hash, count_by_group
1461 assert_equal %w(Date NilClass), count_by_group.keys.collect {|k| k.class.name}.uniq.sort
1504 assert_equal %w(Date NilClass), count_by_group.keys.collect {|k| k.class.name}.uniq.sort
1462 assert_equal %w(Fixnum), count_by_group.values.collect {|k| k.class.name}.uniq
1505 assert_equal %w(Fixnum), count_by_group.values.collect {|k| k.class.name}.uniq
1463 end
1506 end
1464
1507
1465 def test_issue_count_with_nil_group_only
1508 def test_issue_count_with_nil_group_only
1466 Issue.update_all("assigned_to_id = NULL")
1509 Issue.update_all("assigned_to_id = NULL")
1467
1510
1468 q = IssueQuery.new(:name => '_', :group_by => 'assigned_to')
1511 q = IssueQuery.new(:name => '_', :group_by => 'assigned_to')
1469 count_by_group = q.issue_count_by_group
1512 count_by_group = q.issue_count_by_group
1470 assert_kind_of Hash, count_by_group
1513 assert_kind_of Hash, count_by_group
1471 assert_equal 1, count_by_group.keys.size
1514 assert_equal 1, count_by_group.keys.size
1472 assert_nil count_by_group.keys.first
1515 assert_nil count_by_group.keys.first
1473 end
1516 end
1474
1517
1475 def test_issue_ids
1518 def test_issue_ids
1476 q = IssueQuery.new(:name => '_')
1519 q = IssueQuery.new(:name => '_')
1477 order = "issues.subject, issues.id"
1520 order = "issues.subject, issues.id"
1478 issues = q.issues(:order => order)
1521 issues = q.issues(:order => order)
1479 assert_equal issues.map(&:id), q.issue_ids(:order => order)
1522 assert_equal issues.map(&:id), q.issue_ids(:order => order)
1480 end
1523 end
1481
1524
1482 def test_label_for
1525 def test_label_for
1483 set_language_if_valid 'en'
1526 set_language_if_valid 'en'
1484 q = IssueQuery.new
1527 q = IssueQuery.new
1485 assert_equal 'Assignee', q.label_for('assigned_to_id')
1528 assert_equal 'Assignee', q.label_for('assigned_to_id')
1486 end
1529 end
1487
1530
1488 def test_label_for_fr
1531 def test_label_for_fr
1489 set_language_if_valid 'fr'
1532 set_language_if_valid 'fr'
1490 q = IssueQuery.new
1533 q = IssueQuery.new
1491 assert_equal "Assign\xc3\xa9 \xc3\xa0".force_encoding('UTF-8'), q.label_for('assigned_to_id')
1534 assert_equal "Assign\xc3\xa9 \xc3\xa0".force_encoding('UTF-8'), q.label_for('assigned_to_id')
1492 end
1535 end
1493
1536
1494 def test_editable_by
1537 def test_editable_by
1495 admin = User.find(1)
1538 admin = User.find(1)
1496 manager = User.find(2)
1539 manager = User.find(2)
1497 developer = User.find(3)
1540 developer = User.find(3)
1498
1541
1499 # Public query on project 1
1542 # Public query on project 1
1500 q = IssueQuery.find(1)
1543 q = IssueQuery.find(1)
1501 assert q.editable_by?(admin)
1544 assert q.editable_by?(admin)
1502 assert q.editable_by?(manager)
1545 assert q.editable_by?(manager)
1503 assert !q.editable_by?(developer)
1546 assert !q.editable_by?(developer)
1504
1547
1505 # Private query on project 1
1548 # Private query on project 1
1506 q = IssueQuery.find(2)
1549 q = IssueQuery.find(2)
1507 assert q.editable_by?(admin)
1550 assert q.editable_by?(admin)
1508 assert !q.editable_by?(manager)
1551 assert !q.editable_by?(manager)
1509 assert q.editable_by?(developer)
1552 assert q.editable_by?(developer)
1510
1553
1511 # Private query for all projects
1554 # Private query for all projects
1512 q = IssueQuery.find(3)
1555 q = IssueQuery.find(3)
1513 assert q.editable_by?(admin)
1556 assert q.editable_by?(admin)
1514 assert !q.editable_by?(manager)
1557 assert !q.editable_by?(manager)
1515 assert q.editable_by?(developer)
1558 assert q.editable_by?(developer)
1516
1559
1517 # Public query for all projects
1560 # Public query for all projects
1518 q = IssueQuery.find(4)
1561 q = IssueQuery.find(4)
1519 assert q.editable_by?(admin)
1562 assert q.editable_by?(admin)
1520 assert !q.editable_by?(manager)
1563 assert !q.editable_by?(manager)
1521 assert !q.editable_by?(developer)
1564 assert !q.editable_by?(developer)
1522 end
1565 end
1523
1566
1524 def test_visible_scope
1567 def test_visible_scope
1525 query_ids = IssueQuery.visible(User.anonymous).map(&:id)
1568 query_ids = IssueQuery.visible(User.anonymous).map(&:id)
1526
1569
1527 assert query_ids.include?(1), 'public query on public project was not visible'
1570 assert query_ids.include?(1), 'public query on public project was not visible'
1528 assert query_ids.include?(4), 'public query for all projects was not visible'
1571 assert query_ids.include?(4), 'public query for all projects was not visible'
1529 assert !query_ids.include?(2), 'private query on public project was visible'
1572 assert !query_ids.include?(2), 'private query on public project was visible'
1530 assert !query_ids.include?(3), 'private query for all projects was visible'
1573 assert !query_ids.include?(3), 'private query for all projects was visible'
1531 assert !query_ids.include?(7), 'public query on private project was visible'
1574 assert !query_ids.include?(7), 'public query on private project was visible'
1532 end
1575 end
1533
1576
1534 def test_query_with_public_visibility_should_be_visible_to_anyone
1577 def test_query_with_public_visibility_should_be_visible_to_anyone
1535 q = IssueQuery.create!(:name => 'Query', :visibility => IssueQuery::VISIBILITY_PUBLIC)
1578 q = IssueQuery.create!(:name => 'Query', :visibility => IssueQuery::VISIBILITY_PUBLIC)
1536
1579
1537 assert q.visible?(User.anonymous)
1580 assert q.visible?(User.anonymous)
1538 assert IssueQuery.visible(User.anonymous).find_by_id(q.id)
1581 assert IssueQuery.visible(User.anonymous).find_by_id(q.id)
1539
1582
1540 assert q.visible?(User.find(7))
1583 assert q.visible?(User.find(7))
1541 assert IssueQuery.visible(User.find(7)).find_by_id(q.id)
1584 assert IssueQuery.visible(User.find(7)).find_by_id(q.id)
1542
1585
1543 assert q.visible?(User.find(2))
1586 assert q.visible?(User.find(2))
1544 assert IssueQuery.visible(User.find(2)).find_by_id(q.id)
1587 assert IssueQuery.visible(User.find(2)).find_by_id(q.id)
1545
1588
1546 assert q.visible?(User.find(1))
1589 assert q.visible?(User.find(1))
1547 assert IssueQuery.visible(User.find(1)).find_by_id(q.id)
1590 assert IssueQuery.visible(User.find(1)).find_by_id(q.id)
1548 end
1591 end
1549
1592
1550 def test_query_with_roles_visibility_should_be_visible_to_user_with_role
1593 def test_query_with_roles_visibility_should_be_visible_to_user_with_role
1551 q = IssueQuery.create!(:name => 'Query', :visibility => IssueQuery::VISIBILITY_ROLES, :role_ids => [1,2])
1594 q = IssueQuery.create!(:name => 'Query', :visibility => IssueQuery::VISIBILITY_ROLES, :role_ids => [1,2])
1552
1595
1553 assert !q.visible?(User.anonymous)
1596 assert !q.visible?(User.anonymous)
1554 assert_nil IssueQuery.visible(User.anonymous).find_by_id(q.id)
1597 assert_nil IssueQuery.visible(User.anonymous).find_by_id(q.id)
1555
1598
1556 assert !q.visible?(User.find(7))
1599 assert !q.visible?(User.find(7))
1557 assert_nil IssueQuery.visible(User.find(7)).find_by_id(q.id)
1600 assert_nil IssueQuery.visible(User.find(7)).find_by_id(q.id)
1558
1601
1559 assert q.visible?(User.find(2))
1602 assert q.visible?(User.find(2))
1560 assert IssueQuery.visible(User.find(2)).find_by_id(q.id)
1603 assert IssueQuery.visible(User.find(2)).find_by_id(q.id)
1561
1604
1562 assert q.visible?(User.find(1))
1605 assert q.visible?(User.find(1))
1563 assert IssueQuery.visible(User.find(1)).find_by_id(q.id)
1606 assert IssueQuery.visible(User.find(1)).find_by_id(q.id)
1564 end
1607 end
1565
1608
1566 def test_query_with_private_visibility_should_be_visible_to_owner
1609 def test_query_with_private_visibility_should_be_visible_to_owner
1567 q = IssueQuery.create!(:name => 'Query', :visibility => IssueQuery::VISIBILITY_PRIVATE, :user => User.find(7))
1610 q = IssueQuery.create!(:name => 'Query', :visibility => IssueQuery::VISIBILITY_PRIVATE, :user => User.find(7))
1568
1611
1569 assert !q.visible?(User.anonymous)
1612 assert !q.visible?(User.anonymous)
1570 assert_nil IssueQuery.visible(User.anonymous).find_by_id(q.id)
1613 assert_nil IssueQuery.visible(User.anonymous).find_by_id(q.id)
1571
1614
1572 assert q.visible?(User.find(7))
1615 assert q.visible?(User.find(7))
1573 assert IssueQuery.visible(User.find(7)).find_by_id(q.id)
1616 assert IssueQuery.visible(User.find(7)).find_by_id(q.id)
1574
1617
1575 assert !q.visible?(User.find(2))
1618 assert !q.visible?(User.find(2))
1576 assert_nil IssueQuery.visible(User.find(2)).find_by_id(q.id)
1619 assert_nil IssueQuery.visible(User.find(2)).find_by_id(q.id)
1577
1620
1578 assert q.visible?(User.find(1))
1621 assert q.visible?(User.find(1))
1579 assert_nil IssueQuery.visible(User.find(1)).find_by_id(q.id)
1622 assert_nil IssueQuery.visible(User.find(1)).find_by_id(q.id)
1580 end
1623 end
1581
1624
1582 test "#available_filters should include users of visible projects in cross-project view" do
1625 test "#available_filters should include users of visible projects in cross-project view" do
1583 users = IssueQuery.new.available_filters["assigned_to_id"]
1626 users = IssueQuery.new.available_filters["assigned_to_id"]
1584 assert_not_nil users
1627 assert_not_nil users
1585 assert users[:values].map{|u|u[1]}.include?("3")
1628 assert users[:values].map{|u|u[1]}.include?("3")
1586 end
1629 end
1587
1630
1588 test "#available_filters should include users of subprojects" do
1631 test "#available_filters should include users of subprojects" do
1589 user1 = User.generate!
1632 user1 = User.generate!
1590 user2 = User.generate!
1633 user2 = User.generate!
1591 project = Project.find(1)
1634 project = Project.find(1)
1592 Member.create!(:principal => user1, :project => project.children.visible.first, :role_ids => [1])
1635 Member.create!(:principal => user1, :project => project.children.visible.first, :role_ids => [1])
1593
1636
1594 users = IssueQuery.new(:project => project).available_filters["assigned_to_id"]
1637 users = IssueQuery.new(:project => project).available_filters["assigned_to_id"]
1595 assert_not_nil users
1638 assert_not_nil users
1596 assert users[:values].map{|u|u[1]}.include?(user1.id.to_s)
1639 assert users[:values].map{|u|u[1]}.include?(user1.id.to_s)
1597 assert !users[:values].map{|u|u[1]}.include?(user2.id.to_s)
1640 assert !users[:values].map{|u|u[1]}.include?(user2.id.to_s)
1598 end
1641 end
1599
1642
1600 test "#available_filters should include visible projects in cross-project view" do
1643 test "#available_filters should include visible projects in cross-project view" do
1601 projects = IssueQuery.new.available_filters["project_id"]
1644 projects = IssueQuery.new.available_filters["project_id"]
1602 assert_not_nil projects
1645 assert_not_nil projects
1603 assert projects[:values].map{|u|u[1]}.include?("1")
1646 assert projects[:values].map{|u|u[1]}.include?("1")
1604 end
1647 end
1605
1648
1606 test "#available_filters should include 'member_of_group' filter" do
1649 test "#available_filters should include 'member_of_group' filter" do
1607 query = IssueQuery.new
1650 query = IssueQuery.new
1608 assert query.available_filters.keys.include?("member_of_group")
1651 assert query.available_filters.keys.include?("member_of_group")
1609 assert_equal :list_optional, query.available_filters["member_of_group"][:type]
1652 assert_equal :list_optional, query.available_filters["member_of_group"][:type]
1610 assert query.available_filters["member_of_group"][:values].present?
1653 assert query.available_filters["member_of_group"][:values].present?
1611 assert_equal Group.givable.sort.map {|g| [g.name, g.id.to_s]},
1654 assert_equal Group.givable.sort.map {|g| [g.name, g.id.to_s]},
1612 query.available_filters["member_of_group"][:values].sort
1655 query.available_filters["member_of_group"][:values].sort
1613 end
1656 end
1614
1657
1615 test "#available_filters should include 'assigned_to_role' filter" do
1658 test "#available_filters should include 'assigned_to_role' filter" do
1616 query = IssueQuery.new
1659 query = IssueQuery.new
1617 assert query.available_filters.keys.include?("assigned_to_role")
1660 assert query.available_filters.keys.include?("assigned_to_role")
1618 assert_equal :list_optional, query.available_filters["assigned_to_role"][:type]
1661 assert_equal :list_optional, query.available_filters["assigned_to_role"][:type]
1619
1662
1620 assert query.available_filters["assigned_to_role"][:values].include?(['Manager','1'])
1663 assert query.available_filters["assigned_to_role"][:values].include?(['Manager','1'])
1621 assert query.available_filters["assigned_to_role"][:values].include?(['Developer','2'])
1664 assert query.available_filters["assigned_to_role"][:values].include?(['Developer','2'])
1622 assert query.available_filters["assigned_to_role"][:values].include?(['Reporter','3'])
1665 assert query.available_filters["assigned_to_role"][:values].include?(['Reporter','3'])
1623
1666
1624 assert ! query.available_filters["assigned_to_role"][:values].include?(['Non member','4'])
1667 assert ! query.available_filters["assigned_to_role"][:values].include?(['Non member','4'])
1625 assert ! query.available_filters["assigned_to_role"][:values].include?(['Anonymous','5'])
1668 assert ! query.available_filters["assigned_to_role"][:values].include?(['Anonymous','5'])
1626 end
1669 end
1627
1670
1628 def test_available_filters_should_include_custom_field_according_to_user_visibility
1671 def test_available_filters_should_include_custom_field_according_to_user_visibility
1629 visible_field = IssueCustomField.generate!(:is_for_all => true, :is_filter => true, :visible => true)
1672 visible_field = IssueCustomField.generate!(:is_for_all => true, :is_filter => true, :visible => true)
1630 hidden_field = IssueCustomField.generate!(:is_for_all => true, :is_filter => true, :visible => false, :role_ids => [1])
1673 hidden_field = IssueCustomField.generate!(:is_for_all => true, :is_filter => true, :visible => false, :role_ids => [1])
1631
1674
1632 with_current_user User.find(3) do
1675 with_current_user User.find(3) do
1633 query = IssueQuery.new
1676 query = IssueQuery.new
1634 assert_include "cf_#{visible_field.id}", query.available_filters.keys
1677 assert_include "cf_#{visible_field.id}", query.available_filters.keys
1635 assert_not_include "cf_#{hidden_field.id}", query.available_filters.keys
1678 assert_not_include "cf_#{hidden_field.id}", query.available_filters.keys
1636 end
1679 end
1637 end
1680 end
1638
1681
1639 def test_available_columns_should_include_custom_field_according_to_user_visibility
1682 def test_available_columns_should_include_custom_field_according_to_user_visibility
1640 visible_field = IssueCustomField.generate!(:is_for_all => true, :is_filter => true, :visible => true)
1683 visible_field = IssueCustomField.generate!(:is_for_all => true, :is_filter => true, :visible => true)
1641 hidden_field = IssueCustomField.generate!(:is_for_all => true, :is_filter => true, :visible => false, :role_ids => [1])
1684 hidden_field = IssueCustomField.generate!(:is_for_all => true, :is_filter => true, :visible => false, :role_ids => [1])
1642
1685
1643 with_current_user User.find(3) do
1686 with_current_user User.find(3) do
1644 query = IssueQuery.new
1687 query = IssueQuery.new
1645 assert_include :"cf_#{visible_field.id}", query.available_columns.map(&:name)
1688 assert_include :"cf_#{visible_field.id}", query.available_columns.map(&:name)
1646 assert_not_include :"cf_#{hidden_field.id}", query.available_columns.map(&:name)
1689 assert_not_include :"cf_#{hidden_field.id}", query.available_columns.map(&:name)
1647 end
1690 end
1648 end
1691 end
1649
1692
1650 def setup_member_of_group
1693 def setup_member_of_group
1651 Group.destroy_all # No fixtures
1694 Group.destroy_all # No fixtures
1652 @user_in_group = User.generate!
1695 @user_in_group = User.generate!
1653 @second_user_in_group = User.generate!
1696 @second_user_in_group = User.generate!
1654 @user_in_group2 = User.generate!
1697 @user_in_group2 = User.generate!
1655 @user_not_in_group = User.generate!
1698 @user_not_in_group = User.generate!
1656
1699
1657 @group = Group.generate!.reload
1700 @group = Group.generate!.reload
1658 @group.users << @user_in_group
1701 @group.users << @user_in_group
1659 @group.users << @second_user_in_group
1702 @group.users << @second_user_in_group
1660
1703
1661 @group2 = Group.generate!.reload
1704 @group2 = Group.generate!.reload
1662 @group2.users << @user_in_group2
1705 @group2.users << @user_in_group2
1663
1706
1664 @query = IssueQuery.new(:name => '_')
1707 @query = IssueQuery.new(:name => '_')
1665 end
1708 end
1666
1709
1667 test "member_of_group filter should search assigned to for users in the group" do
1710 test "member_of_group filter should search assigned to for users in the group" do
1668 setup_member_of_group
1711 setup_member_of_group
1669 @query.add_filter('member_of_group', '=', [@group.id.to_s])
1712 @query.add_filter('member_of_group', '=', [@group.id.to_s])
1670
1713
1671 assert_find_issues_with_query_is_successful @query
1714 assert_find_issues_with_query_is_successful @query
1672 end
1715 end
1673
1716
1674 test "member_of_group filter should search not assigned to any group member (none)" do
1717 test "member_of_group filter should search not assigned to any group member (none)" do
1675 setup_member_of_group
1718 setup_member_of_group
1676 @query.add_filter('member_of_group', '!*', [''])
1719 @query.add_filter('member_of_group', '!*', [''])
1677
1720
1678 assert_find_issues_with_query_is_successful @query
1721 assert_find_issues_with_query_is_successful @query
1679 end
1722 end
1680
1723
1681 test "member_of_group filter should search assigned to any group member (all)" do
1724 test "member_of_group filter should search assigned to any group member (all)" do
1682 setup_member_of_group
1725 setup_member_of_group
1683 @query.add_filter('member_of_group', '*', [''])
1726 @query.add_filter('member_of_group', '*', [''])
1684
1727
1685 assert_find_issues_with_query_is_successful @query
1728 assert_find_issues_with_query_is_successful @query
1686 end
1729 end
1687
1730
1688 test "member_of_group filter should return an empty set with = empty group" do
1731 test "member_of_group filter should return an empty set with = empty group" do
1689 setup_member_of_group
1732 setup_member_of_group
1690 @empty_group = Group.generate!
1733 @empty_group = Group.generate!
1691 @query.add_filter('member_of_group', '=', [@empty_group.id.to_s])
1734 @query.add_filter('member_of_group', '=', [@empty_group.id.to_s])
1692
1735
1693 assert_equal [], find_issues_with_query(@query)
1736 assert_equal [], find_issues_with_query(@query)
1694 end
1737 end
1695
1738
1696 test "member_of_group filter should return issues with ! empty group" do
1739 test "member_of_group filter should return issues with ! empty group" do
1697 setup_member_of_group
1740 setup_member_of_group
1698 @empty_group = Group.generate!
1741 @empty_group = Group.generate!
1699 @query.add_filter('member_of_group', '!', [@empty_group.id.to_s])
1742 @query.add_filter('member_of_group', '!', [@empty_group.id.to_s])
1700
1743
1701 assert_find_issues_with_query_is_successful @query
1744 assert_find_issues_with_query_is_successful @query
1702 end
1745 end
1703
1746
1704 def setup_assigned_to_role
1747 def setup_assigned_to_role
1705 @manager_role = Role.find_by_name('Manager')
1748 @manager_role = Role.find_by_name('Manager')
1706 @developer_role = Role.find_by_name('Developer')
1749 @developer_role = Role.find_by_name('Developer')
1707
1750
1708 @project = Project.generate!
1751 @project = Project.generate!
1709 @manager = User.generate!
1752 @manager = User.generate!
1710 @developer = User.generate!
1753 @developer = User.generate!
1711 @boss = User.generate!
1754 @boss = User.generate!
1712 @guest = User.generate!
1755 @guest = User.generate!
1713 User.add_to_project(@manager, @project, @manager_role)
1756 User.add_to_project(@manager, @project, @manager_role)
1714 User.add_to_project(@developer, @project, @developer_role)
1757 User.add_to_project(@developer, @project, @developer_role)
1715 User.add_to_project(@boss, @project, [@manager_role, @developer_role])
1758 User.add_to_project(@boss, @project, [@manager_role, @developer_role])
1716
1759
1717 @issue1 = Issue.generate!(:project => @project, :assigned_to_id => @manager.id)
1760 @issue1 = Issue.generate!(:project => @project, :assigned_to_id => @manager.id)
1718 @issue2 = Issue.generate!(:project => @project, :assigned_to_id => @developer.id)
1761 @issue2 = Issue.generate!(:project => @project, :assigned_to_id => @developer.id)
1719 @issue3 = Issue.generate!(:project => @project, :assigned_to_id => @boss.id)
1762 @issue3 = Issue.generate!(:project => @project, :assigned_to_id => @boss.id)
1720 @issue4 = Issue.generate!(:project => @project, :author_id => @guest.id, :assigned_to_id => @guest.id)
1763 @issue4 = Issue.generate!(:project => @project, :author_id => @guest.id, :assigned_to_id => @guest.id)
1721 @issue5 = Issue.generate!(:project => @project)
1764 @issue5 = Issue.generate!(:project => @project)
1722
1765
1723 @query = IssueQuery.new(:name => '_', :project => @project)
1766 @query = IssueQuery.new(:name => '_', :project => @project)
1724 end
1767 end
1725
1768
1726 test "assigned_to_role filter should search assigned to for users with the Role" do
1769 test "assigned_to_role filter should search assigned to for users with the Role" do
1727 setup_assigned_to_role
1770 setup_assigned_to_role
1728 @query.add_filter('assigned_to_role', '=', [@manager_role.id.to_s])
1771 @query.add_filter('assigned_to_role', '=', [@manager_role.id.to_s])
1729
1772
1730 assert_query_result [@issue1, @issue3], @query
1773 assert_query_result [@issue1, @issue3], @query
1731 end
1774 end
1732
1775
1733 test "assigned_to_role filter should search assigned to for users with the Role on the issue project" do
1776 test "assigned_to_role filter should search assigned to for users with the Role on the issue project" do
1734 setup_assigned_to_role
1777 setup_assigned_to_role
1735 other_project = Project.generate!
1778 other_project = Project.generate!
1736 User.add_to_project(@developer, other_project, @manager_role)
1779 User.add_to_project(@developer, other_project, @manager_role)
1737 @query.add_filter('assigned_to_role', '=', [@manager_role.id.to_s])
1780 @query.add_filter('assigned_to_role', '=', [@manager_role.id.to_s])
1738
1781
1739 assert_query_result [@issue1, @issue3], @query
1782 assert_query_result [@issue1, @issue3], @query
1740 end
1783 end
1741
1784
1742 test "assigned_to_role filter should return an empty set with empty role" do
1785 test "assigned_to_role filter should return an empty set with empty role" do
1743 setup_assigned_to_role
1786 setup_assigned_to_role
1744 @empty_role = Role.generate!
1787 @empty_role = Role.generate!
1745 @query.add_filter('assigned_to_role', '=', [@empty_role.id.to_s])
1788 @query.add_filter('assigned_to_role', '=', [@empty_role.id.to_s])
1746
1789
1747 assert_query_result [], @query
1790 assert_query_result [], @query
1748 end
1791 end
1749
1792
1750 test "assigned_to_role filter should search assigned to for users without the Role" do
1793 test "assigned_to_role filter should search assigned to for users without the Role" do
1751 setup_assigned_to_role
1794 setup_assigned_to_role
1752 @query.add_filter('assigned_to_role', '!', [@manager_role.id.to_s])
1795 @query.add_filter('assigned_to_role', '!', [@manager_role.id.to_s])
1753
1796
1754 assert_query_result [@issue2, @issue4, @issue5], @query
1797 assert_query_result [@issue2, @issue4, @issue5], @query
1755 end
1798 end
1756
1799
1757 test "assigned_to_role filter should search assigned to for users not assigned to any Role (none)" do
1800 test "assigned_to_role filter should search assigned to for users not assigned to any Role (none)" do
1758 setup_assigned_to_role
1801 setup_assigned_to_role
1759 @query.add_filter('assigned_to_role', '!*', [''])
1802 @query.add_filter('assigned_to_role', '!*', [''])
1760
1803
1761 assert_query_result [@issue4, @issue5], @query
1804 assert_query_result [@issue4, @issue5], @query
1762 end
1805 end
1763
1806
1764 test "assigned_to_role filter should search assigned to for users assigned to any Role (all)" do
1807 test "assigned_to_role filter should search assigned to for users assigned to any Role (all)" do
1765 setup_assigned_to_role
1808 setup_assigned_to_role
1766 @query.add_filter('assigned_to_role', '*', [''])
1809 @query.add_filter('assigned_to_role', '*', [''])
1767
1810
1768 assert_query_result [@issue1, @issue2, @issue3], @query
1811 assert_query_result [@issue1, @issue2, @issue3], @query
1769 end
1812 end
1770
1813
1771 test "assigned_to_role filter should return issues with ! empty role" do
1814 test "assigned_to_role filter should return issues with ! empty role" do
1772 setup_assigned_to_role
1815 setup_assigned_to_role
1773 @empty_role = Role.generate!
1816 @empty_role = Role.generate!
1774 @query.add_filter('assigned_to_role', '!', [@empty_role.id.to_s])
1817 @query.add_filter('assigned_to_role', '!', [@empty_role.id.to_s])
1775
1818
1776 assert_query_result [@issue1, @issue2, @issue3, @issue4, @issue5], @query
1819 assert_query_result [@issue1, @issue2, @issue3, @issue4, @issue5], @query
1777 end
1820 end
1778
1821
1779 def test_query_column_should_accept_a_symbol_as_caption
1822 def test_query_column_should_accept_a_symbol_as_caption
1780 set_language_if_valid 'en'
1823 set_language_if_valid 'en'
1781 c = QueryColumn.new('foo', :caption => :general_text_Yes)
1824 c = QueryColumn.new('foo', :caption => :general_text_Yes)
1782 assert_equal 'Yes', c.caption
1825 assert_equal 'Yes', c.caption
1783 end
1826 end
1784
1827
1785 def test_query_column_should_accept_a_proc_as_caption
1828 def test_query_column_should_accept_a_proc_as_caption
1786 c = QueryColumn.new('foo', :caption => lambda {'Foo'})
1829 c = QueryColumn.new('foo', :caption => lambda {'Foo'})
1787 assert_equal 'Foo', c.caption
1830 assert_equal 'Foo', c.caption
1788 end
1831 end
1789
1832
1790 def test_date_clause_should_respect_user_time_zone_with_local_default
1833 def test_date_clause_should_respect_user_time_zone_with_local_default
1791 @query = IssueQuery.new(:name => '_')
1834 @query = IssueQuery.new(:name => '_')
1792
1835
1793 # user is in Hawaii (-10)
1836 # user is in Hawaii (-10)
1794 User.current = users(:users_001)
1837 User.current = users(:users_001)
1795 User.current.pref.update_attribute :time_zone, 'Hawaii'
1838 User.current.pref.update_attribute :time_zone, 'Hawaii'
1796
1839
1797 # assume timestamps are stored in server local time
1840 # assume timestamps are stored in server local time
1798 local_zone = Time.zone
1841 local_zone = Time.zone
1799
1842
1800 from = Date.parse '2016-03-20'
1843 from = Date.parse '2016-03-20'
1801 to = Date.parse '2016-03-22'
1844 to = Date.parse '2016-03-22'
1802 assert c = @query.send(:date_clause, 'table', 'field', from, to, false)
1845 assert c = @query.send(:date_clause, 'table', 'field', from, to, false)
1803
1846
1804 # the dates should have been interpreted in the user's time zone and
1847 # the dates should have been interpreted in the user's time zone and
1805 # converted to local time
1848 # converted to local time
1806 # what we get exactly in the sql depends on the local time zone, therefore
1849 # what we get exactly in the sql depends on the local time zone, therefore
1807 # it's computed here.
1850 # it's computed here.
1808 f = User.current.time_zone.local(from.year, from.month, from.day).yesterday.end_of_day.in_time_zone(local_zone)
1851 f = User.current.time_zone.local(from.year, from.month, from.day).yesterday.end_of_day.in_time_zone(local_zone)
1809 t = User.current.time_zone.local(to.year, to.month, to.day).end_of_day.in_time_zone(local_zone)
1852 t = User.current.time_zone.local(to.year, to.month, to.day).end_of_day.in_time_zone(local_zone)
1810 assert_equal "table.field > '#{Query.connection.quoted_date f}' AND table.field <= '#{Query.connection.quoted_date t}'", c
1853 assert_equal "table.field > '#{Query.connection.quoted_date f}' AND table.field <= '#{Query.connection.quoted_date t}'", c
1811 end
1854 end
1812
1855
1813 def test_date_clause_should_respect_user_time_zone_with_utc_default
1856 def test_date_clause_should_respect_user_time_zone_with_utc_default
1814 @query = IssueQuery.new(:name => '_')
1857 @query = IssueQuery.new(:name => '_')
1815
1858
1816 # user is in Hawaii (-10)
1859 # user is in Hawaii (-10)
1817 User.current = users(:users_001)
1860 User.current = users(:users_001)
1818 User.current.pref.update_attribute :time_zone, 'Hawaii'
1861 User.current.pref.update_attribute :time_zone, 'Hawaii'
1819
1862
1820 # assume timestamps are stored as utc
1863 # assume timestamps are stored as utc
1821 ActiveRecord::Base.default_timezone = :utc
1864 ActiveRecord::Base.default_timezone = :utc
1822
1865
1823 from = Date.parse '2016-03-20'
1866 from = Date.parse '2016-03-20'
1824 to = Date.parse '2016-03-22'
1867 to = Date.parse '2016-03-22'
1825 assert c = @query.send(:date_clause, 'table', 'field', from, to, false)
1868 assert c = @query.send(:date_clause, 'table', 'field', from, to, false)
1826 # the dates should have been interpreted in the user's time zone and
1869 # the dates should have been interpreted in the user's time zone and
1827 # converted to utc. March 20 in Hawaii begins at 10am UTC.
1870 # converted to utc. March 20 in Hawaii begins at 10am UTC.
1828 f = Time.new(2016, 3, 20, 9, 59, 59, 0).end_of_hour
1871 f = Time.new(2016, 3, 20, 9, 59, 59, 0).end_of_hour
1829 t = Time.new(2016, 3, 23, 9, 59, 59, 0).end_of_hour
1872 t = Time.new(2016, 3, 23, 9, 59, 59, 0).end_of_hour
1830 assert_equal "table.field > '#{Query.connection.quoted_date f}' AND table.field <= '#{Query.connection.quoted_date t}'", c
1873 assert_equal "table.field > '#{Query.connection.quoted_date f}' AND table.field <= '#{Query.connection.quoted_date t}'", c
1831 ensure
1874 ensure
1832 ActiveRecord::Base.default_timezone = :local # restore Redmine default
1875 ActiveRecord::Base.default_timezone = :local # restore Redmine default
1833 end
1876 end
1834
1877
1835 end
1878 end
General Comments 0
You need to be logged in to leave comments. Login now