Para Birimi Girdisini Biçimlendirmenin Daha İyi Bir Yolu editText?


91

Bir editText'im var, başlangıç ​​değeri 0,00 $. 1'e bastığınızda 0,01 $ olarak değişir. 4'e basın, 0,14 dolara gider. 8'e 1.48 $ basın. Geri tuşuna basın, 0,14 ABD doları vb.

Bu işe yarıyor, sorun şu ki, biri imleci manuel olarak konumlandırırsa, biçimlendirmede sorunlar ortaya çıkar. Ondalık sayıları sileceklerse, geri gelmez. İmleci ondalık sayının önüne koyarlarsa ve 2 yazarlarsa, $ 2.00 yerine $ 02.00 gösterecektir. $ 'I silmeye çalışırlarsa, bunun yerine örneğin bir rakam siler.

İşte kullandığım kod, herhangi bir öneri için minnettarım.

mEditPrice.setRawInputType(Configuration.KEYBOARD_12KEY);
    public void priceClick(View view) {
    mEditPrice.addTextChangedListener(new TextWatcher(){
        DecimalFormat dec = new DecimalFormat("0.00");
        @Override
        public void afterTextChanged(Editable arg0) {
        }
        @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.toString().matches("^\\$(\\d{1,3}(\\,\\d{3})*|(\\d+))(\\.\\d{2})?$"))
            {
                String userInput= ""+s.toString().replaceAll("[^\\d]", "");
                if (userInput.length() > 0) {
                    Float in=Float.parseFloat(userInput);
                    float percen = in/100;
                    mEditPrice.setText("$"+dec.format(percen));
                    mEditPrice.setSelection(mEditPrice.getText().length());
                }
            }
        }
    });

1
Cehaletimi bağışlayın, ancak bu kod parçacığı etkinlik yaşam döngüsü yöntemlerinden birinden mi yoksa sizin yaptığınız özel bir sınıfta mı? Lütfen daha eksiksiz bir kod örneği verebilir misiniz? Teşekkürler!
Argus9

Bu benim için çalışıyor Bu harici
lib'yi

Yanıtlar:


153

Yöntemi test ettim, ancak çok sayıda kullandığımda başarısız oluyor ... Bunu ben oluşturdum:

private String current = "";
@Override
public void onTextChanged(CharSequence s, int start, int before, int count) {
    if(!s.toString().equals(current)){
       [your_edittext].removeTextChangedListener(this);

       String cleanString = s.toString().replaceAll("[$,.]", "");
                
       double parsed = Double.parseDouble(cleanString);
       String formatted = NumberFormat.getCurrencyInstance().format((parsed/100));
                    
       current = formatted;
       [your_edittext].setText(formatted);
       [your_edittext].setSelection(formatted.length());
       
       [your_edittext].addTextChangedListener(this);
    }
}

Kotlin varyantı:

private var current: String = ""

         override fun onTextChanged(
            s: CharSequence,
            start: Int,
            before: Int,
            count: Int
        ) {
            if (s.toString() != current) {
                discount_amount_edit_text.removeTextChangedListener(this)

                val cleanString: String = s.replace("""[$,.]""".toRegex(), "")

                val parsed = cleanString.toDouble()
                val formatted = NumberFormat.getCurrencyInstance().format((parsed / 100))

                current = formatted
                discount_amount_edit_text.setText(formatted)
                discount_amount_edit_text.setSelection(formatted.length)

                discount_amount_edit_text.addTextChangedListener(this)
            }
        }

36
Dolar sembolü varsaymaktansa aşağıdakileri yapmak daha iyi olabilir: String replaceable = String.format("[%s,.]", NumberFormat.getCurrencyInstance().getCurrency().getSymbol()); String cleanString = s.toString().replaceAll(replaceable, "");
craigp

6
: Hmm, aslında replaceAll gelen regex desen sıra sap mekanlara, bu gibi görünmelidir, şimdi bu kendimi denedim String replaceable = String.format("[%s,.\\s]", NumberFormat.getCurrencyInstance().getCurrency().getSymbol());
craigp

6
AfterTextChanged () 'de değişiklik yapılmaması tavsiye edilmiyoronTextChanged() and rather to do so in mu
codinguser

3
Metin değiştirilen dinleyicinin neden kaldırıldığını ve ardından her seferinde yeniden eklendiğini bilmekle ilgileniyorum. Benim için sadece bir kez eklenirse işe yarar (ve değişiklikleri afterTextChanged'e taşıdım)
Daniel Wilson

5
1,00 almak için 1 -> 0 -> 0 koyduğunuzda çalışmıyorum. Bunun nedeni, 0,1'in 010 dizgesine dönüştüğü ve 010'un double10 olduğu noktaya 10 / 100 = 0,1gelmenizdir. Onu geçemezsiniz.
JakubW

30

Yukarıdaki cevaplardan bazılarına dayanarak, aşağıdaki gibi kullanacağınız bir MoneyTextWatcher oluşturdum:

priceEditText.addTextChangedListener(new MoneyTextWatcher(priceEditText));

ve işte sınıf:

public class MoneyTextWatcher implements TextWatcher {
    private final WeakReference<EditText> editTextWeakReference;

    public MoneyTextWatcher(EditText editText) {
        editTextWeakReference = new WeakReference<EditText>(editText);
    }

    @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 editable) {
        EditText editText = editTextWeakReference.get();
        if (editText == null) return;
        String s = editable.toString();
        if (s.isEmpty()) return;
        editText.removeTextChangedListener(this);
        String cleanString = s.replaceAll("[$,.]", "");
        BigDecimal parsed = new BigDecimal(cleanString).setScale(2, BigDecimal.ROUND_FLOOR).divide(new BigDecimal(100), BigDecimal.ROUND_FLOOR);
        String formatted = NumberFormat.getCurrencyInstance().format(parsed);
        editText.setText(formatted);
        editText.setSelection(formatted.length());
        editText.addTextChangedListener(this);
    }
}

Bunu bir süredir kullanıyorum ama son zamanlarda küçük bir sorun buldum, bazı klavyelerde sil düğmesini basılı tutarsanız tüm kelimeyi / metin grubunu siler ve neden olurjava.lang.NumberFormatException: Bad offset/length
BluGeni

1
Benim için mükemmel çalıştı! 'EditText.setSelection (formatted.length ());' dikkatine söz konusu EditText'in 'maxLength' özelliğine uyulmalıdır. maxLength == 13; formatted.length () == 14; "Formatted.length", "maxLength" değerinden büyükse, aşağıdaki hata oluşur: IndexOutOfBoundsException: setSpan (14 ... 14), 13
tk'nin

1
@BluGeni, eğer (s.isEmpty ()) dönerse metin değiştirme dinleyicisini kaldırmadan önce bir s.isEmpty kontrolü eklemeyi düzeltmek için; editText.removeTextChangedListener (this); Ayrıca cleanString satırında, s.toString () gereksiz
Mike Baglio Jr.

