
extra.orx-rabbit-control.0.4.5-alpha6.source-code.RabbitControlServer.kt Maven / Gradle / Ivy
import com.google.zxing.BarcodeFormat
import com.google.zxing.qrcode.QRCodeWriter
import io.ktor.server.engine.*
import io.ktor.server.http.content.*
import io.ktor.server.netty.*
import io.ktor.server.routing.*
import org.openrndr.Extension
import org.openrndr.Program
import org.openrndr.color.ColorRGBa
import org.openrndr.draw.*
import org.openrndr.extra.compositor.*
import org.openrndr.extra.fx.blend.Darken
import org.openrndr.extra.imageFit.FitMethod
import org.openrndr.extra.imageFit.imageFit
import org.openrndr.extra.parameters.Parameter
import org.openrndr.extra.parameters.ParameterType
import org.openrndr.extra.parameters.listParameters
import org.openrndr.math.Vector2
import org.openrndr.math.Vector3
import org.openrndr.math.Vector4
import org.openrndr.math.mix
import org.rabbitcontrol.rcp.RCPServer
import org.rabbitcontrol.rcp.model.interfaces.IParameter
import org.rabbitcontrol.rcp.model.parameter.*
import org.rabbitcontrol.rcp.transport.websocket.server.RabbitHoleWsServerTransporterNetty
import org.rabbitcontrol.rcp.transport.websocket.server.WebsocketServerTransporterNetty
import java.awt.Color
import java.net.InetSocketAddress
import java.net.Socket
import java.net.URI
import java.net.URISyntaxException
import kotlin.reflect.KMutableProperty1
class RabbitControlServer(private val showQRUntilClientConnects: Boolean = true, rcpPort: Int =
10000, staticFilesPort: Int = 8080, rabbithole: String = "") : Extension {
private val rabbitServer = RCPServer()
private val transporter = WebsocketServerTransporterNetty()
private var rabbitholeTransporter: RabbitHoleWsServerTransporterNetty? = null
private var webServer: NettyApplicationEngine? = null
private var parameterMap = mutableMapOf>()
private var qrCodeImage: ColorBuffer? = null
private var qrOverlayComposition: Composite? = null
/**
* Animate the opacity to make it look smooooth
*/
private var currentOpacity = 0.0
private val targetOpacity: Double
get() = if (shouldShowQR) 0.8 else 0.0
private val shouldShowQR
get() = (rabbitServer.connectionCount == 0 && showQRUntilClientConnects) || showQRCode
/**
* Used to manually show and hide the QR code and override the default
* behaviour of (only) showing the code when no clients are connected
*/
var showQRCode = false
init {
rabbitServer.addTransporter(transporter)
transporter.bind(rcpPort)
/**
* add rabbithole transporter
*/
if (rabbithole.isNotEmpty()) {
try {
val rhlTransporter = RabbitHoleWsServerTransporterNetty(URI(rabbithole))
rabbitServer.addTransporter(rhlTransporter)
rhlTransporter.bind(0)
rabbitholeTransporter = rhlTransporter
} catch (e: URISyntaxException) {
//
println("invalid URI for rabbithole: $rabbithole")
}
}
/**
* Start KTOR to serve the static files of the RabbitControl client
*/
val server = embeddedServer(Netty, port = staticFilesPort) {
routing {
static("/rabbit-client") {
resources("rabbit-client")
}
}
}
server.start()
webServer = server
/**
* Print the address
*/
val socket = Socket()
socket.connect(InetSocketAddress("google.com", 80))
val ip = socket.localAddress.toString().replace("/", "")
val clientUrlWithHash = "http://$ip:$staticFilesPort/rabbit-client/index.html#$ip:$rcpPort"
qrCodeImage = getQRCodeImage(barcodeText = clientUrlWithHash)
println("RabbitControl Web Client: $clientUrlWithHash")
/**
* Update the object when it has been updated in RabbitControl
*/
rabbitServer.setUpdateListener {
val (obj, orxParameter) = parameterMap[it]!!
when(it) {
is Int32Parameter -> {
val v = it.value
orxParameter.property.qset(obj, v)
}
is Float64Parameter -> {
val v = it.value
orxParameter.property.qset(obj, v)
}
is BooleanParameter -> {
val v = it.value
orxParameter.property.qset(obj, v)
}
is StringParameter -> {
val v = it.value
orxParameter.property.qset(obj, v)
}
is RGBAParameter -> {
val c = it.value
val cc = ColorRGBa(c.red.toDouble() / 255.0, c.green.toDouble() / 255.0, c.blue.toDouble() / 255.0, c.alpha.toDouble() / 255.0)
orxParameter.property.qset(obj, cc)
}
is Vector2Float32Parameter -> {
val v = it.value
orxParameter.property.qset(obj, Vector2(v.x.toDouble(), v.y.toDouble()))
}
is Vector3Float32Parameter -> {
val v = it.value
orxParameter.property.qset(obj, Vector3(v.x.toDouble(), v.y.toDouble(), v.z.toDouble()))
}
is Vector4Float32Parameter -> {
val v = it.value
orxParameter.property.qset(obj, Vector4(v.x.toDouble(), v.y.toDouble(), v.z.toDouble(), v.t.toDouble()))
}
}
}
}
override fun setup(program: Program) {
/**
* Creating the Composite for the overlay needs to happen in setup(),
* as we need access to [Program.drawer]
*/
qrOverlayComposition = compose {
layer {
draw {
program.drawer.isolated {
fill = ColorRGBa.WHITE.opacify(currentOpacity)
stroke = null
rectangle(0.0,0.0, width.toDouble(), height.toDouble())
}
}
layer {
blend(Darken()) {
clip = true
}
draw {
qrCodeImage?.let {
program.drawer.imageFit(it, program.width / 4.0,program.height / 4.0, program.width * .5, program.height * .5, 0.0,0.0, FitMethod.Contain)
}
}
}
}
}
}
@Suppress("UNCHECKED_CAST")
fun add(objectWithParameters: Any) {
val parameters = objectWithParameters.listParameters()
parameters.forEach {
val rabbitParam = when (it.parameterType) {
ParameterType.Int -> {
val param = rabbitServer.createInt32Parameter(it.label)
param.value = (it.property as KMutableProperty1).get(objectWithParameters)
param
}
ParameterType.Double -> {
val param = rabbitServer.createFloat64Parameter(it.label)
param.value = (it.property as KMutableProperty1).get(objectWithParameters)
param
}
ParameterType.Action -> {
val param = rabbitServer.createBangParameter(it.label)
param.setFunction {
it.function!!.call(objectWithParameters)
}
param
}
ParameterType.Boolean -> {
val param = rabbitServer.createBooleanParameter(it.label)
param.value = (it.property as KMutableProperty1).get(objectWithParameters)
param
}
ParameterType.Text -> {
val param =rabbitServer.createStringParameter(it.label)
param.value = (it.property as KMutableProperty1).get(objectWithParameters)
param
}
ParameterType.Color -> {
val param = rabbitServer.createRGBAParameter(it.label)
val c = (it.property as KMutableProperty1).get(objectWithParameters)
param.value = Color(c.r.toFloat(), c.g.toFloat(), c.b.toFloat(), c.alpha.toFloat())
param
}
ParameterType.Vector2 -> {
val param = rabbitServer.createVector2Float32Parameter(it.label)
val v2 = (it.property as KMutableProperty1).get(objectWithParameters)
param.value = org.rabbitcontrol.rcp.model.types.Vector2(v2.x.toFloat(), v2.y.toFloat())
param
}
ParameterType.Vector3 -> {
val param = rabbitServer.createVector3Float32Parameter(it.label)
val v3 = (it.property as KMutableProperty1).get(objectWithParameters)
param.value = org.rabbitcontrol.rcp.model.types.Vector3(v3.x.toFloat(), v3.y.toFloat(), v3.z.toFloat())
param
}
ParameterType.Vector4 -> {
val param = rabbitServer.createVector4Float32Parameter(it.label)
val v4 = (it.property as KMutableProperty1).get(objectWithParameters)
param.value = org.rabbitcontrol.rcp.model.types.Vector4(v4.x.toFloat(), v4.y.toFloat(), v4.z.toFloat(), v4.w.toFloat())
param
}
else -> rabbitServer.createBangParameter(it.label)
}
// We need to store a mapping from Rabbit parameter to target object + orx parameter
// so we can update the object later
parameterMap[rabbitParam] = Pair(objectWithParameters, it)
}
rabbitServer.update()
}
override var enabled = true
override fun shutdown(program: Program) {
transporter.dispose()
rabbitholeTransporter?.dispose()
webServer?.stop(0, 0)
}
private fun getQRCodeImage(barcodeText: String): ColorBuffer {
val qrCodeWriter = QRCodeWriter()
val bitMatrix = qrCodeWriter.encode(barcodeText, BarcodeFormat.QR_CODE, 30, 30)
val cb = colorBuffer(bitMatrix.width, bitMatrix.height)
cb.filterMag = MagnifyingFilter.NEAREST
val shad = cb.shadow
for (y in 0 until bitMatrix.width) {
for (x in 0 until bitMatrix.height) {
shad[x, y] = if (bitMatrix[x, y]) ColorRGBa.BLACK else ColorRGBa.WHITE
}
}
shad.upload()
return cb
}
override fun afterDraw(drawer: Drawer, program: Program) {
currentOpacity = mix(targetOpacity, currentOpacity, 0.8)
// Don't draw if it isn't necessary
if (currentOpacity > 0.0) {
qrOverlayComposition?.draw(drawer)
}
}
}
fun KMutableProperty1?.qset(obj: Any, value: T) {
@Suppress("UNCHECKED_CAST")
return (this as KMutableProperty1).set(obj, value)
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy