Android应用开发-小巫CSDN博客客户端之显示博文详细内容

上篇博文给大家介绍的是如何嵌入有米广告并且获取收益,本篇博客打算讲讲关于如何在一个ListView里显示博文的详细信息,这个可能是童鞋们比较困惑的,因为一篇博客可能有标题、摘要、图片、代码等等元素组成,我们要怎么在一个界面中显示这些内容并且按照自己的指定的方式显示呢,别急,下面会告诉大家。
   重新整理一下一篇博文可能有以下元素:
  • 标题
  • 摘要
  • 文本内容
  • 图片
  • 粗标题
  • 代码块
在UI篇小巫已经介绍了,博文详细内容的主要控件就是一个ListView,每个元素就是ListView中的一项item,每一项都有自己的布局用于显示特定的元素。效果图如下:
关于UI就不说了,主要来看一下内容适配器:
/BlogClient/src/com/xiaowu/blogclient/adapter/BlogDetailAdapter.java
package com.xiaowu.blogclient.adapter;import java.util.ArrayList;
import java.util.List;import android.content.Context;
import android.graphics.Bitmap;
import android.text.Html;
import android.text.SpannableStringBuilder;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.webkit.WebView;
import android.widget.BaseAdapter;
import android.widget.ImageView;
import android.widget.TextView;import com.nostra13.universalimageloader.core.DisplayImageOptions;
import com.nostra13.universalimageloader.core.ImageLoader;
import com.nostra13.universalimageloader.core.ImageLoaderConfiguration;
import com.nostra13.universalimageloader.core.assist.ImageScaleType;
import com.nostra13.universalimageloader.core.display.FadeInBitmapDisplayer;
import com.xiaowu.blogclient.R;
import com.xiaowu.blogclient.model.Blog;
import com.xiaowu.blogclient.util.Constants;
import com.xiaowu.blogclient.util.FileUtil;
import com.xiaowu.blogclient.util.MyTagHandler;/*** 博客内容适配器* * @author wwj_748* @date 2014/8/10*/
public class BlogDetailAdapter extends BaseAdapter {private ViewHolder holder;private LayoutInflater layoutInflater;private Context context;private List<Blog> list;private SpannableStringBuilder htmlSpannable;private ImageLoader imageLoader = ImageLoader.getInstance();private DisplayImageOptions options;public BlogDetailAdapter(Context context) {super();this.context = context;layoutInflater = LayoutInflater.from(context);list = new ArrayList<Blog>();// 初始化imageLoaderimageLoader.init(ImageLoaderConfiguration.createDefault(context));// imageloader配置options = new DisplayImageOptions.Builder().showStubImage(R.drawable.images).showImageForEmptyUri(R.drawable.images).showImageOnFail(R.drawable.images).cacheInMemory().cacheOnDisc().imageScaleType(ImageScaleType.EXACTLY).bitmapConfig(Bitmap.Config.RGB_565).displayer(new FadeInBitmapDisplayer(300)).build();}public void setList(List<Blog> list) {this.list = list;}public void addList(List<Blog> list) {this.list.addAll(list);}public void clearList() {this.list.clear();}public List<Blog> getList() {return list;}public void removeItem(int position) {if (list.size() > 0) {list.remove(position);}}@Overridepublic int getCount() {return list.size();}@Overridepublic Object getItem(int position) {return list.get(position);}@Overridepublic long getItemId(int position) {return position;}@Overridepublic View getView(int position, View convertView, ViewGroup parent) {Blog item = list.get(position);if (null == convertView) {holder = new ViewHolder();switch (item.getState()) {case Constants.DEF_BLOG_ITEM_TYPE.TITLE:// 显示标题convertView = layoutInflater.inflate(R.layout.article_detail_title_item, null);holder.content = (TextView) convertView.findViewById(R.id.text);break;case Constants.DEF_BLOG_ITEM_TYPE.SUMMARY: // 摘要convertView = layoutInflater.inflate(R.layout.article_detail_summary_item, null);holder.content = (TextView) convertView.findViewById(R.id.text);break;case Constants.DEF_BLOG_ITEM_TYPE.CONTENT: // 内容convertView = layoutInflater.inflate(R.layout.article_detail_item, null);holder.content = (TextView) convertView.findViewById(R.id.text);break;case Constants.DEF_BLOG_ITEM_TYPE.IMG: // 图片convertView = layoutInflater.inflate(R.layout.article_detail_img_item, null);holder.image = (ImageView) convertView.findViewById(R.id.imageView);break;case Constants.DEF_BLOG_ITEM_TYPE.BOLD_TITLE: // 加粗标题convertView = layoutInflater.inflate(R.layout.article_detail_bold_title_item, null);holder.content = (TextView) convertView.findViewById(R.id.text);break;case Constants.DEF_BLOG_ITEM_TYPE.CODE: // 代码convertView = layoutInflater.inflate(R.layout.article_detail_code_item, null);holder.code = (WebView) convertView.findViewById(R.id.code_view);// holder.code.getSettings().setUseWideViewPort(true);// holder.code.getSettings().setJavaScriptEnabled(true);// holder.code.getSettings().setSupportZoom(true);// holder.code.getSettings().setBuiltInZoomControls(false);// holder.code.getSettings().setLoadWithOverviewMode(true);break;}convertView.setTag(holder);} else {holder = (ViewHolder) convertView.getTag();}// System.out.println(item.getContent());if (null != item) {switch (item.getState()) {case Constants.DEF_BLOG_ITEM_TYPE.IMG: // 图片,异步加载imageLoader.displayImage(item.getContent(), holder.image,options);break;case Constants.DEF_BLOG_ITEM_TYPE.CODE: // 代码,格式显示// 读取代码文件和模板文件String code = item.getContent();// String code = FileUtil.getFileContent(context,// "AboutActivity.java");String template = FileUtil.getFileContent(context, "code.html");// 生成结果String html = template.replace("{{code}}", code);holder.code.getSettings().setDefaultTextEncodingName("utf-8");holder.code.getSettings().setSupportZoom(true);holder.code.getSettings().setBuiltInZoomControls(true);// holder.code.loadUrl("file:///android_asset/code.html");holder.code.loadDataWithBaseURL("file:///android_asset/", html,"text/html", "utf-8", null);break;default:holder.content.setText(Html.fromHtml(item.getContent(), null,new MyTagHandler()));break;}}return convertView;}@Overridepublic int getViewTypeCount() {return 6;}@Overridepublic int getItemViewType(int position) {switch (list.get(position).getState()) {case Constants.DEF_BLOG_ITEM_TYPE.TITLE:return 0;case Constants.DEF_BLOG_ITEM_TYPE.SUMMARY:return 1;case Constants.DEF_BLOG_ITEM_TYPE.CONTENT:return 2;case Constants.DEF_BLOG_ITEM_TYPE.IMG:return 3;case Constants.DEF_BLOG_ITEM_TYPE.BOLD_TITLE:return 4;case Constants.DEF_BLOG_ITEM_TYPE.CODE:return 5;}return 1;}@Overridepublic boolean isEnabled(int position) {switch (list.get(position).getState()) {case Constants.DEF_BLOG_ITEM_TYPE.IMG:return true;default:return false;}}private class ViewHolder {TextView id;TextView date;TextView title;TextView content;ImageView image;WebView code;}
}
这里有一个ListView的优化策略,就是图片进行异步加载,小巫这里用到了优秀的开源项目universalimageloader,我们只需要关联依赖项目,就可以在项目中使用它对网络图片进行异步加载,具体使用自己查看上面的代码实现。
/BlogClient/src/com/xiaowu/blogclient/BlogDetailActivity.java
package com.xiaowu.blogclient;import me.maxwin.view.IXListViewLoadMore;
import me.maxwin.view.XListView;
import android.app.Activity;
import android.content.Intent;
import android.os.AsyncTask;
import android.os.Bundle;
import android.view.View;
import android.view.View.OnClickListener;
import android.view.Window;
import android.widget.AdapterView.OnItemClickListener;
import android.widget.AdapterView;
import android.widget.ImageView;
import android.widget.ProgressBar;
import android.widget.Toast;import com.xiaowu.blogclient.adapter.BlogDetailAdapter;
import com.xiaowu.blogclient.util.Constants;
import com.xiaowu.blogclient.util.JsoupUtil;
import com.xiaowu.blogclient.util.HttpUtil;/*** 博客详细内容界面* * @author wwj_748* @date 2014/8/10*/
public class BlogDetailActivity extends Activity implements IXListViewLoadMore {private XListView listView; // 列表控件private BlogDetailAdapter blogDetailAdapter; // 内容适配器private ProgressBar progressBar; // 进度条private ImageView reLoadImageView; // 重新加载的图片private ImageView backBtn; // 回退按钮private ImageView commentBtn; // 评论按钮public static String url; // 博客地址private String filename; // 文件名字@Overrideprotected void onCreate(Bundle savedInstanceState) {requestWindowFeature(Window.FEATURE_NO_TITLE);// 无标题super.onCreate(savedInstanceState);setContentView(R.layout.article_detail);init();initComponent();// 执行异步加载new MainTask().execute(url, Constants.DEF_TASK_TYPE.FIRST);}// 初始化private void init() {blogDetailAdapter = new BlogDetailAdapter(this);url = getIntent().getExtras().getString("blogLink");filename = url.substring(url.lastIndexOf("/") + 1);System.out.println("filename--->" + filename);}// 初始化组件private void initComponent() {progressBar = (ProgressBar) findViewById(R.id.blogContentPro);reLoadImageView = (ImageView) findViewById(R.id.reLoadImage);reLoadImageView.setOnClickListener(new OnClickListener() {@Overridepublic void onClick(View view) {reLoadImageView.setVisibility(View.INVISIBLE);progressBar.setVisibility(View.VISIBLE);}});backBtn = (ImageView) findViewById(R.id.backBtn);backBtn.setOnClickListener(new OnClickListener() {@Overridepublic void onClick(View arg0) {finish();}});commentBtn = (ImageView) findViewById(R.id.comment);commentBtn.setOnClickListener(new OnClickListener() {@Overridepublic void onClick(View view) {Intent i = new Intent();i.setClass(BlogDetailActivity.this, BlogCommentActivity.class);i.putExtra("filename", filename);startActivity(i);overridePendingTransition(R.anim.push_left_in, R.anim.push_no);}});listView = (XListView) findViewById(R.id.listview);listView.setAdapter(blogDetailAdapter);listView.setPullLoadEnable(this);listView.setOnItemClickListener(new OnItemClickListener() {@Overridepublic void onItemClick(AdapterView<?> parent, View view,int position, long id) {// 获取点击列表项的状态int state = blogDetailAdapter.getList().get(position - 1).getState();switch (state) {case Constants.DEF_BLOG_ITEM_TYPE.IMG: // 点击的是图片String url = blogDetailAdapter.getList().get(position - 1).getImgLink();Intent i = new Intent();i.setClass(BlogDetailActivity.this, ImageActivity.class);i.putExtra("url", url);startActivity(i);break;default:break;}}});}@Overridepublic void finish() {super.finish();}private class MainTask extends AsyncTask<String, Void, Integer> {@Overrideprotected Integer doInBackground(String... params) {// 通过http请求url地址,获取html文档String temp = HttpUtil.httpGet(params[0]);if (temp == null) {if (params[1].equals(Constants.DEF_TASK_TYPE.FIRST)) {return Constants.DEF_RESULT_CODE.FIRST;} else {return Constants.DEF_RESULT_CODE.ERROR;}}// 添加解析出来的数据blogDetailAdapter.addList(JsoupUtil.getContent(url, temp));if (params[1].equals(Constants.DEF_TASK_TYPE.FIRST)) {return Constants.DEF_RESULT_CODE.REFRESH;}return Constants.DEF_RESULT_CODE.LOAD;}@Overrideprotected void onPostExecute(Integer result) {if (result == Constants.DEF_RESULT_CODE.FIRST) {Toast.makeText(getApplicationContext(), "网络信号不佳",Toast.LENGTH_LONG).show();reLoadImageView.setVisibility(View.VISIBLE);} else if (result == Constants.DEF_RESULT_CODE.ERROR) {listView.stopLoadMore();} else if (result == Constants.DEF_RESULT_CODE.REFRESH) {blogDetailAdapter.notifyDataSetChanged();} else {blogDetailAdapter.notifyDataSetChanged();listView.stopLoadMore();}progressBar.setVisibility(View.INVISIBLE);super.onPostExecute(result);}}// 加载更多@Overridepublic void onLoadMore() {if (!JsoupUtil.contentLastPage) {new MainTask().execute(url, Constants.DEF_TASK_TYPE.NOR_FIRST);} else {// 在底部显示“--THE END0--"文本listView.stopLoadMore(" -- THE END --");}}}
如果研读了我提供源码的童鞋都会发现,我这里使用AsyncTask来进行网络请求操作,童鞋们也可以使用Thread+Handler的方式来实现异步请求,需要注意的是,耗时操作和网络操作都不能放在主线程,这是Android开发的规范。
这里还提一个技巧,我们更新ListView的数据的时候,并不需要重新new一个adapter,可以像我一样,在适配器类中提供addList的方法,添加数据到adapter中,然后在适当的位置调用notifyDataSetChanged()方法就可以更新数据,不会出现数据重复和效率低下的情况。

Android应用开发-小巫CSDN博客客户端之显示博文详细内容相关推荐

