该文章主要改动于CSDN某大神的一篇文章,本人认为这篇文章的面向对象非常透彻,以下分享例如以下可学习的几点:

Android应用经典主界面框架之中的一个:仿QQ (使用Fragment, 附源代码)

1.通过&符号实现计算优化:(后来通过问同事,说是计算机通过位运算 效率比平时的switch效率高,并解说了该算法的原理。)

public class Constant {

	public static final int SIGN_FRAGMENT_MESSAGE=0x01 <<1;
public static final int SIGN_FRAGMENT_CONTACTS=0x01 <<2;
public static final int SIGN_FRAGMENT_NEWS=0x01 <<3;
public static final int SIGN_FRAGMENT_SETTENGS=0x01 <<4; }
@Override
public void onClickCallBack(int itemID) {
String tag = "";
if ((itemID & Constant.SIGN_FRAGMENT_MESSAGE) != 0) {
tag = Constant.STR_FRAGMENT_MESSAGE;
} else if ((itemID & Constant.SIGN_FRAGMENT_CONTACTS) != 0) {
tag = Constant.STR_FRAGMENT_CONTACTS;
} else if ((itemID & Constant.SIGN_FRAGMENT_NEWS) != 0) {
tag = Constant.STR_FRAGMENT_NEWS;
} else if ((itemID & Constant.SIGN_FRAGMENT_SETTENGS) != 0) {
tag = Constant.STR_FRAGMENT_SETTINGS;
}
mHeaderPanelLayout.setText(tag);
setTabSection(tag);
}

2.通过onLayout对底部栏中间的button进行“动态”调整

@Override
protected void onLayout(boolean changed, int l, int t, int r, int b) {
super.onLayout(changed, l, t, r, b);
layoutItem(l, t, r, b);
} private void layoutItem(int left, int top, int right, int bottom) {
int allChildWidth=0;
int num=getChildCount();
for (int i = 0; i < num; i++) {
allChildWidth+=getChildAt(i).getWidth();
}
int absoluteWidth=right-left-getPaddingLeft()-getPaddingRight();
int blankWidth=(absoluteWidth-allChildWidth)/(num-1);
//设置第2 3个button的间距
LayoutParams params1=(LayoutParams) mContactsBtn.getLayoutParams();
params1.leftMargin=blankWidth;
mContactsBtn.setLayoutParams(params1);
LayoutParams params2=(LayoutParams) mNewsBtn.getLayoutParams();
params2.leftMargin=blankWidth;
mNewsBtn.setLayoutParams(params2);
}

3.两种实例化布局的应用:

1)通过layoutInflater.

	public ImageText(Context context, AttributeSet attrs) {
super(context, attrs);
LayoutInflater.from(context).inflate(R.layout.image_text_layout, this,true);
mImageView=(ImageView) findViewById(R.id.iv_imgae_text);
mTextiew=(TextView) findViewById(R.id.tv_imgae_text);
}

2)通过onFinishInflater()

<?xml version="1.0" encoding="utf-8"?>
<org.lean.ui.BottomPanelLayout
xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:background="#FFF3F3F3"
android:paddingLeft="20dp"
android:paddingRight="20dp"
android:layout_alignParentBottom="true" > <org.lean.ui.ImageText
android:id="@+id/message_btn"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentLeft="true" /> <org.lean.ui.ImageText
android:id="@+id/contacts_btn"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_toRightOf="@id/message_btn" /> <org.lean.ui.ImageText
android:id="@+id/news_btn"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_toRightOf="@id/contacts_btn" /> <org.lean.ui.ImageText
android:id="@+id/settings_btn"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentRight="true" /> </org.lean.ui.BottomPanelLayout>
@Override
protected void onFinishInflate() {
super.onFinishInflate();
mMessageBtn=(ImageText) findViewById(R.id.message_btn);
mContactsBtn=(ImageText) findViewById(R.id.contacts_btn);
mNewsBtn=(ImageText) findViewById(R.id.news_btn);
mSettingsBtn=(ImageText) findViewById(R.id.settings_btn);
initClickEvent();
}

4.代理实现数据传递(IOS中最经常使用的一种设计模式)

public class BottomPanelLayout extends RelativeLayout implements OnClickListener{

	private BottomPanelCallBackProtocal mCallBackProtocal;

