소개 페이지 만들기

·2023년 8월 14일
0

웹 프로젝트

목록 보기
2/18

저번에 만들었던 팀원 소개 페이지가 조금 심심한 것 같아 간단하게 자기소개 페이지를 만들어보았다.

주요 기능 소개

아이콘

전화, 깃허브, 이메일 아이콘에 호버 효과를 넣었다. 깃허브 아이콘은 클릭하면 내 깃허브 페이지로 이동할 수 있다.

프로젝트

Projects 의 메뉴에서 해당하는 버튼을 누르면 그 분야에서 진행했던 프로젝트 페이지로 넘어가게 된다. 메뉴는 호버 css를 추가해서 마우스를 올릴 시

이러한 효과가 나타나도록 했다.

<a href> 태그를 통해 클릭하면 해당 프로젝트 깃허브 페이지로 넘어갈 수 있다.

css 출처
https://inpa.tistory.com/entry/CSS-%F0%9F%92%8D-%EB%A7%81%ED%81%AC-hover-%EB%94%94%EC%9E%90%EC%9D%B8-%F0%9F%96%8C%EF%B8%8F-%EB%AA%A8%EC%9D%8C

태그

내가 가진 기술 스택들을 태그로 추가한 뒤 간단한 호버 효과를 넣어주었다.

댓글 기능

닉네임과 코멘트를 입력하면 아래에 댓글이 보이도록 만들었다. 작성된 댓글은 mongoDB의 hyelee 컬렉션에 저장되며, rewrite를 누르면 수정할 수 있고 delete를 누르면 삭제할 수 있다.



전체 코드