1
tek bir öneriden daha iyi yanıt, .replaceAll ("[$ ...) için -> .replaceAll (" [^ \\ d.] "," ") değiştirmektir; ben başka bir para birimim olduğundan, yalnızca $, benim durumumda olduğu gibi R $ (Brezilya)
user2582318

1
üzgünüm doğru öneri bu -> .replaceAll("[^0-9]", ""), yukarıdakinin sınırı 9.999.999 -_-
user2582318

21

İşte benim geleneğim CurrencyEditText

import android.content.Context;import android.graphics.Rect;import android.text.Editable;import android.text.InputFilter;import android.text.InputType;import android.text.TextWatcher;
import android.util.AttributeSet;import android.widget.EditText;import java.math.BigDecimal;import java.math.RoundingMode;
import java.text.DecimalFormat;import java.text.DecimalFormatSymbols;
import java.util.Locale;

/**
 * Some note <br/>
 * <li>Always use locale US instead of default to make DecimalFormat work well in all language</li>
 */
public class CurrencyEditText extends android.support.v7.widget.AppCompatEditText {
    private static String prefix = "VND ";
    private static final int MAX_LENGTH = 20;
    private static final int MAX_DECIMAL = 3;
    private CurrencyTextWatcher currencyTextWatcher = new CurrencyTextWatcher(this, prefix);

    public CurrencyEditText(Context context) {
        this(context, null);
    }

    public CurrencyEditText(Context context, AttributeSet attrs) {
        this(context, attrs, android.support.v7.appcompat.R.attr.editTextStyle);
    }

    public CurrencyEditText(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        this.setInputType(InputType.TYPE_CLASS_NUMBER | InputType.TYPE_NUMBER_FLAG_DECIMAL);
        this.setHint(prefix);
        this.setFilters(new InputFilter[] { new InputFilter.LengthFilter(MAX_LENGTH) });
    }

    @Override
    protected void onFocusChanged(boolean focused, int direction, Rect previouslyFocusedRect) {
        super.onFocusChanged(focused, direction, previouslyFocusedRect);
        if (focused) {
            this.addTextChangedListener(currencyTextWatcher);
        } else {
            this.removeTextChangedListener(currencyTextWatcher);
        }
        handleCaseCurrencyEmpty(focused);
    }

    /**
     * When currency empty <br/>
     * + When focus EditText, set the default text = prefix (ex: VND) <br/>
     * + When EditText lose focus, set the default text = "", EditText will display hint (ex:VND)
     */
    private void handleCaseCurrencyEmpty(boolean focused) {
        if (focused) {
            if (getText().toString().isEmpty()) {
                setText(prefix);
            }
        } else {
            if (getText().toString().equals(prefix)) {
                setText("");
            }
        }
    }

    private static class CurrencyTextWatcher implements TextWatcher {
        private final EditText editText;
        private String previousCleanString;
        private String prefix;

        CurrencyTextWatcher(EditText editText, String prefix) {
            this.editText = editText;
            this.prefix = prefix;
        }

        @Override
        public void beforeTextChanged(CharSequence s, int start, int count, int after) {
            // do nothing
        }

        @Override
        public void onTextChanged(CharSequence s, int start, int before, int count) {
            // do nothing
        }

        @Override
        public void afterTextChanged(Editable editable) {
            String str = editable.toString();
            if (str.length() < prefix.length()) {
                editText.setText(prefix);
                editText.setSelection(prefix.length());
                return;
            }
            if (str.equals(prefix)) {
                return;
            }
            // cleanString this the string which not contain prefix and ,
            String cleanString = str.replace(prefix, "").replaceAll("[,]", "");
            // for prevent afterTextChanged recursive call
            if (cleanString.equals(previousCleanString) || cleanString.isEmpty()) {
                return;
            }
            previousCleanString = cleanString;

            String formattedString;
            if (cleanString.contains(".")) {
                formattedString = formatDecimal(cleanString);
            } else {
                formattedString = formatInteger(cleanString);
            }
            editText.removeTextChangedListener(this); // Remove listener
            editText.setText(formattedString);
            handleSelection();
            editText.addTextChangedListener(this); // Add back the listener
        }

        private String formatInteger(String str) {
            BigDecimal parsed = new BigDecimal(str);
            DecimalFormat formatter =
                    new DecimalFormat(prefix + "#,###", new DecimalFormatSymbols(Locale.US));
            return formatter.format(parsed);
        }

        private String formatDecimal(String str) {
            if (str.equals(".")) {
                return prefix + ".";
            }
            BigDecimal parsed = new BigDecimal(str);
            // example pattern VND #,###.00
            DecimalFormat formatter = new DecimalFormat(prefix + "#,###." + getDecimalPattern(str),
                    new DecimalFormatSymbols(Locale.US));
            formatter.setRoundingMode(RoundingMode.DOWN);
            return formatter.format(parsed);
        }

        /**
         * It will return suitable pattern for format decimal
         * For example: 10.2 -> return 0 | 10.23 -> return 00, | 10.235 -> return 000
         */
        private String getDecimalPattern(String str) {
            int decimalCount = str.length() - str.indexOf(".") - 1;
            StringBuilder decimalPattern = new StringBuilder();
            for (int i = 0; i < decimalCount && i < MAX_DECIMAL; i++) {
                decimalPattern.append("0");
            }
            return decimalPattern.toString();
        }

        private void handleSelection() {
            if (editText.getText().length() <= MAX_LENGTH) {
                editText.setSelection(editText.getText().length());
            } else {
                editText.setSelection(MAX_LENGTH);
            }
        }
    }
}

XML'de şöyle kullanın

 <...CurrencyEditText
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        />

Aşağıdaki 2 sabiti projenize uygun şekilde düzenlemelisiniz

private static String prefix = "VND ";
private static final int MAX_DECIMAL = 3;

görüntü açıklamasını buraya girin

GitHub'da demo


2
Bu harika!
YTerle

1
Maksimum ondalık basamak sayısına ulaşmak için yazdıktan sonra 5-9 sayısını girmeye çalışmanın son ondalık basamağı 1 artıracağını buldum ... yuvarlanıyor! Benim düzeltme aramak oldu formatter.setRoundingMode(RoundingMode.DOWN);yılında formatDecimalyöntemle.
BW

@bwicks sorunu bulduğunuz için çok teşekkür ederim. Düzenlemenizi onayladım
Phan Van Linh

VND'nin para birimi simgesi örneği nasıl yerleştirilir?
Mayur Karmur

1
Bir başka iyileştirme fikri: Kullanıcı girerse $., ham değeri .Double olarak aldığımızda ve ayrıştırdığımızda NFE verir. Düzeltme için, benim yaptığım formatDecimal()dönüş için prefix + "0.";ve değiştirilemez #,###.için #,##0.içeriden formatDecimal(). Bu ayrıca, kullanıcı yalnızca ondalık basamaklar girdiğinde daha iyi görünür. Sanki gösterir $0.25yerine $.25.
Gökhan Arık

13

Aslında daha önce sağlanan çözüm işe yaramıyor. 100.00 girmek isterseniz çalışmaz.

Değiştirin:

double parsed = Double.parseDouble(cleanString);
String formato = NumberFormat.getCurrencyInstance().format((parsed/100));

İle:

BigDecimal parsed = new BigDecimal(cleanString).setScale(2,BigDecimal.ROUND_FLOOR).divide(new BigDecimal(100),BigDecimal.ROUND_FLOOR);                
String formato = NumberFormat.getCurrencyInstance().format(parsed);

Kodum için bazı değişiklikler yaptığımı söylemeliyim. Önemli olan şu ki, BigDecimal'ın


6

TextWatcher'ı uygulayarak Brezilya para birimi biçimlerini kullanacak ve değeri düzenlerken imleç konumunu ayarlayarak sınıfı değiştiriyorum.

public class MoneyTextWatcher TextWatcher'ı uygular {

    özel EditText editText;

    private String lastAmount = "";

    private int lastCursorPosition = -1;

    public MoneyTextWatcher (EditText editText) {
        Süper();
        this.editText = editText;
    }

    @Override
    public void onTextChanged (CharSequence miktarı, int start, int before, int count) {

        if (! amount.toString (). equals (lastAmount)) {

            String cleanString = clearCurrencyToNumber (amount.toString ());

            Deneyin {

                String formattedAmount = transformToCurrency (cleanString);
                editText.removeTextChangedListener (this);
                editText.setText (formattedAmount);
                editText.setSelection (formattedAmount.length ());
                editText.addTextChangedListener (this);

                if (lastCursorPosition! = lastAmount.length () && lastCursorPosition! = -1) {
                    int lengthDelta = formattedAmount.length () - lastAmount.length ();
                    int newCursorOffset = max (0, min (formattedAmount.length (), lastCursorPosition + lengthDelta));
                    editText.setSelection (newCursorOffset);
                }
            } catch (İstisna e) {
               // bir şeyi günlüğe kaydet
            }
        }
    }

    @Override
    public void afterTextChanged (Düzenlenebilir s) {
    }

    @Override
    public void beforeTextChanged (CharSequence s, int start, int count, int after) {
        Dize değeri = s.toString ();
        eğer (! value.equals ("")) {
            String cleanString = clearCurrencyToNumber (değer);
            String formattedAmount = transformToCurrency (cleanString);
            lastAmount = formattedAmount;
            lastCursorPosition = editText.getSelectionStart ();
        }
    }

    public static String clearCurrencyToNumber (String currencyValue) {
        Dize sonucu = boş;

        if (currencyValue == null) {
            sonuç = "";
        } Başka {
            sonuç = currencyValue.replaceAll ("[(az) | (AZ) | ($ ,.)]", "");
        }
        dönüş sonucu;
    }

    public static boolean isCurrencyValue (String currencyValue, boolean podeSerZero) {
        boole sonucu;

        eğer (currencyValue == null || currencyValue.length () == 0) {
            sonuç = yanlış;
        } Başka {
            eğer (! podeSerZero && currencyValue.equals ("0,00")) {
                sonuç = yanlış;
            } Başka {
                sonuç = doğru;
            }
        }
        dönüş sonucu;
    }

    public static String transformToCurrency (Dize değeri) {
        çift ​​ayrıştırılmış = Double.parseDouble (değer);
        String formatted = NumberFormat.getCurrencyInstance (yeni Yerel Ayar ("pt", "BR")). Format ((ayrıştırılmış / 100));
        formatted = formatted.replaceAll ("[^ (0-9) (.,)]", "");
        biçimlendirilmiş dönüş;
    }
}

