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

简介:

该文章主要改动于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());
		}
	}

改动后的项目源代码
相关文章
|
16天前
|
移动开发 Java Android开发
构建高效Android应用:探究Kotlin与Java的性能差异
【4月更文挑战第3天】在移动开发领域,性能优化一直是开发者关注的焦点。随着Kotlin的兴起,其在Android开发中的地位逐渐上升,但关于其与Java在性能方面的对比,尚无明确共识。本文通过深入分析并结合实际测试数据,探讨了Kotlin与Java在Android平台上的性能表现,揭示了在不同场景下两者的差异及其对应用性能的潜在影响,为开发者在选择编程语言时提供参考依据。
|
17天前
|
数据库 Android开发 开发者
构建高效Android应用:Kotlin协程的实践指南
【4月更文挑战第2天】随着移动应用开发的不断进步,开发者们寻求更流畅、高效的用户体验。在Android平台上,Kotlin语言凭借其简洁性和功能性赢得了开发社区的广泛支持。特别是Kotlin协程,作为一种轻量级的并发处理方案,使得异步编程变得更加简单和直观。本文将深入探讨Kotlin协程的核心概念、使用场景以及如何将其应用于Android开发中,以提高应用性能和响应能力。通过实际案例分析,我们将展示协程如何简化复杂任务,优化资源管理,并为最终用户提供更加流畅的体验。
|
17天前
|
开发框架 安全 Android开发
探索安卓系统的新趋势:智能家居应用的蓬勃发展
随着智能家居概念的兴起,安卓系统在智能家居应用领域的应用日益广泛。本文将探讨安卓系统在智能家居应用开发方面的最新趋势和创新,以及其对用户生活的影响。
13 2
|
20天前
|
缓存 监控 Java
构建高效Android应用:从优化用户体验到提升性能
在竞争激烈的移动应用市场中,为用户提供流畅和高效的体验是至关重要的。本文深入探讨了如何通过多种技术手段来优化Android应用的性能,包括UI响应性、内存管理和多线程处理。同时,我们还将讨论如何利用最新的Android框架和工具来诊断和解决性能瓶颈。通过实例分析和最佳实践,读者将能够理解并实施必要的优化策略,以确保他们的应用在保持响应迅速的同时,还能够有效地利用系统资源。
|
21天前
|
Java Android开发
Android 开发获取通知栏权限时会出现两个应用图标
Android 开发获取通知栏权限时会出现两个应用图标
12 0
|
25天前
|
编解码 算法 Java
构建高效的Android应用:内存优化策略详解
随着智能手机在日常生活和工作中的普及,用户对移动应用的性能要求越来越高。特别是对于Android开发者来说,理解并实践内存优化是提升应用程序性能的关键步骤。本文将深入探讨针对Android平台的内存管理机制,并提供一系列实用的内存优化技巧,以帮助开发者减少内存消耗,避免常见的内存泄漏问题,并确保应用的流畅运行。
|
18天前
|
Java Android开发 开发者
构建高效Android应用:Kotlin协程的实践与优化
在响应式编程范式日益盛行的今天,Kotlin协程作为一种轻量级的线程管理解决方案,为Android开发带来了性能和效率的双重提升。本文旨在探讨Kotlin协程的核心概念、实践方法及其在Android应用中的优化策略,帮助开发者构建更加流畅和高效的应用程序。通过深入分析协程的原理与应用场景,结合实际案例,本文将指导读者如何优雅地解决异步任务处理,避免阻塞UI线程,从而优化用户体验。
|
23天前
|
Java 编译器 Android开发
构建高效Android应用:探究Kotlin与Java的性能差异
在开发高性能的Android应用时,选择合适的编程语言至关重要。近年来,Kotlin因其简洁性和功能性受到开发者的青睐,但其性能是否与传统的Java相比有所不足?本文通过对比分析Kotlin与Java在Android平台上的运行效率,揭示二者在编译速度、运行时性能及资源消耗方面的具体差异,并探讨在实际项目中如何做出最佳选择。
17 4
|
1天前
|
缓存 移动开发 Android开发
构建高效Android应用:从优化用户体验到提升性能表现
【4月更文挑战第18天】 在移动开发的世界中,打造一个既快速又流畅的Android应用并非易事。本文深入探讨了如何通过一系列创新的技术策略来提升应用性能和用户体验。我们将从用户界面(UI)设计的简约性原则出发,探索响应式布局和Material Design的实践,再深入剖析后台任务处理、内存管理和电池寿命优化的技巧。此外,文中还将讨论最新的Android Jetpack组件如何帮助开发者更高效地构建高质量的应用。此内容不仅适合经验丰富的开发者深化理解,也适合初学者构建起对Android高效开发的基础认识。
2 0
|
1天前
|
移动开发 Android开发 开发者
构建高效Android应用:采用Kotlin进行内存优化的策略
【4月更文挑战第18天】 在移动开发领域,性能优化一直是开发者关注的焦点。特别是对于Android应用而言,由于设备和版本的多样性,确保应用流畅运行且占用资源少是一大挑战。本文将探讨使用Kotlin语言开发Android应用时,如何通过内存优化来提升应用性能。我们将从减少不必要的对象创建、合理使用数据结构、避免内存泄漏等方面入手,提供实用的代码示例和最佳实践,帮助开发者构建更加高效的Android应用。
5 0