Skip to content

Latest commit

 

History

History
246 lines (202 loc) · 10.7 KB

README.md

File metadata and controls

246 lines (202 loc) · 10.7 KB

🎬 Moogle, 세상의 모든 영화 🎬

image

1️. 프로젝트 개요

  • Moogle은 TMDB API를 기반으로 아래와 같은 서비스를 제공합니다 🎥
    1. 영화 정보 검색 및 조회
    2. 영화 추천 (유저 좋아요 기반, 현재 상영중, 장르별)
    3. 영화에 대한 리뷰 작성 및 좋아요 남기기
    4. 유저간 팔로우 기능, 리뷰 댓글 작성 등을 통해 다른 유저들과도 소통할 수 있는 영화 커뮤니티

2. 기술 스택

  • Front-End : HTML, CSS, SCSS, Bootstrap, JavaScript, Vue, Vuex

  • Back-End : Django, Python, SQLite

3. 팀 구성

  • 개발 일정 : 11/16 ~ 11/24
  • 역할 분담
김경림 류태규 정효상
Vue, JavaScript, Django, Python Django, Python Vue, JavaScript, Django, Python
FrontEndBackEnd BackEnd FrontEndBackEnd

4. 프로젝트 구조

  • front
📦final-pjt-front
┣ 📂public
┗ 📂src
  ┣ 📂api
  ┣ 📂assets
  ┣ 📂components
  ┃ ┣ 📂movie
  ┃ ┗ 📂profile
  ┣ 📂plugins
  ┣ 📂router
  ┣ 📂store
  ┣ 📂views
  ┃ ┣ 📂accounts
  ┃ ┣ 📂movie
  ┣ 📜App.vue
  ┗ 📜main.js
  • back
📦final-pjt-back
 ┣ 📂accounts
 ┣ 📂final_pjt
 ┣ 📂media
 ┃ ┗ 📂users
 ┣ 📂movies
 ┃ ┗ 📂fixtures
 ┣ 📜db.sqlite3
 ┗ 📜manage.py

5. UI & 주요 기능

0. 메인 화면
main_page
- 기본 테마: 영화들이 돋보일 수 있는 어두운 배경 + 보라색 컬러
- 로고: 네온사인처럼 깜빡이는 효과 + 눈의 피로감을 덜기 위해 잠시 쉬어가는 인터벌
1. 회원가입 2. 로그인

영화 관련 랜덤 이미지 배경

영화 관련 랜덤 이미지 배경
3. 로그아웃 4. 영화추천 (로그인 X)

로그인 안했을 시 현재 인기 있는 영화 추천 + 기본 문구 (당신만의 영화추천을 보고싶다면 로그인)
3-1. 영화추천 (로그인 O, 좋아요 X) 3-2. 영화추천 (로그인 O, 좋아요 O)

로그인 했으나 좋아요 한 영화가 없을 경우 좋아요 유도 문구

로그인 완료 + 좋아요 한 영화가 1개 이상일 경우 맞춤 영화 추천
4. 현재 상영중, 액션, 로맨스 영화 추천 5. 영화 상세 정보
ezgif com-gif-maker (1)
- 줄거리, 장르, 별점, 감독, 배우 등 상세 정보 제공
- 비슷한 영화 추천 제공
6. 영화 좋아요 (로그인 필요) 7. 영화 리뷰 작성
8. 영화 리뷰 수정, 삭제 9. 영화 리뷰 좋아요 및 댓글
review_fix
10. 유저 프로필 (팔로우, 리뷰, 좋아요, 랭킹) 11. 유저간 팔로우, 언팔로우

- 좋아요 기반 추천을 더욱 강화하기 위해 좋아요 개수에 따른 랭킹 시스템 도입
follow

|

6. 주요 기능 코드

선택한 영화가 DB에 있다면 DB 정보 불러오기, 없다면 DB 저장후 해당 정보 불러오기
getMovieDetail() {
  // 먼저 DB에 있는지 확인
  axios({
    method: 'get',
    url: API_URL + `/movies/${this.$route.params.movie_id}/`,
  })
  .then((res) => {
    this.movie = res.data
  })
  .catch((error) => {
    console.log(error)

    // DB에 없으면 TMDB에서 가져온 데이터를 DB에 저장
    axios({
      method: 'get',
      url: API_URL + `/movies/${this.$route.params.movie_id}/`,
      data: {
        id: this.$route.params.movie_id,
      }
    })
      .then((response) => {
        console.log(response)
      })
      .catch((error) => {
        console.log(error)
      })
  })
},

