转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/28441197

如今非常多APP都给ScrollView加入了反弹效果。QQ、小米私密短信等。恰好在网上看到一个类:BounceScrollView ,

原创地址是:http://blog.csdn.net/h7870181/article/details/8960430 。 可惜作者没有提供一个效果图。于是我发现小米短信列表页往下拉。有反弹效果,且拉到1/3以上时,会打开私密短信列表。小米的用户能够试试。

我在作者BounceScrollView 类的基础上改动了一下。写了一个样例,给大家分享下。

效果图:(模拟器的效果不佳,凑合能够看出来效果)

1、首先是布局文件:

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
> <com.example.zhy_bouncescrollview02.BounceScrollView
android:background="@drawable/shakehideimg_man"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:id="@+id/id_scrollView" > <com.example.zhy_bouncescrollview02.MyListView
android:background="#fff"
android:id="@+id/id_listView"
android:layout_width="fill_parent"
android:layout_height="wrap_content" >
</com.example.zhy_bouncescrollview02.MyListView>
</com.example.zhy_bouncescrollview02.BounceScrollView> </RelativeLayout>

一个自己定义的ScrollView内部包括着一个自己定义的ListView,自己定义ScrollView是为了加入反弹效果,自己定义ListView是为了解决ScrollView和ListView嵌套产生的问题。

2、BounceScrollView

package com.example.zhy_bouncescrollview02;

import android.content.Context;
import android.graphics.Rect;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.view.View;
import android.view.animation.Animation;
import android.view.animation.TranslateAnimation;
import android.widget.ScrollView; /**
* 支持上下反弹效果的ScrollView
*
* @author zhy
*
*/
public class BounceScrollView extends ScrollView
{ private boolean isCalled ; private Callback mCallback; /**
* 包括的View
*/
private View mView;
/**
* 存储正常时的位置
*/
private Rect mRect = new Rect(); /**
* y坐标
*/
private int y; private boolean isFirst = true; public BounceScrollView(Context context, AttributeSet attrs)
{
super(context, attrs);
} /***
* 依据 XML 生成视图工作完毕.该函数在生成视图的最后调用,在全部子视图加入完之后. 即使子类覆盖了 onFinishInflate
* 方法。也应该调用父类的方法,使该方法得以运行.
*/
@Override
protected void onFinishInflate()
{
if (getChildCount() > 0)
mView = getChildAt(0);
super.onFinishInflate();
} @Override
public boolean onTouchEvent(MotionEvent ev)
{
if (mView != null)
{
commonOnTouch(ev);
} return super.onTouchEvent(ev);
} private void commonOnTouch(MotionEvent ev)
{
int action = ev.getAction();
int cy = (int) ev.getY();
switch (action)
{
case MotionEvent.ACTION_DOWN:
break;
/**
* 尾随手指移动
*/
case MotionEvent.ACTION_MOVE: int dy = cy - y;
if (isFirst)
{
dy = 0;
isFirst = false;
}
y = cy; if (isNeedMove())
{
if (mRect.isEmpty())
{
/**
* 记录移动前的位置
*/
mRect.set(mView.getLeft(), mView.getTop(),
mView.getRight(), mView.getBottom());
} mView.layout(mView.getLeft(), mView.getTop() + 2 * dy / 3,
mView.getRight(), mView.getBottom() + 2 * dy / 3); if (shouldCallBack(dy))
{
if (mCallback != null)
{
if(!isCalled)
{
isCalled = true ;
resetPosition();
mCallback.callback(); }
}
}
} break;
/**
* 反弹回去
*/
case MotionEvent.ACTION_UP:
if (!mRect.isEmpty())
{
resetPosition();
}
break; }
} /**
* 当从上往下,移动距离达到一半时,回调接口
*
* @return
*/
private boolean shouldCallBack(int dy)
{ if (dy > 0 && mView.getTop() > getHeight() / 2)
return true;
return false;
} private void resetPosition()
{
Animation animation = new TranslateAnimation(0, 0, mView.getTop(),
mRect.top);
animation.setDuration(200);
animation.setFillAfter(true);
mView.startAnimation(animation);
mView.layout(mRect.left, mRect.top, mRect.right, mRect.bottom);
mRect.setEmpty();
isFirst = true;
isCalled = false ;
} /***
* 是否须要移动布局 inner.getMeasuredHeight():获取的是控件的总高度
*
* getHeight():获取的是屏幕的高度
*
* @return
*/
public boolean isNeedMove()
{
int offset = mView.getMeasuredHeight() - getHeight();
int scrollY = getScrollY();
// 0是顶部,后面那个是底部
if (scrollY == 0 || scrollY == offset)
{
return true;
}
return false;
} public void setCallBack(Callback callback)
{
mCallback = callback;
} interface Callback
{
void callback();
} }

