前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Android使用HorizontalScrollView实现水平滚动

Android使用HorizontalScrollView实现水平滚动

作者头像
砸漏
发布2020-11-04 15:38:36
3.2K0
发布2020-11-04 15:38:36
举报
文章被收录于专栏:恩蓝脚本

HorizontalScrollView 和 ScrollView 都是由 FrameLayout 派生出来的。它们就是一个用于为普通组件添加滚动条的组件。且 HorizontalScrollView 和 ScrollView 里面最多只能包含一个组件(当然组件里面还可以嵌套组件)。它们不同的是 HorizontalScrollView 用于添加水平滚动,而 ScrollView 用于添加垂直滚动。

突然间想到 做一个屏幕下方水平滑动,屏幕上方并作出相应的反应的效果。只是在下方滚动时,屏幕上方没有作出理想的反应,点击事件倒是实现了。最终只能在网上搜索,终于找到了一个。于是作出的效果如下:

只是这个效果还有所缺陷,加载了 13 张图片,在屏幕下方水平滚动到最后一页时,第 9 张的图片并没有在上面的显示出来(原作者的也有这个问题);如果图片的数量小于或者等于 4 张时则不能运行。

本例的难点主要在于 MyHorizontalView 类中,并且还有收集而来的注解。

MainActivity.java :

代码语言:javascript
复制
package com.crazy.horizontalscrollviewtest;
import java.io.InputStream;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import com.crazy.horizontalscrollviewtest.MyHorizontalView.CurrentImageChangeListener;
import com.crazy.horizontalscrollviewtest.MyHorizontalView.OnItemClickListener;
import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.Color;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.ImageView;
import android.support.v7.app.AppCompatActivity;
public class MainActivity extends AppCompatActivity {
private ImageView mImageView;
private MyHorizontalView myHorizontalView;
private List<Bitmap  bitmapList;
private MyAdapter adapter;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
init();
}
private void init() {
mImageView = (ImageView)findViewById(R.id.imageView);
bitmapList = new ArrayList< (Arrays.asList(
readBitMap(this, R.drawable.bricks),
readBitMap(this, R.drawable.dog),
readBitMap(this, R.drawable.flower),
readBitMap(this, R.drawable.grass),
readBitMap(this, R.drawable.stones),
readBitMap(this, R.drawable.wood),
readBitMap(this, R.drawable.bg_01),
readBitMap(this, R.drawable.bg_02),
readBitMap(this, R.drawable.bg_03),
readBitMap(this, R.drawable.bg_04),
readBitMap(this, R.drawable.bg_05),
readBitMap(this, R.drawable.bg_06),
readBitMap(this, R.drawable.bg_07)
));
myHorizontalView = (MyHorizontalView)findViewById(R.id.my_horizontal);
adapter = new MyAdapter(this, bitmapList);
//设置适配器
myHorizontalView.initDatas(adapter);
//添加滚动回调
myHorizontalView
.setCurrentImageChangeListener(new CurrentImageChangeListener() {
@Override
public void onCurrentImgChanged(int position, View viewIndicator) {
Log.e("==============","===============  " + position);
mImageView.setImageBitmap(bitmapList.get(position));
viewIndicator.setBackgroundColor(Color.parseColor("#AA024DA4"));
}
});
//添加点击回调
myHorizontalView.setOnItemClickListener(new OnItemClickListener() {
@Override
public void onItemClick(View view, int position) {
mImageView.setImageBitmap(bitmapList.get(position));
view.setBackgroundColor(Color.parseColor("#AA024DA4"));
}
});
}
public static Bitmap readBitMap(Context mContext, int resId) {
BitmapFactory.Options opt = new BitmapFactory.Options();
opt.inPreferredConfig = Bitmap.Config.RGB_565;
opt.inPurgeable = true;
opt.inInputShareable = true;
InputStream is = mContext.getResources().openRawResource(resId);
return BitmapFactory.decodeStream(is, null, opt);
}
}

