diff --git a/.bootstraprc b/.bootstraprc
index ab4a0cccb6882f155316d46dff87435d40a1502a..cd4c5b095e1f3274f4a76081fc0879c080796e86 100644
--- a/.bootstraprc
+++ b/.bootstraprc
@@ -9,6 +9,8 @@
     ],
     "extractStyles": true,
     "styles": {
+        # First tables, due to self dependencies
+        "tables": true,
         "alert": true,
         "button-group": true,
         "buttons": true,
diff --git a/commentaries/factories.py b/commentaries/factories.py
index 31f1354fa2c7c811daccdbfa7bd9a27cdf55850f..60f1625f2f365bd240db1d1aa7e5d7beb2a9fa73 100644
--- a/commentaries/factories.py
+++ b/commentaries/factories.py
@@ -2,11 +2,12 @@ import factory
 
 from scipost.constants import DISCIPLINE_PHYSICS, SCIPOST_SUBJECT_AREAS
 from scipost.factories import ContributorFactory
-from journals.models import SCIPOST_JOURNALS_DOMAINS
+from journals.constants import SCIPOST_JOURNALS_DOMAINS
 from common.helpers import random_arxiv_identifier_with_version_number
 
 from .models import Commentary, COMMENTARY_TYPES
 
+
 class CommentaryFactory(factory.django.DjangoModelFactory):
     class Meta:
         model = Commentary
diff --git a/commentaries/models.py b/commentaries/models.py
index 68860aa7f0763f6f81e6452d414366562d7361cd..eb7af81f0b08132b47e5e907a2ac37784ff524d8 100644
--- a/commentaries/models.py
+++ b/commentaries/models.py
@@ -1,12 +1,16 @@
 from django.db import models
 from django.contrib.postgres.fields import JSONField
+from django.core.urlresolvers import reverse
 from django.template import Template, Context
 
-from journals.models import SCIPOST_JOURNALS_DOMAINS
+from journals.constants import SCIPOST_JOURNALS_DOMAINS
 from scipost.behaviors import ArxivCallable
 from scipost.models import TimeStampedModel, Contributor
 from scipost.constants import SCIPOST_DISCIPLINES, DISCIPLINE_PHYSICS, SCIPOST_SUBJECT_AREAS
 
+from .managers import CommentaryManager
+
+
 COMMENTARY_PUBLISHED = 'published'
 COMMENTARY_PREPRINT = 'preprint'
 COMMENTARY_TYPES = (
@@ -15,14 +19,6 @@ COMMENTARY_TYPES = (
 )
 
 
-class CommentaryManager(models.Manager):
-    def vetted(self, **kwargs):
-        return self.filter(vetted=True, **kwargs)
-
-    def awaiting_vetting(self, **kwargs):
-        return self.filter(vetted=False, **kwargs)
-
-
 class Commentary(ArxivCallable, TimeStampedModel):
     """
     A Commentary contains all the contents of a SciPost Commentary page for a given publication.
@@ -87,49 +83,9 @@ class Commentary(ArxivCallable, TimeStampedModel):
     def same_version_exists(self, identifier):
         return self.objects.filter(arxiv_identifier=identifier).exists()
 
-    def header_as_table(self):
-        # for display in Commentary page itself
-        header = ('<table>'
-                  '<tr><td>Title: </td><td>&nbsp;</td><td>{{ pub_title }}</td></tr>'
-                  '<tr><td>Author(s): </td><td>&nbsp;</td><td>{{ author_list }}</td></tr>'
-                  '<tr><td>As Contributors: </td><td>&nbsp;</td>')
-        if self.authors.all():
-            header += '<td>'
-            for auth in self.authors.all():
-                header += ('<a href="/contributor/' + str(auth.id) + '">' + auth.user.first_name
-                           + ' ' + auth.user.last_name + '</a>,&nbsp;')
-            header += '</td>'
-        else:
-            header += '<td>(none claimed)</td>'
-        header += '</tr>'
-        if self.type == 'published':
-            header += ('<tr><td>Journal ref.: </td><td>&nbsp;</td><td>{{ journal }} {{ volume }}, '
-                       '{{ pages }}</td></tr>'
-                       '<tr><td>DOI: </td><td>&nbsp;</td><td><a href="{{ pub_DOI_link }}" '
-                       'target="_blank">{{ pub_DOI_link }}</a></td></tr>')
-        elif self.type == 'preprint':
-            header += ('<tr><td>arxiv Link: </td><td>&nbsp;</td><td><a href="{{ arxiv_link }}">'
-                       '{{ arxiv_link }}</a></td></tr>')
-        if self.pub_date:
-            header += '<tr><td>Date: </td><td>&nbsp;</td><td>{{ pub_date }}</td></tr>'
-        header += '</table>'
-        template = Template(header)
-        context = Context({
-            'pub_title': self.pub_title, 'author_list': self.author_list,
-        })
-        if self.type == 'published':
-            context['journal'] = self.journal
-            context['volume'] = self.volume
-            context['pages'] = self.pages
-            context['pub_DOI_link'] = self.pub_DOI_link
-            context['pub_date'] = self.pub_date
-        elif self.type == 'preprint':
-            context['arxiv_link'] = self.arxiv_link
-        return template.render(context)
-
     def title_label(self):
         context = Context({
-            'scipost_url': self.scipost_url(),
+            'scipost_url': reverse('commentaries:commentary', args=(self.arxiv_or_DOI_string,)),
             'pub_title': self.pub_title
         })
         template = Template('<a href="{{scipost_url}}" class="pubtitleli">{{pub_title}}</a>')
@@ -146,11 +102,3 @@ class Commentary(ArxivCallable, TimeStampedModel):
 
         if commit:
             self.save()
-
-    def scipost_url(self):
-        """ Returns the url of the SciPost Commentary Page """
-        return '/commentary/' + self.arxiv_or_DOI_string
-
-    def scipost_url_full(self):
-        """ Returns the url of the SciPost Commentary Page """
-        return 'https://scipost.org/commentary/' + self.arxiv_or_DOI_string
diff --git a/commentaries/templates/commentaries/_commentary_card_content.html b/commentaries/templates/commentaries/_commentary_card_content.html
index 6724ee64fb06a93f49d90aac0c5f062297bac03f..a00cd4ca86eda5aeae8d6dd1be4e027158de80b0 100644
--- a/commentaries/templates/commentaries/_commentary_card_content.html
+++ b/commentaries/templates/commentaries/_commentary_card_content.html
@@ -1,6 +1,6 @@
 <div class="card-block">
     <h3 class="card-title">
-        <a href="{{ commentary.scipost_url }}">{{ commentary.pub_title }}</a>
+        <a href="{% url 'commentaries:commentary' commentary.arxiv_or_DOI_string %}">{{ commentary.pub_title }}</a>
     </h3>
     <p class="card-text">
         by {{ commentary.author_list }}{% if commentary.type == 'published' %}, {{ commentary.journal }} {{ commentary.volume }}, {{ commentary.pages }}{% elif commentary.type == 'preprint' %} - <a href="{{ commentary.arxiv_link }}">{{ commentary.arxiv_link }}</a>{% endif %}
diff --git a/commentaries/templates/commentaries/commentary_detail.html b/commentaries/templates/commentaries/commentary_detail.html
index 0e253b730ee81cdb3feed44d1df665b63bf09715..5b47e90f8844ea97816973ff8de2271abd9a271c 100644
--- a/commentaries/templates/commentaries/commentary_detail.html
+++ b/commentaries/templates/commentaries/commentary_detail.html
@@ -38,7 +38,7 @@
 <div class="row">
     <div class="col-12">
         <h2>Original publication: </h2>
-        {{ commentary.header_as_table }}
+        {% include 'commentaries/_commentary_summary.html' with commentary=commentary %}
     </div>
 </div>
 
diff --git a/commentaries/templates/commentaries/vet_commentary_requests.html b/commentaries/templates/commentaries/vet_commentary_requests.html
index 1211f37e610d29a04277ab74fae16951f2150198..70edddb6e8eaca847e8337f0d5b1d003e759daae 100644
--- a/commentaries/templates/commentaries/vet_commentary_requests.html
+++ b/commentaries/templates/commentaries/vet_commentary_requests.html
@@ -16,7 +16,7 @@
   <hr>
   <div class="row">
     <div class="col-8">
-      {{ commentary_to_vet.header_as_table }}
+      {% include 'commentaries/_commentary_summary.html' with commentary=commentary_to_vet %}
       <h3>Abstract:</h3>
       <p>{{ commentary_to_vet.pub_abstract }}</p>
 
diff --git a/comments/admin.py b/comments/admin.py
index 69da3613e39f4bcc268141a5f3cab8d80e60e575..ac6a36e65aab47674d8579434abd6d721b104d8f 100644
--- a/comments/admin.py
+++ b/comments/admin.py
@@ -1,10 +1,19 @@
 from django.contrib import admin
 
-from comments.models import Comment
+from .constants import STATUS_VETTED
+from .models import Comment
+
+
+def comment_is_vetted(comment):
+    '''Check if comment is vetted.'''
+    return comment.status is STATUS_VETTED
 
 
 class CommentAdmin(admin.ModelAdmin):
-    search_fields = ['comment_text', 'author__user__last_name']
+    list_display = ('comment_text', 'author', 'date_submitted', comment_is_vetted)
+    date_hierarchy = 'date_submitted'
+    list_filter = ('status',)
+    comment_is_vetted.boolean = True
 
 
 admin.site.register(Comment, CommentAdmin)
diff --git a/comments/constants.py b/comments/constants.py
index 9361790ed061dcd64341a532a3920e7a000b708e..3b3cdcc44efc65dbf0c861ce2fb40c067464def7 100644
--- a/comments/constants.py
+++ b/comments/constants.py
@@ -1,3 +1,43 @@
 EXTENTIONS_IMAGES = ['.jpg', '.png']
 EXTENTIONS_PDF = ['.pdf']
 EXTENTIONS_FILES = EXTENTIONS_PDF + EXTENTIONS_IMAGES
+
+STATUS_VETTED = 1
+STATUS_PENDING = 0
+STATUS_UNCLEAR = -1
+STATUS_INCORRECT = -2
+STATUS_NOT_USEFUL = -3
+COMMENT_STATUS = (
+    (STATUS_VETTED, 'vetted'),
+    (STATUS_PENDING, 'not yet vetted (pending)'),
+    (STATUS_UNCLEAR, 'rejected (unclear)'),
+    (STATUS_INCORRECT, 'rejected (incorrect)'),
+    (STATUS_NOT_USEFUL, 'rejected (not useful)'),
+)
+
+COMMENT_CATEGORIES = (
+    ('ERR', 'erratum'),
+    ('REM', 'remark'),
+    ('QUE', 'question'),
+    ('ANS', 'answer to question'),
+    ('OBJ', 'objection'),
+    ('REP', 'reply to objection'),
+    ('VAL', 'validation or rederivation'),
+    ('LIT', 'pointer to related literature'),
+    ('SUG', 'suggestion for further work'),
+)
+
+COMMENT_ACTION_ACCEPT = 1
+COMMENT_ACTION_REFUSE = 2
+COMMENT_ACTION_CHOICES = (
+    (COMMENT_ACTION_ACCEPT, 'accept'),
+    (COMMENT_ACTION_REFUSE, 'refuse (give reason below)'),
+)
+
+COMMENT_REFUSAL_EMPTY = 0
+COMMENT_REFUSAL_CHOICES = (
+    (COMMENT_REFUSAL_EMPTY, '-'),
+    (STATUS_UNCLEAR, 'unclear'),
+    (STATUS_INCORRECT, 'incorrect'),
+    (STATUS_NOT_USEFUL, 'not useful'),
+)
diff --git a/comments/forms.py b/comments/forms.py
index f38c9d933f5b177150cb4f43263c983d41c94c8b..8aace94b0c36586c54abc63509f04588f093ff64 100644
--- a/comments/forms.py
+++ b/comments/forms.py
@@ -1,25 +1,12 @@
 from django import forms
 
+from .constants import COMMENT_ACTION_CHOICES, COMMENT_REFUSAL_CHOICES
 from .models import Comment
 
 from crispy_forms.helper import FormHelper
 from crispy_forms.layout import Layout, Div, Field, Fieldset, HTML, Submit
 
 
-COMMENT_ACTION_CHOICES = (
-    (1, 'accept'),
-    (2, 'refuse (give reason below)'),
-    )
-
-COMMENT_REFUSAL_CHOICES = (
-    (0, '-'),
-    (-1, 'unclear'),
-    (-2, 'incorrect'),
-    (-3, 'not useful'),
-    )
-comment_refusal_dict = dict(COMMENT_REFUSAL_CHOICES)
-
-
 class CommentForm(forms.ModelForm):
     class Meta:
         model = Comment
@@ -52,7 +39,8 @@ class CommentForm(forms.ModelForm):
                         style="border: 0px; font-size: 90%"),
                     HTML('<br>'),
                     Div(
-                        Submit('submit', 'Submit your Comment for vetting', css_class="submitButton"),
+                        Submit('submit', 'Submit your Comment for vetting',
+                               css_class="submitButton"),
                         HTML('<p id="goodCommenter"><i>By clicking on Submit, you agree with the '
                              '<a href="{% url \'scipost:terms_and_conditions\' %}">'
                              'Terms and Conditions</a>.</i></p>'),
diff --git a/comments/models.py b/comments/models.py
index eff1cf1f0f6dfc5799433cc6f666a4649a722aa0..d2e5aed35164433256ff1eaf93dcde89cae7d977 100644
--- a/comments/models.py
+++ b/comments/models.py
@@ -1,44 +1,21 @@
 from django.db import models
 from django.shortcuts import get_object_or_404
-from django.template import Template, Context
-from django.utils.safestring import mark_safe
-
-from .behaviors import validate_file_extension, validate_max_file_size
 
 from commentaries.models import Commentary
 from scipost.models import TimeStampedModel, Contributor
 from submissions.models import Submission, Report
 from theses.models import ThesisLink
 
+from .behaviors import validate_file_extension, validate_max_file_size
+from .constants import COMMENT_STATUS, STATUS_PENDING
 from .managers import CommentManager
 
-COMMENT_CATEGORIES = (
-    ('ERR', 'erratum'),
-    ('REM', 'remark'),
-    ('QUE', 'question'),
-    ('ANS', 'answer to question'),
-    ('OBJ', 'objection'),
-    ('REP', 'reply to objection'),
-    ('VAL', 'validation or rederivation'),
-    ('LIT', 'pointer to related literature'),
-    ('SUG', 'suggestion for further work'),
-    )
-
-COMMENT_STATUS = (
-    (1, 'vetted'),
-    (0, 'not yet vetted (pending)'),
-    (-1, 'rejected (unclear)'),
-    (-2, 'rejected (incorrect)'),
-    (-3, 'rejected (not useful)'),
-)
-comment_status_dict = dict(COMMENT_STATUS)
-
 
 class Comment(TimeStampedModel):
     """ A Comment is an unsollicited note, submitted by a Contributor,
     on a particular publication or in reply to an earlier Comment. """
 
-    status = models.SmallIntegerField(default=0)
+    status = models.SmallIntegerField(default=STATUS_PENDING, choices=COMMENT_STATUS)
     vetted_by = models.ForeignKey(
         Contributor,
         blank=True,
@@ -48,16 +25,18 @@ class Comment(TimeStampedModel):
     )
     file_attachment = models.FileField(
         upload_to='uploads/comments/%Y/%m/%d/', blank=True,
-        validators=[validate_file_extension,
-        validate_max_file_size]
+        validators=[validate_file_extension, validate_max_file_size]
     )
     # a Comment is either for a Commentary or Submission or a ThesisLink.
     commentary = models.ForeignKey(Commentary, blank=True, null=True, on_delete=models.CASCADE)
     submission = models.ForeignKey(Submission, blank=True, null=True, on_delete=models.CASCADE)
     thesislink = models.ForeignKey(ThesisLink, blank=True, null=True, on_delete=models.CASCADE)
     is_author_reply = models.BooleanField(default=False)
-    in_reply_to_comment = models.ForeignKey('self', blank=True, null=True, related_name="nested_comments", on_delete=models.CASCADE)
-    in_reply_to_report = models.ForeignKey(Report, blank=True, null=True, on_delete=models.CASCADE)
+    in_reply_to_comment = models.ForeignKey('self', blank=True, null=True,
+                                            related_name="nested_comments",
+                                            on_delete=models.CASCADE)
+    in_reply_to_report = models.ForeignKey(Report, blank=True, null=True,
+                                           on_delete=models.CASCADE)
     author = models.ForeignKey(Contributor, on_delete=models.CASCADE)
     anonymous = models.BooleanField(default=False, verbose_name='Publish anonymously')
     # Categories:
@@ -71,7 +50,8 @@ class Comment(TimeStampedModel):
     is_lit = models.BooleanField(default=False, verbose_name='pointer to related literature')
     is_sug = models.BooleanField(default=False, verbose_name='suggestion for further work')
     comment_text = models.TextField()
-    remarks_for_editors = models.TextField(default='', blank=True, verbose_name='optional remarks for the Editors only')
+    remarks_for_editors = models.TextField(default='', blank=True,
+                                           verbose_name='optional remarks for the Editors only')
     date_submitted = models.DateTimeField('date submitted')
     # Opinions
     nr_A = models.PositiveIntegerField(default=0)
@@ -84,6 +64,7 @@ class Comment(TimeStampedModel):
         related_name='in_disagreement',
         blank=True
     )
+
     objects = CommentManager()
 
     def __str__(self):
@@ -118,136 +99,3 @@ class Comment(TimeStampedModel):
         self.nr_N = self.in_notsure.count()
         self.nr_D = self.in_disagreement.count()
         self.save()
-
-    def opinions_as_ul(self):
-        template = Template('''
-        <ul class="opinionsDisplay">
-        <li style="background-color: #000099">Agree {{ nr_A }}</li>
-        <li style="background-color: #555555">Not sure {{ nr_N }}</li>
-        <li style="background-color: #990000">Disagree {{ nr_D }}</li>
-        </ul>
-        ''')
-        context = Context({'nr_A': self.nr_A, 'nr_N': self.nr_N, 'nr_D': self.nr_D})
-        return template.render(context)
-
-    def opinions_as_ul_tiny(self):
-        template = Template('''
-        <ul class="opinionsDisplay">
-        <li style="background-color: #000099; font-size: 8px; padding: 2px;">Agree {{ nr_A }}</li>
-        <li style="background-color: #555555; font-size: 8px; padding: 2px;">Not sure {{ nr_N }}</li>
-        <li style="background-color: #990000; font-size: 8px; padding: 2px;">Disagree {{ nr_D }}</li>
-        </ul>
-        ''')
-        context = Context({'nr_A': self.nr_A, 'nr_N': self.nr_N, 'nr_D': self.nr_D})
-        return template.render(context)
-
-    def print_identifier(self):
-        # for display
-        output = '<div class="commentid">\n'
-        output += '<h3><a id="comment_id{{ id }}"></a>'
-        context = Context({'id': self.id})
-        if self.is_author_reply:
-            output += 'Author '
-        if not self.anonymous:
-            output += (' <a href="/contributor/{{ author_id }}">'
-                       + '{{ first_name }} {{ last_name }}</a> on ')
-            context['author_id'] = self.author.id
-            context['first_name'] = self.author.user.first_name
-            context['last_name'] = self.author.user.last_name
-        output += '{{ date_comment_submitted }}'
-        context['date_comment_submitted'] = self.date_submitted.strftime("%Y-%m-%d")
-        if self.in_reply_to_comment:
-            output += (' (in reply to <a href="#comment_id{{ in_reply_to_comment_id }}">'
-                       '{{ in_reply_to_comment_first_name }} '
-                       '{{ in_reply_to_comment_last_name }} on '
-                       '{{ in_reply_to_comment_date }}</a>)')
-            context['in_reply_to_comment_id'] = self.in_reply_to_comment_id
-            context['in_reply_to_comment_first_name'] = (self.in_reply_to_comment
-                                                         .author.user.first_name)
-            context['in_reply_to_comment_last_name'] = (self.in_reply_to_comment
-                                                        .author.user.last_name)
-            context['in_reply_to_comment_date'] = (self.in_reply_to_comment
-                                                   .date_submitted.strftime("%Y-%m-%d"))
-        elif self.in_reply_to_report:
-            output += ' (in reply to <a href="#report_id{{ in_reply_to_report_id }}">'
-            context['in_reply_to_report_id'] = self.in_reply_to_report_id
-            if not self.in_reply_to_report.anonymous:
-                output += '{{ in_reply_to_report_first_name }} {{ in_reply_to_report_last_name}}'
-                context['in_reply_to_report_first_name'] = (self.in_reply_to_report
-                                                            .author.user.first_name)
-                context['in_reply_to_report_last_name'] = (self.in_reply_to_report
-                                                           .author.user.last_name)
-            else:
-                output += 'Report {{ in_reply_to_report_id }}'
-                context['in_reply_to_report_id'] = self.in_reply_to_report_id
-            output += ' on {{ date_report_submitted }}</a>)'
-            context['date_report_submitted'] = self.in_reply_to_report.date_submitted.strftime(
-                                                                                        "%Y-%m-%d")
-        output += '</h3></div>'
-        template = Template(output)
-        return template.render(context)
-
-    def print_identifier_for_vetting(self):
-        # for display, same as print_identifier but named even if anonymous, not linked
-        output = '<div class="commentid">\n'
-        output += '<h3>'
-        context = Context()
-        if self.is_author_reply:
-            output += 'Author '
-        output += ' <a href="/contributor/{{ author_id }}">{{ first_name }} {{ last_name }}</a>'
-        context['author_id'] = self.author.id
-        context['first_name'] = self.author.user.first_name
-        context['last_name'] = self.author.user.last_name
-        output += ' on {{ date_submitted }}'
-        context['date_submitted'] = self.date_submitted.strftime("%Y-%m-%d")
-        if self.in_reply_to_comment:
-            output += (' (in reply to <a href="#comment_id{{ in_reply_to_comment_id }}">'
-                       '{{ in_reply_to_comment_first_name }} {{ in_reply_to_comment_last_name }} '
-                       'on {{ in_reply_to_comment_date }}</a>)')
-            context['in_reply_to_comment_id'] = self.in_reply_to_comment_id
-            context['in_reply_to_comment_first_name'] = (self.in_reply_to_comment
-                                                         .author.user.first_name)
-            context['in_reply_to_comment_last_name'] = (self.in_reply_to_comment
-                                                        .author.user.last_name)
-            context['in_reply_to_comment_date'] = (self.in_reply_to_comment
-                                                   .date_submitted.strftime("%Y-%m-%d"))
-        elif self.in_reply_to_report:
-            output += ' (in reply to <a href="#report_id{{ in_reply_to_report_id }}">'
-            context['in_reply_to_report_id'] = self.in_reply_to_report_id
-            if not self.in_reply_to_report.anonymous:
-                output += '{{ in_reply_to_report_first_name }} {{ in_reply_to_report_last_name}}'
-                context['in_reply_to_report_first_name'] = (self.in_reply_to_report
-                                                            .author.user.first_name)
-                context['in_reply_to_report_last_name'] = (self.in_reply_to_report
-                                                           .author.user.last_name)
-            else:
-                output += 'Report {{ in_reply_to_report_id }}'
-                context['in_reply_to_report_id'] = self.in_reply_to_report_id
-            output += '</a> on {{ date_submitted }})'
-            context['date_submitted'] = self.in_reply_to_report.date_submitted.strftime("%Y-%m-%d")
-        output += '</h3></div>'
-        template = Template(output)
-        return template.render(context)
-
-    def categories_as_ul(self):
-        output = '<div class="commentcategorydisplay"><h4>Category:</h4><ul>'
-        if self.is_rem:
-            output += '<li>remark</li>'
-        if self.is_que:
-            output += '<li>question</li>'
-        if self.is_ans:
-            output += '<li>answer to question</li>'
-        if self.is_obj:
-            output += '<li>objection</li>'
-        if self.is_rep:
-            output += '<li>reply to objection</li>'
-        if self.is_cor:
-            output += '<li>correction</li>'
-        if self.is_val:
-            output += '<li>validation or rederivation</li>'
-        if self.is_lit:
-            output += '<li>pointer to related literature</li>'
-        if self.is_sug:
-            output += '<li>suggestion for further work</li>'
-        output += '</ul></div>'
-        return mark_safe(output)
diff --git a/comments/templates/comments/_comment_card_content.html b/comments/templates/comments/_comment_card_content.html
index d4cec6b538ca30ec1de1c5c2f770f2c6bfd9b5d4..fba6a0aee5a47323a4c8a44090b04290a72b5f62 100644
--- a/comments/templates/comments/_comment_card_content.html
+++ b/comments/templates/comments/_comment_card_content.html
@@ -20,7 +20,7 @@
             in commentary on <a href="{% url 'commentaries:commentary' comment.commentary.arxiv_or_DOI_string %}" class="pubtitleli">{{comment.commentary.pub_title}}</a>
             <span class="text-muted">by {{comment.commentary.author_list}}</span>
         {% elif comment.thesislink %}
-            <a href="{% url 'commentaries:commentary' comment.commentary.arxiv_or_DOI_string %}#comment_id{{comment.id}}">
+            <a href="{% url 'theses:thesis' comment.thesislink.id %}#comment_id{{comment.id}}">
                 "{{comment.comment_text|slice:'30'}}{% if comment.comment_text|length > 30 %}...{% endif %}"
             </a>
             </p><p class="card-text pl-md-3">
diff --git a/comments/templates/comments/_single_comment.html b/comments/templates/comments/_single_comment.html
index 84a01f82c7516645d75c49cb00a1fe976da6fb73..1eaaf85a016bf09f61c8b349bd7339cbae35e843 100644
--- a/comments/templates/comments/_single_comment.html
+++ b/comments/templates/comments/_single_comment.html
@@ -8,7 +8,7 @@
 
             <div class="row">
                 <div class="col-12">
-                    {{ comment.print_identifier }}
+                    {% include 'comments/_comment_identifier.html' with comment=comment %}
 
                     {% include 'comments/_comment_categories.html' with comment=comment class='mr-2' %}
 
diff --git a/comments/templates/comments/reply_to_comment.html b/comments/templates/comments/reply_to_comment.html
index 3f2b8e81a153f0ee890db3e4b19657607d52db56..3937e5d0051a557def52eb7728998e2b5f0ccbf3 100644
--- a/comments/templates/comments/reply_to_comment.html
+++ b/comments/templates/comments/reply_to_comment.html
@@ -4,32 +4,10 @@
 
 {% block content %}
 
-<script>
-  $(document).ready(function(){
-
-    var comment_text_input = $("#id_comment_text");
-
-    function set_comment_text(value) {
-      $("#preview-comment_text").text(value)
-    }
-    set_comment_text(comment_text_input.val())
-
-    comment_text_input.keyup(function(){
-      var new_text = $(this).val()
-      set_comment_text(new_text)
-      MathJax.Hub.Queue(["Typeset",MathJax.Hub]);
-    })
-
-  });
-</script>
-
-
 {% if user.is_authenticated %}
     <div class="row">
         <div class="col-12">
-          <div class="panel">
-              <h1>SciPost Reply to Comment Page</h1>
-          </div>
+            <h1 class="highlight">SciPost Reply to Comment Page</h1>
         </div>
     </div>
 
@@ -37,7 +15,7 @@
         <div class="col-12">
             {% if comment.commentary %}
                 <h2>The Commentary concerned:</h2>
-                {{ comment.commentary.header_as_table }}
+                {% include 'commentaries/_commentary_summary.html' with commentary=comment.commentary %}
 
                 <h3 class="mt-3">Abstract:</h3>
                 <p>{{ comment.commentary.pub_abstract }}</p>
@@ -45,21 +23,18 @@
 
             {% if comment.submission %}
                 <h2>The Submission concerned:</h2>
-                {{ comment.submission.header_as_table }}
 
-                <h3 class="mt-3">Abstract:</h3>
-                <p>{{ comment.submission.abstract }}</p>
+                {% include 'submissions/_submission_summary.html' with submission=comment.submission %}
+
             {% endif %}
 
             {% if comment.thesislink %}
                 <h2>The Thesis concerned:</h2>
-                {% include "theses/_header_as_table.html" with thesislink=comment.thesislink %}
-
-                <h3 class="mt-3">Abstract:</h3>
-                <p>{{ comment.thesislink.abstract }}</p>
+                {% include "theses/_thesislink_information.html" with thesislink=comment.thesislink %}
             {% endif %}
         </div>
     </div>
+
     <div class="row">
         <div class="col-12">
             <h2>The Comment you wish to reply to:</h2>
@@ -68,7 +43,7 @@
     <div class="row">
         <div class="col-12">
             <div class="comment">
-                {{ comment.print_identifier }}
+                {% include 'comments/_comment_identifier.html' with comment=comment %}
 
                 {% include 'comments/_comment_categories.html' with comment=comment class='mr-2' %}
 
@@ -83,33 +58,21 @@
 
     <div class="row">
         <div class="col-12">
-            <div class="panel">
-                <h2>Your Reply to this Comment:</h2>
-                {% if is_author %}
-                    <h3>(you are identified as an author, your reply will appear as an author reply)</h3>
-                {% else %}
-                    <h3>(you are not identified as an author of this publication; if you are, you can claim authorship on your Personal Page)</h3>
-                {% endif %}
+            <div class="card card-grey">
+                <div class="card-block">
+                    <h2>Your Reply to this Comment:</h2>
+                    {% if is_author %}
+                        <h3>(you are identified as an author, your reply will appear as an author reply)</h3>
+                    {% else %}
+                        <h3>(you are not identified as an author of this publication; if you are, you can claim authorship on your Personal Page)</h3>
+                    {% endif %}
+                </div>
             </div>
         </div>
     </div>
 
-    <div class="row">
-        <div class="col-12">
-            <form enctype="multipart/form-data" action="{% url 'comments:reply_to_comment' comment.id %}" method="post">
-                {% csrf_token %}
-                {% load crispy_forms_tags %}
-                {% crispy form %}
-            </form>
-        </div>
-    </div>
-
-    <div class="row">
-        <div class="col-12">
-            <h3>Preview of your comment:</h3>
-            <p id="preview-comment_text"></p>
-        </div>
-    </div>
+    {% url 'comments:reply_to_comment' comment.id as add_comment_url %}
+    {% include 'comments/_add_comment_form.html' with url=add_comment_url form=form %}
 
 {% endif %}
 
diff --git a/comments/templates/comments/reply_to_report.html b/comments/templates/comments/reply_to_report.html
index af123a74c633a67803b2dca89b2419e1846a709c..348d82701826f73a80777e21357ef24fbc042db9 100644
--- a/comments/templates/comments/reply_to_report.html
+++ b/comments/templates/comments/reply_to_report.html
@@ -2,70 +2,37 @@
 
 {% block pagetitle %}: reply to report{% endblock pagetitle %}
 
-{% block bodysup %}
-
-<script>
-  $(document).ready(function(){
-
-    var comment_text_input = $("#id_comment_text");
-
-    function set_comment_text(value) {
-      $("#preview-comment_text").text(value)
-    }
-    set_comment_text(comment_text_input.val())
-
-    comment_text_input.keyup(function(){
-      var new_text = $(this).val()
-      set_comment_text(new_text)
-      MathJax.Hub.Queue(["Typeset",MathJax.Hub]);
-    })
-
-  });
-</script>
-
+{% block content %}
 
 {% if user.is_authenticated %}
-<section>
-  <hr class="hr12">
-  <h1>SciPost Reply to Report Page</h1>
-
-  {% if not is_author %}
-    <h3>(you are not identified as an author of this Submission; if you are, you can claim authorship on your Personal Page)</h3>
-  {% else %}
-
-  <hr class="hr12">
-  <h1>The Submission concerned:</h1>
-  {{ report.submission.header_as_table }}
-
-  <h3>Abstract:</h3>
-  <p>{{ report.submission.abstract }}</p>
-  <br>
-
-  <hr class="hr6">
-  <h2>The Report you wish to reply to:</h2>
+    <div class="row">
+        <div class="col-12">
+            <h1 class="highlight">SciPost Reply to Report Page</h1>
+        </div>
+    </div>
 
-  {{ report.print_identifier }}
-  {{ report.print_contents }}
+    <div class="row">
+        <div class="col-12">
+            {% if not is_author %}
+                <h2>(you are not identified as an author of this Submission; if you are, you can claim authorship on your Personal Page)</h2>
+            {% else %}
+                <h2>The Submission concerned:</h2>
 
+                {% include 'submissions/_submission_summary.html' with submission=report.submission %}
 
-  <hr class="hr6">
-  <h1>Your Reply to this Report:</h1>
+            </div>
+        </div>
 
-  <form enctype="multipart/form-data" action="{% url 'comments:reply_to_report' report_id=report.id %}" method="post">
-    {% csrf_token %}
-    {% load crispy_forms_tags %}
-    {% crispy form %}
-  </form>
+        {% include 'submissions/_single_public_report_without_comments.html' with report=report %}
 
-  <div class="row">
-    <div class="col-10">
-      <h3>Preview of your comment:</h3>
-      <p id="preview-comment_text"></p>
+        <div class="row">
+            <div class="col-12">
+                <h2 class="highlight">Your Reply to this Report:</h2>
+                {% url 'comments:reply_to_report' report_id=report.id as add_comment_url %}
+                {% include 'comments/_add_comment_form.html' with url=add_comment_url form=form %}
+            {% endif %}
+        </div>
     </div>
-  </div>
-
-  {% endif %}
-</section>
 {% endif %}
 
-{% endblock bodysup %}
+{% endblock content %}
diff --git a/comments/templates/comments/vet_submitted_comments.html b/comments/templates/comments/vet_submitted_comments.html
index 9300f1ff4820dee5f1385ee77386d714a13ee40d..e3a71b702cb6ab815ad2af54cf9abf1147506ad5 100644
--- a/comments/templates/comments/vet_submitted_comments.html
+++ b/comments/templates/comments/vet_submitted_comments.html
@@ -11,74 +11,79 @@
 {% if not comments_to_vet %}
 <div class="row">
     <div class="col-12">
-        <div class="panel">
-            <h1>There are no comments for you to vet.</h1>
-        </div>
+        <h1 class="highlight">There are no comments for you to vet.</h1>
     </div>
 </div>
 {% else %}
 <div class="row">
     <div class="col-12">
-        <div class="panel">
-            <h1>SciPost Comments to vet:</h1>
-        </div>
+        <h1 class="highlight">SciPost Comments to vet:</h1>
     </div>
 </div>
 
 <div class="row">
-    {% for comment_to_vet in comments_to_vet %}
-        <div class="col-12">
-          {% if comment_to_vet.commentary %}
-            <h2>From Commentary (<a href="{% url 'commentaries:commentary' arxiv_or_DOI_string=comment_to_vet.commentary.arxiv_or_DOI_string %}">link</a>)</h2>
-            {{ comment_to_vet.commentary.header_as_table }}
-            <br />
-          {% endif %}
+    <div class="col-12">
+        {% for comment_to_vet in comments_to_vet %}
+            <div class="card card-vetting">
+                <div class="card-header">
+
+                    {% if comment_to_vet.commentary %}
+                        <h2>From Commentary (<a href="{% url 'commentaries:commentary' arxiv_or_DOI_string=comment_to_vet.commentary.arxiv_or_DOI_string %}">link</a>)</h2>
+                        {% include 'commentaries/_commentary_summary.html' with commentary=comment_to_vet.commentary %}
+                    {% endif %}
+
+                    {% if comment_to_vet.submission %}
+                        <h2>From Submission (<a href="{% url 'submissions:submission' arxiv_identifier_w_vn_nr=comment_to_vet.submission.arxiv_identifier_w_vn_nr %}">link</a>)</h2>
+                        {% include 'submissions/_submission_summary_short.html' with submission=comment_to_vet.submission %}
+                    {% endif %}
+
+                    {% if comment_to_vet.thesislink %}
+                        <h2>From Thesis Link (<a href="{% url 'theses:thesis' comment_to_vet.thesislink.id %}">link</a>)</h2>
+                        {% include 'theses/_thesislink_information.html' with thesislink=comment_to_vet.thesislink %}
+                    {% endif %}
+                </div>
+                <div class="card-block">
 
-          {% if comment_to_vet.submission %}
-            <h2>From Submission (<a href="{% url 'submissions:submission' arxiv_identifier_w_vn_nr=comment_to_vet.submission.arxiv_identifier_w_vn_nr %}">link</a>)</h2>
-            {{ comment_to_vet.submission.header_as_table }}
-            <br />
-          {% endif %}
+                    <h2 class="card-title">The Comment to be vetted:</h2>
 
-          <hr class="small">
-          <h2>The Comment to be vetted:</h2>
+                    <div class="row">
+                        <div class="col-md-6">
+                            {% include 'comments/_comment_identifier_vetting.html' with comment=comment_to_vet %}
+                            <hr class="small">
 
-          <div class="row">
-            <div class="col-md-6">
-                {{ comment_to_vet.print_identifier_for_vetting }}
-                <h3>Comment text:</h3>
-                <p>{{ comment_to_vet.comment_text }}</p>
+                            <h3>Comment text:</h3>
+                            <p>{{ comment_to_vet.comment_text }}</p>
 
-                {% if comment_to_vet.file_attachment %}
-                    <h3>Attachment:</h3>
-                    <p>
-                        <a target="_blank" href="{{ comment_to_vet.file_attachment.url }}">
-                            {% if comment_to_vet.file_attachment|is_image %}
-                                <img class="attachment attachment-comment" src="{{ comment_to_vet.file_attachment.url }}">
-                            {% else %}
-                                {{ comment_to_vet.file_attachment|filename }}<br><small>{{ comment_to_vet.file_attachment.size|filesizeformat }}</small>
+                            {% if comment_to_vet.file_attachment %}
+                                <h3>Attachment:</h3>
+                                <p>
+                                    <a target="_blank" href="{{ comment_to_vet.file_attachment.url }}">
+                                        {% if comment_to_vet.file_attachment|is_image %}
+                                            <img class="attachment attachment-comment" src="{{ comment_to_vet.file_attachment.url }}">
+                                        {% else %}
+                                            {{ comment_to_vet.file_attachment|filename }}<br><small>{{ comment_to_vet.file_attachment.size|filesizeformat }}</small>
+                                        {% endif %}
+                                    </a>
+                                </p>
                             {% endif %}
-                        </a>
-                    </p>
-                {% endif %}
 
-                {% if comment_to_vet.remarks_for_editors %}
-                    <h3>Remarks for Editors only:</h3>
-                    <p>{{ comment_to_vet.remarks_for_editors }}</p>
-                {% endif %}
-            </div>
-            <div class="col-md-6">
-              <form action="{% url 'comments:vet_submitted_comment_ack' comment_id=comment_to_vet.id %}" method="post">
-                {% csrf_token %}
-                {{ form|bootstrap }}
-                <input class="btn btn-primary" type="submit" value="Submit" />
-              </form>
+                            {% if comment_to_vet.remarks_for_editors %}
+                                <h3>Remarks for Editors only:</h3>
+                                <p>{{ comment_to_vet.remarks_for_editors }}</p>
+                            {% endif %}
+                        </div>
+                        <div class="col-md-6">
+                            <form action="{% url 'comments:vet_submitted_comment_ack' comment_id=comment_to_vet.id %}" method="post">
+                                {% csrf_token %}
+                                {{ form|bootstrap }}
+                                <input class="btn btn-primary" type="submit" value="Submit" />
+                            </form>
+                        </div>
+                    </div>
+                </div>
             </div>
-        </div>
-            <hr>
+        {% endfor %}
     </div>
-
-    {% endfor %}
 </div>
 {% endif %}
 
diff --git a/comments/views.py b/comments/views.py
index ce859b86553018a6788aee4a82472c71cfa388f8..af5577b3316f4f7d4508ebf52bf1a2e244d9a760 100644
--- a/comments/views.py
+++ b/comments/views.py
@@ -10,14 +10,15 @@ from django.http import HttpResponseRedirect, Http404
 import strings
 
 from .models import Comment
-from .forms import CommentForm, VetCommentForm, comment_refusal_dict
+from .forms import CommentForm, VetCommentForm
 
-from scipost.models import Contributor, title_dict
+from scipost.models import Contributor
 from theses.models import ThesisLink
 from submissions.utils import SubmissionUtils
 from submissions.models import Submission, Report
 from commentaries.models import Commentary
 
+
 @permission_required('scipost.can_submit_comments', raise_exception=True)
 def new_comment(request, **kwargs):
     if request.method == "POST":
@@ -46,7 +47,7 @@ def new_comment(request, **kwargs):
                 if not thesislink.open_for_commenting:
                     raise PermissionDenied
                 new_comment.thesislink = thesislink
-                redirect_link = reverse('theses:thesis', kwargs={"thesislink_id":thesislink.id})
+                redirect_link = reverse('theses:thesis', kwargs={"thesislink_id": thesislink.id})
             elif type_of_object == "submission":
                 submission = Submission.objects.get(id=object_id)
                 if not submission.open_for_commenting:
@@ -62,7 +63,8 @@ def new_comment(request, **kwargs):
                     raise PermissionDenied
                 new_comment.commentary = commentary
                 redirect_link = reverse(
-                    'commentaries:commentary', kwargs={'arxiv_or_DOI_string': commentary.arxiv_or_DOI_string}
+                    'commentaries:commentary',
+                    kwargs={'arxiv_or_DOI_string': commentary.arxiv_or_DOI_string}
                 )
 
             new_comment.save()
@@ -75,6 +77,7 @@ def new_comment(request, **kwargs):
         # This view is only accessible by POST request
         raise Http404
 
+
 @permission_required('scipost.can_vet_comments', raise_exception=True)
 def vet_submitted_comments(request):
     contributor = Contributor.objects.get(user=request.user)
@@ -95,7 +98,7 @@ def vet_submitted_comment_ack(request, comment_id):
                 comment.status = 1
                 comment.vetted_by = request.user.contributor
                 comment.save()
-                email_text = ('Dear ' + title_dict[comment.author.title] + ' '
+                email_text = ('Dear ' + comment.author.get_title_display() + ' '
                               + comment.author.user.last_name +
                               ', \n\nThe Comment you have submitted, '
                               'concerning publication with title ')
@@ -138,7 +141,7 @@ def vet_submitted_comment_ack(request, comment_id):
                 if comment.status == 0:
                     comment.status == -1
                 comment.save()
-                email_text = ('Dear ' + title_dict[comment.author.title] + ' '
+                email_text = ('Dear ' + comment.author.get_title_display() + ' '
                               + comment.author.user.last_name
                               + ', \n\nThe Comment you have submitted, '
                               'concerning publication with title ')
@@ -151,7 +154,7 @@ def vet_submitted_comment_ack(request, comment_id):
                 elif comment.thesislink is not None:
                     email_text += comment.thesislink.title + ' by ' + comment.thesislink.author
                 email_text += (', has been rejected for the following reason: '
-                               + comment_refusal_dict[comment.status] + '.' +
+                               + comment.get_status_display() + '.' +
                                '\n\nWe copy it below for your convenience.' +
                                '\n\nThank you for your contribution, \n\nThe SciPost Team.')
                 if form.cleaned_data['email_response_field']:
diff --git a/journals/models.py b/journals/models.py
index 2a07b982f244e93be2dcc1641ce93fe93b03948e..1c18db989536bed944afa86e3f67fa380fdf74d9 100644
--- a/journals/models.py
+++ b/journals/models.py
@@ -1,9 +1,13 @@
 from django.contrib.postgres.fields import JSONField
 from django.db import models
-from django.http import Http404
 from django.template import Template, Context
 from django.utils import timezone
 
+from .constants import SCIPOST_JOURNALS, SCIPOST_JOURNALS_DOMAINS,\
+                       STATUS_DRAFT, STATUS_PUBLISHED, ISSUE_STATUSES
+from .helpers import paper_nr_string, journal_name_abbrev_citation
+from .managers import IssueManager, PublicationManager
+
 from scipost.constants import SCIPOST_DISCIPLINES, SCIPOST_SUBJECT_AREAS
 from scipost.models import ChoiceArrayField, Contributor
 
@@ -16,103 +20,6 @@ class UnregisteredAuthor(models.Model):
         return self.last_name + ', ' + self.first_name
 
 
-SCIPOST_JOURNALS = (
-    ('SciPost Physics Select', 'SciPost Physics Select'),
-    ('SciPost Physics', 'SciPost Physics'),
-    ('SciPost Physics Lecture Notes', 'SciPost Physics Lecture Notes'),
-    )
-journals_dict = dict(SCIPOST_JOURNALS)
-
-
-class JournalNameError(Exception):
-    def __init__(self, name):
-        self.name = name
-
-    def __str__(self):
-        return self.name
-
-
-def journal_name_abbrev_citation(journal_name):
-    if journal_name == 'SciPost Physics':
-        return 'SciPost Phys.'
-    elif journal_name == 'SciPost Physics Select':
-        return 'SciPost Phys. Sel.'
-    elif journal_name == 'SciPost Physics Lecture Notes':
-        return 'SciPost Phys. Lect. Notes'
-    else:
-        raise JournalNameError(journal_name)
-
-
-def journal_name_abbrev_doi(journal_name):
-    if journal_name == 'SciPost Physics':
-        return 'SciPostPhys'
-    elif journal_name == 'SciPost Physics Select':
-        return 'SciPostPhysSel'
-    elif journal_name == 'SciPost Physics Lecture Notes':
-        return 'SciPostPhysLectNotes'
-    else:
-        raise JournalNameError(journal_name)
-
-
-class PaperNumberError(Exception):
-    def __init__(self, nr):
-        self.nr = nr
-
-    def __str__(self):
-        return self.nr
-
-
-def paper_nr_string(nr):
-    if nr < 10:
-        return '00' + str(nr)
-    elif nr < 100:
-        return '0' + str(nr)
-    elif nr < 1000:
-        return str(nr)
-    else:
-        raise PaperNumberError(nr)
-
-
-class PaperNumberingError(Exception):
-    def __init__(self, nr):
-        self.nr = nr
-
-    def __str__(self):
-        return self.nr
-
-
-SCIPOST_JOURNALS_SUBMIT = (  # Same as SCIPOST_JOURNALS, but SP Select deactivated
-    ('SciPost Physics', 'SciPost Physics'),
-    ('SciPost Physics Lecture Notes', 'SciPost Physics Lecture Notes'),
-    )
-journals_submit_dict = dict(SCIPOST_JOURNALS_SUBMIT)
-
-SCIPOST_JOURNALS_DOMAINS = (
-    ('E', 'Experimental'),
-    ('T', 'Theoretical'),
-    ('C', 'Computational'),
-    ('ET', 'Exp. & Theor.'),
-    ('EC', 'Exp. & Comp.'),
-    ('TC', 'Theor. & Comp.'),
-    ('ETC', 'Exp., Theor. & Comp.'),
-)
-journals_domains_dict = dict(SCIPOST_JOURNALS_DOMAINS)
-
-SCIPOST_JOURNALS_SPECIALIZATIONS = (
-    ('A', 'Atomic, Molecular and Optical Physics'),
-    ('B', 'Biophysics'),
-    ('C', 'Condensed Matter Physics'),
-    ('F', 'Fluid Dynamics'),
-    ('G', 'Gravitation, Cosmology and Astroparticle Physics'),
-    ('H', 'High-Energy Physics'),
-    ('M', 'Mathematical Physics'),
-    ('N', 'Nuclear Physics'),
-    ('Q', 'Quantum Statistical Mechanics'),
-    ('S', 'Statistical and Soft Matter Physics'),
-    )
-journals_spec_dict = dict(SCIPOST_JOURNALS_SPECIALIZATIONS)
-
-
 class Journal(models.Model):
     name = models.CharField(max_length=100, choices=SCIPOST_JOURNALS,
                             unique=True)
@@ -137,43 +44,6 @@ class Volume(models.Model):
         return str(self.in_journal) + ' Vol. ' + str(self.number)
 
 
-STATUS_DRAFT = 'draft'
-STATUS_PUBLISHED = 'published'
-ISSUE_STATUSES = (
-    (STATUS_DRAFT, 'Draft'),
-    (STATUS_PUBLISHED, 'Published'),
-)
-
-
-class IssueManager(models.Manager):
-    def get_published(self, *args, **kwargs):
-        try:
-            return self.published(*args, **kwargs)[0]
-        except IndexError:
-            raise Http404
-
-    def published(self, journal=None, **kwargs):
-        issues = self.filter(status=STATUS_PUBLISHED, **kwargs)
-        if journal:
-            issues.filter(in_volume__in_journal__name=journal)
-        return issues
-
-    def in_draft(self, journal=None, **kwargs):
-        issues = self.filter(status=STATUS_DRAFT, **kwargs)
-        if journal:
-            issues.filter(in_volume__in_journal__name=journal)
-        return issues
-
-    def get_current_issue(self, *args, **kwargs):
-        return self.published(start_date__lte=timezone.now(),
-                              until_date__gte=timezone.now(),
-                              **kwargs).order_by('-until_date').first()
-
-    def get_last_filled_issue(self, *args, **kwargs):
-        return self.published(publication__isnull=False,
-                              **kwargs).order_by('-until_date').first()
-
-
 class Issue(models.Model):
     in_volume = models.ForeignKey(Volume, on_delete=models.CASCADE)
     number = models.PositiveSmallIntegerField()
@@ -198,7 +68,7 @@ class Issue(models.Model):
 
     def period_as_string(self):
         if self.start_date.month == self.until_date.month:
-            return ' (' + self.until_date.strftime('%B') + ' ' + self.until_date.strftime('%Y') + ')'
+            return ' (%s %s)' % (self.until_date.strftime('%B'), self.until_date.strftime('%Y'))
         else:
             return (' (' + self.start_date.strftime('%B') + '-' + self.until_date.strftime('%B') +
                     ' ' + self.until_date.strftime('%Y') + ')')
@@ -215,20 +85,6 @@ class Issue(models.Model):
         return template.render(context)
 
 
-class PublicationManager(models.Manager):
-    def get_published(self, *args, **kwargs):
-        try:
-            return self.published(*args, **kwargs)[0]
-        except IndexError:
-            raise Http404
-
-    def published(self, **kwargs):
-        return self.filter(in_issue__status=STATUS_PUBLISHED, **kwargs)
-
-    def in_draft(self, **kwargs):
-        return self.filter(in_issue__status=STATUS_DRAFT, **kwargs)
-
-
 class Publication(models.Model):
     accepted_submission = models.OneToOneField('submissions.Submission', on_delete=models.CASCADE)
     in_issue = models.ForeignKey(Issue, on_delete=models.CASCADE)
@@ -296,7 +152,7 @@ class Publication(models.Model):
     def citation_for_web_linked(self):
         citation = ('<a href="{% url \'scipost:publication_detail\' doi_string=doi_string %}">'
                     '{{ abbrev }} <strong>{{ volume_nr }}</strong>'
-                    ', {{ paper_nr }} ({{ year }})')
+                    ', {{ paper_nr }} ({{ year }})</a>')
         template = Template(citation)
         context = Context(
             {'doi_string': self.doi_string,
@@ -307,49 +163,6 @@ class Publication(models.Model):
              'year': self.publication_date.strftime('%Y'), })
         return template.render(context)
 
-    def details(self):
-        """
-        This method is called from the publication_detail template.
-        It provides all the details for a publication.
-        """
-        pub_details = (
-            '<div class="row"><div class="col-12">'
-            '<h3 class="publicationTitle">'
-            '<a href="{% url \'scipost:publication_detail\' doi_string=doi_string %}">{{ title }}</a>'
-            '</h3>'
-            '<p class="publicationAuthors">{{ author_list }}</p>'
-            '<p class="publicationReference">{{ citation }} &nbsp;&nbsp;'
-            '|&nbsp;published {{ pub_date}}</p>'
-            '<ul class="publicationClickables">'
-            '<li>doi:  {{ doi_string }}</li>'
-            '<li class="publicationPDF">'
-            '<a href="{% url \'scipost:publication_pdf\' doi_string=doi_string %}" target="_blank">pdf</a>'
-            '</li>'
-            '<li><a href="#openModal">BiBTeX</a></li>'
-            '<li><a href="{% url \'submissions:submission\' arxiv_identifier_w_vn_nr='
-            'arxiv_identifier_w_vn_nr %}">Submissions/Reports</a></li>'
-            '</ul></div></div>'
-            '<div class="row"><div class="col-12"><hr>'
-            '<h3>Abstract:</h3>'
-            '<p class="publicationAbstract">{{ abstract }}</p>'
-            '<div id="openModal" class="modalDialog"><div>'
-            '<a href="#close" title="Close" class="close">X</a>'
-            '<h2>BiBTeX</h2><p>{{ BiBTeX|linebreaks }}</p></div></div>'
-            '</div></div>'
-        )
-        template = Template(pub_details)
-        context = Context({
-            'title': self.title,
-            'author_list': self.author_list,
-            'citation': self.citation_for_web,
-            'pub_date': self.publication_date.strftime('%d %B %Y'),
-            'abstract': self.abstract,
-            'doi_string': self.doi_string,
-            'BiBTeX': self.BiBTeX_entry,
-            'arxiv_identifier_w_vn_nr': self.accepted_submission.arxiv_identifier_w_vn_nr
-        })
-        return template.render(context)
-
     def citations_as_ul(self):
         output = '<ul>'
         context = Context({})
diff --git a/journals/templates/journals/add_author.html b/journals/templates/journals/add_author.html
index 241de8b1a11ef7a0b1aa86685a8abb8c261400cf..fabb202ea90b856b73e0cbc2154f00e138f615c6 100644
--- a/journals/templates/journals/add_author.html
+++ b/journals/templates/journals/add_author.html
@@ -2,90 +2,102 @@
 
 {% block pagetitle %}: add author to publication{% endblock pagetitle %}
 
-{% block headsup %}
-
 {% load scipost_extras %}
-
-{% endblock headsup %}
-
-{% block bodysup %}
-
-
-<section>
-  <div class="flex-greybox">
-    <h1>Add author to publication</h1>
-  </div>
-  {{ publication.details }}
-</section>
-
-
-<section>
-  <div class="flex-greybox">
-    <h1>Add an (unregistered) author</h1>
-  </div>
-
-  <h3>Current list of authors as contributors:</h3>
-  <p>
-    {% for author in publication.authors.all %}
-      <a href="/contributor/{{ author.id }}">{{ author.user.first_name }} {{ author.user.last_name }}</a>&nbsp;
-    {% endfor %}
-  </p>
-  <h3>Current list of additional authors (unregistered):</h3>
-  <p>
-    {% for author in publication.authors_unregistered.all %}
-      {{ author }}
-    {% endfor %}
-  </p>
-
-  <h3>Search for missing author:</h3>
-  <form action="{% url 'journals:add_author' publication_id=publication.id %}" method="post">
-    {% csrf_token %}
-    {{ form.as_p }}
-    <input type="submit" value="Search">
-  </form>
-
-  {% if contributors_found %}
-    <h3>Identified as contributor:</h3>
-    <table>
-      {% for contributor in contributors_found %}
-        <tr>
-          <td>{{ contributor.user.first_name }} {{ contributor.user.last_name }}</td>
-          <td>&nbsp;</td>
-          <td><a href="{% url 'journals:add_author' publication_id=publication.id contributor_id=contributor.id %}">Add this Contributor as author of this Publication</a></td>
-        </tr>
-      {% endfor %}
-    </table>
-  {% elif form.has_changed %}
-    <p>No Contributor with this name could be identified.</p>
-  {% endif %}
-  <br/>
-
-  {% if unregistered_authors_found %}
-    <h3>Identified as existing unregistered author:</h3>
-    <table>
-      {% for unreg_auth in unregistered_authors_found %}
-        <tr>
-          <td>{{ unreg_auth }}</td>
-          <td>&nbsp;</td>
-          <td><a href="{% url 'journals:add_unregistered_author' publication_id=publication.id unregistered_author_id=unreg_auth.id %}">Add this unregistered author as author of this Publication</a></td>
-        </tr>
-      {% endfor %}
-    </table>
-  {% elif form.has_changed %}
-    <p>No UnregisteredAuthor with this name could be found in the database.</p>
-  {% endif %}
-  <br/>
-
-  <h3>You can otherwise create an UnregisteredAuthor object instance and link it to this publication:</h3>
-  <form action="{% url 'journals:add_new_unreg_author' publication_id=publication.id %}" method="post">
-    {% csrf_token %}
-    {{ new_unreg_author_form.as_p }}
-    <input type="submit" value="Add">
-  </form>
-
-  <h3><a href="{% url 'scipost:publication_detail' doi_string=publication.doi_string %}">Return to the publication's page</a></h3>
-
-</section>
-
-
-{% endblock bodysup %}
+{% load bootstrap %}
+
+{% block content %}
+
+<div class="row">
+    <div class="col-12">
+        <h1 class="highlight">Add author to publication</h1>
+    </div>
+</div>
+
+<div class="row">
+    <div class="col-12">
+        {% include 'journals/_publication_details.html' with publication=publication %}
+    </div>
+</div>
+
+
+<div class="row">
+    <div class="col-12">
+        <h2 class="highlight">Add an (unregistered) author</h2>
+    </div>
+</div>
+
+<div class="row">
+    <div class="col-12">
+      <h3>Current list of authors as contributors:</h3>
+      <ul class="list-group list-group-noborder">
+        {% for author in publication.authors.all %}
+            <li class="list-group-item py-1 pl-2">
+                <a href="{% url 'scipost:contributor_info' author.id %}">{{ author.user.first_name }} {{ author.user.last_name }}</a>
+            </li>
+        {% empty %}
+            No unregistered authors known.
+        {% endfor %}
+      </ul>
+      <h3>Current list of additional authors (unregistered):</h3>
+      <ul class="list-group list-group-noborder">
+        {% for author in publication.authors_unregistered.all %}
+            <li class="list-group-item py-1 pl-2">{{ author }}</li>
+        {% empty %}
+            No unregistered authors known.
+        {% endfor %}
+      </ul>
+      <hr class="small">
+
+      <div class="row">
+          <div class="col-md-8">
+              <h3>Search for missing author:</h3>
+              <form action="{% url 'journals:add_author' publication.id %}" method="post">
+                    {% csrf_token %}
+                    {{form|bootstrap}}
+                    <input class="btn btn-secondary" type="submit" value="Search">
+              </form>
+
+              {% if form.has_changed %}
+                <h3 class="mt-4">Identified as contributor:</h3>
+                <ul class="list-group">
+                  {% for contributor in contributors_found %}
+                      <li class="list-group-item p-2">
+                        <div class="d-block w-100 font-weight-bold">{{ contributor.user.first_name }} {{ contributor.user.last_name }}</div>
+                        <a class="d-block" href="{% url 'journals:add_author' publication_id=publication.id contributor_id=contributor.id %}">Add this Contributor as author of this Publication</a>
+                      </li>
+                  {% empty %}
+                      <span class="text-danger">No Contributor with this name could be identified.</span>
+                  {% endfor %}
+                </ul>
+
+                <h3 class="mt-2">Identified as existing unregistered author:</h3>
+                <ul class="list-group">
+                  {% for unreg_auth in unregistered_authors_found %}
+                    <li class="list-group-item">
+                        <div class="d-block w-100 font-weight-bold">{{ unreg_auth }}
+                        <a class="d-block" href="{% url 'journals:add_unregistered_author' publication_id=publication.id unregistered_author_id=unreg_auth.id %}">Add this unregistered author as author of this Publication</a>
+                    </li>
+                    {% empty %}
+                        <span class="text-danger">No UnregisteredAuthor with this name could be found in the database.</span>
+                    {% endfor %}
+                </ul>
+
+                <h3 class="mt-3">You can otherwise create an UnregisteredAuthor object instance and link it to this publication:</h3>
+                <form action="{% url 'journals:add_new_unreg_author' publication_id=publication.id %}" method="post">
+                    {% csrf_token %}
+                    {{ new_unreg_author_form|bootstrap }}
+                    <input class="btn btn-secondary" type="submit" value="Add">
+                </form>
+
+              {% endif %}
+          </div>
+      </div>
+
+      <h3>
+          <a href="{% url 'scipost:publication_detail' doi_string=publication.doi_string %}">Return to the publication's page</a>
+      </h3>
+    </div>
+</div>
+
+
+{% endblock content %}
diff --git a/journals/templates/journals/publication_detail.html b/journals/templates/journals/publication_detail.html
index 12c8c509f7d5d5edc0d11a5232571a47874c8b6a..8b9d46ecd843d98224c350dad63c5efdb6d6cf4c 100644
--- a/journals/templates/journals/publication_detail.html
+++ b/journals/templates/journals/publication_detail.html
@@ -40,11 +40,7 @@
 
 {% include 'journals/scipost_physics_menu.html' %}
 
-<div class="row">
-    <div class="col-12">
-        {{ publication.details }}
-    </div>
-</div>
+{% include 'journals/_publication_details.html' with publication=publication %}
 
 {% if publication.citedby|length >= 1 %}
 <hr>
diff --git a/journals/templatetags/journals_extras.py b/journals/templatetags/journals_extras.py
index c5625821237805fb08970b9be997c43384996c73..382ac91e2b0e6a2528fbd9c54349ca4a9e0ac332 100644
--- a/journals/templatetags/journals_extras.py
+++ b/journals/templatetags/journals_extras.py
@@ -1,6 +1,6 @@
 from django import template
 
-from journals.models import paper_nr_string
+from journals.helpers import paper_nr_string
 
 register = template.Library()
 
diff --git a/journals/utils.py b/journals/utils.py
index 71d5cae1cd91cb6373334c5fd1d76b607bdb32bf..ccb5852064de8ae78db09de828d791749e7b5b0b 100644
--- a/journals/utils.py
+++ b/journals/utils.py
@@ -1,7 +1,5 @@
 from django.core.mail import EmailMessage
 
-from scipost.models import title_dict
-
 
 class JournalUtils(object):
 
@@ -14,7 +12,7 @@ class JournalUtils(object):
     def send_authors_paper_published_email(cls):
         """ Requires loading 'publication' attribute. """
         email_text = ('Dear '
-                      + title_dict[cls.publication.accepted_submission.submitted_by.title]
+                      + cls.publication.accepted_submission.submitted_by.get_title_display()
                       + ' ' +
                       cls.publication.accepted_submission.submitted_by.user.last_name +
                       ', \n\nWe are happy to inform you that your Submission to SciPost,\n\n' +
diff --git a/journals/views.py b/journals/views.py
index 77c1df2b0b48ffc596807004214966204c6c898a..5a4bf03f75a2007a68dbaf387d081de9907db0d4 100644
--- a/journals/views.py
+++ b/journals/views.py
@@ -8,14 +8,13 @@ import xml.etree.ElementTree as ET
 from django.conf import settings
 from django.utils import timezone
 from django.shortcuts import get_object_or_404, render, redirect
-from django.core.files import File
 from django.core.urlresolvers import reverse
 from django.db import transaction
 from django.http import HttpResponse
 
-from .models import Issue, Publication, PaperNumberingError,\
-                    journal_name_abbrev_doi, paper_nr_string, journal_name_abbrev_citation,\
-                    UnregisteredAuthor
+from .exceptions import PaperNumberingError
+from .helpers import journal_name_abbrev_citation, journal_name_abbrev_doi, paper_nr_string
+from .models import Issue, Publication, UnregisteredAuthor
 from .forms import FundingInfoForm, InitiatePublicationForm, ValidatePublicationForm,\
                    UnregisteredAuthorForm, CreateMetadataXMLForm, CitationListBibitemsForm
 from .utils import JournalUtils
diff --git a/news/models.py b/news/models.py
index 8f73d1be7c615125ae53c98b0f74a6f9b4d78748..a8992a3e7c5464b359a81208f4240c589e901320 100644
--- a/news/models.py
+++ b/news/models.py
@@ -1,5 +1,4 @@
 from django.db import models
-from django.template import Template, Context
 
 
 class NewsItem(models.Model):
@@ -14,39 +13,3 @@ class NewsItem(models.Model):
 
     def __str__(self):
         return self.date.strftime('%Y-%m-%d') + ', ' + self.headline
-
-    def descriptor_full(self):
-        """ For News page. """
-        descriptor = ('<div class="flex-greybox640">'
-                      '<h3 class="NewsHeadline">{{ headline }}</h3>'
-                      '<p>{{ date }}</p>'
-                      '<p>{{ blurb }}</p>'
-                      )
-        context = Context({'headline': self.headline,
-                           'date': self.date.strftime('%Y-%m-%d'),
-                           'blurb': self.blurb, })
-        if self.followup_link:
-            descriptor += '<p><a href="{{ followup_link }}">{{ followup_link_text }}</a></p>'
-            context['followup_link'] = self.followup_link
-            context['followup_link_text'] = self.followup_link_text
-        descriptor += '</div>'
-        template = Template(descriptor)
-        return template.render(context)
-
-    def descriptor_small(self):
-        """ For index page. """
-        descriptor = ('<h3 class="NewsHeadline">{{ headline }}</h3>'
-                      '<div class="p-2">'
-                      '<p>{{ date }}</p>'
-                      '<p>{{ blurb }}</p>'
-                      )
-        context = Context({'headline': self.headline,
-                           'date': self.date.strftime('%Y-%m-%d'),
-                           'blurb': self.blurb, })
-        if self.followup_link:
-            descriptor += '<a href="{{ followup_link }}">{{ followup_link_text }}</a>'
-            context['followup_link'] = self.followup_link
-            context['followup_link_text'] = self.followup_link_text
-        descriptor += '</div>'
-        template = Template(descriptor)
-        return template.render(context)
diff --git a/scipost/constants.py b/scipost/constants.py
index 5175737aab2fe2a7d5c3378996ac8b1f49aa3b57..61396046822991f0f22d8da26a65babe42fe6955 100644
--- a/scipost/constants.py
+++ b/scipost/constants.py
@@ -122,3 +122,29 @@ subject_areas_raw_dict = dict(SCIPOST_SUBJECT_AREAS)
 subject_areas_dict = {}
 for k in subject_areas_raw_dict.keys():
     subject_areas_dict.update(dict(subject_areas_raw_dict[k]))
+
+
+CONTRIBUTOR_STATUS = (
+    # status determine the type of Contributor:
+    # 0: newly registered (unverified; not allowed to submit, comment or vote)
+    # 1: contributor has been vetted through
+    #
+    # Negative status denotes rejected requests or:
+    # -1: not a professional scientist (>= PhD student in known university)
+    # -2: other account already exists for this person
+    # -3: barred from SciPost (abusive behaviour)
+    # -4: disabled account (deceased)
+    (0, 'newly registered'),
+    (1, 'normal user'),
+    (-1, 'not a professional scientist'),
+    (-2, 'other account already exists'),
+    (-3, 'barred from SciPost'),
+    (-4, 'account disabled'),
+    )
+
+TITLE_CHOICES = (
+    ('PR', 'Prof.'),
+    ('DR', 'Dr'),
+    ('MR', 'Mr'),
+    ('MRS', 'Mrs'),
+)
diff --git a/scipost/factories.py b/scipost/factories.py
index 62eb21c81e74b9fa81c6d0c7892f06ef578bbc8c..7381d6173165155478253327f9aacac1665129c7 100644
--- a/scipost/factories.py
+++ b/scipost/factories.py
@@ -6,7 +6,8 @@ from django.contrib.auth.models import Group
 
 from django_countries.data import COUNTRIES
 
-from .models import Contributor, EditorialCollege, EditorialCollegeFellowship, TITLE_CHOICES
+from .models import Contributor, EditorialCollege, EditorialCollegeFellowship
+from .constants import TITLE_CHOICES
 
 
 class ContributorFactory(factory.django.DjangoModelFactory):
diff --git a/scipost/feeds.py b/scipost/feeds.py
index 29ea0a31d6fced705e0c8fd6ac77381e7ed818f8..6edb9e1b7d4b1aaf221382ba83cff76a5064d27b 100644
--- a/scipost/feeds.py
+++ b/scipost/feeds.py
@@ -5,11 +5,12 @@ from django.utils.feedgenerator import Atom1Feed
 from django.core.urlresolvers import reverse
 from django.db.models import Q
 
-from scipost.models import subject_areas_dict
 from comments.models import Comment
 from journals.models import Publication
 from news.models import NewsItem
-from submissions.models import Submission, SUBMISSION_STATUS_PUBLICLY_INVISIBLE
+from scipost.models import subject_areas_dict
+from submissions.constants import SUBMISSION_STATUS_PUBLICLY_INVISIBLE
+from submissions.models import Submission
 
 
 class LatestCommentsFeedRSS(Feed):
diff --git a/scipost/forms.py b/scipost/forms.py
index 3a85ac22032944518aae6cb8a7230eee68005bb9..f18c9e1aa07706765d763659e14a7880313ecbfb 100644
--- a/scipost/forms.py
+++ b/scipost/forms.py
@@ -10,15 +10,14 @@ from captcha.fields import ReCaptchaField
 from crispy_forms.helper import FormHelper
 from crispy_forms.layout import Layout, Div, Field, HTML, Submit
 
-from .constants import SCIPOST_DISCIPLINES, SCIPOST_SUBJECT_AREAS
-from .models import TITLE_CHOICES, SCIPOST_FROM_ADDRESSES,\
+from .constants import SCIPOST_DISCIPLINES, TITLE_CHOICES
+from .models import SCIPOST_FROM_ADDRESSES,\
                     Contributor, DraftInvitation, RegistrationInvitation,\
                     SupportingPartner, SPBMembershipAgreement,\
                     UnavailabilityPeriod, PrecookedEmail
-from virtualmeetings.models import Feedback, Nomination, Motion
 
 from journals.models import Publication
-from submissions.models import SUBMISSION_STATUS_PUBLICLY_UNLISTED
+from submissions.constants import SUBMISSION_STATUS_PUBLICLY_UNLISTED
 from submissions.models import Submission
 
 
diff --git a/scipost/models.py b/scipost/models.py
index 2bb4c15cfb9b6ca2c262301852f55f68dbdad5c1..9d7ed767193d907df22bbc674ff0d5d27f4854f6 100644
--- a/scipost/models.py
+++ b/scipost/models.py
@@ -7,13 +7,12 @@ from django.db import models
 from django.db.models import Q
 from django.template import Template, Context
 from django.utils import timezone
-from django.utils.encoding import force_text
 from django.utils.safestring import mark_safe
 
 from django_countries.fields import CountryField
 
 from .constants import SCIPOST_DISCIPLINES, SCIPOST_SUBJECT_AREAS,\
-    disciplines_dict, subject_areas_dict
+                       disciplines_dict, subject_areas_dict, CONTRIBUTOR_STATUS, TITLE_CHOICES
 from .db.fields import AutoDateTimeField
 
 
@@ -34,33 +33,6 @@ class ChoiceArrayField(ArrayField):
         return super(ArrayField, self).formfield(**defaults)
 
 
-CONTRIBUTOR_STATUS = (
-    # status determine the type of Contributor:
-    # 0: newly registered (unverified; not allowed to submit, comment or vote)
-    # 1: contributor has been vetted through
-    #
-    # Negative status denotes rejected requests or:
-    # -1: not a professional scientist (>= PhD student in known university)
-    # -2: other account already exists for this person
-    # -3: barred from SciPost (abusive behaviour)
-    # -4: disabled account (deceased)
-    (0, 'newly registered'),
-    (1, 'normal user'),
-    (-1, 'not a professional scientist'),
-    (-2, 'other account already exists'),
-    (-3, 'barred from SciPost'),
-    (-4, 'account disabled'),
-    )
-
-TITLE_CHOICES = (
-    ('PR', 'Prof.'),
-    ('DR', 'Dr'),
-    ('MR', 'Mr'),
-    ('MRS', 'Mrs'),
-    )
-title_dict = dict(TITLE_CHOICES)
-
-
 class TimeStampedModel(models.Model):
     """
     All objects should inherit from this abstract model.
@@ -117,7 +89,8 @@ class Contributor(models.Model):
         return '%s, %s' % (self.user.last_name, self.user.first_name)
 
     def get_title(self):
-        return title_dict[self.title]
+        # Please use get_title_display(). To be removed in future
+        return self.get_title_display()
 
     def is_currently_available(self):
         unav_periods = UnavailabilityPeriod.objects.filter(contributor=self)
@@ -145,7 +118,7 @@ class Contributor(models.Model):
             </table>
         ''')
         context = Context({
-            'title': title_dict[self.title],
+            'title': self.get_title_display(),
             'first_name': self.user.first_name,
             'last_name': self.user.last_name,
             'email': self.user.email,
@@ -174,7 +147,7 @@ class Contributor(models.Model):
             </table>
         ''')
         context = Context({
-                'title': title_dict[self.title],
+                'title': self.get_title_display(),
                 'first_name': self.user.first_name,
                 'last_name': self.user.last_name,
                 'email': self.user.email,
diff --git a/scipost/static/scipost/assets/config/preconfig.scss b/scipost/static/scipost/assets/config/preconfig.scss
index 0ca0434f571a37fc3d31d816d25957730b6091f2..e924f4f07db1eaf8e875412d873bf70b3136edb1 100644
--- a/scipost/static/scipost/assets/config/preconfig.scss
+++ b/scipost/static/scipost/assets/config/preconfig.scss
@@ -24,9 +24,11 @@ $card-border-color: rgba(238, 238, 238, 0.5);
 
 // Colors
 //
+$scipost-light: #C3D7EE;
 $scipost-lightblue: #6884C2;
 $scipost-darkblue: #002b49;
 $scipost-orange: #FFA300;
+$green: #6ebb6e;
 
 $blue: $scipost-lightblue !default;
 $body-color: $scipost-darkblue !default;
@@ -63,6 +65,10 @@ $font-size-h4: 0.8rem;
 $font-size-h5: 0.8rem;
 $font-size-h6: 0.8rem;
 
+// Tables
+//
+$table-cell-padding: 0.25rem 0.5rem;
+
 
 // Navbar
 //
diff --git a/scipost/static/scipost/assets/css/_cards.scss b/scipost/static/scipost/assets/css/_cards.scss
index 129e0b047f5212a83c6af050d2aefcbde7813321..78bce205b80996f4d7607e1bbfa19301cccc45e7 100644
--- a/scipost/static/scipost/assets/css/_cards.scss
+++ b/scipost/static/scipost/assets/css/_cards.scss
@@ -28,3 +28,10 @@
 .card-subtitle {
     line-height: 1.5;
 }
+
+.card.card-news {
+    .card-header {
+        background-color: $scipost-darkblue;
+        color: $scipost-light;
+    }
+}
diff --git a/scipost/static/scipost/assets/css/_general.scss b/scipost/static/scipost/assets/css/_general.scss
index 3a7a8095d4f6d2e860e636ba811e23613fca7852..806ff1329bfdd30180da7cd9e74dc87fdc069328 100644
--- a/scipost/static/scipost/assets/css/_general.scss
+++ b/scipost/static/scipost/assets/css/_general.scss
@@ -9,3 +9,7 @@
         color: #FFA300;
     }
 }
+
+.bg-transparent {
+    background-color: transparent;
+}
diff --git a/scipost/static/scipost/assets/css/style.scss b/scipost/static/scipost/assets/css/style.scss
index 17cfeb51944c746c682c388bf64b816cbbab679b..b40f5ca89e040ec6aaa5d37cde766c8709ee29e8 100644
--- a/scipost/static/scipost/assets/css/style.scss
+++ b/scipost/static/scipost/assets/css/style.scss
@@ -23,10 +23,12 @@
 @import "form";
 @import "grid";
 @import "labels";
+@import "list_group";
 @import "messages";
 // @import "modal";
 @import "navbar";
 @import "page_header";
+@import "tables";
 @import "tooltip";
 @import "type";
 
diff --git a/scipost/templates/scipost/index.html b/scipost/templates/scipost/index.html
index 3e750cc98753c4307cf32430c0c3accbdb8ff23e..35fc674ca9abdbd0345ce3764e8a09819ad6996f 100644
--- a/scipost/templates/scipost/index.html
+++ b/scipost/templates/scipost/index.html
@@ -13,15 +13,11 @@
                     <h1 class="card-title mb-0"><a href="{% url 'news:news' %}">News</a><a style="float: right;" href="{% url 'scipost:feeds' %}"><img src="{% static 'scipost/images/feed-icon-14x14.png' %}" alt="Feed logo" width="14"></a></h1>
                     <h4 class="card-subtitle mb-0 pb-0 text-muted">Latest news and announcements.</h4>
                 </div>
-                <ul class="list-group list-group-flush">
                 {% for item in latest_newsitems %}
-                    <li class="list-group-item">
-                        <div class="card-block">
-                            {{ item.descriptor_small }}
-                        </div>
-                    </li>
+                    <div class="card card-news bg-transparent m-2">
+                        {% include 'news/news_card_content.html' with news=item %}
+                    </div>
                 {% endfor %}
-                </ul>
             </div>
         </div>
         {% endif %}
diff --git a/scipost/templates/scipost/news.html b/scipost/templates/scipost/news.html
index f1ff82c7973d63ea39d5871a87f2db21f5527961..71982a644a9dd23d04c4c0090f67b18d4eb5a525 100644
--- a/scipost/templates/scipost/news.html
+++ b/scipost/templates/scipost/news.html
@@ -4,25 +4,22 @@
 
 {% load staticfiles %}
 
-{% block bodysup %}
+{% block content %}
 
-
-<section>
-  <div class="flex-container">
-    <div class="flex-greybox">
-      <h1>SciPost News</h1>
+<div class="row">
+    <div class="col-12">
+        <h1 class="highlight">SciPost News</h1>
     </div>
-  </div>
-
-  <div class="flex-container">
-    <ul class="NewsItemsList">
-      {% for item in newsitems %}
-      <li>{{ item.descriptor_full|linebreaks }}</li>
-      {% endfor %}
-    </ul>
-  </div>
-
-</section>
-
+</div>
+
+<div class="row">
+    <div class="col-12">
+        {% for item in newsitems %}
+            <div class="card card-grey card-news">
+                {% include 'news/news_card_content.html' with news=item %}
+            </div>
+        {% endfor %}
+    </div>
+</div>
 
-{% endblock bodysup %}
+{% endblock content %}
diff --git a/scipost/templatetags/bootstrap.py b/scipost/templatetags/bootstrap.py
index 5f7816e6a69bf51bd32a83e28ffc55d21330e22b..c06ec80d8eba9401d0696df6e668fa82bceda2f7 100644
--- a/scipost/templatetags/bootstrap.py
+++ b/scipost/templatetags/bootstrap.py
@@ -35,8 +35,18 @@ def bootstrap(element, args='2,10'):
 
 
 @register.filter
-def bootstrap_inline(element):
-    markup_classes = {'label': 'sr-only', 'value': '', 'single_value': ''}
+def bootstrap_inline(element, args='2,10'):
+    args = [arg.strip() for arg in args.split(',')]
+    markup_classes = {
+        'label': 'sr-only col-md-%s' % args[0],
+        'value': 'col-md-%s' % args[1],
+        'single_value': ''
+    }
+    try:
+        markup_classes['label'] += ' col-form-label-%s' % args[2]
+        markup_classes['form_control'] = 'form-control-%s' % args[2]
+    except IndexError:
+        markup_classes['form_control'] = ''
     return render(element, markup_classes)
 
 
diff --git a/scipost/utils.py b/scipost/utils.py
index bac564998baf16da8d187141d2d65d5c12214b30..20e45420ea434d53d2921f003f7c24cbcc084405 100644
--- a/scipost/utils.py
+++ b/scipost/utils.py
@@ -8,7 +8,7 @@ from django.core.mail import EmailMultiAlternatives
 from django.template import Context, Template
 from django.utils import timezone
 
-from .models import Contributor, DraftInvitation, RegistrationInvitation, title_dict
+from .models import Contributor, DraftInvitation, RegistrationInvitation
 
 
 SCIPOST_SUMMARY_FOOTER = (
@@ -154,7 +154,7 @@ class Utils(object):
         cls.contributor.key_expires = datetime.datetime.strftime(
             datetime.datetime.now() + datetime.timedelta(days=2), "%Y-%m-%d %H:%M:%S")
         cls.contributor.save()
-        email_text = ('Dear ' + title_dict[cls.contributor.title] + ' ' +
+        email_text = ('Dear ' + cls.contributor.get_title_display() + ' ' +
                       cls.contributor.user.last_name +
                       ', \n\nYour request for registration to the SciPost publication portal' +
                       ' has been received. You now need to validate your email by visiting ' +
@@ -172,7 +172,7 @@ class Utils(object):
             '\n<p>Your registration will thereafter be vetted. Many thanks for your interest.</p>'
             '<p>The SciPost Team.</p>')
         email_context = Context({
-            'title': title_dict[cls.contributor.title],
+            'title': cls.contributor.get_title_display(),
             'last_name': cls.contributor.user.last_name,
             'activation_key': cls.contributor.activation_key,
         })
@@ -220,7 +220,7 @@ class Utils(object):
 
     @classmethod
     def send_registration_invitation_email(cls, renew=False):
-        signature = (title_dict[cls.invitation.invited_by.title] + ' '
+        signature = (cls.invitation.invited_by.get_title_display() + ' '
                      + cls.invitation.invited_by.user.first_name + ' '
                      + cls.invitation.invited_by.user.last_name)
         if not renew:
@@ -253,9 +253,9 @@ class Utils(object):
         email_text += 'Dear '
         email_text_html += 'Dear '
         if cls.invitation.message_style == 'F':
-            email_text += title_dict[cls.invitation.title] + ' ' + cls.invitation.last_name
+            email_text += cls.invitation.get_title_display() + ' ' + cls.invitation.last_name
             email_text_html += '{{ title }} {{ last_name }}'
-            email_context['title'] = title_dict[cls.invitation.title]
+            email_context['title'] = cls.invitation.get_title_display()
             email_context['last_name'] = cls.invitation.last_name
         else:
             email_text += cls.invitation.first_name
@@ -645,10 +645,10 @@ class Utils(object):
         Requires loading the 'notification' attribute.
         """
         email_context = Context({})
-        email_text = ('Dear ' + title_dict[cls.notification.contributor.title] +
+        email_text = ('Dear ' + cls.notification.contributor.get_title_display() +
                       ' ' + cls.notification.contributor.user.last_name)
         email_text_html = 'Dear {{ title }} {{ last_name }}'
-        email_context['title'] = title_dict[cls.notification.contributor.title]
+        email_context['title'] = cls.notification.contributor.get_title_display()
         email_context['last_name'] = cls.notification.contributor.user.last_name
         email_text += ',\n\n'
         email_text_html += ',<br/>'
diff --git a/scipost/views.py b/scipost/views.py
index 311914d0356ea12f32757ec0b628bcefe544f429..efe1d100321911733b52cc12338a6273788b2823 100644
--- a/scipost/views.py
+++ b/scipost/views.py
@@ -29,7 +29,7 @@ from guardian.decorators import permission_required
 from .constants import SCIPOST_SUBJECT_AREAS, subject_areas_raw_dict
 from .models import Contributor, CitationNotification, UnavailabilityPeriod,\
                     DraftInvitation, RegistrationInvitation,\
-                    title_dict, SciPost_from_addresses_dict,\
+                    SciPost_from_addresses_dict,\
                     AuthorshipClaim, SupportingPartner, SPBMembershipAgreement, EditorialCollege, EditorialCollegeFellowship
 from .forms import AuthenticationForm, DraftInvitationForm, UnavailabilityPeriodForm,\
                    RegistrationForm, RegistrationInvitationForm, AuthorshipClaimForm,\
@@ -43,9 +43,9 @@ from commentaries.models import Commentary
 from comments.models import Comment
 from journals.models import Publication, Issue
 from news.models import NewsItem
-from submissions.models import SUBMISSION_STATUS_PUBLICLY_UNLISTED
-from submissions.models import Submission, EditorialAssignment
-from submissions.models import RefereeInvitation, Report, EICRecommendation
+from submissions.constants import SUBMISSION_STATUS_PUBLICLY_UNLISTED
+from submissions.models import Submission, EditorialAssignment, RefereeInvitation,\
+                               Report, EICRecommendation
 from theses.models import ThesisLink
 
 
@@ -327,7 +327,7 @@ def invitation(request, key):
         form.fields['first_name'].initial = invitation.first_name
         form.fields['email'].initial = invitation.email
         errormessage = ''
-        welcome_message = ('Welcome, ' + title_dict[invitation.title] + ' '
+        welcome_message = ('Welcome, ' + invitation.get_title_display() + ' '
                            + invitation.last_name + ', and thanks in advance for '
                            'registering (by completing this form)')
         return render(request, 'scipost/register.html', {
@@ -374,7 +374,7 @@ def request_new_activation_link(request, oldkey):
     contributor.key_expires = datetime.datetime.strftime(
         datetime.datetime.now() + datetime.timedelta(days=2), "%Y-%m-%d %H:%M:%S")
     contributor.save()
-    email_text = ('Dear ' + title_dict[contributor.title] + ' ' + contributor.user.last_name +
+    email_text = ('Dear ' + contributor.get_title_display() + ' ' + contributor.user.last_name +
                   ', \n\n'
                   'Your request for a new email activation link for registration to the SciPost '
                   'publication portal has been received. You now need to visit this link within '
@@ -453,7 +453,7 @@ def vet_registration_request_ack(request, contributor_id):
                 except RefereeInvitation.DoesNotExist:
                     pending_ref_inv_exists = False
 
-                email_text = ('Dear ' + title_dict[contributor.title] + ' '
+                email_text = ('Dear ' + contributor.get_title_display() + ' '
                               + contributor.user.last_name +
                               ', \n\nYour registration to the SciPost publication portal '
                               'has been accepted. '
@@ -472,7 +472,7 @@ def vet_registration_request_ack(request, contributor_id):
                 emailmessage.send(fail_silently=False)
             else:
                 ref_reason = int(form.cleaned_data['refusal_reason'])
-                email_text = ('Dear ' + title_dict[contributor.title] + ' '
+                email_text = ('Dear ' + contributor.get_title_display() + ' '
                               + contributor.user.last_name +
                               ', \n\nYour registration to the SciPost publication portal '
                               'has been turned down, the reason being: '
@@ -1012,7 +1012,7 @@ def personal_page(request):
                          .filter(author=contributor, is_author_reply=True)
                          .order_by('-date_submitted'))
 
-    appellation = title_dict[contributor.title] + ' ' + contributor.user.last_name
+    appellation = contributor.get_title_display() + ' ' + contributor.user.last_name
     context = {
         'contributor': contributor,
         'user_groups': user_groups,
@@ -1300,7 +1300,7 @@ def email_group_members(request):
             # emailmessage.send(fail_silently=False)
             # with mail.get_connection() as connection:
             #     for member in form.cleaned_data['group'].user_set.all():
-            #         email_text = ('Dear ' + title_dict[member.contributor.title] + ' ' +
+            #         email_text = ('Dear ' + member.contributor.get_title_display() + ' ' +
             #                       member.last_name + ', \n\n'
             #                       + form.cleaned_data['email_text'])
             #         mail.EmailMessage(form.cleaned_data['email_subject'],
@@ -1316,7 +1316,7 @@ def email_group_members(request):
                             email_text = ''
                             email_text_html = ''
                             if form.cleaned_data['personalize']:
-                                email_text = ('Dear ' + title_dict[member.contributor.title]
+                                email_text = ('Dear ' + member.contributor.get_title_display()
                                               + ' ' + member.last_name + ', \n\n')
                                 email_text_html = 'Dear {{ title }} {{ last_name }},<br/>'
                             email_text += form.cleaned_data['email_text']
@@ -1334,7 +1334,7 @@ def email_group_members(request):
                                 'emails? <a href="https://scipost.org/unsubscribe/{{ key }}">'
                                 'Unsubscribe</a>.</p>')
                             email_context = Context({
-                                'title': title_dict[member.contributor.title],
+                                'title': member.contributor.get_title_display(),
                                 'last_name': member.last_name,
                                 'email_text': form.cleaned_data['email_text'],
                                 'key': member.contributor.activation_key,
diff --git a/submissions/forms.py b/submissions/forms.py
index bcac8cac71b9590b2e4c30d78aecfb018a45e53e..2a5b87466d12368f58d3c700ddbe3552db98fab1 100644
--- a/submissions/forms.py
+++ b/submissions/forms.py
@@ -1,8 +1,8 @@
 from django import forms
 from django.core.validators import RegexValidator
 
-from .models import ASSIGNMENT_BOOL, ASSIGNMENT_REFUSAL_REASONS,\
-                    Submission, RefereeInvitation, Report, EICRecommendation
+from .constants import ASSIGNMENT_BOOL, ASSIGNMENT_REFUSAL_REASONS
+from .models import Submission, RefereeInvitation, Report, EICRecommendation
 
 from scipost.constants import SCIPOST_SUBJECT_AREAS
 from scipost.models import Contributor
diff --git a/submissions/models.py b/submissions/models.py
index 93399cfad740fbba40a67515038e48918c9ec45d..a750d4834c43cdb62d5abe48e8cb1675fa2ec494 100644
--- a/submissions/models.py
+++ b/submissions/models.py
@@ -2,100 +2,24 @@ import datetime
 
 from django.utils import timezone
 from django.db import models, transaction
-from django.db.models import Q
 from django.contrib.postgres.fields import JSONField
-from django.template import Template, Context
+
+from .constants import ASSIGNMENT_REFUSAL_REASONS, SUBMISSION_STATUS, ASSIGNMENT_NULLBOOL,\
+                       SUBMISSION_TYPE, ED_COMM_CHOICES, REFEREE_QUALIFICATION, QUALITY_SPEC,\
+                       RANKING_CHOICES, REPORT_REC
+from .managers import SubmissionManager, EditorialAssignmentManager, EICRecommendationManager
 
 from scipost.behaviors import ArxivCallable
-from scipost.models import ChoiceArrayField, Contributor, TITLE_CHOICES, title_dict
-from scipost.constants import SCIPOST_DISCIPLINES, SCIPOST_SUBJECT_AREAS, subject_areas_dict
-from journals.models import SCIPOST_JOURNALS_SUBMIT, SCIPOST_JOURNALS_DOMAINS
-from journals.models import journals_submit_dict, journals_domains_dict
+from scipost.constants import TITLE_CHOICES
+from scipost.models import ChoiceArrayField, Contributor
+from scipost.constants import SCIPOST_DISCIPLINES, SCIPOST_SUBJECT_AREAS
+from journals.constants import SCIPOST_JOURNALS_SUBMIT, SCIPOST_JOURNALS_DOMAINS
 from journals.models import Publication
 
 
 ###############
 # Submissions:
 ###############
-SUBMISSION_STATUS = (
-    ('unassigned', 'Unassigned, undergoing pre-screening'),
-    ('assignment_failed', 'Failed to assign Editor-in-charge; manuscript rejected'),
-    ('EICassigned', 'Editor-in-charge assigned, manuscript under review'),
-    ('review_closed', 'Review period closed, editorial recommendation pending'),
-    # If revisions required: resubmission creates a new Submission object
-    ('revision_requested', 'Editor-in-charge has requested revision'),
-    ('resubmitted', 'Has been resubmitted'),
-    ('resubmitted_and_rejected', 'Has been resubmitted and subsequently rejected'),
-    ('resubmitted_and_rejected_visible',
-     'Has been resubmitted and subsequently rejected (still publicly visible)'),
-    # If acceptance/rejection:
-    ('voting_in_preparation', 'Voting in preparation (eligible Fellows being selected)'),
-    ('put_to_EC_voting', 'Undergoing voting at the Editorial College'),
-    ('EC_vote_completed', 'Editorial College voting rounded up'),
-    ('accepted', 'Publication decision taken: accept'),
-    ('rejected', 'Publication decision taken: reject'),
-    ('rejected_visible', 'Publication decision taken: reject (still publicly visible)'),
-    ('published', 'Published'),
-    # If withdrawn:
-    ('withdrawn', 'Withdrawn by the Authors'),
-    )
-submission_status_dict = dict(SUBMISSION_STATUS)
-
-SUBMISSION_STATUS_OUT_OF_POOL = [
-    'assignment_failed',
-    'resubmitted',
-    'published',
-    'withdrawn',
-    'rejected',
-    'rejected_visible',
-]
-
-# Submissions which should not appear in search lists
-SUBMISSION_STATUS_PUBLICLY_UNLISTED = [
-    'unassigned',
-    'assignment_failed',
-    'resubmitted',
-    'resubmitted_rejected',
-    'resubmitted_rejected_visible',
-    'rejected',
-    'published',
-    'withdrawn',
-]
-
-# Submissions which should not be viewable (except by admins, Fellows and authors)
-SUBMISSION_STATUS_PUBLICLY_INVISIBLE = [
-    'unassigned',
-    'assignment_failed',
-    'resubmitted_rejected',
-    'rejected',
-    'withdrawn',
-]
-
-# Submissions for which voting on a related recommendation is deprecated:
-SUBMISSION_STATUS_VOTING_DEPRECATED = [
-    'rejected',
-    'published',
-    'withdrawn',
-]
-
-SUBMISSION_TYPE = (
-    ('Letter', 'Letter (broad-interest breakthrough results)'),
-    ('Article', 'Article (in-depth reports on specialized research)'),
-    ('Review', 'Review (candid snapshot of current research in a given area)'),
-)
-submission_type_dict = dict(SUBMISSION_TYPE)
-
-
-class SubmissionManager(models.Manager):
-    def get_pool(self, user):
-        return self.exclude(status__in=SUBMISSION_STATUS_OUT_OF_POOL)\
-                .exclude(is_current=False)\
-                .exclude(authors=user.contributor)\
-                .exclude(Q(author_list__icontains=user.last_name),
-                         ~Q(authors_false_claims=user.contributor))\
-                .order_by('-submission_date')
-
-
 class Submission(ArxivCallable, models.Model):
     # Main submission fields
     author_comments = models.TextField(blank=True, null=True)
@@ -116,7 +40,8 @@ class Submission(ArxivCallable, models.Model):
     secondary_areas = ChoiceArrayField(
         models.CharField(max_length=10, choices=SCIPOST_SUBJECT_AREAS),
         blank=True, null=True)
-    status = models.CharField(max_length=30, choices=SUBMISSION_STATUS, default='unassigned')  # set by Editors
+    # Status set by Editors
+    status = models.CharField(max_length=30, choices=SUBMISSION_STATUS, default='unassigned')
     subject_area = models.CharField(max_length=10, choices=SCIPOST_SUBJECT_AREAS,
                                     verbose_name='Primary subject area', default='Phys:QP')
     submission_type = models.CharField(max_length=10, choices=SUBMISSION_TYPE,
@@ -234,40 +159,6 @@ class Submission(ArxivCallable, models.Model):
             arxiv_identifier_wo_vn_nr=self.arxiv_identifier_wo_vn_nr
         ).exclude(pk=self.id).order_by('-arxiv_vn_nr')
 
-    def header_as_table(self):
-        # for Submission page
-        header = '<table class="submission_header">'
-        header += '<tr><td>Title: </td><td>{{ title }}</td></tr>'
-        header += '<tr><td>Author(s): </td><td>{{ author_list }}</td></tr>'
-        header += '<tr><td>As Contributors: </td>'
-        if self.authors.all():
-            header += '<td>'
-            for auth in self.authors.all():
-                header += ('<a href="/contributor/' + str(auth.id) + '">' + auth.user.first_name
-                           + ' ' + auth.user.last_name + '</a>')
-            header += '</td>'
-        else:
-            header += '<td>(none claimed)</td>'
-        header += ('</tr>'
-                   '<tr><td>arxiv Link: </td>'
-                   '<td><a href="{{ arxiv_link }}" target="_blank">{{ arxiv_link }}</a></td></tr>'
-                   '<tr><td>Date submitted: </td><td>{{ submission_date }}</td></tr>'
-                   '<tr><td>Submitted by: </td><td>{{ submitted_by }}</td></tr>'
-                   '<tr><td>Submitted to: </td><td>{{ to_journal }}</td></tr>'
-                   '<tr><td>Domain(s): </td><td>{{ domain }}</td></tr>'
-                   '<tr><td>Subject area: </td><td>{{ subject_area }}</td></tr>'
-                   '</table>')
-        template = Template(header)
-        context = Context({
-            'title': self.title, 'author_list': self.author_list,
-            'arxiv_link': self.arxiv_link, 'submission_date': self.submission_date,
-            'submitted_by': self.submitted_by,
-            'to_journal': journals_submit_dict[self.submitted_to_journal],
-            'domain': journals_domains_dict[self.domain],
-            'subject_area': subject_areas_dict[self.subject_area],
-        })
-        return template.render(context)
-
     def count_accepted_invitations(self):
         return self.refereeinvitation_set.filter(accepted=True).count()
 
@@ -292,86 +183,11 @@ class Submission(ArxivCallable, models.Model):
     def count_awaiting_vetting(self):
         return self.reports.filter(status=0).count()
 
-    def refereeing_status_as_p(self):
-        nr_ref_invited = RefereeInvitation.objects.filter(submission=self).count()
-        nr_ref_accepted = RefereeInvitation.objects.filter(submission=self, accepted=True).count()
-        nr_ref_declined = RefereeInvitation.objects.filter(submission=self, accepted=False).count()
-        nr_invited_reports_in = Report.objects.filter(submission=self,
-                                                      status=1, invited=True).count()
-        nr_contrib_reports_in = Report.objects.filter(submission=self,
-                                                      status=1, invited=False).count()
-        nr_reports_awaiting_vetting = Report.objects.filter(submission=self, status=0).count()
-        nr_reports_refused = Report.objects.filter(submission=self, status__lte=-1).count()
-        header = ('<p>Nr referees invited: ' + str(nr_ref_invited) +
-                  ' [' + str(nr_ref_accepted) + ' accepted/ ' +
-                  str(nr_ref_declined) + ' declined/ ' +
-                  str(nr_ref_invited - nr_ref_accepted - nr_ref_declined) +
-                  ' response pending]</p>' +
-                  '<p>Nr reports obtained: ' +
-                  str(nr_invited_reports_in + nr_contrib_reports_in) + ' [' +
-                  str(nr_invited_reports_in) + ' invited/ ' + str(nr_contrib_reports_in) +
-                  ' contributed], nr refused: ' + str(nr_reports_refused) +
-                  ', nr awaiting vetting: ' + str(nr_reports_awaiting_vetting) + '</p>')
-        template = Template(header)
-        context = Context({})
-        return template.render(context)
-
-    def version_info_as_li(self):
-        # for listing all versions of a Submission
-        header = ('<li>'
-                  '<p>'
-                  '<a href="/submission/{{ arxiv_identifier_w_vn_nr }}" '
-                  'class="pubtitleli">version {{ arxiv_vn_nr }}</a>')
-        if self.is_current:
-            header += ' (current version)'
-        else:
-            header += ' (deprecated version {{ arxiv_vn_nr }})'
-        header += ('</p>'
-                   '</li>')
-        context = Context({'arxiv_identifier_w_vn_nr': self.arxiv_identifier_w_vn_nr,
-                           'arxiv_vn_nr': self.arxiv_vn_nr, })
-        template = Template(header)
-        return template.render(context)
-
-    def status_info_as_table(self):
-        header = '<table><tr><td>Current status: </td><td>&nbsp;</td><td>{{ status }}'
-        context = Context({'status': submission_status_dict[self.status], })
-        try:
-            context['citation'] = self.publication.citation_for_web_linked()
-            header += ' as {{ citation }}'
-        except Publication.DoesNotExist:
-            pass
-        header += '</td></tr></table>'
-        template = Template(header)
-        return template.render(context)
-
 
 ######################
 # Editorial workflow #
 ######################
 
-ASSIGNMENT_BOOL = ((True, 'Accept'), (False, 'Decline'))
-ASSIGNMENT_NULLBOOL = ((None, 'Response pending'), (True, 'Accept'), (False, 'Decline'))
-
-ASSIGNMENT_REFUSAL_REASONS = (
-    ('BUS', 'Too busy'),
-    ('VAC', 'Away on vacation'),
-    ('COI', 'Conflict of interest: coauthor in last 5 years'),
-    ('CCC', 'Conflict of interest: close colleague'),
-    ('NIR', 'Cannot give an impartial assessment'),
-    ('NIE', 'Not interested enough'),
-    ('DNP', 'SciPost should not even consider this paper'),
-    )
-assignment_refusal_reasons_dict = dict(ASSIGNMENT_REFUSAL_REASONS)
-
-
-class EditorialAssignmentManager(models.Manager):
-    def get_for_user_in_pool(self, user):
-        return self.exclude(submission__authors=user.contributor)\
-                .exclude(Q(submission__author_list__icontains=user.last_name),
-                         ~Q(submission__authors_false_claims=user.contributor))
-
-
 class EditorialAssignment(models.Model):
     submission = models.ForeignKey(Submission, on_delete=models.CASCADE)
     to = models.ForeignKey(Contributor, on_delete=models.CASCADE)
@@ -391,34 +207,6 @@ class EditorialAssignment(models.Model):
                 self.submission.title[:30] + ' by ' + self.submission.author_list[:30] +
                 ', requested on ' + self.date_created.strftime('%Y-%m-%d'))
 
-    def info_as_li(self):
-        context = Context({'first_name': self.to.user.first_name,
-                           'last_name': self.to.user.last_name,
-                           'date_created': self.date_created.strftime('%Y-%m-%d %H:%M')})
-        info = '<li'
-        if self.accepted:
-            info += ' style="color: green"'
-        elif self.deprecated:
-            info += ' style="color: purple"'
-        elif not self.accepted:
-            if self.refusal_reason == 'NIE' or self.refusal_reason == 'DNP':
-                info += ' style="color: #CC0000"'
-            else:
-                info += ' style="color: #FF7700"'
-        info += '>{{ first_name }} {{ last_name }}, requested {{ date_created }}'
-        if self.accepted:
-            info += ', accepted {{ date_answered }}'
-            context['date_answered'] = self.date_answered.strftime('%Y-%m-%d %H:%M')
-        if self.deprecated:
-            info += ', deprecated'
-        if self.refusal_reason:
-            info += ', declined {{ date_answered }}, reason: {{ reason }}'
-            context['date_answered'] = self.date_answered.strftime('%Y-%m-%d %H:%M')
-            context['reason'] = assignment_refusal_reasons_dict[self.refusal_reason]
-        info += '</li>'
-        template = Template(info)
-        return template.render(context)
-
 
 class RefereeInvitation(models.Model):
     submission = models.ForeignKey(Submission, on_delete=models.CASCADE)
@@ -447,71 +235,11 @@ class RefereeInvitation(models.Model):
                 self.submission.title[:30] + ' by ' + self.submission.author_list[:30] +
                 ', invited on ' + self.date_invited.strftime('%Y-%m-%d'))
 
-    def summary_as_tds(self):
-        context = Context({'first_name': self.first_name, 'last_name': self.last_name,
-                           'date_invited': self.date_invited.strftime('%Y-%m-%d %H:%M')})
-        output = ('<td>{{ first_name }} {{ last_name }}</td><td>invited <br/>'
-                  '{{ date_invited }}</td><td>')
-        if self.cancelled:
-            output += '<strong style="color: red;">cancelled</strong>'
-        elif self.accepted is not None:
-            if self.accepted:
-                output += '<strong style="color: green">task accepted</strong> '
-            else:
-                output += '<strong style="color: red">task declined</strong> '
-            output += '<br/>{{ date_responded }}'
-            context['date_responded'] = self.date_responded.strftime('%Y-%m-%d %H:%M')
-        else:
-            output += 'response pending'
-        if self.fulfilled:
-            output += '<br/><strong style="color: green">task fulfilled</strong>'
-        output += '</td>'
-        template = Template(output)
-        return template.render(context)
-
 
 ###########
 # Reports:
 ###########
 
-REFEREE_QUALIFICATION = (
-    (4, 'expert in this subject'),
-    (3, 'very knowledgeable in this subject'),
-    (2, 'knowledgeable in this subject'),
-    (1, 'generally qualified'),
-    (0, 'not qualified'),
-    )
-ref_qualif_dict = dict(REFEREE_QUALIFICATION)
-
-QUALITY_SPEC = (
-    (6, 'perfect'),
-    (5, 'excellent'),
-    (4, 'good'),
-    (3, 'reasonable'),
-    (2, 'acceptable'),
-    (1, 'below threshold'),
-    (0, 'mediocre'),
-    )
-quality_spec_dict = dict(QUALITY_SPEC)
-
-
-RANKING_CHOICES = (
-    (101, '-'),  # Only values between 0 and 100 are kept, anything outside those limits is discarded.
-    (100, 'top'), (80, 'high'), (60, 'good'), (40, 'ok'), (20, 'low'), (0, 'poor')
-    )
-ranking_choices_dict = dict(RANKING_CHOICES)
-
-REPORT_REC = (
-    (1, 'Publish as Tier I (top 10% of papers in this journal, qualifies as Select) NOTE: SELECT NOT YET OPEN, STARTS EARLY 2017'),
-    (2, 'Publish as Tier II (top 50% of papers in this journal)'),
-    (3, 'Publish as Tier III (meets the criteria of this journal)'),
-    (-1, 'Ask for minor revision'),
-    (-2, 'Ask for major revision'),
-    (-3, 'Reject')
-    )
-report_rec_dict = dict(REPORT_REC)
-
-
 class Report(models.Model):
     """ Both types of reports, invited or contributed. """
     # status: see forms.py:REPORT_REFUSAL_CHOICES
@@ -558,84 +286,11 @@ class Report(models.Model):
         return (self.author.user.first_name + ' ' + self.author.user.last_name + ' on ' +
                 self.submission.title[:50] + ' by ' + self.submission.author_list[:50])
 
-    def print_identifier(self):
-        context = Context({'id': self.id, 'author_id': self.author.id,
-                           'first_name': self.author.user.first_name,
-                           'last_name': self.author.user.last_name,
-                           'date_submitted': self.date_submitted.strftime("%Y-%m-%d")})
-        output = '<div class="reportid">'
-        output += '<h3><a id="report_id{{ id }}"></a>'
-        if self.anonymous:
-            output += 'Anonymous Report {{ id }}'
-        else:
-            output += '<a href="/contributor/{{ author_id }}">{{ first_name }} {{ last_name }}</a>'
-        output += ' on {{ date_submitted }}</h3></div>'
-        template = Template(output)
-        return template.render(context)
-
-    def print_contents(self):
-        context = Context({'strengths': self.strengths, 'weaknesses': self.weaknesses,
-                           'report': self.report, 'requested_changes': self.requested_changes})
-        output = ('<div class="row"><div class="col-12"><h3 class="highlight tight">'
-                  'Strengths</h3><div class="pl-md-4">{{ strengths|linebreaks }}</div></div></div>'
-                  '<div class="row"><div class="col-12"><h3 class="highlight tight">'
-                  'Weaknesses</h3><div class="pl-md-4">{{ weaknesses|linebreaks }}</div></div></div>'
-                  '<div class="row"><div class="col-12"><h3 class="highlight tight">'
-                  'Report</h3><div class="pl-md-4">{{ report|linebreaks }}</div></div></div>'
-                  '<div class="row"><div class="col-12"><h3 class="highlight tight">'
-                  'Requested changes</h3><div class="pl-md-4">'
-                  '<p>{{ requested_changes|linebreaks }}</p>'
-                  '<div class="reportRatings"><ul>'
-                  '<li>validity: ' + ranking_choices_dict[self.validity] + '</li>'
-                  '<li>significance: ' + ranking_choices_dict[self.significance] + '</li>'
-                  '<li>originality: ' + ranking_choices_dict[self.originality] + '</li>'
-                  '<li>clarity: ' + ranking_choices_dict[self.clarity] + '</li>'
-                  '<li>formatting: ' + quality_spec_dict[self.formatting] + '</li>'
-                  '<li>grammar: ' + quality_spec_dict[self.grammar] + '</li>'
-                  '</ul></div></div></div></div>')
-        template = Template(output)
-        return template.render(context)
-
-    def print_contents_for_editors(self):
-        context = Context({
-            'id': self.id, 'author_id': self.author.id,
-            'author_first_name': self.author.user.first_name,
-            'author_last_name': self.author.user.last_name,
-            'date_submitted': self.date_submitted.strftime("%Y-%m-%d"),
-            'remarks_for_editors': self.remarks_for_editors,
-        })
-        output = '<div class="reportid">'
-        output += '<h3>'
-        if self.anonymous:
-            output += '(chose public anonymity) '
-        output += ('<a href="/contributor/{{ author_id }}">'
-                   '{{ author_first_name }} {{ author_last_name }}</a>'
-                   ' on {{ date_submitted }}</h3></div>'
-                   '<div class="row"><div class="col-12"><h3 class="highlight tight">Qualification</h3>'
-                   '<div class="pl-md-4">'
-                   + ref_qualif_dict[self.qualification] + '</div></div></div>')
-        output += self.print_contents()
-        output += '<div class="row"><div class="col-12"><h3>Remarks for editors</h3><div class="pl-md-4">{{ remarks_for_editors }}</div></div></div>'
-        output += '<div class="row"><div class="col-12"><h3>Recommendation</h3><div class="pl-md-4">%s</div></div></div>' % report_rec_dict[self.recommendation]
-        template = Template(output)
-        return template.render(context)
-
 
 ##########################
 # EditorialCommunication #
 ##########################
 
-ED_COMM_CHOICES = (
-    ('EtoA', 'Editor-in-charge to Author'),
-    ('EtoR', 'Editor-in-charge to Referee'),
-    ('EtoS', 'Editor-in-charge to SciPost Editorial Administration'),
-    ('AtoE', 'Author to Editor-in-charge'),
-    ('RtoE', 'Referee to Editor-in-Charge'),
-    ('StoE', 'SciPost Editorial Administration to Editor-in-charge'),
-    )
-ed_comm_choices_dict = dict(ED_COMM_CHOICES)
-
-
 class EditorialCommunication(models.Model):
     """
     Each individual communication between Editor-in-charge
@@ -656,47 +311,11 @@ class EditorialCommunication(models.Model):
                    + self.submission.author_list[:30])
         return output
 
-    def print_contents_as_li(self):
-        context = Context({'timestamp': self.timestamp.strftime("%Y-%m-%d %H:%M"),
-                           'text': self.text})
-        output = '<li><p>'
-        if self.comtype == 'EtoA':
-            output += 'From you to Authors'
-        elif self.comtype == 'EtoR':
-            output += 'From you to Referee '
-            try:
-                output += self.referee.user.first_name + ' ' + self.referee.user.last_name
-            except AttributeError:
-                pass
-        elif self.comtype == 'EtoS':
-            output += 'From you to SciPost Ed Admin'
-        elif self.comtype == 'AtoE':
-            output += 'From Authors to you'
-        elif self.comtype == 'RtoE':
-            output += 'From Referee '
-            try:
-                output += (self.referee.user.first_name + ' ' +
-                           self.referee.user.last_name + ' to you')
-            except AttributeError:
-                pass
-        elif self.comtype == 'StoE':
-            output += 'From SciPost Ed Admin to you'
-        output += ' on {{ timestamp }}</p><p>{{ text }}</p>'
-        template = Template(output)
-        return template.render(context)
-
 
 ############################
 # Editorial Recommendation #
 ############################
 
-class EICRecommendationManager(models.Manager):
-    def get_for_user_in_pool(self, user):
-        return self.exclude(submission__authors=user.contributor)\
-                .exclude(Q(submission__author_list__icontains=user.last_name),
-                         ~Q(submission__authors_false_claims=user.contributor))
-
-
 # From the Editor-in-charge of a Submission
 class EICRecommendation(models.Model):
     submission = models.ForeignKey(Submission, on_delete=models.CASCADE)
@@ -719,7 +338,7 @@ class EICRecommendation(models.Model):
 
     def __str__(self):
         return (self.submission.title[:20] + ' by ' + self.submission.author_list[:30] +
-                ', ' + report_rec_dict[self.recommendation])
+                ', ' + self.get_recommendation_display())
 
     @property
     def nr_for(self):
@@ -732,41 +351,3 @@ class EICRecommendation(models.Model):
     @property
     def nr_abstained(self):
         return self.voted_abstain.count()
-
-    def print_for_authors(self):
-        output = ('<h3>Date: {{ date_submitted }}</h3>'
-                  '<h3>Remarks for authors</h3>'
-                  '<p>{{ remarks_for_authors }}</p>'
-                  '<h3>Requested changes</h3>'
-                  '<p>{{ requested_changes }}</p>'
-                  '<h3>Recommendation</h3>'
-                  '<p>{{ recommendation }}</p>')
-        context = Context({
-            'date_submitted': self.date_submitted.strftime('%Y-%m-%d %H:%M'),
-            'remarks_for_authors': self.remarks_for_authors,
-            'requested_changes': self.requested_changes,
-            'recommendation': report_rec_dict[self.recommendation], })
-        template = Template(output)
-        return template.render(context)
-
-    def print_for_Fellows(self):
-        output = ('<h3>By {{ Fellow }}, formulated on {{ date_submitted }}</h3>'
-                  '<h3>Remarks for authors</h3>'
-                  '<p>{{ remarks_for_authors }}</p>'
-                  '<h3>Requested changes</h3>'
-                  '<p>{{ requested_changes }}</p>'
-                  '<h3>Remarks for Editorial College</h3>'
-                  '<p>{{ remarks_for_editorial_college }}</p>'
-                  '<h3>Recommendation</h3>'
-                  '<p>{{ recommendation }}</p>')
-        context = Context({
-            'Fellow': (title_dict[self.submission.editor_in_charge.title] +
-                       ' ' + self.submission.editor_in_charge.user.first_name +
-                       ' ' + self.submission.editor_in_charge.user.last_name),
-            'date_submitted': self.date_submitted.strftime('%Y-%m-%d %H:%M'),
-            'remarks_for_authors': self.remarks_for_authors,
-            'requested_changes': self.requested_changes,
-            'remarks_for_editorial_college': self.remarks_for_editorial_college,
-            'recommendation': report_rec_dict[self.recommendation], })
-        template = Template(output)
-        return template.render(context)
diff --git a/submissions/templates/submissions/_recommendation_fellow_content.html b/submissions/templates/submissions/_recommendation_fellow_content.html
index 3ee1572de1ee79da41f0751b8a84edf9bf8e7e61..1cb07e3a91404cba5bfef1a2d0c9b5dc61cf1813 100644
--- a/submissions/templates/submissions/_recommendation_fellow_content.html
+++ b/submissions/templates/submissions/_recommendation_fellow_content.html
@@ -1,15 +1,10 @@
-<div class="card-block">
-    <h3 class="card-title text-blue">By {{recommendation.submission.editor_in_charge.get_title_display}} {{recommendation.submission.editor_in_charge.user.first_name}} {{recommendation.submission.editor_in_charge.user.last_name}}, formulated on {{recommendation.date_submitted}}</h3>
-
-    <h3 class="pb-0">Remarks for authors</h3>
-    <p class="pl-md-3">{{recommendation.remarks_for_authors}}</p>
+{% extends 'submissions/_recommendation_author_content.html' %}
 
-    <h3 class="pb-0">Requested changes</h3>
-    <p class="pl-md-3">{{recommendation.requested_changes}}</p>
+{% block recommendation_header %}
+    <h3 class="card-title text-blue">By {{recommendation.submission.editor_in_charge.get_title_display}} {{recommendation.submission.editor_in_charge.user.first_name}} {{recommendation.submission.editor_in_charge.user.last_name}}, formulated on {{recommendation.date_submitted}}</h3>
+{% endblock %}
 
+{% block recommendation_before_recommendation %}
     <h3 class="pb-0">Remarks for Editorial College</h3>
     <p class="pl-md-3">{{recommendation.remarks_for_editorial_college}}</p>
-
-    <h3 class="pb-0">Recommendation</h3>
-    <p class="pl-md-3 mb-0">{{recommendation.get_recommendation_display}}</p>
-</div>
+{% endblock %}
diff --git a/submissions/templates/submissions/_single_report.html b/submissions/templates/submissions/_single_report.html
deleted file mode 100644
index ef15a1b73e89f08adffe69a19aebbd318de8e7e6..0000000000000000000000000000000000000000
--- a/submissions/templates/submissions/_single_report.html
+++ /dev/null
@@ -1,60 +0,0 @@
-{% load scipost_extras %}
-{% load submissions_extras %}
-
-<div class="row">
-    <div class="col-12">
-        <div class="report">
-            {% if user.contributor == submission.editor_in_charge or user|is_in_group:'Editorial Administrators'  and not is_author or user|is_in_group:'Editorial Administrators' and not is_author_unchecked %}
-
-                <div class="reportid">
-                    <h3>{% if report.anonymous %}(chose public anonymity) {% endif %}<a href="{% url 'scipost:contributor_info' report.author.id %}">{{ report.author.user.first_name }} {{ report.author.user.last_name }}</a>
-                        on {{ report.date_submitted|date:'Y-n-j' }}</h3>
-                    </h3>
-                </div>
-
-                {% if report.flagged %}
-                    <h4 class="text-danger font-weight-bold">CAUTION: check if this referee has been flagged by the authors</h4>
-                {% endif %}
-
-                <div class="row">
-                    <div class="col-12">
-                        <h3 class="highlight tight">Qualification</h3>
-                        <div class="pl-md-4">{{ report.get_qualification_display}}</div>
-                    </div>
-                </div>
-
-                {% include 'submissions/_single_report_content.html' with report=report %}
-
-                <div class="row">
-                    <div class="col-12">
-                        <h3>Remarks for editors</h3>
-                        <div class="pl-md-4">{{ report.remarks_for_editors }}</div>
-                    </div>
-                </div>
-                <div class="row">
-                    <div class="col-12">
-                        <h3>Recommendation</h3>
-                        <div class="pl-md-4">{{report.get_recommendation_display}}</div>
-                    </div>
-                </div>
-            {% else %}
-                <div class="reportid">
-                    <h3 id="report_id{{report.id}}">{% if report.anonymous %}Anonymous Report {{report.id}}{% else %}<a href="{% url 'scipost:contributor_info' report.author.id %}">{{ report.author.user.first_name }} {{ report.author.user.last_name }}</a>{% endif %}
-                        on {{ report.date_submitted|date:'Y-n-j' }}</h3>
-                    </h3>
-                </div>
-                {# {{report.print_identifier}}#}
-
-                {# {{ report.print_contents }}#}
-                {% include 'submissions/_single_report_content.html' with report=report %}
-            {% endif %}
-
-            <hr class="small">
-            <h3><a href="{% url 'comments:reply_to_report' report_id=report.id %}">Reply to this Report</a> (authors only)</h3>
-
-            {% for reply in report.comment_set.vetted %}
-                {% include 'comments/_single_comment_with_link.html' with comment=reply perms=perms user=user %}
-            {% endfor %}
-        </div>
-    </div>
-</div>
diff --git a/submissions/templates/submissions/_submission_assignment_request.html b/submissions/templates/submissions/_submission_assignment_request.html
index 8b253bb83047c04ecdc63cc809e3c7a1ea569681..fa5d6099f74988830f2e27c18cac379fc0b2b548 100644
--- a/submissions/templates/submissions/_submission_assignment_request.html
+++ b/submissions/templates/submissions/_submission_assignment_request.html
@@ -1,11 +1,8 @@
 {% load bootstrap %}
 
 <div class="card-block">
-    {{ assignment.submission.header_as_table }}
+    {% include 'submissions/_submission_summary.html' with submission=assignment.submission %}
     <br />
-
-    <h4>Abstract:</h4>
-    <p>{{ assignment.submission.abstract }}</p>
 </div>
 <div class="card-footer">
     <h1>Accept or Decline this Assignment</h1>
diff --git a/submissions/templates/submissions/_submission_card_in_pool.html b/submissions/templates/submissions/_submission_card_in_pool.html
index e4dda343e3a45bc1f82041f584ddf765486a2505..5012644dfd3e3efc3e712ad2271b7232c5879f0a 100644
--- a/submissions/templates/submissions/_submission_card_in_pool.html
+++ b/submissions/templates/submissions/_submission_card_in_pool.html
@@ -42,7 +42,7 @@
             <h4>EIC Assignment requests:</h4>
             <ul>
               {% for assignment in submission.editorialassignment_set.all %}
-                  {{ assignment.info_as_li }}
+                  {% include 'submissions/_assignment_info.html' with assignment=assignment %}
               {% endfor %}
             </ul>
         {% endif %}
diff --git a/submissions/templates/submissions/accept_or_decline_ref_invitations.html b/submissions/templates/submissions/accept_or_decline_ref_invitations.html
index 518a0144693bf276a95899ce0a6651f150e24858..12465e7f61e2510bd50d5b474c0dad6df54fe8c5 100644
--- a/submissions/templates/submissions/accept_or_decline_ref_invitations.html
+++ b/submissions/templates/submissions/accept_or_decline_ref_invitations.html
@@ -33,10 +33,8 @@ $(document).ready(function(){
   </div>
   <br>
   <hr>
-  {{ invitation_to_consider.submission.header_as_table }}
-  <br />
-  <h4>Abstract:</h4>
-  <p>{{ invitation_to_consider.submission.abstract }}</p>
+  {% include 'submissions/_submission_summary.html' with submission=invitation_to_consider.submission %}
+
   <br/>
   <hr>
   <div class="flex-greybox">
diff --git a/submissions/templates/submissions/assign_submission.html b/submissions/templates/submissions/assign_submission.html
index 6365bf7feb95c769e7d3f8fd23f9e55a462faf47..15599863694c23cd653f9bc7a8b8e576f10ce594 100644
--- a/submissions/templates/submissions/assign_submission.html
+++ b/submissions/templates/submissions/assign_submission.html
@@ -10,10 +10,9 @@
 
   <br>
   <hr>
-  {{ submission_to_assign.header_as_table }}
-  <br />
-  <h4>Abstract:</h4>
-  <p>{{ submission_to_assign.abstract }}</p>
+
+  {% include 'submissions/_submission_summary.html' with submission=submission_to_assign %}
+
   <br/>
   <hr/>
 
@@ -29,7 +28,7 @@
 
   <br/>
   <hr>
-  {{ submission_to_assign.status_info_as_table }}
+  {% include 'submissions/_submission_status_block.html' with submission=submission %}
   <hr>
 
   {% if submission_to_assign.editorialassignment_set.all %}
@@ -37,7 +36,7 @@
   <h3>If more than 5 Fellows have declined an assignment for a red-marked reason, the Submission should be rejected.</h3>
   <ul>
     {% for assignment in sub.editorialassignment_set.all %}
-    {{ assignment.info_as_li }}
+        {% include 'submissions/_assignment_info.html' with assignment=assignment %}
     {% endfor %}
   </ul>
   {% endif %}
diff --git a/submissions/templates/submissions/editorial_page.html b/submissions/templates/submissions/editorial_page.html
index 989af1cedd256f936cf5fc3059bba6ad5655488e..e05ae5187a5e3654bfc031a68f6cd24ab1e767ea 100644
--- a/submissions/templates/submissions/editorial_page.html
+++ b/submissions/templates/submissions/editorial_page.html
@@ -2,199 +2,215 @@
 
 {% block pagetitle %}: editorial page for submission{% endblock pagetitle %}
 
-{% block headsup %}
-
 {% load scipost_extras %}
 {% load submissions_extras %}
+{% load bootstrap %}
+
+{% block content %}
+
+<div class="row">
+    <div class="col-12">
+        <div class="card card-grey">
+            <div class="card-block">
+                <h1>Editorial Page for Submission</h1>
+                {% if not submission.is_current %}
+                    <h3 class="text-danger">This is not the editorial page for the current version.</h3>
+                    <p>Please go to the current version's page.</p>
+                {% endif %}
+
+                <h3>(go to the <a href="{% url 'submissions:submission' arxiv_identifier_w_vn_nr=submission.arxiv_identifier_w_vn_nr %}">Submissions Page</a> to view Reports and Comments)</h3>
+            </div>
+        </div>
+    </div>
+</div>
+
+
+{% if other_versions %}
+    <div class="row">
+        <div class="col-12">
+            <h3>Other versions of this Submission exist:</h3>
+            <div class="pl-4">
+                {% for vn in other_versions %}
+                    {% include 'submissions/_submission_version.html' with submission=vn %}
+                {% endfor %}
+            </div>
+        </div>
+    </div>
+{% endif %}
+
+
+<div class="row">
+    <div class="col-12">
+        <h2>Submission:</h2>
+        {% include 'submissions/_submission_summary_short.html' with submission=submission %}
+    </div>
+</div>
+
+<div class="row">
+    <div class="col-12">
+      {% if submission.author_comments %}
+          <h3>Author comments upon resubmission</h3>
+          <p>{{ submission.author_comments|linebreaks }}</p>
+      {% endif %}
+
+      {% if submission.list_of_changes %}
+          <h3>List of changes</h3>
+          <p>{{ submission.list_of_changes|linebreaks }}</p>
+      {% endif %}
+
+      {% if submission.remarks_for_editors %}
+          <h3>Comments for the Editor-in-charge</h3>
+          <p>{{ submission.remarks_for_editors|linebreaks }}</p>
+      {% endif %}
+
+      {% if submission.referees_suggested %}
+          <h3>Referees suggested by authors upon submission:</h3>
+          <p>{{ submission.referees_suggested }}</p>
+      {% endif %}
+
+      {% if submission.referees_flagged %}
+          <h3>Referees flagged upon submission (treat reports with caution):</h3>
+          <p>{{ submission.referees_flagged }}</p>
+      {% endif %}
 
-{% endblock headsup %}
-
-{% block bodysup %}
-
-
-<section>
-  <div class="flex-greybox">
-    <h1>Editorial Page for Submission</h1>
-    {% if not submission.is_current %}
-    <h3 style="color: red;">This is not the editorial page for the current version.</h3>
-    <p>Please go to the current version's page.</p>
-    {% endif %}
-    <h3>(go to the <a href="{% url 'submissions:submission' arxiv_identifier_w_vn_nr=submission.arxiv_identifier_w_vn_nr %}">
-	Submissions Page</a> to view Reports and Comments)</h3>
-  </div>
-
-  {% if other_versions %}
-  <h3>Other versions of this Submission exist:</h3>
-  <ul>
-    {% for vn in other_versions %}
-    {{ vn.version_info_as_li }}
-    {% endfor %}
-  </ul>
-  {% endif %}
-
-  <hr class="hr12">
-  <div class="row">
-    <div class="col-4">
-      <h2>Submission:</h2>
     </div>
-  </div>
-  {{ submission.header_as_table }}
-
-  {% if submission.author_comments %}
-  <h3>Author comments upon resubmission</h3>
-  <p>{{ submission.author_comments|linebreaks }}</p>
-  {% endif %}
-
-  {% if submission.list_of_changes %}
-  <h3>List of changes</h3>
-  <p>{{ submission.list_of_changes|linebreaks }}</p>
-  {% endif %}
-
-  {% if submission.remarks_for_editors %}
-  <h3>Comments for the Editor-in-charge</h3>
-  <p>{{ submission.remarks_for_editors|linebreaks }}</p>
-  {% endif %}
-
-  {% if submission.referees_suggested %}
-  <h3>Referees suggested by authors upon submission:</h3>
-  <p>{{ submission.referees_suggested }}</p>
-  {% endif %}
-
-  {% if submission.referees_flagged %}
-  <h3>Referees flagged upon submission (treat reports with caution):</h3>
-  <p>{{ submission.referees_flagged }}</p>
-  {% endif %}
-</section>
+</div>
 
 {% if recommendation %}
-<section>
-  <div class="flex-greybox">
-    <h1>Editorial Recommendation</h1>
-  </div>
-  {{ recommendation.print_for_authors }}
-</section>
+    <div class="row">
+        <div class="col-12">
+            <h2 class="highlight">Editorial Recommendation</h2>
+            {% include 'submissions/_recommendation_author_content.html' with recommendation=recommendation %}
+        </div>
+    </div>
 {% endif %}
 
-<section>
-  <div class="flex-greybox">
-    <h1>Editorial Workflow</h1>
-    <a href="{% url 'submissions:editorial_workflow' %}">How-to guide: summary of the editorial workflow</a>
-  </div>
-  <h3>Status:</h3>
-  {{ submission.status_info_as_table }}
-  {% if submission|required_actions %}
-  <div class="flex-container">
-    <div class-"flex-whitebox" style="background-color: #ffaaaa;">
-      <h3>Required actions:</h3>
-      <ul>
-	{% for todoitem in submission|required_actions %}
-	<li>{{ todoitem }}</li>
-	{% endfor %}
-      </ul>
+<div class="row">
+    <div class="col-12">
+        <div class="card card-grey">
+            <div class="card-block">
+                <h1 class="card-title">Editorial Workflow</h1>
+                <a href="{% url 'submissions:editorial_workflow' %}">How-to guide: summary of the editorial workflow</a>
+            </div>
+        </div>
     </div>
-  </div>
-  {% endif %}
-  <hr/>
-
-  <h3>Refereeing status summary:</h3>
-  {{ submission.refereeing_status_as_p }}
-
-  <h4>Detail of refereeing invitations:</h4>
-  {% if ref_invitations %}
-      <table class="ref_listing">
-	{% for invitation in ref_invitations %}
-	<tr>
-	  {{ invitation.summary_as_tds }}
-	  {% if not invitation.accepted == False and not invitation.cancelled %}
-	  <td>
-	    {% if invitation.referee %}
-	    <a href="{% url 'submissions:communication' arxiv_identifier_w_vn_nr=submission.arxiv_identifier_w_vn_nr comtype='EtoR' referee_id=invitation.referee.id %}">Write a communication</a>
-	    {% else %}
-	    (not yet registered)
-	    {% endif %}
-	  </td>
-	  <td>
-	    {% if not invitation.fulfilled %}
-	    <a href="{% url 'submissions:ref_invitation_reminder' arxiv_identifier_w_vn_nr=submission.arxiv_identifier_w_vn_nr invitation_id=invitation.id %}">Send reminder email</a>
-	    {% else %}
-	    <strong style="color: green">Report has been delivered</strong>
-	    {% endif %}
-	  </td>
-	  <td>
-	    {% if invitation.nr_reminders > 0 %}
-	    (nr reminders sent: {{ invitation.nr_reminders }}, <br/>last on {{ invitation.date_last_reminded }})
-	    {% endif %}
-	  </td>
-	  <td>
-	    {% if not invitation.fulfilled %}
-	    <a href="{% url 'submissions:cancel_ref_invitation' arxiv_identifier_w_vn_nr=submission.arxiv_identifier_w_vn_nr invitation_id=invitation.id %}">Cancel invitation</a>
-	    {% endif %}
-	  </td>
-	  {% else %}
-	  <td></td><td></td><td></td><td></td>
-	  {% endif %}
-	</tr>
-	{% endfor %}
-      </table>
-  {% else %}
-  <p>You have not invited any referees yet.</p>
-  {% endif %}
-
-  <hr class="hr6"/>
-  <h3>Actions:</h3>
-  {% if not submission.is_current %}
-  <h3 style="color: red;">BEWARE: This is not the editorial page for the current version.</h3>
-  <p>The tools here are thus available only for exceptional circumstances (e.g. vetting a late report on a deprecated version).</p>
-  <p>Please go to the current version's page using the link at the top.</p>
-  <br/>
-  {% endif %}
-  <ul>
-    <li><a href="{% url 'submissions:select_referee' arxiv_identifier_w_vn_nr=submission.arxiv_identifier_w_vn_nr %}">
-	Select an additional referee</a> (bear in mind flagged referees if any)</li>
-    <li>Extend the refereeing deadline (currently {{ submission.reporting_deadline|date:'Y-m-d' }}) by
-      <a href="{% url 'submissions:extend_refereeing_deadline' arxiv_identifier_w_vn_nr=submission.arxiv_identifier_w_vn_nr days=2 %}">2 days</a>,
-      <a href="{% url 'submissions:extend_refereeing_deadline' arxiv_identifier_w_vn_nr=submission.arxiv_identifier_w_vn_nr days=7 %}">1 week</a> or
-      <a href="{% url 'submissions:extend_refereeing_deadline' arxiv_identifier_w_vn_nr=submission.arxiv_identifier_w_vn_nr days=14 %}">2 weeks</a>
-      {% if submission.reporting_deadline_has_passed %}
-      <b style="color: red">THE REPORTING DEADLINE HAS PASSED</b>
-      {% endif %}
-      </li>
-    <li>
-      Set refereeing deadline:
-      <form action="{% url 'submissions:set_refereeing_deadline' arxiv_identifier_w_vn_nr=submission.arxiv_identifier_w_vn_nr %}" method="post">
-	{% csrf_token %}{{ set_deadline_form }}
-	<input type="submit" value="Set deadline"/>
-      </form>
-    </li>
-    <li><a href="{% url 'submissions:vet_submitted_reports' %}">Vet submitted Reports</a> ({{ nr_reports_to_vet }})</li>
-    {% if not submission.reporting_deadline_has_passed %}
-    <li><a href="{% url 'submissions:close_refereeing_round' arxiv_identifier_w_vn_nr=submission.arxiv_identifier_w_vn_nr %}">Close the refereeing round</a> &nbsp;(deactivates submission of new Reports and Comments)</li>
-    {% endif %}
-    <li><a href="{% url 'submissions:communication' arxiv_identifier_w_vn_nr=submission.arxiv_identifier_w_vn_nr comtype='EtoA' %}">Draft and send a communication with the submitting Author</a></li>
-    <li>Communicate with a Referee: see the table above.
-    </li>
-    <li><a href="{% url 'submissions:communication' arxiv_identifier_w_vn_nr=submission.arxiv_identifier_w_vn_nr comtype='EtoS' %}">Draft and send a communication with SciPost Editorial Administration</a></li>
-    <li><a href="{% url 'submissions:eic_recommendation' arxiv_identifier_w_vn_nr=submission.arxiv_identifier_w_vn_nr %}">Formulate an Editorial Recommendation</a>
-    <p>If you recommend revisions, this will be communicated directly to the Authors, who will be asked to resubmit.
-      <br/>If you recommend acceptance or rejection, this will be put to the Editorial College for ratification.</p>
-    </li>
-  </ul>
-</section>
-
-<section>
-  <div class="flex-greybox">
-    <h1>Communications</h1>
-  </div>
-  {% if communications %}
-  <ul>
-    {% for comm in communications %}
-    {{ comm.print_contents_as_li }}
-    {% endfor %}
-  </ul>
-  {% else %}
-  <p>There have been no communications for this Submission.</p>
-  {% endif %}
-</section>
-
-
-{% endblock bodysup %}
+</div>
+
+<div class="row">
+    <div class="col-md-12">
+        {% include 'submissions/_submission_status_block.html' with submission=submission %}
+    </div>
+</div>
+
+<div class="row">
+    <div class="col-md-10 col-lg-8">
+        <div class="card {% if submission|required_actions %}card-danger text-white{% else %}card-outline-success text-success{% endif %}">
+            <div class="card-block">
+                <h3 class="card-title pt-0">Required actions:</h3>
+                <ul class="mb-0">
+                    {% for todoitem in submission|required_actions %}
+                        <li>{{ todoitem }}</li>
+                    {% empty %}
+                        <li>No actions required</li>
+                    {% endfor %}
+                </ul>
+            </div>
+        </div>
+    </div>
+</div>
+
+<div class="row">
+    <div class="col-12">
+        <h3>Refereeing status summary:</h3>
+        {% include 'submissions/_submission_refereeing_status.html' with submission=submission %}
+    </div>
+</div>
+
+<div class="row">
+    <div class="col-12">
+        <h3 class="mb-2">Detail of refereeing invitations:</h3>
+        {% include 'submissions/_submission_refereeing_invitations.html' with submission=submission invitations=ref_invitations %}
+    </div>
+</div>
+
+<hr>
+
+{% if not submission.is_current %}
+<div class="row">
+    <div class="col-12">
+        <div class="card card-outline-warning">
+            <div class="card-block">
+                <h3 class="mb-3 font-weight-bold">BEWARE: This is not the editorial page for the current version!</h3>
+                <p class="mb-0">
+                    The tools here are thus available only for exceptional circumstances (e.g. vetting a late report on a deprecated version).
+                    <br>Please go to the current version's page using the link at the top.
+                </p>
+            </div>
+        </div>
+    </div>
+</div>
+{% endif %}
+
+<div class="row">
+    <div class="col-12">
+        <h3>Actions:</h3>
+            <ul>
+                <li>
+                    <a href="{% url 'submissions:select_referee' arxiv_identifier_w_vn_nr=submission.arxiv_identifier_w_vn_nr %}">Select an additional referee</a> (bear in mind flagged referees if any)
+                </li>
+                <li>Extend the refereeing deadline (currently {{ submission.reporting_deadline|date:'Y-m-d' }}) by
+                    <a href="{% url 'submissions:extend_refereeing_deadline' arxiv_identifier_w_vn_nr=submission.arxiv_identifier_w_vn_nr days=2 %}">2 days</a>,
+                    <a href="{% url 'submissions:extend_refereeing_deadline' arxiv_identifier_w_vn_nr=submission.arxiv_identifier_w_vn_nr days=7 %}">1 week</a> or
+                    <a href="{% url 'submissions:extend_refereeing_deadline' arxiv_identifier_w_vn_nr=submission.arxiv_identifier_w_vn_nr days=14 %}">2 weeks</a>
+                    {% if submission.reporting_deadline_has_passed %}
+                        <span class="ml-1 label label-sm label-outline-danger">THE REPORTING DEADLINE HAS PASSED</span>
+                    {% endif %}
+                </li>
+                <li>
+                    Set refereeing deadline:
+                    <form class="form-inline" action="{% url 'submissions:set_refereeing_deadline' arxiv_identifier_w_vn_nr=submission.arxiv_identifier_w_vn_nr %}" method="post">
+                        {% csrf_token %}{{ set_deadline_form|bootstrap_inline:'0,12' }}
+                        <div class="ml-2 form-group row">
+                            <div class="col-12">
+                                <input class="btn btn-secondary" type="submit" value="Set deadline"/>
+                            </div>
+                        </div>
+                    </form>
+                </li>
+                <li><a href="{% url 'submissions:vet_submitted_reports' %}">Vet submitted Reports</a> ({{ nr_reports_to_vet }})</li>
+                {% if not submission.reporting_deadline_has_passed %}
+                    <li><a href="{% url 'submissions:close_refereeing_round' arxiv_identifier_w_vn_nr=submission.arxiv_identifier_w_vn_nr %}">Close the refereeing round</a> &nbsp;(deactivates submission of new Reports and Comments)</li>
+                {% endif %}
+                <li><a href="{% url 'submissions:communication' arxiv_identifier_w_vn_nr=submission.arxiv_identifier_w_vn_nr comtype='EtoA' %}">Draft and send a communication with the submitting Author</a></li>
+                <li>Communicate with a Referee: see the table above.</li>
+                <li><a href="{% url 'submissions:communication' arxiv_identifier_w_vn_nr=submission.arxiv_identifier_w_vn_nr comtype='EtoS' %}">Draft and send a communication with SciPost Editorial Administration</a></li>
+                <li>
+                    <a href="{% url 'submissions:eic_recommendation' arxiv_identifier_w_vn_nr=submission.arxiv_identifier_w_vn_nr %}">Formulate an Editorial Recommendation</a>
+                    <p>If you recommend revisions, this will be communicated directly to the Authors, who will be asked to resubmit.
+                    <br/>If you recommend acceptance or rejection, this will be put to the Editorial College for ratification.</p>
+                </li>
+          </ul>
+    </div>
+</div>
+
+<div class="row">
+    <div class="col-12">
+        <h1 class="highlight">Communications</h1>
+    </div>
+</div>
+
+<div class="row">
+    <div class="col-lg-6 offset-lg-3 col-md-8 offset-md-2">
+        <ul class="list-group list-group-flush">
+            {% for comm in communications %}
+                <li class="list-group-item">
+                    {% include 'submissions/_editorial_communication_content.html' with communication=comm %}
+                </li>
+            {% empty %}
+                <li class="list-group-item">There have been no communications for this Submission.</li>
+            {% endfor %}
+        </ul>
+    </div>
+</div>
+
+{% endblock content %}
diff --git a/submissions/templates/submissions/eic_recommendation.html b/submissions/templates/submissions/eic_recommendation.html
index 135a636cc4ebbf643a53fad4fa01767bec0bb9d8..c692faa02da942295b07e82cbca56b835c89fdda 100644
--- a/submissions/templates/submissions/eic_recommendation.html
+++ b/submissions/templates/submissions/eic_recommendation.html
@@ -51,7 +51,7 @@
       <h2>Submission:</h2>
     </div>
   </div>
-  {{ submission.header_as_table }}
+  {% include 'submissions/_submission_summary.html' with submission=submission %}
   <h3>Abstract:</h3>
   <p>{{ submission.abstract }}</p>
 
diff --git a/submissions/templates/submissions/pool.html b/submissions/templates/submissions/pool.html
index 4c505ce342baad574f12648e53de04a411a4284d..5cc698d47945e9c2a56e925a49279b00f0b4b18b 100644
--- a/submissions/templates/submissions/pool.html
+++ b/submissions/templates/submissions/pool.html
@@ -41,7 +41,6 @@
                         <h3>Editorial recommendation</h3>
                     </div>
                     <div class="card-block">
-                        {# {{ rec.print_for_Fellows }}#}
                         <div class="card card-outline-secondary">
                             {% include 'submissions/_recommendation_fellow_content.html' with recommendation=rec %}
                         </div>
@@ -154,7 +153,7 @@
                         <h3>Editorial recommendation</h3>
                     </div>
                     <div class="card-block">
-                        {{ rec.print_for_Fellows }}
+                        {% include 'submissions/_recommendation_fellow_content.html' with recommendation=rec %}
                     </div>
                     <div class="card-footer">
                         <h3>Actions:</h3>
@@ -191,7 +190,7 @@
                         <h3>Editorial recommendation</h3>
                     </div>
                     <div class="card-block">
-                        {{ rec.print_for_Fellows }}
+                        {% include 'submissions/_recommendation_fellow_content.html' with recommendation=rec %}
                     </div>
                     <div class="card-footer">
                         <h3>Your position on this recommendation</h3>
diff --git a/submissions/templates/submissions/prepare_for_voting.html b/submissions/templates/submissions/prepare_for_voting.html
index dcd1a2e686a8350d33e18ac302784473f1aad7ca..bb9b8af8999559b01a46ce8fd115e0fd08feb7a5 100644
--- a/submissions/templates/submissions/prepare_for_voting.html
+++ b/submissions/templates/submissions/prepare_for_voting.html
@@ -34,7 +34,9 @@
   <div class="flex-greybox">
     <h3>Editorial recommendation:</h3>
     <ul>
-      <li>{{ recommendation.print_for_Fellows }}</li>
+      <li>
+          {% include 'submissions/_recommendation_fellow_content.html' with recommendation=recommendation %}
+      </li>
     </ul>
   </div>
 
diff --git a/submissions/templates/submissions/submission_detail.html b/submissions/templates/submissions/submission_detail.html
index 15a64f60c15819593496a5c3a0f5a9efc7a77d3a..935acfd7a48ebae50cb4928556263aea5b10f398 100644
--- a/submissions/templates/submissions/submission_detail.html
+++ b/submissions/templates/submissions/submission_detail.html
@@ -38,14 +38,16 @@
 
 <div class="row">
     <div class="col-12">
-        <div class="panel">
-            <h1>SciPost Submission Page</h1>
-            {% if not submission.is_current %}
-                <h3 style="color: red;">This is not the current version.</h3>
-            {% endif %}
-            {% if user.contributor == submission.editor_in_charge %}
-                <h3>(You are the Editor-in-charge, go to the <a href="{% url 'submissions:editorial_page' arxiv_identifier_w_vn_nr=submission.arxiv_identifier_w_vn_nr %}">Editorial Page</a> to take editorial actions)</h3>
-            {% endif %}
+        <div class="card card-grey">
+            <div class="card-block">
+                <h1>SciPost Submission Page</h1>
+                {% if not submission.is_current %}
+                    <h3 style="color: red;">This is not the current version.</h3>
+                {% endif %}
+                {% if user.contributor == submission.editor_in_charge %}
+                    <h3>(You are the Editor-in-charge, go to the <a href="{% url 'submissions:editorial_page' arxiv_identifier_w_vn_nr=submission.arxiv_identifier_w_vn_nr %}">Editorial Page</a> to take editorial actions)</h3>
+                {% endif %}
+            </div>
         </div>
     </div>
 </div>
@@ -53,26 +55,21 @@
 {% if other_versions %}
 <div class="row">
     <div class="col-12">
-      <h3>Other versions of this Submission (with Reports) exist:</h3>
-      <ul>
-        {% for vn in other_versions %}
-            {{ vn.version_info_as_li }}
-        {% endfor %}
-      </ul>
+        <h3>Other versions of this Submission (with Reports) exist:</h3>
+        <div class="pl-4">
+            {% for vn in other_versions %}
+                {% include 'submissions/_submission_version.html' with submission=vn %}
+            {% endfor %}
+        </div>
     </div>
 </div>
 {% endif %}
 
 <div class="row">
     <div class="col-12">
-        <hr>
-        {{ submission.header_as_table }}
+        {% include 'submissions/_submission_summary.html' with submission=submission %}
 
-        <h3 class="mt-3">Abstract:</h3>
-        <p>{{ submission.abstract }}</p>
-
-        <h3>Status:</h3>
-        {{ submission.status_info_as_table }}
+        {% include 'submissions/_submission_status_block.html' with submission=submission %}
 
         {% if submission.author_comments %}
             <h3>Author comments upon resubmission</h3>
@@ -87,16 +84,17 @@
 </div>
 
 {% if is_author or user|is_in_group:'Editorial College' or user|is_in_group:'Editorial Administrators' %}
-{% if recommendation %}
-{% if user|is_in_group:'Editorial College' or user|is_in_group:'Editorial Administrators' or recommendation|is_viewable_by_authors %}
-<div class="row">
-    <div class="col-12">
-        <h2>Editorial Recommendation</h2>
-        {{ recommendation.print_for_authors }}
-    </div>
-</div>
-{% endif %}
-{% endif %}
+    {% if recommendation %}
+        {% if user|is_in_group:'Editorial College' or user|is_in_group:'Editorial Administrators' or recommendation|is_viewable_by_authors %}
+        <hr>
+            <div class="row">
+                <div class="col-12">
+                    <h2>Editorial Recommendation</h2>
+                    {% include 'submissions/_recommendation_author_content.html' with recommendation=recommendation %}
+                </div>
+            </div>
+        {% endif %}
+    {% endif %}
 {% endif %}
 
 
@@ -113,12 +111,17 @@
         <ul>
             {% if submission.open_for_reporting %}
                 {% if perms.scipost.can_referee and not is_author and not is_author_unchecked %}
-                    <li><h3><a href="{% url 'submissions:submit_report' arxiv_identifier_w_vn_nr=submission.arxiv_identifier_w_vn_nr %}">Contribute a Report</a></h3>
-                    <div class="reportingDeadline">Deadline for reporting: {{ submission.reporting_deadline|date:"Y-m-d" }}</div></li>
+                    <li>
+                        <h3><a href="{% url 'submissions:submit_report' arxiv_identifier_w_vn_nr=submission.arxiv_identifier_w_vn_nr %}">Contribute a Report</a></h3>
+                        <div class="text-danger">Deadline for reporting: {{ submission.reporting_deadline|date:"Y-m-d" }}</div>
+                    </li>
                 {% elif is_author_unchecked %}
-                    <li><h3>Contribute a Report [deactivated]: the system flagged you as a potential author of this Submission.
-                    Please go to your <a href="{% url 'scipost:personal_page' %}">personal page</a>
-                    under the Submissions tab to clarify this.</h3></li>
+                    <li>
+                        <h3 class="text-blue">Contribute a Report [deactivated]</h3>
+                        <div>The system flagged you as a potential author of this Submission.
+                        Please go to your <a href="{% url 'scipost:personal_page' %}">personal page</a>
+                        under the Submissions tab to clarify this.</div>
+                    </li>
                 {% endif %}
             {% else %}
                 <li>Reporting for this Submission is closed.</li>
@@ -150,7 +153,7 @@
 
 <div id="invitedreportslist">
     {% for report in invited_reports %}
-        {% include 'submissions/_single_report.html' with report=report user=request.user perms=perms %}
+        {% include 'submissions/_single_public_report.html' with report=report user=request.user perms=perms %}
     {% endfor %}
 </div>
 
@@ -171,7 +174,7 @@
 
 <div id="contributedreportslist">
     {% for report in contributed_reports %}
-        {% include 'submissions/_single_report.html' with report=report user=request.user perms=perms %}
+        {% include 'submissions/_single_public_report.html' with report=report user=request.user perms=perms %}
     {% endfor %}
 </div>
 
diff --git a/submissions/templates/submissions/submit_report.html b/submissions/templates/submissions/submit_report.html
index fad83a5aa691770c3437a6f8df26caa3da55b786..d47385506a57446fc570f74ab22a4b0eca855a8d 100644
--- a/submissions/templates/submissions/submit_report.html
+++ b/submissions/templates/submissions/submit_report.html
@@ -65,10 +65,8 @@
   <div class="flex-greybox">
     <h2>Submission summary and link</h2>
   </div>
-  {{ submission.header_as_table }}
-  <br />
-  <h3>Abstract:</h3>
-  <p>{{ submission.abstract }}</p>
+  {% include 'submissions/_submission_summary.html' with submission=submission %}
+
 
 </section>
 
diff --git a/submissions/templates/submissions/vet_submitted_reports.html b/submissions/templates/submissions/vet_submitted_reports.html
index d88bdeae77764c5e05b341a0f7ddc2865cdaa999..d545a3b09ebd6488f70b8ce3632a806e9b6c2c98 100644
--- a/submissions/templates/submissions/vet_submitted_reports.html
+++ b/submissions/templates/submissions/vet_submitted_reports.html
@@ -2,6 +2,8 @@
 
 {% block pagetitle %}: vet reports{% endblock pagetitle %}
 
+{% load bootstrap %}
+
 {% block headsup %}
 
 <script>
@@ -9,7 +11,7 @@ $(document).ready(function(){
 
   $('#refusal').hide();
 
-  $('#id_action_option').on('change', function() {
+  $('[name="action_option"]').on('change', function() {
       if ($('#id_action_option_1').is(':checked')) {
           $('#refusal').show();
       }
@@ -22,46 +24,40 @@ $(document).ready(function(){
 
 {% endblock headsup %}
 
-{% block bodysup %}
-
-<section>
-  {% if not report_to_vet %}
-  <h1>There are no Reports for you to vet.</h1>
-
-  {% else %}
-
-  <h1>SciPost Report to vet:</h1>
-
-  <br>
-  <hr>
-  <h3>Submission associated to Report:</h3>
-  {{ report_to_vet.submission.header_as_table }}
-  <br />
-  <h4>Abstract:</h4>
-  <p>{{ report_to_vet.submission.abstract }}</p>
-  <hr style="border-style: dotted;" />
-  <div class="flex-greybox">
-    <h3>Report to vet:</h3>
-  </div>
-  {{ report_to_vet.print_contents_for_editors|linebreaks }}
-
-  <hr class="hr6"/>
-  <div class="flex-greaybox">
-    <h2>Please vet this Report:</h2>
-  </div>
-  <form action="{% url 'submissions:vet_submitted_report_ack' report_id=report_to_vet.id %}" method="post">
-    {% csrf_token %}
-    {{ form.action_option }}
-    <div id="refusal">
-      <ul>
-	<li>Refusal reason: {{ form.refusal_reason }}</li>
-	<li>{{ form.email_response_field }}</li>
-      </ul>
+{% block content %}
+
+<div class="row">
+    <div class="col-12">
+        {% if not report_to_vet %}
+            <h1>There are no Reports for you to vet.</h1>
+        {% else %}
+            <h1 class="highlight">SciPost Report to vet:</h1>
+
+            <h2 class="mb-2">Submission associated to Report:</h2>
+            <div class="row">
+                <div class="col-12">
+                    {% include 'submissions/_submission_summary_short.html' with submission=report_to_vet.submission %}
+                </div>
+            </div>
+
+            <h2 class="mb-2">Report to vet:</h2>
+
+            {% include 'submissions/_single_public_report_without_comments.html' with report=report_to_vet %}
+
+            <hr class="small">
+            <h2>Please vet this Report:</h2>
+            <form action="{% url 'submissions:vet_submitted_report_ack' report_id=report_to_vet.id %}" method="post">
+                {% csrf_token %}
+                {{ form.action_option|bootstrap }}
+                <div class="col-md-6" id="refusal">
+                    {{ form.refusal_reason|bootstrap }}
+                    {{ form.email_response_field|bootstrap }}
+                </div>
+                <input class="btn btn-secondary" type="submit" value="Submit" />
+            </form>
+
+        {% endif %}
     </div>
-    <input type="submit" value="Submit" />
-  </form>
-
-  {% endif %}
-</section>
+</div>
 
-{% endblock bodysup %}
+{% endblock content %}
diff --git a/submissions/templatetags/submissions_extras.py b/submissions/templatetags/submissions_extras.py
index b2b785c0357b7be546be77fe2a3eb0d1761022da..2da991f6d3ce3c33c022055bee26bbbbc5d7c597 100644
--- a/submissions/templatetags/submissions_extras.py
+++ b/submissions/templatetags/submissions_extras.py
@@ -2,14 +2,13 @@ import datetime
 
 from django import template
 from django.utils import timezone
-from django.utils.timesince import timesince
-from urllib.parse import urlencode
 
-from submissions.models import SUBMISSION_STATUS_OUT_OF_POOL
+from submissions.constants import SUBMISSION_STATUS_OUT_OF_POOL
 from submissions.models import Submission
 
 register = template.Library()
 
+
 @register.filter(name='is_not_author_of_submission')
 def is_not_author_of_submission(user, arxiv_identifier_w_vn_nr):
     submission = Submission.objects.get(arxiv_identifier_w_vn_nr=arxiv_identifier_w_vn_nr)
@@ -17,7 +16,7 @@ def is_not_author_of_submission(user, arxiv_identifier_w_vn_nr):
             and
             (user.last_name not in submission.author_list
              or
-             user.contributor in submission.authors_false_claims.all() ) )
+             user.contributor in submission.authors_false_claims.all()))
 
 
 @register.filter(name='is_viewable_by_authors')
@@ -26,6 +25,7 @@ def is_viewable_by_authors(recommendation):
                                                 'accepted', 'rejected',
                                                 'published', 'withdrawn']
 
+
 @register.filter(name='required_actions')
 def required_actions(submission):
     """
diff --git a/submissions/utils.py b/submissions/utils.py
index 18fcb7e14d6c46f5df576fe95e17cbb281ea959c..5bfdcdcba3390e9402c41205387a86bb73d48d23 100644
--- a/submissions/utils.py
+++ b/submissions/utils.py
@@ -3,12 +3,9 @@ import datetime
 from django.core.mail import EmailMessage, EmailMultiAlternatives
 from django.template import Context, Template
 
-from journals.models import journals_submit_dict
-from scipost.models import title_dict
 from scipost.utils import EMAIL_FOOTER
 
 from submissions.models import EditorialAssignment
-from submissions.models import assignment_refusal_reasons_dict, ed_comm_choices_dict
 from submissions.forms import report_refusal_choices_dict
 
 
@@ -46,7 +43,7 @@ class SubmissionUtils(object):
     @classmethod
     def send_authors_submission_ack_email(cls):
         """ Requires loading 'submission' attribute. """
-        email_text = ('Dear ' + title_dict[cls.submission.submitted_by.title] + ' ' +
+        email_text = ('Dear ' + cls.submission.submitted_by.get_title_display() + ' ' +
                       cls.submission.submitted_by.user.last_name +
                       ', \n\nWe have received your Submission to SciPost,\n\n' +
                       cls.submission.title + ' by ' + cls.submission.author_list + '.' +
@@ -68,7 +65,7 @@ class SubmissionUtils(object):
             '<p>With many thanks,</p>'
             '<p>The SciPost Team.</p>')
         email_context = Context({
-            'title': title_dict[cls.submission.submitted_by.title],
+            'title': cls.submission.submitted_by.get_title_display(),
             'last_name': cls.submission.submitted_by.user.last_name,
             'sub_title': cls.submission.title,
             'author_list': cls.submission.author_list,
@@ -88,7 +85,7 @@ class SubmissionUtils(object):
     @classmethod
     def send_authors_resubmission_ack_email(cls):
         """ Requires loading 'submission' attribute. """
-        email_text = ('Dear ' + title_dict[cls.submission.submitted_by.title] + ' ' +
+        email_text = ('Dear ' + cls.submission.submitted_by.get_title_display() + ' ' +
                       cls.submission.submitted_by.user.last_name +
                       ', \n\nWe have received your Resubmission to SciPost,\n\n' +
                       cls.submission.title + ' by ' + cls.submission.author_list + '.' +
@@ -107,7 +104,7 @@ class SubmissionUtils(object):
             '<p>With many thanks,</p>'
             '<p>The SciPost Team</p>')
         email_context = Context({
-            'title': title_dict[cls.submission.submitted_by.title],
+            'title': cls.submission.submitted_by.get_title_display(),
             'last_name': cls.submission.submitted_by.user.last_name,
             'sub_title': cls.submission.title,
             'author_list': cls.submission.author_list,
@@ -127,7 +124,7 @@ class SubmissionUtils(object):
     @classmethod
     def send_assignment_request_email(cls):
         """ Requires loading 'assignment' attribute. """
-        email_text = ('Dear ' + title_dict[cls.assignment.to.title] + ' ' +
+        email_text = ('Dear ' + cls.assignment.to.get_title_display() + ' ' +
                       cls.assignment.to.user.last_name +
                       ', \n\nWe have received a Submission to SciPost ' +
                       'for which we would like you to consider becoming Editor-in-charge:\n\n' +
@@ -160,7 +157,7 @@ class SubmissionUtils(object):
             '\n<p>Many thanks in advance for your collaboration,</p>'
             '<p>The SciPost Team.</p>')
         email_context = Context({
-            'title': title_dict[cls.assignment.to.title],
+            'title': cls.assignment.to.get_title_display(),
             'last_name': cls.assignment.to.user.last_name,
             'sub_title': cls.assignment.submission.title,
             'author_list': cls.assignment.submission.author_list,
@@ -180,7 +177,7 @@ class SubmissionUtils(object):
     @classmethod
     def send_EIC_appointment_email(cls):
         """ Requires loading 'assignment' attribute. """
-        email_text = ('Dear ' + title_dict[cls.assignment.to.title] + ' '
+        email_text = ('Dear ' + cls.assignment.to.get_title_display() + ' '
                       + cls.assignment.to.user.last_name
                       + ', \n\nThank you for accepting to become Editor-in-charge '
                       'of the SciPost Submission\n\n'
@@ -216,7 +213,7 @@ class SubmissionUtils(object):
             '<p>Many thanks in advance for your collaboration,</p>'
             '<p>The SciPost Team.</p>')
         email_context = Context({
-            'title': title_dict[cls.assignment.to.title],
+            'title': cls.assignment.to.get_title_display(),
             'last_name': cls.assignment.to.user.last_name,
             'sub_title': cls.assignment.submission.title,
             'author_list': cls.assignment.submission.author_list,
@@ -237,7 +234,7 @@ class SubmissionUtils(object):
     @classmethod
     def send_EIC_reappointment_email(cls):
         """ Requires loading 'submission' attribute. """
-        email_text = ('Dear ' + title_dict[cls.submission.editor_in_charge.title] + ' '
+        email_text = ('Dear ' + cls.submission.editor_in_charge.get_title_display() + ' '
                       + cls.submission.editor_in_charge.user.last_name
                       + ', \n\nThe authors of the SciPost Submission\n\n'
                       + cls.submission.title + ' by '
@@ -278,7 +275,7 @@ class SubmissionUtils(object):
             '<p>Many thanks in advance for your collaboration,</p>'
             '<p>The SciPost Team.</p>')
         email_context = Context({
-            'title': title_dict[cls.submission.editor_in_charge.title],
+            'title': cls.submission.editor_in_charge.get_title_display(),
             'last_name': cls.submission.editor_in_charge.user.last_name,
             'sub_title': cls.submission.title,
             'author_list': cls.submission.author_list,
@@ -299,7 +296,7 @@ class SubmissionUtils(object):
     @classmethod
     def send_author_prescreening_passed_email(cls):
         """ Requires loading 'assignment' attribute. """
-        email_text = ('Dear ' + title_dict[cls.assignment.submission.submitted_by.title] + ' '
+        email_text = ('Dear ' + cls.assignment.submission.submitted_by.get_title_display() + ' '
                       + cls.assignment.submission.submitted_by.user.last_name
                       + ', \n\nWe are pleased to inform you that your recent Submission to SciPost,\n\n'
                       + cls.assignment.submission.title + ' by ' + cls.assignment.submission.author_list
@@ -354,7 +351,7 @@ class SubmissionUtils(object):
             '<p>Sincerely,</p>'
             '<p>The SciPost Team.</p>')
         email_context = Context({
-            'title': title_dict[cls.assignment.submission.submitted_by.title],
+            'title': cls.assignment.submission.submitted_by.get_title_display(),
             'last_name': cls.assignment.submission.submitted_by.user.last_name,
             'sub_title': cls.assignment.submission.title,
             'author_list': cls.assignment.submission.author_list,
@@ -377,7 +374,7 @@ class SubmissionUtils(object):
     @classmethod
     def assignment_failed_email_authors(cls):
         """ Requires loading 'submission' attribute. """
-        email_text = ('Dear ' + title_dict[cls.submission.submitted_by.title] + ' '
+        email_text = ('Dear ' + cls.submission.submitted_by.get_title_display() + ' '
                       + cls.submission.submitted_by.user.last_name
                       + ', \n\nYou recent Submission to SciPost,\n\n'
                       + cls.submission.title + ' by ' + cls.submission.author_list
@@ -406,7 +403,7 @@ class SubmissionUtils(object):
             '<p>Sincerely,</p>'
             '<p>The SciPost Team.</p>')
         email_context = Context({
-            'title': title_dict[cls.submission.submitted_by.title],
+            'title': cls.submission.submitted_by.get_title_display(),
             'last_name': cls.submission.submitted_by.user.last_name,
             'sub_title': cls.submission.title,
             'author_list': cls.submission.author_list,
@@ -433,11 +430,11 @@ class SubmissionUtils(object):
         instead, which calls the send_registration_email method in scipost/utils.
         Requires loading 'invitation' attribute.
         """
-        email_text = ('Dear ' + title_dict[cls.invitation.referee.title] + ' ' +
+        email_text = ('Dear ' + cls.invitation.referee.get_title_display() + ' ' +
                       cls.invitation.referee.user.last_name +
                       ', \n\nWe have received a Submission to SciPost '
                       'which, in view of your expertise and on behalf of the Editor-in-charge '
-                      + title_dict[cls.invitation.submission.editor_in_charge.title] + ' '
+                      + cls.invitation.submission.editor_in_charge.get_title_display() + ' '
                       + cls.invitation.submission.editor_in_charge.user.last_name
                       + ', we would like to invite you to referee:\n\n'
                       + cls.invitation.submission.title + ' by ' + cls.invitation.submission.author_list
@@ -486,9 +483,9 @@ class SubmissionUtils(object):
             'and thank you in advance for your consideration.</p>'
             '<p>The SciPost Team.</p>')
         email_context = Context({
-            'title': title_dict[cls.invitation.referee.title],
+            'title': cls.invitation.referee.get_title_display(),
             'last_name': cls.invitation.referee.user.last_name,
-            'EIC_title': title_dict[cls.invitation.submission.editor_in_charge.title],
+            'EIC_title': cls.invitation.submission.editor_in_charge.get_title_display(),
             'EIC_last_name': cls.invitation.submission.editor_in_charge.user.last_name,
             'sub_title': cls.invitation.submission.title,
             'author_list': cls.invitation.submission.author_list,
@@ -515,10 +512,10 @@ class SubmissionUtils(object):
         This method is used to remind a referee who is not registered as a Contributor.
         It is called from the ref_invitation_reminder method in submissions/views.py.
         """
-        email_text = ('Dear ' + title_dict[cls.invitation.title] + ' '
+        email_text = ('Dear ' + cls.invitation.get_title_display() + ' '
                       + cls.invitation.last_name + ',\n\n'
                       'On behalf of the Editor-in-charge '
-                      + title_dict[cls.invitation.submission.editor_in_charge.title] + ' '
+                      + cls.invitation.submission.editor_in_charge.get_title_display() + ' '
                       + cls.invitation.submission.editor_in_charge.user.last_name
                       + ', we would like to cordially remind you of our recent request to referee\n\n'
                       + cls.invitation.submission.title + ' by '
@@ -584,9 +581,9 @@ class SubmissionUtils(object):
             '<p>Many thanks in advance,</p>'
             '<p>The SciPost Team</p>')
         email_context = Context({
-            'title': title_dict[cls.invitation.title],
+            'title': cls.invitation.get_title_display(),
             'last_name': cls.invitation.last_name,
-            'EIC_title': title_dict[cls.invitation.submission.editor_in_charge.title],
+            'EIC_title': cls.invitation.submission.editor_in_charge.get_title_display(),
             'EIC_last_name': cls.invitation.submission.editor_in_charge.user.last_name,
             'sub_title': cls.invitation.submission.title,
             'author_list': cls.invitation.submission.author_list,
@@ -614,10 +611,10 @@ class SubmissionUtils(object):
         This method is used to inform a referee that his/her services are no longer required.
         It is called from the cancel_ref_invitation method in submissions/views.py.
         """
-        email_text = ('Dear ' + title_dict[cls.invitation.title] + ' '
+        email_text = ('Dear ' + cls.invitation.get_title_display() + ' '
                       + cls.invitation.last_name + ',\n\n'
                       'On behalf of the Editor-in-charge '
-                      + title_dict[cls.invitation.submission.editor_in_charge.title] + ' '
+                      + cls.invitation.submission.editor_in_charge.get_title_display() + ' '
                       + cls.invitation.submission.editor_in_charge.user.last_name
                       + ', we would like to inform you that your report on\n\n'
                       + cls.invitation.submission.title + ' by '
@@ -655,9 +652,9 @@ class SubmissionUtils(object):
                 'after which your registration will be activated, giving you full access to '
                 'the portal\'s facilities (in particular allowing you to provide referee reports).</p>')
         email_context = Context({
-            'title': title_dict[cls.invitation.title],
+            'title': cls.invitation.get_title_display(),
             'last_name': cls.invitation.last_name,
-            'EIC_title': title_dict[cls.invitation.submission.editor_in_charge.title],
+            'EIC_title': cls.invitation.submission.editor_in_charge.get_title_display(),
             'EIC_last_name': cls.invitation.submission.editor_in_charge.user.last_name,
             'sub_title': cls.invitation.submission.title,
             'author_list': cls.invitation.submission.author_list,
@@ -679,9 +676,9 @@ class SubmissionUtils(object):
     @classmethod
     def email_referee_response_to_EIC(cls):
         """ Requires loading 'invitation' attribute. """
-        email_text = ('Dear ' + title_dict[cls.invitation.submission.editor_in_charge.title] + ' ' +
+        email_text = ('Dear ' + cls.invitation.submission.editor_in_charge.get_title_display() + ' ' +
                       cls.invitation.submission.editor_in_charge.user.last_name + ','
-                      '\n\nReferee ' + title_dict[cls.invitation.referee.title] + ' ' +
+                      '\n\nReferee ' + cls.invitation.referee.get_title_display() + ' ' +
                       cls.invitation.referee.user.last_name + ' has ')
         email_text_html = (
             '<p>Dear {{ EIC_title }} {{ EIC_last_name }},</p>'
@@ -693,7 +690,7 @@ class SubmissionUtils(object):
             email_subject = 'SciPost: referee accepts to review'
         elif not cls.invitation.accepted:
             email_text += ('declined (due to reason: '
-                           + assignment_refusal_reasons_dict[cls.invitation.refusal_reason] + ') ')
+                           + cls.invitation.get_refusal_reason_display() + ') ')
             email_text_html += 'declined (due to reason: {{ reason }}) '
 
         email_text += ('to referee Submission\n\n'
@@ -715,16 +712,16 @@ class SubmissionUtils(object):
         email_text_html += ('<p>Many thanks for your collaboration,</p>'
                             '<p>The SciPost Team.</p>')
         email_context = Context({
-            'EIC_title': title_dict[cls.invitation.submission.editor_in_charge.title],
+            'EIC_title': cls.invitation.submission.editor_in_charge.get_title_display(),
             'EIC_last_name': cls.invitation.submission.editor_in_charge.user.last_name,
-            'ref_title': title_dict[cls.invitation.referee.title],
+            'ref_title': cls.invitation.referee.get_title_display(),
             'ref_last_name': cls.invitation.referee.user.last_name,
             'sub_title': cls.invitation.submission.title,
             'author_list': cls.invitation.submission.author_list,
             'arxiv_identifier_w_vn_nr': cls.invitation.submission.arxiv_identifier_w_vn_nr,
         })
         if cls.invitation.refusal_reason:
-            email_context['reason'] = assignment_refusal_reasons_dict[cls.invitation.refusal_reason]
+            email_context['reason'] = cls.invitation.get_refusal_reason_display
         email_text_html += '<br/>' + EMAIL_FOOTER
         html_template = Template(email_text_html)
         html_version = html_template.render(email_context)
@@ -740,9 +737,9 @@ class SubmissionUtils(object):
     @classmethod
     def email_EIC_report_delivered(cls):
         """ Requires loading 'report' attribute. """
-        email_text = ('Dear ' + title_dict[cls.report.submission.editor_in_charge.title] + ' '
+        email_text = ('Dear ' + cls.report.submission.editor_in_charge.get_title_display() + ' '
                       + cls.report.submission.editor_in_charge.user.last_name + ','
-                      '\n\nReferee ' + title_dict[cls.report.author.title] + ' '
+                      '\n\nReferee ' + cls.report.author.get_title_display() + ' '
                       + cls.report.author.user.last_name +
                       ' has delivered a Report for Submission\n\n'
                       + cls.report.submission.title + ' by '
@@ -762,9 +759,9 @@ class SubmissionUtils(object):
             '<p>Many thanks for your collaboration,</p>'
             '<p>The SciPost Team.</p>')
         email_context = Context({
-            'EIC_title': title_dict[cls.report.submission.editor_in_charge.title],
+            'EIC_title': cls.report.submission.editor_in_charge.get_title_display(),
             'EIC_last_name': cls.report.submission.editor_in_charge.user.last_name,
-            'ref_title': title_dict[cls.report.author.title],
+            'ref_title': cls.report.author.get_title_display(),
             'ref_last_name': cls.report.author.user.last_name,
             'sub_title': cls.report.submission.title,
             'author_list': cls.report.submission.author_list,
@@ -784,7 +781,7 @@ class SubmissionUtils(object):
     @classmethod
     def acknowledge_report_email(cls):
         """ Requires loading 'report' attribute. """
-        email_text = ('Dear ' + title_dict[cls.report.author.title] + ' ' +
+        email_text = ('Dear ' + cls.report.author.get_title_display() + ' ' +
                       cls.report.author.user.last_name + ','
                       '\n\nMany thanks for your Report on Submission\n\n' +
                       cls.report.submission.title + ' by '
@@ -837,7 +834,7 @@ class SubmissionUtils(object):
                 '\n<strong>Requested changes</strong>: <br/><p>{{ requested_changes|linebreaks }}</p>'
                 '\n<strong>Remarks for Editors</strong>: <br/><p>{{ remarks_for_editors|linebreaks }}</p>')
         email_context = Context({
-            'ref_title': title_dict[cls.report.author.title],
+            'ref_title': cls.report.author.get_title_display(),
             'ref_last_name': cls.report.author.user.last_name,
             'sub_title': cls.report.submission.title,
             'author_list': cls.report.submission.author_list,
@@ -866,7 +863,7 @@ class SubmissionUtils(object):
     @classmethod
     def send_author_report_received_email(cls):
         """ Requires loading 'report' attribute. """
-        email_text = ('Dear ' + title_dict[cls.report.submission.submitted_by.title] + ' ' +
+        email_text = ('Dear ' + cls.report.submission.submitted_by.get_title_display() + ' ' +
                       cls.report.submission.submitted_by.user.last_name +
                       ', \n\nA Report has been posted on your recent Submission to SciPost,\n\n' +
                       cls.report.submission.title + ' by ' + cls.report.submission.author_list + '.'
@@ -897,7 +894,7 @@ class SubmissionUtils(object):
             '<p>Sincerely,</p>'
             '<p>The SciPost Team.</p>')
         email_context = Context({
-            'auth_title': title_dict[cls.report.submission.submitted_by.title],
+            'auth_title': cls.report.submission.submitted_by.get_title_display(),
             'auth_last_name': cls.report.submission.submitted_by.user.last_name,
             'sub_title': cls.report.submission.title,
             'author_list': cls.report.submission.author_list,
@@ -918,7 +915,7 @@ class SubmissionUtils(object):
     @classmethod
     def send_author_comment_received_email(cls):
         """ Requires loading 'submission' attribute. """
-        email_text = ('Dear ' + title_dict[cls.submission.submitted_by.title] + ' ' +
+        email_text = ('Dear ' + cls.submission.submitted_by.get_title_display() + ' ' +
                       cls.submission.submitted_by.user.last_name +
                       ', \n\nA Comment has been posted on your recent Submission to SciPost,\n\n' +
                       cls.submission.title + ' by ' + cls.submission.author_list + '.'
@@ -939,7 +936,7 @@ class SubmissionUtils(object):
             '<p>Sincerely,</p>'
             '<p>The SciPost Team.</p>')
         email_context = Context({
-            'auth_title': title_dict[cls.submission.submitted_by.title],
+            'auth_title': cls.submission.submitted_by.get_title_display(),
             'auth_last_name': cls.submission.submitted_by.user.last_name,
             'sub_title': cls.submission.title,
             'author_list': cls.submission.author_list,
@@ -971,7 +968,7 @@ class SubmissionUtils(object):
         if cls.communication.comtype in ['AtoE', 'RtoE', 'StoE']:
             recipient_email.append(cls.communication.submission.editor_in_charge.user.email)
             recipient_greeting = ('Dear ' +
-                                  title_dict[cls.communication.submission.editor_in_charge.title] + ' ' +
+                                  cls.communication.submission.editor_in_charge.get_title_display() + ' ' +
                                   cls.communication.submission.editor_in_charge.user.last_name)
             further_action_page = ('https://scipost.org/submission/editorial_page/'
                                    + cls.communication.submission.arxiv_identifier_w_vn_nr)
@@ -981,14 +978,14 @@ class SubmissionUtils(object):
         elif cls.communication.comtype in ['EtoA']:
             recipient_email.append(cls.communication.submission.submitted_by.user.email)
             recipient_greeting = ('Dear ' +
-                                  title_dict[cls.communication.submission.submitted_by.title] + ' ' +
+                                  cls.communication.submission.submitted_by.get_title_display() + ' ' +
                                   cls.communication.submission.submitted_by.user.last_name)
             bcc_emails.append(cls.communication.submission.editor_in_charge)
             bcc_emails.append('submissions@scipost.org')
         elif cls.communication.comtype in ['EtoR']:
             recipient_email.append(cls.communication.referee.user.email)
             recipient_greeting = ('Dear ' +
-                                  title_dict[cls.communication.referee.title] + ' ' +
+                                  cls.communication.referee.get_title_display() + ' ' +
                                   cls.communication.referee.user.last_name)
             bcc_emails.append(cls.communication.submission.editor_in_charge)
             bcc_emails.append('submissions@scipost.org')
@@ -1000,7 +997,7 @@ class SubmissionUtils(object):
 
         email_text = (recipient_greeting +
                       ', \n\nPlease find here a communication (' +
-                      ed_comm_choices_dict[cls.communication.comtype] + ') '
+                      cls.communication.get_comtype_display() + ') '
                       'concerning Submission\n\n' +
                       cls.communication.submission.title + ' by '
                       + cls.communication.submission.author_list + '.'
@@ -1012,7 +1009,7 @@ class SubmissionUtils(object):
                        '\n\nSincerely,' +
                        '\n\nThe SciPost Team.')
         emailmessage = EmailMessage(
-            'SciPost: communication (' + ed_comm_choices_dict[cls.communication.comtype] + ')',
+            'SciPost: communication (' + cls.communication.get_comtype_display() + ')',
             email_text,
             'SciPost Editorial Admin <submissions@scipost.org>',
             recipient_email,
@@ -1023,7 +1020,7 @@ class SubmissionUtils(object):
     @classmethod
     def send_author_revision_requested_email(cls):
         """ Requires loading 'submission' and 'recommendation' attributes. """
-        email_text = ('Dear ' + title_dict[cls.submission.submitted_by.title] + ' ' +
+        email_text = ('Dear ' + cls.submission.submitted_by.get_title_display() + ' ' +
                       cls.submission.submitted_by.user.last_name +
                       ', \n\nThe Editor-in-charge of your recent Submission to SciPost,\n\n' +
                       cls.submission.title + ' by ' + cls.submission.author_list + ','
@@ -1070,7 +1067,7 @@ class SubmissionUtils(object):
             '<p>Sincerely,</p>'
             '<p>The SciPost Team.</p>')
         email_context = Context({
-            'auth_title': title_dict[cls.submission.submitted_by.title],
+            'auth_title': cls.submission.submitted_by.get_title_display(),
             'auth_last_name': cls.submission.submitted_by.user.last_name,
             'sub_title': cls.submission.title,
             'author_list': cls.submission.author_list,
@@ -1092,7 +1089,7 @@ class SubmissionUtils(object):
     @classmethod
     def send_author_College_decision_email(cls):
         """ Requires loading 'submission' and 'recommendation' attributes. """
-        email_text = ('Dear ' + title_dict[cls.submission.submitted_by.title] + ' ' +
+        email_text = ('Dear ' + cls.submission.submitted_by.get_title_display() + ' ' +
                       cls.submission.submitted_by.user.last_name +
                       ', \n\nThe Editorial College of SciPost has taken a decision '
                       'regarding your recent Submission,\n\n' +
@@ -1107,7 +1104,7 @@ class SubmissionUtils(object):
             or cls.recommendation.recommendation == 3):
             email_text += ('We are pleased to inform you that your Submission '
                            'has been accepted for publication in '
-                           + journals_submit_dict[cls.submission.submitted_to_journal])
+                           + cls.submission.get_submitted_to_journal_display())
             email_text_html += (
                 '<p>We are pleased to inform you that your Submission '
                 'has been accepted for publication in <strong>{{ journal }}</strong>')
@@ -1159,12 +1156,12 @@ class SubmissionUtils(object):
                             '<p>Sincerely,</p>'
                             '<p>The SciPost Team.</p>')
         email_context = Context({
-            'auth_title': title_dict[cls.submission.submitted_by.title],
+            'auth_title': cls.submission.submitted_by.get_title_display(),
             'auth_last_name': cls.submission.submitted_by.user.last_name,
             'sub_title': cls.submission.title,
             'author_list': cls.submission.author_list,
             'arxiv_identifier_w_vn_nr': cls.submission.arxiv_identifier_w_vn_nr,
-            'journal': journals_submit_dict[cls.submission.submitted_to_journal],
+            'journal': cls.submission.get_submitted_to_journal_display(),
         })
         email_text_html += '<br/>' + EMAIL_FOOTER
         html_template = Template(email_text_html)
diff --git a/submissions/views.py b/submissions/views.py
index 89a633c257136419cb47da29464d8caf8d900716..c0fc5b703691ef0174d41d696a6dd48258efe2fa 100644
--- a/submissions/views.py
+++ b/submissions/views.py
@@ -10,17 +10,15 @@ from django.http import HttpResponseRedirect
 from django.shortcuts import get_object_or_404, render, redirect
 from django.template import Template, Context
 from django.utils import timezone
-from django.utils.decorators import method_decorator
 
 from guardian.decorators import permission_required_or_403
-from guardian.mixins import LoginRequiredMixin, PermissionRequiredMixin
+from guardian.mixins import PermissionRequiredMixin
 from guardian.shortcuts import assign_perm
 
+from .constants import SUBMISSION_STATUS_PUBLICLY_UNLISTED, SUBMISSION_STATUS_VOTING_DEPRECATED,\
+                       SUBMISSION_STATUS_PUBLICLY_INVISIBLE, SUBMISSION_STATUS, ED_COMM_CHOICES
 from .models import Submission, EICRecommendation, EditorialAssignment,\
-                    RefereeInvitation, Report, EditorialCommunication,\
-                    SUBMISSION_STATUS_PUBLICLY_UNLISTED, SUBMISSION_STATUS_VOTING_DEPRECATED,\
-                    SUBMISSION_STATUS_PUBLICLY_INVISIBLE, SUBMISSION_STATUS_OUT_OF_POOL,\
-                    SUBMISSION_STATUS, submission_status_dict, ed_comm_choices_dict
+                    RefereeInvitation, Report, EditorialCommunication
 from .forms import SubmissionIdentifierForm, SubmissionForm, SubmissionSearchForm,\
                    RecommendationVoteForm, ConsiderAssignmentForm, AssignSubmissionForm,\
                    SetRefereeingDeadlineForm, RefereeSelectForm, RefereeRecruitmentForm,\
@@ -29,9 +27,8 @@ from .forms import SubmissionIdentifierForm, SubmissionForm, SubmissionSearchFor
 from .utils import SubmissionUtils
 
 from comments.models import Comment
-from journals.models import journals_submit_dict
 from scipost.forms import ModifyPersonalMessageForm, RemarkForm
-from scipost.models import Contributor, title_dict, Remark, RegistrationInvitation
+from scipost.models import Contributor, Remark, RegistrationInvitation
 
 from scipost.services import ArxivCaller
 from scipost.utils import Utils
@@ -381,7 +378,7 @@ def pool(request):
 @login_required
 @permission_required('scipost.can_view_pool', raise_exception=True)
 def submissions_by_status(request, status):
-    if status not in submission_status_dict.keys():
+    if status not in dict(SUBMISSION_STATUS).keys():
         errormessage = 'Unknown status.'
         return render(request, 'scipost/error.html', {'errormessage': errormessage})
     submissions_of_status = (Submission.objects.get_pool(request.user)
@@ -470,7 +467,7 @@ def accept_or_decline_assignment_ack(request, assignment_id):
         context = {'errormessage': errormessage}
         return render(request, 'submissions/accept_or_decline_assignment_ack.html', context)
     if assignment.submission.editor_in_charge:
-        errormessage = (title_dict[assignment.submission.editor_in_charge.title] + ' ' +
+        errormessage = (assignment.submission.editor_in_charge.get_title_display() + ' ' +
                         assignment.submission.editor_in_charge.user.last_name +
                         ' has already agreed to be Editor-in-charge of this Submission.')
         context = {'errormessage': errormessage}
@@ -525,7 +522,7 @@ def volunteer_as_EIC(request, arxiv_identifier_w_vn_nr):
         context = {'errormessage': errormessage}
         return render(request, 'submissions/accept_or_decline_assignment_ack.html', context)
     if submission.editor_in_charge:
-        errormessage = (title_dict[submission.editor_in_charge.title] + ' ' +
+        errormessage = (submission.editor_in_charge.get_title_display() + ' ' +
                         submission.editor_in_charge.user.last_name +
                         ' has already agreed to be Editor-in-charge of this Submission.')
         context = {'errormessage': errormessage}
@@ -710,10 +707,10 @@ def recruit_referee(request, arxiv_identifier_w_vn_nr):
             ref_invitation.save()
             # Create and send a registration invitation
             ref_inv_message_head = ('On behalf of the Editor-in-charge ' +
-                                    title_dict[submission.editor_in_charge.title] + ' ' +
+                                    submission.editor_in_charge.get_title_display() + ' ' +
                                     submission.editor_in_charge.user.last_name +
                                     ', we would like to invite you to referee a Submission to ' +
-                                    journals_submit_dict[submission.submitted_to_journal] +
+                                    submission.get_submitted_to_journal_display +
                                     ', namely\n\n' + submission.title +
                                     '\nby ' + submission.author_list +
                                     '\n (see https://scipost.org/submission/'
@@ -917,7 +914,7 @@ def communication(request, arxiv_identifier_w_vn_nr, comtype, referee_id=None):
     """
     submission = get_object_or_404(Submission, arxiv_identifier_w_vn_nr=arxiv_identifier_w_vn_nr)
     errormessage = None
-    if comtype not in ed_comm_choices_dict.keys():
+    if comtype not in dict(ED_COMM_CHOICES).keys():
         errormessage = 'Unknown type of cummunication.'
     # TODO: Verify that this is requested by an authorized contributor (eic, ref, author)
     elif (comtype in ['EtoA', 'EtoR', 'EtoS'] and
@@ -971,7 +968,7 @@ def eic_recommendation(request, arxiv_identifier_w_vn_nr):
     submission = get_object_or_404(Submission, arxiv_identifier_w_vn_nr=arxiv_identifier_w_vn_nr)
     if submission.status not in ['EICassigned', 'review_closed']:
         errormessage = ('This submission\'s current status is: ' +
-                        submission_status_dict[submission.status] + '. '
+                        submission.get_status_display + '. '
                         'An Editorial Recommendation is not required.')
         return render(request, 'scipost/error.html', {'errormessage': errormessage})
     if request.method == 'POST':
diff --git a/theses/forms.py b/theses/forms.py
index 3f5dfcc0e5784131d4e8a0d9f7039bad5b07064a..f211402036e27e1ad7c12f19ecaff8972f9bd512 100644
--- a/theses/forms.py
+++ b/theses/forms.py
@@ -3,13 +3,11 @@ from django.core.mail import EmailMessage
 from django.template.loader import render_to_string
 from django.urls import reverse
 
-from scipost.models import Contributor, title_dict
+from scipost.models import Contributor
 
 from .models import ThesisLink
 from .helpers import past_years
 
-from scipost.models import Contributor, title_dict
-
 
 class RequestThesisLinkForm(forms.ModelForm):
     class Meta:
@@ -64,7 +62,7 @@ class VetThesisLinkForm(RequestThesisLinkForm):
 
     def vet_request(self, thesislink, user):
         mail_params = {
-            'vocative_title': title_dict[thesislink.requested_by.title],
+            'vocative_title': thesislink.requested_by.get_title_display(),
             'thesislink': thesislink,
             'full_url': self.request.build_absolute_uri(
                 reverse('theses:thesis', kwargs={'thesislink_id': thesislink.id}))
diff --git a/theses/models.py b/theses/models.py
index f744431a5831180e6c0c7751ec668379659790b7..241a9409defc8069adee6ca49ccd074f6af3ff9a 100644
--- a/theses/models.py
+++ b/theses/models.py
@@ -1,7 +1,7 @@
 from django.utils import timezone
 from django.db import models
 
-from journals.models import SCIPOST_JOURNALS_DOMAINS
+from journals.constants import SCIPOST_JOURNALS_DOMAINS
 from scipost.constants import SCIPOST_DISCIPLINES, SCIPOST_SUBJECT_AREAS
 from scipost.models import Contributor
 
diff --git a/theses/templatetags/theses_extras.py b/theses/templatetags/theses_extras.py
index f3b441f41f64e3bb5a7ad23f852eb3e627fe9dec..3958f21555cdde6e26465d2d537528f718b0a10e 100644
--- a/theses/templatetags/theses_extras.py
+++ b/theses/templatetags/theses_extras.py
@@ -1,7 +1,6 @@
 from django import template
 
 from scipost.constants import SCIPOST_DISCIPLINES, subject_areas_dict, disciplines_dict
-from journals.models import journals_domains_dict
 
 register = template.Library()
 
@@ -18,7 +17,7 @@ def discipline(thesislink):
 
 @register.filter
 def domain(thesislink):
-    return journals_domains_dict[thesislink.domain]
+    return thesislink.get_domain_display
 
 
 @register.filter
diff --git a/theses/views.py b/theses/views.py
index 9f6a10f83005fa86da1c877ea2a5b45b8222cea7..b91abefdb2170c88a4de62c374fa7095e6093317 100644
--- a/theses/views.py
+++ b/theses/views.py
@@ -13,13 +13,9 @@ from django.utils.decorators import method_decorator
 from .models import ThesisLink
 from .forms import RequestThesisLinkForm, ThesisLinkSearchForm, VetThesisLinkForm
 
-from comments.models import Comment
 from comments.forms import CommentForm
-from scipost.forms import TITLE_CHOICES
-from scipost.models import Contributor
-import strings
 
-title_dict = dict(TITLE_CHOICES)  # Convert titles for use in emails
+import strings
 
 
 ################