Bu satırda "int newCursorOffset = max (0, min (formattedAmount.length (), lastCursorPosition + lengthDelta));" max ve min ne tür bir nesnedir?
Arthur Melo

2
@ArthurMelo Its, Math.max, Math.min Kod için teşekkürler ve edittext'den virgül kaldırılırken başarısız görünüyor.
Marcos Vasconcelos

4

Guilhermes cevabını temel aldım, ancak imlecin konumunu korudum ve ayrıca dönemleri farklı şekilde ele alıyorum - bu şekilde, bir kullanıcı noktadan sonra yazıyorsa, dönemden önceki sayıları etkilemiyor, bunun çok düzgün bir girdi sağladığını düşünüyorum .

    [yourtextfield].addTextChangedListener(new TextWatcher()
    {
        NumberFormat currencyFormat = NumberFormat.getCurrencyInstance();
        private String current = "";

        @Override
        public void onTextChanged(CharSequence s, int start, int before, int count)
        {
            if(!s.toString().equals(current))
            {
                   [yourtextfield].removeTextChangedListener(this);

                   int selection = [yourtextfield].getSelectionStart();


                   // We strip off the currency symbol
                   String replaceable = String.format("[%s,\\s]", NumberFormat.getCurrencyInstance().getCurrency().getSymbol());
                   String cleanString = s.toString().replaceAll(replaceable, "");

                   double price;

                   // Parse the string                     
                   try
                   {
                       price = Double.parseDouble(cleanString);
                   }
                   catch(java.lang.NumberFormatException e)
                   {
                       price = 0;
                   }

                   // If we don't see a decimal, then the user must have deleted it.
                   // In that case, the number must be divided by 100, otherwise 1
                   int shrink = 1;
                   if(!(s.toString().contains(".")))
                   {
                       shrink = 100;
                   }

                   // Reformat the number
                   String formated = currencyFormat.format((price / shrink));

                   current = formated;
                   [yourtextfield].setText(formated);
                   [yourtextfield].setSelection(Math.min(selection, [yourtextfield].getText().length()));

                   [yourtextfield].addTextChangedListener(this);
                }
        }


        @Override
        public void beforeTextChanged(CharSequence s, int start, int count, int after)
        {

        }


        @Override
        public void afterTextChanged(Editable s)
        {
        }
    });

bana çok yardımcı oluyor. Teşekkürler @genixpro
Harin Kaklotar

Fikrinizi beğendim, ancak imleçten sonraki basamak sayısını kaydedip ardından setSelection (uzunluk - sonra) daha düzgün görünür.
Alpha Huang

Çok ilginç! Değiştirilebilir kullanmak fiziksel cihazımda çalıştı, ancak öykünücüde çalışmıyor.
Aliton Oliveira

4

Burada pek çok cevap olsa da en sağlam ve temiz cevap olduğuna inandığım için burada bulduğum bu kodu paylaşmak isterim .

class CurrencyTextWatcher implements TextWatcher {

    boolean mEditing;

    public CurrencyTextWatcher() {
        mEditing = false;
    }

    public synchronized void afterTextChanged(Editable s) {
        if(!mEditing) {
            mEditing = true;

            String digits = s.toString().replaceAll("\\D", "");
            NumberFormat nf = NumberFormat.getCurrencyInstance();
            try{
                String formatted = nf.format(Double.parseDouble(digits)/100);
                s.replace(0, s.length(), formatted);
            } catch (NumberFormatException nfe) {
                s.clear();
            }

            mEditing = false;
        }
    }

    public void beforeTextChanged(CharSequence s, int start, int count, int after) { }

    public void onTextChanged(CharSequence s, int start, int before, int count) { }

}

Umarım yardımcı olur.


Bu ondalık noktayı çıkarmaz mı? Yani 100.00 ile 10.000 $ arasındaki farkı söyleyemezsiniz - bir şeyi kaçırmıyorsam
nasch

2
bu mükemmel cevap! benim için çalıştı. benim, sadece bu cevaplar için ne kadar zaman harcadığımı bir düşünün ve sonunda aşağıya inip istediğimi buldum.
Ge Rong

Sana yardım etmesine sevindim.
Kayvan N

@nasch Bu bir TextWatcher'dır ve metni, bahsettiğiniz vakayı engelleyecek şekilde kullanıcı türleri olarak biçimlendirir.
Kayvan N

@KayvanN TextWatcher'ın ne olduğunu biliyorum. replaceAll("\\D", "")rakam olmayan her şeyi siler, böylece "100,00 $" ve "10.000 $" her ikisi de "10000" olur. Görünüşe göre girdiye sent içerecek şekilde güveniyorsunuz. Bu garantili ise, harika ama değilse, sorunlar çıkacağını düşünüyorum.
nasch

4

