EscSeqReq.cs 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101
  1. #nullable enable
  2. using System.Collections.Concurrent;
  3. namespace Terminal.Gui;
  4. /// <summary>
  5. /// Represents the status of an ANSI escape sequence request made to the terminal using
  6. /// <see cref="EscSeqRequests"/>.
  7. /// </summary>
  8. /// <remarks></remarks>
  9. public class EscSeqReqStatus
  10. {
  11. /// <summary>Creates a new state of escape sequence request.</summary>
  12. /// <param name="ansiRequest">The <see cref="AnsiEscapeSequenceRequest"/> object.</param>
  13. public EscSeqReqStatus (AnsiEscapeSequenceRequest ansiRequest) { AnsiRequest = ansiRequest; }
  14. /// <summary>Gets the Escape Sequence Terminator (e.g. ESC[8t ... t is the terminator).</summary>
  15. public AnsiEscapeSequenceRequest AnsiRequest { get; }
  16. }
  17. // TODO: This class is a singleton. It should use the singleton pattern.
  18. /// <summary>
  19. /// Manages ANSI Escape Sequence requests and responses. The list of <see cref="EscSeqReqStatus"/> contains the
  20. /// status of the request. Each request is identified by the terminator (e.g. ESC[8t ... t is the terminator).
  21. /// </summary>
  22. public class EscSeqRequests
  23. {
  24. /// <summary>
  25. /// Adds a new request for the ANSI Escape Sequence defined by <paramref name="ansiRequest"/>. Adds a
  26. /// <see cref="EscSeqReqStatus"/> instance to <see cref="Statuses"/> list.
  27. /// </summary>
  28. /// <param name="ansiRequest">The <see cref="AnsiEscapeSequenceRequest"/> object.</param>
  29. /// <param name="driver">The driver in use.</param>
  30. public void Add (AnsiEscapeSequenceRequest ansiRequest, ConsoleDriver? driver = null)
  31. {
  32. lock (Statuses)
  33. {
  34. Statuses.Enqueue (new (ansiRequest));
  35. if (driver is null)
  36. {
  37. Console.Out.Write (ansiRequest.Request);
  38. Console.Out.Flush ();
  39. }
  40. else
  41. {
  42. driver.WriteRaw (ansiRequest.Request);
  43. }
  44. }
  45. }
  46. /// <summary>
  47. /// Indicates if a <see cref="EscSeqReqStatus"/> with the <paramref name="terminator"/> exists in the
  48. /// <see cref="Statuses"/> list.
  49. /// </summary>
  50. /// <param name="terminator"></param>
  51. /// <param name="seqReqStatus"></param>
  52. /// <returns><see langword="true"/> if exist, <see langword="false"/> otherwise.</returns>
  53. public bool HasResponse (string terminator, out EscSeqReqStatus? seqReqStatus)
  54. {
  55. lock (Statuses)
  56. {
  57. Statuses.TryPeek (out seqReqStatus);
  58. var result = seqReqStatus?.AnsiRequest.Terminator == terminator;
  59. if (result)
  60. {
  61. return true;
  62. }
  63. seqReqStatus = null;
  64. return false;
  65. }
  66. }
  67. /// <summary>
  68. /// Removes a request defined by <paramref name="seqReqStatus"/>. If a matching <see cref="EscSeqReqStatus"/> is
  69. /// found and the number of outstanding requests is greater than 0, the number of outstanding requests is decremented.
  70. /// If the number of outstanding requests is 0, the <see cref="EscSeqReqStatus"/> is removed from
  71. /// <see cref="Statuses"/>.
  72. /// </summary>
  73. /// <param name="seqReqStatus">The <see cref="EscSeqReqStatus"/> object.</param>
  74. public void Remove (EscSeqReqStatus? seqReqStatus)
  75. {
  76. lock (Statuses)
  77. {
  78. Statuses.TryDequeue (out var request);
  79. if (request != seqReqStatus)
  80. {
  81. throw new InvalidOperationException ("Both EscSeqReqStatus objects aren't equals.");
  82. }
  83. }
  84. }
  85. /// <summary>Gets the <see cref="EscSeqReqStatus"/> list.</summary>
  86. public ConcurrentQueue<EscSeqReqStatus> Statuses { get; } = new ();
  87. }