前面我们已经介绍了如何让你的ScrollView,ListView具有弹性,

今天,我们在前面的基础上,做一下适当的修改,让那些既不是ScrollView,也不是ListView的Activity页面,也能具有弹性。

先上图:

图中是最简单的一个Android测试程序。默认的效果TextView是无法划动的。

进过我们小小的改造,就能让整个UI具有弹性效果。

改动如下:

  1. protected void onCreate(Bundle savedInstanceState) {
  2. super.onCreate(savedInstanceState);
  3. View view = getLayoutInflater().inflate(R.layout.activity_main, null);
  4. view.setOnTouchListener(new ElasticTouchListener());
  5. setContentView(view);
  6. }

我们仅仅只需要在需要弹性的Root VIew 上加上监听。就能达到如果的效果了。

ElasticTouchListener源码:

  1. public class ElasticTouchListener implements OnTouchListener {
  2. private View inner;
  3. View[] children;
  4. private float y;
  5. private Rect normal = new Rect();
  6. private boolean animationFinish = true;
  7. private int[] tops;
  8. private int[] bottoms;
  9. @Override
  10. public boolean onTouch(View v, MotionEvent ev) {
  11. if (inner == null && children == null) {
  12. if (v instanceof ViewGroup) {
  13. ViewGroup group = (ViewGroup) v;
  14. int count = group.getChildCount();
  15. if (count > 0) {
  16. children = new View[count];
  17. tops = new int[count];
  18. bottoms = new int[count];
  19. for (int i = 0; i < count; i++) {
  20. children[i] = group.getChildAt(i);
  21. tops[i] = children[i].getTop();
  22. bottoms[i] = children[i].getBottom();
  23. }
  24. }
  25. }
  26. inner = v;
  27. }
  28. if (animationFinish && (inner != null || children != null)) {
  29. int action = ev.getAction();
  30. switch (action) {
  31. case MotionEvent.ACTION_DOWN:
  32. //              System.out.println("ACTION_DOWN");
  33. y = ev.getY();
  34. break;
  35. case MotionEvent.ACTION_UP:
  36. //              System.out.println("ACTION_UP");
  37. y = 0;
  38. if (isNeedAnimation()) {
  39. animation();
  40. }
  41. inner.invalidate();
  42. break;
  43. case MotionEvent.ACTION_MOVE:
  44. //              System.out.println("ACTION_MOVE");
  45. final float preY = y == 0 ? ev.getY() : y;
  46. float nowY = ev.getY();
  47. int deltaY = (int) (preY - nowY);
  48. y = nowY;
  49. // 当滚动到最上或者最下时就不会再滚动,这时移动布局
  50. if (isNeedMove()) {
  51. if (normal.isEmpty()) {
  52. // 保存正常的布局位置
  53. normal.set(inner.getLeft(), inner.getTop(), inner.getRight(), inner.getBottom());
  54. }
  55. if (children != null) {
  56. View view = null;
  57. for (int i = 0; i < children.length; i++) {
  58. view = children[i];
  59. view.layout(view.getLeft(), view.getTop() - deltaY / 2, view.getRight(), view.getBottom() - deltaY / 2);
  60. }
  61. } else {
  62. // 移动布局
  63. inner.layout(inner.getLeft(), inner.getTop() - deltaY / 2, inner.getRight(), inner.getBottom() - deltaY / 2);
  64. }
  65. }
  66. inner.invalidate();
  67. break;
  68. default:
  69. break;
  70. }
  71. } else {
  72. return false;
  73. }
  74. return true;
  75. }
  76. // 开启动画移动
  77. public void animation() {
  78. if (children == null) {
  79. // 开启移动动画
  80. TranslateAnimation trans = new TranslateAnimation(0, 0, 0, normal.top - inner.getTop());
  81. trans.setDuration(200);
  82. trans.setAnimationListener(new AnimationListener() {
  83. @Override
  84. public void onAnimationStart(Animation animation) {
  85. animationFinish = false;
  86. }
  87. @Override
  88. public void onAnimationRepeat(Animation animation) {
  89. }
  90. @Override
  91. public void onAnimationEnd(Animation animation) {
  92. inner.clearAnimation();
  93. // 设置回到正常的布局位置
  94. inner.layout(normal.left, normal.top, normal.right, normal.bottom);
  95. normal.setEmpty();
  96. animationFinish = true;
  97. }
  98. });
  99. inner.startAnimation(trans);
  100. } else {
  101. for (int i = 0; i < children.length; i++) {
  102. final View view = children[i];
  103. if (view.getVisibility() == View.VISIBLE) {
  104. final int index = i;
  105. // 开启移动动画
  106. TranslateAnimation trans = new TranslateAnimation(0, 0, 0, tops[i] - view.getTop());
  107. trans.setDuration(200);
  108. trans.setAnimationListener(new AnimationListener() {
  109. @Override
  110. public void onAnimationStart(Animation animation) {
  111. animationFinish = false;
  112. }
  113. @Override
  114. public void onAnimationRepeat(Animation animation) {
  115. }
  116. @Override
  117. public void onAnimationEnd(Animation animation) {
  118. view.clearAnimation();
  119. // 设置回到正常的布局位置
  120. view.layout(view.getLeft(), tops[index], view.getRight(), bottoms[index]);
  121. normal.setEmpty();
  122. animationFinish = true;
  123. }
  124. });
  125. view.startAnimation(trans);
  126. }
  127. }
  128. }
  129. }
  130. // 是否需要开启动画
  131. public boolean isNeedAnimation() {
  132. return !normal.isEmpty();
  133. }
  134. // 是否需要移动布局
  135. public boolean isNeedMove() {
  136. //      int offset = inner.getMeasuredHeight() - getHeight();
  137. //      int scrollY = getScrollY();
  138. //      if (scrollY == 0 || scrollY == offset) {
  139. //          return true;
  140. //      }
  141. //      return false;
  142. //      if (children != null && children.length > 0
  143. //              && (children[children.length - 1].getBottom() <= inner.getPaddingTop()/*inner.getTop()*/
  144. //              || children[0].getTop() >= inner.getHeight()
  145. //              )) {
  146. //          return false;
  147. //      }
  148. return true;
  149. }
  150. }

