2018-03-12 16:31:33 -04:00
|
|
|
ardour {
|
2018-03-15 14:36:25 -04:00
|
|
|
["type"] = "EditorAction",
|
|
|
|
name = "Mixer Store",
|
|
|
|
author = "Ardour Lua Taskforce",
|
2018-03-21 12:52:21 -04:00
|
|
|
description = [[Stores the current Mixer state as a file that can be read and recalled arbitrarily.
|
|
|
|
Supports: processor settings, grouping, mute, solo, gain, trim, pan and processor ordering, plus re-adding certain deleted plugins.]]
|
2018-03-12 16:31:33 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
function factory() return function()
|
|
|
|
|
2018-03-19 13:24:46 -04:00
|
|
|
function new_plugin(name)
|
|
|
|
local plugin = nil
|
|
|
|
for x = 0, 6 do
|
|
|
|
plugin = ARDOUR.LuaAPI.new_plugin(Session, name, x, "")
|
|
|
|
if not(plugin:isnil()) then break end
|
|
|
|
end return plugin
|
|
|
|
end
|
|
|
|
|
2018-03-20 14:01:48 -04:00
|
|
|
function group_by_id(id)
|
|
|
|
local group = nil
|
2018-03-21 12:52:21 -04:00
|
|
|
local id = tonumber(id)
|
2018-03-20 14:01:48 -04:00
|
|
|
for g in Session:route_groups():iter() do
|
|
|
|
local group_id = tonumber(g:to_stateful():id():to_s())
|
|
|
|
if group_id == id then group = g end
|
|
|
|
end return group
|
|
|
|
end
|
|
|
|
|
2018-03-21 12:52:21 -04:00
|
|
|
function route_group_interrogate(t)
|
|
|
|
local group = false
|
|
|
|
for g in Session:route_groups():iter() do
|
|
|
|
for r in g:route_list():iter() do
|
|
|
|
if r:name() == t:name() then group = g:to_stateful():id():to_s() end
|
|
|
|
end
|
|
|
|
end return group
|
|
|
|
end
|
|
|
|
|
2018-03-15 14:36:25 -04:00
|
|
|
local path = ARDOUR.LuaAPI.build_filename(Session:path(), "export", "params.lua")
|
|
|
|
function mark()
|
2018-03-20 14:01:48 -04:00
|
|
|
|
2018-03-15 14:36:25 -04:00
|
|
|
local file = io.open(path, "w")
|
|
|
|
file:write("") --empty current file from last run
|
|
|
|
file:close()
|
2018-03-20 14:01:48 -04:00
|
|
|
|
|
|
|
local g_route_str, group_str = "", ""
|
|
|
|
local i = 0
|
2018-03-21 12:52:21 -04:00
|
|
|
for g in Session:route_groups():iter() do --@ToDo: Color, and other bools
|
|
|
|
group_str = "instance = {group_id = " .. g:to_stateful():id():to_s() .. ", name = " .. "\"" .. g:name() .. "\"" .. ", routes = {"
|
2018-03-20 14:01:48 -04:00
|
|
|
for t in g:route_list():iter() do
|
|
|
|
g_route_str = g_route_str .."[".. i .."] = " .. t:to_stateful():id():to_s() .. ","
|
|
|
|
i = i + 1
|
|
|
|
end
|
|
|
|
group_str = group_str .. g_route_str .. "}}"
|
|
|
|
end
|
|
|
|
|
|
|
|
if not(group_str == "") then --sometimes there are no groups in the session
|
|
|
|
file = io.open(path, "a")
|
|
|
|
file:write(group_str, "\r\n")
|
|
|
|
file:close()
|
|
|
|
end
|
|
|
|
|
2018-03-15 14:36:25 -04:00
|
|
|
for r in Session:get_routes():iter() do
|
|
|
|
if r:is_monitor () or r:is_auditioner () then goto nextroute end -- skip special routes
|
2018-03-13 15:51:47 -04:00
|
|
|
|
2018-03-15 14:36:25 -04:00
|
|
|
local order = ARDOUR.ProcessorList()
|
|
|
|
local x = 0
|
|
|
|
repeat
|
|
|
|
local proc = r:nth_processor(x)
|
|
|
|
if not proc:isnil() then
|
|
|
|
order:push_back(proc)
|
|
|
|
end
|
|
|
|
x = x + 1
|
|
|
|
until proc:isnil()
|
2018-03-14 10:58:56 -04:00
|
|
|
|
2018-03-16 12:11:54 -04:00
|
|
|
local route_str, proc_order_str, cache_str = "", "", ""
|
2018-03-15 14:36:25 -04:00
|
|
|
local rid = r:to_stateful():id():to_s()
|
|
|
|
local pan = r:pan_azimuth_control()
|
|
|
|
if pan:isnil() then pan = false else pan = pan:get_value() end --sometimes a route doesn't have pan, like the master.
|
2018-03-14 10:58:56 -04:00
|
|
|
|
2018-03-15 14:36:25 -04:00
|
|
|
local on = 0
|
|
|
|
for p in order:iter() do
|
|
|
|
local pid = p:to_stateful():id():to_s()
|
2018-03-20 14:01:48 -04:00
|
|
|
if not(string.find(p:display_name(), "latcomp")) then
|
2018-03-20 16:41:20 -04:00
|
|
|
proc_order_str = proc_order_str .. "[" .. on .. "] = " .. pid ..","
|
2018-03-20 14:01:48 -04:00
|
|
|
cache_str = cache_str .. "[" .. pid .. "] = " .. "\"" .. p:display_name() .. "\"" ..","
|
|
|
|
end
|
2018-03-15 14:36:25 -04:00
|
|
|
on = on + 1
|
|
|
|
end
|
2018-03-13 15:51:47 -04:00
|
|
|
|
2018-03-21 12:52:21 -04:00
|
|
|
route_str = "instance = {route_id = " .. rid .. ", gain_control = " .. r:gain_control():get_value() .. ", trim_control = " .. r:trim_control():get_value() .. ", pan_control = " .. tostring(pan) .. ", muted = " .. tostring(r:muted()) .. ", soloed = " .. tostring(r:soloed()) .. ", order = {" .. proc_order_str .."}, cache = {" .. cache_str .. "}, group = " .. tostring(route_group_interrogate(r)) .. "}"
|
2018-03-15 14:36:25 -04:00
|
|
|
file = io.open(path, "a")
|
|
|
|
file:write(route_str, "\r\n")
|
|
|
|
file:close()
|
2018-03-14 10:58:56 -04:00
|
|
|
|
2018-03-15 14:36:25 -04:00
|
|
|
local i = 0
|
|
|
|
while true do
|
|
|
|
local params = {}
|
|
|
|
local proc_str, params_str = "", ""
|
|
|
|
local proc = r:nth_plugin (i)
|
|
|
|
if proc:isnil () then break end
|
|
|
|
local active = proc:active()
|
|
|
|
local id = proc:to_stateful():id():to_s()
|
|
|
|
local plug = proc:to_insert ():plugin (0)
|
|
|
|
local n = 0 -- count control-ports
|
|
|
|
for j = 0, plug:parameter_count () - 1 do -- iterate over all plugin parameters
|
|
|
|
if plug:parameter_is_control (j) then
|
|
|
|
local label = plug:parameter_label (j)
|
|
|
|
if plug:parameter_is_input (j) and label ~= "hidden" and label:sub (1,1) ~= "#" then
|
|
|
|
local _, _, pd = ARDOUR.LuaAPI.plugin_automation(proc, n)
|
|
|
|
local val = ARDOUR.LuaAPI.get_processor_param(proc, j, true)
|
2018-03-21 12:52:21 -04:00
|
|
|
print(proc:display_name(), label, val)
|
2018-03-19 13:24:46 -04:00
|
|
|
params[n] = val
|
2018-03-15 14:36:25 -04:00
|
|
|
end
|
|
|
|
n = n + 1
|
|
|
|
end
|
|
|
|
end
|
|
|
|
i = i + 1
|
|
|
|
for k, v in pairs(params) do
|
|
|
|
params_str = params_str .. "[".. k .."] = " .. v .. ","
|
|
|
|
end
|
|
|
|
proc_str = "instance = {plugin_id = " .. id .. ", parameters = {" .. params_str .. "}, active = " .. tostring(active) .. "}"
|
|
|
|
file = io.open(path, "a")
|
|
|
|
file:write(proc_str, "\r\n")
|
|
|
|
file:close()
|
|
|
|
end
|
|
|
|
::nextroute::
|
|
|
|
end
|
|
|
|
end
|
2018-03-16 12:11:54 -04:00
|
|
|
local invalidate = {}
|
2018-03-15 14:36:25 -04:00
|
|
|
function recall()
|
|
|
|
local file = io.open(path, "r")
|
|
|
|
assert(file, "File not found!")
|
|
|
|
for l in file:lines() do
|
2018-03-16 12:11:54 -04:00
|
|
|
--print(l)
|
2018-03-13 15:51:47 -04:00
|
|
|
|
2018-03-20 14:01:48 -04:00
|
|
|
local plugin, route, group = false, false, false
|
2018-03-15 14:36:25 -04:00
|
|
|
local f = load(l)
|
|
|
|
f ()
|
2018-03-13 15:51:47 -04:00
|
|
|
|
2018-03-15 14:36:25 -04:00
|
|
|
if instance["route_id"] ~= nil then route = true end
|
|
|
|
if instance["plugin_id"] ~= nil then plugin = true end
|
2018-03-20 14:01:48 -04:00
|
|
|
if instance["group_id"] ~= nil then group = true end
|
|
|
|
|
|
|
|
if group then
|
|
|
|
local g_id = instance["group_id"]
|
|
|
|
local routes = instance["routes"]
|
2018-03-21 12:52:21 -04:00
|
|
|
local name = instance["name"]
|
2018-03-20 14:01:48 -04:00
|
|
|
local group = group_by_id(g_id)
|
2018-03-21 12:52:21 -04:00
|
|
|
if group == nil then group = Session:new_route_group(name) end
|
2018-03-20 14:01:48 -04:00
|
|
|
for k, v in pairs(routes) do
|
|
|
|
local rt = Session:route_by_id(PBD.ID(v))
|
|
|
|
if not(rt:isnil()) then group:add(rt) end
|
|
|
|
end
|
|
|
|
end
|
2018-03-13 15:51:47 -04:00
|
|
|
|
2018-03-15 14:36:25 -04:00
|
|
|
if route then
|
2018-03-21 12:52:21 -04:00
|
|
|
|
2018-03-15 14:36:25 -04:00
|
|
|
local old_order = ARDOUR.ProcessorList()
|
2018-03-20 14:01:48 -04:00
|
|
|
local r_id = PBD.ID(instance["route_id"])
|
2018-03-21 12:52:21 -04:00
|
|
|
local muted, soloed = instance["muted"], instance["soloed"]
|
2018-03-19 13:24:46 -04:00
|
|
|
local order = instance["order"]
|
|
|
|
local cache = instance["cache"]
|
2018-03-21 12:52:21 -04:00
|
|
|
local group = instance["group"]
|
2018-03-19 13:24:46 -04:00
|
|
|
local gc, tc, pc = instance["gain_control"], instance["trim_control"], instance["pan_control"]
|
|
|
|
|
2018-03-20 14:01:48 -04:00
|
|
|
local rt = Session:route_by_id(r_id)
|
2018-03-15 14:36:25 -04:00
|
|
|
if rt:isnil() then goto nextline end
|
2018-03-21 12:52:21 -04:00
|
|
|
local cur_group_id = route_group_interrogate(rt)
|
|
|
|
if not(group) and (cur_group_id ~= false) then
|
|
|
|
local g = group_by_id(cur_group_id)
|
|
|
|
if g ~= nil then g:remove(rt) end
|
|
|
|
end
|
2018-03-19 13:24:46 -04:00
|
|
|
|
|
|
|
for k, v in pairs(order) do
|
2018-03-16 12:11:54 -04:00
|
|
|
local proc = Session:processor_by_id(PBD.ID(v))
|
|
|
|
if proc:isnil() then
|
2018-03-19 13:24:46 -04:00
|
|
|
for id, name in pairs(cache) do
|
2018-03-16 12:11:54 -04:00
|
|
|
if v == id then
|
|
|
|
proc = new_plugin(name)
|
2018-03-21 12:52:21 -04:00
|
|
|
if not(proc:isnil()) then
|
|
|
|
rt:add_processor_by_index(proc, 0, nil, true)
|
|
|
|
invalidate[v] = proc:to_stateful():id():to_s()
|
|
|
|
old_order:push_back(proc)
|
|
|
|
end
|
2018-03-16 12:11:54 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2018-03-21 12:52:21 -04:00
|
|
|
|
|
|
|
if muted then rt:mute_control():set_value(1, 1) else rt:mute_control():set_value(0, 1) end
|
|
|
|
if soloed then rt:solo_control():set_value(1, 1) else rt:solo_control():set_value(0, 1) end
|
2018-03-15 14:36:25 -04:00
|
|
|
rt:gain_control():set_value(gc, 1)
|
|
|
|
rt:trim_control():set_value(tc, 1)
|
|
|
|
if pc ~= false then rt:pan_azimuth_control():set_value(pc, 1) end
|
|
|
|
rt:reorder_processors(old_order, nil)
|
|
|
|
end
|
2018-03-12 16:31:33 -04:00
|
|
|
|
2018-03-15 14:36:25 -04:00
|
|
|
if plugin then
|
2018-03-19 13:24:46 -04:00
|
|
|
local enable = {}
|
|
|
|
local params = instance["parameters"]
|
|
|
|
local p_id = instance["plugin_id"]
|
|
|
|
local act = instance["active"]
|
|
|
|
|
|
|
|
for k, v in pairs(invalidate) do --invalidate any deleted plugin's id
|
|
|
|
if p_id == k then
|
|
|
|
p_id = v
|
2018-03-16 12:11:54 -04:00
|
|
|
end
|
|
|
|
end
|
2018-03-19 13:24:46 -04:00
|
|
|
|
2018-03-19 14:30:19 -04:00
|
|
|
local proc = Session:processor_by_id(PBD.ID(p_id))
|
2018-03-15 14:36:25 -04:00
|
|
|
if proc:isnil() then goto nextline end
|
2018-03-19 13:24:46 -04:00
|
|
|
local plug = proc:to_insert():plugin(0)
|
|
|
|
|
|
|
|
for k, v in pairs(params) do
|
|
|
|
local label = plug:parameter_label(k)
|
|
|
|
if string.find(label, "Assign") or string.find(label, "Enable") then --@ToDo: Check Plugin type == LADSPA or VST?
|
|
|
|
enable[k] = v --queue any assignments/enables for after the initial parameter recalling to duck the 'in-on-change' feature
|
|
|
|
end
|
|
|
|
ARDOUR.LuaAPI.set_processor_param(proc, k, v)
|
|
|
|
end
|
|
|
|
|
|
|
|
for k, v in pairs(enable) do
|
2018-03-15 14:36:25 -04:00
|
|
|
ARDOUR.LuaAPI.set_processor_param(proc, k, v)
|
|
|
|
end
|
|
|
|
if act then proc:activate() else proc:deactivate() end
|
|
|
|
end
|
|
|
|
::nextline::
|
|
|
|
end
|
|
|
|
end
|
2018-03-12 16:31:33 -04:00
|
|
|
|
2018-03-15 14:36:25 -04:00
|
|
|
local dialog_options = {
|
|
|
|
{ type = "label", colspan= 10, title = "" },
|
|
|
|
{ type = "radio", colspan= 10, key = "select", title = "", values ={ ["1. Mark"] = "mark", ["2. Recall"] = "recall" }, default = "1. Mark"},
|
|
|
|
{ type = "label", colspan= 10, title = "" },
|
|
|
|
}
|
2018-03-12 16:31:33 -04:00
|
|
|
|
2018-03-15 14:36:25 -04:00
|
|
|
local rv = LuaDialog.Dialog("Mixer Store:", dialog_options):run()
|
|
|
|
assert(rv, 'Dialog box was canceled or is ' .. type(rv))
|
|
|
|
local c = rv["select"]
|
|
|
|
if c == "mark" then mark() end
|
|
|
|
if c == "recall" then recall() end
|
2018-03-12 16:31:33 -04:00
|
|
|
|
|
|
|
end end
|