主要就是监听onTouchEvent,当MOVE时,ScrollView中的控件尾随手指移动,UP时恢复原来的位置;当达到1/2时。会调用用户设置的回调,细节就自己看代码了。

3、MyListView

package com.example.zhy_bouncescrollview02;

import android.content.Context;
import android.util.AttributeSet;
import android.widget.ListView; /**
* 解决ScrollView与ListView的嵌套问题
* @author zhy
*
*/
public class MyListView extends ListView
{ public MyListView(Context context, AttributeSet attrs)
{
super(context, attrs);
} public MyListView(Context context, AttributeSet attrs, int defStyle)
{
super(context, attrs, defStyle);
} public MyListView(Context context)
{
super(context);
} @Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec)
{ /**
* 解决ScrollView与ListView的嵌套问题
*/
int expandSpec = MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2,
MeasureSpec.AT_MOST); super.onMeasure(widthMeasureSpec, expandSpec);
} }

4、主Activity

package com.example.zhy_bouncescrollview02;

import java.util.ArrayList;
import java.util.Arrays; import android.app.Activity;
import android.content.Intent;
import android.os.Bundle;
import android.widget.ArrayAdapter;
import android.widget.ListView;
import android.widget.Toast; import com.example.zhy_bouncescrollview02.BounceScrollView.Callback; public class MainActivity extends Activity
{ private ListView mListView;
private BounceScrollView mScrollView; @Override
protected void onCreate(Bundle savedInstanceState)
{
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mScrollView = (BounceScrollView) findViewById(R.id.id_scrollView);
mScrollView.setCallBack(new Callback()
{ @Override
public void callback()
{
Toast.makeText(MainActivity.this, "you can do something!", 0)
.show();
Intent intent = new Intent(MainActivity.this,
SecondActivity.class);
startActivity(intent);
overridePendingTransition(R.anim.fade_in, R.anim.fade_out);
}
});
mListView = (ListView) findViewById(R.id.id_listView);
mListView.setAdapter(new ArrayAdapter<String>(this,
android.R.layout.simple_list_item_1, new ArrayList<String>(
Arrays.asList("Hello", "World", "Welcome", "Java",
"Android", "Lucene", "C++", "C#", "HTML",
"Welcome", "Java", "Android", "Lucene", "C++",
"C#", "HTML"))));
} }

MainActivity代码也非常easy,初始化两个控件,设置了下ScrollView的Callback。

好了,好几天没写代码了。就这样吧。

源代码点击下载

