EscSeqReq.cs 3.6 KB

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