1 use std::borrow::Borrow;
2 use std::i32;
3
4 use super::{Drawable, PointCollection};
5 use crate::style::{FontDesc, FontResult, LayoutBox, TextStyle};
6 use plotters_backend::{BackendCoord, DrawingBackend, DrawingErrorKind};
7
8 /// A single line text element. This can be owned or borrowed string, dependents on
9 /// `String` or `str` moved into.
10 pub struct Text<'a, Coord, T: Borrow<str>> {
11 text: T,
12 coord: Coord,
13 style: TextStyle<'a>,
14 }
15
16 impl<'a, Coord, T: Borrow<str>> Text<'a, Coord, T> {
17 /// Create a new text element
18 /// - `text`: The text for the element
19 /// - `points`: The upper left conner for the text element
20 /// - `style`: The text style
21 /// - Return the newly created text element
new<S: Into<TextStyle<'a>>>(text: T, points: Coord, style: S) -> Self22 pub fn new<S: Into<TextStyle<'a>>>(text: T, points: Coord, style: S) -> Self {
23 Self {
24 text,
25 coord: points,
26 style: style.into(),
27 }
28 }
29 }
30
31 impl<'b, 'a, Coord: 'a, T: Borrow<str> + 'a> PointCollection<'a, Coord> for &'a Text<'b, Coord, T> {
32 type Point = &'a Coord;
33 type IntoIter = std::iter::Once<&'a Coord>;
point_iter(self) -> Self::IntoIter34 fn point_iter(self) -> Self::IntoIter {
35 std::iter::once(&self.coord)
36 }
37 }
38
39 impl<'a, Coord: 'a, DB: DrawingBackend, T: Borrow<str>> Drawable<DB> for Text<'a, Coord, T> {
draw<I: Iterator<Item = BackendCoord>>( &self, mut points: I, backend: &mut DB, _: (u32, u32), ) -> Result<(), DrawingErrorKind<DB::ErrorType>>40 fn draw<I: Iterator<Item = BackendCoord>>(
41 &self,
42 mut points: I,
43 backend: &mut DB,
44 _: (u32, u32),
45 ) -> Result<(), DrawingErrorKind<DB::ErrorType>> {
46 if let Some(a) = points.next() {
47 return backend.draw_text(self.text.borrow(), &self.style, a);
48 }
49 Ok(())
50 }
51 }
52
53 /// An multi-line text element. The `Text` element allows only single line text
54 /// and the `MultiLineText` supports drawing multiple lines
55 pub struct MultiLineText<'a, Coord, T: Borrow<str>> {
56 lines: Vec<T>,
57 coord: Coord,
58 style: TextStyle<'a>,
59 line_height: f64,
60 }
61
62 impl<'a, Coord, T: Borrow<str>> MultiLineText<'a, Coord, T> {
63 /// Create an empty multi-line text element.
64 /// Lines can be append to the empty multi-line by calling `push_line` method
65 ///
66 /// `pos`: The upper left corner
67 /// `style`: The style of the text
new<S: Into<TextStyle<'a>>>(pos: Coord, style: S) -> Self68 pub fn new<S: Into<TextStyle<'a>>>(pos: Coord, style: S) -> Self {
69 MultiLineText {
70 lines: vec![],
71 coord: pos,
72 style: style.into(),
73 line_height: 1.25,
74 }
75 }
76
77 /// Set the line height of the multi-line text element
set_line_height(&mut self, value: f64) -> &mut Self78 pub fn set_line_height(&mut self, value: f64) -> &mut Self {
79 self.line_height = value;
80 self
81 }
82
83 /// Push a new line into the given multi-line text
84 /// `line`: The line to be pushed
push_line<L: Into<T>>(&mut self, line: L)85 pub fn push_line<L: Into<T>>(&mut self, line: L) {
86 self.lines.push(line.into());
87 }
88
89 /// Estimate the multi-line text element's dimension
estimate_dimension(&self) -> FontResult<(i32, i32)>90 pub fn estimate_dimension(&self) -> FontResult<(i32, i32)> {
91 let (mut mx, mut my) = (0, 0);
92
93 for ((x, y), t) in self.layout_lines((0, 0)).zip(self.lines.iter()) {
94 let (dx, dy) = self.style.font.box_size(t.borrow())?;
95 mx = mx.max(x + dx as i32);
96 my = my.max(y + dy as i32);
97 }
98
99 Ok((mx, my))
100 }
101
102 /// Move the location to the specified location
relocate(&mut self, coord: Coord)103 pub fn relocate(&mut self, coord: Coord) {
104 self.coord = coord
105 }
106
layout_lines(&self, (x0, y0): BackendCoord) -> impl Iterator<Item = BackendCoord>107 fn layout_lines(&self, (x0, y0): BackendCoord) -> impl Iterator<Item = BackendCoord> {
108 let font_height = self.style.font.get_size();
109 let actual_line_height = font_height * self.line_height;
110 (0..self.lines.len() as u32).map(move |idx| {
111 let y = f64::from(y0) + f64::from(idx) * actual_line_height;
112 // TODO: Support text alignment as well, currently everything is left aligned
113 let x = f64::from(x0);
114 (x.round() as i32, y.round() as i32)
115 })
116 }
117 }
118
layout_multiline_text<'a, F: FnMut(&'a str)>( text: &'a str, max_width: u32, font: FontDesc<'a>, mut func: F, )119 fn layout_multiline_text<'a, F: FnMut(&'a str)>(
120 text: &'a str,
121 max_width: u32,
122 font: FontDesc<'a>,
123 mut func: F,
124 ) {
125 for line in text.lines() {
126 if max_width == 0 || line.is_empty() {
127 func(line);
128 } else {
129 let mut remaining = &line[0..];
130
131 while !remaining.is_empty() {
132 let mut left = 0;
133 while left < remaining.len() {
134 let width = font.box_size(&remaining[0..=left]).unwrap_or((0, 0)).0 as i32;
135
136 if width > max_width as i32 {
137 break;
138 }
139 left += 1;
140 }
141
142 if left == 0 {
143 left += 1;
144 }
145
146 let cur_line = &remaining[..left];
147 remaining = &remaining[left..];
148
149 func(cur_line);
150 }
151 }
152 }
153 }
154
155 impl<'a, T: Borrow<str>> MultiLineText<'a, BackendCoord, T> {
156 /// Compute the line layout
compute_line_layout(&self) -> FontResult<Vec<LayoutBox>>157 pub fn compute_line_layout(&self) -> FontResult<Vec<LayoutBox>> {
158 let mut ret = vec![];
159 for ((x, y), t) in self.layout_lines(self.coord).zip(self.lines.iter()) {
160 let (dx, dy) = self.style.font.box_size(t.borrow())?;
161 ret.push(((x, y), (x + dx as i32, y + dy as i32)));
162 }
163 Ok(ret)
164 }
165 }
166
167 impl<'a, Coord> MultiLineText<'a, Coord, &'a str> {
168 /// Parse a multi-line text into an multi-line element.
169 ///
170 /// `text`: The text that is parsed
171 /// `pos`: The position of the text
172 /// `style`: The style for this text
173 /// `max_width`: The width of the multi-line text element, the line will break
174 /// into two lines if the line is wider than the max_width. If 0 is given, do not
175 /// do any line wrapping
from_str<ST: Into<&'a str>, S: Into<TextStyle<'a>>>( text: ST, pos: Coord, style: S, max_width: u32, ) -> Self176 pub fn from_str<ST: Into<&'a str>, S: Into<TextStyle<'a>>>(
177 text: ST,
178 pos: Coord,
179 style: S,
180 max_width: u32,
181 ) -> Self {
182 let text = text.into();
183 let mut ret = MultiLineText::new(pos, style);
184
185 layout_multiline_text(text, max_width, ret.style.font.clone(), |l| {
186 ret.push_line(l)
187 });
188 ret
189 }
190 }
191
192 impl<'a, Coord> MultiLineText<'a, Coord, String> {
193 /// Parse a multi-line text into an multi-line element.
194 ///
195 /// `text`: The text that is parsed
196 /// `pos`: The position of the text
197 /// `style`: The style for this text
198 /// `max_width`: The width of the multi-line text element, the line will break
199 /// into two lines if the line is wider than the max_width. If 0 is given, do not
200 /// do any line wrapping
from_string<S: Into<TextStyle<'a>>>( text: String, pos: Coord, style: S, max_width: u32, ) -> Self201 pub fn from_string<S: Into<TextStyle<'a>>>(
202 text: String,
203 pos: Coord,
204 style: S,
205 max_width: u32,
206 ) -> Self {
207 let mut ret = MultiLineText::new(pos, style);
208
209 layout_multiline_text(text.as_str(), max_width, ret.style.font.clone(), |l| {
210 ret.push_line(l.to_string())
211 });
212 ret
213 }
214 }
215
216 impl<'b, 'a, Coord: 'a, T: Borrow<str> + 'a> PointCollection<'a, Coord>
217 for &'a MultiLineText<'b, Coord, T>
218 {
219 type Point = &'a Coord;
220 type IntoIter = std::iter::Once<&'a Coord>;
point_iter(self) -> Self::IntoIter221 fn point_iter(self) -> Self::IntoIter {
222 std::iter::once(&self.coord)
223 }
224 }
225
226 impl<'a, Coord: 'a, DB: DrawingBackend, T: Borrow<str>> Drawable<DB>
227 for MultiLineText<'a, Coord, T>
228 {
draw<I: Iterator<Item = BackendCoord>>( &self, mut points: I, backend: &mut DB, _: (u32, u32), ) -> Result<(), DrawingErrorKind<DB::ErrorType>>229 fn draw<I: Iterator<Item = BackendCoord>>(
230 &self,
231 mut points: I,
232 backend: &mut DB,
233 _: (u32, u32),
234 ) -> Result<(), DrawingErrorKind<DB::ErrorType>> {
235 if let Some(a) = points.next() {
236 for (point, text) in self.layout_lines(a).zip(self.lines.iter()) {
237 backend.draw_text(text.borrow(), &self.style, point)?;
238 }
239 }
240 Ok(())
241 }
242 }
243