Eu também lutei por um bom tempo, mas finalmente encontrei uma solução!
Basta criar uma classe EditText personalizada como:
public class EditTextImeMultiline extends EditText {
public void init() {
addTextChangedListener(new TextWatcher() {
@Override
public void beforeTextChanged(CharSequence s, int start, int count, int after) {
}
@Override
public void onTextChanged(CharSequence s, int start, int before, int count) {
}
@Override
public void afterTextChanged(Editable s) {
for (int i = s.length(); i > 0; i--)
if (s.subSequence(i - 1, i).toString().equals("\n"))
s.replace(i - 1, i, "");
}
});
setSingleLine();
setHorizontallyScrolling(false);
this.getViewTreeObserver().addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
@Override
public void onGlobalLayout() {
EditTextImeMultiline.this.setLines(EditTextImeMultiline.this.getLineCount());
}
});
}
public EditTextImeMultiline(Context context) {
super(context);
init();
}
public EditTextImeMultiline(Context context, AttributeSet attrs) {
super(context, attrs);
init();
}
public EditTextImeMultiline(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
init();
}
@TargetApi(Build.VERSION_CODES.LOLLIPOP)
public EditTextImeMultiline(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {
super(context, attrs, defStyleAttr, defStyleRes);
init();
}
}
Esta classe remove lineBreaks (\ n), quebra o texto como textMultiline faria, E permite que você substitua o botão Enter por um ImeAction;).
Você só precisa chamá-lo em seu XML em vez da classe EditText clássica.
Para explicar a lógica aqui:
- Defina EditText como uma única linha para poder mostrar um botão ImeAction em vez de Enter.
- Remova a rolagem horizontal para fazer o texto ir para a próxima linha ao chegar ao final da visualização.
- Observe as mudanças de layout com o onGlobalLayoutListener e defina o parâmetro "line" para "lineCount" do texto atual mantido pelo editText. Isso é o que atualiza sua altura.