【问题标题】:Executing a "Shared Element Transition" from an Activity to a Fragment执行从 Activity 到 Fragment 的“共享元素转换”
【发布时间】:2017-01-17 23:02:09
【问题描述】:

我已经做了一个 github 项目,就是这个问题。你可以从这里看到它/克隆它/构建它:https://git.io/vMPqb


我正在尝试让共享元素为片段过渡工作。

项目中有两个 FAB - Feather 和 Plane。 Feather 和 Plane 是共享元素。单击 Feather 时,SheetDialog 被打开,Feather 应该动画到 Plane 对话框。目前它没有这样做,我正在尝试确定原因。

值得注意的是,我在 API 24 上运行它,因此版本 21 以下不支持转换的问题不是问题。

谁能告诉我为什么共享元素转换不起作用?

为了回应 repo 中的内容,有四个重要文件:

主要活动

package test.example.fabpop;

import android.os.Bundle;
import android.support.design.widget.FloatingActionButton;
import android.support.transition.ChangeBounds;
import android.support.transition.Fade;
import android.support.v4.app.FragmentTransaction;
import android.support.v7.app.AppCompatActivity;
import android.view.View;

public class MainActivity extends AppCompatActivity {

    FloatingActionButton fab_feather;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        fab_feather = (FloatingActionButton) findViewById(R.id.initial_fab_feather);
    }

    public void fabClick(View view) {
        SheetDialog dialogFragment = new SheetDialog();

        FragmentTransaction transaction = getSupportFragmentManager().beginTransaction();

        // This seemingly has no effect. I am trying to get it to work.
        transaction.addSharedElement(fab_feather, "transition_name_plane");

        dialogFragment.setSharedElementEnterTransition(new ChangeBounds());
        dialogFragment.setSharedElementReturnTransition(new Fade(Fade.OUT));

        dialogFragment.show(transaction, "frag_tag");
    }
}

activity_main.xml 布局

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/activity_main"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:paddingBottom="@dimen/activity_vertical_margin"
    android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    tools:context="test.example.fabpop.MainActivity">

    <LinearLayout
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_alignParentEnd="true"
        android:orientation="vertical">
        <TextView
            android:layout_width="200dp"
            android:layout_height="wrap_content"
            android:text="Transition to a BottomSheetDialogFragment that shares this FAB"
            android:textAlignment="center"/>

        <!--  Feather FAB  -->
        <android.support.design.widget.FloatingActionButton
            android:id="@+id/initial_fab_feather"
            android:layout_width="52dp"
            android:layout_height="52dp"
            android:layout_margin="16dp"
            android:layout_gravity="center"
            android:onClick="fabClick"
            android:transitionName="transition_name_feather"
            android:src="@drawable/ic_feather"
            />
    </LinearLayout>


</RelativeLayout>

SheetDialog

package test.example.fabpop;

import android.os.Bundle;
import android.support.annotation.Nullable;
import android.support.design.widget.BottomSheetDialogFragment;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;

public class SheetDialog extends BottomSheetDialogFragment {

    @Nullable
    @Override
    public View onCreateView(LayoutInflater inflater,
                             @Nullable ViewGroup container,
                             @Nullable Bundle savedInstanceState) {

        return inflater.inflate(R.layout.dialog_sheet, container, false);
    }
}

dialog_sheet.xml 布局

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical" android:layout_width="match_parent"
    android:layout_height="match_parent">

    <!--  Plane FAB  -->
    <android.support.design.widget.FloatingActionButton
        android:id="@+id/final_fab_plane"
        android:layout_width="75dp"
        android:layout_height="75dp"
        android:layout_margin="16dp"
        android:transitionName="transition_name_plane"
        android:src="@drawable/ic_plane"
        />

</LinearLayout>

在一个 Activity 和一个 Fragment 之间的过渡中不能有一个共享元素吗?也许只有在 Activity 到 Activity 或 Fragment 到 Fragment 之间才有可能,但不能跨越两种类型?也许这就是我无法让它工作的原因?

更新:

我现在尝试将&lt;item name="android:windowContentTransitions"&gt;true&lt;/item&gt; 添加到应用程序的主题中。

我现在也在尝试确保两个视图上的两个 transitionName 值相同。

这些都没有帮助解决问题。

【问题讨论】:

  • 您要过渡到的第二个活动是什么?
  • 过渡是从MainActivity(一个Activity)到SheetDialog(一个Fragment)
  • 感谢您的回答。我现在还有其他工作职责,但会在今天晚些时候查看您的答案:)
  • 转换应该将视图从一个活动转换到另一个活动。海事组织,不可能在一项活动中做到这一点。您需要做的是自己为视图设置动画。
  • @azizbekian,不可能从 Activity 动画到 Fragment?你有这方面的消息来源吗?我真的认为你可以,但在尝试让它发挥作用之后,我可能会相信你。

