Python - Django를 이용한 인스타그램 클론 - 7

2022. 11. 8. 01:31How to become a real programmer/Back-End

이전에 회원가입, 로그인 과정을 만들었으니 이제 로그인하여 만든 글을 볼 수 있는 '내 프로필' 화면과 로그아웃을 만들면 될 것 같다. 

우선, 이전에 세션을 통해 사용자 정보를 인식하는 것은 했으니, 프로필과 로그아웃을 만들어야 하는데. 인스타그램은 사용자 창을 누르면 프로필-로그아웃 순으로 드롭다운 방식으로 내려오는 ui를 가지고 있다.

드롭다운을 구현하는 것은 bootstrap에서 카피해 사용하면 될 것 같다. 

 

<main.html>

<!doctype html>
<html lang="en">
<head>
    <meta charset="utf-8">
    <meta name="viewport" content="width=device-width, initial-scale=1">
    {% load static %} <!--불러오기-->
    <title>Bootstrap demo</title>
    <link href="https://cdn.jsdelivr.net/npm/bootstrap@5.2.1/dist/css/bootstrap.min.css" rel="stylesheet"
          integrity="sha384-iYQeCzEYFbKjA/T2uDLTpkwGzCiq6soy8tYaI1GyVh/UjpbCx/TYkiZhlZB6+fzT" crossorigin="anonymous">

    <!--Google Icon-->
    <link
            href="https://fonts.googleapis.com/css?family=Material+Icons|Material+Icons+Outlined|Material+Icons+Two+Tone|Material+Icons+Round|Material+Icons+Sharp"
            rel="stylesheet">

    <!-- Jquery -->
    <script src="http://code.jquery.com/jquery-latest.min.js">

        <link href="css/bootstrap.css" rel="stylesheet">


    </script>
    <style>
        .box {
            width: 50px;
            height: 50px;
            border-radius: 70%;
            overflow: hidden;
        }

        .profile {
            width: 100%;
            height: 100%;
            object-fit: cover;
        }

        .feed_box {
            margin: 20px 0;
            border: solid 1px gray;
            background-color: white;
        }

        .modal_overlay {
            width: 100%;
            height: 100%;
            position: absolute;
            left: 0;
            top: 0;
            display: none;
            flex-direction: column;
            align-items: center;
            justify-content: center;
            background: rgba(0, 0, 0, 0.8);
            backdrop-filter: blur(1.5px);
            -webkit-backdrop-filter: blur(1.5px);
        }

        .modal_window {
            background: white;
            backdrop-filter: blur(13.5px);
            -webkit-backdrop-filter: blur(13.5px);
            border-radius: 10px;
            border: 1px solid rgba(255, 255, 255, 0.18);
            width: 800px;
            height: 600px;
            position: relative;
            padding: 10px;
        }

    </style>
</head>
<body>
<nav class="navbar navbar-expand-lg navbar-light bg-light" style="position: fixed; width:100%;">
    <!--flex의 space-between 활용해서, 또 밑으로 새지 않게 nowrap, 반응형이 아니어도 되기에 최소 화면크기를 1000픽셀 -->
    <div class="container-fluid" style="justify-content: space-between; flex-wrap: nowrap; min-width: 1000px"/>
    <img style="height: 30px; object-fit: contain"
         src="https://www.instagram.com/static/images/web/mobile_nav_type_logo-2x.png/1b47f9d0e595.png">
    <input class="form-control" style="width: 200px" type="search" placeholder="Search" aria-label="Search">
    <div style="display:flex;">
        <span class="material-icons" style="padding-right: 10px">home</span>
        <span id="nav_bar_add_box" class="material-icons-outlined" style="padding-right: 10px">add_box</span>
        <div class="dropdown">
            <a id="dropdownMenuButton" data-toggle="dropdown" aria-haspopup="true" aria-expanded="false">
                <div class="box" style="width: 25px; height: 25px">
                    <img class="profile"
                         src="">
                </div>
            </a>
            <div class="dropdown-menu" style="left: -158px" aria-labelledby="dropdownMenuButton">
                <a class="dropdown-item" href="#">프로필</a>
                <div class="dropdown-divider"></div>
                <a class="dropdown-item" href="/user/logout">로그아웃</a>
            </div>
        </div>
    </div>
