Initial commit: SubManager Flutter App

주요 구현 완료 기능:
- 구독 관리 (추가/편집/삭제/카테고리 분류)
- 이벤트 할인 시스템 (기본값 자동 설정)
- SMS 자동 스캔 및 구독 정보 추출
- 알림 시스템 (타임존 처리 안정화)
- 환율 변환 지원 (KRW/USD)
- 반응형 UI 및 애니메이션
- 다국어 지원 (한국어/영어)

버그 수정:
- NotificationService tz.local 초기화 오류 해결
- MainScreenSummaryCard 레이아웃 오버플로우 수정
- 구독 추가 시 LateInitializationError 완전 해결

🤖 Generated with Claude Code

Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
JiWoong Sul
2025-07-09 14:29:53 +09:00
commit 8619e96739
177 changed files with 23085 additions and 0 deletions

342
lib/temp/test_sms_data.dart Normal file
View File

@@ -0,0 +1,342 @@
// 테스트용 SMS 데이터
// 실제 SMS를 스캔하는 대신 이 테스트 데이터를 사용합니다.
class TestSmsData {
static List<Map<String, dynamic>> getTestData() {
// 현재 날짜 기준으로 미래 날짜 계산
final now = DateTime.now();
final nextMonth = DateTime(now.year, now.month + 1, now.day);
final prevMonth = DateTime(now.year, now.month - 1, now.day);
final String formattedNextMonth =
'${nextMonth.year}-${nextMonth.month.toString().padLeft(2, '0')}-${nextMonth.day.toString().padLeft(2, '0')}';
final String formattedPrevMonth =
'${prevMonth.year}-${prevMonth.month.toString().padLeft(2, '0')}-${prevMonth.day.toString().padLeft(2, '0')}';
final String formattedNow =
'${now.year}-${now.month.toString().padLeft(2, '0')}-${now.day.toString().padLeft(2, '0')}';
// 1년 후 날짜 계산 (연간 구독용)
final nextYear = DateTime(now.year + 1, now.month, now.day);
final prevYear = DateTime(now.year - 1, now.month, now.day);
final String formattedNextYear =
'${nextYear.year}-${nextYear.month.toString().padLeft(2, '0')}-${nextYear.day.toString().padLeft(2, '0')}';
final String formattedPrevYear =
'${prevYear.year}-${prevYear.month.toString().padLeft(2, '0')}-${prevYear.day.toString().padLeft(2, '0')}';
// 기본 테스트 데이터 정의
final baseTestData = [
{
'serviceName': '넷플릭스',
'monthlyCost': 9900.0,
'billingCycle': '월간',
'nextBillingDate': formattedNextMonth,
'isRecurring': true,
'repeatCount': 5,
'sender': '15885000',
'messageDate': formattedNow,
'previousPaymentDate': formattedPrevMonth,
'message': '[넷플릭스] 정기 결제 완료: 9,900원이 결제되었습니다. 다음 결제일: ${nextMonth.day}'
},
{
'serviceName': '유튜브프리미엄',
'monthlyCost': 14900.0,
'billingCycle': '월간',
'nextBillingDate':
'${DateTime(now.year, now.month + 1, 20).year}-${DateTime(now.year, now.month + 1, 20).month.toString().padLeft(2, '0')}-20',
'isRecurring': true,
'repeatCount': 7,
'sender': '15882018',
'messageDate': formattedNow,
'previousPaymentDate':
'${DateTime(now.year, now.month - 1, 20).year}-${DateTime(now.year, now.month - 1, 20).month.toString().padLeft(2, '0')}-20',
'message': '[Google] 유튜브프리미엄 구독료 14,900원이 자동 결제되었습니다.'
},
{
'serviceName': '디즈니플러스',
'monthlyCost': 8900.0,
'billingCycle': '월간',
'nextBillingDate':
'${DateTime(now.year, now.month + 1, 10).year}-${DateTime(now.year, now.month + 1, 10).month.toString().padLeft(2, '0')}-10',
'isRecurring': true,
'repeatCount': 3,
'sender': '15771055',
'messageDate': formattedNow,
'previousPaymentDate':
'${DateTime(now.year, now.month - 1, 10).year}-${DateTime(now.year, now.month - 1, 10).month.toString().padLeft(2, '0')}-10',
'message': '[디즈니플러스] 8,900원 정기결제가 완료되었습니다.'
},
{
'serviceName': '애플 iCloud',
'monthlyCost': 2900.0,
'billingCycle': '월간',
'nextBillingDate':
'${DateTime(now.year, now.month + 1, 5).year}-${DateTime(now.year, now.month + 1, 5).month.toString().padLeft(2, '0')}-05',
'isRecurring': true,
'repeatCount': 12,
'sender': '0802011900',
'messageDate': formattedNow,
'previousPaymentDate':
'${DateTime(now.year, now.month - 1, 5).year}-${DateTime(now.year, now.month - 1, 5).month.toString().padLeft(2, '0')}-05',
'message': 'Apple: iCloud+ 50GB 월 구독(₩2,900)이 자동으로 갱신되었습니다.'
},
{
'serviceName': '멜론',
'monthlyCost': 10900.0,
'billingCycle': '월간',
'nextBillingDate':
'${DateTime(now.year, now.month + 1, 22).year}-${DateTime(now.year, now.month + 1, 22).month.toString().padLeft(2, '0')}-22',
'isRecurring': true,
'repeatCount': 4,
'sender': '16001950',
'messageDate': formattedNow,
'previousPaymentDate':
'${DateTime(now.year, now.month - 1, 22).year}-${DateTime(now.year, now.month - 1, 22).month.toString().padLeft(2, '0')}-22',
'message':
'[멜론] 스트리밍클럽 정기결제 10,900원 완료. 다음 결제일: ${DateTime(now.year, now.month + 1, 22).day}'
},
{
'serviceName': 'Microsoft 365',
'monthlyCost': 12800.0,
'billingCycle': '연간',
'nextBillingDate': formattedNextYear,
'isRecurring': true,
'repeatCount': 2,
'sender': '0801136532',
'messageDate': formattedNow,
'previousPaymentDate': formattedPrevYear,
'message': '[Microsoft] Microsoft 365 연간 구독료 12,800원이 결제되었습니다.'
},
{
'serviceName': '웨이브',
'monthlyCost': 7900.0,
'billingCycle': '월간',
'nextBillingDate':
'${DateTime(now.year, now.month + 1, 15).year}-${DateTime(now.year, now.month + 1, 15).month.toString().padLeft(2, '0')}-15',
'isRecurring': true,
'repeatCount': 2,
'sender': '1800-1234',
'messageDate': formattedNow,
'previousPaymentDate':
'${DateTime(now.year, now.month - 1, 15).year}-${DateTime(now.year, now.month - 1, 15).month.toString().padLeft(2, '0')}-15',
'message': '[웨이브] 구독료 7,900원이 정기결제 되었습니다. 감사합니다.'
},
// 달러 결제 서비스 추가
{
'serviceName': 'Netflix US',
'monthlyCost': 9.99,
'billingCycle': '월간',
'nextBillingDate':
'${DateTime(now.year, now.month + 1, 7).year}-${DateTime(now.year, now.month + 1, 7).month.toString().padLeft(2, '0')}-07',
'isRecurring': true,
'repeatCount': 6,
'sender': '334455',
'messageDate': formattedNow,
'previousPaymentDate':
'${DateTime(now.year, now.month - 1, 7).year}-${DateTime(now.year, now.month - 1, 7).month.toString().padLeft(2, '0')}-07',
'message':
'[Netflix US] Your subscription has been renewed. \$9.99 has been charged to your account. Next billing date: ${DateTime(now.year, now.month + 1, 7).day}'
},
{
'serviceName': 'Spotify Premium',
'monthlyCost': 10.99,
'billingCycle': '월간',
'nextBillingDate':
'${DateTime(now.year, now.month + 1, 12).year}-${DateTime(now.year, now.month + 1, 12).month.toString().padLeft(2, '0')}-12',
'isRecurring': true,
'repeatCount': 4,
'sender': '223344',
'messageDate': formattedNow,
'previousPaymentDate':
'${DateTime(now.year, now.month - 1, 12).year}-${DateTime(now.year, now.month - 1, 12).month.toString().padLeft(2, '0')}-12',
'message':
'[Spotify] Your premium subscription was automatically renewed. USD 10.99 charged to your card. Your next payment will be on ${DateTime(now.year, now.month + 1, 12).day}'
},
{
'serviceName': 'GitHub Pro',
'monthlyCost': 4.00,
'billingCycle': '월간',
'nextBillingDate':
'${DateTime(now.year, now.month + 1, 3).year}-${DateTime(now.year, now.month + 1, 3).month.toString().padLeft(2, '0')}-03',
'isRecurring': true,
'repeatCount': 8,
'sender': '112233',
'messageDate': formattedNow,
'previousPaymentDate':
'${DateTime(now.year, now.month - 1, 3).year}-${DateTime(now.year, now.month - 1, 3).month.toString().padLeft(2, '0')}-03',
'message':
'[GitHub] Your Pro plan has been renewed for \$4.00 USD. View your receipt at github.com/receipt. Next bill on ${DateTime(now.year, now.month + 1, 3).day}'
},
];
// 각 서비스별로 여러 개의 메시지 생성 (그룹화를 위해)
final List<Map<String, dynamic>> resultData = [];
// 각 기본 데이터를 복제하여 여러 개의 메시지 생성
for (final service in baseTestData) {
// 원본 메시지 추가
resultData.add(Map<String, dynamic>.from(service));
// 각 서비스에 대해 과거 메시지 추가 (2개 이상의 메시지가 있어야 반복 구독으로 인식)
for (int i = 2; i <= (service['repeatCount'] as int); i++) {
// 과거 결제일 계산
final pastMonth = DateTime(
now.year,
now.month - i + 1 > 0 ? now.month - i + 1 : now.month - i + 1 + 12,
service['billingCycle'] == '월간' ? now.day : now.day);
final formattedPastMonth =
'${pastMonth.year}-${pastMonth.month.toString().padLeft(2, '0')}-${pastMonth.day.toString().padLeft(2, '0')}';
// 과거 메시지 생성
final pastMessage = Map<String, dynamic>.from(service);
pastMessage['messageDate'] = formattedPastMonth;
pastMessage['nextBillingDate'] =
service['previousPaymentDate']; // 이전 메시지의 다음 결제일은 현재의 이전 결제일
// 과거 메시지의 이전 결제일 계산
final veryPastMonth = DateTime(
pastMonth.year,
pastMonth.month - 1 > 0
? pastMonth.month - 1
: pastMonth.month - 1 + 12,
pastMonth.day);
final formattedVeryPastMonth =
'${veryPastMonth.year}-${veryPastMonth.month.toString().padLeft(2, '0')}-${veryPastMonth.day.toString().padLeft(2, '0')}';
pastMessage['previousPaymentDate'] = formattedVeryPastMonth;
resultData.add(pastMessage);
}
}
print('TestSmsData: 생성된 테스트 메시지 수: ${resultData.length}');
return resultData;
}
// 최근 6개월의 월간 지출 데이터를 반환하는 메서드
static List<Map<String, dynamic>> getMonthlyExpenseData() {
final now = DateTime.now();
final List<Map<String, dynamic>> monthlyData = [];
// 기본 구독 서비스와 금액 (현재 구독 중인 서비스)
final baseServices = [
{'serviceName': '넷플릭스', 'cost': 9900.0},
{'serviceName': '유튜브프리미엄', 'cost': 14900.0},
{'serviceName': '디즈니플러스', 'cost': 8900.0},
{'serviceName': '애플 iCloud', 'cost': 2900.0},
{'serviceName': '멜론', 'cost': 10900.0},
{'serviceName': '웨이브', 'cost': 7900.0},
{'serviceName': 'Netflix US', 'cost': 9.99}, // 달러 결제 서비스 추가
{'serviceName': 'Spotify Premium', 'cost': 10.99}, // 달러 결제 서비스 추가
{'serviceName': 'GitHub Pro', 'cost': 4.00}, // 달러 결제 서비스 추가
];
// Microsoft 365는 연간 구독이므로 월별 비용으로 환산 (1년에 1번만 결제)
final microsoftMonthlyCost = 12800.0 / 12;
// 최근 6개월 데이터 생성
for (int i = 0; i < 6; i++) {
// i개월 전 날짜
final targetMonth = DateTime(now.month - i <= 0 ? now.year - 1 : now.year,
now.month - i <= 0 ? now.month - i + 12 : now.month - i, 1);
// 해당 월의 모든 서비스 데이터 리스트
final List<Map<String, dynamic>> servicesForMonth = [];
// 해당 월의 총 지출
double totalExpense = 0.0;
// 기본 서비스 추가 (일부 서비스는 구독 시작 시점이 다를 수 있음)
for (final service in baseServices) {
// 3개월 전부터 웨이브 구독 시작
if (service['serviceName'] == '웨이브' && i > 3) {
continue;
}
// 2개월 전부터 디즈니플러스 구독 시작
if (service['serviceName'] == '디즈니플러스' && i > 2) {
continue;
}
// 서비스 정보 추가
final Map<String, dynamic> serviceData = {
'serviceName': service['serviceName'],
'cost': service['cost'],
'date':
'${targetMonth.year}-${targetMonth.month.toString().padLeft(2, '0')}-15',
};
servicesForMonth.add(serviceData);
totalExpense += service['cost'] as double;
}
// Microsoft 365는 정확히 4개월 전에 결제됨
if (i == 4) {
servicesForMonth.add({
'serviceName': 'Microsoft 365',
'cost': 12800.0, // 연간 결제 비용
'date':
'${targetMonth.year}-${targetMonth.month.toString().padLeft(2, '0')}-10',
});
totalExpense += 12800.0;
} else {
// 다른 달에는 월 환산 비용으로 계산
totalExpense += microsoftMonthlyCost;
}
// 3개월 전에는 Spotify를 일시적으로 구독했다가 해지
if (i == 3) {
servicesForMonth.add({
'serviceName': 'Spotify',
'cost': 10900.0,
'date':
'${targetMonth.year}-${targetMonth.month.toString().padLeft(2, '0')}-05',
});
totalExpense += 10900.0;
}
// 5개월 전과 4개월 전에는 쿠팡플레이를 구독했다가 해지
if (i == 4 || i == 5) {
servicesForMonth.add({
'serviceName': '쿠팡플레이',
'cost': 4990.0,
'date':
'${targetMonth.year}-${targetMonth.month.toString().padLeft(2, '0')}-18',
});
totalExpense += 4990.0;
}
// 월별 총 지출 데이터 추가
monthlyData.add({
'year': targetMonth.year,
'month': targetMonth.month,
'totalExpense': totalExpense,
'services': servicesForMonth,
'monthName': _getMonthName(targetMonth.month),
});
}
// 최신 달이 먼저 오도록 reverse
return monthlyData.reversed.toList();
}
// 월 숫자를 한글 월 이름으로 변환
static String _getMonthName(int month) {
const monthNames = [
'1월',
'2월',
'3월',
'4월',
'5월',
'6월',
'7월',
'8월',
'9월',
'10월',
'11월',
'12월'
];
return monthNames[month - 1];
}
}