Skip to main content

Command Palette

Search for a command to run...

[React] 클라이언트에서 이미지를 압축해보자

Published
2 min read
[React] 클라이언트에서 이미지를 압축해보자

우리는 서버 비용을 직접 부담하고 있기 때문에 항상 최적화와 비용 절감 방법을 고민하게 된다. 현재 자유게시판이 활성화가 많이 되지는 않았지만 사용자들이 업로드하는 이미지가 평균 5MB 이상일 경우를 대비해서 클라이언트에서 이미지 압축을 진행했다.

웹에서는 크게 두 가지 방법이 있었는데

  • Canvas API를 직접 사용하는 방법

  • browser-image-compression 라이브러리를 활용하는 방법

이 있다.

여러 장점을 고려해 browser-image-compression을 선택했다. Web Worker를 지원해 UI가 막히지 않고 간단한 옵션으로 최대 용량과 크기를 조절할 수 있어 대규모 서비스에도 적합하다고 판단..!(아직 대규모 서비스는 아니지만.. 혹시 몰라 😬)

나는 현재 글 작성시 tiptap 에디터를 사용하고 있어서 이미지 업로드 부분에 적용해 주기로 했다.

설치

//yarn 
yarn add browser-image-compression
//npm 
npm install browser-image-compression

이미지 업로드 코드

import { forwardRef, useCallback, useEffect, useRef } from 'react'
import axios from 'axios'
import { authApi } from '../../../services/api/axios'
import EditorToolbar from './EditorToolbar'
import ImageBubbleMenu from './ImageBubbleMenu'
import useEditorInstance from '../../../hooks/writePost/useEditorInstance'
import { EditorContent } from '@tiptap/react'
import ToastService from '../../../services/toast/ToastService'

const MAX_FILE_SIZE = 5 * 1024 * 1024

const WriteEditor = forwardRef(({ value, initialContent, onChange, onAddFileId }, ref) => {

  const handleFileChange = async (e) => {
    const file = e.target.files[0]
    if (!file) return
    try {
      const url = await uploadImage(file)
      editor.chain().focus().setImage({ src: url }).run()
    } catch (error) {
      ToastService.error('이미지 업로드 실패. 다시 시도해주세요.')
    }
  }

  return (
    <div>
      <input id='file-input' type='file' accept='image/*' onChange={handleFileChange} hidden />
    </div>
  )
})
WriteEditor.displayName = 'WriteEditor'
export default WriteEditor

수정

import { forwardRef, useCallback, useEffect, useRef } from 'react'
import axios from 'axios'
import { authApi } from '../../../services/api/axios'
import EditorToolbar from './EditorToolbar'
import ImageBubbleMenu from './ImageBubbleMenu'
import useEditorInstance from '../../../hooks/writePost/useEditorInstance'
import { EditorContent } from '@tiptap/react'
import ToastService from '../../../services/toast/ToastService'
import imageCompression from 'browser-image-compression'

const MAX_FILE_SIZE = 5 * 1024 * 1024

const WriteEditor = forwardRef(({ value, initialContent, onChange, onAddFileId }, ref) => {

  const handleFileChange = async (e) => {
  const file = e.target.files[0]
  if (!file) return

  try {
    // 브라우저에서 압축
    const options = { maxSizeMB: 1, maxWidthOrHeight: 800, useWebWorker: true }
    const compressedFile = await imageCompression(file, options)

    // 업로드
    const url = await uploadImage(compressedFile)

    // 에디터에 삽입
    editor.chain().focus().setImage({ src: url }).run()
  } catch (error) {
    ToastService.error('이미지 업로드 실패. 다시 시도해주세요.')
  }
}

이런식으로 적용을 해주고 직접 콘솔로 확인해본결과

90% 정도 압축된 것을 직접 확인했다.. 굉장한걸 👍🏻

이미지 압축은 생각보다 굉장히 쉽고 서버 비용을 고려할 때는 필수적인 최적화 과정임을 다시 한번 느꼈다.. !

More from this blog

[회고] 풀스택 개발자로 (곧) 1년...

정말 정말 오랜만에 글을 작성한다 🤭 나는 현재 개발자로 취업한 지 거의 1년이 다 되어간다. 이제 10개월임 ! 처음에는 무조건 프론트엔드 개발자로 취업할거임!! 풀스택 NO ! 라는 마음으로 준비했던 취업.. 하지만 현실은 풀스택으로 취업하게 되었다. 취업을 하고 나니 여러가지 생각과 감정이 들었고 맨날 회고 적어야지 ~ 적어야지 ~ 했는데 10개

May 29, 20263 min read

[책 추천] 프론트엔드 개발자라면 반드시 알아야 할 '웹 접근성' 이야기

내가 생각하는 프론트엔드 개발자는 단순히 보이는 화면만 구현하는 것을 넘어서 누구나 접근할 수 있는 웹을 만드는 데 중요한 역할을 해야 한다고 생각한다. 실무에 바로 적용하는 웹 접근성 가이드북 이번에 이 책을 받고 프엔들이 읽기에 너무 좋다고 생각해서 블로그에 추천 글까지 적게 되었다 ! 목차 Chatper1. 쉽게 이해하는 접근성 Chapter2. 웹 접근성의 기초 Chapter3. HTML 태그, 의미 있게 사용하기 Chpater4...

Jul 27, 20255 min read
[책 추천] 프론트엔드 개발자라면 반드시 알아야 할 '웹 접근성' 이야기

[Next.js] 나만의 학습 블로그 만들기#3-다국어 지원 (Feat.next-i18next)

오늘은 다국어 지원 가능한 기능 구현 과정을 적어보려고 한다. 원래는 댓글기능보다 먼저 구현하려고 했는데 ,, 갑자기 댓글기능 알아보다가 재밌어서 먼저 끝내버렸다,,,🙄 시작하기 전에 정말 어려웠다 ㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋ 어려움이 많아서 기능 적용하는 데 시간이 꽤 걸렸다 ,,, 다국어 지원 기능 나는 영어랑 일본어를 추가해줬다 ! (해외취업도 관심 있기 때문에 나중에 이력서 낼 때 도움이 되지 않을까? ..🤭) 다국어 구현 방법 구현 방...

Jul 15, 20254 min read
[Next.js] 나만의 학습 블로그 만들기#3-다국어 지원 (Feat.next-i18next)

[Next.js] 나만의 학습 블로그 만들기#4 - 댓글 기능(Feat. Giscus)

댓글기능이 있으면 무조건 재밌을 거 같아서 넣어보려고 한다🤭 아무래도 개발에 관련된 학습 블로그라서 개발자들이 많을 것으로 예상했다 ! 그래서 깃허브 이슈 기반 댓글 시스템을 기반으로 기능을 구현해보려고 한다. 👉🏻 도움이 많이 된 블로그 ! 방법 Giscus | (https://giscus.app) utterances | (https://utteranc.es) 알아봤을 때는 이렇게 두 개가 유명하다고 한다. 그럼 각각의 특징과 장...

Jul 14, 20253 min read
[Next.js] 나만의 학습 블로그 만들기#4 - 댓글 기능(Feat. Giscus)
S

subin-dev-blog

28 posts

끊임없이 배우기 위해 노력합니다. 🌱