Tamam, işte Para Birimi biçimleriyle başa çıkmanın daha iyi bir yolu, geri silme tuş vuruşu. Kod, yukarıdaki @androidcurious 'kodunu temel almaktadır ... Ancak, geriye doğru silme ile ilgili bazı problemler ve bazı ayrıştırma istisnaları ile ilgilenir: http://miguelt.blogspot.ca/2013/01/textwatcher-for-currency-masksformatting .html [GÜNCELLEME] Önceki çözümde bazı sorunlar vardı ... Bu daha iyi bir çözümdür: http://miguelt.blogspot.ca/2013/02/update-textwatcher-for-currency.html Ve ... işte detaylar:

Bu yaklaşım, geleneksel Android mekanizmalarını kullandığı için daha iyidir. Buradaki fikir, kullanıcı Görünümü oluşturduktan sonra değerleri biçimlendirmektir.

Sayısal değerleri kısıtlamak için bir InputFilter tanımlayın - bu çoğu durumda gereklidir çünkü ekran uzun EditText görünümlerini barındıracak kadar büyük değildir. Bu, statik bir iç sınıf veya başka bir düz sınıf olabilir:

/** Numeric range Filter. */
class NumericRangeFilter implements InputFilter {
    /** Maximum value. */
    private final double maximum;
    /** Minimum value. */
    private final double minimum;
    /** Creates a new filter between 0.00 and 999,999.99. */
    NumericRangeFilter() {
        this(0.00, 999999.99);
    }
    /** Creates a new filter.
     * @param p_min Minimum value.
     * @param p_max Maximum value. 
     */
    NumericRangeFilter(double p_min, double p_max) {
        maximum = p_max;
        minimum = p_min;
    }
    @Override
    public CharSequence filter(
            CharSequence p_source, int p_start,
            int p_end, Spanned p_dest, int p_dstart, int p_dend
    ) {
        try {
            String v_valueStr = p_dest.toString().concat(p_source.toString());
            double v_value = Double.parseDouble(v_valueStr);
            if (v_value<=maximum && v_value>=minimum) {
                // Returning null will make the EditText to accept more values.
                return null;
            }
        } catch (NumberFormatException p_ex) {
            // do nothing
        }
        // Value is out of range - return empty string.
        return "";
    }
}

View.OnFocusChangeListener'ı uygulayacak bir sınıf (iç statik veya yalnızca bir sınıf) tanımlayın. Utils sınıfı kullandığımı unutmayın - uygulama "Miktarlar, Vergiler" bölümünde bulunabilir.

/** Used to format the amount views. */
class AmountOnFocusChangeListener implements View.OnFocusChangeListener {
    @Override
    public void onFocusChange(View p_view, boolean p_hasFocus) {
        // This listener will be attached to any view containing amounts.
        EditText v_amountView = (EditText)p_view;
        if (p_hasFocus) {
            // v_value is using a currency mask - transfor over to cents.
            String v_value = v_amountView.getText().toString();
            int v_cents = Utils.parseAmountToCents(v_value);
            // Now, format cents to an amount (without currency mask)
            v_value = Utils.formatCentsToAmount(v_cents);
            v_amountView.setText(v_value);
            // Select all so the user can overwrite the entire amount in one shot.
            v_amountView.selectAll();
        } else {
            // v_value is not using a currency mask - transfor over to cents.
            String v_value = v_amountView.getText().toString();
            int v_cents = Utils.parseAmountToCents(v_value);
            // Now, format cents to an amount (with currency mask)
            v_value = Utils.formatCentsToCurrency(v_cents);
            v_amountView.setText(v_value);
        }
    }
}

Bu sınıf, standart mekanizmalara bağlı olarak düzenleme sırasında para birimi biçimini kaldıracaktır. Kullanıcı çıktığı zaman para birimi biçimi yeniden uygulanır.

Örnek sayısını en aza indirmek için bazı statik değişkenler tanımlamak daha iyidir:

   static final InputFilter[] FILTERS = new InputFilter[] {new NumericRangeFilter()};
   static final View.OnFocusChangeListener ON_FOCUS = new AmountOnFocusChangeListener();

Son olarak, onCreateView içinde (...):

   EditText mAmountView = ....
   mAmountView.setFilters(FILTERS);
   mAmountView.setOnFocusChangeListener(ON_FOCUS);

FILTERS ve ON_FOCUS öğesini istediğiniz sayıda EditText görünümünde yeniden kullanabilirsiniz.

İşte Utils sınıfı:

public class Utils {

   private static final NumberFormat FORMAT_CURRENCY = NumberFormat.getCurrencyInstance();
   /** Parses an amount into cents.
    * @param p_value Amount formatted using the default currency. 
    * @return Value as cents.
    */
   public static int parseAmountToCents(String p_value) {
       try {
           Number v_value = FORMAT_CURRENCY.parse(p_value);
           BigDecimal v_bigDec = new BigDecimal(v_value.doubleValue());
           v_bigDec = v_bigDec.setScale(2, BigDecimal.ROUND_HALF_UP);
           return v_bigDec.movePointRight(2).intValue();
       } catch (ParseException p_ex) {
           try {
               // p_value doesn't have a currency format.
               BigDecimal v_bigDec = new BigDecimal(p_value);
               v_bigDec = v_bigDec.setScale(2, BigDecimal.ROUND_HALF_UP);
               return v_bigDec.movePointRight(2).intValue();
           } catch (NumberFormatException p_ex1) {
               return -1;
           }
       }
   }
   /** Formats cents into a valid amount using the default currency.
    * @param p_value Value as cents 
    * @return Amount formatted using a currency.
    */
   public static String formatCentsToAmount(int p_value) {
       BigDecimal v_bigDec = new BigDecimal(p_value);
       v_bigDec = v_bigDec.setScale(2, BigDecimal.ROUND_HALF_UP);
       v_bigDec = v_bigDec.movePointLeft(2);
       String v_currency = FORMAT_CURRENCY.format(v_bigDec.doubleValue());
       return v_currency.replace(FORMAT_CURRENCY.getCurrency().getSymbol(), "").replace(",", "");
   }
   /** Formats cents into a valid amount using the default currency.
    * @param p_value Value as cents 
    * @return Amount formatted using a currency.
    */
   public static String formatCentsToCurrency(int p_value) {
       BigDecimal v_bigDec = new BigDecimal(p_value);
       v_bigDec = v_bigDec.setScale(2, BigDecimal.ROUND_HALF_UP);
       v_bigDec = v_bigDec.movePointLeft(2);
       return FORMAT_CURRENCY.format(v_bigDec.doubleValue());
   }

}

Bu soruya teorik olarak cevap verse de, bağlantılı makalenin temel kısımlarını cevabınıza eklemenizi ve referans için bağlantı sağlamanızı istiyoruz . Bunu yapmamak, cevabı bağlantı çürümesinden dolayı risk altında bırakır.
Kev

Düzenleme metni odağı kaybettiğinde java.lang.NumberFormatException: Geçersiz çift: "$ 12,345.00" alıyorum. Nasıl düzeltilir?
Madhan

4

Aşağıdaki sürümü oluşturmak için, Nathan Leigh'in başvurduğu uygulamayı ve Kayvan N'lerin ve user2582318'in önerilen regex'i rakamlar dışındaki tüm karakterleri kaldırmak için kullandım:

