Android抽屉(SlidingDrawer)的实现发 - 红黑联盟
http://www.2cto.com/kf/201301/182507.html

可动态布局的Android抽屉之基础
http://blog.csdn.net/hellogv/article/details/6789698

android抽屉实现
http://blog.csdn.net/wangkuifeng0118/article/details/7229200

Android 使用动画效果后的控件位置处理 类似系统通知栏下拉动画
http://www.pocketdigi.com/20111028/556.html

下拉通知效果-Android源码下载
http://www.eoeandroid.com/thread-239825-1-1.html

Demo源码 下载

http://pan.baidu.com/s/1c08Esj2

Android抽屉(SlidingDrawer --类似android通知栏下拉效果)_第1张图片

Android抽屉(SlidingDrawer --类似android通知栏下拉效果)_第2张图片


本文来自http://blog.csdn.net/hellogv/,欢迎转摘,引用必须注明出处!

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

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

Android抽屉(SlidingDrawer --类似android通知栏下拉效果)_第3张图片

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

Android抽屉(SlidingDrawer --类似android通知栏下拉效果)_第4张图片

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

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

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

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

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

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

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

main.xml的源码:

[html] view plain copy print ?
  1. <spanstyle="font-family:ComicSansMS;font-size:18px;"><?xmlversion="1.0"encoding="utf-8"?>
  2. <LinearLayoutxmlns: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. <GridViewandroid: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的源码:

[html] view plain copy print ?
  1. <spanstyle="font-family:ComicSansMS;font-size:18px;"><?xmlversion="1.0"encoding="utf-8"?>
  2. <RelativeLayoutxmlns:android="http://schemas.android.com/apk/res/android"
  3. android:layout_height="wrap_content"android:paddingBottom="4dip"
  4. android:layout_width="fill_parent">
  5. <ImageViewandroid:layout_height="wrap_content"android:id="@+id/ItemImage"
  6. android:layout_width="wrap_content"android:layout_centerHorizontal="true">
  7. </ImageView>
  8. <TextViewandroid: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是本文核心,抽屉组件的源码,这个抽屉只实现了从右往左的弹出/从左往右的收缩,读者可以根据自己的需要修改源码来改变抽屉动作的方向:

[java] view plain copy print ?
  1. <spanstyle="font-family:ComicSansMS;font-size:18px;">publicclassPanelextendsLinearLayout{
  2. publicinterfacePanelClosedEvent{
  3. voidonPanelClosed(Viewpanel);
  4. }
  5. publicinterfacePanelOpenedEvent{
  6. voidonPanelOpened(Viewpanel);
  7. }
  8. /**Handle的宽度,与Panel等高*/
  9. privatefinalstaticintHANDLE_WIDTH=30;
  10. /**每次自动展开/收缩的范围*/
  11. privatefinalstaticintMOVE_WIDTH=20;
  12. privateButtonbtnHandle;
  13. privateLinearLayoutpanelContainer;
  14. privateintmRightMargin=0;
  15. privateContextmContext;
  16. privatePanelClosedEventpanelClosedEvent=null;
  17. privatePanelOpenedEventpanelOpenedEvent=null;
  18. /**
  19. *otherView自动布局以适应Panel展开/收缩的空间变化
  20. *@authorGV
  21. *
  22. */
  23. publicPanel(Contextcontext,ViewotherView,intwidth,intheight){
  24. super(context);
  25. this.mContext=context;
  26. //改变Panel附近组件的属性
  27. LayoutParamsotherLP=(LayoutParams)otherView.getLayoutParams();
  28. otherLP.weight=1;//支持压挤
  29. otherView.setLayoutParams(otherLP);
  30. //设置Panel本身的属性
  31. LayoutParamslp=newLayoutParams(width,height);
  32. lp.rightMargin=-lp.width+HANDLE_WIDTH;//Panel的Container在屏幕不可视区域,Handle在可视区域
  33. mRightMargin=Math.abs(lp.rightMargin);
  34. this.setLayoutParams(lp);
  35. this.setOrientation(LinearLayout.HORIZONTAL);
  36. //设置Handle的属性
  37. btnHandle=newButton(context);
  38. btnHandle.setLayoutParams(newLayoutParams(HANDLE_WIDTH,height));
  39. btnHandle.setOnClickListener(newOnClickListener(){
  40. @Override
  41. publicvoidonClick(Viewarg0){
  42. LayoutParamslp=(LayoutParams)Panel.this.getLayoutParams();
  43. if(lp.rightMargin<0)//CLOSE的状态
  44. newAsynMove().execute(newInteger[]{MOVE_WIDTH});//正数展开
  45. elseif(lp.rightMargin>=0)//OPEN的状态
  46. newAsynMove().execute(newInteger[]{-MOVE_WIDTH});//负数收缩
  47. }
  48. });
  49. //btnHandle.setOnTouchListener(HandleTouchEvent);
  50. this.addView(btnHandle);
  51. //设置Container的属性
  52. panelContainer=newLinearLayout(context);
  53. panelContainer.setLayoutParams(newLayoutParams(LayoutParams.FILL_PARENT,
  54. LayoutParams.FILL_PARENT));
  55. this.addView(panelContainer);
  56. }
  57. /**
  58. *定义收缩时的回调函数
  59. *@paramevent
  60. */
  61. publicvoidsetPanelClosedEvent(PanelClosedEventevent)
  62. {
  63. this.panelClosedEvent=event;
  64. }
  65. /**
  66. *定义展开时的回调函数
  67. *@paramevent
  68. */
  69. publicvoidsetPanelOpenedEvent(PanelOpenedEventevent)
  70. {
  71. this.panelOpenedEvent=event;
  72. }
  73. /**
  74. *把View放在Panel的Container
  75. *@paramv
  76. */
  77. publicvoidfillPanelContainer(Viewv)
  78. {
  79. panelContainer.addView(v);
  80. }
  81. /**
  82. *异步移动Panel
  83. *@authorhellogv
  84. */
  85. classAsynMoveextendsAsyncTask<Integer,Integer,Void>{
  86. @Override
  87. protectedVoiddoInBackground(Integer...params){
  88. inttimes;
  89. if(mRightMargin%Math.abs(params[0])==0)//整除
  90. times=mRightMargin/Math.abs(params[0]);
  91. else
  92. //有余数
  93. times=mRightMargin/Math.abs(params[0])+1;
  94. for(inti=0;i<times;i++){
  95. publishProgress(params);
  96. try{
  97. Thread.sleep(Math.abs(params[0]));
  98. }catch(InterruptedExceptione){
  99. //TODOAuto-generatedcatchblock
  100. e.printStackTrace();
  101. }
  102. }
  103. returnnull;
  104. }
  105. @Override
  106. protectedvoidonProgressUpdate(Integer...params){
  107. LayoutParamslp=(LayoutParams)Panel.this.getLayoutParams();
  108. if(params[0]<0)
  109. lp.rightMargin=Math.max(lp.rightMargin+params[0],
  110. (-mRightMargin));
  111. else
  112. lp.rightMargin=Math.min(lp.rightMargin+params[0],0);
  113. if(lp.rightMargin==0&&panelOpenedEvent!=null){//展开之后
  114. panelOpenedEvent.onPanelOpened(Panel.this);//调用OPEN回调函数
  115. }
  116. elseif(lp.rightMargin==-(mRightMargin)&&panelClosedEvent!=null){//收缩之后
  117. panelClosedEvent.onPanelClosed(Panel.this);//调用CLOSE回调函数
  118. }
  119. Panel.this.setLayoutParams(lp);
  120. }
  121. }
  122. }
  123. </span>

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

