仿QQ空间滑动图片放大缩小控件

先来看一下效果:


scrollzoom_listview.gif

一、设计思路与实现步骤
1、本例是通过重写ListView来实现的,头部的图片是ListView的HeadView。定义一个headview.xml布局文件,在这个布局文件中放一个ImageView,并给IamgeView设置一个初始高度
2、实现下拉图片放大
当ListView处于顶部的时候下拉实现图片放大,这里要用的一个核心的方法是overScrollBy方法,当View的滑动超过了正常内容的边界时,该方法被调用。这里也许我解释的不是非常完美,大家可以自己去看一下该方法的注释

    /**
     * Scroll the view with standard behavior for scrolling beyond the normal
     * content boundaries. Views that call this method should override
     * {@link #onOverScrolled(int, int, boolean, boolean)} to respond to the
     * results of an over -scroll operation.
     *
     * Views can use this method to handle any touch or fling -based scrolling.
     *
     * @param deltaX Change in X in pixels
     * @param deltaY Change in Y in pixels
     * @param scrollX Current X scroll value in pixels before applying deltaX
     * @param scrollY Current Y scroll value in pixels before applying deltaY
     * @param scrollRangeX Maximum content scroll range along the X axis
     * @param scrollRangeY Maximum content scroll range along the Y axis
     * @param maxOverScrollX Number of pixels to overscroll by in either direction
     *          along the X axis.
     * @param maxOverScrollY Number of pixels to overscroll by in either direction
     *          along the Y axis.
     * @param isTouchEvent true if this scroll operation is the result of a touch event.
     * @return true if scrolling was clamped to an over-scroll boundary along either
     *          axis, false otherwise.
     */

deltaX表示X轴上过度滑动多出的那部分距离,deltaY表示Y轴上过度滑动多出的那部分距离。那么我们可以得出一个结论:图片的高度=图片原有的高度+|deltaY|,实际上无论是deltaY还是deltaX都是有正负之分的,当ListView处于顶部且的下拉的时候deltaY<0
当ListView处于底部的时候上拉deltaY>0。
3、实现上拉图片缩小恢复
这一步的核心方法是onScrollChanged,不断的监听图片的高度,并且重新绘制界面
图片的高度=图片原有高度-超出屏幕部分的高度
4、实现松手会弹动画
(1)自定义一个动画
(2)重写onTouchEvent方法UP事件的时候启动动画

二、具体实现
1、在重写ListView之前,我们先把准备工作做一下:
MaiinActivity

package lc.com.ui_zoomlistview;

import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.ImageView;

import java.util.ArrayList;
import java.util.List;

public class MainActivity extends AppCompatActivity {

    private ZoomListView zoomListView;
    private ImageView imageView;
    private List<String> list;
    private ItemAdapter adapter;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        initViews();
        initData();
    }

    private void initData() {
        // TODO Auto-generated method stub

        for (int i = 0; i < 20; i++) {
            list.add("测试数据" + i);
        }
        adapter.notifyDataSetChanged();
    }

    private void initViews() {
        // TODO Auto-generated method stub
        list = new ArrayList<String>();
        adapter = new ItemAdapter(list, this);
        zoomListView = (ZoomListView) findViewById(R.id.zoomListView);

        // 加载布局文件
        View header = View.inflate(MainActivity.this, R.layout.headview,
                null);

        imageView = (ImageView) header.findViewById(R.id.layout_header_image);
        // 将ImageView传递给ListView
        zoomListView.setImageView(imageView);
        // 给ListView添加头文件
        zoomListView.addHeaderView(header);
        // 设置适配器
        zoomListView.setAdapter(adapter);
    }
}

activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/activity_main"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context="lc.com.ui_zoomlistview.MainActivity">

    <lc.com.ui_zoomlistview.ZoomListView
        android:id="@+id/zoomListView"
        android:layout_width="match_parent"
        android:layout_height="match_parent" />
</RelativeLayout>

ItemAdapter

package lc.com.ui_zoomlistview;

import java.util.List;

import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.TextView;

public class ItemAdapter extends BaseAdapter {

    private List<String> list;

    private Context mContext;

    private LayoutInflater mInflater;

    public ItemAdapter(List<String> list, Context mContext) {
        super();
        this.list = list;
        this.mContext = mContext;
        mInflater = LayoutInflater.from(mContext);
    }

    @Override
    public int getCount() {
        // TODO Auto-generated method stub
        return list.size();
    }

    @Override
    public Object getItem(int position) {
        // TODO Auto-generated method stub
        return list.get(position);
    }

    @Override
    public long getItemId(int position) {
        // TODO Auto-generated method stub
        return position;
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        // TODO Auto-generated method stub
        ViewHolder mViewHoder = null;
        if (convertView == null) {
            mViewHoder = new ViewHolder();
            convertView = mInflater.inflate(R.layout.item, null);
            mViewHoder.item = (TextView) convertView.findViewById(R.id.item);
            convertView.setTag(mViewHoder);

        } else {
            mViewHoder = (ViewHolder) convertView.getTag();
        }
        mViewHoder.item.setText(getItem(position).toString());
        return convertView;
    }

    class ViewHolder {
        TextView item;
    }
}

item.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical" >

    <TextView
        android:id="@+id/item"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_margin="15dp" />

