如何在旋转后正确恢复视图状态

时间:2015-10-14 12:52:19

标签: android android-view

我想手动处理complex view的旋转,包括恢复道具的位置和大小。在这一刻,我试图在onLayout中做到这一点(更好的想法是velcome)。有时它运作良好,但通常第一次轮换是错位或没有孩子的情况下绘制视图。

private int oldOrientation = -1;

  /**
   * Override method to configure the dragged view and secondView layout properly.
   */
  @Override protected void onLayout(boolean changed, int left, int top, int right, int bottom) {
    Log.e("mylayout", "onLayout " + df.format(new Date(Calendar.getInstance().getTimeInMillis())));
    if (isInEditMode()) {
      super.onLayout(changed, left, top, right, bottom);
    } else {
      dragView.setVisibility(INVISIBLE);
      if (isDragViewAtTop() && (oldOrientation != getResources().getConfiguration().orientation || oldOrientation == -1)) {
        dragView.layout(left, top, right, transformer.getOriginalHeight());
        secondView.layout(left, transformer.getOriginalHeight(), right, bottom);
        ViewHelper.setY(dragView, top);
        ViewHelper.setY(secondView, transformer.getOriginalHeight());
        ViewHelper.setX(dragView, left);
        ViewHelper.setX(secondView, left);
        oldOrientation = getResources().getConfiguration().orientation;
      } else if (isClosedAtLeft() && (
          oldOrientation != getResources().getConfiguration().orientation
              || oldOrientation == -1)) {
        dragView.layout(left, top, right, transformer.getOriginalHeight());
        secondView.layout(left, transformer.getOriginalHeight(), right, bottom);
        ViewHelper.setY(dragView, top);
        ViewHelper.setY(secondView, transformer.getOriginalHeight());
        ViewHelper.setX(dragView, left);
        ViewHelper.setX(secondView, left);
        closeToLeft();
        oldOrientation = getResources().getConfiguration().orientation;
      } else if (isClosedAtRight() && (
          oldOrientation != getResources().getConfiguration().orientation
              || oldOrientation == -1)) {
        dragView.layout(left, top, right, transformer.getOriginalHeight());
        secondView.layout(left, transformer.getOriginalHeight(), right, bottom);
        ViewHelper.setY(dragView, top);
        ViewHelper.setY(secondView, transformer.getOriginalHeight());
        ViewHelper.setX(dragView, left);
        ViewHelper.setX(secondView, left);
        closeToRight();
        oldOrientation = getResources().getConfiguration().orientation;
      } else if ((oldOrientation != getResources().getConfiguration().orientation
          || oldOrientation == -1)) {
        dragView.layout(left, top, right, transformer.getOriginalHeight());
        secondView.layout(left, transformer.getOriginalHeight(), right, bottom);
        ViewHelper.setY(dragView, top);
        ViewHelper.setY(secondView, transformer.getOriginalHeight());
        ViewHelper.setX(dragView, left);
        ViewHelper.setX(secondView, left);
        smoothSlideTo(SLIDE_BOTTOM);
        oldOrientation = getResources().getConfiguration().orientation;
      }
      dragView.setVisibility(VISIBLE);
    }
  }

在这段代码中,我尝试在旋转后恢复初始stte,当调用onLayout然后将其移动到位时,视图在旋转之前(有4个状态,从creen到左边,从屏幕到右边, scren或右下角)。

编辑:

<?xml version="1.0" encoding="utf-8"?>
<manifest
    package="com.github.pedrovgs.sample"
    xmlns:android="http://schemas.android.com/apk/res/android">

    <uses-sdk android:minSdkVersion="8" />

    <!-- Permissions -->

    <uses-permission android:name="android.permission.INTERNET" />
    <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
    <uses-permission android:name="com.google.android.providers.gsf.permission.READ_GSERVICES" />
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />

    <uses-feature
        android:glEsVersion="0x00020000"
        android:required="true" />

    <!-- Application configuration -->

    <application
        android:name=".DraggablePanelApplication"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme">

        <!-- Maps API KEY -->

        <meta-data
            android:name="com.google.android.maps.v2.API_KEY"
            android:value="AIzaSyC1rMU-mkhoyTvBIdTnYU0dss0tU9vtK48" />

        <!-- Main Activity -->

        <activity
            android:name=".activity.MainActivity"
            android:configChanges="keyboardHidden|orientation|screenSize"
            android:label="@string/app_name">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>

        <!-- Places sample -->

        <activity
            android:name=".activity.PlacesSampleActivity"
            android:configChanges="keyboardHidden|orientation|screenSize"

            android:label="@string/places_sample_activity_title" />

        <!-- TV Shows sample -->

        <activity
            android:name=".activity.TvShowsActivity"
            android:label="@string/tv_shows_sample_activity_title" />


        <!-- Youtube Sample -->

        <activity
            android:name=".activity.YoutubeSampleActivity"
            android:configChanges="keyboardHidden|orientation|screenSize"

            android:label="@string/youtube_sample_activity_title" />


        <!-- Video Sample -->

        <activity
            android:name=".activity.VideoSampleActivity"
            android:configChanges="keyboardHidden|orientation|screenSize"

            android:label="@string/video_sample_activity_title" />

        <meta-data
            android:name="com.google.android.gms.version"
            android:value="@integer/google_play_services_version" />

    </application>

