1
0
Fork 0
happy-cat/src/commonMain/kotlin/fi/schro/data/ElgatoLightRepository.kt

129 lines
4.3 KiB
Kotlin

/*
* Copyright (c) 2022 Florian Schrofner
*
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at https://mozilla.org/MPL/2.0/.
*/
package fi.schro.data
import fi.schro.data.ElgatoLightRepository.Companion.MAX_TEMPERATURE
import fi.schro.data.ElgatoLightRepository.Companion.MIN_TEMPERATURE
import fi.schro.ui.LightPowerStatus
import fi.schro.util.clamp
import io.ktor.client.*
import io.ktor.client.call.*
import io.ktor.client.request.*
import io.ktor.client.statement.*
import io.ktor.http.*
import io.ktor.utils.io.core.*
import kotlinx.serialization.SerialName
import kotlinx.serialization.Serializable
import org.koin.core.component.KoinComponent
import org.koin.core.component.get
import kotlin.math.roundToInt
class ElgatoLightRepository: LightRepository, KoinComponent {
companion object {
const val MIN_TEMPERATURE = 143
const val MAX_TEMPERATURE = 344
}
private val PATH_SEPARATOR = "/"
private val DEFAULT_PATH = "elgato"
private val LIGHT_PATH = "lights"
private val ACCESSORY_INFO_PATH = "accessory-info"
//needs a new http client for every request
private val httpClient: HttpClient get() {
return get()
}
val LIGHT_ENDPOINT = listOf(DEFAULT_PATH, LIGHT_PATH)
val ACCESSORY_INFO_ENDPOINT = listOf(DEFAULT_PATH, ACCESSORY_INFO_PATH)
override suspend fun setLightStatus(lightAddress: String, port: Int?, status: LightStatus) {
val elgatoStatus = ElgatoLightStatus.fromLightStatus(status)
httpClient.use {
val response: HttpResponse = it.put(createUrl(lightAddress, port, LIGHT_ENDPOINT)){
contentType(ContentType.Application.Json)
body = elgatoStatus
}
}
}
override suspend fun getLightStatus(lightAddress: String, port: Int?): LightStatus {
httpClient.use {
val status = it.get<ElgatoLightStatus>(createUrl(lightAddress, port, LIGHT_ENDPOINT))
return status.toLightStatus()
}
}
private fun createPath(pathElements: List<String>): String {
return pathElements.joinToString(separator = PATH_SEPARATOR, prefix = PATH_SEPARATOR)
}
private fun createUrl(lightAddress: String, port: Int?, endpoint: List<String>): String {
return "http://" + lightAddress + ":" + (port ?: 9123) + createPath(endpoint)
}
}
@Serializable
data class ElgatoLightStatus(
@SerialName("lights") val lights: List<ElgatoLight>,
@SerialName("numberOfLights") val numberOfLights: Int = lights.size
){
fun toLightStatus(): LightStatus {
return lights.first().toLightStatus()
}
companion object {
fun fromLightStatus(status: LightStatus): ElgatoLightStatus {
return ElgatoLightStatus(listOf(
ElgatoLight.fromLightStatus(status)
))
}
}
}
@Serializable
data class ElgatoLight(
@SerialName("on") val on: Int? = null,
@SerialName("brightness") val brightness: Int? = null,
@SerialName("temperature") val temperature: Int? = null
){
fun toLightStatus(): LightStatus {
return LightStatus(
powerStatus = on?.let { LightPowerStatus.fromInt(it) },
brightness = brightness,
temperature = temperature?.let { convertElgatoTemperatureToKelvin(it) }
)
}
companion object {
fun fromLightStatus(status: LightStatus): ElgatoLight {
return ElgatoLight(
on = status.powerStatus?.intValue,
brightness = status.brightness,
temperature = status.temperature?.let { convertKelvinToElgatoTemperature(it) }
)
}
}
}
//these calculations are based on the official elgato control center app 1.2.0 for android
private fun convertKelvinToElgatoTemperature(kelvinTemperature: Int): Int {
return ((1000000/kelvinTemperature.toFloat()).roundToInt()).clamp(MIN_TEMPERATURE, MAX_TEMPERATURE)
}
private fun convertElgatoTemperatureToKelvin(elgatoTemperature: Int): Int {
var value = (1000000/elgatoTemperature.toFloat()).roundToInt()
//round to closest multiple of 50
val remainder = value % 50
if(remainder > 25) value += 50
return value - remainder
}