mirror of
https://github.com/FlightControl-Master/MOOSE.git
synced 2025-08-15 10:47:21 +00:00
**AIRBOSS** - Stennis now takes dimensions of other super carriers **FLIGHTCONTROL** - Added queue for sounds - Fixed bugs and minor improvements **ZONE** - Added defaults when getting colors if not set **SRS** - Added new class `MSRSQUEUE`
259 lines
8.3 KiB
Lua
259 lines
8.3 KiB
Lua
--- **Utilities** DCS Simple Text-To-Speech (STTS).
|
|
--
|
|
--
|
|
--
|
|
-- @module Utils.STTS
|
|
-- @image MOOSE.JPG
|
|
|
|
--- [DCS Enum world](https://wiki.hoggitworld.com/view/DCS_enum_world)
|
|
-- @type STTS
|
|
-- @field #string DIRECTORY Path of the SRS directory.
|
|
|
|
--- Simple Text-To-Speech
|
|
--
|
|
-- Version 0.4 - Compatible with SRS version 1.9.6.0+
|
|
--
|
|
-- # DCS Modification Required
|
|
--
|
|
-- You will need to edit MissionScripting.lua in DCS World/Scripts/MissionScripting.lua and remove the sanitisation.
|
|
-- To do this remove all the code below the comment - the line starts "local function sanitizeModule(name)"
|
|
-- Do this without DCS running to allow mission scripts to use os functions.
|
|
--
|
|
-- *You WILL HAVE TO REAPPLY AFTER EVERY DCS UPDATE*
|
|
--
|
|
-- # USAGE:
|
|
--
|
|
-- Add this script into the mission as a DO SCRIPT or DO SCRIPT FROM FILE to initialise it
|
|
-- Make sure to edit the STTS.SRS_PORT and STTS.DIRECTORY to the correct values before adding to the mission.
|
|
-- Then its as simple as calling the correct function in LUA as a DO SCRIPT or in your own scripts.
|
|
--
|
|
-- Example calls:
|
|
--
|
|
-- STTS.TextToSpeech("Hello DCS WORLD","251","AM","1.0","SRS",2)
|
|
--
|
|
-- Arguments in order are:
|
|
--
|
|
-- * Message to say, make sure not to use a newline (\n) !
|
|
-- * Frequency in MHz
|
|
-- * Modulation - AM/FM
|
|
-- * Volume - 1.0 max, 0.5 half
|
|
-- * Name of the transmitter - ATC, RockFM etc
|
|
-- * Coalition - 0 spectator, 1 red 2 blue
|
|
-- * OPTIONAL - Vec3 Point i.e Unit.getByName("A UNIT"):getPoint() - needs Vec3 for Height! OR null if not needed
|
|
-- * OPTIONAL - Speed -10 to +10
|
|
-- * OPTIONAL - Gender male, female or neuter
|
|
-- * OPTIONAL - Culture - en-US, en-GB etc
|
|
-- * OPTIONAL - Voice - a specfic voice by name. Run DCS-SR-ExternalAudio.exe with --help to get the ones you can use on the command line
|
|
-- * OPTIONAL - Google TTS - Switch to Google Text To Speech - Requires STTS.GOOGLE_CREDENTIALS path and Google project setup correctly
|
|
--
|
|
--
|
|
-- ## Example
|
|
--
|
|
-- This example will say the words "Hello DCS WORLD" on 251 MHz AM at maximum volume with a client called SRS and to the Blue coalition only
|
|
--
|
|
-- STTS.TextToSpeech("Hello DCS WORLD","251","AM","1.0","SRS",2,null,-5,"male","en-GB")
|
|
--
|
|
-- ## Example
|
|
--
|
|
--This example will say the words "Hello DCS WORLD" on 251 MHz AM at maximum volume with a client called SRS and to the Blue coalition only centered on the position of the Unit called "A UNIT"
|
|
--
|
|
-- STTS.TextToSpeech("Hello DCS WORLD","251","AM","1.0","SRS",2,Unit.getByName("A UNIT"):getPoint(),-5,"male","en-GB")
|
|
--
|
|
-- Arguments in order are:
|
|
--
|
|
-- * FULL path to the MP3 OR OGG to play
|
|
-- * Frequency in MHz - to use multiple separate with a comma - Number of frequencies MUST match number of Modulations
|
|
-- * Modulation - AM/FM - to use multiple
|
|
-- * Volume - 1.0 max, 0.5 half
|
|
-- * Name of the transmitter - ATC, RockFM etc
|
|
-- * Coalition - 0 spectator, 1 red 2 blue
|
|
--
|
|
-- ## Example
|
|
--
|
|
-- This will play that MP3 on 255MHz AM & 31 FM at half volume with a client called "Multiple" and to Spectators only
|
|
--
|
|
-- STTS.PlayMP3("C:\\Users\\Ciaran\\Downloads\\PR-Music.mp3","255,31","AM,FM","0.5","Multiple",0)
|
|
--
|
|
-- @field #STTS
|
|
STTS={
|
|
ClassName="STTS",
|
|
DIRECTORY="",
|
|
SRS_PORT=5002,
|
|
GOOGLE_CREDENTIALS="C:\\Users\\Ciaran\\Downloads\\googletts.json",
|
|
EXECUTABLE="DCS-SR-ExternalAudio.exe",
|
|
}
|
|
|
|
--- FULL Path to the FOLDER containing DCS-SR-ExternalAudio.exe - EDIT TO CORRECT FOLDER
|
|
STTS.DIRECTORY = "D:/DCS/_SRS"
|
|
|
|
--- LOCAL SRS PORT - DEFAULT IS 5002
|
|
STTS.SRS_PORT = 5002
|
|
|
|
--- Google credentials file
|
|
STTS.GOOGLE_CREDENTIALS = "C:\\Users\\Ciaran\\Downloads\\googletts.json"
|
|
|
|
--- DONT CHANGE THIS UNLESS YOU KNOW WHAT YOU'RE DOING
|
|
STTS.EXECUTABLE = "DCS-SR-ExternalAudio.exe"
|
|
|
|
|
|
--- Function for UUID.
|
|
function STTS.uuid()
|
|
local random = math.random
|
|
local template ='yxxx-xxxxxxxxxxxx'
|
|
return string.gsub(template, '[xy]', function (c)
|
|
local v = (c == 'x') and random(0, 0xf) or random(8, 0xb)
|
|
return string.format('%x', v)
|
|
end)
|
|
end
|
|
|
|
--- Round a number.
|
|
-- @param #number x Number.
|
|
-- @param #number n Precision.
|
|
function STTS.round(x, n)
|
|
n = math.pow(10, n or 0)
|
|
x = x * n
|
|
if x >= 0 then x = math.floor(x + 0.5) else x = math.ceil(x - 0.5) end
|
|
return x / n
|
|
end
|
|
|
|
--- Function returns estimated speech time in seconds.
|
|
-- Assumptions for time calc: 100 Words per min, avarage of 5 letters for english word so
|
|
--
|
|
-- * 5 chars * 100wpm = 500 characters per min = 8.3 chars per second
|
|
--
|
|
-- So lengh of msg / 8.3 = number of seconds needed to read it. rounded down to 8 chars per sec map function:
|
|
--
|
|
-- * (x - in_min) * (out_max - out_min) / (in_max - in_min) + out_min
|
|
--
|
|
-- @param #number length can also be passed as #string
|
|
-- @param #number speed Defaults to 1.0
|
|
-- @param #boolean isGoogle We're using Google TTS
|
|
function STTS.getSpeechTime(length,speed,isGoogle)
|
|
|
|
local maxRateRatio = 3
|
|
|
|
speed = speed or 1.0
|
|
isGoogle = isGoogle or false
|
|
|
|
local speedFactor = 1.0
|
|
if isGoogle then
|
|
speedFactor = speed
|
|
else
|
|
if speed ~= 0 then
|
|
speedFactor = math.abs(speed) * (maxRateRatio - 1) / 10 + 1
|
|
end
|
|
if speed < 0 then
|
|
speedFactor = 1/speedFactor
|
|
end
|
|
end
|
|
|
|
local wpm = math.ceil(100 * speedFactor)
|
|
local cps = math.floor((wpm * 5)/60)
|
|
|
|
if type(length) == "string" then
|
|
length = string.len(length)
|
|
end
|
|
|
|
return length/cps --math.ceil(length/cps)
|
|
end
|
|
|
|
--- Text to speech function.
|
|
function STTS.TextToSpeech(message, freqs, modulations, volume, name, coalition, point, speed, gender, culture, voice, googleTTS)
|
|
if os == nil or io == nil then
|
|
env.info("[DCS-STTS] LUA modules os or io are sanitized. skipping. ")
|
|
return
|
|
end
|
|
|
|
speed = speed or 1
|
|
gender = gender or "female"
|
|
culture = culture or ""
|
|
voice = voice or ""
|
|
coalition=coalition or "0"
|
|
name=name or "ROBOT"
|
|
volume=1
|
|
speed=1
|
|
|
|
|
|
message = message:gsub("\"","\\\"")
|
|
|
|
local cmd = string.format("start /min \"\" /d \"%s\" /b \"%s\" -f %s -m %s -c %s -p %s -n \"%s\" -h", STTS.DIRECTORY, STTS.EXECUTABLE, freqs or "305", modulations or "AM", coalition, STTS.SRS_PORT, name)
|
|
|
|
if voice ~= "" then
|
|
cmd = cmd .. string.format(" -V \"%s\"",voice)
|
|
else
|
|
|
|
if culture ~= "" then
|
|
cmd = cmd .. string.format(" -l %s",culture)
|
|
end
|
|
|
|
if gender ~= "" then
|
|
cmd = cmd .. string.format(" -g %s",gender)
|
|
end
|
|
end
|
|
|
|
if googleTTS == true then
|
|
cmd = cmd .. string.format(" -G \"%s\"",STTS.GOOGLE_CREDENTIALS)
|
|
end
|
|
|
|
if speed ~= 1 then
|
|
cmd = cmd .. string.format(" -s %s",speed)
|
|
end
|
|
|
|
if volume ~= 1.0 then
|
|
cmd = cmd .. string.format(" -v %s",volume)
|
|
end
|
|
|
|
if point and type(point) == "table" and point.x then
|
|
local lat, lon, alt = coord.LOtoLL(point)
|
|
|
|
lat = STTS.round(lat,4)
|
|
lon = STTS.round(lon,4)
|
|
alt = math.floor(alt)
|
|
|
|
cmd = cmd .. string.format(" -L %s -O %s -A %s",lat,lon,alt)
|
|
end
|
|
|
|
cmd = cmd ..string.format(" -t \"%s\"",message)
|
|
|
|
if string.len(cmd) > 255 then
|
|
local filename = os.getenv('TMP') .. "\\DCS_STTS-" .. STTS.uuid() .. ".bat"
|
|
local script = io.open(filename,"w+")
|
|
script:write(cmd .. " && exit" )
|
|
script:close()
|
|
cmd = string.format("\"%s\"",filename)
|
|
timer.scheduleFunction(os.remove, filename, timer.getTime() + 1)
|
|
end
|
|
|
|
if string.len(cmd) > 255 then
|
|
env.info("[DCS-STTS] - cmd string too long")
|
|
env.info("[DCS-STTS] TextToSpeech Command :\n" .. cmd.."\n")
|
|
end
|
|
os.execute(cmd)
|
|
|
|
return STTS.getSpeechTime(message,speed,googleTTS)
|
|
end
|
|
|
|
--- Play mp3 function.
|
|
-- @param #string pathToMP3 Path to the sound file.
|
|
-- @param #string freqs Frequencies, e.g. "305, 256".
|
|
-- @param #string modulations Modulations, e.g. "AM, FM".
|
|
-- @param #string volume Volume, e.g. "0.5".
|
|
function STTS.PlayMP3(pathToMP3, freqs, modulations, volume, name, coalition, point)
|
|
|
|
local cmd = string.format("start \"\" /d \"%s\" /b /min \"%s\" -i \"%s\" -f %s -m %s -c %s -p %s -n \"%s\" -v %s -h",
|
|
STTS.DIRECTORY, STTS.EXECUTABLE, pathToMP3, freqs or "305", modulations or "AM", coalition or "0", STTS.SRS_PORT, name or "ROBOT", volume or "1")
|
|
|
|
if point and type(point) == "table" and point.x then
|
|
local lat, lon, alt = coord.LOtoLL(point)
|
|
|
|
lat = STTS.round(lat,4)
|
|
lon = STTS.round(lon,4)
|
|
alt = math.floor(alt)
|
|
|
|
cmd = cmd .. string.format(" -L %s -O %s -A %s",lat,lon,alt)
|
|
end
|
|
|
|
env.info("[DCS-STTS] MP3/OGG Command :\n" .. cmd.."\n")
|
|
os.execute(cmd)
|
|
|
|
end |