package inr.numass.control import hep.dataforge.kodex.KMetaBuilder import hep.dataforge.plots.PlotFrame import hep.dataforge.plots.Plottable import hep.dataforge.plots.fx.PlotContainer import hep.dataforge.plots.jfreechart.JFreeChartFrame import hep.dataforge.values.Value import javafx.beans.value.ObservableValue import javafx.event.EventTarget import javafx.geometry.Orientation import javafx.scene.Node import javafx.scene.layout.BorderPane import javafx.scene.paint.Color import javafx.scene.paint.Paint import javafx.scene.shape.Circle import javafx.scene.shape.StrokeType import org.controlsfx.control.ToggleSwitch import tornadofx.* import java.util.* /** * A pin like indicator fx node */ class Indicator(radius: Double = 10.0) : Circle(radius, Color.GRAY) { private var binding: ObservableValue<*>? = null; init { stroke = Color.BLACK; strokeType = StrokeType.INSIDE; } /** * bind this indicator color to given observable */ fun bind(observable: ObservableValue, transform: (T) -> Paint) { if (binding != null) { throw RuntimeException("Indicator already bound"); } else { binding = observable; fill = transform(observable.value) observable.addListener { _, _, value -> fill = transform(value); } } } /** * bind indicator to the boolean value using default colours */ fun bind(booleanValue: ObservableValue) { bind(booleanValue) { if (it == null) { Color.GRAY } else if (it) { Color.GREEN; } else { Color.RED; } } } fun unbind() { this.binding = null; neutralize(); } /** * return indicator to the neutral state but do not unbind */ fun neutralize() { fill = Color.GRAY; } } fun EventTarget.indicator(radius: Double = 10.0, op: (Indicator.() -> Unit)? = null) = opcr(this, Indicator(radius), op) fun Indicator.bind(connection: DeviceViewConnection<*>, state: String, transform: ((Value) -> Paint)? = null) { tooltip(state) if (transform != null) { bind(connection.getStateBinding(state), transform); } else { bind(connection.getStateBinding(state)) { if (it.isNull) { Color.GRAY } else if (it.booleanValue()) { Color.GREEN; } else { Color.RED; } } } } /** * State name + indicator */ fun EventTarget.deviceStateIndicator(connection: DeviceViewConnection<*>, state: String, showName: Boolean = true, transform: ((Value) -> Paint)? = null) { if (connection.device.hasState(state)) { if (showName) { text("${state.toUpperCase()}: ") } indicator { bind(connection, state, transform); } separator(Orientation.VERTICAL) } else { throw RuntimeException("Device does not support state $state"); } } /** * A togglebutton + indicator for boolean state */ fun Node.deviceStateToggle(connection: DeviceViewConnection<*>, state: String, title: String = state) { if (connection.device.hasState(state)) { togglebutton(title) { isSelected = false selectedProperty().addListener { observable, oldValue, newValue -> if (oldValue != newValue) { connection.device.setState(state, newValue).thenAccept { isSelected = it.booleanValue() } } } } deviceStateIndicator(connection, state, false) } else { throw RuntimeException("Device does not support state $state"); } } fun EventTarget.switch(text: String = "", op: (ToggleSwitch.() -> Unit)? = null): ToggleSwitch { val switch = ToggleSwitch(text) return opcr(this, switch, op) } /** * Add plot */ fun BorderPane.plot(plottables: Iterable = Collections.emptyList(), metaTransform: (KMetaBuilder.() -> Unit)? = null): PlotFrame { val meta = KMetaBuilder("plotFrame"); metaTransform?.invoke(meta) val plot = JFreeChartFrame(meta) plot.addAll(plottables) PlotContainer.centerIn(this).plot = plot return plot; }