egui_map/
map.rs

1use crate::map::animation::Animation;
2use crate::map::objects::{
3    ContextMenuManager, MapBounds, MapLabel, MapLine, MapPoint, MapSettings, RawLine, RawPoint,
4    TextSettings, VisibilitySetting,
5};
6use chrono;
7use egui::{epaint::CircleShape, widgets::*, *};
8use kdtree::distance::squared_euclidean;
9use kdtree::KdTree;
10use std::collections::{HashMap, HashSet};
11use std::fmt::Error;
12use std::rc::Rc;
13use std::time::Instant;
14
15use self::objects::NodeTemplate;
16
17pub mod animation;
18pub mod objects;
19// This can by any object or point with its associated metadata
20/// Struct that contains coordinates to help calculate nearest point in space
21
22#[derive(Clone)]
23pub struct Map {
24    zoom: f32,
25    previous_zoom: f32,
26    points: Option<HashMap<usize, MapPoint>>,
27    lines: Option<HashMap<String, MapLine>>,
28    labels: Vec<MapLabel>,
29    tree: Option<KdTree<f32, usize, [f32; 2]>>,
30    visible_points: Vec<usize>,
31    map_area: Rect,
32    reference: MapBounds,
33    current: MapBounds,
34    style: egui::Style,
35    current_index: usize,
36    entities: HashMap<usize, Instant>,
37    min_size: (Option<f32>, Option<f32>),
38    max_size: (Option<f32>, Option<f32>),
39    pub settings: MapSettings,
40    menu_manager: Option<Rc<dyn ContextMenuManager>>,
41    node_template: Option<Rc<dyn NodeTemplate>>,
42    visible_lines: HashSet<String>,
43    markers: HashMap<usize, usize>,
44}
45
46impl Default for Map {
47    fn default() -> Self {
48        Map::new()
49    }
50}
51
52impl Widget for &mut Map {
53    fn ui(self, ui: &mut egui::Ui) -> Response {
54        let rect = self.calculate_widget_dimentions(ui);
55
56        // we define the initial coordinate as the center of such rectangle
57        self.reference.dist = rect.distance();
58
59        self.assign_visual_style(ui);
60
61        let canvas = egui::Frame::canvas(ui.style()).inner_margin(Margin::symmetric(3, 5));
62
63        let inner_response = canvas.show(ui, |ui| {
64            #[cfg(feature = "puffin")]
65            puffin::profile_scope!("paint_map");
66
67            if ui.is_rect_visible(self.map_area) {
68                let (resp, paint) =
69                    ui.allocate_painter(self.map_area.size(), egui::Sense::click_and_drag());
70                let vec = resp.drag_delta();
71                if vec.length() != 0.0 {
72                    #[cfg(feature = "puffin")]
73                    puffin::profile_scope!("calculating_points_in_visible_area");
74
75                    let coords = RawPoint::from(vec.to_pos2());
76                    let new_pos = self.reference.pos - (coords / self.zoom);
77                    self.set_pos(new_pos.into());
78                }
79                let map_style = self.settings.styles[self.current_index].clone() * self.zoom;
80                if self.zoom < self.settings.line_visible_zoom {
81                    // filling text settings
82                    let mut text_settings = TextSettings {
83                        size: 12.00 * self.zoom * 2.00,
84                        anchor: Align2::CENTER_CENTER,
85                        family: FontFamily::Proportional,
86                        text: String::new(),
87                        position: RawPoint::default(),
88                        text_color: ui.visuals().text_color(),
89                    };
90                    for label in &self.labels {
91                        text_settings.text.clone_from(&label.text);
92                        paint.text(
93                            label.center,
94                            Align2::CENTER_CENTER,
95                            label.text.as_str(),
96                            map_style.font.clone().unwrap(),
97                            ui.visuals().text_color(),
98                        );
99                        self.paint_label(&paint, &text_settings);
100                    }
101                }
102
103                // Here we determine the widget center to print all nodes
104                // let min_point = self.current.pos - RawPoint::try_from([self.map_area.center().x,self.map_area.center().y]).unwrap();
105
106                let rect_midpoint = RawPoint::from(self.map_area.center());
107                let min_point = self.current.pos - rect_midpoint;
108                let vec_points = &self.visible_points;
109                let hashm = &self.points;
110                self.paint_map_lines(&paint, &min_point);
111
112                if let Ok(nodes_to_remove) =
113                    self.paint_map_points(vec_points, hashm, &paint, ui, &min_point, &resp)
114                {
115                    for node in nodes_to_remove {
116                        self.entities.remove(&node);
117                    }
118                }
119
120                for marker in &self.markers {
121                    if let Some(point) = self.points.as_ref().unwrap().get(marker.1) {
122                        let adjusted_point = point.raw_point * self.zoom - min_point;
123                        if let Some(template) = &self.node_template {
124                            template.marker_ui(ui, adjusted_point.into(), self.zoom);
125                        } else {
126                            let mut shapes = Vec::new();
127                            let color = if ui.visuals().dark_mode {
128                                Color32::LIGHT_GREEN
129                            } else {
130                                Color32::GREEN
131                            };
132                            let mut transparency =
133                                (chrono::Local::now().timestamp_millis() % 2550) / 5;
134                            if transparency > 255 {
135                                transparency = 255 - (transparency - 255)
136                            }
137                            let corrected_color = Color32::from_rgba_unmultiplied(
138                                color.r(),
139                                color.g(),
140                                color.b(),
141                                transparency as u8,
142                            );
143                            shapes.push(Shape::Circle(CircleShape::stroke(
144                                adjusted_point.into(),
145                                4.0 * self.zoom,
146                                Stroke::new(9.0 * self.zoom, corrected_color),
147                            )));
148                            ui.ctx().request_repaint();
149                            ui.painter().extend(shapes);
150                        }
151                    }
152                }
153
154                self.paint_sub_components(ui, self.map_area);
155
156                self.capture_mouse_events(ui, &resp);
157
158                if self.zoom != self.previous_zoom {
159                    #[cfg(feature = "puffin")]
160                    puffin::profile_scope!("calculating viewport with zoom");
161                    self.adjust_bounds();
162                    self.calculate_visible_points();
163                    self.previous_zoom = self.zoom;
164                }
165
166                if let Some(ref mut menu_mon) = &mut self.menu_manager {
167                    resp.context_menu(|ui| {
168                        menu_mon.ui(ui);
169                    });
170                }
171
172                if cfg!(debug_assertions) {
173                    self.print_debug_info(paint, resp);
174                }
175            }
176        });
177        ui.allocate_space(self.map_area.size());
178        inner_response.response
179    }
180}
181
182impl Map {
183    pub fn new() -> Self {
184        let settings = MapSettings::default();
185        Self {
186            zoom: 1.0,
187            previous_zoom: 1.0,
188            map_area: Rect::NOTHING,
189            tree: None,
190            points: None,
191            lines: None,
192            labels: Vec::new(),
193            visible_points: Vec::new(),
194            current: MapBounds::default(),
195            reference: MapBounds::default(),
196            settings,
197            min_size: (None, None),
198            max_size: (None, None),
199            current_index: 0,
200            entities: HashMap::new(),
201            style: egui::Style::default(),
202            menu_manager: None,
203            node_template: None,
204            visible_lines: HashSet::new(),
205            markers: HashMap::new(),
206        }
207    }
208
209    fn calculate_widget_dimentions(&mut self, ui: &mut Ui) -> RawLine {
210        self.map_area = ui.available_rect_before_wrap();
211        let mut left_top = RawPoint::from(self.map_area.left_top());
212        let mut right_bottom = RawPoint::from(self.map_area.right_bottom());
213        if self.max_size.0.is_some()
214            && right_bottom.components[0] > self.max_size.0.unwrap_or(0.0f32)
215        {
216            right_bottom.components[0] = self.max_size.0.unwrap();
217        }
218        if self.max_size.1.is_some()
219            && right_bottom.components[1] > self.max_size.1.unwrap_or(0.0f32)
220        {
221            right_bottom.components[1] = self.max_size.1.unwrap();
222        }
223        if self.min_size.0.is_some() && left_top.components[0] < self.min_size.0.unwrap_or(0.0f32) {
224            left_top.components[0] = self.min_size.0.unwrap();
225        }
226        if self.min_size.1.is_some() && left_top.components[1] < self.min_size.1.unwrap_or(0.0f32) {
227            left_top.components[1] = self.min_size.1.unwrap();
228        }
229        RawLine::new(left_top, right_bottom)
230    }
231
232    fn calculate_visible_points(&mut self) {
233        #[cfg(feature = "puffin")]
234        puffin::profile_scope!("calculate_visible_points");
235        if self.current.dist > 0.0 && self.current.dist < f32::INFINITY {
236            if let Some(tree) = &self.tree {
237                let center = self.current.pos / self.zoom;
238                let radius = self.current.dist.powi(2);
239                let point: [f32; 2] = center.into();
240                let vis_pos = tree.within(&point, radius, &squared_euclidean).unwrap();
241                self.visible_points.clear();
242                for point in vis_pos {
243                    self.visible_points.push(*point.1);
244                    let system = self.points.as_ref().unwrap().get(point.1);
245                    for connection in &system.unwrap().connections {
246                        if !self.visible_lines.contains(&connection.clone()) {
247                            self.visible_lines.insert(connection.clone());
248                        }
249                    }
250                }
251            }
252        }
253    }
254
255    pub fn add_hashmap_points(&mut self, hash_map: HashMap<usize, MapPoint>) {
256        #[cfg(feature = "puffin")]
257        puffin::profile_scope!("add_hashmap_points");
258        let mut min = RawPoint::new(f32::INFINITY, f32::INFINITY);
259        let mut max = RawPoint::new(f32::NEG_INFINITY, f32::NEG_INFINITY);
260        let mut tree = KdTree::<f32, usize, [f32; 2]>::new(2);
261        let mut h_map = hash_map.clone();
262
263        for entry in h_map.iter_mut() {
264            for i in 0..min.components.len() {
265                if entry.1.raw_point.components[i] < min.components[i] {
266                    min.components[i] = entry.1.raw_point.components[i];
267                }
268                if entry.1.raw_point.components[i] > max.components[i] {
269                    max.components[i] = entry.1.raw_point.components[i];
270                }
271            }
272            let _result = tree.add(entry.1.raw_point.into(), *entry.0);
273        }
274
275        // We stablish the max and min coordinates in this map, this wont change until we change the point hash map
276        self.reference.min = min;
277        self.reference.max = max;
278        self.points = Some(h_map);
279        self.tree = Some(tree);
280        self.reference.pos = RawLine::new(min, max).midpoint();
281        // we create a rect that include every node in the map
282        // Stupid fix because rect area could be infinite
283        // I need to implement a more elegant fix
284        if self.map_area.area() == 0.0 {
285            self.reference.dist = 3000.00;
286        } else {
287            let rect = RawLine::new(
288                RawPoint::from(self.map_area.left_top()),
289                RawPoint::from(self.map_area.right_bottom()),
290            );
291            self.reference.dist = rect.distance();
292        }
293        self.current = self.reference.clone();
294        self.calculate_visible_points();
295    }
296
297    pub fn set_pos_from_nodeid(&mut self, node_id: usize) {
298        #[cfg(feature = "puffin")]
299        puffin::profile_scope!("set_pos_from_nodeid");
300        if let Some(hash_map) = &self.points {
301            if let Some(map_point) = hash_map.get(&node_id) {
302                self.reference.pos = map_point.raw_point;
303                self.adjust_bounds();
304                self.calculate_visible_points();
305            }
306        }
307    }
308
309    pub fn set_pos(&mut self, position: [f32; 2]) {
310        #[cfg(feature = "puffin")]
311        puffin::profile_scope!("set_pos");
312        let point = RawPoint::from(position);
313        self.reference.pos = point;
314        self.adjust_bounds();
315        self.calculate_visible_points();
316    }
317
318    pub fn get_pos(self) -> [f32; 2] {
319        #[cfg(feature = "puffin")]
320        puffin::profile_scope!("get_pos");
321        self.reference.pos.into()
322    }
323
324    pub fn add_labels(&mut self, labels: Vec<MapLabel>) {
325        #[cfg(feature = "puffin")]
326        puffin::profile_scope!("add_labels");
327        self.labels = labels;
328    }
329
330    pub fn add_lines(&mut self, lines: HashMap<String, MapLine>) {
331        #[cfg(feature = "puffin")]
332        puffin::profile_scope!("add_lines");
333        self.lines = Some(lines);
334    }
335
336    fn adjust_bounds(&mut self) {
337        #[cfg(feature = "puffin")]
338        puffin::profile_scope!("adjust_bounds");
339        self.current.max = self.reference.max * self.zoom;
340        self.current.min = self.reference.min * self.zoom;
341        self.current.dist = self.reference.dist / self.zoom;
342        self.current.pos = self.reference.pos * self.zoom;
343    }
344
345    fn capture_mouse_events(&mut self, ui: &Ui, resp: &Response) {
346        #[cfg(feature = "puffin")]
347        puffin::profile_scope!("capture_mouse_events");
348        // capture MouseWheel Event for Zoom control change
349        if ui.rect_contains_pointer(self.map_area) {
350            ui.input(|x| {
351                #[cfg(feature = "puffin")]
352                puffin::profile_scope!("capture_mouse_events");
353
354                if !x.events.is_empty() {
355                    for event in &x.events {
356                        match event {
357                            Event::MouseWheel {
358                                unit: _,
359                                delta,
360                                modifiers,
361                            } => {
362                                #[cfg(target_os = "macos")]
363                                let zoom_modifier = if modifiers.mac_cmd {
364                                    delta.y / 80.00
365                                } else {
366                                    delta.y / 400.00
367                                };
368
369                                #[cfg(not(target_os = "macos"))]
370                                let zoom_modifier = if modifiers.ctrl {
371                                    delta.y / 8.00
372                                } else {
373                                    delta.y / 40.00
374                                };
375
376                                let mut pre_zoom = self.zoom + zoom_modifier;
377                                if pre_zoom > self.settings.max_zoom {
378                                    pre_zoom = self.settings.max_zoom;
379                                }
380                                if pre_zoom < self.settings.min_zoom {
381                                    pre_zoom = self.settings.min_zoom;
382                                }
383                                self.zoom = pre_zoom;
384                            }
385                            _ => {
386                                continue;
387                            }
388                        };
389                    }
390                }
391            });
392        }
393        if resp.secondary_clicked() {}
394    }
395
396    pub fn set_zoom(mut self, value: f32) {
397        if value >= self.settings.min_zoom && value <= self.settings.max_zoom {
398            self.zoom = value;
399        }
400    }
401
402    pub fn get_zoom(&mut self) -> f32 {
403        self.zoom
404    }
405
406    fn assign_visual_style(&mut self, ui_obj: &mut Ui) {
407        let style_index = ui_obj.visuals().dark_mode as usize;
408
409        if self.current_index != style_index {
410            #[cfg(feature = "puffin")]
411            puffin::profile_scope!("asign_visual_style");
412
413            self.current_index = style_index;
414            self.style = ui_obj.style_mut().clone();
415            self.style.visuals.extreme_bg_color =
416                self.settings.styles[style_index].background_color;
417            self.style.visuals.window_stroke = self.settings.styles[style_index].border.unwrap();
418        }
419    }
420
421    fn print_debug_info(&mut self, paint: Painter, resp: Response) {
422        #[cfg(feature = "puffin")]
423        puffin::profile_scope!("printing debug data");
424
425        let mut init_pos = Pos2::new(
426            self.map_area.left_top().x + 10.00,
427            self.map_area.left_top().y + 10.00,
428        );
429        let mut msg = "MIN:".to_string()
430            + self.current.min.components[0].to_string().as_str()
431            + ","
432            + self.current.min.components[1].to_string().as_str();
433        paint.debug_text(init_pos, Align2::LEFT_TOP, Color32::LIGHT_GREEN, msg);
434        init_pos.y += 15.0;
435        msg = "MAX:".to_string()
436            + self.current.max.components[0].to_string().as_str()
437            + ","
438            + self.current.max.components[1].to_string().as_str();
439        paint.debug_text(init_pos, Align2::LEFT_TOP, Color32::LIGHT_GREEN, msg);
440        init_pos.y += 15.0;
441        msg = "CUR:(".to_string()
442            + self.current.pos.components[0].to_string().as_str()
443            + ","
444            + self.current.pos.components[1].to_string().as_str()
445            + ")";
446        paint.debug_text(init_pos, Align2::LEFT_TOP, Color32::LIGHT_GREEN, msg);
447        init_pos.y += 15.0;
448        msg = "DST:".to_string() + self.current.dist.to_string().as_str();
449        paint.debug_text(init_pos, Align2::LEFT_TOP, Color32::LIGHT_GREEN, msg);
450        init_pos.y += 15.0;
451        msg = "ZOM:".to_string() + self.zoom.to_string().as_str();
452        paint.debug_text(init_pos, Align2::LEFT_TOP, Color32::GREEN, msg);
453        init_pos.y += 15.0;
454        msg = "REC:(".to_string()
455            + self.map_area.left_top().x.to_string().as_str()
456            + ","
457            + self.map_area.left_top().y.to_string().as_str()
458            + "),("
459            + self.map_area.right_bottom().x.to_string().as_str()
460            + ","
461            + self.map_area.right_bottom().y.to_string().as_str()
462            + ")";
463        paint.debug_text(init_pos, Align2::LEFT_TOP, Color32::LIGHT_GREEN, msg);
464        if let Some(points) = &self.points {
465            init_pos.y += 15.0;
466            msg = "NUM:".to_string() + points.len().to_string().as_str();
467            paint.debug_text(init_pos, Align2::LEFT_TOP, Color32::LIGHT_GREEN, msg);
468        }
469        if !self.visible_points.is_empty() {
470            init_pos.y += 15.0;
471            msg = "VIS:".to_string() + self.visible_points.len().to_string().as_str();
472            paint.debug_text(init_pos, Align2::LEFT_TOP, Color32::LIGHT_GREEN, msg);
473        }
474        if let Some(pointer_pos) = resp.hover_pos() {
475            init_pos.y += 15.0;
476            msg = "HVR:".to_string()
477                + pointer_pos.x.to_string().as_str()
478                + ","
479                + pointer_pos.y.to_string().as_str();
480            paint.debug_text(init_pos, Align2::LEFT_TOP, Color32::LIGHT_BLUE, msg);
481        }
482        let vec = resp.drag_delta();
483        if vec.length() != 0.0 {
484            init_pos.y += 15.0;
485            msg = "DRG:".to_string()
486                + vec.to_pos2().x.to_string().as_str()
487                + ","
488                + vec.to_pos2().y.to_string().as_str();
489            paint.debug_text(init_pos, Align2::LEFT_TOP, Color32::GOLD, msg);
490        }
491    }
492
493    fn paint_sub_components(&mut self, ui_obj: &mut Ui, rect: Rect) {
494        #[cfg(feature = "puffin")]
495        puffin::profile_scope!("map_ui_paint_sub_components");
496        let zoom_slider = egui::Slider::new(
497            &mut self.zoom,
498            self.settings.min_zoom..=self.settings.max_zoom,
499        )
500        .show_value(false)
501        //.step_by(0.1)
502        .orientation(SliderOrientation::Vertical);
503        let mut pos1 = rect.right_top();
504        let mut pos2 = rect.right_top();
505        pos1.x -= 80.0;
506        pos1.y += 120.0;
507        pos2.x -= 60.0;
508        pos2.y += 240.0;
509
510        // TODO: Verify if this implementation its correct migrated from allocate_ui_at_rect()
511        let sub_rect = egui::Rect::from_two_pos(pos1, pos2);
512        //ui_obj.allocate_ui_with_layout(sub_rect.size(), egui::Layout::right_to_left(Align::TOP), |ui_obj| {
513        let ui_builder = egui::UiBuilder::new().clone().max_rect(sub_rect);
514        //});
515        ui_obj.scope_builder(ui_builder, |ui_obj| {
516            ui_obj.add(zoom_slider);
517        });
518    }
519
520    fn paint_map_points(
521        &self,
522        vec_points: &Vec<usize>,
523        hashm: &Option<HashMap<usize, MapPoint>>,
524        paint: &Painter,
525        ui_obj: &mut Ui,
526        min_point: &RawPoint,
527        resp: &Response,
528    ) -> Result<Vec<usize>, ()> {
529        let mut nearest_id = None;
530        let mut nodes_to_remove = Vec::new();
531        let mut shape_vec = vec![];
532
533        if hashm.is_none() {
534            return Err(());
535        }
536        if vec_points.is_empty() {
537            return Err(());
538        }
539        // detecting the nearest hover node
540        if self.settings.node_text_visibility == VisibilitySetting::Hover && resp.hovered() {
541            if let Some(point) = resp.hover_pos() {
542                let raw_point = RawPoint::from(point);
543                let hovered_map_point = (*min_point + raw_point) / self.zoom;
544                if let Ok(nearest_node) = self.tree.as_ref().unwrap().nearest(
545                    &hovered_map_point.components,
546                    1,
547                    &squared_euclidean,
548                ) {
549                    nearest_id = Some(nearest_node.first().unwrap().1);
550                }
551            }
552        }
553        // filling text settings
554        let mut text_settings = TextSettings {
555            size: 12.00 * self.zoom,
556            anchor: Align2::LEFT_BOTTOM,
557            family: FontFamily::Proportional,
558            text: String::new(),
559            position: RawPoint::default(),
560            text_color: ui_obj.visuals().text_color(),
561        };
562
563        // Drawing Points
564        for temp_point in vec_points {
565            if let Some(system) = hashm.as_ref().unwrap().get(temp_point) {
566                #[cfg(feature = "puffin")]
567                puffin::profile_scope!("painting_points_m");
568                let viewport_point = system.raw_point * self.zoom - min_point;
569                if let Some(node_template) = &self.node_template {
570                    if nearest_id.unwrap_or(&0usize) == &system.get_id() {
571                        node_template.selection_ui(ui_obj, viewport_point.into(), self.zoom);
572                    }
573                } else if self.zoom > self.settings.label_visible_zoom
574                    && self.settings.node_text_visibility == VisibilitySetting::Allways
575                    || (self.settings.node_text_visibility == VisibilitySetting::Hover
576                        && nearest_id.unwrap_or(&0usize) == &system.get_id())
577                {
578                    let mut viewport_text = viewport_point;
579                    viewport_text.components[0] += 3.0 * self.zoom;
580                    viewport_text.components[1] -= 3.0 * self.zoom;
581                    text_settings.position = viewport_text;
582                    text_settings.text = system.get_name();
583                    self.paint_label(paint, &text_settings);
584                }
585
586                let system_id = system.get_id();
587                if let Some(init_time) = self.entities.get(&system_id) {
588                    if let Some(template) = &self.node_template {
589                        template.notification_ui(
590                            ui_obj,
591                            viewport_point.into(),
592                            self.zoom,
593                            *init_time,
594                            self.settings.styles[self.current_index].alert_color,
595                        );
596                    } else {
597                        match Animation::pulse(
598                            paint,
599                            viewport_point,
600                            self.zoom,
601                            *init_time,
602                            self.settings.styles[self.current_index].alert_color,
603                        ) {
604                            Ok(true) => {
605                                ui_obj.ctx().request_repaint();
606                            }
607                            Ok(false) => nodes_to_remove.push(system_id),
608                            Err(_) => (),
609                        }
610                    }
611                }
612                if let Some(node_template) = &self.node_template {
613                    node_template.node_ui(ui_obj, viewport_point.into(), self.zoom, system);
614                } else {
615                    shape_vec.push(Shape::circle_filled(
616                        viewport_point.into(),
617                        4.00 * self.zoom,
618                        self.settings.styles[self.current_index].fill_color,
619                    ));
620                }
621            }
622        }
623        paint.extend(shape_vec);
624        Ok(nodes_to_remove)
625    }
626
627    fn paint_map_lines(&self, painter: &Painter, min_point: &RawPoint) {
628        #[cfg(feature = "puffin")]
629        puffin::profile_scope!("paint_map_lines");
630
631        // Drawing Lines
632        if self.zoom > self.settings.line_visible_zoom {
633            let mut shape_vec = vec![];
634            let mut stroke = self.settings.styles[self.current_index].line.unwrap();
635            let transparency_range = self.zoom - self.settings.line_visible_zoom;
636            if (0.00..0.80).contains(&transparency_range) {
637                let mut tup_stroke = self.settings.styles[self.current_index]
638                    .line
639                    .unwrap()
640                    .color
641                    .to_tuple();
642                let transparency = (self.zoom - self.settings.line_visible_zoom) / 0.80;
643                tup_stroke.3 = (255.0 * transparency).round() as u8;
644                let color = Color32::from_rgba_unmultiplied(
645                    tup_stroke.0,
646                    tup_stroke.1,
647                    tup_stroke.2,
648                    tup_stroke.3,
649                );
650                stroke = Stroke::new(
651                    self.settings.styles[self.current_index].line.unwrap().width,
652                    color,
653                );
654            }
655            //let stroke = Stroke::new(10.0,Color32::GREEN);
656            for line in &self.visible_lines {
657                if let Some(connection) = self.lines.as_ref().unwrap().get(line) {
658                    let pos_a = connection.raw_line.points[0] * self.zoom - min_point;
659                    let pos_b = connection.raw_line.points[1] * self.zoom - min_point;
660                    //let pos_a = connection.raw_line.points[0] / self.zoom - min_point;
661                    //let pos_b = connection.raw_line.points[1] / self.zoom - min_point;
662                    shape_vec.push(Shape::line_segment([pos_a.into(), pos_b.into()], stroke));
663                    //shape_vec.push(painter.line_segment([pos_a.into(),pos_b.into()], stroke));
664                }
665            }
666            painter.extend(shape_vec);
667        }
668    }
669
670    fn paint_label(&self, paint: &Painter, text_settings: &TextSettings) {
671        #[cfg(feature = "puffin")]
672        puffin::profile_scope!("paint_label");
673        paint.text(
674            text_settings.position.into(),
675            text_settings.anchor,
676            text_settings.text.clone(),
677            FontId::new(text_settings.size, text_settings.family.clone()),
678            text_settings.text_color,
679        );
680    }
681
682    pub fn notify(&mut self, id_node: usize, time: Instant) -> Result<bool, Error> {
683        #[cfg(feature = "puffin")]
684        puffin::profile_scope!("notify");
685        self.entities
686            .entry(id_node)
687            .and_modify(|value| *value = time)
688            .or_insert(time);
689        Ok(true)
690    }
691
692    pub fn set_context_manager(&mut self, manager: Rc<dyn ContextMenuManager>) {
693        self.menu_manager = Some(manager);
694    }
695
696    pub fn set_node_template(&mut self, template: Rc<dyn NodeTemplate>) {
697        self.node_template = Some(template);
698    }
699
700    pub fn update_marker(&mut self, id: usize, node_id: usize) {
701        self.markers
702            .entry(id)
703            .and_modify(|value| *value = node_id)
704            .or_insert(node_id);
705    }
706
707    pub fn allocate_at_least(&mut self, width: Option<f32>, height: Option<f32>) {
708        self.min_size = (width, height);
709    }
710
711    pub fn allocate_at_most(&mut self, width: Option<f32>, height: Option<f32>) {
712        self.max_size = (width, height);
713    }
714}