Master-Detail 用户界面
来源:互联网 发布:逆战刷枪软件安卓版 编辑:程序博客网 时间:2024/05/21 15:03
适合平板设备的用户界面,让用户能同时查看列表和明细界面并与它们进行交互。下图展示了这样的列表明细界面。通常我们也称为主从用户界面(master-detail interface)。
一、使用别名资源
别名资源是一种指向其他资源的特殊资源。它存放在res/values/目录下,并按照约定定义在refs.xml文件中。
我们将分别创建用于手机指向activity_fragment.xml布局的别名资源,以及用于平板指向activity_twopane.xml布局的别名资源.
activity_fragment.xml布局
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:id="@+id/fragmentContainer" android:layout_width="match_parent" android:layout_height="match_parent"/>
activity_twopane.xml布局
<?xml version="1.0" encoding="utf-8"?><LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="horizontal" android:showDividers="middle" > <FrameLayout android:id="@+id/fragmentContainer" android:layout_width="0dp" android:layout_height="match_parent" android:layout_weight="1" /> <FrameLayout android:id="@+id/detailFragmentContainer" android:layout_width="0dp" android:layout_height="match_parent" android:layout_weight="3" /></LinearLayout>
1、创建默认的别名资源值
<?xml version="1.0" encoding="utf-8"?><resources> <item name="activity_masterdetail" type="layout">@layout/activity_fragment</item></resources>
别名资源指向了单版面布局(即activity_fragment.xml布局)资源文件。别名资源自身也具有资源ID:R.layout.activity_masterdetail。注意,别名的type属性决定了资源ID属于什么内部类。即使别名资源自身存放在res/values/目录中,它的资源ID依然归属于R.layout内部类。
2、创建平板设备专用可选资源
存放在res/values/目录下的别名资源是系统默认的别名资源。现在,创建一个可选别名资源,以实现在平板等大屏幕设备上,activity_masterdetail别名资源可以指向activity_twopane.xml双版面布局资源。
在res/目录下新建一个名为values-sw600dp的目录。然后在res/values-sw600dp/目录下创建新的refs.xml文件。
<?xml version="1.0" encoding="utf-8"?><resources> <item name="activity_masterdetail" type="layout">@layout/activity_twopane</item></resources>
配置修饰符-sw600dp是什么意思?SW是small width(最小宽度)的缩写,虽然字面上是宽度的意思,但它实际指的是屏幕的最小尺寸(dimension),因而SW与设备的当前方向无关。
配置修饰符-sw600dp表明:对任何最小尺寸为600dp或更高的设备,都使用该资源。
需要说明的是,Android3.2中才引入了最小宽度配置修饰符。这意味着,运行Android 3.0或Android 3.1系统的平板设备无法识别它。为了解决该问题,可以增加另一种是用-xlarge(仅适用于Android3.2以前的版本)屏幕尺寸修饰符的可选资源。
在res/目录下新建一个名为values–xlarge的目录。然后将res/values-sw600dp/目录下的refs.xml文件复制到res/values–xlarge/目录。
二、Activity:fragment的托管者
让CrimeListActivity可以展示一个完整的双版面用户界面,我们的第一反应可能会认为,只需再为平板设备实现一个CrimeListFragment.onListItemClick(…)监听器方法就可以了。这样,无需启动新的CrimePagerActivity,onListItemClick(…)方法会获取CrimeListActivity的FragmentManager,然后提交一个fragment事务,将CrimeFragment添加到明细fragment容器中。
具体代码如下:
public void onListItemClick(ListView l, View v, int position, long id) { super.onListItemClick(l, v, position, id); Crime crime = (Crime) getListAdapter().getItem(position); Fragment fragment = CrimeFragment.newInstance(crime.getId()); FragmentManager fm = getActivity().getSupportFragmentManager(); fm.beginTransaction().add(R.id.detailFragmentContainer, fragment).commit(); }
以上设想虽然行的通,但做法很老套。fragment天生是一种独立的开发构建。如果要开发一个fragment用来添加其他fragment到activity的FragmentManager,那么这个fragment就必须知道托管activity是如何工作的,这样一来,该fragment就再也无法作为独立的开发构建来使用了。
为保持fragment的独立性,我们可以在fragment中定义回调接口,委托托管
activity来完成那些不应由fragment处理的任务。托管activity将实现回调接口,履行托管fragment的任务。
为CrimeListFragment添加回调接口
private Callbacks mCallbacks; public interface Callbacks{ void onCrimeSelected(Crime crime); } /** * 该方法是在Fragment附加给Activity时调用,将托管activity强制类型转换为Callbacks对象并赋值给Callbacks类型变量。 */ @Override public void onAttach(Activity activity) { super.onAttach(activity); mCallbacks = (Callbacks) activity; } /** * 在生命周期销毁方法中,我们应将Callbacks变量置为null,随后再也无法访问该activity或继续指望该activity存在了。 */ @Override public void onDetach() { super.onDetach(); mCallbacks = null; }
CrimeListActivity必须实现CrimeListFragment.Callbacks接口并重写onCrimeSelected(Crime crime)方法。在onCrimeSelected(Crime crime)方法被调用时,CrimeListActivity需要完成以下二选一的任务:
1、如果使用手机用户界面布局,启动新的CrimePagerActivity;
2、如果使用平板设备用户界面布局,将将CrimeFragment放入detailFragmentContainer中。
为了确定实例化手机还是平板界面布局,可以检查布局ID。但最好最准确的检查方式是检查布局是否包含detailFragmentContainer。
@Override public void onCrimeSelected(Crime crime) { if (findViewById(R.id.detailFragmentContainer) == null) { Intent intent = new Intent(this, CrimePagerActivity.class); // UUID是Serializable对象,我们调用了可接受Serializable对象的putExtra(...)方法,即putExtra(String, // Serializable)方法。 intent.putExtra(CrimeFragment.EXTRA_CRIME_ID, crime.getId()); /** * 从fragment中启动activity,可以调用Fragment.startActivity(Intent)方法。 * * 除了将返回结果从托管activity传递给fragment的额外实现代码之外,Fragment. * startActivityForResult(Intent, int)方法的实现代码与activity的同名方法基本相同。 */ startActivity(intent); } else { FragmentManager fm = getSupportFragmentManager(); FragmentTransaction ft = fm.beginTransaction(); Fragment oldDetail = fm .findFragmentById(R.id.detailFragmentContainer); Fragment newDetail = CrimeFragment.newInstance(crime.getId()); if (oldDetail != null) ft.remove(oldDetail); ft.add(R.id.detailFragmentContainer, newDetail); ft.commit(); } }
三、设备屏幕尺寸的确定
Android3.2之前,屏幕大小修饰符是基于设备的屏幕大小来提供可选资源的。
屏幕大小修饰符将不同的设备分成了四大类别:
顺应允许开发者测试设备尺寸的新修饰符的推出,屏幕大小修饰符在Android3.2中已弃用。独立的屏幕尺寸修饰符如下:
代码地址
- Master-Detail 用户界面
- Master-Detail GridView
- Master-Detail GridView
- ASPxGridView之Master-Detail
- Android编程权威指南(第二版)学习笔记(十七)—— 第17章 Master-Detail 用户界面
- ADO.NET Binding: Master Detail
- OAF中的MASTER-DETAIL关系
- OAF中的MASTER-DETAIL关系
- DevExpress控件XtraGrid的Master-Detail用法
- IOS Master-Detail Template Application Demo 解析
- Master-Detail 查询 表体字符串合并
- 2.25 Presenting Master-Detail Views with UISplitViewController
- HighCharts dynamic-master-detail多线解决方案
- 一步一步学习Angular2(03.示例 MASTER/DETAIL)
- iOS左是master,右边是detail
- Master-Detail Application的纯代码实现
- IOS基础知识记录四(Master-Detail Application)
- 对象关系Lookup与Master-Detail
- ubunt 下搭建ftp服务器
- H5学习之旅-H5列表(8)
- CentOS7安装.tar.gz包
- iOS开发-获取设备电量
- linux找不到动态链接库 .so文件的解决方法
- Master-Detail 用户界面
- 单向电表模块焊接完成
- 获得android手机中的传感器信息
- Redis学习11之jedis问题之1--connect time out
- hdu1074Doing Homework(状态压缩,好题)
- CrossApp环境搭建
- 打开系统相册/相机,选择图片
- hdoj 4496 D-City 【并查集】
- Python多版本管理软件pyenv的安装应用及pip的使用讲解