	//代理协议
public void setCallBackProtocal(BottomPanelCallBackProtocal callBackProtocal) {
this.mCallBackProtocal = callBackProtocal;
} public interface BottomPanelCallBackProtocal{
public void onClickCallBack(int itemID);
} /**
* 1.改动本身样式
* 2.对外声明事件
*/
@Override
public void onClick(View v) {
initBottomPanel();
int index=-1;
switch (v.getId()) {
case R.id.message_btn:
index=Constant.SIGN_FRAGMENT_MESSAGE;
mMessageBtn.setChecked(index);
break;
case R.id.contacts_btn:
index=Constant.SIGN_FRAGMENT_CONTACTS;
mContactsBtn.setChecked(index);
break;
case R.id.news_btn:
index=Constant.SIGN_FRAGMENT_NEWS;
mNewsBtn.setChecked(index);
break;
case R.id.settings_btn:
index=Constant.SIGN_FRAGMENT_SETTENGS;
mSettingsBtn.setChecked(index);
break;
default:
break;
}
if (mCallBackProtocal!=null) {
mCallBackProtocal.onClickCallBack(index);
}
} }
public class MainActivity extends Activity implements
BottomPanelCallBackProtocal {
@Override
public void onClickCallBack(int itemID) {
String tag = "";
if ((itemID & Constant.SIGN_FRAGMENT_MESSAGE) != 0) {
tag = Constant.STR_FRAGMENT_MESSAGE;
} else if ((itemID & Constant.SIGN_FRAGMENT_CONTACTS) != 0) {
tag = Constant.STR_FRAGMENT_CONTACTS;
} else if ((itemID & Constant.SIGN_FRAGMENT_NEWS) != 0) {
tag = Constant.STR_FRAGMENT_NEWS;
} else if ((itemID & Constant.SIGN_FRAGMENT_SETTENGS) != 0) {
tag = Constant.STR_FRAGMENT_SETTINGS;
}
mHeaderPanelLayout.setText(tag);
setTabSection(tag);
} }

5.改动原来Fragment跳转的代码(之前方法ensureTransaction()是在粘贴或者消除的时候都要推断,但作为一个事务,仅仅须要保证事物仅仅有一个開始就可以,而不须要每次都调用)

private void setTabSection(String tag) {
if (TextUtils.equals(tag, currFagTag)) {
return;
}
ensureTransaction();
if (currFagTag != null && !currFagTag.equals("")) {
detachFragment(getFragment(currFagTag));
}
attachFragment(R.id.fragment_panel, getFragment(tag), tag);
commitTransaction();
}
private void ensureTransaction() {
if (mFragmentTransaction == null) {
mFragmentTransaction = mFragmentManager.beginTransaction();
mFragmentTransaction
.setTransition(FragmentTransaction.TRANSIT_FRAGMENT_FADE);
}
}

6.Fragment对Fragment进行跳转并传值的改进。(这里试验从MessageFragment 点击textview跳转到 ContactFragment );

1>在MessageFragment 中

	@Override
public void onActivityCreated(Bundle savedInstanceState) {
super.onActivityCreated(savedInstanceState);
getActivity().findViewById(R.id.msg_tv).setOnClickListener(new OnClickListener() { @Override
public void onClick(View v) {
((MainActivity)getActivity()).setTabSection(Constant.STR_FRAGMENT_CONTACTS);
}
});
}

2>在ContactFragment 中声明数据代理

	//声明一个变量,该变量存储该Fragment所须要的一切參数 当刷新View时手动调用其更新数据
private ContactFragmentCallBack mContactFragmentCallBack; //声明该接口
public interface ContactFragmentCallBack{
//说明该Fragment更新时须要一个String对象
public String getContentStr();
}

3>MessageFragment 实现该代理

public class MessageFragment extends BaseFragment implements ContactFragmentCallBack{
@Override
public String getContentStr() {
return "abc";
}
}

4>在ContactFragment 中回调

	@Override
public void onResume() {
super.onResume();
MainActivity.currFagTag=Constant.STR_FRAGMENT_CONTACTS; //通过取出 存储于上个Fragment中的数据
Fragment f=((MainActivity)getActivity()).getFragment(Constant.STR_FRAGMENT_MESSAGE);
if (f!=null&&f instanceof ContactFragmentCallBack) {
mContactFragmentCallBack=(ContactFragmentCallBack)f;
TextView textView=(TextView) ((MainActivity)getActivity()).findViewById(R.id.contact_tv);
textView.setText(mContactFragmentCallBack.getContentStr());
}
}

改动后的项目源代码

