Compare commits

..

10 Commits

Author SHA1 Message Date
e9b9295a46 removes unnecessary import 2023-09-27 21:27:10 +01:00
2db268551c adds modify bus assignment binds
to configuration section in readme
2023-09-27 21:22:34 +01:00
0be7919f12 add Bus Assignments to README 2023-09-27 21:17:58 +01:00
ab728f0a32 fix bug bind overrides 2023-09-27 19:18:52 +01:00
050e0336b8 add voicemeeter kinds to README 2023-09-27 19:09:09 +01:00
f49b04d4f6 adds Configuration section to README 2023-09-27 18:51:51 +01:00
c9781ff92a bump to 0.5 2023-09-27 18:39:09 +01:00
bfccc323f1 slider mode commands split up
config.py added. loads custom user settings

_make_gestures moved into util.py
2023-09-27 18:38:13 +01:00
86dbe0b335 add Install section to README 2023-09-27 16:53:02 +01:00
82f32643d6 fix typo 2023-09-27 16:47:11 +01:00
7 changed files with 185 additions and 93 deletions

View File

@ -1,8 +1,12 @@
# NVDA Addon Voicemeeter # NVDA Addon Voicemeeter
Control Voicemeeter GUI with customisable hotkeys. Control Voicemeeter with global hotkeys.
## Keybinds ## Install
This addon can be installed through the Add-on store, `Install from external source`. Simply download the [latest Release](https://github.com/onyx-and-iris/nvda-addon-voicemeeter/releases) and load it with NVDA.
## Default Keybinds
### Controllers ### Controllers
@ -22,7 +26,7 @@ Control Voicemeeter GUI with customisable hotkeys.
- `NVDA+alt+g`: Enable gain slider mode. - `NVDA+alt+g`: Enable gain slider mode.
- `NVDA+alt+c`: Enable comp slider mode. - `NVDA+alt+c`: Enable comp slider mode.
- `NVDA+alt+t`: Enable gate sldier mode. - `NVDA+alt+t`: Enable gate slider mode.
- `NVDA+alt+d`: Enable denoiser slider mode. - `NVDA+alt+d`: Enable denoiser slider mode.
- `NVDA+alt+a`: Enable audibility slider mode. - `NVDA+alt+a`: Enable audibility slider mode.
@ -43,7 +47,74 @@ Control Voicemeeter GUI with customisable hotkeys.
- `NVDA+shift+c`: MC - `NVDA+shift+c`: MC
- `NVDA+shift+k`: Karaoke - `NVDA+shift+k`: Karaoke
### Bus Assignments (A1-A5|B1-B3)
- `NVDA+shift+1`: Toggle BUS assignment 1 for a strip
- `NVDA+shift+2`: Toggle BUS assignment 2 for a strip
- `NVDA+shift+3`: Toggle BUS assignment 3 for a strip
- `NVDA+shift+4`: Toggle BUS assignment 4 for a strip
- `NVDA+shift+5`: Toggle BUS assignment 5 for a strip
- `NVDA+shift+6`: Toggle BUS assignment 6 for a strip
- `NVDA+shift+7`: Toggle BUS assignment 7 for a strip
- `NVDA+shift+8`: Toggle BUS assignment 8 for a strip
### Announcements ### Announcements
- `NVDA+shift+q`: Announce current controller. - `NVDA+shift+q`: Announce current controller.
- `NVDA+shift+a`: Announce Voicemeeter kind. - `NVDA+shift+a`: Announce Voicemeeter kind.
## Configuration
By placing a file named `nvda_settings.json` in `User Home Directory / Documents / Voicemeeter` (the same place as your Voicemeeter xml profiles) you can change most of the default keybinds.
The `voicemeeter` key can take one of three values:
- `basic`
- `banana`
- `potato`
example:
```json
{
"voicemeeter": "banana",
"keybinds": {
"NVDA+alt+k": "strip_mode",
"NVDA+alt+l": "bus_mode",
"NVDA+alt+g": "gain_mode",
"NVDA+alt+c": "comp_mode",
"NVDA+alt+t": "gate_mode",
"NVDA+alt+d": "denoiser_mode",
"NVDA+alt+a": "audibility_mode",
"NVDA+shift+q": "announce_controller",
"NVDA+shift+z": "announce_voicemeeter_version",
"NVDA+shift+s": "toggle_solo",
"NVDA+shift+m": "toggle_mute",
"NVDA+shift+c": "toggle_mc",
"NVDA+shift+k": "karaoke",
"NVDA+shift+upArrow": "slider_increase_by_point_one",
"NVDA+shift+downArrow": "slider_decrease_by_point_one",
"NVDA+shift+alt+upArrow": "slider_increase_by_one",
"NVDA+shift+alt+downArrow": "slider_decrease_by_one",
"NVDA+shift+control+upArrow": "slider_increase_by_three",
"NVDA+shift+control+downArrow": "slider_decrease_by_three",
"NVDA+control+1": "bus_assignment",
"NVDA+control+2": "bus_assignment",
"NVDA+control+3": "bus_assignment",
"NVDA+control+4": "bus_assignment",
"NVDA+control+5": "bus_assignment",
"NVDA+control+6": "bus_assignment",
"NVDA+control+7": "bus_assignment",
"NVDA+control+8": "bus_assignment"
}
}
```
Would make the following changes:
- load the plugin in `banana` mode (default is potato)
- change the `strip_mode` and `bus_mode` binds to `NVDA+alt+k` and `NVDA+alt+l` respectively
- change the `announce_voicemeeter_version` bind to `NVDA+shift+z`
- changes the bus assignment binds to `NVDA+control+number`
All other binds would then be defaults.

View File

@ -1,63 +1,16 @@
import json
import time import time
from pathlib import Path
import globalPluginHandler import globalPluginHandler
from logHandler import log
from . import config, util
from .commands import CommandsMixin from .commands import CommandsMixin
from .controller import Controller from .controller import Controller
from .kinds import KindId, request_kind_map from .kinds import KindId, request_kind_map
def _make_gestures():
defaults = {
"kb:NVDA+alt+s": "strip_mode",
"kb:NVDA+alt+b": "bus_mode",
"kb:NVDA+alt+g": "slider_mode", # Gate
"kb:NVDA+alt+c": "slider_mode", # Comp
"kb:NVDA+alt+t": "slider_mode", # Gate
"kb:NVDA+alt+d": "slider_mode", # Denoiser
"kb:NVDA+alt+a": "slider_mode", # Audibility
"kb:NVDA+shift+q": "announce_controller",
"kb:NVDA+shift+a": "announce_voicemeeter_version",
"kb:NVDA+shift+o": "toggle_mono",
"kb:NVDA+shift+s": "toggle_solo",
"kb:NVDA+shift+m": "toggle_mute",
"kb:NVDA+shift+c": "toggle_mc",
"kb:NVDA+shift+k": "karaoke",
"kb:NVDA+shift+upArrow": "slider_increase",
"kb:NVDA+shift+downArrow": "slider_decrease",
"kb:NVDA+shift+alt+upArrow": "slider_increase",
"kb:NVDA+shift+alt+downArrow": "slider_decrease",
"kb:NVDA+shift+control+upArrow": "slider_increase",
"kb:NVDA+shift+control+downArrow": "slider_decrease",
}
overrides = None
pn = Path.home() / "Documents" / "Voicemeeter" / "keybinds.json"
if pn.exists():
with open(pn, "r") as f:
data = json.load(f)
overrides = {f"kb:{v}": k for k, v in data.items()}
log.info("INFO - loading settings from keybinds.json")
if overrides:
return {**defaults, **overrides}
return defaults
def _get_kind_id():
pn = Path.home() / "Documents" / "Voicemeeter" / "settings.json"
if pn.exists():
with open(pn, "r") as f:
data = json.load(f)
return data["voicemeeter"]
return "potato"
class GlobalPlugin(globalPluginHandler.GlobalPlugin, CommandsMixin): class GlobalPlugin(globalPluginHandler.GlobalPlugin, CommandsMixin):
__gestures = _make_gestures() __kind_id = config.get("voicemeeter", "potato")
__kind_id = _get_kind_id() __gestures = util._make_gestures(__kind_id)
def __init__(self, *args, **kwargs): def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs) super().__init__(*args, **kwargs)
@ -65,12 +18,7 @@ class GlobalPlugin(globalPluginHandler.GlobalPlugin, CommandsMixin):
if self.controller.login() == 1: if self.controller.login() == 1:
self.controller.run_voicemeeter(KindId[self.__kind_id.upper()]) self.controller.run_voicemeeter(KindId[self.__kind_id.upper()])
time.sleep(1) time.sleep(1)
self.kind = request_kind_map(self.controller.kind_id) self.kind = request_kind_map(self.__kind_id)
for i in range(1, self.kind.num_strip + 1):
self.bindGesture(f"kb:NVDA+alt+{i}", "index")
for i in range(1, self.kind.phys_out + self.kind.virt_out + 1):
self.bindGesture(f"kb:NVDA+shift+{i}", "bus_assignment")
def terminate(self, *args, **kwargs): def terminate(self, *args, **kwargs):
super().terminate(*args, **kwargs) super().terminate(*args, **kwargs)

