Skip to content

Commit 7f46ff7

Browse files
author
Adriano Santos
committed
feat: added android extensions to support sensors, locations, and camera events inside actors
1 parent a0784cb commit 7f46ff7

File tree

9 files changed

+271
-14
lines changed

9 files changed

+271
-14
lines changed

settings.gradle.kts

+1
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,7 @@ rootProject.name = "synapsys"
1717
include("synapsys-bom")
1818
include("synapsys-core")
1919
include("synapsys-examples")
20+
include("synapsys-android-extensions")
2021
include("synapsys-store-room")
2122
include("synapsys-store-mysql")
2223
include("synapsys-store-postgres")
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,52 @@
1+
plugins {
2+
id("com.android.library")
3+
id("org.jetbrains.kotlin.multiplatform")
4+
}
5+
6+
android {
7+
namespace = "io.eigr.synapsys.extensions.android.sensors"
8+
compileSdk = 34
9+
10+
defaultConfig {
11+
minSdk = 21
12+
}
13+
14+
compileOptions {
15+
sourceCompatibility = JavaVersion.VERSION_17
16+
targetCompatibility = JavaVersion.VERSION_17
17+
}
18+
19+
buildFeatures {
20+
viewBinding = true
21+
}
22+
}
23+
24+
kotlin {
25+
androidTarget {
26+
compilations.all {
27+
kotlinOptions {
28+
jvmTarget = "17"
29+
}
30+
}
31+
}
32+
33+
jvmToolchain(17)
34+
35+
sourceSets {
36+
val commonMain by getting {
37+
dependencies {
38+
implementation(project.dependencies.platform(project(":synapsys-bom")))
39+
implementation(project(":synapsys-core"))
40+
}
41+
}
42+
43+
val androidMain by getting {
44+
dependencies {
45+
implementation("androidx.room:room-runtime")
46+
implementation("androidx.room:room-ktx")
47+
implementation("androidx.sqlite:sqlite-ktx")
48+
//ksp("androidx.room:room-compiler")
49+
}
50+
}
51+
}
52+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,89 @@
1+
package io.eigr.synapsys.extensions.android.sensors.actor
2+
3+
import android.hardware.Sensor
4+
import android.hardware.SensorEvent
5+
import android.hardware.SensorEventListener
6+
import android.hardware.SensorManager
7+
import io.eigr.synapsys.core.actor.Actor
8+
import io.eigr.synapsys.core.actor.ActorPointer
9+
import io.eigr.synapsys.extensions.android.sensors.events.SensorData
10+
import io.eigr.synapsys.extensions.android.sensors.internals.ActorHandler
11+
import kotlinx.coroutines.runBlocking
12+
import android.content.Context as AndroidContext
13+
import io.eigr.synapsys.core.actor.Context as ActorContext
14+
15+
open class SensorActor<S : Any, M : SensorData>(
16+
id: String,
17+
initialState: S?,
18+
private val androidContext: AndroidContext,
19+
private val sensorType: Int,
20+
private val samplingPeriod: Int = SensorManager.SENSOR_DELAY_NORMAL,
21+
) : Actor<S, M, Unit>(
22+
id = "sensor-actor-${id}-${sensorType}",
23+
initialState = initialState
24+
),
25+
SensorEventListener {
26+
27+
private val sensorManager by lazy {
28+
androidContext.getSystemService(AndroidContext.SENSOR_SERVICE) as SensorManager
29+
}
30+
31+
private lateinit var targetActor: ActorPointer<*>
32+
33+
override fun onStart(ctx: ActorContext<S>): ActorContext<S> {
34+
targetActor = ctx.system.actorOf(
35+
id = "processor-$id",
36+
initialState = initialState!!) {id, state -> ActorHandler(id, state) }
37+
38+
val sensor = sensorManager.getDefaultSensor(sensorType)
39+
sensor?.let {
40+
sensorManager.registerListener(this, it, samplingPeriod)
41+
} ?: {
42+
//ctx.system.log.error("Sensor $sensorType not available")
43+
}
44+
45+
return ctx
46+
}
47+
48+
override fun onStop() {
49+
sensorManager.unregisterListener(this)
50+
}
51+
52+
override fun onReceive(message: M, ctx: ActorContext<S>): Pair<ActorContext<S>, Unit> {
53+
return ctx to Unit
54+
}
55+
56+
@Suppress("UNCHECKED_CAST")
57+
override fun onSensorChanged(event: SensorEvent) {
58+
val message = when (event.sensor.type) {
59+
Sensor.TYPE_ACCELEROMETER -> SensorData.AccelerometerData(
60+
x = event.values[0],
61+
y = event.values[1],
62+
z = event.values[2],
63+
timestamp = event.timestamp
64+
)
65+
66+
Sensor.TYPE_GYROSCOPE -> SensorData.GyroscopeData(
67+
x = event.values[0],
68+
y = event.values[1],
69+
z = event.values[2],
70+
timestamp = event.timestamp
71+
)
72+
73+
Sensor.TYPE_HEART_RATE -> SensorData.HeartRateData(
74+
bpm = event.values[0],
75+
timestamp = event.timestamp
76+
)
77+
78+
else -> SensorData.RawSensorData(
79+
sensorType = event.sensor.type,
80+
values = event.values.copyOf(),
81+
timestamp = event.timestamp
82+
)
83+
}
84+
85+
runBlocking { targetActor.send(message as M) }
86+
}
87+
88+
override fun onAccuracyChanged(sensor: Sensor?, accuracy: Int) {}
89+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,79 @@
1+
package io.eigr.synapsys.extensions.android.sensors.events
2+
3+
sealed class SensorData{
4+
abstract val timestamp: Long
5+
6+
data class AccelerometerData(
7+
val x: Float,
8+
val y: Float,
9+
val z: Float,
10+
override val timestamp: Long
11+
) : SensorData()
12+
13+
data class GyroscopeData(
14+
val x: Float,
15+
val y: Float,
16+
val z: Float,
17+
override val timestamp: Long
18+
) : SensorData()
19+
20+
data class HeartRateData(
21+
val bpm: Float,
22+
override val timestamp: Long
23+
) : SensorData()
24+
25+
data class StepCounterData(
26+
val steps: Int,
27+
override val timestamp: Long
28+
) : SensorData()
29+
30+
data class GpsData(
31+
val latitude: Double,
32+
val longitude: Double,
33+
val altitude: Double? = null,
34+
val accuracy: Float? = null,
35+
val speed: Float? = null,
36+
val bearing: Float? = null,
37+
override val timestamp: Long
38+
) : SensorData() {
39+
40+
/*override fun equals(other: Any?): Boolean {
41+
if (this === other) return true
42+
if (javaClass != other?.javaClass) return false
43+
44+
other as Gps
45+
46+
return latitude == other.latitude &&
47+
longitude == other.longitude &&
48+
altitude == other.altitude &&
49+
accuracy == other.accuracy &&
50+
speed == other.speed &&
51+
bearing == other.bearing &&
52+
timestamp == other.timestamp
53+
}*/
54+
55+
override fun hashCode(): Int {
56+
var result = latitude.hashCode()
57+
result = 31 * result + longitude.hashCode()
58+
result = 31 * result + (altitude?.hashCode() ?: 0)
59+
result = 31 * result + (accuracy?.hashCode() ?: 0)
60+
result = 31 * result + (speed?.hashCode() ?: 0)
61+
result = 31 * result + (bearing?.hashCode() ?: 0)
62+
result = 31 * result + timestamp.hashCode()
63+
return result
64+
}
65+
}
66+
67+
data class RawSensorData(
68+
val sensorType: Int,
69+
val values: FloatArray,
70+
override val timestamp: Long
71+
) : SensorData() {
72+
override fun hashCode(): Int {
73+
var result = sensorType
74+
result = 31 * result + values.contentHashCode()
75+
result = 31 * result + timestamp.hashCode()
76+
return result
77+
}
78+
}
79+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
package io.eigr.synapsys.extensions.android.sensors.internals
2+
3+
import io.eigr.synapsys.core.actor.Actor
4+
import io.eigr.synapsys.core.actor.Context
5+
import io.eigr.synapsys.extensions.android.sensors.actor.SensorActor
6+
import io.eigr.synapsys.extensions.android.sensors.events.SensorData
7+
8+
class ActorHandler<S : Any, M : SensorData>(id: String?, initialState: S) : Actor<S, M, Unit>(id, initialState) {
9+
10+
private lateinit var parentActor: SensorActor<S,M>
11+
12+
internal fun setParentActor(parent: SensorActor<S,M>) {
13+
this.parentActor = parent
14+
}
15+
16+
override fun onReceive(message: M, ctx: Context<S>): Pair<Context<S>, Unit> {
17+
return parentActor.onReceive(message, ctx)
18+
}
19+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
<configuration>
2+
<appender name="STDOUT" class="ch.qos.logback.core.ConsoleAppender">
3+
<encoder>
4+
<pattern>%d{HH:mm:ss.SSS} [%thread] %-5level %logger{36} - %msg%n</pattern>
5+
</encoder>
6+
</appender>
7+
8+
<logger name="io.eigr.synapsys" level="INFO"/>
9+
10+
<root level="INFO">
11+
<appender-ref ref="STDOUT"/>
12+
</root>
13+
</configuration>

synapsys-core/src/main/kotlin/io/eigr/synapsys/core/actor/Actor.kt

+13-5
Original file line numberDiff line numberDiff line change
@@ -20,13 +20,15 @@ import io.eigr.synapsys.core.internals.store.Store
2020
* @see Context
2121
* @see Store
2222
*/
23-
abstract class Actor<S : Any, M : Any, R>(val id: String?, private var initialState: S?) {
23+
abstract class Actor<S : Any, M : Any, R>(val id: String?, var initialState: S?) {
2424
private val log = loggerFor(Actor::class.java)
25+
private lateinit var system: ActorSystem
26+
2527
/**
2628
* Current operational context containing the actor's state.
2729
* Initialized with either the provided initial state or loaded from persistence.
2830
*/
29-
var state: Context<S> = Context(this.initialState)
31+
var state: Context<S> = Context(this.initialState, system)
3032

3133
private val stateClass: Class<S>? = initialState?.javaClass
3234

@@ -37,6 +39,12 @@ abstract class Actor<S : Any, M : Any, R>(val id: String?, private var initialSt
3739
*/
3840
internal var store: Store<S>? = null
3941

42+
open fun onStart(ctx: Context<S>): Context<S> {
43+
return ctx
44+
}
45+
46+
open fun onStop() {}
47+
4048
/**
4149
* Core message handling method that must be implemented by concrete actors.
4250
*
@@ -57,10 +65,10 @@ abstract class Actor<S : Any, M : Any, R>(val id: String?, private var initialSt
5765
log.info("[{}] Rehydrating actor state", id)
5866
val oldState = store?.load(id!!, stateClass!!)
5967
if (oldState != null) {
60-
this.state = Context(oldState)
68+
this.state = Context(oldState, system)
6169
this.state
6270
} else {
63-
this.state = Context(this.initialState)
71+
this.state = Context(this.initialState, system)
6472
this.state
6573
}
6674

@@ -75,7 +83,7 @@ abstract class Actor<S : Any, M : Any, R>(val id: String?, private var initialSt
7583
* @throws IllegalStateException If id is null when attempting to save state
7684
*/
7785
internal suspend fun mutate(state: S): S {
78-
this.state = Context(state)
86+
this.state = Context(state, system)
7987
store?.save(id!!, state)
8088
return state
8189
}

synapsys-core/src/main/kotlin/io/eigr/synapsys/core/actor/Context.kt

+5-1
Original file line numberDiff line numberDiff line change
@@ -12,14 +12,18 @@ package io.eigr.synapsys.core.actor
1212
*
1313
* @see Actor
1414
*/
15-
class Context<S : Any>(private var internalState: S?) {
15+
class Context<S : Any>(private var internalState: S?, private val actorSystem : ActorSystem) {
1616

1717
/**
1818
* Read-only access to the current state value.
1919
* Returns null if state hasn't been initialized.
2020
*/
2121
val state get() = internalState
2222

23+
24+
25+
val system get() = actorSystem
26+
2327
/**
2428
* Creates a new Context instance with updated state while maintaining immutability.
2529
* Original instance remains unchanged.

synapsys-core/src/main/kotlin/io/eigr/synapsys/core/actor/SystemConfigurer.kt

-8
This file was deleted.

0 commit comments

Comments
 (0)