Android ViewPagerアダプティブ高さ

4869 ワード

多くのAndroid開発者が直面する問題は、ViewPagerがコンテンツの高さに適応できないことです.次に、コンテンツに基づいてViewPagerの高さを動的に設定し、2つの方法をまとめます.
1つ目は、すべてのviewの最大高さをViewPagerの高さとして取ります.この方法では、すべてのpageページの高さの差が少ない場合を試してみます.そうしないと、次のような空白の部分があります.コードは次のとおりです.
public class AutoHeightViewPager extends ViewPager {  
   
    public AutoHeightViewPager (Context context) {  
        super(context);  
    }  
   
    public AutoHeightViewPager (Context context, AttributeSet attrs) {  
        super(context, attrs);  
    }  
   
    @Override  
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {  
   
        int height = 0;  
        for (int i = 0; i < getChildCount(); i++) {  
            View child = getChildAt(i);  
            child.measure(widthMeasureSpec, MeasureSpec.makeMeasureSpec(0, MeasureSpec.UNSPECIFIED));  
            int h = child.getMeasuredHeight();  
            if (h > height)  
                height = h;  
        }  
   
        heightMeasureSpec = MeasureSpec.makeMeasureSpec(height, MeasureSpec.EXACTLY);  
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);  
    }  
}

使い方も簡単で、xmlレイアウトに元のViewPagerをそのまま置き換えればいいです.
  xxx.xxx.xxx    
  
 

2つ目は、すべてのインタフェースの高さが同じではない場合、高さの低いページに空白が表示される場合です.次に、2つ目の方法で高さを動的に設定する方法について説明します.
public class AutoHeightViewPager extends ViewPager {
    
    public AutoHeightViewPager(Context context, AttributeSet attrs) {
        super(context, attrs);
    }
    
    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
        // find the current child view
        // and you must cache all the child view
        // use setOffscreenPageLimit(adapter.getCount())
        View view = getChildAt(getCurrentItem());
        if (view != null) {
            // measure the current child view with the specified measure spec
            view.measure(widthMeasureSpec, heightMeasureSpec);
        }
        
        setMeasuredDimension(getMeasuredWidth(), measureHeight(heightMeasureSpec, view));
    }
    
    /**
     * Determines the height of this view
     *
     * @param measureSpec A measureSpec packed into an int
     * @param view the base view with already measured height
     *
     * @return The height of the view, honoring constraints from measureSpec
     */
    private int measureHeight(int measureSpec, View view) {
        int result = 0;
        int specMode = MeasureSpec.getMode(measureSpec);
        int specSize = MeasureSpec.getSize(measureSpec);
        
        if (specMode == MeasureSpec.EXACTLY) {
            result = specSize;
        } else {
            // set the height from the base view if available
            if (view != null) {
                result = view.getMeasuredHeight();
            }
            if (specMode == MeasureSpec.AT_MOST) {
                result = Math.min(result, specSize);
            }
        }
        return result;
    }
    
    /**
     *     view    
     *
     * @param view
     */
    public void measeureView(View view) {
        
        int intw = View.MeasureSpec.makeMeasureSpec(0, View.MeasureSpec.UNSPECIFIED);
        int inth = View.MeasureSpec.makeMeasureSpec(0, View.MeasureSpec.UNSPECIFIED);
        //     view
        view.measure(intw, inth);
        
        //   3         
        //view.measure(0,0);
        
        //       view  
        int intwidth = view.getMeasuredWidth();
        int intheight = view.getMeasuredHeight();
    }
}

注釈には詳細な説明があります.使用方法は次のとおりです.
  • xmlレイアウトにおける
  • の構成
    xxx.xxx.xxx代表パス
     
    
  • コードで呼び出す
  • mViewPager.addOnPageChangeListener(new ViewPager.OnPageChangeListener() {
                @Override
                public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
    
                }
    
                @Override
                public void onPageSelected(final int position) {
    	            //        ,    
                   mViewPager.requestLayout();
                }
    
                @Override
                public void onPageScrollStateChanged(int state) {
    
                }
            });
     
    
  • offsetプロパティを設定し、すべてのviewを事前にロードし、ロードされていないviewが高さ0//View=getChildAt(getCurrentItem()であることを防止する.
  • //      ,                ,  offset     1 
    mViewPager.setOffscreenPageLimit(adapter.getCount());
    

    以上、まとめた2つのViewPagerの適応高さの方法ですが、あなたに役に立つことを望んでいます.足りないところは大人に教えてもらいたいです.