Car/Assets/StompyRobot/SRDebugger/Scripts/Services/Implementation/StandardConsoleService.cs

211 lines
5.9 KiB
C#
Raw Permalink Normal View History

2025-01-02 11:32:58 +08:00
using System;
using System.Collections.Generic;
namespace SRDebugger.Services.Implementation
{
using SRF.Service;
using UnityEngine;
[Service(typeof (IConsoleService))]
public class StandardConsoleService : IConsoleService, IDisposable
{
private readonly bool _collapseEnabled;
private bool _hasCleared;
private readonly CircularBuffer<ConsoleEntry> _allConsoleEntries;
private CircularBuffer<ConsoleEntry> _consoleEntries;
private readonly object _threadLock = new object();
private ILogHandler _expectedLogHandler;
public StandardConsoleService()
{
Application.logMessageReceivedThreaded += UnityLogCallback;
_expectedLogHandler = Debug.unityLogger.logHandler;
SRServiceManager.RegisterService<IConsoleService>(this);
_collapseEnabled = Settings.Instance.CollapseDuplicateLogEntries;
_allConsoleEntries = new CircularBuffer<ConsoleEntry>(Settings.Instance.MaximumConsoleEntries);
}
public void Dispose()
{
Application.logMessageReceivedThreaded -= UnityLogCallback;
if (_consoleEntries != null)
{
_consoleEntries.Clear();
}
_allConsoleEntries.Clear();
}
public int ErrorCount { get; private set; }
public int WarningCount { get; private set; }
public int InfoCount { get; private set; }
public event ConsoleUpdatedEventHandler Updated;
public event ConsoleUpdatedEventHandler Error;
public bool LoggingEnabled
{
get { return Debug.unityLogger.logEnabled; }
set { Debug.unityLogger.logEnabled = value; }
}
public bool LogHandlerIsOverriden
{
get
{
return Debug.unityLogger.logHandler != _expectedLogHandler;
}
}
public IReadOnlyList<ConsoleEntry> Entries
{
get
{
if (!_hasCleared)
{
return _allConsoleEntries;
}
return _consoleEntries;
}
}
public IReadOnlyList<ConsoleEntry> AllEntries
{
get { return _allConsoleEntries; }
}
public void Clear()
{
lock (_threadLock)
{
_hasCleared = true;
if (_consoleEntries == null)
{
_consoleEntries = new CircularBuffer<ConsoleEntry>(Settings.Instance.MaximumConsoleEntries);
}
else
{
_consoleEntries.Clear();
}
ErrorCount = WarningCount = InfoCount = 0;
}
OnUpdated();
}
protected void OnEntryAdded(ConsoleEntry entry)
{
if (_hasCleared)
{
// Decrement counters if adding this entry will push another
// entry from the buffer.
if (_consoleEntries.IsFull)
{
AdjustCounter(_consoleEntries.Front().LogType, -1);
_consoleEntries.PopFront();
}
_consoleEntries.PushBack(entry);
}
else
{
if (_allConsoleEntries.IsFull)
{
AdjustCounter(_allConsoleEntries.Front().LogType, -1);
_allConsoleEntries.PopFront();
}
}
_allConsoleEntries.PushBack(entry);
OnUpdated();
}
protected void OnEntryDuplicated(ConsoleEntry entry)
{
entry.Count++;
OnUpdated();
// If has cleared, add this entry again for the current list
if (_hasCleared && _consoleEntries.Count == 0)
{
OnEntryAdded(new ConsoleEntry(entry) {Count = 1});
}
}
private void OnUpdated()
{
if (Updated != null)
{
try
{
Updated(this);
}
catch {}
}
}
private void UnityLogCallback(string condition, string stackTrace, LogType type)
{
//if (condition.StartsWith("[SRConsole]"))
// return;
lock (_threadLock)
{
var prevMessage = _collapseEnabled && _allConsoleEntries.Count > 0
? _allConsoleEntries[_allConsoleEntries.Count - 1]
: null;
AdjustCounter(type, 1);
if (prevMessage != null && prevMessage.LogType == type && prevMessage.Message == condition &&
prevMessage.StackTrace == stackTrace)
{
OnEntryDuplicated(prevMessage);
}
else
{
var newEntry = new ConsoleEntry
{
LogType = type,
StackTrace = stackTrace,
Message = condition
};
OnEntryAdded(newEntry);
}
}
}
private void AdjustCounter(LogType type, int amount)
{
switch (type)
{
case LogType.Assert:
case LogType.Error:
case LogType.Exception:
ErrorCount += amount;
if (Error != null)
{
Error.Invoke(this);
}
break;
case LogType.Warning:
WarningCount += amount;
break;
case LogType.Log:
InfoCount += amount;
break;
}
}
}
}