问题
The code below crashes in Kotlin/Android with the stack trace provided on the bottom. It was converted from Java/Android, which didn't have such a problem. Original Java code is provided as well. The crash happens when I try to add a character to a password field. Editing existing characters works well.
I've two questions about it:
- What was Kotlin's motivation to replace java.lang.CharSequence with a Kotlin's CharSequence? These two are quite different, and I suspect that it causes the crash.
- Is there any way around to make it working in Kotlin?
Kotlin code that crashes Android's Paint
mPwd!!.transformationMethod = object : PasswordTransformationMethod() {
override fun getTransformation(source: CharSequence, view: View): CharSequence {
return PasswordCharSequence(source)
}
internal inner class PasswordCharSequence(private val source: CharSequence)// Store char sequence
: CharSequence {
val mSource = source
public override val length = mSource.length
init {
App.d(TAG, "SOURCE " + mSource + " " + length)
}
override fun get(index: Int): Char {
App.d(TAG, "SOURCE IND " + index + " " + mSource.length)
return if (mChkUnmask!!.isChecked) mSource.get(index) else '*'
}
override fun subSequence(start: Int, end: Int): CharSequence {
App.d(TAG, "SOURCE SUB " + start + " " + end)
return mSource.subSequence(start, end) // Return default
}
}
}
Original Java code that works perfectly well
mPwd.setTransformationMethod(new PasswordTransformationMethod() {
@Override
public CharSequence getTransformation(CharSequence source, View view) {
return new PasswordCharSequence(source);
}
class PasswordCharSequence implements CharSequence {
private CharSequence mSource;
public PasswordCharSequence(CharSequence source) {
mSource = source; // Store char sequence
}
public char charAt(int index) {
return mChkUnmask.isChecked()?mSource.charAt(index):'*';
}
public int length() {
return mSource.length(); // Return default
}
public CharSequence subSequence(int start, int end) {
return mSource.subSequence(start, end); // Return default
}
}
});
Stack Trace from Logcat
06-03 00:35:08.143 16694 16694 E AndroidRuntime: java.lang.IndexOutOfBoundsException
06-03 00:35:08.143 16694 16694 E AndroidRuntime: at android.graphics.Paint.getRunAdvance(Paint.java:2986)
06-03 00:35:08.143 16694 16694 E AndroidRuntime: at android.text.TextLine.handleText(TextLine.java:719)
06-03 00:35:08.143 16694 16694 E AndroidRuntime: at android.text.TextLine.handleRun(TextLine.java:873)
06-03 00:35:08.143 16694 16694 E AndroidRuntime: at android.text.TextLine.measureRun(TextLine.java:387)
06-03 00:35:08.143 16694 16694 E AndroidRuntime: at android.text.TextLine.measure(TextLine.java:277)
06-03 00:35:08.143 16694 16694 E AndroidRuntime: at android.text.TextLine.metrics(TextLine.java:251)
06-03 00:35:08.143 16694 16694 E AndroidRuntime: at android.text.Layout.getLineExtent(Layout.java:1072)
06-03 00:35:08.143 16694 16694 E AndroidRuntime: at android.text.Layout.getLineWidth(Layout.java:1038)
06-03 00:35:08.143 16694 16694 E AndroidRuntime: at android.widget.TextView.desired(TextView.java:8142)
06-03 00:35:08.143 16694 16694 E AndroidRuntime: at android.widget.TextView.onMeasure(TextView.java:8208)
06-03 00:35:08.143 16694 16694 E AndroidRuntime: at android.view.View.measure(View.java:21051)
06-03 00:35:08.143 16694 16694 E AndroidRuntime: at android.widget.RelativeLayout.measureChildHorizontal(RelativeLayout.java:715)
06-03 00:35:08.143 16694 16694 E AndroidRuntime: at android.widget.RelativeLayout.onMeasure(RelativeLayout.java:461)
06-03 00:35:08.143 16694 16694 E AndroidRuntime: at android.view.View.measure(View.java:21051)
06-03 00:35:08.143 16694 16694 E AndroidRuntime: at android.widget.RelativeLayout.measureChildHorizontal(RelativeLayout.java:715)
06-03 00:35:08.143 16694 16694 E AndroidRuntime: at android.widget.RelativeLayout.onMeasure(RelativeLayout.java:461)
06-03 00:35:08.143 16694 16694 E AndroidRuntime: at android.view.View.measure(View.java:21051)
06-03 00:35:08.143 16694 16694 E AndroidRuntime: at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:6459)
06-03 00:35:08.143 16694 16694 E AndroidRuntime: at android.widget.FrameLayout.onMeasure(FrameLayout.java:185)
06-03 00:35:08.143 16694 16694 E AndroidRuntime: at android.support.v7.widget.ContentFrameLayout.onMeasure(ContentFrameLayout.java:141)
06-03 00:35:08.143 16694 16694 E AndroidRuntime: at android.view.View.measure(View.java:21051)
06-03 00:35:08.143 16694 16694 E AndroidRuntime: at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:6459)
06-03 00:35:08.143 16694 16694 E AndroidRuntime: at android.support.v7.widget.ActionBarOverlayLayout.onMeasure(ActionBarOverlayLayout.java:400)
Program output from Log.d(...)
10 is a CharSequence length here. First digit - current index. Crash happens when I was trying to add a char with index 9 from a keyboard. It was never added and didn't show up in the output below
06-03 00:35:08.121 16694 16694 D GAC-STORE: SOURCE IND 810
06-03 00:35:08.121 16694 16694 D GAC-STORE: SOURCE IND 710
06-03 00:35:08.121 16694 16694 D GAC-STORE: SOURCE IND 610
06-03 00:35:08.121 16694 16694 D GAC-STORE: SOURCE IND 510
06-03 00:35:08.121 16694 16694 D GAC-STORE: SOURCE IND 410
06-03 00:35:08.121 16694 16694 D GAC-STORE: SOURCE IND 310
06-03 00:35:08.121 16694 16694 D GAC-STORE: SOURCE IND 210
06-03 00:35:08.121 16694 16694 D GAC-STORE: SOURCE IND 110
06-03 00:35:08.121 16694 16694 D GAC-STORE: SOURCE IND 010
06-03 00:35:08.121 16694 16694 D GAC-STORE: SOURCE IND 010
06-03 00:35:08.121 16694 16694 D GAC-STORE: SOURCE IND 110
06-03 00:35:08.121 16694 16694 D GAC-STORE: SOURCE IND 210
06-03 00:35:08.121 16694 16694 D GAC-STORE: SOURCE IND 310
06-03 00:35:08.121 16694 16694 D GAC-STORE: SOURCE IND 410
06-03 00:35:08.121 16694 16694 D GAC-STORE: SOURCE IND 510
06-03 00:35:08.122 16694 16694 D GAC-STORE: SOURCE IND 610
06-03 00:35:08.122 16694 16694 D GAC-STORE: SOURCE IND 710
06-03 00:35:08.122 16694 16694 D GAC-STORE: SOURCE IND 810
06-03 00:35:08.122 16694 16694 D GAC-STORE: SOURCE IND 010
06-03 00:35:08.122 16694 16694 D GAC-STORE: SOURCE IND 110
06-03 00:35:08.122 16694 16694 D GAC-STORE: SOURCE IND 210
06-03 00:35:08.122 16694 16694 D GAC-STORE: SOURCE IND 310
06-03 00:35:08.122 16694 16694 D GAC-STORE: SOURCE IND 410
06-03 00:35:08.122 16694 16694 D GAC-STORE: SOURCE IND 510
06-03 00:35:08.122 16694 16694 D GAC-STORE: SOURCE IND 610
06-03 00:35:08.122 16694 16694 D GAC-STORE: SOURCE IND 710
06-03 00:35:08.122 16694 16694 D GAC-STORE: SOURCE IND 810
06-03 00:35:08.122 16694 16694 D GAC-STORE: SOURCE IND 810
Layout for mPwd
<EditText
android:id="@+id/txtDlgPwd"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_below="@id/chkStorePwd"
android:hint ="Ecnryption Password"
android:maxLines="1"
android:singleLine="true"
android:inputType="textPassword|textNoSuggestions"
android:imeOptions="actionDone"
android:layout_marginLeft="10dp"
android:layout_marginTop="20dp"
android:textSize="15sp"/>
回答1:
Fortunately, Android/Kotlin allows mixing Java and Kotlin files in a single project. I used this as a work around for the problem.
Created a Java class for the password handling. Please notice, CharSequence is a native Java class here, and this makes a difference.
import java.lang.CharSequence;
public class PasswordHandler implements CharSequence {
private CharSequence mSource;
private Switch mUnmask;
public PasswordHandler(CharSequence source, Switch unmask) {
mSource = source;
mUnmask = unmask;
}
public char charAt(int index) {
return mUnmask.isChecked()?mSource.charAt(index):'*';
}
public int length() {
return mSource.length();
}
public CharSequence subSequence(int start, int end) {
return mSource.subSequence(start, end);
}
@NonNull
@Override
public String toString() {
String star = new String();
int l = length();
if (!mUnmask.isChecked()) {
while (l-- > 0){
star += "*";
}
}
return mUnmask.isChecked()? mSource.toString():star;
}
}
Instantiated this class in getTransfromation method in a Kotlin's class. Kotlin compiler took it well this time. Crashes do not happen anymore.I still would like to see a pure Kotlin's solution for this.
mPwd!!.transformationMethod = object : PasswordTransformationMethod() {
override fun getTransformation(source: CharSequence, view: View): CharSequence {
return PasswordHandler(source, mChkUnmask)
}
}
回答2:
class CustomPasswordSymbole : PasswordTransformationMethod() {
override fun getTransformation(source: CharSequence?, view: View?): CharSequence {
return PasswordCharSequence(source)
}
class PasswordCharSequence(source: CharSequence?) : CharSequence{
val mSource = source
override val length: Int=6 ///manually enter lenght of your password
override fun get(index: Int): Char {
return '*'
}
override fun subSequence(startIndex: Int, endIndex: Int): CharSequence {
return mSource!!.subSequence(startIndex, endIndex)
}
}
}
来源:https://stackoverflow.com/questions/50667607/kotlin-crashes-in-passwordtransformationmethod-while-java-works-fine