Kotlin crashes in PasswordTransformationMethod while Java works fine

南笙酒味 提交于 2020-05-15 06:27:06

问题


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:

  1. 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.
  2. 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

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!