All Downloads are FREE. Search and download functionalities are using the official Maven repository.

commonMain.org.openrndr.Program.kt Maven / Gradle / Ivy

@file:Suppress("unused")

package org.openrndr

import org.openrndr.animatable.Animatable
import org.openrndr.color.ColorRGBa
import org.openrndr.draw.Drawer
import org.openrndr.events.Event
import org.openrndr.internal.Driver
import org.openrndr.math.Vector2
import kotlin.jvm.JvmRecord

expect fun rootClassName(): String

enum class WindowEventType {
    MOVED,
    RESIZED,
    FOCUSED,
    UNFOCUSED,
    MINIMIZED,
    RESTORED,
    CLOSED
}

/**
 * window event message
 */
@JvmRecord
data class WindowEvent(val type: WindowEventType, val position: Vector2, val size: Vector2, val focused: Boolean)

/**
 * window drop item event message
 */
@JvmRecord
data class DropEvent(val position: Vector2, val files: List)

/**
 * program event type
 */
enum class ProgramEventType {
    /**
     * indicates the program has ended
     */
    ENDED
}

/**
 * program event message
 */
@JvmRecord
data class ProgramEvent(val type: ProgramEventType)

@JvmRecord
data class RequestAssetsEvent(val origin: Any, val program: Program)

@JvmRecord
data class ProduceAssetsEvent(val origin: Any, val program: Program, val assetMetadata: AssetMetadata)

@JvmRecord
data class AssetMetadata(
    val programName: String,
    val assetBaseName: String,
    val assetProperties: Map
)

interface InputEvents {
    val mouse: MouseEvents
    val keyboard: KeyEvents
    val pointers: Pointers
}

interface Clipboard {
    var contents: String?
}

interface Clock {
    val seconds: Double
}

interface Program : InputEvents, ExtensionHost, Clock {

    /**
     * A map that can be used to store arbitrary data, including functions
     */
    var userProperties: MutableMap

    var name: String

    var width: Int
    var height: Int
    var isNested: Boolean
    var drawer: Drawer
    var driver: Driver

    val dispatcher: Dispatcher
    val window: Window
    var application: Application

    suspend fun setup()

    fun drawImpl()
    fun draw()

    val produceAssets: Event
    val requestAssets: Event
    var assetMetadata: () -> AssetMetadata
    var assetProperties: MutableMap
    var clock: () -> Double
    var ended: Event
    var backgroundColor: ColorRGBa?
    val frameCount: Int
    val clipboard: ProgramImplementation.ApplicationClipboard

    fun updateFrameSecondsFromClock()
}

interface Window {
    var title: String
    var size: Vector2
    var contentScale: Double
    var presentationMode: PresentationMode
    var multisample: WindowMultisample
    var resizable: Boolean

    fun requestFocus()

    fun requestDraw()

    /**
     * Window focused event, triggered when the window receives focus
     */
    val focused: Event

    /**
     * Window focused event, triggered when the window loses focus
     */
    val unfocused: Event

    /**
     * Window moved event
     */
    val moved: Event

    /**
     * Window sized event
     */
    val sized: Event

    /**
     * Window minimized event
     */
    val minimized: Event

    /**
     * Window restored (from minimization) event
     */
    val restored: Event

    /**
     * Window restored (from minimization) event
     */
    val closed: Event

    /**
     * Drop event, triggered when a file is dropped on the window
     */
    val drop: Event

    /**
     * Window position
     */
    var position: Vector2
}

/**
 * The Program class, this is where most user implementations start.
 */
open class ProgramImplementation(val suspend: Boolean = false) : Program {
    override var width = 0
    override var height = 0

    override val program: Program by lazy { this }
    override var userProperties: MutableMap = mutableMapOf()


    override var name = rootClassName()

    private val animator by lazy { Animatable() }

    override lateinit var drawer: Drawer
    override lateinit var driver: Driver

    override lateinit var application: Application

    /** This is checked at runtime to disallow nesting [extend] blocks. */
    override var isNested: Boolean = false

    /**
     * background color that is used to clear the background every frame
     */
    override var backgroundColor: ColorRGBa? = ColorRGBa.BLACK
    override val dispatcher = Dispatcher()

    /**
     * program ended event
     *
     * The [ended] event is emitted when the program is ended by closing the application window
     */
    override var ended = Event()


    private var firstFrameTime = Double.POSITIVE_INFINITY

    /**
     * clock function. defaults to returning the application time.
     */
    override var clock =
        { if (firstFrameTime == Double.POSITIVE_INFINITY || frameCount <= 0) 0.0 else (application.seconds - firstFrameTime) }

    override var assetProperties = mutableMapOf()
    override var assetMetadata = {
        AssetMetadata(this.name, namedTimestamp(), assetProperties)
    }

    final override val requestAssets = Event()
    final override val produceAssets = Event()

    init {
        requestAssets.listen {
            produceAssets.trigger(
                ProduceAssetsEvent(
                    it.origin, it.program,
                    assetMetadata()
                )
            )
        }
    }


    private var frameSeconds = 0.0
    private var deltaSeconds: Double = 0.0
    private var lastSeconds: Double = -1.0

    override var frameCount = 0

    /**
     * The number of [seconds] since program start, or the time from a custom [clock].
     * value is updated at the beginning of the frame only.
     */
    override val seconds: Double
        get() = frameSeconds

