2

我遇到了一個問題,我知道根本原因,但沒有看到解決問題的方法。如果在活動中多次使用自定義複合組件,則從視圖保存的值將相互覆蓋。爲了更容易解釋,我做了下面的例子。在屏幕上保存狀態在多次使用複合組件時旋轉

新組件的xml,只有一個EditText使它更短。

<?xml version="1.0" encoding="utf-8"?> 
<merge xmlns:android="http://schemas.android.com/apk/res/android" > 

    <EditText 
     android:id="@+id/custom_text" 
     android:layout_width="match_parent" 
     android:layout_height="wrap_content" 
     android:ems="10" 
     android:inputType="number" > 
    </EditText> 

</merge> 

實現新行爲的類只是使佈局膨脹。

public class CustomView extends LinearLayout { 
    public CustomView(Context context) { 
     this(context, null); 
    } 

    public CustomView(Context context, AttributeSet attrs) { 
     this(context, attrs, 0); 
    } 

    public CustomView(Context context, AttributeSet attrs, int defStyle) { 
     super(context, attrs, defStyle); 

     LayoutInflater inflater = (LayoutInflater) context 
       .getSystemService(Context.LAYOUT_INFLATER_SERVICE); 
     inflater.inflate(R.layout.custom_view, this, true); 
    } 
} 

並使用其中的2個佈局。

<?xml version="1.0" encoding="utf-8"?> 
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" 
    android:layout_width="fill_parent" 
    android:layout_height="fill_parent" 
    android:orientation="vertical" > 

    <test.customview.CustomView 
     android:id="@+id/customView1" 
     android:layout_width="fill_parent" 
     android:layout_height="wrap_content" > 
    </test.customview.CustomView> 

    <test.customview.CustomView 
     android:id="@+id/customView2" 
     android:layout_width="fill_parent" 
     android:layout_height="wrap_content" > 
    </test.customview.CustomView> 

</LinearLayout> 

當屏幕旋轉時,第二個視圖的值也會在第一個視圖中恢復。

深入研究框架的代碼我發現從View類中定義的onSaveInstanceState返回的Parcelable對象與關鍵對象的ID放在SparseArray中。因爲我多次包含CustomView,因此ID爲「custom_text」的EditText也會多次添加。具有相同的ID,保存的值將相互覆蓋。

我在尋找任何有關如何實際應用的建議。現在,我看不出有什麼辦法來改變這些標識符。

回答

2

似乎我有這個問題的一些解決方案。我試圖找到它一段時間。

1.首先你必須創建內部類,它擴展了你的CustomView中的BaseSavedState。

CustomView{ 
    String value; //some text value from edittext 

    EditText edittext; 
     ... 

    private static class Save extends BaseSavedState{ 

    String savedValue; 

    public Save(Parcel incoming) { 
     super(incoming); 
     savedValue = incoming.readString(); 
     Log.i("Save", "Parcel"); 
    } 

    public Save(Parcelable parcelable) { 
     super(parcelable); 
     Log.i("Save", "Parcelable"); 
    } 

    @Override 
    public void writeToParcel(Parcel outcoming, int flags) { 
     super.writeToParcel(outcoming, flags); 
     outcoming.writeString(savedValue); 
     Log.i("Save", "writeToParcel"); 
    } 

    public static final Parcelable.Creator<Save> CREATOR = 
       new Creator<CustomView.Save>() { 

     @Override 
     public Save[] newArray(int size) { 
      Log.i("Parcelable.Creator<Save>", "newArray"); 
      return new Save[size]; 
     } 

     @Override 
     public Save createFromParcel(Parcel incoming) { 
      Log.i("Parcelable.Creator<Save>", "createFromParcel"); 
      return new Save(incoming); 
     } 
    }; 
} 

} 

2.然後重寫此兩個方法中CustomView

CustomView{ 
    String value; //some text value from edittext 

    EditText edittext; 

    ... 

    @Override 
protected Parcelable onSaveInstanceState() { 
    Log.i("CustomView", "onSaveInstanceState"); 

    Parcelable p = super.onSaveInstanceState(); 

    Save save = new Save(p); 
    save.savedValue = value; // value is from CustomView class 

    return save; 
} 

@Override 
protected void onRestoreInstanceState(Parcelable state) { 
    Log.i("CustomView", "onRestoreInstanceState"); 
    if(!(state instanceof Save)){ 
     super.onRestoreInstanceState(state); 
     return; 
    } 

    Save save = (Save) state; 
    value = save.savedValue; 
    //setting in this place value to edittext will not do anything. 
      //instead, you have to do this in step 3 
    super.onRestoreInstanceState(save.getSuperState()); 
} 

    ... 
} 

3.override onAttachedToWindow()和設置爲EDITTEXT 「值」。

CustomView{ 
    String value; //some text value from edittext 

    EditText edittext; 
    ... 
    @Override 
    protected void onAttachedToWindow() { 
      edittext.setText(value); 
      super.onAttachedToWindow(); 
    } 
    ... 
} 

,現在你可以有拒絕改變方向自定義視圖的多個實例 - 他們將擁有正確values.I沒有測試在100%這個解決方案,但它似乎也不錯。

+0

謝謝你,你的代碼正在工作:)我花了一段時間才明白爲什麼,但最終我得到了它。不要擔心你的英語,這很好。歡迎來到StackOverflow :) – azertiti 2012-04-03 22:20:51

相關問題