日常开发Bug小记Android开发经验谈我与 Kotlin 的爱恨情仇

Kotlin @JvmOverloads 自定义 View 的坑

2018-12-20  本文已影响9人  IMSk

详细的可以看这一篇文章: 《Do not always trust @JvmOverloads》

@JvmOverloads 注解是用来干撒的?

  /**
 * Instructs the Kotlin compiler to generate overloads for this function that substitute default parameter values.
 *
 * If a method has N parameters and M of which have default values, M overloads are generated: the first one
 * takes N-1 parameters (all but the last one that takes a default value), the second takes N-2 parameters, and so on.
 */
@Target(AnnotationTarget.FUNCTION, AnnotationTarget.CONSTRUCTOR)
@Retention(AnnotationRetention.BINARY)
@MustBeDocumented
public annotation class JvmOverloads

也就是说,如果方法参数中有默认值的,那么这个注解可以帮我们生成多个重载方法,这样方便我们调用,可以省掉一些不关心的参数。

比如这样:

   @JvmOverloads
    fun test(name: String = "hello",  code: Int = 0){
            
     }

最终会帮我们重载三个方法出来

    @JvmOverloads
      public final void test(@NotNull String name, int code) {
         Intrinsics.checkParameterIsNotNull(name, "name");
      }

      @JvmOverloads
      public final void test(@NotNull String name) {
         test$default(this, name, 0, 2, (Object)null);
      }

      @JvmOverloads
      public final void test() {
         test$default(this, (String)null, 0, 3, (Object)null);
      }

@JvmOverloads 自定义View

  class EmoticonView @JvmOverloads constructor(
        context: Context, attrs: AttributeSet? = null, defStyleAttr: Int = 0
) : RelativeLayout(context, attrs, defStyleAttr)

按照上面的介绍,会帮忙自动生成多个重载方法,那么坑来了的,注意这个 defStyleAttr = 0 。。。

我们先看看 EditText 的构造函数, 如果是XML来写的布局,那么会调用第二个构造函数,也就是 public EditText(Context context, AttributeSet attrs) , 那么默认传入的 样式 defStyleAttr 就是 com.android.internal.R.attr.editTextStyle

 public class EditText extends TextView {
    public EditText(Context context) {
        this(context, null);
    }

    public EditText(Context context, AttributeSet attrs) {
        this(context, attrs, com.android.internal.R.attr.editTextStyle);
    }

    public EditText(Context context, AttributeSet attrs, int defStyleAttr) {
        this(context, attrs, defStyleAttr, 0);
    }

    public EditText(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {
        super(context, attrs, defStyleAttr, defStyleRes);
    }

而我们如果使用 IDE 自动提示的 @JvmOverload

image.png

写出来的代码长这样:

 class InputVIew @JvmOverloads constructor(
        context: Context, attrs: AttributeSet? = null, defStyleAttr: Int = 0
) : EditText(context, attrs, defStyleAttr) {
}

看到了么,你的代码 defStyleAttr 默认是 0 而不是 com.android.internal.R.attr.editTextStyle

那么说了这么多,这样会导致什么问题呢?

亲测,焦点没有了的,点击之后键盘也无法弹起。

那么这个坑 如何填呢?

修改默认值即可:

class InputVIew @JvmOverloads constructor(
        context: Context, attrs: AttributeSet? = null, defStyleAttr: Int = android.support.design.R.attr.editTextStyle
) : EditText(context, attrs, defStyleAttr)

注意

像我在开头提到的文章 《Do not always trust @JvmOverloads》 ,在最后写到: Button, EditText, RadioButton, Switch 都会有类似的问题,请大家平时开发过程中要留意。

上一篇下一篇

猜你喜欢

热点阅读