profile.css

  @import url("https://fonts.googleapis.com/css?family=Lato");

  :root {
      /* COLORS */
      --white: #e9e9e9;
      --gray: #f8f8f8;
      --blue: #1c8d51;
      --lightblue: #137b32b8;

      /* RADII */
      /* --button-radius: 0.7rem; */

      /* SIZES */
      --max-width: 758px;
      --max-height: 420px;

      font-size: 24px;
      font-family: 'Noto Serif KR', serif;

  }

  .skill {
      font-size: 35px;
      font-family: 'Noto Serif KR', serif;
      font-family: 'Roboto Mono', monospace;
      margin-bottom: 15px;
  }

  .btn {
      font-family: 'Noto Serif KR', serif;
      font-family: 'Roboto Mono', monospace;
      background-color: #5959598e;
      border-radius: 5px;
      margin-bottom: 150px;
      margin-left: 350px;
  }

  .button-container {
      margin-top: 20px;
      margin-right: 20px;
      display: flex;
      flex-direction: row-reverse;

  }

  .profile-img {
      width: 50px;
      height: 50px;
      border-radius: 70%;
  }

  .skill-container {
      display: flex;
      flex-wrap: wrap;
      margin-bottom: 30px;
  }

  .comment-btn1,
  .comment-btn2 {
      font-family: 'Noto Serif KR', serif;
      font-family: 'Roboto Mono', monospace;
      background-color: #5959598e;
      border-radius: 7px;
      margin-left: 10px;
  }

  .ex-comment {
      margin-left: 50px;
      font-size: 30px;
      font-family: 'Noto Serif KR', serif;
      font-family: 'Roboto Mono', monospace;
  }

  .ex-nick {
      margin-left: 50px;
      font-size: 20px;
      font-family: 'Noto Serif KR', serif;
      font-family: 'Roboto Mono', monospace;
  }

  .profile_img {
      margin-top: 70px;
      display: flex;
      flex-direction: row;
  }

  .bottom {
      display: flex;
      flex-direction: row;
  }

  .menu-pro {
      display: flex;
      flex-direction: column;
  }

  .menu a {
      color: rgba(255, 255, 255, 0.8);
      font-family: Lato;
      font-size: 17pt;
      font-weight: 400;
      padding: 15px 25px;
      /**/
      position: relative;
      display: block;
      text-decoration: none;
      text-transform: uppercase;

  }

  .SMN_effect-2 a:before,
  .SMN_effect-2 a:after {
      position: absolute;
      top: 100%;
      left: 0;
      width: 100%;
      height: 2px;
      background: rgba(255, 255, 255, 0.5);
      content: '';
      -webkit-transition: -webkit-transform 0.3s;
      -moz-transition: -moz-transform 0.3s;
      transition: transform 0.3s;
      -webkit-transform: scale(0.85);
      -moz-transform: scale(0.85);
      transform: scale(0.85);

  }

  .SMN_effect-2 a:after {
      opacity: 0;
      -webkit-transition: top 0.3s, opacity 0.3s, -webkit-transform 0.3s;
      -moz-transition: top 0.3s, opacity 0.3s, -moz-transform 0.3s;
      transition: top 0.3s, opacity 0.3s, transform 0.3s;
  }

  .SMN_effect-2 a:hover:before,
  .SMN_effect-2 a:hover:after,
  .SMN_effect-2 a:focus:before,
  .SMN_effect-2 a:focus:after {
      -webkit-transform: scale(1);
      -moz-transform: scale(1);
      transform: scale(1);
      background: #c8a755;
  }

  .SMN_effect-2 a:hover:after,
  .SMN_effect-2 a:focus:after {
      top: 0%;
      opacity: 1;
  }


  .about {
      margin-top: 0px;
      margin-left: 30px;
      font-size: 40px;
      font-family: 'Noto Serif KR', serif;
      font-family: 'Roboto Mono', monospace;
  }
  .projects{
    color: white; 
    font-family: 'Noto Serif KR', serif;

    font-family: 'Roboto Mono', monospace;
    margin-left: 50px;
    
  }

  .ex {
      margin-top: 15px;
      margin-left: 0px;
      font-size: 30px;
      font-family: 'Noto Serif KR', serif;
      font-family: 'Roboto Mono', monospace;
  }

  .about1 {
      margin-top: 0px;
      margin-left: 30px;
      font-size: 30px;
      font-family: 'Noto Serif KR', serif;
      font-family: 'Roboto Mono', monospace;
  }

  .about2 {
      margin-top: 0px;
      margin-left: 30px;
      font-size: 26px;
      font-family: 'Noto Serif KR', serif;
      font-family: 'Roboto Mono', monospace;
  }

  .icon {
      float: left;
      margin-left: 10px;
  }

  .icon1 {
      float: left;
      margin-left: 10px;
      margin-top: 3px;
  }

  .icon1 img {
      transition: all 0.2s linear;
  }

  .icon1:hover img {
      transform: scale(1.3);
  }

  .icon img {
      transition: all 0.2s linear;
  }

  .icon:hover img {
      transform: scale(1.3);
  }

  .tag {
      display: inline-block;
      border-radius: 3px;
      padding: .2em .5em .3em;
      border-radius: 2px;
      background: var(--tag-bg);
      color: var(--text-color);
      font-weight: 600;
      margin: .25em .1em;
      margin-right: 20px;
      transition: all 0.2s linear;
      font-family: 'Noto Serif KR', serif;
      font-family: 'Roboto Mono', monospace;
  }

  /* .tag::before */

  .tag:hover {
      transform: scale(1.2);
      color: #ffffff;
      ;
      background-color: #0d834a;

  }

  .tag-lg {
      font-size: 1.0em;
      border-radius: 4px
  }

  .tag-js {
      background: #d7d7d7;
      color: #595959
  }

  body {
      align-items: top;
      background-color: var(--white);
      background: url("https://wallpaperboat.com/wp-content/uploads/2020/04/green-aesthetic-wallpaper-download-full-6.jpg");
      background-attachment: fixed;
      background-position: center;
      background-repeat: no-repeat;
      background-size: cover;
      display: grid;
      height: 100vh;
      place-items: right;
      margin-left: 50px;

  }

  .form__title {
      font-weight: 300;
      margin: 0;
      margin-bottom: 2.25rem;
  }

  .link {
      color: var(--gray);
      font-size: 0.9rem;
      margin: 1.5rem 0;
      text-decoration: none;
  }

  .container {
      background-color: var(--white);
      border-radius: var(--button-radius);
      box-shadow: 0 0.9rem 1.7rem rgba(0, 0, 0, 0.25),
          0 0.7rem 0.7rem rgba(0, 0, 0, 0.22);
      height: var(--max-height);
      max-width: var(--max-width);
      overflow: hidden;
      position: relative;
      width: 100%;

  }



  @keyframes show {

      0%,
      49.99% {
          opacity: 0;
          z-index: 1;
      }

      50%,
      100% {
          opacity: 1;
          z-index: 5;
      }
  }