MyAdapter 这部分并不是为 AbsListView 和 AbsSpinner 及其子类提供列表项的。它主要用于为 HorizontalScrollView 提供数据。

MyAdapter.java :

代码语言:javascript
复制
package com.crazy.horizontalscrollviewtest;
import java.util.ArrayList;
import java.util.List;
import android.content.Context;
import android.graphics.Bitmap;
import android.util.Log;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.ImageView;
import android.widget.RelativeLayout;
/**
* Created by antimage on 2016/1/9.
*/
public class MyAdapter extends BaseAdapter {
private List<Bitmap  bitmapList;
private Context mContext;
public MyAdapter(Context context, List<Bitmap  bitmapList) {
mContext = context;
if (bitmapList == null) {
bitmapList = new ArrayList<Bitmap ();
} else {
this.bitmapList = bitmapList;
}
}
@Override
public int getCount() {
return bitmapList.size();
}
@Override
public Object getItem(int position) {
return bitmapList.get(position);
}
@Override
public long getItemId(int position) {
return position;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder viewHolder = null;
View view = null;
// 此处要用相对布局,且与 XML 中的布局相同;
// 如果使用线性布局,则显示不完整
RelativeLayout layout;
if (convertView == null) {
layout = (RelativeLayout) View.inflate(mContext, R.layout.image_item_layout, null);
viewHolder = new ViewHolder();
viewHolder.image = (ImageView) layout.findViewById(R.id.top_image);
layout.setTag(viewHolder);
} else {
layout = (RelativeLayout) convertView;
view = layout;
viewHolder = (ViewHolder) layout.getTag();
Log.e("MyAdapter", "正在检测数据来了没有 ");
}
Bitmap btm = (Bitmap) getItem(position);
viewHolder.image.setImageBitmap(btm);
Log.e("MyAdapter", "信息来了哦!");
return layout;
}
private static class ViewHolder {
ImageView image;
}
}

MyHorizontalView 类主要用于未 MainAcitivity 类提供接口、水平滚动时屏幕上方的反应及相应的点击事件等。该类主要使用了收集而来的代码,并做了相应的调整。

MyHorizontalView.java :

