Source code for jdxi_editor.ui.widgets.adsr.adsr

"""
ADSR Widget for Roland JD-Xi

This widget provides address visual interface for editing ADSR (Attack, Decay, Sustain, Release)
envelope parameters. It includes:
- Interactive sliders for each ADSR parameter
- Visual envelope plot
- Real-time parameter updates
- MIDI parameter integration via SynthParameter objects

The widget supports both analog and digital synth parameters and provides visual feedback
through an animated envelope curve.
"""

from typing import Callable, Dict, Optional

from PySide6.QtCore import Signal
from PySide6.QtWidgets import QGridLayout, QWidget

from jdxi_editor.core.jdxi import JDXi
from jdxi_editor.midi.data.address.address import JDXiSysExAddress
from jdxi_editor.midi.io.helper import MidiIOHelper
from jdxi_editor.midi.sysex.composer import JDXiSysExComposer
from jdxi_editor.ui.widgets.adsr.plot import ADSRPlot
from jdxi_editor.ui.widgets.envelope.base import TOOLTIPS, EnvelopeWidgetBase
from jdxi_editor.ui.widgets.envelope.parameter import EnvelopeParameter
from jdxi_editor.ui.widgets.slider_spinbox.slider_spinbox import AdsrSliderSpinbox
from picomidi.sysex.parameter.address import AddressParameter


