经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » 移动开发 » Android » 查看文章
Android使用HorizontalScrollView实现水平滚动
来源:jb51  时间:2018/11/25 19:38:57  对本文有异议

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

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

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

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

MainActivity.java :

  1. package com.crazy.horizontalscrollviewtest;
  2. import java.io.InputStream;
  3. import java.util.ArrayList;
  4. import java.util.Arrays;
  5. import java.util.List;
  6. import com.crazy.horizontalscrollviewtest.MyHorizontalView.CurrentImageChangeListener;
  7. import com.crazy.horizontalscrollviewtest.MyHorizontalView.OnItemClickListener;
  8. import android.content.Context;
  9. import android.graphics.Bitmap;
  10. import android.graphics.BitmapFactory;
  11. import android.graphics.Color;
  12. import android.os.Bundle;
  13. import android.util.Log;
  14. import android.view.View;
  15. import android.widget.ImageView;
  16. import android.support.v7.app.AppCompatActivity;
  17. public class MainActivity extends AppCompatActivity {
  18. private ImageView mImageView;
  19. private MyHorizontalView myHorizontalView;
  20. private List<Bitmap> bitmapList;
  21. private MyAdapter adapter;
  22. @Override
  23. protected void onCreate(Bundle savedInstanceState) {
  24. super.onCreate(savedInstanceState);
  25. setContentView(R.layout.activity_main);
  26. init();
  27. }
  28. private void init() {
  29. mImageView = (ImageView)findViewById(R.id.imageView);
  30. bitmapList = new ArrayList<>(Arrays.asList(
  31. readBitMap(this, R.drawable.bricks),
  32. readBitMap(this, R.drawable.dog),
  33. readBitMap(this, R.drawable.flower),
  34. readBitMap(this, R.drawable.grass),
  35. readBitMap(this, R.drawable.stones),
  36. readBitMap(this, R.drawable.wood),
  37. readBitMap(this, R.drawable.bg_01),
  38. readBitMap(this, R.drawable.bg_02),
  39. readBitMap(this, R.drawable.bg_03),
  40. readBitMap(this, R.drawable.bg_04),
  41. readBitMap(this, R.drawable.bg_05),
  42. readBitMap(this, R.drawable.bg_06),
  43. readBitMap(this, R.drawable.bg_07)
  44. ));
  45. myHorizontalView = (MyHorizontalView)findViewById(R.id.my_horizontal);
  46. adapter = new MyAdapter(this, bitmapList);
  47. //设置适配器
  48. myHorizontalView.initDatas(adapter);
  49. //添加滚动回调
  50. myHorizontalView
  51. .setCurrentImageChangeListener(new CurrentImageChangeListener() {
  52. @Override
  53. public void onCurrentImgChanged(int position, View viewIndicator) {
  54. Log.e("==============","=============== " + position);
  55. mImageView.setImageBitmap(bitmapList.get(position));
  56. viewIndicator.setBackgroundColor(Color.parseColor("#AA024DA4"));
  57. }
  58. });
  59. //添加点击回调
  60. myHorizontalView.setOnItemClickListener(new OnItemClickListener() {
  61. @Override
  62. public void onItemClick(View view, int position) {
  63. mImageView.setImageBitmap(bitmapList.get(position));
  64. view.setBackgroundColor(Color.parseColor("#AA024DA4"));
  65. }
  66. });
  67. }
  68. public static Bitmap readBitMap(Context mContext, int resId) {
  69. BitmapFactory.Options opt = new BitmapFactory.Options();
  70. opt.inPreferredConfig = Bitmap.Config.RGB_565;
  71. opt.inPurgeable = true;
  72. opt.inInputShareable = true;
  73. InputStream is = mContext.getResources().openRawResource(resId);
  74. return BitmapFactory.decodeStream(is, null, opt);
  75. }
  76. }

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