유저 좋아요 기반 영화 추천
@api_view(['GET'])
def recommend(request,user_pk):
    print(request)
    file_path = "movies/fixtures/movies.json"

    with open(file_path, 'r', encoding="UTF-8") as f:
        data = json.load(f)
    new_data = []
    for d in data:
        new_data.append({
            'pk': d['pk'],
            'adult': d['fields']['adult'],
            'overview': d['fields']['overview'],
            'title': d['fields']['title'],
            'poster_path': d['fields']['poster_path'],
            'genres': d['fields']['genres'],
            'vote_average': d['fields']['vote_average'],
        })

    new_data = pd.DataFrame(new_data)

    new_data['overview'].isnull().sum()
    new_data['overview'] = new_data['overview'].fillna('')
    new_data['overview'].isnull().sum() #0 으로 바뀜 내적하면 모두 0 나옴

    tfidf=TfidfVectorizer(stop_words='english') #불용어 제거
    tfidf_mat=tfidf.fit_transform(new_data['overview']).toarray()

    def cos_sim2(X,Y):
        return np.dot(X,Y)/((norm(X)*norm(Y))+1e-7)

    def top_match_ar2(new_data, name, rank=5,simf=cos_sim2):
        sim=[]
        for i in range(len(new_data)):
            if name != i:
                sim.append((simf(new_data[i],new_data[name]),i))
        sim.sort()
        sim.reverse()
        return sim[:rank]

    
    user = get_object_or_404(get_user_model(),pk = user_pk)
    lst1 = list(user.like_movies.all().values())    # 좋아요 누른 영화 리스트
    lst = []    # 새로운 리스트
    movieList = []

    if len(lst1) >= 3:                      # 좋아요를 누른 영화가 3개 이상일 경우
        for elt in lst1:
            lst.append(elt['title'])
    else:                                   # 좋아요를 누른 영화가 3개 미만일 경우
        lst1.append(list(Movie.objects.all().values),3 )
        for elt in lst1:
            lst.append(elt['title'])                        
        
    movieList = random.sample(lst,3)
    recommend_lst = set()
    res_list = []
    for movie_name in movieList:
        # 여기에 영화 이름 동적으로 할당
        movie_idx = list(new_data['title']).index(movie_name)
        for sim, movie_id in top_match_ar2(tfidf_mat, movie_idx ,20):
            res_list.append(str({'id': new_data.loc[movie_id,'pk'], 'title' :new_data.loc[movie_id,'title'], 'poster_path' :new_data.loc[movie_id,'poster_path'], 'vote_average' :new_data.loc[movie_id,'vote_average']}))
        for res in res_list[:30]:
            recommend_lst.add(res)
    result = []
    for i in recommend_lst:
        i = eval(i)
        result.append(i)
    return Response(result)

7. 개발 이슈 및 트러블 슈팅

  • 매일매일 개발 일지를 작성하며 이슈에 대한 트러블 슈팅도 함께 기록하였습니다.
트러블슈팅 예시

image

image

8. 프로젝트를 마치며

김경림

배운 점

  1. 앱의 모든 기능, 디자인, 컴포넌트 하나하나 직접 담당하여 설계하고 구현해보게 되어 뿌듯했다.
  2. 프론트엔드, 백엔드를 함께 기획하고 구현하는 경험
  3. 협업시에는 서로 기본 용어에 대해 동일하게 정의하고 생각하고 있는지부터 점검해야 소통이 보다 쉬워진다는 것을 재확인하였다. (알고보면 서로 같은 이야기를 하고 있는 경우가 많다!)
  4. SCSS, Vuex 등 추가적으로 도입이 필요한 부분에 대해 고민해보고 당위성을 따져볼 수 있었다.
  5. 모듈화의 중요성, 페이지나 컴포넌트 구조를 정확히 생각하고 짜는 것의 중요성
    • 예상치 못하게 구조에 변경이 필요할 때 했던 일을 다시 하는 시간과 빈도가 현저하게 줄기 때문이다.

아쉬운 점

  • 프로젝트 기간이 너무 짧아 git flow, github flow 등 브랜치 전략 및 github의 이슈 등 기본 기능들을 활용하지 못한점이 아쉽다.
  • 최적화를 어떻게 더 할수 있을지 알고싶다. (Vuex로 중앙 저장소에서 데이터 관리를 적절하게 한 것인지, 불필요하게 재렌더링이 발생한 부분이 있는지)
  • 컴포넌트 구조를 더욱 개선하고 싶다.