wagtail-newsletter-django-backend 0.0.2__tar.gz → 0.2.0__tar.gz

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (72) hide show
  1. {wagtail_newsletter_django_backend-0.0.2 → wagtail_newsletter_django_backend-0.2.0}/.gitignore +2 -1
  2. {wagtail_newsletter_django_backend-0.0.2 → wagtail_newsletter_django_backend-0.2.0}/PKG-INFO +7 -5
  3. wagtail_newsletter_django_backend-0.2.0/demo/manage.py +22 -0
  4. wagtail_newsletter_django_backend-0.2.0/demo/pyproject.toml +35 -0
  5. wagtail_newsletter_django_backend-0.2.0/demo/src/demo/__init__.py +4 -0
  6. wagtail_newsletter_django_backend-0.2.0/demo/src/demo/blog/apps.py +11 -0
  7. wagtail_newsletter_django_backend-0.2.0/demo/src/demo/blog/migrations/0001_initial.py +80 -0
  8. wagtail_newsletter_django_backend-0.2.0/demo/src/demo/blog/migrations/0002_newslettersignup.py +45 -0
  9. wagtail_newsletter_django_backend-0.2.0/demo/src/demo/blog/migrations/__init__.py +0 -0
  10. wagtail_newsletter_django_backend-0.2.0/demo/src/demo/blog/models.py +62 -0
  11. wagtail_newsletter_django_backend-0.2.0/demo/src/demo/blog/signals.py +13 -0
  12. wagtail_newsletter_django_backend-0.2.0/demo/src/demo/blog/templates/blog/blog_index_page.html +19 -0
  13. wagtail_newsletter_django_backend-0.2.0/demo/src/demo/blog/templates/blog/blog_page.html +17 -0
  14. wagtail_newsletter_django_backend-0.2.0/demo/src/demo/blog/templates/blog/blog_page_newsletter.html +19 -0
  15. wagtail_newsletter_django_backend-0.2.0/demo/src/demo/blog/templates/blog/newsletter_signup.html +21 -0
  16. wagtail_newsletter_django_backend-0.2.0/demo/src/demo/home/__init__.py +0 -0
  17. wagtail_newsletter_django_backend-0.2.0/demo/src/demo/home/apps.py +6 -0
  18. wagtail_newsletter_django_backend-0.2.0/demo/src/demo/home/migrations/0001_initial.py +36 -0
  19. wagtail_newsletter_django_backend-0.2.0/demo/src/demo/home/migrations/__init__.py +0 -0
  20. wagtail_newsletter_django_backend-0.2.0/demo/src/demo/home/models.py +7 -0
  21. wagtail_newsletter_django_backend-0.2.0/demo/src/demo/home/static/css/welcome_page.css +184 -0
  22. wagtail_newsletter_django_backend-0.2.0/demo/src/demo/home/templates/home/home_page.html +21 -0
  23. wagtail_newsletter_django_backend-0.2.0/demo/src/demo/home/templates/home/welcome_page.html +52 -0
  24. wagtail_newsletter_django_backend-0.2.0/demo/src/demo/home/tests.py +43 -0
  25. wagtail_newsletter_django_backend-0.2.0/demo/src/demo/search/__init__.py +0 -0
  26. wagtail_newsletter_django_backend-0.2.0/demo/src/demo/search/apps.py +5 -0
  27. wagtail_newsletter_django_backend-0.2.0/demo/src/demo/search/templates/search/search.html +38 -0
  28. wagtail_newsletter_django_backend-0.2.0/demo/src/demo/search/views.py +46 -0
  29. wagtail_newsletter_django_backend-0.2.0/demo/src/demo/settings/__init__.py +0 -0
  30. wagtail_newsletter_django_backend-0.2.0/demo/src/demo/settings/base.py +195 -0
  31. wagtail_newsletter_django_backend-0.2.0/demo/src/demo/settings/dev.py +24 -0
  32. wagtail_newsletter_django_backend-0.2.0/demo/src/demo/settings/production.py +14 -0
  33. wagtail_newsletter_django_backend-0.2.0/demo/src/demo/static/css/demo.css +0 -0
  34. wagtail_newsletter_django_backend-0.2.0/demo/src/demo/static/js/demo.js +0 -0
  35. wagtail_newsletter_django_backend-0.2.0/demo/src/demo/templates/404.html +11 -0
  36. wagtail_newsletter_django_backend-0.2.0/demo/src/demo/templates/500.html +13 -0
  37. wagtail_newsletter_django_backend-0.2.0/demo/src/demo/templates/base.html +54 -0
  38. wagtail_newsletter_django_backend-0.2.0/demo/src/demo/urls.py +36 -0
  39. wagtail_newsletter_django_backend-0.2.0/demo/src/demo/wsgi.py +16 -0
  40. wagtail_newsletter_django_backend-0.2.0/demo/uv.lock +870 -0
  41. {wagtail_newsletter_django_backend-0.0.2 → wagtail_newsletter_django_backend-0.2.0}/pyproject.toml +6 -3
  42. {wagtail_newsletter_django_backend-0.0.2 → wagtail_newsletter_django_backend-0.2.0}/src/wagtail_newsletter_django_backend/__init__.py +1 -1
  43. wagtail_newsletter_django_backend-0.2.0/src/wagtail_newsletter_django_backend/admin.py +3 -0
  44. wagtail_newsletter_django_backend-0.2.0/src/wagtail_newsletter_django_backend/admin_viewsets.py +58 -0
  45. {wagtail_newsletter_django_backend-0.0.2 → wagtail_newsletter_django_backend-0.2.0}/src/wagtail_newsletter_django_backend/apps.py +3 -1
  46. wagtail_newsletter_django_backend-0.2.0/src/wagtail_newsletter_django_backend/campaign_backend.py +180 -0
  47. wagtail_newsletter_django_backend-0.2.0/src/wagtail_newsletter_django_backend/forms.py +97 -0
  48. wagtail_newsletter_django_backend-0.2.0/src/wagtail_newsletter_django_backend/management/__init__.py +0 -0
  49. wagtail_newsletter_django_backend-0.2.0/src/wagtail_newsletter_django_backend/management/commands/__init__.py +0 -0
  50. wagtail_newsletter_django_backend-0.2.0/src/wagtail_newsletter_django_backend/management/commands/send_scheduled_campaigns.py +8 -0
  51. wagtail_newsletter_django_backend-0.2.0/src/wagtail_newsletter_django_backend/migrations/0001_initial.py +291 -0
  52. wagtail_newsletter_django_backend-0.2.0/src/wagtail_newsletter_django_backend/migrations/__init__.py +0 -0
  53. wagtail_newsletter_django_backend-0.2.0/src/wagtail_newsletter_django_backend/models.py +317 -0
  54. wagtail_newsletter_django_backend-0.2.0/src/wagtail_newsletter_django_backend/templates/wagtail_newsletter_django_backend/manage.html +16 -0
  55. wagtail_newsletter_django_backend-0.2.0/src/wagtail_newsletter_django_backend/templates/wagtail_newsletter_django_backend/unsubscribe.html +22 -0
  56. wagtail_newsletter_django_backend-0.2.0/src/wagtail_newsletter_django_backend/tests.py +3 -0
  57. wagtail_newsletter_django_backend-0.2.0/src/wagtail_newsletter_django_backend/urls.py +17 -0
  58. wagtail_newsletter_django_backend-0.2.0/src/wagtail_newsletter_django_backend/views.py +79 -0
  59. wagtail_newsletter_django_backend-0.2.0/src/wagtail_newsletter_django_backend/wagtail_hooks.py +7 -0
  60. {wagtail_newsletter_django_backend-0.0.2 → wagtail_newsletter_django_backend-0.2.0}/uv.lock +12 -1
  61. wagtail_newsletter_django_backend-0.0.2/src/wagtail_newsletter_django_backend/models.py +0 -3
  62. {wagtail_newsletter_django_backend-0.0.2 → wagtail_newsletter_django_backend-0.2.0}/.readthedocs.yaml +0 -0
  63. {wagtail_newsletter_django_backend-0.0.2 → wagtail_newsletter_django_backend-0.2.0}/LICENSE +0 -0
  64. {wagtail_newsletter_django_backend-0.0.2 → wagtail_newsletter_django_backend-0.2.0}/README.md +0 -0
  65. {wagtail_newsletter_django_backend-0.0.2/src/wagtail_newsletter_django_backend/migrations → wagtail_newsletter_django_backend-0.2.0/demo/src/demo/blog}/__init__.py +0 -0
  66. {wagtail_newsletter_django_backend-0.0.2/src/wagtail_newsletter_django_backend → wagtail_newsletter_django_backend-0.2.0/demo/src/demo/blog}/admin.py +0 -0
  67. {wagtail_newsletter_django_backend-0.0.2/src/wagtail_newsletter_django_backend → wagtail_newsletter_django_backend-0.2.0/demo/src/demo/blog}/tests.py +0 -0
  68. {wagtail_newsletter_django_backend-0.0.2/src/wagtail_newsletter_django_backend → wagtail_newsletter_django_backend-0.2.0/demo/src/demo/blog}/views.py +0 -0
  69. {wagtail_newsletter_django_backend-0.0.2 → wagtail_newsletter_django_backend-0.2.0}/docs/Makefile +0 -0
  70. {wagtail_newsletter_django_backend-0.0.2 → wagtail_newsletter_django_backend-0.2.0}/docs/conf.py +0 -0
  71. {wagtail_newsletter_django_backend-0.0.2 → wagtail_newsletter_django_backend-0.2.0}/docs/index.rst +0 -0
  72. {wagtail_newsletter_django_backend-0.0.2 → wagtail_newsletter_django_backend-0.2.0}/docs/make.bat +0 -0
