mirror of
				https://github.com/SoftFever/OrcaSlicer.git
				synced 2025-10-25 01:31:14 -06:00 
			
		
		
		
	SLA support gizmo hotkeys added (A,M,Esc,Enter)
This commit is contained in:
		
							parent
							
								
									01c9b13ade
								
							
						
					
					
						commit
						bb3819fd18
					
				
					 4 changed files with 226 additions and 194 deletions
				
			
		|  | @ -4625,7 +4625,19 @@ void GLCanvas3D::on_char(wxKeyEvent& evt) | ||||||
|         switch (keyCode) |         switch (keyCode) | ||||||
|         { |         { | ||||||
|         // key ESC
 |         // key ESC
 | ||||||
|         case WXK_ESCAPE: { m_gizmos.reset_all_states(); m_dirty = true;  break; } |         case WXK_ESCAPE: { | ||||||
|  |             if (m_gizmos.get_current_type() != Gizmos::SlaSupports || !m_gizmos.mouse_event(SLAGizmoEventType::DiscardChanges)) | ||||||
|  |                 m_gizmos.reset_all_states(); | ||||||
|  |             m_dirty = true; | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         case WXK_RETURN: { | ||||||
|  |             if (m_gizmos.get_current_type() == Gizmos::SlaSupports && m_gizmos.mouse_event(SLAGizmoEventType::ApplyChanges)) | ||||||
|  |                 m_dirty = true; | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
| #ifdef __APPLE__ | #ifdef __APPLE__ | ||||||
|         case WXK_BACK: // the low cost Apple solutions are not equipped with a Delete key, use Backspace instead.
 |         case WXK_BACK: // the low cost Apple solutions are not equipped with a Delete key, use Backspace instead.
 | ||||||
| #else /* __APPLE__ */ | #else /* __APPLE__ */ | ||||||
|  | @ -4648,11 +4660,25 @@ void GLCanvas3D::on_char(wxKeyEvent& evt) | ||||||
|         case '-': { post_event(Event<int>(EVT_GLCANVAS_INCREASE_INSTANCES, -1)); break; } |         case '-': { post_event(Event<int>(EVT_GLCANVAS_INCREASE_INSTANCES, -1)); break; } | ||||||
|         case '?': { post_event(SimpleEvent(EVT_GLCANVAS_QUESTION_MARK)); break; } |         case '?': { post_event(SimpleEvent(EVT_GLCANVAS_QUESTION_MARK)); break; } | ||||||
|         case 'A': |         case 'A': | ||||||
|         case 'a': { post_event(SimpleEvent(EVT_GLCANVAS_ARRANGE)); break; } |         case 'a': { | ||||||
|  |             if (m_gizmos.get_current_type() == Gizmos::SlaSupports) { | ||||||
|  |                 if (m_gizmos.mouse_event(SLAGizmoEventType::AutomaticGeneration)) | ||||||
|  |                     m_dirty = true; | ||||||
|  |             } | ||||||
|  |             else | ||||||
|  |                 post_event(SimpleEvent(EVT_GLCANVAS_ARRANGE)); | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|         case 'B': |         case 'B': | ||||||
|         case 'b': { zoom_to_bed(); break; } |         case 'b': { zoom_to_bed(); break; } | ||||||
|         case 'I': |         case 'I': | ||||||
|         case 'i': { set_camera_zoom(1.0f); break; } |         case 'i': { set_camera_zoom(1.0f); break; } | ||||||
|  |         case 'M': | ||||||
|  |         case 'm': { | ||||||
|  |             if (m_gizmos.get_current_type() == Gizmos::SlaSupports && m_gizmos.mouse_event(SLAGizmoEventType::ManualEditing)) | ||||||
|  |                 m_dirty = true; | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|         case 'O': |         case 'O': | ||||||
|         case 'o': { set_camera_zoom(-1.0f); break; } |         case 'o': { set_camera_zoom(-1.0f); break; } | ||||||
|         case 'Z': |         case 'Z': | ||||||
|  |  | ||||||
|  | @ -125,7 +125,11 @@ enum class SLAGizmoEventType { | ||||||
|     Dragging, |     Dragging, | ||||||
|     Delete, |     Delete, | ||||||
|     SelectAll, |     SelectAll, | ||||||
|     ShiftUp |     ShiftUp, | ||||||
|  |     ApplyChanges, | ||||||
|  |     DiscardChanges, | ||||||
|  |     AutomaticGeneration, | ||||||
|  |     ManualEditing | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -1941,12 +1941,6 @@ bool GLGizmoSlaSupports::is_mesh_update_necessary() const | ||||||
| 
 | 
 | ||||||
|     //if (m_state != On || !m_model_object || m_model_object->instances.empty() || ! m_instance_matrix.isApprox(m_source_data.matrix))
 |     //if (m_state != On || !m_model_object || m_model_object->instances.empty() || ! m_instance_matrix.isApprox(m_source_data.matrix))
 | ||||||
|     //    return false;
 |     //    return false;
 | ||||||
| 
 |  | ||||||
|     // following should detect direct mesh changes (can be removed after the mesh is made completely immutable):
 |  | ||||||
|     /*const float* first_vertex = m_model_object->volumes.front()->get_convex_hull().first_vertex();
 |  | ||||||
|     Vec3d first_point((double)first_vertex[0], (double)first_vertex[1], (double)first_vertex[2]); |  | ||||||
|     if (first_point != m_source_data.mesh_first_point) |  | ||||||
|         return true;*/ |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void GLGizmoSlaSupports::update_mesh() | void GLGizmoSlaSupports::update_mesh() | ||||||
|  | @ -2022,8 +2016,7 @@ Vec3f GLGizmoSlaSupports::unproject_on_mesh(const Vec2d& mouse_pos) | ||||||
| // concludes that the event was not intended for it, it should return false.
 | // concludes that the event was not intended for it, it should return false.
 | ||||||
| bool GLGizmoSlaSupports::mouse_event(SLAGizmoEventType action, const Vec2d& mouse_position, bool shift_down) | bool GLGizmoSlaSupports::mouse_event(SLAGizmoEventType action, const Vec2d& mouse_position, bool shift_down) | ||||||
| { | { | ||||||
|     if (!m_editing_mode) |     if (m_editing_mode) { | ||||||
|         return false; |  | ||||||
| 
 | 
 | ||||||
|         // left down - show the selection rectangle:
 |         // left down - show the selection rectangle:
 | ||||||
|         if (action == SLAGizmoEventType::LeftDown && shift_down) { |         if (action == SLAGizmoEventType::LeftDown && shift_down) { | ||||||
|  | @ -2142,6 +2135,16 @@ bool GLGizmoSlaSupports::mouse_event(SLAGizmoEventType action, const Vec2d& mous | ||||||
|             return true; |             return true; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  |         if (action ==  SLAGizmoEventType::ApplyChanges) { | ||||||
|  |             editing_mode_apply_changes(); | ||||||
|  |             return true; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (action ==  SLAGizmoEventType::DiscardChanges) { | ||||||
|  |             editing_mode_discard_changes(); | ||||||
|  |             return true; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|         if (action == SLAGizmoEventType::RightDown) { |         if (action == SLAGizmoEventType::RightDown) { | ||||||
|             if (m_hover_id != -1) { |             if (m_hover_id != -1) { | ||||||
|                 select_point(NoPoints); |                 select_point(NoPoints); | ||||||
|  | @ -2156,6 +2159,19 @@ bool GLGizmoSlaSupports::mouse_event(SLAGizmoEventType action, const Vec2d& mous | ||||||
|             select_point(AllPoints); |             select_point(AllPoints); | ||||||
|             return true; |             return true; | ||||||
|         } |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     if (!m_editing_mode) { | ||||||
|  |         if (action == SLAGizmoEventType::AutomaticGeneration) { | ||||||
|  |             auto_generate(); | ||||||
|  |             return true; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (action == SLAGizmoEventType::ManualEditing) { | ||||||
|  |             switch_to_editing_mode(); | ||||||
|  |             return true; | ||||||
|  |         } | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|     return false; |     return false; | ||||||
| } | } | ||||||
|  | @ -2241,7 +2257,6 @@ RENDER_AGAIN: | ||||||
| 
 | 
 | ||||||
|     bool force_refresh = false; |     bool force_refresh = false; | ||||||
|     bool remove_selected = false; |     bool remove_selected = false; | ||||||
|     bool old_editing_state = m_editing_mode; |  | ||||||
| 
 | 
 | ||||||
|     if (m_editing_mode) { |     if (m_editing_mode) { | ||||||
|         m_imgui->text(_(L("Left mouse click - add point"))); |         m_imgui->text(_(L("Left mouse click - add point"))); | ||||||
|  | @ -2285,9 +2300,6 @@ RENDER_AGAIN: | ||||||
|         if (apply_changes) { |         if (apply_changes) { | ||||||
|             editing_mode_apply_changes(); |             editing_mode_apply_changes(); | ||||||
|             force_refresh = true; |             force_refresh = true; | ||||||
|             // Recalculate support structures once the editing mode is left.
 |  | ||||||
|             // m_parent.post_event(SimpleEvent(EVT_GLCANVAS_SCHEDULE_BACKGROUND_PROCESS));
 |  | ||||||
|             wxGetApp().plater()->reslice_SLA_supports(*m_model_object); |  | ||||||
|         } |         } | ||||||
|         ImGui::SameLine(); |         ImGui::SameLine(); | ||||||
|         bool discard_changes = m_imgui->button(_(L("Discard changes"))); |         bool discard_changes = m_imgui->button(_(L("Discard changes"))); | ||||||
|  | @ -2305,70 +2317,24 @@ RENDER_AGAIN: | ||||||
|         ImGui::SameLine(); |         ImGui::SameLine(); | ||||||
|         value_changed |= ImGui::InputDouble("%", &m_density, 0.0f, 0.0f, "%.f");*/ |         value_changed |= ImGui::InputDouble("%", &m_density, 0.0f, 0.0f, "%.f");*/ | ||||||
| 
 | 
 | ||||||
|         bool generate = m_imgui->button(_(L("Auto-generate points"))); |         bool generate = m_imgui->button(_(L("Auto-generate points [A]"))); | ||||||
| 
 | 
 | ||||||
|         if (generate) { |         if (generate) | ||||||
| #if SLAGIZMO_IMGUI_MODAL |             auto_generate(); | ||||||
|             ImGui::OpenPopup(_(L("Warning"))); |  | ||||||
|             m_show_modal = true; |  | ||||||
|             force_refresh = true; |  | ||||||
| #else |  | ||||||
|             wxMessageDialog dlg(GUI::wxGetApp().plater(), _(L( |  | ||||||
|                         "Autogeneration will erase all manually edited points.\n\n" |  | ||||||
|                         "Are you sure you want to do it?\n" |  | ||||||
|                         )), _(L("Warning")), wxICON_WARNING | wxYES | wxNO); |  | ||||||
|             if (m_model_object->sla_support_points.empty() || dlg.ShowModal() == wxID_YES) { |  | ||||||
|                 m_model_object->sla_support_points.clear(); |  | ||||||
|                 m_editing_mode_cache.clear(); |  | ||||||
|                 wxGetApp().plater()->reslice_SLA_supports(*m_model_object); |  | ||||||
|             } |  | ||||||
| #endif |  | ||||||
|         } |  | ||||||
| #if SLAGIZMO_IMGUI_MODAL |  | ||||||
|         if (m_show_modal) { |  | ||||||
|             if (ImGui::BeginPopupModal(_(L("Warning")), &m_show_modal/*, ImGuiWindowFlags_NoDecoration*/)) |  | ||||||
|             { |  | ||||||
|                 m_imgui->text(_(L("Autogeneration will erase all manually edited points."))); |  | ||||||
|                 m_imgui->text(""); |  | ||||||
|                 m_imgui->text(_(L("Are you sure you want to do it?"))); |  | ||||||
| 
 | 
 | ||||||
|                 if (m_imgui->button(_(L("Continue")))) |  | ||||||
|                 { |  | ||||||
|                     ImGui::CloseCurrentPopup(); |  | ||||||
|                     m_show_modal = false; |  | ||||||
| 
 |  | ||||||
|                     m_model_object->sla_support_points.clear(); |  | ||||||
|                     m_editing_mode_cache.clear(); |  | ||||||
|                     wxGetApp().plater()->reslice_SLA_supports(*m_model_object); |  | ||||||
|                 } |  | ||||||
|                 ImGui::SameLine(); |  | ||||||
|                 if (m_imgui->button(_(L("Cancel")))) { |  | ||||||
|                     ImGui::CloseCurrentPopup(); |  | ||||||
|                     m_show_modal = false; |  | ||||||
|                 } |  | ||||||
|             ImGui::EndPopup(); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             if (!m_show_modal) |  | ||||||
|                 force_refresh = true; |  | ||||||
|         } |  | ||||||
| #endif |  | ||||||
|         m_imgui->text(""); |         m_imgui->text(""); | ||||||
|         m_imgui->text(""); |         m_imgui->text(""); | ||||||
|         bool editing_clicked = m_imgui->button(_(L("Manual editing"))); |         if (m_imgui->button(_(L("Manual editing [M]")))) | ||||||
|         if (editing_clicked) { |             switch_to_editing_mode(); | ||||||
|             editing_mode_reload_cache(); |  | ||||||
|             m_editing_mode = true; |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     m_imgui->end(); |     m_imgui->end(); | ||||||
| 
 | 
 | ||||||
|     if (m_editing_mode != old_editing_state) { // user just toggled between editing/non-editing mode
 |     if (m_editing_mode != m_old_editing_state) { // user toggled between editing/non-editing mode
 | ||||||
|         m_parent.toggle_sla_auxiliaries_visibility(!m_editing_mode); |         m_parent.toggle_sla_auxiliaries_visibility(!m_editing_mode); | ||||||
|         force_refresh = true; |         force_refresh = true; | ||||||
|     } |     } | ||||||
| 
 |     m_old_editing_state = m_editing_mode; | ||||||
| 
 | 
 | ||||||
|     if (remove_selected) { |     if (remove_selected) { | ||||||
|         force_refresh = false; |         force_refresh = false; | ||||||
|  | @ -2434,18 +2400,13 @@ void GLGizmoSlaSupports::on_set_state() | ||||||
| 
 | 
 | ||||||
|             m_parent.toggle_model_objects_visibility(true); |             m_parent.toggle_model_objects_visibility(true); | ||||||
|             m_editing_mode = false; // so it is not active next time the gizmo opens
 |             m_editing_mode = false; // so it is not active next time the gizmo opens
 | ||||||
| 
 |  | ||||||
| #if SLAGIZMO_IMGUI_MODAL |  | ||||||
|             if (m_show_modal) { |  | ||||||
|                 m_show_modal = false; |  | ||||||
|                 on_render_input_window(0,0,m_parent.get_selection()); // this is necessary to allow ImGui to terminate the modal dialog correctly
 |  | ||||||
|             } |  | ||||||
| #endif |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|     m_old_state = m_state; |     m_old_state = m_state; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| void GLGizmoSlaSupports::on_start_dragging(const GLCanvas3D::Selection& selection) | void GLGizmoSlaSupports::on_start_dragging(const GLCanvas3D::Selection& selection) | ||||||
| { | { | ||||||
|     if (m_hover_id != -1) { |     if (m_hover_id != -1) { | ||||||
|  | @ -2454,6 +2415,8 @@ void GLGizmoSlaSupports::on_start_dragging(const GLCanvas3D::Selection& selectio | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| void GLGizmoSlaSupports::select_point(int i) | void GLGizmoSlaSupports::select_point(int i) | ||||||
| { | { | ||||||
|     if (i == AllPoints || i == NoPoints) { |     if (i == AllPoints || i == NoPoints) { | ||||||
|  | @ -2467,6 +2430,8 @@ void GLGizmoSlaSupports::select_point(int i) | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| void GLGizmoSlaSupports::editing_mode_discard_changes() | void GLGizmoSlaSupports::editing_mode_discard_changes() | ||||||
| { | { | ||||||
|     m_editing_mode_cache.clear(); |     m_editing_mode_cache.clear(); | ||||||
|  | @ -2476,6 +2441,8 @@ void GLGizmoSlaSupports::editing_mode_discard_changes() | ||||||
|     m_unsaved_changes = false; |     m_unsaved_changes = false; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| void GLGizmoSlaSupports::editing_mode_apply_changes() | void GLGizmoSlaSupports::editing_mode_apply_changes() | ||||||
| { | { | ||||||
|     // If there are no changes, don't touch the front-end. The data in the cache could have been
 |     // If there are no changes, don't touch the front-end. The data in the cache could have been
 | ||||||
|  | @ -2487,8 +2454,14 @@ void GLGizmoSlaSupports::editing_mode_apply_changes() | ||||||
|     } |     } | ||||||
|     m_editing_mode = false; |     m_editing_mode = false; | ||||||
|     m_unsaved_changes = false; |     m_unsaved_changes = false; | ||||||
|  | 
 | ||||||
|  |     // Recalculate support structures once the editing mode is left.
 | ||||||
|  |     // m_parent.post_event(SimpleEvent(EVT_GLCANVAS_SCHEDULE_BACKGROUND_PROCESS));
 | ||||||
|  |     wxGetApp().plater()->reslice_SLA_supports(*m_model_object); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| void GLGizmoSlaSupports::editing_mode_reload_cache() | void GLGizmoSlaSupports::editing_mode_reload_cache() | ||||||
| { | { | ||||||
|     m_editing_mode_cache.clear(); |     m_editing_mode_cache.clear(); | ||||||
|  | @ -2497,6 +2470,8 @@ void GLGizmoSlaSupports::editing_mode_reload_cache() | ||||||
|     m_unsaved_changes = false; |     m_unsaved_changes = false; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| void GLGizmoSlaSupports::get_data_from_backend() | void GLGizmoSlaSupports::get_data_from_backend() | ||||||
| { | { | ||||||
|     for (const SLAPrintObject* po : m_parent.sla_print()->objects()) { |     for (const SLAPrintObject* po : m_parent.sla_print()->objects()) { | ||||||
|  | @ -2514,6 +2489,34 @@ void GLGizmoSlaSupports::get_data_from_backend() | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
|  | void GLGizmoSlaSupports::auto_generate() | ||||||
|  | { | ||||||
|  |     wxMessageDialog dlg(GUI::wxGetApp().plater(), _(L( | ||||||
|  |                 "Autogeneration will erase all manually edited points.\n\n" | ||||||
|  |                 "Are you sure you want to do it?\n" | ||||||
|  |                 )), _(L("Warning")), wxICON_WARNING | wxYES | wxNO); | ||||||
|  | 
 | ||||||
|  |     if (m_model_object->sla_support_points.empty() || dlg.ShowModal() == wxID_YES) { | ||||||
|  |         m_model_object->sla_support_points.clear(); | ||||||
|  |         m_editing_mode_cache.clear(); | ||||||
|  |         wxGetApp().plater()->reslice_SLA_supports(*m_model_object); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | void GLGizmoSlaSupports::switch_to_editing_mode() | ||||||
|  | { | ||||||
|  |     editing_mode_reload_cache(); | ||||||
|  |     m_editing_mode = true; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| // GLGizmoCut
 | // GLGizmoCut
 | ||||||
| 
 | 
 | ||||||
| class GLGizmoCutPanel : public wxPanel | class GLGizmoCutPanel : public wxPanel | ||||||
|  |  | ||||||
|  | @ -439,7 +439,6 @@ protected: | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| #define SLAGIZMO_IMGUI_MODAL 0 |  | ||||||
| class GLGizmoSlaSupports : public GLGizmoBase | class GLGizmoSlaSupports : public GLGizmoBase | ||||||
| { | { | ||||||
| private: | private: | ||||||
|  | @ -457,7 +456,7 @@ private: | ||||||
|     igl::AABB<Eigen::MatrixXf,3> m_AABB; |     igl::AABB<Eigen::MatrixXf,3> m_AABB; | ||||||
| 
 | 
 | ||||||
|     struct SourceDataSummary { |     struct SourceDataSummary { | ||||||
|         Vec3d mesh_first_point; |         Geometry::Transformation transformation; | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|     // This holds information to decide whether recalculation is necessary:
 |     // This holds information to decide whether recalculation is necessary:
 | ||||||
|  | @ -491,6 +490,7 @@ private: | ||||||
| 
 | 
 | ||||||
|     bool m_lock_unique_islands = false; |     bool m_lock_unique_islands = false; | ||||||
|     bool m_editing_mode = false; |     bool m_editing_mode = false; | ||||||
|  |     bool m_old_editing_state = false; | ||||||
|     float m_new_point_head_diameter = 0.4f; |     float m_new_point_head_diameter = 0.4f; | ||||||
|     double m_minimal_point_distance = 20.; |     double m_minimal_point_distance = 20.; | ||||||
|     double m_density = 100.; |     double m_density = 100.; | ||||||
|  | @ -504,9 +504,6 @@ private: | ||||||
|     bool m_unsaved_changes = false; |     bool m_unsaved_changes = false; | ||||||
|     bool m_selection_empty = true; |     bool m_selection_empty = true; | ||||||
|     EState m_old_state = Off; // to be able to see that the gizmo has just been closed (see on_set_state)
 |     EState m_old_state = Off; // to be able to see that the gizmo has just been closed (see on_set_state)
 | ||||||
| #if SLAGIZMO_IMGUI_MODAL |  | ||||||
|     bool m_show_modal = false; |  | ||||||
| #endif |  | ||||||
|     int m_canvas_width; |     int m_canvas_width; | ||||||
|     int m_canvas_height; |     int m_canvas_height; | ||||||
| 
 | 
 | ||||||
|  | @ -521,6 +518,8 @@ private: | ||||||
|     void editing_mode_discard_changes(); |     void editing_mode_discard_changes(); | ||||||
|     void editing_mode_reload_cache(); |     void editing_mode_reload_cache(); | ||||||
|     void get_data_from_backend(); |     void get_data_from_backend(); | ||||||
|  |     void auto_generate(); | ||||||
|  |     void switch_to_editing_mode(); | ||||||
| 
 | 
 | ||||||
| protected: | protected: | ||||||
|     void on_set_state() override; |     void on_set_state() override; | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Lukas Matena
						Lukas Matena