因为用于展示短信记录的是一个ListView,但是为了方便,可以直接继承自ListFragment,就可以免去写ListView对应的布局了,只需要写其item对应的布局即可。

item_sended_msg.xml

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="@drawable/sms_item"
android:padding="16dp"> <TextView
android:id="@+id/id_tv_sended_content"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:padding="8dp"/> <com.example.just.festival_sms.view.FlowLayout
android:id="@+id/id_fl_sended_contacts"
android:layout_width="match_parent"
android:layout_height="match_parent">
</com.example.just.festival_sms.view.FlowLayout> <LinearLayout
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="right"
android:orientation="horizontal"> <TextView
android:id="@+id/id_tv_fes"
android:background="@drawable/tag_bg"
android:layout_marginRight="8dp"
android:layout_width="wrap_content"
android:layout_height="wrap_content" /> <TextView
android:id="@+id/id_tv_date"
android:layout_width="wrap_content"
android:layout_height="wrap_content" />
</LinearLayout> </LinearLayout>

还有tag.xml,显示联系人的layout

<TextView xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textColor="#0ffcdd"
android:background="@drawable/tag_bg"
android:layout_margin="4dp"> </TextView>

以及tag_bg.xml,用于显示联系人和节日的TextView的background 

<shape xmlns:android="http://schemas.android.com/apk/res/android">

    <solid android:color="#ffffff"></solid>

    <stroke android:width="2dp" android:color="#0ffcdd"></stroke>

    <corners android:radius="8dp"></corners>

    <padding android:left="8dp" android:right="8dp" android:top="2dp" android:bottom="2dp"></padding>
</shape>

关于shape中的属性: 
http://www.oschina.net/question/166763_34833?fromerr=FBMFjTg7


SmsHistoryFragment.Java

需要注意,某些导入的包可能有多种选择,应该都是v4下的

public class SmsHistoryFragment extends ListFragment {
private static final int LOADER_ID =; private LayoutInflater mInflater;
private CursorAdapter mCursorAdapter; @Override
public void onViewCreated(View view, Bundle savedInstanceState) {
super.onViewCreated(view, savedInstanceState); mInflater=LayoutInflater.from(getActivity()); initLoader(); setupListAdapter(); } private void setupListAdapter() {
mCursorAdapter=new CursorAdapter(getActivity(),null,false) { //并不是每次都被调用的,它只在实例化view的时候调用,数据增加的时候也会调用
//但是在重绘(比如修改条目里的TextView的内容)的时候不会被调用
@Override
public View newView(Context context, Cursor cursor, ViewGroup parent) {
View view=mInflater.inflate(R.layout.item_sended_msg,parent,false);//注意是"包名.R"
return view;
} //在绘制Item之前一定会调用bindView方法,它在重绘的时候也同样被调用
@Override
public void bindView(View view, Context context, Cursor cursor) {
TextView tvContent= (TextView) view.findViewById(R.id.id_tv_sended_content);
FlowLayout flContacts= (FlowLayout) view.findViewById(R.id.id_fl_sended_contacts);
TextView tvFes= (TextView) view.findViewById(R.id.id_tv_fes);
TextView tvDate= (TextView) view.findViewById(R.id.id_tv_date); tvContent.setText(cursor.getString(cursor.getColumnIndex(SendedMsg.COLUMN_CONTENT)));
tvFes.setText(cursor.getString(cursor.getColumnIndex(SendedMsg.COLUMN_FESTIVAL_NAME))); //注意这里的date为long,int型会溢出
long date=cursor.getLong(cursor.getColumnIndex(SendedMsg.COLUMN_DATE));
tvDate.setText(parseDate(date)); String names=cursor.getString(cursor.getColumnIndex(SendedMsg.COLUMN_NAMES));
if(TextUtils.isEmpty(names)) {
return;
} //因为ListView的item有复用的可能性,所以每次都要先除去item中的flContacts在上一次使用时添加的view
flContacts.removeAllViews(); for (String name:names.split(",")) {
addTag(name, flContacts);
}
}
}; setListAdapter(mCursorAdapter);
} private String parseDate(long date) {
DateFormat df=new SimpleDateFormat("yyyy-MM-dd HH:mm");
return df.format(date);
} private void addTag(String name,FlowLayout fl) {
TextView tv= (TextView) mInflater.inflate(R.layout.tag,fl,false);
tv.setText(name);
fl.addView(tv);
} private void initLoader() {
getLoaderManager().initLoader(LOADER_ID,null,new LoaderManager.LoaderCallbacks<Cursor>() { //onCreateLoader是一个工厂方法,用来返回一个新的Loader
//LoaderManager将会在它第一次创建Loader的时候调用该方法
@Override
public Loader<Cursor> onCreateLoader(int id, Bundle args) {
CursorLoader loader=new CursorLoader(getActivity(), SmsProvider.URI_SMS_ALL,null,null,null,null);
return loader;
} //onLoadFinished方法将在Loader创建完毕的时候自动调用
//在数据更新的时候也会调用
@Override
public void onLoadFinished(Loader<Cursor> loader, Cursor data) {
if(loader.getId()==LOADER_ID) {
mCursorAdapter.swapCursor(data);//更新mCursorAdapter的Cursor
}
} @Override
public void onLoaderReset(Loader<Cursor> loader) {
mCursorAdapter.swapCursor(null);
}
});
}
}

