Android 警告 : Window already focused, 忽略焦点增益

标签 android android-spinner

伙计们,我需要你们的眼睛。我已经为我的代码奋斗了很长时间,但我仍然像一个旧架子一样卡住了。

你看,有很多主题与 android 警告相关:

W/InputMethodManagerService: Window already focused, ignoring focus gain of: com.android.internal.view.IInputMethodClient$Stub...

我读了很多,没有找到任何有用的东西,我的眼睛很累。

我正在开发一个微调器组件,第一个项目名为 Add new... 会触发一个对话框以将另一个项目添加到列表中。因为我目前有两个这样的微调器(在一个 Activity 中),所以我为微调器创建了一个包装类,它负责处理逻辑,并且我创建了类,它扩展了 DialogFragment。因为两个微调器都会用 <EditText> 触发类似的对话框, 只是不同的标题。

但在 View 中,只有第一个微调器起作用。第二个旋转器,当choosig选项添加新的... 时,什么也不做,只会产生上面写的警告。

微调包装器

KegAddActivity extends AppCompatActivity 的内部类看起来像这样:

public class ExtendableSpinner implements AdapterView.OnItemSelectedListener, View.OnTouchListener {
    private static final int ADD_NEW_ID = -1;

    private final Spinner _mSpinner;
    private final String _mTableName;
    private final int _mAddNewTitle;
    private final MatrixCursor _mAddOptionsCursor;

    private Uri _mAdapterUri;
    private SimpleCursorAdapter _mAdapter;
    private boolean _mWasTouched;

    public ExtendableSpinner(Spinner spinner, String tableName, int addNewTitle) {
        super();

        _mSpinner = spinner;
        _mTableName = tableName;
        _mAdapterUri = BeerBookUriHandler.getUri(tableName);
        _mAddNewTitle = addNewTitle;

        _mAddOptionsCursor = new MatrixCursor(new String[]{Table.COL_ID, BeerTable.COL_NAME});
        _mAddOptionsCursor.addRow(new String[]{"" + ADD_NEW_ID, getString(addNewTitle) + '\u2026'});

        _mSpinner.setOnTouchListener(this);
        _mSpinner.setOnItemSelectedListener(this);

    }

    public void setAdapterUri(Uri uri) {
        _mAdapterUri = uri;
    }

    @Override
    public boolean onTouch(View v, MotionEvent event) {
        _mWasTouched = true;
        return false;
    }

    @Override
    public void onItemSelected(AdapterView<?> parent, View view, int position, long id) {
        if (_mWasTouched) {
            // event fired by user
            if (id == ADD_NEW_ID) {
                Bundle args = new Bundle();
                args.putInt(AddNewDialogFragment.TITLE, _mAddNewTitle);

                _mAddNewDialogFragment = new AddNewDialogFragment();
                _mAddNewDialogFragment.setArguments(args);
                _mAddNewDialogFragment.show(getSupportFragmentManager(), _mTableName + "AddNewDialog");
            }
        } else {
            // event fired at activity start
            int selection = 0;
            if (_mAdapter.getCount() > 1) {
                selection = 1;
                Uri uri = BeerBookUriHandler.getUri(KegTable.NAME + "/last");
                Cursor cursor = getContentResolver().query(uri, new String[]{
                        _mTableName + "." + Table.COL_ID
                }, null, null, null);
                if (cursor != null && cursor.moveToFirst()) {
                    long lastUsedId = cursor.getLong(0);
                    selection = getPositionForId(lastUsedId);
                }
                cursor.close();
            }
            _mSpinner.setSelection(selection);
        }
    }

    @Override
    public void onNothingSelected(AdapterView<?> parent) {

    }

    public void populate() {
        Cursor cursor = getContentResolver().query(_mAdapterUri,
                BEER_PROJECTION, null, null, BeerTable.COL_NAME);
        Cursor extendedCursor = new MergeCursor(new Cursor[]{ _mAddOptionsCursor, cursor});

        _mAdapter = new SimpleCursorAdapter(getApplicationContext(),
                SPINNER_LAYOUT, extendedCursor, new String[] { BeerTable.COL_NAME },
                new int[] {android.R.id.text1}, 0);
        _mAdapter.setDropDownViewResource(SPINNER_ITEM_LAYOUT);
        _mSpinner.setAdapter(_mAdapter);
    }

    public int getPositionForId(long itemId) {
        int pos = 0;
        for(int i = 1, l = _mAdapter.getCount(); i < l; i++)
            if (itemId == _mAdapter.getItemId(i)) {
                pos = i;
                break;
            }

        return pos;
    }

    public long getSelectedId() {
        return _mSpinner.getSelectedItemId();
    }

    public void setSelectionById(long id) {
        _mSpinner.setSelection(getPositionForId(id));
    }
}

桶添加 Activity

onCreate()方法我像这样实例化微调器