View File

@ -1,7 +1,7 @@
import ui import ui
from logHandler import log from logHandler import log
from . import context, util from . import context
class CommandsMixin: class CommandsMixin:
@ -37,18 +37,24 @@ class CommandsMixin:
ui.message(f"Controller for {self.controller.ctx.strategy} {self.controller.ctx.index + 1}") ui.message(f"Controller for {self.controller.ctx.strategy} {self.controller.ctx.index + 1}")
log.info(f"INFO - {self.controller.ctx.strategy} {self.controller.ctx.index} mode") log.info(f"INFO - {self.controller.ctx.strategy} {self.controller.ctx.index} mode")
def script_slider_mode(self, gesture): def __set_slider_mode(self, mode):
if gesture.displayName.endswith("g"): self.controller.ctx.slider_mode = mode
self.controller.ctx.slider_mode = "gain" ui.message(f"{mode} mode enabled")
elif gesture.displayName.endswith("c"):
self.controller.ctx.slider_mode = "comp" def script_gain_mode(self, _):
elif gesture.displayName.endswith("t"): self.__set_slider_mode("gain")
self.controller.ctx.slider_mode = "gate"
elif gesture.displayName.endswith("d"): def script_comp_mode(self, _):
self.controller.ctx.slider_mode = "denoiser" self.__set_slider_mode("comp")
elif gesture.displayName.endswith("a"):
self.controller.ctx.slider_mode = "audibility" def script_gate_mode(self, _):
ui.message(f"{self.controller.ctx.slider_mode} mode enabled") self.__set_slider_mode("gate")
def script_denoiser_mode(self, _):
self.__set_slider_mode("denoiser")
def script_audibility_mode(self, _):
self.__set_slider_mode("audibility")
### BOOLEAN PARAMETERS ### ### BOOLEAN PARAMETERS ###
@ -89,28 +95,34 @@ class CommandsMixin:
self.controller.ctx.set_bool(output, val) self.controller.ctx.set_bool(output, val)
ui.message("on" if val else "off") ui.message("on" if val else "off")
### SLIDER MODES ### ### CONTROL SLIDERS ###
def script_slider_increase(self, gesture): def script_slider_increase_by_point_one(self, gesture):
op = util.remove_prefix(gesture.displayName, "kb:NVDA+shift+") val = self.controller.ctx.get_float(self.controller.ctx.slider_mode) + 0.1
if op.startswith("alt"):
offset = 0.1
elif op.startswith("ctrl"):
offset = 3
else:
offset = 1
val = self.controller.ctx.get_float(self.controller.ctx.slider_mode) + offset
self.controller.ctx.set_float(self.controller.ctx.slider_mode, val) self.controller.ctx.set_float(self.controller.ctx.slider_mode, val)
ui.message(str(round(val, 1))) ui.message(str(round(val, 1)))
def script_slider_decrease(self, gesture): def script_slider_decrease_by_point_one(self, gesture):
op = util.remove_prefix(gesture.displayName, "kb:NVDA+shift+") val = self.controller.ctx.get_float(self.controller.ctx.slider_mode) - 0.1
if op.startswith("alt"): self.controller.ctx.set_float(self.controller.ctx.slider_mode, val)
offset = 0.1 ui.message(str(round(val, 1)))
elif op.startswith("ctrl"):
offset = 3 def script_slider_increase_by_one(self, gesture):
else: val = self.controller.ctx.get_float(self.controller.ctx.slider_mode) + 1
offset = 1 self.controller.ctx.set_float(self.controller.ctx.slider_mode, val)
val = self.controller.ctx.get_float(self.controller.ctx.slider_mode) - offset ui.message(str(round(val, 1)))
def script_slider_decrease_by_one(self, gesture):
val = self.controller.ctx.get_float(self.controller.ctx.slider_mode) - 1
self.controller.ctx.set_float(self.controller.ctx.slider_mode, val)
ui.message(str(round(val, 1)))
def script_slider_increase_by_three(self, gesture):
val = self.controller.ctx.get_float(self.controller.ctx.slider_mode) + 3
self.controller.ctx.set_float(self.controller.ctx.slider_mode, val)
ui.message(str(round(val, 1)))
def script_slider_decrease_by_three(self, gesture):
val = self.controller.ctx.get_float(self.controller.ctx.slider_mode) - 3
self.controller.ctx.set_float(self.controller.ctx.slider_mode, val) self.controller.ctx.set_float(self.controller.ctx.slider_mode, val)
ui.message(str(round(val, 1))) ui.message(str(round(val, 1)))

