• Android ScrollView滚动实现大众点评、网易云音乐评论悬停效果


    今天听着网易云音乐,写着代码,真是爽翻了。


    http://blog.csdn.net/linshijun33/article/details/47910833
    网易云音乐这个产品亮点应该在评论这一模块,于是我就去看了下评论区,发现一个它是用的一个ScrollView,从上到下可分为三部分。最上面是音乐封面。接着是精彩评论,然后是最新评论。手指在上面滚动一段距离。精彩评论就在布局顶部悬停,再接着滚动,最新评论就替代精彩评论在布局顶部悬停,十分有趣的一个特点。

    这里写图片描写叙述

    然后又去翻了一下大众点评APP,发现也是运用到这个悬停效果。

    这里写图片描写叙述


    价格显示悬停在顶部了
    这一个真是非常用心的举动。价格消费者时时都能够看得到。多么良好的用户体验,能够非常好地提高产品的转化率呢。真是机智。
    确实要在用户交互上面花点功夫呢。


    这里写图片描写叙述

    于是想着怎么去实现这种效果。去找了资料来看。大概是这么做到的。

    老规矩,先上效果图。


    这里写图片描写叙述
    然后是悬停,图片是截取的。有点变形。
    这里写图片描写叙述

    先上布局文件activity_main.xml,设置了控件有四个,一个自己定义MyScrollView(布局后面给出代码),两个ImageView,一个TextView( 用来充字数占空间让屏幕滚动起来的)。

    activity_main.xml

    <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:tools="http://schemas.android.com/tools"
        android:id="@+id/box"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        tools:context=".MainActivity">
    
        <com.topfloat.MyScrollView
            android:id="@+id/myScrollView"
            android:layout_width="fill_parent"
            android:layout_height="match_parent">
    
            <LinearLayout
                android:layout_width="match_parent"
                android:layout_height="wrap_content"
                android:orientation="vertical">
    
                <RelativeLayout
                    android:id="@+id/rlayout"
                    android:layout_width="fill_parent"
                    android:layout_height="wrap_content"
                    >
    
                    <ImageView
                        android:id="@+id/top"
                        android:layout_width="fill_parent"
                        android:layout_height="wrap_content"
                        android:background="@mipmap/top_show"
                        android:layout_alignParentBottom="true" />
                </RelativeLayout>
    
                <LinearLayout
                    android:id="@+id/bind_2"
                    android:layout_width="match_parent"
                    android:layout_height="wrap_content"
                    android:orientation="vertical">
    
                    <ImageView
                        android:id="@+id/buy_price"
                        android:layout_width="fill_parent"
                        android:layout_height="wrap_content"
                        android:background="@mipmap/buy_price"
                        />
                </LinearLayout>
    
                <TextView
                    android:layout_width="match_parent"
                    android:layout_height="wrap_content"
                    android:gravity="center_horizontal"
                    android:text="下面内容自己脑补
    下面内容自己脑补
    下面内容自己脑补
    下面内容自己脑补
    下面内容自己脑补
    下面内容自己脑补
    
                                  下面内容自己脑补
    下面内容自己脑补
    下面内容自己脑补
    下面内容自己脑补
    下面内容自己脑补
    下面内容自己脑补
    
                                  下面内容自己脑补
    下面内容自己脑补
    下面内容自己脑补
    下面内容自己脑补
    下面内容自己脑补
    下面内容自己脑补
    
                                  下面内容自己脑补
    下面内容自己脑补
    下面内容自己脑补
    下面内容自己脑补
    下面内容自己脑补
    下面内容自己脑补
    "
                    android:textSize="30sp"/>
    
            </LinearLayout>
        </com.topfloat.MyScrollView>
    
        <LinearLayout
            android:id="@+id/bing_1"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:orientation="vertical">
    
        </LinearLayout>
    
    </RelativeLayout>
    

    然后给出的是自己定义控件MyScrollView

    MyScrollView

    package com.topfloat;
    
    /**
     * Created by Linshijun on 2015/8/23.
     */
    
    
    import android.content.Context;
    import android.os.Handler;
    import android.util.AttributeSet;
    import android.view.MotionEvent;
    import android.widget.ScrollView;
    
    /*
     * ScrollView并没有实现滚动监听。所以我们必须自行实现对ScrollView的监听。
     * 我们非常自然的想到在onTouchEvent()方法中实现对滚动Y轴进行监听
     * ScrollView的滚动Y值进行监听
     */
    public class MyScrollView extends ScrollView {
        private OnScrollListener onScrollListener;
        /**
         * 主要是用在用户手指离开MyScrollView。MyScrollView还在继续滑动,我们用来保存Y的距离,然后做比較
         */
        private int lastScrollY;
    
        public MyScrollView(Context context) {
            super(context, null);
        }
    
        public MyScrollView(Context context, AttributeSet attrs) {
            super(context, attrs, 0);
        }
    
        public MyScrollView(Context context, AttributeSet attrs, int defStyle) {
            super(context, attrs, defStyle);
        }
    
        /**
         * 设置滚动接口
         *
         * @param onScrollListener
         */
        public void setOnScrollListener(OnScrollListener onScrollListener) {
            this.onScrollListener = onScrollListener;
        }
    
        /**
         * 用于用户手指离开MyScrollView的时候获取MyScrollView滚动的Y距离。然后回调给onScroll方法中
         */
        private Handler handler = new Handler() {
    
            public void handleMessage(android.os.Message msg) {
                int scrollY = MyScrollView.this.getScrollY();
    
                //此时的距离和记录下的距离不相等,在隔5毫秒给handler发送消息
                if (lastScrollY != scrollY) {
                    lastScrollY = scrollY;
                    handler.sendMessageDelayed(handler.obtainMessage(), 5);
                }
                if (onScrollListener != null) {
                    onScrollListener.onScroll(scrollY);
                }
    
            }
    
    
        };
    
        /**
         * 重写onTouchEvent。 当用户的手在MyScrollView上面的时候,
         * 直接将MyScrollView滑动的Y方向距离回调给onScroll方法中,当用户抬起手的时候。
         * MyScrollView可能还在滑动。所以当用户抬起手我们隔5毫秒给handler发送消息,在handler处理
         * MyScrollView滑动的距离
         */
        @Override
        public boolean onTouchEvent(MotionEvent ev) {
            if (onScrollListener != null) {
                onScrollListener.onScroll(lastScrollY = this.getScrollY());
            }
            switch (ev.getAction()) {
                case MotionEvent.ACTION_UP:
                    handler.sendMessageDelayed(handler.obtainMessage(), 5);
                    break;
            }
            return super.onTouchEvent(ev);
        }
    
        /**
         * 滚动的回调接口
         */
        public interface OnScrollListener {
            /**
             * 回调方法。 返回MyScrollView滑动的Y方向距离
             */
            public void onScroll(int scrollY);
        }
    }
    
    

    ScrollView并没有实现滚动监听,所以必须自行实现对ScrollView的监听,由于滚动方向在Y轴,能够使用onTouchEvent()方法实现对滚动Y轴进行监听,而手指离开屏幕时,滚动还在继续,这时就须要选择在用户手指离开的时候每隔5毫秒来推断ScrollView是否停止滑动。并将ScrollView的滚动Y值回调给OnScrollListener接口的onScroll(int scrollY)方法中。

    然后对ScrollView调用setOnScrollListener方法就能监听到滚动的Y值。

    实现了上面两步。MainActivity就比較简单了。主要用了控件的从属关系来addview和removeview,这样做使代码更为简洁,更有适用性。

    MainActivity

    package com.topfloat;
    
    import android.support.v7.app.AppCompatActivity;
    import android.os.Bundle;
    import android.view.Menu;
    import android.view.MenuItem;
    import android.widget.ImageView;
    import android.widget.LinearLayout;
    import android.widget.RelativeLayout;
    
    /**
     * 实现OnScrollListener接口
     */
    public class MainActivity extends AppCompatActivity implements MyScrollView.OnScrollListener{
    
        private ImageView topshow,priceshow;
        private MyScrollView myScrollView;
        private int buypriceTop;
    
        LinearLayout bind_1,bind_2;
    
    
    
        RelativeLayout rlayout;
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_main);
            //初始化控件
            initView();
        }
    
        private void initView() {
            topshow= (ImageView) findViewById(R.id.top);
            priceshow= (ImageView) findViewById(R.id.buy_price);
            myScrollView= (MyScrollView) findViewById(R.id.myScrollView);
            rlayout= (RelativeLayout) findViewById(R.id.rlayout);
            bind_1= (LinearLayout) findViewById(R.id.bing_1);
            bind_2= (LinearLayout) findViewById(R.id.bind_2);
            myScrollView.setOnScrollListener(this);
        }
    
        /**
         * 获取pirce_show的顶部位置,即rlayout的底部位置
         * @param hasFocus
         */
        @Override
        public void onWindowFocusChanged(boolean hasFocus) {
            super.onWindowFocusChanged(hasFocus);
            if(hasFocus){
                buypriceTop=rlayout.getBottom(); //获取pirce_show的顶部位置。即rlayout的底部位置
            }
        }
    
        @Override
        public boolean onCreateOptionsMenu(Menu menu) {
            // Inflate the menu; this adds items to the action bar if it is present.
            getMenuInflater().inflate(R.menu.menu_main, menu);
            return true;
        }
    
        @Override
        public boolean onOptionsItemSelected(MenuItem item) {
            // Handle action bar item clicks here. The action bar will
            // automatically handle clicks on the Home/Up button, so long
            // as you specify a parent activity in AndroidManifest.xml.
            int id = item.getItemId();
    
            //noinspection SimplifiableIfStatement
            if (id == R.id.action_settings) {
                return true;
            }
    
            return super.onOptionsItemSelected(item);
        }
    
        /**
         * 监听滚动Y值变化。通过addView和removeView来实现悬停效果
         * @param scrollY
         */
        @Override
        public void onScroll(int scrollY) {
    
            //推断滚动距离是否在于图片高度
            if(scrollY>=buypriceTop){
                if(priceshow.getParent()!=bind_1)  //推断其现有父类
                {
                    bind_2.removeView(priceshow);  //从现有父类移除
                    bind_1.addView(priceshow);  //加入到目标父类
                }
            }else {
                if(priceshow.getParent()!=bind_2)
                {
                 bind_1.removeView(priceshow);
                    bind_2.addView(priceshow);
                }
            }
        }
    }
    

    凝视里基本都将须要重点都说了,比較好理解。

    Android Studio源代码稍后上传,有须要的能够下载来看看。


    http://download.csdn.net/detail/linshijun33/9039237

  • 相关阅读:
    module.exports = $; $ is not defined
    npm run build 时 报 __webpack_public_path__ = window.webpackPublicPath; 中的windows未定义
    TypeError: Cannot assign to read only property 'exports' of object '#<Object>'
    windows package.json设置多个环境变量
    点击劫持ClickJacking
    判断数组的方法/判断JS数据类型的四种方法
    时间操作
    Ado调用存储过程
    layui表格增删改查与上传图片+Api
    layui表单与上传图片
  • 原文地址:https://www.cnblogs.com/brucemengbm/p/7026323.html
Copyright © 2020-2023  润新知