</manifest>

样本活动xml

<?xml version="1.0" encoding="utf-8"?>

<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:draggable_panel="http://schemas.android.com/apk/res-auto"
    android:id="@+id/fl_container"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

  <!-- Movie Thumbnail -->

  <ImageView
      android:id="@+id/iv_thumbnail"
      android:layout_width="fill_parent"
      android:layout_height="fill_parent"
      style="@style/image_view"/>

  <!-- DraggablePanel -->

  <com.github.pedrovgs.DraggablePanel
      android:id="@+id/draggable_panel"
      android:layout_width="fill_parent"
      android:layout_height="fill_parent"
      draggable_panel:x_scale_factor="@dimen/x_scale_factor"
      draggable_panel:y_scale_factor="@dimen/y_scale_factor"
      draggable_panel:top_fragment_height="@dimen/top_fragment_height"
      draggable_panel:top_fragment_margin_right="@dimen/top_fragment_margin"
      draggable_panel:top_fragment_margin_bottom="@dimen/top_fragment_margin"
      draggable_panel:enable_horizontal_alpha_effect="false"/>

</FrameLayout>

5 个答案:

答案 0 :(得分:2)

如果您正在努力深入了解View,那么View Activity android:configChanges="keyboardHidden|orientation|screenSize"

Activity

如果在旋转后设置,则不在乎是否重新创建ImageView因此重新绘制,重新测量,重新定位重新等等。

因此,如果您在轮换男孩后使用ImageView展示位置[10 100]启动您的应用,那么您的{{1}}将被假定为[10 100],有时它甚至可能会给您带来奇怪的屏幕外坐标。

答案 1 :(得分:1)

您扩展了ViewGroup而不是View,但尚未向xml或代码中的child添加任何DraggableView。我在addChild中没有看到任何DraggableView或任何与儿童相关的代码。您使用了topFragmentbottomFragment等等,并使用addFragmentToView()ViewGroup内进行了片段交易。不好主意!

我知道这很费时间。但是你需要退一步思考这个视图的设计。我强烈建议您将这些topFragment等作为DraggableView的子项处理,并在此视图中没有FragmentManager并执行addTransaction

一旦你开始将片段作为孩子添加到你的ViewGroup,许多并发症就会消失。我从过去的经历中说出这一点。 onLayout()用于在Child Views内排列ViewGroup。旋转设备后,将再次调用onLayout(),并根据需要排列子视图。这很简单。一旦开始这样的概念化,拖动操作将变得非常简单。

如果您认为在Framgents中将ViewGroup作为孩子是荒谬的,请查看ViewPager代码here。这使用片段作为输入,但仍然将它们视为布局的子项。

  1. 尝试处理与onLayout()本身的位置相关的所有内容。
  2. 请勿在{{1​​}}中保存位置数据。仅保存此中的onSaveInstanceState()text数据。
  3. 始终建议不要覆盖drawable
  4. 您现在可能无法执行此操作,但从长远来看,您可能希望重构并以此方式执行此操作。

    记住onCofigurationChanged()的真正目的:

      

    当此视图应指定大小和位置时,从布局调用   每个孩子。带子项的派生类应该覆盖   这个方法和他们每个孩子的呼叫布局。

答案 2 :(得分:0)

尝试使用onConfigrationChange()覆盖方法 这个方法将处理旋转。

答案 3 :(得分:0)

假设您正在创建自己的自定义视图。如果是这样,您应该使用View类提供的标准机制。

@Override
protected void onRestoreInstanceState(Parcelable state) {
    // Restore state
}

@Override
protected Parcelable onSaveInstanceState() {
    // Save state
}

答案 4 :(得分:0)

我想这是一个好方法

@Override
    protected Parcelable onSaveInstanceState() {
        Parcelable superState = super.onSaveInstanceState();
        return new YourParcelableObject(superState, yourValues);
    }

    @Override
    protected void onRestoreInstanceState(Parcelable state) {
        YourParcelableObject savedState = (SavedState) state;
        super.onRestoreInstanceState(savedState.getSuperState());

        //restote other values from YourParcelableObject
    }