  1. Android应用开发-小巫CSDN博客客户端UI篇

    Android应用开发-小巫CSDN博客客户端UI篇 上一篇是给童鞋们介绍整个项目的概况,从这篇博文开始,后续也会详细介绍整个客户端的开发,但不会贴很多代码,我会贴核心代码然后提供实现思路,想看里面更 ...

  2. Android应用开发-小巫CSDN博客客户端之获取评论列表

    Android应用开发-小巫CSDN博客客户端之获取评论列表 上一篇博客介绍了博文详细内容的业务逻辑实现,本篇博客介绍小巫CSDN博客客户端的最后一项功能,获取评论列表,这个功能的实现跟前面获取文章列 ...

  3. Android应用开发-小巫CSDN博客客户端之嵌入有米广告

    Android应用开发-小巫CSDN博客客户端之嵌入有米广告 上一篇博客给大家介绍如何集成友盟社会化组件,本篇继续带来干货,教大家如何嵌入广告到应用中去.小巫自称专业对接30年,熟悉各大渠道SDK的接 ...

  4. Android应用开发-小巫CSDN博客客户端Jsoup篇

    Android应用开发-小巫CSDN博客客户端Jsoup篇 距上一篇博客已经过去了两个星期,小巫也觉得非常抱歉,因为在忙着做另外一个项目,几乎抽不出空来,这不小巫会把剩下的博文全部在国庆补上.本篇博客 ...

