• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 use chrono::offset::{Local, TimeZone};
2 use chrono::{Date, Duration};
3 use plotters::prelude::*;
parse_time(t: &str) -> Date<Local>4 fn parse_time(t: &str) -> Date<Local> {
5     Local
6         .datetime_from_str(&format!("{} 0:0", t), "%Y-%m-%d %H:%M")
7         .unwrap()
8         .date()
9 }
10 const OUT_FILE_NAME: &str = "plotters-doc-data/stock.png";
main() -> Result<(), Box<dyn std::error::Error>>11 fn main() -> Result<(), Box<dyn std::error::Error>> {
12     let data = get_data();
13     let root = BitMapBackend::new(OUT_FILE_NAME, (1024, 768)).into_drawing_area();
14     root.fill(&WHITE)?;
15 
16     let (to_date, from_date) = (
17         parse_time(data[0].0) + Duration::days(1),
18         parse_time(data[29].0) - Duration::days(1),
19     );
20 
21     let mut chart = ChartBuilder::on(&root)
22         .x_label_area_size(40)
23         .y_label_area_size(40)
24         .caption("MSFT Stock Price", ("sans-serif", 50.0).into_font())
25         .build_cartesian_2d(from_date..to_date, 110f32..135f32)?;
26 
27     chart.configure_mesh().light_line_style(WHITE).draw()?;
28 
29     chart.draw_series(
30         data.iter().map(|x| {
31             CandleStick::new(parse_time(x.0), x.1, x.2, x.3, x.4, GREEN.filled(), RED, 15)
32         }),
33     )?;
34 
35     // To avoid the IO failure being ignored silently, we manually call the present function
36     root.present().expect("Unable to write result to file, please make sure 'plotters-doc-data' dir exists under current dir");
37     println!("Result has been saved to {}", OUT_FILE_NAME);
38 
39     Ok(())
40 }
41 
get_data() -> Vec<(&'static str, f32, f32, f32, f32)>42 fn get_data() -> Vec<(&'static str, f32, f32, f32, f32)> {
43     vec![
44         ("2019-04-25", 130.06, 131.37, 128.83, 129.15),
45         ("2019-04-24", 125.79, 125.85, 124.52, 125.01),
46         ("2019-04-23", 124.1, 125.58, 123.83, 125.44),
47         ("2019-04-22", 122.62, 124.0000, 122.57, 123.76),
48         ("2019-04-18", 122.19, 123.52, 121.3018, 123.37),
49         ("2019-04-17", 121.24, 121.85, 120.54, 121.77),
50         ("2019-04-16", 121.64, 121.65, 120.1, 120.77),
51         ("2019-04-15", 120.94, 121.58, 120.57, 121.05),
52         ("2019-04-12", 120.64, 120.98, 120.37, 120.95),
53         ("2019-04-11", 120.54, 120.85, 119.92, 120.33),
54         ("2019-04-10", 119.76, 120.35, 119.54, 120.19),
55         ("2019-04-09", 118.63, 119.54, 118.58, 119.28),
56         ("2019-04-08", 119.81, 120.02, 118.64, 119.93),
57         ("2019-04-05", 119.39, 120.23, 119.37, 119.89),
58         ("2019-04-04", 120.1, 120.23, 118.38, 119.36),
59         ("2019-04-03", 119.86, 120.43, 119.15, 119.97),
60         ("2019-04-02", 119.06, 119.48, 118.52, 119.19),
61         ("2019-04-01", 118.95, 119.1085, 118.1, 119.02),
62         ("2019-03-29", 118.07, 118.32, 116.96, 117.94),
63         ("2019-03-28", 117.44, 117.58, 116.13, 116.93),
64         ("2019-03-27", 117.875, 118.21, 115.5215, 116.77),
65         ("2019-03-26", 118.62, 118.705, 116.85, 117.91),
66         ("2019-03-25", 116.56, 118.01, 116.3224, 117.66),
67         ("2019-03-22", 119.5, 119.59, 117.04, 117.05),
68         ("2019-03-21", 117.135, 120.82, 117.09, 120.22),
69         ("2019-03-20", 117.39, 118.75, 116.71, 117.52),
70         ("2019-03-19", 118.09, 118.44, 116.99, 117.65),
71         ("2019-03-18", 116.17, 117.61, 116.05, 117.57),
72         ("2019-03-15", 115.34, 117.25, 114.59, 115.91),
73         ("2019-03-14", 114.54, 115.2, 114.33, 114.59),
74     ]
75 }
76 #[test]
entry_point()77 fn entry_point() {
78     main().unwrap()
79 }
80