Android使用HorizontalScrollView實(shí)現(xiàn)水平滾動(dòng)
HorizontalScrollView 和 ScrollView 都是由 FrameLayout 派生出來(lái)的。它們就是一個(gè)用于為普通組件添加滾動(dòng)條的組件。且 HorizontalScrollView 和 ScrollView 里面最多只能包含一個(gè)組件(當(dāng)然組件里面還可以嵌套組件)。它們不同的是 HorizontalScrollView 用于添加水平滾動(dòng),而 ScrollView 用于添加垂直滾動(dòng)。
突然間想到 做一個(gè)屏幕下方水平滑動(dòng),屏幕上方并作出相應(yīng)的反應(yīng)的效果。只是在下方滾動(dòng)時(shí),屏幕上方?jīng)]有作出理想的反應(yīng),點(diǎn)擊事件倒是實(shí)現(xiàn)了。最終只能在網(wǎng)上搜索,終于找到了一個(gè)。于是作出的效果如下:

只是這個(gè)效果還有所缺陷,加載了 13 張圖片,在屏幕下方水平滾動(dòng)到最后一頁(yè)時(shí),第 9 張的圖片并沒有在上面的顯示出來(lái)(原作者的也有這個(gè)問題);如果圖片的數(shù)量小于或者等于 4 張時(shí)則不能運(yùn)行。
本例的難點(diǎn)主要在于 MyHorizontalView 類中,并且還有收集而來(lái)的注解。
MainActivity.java :
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);
//設(shè)置適配器
myHorizontalView.initDatas(adapter);
//添加滾動(dòng)回調(diào)
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"));
}
});
//添加點(diǎn)擊回調(diào)
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 及其子類提供列表項(xiàng)的。它主要用于為 HorizontalScrollView 提供數(shù)據(jù)。
MyAdapter.java :
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;
// 此處要用相對(duì)布局,且與 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", "正在檢測(cè)數(shù)據(jù)來(lái)了沒有 ");
}
Bitmap btm = (Bitmap) getItem(position);
viewHolder.image.setImageBitmap(btm);
Log.e("MyAdapter", "信息來(lái)了哦!");
return layout;
}
private static class ViewHolder {
ImageView image;
}
}
MyHorizontalView 類主要用于未 MainAcitivity 類提供接口、水平滾動(dòng)時(shí)屏幕上方的反應(yīng)及相應(yīng)的點(diǎn)擊事件等。該類主要使用了收集而來(lái)的代碼,并做了相應(yīng)的調(diào)整。
MyHorizontalView.java :
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;
/* 當(dāng)前最后一張圖片的index*/
private int mCurrentIndex;
/* 當(dāng)前第一張圖片的下標(biāo)*/
private int mFristIndex;
/* 每屏幕最多顯示的個(gè)數(shù)*/
private int mCountOneScreen;
/* 子元素的寬度*/
private int mChildWidth;
/* 子元素的高度*/
private int mChildHeight;
private MyAdapter mAdapter;
private CurrentImageChangeListener mListener;
private OnItemClickListener mOnItemClickListener;
/**
* 圖片滾動(dòng)時(shí)的回調(diào)接口
*/
public interface CurrentImageChangeListener {
void onCurrentImgChanged(int position, View viewIndicator);
}
/**
* 點(diǎn)擊條目時(shí)的回調(diào)
*/
public interface OnItemClickListener {
void onItemClick(View view, int pos);
}
/* 保存View與位置的鍵值對(duì) */
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;
}
/**
* 初始化數(shù)據(jù),設(shè)置數(shù)據(jù)適配器
*/
public void initDatas(MyAdapter mAdapter) {
if (getChildCount() == 0) {
Log.e(TAG, "必須要有子元素");
}
if (getChildCount() == 0 || mAdapter == null)
return;
this.mAdapter = mAdapter;
parent = (ViewGroup) getChildAt(0);
// 獲得適配器中第一個(gè)View
final View view = mAdapter.getView(0, null, parent);
parent.addView(view);
// 強(qiáng)制計(jì)算當(dāng)前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);
// 計(jì)算每次加載多少個(gè)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();
// 如果當(dāng)前scrollX為view的寬度,加載下一張,移除第一張
if (scrollX >= mChildWidth) {
loadNextImage();
}
// 如果當(dāng)前scrollX = 0, 往前設(shè)置一張,移除最后一張
if (scrollX == 0) {
loadPreImage();
}
break;
}
// 這里無(wú)論返回值是設(shè)置 true 還是 false
// HorizontalScrollView都不會(huì)滑動(dòng)
return super.onTouchEvent(event);
}
/**
* 加載下一張圖片
*/
protected void loadNextImage() {
// 數(shù)組邊界值計(jì)算
if (mCurrentIndex == mAdapter.getCount() - 1) {
return;
}
//移除第一張圖片,且將水平滾動(dòng)位置置0(圖片有寬度,所以為置0)
scrollTo(0, 0);
mViewPos.remove(parent.getChildAt(0));
parent.removeViewAt(0);
//獲取下一張圖片,并且設(shè)置onClick事件,且加入容器中
View view = mAdapter.getView(++mCurrentIndex, null, parent);
Log.e(TAG, "mCurrentIndex ===" + mCurrentIndex);
view.setOnClickListener(this);
parent.addView(view);
mViewPos.put(view, mCurrentIndex);
//當(dāng)前第一張圖片小標(biāo)
mFristIndex++;
//如果設(shè)置了滾動(dòng)監(jiān)聽則觸發(fā)
if (mListener != null) {
notifyCurrentImageChanged();
}
}
/**
* 加載前一張圖片
*/
protected void loadPreImage() {
//如果當(dāng)前已經(jīng)是第一張,則返回
if (mFristIndex == 0)
return;
//獲得當(dāng)前應(yīng)該顯示為第一張圖片的下標(biāo)
int index = mCurrentIndex - mCountOneScreen;
if (index >= 0) {
//移除最后一張
int oldViewPos = parent.getChildCount() - 1;
mViewPos.remove(parent.getChildAt(oldViewPos));
parent.removeViewAt(oldViewPos);
//將此View放入第一個(gè)位置
View view = mAdapter.getView(index, null, parent);
mViewPos.put(view, index);
parent.addView(view, 0);
view.setOnClickListener(this);
//水平滾動(dòng)位置向左移動(dòng)view的寬度個(gè)像素
scrollTo(mChildWidth, 0);
//當(dāng)前位置--,當(dāng)前第一個(gè)顯示的下標(biāo)--
mCurrentIndex--;
mFristIndex--;
//回調(diào)
if (mListener != null) {
notifyCurrentImageChanged();
}
}
}
/**
* 滑動(dòng)時(shí)的回調(diào)
*/
public void notifyCurrentImageChanged() {
int sum = parent.getChildCount();
for (int i = 0; i < sum; i++) {
// 清除所有的背景色,點(diǎn)擊時(shí)會(huì)設(shè)置為藍(lán)色
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;
}
}
該類中的很多數(shù)據(jù)都在 List 集合里面,而集合的下標(biāo)初始值為 0,與 list.size() 不相等。顧猜測(cè),正是由于這個(gè)原因,在 mImageView 顯示圖片時(shí),不能顯示到第 9 張。
在這個(gè)類中 計(jì)算每次加載多少個(gè) View 時(shí)的 mCountOneScreen 計(jì)算方法感覺略有問題,從效果圖中可以看出,屏幕中能加載 3 張多一點(diǎn)的圖片。mCountOneScreen = screenWidth / mChildWidth + 2; 在我的模擬器上計(jì)算得出的結(jié)果等于 5,也就是為什么不能加載小于等于 4 張圖片,如果想要讓該屏幕底部上只顯示 3 張即一個(gè)屏幕也就能顯示完。那就不用水平滾動(dòng)了,那樣就感覺使用 HorizontalScrollView 失去了意義。
所用到的布局文件:
content_main.xml :
<?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 (主要用于提供水平滾動(dòng)的圖片(屏幕底部)):
<?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>
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
解決Android使用Handler造成內(nèi)存泄露問題
內(nèi)存泄露的危害就是會(huì)使虛擬機(jī)占用內(nèi)存過高,導(dǎo)致OOM(內(nèi)存溢出),程序出錯(cuò)。接下來(lái)通過本文給大家分享Android使用Handler造成內(nèi)存泄露問題及解決方法,一起看看吧2017-08-08
Android使用Websocket實(shí)現(xiàn)聊天室
這篇文章主要為大家詳細(xì)介紹了Android使用Websocket實(shí)現(xiàn)聊天室,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-03-03
Android實(shí)現(xiàn)關(guān)機(jī)重啟的方法分享
這篇文章主要介紹了Android實(shí)現(xiàn)關(guān)機(jī)重啟的方法,需要的朋友可以參考下2014-02-02
android實(shí)現(xiàn)切換日期左右無(wú)限滑動(dòng)效果
本篇內(nèi)容給大家分享了android開發(fā)時(shí)候?qū)崿F(xiàn)自定義的日期無(wú)限左右滑動(dòng)效果以及控件使用的技巧。2017-11-11
一款非常簡(jiǎn)單酷炫的LoadingView動(dòng)畫效果
這篇文章主要為大家詳細(xì)介紹了一款非常簡(jiǎn)單酷炫的LoadingView動(dòng)畫效果,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-08-08
Android 中Fragment與Activity通訊的詳解
這篇文章主要介紹了Android 中Fragment與Activity通訊的詳解的相關(guān)資料,希望通過本文能幫助到大家,讓大家理解掌握如何通信的,需要的朋友可以參考下2017-10-10
Retrofit網(wǎng)絡(luò)請(qǐng)求和響應(yīng)處理重點(diǎn)分析講解
這篇文章主要介紹了Retrofit網(wǎng)絡(luò)請(qǐng)求和響應(yīng)處理重點(diǎn)分析,在使用?Retrofit發(fā)起網(wǎng)絡(luò)請(qǐng)求時(shí),我們可以通過定義一個(gè)接口并使用Retrofit的注解來(lái)描述這個(gè)接口中的請(qǐng)求,Retrofit會(huì)自動(dòng)生成一個(gè)實(shí)現(xiàn)該接口的代理對(duì)象2023-03-03

