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
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
use crate::*;
use std::{ops::{Range, RangeBounds, RangeFrom, RangeInclusive, RangeTo, RangeToInclusive}, str::FromStr};



/// Internal utility function
pub fn read_range<T, R>(range: &R, prompt: Option<String>, default: Option<T>, format: fn(&R) -> String) -> BoxResult<T>
where
	T: Display + FromStr + PartialOrd<T>,
	R: RangeBounds<T>,
	<T as FromStr>::Err: Display,
{
	let mut prompt = match prompt {
		Some(v) => v,
		None => format!("Enter a number within the range {}: ", format(range)),
	};
	if let Some(default) = default.as_ref() {
		prompt += &format!(" (default: {default})");
	}
	loop {
		
		print!("{prompt}");
		let output_string = read_stdin()?;
		if output_string.is_empty() && let Some(default) = default {
			return Ok(default);
		}
		
		let output = match output_string.parse::<T>() {
			Ok(v) => v,
			Err(err) => {
				println!();
				println!("Could not parse input (error: {err})");
				continue;
			}
		};
		if range.contains(&output) {
			return Ok(output);
		}
		
		println!();
		println!("Invalid input, not within bounds");
	}
}



impl<T> TryRead for Range<T>
where
	T: Display + FromStr + PartialOrd<T>,
	<T as FromStr>::Err: Display,
{
	type Output = T;
	fn try_read_line(&self, prompt: Option<String>, default: Option<Self::Output>) -> BoxResult<Self::Output> {
		fn format(range: &Range<impl Display>) -> String {
			format!("[{:.1}, {:.1})", range.start, range.end)
		}
		read_range(self, prompt, default, format)
	}
}

impl<T> TryRead for RangeInclusive<T>
where
	T: Display + FromStr + PartialOrd<T>,
	<T as FromStr>::Err: Display,
{
	type Output = T;
	fn try_read_line(&self, prompt: Option<String>, default: Option<Self::Output>) -> BoxResult<Self::Output> {
		fn format(range: &RangeInclusive<impl Display>) -> String {
			format!("[{:.1}, {:.1}]", range.start(), range.end())
		}
		read_range(self, prompt, default, format)
	}
}

impl<T> TryRead for RangeTo<T>
where
	T: Display + FromStr + PartialOrd<T>,
	<T as FromStr>::Err: Display,
{
	type Output = T;
	fn try_read_line(&self, prompt: Option<String>, default: Option<Self::Output>) -> BoxResult<Self::Output> {
		fn format(range: &RangeTo<impl Display>) -> String {
			format!(".., {:.1})", range.end)
		}
		read_range(self, prompt, default, format)
	}
}

impl<T> TryRead for RangeFrom<T>
where
	T: Display + FromStr + PartialOrd<T>,
	<T as FromStr>::Err: Display,
{
	type Output = T;
	fn try_read_line(&self, prompt: Option<String>, default: Option<Self::Output>) -> BoxResult<Self::Output> {
		fn format(range: &RangeFrom<impl Display>) -> String {
			format!("[{:.1}, ..", range.start)
		}
		read_range(self, prompt, default, format)
	}
}

impl<T> TryRead for RangeToInclusive<T>
where
	T: Display + FromStr + PartialOrd<T>,
	<T as FromStr>::Err: Display,
{
	type Output = T;
	fn try_read_line(&self, prompt: Option<String>, default: Option<Self::Output>) -> BoxResult<Self::Output> {
		fn format(range: &RangeToInclusive<impl Display>) -> String {
			format!(".., {:.1}]", range.end)
		}
		read_range(self, prompt, default, format)
	}
}