</LinearLayout>

很简单不做停留。下面我们开始真正的重头戏,ZoomListView的实现:

2、ZoomListView

package lc.com.ui_zoomlistview;

import android.content.Context;
import android.util.AttributeSet;
import android.util.Log;
import android.view.MotionEvent;
import android.view.View;
import android.view.animation.Animation;
import android.view.animation.Transformation;
import android.widget.ImageView;
import android.widget.ListView;

/**
 * Created by HP on 2017/4/13.
 */

public class ZoomListView extends ListView {
    private static final String TAG = "ZoomListView";
    //ImageView的初始的高度
    private int mImageViewHeight;
    private ImageView mImageView;

    public ZoomListView(Context context) {
        this(context, null);
    }

    public ZoomListView(Context context, AttributeSet attrs) {
        this(context, attrs, 0);
    }

    public ZoomListView(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        init();
    }

    /**
     * 初始化一些必要的条件
     */
    private void init() {
        //获取mIageView的初始高度
        mImageViewHeight = getContext().getResources().getDimensionPixelSize(R.dimen.size_default_height);
        Log.e(TAG, "mImageViewHeight:" + mImageViewHeight);
    }

    /**
     * 设置ImageView
     *
     * @param imageView
     */
    public void setImageView(ImageView imageView) {
        this.mImageView = imageView;
    }

    /**
     * 步骤2:实现下拉图片大
     *
     * @param deltaY Y轴上超出的距离
     * @param deltaX X轴上超出的距离
     */
    @Override
    protected boolean overScrollBy(int deltaX, int deltaY, int scrollX, int scrollY, int scrollRangeX, int scrollRangeY, int maxOverScrollX, int maxOverScrollY, boolean isTouchEvent) {
        //当deltaY的时候表示向下过度滑动
        if (deltaY < 0) {
            Log.i(TAG, "deltaY:" + deltaY);
            int imageViewHeight = mImageView.getHeight() - deltaY;
            mImageView.getLayoutParams().height = imageViewHeight;
            //重新绘制ImageView
            mImageView.requestLayout();
        }
        //当deltaY>0的时候表示上拉过度
        else
        {
            if (mImageView.getHeight() > mImageViewHeight) {
                mImageView.getLayoutParams().height = mImageView.getHeight()
                        - deltaY;
                // 重新绘制,或者叫重新摆放
                mImageView.requestLayout();

            }
        }
        return super.overScrollBy(deltaX, deltaY, scrollX, scrollY, scrollRangeX, scrollRangeY, maxOverScrollX, maxOverScrollY, isTouchEvent);
    }

    /**
     * 步骤3:实现图片
     * *@param l
     * @param t
     * @param oldl
     * @param oldt
     */
    @Override
    protected void onScrollChanged(int l, int t, int oldl, int oldt) {
        //得到mImageView的父布局,这里明显是headview
        View parent = (View) mImageView.getParent();
        //实际内容的上边界到父布局上边界的距离
        int top = parent.getTop();
        if (top < 0 && mImageView.getHeight() > mImageViewHeight) {

            //计算图片的高度
            int imageViewHeight = mImageView.getHeight() + top;
            mImageView.getLayoutParams().height = imageViewHeight;
            //重新布局
            parent.layout(parent.getLeft(), 0, parent.getRight(), parent.getHeight());
            mImageView.requestLayout();
        }
        super.onScrollChanged(l, t, oldl, oldt);
    }

    /**
     * 步骤3:实现松手回弹动画
     *
     * @param ev
     * @return
     */
    @Override
    public boolean onTouchEvent(MotionEvent ev) {

        //这里我们只需要实现UP事件
        if (ev.getAction() == MotionEvent.ACTION_UP) {
            Log.w(TAG, "UP");
            //情动回弹动画
            ResetAnimation animation = new ResetAnimation(mImageView);
            animation.setDuration(300);
            mImageView.startAnimation(animation);
        }
        return super.onTouchEvent(ev);
    }

    /**
     * 自定义一个会弹动画
     */
    class ResetAnimation extends Animation {


        private ImageView imageView;
        //高度差
        private int dHeight;

        //图片拉升以后的高度
        private int orginHeight;



        public ResetAnimation(ImageView imageView) {
            this.imageView = imageView;
            orginHeight=this.imageView.getHeight();
            dHeight = orginHeight - mImageViewHeight;
            Log.w(TAG, "dHeight:" + dHeight);
        }

        /**
         * 这个方法是为一个自定义动画需要重写的方法
         *
         * @param interpolatedTime 动画时间0~1.0
         * @param t
         */
        @Override
        protected void applyTransformation(float interpolatedTime, Transformation t) {
            Log.w(TAG, "applyTransformation");
            Log.w(TAG, "interpolatedTime:" + interpolatedTime);
            //设置图片的大小

            //计算图片的高度
            int imageViewHeight = (int) (orginHeight - dHeight * interpolatedTime);
            mImageView.getLayoutParams().height = imageViewHeight;
            mImageView.requestLayout();
            super.applyTransformation(interpolatedTime, t);
        }
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,179评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,229评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,032评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,533评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,531评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,539评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,916评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,574评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,813评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,568评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,654评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,354评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,937评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,918评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,152评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,852评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,378评论 2 342

推荐阅读更多精彩内容