From 627ed6dcbc6f1635f401a9ad88bae1b8e0496a89 Mon Sep 17 00:00:00 2001 From: Kurogoma4D Date: Mon, 26 Jul 2021 16:40:16 +0900 Subject: [PATCH 1/5] add: indicatorHeight property --- lib/src/infinite_scroll_tab_view.dart | 8 ++++++++ lib/src/inner_infinite_scroll_tab_view.dart | 5 ++++- 2 files changed, 12 insertions(+), 1 deletion(-) diff --git a/lib/src/infinite_scroll_tab_view.dart b/lib/src/infinite_scroll_tab_view.dart index c87192a..1c454de 100644 --- a/lib/src/infinite_scroll_tab_view.dart +++ b/lib/src/infinite_scroll_tab_view.dart @@ -29,6 +29,7 @@ class InfiniteScrollTabView extends StatelessWidget { this.backgroundColor, this.onPageChanged, this.indicatorColor = Colors.pinkAccent, + this.indicatorHeight, this.tabHeight = 44.0, this.tabPadding = 12.0, }) : super(key: key); @@ -83,6 +84,12 @@ class InfiniteScrollTabView extends StatelessWidget { /// Defaults to [Colors.pinkAccent], and must not be null. final Color indicatorColor; + /// The height of indicator. + /// + /// If this is null, the indicator height is aligned to [separator] height, or + /// it also null, then fallbacks to 2.0. + final double? indicatorHeight; + /// The height of tab contents. /// /// Defaults to 44.0. @@ -111,6 +118,7 @@ class InfiniteScrollTabView extends StatelessWidget { backgroundColor: backgroundColor, onPageChanged: onPageChanged, indicatorColor: indicatorColor, + indicatorHeight: indicatorHeight, defaultLocale: Localizations.localeOf(context), tabHeight: tabHeight, tabPadding: tabPadding, diff --git a/lib/src/inner_infinite_scroll_tab_view.dart b/lib/src/inner_infinite_scroll_tab_view.dart index 2983c28..1e33544 100644 --- a/lib/src/inner_infinite_scroll_tab_view.dart +++ b/lib/src/inner_infinite_scroll_tab_view.dart @@ -24,6 +24,7 @@ class InnerInfiniteScrollTabView extends StatefulWidget { this.backgroundColor, this.onPageChanged, required this.indicatorColor, + this.indicatorHeight, required this.defaultLocale, required this.tabHeight, required this.tabPadding, @@ -41,6 +42,7 @@ class InnerInfiniteScrollTabView extends StatefulWidget { final Color? backgroundColor; final ValueChanged? onPageChanged; final Color indicatorColor; + final double? indicatorHeight; final Locale defaultLocale; final double tabHeight; final double tabPadding; @@ -85,7 +87,8 @@ class InnerInfiniteScrollTabViewState extends State final List> _tabSizeTweens = []; List> get tabSizeTweens => _tabSizeTweens; - double get indicatorWidth => widget.separator?.width ?? 2.0; + double get indicatorWidth => + widget.indicatorHeight ?? widget.separator?.width ?? 2.0; late final _indicatorAnimationController = AnimationController(vsync: this, duration: _tabAnimationDuration) From b32cd5c9819c0656a33ac80157d51f5e9f4c9bed Mon Sep 17 00:00:00 2001 From: Kurogoma4D Date: Mon, 26 Jul 2021 17:07:43 +0900 Subject: [PATCH 2/5] modify: prevent double-tap on tab --- lib/src/inner_infinite_scroll_tab_view.dart | 76 ++++++++++++--------- 1 file changed, 43 insertions(+), 33 deletions(-) diff --git a/lib/src/inner_infinite_scroll_tab_view.dart b/lib/src/inner_infinite_scroll_tab_view.dart index 1e33544..e892131 100644 --- a/lib/src/inner_infinite_scroll_tab_view.dart +++ b/lib/src/inner_infinite_scroll_tab_view.dart @@ -60,7 +60,7 @@ class InnerInfiniteScrollTabViewState extends State ); late final _pageController = CycledScrollController(); - bool _isContentChangingByTab = false; + final ValueNotifier _isContentChangingByTab = ValueNotifier(false); bool _isTabForceScrolling = false; late double _previousTextScaleFactor = widget.textScaleFactor; @@ -185,7 +185,7 @@ class InnerInfiniteScrollTabViewState extends State }); _pageController.addListener(() { - if (_isContentChangingByTab) return; + if (_isContentChangingByTab.value) return; final currentIndexDouble = _pageController.offset / widget.size.width; final currentIndex = currentIndexDouble.floor(); @@ -213,6 +213,8 @@ class InnerInfiniteScrollTabViewState extends State } void _onTapTab(int modIndex, int rawIndex) async { + _isContentChangingByTab.value = true; + widget.onTabTap?.call(modIndex); widget.onPageChanged?.call(modIndex); @@ -239,7 +241,6 @@ class InnerInfiniteScrollTabViewState extends State .animate(_indicatorAnimationController); _indicatorAnimationController.forward(from: 0); - _isContentChangingByTab = true; // 現在のスクロール位置とページインデックスを取得 final currentOffset = _pageController.offset; final currentModIndex = @@ -257,7 +258,7 @@ class InnerInfiniteScrollTabViewState extends State curve: Curves.ease, ); - _isContentChangingByTab = false; + _isContentChangingByTab.value = false; } @override @@ -268,35 +269,12 @@ class InnerInfiniteScrollTabViewState extends State children: [ SizedBox( height: widget.tabHeight + (widget.separator?.width ?? 0), - child: CycledListView.builder( - scrollDirection: Axis.horizontal, - controller: _tabController, - contentCount: widget.contentLength, - itemBuilder: (context, modIndex, rawIndex) { - return Material( - color: widget.backgroundColor, - child: InkWell( - onTap: () => _onTapTab(modIndex, rawIndex), - child: ValueListenableBuilder( - valueListenable: _selectedIndex, - builder: (context, index, _) => - ValueListenableBuilder( - valueListenable: _isTabPositionAligned, - builder: (context, tab, _) => _TabContent( - isTabPositionAligned: tab, - selectedIndex: index, - indicatorColor: widget.indicatorColor, - tabPadding: widget.tabPadding, - modIndex: modIndex, - tabBuilder: widget.tabBuilder, - separator: widget.separator, - indicatorWidth: indicatorWidth, - ), - ), - ), - ), - ); - }, + child: ValueListenableBuilder( + valueListenable: _isContentChangingByTab, + builder: (context, value, _) => AbsorbPointer( + absorbing: value, + child: _buildTabSection(), + ), ), ), Positioned( @@ -337,6 +315,38 @@ class InnerInfiniteScrollTabViewState extends State ); } + Widget _buildTabSection() { + return CycledListView.builder( + scrollDirection: Axis.horizontal, + controller: _tabController, + contentCount: widget.contentLength, + itemBuilder: (context, modIndex, rawIndex) { + return Material( + color: widget.backgroundColor, + child: InkWell( + onTap: () => _onTapTab(modIndex, rawIndex), + child: ValueListenableBuilder( + valueListenable: _selectedIndex, + builder: (context, index, _) => ValueListenableBuilder( + valueListenable: _isTabPositionAligned, + builder: (context, tab, _) => _TabContent( + isTabPositionAligned: tab, + selectedIndex: index, + indicatorColor: widget.indicatorColor, + tabPadding: widget.tabPadding, + modIndex: modIndex, + tabBuilder: widget.tabBuilder, + separator: widget.separator, + indicatorWidth: indicatorWidth, + ), + ), + ), + ), + ); + }, + ); + } + @override void dispose() { _tabController.dispose(); From 91917833e2491b2adb2d0a2efa6873d5bdd34171 Mon Sep 17 00:00:00 2001 From: Kurogoma4D Date: Mon, 26 Jul 2021 17:51:10 +0900 Subject: [PATCH 3/5] modify: fix calculation for amount of page movement --- lib/src/inner_infinite_scroll_tab_view.dart | 7 +++---- 1 file changed, 3 insertions(+), 4 deletions(-) diff --git a/lib/src/inner_infinite_scroll_tab_view.dart b/lib/src/inner_infinite_scroll_tab_view.dart index e892131..ef6907d 100644 --- a/lib/src/inner_infinite_scroll_tab_view.dart +++ b/lib/src/inner_infinite_scroll_tab_view.dart @@ -219,7 +219,6 @@ class InnerInfiniteScrollTabViewState extends State widget.onPageChanged?.call(modIndex); HapticFeedback.selectionClick(); - _selectedIndex.value = modIndex; _isTabPositionAligned.value = true; final sizeOnIndex = _calculateTabSizeFromIndex(modIndex); @@ -243,15 +242,15 @@ class InnerInfiniteScrollTabViewState extends State // 現在のスクロール位置とページインデックスを取得 final currentOffset = _pageController.offset; - final currentModIndex = - (currentOffset ~/ widget.size.width) % widget.contentLength; // 選択したページまでの距離を計算する // modの境界をまたぐ場合を考慮して、近い方向を指すように正負を調整する final move = calculateMoveIndexDistance( - currentModIndex, modIndex, widget.contentLength); + _selectedIndex.value, modIndex, widget.contentLength); final targetPageOffset = currentOffset + move * widget.size.width; + _selectedIndex.value = modIndex; + await _pageController.animateTo( targetPageOffset, duration: _tabAnimationDuration, From 5b756351bf23305c9f8a085ba3a4e2523e13b043 Mon Sep 17 00:00:00 2001 From: Kurogoma4D Date: Mon, 26 Jul 2021 18:13:50 +0900 Subject: [PATCH 4/5] modify: constraint height of indicator. --- lib/src/infinite_scroll_tab_view.dart | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/lib/src/infinite_scroll_tab_view.dart b/lib/src/infinite_scroll_tab_view.dart index 1c454de..d99def0 100644 --- a/lib/src/infinite_scroll_tab_view.dart +++ b/lib/src/infinite_scroll_tab_view.dart @@ -88,6 +88,8 @@ class InfiniteScrollTabView extends StatelessWidget { /// /// If this is null, the indicator height is aligned to [separator] height, or /// it also null, then fallbacks to 2.0. + /// + /// This must 1.0 or higher. final double? indicatorHeight; /// The height of tab contents. @@ -104,6 +106,10 @@ class InfiniteScrollTabView extends StatelessWidget { @override Widget build(BuildContext context) { + if (indicatorHeight != null) { + assert(indicatorHeight! >= 1.0); + } + return LayoutBuilder( builder: (context, constraint) => InnerInfiniteScrollTabView( size: constraint.biggest, From 5073144882372c0131ee56b774a3b292a026f961 Mon Sep 17 00:00:00 2001 From: Kurogoma4D Date: Mon, 26 Jul 2021 18:14:40 +0900 Subject: [PATCH 5/5] bump up to 1.0.1 --- CHANGELOG.md | 5 +++++ example/pubspec.lock | 2 +- pubspec.yaml | 2 +- 3 files changed, 7 insertions(+), 2 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 67cee3e..1cda9e8 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,3 +1,8 @@ +## 1.0.1 +FIX +- Added `indicatorHeight` property to `InfintieScrollTabView`. It will override indicator height if specified non-null value. +- The tabs now prevent double tap. +- Fixed a bug that tapping tab causes changing page to unexpected index sometime. ## 1.0.0 FEAT - Changed internal structure about indicator. It considers specified `separator`'s size now. diff --git a/example/pubspec.lock b/example/pubspec.lock index 5bcf5a5..1a3a79f 100644 --- a/example/pubspec.lock +++ b/example/pubspec.lock @@ -66,7 +66,7 @@ packages: path: ".." relative: true source: path - version: "1.0.0" + version: "1.0.1" matcher: dependency: transitive description: diff --git a/pubspec.yaml b/pubspec.yaml index 3475942..68fb73c 100644 --- a/pubspec.yaml +++ b/pubspec.yaml @@ -1,6 +1,6 @@ name: infinite_scroll_tab_view description: A Flutter package for tab view component that can scroll infinitely. -version: 1.0.0 +version: 1.0.1 repository: https://github.com/cb-cloud/flutter_infinite_scroll_tab_view environment: