5

我正在使用两个NumberPicker在我的应用程序中显示省(mProvincePicker)和城市(mCityPicker)数据。当用户更改省数据时,城市数据也应相应更改。我重新设置了中的mCityPicker数据NumberPicker.onValueChange(NumberPicker picker, int oldVal, int newVal)。但它不能很好地工作并且java.lang.ArrayIndexOutOfBoundsException应用程序崩溃。

这是我的代码:

public class AreaPickerDialog extends Dialog implements OnValueChangeListener, OnClickListener {
    static final String TAG = "AreaPickerDialog";

    private NumberPicker mProvincePicker;
    private NumberPicker mCityPicker;
    private Button mCancelBtn;
    private Button mOKBtn;

    private AreaUtil mAreaUtil;
    private List<Area> mProvinces;
    private int mAreaId;
    private int mProvinceId;
    private Handler mHandler;

    public AreaPickerDialog(Context context, Handler handler, int areaId) {
        super(context);
        mAreaUtil = AreaUtil.getInstance();
        mProvinces = mAreaUtil.getProvinceList();
        mAreaId = areaId;
        mProvinceId = mAreaUtil.getProvinceIdByAreaId(areaId);
        mHandler = handler;
    }

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.dialog_area_picker);
        mProvincePicker = (NumberPicker) findViewById(R.id.np_province);
        mCityPicker = (NumberPicker) findViewById(R.id.np_city);
        mCancelBtn = (Button) findViewById(R.id.btn_cancel);
        mOKBtn = (Button) findViewById(R.id.btn_ok);

        initProvinceData();

        mProvincePicker.setOnValueChangedListener(this);
        mCityPicker.setOnValueChangedListener(this);
        mCancelBtn.setOnClickListener(this);
        mOKBtn.setOnClickListener(this);
    }

    private void initProvinceData() {
        if (mProvinces != null && mProvinces.size() > 0) {
            String[] provinceNames = new String[mProvinces.size()];
            for (int i = 0; i < provinceNames.length; i++) {
                provinceNames[i] = mProvinces.get(i).getProvincename();
            }
            mProvincePicker.setDisplayedValues(provinceNames );
            mProvincePicker.setMinValue(1);
            mProvincePicker.setMaxValue(mProvinces.size());
        }
        mProvincePicker.setValue(mProvinceId);
        initCityData(mProvincePicker.getValue());
    }

    private void initCityData(int provinceId) {
        List<Arealist> cities = mAreaUtil.getCityListOfProvince(provinceId);
        if (cities != null && cities.size() > 0) {
            try {
                int min = Integer.parseInt(cities.get(0).getAreaid());
                int max = Integer.parseInt(cities.get(cities.size() -1).getAreaid());
                String[] cityNames = new String[cities.size()];
                for (int i = 0; i < cityNames.length; i++) {
                    cityNames[i] = cities.get(i).getName();
                }
                mCityPicker.setValue(0);
                mCityPicker.setDisplayedValues(cityNames);
                mCityPicker.setMinValue(min);
                mCityPicker.setMaxValue(max);
            } catch (NumberFormatException e) {
                ILog.e(TAG, e.getMessage(), e);
            }
        }
    }

    @Override
    public void onValueChange(NumberPicker picker, int oldVal, int newVal) {
        if (mProvincePicker.equals(picker)) {
            initCityData(mProvincePicker.getValue());
        } else if (mCityPicker.equals(picker)) {
            mAreaId = mCityPicker.getValue();
        }
    }

    @Override
    public void onClick(View v) {
        switch (v.getId()) {
        case R.id.btn_cancel:
            dismiss();
            break;

        case R.id.btn_ok:
            Message msg = mHandler.obtainMessage(MineActivity.MSG_UPDATE_AREA);
            msg.arg1 = mAreaId;
            msg.sendToTarget();
            dismiss();
            break;

        default:
            break;
        }
    }
} 

和 logcat 错误跟踪:

