可动态布局的Android抽屉之基础

    以前曾经介绍过 《Android提高第十九篇之"多方向"抽屉》 ,当这个抽屉组件不与周围组件发生压挤的情况下(周围组件布局不变),是比较好使的,但是如果需要对周围组件挤压,则用起来欠缺美观了。

       如下图。在对周围压挤的情况下,抽屉是先把周围的组件一次性压挤,再通过动画效果展开/收缩的,这种做法的好处是快速简单,坏处是如果挤压范围过大,则效果生硬。

 

      本文实现的自定义抽屉组件,主要针对这种压挤效果做出改良,渐进式压挤周围组件,使得过渡效果更加美观。如下图。

 

本文实现的抽屉原理是酱紫:

1.抽屉组件主要在屏幕不可视区域,手柄在屏幕边缘的可视区域。即 抽屉.rightMargin=-XXX + 手柄.width

2.指定一个周围组件为可压挤,即LayoutParams.weight=1;当然用户也可以指定多个View.

3.使用AsyncTask来实现弹出/收缩的动画,弹出:抽屉.rightMargin+=XX,收缩:抽屉.rightMargin-=XX

总结,本文的自定义抽屉虽然对压挤周围组件有过渡效果,但是比较耗资源,读者可以针对不同的情况考虑使用。

本文的源码可以到http://download.csdn.net/detail/hellogv/3615686 下载。

接下来贴出本文全部源代码:

main.xml的源码:

  1. <span style="font-family:Comic Sans MS;font-size:18px;"><?xml version="1.0" encoding="utf-8"?>  
  2. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"  
  3.     android:layout_width="fill_parent" android:layout_height="fill_parent"  
  4.     android:id="@+id/container">  
  5.     <GridView android:id="@+id/gridview" android:layout_width="fill_parent"  
  6.         android:layout_height="fill_parent" android:numColumns="auto_fit"  
  7.         android:verticalSpacing="10dp" android:gravity="center"  
  8.         android:columnWidth="50dip" android:horizontalSpacing="10dip" />  
  9. </LinearLayout></span>  

GridView的Item.xml的源码:

  1. <span style="font-family:Comic Sans MS;font-size:18px;"><?xml version="1.0" encoding="utf-8"?>  
  2. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"  
  3.     android:layout_height="wrap_content" android:paddingBottom="4dip"  
  4.     android:layout_width="fill_parent">  
  5.     <ImageView android:layout_height="wrap_content" android:id="@+id/ItemImage"  
  6.         android:layout_width="wrap_content" android:layout_centerHorizontal="true">  
  7.     </ImageView>  
  8.     <TextView android:layout_width="wrap_content"  
  9.         android:layout_below="@+id/ItemImage" android:layout_height="wrap_content"  
  10.         android:text="TextView01" android:layout_centerHorizontal="true"  
  11.         android:id="@+id/ItemText">  
  12.     </TextView>  
  13. </RelativeLayout>  </span>  

