]> source.dussan.org Git - redmine.git/commitdiff
SortHelper refactoring:
authorJean-Philippe Lang <jp_lang@yahoo.fr>
Tue, 10 Mar 2009 21:11:36 +0000 (21:11 +0000)
committerJean-Philippe Lang <jp_lang@yahoo.fr>
Tue, 10 Mar 2009 21:11:36 +0000 (21:11 +0000)
* multiple columns sort feature (#2871)
* CSS classes instead of an image tag to reflect the state of the column
* examples fixed (#2945)

git-svn-id: svn+ssh://rubyforge.org/var/svn/redmine/trunk@2571 e93f8b46-1217-0410-a6f0-8f06a7374b81

app/helpers/sort_helper.rb
public/stylesheets/application.css
test/unit/helpers/sort_helper_test.rb [new file with mode: 0644]

index f5e9bb4947b9522015ada7e9cb581f3505c3b0ab..d4d8f721ac5308abcdc45a19ee5e006163b132f8 100644 (file)
@@ -1,11 +1,12 @@
 # Helpers to sort tables using clickable column headers.
 #
 # Author:  Stuart Rackham <srackham@methods.co.nz>, March 2005.
+#          Jean-Philippe Lang, 2009
 # License: This source code is released under the MIT license.
 #
 # - Consecutive clicks toggle the column's sort order.
 # - Sort state is maintained by a session hash entry.
-# - Icon image identifies sort column and state.
+# - CSS classes identify sort column and state.
 # - Typically used in conjunction with the Pagination module.
 #
 # Example code snippets:
@@ -17,7 +18,7 @@
 # 
 #   def list
 #     sort_init 'last_name'
-#     sort_update
+#     sort_update %w(first_name, last_name)
 #     @items = Contact.find_all nil, sort_clause
 #   end
 # 
@@ -28,7 +29,7 @@
 # 
 #   def list
 #     sort_init 'last_name'
-#     sort_update
+#     sort_update %w(first_name, last_name)
 #     @contact_pages, @items = paginate :contacts,
 #       :order_by => sort_clause,
 #       :per_page => 10
 #     </tr>
 #   </thead>
 #
-# - The ascending and descending sort icon images are sort_asc.png and
-#   sort_desc.png and reside in the application's images directory.
-# - Introduces instance variables: @sort_name, @sort_default.
-# - Introduces params :sort_key and :sort_order.
+# - Introduces instance variables: @sort_default, @sort_criteria
+# - Introduces param :sort
 #
+
 module SortHelper
+  class SortCriteria
+    
+    def initialize
+      @criteria = []
+    end
+    
+    def available_criteria=(criteria)
+      unless criteria.is_a?(Hash)
+        criteria = criteria.inject({}) {|h,k| h[k] = k; h}
+      end
+      @available_criteria = criteria
+    end
+    
+    def from_param(param)
+      @criteria = param.to_s.split(',').collect {|s| s.split(':')[0..1]}
+      normalize!
+    end
+    
+    def to_param
+      @criteria.collect {|k,o| k + (o ? '' : ':desc')}.join(',')
+    end
+    
+    def to_sql
+      sql = @criteria.collect do |k,o|
+        if s = @available_criteria[k]
+          (o ? s.to_a : s.to_a.collect {|c| "#{c} DESC"}).join(', ')
+        end
+      end.compact.join(', ')
+      sql.blank? ? nil : sql
+    end
+    
+    def add!(key, asc)
+      @criteria.delete_if {|k,o| k == key}
+      @criteria = [[key, asc]] + @criteria
+      normalize!
+    end
+    
+    def add(*args)
+      r = self.class.new.from_param(to_param)
+      r.add!(*args)
+      r
+    end
+    
+    def first_key
+      @criteria.first && @criteria.first.first
+    end
+    
+    def first_asc?
+      @criteria.first && @criteria.first.last
+    end
+    
+    private
+    
+    def normalize!
+      @criteria = @criteria.collect {|s| [s.first, (s.last == false || s.last == 'desc') ? false : true]}
+      @criteria = @criteria.select {|k,o| @available_criteria.has_key?(k)} if @available_criteria
+      @criteria.slice!(3)
+      self
+    end
+  end
 
   # Initializes the default sort column (default_key) and sort order
   # (default_order).
   #
   # - default_key is a column attribute name.
   # - default_order is 'asc' or 'desc'.
-  # - name is the name of the session hash entry that stores the sort state,
-  #   defaults to '<controller_name>_sort'.
   #
-  def sort_init(default_key, default_order='asc', name=nil)
-    @sort_name = name || params[:controller] + params[:action] + '_sort'
-    @sort_default = {:key => default_key, :order => default_order}
+  def sort_init(default_key, default_order='asc')
+    @sort_default = "#{default_key}:#{default_order}"
   end
 
   # Updates the sort state. Call this in the controller prior to calling
   # sort_clause.
-  # sort_keys can be either an array or a hash of allowed keys
-  def sort_update(sort_keys)
-    sort_key = params[:sort_key]
-    sort_key = nil unless (sort_keys.is_a?(Array) ? sort_keys.include?(sort_key) : sort_keys[sort_key])
-
-    sort_order = (params[:sort_order] == 'desc' ? 'DESC' : 'ASC')
-    
-    if sort_key
-      sort = {:key => sort_key, :order => sort_order}
-    elsif session[@sort_name]
-      sort = session[@sort_name]   # Previous sort.
-    else
-      sort = @sort_default
-    end
-    session[@sort_name] = sort
+  # - criteria can be either an array or a hash of allowed keys
+  #
+  def sort_update(criteria)
+    sort_name = controller_name + '_' + action_name + '_sort'
     
-    sort_column = (sort_keys.is_a?(Hash) ? sort_keys[sort[:key]] : sort[:key])
-    @sort_clause = (sort_column.blank? ? nil : [sort_column].flatten.collect {|s| "#{s} #{sort[:order]}"}.join(','))
+    @sort_criteria = SortCriteria.new
+    @sort_criteria.available_criteria = criteria
+    @sort_criteria.from_param(params[:sort] || session[sort_name] || @sort_default)
+    session[sort_name] = @sort_criteria.to_param
   end
 
   # Returns an SQL sort clause corresponding to the current sort state.
   # Use this to sort the controller's table items collection.
   #
   def sort_clause()
-    @sort_clause
+    @sort_criteria.to_sql
   end
 
   # Returns a link which sorts by the named column.
   #
   # - column is the name of an attribute in the sorted record collection.
-  # - The optional caption explicitly specifies the displayed link text.
-  # - A sort icon image is positioned to the right of the sort link.
+  # - the optional caption explicitly specifies the displayed link text.
+  # - 2 CSS classes reflect the state of the link: sort and asc or desc
   #
   def sort_link(column, caption, default_order)
-    key, order = session[@sort_name][:key], session[@sort_name][:order]
-    if key == column
-      if order.downcase == 'asc'
-        icon = 'sort_asc.png'
+    css, order = nil, default_order
+    
+    if column.to_s == @sort_criteria.first_key
+      if @sort_criteria.first_asc?
+        css = 'sort asc'
         order = 'desc'
       else
-        icon = 'sort_desc.png'
+        css = 'sort desc'
         order = 'asc'
       end
-    else
-      icon = nil
-      order = default_order
     end
-    caption = titleize(Inflector::humanize(column)) unless caption
+    caption = column.to_s.humanize unless caption
     
-    sort_options = { :sort_key => column, :sort_order => order }
+    sort_options = { :sort => @sort_criteria.add(column.to_s, order).to_param }
     # don't reuse params if filters are present
     url_options = params.has_key?(:set_filter) ? sort_options : params.merge(sort_options)
     
-    # Add project_id to url_options
+     # Add project_id to url_options
     url_options = url_options.merge(:project_id => params[:project_id]) if params.has_key?(:project_id)
-    
+
     link_to_remote(caption,
                   {:update => "content", :url => url_options, :method => :get},
-                  {:href => url_for(url_options)}) +
-    (icon ? nbsp(2) + image_tag(icon) : '')
+                  {:href => url_for(url_options),
+                   :class => css})
   end
 
   # Returns a table header <th> tag with a sort link for the named column
@@ -150,22 +196,10 @@ module SortHelper
   #   </th>
   #
   def sort_header_tag(column, options = {})
-    caption = options.delete(:caption) || titleize(Inflector::humanize(column))
+    caption = options.delete(:caption) || column.to_s.humanize
     default_order = options.delete(:default_order) || 'asc'
-    options[:title]= l(:label_sort_by, "\"#{caption}\"") unless options[:title]
+    options[:title] = l(:label_sort_by, "\"#{caption}\"") unless options[:title]
     content_tag('th', sort_link(column, caption, default_order), options)
   end
-
-  private
-
-    # Return n non-breaking spaces.
-    def nbsp(n)
-      '&nbsp;' * n
-    end
-
-    # Return capitalized title.
-    def titleize(title)
-      title.split.map {|w| w.capitalize }.join(' ')
-    end
-
 end
+
index 96b61b28c50342e09a8da8d236b6dbc706339e48..88cefadea9d754f8688ccfff34eb3a03ec5f8755 100644 (file)
@@ -142,6 +142,10 @@ table p {margin:0;}
 .odd {background-color:#f6f7f8;}
 .even {background-color: #fff;}
 
+a.sort { padding-right: 16px; background-position: 100% 50%; background-repeat: no-repeat; }
+a.sort.asc  { background-image: url(../images/sort_asc.png); }
+a.sort.desc { background-image: url(../images/sort_desc.png); }
+
 .highlight { background-color: #FCFD8D;}
 .highlight.token-1 { background-color: #faa;}
 .highlight.token-2 { background-color: #afa;}
diff --git a/test/unit/helpers/sort_helper_test.rb b/test/unit/helpers/sort_helper_test.rb
new file mode 100644 (file)
index 0000000..9132beb
--- /dev/null
@@ -0,0 +1,73 @@
+# Redmine - project management software\r
+# Copyright (C) 2006-2009  Jean-Philippe Lang\r
+#\r
+# This program is free software; you can redistribute it and/or\r
+# modify it under the terms of the GNU General Public License\r
+# as published by the Free Software Foundation; either version 2\r
+# of the License, or (at your option) any later version.\r
+# \r
+# This program is distributed in the hope that it will be useful,\r
+# but WITHOUT ANY WARRANTY; without even the implied warranty of\r
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\r
+# GNU General Public License for more details.\r
+# \r
+# You should have received a copy of the GNU General Public License\r
+# along with this program; if not, write to the Free Software\r
+# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.\r
+\r
+require File.dirname(__FILE__) + '/../../test_helper'\r
+\r
+class SortHelperTest < HelperTestCase\r
+  include SortHelper\r
+  \r
+  def test_default_sort_clause_with_array\r
+    sort_init 'attr1', 'desc'\r
+    sort_update(['attr1', 'attr2'])\r
+\r
+    assert_equal 'attr1 DESC', sort_clause\r
+  end\r
+  \r
+  def test_default_sort_clause_with_hash\r
+    sort_init 'attr1', 'desc'\r
+    sort_update({'attr1' => 'table1.attr1', 'attr2' => 'table2.attr2'})\r
+\r
+    assert_equal 'table1.attr1 DESC', sort_clause\r
+  end\r
+  \r
+  def test_params_sort\r
+    @sort_param = 'attr1,attr2:desc'\r
+    \r
+    sort_init 'attr1', 'desc'\r
+    sort_update({'attr1' => 'table1.attr1', 'attr2' => 'table2.attr2'})\r
+\r
+    assert_equal 'table1.attr1, table2.attr2 DESC', sort_clause\r
+    assert_equal 'attr1,attr2:desc', @session['foo_bar_sort']\r
+  end\r
+  \r
+  def test_invalid_params_sort\r
+    @sort_param = 'attr3'\r
+    \r
+    sort_init 'attr1', 'desc'\r
+    sort_update({'attr1' => 'table1.attr1', 'attr2' => 'table2.attr2'})\r
+\r
+    assert_nil sort_clause\r
+    assert_equal '', @session['foo_bar_sort']\r
+  end\r
+  \r
+  def test_invalid_order_params_sort\r
+    @sort_param = 'attr1:foo:bar,attr2'\r
+    \r
+    sort_init 'attr1', 'desc'\r
+    sort_update({'attr1' => 'table1.attr1', 'attr2' => 'table2.attr2'})\r
+\r
+    assert_equal 'table1.attr1, table2.attr2', sort_clause\r
+    assert_equal 'attr1,attr2', @session['foo_bar_sort']\r
+  end\r
+  \r
+  private\r
+  \r
+  def controller_name; 'foo'; end\r
+  def action_name; 'bar'; end\r
+  def params; {:sort => @sort_param}; end\r
+  def session; @session ||= {}; end\r
+end\r