Giter VIP home page Giter VIP logo

disneyplus-clone's Introduction

다시한번 기초부터 시작해보는 프로젝트(.with 패스트캠퍼스)

Disney_plus 클론 코딩

  • 본 프로젝트는 저의 실력향상 및 기초부터 천천히 다시 배워보고자 시작하는 개인 프로젝트입니다.
  • (주의) 패스트캠퍼스 개인 강의를 통해 클론코딩을 할 예정이니 하나씩 천천히 배우고 올릴 예정이니 많이 늦을수 도있습니다. 퐈이팅!!
  • 개인 프로젝트이기 때문에 commit message와 pr, branch의 규칙을 따로 정하지 않았습니다. 참고 부탁드리겠습니다.
  • themovieDB를 사용하여 영화의 정보가 한정적이고 영화를 재생할 수 없어서 iframe 기능을 사용하여 예고편 영상을 불러왔습니다.
  • 베포 : Firebase
  • 베포 주소 <= 클릭시 베포사이트로 넘어갑니다.

1일차

  • 기본 환경 설정 및 axios 설치

    1. create-react-app으로 기본 세팅을 완료하였습니다.
    2. npm i axios --save으로 axios를 설치하였습니다.
    • Axios: Axios는 브라우저, node.js를 위한 Promise API를 활용하는 HTTP 비동기 통신 라이브러리입니다. 쉽게 말하자면 프론트엔드랑 백엔드의 통신을 쉽게하기 위해 Ajax와 더불어 사용합니다.
    1. Axios 인스턴스화를 하는 이유는?
    • 중복된 부분을 계속 입력하지 않아도 되기 때문입니다.
    1. 전체 구조를 setting 하였습니다.
    2. npm i styled-components --save css 사용을 위해 styled-components를 설치하였습니다.
    • styled component는 CSS-in-JS라고 하는 Javascript 파일 안에서 CSS를 처리 할 수 있게 해주는 대표적인 라이브러리입니다.
    1. 상단 nav작업을 시작하였고 이미지와 스크롤을 일정부분 내리면 색상이 변경하는 이벤트를 추가하였습니다.
    const [show, setShow] = useState(false);
    
    useEffect(() => {
      window.addEventListener("scroll", () => {
        if (window.scrollY > 50) {
          setShow(true);
        } else {
          setShow(false);
        }
      });
      return () => {
        window.removeEventListener("scroll", () => {});
      };
    }, []);


    2일차

    • Banner 완성

      1. themovieDB에서 데이터를 가져와 axios를 활용하여 영화의 정보 불러오기
      const fetchData = async () => {
        // 현재 제공되는 영화 정보를 가져오기(여러가지 영화)
        const request = await axios.get(requests.fetchNowPlaying);
        // 여러 영화중 영화 하나의 ID를 가져오기
        const movieId =
          request.data.results[
            Math.floor(Math.random() * request.data.results.length)
          ].id;
        const { data: movieDetail } = await axios.get(`movie/${movieId}`, {
          params: {
            append_to_response: "videos",
          },
        });
        setMovie(movieDetail);
      };
      1. 영화의 상세정보가 100글자를 넘어가면 ...이 표시되도록 기능을 추가하였습니다
      // 설명글이 100자 이상이라면 글자를 자른 후 ...을 붙인다.
      const truncate = (str, n) => {
        return str?.length > n ? str.substring(0, n) + "..." : str;
      };
      <p className="bannerDescription">{truncate(movie?.overview, 100)}</p>
      1. 영상정보가 있는 영화는 iframe기능을 사용하여 예고편식의 영상을 확인할수 있게 만들었습니다.
      if (isClicked) {
        return (
          <>
            <Container>
              <HomeContainer>
                <Iframe
                  src={`https://www.youtube.com/embed/${movie.videos.results[0].key}
                ?controls=0&autoplay=1&loop=1&mute=1&playlist=${movie.videos.results[0].key}`}
                  width="640"
                  height="360"
                  frameborder="0"
                  allow="autoplay; fullscreen"
                ></Iframe>
              </HomeContainer>
            </Container>
            <button onClick={() => setIsClicked(false)}>나가기</button>
          </>
        );
      } else {
        ("기존 배너");
      }



3일차

  • Row 컴포넌트 완성

    1. api폴더에 만들어놓은 request.js를 활용하여 카테고리별로 나타나게끔 만들었습니다
    <Row title="Trending Now" id="TN" fetchUrl={requests.fetchTrending} />
    <Row title="Top Rated" id="TR" fetchUrl={requests.fetchTopRated} />
    <Row title="Action Movies" id="AM" fetchUrl={requests.fetchActionMovies} />
    <Row title="Comedy Movies" id="CM" fetchUrl={requests.fetchComedyMovies} />
    1. arrow 버튼을 클릭시 slide 기능이 되도록 구현하였습니다.
    <span
      className="arrow"
      onClick={() => {
        document.getElementById(id).scrollLeft += window.innerWidth - 80;
      }}
    >
      {">"}
    </span>
    • category 컴포넌트 완성
      1. 제공받은 동영상과 사진으로 카테고리를 만들었습니다.
      2. 카테고리를 hover시 브랜드별로 영상이 재생되게끔 만들었습니다.


