我只是完成数字首选项验证例程的一小步
我替换了ClickListener上的EditTextPreference OK按钮,以防止在条目不是数字的情况下离开对话框
我使用TextWatcher.afterTextChanged来检查条目是否是数字,并将有效的评估放在EditText标记上
一切正常,输入无效时按OK不会离开对话框,输入有效时按OK。。。这里我需要调用Click上的原始OK按钮,但我没有找到如何执行。有一些View函数callOnClick()和performOnClick
@Override
public void onCreate(Bundle savedInstanceState) {
Log.d(TAG, "+ onCreate(savedInstanceState:" + savedInstanceState + ")");
super.onCreate(savedInstanceState);
...
prefMaxLogs = (EditTextPreference) findPreference(getText(R.string.pref_maxLogs_key));
prefMaxLogsEt = prefMaxLogs.getEditText();
prefMaxLogsEt.setSingleLine();
prefMaxLogsEt.setOnFocusChangeListener(ofcl);
prefMaxLogsEt.addTextChangedListener(tw);
...
Log.d(TAG, "- onCreate()");
}
private OnFocusChangeListener ofcl = new OnFocusChangeListener() {
@Override
public void onFocusChange(View v, boolean hasFocus) {
Log.d(TAG, "+ onFocusChange(v:" + v + ", hasFocus:" + hasFocus + ")");
if( hasFocus ) {
prefMaxLogsEt.selectAll();
Dialog dialog = prefMaxLogs.getDialog();
if( dialog != null ) {
dialog.findViewById(android.R.id.button1).setOnClickListener(ocl);
}
}
Log.d(TAG, "- onFocusChange()");
}
};
private OnClickListener ocl = new OnClickListener() {
@Override
public void onClick(View v) {
Log.d(TAG, "+ onClick(v:" + v + ")");
Boolean valid = !(Boolean) prefMaxLogsEt.getTag(R.string.invalidEntry);
if( valid ) {
Log.d(TAG, "Valid et");
}
Log.d(TAG, "- onClick()");
}
};
private TextWatcher tw = new TextWatcher() {
@Override
public void beforeTextChanged(CharSequence s, int start, int count, int after) {}
@Override
public void onTextChanged(CharSequence s, int start, int before, int count) {}
@Override
public void afterTextChanged(Editable s) {
Log.d(TAG, "+ afterTextChanged(s:" + s + ")");
String source = s.toString();
//prefMaxLogsEt.removeTextChangedListener(this);
if( !source.matches("^[0-9]+$") ) {
prefMaxLogsEt.getBackground().setColorFilter(getResources().getColor(R.color.invalid), Mode.SRC_IN);
prefMaxLogsEt.invalidate();
prefMaxLogsEt.setError(getText(R.string.invalidEntry));
prefMaxLogsEt.selectAll();
prefMaxLogsEt.setTag(R.string.invalidEntry, true);
} else {
prefMaxLogsEt.setError(null);
prefMaxLogsEt.getBackground().clearColorFilter();
prefMaxLogsEt.invalidate();
prefMaxLogsEt.setTag(R.string.invalidEntry, false);
}
//prefMaxLogsEt.addTextChangedListener(this);
Log.d(TAG, "- afterTextChanged()");
}
};
onClickListener是在EditText获得焦点时设置的,因为我知道对话框已经显示了
onClickListener防止对话框使用无效条目
我找到了一种不同的方法来验证EditTextPreference。设置OnPreferenceChangeListener并实现onPreferenceChange。
@Override
public boolean onPreferenceChange(Preference preference, Object newValue) {
Log.d(TAG, "+ onPreferenceChange(preference:" + preference + ", newValue:" + newValue + ")");
Boolean rv = true;
if( preference.equals(prefMaxLogs) ) {
String source = newValue.toString();
if( !source.matches("^[0-9]+$") ) {
rv = false;
}
}
Log.d(TAG, "- onPreferenceChange()");
return rv;
}
这个函数的优点是它返回一个布尔值,如果您接受更改,则返回true,如果您拒绝更改,则为false
我为OK按钮删除了onFocusListener和onClickListener,这种方式更简单
我仍然使用setColorFilter和afterTextChanged来警告用户无效的条目。如果用户按"确定",对话框将关闭,但只接受有效值。
这个监听器是为每个首选项单独设置的,所以这是我在onCreate()中的初始化
prefMaxLogs = (EditTextPreference) findPreference(getText(R.string.pref_maxLogs_key));
prefMaxLogsEt = prefMaxLogs.getEditText();
prefMaxLogsEt.setSingleLine();
prefMaxLogsEt.addTextChangedListener(tcl);
prefMaxLogs.setOnPreferenceClickListener(this);