mirror of
https://github.com/onyx-and-iris/nvda-addon-voicemeeter.git
synced 2026-04-07 20:13:30 +00:00
Compare commits
17 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| c3b06cae51 | |||
| 4c176cfd77 | |||
| df023c67ea | |||
| 8b025206b1 | |||
| 8e8e3ce8a5 | |||
| e565283827 | |||
| 2d194e8e67 | |||
| 3aab5922ab | |||
| 1fb435416f | |||
| 40aaeb4c54 | |||
| 17cdd84c51 | |||
| 43379f1b09 | |||
| dc9ac5ddc3 | |||
| 9764b9d827 | |||
| d95a2280c6 | |||
| 5534381981 | |||
| 0522b69420 |
52
.github/workflows/build_addon.yml
vendored
52
.github/workflows/build_addon.yml
vendored
@@ -1,46 +1,35 @@
|
||||
name: build addon
|
||||
name: Build Addon
|
||||
|
||||
on:
|
||||
push:
|
||||
tags: ["*"]
|
||||
# To build on main/master branch, uncomment the following line:
|
||||
# branches: [ main , master ]
|
||||
|
||||
pull_request:
|
||||
branches: [ main, master ]
|
||||
tags:
|
||||
- 'v*.*.*'
|
||||
|
||||
workflow_dispatch:
|
||||
|
||||
jobs:
|
||||
build:
|
||||
|
||||
runs-on: ubuntu-latest
|
||||
|
||||
steps:
|
||||
- uses: actions/checkout@v3
|
||||
- name: Checkout code
|
||||
uses: actions/checkout@v4
|
||||
|
||||
- run: echo -e "pre-commit\nscons\nmarkdown">requirements.txt
|
||||
|
||||
- name: Set up Python
|
||||
uses: actions/setup-python@v4
|
||||
- name: Set up PDM
|
||||
uses: pdm-project/setup-pdm@v4
|
||||
with:
|
||||
python-version: 3.9
|
||||
cache: 'pip'
|
||||
python-version: '3.11'
|
||||
|
||||
- name: Install dependencies
|
||||
run: |
|
||||
python -m pip install --upgrade pip wheel
|
||||
pip install -r requirements.txt
|
||||
sudo apt-get update -y
|
||||
sudo apt-get install -y gettext
|
||||
pdm sync -d -G build
|
||||
|
||||
- name: Code checks
|
||||
run: export SKIP=no-commit-to-branch; pre-commit run --all
|
||||
- name: Build addon
|
||||
run: pdm run scons
|
||||
|
||||
- name: building addon
|
||||
run: scons
|
||||
|
||||
- uses: actions/upload-artifact@v3
|
||||
- name: Upload build artifacts
|
||||
if: success()
|
||||
uses: actions/upload-artifact@v4
|
||||
with:
|
||||
name: packaged_addon
|
||||
path: ./*.nvda-addon
|
||||
@@ -48,17 +37,20 @@ jobs:
|
||||
upload_release:
|
||||
runs-on: ubuntu-latest
|
||||
if: ${{ startsWith(github.ref, 'refs/tags/') }}
|
||||
needs: ["build"]
|
||||
needs: build
|
||||
|
||||
steps:
|
||||
- uses: actions/checkout@v3
|
||||
- name: download releases files
|
||||
- name: Download releases files
|
||||
uses: actions/download-artifact@v4.1.7
|
||||
with:
|
||||
name: packaged_addon
|
||||
|
||||
- name: Display structure of downloaded files
|
||||
run: ls -R
|
||||
run: tree
|
||||
|
||||
- name: Release
|
||||
uses: softprops/action-gh-release@v1
|
||||
with:
|
||||
files: packaged_addon/*.nvda-addon
|
||||
files: ./*.nvda-addon
|
||||
fail_on_unmatched_files: true
|
||||
prerelease: ${{ contains(github.ref, '-') }}
|
||||
|
||||
@@ -1,4 +1,4 @@
|
||||
[](https://pdm.fming.dev)
|
||||
[](https://pdm-project.org)
|
||||
[](https://github.com/astral-sh/ruff)
|
||||
|
||||
# NVDA Addon Voicemeeter
|
||||
@@ -85,6 +85,7 @@ example:
|
||||
```json
|
||||
{
|
||||
"voicemeeter": "banana",
|
||||
"bits": 64,
|
||||
"keybinds": {
|
||||
"NVDA+alt+k": "strip_mode",
|
||||
"NVDA+alt+l": "bus_mode",
|
||||
@@ -120,6 +121,7 @@ example:
|
||||
Would make the following changes:
|
||||
|
||||
- load the plugin in `banana` mode (default is potato)
|
||||
- override the bits of Voicemeeter GUI to 64 (default is 32)
|
||||
- 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`
|
||||
|
||||
41
Taskfile.yml
Normal file
41
Taskfile.yml
Normal file
@@ -0,0 +1,41 @@
|
||||
version: '3'
|
||||
|
||||
vars:
|
||||
SHELL: pwsh
|
||||
|
||||
tasks:
|
||||
default:
|
||||
desc: Build the addon
|
||||
cmds:
|
||||
- task: build
|
||||
|
||||
copy:
|
||||
desc: Copy sources files to scratchpad directory
|
||||
platforms: [windows]
|
||||
vars:
|
||||
SOURCE: addon\globalPlugins\voicemeeter
|
||||
DEST: ${env:appdata}\nvda\scratchpad\globalPlugins\voicemeeter\
|
||||
cmds:
|
||||
- "{{.SHELL}} -Command 'robocopy {{.SOURCE}} {{.DEST}} /NDL'"
|
||||
ignore_error: true
|
||||
|
||||
build:
|
||||
desc: Build the addon
|
||||
platforms: [windows]
|
||||
cmds:
|
||||
- '{{.SHELL}} -Command "pdm run scons"'
|
||||
|
||||
bump:
|
||||
desc: Bump the version number
|
||||
preconditions:
|
||||
- sh: '{{.SHELL}} -c "if (Get-Command bump) { exit 0 } else { exit 1 }"'
|
||||
msg: "The 'bump' command is not available. Run `go install github.com/onyx-and-iris/bump/cmd/bump@latest`."
|
||||
cmds:
|
||||
- |
|
||||
{{if eq .CLI_ARGS "show"}}
|
||||
{{.SHELL}} -c "bump show -f buildVars.py -p \"'addon_version': '(\d+\.\d+\.\d+)'\""
|
||||
{{.SHELL}} -c "bump show -f pyproject.toml -p \"version = .(\d+\.\d+\.\d+).\""
|
||||
{{else}}
|
||||
{{.SHELL}} -c "bump {{.CLI_ARGS}} -w -f buildVars.py -p \"'addon_version': '(\d+\.\d+\.\d+)'\""
|
||||
{{.SHELL}} -c "bump {{.CLI_ARGS}} -w -f pyproject.toml -p \"version = .(\d+\.\d+\.\d+).\""
|
||||
{{end}}
|
||||
@@ -2,7 +2,7 @@ import ctypes as ct
|
||||
from ctypes.wintypes import CHAR, FLOAT, LONG
|
||||
|
||||
from .cdll import libc
|
||||
from .error import VMCAPIError
|
||||
from .error import VMAddonCAPIError
|
||||
|
||||
|
||||
class Binds:
|
||||
@@ -41,5 +41,5 @@ class Binds:
|
||||
def call(self, fn, *args, ok=(0,)):
|
||||
retval = fn(*args)
|
||||
if retval not in ok:
|
||||
raise VMCAPIError(fn.__name__, retval)
|
||||
raise VMAddonCAPIError(fn.__name__, retval)
|
||||
return retval
|
||||
|
||||
@@ -1,16 +1,22 @@
|
||||
import ctypes as ct
|
||||
import platform
|
||||
import winreg
|
||||
from pathlib import Path
|
||||
|
||||
from .error import VMError
|
||||
from .error import VMAddonError
|
||||
|
||||
try:
|
||||
import winreg
|
||||
except ImportError as e:
|
||||
ERR_MSG = 'winreg module not found, only Windows OS supported'
|
||||
raise VMAddonError(ERR_MSG) from e
|
||||
|
||||
# Defense against edge cases where winreg imports but we're not on Windows
|
||||
if platform.system() != 'Windows':
|
||||
ERR_MSG = f'Unsupported OS: {platform.system()}, only Windows OS supported'
|
||||
raise VMAddonError(ERR_MSG)
|
||||
|
||||
BITS = 64 if ct.sizeof(ct.c_voidp) == 8 else 32
|
||||
|
||||
if platform.system() != 'Windows':
|
||||
raise VMError('Only Windows OS supported')
|
||||
|
||||
|
||||
VM_KEY = 'VB:Voicemeeter {17359A74-1236-5467}'
|
||||
REG_KEY = '\\'.join(
|
||||
filter(
|
||||
@@ -35,15 +41,14 @@ def get_vmpath():
|
||||
try:
|
||||
vm_parent = Path(get_vmpath()).parent
|
||||
except FileNotFoundError as e:
|
||||
raise VMError('Unable to fetch DLL path from the registry') from e
|
||||
ERR_MSG = 'Voicemeeter installation not found in registry'
|
||||
raise VMAddonError(ERR_MSG) from e
|
||||
|
||||
DLL_NAME = f'VoicemeeterRemote{"64" if BITS == 64 else ""}.dll'
|
||||
|
||||
dll_path = vm_parent.joinpath(DLL_NAME)
|
||||
if not dll_path.is_file():
|
||||
raise VMError(f'Could not find {dll_path}')
|
||||
ERR_MSG = f'Could not find {dll_path}'
|
||||
raise VMAddonError(ERR_MSG)
|
||||
|
||||
if BITS == 64:
|
||||
libc = ct.CDLL(str(dll_path))
|
||||
else:
|
||||
libc = ct.WinDLL(str(dll_path))
|
||||
|
||||
@@ -8,7 +8,7 @@ class CommandsMixin:
|
||||
### ANNOUNCEMENTS ###
|
||||
|
||||
def script_announce_voicemeeter_version(self, _):
|
||||
ui.message(f'Running Voicemeeter {self.kind} {self.controller.version}')
|
||||
ui.message(f'Running Voicemeeter {self.kind} {self.controller.version} {self.controller.bits} bit')
|
||||
|
||||
def script_announce_controller(self, _):
|
||||
ui.message(f'Controller for {self.controller.ctx.strategy} {self.controller.ctx.index + 1}')
|
||||
|
||||
@@ -2,6 +2,7 @@ import ctypes as ct
|
||||
|
||||
from logHandler import log
|
||||
|
||||
from . import config
|
||||
from .binds import Binds
|
||||
from .cdll import BITS
|
||||
from .context import Context, StripStrategy
|
||||
@@ -11,6 +12,7 @@ from .kinds import KindId
|
||||
class Controller(Binds):
|
||||
def __init__(self):
|
||||
self.ctx = Context(StripStrategy(self, 0))
|
||||
self.bits = config.get('bits', BITS)
|
||||
|
||||
def login(self):
|
||||
retval = self.call(self.bind_login, ok=(0, 1))
|
||||
@@ -40,8 +42,8 @@ class Controller(Binds):
|
||||
|
||||
def run_voicemeeter(self, kind_id):
|
||||
val = kind_id.value
|
||||
if val == 3 and BITS == 64:
|
||||
val = 6
|
||||
if self.bits == 64:
|
||||
val += 3
|
||||
self.call(self.bind_run_voicemeeter, val)
|
||||
|
||||
def __clear(self):
|
||||
|
||||
@@ -1,9 +1,9 @@
|
||||
class VMError(Exception):
|
||||
"""Base voicemeeterlib exception class"""
|
||||
class VMAddonError(Exception):
|
||||
"""Base voicemeeter add-on exception class"""
|
||||
|
||||
|
||||
class VMCAPIError(VMError):
|
||||
"""Exception raised when the C-API returns an error code"""
|
||||
class VMAddonCAPIError(VMAddonError):
|
||||
"""Exception raised when the Voicemeeter C-API returns an error code"""
|
||||
|
||||
def __init__(self, fn_name, code):
|
||||
self.fn_name = fn_name
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
from dataclasses import dataclass
|
||||
from enum import Enum, unique
|
||||
|
||||
from .error import VMError
|
||||
from .error import VMAddonError
|
||||
|
||||
|
||||
@unique
|
||||
@@ -14,11 +14,6 @@ class KindId(Enum):
|
||||
@dataclass
|
||||
class KindMapClass:
|
||||
name: str
|
||||
ins: tuple
|
||||
outs: tuple
|
||||
vban: tuple
|
||||
asio: tuple
|
||||
insert: int
|
||||
|
||||
@property
|
||||
def phys_in(self) -> int:
|
||||
@@ -50,7 +45,6 @@ class KindMapClass:
|
||||
|
||||
@dataclass
|
||||
class BasicMap(KindMapClass):
|
||||
name: str
|
||||
ins: tuple = (2, 1)
|
||||
outs: tuple = (1, 1)
|
||||
vban: tuple = (4, 4, 1, 1)
|
||||
@@ -60,7 +54,6 @@ class BasicMap(KindMapClass):
|
||||
|
||||
@dataclass
|
||||
class BananaMap(KindMapClass):
|
||||
name: str
|
||||
ins: tuple = (3, 2)
|
||||
outs: tuple = (3, 2)
|
||||
vban: tuple = (8, 8, 1, 1)
|
||||
@@ -70,7 +63,6 @@ class BananaMap(KindMapClass):
|
||||
|
||||
@dataclass
|
||||
class PotatoMap(KindMapClass):
|
||||
name: str
|
||||
ins: tuple = (5, 3)
|
||||
outs: tuple = (5, 3)
|
||||
vban: tuple = (8, 8, 1, 1)
|
||||
@@ -86,7 +78,8 @@ def kind_factory(kind_id):
|
||||
elif kind_id == 'potato':
|
||||
_kind_map = PotatoMap
|
||||
else:
|
||||
raise ValueError(f'Unknown Voicemeeter kind {kind_id}')
|
||||
ERR_MSG = f'Unknown Voicemeeter kind {kind_id}'
|
||||
raise ValueError(ERR_MSG)
|
||||
return _kind_map(name=kind_id)
|
||||
|
||||
|
||||
@@ -95,5 +88,5 @@ def request_kind_map(kind_id):
|
||||
try:
|
||||
KIND_obj = kind_factory(kind_id)
|
||||
except ValueError as e:
|
||||
raise VMError(str(e)) from e
|
||||
raise VMAddonError(str(e)) from e
|
||||
return KIND_obj
|
||||
|
||||
25
build.ps1
25
build.ps1
@@ -1,25 +0,0 @@
|
||||
param (
|
||||
[switch]$build
|
||||
)
|
||||
|
||||
function Copy-FilesToScratchpad {
|
||||
$source = Join-Path $PSScriptRoot "addon" "globalPlugins" "voicemeeter"
|
||||
$target = Join-Path $env:appdata "nvda" "scratchpad" "globalPlugins" "voicemeeter"
|
||||
Robocopy $source $target | Out-Null
|
||||
}
|
||||
|
||||
function Build-Addon {
|
||||
"Building add-on" | Write-Host
|
||||
scons
|
||||
}
|
||||
|
||||
function Main {
|
||||
"Copying updated files to Scratchpad" | Write-Host
|
||||
Copy-FilesToScratchpad
|
||||
|
||||
if ($build) {
|
||||
Build-Addon
|
||||
}
|
||||
}
|
||||
|
||||
if ($MyInvocation.InvocationName -ne '.') { Main }
|
||||
@@ -28,7 +28,7 @@ addon_info = {
|
||||
The add-on requires Voicemeeter to be installed."""
|
||||
),
|
||||
# version
|
||||
'addon_version': '1.0.0',
|
||||
'addon_version': '1.1.1',
|
||||
# Author(s)
|
||||
'addon_author': 'onyx-and-iris <code@onyxandiris.online>',
|
||||
# URL for the add-on documentation support
|
||||
@@ -38,7 +38,7 @@ The add-on requires Voicemeeter to be installed."""
|
||||
# Documentation file name
|
||||
'addon_docFileName': 'readme.html',
|
||||
# Minimum NVDA version supported (e.g. "2018.3.0", minor version is optional)
|
||||
'addon_minimumNVDAVersion': '2024.4.0',
|
||||
'addon_minimumNVDAVersion': '2022.1.0',
|
||||
# Last NVDA version supported/tested (e.g. "2018.4.0", ideally more recent than minimum version)
|
||||
'addon_lastTestedNVDAVersion': '2024.4.2',
|
||||
# Add-on update channel (default is None, denoting stable releases,
|
||||
|
||||
@@ -1,27 +1,22 @@
|
||||
[project]
|
||||
name = "nvda-addon-voicemeeter"
|
||||
version = "1.0.0"
|
||||
version = "1.1.1"
|
||||
description = "A GUI-less NVDA Addon for Voicemeeter using the Remote API"
|
||||
authors = [
|
||||
{name = "Onyx and Iris", email = "code@onyxandiris.online"},
|
||||
]
|
||||
authors = [{ name = "Onyx and Iris", email = "code@onyxandiris.online" }]
|
||||
dependencies = []
|
||||
requires-python = "==3.11.*"
|
||||
readme = "README.md"
|
||||
license = { text = "MIT" }
|
||||
|
||||
[dependency-groups]
|
||||
build = [
|
||||
"scons>=4.8.1",
|
||||
"markdown>=3.7",
|
||||
]
|
||||
build = ["scons>=4.8.1", "markdown>=3.7"]
|
||||
|
||||
[tool.pdm]
|
||||
distribution = false
|
||||
|
||||
[tool.pdm.scripts]
|
||||
copy = "pwsh build.ps1"
|
||||
build = "pwsh build.ps1 -build"
|
||||
copy = "task copy"
|
||||
release = "task build"
|
||||
|
||||
[tool.ruff]
|
||||
exclude = [
|
||||
@@ -51,14 +46,16 @@ exclude = [
|
||||
line-length = 120
|
||||
indent-width = 4
|
||||
|
||||
# Assume Python 3.10
|
||||
target-version = "py310"
|
||||
# Assume Python 3.11
|
||||
target-version = "py311"
|
||||
|
||||
[tool.ruff.lint]
|
||||
# Enable Pyflakes (`F`) and a subset of the pycodestyle (`E`) codes by default.
|
||||
# Enable flake8-errmsg (EM) warnings.
|
||||
# Enable flake8-bugbear (B) warnings.
|
||||
# Unlike Flake8, Ruff doesn't enable pycodestyle warnings (`W`) or
|
||||
# McCabe complexity (`C901`) by default.
|
||||
select = ["E4", "E7", "E9", "F"]
|
||||
select = ["E4", "E7", "E9", "EM", "F", "B"]
|
||||
ignore = []
|
||||
|
||||
# Allow fix for all enabled rules (when `--fix`) is provided.
|
||||
@@ -100,7 +97,4 @@ docstring-code-line-length = "dynamic"
|
||||
max-complexity = 10
|
||||
|
||||
[tool.ruff.lint.per-file-ignores]
|
||||
"__init__.py" = [
|
||||
"E402",
|
||||
"F401",
|
||||
]
|
||||
"__init__.py" = ["E402", "F401"]
|
||||
|
||||
Reference in New Issue
Block a user