티스토리 뷰
결과 화면
개요
- 매우 간단한 댓글 컴포넌트를 만들어보았습니다.
- 백엔드 데이터로 작성일자를 받으면 위와 같은 ISO 형식으로 받게됩니다.
- 처음 피그마로 작업할 때는 2023-07-06 12:23 의 형식으로 편집하여 출력하려고 하였으나 핀터레스트, 트위터, 인스타그램 등 유명 SNS의 댓글 UI를 보면 좀 더 직관적으로 날짜를 표시해주는 것을 알게 되었습니다.
- dayjs나 moment와 같은 날짜를 다루기 쉽게 도와주는 라이브러리도 있지만 최대한 직접 기능을 구현해보고 싶어서 라이브러리는 이용하지 않았습니다.
구현
const elapsedTime = (date: Date) => {
const start = new Date(date);
const end = new Date();
const diff = (end.getTime() - start.getTime()) / 1000;
- date를 매개변수로 받는 함수입니다. date는 사용하는 곳에서의 작성일자 혹은 수정일자가 들어가겠죠!
- 대상이 되는 날짜를 start 변수에 담고 현재 날짜는 end 에 담아서 둘의 차이를 계산합니다. 1000으로 나누어주는 이유는 Date()값은 기본적으로 milliseconds로 나오기 때문입니다.
- 자바스크립트의 경우는 end - start를 하면 알아서 number로 형변환한 후 계산을 해주지만
- 타입스크립트는 start와 end의 타입을 Date로 알고 있기 때문에 뺄셈 연산을 하지 못한다고 오류를 뱉어냅니다.
- 따라서
end.getTime()
혹은Number(end)
혹은+end
로 형변환을 명시적으로 해주어야 합니다!
const times = [
{ name: "년", milliSeconds: 60 * 60 * 24 * 365 },
{ name: "개월", milliSeconds: 60 * 60 * 24 * 30 },
{ name: "일", milliSeconds: 60 * 60 * 24 },
{ name: "시간", milliSeconds: 60 * 60 },
{ name: "분", milliSeconds: 60 },
];
for (const value of times) {
const betweenTime = Math.floor(diff / value.milliSeconds);
if (betweenTime > 0) {
return `${betweenTime}${value.name} 전`;
}
}
return "방금 전";
- 대상 시간과 현재 시간의 차이를 구했다면 그 차이에 따라 적절한 문구를 출력해주어야 합니다.
- 년 ~ 분 의 순서로 탐색을 하며 둘의 차이가 나는 경우에 해당 name을 출력해주게 됩니다.
- 경과시간에 milliSeconds를 나눠서 해당 값보다 경과시간이 적을 시, 0보다 작은 소수점 값이 나오게 됩니다. ( 1분 / 1년 ) => 0.0000019...
- 정수가 나올 때까지(0보다 큰 값) 다음 단위로 넘어가서 다시 비교를 반복합니다. 모든 탐색이 끝났음에도 return 되지 않았다는 것은 두 날짜의 차이가 1분보다 작다는 의미이므로 "방금 전"이라는 문자를 반환하도록 하였습니다.
Intl API 이용해보기
Intl API의 주요 내용을 정리하기 위해 본문과 상관없는 내용을 정리해두었습니다. 위 내용과 이어지는 부분이 궁금하시다면 아래쪽에 Intl.RelativeTimeFormat 부분부터 봐주세요!
Intl API란?
- 웹 서비스를 여러가지 언어로 설계하고 구현하는 과정을 국제화(Internalization) 라고 합니다. 단순히 문구를 번역하는 것뿐만 아니라 동일한 데이터를 다른 형식으로 보여주어야 하기도 합니다.
- 예를 들어서 날짜의 경우 한국에서는 연, 월, 일, 요일 순서를 따르지만 영어권에서는 요일, 월, 일, 년 순서를 따릅니다. 시간을 표시할 때도 한국에서는 오전, 오후를 제일 앞에 명시하는 반면 영어에서는 보통 제일 뒤에 명시한다는 차이가 있습니다.
- 이러한 작업을 일일이 하려면 굉장한 노력이 필요하고 외부 라이브러리를 사용한다면 번들 사이즈에 영향이 갈 수밖에 없겠네요!
- 이러한 문제점을 해결하기 위해 자바스크립트에서는 Intl API를 제공하고 있습니다. 이 API는 대부분의 모던 브라우저에서 지원되며 Node.js에서도 사용이 가능합니다.
Intl API 사용법
- Intl API는 전역에서 접근이 가능한 Intl 객체를 통해 사용이 가능합니다.
- 모든 생성자는 공통적으로 2가지 인자를 받는데 첫 번째 인자는 locale 이라는 언어와 지역 정보를 표준화 시킨 코드이며 두 번째 인자는 추가적인 option 을 명시할 수 있습니다.
예시 코드
- 현재 날짜를 한국 이용자들을 위한 형식으로 변환하려고 한다면
DateTimeFormat()
생성자로 생성한 객체의format()
함수를 호출하면 됩니다.
const koDtf = new Intl.DateTimeFormat("ko", { dateStyle: "long" });
kodtf.format(new Date());
// '2023년 7월 7일'
주요 생성자
Intl.DateTimeFormat
- 시간이나 날짜를 쉽게 국제화할 수 있는 생성자입니다.
- 포매팅 옵션으로 dateStyle 과 timeStyle 이 자주 사용되며 full, long, medium, short 을 옵션으로 설정할 수 있습니다.
- locale 속성에 따라 다른 형식으로 출력되는 것을 볼 수 있습니다!
Intl.NumberFormat
- 통화(currency), 백분율, 무게, 길이, 속도, 온도 등 단위가 있는 숫자 데이터를 다룰 때 유용합니다.
- style 옵션을 통해 다양한 스타일을 지원하는데 예를 들어 백분율 데이터를 다룰 때는 style: 'percent' 로 설정하고 format() 함수에 소수를 넘겨주는 형식으로 사용이 가능합니다.
new Intl.NumberFormat('ko', { style: 'percent' }).format(0.7)
// '70%'
- 통화 데이터를 다룰 때는 style: 'currency' 로 설정을 하고 옵션으로 통화 코드를 넘겨줍니다.
new Intl.NumberFormat('ko', { style: 'currency', currency: 'USD' }).format(40.56)
// 'US$40.56'
- 그 외에 다른 단위를 사용할 때는 style: 'unit' 으로 주고 옵션으로 단위 코드를 넘겨주는 방식으로 사용할 수 있습니다.
new Intl.NumberFormat('ko', { style: 'unit', unit: 'kilogram' }).format(50)
// '50kg'
new Intl.NumberFormat('ko', { style: 'unit', unit: 'pound' }).format(110)
// '110lb'
Intl.PluralRules
- 영어에서는 한국어와 다르게 명사의 단복수 개념이 매우 중요합니다.
- 예를 들어서 한국에서는 한 사람, 두 사람 이라고 표현한다면 영어에서는 One person, Two people 로 명확하게 구분을 합니다. 또한 숫자 0은 복수로 취급이 된다는 특징도 있죠!
- 영어와 같이 단복수를 엄격하게 구분하는 언어로 서비스를 한다면 매우 유용하게 사용할 수 있을 것 같습니다.
- PluralRules() 생성자로 만든 객체의 select() 함수에 숫자를 넘기면 'one' 또는 'other'를 반환하는데 이 결과값을 가지고 단수와 복수를 구분하게 됩니다.
let pr = new Intl.PluralRules("en")
pr.select(0)
'other'
pr.select(1)
'one'
pr.select(2)
'other'
pr.select(3)
'other'
- 또한 type option 을 기본값 cardinal 에서 ordinal로 변경을 하면 1st, 2nd, 3rd, 4th 와 같은 서수의 어미 처리를 할 수 있습니다.
let pr = new Intl.PluralRules("en", { type: 'ordinal' })
pr.select(0)
'other'
pr.select(1)
'one'
pr.select(2)
'two'
pr.select(3)
'few'
pr.select(4)
'other'
- 'one'이 반환되면 어미로 'st'를 사용하고(1st) 'two'가 반환되면 'nd'(2nd), 'few' 가 반환되면 'rd'(3rd), 그 외 'other'이 반환되면 'th'(4th)를 어미로 사용하게 됩니다.
Intl.RelativeTimeFormat
- 현재 시간 기준으로 얼마나 시간이 지났는지, 혹은 얼마나 걸릴 것인지를 상대적으로 표현해줍니다.
- numeric 옵션을 auto 혹은 always 로 설정할 수 있는데 auto인 경우는 숫자없이 표기할 수 있는 부분은 최대한 문자를 사용해서 표현해줍니다.
numeric: 'auto'
numeric: 'always'
구현
const formatter = new Intl.RelativeTimeFormat("ko", {
numeric: "always",
});
const times: TimesProps[] = [
{ name: "year", milliSeconds: 60 * 60 * 24 * 365 },
{ name: "month", milliSeconds: 60 * 60 * 24 * 30 },
{ name: "day", milliSeconds: 60 * 60 * 24 },
{ name: "hour", milliSeconds: 60 * 60 },
{ name: "minute", milliSeconds: 60 },
];
- Intl.RelativeTimeFormat() 을 생성해주고 times의 name값들을 API가 지원하는 형식으로 바꾸어줍니다. 해당 형식은 생성자를 타고 들어가서 내부를 들어가보면 다음과 같이 알려줍니다.
- 그리고 format()을 호출하려고 하면 name의 타입이 맞지 않다며 오류를 내버립니다.
- format()은 두 번째 인자로 unit을 받기 때문에 그에 맞는 타입을 적용시켜주어야 합니다.
interface TimesProps {
name: Intl.RelativeTimeFormatUnit;
milliSeconds: number;
}
- 따라서 위와 같이 인터페이스를 선언하고 times에 배열로 적용시켜주면 됩니다!
전체 코드
interface TimesProps {
name: Intl.RelativeTimeFormatUnit;
milliSeconds: number;
}
const elapsedTime = (date: Date) => {
const start = new Date(date);
const end = new Date();
const diff = (end.getTime() - start.getTime()) / 1000;
const formatter = new Intl.RelativeTimeFormat("ko", {
numeric: "always",
});
const times: TimesProps[] = [
{ name: "year", milliSeconds: 60 * 60 * 24 * 365 },
{ name: "month", milliSeconds: 60 * 60 * 24 * 30 },
{ name: "day", milliSeconds: 60 * 60 * 24 },
{ name: "hour", milliSeconds: 60 * 60 },
{ name: "minute", milliSeconds: 60 },
];
for (const value of times) {
const betweenTime = Math.floor(diff / value.milliSeconds);
if (betweenTime > 0) {
return formatter.format(-betweenTime, value.name);
}
}
return "방금 전";
};
export default elapsedTime;
참고 자료
반응형
'개발냥이 > 타입스크립트(Typescript)' 카테고리의 다른 글
[리액트+타입스크립트] 이미지 업로드 구현 & 이미지와 콘텐츠 하나의 객체로 관리하기 (0) | 2023.07.12 |
---|---|
[Next.js + typescript] 페이지네이션(Pagination) 구현해보기 (0) | 2023.07.10 |
[리액트+타입스크립트] 커스텀 셀렉트 박스(custom select box) 만들어보기)_리팩토링 추가(2023-07-29) (0) | 2023.07.06 |
[리액트+타입스크립트] 커스텀 훅(Custom hook) 만들어보기(useInput) (0) | 2023.06.22 |
[타입스크립트] 인터페이스(Interface) (0) | 2023.05.31 |
댓글
반응형
공지사항
최근에 올라온 글
최근에 달린 댓글
- Total
- Today
- Yesterday
링크
TAG
- Comparator
- java
- Spring
- 스프링부트
- 자바
- 프로그래머스
- SQL
- CS
- 이분탐색
- 리액트
- SQLD
- 자바스크립트
- Algorithm
- 형변환
- 자바트리
- 자바bfs
- 알고리즘
- 백준
- JPA
- DP
- 정렬
- Nest
- 해시맵
- dfs
- 자바dp
- Queue
- 타입스크립트
- BFS
- JavaScript
- 스프링
일 | 월 | 화 | 수 | 목 | 금 | 토 |
---|---|---|---|---|---|---|
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 |
글 보관함