Activity托管的两个Fragment间的数据传递

废话不多说,直接上案例,以下案例截取自《Android编程权威指南第二版》第12章 对话框 12.3。

图1 CrimeFragment

图2 DatePickerFragment

如图1为CrimeFragment,图2为点击DETAILS下的第一个日历按钮弹出的日历选择器对话框,其为DatePickerFragment,这两个fragment是托管在一个CrimeActivity上的,现在要实现:

  1. 弹出的日历选择器要获取并显示按钮上的日期信息
  2. 用户在任意选择日期并点击OK后,按钮可以展示用户选择的日期

功能1解决方案(传递数据给DatePickerFragment)

实现思路

DatePickerFragment中新建newInstance(Date)方法,之后将Date作为argument附加给DatePickerFragment,然后在CrimeFragment中调用newInstance(Date)并将按钮日期传递进去,最后在DatePickerFragment中通过getArguments()方法获取Date即可

代码实现

  • 添加newInstance(Date)方法
public class DatePickerFragment extends DialogFragment {
      private static final String ARG_DATE = "date";

      public static DatePickerFragment newInstances(Date date) {
           Bundle args = new Bundle();       
           args.putSerializable(ARG_DATE, date); //ARG_DATE为标记键key,目的是在get时找到对应的值date   
           DatePickerFragment fragment = new DatePickerFragment();
           fragment.setArguments(args);
           return fragment;
      }
......
}
  • 调用newInstance()方法并传递按钮日期Date
public class CrimeFragment extends Fragment {
      @Override
      public View onCreateView(...) {
            mDateButton.setOnClickListener(new View.OnClickListener() {    
                @Override    
                public void onClick(View v) {
                      FragmentManager fragmentManager = getFragmentManager();

                      DatePickerFragment dialog = DatePickerFragment.newInstances(mCrime.getDate());

                      dialog.show(fragmentManager, DIALOG_DATE);
                }
            }
       }
}
  • DatePickerFragment 中直接通过getArguments()方法获取date。DatePickerFragment使用Date中的信息来初始化DatePicker对象,然而,DatePicker对象的初始化需整数形式的年月日。Date是个时间戳,无法直接提供整数形式的年月日。
    要达到目的,必须首先创建一个Calendar对象,然后用Date对象配置它,再从Calendar对象中取回所需信息。
@Override
public Dialog onCreateDialog(Bundle savedInstanceState) {
      Date date = (Date) getArguments().getSerializable(ARG_DATE);

      Calendar calendar = Calendar.getInstance();
      calendar.setTime(date);
      int year = calendar.get(Calendar.YEAR);
      int month = calendar.get(Calendar.MONTH);
      int day = calendar.get(Calendar.DAY_OF_MONTH);

      View v = LayoutInflater.from(getActivity()).inflate(R.layout.dialog_date, null);

      mDatePicker = (DatePicker) v.findViewById(R.id.dialog_date_date_picker);
      mDatePicker.init(year, month, day, null);

      return new AlertDialog.Builder(getActivity())        
             .setView(v)        
             .setTitle(R.string.date_picker_title)  
             .setPositiveButton(android.R.string.ok, null)
             .create();
}

注:使用fragment的argument可以将接收来的数据保存在Bundle中,然后在使用时直接在fragment内部取出即可,不需要依赖外部的activity或者fragment,这可以使fragment的封装性更好,即低耦合,推荐使用。

功能2解决方案(返回数据给CrimeFragment)

实现思路

  • 类似于之前一篇Activity间的数据传递,通过设置目标TargetFragment将两个fragment关联,然后在DatePickerFragment中将所传数据存入Intent,并借用目标fragment的Fragment.onActivityResult()方法,就能实现数据回传

代码实现

  • 设置CrimeFragment为目标fragment
public class CrimeFragment extends Fragment {

      private static final int REQUEST_DATE = 0;