  5. Android应用开发-小巫CSDN博客客户端之集成友盟社会化分享组件

    Android应用开发-小巫CSDN博客客户端之集成友盟社会化分享组件 上一篇博客给大家介绍了如何分析网页并且使用jsoup这个库对html代码进行解析,本篇博客继续给大家介绍如何集成友盟社会化组件, ...

  6. Android应用开发-小巫CSDN博客客户端UI篇,kotlin安卓开发教程视频

    ** ** (图5-博文评论列表) 以上给大家展示的是小巫CSDN博客客户端的主要界面效果,下面来讲解如何布局这样的界面: 启动界面布局 /BlogClient/res/layout/splash.x ...

  7. Android应用开发-小巫CSDN博客客户端之集成友盟社会化分享组件,2021最新Android框架体系架构面试题

    下面是详细的集成过程: 首先来到官网:http://www.umeng.com/ 选择社会化分享,进入查看这个产品的介绍,下载对应平台的SDK,笔者这里自然是选择Android平台: 笔者集成友盟社会 ...

  8. Android应用开发-小巫CSDN博客客户端开发开篇,玩转MySQL

    本篇博客是关于这款应用的开发的起始篇,主要简单介绍一下整个项目的概况,整体大纲如下: 1. 项目起因 2. 项目效果展示 3. 项目文档结构和依赖库说明 4. 项目功能简介 5. 系列博客分享后期计划 ...

