저번에 만들었던 팀원 소개 페이지가 조금 심심한 것 같아 간단하게 자기소개 페이지를 만들어보았다.
전화, 깃허브, 이메일 아이콘에 호버 효과를 넣었다. 깃허브 아이콘은 클릭하면 내 깃허브 페이지로 이동할 수 있다.
Projects 의 메뉴에서 해당하는 버튼을 누르면 그 분야에서 진행했던 프로젝트 페이지로 넘어가게 된다. 메뉴는 호버 css를 추가해서 마우스를 올릴 시
이러한 효과가 나타나도록 했다.
<a href>
태그를 통해 클릭하면 해당 프로젝트 깃허브 페이지로 넘어갈 수 있다.
내가 가진 기술 스택들을 태그로 추가한 뒤 간단한 호버 효과를 넣어주었다.
닉네임과 코멘트를 입력하면 아래에 댓글이 보이도록 만들었다. 작성된 댓글은 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)