      @Override
      public View onCreateView(...) {
            mDateButton.setOnClickListener(new View.OnClickListener() {    
                @Override    
                public void onClick(View v) {
                      FragmentManager fragmentManager = getFragmentManager();
                      DatePickerFragment dialog = DatePickerFragment.newInstances(mCrime.getDate());

                      dialog.setTargetFragment(CrimeFragment.this, REQUEST_DATE); //设置目标fragment

                      dialog.show(fragmentManager, DIALOG_DATE);
                }
            }
       }
}

REQUEST_DATE为请求代码,目标fragment可以通过请求代码确认是哪个fragment在回传数据

  • 创建intent并将日期数据作为extra附加到intent,最后调用CrimeFragment.onActivityResult(...)传递数据给目标fragment
public class DatePickerFragment extends DialogFragment {
        public static final String EXTRA_DATE = "com.bignerdranch.android.criminalintent.date";
......
        private void sendResult(int resultCode, Date date) {    
                if (getTargetFragment() == null) {        
                    return;    
                }    
                Intent intent = new Intent();    
                intent.putExtra(EXTRA_DATE, date);
                getTargetFragment().onActivityResult(getTargetRequestCode(), resultCode, intent);
}
  • 使用sendResult(...)私有方法。用户点击对话框中的positive按钮(即OK按钮)时,需要从日期选择器DatePicker中获取日期并回传给CrimeFragment,在onCreateDailog(...)中,替换setPositiveButton(...)的null参数,实现DialogInterface.OnClickListener监听器接口。在监听器接口中的onClick(...)方法中,获取日期并调用setResult(...)
@Override
public Dialog onCreateDialog(Bundle savedInstanceState) {
......
      return new AlertDialog.Builder(getActivity())        
                  .setView(v)        
                  .setTitle(R.string.date_picker_title)     
                  .setPositiveButton(android.R.string.ok, new DialogInterface.OnClickListener() {            
                        @Override           
                        public void onClick(DialogInterface dialog, int which) {               
                              int year = mDatePicker.getYear();                
                              int month = mDatePicker.getMonth();               
                              int day = mDatePicker.getDayOfMonth();                
                              Date date = new GregorianCalendar(year, month, day).getTime(); 
               
                              sendResult(Activity.RESULT_OK, date);            
                        }        
                   })        
                   .create();
}
  • 在CrimeFragment中,覆盖onActivityResult(...)方法,从extra中获取日期数据并显示
@Override
public void onActivityResult(int requestCode, int resultCode, Intent data) {    
      if (resultCode != Activity.RESULT_OK) {        
          return;    
      }    
      if (requestCode == REQUEST_DATE) {          
          Date date = (Date) data.getSerializableExtra(DatePickerFragment.EXTRA_DATE):
          mCrime.setDate(date);       
          updateDate();    
      }    
 }
private void updateDate() {    
      SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy年MM月dd日 EEEE", Locale.CHINA);
      mDateButton.setText(dateFormat.format(mCrime.getDate()));
}

手机屏幕空间有限,常常使用activity托管全屏的fragment界面,以显示用户输入要求。父Activity的fragment调用startActivityForResult()方法启动子Activity,子Activity销毁后,会将数据通过onActivityResult()传递给父Activity,父Activity会接收到onActivityResult()的调用请求,requestCode和resultCode核对正确后,父Activity会接收并将数据转发给自己托管的fragment

平板设备的屏幕空间比较大,适合以弹出对话框的方式显示信息和接收用户输入。这种情况,应设置目标fragment并调用对话框fragment的show方法。对话框销毁后,对话框的fragment会调用目标fragment的onActivityResult方法。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,884评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,755评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,369评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,799评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,910评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,096评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,159评论 3 411
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,917评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,360评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,673评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,814评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,509评论 4 334
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,156评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,882评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,123评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,641评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,728评论 2 351

推荐阅读更多精彩内容