</nav>


<div style="display: flex; flex-direction: row; justify-content: center; padding-top: 60px; background-color: #fafafa">
        <div style="padding: 20px 60px">
             <div class="box" style="width: 175px; height: 175px;">
                    <img class="profile"
                         src="">
                </div>
        </div>
        <div>
            <div>
                프로필
            </div>
            <div>
                게시물 팔로워
            </div>
            <div>
                이름
            </div>
        </div>
</div>
<!-- 첫 번째 모달 : 이미지 업로드 용 -->
<div id="first_modal" class="modal_overlay">
    <div class="modal_window">
        <div style="display: flex; flex-direction: row; justify-content: space-between">
            <div style="width: 40px">
            </div>
            <div>
                새 이미지 업로드
            </div>
            <div>
                <span class="material-icons">close</span>
            </div>
        </div>
        <div class="img_upload_space" style="border-top: solid 1px gray; margin-top: 10px; width:778px; height: 540px;">
        </div>
    </div>
</div>

<!-- 두 번째 모달 : 내용 탑재용 -->
<div id="second_modal" class="modal_overlay">
    <div class="modal_window">
        <div style="display: flex; flex-direction: row; justify-content: space-between">
            <div style="width: 40px">
            </div>
            <div>
                새 이미지 업로드
            </div>
            <div>
                <span class="material-icons">close</span>
            </div>
        </div>
        <div style="border-top: solid 1px gray; margin-top:10px; display: flex; flex-direction: row;">

            <div class="img_upload_space" style="width:500px; height: 540px;">
            </div>
            <div style="border-left: 1px solid gray">
                <div>
                    <textarea id="input_feed_content" style="width: 276px; height: 400px" class="form-control"
                              rows="5"></textarea>
                </div>
                <button id="feed_create_button" type="button" class="btn btn-primary" style="width:100%">공유하기</button>
            </div>
        </div>
    </div>
</div>


<!-- Jquery -->
<script>

    let files;

    $('#feed_create_button').click(function () {

        let file = files[0];
        let image = files[0].image;
        let content = $('#input_feed_content').val();
        let user_id = "Jin.99";
        let profile_image = "";
// 우선 user_id와 profile_image는 우리가 html에 데이터로 불러온게 아닌 그냥 삽입해둔 상태라 다른 방식을 거치지 않고 그대로 작성해주었다.

        let fd = new FormData();

        fd.append('file', file);
        fd.append('image', image);
        fd.append('content', content);
        fd.append('user_id', user_id);
        fd.append('profile_image', profile_image);
        // form data에 데이터 추가

        $.ajax({
            url: "/content/upload",
            data: fd,
            method: "post",
            processData: false,
            contentType: false,
            success: function (data) {
                console.log("성공");
            },
            error: function (request, status, error) {
                console.log("에러");
            },
            complete: function () {
                console.log("완료");
                location.replace("/main");
            }
        })
    });

    $('#nav_bar_add_box').click(function () {
        $('#first_modal').css({
            display: 'flex'
        });

        $(document.body).css({
            overflowY: 'hidden'
        })

    });

    $('.img_upload_space')
        .on("dragover", dragOver)
        .on("dragleave", dragOver)
        .on("drop", uploadFiles);

    function dragOver(e) {
        e.stopPropagation(); // img_upload_space와 겹쳐진 부분에 탑재되는 것을 막는 역할
        e.preventDefault();

        if (e.type == "dragover") {
            $(e.target).css({
                "background-color": "black",
                "outline-offset": "-20px"
            });
        } else { // dragleave 인 경우
            $(e.target).css({
                "outline-offset": "-10px"
            });
        }
    }

    function uploadFiles(e) {
        e.stopPropagation();
        e.preventDefault();

        e.dataTransfer = e.originalEvent.dataTransfer; // 올린 파일을 업로드 하는
        files = e.target.files || e.dataTransfer.files;
        console.log("뭔가 이미 파일을 올렸네 " + files[0].name); // 파일을 드래그 해서 여러개 올릴 수도 있기에 files내 리스트처럼 접근해야한다
        if (files.length > 1) {
            alert('하나만 올려라.');
            return;
        }


        if (files[0].type.match(/image.*/)) {


            $('#first_modal').css({
                display: 'none'
            });
            $('#second_modal').css({
                display: 'flex'
            });

            $('.img_upload_space').css({
                "background-image": "url(" + window.URL.createObjectURL(files[0]) + ")",
                "outline": "none",
                "background-size": "100%",
                "background-position": "center",
                "background-repeat": "no-repeat"
            }); // 이미지 파일인 경우 그걸로 배경설정
        } else {
            alert('이미지가 아닙니다.');
            return;
        }
    }


