2021-09-04 12:01:03 +00:00
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Threading;
|
|
|
|
|
|
|
|
|
|
namespace Tapeti.Cmd.ConsoleHelper
|
|
|
|
|
{
|
|
|
|
|
public class ConsoleWrapper : IConsole
|
|
|
|
|
{
|
|
|
|
|
private readonly List<TemporaryWriter> temporaryWriters = new();
|
|
|
|
|
private bool temporaryActive;
|
|
|
|
|
|
|
|
|
|
private int temporaryCursorTop;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public ConsoleWrapper()
|
|
|
|
|
{
|
|
|
|
|
temporaryCursorTop = Console.CursorTop;
|
|
|
|
|
|
|
|
|
|
Console.CancelKeyPress += (_, args) =>
|
|
|
|
|
{
|
|
|
|
|
if (Cancelled)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
using var consoleWriter = GetPermanentWriter();
|
|
|
|
|
consoleWriter.WriteLine("Cancelling...");
|
|
|
|
|
|
|
|
|
|
args.Cancel = true;
|
|
|
|
|
Cancelled = true;
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public void Dispose()
|
|
|
|
|
{
|
|
|
|
|
foreach (var writer in temporaryWriters)
|
|
|
|
|
writer.Dispose();
|
|
|
|
|
|
|
|
|
|
Console.CursorVisible = true;
|
|
|
|
|
GC.SuppressFinalize(this);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public bool Cancelled { get; private set; }
|
|
|
|
|
|
|
|
|
|
public IConsoleWriter GetPermanentWriter()
|
|
|
|
|
{
|
|
|
|
|
return new PermanentWriter(this);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public IConsoleWriter GetTemporaryWriter()
|
|
|
|
|
{
|
|
|
|
|
var writer = new TemporaryWriter(this, temporaryWriters.Count);
|
|
|
|
|
temporaryWriters.Add(writer);
|
|
|
|
|
|
|
|
|
|
return writer;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
private void AcquirePermanent()
|
|
|
|
|
{
|
|
|
|
|
if (!temporaryActive)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
foreach (var writer in temporaryWriters)
|
|
|
|
|
{
|
|
|
|
|
Console.SetCursorPosition(0, temporaryCursorTop + writer.RelativePosition);
|
|
|
|
|
writer.Clear();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Console.SetCursorPosition(0, temporaryCursorTop);
|
|
|
|
|
Console.CursorVisible = true;
|
|
|
|
|
temporaryActive = false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
private void ReleasePermanent()
|
|
|
|
|
{
|
|
|
|
|
if (temporaryWriters.Count == 0)
|
|
|
|
|
{
|
|
|
|
|
temporaryCursorTop = Console.CursorTop;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
foreach (var writer in temporaryWriters)
|
|
|
|
|
{
|
|
|
|
|
writer.Restore();
|
|
|
|
|
Console.WriteLine();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Store the cursor position afterwards to account for buffer scrolling
|
|
|
|
|
temporaryCursorTop = Console.CursorTop - temporaryWriters.Count;
|
|
|
|
|
Console.CursorVisible = false;
|
|
|
|
|
temporaryActive = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
private void AcquireTemporary(TemporaryWriter writer)
|
|
|
|
|
{
|
|
|
|
|
Console.SetCursorPosition(0, temporaryCursorTop + writer.RelativePosition);
|
|
|
|
|
|
|
|
|
|
if (temporaryActive)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
Console.CursorVisible = false;
|
|
|
|
|
temporaryActive = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
private void DisposeWriter(BaseWriter writer)
|
|
|
|
|
{
|
|
|
|
|
if (writer is not TemporaryWriter temporaryWriter)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
Console.SetCursorPosition(0, temporaryCursorTop + temporaryWriter.RelativePosition);
|
|
|
|
|
temporaryWriter.Clear();
|
|
|
|
|
|
|
|
|
|
temporaryWriters.Remove(temporaryWriter);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
private abstract class BaseWriter : IConsoleWriter
|
|
|
|
|
{
|
|
|
|
|
protected readonly ConsoleWrapper Owner;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
protected BaseWriter(ConsoleWrapper owner)
|
|
|
|
|
{
|
|
|
|
|
Owner = owner;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public virtual void Dispose()
|
|
|
|
|
{
|
|
|
|
|
Owner.DisposeWriter(this);
|
|
|
|
|
GC.SuppressFinalize(this);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public abstract bool Enabled { get; }
|
|
|
|
|
|
|
|
|
|
public abstract void WriteLine(string value);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public void Confirm(string message)
|
|
|
|
|
{
|
|
|
|
|
WriteLine(message);
|
2021-09-05 08:05:04 +00:00
|
|
|
|
TryReadKey(false, out _);
|
2021-09-04 12:01:03 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public bool ConfirmYesNo(string message)
|
|
|
|
|
{
|
|
|
|
|
WriteLine($"{message} (Y/N) ");
|
|
|
|
|
if (!TryReadKey(true, out var key))
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
return key.KeyChar == 'y' || key.KeyChar == 'Y';
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
private bool TryReadKey(bool showKeyOutput, out ConsoleKeyInfo keyInfo)
|
|
|
|
|
{
|
|
|
|
|
while (!Owner.Cancelled && !Console.KeyAvailable)
|
|
|
|
|
Thread.Sleep(50);
|
|
|
|
|
|
|
|
|
|
if (Owner.Cancelled)
|
|
|
|
|
{
|
|
|
|
|
keyInfo = default;
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
keyInfo = Console.ReadKey(!showKeyOutput);
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
private class PermanentWriter : BaseWriter
|
|
|
|
|
{
|
|
|
|
|
public PermanentWriter(ConsoleWrapper owner) : base(owner)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public override bool Enabled => true;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public override void WriteLine(string value)
|
|
|
|
|
{
|
|
|
|
|
Owner.AcquirePermanent();
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
Console.WriteLine(value);
|
|
|
|
|
}
|
|
|
|
|
finally
|
|
|
|
|
{
|
|
|
|
|
Owner.ReleasePermanent();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
private class TemporaryWriter : BaseWriter
|
|
|
|
|
{
|
|
|
|
|
public int RelativePosition { get; }
|
|
|
|
|
|
|
|
|
|
private bool isActive;
|
|
|
|
|
private string storedValue;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public TemporaryWriter(ConsoleWrapper owner, int relativePosition) : base(owner)
|
|
|
|
|
{
|
|
|
|
|
RelativePosition = relativePosition;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public override bool Enabled => !Console.IsOutputRedirected;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public override void WriteLine(string value)
|
|
|
|
|
{
|
|
|
|
|
if (!Enabled)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
Owner.AcquireTemporary(this);
|
|
|
|
|
Console.Write(value);
|
|
|
|
|
|
|
|
|
|
if (!string.IsNullOrEmpty(storedValue) && storedValue.Length > value.Length)
|
|
|
|
|
// Clear characters remaining from the previous value
|
|
|
|
|
Console.Write(new string(' ', storedValue.Length - value.Length));
|
|
|
|
|
|
|
|
|
|
storedValue = value;
|
|
|
|
|
isActive = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public void Clear()
|
|
|
|
|
{
|
|
|
|
|
if (!isActive)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
if (!string.IsNullOrEmpty(storedValue))
|
|
|
|
|
Console.Write(new string(' ', storedValue.Length));
|
|
|
|
|
|
|
|
|
|
isActive = false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public void Restore()
|
|
|
|
|
{
|
|
|
|
|
if (string.IsNullOrEmpty(storedValue))
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
Console.Write(storedValue);
|
|
|
|
|
isActive = true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|