diff --git a/src/author/__init__.py b/src/author/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/src/author/admin.py b/src/author/admin.py new file mode 100644 index 0000000..9e73a08 --- /dev/null +++ b/src/author/admin.py @@ -0,0 +1,25 @@ + +from django.contrib import admin +from author.models import Author + + +@admin.register(Author) +class AuthorAdmin(admin.ModelAdmin): + list_display = ( + "resume_url", + "github_url", + "email", + "created_at", + "updated_at", + ) + + search_fields = ( + "github_url", + "resume_url", + "email", + ) + + list_filter = ( + "created_at", + "updated_at", + ) \ No newline at end of file diff --git a/src/author/apps.py b/src/author/apps.py new file mode 100644 index 0000000..9606d38 --- /dev/null +++ b/src/author/apps.py @@ -0,0 +1,11 @@ +from django.apps import AppConfig + + +class AuthorConfig(AppConfig): + """ + Конфигурация приложения. + """ + + default_auto_field = "django.db.models.BigAutoField" + name = "author" + verbose_name = "Автор" \ No newline at end of file diff --git a/src/author/context_processors.py b/src/author/context_processors.py new file mode 100644 index 0000000..2f0a2da --- /dev/null +++ b/src/author/context_processors.py @@ -0,0 +1,9 @@ +from django.http import HttpRequest +from author.models import Author + + +def author(request: HttpRequest) -> dict: + """ + Контекстный процессор для информации об авторе + """ + return {"author": Author.objects.last()} \ No newline at end of file diff --git a/src/author/migrations/0001_initial.py b/src/author/migrations/0001_initial.py new file mode 100644 index 0000000..1360315 --- /dev/null +++ b/src/author/migrations/0001_initial.py @@ -0,0 +1,49 @@ +# Generated by Django 4.1.13 on 2024-03-20 15:57 + +from django.db import migrations, models + + +class Migration(migrations.Migration): + + initial = True + + dependencies = [] + + operations = [ + migrations.CreateModel( + name="Author", + fields=[ + ( + "id", + models.BigAutoField( + auto_created=True, + primary_key=True, + serialize=False, + verbose_name="ID", + ), + ), + ( + "created_at", + models.DateTimeField( + auto_now_add=True, verbose_name="Время создания записи" + ), + ), + ( + "updated_at", + models.DateTimeField( + auto_now=True, verbose_name="Время обновления записи" + ), + ), + ("resume_url", models.URLField(verbose_name="Ссылка на резюме")), + ("github_url", models.URLField(verbose_name="Ссылка GitHub")), + ( + "email", + models.EmailField(max_length=254, verbose_name="Email автора"), + ), + ], + options={ + "verbose_name": "Информация об авторе", + "verbose_name_plural": "Информация об авторе", + }, + ), + ] diff --git a/src/author/migrations/__init__.py b/src/author/migrations/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/src/author/models.py b/src/author/models.py new file mode 100644 index 0000000..0a23d86 --- /dev/null +++ b/src/author/models.py @@ -0,0 +1,19 @@ +from django.db import models +from base.models import TimeStampMixin + + +class Author(TimeStampMixin): + """ + Модель для хранения данных об авторе + """ + + resume_url = models.URLField(verbose_name="Ссылка на резюме") + github_url = models.URLField(verbose_name="Ссылка GitHub") + email = models.EmailField(verbose_name="Email автора") + + class Meta: + verbose_name = "Информация об авторе" + verbose_name_plural = "Информация об авторе" + + def __str__(self) -> str: + return f'Объект "автор" (id={self.pk})' \ No newline at end of file diff --git a/src/jobs/migrations/0002_job_detailed_description.py b/src/jobs/migrations/0002_job_detailed_description.py new file mode 100644 index 0000000..972bb2b --- /dev/null +++ b/src/jobs/migrations/0002_job_detailed_description.py @@ -0,0 +1,24 @@ +# Generated by Django 4.1.13 on 2024-03-20 13:51 + +import ckeditor_uploader.fields +from django.db import migrations + + +class Migration(migrations.Migration): + + dependencies = [ + ("jobs", "0001_initial"), + ] + + operations = [ + migrations.AddField( + model_name="job", + name="detailed_description", + field=ckeditor_uploader.fields.RichTextUploadingField( + default="", + help_text="Подробное описание выполненной работы", + verbose_name="Подробное описание", + ), + preserve_default=False, + ), + ] diff --git a/src/jobs/models.py b/src/jobs/models.py index a5525d1..305ccb2 100644 --- a/src/jobs/models.py +++ b/src/jobs/models.py @@ -2,6 +2,7 @@ Модели для приложения "Jobs" (выполненные работы). """ +from ckeditor_uploader.fields import RichTextUploadingField from django.db import models from base.models import TimeStampMixin @@ -23,9 +24,21 @@ class Job(TimeStampMixin): help_text="Краткое описание выполненной работы", ) + detailed_description = RichTextUploadingField( + verbose_name="Подробное описание", + help_text="Подробное описание выполненной работы", + ) + class Meta: verbose_name = "Выполненная работа" verbose_name_plural = "Выполненные работы" def __str__(self) -> str: return f'Объект "Выполненная работа" (id={self.pk})' + + def summary(self) -> str: + """ + Краткое представление подробного описания работы. + :return: + """ + return self.detailed_description[:100] + "..." \ No newline at end of file diff --git a/src/jobs/templates/jobs/job_detail.html b/src/jobs/templates/jobs/job_detail.html index 6b40e3f..3a50298 100644 --- a/src/jobs/templates/jobs/job_detail.html +++ b/src/jobs/templates/jobs/job_detail.html @@ -7,7 +7,11 @@