hyelee.html

<!DOCTYPE html>
<html>

<head>
    <meta charset="UTF-8" />
    <meta http-equiv="X-UA-Compatible" content="IE=edge" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />

    <link href="{{ url_for('static', filename='profile1.css') }}" rel="stylesheet" />
    <script src="https://ajax.googleapis.com/ajax/libs/jquery/3.5.1/jquery.min.js"></script>

    <title>이혜리</title>
    <style>
        .form-floating {
            font-family: 'Noto Serif KR', serif;
            font-family: 'Roboto Mono', monospace;
            margin-bottom: 4px;
            border-radius: 15px;
        }

        .mypost {
            margin: 20px 30px 20px 100px;

            box-shadow: 0px 0px 3px 0px black;
            padding: 10px;
            background-color: transparent;

            font-family: 'Noto Serif KR', serif;
            font-family: 'Roboto Mono', monospace;

            border-radius: 15px;
            border-color: white;
            border-style: outset;
        }

        /* .mypost>button {
            margin-top: 0px;
        } */
    </style>
    <script>
        $(document).ready(function () {
            show_comment();
        });
        function save_comment() {
            let name = $('#name').val()
            let comment = $('#comment').val()
            //formData 객체 생성 전 받아놔야 함, formData 이후에 받으면 데이터가 제대로 저장 x
            let formData = new FormData();

            formData.append("name_give", name);
            formData.append("comment_give", comment);

            fetch('/comment', { method: "POST", body: formData, }).then((res) => res.json()).then((data) => {
                alert(data['msg']);
            });
            window.location.reload()
        }
        function show_comment() {
            fetch('/comment').then((res) => res.json()).then((data) => {
                let rows = data['result']

                $('#comment-list').empty()
                rows.forEach((a) => {
                    let name = a['name']
                    let comment = a['comment']

                    let temp_html = `<div class="mypost" style="width: 420px; height: 120px;" id="${comment}_post">
                                        <div class="form-floating">
                                            <a style="color: white;" type="text" class="ex-comment">
                                                ${comment}
                                            </a>
                                        </div>
                                        <div class="form-floating">
                                            <a style="color: white;" type="text" class="ex-nick">
                                                ㅡ ${name}
                                            </a>
                                        </div>
                                        <div class="button-container">
                                            <button style="color: white;"token interpolation">${name})" type="button" class="comment-btn1" id="${name}_re" value=${comment}>
                                                rewrite
                                            </button>
                                            <button style="color: white;"token interpolation">${name})" type="button" class="comment-btn2" id="${name}" value="${comment}">
                                                delete
                                            </button>
                                        </div>
                                    </div>`

                    $('#comment-list').append(temp_html)
                })
            })
        }

        function delete_comment(name) {

            let give_comment = name.value
            console.log(give_comment)

            let formData = new FormData();

            formData.append("give_comment", give_comment);

            fetch('/delete', { method: "POST", body: formData, }).then((res) => res.json()).then((data) => {
                alert(data['msg'])
            });
            window.location.reload()
        }
        function rewrite_comment(name) {

            let give_comment = name.value
            console.log(give_comment)

            let formData = new FormData();

            formData.append("re_comment", give_comment);

            $('#' + give_comment + '_post').hide();

            let temp_html = `<div class="mypost" style="width: 420px; height: 120px;" id="update-post">
                                <div class="form-floating">
                                    <textarea class="form-control" placeholder="Leave a comment here" id="new-comment"
                                        style="width: 400px; height: 45px; border-radius: 10px"></textarea>
                                </div>
                                <button style="color: white;" type="button" class="btn" id="update">
                                    update
                                </button>
                            </div>`
            $('#comment-list').append(temp_html)

            let updateBtn = document.getElementById("update");

            updateBtn.addEventListener("click", () => {

                let new_comment = $('#new-comment').val()
                $('#update-post').remove()

                formData.append("comment_rewrite", new_comment);
                console.log(new_comment)

                fetch('/rewrite', { method: "POST", body: formData, }).then((res) => res.json()).then((data) => {
                    alert(data['msg']);
                    console.log(data['msg'])
                    //show_comment();
                    window.location.reload()
                });

            });
            //window.location.reload()
        }
    </script>
