cursive/src/view/text_view.rs
Alexandre Bury e79207ac6e Fix TextView size computation
We weren't counting spaces.
2015-05-23 20:12:11 -07:00

162 lines
4.3 KiB
Rust

use std::cmp;
use vec::Vec2;
use view::{View,DimensionRequest,SizeRequest};
use div::*;
use printer::Printer;
/// A simple view showing a fixed text
pub struct TextView {
content: String,
}
/// Returns the number of lines required to display the given text with the
/// specified maximum line width.
fn get_line_span(line: &str, max_width: usize) -> usize {
let mut lines = 1;
let mut length = 0;
for l in line.split(" ").map(|word| word.len()) {
length += l;
if length >= max_width {
length = l;
lines += 1;
}
length += 1;
}
lines
}
impl TextView {
/// Creates a new TextView with the given content.
pub fn new(content: &str) -> Self {
TextView {
content: content.to_string(),
}
}
/// Replace the text in this view.
pub fn set_content(&mut self, content: &str) {
self.content = content.to_string();
}
/// Returns the current text in this view.
pub fn get_content(&self) -> &str {
&self.content
}
/// Returns the number of lines required to display the content
/// with the given width.
fn get_num_lines(&self, max_width: usize) -> usize {
self.content.split("\n")
.map(|line| get_line_span(line, max_width))
.fold(0, |sum, x| sum + x)
}
fn get_num_cols(&self, max_height: usize) -> usize {
(div_up_usize(self.content.len(), max_height)..self.content.len())
.find(|w| self.get_num_lines(*w) <= max_height)
.unwrap()
}
fn get_ideal_size(&self) -> Vec2 {
let mut max_width = 0;
let mut height = 0;
for line in self.content.split("\n") {
height += 1;
max_width = cmp::max(max_width, line.len() as u32);
}
Vec2::new(max_width, height)
}
}
struct LinesIterator<'a> {
line: &'a str,
start: usize,
width: usize,
}
impl <'a> LinesIterator<'a> {
fn new(line: &'a str, width: usize) -> Self {
if line.len() == 0 {
LinesIterator {
line: " ",
width: width,
start: 0,
}
} else {
LinesIterator {
line: line,
width: width,
start: 0,
}
}
}
}
impl <'a> Iterator for LinesIterator<'a> {
type Item = &'a str;
fn next(&mut self) -> Option<&'a str> {
if self.start >= self.line.len() {
None
} else if self.start + self.width >= self.line.len() {
let start = self.start;
self.start = self.line.len();
Some(&self.line[start..])
} else {
let start = self.start;
let (end,skip_space) = match self.line[start..start+self.width].rfind(" ") {
// Hard break
None => (start + self.width, false),
Some(i) => (start+i, true),
};
self.start = end;
if skip_space {
self.start += 1;
}
Some(&self.line[start..end])
}
}
}
impl View for TextView {
fn draw(&mut self, printer: &Printer, _: bool) {
// We don't have a focused view
let lines = self.content.split("\n")
.flat_map(|line| LinesIterator::new(line, printer.size.x as usize));
for (i, line) in lines.enumerate() {
printer.print((0,i), line);
}
}
fn get_min_size(&self, size: SizeRequest) -> Vec2 {
match (size.w,size.h) {
(DimensionRequest::Unknown, DimensionRequest::Unknown) => Vec2::new(self.content.len() as u32, 1),
(DimensionRequest::Fixed(w),_) => {
let h = self.get_num_lines(w as usize) as u32;
Vec2::new(w, h)
},
(_,DimensionRequest::Fixed(h)) => {
let w = self.get_num_cols(h as usize) as u32;
Vec2::new(w, h)
},
(DimensionRequest::AtMost(w),_) => {
let ideal = self.get_ideal_size();
if w >= ideal.x {
ideal
} else {
let h = self.get_num_lines(w as usize) as u32;
Vec2::new(w, h)
}
},
_ => unreachable!(),
}
}
}