{{ blog.title }}

{{ blog.pub_date }}

{{ blog.title }}

{{ blog.content }}

- Вернуться в блог +

{{ job.title }}

+

{{ job.pub_date }}

+

{{ job.description }}

+

{{ job.detailed_description|safe }}

+ На главную {% endblock %} diff --git a/src/jobs/templates/jobs/job_list.html b/src/jobs/templates/jobs/job_list.html index 5765bc2..0375235 100644 --- a/src/jobs/templates/jobs/job_list.html +++ b/src/jobs/templates/jobs/job_list.html @@ -7,10 +7,10 @@
-

Привет!

-

Меня зовут Иван Иванов.
Я профессионально занимаюсь разработкой на Python.

+ +

Меня зовут Шутов Артемий.
Я профессионально занимаюсь разработкой на JavaScript.

- Иван Иванов + Шутов Артемий

Написать мне @@ -33,10 +33,10 @@ alt="job image"/>

-

{{ job.description }}

+

{{ job.summary|safe}}

{{ job.created_at|date:"d E Y" }}

diff --git a/src/jobs/urls.py b/src/jobs/urls.py new file mode 100644 index 0000000..5d079dd --- /dev/null +++ b/src/jobs/urls.py @@ -0,0 +1,7 @@ +from django.urls import path +from jobs.views import IndexJobsListView, IndexJobDetailView + +urlpatterns = [ + path("", IndexJobsListView.as_view(), name='jobs'), + path("/", IndexJobDetailView.as_view(), name='job') +] \ No newline at end of file diff --git a/src/jobs/views.py b/src/jobs/views.py index 17fb444..bea0f93 100644 --- a/src/jobs/views.py +++ b/src/jobs/views.py @@ -1,7 +1,11 @@ -from django.views.generic import ListView +from django.views.generic import ListView, DetailView from jobs.models import Job class IndexJobsListView(ListView): model = Job + + +class IndexJobDetailView(DetailView): + model = Job \ No newline at end of file diff --git a/src/portfolio/settings.py b/src/portfolio/settings.py index dbe6dbd..d114fe1 100644 --- a/src/portfolio/settings.py +++ b/src/portfolio/settings.py @@ -44,6 +44,7 @@ "django.contrib.staticfiles", # созданные приложения "jobs.apps.JobsConfig", + "author.apps.AuthorConfig", "blog.apps.BlogConfig", # установленные приложения "ckeditor", @@ -73,6 +74,7 @@ "django.template.context_processors.request", "django.contrib.auth.context_processors.auth", "django.contrib.messages.context_processors.messages", + "author.context_processors.author" ], }, }, diff --git a/src/portfolio/templates/base.html b/src/portfolio/templates/base.html index 42b2f67..21cc687 100644 --- a/src/portfolio/templates/base.html +++ b/src/portfolio/templates/base.html @@ -36,9 +36,9 @@

Обо мне

Мои контакты

diff --git a/src/portfolio/urls.py b/src/portfolio/urls.py index 11ab691..64a5ab5 100644 --- a/src/portfolio/urls.py +++ b/src/portfolio/urls.py @@ -28,6 +28,7 @@ path("admin/", admin.site.urls), path("ckeditor/", include("ckeditor_uploader.urls")), path("", IndexJobsListView.as_view(), name="home"), + path("job/", include("jobs.urls")), path("blog/", include("blog.urls")), ] urlpatterns.extend(static(settings.MEDIA_URL, document_root=settings.MEDIA_ROOT))