【问题标题】:Xamarin.Forms: how to disable arrow to hamburger transition animation on Android?Xamarin.Forms:如何在 Android 上禁用箭头到汉堡包过渡动画?
【发布时间】:2018-04-18 16:23:12
【问题描述】:

我使用 Xamarin.Forms 创建了一个基本示例应用程序,我尝试在其中复制 Twitter 或 Instagram 等 UI:

  • 一个 MasterDetail 页面作为主页,其中默认的“汉堡包”图标 替换为自定义图标
  • “底部”TabbedPage 作为 MasterDetail 页面的详细信息
  • 允许在 TabbedPage 中实现 top Tabs 的控件
  • ContentPagesNavigationPage 中为每个 TabbedPage 的标签

所以,我的应用的页面架构如下所示:

|-- MasterDetailPage
..|--标签页
....|-- 导航页
......|-- 内容页面

为了实现这一点,我使用了:

  • Naxam BottomTabedPage,在 Android 上实现“底部”TabbedPage(如 BottomNavigationView)
  • Syncfusion SfTabView 控件实现“顶部”选项卡
  • 一个自定义渲染器,用于管理自定义图标作为主级别的“汉堡”图标和子级别的“箭头”图标的使用

这个渲染器看起来像这样:

public class CustomNavigationPageRenderer : MasterDetailPageRenderer
{
    protected override void OnLayout(bool changed, int l, int t, int r, int b)
    {
        base.OnLayout(changed, l, t, r, b);
        var toolbar = FindViewById<Android.Support.V7.Widget.Toolbar>(Resource.Id.toolbar);

        for (var i = 0; i < toolbar.ChildCount; i++)
        {
            var imageButton = toolbar.GetChildAt(i) as ImageButton;
            var drawerArrow = imageButton?.Drawable as DrawerArrowDrawable;
            if (drawerArrow == null)
                continue;

            bool displayBack = false;
            var app = Xamarin.Forms.Application.Current;
            //var navPage = ((app.MainPage.Navigation.ModalStack[0] as MasterDetailPage).Detail as NavigationPage);
            var detailPage = (app.MainPage as MasterDetailPage).Detail;
            if (detailPage.GetType() == typeof(BottomTp.Views.NaxamMainPage))
            {
                var tabPage = detailPage as BottomTabbedPage;
                var curPage = tabPage.CurrentPage;
                var navPageLevel = curPage.Navigation.NavigationStack.Count;
                if (navPageLevel > 1)
                    displayBack = true;
            }

            if (!displayBack)
                ChangeIcon(imageButton, Resource.Drawable.icon);
        }
    }

    private void ChangeIcon(ImageButton imageButton, int id)
    {
        if (Android.OS.Build.VERSION.SdkInt >= Android.OS.BuildVersionCodes.Lollipop)
            imageButton.SetImageDrawable(Context.GetDrawable(id));
        imageButton.SetImageResource(id);
    }
}

这很好用,但是当我“返回”到主页时,还有一个问题

  • 有默认的从箭头到“汉堡”图标的过渡动画
  • 此后,“汉堡”图标被我的自定义图标取代

这里有一个简短的动画来说明这个问题:

有没有办法关闭这个动画?我该如何解决这个问题?

【问题讨论】:

  • 请参考thispublic class MasterDetailPageRenderer : DrawerLayout,这样你就可以使用DrawerLayout来实现你的目标。
  • 嗨@JoeLv-MSFT 我已经看到了这个链接,但我没有看到如何在我的渲染器中使用 DrawerLayout。
  • 嗨,here,在OnElementChanged 方法中,添加DrawerClosedDrawerOpened 事件。能给我演示一下吗?
  • 但是MasterDetailPageRenderer中没有OnElementChanged方法。
  • 请查看here

标签: android animation xamarin.forms hamburger-menu


【解决方案1】:

您需要使用自定义NavigationPage。这是我的demo,基于你的。

实现它的三个步骤:

1)添加一个名为CustomControl的文件夹,创建一个类MyNavigationPage继承自NavigationPage

namespace BottomTp.CustomControl
{
    public class MyNavigationPage : NavigationPage
    {
        public MyNavigationPage() { }
        public MyNavigationPage(Page root) : base(root) { }
    }
}

2) 更改NaxamMainPage.xaml 文件中的标签,

在您的naxam:BottomTabbedPage 标签中添加xmlns:controls="clr-namespace:BottomTp.CustomControl;assembly=BottomTp",然后将您的NavigationPage 标签更改为controls:MyNavigationPage

<controls:MyNavigationPage Title="Browse" Icon="md-view-list"
                x:Name="NP1">
    <x:Arguments>
        <views:SfItemsPage   />
    </x:Arguments>
</controls:MyNavigationPage>
<controls:MyNavigationPage Title="About" Icon="md-help"
                x:Name="NP2">
    <x:Arguments>
        <views:AboutPage />
    </x:Arguments>
</controls:MyNavigationPage>

3) 在Android平台上创建自定义渲染MyCustomNavigationPageRenderer

[assembly: ExportRenderer(typeof(MyNavigationPage), typeof(MyCustomNavigationPageRenderer))]
namespace BottomTp.Droid.Renderers
{
    class MyCustomNavigationPageRenderer : NavigationPageRenderer
    {
        public MyCustomNavigationPageRenderer(Context c) : base(c)
        { }

        protected override Task<bool> OnPopToRootAsync(Page page, bool animated)
        {
            return base.OnPopToRootAsync(page, false);
        }

        protected override Task<bool> OnPopViewAsync(Page page, bool animated)
        {
            return base.OnPopViewAsync(page, false);
        }

        protected override Task<bool> OnPushAsync(Page view, bool animated)
        {
            return base.OnPushAsync(view, false);
        }

    }
}

【讨论】:

    猜你喜欢
    • 2015-03-02
    • 2017-05-10
    • 2015-01-22
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2017-06-11
    • 1970-01-01
    相关资源
    最近更新 更多