代码语言:javascript
复制
package com.crazy.horizontalscrollviewtest;
import java.util.HashMap;
import java.util.Map;
import android.app.Activity;
import android.content.Context;
import android.graphics.Color;
import android.util.AttributeSet;
import android.util.DisplayMetrics;
import android.util.Log;
import android.view.MotionEvent;
import android.view.View;
import android.view.ViewGroup;
import android.widget.HorizontalScrollView;
/**
* Created by antimage on 2016/1/9.
*/
public class MyHorizontalView extends HorizontalScrollView
implements View.OnClickListener {
private String TAG = "MyHorizontalView";
private ViewGroup parent;
private int screenWidth;
/* 当前最后一张图片的index*/
private int mCurrentIndex;
/* 当前第一张图片的下标*/
private int mFristIndex;
/* 每屏幕最多显示的个数*/
private int mCountOneScreen;
/* 子元素的宽度*/
private int mChildWidth;
/* 子元素的高度*/
private int mChildHeight;
private MyAdapter mAdapter;
private CurrentImageChangeListener mListener;
private OnItemClickListener mOnItemClickListener;
/**
* 图片滚动时的回调接口
*/
public interface CurrentImageChangeListener {
void onCurrentImgChanged(int position, View viewIndicator);
}
/**
* 点击条目时的回调
*/
public interface OnItemClickListener {
void onItemClick(View view, int pos);
}
/* 保存View与位置的键值对 */
private Map<View, Integer  mViewPos = new HashMap< ();
public MyHorizontalView(Context context) {
this(context, null);
}
public MyHorizontalView(Context context, AttributeSet attrs) {
this(context, attrs, 0);
}
public MyHorizontalView(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
this.setSmoothScrollingEnabled(true);
DisplayMetrics metrics = new DisplayMetrics();
// 取得窗口属性
((Activity) context).getWindowManager().getDefaultDisplay().getMetrics(metrics);
// 窗口的宽度 (像素)
screenWidth = metrics.widthPixels;
}
/**
* 初始化数据,设置数据适配器
*/
public void initDatas(MyAdapter mAdapter) {
if (getChildCount() == 0) {
Log.e(TAG, "必须要有子元素");
}
if (getChildCount() == 0 || mAdapter == null)
return;
this.mAdapter = mAdapter;
parent = (ViewGroup) getChildAt(0);
// 获得适配器中第一个View
final View view = mAdapter.getView(0, null, parent);
parent.addView(view);
// 强制计算当前View的宽和高
if (mChildWidth == 0 && mChildHeight == 0) {
int w = View.MeasureSpec.makeMeasureSpec(0,
View.MeasureSpec.UNSPECIFIED);
int h = View.MeasureSpec.makeMeasureSpec(0,
View.MeasureSpec.UNSPECIFIED);
view.measure(w, h);
mChildHeight = view.getMeasuredHeight();
mChildWidth = view.getMeasuredWidth();
Log.e(TAG, "子组件的宽:" + mChildWidth + ", 子组件的高:" + mChildHeight);
// 计算每次加载多少个View
mCountOneScreen = screenWidth / mChildWidth + 2;
Log.e(TAG, "mCountOneScreen = " + mCountOneScreen
+ " ,mChildWidth = " + mChildWidth);
}
//初始化第一屏幕的元素
loadFirstChild(mCountOneScreen);
}
/**
* 加载第一屏的View
*/
public void loadFirstChild(int mCountOneScreen) {
parent.removeAllViews();
mViewPos.clear();
for (int i = 0; i < mCountOneScreen; i++) {
View view = mAdapter.getView(i, null, parent);
view.setOnClickListener(this);
parent.addView(view);
mViewPos.put(view, i);
mCurrentIndex = i;
}
if (mListener != null) {
notifyCurrentImageChanged();
}
}
@Override
public boolean onTouchEvent(MotionEvent event) {
switch (event.getAction()) {
case MotionEvent.ACTION_MOVE:
int scrollX = getScrollX();
// 如果当前scrollX为view的宽度,加载下一张,移除第一张
if (scrollX  = mChildWidth) {
loadNextImage();
}
// 如果当前scrollX = 0, 往前设置一张,移除最后一张
if (scrollX == 0) {
loadPreImage();
}
break;
}
// 这里无论返回值是设置 true 还是 false
// HorizontalScrollView都不会滑动
return super.onTouchEvent(event);
}
/**
* 加载下一张图片
*/
protected void loadNextImage() {
// 数组边界值计算
if (mCurrentIndex == mAdapter.getCount() - 1) {
return;
}
//移除第一张图片,且将水平滚动位置置0(图片有宽度,所以为置0)
scrollTo(0, 0);
mViewPos.remove(parent.getChildAt(0));
parent.removeViewAt(0);
//获取下一张图片,并且设置onClick事件,且加入容器中
View view = mAdapter.getView(++mCurrentIndex, null, parent);
Log.e(TAG, "mCurrentIndex ===" + mCurrentIndex);
view.setOnClickListener(this);
parent.addView(view);
mViewPos.put(view, mCurrentIndex);
//当前第一张图片小标
mFristIndex++;
//如果设置了滚动监听则触发
if (mListener != null) {
notifyCurrentImageChanged();
}
}
/**
* 加载前一张图片
*/
protected void loadPreImage() {
//如果当前已经是第一张,则返回
if (mFristIndex == 0)
return;
//获得当前应该显示为第一张图片的下标
int index = mCurrentIndex - mCountOneScreen;
if (index  = 0) {
//移除最后一张
int oldViewPos = parent.getChildCount() - 1;
mViewPos.remove(parent.getChildAt(oldViewPos));
parent.removeViewAt(oldViewPos);
//将此View放入第一个位置
View view = mAdapter.getView(index, null, parent);
mViewPos.put(view, index);
parent.addView(view, 0);
view.setOnClickListener(this);
//水平滚动位置向左移动view的宽度个像素
scrollTo(mChildWidth, 0);
//当前位置--,当前第一个显示的下标--
mCurrentIndex--;
mFristIndex--;
//回调
if (mListener != null) {
notifyCurrentImageChanged();
}
}
}
/**
* 滑动时的回调
*/
public void notifyCurrentImageChanged() {
int sum = parent.getChildCount();
for (int i = 0; i < sum; i++) {
// 清除所有的背景色,点击时会设置为蓝色
parent.getChildAt(i).setBackgroundColor(Color.WHITE);
}
mListener.onCurrentImgChanged(mFristIndex, parent.getChildAt(0));
}
@Override
public void onClick(View v) {
if (mOnItemClickListener != null) {
int sum = parent.getChildCount();
for (int i = 0; i < sum; i++) {
parent.getChildAt(i).setBackgroundColor(Color.WHITE);
}
mOnItemClickListener.onItemClick(v, mViewPos.get(v));
}
}
public void setOnItemClickListener(OnItemClickListener mOnClickListener) {
this.mOnItemClickListener = mOnClickListener;
}
public void setCurrentImageChangeListener(CurrentImageChangeListener mListener) {
this.mListener = mListener;
}
}

