1、效果图
2、实现简介
1、布局结构
(1)、项目结构:
(2)、代码结构
gitosc代码结构链接
一个主Activity,一个PullToZoomScrollView(即自定义的ScrollView),还有一个工具类(可以忽略)
(3)、布局结构
gitosc布局结构链接
主布局:分为两块,一个为TopBar,即标题栏(这个不用说了),另外一个是自定义的ScrollView,即PullToZoomScrollView。既然是ScrollView,它也是ViewGroop,所以它内部也可以包含多个布局。
PullToZoomScrollView布局:也包含两块,一个是上面要下拉可以变大的布局(即含背景图的布局),一个就是下面的文字显示布局。我这里是分别写两个layout布局,在主布局里include进去的。
3、实现代码
代码链接:gitos代码链接
觉得下面代码不好看的,可以点上面的链接去开源中国的Git里代码展示页面看。代码里注释很详细
package app.jiyi.com.pulltozoomscrollview;
import android.animation.ObjectAnimator;
import android.content.Context;
import android.util.AttributeSet;
import android.util.DisplayMetrics;
import android.util.TypedValue;
import android.view.MotionEvent;
import android.view.View;
import android.view.ViewGroup;
import android.view.WindowManager;
import android.widget.LinearLayout;
import android.widget.ScrollView;
/**
* Created by JIYI on 2015/8/10.
*/
public class PullToZoomScrollView extends ScrollView{
private boolean isonce;
private LinearLayout mParentView;
private ViewGroup mTopView;
private int mScreenHeight;
private int mTopViewHeight;
private int mCurrentOffset=0;
private ObjectAnimator oa;
/**
* 初始化获取高度值,并记录
* @param context
* @param attrs
*/
public PullToZoomScrollView(Context context, AttributeSet attrs) {
super(context, attrs);
this.setOverScrollMode(View.OVER_SCROLL_NEVER);
WindowManager wm= (WindowManager) getContext().getSystemService(Context.WINDOW_SERVICE);
DisplayMetrics metrics=new DisplayMetrics();
wm.getDefaultDisplay().getMetrics(metrics);
mScreenHeight=metrics.heightPixels;
mTopViewHeight=mScreenHeight/2-(int) TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, 90, context.getResources().getDisplayMetrics());
}
/**
* 将记录的值设置到控件上,并只让控件设置一次
* @param widthMeasureSpec
* @param heightMeasureSpec
*/
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
super.onMeasure(widthMeasureSpec, heightMeasureSpec);
if(!isonce) {
mParentView = (LinearLayout) this.getChildAt(0);
mTopView = (ViewGroup) mParentView.getChildAt(0);
mTopView.getLayoutParams().height = mTopViewHeight;
isonce=true;
}
}
private float startY=0;
private boolean isBig;
private boolean isTouchOne;
private float distance=0;
@Override
public boolean onTouchEvent(MotionEvent ev) {
int action =ev.getAction();
switch (action){
case MotionEvent.ACTION_DOWN:
break;
case MotionEvent.ACTION_MOVE:
if(mCurrentOffset<=0){
if(!isTouchOne){
startY=ev.getY();
isTouchOne=true;
}
distance=ev.getY()-startY;
if(distance>0){
isBig=true;
setT((int)-distance/4);
}
}
break;
case MotionEvent.ACTION_UP:
if(isBig) {
reset();
isBig=false;
}
isTouchOne=false;
break;
}
return super.onTouchEvent(ev);
}
/**
* 对象动画要有的设置方法
* @param t
*/
public void setT(int t) {
scrollTo(0, 0);
if (t < 0) {
mTopView.getLayoutParams().height = mTopViewHeight-t;
mTopView.requestLayout();
}
}
/**
* 主要用于释放手指后的回弹效果
*/
private void reset() {
if (oa != null && oa.isRunning()) {
return;
}
oa = ObjectAnimator.ofInt(this, "t", (int)-distance / 4, 0);
oa.setDuration(150);
oa.start();
}
/**
* 这个是设置向上滑动时,上半部分View滑动速度让其小于下半部分
* @param l
* @param t
* @param oldl
* @param oldt
*/
@Override
protected void onScrollChanged(int l, int t, int oldl, int oldt) {
super.onScrollChanged(l, t, oldl, oldt);
mCurrentOffset = t;
if (t <= mTopViewHeight&&t>=0&&!isBig) {
mTopView.setTranslationY(t / 2);
}
if(isBig){
scrollTo(0,0);
}
}
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42
- 43
- 44
- 45
- 46
- 47
- 48
- 49
- 50
- 51
- 52
- 53
- 54
- 55
- 56
- 57
- 58
- 59
- 60
- 61
- 62
- 63
- 64
- 65
- 66
- 67
- 68
- 69
- 70
- 71
- 72
- 73
- 74
- 75
- 76
- 77
- 78
- 79
- 80
- 81
- 82
- 83
- 84
- 85
- 86
- 87
- 88
- 89
- 90
- 91
- 92
- 93
- 94
- 95
- 96
- 97
- 98
- 99
- 100
- 101
- 102
- 103
- 104
- 105
- 106
- 107
- 108
- 109
- 110
- 111
- 112
- 113
- 114
- 115
- 116
- 117
- 118
- 119
- 120
- 121
- 122
- 123
- 124
- 125
- 126
- 127
- 128
- 129
- 130
- 131
- 132
- 133
- 134
- 135
- 136
- 137
- 138
- 139
- 140
- 141
- 142
- 143
- 144
- 145
- 146
- 147
- 148
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42
- 43
- 44
- 45
- 46
- 47
- 48
- 49
- 50
- 51
- 52
- 53
- 54
- 55
- 56
- 57
- 58
- 59
- 60
- 61
- 62
- 63
- 64
- 65
- 66
- 67
- 68
- 69
- 70
- 71
- 72
- 73
- 74
- 75
- 76
- 77
- 78
- 79
- 80
- 81
- 82
- 83
- 84
- 85
- 86
- 87
- 88
- 89
- 90
- 91
- 92
- 93
- 94
- 95
- 96
- 97
- 98
- 99
- 100
- 101
- 102
- 103
- 104
- 105
- 106
- 107
- 108
- 109
- 110
- 111
- 112
- 113
- 114
- 115
- 116
- 117
- 118
- 119
- 120
- 121
- 122
- 123
- 124
- 125
- 126
- 127
- 128
- 129
- 130
- 131
- 132
- 133
- 134
- 135
- 136
- 137
- 138
- 139
- 140
- 141
- 142
- 143
- 144
- 145
- 146
- 147
- 148
4、Demo下载
CSDN下载
github下载
gitos下载