[docs] class ADSR(EnvelopeWidgetBase): """ADSR Widget for Roland JD-Xi"""
[docs] envelope_changed = Signal(dict)
def __init__( self, attack_param: AddressParameter, decay_param: AddressParameter, sustain_param: AddressParameter, release_param: AddressParameter, initial_param: Optional[AddressParameter] = None, peak_param: Optional[AddressParameter] = None, create_parameter_slider: Callable = None, midi_helper: Optional[MidiIOHelper] = None, address: Optional[JDXiSysExAddress] = None, controls: Dict[AddressParameter, QWidget] = None, parent: Optional[QWidget] = None, analog: bool = False, ): super().__init__( envelope_keys=[ EnvelopeParameter.ATTACK_TIME, EnvelopeParameter.DECAY_TIME, EnvelopeParameter.SUSTAIN_LEVEL, EnvelopeParameter.RELEASE_TIME, ], create_parameter_slider=create_parameter_slider, parameters=[attack_param, decay_param, sustain_param, release_param], midi_helper=midi_helper, address=address, controls=controls, parent=parent, )
[docs] self.sysex_composer = JDXiSysExComposer()
""" Initialize the ADSR widget :param attack_param: AddressParameter :param decay_param: AddressParameter :param sustain_param: AddressParameter :param release_param: AddressParameter :param initial_param: Optional[AddressParameter] :param peak_param: Optional[AddressParameter] :param midi_helper: Optional[MidiIOHelper] :param address: Optional[RolandSysExAddress] :param parent: Optional[QWidget] """
[docs] self.address = address
[docs] self.midi_helper = midi_helper
if controls is not None: self.controls = controls else: self.controls = {}
[docs] self._create_parameter_slider = create_parameter_slider
[docs] self.envelope = { EnvelopeParameter.ATTACK_TIME: 300.0, EnvelopeParameter.DECAY_TIME: 800.0, EnvelopeParameter.RELEASE_TIME: 500.0, EnvelopeParameter.INITIAL_LEVEL: 0.0, EnvelopeParameter.PEAK_LEVEL: 0.50, EnvelopeParameter.SUSTAIN_LEVEL: 0.8, }
[docs] self.attack_control = AdsrSliderSpinbox( attack_param, min_value=0, max_value=1000, units=" ms", label="Attack", value=self.envelope[EnvelopeParameter.ATTACK_TIME], create_parameter_slider=self._create_parameter_slider, parent=self, )
[docs] self.decay_control = AdsrSliderSpinbox( decay_param, min_value=0, max_value=1000, units=" ms", label="Decay", value=self.envelope[EnvelopeParameter.DECAY_TIME], create_parameter_slider=self._create_parameter_slider, parent=self, )
[docs] self.sustain_control = AdsrSliderSpinbox( sustain_param, min_value=0.0, max_value=1.0, units="", label="Sustain", value=self.envelope[EnvelopeParameter.SUSTAIN_LEVEL], create_parameter_slider=self._create_parameter_slider, parent=self, )
[docs] self.release_control = AdsrSliderSpinbox( release_param, min_value=0, max_value=1000, units=" ms", label="Release", value=self.envelope[EnvelopeParameter.RELEASE_TIME], create_parameter_slider=self._create_parameter_slider, parent=self, )
[docs] self._control_widgets = [ self.attack_control, self.decay_control, self.sustain_control, self.release_control, ]
self.controls[attack_param] = self.attack_control self.controls[decay_param] = self.decay_control self.controls[sustain_param] = self.sustain_control self.controls[release_param] = self.release_control if peak_param: # Always create a new AdsrSliderSpinbox for the ADSR widget # The regular slider from SLIDER_GROUPS will remain in the Controls tab # Both widgets will control the same parameter self.peak_control = AdsrSliderSpinbox( peak_param, min_value=0, max_value=1.0, units="", label="Depth", value=self.envelope[EnvelopeParameter.PEAK_LEVEL], create_parameter_slider=self._create_parameter_slider, parent=self, ) self._control_widgets.append(self.peak_control) # Only store in controls if it doesn't already exist (to avoid overwriting SLIDER_GROUPS slider) if peak_param not in self.controls: self.controls[peak_param] = self.peak_control for key, widget in [ (EnvelopeParameter.ATTACK_TIME, self.attack_control), (EnvelopeParameter.DECAY_TIME, self.decay_control), (EnvelopeParameter.SUSTAIN_LEVEL, self.sustain_control), (EnvelopeParameter.RELEASE_TIME, self.release_control), ]: if tooltip := TOOLTIPS.get(key): widget.setToolTip(tooltip)
[docs] self.attack_parameter = attack_param
[docs] self.decay_parameter = decay_param
[docs] self.sustain_parameter = sustain_param
[docs] self.release_parameter = release_param
[docs] self._control_parameters = [ self.attack_parameter, self.decay_parameter, self.sustain_parameter, self.release_parameter, ]
if peak_param: self._control_parameters.append(peak_param)
[docs] self.layout = QGridLayout()
self.layout.setColumnStretch(0, 1) self.layout.addWidget(self.attack_control, 0, 1) self.layout.addWidget(self.decay_control, 0, 2) self.layout.addWidget(self.sustain_control, 0, 3) self.layout.addWidget(self.release_control, 0, 4) self.setLayout(self.layout)
[docs] self.envelope_spinbox_map = { EnvelopeParameter.ATTACK_TIME: self.attack_control.spinbox, EnvelopeParameter.DECAY_TIME: self.decay_control.spinbox, EnvelopeParameter.SUSTAIN_LEVEL: self.sustain_control.spinbox, EnvelopeParameter.RELEASE_TIME: self.release_control.spinbox, }
# Create layout
[docs] self.plot = ADSRPlot( width=JDXi.UI.Style.ADSR_PLOT_WIDTH, height=JDXi.UI.Style.ADSR_PLOT_HEIGHT, envelope=self.envelope, parent=self, )
if hasattr(self, "peak_control"): self.layout.addWidget(self.peak_control, 0, 5) self.envelope_spinbox_map[EnvelopeParameter.PEAK_LEVEL] = ( self.peak_control.spinbox ) self.layout.addWidget(self.plot, 0, 6, 3, 1) self.layout.setColumnStretch(7, 1) else: self.layout.addWidget(self.plot, 0, 5, 3, 1) self.layout.setColumnStretch(6, 1) self.plot.set_values(self.envelope) for control in self._control_widgets: control.envelope_changed.connect(self.on_control_changed) self.update_controls_from_envelope() JDXi.UI.Theme.apply_adsr_style(self, analog=analog)
[docs] def on_control_changed(self, change: dict): self.envelope.update(change) self.plot.set_values(self.envelope) self.envelope_changed.emit(self.envelope)
[docs] def update_envelope_from_spinboxes(self): """Update envelope values from spin boxes""" self.envelope[EnvelopeParameter.ATTACK_TIME] = self.attack_control.value() self.envelope[EnvelopeParameter.DECAY_TIME] = self.decay_control.value() self.envelope[EnvelopeParameter.SUSTAIN_LEVEL] = self.sustain_control.value() self.envelope[EnvelopeParameter.RELEASE_TIME] = self.release_control.value() self.plot.set_values(self.envelope) self.envelope_changed.emit(self.envelope)
[docs] def update_spinboxes_from_envelope(self): """Update spinboxes from envelope values""" self.attack_control.setValue(self.envelope[EnvelopeParameter.ATTACK_TIME]) self.decay_control.setValue(self.envelope[EnvelopeParameter.DECAY_TIME]) self.sustain_control.setValue(self.envelope[EnvelopeParameter.SUSTAIN_LEVEL]) self.release_control.setValue(self.envelope[EnvelopeParameter.RELEASE_TIME]) self.plot.set_values(self.envelope) self.envelope_changed.emit(self.envelope)
[docs] def refresh_plot_from_controls(self) -> None: """ Sync envelope from current control values and redraw the plot without emitting. Call this after programmatically setting control values (e.g. from incoming SysEx) when blockSignals(True) was used, so the plot reflects the new values. """ self.envelope[EnvelopeParameter.ATTACK_TIME] = self.attack_control.value() self.envelope[EnvelopeParameter.DECAY_TIME] = self.decay_control.value() self.envelope[EnvelopeParameter.SUSTAIN_LEVEL] = self.sustain_control.value() self.envelope[EnvelopeParameter.RELEASE_TIME] = self.release_control.value() if hasattr(self, "peak_control"): self.envelope[EnvelopeParameter.PEAK_LEVEL] = self.peak_control.value() self.plot.set_values(self.envelope)