Android 节日短信送祝福(功能篇:2-短信历史记录Fragment的编写)
因为用于展示短信记录的是一个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的编写)的更多相关文章
- Android 节日短信送祝福(功能篇:1-数据库操作类与自定义ContentProvider)
首先,还是展示一下部分目录结构: 在节日短信送祝福的功能实现方面,为了能够方便直观展示实现过程,小编我以Java文件为基础,一个一个来展示,免得到时候这个java文件写点,一下又跳到另外一个java ...
- Android 节日短信送祝福(UI篇:3-选择短信与发送短信的Activity的实现)
一.ChooseMsgActivity的实现 1.布局文件 <RelativeLayout xmlns:android="http://schemas.android.com/apk/ ...
- Android service 服务的应用之电话监听器以及短信监听器
首先建立一个项目工程文件,如下图所示:
- SpringBoot + Spring Security 学习笔记(五)实现短信验证码+登录功能
在 Spring Security 中基于表单的认证模式,默认就是密码帐号登录认证,那么对于短信验证码+登录的方式,Spring Security 没有现成的接口可以使用,所以需要自己的封装一个类似的 ...
- Android手机使用广播监听手机收到的短信
我们使用的Android手机在收到短信的时候会发出一条系统广播.该条广播中存放着接收到的短信的详细信息.本文将详细介绍如何通过动态注册广播来监听短信. 注册广播有两种方式,一种是动态注册,另一种是静态 ...
- GPRS的短信和打电话功能
短信功能: 发短信设置文本格式就可以了:但收短信可能收到的是乱码,需要编写解码程序才可以: 关于打电话单片机复位功能: 首先要建立黑白名单制度过滤手机号,只运行白名单的手机对的单片机打电话:其它的不响 ...
- 客户注册功能,发短信功能分离 通过ActiveMQ实现
客户注册功能,发短信功能分离 通过ActiveMQ 配置链接工厂, 配置session缓存工厂(引入链接工厂) 2.配置模板对象JmsTemplate 引入缓存工厂 指定消息模式(队列,发布和订 ...
- 项目一:在线下单(补充) activeMQ使用(重点) 重构客户注册功能,发短信功能分离
1 课程计划 1.在线下单(补充) 2.activeMQ使用(重点) n 简介和安装 n activeMQ入门案例 n spring整合activeMQ应用 3.重构客户注册功能,发短信功能分离 n ...
- 基于EasyDarwin EasyPusher实现Android手机直播推送功能
EasyPusher直播推送在之前就已经稳定支持了Windows.Linux.ARM上的RTSP直播推送功能,配合EasyDarwin开源流媒体服务器,延时基本在1s以内,这个技术方案经过一年多时间, ...
随机推荐
- Django加入JS,CSS,图片等外部文件的方法
Django加入JS,CSS.图片等外部文件的方法 By 白熊花田(http://blog.csdn.net/whiterbear) 转载需注明出处,谢谢. 在使用Django搭建站点时,往往须要使用 ...
- IPv6地址表示方法详解
IPv6是互联网协议的第六版:最初它在IETF的 IPng选取过程中胜出时称为互联网新一代网际协议(IPng),IPv6是被正式广泛使用的第二版互联网协议. 现有标准IPv4只支持大概40亿(4×10 ...
- android.graphics.Paint方法setXfermode (Xfermode x...
mPaint = new Paint(); mPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.SCREEN)); 常见的Xfermod ...
- CSS min-height不能解决垂直外边距合并问题
垂直外边距合并有一种情况是嵌套元素的垂直外边距合并,当父级元素没有设定外边距时,在顶部或者底部边缘的子元素的垂直外边距就会和父级的合并,导致父级也有了“隐形”的垂直外边距. 当父级元素的min-hei ...
- sql 高性能存储过程分页
USE [Lyjjr] GO /****** Object: StoredProcedure [dbo].[P_ViewPage] Script Date: 05/29/2015 17:18:56 * ...
- RocketMQ(八):消息发送
匠心零度 转载请注明原创出处,谢谢! RocketMQ网络部署图 NameServer:在系统中是做命名服务,更新和发现 broker服务. Broker-Master:broker 消息主机服务器. ...
- JQuery的index()函数
1.index(),这里的索引从0开始计数. jQueryObject.index( [ object ] ):1.1 如果没有指定参数object,则返回当前元素在其所有同辈元素中的索引位置.1.2 ...
- MES制造系统
给大家介绍一个非常不错的MES专业产品网站,http://www.OrBitMES.com 上面分为初.中.高各级的难得的MES产品学习资料下载, http://www.orbitmes.com/Pr ...
- 为什么出现ORM
ORM(Object Relational Mapping)对象关系映射,是一种为了解决面向对象与关系数据库存在的互不匹配的现象的技术 . 为什么出现ORM? 面向对象的特征:我们通常使用的开发语言J ...
- 四种布局JS
现代 Web 开发在将体验和功能做到极致的同时,对于美观的追求也越来越高.在推荐完图形库之后,再来推荐一些精品的独立 UI 组件.这些组件可组合在一起,形成美观而交互强大的 Web UI . 给 We ...