service categories

This commit is contained in:
Tobias Brunner 2025-01-27 15:23:50 +01:00
parent 79a8c6f280
commit 17b6c4c9ee
No known key found for this signature in database
6 changed files with 154 additions and 5 deletions

View file

@ -1,6 +1,15 @@
from django.contrib import admin from django.contrib import admin
from django.utils.html import format_html from django.utils.html import format_html
from .models import CloudProvider, Country, ServiceLevel, Service from .models import CloudProvider, Country, ServiceLevel, Service, Category
@admin.register(Category)
class CategoryAdmin(admin.ModelAdmin):
list_display = ("name", "slug", "parent", "order")
list_filter = ("parent",)
search_fields = ("name", "description")
prepopulated_fields = {"slug": ("name",)}
ordering = ("order", "name")
@admin.register(CloudProvider) @admin.register(CloudProvider)
@ -30,10 +39,17 @@ class ServiceLevelAdmin(admin.ModelAdmin):
@admin.register(Service) @admin.register(Service)
class ServiceAdmin(admin.ModelAdmin): class ServiceAdmin(admin.ModelAdmin):
list_display = ("name", "cloud_provider", "service_level", "price", "logo_preview") list_display = (
list_filter = ("cloud_provider", "service_level", "countries") "name",
"cloud_provider",
"service_level",
"price",
"logo_preview",
"category_list",
)
list_filter = ("cloud_provider", "service_level", "countries", "categories")
search_fields = ("name", "description") search_fields = ("name", "description")
filter_horizontal = ("countries",) filter_horizontal = ("countries", "categories")
def logo_preview(self, obj): def logo_preview(self, obj):
if obj.logo: if obj.logo:
@ -41,3 +57,8 @@ class ServiceAdmin(admin.ModelAdmin):
'<img src="{}" style="max-height: 50px;"/>', obj.logo.url '<img src="{}" style="max-height: 50px;"/>', obj.logo.url
) )
return "No logo" return "No logo"
def category_list(self, obj):
return ", ".join([cat.name for cat in obj.categories.all()])
category_list.short_description = "Categories"

View file

@ -0,0 +1,53 @@
# Generated by Django 5.1.5 on 2025-01-27 14:19
import django.db.models.deletion
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
("services", "0002_cloudprovider_logo_service_logo"),
]
operations = [
migrations.CreateModel(
name="Category",
fields=[
(
"id",
models.BigAutoField(
auto_created=True,
primary_key=True,
serialize=False,
verbose_name="ID",
),
),
("name", models.CharField(max_length=100)),
("slug", models.SlugField(unique=True)),
("description", models.TextField(blank=True)),
("order", models.IntegerField(default=0)),
(
"parent",
models.ForeignKey(
blank=True,
null=True,
on_delete=django.db.models.deletion.CASCADE,
related_name="children",
to="services.category",
),
),
],
options={
"verbose_name_plural": "Categories",
"ordering": ["order", "name"],
},
),
migrations.AddField(
model_name="service",
name="categories",
field=models.ManyToManyField(
related_name="services", to="services.category"
),
),
]

View file

@ -1,5 +1,6 @@
from django.db import models from django.db import models
from django.core.exceptions import ValidationError from django.core.exceptions import ValidationError
from django.utils.text import slugify
def validate_image_size(value): def validate_image_size(value):
@ -8,6 +9,39 @@ def validate_image_size(value):
raise ValidationError("Maximum file size is 1MB") raise ValidationError("Maximum file size is 1MB")
class Category(models.Model):
name = models.CharField(max_length=100)
slug = models.SlugField(unique=True)
parent = models.ForeignKey(
"self", on_delete=models.CASCADE, null=True, blank=True, related_name="children"
)
description = models.TextField(blank=True)
order = models.IntegerField(default=0)
class Meta:
verbose_name_plural = "Categories"
ordering = ["order", "name"]
def __str__(self):
if self.parent:
return f"{self.parent} > {self.name}"
return self.name
def save(self, *args, **kwargs):
if not self.slug:
self.slug = slugify(self.name)
super().save(*args, **kwargs)
@property
def full_path(self):
path = [self.name]
parent = self.parent
while parent:
path.append(parent.name)
parent = parent.parent
return " > ".join(reversed(path))
class CloudProvider(models.Model): class CloudProvider(models.Model):
name = models.CharField(max_length=100) name = models.CharField(max_length=100)
description = models.TextField(blank=True) description = models.TextField(blank=True)
@ -47,6 +81,7 @@ class Service(models.Model):
description = models.TextField() description = models.TextField()
cloud_provider = models.ForeignKey(CloudProvider, on_delete=models.CASCADE) cloud_provider = models.ForeignKey(CloudProvider, on_delete=models.CASCADE)
service_level = models.ForeignKey(ServiceLevel, on_delete=models.CASCADE) service_level = models.ForeignKey(ServiceLevel, on_delete=models.CASCADE)
categories = models.ManyToManyField(Category, related_name="services")
countries = models.ManyToManyField(Country) countries = models.ManyToManyField(Country)
price = models.DecimalField(max_digits=10, decimal_places=2) price = models.DecimalField(max_digits=10, decimal_places=2)
features = models.TextField() features = models.TextField()

