【问题标题】:Convert java to kotlin breaks contextmenu将 java 转换为 kotlin 会破坏上下文菜单
【发布时间】:2020-09-01 22:19:08
【问题描述】:

我一直在将一个项目转换为 Kotlin 并发现了一个问题。 java 代码中的上下文菜单在生成的 kotlin 中被破坏。
这是对项目源代码的简化测试。它仅由具有单个布局和上下文菜单的主要活动组成。 java 版本有效,但 kotlin 版本崩溃。我能想到的唯一不寻常的是,我正在注册的视图是一个相对布局中的 imageView。

 java.lang.NullPointerException:
        Parameter specified as non-null is null: 
        method kotlin.jvm.internal.Intrinsics.checkNotNullParameter
        , parameter menuInfo
    at com...MainActivity.onCreateContextMenu(MainActivity.kt)
    at android.view.View.createContextMenu(View.java:8392)
    at com.android.internal.view.menu.ContextMenuBuilder
        .show(ContextMenuBuilder.java:81)
    at com.android.internal.policy.impl
        .PhoneWindow$DecorView
        .showContextMenuForChild(PhoneWindow.java:2517)
    at android.view.ViewGroup.showContextMenuForChild(ViewGroup.java:658)

MainActivity.java 是:

public class MainActivity extends AppCompatActivity {
    private static int animationSpeed = 0;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        registerForContextMenu(findViewById(R.id.imageView));
    }
    @Override
    public void onCreateContextMenu(ContextMenu menu, View v,
                ContextMenu.ContextMenuInfo menuInfo) {
        super.onCreateContextMenu(menu, v, menuInfo);
        MenuInflater inflater = getMenuInflater();
        inflater.inflate(R.menu.speed_select, menu);
        menu.getItem(animationSpeed).setChecked(true);
    }
    @Override
    public boolean onContextItemSelected(MenuItem item) {
        int itemId = item.getItemId();
        boolean rv = true;
        switch(itemId) {
            case R.id.animate_slow: animationSpeed = 0; break;
            case R.id.animate_normal: animationSpeed = 1; break;
            case R.id.animate_fast: animationSpeed = 2; break;
            default: Log.d("onContextItemSelected", String.format(
                    "menu item unhandled:0x%08x", itemId)
            );
                rv = false;
        }
        return rv;
    }
}

MainActivity.kt 是:

class MainActivity : AppCompatActivity() {
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)
        registerForContextMenu(findViewById(R.id.imageView))
    }
    override fun onCreateContextMenu(menu: ContextMenu, v: View,
                                     menuInfo: ContextMenuInfo) {
        super.onCreateContextMenu(menu, v, menuInfo)
        val inflater = menuInflater
        inflater.inflate(R.menu.speed_select, menu)
        menu.getItem(animationSpeed).isChecked = true
    }
    override fun onContextItemSelected(item: MenuItem): Boolean {
        val itemId = item.itemId
        var rv = true
        when (itemId) {
            R.id.animate_slow -> animationSpeed = 0
            R.id.animate_normal -> animationSpeed = 1
            R.id.animate_fast -> animationSpeed = 2
            else -> {
                Log.d("onContextItemSelected", String.format(
                        "menu item unhandled:0x%08x", itemId))
                rv = false
            }
        }
        return rv
    }
    companion object {
        private var animationSpeed = 0
    }
}

我的菜单文件是:

<?xml version="1.0" encoding="utf-8"?>
<menu xmlns:android="http://schemas.android.com/apk/res/android" >
    <group 
        android:checkableBehavior="single"
        android:id="@+id/animate_speed" >
       <item android:id="@+id/animate_slow"
             android:title="@string/slow" />
       <item android:id="@+id/animate_normal"
             android:title="@string/normal" />
       <item android:id="@+id/animate_fast"
             android:title="@string/fast" />
       </group>
</menu>

活动布局为:

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/relative_layout"
    android:layout_width="match_parent"
    android:layout_height="match_parent">
    <ImageView
        android:id="@+id/imageView"
        android:layout_width="250dp"
        android:layout_height="250dp"
        android:layout_centerInParent="true"
        android:background="@drawable/andy"
        />
