schof-searchlogic 0.0.2

Sign up to get free protection for your applications and to get access to all the features.
Files changed (159) hide show
  1. data/CHANGELOG.rdoc +302 -0
  2. data/MIT-LICENSE +20 -0
  3. data/Manifest +157 -0
  4. data/README.rdoc +461 -0
  5. data/Rakefile +13 -0
  6. data/TODO.rdoc +4 -0
  7. data/init.rb +1 -0
  8. data/lib/searchlogic.rb +100 -0
  9. data/lib/searchlogic/active_record/associations.rb +52 -0
  10. data/lib/searchlogic/active_record/base.rb +224 -0
  11. data/lib/searchlogic/active_record/connection_adapters/mysql_adapter.rb +176 -0
  12. data/lib/searchlogic/active_record/connection_adapters/postgresql_adapter.rb +172 -0
  13. data/lib/searchlogic/active_record/connection_adapters/sqlite_adapter.rb +80 -0
  14. data/lib/searchlogic/condition/base.rb +165 -0
  15. data/lib/searchlogic/condition/begins_with.rb +17 -0
  16. data/lib/searchlogic/condition/blank.rb +21 -0
  17. data/lib/searchlogic/condition/child_of.rb +11 -0
  18. data/lib/searchlogic/condition/descendant_of.rb +11 -0
  19. data/lib/searchlogic/condition/ends_with.rb +17 -0
  20. data/lib/searchlogic/condition/equals.rb +33 -0
  21. data/lib/searchlogic/condition/greater_than.rb +15 -0
  22. data/lib/searchlogic/condition/greater_than_or_equal_to.rb +15 -0
  23. data/lib/searchlogic/condition/inclusive_descendant_of.rb +10 -0
  24. data/lib/searchlogic/condition/keywords.rb +47 -0
  25. data/lib/searchlogic/condition/less_than.rb +15 -0
  26. data/lib/searchlogic/condition/less_than_or_equal_to.rb +15 -0
  27. data/lib/searchlogic/condition/like.rb +15 -0
  28. data/lib/searchlogic/condition/nested_set.rb +17 -0
  29. data/lib/searchlogic/condition/nil.rb +21 -0
  30. data/lib/searchlogic/condition/not_begin_with.rb +20 -0
  31. data/lib/searchlogic/condition/not_blank.rb +19 -0
  32. data/lib/searchlogic/condition/not_end_with.rb +20 -0
  33. data/lib/searchlogic/condition/not_equal.rb +27 -0
  34. data/lib/searchlogic/condition/not_have_keywords.rb +20 -0
  35. data/lib/searchlogic/condition/not_like.rb +20 -0
  36. data/lib/searchlogic/condition/not_nil.rb +19 -0
  37. data/lib/searchlogic/condition/sibling_of.rb +14 -0
  38. data/lib/searchlogic/conditions/any_or_all.rb +42 -0
  39. data/lib/searchlogic/conditions/base.rb +244 -0
  40. data/lib/searchlogic/conditions/groups.rb +74 -0
  41. data/lib/searchlogic/conditions/magic_methods.rb +286 -0
  42. data/lib/searchlogic/conditions/multiparameter_attributes.rb +105 -0
  43. data/lib/searchlogic/conditions/protection.rb +36 -0
  44. data/lib/searchlogic/config.rb +31 -0
  45. data/lib/searchlogic/config/helpers.rb +338 -0
  46. data/lib/searchlogic/config/search.rb +53 -0
  47. data/lib/searchlogic/core_ext/hash.rb +75 -0
  48. data/lib/searchlogic/core_ext/object.rb +19 -0
  49. data/lib/searchlogic/helpers/control_types/link.rb +310 -0
  50. data/lib/searchlogic/helpers/control_types/links.rb +242 -0
  51. data/lib/searchlogic/helpers/control_types/remote_link.rb +87 -0
  52. data/lib/searchlogic/helpers/control_types/remote_links.rb +72 -0
  53. data/lib/searchlogic/helpers/control_types/remote_select.rb +36 -0
  54. data/lib/searchlogic/helpers/control_types/select.rb +82 -0
  55. data/lib/searchlogic/helpers/form.rb +208 -0
  56. data/lib/searchlogic/helpers/utilities.rb +197 -0
  57. data/lib/searchlogic/modifiers/absolute.rb +15 -0
  58. data/lib/searchlogic/modifiers/acos.rb +11 -0
  59. data/lib/searchlogic/modifiers/asin.rb +11 -0
  60. data/lib/searchlogic/modifiers/atan.rb +11 -0
  61. data/lib/searchlogic/modifiers/avg.rb +15 -0
  62. data/lib/searchlogic/modifiers/base.rb +27 -0
  63. data/lib/searchlogic/modifiers/ceil.rb +15 -0
  64. data/lib/searchlogic/modifiers/char_length.rb +15 -0
  65. data/lib/searchlogic/modifiers/cos.rb +15 -0
  66. data/lib/searchlogic/modifiers/cot.rb +15 -0
  67. data/lib/searchlogic/modifiers/count.rb +11 -0
  68. data/lib/searchlogic/modifiers/day_of_month.rb +15 -0
  69. data/lib/searchlogic/modifiers/day_of_week.rb +15 -0
  70. data/lib/searchlogic/modifiers/day_of_year.rb +15 -0
  71. data/lib/searchlogic/modifiers/degrees.rb +11 -0
  72. data/lib/searchlogic/modifiers/exp.rb +15 -0
  73. data/lib/searchlogic/modifiers/floor.rb +15 -0
  74. data/lib/searchlogic/modifiers/hex.rb +11 -0
  75. data/lib/searchlogic/modifiers/hour.rb +11 -0
  76. data/lib/searchlogic/modifiers/log.rb +15 -0
  77. data/lib/searchlogic/modifiers/log10.rb +11 -0
  78. data/lib/searchlogic/modifiers/log2.rb +11 -0
  79. data/lib/searchlogic/modifiers/lower.rb +15 -0
  80. data/lib/searchlogic/modifiers/ltrim.rb +15 -0
  81. data/lib/searchlogic/modifiers/md5.rb +11 -0
  82. data/lib/searchlogic/modifiers/microseconds.rb +11 -0
  83. data/lib/searchlogic/modifiers/milliseconds.rb +11 -0
  84. data/lib/searchlogic/modifiers/minute.rb +15 -0
  85. data/lib/searchlogic/modifiers/month.rb +15 -0
  86. data/lib/searchlogic/modifiers/octal.rb +15 -0
  87. data/lib/searchlogic/modifiers/radians.rb +11 -0
  88. data/lib/searchlogic/modifiers/round.rb +11 -0
  89. data/lib/searchlogic/modifiers/rtrim.rb +15 -0
  90. data/lib/searchlogic/modifiers/second.rb +15 -0
  91. data/lib/searchlogic/modifiers/sign.rb +11 -0
  92. data/lib/searchlogic/modifiers/sin.rb +11 -0
  93. data/lib/searchlogic/modifiers/square_root.rb +15 -0
  94. data/lib/searchlogic/modifiers/sum.rb +11 -0
  95. data/lib/searchlogic/modifiers/tan.rb +15 -0
  96. data/lib/searchlogic/modifiers/trim.rb +15 -0
  97. data/lib/searchlogic/modifiers/upper.rb +15 -0
  98. data/lib/searchlogic/modifiers/week.rb +11 -0
  99. data/lib/searchlogic/modifiers/year.rb +11 -0
  100. data/lib/searchlogic/search/base.rb +148 -0
  101. data/lib/searchlogic/search/conditions.rb +53 -0
  102. data/lib/searchlogic/search/ordering.rb +244 -0
  103. data/lib/searchlogic/search/pagination.rb +121 -0
  104. data/lib/searchlogic/search/protection.rb +89 -0
  105. data/lib/searchlogic/search/searching.rb +32 -0
  106. data/lib/searchlogic/shared/utilities.rb +56 -0
  107. data/lib/searchlogic/shared/virtual_classes.rb +39 -0
  108. data/lib/searchlogic/version.rb +79 -0
  109. data/searchlogic.gemspec +41 -0
  110. data/test/active_record_tests/associations_test.rb +94 -0
  111. data/test/active_record_tests/base_test.rb +115 -0
  112. data/test/condition_tests/base_test.rb +54 -0
  113. data/test/condition_tests/begins_with_test.rb +11 -0
  114. data/test/condition_tests/blank_test.rb +31 -0
  115. data/test/condition_tests/child_of_test.rb +17 -0
  116. data/test/condition_tests/descendant_of_test.rb +12 -0
  117. data/test/condition_tests/ends_with_test.rb +11 -0
  118. data/test/condition_tests/equals_test.rb +28 -0
  119. data/test/condition_tests/greater_than_or_equal_to_test.rb +11 -0
  120. data/test/condition_tests/greater_than_test.rb +11 -0
  121. data/test/condition_tests/inclusive_descendant_of_test.rb +12 -0
  122. data/test/condition_tests/keywords_test.rb +23 -0
  123. data/test/condition_tests/less_than_or_equal_to_test.rb +11 -0
  124. data/test/condition_tests/less_than_test.rb +11 -0
  125. data/test/condition_tests/like_test.rb +11 -0
  126. data/test/condition_tests/nil_test.rb +31 -0
  127. data/test/condition_tests/not_begin_with_test.rb +8 -0
  128. data/test/condition_tests/not_blank_test.rb +8 -0
  129. data/test/condition_tests/not_end_with_test.rb +8 -0
  130. data/test/condition_tests/not_equal_test.rb +19 -0
  131. data/test/condition_tests/not_have_keywords_test.rb +8 -0
  132. data/test/condition_tests/not_like_test.rb +8 -0
  133. data/test/condition_tests/not_nil_test.rb +13 -0
  134. data/test/condition_tests/sibling_of_test.rb +15 -0
  135. data/test/conditions_tests/any_or_all_test.rb +23 -0
  136. data/test/conditions_tests/base_test.rb +185 -0
  137. data/test/conditions_tests/groups_test.rb +68 -0
  138. data/test/conditions_tests/magic_methods_test.rb +36 -0
  139. data/test/conditions_tests/multiparameter_attributes_test.rb +15 -0
  140. data/test/conditions_tests/protection_test.rb +18 -0
  141. data/test/config_test.rb +23 -0
  142. data/test/fixtures/accounts.yml +12 -0
  143. data/test/fixtures/animals.yml +7 -0
  144. data/test/fixtures/orders.yml +12 -0
  145. data/test/fixtures/user_groups.yml +5 -0
  146. data/test/fixtures/users.yml +45 -0
  147. data/test/libs/awesome_nested_set.rb +545 -0
  148. data/test/libs/awesome_nested_set/compatability.rb +29 -0
  149. data/test/libs/awesome_nested_set/helper.rb +40 -0
  150. data/test/libs/awesome_nested_set/named_scope.rb +140 -0
  151. data/test/libs/rexml_fix.rb +14 -0
  152. data/test/modifier_tests/day_of_month_test.rb +16 -0
  153. data/test/search_tests/base_test.rb +241 -0
  154. data/test/search_tests/conditions_test.rb +21 -0
  155. data/test/search_tests/ordering_test.rb +167 -0
  156. data/test/search_tests/pagination_test.rb +74 -0
  157. data/test/search_tests/protection_test.rb +26 -0
  158. data/test/test_helper.rb +116 -0
  159. metadata +385 -0
