스팀 앱 개발기 #40 - UTC 시간을 로컬 시간으로 바꾸는 코드 만들기
시작하며...
스팀 API에서 주는 데이터 중에는 시간도 있습니다. 그것은 UTC 시간입니다. 앱에서는 그것을 현재 지역 시간으로 바꿔주어야 합니다. 이번 포스트에서는 사전 작업으로서 그것을 코드로 구현하고자 합니다.
UTC 시간을 로컬 시간으로 바꾸는 확장 함수 구현
"yyyy-MM-dd'T'HH:mm:ss" 또는 "yyyy-MM-dd HH:mm:ss" 형식의 UTC 시간을 갖는 문자열을 현재 지역 시간을 나타내는 문자열로 바꾸는 메소드를 String 클래스의 확장 함수로 구현해 보았습니다. 코드는 아래와 같습니다.
// Precondition: The date format of this string is "yyyy-MM-dd'T'HH:mm:ss" or "yyyy-MM-dd HH:mm:ss"
fun String.fromUtcTimeToLocalTime(): String {
try {
val sdf = SimpleDateFormat("yyyy-MM-dd HH:mm:ss")
val date = sdf.parse(this.replace("T", " ").trim())
val longUtcTime = date.getTime()
val offset: Int = TimeZone.getDefault().getOffset(longUtcTime)
val longLocalTime = longUtcTime + offset
val dateLocalTime = Date().apply {
setTime(longLocalTime)
}
return sdf.format(dateLocalTime)
}
catch (e: ParseException) {
e.printStackTrace()
return ""
}
}
테스트 코드 만들기
위에 보여드린 코드가 잘 작동하는지 테스트를 해보았습니다. 테스트 케이스는 3가지를 만들었습니다.
- 포맷이 "yyyy-MM-dd'T'HH:mm:ss"인 UTC 시간 문자열을 현재 지역 시간으로 변환
- 포맷이 "yyyy-MM-dd HH:mm:ss"인 UTC 시간 문자열을 현재 지역 시간으로 변환
- 시간이 아닌 문자열을 현재 지역 시간으로 변환
코드는 다음과 같습니다.
class StringExtTest {
// Test case 1: Trying to convert UTC time of which the format is "yyyy-MM-dd'T'HH:mm:ss"
@Test
fun fromUtcTimeToLocalTime_case1() {
val sampleUtcTime = "2022-10-04T00:00:00"
val convertedLocalTime = sampleUtcTime.fromUtcTimeToLocalTime()
assertEquals("2022-10-04 09:00:00", convertedLocalTime)
}
// Test case 2: Trying to convert UTC time of which the format is "yyyy-MM-dd HH:mm:ss"
@Test
fun fromUtcTimeToLocalTime_case2() {
val sampleUtcTime = "2022-10-04 10:00:00"
val convertedLocalTime = sampleUtcTime.fromUtcTimeToLocalTime()
assertEquals("2022-10-04 19:00:00", convertedLocalTime)
}
// Test case 3: Trying to convert the string that is not a time
@Test
fun fromUtcTimeToLocalTime_case3() {
val sampleUtcTime = "abcdefg"
val convertedLocalTime = sampleUtcTime.fromUtcTimeToLocalTime()
assertTrue(convertedLocalTime.isEmpty())
}
}
테스트 결과, 모두 성공임을 확인했습니다.
GitHub Commit
참고 자료
마치며...
UTC 시간을 지역 시간으로 바꾸는 방법은 예전에도 알아봤었는데, 잘 되지 않았던 적이 있었습니다. 오늘에야 해결이 되었네요. 마친 후 느끼는 뿌듯함은 개발을 계속 할 수 있게 해주는 원동력인 거 같습니다. 지치지 않고 50대에도, 60대에도 개발을 계속 할 수 있기를 바랍니다.
지난 스팀 앱 개발기
- #39 - condenser_api.get_dynamic_global_properties API 실행에 필요한 데이터 클래스 추가
- #38 - 지갑 화면 구현: 예금 보여주기 추가
- #37 - 지갑 화면 구현: WalletViewModel과 WalletFragment를 바인딩
- #36 - 유닛 테스트로 구현한 WalletViewModelTest 클래스에서 공통 부분 추출하여 CommonPartOfViewModelTest 클래스 만들기
- #35 - 유닛 테스트용 WalletViewModelTest 클래스 구현 후 보이는 문제점
- #34 - WalletViewModel 클래스에 스팀 지갑 정보 읽는 코드 작성
- #33 - 추가 리팩토링: 데이터 모듈의 데이터 클래스의 필드들에 nullable 적용
- #32 - 리팩토링: 데이터 클래스 이름 변경
- #31 - 도메인 모듈에서 정의하는 데이터 클래스들이 DTO 패턴을 따르는 것이 맞는가?
- #30 - 진도를 더 나가기 전에 리팩토링 고려 중
- #29 - 뷰모델, Repository 패턴 그리고 Use Case 패턴의 관계
- #28 - Use Case 패턴 적용
- #27 - Repository 패턴 적용
- #26 - condenser_api.get_accounts API로 받은 계정 자료를 SteemitWalletDTO 타입으로 변환하기
- #25 - SteemClient, SteemService 잘 작동하는지 유닛 테스트
- #24 - API 실행을 담당할 SteemService 인터페이스 정의
- #23 - condenser_api.get_accounts API 연동에 필요한 데이터 클래스 정의
- #22 - API 연동에 필요한 라이브러리 추가
- #21 - 사용자 지갑 정보를 담을 SteemitWallet 클래스 정의
- #20 - 클린 아키텍처를 위한 모듈 구성 (2)
- #19 - 클린 아키텍처를 위한 모듈 구성 (1)
- #18 - VEST로부터 STEEM POWER를 계산하는 방법
- #17 - VEST를 STEEM POWER로 변환하기 위해 필요한 get_dynamic_global_properties API
- #16 - 지갑 내용을 읽기 위해 필요한 API
- #15 - 지갑 서브화면(WalletFragment) 레이아웃 구성해 보기
- #14 - 지갑 서브화면(WalletFragment)에서 계정 인식하기
- #13 - 프로파일 서브화면(ProfileFragment)에서 계정 인식하기
- #12 - 태그 서브화면에서 태그 인식하기
- #11 - 검색 레이아웃을 메인 화면에 적용
- #10 - 태그 및 계정 검색 레이아웃 만들기
- #9 - BaseActivity 클래스 정의 그리고 MainActivity 클래스에 적용
- #8 - BaseFragment 클래스 정의 그리고 기존 프래그먼트들에 적용
- #7 - ProfileFragment 그리고 WalletFragment에 데이터 바인딩 적용
- #6 - 태그별 검색 내용을 보여줄 TagsFragment에 데이터 바인딩 적용
- #5 - GitHub에 소스 올리기
- #4 - 하단 내비게이션의 탭 관련 클래스 이름 수정
- #3 - 하단 내비게이션 바의 아이콘, 텍스트 수정
- #2 - 프로젝트 생성
- #1 - 시작하며...
[광고] STEEM 개발자 커뮤니티에 참여 하시면, 다양한 혜택을 받을 수 있습니다.
Upvoted! Thank you for supporting witness @jswit.