我Activity
在 Android 中有一个,有两个元素:
EditText
ListView
当我Activity
开始时,EditText
立即具有输入焦点(闪烁的光标)。我不希望任何控件在启动时具有输入焦点。我试过:
EditText.setSelected(false);
EditText.setFocusable(false);
没运气。我怎样才能说服开始EditText
时不要选择自己Activity
?
我Activity
在 Android 中有一个,有两个元素:
EditText
ListView
当我Activity
开始时,EditText
立即具有输入焦点(闪烁的光标)。我不希望任何控件在启动时具有输入焦点。我试过:
EditText.setSelected(false);
EditText.setFocusable(false);
没运气。我怎样才能说服开始EditText
时不要选择自己Activity
?
像下面的例子一样添加标签android:focusableInTouchMode="true"
和android:focusable="true"
父布局(例如LinearLayout
或ConstraintLayout
),将解决问题。
<!-- 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"/>
实际问题是您根本不希望它有焦点吗?或者您不希望它因为专注于EditText
? 我真的没有看到将EditText
注意力集中在开始上的问题,但是当用户没有明确请求关注EditText
(并因此打开键盘)时,打开 softInput 窗口绝对是一个问题。
如果是虚拟键盘的问题,请参阅AndroidManifest.xml
<activity> 元素文档。
android:windowSoftInputMode="stateHidden"
- 进入活动时始终隐藏它。
或android:windowSoftInputMode="stateUnchanged"
- 不要更改它(例如,如果它尚未显示,则不要显示它,但如果在进入活动时它是打开的,则让它保持打开状态)。
存在一个更简单的解决方案。在您的父布局中设置这些属性:
<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"
.
真的,我们可以看到beforeDescendants
在方法中设置为默认值ViewGroup.initViewGroup()
(Android 2.2.2)。但不等于0。ViewGroup.FOCUS_BEFORE_DESCENDANTS = 0x20000;
感谢纪尧姆。
我找到的唯一解决方案是:
android:focusable="true"
和android:focusableInTouchMode="true"
并且EditText
在开始活动后不会获得焦点
问题似乎来自我只能在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>
使用其他海报提供的信息,我使用了以下解决方案:
在布局 XML
<!-- Dummy item to prevent AutoCompleteTextView from receiving focus -->
<LinearLayout
android:id="@+id/linearLayout_focus"
android:focusable="true"
android:focusableInTouchMode="true"
android:layout_width="0px"
android:layout_height="0px"/>
<!-- AUTOCOMPLETE -->
<AutoCompleteTextView
android:id="@+id/autocomplete"
android:layout_width="200dip"
android:layout_height="wrap_content"
android:layout_marginTop="20dip"
android:inputType="textNoSuggestions|textVisiblePassword"/>
在 onCreate()
private AutoCompleteTextView mAutoCompleteTextView;
private LinearLayout mLinearLayout;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.mylayout);
//get references to UI components
mAutoCompleteTextView = (AutoCompleteTextView) findViewById(R.id.autocomplete);
mLinearLayout = (LinearLayout) findViewById(R.id.linearLayout_focus);
}
最后,在 onResume()
@Override
protected void onResume() {
super.onResume();
//do not give the editbox focus automatically when activity starts
mAutoCompleteTextView.clearFocus();
mLinearLayout.requestFocus();
}
尝试clearFocus()而不是setSelected(false)
. Android中的每个视图都具有可聚焦性和可选择性,我认为您只想清除焦点。
以下将在创建时停止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://stackoverflow.com/a/8639921/15695)
并删除
<requestFocus />
在 EditText
(参考来自 floydaddict https://stackoverflow.com/a/9681809)
迟到但最简单的答案,只需将其添加到 XML 的父布局中。
android:focusable="true"
android:focusableInTouchMode="true"
对你有帮助就点个赞吧!快乐编码:)
这些解决方案都不适合我。我修复自动对焦的方法是:
<activity android:name=".android.InviteFriendsActivity" android:windowSoftInputMode="adjustPan">
<intent-filter >
</intent-filter>
</activity>
简单的解决方案:在AndroidManifest
标签中Activity
使用
android:windowSoftInputMode="stateAlwaysHidden"
您可以在第一个. _ 这样,当活动开始时,将获得焦点,但由于其性质,您不会在屏幕上看到任何焦点,当然,也不会显示键盘...TextView
layout
TextView
以下对我有用Manifest
。写 ,
<activity
android:name=".MyActivity"
android:windowSoftInputMode="stateAlwaysHidden"/>
我需要以编程方式清除所有领域的焦点。我刚刚将以下两个语句添加到我的主要布局定义中。
myLayout.setDescendantFocusability(ViewGroup.FOCUS_BEFORE_DESCENDANTS);
myLayout.setFocusableInTouchMode(true);
就是这样。立即解决了我的问题。谢谢,西尔弗,为我指明了正确的方向。
添加文件android:windowSoftInputMode="stateAlwaysHidden"
的活动标签Manifest.xml
。
如果您对自己的活动有另一种看法,例如 a ListView
,您还可以执行以下操作:
ListView.requestFocus();
在你onResume()
从editText
.
我知道这个问题已经得到解答,但只是提供了一个对我有用的替代解决方案:)
在您的第一个可编辑字段之前尝试此操作:
<TextView
android:id="@+id/dummyfocus"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="@string/foo"
/>
----
findViewById(R.id.dummyfocus).setFocusableInTouchMode(true);
findViewById(R.id.dummyfocus).requestFocus();
在方法中添加以下内容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"
晚了,但也许有帮助。myDummyEditText.requestFocus()
在布局顶部创建一个虚拟 EditText 然后调用onCreate()
<EditText android:id="@+id/dummyEditTextFocus"
android:layout_width="0px"
android:layout_height="0px" />
这似乎符合我的预期。无需处理配置更改等。对于具有冗长 TextView(说明)的 Activity,我需要此功能。
是的,我做了同样的事情——创建一个获得初始焦点的“虚拟”线性布局。此外,我设置了“下一个”焦点 ID,因此用户在滚动一次后无法再对其进行聚焦:
<LinearLayout 'dummy'>
<EditText et>
dummy.setNextFocusDownId(et.getId());
dummy.setNextFocusUpId(et.getId());
et.setNextFocusUpId(et.getId());
很多工作只是为了摆脱对视图的关注..
谢谢
对我来说,在所有设备上都有效的是:
<!-- fake first focusable view, to allow stealing the focus to itself when clearing the focus from others -->
<View
android:layout_width="0px"
android:layout_height="0px"
android:focusable="true"
android:focusableInTouchMode="true" />
只需将此作为一个视图放在有问题的重点视图之前,就是这样。
<TextView
android:id="@+id/TextView01"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:layout_weight="1"
android:singleLine="true"
android:ellipsize="marquee"
android:marqueeRepeatLimit="marquee_forever"
android:focusable="true"
android:focusableInTouchMode="true"
style="@android:style/Widget.EditText"/>
我做的最简单的事情是在 onCreate 中将焦点设置在另一个视图上:
myView.setFocusableInTouchMode(true);
myView.requestFocus();
这停止了软键盘的出现,并且 EditText 中没有光标闪烁。
这是完美和最简单的解决方案。我总是在我的应用程序中使用它。
getWindow().setSoftInputMode(WindowManager.LayoutParams.SOFT_INPUT_STATE_HIDDEN);
将此代码写入您不想打开键盘Manifest
的文件中。Activity
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>
在onCreate
您的活动中,只需clearFocus()
在您的 EditText 元素上添加使用。例如,
edittext = (EditText) findViewById(R.id.edittext);
edittext.clearFocus();
如果您想将焦点转移到另一个元素,请使用requestFocus()
它。例如,
button = (Button) findViewById(R.id.button);
button.requestFocus();
隐藏键盘的最简单方法是使用 setSoftInputMode
getWindow().setSoftInputMode(WindowManager.LayoutParams.SOFT_INPUT_STATE_ALWAYS_HIDDEN);
或者您可以使用 InputMethodManager 并像这样隐藏键盘。
InputMethodManager imm = (InputMethodManager) getSystemService(Context.INPUT_METHOD_SERVICE);
imm.hideSoftInputFromWindow(myEditText.getWindowToken(), 0);
我使用以下代码来阻止 EditText 在按下按钮时窃取焦点。
addButton.setOnClickListener(new OnClickListener() {
public void onClick(View v) {
View focused = internalWrapper.getFocusedChild();
focused.setVisibility(GONE);
v.requestFocus();
addPanel();
focused.setVisibility(VISIBLE);
}
});
基本上,隐藏编辑文本,然后再次显示。这对我有用,因为 EditText不在视图中,所以它是否显示并不重要。
您可以尝试将其隐藏并连续显示,看看是否有助于它失去焦点。
您可以通过创建一个EditText
布局宽度和高度设置为的 dummy 来实现这一点0dp
,并请求该视图的焦点。在您的 xml 布局中添加以下代码片段:
<EditText
android:id="@+id/editText0"
android:layout_width="0dp"
android:layout_height="0dp"
android:hint="@string/dummy"
android:ems="10"
>
<requestFocus />
</EditText>
确保"<requestFocus />"
从EditText
xml 中的标记中删除该行。
<EditText
android:id="@+id/input"
android:layout_width="fill_parent"
android:layout_height="wrap_content">
<requestFocus /> <!-- remove this line -->
</EditText>
View current = getCurrentFocus();
if (current != null)
current.clearFocus();
当您的活动打开时,键盘会自动可见,这会导致 EditText 聚焦。您可以通过在 manifest.xml 文件的活动标记中写入以下行来禁用键盘。
android:windowSoftInputMode="stateAlwaysHidden|adjustPan"
/**
* set focus to top level window
* disposes descendant focus
* disposes softInput
* */
public static void topLevelFocus(Context context){
if(Activity.class.isAssignableFrom(context.getClass())){
ViewGroup tlView = (ViewGroup) ((Activity) context).getWindow().getDecorView();
if(tlView!=null){
tlView.setFocusable(true);
tlView.setFocusableInTouchMode(true);
tlView.setDescendantFocusability(ViewGroup.FOCUS_BEFORE_DESCENDANTS);
}
}
}
您有编辑文本和列表。在 OnStart/On Create 中,您应该将焦点设置在列表视图上:Listview.requestfocus()
可以通过继承EditText
和覆盖来实现onTouchEvent
。
class NonFocusableEditText: EditText {
constructor(context: Context): super(context)
constructor(context: Context, attrs: AttributeSet?): super(context, attrs)
constructor(context: Context, attrs: AttributeSet?, defStyleAttr: Int): super(context, attrs, defStyleAttr)
override fun onTouchEvent(event: MotionEvent?): Boolean {
return if (isFocusable) super.onTouchEvent(event) else false
}
}
然后你可以在正常的布局中使用它EditText
:
<com.yourpackage.NonFocusableEditText
android:id="@+id/editText"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:hint="@string/your_hint"
android:imeOptions="actionDone"
android:inputType="textNoSuggestions" />
只需添加android:focusableInTouchMode="true"
EditText 的父布局,您将摆脱这种尴尬的行为。
尝试
edit.setInputType(InputType.TYPE_NULL);
edit.setEnabled(false);
EditText
aListView
内不能正常工作。使用时最好使用TableLayout
自动生成的行EditText
。
您可以通过使用请求焦点和键盘隐藏代码来指定其他小部件的焦点。
在 onCreate() 中禁用它
final KeyListener edtTxtMessageKeyListener = edtTxtMessage.getKeyListener();
edtTxtMessage.setCursorVisible(false);
edtTxtMessage.setKeyListener(null);
最后在 EditText 的 onClick() 中启用它
edtTxtMessage.setCursorVisible(true);
edtTxtMessage.setKeyListener(edtTxtMessageKeyListener);
但问题是我们必须第一次单击 twise 才能带上 OnScreenKeyboard。
@解决方法
InputMethodManager imm = (InputMethodManager) getActivity().getSystemService(Context.INPUT_METHOD_SERVICE);
imm.toggleSoftInput(InputMethodManager.SHOW_IMPLICIT, 0);
在 onClick() 中也试试这个:)
我遇到了这样的问题,这是由于我的选择器造成的。第一个状态是焦点,即使我的视图被禁用,它仍处于焦点状态,因为它是第一个匹配并使用它的状态。您可以像这样将第一个状态设置为禁用:
<selector xmlns:android="http://schemas.android.com/apk/res/android">
<item android:drawable="@drawable/text_field_disabled" android:state_enabled="false"/>
<item android:drawable="@drawable/text_field_focused" android:state_focused="true"/>
<item android:drawable="@drawable/text_field_normal"/>
<EditText
android:layout_width="match_parent"
android:layout_height="100dp"
android:id="@+id/etComments"
android:hint="Comments.."
android:textSize="14dp"
android:focusable="false"
android:textStyle="italic"/>
最简单的方法是在Manifest.xml文件android:windowSoftInputMode="stateAlwaysHidden"
的活动标签中添加
<requestFocus />
从 xml 文件中的编辑文本中删除
<EditText
android:id="@+id/emailField"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:inputType="textEmailAddress">
//`<requestFocus />` /* <-- remove this tags */
</EditText>`
这样做并完成您的工作!
android:focusable="true"
android:focusableInTouchMode="true"
如果您想在活动开始时隐藏键盘。然后提
安卓:windowSoftInputMode="stateHidden"
到清单文件中的那个活动。问题得到解决。
干杯。
我用提交按钮清除所有焦点
XML文件:
<LinearLayout
...
android:id="@+id/linear_layout"
android:focusableInTouchMode="true"> // 1. make this focusableInTouchMode...
</LinearLayout>
JAVA文件:
private LinearLayout mLinearLayout; // 2. parent layout element
private Button mButton;
mLinearLayout = findViewById(R.id.linear_layout);
mButton = findViewById(R.id.button);
mButton.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
mLinearLayout.requestFocus(); // 3. request focus
}
});
我希望这对你有帮助:)
简单可靠的解决方案,只需覆盖此方法:
@Override
public boolean dispatchTouchEvent(MotionEvent ev) {
View v = getCurrentFocus();
if (v != null &&
(ev.getAction() == MotionEvent.ACTION_UP || ev.getAction() == MotionEvent.ACTION_MOVE) &&
v instanceof EditText &&
!v.getClass().getName().startsWith("android.webkit.")) {
int scrcoords[] = new int[2];
v.getLocationOnScreen(scrcoords);
float x = ev.getRawX() + v.getLeft() - scrcoords[0];
float y = ev.getRawY() + v.getTop() - scrcoords[1];
if (x < v.getLeft() || x > v.getRight() || y < v.getTop() || y > v.getBottom())
hideKeyboard(this);
}
return super.dispatchTouchEvent(ev);
}
public static void hideKeyboard(Activity activity) {
if (activity != null && activity.getWindow() != null && activity.getWindow().getDecorView() != null) {
InputMethodManager imm = (InputMethodManager) activity.getSystemService(Context.INPUT_METHOD_SERVICE);
imm.hideSoftInputFromWindow(activity.getWindow().getDecorView().getWindowToken(), 0);
}
}
已经提供了很多可行的答案,但我认为我们可以通过使用以下简单方法做得更好
//set focus to input field
private fun focusHere() {
findViewById<TextView>(R.id.input).requestFocus()
}
代替 R.id.input 中的输入,使用任何其他视图 ID 将焦点设置到该视图。
您可以将 Editext 设置为禁用焦点属性,现在这可以通过两种方式应用:
您可以禁用可聚焦作为一般属性
或者您可以禁用 FocusableInTouchMode 作为触摸模式下特定视图的属性(触摸屏)
默认情况下,focusable 属性为 true,如果 Editext 位于该活动中的视图堆栈的顶部,例如标题,则它将在活动启动时成为焦点。
要禁用 Focusable,您只需将其布尔值设置为 false。
所以那将是:
android:focusable="false"
要禁用它 FocusableInTouchMode,您只需将其布尔值设置为 false。
所以那将是:
android:focusable="false"
您只需找到要对其应用更改的 Textview 并将相应的代码段添加到 XML 文件中的 xml 规范中。
或者,您可以单击布局编辑器内的 Textview 并找到显示该 Textview 的所有 xml 属性的侧边栏,然后只需向下滚动到声明“Focusable”和“FocusableInTouchMode”的位置并检查它们是否为真或假。
在您提到您的活动的清单文件中添加以下行
android:windowSoftInputMode="stateAlwaysHidden"