@@ -0,0 +1,15 @@
1
+ module Searchlogic
2
+ module Modifiers
3
+ class Second < Base
4
+ class << self
5
+ def modifier_names
6
+ super + ["sec"]
7
+ end
8
+
9
+ def return_type
10
+ :integer
11
+ end
12
+ end
13
+ end
14
+ end
15
+ end
@@ -0,0 +1,11 @@
1
+ module Searchlogic
2
+ module Modifiers
3
+ class Sign < Base
4
+ class << self
5
+ def return_type
6
+ :float
7
+ end
8
+ end
9
+ end
10
+ end
11
+ end
@@ -0,0 +1,11 @@
1
+ module Searchlogic
2
+ module Modifiers
3
+ class Sin < Base
4
+ class << self
5
+ def return_type
6
+ :float
7
+ end
8
+ end
9
+ end
10
+ end
11
+ end
@@ -0,0 +1,15 @@
1
+ module Searchlogic
2
+ module Modifiers
3
+ class SquareRoot < Base
4
+ class << self
5
+ def modifier_names
6
+ super + ["sqrt", "sq_rt"]
7
+ end
8
+
9
+ def return_type
10
+ :float
11
+ end
12
+ end
13
+ end
14
+ end
15
+ end
@@ -0,0 +1,11 @@
1
+ module Searchlogic
2
+ module Modifiers
3
+ class Sum < Base
4
+ class << self
5
+ def return_type
6
+ :float
7
+ end
8
+ end
9
+ end
10
+ end
11
+ end
@@ -0,0 +1,15 @@
1
+ module Searchlogic
2
+ module Modifiers
3
+ class Tan < Base
4
+ class << self
5
+ def modifier_names
6
+ super + ["tangent"]
7
+ end
8
+
9
+ def return_type
10
+ :float
11
+ end
12
+ end
13
+ end
14
+ end
15
+ end
@@ -0,0 +1,15 @@
1
+ module Searchlogic
2
+ module Modifiers
3
+ class Trim < Base
4
+ class << self
5
+ def modifier_names
6
+ super + ["strip"]
7
+ end
8
+
9
+ def return_type
10
+ :string
11
+ end
12
+ end
13
+ end
14
+ end
15
+ end
@@ -0,0 +1,15 @@
1
+ module Searchlogic
2
+ module Modifiers
3
+ class Upper < Base
4
+ class << self
5
+ def modifier_names
6
+ super + ["upcase", "ucase"]
7
+ end
8
+
9
+ def return_type
10
+ :string
11
+ end
12
+ end
13
+ end
14
+ end
15
+ end
@@ -0,0 +1,11 @@
1
+ module Searchlogic
2
+ module Modifiers
3
+ class Week < Base
4
+ class << self
5
+ def return_type
6
+ :integer
7
+ end
8
+ end
9
+ end
10
+ end
11
+ end
@@ -0,0 +1,11 @@
1
+ module Searchlogic
2
+ module Modifiers
3
+ class Year < Base
4
+ class << self
5
+ def return_type
6
+ :integer
7
+ end
8
+ end
9
+ end
10
+ end
11
+ end
@@ -0,0 +1,148 @@
1
+ module Searchlogic #:nodoc:
2
+ module Search #:nodoc:
3
+ # = Searchlogic
4
+ #
5
+ # Please refer the README.rdoc for usage, examples, and installation.
6
+ class Base
7
+ include Searchlogic::Shared::Utilities
8
+ include Searchlogic::Shared::VirtualClasses
9
+
10
+ # Options ActiveRecord allows when searching
11
+ AR_FIND_OPTIONS = ::ActiveRecord::Base.valid_find_options
12
+
13
+ # Options ActiveRecord allows when performing calculations
14
+ AR_CALCULATIONS_OPTIONS = (::ActiveRecord::Base.valid_calculations_options - [:select, :limit, :offset, :order, :group, :include])
15
+
16
+ AR_OPTIONS = (AR_FIND_OPTIONS + AR_CALCULATIONS_OPTIONS).uniq
17
+
18
+ # Options that ActiveRecord doesn't suppport, but Searchlogic does
19
+ SPECIAL_FIND_OPTIONS = [:order_by, :order_as, :page, :per_page, :priority_order, :priority_order_by, :priority_order_as]
20
+
21
+ # Valid options you can use when searching
22
+ OPTIONS = SPECIAL_FIND_OPTIONS + AR_OPTIONS # the order is very important, these options get set in this order
23
+
24
+ attr_accessor *AR_OPTIONS
25
+ attr_writer :scope
26
+
27
+ class << self
28
+ # Used in the ActiveRecord methods to determine if Searchlogic should get involved or not.
29
+ # This keeps Searchlogic out of the way unless it is needed.
30
+ def needed?(model_class, options)
31
+ return false if options.blank?
32
+
33
+ SPECIAL_FIND_OPTIONS.each do |option|
34
+ return true if options.symbolize_keys.keys.include?(option)
35
+ end
36
+
37
+ Searchlogic::Conditions::Base.needed?(model_class, options[:conditions])
38
+ end
39
+ end
40
+
41
+ def initialize(init_options = {})
42
+ self.options = init_options
43
+ end
44
+
45
+ # Flag to determine if searchlogic is acting as a filter for the ActiveRecord search methods.
46
+ # By filter it means that searchlogic is being used on the default ActiveRecord search methods, like all, count, find(:all), first, etc.
47
+ def acting_as_filter=(value)
48
+ @acting_as_filter = value
49
+ end
50
+
51
+ # See acting_as_filter=
52
+ def acting_as_filter?
53
+ @acting_as_filter == true
54
+ end
55
+
56
+ # Specific implementation of cloning
57
+ def clone
58
+ options = {}
59
+ (AR_OPTIONS - [:conditions]).each { |option| options[option] = instance_variable_get("@#{option}") }
60
+ options[:conditions] = conditions.conditions
61
+ SPECIAL_FIND_OPTIONS.each { |option| options[option] = send(option) }
62
+ obj = self.class.new(options)
63
+ obj.protect = protected?
64
+ obj.scope = scope
65
+ obj
66
+ end
67
+ alias_method :dup, :clone
68
+
69
+ # Makes using searchlogic in the console less annoying and keeps the output meaningful and useful
70
+ def inspect
71
+ current_find_options = {}
72
+ (AR_OPTIONS - [:conditions]).each do |option|
73
+ value = send(option)
74
+ next if value.nil?
75
+ current_find_options[option] = value
76
+ end
77
+ conditions_value = conditions.conditions
78
+ current_find_options[:conditions] = conditions_value unless conditions_value.blank?
79
+ current_find_options[:scope] = scope unless scope.blank?
80
+ "#<#{klass}Search #{current_find_options.inspect}>"
81
+ end
82
+
83
+ # Merges all joins together, including the scopes joins for
84
+ def joins
85
+ all_joins = (safe_to_array(conditions.auto_joins) + safe_to_array(order_by_auto_joins) + safe_to_array(priority_order_by_auto_joins) + safe_to_array(@joins)).uniq
86
+ all_joins.size <= 1 ? all_joins.first : all_joins
87
+ end
88
+
89
+ def limit=(value)
90
+ @set_limit = true
91
+ @limit = value.blank? || value == 0 ? nil : value.to_i
92
+ end
93
+
94
+ def limit
95
+ @limit ||= Config.search.per_page if !acting_as_filter? && !@set_limit
96
+ @limit
97
+ end
98
+
99
+ def offset=(value)
100
+ @offset = value.blank? ? nil : value.to_i
101
+ end
102
+
103
+ def options=(values)
104
+ return unless values.is_a?(Hash)
105
+ values.symbolize_keys.fast_assert_valid_keys(OPTIONS)
106
+ values.each { |key, value| send("#{key}=", value) }
107
+ end
108
+
109
+ # Sanitizes everything down into options ActiveRecord::Base.find can understand
110
+ def sanitize(searching = true)
111
+ find_options = {}
112
+
113
+ (searching ? AR_FIND_OPTIONS : AR_CALCULATIONS_OPTIONS).each do |find_option|
114
+ value = send(find_option)
115
+ next if value.blank?
116
+ find_options[find_option] = value
117
+ end
118
+
119
+ find_options
120
+ end
121
+
122
+ def select
123
+ @select ||= "DISTINCT #{klass.connection.quote_table_name(klass.table_name)}.*" if !joins.blank? && Config.search.remove_duplicates? && klass.connection.adapter_name != "PostgreSQL"
124
+ @select
125
+ end
126
+
127
+ def scope
128
+ @scope ||= {}
129
+ end
130
+
131
+ private
132
+ def safe_to_array(o)
133
+ case o
134
+ when NilClass
135
+ []
136
+ when Array
137
+ o
138
+ else
139
+ [o]
140
+ end
141
+ end
142
+
143
+ def array_of_strings?(o)
144
+ o.is_a?(Array) && o.all?{|obj| obj.is_a?(String)}
145
+ end
146
+ end
147
+ end
148
+ end
@@ -0,0 +1,53 @@
1
+ module Searchlogic
2
+ module Search
3
+ # = Searchlogic Conditions
4
+ #
5
+ # Implements all of the conditions functionality into a searchlogic search. All of this functonality is extracted out into its own class Searchlogic::Conditions::Base. This is a separate module to help keep the code
6
+ # clean and organized.
7
+ module Conditions
8
+ def self.included(klass)
9
+ klass.class_eval do
10
+ alias_method_chain :initialize, :conditions
11
+ alias_method_chain :conditions=, :conditions
12
+ alias_method_chain :sanitize, :conditions
13
+ end
14
+ end
15
+
16
+ def initialize_with_conditions(init_options = {})
17
+ self.conditions = Searchlogic::Conditions::Base.create_virtual_class(klass).new
18
+ initialize_without_conditions(init_options)
19
+ end
20
+
21
+ # Sets conditions on the search. Accepts a hash or a Searchlogic::Conditions::Base object.
22
+ #
23
+ # === Examples
24
+ #
25
+ # search.conditions = {:first_name_like => "Ben"}
26
+ # search.conditions = User.new_conditions
27
+ #
28
+ # or to set a scope
29
+ #
30
+ # search.conditions = "user_group_id = 6"
31
+ #
32
+ # now you can create the rest of your search and your "scope" will get merged into your final SQL.
33
+ # What this does is determine if the value a hash or a conditions object, if not it sets it up as a scope.
34
+ def conditions_with_conditions=(values)
35
+ case values
36
+ when Searchlogic::Conditions::Base
37
+ @conditions = values
38
+ else
39
+ @conditions.conditions = values
40
+ end
41
+ end
42
+
43
+ def sanitize_with_conditions(searching = true) # :nodoc:
44
+ find_options = sanitize_without_conditions(searching)
45
+ if conditions_obj = find_options.delete(:conditions)
46
+ new_conditions = conditions_obj.sanitize
47
+ find_options[:conditions] = new_conditions unless new_conditions.blank?
48
+ end
49
+ find_options
50
+ end
51
+ end
52
+ end
53
+ end
@@ -0,0 +1,244 @@
1
+ module Searchlogic
2
+ module Search
3
+ # = Search Ordering
4
+ #
5
+ # The purpose of this module is to provide easy ordering for your searches. All that these options do is
6
+ # build :order for you. This plays a huge part in ordering your data on the interface. See the options and examples below. The readme also touches on ordering. It's pretty simple thought:
7
+ #
8
+ # === Examples
9
+ #
10
+ # search.order_by = :id
11
+ # search.order_by = [:id, :first_name]
12
+ # search.order_by = {:user_group => :name}
13
+ # search.order_by = [:id, {:user_group => :name}]
14
+ # search.order_by = {:user_group => {:account => :name}} # you can traverse through all of your relationships
15
+ #
16
+ # search.order_as = "DESC"
17
+ # search.order_as = "ASC"
18
+ module Ordering
19
+ def self.included(klass)
20
+ klass.class_eval do
21
+ alias_method_chain :order=, :ordering
22
+ alias_method_chain :sanitize, :ordering
23
+ attr_reader :priority_order
24
+ end
25
+ end
26
+
27
+ def order_with_ordering=(value) # :nodoc
28
+ @order_by = nil
29
+ @order_as = nil
30
+ @order_by_auto_joins = nil
31
+ self.order_without_ordering = value
32
+ end
33
+
34
+ # Convenience method for determining if the ordering is ascending
35
+ def asc?
36
+ !desc?
37
+ end
38
+
39
+ # Convenience method for determining if the ordering is descending
40
+ def desc?
41
+ return false if order_as.nil?
42
+ order_as == "DESC"
43
+ end
44
+
45
+ # Determines how the search is being ordered: as DESC or ASC
46
+ def order_as
47
+ return if order.blank?
48
+ return @order_as if @order_as
49
+
50
+ case order
51
+ when /ASC$/i
52
+ @order_as = "ASC"
53
+ when /DESC$/i
54
+ @order_as = "DESC"
55
+ else
56
+ nil
57
+ end
58
+ end
59
+
60
+ # Sets how the results will be ordered: ASC or DESC
61
+ def order_as=(value)
62
+ value = value.blank? ? nil : value.to_s.upcase
63
+ raise(ArgumentError, "order_as only accepts a blank string / nil or a string as 'ASC' or 'DESC'") if !value.blank? && !["ASC", "DESC"].include?(value)
64
+ if @order_by
65
+ @order = order_by_to_order(@order_by, value)
66
+ elsif order
67
+ @order.gsub!(/(ASC|DESC)/i, value)
68
+ end
69
+ @order_as = value
70
+ end
71
+
72
+ # Determines by what columns the search is being ordered. This is nifty in that is reverse engineers the order SQL to determine this, only
73
+ # if you haven't explicitly set the order_by option yourself.
74
+ def order_by
75
+ return if order.blank?
76
+ @order_by ||= order_to_order_by(order)
77
+ end
78
+
79
+ # Lets you set how to order the data
80
+ #
81
+ # === Examples
82
+ #
83
+ # In these examples "ASC" is determined by the value of order_as
84
+ #
85
+ # order_by = :id # => users.id ASC
86
+ # order_by = [:id, name] # => users.id ASC, user.name ASC
87
+ # order_by = [:id, {:user_group => :name}] # => users.id ASC, user_groups.name ASC
88
+ def order_by=(value)
89
+ @order_by_auto_joins = nil
90
+ @order_by = get_order_by_value(value)
91
+ @order = order_by_to_order(@order_by, @order_as)
92
+ @order_by
93
+ end
94
+
95
+ # Returns the joins neccessary for the "order" statement so that we don't get an SQL error
96
+ def order_by_auto_joins
97
+ @order_by_auto_joins ||= build_order_by_auto_joins(order_by)
98
+ end
99
+
100
+ # Let's you set a priority order. Meaning this will get ordered first before anything else, but is unnoticeable and abstracted out from your regular order. For example, lets say you have a model called Product
101
+ # that had a "featured" boolean column. You want to order the products by the price, quantity, etc., but you want the featured products to always be first.
102
+ #
103
+ # Without a priority order your controller would get cluttered and your code would be much more complicated. All of your order_by_link methods would have to be order_by_link [:featured, :price], :text => "Price"
104
+ # Your order_by_link methods alternate between ASC and DESC, so the featured products would jump from the top the bottom. It presents a lot of "work arounds". So priority_order solves this.
105
+ def priority_order=(value)
106
+ @priority_order = value
107
+ end
108
+
109
+ # Same as order_by but for your priority order. See priority_order= for more informaton on priority_order.
110
+ def priority_order_by
111
+ return if priority_order.blank?
112
+ @priority_order_by ||= order_to_order_by(priority_order)
113
+ end
114
+
115
+ # Same as order_by= but for your priority order. See priority_order= for more informaton on priority_order.
116
+ def priority_order_by=(value)
117
+ @priority_order_by_auto_joins = nil
118
+ @priority_order_by = get_order_by_value(value)
119
+ @priority_order = order_by_to_order(@priority_order_by, @priority_order_as)
120
+ @priority_order_by
121
+ end
122
+
123
+ # Same as order_as but for your priority order. See priority_order= for more informaton on priority_order.
124
+ def priority_order_as
125
+ return if priority_order.blank?
126
+ return @priority_order_as if @priority_order_as
127
+
128
+ case priority_order
129
+ when /ASC$/i
130
+ @priority_order_as = "ASC"
131
+ when /DESC$/i
132
+ @priority_order_as = "DESC"
133
+ else
134
+ nil
135
+ end
136
+ end
137
+
138
+ # Same as order_as= but for your priority order. See priority_order= for more informaton on priority_order.
139
+ def priority_order_as=(value)
140
+ value = value.blank? ? nil : value.to_s.upcase
141
+ raise(ArgumentError, "priority_order_as only accepts a blank string / nil or a string as 'ASC' or 'DESC'") if !value.blank? && !["ASC", "DESC"].include?(value)
142
+ if @priority_order_by
143
+ @priority_order = order_by_to_order(@priority_order_by, value)
144
+ elsif priority_order
145
+ @priority_order.gsub!(/(ASC|DESC)/i, value)
146
+ end
147
+ @priority_order_as = value
148
+ end
149
+
150
+ def priority_order_by_auto_joins
151
+ @priority_order_by_auto_joins ||= build_order_by_auto_joins(priority_order_by)
152
+ end
153
+
154
+ def sanitize_with_ordering(searching = true)
155
+ find_options = sanitize_without_ordering(searching)
156
+ unless priority_order.blank?
157
+ order_parts = [priority_order, find_options[:order]].compact
158
+ find_options[:order] = order_parts.join(", ")
159
+ end
160
+ find_options
161
+ end
162
+
163
+ private
164
+ def order_by_to_order(order_by, order_as, alt_klass = nil)
165
+ return if order_by.blank?
166
+
167
+ k = alt_klass || klass
168
+ table_name = k.table_name
169
+ sql_parts = []
170
+
171
+ case order_by
172
+ when Array
173
+ order_by.each { |part| sql_parts << order_by_to_order(part, order_as, alt_klass) }
174
+ when Hash
175
+ raise(ArgumentError, "when passing a hash to order_by you must only have 1 key: {:user_group => :name} not {:user_group => :name, :user_group => :id}. The latter should be [{:user_group => :name}, {:user_group => :id}]") if order_by.keys.size != 1
176
+ key = order_by.keys.first
177
+ reflection = k.reflect_on_association(key.to_sym)
178
+ value = order_by.values.first
179
+ sql_parts << order_by_to_order(value, order_as, reflection.klass)
180
+ when Symbol, String
181
+ part = "#{quote_table_name(table_name)}.#{quote_column_name(order_by)}"
182
+ part += " #{order_as}" unless order_as.blank?
183
+ sql_parts << part
184
+ end
185
+
186
+ sql_parts.join(", ")
187
+ end
188
+
189
+ def order_to_order_by(order)
190
+ # Reversege engineer order, only go 1 level deep with relationships, anything beyond that is probably excessive and not good for performance
191
+ order_parts = order.split(",").collect do |part|
192
+ part.strip!
193
+ part.gsub!(/ (ASC|DESC)$/i, "")
194
+ part.gsub!(/(.*)\./, "")
195
+ table_name = ($1 ? $1.gsub(/[^a-z0-9_]/i, "") : nil)
196
+ part.gsub!(/[^a-z0-9_]/i, "")
197
+ reflection = nil
198
+ if table_name && table_name != klass.table_name
199
+ reflection = klass.reflect_on_association(table_name.to_sym) || klass.reflect_on_association(table_name.singularize.to_sym)
200
+ next unless reflection
201
+ {reflection.name.to_s => part}
202
+ else
203
+ part
204
+ end
205
+ end.compact
206
+ order_parts.size <= 1 ? order_parts.first : order_parts
207
+ end
208
+
209
+ def build_order_by_auto_joins(order_by_value)
210
+ case order_by_value
211
+ when Array
212
+ order_by_value.collect { |value| build_order_by_auto_joins(value) }.uniq.compact
213
+ when Hash
214
+ key = order_by_value.keys.first
215
+ value = order_by_value.values.first
216
+ case value
217
+ when Hash
218
+ {key.to_sym => build_order_by_auto_joins(value)}
219
+ else
220
+ key.to_sym
221
+ end
222
+ else
223
+ nil
224
+ end
225
+ end
226
+
227
+ def get_order_by_value(value)
228
+ Marshal.load(value.unpack("m").first) rescue value
229
+ end
230
+
231
+ def quote_column_name(column_name)
232
+ klass_connection.quote_column_name(column_name)
233
+ end
234
+
235
+ def quote_table_name(table_name)
236
+ klass_connection.quote_table_name(table_name)
237
+ end
238
+
239
+ def klass_connection
240
+ @connection ||= klass.connection
241
+ end
242
+ end
243
+ end
244
+ end