134 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			134 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using FFXIVClientStructs.FFXIV.Client.Game.Character;
 | |
| using MareSynchronos.API.Data.Enum;
 | |
| using MareSynchronos.Factories;
 | |
| using MareSynchronos.Mediator;
 | |
| using MareSynchronos.Models;
 | |
| using MareSynchronos.Utils;
 | |
| 
 | |
| namespace MareSynchronos.Managers;
 | |
| 
 | |
| public class CacheCreationService : MediatorSubscriberBase, IDisposable
 | |
| {
 | |
|     private readonly CharacterDataFactory _characterDataFactory;
 | |
|     private Task? _cacheCreationTask;
 | |
|     private readonly Dictionary<ObjectKind, GameObjectHandler> _cachesToCreate = new();
 | |
|     private readonly CharacterData _lastCreatedData = new();
 | |
|     private readonly CancellationTokenSource _cts = new();
 | |
|     private readonly List<GameObjectHandler> _playerRelatedObjects = new();
 | |
|     private CancellationTokenSource _palettePlusCts = new();
 | |
| 
 | |
|     public unsafe CacheCreationService(MareMediator mediator, CharacterDataFactory characterDataFactory, DalamudUtil dalamudUtil) : base(mediator)
 | |
|     {
 | |
|         _characterDataFactory = characterDataFactory;
 | |
| 
 | |
|         Mediator.Subscribe<CreateCacheForObjectMessage>(this, (msg) =>
 | |
|         {
 | |
|             var actualMsg = (CreateCacheForObjectMessage)msg;
 | |
|             _cachesToCreate[actualMsg.ObjectToCreateFor.ObjectKind] = actualMsg.ObjectToCreateFor;
 | |
|         });
 | |
|         Mediator.Subscribe<ClearCacheForObjectMessage>(this, (msg) =>
 | |
|         {
 | |
|             var actualMsg = (ClearCacheForObjectMessage)msg;
 | |
|             _lastCreatedData.FileReplacements.Remove(actualMsg.ObjectToCreateFor.ObjectKind);
 | |
|             _lastCreatedData.GlamourerString.Remove(actualMsg.ObjectToCreateFor.ObjectKind);
 | |
|             Mediator.Publish(new CharacterDataCreatedMessage(_lastCreatedData));
 | |
|         });
 | |
|         Mediator.Subscribe<FrameworkUpdateMessage>(this, (msg) => UpdatePointers());
 | |
|         Mediator.Subscribe<DelayedFrameworkUpdateMessage>(this, (msg) => ProcessCacheCreation());
 | |
|         Mediator.Subscribe<CustomizePlusMessage>(this, (msg) => CustomizePlusChanged((CustomizePlusMessage)msg));
 | |
|         Mediator.Subscribe<HeelsOffsetMessage>(this, (msg) => HeelsOffsetChanged((HeelsOffsetMessage)msg));
 | |
|         Mediator.Subscribe<PalettePlusMessage>(this, (msg) => PalettePlusChanged((PalettePlusMessage)msg));
 | |
|         Mediator.Subscribe<PenumbraModSettingChangedMessage>(this, (msg) => _cachesToCreate[ObjectKind.Player] = _playerRelatedObjects.First(p => p.ObjectKind == ObjectKind.Player));
 | |
| 
 | |
|         _playerRelatedObjects.AddRange(new List<GameObjectHandler>()
 | |
|         {
 | |
|             new(Mediator, ObjectKind.Player, () => dalamudUtil.PlayerPointer),
 | |
|             new(Mediator, ObjectKind.MinionOrMount, () => (IntPtr)((Character*)dalamudUtil.PlayerPointer)->CompanionObject),
 | |
|             new(Mediator, ObjectKind.Pet, () => dalamudUtil.GetPet()),
 | |
|             new(Mediator, ObjectKind.Companion, () => dalamudUtil.GetCompanion()),
 | |
|         });
 | |
|     }
 | |
| 
 | |
|     private void PalettePlusChanged(PalettePlusMessage msg)
 | |
|     {
 | |
|         if (!string.Equals(msg.Data, _lastCreatedData.PalettePlusPalette, StringComparison.Ordinal))
 | |
|         {
 | |
|             _lastCreatedData.PalettePlusPalette = msg.Data ?? string.Empty;
 | |
| 
 | |
|             _palettePlusCts?.Cancel();
 | |
|             _palettePlusCts?.Dispose();
 | |
|             _palettePlusCts = new();
 | |
|             var token = _palettePlusCts.Token;
 | |
| 
 | |
|             Task.Run(async () =>
 | |
|             {
 | |
|                 await Task.Delay(TimeSpan.FromSeconds(1), token).ConfigureAwait(false);
 | |
|                 Mediator.Publish(new CharacterDataCreatedMessage(_lastCreatedData));
 | |
|             }, token);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     private void HeelsOffsetChanged(HeelsOffsetMessage msg)
 | |
|     {
 | |
|         if (msg.Offset != _lastCreatedData.HeelsOffset)
 | |
|         {
 | |
|             _lastCreatedData.HeelsOffset = msg.Offset;
 | |
|             Mediator.Publish(new CharacterDataCreatedMessage(_lastCreatedData));
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     private void CustomizePlusChanged(CustomizePlusMessage msg)
 | |
|     {
 | |
|         if (!string.Equals(msg.Data, _lastCreatedData.CustomizePlusScale, StringComparison.Ordinal))
 | |
|         {
 | |
|             _lastCreatedData.CustomizePlusScale = msg.Data ?? string.Empty;
 | |
|             Mediator.Publish(new CharacterDataCreatedMessage(_lastCreatedData));
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     private void UpdatePointers()
 | |
|     {
 | |
|         Mediator.Publish(new PlayerRelatedObjectPointerUpdateMessage(_playerRelatedObjects.Select(f => f.CurrentAddress).ToArray()));
 | |
|     }
 | |
| 
 | |
|     private void ProcessCacheCreation()
 | |
|     {
 | |
|         if (_cachesToCreate.Any() && (_cacheCreationTask?.IsCompleted ?? true))
 | |
|         {
 | |
|             var toCreate = _cachesToCreate.ToList();
 | |
|             _cachesToCreate.Clear();
 | |
|             _cacheCreationTask = Task.Run(async () =>
 | |
|             {
 | |
|                 try
 | |
|                 {
 | |
|                     foreach (var obj in toCreate)
 | |
|                     {
 | |
|                         var data = await _characterDataFactory.BuildCharacterData(_lastCreatedData, obj.Value, _cts.Token).ConfigureAwait(false);
 | |
|                     }
 | |
|                     Mediator.Publish(new CharacterDataCreatedMessage(_lastCreatedData));
 | |
|                 }
 | |
|                 catch (Exception ex)
 | |
|                 {
 | |
|                     Logger.Error("Error during Cache Creation Processing", ex);
 | |
|                 }
 | |
|                 finally
 | |
|                 {
 | |
|                     Logger.Debug("Cache Creation complete");
 | |
| 
 | |
|                 }
 | |
|             }, _cts.Token);
 | |
|         }
 | |
|         else if (_cachesToCreate.Any())
 | |
|         {
 | |
|             Logger.Debug("Cache Creation stored until previous creation finished");
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     public override void Dispose()
 | |
|     {
 | |
|         base.Dispose();
 | |
|         _playerRelatedObjects.ForEach(p => p.Dispose());
 | |
|         _cts.Dispose();
 | |
|     }
 | |
| }
 | 
