您最多选择25个主题
主题必须以中文或者字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符
339 行
12 KiB
339 行
12 KiB
using System.Collections.Generic;
|
|
using Unity.UIWidgets.foundation;
|
|
using Unity.UIWidgets.painting;
|
|
using Unity.UIWidgets.rendering;
|
|
using Unity.UIWidgets.ui;
|
|
using Unity.UIWidgets.widgets;
|
|
|
|
namespace Unity.UIWidgets.cupertino {
|
|
public class CupertinoTabController : ChangeNotifier {
|
|
|
|
public CupertinoTabController(int? initialIndex = 0) {
|
|
D.assert(initialIndex != null);
|
|
D.assert(initialIndex >= 0);
|
|
_index = initialIndex.Value;
|
|
}
|
|
|
|
public bool _isDisposed = false;
|
|
|
|
public int index {
|
|
get { return _index; }
|
|
set {
|
|
D.assert(value >= 0);
|
|
if (_index == value) {
|
|
return;
|
|
}
|
|
_index = value;
|
|
notifyListeners();
|
|
}
|
|
}
|
|
int _index;
|
|
public override void dispose() {
|
|
base.dispose();
|
|
_isDisposed = true;
|
|
}
|
|
}
|
|
|
|
public class CupertinoTabScaffold : StatefulWidget {
|
|
public CupertinoTabScaffold(
|
|
Key key = null,
|
|
CupertinoTabBar tabBar = null,
|
|
IndexedWidgetBuilder tabBuilder = null,
|
|
CupertinoTabController controller = null,
|
|
Color backgroundColor = null,
|
|
bool resizeToAvoidBottomInset = true
|
|
) : base(key: key) {
|
|
D.assert(tabBar != null);
|
|
D.assert(tabBuilder != null);
|
|
D.assert(
|
|
controller == null || controller.index < tabBar.items.Count, () =>
|
|
$"The CupertinoTabController's current index {controller.index} is " +
|
|
$"out2 of bounds for the tab bar with {tabBar.items.Count} tabs"
|
|
);
|
|
this.tabBar = tabBar;
|
|
this.controller = controller;
|
|
this.tabBuilder = tabBuilder;
|
|
this.backgroundColor = backgroundColor;
|
|
this.resizeToAvoidBottomInset = resizeToAvoidBottomInset;
|
|
}
|
|
|
|
public readonly CupertinoTabController controller;
|
|
public readonly CupertinoTabBar tabBar;
|
|
|
|
public readonly IndexedWidgetBuilder tabBuilder;
|
|
|
|
public readonly Color backgroundColor;
|
|
|
|
public readonly bool resizeToAvoidBottomInset;
|
|
|
|
public override State createState() {
|
|
return new _CupertinoTabScaffoldState();
|
|
}
|
|
}
|
|
|
|
class _CupertinoTabScaffoldState : State<CupertinoTabScaffold> {
|
|
int _currentPage;
|
|
CupertinoTabController _controller;
|
|
|
|
public override void initState() {
|
|
base.initState();
|
|
_updateTabController();
|
|
}
|
|
void _updateTabController( bool shouldDisposeOldController = false ) {
|
|
CupertinoTabController newController =
|
|
// User provided a new controller, update `_controller` with it.
|
|
widget.controller
|
|
?? new CupertinoTabController(initialIndex: widget.tabBar.currentIndex);
|
|
|
|
if (newController == _controller) {
|
|
return;
|
|
}
|
|
|
|
if (shouldDisposeOldController) {
|
|
_controller?.dispose();
|
|
} else if (_controller?._isDisposed == false) {
|
|
_controller.removeListener(_onCurrentIndexChange);
|
|
}
|
|
|
|
newController.addListener(_onCurrentIndexChange);
|
|
_controller = newController;
|
|
}
|
|
void _onCurrentIndexChange() {
|
|
D.assert(
|
|
_controller.index >= 0 && _controller.index < widget.tabBar.items.Count,()=>
|
|
$"The {GetType()}'s current index {_controller.index} is " +
|
|
$"out of bounds for the tab bar with {widget.tabBar.items.Count} tabs"
|
|
);
|
|
setState(()=> {});
|
|
}
|
|
|
|
public override void didUpdateWidget(StatefulWidget _oldWidget) {
|
|
CupertinoTabScaffold oldWidget = _oldWidget as CupertinoTabScaffold;
|
|
base.didUpdateWidget(oldWidget);
|
|
if (widget.controller != oldWidget.controller) {
|
|
_updateTabController(shouldDisposeOldController: oldWidget.controller == null);
|
|
} else if (_controller.index >= widget.tabBar.items.Count) {
|
|
_controller.index = widget.tabBar.items.Count - 1;
|
|
}
|
|
}
|
|
public override Widget build(BuildContext context) {
|
|
List<Widget> stacked = new List<Widget> { };
|
|
|
|
MediaQueryData existingMediaQuery = MediaQuery.of(context);
|
|
MediaQueryData newMediaQuery = MediaQuery.of(context);
|
|
|
|
Widget content = new _TabSwitchingView(
|
|
currentTabIndex: _controller.index,
|
|
tabNumber: widget.tabBar.items.Count,
|
|
tabBuilder: widget.tabBuilder
|
|
);
|
|
EdgeInsets contentPadding = EdgeInsets.zero;
|
|
|
|
if (widget.resizeToAvoidBottomInset) {
|
|
newMediaQuery = newMediaQuery.removeViewInsets(removeBottom: true);
|
|
contentPadding = EdgeInsets.only(bottom: existingMediaQuery.viewInsets.bottom);
|
|
}
|
|
|
|
if (widget.tabBar != null &&
|
|
(!widget.resizeToAvoidBottomInset ||
|
|
widget.tabBar.preferredSize.height > existingMediaQuery.viewInsets.bottom)) {
|
|
float bottomPadding = widget.tabBar.preferredSize.height + existingMediaQuery.padding.bottom;
|
|
|
|
if (widget.tabBar.opaque(context)) {
|
|
contentPadding = EdgeInsets.only(bottom: bottomPadding);
|
|
newMediaQuery = newMediaQuery.removePadding(removeBottom: true);
|
|
}
|
|
else {
|
|
newMediaQuery = newMediaQuery.copyWith(
|
|
padding: newMediaQuery.padding.copyWith(
|
|
bottom: bottomPadding
|
|
)
|
|
);
|
|
}
|
|
}
|
|
|
|
content = new MediaQuery(
|
|
data: newMediaQuery,
|
|
child: new Padding(
|
|
padding: contentPadding,
|
|
child: content
|
|
)
|
|
);
|
|
stacked.Add(content);
|
|
stacked.Add(
|
|
new MediaQuery(
|
|
data: existingMediaQuery.copyWith(textScaleFactor: 1),
|
|
child: new Align(
|
|
alignment: Alignment.bottomCenter,
|
|
child: widget.tabBar.copyWith(
|
|
currentIndex: _controller.index,
|
|
onTap: (int newIndex) =>{
|
|
_controller.index = newIndex;
|
|
if (widget.tabBar.onTap != null)
|
|
widget.tabBar.onTap(newIndex);
|
|
}
|
|
)
|
|
)
|
|
)
|
|
);
|
|
|
|
return new DecoratedBox(
|
|
decoration: new BoxDecoration(
|
|
color: CupertinoDynamicColor.resolve(widget.backgroundColor, context)
|
|
?? CupertinoTheme.of(context).scaffoldBackgroundColor
|
|
),
|
|
child: new Stack(
|
|
children: stacked
|
|
)
|
|
);
|
|
}
|
|
|
|
public override void dispose() {
|
|
|
|
if (widget.controller == null) {
|
|
_controller?.dispose();
|
|
} else if (_controller?._isDisposed == false) {
|
|
_controller.removeListener(_onCurrentIndexChange);
|
|
}
|
|
|
|
base.dispose();
|
|
}
|
|
|
|
}
|
|
|
|
class _TabSwitchingView : StatefulWidget {
|
|
public _TabSwitchingView(
|
|
int currentTabIndex,
|
|
int tabNumber,
|
|
IndexedWidgetBuilder tabBuilder
|
|
) {
|
|
D.assert(tabNumber > 0);
|
|
D.assert(tabBuilder != null);
|
|
this.currentTabIndex = currentTabIndex;
|
|
this.tabNumber = tabNumber;
|
|
this.tabBuilder = tabBuilder;
|
|
}
|
|
|
|
public readonly int currentTabIndex;
|
|
public readonly int tabNumber;
|
|
public readonly IndexedWidgetBuilder tabBuilder;
|
|
|
|
public override State createState() {
|
|
return new _TabSwitchingViewState();
|
|
}
|
|
}
|
|
|
|
class _TabSwitchingViewState : State<_TabSwitchingView> {
|
|
|
|
public readonly List<bool> shouldBuildTab = new List<bool>();
|
|
public readonly List<FocusScopeNode> tabFocusNodes = new List<FocusScopeNode>();
|
|
public readonly List<FocusScopeNode> discardedNodes = new List<FocusScopeNode>();
|
|
|
|
|
|
public override void initState() {
|
|
base.initState();
|
|
List<bool> tabBool = new List<bool>();
|
|
for (int i = 0; i < widget.tabNumber; i++) {
|
|
tabBool.Add(false);
|
|
}
|
|
foreach (var tab in tabBool) {
|
|
shouldBuildTab.Add(tab);
|
|
}
|
|
}
|
|
|
|
public override void didChangeDependencies() {
|
|
base.didChangeDependencies();
|
|
_focusActiveTab();
|
|
}
|
|
|
|
public override void didUpdateWidget(StatefulWidget _oldWidget) {
|
|
_TabSwitchingView oldWidget = _oldWidget as _TabSwitchingView;
|
|
base.didUpdateWidget(oldWidget);
|
|
int lengthDiff = widget.tabNumber - shouldBuildTab.Count;
|
|
if (lengthDiff > 0) {
|
|
List<bool> tabBool = new List<bool>();
|
|
for (int i = 0; i < lengthDiff; i++) {
|
|
tabBool.Add(false);
|
|
}
|
|
foreach (var tab in tabBool) {
|
|
shouldBuildTab.Add(tab);
|
|
}
|
|
}
|
|
else if (lengthDiff < 0) {
|
|
for (int i = widget.tabNumber; i < shouldBuildTab.Count; i++) {
|
|
shouldBuildTab.RemoveAt(i);
|
|
}
|
|
|
|
}
|
|
_focusActiveTab();
|
|
}
|
|
|
|
void _focusActiveTab() {
|
|
if (tabFocusNodes.Count!= widget.tabNumber) {
|
|
if (tabFocusNodes.Count> widget.tabNumber) {
|
|
for (int i = widget.tabNumber; i < tabFocusNodes.Count; i++) {
|
|
discardedNodes.Add(tabFocusNodes[i]);
|
|
}
|
|
for (int i = widget.tabNumber; i < tabFocusNodes.Count; i++ ){
|
|
tabFocusNodes.RemoveAt(i);
|
|
}
|
|
}
|
|
|
|
else {
|
|
List<FocusScopeNode> scopeNodes = new List<FocusScopeNode>();
|
|
var length = widget.tabNumber - tabFocusNodes.Count;
|
|
for (int i = 0; i < length; i++) {
|
|
scopeNodes.Add(new FocusScopeNode(debugLabel: $"CupertinoTabScaffold Tab {i + tabFocusNodes.Count}")
|
|
);
|
|
}
|
|
tabFocusNodes.AddRange(scopeNodes);
|
|
|
|
}
|
|
|
|
}
|
|
FocusScope.of(context).setFirstFocus(tabFocusNodes[widget.currentTabIndex]);
|
|
}
|
|
|
|
public override void dispose() {
|
|
foreach(FocusScopeNode focusScopeNode in tabFocusNodes) {
|
|
focusScopeNode.dispose();
|
|
}
|
|
foreach( FocusScopeNode focusScopeNode in discardedNodes) {
|
|
focusScopeNode.dispose();
|
|
}
|
|
base.dispose();
|
|
}
|
|
|
|
public override Widget build(BuildContext context) {
|
|
|
|
List<Widget> stages = new List<Widget>();
|
|
int count = widget.tabNumber;
|
|
for (int i = 0; i < count; i++) {
|
|
bool active = i == widget.currentTabIndex;
|
|
shouldBuildTab[i] = active || shouldBuildTab[i];
|
|
int temp = i;
|
|
stages.Add(new Offstage(
|
|
offstage: !active,
|
|
child: new TickerMode(
|
|
enabled: active,
|
|
child: new FocusScope(
|
|
node: tabFocusNodes[i],
|
|
child: new Builder(
|
|
builder: (BuildContext context1) => {
|
|
return shouldBuildTab[temp] ? widget.tabBuilder(context1, temp) : new Container();
|
|
}
|
|
)
|
|
)
|
|
)
|
|
));
|
|
|
|
}
|
|
|
|
return new Stack(
|
|
fit: StackFit.expand,
|
|
children:stages
|
|
|
|
);
|
|
}
|
|
}
|
|
}
|