重写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做了些什么。
一、DecorView为整个Window界面的最顶层View。
二、DecorView只有一个子元素为LinearLayout。代表整个Window界面,包含通知栏,标题栏,内容显示栏三块区域。
三、LinearLayout里有两个FrameLayout子元素。
(20)为标题栏显示界面。只有一个TextView显示应用的名称。也可以自定义标题栏,载入后的自定义标题栏View将加入FrameLayout中。
(21)为内容栏显示界面。就是setContentView()方法载入的布局界面,加入其中。
所以要实现activity具有公共部分的UI,重写setContentView()方法:
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
contentLayout=new LinearLayout(this);
contentLayout.setOrientation(LinearLayout.VERTICAL);
content.addView(contentLayout);
作为新的内容区域。
接下来将公共部分的UI添加进新的内容区域。
LayoutInflater.from(this).inflate(R.layout.common_title_bar, contentLayout, true);
我这里是自定义了一个标题栏作为公共部分,其实如果是标题栏可以不如此麻烦直接用原生的titlebar就行了。
然后重写setContentView,将子类的内容区域从原本该直接挂在到android.R.id.content
上面改为挂在到这个新的内容区域。代码如下:
@Override
public void setContentView(int layoutResID) {
LayoutInflater.from(this).inflate(layoutResID, contentLayout, true);
}