Android模仿淘寶 商品詳情拖動(dòng)查看圖文詳情效果

tx.gif

最近要在公司的項(xiàng)目中實(shí)現(xiàn)這種效果英妓,就自己動(dòng)手實(shí)現(xiàn)了一下挽放。

使用

GraphicDetailsLayout gdLayout = (GraphicDetailsLayout) findViewById(R.id.gdlayout);
gdLayout.addFragment(new Fragment[] {new SpFragment(), new DeFragment()}, getSupportFragmentManager());

還是很簡(jiǎn)單的,把上下兩個(gè)fragment添加到GraphicDetailsLayout 中就可以了

思路

從效果中可以看到上下兩個(gè)控件都是可以滾動(dòng)的蔓纠,初始化狀態(tài)下辑畦,下面的控件是隱藏在屏幕下面的;那我們?cè)O(shè)計(jì)最外面的布局是LinearLayout贺纲,然后LinearLayout里面放兩個(gè)ScrollView航闺,ScrollView滾動(dòng)到頂部或頂部的時(shí)候,告訴LinearLayout攔截事件猴誊,來實(shí)現(xiàn)兩個(gè)ScrollView的上下拖動(dòng)效果。

實(shí)現(xiàn)

首先自定義一個(gè)ScrollView

public class GDScrollView extends ScrollView {    

   private LinearLayout mLl;    
   private int mLlHeight;    
   public static final String TAG_ONE = "up";    
   public static final String TAG_TWO = "down";    
   public static final int ID_ONE = 11111;    
   public static final int ID_TWO = 22222;    
   private GraphicDetailsLayout.ScrollListener mScrollListener;   

   public GDScrollView(Context context) {        
       super(context);    }    

   public GDScrollView(Context context, AttributeSet attrs) {   
       super(context, attrs);    
   }    

   public GDScrollView(Context context, AttributeSet attrs, int defStyleAttr) {        
       super(context, attrs, defStyleAttr);    
   }    

   @TargetApi(Build.VERSION_CODES.LOLLIPOP)    
   public GDScrollView(Context context, AttributeSet attrs, int defStyleAttr, 
    int defStyleRes) {        
       super(context, attrs, defStyleAttr, defStyleRes);    
   }    

   public void setScrollListener(GraphicDetailsLayout.ScrollListener scrollListener) {        
        mScrollListener = scrollListener;    
   }    

   public void addFragment(Fragment fragment, FragmentManager fragmentManager) {        
        FragmentTransaction fragmentTransaction = fragmentManager.beginTransaction();        
        if(mLl == null) mLl = (LinearLayout) getChildAt(0);        
        fragmentTransaction.replace(mLl.getId(), fragment);  
        fragmentTransaction.commit();    }    

   @Override    
   protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {        
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);  
        mLlHeight = mLl.getMeasuredHeight();    
   }    

   @Override    
   protected void onFinishInflate() {        
        super.onFinishInflate();        
        mLl = (LinearLayout) getChildAt(0);    
   }    

   @Override    
   protected void onOverScrolled(int scrollX, int scrollY, boolean clampedX, boolean clampedY) {        
        super.onOverScrolled(scrollX, scrollY, clampedX, clampedY);   
        if(getTag().equals(TAG_ONE)) {            
          //上面的界面滾動(dòng)到底部的時(shí)候            
           if(isScrollBottom()) {                
               criticalPointOperation(false, true, TAG_ONE);           
           }        
         }       

        if(getTag().equals(TAG_TWO)) {    
          //下面的界面滾動(dòng)到頂部的時(shí)候        
          if(getScrollY() <= 0) {                
              criticalPointOperation(false, true, TAG_TWO);           
           }        
         }    
    }    

   private void criticalPointOperation(boolean allow, boolean intercept, String tag) {      
       getParent().requestDisallowInterceptTouchEvent(allow);  
       if(mScrollListener != null) mScrollListener.scrollBottom(intercept, tag);    }    

   public boolean isScrollBottom() {        
       return getScrollY() >= (mLlHeight - getMeasuredHeight());    
   }
}

重寫onOverScrolled方法監(jiān)控滾動(dòng)的狀態(tài)侮措,判斷不同的ScrollView滾動(dòng)到頂部或者頂部觸發(fā)回調(diào)事件懈叹,把觸摸事件交給上層LinarLayout控件,來看看LinearLayout的onTouchEvent方法

