Ultraschall Internals Documentation Reaper Internals Documentation ReaGirl Documentation Downloads Changelog of documentation Impressum and Contact
Ultraschall-logo Functions Engine GFX Engine Doc Engine Video Engine      
  Introduction/Concepts        Concepts      Concepts      Concepts
           Functions      Functions       Functions       Functions 

Jump to Index


^ Introduction

Introduction to the Ultraschall API
How to install Ultraschall API
Introduction: How to use Ultraschall API
Introduction: Bugreporting and Feature Requests
Introduction: License

^ Datatypes

Datatypes: Introduction
Datatypes: Trackstrings
Datatypes: MediaItemArray
Datatypes: MediaItemStateChunkArray
Datatypes: EnvelopePointObject
Datatypes: EnvelopePointArray
Datatypes: ColorTable
Datatypes: RenderTable
Datatypes: Checking Datatypes

^ API-Variables

API-Variables

^ Rendering

Rendering: Introduction
Rendering: About Renderstrings
Rendering: About RenderTables
Rendering: About Rendering-functions
Rendering: Change more render-settings in projectfiles
Rendering: Render Presets
Rendering: Render Queue

^ Arrangeview Snapshots

Arrangeview Snapshots: Introduction
Arrangeview Snapshots: How to store, retrieve, delete
Arrangeview Snapshots: How to restore

^ Navigation

Navigation: Introduction
Navigation: Move Play and Editcursor
Navigation: Go to markers, regionedges and itemedges
Navigation: Center View
Navigation: Autoscroll and Followmode

^ Get/Set Project/Track/MediaItem-States

Get/Set States for Project, Tracks and Items(including StateChunks)

^ MediaItems

MediaItems: Introduction
MediaItems: Getting MediaItems by Time and Tracks
MediaItems: Splitting MediaItems by Time and Tracks
MediaItems: Deleting MediaItems by Time and Tracks
MediaItems: SectionCut, RippleCut, RippleInsert
MediaItems: Moving and Manipulating
MediaItems: Inserting Items and Files
MediaItems: Programming Spectral Edit
MediaItems: Miscellaneous

^ File Management

File Management: Introduction
File Management: Read
File Management: Write
File Management: Analyse
File Management: Background Copying
File Management: Misc

^ Project Management

Project Management: Introduction
Project Management: Check for changed projecttabs

^ Color Management

Color Management: Introduction
Color Management: Native Color Conversion
Color Management: Brightness, Contrast and Colorsaturation
Color Management: Working with Colortables
Color Management: Creating Colortables
Color Management: Applying Colortables

^ Background-Scripts

Background Scripts: Introduction

^ Cough and Mute-Buttons

Cough and Mute Buttons/Actions: Introduction
Cough and Mute Buttons/Actions: Toggling Mute
Cough and Mute Buttons/Actions: Toggling Mute
Cough and Mute Buttons/Actions: Toggling Mute

^ Error Messaging System

Error Messaging System: Introduction
Error Messaging System: Creating Error Messages
Error Messaging System: Getting Error Messages
Error Messaging System: Deleting Error Messages
Error Messaging System: Toggling showing errors in IDE instead
Error Messaging System: Other helpers for Error-Messaging-system

^ TrackStates

Trackstate Management: Introduction

^ Routing

Routing: Introduction
Routing: Send and Receives
Routing: Hardware Outs
Routing: Mass manipulation of Routings

^ ExtState Management

ExtState Management: Introduction
ExtState Management: Ini-Files
ExtState Management: Inifile-Functions
ExtState Management: Ultraschall.ini
ExtState Management: Track and Item-Extstates
ExtState Management: Track Extstates
ExtState Management: Item Extstates
ExtState Management: Marker Extstates

^ Marker Management

Markers and Regions: Introduction
Markers and Regions: General How To
Markers and Regions: Helpers and Manipulation
Markers and Regions: Custom-Markers and Custom-Regions

^ Child Scripts

Child Scripts: Introduction
Child Scripts: The unique ScriptIdentifier
Child Scripts: Running Childscripts
Child Scripts: Passing Parameters and Returnvalues

^ Defer Loops

DeferScripts: Introduction to Ultraschall's Defer-functions
DeferScripts: Special Defer-loops in the Ultraschall-API
DeferScripts: Stopping Defer-loops from in- and outside of scripts
DeferScripts: Manipulating and protecting Defer-loops

^ Localize Scripts

Localize Scripts: Introduction
Localize Scripts: Language-pack-fileformat
Localize Scripts: Functions

^ EventManager

EventManager: Introduction
EventManager: The Basic Concept
EventManager: Basic workflow
EventManager: Which events can be checked?
EventManager: the eventcheck-function for true/false-events
EventManager: Creating Events for transition-events
EventManager: Adding events
EventManager: Alter Events and retrieve settings
EventManager: Debugging Events
EventManager: Working with startup events

^ FX-Management

FX-Management: Introduction
FX-Management: Working with FX-StateChunks
FX-Management: Working with FX-StateChunk's functions
FX-Management: Manipulating individual effects
Parameter Modulation/Learn/Alias: Introduction
Parameter Alias
Parameter Mapping Learn
Parameter LFO Learn
Parameter Modulation Introduction
Parameter Modulation: the ParmModTable
Parameter Modulation: Using ParmModTables
Parameter Modulation: ParmModTable-examples

^ Helper Functions

Helper_Functions: Introduction
Helper_Functions: Clipboard Management
Helper_Functions: Data Manipulation
Helper_Functions: Undo Management
Helper_Functions: Miscellaneous

^ Final Words

Final words


^ Introduction to the Ultraschall API

The Ultraschall-Extension is intended to be an extension for the DAW Reaper, that enhances it with podcast functionalities. Most DAWs are intended to be used by musicians, for music, but podcasters have their own needs to be fulfilled. In fact, in some places their needs differ from the needs of a musician heavily. Ultraschall is intended to optimise the Reaper's workflows, by reworking them with functionalities for the special needs of podcasters.

The Ultraschall-Framework itself is intended to include a set of Lua-functions, that help creating such functionalities. By giving programmers helper functions to get access to each and every corner of Reaper. That way, extending Ultraschall and Reaper is more comfortable to do.

This API was to be used within Ultraschall only, but quickly evolved into a huge 1000+ function-library, that many 3rd-party programmers and scripters may find use in, with many useful features, like:

Happy coding and let's see, what you can do with it :D

Meo-Ada Mespotine (mespotine.de) (ultraschall.fm/api)

For more information about Ultraschall itself, see ultraschall.fm and if you want to support us, see ultraschall.fm/danke for donating to us.

PS: In this documentation, I assume you have some basic knowledge in Lua and in using Reaper's own API-functions. Explaining both of these is beyond the scope of this doc.



^ How to install Ultraschall API

Installing Ultraschall-API is quite easy.

First make sure, you use the right versions of Reaper, SWS and JS-extension: Reaper 7.03 and higher, SWS 2.10.0.1 and higher and Julian Sader's plugin 1.215 and higher. You can download them at reaper.fm and sws-extension.org respectively. Julian Sader's plugin can be installed using ReaPack or gotten from his own github-site

To install the Ultraschall-API, just follow the following steps:

Installation via ReaPack:

The easiest way is to install the Ultraschall-API using ReaPack. ReaPack is a package-manager for extensions and helper-stuff for Reaper, which allows you to easily install and update content done by the Reaper community. Many Scripts, Themes, JSFX-FX, etc are available through that.

  1. Install ReaPack. Get it from https://reapack.com/. There's also a real good User guide for it available, that explains, how to do it.
    Copy the downloaded dll(Windows), dylib(Mac) or so(Linux)-file into the UserPlugins-folder in the Resources-folder of Reaper. You can find the correct Resources-path in the Reaper-menu "Options -> Show REAPER resource path in explorer/finder...".

  2. (Re-)Start Reaper

  3. Go into the Menu Extensions -> ReaPack -> Manage Repositories

  4. Click on Import/export...-button and choose Import repositories

  5. Paste into the dialog the following link

     https://github.com/Ultraschall/ultraschall-lua-api-for-reaper/raw/master/ultraschall_api_index.xml

    and hit OK.

  6. Doubleclick on the Ultraschall-API-entry. A dialog will pop up with a description of the Ultraschall-API

  7. Hit the Install/update Ultraschall-API-button and select Install all packages in this repository

  8. ReaPack will ask you, if you want to install new packages/updates the next time you synchronize ReaPack.
    Hit Yes.

  9. ReaPack installs the Ultraschall-API. If no error appears, it will tell you to restart Reaper, as a new extension has been installed

  10. Restart Reaper.

To update the Ultraschall-API in the future, choose in the menu Extensions -> ReaPack -> Synchronize packages. If an update is available, it will install it automatically.
For more information on ReaPack and it's usage, refer User guide.

Manual Installation:

If you can't or don't want to install ReaPack, you can also install it manually.

  1. Just download the zip-file of the current version from ultraschall_api4.9.zip.
  2. Extract it to the UserPlugins-folder in the Resources-folder of Reaper. You can find the correct Resources-path in the Reaper-menu "Options -> Show REAPER resource path in explorer/finder...".
    The folder UserPlugins should contain the folder ultraschall_api and the files ultraschall_api_readme.txt and ultraschall_api.lua after this step.
  3. Restart Reaper.

In the next chapter, I'll explain, how to use Ultraschall-API.



^ Introduction: How to use Ultraschall API

Usage is really simple. First, create a new script. For this, go into the menu "Actions", hit "Show action list", click the "New action"-button and select "New ReaScript". In the file-requester, you enter a filename for the new script. Just test it by naming it "Test_Ultraschall_Api.lua" Now an editor appears, in which you can enter the code.

Add the following line at the beginning of your script:

        dofile(reaper.GetResourcePath().."/UserPlugins/ultraschall_api.lua")
        

and you can immediately use the functions of Ultraschall-API. For instance, you can test now, if Ultraschall-API works and if installation went alright. Just add the following function underneath the dofile-line:

        ultraschall.ApiTest()

And hit Cmd+S/Ctrl+S. A messagebox should appear, that tells you, that the API is up and running. If an error-message pops up, please try installing again, according to the manual.

If you are already familiar with programming Reaper's own API-functions, you know, that all of Reaper's functions are placed in a table called "reaper."
e.g. reaper.ShowConsoleMsg("msg")

The Ultraschall-API is quite the same, but is using the table "ultraschall." instead.
e.g. ultraschall.ApiTest()

See the functions-reference for all available functions or read on in this documentation for a collection of concepts introduced by the Ultraschall-API.

You can also find all documentations(including the Lua-Reference Manual) when searching for "Ultraschall Help" in the actionlist(when installed via ReaPack).

Oh, before I forget: The ultraschall_api-folder holds a folder "Scripts_Examples" in which you can find some demos and example-scripts using this API. If you have installed Ultraschall-API via ReaPack, you can also run the action "Script: ultraschall_Add_Developertools_To_Reaper.lua" to install them.



^ Introduction: Bugreporting and Feature Requests

If you find any bugs or itches and want to report them, I suggest you the following procedure:

  1. Make notes of: what operating-system you use(Mac, Win, Linux), which Reaper-version, which SWS-Version and which Ultraschall-Framework-Version.

  2. Write down, what you wanted to do, what you expected to happen and what has happened instead. Make it as detailed as possible(a code-fragment that triggers a bug, a screenvideo i.e. would be perfect), as more information helps to find out, where the problem lies. It's always better to write too much, than the other way around. Please keep in mind when sending code-fragments: they need to be able to trigger the bug without any of your other code. And please don't send me hundreds of lines of your code, as I can't debug it for you. Just concentrate on the line(s), that trigger the bug successfully.

  3. Send these notes either as:
    Issue at the GitHub-Repository of the Ultraschall-API(preferred): https://github.com/Ultraschall/Ultraschall-Api-for-Reaper.git
    eMail: lspmp3@yahoo.de(for framework-related stuff only!!)
    Sendegate: sendegate.de into the Ultraschall-section.

Bugreports that contain only a "it doesn't work" and "I expected it to work" will be ignored gracefully ;)

If you have feature-requests, we have open ears. Keep in mind, not everything you find a good idea actually is one. So we may or may not take on your idea, change and rework it into a way, that benefits all, not just your particular use-case. When in doubt, just try it! Keep also in mind: there are limitations. Some cool features we all would love to have, simply aren't implementable.
Que sera, sera...

For your comments just send a mail at: lspmp3@yahoo.de(for framework-related stuff only!!) or go to sendegate.de into the Ultraschall-section.

PS: If you know how to implement impossible things or do things better than the current implementation, you are welcome to donate your improved codes. :)



^ Introduction: License

Copyright (c) 2014-2019 Ultraschall (http://ultraschall.fm)

Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:

The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.

Reaper and the Reaper-Logo are trademarks of Cockos inc and can be found at reaper.fm

The SWS-logo has been taken from the SWS-extension-project, which can be found at sws-extension.org

Ultraschall-API written by Meo-Ada Mespotine mespotine.de with contributions from Udo Sauer and Ralf Stockmann

If you want to donate to our project, head over to ultraschall.fm/danke.

Kudos to lokasenna, who suggested some cool things, that made some functions much faster and sparkled new ones. Cheers!
MakeCopyOfTable-function adapted from Tyler Neylon's (twitter.com/tylerneylon) function, found at Stack Overflow
Thanks to him for allowing me to use it :)



^ Datatypes: Introduction

Parameters and returnvalues are usually of specific types. Let's look at the following function:

        integer number_of_items, array MediaItemArray, array MediaItemStateChunkArray = 
                                        ultraschall.GetMediaItemsAtPosition(number position, string trackstring)

The returnvalues number_of_items is of type integer, MediaItemArray is of type array/type, MediaItemStateChunkArray if of type array. The parameters position is of type number, trackstring is of type string. These parameters/returnvalues only accept/return data of these given types.

The Ultraschall-API uses a lot of the standard-datatypes provided by Lua and Reaper/SWS, which are:

If a datatype has "optional" before it, this parameter/returnvalue is optional. That means, such parameters don't need to be given, such return-values can be nil.

In addition to the already used datatypes, the Ultraschall-API introduces some more datatypes, like:

These Ultraschall-API-specific datatypes are described in more detail in the following chapters.



^ Datatypes: Trackstrings

Many functions allow you to process through multiple tracks. As good as this is, this provided some difficulty in telling a function, to which track it shall be applied to.
Hence the datatype: trackstring

A trackstring is just a simple string with all tracknumbers that you want a certain function to be applied to.
Examplecode:

        trackstring = "1,2,5,7,9"

These tracknumbers must be separated from each other using a comma. Whitespaces are not allowed.

With trackstrings, passing the wanted tracks to a function is really easy.



^ Datatypes: MediaItemArray

Many functions allow you to process through multiple mediaitems. To be able to pass multiple mediaitems at once to a function, I added the datatype MediaItemArray.

A MediaItemArray is an array with many MediaItems, indexed by an integer.
Examplecode:

        -- get the MediaItems
        MediaItem1=reaper.GetMediaItem(0,0)
        MediaItem2=reaper.GetMediaItem(0,1)
        MediaItem3=reaper.GetMediaItem(0,2)
        MediaItem4=reaper.GetMediaItem(0,3)
        
        -- create and fill the MediaItemArray
        MediaItemArray={}
        MediaItemArray[1]=MediaItem1
        MediaItemArray[2]=MediaItem2
        MediaItemArray[3]=MediaItem3
        MediaItemArray[4]=MediaItem4

MediaItemArrays will be read, until an index of the MediaItemArray is nil. In the above example, MediaItemArray[5] would be nil and therefore seen as the end of the array, even if there's a MediaItemArray[6]!

With MediaItemArray, passing the wanted MediaItems to a function is really easy.



^ Datatypes: MediaItemStateChunkArray

Many functions allow you to process through multiple mediaitem-statechunks. To be able to pass multiple mediaitem-statechunks at once to a function, I added the datatype MediaItemStateChunkArray.

A MediaItemStateChunkArray is an array with many MediaItemStateChunks, indexed by an integer.
Examplecode:

        -- get the MediaItems
        MediaItem1=reaper.GetMediaItem(0,0)
        MediaItem2=reaper.GetMediaItem(0,1)
        MediaItem3=reaper.GetMediaItem(0,2)
        MediaItem4=reaper.GetMediaItem(0,3)
        
        -- get the MediaItemStateChunks
        StateChunk1=reaper.GetItemStateChunk(MediaItem1, "", false)
        StateChunk2=reaper.GetItemStateChunk(MediaItem2, "", false)
        StateChunk3=reaper.GetItemStateChunk(MediaItem3, "", false)
        StateChunk4=reaper.GetItemStateChunk(MediaItem4, "", false)
        
        -- create and fill the MediaItemStateChunkArray
        MediaItemStateChunkArray={}
        MediaItemStateChunkArray[1]=StateChunk1
        MediaItemStateChunkArray[2]=StateChunk2
        MediaItemStateChunkArray[3]=StateChunk3
        MediaItemStateChunkArray[4]=StateChunk4

MediaItemStateChunkArrays will be read, until an index of the MediaItemStateChunkArray is nil. In the above example, MediaItemStateChunkArray[5] would be nil and therefore seen as the end of the array, even if there's a MediaItemStateChunkArray[6]!

With MediaItemStateChunkArrays, passing the wanted MediaItemStateChunks to a function is really easy.



^ Datatypes: EnvelopePointObject

When working with envelope-points, handling all the attributes such an envelope-point is quite unhandy. So I introduced the datatype EnvelopePointObject, containing all it's attributes.

An EnvelopePointObject is an array with all attributes an envelope-point has.
Examplecode:

        MediaTrack=reaper.GetTrack(0,0) -- get MediaTrack
        TrackEnvelope=reaper.GetTrackEnvelope(MediaTrack, 0) -- get TrackEnvelope
        retval, EnvelopePointObject = ultraschall.CreateEnvelopePointObject(TrackEnvelope, 1, 20, 10, 0, 0, false) -- create EnvelopePointObject

An EnvelopePointObject is an array with all the attributes of an envelope-point, containing the following values:

        EnvelopePointObject[1] - Trackenvelope; The TrackEnvelope-object, in which the point lies
        EnvelopePointObject[2] - integer; Envelope-idx, with 0 for the first envelope-point; 1, for the second, etc
        EnvelopePointObject[3] - number; the time in seconds
        EnvelopePointObject[4] - number; the raw value of the envelope-point
        EnvelopePointObject[5] - integer; the shape of the envelope-point, with 
                                               0 - Linear
                                               1 - Square
                                               2 - Slow start/end
                                               3 - Fast start
                                               4 - Fast end
                                               5 - Bezier
        EnvelopePointObject[6] - number; tension of the envelope-point; -1 to 1; 0 for no tension
        EnvelopePointObject[7] - boolean; if the envelope-point is selected(true) or not(false)
        EnvelopePointObject[8] - number; dBValue converted from value

EnvelopePointObjects make handling of envelope-points and it's attributes much easier.



^ Datatypes: EnvelopePointArray

When working with multiple EnvelopePointObjects, I introduced the datatype EnvelopePointArray, containing multiple EnvelopePointObjects.

An EnvelopePointArray is an array with multiple EnvelopePointObjects, indexed by an integer.
Examplecode:

        -- create EnvelopePointObjects
        MediaTrack=reaper.GetTrack(0,0) -- get MediaTrack
        TrackEnvelope=reaper.GetTrackEnvelope(MediaTrack, 0) -- get TrackEnvelope
        retval, EnvelopePointObject1 = ultraschall.CreateEnvelopePointObject(TrackEnvelope, 1, 20, 10, 0, 0, false) -- create EnvelopePointObject1
        retval, EnvelopePointObject2 = ultraschall.CreateEnvelopePointObject(TrackEnvelope, 1, 20, 10, 0, 0, false) -- create EnvelopePointObject2
        retval, EnvelopePointObject3 = ultraschall.CreateEnvelopePointObject(TrackEnvelope, 1, 20, 10, 0, 0, false) -- create EnvelopePointObject3
        
        -- creating EnvelopePointArray
        EnvelopePointArray={}
        EnvelopePointArray[1]=EnvelopePointObject1
        EnvelopePointArray[2]=EnvelopePointObject2
        EnvelopePointArray[3]=EnvelopePointObject3

EnvelopePointArray makes passing multiple EnvelopePointObjects to functions much easier.



^ Datatypes: ColorTable

When having to work with many colors, like MediaTrack-colors or the colors set to individual MediaItems, one may wish to have a proper datastructure to store these colorvalues. For that, I introduce the ColorTable.

A ColorTable is an array containing the multiple colors, indexed by an integer.
Structure:

        ColorTable[index]["r"]=red color-value(0-255)
        ColorTable[index]["g"]=green color-value(0-255)
        ColorTable[index]["b"]=blue color-value(0-255)
        ColorTable[index]["gfxr"]=red color-value, useable by gfx-related-functions(0-1)
        ColorTable[index]["gfxg"]=green color-value, useable by gfx-related-functions(0-1)
        ColorTable[index]["gfxb"]=blue color-value, useable by gfx-related-functions(0-1)
        ColorTable[index]["nativecolor"]=the r-g-b-color-value converted to the native-color, used in your system
    

ColorTables can be used to store gradients, or temporary track-colors, or anything related to multiple colors.



^ Datatypes: RenderTable

When working with rendering, you have to deal with dozens of potential options. I tried to add all these options as parameters into the rendering-functions but, I failed miserably.
This is mostly because, there are so many.

So I added a new construct, called: the RenderTable.

A Rendertable is a table, which stores all rendersettings you can set in Reaper. And with all, I mean all!

Important: You should never create RenderTables "by hand" but rather use my functions for that. That way your RenderTables will stay valid when I add new features to them in the future. You can, however, alter them, after you've created them with my functions.

Structure:

  RenderTable["AddToProj"] - Add rendered items to new tracks in project-checkbox; true, checked; false, unchecked
  RenderTable["Brickwall_Limiter_Enabled"] - true, brickwall limiting is enabled; false, brickwall limiting is disabled            
  RenderTable["Brickwall_Limiter_Method"] - brickwall-limiting-mode; 1, peak; 2, true peak
  RenderTable["Brickwall_Limiter_Target"] - the volume of the brickwall-limit
  RenderTable["Bounds"] - 0, Custom time range; 
                          1, Entire project; 
                          2, Time selection; 
                          3, Project regions; 
                          4, Selected Media Items(in combination with Source 32); 
                          5, Selected regions
                          6, Razor edit areas
                          7, All project markers
                          8, Selected markers
  RenderTable["Channels"] - the number of channels in the rendered file; 
                            1, mono; 
                            2, stereo; 
                            higher, the number of channels
  RenderTable["CloseAfterRender"] - true, closes rendering to file-dialog after render; false, doesn't close it
  RenderTable["Dither"] - &1, dither master mix; 
                          &2, noise shaping master mix; 
                          &4, dither stems; 
                          &8, dither noise shaping stems
  RenderTable["EmbedMetaData"] - Embed metadata; true, checked; false, unchecked
  RenderTable["EmbedStretchMarkers"] - Embed stretch markers/transient guides; true, checked; false, unchecked
  RenderTable["EmbedTakeMarkers"] - Embed Take markers; true, checked; false, unchecked                        
  RenderTable["Enable2ndPassRender"] - true, 2nd pass render is enabled; false, 2nd pass render is disabled
  RenderTable["Endposition"] - the endposition of the rendering selection in seconds            
  RenderTable["FadeIn_Enabled"] - true, fade-in is enabled; false, fade-in is disabled
  RenderTable["FadeIn"] - the fade-in-time in seconds
  RenderTable["FadeIn_Shape"] - the fade-in-shape
                         - 0, Linear fade in
                         - 1, Inverted quadratic fade in
                         - 2, Quadratic fade in
                         - 3, Inverted quartic fade in
                         - 4, Quartic fade in
                         - 5, Cosine S-curve fade in
                         - 6, Quartic S-curve fade in
  RenderTable["FadeOut_Enabled"] - true, fade-out is enabled; false, fade-out is disabled
  RenderTable["FadeOut"] - the fade-out time in seconds
  RenderTable["FadeOut_Shape"] - the fade-out-shape
                         - 0, Linear fade in
                         - 1, Inverted quadratic fade in
                         - 2, Quadratic fade in
                         - 3, Inverted quartic fade in
                         - 4, Quartic fade in
                         - 5, Cosine S-curve fade in
                         - 6, Quartic S-curve fade in
  RenderTable["MultiChannelFiles"] - Multichannel tracks to multichannel files-checkbox; true, checked; false, unchecked            
  RenderTable["Normalize_Enabled"] - true, normalization enabled; false, normalization not enabled
  RenderTable["Normalize_Method"] - the normalize-method-dropdownlist
                 0, LUFS-I
                 1, RMS-I
                 2, Peak
                 3, True Peak
                 4, LUFS-M max
                 5, LUFS-S max
  RenderTable["Normalize_Only_Files_Too_Loud"] - Only normalize files that are too loud,checkbox
                                               - true, checkbox checked
                                               - false, checkbox unchecked
  RenderTable["Normalize_Stems_to_Master_Target"] - true, normalize-stems to master target(common gain to stems)
                                                    false, normalize each file individually
  RenderTable["Normalize_Target"] - the normalize-target as dB-value
  RenderTable["NoSilentRender"] - Do not render files that are likely silent-checkbox; true, checked; false, unchecked
  RenderTable["OfflineOnlineRendering"] - Offline/Online rendering-dropdownlist; 
                                          0, Full-speed Offline
                                          1, 1x Offline
                                          2, Online Render
                                          3, Online Render(Idle)
                                          4, Offline Render(Idle)
  RenderTable["OnlyMonoMedia"] - Tracks with only mono media to mono files-checkbox; true, checked; false, unchecked
  RenderTable["OnlyChannelsSentToParent"] - true, option is checked; false, option is unchecked
  RenderTable["ProjectSampleRateFXProcessing"] - Use project sample rate for mixing and FX/synth processing-checkbox; 
                                                 true, checked; false, unchecked
  RenderTable["RenderFile"] - the contents of the Directory-inputbox of the Render to File-dialog
  RenderTable["RenderPattern"] - the render pattern as input into the File name-inputbox of the Render to File-dialog
  RenderTable["RenderQueueDelay"] - Delay queued render to allow samples to load-checkbox; true, checked; false, unchecked
  RenderTable["RenderQueueDelaySeconds"] - the amount of seconds for the render-queue-delay
  RenderTable["RenderResample"] - Resample mode-dropdownlist; 
                                      0, Sinc Interpolation: 64pt (medium quality)
                                      1, Linear Interpolation: (low quality)
                                      2, Point Sampling (lowest quality, retro)
                                      3, Sinc Interpolation: 192pt
                                      4, Sinc Interpolation: 384pt
                                      5, Linear Interpolation + IIR
                                      6, Linear Interpolation + IIRx2
                                      7, Sinc Interpolation: 16pt
                                      8, Sinc Interpolation: 512pt(slow)
                                      9, Sinc Interpolation: 768pt(very slow)
                                      10, r8brain free (highest quality, fast)
  RenderTable["RenderStems_Prefader"] - true, option is checked; false, option is unchecked
  RenderTable["RenderString"] -  the render-cfg-string, that holds all settings of the currently set render-output-format 
                                 as BASE64 string
  RenderTable["RenderString2"] - the render-cfg-string, that holds all settings of the currently set secondary-render-output-format 
                                 as BASE64 string
  RenderTable["RenderTable"]=true - signals, this is a valid render-table
  RenderTable["SampleRate"] - the samplerate of the rendered file(s)
  RenderTable["SaveCopyOfProject"] - the "Save copy of project to outfile.wav.RPP"-checkbox; true, checked; false, unchecked
  RenderTable["SilentlyIncrementFilename"] - Silently increment filenames to avoid overwriting-checkbox; true, checked; false, unchecked
  RenderTable["Source"] - 0, Master mix
                          1, Master mix + stems; 
                          3, Stems (selected tracks); 
                          8, Region render matrix; 
                          32, Selected media items; 
                          64, selected media items via master; 
                          128, selected tracks via master
                          136, Region render matrix via master
                          4096, Razor edit areas
                          4224, Razor edit areas via master
  RenderTable["Startposition"] - the startposition of the rendering selection in seconds
  RenderTable["TailFlag"] - in which bounds is the Tail-checkbox checked
                            &1, custom time bounds; 
                            &2, entire project; 
                            &4, time selection; 
                            &8, all project regions; 
                            &16, selected media items; 
                            &32, selected project regions
                            &64, razor edit areas
  RenderTable["TailMS"] - the amount of milliseconds of the tail
    

This rendertable can be used by, e.g. RenderProject_RenderTable



^ Datatypes: Checking Datatypes

When you work with data of different sources but need to work with data of a specific type, it would be handy to have ways of checking, whether a certain variable has data of a specific type.
Lua provides ways of checking for valid datatypes(type() and math.type()) as well as Reaper(ValidatePtr() and ValidatePtr2()).

However, sometimes you want to have one function to check them all, and all these functions do not include Ultraschall-API-specific datatypes, so I added some stuff for that.

  1. type
    works like Lua's own type-function, but checks for Reaper's own datatypes as well, like:

            Lua: nil, number: integer, number: float, boolean, string, function, table, thread, userdata,  
            Reaper: ReaProject, MediaItem, MediaItem_Take, MediaTrack, TrackEnvelope, AudioAccessor, joystick_device, PCM_source  
            userdata: will be shown, if object isn't of any known type  

    to get of which type a variable is, just use

             datatype_of_variable = ultraschall.type(variable)

    where the return-value datatype_of_variable will hold a string describing the type of the variable.

    Due some API-restrictions, SWS-specific datatypes are not (yet) supported.

  2. Ultraschall-API specific or other Reaper-datatypes
    To check for Ultraschall-API specific or other Reaper-datatypes, you can use the following functions:

    Ultraschall-API-specific:

    Other Reaper-datatypes:

    They will be part of ultraschall.type() someday.

  3. other types of data
    These aren't datatypes, but you may want to check them for validity as well



^ API-Variables

When working with the Ultraschall-API or general programming in Reaper, some additional things may or may not be helpful to know.
So I added some API-variables, like:

  1. Script_Path - which contains the current path to Reaper's-scripts-folder
  2. Separator - which contains the correct separator for paths; on Windows it is \ on Mac and Linux it is /
  3. StartTime - contains the starting time of the Ultraschall-API as used in the current script. That means, the starting time of the now running script.
  4. Api_Path - the current path to the Ultraschall-API folder.
  5. Api_InstallPath - an API-variable that contains the path to the install-folder of the Ultraschall-API

These can be accessed using:

            scriptpath = ultraschall.Script_Path
    

which would put the current-scriptpath to the variable scriptpath.

You could change them as well, but that would be pointless.



^ Rendering: Introduction

The Ultraschall-API provides functions for rendering your projects, without having to use the Rendering-dialog of Reaper. This gives you a wide range of possibilities to customize your rendering-needs.

There are two basic ways to render a file.

The easy way:

  1. creating a renderstring with all format-specific-settings, using the accompanying functions:
    CreateRenderCFG_AIFF, CreateRenderCFG_AudioCD, CreateRenderCFG_DDP ,CreateRenderCFG_FLAC, CreateRenderCFG_M4AMAC, CreateRenderCFG_MP3ABR,CreateRenderCFG_MP3CBR, CreateRenderCFG_MP3MaxQuality, CreateRenderCFG_MP3VBR, CreateRenderCFG_OGG, CreateRenderCFG_Opus, CreateRenderCFG_WAV, CreateRenderCFG_WAVPACK CreateRenderCFG_AVI_Video, CreateRenderCFG_GIF, CreateRenderCFG_LCF, CreateRenderCFG_MKV_Video, CreateRenderCFG_MOVMAC_Video, CreateRenderCFG_MP4MAC_Video, CreateRenderCFG_QTMOVMP4_Video, CreateRenderCFG_WebMVideo, CreateRenderCFG_CAF, CreateRenderCFG_FLV_Video], CreateRenderCFG_MPEG1_Video, CreateRenderCFG_MPEG2_Video

  2. passing the render-string to one of the Render-functions
    RenderProject, RenderProject_Regions

