通过定义BaseActivity来实现项目中代码重用,重写setContentView实现多个Activity部分UI布局相同

安卓应用中不同的activity一般都具有相同的地方,最典型的是标题栏(titlebar),我们只需在每个activity中调用setTitle就可以得到一个除了标题文字不同,其他完全相同的标题栏。

系统已经为我们引进了titlebar这样的功能,但是如果我们还需要一个类似titlebar这样容易copy外形的bottombar呢?

当然是否需要一个bottombar是个问题,我要说的其实是如果我们想让activity共享一部分UI的情况下该怎么做。

很直观的我们会写一个activity的子类,然后将公共部分的UI在这个子类activity中实现,命名为BaseActivity,最后所有要共享此部分UI的activity都继承这个BaseActivity。

思路是这样,但是究竟该如何写这个BaseActivity呢,注意上面蓝色那句话,公共部分的UI如果是通过setContentView来渲染的话那该如果处理BaseActivity子类中其独有的UI呢,合理的情况是在BaseActivity子类中调用setContentView来显示自己独有的界面,但是两次调用setContentView总有一次是会被覆盖的。

现在的情况是,我们想得到公共的UI,但没办法把公共部分和独有部分的UI分开来处理。解决问题的办法是了解activity的布局到底是如何组成的,setContentView做了些什么。

通过定义BaseActivity来实现项目中代码重用,重写setContentView实现多个Activity部分UI布局相同_第1张图片

一、DecorView为整个Window界面的最顶层View。

二、DecorView只有一个子元素为LinearLayout。代表整个Window界面,包含通知栏,标题栏,内容显示栏三块区域。

三、LinearLayout里有两个FrameLayout子元素。

(20)为标题栏显示界面。只有一个TextView显示应用的名称。也可以自定义标题栏,载入后的自定义标题栏View将加入FrameLayout中。

(21)为内容栏显示界面。就是setContentView()方法载入的布局界面,加入其中。

所以要实现activity具有公共部分的UI,重写setContentView()方法:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
import android.app.Activity;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.view.View.OnClickListener;
import android.view.ViewGroup.LayoutParams;
import android.widget.LinearLayout;
import android.widget.TextView;
public class BaseActivity extends Activity {
private TextView mTitleTx;
private View mBack;
private LinearLayout contentLayout;
@Override
protected void onCreate(Bundle savedInstanceState) {
super .onCreate(savedInstanceState);
initContentView();
initTitleBar();
}
public void initTitleBar(){
mTitleTx = (TextView)findViewById(R.id.titlebar_title);
mBack = findViewById(R.id.titlebar_left);
mBack.setOnClickListener( new OnClickListener(){
@Override
public void onClick(View view){
finish();
}
});
}
private void initContentView() {
ViewGroup content = (ViewGroup) findViewById(android.R.id.content);
content.removeAllViews();
contentLayout= new LinearLayout( this );
contentLayout.setOrientation(LinearLayout.VERTICAL);
content.addView(contentLayout);
LayoutInflater.from( this ).inflate(R.layout.common_title_bar, contentLayout, true );
}
@Override
public void setContentView(int layoutResID) {
//View customContentView = LayoutInflater.from(this).inflate(layoutResID,null);
/*this is the same result with
View customContentView = LayoutInflater.from(this).inflate(layoutResID,contentLayout, false);
*/
//contentLayout.addView(customContentView,LayoutParams.FILL_PARENT, LayoutParams.FILL_PARENT);
LayoutInflater.from( this ).inflate(layoutResID, contentLayout, true );
}
@Override
public void setContentView(View customContentView) {
contentLayout.addView(customContentView);
}
@Override
public void setTitle(CharSequence title) {
mTitleTx.setText(title);
}
}

(ViewGroup) findViewById(android.R.id.content)可以获得挂在一个activity内容部分LinearLayout。在这个LinearLayout中添加一个LinearLayout

1
contentLayout= new LinearLayout( this );
1
2
contentLayout.setOrientation(LinearLayout.VERTICAL);
content.addView(contentLayout);

作为新的内容区域。

接下来将公共部分的UI添加进新的内容区域。

1
LayoutInflater.from( this ).inflate(R.layout.common_title_bar, contentLayout, true );

我这里是自定义了一个标题栏作为公共部分,其实如果是标题栏可以不如此麻烦直接用原生的titlebar就行了。

然后重写setContentView,将子类的内容区域从原本该直接挂在到android.R.id.content上面改为挂在到这个新的内容区域。代码如下:

1
2
3
4
5
6
@Override
public void setContentView(int layoutResID) {
LayoutInflater.from( this ).inflate(layoutResID, contentLayout, true );

你可能感兴趣的:(安卓)