Skip to content

Commit 0ee6d4d

Browse files
committed
Add some comments for JS and multiplatform stuff
1 parent 5243d42 commit 0ee6d4d

File tree

12 files changed

+133
-12
lines changed

12 files changed

+133
-12
lines changed

build.gradle

+3
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,9 @@ buildscript {
1111
}
1212

1313
dependencies {
14+
// NOTE: Unfortunately we need to add these build dependencies in the
15+
// top-level gradle file and not in :frontend, otherwise the build breaks.
16+
// This will likely be resolved once the plugins become more stable.
1417
classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version"
1518
classpath "org.jetbrains.kotlin:kotlin-frontend-plugin:0.0.21"
1619
}

domain/src/main/kotlin/domain/Todo.kt

+11
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,16 @@
11
package domain
22

3+
/**
4+
* Shared domain object for Todos.
5+
*
6+
* Note that we're allowed to use all basic data types as well as types
7+
* defined in kotlin-stdlib-common. We could also implement behavior here,
8+
* as long as we don't rely on platform-specific code.
9+
*
10+
* For platform-specific behavior, we would have to create "header"
11+
* declarations and "impl" them for all targets in their respective
12+
* sub-projects. See https://vimeo.com/215556547#t=868s for more information.
13+
*/
314
data class Todo(
415
val id: String,
516
val title: String,

frontend/build.gradle

+25-1
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,16 @@
11
group 'org.example'
22
version '1.0-SNAPSHOT'
33

4+
// NOTE: The plugins below have been declared in the top-level gradle file.
5+
// For some reason, declaring them here breaks the build. It is likely that
6+
// this issue will be resolved once the plugins become more stable.
7+
8+
// Compile (i.e. transpile) kotlin to JS
9+
// We cannot use kotlin2js, as we need some new functionality
410
apply plugin: 'kotlin-platform-js'
11+
// Add npm, webpack and karma support
512
apply plugin: 'org.jetbrains.kotlin.frontend'
13+
// Add dead code elimination for kotlin.js
614
apply plugin: 'kotlin-dce-js'
715

816
repositories {
@@ -21,8 +29,15 @@ compileKotlin2Js {
2129

2230
dependencies {
2331
compile "org.jetbrains.kotlin:kotlin-stdlib-js:$kotlin_version"
32+
33+
// Since we render HTML directly, we use kotlinx-html-js. There is also
34+
// a server-side implementation to render static HTML. For highly dynamic
35+
// though it makes more sense to use a framework like React or CycleJS.
2436
compile "org.jetbrains.kotlinx:kotlinx-html-js:0.6.4"
2537

38+
// Note "implement" instead of "compile" for dependencies to
39+
// platform-common projects. This is only provided by kotlin-platform-
40+
// common, but not by kotlin2js!
2641
implement project(":domain")
2742
}
2843

@@ -33,16 +48,25 @@ kotlin {
3348
}
3449

3550
kotlinFrontend {
51+
// Generate a package.json and install dependencies during build
3652
npm {
3753
dependency "lodash"
3854
}
3955

56+
// Basic webpack configuration. More config goes in webpack.config.d
4057
webpackBundle {
4158
bundleName = "main"
4259
}
4360

61+
// This is where you switch on minification. Regardless, kotlin-dce
62+
// will always run.
4463
define "PRODUCTION", false
4564
}
4665

47-
// Run DCE on kotlin.js before building the final bundle
66+
// Run DCE on kotlin.js before building the final bundle. Sadly, this
67+
// dependency is not set automatically by kotlin-dce-js, so we have to
68+
// add it. If we leave this out, the minified kotlin.js is generated
69+
// too late and we end up with:
70+
// - failing builds from a clean state
71+
// - an outdated min/kotlin.js in incremental builds
4872
bundle.dependsOn runDceKotlinJs

frontend/src/main/kotlin/actions.kt

+15
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,9 @@ import domain.Todo
22
import org.w3c.fetch.RequestInit
33
import kotlin.browser.window
44

5+
/**
6+
* Creates or updates a [todo] on the server and redraws the UI on success
7+
*/
58
private fun putTodo(todo: Todo) {
69
launch {
710
window.fetch("http://localhost:8080/todo/${todo.id}", object : RequestInit {
@@ -13,12 +16,24 @@ private fun putTodo(todo: Todo) {
1316
}
1417
}
1518

19+
/**
20+
* Creates a new todo with the given [title]
21+
*
22+
* This method is asynchronous and automatically redraws the UI on completion. In case of an error,
23+
* the error is printed to the console and the UI is not updated.
24+
*/
1625
fun createTodo(title: String) {
1726
val id = Lodash.uniqueId("todo_")
1827
val todo = Todo(id = id, title = title)
1928
putTodo(todo)
2029
}
2130

31+
/**
32+
* Toggles the completed state of the given [todo]
33+
*
34+
* This method is asynchronous and automatically redraws the UI on completion. In case of an error,
35+
* the error is printed to the console and the UI is not updated.
36+
*/
2237
fun toggleTodo(todo: Todo) {
2338
putTodo(todo.copy(completed = !todo.completed))
2439
}
+12-1
Original file line numberDiff line numberDiff line change
@@ -1,14 +1,25 @@
11
import kotlin.coroutines.experimental.*
22
import kotlin.js.Promise
33

4+
/**
5+
* Suspends the coroutine until the promise resolves, then returns its resolved value
6+
*/
47
suspend fun <T> Promise<T>.await(): T = suspendCoroutine { cont ->
58
then({ cont.resume(it) }, { cont.resumeWithException(it) })
69
}
710

11+
/**
12+
* Launches an asynchronous block as coroutine
13+
*
14+
* The block is executed with an empty context. If it throws an exception, the error is logged to
15+
* the console.
16+
*/
817
fun launch(block: suspend () -> Unit) {
918
block.startCoroutine(object : Continuation<Unit> {
1019
override val context: CoroutineContext get() = EmptyCoroutineContext
1120
override fun resume(value: Unit) {}
12-
override fun resumeWithException(e: Throwable) { console.log("Coroutine failed: $e") }
21+
override fun resumeWithException(e: Throwable) {
22+
console.log("Coroutine failed: $e")
23+
}
1324
})
1425
}

frontend/src/main/kotlin/lodash.kt

+14-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,17 @@
1+
/**
2+
* Binding to the native lodash implementation (installed via npm).
3+
*
4+
* This requires a module loader that can provide the "lodash" module, likely
5+
* installed with "npm install lodash".
6+
*
7+
* Also, if webpack is configured correctly, tree shaking will make sure that
8+
* all unused lodash functions get eliminated in the final bundle to reduce
9+
* asset size.
10+
*/
111
@JsModule("lodash")
212
external object Lodash {
3-
fun uniqueId(text: String = definedExternally): String;
13+
/**
14+
* Generates a unique ID. If prefix is given, the ID is appended to it
15+
*/
16+
fun uniqueId(prefix: String = definedExternally): String;
417
}

frontend/src/main/kotlin/main.kt

+19-3
Original file line numberDiff line numberDiff line change
@@ -2,8 +2,19 @@ import kotlinx.html.dom.append
22
import kotlin.browser.document
33
import kotlin.browser.window
44

5+
/**
6+
* Loads all Todos and renders the main UI
7+
*/
58
suspend fun render() {
9+
// Asynchronously fetch all TODOs from the server. We suspend until the response
10+
// arrives, so we can immediately render the main UI. Usually, we would display
11+
// a loading indicator here, so the user knows what's going on. Also, it makes
12+
// sense to store TODOs in some local state so we don't have to fetch them every
13+
// time we're rendering the App.
614
val response = window.fetch("http://localhost:8080/todo/").await()
15+
16+
// It is important to transform plain JS objects to domain objects here
17+
// Otherwise, we wouldn't be able to call instance methods on them later on
718
val todos = mapJson(response.json().await(), ::deserializeTodo)
819

920
// Remove any previous content
@@ -17,8 +28,13 @@ suspend fun render() {
1728
}
1829
}
1930

31+
/**
32+
* This is the main entry point of the Todo app. All imported files will be compiled
33+
* into a single bundle by the kotlin2js compiler. Since we use certain methods from
34+
* stdlib we also need to load kotlin.js during runtime.
35+
*/
2036
fun main(args: Array<String>) {
21-
launch {
22-
render()
23-
}
37+
// Since render is suspending, we need to call it in a coroutine context.
38+
// This is handled by the launch helper
39+
launch { render() }
2440
}
+16
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,21 @@
11
import domain.Todo
22

3+
/**
4+
* Converts a plain [json] object into a [Todo]
5+
*
6+
* Note that this function does not validate the JSON before converting.
7+
* Incomplete data, such as a missing id or title could cause errors later
8+
* in the program. Kotlin does not runime-check dynamic values at all.
9+
*/
310
fun deserializeTodo(json: dynamic) = Todo(json.id, json.title, json.text, json.completed)
411

12+
/**
13+
* Converts a plain [json] object into an array of [Todo]s.
14+
*
15+
* Note that this function does not validate the JSON before converting. The
16+
* cast to Array<T> is unsafe and will always succeed. Calling map on the
17+
* result can crash during runtime, however, if the [json] parameter does
18+
* not implement a compatible map method. Kotlin neither runtime-checks the
19+
* cast nor the result type.
20+
*/
521
fun <T> mapJson(json: dynamic, transform: (dynamic) -> T) = (json as Array<T>).map(transform)

frontend/src/main/kotlin/ui.kt

+6-1
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,12 @@
1-
21
import domain.Todo
32
import kotlinx.html.*
43
import kotlinx.html.js.onChangeFunction
54
import kotlinx.html.js.onClickFunction
65
import org.w3c.dom.HTMLInputElement
76

7+
/**
8+
* Renders a single [todo]
9+
*/
810
fun <T, C : kotlinx.html.TagConsumer<T>> C.Todo(todo: Todo) = div {
911
onClickFunction = { toggleTodo(todo) }
1012

@@ -17,6 +19,9 @@ fun <T, C : kotlinx.html.TagConsumer<T>> C.Todo(todo: Todo) = div {
1719
}
1820
}
1921

22+
/**
23+
* Renders a list of [todos] and an input box to create new ones
24+
*/
2025
fun <T, C : kotlinx.html.TagConsumer<T>> C.Todos(todos: List<Todo>) = div {
2126
id = "todos"
2227

frontend/src/main/resources/index.html

+3-3
Original file line numberDiff line numberDiff line change
@@ -2,11 +2,10 @@
22
<html lang="en">
33
<head>
44
<meta charset="UTF-8">
5-
<meta name="viewport"
6-
content="width=device-width, user-scalable=no, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0">
5+
<meta name="viewport" content="width=device-width, user-scalable=no, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0">
76
<meta http-equiv="X-UA-Compatible" content="ie=edge">
87

9-
<title>Document</title>
8+
<title>Todos</title>
109

1110
<link rel="stylesheet" href="http://todomvc.com/examples/react/node_modules/todomvc-app-css/index.css">
1211
<style>
@@ -22,6 +21,7 @@
2221
</style>
2322
</head>
2423
<body class="todoapp">
24+
<!-- The URL could be injected during build time -->
2525
<script src="../../bundle/main.bundle.js"></script>
2626
</body>
2727
</html>

frontend/webpack.config.d/minify.js

+9-1
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,15 @@ if (defined.PRODUCTION) {
33
config.plugins.push(new webpack.optimize.ModuleConcatenationPlugin());
44
config.plugins.push(new webpack.optimize.UglifyJsPlugin({ minimize: true }));
55

6-
// Use minified output form kotlin-dce-js
6+
// Use minified output form kotlin-dce-js. This patches webpack to look for
7+
// minified output in the correct directory, as gradle will automatically move
8+
// all assets to that folder.
79
config.context += '/min';
10+
11+
// The kotlin frontend plugin does not know about dce-js and that it puts the
12+
// minified kotlin.js in a different location. However, is set to a fixed
13+
// file URL in the generated package.json (have a look in build/package.json
14+
// to see where it points to). Therefore, we have to tell webpack's resolver
15+
// where to look for the minified version instead.
816
config.resolve.alias = { kotlin: config.context + '/kotlin.js' };
917
}

settings.gradle

-1
Original file line numberDiff line numberDiff line change
@@ -2,4 +2,3 @@ rootProject.name = 'kotlin-multi-todo'
22
include 'frontend'
33
include 'domain'
44
include 'backend'
5-

0 commit comments

Comments
 (0)