fun EditText.addCurrencyFormatter() {

    // Reference: /programming/5107901/better-way-to-format-currency-input-edittext/29993290#29993290
    this.addTextChangedListener(object: TextWatcher {

        private var current = ""

        override fun afterTextChanged(s: Editable?) {
        }

        override fun beforeTextChanged(s: CharSequence?, start: Int, count: Int, after: Int) {
        }

        override fun onTextChanged(s: CharSequence?, start: Int, before: Int, count: Int) {

            if (s.toString() != current) {
                this@addCurrencyFormatter.removeTextChangedListener(this)
                // strip off the currency symbol

                // Reference for this replace regex: /programming/5107901/better-way-to-format-currency-input-edittext/28005836#28005836
                val cleanString = s.toString().replace("\\D".toRegex(), "")
                val parsed = if (cleanString.isBlank()) 0.0 else cleanString.toDouble()
                // format the double into a currency format
                val formated = NumberFormat.getCurrencyInstance()
                        .format(parsed / 100)

                current = formated
                this@addCurrencyFormatter.setText(formated)
                this@addCurrencyFormatter.setSelection(formated.length)

                this@addCurrencyFormatter.addTextChangedListener(this)
            }
        }
    })

}

Bu, Kotlin'de TextWatcher'ı EditText'in TextChangedListener öğesine ekleyen bir uzantı işlevidir.

Kullanmak için sadece:

yourEditText = (EditText) findViewById(R.id.edit_text_your_id);
yourEditText.addCurrencyFormatter()

Umut ediyorum bu yardım eder.


3

Bunu buradan aldım ve Portekiz para birimi formatına uyacak şekilde değiştirdim.

import java.text.NumberFormat;
import java.util.Currency;
import java.util.Locale;

import android.text.Editable;
import android.text.TextWatcher;
import android.widget.EditText;

public class CurrencyTextWatcher implements TextWatcher {

    private String current = "";
    private int index;
    private boolean deletingDecimalPoint;
    private final EditText currency;

    public CurrencyTextWatcher(EditText p_currency) {
        currency = p_currency;
    }


    @Override
    public void beforeTextChanged(CharSequence p_s, int p_start, int p_count, int p_after) {

        if (p_after>0) {
                index = p_s.length() - p_start;
            } else {
                index = p_s.length() - p_start - 1;
            }
            if (p_count>0 && p_s.charAt(p_start)==',') {
                deletingDecimalPoint = true;
            } else {
                deletingDecimalPoint = false;
            }

    }

    @Override
    public void onTextChanged(CharSequence s, int start, int before, int count) {

    }

    @Override
    public void afterTextChanged(Editable p_s) {


         if(!p_s.toString().equals(current)){
                currency.removeTextChangedListener(this);
                if (deletingDecimalPoint) {
                    p_s.delete(p_s.length()-index-1, p_s.length()-index);
                }
                // Currency char may be retrieved from  NumberFormat.getCurrencyInstance()
                String v_text = p_s.toString().replace("€","").replace(",", "");
                v_text = v_text.replaceAll("\\s", "");
                double v_value = 0;
                if (v_text!=null && v_text.length()>0) {
                    v_value = Double.parseDouble(v_text);
                }
                // Currency instance may be retrieved from a static member.
                NumberFormat numberFormat = NumberFormat.getCurrencyInstance(new Locale("pt", "PT"));
                String v_formattedValue = numberFormat.format((v_value/100));
                current = v_formattedValue;
                currency.setText(v_formattedValue);
                if (index>v_formattedValue.length()) {
                    currency.setSelection(v_formattedValue.length());
                } else {
                    currency.setSelection(v_formattedValue.length()-index);
                }
                // include here anything you may want to do after the formatting is completed.
                currency.addTextChangedListener(this);
             }
    }

}

Layout.xml

<EditText
    android:id="@+id/edit_text_your_id"
    ...
    android:text="0,00 €"
    android:inputType="numberDecimal"
    android:digits="0123456789" />

Çalışmasını sağlayın

    yourEditText = (EditText) findViewById(R.id.edit_text_your_id);
    yourEditText.setRawInputType(Configuration.KEYBOARD_12KEY);
    yourEditText.addTextChangedListener(new CurrencyTextWatcher(yourEditText));

2

Benim için böyle çalıştı

 public void onTextChanged(CharSequence s, int start,
                    int before, int count) {
                if(!s.toString().matches("^\\$(\\d{1,3}(\\,\\d{3})*|(\\d+))(\\.\\d{2})?$"))
                {
                    String userInput= ""+s.toString().replaceAll("[^\\d]", "");
                    if (userInput.length() > 2) {
                        Float in=Float.parseFloat(userInput);
                        price = Math.round(in); // just to get an Integer
                        //float percen = in/100;
                        String first, last;
                        first = userInput.substring(0, userInput.length()-2);
                        last = userInput.substring(userInput.length()-2);
                        edEx1.setText("$"+first+"."+last);
                        Log.e(MainActivity.class.toString(), "first: "+first + " last:"+last);
                        edEx1.setSelection(edEx1.getText().length());
                    }
                }
            }

2

InputFilter arayüzünü kullanmak daha iyidir. Normal ifadeler kullanarak her tür girdiyi işlemek çok daha kolay. Para birimi giriş formatı için çözümüm:

public class CurrencyFormatInputFilter implements InputFilter {

Pattern mPattern = Pattern.compile("(0|[1-9]+[0-9]*)(\\.[0-9]{1,2})?");

@Override
public CharSequence filter(
        CharSequence source,
        int start,
        int end,
        Spanned dest,
        int dstart,
        int dend) {

String result = 
        dest.subSequence(0, dstart)
        + source.toString() 
        + dest.subSequence(dend, dest.length());

Matcher matcher = mPattern.matcher(result);

if (!matcher.matches()) return dest.subSequence(dstart, dend);

return null;
}
}

Geçerli: 0.00, 0.0, 10.00, 111.1
Geçersiz: 0, 0.000, 111, 10, 010.00, 01.0

Nasıl kullanılır:

editText.setFilters(new InputFilter[] {new CurrencyFormatInputFilter()});


1

Json para birimi alanınız sayı türündeyse (String değil) 3.1, 3.15 veya sadece 3 olarak gelebilir. Çünkü json sayı alanlarını otomatik olarak yuvarlar.

Bu durumda, düzgün bir görüntü için onu yuvarlamanız gerekebilir (ve daha sonra bir giriş alanında bir maske kullanabilmeniz için):

    NumberFormat nf = NumberFormat.getCurrencyInstance();

    float value = 200 // it can be 200, 200.3 or 200.37, BigDecimal will take care
    BigDecimal valueAsBD = BigDecimal.valueOf(value);
    valueAsBD.setScale(2, BigDecimal.ROUND_HALF_UP);

    String formated = nf.format(valueAsBD);

Bu neden gerekli?

Tüm cevaplar, kuruşları aldığınızı ve böylece dolar + sent / 100 = dolar, sent oluşturmaya karar verirken para birimi simbollarını kaldırmaya işaret ediyor. Ancak, json para birimi alanınız bir sayı türü ise (ve bir Dize değilse) kuruşlarınızı yuvarlayacaktır, 3, 3.1 veya 3.15 olabilir.


1
Tam olarak ihtiyacım olan şey. Teşekkürler!
Erick Engelhardt

come as 3.1 , 3.15 or just 3. Because json automatically round number fields- bunun yuvarlamayla ortak bir yanı yok !
Marcin Orlowski

1

Doubles, BigDecimals vb. İle çok fazla arama yaptıktan ve başarısız olduktan sonra, bu kodu yaptım. Tak ve Çalıştır ile çalışır. Kotlin'de. Öyleyse, benim gibi sıkışan başkalarına yardım etmek için gidelim.

Kod temelde bir textWatcher yerleştirecek ve komayı doğru yere ayarlayacak bir işlevdir.

İlk önce bu işlevi oluşturun:

