The JUCE cross-platform C++ framework, with DISTRHO/KXStudio specific changes
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

2303 lines
86KB

  1. /*
  2. ==============================================================================
  3. This file is part of the JUCE examples.
  4. Copyright (c) 2022 - Raw Material Software Limited
  5. The code included in this file is provided under the terms of the ISC license
  6. http://www.isc.org/downloads/software-support-policy/isc-license. Permission
  7. To use, copy, modify, and/or distribute this software for any purpose with or
  8. without fee is hereby granted provided that the above copyright notice and
  9. this permission notice appear in all copies.
  10. THE SOFTWARE IS PROVIDED "AS IS" WITHOUT ANY WARRANTY, AND ALL WARRANTIES,
  11. WHETHER EXPRESSED OR IMPLIED, INCLUDING MERCHANTABILITY AND FITNESS FOR
  12. PURPOSE, ARE DISCLAIMED.
  13. ==============================================================================
  14. */
  15. /*******************************************************************************
  16. The block below describes the properties of this PIP. A PIP is a short snippet
  17. of code that can be read by the Projucer and used to generate a JUCE project.
  18. BEGIN_JUCE_PIP_METADATA
  19. name: ARAPluginDemo
  20. version: 1.0.0
  21. vendor: JUCE
  22. website: http://juce.com
  23. description: Audio plugin using the ARA API.
  24. dependencies: juce_audio_basics, juce_audio_devices, juce_audio_formats,
  25. juce_audio_plugin_client, juce_audio_processors,
  26. juce_audio_utils, juce_core, juce_data_structures,
  27. juce_events, juce_graphics, juce_gui_basics, juce_gui_extra
  28. exporters: xcode_mac, vs2022
  29. moduleFlags: JUCE_STRICT_REFCOUNTEDPOINTER=1
  30. type: AudioProcessor
  31. mainClass: ARADemoPluginAudioProcessor
  32. documentControllerClass: ARADemoPluginDocumentControllerSpecialisation
  33. useLocalCopy: 1
  34. END_JUCE_PIP_METADATA
  35. *******************************************************************************/
  36. #pragma once
  37. #include <ARA_Library/Utilities/ARAPitchInterpretation.h>
  38. #include <ARA_Library/Utilities/ARATimelineConversion.h>
  39. //==============================================================================
  40. class ARADemoPluginAudioModification : public ARAAudioModification
  41. {
  42. public:
  43. ARADemoPluginAudioModification (ARAAudioSource* audioSource,
  44. ARA::ARAAudioModificationHostRef hostRef,
  45. const ARAAudioModification* optionalModificationToClone)
  46. : ARAAudioModification (audioSource, hostRef, optionalModificationToClone)
  47. {
  48. if (optionalModificationToClone != nullptr)
  49. dimmed = static_cast<const ARADemoPluginAudioModification*> (optionalModificationToClone)->dimmed;
  50. }
  51. bool isDimmed() const { return dimmed; }
  52. void setDimmed (bool shouldDim) { dimmed = shouldDim; }
  53. private:
  54. bool dimmed = false;
  55. };
  56. //==============================================================================
  57. struct PreviewState
  58. {
  59. std::atomic<double> previewTime { 0.0 };
  60. std::atomic<ARAPlaybackRegion*> previewedRegion { nullptr };
  61. };
  62. class SharedTimeSliceThread : public TimeSliceThread
  63. {
  64. public:
  65. SharedTimeSliceThread()
  66. : TimeSliceThread (String (JucePlugin_Name) + " ARA Sample Reading Thread")
  67. {
  68. startThread (Priority::high); // Above default priority so playback is fluent, but below realtime
  69. }
  70. };
  71. class AsyncConfigurationCallback : private AsyncUpdater
  72. {
  73. public:
  74. explicit AsyncConfigurationCallback (std::function<void()> callbackIn)
  75. : callback (std::move (callbackIn)) {}
  76. ~AsyncConfigurationCallback() override { cancelPendingUpdate(); }
  77. template <typename RequiresLock>
  78. auto withLock (RequiresLock&& fn)
  79. {
  80. const SpinLock::ScopedTryLockType scope (processingFlag);
  81. return fn (scope.isLocked());
  82. }
  83. void startConfigure() { triggerAsyncUpdate(); }
  84. private:
  85. void handleAsyncUpdate() override
  86. {
  87. const SpinLock::ScopedLockType scope (processingFlag);
  88. callback();
  89. }
  90. std::function<void()> callback;
  91. SpinLock processingFlag;
  92. };
  93. static void crossfade (const float* sourceA,
  94. const float* sourceB,
  95. float aProportionAtStart,
  96. float aProportionAtFinish,
  97. float* destinationBuffer,
  98. int numSamples)
  99. {
  100. AudioBuffer<float> destination { &destinationBuffer, 1, numSamples };
  101. destination.copyFromWithRamp (0, 0, sourceA, numSamples, aProportionAtStart, aProportionAtFinish);
  102. destination.addFromWithRamp (0, 0, sourceB, numSamples, 1.0f - aProportionAtStart, 1.0f - aProportionAtFinish);
  103. }
  104. class Looper
  105. {
  106. public:
  107. Looper() : inputBuffer (nullptr), pos (loopRange.getStart())
  108. {
  109. }
  110. Looper (const AudioBuffer<float>* buffer, Range<int64> range)
  111. : inputBuffer (buffer), loopRange (range), pos (range.getStart())
  112. {
  113. }
  114. void writeInto (AudioBuffer<float>& buffer)
  115. {
  116. if (loopRange.getLength() == 0)
  117. {
  118. buffer.clear();
  119. return;
  120. }
  121. const auto numChannelsToCopy = std::min (inputBuffer->getNumChannels(), buffer.getNumChannels());
  122. const auto actualCrossfadeLengthSamples = std::min (loopRange.getLength() / 2, (int64) desiredCrossfadeLengthSamples);
  123. for (auto samplesCopied = 0; samplesCopied < buffer.getNumSamples();)
  124. {
  125. const auto [needsCrossfade, samplePosOfNextCrossfadeTransition] = [&]() -> std::pair<bool, int64>
  126. {
  127. if (const auto endOfFadeIn = loopRange.getStart() + actualCrossfadeLengthSamples; pos < endOfFadeIn)
  128. return { true, endOfFadeIn };
  129. return { false, loopRange.getEnd() - actualCrossfadeLengthSamples };
  130. }();
  131. const auto samplesToNextCrossfadeTransition = samplePosOfNextCrossfadeTransition - pos;
  132. const auto numSamplesToCopy = std::min (buffer.getNumSamples() - samplesCopied,
  133. (int) samplesToNextCrossfadeTransition);
  134. const auto getFadeInGainAtPos = [this, actualCrossfadeLengthSamples] (auto p)
  135. {
  136. return jmap ((float) p, (float) loopRange.getStart(), (float) loopRange.getStart() + (float) actualCrossfadeLengthSamples - 1.0f, 0.0f, 1.0f);
  137. };
  138. for (int i = 0; i < numChannelsToCopy; ++i)
  139. {
  140. if (needsCrossfade)
  141. {
  142. const auto overlapStart = loopRange.getEnd() - actualCrossfadeLengthSamples
  143. + (pos - loopRange.getStart());
  144. crossfade (inputBuffer->getReadPointer (i, (int) pos),
  145. inputBuffer->getReadPointer (i, (int) overlapStart),
  146. getFadeInGainAtPos (pos),
  147. getFadeInGainAtPos (pos + numSamplesToCopy),
  148. buffer.getWritePointer (i, samplesCopied),
  149. numSamplesToCopy);
  150. }
  151. else
  152. {
  153. buffer.copyFrom (i, samplesCopied, *inputBuffer, i, (int) pos, numSamplesToCopy);
  154. }
  155. }
  156. samplesCopied += numSamplesToCopy;
  157. pos += numSamplesToCopy;
  158. jassert (pos <= loopRange.getEnd() - actualCrossfadeLengthSamples);
  159. if (pos == loopRange.getEnd() - actualCrossfadeLengthSamples)
  160. pos = loopRange.getStart();
  161. }
  162. }
  163. private:
  164. static constexpr int desiredCrossfadeLengthSamples = 50;
  165. const AudioBuffer<float>* inputBuffer;
  166. Range<int64> loopRange;
  167. int64 pos;
  168. };
  169. //==============================================================================
  170. // Returns the modified sample range in the output buffer.
  171. inline std::optional<Range<int64>> readPlaybackRangeIntoBuffer (Range<double> playbackRange,
  172. const ARAPlaybackRegion* playbackRegion,
  173. AudioBuffer<float>& buffer,
  174. const std::function<AudioFormatReader* (ARAAudioSource*)>& getReader)
  175. {
  176. const auto rangeInAudioModificationTime = playbackRange - playbackRegion->getStartInPlaybackTime()
  177. + playbackRegion->getStartInAudioModificationTime();
  178. const auto audioModification = playbackRegion->getAudioModification<ARADemoPluginAudioModification>();
  179. const auto audioSource = audioModification->getAudioSource();
  180. const auto audioModificationSampleRate = audioSource->getSampleRate();
  181. const Range<int64_t> sampleRangeInAudioModification {
  182. ARA::roundSamplePosition (rangeInAudioModificationTime.getStart() * audioModificationSampleRate),
  183. ARA::roundSamplePosition (rangeInAudioModificationTime.getEnd() * audioModificationSampleRate) - 1
  184. };
  185. const auto inputOffset = jlimit ((int64_t) 0, audioSource->getSampleCount(), sampleRangeInAudioModification.getStart());
  186. // With the output offset it can always be said of the output buffer, that the zeroth element
  187. // corresponds to beginning of the playbackRange.
  188. const auto outputOffset = std::max (-sampleRangeInAudioModification.getStart(), (int64_t) 0);
  189. /* TODO: Handle different AudioSource and playback sample rates.
  190. The conversion should be done inside a specialized AudioFormatReader so that we could use
  191. playbackSampleRate everywhere in this function and we could still read `readLength` number of samples
  192. from the source.
  193. The current implementation will be incorrect when sampling rates differ.
  194. */
  195. const auto readLength = [&]
  196. {
  197. const auto sourceReadLength =
  198. std::min (sampleRangeInAudioModification.getEnd(), audioSource->getSampleCount()) - inputOffset;
  199. const auto outputReadLength =
  200. std::min (outputOffset + sourceReadLength, (int64_t) buffer.getNumSamples()) - outputOffset;
  201. return std::min (sourceReadLength, outputReadLength);
  202. }();
  203. if (readLength == 0)
  204. return Range<int64>();
  205. auto* reader = getReader (audioSource);
  206. if (reader != nullptr && reader->read (&buffer, (int) outputOffset, (int) readLength, inputOffset, true, true))
  207. {
  208. if (audioModification->isDimmed())
  209. buffer.applyGain ((int) outputOffset, (int) readLength, 0.25f);
  210. return Range<int64>::withStartAndLength (outputOffset, readLength);
  211. }
  212. return {};
  213. }
  214. class PossiblyBufferedReader
  215. {
  216. public:
  217. PossiblyBufferedReader() = default;
  218. explicit PossiblyBufferedReader (std::unique_ptr<BufferingAudioReader> readerIn)
  219. : setTimeoutFn ([ptr = readerIn.get()] (int ms) { ptr->setReadTimeout (ms); }),
  220. reader (std::move (readerIn))
  221. {}
  222. explicit PossiblyBufferedReader (std::unique_ptr<AudioFormatReader> readerIn)
  223. : setTimeoutFn(),
  224. reader (std::move (readerIn))
  225. {}
  226. void setReadTimeout (int ms)
  227. {
  228. NullCheckedInvocation::invoke (setTimeoutFn, ms);
  229. }
  230. AudioFormatReader* get() const { return reader.get(); }
  231. private:
  232. std::function<void (int)> setTimeoutFn;
  233. std::unique_ptr<AudioFormatReader> reader;
  234. };
  235. //==============================================================================
  236. class PlaybackRenderer : public ARAPlaybackRenderer
  237. {
  238. public:
  239. using ARAPlaybackRenderer::ARAPlaybackRenderer;
  240. void prepareToPlay (double sampleRateIn,
  241. int maximumSamplesPerBlockIn,
  242. int numChannelsIn,
  243. AudioProcessor::ProcessingPrecision,
  244. AlwaysNonRealtime alwaysNonRealtime) override
  245. {
  246. numChannels = numChannelsIn;
  247. sampleRate = sampleRateIn;
  248. maximumSamplesPerBlock = maximumSamplesPerBlockIn;
  249. tempBuffer.reset (new AudioBuffer<float> (numChannels, maximumSamplesPerBlock));
  250. useBufferedAudioSourceReader = alwaysNonRealtime == AlwaysNonRealtime::no;
  251. for (const auto playbackRegion : getPlaybackRegions())
  252. {
  253. auto audioSource = playbackRegion->getAudioModification()->getAudioSource();
  254. if (audioSourceReaders.find (audioSource) == audioSourceReaders.end())
  255. {
  256. auto reader = std::make_unique<ARAAudioSourceReader> (audioSource);
  257. if (! useBufferedAudioSourceReader)
  258. {
  259. audioSourceReaders.emplace (audioSource,
  260. PossiblyBufferedReader { std::move (reader) });
  261. }
  262. else
  263. {
  264. const auto readAheadSize = jmax (4 * maximumSamplesPerBlock,
  265. roundToInt (2.0 * sampleRate));
  266. audioSourceReaders.emplace (audioSource,
  267. PossiblyBufferedReader { std::make_unique<BufferingAudioReader> (reader.release(),
  268. *sharedTimesliceThread,
  269. readAheadSize) });
  270. }
  271. }
  272. }
  273. }
  274. void releaseResources() override
  275. {
  276. audioSourceReaders.clear();
  277. tempBuffer.reset();
  278. }
  279. bool processBlock (AudioBuffer<float>& buffer,
  280. AudioProcessor::Realtime realtime,
  281. const AudioPlayHead::PositionInfo& positionInfo) noexcept override
  282. {
  283. const auto numSamples = buffer.getNumSamples();
  284. jassert (numSamples <= maximumSamplesPerBlock);
  285. jassert (numChannels == buffer.getNumChannels());
  286. jassert (realtime == AudioProcessor::Realtime::no || useBufferedAudioSourceReader);
  287. const auto timeInSamples = positionInfo.getTimeInSamples().orFallback (0);
  288. const auto isPlaying = positionInfo.getIsPlaying();
  289. bool success = true;
  290. bool didRenderAnyRegion = false;
  291. if (isPlaying)
  292. {
  293. const auto blockRange = Range<int64>::withStartAndLength (timeInSamples, numSamples);
  294. for (const auto& playbackRegion : getPlaybackRegions())
  295. {
  296. // Evaluate region borders in song time, calculate sample range to render in song time.
  297. // Note that this example does not use head- or tailtime, so the includeHeadAndTail
  298. // parameter is set to false here - this might need to be adjusted in actual plug-ins.
  299. const auto playbackSampleRange = playbackRegion->getSampleRange (sampleRate, ARAPlaybackRegion::IncludeHeadAndTail::no);
  300. auto renderRange = blockRange.getIntersectionWith (playbackSampleRange);
  301. if (renderRange.isEmpty())
  302. continue;
  303. // Evaluate region borders in modification/source time and calculate offset between
  304. // song and source samples, then clip song samples accordingly
  305. // (if an actual plug-in supports time stretching, this must be taken into account here).
  306. Range<int64> modificationSampleRange { playbackRegion->getStartInAudioModificationSamples(),
  307. playbackRegion->getEndInAudioModificationSamples() };
  308. const auto modificationSampleOffset = modificationSampleRange.getStart() - playbackSampleRange.getStart();
  309. renderRange = renderRange.getIntersectionWith (modificationSampleRange.movedToStartAt (playbackSampleRange.getStart()));
  310. if (renderRange.isEmpty())
  311. continue;
  312. // Get the audio source for the region and find the reader for that source.
  313. // This simplified example code only produces audio if sample rate and channel count match -
  314. // a robust plug-in would need to do conversion, see ARA SDK documentation.
  315. const auto audioSource = playbackRegion->getAudioModification()->getAudioSource();
  316. const auto readerIt = audioSourceReaders.find (audioSource);
  317. if (std::make_tuple (audioSource->getChannelCount(), audioSource->getSampleRate()) != std::make_tuple (numChannels, sampleRate)
  318. || (readerIt == audioSourceReaders.end()))
  319. {
  320. success = false;
  321. continue;
  322. }
  323. auto& reader = readerIt->second;
  324. reader.setReadTimeout (realtime == AudioProcessor::Realtime::no ? 100 : 0);
  325. // Calculate buffer offsets.
  326. const int numSamplesToRead = (int) renderRange.getLength();
  327. const int startInBuffer = (int) (renderRange.getStart() - blockRange.getStart());
  328. auto startInSource = renderRange.getStart() + modificationSampleOffset;
  329. // Read samples:
  330. // first region can write directly into output, later regions need to use local buffer.
  331. auto& readBuffer = (didRenderAnyRegion) ? *tempBuffer : buffer;
  332. if (! reader.get()->read (&readBuffer, startInBuffer, numSamplesToRead, startInSource, true, true))
  333. {
  334. success = false;
  335. continue;
  336. }
  337. // Apply dim if enabled
  338. if (playbackRegion->getAudioModification<ARADemoPluginAudioModification>()->isDimmed())
  339. readBuffer.applyGain (startInBuffer, numSamplesToRead, 0.25f); // dim by about 12 dB
  340. // Mix output of all regions
  341. if (didRenderAnyRegion)
  342. {
  343. // Mix local buffer into the output buffer.
  344. for (int c = 0; c < numChannels; ++c)
  345. buffer.addFrom (c, startInBuffer, *tempBuffer, c, startInBuffer, numSamplesToRead);
  346. }
  347. else
  348. {
  349. // Clear any excess at start or end of the region.
  350. if (startInBuffer != 0)
  351. buffer.clear (0, startInBuffer);
  352. const int endInBuffer = startInBuffer + numSamplesToRead;
  353. const int remainingSamples = numSamples - endInBuffer;
  354. if (remainingSamples != 0)
  355. buffer.clear (endInBuffer, remainingSamples);
  356. didRenderAnyRegion = true;
  357. }
  358. }
  359. }
  360. // If no playback or no region did intersect, clear buffer now.
  361. if (! didRenderAnyRegion)
  362. buffer.clear();
  363. return success;
  364. }
  365. using ARAPlaybackRenderer::processBlock;
  366. private:
  367. //==============================================================================
  368. // We're subclassing here only to provide a proper default c'tor for our shared resource
  369. SharedResourcePointer<SharedTimeSliceThread> sharedTimesliceThread;
  370. std::map<ARAAudioSource*, PossiblyBufferedReader> audioSourceReaders;
  371. bool useBufferedAudioSourceReader = true;
  372. int numChannels = 2;
  373. double sampleRate = 48000.0;
  374. int maximumSamplesPerBlock = 128;
  375. std::unique_ptr<AudioBuffer<float>> tempBuffer;
  376. };
  377. class EditorRenderer : public ARAEditorRenderer,
  378. private ARARegionSequence::Listener
  379. {
  380. public:
  381. EditorRenderer (ARA::PlugIn::DocumentController* documentController, const PreviewState* previewStateIn)
  382. : ARAEditorRenderer (documentController), previewState (previewStateIn), previewBuffer()
  383. {
  384. jassert (previewState != nullptr);
  385. }
  386. ~EditorRenderer() override
  387. {
  388. for (const auto& rs : regionSequences)
  389. rs->removeListener (this);
  390. }
  391. void didAddPlaybackRegionToRegionSequence (ARARegionSequence*, ARAPlaybackRegion*) override
  392. {
  393. asyncConfigCallback.startConfigure();
  394. }
  395. void didAddRegionSequence (ARA::PlugIn::RegionSequence* rs) noexcept override
  396. {
  397. auto* sequence = static_cast<ARARegionSequence*> (rs);
  398. sequence->addListener (this);
  399. regionSequences.insert (sequence);
  400. asyncConfigCallback.startConfigure();
  401. }
  402. void didAddPlaybackRegion (ARA::PlugIn::PlaybackRegion*) noexcept override
  403. {
  404. asyncConfigCallback.startConfigure();
  405. }
  406. /* An ARA host could be using either the `addPlaybackRegion()` or `addRegionSequence()` interface
  407. so we need to check the other side of both.
  408. The callback must have a signature of `bool (ARAPlaybackRegion*)`
  409. */
  410. template <typename Callback>
  411. void forEachPlaybackRegion (Callback&& cb)
  412. {
  413. for (const auto& playbackRegion : getPlaybackRegions())
  414. if (! cb (playbackRegion))
  415. return;
  416. for (const auto& regionSequence : getRegionSequences())
  417. for (const auto& playbackRegion : regionSequence->getPlaybackRegions())
  418. if (! cb (playbackRegion))
  419. return;
  420. }
  421. void prepareToPlay (double sampleRateIn,
  422. int maximumExpectedSamplesPerBlock,
  423. int numChannels,
  424. AudioProcessor::ProcessingPrecision,
  425. AlwaysNonRealtime alwaysNonRealtime) override
  426. {
  427. sampleRate = sampleRateIn;
  428. previewBuffer = std::make_unique<AudioBuffer<float>> (numChannels, (int) (2 * sampleRateIn));
  429. ignoreUnused (maximumExpectedSamplesPerBlock, alwaysNonRealtime);
  430. }
  431. void releaseResources() override
  432. {
  433. audioSourceReaders.clear();
  434. }
  435. void reset() override
  436. {
  437. previewBuffer->clear();
  438. }
  439. bool processBlock (AudioBuffer<float>& buffer,
  440. AudioProcessor::Realtime realtime,
  441. const AudioPlayHead::PositionInfo& positionInfo) noexcept override
  442. {
  443. ignoreUnused (realtime);
  444. return asyncConfigCallback.withLock ([&] (bool locked)
  445. {
  446. if (! locked)
  447. return true;
  448. const auto fadeOutIfNecessary = [this, &buffer]
  449. {
  450. if (std::exchange (wasPreviewing, false))
  451. {
  452. previewLooper.writeInto (buffer);
  453. const auto fadeOutStart = std::max (0, buffer.getNumSamples() - 50);
  454. buffer.applyGainRamp (fadeOutStart, buffer.getNumSamples() - fadeOutStart, 1.0f, 0.0f);
  455. }
  456. };
  457. if (positionInfo.getIsPlaying())
  458. {
  459. fadeOutIfNecessary();
  460. return true;
  461. }
  462. if (const auto previewedRegion = previewState->previewedRegion.load())
  463. {
  464. const auto regionIsAssignedToEditor = [&]()
  465. {
  466. bool regionIsAssigned = false;
  467. forEachPlaybackRegion ([&previewedRegion, &regionIsAssigned] (const auto& region)
  468. {
  469. if (region == previewedRegion)
  470. {
  471. regionIsAssigned = true;
  472. return false;
  473. }
  474. return true;
  475. });
  476. return regionIsAssigned;
  477. }();
  478. if (regionIsAssignedToEditor)
  479. {
  480. const auto previewTime = previewState->previewTime.load();
  481. const auto previewDimmed = previewedRegion->getAudioModification<ARADemoPluginAudioModification>()
  482. ->isDimmed();
  483. if (lastPreviewTime != previewTime
  484. || lastPlaybackRegion != previewedRegion
  485. || lastPreviewDimmed != previewDimmed)
  486. {
  487. Range<double> previewRangeInPlaybackTime { previewTime - 0.25, previewTime + 0.25 };
  488. previewBuffer->clear();
  489. const auto rangeInOutput = readPlaybackRangeIntoBuffer (previewRangeInPlaybackTime,
  490. previewedRegion,
  491. *previewBuffer,
  492. [this] (auto* source) -> auto*
  493. {
  494. const auto iter = audioSourceReaders.find (source);
  495. return iter != audioSourceReaders.end() ? iter->second.get() : nullptr;
  496. });
  497. if (rangeInOutput)
  498. {
  499. lastPreviewTime = previewTime;
  500. lastPlaybackRegion = previewedRegion;
  501. lastPreviewDimmed = previewDimmed;
  502. previewLooper = Looper (previewBuffer.get(), *rangeInOutput);
  503. }
  504. }
  505. else
  506. {
  507. previewLooper.writeInto (buffer);
  508. if (! std::exchange (wasPreviewing, true))
  509. {
  510. const auto fadeInLength = std::min (50, buffer.getNumSamples());
  511. buffer.applyGainRamp (0, fadeInLength, 0.0f, 1.0f);
  512. }
  513. }
  514. }
  515. }
  516. else
  517. {
  518. fadeOutIfNecessary();
  519. }
  520. return true;
  521. });
  522. }
  523. using ARAEditorRenderer::processBlock;
  524. private:
  525. void configure()
  526. {
  527. forEachPlaybackRegion ([this, maximumExpectedSamplesPerBlock = 1000] (const auto& playbackRegion)
  528. {
  529. const auto audioSource = playbackRegion->getAudioModification()->getAudioSource();
  530. if (audioSourceReaders.find (audioSource) == audioSourceReaders.end())
  531. {
  532. audioSourceReaders[audioSource] = std::make_unique<BufferingAudioReader> (
  533. new ARAAudioSourceReader (playbackRegion->getAudioModification()->getAudioSource()),
  534. *timeSliceThread,
  535. std::max (4 * maximumExpectedSamplesPerBlock, (int) sampleRate));
  536. }
  537. return true;
  538. });
  539. }
  540. const PreviewState* previewState = nullptr;
  541. AsyncConfigurationCallback asyncConfigCallback { [this] { configure(); } };
  542. double lastPreviewTime = 0.0;
  543. ARAPlaybackRegion* lastPlaybackRegion = nullptr;
  544. bool lastPreviewDimmed = false;
  545. bool wasPreviewing = false;
  546. std::unique_ptr<AudioBuffer<float>> previewBuffer;
  547. Looper previewLooper;
  548. double sampleRate = 48000.0;
  549. SharedResourcePointer<SharedTimeSliceThread> timeSliceThread;
  550. std::map<ARAAudioSource*, std::unique_ptr<BufferingAudioReader>> audioSourceReaders;
  551. std::set<ARARegionSequence*> regionSequences;
  552. };
  553. //==============================================================================
  554. class ARADemoPluginDocumentControllerSpecialisation : public ARADocumentControllerSpecialisation
  555. {
  556. public:
  557. using ARADocumentControllerSpecialisation::ARADocumentControllerSpecialisation;
  558. PreviewState previewState;
  559. protected:
  560. ARAAudioModification* doCreateAudioModification (ARAAudioSource* audioSource,
  561. ARA::ARAAudioModificationHostRef hostRef,
  562. const ARAAudioModification* optionalModificationToClone) noexcept override
  563. {
  564. return new ARADemoPluginAudioModification (audioSource,
  565. hostRef,
  566. static_cast<const ARADemoPluginAudioModification*> (optionalModificationToClone));
  567. }
  568. ARAPlaybackRenderer* doCreatePlaybackRenderer() noexcept override
  569. {
  570. return new PlaybackRenderer (getDocumentController());
  571. }
  572. EditorRenderer* doCreateEditorRenderer() noexcept override
  573. {
  574. return new EditorRenderer (getDocumentController(), &previewState);
  575. }
  576. bool doRestoreObjectsFromStream (ARAInputStream& input,
  577. const ARARestoreObjectsFilter* filter) noexcept override
  578. {
  579. // Start reading data from the archive, starting with the number of audio modifications in the archive
  580. const auto numAudioModifications = input.readInt64();
  581. // Loop over stored audio modification data
  582. for (int64 i = 0; i < numAudioModifications; ++i)
  583. {
  584. const auto progressVal = (float) i / (float) numAudioModifications;
  585. getDocumentController()->getHostArchivingController()->notifyDocumentUnarchivingProgress (progressVal);
  586. // Read audio modification persistent ID and analysis result from archive
  587. const String persistentID = input.readString();
  588. const bool dimmed = input.readBool();
  589. // Find audio modification to restore the state to (drop state if not to be loaded)
  590. auto audioModification = filter->getAudioModificationToRestoreStateWithID<ARADemoPluginAudioModification> (persistentID.getCharPointer());
  591. if (audioModification == nullptr)
  592. continue;
  593. const bool dimChanged = (dimmed != audioModification->isDimmed());
  594. audioModification->setDimmed (dimmed);
  595. // If the dim state changed, send a sample content change notification without notifying the host
  596. if (dimChanged)
  597. {
  598. audioModification->notifyContentChanged (ARAContentUpdateScopes::samplesAreAffected(), false);
  599. for (auto playbackRegion : audioModification->getPlaybackRegions())
  600. playbackRegion->notifyContentChanged (ARAContentUpdateScopes::samplesAreAffected(), false);
  601. }
  602. }
  603. getDocumentController()->getHostArchivingController()->notifyDocumentUnarchivingProgress (1.0f);
  604. return ! input.failed();
  605. }
  606. bool doStoreObjectsToStream (ARAOutputStream& output, const ARAStoreObjectsFilter* filter) noexcept override
  607. {
  608. // This example implementation only deals with audio modification states
  609. const auto& audioModificationsToPersist { filter->getAudioModificationsToStore<ARADemoPluginAudioModification>() };
  610. const auto reportProgress = [archivingController = getDocumentController()->getHostArchivingController()] (float p)
  611. {
  612. archivingController->notifyDocumentArchivingProgress (p);
  613. };
  614. const ScopeGuard scope { [&reportProgress] { reportProgress (1.0f); } };
  615. // Write the number of audio modifications we are persisting
  616. const auto numAudioModifications = audioModificationsToPersist.size();
  617. if (! output.writeInt64 ((int64) numAudioModifications))
  618. return false;
  619. // For each audio modification to persist, persist its ID followed by whether it's dimmed
  620. for (size_t i = 0; i < numAudioModifications; ++i)
  621. {
  622. // Write persistent ID and dim state
  623. if (! output.writeString (audioModificationsToPersist[i]->getPersistentID()))
  624. return false;
  625. if (! output.writeBool (audioModificationsToPersist[i]->isDimmed()))
  626. return false;
  627. const auto progressVal = (float) i / (float) numAudioModifications;
  628. reportProgress (progressVal);
  629. }
  630. return true;
  631. }
  632. };
  633. struct PlayHeadState
  634. {
  635. void update (const Optional<AudioPlayHead::PositionInfo>& info)
  636. {
  637. if (info.hasValue())
  638. {
  639. isPlaying.store (info->getIsPlaying(), std::memory_order_relaxed);
  640. timeInSeconds.store (info->getTimeInSeconds().orFallback (0), std::memory_order_relaxed);
  641. isLooping.store (info->getIsLooping(), std::memory_order_relaxed);
  642. const auto loopPoints = info->getLoopPoints();
  643. if (loopPoints.hasValue())
  644. {
  645. loopPpqStart = loopPoints->ppqStart;
  646. loopPpqEnd = loopPoints->ppqEnd;
  647. }
  648. }
  649. else
  650. {
  651. isPlaying.store (false, std::memory_order_relaxed);
  652. isLooping.store (false, std::memory_order_relaxed);
  653. }
  654. }
  655. std::atomic<bool> isPlaying { false },
  656. isLooping { false };
  657. std::atomic<double> timeInSeconds { 0.0 },
  658. loopPpqStart { 0.0 },
  659. loopPpqEnd { 0.0 };
  660. };
  661. //==============================================================================
  662. class ARADemoPluginAudioProcessorImpl : public AudioProcessor,
  663. public AudioProcessorARAExtension
  664. {
  665. public:
  666. //==============================================================================
  667. ARADemoPluginAudioProcessorImpl()
  668. : AudioProcessor (getBusesProperties())
  669. {}
  670. ~ARADemoPluginAudioProcessorImpl() override = default;
  671. //==============================================================================
  672. void prepareToPlay (double sampleRate, int samplesPerBlock) override
  673. {
  674. playHeadState.update (nullopt);
  675. prepareToPlayForARA (sampleRate, samplesPerBlock, getMainBusNumOutputChannels(), getProcessingPrecision());
  676. }
  677. void releaseResources() override
  678. {
  679. playHeadState.update (nullopt);
  680. releaseResourcesForARA();
  681. }
  682. bool isBusesLayoutSupported (const BusesLayout& layouts) const override
  683. {
  684. if (layouts.getMainOutputChannelSet() != AudioChannelSet::mono()
  685. && layouts.getMainOutputChannelSet() != AudioChannelSet::stereo())
  686. return false;
  687. return true;
  688. }
  689. void processBlock (AudioBuffer<float>& buffer, MidiBuffer& midiMessages) override
  690. {
  691. ignoreUnused (midiMessages);
  692. ScopedNoDenormals noDenormals;
  693. auto* audioPlayHead = getPlayHead();
  694. playHeadState.update (audioPlayHead->getPosition());
  695. if (! processBlockForARA (buffer, isRealtime(), audioPlayHead))
  696. processBlockBypassed (buffer, midiMessages);
  697. }
  698. using AudioProcessor::processBlock;
  699. //==============================================================================
  700. const String getName() const override { return "ARAPluginDemo"; }
  701. bool acceptsMidi() const override { return true; }
  702. bool producesMidi() const override { return true; }
  703. double getTailLengthSeconds() const override
  704. {
  705. double tail;
  706. if (getTailLengthSecondsForARA (tail))
  707. return tail;
  708. return 0.0;
  709. }
  710. //==============================================================================
  711. int getNumPrograms() override { return 0; }
  712. int getCurrentProgram() override { return 0; }
  713. void setCurrentProgram (int) override {}
  714. const String getProgramName (int) override { return "None"; }
  715. void changeProgramName (int, const String&) override {}
  716. //==============================================================================
  717. void getStateInformation (MemoryBlock&) override {}
  718. void setStateInformation (const void*, int) override {}
  719. PlayHeadState playHeadState;
  720. private:
  721. //==============================================================================
  722. static BusesProperties getBusesProperties()
  723. {
  724. return BusesProperties().withInput ("Input", AudioChannelSet::stereo(), true)
  725. .withOutput ("Output", AudioChannelSet::stereo(), true);
  726. }
  727. JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (ARADemoPluginAudioProcessorImpl)
  728. };
  729. //==============================================================================
  730. class TimeToViewScaling
  731. {
  732. public:
  733. class Listener
  734. {
  735. public:
  736. virtual ~Listener() = default;
  737. virtual void zoomLevelChanged (double newPixelPerSecond) = 0;
  738. };
  739. void addListener (Listener* l) { listeners.add (l); }
  740. void removeListener (Listener* l) { listeners.remove (l); }
  741. TimeToViewScaling() = default;
  742. void zoom (double factor)
  743. {
  744. zoomLevelPixelPerSecond = jlimit (minimumZoom, minimumZoom * 32, zoomLevelPixelPerSecond * factor);
  745. setZoomLevel (zoomLevelPixelPerSecond);
  746. }
  747. void setZoomLevel (double pixelPerSecond)
  748. {
  749. zoomLevelPixelPerSecond = pixelPerSecond;
  750. listeners.call ([this] (Listener& l) { l.zoomLevelChanged (zoomLevelPixelPerSecond); });
  751. }
  752. int getXForTime (double time) const
  753. {
  754. return roundToInt (time * zoomLevelPixelPerSecond);
  755. }
  756. double getTimeForX (int x) const
  757. {
  758. return x / zoomLevelPixelPerSecond;
  759. }
  760. private:
  761. static constexpr auto minimumZoom = 10.0;
  762. double zoomLevelPixelPerSecond = minimumZoom * 4;
  763. ListenerList<Listener> listeners;
  764. };
  765. class RulersView : public Component,
  766. public SettableTooltipClient,
  767. private Timer,
  768. private TimeToViewScaling::Listener,
  769. private ARAMusicalContext::Listener
  770. {
  771. public:
  772. class CycleMarkerComponent : public Component
  773. {
  774. void paint (Graphics& g) override
  775. {
  776. g.setColour (Colours::yellow.darker (0.2f));
  777. const auto bounds = getLocalBounds().toFloat();
  778. g.drawRoundedRectangle (bounds.getX(), bounds.getY(), bounds.getWidth(), bounds.getHeight(), 6.0f, 2.0f);
  779. }
  780. };
  781. RulersView (PlayHeadState& playHeadStateIn, TimeToViewScaling& timeToViewScalingIn, ARADocument& document)
  782. : playHeadState (playHeadStateIn), timeToViewScaling (timeToViewScalingIn), araDocument (document)
  783. {
  784. timeToViewScaling.addListener (this);
  785. addChildComponent (cycleMarker);
  786. cycleMarker.setInterceptsMouseClicks (false, false);
  787. setTooltip ("Double-click to start playback, click to stop playback or to reposition, drag horizontal range to set cycle.");
  788. startTimerHz (30);
  789. }
  790. ~RulersView() override
  791. {
  792. stopTimer();
  793. timeToViewScaling.removeListener (this);
  794. selectMusicalContext (nullptr);
  795. }
  796. void paint (Graphics& g) override
  797. {
  798. auto drawBounds = g.getClipBounds();
  799. const auto drawStartTime = timeToViewScaling.getTimeForX (drawBounds.getX());
  800. const auto drawEndTime = timeToViewScaling.getTimeForX (drawBounds.getRight());
  801. const auto bounds = getLocalBounds();
  802. g.setColour (getLookAndFeel().findColour (ResizableWindow::backgroundColourId));
  803. g.fillRect (bounds);
  804. g.setColour (getLookAndFeel().findColour (ResizableWindow::backgroundColourId).contrasting());
  805. g.drawRect (bounds);
  806. const auto rulerHeight = bounds.getHeight() / 3;
  807. g.drawRect (drawBounds.getX(), rulerHeight, drawBounds.getRight(), rulerHeight);
  808. g.setFont (Font (12.0f));
  809. const int lightLineWidth = 1;
  810. const int heavyLineWidth = 3;
  811. if (selectedMusicalContext != nullptr)
  812. {
  813. const ARA::PlugIn::HostContentReader<ARA::kARAContentTypeTempoEntries> tempoReader (selectedMusicalContext);
  814. const ARA::TempoConverter<decltype (tempoReader)> tempoConverter (tempoReader);
  815. // chord ruler: one rect per chord, skipping empty "no chords"
  816. const auto chordBounds = drawBounds.removeFromTop (rulerHeight);
  817. const ARA::PlugIn::HostContentReader<ARA::kARAContentTypeSheetChords> chordsReader (selectedMusicalContext);
  818. if (tempoReader && chordsReader)
  819. {
  820. const ARA::ChordInterpreter interpreter (true);
  821. for (auto itChord = chordsReader.begin(); itChord != chordsReader.end(); ++itChord)
  822. {
  823. if (interpreter.isNoChord (*itChord))
  824. continue;
  825. const auto chordStartTime = (itChord == chordsReader.begin()) ? 0 : tempoConverter.getTimeForQuarter (itChord->position);
  826. if (chordStartTime >= drawEndTime)
  827. break;
  828. auto chordRect = chordBounds;
  829. chordRect.setLeft (timeToViewScaling.getXForTime (chordStartTime));
  830. if (std::next (itChord) != chordsReader.end())
  831. {
  832. const auto nextChordStartTime = tempoConverter.getTimeForQuarter (std::next (itChord)->position);
  833. if (nextChordStartTime < drawStartTime)
  834. continue;
  835. chordRect.setRight (timeToViewScaling.getXForTime (nextChordStartTime));
  836. }
  837. g.drawRect (chordRect);
  838. g.drawText (convertARAString (interpreter.getNameForChord (*itChord).c_str()),
  839. chordRect.withTrimmedLeft (2),
  840. Justification::centredLeft);
  841. }
  842. }
  843. // beat ruler: evaluates tempo and bar signatures to draw a line for each beat
  844. const ARA::PlugIn::HostContentReader<ARA::kARAContentTypeBarSignatures> barSignaturesReader (selectedMusicalContext);
  845. if (barSignaturesReader)
  846. {
  847. const ARA::BarSignaturesConverter<decltype (barSignaturesReader)> barSignaturesConverter (barSignaturesReader);
  848. const double beatStart = barSignaturesConverter.getBeatForQuarter (tempoConverter.getQuarterForTime (drawStartTime));
  849. const double beatEnd = barSignaturesConverter.getBeatForQuarter (tempoConverter.getQuarterForTime (drawEndTime));
  850. const int endBeat = roundToInt (std::floor (beatEnd));
  851. RectangleList<int> rects;
  852. for (int beat = roundToInt (std::ceil (beatStart)); beat <= endBeat; ++beat)
  853. {
  854. const auto quarterPos = barSignaturesConverter.getQuarterForBeat (beat);
  855. const int x = timeToViewScaling.getXForTime (tempoConverter.getTimeForQuarter (quarterPos));
  856. const auto barSignature = barSignaturesConverter.getBarSignatureForQuarter (quarterPos);
  857. const int lineWidth = (quarterPos == barSignature.position) ? heavyLineWidth : lightLineWidth;
  858. const int beatsSinceBarStart = roundToInt( barSignaturesConverter.getBeatDistanceFromBarStartForQuarter (quarterPos));
  859. const int lineHeight = (beatsSinceBarStart == 0) ? rulerHeight : rulerHeight / 2;
  860. rects.addWithoutMerging (Rectangle<int> (x - lineWidth / 2, 2 * rulerHeight - lineHeight, lineWidth, lineHeight));
  861. }
  862. g.fillRectList (rects);
  863. }
  864. }
  865. // time ruler: one tick for each second
  866. {
  867. RectangleList<int> rects;
  868. for (auto time = std::floor (drawStartTime); time <= drawEndTime; time += 1.0)
  869. {
  870. const int lineWidth = (std::fmod (time, 60.0) <= 0.001) ? heavyLineWidth : lightLineWidth;
  871. const int lineHeight = (std::fmod (time, 10.0) <= 0.001) ? rulerHeight : rulerHeight / 2;
  872. rects.addWithoutMerging (Rectangle<int> (timeToViewScaling.getXForTime (time) - lineWidth / 2,
  873. bounds.getHeight() - lineHeight,
  874. lineWidth,
  875. lineHeight));
  876. }
  877. g.fillRectList (rects);
  878. }
  879. }
  880. void mouseDrag (const MouseEvent& m) override
  881. {
  882. isDraggingCycle = true;
  883. auto cycleRect = getBounds();
  884. cycleRect.setLeft (jmin (m.getMouseDownX(), m.x));
  885. cycleRect.setRight (jmax (m.getMouseDownX(), m.x));
  886. cycleMarker.setBounds (cycleRect);
  887. }
  888. void mouseUp (const MouseEvent& m) override
  889. {
  890. auto playbackController = araDocument.getDocumentController()->getHostPlaybackController();
  891. if (playbackController != nullptr)
  892. {
  893. const auto startTime = timeToViewScaling.getTimeForX (jmin (m.getMouseDownX(), m.x));
  894. const auto endTime = timeToViewScaling.getTimeForX (jmax (m.getMouseDownX(), m.x));
  895. if (playHeadState.isPlaying.load (std::memory_order_relaxed))
  896. playbackController->requestStopPlayback();
  897. else
  898. playbackController->requestSetPlaybackPosition (startTime);
  899. if (isDraggingCycle)
  900. playbackController->requestSetCycleRange (startTime, endTime - startTime);
  901. }
  902. isDraggingCycle = false;
  903. }
  904. void mouseDoubleClick (const MouseEvent&) override
  905. {
  906. if (auto* playbackController = araDocument.getDocumentController()->getHostPlaybackController())
  907. {
  908. if (! playHeadState.isPlaying.load (std::memory_order_relaxed))
  909. playbackController->requestStartPlayback();
  910. }
  911. }
  912. void selectMusicalContext (ARAMusicalContext* newSelectedMusicalContext)
  913. {
  914. if (auto* oldSelection = std::exchange (selectedMusicalContext, newSelectedMusicalContext);
  915. oldSelection != selectedMusicalContext)
  916. {
  917. if (oldSelection != nullptr)
  918. oldSelection->removeListener (this);
  919. if (selectedMusicalContext != nullptr)
  920. selectedMusicalContext->addListener (this);
  921. repaint();
  922. }
  923. }
  924. void zoomLevelChanged (double) override
  925. {
  926. repaint();
  927. }
  928. void doUpdateMusicalContextContent (ARAMusicalContext*, ARAContentUpdateScopes) override
  929. {
  930. repaint();
  931. }
  932. private:
  933. void updateCyclePosition()
  934. {
  935. if (selectedMusicalContext != nullptr)
  936. {
  937. const ARA::PlugIn::HostContentReader<ARA::kARAContentTypeTempoEntries> tempoReader (selectedMusicalContext);
  938. const ARA::TempoConverter<decltype (tempoReader)> tempoConverter (tempoReader);
  939. const auto loopStartTime = tempoConverter.getTimeForQuarter (playHeadState.loopPpqStart.load (std::memory_order_relaxed));
  940. const auto loopEndTime = tempoConverter.getTimeForQuarter (playHeadState.loopPpqEnd.load (std::memory_order_relaxed));
  941. auto cycleRect = getBounds();
  942. cycleRect.setLeft (timeToViewScaling.getXForTime (loopStartTime));
  943. cycleRect.setRight (timeToViewScaling.getXForTime (loopEndTime));
  944. cycleMarker.setVisible (true);
  945. cycleMarker.setBounds (cycleRect);
  946. }
  947. else
  948. {
  949. cycleMarker.setVisible (false);
  950. }
  951. }
  952. void timerCallback() override
  953. {
  954. if (! isDraggingCycle)
  955. updateCyclePosition();
  956. }
  957. private:
  958. PlayHeadState& playHeadState;
  959. TimeToViewScaling& timeToViewScaling;
  960. ARADocument& araDocument;
  961. ARAMusicalContext* selectedMusicalContext = nullptr;
  962. CycleMarkerComponent cycleMarker;
  963. bool isDraggingCycle = false;
  964. };
  965. class RulersHeader : public Component
  966. {
  967. public:
  968. RulersHeader()
  969. {
  970. chordsLabel.setText ("Chords", NotificationType::dontSendNotification);
  971. addAndMakeVisible (chordsLabel);
  972. barsLabel.setText ("Bars", NotificationType::dontSendNotification);
  973. addAndMakeVisible (barsLabel);
  974. timeLabel.setText ("Time", NotificationType::dontSendNotification);
  975. addAndMakeVisible (timeLabel);
  976. }
  977. void resized() override
  978. {
  979. auto bounds = getLocalBounds();
  980. const auto rulerHeight = bounds.getHeight() / 3;
  981. for (auto* label : { &chordsLabel, &barsLabel, &timeLabel })
  982. label->setBounds (bounds.removeFromTop (rulerHeight));
  983. }
  984. void paint (Graphics& g) override
  985. {
  986. auto bounds = getLocalBounds();
  987. const auto rulerHeight = bounds.getHeight() / 3;
  988. g.setColour (getLookAndFeel().findColour (ResizableWindow::backgroundColourId));
  989. g.fillRect (bounds);
  990. g.setColour (getLookAndFeel().findColour (ResizableWindow::backgroundColourId).contrasting());
  991. g.drawRect (bounds);
  992. bounds.removeFromTop (rulerHeight);
  993. g.drawRect (bounds.removeFromTop (rulerHeight));
  994. }
  995. private:
  996. Label chordsLabel, barsLabel, timeLabel;
  997. };
  998. //==============================================================================
  999. struct WaveformCache : private ARAAudioSource::Listener
  1000. {
  1001. WaveformCache() : thumbnailCache (20)
  1002. {
  1003. }
  1004. ~WaveformCache() override
  1005. {
  1006. for (const auto& entry : thumbnails)
  1007. {
  1008. entry.first->removeListener (this);
  1009. }
  1010. }
  1011. //==============================================================================
  1012. void willDestroyAudioSource (ARAAudioSource* audioSource) override
  1013. {
  1014. removeAudioSource (audioSource);
  1015. }
  1016. AudioThumbnail& getOrCreateThumbnail (ARAAudioSource* audioSource)
  1017. {
  1018. const auto iter = thumbnails.find (audioSource);
  1019. if (iter != std::end (thumbnails))
  1020. return *iter->second;
  1021. auto thumb = std::make_unique<AudioThumbnail> (128, dummyManager, thumbnailCache);
  1022. auto& result = *thumb;
  1023. ++hash;
  1024. thumb->setReader (new ARAAudioSourceReader (audioSource), hash);
  1025. audioSource->addListener (this);
  1026. thumbnails.emplace (audioSource, std::move (thumb));
  1027. return result;
  1028. }
  1029. private:
  1030. void removeAudioSource (ARAAudioSource* audioSource)
  1031. {
  1032. audioSource->removeListener (this);
  1033. thumbnails.erase (audioSource);
  1034. }
  1035. int64 hash = 0;
  1036. AudioFormatManager dummyManager;
  1037. AudioThumbnailCache thumbnailCache;
  1038. std::map<ARAAudioSource*, std::unique_ptr<AudioThumbnail>> thumbnails;
  1039. };
  1040. class PlaybackRegionView : public Component,
  1041. public ChangeListener,
  1042. public SettableTooltipClient,
  1043. private ARAAudioSource::Listener,
  1044. private ARAPlaybackRegion::Listener,
  1045. private ARAEditorView::Listener
  1046. {
  1047. public:
  1048. PlaybackRegionView (ARAEditorView& editorView, ARAPlaybackRegion& region, WaveformCache& cache)
  1049. : araEditorView (editorView), playbackRegion (region), waveformCache (cache), previewRegionOverlay (*this)
  1050. {
  1051. auto* audioSource = playbackRegion.getAudioModification()->getAudioSource();
  1052. waveformCache.getOrCreateThumbnail (audioSource).addChangeListener (this);
  1053. audioSource->addListener (this);
  1054. playbackRegion.addListener (this);
  1055. araEditorView.addListener (this);
  1056. addAndMakeVisible (previewRegionOverlay);
  1057. setTooltip ("Double-click to toggle dim state of the region, click and hold to prelisten region near click.");
  1058. }
  1059. ~PlaybackRegionView() override
  1060. {
  1061. auto* audioSource = playbackRegion.getAudioModification()->getAudioSource();
  1062. audioSource->removeListener (this);
  1063. playbackRegion.removeListener (this);
  1064. araEditorView.removeListener (this);
  1065. waveformCache.getOrCreateThumbnail (audioSource).removeChangeListener (this);
  1066. }
  1067. void mouseDown (const MouseEvent& m) override
  1068. {
  1069. const auto relativeTime = (double) m.getMouseDownX() / getLocalBounds().getWidth();
  1070. const auto previewTime = playbackRegion.getStartInPlaybackTime()
  1071. + relativeTime * playbackRegion.getDurationInPlaybackTime();
  1072. auto& previewState = ARADocumentControllerSpecialisation::getSpecialisedDocumentController<ARADemoPluginDocumentControllerSpecialisation> (playbackRegion.getDocumentController())->previewState;
  1073. previewState.previewTime.store (previewTime);
  1074. previewState.previewedRegion.store (&playbackRegion);
  1075. previewRegionOverlay.update();
  1076. }
  1077. void mouseUp (const MouseEvent&) override
  1078. {
  1079. auto& previewState = ARADocumentControllerSpecialisation::getSpecialisedDocumentController<ARADemoPluginDocumentControllerSpecialisation> (playbackRegion.getDocumentController())->previewState;
  1080. previewState.previewTime.store (0.0);
  1081. previewState.previewedRegion.store (nullptr);
  1082. previewRegionOverlay.update();
  1083. }
  1084. void mouseDoubleClick (const MouseEvent&) override
  1085. {
  1086. // Set the dim flag on our region's audio modification when double-clicked
  1087. auto audioModification = playbackRegion.getAudioModification<ARADemoPluginAudioModification>();
  1088. audioModification->setDimmed (! audioModification->isDimmed());
  1089. // Send a content change notification for the modification and all associated playback regions
  1090. audioModification->notifyContentChanged (ARAContentUpdateScopes::samplesAreAffected(), true);
  1091. for (auto region : audioModification->getPlaybackRegions())
  1092. region->notifyContentChanged (ARAContentUpdateScopes::samplesAreAffected(), true);
  1093. }
  1094. void changeListenerCallback (ChangeBroadcaster*) override
  1095. {
  1096. repaint();
  1097. }
  1098. void didEnableAudioSourceSamplesAccess (ARAAudioSource*, bool) override
  1099. {
  1100. repaint();
  1101. }
  1102. void willUpdatePlaybackRegionProperties (ARAPlaybackRegion*,
  1103. ARAPlaybackRegion::PropertiesPtr newProperties) override
  1104. {
  1105. if (playbackRegion.getName() != newProperties->name
  1106. || playbackRegion.getColor() != newProperties->color)
  1107. {
  1108. repaint();
  1109. }
  1110. }
  1111. void didUpdatePlaybackRegionContent (ARAPlaybackRegion*, ARAContentUpdateScopes) override
  1112. {
  1113. repaint();
  1114. }
  1115. void onNewSelection (const ARAViewSelection& viewSelection) override
  1116. {
  1117. const auto& selectedPlaybackRegions = viewSelection.getPlaybackRegions();
  1118. const bool selected = std::find (selectedPlaybackRegions.begin(), selectedPlaybackRegions.end(), &playbackRegion) != selectedPlaybackRegions.end();
  1119. if (selected != isSelected)
  1120. {
  1121. isSelected = selected;
  1122. repaint();
  1123. }
  1124. }
  1125. void paint (Graphics& g) override
  1126. {
  1127. g.fillAll (convertOptionalARAColour (playbackRegion.getEffectiveColor(), Colours::black));
  1128. const auto* audioModification = playbackRegion.getAudioModification<ARADemoPluginAudioModification>();
  1129. g.setColour (audioModification->isDimmed() ? Colours::darkgrey.darker() : Colours::darkgrey.brighter());
  1130. if (audioModification->getAudioSource()->isSampleAccessEnabled())
  1131. {
  1132. auto& thumbnail = waveformCache.getOrCreateThumbnail (playbackRegion.getAudioModification()->getAudioSource());
  1133. thumbnail.drawChannels (g,
  1134. getLocalBounds(),
  1135. playbackRegion.getStartInAudioModificationTime(),
  1136. playbackRegion.getEndInAudioModificationTime(),
  1137. 1.0f);
  1138. }
  1139. else
  1140. {
  1141. g.setFont (Font (12.0f));
  1142. g.drawText ("Audio Access Disabled", getLocalBounds(), Justification::centred);
  1143. }
  1144. g.setColour (Colours::white.withMultipliedAlpha (0.9f));
  1145. g.setFont (Font (12.0f));
  1146. g.drawText (convertOptionalARAString (playbackRegion.getEffectiveName()),
  1147. getLocalBounds(),
  1148. Justification::topLeft);
  1149. if (audioModification->isDimmed())
  1150. g.drawText ("DIMMED", getLocalBounds(), Justification::bottomLeft);
  1151. g.setColour (isSelected ? Colours::white : Colours::black);
  1152. g.drawRect (getLocalBounds());
  1153. }
  1154. void resized() override
  1155. {
  1156. repaint();
  1157. }
  1158. private:
  1159. class PreviewRegionOverlay : public Component
  1160. {
  1161. static constexpr auto previewLength = 0.5;
  1162. public:
  1163. PreviewRegionOverlay (PlaybackRegionView& ownerIn) : owner (ownerIn)
  1164. {
  1165. }
  1166. void update()
  1167. {
  1168. const auto& previewState = owner.getDocumentController()->previewState;
  1169. if (previewState.previewedRegion.load() == &owner.playbackRegion)
  1170. {
  1171. const auto previewStartTime = previewState.previewTime.load() - owner.playbackRegion.getStartInPlaybackTime();
  1172. const auto pixelPerSecond = owner.getWidth() / owner.playbackRegion.getDurationInPlaybackTime();
  1173. setBounds (roundToInt ((previewStartTime - previewLength / 2) * pixelPerSecond),
  1174. 0,
  1175. roundToInt (previewLength * pixelPerSecond),
  1176. owner.getHeight());
  1177. setVisible (true);
  1178. }
  1179. else
  1180. {
  1181. setVisible (false);
  1182. }
  1183. repaint();
  1184. }
  1185. void paint (Graphics& g) override
  1186. {
  1187. g.setColour (Colours::yellow.withAlpha (0.5f));
  1188. g.fillRect (getLocalBounds());
  1189. }
  1190. private:
  1191. PlaybackRegionView& owner;
  1192. };
  1193. ARADemoPluginDocumentControllerSpecialisation* getDocumentController() const
  1194. {
  1195. return ARADocumentControllerSpecialisation::getSpecialisedDocumentController<ARADemoPluginDocumentControllerSpecialisation> (playbackRegion.getDocumentController());
  1196. }
  1197. ARAEditorView& araEditorView;
  1198. ARAPlaybackRegion& playbackRegion;
  1199. WaveformCache& waveformCache;
  1200. PreviewRegionOverlay previewRegionOverlay;
  1201. bool isSelected = false;
  1202. };
  1203. class RegionSequenceView : public Component,
  1204. public ChangeBroadcaster,
  1205. private TimeToViewScaling::Listener,
  1206. private ARARegionSequence::Listener,
  1207. private ARAPlaybackRegion::Listener
  1208. {
  1209. public:
  1210. RegionSequenceView (ARAEditorView& editorView, TimeToViewScaling& scaling, ARARegionSequence& rs, WaveformCache& cache)
  1211. : araEditorView (editorView), timeToViewScaling (scaling), regionSequence (rs), waveformCache (cache)
  1212. {
  1213. regionSequence.addListener (this);
  1214. for (auto* playbackRegion : regionSequence.getPlaybackRegions())
  1215. createAndAddPlaybackRegionView (playbackRegion);
  1216. updatePlaybackDuration();
  1217. timeToViewScaling.addListener (this);
  1218. }
  1219. ~RegionSequenceView() override
  1220. {
  1221. timeToViewScaling.removeListener (this);
  1222. regionSequence.removeListener (this);
  1223. for (const auto& it : playbackRegionViews)
  1224. it.first->removeListener (this);
  1225. }
  1226. //==============================================================================
  1227. // ARA Document change callback overrides
  1228. void willUpdateRegionSequenceProperties (ARARegionSequence*,
  1229. ARARegionSequence::PropertiesPtr newProperties) override
  1230. {
  1231. if (regionSequence.getColor() != newProperties->color)
  1232. {
  1233. for (auto& pbr : playbackRegionViews)
  1234. pbr.second->repaint();
  1235. }
  1236. }
  1237. void willRemovePlaybackRegionFromRegionSequence (ARARegionSequence*,
  1238. ARAPlaybackRegion* playbackRegion) override
  1239. {
  1240. playbackRegion->removeListener (this);
  1241. removeChildComponent (playbackRegionViews[playbackRegion].get());
  1242. playbackRegionViews.erase (playbackRegion);
  1243. updatePlaybackDuration();
  1244. }
  1245. void didAddPlaybackRegionToRegionSequence (ARARegionSequence*, ARAPlaybackRegion* playbackRegion) override
  1246. {
  1247. createAndAddPlaybackRegionView (playbackRegion);
  1248. updatePlaybackDuration();
  1249. }
  1250. void willDestroyPlaybackRegion (ARAPlaybackRegion* playbackRegion) override
  1251. {
  1252. playbackRegion->removeListener (this);
  1253. removeChildComponent (playbackRegionViews[playbackRegion].get());
  1254. playbackRegionViews.erase (playbackRegion);
  1255. updatePlaybackDuration();
  1256. }
  1257. void didUpdatePlaybackRegionProperties (ARAPlaybackRegion*) override
  1258. {
  1259. updatePlaybackDuration();
  1260. }
  1261. void zoomLevelChanged (double) override
  1262. {
  1263. resized();
  1264. }
  1265. void resized() override
  1266. {
  1267. for (auto& pbr : playbackRegionViews)
  1268. {
  1269. const auto playbackRegion = pbr.first;
  1270. pbr.second->setBounds (
  1271. getLocalBounds()
  1272. .withTrimmedLeft (timeToViewScaling.getXForTime (playbackRegion->getStartInPlaybackTime()))
  1273. .withWidth (timeToViewScaling.getXForTime (playbackRegion->getDurationInPlaybackTime())));
  1274. }
  1275. }
  1276. auto getPlaybackDuration() const noexcept
  1277. {
  1278. return playbackDuration;
  1279. }
  1280. private:
  1281. void createAndAddPlaybackRegionView (ARAPlaybackRegion* playbackRegion)
  1282. {
  1283. playbackRegionViews[playbackRegion] = std::make_unique<PlaybackRegionView> (araEditorView,
  1284. *playbackRegion,
  1285. waveformCache);
  1286. playbackRegion->addListener (this);
  1287. addAndMakeVisible (*playbackRegionViews[playbackRegion]);
  1288. }
  1289. void updatePlaybackDuration()
  1290. {
  1291. const auto iter = std::max_element (
  1292. playbackRegionViews.begin(),
  1293. playbackRegionViews.end(),
  1294. [] (const auto& a, const auto& b) { return a.first->getEndInPlaybackTime() < b.first->getEndInPlaybackTime(); });
  1295. playbackDuration = iter != playbackRegionViews.end() ? iter->first->getEndInPlaybackTime()
  1296. : 0.0;
  1297. sendChangeMessage();
  1298. }
  1299. ARAEditorView& araEditorView;
  1300. TimeToViewScaling& timeToViewScaling;
  1301. ARARegionSequence& regionSequence;
  1302. WaveformCache& waveformCache;
  1303. std::unordered_map<ARAPlaybackRegion*, std::unique_ptr<PlaybackRegionView>> playbackRegionViews;
  1304. double playbackDuration = 0.0;
  1305. };
  1306. class ZoomControls : public Component
  1307. {
  1308. public:
  1309. ZoomControls()
  1310. {
  1311. addAndMakeVisible (zoomInButton);
  1312. addAndMakeVisible (zoomOutButton);
  1313. }
  1314. void setZoomInCallback (std::function<void()> cb) { zoomInButton.onClick = std::move (cb); }
  1315. void setZoomOutCallback (std::function<void()> cb) { zoomOutButton.onClick = std::move (cb); }
  1316. void resized() override
  1317. {
  1318. FlexBox fb;
  1319. fb.justifyContent = FlexBox::JustifyContent::flexEnd;
  1320. for (auto* button : { &zoomInButton, &zoomOutButton })
  1321. fb.items.add (FlexItem (*button).withMinHeight (30.0f).withMinWidth (30.0f).withMargin ({ 5, 5, 5, 0 }));
  1322. fb.performLayout (getLocalBounds());
  1323. }
  1324. private:
  1325. TextButton zoomInButton { "+" }, zoomOutButton { "-" };
  1326. };
  1327. class PlayheadPositionLabel : public Label,
  1328. private Timer
  1329. {
  1330. public:
  1331. PlayheadPositionLabel (PlayHeadState& playHeadStateIn)
  1332. : playHeadState (playHeadStateIn)
  1333. {
  1334. startTimerHz (30);
  1335. }
  1336. ~PlayheadPositionLabel() override
  1337. {
  1338. stopTimer();
  1339. }
  1340. void selectMusicalContext (ARAMusicalContext* newSelectedMusicalContext)
  1341. {
  1342. selectedMusicalContext = newSelectedMusicalContext;
  1343. }
  1344. private:
  1345. void timerCallback() override
  1346. {
  1347. const auto timePosition = playHeadState.timeInSeconds.load (std::memory_order_relaxed);
  1348. auto text = timeToTimecodeString (timePosition);
  1349. if (playHeadState.isPlaying.load (std::memory_order_relaxed))
  1350. text += " (playing)";
  1351. else
  1352. text += " (stopped)";
  1353. if (selectedMusicalContext != nullptr)
  1354. {
  1355. const ARA::PlugIn::HostContentReader<ARA::kARAContentTypeTempoEntries> tempoReader (selectedMusicalContext);
  1356. const ARA::PlugIn::HostContentReader<ARA::kARAContentTypeBarSignatures> barSignaturesReader (selectedMusicalContext);
  1357. if (tempoReader && barSignaturesReader)
  1358. {
  1359. const ARA::TempoConverter<decltype (tempoReader)> tempoConverter (tempoReader);
  1360. const ARA::BarSignaturesConverter<decltype (barSignaturesReader)> barSignaturesConverter (barSignaturesReader);
  1361. const auto quarterPosition = tempoConverter.getQuarterForTime (timePosition);
  1362. const auto barIndex = barSignaturesConverter.getBarIndexForQuarter (quarterPosition);
  1363. const auto beatDistance = barSignaturesConverter.getBeatDistanceFromBarStartForQuarter (quarterPosition);
  1364. const auto quartersPerBeat = 4.0 / (double) barSignaturesConverter.getBarSignatureForQuarter (quarterPosition).denominator;
  1365. const auto beatIndex = (int) beatDistance;
  1366. const auto tickIndex = juce::roundToInt ((beatDistance - beatIndex) * quartersPerBeat * 960.0);
  1367. text += newLine;
  1368. text += String::formatted ("bar %d | beat %d | tick %03d", (barIndex >= 0) ? barIndex + 1 : barIndex, beatIndex + 1, tickIndex + 1);
  1369. text += " - ";
  1370. const ARA::PlugIn::HostContentReader<ARA::kARAContentTypeSheetChords> chordsReader (selectedMusicalContext);
  1371. if (chordsReader && chordsReader.getEventCount() > 0)
  1372. {
  1373. const auto begin = chordsReader.begin();
  1374. const auto end = chordsReader.end();
  1375. auto it = begin;
  1376. while (it->position <= quarterPosition && it != end)
  1377. ++it;
  1378. if (it != begin)
  1379. --it;
  1380. const ARA::ChordInterpreter interpreter (true);
  1381. text += "chord ";
  1382. text += String (interpreter.getNameForChord (*it));
  1383. }
  1384. else
  1385. {
  1386. text += "(no chords provided)";
  1387. }
  1388. }
  1389. }
  1390. setText (text, NotificationType::dontSendNotification);
  1391. }
  1392. // Copied from AudioPluginDemo.h: quick-and-dirty function to format a timecode string
  1393. static String timeToTimecodeString (double seconds)
  1394. {
  1395. auto millisecs = roundToInt (seconds * 1000.0);
  1396. auto absMillisecs = std::abs (millisecs);
  1397. return String::formatted ("%02d:%02d:%02d.%03d",
  1398. millisecs / 3600000,
  1399. (absMillisecs / 60000) % 60,
  1400. (absMillisecs / 1000) % 60,
  1401. absMillisecs % 1000);
  1402. }
  1403. PlayHeadState& playHeadState;
  1404. ARAMusicalContext* selectedMusicalContext = nullptr;
  1405. };
  1406. class TrackHeader : public Component,
  1407. private ARARegionSequence::Listener,
  1408. private ARAEditorView::Listener
  1409. {
  1410. public:
  1411. TrackHeader (ARAEditorView& editorView, ARARegionSequence& regionSequenceIn)
  1412. : araEditorView (editorView), regionSequence (regionSequenceIn)
  1413. {
  1414. updateTrackName (regionSequence.getName());
  1415. onNewSelection (araEditorView.getViewSelection());
  1416. addAndMakeVisible (trackNameLabel);
  1417. regionSequence.addListener (this);
  1418. araEditorView.addListener (this);
  1419. }
  1420. ~TrackHeader() override
  1421. {
  1422. araEditorView.removeListener (this);
  1423. regionSequence.removeListener (this);
  1424. }
  1425. void willUpdateRegionSequenceProperties (ARARegionSequence*, ARARegionSequence::PropertiesPtr newProperties) override
  1426. {
  1427. if (regionSequence.getName() != newProperties->name)
  1428. updateTrackName (newProperties->name);
  1429. if (regionSequence.getColor() != newProperties->color)
  1430. repaint();
  1431. }
  1432. void resized() override
  1433. {
  1434. trackNameLabel.setBounds (getLocalBounds().reduced (2));
  1435. }
  1436. void paint (Graphics& g) override
  1437. {
  1438. const auto backgroundColour = getLookAndFeel().findColour (ResizableWindow::backgroundColourId);
  1439. g.setColour (isSelected ? backgroundColour.brighter() : backgroundColour);
  1440. g.fillRoundedRectangle (getLocalBounds().reduced (2).toFloat(), 6.0f);
  1441. g.setColour (backgroundColour.contrasting());
  1442. g.drawRoundedRectangle (getLocalBounds().reduced (2).toFloat(), 6.0f, 1.0f);
  1443. if (auto colour = regionSequence.getColor())
  1444. {
  1445. g.setColour (convertARAColour (colour));
  1446. g.fillRect (getLocalBounds().removeFromTop (16).reduced (6));
  1447. g.fillRect (getLocalBounds().removeFromBottom (16).reduced (6));
  1448. }
  1449. }
  1450. void onNewSelection (const ARAViewSelection& viewSelection) override
  1451. {
  1452. const auto& selectedRegionSequences = viewSelection.getRegionSequences();
  1453. const bool selected = std::find (selectedRegionSequences.begin(), selectedRegionSequences.end(), &regionSequence) != selectedRegionSequences.end();
  1454. if (selected != isSelected)
  1455. {
  1456. isSelected = selected;
  1457. repaint();
  1458. }
  1459. }
  1460. private:
  1461. void updateTrackName (ARA::ARAUtf8String optionalName)
  1462. {
  1463. trackNameLabel.setText (optionalName ? optionalName : "No track name",
  1464. NotificationType::dontSendNotification);
  1465. }
  1466. ARAEditorView& araEditorView;
  1467. ARARegionSequence& regionSequence;
  1468. Label trackNameLabel;
  1469. bool isSelected = false;
  1470. };
  1471. constexpr auto trackHeight = 60;
  1472. class VerticalLayoutViewportContent : public Component
  1473. {
  1474. public:
  1475. void resized() override
  1476. {
  1477. auto bounds = getLocalBounds();
  1478. for (auto* component : getChildren())
  1479. {
  1480. component->setBounds (bounds.removeFromTop (trackHeight));
  1481. component->resized();
  1482. }
  1483. }
  1484. };
  1485. class VerticalLayoutViewport : public Viewport
  1486. {
  1487. public:
  1488. VerticalLayoutViewport()
  1489. {
  1490. setViewedComponent (&content, false);
  1491. }
  1492. void paint (Graphics& g) override
  1493. {
  1494. g.fillAll (getLookAndFeel().findColour (ResizableWindow::backgroundColourId).brighter());
  1495. }
  1496. std::function<void (Rectangle<int>)> onVisibleAreaChanged;
  1497. VerticalLayoutViewportContent content;
  1498. private:
  1499. void visibleAreaChanged (const Rectangle<int>& newVisibleArea) override
  1500. {
  1501. NullCheckedInvocation::invoke (onVisibleAreaChanged, newVisibleArea);
  1502. }
  1503. };
  1504. class OverlayComponent : public Component,
  1505. private Timer,
  1506. private TimeToViewScaling::Listener
  1507. {
  1508. public:
  1509. class PlayheadMarkerComponent : public Component
  1510. {
  1511. void paint (Graphics& g) override { g.fillAll (Colours::yellow.darker (0.2f)); }
  1512. };
  1513. OverlayComponent (PlayHeadState& playHeadStateIn, TimeToViewScaling& timeToViewScalingIn)
  1514. : playHeadState (playHeadStateIn), timeToViewScaling (timeToViewScalingIn)
  1515. {
  1516. addChildComponent (playheadMarker);
  1517. setInterceptsMouseClicks (false, false);
  1518. startTimerHz (30);
  1519. timeToViewScaling.addListener (this);
  1520. }
  1521. ~OverlayComponent() override
  1522. {
  1523. timeToViewScaling.removeListener (this);
  1524. stopTimer();
  1525. }
  1526. void resized() override
  1527. {
  1528. updatePlayHeadPosition();
  1529. }
  1530. void setHorizontalOffset (int offset)
  1531. {
  1532. horizontalOffset = offset;
  1533. }
  1534. void setSelectedTimeRange (std::optional<ARA::ARAContentTimeRange> timeRange)
  1535. {
  1536. selectedTimeRange = timeRange;
  1537. repaint();
  1538. }
  1539. void zoomLevelChanged (double) override
  1540. {
  1541. updatePlayHeadPosition();
  1542. repaint();
  1543. }
  1544. void paint (Graphics& g) override
  1545. {
  1546. if (selectedTimeRange)
  1547. {
  1548. auto bounds = getLocalBounds();
  1549. bounds.setLeft (timeToViewScaling.getXForTime (selectedTimeRange->start));
  1550. bounds.setRight (timeToViewScaling.getXForTime (selectedTimeRange->start + selectedTimeRange->duration));
  1551. g.setColour (getLookAndFeel().findColour (ResizableWindow::backgroundColourId).brighter().withAlpha (0.3f));
  1552. g.fillRect (bounds);
  1553. g.setColour (Colours::whitesmoke.withAlpha (0.5f));
  1554. g.drawRect (bounds);
  1555. }
  1556. }
  1557. private:
  1558. void updatePlayHeadPosition()
  1559. {
  1560. if (playHeadState.isPlaying.load (std::memory_order_relaxed))
  1561. {
  1562. const auto markerX = timeToViewScaling.getXForTime (playHeadState.timeInSeconds.load (std::memory_order_relaxed));
  1563. const auto playheadLine = getLocalBounds().withTrimmedLeft ((int) (markerX - markerWidth / 2.0) - horizontalOffset)
  1564. .removeFromLeft ((int) markerWidth);
  1565. playheadMarker.setVisible (true);
  1566. playheadMarker.setBounds (playheadLine);
  1567. }
  1568. else
  1569. {
  1570. playheadMarker.setVisible (false);
  1571. }
  1572. }
  1573. void timerCallback() override
  1574. {
  1575. updatePlayHeadPosition();
  1576. }
  1577. static constexpr double markerWidth = 2.0;
  1578. PlayHeadState& playHeadState;
  1579. TimeToViewScaling& timeToViewScaling;
  1580. int horizontalOffset = 0;
  1581. std::optional<ARA::ARAContentTimeRange> selectedTimeRange;
  1582. PlayheadMarkerComponent playheadMarker;
  1583. };
  1584. class DocumentView : public Component,
  1585. public ChangeListener,
  1586. public ARAMusicalContext::Listener,
  1587. private ARADocument::Listener,
  1588. private ARAEditorView::Listener
  1589. {
  1590. public:
  1591. DocumentView (ARAEditorView& editorView, PlayHeadState& playHeadState)
  1592. : araEditorView (editorView),
  1593. araDocument (*editorView.getDocumentController()->getDocument<ARADocument>()),
  1594. rulersView (playHeadState, timeToViewScaling, araDocument),
  1595. overlay (playHeadState, timeToViewScaling),
  1596. playheadPositionLabel (playHeadState)
  1597. {
  1598. if (araDocument.getMusicalContexts().size() > 0)
  1599. selectMusicalContext (araDocument.getMusicalContexts().front());
  1600. addAndMakeVisible (rulersHeader);
  1601. viewport.content.addAndMakeVisible (rulersView);
  1602. viewport.onVisibleAreaChanged = [this] (const auto& r)
  1603. {
  1604. viewportHeightOffset = r.getY();
  1605. overlay.setHorizontalOffset (r.getX());
  1606. resized();
  1607. };
  1608. addAndMakeVisible (viewport);
  1609. addAndMakeVisible (overlay);
  1610. addAndMakeVisible (playheadPositionLabel);
  1611. zoomControls.setZoomInCallback ([this] { zoom (2.0); });
  1612. zoomControls.setZoomOutCallback ([this] { zoom (0.5); });
  1613. addAndMakeVisible (zoomControls);
  1614. invalidateRegionSequenceViews();
  1615. araDocument.addListener (this);
  1616. araEditorView.addListener (this);
  1617. }
  1618. ~DocumentView() override
  1619. {
  1620. araEditorView.removeListener (this);
  1621. araDocument.removeListener (this);
  1622. selectMusicalContext (nullptr);
  1623. }
  1624. //==============================================================================
  1625. // ARADocument::Listener overrides
  1626. void didAddMusicalContextToDocument (ARADocument*, ARAMusicalContext* musicalContext) override
  1627. {
  1628. if (selectedMusicalContext == nullptr)
  1629. selectMusicalContext (musicalContext);
  1630. }
  1631. void willDestroyMusicalContext (ARAMusicalContext* musicalContext) override
  1632. {
  1633. if (selectedMusicalContext == musicalContext)
  1634. selectMusicalContext (nullptr);
  1635. }
  1636. void didReorderRegionSequencesInDocument (ARADocument*) override
  1637. {
  1638. invalidateRegionSequenceViews();
  1639. }
  1640. void didAddRegionSequenceToDocument (ARADocument*, ARARegionSequence*) override
  1641. {
  1642. invalidateRegionSequenceViews();
  1643. }
  1644. void willRemoveRegionSequenceFromDocument (ARADocument*, ARARegionSequence* regionSequence) override
  1645. {
  1646. removeRegionSequenceView (regionSequence);
  1647. }
  1648. void didEndEditing (ARADocument*) override
  1649. {
  1650. rebuildRegionSequenceViews();
  1651. update();
  1652. }
  1653. //==============================================================================
  1654. void changeListenerCallback (ChangeBroadcaster*) override
  1655. {
  1656. update();
  1657. }
  1658. //==============================================================================
  1659. // ARAEditorView::Listener overrides
  1660. void onNewSelection (const ARAViewSelection& viewSelection) override
  1661. {
  1662. auto getNewSelectedMusicalContext = [&viewSelection]() -> ARAMusicalContext*
  1663. {
  1664. if (! viewSelection.getRegionSequences().empty())
  1665. return viewSelection.getRegionSequences<ARARegionSequence>().front()->getMusicalContext();
  1666. else if (! viewSelection.getPlaybackRegions().empty())
  1667. return viewSelection.getPlaybackRegions<ARAPlaybackRegion>().front()->getRegionSequence()->getMusicalContext();
  1668. return nullptr;
  1669. };
  1670. if (auto* newSelectedMusicalContext = getNewSelectedMusicalContext())
  1671. if (newSelectedMusicalContext != selectedMusicalContext)
  1672. selectMusicalContext (newSelectedMusicalContext);
  1673. if (const auto timeRange = viewSelection.getTimeRange())
  1674. overlay.setSelectedTimeRange (*timeRange);
  1675. else
  1676. overlay.setSelectedTimeRange (std::nullopt);
  1677. }
  1678. void onHideRegionSequences (const std::vector<ARARegionSequence*>& regionSequences) override
  1679. {
  1680. hiddenRegionSequences = regionSequences;
  1681. invalidateRegionSequenceViews();
  1682. }
  1683. //==============================================================================
  1684. void paint (Graphics& g) override
  1685. {
  1686. g.fillAll (getLookAndFeel().findColour (ResizableWindow::backgroundColourId).darker());
  1687. }
  1688. void resized() override
  1689. {
  1690. auto bounds = getLocalBounds();
  1691. FlexBox fb;
  1692. fb.justifyContent = FlexBox::JustifyContent::spaceBetween;
  1693. fb.items.add (FlexItem (playheadPositionLabel).withWidth (450.0f).withMinWidth (250.0f));
  1694. fb.items.add (FlexItem (zoomControls).withMinWidth (80.0f));
  1695. fb.performLayout (bounds.removeFromBottom (40));
  1696. auto headerBounds = bounds.removeFromLeft (headerWidth);
  1697. rulersHeader.setBounds (headerBounds.removeFromTop (trackHeight));
  1698. layOutVertically (headerBounds, trackHeaders, viewportHeightOffset);
  1699. viewport.setBounds (bounds);
  1700. overlay.setBounds (bounds.reduced (1));
  1701. const auto width = jmax (timeToViewScaling.getXForTime (timelineLength), viewport.getWidth());
  1702. const auto height = (int) (regionSequenceViews.size() + 1) * trackHeight;
  1703. viewport.content.setSize (width, height);
  1704. viewport.content.resized();
  1705. }
  1706. //==============================================================================
  1707. static constexpr int headerWidth = 120;
  1708. private:
  1709. struct RegionSequenceViewKey
  1710. {
  1711. explicit RegionSequenceViewKey (ARARegionSequence* regionSequence)
  1712. : orderIndex (regionSequence->getOrderIndex()), sequence (regionSequence)
  1713. {
  1714. }
  1715. bool operator< (const RegionSequenceViewKey& other) const
  1716. {
  1717. return std::tie (orderIndex, sequence) < std::tie (other.orderIndex, other.sequence);
  1718. }
  1719. ARA::ARAInt32 orderIndex;
  1720. ARARegionSequence* sequence;
  1721. };
  1722. void selectMusicalContext (ARAMusicalContext* newSelectedMusicalContext)
  1723. {
  1724. if (auto oldContext = std::exchange (selectedMusicalContext, newSelectedMusicalContext);
  1725. oldContext != selectedMusicalContext)
  1726. {
  1727. if (oldContext != nullptr)
  1728. oldContext->removeListener (this);
  1729. if (selectedMusicalContext != nullptr)
  1730. selectedMusicalContext->addListener (this);
  1731. rulersView.selectMusicalContext (selectedMusicalContext);
  1732. playheadPositionLabel.selectMusicalContext (selectedMusicalContext);
  1733. }
  1734. }
  1735. void zoom (double factor)
  1736. {
  1737. timeToViewScaling.zoom (factor);
  1738. update();
  1739. }
  1740. template <typename T>
  1741. void layOutVertically (Rectangle<int> bounds, T& components, int verticalOffset = 0)
  1742. {
  1743. bounds = bounds.withY (bounds.getY() - verticalOffset).withHeight (bounds.getHeight() + verticalOffset);
  1744. for (auto& component : components)
  1745. {
  1746. component.second->setBounds (bounds.removeFromTop (trackHeight));
  1747. component.second->resized();
  1748. }
  1749. }
  1750. void update()
  1751. {
  1752. timelineLength = 0.0;
  1753. for (const auto& view : regionSequenceViews)
  1754. timelineLength = std::max (timelineLength, view.second->getPlaybackDuration());
  1755. resized();
  1756. }
  1757. void addTrackViews (ARARegionSequence* regionSequence)
  1758. {
  1759. const auto insertIntoMap = [](auto& map, auto key, auto value) -> auto&
  1760. {
  1761. auto it = map.insert ({ std::move (key), std::move (value) });
  1762. return *(it.first->second);
  1763. };
  1764. auto& regionSequenceView = insertIntoMap (
  1765. regionSequenceViews,
  1766. RegionSequenceViewKey { regionSequence },
  1767. std::make_unique<RegionSequenceView> (araEditorView, timeToViewScaling, *regionSequence, waveformCache));
  1768. regionSequenceView.addChangeListener (this);
  1769. viewport.content.addAndMakeVisible (regionSequenceView);
  1770. auto& trackHeader = insertIntoMap (trackHeaders,
  1771. RegionSequenceViewKey { regionSequence },
  1772. std::make_unique<TrackHeader> (araEditorView, *regionSequence));
  1773. addAndMakeVisible (trackHeader);
  1774. }
  1775. void removeRegionSequenceView (ARARegionSequence* regionSequence)
  1776. {
  1777. const auto& view = regionSequenceViews.find (RegionSequenceViewKey { regionSequence });
  1778. if (view != regionSequenceViews.cend())
  1779. {
  1780. removeChildComponent (view->second.get());
  1781. regionSequenceViews.erase (view);
  1782. }
  1783. invalidateRegionSequenceViews();
  1784. }
  1785. void invalidateRegionSequenceViews()
  1786. {
  1787. regionSequenceViewsAreValid = false;
  1788. rebuildRegionSequenceViews();
  1789. }
  1790. void rebuildRegionSequenceViews()
  1791. {
  1792. if (! regionSequenceViewsAreValid && ! araDocument.getDocumentController()->isHostEditingDocument())
  1793. {
  1794. for (auto& view : regionSequenceViews)
  1795. removeChildComponent (view.second.get());
  1796. regionSequenceViews.clear();
  1797. for (auto& view : trackHeaders)
  1798. removeChildComponent (view.second.get());
  1799. trackHeaders.clear();
  1800. for (auto* regionSequence : araDocument.getRegionSequences())
  1801. if (std::find (hiddenRegionSequences.begin(), hiddenRegionSequences.end(), regionSequence) == hiddenRegionSequences.end())
  1802. addTrackViews (regionSequence);
  1803. update();
  1804. regionSequenceViewsAreValid = true;
  1805. }
  1806. }
  1807. ARAEditorView& araEditorView;
  1808. ARADocument& araDocument;
  1809. bool regionSequenceViewsAreValid = false;
  1810. TimeToViewScaling timeToViewScaling;
  1811. double timelineLength = 0.0;
  1812. ARAMusicalContext* selectedMusicalContext = nullptr;
  1813. std::vector<ARARegionSequence*> hiddenRegionSequences;
  1814. WaveformCache waveformCache;
  1815. std::map<RegionSequenceViewKey, std::unique_ptr<TrackHeader>> trackHeaders;
  1816. std::map<RegionSequenceViewKey, std::unique_ptr<RegionSequenceView>> regionSequenceViews;
  1817. RulersHeader rulersHeader;
  1818. RulersView rulersView;
  1819. VerticalLayoutViewport viewport;
  1820. OverlayComponent overlay;
  1821. ZoomControls zoomControls;
  1822. PlayheadPositionLabel playheadPositionLabel;
  1823. TooltipWindow tooltip;
  1824. int viewportHeightOffset = 0;
  1825. };
  1826. class ARADemoPluginProcessorEditor : public AudioProcessorEditor,
  1827. public AudioProcessorEditorARAExtension
  1828. {
  1829. public:
  1830. explicit ARADemoPluginProcessorEditor (ARADemoPluginAudioProcessorImpl& p)
  1831. : AudioProcessorEditor (&p),
  1832. AudioProcessorEditorARAExtension (&p)
  1833. {
  1834. if (auto* editorView = getARAEditorView())
  1835. documentView = std::make_unique<DocumentView> (*editorView, p.playHeadState);
  1836. addAndMakeVisible (documentView.get());
  1837. // ARA requires that plugin editors are resizable to support tight integration
  1838. // into the host UI
  1839. setResizable (true, false);
  1840. setSize (800, 300);
  1841. }
  1842. //==============================================================================
  1843. void paint (Graphics& g) override
  1844. {
  1845. g.fillAll (getLookAndFeel().findColour (ResizableWindow::backgroundColourId));
  1846. if (! isARAEditorView())
  1847. {
  1848. g.setColour (Colours::white);
  1849. g.setFont (15.0f);
  1850. g.drawFittedText ("ARA host isn't detected. This plugin only supports ARA mode",
  1851. getLocalBounds(),
  1852. Justification::centred,
  1853. 1);
  1854. }
  1855. }
  1856. void resized() override
  1857. {
  1858. if (documentView != nullptr)
  1859. documentView->setBounds (getLocalBounds());
  1860. }
  1861. private:
  1862. std::unique_ptr<Component> documentView;
  1863. JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (ARADemoPluginProcessorEditor)
  1864. };
  1865. class ARADemoPluginAudioProcessor : public ARADemoPluginAudioProcessorImpl
  1866. {
  1867. public:
  1868. bool hasEditor() const override { return true; }
  1869. AudioProcessorEditor* createEditor() override { return new ARADemoPluginProcessorEditor (*this); }
  1870. };