</script>


<!-- Bootstrap Bundle-->
<script src="https://code.jquery.com/jquery-3.5.1.slim.min.js"
        integrity="sha384-DfXdz2htPH0lsSSs5nCTpuj/zy4C+OGpamoFVy38MVBnE+IbbVYUew+OrCXaRkfj"
        crossorigin="anonymous"></script>
<script src="https://cdn.jsdelivr.net/npm/bootstrap@4.5.3/dist/js/bootstrap.bundle.min.js"
        integrity="sha384-ho+j7jyWK8fNQe+A12Hb8AhRq26LrZ/JpcUGGOn+Y7RsweNrtN/tE3MoK7ZeZDyx"
        crossorigin="anonymous"></script>


</body>


</html>

위 코드를 보면, 드롭다운 메뉴로 프로필로 갈 수 있는 부분 하나랑 로그아웃 버튼을 구현해두었는데

<해당 부분>

<div class="dropdown-menu" style="left: -158px" aria-labelledby="dropdownMenuButton">
                <a class="dropdown-item" href="#">프로필</a>
                <div class="dropdown-divider"></div>
                <a class="dropdown-item" href="/user/logout">로그아웃</a>
            </div>

a 태그로, /user/logout으로 가게 설정해두었다. 아직 /user/logout 을 만들지 않았기에 만들어준다.

user - <views.py>

class LogOut(APIView):
    def get(self, request):
        request.session.flush()
        return render(request, 'user/login.html')

view에서 logout 클래스를 만들어주면 된다. get으로 받아오는 방식이고 우리가 로그인 정보를 세션에 넘겨주어 인스타그램 로그인이 가능하였던 것이기에 해당 세션을 flush()함수를 통해서 없애주면 된다. 그리고 user/login.html을 렌더해주면 된다.

view에 맞게 url 또한 경로 설정해준다.

 

이렇게 하면 로그아웃이 잘 구현될 것이다.

그리고 내 프로필 창에 들어가면 사용자 프로필도 보여주어야 하므로 template-content내에 profile.html 파일을 하나 만들어주고, bootstrap 기본 설정과 nav bar까지 붙여넣어준다.

<profile.html>

<!doctype html>
<html lang="en">
<head>
    <!-- Required meta tags -->
    <meta charset="utf-8">
    <meta name="viewport" content="width=device-width, initial-scale=1">

    <!-- Bootstrap CSS -->
    <link href="https://cdn.jsdelivr.net/npm/bootstrap@5.1.3/dist/css/bootstrap.min.css" rel="stylesheet"
          integrity="sha384-1BmE4kWBq78iYhFldvKuhfTAU6auU8tT94WrHftjDbrCEXSU1oBoqyl2QvZ6jIW3" crossorigin="anonymous">

    <title>Hello, world!</title>
