リストビューがあり、いくつかのロジックに応じて、一時的にスクロールを無効にしたい。 view.setOnScrollListener(null);私はいくつかのコードを書く必要があると思いますが、誰かが私に履歴やスニペットを教えてもらえますか?
ありがとう
CustomListViewで:
@Override
public boolean dispatchTouchEvent(MotionEvent ev){
if(ev.getAction()==MotionEvent.ACTION_MOVE)
return true;
return super.dispatchTouchEvent(ev);
}
その後、ListViewはクリックに反応しますが、スクロール位置は変更しません。
新しいカスタムListViewを作成しない別のオプションは、onTouchListener
をListViewにアタッチし、モーションイベントアクションがACTION_MOVE
の場合、onTouch()
コールバックでtrueを返すことです。
listView.setOnTouchListener(new OnTouchListener() {
public boolean onTouch(View v, MotionEvent event) {
return (event.getAction() == MotionEvent.ACTION_MOVE);
}
});
listview.setEnabled(false)を使用して、リストビューのスクロールを無効にします
注:これにより、行の選択も無効になります
リストアイテムにバインドされたイベントがある場合、これらのソリューションのいずれかでリストをドラッグすると、イベントがトリガーされます。次の方法を使用して、選択したアイテム(ポインターヌルの回答から適応)からドラッグしてイベントをキャンセルするユーザーの期待を考慮します。
@Override
public boolean dispatchTouchEvent(MotionEvent ev) {
final int actionMasked = ev.getActionMasked() & MotionEvent.ACTION_MASK;
if (actionMasked == MotionEvent.ACTION_DOWN) {
// Record the position the list the touch landed on
mPosition = pointToPosition((int) ev.getX(), (int) ev.getY());
return super.dispatchTouchEvent(ev);
}
if (actionMasked == MotionEvent.ACTION_MOVE) {
// Ignore move events
return true;
}
if (actionMasked == MotionEvent.ACTION_UP) {
// Check if we are still within the same view
if (pointToPosition((int) ev.getX(), (int) ev.getY()) == mPosition) {
super.dispatchTouchEvent(ev);
} else {
// Clear pressed state, cancel the action
setPressed(false);
invalidate();
return true;
}
}
return super.dispatchTouchEvent(ev);
}
完全なカスタムビュークラスが利用可能です: https://Gist.github.com/danosipov/649849
CustomListView
を作成します
@Override
public boolean onInterceptTouchEvent(MotionEvent ev) {
if(needToStop){
return false;}
return super.onInterceptTouchEvent(ev);
}
false
で、子はtouchイベントを処理します。必ずif condition
スクロールする必要があるかどうかを確認します。
私にとって最良の答えはダン・オシポフからのものです。このように改善します。 (押された状態を手動で消去する代わりに、CANCELアクションイベントを起動します)。
@Override
public boolean dispatchTouchEvent(MotionEvent ev) {
final int actionMasked = ev.getActionMasked() & MotionEvent.ACTION_MASK;
if (actionMasked == MotionEvent.ACTION_DOWN) {
// Record the position the list the touch landed on
mPosition = pointToPosition((int) ev.getX(), (int) ev.getY());
return super.dispatchTouchEvent(ev);
}
if (actionMasked == MotionEvent.ACTION_MOVE) {
// Ignore move events
return true;
}
if (actionMasked == MotionEvent.ACTION_UP) {
// Check if we are still within the same view
if (pointToPosition((int) ev.getX(), (int) ev.getY()) != mPosition) {
// Clear pressed state, cancel the action
ev.setAction(MotionEvent.ACTION_CANCEL);
}
}
return super.dispatchTouchEvent(ev);
}
リストビューで削除するスワイプのコードを書くとき、スワイプが検出されたら垂直スクロールを防止したかった。条件を削除するスワイプが満たされたら、ACTION_MOVEセクションにブールフラグを設定します。 dispatchTouchEventメソッドはその状態をチェックし、スクロールが機能しないようにします。 ACTION_UPで、フラグをfalseに戻し、スクロールを再度有効にします。
private float mY = Float.NaN;
private boolean mStopScroll = false;
@Override
public boolean onTouch(View view, MotionEvent event) {
if(!mStopScroll) {
mY = event.getY();
}
switch (event.getAction()) {
case MotionEvent.ACTION_MOVE:
if(<condition that stops the vertical scroll>) {
mStopScroll = true;
}
break;
case MotionEvent.ACTION_UP:
mStopScroll = false;
// your code here
return true;
default:
}
return false;
}
@Override
public boolean dispatchTouchEvent(MotionEvent motionEvent) {
if(mStopScroll) {
motionEvent.setLocation(motionEvent.getX(), mY);
}
return super.dispatchTouchEvent(motionEvent);
}
これを試して:
listViewObject.setTranscriptMode(0);
私のために働く。
これは http://danosipov.com/?p=604 でポイントされているコードJoe Blow(OP投稿のコメント)ですが、リンクが孤立している場合に備えてここに投稿しています:
public class ScrollDisabledListView extends ListView {
private int mPosition;
public ScrollDisabledListView(Context context) {
super(context);
}
public ScrollDisabledListView(Context context, AttributeSet attrs) {
super(context, attrs);
}
public ScrollDisabledListView(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
}
@Override
public boolean dispatchTouchEvent(MotionEvent ev) {
final int actionMasked = ev.getActionMasked() & MotionEvent.ACTION_MASK;
if (actionMasked == MotionEvent.ACTION_DOWN) {
// Record the position the list the touch landed on
mPosition = pointToPosition((int) ev.getX(), (int) ev.getY());
return super.dispatchTouchEvent(ev);
}
if (actionMasked == MotionEvent.ACTION_MOVE) {
// Ignore move events
return true;
}
if (actionMasked == MotionEvent.ACTION_UP) {
// Check if we are still within the same view
if (pointToPosition((int) ev.getX(), (int) ev.getY()) == mPosition) {
super.dispatchTouchEvent(ev);
} else {
// Clear pressed state, cancel the action
setPressed(false);
invalidate();
return true;
}
}
return super.dispatchTouchEvent(ev);
}
}
このListViewをレイアウトに追加するときは、パッケージ名を前に付けることを忘れないでください。そうしないと、膨張しようとすると例外がスローされます。
私の答えは、XamarinとMvvmCrossのユーザーにとって興味深いものです。主な概念は以前の投稿と同じなので、主な手順は次のとおりです。
リストビューでスクロールを無効にできるヘルパークラスを次に示します。
using System;
using Cirrious.MvvmCross.Binding.Droid.Views;
using Android.Content;
using Android.Util;
using Android.Views;
using Android.Database;
namespace MyProject.Android.Helpers
{
public class UnscrollableMvxListView
: MvxListView
{
private MyObserver _myObserver;
public UnscrollableMvxListView (Context context, IAttributeSet attrs, MvxAdapter adapter)
: base(context, attrs, adapter)
{
}
protected override void OnAttachedToWindow ()
{
base.OnAttachedToWindow ();
var dtso = new MyObserver(this);
_myObserver = dtso;
Adapter.RegisterDataSetObserver (dtso);
}
protected override void OnDetachedFromWindow ()
{
Log.Debug ("UnscrollableMvxListView", "OnDetachedFromWindow");
if (_myObserver != null) {
Adapter.UnregisterDataSetObserver (_myObserver);
_myObserver = null;
}
base.OnDetachedFromWindow ();
}
//Make List Unscrollable
private int _position;
public override bool DispatchTouchEvent (MotionEvent ev)
{
MotionEventActions actionMasked = ev.ActionMasked & MotionEventActions.Mask;
if (actionMasked == MotionEventActions.Down) {
// Record the position the list the touch landed on
_position = PointToPosition((int) ev.GetX (), (int) ev.GetY());
return base.DispatchTouchEvent(ev);
}
if (actionMasked == MotionEventActions.Move) {
// Ignore move events
return true;
}
if (actionMasked == MotionEventActions.Up) {
// Check if we are still within the same view
if (PointToPosition((int) ev.GetX(), (int) ev.GetY()) == _position) {
base.DispatchTouchEvent(ev);
} else {
// Clear pressed state, cancel the action
Pressed = false;
Invalidate();
return true;
}
}
return base.DispatchTouchEvent(ev);
}
//Make List Flat
public void JustifyListViewHeightBasedOnChildren () {
if (Adapter == null) {
return;
}
var vg = this as ViewGroup;
int totalHeight = 0;
for (int i = 0; i < Adapter.Count; i++) {
View listItem = Adapter.GetView(i, null, vg);
listItem.Measure(0, 0);
totalHeight += listItem.MeasuredHeight;
}
ViewGroup.LayoutParams par = LayoutParameters;
par.Height = totalHeight + (DividerHeight * (Adapter.Count - 1));
LayoutParameters = par;
RequestLayout();
}
}
internal class MyObserver
: DataSetObserver
{
private readonly UnscrollableMvxListView _unscrollableMvxListView;
public MyObserver (UnscrollableMvxListView lst)
{
_unscrollableMvxListView = lst;
}
public override void OnChanged() {
Log.Debug("UnscrollableMvxListView", "OnChanged");
base.OnChanged ();
_unscrollableMvxListView.JustifyListViewHeightBasedOnChildren ();
}
}
}