12-30 23:09:26.560: E/InputEventReceiver(5620): Exception dispatching input event.
12-30 23:09:26.560: W/dalvikvm(5620): threadid=1: thread exiting with uncaught exception (group=0x41ddf438)
12-30 23:09:26.580: E/AndroidRuntime(5620): FATAL EXCEPTION: main
12-30 23:09:26.580: E/AndroidRuntime(5620): java.lang.ArrayIndexOutOfBoundsException: length=14; index=15
12-30 23:09:26.580: E/AndroidRuntime(5620):     at net.simonvt.numberpicker.NumberPicker.updateInputTextView(NumberPicker.java:1840)
12-30 23:09:26.580: E/AndroidRuntime(5620):     at net.simonvt.numberpicker.NumberPicker.setDisplayedValues(NumberPicker.java:1423)
12-30 23:09:26.580: E/AndroidRuntime(5620):     at com.meishai.app.dialog.AreaPickerDialog.initCityData(AreaPickerDialog.java:88)
12-30 23:09:26.580: E/AndroidRuntime(5620):     at com.meishai.app.dialog.AreaPickerDialog.onValueChange(AreaPickerDialog.java:100)
12-30 23:09:26.580: E/AndroidRuntime(5620):     at net.simonvt.numberpicker.NumberPicker.notifyChange(NumberPicker.java:1855)
12-30 23:09:26.580: E/AndroidRuntime(5620):     at net.simonvt.numberpicker.NumberPicker.setValueInternal(NumberPicker.java:1641)
12-30 23:09:26.580: E/AndroidRuntime(5620):     at net.simonvt.numberpicker.NumberPicker.scrollBy(NumberPicker.java:1106)
12-30 23:09:26.580: E/AndroidRuntime(5620):     at net.simonvt.numberpicker.NumberPicker.onTouchEvent(NumberPicker.java:886)
12-30 23:09:26.580: E/AndroidRuntime(5620):     at android.view.View.dispatchTouchEvent(View.java:7127)
12-30 23:09:26.580: E/AndroidRuntime(5620):     at android.view.ViewGroup.dispatchTransformedTouchEvent(ViewGroup.java:2170)
12-30 23:09:26.580: E/AndroidRuntime(5620):     at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:1905)
12-30 23:09:26.580: E/AndroidRuntime(5620):     at net.simonvt.numberpicker.NumberPicker.dispatchTouchEvent(NumberPicker.java:944)
    ....

我正在minValue,maxValue and DisplayedValues重置以重置NumberPicker数据并且无法正常工作,那么正确的方法是什么?

4

5 回答 5

22

在 onClick 中检查 Call setDisplayedValues(null)before 。setMaxValue

参考:https ://stackoverflow.com/a/24322319/2624806

于 2014-08-12T12:14:20.737 回答
7

我有同样的问题,我没有修改就解决了NumberPicker.java。这是供您参考的代码。

private void initCityData(int provinceId) {
        List<Arealist> cities = mAreaUtil.getCityListOfProvince(provinceId);
        if (cities != null && cities.size() > 0) {
            try {
                int min = Integer.parseInt(cities.get(0).getAreaid());
                int max = Integer.parseInt(cities.get(cities.size() -1).getAreaid());
                String[] cityNames = new String[cities.size()];
                for (int i = 0; i < cityNames.length; i++) {
                    cityNames[i] = cities.get(i).getName();
                }
                int maxV = mCityPicker.getMaxValue();
                if (max> maxV){ 
                    mCityPicker.setMinValue(min);
                    mCityPicker.setValue(0);
                    mCityPicker.setDisplayedValues(cityNames);
                    mCityPicker.setMaxValue(max);
                }else{
                    mCityPicker.setMinValue(min);
                    mCityPicker.setValue(0);
                    mCityPicker.setMaxValue(max);
                    mCityPicker.setDisplayedValues(cityNames);
                }

            } catch (NumberFormatException e) {
                ILog.e(TAG, e.getMessage(), e);
            }
        }
    }
于 2014-01-01T11:01:01.917 回答
4

只需在重置数字选择器之前将 displayValues 设置为 null,如下所示:

 pickerWard.displayedValues = null

 pickerWard.maxValue = values.size - 1
 pickerWard.displayedValues = values
于 2019-09-10T06:58:51.417 回答
3