</head>
<body>
<nav class="navbar navbar-expand-lg navbar-light bg-light" style="position: fixed; width:100%;">
    <!--flex의 space-between 활용해서, 또 밑으로 새지 않게 nowrap, 반응형이 아니어도 되기에 최소 화면크기를 1000픽셀 -->
    <div class="container-fluid" style="justify-content: space-between; flex-wrap: nowrap; min-width: 1000px"/>
    <img style="height: 30px; object-fit: contain"
         src="https://www.instagram.com/static/images/web/mobile_nav_type_logo-2x.png/1b47f9d0e595.png">
    <input class="form-control" style="width: 200px" type="search" placeholder="Search" aria-label="Search">
    <div style="display:flex;">
        <span class="material-icons" style="padding-right: 10px">home</span>
        <span id="nav_bar_add_box" class="material-icons-outlined" style="padding-right: 10px">add_box</span>
        <div class="dropdown">
            <a id="dropdownMenuButton" data-toggle="dropdown" aria-haspopup="true" aria-expanded="false">
                <div class="box" style="width: 25px; height: 25px">
                    <img class="profile"
                         src="">
                </div>
            </a>
            <div class="dropdown-menu" style="left: -158px" aria-labelledby="dropdownMenuButton">
                <a class="dropdown-item" href="#">프로필</a>
                <div class="dropdown-divider"></div>
                <a class="dropdown-item" href="/user/logout">로그아웃</a>
            </div>
        </div>
    </div>
</nav>

<!-- Optional JavaScript; choose one of the two! -->

<!-- Option 1: Bootstrap Bundle with Popper -->
<script src="https://cdn.jsdelivr.net/npm/bootstrap@5.1.3/dist/js/bootstrap.bundle.min.js"
        integrity="sha384-ka7Sk0Gln4gmtz2MlQnikT1wXgYsOg+OMhuP+IlRH9sENBO0LRn5q+8nbTov4+1p"
        crossorigin="anonymous"></script>

<!-- Option 2: Separate Popper and Bootstrap JS -->
<!--
<script src="https://cdn.jsdelivr.net/npm/@popperjs/core@2.10.2/dist/umd/popper.min.js" integrity="sha384-7+zCNj/IqJ95wo16oMtfsKbZ9ccEh31eOz1HGyDuCQ6wgnyJNSYdrPa03rtR1zdB" crossorigin="anonymous"></script>
<script src="https://cdn.jsdelivr.net/npm/bootstrap@5.1.3/dist/js/bootstrap.min.js" integrity="sha384-QJHtvGhmr9XOIpI6YVutG+2QOK9T+ZnN4kzFN1RtK3zEFEIsxhlmWl5/YESvpZ13" crossorigin="anonymous"></script>
-->
</body>
</html>

이렇게 우선 해주고, 

content의 views.py에 아래 클래스를 추가

class Profile(APIView):
    def get(self, request):
        return render(request, 'content/profile.html')

클래스를 추가해줬으니 url.py에서도

from django.urls import path
from .views import Main, UploadFeed, Profile

urlpatterns = [
    path('upload', UploadFeed.as_view()),
    path('profile', Profile.as_view())
]

profile을 경로로 등록해주면 로컬호스트:8000/content/profile로 들어가질것이다.

 

기본적인 형태를 만들었다. 현재 까지 profile.html을 실행해보면

킹받고 귀여운 고양이가 반겨주는 기본적인 프레임이다.

저기 사용자 이름이랑 팔로워 등등을 가져오기 위해서 get으로 호출할 때 세션정보를 불러와야한다.

content/views.py에 profile 클래스에 세션정보를 불러올 수 있게 아래와 같이 해주고

class Profile(APIView):
    def get(self, request):
        email = request.session.get('email', None)

        if email is None:
            return render(request, "user/login.html")

        user = User.objects.filter(email=email).first()  # 웹 서버 세션에 등록된 이메일 정보를 통해 유저 데이터 불러온다

        if user is None:
            return render(request, "user/login.html")

        return render(request, 'content/profile.html', context=dict(user=user))

