using System; using System.Collections.Generic; using System.Linq; using Unity.UIWidgets.animation; using Unity.UIWidgets.foundation; using Unity.UIWidgets.gestures; using Unity.UIWidgets.painting; using Unity.UIWidgets.rendering; using Unity.UIWidgets.ui; using Unity.UIWidgets.widgets; using UnityEngine; using Transform = Unity.UIWidgets.widgets.Transform; namespace Unity.UIWidgets.material { class UserAccountsDrawerHeaderUtils { public const float _kAccountDetailsHeight = 56.0f; } class _AccountPictures : StatelessWidget { public _AccountPictures( Key key = null, Widget currentAccountPicture = null, List otherAccountsPictures = null ) : base(key: key) { this.currentAccountPicture = currentAccountPicture; this.otherAccountsPictures = otherAccountsPictures; } public readonly Widget currentAccountPicture; public readonly List otherAccountsPictures; public override Widget build(BuildContext context) { return new Stack( children: new List { new Positioned( top: 0.0f, right: 0.0f, child: new Row( children: (this.otherAccountsPictures ?? new List { }) .GetRange(0, Math.Min(3, this.otherAccountsPictures?.Count ?? 0)) .Select( (Widget picture) => { return new Padding( padding: EdgeInsets.only(left: 8.0f), child: new Container( padding: EdgeInsets.only(left: 8.0f, bottom: 8.0f), width: 48.0f, height: 48.0f, child: picture ) ); }).ToList() ) ), new Positioned( top: 0.0f, child: new SizedBox( width: 72.0f, height: 72.0f, child: this.currentAccountPicture ) ) } ); } } class _AccountDetails : StatefulWidget { public _AccountDetails( Key key = null, Widget accountName = null, Widget accountEmail = null, VoidCallback onTap = null, bool? isOpen = null ) : base(key: key) { D.assert(accountName != null); D.assert(accountEmail != null); this.accountName = accountName; this.accountEmail = accountEmail; this.onTap = onTap; this.isOpen = isOpen; } public readonly Widget accountName; public readonly Widget accountEmail; public readonly VoidCallback onTap; public readonly bool? isOpen; public override State createState() { return new _AccountDetailsState(); } } class _AccountDetailsState : SingleTickerProviderStateMixin<_AccountDetails> { Animation _animation; AnimationController _controller; public override void initState() { base.initState(); this._controller = new AnimationController( value: this.widget.isOpen == true ? 1.0f : 0.0f, duration: new TimeSpan(0, 0, 0, 0, 200), vsync: this ); this._animation = new CurvedAnimation( parent: this._controller, curve: Curves.fastOutSlowIn, reverseCurve: Curves.fastOutSlowIn.flipped ); this._animation.addListener(() => this.setState(() => { })); } public override void dispose() { this._controller.dispose(); base.dispose(); } public override void didUpdateWidget(StatefulWidget _oldWidget) { base.didUpdateWidget(_oldWidget); _AccountDetails oldWidget = _oldWidget as _AccountDetails; if (this._animation.status == AnimationStatus.dismissed || this._animation.status == AnimationStatus.reverse) { this._controller.forward(); } else { this._controller.reverse(); } } public override Widget build(BuildContext context) { D.assert(WidgetsD.debugCheckHasDirectionality(context)); D.assert(MaterialD.debugCheckHasMaterialLocalizations(context)); D.assert(MaterialD.debugCheckHasMaterialLocalizations(context)); ThemeData theme = Theme.of(context); List children = new List { }; if (this.widget.accountName != null) { Widget accountNameLine = new LayoutId( id: _AccountDetailsLayout.accountName, child: new Padding( padding: EdgeInsets.symmetric(vertical: 2.0f), child: new DefaultTextStyle( style: theme.primaryTextTheme.body2, overflow: TextOverflow.ellipsis, child: this.widget.accountName ) ) ); children.Add(accountNameLine); } if (this.widget.accountEmail != null) { Widget accountEmailLine = new LayoutId( id: _AccountDetailsLayout.accountEmail, child: new Padding( padding: EdgeInsets.symmetric(vertical: 2.0f), child: new DefaultTextStyle( style: theme.primaryTextTheme.body1, overflow: TextOverflow.ellipsis, child: this.widget.accountEmail ) ) ); children.Add(accountEmailLine); } if (this.widget.onTap != null) { MaterialLocalizations localizations = MaterialLocalizations.of(context); Widget dropDownIcon = new LayoutId( id: _AccountDetailsLayout.dropdownIcon, child: new SizedBox( height: UserAccountsDrawerHeaderUtils._kAccountDetailsHeight, width: UserAccountsDrawerHeaderUtils._kAccountDetailsHeight, child: new Center( child: Transform.rotate( degree: this._animation.value * Mathf.PI, child: new Icon( Icons.arrow_drop_down, color: Colors.white ) ) ) ) ); children.Add(dropDownIcon); } Widget accountDetails = new CustomMultiChildLayout( layoutDelegate: new _AccountDetailsLayout(), children: children ); if (this.widget.onTap != null) { accountDetails = new InkWell( onTap: this.widget.onTap == null ? (GestureTapCallback) null : () => { this.widget.onTap(); }, child: accountDetails ); } return new SizedBox( height: UserAccountsDrawerHeaderUtils._kAccountDetailsHeight, child: accountDetails ); } } class _AccountDetailsLayout : MultiChildLayoutDelegate { public _AccountDetailsLayout() { } public const string accountName = "accountName"; public const string accountEmail = "accountEmail"; public const string dropdownIcon = "dropdownIcon"; public override void performLayout(Size size) { Size iconSize = null; if (this.hasChild(dropdownIcon)) { iconSize = this.layoutChild(dropdownIcon, BoxConstraints.loose(size)); this.positionChild(dropdownIcon, this._offsetForIcon(size, iconSize)); } string bottomLine = this.hasChild(accountEmail) ? accountEmail : (this.hasChild(accountName) ? accountName : null); if (bottomLine != null) { Size constraintSize = iconSize == null ? size : size - new Offset(iconSize.width, 0.0f); iconSize = iconSize ?? new Size(UserAccountsDrawerHeaderUtils._kAccountDetailsHeight, UserAccountsDrawerHeaderUtils._kAccountDetailsHeight); Size bottomLineSize = this.layoutChild(bottomLine, BoxConstraints.loose(constraintSize)); Offset bottomLineOffset = this._offsetForBottomLine(size, iconSize, bottomLineSize); this.positionChild(bottomLine, bottomLineOffset); if (bottomLine == accountEmail && this.hasChild(accountName)) { Size nameSize = this.layoutChild(accountName, BoxConstraints.loose(constraintSize)); this.positionChild(accountName, this._offsetForName(size, nameSize, bottomLineOffset)); } } } public override bool shouldRelayout(MultiChildLayoutDelegate oldDelegate) { return true; } Offset _offsetForIcon(Size size, Size iconSize) { return new Offset(size.width - iconSize.width, size.height - iconSize.height); } Offset _offsetForBottomLine(Size size, Size iconSize, Size bottomLineSize) { float y = size.height - 0.5f * iconSize.height - 0.5f * bottomLineSize.height; return new Offset(0.0f, y); } Offset _offsetForName(Size size, Size nameSize, Offset bottomLineOffset) { float y = bottomLineOffset.dy - nameSize.height; return new Offset(0.0f, y); } } public class UserAccountsDrawerHeader : StatefulWidget { public UserAccountsDrawerHeader( Key key = null, Decoration decoration = null, EdgeInsets margin = null, Widget currentAccountPicture = null, List otherAccountsPictures = null, Widget accountName = null, Widget accountEmail = null, VoidCallback onDetailsPressed = null ) : base(key: key) { D.assert(accountName != null); D.assert(accountEmail != null); this.decoration = decoration; this.margin = margin ?? EdgeInsets.only(bottom: 8.0f); this.currentAccountPicture = currentAccountPicture; this.otherAccountsPictures = otherAccountsPictures; this.accountName = accountName; this.accountEmail = accountEmail; this.onDetailsPressed = onDetailsPressed; } public readonly Decoration decoration; public readonly EdgeInsets margin; public readonly Widget currentAccountPicture; public readonly List otherAccountsPictures; public readonly Widget accountName; public readonly Widget accountEmail; public readonly VoidCallback onDetailsPressed; public override State createState() { return new _UserAccountsDrawerHeaderState(); } } class _UserAccountsDrawerHeaderState : State { bool _isOpen = false; void _handleDetailsPressed() { this.setState(() => { this._isOpen = !this._isOpen; }); this.widget.onDetailsPressed(); } public override Widget build(BuildContext context) { D.assert(MaterialD.debugCheckHasMaterial(context)); D.assert(MaterialD.debugCheckHasMaterialLocalizations(context)); return new DrawerHeader( decoration: this.widget.decoration ?? new BoxDecoration( color: Theme.of(context).primaryColor ), margin: this.widget.margin, padding: EdgeInsets.only(top: 16.0f, left: 16.0f), child: new SafeArea( bottom: false, child: new Column( crossAxisAlignment: CrossAxisAlignment.stretch, children: new List { new Expanded( child: new Padding( padding: EdgeInsets.only(right: 16.0f), child: new _AccountPictures( currentAccountPicture: this.widget.currentAccountPicture, otherAccountsPictures: this.widget.otherAccountsPictures ) ) ), new _AccountDetails( accountName: this.widget.accountName, accountEmail: this.widget.accountEmail, isOpen: this._isOpen, onTap: this.widget.onDetailsPressed == null ? (VoidCallback) null : () => { this._handleDetailsPressed(); }) } ) ) ); } } }