Kotlin @JvmOverloads 自定义 View 的坑

详细的可以看这一篇文章: 《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

idea

写出来的代码长这样:

 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 都会有类似的问题,请大家平时开发过程中要留意。

更多内容,请上车https://eicky.com

发布了33 篇原创文章 · 获赞 34 · 访问量 381万+
展开阅读全文

没有更多推荐了,返回首页

©️2019 CSDN 皮肤主题: 大白 设计师: CSDN官方博客

分享到微信朋友圈

×

扫一扫,手机浏览