From e3fdc57f2ffacf7986ec0b847b2b655d5114e6fe Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?Jean-S=C3=A9bastien=20Caux?= <git@jscaux.org>
Date: Wed, 16 Feb 2022 08:33:24 +0100
Subject: [PATCH] Add more navs: reports, comments, commentaries, theses

---
 scipost_django/commentaries/forms.py          | 61 ++++++++++++++++++-
 scipost_django/commentaries/models.py         |  2 +-
 .../_commentary_card_content.html             |  2 +-
 .../commentaries/tests/test_views.py          |  2 +-
 scipost_django/commentaries/views.py          |  4 +-
 scipost_django/comments/forms.py              |  4 +-
 .../scipost/templates/scipost/navbar.html     | 59 ++++++++++++++++++
 .../scipost/portal/_hx_commentaries.html      | 42 +++++++++++++
 .../scipost/portal/_hx_commentaries_page.html | 28 +++++++++
 .../scipost/portal/_hx_comments.html          |  2 +-
 .../templates/scipost/portal/_hx_reports.html | 10 +--
 .../scipost/portal/_hx_submissions_base.html  |  5 +-
 .../templates/scipost/portal/_hx_theses.html  | 37 +++++++++++
 .../scipost/portal/_hx_theses_page.html       | 28 +++++++++
 .../templates/scipost/portal/portal.html      | 20 ++++++
 scipost_django/scipost/tests/test_views.py    |  1 -
 scipost_django/scipost/urls.py                | 16 +++++
 scipost_django/scipost/views.py               | 58 +++++++++++++++++-
 scipost_django/submissions/forms.py           |  6 +-
 scipost_django/theses/forms.py                | 56 +++++++++++++++++
 .../theses/_thesislink_card_content.html      |  2 +-
 21 files changed, 422 insertions(+), 23 deletions(-)
 create mode 100644 scipost_django/scipost/templates/scipost/portal/_hx_commentaries.html
 create mode 100644 scipost_django/scipost/templates/scipost/portal/_hx_commentaries_page.html
 create mode 100644 scipost_django/scipost/templates/scipost/portal/_hx_theses.html
 create mode 100644 scipost_django/scipost/templates/scipost/portal/_hx_theses_page.html

diff --git a/scipost_django/commentaries/forms.py b/scipost_django/commentaries/forms.py
index af9c3ed57..0271ad453 100644
--- a/scipost_django/commentaries/forms.py
+++ b/scipost_django/commentaries/forms.py
@@ -6,6 +6,10 @@ from django import forms
 from django.utils.safestring import mark_safe
 from django.template.loader import get_template
 
+from crispy_forms.helper import FormHelper
+from crispy_forms.layout import Layout, Div
+from crispy_bootstrap5.bootstrap5 import FloatingField
+
 from .models import Commentary
 from .constants import COMMENTARY_PUBLISHED, COMMENTARY_PREPRINT
 
@@ -315,7 +319,62 @@ class VetCommentaryForm(forms.Form):
 
 
 class CommentarySearchForm(forms.Form):
-    """Search for Commentary specified by user"""
+    author = forms.CharField(
+        max_length=100,
+        required=False,
+        label="On publication with author(s)"
+    )
+    title = forms.CharField(
+        max_length=100,
+        required=False,
+        label="On publication with title"
+    )
+    abstract = forms.CharField(
+        max_length=1000,
+        required=False,
+        label="On publication with abstract"
+    )
+
+    def __init__(self, *args, **kwargs):
+        self.acad_field_slug = kwargs.pop("acad_field_slug")
+        self.specialty_slug = kwargs.pop("specialty_slug")
+        super().__init__(*args, **kwargs)
+        self.helper = FormHelper()
+        self.helper.layout = Layout(
+            Div(
+                FloatingField("author"),
+                FloatingField("title"),
+                FloatingField("abstract"),
+            ),
+        )
+
+    def search_results(self):
+        """Return all Commentary objects according to search"""
+        commentaries = Commentary.objects.vetted()
+        if self.acad_field_slug and self.acad_field_slug != "all":
+            commentaries = commentaries.filter(acad_field__slug=self.acad_field_slug)
+            if self.specialty_slug and self.specialty_slug != "all":
+                commentaries = commentaries.filter(
+                    specialties__slug=self.specialty_slug
+                )
+        if hasattr(self, "cleaned_data"):
+            if "title" in self.cleaned_data:
+                commentaries = commentaries.filter(
+                    title__icontains=self.cleaned_data["title"],
+                )
+            if "abstract" in self.cleaned_data:
+                commentaries = commentaries.filter(
+                    pub_abstract__icontains=self.cleaned_data["abstract"],
+                )
+            if "author" in self.cleaned_data:
+                commentaries = commentaries.filter(
+                    author_list__icontains=self.cleaned_data["author"],
+                )
+        return commentaries.order_by("-pub_date")
+
+
+class CommentaryListSearchForm(forms.Form):
+    """Search for Commentary specified by user (for old CommentaryListView)"""
 
     author = forms.CharField(max_length=100, required=False, label="Author(s)")
     title = forms.CharField(max_length=100, required=False, label="Title")
