这个问题应该涉及到所有实现了Tab风格的应用的设计
android的SDK提供了TabHost,TabWidget控件和TabActivity来支持这种方式,但是我感觉限制比较大,就尝试使用ActivityGroup去做。
在布局中,总要设置一个导航的菜单栏,iphone的是放在下面的。
总体上的布局都是如此的,整体的布局一般是一个RelativeLayout,下面放一个width是fill_parent的菜单栏。
contain是一个容器,我自己用过很多,比如LinearLayout,ViewAnimator等等,后者支持动画,不过其实这样的布局,上面的内容有点动画,看起来会有点怪。
ActivityGroup里面有个非常重要的成员,它是负责subActivity管理的——LocalActivityManager,可以通过getLocalActivityManager()来获的。
来看一下这个成员有什么作用,当然从老农的博客里面可以去寻找这个类的中文API啦~
public Activity getCurrentActivity()
返回当前的activity,也就是状态是running的subActivity,也就是getAcitivity(getCurrentId())的实现
public String getCurrentId()
见上条
public void removeAllActivities()
销毁所有的subActivity
public Activity getActivity(String id)
通过acitivityId找到这个group对应的Activity对象。
public Window destroyActivity(String id, boolean finish)
这个是负责结束某一个subActivity的,在每次我们开启一个新的activity的时候,往往给他一个activityId,然后我们就可以通过这个id去销毁他。
public Window startActivity(String id, Intent intent)
在当前的group里面运行一个新的subActivity,必须传入一个id去映射之,做为索引。这边的Intent和Activity里的Intent一样,我们也可以通过这个来传递参数。
这边有一个需要注意的就是subActivity的启动方式方面的问题:
SingleTop和平时不同,举个例子:a启动b,b启动a,这时候在acitivity的栈里面有个就有了两个a,而在acitivityGroup管理下的SingleTop启动,则是a->b,再b->a,则不会有两个a,则是跳入a的时候call他的newIntent()方法
所以在这边如果你要复用这个界面,我个人觉得可以用singleTop,不仅可以保存它的状态,也可以把后续的操作放在onNewIntent(Intent) 里面,和onCreate区分开来。
给出一段示例代码:
public class AcitivityGroupTestActivity extends ActivityGroup {
LocalActivityManager activityManager;
Button b1, b2;
LinearLayout contain;
/** Called when the activity is first created. */
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
activityManager = getLocalActivityManager();
b1 = (Button) findViewById(R.id.button1);
b2 = (Button) findViewById(R.id.button2);
contain = (LinearLayout) findViewById(R.id.contain);
b1.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
contain.removeAllViews();
Intent in = new Intent(AcitivityGroupTestActivity.this, Activity01.class);
in.addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP);
//in.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
Window mWindow = activityManager.startActivity("1", in);
contain.addView(mWindow.getDecorView());
Activity a1 = activityManager.getActivity("1");
Activity a2 = activityManager.getActivity("2");
System.out.println("a1:" + a1 + ",a2:" + a2);
}
});
b2.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
contain.removeAllViews();
Intent in = new Intent(AcitivityGroupTestActivity.this, Activity02.class);
in.addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP);
//in.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
Window mWindow = activityManager.startActivity("2", in);
contain.addView(mWindow.getDecorView());
Activity a1 = activityManager.getActivity("1");
Activity a2 = activityManager.getActivity("2");
System.out.println("a1:" + a1 + ",a2:" + a2);
}
});
}
}
布局文件 如下 :
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
>
<LinearLayout
android:orientation="horizontal"
android:layout_width="fill_parent"
android:layout_height="wrap_content">
<Button
android:id="@+id/button1"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="..1..">
</Button>
<Button
android:id="@+id/button2"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="..2..">
</Button>
</LinearLayout>
<LinearLayout
android:id="@+id/contain"
android:layout_width="fill_parent"
android:layout_height="fill_parent">
</LinearLayout>
</LinearLayout>
如果我们要给他们加上跳转,怎么做呢?
一种做法是做在subActivity里面的:
在subActiivity里面获取 getParent() 则就是我们的ActivityGroup,然后通过这个来跳转。
比如在subActivity01里面按button01,跳转到subActivity02,我们可以这么写。
// for parent
AcitivityGroupTestActivity parent = (AcitivityGroupTestActivity) getParent();
final LinearLayout contain = (LinearLayout) parent.findViewById(R.id.contain);
contain.removeAllViews();
Intent in = new Intent(getParent(), Activity02.class);
in.addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP);
LocalActivityManager manager = parent.getLocalActivityManager();
String currentId = manager.getCurrentId();
Window window = manager.startActivity("2", in);
contain.addView(window.getDecorView());
// 如果有必要可以杀死之前的activity
manager.destroyActivity(currentId, true);
但是要做的通用点,则需要想办法了。
我个人的做法是在ActivityGroup里面注册一个receiver去监听跳转的信息,并且获取参数,我的参数全部封装在一个bundle里,这个是一个统一的格式,当然也可以用自己喜欢的方式。
final static public String LOAD_SUBACTIVITY = "load_subActivity";
final static public String SUBACTIVITY_NAME = "subActivity_name";
final static public String SUBACTIVITY_01 = "subActivity_01";
final static public String SUBACTIVITY_02 = "subActivity_02";
final static public String SUBACTIVITY_PARAM = "subActivity_param";
private BroadcastReceiver receiver = new BroadcastReceiver() {
@Override
public void onReceive(Context context, Intent intent) {
// 去接收
String action = intent.getAction();
if(action.equals(LOAD_SUBACTIVITY)) {
String subActivityName = intent.getStringExtra(SUBACTIVITY_NAME);
Bundle param = intent.getBundleExtra(SUBACTIVITY_PARAM);
loadSubActivity(subActivityName, param);
}
}
};
当然你要注意在ActivityGroup里面去的onCreate去注册监听,和在onDestroy注销之。
public void onCreate(Bundle savedInstanceState) {
...
registerReceiver(receiver, new IntentFilter(LOAD_SUBACTIVITY));
}
protected void onDestroy() {
unregisterReceiver(receiver);
}
如果大家有别的思路或者更好的做法可以交流。
mailto:yanyin1986@gmail.com