为NumberPicker捕获键盘'Done'

时间:2013-02-10 21:38:14

标签: android numberpicker

我有一个AlertDialog只有一些文字,NumberPicker,一个确定和一个取消。

package org.dyndns.schep.example;
import android.os.Bundler;
import android.view.View;
import android.widget.NumberPicker;
import android.app.Activity;
import android.app.AlertDialog;
import android.app.Dialog;
import android.app.DialogFragment;
import android.content.DialogInterface;

public class FooFragment extends DialogFragment {
    @Override
    public void onAttach(Activity activity) {
        super.onAttach(activity);
        mParent = (MainActivity) activity;
    }

    @Override
    public Dialog onCreateDialog(Bundle savedInstanceState) {
        AlertDialog.Builder builder = new AlertDialog.Builder(getActivity());
        builder.setPositiveButton(android.R.string.ok,
                new DialogInterface.OnClickListener() {
            @Override
            public void onClick(DialogInterface dialog, int id) {
                mParent.setFoo(foo());
            }
        })
        .setNegativeButton(android.R.string.cancel, null);

        View view = getActivity().getLayoutInflater.inflate(
                R.layout.dialog_foo, null);
        mPicker = (NumberPicker) view.findViewById(R.id.numberPicker1);
        mPicker.setValue(mParent.getFoo());
        builder.setView(view);
        return builder.create();
    }

    public int foo() {
        return mPicker.getValue();
    }

    private MainActivity mParent;
    private NumberPicker mPicker;
}

(我知道,此对话框尚未执行暂停和恢复时应保留状态的操作。)

我想在软键盘或其他IME上执行“完成”操作以关闭对话框,就像按下了“确定”一样,因为只有一个小部件可以编辑。

看起来处理IME“完成”的最佳方式通常是setOnEditorActionListener上的TextView。但我没有任何TextView变量,NumberPicker显然没有公开任何TextView或类似的编辑器回调。 (也许NumberPicker包含TextView,其中包含我可以使用findViewById搜索的常量ID?)

NumberPicker.setOnValueChangedListener会在“完成”操作中触发,但在点击或轻拂数字列表时也会触发,这绝对不应该忽略对话框。

基于this问题,我尝试检出setOnKeyListener,但在使用软键盘时该界面根本没有触发。这并不奇怪,因为KeyEvent文档表明它对硬件事件的意义更大,而在最近的API中,软键盘根本不会发送它们。

如何将IME“完成”连接到对话框的“确定”操作?

修改:从源代码的外观来看,NumberPicker布局确实包含EditText,但其id/numberpicker_input包中的ID为com.android.internal。使用它并不容易,显然不鼓励。但似乎只有黑客可以获得我想要的行为。

2 个答案:

答案 0 :(得分:9)

  

如何将IME“完成”连接到对话框的“确定”操作?

问题是,如果您在TextView窗口小部件上没有设置侦听器,而您当前可以使用IME,则无法传递IME的事件。做你想做的事情的一种方法是将我们自己的逻辑挂钩到与{IME一起工作的NumberPicker的孩子(就像你在问题的最后部分已经讨论过的那样)。为了避免使用某些ID或其他布局技巧(可能有问题)来获取该窗口小部件,您可以使用贪婪的策略,将侦听器设置为NumberPicker中可能触发所需事件的任何窗口小部件( TextViewsTextView的任何子类。像这样:

    private AlertDialog mCurrentDialog;
    private List<TextView> mTargets = new ArrayList<TextView>();
    private OnEditorActionListener mListener = new OnEditorActionListener() {

        @Override
        public boolean onEditorAction(TextView v, int actionId,
                KeyEvent event) {
            if (actionId == EditorInfo.IME_ACTION_DONE) {
                // if a child of NumberPicker triggers the DONE editor event
                // get a reference to the positive button(which you use in your
                // code) and click it
               mCurrentDialog.getButton(Dialog.BUTTON_POSITIVE).performClick();
            }
            return false;
        }
    };

@Override
public Dialog onCreateDialog(Bundle savedInstanceState) {
     // ...
     mPicker = (NumberPicker) view.findViewById(R.id.numberPicker1);
        mPicker.setValue(mParent.getFoo());
        // clear any previous targets
        mTargets.clear();
        // find possible targets in the NumberPicker 
        findTextViews(mPicker);
        // setup those possible targets with our own logic 
        setupEditorListener();          
        builder.setView(view);
        // get a reference to the current showed dialog 
        mCurrentDialog = builder.create();          
        return mCurrentDialog;
    }

方法是:

private void findTextViews(ViewGroup parent) {
        final int count = parent.getChildCount();
        for (int i = 0; i < count; i++) {
            final View child = parent.getChildAt(i);
            if (child instanceof ViewGroup) {
                findTextViews((ViewGroup) child);
            } else if (child instanceof TextView) {
                mTargets.add((TextView) child);
            }
        }
    }

    private void setupEditorListener() {
        final int count = mTargets.size();
        for (int i = 0; i < count; i++) {
            final TextView target = mTargets.get(i);
            target.setOnEditorActionListener(mListener);
        }
    }

另一种可能的(也是合理的)解决方案(就像Naveen在他的评论中已经提到过的)是使用NumberPicker类的一个端口(或修改SDK中的一个)然后插入你自己的小部件ID(这将使得对小部件的引用成为一项简单的任务)。这样现在更容易实现,但从长远来看维护起来不方便。

答案 1 :(得分:0)

我的搜索和实验也空手而归。这就是说我不会鼓励这种行为,因为它没有多大意义。

如果您需要NumberPicker,那是因为您希望滚动的功能为您需要的值,而不是输入它(尽管可能)。如果你想输入你使用EditText的值,你就可以毫无问题地实现你所需要的。

或者,您必须从源复制NumberPicker实现,然后根据您的需要进行更改(通过添加ime选项来更新)。