schof-searchlogic 0.0.2
Sign up to get free protection for your applications and to get access to all the features.
- data/CHANGELOG.rdoc +302 -0
- data/MIT-LICENSE +20 -0
- data/Manifest +157 -0
- data/README.rdoc +461 -0
- data/Rakefile +13 -0
- data/TODO.rdoc +4 -0
- data/init.rb +1 -0
- data/lib/searchlogic.rb +100 -0
- data/lib/searchlogic/active_record/associations.rb +52 -0
- data/lib/searchlogic/active_record/base.rb +224 -0
- data/lib/searchlogic/active_record/connection_adapters/mysql_adapter.rb +176 -0
- data/lib/searchlogic/active_record/connection_adapters/postgresql_adapter.rb +172 -0
- data/lib/searchlogic/active_record/connection_adapters/sqlite_adapter.rb +80 -0
- data/lib/searchlogic/condition/base.rb +165 -0
- data/lib/searchlogic/condition/begins_with.rb +17 -0
- data/lib/searchlogic/condition/blank.rb +21 -0
- data/lib/searchlogic/condition/child_of.rb +11 -0
- data/lib/searchlogic/condition/descendant_of.rb +11 -0
- data/lib/searchlogic/condition/ends_with.rb +17 -0
- data/lib/searchlogic/condition/equals.rb +33 -0
- data/lib/searchlogic/condition/greater_than.rb +15 -0
- data/lib/searchlogic/condition/greater_than_or_equal_to.rb +15 -0
- data/lib/searchlogic/condition/inclusive_descendant_of.rb +10 -0
- data/lib/searchlogic/condition/keywords.rb +47 -0
- data/lib/searchlogic/condition/less_than.rb +15 -0
- data/lib/searchlogic/condition/less_than_or_equal_to.rb +15 -0
- data/lib/searchlogic/condition/like.rb +15 -0
- data/lib/searchlogic/condition/nested_set.rb +17 -0
- data/lib/searchlogic/condition/nil.rb +21 -0
- data/lib/searchlogic/condition/not_begin_with.rb +20 -0
- data/lib/searchlogic/condition/not_blank.rb +19 -0
- data/lib/searchlogic/condition/not_end_with.rb +20 -0
- data/lib/searchlogic/condition/not_equal.rb +27 -0
- data/lib/searchlogic/condition/not_have_keywords.rb +20 -0
- data/lib/searchlogic/condition/not_like.rb +20 -0
- data/lib/searchlogic/condition/not_nil.rb +19 -0
- data/lib/searchlogic/condition/sibling_of.rb +14 -0
- data/lib/searchlogic/conditions/any_or_all.rb +42 -0
- data/lib/searchlogic/conditions/base.rb +244 -0
- data/lib/searchlogic/conditions/groups.rb +74 -0
- data/lib/searchlogic/conditions/magic_methods.rb +286 -0
- data/lib/searchlogic/conditions/multiparameter_attributes.rb +105 -0
- data/lib/searchlogic/conditions/protection.rb +36 -0
- data/lib/searchlogic/config.rb +31 -0
- data/lib/searchlogic/config/helpers.rb +338 -0
- data/lib/searchlogic/config/search.rb +53 -0
- data/lib/searchlogic/core_ext/hash.rb +75 -0
- data/lib/searchlogic/core_ext/object.rb +19 -0
- data/lib/searchlogic/helpers/control_types/link.rb +310 -0
- data/lib/searchlogic/helpers/control_types/links.rb +242 -0
- data/lib/searchlogic/helpers/control_types/remote_link.rb +87 -0
- data/lib/searchlogic/helpers/control_types/remote_links.rb +72 -0
- data/lib/searchlogic/helpers/control_types/remote_select.rb +36 -0
- data/lib/searchlogic/helpers/control_types/select.rb +82 -0
- data/lib/searchlogic/helpers/form.rb +208 -0
- data/lib/searchlogic/helpers/utilities.rb +197 -0
- data/lib/searchlogic/modifiers/absolute.rb +15 -0
- data/lib/searchlogic/modifiers/acos.rb +11 -0
- data/lib/searchlogic/modifiers/asin.rb +11 -0
- data/lib/searchlogic/modifiers/atan.rb +11 -0
- data/lib/searchlogic/modifiers/avg.rb +15 -0
- data/lib/searchlogic/modifiers/base.rb +27 -0
- data/lib/searchlogic/modifiers/ceil.rb +15 -0
- data/lib/searchlogic/modifiers/char_length.rb +15 -0
- data/lib/searchlogic/modifiers/cos.rb +15 -0
- data/lib/searchlogic/modifiers/cot.rb +15 -0
- data/lib/searchlogic/modifiers/count.rb +11 -0
- data/lib/searchlogic/modifiers/day_of_month.rb +15 -0
- data/lib/searchlogic/modifiers/day_of_week.rb +15 -0
- data/lib/searchlogic/modifiers/day_of_year.rb +15 -0
- data/lib/searchlogic/modifiers/degrees.rb +11 -0
- data/lib/searchlogic/modifiers/exp.rb +15 -0
- data/lib/searchlogic/modifiers/floor.rb +15 -0
- data/lib/searchlogic/modifiers/hex.rb +11 -0
- data/lib/searchlogic/modifiers/hour.rb +11 -0
- data/lib/searchlogic/modifiers/log.rb +15 -0
- data/lib/searchlogic/modifiers/log10.rb +11 -0
- data/lib/searchlogic/modifiers/log2.rb +11 -0
- data/lib/searchlogic/modifiers/lower.rb +15 -0
- data/lib/searchlogic/modifiers/ltrim.rb +15 -0
- data/lib/searchlogic/modifiers/md5.rb +11 -0
- data/lib/searchlogic/modifiers/microseconds.rb +11 -0
- data/lib/searchlogic/modifiers/milliseconds.rb +11 -0
- data/lib/searchlogic/modifiers/minute.rb +15 -0
- data/lib/searchlogic/modifiers/month.rb +15 -0
- data/lib/searchlogic/modifiers/octal.rb +15 -0
- data/lib/searchlogic/modifiers/radians.rb +11 -0
- data/lib/searchlogic/modifiers/round.rb +11 -0
- data/lib/searchlogic/modifiers/rtrim.rb +15 -0
- data/lib/searchlogic/modifiers/second.rb +15 -0
- data/lib/searchlogic/modifiers/sign.rb +11 -0
- data/lib/searchlogic/modifiers/sin.rb +11 -0
- data/lib/searchlogic/modifiers/square_root.rb +15 -0
- data/lib/searchlogic/modifiers/sum.rb +11 -0
- data/lib/searchlogic/modifiers/tan.rb +15 -0
- data/lib/searchlogic/modifiers/trim.rb +15 -0
- data/lib/searchlogic/modifiers/upper.rb +15 -0
- data/lib/searchlogic/modifiers/week.rb +11 -0
- data/lib/searchlogic/modifiers/year.rb +11 -0
- data/lib/searchlogic/search/base.rb +148 -0
- data/lib/searchlogic/search/conditions.rb +53 -0
- data/lib/searchlogic/search/ordering.rb +244 -0
- data/lib/searchlogic/search/pagination.rb +121 -0
- data/lib/searchlogic/search/protection.rb +89 -0
- data/lib/searchlogic/search/searching.rb +32 -0
- data/lib/searchlogic/shared/utilities.rb +56 -0
- data/lib/searchlogic/shared/virtual_classes.rb +39 -0
- data/lib/searchlogic/version.rb +79 -0
- data/searchlogic.gemspec +41 -0
- data/test/active_record_tests/associations_test.rb +94 -0
- data/test/active_record_tests/base_test.rb +115 -0
- data/test/condition_tests/base_test.rb +54 -0
- data/test/condition_tests/begins_with_test.rb +11 -0
- data/test/condition_tests/blank_test.rb +31 -0
- data/test/condition_tests/child_of_test.rb +17 -0
- data/test/condition_tests/descendant_of_test.rb +12 -0
- data/test/condition_tests/ends_with_test.rb +11 -0
- data/test/condition_tests/equals_test.rb +28 -0
- data/test/condition_tests/greater_than_or_equal_to_test.rb +11 -0
- data/test/condition_tests/greater_than_test.rb +11 -0
- data/test/condition_tests/inclusive_descendant_of_test.rb +12 -0
- data/test/condition_tests/keywords_test.rb +23 -0
- data/test/condition_tests/less_than_or_equal_to_test.rb +11 -0
- data/test/condition_tests/less_than_test.rb +11 -0
- data/test/condition_tests/like_test.rb +11 -0
- data/test/condition_tests/nil_test.rb +31 -0
- data/test/condition_tests/not_begin_with_test.rb +8 -0
- data/test/condition_tests/not_blank_test.rb +8 -0
- data/test/condition_tests/not_end_with_test.rb +8 -0
- data/test/condition_tests/not_equal_test.rb +19 -0
- data/test/condition_tests/not_have_keywords_test.rb +8 -0
- data/test/condition_tests/not_like_test.rb +8 -0
- data/test/condition_tests/not_nil_test.rb +13 -0
- data/test/condition_tests/sibling_of_test.rb +15 -0
- data/test/conditions_tests/any_or_all_test.rb +23 -0
- data/test/conditions_tests/base_test.rb +185 -0
- data/test/conditions_tests/groups_test.rb +68 -0
- data/test/conditions_tests/magic_methods_test.rb +36 -0
- data/test/conditions_tests/multiparameter_attributes_test.rb +15 -0
- data/test/conditions_tests/protection_test.rb +18 -0
- data/test/config_test.rb +23 -0
- data/test/fixtures/accounts.yml +12 -0
- data/test/fixtures/animals.yml +7 -0
- data/test/fixtures/orders.yml +12 -0
- data/test/fixtures/user_groups.yml +5 -0
- data/test/fixtures/users.yml +45 -0
- data/test/libs/awesome_nested_set.rb +545 -0
- data/test/libs/awesome_nested_set/compatability.rb +29 -0
- data/test/libs/awesome_nested_set/helper.rb +40 -0
- data/test/libs/awesome_nested_set/named_scope.rb +140 -0
- data/test/libs/rexml_fix.rb +14 -0
- data/test/modifier_tests/day_of_month_test.rb +16 -0
- data/test/search_tests/base_test.rb +241 -0
- data/test/search_tests/conditions_test.rb +21 -0
- data/test/search_tests/ordering_test.rb +167 -0
- data/test/search_tests/pagination_test.rb +74 -0
- data/test/search_tests/protection_test.rb +26 -0
- data/test/test_helper.rb +116 -0
- metadata +385 -0
@@ -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
|