android edittext焦点 listview - 阻止EditText在Activity启动时获得关注




15 Answers

Luc和Mark的优秀答案却缺少一个好的代码示例。 将标记android:focusableInTouchMode="true"<LinearLayout> ,如下例所示将解决问题。

<!-- Dummy item to prevent AutoCompleteTextView from receiving focus -->
<LinearLayout
    android:focusable="true" 
    android:focusableInTouchMode="true"
    android:layout_width="0px" 
    android:layout_height="0px"/>

<!-- :nextFocusUp and :nextFocusLeft have been set to the id of this component
to prevent the dummy from receiving focus again -->
<AutoCompleteTextView android:id="@+id/autotext"
    android:layout_width="fill_parent" 
    android:layout_height="wrap_content"
    android:nextFocusUp="@id/autotext" 
    android:nextFocusLeft="@id/autotext"/>

我在Android中有一个Activity ,有两个元素:

  1. EditText
  2. ListView

当我的Activity启动时, EditText立即具有输入焦点(闪烁光标)。 我不希望任何控件在启动时具有输入焦点。 我试过了:

EditText.setSelected(false);

没运气。 如何在Activity开始时说服EditText不选择自己?




存在更简单的解决方案。 在父布局中设置这些属性:

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/mainLayout"
    android:descendantFocusability="beforeDescendants"
    android:focusableInTouchMode="true" >

现在,当活动开始时,此主要布局将默认获得焦点。

此外,我们可以通过再次将焦点放在主布局上,在运行时从子视图中删除焦点(例如,在完成子编辑之后),如下所示:

findViewById(R.id.mainLayout).requestFocus();

Guillaume Perrot的 好评

android:descendantFocusability="beforeDescendants"似乎是默认值(整数值为0)。 它只是通过添加android:focusableInTouchMode="true"

实际上,我们可以看到beforeDescendantsViewGroup.initViewGroup()方法(Android 2.2.2)中设置为默认值。 但不等于0. ViewGroup.FOCUS_BEFORE_DESCENDANTS = 0x20000;

感谢纪尧姆。




问题似乎来自于我只能在布局的XML form中看到的属性。

确保在EditText XML标记内的声明末尾删除此行:

<requestFocus />

这应该是这样的:

<EditText
   android:id="@+id/emailField"
   android:layout_width="fill_parent"
   android:layout_height="wrap_content"
   android:inputType="textEmailAddress">

   //<requestFocus /> /* <-- without this line */
</EditText>



以下内容将阻止edittext在创建时获得焦点,但在触摸时抓住它。

<EditText
    android:id="@+id/et_bonus_custom"
    android:focusable="false" />

所以你在xml中将focusable设置为false,但关键是在java中,你添加了以下监听器:

etBonus.setOnTouchListener(new View.OnTouchListener() {
    @Override
    public boolean onTouch(View v, MotionEvent event) {
        v.setFocusable(true);
        v.setFocusableInTouchMode(true);
        return false;
    }
});

因为您返回false,即不消耗该事件,所以聚焦行为将像平常一样进行。




我曾单独尝试了几个答案,但重点仍然是EditText。 我只是通过使用以下两个解决方案来设法解决它。

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
  android:id="@+id/mainLayout"
  android:descendantFocusability="beforeDescendants"
  android:focusableInTouchMode="true" >

(来自Silver的参考https://.com/a/8639921/15695 )

并删除

 <requestFocus />

在EditText

(参考文献floydaddict share )




简单的解决方案:在AndroidManifest中使用Activity标签

android:windowSoftInputMode="stateAlwaysHidden"



以下在Manifest为我工作。 写,

<activity
android:name=".MyActivity"
android:windowSoftInputMode="stateAlwaysHidden"/>



Manifest.xml文件的activity标记中添加android:windowSoftInputMode="stateAlwaysHidden"

Source




如果您的活动有另一个视图,如ListView ,您还可以:

ListView.requestFocus(); 

在你的onResume()中从editText获取焦点。

我知道这个问题已经得到解答,但只是提供了一个对我有用的替代解决方案:)




onCreate方法中添加以下内容:

this.getWindow().setSoftInputMode(WindowManager.LayoutParams.SOFT_INPUT_STATE_ALWAYS_HIDDEN);



由于我不喜欢用与功能相关的东西污染XML,我创建了这个方法,“透明地”从第一个可聚焦视图中窃取焦点,然后确保在必要时自行移除!

public static View preventInitialFocus(final Activity activity)
{
    final ViewGroup content = (ViewGroup)activity.findViewById(android.R.id.content);
    final View root = content.getChildAt(0);
    if (root == null) return null;
    final View focusDummy = new View(activity);
    final View.OnFocusChangeListener onFocusChangeListener = new View.OnFocusChangeListener()
    {
        @Override
        public void onFocusChange(View view, boolean b)
        {
            view.setOnFocusChangeListener(null);
            content.removeView(focusDummy);
        }
    };
    focusDummy.setFocusable(true);
    focusDummy.setFocusableInTouchMode(true);
    content.addView(focusDummy, 0, new LinearLayout.LayoutParams(0, 0));
    if (root instanceof ViewGroup)
    {
        final ViewGroup _root = (ViewGroup)root;
        for (int i = 1, children = _root.getChildCount(); i < children; i++)
        {
            final View child = _root.getChildAt(i);
            if (child.isFocusable() || child.isFocusableInTouchMode())
            {
                child.setOnFocusChangeListener(onFocusChangeListener);
                break;
            }
        }
    }
    else if (root.isFocusable() || root.isFocusableInTouchMode())
        root.setOnFocusChangeListener(onFocusChangeListener);

    return focusDummy;
}



在父布局中写下这一行......

 android:focusableInTouchMode="true"



这是最完美,最简单的解决方案。我总是在我的应用程序中使用它。

getWindow().setSoftInputMode(WindowManager.LayoutParams.SOFT_INPUT_STATE_HIDDEN);



将此代码写入您不想打开键盘的Activity中的Manifest文件中。

android:windowSoftInputMode="stateHidden"

清单文件:

 <?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.projectt"
    android:versionCode="1"
    android:versionName="1.0" >

    <uses-sdk
        android:minSdkVersion="8"
        android:targetSdkVersion="24" />

    <application
        android:allowBackup="true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme" >
         <activity
            android:name=".Splash"
            android:label="@string/app_name" >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
        <activity
            android:name=".Login"
            **android:windowSoftInputMode="stateHidden"**
            android:label="@string/app_name" >
        </activity>

    </application>

</manifest>



在Activity的onCreate中,只需在EditText元素上添加clearFocus() 。 例如,

edittext = (EditText) findViewById(R.id.edittext);
edittext.clearFocus();

如果你想将焦点转移到另一个元素,请使用requestFocus() 。 例如,

button = (Button) findViewById(R.id.button);
button.requestFocus();



Related