1

我有一个CustomButton类 (extends LinearLayout),在其中我膨胀了一个包含 a 的布局ToggleButton(实际上这更复杂,但我在这里简化了问题)。

public class CustomButton extends LinearLayout {

    private ToggleButton toggleOnOffButton;

    public CustomButton(Context context, AttributeSet attrs) {
        super(context, attrs);
        LayoutInflater.from(context).inflate(R.layout.custom_button_layout, this);
    }

    @Override
    protected void onFinishInflate() {
        toggleOnOffButton = (ToggleButton) findViewById(R.id.toggle_on_off_button);
        super.onFinishInflate();
    }

    public ToggleButton getToggleOnOffButton() {
        return toggleOnOffButton;
    }
}

custom_button_layout.xml

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
                android:layout_width="fill_parent"
                android:layout_height="wrap_content">

    <ToggleButton android:id="@+id/toggle_on_off_button"
                  android:layout_width="wrap_content"
                  android:layout_height="wrap_content"
                  android:textOff="Off"
                  android:textOn="On"
                  android:layout_alignParentRight="true"
            />
</RelativeLayout>

我有一个活动,我用 2 CustomButton-s 为布局充气。第一个 toggleButton 的开/关状态保存在共享首选项中,我在onCreate方法中从那里加载值。

public class FirstActivity extends Activity
{
    private CustomButton customButton;
    private ToggleButton toggleBut;

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

        customButton = (CustomButton)findViewById(R.id.toggleButton);
        toggleBut = customButton.getToggleOnOffButton();

        boolean saved = loadPreferences("toggleBut");
        toggleBut.setChecked(saved);
        toggleBut.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                boolean checked = toggleBut.isChecked();
                savePreferences("toggleBut", checked);
            }
        });
    }

    private void savePreferences(String key, boolean value){
        SharedPreferences sharedPreferences = getPreferences(MODE_PRIVATE);
        SharedPreferences.Editor editor = sharedPreferences.edit();
        editor.putBoolean(key, value);
        editor.commit();
    }

    private boolean loadPreferences(String key){
        SharedPreferences sharedPreferences = getPreferences(MODE_PRIVATE);
        return sharedPreferences.getBoolean(key, true);
    }
}

主要的.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
              android:orientation="vertical"
              android:layout_width="fill_parent"
              android:layout_height="fill_parent"> 
    <com.example.example.cs.ssd.custom.CustomButton
            android:id="@+id/toggleButton"
            android:layout_width="fill_parent"
            android:layout_height="wrap_content"
            />

    <com.example.example.cs.ssd.custom.CustomButton
            android:id="@+id/toggleButton2"
            android:layout_width="fill_parent"
            android:layout_height="wrap_content"
            />
</LinearLayout>

当我启动应用程序时,第一个 toggleButton 为 ON。当我更改屏幕的方向时,第一个 toggleButton 自动变为 Off,甚至saved具有值true并被调用toggleBut.setChecked(saved);,我认为这与CutomButton我创建的有关,因为如果main.xml布局仅包含 1 CustomButton,则此问题不会重现。我不确定我做错了什么......这是带有上述代码的存档(作为一个项目):存档

4

1 回答 1

6

如果您希望 CustomButton 在方向更改后保持其当前状态,只需覆盖 onSaveInstanceState() 和 onRestoreInstanceState()。

一个办法

我浏览了您的代码并注意到toggleBut在 onActivityCreated() 之后但在 onStart() 之前的状态正在更改。为了避免这些方法中的任何一个覆盖您的切换设置,我只是将这些行从 onViewCreated() 中移出:

boolean saved = loadPreferences("toggleBut");
toggleBut.setChecked(saved);

并将它们放入 onResume() 中。希望有帮助!

更好的解决方案

当系统尝试恢复默认值时,您的 ToggleButton 设置将被覆盖saveInstanceState,可能在 Fragment.onActivityCreated() 中。

在 CustomButton 中,像这样覆盖这些函数:

@Override
protected Parcelable onSaveInstanceState() {
    Bundle state = new Bundle();
    state.putParcelable("default", super.onSaveInstanceState());
    state.putParcelable("toggle", toggleOnOffButton.onSaveInstanceState());
    return state;
}

@Override
protected void onRestoreInstanceState(Parcelable state) {
    Bundle bundle = (Bundle) state;
    super.onRestoreInstanceState(bundle.getParcelable("default"));
    toggleOnOffButton.onRestoreInstanceState(bundle.getParcelable("toggle"));
};

了解系统仍会更改 ToggleButton 状态,而无需多做一件事。但让我试着解释一下发生了什么:

  • onActivityCreated(Bundle savedInstanceState)savedInstanceState通过调用“onRestoreInstanceState(Bundle savedInstanceState)”将它传递给每个布局元素。
  • onRestoreInstanceState() 首先从布局的根元素开始并向上遍历布局的层次结构(在这种情况下,它最后设置每个 ToggleButton 的选中状态)。
  • 由于默认方法显然不起作用,我们需要为 ToggleButtons 定义我们自己的保存/恢复方法。否则我们在系统调用 onRestoreInstanceState() 之前所做的任何更改都将被系统再次更改...

因此,最后我们将通过在 CustomButton.onFinishInflate() 中添加以下行来从默认行为中排除 ToggleButtons:

toggleOnOffButton.setSaveEnabled(false);

瞧,您的 CustomButtons 会自动保留它们的状态。

于 2012-06-26T18:47:13.477 回答