feat: 다국어 지원 및 다중 통화 환율 변환 기능 확대

- ExchangeRateService에 JPY, CNY 환율 지원 추가
- 구독 서비스별 다국어 표시 이름 지원
- 분석 화면 차트 및 UI/UX 개선
- 설정 화면 전면 리팩토링
- SMS 스캔 기능 사용성 개선
- 전체 앱 다국어 번역 확대

🤖 Generated with [Claude Code](https://claude.ai/code)

Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
JiWoong Sul
2025-07-16 17:34:32 +09:00
parent 4d1c0f5dab
commit 0f0b02bf08
55 changed files with 4100 additions and 1197 deletions

View File

@@ -17,6 +17,8 @@ class ExchangeRateService {
// 캐싱된 환율 정보
double? _usdToKrwRate;
double? _usdToJpyRate;
double? _usdToCnyRate;
DateTime? _lastUpdated;
// API 요청 URL (ExchangeRate-API 사용)
@@ -24,18 +26,20 @@ class ExchangeRateService {
// 기본 환율 상수
static const double DEFAULT_USD_TO_KRW_RATE = 1350.0;
static const double DEFAULT_USD_TO_JPY_RATE = 150.0;
static const double DEFAULT_USD_TO_CNY_RATE = 7.2;
// 캐싱된 환율 반환 (동기적)
double? get cachedUsdToKrwRate => _usdToKrwRate;
/// 현재 USD to KRW 환율 정보를 가져옵니다.
/// 최근 6시간 이내 조회했던 정보가 있다면 캐싱된 정보를 반환합니다.
Future<double?> getUsdToKrwRate() async {
// 캐싱된 데이터 있고 6시간 이내면 캐싱된 데이터 반환
if (_usdToKrwRate != null && _lastUpdated != null) {
/// 모든 환율 정보를 한 번에 가져옵니다.
/// 최근 6시간 이내 조회했던 정보가 있다면 캐싱된 정보를 사용합니다.
Future<void> _fetchAllRatesIfNeeded() async {
// 캐싱된 데이터 있고 6시간 이내면 스킵
if (_lastUpdated != null) {
final difference = DateTime.now().difference(_lastUpdated!);
if (difference.inHours < 6) {
return _usdToKrwRate;
return;
}
}
@@ -45,19 +49,22 @@ class ExchangeRateService {
if (response.statusCode == 200) {
final data = json.decode(response.body);
_usdToKrwRate = data['rates']['KRW'].toDouble();
_usdToKrwRate = data['rates']['KRW']?.toDouble();
_usdToJpyRate = data['rates']['JPY']?.toDouble();
_usdToCnyRate = data['rates']['CNY']?.toDouble();
_lastUpdated = DateTime.now();
return _usdToKrwRate;
} else {
// 실패 시 캐싱된 값이라도 반환
return _usdToKrwRate;
}
} catch (e) {
// 오류 발생 시 캐싱된 값이라도 반환
return _usdToKrwRate;
// 오류 발생 시 기본값 사용
}
}
/// 현재 USD to KRW 환율 정보를 가져옵니다.
Future<double?> getUsdToKrwRate() async {
await _fetchAllRatesIfNeeded();
return _usdToKrwRate;
}
/// USD 금액을 KRW로 변환합니다.
Future<double?> convertUsdToKrw(double usdAmount) async {
final rate = await getUsdToKrwRate();
@@ -67,6 +74,48 @@ class ExchangeRateService {
return null;
}
/// USD 금액을 지정된 통화로 변환합니다.
Future<double?> convertUsdToTarget(double usdAmount, String targetCurrency) async {
await _fetchAllRatesIfNeeded();
switch (targetCurrency) {
case 'KRW':
final rate = _usdToKrwRate ?? DEFAULT_USD_TO_KRW_RATE;
return usdAmount * rate;
case 'JPY':
final rate = _usdToJpyRate ?? DEFAULT_USD_TO_JPY_RATE;
return usdAmount * rate;
case 'CNY':
final rate = _usdToCnyRate ?? DEFAULT_USD_TO_CNY_RATE;
return usdAmount * rate;
case 'USD':
return usdAmount;
default:
return null;
}
}
/// 지정된 통화를 USD로 변환합니다.
Future<double?> convertTargetToUsd(double amount, String sourceCurrency) async {
await _fetchAllRatesIfNeeded();
switch (sourceCurrency) {
case 'KRW':
final rate = _usdToKrwRate ?? DEFAULT_USD_TO_KRW_RATE;
return amount / rate;
case 'JPY':
final rate = _usdToJpyRate ?? DEFAULT_USD_TO_JPY_RATE;
return amount / rate;
case 'CNY':
final rate = _usdToCnyRate ?? DEFAULT_USD_TO_CNY_RATE;
return amount / rate;
case 'USD':
return amount;
default:
return null;
}
}
/// 현재 환율 정보를 포맷팅하여 텍스트로 반환합니다.
Future<String> getFormattedExchangeRateInfo() async {
final rate = await getUsdToKrwRate();
@@ -76,11 +125,42 @@ class ExchangeRateService {
symbol: '',
decimalDigits: 0,
).format(rate);
return '오늘 기준 환율 : $formattedRate';
return formattedRate;
}
return '';
}
/// 언어별 환율 정보를 포맷팅하여 반환합니다.
Future<String> getFormattedExchangeRateInfoForLocale(String locale) async {
await _fetchAllRatesIfNeeded();
switch (locale) {
case 'ko':
final rate = _usdToKrwRate ?? DEFAULT_USD_TO_KRW_RATE;
return NumberFormat.currency(
locale: 'ko_KR',
symbol: '',
decimalDigits: 0,
).format(rate);
case 'ja':
final rate = _usdToJpyRate ?? DEFAULT_USD_TO_JPY_RATE;
return NumberFormat.currency(
locale: 'ja_JP',
symbol: '¥',
decimalDigits: 0,
).format(rate);
case 'zh':
final rate = _usdToCnyRate ?? DEFAULT_USD_TO_CNY_RATE;
return NumberFormat.currency(
locale: 'zh_CN',
symbol: '¥',
decimalDigits: 2,
).format(rate);
default:
return '';
}
}
/// USD 금액을 KRW로 변환하여 포맷팅된 문자열로 반환합니다.
Future<String> getFormattedKrwAmount(double usdAmount) async {
final krwAmount = await convertUsdToKrw(usdAmount);
@@ -94,4 +174,46 @@ class ExchangeRateService {
}
return '';
}
/// USD 금액을 지정된 언어의 기본 통화로 변환하여 포맷팅된 문자열로 반환합니다.
Future<String> getFormattedAmountForLocale(double usdAmount, String locale) async {
String targetCurrency;
String localeCode;
String symbol;
int decimalDigits;
switch (locale) {
case 'ko':
targetCurrency = 'KRW';
localeCode = 'ko_KR';
symbol = '';
decimalDigits = 0;
break;
case 'ja':
targetCurrency = 'JPY';
localeCode = 'ja_JP';
symbol = '¥';
decimalDigits = 0;
break;
case 'zh':
targetCurrency = 'CNY';
localeCode = 'zh_CN';
symbol = '¥';
decimalDigits = 2;
break;
default:
return '\$$usdAmount';
}
final convertedAmount = await convertUsdToTarget(usdAmount, targetCurrency);
if (convertedAmount != null) {
final formattedAmount = NumberFormat.currency(
locale: localeCode,
symbol: symbol,
decimalDigits: decimalDigits,
).format(convertedAmount);
return '($formattedAmount)';
}
return '';
}
}