寫了一個(gè)非常小的閱讀器续誉。在實(shí)現(xiàn)分頁功能時(shí)哲银,一直沒有思路扛吞。后來想了一個(gè)非常特別的方法。經(jīng)過測試可以完美的實(shí)現(xiàn)分頁功能荆责。
主要思路:
- 將文本內(nèi)容填充到TextView中滥比,調(diào)用setText一句搞定。
- 計(jì)算TextView的高度范圍內(nèi)可顯示的行數(shù)做院。如果TextView占據(jù)整個(gè)屏幕則計(jì)算屏幕范圍可顯示的的函數(shù)盲泛。利用TextView 的getLineBounds 函數(shù)可以計(jì)算每行占據(jù)的高度h。利用 h 和TextView的高度 H 就可以很方便計(jì)算可顯示的行數(shù)键耕。
- 最關(guān)鍵的一步寺滚。計(jì)算TextView n 行顯示的字體個(gè)數(shù)。這是最關(guān)鍵的一個(gè)API屈雄,能夠?qū)崿F(xiàn)這個(gè)功能主要靠它村视。而且TextView本身也是借助這個(gè)API實(shí)現(xiàn)自動(dòng)換行的。
這就是StatiLayout酒奶。StatiLayout有一個(gè)函數(shù)getLineEnd(n)可以計(jì)算從0到n行字體的個(gè)數(shù)蚁孔。TextView 一頁顯示的行數(shù)是固定的,
分頁的難點(diǎn)就是每行的字體個(gè)數(shù)不固定惋嚎。通過getLineEnd 就可以非常簡單的計(jì)算每頁的字體個(gè)數(shù)杠氢。 - 通過每頁的字體個(gè)數(shù)從文本內(nèi)容中截取每頁的內(nèi)容。
使用了一個(gè)PagerAdapter 將文本內(nèi)容創(chuàng)建為一個(gè)TextView另伍,這樣就可以滑動(dòng)分頁了.
關(guān)鍵代碼:
說明:代碼主要是說明分頁思路鼻百,其中有不少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摆尝,為了節(jié)省資源温艇,對TextView進(jìn)行了復(fù)用。
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); }}
最后:
如果文本的內(nèi)容比較大结榄,可以采用分段載入的方法中贝,這樣可以加快打開速度。 即先加載一部分文本用來顯示臼朗,然后在后臺線程加載剩余的文本邻寿。