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

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

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

  1. 爬取网易云音乐评论!python 爬虫入门实战(六)selenium 入门!

    说到爬虫,第一时间可能就会想到网易云音乐的评论.网易云音乐评论里藏了许多宝藏,那么让我们一起学习如何用 python 挖宝藏吧! 既然是宝藏,肯定是用要用钥匙加密的.打开 Chrome 分析 Head ...

  2. Android项目实战之高仿网易云音乐项目介绍

    这一节我们来讲解这个项目所用到的一些技术,以及一些实现的效果图,让大家对该项目有一个整体的认识,推荐大家收藏该文章,因为我们发布文章后会在该文章里面加入链接,这样大家找着就很方便. 目录 第1章 前期 ...

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

    imageMogr2/auto-orient/strip%7CimageView2/2/w/1240" alt="唱片机" title=""> ...

  4. NetCloud——一个网易云音乐评论抓取和分析的Python库

    在17的四月份,我曾经写了一篇关于网易云音乐爬虫的文章,还写了一篇关于评论数据可视化的文章.在这大半年的时间里,有时会有一些朋友给我发私信询问一些关于代码方面的问题.所以我最近抽空干脆将原来的代码整理 ...

  5. python爬虫+词云图,爬取网易云音乐评论

    又到了清明时节,用python爬取了网易云音乐<清明雨上>的评论,统计词频和绘制词云图,记录过程中遇到一些问题 爬取网易云音乐的评论 一开始是按照常规思路,分析网页ajax的传参情况.看到 ...

  6. Android项目实战之高仿网易云音乐创建项目和配置

    这一节我们来讲解创建项目:说道大家可能就会说了,创建项目还有谁不会啊,还需要讲吗,别急听我慢慢到来,肯定有你不知道的. 使用项目Android Studio创建项目我们这里就不讲解了,主要是讲解如何配 ...

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

    Android精选源码 复现网易云音乐引导页效果 高仿书旗小说 Flutter版,支持iOS.Android Android Srt和Ass字幕解析器 Material Design ViewPage ...

  8. 网易云音乐 歌词制作软件 BesLyric

    导读 哈哈,喜欢网易云音乐,又愁于制作歌词的童鞋有福啦! BesLyric 为你排忧解难! 上个周末在用网易云音乐听歌,发现一些喜欢的歌还没有滚动歌词,然而网易云音乐还没有自带的歌词编辑功能,要制作歌 ...

  9. 实现一个网易云音乐的 BottomSheetDialog

    作者:林冠宏 / 指尖下的幽灵 掘金:https://juejin.im/user/587f0dfe128fe100570ce2d8 博客:http://www.cnblogs.com/linguan ...

随机推荐

  1. 冒泡,简单选择,直接插入排序(Java版)

    冒泡.简单选择,直接插入这三种排序都是简单排序. 工具类 package Utils; import java.util.Arrays; public class SortUtils { public ...

  2. 【转】webshell检测——使用auditd进行system调用审计

    本文档将介绍:如何通过Linux审计系统auditd监测WebShell执行系统命令的行为. 测试环境:CentOS7.0_x64 auditd简介 Linux审计系统提供了一种跟踪系统上与安全相关的 ...

  3. 安卓开发--WebView

    package com.zhangxi.test01; import android.app.Activity;import android.app.ProgressDialog;import and ...

  4. MySQL修改最大连接数,没有my.ini文件,只有my-default,这怎么改呀?

    # For advice on how to change settings please see # http://dev.mysql.com/doc/refman/5.6/en/server-co ...

  5. <Sicily>Funny Game

    一.题目描述 Two players, Singa and Suny, play, starting with two natural numbers. Singa, the first player ...

  6. Debian9.5下sftp配置和scp用法

    基于 ssh 的 sftp 服务相比 ftp 有更好的安全性(非明文帐号密码传输)和方便的权限管理(限制用户的活动目录). 1.如果只想让某些用户只能使用 sftp 操作文件, 而不能通过ssh进行服 ...

  7. caffe(1) 网络结构层参数详解

    prototxt文件是caffe的配置文件,用于保存CNN的网络结构和配置信息.prototxt文件有三种,分别是deploy.prototxt,train_val.prototxt和solver.p ...

  8. JAVA-截取字符串两边指定字符

    工具类: /** * 工具类 */ public class Tool { /** * 截取两边指定的字符 * @param character * @param symbol * @return * ...

  9. PHP取不定个数数组交集

    最近有个需求,有一个N个二维数组,N是动态的,不固定个数,现需取这N个数组的交集内容. 用到的函数是array_intersect_assoc 用法 $result_arr = array_inter ...

  10. 题解 P2195 【HXY造公园】

    天哪这道题竟然只有一篇题解! emm,首先读题看完两个操作就已经有很明确的思路了,显然是并查集+树的直径 一波解决. 并查集不多说了,如果不了解的可以看这里. 树的直径的思路很朴实,就是两边DFS(B ...