fun CurrencyWatcher( editText:EditText) {

    editText.addTextChangedListener(object : TextWatcher {
        //this will prevent the loop
        var changed: Boolean = false

        override fun afterTextChanged(p0: Editable?) {
            changed = false

        }

        override fun beforeTextChanged(p0: CharSequence?, p1: Int, p2: Int, p3: Int) {

            editText.setSelection(p0.toString().length)
        }

        @SuppressLint("SetTextI18n")
        override fun onTextChanged(p0: CharSequence?, p1: Int, p2: Int, p3: Int) {
            if (!changed) {
                changed = true

                var str: String = p0.toString().replace(",", "").trim()
                var element0: String = str.elementAt(0).toString()
                var element1: String = "x"
                var element2: String = "x"
                var element3: String = "x"
                var element4: String = "x"
                var element5: String = "x"
                var element6: String = "x"

                //this variables will store each elements of the initials data for the case we need to move this numbers like: 0,01 to 0,11 or 0,11 to 0,01
                if (str.length >= 2) {
                    element1 = str.elementAt(1).toString()
                }
                if (str.length >= 3) {
                    element2 = str.elementAt(2).toString()
                }

                editText.removeTextChangedListener(this)


                //this first block of code will take care of the case
                //where the number starts with 0 and needs to adjusta the 0 and the "," place
                if (str.length == 1) {
                    str = "0,0" + str
                    editText.setText(str)

                } else if (str.length <= 3 && str == "00") {

                    str = "0,00"
                    editText.setText(str)
                    editText.setSelection(str.length)
                } else if (element0 == "0" && element1 == "0" && element2 == "0") {
                    str = str.replace("000", "")
                    str = "0,0" + str
                    editText.setText(str)
                } else if (element0 == "0" && element1 == "0" && element2 != "0") {
                    str = str.replace("00", "")
                    str = "0," + str
                    editText.setText(str)
                } else {

                    //This block of code works with the cases that we need to move the "," only because the value is bigger
                    //lets get the others elements
                    if (str.length >= 4) {
                        element3 = str.elementAt(3).toString()
                    }
                    if (str.length >= 5) {
                        element4 = str.elementAt(4).toString()
                    }
                    if (str.length >= 6) {
                        element5 = str.elementAt(5).toString()
                    }
                    if (str.length == 7) {
                        element6 = str.elementAt(6).toString()
                    }


                    if (str.length >= 4 && element0 != "0") {

                        val sb: StringBuilder = StringBuilder(str)
                        //set the coma in right place
                        sb.insert(str.length - 2, ",")
                        str = sb.toString()
                    }

                    //change the 0,11 to 1,11
                    if (str.length == 4 && element0 == "0") {

                        val sb: StringBuilder = StringBuilder(str)
                        //takes the initial 0 out
                        sb.deleteCharAt(0);
                        str = sb.toString()

                        val sb2: StringBuilder = StringBuilder(str)
                        sb2.insert(str.length - 2, ",")
                        str = sb2.toString()
                    }

                    //this will came up when its like 11,11 and the user delete one, so it will be now 1,11
                    if (str.length == 3 && element0 != "0") {
                        val sb: StringBuilder = StringBuilder(str)
                        sb.insert(str.length - 2, ",")
                        str = sb.toString()
                    }

                    //came up when its like 0,11 and the user delete one, output will be 0,01
                    if (str.length == 2 && element0 == "0") {
                        val sb: StringBuilder = StringBuilder(str)
                        //takes 0 out
                        sb.deleteCharAt(0);
                        str = sb.toString()

                        str = "0,0" + str

                    }

                    //came up when its 1,11 and the user delete, output will be 0,11
                    if (str.length == 2 && element0 != "0") {
                        val sb: StringBuilder = StringBuilder(str)
                        //retira o 0 da frente
                        sb.insert(0, "0,")
                        str = sb.toString()

                    }


                    editText.setText(str)
                }

                //places the selector at the end to increment the number
                editText.setSelection(str.length)
                editText.addTextChangedListener(this)
            }

        }
    })
}

Ve sonra bu işlevi bu şekilde adlandırırsınız

val etVal:EditText = findViewById(R.id.etValue)

CurrencyWatcher(etVal)

1

Başka bir yaklaşım, ancak Guilherme cevabına dayanıyor . Bu yaklaşım, ülke yerel ayarınız bulunmadığında veya özel para birimi simgeleri kullanmak istediğinizde yararlıdır. Bu uygulama yalnızca ondalık olmayan pozitif içindir.

KOTLIN birinci, temsilci bu kod setMaskingMoneyiçinEditText

fun EditText.setMaskingMoney(currencyText: String) {
    this.addTextChangedListener(object: MyTextWatcher{
        val editTextWeakReference: WeakReference<EditText> = WeakReference<EditText>(this@setMaskingMoney)
        override fun afterTextChanged(editable: Editable?) {
            val editText = editTextWeakReference.get() ?: return
            val s = editable.toString()
            editText.removeTextChangedListener(this)
            val cleanString = s.replace("[Rp,. ]".toRegex(), "")
            val newval = currencyText + cleanString.monetize()

            editText.setText(newval)
            editText.setSelection(newval.length)
            editText.addTextChangedListener(this)
        }
    })
}

Daha sonra MyTextWatcherarayüzden genişletilmelidir TextWatcher. Sadece afterTextChangedmetoda ihtiyacımız olduğundan , diğer metotların bu arayüzde geçersiz kılınması gerekir.

interface MyTextWatcher: TextWatcher {
    override fun onTextChanged(p0: CharSequence?, p1: Int, p2: Int, p3: Int) {}
    override fun beforeTextChanged(p0: CharSequence?, p1: Int, p2: Int, p3: Int) {}
}

ve para kazanma yöntemleri:

fun String.monetize(): String = if (this.isEmpty()) "0"
    else DecimalFormat("#,###").format(this.replace("[^\\d]".toRegex(),"").toLong())

Tam uygulamalar:

fun EditText.setMaskingMoney(currencyText: String) {
    this.addTextChangedListener(object: MyTextWatcher{
        val editTextWeakReference: WeakReference<EditText> = WeakReference<EditText>(this@setMaskingMoney)
        override fun afterTextChanged(editable: Editable?) {
            val editText = editTextWeakReference.get() ?: return
            val s = editable.toString()
            editText.removeTextChangedListener(this)
            val cleanString = s.replace("[Rp,. ]".toRegex(), "")
            val newval = currencyText + cleanString.monetize()

            editText.setText(newval)
            editText.setSelection(newval.length)
            editText.addTextChangedListener(this)
        }
    })
}

interface MyTextWatcher: TextWatcher {
    override fun onTextChanged(p0: CharSequence?, p1: Int, p2: Int, p3: Int) {}
    override fun beforeTextChanged(p0: CharSequence?, p1: Int, p2: Int, p3: Int) {}
}


fun String.monetize(): String = if (this.isEmpty()) "0"
    else DecimalFormat("#,###").format(this.replace("[^\\d]".toRegex(),"").toLong())

ve onCreate yönteminde bir yerde:

yourTextView.setMaskingMoney("Rp. ")

0

Farklı şekillerde üzerinde StackOverflow mesajların çoğu baktıktan sonra bir kullanarak bunu başarmak için TextWatcher, InputFilterya da benzeri kütüphane CurrencyEditText I bir kullanarak bu basit bir çözüm oturttuktan OnFocusChangeListener.

Mantık, EditTextodaklandığında bir sayıya ayrıştırmak ve odağı kaybettiğinde yeniden biçimlendirmektir.

