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

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值