MyAdapter.java :

  1. package com.crazy.horizontalscrollviewtest;
  2. import java.util.ArrayList;
  3. import java.util.List;
  4. import android.content.Context;
  5. import android.graphics.Bitmap;
  6. import android.util.Log;
  7. import android.view.View;
  8. import android.view.ViewGroup;
  9. import android.widget.BaseAdapter;
  10. import android.widget.ImageView;
  11. import android.widget.RelativeLayout;
  12. /**
  13. * Created by antimage on 2016/1/9.
  14. */
  15. public class MyAdapter extends BaseAdapter {
  16. private List<Bitmap> bitmapList;
  17. private Context mContext;
  18. public MyAdapter(Context context, List<Bitmap> bitmapList) {
  19. mContext = context;
  20. if (bitmapList == null) {
  21. bitmapList = new ArrayList<Bitmap>();
  22. } else {
  23. this.bitmapList = bitmapList;
  24. }
  25. }
  26. @Override
  27. public int getCount() {
  28. return bitmapList.size();
  29. }
  30. @Override
  31. public Object getItem(int position) {
  32. return bitmapList.get(position);
  33. }
  34. @Override
  35. public long getItemId(int position) {
  36. return position;
  37. }
  38. @Override
  39. public View getView(int position, View convertView, ViewGroup parent) {
  40. ViewHolder viewHolder = null;
  41. View view = null;
  42. // 此处要用相对布局,且与 XML 中的布局相同;
  43. // 如果使用线性布局,则显示不完整
  44. RelativeLayout layout;
  45. if (convertView == null) {
  46. layout = (RelativeLayout) View.inflate(mContext, R.layout.image_item_layout, null);
  47. viewHolder = new ViewHolder();
  48. viewHolder.image = (ImageView) layout.findViewById(R.id.top_image);
  49. layout.setTag(viewHolder);
  50. } else {
  51. layout = (RelativeLayout) convertView;
  52. view = layout;
  53. viewHolder = (ViewHolder) layout.getTag();
  54. Log.e("MyAdapter", "正在检测数据来了没有 ");
  55. }
  56. Bitmap btm = (Bitmap) getItem(position);
  57. viewHolder.image.setImageBitmap(btm);
  58. Log.e("MyAdapter", "信息来了哦!");
  59. return layout;
  60. }
  61. private static class ViewHolder {
  62. ImageView image;
  63. }
  64. }

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

