第03讲 fragment

Posted duhuamei

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了第03讲 fragment相关的知识,希望对你有一定的参考价值。

Fragment

官网文档:https://developer.android.google.cn/guide/components/fragments

什么是Fragment

在手机上,Activity作为用户和应用交互的界面。如果界面上的View较多较复杂时,Activity会较复杂。Fragment提供了一种解决方案,把view放到几个Fragment中,然后再由这些Fragment拼成一个完整的Activity。

如果多个界面中有相同的部分,可以将重复部分放在fragment中,然后在多个activity中使用该fragment,达到风格统一,代码重用的目的。

 

Fragment的应用场景:

 

Fragment的使用

静态添加Fragment

1.      准备Fragment使用的layout文件

2.      新建Fragment类,继承自Fragment

注意:需要重写onCreateView方法;  推荐使用support-v4包中的Fragment类。

public class LeftFragment extends Fragment {

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

      View view=inflater.inflate(R.layout.left_fragment,container,false);

      return view;

 }

}

这两步完成了一个Fragment,其与Activity非常类似,同样是写好对应的layout,然后在生命周期方法中加载layout。

3.      将该Fragment作为一个普通的控件添加到Activity对应的layout中

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"

    xmlns:tools="http://schemas.android.com/tools">

    <fragment

        android:id="@+id/left_fragment"

        android:name="com.ncst.coursecode.LeftFragment"

        android:layout_width="wrap_content"

        android:layout_height="match_parent"/>

</LinearLayout>

注意:必须显式指明id和name属性。name属性即为碎片的类名。

另外,包含碎片的Activity需要继承自FragmentActivity。而我们默认使用的AppCompatActivity正是继承自FragmentActivity。

动态添加Fragment

1,2步骤与上面相同(即准备一个Fragment)。

3.      在Activity对应的layout文件中预留fragment将要放置的容器。

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"

    xmlns:tools="http://schemas.android.com/tools">

       <FrameLayout

        android:id="@+id/fragment_container"

        android:layout_width=" match_parent "

        android:layout_height="match_parent"/>

</LinearLayout>

4.      在Acitivity中动态添加Fragment

Fragment fragment=new RightFragment();

        FragmentManager manager=getSupportFragmentManager();

        FragmentTransaction transaction=manager.beginTransaction();

        transaction.add(R.id. fragment_container,fragment);

        transaction.commit();

另外两个方法:replace(R.id. fragment_container,fragment);移除容器中已有的所有fragment,并添加本fragement。remove(R.id. fragment_container,fragment);移除容器中的该fragment。

注意:在XML中静态添加的fragment不能通过上述两个方法移除。

addToBackStack()将事务添加到返回栈,该返回栈由Activity管理,以便用户能够通过按返回按钮撤消事务并回退到上一片段。

Fragment与activity进行通信

在活动中获取碎片实例:

getFragmentManager().findFragmentById(R.id.right_fragment);

在碎片中获取和当前碎片相关联的活动实例:

getActivity();

在一个碎片中获取和当前碎片相关联的活动中的另外一个碎片:

getActivity().getFragmentManager().findFragmentById(R.id.left_fragment);

Fragment生命周期

片段必须始终嵌入在 Activity 中,其生命周期直接受宿主 Activity 生命周期的影响。例如,当 Activity 暂停时,其中的所有片段也会暂停;当 Activity 被销毁时,所有片段也会被销毁。

 

Fragment例子(手机和平板显示不同界面,复用相同的Fragment)

 

Fragment应用的几个例子

Fragment实现底部导航(类似微信界面)

图片:

 

 

分析:页面分为两部分:底部(RadioGroup),上面部分(放置Fragment)

 

  1. 准备两个Fragment(ChatFragment,MomentFragment)
  2. 编写Activity对应的layout(activity_wechat)
  3. 在Activity中动态将Fragment添加到ViewGroup中。(WechatActivity)
  4. 给RadioGroup添加事件监听,切换Fragment。

ViewPager实现左右滑动界面

ViewPager功能:通过手势滑动可以完成View的切换,一般是用来做APP 的引导页或者实现图片轮播, ViewPager在:android.support.v4.view.ViewPager。