  9. Android应用开发-小巫CSDN博客客户端之嵌入有米广告,android音视频开发面试

    然而横幅广告有两种形式,一种是以XML形式布局,另一种是以代码形式布局. 配置有米权限: 有米必要的配置: <activity android:name="net.youmi.andr ...

最新文章

  1. java for循环返回值_Java中的for循环——通过示例学习Java编程(9)
  2. mybatis3进行模糊查询的总结
  3. LeetCode Wiggle Sort II(快排)
  4. ajax分页实现(php)
  5. spark on yarn相关脚本整理20210524
  6. VTK:旋转球体用法实战
  7. 都说雪花ID不能做MySQL的主键,你确定?
  8. Visual Studio编译选项简介
  9. Ubuntu kylin 14.04 LTS上安装opencv
  10. 反馈页面设计_获得设计反馈的艺术
  11. ASP.NET WebApi 基于JWT实现Token签名认证
  12. [Hei-Ocelot-Gateway ].Net Core Api网关Ocelot的开箱即用版本
  13. C#LeetCode刷题之#680-验证回文字符串 Ⅱ​​​​​​​(Valid Palindrome II)
  14. Maven查看插件信息
  15. 接口测试要测试什么?
  16. mvdr波束形成原理_有了波束赋形这个5G黑科技,让你畅享飞一样的网速
  17. 单元测试的思考与实践
  18. python百度文库下载器_真正百度文库免费下载,比冰点文库下载器还更牛x!
  19. cmake practice学习记录
  20. 互联网大数据面试题集锦

热门文章

  1. (七)Linux内存管理 - zoned page frame allocator - 2
  2. Alibaba Java Coding Guidelines安装使用教程
  3. 中国第三方支付产业投资项目及十四五机会战略分析报告2022~2028年
  4. java反射各种属性获取
  5. tensorflow63 《深度学习原理与TensorFlow实战》03 Hello TensorFlow
  6. S5PV210开发系列五_sd卡驱动实现
  7. 100 个知名网站源码
  8. 学生为什么要学python_碎碎念|为什么要学Python
  9. android前端开发工具,分享七个非常有用的Android开发工具和工具包
  10. Infortrend数据库存储解决方案