标签: java android android-layout android-fragments android-dialogfragment


【解决方案1】:

在我们开始之前...

让我们先快速回顾一下android框架是如何实现神奇的Shared Element Transition的。

共享元素转换实际上只是android框架的谎言之一。事实是,当您进行共享元素转换时,您实际上并没有在 Activities 之间共享任何视图,也就是说,您正在处理两个单独的视图。那是因为每个Activity 都有自己独立的视图树。

啊?!

假设您尝试将由view_a 标识的视图从ActivityA 转换为ActivityB 中的view_b。框架的作用是,它首先查找view_a 的某些属性,例如ActivityA 中的大小(widthheight)和位置(xy)。然后它将这些信息传递给ActivityB,在view_b 上应用这些属性,以便在ActivityA 关闭时它占据与view_a 完全相同的位置。之后,框架通过将您的view_b 反向动画到它在ActivityB 中的外观来开始转换。这就是创建共享视图的错觉的方式。魔法!

因此,

从上面我们可以推断,在开始任何动画之前,必须确保view_b已经在ActivityB上创建,否则这是不可能的。

在使用Fragment的情况下,

调用FragmentTransaction.commit()只会安排您的片段事务(片段不会在其包含的活动创建后立即创建)。

因此,在您的情况下,当您的 ActivityB 创建时,您的 view_b 丢失(正如解释的那样,这是因为尚未创建其包含的片段)。

解决方案

确保在动画开始之前创建您的view_b

为此,您必须找到一种方法来告诉框架不要做通常的事情,而是在创建动画之前等待您的信号。

实现这一点的一种方法是将您的代码更改为与此类似的代码:

活动B

class ActivityB extends Activity{
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        // ...
        // Tell the framework to wait.
        postponeEnterTransition();
    }
}

片段B

class FragmentB extends Fragment {

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
    View viewB = getView().findViewById(R.id.view_b);
    sharedview.getViewTreeObserver().addOnPreDrawListener(new ViewTreeObserver.OnPreDrawListener() {
        @Override
        public boolean onPreDraw() {
                // Tell the framework to start.
                sharedview.getViewTreeObserver().removeOnPreDrawListener(this);
                getActivity().startPostponedEnterTransition(); 
                return true;
         }
       });
       ...
    }
}

进一步阅读

Getting Started with Activity & Fragment Transitions

【讨论】:

  • 感谢您的回答,但我无法使其正常工作。您的答案使用了两个活动和两个片段,而我的项目只有一个。我确实尝试过应用这些概念,但在尝试了几种不同的方法后,我仍然无法让它发挥作用。
  • Fragment2 不返回任何值,这段代码不会编译。
  • @AlexBeggs thanx 指出了这一点,但其余代码完全取决于用户需求(返回膨胀视图将在普通场景中完成)。
  • @AnixPasBesoin 可以请你帮我解决这个问题stackoverflow.com/q/59431465/4291272
  • 这是一个很棒的解释,谢谢
【解决方案2】:

您正在寻找的是这个例子:https://github.com/hujiaweibujidao/FabDialogMorph。 请注意,您要实现的不是标准的 Android 过渡,您需要根据 ChangeBounds 过渡创建自己的变形过渡。最初,它已在Plaid 中显示,感谢Nick Butcher,因此您可以查看它以获取更多提示和背景。

【讨论】:

  • 感谢您的回答。我会试一试,但我有点怀疑这是实现我需要的适当方式。我不想做任何变形,只是将视图从我的活动转移到片段。
【解决方案3】:

只是引用Android Docs

使用共享元素启动活动

  1. 使用 android:transitionName 属性为两种布局中的共享元素指定一个通用名称。

据我所知,您的两个 XML 不共享相同的 android:transitionName

尝试将 XML 布局(activity_main.xml 和 dialog_sheet.xml)中的 android:transitionName 都更改为相同的字符串,例如:transition_plane_feather

我自己没有测试过代码,但我认为这可能是一个很好的起点。

奖金提示

如果您计划在不兼容较低 API 级别(Lollipop 之前)的情况下完全实现 Material Design,我强烈建议您查看 Google UI/UX 设计师制作的示例:https://github.com/nickbutcher/plaid

另外,请查看该示例的优秀补充 YouTube 视频:https://www.youtube.com/watch?v=EjTJIDKT72M

它向您展示了一些可用于完全实现 Material Design 以及共享元素转换的最佳技巧和实践。

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 2017-01-12
    • 2021-04-05
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2015-11-08
    相关资源
    最近更新 更多