profile.html에서는 각 닉네임 부분이나 이름부분을

{{ user.nickname }}
{{ user.name }}

으로 가져와 해당 부분에 적용해주면 된다.

그리고 이름 크기를 키우고, 마진을 좀 주고 하는 등의 사소한 처리를 거친 후, 

위 '프로필 편집' 부분을 누르면 바로 이미지를 변경할 수 있도록 하고싶은데, 이를 위해선 type이 file인 input을 만들어서 display:none으로 설정 후,  프로필 편집 버튼이 눌러졌을 때 함께 실행되도록 해주면 되겠다.

<button id="button_profile_upload" style="margin-right: 40px">프로필 편집</button>
            <input type="file" id="input_fileupload" style="display: none;">
$('#button_profile_upload').click(function(){
       $('#input_fileupload').click();
    });

그러고 나면, 

위와같이 파일 선택창이 나온다.

실제로 이러한 파일 선택창에서 이미지 파일을 선택을 해도 프로필은 아직 바뀌지 않는다. 프로필 사진을 바꾸기 위해서는 제이쿼리의 onchange함수를 써서 파일이 탑재될 때 해당 이미지 파일을 불러와서 적용시켜주는 절차를 추가해줘야 한다.

<input type="file" id="input_fileupload" style="display: none;" onchange="profile_upload();">
$('#button_profile_upload').click(function(){
       $('#input_fileupload').click();
    });

    function profile_upload(){
        let file = $('#input_fileupload')[0].files[0]
        console.log(file);
    }

이렇게 한 후 콘솔로그로 file명이 잘 찍히나 확인해본다

찍힌다. 솔직히 $('#input_fileupload')에서 왜 [0]으로, 첫 번째 인자를 지정해줘야하는진 모르겠지만 리스트 형태로 불러와지는 것 같다. 쨌든 이렇게 파일을 불러와지는건 확인했으므로, 계속한다.

내 계정의 이미지 파일을 변경하려면 파일만 탑재하는게 아니라 내 아이디 정보 또한 함께 실어줘야한다.

    $('#button_profile_upload').click(function(){
       $('#input_fileupload').click();
    });

    function profile_upload(){
        let file = $('#input_fileupload')[0].files[0]
        let email = "{{ user.email }}";

        let fd = new FormData()

        fd.append('file', file);
        fd.append('email', email);
        
        $.ajax({
            url: "/user/profile/upload",
            data: fd,
            method: "post",
            processData: false,
            contentType: false,
            success: function (data) {
                console.log("성공");
            },
            error: function (request, status, error) {
                console.log("에러");
            },
            complete: function () {
                console.log("완료");
                location.replace("/content/profile");
            }
        })
    }

요로코롬 function 코드에 email 정보를 추가해주고 FormData를 통해서 두 정보를 FormData에 추가한다.

그리고 이미지 파일을 변경하기 위해서 user/views.py에 UploadProfile 클래스를 추가해준다

class UploadProfile(APIView):
    def post(self, request):

        # 일단 파일 불러와
        file = request.FILES['file']
        email = request.data.get('email')
        uuid_name = uuid4().hex # 이미지에 고유한 id부여를 위해 랜덤하게 글자를 정함
        save_path = os.path.join(MEDIA_ROOT, uuid_name)

        with open(save_path, "wb+") as destination: # with open~ 부분이 실제로 파일을 저장하는 부분, save_path에 파일을 열어 조각조각 가져와 쓴다 - 그냥 알고만 있기
            for chunk in file.chunks(): # 파일변수명.chunks로 가져올 수 있다잉
                destination.write(chunk)
        # 이렇게 저장된 경로를 담아주기 - 이미지에
        profile_image = uuid_name

        user = User.objects.filter(email=email).first()
        user.profile_image = profile_image
        user.save() # create일 때는 안해두 되지만 객체를 불러와서 수정과정을 거친 후에는 save()를 해주어야 한다

        return Response(status=200) #http response -- http:200(=success를 의미)

