Project using python/Cloning Airbnb

django reservation-detail by FBV

Cog Factory 2021. 3. 17. 05:54

url 설정

config/urls.py

urlpatterns = [
    path("reservations/", include("reservations.urls", namespace="reservations"))
]

reservations/urls.py

from django.urls import path
from reservations import views

app_name = "reservations"

urlpatterns = [
    path(
        "<int:pk>/",
        views.reservationDetail,
        name="detail",
    ),
]

FBV(Function Based View)

reservations/views.py

# reservations/views.py

from django.shortcuts import render, redirect, reverse
from django.contrib.auth.decorators import login_required
from django.contrib import messages
from django.http import Http404

from reservations import models as reservation_models


@login_required
def reservationDetail(request, pk):
    reservation = reservation_models.Reservation.objects.get_or_none(pk=pk)
    if reservation is None:
        messages.error(request, "Reservation does not exist")
        return redirect(reverse("core:home"))
    if reservation.guest.pk != request.user.pk:
        raise Http404()
    bookedDays = reservation.bookedDays.all()
    days = []

    for day in bookedDays:
        day = str(day)
        day = int(day.split("-")[2])
        days.append(day)

    room = reservation.room
    return render(
        request,
        "pages/reservations/reservation_detail.html",
        context={"room": room, "reservation": reservation, "days": days},
    )

templates

pages/reservations/reservation_detail.html

{% extends 'base.html' %}

{% load remainder_op %}

{% block page_title %}
    Reservation-{{reservation}}
{% endblock page_title %}

{% block content %}
    <div class="flex flex-col items-center justify-items-center">
        <div class="w-10/12">
            <div class="grid grid-cols-2 grid-rows-2-200px gap-2 mb-12">
                {% if room.get_first_photo is not None %}
                    <img src="{{room.get_first_photo}}" class="row-span-full rounded-tl-lg rounded-bl-lg h-full w-full bg-cover bg-center" alt="Room Photo">
                {% endif %}
                <div class="grid grid-cols-2 grid-rows-2-200px gap-2">
                    {% for index, photo in room.get_four_photo %}
                        {% remainder_op index as remainder %}
                        <img src="{{photo.file.url}}" class="h-full w-full bg-cover bg-center {% if remainder == 1 %}rounded-tr-lg{% elif remainder == 3 %}rounded-br-lg{% endif %}" alt="Room Photo">
                    {% endfor %}
                </div>
            </div>
            <main class="flex">
                <div class="flex flex-col w-3/5">
                    <div class="flex items-center ">
                        <h1 class="text-3xl font-bold mb-4">{{room.name}}</h1>
                        <span class="text-lg font-bold mb-2 ml-4 {% if reservation.status == 'pending' %} text-yellow-500 {% elif reservation.status == 'canceled' %} text-red-600 {% else %} text-green-600 {% endif %}">{{reservation.status|capfirst}}</span>
                    </div>
                    <div class="flex text-gray-500 mb-5">
                        <div class="rating">
                            <i class="fas fa-star text-red-500"></i>
                            <span class="font-bold">{{room.get_review_points}}</span>
                            <span class=" font-normal">({{room.reviews.count}})</span>
                            <span class="mx-1">&centerdot;</span>
                        </div>
                        {% if room.host.superhost %}
                            <span class="ml-1">Superhost</span>
                            <span class="mx-2">&centerdot;</span>
                        {% endif %}
                        <div class="font-medium">
                            <span class="mr-1">{{room.address}},</span>
                            <span class="mr-1">{{room.city}},</span>
                            <span>{{room.country.name}}</span>
                        </div>
                    </div>
                    <div class="flex mb-10 items-center">
                        <button class="button w-48 bg-red-500">
                            <a  href="#">Cancel Reservation</a>
                        </button>
                    </div>
                    <div class="w-11/12 grid grid-cols-2 gap-10 border-b border-gray-300 mb-6 pb-6">
                        {% include 'mixins/calendar.html' with page="reservation_page" reservation="reservation" days=days %}
                    </div>
                </div>
                <div class="w-2/5">
                    
                    {% if reservation.status == 'confirmed' and reservation.is_finished %}
                        <span class="font-medium text-2xl text-center w-full block mb-5">Write your review</span>
                        <form action="{% url 'reviews:create' reservation.pk %}" method="post" class="w-full">
                            {% csrf_token %}
                            <div class="border p-3 border-gray-400 rounded">
                                <textarea name="review" id="" rows="5" placeholder="Write a review" class="w-full resize-none"></textarea>
                            </div>
                            <div class="grid grid-cols-2 gap-1 mt-1">
                                <div class="border p-3 border-gray-400 rounded">
                                    <input type="number" name="accuracy" placeholder="Accuracy Points" min=0 max=5 class="outline-none w-full">
                                </div>
                                <div class="border p-3 border-gray-400 rounded">
                                    <input type="number" name="communication" placeholder="Communication Points" min=0 max=5 class="outline-none w-full">
                                </div>
                                <div class="border p-3 border-gray-400 rounded">
                                    <input type="number" name="cleanliness" placeholder="Cleanliness Points" min=0 max=5 class="outline-none w-full">
                                </div>
                                <div class="border p-3 border-gray-400 rounded">
                                    <input type="number" name="location" placeholder="Location Points" min=0 max=5 class="outline-none w-full">
                                </div>
                                <div class="border p-3 border-gray-400 rounded">
                                    <input type="number" name="check_in" placeholder="Check In Points" min=0 max=5 class="outline-none w-full">
                                </div>
                                <div class="border p-3 border-gray-400 rounded">
                                    <input type="number" name="value" placeholder="Value Points" min=0 max=5 class="outline-none w-full">
                                </div>
                            </div>
                            <button class="form_button mt-5">Submit Review</button>
                        </form>
                    {% endif %}
                        
                </div>
            </main>
        </div>
    </div>
{% endblock content %}

 

참고 자료

  • 노마드 코더의 Airbnb 클론 강의

소스 코드

github.com/zpskek/airbnb-clone-v3/commit/a1675af32d7ba36d5907828836aef5501ff573b4