카카오톡 채팅 기능을 구현
Socket.io를 이용하여 실시간 양방향 통신 기능을 구현해보고 싶었습니다. 그러기에 가장 적합한 것이 채팅이라고 생각하였습니다. 그래서 가장 흔하게 사용하고 있는 카카오톡 채팅 기능을 구현해보자는 생각으로 이 프로젝트를 진행하였습니다.
Ⅰ) 기술 스택
Ⅱ) 프로젝트 사용해보기
Ⅲ) 주요 기능
Ⅳ) UI/UX
Ⅴ) 프로젝트 구현 기술
- TypeScript
- React V16
- React Router V5
- Redux
- Redux-saga
- Styled-components
- Socket.io-client
- Webpack
- Node.js - Express
- Sequelize
- Socket.io
- MySQL
-
http://kakaoclonechat.cafe24app.com/ 에서 서비스를 이용할 수 있습니다.
-
다음 ID로 로그인하여 서비스 이용이 가능합니다.
- ID : test01 또는 test02
- PW : 12345678
-
회원가입을 통해서도 서비스 이용이 가능합니다.
-
Session Storage를 사용하였기 때문에, 여러 창에서 로그인하여 서로 대화가 가능합니다. 또는 서로 다른 컴퓨터에서 접속하여 대화를 해보시기 바랍니다.
※ 프로젝트에 사용된 ERD & API는 다음 사이트에 정리하였습니다.
- ERD Cloud : https://www.erdcloud.com/d/dqzfcGwjsQFq8BiG7
- Gitbook: https://eastshine94.gitbook.io/kakaoclone/
-
프로필 창에서 상태메시지, 배경사진, 프로필 사진 등을 변경할 수 있습니다. 또한, 친구의 이름을 변경할 수 있습니다.
-
사용자들이 채팅방을 만들어 1:1 대화가 가능합니다. 채팅이 온 것을 사용자가 알 수 있도록, 읽지 않은 채팅 숫자를 화면에 표시해줍니다.
-
또한, 채팅방에서 상대방이 채팅을 읽었는지 확인할 수 있도록, 채팅 옆에 읽지 않은 참가자 수만큼 숫자가 표시되도록 하였습니다.
-
무한 스크롤을 이용하여, 채팅방에서 스크롤을 위로 올리면, 이전의 채팅이 나타나도록 하였습니다.
-
채팅방에서 스크롤을 기준치 이상 올릴 경우, 페이지 맨 아래로 내리는 버튼이 나오게 됩니다. 또한, 스크롤이 위로 올라간 상태에서 상대방이 채팅을 보내면, 채팅이 왔다는 것을 알려주는 창이 나오도록 하였습니다.
간단한 입력을 통해 회원가입이 가능합니다.
회원가입한 정보로 로그인할 수 있습니다.
- 해당 메뉴에서는 친구 목록 확인 및 친구 추가, 검색이 가능합니다.
- 친구는 이름순으로 정렬되어 있습니다.
- 친구를 더블 클릭 시 채팅방에 입장하게 됩니다.
친구 ID를 입력하여 친구를 추가할 수 있습니다. 만약 이미 친구라면 1:1 채팅 버튼이 나옵니다.
검색 창에 입력한 단어가 이름에 있는 친구들을 찾습니다.
- 해당 메뉴에서는 채팅방 목록 확인 및 검색 등이 가능합니다.
- 채팅방은 최근 수신한 채팅 날짜 순으로 정렬되어 있으며, 읽지 않은 채팅 수가 표시됩니다.
- 채팅방을 더블 클릭 시 채팅방에 입장하게 됩니다.
- 친구 목록에서 채팅 할 대상을 선택하여 대화할 수 있습니다.
- 검색을 통해 대화할 친구를 찾을 수도 있습니다.
검색 단어가 방 참가자 또는 방 이름에 있는 채팅방들을 나타냅니다.
- 채팅방에서 다른 사용자와 대화를 할 수 있습니다.
- 친구가 아닐 경우, 경고창이 뜨고 원하면 친구 추가를 할 수 있습니다.
- 스크롤이 위로 올라가 있을 때, 상대방이 메시지를 보내면 알려줍니다.
친구 메뉴, 채팅방 등에서 사진을 클릭 시 프로필 창이 등장합니다. 해당 창에서 사용자 정보를 변경할 수 있습니다.
- 나의 프로필
- 친구 프로필
친구의 경우, 이름 변경만 가능합니다.
CRA(create-react-app)를 통해 프로젝트를 진행하지 않고, Webpack을 이용하여 직접 개발환경을 설정하였습니다. 이를 통해 Customize하게 개발 환경을 구성할 수 있습니다.
mode: process.env.NODE_ENV,
entry: "./src/index.tsx",
resolve: {
extensions: [".ts", ".tsx", '.js'],
plugins: [new TsconfigPathsPlugin({ configFile: "./tsconfig.json" })]
},
module: {
rules: [
{
test: /\.tsx?$/,
loader: "babel-loader",
exclude: /node_modules/
},
{
test: /\.tsx?$/,
loader: 'ts-loader'
}
]
},
......
Styled-Components를 사용하여 CSS-in-JS를 구현하였습니다. 이를 통해 CSS 모델을 문서 레벨이 아니라 컴포넌트 레벨로 추상화하여, 스타일 시트를 더 이상 유지 보수할 필요가 없도록 하였습니다.
글로벌적으로 설정해야 하는 스타일(ex) body, div, input 등)이나, 재사용할 수 있는 스타일은 styles 폴더에 따로 분리하였습니다. 그 후 다른 컴포넌트에서 import하여 사용하였습니다.
const GlobalStyle = createGlobalStyle`
* {
box-sizing: border-box;
}
body {
width: 100%;
height: 100%;
}
body, div, ul, li, dl, dd, dt, ol, h1, h2, h3, h4, h5, h6, input, fieldset, legend, p, select, table, th, td, tr, textarea, button, form, figure, figcaption {
padding: 0;
margin: 0;
}
......
채팅방에 스크롤 페이징 기술을 접목하였습니다. 이를 통해 처음부터 모든 채팅 내용을 서버에서 가져오는 것이 아니라, 사용자가 원할 때만 이전 채팅 내용을 가져오기 때문에 리소스 낭비를 막을 수 있습니다.
- 채팅의 실시간 양방향 통신을 위하여 Socket.io를 사용하였습니다.
- 채팅의 전반적인 기능(메시지 송수신, 알림 등)에 사용하여 실시간으로 채팅이 이루어지도록 구현하였습니다.
ORM(Object Relational Mapping)인 Sequelize를 이용하여 객체와 관계형 데이터베이스의 데이터를 매핑하였습니다. 이를 통해 SQL Query가 아닌 직관적인 코드로 데이터를 조작할 수 있기 때문에, 더 수월하게 개발이 가능합니다.
Object(객체)
class Friend extends Model {
public id!: number;
public my_id!: number;
public friend_id!: number;
public friend_name!: string;
public readonly User?: User;
public readonly createdAt!: Date;
public readonly updatedAt!: Date;
}
Object와 RDB를 Mapping
Friend.init({
id: {
type: DataTypes.INTEGER.UNSIGNED,
autoIncrement: true,
primaryKey: true,
allowNull: false,
},
my_id: {
type: DataTypes.INTEGER.UNSIGNED,
allowNull: false,
},
friend_id: {
type: DataTypes.INTEGER.UNSIGNED,
allowNull: false,
},
friend_name: {
type: new DataTypes.STRING(20),
allowNull: false,
},
createdAt: {
type: DataTypes.DATE,
allowNull: false,
},
updatedAt: {
type: DataTypes.DATE,
allowNull: false,
},
}, {
sequelize,
tableName: 'friend',
engine: 'InnoDB',
charset: 'utf8',
freezeTableName: true,
indexes: [
{
unique: true,
fields: ['my_id', 'friend_id']
}
]
});
객체를 이용한 CRUD
Friend.findAll({
attributes: ["friend_id", "friend_name"],
where: {my_id: 1},
});
Friend.create({
my_id: 1,
friend_id: 2,
friend_name: "김갑수"
});
Friend.update({
friend_name: "홍길동"
}, {
where: {my_id, friend_id}
});