[Mobile]/[Flutter]
[Flutter] TableCalendar EventHandler
HiSmith
2023. 9. 17. 16:49
반응형
플러터에서 달력에 대한 오픈소스를 사용하면서 이벤트 관련 내용을 개발했다.
개발 기능은 아래와 같다.
1. 일정 내용을 로드
2. 달력 컴포넌트에 일정을 표시
3. 일정을 클릭시 해당 일정에 대한 상세 내용 확인
4. 해당 상세 내용 클릭시, 팝업으로 그룹에 내용 표기
무튼 소스 내용은 아래와 같다.
1. 화면소스
import 'dart:collection';
import 'package:flutter/material.dart';
import 'package:flutter_naver_map/flutter_naver_map.dart';
import 'package:get/get.dart';
import 'package:goodshot/global-provider.dart';
import 'package:goodshot/util/const.dart';
import 'package:goodshot/view/group/http/group-api.dart';
import 'package:goodshot/view/group/http/group-join-api.dart';
import 'package:goodshot/view/group/popup/group-detail-popup.dart';
import 'package:goodshot/view/group/popup/group-schedule-detail-popup.dart';
import 'package:goodshot/view/schedule/dto/schedule.dart';
import 'package:goodshot/view/schedule/http/schedule-api.dart';
import 'package:goodshot/view/schedule/utils/schedule-util.dart';
import 'package:table_calendar/table_calendar.dart';
final GlobalProvider globalProvider = Get.find();
class TableCalendarScreen extends StatefulWidget {
//Future.sync(() => getScheduleList(globalProvider.getUserId()).then((value) => eventList=value));
@override
_TableEventsExampleState createState() => _TableEventsExampleState();
}
class _TableEventsExampleState extends State<TableCalendarScreen> {
Map<DateTime,List<Event>> eventList = Map();
late ValueNotifier<List<Event>> _selectedEvents;
CalendarFormat _calendarFormat = CalendarFormat.month;
RangeSelectionMode _rangeSelectionMode = RangeSelectionMode
.toggledOff; // Can be toggled on/off by longpressing a date
DateTime _focusedDay = DateTime.now();
DateTime? _selectedDay;
DateTime? _rangeStart;
DateTime? _rangeEnd;
@override
void initState() {
Future.sync(() => getScheduleList(globalProvider.getUserId()).then((value) => eventList = value)).then((value) =>{
setState(() {
eventList = value;
_selectedDay = _focusedDay;
_selectedEvents = ValueNotifier(_getEventsForDay(_selectedDay!));
})
}
);
_selectedDay = _focusedDay;
_selectedEvents = ValueNotifier(_getEventsForDay(_selectedDay!));
super.initState();
}
@override
void dispose() {
_selectedEvents.dispose();
super.dispose();
}
List<Event> _getEventsForDay(DateTime day) {
return eventList[DateTime(day.year,day.month,day.day)] ?? [];
}
List<Event> _getEventsForRange(DateTime start, DateTime end) {
// Implementation example
final days = daysInRange(start, end);
return [
for (final d in days) ..._getEventsForDay(d),
];
}
void _onDaySelected(DateTime selectedDay, DateTime focusedDay) {
print('select');
print(eventList);
if (!isSameDay(_selectedDay, selectedDay)) {
setState(() {
_selectedDay = selectedDay;
_focusedDay = focusedDay;
_rangeStart = null; // Important to clean those
_rangeEnd = null;
_rangeSelectionMode = RangeSelectionMode.toggledOff;
});
_selectedEvents.value = _getEventsForDay(selectedDay);
}
}
void _onRangeSelected(DateTime? start, DateTime? end, DateTime focusedDay) {
setState(() {
_selectedDay = null;
_focusedDay = focusedDay;
_rangeStart = start;
_rangeEnd = end;
_rangeSelectionMode = RangeSelectionMode.toggledOn;
});
// `start` or `end` could be null
if (start != null && end != null) {
_selectedEvents.value = _getEventsForRange(start, end);
} else if (start != null) {
_selectedEvents.value = _getEventsForDay(start);
} else if (end != null) {
_selectedEvents.value = _getEventsForDay(end);
}
}
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: Text('굿샷 모임 캘린더'),
),
body: Column(
children: [
TableCalendar<Event>(
firstDay: kFirstDay,
lastDay: kLastDay,
focusedDay: _focusedDay,
selectedDayPredicate: (day) => isSameDay(_selectedDay, day),
rangeStartDay: _rangeStart,
rangeEndDay: _rangeEnd,
calendarFormat: _calendarFormat,
rangeSelectionMode: _rangeSelectionMode,
eventLoader: _getEventsForDay,
startingDayOfWeek: StartingDayOfWeek.monday,
locale: "ko-KR",
headerStyle: HeaderStyle(
formatButtonVisible: false,
titleCentered: true,
),
calendarStyle: CalendarStyle(
// Use `CalendarStyle` to customize the UI
outsideDaysVisible: false,
),
onDaySelected: _onDaySelected,
onRangeSelected: _onRangeSelected,
onFormatChanged: (format) {
if (_calendarFormat != format) {
setState(() {
_calendarFormat = format;
});
}
},
onPageChanged: (focusedDay) {
_focusedDay = focusedDay;
},
),
const SizedBox(height: 8.0),
Expanded(
child: ValueListenableBuilder<List<Event>>(
valueListenable: _selectedEvents,
builder: (context, value, _) {
return ListView.builder(
itemCount: value.length,
itemBuilder: (context, index) {
return Container(
margin: const EdgeInsets.symmetric(
horizontal: 12.0,
vertical: 4.0,
),
decoration: BoxDecoration(
border: Border.all(),
borderRadius: BorderRadius.circular(12.0),
),
child: ListTile(
onTap: () {
//print('${value[index].groupNo}');
getGroupDetail(value[index].groupNo).then(
(value)=>
{
showDialog(context: context,
builder: (BuildContext context){
return GetGroupScheduleDetailPopup(getGroupItem: value);
})
});
},
title: Text('${value[index]}'),
),
);
},
);
},
),
),
],
),
);
}
}
테이블 위젯을 로드하고 위젯 리스트를 불러온다.
그리고 해당 위젯 리스트는
LinkedHashMap<DateTime,List<Event>>의 형태로 가져온다.
해당 DateTime으로 그룹화를 시킨다.
해당 내용을 지원하는 유틸이다
import 'dart:collection';
import 'package:get/get.dart';
import 'package:goodshot/global-provider.dart';
import 'package:goodshot/view/schedule/dto/schedule.dart';
import 'package:goodshot/view/schedule/http/schedule-api.dart';
import 'package:table_calendar/table_calendar.dart';
/// Example events.
///
/// Using a [LinkedHashMap] is highly recommended if you decide to use a map.
// final events = LinkedHashMap(
// equals: isSameDay,
//
// )..addAll(eventSource(String userId));
//late Map<DateTime,List<Event>> eventList = Map<DateTime,List<Event>>();
Map<DateTime,List<Event>> settingSchedule(String userId) {
late Map<DateTime,List<Event>> eventList = Map();
getScheduleList(userId).then((value)=>{
eventList = value,
});
return eventList;
}
//Map<DateTime,List<Event>> eventList
LinkedHashMap<DateTime, List<Event>> getEventStore(Map<DateTime,List<Event>> eventList) {
LinkedHashMap<DateTime, List<Event>> kEvents = LinkedHashMap<DateTime, List<Event>>(
equals: isSameDay,
hashCode: getHashCode,
)..addAll(eventList);
return kEvents;
}
int getHashCode(DateTime key) {
return key.day * 1000000 + key.month * 10000 + key.year;
}
/// Returns a list of [DateTime] objects from [first] to [last], inclusive.
List<DateTime> daysInRange(DateTime first, DateTime last) {
final dayCount = last.difference(first).inDays + 1;
return List.generate(
dayCount,
(index) => DateTime.utc(first.year, first.month, first.day + index),
);
}
final kToday = DateTime.now();
final kFirstDay = DateTime(kToday.year, kToday.month - 3, kToday.day);
final kLastDay = DateTime(kToday.year, kToday.month + 3, kToday.day);
나는 연,월,일에 대해서만 해당 값을 리턴하기 때문에 해당값에 대한 것들을 세팅한다.
마지막으로 api 와 dto이다.
DateTime dateOnly(DateTime dateTime) {
if (dateTime.isUtc) {
return DateTime.utc(dateTime.year, dateTime.month, dateTime.day);
} else {
return DateTime(dateTime.year, dateTime.month, dateTime.day);
}
}
Future<Map<DateTime, List<Event>>> getScheduleList(String userId) async{
Iterable l;
List<Event> eventList;
var eventDayMap = <DateTime, List<Event>>{};
var url = Uri.http(backendHost, 'get-schedule',{"userId":userId});
await http.get(url).then((value) =>
{
l = json.decode(value.body),
eventList = List<Event>.from(l.map((model)=> Event.fromJson(model))),
for (var event in eventList) {
(eventDayMap[dateOnly(event.date)] ??= []).add(event),
}
});
return eventDayMap;
}
class Event {
final String title;
final DateTime date;
final int groupNo;
const Event({required this.date, required this.title,required this.groupNo});
@override
String toString() => title;
factory Event.fromJson(dynamic json){
DateTime time = DateTime.parse(json?['date']);
Event event = Event(
title: json?['title'],
groupNo: json?['group_no'],
date: DateTime(time.year,time.month,time.day)
);
return event;
}
}
해당 dto와 api로 값을 세팅하고, 사용하게 된다.
무튼 해당 방식으로 진행이 되고 아래와 같은 결과를 얻게 된다.
소스를 그대로 가져다 쓰면 오류는 안날 것이다
무튼 해당 관련 모듈구현은 끝났다.
소스를 좀 정리하고 다시 올려야겠다.
반응형