@Override
public boolean onTouchEvent(MotionEvent event) {    
   switch (event.getAction()) {        
     case MotionEvent.ACTION_MOVE:            
       if(mInitY == 0) {                
          mInitY = event.getY();            
       } else {                
          int offset = (int) Math.abs(event.getY() - mInitY);             
          if(offset > mTouchSlop) {                    
            int delayOffset = offset * 7 / 10;    
            if(mCurrentTag.equals(GDScrollView.TAG_ONE)) {   
                  mUpSVMarginTop = mInitMarginTop - delayOffset;      
              } else {                        
                  mUpSVMarginTop = - halfHeight + delayOffset;       
             }             
       
          if(mUpSVMarginTop > 0) mUpSVMarginTop = 0;      
            requestLayout();                
         }            
      }            
      break;        
    case MotionEvent.ACTION_UP:            
       mIntercept = false;            
       mInitY = 0;   
       if(mCurrentTag.equals(GDScrollView.TAG_ONE)) {          
           if(Math.abs(mUpSVMarginTop) > halfHeight / 3) {   
              startAnimation(mUpSVMarginTop, halfHeight - Math.abs(mUpSVMarginTop), false);                
           } else {                    
              startAnimation(mUpSVMarginTop, Math.abs(mUpSVMarginTop), true);                
           }            
        } else {                
           if(Math.abs(mUpSVMarginTop) < halfHeight * 2 / 3) {        
                startAnimation(mUpSVMarginTop, Math.abs(mUpSVMarginTop), true);                
            } else {                    
                startAnimation(mUpSVMarginTop, halfHeight - Math.abs(mUpSVMarginTop), false);                
            }            
        }            
         requestLayout();            
         break;    
      }    
      return true;
  }

LinearLayout攔截到事件以后重寫onTouchEvent方法分扎,通過手勢(shì)拖動(dòng)來不斷的計(jì)算ScrooView距離頂部的高度mUpSVMarginTop澄成,調(diào)用requestLayout方法發(fā)起重新布局,重寫onLayout方法

@Override
protected void onLayout(boolean changed, int l, int t, int r, int b) {    
    super.onLayout(changed, l, t, r, b);    
    halfHeight = getMeasuredHeight() / 2;    
    mUpScrollView.layout(0, mUpSVMarginTop, getMeasuredWidth(), mUpSVMarginTop + halfHeight);    
    mBottomScrollView.layout(0, mUpSVMarginTop + halfHeight , getMeasuredWidth(), mUpSVMarginTop + getMeasuredHeight());}

大致的思路和實(shí)現(xiàn)已經(jīng)講解完成了畏吓,想看具體實(shí)現(xiàn)代碼
https://github.com/chenpengfei88/GraphicDetailsLayout
歡迎start墨状,follow。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末菲饼,一起剝皮案震驚了整個(gè)濱河市肾砂,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌宏悦,老刑警劉巖镐确,帶你破解...
    沈念sama閱讀 218,858評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件包吝,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡源葫,警方通過查閱死者的電腦和手機(jī)诗越,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,372評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來息堂,“玉大人嚷狞,你說我怎么就攤上這事∪傺撸” “怎么了感耙?”我有些...
    開封第一講書人閱讀 165,282評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)持隧。 經(jīng)常有香客問我即硼,道長(zhǎng),這世上最難降的妖魔是什么屡拨? 我笑而不...
    開封第一講書人閱讀 58,842評(píng)論 1 295
  • 正文 為了忘掉前任只酥,我火速辦了婚禮,結(jié)果婚禮上呀狼,老公的妹妹穿的比我還像新娘裂允。我一直安慰自己,他們只是感情好哥艇,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,857評(píng)論 6 392
  • 文/花漫 我一把揭開白布绝编。 她就那樣靜靜地躺著,像睡著了一般貌踏。 火紅的嫁衣襯著肌膚如雪十饥。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,679評(píng)論 1 305
  • 那天祖乳,我揣著相機(jī)與錄音逗堵,去河邊找鬼。 笑死眷昆,一個(gè)胖子當(dāng)著我的面吹牛蜒秤,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播亚斋,決...
    沈念sama閱讀 40,406評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼作媚,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了帅刊?” 一聲冷哼從身側(cè)響起纸泡,我...
    開封第一講書人閱讀 39,311評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎厚掷,沒想到半個(gè)月后弟灼,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體级解,經(jīng)...
    沈念sama閱讀 45,767評(píng)論 1 315
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,945評(píng)論 3 336
  • 正文 我和宋清朗相戀三年田绑,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了勤哗。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,090評(píng)論 1 350
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡掩驱,死狀恐怖芒划,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情欧穴,我是刑警寧澤民逼,帶...
    沈念sama閱讀 35,785評(píng)論 5 346
  • 正文 年R本政府宣布,位于F島的核電站涮帘,受9級(jí)特大地震影響拼苍,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜调缨,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,420評(píng)論 3 331
  • 文/蒙蒙 一疮鲫、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧弦叶,春花似錦俊犯、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,988評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至立莉,卻和暖如春绢彤,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背桃序。 一陣腳步聲響...
    開封第一講書人閱讀 33,101評(píng)論 1 271
  • 我被黑心中介騙來泰國(guó)打工杖虾, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人媒熊。 一個(gè)月前我還...
    沈念sama閱讀 48,298評(píng)論 3 372
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像坟比,于是被迫代替她去往敵國(guó)和親芦鳍。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,033評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容