mirror of
https://github.com/onyx-and-iris/voicemeeter-api-python.git
synced 2025-04-03 20:33:44 +01:00
Compare commits
9 Commits
5b99f8aae3
...
3036cdff2f
Author | SHA1 | Date | |
---|---|---|---|
3036cdff2f | |||
b02f3af665 | |||
145f85b7cd | |||
71f77b7830 | |||
4415851816 | |||
8b63cbfe8d | |||
de4ce850eb | |||
ee3fa0a372 | |||
f92bb1e457 |
13
CHANGELOG.md
13
CHANGELOG.md
@ -11,6 +11,19 @@ Before any major/minor/patch bump all unit tests will be run to verify they pass
|
|||||||
|
|
||||||
- [x]
|
- [x]
|
||||||
|
|
||||||
|
## [2.1.0] - 2023-07-01
|
||||||
|
|
||||||
|
### Added
|
||||||
|
|
||||||
|
- RecorderMode added to Recorder class. See Recorder section in README for new properties and methods.
|
||||||
|
- recorder.loop is now a forwarder method for recorder.mode.loop for backwards compatibility
|
||||||
|
|
||||||
|
- RecorderArmStrip, RecorderArmBus mixed into Recorder class.
|
||||||
|
|
||||||
|
### Removed
|
||||||
|
|
||||||
|
- Recorder.loop removed from documentation
|
||||||
|
|
||||||
## [2.0.0] - 2023-06-25
|
## [2.0.0] - 2023-06-25
|
||||||
|
|
||||||
Where possible I've attempted to make the changes backwards compatible. The breaking changes affect two higher classes, Strip and Bus, as well as the behaviour of events. All other changes are additive or QOL aimed at giving more options to the developer. For example, every low-level CAPI call is now logged and error raised on Exception, you can now register callback functions as well as observer classes, extra examples to demonstrate different use cases etc.
|
Where possible I've attempted to make the changes backwards compatible. The breaking changes affect two higher classes, Strip and Bus, as well as the behaviour of events. All other changes are additive or QOL aimed at giving more options to the developer. For example, every low-level CAPI call is now logged and error raised on Exception, you can now register callback functions as well as observer classes, extra examples to demonstrate different use cases etc.
|
||||||
|
49
README.md
49
README.md
@ -398,13 +398,19 @@ The following methods are available
|
|||||||
- `record()`
|
- `record()`
|
||||||
- `ff()`
|
- `ff()`
|
||||||
- `rew()`
|
- `rew()`
|
||||||
- `load(<filepath>)`: string
|
- `load(filepath)`: raw string
|
||||||
|
- `goto(time_string)`: time string in format `hh:mm:ss`
|
||||||
|
- `filetype(filetype)`: string, ("wav", "aiff", "bwf", "mp3")
|
||||||
|
|
||||||
The following properties are available
|
The following properties are available
|
||||||
|
|
||||||
- `loop`: boolean
|
|
||||||
- `A1 - A5`: boolean
|
- `A1 - A5`: boolean
|
||||||
- `B1 - A3`: boolean
|
- `B1 - A3`: boolean
|
||||||
|
- `samplerate`: int, (22050, 24000, 32000, 44100, 48000, 88200, 96000, 176400, 192000)
|
||||||
|
- `bitresolution`: int, (8, 16, 24, 32)
|
||||||
|
- `channel`: int, from 1 to 8
|
||||||
|
- `kbps`: int, (32, 40, 48, 56, 64, 80, 96, 112, 128, 160, 192, 224, 256, 320)
|
||||||
|
- `gain`: float, from -60.0 to 12.0
|
||||||
|
|
||||||
example:
|
example:
|
||||||
|
|
||||||
@ -412,17 +418,46 @@ example:
|
|||||||
vm.recorder.play()
|
vm.recorder.play()
|
||||||
vm.recorder.stop()
|
vm.recorder.stop()
|
||||||
|
|
||||||
# Enable loop play
|
|
||||||
vm.recorder.loop = True
|
|
||||||
|
|
||||||
# Disable recorder out channel B2
|
# Disable recorder out channel B2
|
||||||
vm.recorder.B2 = False
|
vm.recorder.B2 = False
|
||||||
|
|
||||||
# filepath as raw string
|
# filepath as raw string
|
||||||
vm.recorder.load(r'C:\music\mytune.mp3')
|
vm.recorder.load(r'C:\music\mytune.mp3')
|
||||||
|
|
||||||
|
# set the goto time to 1m 30s
|
||||||
|
vm.recorder.goto("00:01:30")
|
||||||
```
|
```
|
||||||
|
|
||||||
Recorder properties are defined as write only.
|
#### Recorder.Mode
|
||||||
|
|
||||||
|
The following properties are available
|
||||||
|
|
||||||
|
- `recbus`: boolean
|
||||||
|
- `playonload`: boolean
|
||||||
|
- `loop`: boolean
|
||||||
|
- `multitrack`: boolean
|
||||||
|
|
||||||
|
example:
|
||||||
|
|
||||||
|
```python
|
||||||
|
# Enable loop play
|
||||||
|
vm.recorder.mode.loop = True
|
||||||
|
```
|
||||||
|
|
||||||
|
#### Recorder.ArmStrip[i]|ArmBus[i]
|
||||||
|
|
||||||
|
The following method is available
|
||||||
|
|
||||||
|
- `set(val)`: boolean
|
||||||
|
|
||||||
|
example:
|
||||||
|
|
||||||
|
```python
|
||||||
|
# Arm strip 3
|
||||||
|
vm.recorder.armstrip[3].set(True)
|
||||||
|
# Arm bus 0
|
||||||
|
vm.recorder.armbus[0].set(True)
|
||||||
|
```
|
||||||
|
|
||||||
### VBAN
|
### VBAN
|
||||||
|
|
||||||
@ -664,7 +699,7 @@ vm.vban.outstream[0].apply(on: True, name: 'streamname', bit: 24)
|
|||||||
|
|
||||||
## Config Files
|
## Config Files
|
||||||
|
|
||||||
`vm.apply_config(<configname>)`
|
`vm.apply_config(configname)`
|
||||||
|
|
||||||
You may load config files in TOML format.
|
You may load config files in TOML format.
|
||||||
Three example configs have been included with the package. Remember to save
|
Three example configs have been included with the package. Remember to save
|
||||||
|
@ -6,36 +6,50 @@ import voicemeeterlib
|
|||||||
from voicemeeterlib.kinds import KindId
|
from voicemeeterlib.kinds import KindId
|
||||||
from voicemeeterlib.kinds import request_kind_map as kindmap
|
from voicemeeterlib.kinds import request_kind_map as kindmap
|
||||||
|
|
||||||
# let's keep things random
|
|
||||||
KIND_ID = random.choice(tuple(kind_id.name.lower() for kind_id in KindId))
|
|
||||||
vm = voicemeeterlib.api(KIND_ID)
|
|
||||||
kind = kindmap(KIND_ID)
|
|
||||||
|
|
||||||
|
|
||||||
@dataclass
|
@dataclass
|
||||||
class Data:
|
class Data:
|
||||||
"""bounds data to map tests to a kind"""
|
"""bounds data to map tests to a kind"""
|
||||||
|
|
||||||
name: str = kind.name
|
name: str
|
||||||
phys_in: int = kind.ins[0] - 1
|
phys_in: int
|
||||||
virt_in: int = kind.ins[0] + kind.ins[-1] - 1
|
virt_in: int
|
||||||
phys_out: int = kind.outs[0] - 1
|
phys_out: int
|
||||||
virt_out: int = kind.outs[0] + kind.outs[-1] - 1
|
virt_out: int
|
||||||
vban_in: int = kind.vban[0] - 1
|
vban_in: int
|
||||||
vban_out: int = kind.vban[-1] - 1
|
vban_out: int
|
||||||
button_lower: int = 0
|
button_lower: int
|
||||||
button_upper: int = 79
|
button_upper: int
|
||||||
asio_in: int = kind.asio[0] - 1
|
asio_in: int
|
||||||
asio_out: int = kind.asio[-1] - 1
|
asio_out: int
|
||||||
insert_lower: int = 0
|
insert_lower: int
|
||||||
insert_higher: int = kind.insert - 1
|
insert_higher: int
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def channels(self):
|
def channels(self):
|
||||||
return (2 * self.phys_in) + (8 * self.virt_in)
|
return (2 * self.phys_in) + (8 * self.virt_in)
|
||||||
|
|
||||||
|
|
||||||
data = Data()
|
# let's keep things random
|
||||||
|
KIND_ID = random.choice(tuple(kind_id.name.lower() for kind_id in KindId))
|
||||||
|
vm = voicemeeterlib.api(KIND_ID)
|
||||||
|
kind = kindmap(KIND_ID)
|
||||||
|
|
||||||
|
data = Data(
|
||||||
|
name=kind.name,
|
||||||
|
phys_in=kind.ins[0] - 1,
|
||||||
|
virt_in=kind.ins[0] + kind.ins[-1] - 1,
|
||||||
|
phys_out=kind.outs[0] - 1,
|
||||||
|
virt_out=kind.outs[0] + kind.outs[-1] - 1,
|
||||||
|
vban_in=kind.vban[0] - 1,
|
||||||
|
vban_out=kind.vban[-1] - 1,
|
||||||
|
button_lower=0,
|
||||||
|
button_upper=79,
|
||||||
|
asio_in=kind.asio[0] - 1,
|
||||||
|
asio_out=kind.asio[-1] - 1,
|
||||||
|
insert_lower=0,
|
||||||
|
insert_higher=kind.insert - 1,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
def setup_module():
|
def setup_module():
|
||||||
|
@ -1 +1 @@
|
|||||||
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="68" height="20" role="img" aria-label="tests: 139"><title>tests: 139</title><linearGradient id="s" x2="0" y2="100%"><stop offset="0" stop-color="#bbb" stop-opacity=".1"/><stop offset="1" stop-opacity=".1"/></linearGradient><clipPath id="r"><rect width="68" height="20" rx="3" fill="#fff"/></clipPath><g clip-path="url(#r)"><rect width="37" height="20" fill="#555"/><rect x="37" width="31" height="20" fill="#4c1"/><rect width="68" height="20" fill="url(#s)"/></g><g fill="#fff" text-anchor="middle" font-family="Verdana,Geneva,DejaVu Sans,sans-serif" text-rendering="geometricPrecision" font-size="110"><text aria-hidden="true" x="195" y="150" fill="#010101" fill-opacity=".3" transform="scale(.1)" textLength="270">tests</text><text x="195" y="140" transform="scale(.1)" fill="#fff" textLength="270">tests</text><text aria-hidden="true" x="515" y="150" fill="#010101" fill-opacity=".3" transform="scale(.1)" textLength="210">139</text><text x="515" y="140" transform="scale(.1)" fill="#fff" textLength="210">139</text></g></svg>
|
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="68" height="20" role="img" aria-label="tests: 155"><title>tests: 155</title><linearGradient id="s" x2="0" y2="100%"><stop offset="0" stop-color="#bbb" stop-opacity=".1"/><stop offset="1" stop-opacity=".1"/></linearGradient><clipPath id="r"><rect width="68" height="20" rx="3" fill="#fff"/></clipPath><g clip-path="url(#r)"><rect width="37" height="20" fill="#555"/><rect x="37" width="31" height="20" fill="#4c1"/><rect width="68" height="20" fill="url(#s)"/></g><g fill="#fff" text-anchor="middle" font-family="Verdana,Geneva,DejaVu Sans,sans-serif" text-rendering="geometricPrecision" font-size="110"><text aria-hidden="true" x="195" y="150" fill="#010101" fill-opacity=".3" transform="scale(.1)" textLength="270">tests</text><text x="195" y="140" transform="scale(.1)" fill="#fff" textLength="270">tests</text><text aria-hidden="true" x="515" y="150" fill="#010101" fill-opacity=".3" transform="scale(.1)" textLength="210">155</text><text x="515" y="140" transform="scale(.1)" fill="#fff" textLength="210">155</text></g></svg>
|
Before Width: | Height: | Size: 1.1 KiB After Width: | Height: | Size: 1.1 KiB |
@ -156,6 +156,7 @@ class TestSetAndGetBoolHigher:
|
|||||||
[("A1"), ("B2")],
|
[("A1"), ("B2")],
|
||||||
)
|
)
|
||||||
def test_it_sets_and_gets_recorder_bool_params(self, param, value):
|
def test_it_sets_and_gets_recorder_bool_params(self, param, value):
|
||||||
|
assert hasattr(vm.recorder, param)
|
||||||
setattr(vm.recorder, param, value)
|
setattr(vm.recorder, param, value)
|
||||||
assert getattr(vm.recorder, param) == value
|
assert getattr(vm.recorder, param) == value
|
||||||
|
|
||||||
@ -168,7 +169,56 @@ class TestSetAndGetBoolHigher:
|
|||||||
[("loop")],
|
[("loop")],
|
||||||
)
|
)
|
||||||
def test_it_sets_recorder_bool_params(self, param, value):
|
def test_it_sets_recorder_bool_params(self, param, value):
|
||||||
|
assert hasattr(vm.recorder, param)
|
||||||
setattr(vm.recorder, param, value)
|
setattr(vm.recorder, param, value)
|
||||||
|
assert getattr(vm.recorder, param) == value
|
||||||
|
|
||||||
|
""" recoder.mode tests """
|
||||||
|
|
||||||
|
@pytest.mark.skipif(
|
||||||
|
data.name == "basic",
|
||||||
|
reason="Skip test if kind is basic",
|
||||||
|
)
|
||||||
|
@pytest.mark.parametrize(
|
||||||
|
"param",
|
||||||
|
[("loop"), ("recbus")],
|
||||||
|
)
|
||||||
|
def test_it_sets_recorder_mode_bool_params(self, param, value):
|
||||||
|
assert hasattr(vm.recorder.mode, param)
|
||||||
|
setattr(vm.recorder.mode, param, value)
|
||||||
|
assert getattr(vm.recorder.mode, param) == value
|
||||||
|
|
||||||
|
""" recorder.armstrip """
|
||||||
|
|
||||||
|
@pytest.mark.skipif(
|
||||||
|
data.name == "basic",
|
||||||
|
reason="Skip test if kind is basic",
|
||||||
|
)
|
||||||
|
@pytest.mark.parametrize(
|
||||||
|
"index",
|
||||||
|
[
|
||||||
|
(data.phys_out),
|
||||||
|
(data.virt_out),
|
||||||
|
],
|
||||||
|
)
|
||||||
|
def test_it_sets_recorder_armstrip_bool_params(self, index, value):
|
||||||
|
vm.recorder.armstrip[index].set(value)
|
||||||
|
|
||||||
|
""" recorder.armbus """
|
||||||
|
|
||||||
|
@pytest.mark.skipif(
|
||||||
|
data.name == "basic",
|
||||||
|
reason="Skip test if kind is basic",
|
||||||
|
)
|
||||||
|
@pytest.mark.parametrize(
|
||||||
|
"index",
|
||||||
|
[
|
||||||
|
(data.phys_out),
|
||||||
|
(data.virt_out),
|
||||||
|
],
|
||||||
|
)
|
||||||
|
def test_it_sets_recorder_armbus_bool_params(self, index, value):
|
||||||
|
vm.recorder.armbus[index].set(True)
|
||||||
|
|
||||||
""" fx tests """
|
""" fx tests """
|
||||||
|
|
||||||
@ -323,6 +373,26 @@ class TestSetAndGetIntHigher:
|
|||||||
vm.option.delay[index].set(value)
|
vm.option.delay[index].set(value)
|
||||||
assert vm.option.delay[index].get() == value
|
assert vm.option.delay[index].get() == value
|
||||||
|
|
||||||
|
""" recorder tests """
|
||||||
|
|
||||||
|
@pytest.mark.skipif(
|
||||||
|
data.name == "basic",
|
||||||
|
reason="Skip test if kind is basic",
|
||||||
|
)
|
||||||
|
@pytest.mark.parametrize(
|
||||||
|
"param,value",
|
||||||
|
[
|
||||||
|
("samplerate", 32000),
|
||||||
|
("samplerate", 96000),
|
||||||
|
("bitresolution", 16),
|
||||||
|
("bitresolution", 32),
|
||||||
|
],
|
||||||
|
)
|
||||||
|
def test_it_sets_and_gets_recorder_int_params(self, param, value):
|
||||||
|
assert hasattr(vm.recorder, param)
|
||||||
|
setattr(vm.recorder, param, value)
|
||||||
|
assert getattr(vm.recorder, param) == value
|
||||||
|
|
||||||
|
|
||||||
class TestSetAndGetFloatHigher:
|
class TestSetAndGetFloatHigher:
|
||||||
__test__ = True
|
__test__ = True
|
||||||
|
@ -1,3 +1,5 @@
|
|||||||
|
import re
|
||||||
|
|
||||||
from .error import VMError
|
from .error import VMError
|
||||||
from .iremote import IRemote
|
from .iremote import IRemote
|
||||||
from .kinds import kinds_all
|
from .kinds import kinds_all
|
||||||
@ -19,9 +21,10 @@ class Recorder(IRemote):
|
|||||||
Returns a Recorder class of a kind.
|
Returns a Recorder class of a kind.
|
||||||
"""
|
"""
|
||||||
CHANNELOUTMIXIN_cls = _make_channelout_mixins[remote.kind.name]
|
CHANNELOUTMIXIN_cls = _make_channelout_mixins[remote.kind.name]
|
||||||
|
ARMCHANNELMIXIN_cls = _make_armchannel_mixins(remote)[remote.kind.name]
|
||||||
REC_cls = type(
|
REC_cls = type(
|
||||||
f"Recorder{remote.kind}",
|
f"Recorder{remote.kind}",
|
||||||
(cls, CHANNELOUTMIXIN_cls),
|
(cls, CHANNELOUTMIXIN_cls, ARMCHANNELMIXIN_cls),
|
||||||
{
|
{
|
||||||
**{
|
**{
|
||||||
param: action_fn(param)
|
param: action_fn(param)
|
||||||
@ -35,6 +38,7 @@ class Recorder(IRemote):
|
|||||||
"rew",
|
"rew",
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
|
"mode": RecorderMode(remote),
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
return REC_cls(remote)
|
return REC_cls(remote)
|
||||||
@ -46,20 +50,183 @@ class Recorder(IRemote):
|
|||||||
def identifier(self) -> str:
|
def identifier(self) -> str:
|
||||||
return "recorder"
|
return "recorder"
|
||||||
|
|
||||||
|
@property
|
||||||
|
def samplerate(self) -> int:
|
||||||
|
return int(self.getter("samplerate"))
|
||||||
|
|
||||||
|
@samplerate.setter
|
||||||
|
def samplerate(self, val: int):
|
||||||
|
opts = (22050, 24000, 32000, 44100, 48000, 88200, 96000, 176400, 192000)
|
||||||
|
if val not in opts:
|
||||||
|
self.logger.warning(f"samplerate got: {val} but expected a value in {opts}")
|
||||||
|
self.setter("samplerate", val)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def bitresolution(self) -> int:
|
||||||
|
return int(self.getter("bitresolution"))
|
||||||
|
|
||||||
|
@bitresolution.setter
|
||||||
|
def bitresolution(self, val: int):
|
||||||
|
opts = (8, 16, 24, 32)
|
||||||
|
if val not in opts:
|
||||||
|
self.logger.warning(
|
||||||
|
f"bitresolution got: {val} but expected a value in {opts}"
|
||||||
|
)
|
||||||
|
self.setter("bitresolution", val)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def channel(self) -> int:
|
||||||
|
return int(self.getter("channel"))
|
||||||
|
|
||||||
|
@channel.setter
|
||||||
|
def channel(self, val: int):
|
||||||
|
if not 1 <= val <= 8:
|
||||||
|
self.logger.warning(f"channel got: {val} but expected a value from 1 to 8")
|
||||||
|
self.setter("channel", val)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def kbps(self):
|
||||||
|
return int(self.getter("kbps"))
|
||||||
|
|
||||||
|
@kbps.setter
|
||||||
|
def kbps(self, val: int):
|
||||||
|
opts = (32, 40, 48, 56, 64, 80, 96, 112, 128, 160, 192, 224, 256, 320)
|
||||||
|
if val not in opts:
|
||||||
|
self.logger.warning(f"kbps got: {val} but expected a value in {opts}")
|
||||||
|
self.setter("kbps", val)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def gain(self) -> float:
|
||||||
|
return round(self.getter("gain"), 1)
|
||||||
|
|
||||||
|
@gain.setter
|
||||||
|
def gain(self, val: float):
|
||||||
|
self.setter("gain", val)
|
||||||
|
|
||||||
def load(self, file: str):
|
def load(self, file: str):
|
||||||
try:
|
try:
|
||||||
self.setter("load", file)
|
self.setter("load", file)
|
||||||
except UnicodeError:
|
except UnicodeError:
|
||||||
raise VMError("File full directory must be a raw string")
|
raise VMError("File full directory must be a raw string")
|
||||||
|
|
||||||
def set_loop(self, val: bool):
|
# loop forwarder methods, for backwards compatibility
|
||||||
self.setter("mode.loop", 1 if val else 0)
|
@property
|
||||||
|
def loop(self):
|
||||||
|
return self.mode.loop
|
||||||
|
|
||||||
loop = property(fset=set_loop)
|
@loop.setter
|
||||||
|
def loop(self, val: bool):
|
||||||
|
self.mode.loop = val
|
||||||
|
|
||||||
|
def goto(self, time_str):
|
||||||
|
def get_sec():
|
||||||
|
"""Get seconds from time string"""
|
||||||
|
h, m, s = time_str.split(":")
|
||||||
|
return int(h) * 3600 + int(m) * 60 + int(s)
|
||||||
|
|
||||||
|
time_str = str(time_str) # coerce the type
|
||||||
|
if (
|
||||||
|
match := re.match(
|
||||||
|
r"^(?:[01]\d|2[0123]):(?:[012345]\d):(?:[012345]\d)$",
|
||||||
|
time_str,
|
||||||
|
)
|
||||||
|
is not None
|
||||||
|
):
|
||||||
|
self.setter("goto", get_sec())
|
||||||
|
else:
|
||||||
|
self.logger.warning(
|
||||||
|
f"goto expects a string that matches the format 'hh:mm:ss'"
|
||||||
|
)
|
||||||
|
|
||||||
|
def filetype(self, val: str):
|
||||||
|
opts = {"wav": 1, "aiff": 2, "bwf": 3, "mp3": 100}
|
||||||
|
try:
|
||||||
|
self.setter("filetype", opts[val.lower()])
|
||||||
|
except KeyError:
|
||||||
|
self.logger.warning(
|
||||||
|
f"filetype got: {val} but expected a value in {list(opts.keys())}"
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class RecorderMode(IRemote):
|
||||||
|
@property
|
||||||
|
def identifier(self):
|
||||||
|
return "recorder.mode"
|
||||||
|
|
||||||
|
@property
|
||||||
|
def recbus(self) -> bool:
|
||||||
|
return self.getter("recbus") == 1
|
||||||
|
|
||||||
|
@recbus.setter
|
||||||
|
def recbus(self, val: bool):
|
||||||
|
self.setter("recbus", 1 if val else 0)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def playonload(self) -> bool:
|
||||||
|
return self.getter("playonload") == 1
|
||||||
|
|
||||||
|
@playonload.setter
|
||||||
|
def playonload(self, val: bool):
|
||||||
|
self.setter("playonload", 1 if val else 0)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def loop(self) -> bool:
|
||||||
|
return self.getter("loop") == 1
|
||||||
|
|
||||||
|
@loop.setter
|
||||||
|
def loop(self, val: bool):
|
||||||
|
self.setter("loop", 1 if val else 0)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def multitrack(self) -> bool:
|
||||||
|
return self.getter("multitrack") == 1
|
||||||
|
|
||||||
|
@multitrack.setter
|
||||||
|
def multitrack(self, val: bool):
|
||||||
|
self.setter("multitrack", 1 if val else 0)
|
||||||
|
|
||||||
|
|
||||||
|
class RecorderArmChannel(IRemote):
|
||||||
|
def __init__(self, remote, i):
|
||||||
|
super().__init__(remote)
|
||||||
|
self._i = i
|
||||||
|
|
||||||
|
def set(self, val: bool):
|
||||||
|
self.setter("", 1 if val else 0)
|
||||||
|
|
||||||
|
|
||||||
|
class RecorderArmStrip(RecorderArmChannel):
|
||||||
|
@property
|
||||||
|
def identifier(self):
|
||||||
|
return f"recorder.armstrip[{self._i}]"
|
||||||
|
|
||||||
|
|
||||||
|
class RecorderArmBus(RecorderArmChannel):
|
||||||
|
@property
|
||||||
|
def identifier(self):
|
||||||
|
return f"recorder.armbus[{self._i}]"
|
||||||
|
|
||||||
|
|
||||||
|
def _make_armchannel_mixin(remote, kind):
|
||||||
|
"""Creates an armchannel out mixin"""
|
||||||
|
return type(
|
||||||
|
f"ArmChannelMixin{kind}",
|
||||||
|
(),
|
||||||
|
{
|
||||||
|
"armstrip": tuple(
|
||||||
|
RecorderArmStrip(remote, i) for i in range(kind.num_strip)
|
||||||
|
),
|
||||||
|
"armbus": tuple(RecorderArmBus(remote, i) for i in range(kind.num_bus)),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def _make_armchannel_mixins(remote):
|
||||||
|
return {kind.name: _make_armchannel_mixin(remote, kind) for kind in kinds_all}
|
||||||
|
|
||||||
|
|
||||||
def _make_channelout_mixin(kind):
|
def _make_channelout_mixin(kind):
|
||||||
"""Creates a channel out property mixin"""
|
"""Creates a channel out mixin"""
|
||||||
return type(
|
return type(
|
||||||
f"ChannelOutMixin{kind}",
|
f"ChannelOutMixin{kind}",
|
||||||
(),
|
(),
|
||||||
|
Loading…
x
Reference in New Issue
Block a user