formtastic 1.2.5 → 2.0.0.rc1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (134) hide show
  1. data/.gitignore +13 -0
  2. data/.rspec +2 -0
  3. data/.yardopts +1 -0
  4. data/CHANGELOG +279 -0
  5. data/Gemfile +3 -0
  6. data/README.textile +155 -172
  7. data/RELEASE_PROCESS +7 -0
  8. data/Rakefile +52 -0
  9. data/app/assets/stylesheets/formtastic.css +275 -0
  10. data/app/assets/stylesheets/formtastic_ie6.css +27 -0
  11. data/app/assets/stylesheets/formtastic_ie7.css +17 -0
  12. data/formtastic.gemspec +51 -0
  13. data/lib/formtastic.rb +21 -1960
  14. data/lib/formtastic/engine.rb +7 -0
  15. data/lib/formtastic/form_builder.rb +83 -0
  16. data/lib/formtastic/helpers.rb +16 -0
  17. data/lib/formtastic/helpers/buttons_helper.rb +277 -0
  18. data/lib/formtastic/helpers/errors_helper.rb +113 -0
  19. data/lib/formtastic/helpers/fieldset_wrapper.rb +75 -0
  20. data/lib/formtastic/helpers/file_column_detection.rb +16 -0
  21. data/lib/formtastic/helpers/form_helper.rb +198 -0
  22. data/lib/formtastic/helpers/input_helper.rb +366 -0
  23. data/lib/formtastic/helpers/inputs_helper.rb +392 -0
  24. data/lib/formtastic/helpers/reflection.rb +33 -0
  25. data/lib/formtastic/helpers/semantic_form_helper.rb +11 -0
  26. data/lib/formtastic/html_attributes.rb +21 -0
  27. data/lib/formtastic/i18n.rb +1 -0
  28. data/lib/formtastic/inputs.rb +31 -0
  29. data/lib/formtastic/inputs/base.rb +61 -0
  30. data/lib/formtastic/inputs/base/associations.rb +31 -0
  31. data/lib/formtastic/inputs/base/choices.rb +103 -0
  32. data/lib/formtastic/inputs/base/collections.rb +94 -0
  33. data/lib/formtastic/inputs/base/database.rb +17 -0
  34. data/lib/formtastic/inputs/base/errors.rb +58 -0
  35. data/lib/formtastic/inputs/base/fileish.rb +23 -0
  36. data/lib/formtastic/inputs/base/grouped_collections.rb +77 -0
  37. data/lib/formtastic/inputs/base/hints.rb +31 -0
  38. data/lib/formtastic/inputs/base/html.rb +52 -0
  39. data/lib/formtastic/inputs/base/labelling.rb +55 -0
  40. data/lib/formtastic/inputs/base/naming.rb +42 -0
  41. data/lib/formtastic/inputs/base/options.rb +18 -0
  42. data/lib/formtastic/inputs/base/stringish.rb +35 -0
  43. data/lib/formtastic/inputs/base/timeish.rb +128 -0
  44. data/lib/formtastic/inputs/base/validations.rb +166 -0
  45. data/lib/formtastic/inputs/base/wrapping.rb +40 -0
  46. data/lib/formtastic/inputs/boolean_input.rb +96 -0
  47. data/lib/formtastic/inputs/check_boxes_input.rb +179 -0
  48. data/lib/formtastic/inputs/country_input.rb +66 -0
  49. data/lib/formtastic/inputs/date_input.rb +14 -0
  50. data/lib/formtastic/inputs/datetime_input.rb +9 -0
  51. data/lib/formtastic/inputs/email_input.rb +40 -0
  52. data/lib/formtastic/inputs/file_input.rb +42 -0
  53. data/lib/formtastic/inputs/hidden_input.rb +66 -0
  54. data/lib/formtastic/inputs/number_input.rb +118 -0
  55. data/lib/formtastic/inputs/numeric_input.rb +21 -0
  56. data/lib/formtastic/inputs/password_input.rb +40 -0
  57. data/lib/formtastic/inputs/phone_input.rb +41 -0
  58. data/lib/formtastic/inputs/radio_input.rb +157 -0
  59. data/lib/formtastic/inputs/range_input.rb +119 -0
  60. data/lib/formtastic/inputs/search_input.rb +40 -0
  61. data/lib/formtastic/inputs/select_input.rb +210 -0
  62. data/lib/formtastic/inputs/string_input.rb +34 -0
  63. data/lib/formtastic/inputs/text_input.rb +47 -0
  64. data/lib/formtastic/inputs/time_input.rb +14 -0
  65. data/lib/formtastic/inputs/time_zone_input.rb +48 -0
  66. data/lib/formtastic/inputs/url_input.rb +40 -0
  67. data/lib/formtastic/localized_string.rb +105 -0
  68. data/lib/formtastic/railtie.rb +5 -7
  69. data/lib/formtastic/semantic_form_builder.rb +11 -0
  70. data/lib/formtastic/util.rb +6 -19
  71. data/lib/formtastic/version.rb +3 -0
  72. data/lib/generators/formtastic/install/install_generator.rb +28 -6
  73. data/lib/generators/templates/_form.html.erb +10 -4
  74. data/lib/generators/templates/_form.html.haml +8 -4
  75. data/lib/generators/templates/formtastic.rb +25 -32
  76. data/lib/locale/en.yml +1 -0
  77. data/lib/tasks/verify_rcov.rb +44 -0
  78. data/sample/basic_inputs.html +182 -0
  79. data/sample/config.ru +69 -0
  80. data/sample/index.html +14 -0
  81. data/spec/builder/custom_builder_spec.rb +109 -0
  82. data/spec/builder/error_proc_spec.rb +27 -0
  83. data/spec/builder/errors_spec.rb +193 -0
  84. data/spec/builder/semantic_fields_for_spec.rb +88 -0
  85. data/spec/helpers/buttons_helper_spec.rb +150 -0
  86. data/spec/helpers/commit_button_helper_spec.rb +470 -0
  87. data/spec/helpers/form_helper_spec.rb +135 -0
  88. data/spec/helpers/input_helper_spec.rb +837 -0
  89. data/spec/helpers/inputs_helper_spec.rb +562 -0
  90. data/spec/helpers/semantic_errors_helper_spec.rb +112 -0
  91. data/spec/i18n_spec.rb +199 -0
  92. data/spec/inputs/boolean_input_spec.rb +184 -0
  93. data/spec/inputs/check_boxes_input_spec.rb +375 -0
  94. data/spec/inputs/country_input_spec.rb +133 -0
  95. data/spec/inputs/custom_input_spec.rb +52 -0
  96. data/spec/inputs/date_input_spec.rb +110 -0
  97. data/spec/inputs/datetime_input_spec.rb +115 -0
  98. data/spec/inputs/email_input_spec.rb +55 -0
  99. data/spec/inputs/file_input_spec.rb +59 -0
  100. data/spec/inputs/hidden_input_spec.rb +120 -0
  101. data/spec/inputs/include_blank_spec.rb +70 -0
  102. data/spec/inputs/label_spec.rb +104 -0
  103. data/spec/inputs/number_input_spec.rb +487 -0
  104. data/spec/inputs/numeric_input_spec.rb +41 -0
  105. data/spec/inputs/password_input_spec.rb +69 -0
  106. data/spec/inputs/phone_input_spec.rb +55 -0
  107. data/spec/inputs/placeholder_spec.rb +71 -0
  108. data/spec/inputs/radio_input_spec.rb +234 -0
  109. data/spec/inputs/range_input_spec.rb +477 -0
  110. data/spec/inputs/search_input_spec.rb +55 -0
  111. data/spec/inputs/select_input_spec.rb +545 -0
  112. data/spec/inputs/string_input_spec.rb +163 -0
  113. data/spec/inputs/text_input_spec.rb +158 -0
  114. data/spec/inputs/time_input_spec.rb +155 -0
  115. data/spec/inputs/time_zone_input_spec.rb +87 -0
  116. data/spec/inputs/url_input_spec.rb +55 -0
  117. data/spec/spec.opts +2 -0
  118. data/spec/spec_helper.rb +361 -0
  119. data/spec/support/custom_macros.rb +656 -0
  120. data/spec/support/deferred_garbage_collection.rb +21 -0
  121. data/spec/support/deprecation.rb +6 -0
  122. data/spec/support/test_environment.rb +30 -0
  123. metadata +306 -88
  124. data/generators/form/USAGE +0 -16
  125. data/generators/form/form_generator.rb +0 -111
  126. data/generators/formtastic/formtastic_generator.rb +0 -26
  127. data/init.rb +0 -5
  128. data/lib/formtastic/layout_helper.rb +0 -12
  129. data/lib/generators/formtastic/form/form_generator.rb +0 -84
  130. data/lib/generators/templates/formtastic.css +0 -145
  131. data/lib/generators/templates/formtastic_changes.css +0 -14
  132. data/lib/generators/templates/rails2/_form.html.erb +0 -5
  133. data/lib/generators/templates/rails2/_form.html.haml +0 -4
  134. data/rails/init.rb +0 -2
