Wer, Wie, Was?
Spieleprogrammierung in Kotlin
Tobse Fritz
data:image/s3,"s3://crabby-images/1b303/1b30345289e57fa6e653531f8d2ae2077813e9c1" alt=""
data:image/s3,"s3://crabby-images/21d18/21d181dbb3ff8fe028e8f9974b1e38587de8ce9c" alt=""
data:image/s3,"s3://crabby-images/307e2/307e2093331c12720923ed4890869fb4138cb7a1" alt=""
ITscope GmbH
Java Entwickler
Kotlin Fanboy
Hobby Fotograf
Gaming Nerd
3D Printer
Daddy
Mit ITscope vereinfachen Systemhäuser
Ihre Verkaufsprozesse & IT Beschaffung
data:image/s3,"s3://crabby-images/e19a0/e19a01c0d8f3ea9d2bd794cb76bd7d02a58d30da" alt=""
WO
WO
data:image/s3,"s3://crabby-images/6d419/6d419759e72e452ec26f13b74c8cae335b9cfbf2" alt=""
WO
data:image/s3,"s3://crabby-images/6d419/6d419759e72e452ec26f13b74c8cae335b9cfbf2" alt=""
data:image/s3,"s3://crabby-images/349ee/349ee03c5822978d33004b03a30022bb6c9f661b" alt=""
WO
data:image/s3,"s3://crabby-images/349ee/349ee03c5822978d33004b03a30022bb6c9f661b" alt=""
data:image/s3,"s3://crabby-images/f4f1e/f4f1e6c528c7f4a7505dda29cb8e44ef60ca45b0" alt=""
2003
Gründung
62
Mitarbeiter
7.800
Aktive Nutzer
420+
Distributoren
3 Mio
Produkte
Fakten
Tech Stack
Spieleprogrammierung in Kotlin
My Games
data:image/s3,"s3://crabby-images/3d732/3d7323742fe13e6a768eb731272857208224244b" alt=""
Hospital
data:image/s3,"s3://crabby-images/a427b/a427bf2ad24cdbbd7a350cb1a6c6e8ee7990f883" alt=""
Kathis Kleiner Sattelit
My Games
Swing Plus Remake
data:image/s3,"s3://crabby-images/e97b7/e97b73047c8bd698fe8131a1b9aad956563d961a" alt=""
My Games
Hit Klack Remake
data:image/s3,"s3://crabby-images/5ed25/5ed25a95c3fa24c132692ec97de440c5aa40e5d5" alt=""
data:image/s3,"s3://crabby-images/c05f5/c05f5f7dcf6d72b1a1f2034fe3e9a28d20accbc3" alt=""
My Games
My Games
data:image/s3,"s3://crabby-images/d5173/d517371079b9af88d835e588886c3c68a78f33ee" alt=""
Candy Crush Clone
Magic Maze Online Multiplayer
data:image/s3,"s3://crabby-images/86f5b/86f5bfaefd3ec5a028c178b2f885a160f4f7f961" alt=""
My Games
K
or
G
E
K
or
otlin
G
E
K
or
otlin
C
outines
G
E
K
or
otlin
C
outines
G
E
ame
ngine
K
or
G
E
Kotlin Multiplatform
JVM
Win
Mac
Linux
JS
Browser
Native
Android
IOS
K
or
G
E
JVM
Win
Mac
Linux
JS
Browser
Win
Mac
Linux
Native
Android
IOS
Common
Code
Kotlin
Compiler
LLVM IR
Kotlin Multiplatform
K
or
G
E
Kotlin Multiplatform
JVM
Win
Mac
Linux
JS
Browser
Win
Mac
Linux
Native
Android
IOS
KorGE
Libs
Common
Code
Kotlin
Compiler
LLVM IR
K
or
G
E
Kotlin Multiplatform
JVM
Win
Mac
Linux
JS
Browser
Win
Mac
Linux
Native
Android
IOS
KorGE
Libs
Common
Code
data:image/s3,"s3://crabby-images/a5a1f/a5a1f7162e572350f6a46f35e382fef6fb2538f9" alt=""
expect
expect fun play(file: Sound)
actual
actual
actual
actual
actual
K
or
G
E
.org
K
or
G
E
- 2D Game Engine
- Open Source
- Native Multiplatform
- Live Debugging
- IDE Integration
- Visual editor
- 100 % Kotlin
⭐Features
Kotlin
Spiele... warum in
Kotlin
- Prägnant
- Sicher
- Interoperabel
- Optimales Tooling
- Frei & OpenSource
Features
Kotlin
public class Person {
private final String firstName;
private final String lastName;
private final LocalDate birthDay;
public Person(String firstName, String lastName,
LocalDate birthDay) {
this.firstName = firstName;
this.lastName = lastName;
this.birthDay = birthDay;
}
public String getFirstName() {
return firstName;
}
public String getLastName() {
return lastName;
}
public LocalDate getBirthDay() {
return birthDay;
}
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
Person person = (Person) o;
return Objects.equals(firstName, person.firstName) &&
Objects.equals(lastName, person.lastName) &&
Objects.equals(birthDay, person.birthDay);
}
@Override
public int hashCode() {
return Objects.hash(firstName, lastName, birthDay);
}
@Override
public String toString() {
return "Person{" +
"firstName='" + firstName + '\'' +
", lastName='" + lastName + '\'' +
", birthDay=" + birthDay +
'}';
}
}
data class PersonK(
val firstName: String,
val lastName: String,
val birthDay: LocalDate
)
Java
Kotlin
Prägnanz
Kotlin
public class RectangleJava {
private int x = 0;
private int y = 0;
private int width = 0;
private int height = 0;
public boolean contains(int x, int y) {
var inWidth = (x >= this.x) && (x <= this.x + width);
var inHeight = (y >= this.y) && (y <= this.y + height);
return inWidth && inHeight;
}
}
Java
Kotlin
public class RectangleKotlin {
private val x = 0;
private val y = 0;
private val width = 0;
private val height = 0;
public fun contains(x: Int, y: Int): Boolean {
val inWidth = (x >= this.x) && (x <= this.x + width);
val inHeight = (y >= this.y) && (y <= this.y + height);
return inWidth && inHeight;
}
}
Kotlin
public fun public private fun
new
;
Kotlin
class RectangleKotlinNice {
private val x = 0
private val y = 0
private val width = 0
private val height = 0
fun contains(x: Int, y: Int): Boolean {
val inWidth = x in (x .. this.x + width)
val inHeight = y in (y .. this.y + height)
return inWidth && inHeight
}
}
Kotlin
Kotlin
- Prägnant
- Sicher
- Interoperabel
- Optimales Tooling
- Frei & OpenSource
Features
Kotlin
Prägnant
Sicher
Interoperabel
Optimales Tooling
Frei & OpenSource
Inline Classes
Extensions
Contracts
Operator Overloading
Gradle DSL
Data Classes
Coroutines
KorGE
The Tree Game
data:image/s3,"s3://crabby-images/3fdc1/3fdc1cf7b975ac83159da0ff4bfc7e6eeb8b7f19" alt=""
KorGE
The Tree Game
The Tree Game
Lesson learned
data:image/s3,"s3://crabby-images/32214/32214c0307281d0aac87ed2dc6983a066344ee9b" alt=""
data:image/s3,"s3://crabby-images/a9a80/a9a8020167972673d352e1ae35d375783e0b9d0e" alt=""
data:image/s3,"s3://crabby-images/9eb93/9eb93c15d144576d65f5a3609046efc94c7c7a90" alt=""
image(backgroundBitmap){
size(gameWidth, gameHeight)
}
image(treeBitmap).centerOnStage()
Image(appleBitmap).addTo(this)
container {
name = "Happy Sun"
circle(radius = 50.0, fill = YELLOW)
circle(radius = 8.0, fill = BLACK).position(60, 20)
circle(radius = 8.0, fill = BLACK).position(24, 20)
circle(radius = 10.0, fill = RED).position(50, 60)
position(50, 50)
}
The Tree Game
private suspend fun hit() {
tween(this::y[gameHeight],
time = 1.seconds, easing = EASE_IN)
tween(this::rotation[Random[-40, 40].degrees],
time = 100.milliseconds, easing = EASE_IN)
}
data:image/s3,"s3://crabby-images/d821f/d821f14bf416f83d403763eea2ff6642ff13ab7f" alt=""
EASE_IN_ELASTIC EASE_OUT_ELASTIC EASE_OUT_BOUNCE LINEAR EASE_IN EASE_OUT EASE_IN_OUT EASE_OUT_IN EASE_IN_BACK EASE_OUT_BACK
EASE_IN_OUT_BACK EASE_OUT_IN_BACK EASE_IN_OUT_ELASTIC EASE_OUT_IN_ELASTIC EASE_IN_BOUNCE EASE_IN_OUT_BOUNCE EASE_OUT_IN_BOUNCE EASE_IN_QUAD EASE_OUT_QUAD EASE_IN_OUT_QUAD
Lesson learned
The Tree Game
val bus = GlobalBus()
class GameOverEvent()
private fun eatApple() {
if (apples == 0) {
bus.send(GameOverEvent())
}
}
bus.register<GameOverEvent> {
text("Game Over", textSize = 64.0).centerOnStage()
}
private suspend fun hit() {
tween(this::y[gameHeight],
time = 1.seconds, easing = EASE_IN)
tween(this::rotation[Random[-40, 40].degrees],
time = 100.milliseconds, easing = EASE_IN)
}
Game Over
Lesson learned
The Tree Game
data:image/s3,"s3://crabby-images/9eb93/9eb93c15d144576d65f5a3609046efc94c7c7a90" alt=""
class Bird(...){
init {
onClick {
hit()
}
}
class Bird(...){
init {
onCollision(filter = { it is Image}) {
if (it.name == "apple") {
it.removeFromParent()
eatApple()
}
}
}
}
Lesson learned
resourcesVfs["bird.mp3"].readSound().play()
The Tree Game
JS
Browser
runAndroidRelease
runJs
runJvm
runNativeRelease
Native
Android
Native
IOS
Win
Mac
Linux
Native
JVM
Win
Mac
Linux
Tasks > run
iosRunSimulatorDebug
Lesson learned
The Tree Game
Coming soon
🆕
Jetzt mit Hörnchen!
The Tree Game
gamepad.down(0, GameButton.LEFT) {
x -= 40
}
gamepad.down(0, GameButton.RIGHT) {
x += 40
}
gamepad.down(0, GameButton.BUTTON0) {
shoot()
}
keys {
down {
if (it.key == Key.LEFT) {
x -= 20
}
if (it.key == Key.RIGHT) {
x += 20
}
if (it.key == Key.SPACE) {
shoot()
}
}
}
Coming soon
Box2D
solidRect(50, 50, Colors.RED)
solidRect(50, 50, Colors.RED)
solidRect(50, 50, Colors.RED)
solidRect(600, 100, Colors.WHITE)
Box2D
solidRect(50, 50, Colors.RED)
.registerBodyWithFixture(type = BodyType.DYNAMIC, density = 2, friction = 0.01)
solidRect(50, 50, Colors.RED)
.registerBodyWithFixture(type = BodyType.DYNAMIC)
solidRect(50, 50, Colors.RED)
.registerBodyWithFixture(type = BodyType.DYNAMIC)
solidRect(600, 100, Colors.WHITE)
.registerBodyWithFixture(
type = BodyType.STATIC,
friction = 0.2
)
Box2D
solidRect(50, 50, Colors.RED)
.registerBodyWithFixture(type = BodyType.DYNAMIC, density = 2, friction = 0.01)
solidRect(50, 50, Colors.RED)
.registerBodyWithFixture(type = BodyType.DYNAMIC)
solidRect(50, 50, Colors.RED)
.registerBodyWithFixture(type = BodyType.DYNAMIC)
solidRect(600, 100, Colors.WHITE)
.registerBodyWithFixture(
type = BodyType.STATIC,
friction = 0.2
)
onClick {
val pos = it.currentPosLocal
solidRect(50, 50, Colors.RED).position(pos.x, pos.y).rotation(randomAngle())
.registerBodyWithFixture(type = BodyType.DYNAMIC)
}
fun randomAngle(): Angle = Random.nextInt(0, 90).degrees
Box2D
data:image/s3,"s3://crabby-images/f4011/f4011b38274a164b50ae9a68f7ad61710afe965b" alt=""
Kor
UI
val container = fixedSizeContainer(width, height)
container.korui {
vertical {
layoutChildrenPadding = 2
horizontal {
preferredWidth = 100.percent
button("HELLO", {
preferredWidth = 70.percent
})
button("WORLD", {
preferredWidth = 30.percent
preferredHeight = 32.pt
})
}
button("TEST")
checkBox("CheckBox", checked = true)
comboBox("test", listOf("test", "demo"))
}
}
DSL
data:image/s3,"s3://crabby-images/f4011/f4011b38274a164b50ae9a68f7ad61710afe965b" alt=""
Kor
UI
Absolute Positionierung
uiButton(256.0, 32.0) {
text = "Disabled Button"
position(128, 128)
}
val emitter = resourcesVfs["particle2.pex"].readParticleEmitter()
particleEmitter(emitter)
Particles
data:image/s3,"s3://crabby-images/5b6f6/5b6f693a092de3cdde2c1c7241f414f810eeeec7" alt=""
Particles
Hiero
data:image/s3,"s3://crabby-images/79cf6/79cf628960ae53fdddf8a88e09384979d83ef875" alt=""
data:image/s3,"s3://crabby-images/87bbe/87bbe2bfea9cfa9621f4788f189cf7ed58f67247" alt=""
data:image/s3,"s3://crabby-images/ce872/ce8724d5e0ebe6b67f78f09a5edf3a61d4ed474d" alt=""
data:image/s3,"s3://crabby-images/0ac7a/0ac7a801a2b8bfc09091b5375db0a5717e2672b0" alt=""
data:image/s3,"s3://crabby-images/ae209/ae209da4b064eafa7fb55c0ccac0ba7ccec7663b" alt=""
data:image/s3,"s3://crabby-images/a9321/a932140b6ff8f48fcc97f48a0af6b0454df3bbcb" alt=""
data:image/s3,"s3://crabby-images/b2a15/b2a15ef16cd609ffa06b23da297d9aaac4443a1c" alt=""
data:image/s3,"s3://crabby-images/a9321/a932140b6ff8f48fcc97f48a0af6b0454df3bbcb" alt=""
Axiom Verge
Brigador
2016
2015
Edit Worlds
Performance
10K
Performance
10K
1Mio
100K
need
more?
Performance
data:image/s3,"s3://crabby-images/cb83d/cb83de3977627c97a17000ad8aa56ffaed0a19a3" alt=""
https://store.steampowered.com/app/379720/DOOM/
- 😁Weil es Spaß macht
- 💡 Programmier-Skills verbessern
- 🔭Neues ausprobieren
- 👨👦Für Entwicklung begeistern
- 😊Wertschätzung für eigene Arbeit
- 🤝Zusammenarbeit im Team
- 👾Kleine abgeschlossene Projekte
Warum ein Spiel programmieren?
data:image/s3,"s3://crabby-images/cbca4/cbca4c590f200fccc374fb2dddc6df68b9c1b0b1" alt=""
https://produktivkeller.com/game-jams
29 - 31 Oktober 2021
data:image/s3,"s3://crabby-images/544b3/544b33a4f3774af8d6d321a76757534792802e48" alt=""
https://globalgamejam.org
28 - 30 Januar 2022
28 000 creators
104 Countries
6 000 Games
Warum ein Spiel programmieren?
Warum ein Spiel programmieren?
data:image/s3,"s3://crabby-images/cbca4/cbca4c590f200fccc374fb2dddc6df68b9c1b0b1" alt=""
data:image/s3,"s3://crabby-images/9c4f6/9c4f60f3a17e7b994c5759fc807fdba1f95aadd7" alt=""
Jetzt bist Du dran!
https://globalgamejam.org/news/poster-showcase-2018-0
data:image/s3,"s3://crabby-images/9c4f6/9c4f60f3a17e7b994c5759fc807fdba1f95aadd7" alt=""
git clone https://github.com/korlibs/korge-hello-world
data:image/s3,"s3://crabby-images/9c4f6/9c4f60f3a17e7b994c5759fc807fdba1f95aadd7" alt=""
data:image/s3,"s3://crabby-images/9c4f6/9c4f60f3a17e7b994c5759fc807fdba1f95aadd7" alt=""
Tools
- Tiled (Level Editor)
- Spine (Paid, Character Animation)
- Free Sprite Sheet Packer (Web App)
- Texture Packer (Paid)
- Hiero (Bitmap Font Editor)
-
KorGE IDE Plugin
- Visual Editor
- Particle Editor
- Level Editor
data:image/s3,"s3://crabby-images/9c4f6/9c4f60f3a17e7b994c5759fc807fdba1f95aadd7" alt=""
data:image/s3,"s3://crabby-images/4be66/4be668577d0f31801e3515749164d754176fd47c" alt=""
data:image/s3,"s3://crabby-images/61d9f/61d9f7b62fa8b062ab93bd85260768d87b608a8e" alt=""
Werde Teil der ITscope Familie
Werde Teil der ITscope Familie
data:image/s3,"s3://crabby-images/61d9f/61d9f7b62fa8b062ab93bd85260768d87b608a8e" alt=""
Java Backend Developer (m/w/d)
Festanstellung, Vollzeit
Informatikstudenten (m/w/d)
Werkstudent, Teilzeit
Werkstudent (m/w/d) im Marketing
Werkstudent, Teilzeit
Werde Teil der ITscope Familie
data:image/s3,"s3://crabby-images/61d9f/61d9f7b62fa8b062ab93bd85260768d87b608a8e" alt=""
ITscope - Carrer Contacts
By Tobse Fritz
ITscope - Carrer Contacts
Spieleprogrammierung mit KorGE - Der Kotlin Multiplatform Game Engine
- 750