@@ -60,7 +60,7 @@ cover/
60
60
  *.log
61
61
  local_settings.py
62
62
  db.sqlite3
63
- db.sqlite3-journal
63
+ db.sqlite3-*
64
64
 
65
65
  # Flask stuff:
66
66
  instance/
@@ -162,3 +162,4 @@ cython_debug/
162
162
  # option (not recommended) you can uncomment the following to ignore the entire idea folder.
163
163
  #.idea/
164
164
 
165
+ demo/media/
@@ -1,14 +1,16 @@
1
- Metadata-Version: 2.3
1
+ Metadata-Version: 2.4
2
2
  Name: wagtail-newsletter-django-backend
3
- Version: 0.0.2
3
+ Version: 0.2.0
4
4
  Summary: An internal Django backend to wagtail-newsletter.
5
5
  Author-email: "Taylor C. Richberger" <taylor@axfive.net>
6
6
  Maintainer-email: "Taylor C. Richberger" <taylor@axfive.net>
7
7
  Requires-Python: >= 3.10
8
8
  Description-Content-Type: text/markdown
9
- Requires-Dist: Django>=5.2,<5.3
10
- Requires-Dist: wagtail>=7.1,<7.2
11
- Requires-Dist: wagtail-newsletter>=0.2.2,<0.3.0
9
+ License-File: LICENSE
10
+ Requires-Dist: Django >=5.2, <5.3
11
+ Requires-Dist: wagtail >=7.1, <7.2
12
+ Requires-Dist: wagtail-newsletter >=0.2.2, <0.3.0
13
+ Requires-Dist: html2text >= 2025.4.15
12
14
  Project-URL: documentation, https://wagtail-newsletter-django-backend.readthedocs.io/en/latest/