위와 같이 추가한다. 잠깐 설명을 하자면 files로 넘어오는 변수 'file'을 변수 'file'에 저장하고, email이라는 이름으로 받은 정보를 email 변수에 저장. 이후 파일은 이름을 변경하고 MEDIA 폴더에 저장해놓는다. 이후에 profile_image를 해당 이미지로 설정, 이를 user의 사용자 정보를 email로 불러온 user변수를 활용해 profile_image로 설정해준다. 이후에 save()까지 해주면 변경된 사용자 이미지가 저장될 것이다.

그리고 이러한 view를 불러올 수 있게 url에도 추가해준다.

from django.urls import path
from .views import Join, Login, LogOut, UploadProfile

urlpatterns=[
    path('join', Join.as_view()),
    path('login', Login.as_view()),
    path('logout', LogOut.as_view()),
    path('profile/upload', UploadProfile.as_view())
]

 

<profile.html>

<!doctype html>
<html lang="en">
<head>
    <meta charset="utf-8">
    <meta name="viewport" content="width=device-width, initial-scale=1">
    {% load static %} <!--불러오기-->
    <title>Bootstrap demo</title>
    <link href="https://cdn.jsdelivr.net/npm/bootstrap@5.2.1/dist/css/bootstrap.min.css" rel="stylesheet"
          integrity="sha384-iYQeCzEYFbKjA/T2uDLTpkwGzCiq6soy8tYaI1GyVh/UjpbCx/TYkiZhlZB6+fzT" crossorigin="anonymous">

    <!--Google Icon-->
    <link
            href="https://fonts.googleapis.com/css?family=Material+Icons|Material+Icons+Outlined|Material+Icons+Two+Tone|Material+Icons+Round|Material+Icons+Sharp"
            rel="stylesheet">
    <link rel="stylesheet" href="https://fonts.googleapis.com/css2?family=Material+Symbols+Outlined:opsz,wght,FILL,GRAD@20..48,100..700,0..1,-50..200" />

    <!-- Jquery -->
    <script src="http://code.jquery.com/jquery-latest.min.js">

        <link href="css/bootstrap.css" rel="stylesheet">


    </script>
    <style>
        .box {
            width: 50px;
            height: 50px;
            border-radius: 70%;
            overflow: hidden;
        }

        .profile {
            width: 100%;
            height: 100%;
            object-fit: cover;
        }

        .feed_box {
            margin: 20px 0;
            border: solid 1px gray;
            background-color: white;
        }

        .modal_overlay {
            width: 100%;
            height: 100%;
            position: absolute;
            left: 0;
            top: 0;
            display: none;
            flex-direction: column;
            align-items: center;
            justify-content: center;
            background: rgba(0, 0, 0, 0.8);
            backdrop-filter: blur(1.5px);
            -webkit-backdrop-filter: blur(1.5px);
        }

        .modal_window {
            background: white;
            backdrop-filter: blur(13.5px);
            -webkit-backdrop-filter: blur(13.5px);
            border-radius: 10px;
            border: 1px solid rgba(255, 255, 255, 0.18);
            width: 800px;
            height: 600px;
            position: relative;
            padding: 10px;
        }

    </style>
