Я пытался создать NumberPicker, который удовлетворял бы мои потребности, но я наткнулся на что-то, и я не понимаю, что это такое.
Поведение простое, у меня есть три средства выбора чисел, и каждое из них может иметь значение от -15 до 15; когда пользователь нажимает кнопку Ok, выбранные значения, если они допустимы, сохраняются в "структуре" и в SharedPreferences для удобства.
Проблема показана на этом снимке экрана.
Когда я сохраняю число, отличное от -1, оно работает без проблем. Если я сохраню -1, результат будет таким же, как указано выше (это происходит в каждом средстве выбора чисел).
Несмотря на то, что отображается -15, фактическое значение, которое сохраняется, равно -1: выше и ниже -15 значения верны, если я сдвину один пиксель, значение станет -1, в OnClick анализируемое значение равно -1, а значение, возвращаемое
value < 0 ? value + maxValue + 1 : value - 1
в setPickers() правильно (29, последний индекс массива).
Я попытался перенести различную инициализацию в разные части кода и заставил ее работать на трех устройствах; два физических (с Android 5.0.2 и 5.1.1) и один, эмулированный с помощью Android Studio (с использованием образа KitKat), и проблема проявляется всегда.
Теперь я начинаю думать, что либо я что-то упускаю действительно простой или полная противоположность.
Спасибо за любую помощь, которую вы можете предоставить.
.java:
public class CoefficientsDialog extends DialogFragment {
private Activity activity;
private MyView myView;
private AlertDialog alertDialog;
private AlertDialog.Builder builder;
private NumberPicker startPicker, endPicker, stepPicker;
final private String C = "C_Coefficient";
final private String Gamma = "Gamma_Coefficient";
private String cStartKey;
private String cEndKey;
private String cStepKey;
private String gStartKey;
private String gEndKey;
private String gStepKey;
String[] defaultValues = new String[]{
"1", "2", "3", "4", "5", "6", "7", "8", "9", "10", "11",
"12", "13", "14", "15", "-15", "-14", "-13", "-12", "-11",
"-10", "-9", "-8", "-7", "-6", "-5", "-4", "-3", "-2", "-1"
};
final int minValue = 0;
final int maxValue = defaultValues.length - 1;
private SharedPreferences settings;
@Override
public Dialog onCreateDialog(Bundle savedInstanceState) {
final String tag = getTag();
cStartKey = getString(R.string.c_start_key);
cEndKey = getString(R.string.c_end_key);
cStepKey = getString(R.string.c_step_key);
gStartKey = getString(R.string.g_start_key);
gEndKey = getString(R.string.g_end_key);
gStepKey = getString(R.string.g_step_key);
activity = getActivity();
settings = PreferenceManager.getDefaultSharedPreferences(activity);
builder = new AlertDialog.Builder(activity);
myView = new MyView(activity);
setPickers();
switch (tag) {
case (C):
String cTitle = "C coefficient range";
init(cTitle, tag);
break;
case (Gamma):
String gTitle = "Gamma coefficient range";
init(gTitle, tag);
break;
default:
dismiss();
break;
}
return alertDialog;
}
private void init(String title, final String tag) {
builder
.setView(myView)
.setTitle(title)
.setPositiveButton("Ok", null)
.setNegativeButton("Back", null);
alertDialog = builder.create();
alertDialog.setOnShowListener(new DialogInterface.OnShowListener() {
@Override
public void onShow(final DialogInterface dialog) {
Button button = alertDialog.getButton(AlertDialog.BUTTON_POSITIVE);
button.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
int start = startPicker.getValue();
int end = endPicker.getValue();
int step = stepPicker.getValue();
int middle = defaultValues.length / 2;
// The index is different from the actual value, method valid only if the
// array is antisymmetric
start = start <= middle ? start + 1 : start - maxValue - 1;
end = end <= middle ? end + 1 : end - maxValue - 1;
step = step <= middle ? step + 1 : step - maxValue - 1;
// If true, the user is trying to go from a positive value to a negative
// value with a positive step or vice versa
if ((end - start) / step < 0)
Toast.makeText(activity, R.string.wrong_direction, Toast.LENGTH_SHORT)
.show();
// If true, start + (n * step) isn't equal to end given any value of n
else if ((end - start) % step != 0)
Toast.makeText(activity, R.string.step_inadequate, Toast.LENGTH_SHORT)
.show();
else if (end == start)
Toast.makeText(activity, R.string.one_value_validation,
Toast.LENGTH_SHORT)
.show();
else {
savePreferences(tag, start, end, step);
alertDialog.dismiss();
}
}
});
}
});
}
private void savePreferences(final String tag, int start, int end, int step) {
SharedPreferences.Editor editor = settings.edit();
switch (tag) {
case (C): {
ModelingStructure.cStart = start;
ModelingStructure.cEnd = end;
ModelingStructure.cStep = step;
editor.putInt(cStartKey, start);
editor.putInt(cEndKey, end);
editor.putInt(cStepKey, step);
editor.apply();
break;
}
case (Gamma): {
ModelingStructure.gStart = start;
ModelingStructure.gEnd = end;
ModelingStructure.gStep = step;
editor.putInt(gStartKey, start);
editor.putInt(gEndKey, end);
editor.putInt(gStepKey, step);
editor.apply();
break;
}
}
}
private void setPickers() {
// Initialize startPicker
startPicker = (NumberPicker) myView.findViewById(R.id.cStartPicker);
startPicker.setMaxValue(maxValue);
startPicker.setMinValue(minValue);
startPicker.setDisplayedValues(defaultValues);
// Initialize endPicker
endPicker = (NumberPicker) myView.findViewById(R.id.cEndPicker);
endPicker.setMinValue(minValue);
endPicker.setMaxValue(maxValue);
endPicker.setDisplayedValues(defaultValues);
// Initialize stepPicker
stepPicker = (NumberPicker) myView.findViewById(R.id.cStepPicker);
stepPicker.setMinValue(minValue);
stepPicker.setMaxValue(maxValue);
stepPicker.setDisplayedValues(defaultValues);
// Check if the pickers were previously changed
int start = settings.getInt(cStartKey, 1);
int end = settings.getInt(cEndKey, 1);
int step = settings.getInt(cStepKey, 1);
// The index is different from the actual value
startPicker.setValue(start < 0 ? start + maxValue + 1 : start - 1);
endPicker.setValue (end < 0 ? end + maxValue + 1 : end - 1);
stepPicker.setValue (step < 0 ? step + maxValue + 1 : step - 1);
}
public void showDialog() {
alertDialog.show();
}
[...]
}
.xml:
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="horizontal" android:layout_width="match_parent"
android:layout_height="match_parent"
android:layout_gravity="center">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/c_start"
android:id="@+id/cStartView"
android:layout_alignStart="@+id/cStartPicker"
android:layout_alignEnd="@+id/cStartPicker"
android:gravity="center_horizontal" />
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/c_end"
android:id="@+id/cEndView"
android:layout_alignStart="@+id/cEndPicker"
android:layout_alignEnd="@+id/cEndPicker"
android:gravity="center_horizontal" />
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/c_step"
android:id="@+id/cStepView"
android:gravity="center_horizontal"
android:layout_alignStart="@+id/cStepPicker"
android:layout_alignEnd="@+id/cStepPicker" />
<NumberPicker
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:id="@+id/cEndPicker"
android:layout_centerInParent="true" />
<NumberPicker
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:id="@+id/cStartPicker"
android:layout_centerInParent="true"
android:layout_toStartOf="@id/cEndPicker"
android:layout_marginEnd="10dp" />
<NumberPicker
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:id="@+id/cStepPicker"
android:layout_centerInParent="true"
android:layout_toEndOf="@id/cEndPicker"
android:layout_marginStart="10dp" />
</RelativeLayout>