2024-10-02 18:18:00 +00:00
|
|
|
using System;
|
2024-10-05 17:38:48 +00:00
|
|
|
using System.Collections.Immutable;
|
|
|
|
using System.Collections.ObjectModel;
|
2024-10-02 18:18:00 +00:00
|
|
|
using System.ComponentModel;
|
2024-10-05 17:38:48 +00:00
|
|
|
using System.Reactive;
|
2024-10-02 18:18:00 +00:00
|
|
|
using System.Reactive.Linq;
|
|
|
|
using System.Text.RegularExpressions;
|
2024-10-05 17:38:48 +00:00
|
|
|
using Avalonia.Controls.Platform;
|
2024-10-02 18:18:00 +00:00
|
|
|
using DynamicData;
|
|
|
|
using DynamicData.Binding;
|
2024-10-05 17:38:48 +00:00
|
|
|
using ReactiveUI;
|
2024-10-02 18:18:00 +00:00
|
|
|
using Splat;
|
|
|
|
using StarsAssistant.Model;
|
|
|
|
using StarsAssistant.ViewModels;
|
|
|
|
|
|
|
|
namespace StarsAssistant.Services;
|
|
|
|
|
2024-10-05 17:38:48 +00:00
|
|
|
public class FleetManager : IEnableLogger
|
2024-10-02 18:18:00 +00:00
|
|
|
{
|
|
|
|
protected Services.Game Game = Locator.Current.GetService<Services.Game>()!;
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// SourceCache for DynamicData views
|
|
|
|
/// </summary>
|
2024-10-05 16:47:53 +00:00
|
|
|
private SourceList<Fleet> _fleets = new();
|
2024-10-02 18:18:00 +00:00
|
|
|
|
|
|
|
public FleetManager() {}
|
|
|
|
|
2024-10-04 17:05:23 +00:00
|
|
|
public class FleetSummary
|
|
|
|
{
|
|
|
|
public string Destination { get; set; } = string.Empty;
|
|
|
|
public int TotalIronium { get; set; }
|
|
|
|
public int TotalBoranium { get; set; }
|
|
|
|
public int TotalGermanium { get; set; }
|
|
|
|
public int TotalColonists { get; set; }
|
2024-10-05 17:38:48 +00:00
|
|
|
|
|
|
|
public override string ToString() => $"To {Destination}: {TotalIronium}I/{TotalBoranium}B/{TotalGermanium}G/{TotalColonists}C";
|
2024-10-04 17:05:23 +00:00
|
|
|
}
|
2024-10-08 17:26:34 +00:00
|
|
|
|
2024-10-05 17:38:48 +00:00
|
|
|
private IObservable<IChangeSet<FleetSummary>>? _fleetSummaries;
|
|
|
|
|
|
|
|
private ReadOnlyObservableCollection<FleetSummary>? summaries;
|
|
|
|
|
|
|
|
private IDisposable? d1;
|
|
|
|
|
|
|
|
private IDisposable? d2;
|
|
|
|
|
2024-10-04 17:05:23 +00:00
|
|
|
|
|
|
|
public void test()
|
|
|
|
{
|
2024-10-05 17:38:48 +00:00
|
|
|
_fleetSummaries = _fleets.Connect()
|
|
|
|
.Filter(fleet => fleet.TrueDestination != "-- ")
|
2024-10-05 16:47:53 +00:00
|
|
|
.GroupOn(fleet => fleet.TrueDestination)
|
2024-10-05 17:38:48 +00:00
|
|
|
.Log(this, $"{DateTime.Now.ToLongTimeString()} fleetWatcher", grp => $"{grp.TotalChanges} detected")
|
|
|
|
.Transform(group => new FleetSummary
|
|
|
|
{
|
|
|
|
Destination = group.GroupKey,
|
|
|
|
TotalIronium = group.List.Items.Sum(f => f.Ironium),
|
|
|
|
TotalBoranium = group.List.Items.Sum(f => f.Boranium),
|
|
|
|
TotalGermanium = group.List.Items.Sum(f => f.Germanium),
|
|
|
|
TotalColonists = group.List.Items.Sum(f => f.Colonists)
|
|
|
|
})
|
|
|
|
;
|
2024-10-08 17:26:34 +00:00
|
|
|
|
|
|
|
// Demo only
|
|
|
|
var sourceCache = new SourceCache<FleetSummary, string>(fs => fs.Destination);
|
|
|
|
var tmp = _fleetSummaries
|
|
|
|
.AddKey(fs => fs.Destination)
|
|
|
|
.PopulateInto(sourceCache);
|
2024-10-05 17:38:48 +00:00
|
|
|
|
2024-10-08 17:26:34 +00:00
|
|
|
|
2024-10-05 17:38:48 +00:00
|
|
|
_fleetSummaries
|
|
|
|
.ObserveOn(RxApp.MainThreadScheduler)
|
|
|
|
.Bind(out summaries)
|
|
|
|
.DisposeMany()
|
|
|
|
.Subscribe();
|
|
|
|
|
|
|
|
d1 = summaries
|
|
|
|
.Subscribe(Observer.Create<FleetSummary>(f =>
|
2024-10-05 16:47:53 +00:00
|
|
|
{
|
2024-10-05 17:38:48 +00:00
|
|
|
this.Log().Debug($"FleetSummary observed: {f}");
|
|
|
|
}
|
|
|
|
));
|
|
|
|
|
|
|
|
d2 = _fleetSummaries
|
|
|
|
.ObserveOn(RxApp.MainThreadScheduler)
|
|
|
|
.Subscribe(x =>
|
|
|
|
{
|
|
|
|
var lst = x.ToList();
|
|
|
|
foreach (var f in lst)
|
|
|
|
{
|
|
|
|
this.Log().Debug($"Reason {f.Reason}, Type {f.Type}: {f.Item.Current}");
|
|
|
|
}
|
|
|
|
}
|
2024-10-05 16:47:53 +00:00
|
|
|
);
|
2024-10-04 17:05:23 +00:00
|
|
|
}
|
|
|
|
|
2024-10-02 18:18:00 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Load the fleet records from the database and push them into our source cache.
|
|
|
|
/// If the data has been freshly imported, call PostProcessImportedData first.
|
|
|
|
/// </summary>
|
|
|
|
public void InitFromDatabase()
|
|
|
|
{
|
|
|
|
using var db = Locator.Current.GetService<StarsDatabase>()!;
|
|
|
|
var allFleets = db.Fleet.ToList();
|
2024-10-04 17:05:23 +00:00
|
|
|
_fleets.Edit(innerCache =>
|
|
|
|
{
|
|
|
|
innerCache.Clear();
|
2024-10-05 16:47:53 +00:00
|
|
|
innerCache.Add(allFleets);
|
2024-10-04 17:05:23 +00:00
|
|
|
}
|
|
|
|
);
|
2024-10-02 18:18:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Helper to fill up missing data from the original import. Tries to deduce
|
|
|
|
/// missing properties using heuristics.
|
|
|
|
/// </summary>
|
|
|
|
public static void PostProcessImportedData()
|
|
|
|
{
|
|
|
|
using var db = Locator.Current.GetService<StarsDatabase>()!;
|
|
|
|
|
|
|
|
// Check for all cases where we're targeting another fleet instead of
|
|
|
|
// a planet. Update the DB accordingly.
|
|
|
|
var playerFleets = from flt in db.Fleet
|
|
|
|
where flt.OwnerFileId == Game.Player.PlayerFileId
|
|
|
|
select flt;
|
|
|
|
|
|
|
|
Regex shipPattern = new ($@"^{Regex.Escape(Game.Player.Name)} (?<ShipType>.+) #\d+$");
|
|
|
|
|
|
|
|
foreach (Fleet flt in playerFleets)
|
|
|
|
{
|
|
|
|
if (flt.Destination != "-- ")
|
|
|
|
{
|
|
|
|
var trueDest = db.Fleet
|
|
|
|
.FirstOrDefault(f => f.FleetName == $"{Game.Player.Name} {flt.Destination}"
|
|
|
|
&& f.Planet != String.Empty);
|
|
|
|
flt.TrueDestination = trueDest?.Planet ?? flt.Destination;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
flt.TrueDestination = "-- ";
|
|
|
|
}
|
|
|
|
|
|
|
|
Match m = shipPattern.Match(flt.FleetName);
|
|
|
|
if (m.Success)
|
|
|
|
flt.ShipTypeGuess = m.Groups[1].Value;
|
|
|
|
|
|
|
|
db.Update(flt);
|
|
|
|
}
|
|
|
|
|
|
|
|
db.SaveChanges();
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|