Panel.java是本文核心,抽屉组件的源码,这个抽屉只实现了从右往左的弹出/从左往右的收缩,读者可以根据自己的需要修改源码来改变抽屉动作的方向:

  1. <span style="font-family:Comic Sans MS;font-size:18px;">public class Panel extends LinearLayout{  
  2.       
  3.     public interface PanelClosedEvent {  
  4.         void onPanelClosed(View panel);  
  5.     }  
  6.       
  7.     public interface PanelOpenedEvent {  
  8.         void onPanelOpened(View panel);  
  9.     }  
  10.     /**Handle的宽度,与Panel等高*/  
  11.     private final static int HANDLE_WIDTH=30;  
  12.     /**每次自动展开/收缩的范围*/  
  13.     private final static int MOVE_WIDTH=20;  
  14.     private Button btnHandle;  
  15.     private LinearLayout panelContainer;  
  16.     private int mRightMargin=0;  
  17.     private Context mContext;  
  18.     private PanelClosedEvent panelClosedEvent=null;  
  19.     private PanelOpenedEvent panelOpenedEvent=null;  
  20.   
  21.     /** 
  22.      * otherView自动布局以适应Panel展开/收缩的空间变化 
  23.      * @author GV 
  24.      * 
  25.      */   
  26.     public Panel(Context context,View otherView,int width,int height) {  
  27.         super(context);  
  28.         this.mContext=context;  
  29.       
  30.         //改变Panel附近组件的属性  
  31.         LayoutParams otherLP=(LayoutParams) otherView.getLayoutParams();  
  32.         otherLP.weight=1;//支持压挤  
  33.         otherView.setLayoutParams(otherLP);  
  34.           
  35.         //设置Panel本身的属性  
  36.         LayoutParams lp=new LayoutParams(width, height);  
  37.         lp.rightMargin=-lp.width+HANDLE_WIDTH;//Panel的Container在屏幕不可视区域,Handle在可视区域  
  38.         mRightMargin=Math.abs(lp.rightMargin);  
  39.         this.setLayoutParams(lp);  
  40.         this.setOrientation(LinearLayout.HORIZONTAL);  
  41.           
  42.         //设置Handle的属性  
  43.         btnHandle=new Button(context);  
  44.         btnHandle.setLayoutParams(new LayoutParams(HANDLE_WIDTH,height));  
  45.         btnHandle.setOnClickListener(new OnClickListener(){  
  46.   
  47.             @Override  
  48.             public void onClick(View arg0) {  
  49.                 LayoutParams lp = (LayoutParams) Panel.this.getLayoutParams();  
  50.                 if (lp.rightMargin < 0)// CLOSE的状态  
  51.                     new AsynMove().execute(new Integer[] { MOVE_WIDTH });// 正数展开  
  52.                 else if (lp.rightMargin >= 0)// OPEN的状态  
  53.                     new AsynMove().execute(new Integer[] { -MOVE_WIDTH });// 负数收缩  
  54.             }  
  55.               
  56.         });  
  57.         //btnHandle.setOnTouchListener(HandleTouchEvent);  
  58.         this.addView(btnHandle);  
  59.           
  60.         //设置Container的属性  
  61.         panelContainer=new LinearLayout(context);  
  62.         panelContainer.setLayoutParams(new LayoutParams(LayoutParams.FILL_PARENT,  
  63.                 LayoutParams.FILL_PARENT));  
  64.         this.addView(panelContainer);  
  65.     }  
  66.   
  67.     /** 
  68.      * 定义收缩时的回调函数 
  69.      * @param event 
  70.      */  
  71.     public void setPanelClosedEvent(PanelClosedEvent event)  
  72.     {  
  73.         this.panelClosedEvent=event;  
  74.     }  
  75.       
  76.     /** 
  77.      * 定义展开时的回调函数 
  78.      * @param event 
  79.      */  
  80.     public void setPanelOpenedEvent(PanelOpenedEvent event)  
  81.     {  
  82.         this.panelOpenedEvent=event;  
  83.     }  
  84.       
  85.     /** 
  86.      * 把View放在Panel的Container 
  87.      * @param v 
  88.      */  
  89.     public void fillPanelContainer(View v)  
  90.     {  
  91.         panelContainer.addView(v);  
  92.     }  
  93.       
  94.     /** 
  95.      * 异步移动Panel 
  96.      * @author hellogv  
  97.      */  
  98.     class AsynMove extends AsyncTask<Integer, Integer, Void> {  
  99.   
  100.         @Override  
  101.         protected Void doInBackground(Integer... params) {  
  102.             int times;  
  103.             if (mRightMargin % Math.abs(params[0]) == 0)// 整除  
  104.                 times = mRightMargin / Math.abs(params[0]);  
  105.             else  
  106.                 // 有余数  
  107.                 times = mRightMargin / Math.abs(params[0]) + 1;  
  108.   
  109.             for (int i = 0; i < times; i++) {  
  110.                 publishProgress(params);  
  111.                 try {  
  112.                     Thread.sleep(Math.abs(params[0]));  
  113.                 } catch (InterruptedException e) {  
  114.                     // TODO Auto-generated catch block  
  115.                     e.printStackTrace();  
  116.                 }  
  117.             }  
  118.             return null;  
  119.         }  
  120.   
  121.         @Override  
  122.         protected void onProgressUpdate(Integer... params) {  
  123.             LayoutParams lp = (LayoutParams) Panel.this.getLayoutParams();  
  124.             if (params[0] < 0)  
  125.                 lp.rightMargin = Math.max(lp.rightMargin + params[0],  
  126.                         (-mRightMargin));  
  127.             else  
  128.                 lp.rightMargin = Math.min(lp.rightMargin + params[0], 0);  
  129.   
  130.             if(lp.rightMargin==0 && panelOpenedEvent!=null){//展开之后  
  131.                 panelOpenedEvent.onPanelOpened(Panel.this);//调用OPEN回调函数  
  132.             }  
  133.             else if(lp.rightMargin==-(mRightMargin) && panelClosedEvent!=null){//收缩之后  
  134.                 panelClosedEvent.onPanelClosed(Panel.this);//调用CLOSE回调函数  
  135.             }  
  136.             Panel.this.setLayoutParams(lp);  
  137.         }  
  138.     }  
  139.   
  140. }  
  141. </span>  

 