android-改进&lt;&lt;仿QQ&gt;&gt;框架源代码的更多相关文章

  1. android 自定义控件二之仿QQ长按删除

    自定义Dialog 1.先上个效果图:

  2. Android应用经典主界面框架之中的一个:仿QQ (使用Fragment, 附源代码)

    备注:代码已传至https://github.com/yanzi1225627/FragmentProject_QQ 欢迎fork,如今来审视这份代码,非常多地方写的不太好,欢迎大家指正.有时间我会继 ...

  3. Android自定义View实现仿QQ实现运动步数效果

    效果图: 1.attrs.xml中 <declare-styleable name="QQStepView"> <attr name="outerCol ...

  4. android桌面悬浮窗仿QQ手机管家加速效果

    主要还是用到了WindowManager对桌面悬浮进行管理. 需要一个火箭的悬浮窗 一个发射台悬浮窗  ,判断火箭是否放到了发射台,如果放上了,则使用AsyTask 慢慢将火箭的图片往上移.结束后., ...

  5. android 自定义scrollview 仿QQ空间效果 下拉伸缩顶部图片,上拉回弹 上拉滚动顶部title 颜色渐变

    首先要知道  自定义scrollview 仿QQ效果 下拉伸缩放大顶部图片 的原理是监听ontouch事件,在MotionEvent.ACTION_MOVE事件时候,使用不同倍数的系数,重置布局位置[ ...

  6. Android实现高仿QQ附近的人搜索展示

    本文主要实现了高仿QQ附近的人搜索展示,用到了自定义控件的方法 最终效果如下 1.下面展示列表我们可以使用ViewPager来实现(当然如果你不觉得麻烦,你也可以用HorizontalScrollVi ...

  7. Android项目源码界面超级华丽的仿QQ最新版本

    这是一个我们比较熟悉的一款应用,高仿专仿最新QQ应用源码,也是一个高仿QQ最新版本的项目,界面超级华丽,使用了大量的自定义控件,项目里实现了部分功能,例如WIFI-FTP(把手机变成FTP服务端,可以 ...

  8. Android插件化的思考——仿QQ一键换肤,思考比实现更重要!

    Android插件化的思考--仿QQ一键换肤,思考比实现更重要! 今天群友希望写一个关于插件的Blog,思来想去,插件也不是很懂,只是用大致的思路看看能不能模拟一个,思路还是比较重要的,如果你有兴趣的 ...

  9. Android特效专辑(六)——仿QQ聊天撒花特效,无形装逼,最为致命

    Android特效专辑(六)--仿QQ聊天撒花特效,无形装逼,最为致命 我的关于特效的专辑已经在CSDN上申请了一个专栏--http://blog.csdn.net/column/details/li ...

随机推荐

  1. Migration data on SQL

    从表里面导出数据XML: -- export declare @xml xml set @xml = (select * from ( select TableName = 'Schema', xml ...

  2. SQL数据库关键字和列名冲突处理

    在设计SQL数据库的时候可能由于考虑不全,使列名和数据库内关键字冲突,可能导致Query不能被正确识别,对列名要加[]处理.

  3. vs2013出现ISO C++ conformant解决办法

    出现:错误    1    error C4996: 'strnset': The POSIX name for this item is deprecated. Instead, use the I ...

  4. 我用过的Linux命令--关闭防火墙

    关闭防火墙: 防火墙的弄能是限制某一些端口的使用,可以通过linux命令关系它,相应的指令: 查看防火墙信息: #service iptables status 就能看到防火墙的状态: 关闭防火墙: ...

  5. HTTP协议(超文本传输协议)

    一.HTTP的简介: 超文本传输协议. 它是基于TCP连接的(默认端口号是80).所以在传输数据前客户端需向服务器发送连接请求.当服务器同意连接请求,建立连接后才可以发送数据报文. 二.HTTP的报文 ...

  6. centos6.5 升级python 到 python 2.7.11 安装 pip

    1.首先官方下载源码,然后安装(./configure,make all,make install,make clean,make distclean) 注意:需要先安装zlib-devel,open ...

  7. Loadrunner11点击录制脚本无响应,IE页面弹不出——解决方案汇总

    以前用Loadrunner的时候都没有遇到过这个问题,后来将服务器重装系统(win7)后,重新安装Loadrunner11,浏览器版本刚开始为IE11,后来降为IE8,IE访问部署在虚拟机里的平台能正 ...

  8. android application 的使用

    参考http://oyeal.iteye.com/blog/941183 由于intent能够传送的对象类型非常有限  因此有些很多类都要用到的变量我们放在Application中  很像web中的s ...

  9. angularjs学习总结(快速预览版)

    对html标签的增强 -> 指令 指令的本质是什么 声明的方式调用相应的脚本,实现一些操作,声明的所在的dom就是脚本的执行上下文? 自定义标签 -- 标签指令自定义属性 -- 属性指令特定格式 ...

  10. oc拨打电话方法

    1,这种方法,拨打完电话回不到原来的应用,会停留在通讯录里,而且是直接拨打,不弹出提示NSMutableString * str=[[NSMutableString alloc] initWithFo ...