-- AudioScience Iyo Dante/AES67 QSC plugin -- November 2018 PluginInfo = { Name = "AudioScience~Iyo Dante", Version = "1.0.0", Id = "b76fdbd7-72d3-408b-9bb8-0d3e2e018580", ShowDebug = false, Description = "AudioScience Iyo Dante plugin" } function GetPrettyName(props) return "AudioScience Iyo Dante (AES67 mode) 1.0.0" end function GetColor(props) return { 190, 83, 28 } end function GetProperties() props = { { Name = "ModelType", Type = "enum", Choices = {"8.8M", "16.16M", "32.32M", "16.0M", "0.16L", "32.0M", "0.32L"}, Value = "32.32M" -- default }, } return props end -- Get the I/O count from the model type property function GetIOCount(model_type) local input_count, output_count = 0 if model_type == "8.8M" then input_count = 8 output_count = 8 elseif model_type == "16.16M" then input_count = 16 output_count = 16 elseif model_type == "32.32M" then input_count = 32 output_count = 32 elseif model_type == "16.0M" then input_count = 16 elseif model_type == "0.16L" then output_count = 16 elseif model_type == "32.0M" then input_count = 32 elseif model_type == "0.32L" then output_count = 32 end return input_count, output_count end function ModelTypeToNo(model_type) local model_no = 0 if model_type == "8.8M" then model_no = 2701 elseif model_type == "16.16M" then model_no = 2702 elseif model_type == "32.32M" then model_no = 2703 elseif model_type == "16.0M" then model_no = 2705 elseif model_type == "0.16L" then model_no = 2708 elseif model_type == "32.0M" then model_no = 2706 elseif model_type == "0.32L" then model_no = 2709 end return model_no end -- Create additional channel group pages as necessary for the selected model type function GetPages(props) local pages = {} local io_pagenames = {'Channels 1 - 8', 'Channels 9 - 16', 'Channels 17 - 24', 'Channels 25 - 32'} local io_only_pagenames = {'Channels 1 - 16', 'Channels 17 - 32'} local model_type = props["ModelType"].Value if model_type == "8.8M" then table.insert( pages, { name = io_pagenames[1] } ) elseif model_type == "16.16M" then for idx = 1, 2 do table.insert( pages, { name = io_pagenames[idx] } ) end elseif model_type == "32.32M" then for idx = 1, 4 do table.insert( pages, { name = io_pagenames[idx] } ) end elseif model_type == "16.0M" or model_type == "0.16L" then table.insert( pages, { name = io_only_pagenames[1] } ) elseif model_type == "32.0M" or model_type == "0.32L" then for idx = 1, 2 do table.insert( pages, { name = io_only_pagenames[idx] } ) end end return pages end -- Define Controls function GetControls(props) input_count, output_count = GetIOCount(props["ModelType"].Value) return { -- 'Select a Device' group -- Device selection combobox { Name = "SelectedDevice", ControlType = "Text", UserPin = false, Count = 1, }, -- Selected device connection status text { Name = "SelectedDeviceStatus", ControlType = "Text", UserPin = false, Count = 1, }, -- 'Device' group -- Device info parameters text { Name = "DeviceInfo", ControlType = "Text", UserPin = true, PinStyle = "Output", Count = 5, }, -- 'Firmware' group -- Firmware Info parameters text { Name = "FirmwareInfo", ControlType = "Text", UserPin = true, PinStyle = "Output", Count = 2, }, -- 'Status' group -- Status Info parameters text { Name = "StatusInfo", ControlType = "Text", UserPin = true, PinStyle = "Output", Count = 2, }, -- Status LEDs { Name = "StatusLEDs", ControlType = "Indicator", IndicatorType = "Led", UserPin = false, Count = 2, }, -- Identify button { Name = "Identify", ControlType = "Button", ButtonType = "Momentary", UserPin = true, PinStyle = "Both", Count = 1, }, -- 'Settings' group -- LED Brightness combobox { Name = "LED_Brightness", ControlType = "Text", UserPin = true, PinStyle = "Both", Count = 1, }, -- 'Input/Transmit AES67 Details' group -- AES67 TX details listbox { Name = "TXAES67Details", ControlType = "Text", UserPin = false, Count = 1, }, -- 'Receive/Output AES67 Details' group -- AES67 RX Details listbox { Name = "RXAES67Details", ControlType = "Text", UserPin = false, Count = 1, }, -- 'Channel Strip' controls -- TX channel name { Name = "TXChannelName", ControlType = "Text", UserPin = false, Count = input_count, }, -- RX channel name { Name = "RXChannelName", ControlType = "Text", UserPin = false, Count = output_count, }, -- TX meter peak text { Name = "TXChanneldBFS", ControlType = "Text", UserPin = true, PinStyle = "Output", Count = input_count, }, -- RX meter peak text { Name = "RXChanneldBFS", ControlType = "Text", UserPin = true, PinStyle = "Output", Count = output_count, }, -- TX meter horz bar { Name = "TXChannelMeter", ControlType = "Knob", ControlUnit = "Integer", Min = -60, Max = 0, UserPin = false, Count = input_count, }, -- RX meter horz bar { Name = "RXChannelMeter", ControlType = "Knob", ControlUnit = "Integer", Min = -60, Max = 0, UserPin = false, Count = output_count, }, -- TX Gain text { Name = "TXChannelGain", ControlType = "Text", UserPin = false, Count = input_count, }, -- TX Gain knob { Name = "TXChannelLevelKnob", ControlType = "Knob", ControlUnit = "Integer", Min = 0, Max = 84, UserPin = true, PinStyle = "Both", Count = input_count, }, -- RX Level knob { Name = "RXChannelLevelKnob", ControlType = "Knob", ControlUnit = "Integer", Min = -10, Max = 24, UserPin = true, PinStyle = "Both", Count = output_count, }, -- TX Level text { Name = "TXChannelLevel", ControlType = "Text", UserPin = true, PinStyle = "Output", Count = input_count, }, -- RX Level text { Name = "RXChannelLevel", ControlType = "Text", UserPin = false, Count = output_count, }, -- TX phantom power button { Name = "TXChannelPhantom", ControlType = "Button", ButtonType = "Toggle", UserPin = true, PinStyle = "Both", Count = input_count, }, -- TX AES67 status { Name = "TXChannelAES67Status", ControlType = "Text", UserPin = false, Count = input_count, }, -- RX AES67 status { Name = "RXChannelAES67Status", ControlType = "Text", UserPin = false, Count = output_count, }, -- TX AES67 toggle button { Name = "TXChannelAES67Tgl", ControlType = "Button", ButtonType = "Toggle", UserPin = true, PinStyle = "Both", Count = input_count, }, -- RX AES67 transmitter selection combobox { Name = "SelectedAES67TX", ControlType = "Text", UserPin = true, PinStyle = "Both", Count = output_count, }, } end function GetControlLayout(props) layout = { } graphics = { } local device_info_labels = { "Model Name", "Model No.", "Hardware Rev.", "Serial No.", "Primary MAC" } local fw_vendors = { "AudioScience", "XMOS" } local status_types = { "Sync", "Sys" } local aes67_tx_labels = { 'Stream Name', 'Transport IP'} local aes67_rx_labels = { 'Channel', 'Connected To', 'Stream Ch.'} local model_type = props["ModelType"].Value -- White background table.insert(graphics, { Type = "GroupBox", StrokeWidth = 0, Position = {0, 0}, Size = {680, 680}, Fill = {255, 255, 255}, }) -- Logos table.insert(graphics, --ASI Logo { Type = "Svg", Image = "<svg      version="1.1"      xmlns="http://www.w3.org/2000/svg"      xmlns:xlink="http://www.w3.org/1999/xlink"      width="120px" height="55px"      viewBox="0 0 120 55" preserveAspectRatio="none">    <g> <image width="120" height="55" xlink:href="data:image/png;base64, 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"/>    </g> </svg>", Position = {20, 4}, Size = {120, 55} }) table.insert(graphics, --Iyo Logo { Type = "Svg", Image = "<svg      version="1.1"      xmlns="http://www.w3.org/2000/svg"      xmlns:xlink="http://www.w3.org/1999/xlink"      width="60px" height="27px"      viewBox="0 0 60 27" preserveAspectRatio="none">    <g> <image width="60" height="27" xlink:href="data:image/png;base64, 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"/>    </g> </svg>", Position = {500, 20}, Size = {60, 27} }) table.insert(graphics, --Dante Logo { Type = "Svg", Image = "PHN2ZyAgICAgIHZlcnNpb249IjEuMSIgICAgICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciICAgICAgeG1sbnM6eGxpbms9Imh0dHA6Ly93d3cudzMub3JnLzE5OTkveGxpbmsiICAgICAgd2lkdGg9Ijc2cHgiIGhlaWdodD0iMThweCIgICAgICB2aWV3Qm94PSIwIDAgNzYgMTgiIHByZXNlcnZlQXNwZWN0UmF0aW89Im5vbmUiPiAgICA8Zz4gPGltYWdlIHdpZHRoPSI3NiIgaGVpZ2h0PSIxOCIgeGxpbms6aHJlZj0iZGF0YTppbWFnZS9wbmc7YmFzZTY0LCBpVkJPUncwS0dnb0FBQUFOU1VoRVVnQUFBRXdBQUFBU0NBWUFBQUR4RXppc0FBQUFBWE5TUjBJQXJzNGM2UUFBQUFSblFVMUJBQUN4and2OFlRVUFBQUFKY0VoWmN3QUFEc1FBQUE3RUFaVXJEaHNBQUFiN1NVUkJWRmhIMVZnSmJFMWJGTjN2OVhYUVFhdGY2VUJOYlJOampWRVJmbUlJTWNkTXBBaGlUTkJHZm1JSUVwV28xQmhLUkVsRWtDQWlwaUJCRUxNbXFDbG1xVkl0V25OVno5OXI5NXo3N3F2WFZQOHZpWldjM0gzVzJlL2NlL2ZkMDNrT3hhQXE4T1RKRTNyMDZCRWRQSGlRVHA4K0xmT2lvaUxhc1dNSGpSczNUblErZi81TXZYcjFvcXRYcjFKd2NEQmhXNGZEUVpHUmtaU1ltRWdEQmd5Z1RwMDZVWHg4dk9qL3NZREJLc09KRXlmVXhJa1RZVkN2SXlzclMyc3E5ZkhqUjlXMmJWdXZldll4Y3VSSXhjYlh2L3J6VUtuQit2YnQ2L0dpb2FHaGF1WEtsZXJXclZzeWJ0eTRvWXFMaTdWMnVjSGF0Mjh2dXJWcTFWSlRwa3hSQXdjT1ZQWHIxL2ZZQjhQWDExZmw1dWJxWC80ZW5EdDNUZzBmUGx6R3k1Y3ZOVnQ5L0dDdzE2OWZxOERBUU92bGtwS1M5SW9uUG4zNnBNckt5dlRNMDJBeE1UR2FkZVBhdFd1cWVmUG0xcjRZSE5wNjlkZGorL2J0MW4zeHNhc0R2T3Y5Ky9kRmR2SUdGdkx6ODRuRGlsaUIyS1BvNU1tVGRPSENCVmw3OWVvVnNZZlIzTGx6cVgvLy90U3hZMGM2Y3VTSXJGWEU5Ky9mdGVRR0c1TnUzNzVOZ3djUDFneForZTkzd00vUFQwdEU3T0ZhK2pta3BxYlMyclZyYWVmT25XeHVHeEJLdkM2RGs3cG1sUm8xYXBTcVc3ZXV0V2JHMXExYnRZYW5oM0dpMTZ4M2NQSzM5amgyN0pobWZ5MXdIM05QL3ZpYXJScHNLTFYzNzE3Vm8wY1BOV2ZPSExlSGRlalFRU3BkUkVRRThjdFQ3ZHExNmZyMTYrUnl1V2pQbmozMDVzMGJpb3VMRXlzUEd6Wk1mbVAvYXRYQi9Qbnp0VVIwOCtaTkxSRzlmLzllS20rVEprMmt3clp1M1pvT0h6NHNGWG5Ka2lXVWtwSkNaOCtlRmQwclY2Nkl0Mk92dkx3OFdyNTh1VHd6bm9sekoxMjhlRkgwQURZV3JWNjlXcytJbGkxYlJ2UG16YU8zYjkvS0hGVi85dXpaRkJJU0l1L0xxWVAyNzk4dmtUSjA2RkE2Zi80OFRaMDZsWnhPTmhlc2VPREFBYkU4UDZUaUJ4TEw4ZzFWVUZDUThOMjZkVk9uVHAwU0h1QWZDODh2cDVucWVSZ0FQUXdPVVptL2VQRkN0V2pSd3VMdEExL1h5R3dZMGQrNGNhUEYxYWxUeDVMdGcxc2MwUjB6Wm96WGRTUi9GREp2YXhqc1JQTDdvMGVQcWsyYk5rblVpY0ZRT2FBd2FOQWdVUUJnUEhCc1ljMjRNWDc4ZUZtckNZTjE3dHhaNXR6RFdSekcyTEZqNWQ1MkRtUFZxbFdpajNSZzV4czFhcVJtelpxbG1qVnJabkY5K3ZRUjNYNzkrbm5vbXNIZTdUSEgreTljdUZENStQaFkzSkFoUTJRUEE3cDM3NTYxYU5DbFN4ZVo5K3paVXpPZW1EQmhncXpYaE1HNGdLam56NTliY3c1RHJWRU9WS2lFaEFScjNadkJKazJhSkp6QmlCRWpoRy9Zc0tGNjhPQ0JjSWdZbzE5YVdpcGM0OGFOWlE3bktDZ29FTTZnZCsvZWxyNGRUaTZYekJHMWF0Vktyc2dwZCs3Y0liWXlMVjY4V0xpYUJqK2Nsb2k2ZCs4dU9jWGcrUEhqV2lvSEZ5SmF0R2lSbm5rSGU0R1d5bEd2WGoyNUloZWJQUFh3NFVPNUFqaTE4THZUbHk5ZlpNNnBnRFpzMkVCTGx5NlZYSm1XbGliNTA0QmJFaTBST1hIY0FaQkFBU1RJZCsvZVNZTG4zQ1ZjUmVCbS93ZnIxNi9YRXNseDZjeVpNeUxqU0lXalZFV01IajJhQWdJQzlPeEhORzNhVkV2bHNMYzEzQ3ZLVlJLMkJ0b0tPTVdIRHg5a25wT1RJODRCWThGb0hKYkVuaXhyQUlxZmdmUHk1Y3NpNE1HQng0OGZ5eFVlaCtySXlaVXlNek5sckZtelJ2cXlzTEF3MGZrdmVQYnNHVzNidGsxa2JsWGtuSW5xREZUV0g2RnkyVis0SXFyYlZ3SGZ2bjJ6akJrYkd5dnZqMmN4QTMxajE2NWRxV1hMbHBiSEFxNm5UNStLZ0EwQU05KzNiNStNaWtDWk5lRmJVbElpMTU5RmRuWTI4WkdKT0dmSkhJMXJlSGk0UEJ6dWkvRDUrdlVyK2Z2N3k3ckJvVU9IcEptdUROWDFlSGdnL2dTQVIyTmY3ak1wUFQxZHI3ckIvUmR4UWRTemNqampFeElvaFBzWDQvTG93eUtqb2doOEhHOXFCdVlSYk9sRVBna0VjNzhDNEdXOW9XSi9oajRHdlJIeUZSK1JoSVBIbU41b3dZSUZjZ1U0MldySmpYWHIxbW1wWnNCSm4vajRaMzF3bkdCd2tyRURxWW1MaDBUVDVzMmJOY3RBNXE4SjJLdGtWWU05VlAvS0RWUkhzODROcU5xeVpZdDAyZndoUFg3cnJVcWkwdHN4YmRvMDRmbURxa3VYTGdtM2E5Y3VTLy91M2J2QzRYeG9PQXcrQXFuZHUzZi8wTTRVRmhhS1B1QXFURTZoMHZ3OENzOU1KMWRzREJXbFpWQkpkZzQ1L0NwUHN0eUdVMWx4QVlYK2swb0JmeWNKQlRjdkxpNFd1VEtnaUV5ZVBKbVNrNU0xNHdhK0tLb1Z1bnJzZzlBMTRON0txbklJV2NDZURpcW1CcU9EazROWk14eGdaSVFsT241NEdKQ1JrU0ZYQTRSc1ZsYVdSeVM1ZkdJaVNmazR5T25uU3c0ZkovbUUvVVd1cUdoeStGZGhzQUIvY2dRRmFvSWtwR2ZPbkVtNXVia2VJUm5GNFkwRFBSSW5ONWMvNUNlRDZPaG9LUWdvN3lnd3lIUFFSN2dpTkZhc1dDR0dURW9xLzBEdDJyV2pHVE5tU0VIQVBlemdSbFdPT2ZnRGdYc3Q0ZHEwYVVQVHAwK1hJeGZ1WllEY3hYMmxISzdaQzZWeU5talFRTG9HOEh5SzBKb0EwYi8yQnFxRmNjVFdrd0FBQUFCSlJVNUVya0pnZ2c9PSIvPiAgICA8L2c+IDwvc3ZnPg==", Position = {595, 30}, Size = {57, 14} }) -- Device controls (Left column) -- 'Select a Device' grouping -- Groupbox table.insert(graphics, { Type = "GroupBox", Text = "Select a Device", StrokeWidth = 1, Position = {2, 72}, Size = {208, 70}, Fill = {255, 255, 255}, CornerRadius = 3, HTextAlign = "Center", }) -- Device selection combobox layout["SelectedDevice"] = { Style = "ComboBox", PrettyName = "Select a device", Position = {10, 92}, Size = {192, 20}, Fill = {255, 255, 255}, TextSize = 10, } -- Selected device connection status text layout["SelectedDeviceStatus"] = { Style = "Text", PrettyName = "Selected device connection status", Position = {10, 117}, Size = {192, 20}, Color = {255, 255, 255}, IsReadOnly = true, TextSize = 10, HTextAlign = "Center", StrokeWidth = 0, } -- 'Device' grouping -- Groupbox table.insert(graphics, { Type = "GroupBox", Text = "Device", StrokeWidth = 1, Position = {2, 150}, Size = {208, 100}, Fill = {255, 255, 255}, CornerRadius = 3, HTextAlign = "Center", }) for k, v in ipairs(device_info_labels) do -- Parameter label table.insert(graphics, { Type = "Label", Text = v, Position = {26, 151 + (15 * k)}, Size = {63, 15}, TextSize = 10, Color = {0, 0, 0}, HTextAlign = "Right", }) -- Value text layout["DeviceInfo "..k] = { Style = "Text", PrettyName = "Device~Info~"..v, Position = {98, 151 + (15 * k)}, Size = {85, 15}, TextSize = 10, Color = {255, 255, 255}, IsReadOnly = true, HTextAlign = "Right", StrokeWidth = 0, } end -- 'Firmware' grouping -- Groupbox table.insert(graphics, { Type = "GroupBox", Text = "Firmware", StrokeWidth = 1, Position = {2, 259}, Size = {208, 60}, Fill = {255, 255, 255}, CornerRadius = 3, HTextAlign = "Center", }) for k, v in ipairs(fw_vendors) do -- Parameter label table.insert(graphics, { Type = "Label", Text = v, Position = {6, 265 + (15 * k)}, Size = {63, 15}, TextSize = 10, Color = {0, 0, 0}, HTextAlign = "Right", }) -- Value text layout["FirmwareInfo "..k] = { Style = "Text", PrettyName = "Device~Firmware~"..v, Position = {78, 265 + (15 * k)}, Size = {125, 15}, TextSize = 10, Color = {255, 255, 255}, IsReadOnly = true, HTextAlign = "Right", StrokeWidth = 0, } end -- 'Status' grouping -- Groupbox table.insert(graphics, { Type = "GroupBox", Text = "Status", StrokeWidth = 1, Position = {2, 329}, Size = {208, 100}, Fill = {255, 255, 255}, CornerRadius = 3, HTextAlign = "Center", }) for k, v in ipairs(status_types) do -- Parameter label table.insert(graphics, { Type = "Label", Text = v, Position = {6, 343 + (15 * k)}, Size = {63, 15}, TextSize = 10, Color = {0, 0, 0}, HTextAlign = "Right", }) -- LEDs layout["StatusLEDs "..k] = { Style = "LED", PrettyName = "Device~Status~"..v, Position = {80, 343 + (15 * k)}, Size = {16, 16}, CornerRadius = 8, Color = {99, 99, 99}, OffColor = {99, 99, 99}, UnlinkOffColor = true, } -- Values text layout["StatusInfo "..k] = { Style = "Text", PrettyName = "Device~Status~"..v, Position = {98, 343 + (15 * k)}, Size = {65, 15}, TextSize = 10, Color = {255, 255, 255}, IsReadOnly = true, HTextAlign = "Right", StrokeWidth = 0, } end -- Identify button layout["Identify"] = { PrettyName = "Device~Status~Identify", Style = "Button", ButtonStyle = "Momentary", Legend = "Identify", Position = {79, 394}, Color = {220, 220, 220}, OffColor = {220, 220, 220}, UnlinkOffColor = true, Size = {55, 25}, TextFontSize = 10, } -- 'Settings' grouping -- Groupbox table.insert(graphics, { Type = "GroupBox", Text = "Settings", StrokeWidth = 1, Position = {2, 438}, Size = {208, 60}, Fill = {255, 255, 255}, CornerRadius = 3, HTextAlign = "Center", }) -- 'LED Brightness' label table.insert(graphics, { Type = "Label", Text = "LED Brightness", Position = {6, 468}, Size = {80, 15}, TextSize = 10, Color = {0, 0, 0}, HTextAlign = "Right", }) -- 'LED Brightness' combobox layout["LED_Brightness"] = { Style = "ComboBox", PrettyName = "Device~Settings~LED Brightness", Position = {105, 466}, Size = {85, 20}, Fill = {255, 255, 255}, TextSize = 10, } -- 'Input/Transmit AES67 Details' grouping -- Groupbox table.insert(graphics, { Type = "GroupBox", Text = "Input/Transmit AES67 Details", StrokeWidth = 1, Position = {2, 508}, Size = {208, 80}, Fill = {255, 255, 255}, CornerRadius = 3, HTextAlign = "Center", }) -- Parameter labels for k, v in pairs(aes67_tx_labels) do table.insert(graphics, { Type = "Label", Text = v, Position = {30 + 80 * (k - 1), 523}, Size = {70, 15}, TextSize = 10, Color = {0, 0, 0}, HTextAlign = "Center", }) end -- AES67 TX details listbox layout["TXAES67Details"] = { Style = "ListBox", PrettyName = "", StrokeWidth = 0, Position = {5, 536}, Size = {201, 49}, Fill = {255, 255, 255}, TextSize = 10, } -- 'Receive/Output AES67 Details' grouping -- Groupbox table.insert(graphics, { Type = "GroupBox", Text = "Receive/Output AES67 Details", StrokeWidth = 1, Position = {2, 598}, Size = {208, 80}, Fill = {255, 255, 255}, CornerRadius = 3, HTextAlign = "Center", }) -- Parameter Labels for k, v in pairs(aes67_rx_labels) do table.insert(graphics, { Type = "Label", Text = v, Position = {70 * (k - 1), 613}, Size = {70, 15}, TextSize = 10, Color = {0, 0, 0}, HTextAlign = "Center", }) end -- AES67 RX details listbox layout["RXAES67Details"] = { Style = "ListBox", PrettyName = "", StrokeWidth = 0, Position = {5, 627}, Size = {201, 49}, Fill = {255, 255, 255}, TextSize = 10, } -- Channel Strip controls -- Lay out 2 8-Channel banks per page -- Determine which channels to display based on the model type and the currently selected page, e.g.: -- 8.8M -> Page 1: Bank 1 (Mic/Line 1-8), Bank 2 (Line 1-8) -- 16.0M -> Page 1: Bank 1 (Mic/Line 1-8), Bank 2 (Mic/Line 9-16) -- 0.32L -> Page 1: Bank 1 (Line 1-8), Bank 2 (Line 9-16) Page 2: Bank 1 (Line 17-24), Bank 2 (Line 25-32) for i = 1, 2 do local start_ch, end_ch = 0 local ctrl_prefix = nil local vert_offset = (i == 2) and 300 or 0 local current_page_idx = props["page_index"].Value if model_type == '8.8M' or model_type == '16.16M' or model_type == '32.32M' then start_ch = ((current_page_idx - 1) * 8) + 1 end_ch = start_ch + 7 ctrl_prefix = (i == 2) and 'RX' or 'TX' elseif model_type == '16.0M' or model_type == '0.16L' then start_ch = ((i - 1) * 8) + 1 end_ch = start_ch + 7 ctrl_prefix = (model_type == '16.0M') and 'TX' or 'RX' elseif model_type == '32.0M' or model_type == '0.32L' then start_ch = ((current_page_idx - 1) * 16) + 1 + ((i - 1) * 8) end_ch = start_ch + 7 ctrl_prefix = (model_type == '32.0M') and 'TX' or 'RX' end -- Labels -- Channel bank label table.insert(graphics, { Type = "GroupBox", Text = (ctrl_prefix == 'TX') and "Mic/Line In" or "Line Out", StrokeWidth = 1, Position = {220, 101 + vert_offset}, Size = {458, 252}, Fill = {255, 255, 255}, CornerRadius = 5, HTextAlign = "Left", }) -- Peak Meter label table.insert(graphics, { Type = "Label", Text = "Meter Peak", Position = {227, 143 + vert_offset}, Size = {83, 15}, Color = {0, 0, 0}, HTextAlign = "Right", }) -- Gain in dB on Input/Transmit side only if ctrl_prefix == 'TX' then -- TX Gain Label table.insert(graphics, { Type = "Label", Text = "Gain", Position = {227, 183 + vert_offset}, Size = {83, 15}, Color = {0, 0, 0}, HTextAlign = "Right", }) end -- Level Label table.insert(graphics, { Type = "Label", Text = "Level", Position = {227, 235 + vert_offset}, Size = {83, 15}, Color = {0, 0, 0}, HTextAlign = "Right", }) -- Phantom power only on Input/Transmit if ctrl_prefix == 'TX' then -- Phantom power label table.insert(graphics, { Type = "Label", Text = "Phantom Power", Position = {227, 260 + vert_offset}, Size = {83, 25}, Color = {0, 0, 0}, HTextAlign = "Right", }) end -- AES67 Status Label table.insert(graphics, { Type = "Label", Text = "AES67 Status", Position = {227, 295 + vert_offset}, Size = {83, 15}, Color = {0, 0, 0}, HTextAlign = "Right", }) -- AES67 Streaming Label table.insert(graphics, { Type = "Label", Text = "AES67", Position = {227, 315 + vert_offset}, Size = {83, 25}, Color = {0, 0, 0}, HTextAlign = "Right", }) -- iterate through the visible channels for the bank for idx = start_ch, end_ch do local horz_offset = 43 * ((idx - 1) % 8) -- Channel strip background table.insert(graphics, { Type = "GroupBox", Position = {320 + horz_offset, 118 + vert_offset}, Size = {40, 230}, Fill = {240, 240, 240}, Color = {101, 101, 81}, CornerRadius = 4, TextSize = 14, }) -- Channel name layout[ctrl_prefix.."ChannelName "..idx] = { Style = "Text", PrettyName = "", Position = {323 + horz_offset, 123 + vert_offset}, Size = {35, 13}, Color = {240, 240, 240}, IsReadOnly = true, WordWrap = true, TextSize = 10, HTextAlign = "Center", StrokeWidth = 0, } -- Channel peak meter (dbFS) text local dBFS_pn = '' if ctrl_prefix == 'TX' then dBFS_pn = "Input/Transmit~Peak Meter (dBFS)~"..idx else dBFS_pn = "Receive/Output~Peak Meter (dBFS)~"..idx end layout[ctrl_prefix.."ChanneldBFS "..idx] = { Style = "Text", PrettyName = dBFS_pn, Position = {321 + horz_offset, 143 + vert_offset}, Size = {38, 15}, Color = {224, 224, 224}, IsReadOnly = true, TextSize = 8, HTextAlign = "Center", CornerRadius = 2, StrokeWidth = 0, } -- Channel bar meter layout[ctrl_prefix.."ChannelMeter "..idx] = { Style = "Meter", MeterStyle = "Level", PrettyName = "Input/Transmit~Peak Meter (dBFS) "..idx, Position = {322 + horz_offset, 164 + vert_offset}, Size = {36, 5}, CornerRadius = 0, Color = {99, 99, 99}, OffColor = {99, 99, 99}, UnlinkOffColor = true, } -- Channel gain (dB) (only on Input/Transmit) if ctrl_prefix == 'TX' then layout[ctrl_prefix.."ChannelGain "..idx] = { Style = "Text", PrettyName = "Input/Transmit~Gain (dB) "..idx, Position = {321 + horz_offset, 183 + vert_offset}, Size = {38, 15}, Color = {224, 224, 224}, IsReadOnly = true, TextSize = 10, HTextAlign = "Center", CornerRadius = 1, StrokeWidth = 0, } end -- Channel Gain/Level Knob (Gain (dB) adjuster for Input/Transmit, Level (dBu) adjuster for Receive/Output) local gain_knob_pn = '' if ctrl_prefix == 'TX' then gain_knob_pn = "Input/Transmit~Gain/Level~Gain (dB)~ "..idx else gain_knob_pn = "Receive/Output~Level (dBu)~ "..idx end layout[ctrl_prefix.."ChannelLevelKnob "..idx] = { Style = "Knob", PrettyName = gain_knob_pn, Position = {325 + horz_offset, 203 + vert_offset}, Size = {30, 30}, Color = {190, 83, 28}, ShowTextBox = false, TextSize = 10, } -- Level (dBu) text local level_pn = '' if ctrl_prefix == 'TX' then level_pn = "Input/Transmit~Gain/Level~Level (dBu)~ "..idx else level_pn = "Receive/Output~Level (dBu)~ "..idx end layout[ctrl_prefix.."ChannelLevel "..idx] = { Style = "Text", PrettyName = level_pn, Position = {321 + horz_offset, 235 + vert_offset}, Size = {38, 15}, Color = {224, 224, 224}, IsReadOnly = true, TextSize = 10, HTextAlign = "Center", CornerRadius = 1, StrokeWidth = 0, } -- Phantom power button (Input/Transmit only) if ctrl_prefix == 'TX' then layout["TXChannelPhantom "..idx] = { Style = "Button", Legend = "48V", PrettyName = "Input/Transmit~Phantom Power~ "..idx, Position = {322 + horz_offset, 260 + vert_offset}, Color = {255, 0, 0}, OffColor = {220, 220, 220}, UnlinkOffColor = true, Size = {35, 25}, TextFontSize = 10, } end -- AES67 Status local aes67_pn = '' if ctrl_prefix == 'TX' then aes67_pn = "Input/Transmit~AES67 Status~ "..idx else aes67_pn = "Receive/Output~AES67 Status~ "..idx end layout[ctrl_prefix.."ChannelAES67Status "..idx] = { Style = "Text", PrettyName = aes67_pn, Position = {321 + horz_offset, 295 + vert_offset}, Size = {38, 15}, Color = {224, 224, 224}, IsReadOnly = true, TextSize = 10, HTextAlign = "Center", CornerRadius = 1, StrokeWidth = 0, } if ctrl_prefix == 'TX' then -- AES67 toggle button (Input/Transmit only) layout["TXChannelAES67Tgl "..idx] = { PrettyName = "Input/Transmit~AES67 State~ "..idx, Style = "Button", Legend = "Off", Position = {322 + horz_offset, 315 + vert_offset}, Color = {0, 0, 255}, OffColor = {220, 220, 220}, UnlinkOffColor = true, Size = {35, 25}, TextFontSize = 10, } else -- AES67 transmitter selection combobox (Receive/Output only) layout["SelectedAES67TX "..idx] = { PrettyName = "Receive/Output~Select AES67 Transmitter~ "..idx, Style = "ComboBox", Position = {321 + horz_offset, 320 + vert_offset}, Size = {38, 15}, Fill = {255, 255, 255}, TextSize = 10, } end end end return layout, graphics end -- Start of Lua runtime code if Controls then rapidjson = require("rapidjson") e = System.IsEmulating -- Device selection control aliases cb_Dev = Controls.SelectedDevice text_DevStatus = Controls.SelectedDeviceStatus text_DevStatus.String = '' -- Device control aliases text_DeviceInfo = Controls.DeviceInfo text_FirmwareInfo = Controls.FirmwareInfo text_StatusInfo = Controls.StatusInfo led_StatusLEDs = Controls.StatusLEDs btn_Identify = Controls.Identify cb_LED_Brightness = Controls.LED_Brightness -- Input / Transmit control aliases text_TXNameLabel = Controls.TXChannelName text_TXChanneldBFS = Controls.TXChanneldBFS meter_TXChannelMeter = Controls.TXChannelMeter knob_TXChannelGain = Controls.TXChannelLevelKnob text_TXChannelGain = Controls.TXChannelGain text_TXChannelLevel = Controls.TXChannelLevel btn_Phantom = Controls.TXChannelPhantom text_TXChannelAES67Status = Controls.TXChannelAES67Status btn_TXChannelAES67Tgl = Controls.TXChannelAES67Tgl -- Receive / Output control aliases text_RXNameLabel = Controls.RXChannelName knob_RXChannelLevel = Controls.RXChannelLevelKnob text_RXChannelLevel = Controls.RXChannelLevel text_RXChanneldBFS = Controls.RXChanneldBFS meter_RXChannelMeter = Controls.RXChannelMeter text_RXChannelAES67Status = Controls.RXChannelAES67Status cb_SelectedAES67TX = Controls.SelectedAES67TX -- Globals DiscoveredIPLookup = {} SelectedDevIP = '' WASPControls = {} QSYSCtrlIdxToWASPCtrl = {} DiscAES67TXFlows = {} QSysCtrlIdxToChIdx = {} DiscAES67TXChannels = {} AES67IPToSessionName = {} AES67SessionNameToIP = {} Timers = {} OutputNames = {} InputCtrls = { text_TXChanneldBFS, meter_TXChannelMeter, knob_TXChannelGain, text_TXChannelGain, text_TXChannelLevel, btn_Phantom, text_TXChannelAES67Status, btn_TXChannelAES67Tgl } OutputCtrls = { knob_RXChannelLevel, text_RXChannelLevel, text_RXChanneldBFS, meter_RXChannelMeter, text_RXChannelAES67Status, cb_SelectedAES67TX } -- Colors RED = '#ff0000' ORANGE = '#ff8c00' GRAY = '#e0e0e0' ALT_GRAY = '#dcdcdc' GREEN = '#00ff00' BLUE = '#0000ff' QSYS_GREEN = '#007c00' -- Enable/Disable all controls function enable_all_ctrls(do_enable) for k, v in ipairs(InputCtrls) do for key, val in ipairs(v) do val.IsDisabled = not do_enable end end for k, v in ipairs(OutputCtrls) do for key, val in ipairs(v) do val.IsDisabled = not do_enable end end btn_Identify.IsDisabled = not do_enable cb_LED_Brightness.IsDisabled = not do_enable end -- Helper functions function split(s, delimiter) result = {} for match in (s..delimiter):gmatch("(.-)"..delimiter) do table.insert(result, match); end return result; end function path_to_ch_idx(s) local path_split = split(s, '/') return path_split[#path_split] end function display_error(str) text_DevStatus.String = str text_DevStatus.Color = RED end -- Event Handlers -- Identify btn_Identify.EventHandler = function() btn_Identify.IsDisabled = true HttpClient.Upload { Url = "http://"..SelectedDevIP.."/asi/r1/status/identify", Method = "PUT", Timeout = 10, Headers = { ["Content-Type"] = "application/json", ["X-HTTP-Method-Override"] = "PATCH" }, Data = "", EventHandler = function(tbl, code, d, e) if code ~= 200 then display_error('Identify device request error') end btn_Identify.IsDisabled = false end } end -- Front panel brightness cb_LED_Brightness.Choices = { '100%', '75%', '50%', '25%' } cb_LED_Brightness.EventHandler = function(ctrl) local brightness_pct = tonumber(cb_LED_Brightness.String:sub(1, -2)) -- remove '%' HttpClient.Upload { Url = "http://"..SelectedDevIP.."/asi/r1/"..QSYSCtrlIdxToWASPCtrl[ctrl.Index]['_self'], Method = "PUT", Timeout = 10, Data = rapidjson.encode({brightness = brightness_pct}), Headers = { ["Content-Type"] = "application/json", ["X-HTTP-Method-Override"] = "PATCH" }, EventHandler = function(tbl, code, d, e) if code ~= 202 then display_error('PATCH LED Brightness error') end end } end -- Event handler for timers associated with gain/level knobs -- Gain/Level changes are submitted after 500ms of inactivity function OnLevelChangeTimeout(timer) for k, v in pairs(Timers) do if v == timer then -- stop the timer v:Stop() -- look up the control local level_dBu = nil local ch_idx = QSysCtrlIdxToChIdx[k] local ctrl = QSYSCtrlIdxToWASPCtrl[k] if string.find(ctrl['loc'][1], "io/analog/in") then level_dBu = 24 - knob_TXChannelGain[ch_idx].Value else level_dBu = knob_RXChannelLevel[ch_idx].Value end -- submit the level change HttpClient.Upload { Url = "http://"..SelectedDevIP.."/asi/r1"..ctrl['_self'], Method = "PUT", Timeout = 10, Data = rapidjson.encode({level = level_dBu}), Headers = { ["Content-Type"] = "application/json", ["X-HTTP-Method-Override"] = "PATCH" }, EventHandler = function(tbl, code, d, e) if code ~= 202 then display_error('PATCH level error') end end } end end end -- Gain/Level knobs for k, v in ipairs(knob_TXChannelGain) do -- Create timer Timers[v.Index] = Timer.New() local new_timer = Timers[v.Index] new_timer.EventHandler = OnLevelChangeTimeout v.EventHandler = function(ctrl) -- display Gain (dB) and Level (dBu) text_TXChannelGain[k].String = ctrl.String.." dB" local level_dBu = 24 - tonumber(ctrl.String) text_TXChannelLevel[k].String = tostring(level_dBu).." dBu" -- disable phantom power btn if level > 10dBu (14dB) btn_Phantom[k].IsDisabled = (ctrl.Value < 14) new_timer:Stop() new_timer:Start(0.5) -- 500ms timeout end end for k, v in ipairs(knob_RXChannelLevel) do -- Create timer Timers[v.Index] = Timer.New() local new_timer = Timers[v.Index] new_timer.EventHandler = OnLevelChangeTimeout v.EventHandler = function(ctrl) -- display Level (dBu) text_RXChannelLevel[k].String = ctrl.String.." dBu" new_timer:Stop() new_timer:Start(0.5) -- 500ms timeout end end -- Phantom power for k,v in ipairs(btn_Phantom) do v.EventHandler = function(ctrl) -- get the Phantom ctrl channel index, WASP control index, state local ch_idx = QSysCtrlIdxToChIdx[ctrl.Index] local ctrl = QSYSCtrlIdxToWASPCtrl[ctrl.Index] local state = btn_Phantom[ch_idx].Boolean -- disable the button while submitting the request btn_Phantom[ch_idx].IsDisabled = true -- submit HttpClient.Upload { Url = "http://"..SelectedDevIP.."/asi/r1"..ctrl['_self'], Method = "PUT", Timeout = 10, Data = rapidjson.encode({active = state}), Headers = { ["Content-Type"] = "application/json", ["X-HTTP-Method-Override"] = "PATCH" }, EventHandler = function(tbl, code, d, e) btn_Phantom[ch_idx].IsDisabled = false if code ~= 202 then display_error('PATCH phantom power error') end end } end end -- Input/Transmit AES67 toggle buttons for k,v in ipairs(btn_TXChannelAES67Tgl) do v.EventHandler = function(ctrl) local ch_idx = k - 1 local new_state = v.Boolean btn_TXChannelAES67Tgl[k].Legend = "..." HttpClient.Upload { Url = "http://"..SelectedDevIP.."/asi/r1/aes67/tx/toggle", Method = "PUT", Timeout = 10, Data = rapidjson.encode({index = ch_idx, enable = new_state}), Headers = { ["Content-Type"] = "application/json", ["X-HTTP-Method-Override"] = "PATCH" }, EventHandler = function(tbl, code, d, e) if code ~= 200 then display_error('PATCH TX AES67 status error') end end } end end -- Event handler for Receive/Output AES67 transmitter selection combobox for k,v in ipairs(cb_SelectedAES67TX) do v.String = '...' v.EventHandler = function(ctrl) local rx_ch_idx = k - 1 local transport_ip = '0.0.0.0' local tx_flow_idx = 0 if v.String ~= 'None' then local tx_flow = split(v.String, ':') transport_ip = AES67SessionNameToIP[tx_flow[1]] tx_flow_idx = tonumber(tx_flow[2]) - 1 end v.String = '...' HttpClient.Upload { Url = "http://"..SelectedDevIP.."/asi/r1/aes67/rx/subscribe", Method = "PUT", Timeout = 10, Data = rapidjson.encode({rx_ch_idx = rx_ch_idx, tx_flow_ip = transport_ip, tx_flow_idx = tx_flow_idx}), Headers = { ["Content-Type"] = "application/json", ["X-HTTP-Method-Override"] = "PATCH" }, EventHandler = function(tbl, code, d, e) if code ~= 200 then display_error('PATCH RX AES67 selected transmitter error') end end } end end -- Control update functions function update_meter(ctrl, peak_dBFS) local ch_idx = path_to_ch_idx(ctrl['loc'][1]) + 1 local dbFS_ctrl, meter_ctrl = nil local is_input = string.find(ctrl['loc'][1], 'io/analog/in') if is_input then dbFS_ctrl = text_TXChanneldBFS[ch_idx] meter_ctrl = meter_TXChannelMeter[ch_idx] else dbFS_ctrl = text_RXChanneldBFS[ch_idx] meter_ctrl = meter_RXChannelMeter[ch_idx] end -- dBFS control text and background local dBFS_text = '-'..tostring(peak_dBFS)..'dBFS' local idx = ctrl['_self'] if peak_dBFS == 0 then dBFS_text = 'Clip' dbFS_ctrl.Color = RED -- hold clip for 1s Timers[idx] = Timer.New() Timers[idx].EventHandler = function(timer) if timer == Timers[idx] then dbFS_ctrl.Color = GRAY end end Timers[idx]:Start(1) elseif peak_dBFS > 60 then dBFS_text = 'Low' end dbFS_ctrl.String = dBFS_text meter_ctrl.Value = -peak_dBFS end function update_input_level(ch_idx, level_dBu) --update knob knob_TXChannelGain[ch_idx].Value = (24 - level_dBu) --dBu -> dB -- display Gain (dB) and Level (dBu) text_TXChannelLevel[ch_idx].String = tostring(level_dBu).." dBu" text_TXChannelGain[ch_idx].String = tostring(24 - level_dBu).." dB" -- disable phantom power btn if level > 10dBu btn_Phantom[ch_idx].IsDisabled = (level_dBu > 10) end function update_output_level(ch_idx, level_dBu) -- update knob knob_RXChannelLevel[ch_idx].Value = level_dBu -- display Level (dBu) text_RXChannelLevel[ch_idx].String = tostring(level_dBu).." dBu" end function update_tx_aes67_ctrls(ch_idx, enabled) btn_TXChannelAES67Tgl[ch_idx].Color = enabled and BLUE or ALT_GRAY btn_TXChannelAES67Tgl[ch_idx].Legend = enabled and 'On' or 'Off' btn_TXChannelAES67Tgl[ch_idx].Boolean = enabled text_TXChannelAES67Status[ch_idx].String = enabled and 'Active' or 'Inactive' text_TXChannelAES67Status[ch_idx].Color = enabled and BLUE or ALT_GRAY end function update_rx_aes67_ctrls(ch_idx, enabled) text_RXChannelAES67Status[ch_idx].String = enabled and 'Active' or 'Inactive' text_RXChannelAES67Status[ch_idx].Color = enabled and BLUE or ALT_GRAY if not enabled then cb_SelectedAES67TX[ch_idx].String = 'None' end end -- Polled functions -- Device status function poll_sync_sys() local reconnected_needed = false HttpClient.Download { Url = "http://"..SelectedDevIP.."/asi/r1/status/sync_sys", Headers = { ["Content-Type"] = "application/json" }, Timeout = 10, -- mark the device as disconnected if this request fails EventHandler = function(tbl, code, data, err, headers) if code ~= 200 then reconnect_needed = true display_error('Error - Trying to Reconnect...') enable_all_ctrls(false) do return end end -- request succeeded, update the status fields and reconnect to the update stream, if necessary text_DevStatus.String = 'OK' text_DevStatus.Color = QSYS_GREEN if reconnect_needed then sock:Disconnect() sock:Connect(SelectedDevIP, 80) reconnect_needed = false enable_all_ctrls(true) end -- update status parameters and LEDs local obj = rapidjson.decode(rapidjson.decode(data)) for k, v in pairs(obj) do local color, text = nil if k == 'sync' then if v == 0 then color = GREEN text = 'PTP Slave' elseif v == 1 then color = ORANGE text = 'PTP Syncing' elseif v == 2 then color = BLUE text = 'PTP Master' elseif v == 3 then color = RED text = 'PTP Error' else color = ALT_GRAY text = 'PTP State Unknown' end led_StatusLEDs[1].Color = color text_StatusInfo[1].String = text elseif k == 'sys' then if v == 0 then color = GREEN text = 'OK' elseif v == 1 then color = ORANGE text = 'Waiting' elseif v == 2 then color = BLUE text = 'Hardware Error' else color = ALT_GRAY text = 'System State Unknown' end led_StatusLEDs[2].Color = color text_StatusInfo[2].String = text end end end } end -- Input/Transmit channels AES67 status function poll_tx_aes67_status() HttpClient.Download { Url = "http://"..SelectedDevIP.."/asi/r1/aes67/tx/status", Headers = { ["Content-Type"] = "application/json" }, Timeout = 10, EventHandler = function(tbl, code, data, err, headers) if code ~= 200 then display_error('GET TX AES67 status request error') do return end end local choices = {''} local session_names = {'Mic/Line In 1-8', 'Mic/Line In 9-16', 'Mic/Line In 17-24', 'Mic/Line In 25-32'} input_count, output_count = GetIOCount(Properties.ModelType.Value) local obj = rapidjson.decode(data) for k, v in pairs(obj) do if k <= input_count then local active = false local tmp = rapidjson.decode(v) local transport_ip = tmp['transport_ip'] if transport_ip ~= '0.0.0.0' then active = true local session_name = session_names[math.floor((k - 1) / 8) + 1] local new_selection = string.format("%20.20s", session_name)..string.format("%20.20s", transport_ip) local found = false for k, v in pairs(choices) do if v == new_selection then found = true end end if not found then table.insert(choices, new_selection) end end update_tx_aes67_ctrls(k, active) end end Controls.TXAES67Details.Choices = choices end } end -- Advertised AES67 transmitters function poll_aes67_disc_tx_flows() HttpClient.Download { Url = "http://"..SelectedDevIP.."/asi/r1/aes67/tx/adv", Headers = { ["Content-Type"] = "application/json" }, Timeout = 10, EventHandler = function(tbl, code, data, err, headers) if code ~= 200 then display_error('GET AES67 advertised flows request error') do return end end DiscAES67TXFlows = rapidjson.decode(data) DiscAES67TXChannels = {'None'} for k, v in pairs(DiscAES67TXFlows) do AES67SessionNameToIP[k] = v['transport_ip_str'] AES67IPToSessionName[v['transport_ip_str']] = k for i = 1, v['channel_count'] do table.insert(DiscAES67TXChannels, k..':'..tostring(i)) end end for k, v in pairs(cb_SelectedAES67TX) do cb_SelectedAES67TX[tonumber(k)].Choices = DiscAES67TXChannels end end } end -- Receive/Output channels AES67 status function poll_rx_aes67_status() HttpClient.Download { Url = "http://"..SelectedDevIP.."/asi/r1/aes67/rx/status", Headers = { ["Content-Type"] = "application/json" }, Timeout = 10, EventHandler = function(tbl, code, data, err, headers) if code ~= 200 then display_error('GET RX AES67 status request error') do return end end local choices = {''} local obj = rapidjson.decode(data) input_count, output_count = GetIOCount(Properties.ModelType.Value) for k, v in pairs(obj) do if k <= output_count then local active = false local aes67_rx_ch = rapidjson.decode(v) local sub_transport_ip = aes67_rx_ch['sub_transport_ip'] local sub_ch_idx = aes67_rx_ch['sub_ch_idx'] if sub_transport_ip ~= '0.0.0.0' then active = true for key, value in pairs(DiscAES67TXChannels) do local session_name = AES67IPToSessionName[sub_transport_ip] if session_name ~= nil then local found = false local new_option = string.format("%9.9s", OutputNames[tonumber(k) - 1])..string.format("%20.20s", session_name)..string.format("%15.15s", tostring(sub_ch_idx + 1)) for k, v in pairs(choices) do if v == new_option then found = true end end if not found then table.insert(choices, new_option) end local option = session_name..":"..tostring(sub_ch_idx + 1) if value == option then local session = split(value, ':') local session_substr = string.sub(session[1], 0, 3) cb_SelectedAES67TX[k].String = session_substr.."...:"..session[2] end end end end update_rx_aes67_ctrls(k, active) end end Controls.RXAES67Details.Choices = choices end } end -- Functions/event handlers called/triggered on device selection -- update stream (meters and unsolicited DOM changes) sock = TcpSocket.New() function update_stream() enable_all_ctrls(true) -- enable all controls sendData = 'GET /asi/u1/controls HTTP/1.1\r\nHost: '..SelectedDevIP..'\r\n\r\n' sock.Connected = function(sock) sock:Write(sendData) end sock.Data = function(sock) --read from update stream local data = sock:Read(sock.BufferLength) data = split(data, '\r\n---\n')[1] if string.sub(data, 1, 1) == '{' then -- JSON -> Lua local obj = rapidjson.decode(data) -- meters if obj['_type'] == '/meta/update/group_prefix' then for wasp_ctrl_idx, peak in pairs(obj['body']) do for k, v in pairs(WASPControls) do if v['_self'] == "/controls/"..wasp_ctrl_idx then update_meter(v, -peak) end end end -- input / output level, phantom power elseif obj['_type'] == '/meta/update/control' then for k, v in pairs(WASPControls) do if v['_self'] == obj['path'] then if v['loc'][1] ~= '/adapter' then local ch_idx = tonumber(path_to_ch_idx(v['loc'][1])) + 1 -- look up the control type if v['_type'] == '/meta/controls/input_trim' then for k, level_dBu in pairs(obj['body']) do update_input_level(ch_idx, level_dBu) end elseif v['_type'] == '/meta/controls/output_trim' then for k, level_dBu in pairs(obj['body']) do update_output_level(ch_idx, level_dBu) end elseif v['_type'] == '/meta/controls/phantom' then for param, state in pairs(obj['body']) do btn_Phantom[ch_idx].Boolean = state end end end end end end end end sock.Closed = function(sock) sock:Connect(SelectedDevIP, 80) -- WASP 1.0 socket closes after 60s, reconnect when this occurs end sock.Error = function(sock, err) display_error("TCP socket had an error:"..err) end sock.Timeout = function(sock, err) display_error("TCP socket timed out"..err) end sock:Connect(SelectedDevIP, 80) end -- All WASP controls enumerated function on_get_controls_done(tbl, code, data, err, headers) if code ~= 200 then display_error('GET controls request error') do return end end local dom = rapidjson.decode(data) WASPControls = dom.controls for k, v in pairs(WASPControls) do -- Initialize and Store controls, Q-SYS control Index -> WASP control Index -- Q-SYS control Index -> channel index (if applicable) local qsys_idx = nil local ch_idx = tonumber(path_to_ch_idx(v['loc'][1])) if v['_type'] == '/meta/controls/input_trim' then qsys_idx = knob_TXChannelGain[ch_idx + 1].Index update_input_level(ch_idx + 1, v['level']) elseif v['_type'] == '/meta/controls/output_trim' then qsys_idx = knob_RXChannelLevel[ch_idx + 1].Index update_output_level(ch_idx + 1, v['level']) elseif v['_type'] == '/meta/controls/phantom' then qsys_idx = btn_Phantom[ch_idx + 1].Index btn_Phantom[ch_idx + 1].Boolean = v['active'] elseif v['_type'] == '/meta/controls/front_panel' then local brightness_pct = v['brightness'] cb_LED_Brightness.String = tostring(brightness_pct)..'%' qsys_idx = cb_LED_Brightness.Index end if qsys_idx ~= nil then QSYSCtrlIdxToWASPCtrl[qsys_idx] = v if ch_idx ~= nil then QSysCtrlIdxToChIdx[qsys_idx] = ch_idx + 1 end end end -- Iterate through I/O controls local io = dom.io.analog for k, v in pairs(io) do if k == 'out' then for idx, obj in pairs(v) do OutputNames[obj['idx']] = obj['label'] text_RXNameLabel[idx].String = tostring(idx) end else for idx, obj in pairs(v) do text_TXNameLabel[idx].String = tostring(idx) end end end -- Polling timer local poll_timer = Timer.New() poll_timer.EventHandler = function(timer) if timer == poll_timer then poll_tx_aes67_status() poll_aes67_disc_tx_flows() poll_rx_aes67_status() poll_sync_sys() end end poll_timer:Start(10) -- 10s -- initiate the control update / meters stream update_stream() end -- WASP adapter info enumerated function on_get_adapter_done(tbl, code, data, err, headers) if code ~= 200 then display_error('GET adapter request error') do return end end obj = rapidjson.decode(data) -- Device Info local dev_info = { 'product_name', 'model_id', 'hw_rev', 'serial' } for k, v in ipairs(dev_info) do text_DeviceInfo[k].String = obj.info[v] end -- Firmware local asi_fw = obj.software['AudioScience']['firmware'] asi_fw = split(asi_fw, '+')[1] local xmos_fw = obj.software['XMOS']['firmware'] xmos_fw = split(xmos_fw, '+')[1] text_FirmwareInfo[1].String = asi_fw text_FirmwareInfo[2].String = xmos_fw -- Get MAC Address HttpClient.Download { Url = "http://"..SelectedDevIP.."/asi/r1/status/mac", Headers = { ["Content-Type"] = "application/json" }, Timeout = 10, EventHandler = function(tbl, code, data, err, headers) if code ~= 200 then display_error('GET MAC Address request error') do return end end text_DeviceInfo[5].String = rapidjson.decode(data) end } -- Get all Controls HttpClient.Download { Url = "http://"..SelectedDevIP.."/asi/r1", Headers = { ["Content-Type"] = "application/json" }, Timeout = 10, EventHandler = on_get_controls_done } end -- Create Device Selection List function CreateDeviceList() deviceList = {} for k, v in pairs(DiscoveredIPLookup) do if k ~= nil and v ~= nil then table.insert(deviceList, k) end end cb_Dev.Choices = deviceList cb_Dev.EventHandler = function() text_DevStatus.String = 'Connecting...' text_DevStatus.Color = BLUE SelectedDevIP = DiscoveredIPLookup[cb_Dev.String] HttpClient.Download { Url = "http://"..SelectedDevIP.."/asi/r1/adapter", Headers = { ["Content-Type"] = "application/json" }, Timeout = 10, EventHandler = on_get_adapter_done } end end -- Bonjour Bits on Core local ResolveBonjourName local function SetRBN() local sixoh = Qlib.gethostbyname~=nil local RBN = { "140000003500000007001117000000c6014000c741c003e48180000682400007c24004248280004102010080020000c142010000030004418301005d428304840200016c030000ca4183836c4300008c03c20300040003a443800180038006c0038004a44300012600800009000000040654696d657204044e657704036f730408746d706e616d6504146e6f68757020646e732d7364202d47207634200404203e2004032026040d4576656e7448616e646c65720406537461727401000000000002000000001a0000002c0000000000085400000006404000078040004500800081c000002480800108000080060040000cc041008100020024008101088000830840808208000082060040000c404200850080002440800106004100224000001e4001800680420041c00200864041005d808000868041002440800106004300074043004500800024400001060043000780430041c0030024000101c6404400c780c4010601410041c10400e480800108c00088220000001e000980c6004100dc0080015f00c5011e000880c5000001cc40c501e4400001c6004400c6c080015f80c5011e000180c5008002c6c00002060144005f0081011e000680c60044000bc10000450180020a41818b450100034c81c402c1410600648180010a41018c450180030a41018dc8008101c50080020601440008018101c50000045f80c5011e800180c5000004e44080001ec00080c680420001c1060045010003e4408001260080001c000000040566696c650403696f04056f70656e040272040472657304046d73670405636f646504057265616404032a610406636c6f736504067072696e7404084572726f723a2004036f73040772656d6f7665040865786563757465041d706772657020646e732d7364207c207861726773206b696c6c202d39040c63616e64696461746549500407737472696e6704066d6174636814365b3031325d3f25643f25642b252e5b3031325d3f25643f25642b252e5b3031325d3f25643f25642b252e5b3031325d3f25643f2564130000000000000000040553746f700004054e616d65040844656661756c74040b282e2b292e6c6f63616c04064d6f64656c04114950206e6f74207265736f6c766564210900000000000108010701030104010101000102010500000000540000001b0000001b0000001b0000001b0000001b0000001b0000001c0000001c0000001c0000001c0000001c0000001c0000001c0000001d0000001d0000001d0000001d0000001e0000001e0000001e0000001e0000001e0000001e0000001e0000001e0000001e0000001f0000001f0000001f0000001f000000200000002000000020000000200000002100000021000000210000002100000021000000210000002200000022000000220000002200000022000000220000002300000023000000230000002400000024000000240000002400000024000000240000002400000024000000240000002500000025000000250000002500000025000000250000002500000025000000250000002500000025000000250000002600000026000000260000002700000027000000270000002700000027000000280000002a0000002a0000002a0000002a0000002c000000030000000873756363657373220000005400000005657865632200000054000000077369676e616c220000005400000009000000055f454e5604746d700a6b696c6c74696d65720a446973635461626c650a49504c755461626c6509667269656e646c79056e616d65066d6f64656c106d616b65646973706c61794c697374002e0000003200000003000c12000000c100000006414000078140024001000081c10000c50180005dc1810224010101c6014100000200024642410081820100c00280029dc20205c002000364028001e44100002600800007000000040104036f730408657865637574650404203e2004076173736572740409746f737472696e67041145786563757465204661696c65643a20020000000000010800000000120000002f00000030000000300000003000000030000000300000003000000030000000310000003100000031000000310000003100000031000000310000003100000031000000320000000700000008636d646c696e650000000012000000087061747465726e0000000012000000086361707475726500000000120000000872737472696e6701000000120000000873756363657373080000001200000005657865630800000012000000077369676e616c080000001200000002000000055f454e5604746d7017000000150000001500000015000000160000001600000016000000170000001700000017000000170000001700000017000000180000002c0000002c00000032000000330000003300000033000000340000003400000034000000350000000e000000056e616d65000000001700000009667269656e646c790000000017000000066d6f64656c00000000170000000a446973635461626c6500000000170000000a49504c755461626c650000000017000000106d616b65646973706c61794c6973740000000017000000096b696c6c74696d6500000000170000000a6b696c6c74696d6572030000001700000004746d70060000001700000008636d646c696e650c00000017000000087061747465726e0d0000001700000008636170747572650d000000170000000872737472696e670d0000001700000004436d64100000001700000001000000055f454e56", "400000004200000001000308000000460040004740c00080000000648000014780c00047c0c0006600000126008000040000000405516c6962040e676574686f737462796e616d65040a6164647265737365731301000000000000000100000000000000000008000000410000004100000041000000410000004100000041000000410000004200000001000000056e616d65000000000800000001000000055f454e56"} local tmpFn = load(bitstring.fromhexstream("1b4c7561530019930d0a1a0a04080408087856000000000000000000000028774001023d"..RBN[sixoh and 2 or 1])) if sixoh then ResolveBonjourName = function(name, friendlyName) DiscoveredIPLookup[friendlyName] = tmpFn(name) CreateDeviceList() end else ResolveBonjourName = tmpFn end end if not e then SetRBN() end -- Delete Bonjour Entry function DeleteBonjourEntry(name) DiscoveredIPLookup[name] = nil if cb_Dev.String == name then cb_Dev.String = "None" end CreateDeviceList() end -- Bonjour Browse function mDNSBrowse(selected_model_type) if not e then mDNS = BonjourBrowser.New() mDNS.EventHandler = function( record ) if record.Type == "_netaudio-cmc._udp." then if record.Added then --resolve record local r = BonjourResolver.New() r.EventHandler = function( rc ) local mf = rc.Values.mf if mf == "AudioSci" then local discovered_model = string.sub(rc.Values.model, -4) local selected_model = ModelTypeToNo(selected_model_type) if tonumber(discovered_model) == selected_model then ResolveBonjourName(rc.Target, record.Name) end end end r:Resolve( record ) else --remove record DeleteBonjourEntry(record.Name) end end end mDNS:Start( "_netaudio-cmc._udp", 0, "local" ) end end if not e then cb_Dev.String = "None" enable_all_ctrls(false) mDNSBrowse(Properties.ModelType.Value) end end