summaryrefslogtreecommitdiffstats
path: root/src/utils.rs
blob: 669b926459629fc1c75b2161904ac10fbc8230d1 (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
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
//! Internal only utilities
use std::io::{Error, ErrorKind, Write};
use std::str;

use unicode_width::UnicodeWidthStr;

use super::format::Alignment;

#[cfg(not(windows))]
pub static NEWLINE: &'static [u8] = b"\n";
#[cfg(windows)]
pub static NEWLINE: &'static [u8] = b"\r\n";

/// Internal utility for writing data into a string
pub struct StringWriter {
	string: String
}

impl StringWriter {
	/// Create a new `StringWriter`
	pub fn new() -> StringWriter {
		return StringWriter{string: String::new()};
	}

	/// Return a reference to the internally written `String`
	pub fn as_string(&self) -> &str {
		return &self.string;
	}
}

impl Write for StringWriter {
	fn write(&mut self, data: &[u8]) -> Result<usize, Error> {
		let string = match str::from_utf8(data) {
			Ok(s) => s,
			Err(e) => return Err(Error::new(ErrorKind::Other, format!("Cannot decode utf8 string : {}", e)))
		};
		self.string.push_str(string);
		return Ok(data.len());
	}

	fn flush(&mut self) -> Result<(), Error> {
		// Nothing to do here
		return Ok(());
	}
}

/// Align/fill a string and print it to `out`
pub fn print_align<T: Write+?Sized>(out: &mut T, align: Alignment, text: &str, fill: char, size: usize) -> Result<(), Error> {
	let text_len = UnicodeWidthStr::width(text);
	let mut nfill = if text_len < size { size - text_len } else { 0 };
	let n = match align {
		Alignment::LEFT => 0,
		Alignment:: RIGHT => nfill,
		Alignment:: CENTER => nfill/2
	};
	if n > 0 {
		try!(out.write(&vec![fill as u8; n]));
		nfill -= n;
	}
	try!(out.write(text.as_bytes()));
	if nfill > 0 {
		try!(out.write(&vec![fill as u8; nfill]));
	}
	return Ok(());
}

#[cfg(test)]
mod tests {
	use super::*;
	use format::Alignment;
	use std::io::Write;

	#[test]
	fn string_writer() {
		let mut out = StringWriter::new();
		out.write("foo".as_bytes()).unwrap();
		out.write(" ".as_bytes()).unwrap();
		out.write("".as_bytes()).unwrap();
		out.write("bar".as_bytes()).unwrap();
		assert_eq!(out.as_string(), "foo bar");
	}

	#[test]
	fn fill_align() {
		let mut out = StringWriter::new();
		print_align(&mut out, Alignment::RIGHT, "foo", '*', 10).unwrap();
		assert_eq!(out.as_string(), "*******foo");

		let mut out = StringWriter::new();
		print_align(&mut out, Alignment::LEFT, "foo", '*', 10).unwrap();
		assert_eq!(out.as_string(), "foo*******");

		let mut out = StringWriter::new();
		print_align(&mut out, Alignment::CENTER, "foo", '*', 10).unwrap();
		assert_eq!(out.as_string(), "***foo****");

		let mut out = StringWriter::new();
		print_align(&mut out, Alignment::CENTER, "foo", '*', 1).unwrap();
		assert_eq!(out.as_string(), "foo");
	}
}