将光标放在EditText的文本末尾


883

我正在更改EditTexton 的值keyListener

但是,当我更改文本时,光标会移到的开头EditText。我需要将光标放在文本的末尾。

如何将光标移动到文本中的末尾EditText


3
我遇到了同样的问题。但是我更好地问自己是为什么会这样,所以我可以先解决问题,而不必自己移动插入符号。
kaneda

3
@kaneda我完全同意,但是如果您添加了实际的解决方案将对您有所帮助。
AgentKnopf

1
@Zainodis这只是一个想法。就像我说的那样,我遇到了相同的问题,这不一定意味着我已经找到了解决方案。就我而言,EditTexts 属于s 的问题ListView。至于实验,我对ListView源代码本身(这是一个相当复杂的野兽)进行了一些更改,并在模拟器上进行了测试。它与组件进行的焦点控制管理有关。所以,这当然不是我可以帮助我们朋友的解决方案。:)
kaneda 2013年


这在OnFocusChanged回调内部不起作用。那里的解决方案是将setSelection放在可运行的内部并在主线程上运行它。在此处检查stackoverflow.com/a/32156989/4514796
Ali Nem

Answers:


1317

尝试这个:

EditText et = (EditText)findViewById(R.id.inbox);
et.setSelection(et.getText().length());

15
在我的情况下这不起作用。setSelection()方法似乎无效。我的EditText视图包含ImageSpans。还有其他解决方法吗?
toobsco42

@marqss,我遇到了同样的问题,对我来说工作非常完美。我在对话框上使用EditText,并在主屏幕上预先填充了文本。发生这种情况时,光标停留在开头而不是结尾,但是在我尝试了您的建议之后,一切都很好,就是我想要的方式。感谢您发布此信息。

4
嘿,@ Mullay,我现在通过重写类的onSelectionChanged()方法来处理它EditTextpublic void onSelectionChanged(int start, int end) { CharSequence text = getText(); if (text != null) { if (start != text.length() || end != text.length()) { setSelection(text.length(), text.length()); return; } } super.onSelectionChanged(start, end); }
toobsco42 2013年