So the following code should render the current project into an MP3 with a Constant Bitrate of 128kbps.

       -- create Render-string
       render_cfg_string = ultraschall.CreateRenderCFG_MP3CBR(11, 2)
       
       -- Pass this Render-string to the rendering-function
       retval, rendered_files_count, rendered_files_MediaItemStateChunk, rendered_files_array 
                      = ultraschall.RenderProject(nil, "c:\\exportfile.mp3", 0, -1, false, false, false, render_cfg_string, nil)

The full control way:

  1. creating a RenderTable, which holds all settings you can set for the rendering-process, using one of the following functions:
    CreateNewRenderTable(new rendertable), GetRenderTable_Project(create from the current project), GetRenderTable_ProjectFile(create from an rpp-projectfile), GetRenderPreset_RenderTable(create from render-preset)

  2. optionally exchanging the renderstring(which holds all format-specific-settings) in the RenderTable, using the accompanying functions:
    CreateRenderCFG_AIFF, CreateRenderCFG_AudioCD, CreateRenderCFG_DDP ,CreateRenderCFG_FLAC, CreateRenderCFG_M4AMAC, CreateRenderCFG_MP3ABR,CreateRenderCFG_MP3CBR, CreateRenderCFG_MP3MaxQuality, CreateRenderCFG_MP3VBR, CreateRenderCFG_OGG, CreateRenderCFG_Opus, CreateRenderCFG_WAV, CreateRenderCFG_WAVPACK CreateRenderCFG_AVI_Video, CreateRenderCFG_GIF, CreateRenderCFG_LCF, CreateRenderCFG_MKV_Video, CreateRenderCFG_MOVMAC_Video, CreateRenderCFG_MP4MAC_Video, CreateRenderCFG_QTMOVMP4_Video, CreateRenderCFG_WebMVideo, CreateRenderCFG_CAF, CreateRenderCFG_FLV_Video], CreateRenderCFG_MPEG1_Video, CreateRenderCFG_MPEG2_Video

  3. Render the project, using the function:
    RenderProject_RenderTable

So the following code should render the current project, using the settings stored in a projectfile "c:\testproject.rpp", into an MP3 with a Constant Bitrate of 128kbps.

    -- create RenderTable
    RenderTable = ultraschall.GetRenderTable_ProjectFile("c:\\testproject.rpp")
    
    -- create Render-string
    RenderTable["RenderString"] = ultraschall.CreateRenderCFG_MP3CBR(11, 2)
    
    -- Render project using that RenderTable
    rendered_files_count, rendered_files_MediaItemStateChunk, rendered_files_array 
                                                = ultraschall.RenderProject_RenderTable(nil, RenderTable, false, false, false)

Read the accompanying documentation-entries for CreateRenderCFG_MP3CBR and RenderProject for more details and/or the following chapters.



^ Rendering: About Renderstrings

Render-strings(or render_cfg, as they are named in Reaper) are strings, that contain all settings for a specific Render-Output-format, as MP3, WAV, AIF, FLAC, etc.
They are usually stored into RPP-projectfiles and are quite cryptic(for those of you, who know what that means: BASE64-encoded).
These are essential for rendering a project using the provided rendering-functions, so the Ultraschall-API provides functions who create or analyze such render-strings.

All these functions start with CreateRenderCFG_ in their name, so if you are looking for a specific format, use CreateRenderCFG_audioformat (like CreateRenderCFG_FLAC, etc).
The parameters of these functions represent all format-options as you are used from the Render-dialog. They should be fairly complete, though some formats (Video, OGG) are limited with some of their options(fps only supported up to 2000.00 in Ultraschall-API for API-limitations).

Example for FLAC:

        render_string = ultraschall.CreateRenderCFG_FLAC(integer BitDepth, integer EncSpeed)

creates the render-string for the FLAC-fileformat. Just provide the BitDepth(e.g 0 for 24 Bit) and the encoding-speed(EncSpeed) (e.g 5 for the default encoding speed).
Have a look into the functions-reference to get the possible values for these parameters.

        render_string = ultraschall.CreateRenderCFG_FLAC(0, 5)

This will create a render-string for FLAC with 24bit-depth and the encoding-speed of 5(which is the default-setting in the render-dialog).

This renderstring can then be passed over to SetProject_RenderCFG to set it into a project-file, or to RenderProject to render a projectfile using the format-settings in the Render-string or put into RenderTables, that can be used for the rendering function RenderProject_RenderTable.

The following render-string-functions are available in Ultraschall-API:

CreateRenderCFG_AIFF, CreateRenderCFG_AudioCD, CreateRenderCFG_DDP ,CreateRenderCFG_FLAC, CreateRenderCFG_M4AMAC, CreateRenderCFG_MP3ABR,CreateRenderCFG_MP3CBR, CreateRenderCFG_MP3MaxQuality, CreateRenderCFG_MP3VBR, CreateRenderCFG_OGG, CreateRenderCFG_Opus, CreateRenderCFG_WAV, CreateRenderCFG_WAVPACK, CreateRenderCFG_AVI_Video, CreateRenderCFG_GIF, CreateRenderCFG_LCF, CreateRenderCFG_MKV_Video, CreateRenderCFG_MOVMAC_Video, CreateRenderCFG_MP4MAC_Video, CreateRenderCFG_QTMOVMP4_Video, CreateRenderCFG_WebMVideo CreateRenderCFG_CAF, CreateRenderCFG_FLV_Video], CreateRenderCFG_MPEG1_Video, CreateRenderCFG_MPEG2_Video

If you have already a Base64-encoded-renderstring but would love to know, which settings it holds, you can analyze them as well. First, you need to determine, which format is stored in this render-string, using:

GetOutputFormat_RenderCfg

        string outputformat, string renderstring = ultraschall.GetOutputFormat_RenderCfg(string Renderstring, optional ReaProject ReaProject)
    

This allows you to get the output-format stored in a Renderstring. If you set it to nil, it will return the output-format set in the current active project.
You can also pass a ReaProject as optional second parameter, so the function will return the output-format of it as well.
To make life easier, it also returns the analysed render-string. So if you want to know the renderoutput-format of a ReaProject, you also get the render-string to work on further.

Now that we know, of which format the renderstring is, we can get the settings stored in it. So let's assume, it is a renderstring of the format FLAC, you can use the following function to analyse it's settings:

GetRenderCFG_Settings_FLAC

        integer encoding_depth, integer compression = ultraschall.GetRenderCFG_Settings_FLAC(string rendercfg)

Pass to this function the renderstring of the flac-format and it will return the settings for encoding_depth and compression, as settable in the Render to File-dialog.

The following functions are available for analysing the render-strings in all kinds of formats:

GetRenderCFG_Settings_AIFF, GetRenderCFG_Settings_AudioCD, GetRenderCFG_Settings_DDP, GetRenderCFG_Settings_FLAC, GetRenderCFG_Settings_M4AMac, GetRenderCFG_Settings_MP3, GetRenderCFG_Settings_MP3ABR, GetRenderCFG_Settings_MP3CBR, GetRenderCFG_Settings_MP3MaxQuality, GetRenderCFG_Settings_MP3VBR, GetRenderCFG_Settings_OGG, GetRenderCFG_Settings_OPUS, GetRenderCFG_Settings_WAV, GetRenderCFG_Settings_WAVPACK, GetRenderCFG_Settings_AVI_Video, GetRenderCFG_Settings_GIF, GetRenderCFG_Settings_LCF, GetRenderCFG_Settings_MKV_Video, GetRenderCFG_Settings_MOVMac_Video, GetRenderCFG_Settings_MP4Mac_Video, GetRenderCFG_Settings_QTMOVMP4_Video, GetRenderCFG_Settings_WebMVideo GetRenderCFG_CAF, GetRenderCFG_FLV_Video], GetRenderCFG_MPEG1_Video, GetRenderCFG_MPEG2_Video



^ Rendering: About RenderTables

RenderTables are tables, who hold all settings, that one can set in Reaper for the rendering-process. Most of the stuff about them, I've told already in the chapter RenderTables.
The main goal behind them is, to make passing the dozens of render-settings much easier and comfortable.

But even then, creating one is very painful, so I made functions, who create valid RenderTables.

It is always a good idea to use the US-API-functions for creating RenderTables, as I may add more settings into RenderTables(when Reaper gets new ones). If you would create a RenderTable by hand without my functions, you not only miss out the new features. In fact, your old RenderTable would be seen as invalid by the API, as it's missing new settings.
So be advised to always use the CreateRenderTable-functions from the API. They will always create valid RenderTables, setting unset settings to useful defaults.
That way, your old scripts will not break, even if you don't update them to use new settings.
After a RenderTable is created, you can safely alter it's settings to valid values, but creating them must be done by US-API's functions.

I wrote numerous functions to create and work with RenderTables.

Create new RenderTable:

Get render-settings a RenderTable:

Apply settings from a RenderTable:

Check for validity:

With that, you can interoperate with RenderTables and render-settings from/to projects, projectfiles and presets in any way you wish.



^ Rendering: About Rendering-functions

The rendering-functions let you render a project, either a stored rpp-project-file or the currently opened one.
They provide you with a wide range of functionality, so the rendering process should be quite close to Reaper's "official"-own-process using the Render to File-dialog.
They also return the filenames of the rendered files, as well as MediaItemStateChunks of all rendered files, so you can easily import them into your project.

There are currently three different render-functions available:

  1. RenderProject - with this one, you can render a whole project or just timeselection or from startposition to endposition in seconds
  2. RenderProject_Regions - with this one, you can render specific regions of a project
  3. RenderProject_RenderTable - with this one, you can render a project using a RenderTable, which allows you full control over all render-settings

RenderProject and RenderProject_Regions

are basically the same, with the only difference, that with the first one you can set a specific startposition and endposition, while the second one, you set it to a region-number.
That said, a lot of the parameters are the same.

         projectfilename_with_path - the projectfile with path, that you want to render. Set it to nil, if you want to render the 
                                     currently opened project
         renderfilename_with_path  - the filename with path of the output-file
         overwrite_without_asking  - if you want to overwrite already existing outputfiles, set this to true; else, set it to false

The following two are directly connected to some of Reaper's dialogs:

         renderclosewhendone - the render-progress-window, that is shown during the actual rendering can be closed automatically after  
                               rendering is finished(there's a checkbox in that window to set this). Set this to true to automatically 
                               close it; set it to false to keep it shown; set it to nil and it will use the setting the user set with 
                               the checkbox
         filenameincrease    - another of Reaper's dialogs, that will pop up, when an output-file already exists and overwrite_without_asking 
                               is set to false. It will ask you to automatically increase the filename with a number to prevent accidental 
                               overwriting. Set this to true to automatically increase filename; 
                               set it to false to show the dialog; set it to nil to use the settings the user chose

The last parameter is the place for your render-string:

         rendercfg           - the renderstring, as created using a CreateRenderCFG_XXX-function, as described in 
                               the chapter "Rendering: About Renderstrings"
                               if you omit it or set it to nil, it will use the format-settings already set in the project or projectfile

         rendercfg2         - the same like rendercfg, but for the secondary render-format(see Render to File-dialog for what that means)
         

It also returns some interesting return-values:

        retval                   - 0, if rendering was successful; -1, in case of an error(like invalid parametervalues or user aborted rendering)  
        renderfilecount          - the number of rendered files. Usually 1, but can be higher, when rendering stems as well  
        MediaItemStateChunkArray - an array with MediaItemStateChunks of the rendered projects, ready to include into a project of your choice  
                                   the first entry is for the master-track-rendered-file  
        Filearray                - an array with filenames-with-path of all rendered files, with the first entry being the one of  
                                   the master-track-rendered-file

RenderProject_RenderTable

works different and uses a RenderTable, which stores all rendersettings(and I mean: ALL).
This gives you full control over the rendering-process and you can customize it in anyway you want.

You can set the following parameters:

        projectfilename_with_path - You can pass here the filename+path of the project that you want to render.  
                                    To render the current active project, set this parameter to nil.  
        RenderTable               - a RenderTable, which holds all render-settings you want to set; to use the already set settings, set this to nil  
                                    This RenderTable allows you to control all render-settings.  
        AddToProj                 - when rendering the current project, shall the rendered files be added to the project(true) or not(false)?  

The following two are directly connected to some of Reaper's dialogs:

        CloseAfterRender          - the render-progress-window, that is shown during the actual rendering can be closed automatically after  
                                    rendering is finished(there's a checkbox in that window to set this). Set this to true to automatically  
                                    close it; set it to false to keep it shown; set it to nil and it will use the setting the user set with  
                                    the checkbox  
        SilentlyIncrementFilename - another of Reaper's dialogs, that will pop up, when an output-file already exists and overwrite_without_asking   
                                    is set to false. It will ask you to automatically increase the filename with a number to prevent accidental  
                                    overwriting. Set this to true to automatically increase filename;   
                                    set it to false to show the dialog; set it to nil to use the settings the user chose  

The return-values should be quite familiar to you from the other render-functions:

        renderfilecount          - the number of rendered files. Usually 1, but can be higher, when rendering stems as well  
        MediaItemStateChunkArray - an array with MediaItemStateChunks of the rendered projects, ready to include into a project of your choice  
                                   the first entry is for the master-track-rendered-file  
        Filearray                - an array with filenames-with-path of all rendered files, with the first entry being the one of  
                                   the master-track-rendered-file

With that, you should be able to successfully render your project and do some neat stuff afterwards.

For more enhanced customization of projectfiles for Rendering, see Rendering: Change more render-settings in projectfiles.



^ Rendering: Change more render-settings in projectfiles

Sometimes, you want to alter Render-Settings in a projectfile by hand, without using RenderTables for that. So, the API has many useful functions as well, starting with SetProject_ You can alter a projectfile using the following Ultraschall-Framework-functions, that represent certain elements from Reaper's Render-Dialog:

If you want to alter the currently opened project in its rpp-projectfile, you need to save it first. After that, use:

        retval, projectfilename_with_path = reaper.EnumProjects(-1,"")

to get the projectfilename_with_path of the current project as returnvalue. Use projectfilename_with_path as parameter for the SetProject-functions needed for you alterations.

The functions are just a small selection of the functions to alter project-files, as provided by the Ultraschall-API. Browse through the functions in the "Project-Files"-section of the index of the Ultraschall-API-Functions-Reference for many more of them.



^ Rendering: Render Presets

Reaper has the concept of render-presets, who hold all render-settings you can set in the Render to File-dialog and you want to store for quick retrieval.
These can be reused within the Render to File-dialog, when clicking the Preset-button.
Render-presets are split into two main parts: Bounds and Options and Formats, who each hold parts of the full render-settings-package.

Options and Format holds the following settings:
OutputAndFormatPresetname, SampleRate, Channels, offline_online_dropdownlist, useprojectsamplerate_checkbox, resamplemode_dropdownlist, various_checkboxes and rendercfg

Bounds holds the following settings: BoundsPresetname, bounds_dropdownlist, start_position, endposition, source_dropdownlist_and_checkboxes, unknown, outputfilename_renderpattern, tail_checkbox

That means, to get the full render-settings as a Rendertable, you need to get both, the settings for a Renderformat-preset and a Bounds-preset.
And to make that easier, I added numerous functions:

GetRenderPreset_Names

        integer bounds_presets, table bounds_names, integer options_format_presets, table options_format_names, 
                                      integer both_presets, table both_names = ultraschall.GetRenderPreset_Names()

This gets all Bounds-presetnames and all Options and Format-presetnames. These are important, as you need them to for getting them as RenderTable using GetRenderPreset_RenderTable.
As a general rule: RenderPresets who were stored using "All Settings" in the Render-Presets-menu of the Render to File-dialog, share the same presetname.

If you know the preset-names, you can create a RenderTable, that stores all render-settings stored in their presets.

GetRenderPreset_RenderTable

        RenderTable RenderTable = ultraschall.GetRenderPreset_RenderTable(string Bounds_Name, string Options_and_Format_Name)

This function returns a RenderTable from render-presets. You simply pass to the function the Bounds-presetname and the options and format-presetname and it will take these settings and put them into a RenderTable.

Some settings aren't stored in Render-Presets and therefore will get default values:
TailMS=0, SilentlyIncrementFilename=false, AddToProj=false, SaveCopyOfProject=false, RenderQueueDelay=false, RenderQueueDelaySeconds=false

This RenderTable can be used for the RenderProject_RenderTable-function.

Using already existing render-presets is nice, but being able to change them is even nicer. Once they are added/set, they are available after (re-)opening the Render to File-dialog.

AddRenderPreset

        boolean retval = ultraschall.AddRenderPreset(string Bounds_Name, string Options_and_Format_Name, 
                                                                                  RenderTable RenderTable)

With this one, you can add a new render-setting, the usage is simple: pass to it a new Bounds-presetname, a new Options and Format-presetname and a RenderTable, which holds all render-settings. If you want to just store one of the render-presets, you pass nil to the other presetname. So if you want to add only a Bounds-preset, the functioncall could look like that:

        retval = ultraschall.AddRenderPreset("My new bounds-render-preset", nil, RenderTable)

The function will check, whether the chosen name is already in use and return false in that case.

SetRenderPreset

        boolean retval = ultraschall.SetRenderPreset(string Bounds_Name, string Options_and_Format_Name, 
                                                                                  RenderTable RenderTable)

As you can see, it basically works like the aforementioned AddRenderPreset. The only difference is that is checks, whether the preset-name does not exists. In that case, it returns false, otherwise it exchanges the render-preset-settings with the settings stored in the RenderTable.

Now we have get, add and set, so what is still missing? You're right: deleting them.

DeleteRenderPreset_Bounds

        boolean retval = ultraschall.DeleteRenderPreset_Bounds(string Bounds_Name)

This function deletes an already stored bounds-render-preset. Just pass the Bounds-presetname and it deletes it, if existing. If the preset does not exist, it returns false, otherwise it returns true.

DeleteRenderPreset_FormatOptions

        boolean retval = ultraschall.DeleteRenderPreset_FormatOptions(string Options_and_Format_Name)

Basically like DeleteRenderPreset_Bounds, but deletes an Options and Format-renderpreset. Just pass the Options and Format-presetname and it deletes it from the presets. If the preset does not exist, it returns false, otherwise it returns true.

This should give you a lot of control over render-presets.



^ Rendering: Render Queue

Another way to render projects is using the RenderQueue, which is often practical. For that, I included some functions to deal with the render-queue:

GetRenderQueuedProjects

        integer Filecount, array Filearray = ultraschall.GetRenderQueuedProjects()

This function allows you to get the projects currently stored in RenderQueue and the number of queued projects.
The order of the projects in FileArray represent the index, which can be used in RenderProject_RenderQueue, with the first project being index 1, the second being index 2, etc.
You can also render these projects using the other render-functions as well.

RenderProject_RenderQueue

        boolean retval = ultraschall.RenderProject_RenderQueue(integer index)

This renders a project in the render-queue. The index represents the project within the render-queue, with 1 for the first.
To know, which queued project has which index, use GetRenderQueuedProjects.
If you set index to -1, it will attempt rendering all projects in the render-queue. Due API-limitations, this function can not return the rendered files(I hope I can circumvent this limitation at some point).

AddProjectFileToRenderQueue

        boolean retval = ultraschall.AddProjectFileToRenderQueue(string projectfilename_with_path)

If you have a project, which you want to add to the RenderQueue, but is not currently opened yet, you can use this function.
Just pass to it the projectfilename+path of the project, that you want to add to the RenderQueue and it will do what is needed for that.



^ Arrangeview Snapshots: Introduction

When working with big or complex projects:

Sometimes it's a good thing to have quick access to certain parts of the project, certain view-settings, zoom-factors. Arrangeview-snapshots are meant to help with that.

Arrangeview-Snapshots are snapshots that store the current position of the arrangeview as well as it's zoom-factor. You can decide, whether to store only the zoom-factor or the position. You can also give a short description to a ArrangeView-Snapshot, so you can store, what to expect from a certain snapshot.
They can be retrieved and the arrange-view can be set to these settings. That way, quick navigation through often accessed parts of the project is fast and easy.

Arrangeview-Snapshots are stored as ProjExtStates, which means, that the settings are stored in the project itself and can be retrieved the next time the project is loaded.

Due limitations with Reaper's own API, storing the vertical-scroll-position of the arrangeview isn't possible yet. This will change as soon as the limitation is raised from Reaper's own API.



^ Arrangeview Snapshots: How to store, retrieve, delete

If you want to store the current position and zoom-factor of the Arrangeview, you can use the function StoreArrangeviewSnapshot.

It accepts the following parameters:

        slot        - the slot for the snapshot, which must be an integer. The function will overwrite an already existing snapshot. To prevent that,
                      use ultraschall.IsValidArrangeviewSnapshot() to check, if it's already existing.
        description - a short description, what the snapshot contains so you know, what to expect from it
        position    - set to true to store the startposition and endposition of the arrangeview. Otherwise(false), 
                      it will only store the current horizontal zoom-factor
        vzoom       - set to true, if you want to store the vertical zoom-factor as well; set to false, if you don't want it to be stored.

If you want to retrieve the settings of a certain Arrange-View-Snapshot, you can use RetrieveArrangeviewSnapshot, which will return all settings from an Arrangeview-snapshot. The return-values of RetrieveArrangeviewSnapshot basically work the same as the parameters of StoreArrangeviewSnapshot.

If you want to check, whether a slot is already used, you can use IsValidArrangeviewSnapshot, which will return true in that case and false, if the slot is unused.

To delete a certain slot, just use DeleteArrangeviewSnapshot.



^ Arrangeview Snapshots: How to restore

When having stored an Arrangeview-Snapshot into a slot, you certainly want to restore it at one point. For that, use RestoreArrangeviewSnapshot.

This function let's you restore an earlier arrange-view completely, but also allows you to individually set, what you want to restore, using the parameters:

        slot        - is the Arrangeview-Snapshot you want to restore

the other parameters are optional, means, if you omit them or set them to nil, they will restore the setting from the snapshot or use a default setting

        position    - true, restore the start and endposition of the arrange-view; false, just restore the horizontal-zoom-factor
        vzoom       - set to true to restore the vertical zoom-factor or set to false to keep the current one
        hcentermode - this decides, what shall be in the center of the arrangeview, when position is set to false, with several options possible:
                       nil, keeps center of view in the center during zoom(default setting)
                        -1, default selection, as set in the reaper-prefs, 
                         0, edit-cursor or playcursor(if it's in the current zoomfactor of the view during playback/recording) in center,
                         1, keeps edit-cursor in center of zoom
                         2, keeps center of view in the center during zoom
                         3, keeps in center of zoom, what is beneath the mousecursor
         

This should give you full control in what to restore from an Arrangeview-Snapshot and what to ignore.



^ Navigation: Introduction

When editing and postproducing a project, navigating through it is essential. For that, I added some functions that are not part of Reaper's own API, to help navigation, with functions for:

  1. more control about moving the playcursor and the editcursor
  2. jumping to the next/previous closest marker/regionedge/itemedge
  3. centering the view to several possible center-positions selectable(mousecursor, editcursor, playcursor)
  4. Followmode, aka autoscrolling

This should give you more control about programming faster and quicker navigation-capabilities.



^ Navigation: Move Play and Editcursor

The Ultraschall-API provides you with many functions regarding changing the position of the playcursor and the editcursor.
For that we have numerous functions:



^ Navigation: Go to markers, regionedges and itemedges

Markers and items provide you with much additional helpful information regarding the project. They also provide you with an additional information: useful positions to navigate through.
To make use of that, I added some functions for that:

  1. GetClosestPreviousMarker, GetClosestNextMarker - Get the previous/next closest marker at a given position
  2. GetClosestPreviousRegionEdge, GetClosestNextRegionEdge - Get the previous/next closest regionedge at a given position
  3. GetPreviousClosestItemEdge, GetNextClosestItemEdge - Get the previous/next closest itemedge at a given position
  4. GetClosestGoToPoints - get previous/next markers/regionedges/itemedges/projectstart/projectend from position, for those who need the full marker/region/item-position-package

Let's go into more detail, by examining GetClosestPreviousRegionEdge.

        number markerindex, number position, string markername, string edge_type = 
                                                    ultraschall.GetClosestPreviousRegionEdge(integer cursor_type, optional number time_position)

This function allows you to get, which is the previous closest region-edge-position(either the start or the end of a region) as seen from a given position. To set that position, you need to set the parameter cursor_type:

        0 - Edit Cursor,   
        1 - Play Cursor,   
        2 - Mouse Cursor, or  
        3 - Timeposition  

If you set it to 3, you can use the optional parameter time_position, with which you can set any position, at which you want to know the previous closest region-edge.

If you run that function, it will return the markerindex, which is the index of all markers in your project, the position at which the regionedge is located, the markername and the type of the edge, which is either "beg" or "end".

The Marker-functions (from 1) ) work the same, the Item-edge-functions (from 3) ) however have an additional parameter trackstring, with which you can set, from which tracks you want to get the next/previous closest item-edge-position.

The function GetClosestGoToPoints is the combination of all of these functions, which let's you decide fully, which edges/positions you want to check for.
It will also check, if the next/previous closest edge is the beginning or the end of the project.



^ Navigation: Center View

Sometimes it's a good idea to center the arrangeview to a certain point, may it be different points of interests within your project or just to get back to the playcursor/editcursor out of the view.

For that, I added the function CenterViewToCursor.
Let's have a look at it:

        ultraschall.CenterViewToCursor(integer cursortype, optional number position) 

It has two parameters, of which cursortype allows you to give the type of the cursor to center around:

        1 - change arrangeview with edit-cursor centered
        2 - change arrangeview with play-cursor centered
        3 - change arrangeview with mouse-cursor-position centered
        4 - centers arrangeview around the position given with parameter position

The second parameter position is an optional one and only used, if cursortype is set to 4. It allows you to give a specific position in seconds, which the arrangeview shall be centered around.

This functions only centers the given position/cursor-position to the arrangeview. It keeps the zoom-factor intact.



^ Navigation: Autoscroll and Followmode

Reaper allows you to set autoscrolling during playback/recording. It allows you to set it to continuous scrolling or to "page-wise"-scrolling.
This is quite flexible but hidden somewhat within the actions of Reaper. So I added the function ToggleScrollingDuringPlayback.
It turns on autoscrolling for playback and recording AND continuous scrolling.
Let's have a look at it:

        ultraschall.ToggleScrollingDuringPlayback(integer scrolling_switch, boolean move_editcursor, boolean goto_playcursor)

The parameter scrolling_switch allows you to turn on/off autoscrolling completely, that means, it will turn on autoscroll for playback and recording AND it sets autoscrolling to continuous scrolling.
The parameter move_edit_cursor allows you to set, if the editcursor shall be moved to the current playposition. This has an effect only, if scrolling_switch is set to 1(off).
The last parameter goto_playcursor allows you to change the view to the current playcursor-position, if you turn on autoscrolling. This has an effect only, if scrolling_switch is set to 0(off).

When running the function, it changes, if neccessary, the toggle-states of the actions

        41817(View: Continuous scrolling during playback),  
        40036(View: Toggle auto-view-scroll during playback) and   
        40262(View: Toggle auto-view-scroll while recording), 

which means, it sets if a certain autoscrolling behavior is turned on or not.
If you have your own custom actions toggling these actions, you probably shouldn't use this function. Otherwise it probably messes up your workflows.



^ Get/Set States for Project, Tracks and Items(including StateChunks)

One of the long-term-goals of the Ultraschall-API is full access to all states within projects, tracks, items, envelopes, including all states only available in StateChunks.
As of the current version, all track-states and many project and item-states are get and settable.

The functions all work after the same principle, with the first parameter being the object to get/set the state from/to (Projectfile, MediaItem-object, MediaTrack-object and TrackEnvelope-Object).

The last parameter, which is an optional one, can be a StateChunk-representation of the object, like ProjectStateChunk, MediaItemStateChunk, TrackStateChunk, TrackEnvelopeStateChunk. This optional parameter will only be seen, when the first parameter(for the object) is set to nil.
That way, you can decide, whether to use the original-object or the StateChunk, whatever works better for you.

In Setting-State-functions, it is basically the same: the first parameter the object and the last parameter is the optional StateChunk(when the first parameter is set to nil). The parameters inbetween set the individual settings for that state.

Let's have a look at two example functions GetProject_CursorPos and SetProject_CursorPos:
Get Project-State:

        number cursorpos = ultraschall.GetProject_CursorPos(string projectfilename_with_path, optional string ProjectStateChunk) 

The first parameter is the filename with path to the RPP-Projectfile. When this is set to nil, you can pass a ProjectStateChunk(which is basically the content of the RPP-projectfile).

Set Project-State:

        integer retval = ultraschall.SetProject_CursorPos(string projectfilename_with_path, number cursorpos, optional string ProjectStateChunk) 

The first parameter is the filename with path to the RPP-Projectfile. When this is set to nil, you can pass a ProjectStateChunk(which is basically the content of the RPP-projectfile). The parameter in the middle, cursorpos, can be set by you. That way, the cursorposition of the Project/ProjectStateChunk can be set to the position you prefer.

The same principle is for MediaTrack-states and MediaItem-states:

        -- the first parameter either MediaItem or nil, the last parameter can be a MediaItemStateChunk when first parameter is nil
        number length = ultraschall.GetItemLength(MediaItem MediaItem, optional string MediaItemStateChunk)
        string MediaItemStateChunk = ultraschall.SetItemLength(MediaItem MediaItem, integer length, string MediaItemStateChunk)
        
        -- the first parameter either MediaTrack or nil, the last parameter can be a TrackStateChunk when first parameter is nil
        integer lockedstate = ultraschall.GetTrackLockState(integer tracknumber, optional string TrackStateChunk)
        boolean retval, string TrackStateChunk = ultraschall.SetTrackLockState(integer tracknumber, integer LockedState, optional string TrackStateChunk) 

StateChunks can be gotten using reaper.GetTrackStateChunk(), reaper.SetTrackStateChunk(), reaper.GetItemStateChunk(), reaper.SetItemStateChunk(), reaper.GetEnvelopeStateChunk(), reaper.SetEnvelopeStateChunk().



^ MediaItems: Introduction

When working with MediaItems, it often was frustrating for me to code, how to get their MediaItem-objects. Especially when "mass-working" with dozens and more MediaItems.
So I wrote a set of functions to work with MediaItems more comfortably.

These functions include getting MediaItem-objects by time AND track, getting MediaItem-states, editing, inserting, manipulating, spectral edit, previewing, RippleCut, RippleInsert, SectionCut, working with locked, selected items, applying Reaper-actions to MediaItems, etc.

When working with masses of MediaItems, I either use the datatypes MediaItemArrays or MediaItemStateChunkArrays.
When passing over the tracks wanted, I use the datatype trackstring.

All this stuff should help you getting and manipulating MediaItems much more easier.

Let's begin with getting items by time(range) and tracks.



^ MediaItems: Getting MediaItems by Time and Tracks

Let's face it, when editing items of a project in Reaper, you either click on the items or select them in a 2D-way, by drawing a boundary box around the items of your choice or using a time-selection.
What you do by that is selecting the items by time. What you also do is, selecting the items by track, as your boundary box may go over several tracks. Or you use a track-selection by clicking on the tracks you want.
In either way, you select them in a 2Dimensional way. Not with Reaper's own API. Sure, you can somehow choose the MediaItems by track or by project, but you can't select them by multiple tracks. And certainly not by a time-range.

This was annoying for me, so to address this, I wrote the two functions GetMediaItemsAtPosition and GetAllMediaItemsBetween (my favorite ones in this api, I have to admit ;) ).

Let's have a closer look at them.

GetMediaItemsAtPosition:

        integer number_of_items, array MediaItemArray, array MediaItemStateChunkArray 
                                                       = ultraschall.GetMediaItemsAtPosition(number position, string trackstring)

This function gives you all items at position passed with parameter position and within the tracks given by parameter trackstring.
It returns the number of items, an array with all MediaItems and an array with all StateChunks of the MediaItems returned.
With this function, you can easily get the items from a certain position, without having to deal with looking into the MediaItem-objects for the correct time-position, or even have to care, where to get the corresponding tracks from an item.
This function does this for you.

But what, if you want to get the MediaItems inbetween a startingposition and an endposition?
For this, I wrote the function