amount.setOnFocusChangeListener(new View.OnFocusChangeListener() {
        @Override
        public void onFocusChange(View view, boolean hasFocus) {
            Number numberAmount = 0f;
            try {
                numberAmount = Float.valueOf(amount.getText().toString());
            } catch (NumberFormatException e1) {
                e1.printStackTrace();
                try {
                    numberAmount = NumberFormat.getCurrencyInstance().parse(amount.getText().toString());
                } catch (ParseException e2) {
                    e2.printStackTrace();
                }
            }
            if (hasFocus) {
                amount.setText(numberAmount.toString());
            } else {
                amount.setText(NumberFormat.getCurrencyInstance().format(numberAmount));
            }
        }
    });

0

Bir Kotlin + Rx sürümü uyguladım.

Brezilya'nın para birimi içindir (ör. 1,500.00 - 5,21 - 192,90), ancak diğer biçimlere kolayca uyum sağlayabilirsiniz.

Umarım başka biri onu yararlı bulur.

RxTextView
            .textChangeEvents(fuel_price) // Observe text event changes
            .filter { it.text().isNotEmpty() } // do not accept empty text when event first fires
            .flatMap {
                val onlyNumbers = Regex("\\d+").findAll(it.text()).fold(""){ acc:String,it:MatchResult -> acc.plus(it.value)}
                Observable.just(onlyNumbers)
            }
            .distinctUntilChanged()
            .map { it.trimStart('0') }
            .map { when (it.length) {
                        1-> "00"+it
                        2-> "0"+it
                        else -> it }
            }
            .subscribe {
                val digitList = it.reversed().mapIndexed { i, c ->
                    if ( i == 2 ) "${c},"
                    else if ( i < 2 ) c
                    else if ( (i-2)%3==0 ) "${c}." else c
                }

                val currency = digitList.reversed().fold(""){ acc,it -> acc.toString().plus(it) }
                fuel_price.text = SpannableStringBuilder(currency)
                fuel_price.setSelection(currency.length)
            }

0

CurrencyTextWatcher.java

public class CurrencyTextWatcher implements TextWatcher {

    private final static String DS = "."; //Decimal Separator
    private final static String TS = ","; //Thousands Separator
    private final static String NUMBERS = "0123456789"; //Numbers
    private final static int MAX_LENGTH = 13; //Maximum Length

    private String format;

    private DecimalFormat decimalFormat;
    private EditText editText;

    public CurrencyTextWatcher(EditText editText) {
        String pattern = "###" + TS + "###" + DS + "##";
        decimalFormat = new DecimalFormat(pattern);
        this.editText = editText;
        this.editText.setInputType(InputType.TYPE_CLASS_NUMBER);
        this.editText.setKeyListener(DigitsKeyListener.getInstance(NUMBERS + DS));
        this.editText.setFilters(new InputFilter[]{new InputFilter.LengthFilter(MAX_LENGTH)});
    }

    @Override
    public void beforeTextChanged(CharSequence charSequence, int i, int i1, int i2) {

    }

    @Override
    public void onTextChanged(CharSequence charSequence, int i, int i1, int i2) {

    }

    @Override
    public void afterTextChanged(Editable editable) {

        editText.removeTextChangedListener(this);
        String value = editable.toString();
        if (!value.isEmpty()) {
            value = value.replace(TS, "");
            try {
                format = decimalFormat.format(Double.parseDouble(value));
                format = format.replace("0", "");
            } catch (Exception e) {
                System.out.println(e.getMessage());
            }

            editText.setText(format);
        }

        editText.addTextChangedListener(this);
    }
}

EditTextCurrency.java

public class EditTextCurrency extends AppCompatEditText {
    public EditTextCurrency(Context context) {
        super(context);
    }

    public EditTextCurrency(Context context, AttributeSet attrs) {
        super(context, attrs);
        addTextChangedListener(new CurrencyTextWatcher(this));
    }
}

görüntü açıklamasını buraya girin


0

İşte bir EditText'te, uygulanması kolay ve her yerde çılgın semboller potansiyeli olmadan kullanıcı için iyi çalışan bir para birimini nasıl görüntüleyebildim. Bu, EditText artık odakta kalmayıncaya kadar herhangi bir biçimlendirme yapmaya çalışmayacaktır. Kullanıcı, biçimlendirmeyi tehlikeye atmadan geri dönebilir ve herhangi bir düzenleme yapabilir. Yalnızca görüntüleme için 'formattedPrice' değişkenini ve hesaplamalar için sakladığım / kullandığım değer olarak 'itemPrice' değişkenini kullanıyorum.

Gerçekten iyi çalışıyor gibi görünüyor, ancak ben sadece birkaç haftadır buradayım, bu yüzden her türlü yapıcı eleştiriye kesinlikle açığız!

Xml'deki EditText görünümü aşağıdaki özniteliğe sahiptir:

android:inputType="numberDecimal"

Global değişkenler:

private String formattedPrice;
private int itemPrice = 0;

OnCreate yönteminde:

EditText itemPriceInput = findViewById(R.id.item_field_price);

itemPriceInput.setOnFocusChangeListener(new View.OnFocusChangeListener() {
    @Override
    public void onFocusChange(View v, boolean hasFocus) {
        String priceString = itemPriceInput.getText().toString();

        if (! priceString.equals("")) {
            itemPrice = Double.parseDouble(priceString.replaceAll("[$,]", ""));
            formattedPrice = NumberFormat.getCurrencyInstance().format(itemPrice);
            itemPriceInput.setText(formattedPrice);
        }
    }
});

0

Birisinin bunu yapmanın bir yolunu RxBinding ve Kotlin kullanarak ilgilendirmesi durumunda:

var isEditing = false

RxTextView.textChanges(dollarValue)
            .filter { !isEditing }
            .filter { it.isNotBlank() }
            .map { it.toString().filter { it.isDigit() } }
            .map { BigDecimal(it).setScale(2, BigDecimal.ROUND_FLOOR).divide(100.toBigDecimal(), BigDecimal.ROUND_FLOOR) }
            .map { NumberFormat.getCurrencyInstance(Locale("pt", "BR")).format(it) }
            .subscribe {
                isEditing = true
                dollarValue.text = SpannableStringBuilder(it)
                dollarValue.setSelection(it.length)
                isEditing = false
            }

0

sadece onaylanan cevaba ek bir yorum. Ayrıştırma nedeniyle imleci edittext alanında hareket ettirirken bir çökme yaşayabilirsiniz. Bir deneme yakalama ifadesi yaptım, ancak kendi kodunuzu uyguladım.

@Override public void onTextChanged(CharSequence s, int start, int before, int count) {
        if(!s.toString().equals(current)){
        amountEditText.removeTextChangedListener(this);

            String cleanString = s.toString().replaceAll("[$,.]", "");

            try{
                double parsed = Double.parseDouble(cleanString);
                String formatted = NumberFormat.getCurrencyInstance().format((parsed/100));
                current = formatted;
                amountEditText.setText(formatted);
                amountEditText.setSelection(formatted.length());
            } catch (Exception e) {

            }

            amountEditText.addTextChangedListener(this);
        }
    }

0

bu yöntemleri kullanabilirsin

import android.text.Editable
import android.text.TextWatcher
import android.widget.EditText
import android.widget.TextView
import java.text.NumberFormat
import java.util.*