13
15
  Project-URL: repository, https://forge.axfive.net/Taylor/wagtail-newsletter-django-backend
14
16
 
@@ -0,0 +1,22 @@
1
+ #!/usr/bin/env python
2
+ """Django's command-line utility for administrative tasks."""
3
+ import os
4
+ import sys
5
+
6
+
7
+ def main():
8
+ """Run administrative tasks."""
9
+ os.environ.setdefault("DJANGO_SETTINGS_MODULE", "demo.settings.dev")
10
+ try:
11
+ from django.core.management import execute_from_command_line
12
+ except ImportError as exc:
13
+ raise ImportError(
14
+ "Couldn't import Django. Are you sure it's installed and "
15
+ "available on your PYTHONPATH environment variable? Did you "
16
+ "forget to activate a virtual environment?"
17
+ ) from exc
18
+ execute_from_command_line(sys.argv)
19
+
20
+
21
+ if __name__ == '__main__':
22
+ main()
@@ -0,0 +1,35 @@
1
+ [build-system]
2
+ requires = ['flit_core >= 3.2, <4']
3
+ build-backend = 'flit_core.buildapi'
4
+
5
+ [project]
6
+ name = "demo"
7
+ dynamic = ['version', 'description']
8
+ dependencies = [
9
+ 'Django>=5.2,<5.3',
10
+ 'wagtail>=7.1,<7.2',
11
+ "wagtail-newsletter[mrml]>=0.2.2,<0.3.0",
12
+ "wagtail-newsletter-django-backend",
13
+ 'python-dotenv',
14
+ "django-extensions",
15
+ 'requests >= 2.32.5, < 3',
16
+ ]
17
+ requires-python = ">=3.10"
18
+
19
+ [dependency-groups]
20
+ dev = [
21
+ "basedpyright>=1.31.2",
22
+ "django-types>=0.22.0",
23
+ "ptpython",
24
+ ]
25
+
26
+ [[project.authors]]
27
+ name = 'Taylor C. Richberger'
28
+ email = 'taylor@axfive.net'
29
+
30
+ [[project.maintainers]]
31
+ name = 'Taylor C. Richberger'
32
+ email = 'taylor@axfive.net'
33
+
34
+ [tool.uv.sources]
35
+ wagtail-newsletter-django-backend = { path = "../", editable = true }
@@ -0,0 +1,4 @@
1
+ '''A demo site for wagtail-newsletter-django-backend.
2
+ '''
3
+
4
+ __version__ = '0.0.1'
@@ -0,0 +1,11 @@
1
+ from typing import override
2
+ from django.apps import AppConfig
3
+
4
+
5
+ class BlogConfig(AppConfig):
6
+ default_auto_field = "django.db.models.BigAutoField"
7
+ name = "demo.blog"
8
+
9
+ @override
10
+ def ready(self):
11
+ from . import signals
@@ -0,0 +1,80 @@
1
+ # Generated by Django 5.2.5 on 2025-09-01 20:37
2
+
3
+ import django.db.models.deletion
4
+ import wagtail.fields
5
+ from django.db import migrations, models
6
+
7
+
8
+ class Migration(migrations.Migration):
9
+
10
+ initial = True
11
+
12
+ dependencies = [
13
+ ("wagtail_newsletter", "0001_initial"),
14
+ ("wagtailcore", "0095_groupsitepermission"),
15
+ ]
16
+
17
+ operations = [
18
+ migrations.CreateModel(
19
+ name="BlogIndexPage",
20
+ fields=[
21
+ (
22
+ "page_ptr",
23
+ models.OneToOneField(
24
+ auto_created=True,
25
+ on_delete=django.db.models.deletion.CASCADE,
26
+ parent_link=True,
27
+ primary_key=True,
28
+ serialize=False,
29
+ to="wagtailcore.page",
30
+ ),
31
+ ),
32
+ ("intro", wagtail.fields.RichTextField(blank=True)),
33
+ ],
34
+ options={
35
+ "abstract": False,
36
+ },
37
+ bases=("wagtailcore.page",),
38
+ ),
39
+ migrations.CreateModel(
40
+ name="BlogPage",
41
+ fields=[
42
+ (
43
+ "page_ptr",
44
+ models.OneToOneField(
45
+ auto_created=True,
46
+ on_delete=django.db.models.deletion.CASCADE,
47
+ parent_link=True,
48
+ primary_key=True,
49
+ serialize=False,
50
+ to="wagtailcore.page",
51
+ ),
52
+ ),
53
+ (
54
+ "newsletter_subject",
55
+ models.CharField(
56
+ blank=True,
57
+ help_text="Subject for the newsletter. Defaults to page title if blank.",
58
+ max_length=1000,
59
+ ),
60
+ ),
61
+ ("newsletter_campaign", models.CharField(blank=True, max_length=1000)),
62
+ ("date", models.DateField(verbose_name="Post date")),
63
+ ("intro", models.CharField(max_length=250)),
64
+ ("body", wagtail.fields.RichTextField(blank=True)),
65
+ (
66
+ "newsletter_recipients",
67
+ models.ForeignKey(
68
+ blank=True,
69
+ null=True,
70
+ on_delete=django.db.models.deletion.SET_NULL,
71
+ to="wagtail_newsletter.newsletterrecipients",
72
+ ),
73
+ ),
74
+ ],
75
+ options={
76
+ "abstract": False,
77
+ },
78
+ bases=("wagtailcore.page",),
79
+ ),
80
+ ]
@@ -0,0 +1,45 @@
1
+ # Generated by Django 5.2.5 on 2025-09-01 22:31
2
+
3
+ import django.db.models.deletion
4
+ import wagtail.fields
5
+ from django.db import migrations, models
6
+
7
+
8
+ class Migration(migrations.Migration):
9
+
10
+ dependencies = [
11
+ ("blog", "0001_initial"),
12
+ ("wagtail_newsletter_django_backend", "0001_initial"),
13
+ ("wagtailcore", "0095_groupsitepermission"),
14
+ ]
15
+
16
+ operations = [
17
+ migrations.CreateModel(
18
+ name="NewsletterSignup",
19
+ fields=[
20
+ (
21
+ "page_ptr",
22
+ models.OneToOneField(
23
+ auto_created=True,
24
+ on_delete=django.db.models.deletion.CASCADE,
25
+ parent_link=True,
26
+ primary_key=True,
27
+ serialize=False,
28
+ to="wagtailcore.page",
29
+ ),
30
+ ),
31
+ ("intro", wagtail.fields.RichTextField(blank=True)),
32
+ (
33
+ "audience",
34
+ models.ForeignKey(
35
+ on_delete=django.db.models.deletion.CASCADE,
36
+ to="wagtail_newsletter_django_backend.audience",
37
+ ),
38
+ ),
39
+ ],
40
+ options={
41
+ "abstract": False,
42
+ },
43
+ bases=("wagtailcore.page",),
44
+ ),
45
+ ]
@@ -0,0 +1,62 @@
1
+ from django.contrib import messages
2
+ from django.db import models
3
+ from django.shortcuts import redirect
4
+ from django.conf import settings
5
+ from wagtail import models as wt_models
6
+ from wagtail import fields as wt_fields
7
+ from wagtail.models.pages import HttpRequest, HttpResponse
8
+ from wagtail_newsletter.models import NewsletterPageMixin
9
+ import requests
10
+ from wagtail_newsletter_django_backend.forms import SignupForm
11
+ from wagtail_newsletter_django_backend.models import Audience, Subscriber
12
+
13
+ class BlogIndexPage(wt_models.Page):
14
+ intro = wt_fields.RichTextField(blank=True)
15
+
16
+ content_panels = wt_models.Page.content_panels + ["intro"]
17
+
18
+ class BlogPage(NewsletterPageMixin, wt_models.Page):
19
+ date = models.DateField("Post date")
20
+ intro = models.CharField(max_length=250)
21
+ body = wt_fields.RichTextField(blank=True)
22
+
23
+ content_panels = wt_models.Page.content_panels + ["date", "intro", "body"]
24
+
25
+ newsletter_template = 'blog/blog_page_newsletter.html'
26
+
27
+ class NewsletterSignup(wt_models.Page):
28
+ intro = wt_fields.RichTextField(blank=True)
29
+
30
+ audience = models.ForeignKey(Audience, blank=False, null=False, on_delete=models.CASCADE)
31
+
32
+ content_panels = wt_models.Page.content_panels + ["intro", 'audience']
33
+
34
+ form: SignupForm | None = None
35
+
36
+ def serve(self, request: HttpRequest) -> HttpResponse:
37
+ if request.method != 'POST':
38
+ self.form = SignupForm(initial={'audience': self.audience}, audience=self.audience)
39
+ return super().serve(request=request)
40
+ self.form = SignupForm(request.POST, audience=self.audience)
41
+ token = request.POST['h-captcha-response']
42
+ params = {
43
+ "secret": settings.HCAPTCHA_SECRET,
44
+ "response": token
45
+ }
46
+ # TODO: add remoteip: https://docs.hcaptcha.com/#server
47
+ response = requests.post("https://hcaptcha.com/siteverify", data=params)
48
+ response.raise_for_status()
49
+ data = response.json()
50
+ if data['success']:
51
+ if self.form.is_valid():
52
+ _ = self.form.save()
53
+ messages.success(request, 'You have successfully signed up for the newsletter, and a welcome email has been sent. You will not receive any messages until you validate your email address.')
54
+ else:
55
+ messages.error(request, 'form failed validation')
56
+ else:
57
+ messages.error(request, 'Captcha validation failed. Error codes: ' + '\n'.join(data['error-codes']))
58
+ return super().serve(request=request)
59
+
60
+ def hcaptcha_site_id(self):
61
+ return settings.HCAPTCHA_SITE_ID
62
+
@@ -0,0 +1,13 @@
1
+ import sqlite3
2
+ from typing import Any
3
+ from django.db.backends.signals import connection_created
4
+ from django.dispatch import receiver
5
+
6
+ @receiver(connection_created)
7
+ def sqlite_pragmas(sender, connection: Any, **kwargs):
8
+ """Enable integrity constraint with sqlite."""
9
+ if connection.vendor == 'sqlite':
10
+ cursor = connection.cursor()
11
+ _ = cursor.execute('PRAGMA foreign_keys = ON')
12
+ _ = cursor.execute('PRAGMA journal_mode = WAL')
13
+ _ = cursor.execute('PRAGMA synchronous = NORMAL')
@@ -0,0 +1,19 @@
1
+ {% extends "base.html" %}
2
+
3
+ {% load wagtailcore_tags %}
4
+
5
+ {% block body_class %}template-blogindexpage{% endblock %}
6
+
7
+ {% block content %}
8
+ <h1>{{ page.title }}</h1>
9
+
10
+ <div class="intro">{{ page.intro|richtext }}</div>
11
+
12
+ {% for post in page.get_children %}
13
+ <h2><a href="{% pageurl post %}">{{ post.title }}</a></h2>
14
+ {{ post.specific.intro }}
15
+ {{ post.specific.body|richtext }}
16
+ {% endfor %}
17
+
18
+ {% endblock %}
19
+
@@ -0,0 +1,17 @@
1
+ {% extends "base.html" %}
2
+
3
+ {% load wagtailcore_tags %}
4
+
5
+ {% block body_class %}template-blogpage{% endblock %}
6
+
7
+ {% block content %}
8
+ <h1>{{ page.title }}</h1>
9
+ <p class="meta">{{ page.date }}</p>
10
+
11
+ <div class="intro">{{ page.intro }}</div>
12
+
13
+ {{ page.body|richtext }}
14
+
15
+ <p><a href="{{ page.get_parent.url }}">Return to blog</a></p>
16
+
17
+ {% endblock %}
@@ -0,0 +1,19 @@
1
+ {% load wagtail_newsletter %}
2
+
3
+ {% mrml %}
4
+ <mjml>
5
+ <mj-body>
6
+ <mj-text>
7
+ <h1>{{ page.title }}</h1>
8
+ <p><a href="{{ page.full_url }}">Read this in a web browser.</a></p>
9
+ <div class="intro">{{ page.intro }}</div>
10
+ </mj-text>
11
+ {{ page.body|newsletter_richtext }}
12
+ <mj-text>
13
+ <p><a href="[[unsubscribe]]">Unsubscribe from all further emails.</a></p>
14
+ <p><a href="[[manage]]">Manage your subscription.</a></p>
15
+ </mj-text>
16
+ </mj-body>
17
+ </mjml>
18
+ {% endmrml %}
19
+
@@ -0,0 +1,21 @@
1
+ {% extends "base.html" %}
2
+
3
+ {% load wagtailcore_tags %}
4
+
5
+ {% block body_class %}template-blogindexpage{% endblock %}
6
+ {% block extra_js %}
7
+ <script src="https://js.hcaptcha.com/1/api.js" async defer></script>
8
+ {% endblock %}
9
+
10
+ {% block content %}
11
+ <h1>{{ page.title }}</h1>
12
+
13
+ <div class="intro">{{ page.intro|richtext }}</div>
14
+
15
+ <form method="POST">{% csrf_token %}
16
+ {{ page.form.as_p }}
17
+ <div class="h-captcha" data-sitekey="{{ page.hcaptcha_site_id }}"></div>
18
+ <input type="submit" value="Sign Up">
19
+ </form>
20
+ {% endblock %}
21
+
@@ -0,0 +1,6 @@
1
+ from django.apps import AppConfig
2
+
3
+
4
+ class HomeConfig(AppConfig):
5
+ default_auto_field = "django.db.models.BigAutoField"
6
+ name = "demo.home"
@@ -0,0 +1,36 @@
1
+ # Generated by Django 5.2.5 on 2025-09-01 20:37
2
+
3
+ import django.db.models.deletion
4
+ from django.db import migrations, models
5
+
6
+
7
+ class Migration(migrations.Migration):
8
+
9
+ initial = True
10
+
11
+ dependencies = [
12
+ ("wagtailcore", "0095_groupsitepermission"),
13
+ ]
14
+
15
+ operations = [
16
+ migrations.CreateModel(
17
+ name="HomePage",
18
+ fields=[
19
+ (
20
+ "page_ptr",
21
+ models.OneToOneField(
22
+ auto_created=True,
23
+ on_delete=django.db.models.deletion.CASCADE,
24
+ parent_link=True,
25
+ primary_key=True,
26
+ serialize=False,
27
+ to="wagtailcore.page",
28
+ ),
29
+ ),
30
+ ],
31
+ options={
32
+ "abstract": False,
33
+ },
34
+ bases=("wagtailcore.page",),
35
+ ),
36
+ ]
@@ -0,0 +1,7 @@
1
+ from django.db import models
2
+
3
+ from wagtail.models import Page
4
+
5
+
6
+ class HomePage(Page):
7
+ pass
@@ -0,0 +1,184 @@
1
+ html {
2
+ box-sizing: border-box;
3
+ }
4
+
5
+ *,
6
+ *:before,
7
+ *:after {
8
+ box-sizing: inherit;
9
+ }
10
+
11
+ body {
12
+ max-width: 960px;
13
+ min-height: 100vh;
14
+ margin: 0 auto;
15
+ padding: 0 15px;
16
+ color: #231f20;
17
+ font-family: 'Helvetica Neue', 'Segoe UI', Arial, sans-serif;
18
+ line-height: 1.25;
19
+ }
20
+
21
+ a {
22
+ background-color: transparent;
23
+ color: #308282;
24
+ text-decoration: underline;
25
+ }
26
+
27
+ a:hover {
28
+ color: #ea1b10;
29
+ }
30
+
31
+ h1,
32
+ h2,
33
+ h3,
34
+ h4,
35
+ h5,
36
+ p,
37
+ ul {
38
+ padding: 0;
39
+ margin: 0;
40
+ font-weight: 400;
41
+ }
42
+
43
+ svg:not(:root) {
44
+ overflow: hidden;
45
+ }
46
+
47
+ .header {
48
+ display: flex;
49
+ justify-content: space-between;
50
+ align-items: center;
51
+ padding-top: 20px;
52
+ padding-bottom: 10px;
53
+ border-bottom: 1px solid #e6e6e6;
54
+ }
55
+
56
+ .logo {
57
+ width: 150px;
58
+ margin-inline-end: 20px;
59
+ }
60
+
61
+ .logo a {
62
+ display: block;
63
+ }
64
+
65
+ .figure-logo {
66
+ max-width: 150px;
67
+ max-height: 55.1px;
68
+ }
69
+
70
+ .release-notes {
71
+ font-size: 14px;
72
+ }
73
+
74
+ .main {
75
+ padding: 40px 0;
76
+ margin: 0 auto;
77
+ text-align: center;
78
+ }
79
+
80
+ .figure-space {
81
+ max-width: 265px;
82
+ }
83
+
84
+ @keyframes pos {
85
+ 0%, 100% {
86
+ transform: rotate(-6deg);
87
+ }
88
+ 50% {
89
+ transform: rotate(6deg);
90
+ }
91
+ }
92
+
93
+ .egg {
94
+ fill: #43b1b0;
95
+ animation: pos 3s ease infinite;
96
+ transform: translateY(50px);
97
+ transform-origin: 50% 80%;
98
+ }
99
+
100
+ .main-text {
101
+ max-width: 400px;
102
+ margin: 5px auto;
103
+ }
104
+
105
+ .main-text h1 {
106
+ font-size: 22px;
107
+ }
108
+
109
+ .main-text p {
110
+ margin: 15px auto 0;
111
+ }
112
+
113
+ .footer {
114
+ display: flex;
115
+ flex-wrap: wrap;
116
+ justify-content: space-between;
117
+ border-top: 1px solid #e6e6e6;
118
+ padding: 10px;
119
+ }
120
+
121
+ .option {
122
+ display: block;
123
+ padding: 10px 10px 10px 34px;
124
+ position: relative;
125
+ text-decoration: none;
126
+ }
127
+
128
+ .option svg {
129
+ width: 24px;
130
+ height: 24px;
131
+ fill: gray;
132
+ border: 1px solid #d9d9d9;
133
+ padding: 5px;
134
+ border-radius: 100%;
135
+ top: 10px;
136
+ inset-inline-start: 0;
137
+ position: absolute;
138
+ }
139
+
140
+ .option h2 {
141
+ font-size: 19px;
142
+ text-decoration: underline;
143
+ }
144
+
145
+ .option p {
146
+ padding-top: 3px;
147
+ color: #231f20;
148
+ font-size: 15px;
149
+ font-weight: 300;
150
+ }
151
+
152
+ @media (max-width: 996px) {
153
+ body {
154
+ max-width: 780px;
155
+ }
156
+ }
157
+
158
+ @media (max-width: 767px) {
159
+ .option {
160
+ flex: 0 0 50%;
161
+ }
162
+ }
163
+
164
+ @media (max-width: 599px) {
165
+ .main {
166
+ padding: 20px 0;
167
+ }
168
+
169
+ .figure-space {
170
+ max-width: 200px;
171
+ }
172
+
173
+ .footer {
174
+ display: block;
175
+ width: 300px;
176
+ margin: 0 auto;
177
+ }
178
+ }
179
+
180
+ @media (max-width: 360px) {
181
+ .header-link {
182
+ max-width: 100px;
183
+ }
184
+ }
@@ -0,0 +1,21 @@
1
+ {% extends "base.html" %}
2
+ {% load static %}
3
+
4
+ {% block body_class %}template-homepage{% endblock %}
5
+
6
+ {% block extra_css %}
7
+
8
+ {% comment %}
9
+ Delete the line below if you're just getting started and want to remove the welcome screen!
10
+ {% endcomment %}
11
+ <link rel="stylesheet" href="{% static 'css/welcome_page.css' %}">
12
+ {% endblock extra_css %}
13
+
14
+ {% block content %}
15
+
16
+ {% comment %}
17
+ Delete the line below if you're just getting started and want to remove the welcome screen!
18
+ {% endcomment %}
19
+ {% include 'home/welcome_page.html' %}
20
+
21
+ {% endblock content %}