перетаскивание в Android 3.x вызывает FormalStateException после небольшого числа при перетаскивании - PullRequest
14 голосов
/ 05 марта 2012

есть проблема с механизмом перетаскивания в Android 3.x: после некоторых перетаскиваний (скажем, 30 перетаскиваний) возникает исключение (см. Прикрепленную ссылку)

https://groups.google.com/forum/#!msg/android-platform/2APvO248NNY/rKI-5dCT8XcJ (I 'я получаю в журнале то же самое, что и прикрепленный к этому сообщению ..)

техник по Android отвечает там, что это ошибка в API, и говорит, что единственный способ избежать этой проблемы - вызвать сборщик мусора.

Я сделал это.исключение больше не генерируется, но через некоторое время (скажем, 30-40 перетаскиваний) Android перестает вызывать событие сброса по какой-то причине.

Я попытался «обновить» все представления, освободив все ресурсы / canvas /рисовать кэш / перерабатывать растровые изображения и заново создавать их, и это не помогло (больше не генерировало исключение - но все же после некоторых перетаскиваний событие сброса не работает)

единственное, что «помогает»"закрыть действие и перезапустить его снова.

кто-нибудь решил эту проблему как-нибудь, или есть хорошая простая альтернатива ???(помимо реализации моей собственной функции перетаскивания ..)

Я хотел бы получить решение, которое не заставит меня перезапускать или заново создавать что-либо, что не предполагает ..

вот пример кода , который демонстрирует ошибку (не демонстрирует ту часть, которую я сказал о проблеме с событием drop после использования System.GC):

public class DragandDropExampleActivity extends Activity {

private boolean mIsBeenDragged = false;

@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.main);

    final ImageView imageViewToDRag = (ImageView) findViewById(R.id.image_view_to_drag);

    imageViewToDRag.setClickable(true);

    imageViewToDRag.setOnTouchListener(new OnTouchListener() {

        @Override
        public boolean onTouch(View v, MotionEvent event) {

            if (event.getAction() == MotionEvent.ACTION_DOWN) {
                mIsBeenDragged = true;
                DragShadowBuilder shadowBuilder = new DragShadowBuilder(imageViewToDRag);
                imageViewToDRag.startDrag(null, shadowBuilder, imageViewToDRag, 0);
            } else if (event.getAction() == MotionEvent.ACTION_UP) {

                mIsBeenDragged = false; 
            }
            return false;
        }
    });

}
}

этоxml:

<?xml version="1.0" encoding="utf-8"?>
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/main_frame"
android:layout_width="fill_parent"
android:layout_height="fill_parent" >

<ImageView
    android:id="@+id/image_view_to_drag"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:src="@drawable/ic_launcher" >

</ImageView>

это трассировка стека:

06-04 13:34:32.730: E/View(8061):
java.lang.IllegalArgumentException
    at android.view.Surface.lockCanvasNative(Native Method)
    at android.view.Surface.lockCanvas(Surface.java:350)
    at android.view.View.startDrag(View.java:11489)
    at com.show.dragandrop.DragandDropExampleActivity$1.onTouch(DragandDropExampleActivity.java:32)
    at android.view.View.dispatchTouchEvent(View.java:4617)
    at android.view.ViewGroup.dispatchTransformedTouchEvent(ViewGroup.java:1560)
    at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:1291)
    at android.view.ViewGroup.dispatchTransformedTouchEvent(ViewGroup.java:1560)
    at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:1291)
    at android.view.ViewGroup.dispatchTransformedTouchEvent(ViewGroup.java:1560)
    at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:1291)
    at android.view.ViewGroup.dispatchTransformedTouchEvent(ViewGroup.java:1560)
    at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:1291)
    at com.android.internal.policy.impl.PhoneWindow$DecorView.superDispatchTouchEvent(PhoneWindow.java: 1862)
    at com.android.internal.policy.impl.PhoneWindow.superDispatchTouchEvent(PhoneWindow.java:1286)
    at android.app.Activity.dispatchTouchEvent(Activity.java:2315)
    at com.android.internal.policy.impl.PhoneWindow$DecorView.dispatchTouchEvent(PhoneWindow.java:1835)
    at android.view.View.dispatchPointerEvent(View.java:4689)
    at android.view.ViewRoot.deliverPointerEvent(ViewRoot.java:2415)
    at android.view.ViewRoot.handleMessage(ViewRoot.java:2077)
    at android.os.Handler.dispatchMessage(Handler.java:99)
    at android.os.Looper.loop(Looper.java:132)
    at android.app.ActivityThread.main(ActivityThread.java:4126)
    at java.lang.reflect.Method.invokeNative(Native Method)
    at java.lang.reflect.Method.invoke(Method.java:491)
    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:844)
    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:602)
    at dalvik.system.NativeStart.main(Native Method)