</RelativeLayout>

我尝试过闯入 onCreateContextMenu,但从未到达那里。
我正在使用 Kotlin 1.40、AndroidStudio 4.01、SDK 30 和 gradle 4.01。几天来我一直在查看文档和代码,对我来说,生成的 kotlin 看起来不错。
谢谢!

感谢下面的 John Healy,此问题已解决。 John 说他认为这可能是在 Kotlin 的零安全处理中。我对此表示怀疑,因此我在工作 Java 代码中添加了一条日志语句,并且 menuInfo 以 null 的形式出现。我在给我的 Java 声明中添加了一个 @Nullable 注释:

public void onCreateContextMenu(
    ContextMenu menu, View v,
    @Nullable ContextMenu.ContextMenuInfo menuInfo)

对 Java 代码的测试表明编译器和 lint 都很满意,并且代码仍然可以运行。我再次通过转换过程运行了 jave,结果函数的 kotlin 签名是:

override fun onCreateContextMenu(
    menu: ContextMenu, v: View,
    menuInfo: ContextMenuInfo?)

我测试了 Kotlin,它现在也可以工作了!

注意:为了您的启迪和娱乐,我将来源发布在 git hub.

【问题讨论】:

  • Java代码有ContextMenu.ContextMenuInfo menuInfo,指定ContextMenuInfoContextMenu的静态成员类,而Kotlin代码只有menuInfo: ContextMenuInfo有关系吗?在MainActivity#onCreateContextMenu.
  • 我想我找到了问题 - 检查 stackoverflow.com/questions/53794754/… - 所以它应该是 menuInfo: ContextMenuInfo? - 我想这只是向编译器发出信号的一种方式,你知道 null 是一个有效值参数。如果 Kotlin 知道值永远不会是 null,它可能会优化您的代码。
  • 谢谢@JonnyHenly!我永远不会发现。我会在我的问题中发布一个说明,解释解决方案。那么 - 这是一个错误吗?如果是这样(对我来说这似乎是一个错误)我应该向谁报告?
  • 所以它确实闻起来像一个错误,特别是对于 Android SDK 中这样一个被覆盖的方法,但我认为它更多的是 Java -> Kotlin 转换器的限制。解析该方法是否在早期检查 if (menuItem == null) 并随后将 ? 附加到参数类型,这似乎很简单。同时,我从未尝试过创建转换器,所以我确信有许多警告和异常值会使此评论看起来很愚蠢。
  • 我对此进行了更多研究。我认为我遇到问题的原因是我没有创建菜单项,而是使用 menu.xml 布局并对其进行膨胀,所以我不在这里查看单个 menuItems。取而代之的是,我在 onContextItemSelected 中捕获了菜单选择。所以——在我的例子中,菜单项是空的,因为没有单独的菜单项,并且该字段应该是@Nullable。我寻找 Activity.onCreateContextMenu 的来源,它似乎并不在线。在任何情况下,修复都有效。如果有人发现信息:相反的menuItem,请告诉我。

标签: java android kotlin contextmenu


【解决方案1】:

我只是对我的问题进行此回复,因此人们会注意到由于评论者而找到了解决方案。此修复仅适用于您使用菜单列表而不是创建单个菜单项的情况,并且该修复仅对 Kotlin 是必需的,因为 Kotlin 处理空值安全的方式。请看问题的结尾和我对该问题的评论,
史蒂夫·S.

【讨论】:

  • 在我看来,一般来说最好让评论者提交和回答,这样你就可以投票并接受它。将答案放在答案中而不是在问题上也更干净。 (刚才说了,如果你愿意,你可以考虑下一个问题)。问候。
猜你喜欢
  • 1970-01-01
  • 2021-01-30
  • 2016-07-07
  • 2020-07-27
  • 1970-01-01
  • 1970-01-01
  • 2016-01-20
  • 2018-11-26
  • 2015-04-07
相关资源
最近更新 更多