4일차

  • React Router Dom

    • React Router Dom을 사용하면 웹 앱에서 동적 라우팅을 구현할수 있다. 라우팅이 실행 중인 앱 외부의 구성에서 처리되는 기존 라우팅 아키텍처와 달리 React Router Dom은 앱 및 플랫폼의 요구 사항에 따라 컴포넌트 기반 라우팅을 용이하게 한다.
    • 설치 방법 : npm i react-router-dom --save
  • Single Page Application(SPA)

    • 리액트는 SPA이기 때문에 하나의 index.html 템플릿 파일을 가지고 있습니다. 하나의 템플릿에 js를 이용해서 다른 컴포넌트를 이 index.html 템플릿에 넣으므로 페이지를 변경해주게 된다. 이때 이 React Router Dom 라이브러리가 새 컴포넌트로 라우팅 및 탐색을 하고 렌더링 하는데 도움을 주게 된다.
  • Modal 컴포넌트 완성

    1. props를 활용하여 영화에 대한 상세페이지를 modal창으로 나타나게 만들었습니다.
    • 모달창을 제외한 어느곳을 눌러도 모달창이 닫히게 설정하였습니다.
    const MovieModal = ({
      backdrop_path,
      title,
      overview,
      name,
      release_date,
      first_air_date,
      vote_average,
      setModalOpen
    }) => {}
    1. React Router Dom을 사용하여 라우팅을 위한 파일을 변경하였습니다


5일차

  • URLSearchParams

    • URLSearchParams인터페이스는 URL의 쿼리 문자열을 대상으로 작업할 수 있는 유틸리티 메서드를 정의한다.
  • Debounce

    • Debounce function은 사용자가 미리 결정된 시간 동안 타이핑을 멈출 때까지 keyup이벤트의 처리를 지연시킵니다.
  • searchPage

    1. URLSearchParams를 활용하여 검색페이지 데이터를 불러오는데 성공하였습니다.
    1. 검색결과에 맞게 영화가 표시되도록 css를 적용 시켰고 결과가 없을시 결과가 없다는 내용이 나타나게 적용시켰습니다.
    1. useDebounce hook을 만들어서 search페이지에 적용시켰습니다
    import { useState, useEffect } from 'react'
    
    export const useDebounce = (value, delay) => {
      const [debounceValue, setDebouncedValue] = useState(value)
    
      useEffect(() => {
        const handler = setTimeout(() => {
          setDebouncedValue(value)
        }, delay);
        return () => {
          clearTimeout(handler)
        }
      }, [value, delay])
    
      return debounceValue
    }


6일차

  • useParams

    • useParams는 URL에 포함되어있는 Key, Value 형식의 객체를 반환해주는 역할을 한다. Route 부분에서 Key를 지정해주고, 해당하는 Key에 적합한 Value를 넣어 URL을 변경시키면, useParams를 통해 Key, Value 객체를 반환받아 확인할 수 있다.
  • DetailPage

    1. useParams를 사용해 이미지를 불러왔고 상세페이지에 나타나도록 했습니다 (이미지만 불러오고 다른작업은 추후 천천히 완성할 예정입니다.)
  • Modal창 외부 클릭시 닫기 기능

    1. custom Hooks를 만들어 기능을 완성하였습니다.
    export default function useOnClickOutside(ref, handler) {
      useEffect(() => {
        const listener = (event) => {
          if(!ref.current || ref.current.contains(event.target)) {
            return
          }
          handler()
        }
        document.addEventListener("mousedown", listener)
        document.addEventListener("touchstart", listener)
        return () => {
          document.removeEventListener("mousedown", listener)
          document.removeEventListener("touchstart", listener)
        }
      }, [ref, handler])
    }
  • swiper 모듈을 이용한 터치 슬라이드 구현

    • 설치 방법 : npm i swiper



7일차

  • FireBase를 사용하여 베포 와 로그인, 인증을 진행할 예정입니다.

    • 설치방법 : npm i firebase --save
  • loginPage

    1. loginPage 마크업 및 스타일링
    2. login 버튼을 클릭시 handle Auth 함수를 호출하게 했습니다 (firebase의 google 로그인을 사용)
    import { getAuth, GoogleAuthProvider, signInWithPopup } from 'firebase/auth'
    
    const Nav = () => {
      const auth = getAuth()
      const provider = new GoogleAuthProvider()
      const handleAuth = () => {
        signInWithPopup(auth, provider)
        .then(result => {
        
        })
        .catch(error => {
          console.log(error);
        })
      }
      return (
        <Login onClick={handleAuth}>Login</Login>
      )
    }
    1. Firebase를 이용한 인증 체크
    • Firebase를 통해 로그인 및 비로그인 상태를 인증하는 기능을 적용시켰습니다.
    useEffect(() => {
      onAuthStateChanged(auth, (user) => {
        if(user) {
          if(pathname === '/') {
            navigate('/main')
          }
        } else {
          navigate('/')
        }
      })
    }, [auth, navigate, pathname])


