Java Android PostextChanged赢得';t每4个字符后输入一个空格

Java 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

我正在尝试设计一个简单的信用卡/借记卡表单。我有一个cardNumber文本字段。当用户开始输入时,我希望我的应用程序在每4位数字后插入一个空格。我从教程中获得了帮助并做了一些修改,但它不起作用。它不是在每4位数字后输入空格

MainActivity.java

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"