我正在尝试制作一个圆形 ListView,其中列表项排列在半圆上。它应该看起来像这样:
有一个相关的帖子,但它已关闭。
我制作了自己的 Circular Custom ListView,它工作正常,但我的问题是我无法按照图像上显示的方式排列 List Items Half Circle 方式。我尝试了几件事,但它没有用,我不知道该怎么做。
因此,当我制作示例应用程序进行演示时,我必须做两件事。
首先,onDraw(Canvas)
在我的自定义视图上编辑。这可以是任何视图,为了简单起见,我将其设为 TextView。这使我可以根据一个方程来推动视图。
public class MyView extends TextView {
private static final int MAX_INDENT = 300;
private static final String TAG = MyView.class.getSimpleName();
public MyView(Context context) {
super(context);
}
public void onDraw(Canvas canvas){
canvas.save();
float indent = getIndent(getY());
canvas.translate(indent, 0);
super.onDraw(canvas);
canvas.restore();
}
public float getIndent(float distance){
float x_vertex = MAX_INDENT;
DisplayMetrics displayMetrics = getContext().getResources().getDisplayMetrics();
float y_vertex = displayMetrics.heightPixels / 2 / displayMetrics.density;
double a = ( 0 - x_vertex ) / ( Math.pow(( 0 - y_vertex), 2) ) ;
float indent = (float) (a * Math.pow((distance - y_vertex), 2) + x_vertex);
return indent;
}
}
我要做的第二件事是重写 ListView 类,使其实现 OnScrollListener 并调用setOnScrollListener(this);
. 现在我可以滚动列表,它遵循我放在视图中的等式。
public class HalfCircleListView extends ListView implements AbsListView.OnScrollListener {
public HalfCircleListView(Context context) {
super(context);
setOnScrollListener(this);
}
@Override
public void onScrollStateChanged(AbsListView absListView, int i) {
//Ignored
}
@Override
public void onScroll(AbsListView absListView, int i, int i2, int i3) {
absListView.invalidateViews();
}
}
您可以从我的Gist下载完整的源代码。
初始状态 滚动状态
如您所见,我的数学有点偏离...我使用抛物线与圆,因此必须进行更改。
您可以增加/减少适配器的 getView() 中返回的每个视图的左边距。因此,例如,对于您的前半部分视图,您将每个项目的边距增加 20 像素(int margin = index * 20),并在后半部分相应地减少它。
当然,这需要大量微调才能真正看起来像一个循环列表,但我想你明白了。
经过一番摸索,我找到了解决办法。
它像普通的 ListView 一样相对优化和可配置。
以下是主要代码片段:
CircularListView.java
package com.makotokw.android.widget;
import android.annotation.TargetApi;
import android.content.Context;
import android.database.DataSetObserver;
import android.os.Build;
import android.util.AttributeSet;
import android.view.KeyEvent;
import android.view.View;
import android.view.ViewGroup;
import android.widget.AbsListView;
import android.widget.ListAdapter;
import android.widget.ListView;
public class CircularListView extends ListView implements AbsListView.OnScrollListener {
private static final String TAG = CircularListView.class.getSimpleName();
private static final int REPEAT_COUNT = 3;
private int mItemHeight = 0;
private CircularListViewListener mCircularListViewListener;
private InfiniteListAdapter mInfiniteListAdapter;
private boolean mEnableInfiniteScrolling = true;
private CircularListViewContentAlignment mCircularListViewContentAlignment = CircularListViewContentAlignment.Left;
private double mRadius = -1;
private int mSmoothScrollDuration = 80;
public CircularListView(Context context) {
this(context, null);
}
public CircularListView(Context context, AttributeSet attrs) {
this(context, attrs, android.R.attr.listViewStyle);
}
public CircularListView(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
setOnScrollListener(this);
setClipChildren(false);
setEnableInfiniteScrolling(true);
}
public void setAdapter(ListAdapter adapter) {
mInfiniteListAdapter = new InfiniteListAdapter(adapter);
mInfiniteListAdapter.setEnableInfiniteScrolling(mEnableInfiniteScrolling);
super.setAdapter(mInfiniteListAdapter);
}
public CircularListViewListener getCircularListViewListener() {
return mCircularListViewListener;
}
public void setCircularListViewListener(CircularListViewListener circularListViewListener) {
this.mCircularListViewListener = circularListViewListener;
}
public void setEnableInfiniteScrolling(boolean enableInfiniteScrolling) {
mEnableInfiniteScrolling = enableInfiniteScrolling;
if (mInfiniteListAdapter != null) {
mInfiniteListAdapter.setEnableInfiniteScrolling(enableInfiniteScrolling);
}
if (mEnableInfiniteScrolling) {
setHorizontalScrollBarEnabled(false);
setVerticalScrollBarEnabled(false);
}
}
public CircularListViewContentAlignment getCircularListViewContentAlignment() {
return mCircularListViewContentAlignment;
}
public void setCircularListViewContentAlignment(
CircularListViewContentAlignment circularListViewContentAlignment) {
if (mCircularListViewContentAlignment != circularListViewContentAlignment) {
mCircularListViewContentAlignment = circularListViewContentAlignment;
requestLayout();
}
}
public double getRadius() {
return mRadius;
}
public void setRadius(double radius) {
if (this.mRadius != radius) {
this.mRadius = radius;
requestLayout();
}
}
public int getCentralPosition() {
double vCenterPos = getHeight() / 2.0f;
for (int i = 0; i < getChildCount(); i++) {
View child = getChildAt(i);
if (child != null) {
if (child.getTop() <= vCenterPos
&& child.getTop() + child.getHeight() >= vCenterPos) {
return getFirstVisiblePosition() + i;
}
}
}
return -1;
}
public View getCentralChild() {
int pos = getCentralPosition();
if (pos != -1) {
return getChildAt(pos - getFirstVisiblePosition());
}
return null;
}
public void scrollFirstItemToCenter() {
if (!mEnableInfiniteScrolling) {
return;
}
int realTotalItemCount = mInfiniteListAdapter.getRealCount();
if (realTotalItemCount > 0) {
setSelectionFromTop(realTotalItemCount, getBaseCentralChildTop());
}
}
public int getBaseCentralChildTop() {
int itemHeight = getItemHeight();
if (itemHeight > 0) {
return getHeight() / 2 - itemHeight / 2;
}
return 0;
}
public int getItemHeight() {
if (mItemHeight == 0) {
View child = getChildAt(0);
if (child != null) {
mItemHeight = child.getHeight();
}
}
return mItemHeight;
}
public void setSelectionAndMoveToCenter(int position) {
if (!mEnableInfiniteScrolling) {
return;
}
int realTotalItemCount = mInfiniteListAdapter.getRealCount();
if (realTotalItemCount == 0) {
return;
}
position = position % realTotalItemCount;
int centralPosition = getCentralPosition() % realTotalItemCount;
int y = getBaseCentralChildTop();
if (centralPosition == position) {
View centralView = getCentralChild();
y = centralView.getTop();
}
setSelectionFromTop(position + realTotalItemCount, y);
}
@TargetApi(Build.VERSION_CODES.FROYO)
@Override
public boolean dispatchKeyEvent(KeyEvent event) {
if (mEnableInfiniteScrolling) {
if (event.getAction() == KeyEvent.ACTION_DOWN) {
switch (event.getKeyCode()) {
case KeyEvent.KEYCODE_DPAD_UP:
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.FROYO) {
smoothScrollBy(mItemHeight, mSmoothScrollDuration);
return true;
}
break;
case KeyEvent.KEYCODE_DPAD_DOWN:
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.FROYO) {
smoothScrollBy(-mItemHeight, mSmoothScrollDuration);
return true;
}
break;
default:
break;
}
}
}
return super.dispatchKeyEvent(event);
}
@Override
public void onScrollStateChanged(AbsListView view, int scrollState) {
if (scrollState == SCROLL_STATE_IDLE) {
if (!isInTouchMode()) {
setSelectionAndMoveToCenter(getCentralPosition());
}
}
}
@Override
public void onScroll(AbsListView view, int firstVisibleItem, int visibleItemCount,
int totalItemCount) {
if (!mEnableInfiniteScrolling) {
return;
}
View itemView = this.getChildAt(0);
if (itemView == null) {
return;
}
int realTotalItemCount = mInfiniteListAdapter.getRealCount();
if (realTotalItemCount == 0) {
return;
}
if (mItemHeight == 0) {
mItemHeight = itemView.getHeight();
}
if (firstVisibleItem == 0) {
// scroll one unit
this.setSelectionFromTop(realTotalItemCount, itemView.getTop());
}
if (totalItemCount == firstVisibleItem + visibleItemCount) {
// back one unit
this.setSelectionFromTop(firstVisibleItem - realTotalItemCount,
itemView.getTop());
}
if (mCircularListViewContentAlignment != CircularListViewContentAlignment.None) {
double viewHalfHeight = view.getHeight() / 2.0f;
double vRadius = view.getHeight();
double hRadius = view.getWidth();
double yRadius = (view.getHeight() + mItemHeight) / 2.0f;
double xRadius = (vRadius < hRadius) ? vRadius : hRadius;
if (mRadius > 0) {
xRadius = mRadius;
}
for (int i = 0; i < visibleItemCount; i++) {
itemView = this.getChildAt(i);
if (itemView != null) {
double y = Math.abs(viewHalfHeight - (itemView.getTop() + (itemView.getHeight() / 2.0f)));
y = Math.min(y, yRadius);
double angle = Math.asin(y / yRadius);
double x = xRadius * Math.cos(angle);
if (mCircularListViewContentAlignment == CircularListViewContentAlignment.Left) {
x -= xRadius;
} else {
x = xRadius / 2 - x;
}
itemView.scrollTo((int) x, 0);
}
}
} else {
for (int i = 0; i < visibleItemCount; i++) {
itemView = this.getChildAt(i);
if (itemView != null) {
itemView.scrollTo(0, 0);
}
}
}
if (mCircularListViewListener != null) {
mCircularListViewListener.onCircularLayoutFinished(this, firstVisibleItem, visibleItemCount, totalItemCount);
}
}
class InfiniteListAdapter implements ListAdapter {
private boolean mEnableInfiniteScrolling = true;
private ListAdapter mCoreAdapter;
public InfiniteListAdapter(ListAdapter coreAdapter) {
mCoreAdapter = coreAdapter;
}
private void setEnableInfiniteScrolling(boolean enableInfiniteScrolling) {
mEnableInfiniteScrolling = enableInfiniteScrolling;
}
public int getRealCount() {
return mCoreAdapter.getCount();
}
public int positionToIndex(int position) {
int count = mCoreAdapter.getCount();
return (count == 0) ? 0 : position % count;
}
@Override
public void registerDataSetObserver(DataSetObserver observer) {
mCoreAdapter.registerDataSetObserver(observer);
}
@Override
public void unregisterDataSetObserver(DataSetObserver observer) {
mCoreAdapter.unregisterDataSetObserver(observer);
}
@Override
public int getCount() {
int count = mCoreAdapter.getCount();
return (mEnableInfiniteScrolling) ? count * REPEAT_COUNT : count;
}
@Override
public Object getItem(int position) {
return mCoreAdapter.getItem(this.positionToIndex(position));
}
@Override
public long getItemId(int position) {
return mCoreAdapter.getItemId(this.positionToIndex(position));
}
@Override
public boolean hasStableIds() {
return mCoreAdapter.hasStableIds();
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
return mCoreAdapter.getView(this.positionToIndex(position), convertView, parent);
}
@Override
public int getItemViewType(int position) {
return mCoreAdapter.getItemViewType(this.positionToIndex(position));
}
@Override
public int getViewTypeCount() {
return mCoreAdapter.getViewTypeCount();
}
@Override
public boolean isEmpty() {
return mCoreAdapter.isEmpty();
}
@Override
public boolean areAllItemsEnabled() {
return mCoreAdapter.areAllItemsEnabled();
}
@Override
public boolean isEnabled(int position) {
return mCoreAdapter.isEnabled(this.positionToIndex(position));
}
}
}
**CircularListViewContentAlignment.java**
package com.makotokw.android.widget;
public enum CircularListViewContentAlignment {
None,
Left,
Right
}
**CircularListViewListener.java**
package com.makotokw.android.widget;
public interface CircularListViewListener {
void onCircularLayoutFinished(CircularListView circularListView,
int firstVisibleItem,
int visibleItemCount,
int totalItemCount);
}
如需更多说明,您还可以查看我的博客文章并对此发表评论。您可以下载 Eclipse 的示例应用程序。
我的博客是:
http ://androidpantiii.blogspot.in/2015/11/half-circular-list-view-there-were.html