MyHorizontalView.java :

  1. package com.crazy.horizontalscrollviewtest;
  2. import java.util.HashMap;
  3. import java.util.Map;
  4. import android.app.Activity;
  5. import android.content.Context;
  6. import android.graphics.Color;
  7. import android.util.AttributeSet;
  8. import android.util.DisplayMetrics;
  9. import android.util.Log;
  10. import android.view.MotionEvent;
  11. import android.view.View;
  12. import android.view.ViewGroup;
  13. import android.widget.HorizontalScrollView;
  14. /**
  15. * Created by antimage on 2016/1/9.
  16. */
  17. public class MyHorizontalView extends HorizontalScrollView
  18. implements View.OnClickListener {
  19. private String TAG = "MyHorizontalView";
  20. private ViewGroup parent;
  21. private int screenWidth;
  22. /* 当前最后一张图片的index*/
  23. private int mCurrentIndex;
  24. /* 当前第一张图片的下标*/
  25. private int mFristIndex;
  26. /* 每屏幕最多显示的个数*/
  27. private int mCountOneScreen;
  28. /* 子元素的宽度*/
  29. private int mChildWidth;
  30. /* 子元素的高度*/
  31. private int mChildHeight;
  32. private MyAdapter mAdapter;
  33. private CurrentImageChangeListener mListener;
  34. private OnItemClickListener mOnItemClickListener;
  35. /**
  36. * 图片滚动时的回调接口
  37. */
  38. public interface CurrentImageChangeListener {
  39. void onCurrentImgChanged(int position, View viewIndicator);
  40. }
  41. /**
  42. * 点击条目时的回调
  43. */
  44. public interface OnItemClickListener {
  45. void onItemClick(View view, int pos);
  46. }
  47. /* 保存View与位置的键值对 */
  48. private Map<View, Integer> mViewPos = new HashMap<>();
  49. public MyHorizontalView(Context context) {
  50. this(context, null);
  51. }
  52. public MyHorizontalView(Context context, AttributeSet attrs) {
  53. this(context, attrs, 0);
  54. }
  55. public MyHorizontalView(Context context, AttributeSet attrs, int defStyleAttr) {
  56. super(context, attrs, defStyleAttr);
  57. this.setSmoothScrollingEnabled(true);
  58. DisplayMetrics metrics = new DisplayMetrics();
  59. // 取得窗口属性
  60. ((Activity) context).getWindowManager().getDefaultDisplay().getMetrics(metrics);
  61. // 窗口的宽度 (像素)
  62. screenWidth = metrics.widthPixels;
  63. }
  64. /**
  65. * 初始化数据,设置数据适配器
  66. */
  67. public void initDatas(MyAdapter mAdapter) {
  68. if (getChildCount() == 0) {
  69. Log.e(TAG, "必须要有子元素");
  70. }
  71. if (getChildCount() == 0 || mAdapter == null)
  72. return;
  73. this.mAdapter = mAdapter;
  74. parent = (ViewGroup) getChildAt(0);
  75. // 获得适配器中第一个View
  76. final View view = mAdapter.getView(0, null, parent);
  77. parent.addView(view);
  78. // 强制计算当前View的宽和高
  79. if (mChildWidth == 0 && mChildHeight == 0) {
  80. int w = View.MeasureSpec.makeMeasureSpec(0,
  81. View.MeasureSpec.UNSPECIFIED);
  82. int h = View.MeasureSpec.makeMeasureSpec(0,
  83. View.MeasureSpec.UNSPECIFIED);
  84. view.measure(w, h);
  85. mChildHeight = view.getMeasuredHeight();
  86. mChildWidth = view.getMeasuredWidth();
  87. Log.e(TAG, "子组件的宽:" + mChildWidth + ", 子组件的高:" + mChildHeight);
  88. // 计算每次加载多少个View
  89. mCountOneScreen = screenWidth / mChildWidth + 2;
  90. Log.e(TAG, "mCountOneScreen = " + mCountOneScreen
  91. + " ,mChildWidth = " + mChildWidth);
  92. }
  93. //初始化第一屏幕的元素
  94. loadFirstChild(mCountOneScreen);
  95. }
  96. /**
  97. * 加载第一屏的View
  98. */
  99. public void loadFirstChild(int mCountOneScreen) {
  100. parent.removeAllViews();
  101. mViewPos.clear();
  102. for (int i = 0; i < mCountOneScreen; i++) {
  103. View view = mAdapter.getView(i, null, parent);
  104. view.setOnClickListener(this);
  105. parent.addView(view);
  106. mViewPos.put(view, i);
  107. mCurrentIndex = i;
  108. }
  109. if (mListener != null) {
  110. notifyCurrentImageChanged();
  111. }
  112. }
  113. @Override
  114. public boolean onTouchEvent(MotionEvent event) {
  115. switch (event.getAction()) {
  116. case MotionEvent.ACTION_MOVE:
  117. int scrollX = getScrollX();
  118. // 如果当前scrollX为view的宽度,加载下一张,移除第一张
  119. if (scrollX >= mChildWidth) {
  120. loadNextImage();
  121. }
  122. // 如果当前scrollX = 0, 往前设置一张,移除最后一张
  123. if (scrollX == 0) {
  124. loadPreImage();
  125. }
  126. break;
  127. }
  128. // 这里无论返回值是设置 true 还是 false
  129. // HorizontalScrollView都不会滑动
  130. return super.onTouchEvent(event);
  131. }
  132. /**
  133. * 加载下一张图片
  134. */
  135. protected void loadNextImage() {
  136. // 数组边界值计算
  137. if (mCurrentIndex == mAdapter.getCount() - 1) {
  138. return;
  139. }
  140. //移除第一张图片,且将水平滚动位置置0(图片有宽度,所以为置0)
  141. scrollTo(0, 0);
  142. mViewPos.remove(parent.getChildAt(0));
  143. parent.removeViewAt(0);
  144. //获取下一张图片,并且设置onClick事件,且加入容器中
  145. View view = mAdapter.getView(++mCurrentIndex, null, parent);
  146. Log.e(TAG, "mCurrentIndex ===" + mCurrentIndex);
  147. view.setOnClickListener(this);
  148. parent.addView(view);
  149. mViewPos.put(view, mCurrentIndex);
  150. //当前第一张图片小标
  151. mFristIndex++;
  152. //如果设置了滚动监听则触发
  153. if (mListener != null) {
  154. notifyCurrentImageChanged();
  155. }
  156. }
  157. /**
  158. * 加载前一张图片
  159. */
  160. protected void loadPreImage() {
  161. //如果当前已经是第一张,则返回
  162. if (mFristIndex == 0)
  163. return;
  164. //获得当前应该显示为第一张图片的下标
  165. int index = mCurrentIndex - mCountOneScreen;
  166. if (index >= 0) {
  167. //移除最后一张
  168. int oldViewPos = parent.getChildCount() - 1;
  169. mViewPos.remove(parent.getChildAt(oldViewPos));
  170. parent.removeViewAt(oldViewPos);
  171. //将此View放入第一个位置
  172. View view = mAdapter.getView(index, null, parent);
  173. mViewPos.put(view, index);
  174. parent.addView(view, 0);
  175. view.setOnClickListener(this);
  176. //水平滚动位置向左移动view的宽度个像素
  177. scrollTo(mChildWidth, 0);
  178. //当前位置--,当前第一个显示的下标--
  179. mCurrentIndex--;
  180. mFristIndex--;
  181. //回调
  182. if (mListener != null) {
  183. notifyCurrentImageChanged();
  184. }
  185. }
  186. }
  187. /**
  188. * 滑动时的回调
  189. */
  190. public void notifyCurrentImageChanged() {
  191. int sum = parent.getChildCount();
  192. for (int i = 0; i < sum; i++) {
  193. // 清除所有的背景色,点击时会设置为蓝色
  194. parent.getChildAt(i).setBackgroundColor(Color.WHITE);
  195. }
  196. mListener.onCurrentImgChanged(mFristIndex, parent.getChildAt(0));
  197. }
  198. @Override
  199. public void onClick(View v) {
  200. if (mOnItemClickListener != null) {
  201. int sum = parent.getChildCount();
  202. for (int i = 0; i < sum; i++) {
  203. parent.getChildAt(i).setBackgroundColor(Color.WHITE);
  204. }
  205. mOnItemClickListener.onItemClick(v, mViewPos.get(v));
  206. }
  207. }
  208. public void setOnItemClickListener(OnItemClickListener mOnClickListener) {
  209. this.mOnItemClickListener = mOnClickListener;
  210. }
  211. public void setCurrentImageChangeListener(CurrentImageChangeListener mListener) {
  212. this.mListener = mListener;
  213. }
  214. }

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

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