View File

@ -0,0 +1,20 @@
import json
from pathlib import Path
def config_from_json():
pn = Path.home() / "Documents" / "Voicemeeter" / "nvda_settings.json"
data = None
if pn.exists():
with open(pn, "r") as f:
data = json.load(f)
return data or {}
__config = config_from_json()
def get(name, default=None):
if name in __config:
return __config[name]
return default

View File

@ -1,3 +1,7 @@
from . import config
from .kinds import request_kind_map
def remove_prefix(input_string, prefix): def remove_prefix(input_string, prefix):
if prefix and input_string.startswith(prefix): if prefix and input_string.startswith(prefix):
return input_string[len(prefix) :] return input_string[len(prefix) :]
@ -8,3 +12,40 @@ def remove_suffix(input_string, suffix):
if suffix and input_string.endswith(suffix): if suffix and input_string.endswith(suffix):
return input_string[: -len(suffix)] return input_string[: -len(suffix)]
return input_string return input_string
def _make_gestures(kind_id):
kind = request_kind_map(kind_id)
defaults = {
"kb:NVDA+alt+s": "strip_mode",
"kb:NVDA+alt+b": "bus_mode",
"kb:NVDA+alt+g": "gain_mode",
"kb:NVDA+alt+c": "comp_mode",
"kb:NVDA+alt+t": "gate_mode",
"kb:NVDA+alt+d": "denoiser_mode",
"kb:NVDA+alt+a": "audibility_mode",
"kb:NVDA+shift+q": "announce_controller",
"kb:NVDA+shift+v": "announce_voicemeeter_version",
"kb:NVDA+shift+o": "toggle_mono",
"kb:NVDA+shift+s": "toggle_solo",
"kb:NVDA+shift+m": "toggle_mute",
"kb:NVDA+shift+c": "toggle_mc",
"kb:NVDA+shift+k": "karaoke",
"kb:NVDA+shift+upArrow": "slider_increase_by_point_one",
"kb:NVDA+shift+downArrow": "slider_decrease_by_point_one",
"kb:NVDA+shift+alt+upArrow": "slider_increase_by_one",
"kb:NVDA+shift+alt+downArrow": "slider_decrease_by_one",
"kb:NVDA+shift+control+upArrow": "slider_increase_by_three",
"kb:NVDA+shift+control+downArrow": "slider_decrease_by_three",
}
for i in range(1, kind.num_strip + 1):
defaults[f"kb:NVDA+alt+{i}"] = "index"
for i in range(1, kind.phys_out + kind.virt_out + 1):
defaults[f"kb:NVDA+shift+{i}"] = "bus_assignment"
abc = config.get("keybinds")
if abc:
overrides = {f"kb:{remove_prefix(k, 'kb:')}": v for k, v in abc.items()}
matching_values = set(defaults.values()).intersection(set(overrides.values()))
defaults = {k: v for k, v in defaults.items() if v not in matching_values}
return {**defaults, **overrides}
return defaults

View File

@ -5,7 +5,7 @@ param(
function Copy-FilestoScratchpad { function Copy-FilestoScratchpad {
$source = Join-Path $PSScriptRoot "addon" "globalPlugins" "voicemeeter" $source = Join-Path $PSScriptRoot "addon" "globalPlugins" "voicemeeter"
$target = Join-Path $env:appdata "nvda" "scratchpad" "globalPlugins" "voicemeeter" $target = Join-Path $env:appdata "nvda" "scratchpad" "globalPlugins" "voicemeeter"
Robocopy $source $target /MIR /NFL /NDL /NJH /NJS /nc /ns /np Robocopy $source $target | Out-Null
} }
function main { function main {

View File

@ -28,7 +28,7 @@ addon_info = {
The add-on requires Voicemeeter to be installed.""" The add-on requires Voicemeeter to be installed."""
), ),
# version # version
"addon_version": "0.4", "addon_version": "0.5",
# Author(s) # Author(s)
"addon_author": "onyx-and-iris <code@onyxandiris.online>", "addon_author": "onyx-and-iris <code@onyxandiris.online>",
# URL for the add-on documentation support # URL for the add-on documentation support