    /**
     * The elapsed time since the last draw loop
     */
    val deltaTime: Double
        get() = deltaSeconds

    inner class ApplicationClipboard : Clipboard {
        override var contents: String?
            get() {
                return application.clipboardContents
            }
            set(value) {
                application.clipboardContents = value
            }
    }

    override val clipboard = ApplicationClipboard()

    /**
     * list of installed extensions
     */
    override val extensions = mutableListOf()
        get() {
            if (field.isEmpty()) isNested = false
            return field
        }

    /**
     * install an [Extension]
     * @param extension the [Extension] to install
     */
    override fun  extend(extension: T): T {
        extensions.add(extension)
        extension.setup(this)
        return extension
    }

    /**
     * install an [Extension] and configure it
     * @param extension the [Extension] to install
     * @param configure a configuration function to called with [extension] as its receiver
     * @return the installed [Extension]
     */
    override fun  extend(extension: T, configure: T.() -> Unit): T {
        extensions.add(extension)
        extension.configure()
        extension.setup(this)
        return extension
    }

    /**
     * install an extension function for the given [ExtensionStage]
     */
    override fun extend(stage: ExtensionStage, userDraw: Program.() -> Unit) {
        if (isNested) error("Cannot nest extend blocks within extend blocks")
        val functionExtension = when (stage) {
            ExtensionStage.SETUP ->
                object : Extension {
                    override var enabled: Boolean = true
                    override fun setup(program: Program) {
                        program.isNested = true
                        program.userDraw()
                    }
                }

            ExtensionStage.BEFORE_DRAW ->
                object : Extension {
                    override var enabled: Boolean = true
                    override fun beforeDraw(drawer: Drawer, program: Program) {
                        program.isNested = true
                        program.userDraw()
                    }
                }

            ExtensionStage.AFTER_DRAW ->
                object : Extension {
                    override var enabled: Boolean = true
                    override fun afterDraw(drawer: Drawer, program: Program) {
                        program.isNested = true
                        program.userDraw()
                    }
                }
        }
        extensions.add(functionExtension)
    }

    /**
     * Simplified window interface
     */
    inner class Window : org.openrndr.Window {
        override var title: String
            get() = application.windowTitle
            set(value) {
                application.windowTitle = value
            }

        override var size
            get() = application.windowSize
            set(value) {
                application.windowSize = value
            }

        override var contentScale
            get() = application.windowContentScale
            set(value) {
                application.windowContentScale = value
            }

        override var presentationMode: PresentationMode
            get() = application.presentationMode
            set(value) {
                application.presentationMode = value
            }

        override var multisample: WindowMultisample
            get() {
                return application.windowMultisample
            }
            set(value) {
                application.windowMultisample = value
            }

        override var resizable: Boolean
            get() {
                return application.windowResizable
            }
            set(value) {
                application.windowResizable = value
            }

        override fun requestFocus() = application.requestFocus()

        override fun requestDraw() = application.requestDraw()

        /**
         * Window focused event, triggered when the window receives focus
         */
        override val focused = Event("window-focused", postpone = true)

        /**
         * Window focused event, triggered when the window loses focus
         */
        override val unfocused = Event("window-unfocused", postpone = true)

        /**
         * Window moved event
         */
        override val moved = Event("window-moved", postpone = true)

        /**
         * Window sized event
         */
        override val sized = Event("window-sized", postpone = true)

        /**
         * Window minimized event
         */
        override val minimized = Event("window-minimized", postpone = true)

        /**
         * Window restored (from minimization) event
         */
        override val restored = Event("window-restored", postpone = true)

        /**
         * Window restored (from minimization) event
         */
        override val closed = Event("window-closed", postpone = true)

        /**
         * Drop event, triggered when a file is dropped on the window
         */
        override val drop = Event("window-drop", postpone = true)

        /**
         * Window position
         */
        override var position: Vector2
            get() = application.windowPosition
            set(value) {
                application.windowPosition = value
            }
    }

    override val window = Window()


    override val keyboard by lazy { Keyboard() }
    override val mouse by lazy { ApplicationMouse(application = { application }) }
    override val pointers by lazy { Pointers(application = { application }) }

    /**
     * This runs exactly once before the first call to draw()
     */
    override suspend fun setup() {}

    /**
     * This is the draw call that is called by Application. It takes care of handling extensions.
     */
    override fun drawImpl() {
        if (frameCount == 0) {
            firstFrameTime = application.seconds
        }
        animator.updateAnimation()
        updateFrameSecondsFromClock()

        if (lastSeconds == -1.0) lastSeconds = seconds

        deltaSeconds = frameSeconds - lastSeconds
        lastSeconds = frameSeconds


        backgroundColor?.let {
            drawer.clear(it)
        }
        extensions.filter { it.enabled }.forEach { it.beforeDraw(drawer, this) }
        draw()
        extensions.reversed().filter { it.enabled }.forEach { it.afterDraw(drawer, this) }
        frameCount++
    }

    fun animate(animationFunction: Animatable.() -> Unit) {
        animator.animationFunction()
    }

    /**
     * This is the user facing draw call. It should be overridden by the user.
     */
    override fun draw() {}

    override fun updateFrameSecondsFromClock() {
        frameSeconds = clock()
    }
}






expect fun Program.namedTimestamp(extension: String = "", path: String? = null): String




© 2015 - 2025 Weber Informatics LLC | Privacy Policy