GetAllMediaItemsBetween:

         integer count, array MediaItemArray, array MediaItemStateChunkArray = 
                     ultraschall.GetAllMediaItemsBetween(number startposition, number endposition, string trackstring, boolean inside) 

which basically returns the same things, as GetMediaItemsAtPosition. The difference lies in the parameters.
You can pass to the function a startposition and an endposition(which must be bigger than or equal startposition), trackstrings, which is a string with all tracks, separated by commas as well as inside as parameters. When you set inside to true, it will return only items that are completely within startposition and endposition. When setting inside to false, it will also return items, that are partially within start- and endposition, like items beginning before startposition or ending after endposition.

With these two functions, getting items is much, much easier than before.

The returned MediaItems, MediaItemArrays and MediaItemStateChunkArrays can then be passed over to other functions, who accept them, for "mass manipulation" of the MediaItems.

In addition to them, I also added some more functions for getting MediaItems, namely:



^ MediaItems: Splitting MediaItems by Time and Tracks

Getting MediaItems by time and tracks is cool. Editing them by time and tracks is even better. For that, I also added some functions:
The easiest ones are SplitMediaItems_Position and SplitItemsAtPositionFromArray
Let's have a look at:

SplitMediaItems_Position:

            boolean retval, array MediaItemArray = ultraschall.SplitMediaItems_Position(number position, string trackstring, boolean crossfade)

This splits all items at position, that are in the tracks given by parameter trackstring. If you want to have the items So if you want to split all items in tracks 1,3,4 at position 22, you type:

            retval, MediaItemArray = ultraschall.SplitMediaItems_Position(22, "1,3,4", false)

There's another parameter crossfade. If you have it set to true or nil and have Automatic-Crossfade enabled (Preferences -> Media Item Defaults -> Overlap and crossfade items when splitting, length), crossfade will be done at the split.
If you want to avoid that, set it to false and a normal split with fadein/fadeout will appear(if set in the preferences: Preferences -> Media Item Defaults -> Create automatic fade-in/fade-out for new items, length).

This function returns, if splitting was successful and the newly "created"-right-hand-split-items as an MediaItemAray.

SplitItemsAtPositionFromArray:

The function SplitItemsAtPositionFromArray works quite similar, but with the difference, that you don't give tracks, but items to the function, that shall be split at position.

            boolean retval, array MediaItemArray = ultraschall.SplitItemsAtPositionFromArray(number position, array MediaItemArray, boolean crossfade)

position is the position, at which an item shall be split. MediaItemArray is an array with all MediaItems, that shall be split, if possible. crossfade sets if automatic crossfade shall be applied to or not, just as in SplitMediaItems_Position above.
This function will split only items, that have the position in them somewhere. That means, if you want to split at position 22 seconds, an item, that goes from 1 to 4 will not be split, an item from 18 to 25 will be split.

This function returns, if splitting was successful and the newly "created"-right-hand-split-items as an MediaItemAray. Only the right-hand-split-MediaItem of MediaItems, that could be split, will be returned. If a MediaItem could not be split(position outside MediaItemstart and MediaItemEnd), there will be no returned MediaItem for it then.



^ MediaItems: Deleting MediaItems by Time and Tracks

Deleting of MediaItems is often a useful thing. Unfortunately, this is inconvenient to do within the Reaper-API. The only such function is DeleteTrackMediaItem, but it requires you to give the function the track in which the MediaItem lies too. But often, you want to have a function that simply deletes a MediaItem-object or deletes items at position from numerous tracks, etc.
So I added some functions, that make life easier: DeleteMediaItem, DeleteMediaItemsFromArray, DeleteMediaItems_Position and DeleteMediaItemsBetween

In addition to deleting the MediaItems, all these functions return the statechunks of the deleted MediaItems. These statechunks contain an additional field

            "ULTRASCHALL_TRACKNUMBER"

which contains the track, in which the MediaItem was located before deleting it.
This may help doing cut and paste functions, as otherwise, you lose the information, in which track a certain MediaItem was located.

DeleteMediaItem:

            boolean retval, string MediaItemStateChunk = ultraschall.DeleteMediaItem(MediaItem MediaItem)

This function is simple. Just pass the MediaItem that you want to delete to it and it will delete it. If no such MediaItem exists, it will return false.

DeleteMediaItemsFromArray:

            boolean retval, array MediaItemArray = ultraschall.DeleteMediaItemsFromArray(array MediaItemArray)

This function might be more interesting if you want to delete a number of MediaItems at once. Just pass to it a MediaItemArray(like the one returned by functions like GetAllMediaItemsBetween)

DeleteMediaItems_Position:

            boolean retval, array MediaItemStateChunkArray = ultraschall.DeleteMediaItems_Position(number position, string trackstring)

This function deletes all items at position in the tracks, given by trackstring.
If you want to delete all items at position 22, within track 1,4,8 and 9, you just type:

            retval = ultraschall.DeleteMediaItems_Position(22, "1,4,8,9")

DeleteMediaItems_Position:

            boolean retval, array MediaItemStateChunkArray  = 
                        ultraschall.DeleteMediaItems_Between(number startposition, number endposition, string trackstring, boolean inside)

This function deletes MediaItems between start and endposition, within the track given by parameter trackstring. You can use the parameter inside to set, if you want to delete only items that are completely within start and endposition(true) or also include items, that are only partially within start and endposition.
This function works like GetAllMediaItemsBetween, with the additional benefit of deleting the MediaItems.
Let's assume, you want to delete all MediaItems between position 33 and 98, within the tracks 3, 5, 10 and 14 and only the items that are completely within the position 33 and 98, you type:

            retval, MediaItemStateChunkArray  = ultraschall.DeleteMediaItems_Between(33, 98, "3,5,10,14", true)

With all these functions, deleting MediaItems is now comfortable to do.



^ MediaItems: SectionCut, RippleCut, RippleInsert

One of Reaper's real great features is Ripple-Edit. On of the big shortcomings, it only allows Ripple Cut all tracks, Ripple Cut one track, no Ripple Cut.
But what if you want to RippleCut two or more tracks, but not all of them? What, if you want to RippleCut only selected tracks? What if you just want to cut a section without rippling?
Impossible you say!
With naked Reaper, yes. But possible with the Ultraschall-API.

For that I added the following functions: RippleCut, RippleCut_Reverse, RippleInsert, SectionCut, SectionCut_Inverse
All these functions return a MediaItemStateChunkArray, where every StateChunk includes an additional entry "ULTRASCHALL_TRACKNUMBER", which holds the tracknumber, in which the cut MediaItem/piece of a MediaItem was originally located.

RippleCut:

            integer number_items, array MediaItemArray_StateChunk 
                 = ultraschall.RippleCut(number startposition, number endposition, string trackstring, boolean moveenvelopepoints, boolean add_to_clipboard)

With this function, you can RippleCut between startposition, endposition within the tracks as given in trackstring. You can also decide, whether to move the envelope-points as well.
You can also decide, whether the cut items shall be put into the clipboard as well. This should give you total control in how RippleCut is behaving for your needs.
Let's take an example. If you want to cut between seconds 20 and 50 in track 1,4,5,7, not moving the markers but the envelope-points, you type in this Example:

            number_items, MediaItemArray_StateChunk = ultraschall.RippleCut(20, 50, "1,4,5,7", true, true)

This cuts out the section between seconds 20 and 50 and moves everything after that toward the beginning of the project, to fill the gap of the cut section. It also puts the cut items into the clipboard.

RippleCut_Reverse:

            integer number_items, array MediaItemArray_StateChunk = 
                 ultraschall.RippleCut_Reverse(number startposition, number endposition, string trackstring, 
                                               boolean moveenvelopepoints, boolean add_to_clipboard)

With this function, you can RippleCut, but unlike RippleCut above, RippleCut_Reverse moves everything BEFORE the cut towards the end to fill the cut.
Everything else is just the same as RippleCut.

RippleInsert:

            integer number_of_items, array MediaItemArray, number endpos_inserted_items = 
                ultraschall.RippleInsert(number position, array MediaItemArray, string trackstring, boolean moveenvelopepoints, boolean movemarkers)

This function inserts the items in MediaItemArray at position within the tracks, given by trackstring. You can also decide, whether markers and envelope-points shall be moved.
This is quite the opposite of RippleCut: it will split the items at position, move the items after the split towards the end of the project and include the MediaItems in MediaItemArray.
The length of the movement is according the overall length of all MediaItems, beginning with the earliest and ending with the latest MediaItem-length in MediaItemArray.
One additional note: MediaItems will only be included into the tracks they were originally located in, means: an item from track 1 will be included into track 1. The parameter trackstring can only be used to exclude items from certain tracks.
So a trackstring "1,3,4" will only insert all items from tracks 1,3 and 4, leaving out all of the MediaItems from track 2.
Example:

            number_of_items, MediaItemArray, endpos_inserted_items = ultraschall.RippleInsert(20, MediaItemArray, "1,4,9,10" false, false)

This will insert all MediaItems from MediaItemArray, ordered by their relative position, at position 20 seconds. Only the MediaItems from tracks 1,4,9,10 will be included. All others will be ignored.
Markers and Envelopepoints will not move in this example.

SectionCut:

            integer number_items, array MediaItemArray_StateChunk = 
                              ultraschall.SectionCut(number startposition, number endposition, string trackstring, boolean add_to_clipboard)

This function just cuts out the section between start and endposition in the tracks, given in trackstring, leaving a "gap" in it. Useful, when you don't want to ripple stuff.
You can also decide, whether to put the cut items into the clipboard. If you want to cut between seconds 77 and 99 in tracks 1,2,4,6 do it like in this Example:

            number_items, MediaItemArray_StateChunk = ultraschall.SectionCut(77, 99, "1,2,4,6", false)

SectionCut_Inverse:

            integer number_items_beforestart, array MediaItemArray_StateChunk_beforestart, 
            integer number_items_afterend, array MediaItemArray_StateChunk_afterend 
                                   = ultraschall.SectionCut_Inverse(number startposition, number endposition, string trackstring, boolean add_to_clipboard)

This function cuts everything BEFORE AND AFTER start and endposition within the tracks given by trackstring. This is comparable to crop-functionality in graphic-applications like Photoshop, applied to MediaItems.
If you have a 10 minute project, but want to use only the audio from seconds 60 to 89 in tracks 1,2,7,8 you type in this Example:

            number_items_beforestart, MediaItemArray_StateChunk_beforestart, number_items_afterend, MediaItemArray_StateChunk_afterend = 
                                                                                            ultraschall.SectionCut_Inverse(60, 89, "1,2,7,8", false)

With that, everything before second 60 and everything after second 89 in tracks 1,2,7,8 will be deleted.
In addition to that, the function returns a MediaItemStateChunkArray for both, the items cut before startposition and one for the items cut after endposition.

This should give you many additional use-cases into your hands.



^ MediaItems: Moving and Manipulating

Getting, splitting, editing and deleting MediaItems isn't enough. In fact, you also want to manipulate them. And you also want to be able to manipulate many of them at once.
Hence, I added lots of functions to manipulate MediaItems, like:

Moving

Let's have a look at the moving-MediaItems-functions.

MoveMediaItemsAfter_By

            boolean retval = ultraschall.MoveMediaItemsAfter_By(number old_position, number change_position_by, string trackstring)

This function moves all MediaItems from old_position and later by a number of seconds, as given by the parameter change_position_by.
If change_position_by is negative, the MediaItems will be move towards the beginning of the project; a positive value will move the MediaItems toward the end.
The parameter trackstring tells the function, in which tracks the MediaItems shall be moved.

MoveMediaItemsBefore_By

            boolean retval = ultraschall.MoveMediaItemsBefore_By(number old_position, number change_position_by, string trackstring)

This basically works like the MoveMediaItemsAfter_By above, with the difference, that it moves the MediaItems BEFORE old_position.

MoveMediaItemsBetween_To

            boolean retval = ultraschall.MoveMediaItemsBetween_To(number startposition, number endposition, number newposition, 
                                                                  string trackstring, boolean inside)

This also moves MediaItems, but the MediaItems between startposition and endposition. Unlike the functions above, you give the new position in seconds, at which the MediaItems shall start. The relative-positions of the MediaItems will stay intact.
The parameter inside allows you to tell the function, whether to include only MediaItems completely within start and endposition(true) or also MediaItems that are partially within start and endposition(false).

MoveMediaItems_FromArray

            integer retval, number earliest_itemtime, number latest_itemtime = ultraschall.MoveMediaItems_FromArray(array MediaItemArray, number newposition)

This moves the MediaItems in MediaItemArray to the newposition. It will retain the relative positions of the MediaItems as well.

Length

You can change the length of the MediaItems with functioncalls of Reaper's own API already. I added functions, that allow you to change the length of multiple MediaItems at once, using MediaItemArray. And not just the length to a given length, but also a deltalength. Let's have a look.

ChangeLengthOfMediaItems_FromArray

            boolean retval = ultraschall.ChangeLengthOfMediaItems_FromArray(array MediaItemArray, number newlength)

This changes the length of all MediaItems in the MediaItemArray to newlength in seconds.
Example:

            MediaItemArray={}
            MediaItemArray[1]=reaper.GetMediaItem(0,0)
            MediaItemArray[2]=reaper.GetMediaItem(0,1)
            MediaItemArray[3]=reaper.GetMediaItem(0,2)
        
            retval = ultraschall.ChangeLengthOfMediaItems_FromArray(MediaItemArray, 3)

This examplecode will change the length of all MediaItems in MediaItemArray to a length of 3 seconds.

ChangeDeltaLengthOfMediaItems_FromArray

            boolean retval = ultraschall.ChangeDeltaLengthOfMediaItems_FromArray(array MediaItemArray, number deltalength)

This changes the length of the MediaItems in MediaItemArray as well, BUT it will change the length BY deltalength in seconds. That means, if deltalength is 4, all MediaItems in the MediaItemArray will become longer by 4 seconds, if deltalength is -3, all MediaItems in MediaItemArray will become 3 seconds shorter(!)
Example:

            MediaItemArray={}
            MediaItemArray[1]=reaper.GetMediaItem(0,0) -- let's assume, this MediaItem is 10 seconds long
            MediaItemArray[2]=reaper.GetMediaItem(0,1) -- let's assume, this MediaItem is 30 seconds long
            MediaItemArray[3]=reaper.GetMediaItem(0,2) -- let's assume, this MediaItem is 5 seconds long
        
            retval = ultraschall.ChangeDeltaLengthOfMediaItems_FromArray(MediaItemArray, 4)

This examplecode will change the length of all MediaItems in MediaItemArray by 4 seconds, so the first item is now 14 seconds long, the second 34 and the third 9 seconds.

Offset

Just like the length of MediaItems, you can change the offset as well with functioncalls of Reaper's own API. I added functions, that allow you to change the length of multiple MediaItems at once, using MediaItemArray. And not just the length to a given length, but also a deltalength. Let's have a look.

ChangeOffsetOfMediaItems_FromArray

            boolean retval = ultraschall.ChangeOffsetOfMediaItems_FromArray(array MediaItemArray, number newoffset)

This changes the offset of all MediaItems in the MediaItemArray to newoffset in seconds.
Example:

            MediaItemArray={}
            MediaItemArray[1]=reaper.GetMediaItem(0,0)
            MediaItemArray[2]=reaper.GetMediaItem(0,1)
            MediaItemArray[3]=reaper.GetMediaItem(0,2)
        
            retval = ultraschall.ChangeOffsetOfMediaItems_FromArray(MediaItemArray, 3)

This examplecode will change the offset of all MediaItems in MediaItemArray to a the new offset of 3 seconds.

ChangeDeltaOffsetOfMediaItems_FromArray

            boolean retval = ultraschall.ChangeDeltaOffsetOfMediaItems_FromArray(array MediaItemArray, number deltaoffset)

This changes the offset of the MediaItems in MediaItemArray as well, BUT it will change the offset BY deltaoffset in seconds. That means, if deltaoffset is 4, all MediaItems in the MediaItemArray will start 4 seconds later, if deltaoffset is -3, all MediaItems in MediaItemArray will start 3 seconds earlier(!)
Example:

            MediaItemArray={}
            MediaItemArray[1]=reaper.GetMediaItem(0,0) -- let's assume, this MediaItem's offset starts at 10 seconds
            MediaItemArray[2]=reaper.GetMediaItem(0,1) -- let's assume, this MediaItem's offset starts at 30 seconds
            MediaItemArray[3]=reaper.GetMediaItem(0,2) -- let's assume, this MediaItem's offset starts at 0 seconds
            
            retval = ultraschall.ChangeDeltaOffsetOfMediaItems_FromArray(MediaItemArray, 4)

This examplecode will change the offset of all MediaItems in MediaItemArray by 4 seconds, so the first item's offset starts now at 14 seconds, the second 34 and the third 4 seconds.

Other functions for manipulating MediaItems:



^ MediaItems: Inserting Items and Files

Last, but not least, it would be nice to be able to insert MediaItems as well. And not just inserting them from a project, but also from files.
For that, I made: InsertMediaItem_MediaItem, InsertMediaItem_MediaItemStateChunk, InsertMediaItemArray, InsertMediaItemStateChunkArray, InsertMediaItemFromFile, InsertImageFile

Lets have a look at inserting MediaItems.

InsertMediaItem_MediaItem

            integer retval, MediaItem MediaItem, number startposition, number endposition, number length 
                        = ultraschall.InsertMediaItem_MediaItem(number position, MediaItem MediaItem, MediaTrack MediaTrack)

With this function, you can make a copy of an already existing MediaItem and insert at position into a certain track. It allows using MediaItems and MediaTracks of other projects than the current one, as well.
Just give the position, at which to insert the MediaItem, the MediaItem to be included and the MediaTrack into which to include the MediaItem.
It will return the newly created MediaItem, it's startposition, endposition and the length.

InsertMediaItem_MediaItemStateChunk

            integer retval, MediaItem MediaItem 
                        = ultraschall.InsertMediaItem_MediaItemStateChunk(number position, string MediaItemStateChunk, MediaTrack MediaTrack)

This is like InsertMediaItem_MediaItem, but uses a MediaItemStateChunk instead. The rest is just the same, including the possibility to insert the new MediaItem into a MediaTrack in another project than the current one.

InsertMediaItemArray

            integer number_of_items, array MediaItemArray = ultraschall.InsertMediaItemArray(number position, array MediaItemArray, string trackstring)
            

This allows you to insert multiple items at once, that are stored in a MediaItemArray at position. With trackstring you can set, into which tracks to insert the MediaItems.
There's a limitation, however: MediaItems will only be inserted into the tracks from where they originated from. That means, if you have MediaItems located in tracks 1-5 and you set trackstring to "1,2", only the MediaItems originating in tracks 1 and 2 will be inserted.
I'm still looking into a better way to provide the track, in which to insert the MediaItems into other MediaTracks as well.

InsertMediaItemStateChunkArray

            integer number_of_items, array MediaItemArray 
                            = ultraschall.InsertMediaItemStateChunkArray(number position, array MediaItemStateChunkArray, string trackstring)

This works like InsertMediaItemArray, but it's inserting MediaItemStateChunks from the MediaItemStateChunkArray instead.
In addition to that, it also lifts the track-limitation, when you insert the tracknumber into each MediaItemStateChunk using SetItemUSTRackNumber_StateChunk, which is automatically done by GetItem-functions from the Ultraschall-API.
The addition of the tracknumber is mandatory, otherwise the MediaItemStateChunk will not be inserted!

InsertMediaItemFromFile

            integer retval, MediaItem item, number endposition, integer numchannels, integer Samplerate, string Filetype
               = ultraschall.InsertMediaItemFromFile(string filename, integer track, number position, number endposition, 
                                                     integer editcursorpos, optional number offset)
               

This function allows you to insert a file as a new MediaItem into your project, including the tracknumber, position, endposition(if wanted), the editcursorposition and the offset, if wanted.
Example:

            retval, item, endposition, numchannels, Samplerate, Filetype = ultraschall.InsertMediaItemFromFile("C:\\file.wav", 1, 20, -1, 2, 0)
       

This example inserts the file file.wav into track 1, at position 20, with the length set to the length of the audio-file, the editcursorposition being put at the end of the new MediaItem and no offset-changes.

With that, you can easily insert files into your project.

InsertImageFile

            boolean retval, MediaItem item = ultraschall.InsertImageFile(string filename_with_path, integer track, number position, 
                                                                         number length, boolean looped)

This is a special function, focusing on inserting image-files into the project. You can set the track, position, length of the image-MediaItem and, if it shall be looped.
If you don't loop it(looped=false), the MediaItem will have the length anyway, but the image will be shown only for 1 second in the VideoProcessor-window.



^ MediaItems: Programming Spectral Edit

With Reaper v5.50c, the devs introduced a new feature, called, spectral editing, which is a cool feature to influence frequencies in a spectral view of a MediaItem.
But they didn't include some functions to program this feature. So I added them myself.

The functions for using, manipulating, adding, deleting spectral-edits are: AddItemSpectralEdit, CountItemSpectralEdits, DeleteItemSpectralEdit, GetItemSpectralConfig, GetItemSpectralEdit, GetItemSpectralVisibilityState, SetItemSpectralConfig, SetItemSpectralEdit, SetItemSpectralVisibilityState

Let's go into this in more detail.

SetItemSpectralVisibilityState

To use the SpectralEdit-mode, you need to first enable visibility of it in a MediaItem. You can do this later too, but you will not see any of your changes, until you enable visibility first.

             string MediaItemStateChunk = ultraschall.SetItemSpectralVisibilityState(integer itemidx, integer state, optional string MediaItemStateChunk)
             

This function enables visibility of Spectral-Edit of item itemidx. The parameter state must be set to 1 to set to visible, or set to 0 to turn visibility off.
The function returns the altered MediaItemStateChunk in any way.
If you set itemidx to -1, you can use the optional parameter MediaItemStateChunk instead. This will add the corresponding entry for the visibility into the MediaItemStateChunk and returns the modified one.

Now that we have toggled the visibility of SpectralEdit, we might want to add SpectralEdit-instances to the MediaItem. For that we use:

AddItemSpectralEdit

             boolean retval, MediaItemStateChunk statechunk 
                         = ultraschall.AddItemSpectralEdit(integer itemidx, number start_pos, number end_pos, number gain, number fade, 
                                                           number freq_fade, number freq_range_bottom, number freq_range_top, integer h, 
                                                           integer byp_solo, number gate_thres, number gate_floor, number comp_thresh, 
                                                           number comp_exp_ratio, number n, number o, number fade2, number freq_fade2, 
                                                           string MediaItemStateChunk)
                                                           

This will add a SpectralEdit-instance into your MediaItem. You can add as many individual instances, as you want.

As you can see, you can influence a hell lot of parameters for such a SpectralEdit-instance, so I will not explain them in detail here. I suggest you to read the accompanying doc-entry for AddItemSpectralEdit, which explains the parameters in more detail.
Just some bits: Every Spectral-Edit-instance will be shown as a square/rectangle on top of the MediaItem. You can influence this rectangle's position and length, the frequency-ranges covered, the fades as well as all settings of all knobs appearing in it.
And as the cherry on the top: you can also bypass and solo it.
And, as the SetItemSpectralVisibilityState-function above, if you set itemidx to -1, you can add the Spectral-Edit-instance to a MediaItemStateChunk instead.

But what if you want to modify an already existing SpectralEdit-instance? Good question and I have a good answer to that:

SetItemSpectralEdit

             string MediaItemStateChunk 
                         = ultraschall.SetItemSpectralEdit(integer itemix, integer spectralidx, number start_pos, number end_pos, number gain, 
                                                           number fade, number freq_fade, number freq_range_bottom, number freq_range_top, 
                                                           integer h, integer byp_solo, number gate_thres, number gate_floor, number comp_thresh, 
                                                           number comp_exp_ratio, number n, number o, number fade2, number freq_fade2, 
                                                           string MediaItemStateChunk)
                                                           

This sets an already existing SpectralEdit-instance, and it probably reminds you very much of the AddItemSpectralEdit-function.
However, there is a small difference in it, the second parameter spectralidx, which tells the function, which spectral-edit-instance you want to change, with 1 for the first.
The rest is like AddItemSpectralEdit.

To delete such an instance, you can use the function:

DeleteItemSpectralEdit

             boolean retval, string MediaItemStateChunk = ultraschall.DeleteItemSpectralEdit(integer itemidx, integer spectralidx, 
                                                                                             string MediaItemStateChunk)
             

With that, you can easily delete a SpectralEdit-instance, by giving the item's idx(itemidx) and the number of the SpectralEdit-instance(spectralidx).
And, as most of the functions before: when setting itemidx to -1, you can use the optional parameter MediaItemStateChunk

To successfully set an already existing instance, you probably want to know, what current settings are in a SpectralEdit-instance.
For that, there is:

GetItemSpectralEdit

             number start_pos, number end_pos, number gain, number fade, number freq_fade, 
             number freq_range_bottom, number freq_range_top, integer h, integer byp_solo, 
             number gate_thres, number gate_floor, number comp_thresh, number comp_exp_ratio, 
             number n, number o, number fade2, number freq_fade2 
                         = ultraschall.GetItemSpectralEdit(integer itemidx, integer spectralidx, string MediaItemStateChunk)
                         

This returns all settings you can set with AddItemSpectralEdit and SetItemSpectralEdit, by giving the item's idx within the project(itemidx) and the SpectralEdit-instance(spectralidx).
And you know the drill: when setting itemidx to -1, you can pass a MediaItemStateChunk to the function.

Some other functions for SpectralEdit-management are:



^ MediaItems: Miscellaneous

Still not enough? Well, I've added numerous other functions, and I want to introduce you to some of the gems included.
Feel free to browse through the Functions-Reference to find more.

What about, previewing MediaItems and files? Use this:

PreviewMediaItem

            boolean retval = ultraschall.PreviewMediaItem(MediaItem MediaItem, integer Previewtype)
            

This previews an existing MediaItem, which means: Reaper will play it, regardless of what you hear in your project currently.
You can also set, where you want to have it previewed, through the MediaExplorer, the MediaItem itself, through the volume-settings of the track, in which it lies and through the track, in which it lies(including FX and such).
You can just play one MediaItem at a time, unless you play one through the MediaExplorer and one through another previewing-type.

If you want to preview a file not in the current project, you can use:

PreviewMediaFile

            boolean retval = ultraschall.PreviewMediaFile(string filename_with_path)
            

which will simply play the file you gave using filename_with_path.

To stop any preview, just use

StopAnyPreview

            ultraschall.StopAnyPreview()
            

which stops any previewing, be it from a MediaItem or an external mediafile.

What about applying Actions to MediaItems? Use this:

ApplyActionToMediaItem

            boolean retval = ultraschall.ApplyActionToMediaItem(MediaItem MediaItem, string actioncommandid, 
                                                                integer repeat_action, boolean midi, optional HWND MIDI_hwnd)
                                                                

which allows applying main and midi-editor-actions to MediaItem. Just pass the command_id/action_command_id to the parameter.
With parameter repeat_action, you can set, how often the action shall be applied to the MediaItem.
To apply MIDI-Editor, actions, set midi=true and pass over a HWND of the used MIDI-Editor, using Reaper's own API function MIDIEditor_GetActive.

To apply action to multiple MediaItems, use:

ApplyActionToMediaItemArray

            boolean retval = ultraschall.ApplyActionToMediaItemArray(MediaItemArray MediaItemArray, string actioncommandid, 
                                                                     integer repeat_action, boolean midi, optional HWND MIDI_hwnd)
            

which works just the same as ApplyActionToMediaItem, but uses a MediaItemArray that includes the MediaItems to be affected.

In addition to actions, you can also apply functions to MediaItems:

ApplyFunctionToMediaItemArray

            table returnvalues 
                    = ultraschall.ApplyFunctionToMediaItemArray(MediaItemArray MediaItemArray, function functionname, 
                                                                        functionparameters1, ..., functionparametersn)

You just pass to it the MediaItemArray, the functionname, the parameters for the function functionname.
Keep in mind: if a parameter of functionname shall hold the MediaItem, you need to set the accompanying parameter to nil, ApplyFunctionToMediaItemArray will automatically insert the appropriate MediaItem at this nil parameter.

What else? What about Normalizing MediaItems? Use this:

NormalizeItems

            integer retval = ultraschall.NormalizeItems(array MediaItemArray)
            

Just pass to it a MediaItemArray, that holds all MediaItems to be normalized.

And last, but not least: What about applying MediaItemStateChunks to MediaItems? Use this:

ApplyStateChunkToItems

            boolean retval, integer skippeditemscount, array skipped_MediaItemStateChunkArray 
                    = ultraschall.ApplyStateChunkToItems(array MediaItemStateChunkArray, boolean undostate)
                    

This applies the MediaItemStateChunks in MediaItemStateChunkArray to the appropriate MediaItems. That means, if a StateChunk is of a certain, existing MediaItem, the function will apply the StateChunk to the MediaItem.
This function is especially helpful, when mass manipulating StateChunks and wanting to mass-apply the changed ones back.
Easy to do now.



^ File Management: Introduction

Even if file-management isn't that hard to program in Lua, it is quite inconvenient. Especially for "normal" use-cases, it is often a drag to always go through the four steps, checking if file exists, open file, read/write, close file.
Wouldn't it be cool, to have functions to do it for you?

Well now, there are, as the Ultraschall-API includes 28 functions to do it for you.
These functions include functions for reading and writing, copying them.
You can also check for valid filetypes, for valid directories, can count files and directories in paths, get paths and files in a path, and get length or number of lines in a file.

With that, you can do a lot of file-operation quite easy, without having to dig through the little details of the Lua-Reference-Manual.

Let's start with reading files.



^ File Management: Read

To read files, I included some nice functions, like: ReadFullFile, ReadBinaryFile, ReadBinaryFileFromPattern, ReadBinaryFileUntilPattern,
ReadFileAsLines_Array, ReadLinerangeFromFile, ReadValueFromFile, ReadBinaryFile_Offset

Let's start with the function probably used the most:

ReadFullFile

            string contents, integer length_of_file, integer number_of_lines = ultraschall.ReadFullFile(string filename_with_path, boolean binary)

This reads a file fully and returns it's contents to the return-variable contents. It will also return the length of the file.
If you set the parameter binary to true, it will read the files as binary files; if set to false or nil, it will read the file either until the end or until a eof-character comes up.
If you're dealing with textfiles, set it to false or nil, otherwise to true.

If you want to return all lines from a textfile, that have a certain character-pattern in them, use this function:

ReadValueFromFile

            string contents, string linenumbers, integer numberoflines, integer number_of_foundlines = 
                                                ultraschall.ReadValueFromFile(string filename_with_path, string value)
            

This reads a file as textfile and returns it. When you give parameter value a string, it will return all lines from the file, that contain this string.
The returned values are contents, the linenumbers returned as a comma-separated-csv, the total number of lines in the file and the number of lines found and returned
This should help you to read only the lines useful for you, however, it is much slower than ReadFullFile() due the massive pattern-matching used in it.
So, even if you can read the full file with that, better use ReadFullFile when you want the full file returned.

If you want to return everything from a pattern to the end of a file, use:

ReadBinaryFileFromPattern

            integer length, string content = ultraschall.ReadBinaryFileFromPattern(string input_filename_with_path, string pattern)

with this function, you can read a file from a pattern onwards. That means, the function searches for the first(!) instance of pattern and returns the file from that pattern until the end of the file.

Similar to this is:

ReadBinaryFileUntilPattern

            integer length, string content = ultraschall.ReadBinaryFileUntilPattern(string input_filename_with_path, string pattern)

which reads a file until(!) a certain pattern is found in it and returns this. That means, it returns the file from it's start until the pattern.

But what if you want to read a file from a start-offset to an endoffset? Use this:

ReadBinaryFile_Offset

            integer length, string content = 
                    ultraschall.ReadBinaryFile_Offset(string input_filename_with_path, integer startoffset, integer numberofbytes)

