222 lines
7.8 KiB
C#
222 lines
7.8 KiB
C#
|
#nullable enable
|
|||
|
|
|||
|
using System;
|
|||
|
using alterNERDtive.util;
|
|||
|
|
|||
|
namespace RatAttack
|
|||
|
{
|
|||
|
public class RatAttack
|
|||
|
{
|
|||
|
private static RatCase[] CaseList { get; } = new RatCase[20];
|
|||
|
private static dynamic? VA { get; set; }
|
|||
|
private static alterNERDtive.util.PipeServer<Ratsignal> RatsignalPipe
|
|||
|
=> ratsignalPipe ??= new alterNERDtive.util.PipeServer<Ratsignal>(Log, "RatAttack",
|
|||
|
new alterNERDtive.util.PipeServer<Ratsignal>.SignalHandler(On_Ratsignal));
|
|||
|
private static alterNERDtive.util.PipeServer<Ratsignal>? ratsignalPipe;
|
|||
|
|
|||
|
private static VoiceAttackLog Log
|
|||
|
=> log ??= new VoiceAttackLog(VA, "RatAttack");
|
|||
|
private static VoiceAttackLog? log;
|
|||
|
|
|||
|
private static VoiceAttackCommands Commands
|
|||
|
=> commands ??= new VoiceAttackCommands(VA, Log);
|
|||
|
private static VoiceAttackCommands? commands;
|
|||
|
|
|||
|
private class RatCase
|
|||
|
{
|
|||
|
public string Cmdr;
|
|||
|
public string System;
|
|||
|
public string Platform;
|
|||
|
public bool CodeRed;
|
|||
|
public int Number;
|
|||
|
|
|||
|
public RatCase(string cmdr, string system, string platform, bool codeRed, int number)
|
|||
|
=> (Cmdr, System, Platform, CodeRed, Number) = (cmdr, system, platform, codeRed, number);
|
|||
|
|
|||
|
public string ShortInfo
|
|||
|
{
|
|||
|
get
|
|||
|
{
|
|||
|
string type = CodeRed ? ", code red" : "";
|
|||
|
return ($"#{Number}, {Platform}, {System}{type}");
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
public override string ToString()
|
|||
|
{
|
|||
|
return ShortInfo;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
public class Ratsignal : IPipable
|
|||
|
{
|
|||
|
public string Signal { get; set; }
|
|||
|
public bool Announce { get; set; }
|
|||
|
|
|||
|
public Ratsignal()
|
|||
|
=> (Signal, Announce) = ("", false);
|
|||
|
|
|||
|
public Ratsignal(string signal, bool announce)
|
|||
|
=> (Signal, Announce) = (signal, announce);
|
|||
|
|
|||
|
public void ParseString(string serialization)
|
|||
|
{
|
|||
|
try
|
|||
|
{
|
|||
|
string[] parts = serialization.Split('|');
|
|||
|
Signal = parts[0];
|
|||
|
Announce = Boolean.Parse(parts[1]);
|
|||
|
}
|
|||
|
catch (Exception e)
|
|||
|
{
|
|||
|
throw new ArgumentException($"Invalid serialized RATSIGNAL: '{serialization}'", e);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
public override string ToString()
|
|||
|
=> $"{Signal}|{Announce}";
|
|||
|
}
|
|||
|
|
|||
|
private static int ParseRatsignal(string ratsignal)
|
|||
|
{
|
|||
|
char delimiter = (char)0x02;
|
|||
|
string[] parts = ratsignal.Split(delimiter);
|
|||
|
|
|||
|
try
|
|||
|
{
|
|||
|
string cmdr = parts[1];
|
|||
|
string system = parts[3];
|
|||
|
string platform = parts[5];
|
|||
|
if (platform.Contains("PS4")) { platform = "PS4"; }
|
|||
|
else if (platform.Contains("XB")) { platform = "XB"; }
|
|||
|
bool codeRed = parts.Length == 9;
|
|||
|
int start = ratsignal.LastIndexOf('#') + 1;
|
|||
|
int end = ratsignal.LastIndexOf(") (");
|
|||
|
int number = int.Parse(ratsignal.Substring(start, end - start));
|
|||
|
|
|||
|
CaseList[number] = new RatCase(cmdr, system, platform, codeRed, number);
|
|||
|
|
|||
|
return number;
|
|||
|
}
|
|||
|
catch(Exception e)
|
|||
|
{
|
|||
|
throw new ArgumentException($"Invalid RATSIGNAL format: '{ratsignal}'.", "ratsignal", e);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
private static void On_Ratsignal(Ratsignal ratsignal)
|
|||
|
{
|
|||
|
try
|
|||
|
{
|
|||
|
int number = ParseRatsignal(ratsignal.Signal);
|
|||
|
Log.Notice($"New rat case: {CaseList[number]}.");
|
|||
|
Commands.TriggerEvent("RatAttack.incomingCase", parameters: new dynamic[] { new int[] { number }, new bool[] { ratsignal.Announce } });
|
|||
|
}
|
|||
|
catch (ArgumentException e)
|
|||
|
{
|
|||
|
Log.Error(e.Message);
|
|||
|
}
|
|||
|
catch (Exception e)
|
|||
|
{
|
|||
|
Log.Error($"Unhandled exception while parsing RATSIGNAL: '{e.Message}'.");
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
/*================\
|
|||
|
| plugin contexts |
|
|||
|
\================*/
|
|||
|
|
|||
|
private static void Context_GetCaseData(dynamic vaProxy)
|
|||
|
{
|
|||
|
int cn = vaProxy.GetInt("~caseNumber");
|
|||
|
RatCase rc = CaseList[cn];
|
|||
|
|
|||
|
vaProxy.SetInt("~~caseNumber", rc?.Number);
|
|||
|
vaProxy.SetText("~~cmdr", rc?.Cmdr);
|
|||
|
vaProxy.SetText("~~system", rc?.System);
|
|||
|
vaProxy.SetText("~~platform", rc?.Platform);
|
|||
|
vaProxy.SetBoolean("~~codeRed", rc?.CodeRed);
|
|||
|
}
|
|||
|
|
|||
|
private static void Context_Startup(dynamic vaProxy)
|
|||
|
{
|
|||
|
Log.Notice("Starting up …");
|
|||
|
VA = vaProxy;
|
|||
|
_ = RatsignalPipe.Run();
|
|||
|
Log.Notice("Finished startup.");
|
|||
|
}
|
|||
|
|
|||
|
private static void Context_ParseRatsignal(dynamic vaProxy)
|
|||
|
{
|
|||
|
Log.Warn("Passing a RATSIGNAL from VoiceAttack (through the clipboard or a file) is DEPRECATED and will no longer be supported in the future.");
|
|||
|
On_Ratsignal(new Ratsignal(vaProxy.GetText("~ratsignal"), vaProxy.GetBoolean("~announceRatsignal")));
|
|||
|
}
|
|||
|
|
|||
|
/*========================================\
|
|||
|
| required VoiceAttack plugin shenanigans |
|
|||
|
\========================================*/
|
|||
|
|
|||
|
static readonly string VERSION = "0.0.1";
|
|||
|
|
|||
|
public static Guid VA_Id()
|
|||
|
=> new Guid("{F2ADF0AE-4837-4E4A-9C87-8A7E2FA63E5F}");
|
|||
|
public static string VA_DisplayName()
|
|||
|
=> $"RatAttack {VERSION}";
|
|||
|
public static string VA_DisplayInfo()
|
|||
|
=> "RatAttack: a plugin to handle FuelRats cases.";
|
|||
|
|
|||
|
public static void VA_Init1(dynamic vaProxy)
|
|||
|
{
|
|||
|
VA = vaProxy;
|
|||
|
Log.Notice("Initializing …");
|
|||
|
Log.Notice("Init successful.");
|
|||
|
}
|
|||
|
|
|||
|
public static void VA_Invoke1(dynamic vaProxy)
|
|||
|
{
|
|||
|
string context = vaProxy.Context.ToLower();
|
|||
|
Log.Debug($"Running context '{context}' …");
|
|||
|
try
|
|||
|
{
|
|||
|
switch (context)
|
|||
|
{
|
|||
|
// plugin methods
|
|||
|
case "getcasedata":
|
|||
|
Context_GetCaseData(vaProxy);
|
|||
|
break;
|
|||
|
case "parseratsignal":
|
|||
|
Context_ParseRatsignal(vaProxy);
|
|||
|
break;
|
|||
|
case "startup":
|
|||
|
Context_Startup(vaProxy);
|
|||
|
break;
|
|||
|
// plugin settings
|
|||
|
// NYI
|
|||
|
// invalid
|
|||
|
default:
|
|||
|
Log.Error($"Invalid plugin context '{vaProxy.Context}'.");
|
|||
|
break;
|
|||
|
}
|
|||
|
}
|
|||
|
catch (ArgumentNullException e)
|
|||
|
{
|
|||
|
Log.Error($"Missing parameter '~{e.ParamName}' for context '{context}'");
|
|||
|
}
|
|||
|
catch (Exception e)
|
|||
|
{
|
|||
|
Log.Error($"Unhandled exception while executing plugin context '{context}'. ({e.Message})");
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
public static void VA_Exit1(dynamic vaProxy)
|
|||
|
{
|
|||
|
Log.Debug("Starting teardown …");
|
|||
|
Log.Debug("Closing RATSIGNAL pipe …");
|
|||
|
RatsignalPipe!.Stop();
|
|||
|
Log.Debug("Teardown finished.");
|
|||
|
}
|
|||
|
|
|||
|
public static void VA_StopCommand() { }
|
|||
|
}
|
|||
|
}
|