简单方法实现Android阅读器分页
2016-08-23 20:18
483 查看
写了一个非常小的阅读器。在实现分页功能时,一直没有思路。后来想了一个非常特别的方法。经过测试可以完美的实现分页功能。
主要思路:
1、将文本内容填充到TextView中,调用setText一句搞定。
2、计算TextView的高度范围内可显示的行数。如果TextView占据整个屏幕则计算屏幕范围可显示的的函数。
利用TextView 的getLineBounds 函数可以计算每行占据的高度h。利用 h 和TextView的高度 H 就可以很方便计算可显示的行数。
3、最关键的一步。计算TextView n 行显示的字体个数。
这是最关键的一个API,能够实现这个功能主要靠它。而且TextView本身也是借助这个API实现自动换行的。
这就是StatiLayout。StatiLayout有一个函数getLineEnd(n)可以计算从0到n行字体的个数。TextView 一页显示的行数是固定的,
分页的难点就是每行的字体个数不固定。通过getLineEnd 就可以非常简单的计算每页的字体个数。
4、通过每页的字体个数从文本内容中截取每页的内容。
使用了一个PagerAdapter 将文本内容创建为一个TextView,这样就可以滑动分页了.
关键代码:
说明:代码主要是说明分页思路,其中有不少bug。
下面是 分页用的PagerAdapter,为了节省资源,对TextView进行了复用。
最后:
如果文本的内容比较大,可以采用分段载入的方法,这样可以加快打开速度。 即先加载一部分文本用来显示,然后在后台线程加载剩余的文本。
主要思路:
1、将文本内容填充到TextView中,调用setText一句搞定。
2、计算TextView的高度范围内可显示的行数。如果TextView占据整个屏幕则计算屏幕范围可显示的的函数。
利用TextView 的getLineBounds 函数可以计算每行占据的高度h。利用 h 和TextView的高度 H 就可以很方便计算可显示的行数。
3、最关键的一步。计算TextView n 行显示的字体个数。
这是最关键的一个API,能够实现这个功能主要靠它。而且TextView本身也是借助这个API实现自动换行的。
这就是StatiLayout。StatiLayout有一个函数getLineEnd(n)可以计算从0到n行字体的个数。TextView 一页显示的行数是固定的,
分页的难点就是每行的字体个数不固定。通过getLineEnd 就可以非常简单的计算每页的字体个数。
4、通过每页的字体个数从文本内容中截取每页的内容。
使用了一个PagerAdapter 将文本内容创建为一个TextView,这样就可以滑动分页了.
关键代码:
说明:代码主要是说明分页思路,其中有不少bug。
public int[] getPage( TextView textView){ int count=textView.getLineCount(); textView.setText(mContent); int pCount=getPageLineCount(textView); int pageNum=count/pCount; int page[]=new int[pageNum]; for(int i=0;i<pageNum;i++){ page[i]=textView.getLayout().getLineEnd((i+1)*pCount-1); } return page; } private int getPageLineCount(TextView view){ /* * The first row's height is different from other row. */ int h=view.getBottom()-view.getTop()-view.getPaddingTop(); int firstH=getLineHeight(0,view); int otherH=getLineHeight(1,view); return (h-firstH)/otherH + 1 ; } private int getLineHeight(int line,TextView view){ Rect rect=new Rect(); view.getLineBounds(line,rect); return rect.bottom-rect.top; }
下面是 分页用的PagerAdapter,为了节省资源,对TextView进行了复用。
public class ContentAdapter extends PagerAdapter { List mCache; private int[] mPage; private String mContent; public ContentAdapter(int[] page, String content){ mPage=page; mContent=content; } @Override public int getCount() { return mPage.length; } @Override public boolean isViewFromObject(View view, Object object) { return view==object; } private String getText(int page){ if(page==0){ return mContent.substring(0,mPage[0]); } return mContent.substring(mPage[page-1],mPage[page]); } @Override public Object instantiateItem(ViewGroup container, int position) { TextView textView=null; if(mCache==null){ mCache=new LinkedList(); } if(mCache.size()>0){ textView=(TextView) mCache.remove(0); }else { textView=new TextView(container.getContext()); } textView.setText(getText(position)); container.addView(textView); return textView; } @Override public void destroyItem(ViewGroup container, int position, Object object) { container.removeView((View)object); mCache.add(object); } }
最后:
如果文本的内容比较大,可以采用分段载入的方法,这样可以加快打开速度。 即先加载一部分文本用来显示,然后在后台线程加载剩余的文本。
相关文章推荐
- 使用C++实现JNI接口需要注意的事项
- Android IPC进程间通讯机制
- Android Manifest 用法
- [转载]Activity中ConfigChanges属性的用法
- Android之获取手机上的图片和视频缩略图thumbnails
- Android之使用Http协议实现文件上传功能
- Android学习笔记(二九):嵌入浏览器
- android string.xml文件中的整型和string型代替
- i-jetty环境搭配与编译
- android之定时器AlarmManager
- android wifi 无线调试
- Android Native 绘图方法
- Android java 与 javascript互访(相互调用)的方法例子
- android 代码实现控件之间的间距
- android FragmentPagerAdapter的“标准”配置
- Android"解决"onTouch和onClick的冲突问题
- android:installLocation简析
- android searchView的关闭事件
- SourceProvider.getJniDirectories