ScrollView反弹效果 仿小米私密短信效果的更多相关文章

  1. jQ效果:jQuery之插件开发短信发送倒计时功能

    实现的主要功能如下: 1.点击按钮的时候,可以进行倒计时,倒计时自定义. 2.当接收短信失败后,倒计时停止,可点击重新发送短信. 3.点击的元素支持一般标签和input标签. html代码: < ...

  2. discuz! 设置私密论坛版块的方法

    Discuz!的强大功能不用细说, 话说对于有一部分需要设置具有一定访问权限的用户才能浏览的版块内容的话. 可能很多朋友不太清楚, 为了解决这个问题, 第一步以管理员的身份登陆, 然后 论坛-> ...

  3. android sim 卡短信读写

    因为对短信读写操作的api 被隐藏了 , 我们须要使用<Java反射机制的学习>一文中提到的反射的方法得到隐藏API . 这有一个用例大家能够下载http://zhushou.360.cn ...

  4. Android收发短信

    效果:点击发送短信开始发送短信 收到短信时将短信的内容显示出来 代码如下: 一.权限声明 <uses-permission android:name="android.permissi ...

  5. java实现发短信功能---腾讯云短信

    目录 java实现发短信功能 前言 开发环境 腾讯云 ---短信 代码 效果 结束语 java实现发短信功能 前言 如今发短信功能已经成为互联网公司的标配,本篇文章将一步步实现java发送短信 考察了 ...

  6. ScrollView反弹效果的实现

    发现非常多APP的界面都能够滑动,QQ.微信等等,自己琢磨了下.效果例如以下: 代码:ScrollView package com.wsj.wsjdemo; import android.conten ...

  7. uni-app仿抖音APP短视频+直播+聊天实例|uniapp全屏滑动小视频+直播

    基于uniapp+uView-ui跨端H5+小程序+APP短视频|直播项目uni-ttLive. uni-ttLive一款全新基于uni-app技术开发的仿制抖音/快手短视频直播项目.支持全屏丝滑般上 ...

  8. iOS仿支付宝首页的刷新布局效果

    代码地址如下:http://www.demodashi.com/demo/12753.html XYAlipayRefreshDemo 运行效果 动画效果分析 1.UI需要变动,向上滑动的时候,顶部部 ...

  9. 【Github】 Github访问不是私密连接问题

    前言 GitHub是一个软件项目的托管平台,是我们经常需要访问的,我原本在学校时候虽然网速比较慢,但是还以能够满足一些代码下载和上传的,在暑假回到家,再去访问的时候就出现了不能访问的问题. 问题描述 ...

随机推荐

  1. STL源码剖析读书笔记--第6章&第7章--算法与仿函数

    老实说,这两章内容还蛮多的,但是其实在应用中一点点了解比较好.所以我决定这两张在以后使用过程中零零散散地总结,这个时候就说些基本概念好了.实际上,这两个STL组件都及其重要,我不详述一方面是自己偷懒, ...

  2. jquery属性选择器中|value和^value的区别

    jquery的属性选择中有两个比较混淆:一个是[attribute^value], 另一个是[attribute|value]. 先看解释: [attribute^value]:选取属性的值以valu ...

  3. Mellanox OFED2.1-X安装记录

    ---恢复内容开始--- 1,tcl,tk,gcc-gfortran,libnl-devel依赖包

  4. 给Asp.net MVC Forms 验证设置角色访问控制

    当我们使用Asp.net MVC Forms方式验证用户, 然后设置Controller 或 Action 的 Authorize属性时, 默认情况下只有Users属性可以设置(这里的Users通常是 ...

  5. sql的join用法

    SQL join 用于把来自两个或多个表的行结合起来,sql join主要包括inner join. left join .right join .full outer join. 先介绍一下表里面的 ...

  6. apache配置虚拟主机后,启动速度慢

    apache配置虚拟主机后,启动速度慢且提示“the requested operation has failed” 可以通过在cmd下启动,来查找问题(命令中的“apache2.2”,是服务名,根据 ...

  7. 关于 3750 vlan 设置

    >#config int vlan 1                \\进入vlan1,vlan 1 是默认存在的,你不能改名字,也不能删除.你也可以建个新的vlan做你的管理口. ip ad ...

  8. linux运维工程师

    本人是linux运维工程师,对这方面有点心得,现在我说说要掌握哪方面的工具吧说到工具,在行外可以说是技能,在行内我们一般称为工具,就是运维必须要掌握的工具.我就大概列出这几方面,这样入门就基本没问题了 ...

  9. android AsyncHttpClient 开源框架的使用

    AsyncHttpClient 1.在很多时候android都需要进行网络的操作,而android自带的HttpClient可以实现,但要进行很多网络连接的时候(如:下载很多图片),就需要线程池来进行 ...

  10. js打开新页面 关闭当前页 关闭父页面

    js打开新页面.关闭当前页.关闭父页面 2010-04-29 14:04:13|  分类: 页面与JavaScript |  标签: |字号大中小 订阅     //关闭当前页面,并且打开新页面,(不 ...