mirror of
https://github.com/daniviga/django-ram.git
synced 2026-02-04 18:10:38 +01:00
Compare commits
11 Commits
8087ab5997
...
v0.19.6
| Author | SHA1 | Date | |
|---|---|---|---|
|
49c8d804d6
|
|||
|
2ab2d00585
|
|||
|
c95064ddec
|
|||
|
16bd82de39
|
|||
|
2ae7f2685d
|
|||
|
29f9a213b4
|
|||
|
884661d4e1
|
|||
|
c7cace96f7
|
|||
|
d3c099c05b
|
|||
|
903633b5a7
|
|||
|
ee775d737e
|
@@ -59,20 +59,34 @@ class MagazineIssueDocInline(BookDocInline):
|
|||||||
model = MagazineIssueDocument
|
model = MagazineIssueDocument
|
||||||
|
|
||||||
|
|
||||||
|
class BookTocInline(admin.TabularInline):
|
||||||
|
model = TocEntry
|
||||||
|
min_num = 0
|
||||||
|
extra = 0
|
||||||
|
fields = (
|
||||||
|
"title",
|
||||||
|
"subtitle",
|
||||||
|
"authors",
|
||||||
|
"page",
|
||||||
|
"featured",
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
@admin.register(Book)
|
@admin.register(Book)
|
||||||
class BookAdmin(SortableAdminBase, admin.ModelAdmin):
|
class BookAdmin(SortableAdminBase, admin.ModelAdmin):
|
||||||
inlines = (
|
inlines = (
|
||||||
|
BookTocInline,
|
||||||
BookPropertyInline,
|
BookPropertyInline,
|
||||||
BookImageInline,
|
BookImageInline,
|
||||||
BookDocInline,
|
BookDocInline,
|
||||||
)
|
)
|
||||||
list_display = (
|
list_display = (
|
||||||
"published",
|
|
||||||
"title",
|
"title",
|
||||||
"get_authors",
|
"get_authors",
|
||||||
"get_publisher",
|
"get_publisher",
|
||||||
"publication_year",
|
"publication_year",
|
||||||
"number_of_pages",
|
"number_of_pages",
|
||||||
|
"published",
|
||||||
)
|
)
|
||||||
autocomplete_fields = ("authors", "publisher", "shop")
|
autocomplete_fields = ("authors", "publisher", "shop")
|
||||||
readonly_fields = ("invoices", "creation_time", "updated_time")
|
readonly_fields = ("invoices", "creation_time", "updated_time")
|
||||||
@@ -364,23 +378,10 @@ class CatalogAdmin(SortableAdminBase, admin.ModelAdmin):
|
|||||||
actions = [publish, unpublish, download_csv]
|
actions = [publish, unpublish, download_csv]
|
||||||
|
|
||||||
|
|
||||||
class MagazineIssueToc(admin.TabularInline):
|
|
||||||
model = TocEntry
|
|
||||||
min_num = 0
|
|
||||||
extra = 0
|
|
||||||
fields = (
|
|
||||||
"title",
|
|
||||||
"subtitle",
|
|
||||||
"authors",
|
|
||||||
"page",
|
|
||||||
"featured",
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
@admin.register(MagazineIssue)
|
@admin.register(MagazineIssue)
|
||||||
class MagazineIssueAdmin(SortableAdminBase, admin.ModelAdmin):
|
class MagazineIssueAdmin(SortableAdminBase, admin.ModelAdmin):
|
||||||
inlines = (
|
inlines = (
|
||||||
MagazineIssueToc,
|
BookTocInline,
|
||||||
BookPropertyInline,
|
BookPropertyInline,
|
||||||
BookImageInline,
|
BookImageInline,
|
||||||
MagazineIssueDocInline,
|
MagazineIssueDocInline,
|
||||||
|
|||||||
@@ -0,0 +1,28 @@
|
|||||||
|
# Generated by Django 6.0 on 2025-12-31 13:47
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
("bookshelf", "0030_tocentry"),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name="tocentry",
|
||||||
|
name="authors",
|
||||||
|
field=models.CharField(blank=True),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name="tocentry",
|
||||||
|
name="subtitle",
|
||||||
|
field=models.CharField(blank=True),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name="tocentry",
|
||||||
|
name="title",
|
||||||
|
field=models.CharField(),
|
||||||
|
),
|
||||||
|
]
|
||||||
@@ -254,9 +254,9 @@ class TocEntry(BaseModel):
|
|||||||
book = models.ForeignKey(
|
book = models.ForeignKey(
|
||||||
BaseBook, on_delete=models.CASCADE, related_name="toc"
|
BaseBook, on_delete=models.CASCADE, related_name="toc"
|
||||||
)
|
)
|
||||||
title = models.CharField(max_length=200)
|
title = models.CharField()
|
||||||
subtitle = models.CharField(max_length=200, blank=True)
|
subtitle = models.CharField(blank=True)
|
||||||
authors = models.CharField(max_length=256, blank=True)
|
authors = models.CharField(blank=True)
|
||||||
page = models.SmallIntegerField()
|
page = models.SmallIntegerField()
|
||||||
featured = models.BooleanField(
|
featured = models.BooleanField(
|
||||||
default=False,
|
default=False,
|
||||||
@@ -268,9 +268,15 @@ class TocEntry(BaseModel):
|
|||||||
verbose_name_plural = "Table of Contents Entries"
|
verbose_name_plural = "Table of Contents Entries"
|
||||||
|
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return f"{self.title} (p. {self.page})"
|
if self.subtitle:
|
||||||
|
title = f"{self.title}: {self.subtitle}"
|
||||||
|
else:
|
||||||
|
title = self.title
|
||||||
|
return f"{title} (p. {self.page})"
|
||||||
|
|
||||||
def clean(self):
|
def clean(self):
|
||||||
|
if self.page is None:
|
||||||
|
raise ValidationError("Page number is required.")
|
||||||
if self.page < 1:
|
if self.page < 1:
|
||||||
raise ValidationError("Page number is invalid.")
|
raise ValidationError("Page number is invalid.")
|
||||||
try:
|
try:
|
||||||
|
|||||||
18
ram/consist/migrations/0019_consistitem_load.py
Normal file
18
ram/consist/migrations/0019_consistitem_load.py
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
# Generated by Django 6.0 on 2026-01-03 12:31
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
("consist", "0018_alter_consist_scale"),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AddField(
|
||||||
|
model_name="consistitem",
|
||||||
|
name="load",
|
||||||
|
field=models.BooleanField(default=False),
|
||||||
|
),
|
||||||
|
]
|
||||||
@@ -43,10 +43,10 @@ class Consist(BaseModel):
|
|||||||
|
|
||||||
@property
|
@property
|
||||||
def length(self):
|
def length(self):
|
||||||
return self.consist_item.count()
|
return self.consist_item.filter(load=False).count()
|
||||||
|
|
||||||
def get_type_count(self):
|
def get_type_count(self):
|
||||||
return self.consist_item.annotate(
|
return self.consist_item.filter(load=False).annotate(
|
||||||
type=models.F("rolling_stock__rolling_class__type__type")
|
type=models.F("rolling_stock__rolling_class__type__type")
|
||||||
).values(
|
).values(
|
||||||
"type"
|
"type"
|
||||||
@@ -69,6 +69,7 @@ class ConsistItem(models.Model):
|
|||||||
Consist, on_delete=models.CASCADE, related_name="consist_item"
|
Consist, on_delete=models.CASCADE, related_name="consist_item"
|
||||||
)
|
)
|
||||||
rolling_stock = models.ForeignKey(RollingStock, on_delete=models.CASCADE)
|
rolling_stock = models.ForeignKey(RollingStock, on_delete=models.CASCADE)
|
||||||
|
load = models.BooleanField(default=False)
|
||||||
order = models.PositiveIntegerField(blank=False, null=False)
|
order = models.PositiveIntegerField(blank=False, null=False)
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
@@ -92,10 +93,15 @@ class ConsistItem(models.Model):
|
|||||||
# because the consist is not saved yet and it must be moved
|
# because the consist is not saved yet and it must be moved
|
||||||
# to the admin form validation via InlineFormSet.clean()
|
# to the admin form validation via InlineFormSet.clean()
|
||||||
consist = self.consist
|
consist = self.consist
|
||||||
if rolling_stock.scale != consist.scale:
|
# Scale must match, but allow loads of any scale
|
||||||
|
if rolling_stock.scale != consist.scale and not self.load:
|
||||||
raise ValidationError(
|
raise ValidationError(
|
||||||
"The rolling stock and consist must be of the same scale."
|
"The rolling stock and consist must be of the same scale."
|
||||||
)
|
)
|
||||||
|
if self.load and rolling_stock.scale.ratio != consist.scale.ratio:
|
||||||
|
raise ValidationError(
|
||||||
|
"The load and consist must be of the same scale ratio."
|
||||||
|
)
|
||||||
if self.consist.published and not rolling_stock.published:
|
if self.consist.published and not rolling_stock.published:
|
||||||
raise ValidationError(
|
raise ValidationError(
|
||||||
"You must unpublish the the consist before using this item."
|
"You must unpublish the the consist before using this item."
|
||||||
|
|||||||
@@ -20,6 +20,7 @@ class SiteConfigurationAdmin(SingletonModelAdmin):
|
|||||||
"about",
|
"about",
|
||||||
"items_per_page",
|
"items_per_page",
|
||||||
"items_ordering",
|
"items_ordering",
|
||||||
|
"featured_items_ordering",
|
||||||
"currency",
|
"currency",
|
||||||
"footer",
|
"footer",
|
||||||
"footer_extended",
|
"footer_extended",
|
||||||
|
|||||||
@@ -0,0 +1,43 @@
|
|||||||
|
# Generated by Django 6.0 on 2026-01-02 23:41
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
("portal", "0020_alter_flatpage_options"),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AddField(
|
||||||
|
model_name="siteconfiguration",
|
||||||
|
name="featured_items_ordering",
|
||||||
|
field=models.CharField(
|
||||||
|
choices=[
|
||||||
|
("type", "By rolling stock type and company"),
|
||||||
|
("class", "By rolling stock type and class"),
|
||||||
|
("company", "By company and type"),
|
||||||
|
("country", "By country and type"),
|
||||||
|
("cou+com", "By country and company"),
|
||||||
|
],
|
||||||
|
default="type",
|
||||||
|
max_length=11,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name="siteconfiguration",
|
||||||
|
name="items_ordering",
|
||||||
|
field=models.CharField(
|
||||||
|
choices=[
|
||||||
|
("type", "By rolling stock type and company"),
|
||||||
|
("class", "By rolling stock type and class"),
|
||||||
|
("company", "By company and type"),
|
||||||
|
("country", "By country and type"),
|
||||||
|
("cou+com", "By country and company"),
|
||||||
|
],
|
||||||
|
default="type",
|
||||||
|
max_length=11,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
]
|
||||||
@@ -22,14 +22,17 @@ class SiteConfiguration(SingletonModel):
|
|||||||
default="6",
|
default="6",
|
||||||
)
|
)
|
||||||
items_ordering = models.CharField(
|
items_ordering = models.CharField(
|
||||||
max_length=10,
|
max_length=11,
|
||||||
choices=[
|
choices=[
|
||||||
("type", "By rolling stock type"),
|
("type", "By rolling stock type and company"),
|
||||||
("company", "By company name"),
|
("class", "By rolling stock type and class"),
|
||||||
("identifier", "By rolling stock class"),
|
("company", "By company and type"),
|
||||||
|
("country", "By country and type"),
|
||||||
|
("cou+com", "By country and company"),
|
||||||
],
|
],
|
||||||
default="type",
|
default="type",
|
||||||
)
|
)
|
||||||
|
featured_items_ordering = items_ordering.clone()
|
||||||
currency = models.CharField(max_length=3, default="EUR")
|
currency = models.CharField(max_length=3, default="EUR")
|
||||||
footer = tinymce.HTMLField(blank=True)
|
footer = tinymce.HTMLField(blank=True)
|
||||||
footer_extended = tinymce.HTMLField(blank=True)
|
footer_extended = tinymce.HTMLField(blank=True)
|
||||||
|
|||||||
26
ram/portal/templates/_includes/documents.html
Normal file
26
ram/portal/templates/_includes/documents.html
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
{% if documents %}
|
||||||
|
<table class="table table-striped">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th colspan="3" scope="row">{{ header|default:"Documents" }}</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody class="table-group-divider">
|
||||||
|
{% for d in documents.all %}
|
||||||
|
<tr>
|
||||||
|
<td class="w-33">{{ d.description }}</td>
|
||||||
|
<td class="text-nowrap">
|
||||||
|
{% if d.private %}
|
||||||
|
<i class="bi bi-file-earmark-lock2"></i>
|
||||||
|
{% else %}
|
||||||
|
<i class="bi bi-file-earmark-text"></i>
|
||||||
|
{% endif %}
|
||||||
|
<a href="{{ d.file.url }}" target="_blank">{{ d.filename }}</a>
|
||||||
|
</td>
|
||||||
|
<td class="text-end">{{ d.file.size | filesizeformat }}</td>
|
||||||
|
</tr>
|
||||||
|
{% endfor %}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
26
ram/portal/templates/_modules/documents.html
Normal file
26
ram/portal/templates/_modules/documents.html
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
{% if documents %}
|
||||||
|
<table class="table table-striped">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th colspan="3" scope="row">{{ header|default:"Documents" }}</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody class="table-group-divider">
|
||||||
|
{% for d in documents.all %}
|
||||||
|
<tr>
|
||||||
|
<td class="w-33">{{ d.description }}</td>
|
||||||
|
<td class="text-nowrap">
|
||||||
|
{% if d.private %}
|
||||||
|
<i class="bi bi-file-earmark-lock2"></i>
|
||||||
|
{% else %}
|
||||||
|
<i class="bi bi-file-earmark-text"></i>
|
||||||
|
{% endif %}
|
||||||
|
<a href="{{ d.file.url }}" target="_blank">{{ d.filename }}</a>
|
||||||
|
</td>
|
||||||
|
<td class="text-end">{{ d.file.size | filesizeformat }}</td>
|
||||||
|
</tr>
|
||||||
|
{% endfor %}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
18
ram/portal/templates/_modules/properties.html
Normal file
18
ram/portal/templates/_modules/properties.html
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
{% if properties %}
|
||||||
|
<table class="table table-striped">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th colspan="2" scope="row">Properties</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody class="table-group-divider">
|
||||||
|
{% for p in properties %}
|
||||||
|
<tr>
|
||||||
|
<th class="w-33" scope="row">{{ p.property }}</th>
|
||||||
|
<td>{{ p.value }}</td>
|
||||||
|
</tr>
|
||||||
|
{% endfor %}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
29
ram/portal/templates/_modules/purchase_data.html
Normal file
29
ram/portal/templates/_modules/purchase_data.html
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
{% if request.user.is_staff %}
|
||||||
|
{% if data.shop or data.purchase_date or data.price %}
|
||||||
|
<table class="table table-striped">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th colspan="2" scope="row">Purchase</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody class="table-group-divider">
|
||||||
|
<tr>
|
||||||
|
<th class="w-33" scope="row">Shop</th>
|
||||||
|
<td>
|
||||||
|
{{ data.shop|default:"-" }}
|
||||||
|
{% if data.shop.website %} <a href="{{ data.shop.website }}" target="_blank"><i class="bi bi-box-arrow-up-right"></i></a>{% endif %}
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<th class="w-33" scope="row">Purchase date</th>
|
||||||
|
<td>{{ data.purchase_date|default:"-" }}</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<th scope="row">Price ({{ site_conf.currency }})</th>
|
||||||
|
<td>{{ data.price|default:"-" }}</td>
|
||||||
|
</tr>
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
{% endif %}
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
@@ -148,7 +148,7 @@
|
|||||||
<strong>{{ site_conf.site_name }}</strong>
|
<strong>{{ site_conf.site_name }}</strong>
|
||||||
</a>
|
</a>
|
||||||
</div>
|
</div>
|
||||||
{% include 'includes/login.html' %}
|
{% include '_includes/login.html' %}
|
||||||
</div>
|
</div>
|
||||||
</nav>
|
</nav>
|
||||||
</header>
|
</header>
|
||||||
@@ -186,7 +186,7 @@
|
|||||||
{% show_bookshelf_menu %}
|
{% show_bookshelf_menu %}
|
||||||
{% show_flatpages_menu user %}
|
{% show_flatpages_menu user %}
|
||||||
</ul>
|
</ul>
|
||||||
{% include 'includes/search.html' %}
|
{% include '_includes/search.html' %}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</nav>
|
</nav>
|
||||||
@@ -211,9 +211,9 @@
|
|||||||
<div class="container">{% block pagination %}{% endblock %}</div>
|
<div class="container">{% block pagination %}{% endblock %}</div>
|
||||||
</div>
|
</div>
|
||||||
{% block extra_content %}{% endblock %}
|
{% block extra_content %}{% endblock %}
|
||||||
{% include 'includes/symbols.html' %}
|
{% include '_includes/symbols.html' %}
|
||||||
</main>
|
</main>
|
||||||
{% include 'includes/footer.html' %}
|
{% include '_includes/footer.html' %}
|
||||||
{% if site_conf.use_cdn %}
|
{% if site_conf.use_cdn %}
|
||||||
<script src="https://cdn.jsdelivr.net/npm/bootstrap@5.3.8/dist/js/bootstrap.bundle.min.js"></script>
|
<script src="https://cdn.jsdelivr.net/npm/bootstrap@5.3.8/dist/js/bootstrap.bundle.min.js"></script>
|
||||||
{% else %}
|
{% else %}
|
||||||
|
|||||||
@@ -147,51 +147,10 @@
|
|||||||
{% endif %}
|
{% endif %}
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
{% if request.user.is_staff %}
|
{% include "_modules/purchase_data.html" %}
|
||||||
<table class="table table-striped">
|
{% include "_modules/properties.html" %}
|
||||||
<thead>
|
|
||||||
<tr>
|
|
||||||
<th colspan="2" scope="row">Purchase</th>
|
|
||||||
</tr>
|
|
||||||
</thead>
|
|
||||||
<tbody class="table-group-divider">
|
|
||||||
<tr>
|
|
||||||
<th class="w-33" scope="row">Shop</th>
|
|
||||||
<td>
|
|
||||||
{{ data.shop|default:"-" }}
|
|
||||||
{% if data.shop.website %} <a href="{{ data.shop.website }}" target="_blank"><i class="bi bi-box-arrow-up-right"></i></a>{% endif %}
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
<tr>
|
|
||||||
<th class="w-33" scope="row">Purchase date</th>
|
|
||||||
<td>{{ data.purchase_date|default:"-" }}</td>
|
|
||||||
</tr>
|
|
||||||
<tr>
|
|
||||||
<th scope="row">Price ({{ site_conf.currency }})</th>
|
|
||||||
<td>{{ data.price|default:"-" }}</td>
|
|
||||||
</tr>
|
|
||||||
</tbody>
|
|
||||||
</table>
|
|
||||||
{% endif %}
|
|
||||||
{% if properties %}
|
|
||||||
<table class="table table-striped">
|
|
||||||
<thead>
|
|
||||||
<tr>
|
|
||||||
<th colspan="2" scope="row">Properties</th>
|
|
||||||
</tr>
|
|
||||||
</thead>
|
|
||||||
<tbody class="table-group-divider">
|
|
||||||
{% for p in properties %}
|
|
||||||
<tr>
|
|
||||||
<th class="w-33" scope="row">{{ p.property }}</th>
|
|
||||||
<td>{{ p.value }}</td>
|
|
||||||
</tr>
|
|
||||||
{% endfor %}
|
|
||||||
</tbody>
|
|
||||||
</table>
|
|
||||||
{% endif %}
|
|
||||||
</div>
|
</div>
|
||||||
<div class="tab-pane" id="nav-toc" role="tabpanel" aria-labelledby="nav-toc-tab">
|
<div class="tab-pane table-responsive" id="nav-toc" role="tabpanel" aria-labelledby="nav-toc-tab">
|
||||||
<table class="table table-striped">
|
<table class="table table-striped">
|
||||||
<thead>
|
<thead>
|
||||||
<tr>
|
<tr>
|
||||||
@@ -216,22 +175,7 @@
|
|||||||
</table>
|
</table>
|
||||||
</div>
|
</div>
|
||||||
<div class="tab-pane" id="nav-documents" role="tabpanel" aria-labelledby="nav-documents-tab">
|
<div class="tab-pane" id="nav-documents" role="tabpanel" aria-labelledby="nav-documents-tab">
|
||||||
<table class="table table-striped">
|
{% include "_modules/documents.html" %}
|
||||||
<thead>
|
|
||||||
<tr>
|
|
||||||
<th colspan="3" scope="row">Documents</th>
|
|
||||||
</tr>
|
|
||||||
</thead>
|
|
||||||
<tbody class="table-group-divider">
|
|
||||||
{% for d in documents.all %}
|
|
||||||
<tr>
|
|
||||||
<td class="w-33">{{ d.description }}</td>
|
|
||||||
<td><a href="{{ d.file.url }}" target="_blank">{{ d.filename }}</a></td>
|
|
||||||
<td class="text-end">{{ d.file.size | filesizeformat }}</td>
|
|
||||||
</tr>
|
|
||||||
{% endfor %}
|
|
||||||
</tbody>
|
|
||||||
</table>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="d-grid gap-2 d-md-flex justify-content-md-end">
|
<div class="d-grid gap-2 d-md-flex justify-content-md-end">
|
||||||
|
|||||||
@@ -7,11 +7,11 @@
|
|||||||
{{ t.name }}</a>{# new line is required #}
|
{{ t.name }}</a>{# new line is required #}
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</p>
|
</p>
|
||||||
|
{% endif %}
|
||||||
{% if not consist.published %}
|
{% if not consist.published %}
|
||||||
<span class="badge text-bg-warning">Unpublished</span> |
|
<span class="badge text-bg-warning">Unpublished</span> |
|
||||||
{% endif %}
|
{% endif %}
|
||||||
<small class="text-body-secondary">Updated {{ consist.updated_time | date:"M d, Y H:i" }}</small>
|
<small class="text-body-secondary">Updated {{ consist.updated_time | date:"M d, Y H:i" }}</small>
|
||||||
{% endif %}
|
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
{% block carousel %}
|
{% block carousel %}
|
||||||
{% if consist.image %}
|
{% if consist.image %}
|
||||||
@@ -26,6 +26,33 @@
|
|||||||
</div>
|
</div>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
{% block cards_layout %}
|
||||||
|
<div class="row row-cols-1 row-cols-md-2 row-cols-lg-3 g-3">
|
||||||
|
{% block cards %}
|
||||||
|
{% for d in data %}
|
||||||
|
{% include "cards/roster.html" %}
|
||||||
|
{% endfor %}
|
||||||
|
{% endblock %}
|
||||||
|
</div>
|
||||||
|
<div class="accordion shadow-sm mt-4" id="accordionLoads">
|
||||||
|
<div class="accordion-item">
|
||||||
|
<h2 class="accordion-header">
|
||||||
|
<button class="accordion-button collapsed" type="button" data-bs-toggle="collapse" data-bs-target="#collapseLoads" aria-expanded="false" aria-controls="collapseLoads">
|
||||||
|
<i class="bi bi-download"></i> Rolling Stock loaded on freight cars
|
||||||
|
</button>
|
||||||
|
</h2>
|
||||||
|
<div id="collapseLoads" class="accordion-collapse collapse" data-bs-parent="#accordionLoads">
|
||||||
|
<div class="accordion-body">
|
||||||
|
<div class="row row-cols-1 row-cols-md-2 row-cols-lg-3 g-3">
|
||||||
|
{% for l in loads %}
|
||||||
|
{% include "cards/roster.html" with d=l %}
|
||||||
|
{% endfor %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endblock %}
|
||||||
{% block pagination %}
|
{% block pagination %}
|
||||||
{% if data.has_other_pages %}
|
{% if data.has_other_pages %}
|
||||||
<nav aria-label="Page navigation">
|
<nav aria-label="Page navigation">
|
||||||
@@ -76,7 +103,7 @@
|
|||||||
<option value="nav-summary" selected>Summary</option>
|
<option value="nav-summary" selected>Summary</option>
|
||||||
</select>
|
</select>
|
||||||
<div class="tab-content" id="nav-tabContent">
|
<div class="tab-content" id="nav-tabContent">
|
||||||
<div class="tab-pane show active" id="nav-summary" role="tabpanel" aria-labelledby="nav-summary-tab">
|
<div class="tab-pane show active table-responsive" id="nav-summary" role="tabpanel" aria-labelledby="nav-summary-tab">
|
||||||
<table class="table table-striped">
|
<table class="table table-striped">
|
||||||
<thead>
|
<thead>
|
||||||
<tr>
|
<tr>
|
||||||
@@ -113,7 +140,7 @@
|
|||||||
</tr>
|
</tr>
|
||||||
<tr>
|
<tr>
|
||||||
<th scope="row">Composition</th>
|
<th scope="row">Composition</th>
|
||||||
<td>{% for t in consist.get_type_count %}{{ t.count }}x {{ t.type }} {{t.category }}{% if not forloop.last %} » {% endif %}{% endfor %}</td>
|
<td>{% for t in consist.get_type_count %}{{ t.count }}x {{ t.type }} {{t.category }}{% if not forloop.last %} » {% endif %}{% endfor %}{% if loads %} | <i class="bi bi-download"></i> {{ loads|length }}x Load{{ loads|pluralize }}{% endif %}</td>
|
||||||
</tr>
|
</tr>
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
|
|||||||
@@ -217,49 +217,8 @@
|
|||||||
{% endif %}
|
{% endif %}
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
{% if request.user.is_staff %}
|
{% include "_modules/purchase_data.html" with data=rolling_stock %}
|
||||||
<table class="table table-striped">
|
{% include "_modules/properties.html" %}
|
||||||
<thead>
|
|
||||||
<tr>
|
|
||||||
<th colspan="2" scope="row">Purchase</th>
|
|
||||||
</tr>
|
|
||||||
</thead>
|
|
||||||
<tbody class="table-group-divider">
|
|
||||||
<tr>
|
|
||||||
<th class="w-33" scope="row">Shop</th>
|
|
||||||
<td>
|
|
||||||
{{ rolling_stock.shop | default:"-" }}
|
|
||||||
{% if rolling_stock.shop.website %} <a href="{{ rolling_stock.shop.website }}" target="_blank"><i class="bi bi-box-arrow-up-right"></i></a>{% endif %}
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
<tr>
|
|
||||||
<th class="w-33" scope="row">Purchase date</th>
|
|
||||||
<td>{{ rolling_stock.purchase_date | default:"-" }}</td>
|
|
||||||
</tr>
|
|
||||||
<tr>
|
|
||||||
<th scope="row">Price ({{ site_conf.currency }})</th>
|
|
||||||
<td>{{ rolling_stock.price | default:"-" }}</td>
|
|
||||||
</tr>
|
|
||||||
</tbody>
|
|
||||||
</table>
|
|
||||||
{% endif %}
|
|
||||||
{% if properties %}
|
|
||||||
<table class="table table-striped">
|
|
||||||
<thead>
|
|
||||||
<tr>
|
|
||||||
<th colspan="2" scope="row">Properties</th>
|
|
||||||
</tr>
|
|
||||||
</thead>
|
|
||||||
<tbody class="table-group-divider">
|
|
||||||
{% for p in properties %}
|
|
||||||
<tr>
|
|
||||||
<th class="w-33" scope="row">{{ p.property }}</th>
|
|
||||||
<td>{{ p.value }}</td>
|
|
||||||
</tr>
|
|
||||||
{% endfor %}
|
|
||||||
</tbody>
|
|
||||||
</table>
|
|
||||||
{% endif %}
|
|
||||||
</div>
|
</div>
|
||||||
<div class="tab-pane" id="nav-class" role="tabpanel" aria-labelledby="nav-class-tab">
|
<div class="tab-pane" id="nav-class" role="tabpanel" aria-labelledby="nav-class-tab">
|
||||||
<table class="table table-striped">
|
<table class="table table-striped">
|
||||||
@@ -296,23 +255,7 @@
|
|||||||
{% endif %}
|
{% endif %}
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
{% if class_properties %}
|
{% include "_modules/properties.html" with properties=class_properties %}
|
||||||
<table class="table table-striped">
|
|
||||||
<thead>
|
|
||||||
<tr>
|
|
||||||
<th colspan="2" scope="row">Properties</th>
|
|
||||||
</tr>
|
|
||||||
</thead>
|
|
||||||
<tbody class="table-group-divider">
|
|
||||||
{% for p in class_properties %}
|
|
||||||
<tr>
|
|
||||||
<th class="w-33" scope="row">{{ p.property }}</th>
|
|
||||||
<td>{{ p.value }}</td>
|
|
||||||
</tr>
|
|
||||||
{% endfor %}
|
|
||||||
</tbody>
|
|
||||||
</table>
|
|
||||||
{% endif %}
|
|
||||||
</div>
|
</div>
|
||||||
<div class="tab-pane" id="nav-company" role="tabpanel" aria-labelledby="nav-company-tab">
|
<div class="tab-pane" id="nav-company" role="tabpanel" aria-labelledby="nav-company-tab">
|
||||||
<table class="table table-striped">
|
<table class="table table-striped">
|
||||||
@@ -402,43 +345,9 @@
|
|||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
</div>
|
</div>
|
||||||
<div class="tab-pane" id="nav-documents" role="tabpanel" aria-labelledby="nav-documents-tab">
|
<div class="tab-pane table-responsive" id="nav-documents" role="tabpanel" aria-labelledby="nav-documents-tab">
|
||||||
{% if documents %}
|
{% include "_modules/documents.html" %}
|
||||||
<table class="table table-striped">
|
{% include "_modules/documents.html" with documents=decoder_documents header="Decoder documents" %}
|
||||||
<thead>
|
|
||||||
<tr>
|
|
||||||
<th colspan="3" scope="row">Documents</th>
|
|
||||||
</tr>
|
|
||||||
</thead>
|
|
||||||
<tbody class="table-group-divider">
|
|
||||||
{% for d in documents.all %}
|
|
||||||
<tr>
|
|
||||||
<td class="w-33">{{ d.description }}</td>
|
|
||||||
<td><a href="{{ d.file.url }}" target="_blank">{{ d.filename }}</a></td>
|
|
||||||
<td class="text-end">{{ d.file.size | filesizeformat }}</td>
|
|
||||||
</tr>
|
|
||||||
{% endfor %}
|
|
||||||
</tbody>
|
|
||||||
</table>
|
|
||||||
{% endif %}
|
|
||||||
{% if decoder_documents %}
|
|
||||||
<table class="table table-striped">
|
|
||||||
<thead>
|
|
||||||
<tr>
|
|
||||||
<th colspan="3" scope="row">Decoder documents</th>
|
|
||||||
</tr>
|
|
||||||
</thead>
|
|
||||||
<tbody class="table-group-divider">
|
|
||||||
{% for d in decoder_documents.all %}
|
|
||||||
<tr>
|
|
||||||
<td class="w-33">{{ d.description }}</td>
|
|
||||||
<td><a href="{{ d.file.url }}" target="_blank">{{ d.filename }}</a></td>
|
|
||||||
<td class="text-end">{{ d.file.size | filesizeformat }}</td>
|
|
||||||
</tr>
|
|
||||||
{% endfor %}
|
|
||||||
</tbody>
|
|
||||||
</table>
|
|
||||||
{% endif %}
|
|
||||||
</div>
|
</div>
|
||||||
<div class="tab-pane" id="nav-journal" role="tabpanel" aria-labelledby="nav-journal-tab">
|
<div class="tab-pane" id="nav-journal" role="tabpanel" aria-labelledby="nav-journal-tab">
|
||||||
<table class="table table-striped">
|
<table class="table table-striped">
|
||||||
|
|||||||
@@ -36,25 +36,29 @@ def get_items_per_page():
|
|||||||
return int(items_per_page)
|
return int(items_per_page)
|
||||||
|
|
||||||
|
|
||||||
def get_order_by_field():
|
def get_items_ordering(config="items_ordering"):
|
||||||
try:
|
try:
|
||||||
order_by = get_site_conf().items_ordering
|
order_by = getattr(get_site_conf(), config)
|
||||||
except (OperationalError, ProgrammingError):
|
except (OperationalError, ProgrammingError):
|
||||||
order_by = "type"
|
order_by = "type"
|
||||||
|
|
||||||
fields = [
|
fields = [
|
||||||
"rolling_class__type",
|
"rolling_class__type", # 0
|
||||||
"rolling_class__company",
|
"rolling_class__company", # 1
|
||||||
"rolling_class__identifier",
|
"rolling_class__company__country", # 2
|
||||||
"road_number_int",
|
"rolling_class__identifier", # 3
|
||||||
|
"road_number_int", # 4
|
||||||
]
|
]
|
||||||
|
|
||||||
if order_by == "type":
|
order_map = {
|
||||||
return (fields[0], fields[1], fields[2], fields[3])
|
"type": (0, 1, 3, 4),
|
||||||
elif order_by == "company":
|
"company": (1, 0, 3, 4),
|
||||||
return (fields[1], fields[0], fields[2], fields[3])
|
"country": (2, 0, 1, 3, 4),
|
||||||
elif order_by == "identifier":
|
"cou+com": (2, 1, 0, 3, 4),
|
||||||
return (fields[2], fields[0], fields[1], fields[3])
|
"class": (0, 3, 1, 4),
|
||||||
|
}
|
||||||
|
|
||||||
|
return tuple(fields[i] for i in order_map.get(order_by, "type"))
|
||||||
|
|
||||||
|
|
||||||
class Render404(View):
|
class Render404(View):
|
||||||
@@ -70,7 +74,7 @@ class GetData(View):
|
|||||||
def get_data(self, request):
|
def get_data(self, request):
|
||||||
return (
|
return (
|
||||||
RollingStock.objects.get_published(request.user)
|
RollingStock.objects.get_published(request.user)
|
||||||
.order_by(*get_order_by_field())
|
.order_by(*get_items_ordering())
|
||||||
.filter(self.filter)
|
.filter(self.filter)
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -107,7 +111,9 @@ class GetHome(GetData):
|
|||||||
return (
|
return (
|
||||||
RollingStock.objects.get_published(request.user)
|
RollingStock.objects.get_published(request.user)
|
||||||
.filter(featured=True)
|
.filter(featured=True)
|
||||||
.order_by(*get_order_by_field())[:max_items]
|
.order_by(*get_items_ordering(config="featured_items_ordering"))[
|
||||||
|
:max_items
|
||||||
|
]
|
||||||
) or super().get_data(request)
|
) or super().get_data(request)
|
||||||
|
|
||||||
|
|
||||||
@@ -174,7 +180,7 @@ class SearchObjects(View):
|
|||||||
RollingStock.objects.get_published(request.user)
|
RollingStock.objects.get_published(request.user)
|
||||||
.filter(query)
|
.filter(query)
|
||||||
.distinct()
|
.distinct()
|
||||||
.order_by(*get_order_by_field())
|
.order_by(*get_items_ordering())
|
||||||
)
|
)
|
||||||
data = list(roster)
|
data = list(roster)
|
||||||
|
|
||||||
@@ -196,6 +202,7 @@ class SearchObjects(View):
|
|||||||
Q(
|
Q(
|
||||||
Q(title__icontains=search)
|
Q(title__icontains=search)
|
||||||
| Q(description__icontains=search)
|
| Q(description__icontains=search)
|
||||||
|
| Q(toc__title__icontains=search)
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
.distinct()
|
.distinct()
|
||||||
@@ -217,6 +224,7 @@ class SearchObjects(View):
|
|||||||
Q(
|
Q(
|
||||||
Q(magazine__name__icontains=search)
|
Q(magazine__name__icontains=search)
|
||||||
| Q(description__icontains=search)
|
| Q(description__icontains=search)
|
||||||
|
| Q(toc__title__icontains=search)
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
.distinct()
|
.distinct()
|
||||||
@@ -299,7 +307,7 @@ class GetManufacturerItem(View):
|
|||||||
if search != "all":
|
if search != "all":
|
||||||
roster = get_list_or_404(
|
roster = get_list_or_404(
|
||||||
RollingStock.objects.get_published(request.user).order_by(
|
RollingStock.objects.get_published(request.user).order_by(
|
||||||
*get_order_by_field()
|
*get_items_ordering()
|
||||||
),
|
),
|
||||||
Q(
|
Q(
|
||||||
Q(manufacturer=manufacturer)
|
Q(manufacturer=manufacturer)
|
||||||
@@ -321,7 +329,7 @@ class GetManufacturerItem(View):
|
|||||||
| Q(rolling_class__manufacturer=manufacturer)
|
| Q(rolling_class__manufacturer=manufacturer)
|
||||||
)
|
)
|
||||||
.distinct()
|
.distinct()
|
||||||
.order_by(*get_order_by_field())
|
.order_by(*get_items_ordering())
|
||||||
)
|
)
|
||||||
catalogs = Catalog.objects.get_published(request.user).filter(
|
catalogs = Catalog.objects.get_published(request.user).filter(
|
||||||
manufacturer=manufacturer
|
manufacturer=manufacturer
|
||||||
@@ -374,7 +382,7 @@ class GetObjectsFiltered(View):
|
|||||||
RollingStock.objects.get_published(request.user)
|
RollingStock.objects.get_published(request.user)
|
||||||
.filter(query)
|
.filter(query)
|
||||||
.distinct()
|
.distinct()
|
||||||
.order_by(*get_order_by_field())
|
.order_by(*get_items_ordering())
|
||||||
)
|
)
|
||||||
|
|
||||||
data = list(roster)
|
data = list(roster)
|
||||||
@@ -478,7 +486,7 @@ class GetRollingStock(View):
|
|||||||
& Q(set=True)
|
& Q(set=True)
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
.order_by(*get_order_by_field())
|
.order_by(*get_items_ordering())
|
||||||
)
|
)
|
||||||
|
|
||||||
return render(
|
return render(
|
||||||
@@ -518,7 +526,13 @@ class GetConsist(View):
|
|||||||
RollingStock.objects.get_published(request.user).get(
|
RollingStock.objects.get_published(request.user).get(
|
||||||
uuid=r.rolling_stock_id
|
uuid=r.rolling_stock_id
|
||||||
)
|
)
|
||||||
for r in consist.consist_item.all()
|
for r in consist.consist_item.filter(load=False)
|
||||||
|
)
|
||||||
|
loads = list(
|
||||||
|
RollingStock.objects.get_published(request.user).get(
|
||||||
|
uuid=r.rolling_stock_id
|
||||||
|
)
|
||||||
|
for r in consist.consist_item.filter(load=True)
|
||||||
)
|
)
|
||||||
paginator = Paginator(data, get_items_per_page())
|
paginator = Paginator(data, get_items_per_page())
|
||||||
data = paginator.get_page(page)
|
data = paginator.get_page(page)
|
||||||
@@ -533,6 +547,7 @@ class GetConsist(View):
|
|||||||
"title": consist,
|
"title": consist,
|
||||||
"consist": consist,
|
"consist": consist,
|
||||||
"data": data,
|
"data": data,
|
||||||
|
"loads": loads,
|
||||||
"page_range": page_range,
|
"page_range": page_range,
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
from ram.utils import git_suffix
|
from ram.utils import git_suffix
|
||||||
|
|
||||||
__version__ = "0.19.2"
|
__version__ = "0.19.6"
|
||||||
__version__ += git_suffix(__file__)
|
__version__ += git_suffix(__file__)
|
||||||
|
|||||||
@@ -1,22 +1,60 @@
|
|||||||
from django.contrib import admin
|
|
||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
|
from django.contrib import admin
|
||||||
|
from django.core.cache import cache
|
||||||
|
|
||||||
admin.site.site_header = settings.SITE_NAME
|
admin.site.site_header = settings.SITE_NAME
|
||||||
|
|
||||||
|
|
||||||
def publish(modeladmin, request, queryset):
|
def publish(modeladmin, request, queryset):
|
||||||
for obj in queryset:
|
queryset.update(published=True)
|
||||||
obj.published = True
|
cache.clear()
|
||||||
obj.save()
|
|
||||||
|
|
||||||
|
|
||||||
publish.short_description = "Publish selected items"
|
publish.short_description = "Publish selected items"
|
||||||
|
|
||||||
|
|
||||||
def unpublish(modeladmin, request, queryset):
|
def unpublish(modeladmin, request, queryset):
|
||||||
for obj in queryset:
|
queryset.update(published=False)
|
||||||
obj.published = False
|
cache.clear()
|
||||||
obj.save()
|
|
||||||
|
|
||||||
|
|
||||||
unpublish.short_description = "Unpublish selected items"
|
unpublish.short_description = "Unpublish selected items"
|
||||||
|
|
||||||
|
|
||||||
|
def set_featured(modeladmin, request, queryset):
|
||||||
|
count = queryset.count()
|
||||||
|
if count > settings.FEATURED_ITEMS_MAX:
|
||||||
|
modeladmin.message_user(
|
||||||
|
request,
|
||||||
|
"You can only mark up to {} items as featured.".format(
|
||||||
|
settings.FEATURED_ITEMS_MAX
|
||||||
|
),
|
||||||
|
level="error",
|
||||||
|
)
|
||||||
|
return
|
||||||
|
featured = modeladmin.model.objects.filter(featured=True).count()
|
||||||
|
if featured + count > settings.FEATURED_ITEMS_MAX:
|
||||||
|
modeladmin.message_user(
|
||||||
|
request,
|
||||||
|
"There are already {} featured items. You can only mark {} more items as featured.".format( # noqa: E501
|
||||||
|
featured,
|
||||||
|
settings.FEATURED_ITEMS_MAX - featured,
|
||||||
|
),
|
||||||
|
level="error",
|
||||||
|
)
|
||||||
|
return
|
||||||
|
queryset.update(featured=True)
|
||||||
|
cache.clear()
|
||||||
|
|
||||||
|
|
||||||
|
set_featured.short_description = "Mark selected items as featured"
|
||||||
|
|
||||||
|
|
||||||
|
def unset_featured(modeladmin, request, queryset):
|
||||||
|
queryset.update(featured=False)
|
||||||
|
cache.clear()
|
||||||
|
|
||||||
|
|
||||||
|
unset_featured.short_description = (
|
||||||
|
"Unmark selected items as featured"
|
||||||
|
)
|
||||||
|
|||||||
@@ -6,8 +6,8 @@ from django.utils.html import format_html, format_html_join, strip_tags
|
|||||||
|
|
||||||
from adminsortable2.admin import SortableAdminBase, SortableInlineAdminMixin
|
from adminsortable2.admin import SortableAdminBase, SortableInlineAdminMixin
|
||||||
|
|
||||||
from ram.admin import publish, unpublish
|
|
||||||
from ram.utils import generate_csv
|
from ram.utils import generate_csv
|
||||||
|
from ram.admin import publish, unpublish, set_featured, unset_featured
|
||||||
from repository.models import RollingStockDocument
|
from repository.models import RollingStockDocument
|
||||||
from portal.utils import get_site_conf
|
from portal.utils import get_site_conf
|
||||||
from roster.models import (
|
from roster.models import (
|
||||||
@@ -303,37 +303,4 @@ class RollingStockAdmin(SortableAdminBase, admin.ModelAdmin):
|
|||||||
|
|
||||||
download_csv.short_description = "Download selected items as CSV"
|
download_csv.short_description = "Download selected items as CSV"
|
||||||
|
|
||||||
def set_featured(modeladmin, request, queryset):
|
|
||||||
count = queryset.count()
|
|
||||||
if count > settings.FEATURED_ITEMS_MAX:
|
|
||||||
modeladmin.message_user(
|
|
||||||
request,
|
|
||||||
"You can only mark up to {} items as featured.".format(
|
|
||||||
settings.FEATURED_ITEMS_MAX
|
|
||||||
),
|
|
||||||
level="error",
|
|
||||||
)
|
|
||||||
return
|
|
||||||
featured = RollingStock.objects.filter(featured=True).count()
|
|
||||||
if featured + count > settings.FEATURED_ITEMS_MAX:
|
|
||||||
modeladmin.message_user(
|
|
||||||
request,
|
|
||||||
"There are already {} featured items. You can only mark {} more items as featured.".format( # noqa: E501
|
|
||||||
featured,
|
|
||||||
settings.FEATURED_ITEMS_MAX - featured,
|
|
||||||
),
|
|
||||||
level="error",
|
|
||||||
)
|
|
||||||
return
|
|
||||||
queryset.update(featured=True)
|
|
||||||
|
|
||||||
set_featured.short_description = "Mark selected rolling stock as featured"
|
|
||||||
|
|
||||||
def unset_featured(modeladmin, request, queryset):
|
|
||||||
queryset.update(featured=False)
|
|
||||||
|
|
||||||
unset_featured.short_description = (
|
|
||||||
"Unmark selected rolling stock as featured"
|
|
||||||
)
|
|
||||||
|
|
||||||
actions = [publish, unpublish, set_featured, unset_featured, download_csv]
|
actions = [publish, unpublish, set_featured, unset_featured, download_csv]
|
||||||
|
|||||||
Reference in New Issue
Block a user