2
0
mirror of https://github.com/9ParsonsB/Pulsar.git synced 2025-04-05 17:39:39 -04:00
F K 4f4ba88878
Add event-based LogMonitor state changes to better handle batch reads (#59)
* Add event-based LogMonitor state changes to better handle batch reads

Pre-reading hackily used read-all to suppress notifications. But that broke some assumptions about what read-all meant. Furthermore, the Core UI told plugins about read-all rather than the log monitor telling them -- which is really what should be telling them.

To address these concerns, LogMonitor now provides an event that both the PluginCore and PluginEventHandler listens to or tracking logging state allowing more granular information about the activities of LogMonitor, including distinguishing between ReadAll and Pre-read batches. Plugins no longer need to track if LogMonitor is in batch-read mode or not -- PluginCore now provides it. 

I've also converted all built-in plugins to use the new event-based system. The old system is marked deprecated and will go away once other known contributed plugins have converted to the new system.

* Change LogMonitorState enum to [Flags], drop 'None' state

As requested, and did associated simplifications and cleanup that followed.
2022-03-03 16:39:49 -03:30

231 lines
9.6 KiB
C#

using Observatory.Framework;
using Observatory.Framework.Files;
using Observatory.Framework.Files.Journal;
using Observatory.Framework.Interfaces;
using Observatory.Framework.Files.ParameterTypes;
using System.Collections.Generic;
using System.Linq;
using System;
using System.Collections.ObjectModel;
namespace Observatory.Botanist
{
public class Botanist : IObservatoryWorker
{
private IObservatoryCore Core;
private bool OdysseyLoaded = false;
private Dictionary
<
(
ulong systemAddress,
int bodyID
),
(
string bodyName,
int bioTotal,
List<string> speciesFound,
List<string> speciesAnalysed
)
> BioPlanets;
ObservableCollection<object> GridCollection;
private PluginUI pluginUI;
private Guid? samplerStatusNotification = null;
private BotanistSettings botanistSettings = new()
{
OverlayEnabled = true,
};
public string Name => "Observatory Botanist";
public string ShortName => "Botanist";
public string Version => typeof(Botanist).Assembly.GetName().Version.ToString();
public PluginUI PluginUI => pluginUI;
public object Settings { get => botanistSettings; set { botanistSettings = (BotanistSettings)value; } }
public void JournalEvent<TJournal>(TJournal journal) where TJournal : JournalBase
{
switch (journal)
{
case LoadGame loadGame:
OdysseyLoaded = loadGame.Odyssey;
break;
case SAASignalsFound signalsFound:
{
var systemBodyId = (signalsFound.SystemAddress, signalsFound.BodyID);
if (OdysseyLoaded && !BioPlanets.ContainsKey(systemBodyId))
{
var bioSignals = from signal in signalsFound.Signals
where signal.Type == "$SAA_SignalType_Biological;"
select signal;
if (bioSignals.Any())
{
if (!BioPlanets.ContainsKey(systemBodyId))
{
BioPlanets.Add(
systemBodyId,
(signalsFound.BodyName, bioSignals.First().Count, new List<string>(), new List<string>())
);
}
else
{
var bioPlanet = BioPlanets[systemBodyId];
bioPlanet.bodyName = signalsFound.BodyName;
bioPlanet.bioTotal = bioSignals.First().Count;
}
}
}
}
break;
case ScanOrganic scanOrganic:
{
var systemBodyId = (scanOrganic.SystemAddress, scanOrganic.Body);
if (!BioPlanets.ContainsKey(systemBodyId))
{
// Unlikely to ever end up in here, but just in case create a new planet entry.
List<string> genus = new();
List<string> species = new();
genus.Add(scanOrganic.Genus_Localised);
species.Add(scanOrganic.Species_Localised);
var bioPlanet = (string.Empty, 0, genus, species);
BioPlanets.Add(systemBodyId, bioPlanet);
}
else
{
var bioPlanet = BioPlanets[systemBodyId];
switch (scanOrganic.ScanType)
{
case ScanOrganicType.Log:
case ScanOrganicType.Sample:
if (!Core.IsLogMonitorBatchReading && botanistSettings.OverlayEnabled)
{
NotificationArgs args = new()
{
Title = scanOrganic.Species_Localised,
Detail = string.Format("Sample {0} of 3", scanOrganic.ScanType == ScanOrganicType.Log ? 1 : 2),
Rendering = NotificationRendering.NativeVisual,
Timeout = 0,
};
if (samplerStatusNotification == null)
{
samplerStatusNotification = Core.SendNotification(args);
}
else
{
Core.UpdateNotification(samplerStatusNotification.Value, args);
}
}
if (!bioPlanet.speciesFound.Contains(scanOrganic.Species_Localised))
{
bioPlanet.speciesFound.Add(scanOrganic.Species_Localised);
}
break;
case ScanOrganicType.Analyse:
if (!bioPlanet.speciesAnalysed.Contains(scanOrganic.Species_Localised))
{
bioPlanet.speciesAnalysed.Add(scanOrganic.Species_Localised);
}
MaybeCloseSamplerStatusNotification();
break;
}
}
UpdateUIGrid();
}
break;
case LeaveBody:
case FSDJump:
case Shutdown:
// These are all good reasons to kill any open notification. Note that SupercruiseEntry is NOT a
// suitable reason to close the notification as the player hopping out only to double check the
// DSS map for another location. Note that a game client crash will not close the status notification.
MaybeCloseSamplerStatusNotification();
break;
}
}
private void MaybeCloseSamplerStatusNotification()
{
if (samplerStatusNotification != null)
{
Core.CancelNotification(samplerStatusNotification.Value);
samplerStatusNotification = null;
}
}
public void Load(IObservatoryCore observatoryCore)
{
GridCollection = new();
BotanistGrid uiObject = new();
GridCollection.Add(uiObject);
pluginUI = new PluginUI(GridCollection);
BioPlanets = new();
Core = observatoryCore;
}
public void LogMonitorStateChanged(LogMonitorStateChangedEventArgs args)
{
if (LogMonitorStateChangedEventArgs.IsBatchRead(args.NewState))
{
// Beginning a batch read. Clear grid.
Core.ClearGrid(this, new BotanistGrid());
}
else if (LogMonitorStateChangedEventArgs.IsBatchRead(args.PreviousState))
{
// Batch read is complete. Show data.
UpdateUIGrid();
}
}
private void UpdateUIGrid()
{
// Suppress repainting the entire contents of the grid on every ScanOrganic record we read.
if (Core.IsLogMonitorBatchReading) return;
BotanistGrid uiObject = new();
Core.ClearGrid(this, uiObject);
foreach (var bioPlanet in BioPlanets.Values)
{
if (bioPlanet.speciesFound.Count == 0)
{
var planetRow = new BotanistGrid()
{
Body = bioPlanet.bodyName,
BioTotal = bioPlanet.bioTotal.ToString(),
Species = "(NO SAMPLES TAKEN)",
Analysed = string.Empty
};
Core.AddGridItem(this, planetRow);
}
for (int i = 0; i < bioPlanet.speciesFound.Count; i++)
{
var speciesRow = new BotanistGrid()
{
Body = i == 0 ? bioPlanet.bodyName : string.Empty,
BioTotal = i == 0 ? bioPlanet.bioTotal.ToString() : string.Empty,
Species = bioPlanet.speciesFound[i],
Analysed = bioPlanet.speciesAnalysed.Contains(bioPlanet.speciesFound[i]) ? "✓" : ""
};
Core.AddGridItem(this, speciesRow);
}
}
}
}
public class BotanistGrid
{
public string Body { get; set; }
public string BioTotal { get; set; }
public string Species { get; set; }
public string Analysed { get; set; }
}
}