J'ai une vue de liste et selon une logique, je souhaite désactiver temporairement le défilement. view.setOnScrollListener (null); ne m'aide pas, je suppose que j'ai besoin d'écrire du code, quelqu'un peut-il me donner un hist ou un extrait?
Merci
Dans votre CustomListView:
@Override
public boolean dispatchTouchEvent(MotionEvent ev){
if(ev.getAction()==MotionEvent.ACTION_MOVE)
return true;
return super.dispatchTouchEvent(ev);
}
ListView réagira alors aux clics, mais ne changera pas la position du défilement.
Une autre option sans créer un nouveau ListView personnalisé serait d'attacher un onTouchListener
à votre ListView et de renvoyer true dans la fonction de rappel onTouch()
si l'action d'événement de mouvement est ACTION_MOVE
.
listView.setOnTouchListener(new OnTouchListener() {
public boolean onTouch(View v, MotionEvent event) {
return (event.getAction() == MotionEvent.ACTION_MOVE);
}
});
utilisez listview.setEnabled (false) pour désactiver le défilement listview
Remarque: cela désactive également les sélections de lignes
Si vous avez un événement lié aux éléments de la liste, faire glisser la liste avec l'une de ces solutions déclenchera toujours l'événement. Vous souhaitez utiliser la méthode suivante pour tenir compte des attentes des utilisateurs d'annuler l'événement en les éloignant de l'élément sélectionné (adapté de la réponse de Pointer Null):
@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);
}
Une classe de vue personnalisée complète est disponible: https://Gist.github.com/danosipov/649849
faites votre CustomListView
@Override
public boolean onInterceptTouchEvent(MotionEvent ev) {
if(needToStop){
return false;}
return super.onInterceptTouchEvent(ev);
}
sur false
les enfants géreront l'événement tactile, assurez-vous de mettre votre if condition
pour vérifier que vous devez faire défiler ou non.
La meilleure réponse pour moi est celle de Dan Osipov. Je l'améliorerais comme ça. (déclencher un événement d'action CANCEL au lieu d'effacer manuellement l'état appuyé).
@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);
}
Lors de l'écriture du code de balayage à supprimer sur une vue de liste, je voulais empêcher le défilement vertical une fois le balayage détecté. J'ai défini un indicateur booléen dans la section ACTION_MOVE une fois que les conditions de balayage pour supprimer ont été remplies. La méthode dispatchTouchEvent vérifie cette condition et empêche le défilement de fonctionner. Dans ACTION_UP, je remets le drapeau sur false pour réactiver le défilement.
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);
}
Essaye ça:
listViewObject.setTranscriptMode(0);
Travaille pour moi.
C'est le code Joe Blow (commentaire sur le post OP) indiqué sur http://danosipov.com/?p=604 mais je le poste ici pour le conserver au cas où le lien serait orphelin :
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);
}
}
Lorsque vous ajoutez ce ListView à votre mise en page, n'oubliez pas de le faire précéder de son nom de package, sinon une exception sera levée lorsque vous essayez de le gonfler.
Ma réponse sera intéressante pour les utilisateurs de Xamarin et MvvmCross. Le concept principal est le même que dans les articles précédents, les principales étapes sont donc les suivantes:
Ici, vous êtes une classe d'assistance, qui permet de désactiver le défilement dans la vue liste:
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 ();
}
}
}