2
0
mirror of https://github.com/9ParsonsB/Pulsar.git synced 2025-04-05 17:39:39 -04:00

Add colony distance support to Botanist grid and notification (#69)

Fixes Xjph/ObservatoryCore/67.

Tested via read-all and realtime while collecting a sample. 

Required a bit of refactoring to use discrete classes for the main data structure vs. tuples. I think  worked out nicely though.
This commit is contained in:
F K 2022-03-23 06:47:34 -04:00 committed by GitHub
parent fcf6f71abf
commit a6408814f7
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23

View File

@ -14,19 +14,29 @@ namespace Observatory.Botanist
{ {
private IObservatoryCore Core; private IObservatoryCore Core;
private bool OdysseyLoaded = false; private bool OdysseyLoaded = false;
private Dictionary private Dictionary<BodyAddress, BioPlanetDetail> BioPlanets;
<
( // Note: This only explicitly contains Odyssey bios. Everything else is assumed to be 100.
ulong systemAddress, // (This is partly because names for the genus for legacy/Horizons bios are somewhat inconsistent.)
int bodyID private readonly Dictionary<String, int> ColonyDistancesByGenus = new() {
), { "Aleoida", 150 },
( { "Bacterium", 500 },
string bodyName, { "Cactoida", 300 },
int bioTotal, { "Clypeus", 150 },
List<string> speciesFound, { "Concha", 150 },
List<string> speciesAnalysed { "Electricae", 1000 },
) { "Fonticulua", 500 },
> BioPlanets; { "Frutexa", 150 },
{ "Fumerola", 100 },
{ "Fungoida", 300 },
{ "Osseus", 800 },
{ "Recepta", 150 },
{ "Stratum", 500 },
{ "Tubus", 800 },
{ "Tussock", 200 },
};
private const int DEFAULT_COLONY_DISTANCE = 100;
ObservableCollection<object> GridCollection; ObservableCollection<object> GridCollection;
private PluginUI pluginUI; private PluginUI pluginUI;
private Guid? samplerStatusNotification = null; private Guid? samplerStatusNotification = null;
@ -53,7 +63,11 @@ namespace Observatory.Botanist
break; break;
case SAASignalsFound signalsFound: case SAASignalsFound signalsFound:
{ {
var systemBodyId = (signalsFound.SystemAddress, signalsFound.BodyID); BodyAddress systemBodyId = new()
{
SystemAddress = signalsFound.SystemAddress,
BodyID = signalsFound.BodyID
};
if (OdysseyLoaded && !BioPlanets.ContainsKey(systemBodyId)) if (OdysseyLoaded && !BioPlanets.ContainsKey(systemBodyId))
{ {
var bioSignals = from signal in signalsFound.Signals var bioSignals = from signal in signalsFound.Signals
@ -62,36 +76,40 @@ namespace Observatory.Botanist
if (bioSignals.Any()) if (bioSignals.Any())
{ {
if (!BioPlanets.ContainsKey(systemBodyId)) BioPlanets.Add(
{ systemBodyId,
BioPlanets.Add( new() {
systemBodyId, BodyName = signalsFound.BodyName,
(signalsFound.BodyName, bioSignals.First().Count, new List<string>(), new List<string>()) BioTotal = bioSignals.First().Count,
); SpeciesFound = new()
} }
else );
{
var bioPlanet = BioPlanets[systemBodyId];
bioPlanet.bodyName = signalsFound.BodyName;
bioPlanet.bioTotal = bioSignals.First().Count;
}
} }
} }
} }
break; break;
case ScanOrganic scanOrganic: case ScanOrganic scanOrganic:
{ {
var systemBodyId = (scanOrganic.SystemAddress, scanOrganic.Body); BodyAddress systemBodyId = new()
{
SystemAddress = scanOrganic.SystemAddress,
BodyID = scanOrganic.Body
};
if (!BioPlanets.ContainsKey(systemBodyId)) if (!BioPlanets.ContainsKey(systemBodyId))
{ {
// Unlikely to ever end up in here, but just in case create a new planet entry. // Unlikely to ever end up in here, but just in case create a new planet entry.
List<string> genus = new(); Dictionary<string, BioSampleDetail> bioSampleDetails = new();
List<string> species = new(); bioSampleDetails.Add(scanOrganic.Species_Localised, new()
genus.Add(scanOrganic.Genus_Localised); {
species.Add(scanOrganic.Species_Localised); Genus = scanOrganic.Genus_Localised,
var bioPlanet = (string.Empty, 0, genus, species); Analysed = false
BioPlanets.Add(systemBodyId, bioPlanet); });
BioPlanets.Add(systemBodyId, new() {
BodyName = string.Empty,
BioTotal = 0,
SpeciesFound = bioSampleDetails
});
} }
else else
{ {
@ -103,10 +121,12 @@ namespace Observatory.Botanist
case ScanOrganicType.Sample: case ScanOrganicType.Sample:
if (!Core.IsLogMonitorBatchReading && botanistSettings.OverlayEnabled) if (!Core.IsLogMonitorBatchReading && botanistSettings.OverlayEnabled)
{ {
var colonyDistance = GetColonyDistance(scanOrganic);
var sampleNum = scanOrganic.ScanType == ScanOrganicType.Log ? 1 : 2;
NotificationArgs args = new() NotificationArgs args = new()
{ {
Title = scanOrganic.Species_Localised, Title = scanOrganic.Species_Localised,
Detail = string.Format("Sample {0} of 3", scanOrganic.ScanType == ScanOrganicType.Log ? 1 : 2), Detail = $"Sample {sampleNum} of 3{Environment.NewLine}Colony distance: {colonyDistance} m",
Rendering = NotificationRendering.NativeVisual, Rendering = NotificationRendering.NativeVisual,
Timeout = 0, Timeout = 0,
}; };
@ -120,15 +140,19 @@ namespace Observatory.Botanist
} }
} }
if (!bioPlanet.speciesFound.Contains(scanOrganic.Species_Localised)) if (!bioPlanet.SpeciesFound.ContainsKey(scanOrganic.Species_Localised))
{ {
bioPlanet.speciesFound.Add(scanOrganic.Species_Localised); bioPlanet.SpeciesFound.Add(scanOrganic.Species_Localised, new()
{
Genus = scanOrganic.Genus_Localised,
Analysed = false
});
} }
break; break;
case ScanOrganicType.Analyse: case ScanOrganicType.Analyse:
if (!bioPlanet.speciesAnalysed.Contains(scanOrganic.Species_Localised)) if (!bioPlanet.SpeciesFound[scanOrganic.Species_Localised].Analysed)
{ {
bioPlanet.speciesAnalysed.Add(scanOrganic.Species_Localised); bioPlanet.SpeciesFound[scanOrganic.Species_Localised].Analysed = true;
} }
MaybeCloseSamplerStatusNotification(); MaybeCloseSamplerStatusNotification();
break; break;
@ -148,6 +172,23 @@ namespace Observatory.Botanist
} }
} }
private object GetColonyDistance(ScanOrganic scan)
{
// Try find a genus name.
string genusName = null;
if (scan.Genus_Localised != null)
{
genusName = scan.Genus_Localised;
}
else if (scan.Genus == "$Codex_Ent_Clepeus_Genus_Name;")
{
// Odyssey had a bug until Update 9 where Clypeus ScanOrganic entries were missing the Genus_Localised property.
genusName = "Clypeus";
}
return (genusName != null && ColonyDistancesByGenus.ContainsKey(genusName)) ? ColonyDistancesByGenus[genusName] : DEFAULT_COLONY_DISTANCE;
}
private void MaybeCloseSamplerStatusNotification() private void MaybeCloseSamplerStatusNotification()
{ {
if (samplerStatusNotification != null) if (samplerStatusNotification != null)
@ -193,38 +234,93 @@ namespace Observatory.Botanist
Core.ClearGrid(this, uiObject); Core.ClearGrid(this, uiObject);
foreach (var bioPlanet in BioPlanets.Values) foreach (var bioPlanet in BioPlanets.Values)
{ {
if (bioPlanet.speciesFound.Count == 0) if (bioPlanet.SpeciesFound.Count == 0)
{ {
var planetRow = new BotanistGrid() var planetRow = new BotanistGrid()
{ {
Body = bioPlanet.bodyName, Body = bioPlanet.BodyName,
BioTotal = bioPlanet.bioTotal.ToString(), BioTotal = bioPlanet.BioTotal.ToString(),
Species = "(NO SAMPLES TAKEN)", Species = "(NO SAMPLES TAKEN)",
Analysed = string.Empty Analysed = string.Empty,
ColonyDistance = string.Empty,
}; };
Core.AddGridItem(this, planetRow); Core.AddGridItem(this, planetRow);
} }
for (int i = 0; i < bioPlanet.speciesFound.Count; i++) bool firstRow = true;
foreach (var entry in bioPlanet.SpeciesFound)
{ {
int colonyDistance = DEFAULT_COLONY_DISTANCE;
if (entry.Value.Genus != null && ColonyDistancesByGenus.ContainsKey(entry.Value.Genus))
{
colonyDistance = ColonyDistancesByGenus[entry.Value.Genus];
}
var speciesRow = new BotanistGrid() var speciesRow = new BotanistGrid()
{ {
Body = i == 0 ? bioPlanet.bodyName : string.Empty, Body = firstRow ? bioPlanet.BodyName : string.Empty,
BioTotal = i == 0 ? bioPlanet.bioTotal.ToString() : string.Empty, BioTotal = firstRow ? bioPlanet.BioTotal.ToString() : string.Empty,
Species = bioPlanet.speciesFound[i], Species = entry.Key,
Analysed = bioPlanet.speciesAnalysed.Contains(bioPlanet.speciesFound[i]) ? "✓" : "" Analysed = entry.Value.Analysed ? "✓" : "",
ColonyDistance = $"{colonyDistance}m",
}; };
Core.AddGridItem(this, speciesRow); Core.AddGridItem(this, speciesRow);
firstRow = false;
} }
} }
} }
} }
class BodyAddress
{
public ulong SystemAddress { get; set; }
public int BodyID { get; set; }
public override bool Equals(object obj)
{
// We want value equality here.
//
// See the full list of guidelines at
// http://go.microsoft.com/fwlink/?LinkID=85237
// and also the guidance for operator== at
// http://go.microsoft.com/fwlink/?LinkId=85238
//
if (obj == null || GetType() != obj.GetType())
{
return false;
}
BodyAddress other = (BodyAddress)obj;
return other.SystemAddress == SystemAddress
&& other.BodyID == BodyID;
}
public override int GetHashCode()
{
return HashCode.Combine(SystemAddress, BodyID);
}
}
class BioPlanetDetail
{
public string BodyName { get; set; }
public int BioTotal { get; set; }
public Dictionary<string, BioSampleDetail> SpeciesFound { get; set; }
}
class BioSampleDetail
{
public string Genus { get; set; }
public bool Analysed { get; set; }
}
public class BotanistGrid public class BotanistGrid
{ {
public string Body { get; set; } public string Body { get; set; }
public string BioTotal { get; set; } public string BioTotal { get; set; }
public string Species { get; set; } public string Species { get; set; }
public string Analysed { get; set; } public string Analysed { get; set; }
public string ColonyDistance { get; set; }
} }
} }