8일차

  • localStorage는 웹 브라우저에서 제공하는 저장소의 일종입니다. 이를 사용하면 웹 페이지가 닫히거나 리로드되어도 데이터가 유지됩니다. 즉, 영구적인 데이터 저장을 가능하게 해줍니다.

    1. SignOut 기능을 완성하였습니다.
      const handleLogOut = () => {
        signOut(auth).then(() => {
          setUserData({})
          navigate(`/`)
        }).catch((error) => {
          alert(error.message)
        })
      }
    
      <SignOut>
        <UserImg src={userData.photoURL} alt={userData.displayName} />
        <DropDown>
          <span onClick={handleLogOut}>Sign Out</span>
        </DropDown>
      </SignOut>
    1. localStorage에 userData 담기
    • localStorage에 userData 값을 담아서 페이지를 refresh 해도 userData가 계속 남아 있을 수 있게 해주기
    const initialUserData = localStorage.getItem('userData') ?
      JSON.parse(localStorage.getItem('userData')) : {}
    const [userData, setUserData] = useState(initialUserData)
    
    const handleAuth = () => {
      signInWithPopup(auth, provider)
      .then(result => {
        console.log('result.user', result.user);
        setUserData(result.user)
        localStorage.setItem('userData', JSON.stringify(result.user))
      })
      .catch(error => {
        console.log(error)
        alert(error.message)
      })
    }


트러블 슈팅

  • 해당 error를 발견하였고 문제는 간단하게 해결하였습니다.
    • 코드에서 'show'라는 속성을 사용하고 있는데, 이는 boolean형 값입니다. HTML에서는 boolean형의 속성을 인식하지 않기 때문에 이러한 error가 나타났던 것이었습니다. 이 error는 NavWrapper 컴포넌트에서 발생하였고, 'show'라는 prop을 받아서 스타일링에 사용하고 있는 것이었습니다.

수정방법 : <NavWrapper show={show}> => <NavWrapper show={show ? "true" : undefined}>

'show'가 스타일링에 필요한 boolean 값이기 떄문에, 이를 문자열로 변환하거나 undefined를 전달하는 방식으로 해결하였고 error는 사라졌습니다.

  • DropDown을 찾지 못한다는 에러를 발견하였습니다.
    • 해당 error는 DropDown 스타일 컴포넌트를 선언한 코드의 위치가 문제였습니다. SignOut 스타일 컴포넌트 안에서 DropDown 스타일을 참조하고 있지만, 실제로 DropDown의 선언은 SignOut 이후에 이루어져 있었습니다. 이로 인해 DropDown 참조 시점에는 DropDown이 아직 정의되지 않았기 때문에 에러가 발생했던 것이었습니다.

수정방법 : DropDown 스타일 컴포넌트를 SignOut 선언 전에 위치시키는 것이 해결방법이었습니다.



느낀점

기초부터 다시한번 천천히 앞으로 나아가며 놓친부분을 공부할 수 있었고 앞으로도 계속 부족한 부분을 채워 나가도록 노력할 것입니다.



프로젝트 사용법

  • 패키지 설치
npm i 또는 npm install
  • 클라이언트 실행
npm run start


disneyplus-clone's People

Contributors

lkj970524 avatar

Watchers

 avatar

Recommend Projects

  • React photo React

    A declarative, efficient, and flexible JavaScript library for building user interfaces.

  • Vue.js photo Vue.js

    🖖 Vue.js is a progressive, incrementally-adoptable JavaScript framework for building UI on the web.

  • Typescript photo Typescript

    TypeScript is a superset of JavaScript that compiles to clean JavaScript output.

  • TensorFlow photo TensorFlow

    An Open Source Machine Learning Framework for Everyone

  • Django photo Django

    The Web framework for perfectionists with deadlines.

  • D3 photo D3

    Bring data to life with SVG, Canvas and HTML. 📊📈🎉

Recommend Topics

  • javascript

    JavaScript (JS) is a lightweight interpreted programming language with first-class functions.

  • web

    Some thing interesting about web. New door for the world.

  • server

    A server is a program made to process requests and deliver data to clients.

  • Machine learning

    Machine learning is a way of modeling and interpreting data that allows a piece of software to respond intelligently.

  • Game

    Some thing interesting about game, make everyone happy.

Recommend Org

  • Facebook photo Facebook

    We are working to build community through open source technology. NB: members must have two-factor auth.

  • Microsoft photo Microsoft

    Open source projects and samples from Microsoft.

  • Google photo Google

    Google ❤️ Open Source for everyone.

  • D3 photo D3

    Data-Driven Documents codes.