</head>
<body>
<nav class="navbar navbar-expand-lg navbar-light bg-light" style="position: fixed; width:100%;">
    <!--flex의 space-between 활용해서, 또 밑으로 새지 않게 nowrap, 반응형이 아니어도 되기에 최소 화면크기를 1000픽셀 -->
    <div class="container-fluid" style="justify-content: space-between; flex-wrap: nowrap; min-width: 1000px"/>
    <img style="height: 30px; object-fit: contain"
         src="https://www.instagram.com/static/images/web/mobile_nav_type_logo-2x.png/1b47f9d0e595.png">
    <input class="form-control" style="width: 200px" type="search" placeholder="Search" aria-label="Search">
    <div style="display:flex;">
        <span class="material-icons" style="padding-right: 10px">home</span>
        <span id="nav_bar_add_box" class="material-icons-outlined" style="padding-right: 10px">add_box</span>
        <div class="dropdown">
            <a id="dropdownMenuButton" data-toggle="dropdown" aria-haspopup="true" aria-expanded="false">
                <div class="box" style="width: 25px; height: 25px">
                    <img class="profile"
                         src="">
                </div>
            </a>
            <div class="dropdown-menu" style="left: -158px" aria-labelledby="dropdownMenuButton">
                <a class="dropdown-item" href="#">프로필</a>
                <div class="dropdown-divider"></div>
                <a class="dropdown-item" href="/user/logout">로그아웃</a>
            </div>
        </div>
    </div>
</nav>


<div style="display: flex; flex-direction: row; justify-content: center; padding-top: 60px; background-color: #fafafa">
    <div style="padding: 20px 60px">
        <div class="box" style="width: 175px; height: 175px;">
            <img class="profile"
                 src="">
        </div>
    </div>
    <div style="text-align: left;">
        <div style="display: flex; flex-display:row; margin: 20px 0px;">
            <div style="font-size: 26px; margin-right: 40px">
                {{ user.nickname }}
            </div>
            <button id="button_profile_upload" style="margin-right: 40px">프로필 사진 편집</button>
            <input type="file" id="input_fileupload" style="display: none;" onchange="profile_upload;">
            <span style="margin-right:40px;" class="material-symbols-outlined">settings</span>
        </div>
        <div style="margin-bottom: 20px">
            게시물 내 게시물 건수 팔로워 111 팔로잉 111
        </div>
        <div>
            <b>{{ user.name }}</b>
        </div>
    </div>
</div>
<!-- 첫 번째 모달 : 이미지 업로드 용 -->
<div id="first_modal" class="modal_overlay">
    <div class="modal_window">
        <div style="display: flex; flex-direction: row; justify-content: space-between">
            <div style="width: 40px">
            </div>
            <div>
                새 이미지 업로드
            </div>
            <div>
                <span class="material-icons">close</span>
            </div>
        </div>
        <div class="img_upload_space" style="border-top: solid 1px gray; margin-top: 10px; width:778px; height: 540px;">
        </div>
    </div>
</div>

<!-- 두 번째 모달 : 내용 탑재용 -->
<div id="second_modal" class="modal_overlay">
    <div class="modal_window">
        <div style="display: flex; flex-direction: row; justify-content: space-between">
            <div style="width: 40px">
            </div>
            <div>
                새 이미지 업로드
            </div>
            <div>
                <span class="material-icons">close</span>
            </div>
        </div>
        <div style="border-top: solid 1px gray; margin-top:10px; display: flex; flex-direction: row;">

            <div class="img_upload_space" style="width:500px; height: 540px;">
            </div>
            <div style="border-left: 1px solid gray">
                <div>
                    <textarea id="input_feed_content" style="width: 276px; height: 400px" class="form-control"
                              rows="5"></textarea>
                </div>
                <button id="feed_create_button" type="button" class="btn btn-primary" style="width:100%">공유하기</button>
            </div>
        </div>
    </div>
</div>


