Built Twitter Like App - 17

Install Spinners

1
npm install react-spinners

Add User Profile Page

add file pages/users/[userId].tsx

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
import { useRouter } from "next/router";
import { ClipLoader } from "react-spinners"

import Header from "@/components/Header";
import useUser from "@/hooks/useUser";
import UserHero from "@/components/users/UserHero";

const UserView = () => {
const router = useRouter();
const { userId } = router.query;

const { data: fetechedUser, isLoading } = useUser(Number(userId));

if (isLoading || !fetechedUser) {
return (
<div className="
flex
justify-center
items-center
h-full
">
<ClipLoader color="lightblue" size={90} />
</div>
)
}

return (
<>
<Header showBackArrow label={fetechedUser?.name} />
<UserHero userId={ userId as string } />
</>
);
}

export default UserView;
More...

Built Twitter Like App - 16

Add User List to FollowBar

modify components/layout/FollowBar.tsx

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
import useUsers from "../../hooks/useUsers";
import Avatar from "../Avatar";

const FollowBar =() =>{
const { data: users = [] } = useUsers();

if(users.length === 0) {
return null;
}

return (
<div className="px-6 py-4 hidden lg:block">
<div className="bg-neutral-800 rounded-xl p-4">
<h2 className="text-white text-xl font-semibold">Who to follow</h2>
<div className="flex flex-col gap-6 mt-4">
{users.map((user:Record<string,any>) => (
<div key={user.id} className="flex flex-row gap-4">
<Avatar userId={user.id} />
<div className="flex flex-col">
<p className="
text-white
font-semibold
text-m
">{user.name}</p>
<p className="text-neutral-400 text-sm">
@{user.username}
</p>
</div>
</div>
))}
</div>
</div>
</div>
);
}

export default FollowBar
More...

Built Twitter Like App - 15

Add Avatar Component

Add file components/Avatar.tsx

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
import React from "react";

interface AvatarProps {
userId: string;
isLarge: boolean;
hasBorder: boolean;
}

const Avatar: React.FC<AvatarProps> =({
userId,
isLarge,
hasBorder
}) =>{
return (
<div> </div>
);
}

export default Avatar;
More...

Built Twitter Like App - 14

Add User API

Add file pages/api/users/index.tsx

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
import { NextApiRequest, NextApiResponse } from "next";

import prisma from '../../../libs/prismadb';

export default async function handler(
req: NextApiRequest,
res: NextApiResponse
) {
if( req.method !=='GET'){
return res.status(405).end();
}

try{
const users = await prisma.user.findMany({
orderBy:{
createdAt:'desc'
}
});

return res.status(200).json(users);
} catch(error){
console.log(error);
return res.status(400).end();
}
}
More...

Built Twitter Like App - 13

Add Login/Logout Logic

modify file components/layout/Sidebar.tsx

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
...
import { signOut } from 'next-auth/react';
import useCurrentUser from '../../hooks/useCurrentUser';
...
const Sidebar = () => {
const { data: currentUser } = useCurrentUser();
...
return(
<div className='col-span-1 h-full pr-4 pr-6'>
<div className='flex flex-col items-end'>
<div className='space-y-2 lg:w-[230px]'>
<SidebarLog />
{items.map((item) =>(
<SidebarItem
key={item.href}
href={item.href}
label={item.label}
icon={item.icon}
/>
))}
{ currentUser &&
<SidebarItem onClick={() => signOut()} icon={ BiLogOut } label='Logout' href='/' />
}
<SiderbarTweetButton />
</div>
</div>
</div>
);
...
More...

Built Twitter Like App - 12

Install toast

1
npm install react-hot-toast

Add Register Logic in Modal

modify file pages/_app.tsx

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
import type { AppProps } from 'next/app';
import { Toaster } from 'react-hot-toast';
import { SessionProvider } from 'next-auth/react';

import Layout from '../components/Layout'
import LoginModal from '../components/modals/LoginModal'
import RegisterModal from '../components/modals/RegisterModal'
import '../styles/globals.css';

export default function App({ Component, pageProps }: AppProps) {
return (
<SessionProvider session={ pageProps.session }>
<Toaster />
<RegisterModal />
<LoginModal />
<Layout>
<Component {...pageProps} />
</Layout>
</SessionProvider>
)
}
More...

Built Twitter Like App - 11

Install SWR & axios

1
2
npm install swr
npm install axios

Add Hooks for CurrentUser

Add file libs/fetcher.ts

1
2
3
4
5
import axios from "axios";

const fetcher = (url: string) => axios.get(url).then( (res) => res.data);

export default fetcher;

Add file hooks/useCurrentUser.ts

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
import useSWR from 'swr';

import fetcher from '../libs/fetcher';

const useCurrentUser = () => {
const { data, error, isLoading, mutate } = useSWR('/api/current', fetcher)

return {
data,
error,
isLoading,
mutate
}
};

export default useCurrentUser;

Built Twitter Like App - 10

Add Register Function Code

Add file pages/api/register.ts

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
import bcrypt, { compareSync } from 'bcrypt';

import prisma from '../../libs/prismadb';
import { NextApiRequest, NextApiResponse } from 'next';

export default async function handler(
req: NextApiRequest,
res: NextApiResponse
) {
if ( req.method !== 'POST'){
return res.status(405).end();
}

try{
const { email,username, name, password } = req.body

const hashedPassword = await bcrypt.hash(password, 12);

const user = await prisma.user.create({
data: {
email,
username,
name,
hashedPassword
}
});

return res.status(200).json(user)
} catch (error){
console.log(error);
return res.status(400).end();
}
}
More...

请我喝杯咖啡吧~

支付宝
微信