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