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
    评论
Kotlin自定义 View 的步骤与 Java 中基本相同。下面是一个简单的自定义 View 的示例: 1. 创建一个新的 Kotlin 类,扩展自 View 类。 ``` class MyCustomView(context: Context, attrs: AttributeSet?) : View(context, attrs) { // View 的绘制代码将在这里添加 } ``` 2. 实现 `onMeasure()` 方法来定义 View 的大小。在这个方法中,使用 `setMeasuredDimension()` 方法来设置 View 的宽度和高度。 ``` override fun onMeasure(widthMeasureSpec: Int, heightMeasureSpec: Int) { super.onMeasure(widthMeasureSpec, heightMeasureSpec) val desiredWidth = 100 // 期望的宽度 val desiredHeight = 100 // 期望的高度 val widthMode = MeasureSpec.getMode(widthMeasureSpec) val widthSize = MeasureSpec.getSize(widthMeasureSpec) val heightMode = MeasureSpec.getMode(heightMeasureSpec) val heightSize = MeasureSpec.getSize(heightMeasureSpec) val width = when (widthMode) { MeasureSpec.EXACTLY -> widthSize MeasureSpec.AT_MOST -> min(desiredWidth, widthSize) else -> desiredWidth } val height = when (heightMode) { MeasureSpec.EXACTLY -> heightSize MeasureSpec.AT_MOST -> min(desiredHeight, heightSize) else -> desiredHeight } setMeasuredDimension(width, height) } ``` 3. 实现 `onDraw()` 方法来绘制 View。在这个方法中,可以使用 Canvas API 来绘制所需的视图元素。 ``` override fun onDraw(canvas: Canvas) { super.onDraw(canvas) // 绘制一个圆形 val centerX = width / 2f val centerY = height / 2f val radius = min(centerX, centerY) canvas.drawCircle(centerX, centerY, radius, Paint().apply { color = Color.BLUE style = Paint.Style.FILL }) } ``` 4. 在 XML 布局文件中添加自定义 View。 ``` <com.example.MyCustomView android:layout_width="match_parent" android:layout_height="match_parent" /> ``` 这样就完成了一个简单的自定义 View。当然,这只是一个简单的示例,实际上自定义 View 的实现可能会更加复杂,具体实现方式取决于您的需求。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值