写了一个非常小的阅读器。在实现分页功能时,一直没有思路。后来想了一个非常特别的方法。经过测试可以完美的实现分页功能。
主要思路:
- 将文本内容填充到TextView中,调用setText一句搞定。
- 计算TextView的高度范围内可显示的行数。如果TextView占据整个屏幕则计算屏幕范围可显示的的函数。利用TextView 的getLineBounds 函数可以计算每行占据的高度h。利用 h 和TextView的高度 H 就可以很方便计算可显示的行数。
- 最关键的一步。计算TextView n 行显示的字体个数。这是最关键的一个API,能够实现这个功能主要靠它。而且TextView本身也是借助这个API实现自动换行的。
这就是StatiLayout。StatiLayout有一个函数getLineEnd(n)可以计算从0到n行字体的个数。TextView 一页显示的行数是固定的,
分页的难点就是每行的字体个数不固定。通过getLineEnd 就可以非常简单的计算每页的字体个数。 - 通过每页的字体个数从文本内容中截取每页的内容。
使用了一个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); }}
最后:
如果文本的内容比较大,可以采用分段载入的方法,这样可以加快打开速度。 即先加载一部分文本用来显示,然后在后台线程加载剩余的文本。