ViewPager就是一个简单的页面切换组件,我们可以往里面填充多个View,然后我们可以左右滑动,从而切换不同的View,和前面学的ListView,RecyclerView一样,我们也需要一个Adapter (适配器)将我们的View填充到ViewPager,ViewPager有一个特定的Adapter—— PagerAdapter!另外,Google官方是建议我们使用Fragment来填充ViewPager的,这样可以更加方便的生成每个Page,以及管理每个Page的生命周期!给我们提供了两个Fragment 专用的Adapter:FragmentPageAdapter和FragmentStatePagerAdapter 。

使用PagerAdapter

  1. 必须重写的四个方法:

instantiateItem(ViewGroup, int):向父布局添加View

destroyItem(ViewGroup, int, Object):移除一个给定位置的View。

getCount():获得viewpager中有多少个view

isViewFromObject(View, Object):一般的实现:return view == object;

 

一个典型的PagerAdapter类:

public class MyPagerAdapter extends PagerAdapter {

    List<View> viewList;

    public MyPagerAdapter(List<View> viewList) {

        this.viewList = viewList;

    }

    public int getCount() {

        return viewList.size();

    }

    public boolean isViewFromObject(View view, Object o) {

        return view==o;

    }

    public Object instantiateItem(ViewGroup container, int position) {

        container.addView(viewList.get(position));

        return viewList.get(position);

    }

    public void destroyItem(ViewGroup container, int position, Object object) {

        container.removeView(viewList.get(position));

    }

}

  1. 定义Activity的Layout文件,其中包含ViewPager

<android.support.v4.view.ViewPager

        android:id="@+id/view_pager"

        android:layout_width="match_parent"

        android:layout_height="match_parent">

    </android.support.v4.view.ViewPager>

  1. 在Activity中设置ViewPager的Adapter

 

另外,PagerTabStrip 作为ViewPager的交互指针。只需在XML文档中作为ViewPager的子项即可。并且在PagerAdapter需要重写getPageTitle()方法,用于获取指定页面的标题。

PagerTitleStrip 与PagerTabStrip 类似,区别是不能交互。

  <android.support.v4.view.ViewPager

        android:id="@+id/view_pager"

        android:layout_width="match_parent"

        android:layout_height="match_parent">

 

        <android.support.v4.view.PagerTabStrip

            android:layout_width="match_parent"

            android:layout_height="wrap_content"

            android:layout_gravity="top" />

    </android.support.v4.view.ViewPager>

 

官方文档:https://developer.android.google.cn/reference/android/support/v4/view/PagerAdapter

使用FragmentPagerAdapter

是PagerAdapter的实现,每个页面是一个Fragment。

This version of the pager is best for use when there are a handful of typically more static fragments to be paged through, such as a set of tabs. The fragment of each page the user visits will be kept in memory, though its view hierarchy may be destroyed when not visible. This can result in using a significant amount of memory since fragment instances can hold on to an arbitrary amount of state. For larger sets of pages, consider FragmentStatePagerAdapter.

FragmentPagerAdapter适合少量固定数量的页面情况。页面多且不固定时会耗费内存,这时应考虑使用FragmentStatePagerAdapter。

 

实现FragmentPagerAdapter时只需重写:getItem(int) getCount()

class MyAdapter extends FragmentPagerAdapter {

        public MyAdapter(FragmentManager fm) {

            super(fm);

        }

        public Fragment getItem(int i) {

            return MyFragment.newInstance(i);

        }

        public int getCount() {

            return NUM_ITEMS;

        }

}

 

Android提供了android.support.design.widget.TabLayout控件,用来实现顶部标题的左右滑动功能。

  1. 在布局文件中添加控件:

<android.support.design.widget.TabLayout

        android:id="@+id/tab_layout"

        android:layout_width="match_parent"

        android:layout_height="wrap_content" />

  1. 让TabLayout与ViewPager关联到一起,由ViewPager管理TabLayout:

TabLayout tabLayout = (TabLayout) findViewById(R.id.tab_layout);

tabLayout.setupWithViewPager(viewPager);

  1. 当然在Adapter中要重写getPageTitle()方法。

 

官方文档:

https://developer.android.google.cn/reference/android/support/v4/app/FragmentPagerAdapter.html

以上是关于第03讲 fragment的主要内容,如果未能解决你的问题,请参考以下文章

用C#编写百度地图Android手机应用程序(第3讲)

setUserVisibleHint with Fragments for android

如何将数据从一个 Fragment 发送到另一个 Fragment?

显示/隐藏片段并以编程方式更改可见性属性

如何在 Fragment 之间传递值

我的片段无法转换为 android.support.v4.app.Fragment