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

commonMain.dev.atsushieno.ktmidi.MidiMachine.kt Maven / Gradle / Ivy

package dev.atsushieno.ktmidi

@Deprecated("At this state there is only deprecated use, and it has old naming")
fun interface OnMidiEventListener {
    fun onEvent(e: MidiEvent)
}

@Deprecated("Use Midi1Machine instead which resolves several design issues")
class MidiMachine {
    private val event_received_handlers = arrayListOf()

    fun addOnEventReceivedListener(listener: OnMidiEventListener) {
        event_received_handlers.add(listener)
    }

    fun removeOnEventReceivedListener(listener: OnMidiEventListener) {
        event_received_handlers.remove(listener)
    }

    var channels = Array(16) { MidiMachineChannel() }

    fun processEvent(evt: MidiEvent) {
        when (evt.eventType.toUnsigned()) {
            MidiChannelStatus.NOTE_ON ->
                channels[evt.channel.toUnsigned()].noteVelocity[evt.msb.toUnsigned()] = evt.lsb

            MidiChannelStatus.NOTE_OFF ->
                channels[evt.channel.toUnsigned()].noteVelocity[evt.msb.toUnsigned()] = 0
            MidiChannelStatus.PAF ->
                channels[evt.channel.toUnsigned()].pafVelocity[evt.msb.toUnsigned()] = evt.lsb
            MidiChannelStatus.CC -> {
                // FIXME: handle RPNs and NRPNs by DTE
                when (evt.msb.toInt()) {
                    MidiCC.NRPN_MSB,
                    MidiCC.NRPN_LSB ->
                        channels[evt.channel.toUnsigned()].dteTarget = DteTarget.NRPN
                    MidiCC.RPN_MSB,
                    MidiCC.RPN_LSB ->
                        channels[evt.channel.toUnsigned()].dteTarget = DteTarget.RPN

                    MidiCC.DTE_MSB ->
                        channels[evt.channel.toUnsigned()].processDte(evt.lsb, true)
                    MidiCC.DTE_LSB ->
                        channels[evt.channel.toUnsigned()].processDte(evt.lsb, false)
                    MidiCC.DTE_INCREMENT ->
                        channels[evt.channel.toUnsigned()].processDteIncrement()
                    MidiCC.DTE_DECREMENT ->
                        channels[evt.channel.toUnsigned()].processDteDecrement()
                }
                channels[evt.channel.toUnsigned()].controls[evt.msb.toUnsigned()] = evt.lsb
            }
            MidiChannelStatus.PROGRAM ->
                channels[evt.channel.toUnsigned()].program = evt.msb
            MidiChannelStatus.CAF ->
                channels[evt.channel.toUnsigned()].caf = evt.msb
            MidiChannelStatus.PITCH_BEND ->
                channels[evt.channel.toUnsigned()].pitchbend = ((evt.msb.toUnsigned() shl 7) + evt.lsb).toShort()
        }
        for (receiver in event_received_handlers)
            receiver.onEvent(evt)
    }
}

class MidiMachineChannel {
    val noteVelocity = ByteArray(128)
    val pafVelocity = ByteArray(128)
    val controls = ByteArray(128)
    val rpns = ShortArray(128) // only 5 should be used though
    val nrpns = ShortArray(128)
    var program: Byte = 0
    var caf: Byte = 0
    var pitchbend: Short = 8192
    var dteTarget: DteTarget = DteTarget.RPN
    private var dte_target_value: Byte = 0

    val rpnTarget: Short
        get() = ((controls[MidiCC.RPN_MSB].toUnsigned() shl 7) + controls[MidiCC.RPN_LSB]).toShort()


    fun processDte(value: Byte, isMsb: Boolean) {
        var arr: ShortArray
        when (dteTarget) {
            DteTarget.RPN -> {
                dte_target_value = controls[(if (isMsb) MidiCC.RPN_MSB else MidiCC.RPN_LSB)]
                arr = rpns
            }
            DteTarget.NRPN -> {
                dte_target_value = controls[(if (isMsb) MidiCC.NRPN_MSB else MidiCC.NRPN_LSB)]
                arr = nrpns
            }
        }
        var cur = arr[dte_target_value.toUnsigned()].toInt()
        if (isMsb)
            arr[dte_target_value.toUnsigned()] = (cur and 0x007F + ((value.toUnsigned() and 0x7F) shl 7)).toShort()
        else
            arr[dte_target_value.toUnsigned()] = (cur and 0x3FF0 + (value.toUnsigned() and 0x7F)).toShort()
    }

    fun processDteIncrement() {
        when (dteTarget) {
            DteTarget.RPN -> rpns[dte_target_value.toUnsigned()]++
            DteTarget.NRPN -> nrpns[dte_target_value.toUnsigned()]++
        }
    }

    fun processDteDecrement() {
        when (dteTarget) {
            DteTarget.RPN -> rpns[dte_target_value.toUnsigned()]--
            DteTarget.NRPN -> nrpns[dte_target_value.toUnsigned()]--
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy