Android教程網
  1. 首頁
  2. Android 技術
  3. Android 手機
  4. Android 系統教程
  5. Android 游戲
 Android教程網 >> Android技術 >> Android開發實例 >> Android實現可動態布局的抽屜(一)

Android實現可動態布局的抽屜(一)

編輯:Android開發實例

      以前曾經介紹過 Android實現類似Launch的“多方向”抽屜效果  ,當這個抽屜組件不與周圍組件發生壓擠的情況下(周圍組件布局不變),是比較好使的,但是如果需要對周圍組件擠壓,則用起來欠缺美觀了。

       如下圖。在對周圍壓擠的情況下,抽屜是先把周圍的組件一次性壓擠,再通過動畫效果展開/收縮的,這種做法的好處是快速簡單,壞處是如果擠壓范圍過大,則效果生硬。

 

      本文實現的自定義抽屜組件,主要針對這種壓擠效果做出改良,漸進式壓擠周圍組件,使得過渡效果更加美觀。如下圖。

 

 

本文實現的抽屜原理是醬紫:

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. <?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>  

Panel.java是本文核心,抽屜組件的源碼,這個抽屜只實現了從右往左的彈出/從左往右的收縮,讀者可以根據自己的需要修改源碼來改變抽屜動作的方向:

 

  1. 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.  

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加入拖拉效果的處理!

 

  1. 上一頁:
  2. 下一頁:
熱門文章
閱讀排行版
Copyright © Android教程網 All Rights Reserved