, чтобы получить исключение - просто перетащите изображение в некоторую точку наэкран и оставить палец.повторить это ровно 30 раз, и исключение выдается.Я сделал этот очень простой пример, чтобы продемонстрировать, что исключение выдается без каких-либо издержек, вызванных моим приложением.

TIA

Ответы [ 5 ]

1 голос
/ 20 июля 2012

Попробуйте это: -

private OnTouchListener drag = new OnTouchListener() {

    @Override
    public boolean onTouch(View v, MotionEvent event) {
        if (event.getAction() == MotionEvent.ACTION_MOVE) {
            setViewPosition(v, Math.round(event.getRawX()),
                    Math.round(event.getRawY()));
        }
        return false;
    }
};


private void setViewPosition(View v, int x, int y) {
    if (v != null) {
        LayoutParams params = new LayoutParams(LayoutParams.WRAP_CONTENT,
                LayoutParams.WRAP_CONTENT);
        params.leftMargin = (x - v.getMeasuredWidth() / 2);
        params.topMargin = (y - v.getMeasuredHeight() / 2);
        v.setLayoutParams(params);
        v.invalidate();
    }
}
0 голосов
/ 12 января 2013

прошло много времени с тех пор, как я задал этот вопрос. Когда был выпущен Android ICS - эта ошибка больше не существует. так что с Android ICS действительно нет причин писать приложения, ориентированные на Honeycomb. В любом случае, только 0,3% от общего числа устройств Android, работающих под управлением этой версии ОС,

так что для меня ответ - просто не использовать соты! : ->

0 голосов
/ 04 июня 2012

Попробуйте:

imageViewToDRag.startDrag(null, shadowBuilder, imageViewToDRag, 0);
System.gc();

У меня сработало на Android 3.2!

0 голосов
/ 04 июня 2012

Из комментария к вопросу , касающемуся проблемы, которую вы опубликовали в обсуждении групп Google по поводу:

Эта проблема возникает из-за того, что GC не выполняет сбордраг.Но если окно перерисовывается, GC выполняет сбор драг.Окно будет перерисовано после того, как будет открыта программная клавиатура или обновлено некоторое ImageView или что-то в этом роде.

Похоже, вы можете заставить сборщик мусора собирать перетаскиваемые объекты, принудительно перерисовывая содержащую View.У меня нет удобного устройства 3.0, но, возможно, что-то подобное стоит попробовать:

        } else if (event.getAction() == MotionEvent.ACTION_UP) {
           findViewById(R.id.main_frame).invalidate();
        }

Возможно, вам также понадобится подсказка по сбору мусора:

        } else if (event.getAction() == MotionEvent.ACTION_UP) {
           findViewById(R.id.main_frame).invalidate();
           System.gc();
        }

но я думаю, что вы правы, если хотите, хотите избежать этого, поэтому я бы попробовал это сделать, только если это не сработает без него.

Кроме того, я думаю, что onTouch() должно вернуть true, так как вы использовали событие касания.

0 голосов
/ 14 марта 2012

Если перезапуск занятия помогает, попробуйте перезапустить представление, удалив его и добавив снова (возможно, это новый экземпляр).Также вы можете попробовать штатный персонал (который обычно не помогает), например, invalidate (), переопределить onDraw () и посмотреть, почему он вызывается или не вызывается.

...