Zrythm v2.0.0-alpha.1
a highly automated and intuitive digital audio workstation
Loading...
Searching...
No Matches
automation_track.h
1// SPDX-FileCopyrightText: © 2018-2022, 2024-2025 Alexandros Theodotou <alex@zrythm.org>
2// SPDX-License-Identifier: LicenseRef-ZrythmLicense
3
4#pragma once
5
6#include <utility>
7
8#include "dsp/processor_base.h"
9#include "dsp/tempo_map_qml_adapter.h"
10#include "structure/arrangement/arranger_object_owner.h"
11#include "structure/arrangement/automation_region.h"
12#include "structure/arrangement/timeline_data_provider.h"
13#include "structure/tracks/playback_cache_activity_tracker.h"
14#include "utils/playback_cache_scheduler.h"
15#include "utils/units.h"
16
17#include <QtQmlIntegration/qqmlintegration.h>
18
19namespace zrythm::structure::tracks
20{
22 : public QObject,
23 public arrangement::ArrangerObjectOwner<arrangement::AutomationRegion>
24{
25 Q_OBJECT
26 QML_ELEMENT
27 Q_PROPERTY (
28 zrythm::dsp::ProcessorParameter * parameter READ parameter CONSTANT)
29 Q_PROPERTY (
30 AutomationMode automationMode READ getAutomationMode WRITE setAutomationMode
31 NOTIFY automationModeChanged)
32 Q_PROPERTY (
33 AutomationRecordMode recordMode READ getRecordMode WRITE setRecordMode
34 NOTIFY recordModeChanged)
35 DEFINE_ARRANGER_OBJECT_OWNER_QML_PROPERTIES (
37 regions,
39 Q_PROPERTY (
41 playbackCacheActivityTracker READ playbackCacheActivityTracker CONSTANT)
42 QML_UNCREATABLE ("")
43
44public:
45 using AutomationRegion = arrangement::AutomationRegion;
46 using AutomationPoint = arrangement::AutomationPoint;
47
49 static constexpr int AUTOMATION_RECORDING_TOUCH_REL_MS = 800;
50
51 enum class AutomationMode : uint8_t
52 {
53 Read,
54 Record,
55 Off,
56 };
57 Q_ENUM (AutomationMode)
58
59 enum class AutomationRecordMode : uint8_t
60 {
61 Touch,
62 Latch,
63 };
64 Q_ENUM (AutomationRecordMode)
65
66public:
69 const dsp::TempoMapWrapper &tempo_map,
70 utils::IObjectRegistry &registry,
71 dsp::ProcessorParameterUuidReference param_id,
72 QObject * parent = nullptr);
73
74public:
75 // ========================================================================
76 // QML Interface
77 // ========================================================================
78
79 dsp::ProcessorParameter * parameter () const { return param_id_.get (); }
80
81 AutomationMode getAutomationMode () const { return automation_mode_.load (); }
82 void setAutomationMode (AutomationMode automation_mode);
83 Q_SIGNAL void automationModeChanged (AutomationMode automation_mode);
84
85 AutomationRecordMode getRecordMode () const { return record_mode_; }
86 void setRecordMode (AutomationRecordMode record_mode);
87 Q_INVOKABLE void swapRecordMode ();
88 Q_SIGNAL void recordModeChanged (AutomationRecordMode record_mode);
89
96 Q_INVOKABLE void
102 Q_SIGNAL void
104
105 // cache activity tracking
106 [[nodiscard]] PlaybackCacheActivityTracker *
107 playbackCacheActivityTracker () const
108 {
109 return playback_cache_activity_tracker_.get ();
110 }
111
112 // ========================================================================
113
117 [[gnu::hot]] bool should_read_automation () const;
118
133 [[gnu::hot]] bool should_be_recording (bool record_aps) const;
134
144 AutomationPoint * get_automation_point_around (
145 units::precise_tick_t position_ticks,
146 units::precise_tick_t delta_ticks,
147 bool search_only_backwards = false);
148
158 AutomationPoint * get_automation_point_before (
159 units::sample_t timeline_position,
160 bool search_only_regions_enclosing_position) const;
161
171 auto get_region_before (
172 units::sample_t pos_samples,
173 bool search_only_regions_enclosing_position) const -> AutomationRegion *;
174
183 std::optional<float> get_normalized_value (
184 units::sample_t timeline_frames,
185 bool search_only_regions_enclosing_position) const;
186
187 bool contains_automation () const { return !get_children_vector ().empty (); }
188
189 std::string
190 get_field_name_for_serialization (const AutomationRegion *) const override
191 {
192 return "regions";
193 }
194
195public:
196 static constexpr auto kParameterKey = "parameter"sv;
197
198private:
199 static constexpr auto kAutomationModeKey = "automationMode"sv;
200 static constexpr auto kRecordModeKey = "recordMode"sv;
201 friend void to_json (nlohmann::json &j, const AutomationTrack &track);
202 friend void from_json (const nlohmann::json &j, AutomationTrack &track);
203
204 friend void init_from (
205 AutomationTrack &obj,
206 const AutomationTrack &other,
207 utils::ObjectCloneType clone_type);
208
209private:
210 const dsp::TempoMapWrapper &tempo_map_;
211 utils::IObjectRegistry &registry_;
212
214 dsp::ProcessorParameterUuidReference param_id_;
215
217 std::atomic<AutomationMode> automation_mode_{ AutomationMode::Read };
218
220 AutomationRecordMode record_mode_{};
221
225 utils::QObjectUniquePtr<arrangement::AutomationTimelineDataProvider>
226 automation_data_provider_;
227
231 utils::QObjectUniquePtr<utils::PlaybackCacheScheduler>
232 automation_cache_request_debouncer_;
233
234 utils::QObjectUniquePtr<PlaybackCacheActivityTracker>
235 playback_cache_activity_tracker_;
236};
237
241inline void
242generate_automation_tracks_for_processor (
243 std::vector<utils::QObjectUniquePtr<AutomationTrack>> &ret,
244 const dsp::ProcessorBase &processor,
245 const dsp::TempoMapWrapper &tempo_map,
246 utils::IObjectRegistry &registry)
247{
248 z_debug ("generating automation tracks for {}...", processor.get_node_name ());
249 for (const auto &param_ref : processor.get_parameters ())
250 {
251 auto * const param = param_ref.get ();
252 if (!param->automatable ())
253 continue;
254
255 ret.emplace_back (
256 utils::make_qobject_unique<AutomationTrack> (
257 tempo_map, registry, param_ref));
258 }
259}
260}
261
262DEFINE_ENUM_FORMATTER (
263 zrythm::structure::tracks::AutomationTrack::AutomationRecordMode,
264 AutomationRecordMode,
265 QT_TR_NOOP_UTF8 ("Touch"),
266 QT_TR_NOOP_UTF8 ("Latch"));
267
268DEFINE_ENUM_FORMATTER (
269 zrythm::structure::tracks::AutomationTrack::AutomationMode,
270 AutomationMode,
271 "On",
272 "Rec",
273 "Off");
Processor parameter that accepts automation and modulation sources and integrates with QML and the DS...
Definition parameter.h:255
An automation point inside an AutomationTrack.
Represents an automation region, which contains a collection of automation points.
Q_SIGNAL void automationObjectsNeedRecache(utils::ExpandableTickRange affectedRange)
Fired when a new cache is required.
AutomationTrack(const dsp::TempoMapWrapper &tempo_map, utils::IObjectRegistry &registry, dsp::ProcessorParameterUuidReference param_id, QObject *parent=nullptr)
Creates an automation track for the given parameter.
bool should_read_automation() const
Returns whether the automation in the automation track should be read.
std::optional< float > get_normalized_value(units::sample_t timeline_frames, bool search_only_regions_enclosing_position) const
Returns the normalized parameter value at the given timeline position.
bool should_be_recording(bool record_aps) const
Returns if the automation track should currently be recording data.
AutomationPoint * get_automation_point_before(units::sample_t timeline_position, bool search_only_regions_enclosing_position) const
Returns the last automation point before the given timeline position.
AutomationPoint * get_automation_point_around(units::precise_tick_t position_ticks, units::precise_tick_t delta_ticks, bool search_only_backwards=false)
Find an automation point near the given position position_ticks with a tolerance of delta_ticks.
Q_INVOKABLE void regeneratePlaybackCaches(utils::ExpandableTickRange affectedRange)
To be connected to to notify of any changes to the automation content.
auto get_region_before(units::sample_t pos_samples, bool search_only_regions_enclosing_position) const -> AutomationRegion *
Returns the active region at a given position, if any.
Helper that manages cache activity state for a cache-holding object.
Abstract interface for a UUID-keyed object registry.