main.java是主控部分,演示了Panel的使用:

  1. <span style="font-family:Comic Sans MS;font-size:18px;">public class main extends Activity {  
  2.     public Panel panel;  
  3.     public LinearLayout container;  
  4.     public GridView gridview;  
  5.     public void onCreate(Bundle savedInstanceState) {  
  6.         super.onCreate(savedInstanceState);  
  7.         setContentView(R.layout.main);  
  8.         this.setTitle("“可动态布局”的抽屉组件之构建基础-----hellogv");  
  9.         gridview = (GridView) findViewById(R.id.gridview);  
  10.         container=(LinearLayout)findViewById(R.id.container);  
  11.         panel=new Panel(this,gridview,200,LayoutParams.FILL_PARENT);  
  12.         container.addView(panel);//加入Panel控件  
  13.           
  14.         //新建测试组件  
  15.         TextView tvTest=new TextView(this);  
  16.         tvTest.setLayoutParams(new LayoutParams(LayoutParams.FILL_PARENT,LayoutParams.FILL_PARENT));  
  17.         tvTest.setText("测试组件,红字白底");  
  18.         tvTest.setTextColor(Color.RED);  
  19.         tvTest.setBackgroundColor(Color.WHITE);  
  20.         //加入到Panel里面  
  21.         panel.fillPanelContainer(tvTest);  
  22.           
  23.         panel.setPanelClosedEvent(panelClosedEvent);  
  24.         panel.setPanelOpenedEvent(panelOpenedEvent);  
  25.           
  26.         //往GridView填充测试数据  
  27.         ArrayList<HashMap<String, Object>> lstImageItem = new ArrayList<HashMap<String, Object>>();  
  28.         for (int i = 0; i < 100; i++) {  
  29.             HashMap<String, Object> map = new HashMap<String, Object>();  
  30.             map.put("ItemImage", R.drawable.icon);  
  31.             map.put("ItemText""NO." + String.valueOf(i));  
  32.             lstImageItem.add(map);  
  33.         }  
  34.   
  35.         SimpleAdapter saImageItems = new SimpleAdapter(this,   
  36.                 lstImageItem,  
  37.                 R.layout.item,   
  38.                 new String[] { "ItemImage""ItemText" },  
  39.                 new int[] { R.id.ItemImage, R.id.ItemText });  
  40.         gridview.setAdapter(saImageItems);  
  41.         gridview.setOnItemClickListener(new ItemClickListener());  
  42.           
  43.     }  
  44.   
  45.     PanelClosedEvent panelClosedEvent =new PanelClosedEvent(){  
  46.   
  47.         @Override  
  48.         public void onPanelClosed(View panel) {  
  49.             Log.e("panelClosedEvent","panelClosedEvent");  
  50.         }  
  51.           
  52.     };  
  53.       
  54.     PanelOpenedEvent panelOpenedEvent =new PanelOpenedEvent(){  
  55.   
  56.         @Override  
  57.         public void onPanelOpened(View panel) {  
  58.             Log.e("panelOpenedEvent","panelOpenedEvent");  
  59.         }  
  60.           
  61.     };  
  62.       
  63.     class ItemClickListener implements OnItemClickListener {  
  64.         @Override  
  65.         public void onItemClick(AdapterView<?> arg0,View arg1, int arg2, long arg3) {  
  66.             @SuppressWarnings("unchecked")  
  67.             HashMap<String, Object> item = (HashMap<String, Object>) arg0  
  68.                     .getItemAtPosition(arg2);  
  69.             setTitle((String) item.get("ItemText"));  
  70.         }  
  71.   
  72.     }</span>  

后面还会继续介绍如何在Panel加入拖拉效果的处理!

______________________________________________________________________________

上次介绍了基础篇,讲解了自定义抽屉控件的基础实现,这次就在基础篇的基础上加入拖拉功能。拖拉功能基于GestureDetector,GestureDetector的基本使用方式不是本文介绍的重点,有兴趣的童鞋可以上网查询相关的教程。

       本文的抽屉控件相对于基础篇的抽屉控件多了以下功能:

1.支持手势拖拉

2.拖拉到一半时,可以自动展开或者收缩。
具体如下图:

 

 

本文的源码可以到这里下载:http://download.csdn.net/detail/hellogv/3642418

只贴出抽屉组件的源码,其他源文件与基础篇的一样:

  1. <span style="font-family:Comic Sans MS;font-size:18px;">public class Panel extends LinearLayout implements GestureDetector.OnGestureListener{  
  2.       
  3.     public interface PanelClosedEvent {  
  4.         void onPanelClosed(View panel);  
  5.     }  
  6.       
  7.     public interface PanelOpenedEvent {  
  8.         void onPanelOpened(View panel);  
  9.     }  
  10.       
  11.     private final static int HANDLE_WIDTH=30;  
  12.     private final static int MOVE_WIDTH=20;  
  13.     private Button btnHandler;  
  14.     private LinearLayout panelContainer;  
  15.     private int mRightMargin=0;  
  16.     private Context mContext;  
  17.     private GestureDetector mGestureDetector;  
  18.     private boolean mIsScrolling=false;  
  19.     private float mScrollX;  
  20.     private PanelClosedEvent panelClosedEvent=null;  
  21.     private PanelOpenedEvent panelOpenedEvent=null;  
  22.       
  23.     public Panel(Context context,View otherView,int width,int height) {  
  24.         super(context);  
  25.         this.mContext=context;  
  26.           
  27.         //定义手势识别  
  28.         mGestureDetector = new GestureDetector(mContext,this);  
  29.         mGestureDetector.setIsLongpressEnabled(false);  
  30.           
  31.         //改变Panel附近组件的属性  
  32.         LayoutParams otherLP=(LayoutParams) otherView.getLayoutParams();  
  33.         otherLP.weight=1;  
  34.         otherView.setLayoutParams(otherLP);  
  35.           
  36.         //设置Panel本身的属性  
  37.         LayoutParams lp=new LayoutParams(width, height);  
  38.         lp.rightMargin=-lp.width+HANDLE_WIDTH;  
  39.         mRightMargin=Math.abs(lp.rightMargin);  
  40.         this.setLayoutParams(lp);  
  41.         this.setOrientation(LinearLayout.HORIZONTAL);  
  42.           
  43.         //设置Handler的属性  
  44.         btnHandler=new Button(context);  
  45.         btnHandler.setLayoutParams(new LayoutParams(HANDLE_WIDTH,height));  
  46.         //btnHandler.setOnClickListener(handlerClickEvent);  
  47.         btnHandler.setOnTouchListener(handlerTouchEvent);  
  48.         this.addView(btnHandler);  
  49.           
  50.         //设置Container的属性  
  51.         panelContainer=new LinearLayout(context);  
  52.         panelContainer.setLayoutParams(new LayoutParams(LayoutParams.FILL_PARENT,  
  53.                 LayoutParams.FILL_PARENT));  
  54.         this.addView(panelContainer);  
  55.   
  56.     }  
  57.   
  58.     private View.OnTouchListener handlerTouchEvent=new View.OnTouchListener() {  
  59.           
  60.         @Override  
  61.         public boolean onTouch(View v, MotionEvent event) {  
  62.             if(event.getAction()==MotionEvent.ACTION_UP && //onScroll时的ACTION_UP  
  63.                     mIsScrolling==true)  
  64.             {  
  65.                 LayoutParams lp=(LayoutParams) Panel.this.getLayoutParams();  
  66.                 if (lp.rightMargin >= (-mRightMargin/2)) {//往左超过一半  
  67.                     new AsynMove().execute(new Integer[] { MOVE_WIDTH });// 正数展开  
  68.                 }   
  69.                 else if (lp.rightMargin < (-mRightMargin/2)) {//往右拖拉  
  70.                     new AsynMove().execute(new Integer[] { -MOVE_WIDTH });// 负数收缩  
  71.                 }  
  72.             }  
  73.             return mGestureDetector.onTouchEvent(event);   
  74.         }  
  75.     };  
  76.   
  77.     /** 
  78.      * 定义收缩时的回调函数 
  79.      * @param event 
  80.      */  
  81.     public void setPanelClosedEvent(PanelClosedEvent event)  
  82.     {  
  83.         this.panelClosedEvent=event;  
  84.     }  
  85.       
  86.     /** 
  87.      * 定义展开时的回调函数 
  88.      * @param event 
  89.      */  
  90.     public void setPanelOpenedEvent(PanelOpenedEvent event)  
  91.     {  
  92.         this.panelOpenedEvent=event;  
  93.     }  
  94.       
  95.     /** 
  96.      * 把View放在Panel的Container 
  97.      * @param v 
  98.      */  
  99.     public void fillPanelContainer(View v)  
  100.     {  
  101.         panelContainer.addView(v);  
  102.     }  
  103.       
  104.     /** 
  105.      * 异步移动Panel 
  106.      * @author hellogv  
  107.      */  
  108.     class AsynMove extends AsyncTask<Integer, Integer, Void> {  
  109.   
  110.         @Override  
  111.         protected Void doInBackground(Integer... params) {  
  112.             int times;  
  113.             if (mRightMargin % Math.abs(params[0]) == 0)// 整除  
  114.                 times = mRightMargin / Math.abs(params[0]);  
  115.             else  
  116.                 // 有余数  
  117.                 times = mRightMargin / Math.abs(params[0]) + 1;  
  118.   
  119.             for (int i = 0; i < times; i++) {  
  120.                 publishProgress(params);  
  121.                 try {  
  122.                     Thread.sleep(Math.abs(params[0]));  
  123.                 } catch (InterruptedException e) {  
  124.                     // TODO Auto-generated catch block  
  125.                     e.printStackTrace();  
  126.                 }  
  127.             }  
  128.             return null;  
  129.         }  
  130.   
  131.         @Override  
  132.         protected void onProgressUpdate(Integer... params) {  
  133.             LayoutParams lp = (LayoutParams) Panel.this.getLayoutParams();  
  134.             if (params[0] < 0)  
  135.                 lp.rightMargin = Math.max(lp.rightMargin + params[0],  
  136.                         (-mRightMargin));  
  137.             else  
  138.                 lp.rightMargin = Math.min(lp.rightMargin + params[0], 0);  
  139.   
  140.             if(lp.rightMargin==0 && panelOpenedEvent!=null){//展开之后  
  141.                 panelOpenedEvent.onPanelOpened(Panel.this);//调用OPEN回调函数  
  142.             }  
  143.             else if(lp.rightMargin==-(mRightMargin) && panelClosedEvent!=null){//收缩之后  
  144.                 panelClosedEvent.onPanelClosed(Panel.this);//调用CLOSE回调函数  
  145.             }  
  146.             Panel.this.setLayoutParams(lp);  
  147.         }  
  148.     }  
  149.   
  150.     @Override  
  151.     public boolean onDown(MotionEvent e) {  
  152.         mScrollX=0;  
  153.         mIsScrolling=false;  
  154.         return false;  
  155.     }  
  156.   
  157.     @Override  
  158.     public boolean onSingleTapUp(MotionEvent e) {  
  159.         LayoutParams lp = (LayoutParams) Panel.this.getLayoutParams();  
  160.         if (lp.rightMargin < 0)// CLOSE的状态  
  161.             new AsynMove().execute(new Integer[] { MOVE_WIDTH });// 正数展开  
  162.         else if (lp.rightMargin >= 0)// OPEN的状态  
  163.             new AsynMove().execute(new Integer[] { -MOVE_WIDTH });// 负数收缩  
  164.         return false;  
  165.     }  
  166.       
  167.     @Override  
  168.     public boolean onScroll(MotionEvent e1, MotionEvent e2, float distanceX,  
  169.             float distanceY) {  
  170.         mIsScrolling=true;  
  171.         mScrollX+=distanceX;  
  172.           
  173.         LayoutParams lp=(LayoutParams) Panel.this.getLayoutParams();  
  174.         if (lp.rightMargin < -1 && mScrollX > 0) {//往左拖拉  
  175.             lp.rightMargin = Math.min((lp.rightMargin + (int) mScrollX),0);  
  176.             Panel.this.setLayoutParams(lp);  
  177.             Log.e("onScroll",lp.rightMargin+"");  
  178.         }   
  179.         else if (lp.rightMargin > -(mRightMargin) && mScrollX < 0) {//往右拖拉  
  180.             lp.rightMargin = Math.max((lp.rightMargin + (int) mScrollX),-mRightMargin);  
  181.             Panel.this.setLayoutParams(lp);  
  182.         }  
  183.           
  184.         if(lp.rightMargin==0 && panelOpenedEvent!=null){//展开之后  
  185.             panelOpenedEvent.onPanelOpened(Panel.this);//调用OPEN回调函数  
  186.         }  
  187.         else if(lp.rightMargin==-(mRightMargin) && panelClosedEvent!=null){//收缩之后  
  188.             panelClosedEvent.onPanelClosed(Panel.this);//调用CLOSE回调函数  
  189.         }  
  190.         Log.e("onScroll",lp.rightMargin+"");  
  191.           
  192.         return false;  
  193.     }  
  194.       
  195.     @Override  
  196.     public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX,  
  197.             float velocityY) {return false;}  
  198.     @Override  
  199.     public void onLongPress(MotionEvent e) {}  
  200.     @Override  
  201.     public void onShowPress(MotionEvent e) {}  
  202.   
  203. }  
  204. </span> 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值