Java Android PostextChanged赢得';t每4个字符后输入一个空格
我正在尝试设计一个简单的信用卡/借记卡表单。我有一个cardNumber文本字段。当用户开始输入时,我希望我的应用程序在每4位数字后插入一个空格。我从教程中获得了帮助并做了一些修改,但它不起作用。它不是在每4位数字后输入空格 MainActivity.javaJava Android PostextChanged赢得';t每4个字符后输入一个空格,java,android,Java,Android,我正在尝试设计一个简单的信用卡/借记卡表单。我有一个cardNumber文本字段。当用户开始输入时,我希望我的应用程序在每4位数字后插入一个空格。我从教程中获得了帮助并做了一些修改,但它不起作用。它不是在每4位数字后输入空格 MainActivity.java package ************************* import android.content.pm.ActivityInfo; import android.os.Bundle; import android.sup
package *************************
import android.content.pm.ActivityInfo;
import android.os.Bundle;
import android.support.design.widget.FloatingActionButton;
import android.support.design.widget.Snackbar;
import android.support.v7.app.AppCompatActivity;
import android.support.v7.widget.Toolbar;
import android.text.Editable;
import android.text.TextUtils;
import android.text.TextWatcher;
import android.view.KeyEvent;
import android.view.View;
import android.view.Menu;
import android.view.MenuItem;
import android.widget.EditText;
import java.security.Key;
public class MainActivity extends AppCompatActivity {
private EditText cardNumber, expiryDate, CVV, nameOnCard;
private boolean flag = false;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
setSupportActionBar(toolbar);
setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_LOCKED);//Locks the screen orientation
init(); //Initializes the variables
typefunc();
}
@Override
public boolean onCreateOptionsMenu(Menu menu) {
// Inflate the menu; this adds items to the action bar if it is present.
getMenuInflater().inflate(R.menu.menu_main, menu);
return true;
}
@Override
public boolean onOptionsItemSelected(MenuItem item) {
// Handle action bar item clicks here. The action bar will
// automatically handle clicks on the Home/Up button, so long
// as you specify a parent activity in AndroidManifest.xml.
int id = item.getItemId();
//noinspection SimplifiableIfStatement
if (id == R.id.action_settings) {
return true;
}
return super.onOptionsItemSelected(item);
}
public void init()
{
cardNumber = (EditText) findViewById(R.id.cardNumber); //Field for storing the card number
expiryDate = (EditText) findViewById(R.id.expiryDate); //For storing the Expiry Date
CVV = (EditText) findViewById(R.id.CVV); //For storing the CVV
nameOnCard = (EditText) findViewById(R.id.nameOnCard); //For storing the name of the Cardholder
}
public void typefunc()
{
cardNumber.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)
{
if(s.length()==16)
{
expiryDate.setVisibility(View.VISIBLE);
expiryDate.requestFocus();
CVV.setVisibility(View.VISIBLE);
}
}
@Override
public void afterTextChanged(Editable s)
{
char space = ' ';
if (s.length() > 0 && (s.length() % 5) == 0)
{
char c = s.charAt(s.length() - 1);
if (Character.isDigit(c))
{
s.insert(s.length() - 1, String.valueOf(space));
}
}
}
});
expiryDate.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)
{
if(s.length()==6)
CVV.requestFocus();
}
@Override
public void afterTextChanged(Editable s) {
}
});
CVV.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)
{
if(s.length()==3) {
nameOnCard.setVisibility(View.VISIBLE);
}
}
@Override
public void afterTextChanged(Editable s)
{
if(s.length()==3)
nameOnCard.requestFocus();
}
});
}
}
这是因为
postTextChanged()
-文本更改后,这将异常!
调用此方法是为了通知您,在s中的某个位置,文本已更改 -文本更改时,这将异常
调用此方法是为了通知您,在s内,从开始处开始的计数字符刚刚替换了以前具有长度的旧文本。
更多信息 所以你可以这样做:
cardNumber.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)
{
if(cardNumber.length() == 4)
{
cardNumber.setText(cardNumber.getText() + " ");
}
}
@Override
public void afterTextChanged(Editable s)
{
}
});
注意:
setText()
也会触发onTextChanged()
,但由于长度现在是5,所以应该无关紧要。将代码从后文本更改方法中删除,并在onTextChanged方法中删除,如下所示
@Override
public void onTextChanged(CharSequence s, int start, int before, int count)
{
char space = ' ';
if (s.length() > 0 && (s.length() % 5) == 0)
{
char c = s.charAt(s.length() - 1);
if (Character.isDigit(c))
{
//s.insert(s.length() - 1, String.valueOf(space));
cardNumber.append(String.valueOf(space));
}
}
if(s.length()==16)
{
expiryDate.setVisibility(View.VISIBLE);
expiryDate.requestFocus();
CVV.setVisibility(View.VISIBLE);
}
}
试试这个:
cardNumber.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) {
if(s.length()==16) {
expiryDate.setVisibility(View.VISIBLE);
expiryDate.requestFocus();
CVV.setVisibility(View.VISIBLE);
}
}
@Override
public void afterTextChanged(Editable s) {
String txt = s.toString();
s.clear();
s.insert(0, txt.replaceAll("\\D","").replaceAll("(\\d{4}(?!$))","$1-"));
}
});
这很好用
editText.addTextChangedListener(new TextWatcher() {
private static final char space = ' ';
@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) {
// Remove spacing char
if (s.length() > 0 && (s.length() % 5) == 0) {
final char c = s.charAt(s.length() - 1);
if (space == c) {
s.delete(s.length() - 1, s.length());
}
}
// Insert char where needed.
if (s.length() > 0 && (s.length() % 5) == 0) {
char c = s.charAt(s.length() - 1);
// Only if its a digit where there should be a space we insert a space
if (Character.isDigit(c) && TextUtils.split(s.toString(), String.valueOf(space)).length <= 3) {
s.insert(s.length() - 1, String.valueOf(space));
}
}
}
});
希望这有帮助:)当用户更改内容时,您想要空间,而不是在e上,它已经完成了,对吗?!那么为什么不把它放在onTextChanged
s.length()%5
返回0
如果s.length()
是5
的倍数而不是4
@silverFoxA,则不允许从onTextChanged()
更改s
的内容。只能从PostTextChanged()
完成。为什么要修改??您提到的问题与您的目的相同,即在android中以编辑文本的形式格式化信用卡。那么问题出在哪里?可能与onTextChanged()
文档中的重复:“尝试从此回调对s
进行更改是错误的。”。也看看这个答案:我不明白你的意思onTextChanged()
用于立即对文本进行更改(当文本正在更改时)。不,它不是用于更改文本,而是用于观察更改。@Strider试图从onTextChanged()
更改s
,会引发错误<代码>s只能从后文本更改()
插入()
函数不允许从上文本更改()
直接使用cardNumber.setText()而不是插入检查我的应答输入类型“phone”不是有效类型。因为它也允许特殊的字符。
android:inputType="phone" android:maxLength="19"