如何在Android上制作EditText,以使用户可能无法输入多行文本,但显示仍然是多行(即,有自动换行而不是在右侧显示文本)?
它类似于内置的SMS应用程序,在该应用程序中我们无法输入换行符,但文本显示为多行。
如何在Android上制作EditText,以使用户可能无法输入多行文本,但显示仍然是多行(即,有自动换行而不是在右侧显示文本)?
它类似于内置的SMS应用程序,在该应用程序中我们无法输入换行符,但文本显示为多行。
Answers:
我将子类化为小部件并重写键事件处理,以阻止Enter
键:
class MyTextView extends EditText
{
...
@Override
public boolean onKeyDown(int keyCode, KeyEvent event)
{
if (keyCode==KeyEvent.KEYCODE_ENTER)
{
// Just ignore the [Enter] key
return true;
}
// Handle all other keys in the default way
return super.onKeyDown(keyCode, event);
}
}
return super.onKeyDown(keyCode, event);
应该在onKeyDown还是我在想象中?:P
这是一种方法,您不必重写EditText类。您只需捕获并用空字符串替换换行符。
edittext.addTextChangedListener(new TextWatcher() {
public void onTextChanged(CharSequence s, int start, int before, int count) {
}
public void beforeTextChanged(CharSequence s, int start, int count, int after) {
}
public void afterTextChanged(Editable s) {
/*
* The loop is in reverse for a purpose,
* each replace or delete call on the Editable will cause
* the afterTextChanged method to be called again.
* Hence the return statement after the first removal.
* http://developer.android.com/reference/android/text/TextWatcher.html#afterTextChanged(android.text.Editable)
*/
for(int i = s.length()-1; i >= 0; i--){
if(s.charAt(i) == '\n'){
s.delete(i, i + 1);
return;
}
}
}
});
感谢罗尔夫对较早的答案改善。
i > 0
而不是i >= 0
。什么时候i == 0
,s.subSequence(i-1, i)
得到一个IndexOutOfBoundsException
。
XML属性
android:lines="5"
android:inputType="textPersonName"
editText.setSingleLine(false);
如果可以,请改用它。
这对我有用:
<EditText
android:inputType="textShortMessage|textMultiLine"
android:minLines="3"
... />
它显示一个笑脸而不是Enter键。
@Andreas Rudolph提供的答案包含一个严重的错误,不应使用。IndexOutOfBoundsException
当您在EditText
其中包含多个换行符的文本过去时,该代码会导致。这是由所使用的循环类型引起的,Editable
对象将在afterTextChanged
其内容更改(替换,删除,插入)后立即调用该方法。
正确的代码:
edittext.addTextChangedListener(new TextWatcher() {
public void onTextChanged(CharSequence s, int start, int before, int count) {
}
public void beforeTextChanged(CharSequence s, int start, int count, int after) {
}
public void afterTextChanged(Editable s) {
/*
* The loop is in reverse for a purpose,
* each replace or delete call on the Editable will cause
* the afterTextChanged method to be called again.
* Hence the return statement after the first removal.
* http://developer.android.com/reference/android/text/TextWatcher.html#afterTextChanged(android.text.Editable)
*/
for(int i = s.length()-1; i >= 0; i--){
if(s.charAt(i) == '\n'){
s.delete(i, i + 1);
return;
}
}
}
});
i>=0
应该可以使用)^^
这是一个更正确的答案,该答案未在IME键盘上显示Enter键:
// IMPORTANT, do this before any of the code following it
myEditText.setSingleLine(true);
// IMPORTANT, to allow wrapping
myEditText.setHorizontallyScrolling(false);
// IMPORTANT, or else your edit text would wrap but not expand to multiple lines
myEditText.setMaxLines(6);
另外,您可以用XML布局文件上的setSingleLine(true)
显式android:inputType
或setInputType(InputType.*)
代码上的显式代替-在其中使用的输入类型是您所知道的将输入限制为仅一行的任何内容(即,setSingleLine(true)
已经隐式调用的任何内容)。
说明:
什么setSingleLine(true)
做呼吁setHorizontallyScrolling(true)
和setLines(1)
含蓄,与一起改变一些输入法键盘设置禁用回车键。
反过来,调用setLines(1)
像调用setMinLines(1)
和setMaxLines(1)
一个电话。
某些输入类型(即from中的常量InputType.TYPE_*
)会setSingleLine(true)
隐式调用,或至少会达到相同的效果。
结论:
因此,要实现OP所需的功能,我们只需通过还原那些隐式调用来对那些隐式设置进行计数。
我正在对此进行测试,它似乎可以工作:
EditText editText = new EditText(context);
editText.setSingleLine(false);
editText.setInputType(android.text.InputType.TYPE_CLASS_TEXT | android.text.InputType.TYPE_TEXT_VARIATION_EMAIL_SUBJECT);
editText.setSingleLine(false);
)。
尝试这个:
@Override
public boolean onKeyDown(int keyCode, KeyEvent event) {
if (keyCode == KeyEvent.KEYCODE_ENTER)
{
//Nothing
return true;
}
return super.onKeyDown(keyCode, event);
}
您可以从xml进行设置,如下所示:
android:imeOptions="actionDone"
android:inputType="text"
android:maxLines="10"
别忘了android:inputType="text"
,如果您不设置它,它将不起作用。我不知道为什么。另外,不要忘记更改maxLines
为您的首选值。
这是解决方案...
<EditText
android:id="@+id/editText"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:maxLength="150"
android:textSize="15dp"
android:imeOptions="actionDone"
android:inputType="text|textMultiLine"/>
java类中的用法
editText.setOnKeyListener(new View.OnKeyListener() {
@Override
public boolean onKey(View view, int keyCode, KeyEvent event) {
if (keyCode==KeyEvent.KEYCODE_ENTER)
{
// Just ignore the [Enter] key
return true;
}
// Handle all other keys in the default way
return (keyCode == KeyEvent.KEYCODE_ENTER);
}
});
Here is the solution...
?其他一些背景信息也有帮助。谢谢。
在我将带有换行符的文本复制到EditText中之前,可接受的答案一直很好。因此,我添加了onTextContextMenuItem来监视粘贴操作。
@Override
public boolean onTextContextMenuItem(int id) {
boolean ret = super.onTextContextMenuItem(id);
switch (id) {
case android.R.id.paste:
onTextPaste();
break;
}
return ret;
}
public void onTextPaste() {
if (getText() == null)
return;
String text = getText().toString();
text = text.replaceAll(System.getProperty("line.separator"), " ");
text = text.replaceAll("\\s+", " ");
setText(text);
}
<EditText
android:id="@+id/Msg"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:layout_marginTop="5dip"
android:lines="5"
android:selectAllOnFocus="true"
android:hint="Skriv meddelande...\n(max 100tkn)"/>
EditText et = (EditText)findViewById(R.id.Msg);
String strTmp = et.getText().toString();
strTmp = strTmp.replaceAll("\\n"," ");
EditText textView = new EditText(activity);
...
textView.setOnEditorActionListener(new TextView.OnEditorActionListener() {
@Override
public boolean onEditorAction(TextView textView, int i, KeyEvent keyEvent) {
if(KeyEvent.KEYCODE_ENTER == keyEvent.getKeyCode()) {
return false;
}
.......
}
});
我将提供另一个选择,这样您就不必继承EditText的子类。创建一个InputFilter
可过滤掉换行符的。然后使用EditText.addInputFilter
。
这样的输入过滤器的源代码在这里:https : //gist.github.com/CapnSpellcheck/7c72830e43927380daf5205100c93977
您可以在构造函数中传递0,并且不允许任何换行符。另外,您可以将其与其他调整之一结合使用,例如android:imeOptions="actionDone"
,因为这将有助于改善某些设备上的体验。