Compare commits
20 Commits
5f76e51dec
...
20aadb5eb8
| Author | SHA1 | Date | |
|---|---|---|---|
| 20aadb5eb8 | |||
| 4d9c170ab4 | |||
|
|
e53ee300ee | ||
| d600c39c4e | |||
|
|
de197d810b | ||
| 536837991c | |||
| bf48f67f62 | |||
| 063e3f8fb8 | |||
|
|
254501139f | ||
| f584d3f56d | |||
| 007658b326 | |||
| 37a60970dd | |||
| bf4b9dd7da | |||
| e08c71983b | |||
| f0f12502fb | |||
| a5cda0bd0b | |||
| d2dc67dce2 | |||
| 91ba924b46 | |||
| 26be8b675b | |||
| 9ae840d047 |
@ -82,8 +82,12 @@ WSGI_APPLICATION = 'CineSync.wsgi.application'
|
|||||||
|
|
||||||
DATABASES = {
|
DATABASES = {
|
||||||
'default': {
|
'default': {
|
||||||
'ENGINE': 'django.db.backends.sqlite3',
|
'ENGINE': 'django.db.backends.postgresql',
|
||||||
'NAME': BASE_DIR / 'db.sqlite3',
|
'NAME': os.getenv('DJANGO_DB_NAME', default='cinesync_db'),
|
||||||
|
'USER': os.getenv('DJANGO_DB_USER', default='cinesync'),
|
||||||
|
'PASSWORD': os.getenv('DJANGO_DB_PASSWORD'),
|
||||||
|
'HOST': os.getenv('DJANGO_DB_HOST', default='localhost'),
|
||||||
|
'PORT': os.getenv('DJANGO_DB_PORT', default='5432'),
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -116,9 +120,7 @@ USE_TZ = True
|
|||||||
|
|
||||||
STATIC_URL = '/static/'
|
STATIC_URL = '/static/'
|
||||||
|
|
||||||
STATICFILES_DIRS = [
|
STATIC_ROOT = BASE_DIR / 'static'
|
||||||
BASE_DIR / 'static',
|
|
||||||
]
|
|
||||||
|
|
||||||
DEFAULT_AUTO_FIELD = 'django.db.models.BigAutoField'
|
DEFAULT_AUTO_FIELD = 'django.db.models.BigAutoField'
|
||||||
|
|
||||||
|
|||||||
@ -3,7 +3,6 @@ from django.conf.urls.static import static
|
|||||||
from django.contrib import admin
|
from django.contrib import admin
|
||||||
from django.urls import include, path
|
from django.urls import include, path
|
||||||
|
|
||||||
|
|
||||||
urlpatterns = [
|
urlpatterns = [
|
||||||
path('', include('home.urls')),
|
path('', include('home.urls')),
|
||||||
path('admin/', admin.site.urls),
|
path('admin/', admin.site.urls),
|
||||||
@ -14,13 +13,12 @@ urlpatterns = [
|
|||||||
path('auth/', include('django.contrib.auth.urls'), name='auth'),
|
path('auth/', include('django.contrib.auth.urls'), name='auth'),
|
||||||
]
|
]
|
||||||
|
|
||||||
urlpatterns += static(settings.STATIC_URL, document_root=settings.STATIC_ROOT)
|
|
||||||
urlpatterns += static(
|
|
||||||
settings.MEDIA_URL,
|
|
||||||
document_root=settings.MEDIA_ROOT,
|
|
||||||
)
|
|
||||||
|
|
||||||
if settings.DEBUG:
|
if settings.DEBUG:
|
||||||
import debug_toolbar
|
import debug_toolbar
|
||||||
|
|
||||||
urlpatterns += [path('__debug__/', include(debug_toolbar.urls))]
|
urlpatterns += [path('__debug__/', include(debug_toolbar.urls))]
|
||||||
|
urlpatterns += static(settings.STATIC_URL, document_root=settings.STATIC_ROOT)
|
||||||
|
urlpatterns += static(
|
||||||
|
settings.MEDIA_URL,
|
||||||
|
document_root=settings.MEDIA_ROOT,
|
||||||
|
)
|
||||||
|
|||||||
0
CineSync/core/__init__.py
Normal file
0
CineSync/core/__init__.py
Normal file
21
CineSync/core/functions.py
Normal file
21
CineSync/core/functions.py
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
from timetable.models import FilmSession
|
||||||
|
|
||||||
|
|
||||||
|
def get_film_to_sessions():
|
||||||
|
film_sessions = FilmSession.objects.nearest_timetable()
|
||||||
|
sessions_by_date_and_film = {}
|
||||||
|
|
||||||
|
for session in film_sessions:
|
||||||
|
session_date = session.start_datetime.date()
|
||||||
|
session_film = session.film
|
||||||
|
sessions_by_date_and_film.setdefault(session_date, {}).setdefault(
|
||||||
|
session_film, []
|
||||||
|
).append(session)
|
||||||
|
|
||||||
|
for session_date in sessions_by_date_and_film:
|
||||||
|
for session_film in sessions_by_date_and_film[session_date]:
|
||||||
|
sessions_by_date_and_film[session_date][session_film].sort(
|
||||||
|
key=lambda x: x.start_datetime
|
||||||
|
)
|
||||||
|
|
||||||
|
return sessions_by_date_and_film
|
||||||
0
CineSync/core/migrations/__init__.py
Normal file
0
CineSync/core/migrations/__init__.py
Normal file
@ -2,7 +2,7 @@ from django.http import HttpResponse
|
|||||||
from django.shortcuts import get_object_or_404, render
|
from django.shortcuts import get_object_or_404, render
|
||||||
|
|
||||||
from films.models import Film
|
from films.models import Film
|
||||||
from timetable.models import FilmSession
|
from core.functions import get_film_to_sessions
|
||||||
|
|
||||||
|
|
||||||
def films_list(request: HttpResponse) -> HttpResponse:
|
def films_list(request: HttpResponse) -> HttpResponse:
|
||||||
@ -19,28 +19,7 @@ def film_details(request: HttpResponse, film_id: int) -> HttpResponse:
|
|||||||
Film.objects.released(),
|
Film.objects.released(),
|
||||||
id=film_id,
|
id=film_id,
|
||||||
)
|
)
|
||||||
film_sessions = FilmSession.objects.nearest_timetable().filter(
|
sessions_by_date_and_film = get_film_to_sessions()
|
||||||
film_id=film_id,
|
|
||||||
)
|
|
||||||
sessions_by_date_and_film = {}
|
|
||||||
|
|
||||||
for session in film_sessions:
|
|
||||||
session_date = session.start_datetime.date()
|
|
||||||
if session_date not in sessions_by_date_and_film:
|
|
||||||
sessions_by_date_and_film[session_date] = {}
|
|
||||||
|
|
||||||
film_sessions_for_date = sessions_by_date_and_film[session_date]
|
|
||||||
|
|
||||||
if session.film not in film_sessions_for_date:
|
|
||||||
film_sessions_for_date[session.film] = []
|
|
||||||
|
|
||||||
film_sessions_for_date[session.film].append(session)
|
|
||||||
|
|
||||||
for session_date, session_films in sessions_by_date_and_film.items():
|
|
||||||
for session_film in session_films:
|
|
||||||
sessions_by_date_and_film[session_date][session_film].sort(
|
|
||||||
key=lambda sorted_session: sorted_session.start_datetime,
|
|
||||||
)
|
|
||||||
|
|
||||||
context = {
|
context = {
|
||||||
'film_sessions': sessions_by_date_and_film,
|
'film_sessions': sessions_by_date_and_film,
|
||||||
|
|||||||
@ -5,23 +5,12 @@ from django.http import HttpResponse
|
|||||||
from django.shortcuts import render
|
from django.shortcuts import render
|
||||||
|
|
||||||
from films.models import Film
|
from films.models import Film
|
||||||
from timetable.models import FilmSession
|
from core.functions import get_film_to_sessions
|
||||||
|
|
||||||
|
|
||||||
def homepage(request):
|
def homepage(request):
|
||||||
films = Film.objects.on_main()
|
films = Film.objects.on_main()
|
||||||
film_sessions = FilmSession.objects.nearest_timetable()
|
sessions_by_date_and_film = get_film_to_sessions()
|
||||||
sessions_by_date_and_film = {}
|
|
||||||
|
|
||||||
for session in film_sessions:
|
|
||||||
session_date = session.start_datetime.date()
|
|
||||||
session_film = session.film
|
|
||||||
sessions_by_date_and_film.setdefault(session_date, {}).setdefault(session_film, []).append(session)
|
|
||||||
|
|
||||||
for session_date in sessions_by_date_and_film:
|
|
||||||
for session_film in sessions_by_date_and_film[session_date]:
|
|
||||||
sessions_by_date_and_film[session_date][session_film].sort(key=lambda x: x.start_datetime)
|
|
||||||
|
|
||||||
template = render(
|
template = render(
|
||||||
request,
|
request,
|
||||||
'home/homepage.html',
|
'home/homepage.html',
|
||||||
|
|||||||
@ -21,3 +21,8 @@
|
|||||||
.btn {
|
.btn {
|
||||||
margin: 5px !important;
|
margin: 5px !important;
|
||||||
}
|
}
|
||||||
|
.user_image {
|
||||||
|
width: 20vw;
|
||||||
|
border-radius: 2vw;
|
||||||
|
margin-bottom: 1.5vw;
|
||||||
|
}
|
||||||
BIN
CineSync/static/img/qr-site.png
Normal file
BIN
CineSync/static/img/qr-site.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 219 KiB |
@ -21,7 +21,7 @@
|
|||||||
{% endfor %}
|
{% endfor %}
|
||||||
</div>
|
</div>
|
||||||
{% else %}
|
{% else %}
|
||||||
<p class="not_found_label">Информации о фильмах в прокате не найдена :(</p>
|
<p class="not_found_label">Информация о фильмах в прокате не найдена :(</p>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
</div>
|
</div>
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
@ -37,7 +37,7 @@
|
|||||||
<span class="visually-hidden">Next</span>
|
<span class="visually-hidden">Next</span>
|
||||||
</button>
|
</button>
|
||||||
{% else %}
|
{% else %}
|
||||||
<p class="not_found_label">Информации о фильмах в прокате не найдена :(</p>
|
<p class="not_found_label">Информация о фильмах в прокате не найдена :(</p>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
</div>
|
</div>
|
||||||
<div class="timetable_block">
|
<div class="timetable_block">
|
||||||
@ -62,7 +62,7 @@
|
|||||||
{% endfor %}
|
{% endfor %}
|
||||||
</div>
|
</div>
|
||||||
{% else %}
|
{% else %}
|
||||||
<p class="not_found_label">Информации о расписании не найдена :(</p>
|
<p class="not_found_label">Информация о расписании не найдена :(</p>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@ -38,6 +38,9 @@
|
|||||||
<div id="help" class="form-text">{{ field.help_text }}</div>
|
<div id="help" class="form-text">{{ field.help_text }}</div>
|
||||||
</div>
|
</div>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
|
{% if user.profile.image %}
|
||||||
|
<img class="user_image" src="{{ user.profile.image.url }}">
|
||||||
|
{% endif %}
|
||||||
{% if messages %}
|
{% if messages %}
|
||||||
{% for message in messages %}
|
{% for message in messages %}
|
||||||
<p> {{ message }} </p>
|
<p> {{ message }} </p>
|
||||||
|
|||||||
@ -1,7 +1,5 @@
|
|||||||
import datetime
|
import datetime
|
||||||
from collections import defaultdict
|
|
||||||
from datetime import date
|
from datetime import date
|
||||||
from operator import attrgetter
|
|
||||||
|
|
||||||
from django.contrib.auth.decorators import login_required
|
from django.contrib.auth.decorators import login_required
|
||||||
from django.db import transaction
|
from django.db import transaction
|
||||||
@ -23,11 +21,15 @@ def timetable_view(request):
|
|||||||
for session in film_sessions:
|
for session in film_sessions:
|
||||||
session_date = session.start_datetime.date()
|
session_date = session.start_datetime.date()
|
||||||
session_film = session.film
|
session_film = session.film
|
||||||
sessions_by_date_and_film.setdefault(session_date, {}).setdefault(session_film, []).append(session)
|
sessions_by_date_and_film.setdefault(session_date, {}).setdefault(
|
||||||
|
session_film, []
|
||||||
|
).append(session)
|
||||||
|
|
||||||
for session_date in sessions_by_date_and_film:
|
for session_date in sessions_by_date_and_film:
|
||||||
for session_film in sessions_by_date_and_film[session_date]:
|
for session_film in sessions_by_date_and_film[session_date]:
|
||||||
sessions_by_date_and_film[session_date][session_film].sort(key=lambda x: x.start_datetime)
|
sessions_by_date_and_film[session_date][session_film].sort(
|
||||||
|
key=lambda x: x.start_datetime
|
||||||
|
)
|
||||||
|
|
||||||
template = render(
|
template = render(
|
||||||
request,
|
request,
|
||||||
|
|||||||
@ -53,6 +53,17 @@ class SignUpForm(UserCreationForm):
|
|||||||
class Meta(UserCreationForm.Meta):
|
class Meta(UserCreationForm.Meta):
|
||||||
fields = ('username', 'email')
|
fields = ('username', 'email')
|
||||||
|
|
||||||
|
def clean_username(self):
|
||||||
|
username = self.cleaned_data['username']
|
||||||
|
if len(username) > 150:
|
||||||
|
raise forms.ValidationError('Максимальная длина 150 символов.')
|
||||||
|
if not all(char.isalnum() or char in '@/./+/-/_' for char in username):
|
||||||
|
raise forms.ValidationError(
|
||||||
|
'Можно использовать только буквы, цифры и символы @/./+/-/_.'
|
||||||
|
)
|
||||||
|
|
||||||
|
return username
|
||||||
|
|
||||||
|
|
||||||
class ProfileForm(ModelForm):
|
class ProfileForm(ModelForm):
|
||||||
def __init__(self, *args, **kwargs) -> None:
|
def __init__(self, *args, **kwargs) -> None:
|
||||||
@ -99,3 +110,14 @@ class UserForm(forms.ModelForm):
|
|||||||
model.first_name.field.name,
|
model.first_name.field.name,
|
||||||
model.last_name.field.name,
|
model.last_name.field.name,
|
||||||
]
|
]
|
||||||
|
|
||||||
|
def clean_username(self):
|
||||||
|
username = self.cleaned_data['username']
|
||||||
|
if len(username) > 150:
|
||||||
|
raise forms.ValidationError('Максимальная длина 150 символов.')
|
||||||
|
if not all(char.isalnum() or char in '@/./+/-/_' for char in username):
|
||||||
|
raise forms.ValidationError(
|
||||||
|
'Можно использовать только буквы, цифры и символы @/./+/-/_.'
|
||||||
|
)
|
||||||
|
|
||||||
|
return username
|
||||||
|
|||||||
@ -1,3 +1,8 @@
|
|||||||
|
# P.S.
|
||||||
|
|
||||||
|
Этот проект я разрабатывал, как итоговый проект курсов от Яндекса: "Веб разработка на Django", вместе с однокурсником с
|
||||||
|
этих курсов.
|
||||||
|
|
||||||
# CineSync
|
# CineSync
|
||||||
|
|
||||||
Проект представляет собой интерфейс для работы с системой кинотеатра
|
Проект представляет собой интерфейс для работы с системой кинотеатра
|
||||||
@ -8,7 +13,7 @@
|
|||||||
заказывать их. Можно просматривать информацию о конкретном фильме. На его странице будет отображаться описание, дата
|
заказывать их. Можно просматривать информацию о конкретном фильме. На его странице будет отображаться описание, дата
|
||||||
выхода, режиссер, жанры, ближайшие сеансы и другая информация. Также на сайте реализована возможность регистрации,
|
выхода, режиссер, жанры, ближайшие сеансы и другая информация. Также на сайте реализована возможность регистрации,
|
||||||
авторизации, редактирования профиля.
|
авторизации, редактирования профиля.
|
||||||
Сайт расположен по адресу: https://cinesync.numerum.site/
|
Сайт расположен по адресу: https://cinesync.numerum.team/
|
||||||
|
|
||||||
## Перед запуском
|
## Перед запуском
|
||||||
|
|
||||||
|
|||||||
BIN
Документы/Django CineSync - Презентация.pdf
Normal file
BIN
Документы/Django CineSync - Презентация.pdf
Normal file
Binary file not shown.
@ -1,9 +1,9 @@
|
|||||||
CREATE OR REPLACE FUNCTION populate_film_sessions(start_date_session date)
|
CREATE OR REPLACE FUNCTION populate_film_sessions_with_count(start_date_session date, count_days int)
|
||||||
RETURNS VOID AS $$
|
RETURNS VOID AS $$
|
||||||
DECLARE
|
DECLARE
|
||||||
film_count INT := 26; -- Общее количество фильмов
|
film_count INT := 26; -- Общее количество фильмов
|
||||||
auditorium_count INT := 7; -- Общее количество залов
|
auditorium_count INT := 7; -- Общее количество залов
|
||||||
days_to_populate INT := 5; -- Количество дней для заполнения
|
days_to_populate INT := count_days; -- Количество дней для заполнения
|
||||||
start_date DATE := start_date_session + INTERVAL '1 day'; -- Начальная дата заполнения (завтра)
|
start_date DATE := start_date_session + INTERVAL '1 day'; -- Начальная дата заполнения (завтра)
|
||||||
session_times TIME[] := ARRAY['10:00'::TIME, '13:30'::TIME, '17:00'::TIME, '20:30'::TIME]; -- Время сеансов
|
session_times TIME[] := ARRAY['10:00'::TIME, '13:30'::TIME, '17:00'::TIME, '20:30'::TIME]; -- Время сеансов
|
||||||
film_id INT;
|
film_id INT;
|
||||||
@ -29,3 +29,39 @@ BEGIN
|
|||||||
END LOOP;
|
END LOOP;
|
||||||
END;
|
END;
|
||||||
$$ LANGUAGE plpgsql;
|
$$ LANGUAGE plpgsql;
|
||||||
|
|
||||||
|
select populate_film_sessions_with_count('2024-05-28', 60);
|
||||||
|
|
||||||
|
--select populate_film_sessions('2024-05-23');
|
||||||
|
|
||||||
|
--CREATE OR REPLACE FUNCTION auto_fill_films_sessions()
|
||||||
|
--RETURNS VOID AS $$
|
||||||
|
--DECLARE
|
||||||
|
-- film_count INT := 26; -- Общее количество фильмов
|
||||||
|
-- auditorium_count INT := 7; -- Общее количество залов
|
||||||
|
-- days_to_populate INT := 5; -- Количество дней для заполнения
|
||||||
|
-- start_date DATE := CURRENT_DATE + INTERVAL '1 day'; -- Начальная дата заполнения (завтра)
|
||||||
|
-- session_times TIME[] := ARRAY['10:00'::TIME, '13:30'::TIME, '17:00'::TIME, '20:30'::TIME]; -- Время сеансов
|
||||||
|
-- film_id INT;
|
||||||
|
-- start_datetime TIMESTAMP;
|
||||||
|
-- end_datetime TIMESTAMP;
|
||||||
|
--BEGIN
|
||||||
|
-- FOR i IN 0..(days_to_populate - 1) LOOP
|
||||||
|
-- FOR j IN 1..auditorium_count LOOP
|
||||||
|
-- FOR k IN 1..4 LOOP
|
||||||
|
-- -- Выбор случайного фильма
|
||||||
|
-- PERFORM setseed(random());
|
||||||
|
-- film_id := (SELECT id FROM films_films ORDER BY random() LIMIT 1);
|
||||||
|
--
|
||||||
|
-- -- Вычисление даты и времени начала сеанса
|
||||||
|
-- start_datetime := COALESCE(start_date + i + session_times[k], start_date);
|
||||||
|
-- end_datetime := COALESCE(start_datetime + INTERVAL '2 hours', start_datetime); -- Длительность фильма 2 часа
|
||||||
|
--
|
||||||
|
-- -- Вставка данных в таблицу
|
||||||
|
-- INSERT INTO timetable_film_sessions (start_datetime, end_datetime, price, film_id, auditorium_id)
|
||||||
|
-- VALUES (start_datetime, end_datetime, round(random() * 500 + 200), film_id, j);
|
||||||
|
-- END LOOP;
|
||||||
|
-- END LOOP;
|
||||||
|
-- END LOOP;
|
||||||
|
--END;
|
||||||
|
--$$ LANGUAGE plpgsql;
|
||||||
|
|||||||
BIN
Документы/Текст к презентации.docx
Normal file
BIN
Документы/Текст к презентации.docx
Normal file
Binary file not shown.
Loading…
x
Reference in New Issue
Block a user