This function returns the contents of the file from startoffset(in Bytes) until startoffset+numberofbytes.
If you set number of bytes to -1, the function will return the file from startoffset to it's end.
Positive value in startoffset will be related to the beginning of the file. If you want to return the file from a startoffset related to the end of the file,
use negative value for startoffset.
A startoffset of -1 is for the end of the file, -2 for the last byte in the file, -3 for the second to last byte in the file, etc.
Even with negative startoffset, parameter numberofbytes will count from the startoffset to startoffset+numberofbytes.
If the startoffset+numberofbytes reach or cross the end of the file, it will return a shorter string.
To check, whether it returned the requested length, check the returnvalue length.
Returnvalue content holds the requested part of the file

When you have a textfile and you want to work with it on an individual line-basis, use:

ReadFileAsLines_Array

            array contents, boolean correctnumberoflines, integer number_of_lines = 
                    ultraschall.ReadFileAsLines_Array(string filename_with_path, integer firstlinenumber, integer lastlinenumber)

This returns a textfile, split into it's individual lines put into an array. You can also set the linenumbers, that you want to have returned.
If the linenumbers returned are fewer than you requested, correctnumberoflines will be false, otherwise it will be true.

If you want to have the lines returned in one string, use the following instead:

ReadLinerangeFromFile

            string contents, boolean correctnumberoflines, integer number_of_lines = 
                     ultraschall.ReadLinerangeFromFile(string filename_with_path, integer firstlinenumber, integer lastlinenumber)

This works basically the same as ReadFileAsLines_Array, but returns the found-lines as a newline-separated string.

These functions should fulfill most of your daily file-read-usecases.



^ File Management: Write

Reading files is cool, writing files is cool as well, so I added some for exactly that: WriteValueToFile, WriteValueToFile_Insert, WriteValueToFile_Replace, WriteValueToFile_InsertBinary, WriteValueToFile_ReplaceBinary

Let's start with the function, that you'll probably use the most for writing:

WriteValueToFile

            integer retval = ultraschall.WriteValueToFile(string filename_with_path, string value, optional boolean binarymode, optional boolean append)
            

This writes a value to filename_with_path. Optionally, you can control, if the file shall be written as binary-file and if value shall be appended to the current contents of the file.
Default is, file will be stored as binary and value replaces the current contents of the file.

To insert values into a textfile, use:

WriteValueToFile_Insert

            integer retval = ultraschall.WriteValueToFile_Insert(string filename_with_path, integer linenumber, string value)
            

This inserts value after the line, given by parameter linenumber. This works only for textfiles, not for binary-files.

To replace parts of a textfile, use:

WriteValueToFile_Replace

            integer retval = ultraschall.WriteValueToFile_Replace(string filename_with_path, integer startlinenumber, integer endlinenumber, string value)

This replaces the lines between (including)-startnumber and (including)-endlinenumber with the parameter value. This works only for textfiles, not for binary-files.

For binary-files, we have dedicated functions for that as well:

WriteValueToFile_InsertBinary

            integer retval = ultraschall.WriteValueToFile_InsertBinary(string filename_with_path, integer byteposition, string value)
            

This inserts parameter value at the fileoffset given by parameter byteposition. This works for binary-files and for textfiles(though textfiles may cause issues at some points).

To replace contents of a binaryfile, you can use:

WriteValueToFile_ReplaceBinary

            integer retval = ultraschall.WriteValueToFile_ReplaceBinary(string filename_with_path, integer startbyteposition, 
                                                                        integer endbyteposition, string value)
            

This inserts parameter value between (including) startbyteposition and (including)endbyteposition. This works for binary-files and for textfiles(though textfiles may cause issues at some points).

This should make saving files easier than having to code it completely yourself by hand.



^ File Management: Analyse

Sometimes, you want to know more about the contents of a file. For that, I included some functions as well: CountLinesInFile, GetLengthOfFile, CheckForValidFileFormats, OnlyFilesOfCertainType

For reading files, you probably want to know the length of the files or number of lines in it. For that you can use:

GetLengthOfFile

            integer lengthoffile = ultraschall.GetLengthOfFile(string filename_with_path)

This returns the length of the file in bytes.

CountLinesInFile

            integer linesinfile = ultraschall.CountLinesInFile(string filename_with_path)

This returns the number of lines in a textfile.

If you want to know, what type a certain file is, you can use:

CheckForValidFileFormats

            string fileformat, boolean supported_by_reaper, string mediatype = ultraschall.CheckForValidFileFormats(string filename_with_path)

This returns the type of a file, which is either

            JPG, PNG, GIF, LCF, ICO, WAV, AIFF, ASF/WMA/WMV, MP3, MP3 -ID3TAG, FLAC, MKV/MKA/MKS/MK3D/WEBM, AVI, RPP_PROJECT, unknown 

and it returns, if the filetype is supported by Reaper and what kind of mediatype the file is, which is either

            Image, Audio, Audio/Video, Video, Reaper
            

But what, if you want to know all files of a certain types in a filelist(like the file in a directory)? Use this:

OnlyFilesOfCertainType

            integer foundfilecount, array foundfilearray = ultraschall.OnlyFilesOfCertainType(array filearray, string filetype)
  

This returns all files from parameter filearray, that are of a certain filetype. Create an array with all filenames you want to check for and use this for parameter filearray. In the parameter filetype, you can use one of:

            JPG, PNG, GIF, LCF, ICO, WAV, AIFF, ASF/WMA/WMV, MP3, MP3 -ID3TAG, FLAC, MKV/MKA/MKS/MK3D/WEBM, AVI, RPP_PROJECT, unknown 

After that, the function returns an array with all files, that are of the filetype you gave to parameter filetype.



^ File Management: Background Copying

If you have a defer-script and would like to copy files in the background while doing other stuff in the meantime, or if you want to copy huge files in Lua, you'll probably face the problem, that copying huge files may block Reaper.
This'll happen, when you have huge files and you try to copy the whole file at once. You can cirvumvent it by reading and writing the file piece by piece, which is programmable in Lua, which needs some kind of defer-loop as well to work on big files; every defer-cycle reading and writing a piece of the file.
But, as I'm too lazy to do that time and again(and you're probably too), I programmed a bunch of functions and mechanism, that'll do it for you.

Basically it works like that: 1) add files to the copying-queue, using CopyFile_AddFileToQueue 2) start copying in the background using CopyFile_StartCopying

and that will copy files in the background(1MB per defer-cycle) and you'll need to do nothing anymore, EXCEPT: not having more than 900-defer-cycles in your script currently running.

Wait, are you saying, you want more control over the copying? Want to set the copy-buffer-size? Want to pause or stop the processing? Want to know, when the copying is done? Okokok, I've heard you, so here's a bunch of functions, who could help you having control over the background-copying.

CopyFile_AddFileToQueue

            integer current_copyqueue_position = ultraschall.CopyFile_AddFileToQueue(string filename, string targetfilename, boolean overwrite)

This adds a file into the copy-queue. Just add the filename and the new targetname into it. You can also decide, whether a possibly existing file shall be overwritten.
If you have started the copying already and it's still running, this file will be copied immediately.
The returned value in current_copyqueue_position allows you get the current copying status and possible error-messages using CopyFile_GetCopiedStatus

CopyFile_GetBufferSize

            integer buffer_size = ultraschall.CopyFile_GetBufferSize()

This returns the size of the copying-buffer. Default is 1MB but you can raise or lower it using CopyFile_SetBufferSize, depending on the workload the copying shall cause.
1MB seems to be a good value for background-copying files even on slower systems.

CopyFile_SetBufferSize

            boolean retval = ultraschall.CopyFile_SetBufferSize(integer buffersize)

This sets the size of the copying-buffer. Default is 1MB but you can raise or lower it depending on the workload the copying shall cause.
1MB seems to be a good value for background-copying files even on slower systems.

To start/stop/pause copying, you can use these functions:

CopyFile_StartCopying

            integer instance_number = ultraschall.CopyFile_StartCopying()

This starts a new copying-instance, which processes the file-copy-queue. You can have actually up to 30 running in the background, which means: the more, the more stuff can be copied.
So if the copy-buffer-size is 1MB, you can have up to 30 copying-instances, with each copying 1MB/second so the amount of data copied per second is 30MB with 30 started copying instances.
You need to be careful though, if you've set the copy-buffer-size too high, having multiple copying-instances will lead to a laggy or even frozen GUI of Reaper, until the queue is finished in processing.
When the queue is finished in processing, each copying-instance will stop itself automatically.

CopyFile_StopCopying

            ultraschall.CopyFile_StopCopying()

This stops immediately the processing of the copying-queue. If background-copying was in the middle of the copying, the last file processed remains incompletely copied.

CopyFile_IsCurrentlyCopying

            boolean retval = ultraschall.CopyFile_IsCurrentlyCopying()

This returns, if there's any background-copying-instance running. So you can check, whether to start a background-instance after having added new files to the copy-file-queue.

CopyFile_GetCurrentlyRunningCopyInstances

            integer number_of_instances = ultraschall.CopyFile_GetCurrentlyRunningCopyInstances()

This returns, how many background-copy-instances are currently running.

CopyFile_Pause

            boolean retval = ultraschall.CopyFile_Pause(boolean toggle)

This allows you to pause/unpause ALL copying-instances to save ressources. Keep in mind: if you pause the copying-queue and finish your script, the copying-instances still run paused in the background.
That said, if you pause them, you MUST unpause/stop them before exiting your script or your script will run indefinitely.

CopyFile_GetPausedState

            boolean retval = ultraschall.CopyFile_GetPausedState()

This will return, if the copying-instances are currently paused or not.

Now that you know, how to add files to the queue and how to control the copying-instances, you probably would also want to know the current copy-status of files, including possible error-messages.
For that, I need to explain more details. The Background-Copy-feature of Ultraschall-API manages two queues inside of it:

  1. the actual queue, which will be processed and eventually become empty
  2. a file-queue-statelist, which will store the information about the current copied-status, error-messages, etc. This will only become empty, if you flush it yourself from time to time.

To access the states you need, you should have stored the current_copyqueue_position returned by the CopyFile_AddFileToQueue-function, as this is your golden ticket to get every information you want.

CopyFile_GetCopiedStatus

            string filename, boolean already_processed, string error_message, string error_code = ultraschall.CopyFile_GetCopiedStatus(integer fileindex)
  

This returns all copy-states of a certain file in the queue. You just pass as parameter the current_copyqueue_position returned by the CopyFile_AddFileToQueue-function and it will return everything.
For instance, the filename in question, if it has been already processed(no matter if successfully or unsuccessfully), a possible error-message(which is "" if everything went fine) and the possible error-code.

If you've finished copying and want to get rid of the old states, you can flush the file-queue-statelist:

CopyFile_FlushCopiedFiles

            ultraschall.CopyFile_FlushCopiedFiles()
  

This flushes the file-queue-statelist completely. From that moment on, all current_copyqueue_position returned by the CopyFile_AddFileToQueue up to this point become invalid.
This will not affect background-copy, so if you want to stop processing the file-queue, you need to use CopyFile_StopCopying.

But that's not all control and information, you probably want

CopyFile_GetCurrentlyCopiedFile

            integer number_of_remaining_files, string filename, integer remaining_bytes_to_copy, integer percentage 
                                                                    = ultraschall.CopyFile_GetCurrentlyCopiedFile()
  

This give you information about the file currently in processing. So you can get, how many files are remaining in the queue, the filename currently processed, the remaining bytes to copy of this file and a percentage of how far copying has progressed.

CopyFile_GetRemainingFilesToCopy

            integer filecount = ultraschall.CopyFile_GetRemainingFilesToCopy()
  

And this returns the number of files, who haven't been processed yet in the background-copy-queue.

With that, doing extensive copying-work should be possible for you, even writing a file-manager if you wish.



^ File Management: Misc

Last, but not least, I added many other useful functions regarding file-management, like:



^ Project Management: Introduction

Beside of many functions to read and set project-states in RPP-files and ProjectStateChunks, I also added some other things to work with projects.



^ Project Management: Check for changed projecttabs

Sometimes, you want to know, if projecttabs have been reordered, closed, opened, created. For this, I created the function

CheckForChangedProjectTabs

            boolean retval, integer countReorderedProj, array reorderedProj, integer countNewProj, 
                                         array newProj, integer countClosedProj, array closedProj 
                                                      = ultraschall.CheckForChangedProjectTabs(boolean update)

This checks, if there are changed projecttabs and the number of these changes.
Let's go into details:
When you run a script, that includes the Ultraschall-API, the API creates a list of the currently existing projecttabs.
When you run this function, it will check this internal list and compare, if projects have been added, reordered or closed since them.

This function returns if there's a change, into the return-value retval. The other return-values return

  • the number of reordered projects(countReorderedProj),
  • an array with all reorderd Projects(reorderedProj),
  • the number of new projects(countNewProj),
  • an array with all newly created/opened projects(newProj),
  • the number of closed projects(countClosedProj) and
  • an array with all closed projects(closedProj)

When you want to update this internal list, you should set parameter update to true, otherwise set it to false.

That way, you can for instance work with newly created projects, to automatically add things to projects, that can't be added using TemplateProjects.



^ Color Management: Introduction

Reaper has a lot of cool themeing abilities, which allows you to customize most of the design of Reaper yourself.
One of these things is: customized colors.

Unfortunately, Reaper has it's own way to deal with color. In fact there are two main ways of dealing with color:

The r,g,b-color is, as you are used to it: you have a red-color-value, a green-color-value and a blue-color-value, each going from 0 to 255.
The higher the value of one of these colors, the brighter that part of the color becomes.

The native-color is a system-dependent color-value used by Reaper. It is more difficult to understand, but basically it is

    red+(green\*green)+(blue\*blue\*blue)|0x1000000

Or it isn't, because on MacOS, you need to reverse red and blue, so it becomes

    blue+(green\*green)+(red\*red\*red)|0x1000000  

Why that is, is a mystery to me, but there's a function by Reaper, that does the correct conversion for you automatically: reaper.ColorToNative(r,g,b)|0x1000000
But you always need to add that |0x1000000 after the function, and I usually forget, how to write that properly.

There's also some other nice stuff in many gfx-applications, that is completely missing from Reaper's own color-functions: adjusting brightness, saturation and contrast of colors, which would be helpful for adjusting track-colors and item-colors and such.

And what about manipulating multiple colors at once?

So I thought, why not adding functions, that make that stuff easier?



^ Color Management: Native Color Conversion

For color-conversion into a system-dependent native-color, there are the functions:

ConvertColor, ConvertColorReverse, ConvertColorToMac, ConvertColorFromMac, ConvertColorToWin, ConvertColorFromWin, ConvertGFXToColor, ConvertColorToGFX

Let's start with standard-color-conversion, from/to system-dependent native-colors:

ConvertColor

                integer colorvalue, boolean retval = ultraschall.ConvertColor(integer r, integer g, integer b)
            

This function converts red, green and blue-values into the native-color of your system. Unlike Reaper's own function, you don't need to add |0x1000000, which makes it easier to use.
If for one reason or another the conversion fails, it will return 0 and false as returnvalues.
If you want to convert it into a Mac-native-color-value while on Windows or a Windows one while on Mac, just swap the r and the b values.

ConvertColorReverse

                integer r, integer g, integer b, boolean retval = ultraschall.ConvertColorReverse(integer colorvalue)

This one converts a native-color-value into it's original red, green and blue-colorvalues. If color-conversion failed, the returnvalue retval is false, else it is true.

But what if you want to convert a color from/to a native-Mac-color, even if you're using Windows or Linux? And what if you're using Mac and want to convert from/to the native-color for Windows/Linux?
For that, I added four functions:

ConvertColorToMac

                integer mac_colorvalue, boolean retval = ultraschall.ConvertColorToMac(integer red, integer green, integer blue)

This converts a red, green, blue-value to a Mac-native-colorvalue. This works on Windows, Mac and Linux the same way.
Will set returnvalue retval to false, if conversion failed; if conversion succeeded, it will be set to true.

ConvertColorFromMac

                integer red, integer green, integer blue, boolean retval = ultraschall.ConvertColorFromMac(integer mac_colorvalue)

This converts a Mac-native colorvalue into it's red, green and blue-values. If retval is true then conversion was successful; if false, conversion failed.

ConvertColorToWin

                integer win_linux_colorvalue, boolean retval = ultraschall.ConvertColorToWin(integer red, integer green, integer blue)

This converts a red, green, blue-value to a Windows/Linus-native-colorvalue. This works on Windows, Mac and Linux the same way.
Will set returnvalue retval to false, if conversion failed; if conversion succeeded, it will be set to true.

ConvertColorFromWin

                integer red, integer green, integer blue, boolean retval = ultraschall.ConvertColorFromWin(integer win_colorvalue)
                

This converts a Windows/Linux-native colorvalue into it's red, green and blue-values. If retval is true then conversion was successful; if false, conversion failed.

If you want to convert the red, green, blue, alpha-colorvalues with a range from 0 to 255, into colorvalues useable by the gfx.functions for gfx.init-windows (range 0 ... 1), you can use the following functions:

ConvertColorToGFX

                number r, number g, number b, number a = ultraschall.ConvertColorToGFX(integer r, integer g, integer b, integer a)

This converts red, green, blue, alpha-values(0-255) into gfx-useable functions(0-1). Simply pass the values between 0 to 255 to the function. These returned values can be used by gfx.set.

ConvertGFXToColor

                integer r, integer g, integer b, integer a = ultraschall.ConvertGFXToColor(number r, number g, number b, number a)

This converts red, green, blue, alpha-values(0-255) into gfx-useable functions(0-1). Simply pass the values between 0 to 255 to the function.

With that, you can convert color-values into all color-values you ever want to have.



^ Color Management: Brightness, Contrast and Colorsaturation

Sometimes, you want to alter brightness, contrast or saturation of a color but have no idea, how to do that. For that, I added three functions that do that for you: ChangeColorBrightness, ChangeColorContrast, ChangeColorSaturation

Let's start with adjusting the brightness:

ChangeColorBrightness

                integer red, integer green, integer blue, boolean retval = 
                        ultraschall.ChangeColorBrightness(integer r, integer g, integer b, 
                                                          integer bright_r, optional integer bright_g, optional integer bright_b)
                                                          

This function alters the brightness of a color. Just pass to it the old r,g,b-values and the by how much the color shall be brightened/darkened.
To do this, set bright_r, bright_g and bright_b to the new deltavalue. If these delta-values are negative, the color will become darker, if positive, it will become brighter.
If you pass only bright_r and omit bright_g and bright_b, the deltavalue set by bright_r will be applied to red, green and blue at the same time.
To prevent that, set bright_g and bright_b to 0.
It returns the changed colorvalues and the returnvalue retval, which will tell you, if changing saturation was successful(true) or not(false).

Sometimes adjusting brightness is not enough, so let's see, how we can alter contrast of a color:

ChangeColorContrast

                integer red, integer green, integer blue, boolean retval = 
                        ultraschall.ChangeColorContrast(integer r, integer g, integer b, 
                                                        integer Minimum_r, optional integer Maximum_r, 
                                                        optional integer Minimum_g, optional integer Maximum_g, 
                                                        optional integer Minimum_b, optional integer Maximum_b)

This function alters the contrast of a color. It does it by assuming, that the color-value you pass to it, will be seen as the center of the brightness-range, while Minimum_color and Maximum_color are the minimum and maximum of the color-range available.
For example: If you pass as parameter r the value 100, the function will assume, that the current minimum is at 0 and the current maximum is at 255. When you pass now Minimum_r as 0 and Maximum_r as 200, it will calculate the red-colorvalue in relation to the new minimum and maximum. That means, it will divide the new range of 200 by 255 and multiply this value by the old red-value.

               new_redcolor = ((Maximum_r-Minimum_r)/255)*r
               

The more apart Minimum and Maximum become, the stronger the contrast, the closer they become to each other, the weaker the contrast.
You can also influence the brightness by making Maximum and Minimum higher(making it brighter) or lower(making it darker).

If you use only the Minimum_r and Maximum_r-parameters, these will be applied to red, green and blue at the same time.
To prevent that, set Minimum_g to 0 and Minimum_b to 0, Maximum_g to 255 and Maximum_b to 255.

To intensify or desaturate color, you can use the following function:

ChangeColorSaturation

                integer red, integer green, integer blue, number median, boolean retval = 
                        ultraschall.ChangeColorSaturation(integer r, integer g, integer b, integer delta)

This saturates/desaturates a color-value.
Using it is easy, just pass red, green and blue to it, as well as a delta-value that affects the saturation.
To desaturate, set the value for delta negative, to saturate, make delta a positive value.
It will return the new color-values(r,g,b), as well as the median. It also returns retval, which will tell you, if changing saturation was successful(true) or not(false).

The function calculates the new saturation-value by first calculating a median-brightness-value from the red, green and blue-value.
For desaturation: after that, it will add delta to values below median and subtract delta from values above delta.
For saturation: after that, it will subtract delta from values below median and add delta to values above delta.

By that, it will desaturate or saturate.



^ Color Management: Working with Colortables

When working with multiple colors at once, you can use a ColorTable, which can hold multiple colors, as 0-255 integer- and 0-1 float-representation, as well as the current native-color. Such colortables can be used to apply colors to track-colors or item-colors. This is still work in progress but will become more elaborated over time.

There are currently multiple color-table-functions available, like: CreateColorTable, CreateSonicRainboomColorTable, IsValidColorTable, ApplyColorTableToTrackColors

Over time, I intend to add functions to these colortables as well, so adding, removing, altering and changing colors in the colortable(like brightness, saturation and such) is possible more easily.



^ Color Management: Creating Colortables

Let's create such a new ColorTable:

CreateColorTable

                array ColorTable = ultraschall.CreateColorTable(integer startr, integer startg, integer startb, 
                                                                integer endr, integer endg, integer endb, integer number_of_steps)

This creates a new ColorTable with colors from a given color-range. You set startr, startg, startb to the first color, endr, endg and endb to the last color and the number_of_steps from the first to the last color.
After that, you'll have a ColorTable with number_of_steps-colors from startcolor to endcolor. So the following code returns a colortable with 50 shades of gray from black to white:

                ColorTable = ultraschall.CreateColorTable(0, 0, 0, 255, 255, 255, 50)

If you've created multiple ColorTables and want to have them combined into one, just combine them using the function ConcatIntegerIndexedTables.

CreateSonicRainboomColorTable

                array ColorTable = ultraschall.CreateSonicRainboomColorTable()
                

This is a simple function, that creates a ColorTable in Ultraschall's "Sonic Rainboom"-style.

If you want to check, whether a certain table is a valid ColorTable, you can easily check this, using the function IsValidColorTable:

IsValidColorTable

                boolean retval = ultraschall.IsValidColorTable(array ColorTable)
                

Simply pass a table to check for and it will return, whether it is a valid ColorTable(true) or nor(false)



^ Color Management: Applying Colortables

Now, that we've created a new ColorTable, we might want to apply it somewhere, like tracks or items. For that, we have: ApplyColorTableToTrackColors and ApplyColorTableToItemColors

Let's bring some colors into our lives:

ApplyColorTableToTrackColors

                boolean retval = ultraschall.ApplyColorTableToTrackColors(array ColorTable, integer Spread, integer StartTrack, integer EndTrack)

This applies a ColorTable to tracks to colorize them. To do that, create a Colortable and pass it to this function to parameter ColorTable.
The parameter Spread decides, whether to apply a ColorTable once(0), or cyclic/repeating(1 or nil) or whether to spread the colors over all tracks equally(2).
The final parameters decide, to which tracks to apply this, starting from track StartTrack to EndTrack. Use nil as StartTrack to automatically use the first, nil as EndTrack to automatically use the last track in the project.

ApplyColorTableToItemColors

                boolean retval = ultraschall.ApplyColorTableToItemColors(array ColorTable, integer Spread, MediaItemArray MediaItemArray)
                

This applies a ColorTable to MediaItems and works basically as the ApplyColorTableToTrackColors-function above, with the exception, that you pass MediaItems as MediaItemArray.
The rest is the same, pass to the function a ColorTable and in the parameter Spread, how to spread/cycle the colors over the MediaItems.



^ Background Scripts: Introduction

Some things in Reaper can't be solved easily without having something monitoring in the background. So to include some new features otherwise impossible, I added some background-scripts, that can be easily run using:

RunBackgroundHelperFeatures

                ultraschall.RunBackgroundHelperFeatures(true)
                

This starts the background-scripts, that provide additional features. To stop them again, set the parameter to false.

Without having the background-scripts started, some functions will always produce error-codes.
Features that use these background-scripts are:

GetLastCursorPosition - gets the last editcursor-position before the current one. Helpful for left-click-triggered scripts, who change the editcursorposition due the mouseclick
GetLastPlayState - gets the last playstate before the current one. Helpful for monitoring "event-based"-statechanges
GetLastLoopState - gets the last loopstate before the current one. Helpful for monitoring "event-based"-statechanges

With this, you can create event-managers, reacting to e.g. playstate-changes.

Such a script could look like this:

    -- Ultraschall-API demoscript by Meo-Ada Mespotine 26.02.2019
    -- 
    -- a simple event-manager, that employs background-helper-scripts for listening for statechanges and
    -- the Defer-functions in Ultraschall-API, who allow you to control, how often the deferred code
    -- shall be executed (every x seconds/every x defer-cycles).

    dofile(reaper.GetResourcePath().."/UserPlugins/ultraschall_api.lua")

    -- the following code will stop the Background-listener-scripts when script is exited
    function exit()
      ultraschall.RunBackgroundHelperFeatures(false)
    end

    reaper.atexit(exit)


    -- start background-listener scripts
    ultraschall.RunBackgroundHelperFeatures(true)

    function main()
      -- get the last/current playstates
      last_play_state, new_play_state, statechangetime = ultraschall.GetLastPlayState()
      
      -- if the state has been changed in general since last time calling GetLastPlayState:
      if statechangetime~=oldstatechangetime then
        -- show the old and new playstate in the console
        print("Old Playstate: "..last_play_state.." - New Playstate: "..new_play_state)
        
        -- if the state has changed from REC or RECPAUSE to STOP then
        if (last_play_state=="RECPAUSE" or last_play_state=="REC") and new_play_state=="STOP" then
          -- show a messagebox that displays that recording has stopped
          print2("Recording or Rec+Pause has stopped!")
        end
      end  
      
      -- store the current statechangetime as the old one for comparison in the next defer-cycle
      oldstatechangetime=statechangetime
      
      -- run the function main in mode 2(every x seconds) every 1 second
      -- Defer1(functionname, mode(0, normaldefer; 1, every x cycles; 2, every x seconds), 
      -- counter(number of seconds or cycles))
      ultraschall.Defer(main,"Simple Event Manager", 2,1)
    end

    -- run the fun
    main()
    

This script displays into the ReaScript-console the message "Recording or Rec+Pause has stopped!" everytime recording and/or recording+pause changes to STOP It also displays playstate-changes in general, when they happen, into the ReaScript-console.

Refer DeferScripts: Introduction to Ultraschall's Defer-functions for more information on the Defer1-function used in this demo-script.



^ Cough and Mute Buttons/Actions: Introduction

When recording over a long period of time, especially when recording interviews for podcasts, radio and such, you probably run into the issue that someone is coughing or drinking something. Though both of these things are important, a recording of it is usually not wished.
For that, Reaper provides a Mute-Envelope for each track, which allows to mute a track for a period of time and unmute it again. This muting can be recorded, so it will remain even after the recording is finished.
Problem is: You need to "arm" the mute-envelope first, otherwise any mute-information as send by e.g. MIDI-devices will not be added to the mute-envelope and therefor is lost after the recording.
In some cases, this behavior is a good thing and gives you flexibility when to record the mutes and when not, but for casual users, who do not want to have this kind of freedom, this can be really hard to use.

So I added some functions, that add mute-points into a mute-envelope regardless of it's arming-state. This should make using the mute-envelope for your benefits much easier.

Let's go into more detail.



^ Cough and Mute Buttons/Actions: Toggling Mute

The first and foremost thing you probably want is actually muting a mute-envelope of a track, for that there are the two functions: ToggleMute, ToggleMute_TrackObject

ToggleMute

                    integer retval = ultraschall.ToggleMute(integer track, number position, integer state)
                    

This mutes a given track using the mute-envelope of the track(not the mute-buttons in the TCP/MCP).
Just pass over the tracknumber(with 1 for the first track, 2 for the second, etc!), the position, at which to insert the mute-point and the state of muting.
The state of muting can be either muted(0) or unmuted(1)

To mute a certain MediaTrack-object, you can use the function:

ToggleMute_TrackObject

                    integer retval = ultraschall.ToggleMute_TrackObject(MediaTrack trackobject, number position, integer state)
                    

This works the same as ToggleMute, but accepts a MediaTrack instead of a tracknumber. The rest is the same, position for the position of the new mutepoint and state for mute(0) or unmute(1).



^ Cough and Mute Buttons/Actions: Toggling Mute

Toggling is cool, but you probably want to know, if toggling is even necessary of where the next/previous mute-object currently is.
For that, I added the functions: IsMuteAtPosition, IsMuteAtPosition_TrackObject, GetNextMuteState, GetNextMuteState_TrackObject, GetPreviousMuteState, GetPreviousMuteState_TrackObject

If you want to know, how many mute-points exist in a track, you can use:

CountMuteEnvelopePoints

                    integer retval = ultraschall.CountMuteEnvelopePoints(integer track)
                    

This counts the envelope-points of a mute-envelope in a given track.

To check, if there's already a mute-point at a given position in a track, you can use the IsMuteAtPosition-function.

IsMuteAtPosition

                    boolean retval, optional integer envIDX, optional number envVal  = ultraschall.IsMuteAtPosition(integer tracknumber, number position)
                    

This checks, whether there is a mute-point in track tracknumber at position. Parameter tracknumber is 1-based, with 1 for the first track, 2 for the second, etc.
If there is a mutepoint at the position, it will return the index of the mute-point and it's current value.
It will return false in case of an error

With this, you can check, whether adding/toggling at the position is actually necessary.

If you want to know, which is the next or previous mute-point, you can use:

GetNextMuteState

                    integer envIDX, number envVal, number envPosition = ultraschall.GetNextMuteState(integer track, number position)
                    

Returns the attributes of the next mute-point from position. Just give the tracknumber and the position and it returns the corresponding indexnumber, value and position.
It will return -1, if no such mute-point exists

GetPreviousMuteState

                    integer envIDX, number envVal, number envPosition = ultraschall.GetPreviousMuteState(integer track, number position)
                    

Returns the attributes of the previous mute-point from position. Just give the tracknumber and the position and it returns the corresponding indexnumber, value and position.
It will return -1, if no such mute-point exists

If you want to use a trackobject rather than the tracknumber, you can use the functions:

With that, you should get a good overview over the mute-points in your project.



^ Cough and Mute Buttons/Actions: Toggling Mute

Being able to set and toggle and find mute-points is quite good, but sometimes, you want to get rid of them as well.
For that, I added the functions: DeleteMuteState, DeleteMuteState_TrackObject

DeleteMuteState

                    boolean retval = ultraschall.DeleteMuteState(integer tracknumber, number position)
                    

This deletes a mute-point in track with tracknumber at position. It returns false, if there is no mute-point to delete. Parameter tracknumber is 1-based, with 1 for track 1, 2 for track 2, etc.

To delete a mute-point using a MediaTrack-object, you can use instead:

DeleteMuteState_TrackObject

                    boolean retval = ultraschall.DeleteMuteState_TrackObject(MediaTrack MediaTrack, number position)
                    

Works exactly like DeleteMuteState, but expects a MediaTrack-object instead of a tracknumber.



^ Error Messaging System: Introduction

When coding, one of the most boring things, beside documentation, is debugging.
This is especially true, when the development-system used itself creates error- and bugmessages from hell, that just tell you that something went wrong, but not what.

And as I'm known to be lazy, I thought "When creating many API-functions, they should be able to tell me, where I went wrong, what went wrong and when I'm holding it wrong."

Thus, I created an error-messaging-system that tells you exactly that: Which parameter caused which issue and a hint in how to solve this, if needed.
And furthermore, I worked this error-messaging system out in a way, that it does not stop script-execution if you do not want to. In fact, if you want to reuse error-messages created by an Ultraschall-API-function you used, you can do that.
So using error checks from other Ultraschall-API-functions for your benefits is quite easy.

Let's go into more details on that.



^ Error Messaging System: Creating Error Messages

