232 lines
8.5 KiB
PowerShell
232 lines
8.5 KiB
PowerShell
# Change the next line to $true to enable the script
|
|
$useTextToSpeech = $true
|
|
# Set this to $true to use SAPI, $false to use nvdaControllerClient or fall back to clipboard
|
|
$useSAPI = $false
|
|
# Set the speech rate. -10 to 10, default 0
|
|
$speechRate = 0
|
|
|
|
# Do not edit below this line.
|
|
######################################################################
|
|
|
|
# Exit if text-to-speech is disabled
|
|
if (-not $useTextToSpeech) {
|
|
exit
|
|
}
|
|
|
|
# Define antigrep, and sed patterns
|
|
$antigrepPatterns = @(
|
|
'^P_StartScript:',
|
|
'^[Ff]luidsynth:',
|
|
'^(Facing|INTRO|MAP[0-9]+|README)',
|
|
'^ *TITLEMAP',
|
|
'key card',
|
|
'^\[Toby Accessibility Mod\] (INTRO|READMe)([0-9]+).*',
|
|
"^(As |Computer Voice:|I |I've|Monorail|Ugh|What|Where)",
|
|
[regex]::Escape('Ugh... Huh? What the hell was that?! Better go check it out...')
|
|
)
|
|
|
|
$sedPatterns = @(
|
|
@("\[Toby Accessibility Mod\] M_", "[Toby Accessibility Mod] "),
|
|
@("^\[Toby Accessibility Mod\] ", ""),
|
|
@("^MessageBoxMenu$", "Confirmation menu: Press Y for yes or N for no"),
|
|
@("^Mainmenu$", "Main menu"),
|
|
@("^Playerclassmenu$", "Player class menu"),
|
|
@("^Skillmenu$", "Difficulty menu"),
|
|
@("^NGAME", "New game"),
|
|
@("^LOADG$", "Load game"),
|
|
@("^SAVEG$", "Save game"),
|
|
@("^QUITG$", "Quit game"),
|
|
@('"cl_run" = "true"', "Run"),
|
|
@('"cl_run" = "false"', "Walk"),
|
|
@('.*/:Game saved. \(', ""),
|
|
@('^\*\*\*', ""),
|
|
@('^\+', "")
|
|
)
|
|
|
|
# Check PowerShell version
|
|
$requiredPowershellVersion = [Version]"5.1"
|
|
$currentPowershellVersion = $PSVersionTable.PSVersion
|
|
|
|
$logFile = ".\DoomTTS.log"
|
|
Set-Content -Path $logFile -Value "Logging started $(Get-Date -Format 'dddd MMMM dd, yyyy') at $(Get-Date -Format 'hh:mmtt')"
|
|
Add-Content -Path $logFile -Value "Powershell $currentPowershellVersion"
|
|
if ($currentVersion -lt $requiredVersion) {
|
|
Add-Content -Path $logFile -Value "PowerShell version $requiredPowershellVersion or later is required. Exiting."
|
|
exit
|
|
}
|
|
|
|
# Function for logging
|
|
function Write-Log {
|
|
param (
|
|
[string]$Message,
|
|
[string]$Type = "INFO" # Can be INFO, ERROR, or SPEECH
|
|
)
|
|
|
|
$timestamp = Get-Date -Format "yyyy-MM-dd HH:mm:ss"
|
|
$logMessage = "$Message [$Type] [$timestamp]"
|
|
|
|
# Append the message to the log file
|
|
Add-Content -Path $logFile -Value $logMessage
|
|
|
|
# If it's an error, also write to the console
|
|
if ($Type -eq "ERROR") {
|
|
Write-Host $logMessage -ForegroundColor Red
|
|
}
|
|
}
|
|
|
|
# Function to load NVDA DLL and check if NVDA is running
|
|
function Initialize-NVDA {
|
|
try {
|
|
# Declare the P/Invoke signatures
|
|
$signature = @"
|
|
[DllImport("kernel32.dll")]
|
|
public static extern IntPtr LoadLibrary(string dllToLoad);
|
|
|
|
[DllImport("kernel32.dll")]
|
|
public static extern IntPtr GetProcAddress(IntPtr hModule, string procedureName);
|
|
|
|
[DllImport("kernel32.dll")]
|
|
public static extern bool FreeLibrary(IntPtr hModule);
|
|
|
|
[UnmanagedFunctionPointer(CallingConvention.StdCall)]
|
|
public delegate int NvdaTestIfRunning();
|
|
|
|
[UnmanagedFunctionPointer(CallingConvention.StdCall)]
|
|
public delegate void NvdaSpeakText([MarshalAs(UnmanagedType.LPWStr)] string text);
|
|
|
|
[UnmanagedFunctionPointer(CallingConvention.StdCall)]
|
|
public delegate void NvdaBrailleMessage([MarshalAs(UnmanagedType.LPWStr)] string message);
|
|
|
|
[UnmanagedFunctionPointer(CallingConvention.StdCall)]
|
|
public delegate void NvdaCancelSpeech();
|
|
"@
|
|
|
|
Add-Type -MemberDefinition $signature -Name "NvdaFunctions" -Namespace "Win32Functions"
|
|
|
|
# Load the NVDA client library
|
|
$dllPath = ".\nvdaControllerClient.dll"
|
|
$nvdaDll = [Win32Functions.NvdaFunctions]::LoadLibrary($dllPath)
|
|
if ($nvdaDll -eq [IntPtr]::Zero) {
|
|
throw "Failed to load nvdaControllerClient.dll"
|
|
}
|
|
|
|
# Define function pointers
|
|
$nvdaTestIfRunning = [System.Runtime.InteropServices.Marshal]::GetDelegateForFunctionPointer(
|
|
[Win32Functions.NvdaFunctions]::GetProcAddress($nvdaDll, "nvdaController_testIfRunning"),
|
|
[Type][Win32Functions.NvdaFunctions+NvdaTestIfRunning]
|
|
)
|
|
|
|
$nvdaSpeakText = [System.Runtime.InteropServices.Marshal]::GetDelegateForFunctionPointer(
|
|
[Win32Functions.NvdaFunctions]::GetProcAddress($nvdaDll, "nvdaController_speakText"),
|
|
[Type][Win32Functions.NvdaFunctions+NvdaSpeakText]
|
|
)
|
|
|
|
$nvdaBrailleMessage = [System.Runtime.InteropServices.Marshal]::GetDelegateForFunctionPointer(
|
|
[Win32Functions.NvdaFunctions]::GetProcAddress($nvdaDll, "nvdaController_brailleMessage"),
|
|
[Type][Win32Functions.NvdaFunctions+NvdaBrailleMessage]
|
|
)
|
|
|
|
$nvdaCancelSpeech = [System.Runtime.InteropServices.Marshal]::GetDelegateForFunctionPointer(
|
|
[Win32Functions.NvdaFunctions]::GetProcAddress($nvdaDll, "nvdaController_cancelSpeech"),
|
|
[Type][Win32Functions.NvdaFunctions+NvdaCancelSpeech]
|
|
)
|
|
|
|
# Test if NVDA is running
|
|
$res = $nvdaTestIfRunning.Invoke()
|
|
if ($res -ne 0) {
|
|
$errorMessage = [ComponentModel.Win32Exception]::new([System.Runtime.InteropServices.Marshal]::GetLastWin32Error()).Message
|
|
throw "NVDA is not running or communication failed. Error: $errorMessage"
|
|
}
|
|
|
|
return @{
|
|
SpeakText = $nvdaSpeakText
|
|
BrailleMessage = $nvdaBrailleMessage
|
|
CancelSpeech = $nvdaCancelSpeech
|
|
}
|
|
}
|
|
catch {
|
|
Write-Log -Message "Error initializing NVDA: $_" -Type "ERROR"
|
|
return $null
|
|
}
|
|
}
|
|
|
|
# Function to speak text using SAPI, NVDA, or copy to clipboard
|
|
function Speak-Text {
|
|
param (
|
|
[string]$text
|
|
)
|
|
|
|
Write-Log -Message "Speaking: $text" -Type "SPEECH"
|
|
|
|
if ($useSAPI) {
|
|
try {
|
|
$tts = New-Object -ComObject SAPI.SPVoice
|
|
$tts.Rate = $speechRate
|
|
$tts.Speak($text)
|
|
} catch {
|
|
Write-Log -Message "Error using SAPI: $_" -Type "ERROR"
|
|
}
|
|
} else {
|
|
$nvdaFunctions = Initialize-NVDA
|
|
if ($nvdaFunctions) {
|
|
try {
|
|
$nvdaFunctions.SpeakText.Invoke($text)
|
|
} catch {
|
|
Write-Log -Message "Error using nvdaControllerClient.dll: $_" -Type "ERROR"
|
|
Set-Clipboard -Value $text
|
|
}
|
|
} else {
|
|
Set-Clipboard -Value $text
|
|
Write-Log -Message "Failed to initialize NVDA, text copied to clipboard" -Type "INFO"
|
|
}
|
|
}
|
|
}
|
|
|
|
# Process the output with grep, antigrep, and sed-like functionality
|
|
function Process-Output {
|
|
param (
|
|
[string]$line,
|
|
[string[]]$antigrepPatterns,
|
|
[array]$sedPatterns
|
|
)
|
|
|
|
# Apply antigrep (exclude lines)
|
|
foreach ($pattern in $antigrepPatterns) {
|
|
if ($line -match $pattern) {
|
|
return # Skip this line
|
|
}
|
|
}
|
|
|
|
# Apply sed (modify lines)
|
|
foreach ($pattern in $sedPatterns) {
|
|
$line = $line -replace $pattern[0], $pattern[1]
|
|
}
|
|
|
|
return $line
|
|
}
|
|
|
|
|
|
# Start reading the piped output
|
|
$stream = [System.IO.StreamReader]::new([Console]::OpenStandardInput(), [System.Text.Encoding]::UTF8)
|
|
# Use the 40 - line to let us know when to start speaking.
|
|
$startProcessing = $false
|
|
|
|
while ($null -ne ($line = $stream.ReadLine())) {
|
|
Write-Log -Message "Raw input: $line" -Type "INFO"
|
|
|
|
# Check for the separator
|
|
if ($line -match '^-{5,}$') {
|
|
$startProcessing = $true
|
|
continue # Skip the separator
|
|
}
|
|
|
|
# Only process lines after we've seen the separator
|
|
if ($startProcessing) {
|
|
$processedLine = Process-Output -line $line -antigrepPatterns $antigrepPatterns -sedPatterns $sedPatterns
|
|
if ($processedLine) {
|
|
Write-Log -Message "Processed line: $processedLine" -Type "INFO"
|
|
Speak-Text -text $processedLine
|
|
}
|
|
}
|
|
}
|