SmsHistoryFragment中,涉及到了LoaderManager与Loader,而关于这个知识点,推荐一篇优质博文,看完之后肯定会有所收获的。 
http://blog.csdn.net/murphykwu/article/details/35287303

还是回到本文中来,因为Android的设计之中,任何耗时的操作都不能放在UI主线程之中。所以类似于网络操作等等耗时的操作都需要使用异步的实现。而在ContentProvider之中,也有可能存在耗时的操作(当查询的数据量很大的时候),这个时候我们也需要使用异步的调用来完成数据的查。当使用异步的query的时候,我们就需要使用LoaderManager了。使用LoaderManager就可以在不阻塞UI主线程的情况下完成数据的加载。

还记得上一篇文章最后提及的两行代码吗?就是为了实时的更新数据用的,更进一步说,是为了同步ListView中显示数据(历史记录),两行代码缺一不可。(检测数据源是Loader的工作,Loader也会执行实际的同步载入操作,而在这之中,那两句代码就起到了一定的作用

另外,关于还有关于CursorAdapter,可以去看看这篇博文: 
http://www.bubuko.com/infodetail-734550.html

对了,小编在刚开始解读源码的时候有一个疑问,mCursorAdapter是在setupListAdapter中初始化的,但是initLoader中就用到了mCursorAdapter,为什么程序可以运行(因为这里是先initLoader再setupListAdapter)(但是实际中也可以先setupListAdapter再initLoader) 
后来实际测试了一下,发现部分具体的流程如下: 
onCreateLoader() -> 初始化mCursorAdapter ->setListAdapter(mCursorAdapter) 
-> onLoadFinished() 
这是因为getLoaderManager().initLoader()中传入的第三个参数是一个回调接口。

Android 节日短信送祝福(功能篇:2-短信历史记录Fragment的编写)的更多相关文章

  1. Android 节日短信送祝福(功能篇:1-数据库操作类与自定义ContentProvider)

    首先,还是展示一下部分目录结构:  在节日短信送祝福的功能实现方面,为了能够方便直观展示实现过程,小编我以Java文件为基础,一个一个来展示,免得到时候这个java文件写点,一下又跳到另外一个java ...

  2. Android 节日短信送祝福(UI篇:3-选择短信与发送短信的Activity的实现)

    一.ChooseMsgActivity的实现 1.布局文件 <RelativeLayout xmlns:android="http://schemas.android.com/apk/ ...

  3. Android service 服务的应用之电话监听器以及短信监听器

    首先建立一个项目工程文件,如下图所示:

  4. SpringBoot + Spring Security 学习笔记(五)实现短信验证码+登录功能

    在 Spring Security 中基于表单的认证模式,默认就是密码帐号登录认证,那么对于短信验证码+登录的方式,Spring Security 没有现成的接口可以使用,所以需要自己的封装一个类似的 ...

  5. Android手机使用广播监听手机收到的短信

    我们使用的Android手机在收到短信的时候会发出一条系统广播.该条广播中存放着接收到的短信的详细信息.本文将详细介绍如何通过动态注册广播来监听短信. 注册广播有两种方式,一种是动态注册,另一种是静态 ...

  6. GPRS的短信和打电话功能

    短信功能: 发短信设置文本格式就可以了:但收短信可能收到的是乱码,需要编写解码程序才可以: 关于打电话单片机复位功能: 首先要建立黑白名单制度过滤手机号,只运行白名单的手机对的单片机打电话:其它的不响 ...

  7. 客户注册功能,发短信功能分离 通过ActiveMQ实现

    客户注册功能,发短信功能分离 通过ActiveMQ 配置链接工厂, 配置session缓存工厂(引入链接工厂) 2.配置模板对象JmsTemplate 引入缓存工厂    指定消息模式(队列,发布和订 ...

  8. 项目一:在线下单(补充) activeMQ使用(重点) 重构客户注册功能,发短信功能分离

    1 课程计划 1.在线下单(补充) 2.activeMQ使用(重点) n 简介和安装 n activeMQ入门案例 n spring整合activeMQ应用 3.重构客户注册功能,发短信功能分离 n  ...

  9. 基于EasyDarwin EasyPusher实现Android手机直播推送功能

    EasyPusher直播推送在之前就已经稳定支持了Windows.Linux.ARM上的RTSP直播推送功能,配合EasyDarwin开源流媒体服务器,延时基本在1s以内,这个技术方案经过一年多时间, ...

随机推荐

  1. <meta-data>的使用

    在AndroidManifest.xml中,<meta-data>元素可以作为子元素,被包含在<activity>.<application> .<servi ...

  2. elasticsearch java 客户端之Client简介

    elasticsearch通过构造一个client对外提供了一套丰富的java调用接口.总体来说client分为两类cluster信息方面的client及数据(index)方面的client.这两个大 ...

  3. java 三次样条插值 画光滑曲线 例子

    java 三次样条插值 画光滑曲线 例子 主要是做数值拟合,根据sin函数采点,取得数据后在java中插值并在swing中画出曲线,下面为截图  不光滑和光滑曲线前后对比:    代码: 执行类: p ...

  4. ACM 中JAVA的应用

    原文地址:http://www.cppblog.com/vontroy/archive/2010/05/24/116233.html 先说一下Java对于ACM的一些优点吧: (1) 对于熟悉C/C+ ...

  5. BZOJ5020: [THUWC 2017]在美妙的数学王国中畅游(LCT,泰勒展开,二项式定理)

    Description 数字和数学规律主宰着这个世界.   机器的运转,   生命的消长,   宇宙的进程,   这些神秘而又美妙的过程无不可以用数学的语言展现出来.   这印证了一句古老的名言:   ...

  6. 洛谷—— P1080 国王游戏

    https://www.luogu.org/problem/show?pid=1080 题目描述 恰逢 H 国国庆,国王邀请 n 位大臣来玩一个有奖游戏.首先,他让每个大臣在左.右手上面分别写下一个整 ...

  7. ZOJ 题目3587 Marlon&#39;s String(KMP)

    Marlon's String Time Limit: 2 Seconds      Memory Limit: 65536 KB Long long ago, there was a coder n ...

  8. Java Web学习总结(13)——JSP入门

    一.什么是JSP? JSP全称是Java Server Pages,它和servle技术一样,都是SUN公司定义的一种用于开发动态web资源的技术. JSP这门技术的最大的特点在于,写jsp就像在写h ...

  9. SQl 事务增加数据

    连SQl简单地事务处理多表的情况 begin transaction declare @error int declare @QID int set @error = 0 insert into HW ...

  10. python获取序列中最大值

    test =[ [1, 2, 3], [4, 5, 6], [7, 8, 9]]   #这个就可以看做是二维数组了,直接创建print(test)print(test[:][1])           ...