DigitsKeyListenerWithComma -> kt
This commit is contained in:
parent
75a9083abd
commit
588c21fcfd
2 changed files with 145 additions and 197 deletions
|
@ -1,197 +0,0 @@
|
|||
package info.nightscout.core.ui.elements;
|
||||
|
||||
import android.text.InputType;
|
||||
import android.text.SpannableStringBuilder;
|
||||
import android.text.Spanned;
|
||||
import android.text.method.NumberKeyListener;
|
||||
import android.view.KeyEvent;
|
||||
|
||||
import androidx.annotation.NonNull;
|
||||
|
||||
class DigitsKeyListenerWithComma extends NumberKeyListener {
|
||||
|
||||
/**
|
||||
* The characters that are used.
|
||||
*
|
||||
* @see KeyEvent#getMatch
|
||||
* @see #getAcceptedChars
|
||||
*/
|
||||
private static final char[][] CHARACTERS = new char[][]{
|
||||
new char[]{'0', '1', '2', '3', '4', '5', '6', '7', '8', '9'},
|
||||
new char[]{'0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '-'},
|
||||
new char[]{'0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '.', ','},
|
||||
new char[]{'0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '-', '.', ','},
|
||||
};
|
||||
|
||||
private char[] mAccepted;
|
||||
private final boolean mSign;
|
||||
private final boolean mDecimal;
|
||||
|
||||
private static final int SIGN = 1;
|
||||
private static final int DECIMAL = 2;
|
||||
|
||||
private static final DigitsKeyListenerWithComma[] sInstance = new DigitsKeyListenerWithComma[4];
|
||||
|
||||
@Override @NonNull
|
||||
protected char[] getAcceptedChars() {
|
||||
return mAccepted;
|
||||
}
|
||||
|
||||
/**
|
||||
* Allocates a DigitsKeyListener that accepts the digits 0 through 9.
|
||||
*/
|
||||
public DigitsKeyListenerWithComma() {
|
||||
this(false, false);
|
||||
}
|
||||
|
||||
/**
|
||||
* Allocates a DigitsKeyListener that accepts the digits 0 through 9,
|
||||
* plus the minus sign (only at the beginning) and/or decimal point
|
||||
* (only one per field) if specified.
|
||||
*/
|
||||
public DigitsKeyListenerWithComma(boolean sign, boolean decimal) {
|
||||
mSign = sign;
|
||||
mDecimal = decimal;
|
||||
|
||||
int kind = (sign ? SIGN : 0) | (decimal ? DECIMAL : 0);
|
||||
mAccepted = CHARACTERS[kind];
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns a DigitsKeyListener that accepts the digits 0 through 9.
|
||||
*/
|
||||
public static DigitsKeyListenerWithComma getInstance() {
|
||||
return getInstance(false, false);
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns a DigitsKeyListener that accepts the digits 0 through 9,
|
||||
* plus the minus sign (only at the beginning) and/or decimal point
|
||||
* (only one per field) if specified.
|
||||
*/
|
||||
public static DigitsKeyListenerWithComma getInstance(boolean sign, boolean decimal) {
|
||||
int kind = (sign ? SIGN : 0) | (decimal ? DECIMAL : 0);
|
||||
|
||||
if (sInstance[kind] != null)
|
||||
return sInstance[kind];
|
||||
|
||||
sInstance[kind] = new DigitsKeyListenerWithComma(sign, decimal);
|
||||
return sInstance[kind];
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns a DigitsKeyListener that accepts only the characters
|
||||
* that appear in the specified String. Note that not all characters
|
||||
* may be available on every keyboard.
|
||||
*/
|
||||
public static DigitsKeyListenerWithComma getInstance(String accepted) {
|
||||
// TODO: do we need a cache of these to avoid allocating?
|
||||
|
||||
DigitsKeyListenerWithComma dim = new DigitsKeyListenerWithComma();
|
||||
|
||||
dim.mAccepted = new char[accepted.length()];
|
||||
accepted.getChars(0, accepted.length(), dim.mAccepted, 0);
|
||||
|
||||
return dim;
|
||||
}
|
||||
|
||||
public int getInputType() {
|
||||
int contentType = InputType.TYPE_CLASS_NUMBER;
|
||||
if (mSign) {
|
||||
contentType |= InputType.TYPE_NUMBER_FLAG_SIGNED;
|
||||
}
|
||||
if (mDecimal) {
|
||||
contentType |= InputType.TYPE_NUMBER_FLAG_DECIMAL;
|
||||
}
|
||||
return contentType;
|
||||
}
|
||||
|
||||
@Override
|
||||
public CharSequence filter(CharSequence source, int start, int end,
|
||||
Spanned dest, int dstart, int dend) {
|
||||
CharSequence out = super.filter(source, start, end, dest, dstart, dend);
|
||||
|
||||
if (!mSign && !mDecimal) {
|
||||
return out;
|
||||
}
|
||||
|
||||
if (out != null) {
|
||||
source = out;
|
||||
start = 0;
|
||||
end = out.length();
|
||||
}
|
||||
|
||||
int sign = -1;
|
||||
int decimal = -1;
|
||||
int dlen = dest.length();
|
||||
|
||||
/*
|
||||
* Find out if the existing text has '-' or '.' characters.
|
||||
*/
|
||||
|
||||
for (int i = 0; i < dstart; i++) {
|
||||
char c = dest.charAt(i);
|
||||
|
||||
if (c == '-') {
|
||||
sign = i;
|
||||
} else if (c == '.' || c == ',') {
|
||||
decimal = i;
|
||||
}
|
||||
}
|
||||
for (int i = dend; i < dlen; i++) {
|
||||
char c = dest.charAt(i);
|
||||
|
||||
if (c == '-') {
|
||||
return ""; // Nothing can be inserted in front of a '-'.
|
||||
} else if (c == '.' || c == ',') {
|
||||
decimal = i;
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
* If it does, we must strip them out from the source.
|
||||
* In addition, '-' must be the very first character,
|
||||
* and nothing can be inserted before an existing '-'.
|
||||
* Go in reverse order so the offsets are stable.
|
||||
*/
|
||||
|
||||
SpannableStringBuilder stripped = null;
|
||||
|
||||
for (int i = end - 1; i >= start; i--) {
|
||||
char c = source.charAt(i);
|
||||
boolean strip = false;
|
||||
|
||||
if (c == '-') {
|
||||
if (i != start || dstart != 0) {
|
||||
strip = true;
|
||||
} else if (sign >= 0) {
|
||||
strip = true;
|
||||
} else {
|
||||
sign = i;
|
||||
}
|
||||
} else if (c == '.' || c == ',') {
|
||||
if (decimal >= 0) {
|
||||
strip = true;
|
||||
} else {
|
||||
decimal = i;
|
||||
}
|
||||
}
|
||||
|
||||
if (strip) {
|
||||
if (end == start + 1) {
|
||||
return ""; // Only one character, and it was stripped.
|
||||
}
|
||||
|
||||
if (stripped == null) {
|
||||
stripped = new SpannableStringBuilder(source, start, end);
|
||||
}
|
||||
|
||||
stripped.delete(i - start, i + 1 - start);
|
||||
}
|
||||
}
|
||||
|
||||
if (stripped != null) {
|
||||
return stripped;
|
||||
} else return out;
|
||||
}
|
||||
}
|
|
@ -0,0 +1,145 @@
|
|||
package info.nightscout.core.ui.elements
|
||||
|
||||
import android.text.InputType
|
||||
import android.text.SpannableStringBuilder
|
||||
import android.text.Spanned
|
||||
import android.text.method.NumberKeyListener
|
||||
|
||||
internal class DigitsKeyListenerWithComma @JvmOverloads constructor(private val sign: Boolean = false, private val decimal: Boolean = false) : NumberKeyListener() {
|
||||
|
||||
private var accepted: CharArray
|
||||
override fun getAcceptedChars(): CharArray = accepted
|
||||
/**
|
||||
* Allocates a DigitsKeyListener that accepts the digits 0 through 9,
|
||||
* plus the minus sign (only at the beginning) and/or decimal point
|
||||
* (only one per field) if specified.
|
||||
*/
|
||||
/**
|
||||
* Allocates a DigitsKeyListener that accepts the digits 0 through 9.
|
||||
*/
|
||||
init {
|
||||
val kind = (if (sign) SIGN else 0) or if (decimal) DECIMAL else 0
|
||||
accepted = CHARACTERS[kind]
|
||||
}
|
||||
|
||||
override fun getInputType(): Int {
|
||||
var contentType = InputType.TYPE_CLASS_NUMBER
|
||||
if (sign) {
|
||||
contentType = contentType or InputType.TYPE_NUMBER_FLAG_SIGNED
|
||||
}
|
||||
if (decimal) {
|
||||
contentType = contentType or InputType.TYPE_NUMBER_FLAG_DECIMAL
|
||||
}
|
||||
return contentType
|
||||
}
|
||||
|
||||
override fun filter(
|
||||
source: CharSequence, start: Int, end: Int,
|
||||
dest: Spanned, dstart: Int, dend: Int
|
||||
): CharSequence {
|
||||
var sourceSequence = source
|
||||
var startIndex = start
|
||||
var endIndex = end
|
||||
val out = super.filter(sourceSequence, startIndex, endIndex, dest, dstart, dend)
|
||||
if (!sign && !decimal) return out
|
||||
if (out != null) {
|
||||
sourceSequence = out
|
||||
startIndex = 0
|
||||
endIndex = out.length
|
||||
}
|
||||
var sign = -1
|
||||
var decimal = -1
|
||||
val dLen = dest.length
|
||||
|
||||
/*
|
||||
* Find out if the existing text has '-' or '.' characters.
|
||||
*/
|
||||
for (i in 0 until dstart) {
|
||||
val c = dest[i]
|
||||
if (c == '-') sign = i
|
||||
else if (c == '.' || c == ',') decimal = i
|
||||
}
|
||||
for (i in dend until dLen) {
|
||||
val c = dest[i]
|
||||
if (c == '-') return "" // Nothing can be inserted in front of a '-'.
|
||||
else if (c == '.' || c == ',') decimal = i
|
||||
}
|
||||
|
||||
/*
|
||||
* If it does, we must strip them out from the source.
|
||||
* In addition, '-' must be the very first character,
|
||||
* and nothing can be inserted before an existing '-'.
|
||||
* Go in reverse order so the offsets are stable.
|
||||
*/
|
||||
var stripped: SpannableStringBuilder? = null
|
||||
for (i in endIndex - 1 downTo startIndex) {
|
||||
val c = sourceSequence[i]
|
||||
var strip = false
|
||||
if (c == '-') {
|
||||
if (i != startIndex || dstart != 0) strip = true
|
||||
else if (sign >= 0) strip = true
|
||||
else sign = i
|
||||
} else if (c == '.' || c == ',') {
|
||||
if (decimal >= 0) strip = true
|
||||
else decimal = i
|
||||
}
|
||||
if (strip) {
|
||||
if (endIndex == startIndex + 1) return "" // Only one character, and it was stripped.
|
||||
if (stripped == null) stripped = SpannableStringBuilder(sourceSequence, startIndex, endIndex)
|
||||
stripped.delete(i - startIndex, i + 1 - startIndex)
|
||||
}
|
||||
}
|
||||
return stripped ?: out
|
||||
}
|
||||
|
||||
companion object {
|
||||
|
||||
/**
|
||||
* The characters that are used.
|
||||
*
|
||||
* @see android.view.KeyEvent.getMatch
|
||||
*
|
||||
* @see .getAcceptedChars
|
||||
*/
|
||||
private val CHARACTERS =
|
||||
arrayOf(
|
||||
charArrayOf('0', '1', '2', '3', '4', '5', '6', '7', '8', '9'),
|
||||
charArrayOf('0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '-'),
|
||||
charArrayOf('0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '.', ','),
|
||||
charArrayOf('0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '-', '.', ',')
|
||||
)
|
||||
private const val SIGN = 1
|
||||
private const val DECIMAL = 2
|
||||
private val sInstance = arrayOfNulls<DigitsKeyListenerWithComma>(4)
|
||||
val instance: DigitsKeyListenerWithComma?
|
||||
/**
|
||||
* Returns a DigitsKeyListener that accepts the digits 0 through 9.
|
||||
*/
|
||||
get() = getInstance(sign = false, decimal = false)
|
||||
|
||||
/**
|
||||
* Returns a DigitsKeyListener that accepts the digits 0 through 9,
|
||||
* plus the minus sign (only at the beginning) and/or decimal point
|
||||
* (only one per field) if specified.
|
||||
*/
|
||||
fun getInstance(sign: Boolean, decimal: Boolean): DigitsKeyListenerWithComma? {
|
||||
val kind = (if (sign) SIGN else 0) or if (decimal) DECIMAL else 0
|
||||
if (sInstance[kind] != null) return sInstance[kind]
|
||||
sInstance[kind] = DigitsKeyListenerWithComma(sign, decimal)
|
||||
return sInstance[kind]
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns a DigitsKeyListener that accepts only the characters
|
||||
* that appear in the specified String. Note that not all characters
|
||||
* may be available on every keyboard.
|
||||
*/
|
||||
fun getInstance(accepted: String): DigitsKeyListenerWithComma {
|
||||
// TODO: do we need a cache of these to avoid allocating?
|
||||
val dim = DigitsKeyListenerWithComma()
|
||||
dim.accepted = CharArray(accepted.length)
|
||||
accepted.toCharArray(dim.accepted, 0, 0, accepted.length)
|
||||
return dim
|
||||
}
|
||||
}
|
||||
}
|
Loading…
Reference in a new issue