<!-- Jquery -->
<script>

    let files;

    $('#feed_create_button').click(function () {

        let file = files[0];
        let image = files[0].image;
        let content = $('#input_feed_content').val();
        let user_id = "Jin.99";
        let profile_image = "";
// 우선 user_id와 profile_image는 우리가 html에 데이터로 불러온게 아닌 그냥 삽입해둔 상태라 다른 방식을 거치지 않고 그대로 작성해주었다.

        let fd = new FormData();

        fd.append('file', file);
        fd.append('image', image);
        fd.append('content', content);
        fd.append('user_id', user_id);
        fd.append('profile_image', profile_image);
        // form data에 데이터 추가

        $.ajax({
            url: "/content/upload",
            data: fd,
            method: "post",
            processData: false,
            contentType: false,
            success: function (data) {
                console.log("성공");
            },
            error: function (request, status, error) {
                console.log("에러");
            },
            complete: function () {
                console.log("완료");
                location.replace("/main");
            }
        })
    });

    $('#nav_bar_add_box').click(function () {
        $('#first_modal').css({
            display: 'flex'
        });

        $(document.body).css({
            overflowY: 'hidden'
        })

    });

    $('.img_upload_space')
        .on("dragover", dragOver)
        .on("dragleave", dragOver)
        .on("drop", uploadFiles);

    function dragOver(e) {
        e.stopPropagation(); // img_upload_space와 겹쳐진 부분에 탑재되는 것을 막는 역할
        e.preventDefault();

        if (e.type == "dragover") {
            $(e.target).css({
                "background-color": "black",
                "outline-offset": "-20px"
            });
        } else { // dragleave 인 경우
            $(e.target).css({
                "outline-offset": "-10px"
            });
        }
    }

    function uploadFiles(e) {
        e.stopPropagation();
        e.preventDefault();

        e.dataTransfer = e.originalEvent.dataTransfer; // 올린 파일을 업로드 하는
        files = e.target.files || e.dataTransfer.files;
        console.log("뭔가 이미 파일을 올렸네 " + files[0].name); // 파일을 드래그 해서 여러개 올릴 수도 있기에 files내 리스트처럼 접근해야한다
        if (files.length > 1) {
            alert('하나만 올려라.');
            return;
        }


        if (files[0].type.match(/image.*/)) {


            $('#first_modal').css({
                display: 'none'
            });
            $('#second_modal').css({
                display: 'flex'
            });

            $('.img_upload_space').css({
                "background-image": "url(" + window.URL.createObjectURL(files[0]) + ")",
                "outline": "none",
                "background-size": "100%",
                "background-position": "center",
                "background-repeat": "no-repeat"
            }); // 이미지 파일인 경우 그걸로 배경설정
        } else {
            alert('이미지가 아닙니다.');
            return;
        }
    }

    $('#button_profile_upload').click(function(){
       $('#input_fileupload').click();
    });

    function profile_upload(){
        let file = $('#input_fileupload')[0].files[0]
        let email = "{{ user.email }}";

        let fd = new FormData()

        fd.append('file', file);
        fd.append('email', email);

        $.ajax({
            url: "/user/profile/upload",
            data: fd,
            method: "post",
            processData: false,
            contentType: false,
            success: function (data) {
                console.log("성공");
            },
            error: function (request, status, error) {
                console.log("에러");
            },
            complete: function () {
                console.log("완료");
                location.replace("/content/profile");
            }
        })
    }

</script>


<!-- Bootstrap Bundle-->
<script src="https://code.jquery.com/jquery-3.5.1.js"></script>
<script src="https://cdn.jsdelivr.net/npm/bootstrap@4.5.3/dist/js/bootstrap.bundle.min.js"
        integrity="sha384-ho+j7jyWK8fNQe+A12Hb8AhRq26LrZ/JpcUGGOn+Y7RsweNrtN/tE3MoK7ZeZDyx"
        crossorigin="anonymous"></script>


</body>


</html>

잘 수행되었다는 200번 코드를 확인할 수 있다.

이제 우리가 확인해야할 것은 프로필이 바뀌었는지 db에서 확인해본다. 바로 프로필 페이지에 안뜨는 이유는 하드코딩해두었기 때문이다.

DB의 user에 가보면 profile_image가 기존 default_profile이 아닌 다른 파일명으로 바뀐것을 볼 수 있다.

 

나머지 부분은 재빨리 다음 포스트에서 올리겠다.

시험기간을 핑계로 며칠 글을 쓰지 못했는데 더 부지런하고 똑똑하게 블로그를 써야겠다.