add consulting partners
This commit is contained in:
parent
022f0ad60f
commit
ca251218e5
8 changed files with 261 additions and 9 deletions
|
@ -1,6 +1,13 @@
|
||||||
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, Category
|
from .models import (
|
||||||
|
CloudProvider,
|
||||||
|
ConsultingPartner,
|
||||||
|
Country,
|
||||||
|
ServiceLevel,
|
||||||
|
Service,
|
||||||
|
Category,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
@admin.register(Category)
|
@admin.register(Category)
|
||||||
|
@ -42,17 +49,23 @@ class ServiceLevelAdmin(admin.ModelAdmin):
|
||||||
class ServiceAdmin(admin.ModelAdmin):
|
class ServiceAdmin(admin.ModelAdmin):
|
||||||
list_display = (
|
list_display = (
|
||||||
"name",
|
"name",
|
||||||
"slug",
|
|
||||||
"cloud_provider",
|
"cloud_provider",
|
||||||
"service_level",
|
"service_level",
|
||||||
"price",
|
"price",
|
||||||
"logo_preview",
|
"logo_preview",
|
||||||
"category_list",
|
"category_list",
|
||||||
|
"partner_list",
|
||||||
)
|
)
|
||||||
list_filter = ("cloud_provider", "service_level", "countries", "categories")
|
list_filter = (
|
||||||
|
"cloud_provider",
|
||||||
|
"service_level",
|
||||||
|
"countries",
|
||||||
|
"categories",
|
||||||
|
"consulting_partners",
|
||||||
|
)
|
||||||
|
filter_horizontal = ("countries", "categories", "consulting_partners")
|
||||||
search_fields = ("name", "description", "slug")
|
search_fields = ("name", "description", "slug")
|
||||||
prepopulated_fields = {"slug": ("name",)}
|
prepopulated_fields = {"slug": ("name",)}
|
||||||
filter_horizontal = ("countries", "categories")
|
|
||||||
|
|
||||||
def logo_preview(self, obj):
|
def logo_preview(self, obj):
|
||||||
if obj.logo:
|
if obj.logo:
|
||||||
|
@ -64,4 +77,22 @@ class ServiceAdmin(admin.ModelAdmin):
|
||||||
def category_list(self, obj):
|
def category_list(self, obj):
|
||||||
return ", ".join([cat.name for cat in obj.categories.all()])
|
return ", ".join([cat.name for cat in obj.categories.all()])
|
||||||
|
|
||||||
|
def partner_list(self, obj):
|
||||||
|
return ", ".join([partner.name for partner in obj.consulting_partners.all()])
|
||||||
|
|
||||||
|
partner_list.short_description = "Consulting Partners"
|
||||||
category_list.short_description = "Categories"
|
category_list.short_description = "Categories"
|
||||||
|
|
||||||
|
|
||||||
|
@admin.register(ConsultingPartner)
|
||||||
|
class ConsultingPartnerAdmin(admin.ModelAdmin):
|
||||||
|
list_display = ("name", "website", "logo_preview")
|
||||||
|
search_fields = ("name", "description")
|
||||||
|
prepopulated_fields = {"slug": ("name",)}
|
||||||
|
|
||||||
|
def logo_preview(self, obj):
|
||||||
|
if obj.logo:
|
||||||
|
return format_html(
|
||||||
|
'<img src="{}" style="max-height: 50px;"/>', obj.logo.url
|
||||||
|
)
|
||||||
|
return "No logo"
|
||||||
|
|
|
@ -0,0 +1,50 @@
|
||||||
|
# Generated by Django 5.1.5 on 2025-01-28 07:49
|
||||||
|
|
||||||
|
import services.models
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
("services", "0006_service_slug"),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.CreateModel(
|
||||||
|
name="ConsultingPartner",
|
||||||
|
fields=[
|
||||||
|
(
|
||||||
|
"id",
|
||||||
|
models.BigAutoField(
|
||||||
|
auto_created=True,
|
||||||
|
primary_key=True,
|
||||||
|
serialize=False,
|
||||||
|
verbose_name="ID",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
("name", models.CharField(max_length=200)),
|
||||||
|
("slug", models.SlugField(unique=True)),
|
||||||
|
("description", models.TextField(blank=True)),
|
||||||
|
(
|
||||||
|
"logo",
|
||||||
|
models.ImageField(
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
upload_to="partner_logos/",
|
||||||
|
validators=[services.models.validate_image_size],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
("website", models.URLField(blank=True)),
|
||||||
|
("created_at", models.DateTimeField(auto_now_add=True)),
|
||||||
|
("updated_at", models.DateTimeField(auto_now=True)),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name="service",
|
||||||
|
name="consulting_partners",
|
||||||
|
field=models.ManyToManyField(
|
||||||
|
blank=True, related_name="services", to="services.consultingpartner"
|
||||||
|
),
|
||||||
|
),
|
||||||
|
]
|
|
@ -68,6 +68,32 @@ class CloudProvider(models.Model):
|
||||||
return reverse("services:provider_detail", kwargs={"slug": self.slug})
|
return reverse("services:provider_detail", kwargs={"slug": self.slug})
|
||||||
|
|
||||||
|
|
||||||
|
class ConsultingPartner(models.Model):
|
||||||
|
name = models.CharField(max_length=200)
|
||||||
|
slug = models.SlugField(unique=True)
|
||||||
|
description = ProseEditorField(blank=True)
|
||||||
|
logo = models.ImageField(
|
||||||
|
upload_to="partner_logos/",
|
||||||
|
validators=[validate_image_size],
|
||||||
|
null=True,
|
||||||
|
blank=True,
|
||||||
|
)
|
||||||
|
website = models.URLField(blank=True)
|
||||||
|
created_at = models.DateTimeField(auto_now_add=True)
|
||||||
|
updated_at = models.DateTimeField(auto_now=True)
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return self.name
|
||||||
|
|
||||||
|
def save(self, *args, **kwargs):
|
||||||
|
if not self.slug:
|
||||||
|
self.slug = slugify(self.name)
|
||||||
|
super().save(*args, **kwargs)
|
||||||
|
|
||||||
|
def get_absolute_url(self):
|
||||||
|
return reverse("services:partner_detail", kwargs={"slug": self.slug})
|
||||||
|
|
||||||
|
|
||||||
class Country(models.Model):
|
class Country(models.Model):
|
||||||
name = models.CharField(max_length=100)
|
name = models.CharField(max_length=100)
|
||||||
code = models.CharField(max_length=2)
|
code = models.CharField(max_length=2)
|
||||||
|
@ -93,6 +119,9 @@ class Service(models.Model):
|
||||||
slug = models.SlugField(max_length=250, unique=True)
|
slug = models.SlugField(max_length=250, unique=True)
|
||||||
description = ProseEditorField()
|
description = ProseEditorField()
|
||||||
cloud_provider = models.ForeignKey(CloudProvider, on_delete=models.CASCADE)
|
cloud_provider = models.ForeignKey(CloudProvider, on_delete=models.CASCADE)
|
||||||
|
consulting_partners = models.ManyToManyField(
|
||||||
|
ConsultingPartner, related_name="services", blank=True
|
||||||
|
)
|
||||||
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")
|
categories = models.ManyToManyField(Category, related_name="services")
|
||||||
countries = models.ManyToManyField(Country)
|
countries = models.ManyToManyField(Country)
|
||||||
|
|
77
hub/services/templates/services/partner_detail.html
Normal file
77
hub/services/templates/services/partner_detail.html
Normal file
|
@ -0,0 +1,77 @@
|
||||||
|
{% extends 'services/base.html' %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<div class="card">
|
||||||
|
<div class="card-body">
|
||||||
|
<div class="d-flex align-items-start mb-4">
|
||||||
|
{% if partner.logo %}
|
||||||
|
<img src="{{ partner.logo.url }}" alt="{{ partner.name }} logo" class="me-4"
|
||||||
|
style="max-height: 120px; max-width: 240px; object-fit: contain;">
|
||||||
|
{% endif %}
|
||||||
|
<div>
|
||||||
|
<h2 class="card-title mb-3">{{ partner.name }}</h2>
|
||||||
|
{% if partner.website %}
|
||||||
|
<a href="{{ partner.website }}" class="btn btn-outline-primary mb-3" target="_blank">
|
||||||
|
Visit Website
|
||||||
|
</a>
|
||||||
|
{% endif %}
|
||||||
|
<div class="rich-text-content">
|
||||||
|
{{ partner.description|safe }}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<h3 class="mb-4">Available Services</h3>
|
||||||
|
<div class="row row-cols-1 row-cols-md-2 g-4">
|
||||||
|
{% for service in services %}
|
||||||
|
<div class="col">
|
||||||
|
<div class="card h-100">
|
||||||
|
<div class="card-body">
|
||||||
|
<div class="d-flex align-items-center mb-3">
|
||||||
|
{% if service.logo %}
|
||||||
|
<img src="{{ service.logo.url }}" alt="{{ service.name }} logo"
|
||||||
|
class="me-3" style="max-height: 50px; max-width: 100px; object-fit: contain;">
|
||||||
|
{% endif %}
|
||||||
|
<div>
|
||||||
|
<h5 class="card-title mb-0">{{ service.name }}</h5>
|
||||||
|
<div class="d-flex align-items-center mt-2">
|
||||||
|
{% if service.cloud_provider.logo %}
|
||||||
|
<img src="{{ service.cloud_provider.logo.url }}" alt="{{ service.cloud_provider.name }} logo"
|
||||||
|
class="me-2" style="max-height: 25px; max-width: 50px; object-fit: contain;">
|
||||||
|
{% endif %}
|
||||||
|
<h6 class="card-subtitle text-muted mb-0">
|
||||||
|
<a href="{{ service.cloud_provider.get_absolute_url }}"
|
||||||
|
class="text-decoration-none text-muted">{{ service.cloud_provider.name }}</a>
|
||||||
|
</h6>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="card-text description-preview mb-3">
|
||||||
|
{{ service.description|safe|truncatewords_html:30 }}
|
||||||
|
</div>
|
||||||
|
<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">
|
||||||
|
<small class="text-muted">
|
||||||
|
Service Level: {{ service.service_level.name }}<br>
|
||||||
|
Price: ${{ service.price }}
|
||||||
|
</small>
|
||||||
|
</p>
|
||||||
|
<a href="{{ service.get_absolute_url }}" class="btn btn-primary">View Details</a>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% empty %}
|
||||||
|
<div class="col-12">
|
||||||
|
<div class="alert alert-info">
|
||||||
|
No services available from this partner yet.
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endfor %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endblock %}
|
|
@ -63,4 +63,35 @@
|
||||||
<a href="{% url 'services:service_list' %}" class="btn btn-secondary">Back to Services</a>
|
<a href="{% url 'services:service_list' %}" class="btn btn-secondary">Back to Services</a>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
{% if service.consulting_partners.exists %}
|
||||||
|
<div class="card mb-3">
|
||||||
|
<div class="card-body">
|
||||||
|
<h5 class="card-title">Consulting Partners</h5>
|
||||||
|
<div class="row row-cols-1 row-cols-md-2 g-4">
|
||||||
|
{% for partner in service.consulting_partners.all %}
|
||||||
|
<div class="col">
|
||||||
|
<div class="d-flex align-items-center">
|
||||||
|
{% if partner.logo %}
|
||||||
|
<img src="{{ partner.logo.url }}" alt="{{ partner.name }} logo"
|
||||||
|
class="me-2" style="max-height: 40px; max-width: 80px; object-fit: contain;">
|
||||||
|
{% endif %}
|
||||||
|
<div>
|
||||||
|
<a href="{{ partner.get_absolute_url }}" class="text-decoration-none">
|
||||||
|
{{ partner.name }}
|
||||||
|
</a>
|
||||||
|
{% if partner.website %}
|
||||||
|
<br>
|
||||||
|
<a href="{{ partner.website }}" class="small text-muted" target="_blank">
|
||||||
|
Visit Website
|
||||||
|
</a>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endfor %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
|
@ -40,6 +40,18 @@
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</select>
|
</select>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div class="mb-3">
|
||||||
|
<label for="consulting_partner" class="form-label">Consulting Partner</label>
|
||||||
|
<select class="form-select" id="consulting_partner" name="consulting_partner">
|
||||||
|
<option value="">All Partners</option>
|
||||||
|
{% for partner in consulting_partners %}
|
||||||
|
<option value="{{ partner.id }}" {% if request.GET.consulting_partner == partner.id|stringformat:'i' %}selected{% endif %}>
|
||||||
|
{{ partner.name }}
|
||||||
|
</option>
|
||||||
|
{% endfor %}
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="mb-3">
|
<div class="mb-3">
|
||||||
<label for="country" class="form-label">Country</label>
|
<label for="country" class="form-label">Country</label>
|
||||||
|
|
|
@ -9,4 +9,5 @@ urlpatterns = [
|
||||||
path("service/<slug:slug>/interest/", views.create_lead, name="create_lead"),
|
path("service/<slug:slug>/interest/", views.create_lead, name="create_lead"),
|
||||||
path("service/<slug:slug>/thank-you/", views.thank_you, name="thank_you"),
|
path("service/<slug:slug>/thank-you/", views.thank_you, name="thank_you"),
|
||||||
path("provider/<slug:slug>/", views.provider_detail, name="provider_detail"),
|
path("provider/<slug:slug>/", views.provider_detail, name="provider_detail"),
|
||||||
|
path("partner/<slug:slug>/", views.partner_detail, name="partner_detail"),
|
||||||
]
|
]
|
||||||
|
|
|
@ -4,7 +4,14 @@ from django.conf import settings
|
||||||
from django.shortcuts import render, get_object_or_404, redirect
|
from django.shortcuts import render, get_object_or_404, redirect
|
||||||
from django.contrib import messages
|
from django.contrib import messages
|
||||||
from django.db.models import Q
|
from django.db.models import Q
|
||||||
from .models import Service, CloudProvider, Country, ServiceLevel, Category
|
from .models import (
|
||||||
|
Service,
|
||||||
|
CloudProvider,
|
||||||
|
ConsultingPartner,
|
||||||
|
Country,
|
||||||
|
ServiceLevel,
|
||||||
|
Category,
|
||||||
|
)
|
||||||
|
|
||||||
from .forms import LeadForm
|
from .forms import LeadForm
|
||||||
from .odoo import OdooAPI
|
from .odoo import OdooAPI
|
||||||
|
@ -17,9 +24,9 @@ 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
|
categories = Category.objects.filter(parent=None)
|
||||||
|
consulting_partners = ConsultingPartner.objects.all()
|
||||||
|
|
||||||
# Filter handling
|
|
||||||
if request.GET.get("cloud_provider"):
|
if request.GET.get("cloud_provider"):
|
||||||
services = services.filter(cloud_provider_id=request.GET.get("cloud_provider"))
|
services = services.filter(cloud_provider_id=request.GET.get("cloud_provider"))
|
||||||
|
|
||||||
|
@ -32,13 +39,16 @@ def service_list(request):
|
||||||
if request.GET.get("category"):
|
if request.GET.get("category"):
|
||||||
category_id = request.GET.get("category")
|
category_id = request.GET.get("category")
|
||||||
category = Category.objects.get(id=category_id)
|
category = Category.objects.get(id=category_id)
|
||||||
# Get all subcategories
|
|
||||||
subcategories = Category.objects.filter(parent=category)
|
subcategories = Category.objects.filter(parent=category)
|
||||||
# Filter services in this category or its subcategories
|
|
||||||
services = services.filter(
|
services = services.filter(
|
||||||
Q(categories=category) | Q(categories__in=subcategories)
|
Q(categories=category) | Q(categories__in=subcategories)
|
||||||
).distinct()
|
).distinct()
|
||||||
|
|
||||||
|
if request.GET.get("consulting_partner"):
|
||||||
|
services = services.filter(
|
||||||
|
consulting_partners__id=request.GET.get("consulting_partner")
|
||||||
|
)
|
||||||
|
|
||||||
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(
|
||||||
|
@ -51,6 +61,7 @@ def service_list(request):
|
||||||
"countries": countries,
|
"countries": countries,
|
||||||
"service_levels": service_levels,
|
"service_levels": service_levels,
|
||||||
"categories": categories,
|
"categories": categories,
|
||||||
|
"consulting_partners": consulting_partners,
|
||||||
}
|
}
|
||||||
return render(request, "services/service_list.html", context)
|
return render(request, "services/service_list.html", context)
|
||||||
|
|
||||||
|
@ -70,6 +81,16 @@ def provider_detail(request, slug):
|
||||||
return render(request, "services/provider_detail.html", context)
|
return render(request, "services/provider_detail.html", context)
|
||||||
|
|
||||||
|
|
||||||
|
def partner_detail(request, slug):
|
||||||
|
partner = get_object_or_404(ConsultingPartner, slug=slug)
|
||||||
|
services = Service.objects.filter(consulting_partners=partner)
|
||||||
|
return render(
|
||||||
|
request,
|
||||||
|
"services/partner_detail.html",
|
||||||
|
{"partner": partner, "services": services},
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
def thank_you(request, slug):
|
def thank_you(request, slug):
|
||||||
service = get_object_or_404(Service, slug=slug)
|
service = get_object_or_404(Service, slug=slug)
|
||||||
return render(request, "services/thank_you.html", {"service": service})
|
return render(request, "services/thank_you.html", {"service": service})
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue