- Schedule 모델: dynamic id로 변경 (생일/기념일 문자열 ID 지원) - 생일/데뷔/기념일 특별 필드 추가 (isBirthday, isDebut 등) - BirthdayCard: 핑크-보라 그라데이션, 멤버 사진, 케이크 이모지 - DebutCard: 블루 그라데이션, DEBUT/N YEARS 아이콘, 별 장식 - 일정 목록에서 특별 일정 카드 자동 분기 렌더링 - 특별 일정 클릭 시 상세 라우팅 방지 - 달력 그리드 그림자 클리핑 수정 Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
196 lines
5.4 KiB
Dart
196 lines
5.4 KiB
Dart
/// 일정 모델
|
|
library;
|
|
|
|
/// 멤버 정보
|
|
class ScheduleMember {
|
|
final int id;
|
|
final String name;
|
|
|
|
ScheduleMember({required this.id, required this.name});
|
|
|
|
factory ScheduleMember.fromJson(Map<String, dynamic> json) {
|
|
return ScheduleMember(
|
|
id: json['id'] as int,
|
|
name: json['name'] as String,
|
|
);
|
|
}
|
|
}
|
|
|
|
/// 관련 일정 (콘서트 회차 등)
|
|
class RelatedDate {
|
|
final int id;
|
|
final String date;
|
|
final String? time;
|
|
|
|
RelatedDate({required this.id, required this.date, this.time});
|
|
|
|
factory RelatedDate.fromJson(Map<String, dynamic> json) {
|
|
return RelatedDate(
|
|
id: json['id'] as int,
|
|
date: json['date'] as String,
|
|
time: json['time'] as String?,
|
|
);
|
|
}
|
|
}
|
|
|
|
/// 일정 상세 모델
|
|
class ScheduleDetail {
|
|
final int id;
|
|
final String title;
|
|
final String date;
|
|
final String? time;
|
|
final String? description;
|
|
final int categoryId;
|
|
final String? categoryName;
|
|
final String? categoryColor;
|
|
final String? sourceUrl;
|
|
final String? sourceName;
|
|
final String? imageUrl;
|
|
final List<String> images;
|
|
final String? locationName;
|
|
final String? locationAddress;
|
|
final String? locationLat;
|
|
final String? locationLng;
|
|
final List<ScheduleMember> members;
|
|
final List<RelatedDate> relatedDates;
|
|
|
|
ScheduleDetail({
|
|
required this.id,
|
|
required this.title,
|
|
required this.date,
|
|
this.time,
|
|
this.description,
|
|
required this.categoryId,
|
|
this.categoryName,
|
|
this.categoryColor,
|
|
this.sourceUrl,
|
|
this.sourceName,
|
|
this.imageUrl,
|
|
this.images = const [],
|
|
this.locationName,
|
|
this.locationAddress,
|
|
this.locationLat,
|
|
this.locationLng,
|
|
this.members = const [],
|
|
this.relatedDates = const [],
|
|
});
|
|
|
|
factory ScheduleDetail.fromJson(Map<String, dynamic> json) {
|
|
return ScheduleDetail(
|
|
id: json['id'] as int,
|
|
title: json['title'] as String,
|
|
date: json['date'] as String,
|
|
time: json['time'] as String?,
|
|
description: json['description'] as String?,
|
|
categoryId: json['category_id'] as int,
|
|
categoryName: json['category_name'] as String?,
|
|
categoryColor: json['category_color'] as String?,
|
|
sourceUrl: json['source_url'] as String?,
|
|
sourceName: json['source_name'] as String?,
|
|
imageUrl: json['image_url'] as String?,
|
|
images: (json['images'] as List<dynamic>?)?.cast<String>() ?? [],
|
|
locationName: json['location_name'] as String?,
|
|
locationAddress: json['location_address'] as String?,
|
|
locationLat: json['location_lat'] as String?,
|
|
locationLng: json['location_lng'] as String?,
|
|
members: (json['members'] as List<dynamic>?)
|
|
?.map((m) => ScheduleMember.fromJson(m))
|
|
.toList() ??
|
|
[],
|
|
relatedDates: (json['related_dates'] as List<dynamic>?)
|
|
?.map((r) => RelatedDate.fromJson(r))
|
|
.toList() ??
|
|
[],
|
|
);
|
|
}
|
|
|
|
/// 시간 포맷 (HH:mm)
|
|
String? get formattedTime {
|
|
if (time == null) return null;
|
|
return time!.length >= 5 ? time!.substring(0, 5) : time;
|
|
}
|
|
}
|
|
|
|
class Schedule {
|
|
/// ID (일반 일정: int, 생일/기념일: String)
|
|
final dynamic id;
|
|
final String title;
|
|
final String date;
|
|
final String? time;
|
|
final int? categoryId;
|
|
final String? categoryName;
|
|
final String? categoryColor;
|
|
final List<String> members;
|
|
final String? sourceName;
|
|
final String? sourceUrl;
|
|
// 특별 일정 필드
|
|
final bool isBirthday;
|
|
final bool isDebut;
|
|
final bool isAnniversary;
|
|
final String? memberImage;
|
|
final int? anniversaryYear;
|
|
|
|
Schedule({
|
|
required this.id,
|
|
required this.title,
|
|
required this.date,
|
|
this.time,
|
|
this.categoryId,
|
|
this.categoryName,
|
|
this.categoryColor,
|
|
this.members = const [],
|
|
this.sourceName,
|
|
this.sourceUrl,
|
|
this.isBirthday = false,
|
|
this.isDebut = false,
|
|
this.isAnniversary = false,
|
|
this.memberImage,
|
|
this.anniversaryYear,
|
|
});
|
|
|
|
factory Schedule.fromJson(Map<String, dynamic> json) {
|
|
// category 중첩 객체 파싱
|
|
final category = json['category'] as Map<String, dynamic>?;
|
|
|
|
// members 배열 파싱
|
|
final membersList = (json['members'] as List<dynamic>?)
|
|
?.map((m) => m is String ? m : m.toString())
|
|
.toList() ?? [];
|
|
|
|
// source 중첩 객체 파싱
|
|
final source = json['source'] as Map<String, dynamic>?;
|
|
|
|
return Schedule(
|
|
id: json['id'], // int 또는 String (생일/기념일)
|
|
title: json['title'] as String,
|
|
date: json['date'] as String,
|
|
time: json['time'] as String?,
|
|
categoryId: category?['id'] as int?,
|
|
categoryName: category?['name'] as String?,
|
|
categoryColor: category?['color'] as String?,
|
|
members: membersList,
|
|
sourceName: source?['name'] as String?,
|
|
sourceUrl: source?['url'] as String?,
|
|
isBirthday: json['is_birthday'] == true,
|
|
isDebut: json['is_debut'] == true,
|
|
isAnniversary: json['is_anniversary'] == true,
|
|
memberImage: json['member_image'] as String?,
|
|
anniversaryYear: (json['anniversary_year'] as num?)?.toInt(),
|
|
);
|
|
}
|
|
|
|
/// 특별 일정 여부 (생일, 데뷔, 기념일)
|
|
bool get isSpecial => isBirthday || isDebut || isAnniversary;
|
|
|
|
/// 일반 일정의 int ID (특별 일정은 null)
|
|
int? get numericId => id is int ? id : null;
|
|
|
|
/// 멤버 리스트 반환
|
|
List<String> get memberList => members;
|
|
|
|
/// 시간 포맷 (HH:mm)
|
|
String? get formattedTime {
|
|
if (time == null) return null;
|
|
return time!.length >= 5 ? time!.substring(0, 5) : time;
|
|
}
|
|
}
|