Kotlin-49.JavaScript模块(JavaScript Module)

官方文档: http://kotlinlang.org/docs/reference/js-modules.html

1.JavaScript模块(JavaScript Modules)

Kotlin允许把Kotlin项目编译为JavaScript模块,以下列表是可用js模块选项:
    1.默认选项: Plain,即不编译成任何模块,在全局作用域中以其名称访问模块;
    2.异步模块定义(Asynchronous Module Definition,简称AMD),常被require.js库使用;
    3.CommonJS惯例约定,广泛用于node.js/npm (require函数和module.exports对象)
    4.统一模块定义(Unified Module Definitions,简称UMD),与AMD,CommonJS兼容,
      当在运行时AMD和CommonJS都不可用时,作为plain使用!

2.选择目标模块系统(Choosing the Target Module System)

目标模块系统取决于构建环境(即项目依赖,编译环境):
1.在IntelliJ IDEA中设置JS模块系统
    设置单个模块:
        打开"File -> Project Structure…",在"Modules"中找到模块,并选择"Kotlin",
        在"Module kind"字段中选择合适的模块系统!

    设置整个项目:
        打开"File -> Settings",选择"Build, Execution, Deployment" -> "Compiler" -> "Kotlin compiler",
        在"Module kind"字段中选择合适的模块系统!

2.在Maven中设置JS模块系统
    在pom.xml设置moduleKind属性,选择模块系统
        <plugin>
            <artifactId>kotlin-maven-plugin</artifactId>
            <groupId>org.jetbrains.kotlin</groupId>
            <version>${kotlin.version}</version>
            <executions>
                <execution>
                    <id>compile</id>
                    <goals>
                        <goal>js</goal>
                    </goals>
                </execution>
            </executions>

            <!-- moduleKind可用值: plain, amd, commonjs, umd -->
            <configuration>
                <moduleKind>commonjs</moduleKind>
            </configuration>          
        </plugin>

3.在Gradle中设置JS模块系统
    // 设置moduleKind属性, moduleKind可用值: plain, amd, commonjs, umd
    compileKotlin2Js.kotlinOptions.moduleKind = "commonjs"

3.@JsModule注解(@JsModule annotation)

1.使用@JsModule注解,通知Kotlin一个external类/包/函数/属性是一个JavaScript模块!    
    // CommonJS模块,名叫hello
    module.exports.sayHello = function(name) { alert("Hello, " + name); }

    // 在Kotlin中声明
    @JsModule("hello")
    external fun sayHello(name: String)

2.将@JsModule应用到包(Applying @JsModule to packages)
    一些JavaScript库导出包package(命名空间)而不是函数和类,
    从JavaScript角度看,包package是一个对象(成员是类/函数/属性),把包作为Kotlin对象导入,很不自然! 
    所以编译器允许使用以下notation将导入的JavaScript包映射到Kotlin包:
        // JavaScript模块声明, extModule
        module.exports = {
            foo: {                
            },
            C: {                
            }
        }

        // kotlin
        @file:JsModule("extModule")
        package ext.jspackage.name

        external fun foo()
        external class C

提示:有@file:JsModule注解的文件不能声明非外部成员(non external)
        @file:JsModule("extModule")
        package ext.jspackage.name

        external fun foo()
        fun bar() = "!" + foo() + "!" // 编译期报错


3.导入更深层次的包(Importing deeper package hierarchies)
    在前文示例中JavaScript模块导出单个包,但一些JavaScript库还会从模块中导出多个包,
    Kotlin也支持这种场景,但必须为每个导入的包声明一个新的.kt文件! 
    示例:
        // js模块
        module.exports = {
            mylib: {
                pkg1: {
                    foo: function() {                        
                    },
                    bar: function() {                        
                    }
                },
                pkg2: {
                    baz: function() {                        
                    }
                }
            }
        }

        // 在Kotlin中导入该js模块,必须编写两个Kotlin源文件
            @file:JsModule("extModule")
            @file:JsQualifier("mylib.pkg1")
            package extlib.pkg1

            external fun foo()
            external fun bar()

        // kotlin文件2
            @file:JsModule("extModule")
            @file:JsQualifier("mylib.pkg2")
            package extlib.pkg2

            external fun baz()

4.@JsNonModule注解

当一个声明有@JsModule注解,如果不把它编译成JavaScript模块,就不能在Kotlin中使用它! 
开发人员经常将js库既作为JavaScript模块,也作为可下载的.js文件分发到项目静态资源下,通过<script>元素导入! 
使用@JsNonModule注解,可以让kotlin在非JavaScript模块环境中使用@JsModule声明:
    // JavaScript代码
    function topLevelSayHello(name) { alert("Hello, " + name); }
    if (module && module.exports) {
        module.exports = topLevelSayHello;
    }

    // kotlin
    @JsModule("hello")
    @JsNonModule
    @JsName("topLevelSayHello")
    external fun sayHello(name: String)

提示:

Kotlin把[kotlin.js标准库]作为单个文件分发,该文件本身被编译成UMD模块,
因此可以使用上述任何模块系统,也可以在NPM上使用kotlin package

简书:http://www.jianshu.com/p/6dddb7d4508c
CSDN博客: http://blog.csdn.net/qq_32115439/article/details/75948979
GitHub博客: http://lioil.win/2017/07/23/Kotlin-js-module.html
Coding博客: http://c.lioil.win/2017/07/23/Kotlin-js-module.html

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,378评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,356评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,702评论 0 342
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,259评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,263评论 5 371
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,036评论 1 285
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,349评论 3 400
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,979评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,469评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,938评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,059评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,703评论 4 323
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,257评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,262评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,485评论 1 262
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,501评论 2 354
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,792评论 2 345

推荐阅读更多精彩内容