Android 在布局中定义的 EditText onClick 侦听器失败并出现模糊异常

Android EditText onClick Listener defined in Layout fails with obscure Exception

我想为可能 activity 的 Layout-XML 中的 EditText 定义一个 onClick 侦听器,但它总是失败并出现一个模糊的异常。

Layout 在我的 activity 的 onCreate-Method 中用 setContentView() 注入。我没有在这里使用片段,我很清楚XML定义的onClick侦听器不适用于片段。

出于测试目的,我将相同的处理程序方法添加到位于 EditText 旁边的 ImageView。处理程序在那里工作,在 EditText 上它失败了。所以这是 EditText 的特殊之处,而不是错误定位处理程序方法的一般问题。

这是我的布局文件的相关部分:

<ImageView
     android:layout_gravity="center_horizontal|top"
     android:layout_rowSpan="3"
     android:src="@drawable/ic_action_event"
     android:onClick="onCreationClicked"
     />

<EditText
     android:id="@+id/creation_edit"
     android:focusable="false"
     android:clickable="true"
     android:layout_gravity="fill_horizontal"
     android:hint="@string/enter_creation"
     android:onClick="onCreationClicked"
     style="@style/PickerEditText"
     />

当我单击 ImageView 时,将执行关联的方法并启动 DialogFragment。当我单击 EditText 时,出现以下异常:

    java.lang.IllegalStateException: Could not find a method onCreationClicked(View) in the activity class android.support.v7.internal.widget.TintContextWrapper for onClick handler on view class android.support.v7.widget.AppCompatEditText with id 'creation_edit'
            at android.view.View.onClick(View.java:3994)
            at android.view.View.performClick(View.java:4756)
            at android.view.View$PerformClick.run(View.java:19749)
            at android.os.Handler.handleCallback(Handler.java:739)
            at android.os.Handler.dispatchMessage(Handler.java:95)
            at android.os.Looper.loop(Looper.java:135)
            at android.app.ActivityThread.main(ActivityThread.java:5221)
            at java.lang.reflect.Method.invoke(Native Method)
            at java.lang.reflect.Method.invoke(Method.java:372)
            at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:899)
            at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:694)
     Caused by: java.lang.NoSuchMethodException: onCreationClicked [class android.view.View]
            at java.lang.Class.getMethod(Class.java:664)
            at java.lang.Class.getMethod(Class.java:643)
            at android.view.View.onClick(View.java:3987)
            at android.view.View.performClick(View.java:4756)
            at android.view.View$PerformClick.run(View.java:19749)
            at android.os.Handler.handleCallback(Handler.java:739)
            at android.os.Handler.dispatchMessage(Handler.java:95)
            at android.os.Looper.loop(Looper.java:135)
            at android.app.ActivityThread.main(ActivityThread.java:5221)
            at java.lang.reflect.Method.invoke(Native Method)
            at java.lang.reflect.Method.invoke(Method.java:372)
            at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:899)
            at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:694)

我目前正在测试三星 S4 (Android 5.0.1) API 21 和 AppCompat 库 v7 22.1.0

onCreate 方法中以编程方式添加 onClick 侦听器将是一个解决方案,但我想避免这种情况,因为它会导致单个 onClick 方法有一个大开关语句和基于 XML 的 onClick-Handler 产生了我更喜欢的更具可读性和明确性的代码。

做了额外的研究:

现在我可以重现错误行为和正确行为(见下文)。这似乎是 appcompat-v7 22.1.0 & 22.1.1 引入的错误(或功能)。

我使用 Android Studio with Blank Activity 创建了一个新项目。 minSdk 17, targetSdk 21. 如下所示将 EditText 添加到空白 Activity 并将处理程序方法添加到 Activity 如下所示:

public void onCreationClicked(View view) {
    Toast.makeText(this,"Event Handled",Toast.LENGTH_LONG).show();
}

启动应用触及 EditText --> 崩溃

将我的 build.gradle 更改为:

    compile 'com.android.support:appcompat-v7:22.1.1'

    compile 'com.android.support:appcompat-v7:22.0.0'

并显示 Toast

有什么想法吗?

您需要在 Activity 而不是 Fragment 中定义方法 onCreationClicked(View)。如果您希望 Fragment 处理点击,您需要删除 XML 中的 onClick 并以编程方式处理侦听器。

也许这可以帮助其他人节省搜索解决方案的时间。我为 appcompat7 提出了一个问题,该问题已被接受并收到以下评论:

This has already been fixed for the next platform release, but it may not be feasible to fix this in appcompat.

详情: https://code.google.com/p/android/issues/detail?id=174871

所以这确实是 appcompat 的错误/问题,我将回退到以编程方式添加 onClick 侦听器。

我在 Issue 174871 上使用了 #5 中的以下技巧:

改变: <EditText .../><android.widget.EditText .../>

所以 Appcompat 不再处理 onClick