您最多选择25个主题
主题必须以中文或者字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符
1908 行
74 KiB
1908 行
74 KiB
using System;
|
|
using System.Collections.Generic;
|
|
using Unity.UIWidgets.animation;
|
|
using Unity.UIWidgets.foundation;
|
|
using Unity.UIWidgets.painting;
|
|
using Unity.UIWidgets.rendering;
|
|
using Unity.UIWidgets.service;
|
|
using Unity.UIWidgets.ui;
|
|
using Unity.UIWidgets.widgets;
|
|
using UnityEngine;
|
|
using Color = Unity.UIWidgets.ui.Color;
|
|
using Rect = Unity.UIWidgets.ui.Rect;
|
|
using TextStyle = Unity.UIWidgets.painting.TextStyle;
|
|
using Transform = Unity.UIWidgets.widgets.Transform;
|
|
|
|
namespace Unity.UIWidgets.cupertino {
|
|
class NavBarUtils {
|
|
public const float _kNavBarPersistentHeight = 44.0f;
|
|
|
|
public const float _kNavBarLargeTitleHeightExtension = 52.0f;
|
|
|
|
public const float _kNavBarShowLargeTitleThreshold = 10.0f;
|
|
|
|
public const float _kNavBarEdgePadding = 16.0f;
|
|
|
|
public const float _kNavBarBackButtonTapWidth = 50.0f;
|
|
|
|
public static readonly TimeSpan _kNavBarTitleFadeDuration = TimeSpan.FromMilliseconds(150);
|
|
|
|
public static readonly Color _kDefaultNavBarBorderColor = new Color(0x4C000000);
|
|
|
|
public static readonly Border _kDefaultNavBarBorder = new Border(
|
|
bottom: new BorderSide(
|
|
color: _kDefaultNavBarBorderColor,
|
|
0.0f, // One physical pixel.
|
|
style: BorderStyle.solid
|
|
)
|
|
);
|
|
|
|
public static readonly _HeroTag _defaultHeroTag = new _HeroTag(null);
|
|
|
|
public static HeroFlightShuttleBuilder _navBarHeroFlightShuttleBuilder =
|
|
(flightContext, animation, flightDirection, fromHeroContext, toHeroContext) => {
|
|
D.assert(animation != null);
|
|
D.assert(fromHeroContext != null);
|
|
D.assert(toHeroContext != null);
|
|
D.assert(fromHeroContext.widget is Hero);
|
|
D.assert(toHeroContext.widget is Hero);
|
|
|
|
var fromHeroWidget = (Hero) fromHeroContext.widget;
|
|
var toHeroWidget = (Hero) toHeroContext.widget;
|
|
|
|
D.assert(fromHeroWidget.child is _TransitionableNavigationBar);
|
|
D.assert(toHeroWidget.child is _TransitionableNavigationBar);
|
|
|
|
var fromNavBar = (_TransitionableNavigationBar) fromHeroWidget.child;
|
|
var toNavBar = (_TransitionableNavigationBar) toHeroWidget.child;
|
|
D.assert(fromNavBar.componentsKeys != null);
|
|
D.assert(toNavBar.componentsKeys != null);
|
|
|
|
D.assert(
|
|
fromNavBar.componentsKeys.navBarBoxKey.currentContext.owner != null,
|
|
() => "The from nav bar to Hero must have been mounted in the previous frame"
|
|
);
|
|
|
|
D.assert(
|
|
toNavBar.componentsKeys.navBarBoxKey.currentContext.owner != null,
|
|
() => "The to nav bar to Hero must have been mounted in the previous frame"
|
|
);
|
|
|
|
switch (flightDirection) {
|
|
case HeroFlightDirection.push:
|
|
return new _NavigationBarTransition(
|
|
animation: animation,
|
|
bottomNavBar: fromNavBar,
|
|
topNavBar: toNavBar
|
|
);
|
|
case HeroFlightDirection.pop:
|
|
return new _NavigationBarTransition(
|
|
animation: animation,
|
|
bottomNavBar: toNavBar,
|
|
topNavBar: fromNavBar
|
|
);
|
|
}
|
|
|
|
return null;
|
|
};
|
|
|
|
public static CreateRectTween _linearTranslateWithLargestRectSizeTween = (begin, end) => {
|
|
var largestSize = new Size(
|
|
Mathf.Max(a: begin.size.width, b: end.size.width),
|
|
Mathf.Max(a: begin.size.height, b: end.size.height)
|
|
);
|
|
return new RectTween(
|
|
begin.topLeft & largestSize,
|
|
end.topLeft & largestSize
|
|
);
|
|
};
|
|
|
|
public static HeroPlaceholderBuilder _navBarHeroLaunchPadBuilder = (context, heroSize, child) => {
|
|
D.assert(child is _TransitionableNavigationBar);
|
|
return new Visibility(
|
|
maintainSize: true,
|
|
maintainAnimation: true,
|
|
maintainState: true,
|
|
visible: false,
|
|
child: child
|
|
);
|
|
};
|
|
|
|
public static Widget _wrapWithBackground(
|
|
Border border = null,
|
|
Color backgroundColor = null,
|
|
Brightness? brightness = null,
|
|
Widget child = null,
|
|
bool updateSystemUiOverlay = true
|
|
) {
|
|
var result = child;
|
|
if (updateSystemUiOverlay) {
|
|
var isDark = backgroundColor.computeLuminance() < 0.179;
|
|
var newBrightness = brightness ?? (isDark ? Brightness.dark : Brightness.light);
|
|
SystemUiOverlayStyle overlayStyle;
|
|
switch (newBrightness) {
|
|
case Brightness.dark:
|
|
overlayStyle = SystemUiOverlayStyle.light;
|
|
break;
|
|
case Brightness.light:
|
|
default:
|
|
overlayStyle = SystemUiOverlayStyle.dark;
|
|
break;
|
|
}
|
|
|
|
result = new AnnotatedRegion<SystemUiOverlayStyle>(
|
|
value: overlayStyle,
|
|
sized: true,
|
|
child: result
|
|
);
|
|
}
|
|
|
|
var childWithBackground = new DecoratedBox(
|
|
decoration: new BoxDecoration(
|
|
border: border,
|
|
color: backgroundColor
|
|
),
|
|
child: result
|
|
);
|
|
if (backgroundColor.alpha == 0xFF) {
|
|
return childWithBackground;
|
|
}
|
|
|
|
return new ClipRect(
|
|
child: new BackdropFilter(
|
|
filter: ImageFilter.blur(10.0f, 10.0f),
|
|
child: childWithBackground
|
|
)
|
|
);
|
|
}
|
|
|
|
public static Widget _wrapActiveColor(Color color, BuildContext context, Widget child) {
|
|
if (color == null) {
|
|
return child;
|
|
}
|
|
|
|
return new CupertinoTheme(
|
|
data: CupertinoTheme.of(context: context).copyWith(primaryColor: color),
|
|
child: child
|
|
);
|
|
}
|
|
|
|
public static bool _isTransitionable(BuildContext context) {
|
|
var route = ModalRoute.of(context: context);
|
|
return route is PageRoute && !(route as PageRoute).fullscreenDialog;
|
|
}
|
|
}
|
|
|
|
|
|
class _HeroTag {
|
|
public readonly NavigatorState navigator;
|
|
|
|
public _HeroTag(
|
|
NavigatorState navigator
|
|
) {
|
|
this.navigator = navigator;
|
|
}
|
|
|
|
public override string ToString() {
|
|
return $"Default Hero tag for Cupertino navigation bars with navigator {navigator}";
|
|
}
|
|
|
|
public bool Equals(_HeroTag other) {
|
|
if (ReferenceEquals(null, objB: other)) {
|
|
return false;
|
|
}
|
|
|
|
if (ReferenceEquals(this, objB: other)) {
|
|
return true;
|
|
}
|
|
|
|
return other is _HeroTag && navigator == other.navigator;
|
|
}
|
|
|
|
public override bool Equals(object obj) {
|
|
if (ReferenceEquals(null, objB: obj)) {
|
|
return false;
|
|
}
|
|
|
|
if (ReferenceEquals(this, objB: obj)) {
|
|
return true;
|
|
}
|
|
|
|
if (obj.GetType() != GetType()) {
|
|
return false;
|
|
}
|
|
|
|
return Equals((_HeroTag) obj);
|
|
}
|
|
|
|
public override int GetHashCode() {
|
|
return navigator.GetHashCode();
|
|
}
|
|
|
|
public static bool operator ==(_HeroTag left, _HeroTag right) {
|
|
return Equals(objA: left, objB: right);
|
|
}
|
|
|
|
public static bool operator !=(_HeroTag left, _HeroTag right) {
|
|
return !Equals(objA: left, objB: right);
|
|
}
|
|
}
|
|
|
|
public class CupertinoNavigationBar : ObstructingPreferredSizeWidget {
|
|
public readonly Color actionsForegroundColor;
|
|
|
|
public readonly bool automaticallyImplyLeading;
|
|
|
|
public readonly bool automaticallyImplyMiddle;
|
|
public readonly Color backgroundColor;
|
|
|
|
public readonly Border border;
|
|
public readonly Brightness? brightness;
|
|
|
|
public readonly object heroTag;
|
|
|
|
public readonly Widget leading;
|
|
|
|
public readonly Widget middle;
|
|
|
|
public readonly EdgeInsetsDirectional padding;
|
|
|
|
public readonly string previousPageTitle;
|
|
|
|
public readonly Widget trailing;
|
|
|
|
public readonly bool transitionBetweenRoutes;
|
|
|
|
public CupertinoNavigationBar(
|
|
Key key = null,
|
|
Widget leading = null,
|
|
bool automaticallyImplyLeading = true,
|
|
bool automaticallyImplyMiddle = true,
|
|
string previousPageTitle = null,
|
|
Widget middle = null,
|
|
Widget trailing = null,
|
|
Border border = null,
|
|
Color backgroundColor = null,
|
|
Brightness? brightness = null,
|
|
EdgeInsetsDirectional padding = null,
|
|
Color actionsForegroundColor = null,
|
|
bool transitionBetweenRoutes = true,
|
|
object heroTag = null
|
|
) : base(key: key) {
|
|
this.leading = leading;
|
|
this.automaticallyImplyLeading = automaticallyImplyLeading;
|
|
this.automaticallyImplyMiddle = automaticallyImplyMiddle;
|
|
this.previousPageTitle = previousPageTitle;
|
|
this.middle = middle;
|
|
this.trailing = trailing;
|
|
this.border = border ?? NavBarUtils._kDefaultNavBarBorder;
|
|
this.backgroundColor = backgroundColor;
|
|
this.brightness = brightness;
|
|
this.padding = padding;
|
|
this.actionsForegroundColor = actionsForegroundColor;
|
|
this.transitionBetweenRoutes = transitionBetweenRoutes;
|
|
this.heroTag = heroTag ?? NavBarUtils._defaultHeroTag;
|
|
|
|
D.assert(
|
|
this.heroTag != null,
|
|
() => "heroTag cannot be null. Use transitionBetweenRoutes = false to " +
|
|
"disable Hero transition on this navigation bar."
|
|
);
|
|
|
|
D.assert(
|
|
!transitionBetweenRoutes || ReferenceEquals(objA: this.heroTag, objB: NavBarUtils._defaultHeroTag),
|
|
() => "Cannot specify a heroTag override if this navigation bar does not " +
|
|
"transition due to transitionBetweenRoutes = false."
|
|
);
|
|
}
|
|
|
|
public override Size preferredSize {
|
|
get { return Size.fromHeight(height: NavBarUtils._kNavBarPersistentHeight); }
|
|
}
|
|
|
|
public override bool shouldFullyObstruct(BuildContext context) {
|
|
var backgroundColor = CupertinoDynamicColor.resolve(resolvable: this.backgroundColor, context: context)
|
|
?? CupertinoTheme.of(context: context).barBackgroundColor;
|
|
return backgroundColor.alpha == 0xFF;
|
|
}
|
|
|
|
public override State createState() {
|
|
return new _CupertinoNavigationBarState();
|
|
}
|
|
}
|
|
|
|
class _CupertinoNavigationBarState : State<CupertinoNavigationBar> {
|
|
_NavigationBarStaticComponentsKeys keys;
|
|
|
|
public override void initState() {
|
|
base.initState();
|
|
keys = new _NavigationBarStaticComponentsKeys();
|
|
}
|
|
|
|
public override Widget build(BuildContext context) {
|
|
var backgroundColor =
|
|
CupertinoDynamicColor.resolve(resolvable: widget.backgroundColor, context: context) ??
|
|
CupertinoTheme.of(context: context).barBackgroundColor;
|
|
|
|
var components = new _NavigationBarStaticComponents(
|
|
keys: keys,
|
|
route: ModalRoute.of(context: context),
|
|
userLeading: widget.leading,
|
|
automaticallyImplyLeading: widget.automaticallyImplyLeading,
|
|
automaticallyImplyTitle: widget.automaticallyImplyMiddle,
|
|
previousPageTitle: widget.previousPageTitle,
|
|
userMiddle: widget.middle,
|
|
userTrailing: widget.trailing,
|
|
padding: widget.padding,
|
|
userLargeTitle: null,
|
|
large: false
|
|
);
|
|
|
|
var navBar = NavBarUtils._wrapWithBackground(
|
|
border: widget.border,
|
|
backgroundColor: backgroundColor,
|
|
brightness: widget.brightness,
|
|
new DefaultTextStyle(
|
|
style: CupertinoTheme.of(context: context).textTheme.textStyle,
|
|
child: new _PersistentNavigationBar(
|
|
components: components,
|
|
padding: widget.padding
|
|
)
|
|
)
|
|
);
|
|
var actionsForegroundColor = CupertinoDynamicColor.resolve(
|
|
resolvable: widget.actionsForegroundColor, // ignore: deprecated_member_use_from_same_package
|
|
context: context
|
|
);
|
|
|
|
if (!widget.transitionBetweenRoutes || !NavBarUtils._isTransitionable(context: context)) {
|
|
return NavBarUtils._wrapActiveColor(color: actionsForegroundColor, context: context, child: navBar);
|
|
}
|
|
|
|
return NavBarUtils._wrapActiveColor(
|
|
color: actionsForegroundColor,
|
|
context: context,
|
|
new Builder(
|
|
builder: _context => {
|
|
return new Hero(
|
|
tag: widget.heroTag as _HeroTag == NavBarUtils._defaultHeroTag
|
|
? new _HeroTag(Navigator.of(context: _context))
|
|
: widget.heroTag,
|
|
createRectTween: NavBarUtils._linearTranslateWithLargestRectSizeTween,
|
|
placeholderBuilder: NavBarUtils._navBarHeroLaunchPadBuilder,
|
|
flightShuttleBuilder: NavBarUtils._navBarHeroFlightShuttleBuilder,
|
|
transitionOnUserGestures: true,
|
|
child: new _TransitionableNavigationBar(
|
|
componentsKeys: keys,
|
|
backgroundColor: backgroundColor,
|
|
backButtonTextStyle: CupertinoTheme.of(context: _context).textTheme.navActionTextStyle,
|
|
titleTextStyle: CupertinoTheme.of(context: _context).textTheme.navTitleTextStyle,
|
|
null,
|
|
border: widget.border,
|
|
widget.middle != null,
|
|
false,
|
|
child: navBar
|
|
)
|
|
);
|
|
}
|
|
)
|
|
);
|
|
}
|
|
}
|
|
|
|
public class CupertinoSliverNavigationBar : StatefulWidget {
|
|
public readonly Color actionsForegroundColor;
|
|
|
|
public readonly bool automaticallyImplyLeading;
|
|
|
|
public readonly bool automaticallyImplyTitle;
|
|
|
|
public readonly Color backgroundColor;
|
|
|
|
public readonly Border border;
|
|
|
|
public readonly Brightness? brightness;
|
|
|
|
public readonly object heroTag;
|
|
|
|
public readonly Widget largeTitle;
|
|
|
|
public readonly Widget leading;
|
|
|
|
public readonly Widget middle;
|
|
|
|
public readonly EdgeInsetsDirectional padding;
|
|
|
|
public readonly string previousPageTitle;
|
|
|
|
public readonly Widget trailing;
|
|
|
|
public readonly bool transitionBetweenRoutes;
|
|
|
|
public CupertinoSliverNavigationBar(
|
|
Key key = null,
|
|
Widget largeTitle = null,
|
|
Widget leading = null,
|
|
bool automaticallyImplyLeading = true,
|
|
bool automaticallyImplyTitle = true,
|
|
string previousPageTitle = null,
|
|
Widget middle = null,
|
|
Widget trailing = null,
|
|
Border border = null,
|
|
Color backgroundColor = null,
|
|
Brightness? brightness = null,
|
|
EdgeInsetsDirectional padding = null,
|
|
Color actionsForegroundColor = null,
|
|
bool transitionBetweenRoutes = true,
|
|
object heroTag = null
|
|
) : base(key: key) {
|
|
D.assert(
|
|
automaticallyImplyTitle || largeTitle != null,
|
|
() => "No largeTitle has been provided but automaticallyImplyTitle is also " +
|
|
"false. Either provide a largeTitle or set automaticallyImplyTitle to " +
|
|
"true."
|
|
);
|
|
this.largeTitle = largeTitle;
|
|
this.leading = leading;
|
|
this.automaticallyImplyLeading = automaticallyImplyLeading;
|
|
this.automaticallyImplyTitle = automaticallyImplyTitle;
|
|
this.previousPageTitle = previousPageTitle;
|
|
this.middle = middle;
|
|
this.trailing = trailing;
|
|
this.border = border ?? NavBarUtils._kDefaultNavBarBorder;
|
|
this.backgroundColor = backgroundColor;
|
|
this.brightness = brightness;
|
|
this.padding = padding;
|
|
this.actionsForegroundColor = actionsForegroundColor;
|
|
this.transitionBetweenRoutes = transitionBetweenRoutes;
|
|
this.heroTag = heroTag ?? NavBarUtils._defaultHeroTag;
|
|
}
|
|
|
|
public bool opaque {
|
|
get { return backgroundColor.alpha == 0xFF; }
|
|
}
|
|
|
|
public override State createState() {
|
|
return new _CupertinoSliverNavigationBarState();
|
|
}
|
|
}
|
|
|
|
class _CupertinoSliverNavigationBarState : State<CupertinoSliverNavigationBar> {
|
|
_NavigationBarStaticComponentsKeys keys;
|
|
|
|
public override void initState() {
|
|
base.initState();
|
|
keys = new _NavigationBarStaticComponentsKeys();
|
|
}
|
|
|
|
public override Widget build(BuildContext context) {
|
|
var actionsForegroundColor =
|
|
CupertinoDynamicColor.resolve(resolvable: widget.actionsForegroundColor,
|
|
context: context) // ignore: deprecated_member_use_from_same_package
|
|
?? CupertinoTheme.of(context: context).primaryColor;
|
|
var components = new _NavigationBarStaticComponents(
|
|
keys: keys,
|
|
route: ModalRoute.of(context: context),
|
|
userLeading: widget.leading,
|
|
automaticallyImplyLeading: widget.automaticallyImplyLeading,
|
|
automaticallyImplyTitle: widget.automaticallyImplyTitle,
|
|
previousPageTitle: widget.previousPageTitle,
|
|
userMiddle: widget.middle,
|
|
userTrailing: widget.trailing,
|
|
userLargeTitle: widget.largeTitle,
|
|
padding: widget.padding,
|
|
large: true
|
|
);
|
|
|
|
return NavBarUtils._wrapActiveColor(
|
|
color: actionsForegroundColor,
|
|
context: context,
|
|
new MediaQuery(
|
|
data: MediaQuery.of(context: context).copyWith(textScaleFactor: 1),
|
|
child: new SliverPersistentHeader(
|
|
pinned: true, // iOS navigation bars are always pinned.
|
|
del: new _LargeTitleNavigationBarSliverDelegate(
|
|
keys: keys,
|
|
components: components,
|
|
userMiddle: widget.middle,
|
|
CupertinoDynamicColor.resolve(resolvable: widget.backgroundColor, context: context) ??
|
|
CupertinoTheme.of(context: context).barBackgroundColor,
|
|
brightness: widget.brightness,
|
|
border: widget.border,
|
|
padding: widget.padding,
|
|
actionsForegroundColor: actionsForegroundColor,
|
|
transitionBetweenRoutes: widget.transitionBetweenRoutes,
|
|
heroTag: widget.heroTag,
|
|
NavBarUtils._kNavBarPersistentHeight + MediaQuery.of(context: context).padding.top,
|
|
widget.middle != null
|
|
)
|
|
)
|
|
)
|
|
);
|
|
}
|
|
}
|
|
|
|
class _LargeTitleNavigationBarSliverDelegate : SliverPersistentHeaderDelegate {
|
|
public readonly Color actionsForegroundColor;
|
|
public readonly bool alwaysShowMiddle;
|
|
public readonly Color backgroundColor;
|
|
public readonly Border border;
|
|
public readonly Brightness? brightness;
|
|
public readonly _NavigationBarStaticComponents components;
|
|
public readonly object heroTag;
|
|
|
|
public readonly _NavigationBarStaticComponentsKeys keys;
|
|
public readonly EdgeInsetsDirectional padding;
|
|
public readonly float persistentHeight;
|
|
public readonly bool transitionBetweenRoutes;
|
|
public readonly Widget userMiddle;
|
|
|
|
public _LargeTitleNavigationBarSliverDelegate(
|
|
_NavigationBarStaticComponentsKeys keys = null,
|
|
_NavigationBarStaticComponents components = null,
|
|
Widget userMiddle = null,
|
|
Color backgroundColor = null,
|
|
Brightness? brightness = null,
|
|
Border border = null,
|
|
EdgeInsetsDirectional padding = null,
|
|
Color actionsForegroundColor = null,
|
|
bool transitionBetweenRoutes = false,
|
|
object heroTag = null,
|
|
float persistentHeight = 0.0f,
|
|
bool alwaysShowMiddle = false
|
|
) {
|
|
this.keys = keys;
|
|
this.components = components;
|
|
this.userMiddle = userMiddle;
|
|
this.backgroundColor = backgroundColor;
|
|
this.border = border;
|
|
this.brightness = brightness;
|
|
this.padding = padding;
|
|
this.actionsForegroundColor = actionsForegroundColor;
|
|
this.transitionBetweenRoutes = transitionBetweenRoutes;
|
|
this.heroTag = heroTag;
|
|
this.persistentHeight = persistentHeight;
|
|
this.alwaysShowMiddle = alwaysShowMiddle;
|
|
}
|
|
|
|
public override float? minExtent {
|
|
get { return persistentHeight; }
|
|
}
|
|
|
|
public override float? maxExtent {
|
|
get { return persistentHeight + NavBarUtils._kNavBarLargeTitleHeightExtension; }
|
|
}
|
|
|
|
public override Widget build(BuildContext context, float shrinkOffset, bool overlapsContent) {
|
|
var showLargeTitle =
|
|
shrinkOffset < maxExtent - minExtent - NavBarUtils._kNavBarShowLargeTitleThreshold;
|
|
|
|
var persistentNavigationBar =
|
|
new _PersistentNavigationBar(
|
|
components: components,
|
|
padding: padding,
|
|
middleVisible: alwaysShowMiddle ? null : (bool?) !showLargeTitle
|
|
);
|
|
|
|
var navBar = NavBarUtils._wrapWithBackground(
|
|
border: border,
|
|
CupertinoDynamicColor.resolve(resolvable: backgroundColor, context: context),
|
|
brightness: brightness,
|
|
new DefaultTextStyle(
|
|
style: CupertinoTheme.of(context: context).textTheme.textStyle,
|
|
child: new Stack(
|
|
fit: StackFit.expand,
|
|
children: new List<Widget> {
|
|
new Positioned(
|
|
top: persistentHeight,
|
|
left: 0.0f,
|
|
right: 0.0f,
|
|
bottom: 0.0f,
|
|
child: new ClipRect(
|
|
child: new OverflowBox(
|
|
minHeight: 0.0f,
|
|
maxHeight: float.PositiveInfinity,
|
|
alignment: AlignmentDirectional.bottomStart,
|
|
child: new Padding(
|
|
padding: EdgeInsetsDirectional.only(
|
|
start: NavBarUtils._kNavBarEdgePadding,
|
|
bottom: 8.0f
|
|
),
|
|
child: new SafeArea(
|
|
top: false,
|
|
bottom: false,
|
|
child: new AnimatedOpacity(
|
|
opacity: showLargeTitle ? 1.0f : 0.0f,
|
|
duration: NavBarUtils._kNavBarTitleFadeDuration,
|
|
child: new DefaultTextStyle(
|
|
style: CupertinoTheme.of(context: context).textTheme
|
|
.navLargeTitleTextStyle,
|
|
maxLines: 1,
|
|
overflow: TextOverflow.ellipsis,
|
|
child: components.largeTitle
|
|
)
|
|
)
|
|
)
|
|
)
|
|
)
|
|
)
|
|
),
|
|
new Positioned(
|
|
left: 0.0f,
|
|
right: 0.0f,
|
|
top: 0.0f,
|
|
child: persistentNavigationBar
|
|
)
|
|
}
|
|
)
|
|
)
|
|
);
|
|
|
|
if (!transitionBetweenRoutes || !NavBarUtils._isTransitionable(context: context)) {
|
|
return navBar;
|
|
}
|
|
|
|
return new Hero(
|
|
tag: heroTag as _HeroTag == NavBarUtils._defaultHeroTag
|
|
? new _HeroTag(Navigator.of(context: context))
|
|
: heroTag,
|
|
createRectTween: NavBarUtils._linearTranslateWithLargestRectSizeTween,
|
|
flightShuttleBuilder: NavBarUtils._navBarHeroFlightShuttleBuilder,
|
|
placeholderBuilder: NavBarUtils._navBarHeroLaunchPadBuilder,
|
|
transitionOnUserGestures: true,
|
|
child: new _TransitionableNavigationBar(
|
|
componentsKeys: keys,
|
|
CupertinoDynamicColor.resolve(resolvable: backgroundColor, context: context),
|
|
backButtonTextStyle: CupertinoTheme.of(context: context).textTheme.navActionTextStyle,
|
|
titleTextStyle: CupertinoTheme.of(context: context).textTheme.navTitleTextStyle,
|
|
largeTitleTextStyle: CupertinoTheme.of(context: context).textTheme.navLargeTitleTextStyle,
|
|
border: border,
|
|
userMiddle != null,
|
|
largeExpanded: showLargeTitle,
|
|
child: navBar
|
|
)
|
|
);
|
|
}
|
|
|
|
public override bool shouldRebuild(SliverPersistentHeaderDelegate _oldDelegate) {
|
|
var oldDelegate = _oldDelegate as _LargeTitleNavigationBarSliverDelegate;
|
|
return components != oldDelegate.components
|
|
|| userMiddle != oldDelegate.userMiddle
|
|
|| backgroundColor != oldDelegate.backgroundColor
|
|
|| border != oldDelegate.border
|
|
|| padding != oldDelegate.padding
|
|
|| actionsForegroundColor != oldDelegate.actionsForegroundColor
|
|
|| transitionBetweenRoutes != oldDelegate.transitionBetweenRoutes
|
|
|| persistentHeight != oldDelegate.persistentHeight
|
|
|| alwaysShowMiddle != oldDelegate.alwaysShowMiddle
|
|
|| heroTag != oldDelegate.heroTag;
|
|
}
|
|
}
|
|
|
|
class _PersistentNavigationBar : StatelessWidget {
|
|
public readonly _NavigationBarStaticComponents components;
|
|
public readonly bool? middleVisible;
|
|
public readonly EdgeInsetsDirectional padding;
|
|
|
|
public _PersistentNavigationBar(
|
|
Key key = null,
|
|
_NavigationBarStaticComponents components = null,
|
|
EdgeInsetsDirectional padding = null,
|
|
bool? middleVisible = null
|
|
) : base(key: key) {
|
|
this.components = components;
|
|
this.padding = padding;
|
|
this.middleVisible = middleVisible;
|
|
}
|
|
|
|
public override Widget build(BuildContext context) {
|
|
Widget middle = components.middle;
|
|
|
|
if (middle != null) {
|
|
middle = new DefaultTextStyle(
|
|
style: CupertinoTheme.of(context: context).textTheme.navTitleTextStyle,
|
|
child: middle
|
|
);
|
|
middle = middleVisible == null
|
|
? middle
|
|
: new AnimatedOpacity(
|
|
opacity: middleVisible.Value ? 1.0f : 0.0f,
|
|
duration: NavBarUtils._kNavBarTitleFadeDuration,
|
|
child: middle
|
|
);
|
|
}
|
|
|
|
Widget leading = components.leading;
|
|
Widget backChevron = components.backChevron;
|
|
Widget backLabel = components.backLabel;
|
|
|
|
if (leading == null && backChevron != null && backLabel != null) {
|
|
leading = CupertinoNavigationBarBackButton._assemble(
|
|
_backChevron: backChevron,
|
|
_backLabel: backLabel
|
|
);
|
|
}
|
|
|
|
Widget paddedToolbar = new NavigationToolbar(
|
|
leading: leading,
|
|
middle: middle,
|
|
trailing: components.trailing,
|
|
centerMiddle: true,
|
|
middleSpacing: 6.0f
|
|
);
|
|
|
|
if (padding != null) {
|
|
paddedToolbar = new Padding(
|
|
padding: EdgeInsets.only(
|
|
top: padding.top,
|
|
bottom: padding.bottom
|
|
),
|
|
child: paddedToolbar
|
|
);
|
|
}
|
|
|
|
return new SizedBox(
|
|
height: NavBarUtils._kNavBarPersistentHeight + MediaQuery.of(context: context).padding.top,
|
|
child: new SafeArea(
|
|
bottom: false,
|
|
child: paddedToolbar
|
|
)
|
|
);
|
|
}
|
|
}
|
|
|
|
class _NavigationBarStaticComponentsKeys {
|
|
public readonly GlobalKey backChevronKey;
|
|
public readonly GlobalKey backLabelKey;
|
|
public readonly GlobalKey largeTitleKey;
|
|
public readonly GlobalKey leadingKey;
|
|
public readonly GlobalKey middleKey;
|
|
|
|
public readonly GlobalKey navBarBoxKey;
|
|
public readonly GlobalKey trailingKey;
|
|
|
|
public _NavigationBarStaticComponentsKeys() {
|
|
navBarBoxKey = GlobalKey.key("Navigation bar render box");
|
|
leadingKey = GlobalKey.key("Leading");
|
|
backChevronKey = GlobalKey.key("Back chevron");
|
|
backLabelKey = GlobalKey.key("Back label");
|
|
middleKey = GlobalKey.key("Middle");
|
|
trailingKey = GlobalKey.key("Trailing");
|
|
largeTitleKey = GlobalKey.key("Large title");
|
|
}
|
|
}
|
|
|
|
class _NavigationBarStaticComponents {
|
|
public readonly KeyedSubtree backChevron;
|
|
|
|
public readonly KeyedSubtree backLabel;
|
|
|
|
public readonly KeyedSubtree largeTitle;
|
|
|
|
public readonly KeyedSubtree leading;
|
|
|
|
public readonly KeyedSubtree middle;
|
|
|
|
public readonly KeyedSubtree trailing;
|
|
|
|
public _NavigationBarStaticComponents(
|
|
bool automaticallyImplyLeading,
|
|
bool automaticallyImplyTitle,
|
|
bool large,
|
|
_NavigationBarStaticComponentsKeys keys = null,
|
|
ModalRoute route = null,
|
|
Widget userLeading = null,
|
|
string previousPageTitle = null,
|
|
Widget userMiddle = null,
|
|
Widget userTrailing = null,
|
|
Widget userLargeTitle = null,
|
|
EdgeInsetsDirectional padding = null
|
|
) {
|
|
leading = createLeading(
|
|
leadingKey: keys.leadingKey,
|
|
userLeading: userLeading,
|
|
route: route,
|
|
automaticallyImplyLeading: automaticallyImplyLeading,
|
|
padding: padding
|
|
);
|
|
backChevron = createBackChevron(
|
|
backChevronKey: keys.backChevronKey,
|
|
userLeading: userLeading,
|
|
route: route,
|
|
automaticallyImplyLeading: automaticallyImplyLeading
|
|
);
|
|
backLabel = createBackLabel(
|
|
backLabelKey: keys.backLabelKey,
|
|
userLeading: userLeading,
|
|
route: route,
|
|
previousPageTitle: previousPageTitle,
|
|
automaticallyImplyLeading: automaticallyImplyLeading
|
|
);
|
|
middle = createMiddle(
|
|
middleKey: keys.middleKey,
|
|
userMiddle: userMiddle,
|
|
userLargeTitle: userLargeTitle,
|
|
route: route,
|
|
automaticallyImplyTitle: automaticallyImplyTitle,
|
|
large: large
|
|
);
|
|
trailing = createTrailing(
|
|
trailingKey: keys.trailingKey,
|
|
userTrailing: userTrailing,
|
|
padding: padding
|
|
);
|
|
largeTitle = createLargeTitle(
|
|
largeTitleKey: keys.largeTitleKey,
|
|
userLargeTitle: userLargeTitle,
|
|
route: route,
|
|
automaticImplyTitle: automaticallyImplyTitle,
|
|
large: large
|
|
);
|
|
}
|
|
|
|
static Widget _derivedTitle(
|
|
bool automaticallyImplyTitle,
|
|
ModalRoute currentRoute = null
|
|
) {
|
|
if (automaticallyImplyTitle &&
|
|
currentRoute is CupertinoPageRoute route &&
|
|
route.title != null) {
|
|
return new Text(data: route.title);
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
static KeyedSubtree createLeading(
|
|
bool automaticallyImplyLeading,
|
|
GlobalKey leadingKey = null,
|
|
Widget userLeading = null,
|
|
ModalRoute route = null,
|
|
EdgeInsetsDirectional padding = null
|
|
) {
|
|
Widget leadingContent = null;
|
|
|
|
if (userLeading != null) {
|
|
leadingContent = userLeading;
|
|
}
|
|
else if (
|
|
automaticallyImplyLeading &&
|
|
route is PageRoute pageRoute &&
|
|
route.canPop &&
|
|
pageRoute.fullscreenDialog
|
|
) {
|
|
leadingContent = new CupertinoButton(
|
|
child: new Text("Close"),
|
|
padding: EdgeInsets.zero,
|
|
onPressed: () => { route.navigator.maybePop<object>(); }
|
|
);
|
|
}
|
|
|
|
if (leadingContent == null) {
|
|
return null;
|
|
}
|
|
|
|
return new KeyedSubtree(
|
|
key: leadingKey,
|
|
new Padding(
|
|
padding: EdgeInsetsDirectional.only(
|
|
padding?.start ?? NavBarUtils._kNavBarEdgePadding
|
|
),
|
|
child: IconTheme.merge(
|
|
data: new IconThemeData(
|
|
size: 32.0f
|
|
),
|
|
child: leadingContent
|
|
)
|
|
)
|
|
);
|
|
}
|
|
|
|
static KeyedSubtree createBackChevron(
|
|
bool automaticallyImplyLeading,
|
|
GlobalKey backChevronKey = null,
|
|
Widget userLeading = null,
|
|
ModalRoute route = null
|
|
) {
|
|
if (
|
|
userLeading != null ||
|
|
!automaticallyImplyLeading ||
|
|
route == null ||
|
|
!route.canPop ||
|
|
route is PageRoute pageRoute && pageRoute.fullscreenDialog
|
|
) {
|
|
return null;
|
|
}
|
|
|
|
return new KeyedSubtree(key: backChevronKey, new _BackChevron());
|
|
}
|
|
|
|
static KeyedSubtree createBackLabel(
|
|
bool automaticallyImplyLeading,
|
|
GlobalKey backLabelKey = null,
|
|
Widget userLeading = null,
|
|
ModalRoute route = null,
|
|
string previousPageTitle = null
|
|
) {
|
|
if (
|
|
userLeading != null ||
|
|
!automaticallyImplyLeading ||
|
|
route == null ||
|
|
!route.canPop ||
|
|
route is PageRoute pageRoute && pageRoute.fullscreenDialog
|
|
) {
|
|
return null;
|
|
}
|
|
|
|
return new KeyedSubtree(
|
|
key: backLabelKey,
|
|
new _BackLabel(
|
|
specifiedPreviousTitle: previousPageTitle,
|
|
route: route
|
|
)
|
|
);
|
|
}
|
|
|
|
static KeyedSubtree createMiddle(
|
|
bool large,
|
|
bool automaticallyImplyTitle,
|
|
GlobalKey middleKey = null,
|
|
Widget userMiddle = null,
|
|
Widget userLargeTitle = null,
|
|
ModalRoute route = null
|
|
) {
|
|
var middleContent = userMiddle;
|
|
|
|
if (large) {
|
|
middleContent = middleContent ?? userLargeTitle;
|
|
}
|
|
|
|
middleContent = middleContent ??
|
|
_derivedTitle(automaticallyImplyTitle: automaticallyImplyTitle, currentRoute: route);
|
|
|
|
if (middleContent == null) {
|
|
return null;
|
|
}
|
|
|
|
return new KeyedSubtree(
|
|
key: middleKey,
|
|
child: middleContent
|
|
);
|
|
}
|
|
|
|
static KeyedSubtree createTrailing(
|
|
GlobalKey trailingKey = null,
|
|
Widget userTrailing = null,
|
|
EdgeInsetsDirectional padding = null
|
|
) {
|
|
if (userTrailing == null) {
|
|
return null;
|
|
}
|
|
|
|
return new KeyedSubtree(
|
|
key: trailingKey,
|
|
new Padding(
|
|
padding: EdgeInsetsDirectional.only(
|
|
end: padding?.end ?? NavBarUtils._kNavBarEdgePadding
|
|
),
|
|
child: IconTheme.merge(
|
|
data: new IconThemeData(
|
|
size: 32.0f
|
|
),
|
|
child: userTrailing
|
|
)
|
|
)
|
|
);
|
|
}
|
|
|
|
static KeyedSubtree createLargeTitle(
|
|
bool large,
|
|
bool automaticImplyTitle,
|
|
GlobalKey largeTitleKey = null,
|
|
Widget userLargeTitle = null,
|
|
ModalRoute route = null
|
|
) {
|
|
if (!large) {
|
|
return null;
|
|
}
|
|
|
|
var largeTitleContent = userLargeTitle ?? _derivedTitle(
|
|
automaticallyImplyTitle: automaticImplyTitle,
|
|
currentRoute: route);
|
|
D.assert(
|
|
largeTitleContent != null,
|
|
() => "largeTitle was not provided and there was no title from the route."
|
|
);
|
|
return new KeyedSubtree(
|
|
key: largeTitleKey,
|
|
child: largeTitleContent
|
|
);
|
|
}
|
|
}
|
|
|
|
public class CupertinoNavigationBarBackButton : StatelessWidget {
|
|
public readonly Widget _backChevron;
|
|
|
|
public readonly Widget _backLabel;
|
|
|
|
public readonly Color color;
|
|
|
|
public readonly VoidCallback onPressed;
|
|
|
|
public readonly string previousPageTitle;
|
|
|
|
public CupertinoNavigationBarBackButton(
|
|
Key key = null,
|
|
Color color = null,
|
|
string previousPageTitle = null,
|
|
VoidCallback onPressed = null
|
|
) : base(key: key) {
|
|
_backChevron = null;
|
|
_backLabel = null;
|
|
this.color = color;
|
|
this.previousPageTitle = previousPageTitle;
|
|
this.onPressed = onPressed;
|
|
}
|
|
|
|
internal CupertinoNavigationBarBackButton(
|
|
Widget backChevron,
|
|
Widget backLabel,
|
|
Color color = null,
|
|
string previousPageTitle = null,
|
|
VoidCallback onPressed = null
|
|
) {
|
|
_backChevron = backChevron;
|
|
_backLabel = backLabel;
|
|
this.color = color;
|
|
this.previousPageTitle = previousPageTitle;
|
|
this.onPressed = onPressed;
|
|
}
|
|
|
|
public static CupertinoNavigationBarBackButton _assemble(
|
|
Widget _backChevron,
|
|
Widget _backLabel
|
|
) {
|
|
return new CupertinoNavigationBarBackButton(
|
|
backChevron: _backChevron,
|
|
backLabel: _backLabel
|
|
);
|
|
}
|
|
|
|
public override Widget build(BuildContext context) {
|
|
var currentRoute = ModalRoute.of(context: context);
|
|
if (onPressed == null) {
|
|
D.assert(
|
|
currentRoute?.canPop == true,
|
|
() => "CupertinoNavigationBarBackButton should only be used in routes that can be popped"
|
|
);
|
|
}
|
|
|
|
var actionTextStyle = CupertinoTheme.of(context: context).textTheme.navActionTextStyle;
|
|
if (color != null) {
|
|
actionTextStyle =
|
|
actionTextStyle.copyWith(color: CupertinoDynamicColor.resolve(resolvable: color, context: context));
|
|
}
|
|
|
|
return new CupertinoButton(
|
|
child: new DefaultTextStyle(
|
|
style: actionTextStyle,
|
|
child: new ConstrainedBox(
|
|
constraints: new BoxConstraints(minWidth: NavBarUtils._kNavBarBackButtonTapWidth),
|
|
child: new Row(
|
|
mainAxisSize: MainAxisSize.min,
|
|
mainAxisAlignment: MainAxisAlignment.start,
|
|
children: new List<Widget> {
|
|
new Padding(padding: EdgeInsetsDirectional.only(8.0f)),
|
|
_backChevron ?? new _BackChevron(),
|
|
new Padding(padding: EdgeInsetsDirectional.only(6.0f)),
|
|
new Flexible(
|
|
child: _backLabel ?? new _BackLabel(
|
|
specifiedPreviousTitle: previousPageTitle,
|
|
route: currentRoute
|
|
)
|
|
)
|
|
}
|
|
)
|
|
)
|
|
),
|
|
padding: EdgeInsets.zero,
|
|
onPressed: () => {
|
|
if (onPressed != null) {
|
|
onPressed();
|
|
}
|
|
else {
|
|
Navigator.maybePop<object>(context: context);
|
|
}
|
|
}
|
|
);
|
|
}
|
|
}
|
|
|
|
|
|
class _BackChevron : StatelessWidget {
|
|
public _BackChevron(Key key = null) : base(key: key) {
|
|
}
|
|
|
|
public override Widget build(BuildContext context) {
|
|
var textDirection = Directionality.of(context: context);
|
|
var textStyle = DefaultTextStyle.of(context: context).style;
|
|
|
|
Widget iconWidget = Text.rich(
|
|
new TextSpan(
|
|
new string(new[] {(char) CupertinoIcons.back.codePoint}),
|
|
new TextStyle(
|
|
false,
|
|
color: textStyle.color,
|
|
fontSize: 34.0f,
|
|
fontFamily: CupertinoIcons.back.fontFamily
|
|
//package: CupertinoIcons.back.fontPackage
|
|
)
|
|
)
|
|
);
|
|
var matrix = Matrix4.identity();
|
|
matrix.scale(-1.0f, 1.0f, 1.0f);
|
|
switch (textDirection) {
|
|
case TextDirection.rtl:
|
|
iconWidget = new Transform(
|
|
transform: matrix,
|
|
alignment: Alignment.center,
|
|
transformHitTests: false,
|
|
child: iconWidget
|
|
);
|
|
break;
|
|
case TextDirection.ltr:
|
|
break;
|
|
}
|
|
|
|
return iconWidget;
|
|
}
|
|
}
|
|
|
|
class _BackLabel : StatelessWidget {
|
|
public readonly ModalRoute route;
|
|
|
|
public readonly string specifiedPreviousTitle;
|
|
|
|
public _BackLabel(
|
|
Key key = null,
|
|
string specifiedPreviousTitle = null,
|
|
ModalRoute route = null
|
|
) : base(key: key) {
|
|
this.specifiedPreviousTitle = specifiedPreviousTitle;
|
|
this.route = route;
|
|
}
|
|
|
|
Widget _buildPreviousTitleWidget(BuildContext context, string previousTitle, Widget child) {
|
|
if (previousTitle == null) {
|
|
return new SizedBox(height: 0.0f, width: 0.0f);
|
|
}
|
|
|
|
var textWidget = new Text(
|
|
data: previousTitle,
|
|
maxLines: 1,
|
|
overflow: TextOverflow.ellipsis
|
|
);
|
|
|
|
if (previousTitle.Length > 12) {
|
|
textWidget = new Text("Back");
|
|
}
|
|
|
|
return new Align(
|
|
alignment: AlignmentDirectional.centerStart,
|
|
widthFactor: 1.0f,
|
|
child: textWidget
|
|
);
|
|
}
|
|
|
|
public override Widget build(BuildContext context) {
|
|
if (specifiedPreviousTitle != null) {
|
|
return _buildPreviousTitleWidget(context: context, previousTitle: specifiedPreviousTitle, null);
|
|
}
|
|
|
|
if (route is CupertinoPageRoute && !route.isFirst) {
|
|
var cupertinoRoute = route as CupertinoPageRoute;
|
|
return new ValueListenableBuilder<string>(
|
|
valueListenable: cupertinoRoute.previousTitle,
|
|
builder: _buildPreviousTitleWidget
|
|
);
|
|
}
|
|
|
|
return new SizedBox(height: 0.0f, width: 0.0f);
|
|
}
|
|
}
|
|
|
|
class _TransitionableNavigationBar : StatelessWidget {
|
|
public readonly TextStyle backButtonTextStyle;
|
|
public readonly Color backgroundColor;
|
|
public readonly Border border;
|
|
public readonly Widget child;
|
|
|
|
public readonly _NavigationBarStaticComponentsKeys componentsKeys;
|
|
public readonly bool? hasUserMiddle;
|
|
public readonly bool? largeExpanded;
|
|
public readonly TextStyle largeTitleTextStyle;
|
|
public readonly TextStyle titleTextStyle;
|
|
|
|
public _TransitionableNavigationBar(
|
|
_NavigationBarStaticComponentsKeys componentsKeys = null,
|
|
Color backgroundColor = null,
|
|
TextStyle backButtonTextStyle = null,
|
|
TextStyle titleTextStyle = null,
|
|
TextStyle largeTitleTextStyle = null,
|
|
Border border = null,
|
|
bool? hasUserMiddle = null,
|
|
bool? largeExpanded = null,
|
|
Widget child = null
|
|
) : base(key: componentsKeys.navBarBoxKey) {
|
|
D.assert(largeExpanded != null);
|
|
D.assert(!largeExpanded.Value || largeTitleTextStyle != null);
|
|
|
|
this.componentsKeys = componentsKeys;
|
|
this.backgroundColor = backgroundColor;
|
|
this.backButtonTextStyle = backButtonTextStyle;
|
|
this.titleTextStyle = titleTextStyle;
|
|
this.largeTitleTextStyle = largeTitleTextStyle;
|
|
this.border = border;
|
|
this.hasUserMiddle = hasUserMiddle;
|
|
this.largeExpanded = largeExpanded;
|
|
this.child = child;
|
|
}
|
|
|
|
public RenderBox renderBox {
|
|
get {
|
|
var box = componentsKeys.navBarBoxKey.currentContext.findRenderObject() as RenderBox;
|
|
D.assert(
|
|
result: box.attached,
|
|
() => "_TransitionableNavigationBar.renderBox should be called when building " +
|
|
"hero flight shuttles when the from and the to nav bar boxes are already " +
|
|
"laid out and painted."
|
|
);
|
|
return box;
|
|
}
|
|
}
|
|
|
|
public override Widget build(BuildContext context) {
|
|
D.assert(() => {
|
|
bool? inHero = null;
|
|
context.visitAncestorElements(ancestor => {
|
|
if (ancestor is ComponentElement) {
|
|
D.assert(
|
|
ancestor.widget.GetType() != typeof(_NavigationBarTransition),
|
|
() => "_TransitionableNavigationBar should never re-appear inside " +
|
|
"_NavigationBarTransition. Keyed _TransitionableNavigationBar should " +
|
|
"only serve as anchor points in routes rather than appearing inside " +
|
|
"Hero flights themselves."
|
|
);
|
|
if (ancestor.widget.GetType() == typeof(Hero)) {
|
|
inHero = true;
|
|
}
|
|
}
|
|
|
|
inHero = inHero ?? false;
|
|
return true;
|
|
});
|
|
D.assert(
|
|
inHero == true,
|
|
() => "_TransitionableNavigationBar should only be added as the immediate " +
|
|
"child of Hero widgets."
|
|
);
|
|
return true;
|
|
});
|
|
return child;
|
|
}
|
|
}
|
|
|
|
class _NavigationBarTransition : StatelessWidget {
|
|
public readonly Animation<float> animation;
|
|
public readonly ColorTween backgroundTween;
|
|
public readonly BorderTween borderTween;
|
|
public readonly _TransitionableNavigationBar bottomNavBar;
|
|
|
|
public readonly FloatTween heightTween;
|
|
public readonly _TransitionableNavigationBar topNavBar;
|
|
|
|
public _NavigationBarTransition(
|
|
Animation<float> animation = null,
|
|
_TransitionableNavigationBar topNavBar = null,
|
|
_TransitionableNavigationBar bottomNavBar = null
|
|
) {
|
|
this.animation = animation;
|
|
this.topNavBar = topNavBar;
|
|
this.bottomNavBar = bottomNavBar;
|
|
|
|
heightTween = new FloatTween(
|
|
begin: bottomNavBar.renderBox.size.height,
|
|
end: topNavBar.renderBox.size.height
|
|
);
|
|
backgroundTween = new ColorTween(
|
|
begin: bottomNavBar.backgroundColor,
|
|
end: topNavBar.backgroundColor
|
|
);
|
|
borderTween = new BorderTween(
|
|
begin: bottomNavBar.border,
|
|
end: topNavBar.border
|
|
);
|
|
}
|
|
|
|
public override Widget build(BuildContext context) {
|
|
var componentsTransition = new _NavigationBarComponentsTransition(
|
|
animation: animation,
|
|
bottomNavBar: bottomNavBar,
|
|
topNavBar: topNavBar,
|
|
Directionality.of(context: context)
|
|
);
|
|
|
|
var children = new List<Widget> {
|
|
new AnimatedBuilder(
|
|
animation: animation,
|
|
(_context, child) => {
|
|
return NavBarUtils._wrapWithBackground(
|
|
updateSystemUiOverlay: false,
|
|
backgroundColor: backgroundTween.evaluate(animation: animation),
|
|
border: borderTween.evaluate(animation: animation),
|
|
child: new SizedBox(
|
|
height: heightTween.evaluate(animation: animation),
|
|
width: float.PositiveInfinity
|
|
)
|
|
);
|
|
}
|
|
),
|
|
componentsTransition.bottomBackChevron,
|
|
componentsTransition.bottomBackLabel,
|
|
componentsTransition.bottomLeading,
|
|
componentsTransition.bottomMiddle,
|
|
componentsTransition.bottomLargeTitle,
|
|
componentsTransition.bottomTrailing,
|
|
componentsTransition.topLeading,
|
|
componentsTransition.topBackChevron,
|
|
componentsTransition.topBackLabel,
|
|
componentsTransition.topMiddle,
|
|
componentsTransition.topLargeTitle,
|
|
componentsTransition.topTrailing
|
|
};
|
|
|
|
children.RemoveAll(child => child == null);
|
|
|
|
return new SizedBox(
|
|
height: Mathf.Max(a: heightTween.begin, b: heightTween.end) +
|
|
MediaQuery.of(context: context).padding.top,
|
|
width: float.PositiveInfinity,
|
|
child: new Stack(
|
|
children: children
|
|
)
|
|
);
|
|
}
|
|
}
|
|
|
|
class _NavigationBarComponentsTransition {
|
|
public static Animatable<float> fadeOut = new FloatTween(
|
|
1.0f,
|
|
0.0f
|
|
);
|
|
|
|
public static Animatable<float> fadeIn = new FloatTween(
|
|
0.0f,
|
|
1.0f
|
|
);
|
|
|
|
public readonly Animation<float> animation;
|
|
|
|
public readonly TextStyle bottomBackButtonTextStyle;
|
|
public readonly _NavigationBarStaticComponentsKeys bottomComponents;
|
|
|
|
public readonly bool? bottomHasUserMiddle;
|
|
public readonly bool? bottomLargeExpanded;
|
|
public readonly TextStyle bottomLargeTitleTextStyle;
|
|
|
|
public readonly RenderBox bottomNavBarBox;
|
|
public readonly TextStyle bottomTitleTextStyle;
|
|
|
|
public readonly float forwardDirection;
|
|
public readonly TextStyle topBackButtonTextStyle;
|
|
public readonly _NavigationBarStaticComponentsKeys topComponents;
|
|
public readonly bool? topHasUserMiddle;
|
|
public readonly bool? topLargeExpanded;
|
|
public readonly TextStyle topLargeTitleTextStyle;
|
|
public readonly RenderBox topNavBarBox;
|
|
public readonly TextStyle topTitleTextStyle;
|
|
|
|
public readonly Rect transitionBox;
|
|
|
|
public _NavigationBarComponentsTransition(
|
|
Animation<float> animation = null,
|
|
_TransitionableNavigationBar bottomNavBar = null,
|
|
_TransitionableNavigationBar topNavBar = null,
|
|
TextDirection? directionality = null
|
|
) {
|
|
this.animation = animation;
|
|
bottomComponents = bottomNavBar.componentsKeys;
|
|
topComponents = topNavBar.componentsKeys;
|
|
bottomNavBarBox = bottomNavBar.renderBox;
|
|
topNavBarBox = topNavBar.renderBox;
|
|
bottomBackButtonTextStyle = bottomNavBar.backButtonTextStyle;
|
|
topBackButtonTextStyle = topNavBar.backButtonTextStyle;
|
|
bottomTitleTextStyle = bottomNavBar.titleTextStyle;
|
|
topTitleTextStyle = topNavBar.titleTextStyle;
|
|
bottomLargeTitleTextStyle = bottomNavBar.largeTitleTextStyle;
|
|
topLargeTitleTextStyle = topNavBar.largeTitleTextStyle;
|
|
bottomHasUserMiddle = bottomNavBar.hasUserMiddle;
|
|
topHasUserMiddle = topNavBar.hasUserMiddle;
|
|
bottomLargeExpanded = bottomNavBar.largeExpanded;
|
|
topLargeExpanded = topNavBar.largeExpanded;
|
|
transitionBox =
|
|
bottomNavBar.renderBox.paintBounds.expandToInclude(other: topNavBar.renderBox.paintBounds);
|
|
forwardDirection = directionality == TextDirection.ltr ? 1.0f : -1.0f;
|
|
}
|
|
|
|
public Widget bottomLeading {
|
|
get {
|
|
var bottomLeading = bottomComponents.leadingKey.currentWidget as KeyedSubtree;
|
|
if (bottomLeading == null) {
|
|
return null;
|
|
}
|
|
|
|
return Positioned.fromRelativeRect(
|
|
rect: positionInTransitionBox(key: bottomComponents.leadingKey, from: bottomNavBarBox),
|
|
child: new FadeTransition(
|
|
opacity: fadeOutBy(0.4f),
|
|
child: bottomLeading.child
|
|
)
|
|
);
|
|
}
|
|
}
|
|
|
|
public Widget bottomBackChevron {
|
|
get {
|
|
var bottomBackChevron = bottomComponents.backChevronKey.currentWidget as KeyedSubtree;
|
|
if (bottomBackChevron == null) {
|
|
return null;
|
|
}
|
|
|
|
return Positioned.fromRelativeRect(
|
|
rect: positionInTransitionBox(key: bottomComponents.backChevronKey, from: bottomNavBarBox),
|
|
child: new FadeTransition(
|
|
opacity: fadeOutBy(0.6f),
|
|
child: new DefaultTextStyle(
|
|
style: bottomBackButtonTextStyle,
|
|
child: bottomBackChevron.child
|
|
)
|
|
)
|
|
);
|
|
}
|
|
}
|
|
|
|
public Widget bottomBackLabel {
|
|
get {
|
|
var bottomBackLabel = bottomComponents.backLabelKey.currentWidget as KeyedSubtree;
|
|
|
|
if (bottomBackLabel == null) {
|
|
return null;
|
|
}
|
|
|
|
var from =
|
|
positionInTransitionBox(key: bottomComponents.backLabelKey, from: bottomNavBarBox);
|
|
|
|
var positionTween = new RelativeRectTween(
|
|
begin: from,
|
|
from.shift(
|
|
new Offset(forwardDirection * (-bottomNavBarBox.size.width / 2.0f),
|
|
0.0f
|
|
)
|
|
)
|
|
);
|
|
|
|
return new PositionedTransition(
|
|
animation.drive(child: positionTween),
|
|
child: new FadeTransition(
|
|
opacity: fadeOutBy(0.2f),
|
|
child: new DefaultTextStyle(
|
|
style: bottomBackButtonTextStyle,
|
|
child: bottomBackLabel.child
|
|
)
|
|
)
|
|
);
|
|
}
|
|
}
|
|
|
|
public Widget bottomMiddle {
|
|
get {
|
|
var bottomMiddle = bottomComponents.middleKey.currentWidget as KeyedSubtree;
|
|
var topBackLabel = topComponents.backLabelKey.currentWidget as KeyedSubtree;
|
|
var topLeading = topComponents.leadingKey.currentWidget as KeyedSubtree;
|
|
|
|
if (bottomHasUserMiddle != true && bottomLargeExpanded == true) {
|
|
return null;
|
|
}
|
|
|
|
if (bottomMiddle != null && topBackLabel != null) {
|
|
return new PositionedTransition(
|
|
animation.drive(slideFromLeadingEdge(
|
|
fromKey: bottomComponents.middleKey,
|
|
fromNavBarBox: bottomNavBarBox,
|
|
toKey: topComponents.backLabelKey,
|
|
toNavBarBox: topNavBarBox
|
|
)),
|
|
child: new FadeTransition(
|
|
opacity: fadeOutBy(bottomHasUserMiddle == true ? 0.4f : 0.7f),
|
|
child: new Align(
|
|
alignment: AlignmentDirectional.centerStart,
|
|
child: new DefaultTextStyleTransition(
|
|
animation.drive(new TextStyleTween(
|
|
begin: bottomTitleTextStyle,
|
|
end: topBackButtonTextStyle
|
|
)),
|
|
child: bottomMiddle.child
|
|
)
|
|
)
|
|
)
|
|
);
|
|
}
|
|
|
|
if (bottomMiddle != null && topLeading != null) {
|
|
return Positioned.fromRelativeRect(
|
|
rect: positionInTransitionBox(key: bottomComponents.middleKey, from: bottomNavBarBox),
|
|
child: new FadeTransition(
|
|
opacity: fadeOutBy(bottomHasUserMiddle == true ? 0.4f : 0.7f),
|
|
child: new DefaultTextStyle(
|
|
style: bottomTitleTextStyle,
|
|
child: bottomMiddle.child
|
|
)
|
|
)
|
|
);
|
|
}
|
|
|
|
return null;
|
|
}
|
|
}
|
|
|
|
public Widget bottomLargeTitle {
|
|
get {
|
|
var bottomLargeTitle = (KeyedSubtree) bottomComponents.largeTitleKey.currentWidget;
|
|
var topBackLabel = (KeyedSubtree) topComponents.backLabelKey.currentWidget;
|
|
var topLeading = (KeyedSubtree) topComponents.leadingKey.currentWidget;
|
|
|
|
if (bottomLargeTitle == null || bottomLargeExpanded != true) {
|
|
return null;
|
|
}
|
|
|
|
if (bottomLargeTitle != null && topBackLabel != null) {
|
|
return new PositionedTransition(
|
|
animation.drive(slideFromLeadingEdge(
|
|
fromKey: bottomComponents.largeTitleKey,
|
|
fromNavBarBox: bottomNavBarBox,
|
|
toKey: topComponents.backLabelKey,
|
|
toNavBarBox: topNavBarBox
|
|
)),
|
|
child: new FadeTransition(
|
|
opacity: fadeOutBy(0.6f),
|
|
child: new Align(
|
|
alignment: AlignmentDirectional.centerStart,
|
|
child: new DefaultTextStyleTransition(
|
|
animation.drive(new TextStyleTween(
|
|
begin: bottomLargeTitleTextStyle,
|
|
end: topBackButtonTextStyle
|
|
)),
|
|
maxLines: 1,
|
|
overflow: TextOverflow.ellipsis,
|
|
child: bottomLargeTitle.child
|
|
)
|
|
)
|
|
)
|
|
);
|
|
}
|
|
|
|
if (bottomLargeTitle != null && topLeading != null) {
|
|
var from = positionInTransitionBox(key: bottomComponents.largeTitleKey, from: bottomNavBarBox);
|
|
|
|
var positionTween = new RelativeRectTween(
|
|
begin: from,
|
|
from.shift(
|
|
new Offset(forwardDirection * bottomNavBarBox.size.width / 4.0f,
|
|
0.0f
|
|
)
|
|
)
|
|
);
|
|
|
|
return new PositionedTransition(
|
|
animation.drive(child: positionTween),
|
|
child: new FadeTransition(
|
|
opacity: fadeOutBy(0.4f),
|
|
child: new DefaultTextStyle(
|
|
style: bottomLargeTitleTextStyle,
|
|
child: bottomLargeTitle.child
|
|
)
|
|
)
|
|
);
|
|
}
|
|
|
|
return null;
|
|
}
|
|
}
|
|
|
|
public Widget bottomTrailing {
|
|
get {
|
|
var bottomTrailing = (KeyedSubtree) bottomComponents.trailingKey.currentWidget;
|
|
|
|
if (bottomTrailing == null) {
|
|
return null;
|
|
}
|
|
|
|
return Positioned.fromRelativeRect(
|
|
rect: positionInTransitionBox(key: bottomComponents.trailingKey, from: bottomNavBarBox),
|
|
child: new FadeTransition(
|
|
opacity: fadeOutBy(0.6f),
|
|
child: bottomTrailing.child
|
|
)
|
|
);
|
|
}
|
|
}
|
|
|
|
public Widget topLeading {
|
|
get {
|
|
var topLeading = (KeyedSubtree) topComponents.leadingKey.currentWidget;
|
|
|
|
if (topLeading == null) {
|
|
return null;
|
|
}
|
|
|
|
return Positioned.fromRelativeRect(
|
|
rect: positionInTransitionBox(key: topComponents.leadingKey, from: topNavBarBox),
|
|
child: new FadeTransition(
|
|
opacity: fadeInFrom(0.6f),
|
|
child: topLeading.child
|
|
)
|
|
);
|
|
}
|
|
}
|
|
|
|
public Widget topBackChevron {
|
|
get {
|
|
var topBackChevron = (KeyedSubtree) topComponents.backChevronKey.currentWidget;
|
|
var bottomBackChevron = (KeyedSubtree) bottomComponents.backChevronKey.currentWidget;
|
|
|
|
if (topBackChevron == null) {
|
|
return null;
|
|
}
|
|
|
|
var to = positionInTransitionBox(key: topComponents.backChevronKey, from: topNavBarBox);
|
|
var from = to;
|
|
|
|
if (bottomBackChevron == null) {
|
|
var topBackChevronBox =
|
|
(RenderBox) topComponents.backChevronKey.currentContext.findRenderObject();
|
|
from = to.shift(
|
|
new Offset(forwardDirection * topBackChevronBox.size.width * 2.0f,
|
|
0.0f
|
|
)
|
|
);
|
|
}
|
|
|
|
var positionTween = new RelativeRectTween(
|
|
begin: from,
|
|
end: to
|
|
);
|
|
|
|
return new PositionedTransition(
|
|
animation.drive(child: positionTween),
|
|
child: new FadeTransition(
|
|
opacity: fadeInFrom(bottomBackChevron == null ? 0.7f : 0.4f),
|
|
child: new DefaultTextStyle(
|
|
style: topBackButtonTextStyle,
|
|
child: topBackChevron.child
|
|
)
|
|
)
|
|
);
|
|
}
|
|
}
|
|
|
|
public Widget topBackLabel {
|
|
get {
|
|
var bottomMiddle = (KeyedSubtree) bottomComponents.middleKey.currentWidget;
|
|
var bottomLargeTitle = (KeyedSubtree) bottomComponents.largeTitleKey.currentWidget;
|
|
var topBackLabel = (KeyedSubtree) topComponents.backLabelKey.currentWidget;
|
|
|
|
if (topBackLabel == null) {
|
|
return null;
|
|
}
|
|
|
|
var topBackLabelOpacity =
|
|
topComponents.backLabelKey.currentContext?.findAncestorRenderObjectOfType<RenderAnimatedOpacity>();
|
|
|
|
Animation<float> midClickOpacity = null;
|
|
if (topBackLabelOpacity != null && topBackLabelOpacity.opacity.value < 1.0f) {
|
|
midClickOpacity = animation.drive(new FloatTween(
|
|
0.0f,
|
|
end: topBackLabelOpacity.opacity.value
|
|
));
|
|
}
|
|
|
|
if (bottomLargeTitle != null &&
|
|
topBackLabel != null && bottomLargeExpanded.Value) {
|
|
return new PositionedTransition(
|
|
animation.drive(slideFromLeadingEdge(
|
|
fromKey: bottomComponents.largeTitleKey,
|
|
fromNavBarBox: bottomNavBarBox,
|
|
toKey: topComponents.backLabelKey,
|
|
toNavBarBox: topNavBarBox
|
|
)),
|
|
child: new FadeTransition(
|
|
opacity: midClickOpacity ?? fadeInFrom(0.4f),
|
|
child: new DefaultTextStyleTransition(
|
|
animation.drive(new TextStyleTween(
|
|
begin: bottomLargeTitleTextStyle,
|
|
end: topBackButtonTextStyle
|
|
)),
|
|
maxLines: 1,
|
|
overflow: TextOverflow.ellipsis,
|
|
child: topBackLabel.child
|
|
)
|
|
)
|
|
);
|
|
}
|
|
|
|
if (bottomMiddle != null && topBackLabel != null) {
|
|
return new PositionedTransition(
|
|
animation.drive(slideFromLeadingEdge(
|
|
fromKey: bottomComponents.middleKey,
|
|
fromNavBarBox: bottomNavBarBox,
|
|
toKey: topComponents.backLabelKey,
|
|
toNavBarBox: topNavBarBox
|
|
)),
|
|
child: new FadeTransition(
|
|
opacity: midClickOpacity ?? fadeInFrom(0.3f),
|
|
child: new DefaultTextStyleTransition(
|
|
animation.drive(new TextStyleTween(
|
|
begin: bottomTitleTextStyle,
|
|
end: topBackButtonTextStyle
|
|
)),
|
|
child: topBackLabel.child
|
|
)
|
|
)
|
|
);
|
|
}
|
|
|
|
return null;
|
|
}
|
|
}
|
|
|
|
public Widget topMiddle {
|
|
get {
|
|
var topMiddle = (KeyedSubtree) topComponents.middleKey.currentWidget;
|
|
|
|
if (topMiddle == null) {
|
|
return null;
|
|
}
|
|
|
|
if (topHasUserMiddle != true && topLargeExpanded == true) {
|
|
return null;
|
|
}
|
|
|
|
var to = positionInTransitionBox(key: topComponents.middleKey, from: topNavBarBox);
|
|
|
|
var positionTween = new RelativeRectTween(
|
|
to.shift(
|
|
new Offset(forwardDirection * topNavBarBox.size.width / 2.0f,
|
|
0.0f
|
|
)
|
|
),
|
|
end: to
|
|
);
|
|
|
|
return new PositionedTransition(
|
|
animation.drive(child: positionTween),
|
|
child: new FadeTransition(
|
|
opacity: fadeInFrom(0.25f),
|
|
child: new DefaultTextStyle(
|
|
style: topTitleTextStyle,
|
|
child: topMiddle.child
|
|
)
|
|
)
|
|
);
|
|
}
|
|
}
|
|
|
|
public Widget topTrailing {
|
|
get {
|
|
var topTrailing = (KeyedSubtree) topComponents.trailingKey.currentWidget;
|
|
|
|
if (topTrailing == null) {
|
|
return null;
|
|
}
|
|
|
|
return Positioned.fromRelativeRect(
|
|
rect: positionInTransitionBox(key: topComponents.trailingKey, from: topNavBarBox),
|
|
child: new FadeTransition(
|
|
opacity: fadeInFrom(0.4f),
|
|
child: topTrailing.child
|
|
)
|
|
);
|
|
}
|
|
}
|
|
|
|
public Widget topLargeTitle {
|
|
get {
|
|
var topLargeTitle = (KeyedSubtree) topComponents.largeTitleKey.currentWidget;
|
|
|
|
if (topLargeTitle == null || topLargeExpanded != true) {
|
|
return null;
|
|
}
|
|
|
|
var to = positionInTransitionBox(key: topComponents.largeTitleKey, from: topNavBarBox);
|
|
|
|
var positionTween = new RelativeRectTween(
|
|
to.shift(
|
|
new Offset(forwardDirection * topNavBarBox.size.width,
|
|
0.0f
|
|
)
|
|
),
|
|
end: to
|
|
);
|
|
|
|
return new PositionedTransition(
|
|
animation.drive(child: positionTween),
|
|
child: new FadeTransition(
|
|
opacity: fadeInFrom(0.3f),
|
|
child: new DefaultTextStyle(
|
|
style: topLargeTitleTextStyle,
|
|
maxLines: 1,
|
|
overflow: TextOverflow.ellipsis,
|
|
child: topLargeTitle.child
|
|
)
|
|
)
|
|
);
|
|
}
|
|
}
|
|
|
|
public RelativeRect positionInTransitionBox(
|
|
GlobalKey key = null,
|
|
RenderBox from = null
|
|
) {
|
|
var componentBox = key.currentContext.findRenderObject() as RenderBox;
|
|
D.assert(result: componentBox.attached);
|
|
return RelativeRect.fromRect(
|
|
componentBox.localToGlobal(point: Offset.zero, ancestor: from) & componentBox.size,
|
|
container: transitionBox
|
|
);
|
|
}
|
|
|
|
public RelativeRectTween slideFromLeadingEdge(
|
|
GlobalKey fromKey = null,
|
|
RenderBox fromNavBarBox = null,
|
|
GlobalKey toKey = null,
|
|
RenderBox toNavBarBox = null
|
|
) {
|
|
var fromRect = positionInTransitionBox(key: fromKey, from: fromNavBarBox);
|
|
var fromBox = fromKey.currentContext.findRenderObject() as RenderBox;
|
|
var toBox = toKey.currentContext.findRenderObject() as RenderBox;
|
|
var toRect =
|
|
toBox.localToGlobal(
|
|
point: Offset.zero,
|
|
ancestor: toNavBarBox
|
|
).translate(
|
|
0.0f,
|
|
-fromBox.size.height / 2 + toBox.size.height / 2
|
|
) & fromBox.size; // Keep the from render object"s size.
|
|
|
|
if (forwardDirection < 0) {
|
|
toRect = toRect.translate(-fromBox.size.width + toBox.size.width, 0.0f);
|
|
}
|
|
|
|
return new RelativeRectTween(
|
|
begin: fromRect,
|
|
RelativeRect.fromRect(rect: toRect, container: transitionBox)
|
|
);
|
|
}
|
|
|
|
public Animation<float> fadeInFrom(float t, Curve curve = null) {
|
|
return animation.drive(fadeIn.chain(
|
|
new CurveTween(new Interval(begin: t, 1.0f, curve ?? Curves.easeIn))
|
|
));
|
|
}
|
|
|
|
public Animation<float> fadeOutBy(float t, Curve curve = null) {
|
|
return animation.drive(fadeOut.chain(
|
|
new CurveTween(new Interval(0.0f, end: t, curve ?? Curves.easeOut))
|
|
));
|
|
}
|
|
}
|
|
}
|