android ActivityGroup认识
1)ActivityGroup
根据SDK的解释其功能“A screen that contains and runs multiple embedded activities.”。翻译成汉语也就是大概"在Android中,ActivityGroup类是Activity的容器,可以包含多个嵌套进来的Activitys".接下来依然采用源码分析的方式来了解该类的内部实现。
首先,从SDK中和源码中都可以获知,ActivityGroup类的父类是Activity,也就是说二者具有相同的接口和生命周期,同Activity一样,也有onCreate()、onPause()等函数可供我们重载。在ActivityGroup的源码中有成员变量
protected LocalActivityManager mLocalActivityManager;
该变量在ActivityGroup的构造函数中创建并初始化,可见,ActivityGroup的功能实现肯定是要委托给这个对象来完成了。为了给用户开放对此对象的访问,ActivityGroup提供了
12345 | public final LocalActivityManager getLocalActivityManager() { return mLocalActivityManager; } |
通过浏览ActivityGroup的源码可以发现,几乎全部是以通过LocalActivityManager对象来完成的具体动作,比如:
1234567891011 | protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); Bundle states = savedInstanceState != null ? (Bundle) savedInstanceState.getBundle(STATES_KEY) : null; mLocalActivityManager.dispatchCreate(states);} |
下面,我们就来看一下LocalActivityManager的源码。在该类中,提供了一个私有类
123456789101112131415161718192021222324252627 | private static class LocalActivityRecord extends Binder { LocalActivityRecord(String _id, Intent _intent) { id = _id; intent = _intent; } final String id; // Unique name of this record. Intent intent; // Which activity to run here. ActivityInfo activityInfo; // Package manager info about activity. Activity activity; // Currently instantiated activity. Window window; // Activity's top-level window. Bundle instanceState; // Last retrieved freeze state. int curState = RESTORED; // Current state the activity is in. } |
用于保存Activity的信息,并提供了
1234567 | private final Map<String, LocalActivityRecord> mActivities = new HashMap<String, LocalActivityRecord>(); private final ArrayList<LocalActivityRecord> mActivityArray = new ArrayList<LocalActivityRecord>(); |
采用这样的数据结构用于对所有嵌入的子Activity信息进行保存处理。其中前者用于通过String快速查找,后者用于以数组的方式快速访问,是典型的以空间换时间的的方式。
1234567891011121314151617181920212223242526272829303132333435363738394041424344454647 | public void dispatchCreate(Bundle state) { if (state != null) { final Iterator<String> i = state.keySet().iterator(); while (i.hasNext()) { try { final String id = i.next(); final Bundle astate = state.getBundle(id); LocalActivityRecord r = mActivities.get(id); if (r != null) { r.instanceState = astate; } else { r = new LocalActivityRecord(id, null); r.instanceState = astate; mActivities.put(id, r); mActivityArray.add(r); } } catch (Exception e) {…… } } } mCurState = CREATED; } |
从这里我们可以看出,当有一个ActivityGroup被Create的时候,就会有对应的Activity信息被保存到数组中。
当我们调用LocalActivityManager的startActivity()以产生Window的时候,我们也可以看到
123456789101112131415161718192021222324252627 | public Window startActivity(String id, Intent intent) { …… LocalActivityRecord r = mActivities.get(id); if (r == null) { r = new LocalActivityRecord(id, intent); adding = true; } …… if (adding) { mActivities.put(id, r); mActivityArray.add(r); } …… } |
有了这个数组,就可以遍历到ActivityGroup中嵌入的Activitys了,从而可以实现ActivityGroup的功能。
以上的分析结果产生的类图如下:
其核心的方面主要体现在:
Intent i = new Intent(MainActivity.this,IndexActivity.class);
Window indexWindow = localManager.startActivity(INDEX_ID, i);
indexDecorView = indexWindow.getDecorView();
localManager = this.getLocalActivityManager();
然后将view加进你想要用的模块,多个view之间控制好隐藏和显示,就可以了,用activityGroup你会被焦点的问题搞郁闷,我现在一直纠结,如何在window 之间切换,或者request到focus,有这个方面的经验可以谈下。
范例1使用ActivityGroup来切换Activity和Layout
在一个主界面中做Activity切换一般都会用TabActivity,使用方便,Activity互相之间相对独立,但是可定制性不强,而且修改起来很麻烦。当然也可以把layout分开,把逻辑代码全写在主界面的逻辑代码中,但是很明显可维护性相当差,这里通过ActivityGroup来解决这个问题。
一、效果图
要求点击底部不同图片按钮切换不同的Activity,并在中间显示Activity对应的ContentView。
二、 实现代码
2.1 layout.xml
<?xmlversion="1.0"encoding="utf-8"?><LinearLayoutxmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width ="fill_parent"android:orientation="vertical"
android:layout_height ="fill_parent">
<LinearLayoutandroid:gravity="center_horizontal"
android:background ="@drawable/myinfor2"android:layout_width="fill_parent"
android:layout_height ="wrap_content">
<TextViewandroid:id="@+id/cust_title"android:textColor="@android:color/white"
android:textSize ="28sp"android:text="模块1"android:layout_width="wrap_content"
android:layout_height ="wrap_content"></TextView>
</LinearLayout>
<!--中间动态加载View-->
<ScrollViewandroid:measureAllChildren="true"android:id="@+id/containerBody"
android:layout_weight ="1"android:layout_height="fill_parent"
android:layout_width ="fill_parent">
</ScrollView>
<LinearLayoutandroid:background="@android:color/black"
android:layout_gravity ="bottom"android:orientation="horizontal"
android:layout_width ="fill_parent"android:layout_height="wrap_content">
<!--功能模块按钮1-->
<ImageViewandroid:id="@+id/btnModule1"android:src="@android:drawable/ic_dialog_dialer"
android:layout_marginLeft ="7dp"android:layout_marginTop="3dp"
android:layout_marginBottom ="3dp"android:layout_width="wrap_content"
android:layout_height ="wrap_content"/>
<!--功能模块按钮2-->
<ImageViewandroid:id="@+id/btnModule2"android:src="@android:drawable/ic_dialog_info"
android:layout_marginLeft ="7dp"android:layout_marginTop="3dp"
android:layout_marginBottom ="3dp"android:layout_width="wrap_content"
android:layout_height ="wrap_content"/>
<!--功能模块按钮3-->
<ImageViewandroid:id="@+id/btnModule3"android:src="@android:drawable/ic_dialog_alert"
android:layout_marginLeft ="7dp"android:layout_marginTop="3dp"
android:layout_marginBottom ="3dp"android:layout_width="wrap_content"
android:layout_height ="wrap_content"/>
</LinearLayout>
</LinearLayout>
2.2 TestView.java
publicclassTestViewextends ActivityGroup{
privateScrollViewcontainer=null ;
@Override
protectedvoid onCreate(BundlesavedInstanceState){
super .onCreate(savedInstanceState);
//隐藏标题栏
requestWindowFeature(Window.FEATURE_NO_TITLE);
//设置视图
setContentView(R.layout.layout);
container= (ScrollView)findViewById(R.id.containerBody);
//模块1
ImageViewbtnModule1= (ImageView)findViewById(R.id.btnModule1);
btnModule1.setOnClickListener(new OnClickListener(){
@Override
publicvoid onClick(Viewv){
container.removeAllViews();
container.addView(getLocalActivityManager().startActivity(
"Module1" ,
newIntent(TestView.this,ModuleView1.class )
.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP))
.getDecorView());
}
})
//模块2
ImageViewbtnModule2= (ImageView)findViewById(R.id.btnModule2);
btnModule2.setOnClickListener(new OnClickListener(){
@Override
publicvoid onClick(Viewv){
container.removeAllViews();
container.addView(getLocalActivityManager().startActivity(
"Module2" ,
newIntent(TestView.this,ModuleView2.class )
.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP))
.getDecorView());
}
});
//模块3
ImageViewbtnModule3= (ImageView)findViewById(R.id.btnModule3);
btnModule3.setOnClickListener(new OnClickListener(){
@Override
publicvoid onClick(Viewv){
container.removeAllViews();
container.addView(getLocalActivityManager().startActivity(
"Module3" ,
newIntent(TestView.this,ModuleView3.class )
.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP))
.getDecorView());
}
});
}
}
代码说明:
a). ModuleView1、ModuleView2、ModuleView3分别继承自Activity。
b). 想动态改变标题可以通过cust_title获取TextView进行设置。
注释:想代码实现将子activity的所用的layout加入到主activty中的layout
public class FormActivity extends ActivityGroup {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.form);
LocalActivityManager m = getLocalActivityManager();
Intent intent = new Intent().setClass(this, ContactFieldActivity.class);
Window w = m.startActivity("tratat", intent);
View v = w.getDecorView();
LinearLayout container = (LinearLayout)findViewById(R.id.fieldsContainer);
container.addView(v);
}
}
范例2ActivityGroup + GridView 实现Tab分页标签
http://blog.csdn.net/hellogv/article/details/6057174
2) tabActivity
tabActivity继承自Activity,其内部定义好了TabHost,可以通过getTabHost()获取。TabHost 包含了两种子元素:一些可以自由选择的Tab 和tab对应的内容tabContentto,在Layout的<TabHost>下它们分别对应 TabWidget和FrameLayout。 使用TabActivity可以让同一个界面容纳更多的内容。我们将按照Standup Timer里的TeamDetailsActivity来讲述TabActivity的使用。在该例中,包含了两个Tab一个用于展示team的统计信息,一个用于展示team所参加的会议的列表(这是一个ListView)。创建Layout
这里需要注意的是不管你是使用TabActivity 还是自定义TabHost,都要求以TabHost作为XML布局文件的根。 <? xml version="1.0" encoding="utf-8" ?>< TabHost xmlns:android ="http://schemas.android.com/apk/res/android"
android:id ="@android:id/tabhost" android:layout_width ="fill_parent"
android:layout_height ="fill_parent" >
< LinearLayout android:orientation ="vertical"
android:layout_width ="fill_parent" android:layout_height ="fill_parent" >
< TabWidget android:id ="@android:id/tabs"
android:layout_width ="fill_parent" android:layout_height ="wrap_content" />
< FrameLayout android:id ="@android:id/tabcontent"
android:layout_width ="fill_parent" android:layout_height ="fill_parent" >
<!-- 省略部分代码 -->
< TextView android:id ="@+id/no_team_meetings"
android:textSize ="18sp" android:layout_width ="fill_parent"
android:layout_height ="fill_parent" />
< TextView android:id ="@+id/no_team_meeting_stats"
android:textSize ="18sp" android:layout_width ="fill_parent"
android:layout_height ="fill_parent" />
</ FrameLayout >
</ LinearLayout >
</ TabHost >
通常我们采用线性布局所以<TabHost> 的子元素是 <LinearLayout>。<TabWidger>对应Tab。<FrameLayout>则用于包含Tab需要展示的内容。需要注意的是<TabWidger> 和<FrameLayout>的Id 必须使用系统id,分别为android:id/tabs 和 android:id/tabcontent 。因为系统会使用者两个id来初始化TabHost的两个实例变量(mTabWidget 和 mTabContent)。
编写Java代码
我们可以采用两种方法编写标签页:一种是继承TabActivity ,然后使用getTabHost()获取TabHost对象;第二种方法是使用自定的TabHost在布局文件上<TabHost>的自定义其ID,然后通过findViewById(),方法获得TabHost对象。 本文中采用继承TabActivity的方法。 private void createTabs() {TabHost tabhost = getTabHost();
tabhost.addTab(tabhost.newTabSpec( " stats_tab " ).
setIndicator( this .getString(R.string.stats)).
setContent(createMeetingDetails(team)));
tabhost.addTab(tabhost.newTabSpec( " meetings_tab " ).
setIndicator( this .getString(R.string.meetings)).
setContent(createMeetingList()));
getTabHost().setCurrentTab( 0 );
}
Java代码中我们首先需要做的是获取TabHost对象,可以通过TabActivtiy里的getTabHsot()方法。如果是自定义TabHost,在添加Tabs前应该调用 setUp()方法。 mTabHost = (TabHost)findViewById(R.id.tabhost);
mTabHost.setup();
mTabHost.addTab(TAB_TAG_1, " Hello, world! " , " Tab 1 " );
SDK上的原文:
Call setup() before adding tabs if loading TabHost using findViewById().However:You do not need to call setup() after getTabHost() inTabActivity
.
接着向TabHost添加tabs.即调用tabHost.addTab(TabSpec) 方法。 TabSpec主要包含了setIndicator 和 setContent 方法,通过这两个方法来指定Tab 和 TanContent。 TabSpec 通过 .newTabSpec(String tag )来创建实例。实例化后对其属性进行设置。setIndicator()设置tab,它有3个重载的函数
- public TabHost.TabSpec setIndicatior(CharSwquence label,Drawable icon).指定tab的标题和图标。
- public TabHost.TabSpec (View view)通过View来自定义tab
- public TabHost.TabSpec(CharSequence label) 指定tab的标题,此时无图标。
- public TabHost.TabSpec setContent(TabHost.TabContentFactory )
- public TabHost.TabSpec setContent(int ViewId)
- public TabHost.TabSpec setContent(Intent intent)
return new TabHost.TabContentFactory() {
@Override
public View createTabContent(String tag) {
//设置View
setStatsTabContent();
return findViewById(R.id.teamStats);
}
};
}
private TabHost.TabContentFactory createMeetingList()
{
return new TabHost.TabContentFactory() {
@Override
public View createTabContent(String tag) {
meetingListAdapter = createMeetingListAdapter();
meetingList.setAdapter(meetingListAdapter);
return meetingList;
}
};
}
事先声明好的 private ListView meetingList = null ;
private ArrayAdapter < String > meetingListAdapter = null ;
我们也可以让TabActivity去实现TabContentFactory 接口 public class Tabs2 extends TabActivity implements TabHost.TabContentFactory
然后在TabActiviy类中实现createTabContent方法 @Override
public View createTabContent(String tag) {
final TextView tv = new TextView( this );
tv.setText( " Content for tab with tag " + tag);
return tv;
}
setStatsTabContent();方法
最后将TabSpec 添加到 TabHost上,即tabHost.addTab(tabSpec)。我们发现TabSpec 的setIndicator 和 setContent 方法返回的都是 TabSpec 自身所以可以使用窜的方式编写代码:
tabhost.addTab(tabhost.newTabSpec( " stats_tab " )
.setIndicator( this .getString(R.string.stats))
.setContent(createMeetingDetails(team))); 3)两者之间的差异性
在一个主界面中做Activity切换一般都会用TabActivity,使用方便,Activity互相之间相对独立,但是可定制性不强,而且修改起来很麻烦。当然也可以把layout分开,把逻辑代码全写在主界面的逻辑代码中,但是很明显可维护性相当差,这里通过ActivityGroup来解决这个问题。
Android ActivityGroup认识
1,ActivityGroup是继承自Activity的一个类,当然拥有activity类的所有特性,从字面上看,activityGroup就是一组activity的集合,的确该类就是activity的集合。
2,ActivityGroup类所有操作都是用其内部成员LocalActivityManager来实现的。
3,虽然在activityGroup内部已经使用一个数组和一个hash表存储了每个activity和其对应的string id,但是却不提供在组内的back操作,因此back操作需要我们手动完成,同时在组内不提共activity之间的转换,因此我们需要记住activity的先后存放顺序,然后可以根据顺序完成我们想要的操作。
4,在组内部每个字activity都是组的子activity,因此在每个activity内部调用个头parent()就可以实现parent都是activityGroup。
5,使用activity和tablayout可以很好地解决实际问题,比使用gradview和activityGroup和scrollview更好使用。前者我们只要使用activityGroup管理一个tab的activity就可以了,后者我们不仅要考虑gradview中各个view点击的切换,而且是所有的activity都共用scrollview的位置来使用activityGroup,也就是activity没有按照gradview的各个view分类。
更多相关文章
- android获取时间差的方法
- 浅析Android中的visibility属性
- App优化之提升你的App启动速度之实例挑战
- Android(安卓)4.4源码编译环境升级gcc版本的方法
- android 内置的短信等调用代码
- Android实现异步加载图片 ListView
- Android(安卓)调用Remote Service方法
- Android中Cursor 的一些方法
- Android(安卓)第三方 Jar 的 Link 与问题:Unable to execute dex