@@ -1,6 +1,7 @@
1
1
  # encoding: utf-8
2
2
 
3
3
  module Formtastic
4
+ # @private
4
5
  module I18n
5
6
 
6
7
  DEFAULT_SCOPE = [:formtastic].freeze
@@ -0,0 +1,31 @@
1
+ module Formtastic
2
+ module Inputs
3
+ extend ActiveSupport::Autoload
4
+
5
+ autoload :Base
6
+ autoload :Basic
7
+ autoload :BooleanInput
8
+ autoload :CheckBoxesInput
9
+ autoload :CountryInput
10
+ autoload :DateInput
11
+ autoload :DatetimeInput
12
+ autoload :EmailInput
13
+ autoload :FileInput
14
+ autoload :HiddenInput
15
+ autoload :NumberInput
16
+ autoload :NumericInput
17
+ autoload :PasswordInput
18
+ autoload :PhoneInput
19
+ autoload :RadioInput
20
+ autoload :RangeInput
21
+ autoload :SearchInput
22
+ autoload :SelectInput
23
+ autoload :StringInput
24
+ autoload :TextInput
25
+ autoload :TimeInput
26
+ autoload :TimeZoneInput
27
+ autoload :Timeish
28
+ autoload :UrlInput
29
+ end
30
+ end
31
+
@@ -0,0 +1,61 @@
1
+ module Formtastic
2
+ module Inputs
3
+ module Base
4
+
5
+ attr_accessor :builder, :template, :object, :object_name, :method, :options
6
+
7
+ def initialize(builder, template, object, object_name, method, options)
8
+ @builder = builder
9
+ @template = template
10
+ @object = object
11
+ @object_name = object_name
12
+ @method = method
13
+ @options = options.dup
14
+
15
+ warn_and_correct_option!(:label_method, :member_label)
16
+ warn_and_correct_option!(:value_method, :member_value)
17
+ warn_and_correct_option!(:group_label_method, :group_label)
18
+ end
19
+
20
+ def warn_and_correct_option!(old_option_name, new_option_name)
21
+ if options.key?(old_option_name)
22
+ ::ActiveSupport::Deprecation.warn("The :#{old_option_name} option is deprecated in favour of :#{new_option_name} and will be removed from Formtastic after 2.0")
23
+ options[new_option_name] = options.delete(old_option_name)
24
+ end
25
+ end
26
+
27
+ extend ActiveSupport::Autoload
28
+
29
+ autoload :Associations
30
+ autoload :Collections
31
+ autoload :Choices
32
+ autoload :Database
33
+ autoload :Errors
34
+ autoload :Fileish
35
+ autoload :GroupedCollections
36
+ autoload :Hints
37
+ autoload :Html
38
+ autoload :Labelling
39
+ autoload :Naming
40
+ autoload :Options
41
+ autoload :Stringish
42
+ autoload :Timeish
43
+ autoload :Validations
44
+ autoload :Wrapping
45
+
46
+ include Html
47
+ include Options
48
+ include Database
49
+ include Errors
50
+ include Hints
51
+ include Naming
52
+ include Validations
53
+ include Fileish
54
+ include Associations
55
+ include Labelling
56
+ include Wrapping
57
+
58
+ end
59
+ end
60
+ end
61
+
@@ -0,0 +1,31 @@
1
+ module Formtastic
2
+ module Inputs
3
+ module Base
4
+ module Associations
5
+ include Formtastic::Helpers::Reflection
6
+
7
+ # :belongs_to, etc
8
+ def association
9
+ @association ||= association_macro_for_method(method)
10
+ end
11
+
12
+ def reflection
13
+ @reflection ||= reflection_for(method)
14
+ end
15
+
16
+ def belongs_to?
17
+ association == :belongs_to
18
+ end
19
+
20
+ def has_many?
21
+ association == :has_many
22
+ end
23
+
24
+ def association_primary_key
25
+ association_primary_key_for_method(method)
26
+ end
27
+
28
+ end
29
+ end
30
+ end
31
+ end
@@ -0,0 +1,103 @@
1
+ module Formtastic
2
+ module Inputs
3
+ module Base
4
+ module Choices
5
+
6
+ def choices_wrapping(&block)
7
+ template.content_tag(:fieldset,
8
+ template.capture(&block),
9
+ choices_wrapping_html_options
10
+ )
11
+ end
12
+
13
+ def choices_wrapping_html_options
14
+ { :class => "choices" }
15
+ end
16
+
17
+ def choices_group_wrapping(&block)
18
+ template.content_tag(:ol,
19
+ template.capture(&block),
20
+ choices_group_wrapping_html_options
21
+ )
22
+ end
23
+
24
+ def choices_group_wrapping_html_options
25
+ { :class => "choices-group" }
26
+ end
27
+
28
+ def choice_wrapping(html_options, &block)
29
+ template.content_tag(:li,
30
+ template.capture(&block),
31
+ html_options
32
+ )
33
+ end
34
+
35
+ def choice_wrapping_html_options(choice)
36
+ classes = ['choice']
37
+ classes << "#{sanitized_method_name.singularize}_#{choice_html_safe_value(choice)}" if value_as_class?
38
+
39
+ { :class => classes.join(" ") }
40
+ end
41
+
42
+ def choice_html(choice)
43
+ raise "choice_html() needs to be implemented when including Formtastic::Inputs::Base::Choices"
44
+ end
45
+
46
+ def choice_label(choice)
47
+ choice.is_a?(Array) ? choice.first : choice
48
+ end
49
+
50
+ def choice_value(choice)
51
+ choice.is_a?(Array) ? choice[1] : choice
52
+ end
53
+
54
+ def choice_html_options(choice)
55
+ custom_choice_html_options(choice).merge(default_choice_html_options(choice))
56
+ end
57
+
58
+ def default_choice_html_options(choice)
59
+ { :id => choice_input_dom_id(choice) }
60
+ end
61
+
62
+ def custom_choice_html_options(choice)
63
+ (choice.is_a?(Array) && choice.size > 2) ? choice.last : {}
64
+ end
65
+
66
+ def choice_html_safe_value(choice)
67
+ choice_value(choice).to_s.gsub(/\s/, '_').gsub(/[^\w-]/, '').downcase
68
+ end
69
+
70
+ def choice_input_dom_id(choice)
71
+ [
72
+ builder.custom_namespace,
73
+ sanitized_object_name,
74
+ association_primary_key || method,
75
+ choice_html_safe_value(choice)
76
+ ].compact.reject { |i| i.blank? }.join("_")
77
+ end
78
+
79
+ def value_as_class?
80
+ options[:value_as_class]
81
+ end
82
+
83
+ def legend_html
84
+ if render_label?
85
+ template.content_tag(:legend,
86
+ template.content_tag(:label, label_text),
87
+ label_html_options.merge(:class => "label")
88
+ )
89
+ else
90
+ "".html_safe
91
+ end
92
+ end
93
+
94
+ # Override to remove the for attribute since this isn't associated with any element, as it's
95
+ # nested inside the legend.
96
+ def label_html_options
97
+ super.merge(:for => nil)
98
+ end
99
+
100
+ end
101
+ end
102
+ end
103
+ end
@@ -0,0 +1,94 @@
1
+ module Formtastic
2
+ module Inputs
3
+ module Base
4
+ module Collections
5
+
6
+ def label_method
7
+ label_and_value_method(raw_collection).first
8
+ end
9
+
10
+ def value_method
11
+ label_and_value_method(raw_collection).last
12
+ end
13
+
14
+ def label_and_value_method(_collection, grouped=false)
15
+ sample = _collection.first || _collection.last
16
+
17
+ case sample
18
+ when Array
19
+ label, value = :first, :last
20
+ when Integer
21
+ label, value = :to_s, :to_i
22
+ when String, NilClass
23
+ label, value = :to_s, :to_s
24
+ end
25
+
26
+ # Order of preference: user supplied method, class defaults, auto-detect
27
+ label = (grouped ? options[:grouped_label_method] : options[:member_label]) || label || builder.collection_label_methods.find { |m| sample.respond_to?(m) }
28
+ value = (grouped ? options[:grouped_value_method] : options[:member_value]) || value || builder.collection_value_methods.find { |m| sample.respond_to?(m) }
29
+
30
+ [label, value]
31
+ end
32
+
33
+ def raw_collection
34
+ @raw_collection ||= (collection_from_options || collection_from_association || collection_for_boolean)
35
+ end
36
+
37
+ def collection
38
+ # Return if we have a plain string
39
+ return raw_collection if raw_collection.instance_of?(String) || raw_collection.instance_of?(ActiveSupport::SafeBuffer)
40
+
41
+ # Return if we have an Array of strings, fixnums or arrays
42
+ return raw_collection if (raw_collection.instance_of?(Array) || raw_collection.instance_of?(Range)) &&
43
+ [Array, Fixnum, String, Symbol].include?(raw_collection.first.class) &&
44
+ !(options.include?(:member_label) || options.include?(:member_value))
45
+
46
+ raw_collection.map { |o| [send_or_call(label_method, o), send_or_call(value_method, o)] }
47
+ end
48
+
49
+ def collection_from_options
50
+ items = options[:collection]
51
+ items = items.to_a if items.is_a?(Hash)
52
+ items
53
+ end
54
+
55
+ def collection_from_association
56
+ if reflection
57
+ raise PolymorphicInputWithoutCollectionError.new("A collection must be supplied for #{method} input. Collections cannot be guessed for polymorphic associations.") if reflection.options && reflection.options[:polymorphic] == true
58
+
59
+ find_options_from_options = options[:find_options] || {}
60
+ conditions_from_options = find_options_from_options[:conditions] || {}
61
+ conditions_from_reflection = reflection.options && reflection.options[:conditions] || {}
62
+
63
+ if conditions_from_options.any?
64
+ reflection.klass.where(
65
+ conditions_from_reflection.merge(conditions_from_options)
66
+ )
67
+ else
68
+ find_options_from_options.merge!(:include => group_by) if self.respond_to?(:group_by) && group_by
69
+ reflection.klass.where(conditions_from_reflection.merge(find_options_from_options))
70
+ end
71
+ end
72
+ end
73
+
74
+ def collection_for_boolean
75
+ true_text = options[:true] || Formtastic::I18n.t(:yes)
76
+ false_text = options[:false] || Formtastic::I18n.t(:no)
77
+
78
+ # TODO options[:value_as_class] = true unless options.key?(:value_as_class)
79
+
80
+ [ [true_text, true], [false_text, false] ]
81
+ end
82
+
83
+ def send_or_call(duck, object)
84
+ if duck.respond_to?(:call)
85
+ duck.call(object)
86
+ else
87
+ object.send(duck)
88
+ end
89
+ end
90
+
91
+ end
92
+ end
93
+ end
94
+ end
@@ -0,0 +1,17 @@
1
+ module Formtastic
2
+ module Inputs
3
+ module Base
4
+ module Database
5
+
6
+ def column
7
+ object.column_for_attribute(method) if object.respond_to?(:column_for_attribute)
8
+ end
9
+
10
+ def column?
11
+ !column.nil?
12
+ end
13
+
14
+ end
15
+ end
16
+ end
17
+ end
@@ -0,0 +1,58 @@
1
+ module Formtastic
2
+ module Inputs
3
+ module Base
4
+ module Errors
5
+
6
+ def error_html
7
+ errors? ? send(:"error_#{builder.inline_errors}_html") : ""
8
+ end
9
+
10
+ def error_sentence_html
11
+ error_class = options[:error_class] || builder.default_inline_error_class
12
+ template.content_tag(:p, Formtastic::Util.html_safe(errors.to_sentence.html_safe), :class => error_class)
13
+ end
14
+
15
+ def error_list_html
16
+ error_class = options[:error_class] || builder.default_error_list_class
17
+ list_elements = []
18
+ errors.each do |error|
19
+ list_elements << template.content_tag(:li, Formtastic::Util.html_safe(error.html_safe))
20
+ end
21
+ template.content_tag(:ul, Formtastic::Util.html_safe(list_elements.join("\n")), :class => error_class)
22
+ end
23
+
24
+ def error_first_html
25
+ error_class = options[:error_class] || builder.default_inline_error_class
26
+ template.content_tag(:p, Formtastic::Util.html_safe(errors.first.untaint), :class => error_class)
27
+ end
28
+
29
+ def error_none_html
30
+ ""
31
+ end
32
+
33
+ def errors?
34
+ !errors.blank?
35
+ end
36
+
37
+ def errors
38
+ errors = []
39
+ if object && object.respond_to?(:errors)
40
+ error_keys.each do |key|
41
+ errors << object.errors[key] unless object.errors[key].blank?
42
+ end
43
+ end
44
+ errors.flatten.compact.uniq
45
+ end
46
+
47
+ def error_keys
48
+ keys = [method.to_sym]
49
+ keys << builder.file_metadata_suffixes.map{|suffix| "#{method}_#{suffix}".to_sym} if file?
50
+ keys << association_primary_key if belongs_to? || has_many?
51
+ keys.flatten.compact.uniq
52
+ end
53
+
54
+ end
55
+ end
56
+ end
57
+ end
58
+
@@ -0,0 +1,23 @@
1
+ module Formtastic
2
+ module Inputs
3
+ module Base
4
+ module Fileish
5
+
6
+ def file?
7
+ @file ||= begin
8
+ # TODO return true if self.is_a?(Formtastic::Inputs::FileInput::Woo)
9
+ object && object.respond_to?(method) && builder.file_methods.any? { |m| object.send(method).respond_to?(m) }
10
+ end
11
+ end
12
+
13
+ end
14
+ end
15
+ end
16
+ end
17
+
18
+
19
+
20
+
21
+
22
+
23
+
@@ -0,0 +1,77 @@
1
+ module Formtastic
2
+ module Inputs
3
+ module Base
4
+ module GroupedCollections
5
+
6
+ def raw_grouped_collection
7
+ @raw_grouped_collection ||= raw_collection.map { |option| option.send(options[:group_by]) }.uniq
8
+ end
9
+
10
+ def grouped_collection
11
+ @grouped_collection ||= raw_grouped_collection.sort_by { |group_item| group_item.send(group_label_method) }
12
+ end
13
+
14
+ def group_label_method
15
+ @group_label_method ||= (group_label_method_from_options || group_label_method_from_grouped_collection)
16
+ end
17
+
18
+ def group_label_method_from_options
19
+ options[:group_label]
20
+ end
21
+
22
+ def group_label_method_from_grouped_collection
23
+ label_and_value_method(raw_grouped_collection, true).first
24
+ end
25
+
26
+ def group_association
27
+ @group_association ||= (group_association_from_options || group_association_from_reflection)
28
+ end
29
+
30
+ def group_association_from_options
31
+ options[:group_association]
32
+ end
33
+
34
+ def group_by
35
+ options[:group_by]
36
+ end
37
+
38
+ def group_association_from_reflection
39
+ method_to_group_association_by = reflection.klass.reflect_on_association(group_by)
40
+ group_class = method_to_group_association_by.klass
41
+
42
+ # This will return in the normal case
43
+ return method.to_s.pluralize.to_sym if group_class.reflect_on_association(method.to_s.pluralize)
44
+
45
+ # This is for belongs_to associations named differently than their class
46
+ # form.input :parent, :group_by => :customer
47
+ # eg.
48
+ # class Project
49
+ # belongs_to :parent, :class_name => 'Project', :foreign_key => 'parent_id'
50
+ # belongs_to :customer
51
+ # end
52
+ # class Customer
53
+ # has_many :projects
54
+ # end
55
+ group_method = group_class.to_s.underscore.pluralize.to_sym
56
+ return group_method if group_class.reflect_on_association(group_method) # :projects
57
+
58
+ # This is for has_many associations named differently than their class
59
+ # eg.
60
+ # class Project
61
+ # belongs_to :parent, :class_name => 'Project', :foreign_key => 'parent_id'
62
+ # belongs_to :customer
63
+ # end
64
+ # class Customer
65
+ # has_many :tasks, :class_name => 'Project', :foreign_key => 'customer_id'
66
+ # end
67
+ possible_associations = group_class.reflect_on_all_associations(:has_many).find_all {|assoc| assoc.klass == reflection.klass }
68
+ return possible_associations.first.name.to_sym if possible_associations.count == 1
69
+
70
+ raise "Cannot infer group association for #{method} grouped by #{group_by}, there were #{possible_associations.empty? ? 'no' : possible_associations.size} possible associations. Please specify using :group_association"
71
+ end
72
+
73
+ end
74
+ end
75
+ end
76
+ end
77
+