好吧,我查看了NumberPicker.java的源代码,发现它会重新计算数据并刷新和中的setMinValue(int minValue)UI setMaxValue(int minValue)。当设置新的 minValue 时,它​​将使用新的 minValue 和旧的 maxValue 进行计算,然后发生错误。

所以,我修改了一点:只在andNumberPicker中设置值,并在. 当数据发生变化时,只需重新设置 minValue、maxVaule 和 displayValues。请注意,必须最后调用。setMinValue(int minValue)setMaxValue(int maxValue)setDisplayedValues(String[] displayedValues)NumberPickersetDisplayedValues(String[] displayedValues)

这是我修改的内容:

NumberPicker.java

public void setMinValue(int minValue) {
    if (mMinValue == minValue) {
        return;
    }
    if (minValue < 0) {
        throw new IllegalArgumentException("minValue must be >= 0");
    }
    mMinValue = minValue;
    if (mMinValue > mValue) {
        mValue = mMinValue;
    }
    //boolean wrapSelectorWheel = mMaxValue - mMinValue > mSelectorIndices.length;
    //setWrapSelectorWheel(wrapSelectorWheel);
    //initializeSelectorWheelIndices();
    //updateInputTextView();
    //tryComputeMaxWidth();
    invalidate();
}

public void setMaxValue(int maxValue) {
    if (mMaxValue == maxValue) {
        return;
    }
    if (maxValue < 0) {
        throw new IllegalArgumentException("maxValue must be >= 0");
    }
    mMaxValue = maxValue;
    if (mMaxValue < mValue) {
        mValue = mMaxValue;
    }
    //boolean wrapSelectorWheel = mMaxValue - mMinValue > mSelectorIndices.length;
    //setWrapSelectorWheel(wrapSelectorWheel);
    //initializeSelectorWheelIndices();
    //updateInputTextView();
    //tryComputeMaxWidth();
    invalidate();
}

public void setDisplayedValues(String[] displayedValues) {
    if (mDisplayedValues == displayedValues) {
        return;
    }
    mDisplayedValues = displayedValues;
    if (mDisplayedValues != null) {
        // Allow text entry rather than strictly numeric entry.
        mInputText.setRawInputType(InputType.TYPE_CLASS_TEXT
                | InputType.TYPE_TEXT_FLAG_NO_SUGGESTIONS);
    } else {
        mInputText.setRawInputType(InputType.TYPE_CLASS_NUMBER);
    }
    boolean wrapSelectorWheel = mMaxValue - mMinValue > mSelectorIndices.length;
    setWrapSelectorWheel(wrapSelectorWheel);
    updateInputTextView();
    initializeSelectorWheelIndices();
    tryComputeMaxWidth();
}

AreaPickerDialog.java

private void initCityData(int provinceId) {
    List<Arealist> cities = mAreaUtil.getCityListOfProvince(provinceId);
    if (cities != null && cities.size() > 0) {
        try {
            int min = Integer.parseInt(cities.get(0).getAreaid());
            int max = Integer.parseInt(cities.get(cities.size() -1).getAreaid());
            String[] cityNames = new String[cities.size()];
            for (int i = 0; i < cityNames.length; i++) {
                cityNames[i] = cities.get(i).getName();
            }
            mCityPicker.setMinValue(min);
            mCityPicker.setMaxValue(max);
            mCityPicker.setDisplayedValues(cityNames);
        } catch (NumberFormatException e) {
            ILog.e(TAG, e.getMessage(), e);
        }
    }
}

这是一个丑陋但有效的解决方案。

于 2013-12-31T02:26:24.410 回答
0

查看您正在使用的 NumberPicker 的代码:

private boolean updateInputTextView() {
/*
* If we don't have displayed values then use the current number else
* find the correct value in the displayed values for the current
* number.
*/
        String text = (mDisplayedValues == null) ? formatNumber(mValue)
                : mDisplayedValues[mValue - mMinValue];

您可以在调用 setMinValue 之前看到它正在使用 mminValue。

在设置显示值之前尝试设置 minValue 和 maxValue。

于 2013-12-30T16:08:29.487 回答