protected void onCreate(final Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);

    ...

    final Spinner beerSpinner = (Spinner) findViewById(R.id.keg_add_beer);
    _mBeerSpinner = new ExtendableSpinner(beerSpinner, BeerTable.NAME, R.string.beer_add);
    // initially disable, wait for brewery selection
    beerSpinner.setEnabled(false);

    final Spinner brewerySpinner = (Spinner) findViewById(R.id.keg_add_brewery);
    _mBrewerySpinner = new ExtendableSpinner(brewerySpinner, BreweryTable.NAME, R.string.brewery_add) {
        @Override
        public void onItemSelected(AdapterView<?> parent, View view, int position, long id) {
            super.onItemSelected(parent, view, position, id);
            if (id > 0) {
                // brewery selected, populate beerSpinner
                Uri uri = BeerBookUriHandler.getUri(BreweryTable.NAME + "/" +
                        getSelectedId() + "/" + BeerTable.NAME);
                _mBeerSpinner.setAdapterUri(uri);
                _mBeerSpinner.populate();
                beerSpinner.setEnabled(true);
            }
        }
    };
    _mBrewerySpinner.populate();

    ...
}

我目前的怀疑

因为第一个微调器工作而第二个微调器不工作并且它们非常相似,我认为微调器以某种方式并发聚焦,但我真的不知道。 我试过替换 DialogFragment每个微调器都有单独的 Activity ,并且 onItemSelected旋转器的事件然后称为 startActivityForResult()但这导致了同样的问题,第一个微调器运行良好,第二个微调器失败并出现警告,因此问题可能出在微调器的代码中。

终于有了一些进展!

幸运的是,我找到了 constellation 它起作用的地方,然后能够找出它不起作用的原因。

如果 Add new... 的项目当前在微调器中被选中,则再次选择 Add new... 项目会产生 ignoring focus gain警告。但是,如果您在微调器中选择其他一些项目,然后选择添加新...,它就会起作用。问题是,如果微调器中没有其他项目,则无法执行此类重新选择。好像是 onItemSelected如果您选择一个已选择的选项,则不会传播事件。但这是个大问题,因为用户可以在 AddNewDialog 处点击取消返回主视图,然后 Add new... 将保持选中状态并且可能是唯一的选项,因此无法打开 AddNewDialog再次。

如何使其工作的解决方案是添加另一个默认项,例如从列表中选择... 项,如果用户取消对话框,则将微调器重新选择为默认选项。我会测试它并保持这篇文章的更新。

结论

如果您选择 Spinner之前选择的项目,itemSelected事件未传播并且W/InputMethodManagerService: Window already focused, ignoring focus gain警告出现在您的日志中。

Spinner不支持 onItemClick事件,我发现唯一可行的解​​决方法是定义一个默认选项从列表中选择... 如果用户取消 AddNewDialog onDialogNegativeClick监听器将微调器重置为默认的 Choose from... 项目,因此在点击 Add new... 后,选择发生变化并且事件被传播。

最佳答案

我们遇到了同样的问题,并确定解决这个问题的唯一方法是创建一个调用 onItemSelected() 的自定义 Spinner,即使选择了相同的项目也是如此:

public class CustomSpinner extends Spinner {

        public CustomSpinner(Context contextArg){
            super(contextArg);
        }

        public CustomSpinner(Context contextArg, AttributeSet attributeSetArg){
            super(contextArg, attributeSetArg);
        }

        public CustomSpinner(Context contextArg, AttributeSet attributeSetArg, int styleArg){
            super(contextArg, attributeSetArg, styleArg);
        }

        @Override
        public void setSelection(int positionArg){
            boolean samePosition = positionArg == getSelectedItemPosition();
            super.setSelection(positionArg, false);
            // here we modifiy the Spinner's default behavior
            if(samePosition){
                // we dispatch the event, even if the position is the same
                OnItemSelectedListener onItemSelectedListener = getOnItemSelectedListener();
                if(onItemSelectedListener != null){
                    onItemSelectedListener.onItemSelected(this,
                            getSelectedView(),
                            positionArg,
                            getSelectedItemId());
                }
            }
        }

        @Override
        public void setSelection(int positionArg, boolean animateArg){
            boolean samePosition = positionArg == getSelectedItemPosition();
            super.setSelection(positionArg, animateArg);
            // here we modifiy the Spinner's default behavior
            if(samePosition){
                // we dispatch the event, even if the position is the same
                OnItemSelectedListener onItemSelectedListener = getOnItemSelectedListener();
                if(onItemSelectedListener != null){
                    onItemSelectedListener.onItemSelected(this,
                            getSelectedView(),
                            positionArg,
                            getSelectedItemId());
                }
            }
        }
}

关于Android 警告 : Window already focused, 忽略焦点增益,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33986258/

相关文章:

android - 不幸的是,<App name> 已停止工作。 Android?Eclipse应用

android - AppCompat v21 微调器样式

android - 如何在 TextView 下方正确显示弹出菜单,类似于 Spinner?

android - 自定义适配器 Listview 中的 Spinner OnItemSelectedListener 递归调用而不选择任何项目

android - 那么如何在 Jelly Bean 中查看日志呢?

android - 这是所有android 2.1及更高版本的默认字体

android - 在第一个微调器 android 中选择时如何更改第二个微调器数组

Android - 列表导航样式

java - 通过蓝牙低功耗连接从腕带心率传感器中检索数据到 Android 应用程序

android - 语言环境更改后图像未加载