diff --git a/scipost_django/commentaries/models.py b/scipost_django/commentaries/models.py
index 24455ae60..931e64107 100644
--- a/scipost_django/commentaries/models.py
+++ b/scipost_django/commentaries/models.py
@@ -16,7 +16,7 @@ from .managers import CommentaryManager
 
 class Commentary(TimeStampedModel):
     """
-    A Commentary contains all the contents of a SciPost Commentary page for a given publication.
+    A Commentary page for a given publication.
     """
 
     requested_by = models.ForeignKey(
diff --git a/scipost_django/commentaries/templates/commentaries/_commentary_card_content.html b/scipost_django/commentaries/templates/commentaries/_commentary_card_content.html
index 61d51d713..aabbaf788 100644
--- a/scipost_django/commentaries/templates/commentaries/_commentary_card_content.html
+++ b/scipost_django/commentaries/templates/commentaries/_commentary_card_content.html
@@ -1,4 +1,4 @@
-<div class="card-body px-0">
+<div class="card-body">
   <div class="li commentary">
     <h3 class="title"><a href="{{ commentary.get_absolute_url }}">{{ commentary.title }}</a></h3>
     <p class="authors">
diff --git a/scipost_django/commentaries/tests/test_views.py b/scipost_django/commentaries/tests/test_views.py
index a1177d7db..0f095812d 100644
--- a/scipost_django/commentaries/tests/test_views.py
+++ b/scipost_django/commentaries/tests/test_views.py
@@ -15,7 +15,7 @@ from ..factories import (
     UnpublishedCommentaryFactory,
     UnvettedUnpublishedCommentaryFactory,
 )
-from ..forms import CommentarySearchForm, RequestPublishedArticleForm
+from ..forms import RequestPublishedArticleForm
 from ..models import Commentary
 from ..views import RequestPublishedArticle, prefill_using_DOI, RequestArxivPreprint
 from common.helpers.test import add_groups_and_permissions
diff --git a/scipost_django/commentaries/views.py b/scipost_django/commentaries/views.py
index 20702e4b0..e9ac27b9f 100644
--- a/scipost_django/commentaries/views.py
+++ b/scipost_django/commentaries/views.py
@@ -22,7 +22,7 @@ from .forms import (
     ArxivQueryForm,
     VetCommentaryForm,
     RequestCommentaryForm,
-    CommentarySearchForm,
+    CommentaryListSearchForm,
     RequestPublishedArticleForm,
     RequestArxivPreprintForm,
     CommentSciPostPublication,
@@ -224,7 +224,7 @@ def modify_commentary_request(request, commentary_id):
 
 class CommentaryListView(PaginationMixin, ListView):
     model = Commentary
-    form = CommentarySearchForm
+    form = CommentaryListSearchForm
     paginate_by = 10
     context_object_name = "commentary_list"
 
diff --git a/scipost_django/comments/forms.py b/scipost_django/comments/forms.py
index 29a9eefca..3419e2d3d 100644
--- a/scipost_django/comments/forms.py
+++ b/scipost_django/comments/forms.py
@@ -109,13 +109,13 @@ class CommentSearchForm(forms.Form):
 
     def search_results(self):
         comments = Comment.objects.vetted()
-        if self.acad_field_slug != "all":
+        if self.acad_field_slug and self.acad_field_slug != "all":
             comments = comments.filter(
                 Q(submissions__acad_field__slug=self.acad_field_slug)
                 | Q(reports__submission__acad_field__slug=self.acad_field_slug)
                 | Q(commentaries__acad_field__slug=self.acad_field_slug)
             )
-            if self.specialty_slug:
+            if self.specialty_slug and self.specialty_slug != "all":
                 comments = comments.filter(
                     Q(submissions__specialties__slug=self.specialty_slug)
                     | Q(reports__submission__specialties__slug=self.specialty_slug)
diff --git a/scipost_django/scipost/templates/scipost/navbar.html b/scipost_django/scipost/templates/scipost/navbar.html
index 743595cf9..c21604b37 100644
--- a/scipost_django/scipost/templates/scipost/navbar.html
+++ b/scipost_django/scipost/templates/scipost/navbar.html
@@ -87,6 +87,65 @@
 	    </a>
 	  {% endif %}
 	</li>
+
+	<li class="nav-item dropdown">
+          <a class="nav-link dropdown-toggle" href="#" id="MoreDropdown" role="button" data-bs-toggle="dropdown" aria-haspopup="true" aria-expanded="false"  data-trigger="hover">More</a>
+          <ul class="dropdown-menu" aria-labelledby="MoreDropdown">
+	    <li>
+	      {% if request.path == '/portal' %}
+		<button class="nav-link{% if request.GET.tab == 'reports' %} active{% endif %}"
+			       id="reports-tab"
+			       data-bs-toggle="tab" data-bs-target="#reports"
+			       type="button" role="tab"
+			       aria-controls="reports" aria-selected="true">
+		  Reports
+		</button>
+	      {% else %}
+		<a href="{% url 'scipost:portal' %}?tab=reports">Reports</a>
+	      {% endif %}
+	    </li>
+            <li>
+	      {% if request.path == '/portal' %}
+		<button class="nav-link{% if request.GET.tab == 'comments' %} active{% endif %}"
+			       id="comments-tab"
+			       data-bs-toggle="tab" data-bs-target="#comments"
+			       type="button" role="tab"
+			       aria-controls="comments" aria-selected="true">
+		  Comments
+		</button>
+	      {% else %}
+		<a href="{% url 'scipost:portal' %}?tab=comments">Comments</a>
+	      {% endif %}
+            </li>
+            <li>
+	      {% if request.path == '/portal' %}
+		<button class="nav-link{% if request.GET.tab == 'commentaries' %} active{% endif %}"
+			id="commentaries-tab"
+			data-bs-toggle="tab" data-bs-target="#commentaries"
+			type="button" role="tab"
+			aria-controls="commentaries" aria-selected="true">
+		  Commentaries
+		</button>
+	      {% else %}
+		<a href="{% url 'scipost:portal' %}?tab=commentaries">Commentaries</a>
+	      {% endif %}
+            </li>
+            <li>
+	      {% if request.path == '/portal' %}
+		<button class="nav-link{% if request.GET.tab == 'theses' %} active{% endif %}"
+			id="theses-tab"
+			data-bs-toggle="tab" data-bs-target="#theses"
+			type="button" role="tab"
+			aria-controls="theses" aria-selected="true">
+		  Theses
+		</button>
+	      {% else %}
+		<a href="{% url 'scipost:portal' %}?tab=theses">Theses</a>
+	      {% endif %}
+            </li>
+          </ul>
+	</li>
+
 	<!--
 	     <li class="nav-item" role="presentation">
 	     {% if request.path == '/portal' %}
diff --git a/scipost_django/scipost/templates/scipost/portal/_hx_commentaries.html b/scipost_django/scipost/templates/scipost/portal/_hx_commentaries.html
new file mode 100644
index 000000000..716bee0b7
--- /dev/null
+++ b/scipost_django/scipost/templates/scipost/portal/_hx_commentaries.html
@@ -0,0 +1,42 @@
+{% load crispy_forms_tags %}
+
+<div class="p-3 mb-3 bg-light scipost-bar border">
+  <h1 class="mb-3">SciPost Commentaries</h1>
+  <h2>
+    <a href="{% url 'commentaries:howto' %}">SciPost Commentaries how-to</a>
+  </h2>
+  <h2>
+    <a href="{% url 'commentaries:request_commentary' %}">Request a new Commentary Page</a>
+  </h2>
+</div>
+
+<div class="d-flex justify-content-between">
+  <button class="btn btn-outline-primary" data-bs-toggle="collapse" data-bs-target="#commentariesSearch" aria-expanded="false" aria-controls="commentariesSearch">
+    {% include 'bi/search.html' %}&emsp;Simple search / filter
+  </button>
+  <a class="btn btn-outline-primary ms-2" href="{% url 'scipost:search' %}">
+    {% include 'bi/binoculars-fill.html' %}... or use our advanced search API&emsp;{% include 'bi/arrow-right.html' %}
+  </a>
+</div>
+<div class="collapse" id="commentariesSearch">
+  <div class="card card-body">
+    <form
+	hx-post="{% url 'scipost:portal_hx_commentaries_page' %}?page=1"
+	hx-trigger="load, keyup delay:500ms, change"
+	hx-target="#commentaries-search-results"
+	hx-indicator="#indicator-commentaries-search"
+    >
+      <div id="commentaries-search-form">{% crispy commentaries_search_form %}</div>
+    </form>
+  </div>
+  <div id="indicator-commentaries-search" class="htmx-indicator p-2">
+    <button class="btn btn-warning" type="button" disabled>
+      <strong>Loading...</strong>
+      <div class="spinner-grow spinner-grow-sm ms-2" role="status" aria-hidden="true"></div>
+    </button>
+  </div>
+</div>
+
+<h2 class="highlight mb-0">Commentaries{% if session_acad_field %} in {{ session_acad_field }}: {% if session_specialty %}{{ session_specialty }}{% else %}(all specialties){% endif %}{% else %} (all fields){% endif %}</h2>
+
+<ul id="commentaries-search-results" class="list-unstyled pool-list mt-2"></ul>
diff --git a/scipost_django/scipost/templates/scipost/portal/_hx_commentaries_page.html b/scipost_django/scipost/templates/scipost/portal/_hx_commentaries_page.html
new file mode 100644
index 000000000..1dd4a71a8
--- /dev/null
+++ b/scipost_django/scipost/templates/scipost/portal/_hx_commentaries_page.html
@@ -0,0 +1,28 @@
+{% for commentary in page_obj %}
+  <li class="list-group-item py-2">
+    {% include 'commentaries/_commentary_card_content.html' with commentary=commentary %}
+  </li>
+{% empty %}
+  <li class="list-group-item py-2">
+    None found
+  </li>
+{% endfor %}
+{% if page_obj.has_next %}
+  <li id="next-commentaries-{{ page_obj.number }}">
+    <button class="btn btn-primary m-2" type="button"
+	    hx-post="{% url 'scipost:portal_hx_commentaries_page' %}?page={{ page_obj.next_page_number }}"
+	    hx-include="#commentaries-search-form"
+	    hx-target="#next-commentaries-{{ page_obj.number }}"
+	    hx-swap="outerHTML"
+	    hx-indicator="#indicator-commentaries-page-{{ page_obj.number }}"
+    >
+      Load page {{ page_obj.next_page_number }} (out of {{ page_obj.paginator.num_pages }})
+    </button>
+    <span id="indicator-commentaries-page-{{ page_obj.number }}" class="htmx-indicator p-2">
+      <button class="btn btn-warning" type="button" disabled>
+	<strong>Loading page {{ page_obj.next_page_number }} out of {{ page_obj.paginator.num_pages }}</strong>
+	<div class="spinner-grow spinner-grow-sm ms-2" role="status" aria-hidden="true"></div>
+      </button>
+    </span>
+  </li>
+{% endif %}
diff --git a/scipost_django/scipost/templates/scipost/portal/_hx_comments.html b/scipost_django/scipost/templates/scipost/portal/_hx_comments.html
index 1875b2b36..87f2163c2 100644
--- a/scipost_django/scipost/templates/scipost/portal/_hx_comments.html
+++ b/scipost_django/scipost/templates/scipost/portal/_hx_comments.html
@@ -27,6 +27,6 @@
   </div>
 </div>
 
-<h2 class="highlight mb-0">Comments on objects in {{ session_acad_field }}: {% if session_specialty %}{{ session_specialty }}{% else %}(all specialties){% endif %}</h2>
+<h2 class="highlight mb-0">Comments on objects{% if session_acad_field %} in {{ session_acad_field }}: {% if session_specialty %}{{ session_specialty }}{% else %}(all specialties){% endif %}{% else %} (all fields){% endif %}</h2>
 
 <ul id="comments-search-results" class="list-unstyled pool-list mt-2"></ul>
diff --git a/scipost_django/scipost/templates/scipost/portal/_hx_reports.html b/scipost_django/scipost/templates/scipost/portal/_hx_reports.html
index d4eba9664..c12a3f7fa 100644
--- a/scipost_django/scipost/templates/scipost/portal/_hx_reports.html
+++ b/scipost_django/scipost/templates/scipost/portal/_hx_reports.html
@@ -11,10 +11,10 @@
 <div class="collapse" id="reportsSearch">
   <div class="card card-body">
     <form
-      hx-post="{% url 'scipost:portal_hx_reports_page' %}?page=1"
-	       hx-trigger="load, keyup delay:500ms, change"
-	       hx-target="#reports-search-results"
-	       hx-indicator="#indicator-reports-search"
+	hx-post="{% url 'scipost:portal_hx_reports_page' %}?page=1"
+	hx-trigger="load, keyup delay:500ms, change"
+	hx-target="#reports-search-results"
+	hx-indicator="#indicator-reports-search"
     >
       <div id="reports-search-form">{% crispy reports_search_form %}</div>
     </form>
@@ -27,6 +27,6 @@
   </div>
 </div>
 
-<h2 class="highlight mb-0">Reports on Submissions in {{ session_acad_field }}: {% if session_specialty %}{{ session_specialty }}{% else %}(all specialties){% endif %}</h2>
+<h2 class="highlight mb-0">Reports on Submissions{% if session_acad_field %} in {{ session_acad_field }}: {% if session_specialty %}{{ session_specialty }}{% else %}(all specialties){% endif %}{% else %} (all fields){% endif %}</h2>
 
 <ul id="reports-search-results" class="list-unstyled pool-list mt-2"></ul>
diff --git a/scipost_django/scipost/templates/scipost/portal/_hx_submissions_base.html b/scipost_django/scipost/templates/scipost/portal/_hx_submissions_base.html
index 6332e1c88..b97791602 100644
--- a/scipost_django/scipost/templates/scipost/portal/_hx_submissions_base.html
+++ b/scipost_django/scipost/templates/scipost/portal/_hx_submissions_base.html
@@ -39,8 +39,9 @@
       </h2>
     </div>
     <div class="card-body">
-      <h3>Please consider contributing a Report so we can minimize delays in editorial processing.</h3>
-      <p>To do so, please navigate to the Submission's page and click on the <strong>Contribute a Report</strong> link {% if not user.is_authenticated %}(login required){% endif %}
+      <h3><strong>Please consider contributing one</strong> (even if not explicitly invited to do so)!</h3>
+      <h3>Authors will be grateful, and our editorial processing will remain free of undue delays.</h3>
+      <p>To contribute a Report, please navigate to the Submission's page and click on the <strong>Contribute a Report</strong> link {% if not user.is_authenticated %}(login required){% endif %}
     </div>
   </div>
 
diff --git a/scipost_django/scipost/templates/scipost/portal/_hx_theses.html b/scipost_django/scipost/templates/scipost/portal/_hx_theses.html
new file mode 100644
index 000000000..f3d2a3545
--- /dev/null
+++ b/scipost_django/scipost/templates/scipost/portal/_hx_theses.html
@@ -0,0 +1,37 @@
+{% load crispy_forms_tags %}
+
+<div class="p-3 mb-3 bg-light scipost-bar border">
+  <h1 class="mb-3">SciPost Theses</h1>
+  <h2><a href="{% url 'theses:request_thesislink' %}">Request a new Thesis Link</a></h2>
+</div>
+
+<div class="d-flex justify-content-between">
+  <button class="btn btn-outline-primary" data-bs-toggle="collapse" data-bs-target="#thesesSearch" aria-expanded="false" aria-controls="thesesSearch">
+    {% include 'bi/search.html' %}&emsp;Simple search / filter
+  </button>
+  <a class="btn btn-outline-primary ms-2" href="{% url 'scipost:search' %}">
+    {% include 'bi/binoculars-fill.html' %}... or use our advanced search API&emsp;{% include 'bi/arrow-right.html' %}
+  </a>
+</div>
+<div class="collapse" id="thesesSearch">
+  <div class="card card-body">
+    <form
+      hx-post="{% url 'scipost:portal_hx_theses_page' %}?page=1"
+	       hx-trigger="load, keyup delay:500ms, change"
+	       hx-target="#theses-search-results"
+	       hx-indicator="#indicator-theses-search"
+    >
+      <div id="theses-search-form">{% crispy theses_search_form %}</div>
+    </form>
+  </div>
+  <div id="indicator-theses-search" class="htmx-indicator p-2">
+    <button class="btn btn-warning" type="button" disabled>
+      <strong>Loading...</strong>
+      <div class="spinner-grow spinner-grow-sm ms-2" role="status" aria-hidden="true"></div>
+    </button>
+  </div>
+</div>
+
+<h2 class="highlight mb-0">Thesis links{% if session_acad_field %} in {{ session_acad_field }}: {% if session_specialty %}{{ session_specialty }}{% else %}(all specialties){% endif %}{% else %} (all fields){% endif %}</h2>
+
+<ul id="theses-search-results" class="list-unstyled pool-list mt-2"></ul>
diff --git a/scipost_django/scipost/templates/scipost/portal/_hx_theses_page.html b/scipost_django/scipost/templates/scipost/portal/_hx_theses_page.html
new file mode 100644
index 000000000..d48e488ab
--- /dev/null
+++ b/scipost_django/scipost/templates/scipost/portal/_hx_theses_page.html
@@ -0,0 +1,28 @@
+{% for thesislink in page_obj %}
+  <li class="list-group-item py-2">
+    {% include 'theses/_thesislink_card_content.html' with thesislink=thesislink %}
+  </li>
+{% empty %}
+  <li class="list-group-item py-2">
+    None found
+  </li>
+{% endfor %}
+{% if page_obj.has_next %}
+  <li id="next-theses-{{ page_obj.number }}">
+    <button class="btn btn-primary m-2" type="button"
+	    hx-post="{% url 'scipost:portal_hx_theses_page' %}?page={{ page_obj.next_page_number }}"
+	    hx-include="#theses-search-form"
+	    hx-target="#next-theses-{{ page_obj.number }}"
+	    hx-swap="outerHTML"
+	    hx-indicator="#indicator-theses-page-{{ page_obj.number }}"
+    >
+      Load page {{ page_obj.next_page_number }} (out of {{ page_obj.paginator.num_pages }})
+    </button>
+    <span id="indicator-theses-page-{{ page_obj.number }}" class="htmx-indicator p-2">
+      <button class="btn btn-warning" type="button" disabled>
+	<strong>Loading page {{ page_obj.next_page_number }} out of {{ page_obj.paginator.num_pages }}</strong>
+	<div class="spinner-grow spinner-grow-sm ms-2" role="status" aria-hidden="true"></div>
+      </button>
+    </span>
+  </li>
+{% endif %}
diff --git a/scipost_django/scipost/templates/scipost/portal/portal.html b/scipost_django/scipost/templates/scipost/portal/portal.html
index 2b2929cd4..eb642eecb 100644
--- a/scipost_django/scipost/templates/scipost/portal/portal.html
+++ b/scipost_django/scipost/templates/scipost/portal/portal.html
@@ -90,6 +90,26 @@
       </div>
     </div>
 
+    <div class="tab-pane portal-tab fade{% if request.GET.tab == 'commentaries' %} show active{% endif %}"
+	 id="commentaries"
+	 role="tabpanel" aria-labelledby="commentaries-tab">
+      <div hx-get="{% url 'scipost:portal_hx_commentaries' %}"
+	   hx-trigger="{% if request.GET.tab == 'commentaries' %}load, {% endif %}click delay:200ms from:#commentaries-tab, session-acad-field-set from:body, session-specialty-set from:body"
+	   hx-push-url="{% url 'scipost:portal' %}?tab=commentaries"
+      >
+      </div>
+    </div>
+
+    <div class="tab-pane portal-tab fade{% if request.GET.tab == 'theses' %} show active{% endif %}"
+	 id="theses"
+	 role="tabpanel" aria-labelledby="theses-tab">
+      <div hx-get="{% url 'scipost:portal_hx_theses' %}"
+	   hx-trigger="{% if request.GET.tab == 'theses' %}load, {% endif %}click delay:200ms from:#theses-tab, session-acad-field-set from:body, session-specialty-set from:body"
+	   hx-push-url="{% url 'scipost:portal' %}?tab=theses"
+      >
+      </div>
+    </div>
+
   </div>
 
 {% endblock %}
diff --git a/scipost_django/scipost/tests/test_views.py b/scipost_django/scipost/tests/test_views.py
index 81d83ed49..2e5f2b550 100644
--- a/scipost_django/scipost/tests/test_views.py
+++ b/scipost_django/scipost/tests/test_views.py
@@ -13,7 +13,6 @@ from commentaries.factories import (
     CommentaryFactory,
     UnpublishedCommentaryFactory,
 )
-from commentaries.forms import CommentarySearchForm
 from commentaries.models import Commentary
 
 from ..factories import ContributorFactory
diff --git a/scipost_django/scipost/urls.py b/scipost_django/scipost/urls.py
index 419ff30db..7b3341625 100644
--- a/scipost_django/scipost/urls.py
+++ b/scipost_django/scipost/urls.py
@@ -124,6 +124,22 @@ urlpatterns = [
         views.portal_hx_comments_page,
         name="portal_hx_comments_page",
     ),
+    path(
+        "portal/_hx_commentaries",
+        views.portal_hx_commentaries,
+        name="portal_hx_commentaries"
+    ),
+    path(
+        "portal/_hx_commentaries_page",
+        views.portal_hx_commentaries_page,
+        name="portal_hx_commentaries_page",
+    ),
+    path("portal/_hx_theses", views.portal_hx_theses, name="portal_hx_theses"),
+    path(
+        "portal/_hx_theses_page",
+        views.portal_hx_theses_page,
+        name="portal_hx_theses_page",
+    ),
     path("_hx_news", views._hx_news, name="_hx_news"),
     path(
         "_hx_participates_in",
diff --git a/scipost_django/scipost/views.py b/scipost_django/scipost/views.py
index 6a9f82732..1be2891a3 100644
--- a/scipost_django/scipost/views.py
+++ b/scipost_django/scipost/views.py
@@ -68,6 +68,7 @@ from .utils import EMAIL_FOOTER, SCIPOST_SUMMARY_FOOTER, SCIPOST_SUMMARY_FOOTER_
 
 from colleges.permissions import fellowship_or_admin_required
 from commentaries.models import Commentary
+from commentaries.forms import CommentarySearchForm
 from comments.models import Comment
 from comments.forms import CommentSearchForm
 from invitations.constants import STATUS_REGISTERED
@@ -83,6 +84,7 @@ from profiles.models import Profile
 from submissions.models import Submission, RefereeInvitation, Report, EICRecommendation
 from submissions.forms import SubmissionSearchForm, ReportSearchForm
 from theses.models import ThesisLink
+from theses.forms import ThesisSearchForm
 
 
 ###########
@@ -344,7 +346,7 @@ def portal_hx_reports_page(request):
         reports = Report.objects.accepted()
     if session_acad_field_slug and session_acad_field_slug != "all":
         reports = reports.filter(submission__acad_field__slug=session_acad_field_slug)
-        if session_specialty_slug:
+        if session_specialty_slug and session_specialty_slug != "all":
             reports = reports.filter(
                 submission__specialties__slug=session_specialty_slug
             )
@@ -382,7 +384,7 @@ def portal_hx_comments_page(request):
             | Q(reports__submission__acad_field__slug=session_acad_field_slug)
             | Q(commentaries__acad_field__slug=session_acad_field_slug)
         )
-        if session_specialty_slug:
+        if session_specialty_slug and session_specialty_slug != "all":
             comments = comments.filter(
                 Q(submissions__specialties__slug=session_specialty_slug)
                 | Q(reports__submission__specialties__slug=session_specialty_slug)
@@ -395,6 +397,58 @@ def portal_hx_comments_page(request):
     return render(request, "scipost/portal/_hx_comments_page.html", context)
 
 
+def portal_hx_commentaries(request):
+    form = CommentarySearchForm(
+        acad_field_slug=request.session.get("session_acad_field_slug", None),
+        specialty_slug=request.session.get("session_specialty_slug", None),
+    )
+    context = {"commentaries_search_form": form}
+    return render(request, "scipost/portal/_hx_commentaries.html", context)
+
+
+def portal_hx_commentaries_page(request):
+    session_acad_field_slug = request.session.get("session_acad_field_slug", None)
+    session_specialty_slug = request.session.get("session_specialty_slug", None)
+    form = CommentarySearchForm(
+        request.POST or None,
+        acad_field_slug=session_acad_field_slug,
+        specialty_slug=session_specialty_slug,
+    )
+    form.is_valid() # trigger validation to get filtering
+    commentaries = form.search_results()
+    paginator = Paginator(commentaries, 10)
+    page_nr = request.GET.get("page")
+    page_obj = paginator.get_page(page_nr)
+    context = {"page_obj": page_obj}
+    return render(request, "scipost/portal/_hx_commentaries_page.html", context)
+
+
+def portal_hx_theses(request):
+    form = ThesisSearchForm(
+        acad_field_slug=request.session.get("session_acad_field_slug", None),
+        specialty_slug=request.session.get("session_specialty_slug", None),
+    )
+    context = {"theses_search_form": form}
+    return render(request, "scipost/portal/_hx_theses.html", context)
+
+
+def portal_hx_theses_page(request):
+    session_acad_field_slug = request.session.get("session_acad_field_slug", None)
+    session_specialty_slug = request.session.get("session_specialty_slug", None)
+    form = ThesisSearchForm(
+        request.POST or None,
+        acad_field_slug=session_acad_field_slug,
+        specialty_slug=session_specialty_slug,
+    )
+    form.is_valid() # trigger validation to get filtering
+    theses = form.search_results()
+    paginator = Paginator(theses, 10)
+    page_nr = request.GET.get("page")
+    page_obj = paginator.get_page(page_nr)
+    context = {"page_obj": page_obj}
+    return render(request, "scipost/portal/_hx_theses_page.html", context)
+
+
 def _hx_news(request):
     if NewsItem.objects.homepage().exists():
         latest_newsitem_id = NewsItem.objects.homepage().order_by("-date").first().id
diff --git a/scipost_django/submissions/forms.py b/scipost_django/submissions/forms.py
index b3b1ee705..9a33606db 100644
--- a/scipost_django/submissions/forms.py
+++ b/scipost_django/submissions/forms.py
@@ -143,7 +143,7 @@ class SubmissionSearchForm(forms.Form):
         Return all Submission objects fitting search criteria.
         """
         submissions = Submission.objects.public_newest().unpublished()
-        if self.acad_field_slug != "all":
+        if self.acad_field_slug and self.acad_field_slug != "all":
             submissions = submissions.filter(acad_field__slug=self.acad_field_slug)
             if self.specialty_slug and self.specialty_slug != "all":
                 submissions = submissions.filter(specialties__slug=self.specialty_slug)
@@ -412,9 +412,9 @@ class ReportSearchForm(forms.Form):
 
     def search_results(self):
         reports = Report.objects.accepted()
-        if self.acad_field_slug != "all":
+        if self.acad_field_slug and self.acad_field_slug != "all":
             reports = reports.filter(submission__acad_field__slug=self.acad_field_slug)
-            if self.specialty_slug:
+            if self.specialty_slug and self.specialty_slug != "all":
                 reports = reports.filter(
                     submission__specialties__slug=self.specialty_slug
                 )
diff --git a/scipost_django/theses/forms.py b/scipost_django/theses/forms.py
index 1120deb0c..70ef57f20 100644
--- a/scipost_django/theses/forms.py
+++ b/scipost_django/theses/forms.py
@@ -7,6 +7,10 @@ from django.contrib.sites.models import Site
 from django.core.mail import EmailMessage
 from django.template.loader import render_to_string
 
+from crispy_forms.helper import FormHelper
+from crispy_forms.layout import Layout, Div
+from crispy_bootstrap5.bootstrap5 import FloatingField
+
 from scipost.models import Contributor
 from scipost.utils import build_absolute_uri_using_site
 
@@ -138,3 +142,55 @@ class ThesisLinkSearchForm(forms.Form):
         max_length=1000, required=False, label="Abstract"
     )
     supervisor = forms.CharField(max_length=100, required=False, label="Supervisor")
+
+
+class ThesisSearchForm(forms.Form):
+    author = forms.CharField(max_length=100, required=False, label="Author")
+    title = forms.CharField(max_length=100, label="Title", required=False)
+    abstract = forms.CharField(
+        max_length=1000, required=False, label="Abstract"
+    )
+    supervisor = forms.CharField(max_length=100, required=False, label="Supervisor")
+
+    def __init__(self, *args, **kwargs):
+        self.acad_field_slug = kwargs.pop("acad_field_slug")
+        self.specialty_slug = kwargs.pop("specialty_slug")
+        super().__init__(*args, **kwargs)
+        self.helper = FormHelper()
+        self.helper.layout = Layout(
+            Div(
+                FloatingField("author"),
+                FloatingField("title"),
+                FloatingField("abstract"),
+                FloatingField("supervisor"),
+            ),
+        )
+
+    def search_results(self):
+        """Return all ThesisLink objects fitting search"""
+        theses = ThesisLink.objects.vetted()
+        if self.acad_field_slug and self.acad_field_slug != "all":
+            theses = theses.filter(acad_field__slug=self.acad_field_slug)
+            if self.specialty_slug and self.specialty_slug != "all":
+                theses = theses.filter(
+                    specialties__slug=self.specialty_slug
+                )
+        if hasattr(self, "cleaned_data"):
+            if "title" in self.cleaned_data:
+                theses = theses.filter(
+                    title__icontains=self.cleaned_data["title"],
+                )
+                len(theses)
+            if "abstract" in self.cleaned_data:
+                theses = theses.filter(
+                    abstract__icontains=self.cleaned_data["abstract"],
+                )
+            if "author" in self.cleaned_data:
+                theses = theses.filter(
+                    author__icontains=self.cleaned_data["author"],
+                )
+            if "supervisor" in self.cleaned_data:
+                theses = theses.filter(
+                    supervisor__icontains=self.cleaned_data["supervisor"],
+                )
+        return theses.order_by("-defense_date")
diff --git a/scipost_django/theses/templates/theses/_thesislink_card_content.html b/scipost_django/theses/templates/theses/_thesislink_card_content.html
index 7d37452df..e1520ae23 100644
--- a/scipost_django/theses/templates/theses/_thesislink_card_content.html
+++ b/scipost_django/theses/templates/theses/_thesislink_card_content.html
@@ -1,4 +1,4 @@
-<div class="card-body px-0">
+<div class="card-body">
   <div class="li thesis">
     <h3 class="specialties">{{ thesislink.acad_field }}</h3>
     <ul class="list-inline">
-- 
GitLab