diff --git a/CHANGELOG.md b/CHANGELOG.md index 43e05dd8..1bfd69c6 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,4 +1,5 @@ - # [1.1.1] (UnReleased) +- Fixed synchronization of scroll between pages in day and week view. [#186](https://github.com/SimformSolutionsPvtLtd/flutter_calendar_view/issues/186) - Added showWeekTileBorder field whether to show border for header in month view. [#306](https://github.com/SimformSolutionsPvtLtd/flutter_calendar_view/issues/306) - Fixed an issue related to hiding day, which is not in the current month in MonthView. [#328](https://github.com/SimformSolutionsPvtLtd/flutter_calendar_view/issues/328) - Added header title for full day events in week view. [#308](https://github.com/SimformSolutionsPvtLtd/flutter_calendar_view/issues/308) diff --git a/README.md b/README.md index 52a0b364..994035a4 100644 --- a/README.md +++ b/README.md @@ -186,7 +186,8 @@ DayView( hourLinePainter: (lineColor, lineHeight, offset, minuteHeight, showVerticalLine, verticalLineOffset) { return //Your custom painter. }, - dayTitleBuilder: DayHeader.hidden // To Hide day header + dayTitleBuilder: DayHeader.hidden, // To Hide day header + keepScrollOffset: true, // To maintain scroll offset while scrolling ); ``` @@ -227,6 +228,7 @@ WeekView( textOverflow: TextOverflow.ellipsis, maxLines: 2, ), // To set full day events header text config + keepScrollOffset: true, // To maintain scroll offset when the page changes ); ``` diff --git a/lib/src/components/_internal_components.dart b/lib/src/components/_internal_components.dart index 48ce19b3..cebddc29 100644 --- a/lib/src/components/_internal_components.dart +++ b/lib/src/components/_internal_components.dart @@ -98,9 +98,12 @@ class _LiveTimeIndicatorState extends State { /// to set dy offset of live time indicator final startMinutes = widget.startHour * 60; - /// Check if live time is not between startHour and endHour then + /// Check if live time is not between startHour and endHour if it is then /// don't show live time indicator - if (_currentTime.hour > widget.startHour || + /// + /// e.g. startHour : 1:00, endHour : 13:00 and live time is 17:00 + /// then no need to display live time indicator on timeline + if (_currentTime.hour > widget.startHour && widget.endHour < _currentTime.hour) { return SizedBox.shrink(); } diff --git a/lib/src/day_view/_internal_day_view_page.dart b/lib/src/day_view/_internal_day_view_page.dart index cf37908d..804ea239 100644 --- a/lib/src/day_view/_internal_day_view_page.dart +++ b/lib/src/day_view/_internal_day_view_page.dart @@ -101,6 +101,8 @@ class InternalDayViewPage extends StatefulWidget { /// Display full day events. final FullDayEventBuilder fullDayEventBuilder; + final ScrollController dayViewScrollController; + /// Flag to display half hours. final bool showHalfHours; @@ -116,10 +118,11 @@ class InternalDayViewPage extends StatefulWidget { /// Settings for half hour indicator lines. final HourIndicatorSettings quarterHourIndicatorSettings; + /// Scroll listener to set every page's last offset final void Function(ScrollController) scrollListener; - /// Scroll offset of day view page. - final double scrollOffset; + /// Last scroll offset of day view page. + final double lastScrollOffset; /// Emulate vertical line offset from hour line starts. final double emulateVerticalOffsetBy; @@ -127,6 +130,9 @@ class InternalDayViewPage extends StatefulWidget { /// This field will be used to set end hour for day view final int endHour; + /// Flag to keep scrollOffset of pages on page change + final bool keepScrollOffset; + /// Defines a single day page. const InternalDayViewPage({ Key? key, @@ -154,8 +160,9 @@ class InternalDayViewPage extends StatefulWidget { required this.minuteSlotSize, required this.scrollNotifier, required this.fullDayEventBuilder, + required this.dayViewScrollController, required this.scrollListener, - this.scrollOffset = 0.0, + this.lastScrollOffset = 0.0, required this.dayDetectorBuilder, required this.showHalfHours, required this.showQuarterHours, @@ -165,6 +172,7 @@ class InternalDayViewPage extends StatefulWidget { required this.quarterHourIndicatorSettings, required this.emulateVerticalOffsetBy, required this.onTileDoubleTap, + this.keepScrollOffset = false, }) : super(key: key); @override @@ -179,7 +187,7 @@ class _InternalDayViewPageState void initState() { super.initState(); scrollController = ScrollController( - initialScrollOffset: widget.scrollOffset, + initialScrollOffset: widget.lastScrollOffset, ); scrollController.addListener(_scrollControllerListener); } @@ -198,21 +206,23 @@ class _InternalDayViewPageState @override Widget build(BuildContext context) { - final fullDayEventList = controller.getFullDayEvent(date); + final fullDayEventList = widget.controller.getFullDayEvent(widget.date); return Container( height: widget.height, width: widget.width, child: Column( children: [ fullDayEventList.isEmpty - ? SizedBox.shrink() - :widget.fullDayEventBuilder( - widget.controller.getFullDayEvent(widget.date), - widget.date, - ), + ? SizedBox.shrink() + : widget.fullDayEventBuilder( + widget.controller.getFullDayEvent(widget.date), + widget.date, + ), Expanded( child: SingleChildScrollView( - controller: scrollController, + controller: widget.keepScrollOffset + ? scrollController + : widget.dayViewScrollController, child: SizedBox( height: widget.height, width: widget.width, @@ -228,60 +238,64 @@ class _InternalDayViewPageState minuteHeight: widget.heightPerMinute, verticalLineOffset: widget.verticalLineOffset, showVerticalLine: widget.showVerticalLine, - lineStyle: hourIndicatorSettings.lineStyle, - dashWidth: hourIndicatorSettings.dashWidth, - dashSpaceWidth: hourIndicatorSettings.dashSpaceWidth, - emulateVerticalOffsetBy: emulateVerticalOffsetBy, - startHour: startHour, - endHour: endHour, + lineStyle: widget.hourIndicatorSettings.lineStyle, + dashWidth: widget.hourIndicatorSettings.dashWidth, + dashSpaceWidth: + widget.hourIndicatorSettings.dashSpaceWidth, + emulateVerticalOffsetBy: widget.emulateVerticalOffsetBy, + startHour: widget.startHour, + endHour: widget.endHour, ), ), - if (showHalfHours) + if (widget.showHalfHours) CustomPaint( - size: Size(width, height), + size: Size(widget.width, widget.height), painter: HalfHourLinePainter( - lineColor: halfHourIndicatorSettings.color, - lineHeight: halfHourIndicatorSettings.height, - offset: - timeLineWidth + halfHourIndicatorSettings.offset, - minuteHeight: heightPerMinute, - lineStyle: halfHourIndicatorSettings.lineStyle, - dashWidth: halfHourIndicatorSettings.dashWidth, + lineColor: widget.halfHourIndicatorSettings.color, + lineHeight: widget.halfHourIndicatorSettings.height, + offset: widget.timeLineWidth + + widget.halfHourIndicatorSettings.offset, + minuteHeight: widget.heightPerMinute, + lineStyle: widget.halfHourIndicatorSettings.lineStyle, + dashWidth: widget.halfHourIndicatorSettings.dashWidth, dashSpaceWidth: - halfHourIndicatorSettings.dashSpaceWidth, - startHour: startHour, - endHour: endHour, + widget.halfHourIndicatorSettings.dashSpaceWidth, + startHour: widget.startHour, + endHour: widget.endHour, ), ), - if (showQuarterHours) + if (widget.showQuarterHours) CustomPaint( - size: Size(width, height), + size: Size(widget.width, widget.height), painter: QuarterHourLinePainter( - lineColor: quarterHourIndicatorSettings.color, - lineHeight: quarterHourIndicatorSettings.height, - offset: timeLineWidth + - quarterHourIndicatorSettings.offset, - minuteHeight: heightPerMinute, - lineStyle: quarterHourIndicatorSettings.lineStyle, - dashWidth: quarterHourIndicatorSettings.dashWidth, - dashSpaceWidth: - quarterHourIndicatorSettings.dashSpaceWidth, + lineColor: widget.quarterHourIndicatorSettings.color, + lineHeight: + widget.quarterHourIndicatorSettings.height, + offset: widget.timeLineWidth + + widget.quarterHourIndicatorSettings.offset, + minuteHeight: widget.heightPerMinute, + lineStyle: + widget.quarterHourIndicatorSettings.lineStyle, + dashWidth: + widget.quarterHourIndicatorSettings.dashWidth, + dashSpaceWidth: widget + .quarterHourIndicatorSettings.dashSpaceWidth, ), ), - dayDetectorBuilder( - width: width, - height: height, - heightPerMinute: heightPerMinute, - date: date, - minuteSlotSize: minuteSlotSize, + widget.dayDetectorBuilder( + width: widget.width, + height: widget.height, + heightPerMinute: widget.heightPerMinute, + date: widget.date, + minuteSlotSize: widget.minuteSlotSize, ), Align( alignment: Alignment.centerRight, child: EventGenerator( height: widget.height, date: widget.date, - onTileLongTap: onTileLongTap, - onTileDoubleTap: onTileDoubleTap, + onTileLongTap: widget.onTileLongTap, + onTileDoubleTap: widget.onTileDoubleTap, onTileTap: widget.onTileTap, eventArranger: widget.eventArranger, events: widget.controller.getEventsOnDay( @@ -291,8 +305,8 @@ class _InternalDayViewPageState heightPerMinute: widget.heightPerMinute, eventTileBuilder: widget.eventTileBuilder, scrollNotifier: widget.scrollNotifier, - startHour: startHour, - endHour: endHour, + startHour: widget.startHour, + endHour: widget.endHour, width: widget.width - widget.timeLineWidth - widget.hourIndicatorSettings.offset - @@ -305,12 +319,13 @@ class _InternalDayViewPageState timeLineBuilder: widget.timeLineBuilder, timeLineOffset: widget.timeLineOffset, timeLineWidth: widget.timeLineWidth, - showHalfHours: showHalfHours, - startHour: startHour, - endHour: endHour, - showQuarterHours: showQuarterHours, + showHalfHours: widget.showHalfHours, + startHour: widget.startHour, + endHour: widget.endHour, + showQuarterHours: widget.showQuarterHours, key: ValueKey(widget.heightPerMinute), - liveTimeIndicatorSettings: liveTimeIndicatorSettings, + liveTimeIndicatorSettings: + widget.liveTimeIndicatorSettings, ), if (widget.showLiveLine && widget.liveTimeIndicatorSettings.height > 0) @@ -322,8 +337,8 @@ class _InternalDayViewPageState height: widget.height, heightPerMinute: widget.heightPerMinute, timeLineWidth: widget.timeLineWidth, - startHour: startHour, - endHour: endHour, + startHour: widget.startHour, + endHour: widget.endHour, ), ), ], diff --git a/lib/src/day_view/day_view.dart b/lib/src/day_view/day_view.dart index f2c14d83..076025f9 100644 --- a/lib/src/day_view/day_view.dart +++ b/lib/src/day_view/day_view.dart @@ -226,6 +226,9 @@ class DayView extends StatefulWidget { /// This field will be used to set end hour for day view final int endHour; + /// Flag to keep scrollOffset of pages on page change + final bool keepScrollOffset; + /// Main widget for day view. const DayView({ Key? key, @@ -275,6 +278,7 @@ class DayView extends StatefulWidget { this.emulateVerticalOffsetBy = 0, this.onEventDoubleTap, this.endHour = Constants.hoursADay, + this.keepScrollOffset = false, }) : assert(!(onHeaderTitleTap != null && dayTitleBuilder != null), "can't use [onHeaderTitleTap] & [dayTitleBuilder] simultaneously"), assert(timeLineOffset >= 0, @@ -350,7 +354,7 @@ class DayViewState extends State> { @override void initState() { super.initState(); - _lastScrollOffset = widget.scrollOffset; + _lastScrollOffset = widget.scrollOffset ?? 0.0; _reloadCallback = _reload; _setDateRange(); @@ -486,8 +490,6 @@ class DayViewState extends State> { minuteSlotSize: widget.minuteSlotSize, scrollNotifier: _scrollConfiguration, fullDayEventBuilder: _fullDayEventBuilder, - scrollOffset: _lastScrollOffset, - scrollListener: _scrollPageListener, showHalfHours: widget.showHalfHours, showQuarterHours: widget.showQuarterHours, halfHourIndicatorSettings: @@ -498,6 +500,10 @@ class DayViewState extends State> { _quarterHourIndicatorSettings, emulateVerticalOffsetBy: widget.emulateVerticalOffsetBy, + lastScrollOffset: _lastScrollOffset, + dayViewScrollController: _scrollController, + scrollListener: _scrollPageListener, + keepScrollOffset: widget.keepScrollOffset, ), ); }, @@ -745,7 +751,9 @@ class DayViewState extends State> { _currentIndex = index; }); } - animateToDuration(widget.startDuration); + if (!widget.keepScrollOffset) { + animateToDuration(widget.startDuration); + } widget.onPageChange?.call(_currentDate, _currentIndex); } diff --git a/lib/src/week_view/_internal_week_view_page.dart b/lib/src/week_view/_internal_week_view_page.dart index 33a5f5b0..e3b28255 100644 --- a/lib/src/week_view/_internal_week_view_page.dart +++ b/lib/src/week_view/_internal_week_view_page.dart @@ -125,6 +125,8 @@ class InternalWeekViewPage extends StatefulWidget { /// Display full day events. final FullDayEventBuilder fullDayEventBuilder; + final ScrollController weekViewScrollController; + /// First hour displayed in the layout final int startHour; @@ -149,10 +151,14 @@ class InternalWeekViewPage extends StatefulWidget { /// Defines full day events header text config final FullDayHeaderTextConfig fullDayHeaderTextConfig; + /// Scroll listener to set every page's last offset final void Function(ScrollController) scrollListener; - /// Scroll offset of week view page. - final double scrollOffset; + /// Last scroll offset of week view page. + final double lastScrollOffset; + + /// Flag to keep scrollOffset of pages on page change + final bool keepScrollOffset; /// A single page for week view. const InternalWeekViewPage({ @@ -199,7 +205,9 @@ class InternalWeekViewPage extends StatefulWidget { this.fullDayHeaderTitle = '', required this.fullDayHeaderTextConfig, required this.scrollListener, - this.scrollOffset = 0.0, + required this.weekViewScrollController, + this.lastScrollOffset = 0.0, + this.keepScrollOffset = false, }) : super(key: key); @override @@ -214,7 +222,7 @@ class _InternalWeekViewPageState void initState() { super.initState(); scrollController = ScrollController( - initialScrollOffset: widget.scrollOffset, + initialScrollOffset: widget.lastScrollOffset, ); scrollController.addListener(_scrollControllerListener); } @@ -238,8 +246,9 @@ class _InternalWeekViewPageState height: widget.height + widget.weekTitleHeight, width: widget.width, child: Column( - verticalDirection: - showWeekDayAtBottom ? VerticalDirection.up : VerticalDirection.down, + verticalDirection: widget.showWeekDayAtBottom + ? VerticalDirection.up + : VerticalDirection.down, crossAxisAlignment: CrossAxisAlignment.end, children: [ SizedBox( @@ -321,7 +330,9 @@ class _InternalWeekViewPageState ), Expanded( child: SingleChildScrollView( - controller: scrollController, + controller: widget.keepScrollOffset + ? scrollController + : widget.weekViewScrollController, child: SizedBox( height: widget.height, width: widget.width, @@ -337,41 +348,44 @@ class _InternalWeekViewPageState minuteHeight: widget.heightPerMinute, verticalLineOffset: widget.verticalLineOffset, showVerticalLine: widget.showVerticalLine, - startHour: startHour, - emulateVerticalOffsetBy: emulateVerticalOffsetBy, - endHour: endHour, + startHour: widget.startHour, + emulateVerticalOffsetBy: widget.emulateVerticalOffsetBy, + endHour: widget.endHour, ), ), - if (showHalfHours) + if (widget.showHalfHours) CustomPaint( - size: Size(width, height), + size: Size(widget.width, widget.height), painter: HalfHourLinePainter( - lineColor: halfHourIndicatorSettings.color, - lineHeight: halfHourIndicatorSettings.height, - offset: - timeLineWidth + halfHourIndicatorSettings.offset, - minuteHeight: heightPerMinute, - lineStyle: halfHourIndicatorSettings.lineStyle, - dashWidth: halfHourIndicatorSettings.dashWidth, + lineColor: widget.halfHourIndicatorSettings.color, + lineHeight: widget.halfHourIndicatorSettings.height, + offset: widget.timeLineWidth + + widget.halfHourIndicatorSettings.offset, + minuteHeight: widget.heightPerMinute, + lineStyle: widget.halfHourIndicatorSettings.lineStyle, + dashWidth: widget.halfHourIndicatorSettings.dashWidth, dashSpaceWidth: - halfHourIndicatorSettings.dashSpaceWidth, - startHour: halfHourIndicatorSettings.startHour, - endHour: endHour, + widget.halfHourIndicatorSettings.dashSpaceWidth, + startHour: widget.halfHourIndicatorSettings.startHour, + endHour: widget.endHour, ), ), - if (showQuarterHours) + if (widget.showQuarterHours) CustomPaint( - size: Size(width, height), + size: Size(widget.width, widget.height), painter: QuarterHourLinePainter( - lineColor: quarterHourIndicatorSettings.color, - lineHeight: quarterHourIndicatorSettings.height, - offset: timeLineWidth + - quarterHourIndicatorSettings.offset, - minuteHeight: heightPerMinute, - lineStyle: quarterHourIndicatorSettings.lineStyle, - dashWidth: quarterHourIndicatorSettings.dashWidth, - dashSpaceWidth: - quarterHourIndicatorSettings.dashSpaceWidth, + lineColor: widget.quarterHourIndicatorSettings.color, + lineHeight: + widget.quarterHourIndicatorSettings.height, + offset: widget.timeLineWidth + + widget.quarterHourIndicatorSettings.offset, + minuteHeight: widget.heightPerMinute, + lineStyle: + widget.quarterHourIndicatorSettings.lineStyle, + dashWidth: + widget.quarterHourIndicatorSettings.dashWidth, + dashSpaceWidth: widget + .quarterHourIndicatorSettings.dashSpaceWidth, ), ), Align( @@ -384,7 +398,7 @@ class _InternalWeekViewPageState ...List.generate( filteredDates.length, (index) => Container( - decoration: showVerticalLine + decoration: widget.showVerticalLine ? BoxDecoration( border: Border( right: BorderSide( @@ -400,31 +414,31 @@ class _InternalWeekViewPageState width: widget.weekTitleWidth, child: Stack( children: [ - weekDetectorBuilder( - width: weekTitleWidth, - height: height, - heightPerMinute: heightPerMinute, - date: dates[index], - minuteSlotSize: minuteSlotSize, + widget.weekDetectorBuilder( + width: widget.weekTitleWidth, + height: widget.height, + heightPerMinute: widget.heightPerMinute, + date: widget.dates[index], + minuteSlotSize: widget.minuteSlotSize, ), EventGenerator( height: widget.height, date: filteredDates[index], onTileTap: widget.onTileTap, - onTileLongTap: onTileLongTap, - onTileDoubleTap: onTileDoubleTap, + onTileLongTap: widget.onTileLongTap, + onTileDoubleTap: widget.onTileDoubleTap, width: widget.weekTitleWidth, eventArranger: widget.eventArranger, eventTileBuilder: widget.eventTileBuilder, scrollNotifier: widget.scrollConfiguration, - startHour: startHour, + startHour: widget.startHour, events: widget.controller.getEventsOnDay( filteredDates[index], includeFullDayEvents: false, ), heightPerMinute: widget.heightPerMinute, - endHour: endHour, + endHour: widget.endHour, ), ], ), @@ -440,21 +454,24 @@ class _InternalWeekViewPageState height: widget.height, timeLineOffset: widget.timeLineOffset, timeLineBuilder: widget.timeLineBuilder, - startHour: startHour, - showHalfHours: showHalfHours, - showQuarterHours: showQuarterHours, - liveTimeIndicatorSettings: liveTimeIndicatorSettings, - endHour: endHour, + startHour: widget.startHour, + showHalfHours: widget.showHalfHours, + showQuarterHours: widget.showQuarterHours, + liveTimeIndicatorSettings: + widget.liveTimeIndicatorSettings, + endHour: widget.endHour, ), - if (showLiveLine && liveTimeIndicatorSettings.height > 0) + if (widget.showLiveLine && + widget.liveTimeIndicatorSettings.height > 0) LiveTimeIndicator( - liveTimeIndicatorSettings: liveTimeIndicatorSettings, - width: width, - height: height, - heightPerMinute: heightPerMinute, - timeLineWidth: timeLineWidth, - startHour: startHour, - endHour: endHour, + liveTimeIndicatorSettings: + widget.liveTimeIndicatorSettings, + width: widget.width, + height: widget.height, + heightPerMinute: widget.heightPerMinute, + timeLineWidth: widget.timeLineWidth, + startHour: widget.startHour, + endHour: widget.endHour, ), ], ), diff --git a/lib/src/week_view/week_view.dart b/lib/src/week_view/week_view.dart index 321556d0..9b549e6e 100644 --- a/lib/src/week_view/week_view.dart +++ b/lib/src/week_view/week_view.dart @@ -237,6 +237,9 @@ class WeekView extends StatefulWidget { /// Defines full day events header text config final FullDayHeaderTextConfig? fullDayHeaderTextConfig; + /// Flag to keep scrollOffset of pages on page change + final bool keepScrollOffset; + /// Main widget for week view. const WeekView({ Key? key, @@ -294,6 +297,7 @@ class WeekView extends StatefulWidget { this.endHour = Constants.hoursADay, this.fullDayHeaderTitle = '', this.fullDayHeaderTextConfig, + this.keepScrollOffset = false, }) : assert(!(onHeaderTitleTap != null && weekPageHeaderBuilder != null), "can't use [onHeaderTitleTap] & [weekPageHeaderBuilder] simultaneously"), assert((timeLineOffset) >= 0, @@ -381,6 +385,9 @@ class WeekViewState extends State> { super.initState(); _lastScrollOffset = widget.scrollOffset; + _scrollController = + ScrollController(initialScrollOffset: widget.scrollOffset); + _startHour = widget.startHour; _endHour = widget.endHour; @@ -534,7 +541,7 @@ class WeekViewState extends State> { showVerticalLine: widget.showVerticalLines, controller: controller, hourHeight: _hourHeight, - scrollController: _scrollController, + weekViewScrollController: _scrollController, eventArranger: _eventArranger, weekDays: _weekDays, minuteSlotSize: widget.minuteSlotSize, @@ -549,8 +556,9 @@ class WeekViewState extends State> { endHour: _endHour, fullDayHeaderTitle: _fullDayHeaderTitle, fullDayHeaderTextConfig: _fullDayHeaderTextConfig, - scrollOffset: _lastScrollOffset, + lastScrollOffset: _lastScrollOffset, scrollListener: _scrollPageListener, + keepScrollOffset: widget.keepScrollOffset, ), ); },