该类中的很多数据都在 List 集合里面,而集合的下标初始值为 0,与 list.size() 不相等。顾猜测,正是由于这个原因,在 mImageView 显示图片时,不能显示到第 9 张。

在这个类中 计算每次加载多少个 View 时的 mCountOneScreen 计算方法感觉略有问题,从效果图中可以看出,屏幕中能加载 3 张多一点的图片。mCountOneScreen = screenWidth / mChildWidth + 2; 在我的模拟器上计算得出的结果等于 5,也就是为什么不能加载小于等于 4 张图片,如果想要让该屏幕底部上只显示 3 张即一个屏幕也就能显示完。那就不用水平滚动了,那样就感觉使用 HorizontalScrollView 失去了意义。

所用到的布局文件:

content_main.xml :

代码语言:javascript
复制
<?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:paddingLeft="5dp"
android:paddingTop="5dp"
android:paddingRight="5dp"
android:paddingBottom="5dp"
app:layout_behavior="@string/appbar_scrolling_view_behavior"
tools:context="com.crazy.horizontalscrollviewtest.MainActivity"
tools:showIn="@layout/activity_main" 
<ImageView
android:id="@+id/imageView"
android:layout_width="match_parent"
android:layout_height="380dp"
android:scaleType="centerCrop" / 
<com.crazy.horizontalscrollviewtest.MyHorizontalView
android:id="@+id/my_horizontal"
android:layout_below="@id/imageView"
android:layout_alignParentBottom="true"
android:scrollbars="none"
android:layout_width="match_parent"
android:layout_height="wrap_content" 
<LinearLayout
android:id="@+id/linear_layout"
android:orientation="horizontal"
android:layout_gravity="center_vertical"
android:layout_width="wrap_content"
android:layout_height="wrap_content" 
</LinearLayout 
</com.crazy.horizontalscrollviewtest.MyHorizontalView 
</RelativeLayout 

image_item_layout.xml (主要用于提供水平滚动的图片(屏幕底部)):

代码语言:javascript
复制
<?xml version="1.0" encoding="utf-8"? 
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"  
<ImageView 
android:id="@+id/top_image"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_margin="10dp"
android:scaleType="centerCrop"/ 
</RelativeLayout 

以上就是本文的全部内容,希望对大家的学习有所帮助。

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2020-09-11 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
相关产品与服务
容器服务
腾讯云容器服务(Tencent Kubernetes Engine, TKE)基于原生 kubernetes 提供以容器为核心的、高度可扩展的高性能容器管理服务,覆盖 Serverless、边缘计算、分布式云等多种业务部署场景,业内首创单个集群兼容多种计算节点的容器资源管理模式。同时产品作为云原生 Finops 领先布道者,主导开源项目Crane,全面助力客户实现资源优化、成本控制。
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档