Before you can use error-messages, they need to be created first. For that, you can use the function: AddErrorMessage.

Let's create an error-message.

AddErrorMessage

                    boolean retval, integer errorcount = 
                        ultraschall.AddErrorMessage(string functionname, string parametername, string errormessage, integer errorcode)
                        

This creates a new error-message, that will be fed into the error-messaging-system.
The parameters are the functionname, the name of the parameter(set it to "" if you don't want to set this parameter), and error-message and an error-code.

The functionname is for the programmer who uses your function, so they know, which function caused this error.
The parametername is simply the name of the parameter, so if a validity-check for a parameter didn't go well, set this to the name of the problematic parameter.
The errormessage should tell the programmer, what went wrong. It should be easy to understand. Tell, in a few words, what went wrong and what was expected instead.
So if checking for an integer-parameter went wrong because the user passed a string instead, the errormessage should be "Must be an integer." The errorcode must be unique, means, every error-message in your function must get it's own errorcode, even if two error-messages are quite similar. This is to help the user to determine, which error happened, without having to check the error-message itself.
A unique errorcode has also other benefits for you as function-coder: if you want to change the error-message to be more descriptive, a check for the errorcode would still work. Another thing: a unique errorcode will help to determine possible bugs in your function more easily, as you immediately know, which errormessage caused it.
I usually use negative numbers for Ultraschall-API-errorcodes, but you are free to choose them anyway you like. But once you decided on one, you must keep it forever until it becomes obsolete!

The function returns two parameters, retval and errorcount. Returnvalue retval tells you, if adding the error-message worked well. If not(usually when passing invalid parameters) it returns false, otherwise true.
Returnvalue errorcount tells you, which number the newly created errormessage has withing the error-messaging-system. This can be used to get the errormessage directly(more on that later).

Example: Let's assume, you have a function with one parameter which shall be a string and you want to check for it to be valid:

                    function myfunction(this_is_a_string)
                        if type(this_is_a_string)~="string" then
                            retval, errorcount = ultraschall.AddErrorMessage("myfunction", "this_is_a_string", "must be a string", -1)
                            return false, errorcount
                        end
                    end
                    

This checks, if the parameter thisisa_string is actually a string. If not, it adds an error-message to the error-messaging-system.
After that, the function returns false and the index of the error-message, so the user can use this number to retrieve the error-message directly from the Ultraschall-API.

Now, we've created a new error-message, which is now kept by the Ultraschall-API-instance locally within your script.



^ Error Messaging System: Getting Error Messages

Now that we've created a new error-message, we probably want to retrieve it somehow.
For that, I added the following functions: CountErrorMessages, GetLastErrorMessage, GetLastErrorMessage2, ReadErrorMessage, ShowLastErrorMessage

ShowLastErrorMessage

                    ultraschall.ShowLastErrorMessage(optional integer dunk, optional integer target, optional integer message_type)
                    
                    or 
                    
                    SLEM(optional integer dunk, optional integer target, optional integer message_type)
                    

This is probably the most useful for debugging purposes. It simply shows the last added error-message stored in the error-messaging-system, including the functionname, the parametername, the errormessage and the errorcode. You can optionally select, which other "last" errormessage you want to display, as well as its target. That means, you can select if the error-message shall be displayed in a messagebox, the reascript-console, the clipboard or if certain error-messages shall be returned as string be the function. You can also select the default-output-target in the Ultraschall-API-settings-dialog(just look for "Script: Ultraschall_API_Settings.lua" in the actionslist). Add this at the end of your script or within a function, after the code where problems arise to see, which function caused the trouble.

GetLastErrorMessage

                    boolean retval, integer errcode, string functionname, string parmname, string errormessage, 
                    string lastreadtime, string err_creation_date, string err_creation_timestamp, integer errorcounter 
                                                                                            = ultraschall.GetLastErrorMessage()

This returns you the last error-message without opening a messagebox. It also returns attributes associated with the errormessage. Returnvalue retval returns, if an errormessage exists(true) or not(false). The other returnvalues are

  • errcode, which contains the errorcode of the errormessage. This is unique and always associated with the same error, so you can check for the errorcode directly, rather than the errormessage.
  • functionname, which contains the name of the function, in which the error occured
  • parmname, which contains the name of the parameter, which caused the trouble. It is "", if the problem is unrelated to a parameter
  • errormessage, which is the actual errormessage
  • lastreadtime, the time, at which the errormessage was read from the error-messaging-system the last time. Is "unread" if the errormessage hasn't been read yet.
  • err_creation_date, the date, at which the error was created
  • err_creation_timestamp, the time, at which the error was created
  • errorcounter, the indexnumber of the errormessage within the error-messaging-system.

GetLastErrorMessage2

                    boolean retval, array ErrorMessages = ultraschall.GetLastErrorMessage2(integer count, boolean setread)
                    

Very much like GetLastErrorMessage, but returns the values as an array. You can also get the last x error-messages(as set by parameter count) and if the read-status of an errormessage shall be set to true.
When setread is set to true, ShowLastErrorMessage will not show this errormessage, even if it's the last one.

For more sophisticated use-cases, where you need to retrieve one specific errormessage, you can use the following:

ReadErrorMessage

                    boolean retval, integer errcode, string functionname, string parmname, string errormessage, 
                    string lastreadtime, string err_creation_date, string err_creation_timestamp = 
                                                                        ultraschall.ReadErrorMessage(integer errornumber, optional boolean keep_unread)
                                                                        

Just like GetLastErrorMessage, but returns a specific errormessage, as specified by the parameter errornumber. You can also keep the error-message state as "unread" if you want to build a function that peeks into an error-message without leaving a trace.

In addition to that, there's also:



^ Error Messaging System: Deleting Error Messages

If you want to get rid of old errormessages, you can delete them using: DeleteLastErrorMessage, DeleteAllErrorMessages, DeleteErrorMessage
These functions work in the same vein, as the read-functions.

DeleteLastErrorMessage

                    boolean retval = ultraschall.DeleteLastErrorMessage()
                    

This simply deletes the last error-message from the error-messaging system. It returns true, if it succeeded.

DeleteAllErrorMessages

                    boolean retval = ultraschall.DeleteAllErrorMessages()
                    

Just like DeleteLastErrorMessage, but deletes all errormessages from the error-messaging system. It also returns true, if it succeeded.

DeleteErrorMessage

                    boolean retval = ultraschall.DeleteErrorMessage(integer errornumber)
                    

This deletes a specific errormessage, as specified by errornumber. Remember, to get the current number of error-messages available, use CountErrorMessages!

And that's all you need to know about deleting errormessages.



^ Error Messaging System: Toggling showing errors in IDE instead

As cool, as the error-messaging-system is for debugging, you may prefer getting your errors like any other Lua/Reaper-API-error at the bottom of the IDE or in Reaper's error-window.
So I included toggling exactly that, using: ToggleIDE_Errormessages

ToggleIDE_Errormessages

                    boolean retval = ultraschall.ToggleIDE_Errormessages(optional boolean togglevalue)
                    

This toggles, whether to display error-messages at the bottom of the ReaScript-IDE-window/Reaper's error-window or if you prefer using Ultraschall-API's own functions for error-handling.
Optionally, you can set it by setting togglevalue to either true(show errormessages at the bottom of the IDE) or false(use the Ultraschall-API error-management-behavior).



^ Error Messaging System: Other helpers for Error-Messaging-system

For other use-cases, I'm adding other error-messaging-system-features as well. If you need more, feel free to ask me.

ShowErrorMessagesInReascriptConsole

                    ultraschall.ShowErrorMessagesInReascriptConsole(boolean state)
                    

This toggles, whether error-messages shall be shown in the ReaScript-console, WHEN THEY HAPPEN.
That way you can immediately see, which functions complained and how often. This is more detailed, than ShowLastErrorMessage(), which only shows the last error that has happened.



^ Trackstate Management: Introduction

Many things in regards of customizations have also to do with influencing states of MediaTracks. So I added tons of stuff for that too.
I also thought, when I'm at it, I change some of the confusing behavior of Reaper's own API to something more viable.
For instance, when passing a tracknumber to a function, the first track will always be 1, the second always be 2, etc. The master-track, if applicable, will always be 0. That way, you can always be sure, which track in a function is which track.

Another thing is working with trackstrings, which allow you to easily pass over a number of tracks at once.
They are basically a string with the tracknumbers wished, separated by commas. Example:

                    trackstring_of_track1_to_3 = "1,2,3"
                    

Many Ultraschall-API-functions support trackstrings and save you from the pain of having to loop through all tracks you want to pass and influence.
There are also many functions, who create you trackstrings, like for locked tracks, selected tracks, all tracks, etc.

I talked about Trackstates earlier. For that I added full access to all(!) TrackStates available, even those, who are only available through TrackStateChunks. Access means, getting and setting them. Read more on this subject in the chapter Get/Set States for Project, Tracks and Items.

Some other things you can do now easily are:

and many more related to working with TrackStateChunks.



^ Routing: Introduction

In the past, I had numerous situations, where I wanted to influence my routing-settings programmatically. But every time I started, I stumbled and shuddered. Not because it is impossible with Reaper's own API, but because it's so complicated, irritating, confusing and so far away from the way the routing is managed with the user-interface of Reaper, where you can see easily all individual sends/receives.
This problem applies with the whole management of HardwareOuts as well, which is also painful to program.
So I desired some functions that keep programming the routing simple and down to it's basic components while retaining the flexibility of the full routing.

All these functions, HWOut as well as Routing(Send/Receive) are based on a simple concept, as seen in the user-interface of routing-settings:

In addition to that, I also added numerous functions, that allow "mass manipulation" of the Routings. Let's start with the routing.



^ Routing: Send and Receives

Before we start, some basics, if you're not familiar with routing.

Routing is a way to direct the output of one channel/track/fx to another channel/track/fx.
In our case of track-routing, that means, one track sends a signal and another track receives it. If you look into the routing-settings or routing-matrix of Reaper, you can exactly see that: one track sends a signal which can be received by one or many tracks.
The most common of such routings is: Track on sends a signal, and the master-channel receives it to output it.
This allows you to build complex audio-settings and configurations, where one output-signal can be send to a reverb-effect, while at the same time being send to a flanger; in parallel, so the reverbed signal isn't flanged, vice versa.

One more thing: To make send/receive work, you need to understand, that you have to set it in the track, that receives the signal(means, to the track who listens to the sending track)! That means, you apply the following functions to the MediaTrack that receives the send-signal.

To do the Send/Receive-stuff, I added the following functions: AddTrackAUXSendReceives, GetTrackAUXSendReceives, SetTrackAUXSendReceives, DeleteTrackAUXSendReceives, CountTrackAUXSendReceives
Let's create a new send/receive-setting.

AddTrackAUXSendReceives

                   boolean retval, optional string TrackStateChunk 
                                    = ultraschall.AddTrackAUXSendReceives(integer tracknumber, integer recv_tracknumber, integer post_pre_fader, 
                                                                          number volume, number pan, integer mute, integer mono_stereo, integer phase, 
                                                                          integer chan_src, integer snd_chan, number pan_law, integer midichanflag, 
                                                                          integer automation, optional string TrackStateChunk)

This adds a receive to a specific track or TrackStateChunk. To tell the function, from which track the received signal comes from, you need to set the parameter recv_tracknumber to the tracknumber of the sending track.
The other parameters are simply all other settings for this send/receive, like

  • post_pre_fader - shall the signal be post, pre fader or even pre-fx
  • volume - the volume of the received signal
  • pan - the panning of the received signal
  • mute - shall it be muted or not
  • mono_stereo - shall the signal be mono or stereo
  • phase - the phase of the received signal
  • chan_src - the source-channel to receive from. That is the sending-track's-channel!
  • snd_chan - the track's own channel to pass the received signal to. This is in the receiving track's channels!
  • pan_law - the pan-law-setting for this AUXSendReceive
  • midichanflag - manages the whole MIDI-receiving settings, like to receive from which MIDI-source-channel and to send to which of the track's own MIDI-channels. Includes also the MIDI-On-switching as well as no channels at all.
  • automation - the automation-mode of the receive
  • TrackStateChunk - set this, if you want to add AUXSendReceives to a TrackStateChunk; set tracknumber=-1 in that case

Setting a receive-setting works basically the same:

SetTrackAUXSendReceives

                    boolean retval = ultraschall.SetTrackAUXSendReceives(integer tracknumber, integer idx, integer recv_tracknumber, 
                                                                         integer post_pre_fader, number volume, number pan, integer mute, 
                                                                         integer mono_stereo, integer phase, integer chan_src, integer snd_chan, 
                                                                         number pan_law, integer midichanflag, integer automation, 
                                                                         optional string TrackStateChunk)

This works the same as AddTrackAUXSendReceives, but changes an already existing setting.

Setting is one thing, but what if you want to know the settings of a receive?

GetTrackAUXSendReceives

                    integer recv_tracknumber, integer post_pre_fader, number volume, 
                    number pan, integer mute, integer mono_stereo, integer phase, integer chan_src, 
                    integer snd_chan, number pan_law, integer midichanflag, integer automation 
                              = ultraschall.GetTrackAUXSendReceives(integer tracknumber, integer idx, optional string TrackStateChunk)

This receives all the settings of a receive of a track or a TrackStateChunk. The same rules for the former functions (AddTrackAUXSendReceives and SetTrackAUXSendReceives) apply also here.

And when you want to delete a receive, you can use the following function:

DeleteTrackAUXSendReceives

                    boolean retval = ultraschall.DeleteTrackAUXSendReceives(integer tracknumber, integer idx, optional string TrackStateChunk)
                    

This deletes a receive-setting from a track or a TrackStateChunk. Just pass to it the tracknumber, which receives and the idx of the receive-setting. To know, which setting to delete, you can use GetTrackAUXSendReceives. Set tracknumber=-1 to use the parameter TrackStateChunk to delete an AUXSendReceive from a TrackStateChunk.

If you want to know, how many receives a track has, you can use:

CountTrackAUXSendReceives

                    integer count_SendReceives = ultraschall.CountTrackAUXSendReceives(integer tracknumber, optional string TrackStateChunk)
                    

This returns the number of received tracks a the track tracknumber has.

Another thing: Routing-settings of the master-track aren't possible that way. You need to use GetTrackMainSendState and SetTrackMainSendState to route the signal of a track to the master-channel. Unlike the former functions, MainSend must be set in the sending track!



^ Routing: Hardware Outs

Send and Receive is one thing, but if you want to hear anything, you need to send the audiosignal to actual audiohardware. To do that, you can set HardwareOuts(HWOuts) to tracks, as well as the MasterTrack.
For that, I included the functions: AddTrackHWOut, SetTrackHWOut, GetTrackHWOut, DeleteTrackHWOut, CountTrackHWOuts

AddTrackHWOut

                    boolean retval = ultraschall.AddTrackHWOut(integer tracknumber, integer outputchannel, integer post_pre_fader, 
                                                               number volume, number pan, integer mute, integer phase, integer source, 
                                                               number pan_law, integer automationmode, optional string TrackStateChunk)
                                                               

This adds a new HWOut to a track or a TrackStateChunk. The tracknumber is either 0 for the master track or 1 and higher for track 1 or higher or -1, if you want to pass a TrackStateChunk.
The other options are to some extend familiar to the AUXSendReceive-functions:

  • outputchannel - the outputchannel, either single tracks or stereo-tracks
  • post_pre_fader - shall the signal be post, pre fader or even pre-fx
  • volume - the volume of the HWOut
  • pan - the panning of the HWOut
  • mute - shall it be muted or not
  • phase - the phase of the HWOut
  • source - the channel(s) to output to the outputchannels
  • pan_law - the pan-law-setting of the HWOut
  • automationmode - the automation of the HWOut
  • TrackStateChunk - the trackstatechunk, into which you want to add HWOut-settings; only available when tracknumber=-1

Setting an already existing HWOut can be done using the function:

SetTrackHWOut

                    boolean retval = ultraschall.SetTrackHWOut(integer tracknumber, integer idx, integer outputchannel, 
                                                               integer post_pre_fader, number volume, number pan, integer mute, 
                                                               integer phase, integer source, number pan_law, integer automationmode, 
                                                               optional string TrackStateChunk)

This sets the settings of an already existing HWOut. The rest is the same, as the parameters for AddTrackHWOut.

To set HWOuts, you probably want to know, what the old values are of a HWOut. For that, use the function:

GetTrackHWOut

                    integer outputchannel, integer post_pre_fader, number volume, number pan, 
                    integer mute, integer phase, integer source, number pan_law, integer automationmode 
                                = ultraschall.GetTrackHWOut(integer tracknumber, integer idx, optional string TrackStateChunk)
                                                                

This returns the settings of a HWOut. The tracknumbers are the same, with 0 for the master-track; 1 and higher for track 1 and higher; -1 if you want to use the parameter TrackStateChunk. The idx is the index of the HWOut-setting of the track.

To delete it, use:

DeleteTrackHWOut

                    boolean retval = ultraschall.DeleteTrackHWOut(integer tracknumber, integer idx, optional string TrackStateChunk)

This deletes a HWOut-setting from a track.

To get the number of HWOuts of a certain track, use:

CountTrackHWOuts

                    integer count_HWOuts = ultraschall.CountTrackHWOuts(integer tracknumber, optional string TrackStateChunk)


^ Routing: Mass manipulation of Routings

Sometimes, working with individual settings isn't enough. Sometimes, you want to work with all Routing-Settings at once.
For that, I included the functions: ClearRoutingMatrix, GetAllHWOuts, ApplyAllHWOuts, AreHWOutsTablesEqual, GetAllAUXSendReceives, ApplyAllAUXSendReceives, AreAUXSendReceivesTablesEqual, GetAllMainSendStates, ApplyAllMainSendStates, AreMainSendsTablesEqual

ClearRoutingMatrix

                    boolean retval = ultraschall.ClearRoutingMatrix(boolean ClearHWOuts, boolean ClearAuxRecvs, boolean ClearTrackMasterSends, 
                                                                    boolean ClearMasterTrack, boolean undo)
                                                               

This clears the routing-matrix. You can control, what parts of the routing-matrix shall be cleared(MainSends, AUXSendReceives, HWOuts) and if this shall include the master-track as well.

To mass-manipulate all HWOuts, see the following functions:

GetAllHWOuts

                    table AllHWOuts, integer number_of_tracks = ultraschall.GetAllHWOuts()
                                                               

This returns all HWOut-settings as a nice handy table, which you can alter. The name of each index reflects the name of the parameters of the add/get/set-HWOut-functions.
returned table is of structure:
table["HWOuts"]=true - signals, this is a HWOuts-table; don't change that!
table["number_of_tracks"] - the number of tracks in this table, from track 0(master) to track n
table[tracknumber]["HWOut_count"] - the number of HWOuts of tracknumber, beginning with 1
table[tracknumber]["TrackID"] - the unique id of the track as guid; can be used to get the MediaTrack using reaper.BR_GetMediaTrackByGUID(0, guid)
table[tracknumber][HWOutIndex]["outputchannel"] - the number of outputchannels of this HWOutIndex of tracknumber
table[tracknumber][HWOutIndex]["postprefader"] - the setting of post-pre-fader of this HWOutIndex of tracknumber
table[tracknumber][HWOutIndex]["volume"] - the volume of this HWOutIndex of tracknumber
table[tracknumber][HWOutIndex]["pan"] - the panning of this HWOutIndex of tracknumber
table[tracknumber][HWOutIndex]["mute"] - the mute-setting of this HWOutIndex of tracknumber
table[tracknumber][HWOutIndex]["phase"] - the phase-setting of this HWOutIndex of tracknumber
table[tracknumber][HWOutIndex]["source"] - the source/input of this HWOutIndex of tracknumber
table[tracknumber][HWOutIndex]["pan_law"] - pan-law, default is -1
table[tracknumber][HWOutIndex]["automationmode"] - the automation-mode of this HWOutIndex of tracknumber

You can manipulate these entries now. For the valid value-ranges, see the GetTrackHWOut-function.
After that, you can add them back to the Routings at once, using:

ApplyAllHWOuts

                    boolean retval = ultraschall.ApplyAllHWOuts(table AllHWOuts, optional integer option)
                                                               

This applies the settings of the AllHWOuts-table to all the tracks.
The parameter option allows you to set, whether the settings shall be applied by tracknumber or by track-guid.
Each MediaTrack in a project has it's own ID, stored as a 64-digits-string. This allows one to know a certain track, even if the order or the trackname change.
If you choose option=2, this function will look into the table and retrieve the stored guid of a track and apply the setting to exact this track. If you choose option=1 or option=nil, then it will apply it to the tracknumber, even if the tracknumber doesn't reflect the tracknumber of the track with the guid.
So depending on how much you want to "hook" the settings to a specific track or a tracknumber, you set option to 2 or to nil.

Let's assume, you have two AllHWOuts-tables and want to compare, if they are the same or if they are different, you can use:

AreHWOutsTablesEqual

                    boolean retval = ultraschall.AreHWOutsTablesEqual(table AllHWOuts, table AllHWOuts2, optional integer option)
                                                               

This compares two AllHWOut-tables and returns false, if they are different.
With the parameter option, you can choose, whether to check for the guids as well(option=2) or if they shall be ignored in the comparison(option=nil).

The same principles can be used for MainSendStates and AUXSendReceives, who also have the same functions:

Regarding the guid, versus tracknumber-difference

To get an idea of the differences between applying settings by guid versus applying settings by tracknumbers, consider the following example:
Just create a new project with 5 tracks in them and name them A, B, C, D and E.
The routing-matrix could look something like this:

Routing Matrix - before

Now, we look at the following two code-snippets, who get all MainSend-states of a track, add two new tracks before the first track, toggle the states of the MainSends in the AllMainSends-table and reapply them. The only difference is, how they use the parameter option.

Using option=nil:

        dofile(reaper.GetResourcePath().."/UserPlugins/ultraschall_api.lua")

        -- get all current MainSendStates
        AllMainSends_Table, NumberOfTracksInTable = ultraschall.GetAllMainSendStates()
        -- insert two track before the first track
        ultraschall.InsertTrackAtIndex(0, 2, true)

        -- toggle the Main-Sends-states.
        if AllMainSends_Table[1]["MainSendOn"]==0 then newstate=1 else newstate=0 end
        for i=1, NumberOfTracksInTable do
          AllMainSends_Table[i]["MainSendOn"]=newstate
        end

        -- apply them to all track using their tracknumbers(option=nil)
        ultraschall.ApplyAllMainSendStates(AllMainSends_Table,nil)

This changes the main-send-states, beginning with Track 1, although the first two tracks are not A, B, C, D or E.

The resulting routing-matrix will look like this:
Routing Matrix - after, using option=nil

Using option=2:

        dofile(reaper.GetResourcePath().."/UserPlugins/ultraschall_api.lua")

        -- get all current MainSendStates
        AllMainSends_Table, NumberOfTracksInTable = ultraschall.GetAllMainSendStates()
        -- insert two track before the first track
        ultraschall.InsertTrackAtIndex(0, 2, true)

        -- toggle the Main-Sends-states.
        if AllMainSends_Table[1]["MainSendOn"]==0 then newstate=1 else newstate=0 end
        for i=1, NumberOfTracksInTable do
          AllMainSends_Table[i]["MainSendOn"]=newstate
        end

        -- apply them to all track using their guids(option=2)
        ultraschall.ApplyAllMainSendStates(AllMainSends_Table,2)

This changes the main-send-states only at tracks A, B, C, D, E, leaving the newly inserted tracks untouched.

The resulting routing-matrix will look like this:
Routing Matrix - after, using option=2



^ ExtState Management: Introduction

One of the concepts of Reaper to exchange data between scripts, functions, instances are so called extstates.
Such extstates are usually key-value-stores, in which you can store strings. They can be stored either as ini-files or in Reaper's memory itself.

In the Ultraschall-API, I added numerous functions to get/set/enumerate values in ini-files as well as analyzing them for certain patterns and information.
You can find all functions related to storing/analyzing ini-files in the index of the functions-reference Configuration-Files Management -> Ini-Files.

Another thing I added is extstate-management for tracks and items. That means, you can easily store additional information about MediaItems or MediaTracks in your project, just like ItemNotes or ProjectNotes, but more flexible.
That way, you can do things like storing additional metadata for items, without having to store them into the item-notes field.
You can find all these functions for item/track-extstates in the functions-reference-index under Metadata Management -> Extension States(Guid) and Extension States

Let's go into more detail on these concepts.



^ ExtState Management: Ini-Files

Ini-files are an easy way to store information. Many of Reaper's own configuration-files are ini-files.
They all work under the same structural concept:

    [section]
    key=value
    anotherkey=anothervalue
    
    [anothersection]
    morekeyadditions=with a value
    thelaskey=and the last value
    

As you can see, you have one or more sections in the file, usually written [sectionname], with one or more key=values associated with them.

Sections are used to create a basic semantical structure in the ini-file.
So if you want to create an ini-file for yourself, a sectionname should reflect a little, what kind of nature the key-values are, that you store in such a section.
The same goes for the keys, but their names should be orientated on the nature of the value stored with this specific key.

Names of sections should be unique within a file. Names of keys should be unique within a section(!), values can be anything.
You should avoid an = in a section name and brackets [ ] in a keyname, as this could confuse reading of an ini-file.

You can create such ini-files by hand and edit them by hand.

These key-value-stores can be read and set using Ultraschall-API-functions.

Note: Reaper's own configuration-files can be changed only, when Reaper isn't running. To be more precise: you can change them, but Reaper will ignore these changes until restart, sometimes even overwrite them.
Changing Reaper's config-files needs a restart of Reaper for the changes to take effect.
Some of the configuration-settings can be set also at runtime, but need another approach, which you can find on the Reaper-Internals-page for Configuration Variables



^ ExtState Management: Inifile-Functions

Let's start with setting and getting key-values in ini-files.

GetIniFileExternalState

                   string value = ultraschall.GetIniFileExternalState(string section, string key, string ini_filename_with_path)
                   

This gets a value stored in a key within a specific section. Just pass to the function the name of the section, the name of the key and the ini-file, from where to read the value.

Setting a key-value is also quite simple. Just use:

SetIniFileExternalState

                   boolean retval = ultraschall.SetIniFileExternalState(string section, string key, string value, string ini_filename_with_path)
                   

This sets a value into a specific key within a specific section. Just pass over sectionname, keyname, the value to be stored(as string!) and the filename with path of the ini-file.

This is easy, but sometimes you have to deal with unknown ini-files or want to search through it for sections, keys and values stored inside of ini-files, without haveing to guess.
For that, I added numerous functions for:

Count Ini-File-ExtStates

Let's count extstates first. There are two kinds of counting-functions, two for counting sections in the ini-file and two for counting the keys in a specific section.
They also split into two functions, with one for counting all of them, with the other only counting the ones, that contain a specific string in them, so e.g. you can count the sections with the string ultra in them.
Let's have a look at some of them:

CountIniFileExternalState_sec

                   integer sectioncount = ultraschall.CountIniFileExternalState_sec(string ini_filename_with_path)
                   

This counts all sections contained within the ini-file.

CountSectionsByPattern

                   integer number_of_sections, string sectionnames 
                                       = ultraschall.CountSectionsByPattern(string pattern, string ini_filename_with_path)
                                       

This counts all sections in the ini-file, that have a specific string in them, specified by the parameter pattern.
It also returns a list of all sections with the pattern in them found.

In addition to that, I also added a function, that allows you to count the values within an ini-file including a certain pattern

CountValuesByPattern

                   integer number_of_keys, string sections_and_keys 
                                           = ultraschall.CountValuesByPattern(string pattern, string ini_filename_with_path)
                                           

Pass to it the pattern to look for during counting and the ini-filename with path.
It also returns a string with the sections, keys and values found, that fit the pattern.
It is a comma-separated string of the format: [section1],key1=,value,key4=,value,[section4],key2=,value

The other functions work quite the same, but focused on a key within a specific section.

Enumerate Ini-Files by number

These functions are meant to enumerate within sections and keys.

EnumerateIniFileExternalState_sec

                   string sectionname = ultraschall.EnumerateIniFileExternalState_sec(integer number_of_section, string ini_filename_with_path)

This enumerates the sections within an ini-file. You just pass the index-number to the function into parameter number_of_section and the ini-filename with path and it returns the appropriate sectionname.

EnumerateIniFileExternalState_key

                   string keyname = ultraschall.EnumerateIniFileExternalState_key(string section, integer number, string ini_filename_with_path)
                   

This enumerates the keys within a given section. Just pass over to it the sectionname, the index-number of the key and the ini-filename and it returns the correct name.

Enumerate Ini-Files by pattern

These functions are meant to to enumerate sections, keys and values in a file, that follow a certain name-pattern.

EnumerateSectionsByPattern

                   string sectionname = ultraschall.EnumerateSectionsByPattern(string pattern, integer id, string ini_filename_with_path)
                   

This enumerates the name of a section, that follows a certain pattern. Just pass over to it the string-pattern the name shall contain, the index of the section you want to have returned and the ini-filename. Example:

                   string sectionname = ultraschall.EnumerateSectionsByPattern("Hawaii", 3, "c:\\test.ini")
                   

This returns from the file c:\test.ini the third section, that contains the string Hawaii in it.

EnumerateKeysByPattern

                   string keyname = ultraschall.EnumerateKeysByPattern(string pattern, string section, integer id, string ini_filename_with_path)
                   

This enumerates the name of a key within a certain section, that follows a certain pattern. Pass to it the string-pattern, the sectionname, the index of the key you want to have returned and the ini-filename. Like with EnumerateSectionsByPattern, you can use the parameter id to determine, that only the id'th key shall be returned, that follows the string-pattern.

EnumerateValuesByPattern

                   string value, string keyname = ultraschall.EnumerateValuesByPattern(string pattern, string section, 
                                                                                       string id, string ini_filename_with_path)

Just like EnumerateKeysByPattern, but this enumerates through the values within a given section. It will return the value and it's corresponding keyname found.



^ ExtState Management: Ultraschall.ini

The Ultraschall-Podcasting-Extension itself makes use of it's own configuration-file, called ultraschall.ini, which is located in the resourcesfolder of Reaper.
This file can be used by you as well, even if it's mainly intended for an installed Ultraschall-installation. The Ultraschall-API has some functions to deal with the ultraschall.ini.

You need to know, however: sectionnames must contain a leading signature, that is used purely by you, like MaxMiller_sectionname.
All signature variants of Ultraschall_ or US_ or ULT_ (no matter if upper/lower/camelcase) as well as sections with no leading signature are reserved for us.
That way, you don't risk naming conflicts with Ultraschall's own config-settings.
This is only for the ultraschall.ini-file, all other ini-files can be used by you the way you want.

The functions themselves basically work, as the functions described in the chapter ExtState Management: Inifile-Functions.



^ ExtState Management: Track and Item-Extstates

Note: This is deprecated, as Reaper will have the same feature in better in a future version!

Another concept, I introduce with the Ultraschall-API are Track and Item-Extstates.

To store them, I make use of the fact, that tracks and items have their own guids. I use these guids as basis for the section-name, while the key and value can be freely chosen by you.
A valid guid is a string that follows the following pattern {........-....-....-....-............} where . is a hexadecimal value(0-F) These extstates are stored as ProjectExtStates, that means: when you save the project, these extstates will be saved with it.

To make this possible in general, I included two functions: GetGuidExtState, SetGuidExtState
Let's go into more detail:

SetGuidExtState

                  integer retval = ultraschall.SetGuidExtState(string guid, string key, string value, integer savelocation, 
                                                               boolean overwrite, boolean persists)

This sets an extstate using a guid. So if you have an object, that provides a guid, use that one. You can add additional characters before or after the guid, but it must contain a valid guid.
After that, set the key and the value.
With this function, you can also choose, whether to store the extstate as a global extstate that can be used from everywhere in Reaper or only within the current project.
You can choose, whether to overwrite it, if it already exists and if the state shall persist after existing Reaper(only when storing it as a global extstate).

To get such a stored state, just use:

GetGuidExtState

                  integer retval, string value = ultraschall.GetGuidExtState(string guid, string key, integer savelocation)
                  

With this one, you can simply get this extstate again. Just pass to it the guid, the key and the savelocation(either global or in the project) and it returns a success-indicator-return value as well as the stored value itself.

With these two functions as a base, I made variants for tracks and items.

One important thing to mention: these extstates are bound to the project(if they aren't stored globally via parameter savelocation), so when copying an item with extstates into another project, the extstates will not be part of the new project.
I will look into that problem at some point.



^ ExtState Management: Track Extstates

**Note: This is deprecated, use Reaper's own reaper.GetSetTrackSendInfo_String instead**

Let's store some extstates for tracks. For that, I added the functions: GetTrackExtState, SetTrackExtState
TrackExtStates are stored as ProjectExtState within the projectfile.

SetTrackExtState

                  boolean retval = ultraschall.SetTrackExtState(MediaTrack track, string key, string value, boolean overwrite)
                  

This sets an extstate to a certain MediaTrack. Just pass to it a MediaTrack-object, a key and the value to store. You can also decide, whether you want to overwrite an already existing trackextstate.

GetTrackExtState

                  boolean retval, string value = ultraschall.GetTrackExtState(MediaTrack track, string key)

This returns a stored trackextstate. Just pass to it a MediaTrack-object and the key, whose value you want. It will return if getting the value was successful or not with return-value retval, as well as the value with value.



^ ExtState Management: Item Extstates

**Note: This is deprecated, use Reaper's own reaper.GetSetMediaItemInfo_String instead**

Let's store some extstates for items. For that, I added the functions: GetItemExtState, SetItemExtState
ItemExtStates are stored as ProjectExtState within the projectfile. These functions work basically like the ones for TrackExtStates

SetItemExtState

                  boolean retval = ultraschall.SetItemExtState(MediaItem item, string key, string value, boolean overwrite)
                  

This sets an extstate to a certain MediaItem. Just pass to it a MediaItem-object, a key and the value to store. You can also decide, whether you want to overwrite an already existing itemextstate.

GetItemExtState

                  boolean retval, string value = ultraschall.GetItemExtState(MediaItem item, string key)

This returns a stored itemextstate. Just pass to it a MediaItem-object and the key, whose value you want. It will return if getting the value was successful or not with return-value retval, as well as the value with value.



^ ExtState Management: Marker Extstates

Sometimes, you would love to store additional information into markers/regions, but you refrain from that, as you don't want to risk overcrowding the marker/regiontitle.
So I added some functions, who work like regular extstates, but: for markers/regions: GetMarkerExtState, SetMarkerExtState

SetMarkerExtState

                  integer retval = ultraschall.SetMarkerExtState(integer index, string key, string value)
                  

This function allows you to store an extstate to a marker. Its usage is simple: pass to it the marker_region-index(1-based) a keyname and its associated value.
Important: the marker_region-index is the index, which covers all markers and regions at the same time.
You can use functions like reaper.EnumProjectMarkers to get the appropriate marker-indices. You just need to increment the index by 1 for usage with SetMarkerExtState's parameter index.
Toy around with it, you quickly get the idea.

GetMarkerExtState

                  string value = ultraschall.GetMarkerExtState(integer index, string key)

This returns a stored markerextstate. Just pass to it the index of a marker/region and the key, whose value you want.



^ Markers and Regions: Introduction

In the Ultraschall-extension, we make heavy use of markers and regions for all kinds of things. So, to simplify programming for markers, I added a number of functions to deal with them.

The markers used by Ultraschall and represented in the Ultraschall-API, are: normal markers, chapter markers, planned chapter markers, edit-markers, edit-regions, time-markers and PodRangeRegions.

These kind of markers get the most attention, though regular marker/region/tempo-time signature marker-management is part of this API as well.



^ Markers and Regions: General How To

All functions for all kinds of markers settable in the Ultraschall-Api share the same basic subset of workflows.
All markers can be added, set, deleted, enumerated, counted, imported and exported from/to a file, checked for being a certain markertype and converted into a certain markertype(if possible).
Let's take edit-markers as an example, who have the functions: AddEditMarker, SetEditMarker, DeleteEditMarker, EnumerateEditMarkers, CountEditMarkers, ImportEditFromFile, ExportEditMarkersToFile, IsMarkerEdit, MarkerToEditMarker, EditToMarker

AddEditMarker

                  integer marker_number, string guid = ultraschall.AddEditMarker(number position, integer shown_number, string edittitle)
                  

This adds a new edit-marker. Give it a position, the shown number and an edittitle, which should hint at the reason of having to edit later.
The title will be shown in the marker as "_edit: Title". You can change it manually in Reaper, but should retain "_edit: ", as this signals the
Ultraschall-API, if this is an edit-marker. The editmarker also is red (255 0 0).
It will return the marker-number, which is the index of all markers in the project, means: markers and regions, as well as a unique guid for this marker, which can be used for ExtStates, etc.

SetEditMarker

                  boolean retval = ultraschall.SetEditMarker(integer edit_index, number position, integer shown_number, string edittitle)
                  

This sets an already existing edit-marker. Pass to it the index-number of the edit-marker, which is the index for edit-markers only(!).
The rest is like AddEditMarker.
It will return the true, if setting it was successful.

DeleteEditMarker

                  boolean retval = ultraschall.boolean retval = ultraschall.DeleteEditMarker(integer edit_index)
                  

This deletes an edit-marker. Pass to it the index-number of the edit-marker, which is the index for edit-markers only(!). That means, if you have
five markers with the last one being an edit-marker and you want to delete that, the number is 1, not 5!
It will return the true, if deleting it was successful.

EnumerateEditMarkers

                  integer retnumber, integer shown_number, number position, string edittitle, string guid 
                                                             = ultraschall.EnumerateEditMarkers(integer edit_index)
                  

This enumerates an edit-marker and all of it's attributes. Just pass over to it the edit-index-number and it will return the overall marker-index-number(markers and regions)
the shown-number as well as it's position and the title of the edit(without the _edit: ) and the guid of the edit-marker.

CountEditMarkers

                  integer number_of_edit_markers = ultraschall.CountEditMarkers()
                  

This returns the number of edit-markers in your project. With that information, you know, how many of these editmarkers can be enumerated.

ExportEditMarkersToFile

                  integer retval = ultraschall.ExportEditMarkersToFile(string filename_with_path, number PodRangeStart, number PodRangeEnd)
                  

This exports the edit-markers into an exportfile. You can also pass to it from which startposition to which endposition the markers shall be exported.
The file created contains the attributes for one edit-marker each line. The format for each line in the file is: hh:mm:ss.mss Title
This file is generic, so it can be reimported as other types or markers, like normal markers, as well.
Speaking of importing:

ImportEditFromFile

                  array editmarkers = ultraschall.ImportEditFromFile(string filename_with_path, PodRangestart)
                  

This imports a markerexportfile, as created by functions like ExportEditMarkersToFile into the project. Podrangestart is for adding an offset to the edit-marker-positions.

IsMarkerEdit

                  boolean retval = ultraschall.IsMarkerEdit(integer markerid)
                  

Checks, if a certain marker is an edit-marker or not. The markerid is the index for all markers and regions, not only edit-markers.
Returns true, if it's an edit-marker, false if not.

MarkerToEditMarker

                  integer idx, integer shown_number, number position, string markertitle = ultraschall.MarkerToEditMarker(integer markerindex)
                  

This converts a marker into an edit-marker, which usually means, it colors it red (255, 0, 0) and adds "_edit: " at the beginning of the title.
The markerindex is the one for all markers. It returns the overall-marker-indexnumber, the shown number as well as it's position and the new markertitle.

EditToMarker

                  integer idx, integer shown_number, number position, string markertitle = ultraschall.EditToMarker(integer edit_index)
                  

This also converts, but this time the other way around: From edit-marker to a normal marker, which means: coloring it with the standard-color and removing the _edit: from the beginning of the title.

The other marker-types follow the same lead, so if you understood it for this marker, you understood it for all others as well.
One limitation though, you can't (yet) export/import regions into/from an exportfile.



^ Markers and Regions: Helpers and Manipulation

Adding and getting markers is quite good, but sometimes, you want to do more with them. For that, I added numerous functions as well, like:



^ Markers and Regions: Custom-Markers and Custom-Regions

Even if I already included many markertypes, sometimes this isn't enough. In fact, you might want to have other markertypes beside _Edit: or _PodRange:
For that, I added custom-markers and custom-regions.

A custom-marker/custom-region is like a normal marker/region, but their name start with a certain signature. You may have seen it from edit-markers already.

        _Edit:

Which is basically a custom-marker of the type Edit. Such a signature starts with an underscore _ and ends with a : Custom-markers and custom-regions work like the same, but you can insert your own custom-marker-name into it.
For example, if you want to have a custom-marker-type called foobar, it would look like:

        _foobar:

Now, doing that manually is very inconvenient to do, so I made a bunch of functions, who do that for you. All you need to do is to declare the name of the custom-marker/region and pass that over to the function and it does the rest.

Let's take the example above with foobar and use ultraschall-functions for adding such a marker.

        retval, markernumber, guid = ultraschall.AddCustomMarker("foobar", 20, "I am the text of this marker", 1, 0)
    

This would produce a custom-marker at position 20. And it's markertext would look like:

        _foobar: I am the text of this marker
    

To set this to another text, you could use the following function. As this is our only foobar-custom-marker, we use 0 as it's index:

        retval = ultraschall.SetCustomMarker("foobar", 0, 20, "I am the new text, pinky!", 1, 0)
    

This results in a custom-marker name:

        _foobar: I am the new text, pinky!
    

Now here's the thing: you can add other foobar-markers as well, can set them, enumerate them, count them, just as a regular marker.
The only difference being, that with custom-markers, you tell the functions, which name a certain custom-marker-type shall have, like foobar in our example.
Now you can work, manipulate, and work with all foobar-custom-markers, without having to deal with other ones.

The only thing you need to take care of: when deleting numerous custom-markers, you should start with the last one toward the first, as otherwise you will delete markers you wanted to keep.

The things I said about custom-markers are also valid for custom-regions. They work like custom-markers, with the custom-marker-name as starting signature in their name. The only difference from markers: it is regions.

The following functions I added to the Ultraschall-API for:

Markers:

Regions:



^ Child Scripts: Introduction

One of the cool features in Reaper is the possibility to run scripts at your mouse- or keyclick. One of the biggest drawbacks though, is that you can't individualise the scripts, or better: you can pass parameters to scripts to tell them, how to behave.
There are some very limited ways of getting input, like some MIDI-values or current states of many Reaper-elements, but they are either very limited, or they involve a heavy load of guessing what the states mean and hoping, you guessed right.

It also annoyed me, so I had for some time the idea of child-scripts in my mind.

Child scripts in the sense of the Ultraschall-API, are scripts, that can be run from another script, BUT you can also pass parameters and returnvalues back and forth. Even better, if you run a child-script, you will also get a unique datastructure called ScriptIdentifier, which is an identifier for a script that can be used as extstate to pass over information back and forth. And as this identifier is unique, you can run two or more instances of a defer script and communicate with all of them individually, as they all have their own ScriptIdentifier that they can listen to. More on ScriptIdentifiers in the next chapter.

With that, you can finally pass over as many parameters/returnvalues/other information over to a child-script as you want.



^ Child Scripts: The unique ScriptIdentifier

Every script, that uses the Ultraschall-API, gets its own ScriptIdentifier, that is something like a unique name that your script. Such a ScriptIdentifier can then be used to communicate with that given script. So, if you want to run multiple instances of a defer-script and this script uses the Ultraschall-API, you can use these script-identifiers to communicate with each of these instances without disturbing the other instances of the defer-script that are running at the same time.

A ScriptIdentifier is of the following format:

    ScriptIdentifier:path/scriptname-{guid}.lua
    

For example, if I have a script called my_script.lua in the scripts-folder of my Reaper-installation and this Reaper-installation is located at C:\Reaper, the ScriptIdentifier could look like this:

    ScriptIdentifier:C:/Reaper/Scripts/my_script-{E0020927-08C4-4BE3-B8C6-4A52D042ED60}.lua
    

Why do I say, "could" look like this. You may have noticed, theres this cryptic string "{E0020927-08C4-4BE3-B8C6-4A52D042ED60}" inside of the ScriptIdentifier.
This is a so called guid, which is a unique number. This can be created by a Reaper-function called reaper.genGuid("") and has the cool feature, that it is quite unique.
And this will be created new, every time a script is started, that uses the Ultraschall-API to create this unique ScriptIdentifier, making this scriptinstance unique in itself.

Now here's the thing: If you know this ScriptIdentifier, you can pass information to it, like with using extstates:

        reaper.SetExtState("ScriptIdentifier:C:/Reaper/Scripts/my_script-{E0020927-08C4-4BE3-B8C6-4A52D042ED60}.lua", 
                           "I'll tell you a secret", 
                           "I am your father, child-script!"), 
                           false)

and as long as the script with this ScriptIdentifier is running, it can read the extstate for themselves, by using

        secret = reaper.GetExtState("ScriptIdentifier:C:/Reaper/Scripts/my_script-{E0020927-08C4-4BE3-B8C6-4A52D042ED60}.lua", 
                                    "I'll tell you a secret")

That way you can pass messages to this scriptinstance back and forth, which is quite cool.

To get the ScriptIdentifier for your own script, you can use GetScriptIdentifier, which will return your unique ScriptIdentifier. Write a script that returns this ScriptIdentifier and run it a few times and you will see, that the guid-part is changing every time you run it.
That way you can be sure, that every script-instance that you create has their own ScriptIdentifier.

But you may ask: how can I know, what kind of a ScriptIdentifier a script has, that I want to run?

Good question. For that I added another feature, that is actually used by the ultraschall.Main_OnCommandByFilename-function of the Ultraschall-API itself.

You can, if you want, influence this ScriptIdentifier yourself, by influencing the filename of the script. So, what you should do is, create a copy of the script, using MakeCopyOfFile and give it a different name of the following scheme:

        filename-{guid}.lua

So if you have a script called "testscript_that_displays_stuff.lua", this copy chould be called

        testscript_that_displays_stuff-{A6A79CC4-9DE2-4791-A5F8-62EEF4ABEAC3}.lua

Now register this file as new script, run it using Reaper's own reaper.Main_OnCommand-function and unregister it again.

When "testscript_that_displays_stuff-{A6A79CC4-9DE2-4791-A5F8-62EEF4ABEAC3}.lua" is using the Ultraschall-API, it will not only create a ScriptIdentifier, but one, that includes the guid you added to the filename. That way, you can decide, which ScriptIdentifier the script shall have, that you want to run, as the script knows the same ScriptIdentifier that you can now know, as you have the guid now.

This is too difficult to do for you? No problem, I created functions that'll do that just for you.



^ Child Scripts: Running Childscripts

Now that you know a lot about ScriptIdentifiers, you probably want to know, how to run such a child-script and get its ScriptIdentifier. For that, I added mainly two functions: Main_OnCommandByFilename and MIDI_OnCommandByFilename

These two functions allow you to run a script and get the corresponding ScriptIdentifier returned.
Let's have a look:

Main_OnCommandByFilename

                    boolean retval, string script_identifier = ultraschall.Main_OnCommandByFilename(string filename, string ...)
                    

This function runs a script in the Main-section-context of Reaper. Just pass to it the filename of the script you want to run.
You may also note the string ... parameter after filename. That can be used to pass over parameters to the script you are about to run.
These must be strings. If you pass anything else, it will be converted to string. Nil-values will be considered as the last parameter, so you can't pass nil and hope it gets converted!

This function also returns a boolean that signals, if the script has been started or not. It also returns the ScriptIdentifier for the script, you've just started.

This is good, but if you want to have such a script started within a Midi-Editor-context, you should use:

MIDI_OnCommandByFilename

                    boolean retval, string script_identifier 
                            = ultraschall.MIDI_OnCommandByFilename(string filename, optional HWND Midi_EditorHWND, string ...)
                    

This does the same, but runs the script in a MIDI-editor-context. To pass over a MIDI-Editor, you set the parameter Midi_EditorHWND using the HWND-object of the Midi-Editor of your choice.
To select the last used open MidiEditor, set this parameter to nil. If no MidiEditor is opened, it will not(!) run the script.
In regards of parameters, we have the same rules as above: replace string ... with the parameters you want to pass.
You can pass more than one parameter and it must be a string.

Like Main_OnCommandByFilename, this also returns a return value that signals if starting the script was successful. And the script-identifier for the started script as well.

Now, let's have a closer look on passing parameters and returnvalues between caller-scripts and called child-scripts.



^ Child Scripts: Passing Parameters and Returnvalues

Sometimes you want to pass parameters after you've started a script. Or you are a started script and want to return values to the script that started you.

For that, I added the functions: GetScriptParameters, SetScriptParameters, GetScriptReturnvalues, SetScriptReturnvalues, GetScriptReturnvalues_Sender

If you remember from the previous chapter about running child-scripts, the function returns a unique identifier for that started script.
Don't lose that identifier, as we'll need that to communicate with it, like for passing parameters to the childscript.

GetScriptParameters

    integer num_params, array params, string caller_script_identifier 
           = ultraschall.GetScriptParameters(optional string script_identifier, optional boolean remove)
    

If you've passed parameters to a child-script, you probably would want this child-script to actually get the parameters. And, of course, who they got them from.
This function does all that. It returns an array with the parameters as string. It also returns the ScriptIdentifier of the script, that started the child-script.
If you want to retrieve the parameters for your own script, leave the parameter script_identifier as nil, otherwise pass here the ScriptIdentifier, whose currently set parameters you want to retrieve.
As parameters are passed as extstates and you probably want to save resources, you can either set the parameter remove to nil or true to remove the extstates from memory, after your retrieval.
You can set remove to false, if you need the parameters for later use, but remove them, when you exit your script(just run this function again with remove set to nil or true).

If, for some reason, you would love to change parameters after the childscript is started, you can use the function:

SetScriptParameters

    boolean retval, string script_identifier = ultraschall.SetScriptParameters(string script_identifier, string ...)
    

Pass to it the ScriptIdentifier of your child-script and the parameters you want to pass. The same parameter-rules for Main_OnCommandByFilename apply here:
Parameters must be strings. Everything else will be converted into a string, except a nil, which is seen as the last parameter, no matter if parameters would follow.

Passing parameters is quite cool, but if you're a child-script, you may want to return something to your parent/caller-script.
For that, I included:

SetScriptReturnvalues

    boolean retval, string script_identifier = ultraschall.SetScriptReturnvalues(string script_identifier, string ...)

This allows you to set return-values to a specific script with a specific ScriptIdentifier. It basically works like SetScriptParameters, so the rules for SetScriptParameters apply here too.
With that, you can return values to the caller-script or any other script, whose ScriptIdentifier you have.

And if you're a parent/caller-script and would love to hear from you child-script, you can use:

GetScriptReturnvalues

    integer num_params, array retvals = ultraschall.GetScriptReturnvalues(string script_identifier, optional boolean remove)
    

This returns the return-values stored for the current script by another script. Just pass to it the ScriptIdentifier of the script that sent the returnvalues.
The rules for the remove-parameter apply also here: as the return-values are stored as extstates, these should be removed after usage to prevent wasting of resources.
To do that, either set remove to nil or to true. If for some reason you need these returnvalues for longer, set remove to false. Keep in mind, to save resources, that you should remove the retvals later, before exiting the script. Just run the function again with remove set to nil or true in that case.

Now, there's only one question left: How do I know, which script has sent me returnvalues and what ScriptIdentifier it has? Well, use this function:

GetScriptReturnvalues_Sender

    integer count, array retval_sender = ultraschall.GetScriptReturnvalues_Sender()
    

This returns the number of scripts, that have sent returnvalues and their corresponding ScriptIdentifier. As the ScriptIdentifier also includes the filename of the sender-script, you can use this to get, if a script that you've started, send you the returnvalues, if you don't have the ScriptIdentifier available.

This should give you a lot to work with, when working with child-scripts.



^ DeferScripts: Introduction to Ultraschall's Defer-functions

Sometimes, when programming Lua-scripts in Reaper, you would love to have scripts, that run continuously. Normally, you would do that using a while-loop, but this blocks the userinterface of Reaper after a few seconds.
So the developers of Reaper added another structure, called defer-loops.

These defer-loops create nodes in your script, that means: you pass over to the reaper.defer-function a function, that you would love to run continuously.
This continuous running of the function works like that:

etc.
You get the idea.

Now here's the problem: it runs the deferred function everytime. Even if you need it to run only every few seconds, it will be run about 30 times a second.
And if you have a condition, under which the defer-loop should end, you need to construct more or less complex structures to take care of that.
Rule of thumb: the decision-making of how often the deferred function shall be run, is complex. Quitting a defer-loop is complex as well(for beginners at least), especially, if you want to stop the defer-loop from the outside of the script-instance.

For that, I created my own defer-loops, that take care of that.



^ DeferScripts: Special Defer-loops in the Ultraschall-API

It always bugged me, when I wanted to write a simple defer-loop that should run every 3 seconds or so, as the logic to do that is sometimes very annoying to code time and again, sometimes more than the actual code itself.
So, I added my own defer-loop-function, that takes care of that.

Let's have a look at a regular defer-loop in Reaper:

        A=1
        
        function main()
            reaper.ShowConsoleMsg(A.."\n")
            A=A+1
            reaper.defer(main)
        end
        
        main()

This code counts up the variable A in every defer-cycle, which is approximately 30 times a second(depending on how much other stuff Reaper has to do inbetween two defer-cycles). It also displays the value of variable A into the ReaScript-console.
Now, if I want to have this code in the function only executed every 30th cycle(as it would eat too much resources otherwise), I need to add extra structures to do that:

        A=1
        cycle_counter=1
        
        function main()
            
            if cycle_counter==30 then
                -- do this part only every 30th defer-cycler
                cycle_counter=1
                reaper.ShowConsoleMsg(A.."\n")
                A=A+1
            end
            cycle_counter=cycle_counter+1
            
            reaper.defer(main)
        end
        
        main()

This code counts up the variable A every 30 cycles, only. (It also displays the value of variable A into the ReaScript-console.)
The same goes with executing the code only every x seconds.

        A=1
        seconds_counter=reaper.time_precise()+3
        
        function main()
            if reaper.time_precise()>seconds_counter then
                -- do this part only every three seconds
                seconds_counter=reaper.time_precise()+3
                reaper.ShowConsoleMsg(A.."\n")
                A=A+1
            end
            
            reaper.defer(main)
        end
        
        main()

This code counts up variable A every 3 seconds and shows it to the ReaScript-console.
It quickly becomes messy or at least harder to read. Especially, if you need to add more code, the "skipping mechanics" are often spread everywhere, making the code hard to debug and read. And I hate my code being hard to debug and read.

The Ultraschall-API has for that the functions Defer

Defer

        boolean retval = ultraschall.Defer(function func, string deferidentifier, optional integer mode, 
                                           optional number timer_counter, optional boolean protected)
    

This works like reaper.defer, but has additional benefits, like the parameters mode and timer_counter, who allow you to specify, how often the defer-cycle shall be executed.
Either all the time(mode=0), every nth-cycle(mode=1) or every nth-second(mode=2). If mode is 1 or 2, you should set timer_counter to the amount of cycles/seconds to wait for the next execution of the defer-loop.
You can also specify a defer_identifier, which can be used by StopDeferCycle to stop the defer-loop from inside and outside of the script(more on that later).
The parameter protected allows you to prevent this defer-loop from being stopped or influenced from the outside. More on that later.

Now, let's get back to our example and see, how it would look with these new Defer-functions.

        dofile(reaper.GetResourcePath().."/UserPlugins/ultraschall_api.lua")
        
        A=1
         
        function main()
           reaper.ShowConsoleMsg(A.."\n")
           A=A+1
           
           ultraschall.Defer(main, "MyDeferLoop", 1, 30)
        end
        
        main()

Like our previous example, this counts variable A up every 30th cycle and shows its value in the ReaScript console. But as you can see, the code is much cleaner due fewer lines. You can also see, that ultraschall.Defer which defers the function main, is set to mode 1(execute every nth cycle) and the number of cycles to wait inbetween execution is 30 cycles.

Now let's see, whether we can modify the other example as well, that counts variable A up every 3 seconds:

        dofile(reaper.GetResourcePath().."/UserPlugins/ultraschall_api.lua")
        A=1
         
        function main()
           reaper.ShowConsoleMsg(A.."\n")
           A=A+1
         
           ultraschall.Defer(main, "MyDeferLoop", 2, 3)
        end
        
        main()

You see, it is basically the same code. All that changed was the parameters for Defer, which is now set to mode 2(execute code every nth seconds) and the number of seconds to wait(3 in this case).

This makes it much easier to run code of defer-scripts only when it is needed(and keep the code cleaner).
Save computer-resources, save electrical power, save the world!

In the next chapter, I will show you, how to stop such a defer-loop from within and outside the current script-instance.



^ DeferScripts: Stopping Defer-loops from in- and outside of scripts

I was talking about being able to stop these Defer-loops I added to the Ultraschall-API.
For that, we should have another look at the functions-description of Defer.

Defer

        boolean retval = ultraschall.Defer(function func, string deferidentifier, optional integer mode, optional number timer_counter)
    

As you can see, you can pass a parameter called defer_identifier. This is a unique identifier for this defer-loop, which can be used to stop this exact defer-loop from the inside by the script who runs he Defer-cycle or from the outside by another script.

To stop a defer-loop, I added another function, called StopDeferCycle.

StopDeferCycle

        boolean retval = ultraschall.StopDeferCycle(string defer_identifier)
    

Just pass to it the defer_identifier that the Defer-function returned, and it will stop that running loop.

Now here's the clou: you can do that from inside of the script, that is running the Defer-Cycle, BUT: you can also do this from another script.

Let's take the following example:

        dofile(reaper.GetResourcePath().."/UserPlugins/ultraschall_api.lua")
        A=1
        defer_identifier="MyDeferLoop"
         
         
        function main()
           reaper.ShowConsoleMsg(A.."\n")
           A=A+1
         
           retval = ultraschall.Defer(main, defer_identifier, 2, 3) -- defer me every 
           reaper.CF_SetClipboard(defer_identifier)
        end
        
        main()

This code puts the defer_identifier into the clipboard and runs the Defer-script, that counts variable A up every 3 seconds and shows its value in the ReaScript-console.
If we write now another script, that does the following:

        dofile(reaper.GetResourcePath().."/UserPlugins/ultraschall_api.lua")

        defer_identifier_from_clipboard = reaper.CF_GetClipboard("")

        retval = ultraschall.StopDeferCycle(defer_identifier_from_clipboard)

We can stop this defer-cycle, immediately. Try this for yourself.

IMPORTANT: The defer_identifier must be a unique one. If you use the defer_identifier for multiple defer-loops, all these Defer-loops will be stopped altogether, when using StopDeferCycle!

One last note: in the example above, we stopped the script from the outside, but you can stop that script from the inside of the script as well.
So, if you have two defer-loops and the second defer-loop shall control, if the first defer-loop is running or not:
Run the function, that shall be deferred, use ultraschall.Defer and pass to it your defer_identifier and stop it using StopDeferCycle and the defer_identifier.



^ DeferScripts: Manipulating and protecting Defer-loops

Sometimes you want to be able to alter the behavior of defer-loops from the outside. So if you have numerous defer-loops, you may want to have a central control-gui-script, with which you can set, how often to run a function, etc.
For this, I added a bunch of functions: GetDeferCycleSettings and SetDeferCycleSettings

GetDeferCycleSettings

        integer mode, integer timer_counter = ultraschall.GetDeferCycleSettings(string deferidentifier)
        

This function allows you to get the current settings of a defer-cycle with a certain deferidentifier. Just pass to it the deferidentifier and you'll get the current mode(0, run always; 1, run every x cycle; 2, run every x second) as well as the counter/timer for the number of cycles/seconds to wait, until the function is called the next time.
To set things, you need to use:

SetDeferCycleSettings

        boolean retval = ultraschall.SetDeferCycleSettings(string deferidentifier, optional integer mode, optional number timer_counter)
        

With this function, you can set the new mode/timer-settings of a specific defer-cycle. You just need to pass to it the deferidentifier and optionally the mode(0, run always; 1, run every x cycle; 2, run every x second) and the counter/timer for the number of cycles, seconds to wait until running the deferred-function the next time.
If you pass to both, you can reset the settings to the original ones, with which the defer-cycle has been called initially.

This should give you ways of controlling, how often your deferred-functions shall be run and of saving computing-resources by that.

Now there's one thing still missing: What if I want to prevent my Defer-Loop from being influenced by the outside at all?
Let's get back to Defer, specifically the parameter protected.

Defer

        boolean retval = ultraschall.Defer(function func, string deferidentifier, optional integer mode, 
                                           optional number timer_counter, optional boolean protected)
        

If you set the parameter protected to true, this defer-cycle can not be influenced from the outside. The only thing possible is getting its current mode and timer-settings.



^ Localize Scripts: Introduction

Reaper has (among thousands of others) one great thing: Localization. That means, you can translate the whole software into many languages.
Unfortunately, this isn't possible for scripts, or better: Not that easy.

So I thought, that adding such functionalities for Lua-Scripts would be a good idea too.

The whole process consists of two main things:

  1. The USLangPack-file, which holds all the original texts and their translated counterparts
  2. The functions, with whom you can easily translate text within your scripts.

Let's start with the first one, the translationfile.



^ Localize Scripts: Language-pack-fileformat

A USLangPack-translationfile is a simple ini-file of the format:

            [section]
            key=value

In our case, the section is the scriptname, the key is the original string and the value is the translated string.

            [myluascript.lua]
            Hello World=Hallo Welt

Basics

In this example, the text "Hello World" in myluascript.lua would be translated to the german translation "Hallo Welt".
If you need to add comments, you can simply write a semicolon. Everything written in that line after the semicolon will be ignored.

            [myluascript.lua]
            Hello World=Hallo Welt ;It looks like part of the translation, but it's not!
    

Now here's the thing, you may want to add = into your text, but ini-files use this as separator between key and value.
So, to write an = you need to escape it, using \=

            [myluascript.lua]
            Hello\=World=Hallo\=Welt
    

In the above example, the text holds now an =
The escape-symbol will be removed when reading it from the translationfile automatically. If you don't want that, write \= and it will become \= in the text.

Newlines are also possible. Just write \n in the places, where you want to have a newline:

            [myluascript.lua]
            Hello\nWorld=Hallo\nWelt
    

The newlines (\n) in the above example, will be replaced by the Localize-function automatically into newlines.

Advanced

Sometimes, you want to have substitutions, like filenames, variable numbers, etc. As this is very important, I added ways of adding them into the translation as well.
Substitutions are written into the text using %sxxx, where xxx is a number between 000 and 999.
The number between 000 and 999 reflects the order of the substitutions.
Let's assume the following translationentry:

            [myluascript.lua]
            The filename %s000 is the one that you have selected.=Du hast den Dateinamen %s000 ausgewählt.
    

In this example, the text has the substituion of %s000, which will represent the filename.
Now look closely: the substitution %s000 in the english text is on the third position, while in the german translation, it is on the fifth!
That's why substitutions are numbered, as the translator may need to rearrange the substitutions per requirements of the other language.
Especially, if you have a number of substitutions, the order in the translated version can be "disordered", compared to the original one.
The Localize-function takes care of substituting the right words into the substitutions.

As a good practice, you should add a comment directly above the line, which holds the substitution, which explains the meaning of the substitution(s).

            [myluascript.lua]
            ;%s000 - this is a filename, which could also hold the path to the file
            The filename %s000 is the one that you have selected.=Du hast den Dateinamen %s000 ausgewählt.
    

That way, the translator knows, what to expect from it, which improves quality of the translation.

Different Languages

Now you can do one translation, but there are many more languages. So how to make that work?
Via the filename.

The used languages in a translationfile is always added to the USLangPack-file:

            mylangpack_language.USLangPack
    

So an english, german and spanish version would be named like this:

            mylangpack_en.USLangPack
            mylangpack_de.USLangPack
            mylangpack_es.USLangPack
        

The Localize_UseFile-function will load the right LangPack for you.

Best Practices

I will add some final notes on how to translate properly, for scripters as well as for translators:

Scripters:
You should never assume a specific length of your string, as other languages may need longer texts in their translation.
English vs German is such an example. In many cases, English is much more efficient than German, leading to longer translated texts.
If you don't take care of that, texts in your script will look ugly, maybe even unreadable.
So keep your texts as short as possible(no novels please!) and make the displayed text more flexible for different stringlengths.
This makes life and work much more easier for the translator.

Translators:
Even though it is good practice, if displayed strings allow certain flexibility in length, you can't rely on that.
So if you want to translate the texts for a script, try to make the translation not longer than the original string and check, whether your translation will be displayed at all.
If there's no way to translate the text in any way, without clipped texts, etc, ask the scripter to modify the script for more flexibility.
So what I wrote for scripters is also valid for you: So keep your texts as short as possible(no novels please!).
This makes life easier for the user and the scripter.



^ Localize Scripts: Functions

Using a translationfile to localize texts in your script is very easy. There are only three functions to deal with that:
Localize_UseFile, Localize_RefreshFile, Localize

Let's go into more detail:

Localize_UseFile

            boolean retval = ultraschall.Localize_UseFile(string filename, string section, string language)
    

This function loads the contents of a translationfile into your script.
You need to pass the filename of the translationfile. If you omit the path, it will look in the Resource-folder/LangPack/ for the file.
The next part is the section of the file. As you may remember from the previous chapter, you should give the section your scriptname.
That way, you can have one translationfile for all your scripts, with each section dedicated to one of your scripts.
The last parameter is the used languages. You should use the common abbreviations, like en, es, de, it and so on.
This function will load the right translationfile with the right language and the translations from a specific section. If it can't, it will return false.

Localize_RefreshFile

            boolean retval = ultraschall.Localize_RefreshFile()
    

If the translationfile has been updated, you might want to reload it. Just run this function, and it reloads an already loaded translation.

Now there's one thing remaining: How to actually translate/localize a string? With this function:

Localize

            string translated_string, boolean translated = ultraschall.Localize(string original_string, ...)
        

or alternatively

            string translated_string, boolean translated = Localize(string original_string, ...)
    

This localizes a string, using the loaded translation. Just pass the original-string into the first parameter.
If you have substitutions, you pass them as additional parameters in the order of appearance.
That means, if you have a string "I am a filename %s000 and a number %s001" the Localize-functioncall will be

            translated_string, translated = Localize("I am a filename %s000 and a number %s001", "name_of_the_file0ab.txt", "135")

This will return the localized string with the substitutions in the right places.
If for one reason or another, no translation can be done(no loaded translationfile, etc), the second returnvalue translated will be set to false.

For convenience, you can use Localize without ultraschall. at the beginning.

This should make having your scripts localized an easy and convenient task.



^ EventManager: Introduction

Sometimes you run into the situation, where you want to run one or more actions when certain events happen.

For instance, you want to output a warning when trying to start recording and project-playrate is set to higher than 1.x, maybe even stop recording immediately in that case.
This could be done, if you build custom actions for that case, but: if you have dozens of scripts dealing with recording-situations, you would need to update all of them or build custom actions for all of them, just to include this warning.
And if you want to temporarily remove this warning, you are doomed.

Wouldn't it be nice, if there's something, that you can feed with actions, that shall be run under certain conditions?
Something, that reacts to certain events automatically, like playrate greater than 1.0 and recording-state is set to recording?
And wouldn't it be nice, if you could decide, whether this check shall be run or paused the way you want it?

Wouldn't it be nice to have an EventManager in Reaper, that deals with all that stuff?

Behold! It's here!

The Ultraschall-Api-EventManager came to do all of that for you.



^ EventManager: The Basic Concept

The EventManager is a script I wrote. It runs in the background, after you've started it.
You can add events to it, which contain

  1. an eventcheck-function, which returns true if the event happens, otherwise false
  2. the attributes, how long and how often the event shall be checked for
  3. a list of actions who are run, if the event occurred, which means, the eventcheck-function from 1 returns true

The eventcheck-function is one, that you write. That's right, you write a function that'll check, if the event has occured. That way, you have full control on how the event shall be checked.

The attributes include things like, how often to check(every x seconds), how long to check for(for x seconds, then stop checking), if the event shall be paused or run right away, and a name which you can give the event, so you know, what its about. You can also set, if the actionlist shall be run as long as the eventcheck-function returns true(again and again) or only once when the returned state of the check-function goes from false to true(once).

The list of actions is also common to you: you pass over the sections and the accompanying action-command-ids to the eventmanager, as you can find them in the action-list.

Now, if you've passed them over, the EventManager registers it within itself and does the checking. And if the event occurs, it runs the actions from the actionlist in the order that you've passed them. And it does it all automatically for you in the background.

But, what if you've added an event and you change your mind later on how the attributes shall be set, or if you want to change the list of actions to run? Don't panic: you can alter them again after the fact.

And if you don't want to add them all the time manually, you can even add the events as startup-events. That means, you can add events, who are added to the EventManager immediately after the EventManager has been started. So all you need in that case is simply starting the EventManager once, and it will check for the events, you've added as startup events right away.

That's all nice and such, but: how to actually do it?

Read on, my dear.



^ EventManager: Basic workflow

The basic workflow to use the EventManager is quite simple:

  1. Write a checkingfunction, that returns true, if the event occured and false, if not. (No global variables are allowed!) This function can safely use Ultraschall-API-functions.

  2. Start the EventManager with ultraschall.EventManager_Start()

  3. Add the new event to the Ultraschall-Api-EventManager using ultraschall.EventManager_AddEvent(), which returns an EventIdentifier-string.

Voila, the EventManager checks for your event.

And if at some point you need to stop the EventManager, just use EventManager_Stop:

        ultraschall.EventManager_Stop()

The EventIdentifier returned by EventManager_AddEvent is like a unique name for your event, until the event gets removed again from the EventManager.
And this EventIdentifier can be used, if you want, to

So you can influence and alter the event, after you've added it to the EventManager, just using the EventIdentifier. In addition to that, you can

And if you want to remove all Events your script has registered to the EventManager, use ultraschall.EventManager_Stop().
This function allows you to stop all events registered by a different script, using the ScriptIdentifier of that script or even stop all of them when setting parameter force=true.

With that, you can populate the EventManager without a problem.
Now, let's go into more detail, what "events" means, how to build a proper statecheck-function and how to add events at startup of the EventManager.

Let's start with the states, who can be checked.



^ EventManager: Which events can be checked?

Basically there are two types of events you can check for:

  1. True/False events, like "Is the mute-button of the master-track muted(true) or not(false)?"

  2. Transitioning, from > to events, like "Does the playstate change from Play to PlayPause?"

The True/False events are the easiest to check for. All you need to do is write a checking-function, that returns true, if the condition is met(e.g. mastertrack is muted) or false if the condition is not met(e.g. mastertrack is NOT muted).
So everytime something is true, the EventManager runs the actions associated with this event; if it's false, it will not run the actions.
This kind of event is usually the best, if there are only two possible conditions who could be met, like "button is pressed" and "button is unpressed".

The Transitioning, from > to events are a little bit more difficult to code(but not that much).

First you need to find out the event-transition you want to check for. Then you write a function, which returns true, if the transition has happened or false, if not. For that, you need to keep track of the oldstate and the newstate. You are not allowed to use global variables for keeping track of that, but eventcheck-functions get their own userspace to store their information(will explain how that works in the next chapter).
This kind of event is usually the best, if there are more than two conditions who could be met and you want to run actions only when transitioning between two specific conditions.
Like the playstate, who could be stop, play, playpause, rec, recpause. If you want to only run the actions when play changes from play to playpause, but not if play changes from play to stop, transitioning-events are the way to go.

Now I'll explain to you, how to write a proper eventcheck-function, as you need to take some things into consideration for that.



^ EventManager: the eventcheck-function for true/false-events

The eventcheck-function for true/false-events can be made quite simply.
Let's take the aforementioned example, if the mastertrack is muted or not.

        function IsMasterTrackMuted()
           -- get the current mute-state of the master-track
           -- always make needed variables local !
           local mutestate=reaper.GetMasterMuteSoloFlags()&1 
           
           -- return false, if mutestate==0, else return true.
           if mutestate==0 then 
             return false
           else
             return true
           end
        end

This would be fully sufficient as the check-function for such true/false-events and would be fully accepted by the EventManager.
Note, how I made the needed variable mutestate local by adding local before it. This is important to do, as you may risk to overwrite internal structures of the EventManager otherwise!

Another important thing to do is: make the eventcheck-function as simple and fast as possible. Try to avoid resourceintensive operations like heavy patternmatching and such stuff. The longer your eventcheck-function needs to execute, the more laggy the EventManager becomes!



^ EventManager: Creating Events for transition-events

The eventcheck-function for transition-events is a little more difficult to do.

Let's take the other example, where I would do a eventcheck-function, which checks if playstate transitions from play to playpause happen.
For that, we need to store the old playstate. Usually, we would do that into a global variable, but as global variables are forbidden in the EventManager, I've added something else into the EventManager: UserSpaces.

UserSpaces are tables, that I pass to your eventcheck-function as first parameter. This table can be populated in any way you want.
Let's take an example of that:

        function StoreTimeIntoUserSpace(userspace)
            -- get the current time from the system
            local current_time=os.time()
            
            -- if the time has changed, since last time calling this function, return true
            -- the old time is stored into the userspace-table:
            --      userspace["previous_time"]
            -- which is, what we use for comparisions in the following if-statement
            if current_time~=userspace["previous_time"] then
                userspace["previous_time"]=os.time()
                return true
            else
                return false
            end
        end

This function stores the old time of your system into userspace["previous_time"], which can be accessed again the next time this function is called.
Note, that the table we use is passed as first parameter by the EventManager. That means, even if you don't use it, you'll get the UserSpace to store things into it. You can treat it, as any other table you are used to.

Ok, let's return to our intended check-function for the transition-check between play and playpause.

        function TransitionPlayToPlaypause(userspace)
          -- get the current playstate
          local current_playstate=reaper.GetPlayState()
          
          -- if the current playstate==2(playpause) and the old playstate==1(play) then 
          -- update old_playstate in the userspace and return true
          -- in any other case, only update old_playstate in the userspace and return false
          if current_playstate==2 and userspace["old_playstate"]==1 then
            userspace["old_playstate"]=current_playstate
            return true
          else
            userspace["old_playstate"]=current_playstate
            return false
          end
        end

This function only returns true, when playstate changes from play to playpause, in all other cases it will return false.
You can modify it even further. The function above returns true only once, when changing from play to playpause, but you can make it return true, until the playstate changes again.
Feel free to modify and experiment with the code.

To test such transitioning-functions, you can build a simple test-environment, like the following code:

        UserSpace={} -- create a temporary userspace-table, we use to check, if the function is working correctly
        
        -- our eventcheck-function
        function TransitionPlayToPlaypause(userspace)
          -- get the current playstate
          local current_playstate=reaper.GetPlayState()
          
          -- if the current playstate==2(playpause) and the old playstate==1(play) then 
          -- update old_playstate in the userspace and return true
          -- in any other case, only update old_playstate in the userspace and return false
          if current_playstate==2 and userspace["old_playstate"]==1 then
            userspace["old_playstate"]=current_playstate
            return true
          else
            userspace["old_playstate"]=current_playstate
            return false
          end
        end

        -- A defer-loop, which runs the function time and again;
        -- if you run this code in the ReaScript-IDE, you can see in the variable-watchlist, how the
        -- variable "eventcheck_returnvalue" holds true(for a short period of time), every time you switch from play to playpause,
        -- in any other case, "eventcheck_returnvalue" only holds false
        function main()
          eventcheck_returnvalue=TransitionPlayToPlaypause(UserSpace)
          reaper.defer(main)
        end
        
        main()

This allows you to check for quite complex and individual events in any way you want. (See EventManager_Debugging for more debugging and testing strategies).

Note here as well, how I made the variable current_playstate in the eventcheck-function "TransitionPlayToPlaypause" local, as this is important here as well!

And for transition-events, we have the same performance-related mantra: make the eventcheck-function as simple and fast as possible. Try to avoid resourceintensive operations like heavy patternmatching and such stuff. The longer your eventcheck-function needs to execute, the more laggy the EventManager becomes!

Now, that we have our eventcheck-function, we can add a new event to the EventManager.



^ EventManager: Adding events

Now, as we have the play to playpause eventcheck-function, we should try to make something with it.
So, let's add a marker, everytime we change from play to playpause.
Before we start, we need to get the ActionCommandID of the action, we would love to run in these cases.
In our case, we use:

        40157 - Markers: Insert marker at current position

from the main-section: 0

Now, we need to start the EventManager:

        ultraschall.EventManager_Start()

Now let's add the new event, with the following attributes set:

        EventIdentifier = ultraschall.EventManager_AddEvent(
                              "Insert Marker When Play -> PlayPause", -- a descriptive name for the event
                              0,                                      -- how often to check in seconds; 0, means as often as possible 
                              0,                                      -- how long to check for it in seconds; 0, means forever
                              true,                                   -- shall the actions be run as long as the eventcheck-function 
                                                                      --       returns true(false) or not(true)
                              false,                                  -- shall the event be paused(true) or checked for right away(true)
                              TransitionPlayToPlaypause,              -- the eventcheck-functionname, 
                              {"40157, 0"}                            -- a table, which hold all actions and their corresponding sections
                                                                      --       in this case action 40157 from section 0
                                                                      --       note: both must be written as string "40157, 0"
                                                                      --             if you want to add numerous actions, you can write them like
                                                                      --             {"40157, 0", "40171,0"}, which would add a marker and open 
                                                                      --                                      the input-markername-dialog
                              )

This function returns now an EventIdentifier, which(as previously mentioned) can be used to set, pause, resume, remove the event again from the EventManager.

The full example-code looks like this:

        dofile(reaper.GetResourcePath().."/UserPlugins/ultraschall_api.lua")

        -- First: Write a check-function, which must return either 
        -- true(condition has been met) or false(condition hasn't been met)
        -- If you need to store information for the next time the checkfunction 
        -- is called, use the userspace-table, which is passed as first parameter.

        function TransitionPlayToPlaypause(userspace)
          -- get the current playstate
          local current_playstate=reaper.GetPlayState()
          
          -- if the current playstate==2(playpause) and the old playstate==1(play) then 
          -- update old_playstate in the userspace and return true
          -- in any other case, only update old_playstate in the userspace and return false
          if current_playstate==2 and userspace["old_playstate"]==1 then
            userspace["old_playstate"]=current_playstate
            return true
          else
            userspace["old_playstate"]=current_playstate
            return false
          end
        end

        -- Second: start the EventManager
        ultraschall.EventManager_Start()

        -- Third: add a new event to the EventManager
        --        this event adds a marker at playposition every time, you change from play to playpause
        EventIdentifier = ultraschall.EventManager_AddEvent(
            "Insert Marker When Play -> PlayPause", -- a descriptive name for the event
            0,                                      -- how often to check in seconds; 0, means as often as possible 
            0,                                      -- how long to check for it in seconds; 0, means forever
            true,                                   -- shall the actions be run as long as the eventcheck-function 
                                                    --       returns true(false) or not(true)
            false,                                  -- shall the event be paused(true) or checked for right away(true)
            TransitionPlayToPlaypause,              -- the eventcheck-functionname, 
            {"40157, 0",                            -- a table, which hold all actions and their corresponding sections
             "40171, 0"}                            --       in this case action 40157 from section 0 and 40171 from section 0
                                                    --       note: both must be written as string "40157, 0"
                                                    --             if you want to add numerous actions, you can write them like
                                                    --             {  "40157, 0", "40171,0"}, which would add a marker and open 
                                                    --                                        the input-markername-dialog
                                                    --             
                                                    --       ActionCommandIDs with an underscore at the beginning are allowed.
                                                    --       like: _SWS_AWSELGROUPIFGROUP or _RS1234567890, etc
        )


^ EventManager: Alter Events and retrieve settings

Now we've added a new event. But what about altering it? Well, this is as easy as adding an event.
So let's assume, you want to alter how often to check from 0(all the time) to 3(every third second) and you want to change the name from "Insert Marker When Play -> PlayPause" to "Barracuda".
You can do it like this:

        EventIdentifier = ultraschall.EventManager_SetEvent(
                        EventIdentifier,                        -- the identifier of the Event, that you want to alter
                        "Barracuda",                            -- a descriptive name for the event
                        3,                                      -- how often to check in seconds; 0, means as often as possible 
                        0,                                      -- how long to check for it in seconds; 0, means forever
                        true,                                   -- shall the actions be run as long as the eventcheck-function 
                                                                --       returns true(false) or not(true)
                        false,                                  -- shall the event be paused(true) or checked for right away(true)
                        TransitionPlayToPlaypause,              -- the eventcheck-functionname, 
                        {"40157, 0",                            -- a table, which hold all actions and their corresponding sections
                         "40171, 0"}                            --       in this case action 40157 from section 0 and 40171 from section 0
                                                                --       note: both must be written as string "40157, 0"
                                                                --             if you want to add numerous actions, you can write them like
                                                                --             {  "40157, 0", "40171,0"}, which would add a marker and open 
                                                                --                                        the input-markername-dialog
                                                                --             
                                                                --       ActionCommandIDs with an underscore at the beginning are allowed.
                                                                --       like: _SWS_AWSELGROUPIFGROUP or _RS1234567890, etc
                    )
        

This works basically as the AddEvent-function. The only difference is, that you need to pass as first parameter the identifier of the event, so the EventManager knows, which event shall be altered.

To get the EventIdentifiers, you can either store them somewhere or retrieve them using EventManager_GetEventIdentifier, EventManager_EnumerateEvents.

            EventIdentifier = ultraschall.EventManager_GetEventIdentifier(1)
    

This returns the EventIdentifier of the first registered event in the EventManager.

I already mentioned, that you can pause events as well. That means, they are not checked for, while they are paused. I included the functions EventManager_PauseEvent and EventManager_ResumeEvent

            retval = ultraschall.EventManager_PauseEvent(EventIdentifier)
    

pauses the event, which has the EventIdentifier and

            retval = ultraschall.EventManager_ResumeEvent(EventIdentifier)
    

resumes the event, which has the EventIdentifier.

There are even more things, like:



^ EventManager: Debugging Events

Ok, now you know, how to add, get and set the events in the EventManager. But sometimes, especially during development, this isn't enough. In fact, you want to have full insight in what's going on with your event.

Does the eventcheck-function return the right returnvalues?

Does the userspace contain the right values?

For such debugging purposes, I've added some additional nice things.

Returnvalues of Eventcheck-functions

To monitor the currently returned value of the eventcheck-function, just use EventManager_GetLastCheckfunctionState and EventManager_GetLastCheckfunctionState2

            boolean check_state, number last_statechange_precise_time = 
                            ultraschall.EventManager_GetLastCheckfunctionState(integer id)
    

returns the returned value of the checkfunction of an event by id. That means, 1 for the first registered event, 2 for the second, etc.

            boolean check_state, number last_statechange_precise_time = 
                            ultraschall.EventManager_GetLastCheckfunctionState2(string EventIdentifier)
    

just like EventManager_GetLastCheckfunctionState, but you need to pass the EventIdentifier of the event as parameter.

DebugMode

This gives you even more access to internal states of the EventManager, like the UserSpace of an eventcheck-function. However, this mode is resourceintensive and intended for debugging only. So you should keep it deactivated for productive use of the event!

First, you need to turn on the DebugMode:

            ultraschall.EventManager_DebugMode(true)
    

After that, you can get the current contents of the userspace a certain checkfunction of an event uses:

            userspace_count, userspace_table = 
                                    ultraschall.EventManager_DebugMode_UserSpace(1)
    

This returns the contents of the userspace as a handy table. Please note, that userdata-objects, like MediaItems, MediaTracks, etc are just returned as strings! With that, you should be able to see, whether the userspace contains the values you expect.

If you're finished with debugging, you should turn off the DebugMode again:

            ultraschall.EventManager_DebugMode(true)
    

This should give you a lot power to make your events working.



^ EventManager: Working with startup events

Adding events is great, but wouldn't it be awesome, if I could automatically add certain event to the EventManager as soon as I start it, without having to run dozens of event-adding-scripts?

Yes it would be, so I added startup-events, who do this exact thing: being started and checked right away, as soon as you start the eventmanager.
So if you have a __startup.lua in your scripts-folder, which loads the Ultraschall-API and run the function

ultraschall.EventManager_Start()

they are loaded at Reaper startup-time.

EventManager-startup-events are added into ResourcesFolder/UserPlugins/ultraschall_api/IniFiles/EventManager_Startup.ini

You can populate the startup-events the same way, as events into a running EventManager.
That means, all rules for regular events also apply here, but you don't need to run the EventManager for adding startup-events.

And if you need more general functions for that as well:

With that in your pocket, you have a lot of possibilities to control your full event-based workflows.



^ FX-Management: Introduction

When working with audio, it's almost impossible to not use FX on items and tracks, using parameter modulation and learn.
But sometimes, the way Reaper's user interface manages things is not enough. You want to code the stuff yourself to make it behave to your needs.

So I simply had to implement a lot of features for manipulation of FX, including some stuff never been codeable before.

So let's dive into the magical wonderful world of fx-management.



^ FX-Management: Working with FX-StateChunks

A lot of fx-possibilities are already possible with Reaper's native API-functions, but sometimes you need the real deal, the full control.
And Reaper has such a datastructure for that, called FXStateChunk.

FXStateChunks are parts of either MediaItemStateChunks or TrackStateChunks. What they basically do is hold all information, settings, etc of all fx in the FXChain of the MediaItem or MediaTrack.
These hold bypass-settings, state-binary strings, aliasnames of the fx, parameter-learn/parameter modulation, etc. So if you want to manipulate specific fx beyond Reaper's own api-functions, manipulating FXStateChunks is the way to go.

Getting an FXStateChunk is quite easy.

  1. Get the item/track-statechunk, using GetTrackStateChunk_Tracknumber(for trackfx) or reaper.GetItemStateChunk(item/take-fx).
  2. Get the FXStateChunk from the item/track-statechunk with GetFXStateChunk

Now you can manipulate it to your needs. When you're finished manipulating the FXStateChunk, you can set it back into the original item or track-statechunk.

To set it again, you go the path into the other direction 1. Set the FXStateChunk in the item/take-statechunk via SetFXStateChunk 2. Set the item/track-statechunk into the item/track, using SetTrackStateChunk_Tracknumber(for trackfx) or reaper.SetItemStateChunk(item/take-fx).

That's the basic workflow to toy around with FXStateChunks. More details in the next chapters



^ FX-Management: Working with FX-StateChunk's functions

Now let's go into more detail on numerous functions regarding FXStateChunks.

The most important ones are the functions for getting and setting a FXStateChunk from/into track/item-statechunks.

GetFXStateChunk

                    string FXStateChunk, integer linenumber = 
                                    ultraschall.GetFXStateChunk(string StateChunk, optional integer TakeFXChain_id)
                    

This gets the FXStateChunk from a track or item-statechunk and returns also its linenumber within the original statechunk. There's also an optional parameter TakeFXChain_id, which is helpful in item-statechunks, as they can have multiple FXStateChunks, with one for each existing take. Just set it to the take-id(beginning with 1) which shall get the FXStateChunk. Unlike Reaper's own Get/Set-TrackStateChunk-functions, this formats the StateChunk properly, so using the fileformat for string-substitution is possible to do.

SetFXStateChunk

                    boolean retval, optional string alteredStateChunk = 
                                    ultraschall.SetFXStateChunk(string StateChunk, string FXStateChunk, optional integer TakeFXChain_id)
                    

This sets the FXStateChunk in a track or item-statechunk and returns the altered Item/Track-StateChunk.
There's also an optional parameter TakeFXChain_id, which is helpful in item-statechunks, as they can have multiple FXStateChunks, with one for each existing take. Just set it to the take-id(beginning with 1) which shall get the FXStateChunk.

There are more helper-functions included, like:

CountFXStateChunksInStateChunk

                    integer count_of_takefx_statechunks, integer count_of_trackfx_statechunks 
                                                = ultraschall.CountFXStateChunksInStateChunk(string StateChunk)
                    

This counts all available FXStateChunks existing within an item/track/project-statechunk.

IsValidFXStateChunk

                    boolean retval = ultraschall.IsValidFXStateChunk(string StateChunk)
                    

This checks, if a statechunk is a valid FXStateChunk and returns true in that case.

RemoveFXStateChunkFromItemStateChunk

                    string alteredItemStateChunk = ultraschall.RemoveFXStateChunkFromItemStateChunk(
                                                                        string ItemStateChunk, integer take_id)
                                                                        

This removes an FXStateChunk from an ItemStateChunk. Just pass to it the ItemStateChunk as well as the take-number. It returns the StateChunk without the FXStateChunk of the take with take_id.

RemoveFXStateChunkFromTrackStateChunk

                    string altered_TrackStateChunk = ultraschall.RemoveFXStateChunkFromTrackStateChunk(
                                                                                        string TrackStateChunk)
                                                                        

This removes an FXStateChunk from a TrackStateChunk. It returns the StateChunk without the FXStateChunk.

Now there are numerous sources for FXChains, including RFXChain-files, who are files you can store in Reaper, who hold all settings of an FXChain. So if you want to deal with these files, I've included functions for it:

LoadFXStateChunkFromRFXChainFile

                    string FXStateChunk = ultraschall.LoadFXStateChunkFromRFXChainFile(string filename, 
                                                                                        integer trackfx_or_takefx)
                                                                        

This loads an FXChain from an RFXChain-file. You can set the whole path to it but if you only set the filename, it will try to read the RFXChain-file from the folder Resourcepath/FXChains/ It's important to tell the function, whether it shall be used as track or take-fx, as there are small detail-differences between the two FXChain-types.

SaveFXStateChunkAsRFXChainfile

                    integer retval = ultraschall.SaveFXStateChunkAsRFXChainfile(
                                                                              string filename, string FXStateChunk)
                                                                        

This saves an FXChain into an RFXChain-file. You can set the whole path to the target-file but if you only set the filename, it will store the RFXChain-file into the folder Resourcepath/FXChains/



^ FX-Management: Manipulating individual effects

Now that you have the FXStateChunk, you might want to manipulate individual FX. Problem is, you need to parse them out of the FXChain by hand.
Or you use the following way to manipulate individual effects because...you can already guess: I made some functions for that.

Manipulating individual effects needs some special treatment, but don't be afraid, it's not that hard to manage.

First, get the FXStateChunk, like described in the previous chapters.
Then you can get the lines of the FXStateChunk for a specific fx by using GetFXFromFXStateChunk.

                    string fx_lines, integer startoffset, integer endoffset = 
                            ultraschall.GetFXFromFXStateChunk(string FXStateChunk, integer fxindex)
                            

Just pass to it the FXStateChunk and the index of the fx, whose lines you want to get.
It will return these lines as a string and the start and endoffset of these lines within the FXStateChunk.
Keep these values, we'll need them later.

Now that we have the string "fx_lines", you can manipulate this string, using Lua's string-manipulation-functions, string.gsub for instance.
So to set the bypass-state of the fx-lines, you can use:

                    fx_lines=string.gsub(fx_lines, "BYPASS.-\n", "BYPASS 1 0 0\n")
                    

This will replace the part of the string beginning with BYPASS and ending with the next newline(\n) with BYPASS 1 0 0, where the 1 signals,
that this fx is not bypassed anymore.

Now if you've changed the fx-lines accordingly, you can put it into the FXStateChunk quite easily. We just need to use the returnvalues returned by GetFXFromFXStateChunk that you hopefully kept.

                    FXStateChunk=FXStateChunk:sub(1,startoffset)..fx_lines..FXStateChunk:sub(endoffset, -1)
                    

and that's it. You've successfully set the bypass-state of the fx you've got via GetFXFromFXStateChunk. You can set this FXStateChunk back into its original statechunk and then commit it to the track/item of your choice.

The following example-script toggles bypass for the first FX in the first track of your project:

                    -- toggle bypass-state of the first fx in the first track on and off
                    dofile(reaper.GetResourcePath().."/UserPlugins/ultraschall_api.lua")

                    -- get the FXStateChunk
                    retval, TrackStateChunk = ultraschall.GetTrackStateChunk_Tracknumber(1)
                    if retval==false then 
                        print2("An error has happened: No track available") 
                        return 
                    end
                    FXStateChunk = ultraschall.GetFXStateChunk(TrackStateChunk)

                    -- get the lines of the first fx in the FXStateChunk
                    fx_lines, startoffset, endoffset =  ultraschall.GetFXFromFXStateChunk(FXStateChunk, 1)
                    if fx_lines==nil then 
                        print2("An error has happened: No such FX available") 
                        return 
                    end

                    -- see, if the first number in the BYPASS-line is 1(not bypassed) or not
                    -- and toggle accordingly (when 1 newtoggle=0 else newtoggle=1)
                    if fx_lines:match("BYPASS 1") then newtoggle="0" else newtoggle="1" end

                    -- set the new bypassed-value into the fx-lines
                    fx_lines=string.gsub(fx_lines, "BYPASS.-\n", "BYPASS "..newtoggle.." 0 0\n")

                    -- use the fx_lines and put them into the right place of the FXStateChunk
                    FXStateChunk=FXStateChunk:sub(1,startoffset)
                                 ..fx_lines
                                 ..FXStateChunk:sub(endoffset, -1)

                    -- Put the altered FXStateChunk back into the TrackStateChunk
                    retval, TrackStateChunk=ultraschall.SetFXStateChunk(TrackStateChunk, FXStateChunk)

                    -- commit the TrackStateChunk to the first track so we actually toggle 
                    -- the bypass-state of the first fx of track 1
                    ultraschall.SetTrackStateChunk_Tracknumber(1, TrackStateChunk)


^ Parameter Modulation/Learn/Alias: Introduction

For those, who want more control on how fx are managed, Reaper has this feature called Parameter Modulation as well as Parameter Learn and Parameter Alias and there's a lot of stuff you can do with it.
But: it wasn't programmable yet.
And as you all know this: if it's not yet programmable and you're asking for it, I try to make it happen.
Hence, Ladies and Gentlemen and varieties inbetween, please welcome: Parameter Learn, Alias and Modulation.

First things first: to influence one of these, you need to get FXStateChunks and, if needed, set them after your changes.
So if you don't know yet on how to work with FXStateChunks, read the chapters on FXStateChunks

Let's start with the easiest of the three: Parameter Alias



^ Parameter Alias

Parameter Aliases allow you to give new names to parameters. For instance, you have added ReaEQ as first effect in the FXChain and you have this parameter called "Freq-Band 1".
You may want to have a name, which describes more what the parameter does, like "Humming Removal".

In that case, you can set parameter-alias-names.

  1. First you need to get the FXStateChunk, into which you want to set the parameter-alias-name
  2. Then you can use AddParmAlias_FXStateChunk to add a new aliasname or SetParmAlias2_FXStateChunk to set an existing aliasname in the FXStateChunk
  3. Set the FXStateChunk back into the original statechunk

In the following example, I assume that track 1 has ReaEQ in its FXChain as first parameter and the "Reset to factory defaults"-preset selected.
The first parameter of ReaEQ is named "Freq-Low Shelf".
As there is no aliasname available yet, we need to add it:

                    dofile(reaper.GetResourcePath().."/UserPlugins/ultraschall_api.lua")
                    
                    aliasname="The new parameter-alias-name"
                    
                    -- Get the trackstatechunk and the FXStateChunk from it
                    retval,TrackStateChunk = ultraschall.GetTrackStateChunk_Tracknumber(1)
                    FXStateChunk=ultraschall.GetFXStateChunk(TrackStateChunk)

                    -- add the new parameter-aliasname, and show an error-message, if there is already one
                    retval, alteredFXStateChunk = ultraschall.AddParmAlias_FXStateChunk(FXStateChunk, 1, 1, aliasname)
                    if retval==false then print2("Parameter-has already an aliasname. Please use "..
                                                 "SetParmAlias2_FXStateChunk to set it to a new one.") return end

                    -- set the altered FXStateChunk to the TrackStateChunk and commit it to the first track
                    retval, TrackStateChunk=ultraschall.SetFXStateChunk(TrackStateChunk, alteredFXStateChunk)
                    retval = ultraschall.SetTrackStateChunk_Tracknumber(1, TrackStateChunk)

If you want to set it to a new name, you can use SetParmAlias2_FXStateChunk. The following code sets the parameter-aliasname to "Totally new name"

                    dofile(reaper.GetResourcePath().."/UserPlugins/ultraschall_api.lua")
                    
                    aliasname="Totally new name"
                    
                    -- Get the trackstatechunk and the FXStateChunk from it
                    retval,TrackStateChunk = ultraschall.GetTrackStateChunk_Tracknumber(1)
                    FXStateChunk=ultraschall.GetFXStateChunk(TrackStateChunk)

                    -- set the new parameter-aliasname
                    retval, alteredFXStateChunk = ultraschall.SetParmAlias2_FXStateChunk(FXStateChunk, 1, 1, aliasname)

                    -- set the altered FXStateChunk to the TrackStateChunk and commit it to the first track
                    retval, TrackStateChunk=ultraschall.SetFXStateChunk(TrackStateChunk, alteredFXStateChunk)
                    retval = ultraschall.SetTrackStateChunk_Tracknumber(1, TrackStateChunk)

There are more functions related to Parameter Aliases, that give you more possibilities:

GetAllParmAliasNames_FXStateChunk - this gets all aliasnames from an fx within an FXStateChunk
CountParmAlias_FXStateChunk - this counts all aliasnames of an fx within an FXStateChunk
DeleteParmAlias_FXStateChunk - this deletes an aliasname of an fx within an FXStateChunk, indexed by existing aliasnames
DeleteParmAlias2_FXStateChunk - this deletes an aliasname of an fx within an FXStateChunk, indexed by parameterindex
SetParmAlias_FXStateChunk - this sets an aliasname of an fx within an FXStateChunk, indexed by existing aliasnames
SetParmAlias2_FXStateChunk - this sets an aliasname of an fx within an FXStateChunk, indexed by parameterindex
GetParmAlias_FXStateChunk - this gets an aliasname of an fx within an FXStateChunk, indexed by existing aliasnames
GetParmAlias2_FXStateChunk - this gets an aliasname of an fx within an FXStateChunk, indexed by parameterindex

In the next chapter, we'll discuss, how to map MIDI and OSC by learning to parameters.



^ Parameter Mapping Learn

Another thing I implemented is Parameter-Mapping Learn. This is the dialog you can reach, when opening the FXChain, hitting the Param-menu -> Learn -> one of the parameters.
These work in principle with the same approach as ParmAlias in the previous chapter:

To add a learn for a parameter, you

  1. first need to get the FXStateChunk, in which you want to add/get/set parameter-learns
  2. then you can use AddParmLearn_FXStateChunk to add a new parameter-learn for a parameter in the FXStateChunk
  3. Set the FXStateChunk back into the original statechunk(TrackStateChunk or MediaItemStateChunk)

Means, first get the FXStateChunk, then you can manipulate it via the ParmLearn-functions and then you set the manipulated one back into it.

There are more functions related to ParameterLearn, that give you more possibilities:

CountParmLearn_FXStateChunk - this counts all ParmLearns available of an fx within an FXStateChunk
DeleteParmLearn_FXStateChunk - this deletes a ParmLearn of an fx within an FXStateChunk, indexed by existing ParmLearns
SetParmLearn_FXStateChunk - this sets a ParmLearn of an fx within an FXStateChunk, indexed by existing ParmLearns
GetParmLearn_FXStateChunk - this gets a ParmLearn of an fx within an FXStateChunk, indexed by existing ParmLearns

These should allow you to add/set/delete ParmLearns with all possible options you can set in the dialog.

Due API-limitations, I haven't got around some get/set/delete-functions, that allow you to index by parameter-index, only by already existing ones.
So for the moment, if you want to set/delete a ParmLearn-entry for a specific parameter-index, you should check all of them via GetParmLearn_FXStateChunk(which returns the parameter-index as well) and use its "order-index".
For instance: if you want to delete a possibly existing ParmLearn-entry for parameter 16, check all existing ones, if you find one with parameter-index 16.
If that is the case with the second ParmLearn, you delete the second one. If you can't find one, then you don't need to delete it.
Will try to make this more convenient in the future. I promise.



^ Parameter LFO Learn

LFO-Learn is something, that can be set via the Learn Phase-button in the LFO-section of the ParameterModulation-dialog(more on that later).
And it can be set as well, in very much the same way, as ParmLearn in the previous chapter.

To add a learn for a parameter, you

  1. first need to get the FXStateChunk, in which you want to add/get/set parameter-learns
  2. then you can use AddParmLFOLearn_FXStateChunk to add a new LFO-learn for a parameter in the FXStateChunk
  3. Set the FXStateChunk back into the original statechunk(TrackStateChunk or MediaItemStateChunk)

Means, first get the FXStateChunk, then you can manipulate it via the ParmLFOLearn-functions and then you set the manipulated one back into it.

There are more functions related to ParameterLFOLearn, that give you more possibilities:

CountParmLFOLearn_FXStateChunk - this counts all LFOLearns available of an fx within an FXStateChunk
DeleteParmLFOLearn_FXStateChunk - this deletes a LFOLearn of an fx within an FXStateChunk, indexed by existing LFOLearns
SetParmLFOLearn_FXStateChunk - this sets a LFOLearn of an fx within an FXStateChunk, indexed by existing LFOLearns
GetParmLFOLearn_FXStateChunk - this gets a LFOLearn of an fx within an FXStateChunk, indexed by existing LFOLearns

These should allow you to add/set/delete LFOLearns with all possible options you can set in the dialog.

Important: the parameter midi_note some of these functions have, is a multibyte value, which allows you setting MIDI-Notes as well as Chan/CC.
So if you want to add MIDI chan1-Note 0, you set the value to
144, MIDI Chan 1 Note 1 (Byte1=144, Byte2=0)
while MIDI-chan1-Note 1 must be set as
400, MIDI Chan 1 Note 1 (Byte1=144, Byte2=1)

Due API-limitations, I haven't got around some get/set/delete-functions, that allow you to index by parameter-index, only by already existing ones.
So for the moment, if you want to set/delete an LFOLearn-entry for a specific parameter-index, you should check all of them via GetParmLFOLearn_FXStateChunk(which returns the parameter-index as well) and use its "order-index".
For instance: if you want to delete a possibly existing LFOLearn-entry for parameter 16, check all existing ones, if you find one with parameter-index 16.
If that is the case with the second LFOLearn, you delete the second one. If you can't find one, then you don't need to delete it.
Will try to make this more convenient in the future. I promise.



^ Parameter Modulation Introduction

You already learnt about Learn and LFO Learn. You also learnt about giving parameters different aliasnames.
But: you can do even more with parameter-modulation, which also allows linking of parameters.
This can be achieved by openeing the FXChain, hitting the Param-button -> FX parameter list -> Parameter modulation/MIDI link.

If you do that, a nice little window appears, which allows you all sorts of automation/modulation/linking of parameters.

Up until now, this wasn't programmable.

It's time to leave the "Up until now" in the past.

Welcome to parameter-modulation!

PS: It's good to have some experience in using ParameterModulation before attempting to script it.



^ Parameter Modulation: the ParmModTable

A ParmModTable is a table, which holds all settings of a parameter-modulation in one table.
This allows you to easily set settings the way you want. Just create one, alter it, commit it to a parameter.
Or get one from an already existing ParameterModulation.

But before we get into how to work with them, let's have a look at the table itself.

The table has numerous entries, that can be set or unset. The following screenshot shows you, which setting is stored with which entry:

Parameter Modulation-dialog and all ParmModTable-entries shown at their respective settings

As you can see, the checkbox for LFO is stored in the ParmModTable-entry "LFO".
The attack-slider of the Audio Control Signal-section is stored in the ParmModTable-entry "AUDIOCONTROL_ATTACK", etc.
So if you want to know, which setting that you want to alter is stored in which ParmModTable-entry, just have a look at this picture to find it.

Now, what are the actual entries of the ParmModTable and their meanings? Glad you ask:

ParmModTable["PARAM_NR"]               - the index of the parameter that you want to modulate; 1 for the first, 2 for the second, etc
ParmModTable["PARAM_TYPE"]             - the type of the parameter, usually "", "wet" or "bypass"

ParmModTable["PARAMOD_ENABLE_PARAMETER_MODULATION"]  
                                        - Enable parameter modulation, baseline value(envelope overrides)-checkbox; 
                                          true, checked; false, unchecked
ParmModTable["PARAMOD_BASELINE"]       - Enable parameter modulation, baseline value(envelope overrides)-slider; 
                                            0.000 to 1.000

ParmModTable["AUDIOCONTROL"]           - is the Audio control signal(sidechain)-checkbox checked; true, checked; false, unchecked
                                            Note: if true, this needs all other AUDIOCONTROL_-entries to be set
ParmModTable["AUDIOCONTROL_CHAN"]      - the Track audio channel-dropdownlist; When stereo, the first stereo-channel; 
                                          nil, if not available
ParmModTable["AUDIOCONTROL_STEREO"]    - 0, just use mono-channels; 1, use the channel AUDIOCONTROL_CHAN plus 
                                            AUDIOCONTROL_CHAN+1; nil, if not available
ParmModTable["AUDIOCONTROL_ATTACK"]    - the Attack-slider of Audio Control Signal; 0-1000 ms; nil, if not available
ParmModTable["AUDIOCONTROL_RELEASE"]   - the Release-slider; 0-1000ms; nil, if not available
ParmModTable["AUDIOCONTROL_MINVOLUME"] - the Min volume-slider; -60dB to 11.9dB; must be smaller than AUDIOCONTROL_MAXVOLUME; 
                                          nil, if not available
ParmModTable["AUDIOCONTROL_MAXVOLUME"] - the Max volume-slider; -59.9dB to 12dB; must be bigger than AUDIOCONTROL_MINVOLUME; 
                                          nil, if not available
ParmModTable["AUDIOCONTROL_STRENGTH"]  - the Strength-slider; 0(0%) to 1000(100%)
ParmModTable["AUDIOCONTROL_DIRECTION"] - the direction-radiobuttons; -1, negative; 0, centered; 1, positive

ParmModTable["LFO"]                    - if the LFO-checkbox checked; true, checked; false, unchecked
                                            Note: if true, this needs all LFO_-entries to be set
ParmModTable["LFO_SHAPE"]              - the LFO Shape-dropdownlist; 
                                            0, sine; 1, square; 2, saw L; 3, saw R; 4, triangle; 5, random
                                            nil, if not available
ParmModTable["LFO_SHAPEOLD"]           - use the old-style of the LFO_SHAPE; 
                                            0, use current style of LFO_SHAPE; 
                                            1, use old style of LFO_SHAPE; 
                                            nil, if not available
ParmModTable["LFO_TEMPOSYNC"]          - the Tempo sync-checkbox; true, checked; false, unchecked
ParmModTable["LFO_SPEED"]              - the LFO Speed-slider; 0(0.0039Hz) to 1(8.0000Hz); nil, if not available
ParmModTable["LFO_STRENGTH"]           - the LFO Strength-slider; 0.000(0.0%) to 1.000(100.0%)
ParmModTable["LFO_PHASE"]              - the LFO Phase-slider; 0.000 to 1.000; nil, if not available
ParmModTable["LFO_DIRECTION"]          - the LFO Direction-radiobuttons; -1, Negative; 0, Centered; 1, Positive
ParmModTable["LFO_PHASERESET"]         - the LFO Phase reset-dropdownlist; 
                                            0, On seek/loop(deterministic output)
                                            1, Free-running(non-deterministic output)
                                            nil, if not available

ParmModTable["PARMLINK"]               - the Link from MIDI or FX parameter-checkbox
                                          true, checked; false, unchecked
ParmModTable["PARMLINK_LINKEDPLUGIN"]  - the selected plugin; nil, if not available
                                         Will be ignored, when PARMLINK_LINKEDPLUGIN_RELATIVE is set.
                                            -1, nothing selected yet
                                            -100, MIDI-parameter-settings
                                            1 - the first fx-plugin
                                            2 - the second fx-plugin
                                            3 - the third fx-plugin, etc
ParmModTable["PARMLINK_LINKEDPLUGIN_RELATIVE"] - the linked plugin relative to the current one in the FXChain
                                               - 0, use parameter of the current fx-plugin
                                               - negative, use parameter of a plugin above of the current plugin(-1, the one above; -2, two above, etc)
                                               - positive, use parameter of a plugin below the current plugin(1, the one below; 2, two below, etc)
                                               - nil, uses only the plugin linked absolute(the one linked with PARMLINK_LINKEDPARMIDX)
ParmModTable["PARMLINK_LINKEDPARMIDX"] - the id of the linked parameter; -1, if none is linked yet; nil, if not available
                                            When MIDI, this is irrelevant.
                                            When FX-parameter:
                                              0 to n; 0 for the first; 1, for the second, etc                                                               
ParmModTable["PARMLINK_OFFSET"]        - the Offset-slider; -1.00(-100%) to 1.00(+100%); nil, if not available
ParmModTable["PARMLINK_SCALE"]         - the Scale-slider; -1.00(-100%) to 1.00(+100%); nil, if not available

ParmModTable["MIDIPLINK"]              - true, if any parameter-linking with MIDI-stuff; false, if not
                                            Note: if true, this needs all MIDIPLINK_-entries and PARMLINK_LINKEDPLUGIN=-100 to be set
ParmModTable["MIDIPLINK_BUS"]          - the MIDI-bus selected in the button-menu; 
                                            0 to 15 for bus 1 to 16; 
                                            nil, if not available
ParmModTable["MIDIPLINK_CHANNEL"]      - the MIDI-channel selected in the button-menu; 
                                            0, omni; 1 to 16 for channel 1 to 16; 
                                            nil, if not available
ParmModTable["MIDIPLINK_MIDICATEGORY"] - the MIDI_Category selected in the button-menu; nil, if not available
                                            144, MIDI note
                                            160, Aftertouch
                                            176, CC 14Bit and CC
                                            192, Program Change
                                            208, Channel Pressure
                                            224, Pitch
ParmModTable["MIDIPLINK_MIDINOTE"]     - the MIDI-note selected in the button-menu; nil, if not available
                                          When MIDI note:
                                               0(C-2) to 127(G8)
                                          When Aftertouch:
                                               0(C-2) to 127(G8)
                                          When CC14 Bit:
                                               128 to 159; see dropdownlist for the commands(the order of the list 
                                               is the same as this numbering)
                                          When CC:
                                               0 to 119; see dropdownlist for the commands(the order of the list 
                                               is the same as this numbering)
                                          When Program Change:
                                               0
                                          When Channel Pressure:
                                               0
                                          When Pitch:
                                               0
ParmModTable["WINDOW_ALTERED"]         - false, if the windowposition hasn't been altered yet; true, if the window has been altered
                                            Note: if true, this needs all WINDOW_-entries to be set
ParmModTable["WINDOW_ALTEREDOPEN"]     - if the position of the ParmMod-window is altered and currently open; 
                                            nil, unchanged; 0, unopened; 1, open
ParmModTable["WINDOW_XPOS"]            - the x-position of the altered ParmMod-window in pixels; nil, default position
ParmModTable["WINDOW_YPOS"]            - the y-position of the altered ParmMod-window in pixels; nil, default position
ParmModTable["WINDOW_RIGHT"]           - the right-position of the altered ParmMod-window in pixels; 
                                            nil, default position; only readable
ParmModTable["WINDOW_BOTTOM"]          - the bottom-position of the altered ParmMod-window in pixels; 
                                            nil, default position; only readable
    

Some entries depend on each other, which is important to know:
If you've set AUDIOCONTROL=true then you need to set all AUDIOCONTROL_-entries as well as X2 and Y2 to useful values. Same goes for LFO=true -> set all LFO_-entries to useful values, and PARMLINK=true, which needs all PARMLINK_-entries being set.
For PARMLINK you also need to keep in mind, that setting PARMLINK_LINKEDPLUGIN=-100(link to MIDI) means, that you need to set all MIDIPLINK-entries as well. But don't worry. If you set an entry to an invalid value or have invalid combinations, the ParmMod-functions, especially IsValidParmModTable, will return useful error-messages.
So just run SLEM() after calling one of the ParmMod-functions and it will tell you, what you did wrong.

Another note on plugin-linking:
You can index the linked plugin in two ways, absolute or relative.
To link a plugin absolute(by its FXChain-index), set the following entries the following ways:

ParmModTable["PARMLINK_LINKEDPLUGIN"]=the index of the plugin in the FXChain  
ParmModTable["PARMLINK_LINKEDPLUGIN_RELATIVE"]=nil  
    

To link a plugin relative to the current plugin's position, set the following entries the following way:

ParmModTable["PARMLINK_LINKEDPLUGIN"]=any integer number  
ParmModTable["PARMLINK_LINKEDPLUGIN_RELATIVE"]=0 for a parameter of the current plugin; 
                                               negative, to index a plugin above the current plugin; 
                                               positive, to index a plugin below the current plugin in the FXChain.  

Note: If you commit a ParmModTable to a currently opened Parameter Modulation, some things may not be updated in the gui. If you change the state of the AUDIOCONTROL/LFO/PARMLINK-checkbox or the Track audio channel-dropdownlist, it will only be updated after closing and reopening the window again. This affects only the gui, so if i.e. you enable the LFO-checkbox, it will be recognized as activated, even though the gui does not reflect that.

Take your time to toy around with it, as it's a little complex in the beginning but you will quickly get an idea of how it works.
To assist you with developing, there's a developer-tool ultraschall_developertool_MonitorParmModulation.lua, which shows the values of all ParmModTable-entries a certain Parameter Modulation of an fx in a certain track has.
Install it by running the ultraschall_Add_Developertools_To_Reaper.lua-action in the action-list.

Now that you know about Parameter-Modulation, you probably want to know, how to code it.



^ Parameter Modulation: Using ParmModTables

Let's assume, there is no parameter-modulation yet in any track or take. The best way to start is to create a new ParmModTable, which you can alter to your needs.

CreateDefaultParmModTable

                    table ParmModTable = ultraschall.CreateDefaultParmModTable()
                    

This returns a new ParmModTable with all settings set to their defaults, as if you would add a completely new ParameterModulation to your project. You must enable either AudioControl, LFO and Parameter/MIDI Linking-sections by altering the following entries accordingly, or applying the ParmModTable has no effect: ParmModTable["AUDIOCONTROL"] - the checkbox for "Audio control signal (sidechain)" ParmModTable["LFO"] - the checkbox for "LFO" ParmModTable["PARMLINK"] - the checkbox for "Link from MIDI or FX parameter" ParmModTable["PARAM_NR"] - the index of the fx-parameter for which the parameter-modulation-table is intended

this returned table can be altered to your needs. Now, if you have altered it, you probably want to check, if you changed it right or made some mistakes. For that, there's the function:

IsValidParmModTable

                    boolean retval = ultraschall.IsValidParmModTable(table ParmModTable)
                    

This checks, if a table is a valid ParmModTable. If not(retval=false) then its error-message will give you a hint at what you set wrong.
Use SLEM() to get the error-message to correct the wrong entries.
It will also take care of dependencies. So if you set certain settings which depends on another setting being set to a certain value, the error-message will tell you exactly which one must be set as well.
Use this extensively while debugging or you risk creating invalid ParmModTables!

Now the best ParmModTable doesn't do anything good, if you haven't put it to an FX. If you've read the previous chapters about LFOLearn or ParameterAliases, you might already know the drill:

  1. get the statechunk of a track or an item
  2. you need to get the FXStateChunk from the statechunk, in which you want to add/get/set the parameter modulation
  3. then you can commit the ParmModTable with AddParmModulationTable to add a new ParameterModulation for a parameter in the FXStateChunk
  4. Set the FXStateChunk back into the original statechunk(TrackStateChunk or MediaItemStateChunk)
  5. commit the statechunk to the original track/item

There are more functions to deal with ParmModulation, like:

Keep in mind, that you will not neccessarily be able to get a ParmModTable, alter it with one setting and set it again right away. As mentioned before in the previous chapter, some settings need other settings being set. So if you turn on LFO, you need to set all LFO_-entries of the ParmModTable as well before setting it, or it will not work!

Let's do some basic-examples.



^ Parameter Modulation: ParmModTable-examples

Let's assume you have one track in your project with ReaEQ set as first fx in the FXChain and this ReaEQ is set to the factory default-preset.

Let's add a parameter-modulation to the 13th parameter "bypass", which will be opened automatically and have LFO activated:

                    dofile(reaper.GetResourcePath().."/UserPlugins/ultraschall_api.lua")

                    -- let's create a default ParmModTable
                    ParmModTable = ultraschall.CreateDefaultParmModTable()

                    -- let's activate LFO; in the default ParmModTable, all LFO_-entries are set 
                    -- to Reaper's default already, so setting them isn't necessary in this case
                    ParmModTable["LFO"]=true

                    -- let's open the ParmModWindow
                    ParmModTable["WINDOW_ALTERED"]=true

                    -- apply it to the 13th parameter, which is the bypass parameter
                    ParmModTable["PARAM_NR"]=13
                    ParmModTable["PARAM_TYPE"]="bypass"

                    -- Now, let's do the magic:

                    -- get the TrackStateChunk from track 1
                    retval, TrackStateChunk = ultraschall.GetTrackStateChunk_Tracknumber(1)
                    -- get the FXStateChunk
                    FXStateChunk = ultraschall.GetFXStateChunk(TrackStateChunk)
                    -- add the new parameter modulation
                    alteredFXStateChunk = ultraschall.AddParmMod_ParmModTable(FXStateChunk, 1, ParmModTable) 
                    -- set the FXStateChunk back into the TrackStateChunk
                    retval, TrackStateChunk = ultraschall.SetFXStateChunk(TrackStateChunk, alteredFXStateChunk)
                    -- commit the TrackStateChunk back into the track 1
                    retval=ultraschall.SetTrackStateChunk_Tracknumber(1, TrackStateChunk)

This should have added now the parameter modulation for track 1, FX1(ReaEQ), parameter 13(bypass) and have opened the parameter modulation window.

Now, you may want to alter it, by enabling AudioControl. Here's how you do it:

                    dofile(reaper.GetResourcePath().."/UserPlugins/ultraschall_api.lua")

                    -- get the TrackStateChunk from track 1
                    retval, TrackStateChunk = ultraschall.GetTrackStateChunk_Tracknumber(1)
                    -- get the FXStateChunk
                    FXStateChunk = ultraschall.GetFXStateChunk(TrackStateChunk)
                    -- get the new parameter modulation
                    ParmModTable = ultraschall.GetParmModTable_FXStateChunk(FXStateChunk, 1, 13)

                    -- Let's enable AudioControl by setting the needed values of the ParmModTable.
                    -- As they are not set yet, we need to fill them ourselves.
                    -- In this example, I will fill them with Reaper's default values, but you can
                    -- choose others as well, as long as they are in their valid valueranges.
                    -- See description of CreateDefaultParmModTable() to get a list of Reaper's default values

                    ParmModTable["AUDIOCONTROL"]=true        -- enable AUDIOCONTROL
                    ParmModTable["AUDIOCONTROL_CHAN"]=1      -- use channel 1
                    ParmModTable["AUDIOCONTROL_STEREO"]=0    -- use only mono-channels
                    ParmModTable["AUDIOCONTROL_ATTACK"]=300  -- attack 300 ms
                    ParmModTable["AUDIOCONTROL_RELEASE"]=300 -- release 300 ms
                    ParmModTable["AUDIOCONTROL_MINVOLUME"]=-24 -- minvolume=-24dB
                    ParmModTable["AUDIOCONTROL_MAXVOLUME"]=0 -- maxvolume = 0dB
                    ParmModTable["AUDIOCONTROL_STRENGTH"]=1  -- strength = 1%
                    ParmModTable["X2"]=0.5 -- audio control signal shaping x-coordinate
                    ParmModTable["Y2"]=0.5 -- audio control signal shaping y-coordinate

                    -- set the altered ParmModTable back into the FXStateChunk
                    alteredFXStateChunk = ultraschall.SetParmMod_ParmModTable(FXStateChunk, 1, ParmModTable)

                    -- set the FXStateChunk back into the TrackStateChunk
                    retval, TrackStateChunk = ultraschall.SetFXStateChunk(TrackStateChunk, alteredFXStateChunk)
                    -- commit the TrackStateChunk back into the track 1
                    retval=ultraschall.SetTrackStateChunk_Tracknumber(1, TrackStateChunk)

And if you want to remove it again, you run the following example-code:

                    dofile(reaper.GetResourcePath().."/UserPlugins/ultraschall_api.lua")

                    -- get the TrackStateChunk from track 1
                    retval, TrackStateChunk = ultraschall.GetTrackStateChunk_Tracknumber(1)
                    -- get the FXStateChunk
                    FXStateChunk = ultraschall.GetFXStateChunk(TrackStateChunk)

                    -- delete the parameter modulation for fx=1, parameter=13(bypass) from the FXStateChunk
                    alteredFXStateChunk = ultraschall.DeleteParmModFromFXStateChunk(FXStateChunk, 1, 13)

                    -- set the FXStateChunk back into the TrackStateChunk
                    retval, TrackStateChunk = ultraschall.SetFXStateChunk(TrackStateChunk, alteredFXStateChunk)

                    -- commit the TrackStateChunk back into the track 1
                    retval=ultraschall.SetTrackStateChunk_Tracknumber(1, TrackStateChunk)

And that's it. This should bring you through parameter modulation in every way you wish.



^ Helper_Functions: Introduction

Beside all the concepts you just read about, I also added tons of functions and things to the API, that are helpful in one way or another in your everyday coding.
Manipulation of Lua's datastructures, clipboard-functions, stuff for undo-management, checking for the current system being either Mac, Windows or Other, Reaper's Configuration-Settings, User Interface-stuff, applying actions to MediaTracks and MediaItems and many other things.

In this chapter, I want you to introduce you to some of these things.



^ Helper_Functions: Clipboard Management

Getting and setting things to the clipboard is really helpful in many ways. With SWS 2.9.7 we got clipboard-functions, Reaper itself has clipboard-functions for items and such.
But: it's in parts not really convenient to use.
reaper.CF_SetClipboard() needs an obscure datastructure called Faststrings to circumvent a Lua-restriction of strings with a length of only 1023 bytes. Inconvenient to use, as you always need to create and destroy these FastStrings or they'll eat up memory for no good.
Putting MediaItems into the clipboard or finding out, which MediaItems are located in the clipboard is also possible but really inconvenient.
So I added functions to deal with that.

GetStringFromClipboard_SWS

                    string clipboard_string = ultraschall.GetStringFromClipboard_SWS()
                    

This is like reaper.CF_SetClipboard(), but is doing all the FastString-management for you in the background. In other words: it simply returns the string-contents from the clipboard.

I also added some faster ones, like:

FromClip

                    string clipboard_string = FromClip()
                    

This allows getting the current string stored in the clipboard. It's faster to type and takes care of all the SWS-management.

ToClip

                    ToClip(string toclipstring)
                    

This allows setting the current string in the clipboard. It's faster to type and takes care of all the SWS-management.
Important: if you have \0-characters in them, you need to convert them to something else, as otherwise the string could be truncated at the \0-character. So everything after a \0-character will be thrown away. This is important, if you want to store binary-strings into the clipboard.

When dealing with MediaItems, I also created functions for that. Let's put something into the clipboard.

PutMediaItemsToClipboard_MediaItemArray

                    boolean retval = ultraschall.PutMediaItemsToClipboard_MediaItemArray(MediaItemArray MediaItemArray)
                    

This function adds all MediaItems in MediaItem into the clipboard. It returns true, in case of success.

Putting MediaItems into the clipboard is cool, but sometimes you simply want to know, what already is in the clipboard.

GetMediaItemsFromClipboard

                    integer count, array MediaItemStateChunkArray = ultraschall.GetMediaItemsFromClipboard()
                    

This function returns the number of MediaItems in the clipboard as well as an array with all StateChunks of the MediaItems in the clipboard.
Why not the MediaItem-objects? Because, they only work within the current project, but MediaItems in the clipboard can also be from another project, not having any references to the current project. Working with StateChunks instead makes them independent from a certain project.
But what if you want to insert them into a project? Use InsertMediaItem_MediaItemStateChunk to insert a MediaItem into a project using it's StateChunk.
How does GetMediaItemsFromClipboard work? It inserts the Items from the clipboard at the end of the project, gets them, using GetAllMediaItemsBetween and deletes them again.



^ Helper_Functions: Data Manipulation

One of the bread and butter things you have to do in your daily programming is to deal with data-structures of many kinds, be it tables, strings, integers, bits, etc.
Dealing with them can sometimes be a pain in the butt, when using pure Lua. And as I'm lazy(did I already mention that?), I wrote me some functions to make things easier to deal with, especially with datastructures existing and fundamental in Reaper.

Dealing with Numbers and Bits

Dealing with Strings

Dealing with Tables



^ Helper_Functions: Undo Management

Undo-Management can be difficult to program in Reaper at times, so I added some things to deal with that to some extend.

PreventCreatingUndoPoint

                    ultraschall.PreventCreatingUndoPoint()
                    

This function prevents creating an undo-point in non-defer-scripts, which is sometimes desireable.

MakeFunctionUndoable

                    boolean retval, string current_UndoMessage, retvals_1, ..., retvals_2 
                                            = ultraschall.MakeFunctionUndoable(function Func, string UndoMessage, integer Flag, 
                                                                                            Func_parameters_1, ... Func_parameters_n)
                                                                                            

With this function, you can create an undopoint for a specific function.
You need to pass the function itself, an undo-message that is shown in the undo-history and an undo-flag.
The other parameters are all the parameters the function Func needs to run. Just write them in the order you would do, if you would run the function Func directly.
It will return a boolean(true for success), the current undo-message and alle the return-values returned by function Func.



^ Helper_Functions: Miscellaneous

In this final chapter, I show you around some nice things that didn't fit anywhere else in the docs. They may or may not be of use for you.

Checking for the current operating system

Reaper-related stuff

Ultraschall-Api-Variables

Configuration Settings

User Interface stuff

Apply actions to MediaTracks and MediaItems

ReaScript-console-stuff

print replacements

miscellaneous

and more and more and more....



^ Final words

I can't believe it, but after over one and a half years, I can finally write these last words for this mammoth-project.
But what can I say, what I haven't said in this docs?

Feel free to use it. Feel free to code with it. Feel free to learn from the functions I wrote. Dig into the functions-reference, which holds much much more.

If you like it, you can donate to our project at: ultraschall.fm/danke.

Let me take some more time to thank to the many Reaper-Forum-users, who provided me with ideas, hints, bits and pieces and tons of information, which itself informed this API.
I don't want to pick up anyone specific, so if you think, I want to thank you for your ideas and contributions, help or whatever, I do exactly that:
Thank you :)

Let me also thank the Reaper-guys Justin and Schwa, who created this piece of great software. Even though I constantly throw more and more Feature-requests into the forum, I'm deeply impressed with this piece of software and the more I dive into it, the more I can find.
Reaper is like a fractalised fractal: the closer you get, the more details are revealed.

More thanks also to the whole podcast community in Germany, be it at PodStock, ChaosCommunicationCongress, Sendegate, which gave me more than just countless hours of good entertainment and information. You probably saved my life countless times and I don't know how many podcasts I listened through in the endless hours of coding this monster.
Without you, I wouldn't have survived coding this.

Also thanks to the wonderful Ultraschall-team, whose focus on improving Reaper for podcasters shows, what is possible in Reaper and how powerful this software is. And who answered ma tons of questions.
And who survived my constantly being annyoing with this API ;)
I want to thank in particular Ralf Stockmann, whose rant in the Sendegate brought me into the project and back into programming.
Though, after not having seen the lights of the days for months now, I'm not sure, whether this was a good thing ;)

Thanks also to all the other people who supported me over the years. I dedicate this thing to you. And for someone special even a whole function, yay :D

So, that's all for now. Feel free to work with it and have fun.

I go to sleep now. I think, I deserved it :)

Meo-Ada Mespotine, Leipzig(Germany) 30th of November 2018



  Automatically generated by Ultraschall-API 5.1 - 121 elements available