本次教程就到此,

经过有弹性的ScrollView有弹性的ListView,以及本文的介绍,就可以让你的Android应用在每一个角落都拥有弹性啦。

Android自定义控件(四)——让每一个Activity UI都具有弹性的更多相关文章

  1. android activity改变另一个activity ui

    android开发之在activity中控制另一个activity的UI更新   转自:http://www.cnblogs.com/ycxyyzw/p/3875544.html 第一种方法: 遇到一 ...

  2. Android开发:向下一个activity传递数据,返回数据给上一个activity

    1.向下一个activity传递数据 activity1 Button button=(Button) findViewById(R.id.button1); button.setOnClickLis ...

  3. android 案例:从另一个activity选择信息并获取返回值

    主窗口: package com.example.test; import android.app.Activity; import android.app.AlertDialog; import a ...

  4. [android] 显示意图激活另外一个activity

    可以使用跳转的方式类似javaweb来实现界面转换 显示意图就是必须要指定开启组件的具体信息,包名,组件名,组件的class 新建一个类TwoActivity ,继承Activity类,重写onCre ...

  5. Eclipse中在android项目中出现新建一个Activity后,出现整个project的报错以及包导入以后无法执行等等情况分析。

    今天用Eclipse去写android项目,然后后面须要建一个Blank  Activity后,非常正常的建立的.然后那个Activity是基于ActionBarAtivity,要导入v7,结果由于这 ...

  6. android 利用Handler触发另一个activity方法

    如activityA代码: activityB = new ActivtyB(mHandler,CLOSE_SEARCH_MSG);//新建对像B传递,一个handler和Message,然后在act ...

  7. android自定义控件(四) View中的方法

    onFinishInflate() 当View中所有的子控件 均被映射成xml后触发 onMeasure(int, int) 确定所有子元素的大小 onLayout(boolean, int, int ...

  8. 【Android】12.3 在当前Activity中获取另一个Activity的返回值

    分类:C#.Android.VS2015: 创建日期:2016-02-23 一.简介 在上一节的示例中,通过StartActivity(Intent)方法启动另一个Activity后,这两个Activ ...

  9. Android基础整理之四大组件Activity

    最近准备系统的重新整理复习一下Android的各方面的知识,本着知识分享的原则,我就把梳理过程中一些东西给记录下来,权当一个学习笔记吧. 下面步入正题..... 什么是Activity Activit ...

随机推荐

  1. cocos2d-x CCAction(转载)

    接触开发2d后,越来越多的用到动作的内容,看到一篇关于动作比较完整的文章,最主要的是动作的类图,从类图可以更加的理解各个类之间的继承的关系,以及使用更容易的去应用 . 文章有一些方法已经被修改了,现在 ...

  2. 新手上路Tomcat 7.x和JDK的配置

    前言:这段时间又复习了JAVA以前做东西总是在IDE里面好多基础的东西都不知道,这次买了本书让自己重新认识下JAVA! 说明:以前装JDK时,我是和Mars老师学的那种开发JAVA应用程序进行配置的环 ...

  3. Java多线程练习二

    public class ex3 { public static void main(String [] args) { thread2 t1 = new thread2("hello&qu ...

  4. DG下手工处理v$archive_gap方法

    从9i以后,oracle dataguard 备库一般都不需要手工处理丢失的日志,FAL自动会帮我们处理,下面通过个案例来讲下手工处理丢失的日志的方法: 1.在备库查询有哪些日志丢失,没应用到备库 S ...

  5. angular请求传递不了数据

    var data={ 'id':ztreeParent.id } $http({ url:'/rcCategoryControler/deleteRcCategoryById', method:'GE ...

  6. AD DIV 层的知识 和 行为特效

    1.AP(绝对定位) 2.使用AP DIV层和表格结合起来完美布局网页 3.层的Z轴值越大,该层就位于比较顶的位置 4.层有可见性的属性,层溢出,层的裁剪, 5层嵌套,先把光标定位在外层里面,然后拖多 ...

  7. Visual Studio发布项目到远程服务器的步骤

    第一步: 需要远程服务器上安装Web Deploy ,下载地址:http://www.iis.net/downloads/microsoft/web-deploy PS.安装时选择完全安装. 第二步: ...

  8. flask开发restful api系列(2)

    继续上一章所讲,上一章我们最后面说道,虽然这个是很小的程序,但还有好几个要优化的地方.先复制一下老的view.py代码. # coding:utf-8 from flask import Flask, ...

  9. Python Udp Socket

    socket(套接字),传输层通信的端点,由IP和端口号组成(IP,Port),可以通过socket精确地找到服务器上的进程并与之通信 python2.6实现,基于AF_INET(网络套接字) 类型S ...

  10. 自定义Window进入和退出效果(转)

    看了android的源代码和资源文件,终于明白如何去修改设置Dialog和Activity的进入和退出效果了. 设置Dialog首先通过getWindow()方法获取它的窗口, 然后通过getAttr ...