fun TextView.currencyFormat() {
    addTextChangedListener(object : TextWatcher {
        override fun afterTextChanged(s: Editable?) {}

        override fun beforeTextChanged(s: CharSequence?, start: Int, count: Int, after: Int) {}

        override fun onTextChanged(s: CharSequence?, start: Int, before: Int, count: Int) {
            removeTextChangedListener(this)
            text = if (s?.toString().isNullOrBlank()) {
                ""
            } else {
                s.toString().currencyFormat()
            }
            if(this@currencyFormat is EditText){
                setSelection(text.toString().length)
            }
            addTextChangedListener(this)
        }
    })
}

fun String.currencyFormat(): String {
    var current = this
    if (current.isEmpty()) current = "0"
    return try {
        if (current.contains('.')) {
            NumberFormat.getNumberInstance(Locale.getDefault()).format(current.replace(",", "").toDouble())
        } else {
            NumberFormat.getNumberInstance(Locale.getDefault()).format(current.replace(",", "").toLong())
        }
    } catch (e: Exception) {
        "0"
    }
}

0

Kotlin versiyonu:

    var current = ""

    editText.addTextChangedListener(object: TextWatcher {
        override fun afterTextChanged(s: Editable?) {}
        override fun beforeTextChanged(s: CharSequence?, start: Int, count: Int, after: Int) {}

        override fun onTextChanged(s: CharSequence?, start: Int, before: Int, count: Int) {
            val stringText = s.toString()

            if(stringText != current) {
                editText.removeTextChangedListener(this)

                val locale: Locale = Locale.UK
                val currency = Currency.getInstance(locale)
                val cleanString = stringText.replace("[${currency.symbol},.]".toRegex(), "")
                val parsed = cleanString.toDouble()
                val formatted = NumberFormat.getCurrencyInstance(locale).format(parsed / 100)

                current = formatted
                editText.setText(formatted)
                editText.setSelection(formatted.length)
                editText.addTextChangedListener(this)
            }
        }
    })

0
public class MoneyEditText extends android.support.v7.widget.AppCompatEditText{
public MoneyEditText(Context context) {
    super(context);
    addTextChangedListener(MoneySplitter());
}
public MoneyEditText(Context context, AttributeSet attrs) {
    super(context, attrs);
    addTextChangedListener(MoneySplitter());
}
public MoneyEditText(Context context, AttributeSet attrs, int defStyleAttr) {
    super(context, attrs, defStyleAttr);
    addTextChangedListener(MoneySplitter());
}
public TextWatcher MoneySplitter() {
    TextWatcher textWatcher = new TextWatcher() {
        @Override
        public void onTextChanged(CharSequence s, int start, int before, int count) {
            try
            {
                removeTextChangedListener(this);
                String value = s.toString();
                if (!value.equals(""))
                {
                        if(!TextUtils.isEmpty(value))
                            setText(formatPrice(Double.parseDouble(value)));
                        setSelection(getText().toString().length());

                }
                addTextChangedListener(this);
            }
            catch (Exception ex)
            {
                ex.printStackTrace();
                addTextChangedListener(this);
            }
        }
        @Override
        public void beforeTextChanged(CharSequence s, int start, int count, int after) {
        }
        @Override
        public void afterTextChanged(Editable s) {
        }
    };
    return textWatcher;
}

public static String formatPrice(double value){
        int DecimalPointNumber = 2;
        Locale locale = Locale.getDefault();
        DecimalFormat myFormatter = (DecimalFormat) NumberFormat.getCurrencyInstance(locale);
        StringBuilder sb = new StringBuilder();
        if(DecimalPointNumber>0){
            for (int i = 0; i < DecimalPointNumber; i++) {
                sb.append("#");
            }
            myFormatter.applyPattern("###,###."+ sb.toString());
        }else
            myFormatter.applyPattern("###,###"+ sb.toString());

            return Currency.getInstance(Locale.getDefault()).getSymbol() + myFormatter.format(value);
    }
}

ve sonra bu bloğu editText'iniz olarak kullanın

   <MoneyEditText
   android:id="@+id/txtPrice"
   android:layout_width="match_parent"
   android:layout_height="64dp"
   android:digits="0123456789.,"
   android:inputType="numberDecimal"
   android:selectAllOnFocus="true"
   android:singleLine="true" />

Giriş metnini istediğiniz gibi biçimlendirmek için bu özelleştirilmiş düzenleme metnini kullanabilirsiniz.
Saeid Mohammadi'ye

Negatif sayıları kabul etmek için bu sınıfı değiştirdim. Kod bir cevap olarak aşağıdadır.
Michel Fernandes

0

Bu Saeid Mohammadi'nin cevabı gibidir, ancak negatif sayıları kabul etmek için değiştirdim.

  package com.example.liberdade.util
    
    import android.text.Editable
    import android.text.TextWatcher
    import android.widget.EditText
    import java.lang.ref.WeakReference
    import java.math.BigDecimal
    import java.text.NumberFormat
    import java.util.*
    
    
    class MoneyTextWatcher : TextWatcher {
    
    
    
        private val editTextWeakReference: WeakReference<EditText?>?
        private val locale: Locale = Locale("pt", "BR")
        //private final Locale locale;
    
        constructor(editText: EditText?, locale: Locale?) {
            editTextWeakReference = WeakReference<EditText?>(editText)
            //this.locale = if (locale != null) locale else Locale.getDefault()
        }
    
        constructor(editText: EditText?) {
            editTextWeakReference = WeakReference<EditText?>(editText)
            //locale = Locale.getDefault()
        }
    
        override fun beforeTextChanged(
            s: CharSequence?,
            start: Int,
            count: Int,
            after: Int
        ) {
        }
    
        override fun onTextChanged(
            s: CharSequence?,
            start: Int,
            before: Int,
            count: Int
        ) {
        }
    
        override fun afterTextChanged(editable: Editable?) {
            val editText: EditText = editTextWeakReference?.get() ?: return
            editText.removeTextChangedListener(this)
    
            var isNegative = false
            var editableString = editable.toString()
            if (editable != null) {
                if (editableString.contains('-')) {
                    isNegative = true
                    if (editable != null) {
                        editableString = editableString.replace("-","")
                    }
                }
            }
    
            val parsed: BigDecimal? = parseToBigDecimal(editableString, locale)
            //val parsed: BigDecimal? = parseToBigDecimal(editable.toString(), locale)
            var formatted: String = NumberFormat.getCurrencyInstance(locale).format(parsed)
    
            if (isNegative && !(formatted.equals("R\$ 0,00") || formatted.equals("-R\$ 0,00"))) formatted = "-${formatted}"
            editText.setText(formatted)
            editText.setSelection(formatted.length)
            editText.addTextChangedListener(this)
        }
    
        private fun parseToBigDecimal(value: String?, locale: Locale?): BigDecimal? {
            val replaceable = java.lang.String.format(
                "[%s,.\\s]",
                NumberFormat.getCurrencyInstance(locale).currency.symbol
            )
            val cleanString = value!!.replace(replaceable.toRegex(), "")
            return BigDecimal(cleanString).setScale(
                2, BigDecimal.ROUND_FLOOR
            ).divide(
                BigDecimal(100), BigDecimal.ROUND_FLOOR
            )
        }
    }
    
    //como invocar
    //binding.editTextValorCaixa.addTextChangedListener(MoneyTextWatcher(binding.editTextValorCaixa, Locale("pt", "BR")))
Sitemizi kullandığınızda şunları okuyup anladığınızı kabul etmiş olursunuz: Çerez Politikası ve Gizlilik Politikası.
Licensed under cc by-sa 3.0 with attribution required.