所用到的布局文件:

content_main.xml :

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
  3. xmlns:app="http://schemas.android.com/apk/res-auto"
  4. xmlns:tools="http://schemas.android.com/tools"
  5. android:layout_width="match_parent"
  6. android:layout_height="match_parent"
  7. android:paddingLeft="5dp"
  8. android:paddingTop="5dp"
  9. android:paddingRight="5dp"
  10. android:paddingBottom="5dp"
  11. app:layout_behavior="@string/appbar_scrolling_view_behavior"
  12. tools:context="com.crazy.horizontalscrollviewtest.MainActivity"
  13. tools:showIn="@layout/activity_main">
  14. <ImageView
  15. android:id="@+id/imageView"
  16. android:layout_width="match_parent"
  17. android:layout_height="380dp"
  18. android:scaleType="centerCrop" />
  19. <com.crazy.horizontalscrollviewtest.MyHorizontalView
  20. android:id="@+id/my_horizontal"
  21. android:layout_below="@id/imageView"
  22. android:layout_alignParentBottom="true"
  23. android:scrollbars="none"
  24. android:layout_width="match_parent"
  25. android:layout_height="wrap_content">
  26. <LinearLayout
  27. android:id="@+id/linear_layout"
  28. android:orientation="horizontal"
  29. android:layout_gravity="center_vertical"
  30. android:layout_width="wrap_content"
  31. android:layout_height="wrap_content">
  32. </LinearLayout>
  33. </com.crazy.horizontalscrollviewtest.MyHorizontalView>
  34. </RelativeLayout>

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

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
  3. android:layout_width="match_parent"
  4. android:layout_height="match_parent"
  5. android:orientation="vertical" >
  6. <ImageView
  7. android:id="@+id/top_image"
  8. android:layout_width="wrap_content"
  9. android:layout_height="wrap_content"
  10. android:layout_margin="10dp"
  11. android:scaleType="centerCrop"/>
  12. </RelativeLayout>

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

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号