28 KiB
IMDB Clone React Website
Project Description
We will be creating an IMDB clone, where we will fetch Real-time trending movies data and will show it in grid format, we will be designing the whole application by using Tailwind CSS.
Features of the project
The following will be the features of our IMDB clone:
- The user will be able to view all the latest & trneding movies (IMDB API)
- User can create his own separate watchlist
- User can filter movies according to genre
- User can sort movies according to ratings
- Pagination will be implemented to move from one page to another to get updated data of other movies
- Search feature will be there for movies.
- We will be deploying this to netlify
Wireframe
Implementation
What is tailwind css?
Tailwind CSS is a highly popular and innovative utility-first CSS framework for web development. Unlike traditional CSS frameworks that provide pre-designed components, Tailwind CSS focuses on providing a comprehensive set of utility classes that make it easier to create custom and responsive designs without the need for writing custom CSS.
Following are features of Tailwind CSS:
- Utility-First Approach: Tailwind CSS adopts a utility-first approach, offering a vast collection of small, single-purpose utility classes that can be combined to create complex designs and layouts. This approach promotes code reusability and rapid development.
- Customizable and Configurable: Tailwind is highly customizable through a configuration file, allowing developers to tailor the framework to match their project's specific design requirements. You can customize everything from colors and spacing to typography and breakpoints.
- Responsive Design Made Easy: Creating responsive web designs is simplified with Tailwind CSS. It provides responsive variants of utility classes, enabling developers to adapt the layout and styling of their websites for various screen sizes and devices effortlessly.
- Performance Optimization: Tailwind CSS is designed with performance in mind. It generates optimized CSS files by purging unused classes, resulting in smaller file sizes and faster loading times for web pages.
- Active Community and Ecosystem: Tailwind CSS has a thriving community of developers who contribute to its growth and share resources. Additionally, there are numerous plugins and extensions available that enhance Tailwind's capabilities, making it suitable for a wide range of web development projects.
In this application we will be integrating tailwind to our react app.
- tailwind.confing.js
/** @type {import('tailwindcss').Config} */
module.exports = {
content: [
"./src/**/*.{js,jsx,ts,tsx}",
],
theme: {
extend: {},
},
plugins: [],
}
Src
-
index.js:
- In this code snippet, we're diving into the world of web development with React, a popular JavaScript library for building user interfaces.
- We start by importing some essential tools:
React
andReactDOM
. These are the building blocks of React applications. Think of them as the magic wand and canvas, respectively, for creating interactive web experiences. - We also import a CSS file, './index.css'. This file likely contains styles to make our IMDb web page look visually appealing and engaging.
- The star of our show is the
App
component, which represents the heart and soul of our IMDb website. We import it from './App', indicating that this component will be responsible for displaying our movie and TV show information. - Now, let's set the stage. We create a special variable called
root
usingReactDOM.createRoot(document.getElementById('root'))
. This variable acts as our spotlight, focusing on a specific area of the webpage where our IMDb content will appear. - Finally, it's showtime! We call
root.render(<App />)
, which is the grand moment when ourApp
component takes center stage and begins rendering. This means it starts building the entire IMDb web page, showing movie posters, cast lists, and everything else our users want to see.
Code:
import React from 'react';
import ReactDOM from 'react-dom/client';
import './index.css';
import App from './App';
const root = ReactDOM.createRoot(document.getElementById('root'));
root.render(
<App />
);
- index.css:
Certainly, here's the provided CSS code snippet with an explanatory introduction:
-
index.css
- This CSS code snippet is all about styling our IMDb web page. CSS, or Cascading Style Sheets, is used to control the visual appearance of web content.
- The code begins with
@tailwind
directives. These directives are used in conjunction with the Tailwind CSS framework. They help in defining the base styles, components, and utilities for our web page. Tailwind CSS is a utility-first CSS framework that makes it easy to style web applications. - Following the Tailwind directives, we have styles for the
body
element. These styles define how the main content of our IMDb webpage should look. Key aspects include:- Setting the margin to 0 to remove any default spacing around the page.
- Specifying a font-family to ensure a consistent and readable text display on various devices and browsers.
- Enabling font-smoothing and grayscale to enhance text readability and visual quality.
- Next, there are styles for the
code
element. This is likely used for displaying code snippets or inline code within our IMDb web page. The specified font-family ensures that code is displayed in a monospace or fixed-width font, making it easy to read and distinguish from regular text.
Code:
@tailwind base;
@tailwind components;
@tailwind utilities;
body {
margin: 0;
font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', 'Roboto', 'Oxygen',
'Ubuntu', 'Cantarell', 'Fira Sans', 'Droid Sans', 'Helvetica Neue',
sans-serif;
-webkit-font-smoothing: antialiased;
-moz-osx-font-smoothing: grayscale;
}
code {
font-family: source-code-pro, Menlo, Monaco, Consolas, 'Courier New',
monospace;
}
- App.js:
Certainly, here's the code explanation for the provided App.js
code:
-
App.js
- This JavaScript file is a central part of our IMDb web application. It's where we define how the entire app works and what components it consists of.
- We start by importing several components and libraries that our app depends on:
import './App.css';
imports the CSS styles specific to ourApp
component, which are likely used to style the layout and appearance of the app.Movies
,WatchList
,Navbar
, andBanner
are components imported from their respective files. These components are used to build various parts of our IMDb app, such as displaying movies, the watchlist, the navigation bar, and a banner.BrowserRouter
,Route
, andRoutes
are imported from 'react-router-dom'. These are essential for setting up client-side routing, allowing us to navigate between different pages in our app without a full page reload.
- Inside the
App
function, we set up various pieces of functionality for our IMDb app:- We use
useState
to create two state variables,watchList
andpageNo
. These states help manage the user's watchlist and the current page number. - There are also functions like
handlePrev
andhandleNext
to handle pagination by changing thepageNo
state when users click on previous or next buttons. handleAddToWatchList
andhandleRemoveFromWatchList
functions allow users to add or remove movies from their watchlist. These functions update thewatchList
state and store the updated watchlist in local storage.- The
useEffect
hook is used to retrieve the watchlist data from local storage when the component mounts, ensuring that users don't lose their watchlist when they revisit the app.
- We use
- The
return
statement defines the structure of our IMDb app using theBrowserRouter
andRoutes
. Inside the routes, we render various components based on the route path:- On the root path ('/'), we render the
Banner
andMovies
components, passing them necessary props likewatchList
,pageNo
, and functions for adding/removing from the watchlist. - On the '/watchlist' path, we render the
WatchList
component, providing it with the user's watchlist and functions to remove movies from it.
- On the root path ('/'), we render the
- The entire
App
function represents the main structure of our IMDb web application, orchestrating the rendering of different components and handling user interactions.
Code:
import './App.css';
import Movies from './Components/Movies';
import WatchList from './Components/WatchList';
import Navbar from './Components/Navbar';
import Banner from './Components/Banner';
import { BrowserRouter, Route, Routes } from 'react-router-dom';
import {useState,useEffect} from "react"
function App() {
let [watchList,setWatchList] = useState([]);
let [pageNo,setPageNo] = useState(1);
let handlePrev = ()=>{
if(pageNo>1)
setPageNo(pageNo-1)
}
let handleNext = ()=>{
setPageNo(pageNo+1);
}
let handleAddToWatchList = (movieObj)=>{
// console.log("Inside add to watchlist");
// console.log(movieId);
// watchList.push(movieId); // it will not work since the reference is same
let newWatchList = [...watchList,movieObj];
localStorage.setItem("movieApp",JSON.stringify(newWatchList));
setWatchList(newWatchList);
}
let handleRemoveFromWatchList = (movieObj)=>{
let filteredWatchList = watchList.filter((movie)=>{
return movie.id != movieObj.id;
})
localStorage.setItem("movieApp",JSON.stringify(filteredWatchList));
setWatchList(filteredWatchList);
}
useEffect(()=>{
let moviesFromLocalStorage = localStorage.getItem("movieApp");
if(!moviesFromLocalStorage){
return;
}
setWatchList(JSON.parse(moviesFromLocalStorage));
},[])
return (
<BrowserRouter>
<Navbar/>
<Routes>
<Route path='/' element={
<>
<Banner/>
<Movies watchList={watchList}
setWatchList={setWatchList}
handleAddToWatchList={handleAddToWatchList}
handleRemoveFromWatchList={handleRemoveFromWatchList}
pageNo={pageNo}
handleNext={handleNext}
handlePrev={handlePrev}/>
</>
}></Route>
<Route path='/watchlist' element={
<WatchList watchList={watchList}
setWatchList={setWatchList}
handleRemoveFromWatchList={handleRemoveFromWatchList}/>
}
></Route>
</Routes>
</BrowserRouter>
// <>
// <Navbar/>
// <Banner/>
// <Movies/>
// {/* <WatchList/> */}
// </>
);
}
export default App;
-
App.css
- This CSS code snippet is used to style elements within our IMDb web application. It defines various styles to ensure that the app looks visually appealing and functions correctly.
- The
.App
class sets the text alignment to the center. This class is likely associated with the top-level element of our app. .App-logo
is used to style the IMDb logo. It sets the height to 40vmin (a relative unit based on the viewport size) and disables pointer events. This is often used for logos or images.- The
@media
query checks if the user prefers reduced motion. If not, it applies an animation to.App-logo
to make it spin infinitely for 20 seconds in a linear fashion. This adds a dynamic touch to the logo when motion is allowed. .App-header
styles the header section of our IMDb app. It sets the background color, minimum height, and configures it to be a flex container with content centered both horizontally and vertically. The font size is adjusted relative to the viewport size, and the text color is set to white..App-link
styles links in the app with a blue color (#61dafb). This is a common styling for clickable links.- Finally,
@keyframes App-logo-spin
defines an animation that smoothly rotates an element from 0 degrees to 360 degrees. This is used in conjunction with.App-logo
to create the spinning effect of the IMDb logo.
Code:
.App {
text-align: center;
}
.App-logo {
height: 40vmin;
pointer-events: none;
}
@media (prefers-reduced-motion: no-preference) {
.App-logo {
animation: App-logo-spin infinite 20s linear;
}
}
.App-header {
background-color: #282c34;
min-height: 100vh;
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
font-size: calc(10px + 2vmin);
color: white;
}
.App-link {
color: #61dafb;
}
@keyframes App-logo-spin {
from {
transform: rotate(0deg);
}
to {
transform: rotate(360deg);
}
}
Utility
const genreids = {
28: "Action",
12: "Adventure",
16: "Animation",
35: "Comedy",
80: "Crime",
99: "Documentary",
18: "Drama",
10751: "Family",
14: "Fantasy",
36: "History",
27: "Horror",
10402: "Music",
9648: "Mystery",
10749: "Romance",
878: "Sci-Fi",
10770: "TV",
53: "Thriller",
10752: "War",
37: "Western",
};
export default genreids;
Components
-
Banner.jsx:
- This JavaScript file defines the
Banner
component for our IMDb web application. TheBanner
typically appears at the top of the web page and often showcases a prominent image or information related to the content. - Inside the component function, we use the
useState
anduseEffect
hooks from React to manage component state and perform side effects. useState
initializes a state variable calleddetail
with an initial value ofundefined
. This variable will hold data related to the movie we want to display in the banner.- The
useEffect
hook is used to make an HTTP GET request to the 'https://api.themoviedb.org/3/movie/upcoming' endpoint. This endpoint likely retrieves information about upcoming movies from a movie database. The API key '2816c138913c6ef73d40c883d36fbe56' is included in the request for authentication. - When the response is received successfully, it extracts data from the first movie in the results array and sets it to the
detail
state variable. - The
if
statement checks ifdetail
is still undefined. If it is, the component returns nothing (undefined). This likely prevents the component from rendering until it has retrieved movie data from the API. - If
detail
has a value (i.e., movie data has been fetched), the component renders:- A
div
element with a class that sets the background image of the banner. Theurl()
function in thestyle
attribute dynamically sets the background image using theposter_path
property from thedetail
data. - Inside this
div
, there's anotherdiv
element that contains text. This is likely the movie title displayed on top of the banner image. - The styling includes making the text white, applying a semi-transparent gray background, and centering the text both horizontally and vertically. The
text-xl
class sets the font size to extra-large.
- A
- The
Banner
component fetches data about an upcoming movie and displays it as a banner on the IMDb webpage, creating an attractive and engaging visual element for users.
- This JavaScript file defines the
Code:
import axios from "axios";
import { useEffect, useState } from "react";
function Banner(){
let [detail,setDetail] = useState(undefined)
useEffect(()=>{
axios.get('https://api.themoviedb.org/3/movie/upcoming?api_key=2816c138913c6ef73d40c883d36fbe56')
.then(function(res){
// console.log(res);
// console.log(res.data.results[0].poster_path);
let data = res.data.results[0]
setDetail(data);
})
},[]);
if(detail == undefined){
return
}
return(
<div className="h-[20vh] md:h-[70vh] bg-cover bg-center flex items-end"
style={{backgroundImage:`url(https://image.tmdb.org/t/p/original/${detail.poster_path})`}}>
<div className="text-xl text-white bg-gray-900/60 w-full p-4 text-center ">
{detail.title}
</div>
</div>
)
}
export default Banner;
-
MovieCard.jsx:
- This JavaScript file defines the
MovieCard
component. AMovieCard
is a visual representation of a movie that users can interact with in our IMDb web application. - The component function takes several props:
movieObj
: An object containing information about the movie, including itsid
.handleAddToWatchList
: A function to add the movie to the user's watchlist.handleRemoveFromWatchList
: A function to remove the movie from the user's watchlist.name
: The name or title of the movie.watchList
: An array representing the user's watchlist.poster_path
: A URL to the movie's poster image.
- Inside the component, there's a function
isContain(movieObj)
that checks if themovieObj
is already in the user's watchlist. It iterates through thewatchList
array and returnstrue
if it finds a movie with the sameid
, indicating that the movie is in the watchlist. Otherwise, it returnsfalse
. - The component's
return
statement defines the structure and appearance of aMovieCard
. It's adiv
element with the following features:- A fixed height and width (
h-[40vh]
andw-[200px]
) to maintain a consistent size for movie cards. - The
bg-center
andbg-cover
classes ensure that the background image (the movie poster) is centered and covers the entire card. - The
rounded-xl
class rounds the corners of the card, giving it a pleasant look. - On hover, the card slightly scales up (
hover:scale-110
) and the cursor changes to a pointer (hover:cursor-pointer
), providing a visual cue that the card is interactive.
- A fixed height and width (
- Inside the card, there's a dynamic icon that changes based on whether the movie is in the user's watchlist or not. If the movie is in the watchlist, it displays a "remove" icon (❌), and clicking it calls the
handleRemoveFromWatchList
function. If the movie is not in the watchlist, it displays an "add" icon (😁), and clicking it calls thehandleAddToWatchList
function. - Below the icon, there's a text element displaying the movie's name or title. It has a white text color, a semi-transparent gray background, and is centered both horizontally and vertically. The
text-xl
class sets the font size to extra-large. - This
MovieCard
component is a reusable card for displaying movie information and interacting with the user's watchlist. It provides a visually appealing and user-friendly way to add or remove movies from the watchlist.
- This JavaScript file defines the
Code:
export default function MovieCard(props){
let {movieObj,handleAddToWatchList,handleRemoveFromWatchList,name,watchList,poster_path} = props;
function isContain(movieObj){
for(let i=0;i<watchList.length;i++){
if(watchList[i].id == movieObj.id){
return true;
}
}
return false;
}
return(
<div className="h-[40vh] w-[200px] bg-center bg-cover
rounded-xl hover:scale-110 duration-300 hover:cursor-pointer
flex flex-col justify-between items-end overflow-hidden"
style={{
backgroundImage: `url(https://image.tmdb.org/t/p/original/${poster_path})`,
}}>
{isContain(movieObj)?
<div onClick={()=>handleRemoveFromWatchList(movieObj)} className="m-4 bg-gray-900
flex justify-center items-center
h-8 w-8 rounded-lg">
❌
</div>
:<div onClick={()=>handleAddToWatchList(movieObj)}
className="m-4 bg-gray-900
flex justify-center items-center
h-8 w-8 rounded-lg">
😍
</div>
}
<div className="text-xl text-white
bg-gray-900/60 w-full p-2 text-center ">
{name}
</div>
</div>
)
}
-
Movies.jsx
- This JavaScript file defines the
Movies
component, which is responsible for displaying a list of trending movies on our IMDb web application. - The component function takes several props:
watchList
andsetWatchList
: These props are used to manage the user's watchlist.handleAddToWatchList
andhandleRemoveFromWatchList
: Functions for adding and removing movies from the watchlist.pageNo
: The current page number, which is used for pagination.handleNext
andhandlePrev
: Functions for navigating to the next and previous pages of trending movies.
- Inside the component, there's a
useState
hook that initializes a state variablemovies
as an empty array. This variable will store the list of trending movies fetched from an API. - The
useEffect
hook is used to make an HTTP GET request to the 'https://api.themoviedb.org/3/trending/movie/day' endpoint. ThepageNo
prop is included in the request to retrieve a specific page of trending movies. The API key '2816c138913c6ef73d40c883d36fbe56' is used for authentication. - When the response is received successfully, it extracts the list of trending movies from
res.data.results
and sets it to themovies
state variable. - The component's
return
statement defines the structure of the Movies component:- A title ("Trending Movies") displayed in a large and bold font at the top.
- A flex container (
flex flex-wrap justify-around gap-8
) that arranges MovieCard components in a responsive grid layout. EachMovieCard
represents a trending movie.
- Inside the flex container, a
.map()
function is used to iterate through themovies
array and generate aMovieCard
component for each movie. Thekey
prop is set to the movie'sid
for React's reconciliation. - The
MovieCard
component is passed various props, including movie information, watchlist data, and functions for adding and removing movies from the watchlist. - After displaying the list of movies, a
Pagination
component is rendered. This component allows users to navigate between pages of trending movies using thepageNo
,handleNext
, andhandlePrev
props. - The
Movies
component is responsible for rendering trending movies and providing user-friendly interactions for adding/removing movies from the watchlist and pagination.
- This JavaScript file defines the
Code:
import MovieCard from "./MovieCard";
import axios from "axios"
import {useEffect, useState} from "react"
import Pagination from "./Pagination";
function Movies(props){
let {watchList,setWatchList,
handleAddToWatchList,handleRemoveFromWatchList,
pageNo,handleNext,handlePrev} = props;
let [movies,setMovies] = useState([]);
useEffect(()=>{
axios.get(`https://api.themoviedb.org/3/trending/movie/day?api_key=2816c138913c6ef73d40c883d36fbe56&page=${pageNo}`)
.then(function(res){
// console.log(res);
// console.log(res.data.results);
setMovies(res.data.results);
})
},[pageNo])
return(
<div className="p-5">
<div className="text-2xl m-5 font-bold text-center">
Trending Movies
</div>
<div className="flex flex-wrap justify-around gap-8">
{movies.map((movieObj)=>{
// console.log(movieObj);
return <MovieCard key={movieObj.id}
movieObj={movieObj}
name={movieObj.title}
poster_path={movieObj.poster_path}
watchList = { watchList}
handleAddToWatchList = {handleAddToWatchList}
handleRemoveFromWatchList = {handleRemoveFromWatchList}/>
})}
</div>
<Pagination pageNo={pageNo}
handleNext={handleNext}
handlePrev={handlePrev}/>
</div>
)
}
export default Movies;
-
Navbar.jsx
- This JavaScript file defines the
Navbar
component, which represents the navigation bar at the top of the IMDb web application. The navigation bar typically contains links to different sections or pages of the website. - Inside the component function, we use the
Link
component from the 'react-router-dom' library to create navigation links. This allows users to navigate between different sections of the IMDb website without a full page reload, providing a seamless user experience. - The component renders the following elements:
- An
img
element with a source (src
) that appears to be a base64-encoded image. This is likely the IMDb logo. Thealt
attribute is empty, indicating that this is a decorative image without alternative text. - A
Link
component with ato
prop pointing to the root path ("/") of the application. This link represents the "Movies" section of the website. It has a large font size, a bold font-weight, and a hover effect that changes the cursor to a pointer, making it clear that it's clickable. - Another
Link
component with ato
prop pointing to the "/watchlist" path. This link represents the "Watchlist" section of the website. Similar to the "Movies" link, it has a large font size, bold font-weight, and a hover effect.
- An
- The
Navbar
component provides essential navigation options, allowing users to easily switch between the "Movies" and "Watchlist" sections of the IMDb website, enhancing the overall user experience.
- This JavaScript file defines the
import { Link } from "react-router-dom";
function Navbar(){
return(
<div className="flex items-center p-3">
<img className="h-[50px] mx-6" src="" alt="" />
<Link to={"/"} className="text-3xl mx-6 font-bold text-sky-600 hover:cursor-pointer">Movies</Link>
<Link to={"/watchlist"} className="mx-6 text-3xl font-bold text-sky-600 hover:cursor-pointer">WatchList</Link>
</div>
)
}
export default Navbar;