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="下面内容自己脑补\n下面内容自己脑补\n下面内容自己脑补\n下面内容自己脑补\n下面内容自己脑补\n下面内容自己脑补\n
                              下面内容自己脑补\n下面内容自己脑补\n下面内容自己脑补\n下面内容自己脑补\n下面内容自己脑补\n下面内容自己脑补\n
                              下面内容自己脑补\n下面内容自己脑补\n下面内容自己脑补\n下面内容自己脑补\n下面内容自己脑补\n下面内容自己脑补\n
                              下面内容自己脑补\n下面内容自己脑补\n下面内容自己脑补\n下面内容自己脑补\n下面内容自己脑补\n下面内容自己脑补\n"
                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

时间: 2024-10-18 13:15:41

Android ScrollView滚动实现大众点评、网易云音乐评论悬停效果的相关文章

Android实战之酷云--&gt;仿网易云音乐开发

我的个人网站 Xuejianxin's Blog Google Blog Xuejianxin's Blog Android自定义View学习 Android自定义View之常用工具源码分析 Android自定义View之onMeasure()源码分析 Android自定义View之onLayout()源码分析 Android自定义View之对TouchEvent的处理 Android自定义View之draw原理分析 如果觉得我的文章还行的话,也可以关注我的公众号,里面也会第一时间更新,并且会有

用RotateDrawable实现网易云音乐唱片机效果

有一段时间没有更新文章了,记得上一篇文章讲的是<用ClipDrawable实现音频录制麦克风讲话效果>,用户反响也都还不错,自己也是深受鼓舞.其实从那之后就一直想写一篇关于RotateDrawable的文章,原因很简单,RotateDrawable其实和上一篇文章中的ClipDrawable很相似.正愁着不知道以什么样的方式向大家介绍,也正是这个原因吧,一直没有发表新的文章.赶巧了,在用朋友手机的时候发现了一款名叫'网易云音乐'的APP,在主播放页面有一个唱片机的功能感觉不错诶,于是乎,把玩了

网易云音乐评论爬虫(2):歌曲的全部评论

用过网易云音乐听歌的朋友都知道,网易云音乐每首歌曲后面都有很多评论,热门歌曲的评论更是接近百万或者是超过百万条.现在我就来分享一下如何爬取网易云音乐歌曲的全部评论,由于网易云音乐的评论都做了混淆加密处理,因此我们需要深入了解它的加密过程之后才能爬取到网易云音乐歌曲的全部评论. 加qq群813622576或vx;tanzhouyiwan免费领取Python学习资料 一,首先分析数据的请求方式 网易云音乐歌曲页面的URL形式为https://music.163.com/#/song?id=歌曲id号

网易云音乐评论

<爱河> 小词u: 有个男的早年跟老婆一起创业 最穷的时候都带着老婆卖过血.后来创业成功暴富了 几乎不带犹豫地找了个年轻漂亮的小网红,跟老婆离婚了.老婆分了几千万家产,结果他老婆一离婚就自杀了.有人说这个女人傻,可是后来想想,跟他一起吃过苦的人怎么会在乎钱呢. 昨晚你真坏丶: 我交了一个三年的女网友.在他身上花了三四万.结果一见面是个男的.不过我也没吃亏.还是把他上了. 成赐: 刚开学我上课的时候戴耳机听歌,被班长发现.我舍友开玩笑说我戴的是助听器,班长当真了.之后我和班长在一起的时候就一直带

爬取网易云音乐评论并使用词云展示

最近听到一首很喜欢的歌,许薇的<我以为>,评论也很有趣,遂有想爬取该歌曲下的所有评论并用词云工具展示. 我们使用chrome开发者工具,发现歌曲的评论都隐藏在以 R_SO_4 开头的 XHR 文件中 接下来思路就很明确,拿到该文件,解析该文件的 json 数据,拿到全部评论. 我们可以看到该文件有两个用JS加密的参数 params 和 encSecKey ,关于这两个加密参数,参考了知乎用户的解答:https://www.zhihu.com/question/36081767 . 步骤: 1.

android仿网易云音乐引导页、仿书旗小说Flutter版、ViewPager切换、爆炸菜单、风扇叶片效果等源码

Android精选源码 复现网易云音乐引导页效果 高仿书旗小说 Flutter版,支持iOS.Android Android Srt和Ass字幕解析器 Material Design ViewPager切换变色美观效果 卡证识别 相机样式 Android炫酷爆炸效果的菜单源码 Android简洁优雅可点击的toast控件,仿手机百度 Android实现水平列表,其中的项目像风扇叶片一样移动效果 让你的RecyclerView秀出传送带效果,支持横向和纵向两种选... 一种流式布局的效果,很像我们

Python 获取网易云音乐热门评论

--> html { line-height: 1.6 } body { font-family: -apple-system-font, "Helvetica Neue", "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", sans-serif; background-color: #f3f3f3; line-height: inherit } body.ric

python3.基础爬取网易云音乐【超详细版】

简单学习了python爬虫之后,我们就可以嘿咻嘿咻了...因为平时就是用网易云听的歌,也喜欢看歌里的评论,所以就爬网易云音乐评论吧! 正式进入主题 首先还是去找目标网页并开始分析网页结构,如下 上面的三个箭头都是所要找的数据,分别是评论用户,评论和点赞数,都可以用正则表达式找出来,接下来继续找怎样找到下一页的数据,还是用开发者工具,但是当点击下一页的时候,网页的url没有变,说明网页是动态加载,所以就不能在当前网页找数据了,应该在他的xhr文件里找,所以点入network看看,然后也点击下一页一

Android ScrollView 滚动机制

Android ScrollView 滚动机制 我们都知道通过View#scrollTo(x,y)既可以实现将View滚动的效果,如果再添加Scroller类,就可以实现滚到效果.但是,这背后是如何实现的呢?这个问题涉及到View的绘图机制.我们先看看View的绘图的基本流程 (图片来自于网上比较常见的view绘图流程图) 关于三个阶段的简单描述: 1. measure:预估计ViewTree的各个View的占用空间. 2. layout :   确定ViewTree中各个View所处的空间位置