</head>

<body>
    <div>
        <h1 style="color: white;">Hyelee Lee</h1>
        <hr align="left" width=98% color="white">
        <div class="icon1">
            <!--여기에 아이콘추가!-->
            <img style="height: 35px;width: 35px; object-fit: contain"
                src="">
        </div>
        <div class="icon">
            <img style="height: 40px;width: 40px; object-fit: contain"
                src="https://icon-library.com/images/github-icon-white/github-icon-white-6.jpg"
                onclick="location.href='https://github.com/hyeleenotharry';">
        </div>
        <div class="icon">
            <img style="height: 40px;width: 40px; object-fit: contain"
                src="">
        </div>
        <div class="profile_img">
            <img style="height: 240px;width: 220px; object-fit: contain"
                src="https://images6.fanpop.com/image/photos/40600000/Water-and-Flowers-aesthetic-40635010-540-456.jpg">
            <div>
                <h2 class="about" style="color: white;">About Me</h2>
                <h3 class="about1" style="color: rgb(201, 201, 201);">Something Cool</h3>
                <h4 class="about2" style="color: rgb(255, 255, 255);">Easiest to make</h4>
            </div>
            <div style="margin-left: 300px;">
                <h2 class="about" style="color: white;">Projects</h2>
                <ul class="menu align-center expanded text-center SMN_effect-2" id="menu-pro"
                    style="list-style-type: none;">
                    <li><a href="">Database</a></li>
                    <li><a href="">Algorithm</a></li>
                    <li><a href="">Demo</a></li>
                    <li><a href="">Web</a></li>
                </ul>
            </div>
        </div>
    </div>
    <div class="skill" style="color: white;">What Skill I Have</div>
    <div class="skill-container">
        <a><span class="tag tag-js tag-lg">#javascript</span></a>
        <a><span class="tag tag-js tag-lg">#c</span></a>
        <a><span class="tag tag-js tag-lg">#java</span></a>
        <a><span class="tag tag-js tag-lg">#python</span></a>
        <a><span class="tag tag-js tag-lg">#flask</span></a>
        <a><span class="tag tag-js tag-lg">#html</span></a>
        <a><span class="tag tag-js tag-lg">#CSS</span></a>
        <a><span class="tag tag-js tag-lg">#node.js</span></a>
    </div>
    <div class="bottom">
        <div>
            <h3 class="ex" style="color: rgb(201, 201, 201);">Experience</h3>
            <hr align="left" width=100%>
            <div class="ex" style="color: white;">
                Ewha Woman University
            </div>
            <hr align="left" width=100%>
        </div>
        <div>
            <div class="mypost" style="width: 420px; height: 120px;">
                <div class="form-floating">
                    <input type="text" class="form-control" id="name" placeholder="nickname" style="width: 400px; height: 20px; border-radius: 10px; font-family: 'Noto Serif KR', serif;
                            font-family: 'Roboto Mono', monospace;" />
                </div>
                <div class="form-floating">
                    <textarea class="form-control" placeholder="Leave a comment here" id="comment"
                        style="width: 400px; height: 45px; border-radius: 10px"></textarea>
                </div>
                <button style="color: white;" onclick="save_comment()" type="button" class="btn">
                    share
                </button>
            </div>
            <div id="comment-list">
                <div class="mypost" style="width: 420px; height: 120px;">
                    <div class="form-floating">
                        <a style="color: white;" type="text" class="ex-comment">
                            beautiful!
                        </a>
                    </div>
                    <div class="form-floating">
                        <a style="color: white;" type="text" class="ex-nick">
                            ㅡ abc
                        </a>
                    </div>
                    <div class="button-container">
                        <button style="color: white;" onclick="save_comment()" type="button" class="comment-btn1">
                            rewrite
                        </button>
                        <button style="color: white;" onclick="save_comment()" type="button" class="comment-btn2">
                            delete
                        </button>
                    </div>
                </div>
            </div>
        </div>
    </div>
</body>

</html>

app.py

from flask import Flask, render_template, request, jsonify, after_this_request,redirect,url_for
app = Flask(__name__)
from pymongo import MongoClient
import certifi