19
您可能必须使用et.post(new Runnable({... et.setSel ...才能进入队列。这是因为android等待通过发布来进行一些布局工作直到更好的时间,所以如果您尝试setSelection该系统完成之前,将撤消你的工作。
MinceMan

1
它在这里有效,感谢您的回答。但是您应该这样做:editText.setText(s); editText.setSelection(editText.getText()。length()); // setText之后
smileVann 2014年

223

ediitext有一个名为append的函数,该函数将字符串值附加到当前的edittext值,并将光标置于该值的末尾。您可以将字符串值作为当前ediitext值本身,并调用append();。

myedittext.append("current_this_edittext_string"); 

1
如果您的EditText中已经有文本,只需编写myedittext.append("");
-ymerdrengene

4
@ymerdrengene只是申请myedittext.append("");对我没有用!
Chintan Shah 2015年

4
这是完美的,因为它仅替换了我的代码行。正在使用edittextfield.setText并将其交换为追加。很简单。
–blblitz

取决于用法是否可行。对我来说不是。使用Selection是实现该IMO的正确方法。
sud007

139

科特林

将光标设置到起始位置

val editText = findViewById(R.id.edittext_id) as EditText
editText.setSelection(0)

将光标设置在EditText末尾

val editText = findViewById(R.id.edittext_id) as EditText
editText.setSelection(editText.text.length)

下面的代码是将光标放在第二个字符之后:

val editText = findViewById(R.id.edittext_id) as EditText
editText.setSelection(2)

JAVA

将光标设置到起始位置

 EditText editText = (EditText)findViewById(R.id.edittext_id);
 editText.setSelection(0);

将光标设置在EditText末尾

EditText editText = (EditText)findViewById(R.id.edittext_id);
editText.setSelection(editText.getText().length());

下面的代码是将光标放在第二个字符之后:

EditText editText = (EditText)findViewById(R.id.edittext_id);
editText.setSelection(2);

1
如果使用的话,可以替换为铸模findViewById(R.id.edittext_id) as EditTextfindViewById<EditText>(R.id.edittext_id)或者只是避免铸模API 26+
Evin1_18年

95

如果您setText之前调用过,而新文本没有setSelection在由触发的单独可运行对象中进行布局阶段调用View.post(Runnable)(请从主题重新发布)。

因此,对我来说,这段代码有效:

editText.setText("text");
editText.post(new Runnable() {
         @Override
         public void run() {
             editText.setSelection(editText.getText().length());
         }
});

编辑05/16/2019:现在我正在为此使用Kotlin扩展名:

fun EditText.placeCursorToEnd() {
    this.setSelection(this.text.length)
}

然后-editText.placeCursorToEnd()。


谢谢,这真的为我工作。我正在尝试调用setSelection,但是它不起作用。就我而言,我试图设置一个固定的前缀来编辑文本,因此在设置前缀之后,我需要将光标设置为前缀的末尾,以便用户可以在前缀之后输入数据。由于我在TextWatcher的构造函数中进行了setSelection操作,因此可能就是这样。但是您的解决方案效果很好!
Wand Maker

我在MultiAutoCompleteTextView的Clickablespan方法中使用了此post方法,它可以工作。..我希望光标一旦在文本范围内单击删除图标后就在最后移动。它适用于所有最新的android手机。 ..但是在旧版本中,第一次点击后需要一些时间...我所有其他项目都无法点击...任何提示?
VijayRaj

1
我不知道为什么,但这是唯一对我有用的!我的编辑文本具有焦点/更改了文本的侦听器,并包装在TextInputLayout中。这些事情之一就是搞砸了。
丹尼尔·威尔逊,

42

您还可以将光标放在EditText视图中文本的末尾,如下所示:

EditText et = (EditText)findViewById(R.id.textview);
int textLength = et.getText().length();
et.setSelection(textLength, textLength);

27
editText.setOnKeyListener(new View.OnKeyListener() {
    @Override
    public boolean onKey(View v, int keyCode, KeyEvent event) {
        editText.setSelection(editText.getText().length());
        return false;
    }
});

21

这是另一种可能的解决方案:

et.append("");

如果由于某种原因它不起作用,请尝试以下解决方案:

et.setSelection(et.getText().length());

12
/**
 * Set cursor to end of text in edittext when user clicks Next on Keyboard.
 */
View.OnFocusChangeListener onFocusChangeListener = new View.OnFocusChangeListener() {
    @Override
    public void onFocusChange(View view, boolean b) {
        if (b) {
            ((EditText) view).setSelection(((EditText) view).getText().length());
        }
    }
};

mEditFirstName.setOnFocusChangeListener(onFocusChangeListener); 
mEditLastName.setOnFocusChangeListener(onFocusChangeListener);

它对我有用!


12

就我而言,我创建了以下kotlin分机。功能,可能对某人有用

 private fun EditText.focus(){
        requestFocus()
        setSelection(length())
    }

然后使用如下

mEditText.focus()

12

如果您的EditText不清楚:

editText.setText("");
editText.append("New text");

要么

editText.setText(null);
editText.append("New text");


10

我认为这可以实现您想要的。

 Editable etext = mSubjectTextEditor.getText();
 Selection.setSelection(etext, etext.length());

如果我的观察正确,则为每个小部件生成一个android.widget.TextView $ IClipboardDataPasteEventImpl实例。我最终得到了很多。显然,它们最终得到了清理,但是似乎增加了内存消耗。我有一个对话框的12个实例,它只是上述实例的主导者(保持它处于活动状态)。
AgentKnopf

将这段代码放置在代码中的适当位置(在此处将文本直接附加到EText)。谢谢!
sud007'7

9

这个问题已经解决了,但是如果您想使用新发布的Android数据绑定工具,只需在XML中进行设置即可,获得这个答案可能会很有用:

<data>
    <variable name="stringValue" type="String"/>
</data>
...
<EditText
        ...
        android:text="@={stringValue}"
        android:selection="@{stringValue.length()}"
        ...
/>

6

在此处输入图片说明

你好试试看

 <EditText
       android:id="@+id/edt_text"
       android:layout_width="match_parent"
       android:layout_height="wrap_content"
       android:text="Hello World!"
       android:cursorVisible="true"/>

EditText editText = findViewById(R.id.editText); editText.setSelection(editText.getText().length()); // End point光标


5

如果要选择所有文本,而只输入新文本而不是旧文本,则可以使用

    android:selectAllOnFocus="true"

5

这有效

Editable etext = edittext.getText();
Selection.setSelection(etext,edittext.getText().toString().length());

5

editText.setSelection是这里的魔力。基本上,选择可使您将光标放置在所需的任何位置。

EditText editText = findViewById(R.id.editText);
editText.setSelection(editText.getText().length());

这会将光标置于EditText的末尾。基本上editText.getText().length()给您文本长度。然后,您可以使用setSelection长度。

editText.setSelection(0);

用于将光标设置在起始位置(0)。


4

我测试过的所有其他代码均无法正常工作,原因是用户仍然可以将插入号/光标放置在字符串中间的任意位置(例如:12 | 3.00-其中|是光标)。每当在EditText上发生触摸时,我的解决方案始终将光标置于字符串的末尾。

最终的解决方案是:

// For a EditText like:
<EditText
                android:id="@+id/EditTextAmount"
                android:layout_height="wrap_content"
                android:layout_width="fill_parent"
                android:hint="@string/amount"
                android:layout_weight="1"
                android:text="@string/zero_value"
                android:inputType="text|numberDecimal"
                android:maxLength="13"/>

@ string / amount =“ 0.00” @ string / zero_value =“ 0.00”

// Create a Static boolean flag
private static boolean returnNext; 


// Set caret/cursor to the end on focus change
EditTextAmount.setOnFocusChangeListener(new View.OnFocusChangeListener() {
            @Override
            public void onFocusChange(View editText, boolean hasFocus) {
                if(hasFocus){
                    ((EditText) editText).setSelection(((EditText) editText).getText().length());
                }
            }
        });

// Create a touch listener and put caret to the end (no matter where the user touched in the middle of the string)
EditTextAmount.setOnTouchListener(new View.OnTouchListener() {
            @Override
            public boolean onTouch(View editText, MotionEvent event) {
                ((EditText) editText).onTouchEvent(event);
                ((EditText) editText).setSelection(((EditText) editText).getText().length());
                return true;
            }
        });


// Implement a Currency Mask with addTextChangedListener
EditTextAmount.addTextChangedListener(new TextWatcher() {
            @Override
            public void onTextChanged(CharSequence s, int start, int before, int count) {
                String input = s.toString();
                String output = new String();
                String buffer = new String();
                String decimals = new String();
                String numbers = Integer.toString(Integer.parseInt(input.replaceAll("[^0-9]", "")));

                if(returnNext){
                    returnNext = false;
                    return;
                }

                returnNext = true;

                if (numbers.equals("0")){
                    output += "0.00";
                }
                else if (numbers.length() <= 2){
                    output += "0." + String.format("%02d", Integer.parseInt(numbers));
                }
                else if(numbers.length() >= 3){
                    decimals = numbers.substring(numbers.length() - 2);
                    int commaCounter = 0;
                    for(int i=numbers.length()-3; i>=0; i--){
                        if(commaCounter == 3){
                            buffer += ",";
                            commaCounter = 0;
                        }
                        buffer += numbers.charAt(i);
                        commaCounter++;
                    }
                    output = new StringBuilder(buffer).reverse().toString() + "." + decimals;
                }
                EditTextAmount.setText(output);
                EditTextAmount.setSelection(EditTextAmount.getText().length());
            }

            @Override
            public void beforeTextChanged(CharSequence s, int start, int count, int after) {
                /*String input = s.toString();
                if(input.equals("0.0")){
                    EditTextAmount.setText("0.00");
                    EditTextAmount.setSelection(EditTextAmount.getText().length());
                    return;
                }*/
            }

            @Override
            public void afterTextChanged(Editable s) {

            }
        });

希望能帮助到你!


3

对于ViewModel,LiveData和数据绑定

我需要EditText在Notes应用程序中提供多行支持的功能。当用户导航到具有注释文本的片段时,我希望光标位于文本的末尾。

djleop建议的解决方案接近。但是,这样做的问题是,如果用户将光标放在文本中间的某个位置以进行编辑并开始输入,则光标会再次跳到文本结尾。发生这种情况是因为LiveData会发出新值,并且光标会再次跳到文本的末尾,导致用户无法在中间的某个位置编辑文本。

为了解决这个问题,我使用一个标志并仅为MediatorLiveData其分配了String一次长度。这将使LiveData仅读取一次值,即,当用户导航到片段时。之后,用户可以将光标放置在要在其中编辑文本的任何位置。

视图模型

private var accessedPosition: Boolean = false

val cursorPosition = MediatorLiveData<Event<Int>>().apply {
    addSource(yourObject) { value ->
        if(!accessedPosition) {
            setValue(Event(yourObject.note.length))
            accessedPosition = true
        }
    }
}

这里yourObject是从数据库检索的另一个LiveData,其中包含您在中显示的String文本EditText

然后MediatorLiveData使用绑定适配器将此绑定到您的EditText。

XML格式

使用双向数据绑定来显示文本以及接受文本输入。

<!-- android:text must be placed before cursorPosition otherwise we'll get IndexOutOfBounds exception-->
<EditText
    android:text="@={viewModel.noteText}"
    cursorPosition="@{viewModel.cursorPosition}" />

绑定适配器

@BindingAdapter("cursorPosition")
fun bindCursorPosition(editText: EditText, event: Event<Int>?) {
    event?.getContentIfNotHandled()?.let { editText.setSelection(it) }
}

Event

Event这里的类就像Google的JoseAlcérreca编写的SingleLiveEvent。我在这里使用它来照顾屏幕旋转。Event当用户在中间的某个位置编辑文本并旋转屏幕时,使用单个将确保光标不会跳到文本的末尾。屏幕旋转时它将保持相同的位置。

这是Event课程:

open class Event<out T>(private val content: T) {

    var hasBeenHandled = false
        private set // Allow external read but not write

    /**
     * Returns the content and prevents its use again.
     */
    fun getContentIfNotHandled(): T? {
        return if (hasBeenHandled) {
            null
        } else {
            hasBeenHandled = true
            content
        }
    }

    /**
     * Returns the content, even if it's already been handled.
     */
    fun peekContent(): T = content
}

这是对我有效的解决方案,并提供了良好的用户体验。希望它也对您的项目有所帮助。


这个答案需要提高。非常感谢
Abdul Rahman Shamair

2

类似于@Anh Duy的答案,但对我没有用。我还需要仅当用户点击编辑文本时光标才移到末尾,并且之后仍然能够选择光标的位置,这是唯一对我有用的代码

boolean textFocus = false; //define somewhere globally in the class

//in onFinishInflate() or somewhere
editText.setOnTouchListener(new OnTouchListener() {
    @Override
    public boolean onTouch(View v, MotionEvent event) {

        editText.onTouchEvent(event);

        if(!textFocus) {
            editText.setSelection(editText.getText().length());
            textFocus = true;
        }

        return true;
    }
});

editText.setOnFocusChangeListener(new OnFocusChangeListener() {
    @Override
    public void onFocusChange(View v, boolean hasFocus) {
        textFocus = false;
    }
});



2

以上答案对我没有用。所以我找到了一个新的解决方案。这可能对某人有帮助。我一直在使用最新版本的Android Studio,即当前日期为3.5。也许这就是上述答案未显示任何效果的原因。

码:

EditText available_seats = findViewById(R.id.available_seats);
Selection.setSelection(available_seats.getText(),available_seats.getText().length());

在这里,第一个参数是您要播放的Spannable文本值,而第二个参数是索引值。因为我们希望光标位于文本的末尾,所以我们使用了getText()。length()来返回文本的长度。


1
我的荣幸!编码愉快!
Prajwal W

1
public class CustomEditText extends EditText {
    public CustomEditText(Context context, AttributeSet attrs) {
        super(context, attrs);
    }

    public CustomEditText(Context context) {
        super(context);
    }

    public CustomEditText(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
    }


    @Override
    protected void onFocusChanged(boolean focused, int direction, Rect previouslyFocusedRect) {
        super.onFocusChanged(focused, direction, previouslyFocusedRect);
        this.setSelection(this.getText().length());
    }

    @Override
    protected void onSelectionChanged(int selStart, int selEnd) {

    }
}

在XML文件中将此CustomEditText用户使用时,它将起作用。我已经对此进行了测试,它正在为我工​​作。


1

如果要将光标放在EditText视图中的文本末尾

 EditText rename;
 String title = "title_goes_here";
 int counts = (int) title.length();
 rename.setSelection(counts);
 rename.setText(title);
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.