[java] view plain copy print ?
  1. <spanstyle="font-family:ComicSansMS;font-size:18px;">publicclassmainextendsActivity{
  2. publicPanelpanel;
  3. publicLinearLayoutcontainer;
  4. publicGridViewgridview;
  5. publicvoidonCreate(BundlesavedInstanceState){
  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=newPanel(this,gridview,200,LayoutParams.FILL_PARENT);
  12. container.addView(panel);//加入Panel控件
  13. //新建测试组件
  14. TextViewtvTest=newTextView(this);
  15. tvTest.setLayoutParams(newLayoutParams(LayoutParams.FILL_PARENT,LayoutParams.FILL_PARENT));
  16. tvTest.setText("测试组件,红字白底");
  17. tvTest.setTextColor(Color.RED);
  18. tvTest.setBackgroundColor(Color.WHITE);
  19. //加入到Panel里面
  20. panel.fillPanelContainer(tvTest);
  21. panel.setPanelClosedEvent(panelClosedEvent);
  22. panel.setPanelOpenedEvent(panelOpenedEvent);
  23. //往GridView填充测试数据
  24. ArrayList<HashMap<String,Object>>lstImageItem=newArrayList<HashMap<String,Object>>();
  25. for(inti=0;i<100;i++){
  26. HashMap<String,Object>map=newHashMap<String,Object>();
  27. map.put("ItemImage",R.drawable.icon);
  28. map.put("ItemText","NO."+String.valueOf(i));
  29. lstImageItem.add(map);
  30. }
  31. SimpleAdaptersaImageItems=newSimpleAdapter(this,
  32. lstImageItem,
  33. R.layout.item,
  34. newString[]{"ItemImage","ItemText"},
  35. newint[]{R.id.ItemImage,R.id.ItemText});
  36. gridview.setAdapter(saImageItems);
  37. gridview.setOnItemClickListener(newItemClickListener());
  38. }
  39. PanelClosedEventpanelClosedEvent=newPanelClosedEvent(){
  40. @Override
  41. publicvoidonPanelClosed(Viewpanel){
  42. Log.e("panelClosedEvent","panelClosedEvent");
  43. }
  44. };
  45. PanelOpenedEventpanelOpenedEvent=newPanelOpenedEvent(){
  46. @Override
  47. publicvoidonPanelOpened(Viewpanel){
  48. Log.e("panelOpenedEvent","panelOpenedEvent");
  49. }
  50. };
  51. classItemClickListenerimplementsOnItemClickListener{
  52. @Override
  53. publicvoidonItemClick(AdapterView<?>arg0,Viewarg1,intarg2,longarg3){
  54. @SuppressWarnings("unchecked")
  55. HashMap<String,Object>item=(HashMap<String,Object>)arg0
  56. .getItemAtPosition(arg2);
  57. setTitle((String)item.get("ItemText"));
  58. }
  59. }</span>

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



更多相关文章

  1. Android TextView 跑马灯效果
  2. Android学习笔记3之基本组件API
  3. android 开发技巧(9)--为文本添加发亮的效果
  4. Android深入四大组件(一)应用程序启动过程
  5. android常用组件之TextView组件
  6. 基本的UI组件(TextView、EditText、Button、ImageView、SeekBar)
  7. 【Android 界面效果39】android:inputType问题
  8. Android利用TextView制作跑马灯该效果
  9. 安卓开发小效果--走马灯

随机推荐

  1. Android Adapter
  2. [Android]ListView中分割线的设置
  3. Android 人脸检测
  4. Android中文SDK-----Android新手入门
  5. android 分割线渐变色和几种颜色值定义
  6. Android中设置文本颜色的三种方法
  7. Android 获取界面上的元素
  8. 解析Android消息处理机制 ——Handler/Th
  9. Android操作系统默认图片
  10. eclipse android集成开发环境分享 32位 6