blob: 1c0820a903b887b5c08445496702d87b364e8d42 (
plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
|
// file : libbutl/prompt.cxx -*- C++ -*-
// license : MIT; see accompanying LICENSE file
#ifndef __cpp_modules_ts
#include <libbutl/prompt.mxx>
#endif
#ifndef __cpp_lib_modules_ts
#include <string>
#include <iostream>
#endif
// Other includes.
#ifdef __cpp_modules_ts
module butl.prompt;
// Only imports additional to interface.
#ifdef __clang__
#ifdef __cpp_lib_modules_ts
import std.core;
import std.io;
#endif
#endif
import butl.diagnostics;
#else
#include <libbutl/diagnostics.mxx> // diag_stream
#endif
using namespace std;
namespace butl
{
bool
yn_prompt (const string& prompt, char def)
{
// Writing a robust Y/N prompt is more difficult than one would expect.
//
string a;
do
{
*diag_stream << prompt << ' ';
// getline() will set the failbit if it failed to extract anything,
// not even the delimiter and eofbit if it reached eof before seeing
// the delimiter.
//
getline (cin, a);
bool f (cin.fail ());
bool e (cin.eof ());
if (f || e)
*diag_stream << endl; // Assume no delimiter (newline).
if (f)
throw ios_base::failure ("unable to read y/n answer from stdout");
if (a.empty () && def != '\0')
{
// Don't treat eof as the default answer. We need to see the actual
// newline.
//
if (!e)
a = def;
}
} while (a != "y" && a != "n");
return a == "y";
}
}
|