改进翻译

Working with JavaScript libraries

最近更新 2017-02-27
A look at how we use JavaScript libraries with Kotlin/JS.

Warning: this tutorial does not reflect all newest changes for Kotlin 1.3.71. To get started with writing modern Kotlin/JS projects, please see Setting up a Kotlin/JS project

在本教程中,我们将了解如何

使用 dukat 生成用于 Kotlin 的头文件

标准库中提供了一系列有关 DOM 的包装器以及使用静态类型与 JavaScript 配合使用的函数。 但是,当使用 jQuery 之类的库时会发生什么?对于 JavaScript 生态系统上可用的所有不同库,Kotlin 没有自己的“头”文件,但是 TypeScript 有。 Definitely Typed 仓库中提供了大量的头文件可供选择。

使用 dukat 工具可以将任何 TypeScript 声明文件转换为 Kotlin 的。要安装该工具,可以使用 npm

npm -g install dukat

要转换文件,只需提供输入文件,还可以提供输出目录。 以下命令将在当前文件夹中转换文件 jquery.d.ts,该文件是先前从 Definitely Typed 仓库中下载到输出文件夹 headers 的:

dukat -d headers jquery.d.ts 

生成文件后,便可简单地将其包含在项目中并使用它:

jQuery("#area").hover { window.alert("Hello!") }

注意,jQuery 需要包含在相应的 HTML 中:

<script type="text/javascript" src="js/jquery.js"></script>

<!-- 其他脚本文件…… -->

遮蔽头文件

头文件仅包含在运行时定义的函数的函数声明。例如,可以像这样定义一个 jQuery 函数:

@JsName("$")
public external fun jQuery(selector: String): JQuery

上面的代码表明该函数是在外部定义的。@JsName("$") 注解允许在运行时将名称映射到 $。 有关外部声明的更多详细信息,请参阅 JavaScript 互操作文档

请注意,TypeScript 与 Kotlin 的类型系统不完全匹配, 因此,如果在使用 Kotlin 的 API 时遇到困难,则可能需要编辑生成的头文件。

使用 Dynamic

尽管上述解决方案有相应头文件(是自己定义的,或从 TypeScript 头转换而来的文件)的情况下很好用,但通常需要使用一些没有头文件的库。 例如,要使用 jQuery 插件,该插件可让我们将 HTML 表转换为漂亮的可导航网格。

通过 JavaScript 使用的方式是在相应的 <table> 元素上调用 dataTable()

<table id="data" class="display" cellspacing="0" width="100%">
    <thead>
    <tr>
        <th>Name</th>
        <th>Position</th>
        <th>Office</th>
        <th>Age</th>
    </tr>
    </thead>
    <tbody>
    <tr>
        <td>Tiger Nixon</td>
        <td>System Architect</td>
        <td>Edinburgh</td>
        <td>61</td>
    </tr>
    <tr>
        <td>Garrett Winters</td>
        <td>Accountant</td>
        <td>Tokyo</td>
        <td>63</td>
    </tr>
    <tr>
        <td>Ashton Cox</td>
        <td>Junior Technical Author</td>
        <td>San Francisco</td>
        <td>66</td>
    </tr>
    
      . . . 
    
    </tbody>
</table>

在 JavaScript 中调用以下内容

$("#data").dataTable()

假设函数 dataTable() 不存在,该如何从 Kotlin 中做到这一点?调用它会导致编译器错误?

在这种情况下,可以使用 dynamic,该动态类型允许在面向 JavaScript 编程时使用动态类型。 以下变量被声明为 dynamic,这意味着对其进行的调用均不会导致编译时错误:

val myObject: dynamic = null

{ ... }

myObject.callAnything()

上面的代码进行编译。 但是,如果在使用前未正确初始化对象或在运行时未定义 callAnything(),将会产生运行时错误。

标准库定义了一个名为 asDynamic() 的函数,该函数将值转换为动态类型。 在前面的示例中,使用了 jQuery 处理 DOM 元素,现在可以将其与 asDynamic() 结合起来,然后在结果上调用 dataTable()

$("#data").asDynamic().dataTable()

重要的是要理解,就像在 callAnything() 的情况下一样,dataTable() 函数必须在运行时存在。 这种情况下,需要确保包含插件的相应脚本文件:

<script type="text/javascript" src="js/jquery.js"></script>
<script type="text/javascript" src="js/jquery.dataTables.js"></script>

<!-- 其他脚本文件…… -->

有关 dynamic 的更多信息,请参见动态类型