Added the powershell script so it will be available for generated scripts.
This commit is contained in:
parent
9e73910121
commit
d10312c0d2
231
DoomTTS.ps1
Normal file
231
DoomTTS.ps1
Normal file
@ -0,0 +1,231 @@
|
||||
# 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
|
||||
}
|
||||
}
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user