Android仿简书动态searchview搜索栏效果
发布时间 - 2026-01-11 01:59:33 点击率:次简书的动态搜索栏效果是这样的,挺高大上的感觉。

ezgif.com-resize.gif
仔细想一下,其实实现起来非常简单,这是我做的效果,基本完美还原。
ezgif.com-resize (2).gif
实现这个效果, 只要关注几个点
1.搜索栏伸展和收缩动画效果实现
2.搜索栏伸展和收缩的时机
3.顶部透明度的渐变
搜索栏伸展和收缩动画效果实现:
我们只要明确,使用系统为我们提供的Transition框架,就可以轻而易举的实现了。
首先要引入依赖compile 'com.android.support:design:25.3.1',要知道我们使用到的这部分Transition效果只是封装了属性动画的内容,是可以兼容到5.0之前的。
private void expand() {
//设置伸展状态时的布局
tvSearch.setText("搜索简书的内容和朋友");
RelativeLayout.LayoutParams LayoutParams = (RelativeLayout.LayoutParams) mSearchLayout.getLayoutParams();
LayoutParams.width = LayoutParams.MATCH_PARENT;
LayoutParams.setMargins(dip2px(10), dip2px(10), dip2px(10), dip2px(10));
mSearchLayout.setLayoutParams(LayoutParams);
//设置动画
beginDelayedTransition(mSearchLayout);
}
private void reduce() {
//设置收缩状态时的布局
tvSearch.setText("搜索");
RelativeLayout.LayoutParams LayoutParams = (RelativeLayout.LayoutParams) mSearchLayout.getLayoutParams();
LayoutParams.width = dip2px(80);
LayoutParams.setMargins(dip2px(10), dip2px(10), dip2px(10), dip2px(10));
mSearchLayout.setLayoutParams(LayoutParams);
//设置动画
beginDelayedTransition(mSearchLayout);
}
void beginDelayedTransition(ViewGroup view) {
mSet = new AutoTransition();
//设置动画持续时间
mSet.setDuration(300);
// 开始表演
TransitionManager.beginDelayedTransition(view, mSet);
}
其中mSearchLayout就是搜索框的布局,只需要动态设置一下伸展和收缩的布局大小和其中显示的文字,剩下的就交给Transition吧~ 这样搜索框就可以来回摇摆了。。
搜索栏伸展和收缩的时机:
观察一下效果,伸展的时机是当顶部完全盖住banner的时候开始的,收缩的时机是滚动到顶部的时候触发。需要我们监听scllerview的滚动状态。这里的顶部我是用了自定义布局的toolbar,然后用一个imageview代替了banner。
//scrollview滚动状态监听
mScrollView.getViewTreeObserver().addOnScrollChangedListener(new ViewTreeObserver.OnScrollChangedListener() {
@Override
public void onScrollChanged() {
//改变toolbar的透明度
changeToolbarAlpha();
//滚动距离>=大图高度-toolbar高度 即toolbar完全盖住大图的时候 且不是伸展状态 进行伸展操作
if (mScrollView.getScrollY() >=ivImg.getHeight() - toolbar.getHeight() && !isExpand) {
expand();
isExpand = true;
}
//滚动距离<=0时 即滚动到顶部时 且当前伸展状态 进行收缩操作
else if (mScrollView.getScrollY()<=0&& isExpand) {
reduce();
isExpand = false;
}
}
});
}
当然简书的整个布局是基于recyclerview的,这里我为了方便使用了scrollerview。recyclerview也只需监听相应的滚动状态即可。
顶部透明度的渐变
直接上代码
private void changeToolbarAlpha() {
int scrollY = mScrollView.getScrollY();
//快速下拉会引起瞬间scrollY<0
if(scrollY<0){
toolbar.getBackground().mutate().setAlpha(0);
return;
}
//计算当前透明度比率
float radio= Math.min(1,scrollY/(ivImg.getHeight()-toolbar.getHeight()*1f));
//设置透明度
toolbar.getBackground().mutate().setAlpha( (int)(radio * 0xFF));
}
注意刚才监听滚动事件的时候调用changeToolbarAlpha()方法,并且需要初始设置为全透明
toolbar.getBackground().mutate().setAlpha(0);
好了关键代码就这么多点了~
下面附上完整代码
布局文件
<?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" android:background="#c2c0c0" > <ScrollView android:id="@+id/scrollView" android:layout_width="match_parent" android:layout_height="wrap_content" android:scrollbars="none"> <FrameLayout android:layout_width="match_parent" android:layout_height="wrap_content"> <FrameLayout android:layout_width="match_parent" android:layout_height="1500dp"> <ImageView android:id="@+id/iv_img" android:layout_width="match_parent" android:layout_height="180dp" android:scaleType="centerCrop" android:src="@drawable/night1" /> </FrameLayout> </FrameLayout> </ScrollView> <android.support.v7.widget.Toolbar android:layout_width="match_parent" android:layout_height="wrap_content" android:id="@+id/toolbar" android:background="@android:color/white" android:fitsSystemWindows="true"> <RelativeLayout android:layout_width="match_parent" android:layout_height="wrap_content" > <LinearLayout android:id="@+id/ll_search" android:layout_width="80dp" android:layout_height="30dp" android:layout_alignParentRight="true" android:layout_marginRight="10dp" android:layout_marginTop="10dp" android:layout_marginBottom="10dp" android:background="@drawable/shape_bg" android:gravity="center"> <TextView android:id="@+id/tv_search" android:layout_width="wrap_content" android:layout_height="wrap_content" android:drawableLeft="@drawable/search" android:gravity="center_vertical" android:text="搜索" android:textColor="#8A8A8A" /> </LinearLayout> </RelativeLayout> </android.support.v7.widget.Toolbar> </RelativeLayout>
就一个activity
public class MainActivity extends AppCompatActivity {
@Bind(R.id.tv_search)
TextView tvSearch;
@Bind(R.id.ll_search)
LinearLayout mSearchLayout;
@Bind(R.id.scrollView)
ScrollView mScrollView;
boolean isExpand = false;
@Bind(R.id.iv_img)
ImageView ivImg;
@Bind(R.id.toolbar)
Toolbar toolbar;
private TransitionSet mSet;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
ButterKnife.bind(this);
//设置全屏透明状态栏
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
getWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
ViewGroup rootView = (ViewGroup) ((ViewGroup)findViewById(android.R.id.content)).getChildAt(0);
ViewCompat.setFitsSystemWindows(rootView,false);
rootView.setClipToPadding(true);
}
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
getWindow().clearFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS|
WindowManager.LayoutParams.FLAG_TRANSLUCENT_NAVIGATION);
getWindow().getDecorView().setSystemUiVisibility(View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN| View.SYSTEM_UI_FLAG_LAYOUT_STABLE);
getWindow().addFlags(WindowManager.LayoutParams.FLAG_DRAWS_SYSTEM_BAR_BACKGROUNDS);
getWindow().setStatusBarColor(Color.TRANSPARENT);
}
//设置toolbar初始透明度为0
toolbar.getBackground().mutate().setAlpha(0);
//scrollview滚动状态监听
mScrollView.getViewTreeObserver().addOnScrollChangedListener(new ViewTreeObserver.OnScrollChangedListener() {
@Override
public void onScrollChanged() {
//改变toolbar的透明度
changeToolbarAlpha();
//滚动距离>=大图高度-toolbar高度 即toolbar完全盖住大图的时候 且不是伸展状态 进行伸展操作
if (mScrollView.getScrollY() >=ivImg.getHeight() - toolbar.getHeight() && !isExpand) {
expand();
isExpand = true;
}
//滚动距离<=0时 即滚动到顶部时 且当前伸展状态 进行收缩操作
else if (mScrollView.getScrollY()<=0&& isExpand) {
reduce();
isExpand = false;
}
}
});
}
private void changeToolbarAlpha() {
int scrollY = mScrollView.getScrollY();
//快速下拉会引起瞬间scrollY<0
if(scrollY<0){
toolbar.getBackground().mutate().setAlpha(0);
return;
}
//计算当前透明度比率
float radio= Math.min(1,scrollY/(ivImg.getHeight()-toolbar.getHeight()*1f));
//设置透明度
toolbar.getBackground().mutate().setAlpha( (int)(radio * 0xFF));
}
private void expand() {
//设置伸展状态时的布局
tvSearch.setText("搜索简书的内容和朋友");
RelativeLayout.LayoutParams LayoutParams = (RelativeLayout.LayoutParams) mSearchLayout.getLayoutParams();
LayoutParams.width = LayoutParams.MATCH_PARENT;
LayoutParams.setMargins(dip2px(10), dip2px(10), dip2px(10), dip2px(10));
mSearchLayout.setLayoutParams(LayoutParams);
//开始动画
beginDelayedTransition(mSearchLayout);
}
private void reduce() {
//设置收缩状态时的布局
tvSearch.setText("搜索");
RelativeLayout.LayoutParams LayoutParams = (RelativeLayout.LayoutParams) mSearchLayout.getLayoutParams();
LayoutParams.width = dip2px(80);
LayoutParams.setMargins(dip2px(10), dip2px(10), dip2px(10), dip2px(10));
mSearchLayout.setLayoutParams(LayoutParams);
//开始动画
beginDelayedTransition(mSearchLayout);
}
void beginDelayedTransition(ViewGroup view) {
mSet = new AutoTransition();
mSet.setDuration(300);
TransitionManager.beginDelayedTransition(view, mSet);
}
private int dip2px(float dpVale) {
final float scale = getResources().getDisplayMetrics().density;
return (int) (dpVale * scale + 0.5f);
}
}
更完整的在这里https://github.com/yanyiqun001/dymicSearchview 希望大家多多支持
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
# Android
# 搜索栏
# searchview
# Android Studio搜索功能(查找功能)及快捷键图文详解
# Android实现搜索功能并本地保存搜索历史记录
# Android自定义View实现搜索框(SearchView)功能
# Android百度地图实现搜索和定位及自定义图标绘制并点击时弹出泡泡
# Android搜索框SearchView属性和用法详解
# Android SearchView搜索框组件的使用方法
# Android遍历所有文件夹和子目录搜索文件
# Android搜索框组件SearchView的基本使用方法
# Android ListView用EditText实现搜索功能效果
# Android实现简单动态搜索功能
# 大家多多
# 这是
# 我是
# 瞬间
# 几个
# 在这里
# 好了
# 是这样
# 这么多
# 只需
# 这部
# 点了
# 用了
# 要知道
# 自定义
# 我做
# 只需要
# 设置为
# 轻而易举
# 全屏
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
linux写shell需要注意的问题(必看)
如何快速搭建高效简练网站?
Android自定义listview布局实现上拉加载下拉刷新功能
javascript和jQuery中的AJAX技术详解【包含AJAX各种跨域技术】
ChatGPT 4.0官网入口地址 ChatGPT在线体验官网
laravel怎么为API路由添加签名中间件保护_laravel API路由签名中间件保护方法
Win11怎么恢复误删照片_Win11数据恢复工具使用【推荐】
教学论文网站制作软件有哪些,写论文用什么软件
?
儿童网站界面设计图片,中国少年儿童教育网站-怎么去注册?
香港服务器网站生成指南:免费资源整合与高速稳定配置方案
Laravel怎么创建自己的包(Package)_Laravel扩展包开发入门到发布
免费视频制作网站,更新又快又好的免费电影网站?
高端智能建站公司优选:品牌定制与SEO优化一站式服务
如何批量查询域名的建站时间记录?
Laravel Eloquent关联是什么_Laravel模型一对一与一对多关系精讲
如何用狗爹虚拟主机快速搭建网站?
Linux系统运维自动化项目教程_Ansible批量管理实战
EditPlus中的正则表达式实战(6)
进行网站优化必须要坚持的四大原则
宙斯浏览器怎么屏蔽图片浏览 节省手机流量使用设置方法
Laravel如何使用查询构建器?(Query Builder高级用法)
Laravel怎么进行浏览器测试_Laravel Dusk自动化浏览器测试入门
Laravel观察者模式如何使用_Laravel Model Observer配置
宙斯浏览器文件分类查看教程 快速筛选视频文档与图片方法
网站视频制作书签怎么做,ie浏览器怎么将网站固定在书签工具栏?
香港服务器租用费用高吗?如何避免常见误区?
Gemini手机端怎么发图片_Gemini手机端发图方法【步骤】
Laravel如何实现本地化和多语言支持?(i18n教程)
JavaScript如何实现路由_前端路由原理是什么
WEB开发之注册页面验证码倒计时代码的实现
高性价比服务器租赁——企业级配置与24小时运维服务
微信h5制作网站有哪些,免费微信H5页面制作工具?
网站建设要注意的标准 促进网站用户好感度!
佛山企业网站制作公司有哪些,沟通100网上服务官网?
北京企业网站设计制作公司,北京铁路集团官方网站?
Laravel如何集成第三方登录_Laravel Socialite实现微信QQ微博登录
实例解析Array和String方法
海南网站制作公司有哪些,海口网是哪家的?
Laravel如何实现API资源集合?(Resource Collection教程)
JavaScript模板引擎Template.js使用详解
湖南网站制作公司,湖南上善若水科技有限公司做什么的?
Laravel如何实现用户密码重置功能?(完整流程代码)
谷歌Google入口永久地址_Google搜索引擎官网首页永久入口
Python进程池调度策略_任务分发说明【指导】
详解免费开源的DotNet二维码操作组件ThoughtWorks.QRCode(.NET组件介绍之四)
香港服务器WordPress建站指南:SEO优化与高效部署策略
如何在宝塔面板创建新站点?
Android Socket接口实现即时通讯实例代码
Windows10如何删除恢复分区_Win10 Diskpart命令强制删除分区
大连企业网站制作公司,大连2025企业社保缴费网上缴费流程?