ca = certifi.where()

client = MongoClient('mongodb+srv://sparta:test@cluster0.eh7wfh6.mongodb.net/?retryWrites=true&w=majority', tlsCAFile=ca)
db = client.dbsparta

# @app.after_request
# def add_no_cache_header(response):
#     print("No cache")
#     response.headers['Cache-Control'] = 'no-cache, no-store, must-revalidate'
#     response.headers['Pragma'] = 'no-cache'
#     response.headers['Expires'] = '0'
#     return response

@app.route('/')
def home():
   return render_template('index.html')

@app.route('/hyelee')
def hyelee():
   return render_template('hyelee.html')

@app.route("/comment", methods=["POST"])
def guestbook_post():
    name_receive = request.form['name_give']
    comment_receive = request.form['comment_give']

    doc = {
        'name':name_receive,
        'comment':comment_receive
    }
    db.hyelee.insert_one(doc)
    
    return jsonify({'msg': '저장 완료!'})

@app.route("/comment", methods=["GET"])
def guestbook_get():
    all_fans = list(db.hyelee.find({},{'_id':False}))
    return jsonify({'result': all_fans})

@app.route("/delete", methods=["POST"])
def guestbook_delete():
    recieve_comment = request.form['give_comment']
    print(recieve_comment)

    db.hyelee.delete_one({'comment': recieve_comment})
    return jsonify({'msg': '삭제 완료'})

@app.route("/rewrite", methods=["POST"])
def guestbook_rewrite():
    recieve_comment = request.form.get('re_comment',False)
    print("recieve_comment :%s"%recieve_comment)

    if(recieve_comment==False):
       return jsonify({'msg':'값을 입력하십시오'})
    
    rewrite_comment = request.form.get('comment_rewrite',False)
    print("rewrite_comment :%s"%rewrite_comment)

    db.hyelee.update_one({'comment':recieve_comment},{'$set':{'comment':rewrite_comment}})
    return jsonify({'msg': '업데이트 완료'})

@app.route('/webstagram')
def webstagram():
   return render_template('instagram.html')

@app.route('/signUp', methods=["POST"])
def sign_up():
   id_receive = request.form['id_give']
   pwd_receive = request.form['pwd_give']

   doc = {
        'id':id_receive,
        'pwd':pwd_receive
   }
   db.user.insert_one(doc)
    
   return jsonify({'msg': '저장 완료!'})


@app.route('/signIn', methods=["GET","POST"])
def sign_in():
   id_log = request.form.get('id_log', False)
   pwd_log = request.form.get('pwd_log', False)

   id = db.user.find_one({'id':id_log})
   pwd = db.user.find_one({'pwd':pwd_log})

   if(id and pwd):
      print("success")
      @after_this_request
      def add_no_cache_header(response):
          response.headers['Cache-Control'] = 'no-cache, no-store, must-revalidate'
          response.headers['Pragma'] = 'no-cache'
          response.headers['Expires'] = '0'
          return response
      #return redirect('/webstagram')
      return jsonify({'msg':'로그인 성공!'})
   #다른 페이지로 redirect
      
   else:
      print("fail")
      return jsonify({'msg':'사용자를 찾지 못했습니다. 회원가입을 진행해주세요.'})

@app.route('/profile', methods=["POST"])
def profile_post():
   name_receive=request.form['name_give']
   mbti_receive=request.form['mbti_give']
   hobby_receive=request.form['hobby_give']
   profile_receive=request.form['profile_give']
   feed_receive=request.form['feed_give']
   comment_receive=request.form['comment_give']

   doc = {
      'name':name_receive,
      'mbti':mbti_receive,
      'hobby':hobby_receive,
      'profile':profile_receive,
      'feed':feed_receive,
      'comment':comment_receive
   }
   db.profiles.insert_one(doc)

   return jsonify({'msg':'Profile 등록 완료!'})

@app.route('/profile', methods=["GET"])
def profile_get():
   all_profiles = list(db.profiles.find({},{'_id':False}))
   
   return jsonify({'result':all_profiles})

if __name__ == '__main__':
   app.run('0.0.0.0', port=5001, debug=True)
profile
공부 중

0개의 댓글