Order search results properly, paginate them too
authorCharles Connell <charles@connells.org>
Fri, 3 Jan 2014 23:03:20 +0000 (18:03 -0500)
committerCharles Connell <charles@connells.org>
Fri, 3 Jan 2014 23:03:20 +0000 (18:03 -0500)
karmaworld/apps/notes/search.py
karmaworld/apps/notes/views.py
karmaworld/assets/css/search.css
karmaworld/templates/notes/search_results.html

index 39081807d805106a5baf3bebe3dadc663c746466..48394d00c27ac7a0f1b6f9c54e25e501c1d24967 100644 (file)
@@ -9,6 +9,8 @@ import karmaworld.secret.indexden as secret
 
 import logging
 
+PAGE_SIZE = 10
+
 logging.basicConfig()
 logger = logging.getLogger(__name__)
 
@@ -27,6 +29,13 @@ while not index.has_started():
 # "Relevance" is a black box provided by IndexDen.
 index.add_function(0, 'relevance * log(doc.var[0])')
 
+class SearchResult(object):
+
+    def __init__(self, ordered_ids, snippet_dict, has_more):
+        self.ordered_ids = ordered_ids
+        self.snippet_dict = snippet_dict
+        self.has_more = has_more
+
 def note_to_dict(note):
     d = {
         'name': note.name,
@@ -48,12 +57,12 @@ def add_document(note):
     if note.text:
         index.add_document(note.id, note_to_dict(note), variables={0: note.thanks})
     else:
-        logger.warn("Note {n} has no text or thanks value, will not add to IndexDen".format(n=note))
+        logger.warn("Note {n} has no text, will not add to IndexDen".format(n=note))
 
 def remove_document(note):
     index.delete_document(note.id)
 
-def search(query, course_id=None):
+def search(query, course_id=None, page=0):
     """Returns note IDs matching the given query,
     filtered by course ID if given"""
     if course_id:
@@ -61,8 +70,13 @@ def search(query, course_id=None):
     else:
         real_query = '"%s" OR name:"%s"' % (query, query)
 
-    raw_results = index.search(real_query, scoring_function=0, snippet_fields=['text'])
+    raw_results = index.search(real_query, snippet_fields=['text'],
+                               length=PAGE_SIZE, start=(page*PAGE_SIZE))
+
+    ordered_ids = [int(r['docid']) for r in raw_results['results']]
+    snippet_dict = {int(r['docid']): r['snippet_text'] for r in raw_results['results']}
 
-    results = {r['docid']: r['snippet_text'] for r in raw_results['results']}
+    # Are there more results to show the user if they want?
+    has_more = True if int(raw_results['matches']) > ((page+1) * PAGE_SIZE) else False
 
-    return results
+    return SearchResult(ordered_ids, snippet_dict, has_more)
index 647cb97f56f10620b6c52e4dd73fe9210cb0a631..192872194250f4140e90cf888c809888e0817989 100644 (file)
@@ -157,16 +157,25 @@ class NoteSearchView(ListView):
         if not 'query' in self.request.GET:
             return Note.objects.none()
 
+        if 'page' in self.request.GET:
+            page = int(self.request.GET['page'])
+        else:
+            page = 0
+
         if 'course_id' in self.request.GET:
-            matching_notes = search.search(self.request.GET['query'],
-                                              self.request.GET['course_id'])
+            raw_results = search.search(self.request.GET['query'],
+                                              self.request.GET['course_id'],
+                                              page=page)
         else:
-            matching_notes = search.search(self.request.GET['query'])
+            raw_results = search.search(self.request.GET['query'],
+                                        page=page)
 
-        instances = Note.objects.filter(id__in=matching_notes.viewkeys())
+        instances = Note.objects.in_bulk(raw_results.ordered_ids)
         results = []
-        for i in instances:
-            results.append((i, matching_notes[str(i.id)]))
+        for id in raw_results.ordered_ids:
+            if id in instances:
+                results.append((instances[id], raw_results.snippet_dict[id]))
+        self.has_more = raw_results.has_more
 
         return results
 
@@ -177,6 +186,22 @@ class NoteSearchView(ListView):
         if 'course_id' in self.request.GET:
             kwargs['course'] = Course.objects.get(id=self.request.GET['course_id'])
 
+        # If query returned more search results than could
+        # fit on one page, show "Next" button
+        if self.has_more:
+            kwargs['has_next'] = True
+            if 'page' in self.request.GET:
+                kwargs['next_page'] = int(self.request.GET['page']) + 1
+            else:
+                kwargs['next_page'] = 1
+
+        # If the user is looking at a search result page
+        # that isn't the first one, show "Prev" button
+        if 'page' in self.request.GET and \
+            int(self.request.GET['page']) > 0:
+            kwargs['has_prev'] = True
+            kwargs['prev_page'] = int(self.request.GET['page']) - 1
+
         return super(NoteSearchView, self).get_context_data(**kwargs)
 
 
index 821f1103197443fd9c0ed6144a9cfcb6d26a86d3..d7a3755755f0943af58f4c4bc5be5c66365d5ff3 100644 (file)
   font-size: 10px;
   color: #8e8e8e;
   margin-bottom: 20px;
+}
+
+#search_nav_buttons
+{
+  font-family: "MuseoSlab-900";
+  font-size: 25px;
+  color: #8e8e8e;
+  margin-bottom: 20px;
 }
\ No newline at end of file
index d1ded580ab8d3139b84394f152f31200576e72e0..65e408bf48794cf3e9f2bdac0e1f6d79c845353a 100644 (file)
 
   <div id="results_header" class="hero_gradient_bar">
 
-    <div class="row">
-      <div class="twelve columns header_subhead">
-        <a href="{{ course.get_absolute_url}}">
-          <i class="fa fa-arrow-left"></i>&nbsp;back to {{ course.name }}
-        </a>
+    {% if course %}
+      <div class="row">
+        <div class="twelve columns header_subhead">
+          <a href="{{ course.get_absolute_url}}">
+            <i class="fa fa-arrow-left"></i>&nbsp;back to {{ course.name }}
+          </a>
+        </div>
       </div>
-    </div>
+    {% endif %}
 
     <div class="row">
       <div class="small-12 columns header_subhead">
           </div>
         {% endif %}
     </div>
+
+    <div id="search_nav_buttons" class="row">
+      <div class="small-12 columns center">
+        {% if has_prev %}
+          <a href="{% url 'note_search' %}?query={{ query|urlencode }}&course_id={{ course.id }}&page={{ prev_page }}">
+            <i class="fa fa-caret-left"></i>&nbsp;Previous</a>
+        {% endif %}
+        {% if has_next %}
+          <a href="{% url 'note_search' %}?query={{ query|urlencode }}&course_id={{ course.id }}&page={{ next_page }}">
+            Next&nbsp;<i class="fa fa-caret-right"></i></a>
+        {% endif %}
+      </div>
+    </div>
   </div><!-- /course_container -->
 
 </section>