2017-04-13 18:11:50 +03:00
|
|
|
using System;
|
2016-05-10 20:15:47 +03:00
|
|
|
using Android.App;
|
|
|
|
using Android.Content;
|
2016-10-06 20:17:51 +03:00
|
|
|
using Android.OS;
|
2016-05-10 20:15:47 +03:00
|
|
|
using Fragment = Android.Support.V4.App.Fragment;
|
|
|
|
using FragmentManager = Android.Support.V4.App.FragmentManager;
|
|
|
|
using FragmentTransaction = Android.Support.V4.App.FragmentTransaction;
|
|
|
|
|
|
|
|
namespace Xamarin.Forms.Platform.Android.AppCompat
|
|
|
|
{
|
|
|
|
internal class MasterDetailContainer : Xamarin.Forms.Platform.Android.MasterDetailContainer, IManageFragments
|
|
|
|
{
|
|
|
|
PageContainer _pageContainer;
|
|
|
|
FragmentManager _fragmentManager;
|
|
|
|
readonly bool _isMaster;
|
2016-10-04 20:47:02 +03:00
|
|
|
MasterDetailPage _parent;
|
2016-06-27 18:20:47 +03:00
|
|
|
Fragment _currentFragment;
|
2016-10-04 20:47:02 +03:00
|
|
|
bool _disposed;
|
2016-05-10 20:15:47 +03:00
|
|
|
|
|
|
|
public MasterDetailContainer(MasterDetailPage parent, bool isMaster, Context context) : base(parent, isMaster, context)
|
|
|
|
{
|
2017-06-23 01:33:03 +03:00
|
|
|
Id = Platform.GenerateViewId();
|
2016-05-10 20:15:47 +03:00
|
|
|
_parent = parent;
|
|
|
|
_isMaster = isMaster;
|
|
|
|
}
|
|
|
|
|
2018-07-13 09:00:25 +03:00
|
|
|
public bool MarkedForDispose { get; internal set; } = false;
|
|
|
|
|
2016-05-10 20:15:47 +03:00
|
|
|
FragmentManager FragmentManager => _fragmentManager ?? (_fragmentManager = ((FormsAppCompatActivity)Context).SupportFragmentManager);
|
|
|
|
|
|
|
|
protected override void OnLayout(bool changed, int l, int t, int r, int b)
|
|
|
|
{
|
|
|
|
base.OnLayout(changed, l, t, r, b);
|
|
|
|
|
|
|
|
// If we're using a PageContainer (i.e., we've wrapped our contents in a Fragment),
|
|
|
|
// Make sure that it gets laid out
|
|
|
|
if (_pageContainer != null)
|
|
|
|
{
|
|
|
|
if (_isMaster)
|
|
|
|
{
|
2016-05-10 20:37:16 +03:00
|
|
|
var controller = (IMasterDetailPageController)_parent;
|
|
|
|
var width = (int)Context.ToPixels(controller.MasterBounds.Width);
|
|
|
|
// When the base class computes the size of the Master container, it starts at the top of the
|
|
|
|
// screen and adds padding (_parent.MasterBounds.Top) to leave room for the status bar
|
|
|
|
// When this container is laid out, it's already starting from the adjusted y value of the parent,
|
|
|
|
// so we subtract _parent.MasterBounds.Top from our starting point (to get 0) and add it to the
|
|
|
|
// bottom (so the master page stretches to the bottom of the screen)
|
|
|
|
var height = (int)Context.ToPixels(controller.MasterBounds.Height + controller.MasterBounds.Top);
|
2016-05-10 20:15:47 +03:00
|
|
|
_pageContainer.Layout(0, 0, width, height);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
_pageContainer.Layout(l, t, r, b);
|
|
|
|
}
|
|
|
|
|
|
|
|
_pageContainer.Child.UpdateLayout();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
protected override void AddChildView(VisualElement childView)
|
|
|
|
{
|
|
|
|
_pageContainer = null;
|
|
|
|
|
|
|
|
Page page = childView as NavigationPage ?? (Page)(childView as TabbedPage);
|
|
|
|
|
|
|
|
if (page == null)
|
|
|
|
{
|
2016-06-27 18:20:47 +03:00
|
|
|
// The thing we're adding is not a NavigationPage or TabbedPage, so we can just use the old AddChildView
|
|
|
|
|
|
|
|
if (_currentFragment != null)
|
|
|
|
{
|
2018-07-13 09:00:25 +03:00
|
|
|
if (!_parent.IsAttachedToRoot())
|
|
|
|
return;
|
|
|
|
|
2016-06-27 18:20:47 +03:00
|
|
|
// But first, if the previous occupant of this container was a fragment, we need to remove it properly
|
2018-07-13 09:00:25 +03:00
|
|
|
FragmentTransaction transaction = FragmentManager.BeginTransactionEx();
|
|
|
|
transaction.RemoveEx(_currentFragment);
|
|
|
|
transaction.SetTransitionEx((int)FragmentTransit.None);
|
2016-11-16 23:43:33 +03:00
|
|
|
|
|
|
|
// This is a removal of a fragment that's not going on the back stack; there's no reason to care
|
|
|
|
// whether its state gets successfully saved, since we'll never restore it. Ergo, CommitAllowingStateLoss
|
2018-07-13 09:00:25 +03:00
|
|
|
transaction.CommitAllowingStateLossEx();
|
2016-06-27 18:20:47 +03:00
|
|
|
|
|
|
|
_currentFragment = null;
|
|
|
|
}
|
2018-07-13 09:00:25 +03:00
|
|
|
|
2016-05-10 20:15:47 +03:00
|
|
|
base.AddChildView(childView);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-07-13 09:00:25 +03:00
|
|
|
if (!_parent.IsAttachedToRoot())
|
|
|
|
return;
|
|
|
|
|
2016-05-10 20:15:47 +03:00
|
|
|
// The renderers for NavigationPage and TabbedPage both host fragments, so they need to be wrapped in a
|
|
|
|
// FragmentContainer in order to get isolated fragment management
|
|
|
|
Fragment fragment = FragmentContainer.CreateInstance(page);
|
2018-07-13 09:00:25 +03:00
|
|
|
|
2016-05-10 20:15:47 +03:00
|
|
|
var fc = fragment as FragmentContainer;
|
2016-06-27 18:20:47 +03:00
|
|
|
|
2016-05-10 20:15:47 +03:00
|
|
|
fc?.SetOnCreateCallback(pc =>
|
|
|
|
{
|
|
|
|
_pageContainer = pc;
|
|
|
|
SetDefaultBackgroundColor(pc.Child);
|
|
|
|
});
|
|
|
|
|
2018-07-13 09:00:25 +03:00
|
|
|
FragmentTransaction transaction = FragmentManager.BeginTransactionEx();
|
2016-06-27 18:20:47 +03:00
|
|
|
|
|
|
|
if (_currentFragment != null)
|
2018-07-13 09:00:25 +03:00
|
|
|
transaction.RemoveEx(_currentFragment);
|
2016-06-27 18:20:47 +03:00
|
|
|
|
2018-07-13 09:00:25 +03:00
|
|
|
transaction.AddEx(Id, fragment);
|
|
|
|
transaction.SetTransitionEx((int)FragmentTransit.None);
|
2016-11-16 23:43:33 +03:00
|
|
|
|
|
|
|
// We don't currently support fragment restoration
|
|
|
|
// So we don't need to worry about loss of this fragment's state
|
2018-07-13 09:00:25 +03:00
|
|
|
transaction.CommitAllowingStateLossEx();
|
2016-06-27 18:20:47 +03:00
|
|
|
|
|
|
|
_currentFragment = fragment;
|
2016-05-10 20:15:47 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-10-04 20:47:02 +03:00
|
|
|
protected override void Dispose(bool disposing)
|
|
|
|
{
|
|
|
|
if (_disposed)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
_disposed = true;
|
|
|
|
|
|
|
|
if (disposing)
|
|
|
|
{
|
2017-10-11 12:12:43 +03:00
|
|
|
if (_currentFragment != null && !FragmentManager.IsDestroyed)
|
2016-10-04 20:47:02 +03:00
|
|
|
{
|
2018-07-13 09:00:25 +03:00
|
|
|
FragmentTransaction transaction = FragmentManager.BeginTransactionEx();
|
|
|
|
transaction.RemoveEx(_currentFragment);
|
|
|
|
transaction.SetTransitionEx((int)FragmentTransit.None);
|
|
|
|
transaction.CommitAllowingStateLossEx();
|
|
|
|
FragmentManager.ExecutePendingTransactionsEx();
|
2016-10-04 20:47:02 +03:00
|
|
|
|
|
|
|
_currentFragment = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
_parent = null;
|
|
|
|
_pageContainer = null;
|
|
|
|
_fragmentManager = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
base.Dispose(disposing);
|
|
|
|
}
|
|
|
|
|
2016-05-10 20:15:47 +03:00
|
|
|
public void SetFragmentManager(FragmentManager fragmentManager)
|
|
|
|
{
|
|
|
|
if (_fragmentManager == null)
|
|
|
|
_fragmentManager = fragmentManager;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|