1

ユーザーに複数の入力を求めるコンソール アプリがあります。プロンプトが表示された後、ユーザーがエスケープを押して操作をキャンセルできるようにしたいと思います。

何かのようなもの:

if (Console.ReadKey().Key != ConsoleKey.Escape) {
  string input = Console.ReadLine();
  ...
}

ただし、これの問題は、エスケープ以外のキーが押された場合、入力の一部にならないことです (から返されReadLineます)。

次のキーを「覗く」方法はありますか、それ以外の場合はこれを行いますか?

4

3 に答える 3

6
[StructLayout(LayoutKind.Sequential, CharSet = CharSet.Auto)]
internal struct InputRecord
{
  internal short eventType;
  internal KeyEventRecord keyEvent;
}
[StructLayout(LayoutKind.Sequential, CharSet = CharSet.Auto)]
internal struct KeyEventRecord
{
  internal bool keyDown;
  internal short repeatCount;
  internal short virtualKeyCode;
  internal short virtualScanCode;
  internal char uChar;
  internal int controlKeyState;
}

[DllImport("kernel32.dll", SetLastError = true)]
internal static extern bool PeekConsoleInput(IntPtr hConsoleInput, out InputRecord buffer, int numInputRecords_UseOne, out int numEventsRead);

[DllImport("kernel32.dll", SetLastError = true)]
internal static extern bool ReadConsoleInput(IntPtr hConsoleInput, out InputRecord buffer, int numInputRecords_UseOne, out int numEventsRead);

[DllImport("kernel32.dll", SetLastError = true)]
internal static extern IntPtr GetStdHandle(int nStdHandle);

static Nullable<InputRecord> PeekConsoleEvent()
{
  InputRecord ir;
  int num;
  if (!PeekConsoleInput(GetStdHandle(-10), out ir, 1, out num))
  {
    //TODO process error
  }
  if (num != 0)
    return ir;
  return null;
}
static InputRecord ReadConsoleInput()
{
  InputRecord ir;
  int num;
  if (!ReadConsoleInput(GetStdHandle(-10), out ir, 1, out num))
  {
    //TODO process error
  }
  return ir;
}
static bool PeekEscapeKey()
{
  for (; ; )
  {
    var ev = PeekConsoleEvent();
    if (ev == null)
    {
      Thread.Sleep(10);
      continue;
    }
    if (ev.Value.eventType == 1 && ev.Value.keyEvent.keyDown && ev.Value.keyEvent.virtualKeyCode == 27)
    {
      ReadConsoleInput();
      return true;
    }
    if (ev.Value.eventType == 1 && ev.Value.keyEvent.keyDown)
      return false;
    ReadConsoleInput();
  }
}

使用例:

  for (; ; )
  {
    Console.Write("?");
    if (PeekEscapeKey())
    {
      Console.WriteLine("esc");          
      continue;
    }
    Console.Write(">");
    var line = Console.ReadLine();
    Console.WriteLine(line);
  }
于 2012-04-27T15:01:49.273 に答える
1

ピークするには、TextReader.Peek を使用する必要があります。コンソールでそれを使用するには、Console.In.Peek(). 考えているコードを使用するには:

if(Console.In.Peek() != "\0x1B") // 0x1B is the Escape Key
{
        string input = Console.ReadLine();
        ....
}

これはうまくいくはずです。私はそれをテストしていませんが、正しいようです。

于 2012-04-26T22:47:12.030 に答える
1

または、@ user823682よりもわずかにエレガントではない方法の場合

ConsoleKeyInfo key = Console.ReadKey().Key;
if (key != ConsoleKey.Escape) {
  string input = String.Concat(key.KeyChar,Console.ReadLine());
  ...
}
于 2012-04-26T22:53:05.647 に答える