Android 仿 窗帘效果 和 登录界面拖动效果 (Scroller类的应用) 附 2个DEMO及源码
在android学习中,动作交互是软件中重要的一部分,其中的Scroller就是提供了拖动效果的类,在网上,比如说一些Launcher实现滑屏都可以通过这个类去实现。下面要说的就是上次Scroller类学习的后的实践了。
如果你还不了解Scroller类,那请先点击:Android 界面滑动实现---Scroller类 从源码和开发文档中学习(让你的布局动起来)
了解之后再阅读以下内容,你会发现原来实现起来很简单。
- public class CurtainView extends RelativeLayout implements OnTouchListener{
- private static String TAG = "CurtainView";
- private Context mContext;
- /** Scroller 拖动类 */
- private Scroller mScroller;
- /** 屏幕高度 */
- private int mScreenHeigh = 0;
- /** 屏幕宽度 */
- private int mScreenWidth = 0;
- /** 点击时候Y的坐标*/
- private int downY = 0;
- /** 拖动时候Y的坐标*/
- private int moveY = 0;
- /** 拖动时候Y的方向距离*/
- private int scrollY = 0;
- /** 松开时候Y的坐标*/
- private int upY = 0;
- /** 广告幕布的高度*/
- private int curtainHeigh = 0;
- /** 是否 打开*/
- private boolean isOpen = false;
- /** 是否在动画 */
- private boolean isMove = false;
- /** 绳子的图片*/
- private ImageView img_curtain_rope;
- /** 广告的图片*/
- private ImageView img_curtain_ad;
- /** 上升动画时间 */
- private int upDuration = 1000;
- /** 下落动画时间 */
- private int downDuration = 500;
- public CurtainView(Context context) {
- super(context);
- init(context);
- }
- public CurtainView(Context context, AttributeSet attrs, int defStyle) {
- super(context, attrs, defStyle);
- init(context);
- }
- public CurtainView(Context context, AttributeSet attrs) {
- super(context, attrs);
- init(context);
- }
- /** 初始化 */
- private void init(Context context) {
- this.mContext = context;
- //Interpolator 设置为有反弹效果的 (Bounce:反弹)
- Interpolator interpolator = new BounceInterpolator();
- mScroller = new Scroller(context, interpolator);
- mScreenHeigh = BaseTools.getWindowHeigh(context);
- mScreenWidth = BaseTools.getWindowWidth(context);
- // 背景设置成透明
- this.setBackgroundColor(Color.argb(0, 0, 0, 0));
- final View view = LayoutInflater.from(mContext).inflate(R.layout.curtain, null);
- img_curtain_ad = (ImageView)view.findViewById(R.id.img_curtain_ad);
- img_curtain_rope = (ImageView)view.findViewById(R.id.img_curtain_rope);
- addView(view);
- img_curtain_ad.post(new Runnable() {
- @Override
- public void run() {
- // TODO Auto-generated method stub
- curtainHeigh = img_curtain_ad.getHeight();
- Log.d(TAG, "curtainHeigh= " + curtainHeigh);
- CurtainView.this.scrollTo(0, curtainHeigh);
- //注意scrollBy和scrollTo的区别
- }
- });
- img_curtain_rope.setOnTouchListener(this);
- }
- /**
- * 拖动动画
- * @param startY
- * @param dy 垂直距离, 滚动的y距离
- * @param duration 时间
- */
- public void startMoveAnim(int startY, int dy, int duration) {
- isMove = true;
- mScroller.startScroll(0, startY, 0, dy, duration);
- invalidate();//通知UI线程的更新
- }
- @Override
- protected void onLayout(boolean changed, int l, int t, int r, int b) {
- // TODO Auto-generated method stub
- super.onLayout(changed, l, t, r, b);
- }
- @Override
- public void computeScroll() {
- //判断是否还在滚动,还在滚动为true
- if (mScroller.computeScrollOffset()) {
- scrollTo(mScroller.getCurrX(), mScroller.getCurrY());
- //更新界面
- postInvalidate();
- isMove = true;
- } else {
- isMove = false;
- }
- super.computeScroll();
- }
- @Override
- public boolean onTouch(View v, MotionEvent event) {
- // TODO Auto-generated method stub
- if (!isMove) {
- int offViewY = 0;//屏幕顶部和该布局顶部的距离
- switch (event.getAction()) {
- case MotionEvent.ACTION_DOWN:
- downY = (int) event.getRawY();
- offViewY = downY - (int)event.getX();
- return true;
- case MotionEvent.ACTION_MOVE:
- moveY = (int) event.getRawY();
- scrollY = moveY - downY;
- if (scrollY < 0) {
- // 向上滑动
- if(isOpen){
- if(Math.abs(scrollY) <= img_curtain_ad.getBottom() - offViewY){
- scrollTo(0, -scrollY);
- }
- }
- } else {
- // 向下滑动
- if(!isOpen){
- if (scrollY <= curtainHeigh) {
- scrollTo(0, curtainHeigh - scrollY);
- }
- }
- }
- break;
- case MotionEvent.ACTION_UP:
- upY = (int) event.getRawY();
- if(Math.abs(upY - downY) < 10){
- onRopeClick();
- break;
- }
- if (downY > upY) {
- // 向上滑动
- if(isOpen){
- if (Math.abs(scrollY) > curtainHeigh / 2) {
- // 向上滑动超过半个屏幕高的时候 开启向上消失动画
- startMoveAnim(this.getScrollY(),
- (curtainHeigh - this.getScrollY()), upDuration);
- isOpen = false;
- } else {
- startMoveAnim(this.getScrollY(), -this.getScrollY(),upDuration);
- isOpen = true;
- }
- }
- } else {
- // 向下滑动
- if (scrollY > curtainHeigh / 2) {
- // 向上滑动超过半个屏幕高的时候 开启向上消失动画
- startMoveAnim(this.getScrollY(), -this.getScrollY(),upDuration);
- isOpen = true;
- } else {
- startMoveAnim(this.getScrollY(),(curtainHeigh - this.getScrollY()), upDuration);
- isOpen = false;
- }
- }
- break;
- default:
- break;
- }
- }
- return false;
- }
- /**
- * 点击绳索开关,会展开关闭
- * 在onToch中使用这个中的方法来当点击事件,避免了点击时候响应onTouch的衔接不完美的影响
- */
- public void onRopeClick(){
- if(isOpen){
- CurtainView.this.startMoveAnim(0, curtainHeigh, upDuration);
- }else{
- CurtainView.this.startMoveAnim(curtainHeigh,-curtainHeigh, downDuration);
- }
- isOpen = !isOpen;
- }
- }
- public class LoginView extends RelativeLayout {
- /** Scroller 拖动类 */
- private Scroller mScroller;
- /** 屏幕高度 */
- private int mScreenHeigh = 0;
- /** 屏幕宽度 */
- private int mScreenWidth = 0;
- /** 点击时候Y的坐标*/
- private int downY = 0;
- /** 拖动时候Y的坐标*/
- private int moveY = 0;
- /** 拖动时候Y的方向距离*/
- private int scrollY = 0;
- /** 松开时候Y的坐标*/
- private int upY = 0;
- /** 是否在移动*/
- private Boolean isMoving = false;
- /** 布局的高度*/
- private int viewHeight = 0;
- /** 是否打开*/
- public boolean isShow = false;
- /** 是否可以拖动*/
- public boolean mEnabled = true;
- /** 点击外面是否关闭该界面*/
- public boolean mOutsideTouchable = true;
- /** 上升动画时间 */
- private int mDuration = 800;
- private final static String TAG = "LoginView";
- public LoginView(Context context) {
- super(context);
- init(context);
- }
- public LoginView(Context context, AttributeSet attrs) {
- super(context, attrs);
- init(context);
- }
- public LoginView(Context context, AttributeSet attrs, int defStyle) {
- super(context, attrs, defStyle);
- init(context);
- }
- private void init(Context context) {
- setDescendantFocusability(FOCUS_AFTER_DESCENDANTS);
- setFocusable(true);
- mScroller = new Scroller(context);
- mScreenHeigh = BaseTools.getWindowHeigh(context);
- mScreenWidth = BaseTools.getWindowWidth(context);
- // 背景设置成透明
- this.setBackgroundColor(Color.argb(0, 0, 0, 0));
- final View view = LayoutInflater.from(context).inflate(R.layout.view_login,null);
- LayoutParams params = new LayoutParams(LayoutParams.MATCH_PARENT,LayoutParams.WRAP_CONTENT);// 如果不给他设这个,它的布局的MATCH_PARENT就不知道该是多少
- addView(view, params);// ViewGroup的大小,
- // 背景设置成透明
- this.setBackgroundColor(Color.argb(0, 0, 0, 0));
- view.post(new Runnable() {
- @Override
- public void run() {
- // TODO Auto-generated method stub
- viewHeight = view.getHeight();
- }
- });
- LoginView.this.scrollTo(0, mScreenHeigh);
- ImageView btn_close = (ImageView)view.findViewById(R.id.btn_close);
- btn_close.setOnClickListener(new OnClickListener() {
- @Override
- public void onClick(View v) {
- // TODO Auto-generated method stub
- dismiss();
- }
- });
- }
- @Override
- public boolean onInterceptTouchEvent(MotionEvent ev) {
- if(!mEnabled){
- return false;
- }
- return super.onInterceptTouchEvent(ev);
- }
- @Override
- public boolean onTouchEvent(MotionEvent event) {
- // TODO Auto-generated method stub
- switch (event.getAction()) {
- case MotionEvent.ACTION_DOWN:
- downY = (int) event.getY();
- Log.d(TAG, "downY = " + downY);
- //如果完全显示的时候,让布局得到触摸监听,如果不显示,触摸事件不拦截,向下传递
- if(isShow){
- return true;
- }
- break;
- case MotionEvent.ACTION_MOVE:
- moveY = (int) event.getY();
- scrollY = moveY - downY;
- //向下滑动
- if (scrollY > 0) {
- if(isShow){
- scrollTo(0, -Math.abs(scrollY));
- }
- }else{
- if(mScreenHeigh - this.getTop() <= viewHeight && !isShow){
- scrollTo(0, Math.abs(viewHeight - scrollY));
- }
- }
- break;
- case MotionEvent.ACTION_UP:
- upY = (int) event.getY();
- if(isShow){
- if( this.getScrollY() <= -(viewHeight /2)){
- startMoveAnim(this.getScrollY(),-(viewHeight - this.getScrollY()), mDuration);
- isShow = false;
- Log.d("isShow", "false");
- } else {
- startMoveAnim(this.getScrollY(), -this.getScrollY(), mDuration);
- isShow = true;
- Log.d("isShow", "true");
- }
- }
- Log.d("this.getScrollY()", ""+this.getScrollY());
- changed();
- break;
- case MotionEvent.ACTION_OUTSIDE:
- Log.d(TAG, "ACTION_OUTSIDE");
- break;
- default:
- break;
- }
- return super.onTouchEvent(event);
- }
- /**
- * 拖动动画
- * @param startY
- * @param dy 移动到某点的Y坐标距离
- * @param duration 时间
- */
- public void startMoveAnim(int startY, int dy, int duration) {
- isMoving = true;
- mScroller.startScroll(0, startY, 0, dy, duration);
- invalidate();//通知UI线程的更新
- }
- @Override
- public void computeScroll() {
- if (mScroller.computeScrollOffset()) {
- scrollTo(mScroller.getCurrX(), mScroller.getCurrY());
- // 更新界面
- postInvalidate();
- isMoving = true;
- } else {
- isMoving = false;
- }
- super.computeScroll();
- }
- /** 开打界面 */
- public void show(){
- if(!isShow && !isMoving){
- LoginView.this.startMoveAnim(-viewHeight, viewHeight, mDuration);
- isShow = true;
- Log.d("isShow", "true");
- changed();
- }
- }
- /** 关闭界面 */
- public void dismiss(){
- if(isShow && !isMoving){
- LoginView.this.startMoveAnim(0, -viewHeight, mDuration);
- isShow = false;
- Log.d("isShow", "false");
- changed();
- }
- }
- /** 是否打开 */
- public boolean isShow(){
- return isShow;
- }
- /** 获取是否可以拖动*/
- public boolean isSlidingEnabled() {
- return mEnabled;
- }
- /** 设置是否可以拖动*/
- public void setSlidingEnabled(boolean enabled) {
- mEnabled = enabled;
- }
- /**
- * 设置监听接口,实现遮罩层效果
- */
- public void setOnStatusListener(onStatusListener listener){
- this.statusListener = listener;
- }
- public void setOutsideTouchable(boolean touchable) {
- mOutsideTouchable = touchable;
- }
- /**
- * 显示状态发生改变时候执行回调借口
- */
- public void changed(){
- if(statusListener != null){
- if(isShow){
- statusListener.onShow();
- }else{
- statusListener.onDismiss();
- }
- }
- }
- /** 监听接口*/
- public onStatusListener statusListener;
- /**
- * 监听接口,来在主界面监听界面变化状态
- */
- public interface onStatusListener{
- /** 开打状态 */
- public void onShow();
- /** 关闭状态 */
- public void onDismiss();
- }
- @Override
- protected void onLayout(boolean changed, int l, int t, int r, int b) {
- // TODO Auto-generated method stub
- super.onLayout(changed, l, t, r, b);
- }
- }
版权声明:本文为博主原创文章,未经博主允许不得转载。
Android 仿 窗帘效果 和 登录界面拖动效果 (Scroller类的应用) 附 2个DEMO及源码的更多相关文章
- Android 仿 窗帘效果 和 登录界面拖动效果 (Scroller类的应用) 附 2个DEMO及源代码
在android学习中,动作交互是软件中重要的一部分.当中的Scroller就是提供了拖动效果的类,在网上.比方说一些Launcher实现滑屏都能够通过这个类去实现.以下要说的就是上次Scroller ...
- Android 界面滑动实现---Scroller类 从源码和开发文档中学习(让你的布局动起来)
在android学习中,动作交互是软件中重要的一部分,其中的Scroller就是提供了拖动效果的类,在网上,比如说一些Launcher实现滑屏都可以通过这个类去实现.. 例子相关博文:Androi ...
- 仿知乎app登录界面(Material Design设计框架拿来就用的TexnInputLayout)
在我脑子里还没有Material Design这种概念,就我个人而言,PC端应用扁平化设计必须成为首选,手当其冲的两款即时通讯旺旺和QQ早就完成UI扁平化的更新,然而客户端扁平化的设计本身就存在天生的 ...
- Android studio 开发一个用户登录界面
Android studio 开发一个用户登录界面 activity_main.xml <?xml version="1.0" encoding="utf-8&qu ...
- Android开发实例之miniTwitter登录界面的实现
原文: http://www.jizhuomi.com/android/example/134.html 本文要演示的Android开发实例是如何完成一个Android中的miniTwitter登录界 ...
- Android 仿 新闻阅读器 菜单弹出效果(附源码DEMO)
这一系列博文都是:(android高仿系列)今日头条 --新闻阅读器 (一) 开发中碰到问题之后实现的,觉得可能有的开发者用的到或则希望独立成一个小功能DEMO,所以就放出来这么一个DEMO. 原本觉 ...
- Android 比较好看的注册登录界面
各位看官姥爷: 对于一款android手机app而言,美观的界面使得用户有好的使用体验,而一款好看的注册登录界面也会给用户好的用户体验,那么话不多说,直接上代码 首先是一款简单的界面展示 1.登陆界面 ...
- Android中滑屏实现----触摸滑屏以及Scroller类详解 .
转:http://blog.csdn.net/qinjuning/article/details/7419207 知识点一: 关于scrollTo()和scrollBy()以及偏移坐标的设置/取值问 ...
- 商城项目实战 | 1.1 Android 仿京东商城底部布局的选择效果 —— Selector 选择器的实现
前言 本文为菜鸟窝作者刘婷的连载."商城项目实战"系列来聊聊仿"京东淘宝的购物商城"如何实现. 京东商城的底部布局的选择效果看上去很复杂,其实很简单,这主要是要 ...
随机推荐
- 【Web】CXF WebService 服务端和客户端 环境搭建及测试
cxf服务端 1.去官方下载对应的jar包:http://cxf.apache.org/ 2.maven配置相应jar包 3.修改web.xml,完成spring和cxf配置 <!-- Spri ...
- c++ ,protected 和 private修饰的构造函数
c++ protected 和 private修饰的构造函数: 1.在类的外部创建对象时,不能调用protected或private修饰的构造函数. 2.当子类中的构造函数调用父类的private构造 ...
- [Swust OJ 247]--皇帝的新衣(组合数+Lucas定理)
题目链接:http://acm.swust.edu.cn/problem/0247/ Time limit(ms): 1000 Memory limit(kb): 65535 Descriptio ...
- BZOJ 1263: [SCOI2006]整数划分( 高精度 )
yy一下发现好像越小越好...分解成3*3*3*3……这种形式是最好的...然后就是高精度了 ----------------------------------------------------- ...
- C#路径,文件,目录,I/O常见操作
C#路径,文件,目录,I/O常见操作 文件操作是程序中非常基础和重要的内容,而路径.文件.目录以及I/O都是在进行文件操作时的常见主题,这里想把这些常见的问题作个总结,对于每个问题,尽量提供 ...
- 【STL__set_的应用】
1.关于set C++ STL 之所以得到广泛的赞誉,也被很多人使用,不只是提供了像vector, string, list等方便的容器, 更重要的是STL封装了许多复杂的数据结构算法和大量常用数据结 ...
- 基于visual Studio2013解决算法导论之051区间树
题目 区间树 解决代码及点评 #include <stdio.h> #include <string.h> #include <iostream> #def ...
- Pencil OJ 02 安装
Mongo 官方的安装方法 官方教程已经很好啦,这里就不罗嗦了. 源码编译 待补.我是从这里看到的. 遇到的问题 启动时的警告信息 2015-03-06T21:01:15.526-0800 I CON ...
- sprintf,你知道多少?
选自<CSDN 社区电子杂志——C/C++杂志>http://emag.csdn.net 2005 年1 月 总第1 期 - 93 -本文作者:steedhorse(晨星)printf 可 ...
- 开源 免费 java CMS - FreeCMS1.9 职位管理
项目地址:http://code.google.com/p/freecms/ 职位管理 管理职位,实现招聘功能. 1. 职位管理 从左側管理菜单点击职位管理进入. 2. 加入职位 在职位列表下方点击& ...