View file

@ -3,6 +3,12 @@
{% block content %} {% block content %}
<div class="card"> <div class="card">
<div class="card-body"> <div class="card-body">
<h5 class="card-title">Categories</h5>
{% for category in service.categories.all %}
<div class="mb-1">
<span class="badge bg-secondary">{{ category.full_path }}</span>
</div>
{% endfor %}
<div class="d-flex align-items-start mb-4"> <div class="d-flex align-items-start mb-4">
{% if service.logo %} {% if service.logo %}
<img src="{{ service.logo.url }}" alt="{{ service.name }} logo" class="me-4" style="max-height: 100px; max-width: 200px; object-fit: contain;"> <img src="{{ service.logo.url }}" alt="{{ service.name }} logo" class="me-4" style="max-height: 100px; max-width: 200px; object-fit: contain;">

View file

@ -12,6 +12,23 @@
<input type="text" class="form-control" id="search" name="search" value="{{ request.GET.search }}"> <input type="text" class="form-control" id="search" name="search" value="{{ request.GET.search }}">
</div> </div>
<div class="mb-3">
<label for="category" class="form-label">Category</label>
<select class="form-select" id="category" name="category">
<option value="">All Categories</option>
{% for category in categories %}
<option value="{{ category.id }}" {% if request.GET.category == category.id|stringformat:'i' %}selected{% endif %}>
{{ category.name }}
</option>
{% for subcategory in category.children.all %}
<option value="{{ subcategory.id }}" {% if request.GET.category == subcategory.id|stringformat:'i' %}selected{% endif %}>
&nbsp;&nbsp;&nbsp;{{ subcategory.name }}
</option>
{% endfor %}
{% endfor %}
</select>
</div>
<div class="mb-3"> <div class="mb-3">
<label for="cloud_provider" class="form-label">Cloud Provider</label> <label for="cloud_provider" class="form-label">Cloud Provider</label>
<select class="form-select" id="cloud_provider" name="cloud_provider"> <select class="form-select" id="cloud_provider" name="cloud_provider">
@ -76,6 +93,11 @@
</div> </div>
</div> </div>
<p class="card-text">{{ service.description|truncatewords:30 }}</p> <p class="card-text">{{ service.description|truncatewords:30 }}</p>
<div class="mb-2">
{% for category in service.categories.all %}
<span class="badge bg-secondary me-1">{{ category.full_path }}</span>
{% endfor %}
</div>
<p class="card-text"> <p class="card-text">
<small class="text-muted"> <small class="text-muted">
Service Level: {{ service.service_level.name }}<br> Service Level: {{ service.service_level.name }}<br>

View file

@ -1,6 +1,6 @@
from django.shortcuts import render, get_object_or_404 from django.shortcuts import render, get_object_or_404
from django.db.models import Q from django.db.models import Q
from .models import Service, CloudProvider, Country, ServiceLevel from .models import Service, CloudProvider, Country, ServiceLevel, Category
def service_list(request): def service_list(request):
@ -8,6 +8,7 @@ def service_list(request):
cloud_providers = CloudProvider.objects.all() cloud_providers = CloudProvider.objects.all()
countries = Country.objects.all() countries = Country.objects.all()
service_levels = ServiceLevel.objects.all() service_levels = ServiceLevel.objects.all()
categories = Category.objects.filter(parent=None) # Top-level categories
# Filter handling # Filter handling
if request.GET.get("cloud_provider"): if request.GET.get("cloud_provider"):
@ -19,6 +20,16 @@ def service_list(request):
if request.GET.get("service_level"): if request.GET.get("service_level"):
services = services.filter(service_level_id=request.GET.get("service_level")) services = services.filter(service_level_id=request.GET.get("service_level"))
if request.GET.get("category"):
category_id = request.GET.get("category")
category = Category.objects.get(id=category_id)
# Get all subcategories
subcategories = Category.objects.filter(parent=category)
# Filter services in this category or its subcategories
services = services.filter(
Q(categories=category) | Q(categories__in=subcategories)
).distinct()
if request.GET.get("search"): if request.GET.get("search"):
query = request.GET.get("search") query = request.GET.get("search")
services = services.filter( services = services.filter(
@ -30,6 +41,7 @@ def service_list(request):
"cloud_providers": cloud_providers, "cloud_providers": cloud_providers,
"countries": countries, "countries": countries,
"service_levels": service_levels, "service_levels": service_levels,
"categories": categories,
} }
return render(request, "services/service_list.html", context) return render(request, "services/service_list.html", context)