mirror of
				https://github.com/SoftFever/OrcaSlicer.git
				synced 2025-10-22 08:11:11 -06:00 
			
		
		
		
	Merge remote-tracking branch 'origin/master' into feature_arrange_with_libnest2d
# Conflicts: # CMakeLists.txt # lib/Slic3r/GUI/MainFrame.pm
This commit is contained in:
		
						commit
						85474e5803
					
				
					 94 changed files with 12419 additions and 4716 deletions
				
			
		
										
											
												File diff suppressed because it is too large
												Load diff
											
										
									
								
							|  | @ -6,8 +6,10 @@ | |||
| #include "../../libslic3r/Line.hpp" | ||||
| #include "../../libslic3r/TriangleMesh.hpp" | ||||
| #include "../../libslic3r/Utils.hpp" | ||||
| #include "../../slic3r/GUI/GLCanvas3DManager.hpp" | ||||
| 
 | ||||
| class wxBitmap; | ||||
| class wxWindow; | ||||
| 
 | ||||
| namespace Slic3r { | ||||
| 
 | ||||
|  | @ -17,6 +19,11 @@ class Model; | |||
| class ModelObject; | ||||
| class GCodePreviewData; | ||||
| class DynamicPrintConfig; | ||||
| class ExtrusionPath; | ||||
| class ExtrusionMultiPath; | ||||
| class ExtrusionLoop; | ||||
| class ExtrusionEntity; | ||||
| class ExtrusionEntityCollection; | ||||
| 
 | ||||
| // A container for interleaved arrays of 3D vertices and normals,
 | ||||
| // possibly indexed by triangles and / or quads.
 | ||||
|  | @ -305,9 +312,9 @@ public: | |||
|     // Boolean: Is mouse over this object?
 | ||||
|     bool                hover; | ||||
|     // Wheter or not this volume has been generated from a modifier
 | ||||
|     bool                 is_modifier; | ||||
|     bool                is_modifier; | ||||
|     // Wheter or not this volume has been generated from the wipe tower
 | ||||
|     bool                 is_wipe_tower; | ||||
|     bool                is_wipe_tower; | ||||
| 
 | ||||
|     // Interleaved triangles & normals with indexed triangles & quads.
 | ||||
|     GLIndexedVertexArray        indexed_vertex_array; | ||||
|  | @ -437,35 +444,6 @@ private: | |||
| 
 | ||||
| class _3DScene | ||||
| { | ||||
|     struct GCodePreviewVolumeIndex | ||||
|     { | ||||
|         enum EType | ||||
|         { | ||||
|             Extrusion, | ||||
|             Travel, | ||||
|             Retraction, | ||||
|             Unretraction, | ||||
|             Shell, | ||||
|             Num_Geometry_Types | ||||
|         }; | ||||
| 
 | ||||
|         struct FirstVolume | ||||
|         { | ||||
|             EType type; | ||||
|             unsigned int flag; | ||||
|             // Index of the first volume in a GLVolumeCollection.
 | ||||
|             unsigned int id; | ||||
| 
 | ||||
|             FirstVolume(EType type, unsigned int flag, unsigned int id) : type(type), flag(flag), id(id) {} | ||||
|         }; | ||||
| 
 | ||||
|         std::vector<FirstVolume> first_volumes; | ||||
| 
 | ||||
|         void reset() { first_volumes.clear(); } | ||||
|     }; | ||||
| 
 | ||||
|     static GCodePreviewVolumeIndex s_gcode_preview_volume_index; | ||||
| 
 | ||||
|     class TextureBase | ||||
|     { | ||||
|     protected: | ||||
|  | @ -525,12 +503,106 @@ class _3DScene | |||
| 
 | ||||
|     static LegendTexture s_legend_texture; | ||||
|     static WarningTexture s_warning_texture; | ||||
|     static GUI::GLCanvas3DManager s_canvas_mgr; | ||||
| 
 | ||||
| public: | ||||
|     static void _glew_init(); | ||||
|     static void init_gl(); | ||||
|     static std::string get_gl_info(bool format_as_html, bool extensions); | ||||
|     static bool use_VBOs(); | ||||
| 
 | ||||
|     static void load_gcode_preview(const Print* print, const GCodePreviewData* preview_data, GLVolumeCollection* volumes, const std::vector<std::string>& str_tool_colors, bool use_VBOs); | ||||
|     static bool add_canvas(wxGLCanvas* canvas); | ||||
|     static bool remove_canvas(wxGLCanvas* canvas); | ||||
|     static void remove_all_canvases(); | ||||
| 
 | ||||
|     static bool init(wxGLCanvas* canvas); | ||||
| 
 | ||||
|     static void set_as_dirty(wxGLCanvas* canvas); | ||||
| 
 | ||||
|     static unsigned int get_volumes_count(wxGLCanvas* canvas); | ||||
|     static void reset_volumes(wxGLCanvas* canvas); | ||||
|     static void deselect_volumes(wxGLCanvas* canvas); | ||||
|     static void select_volume(wxGLCanvas* canvas, unsigned int id); | ||||
|     static void update_volumes_selection(wxGLCanvas* canvas, const std::vector<int>& selections); | ||||
|     static bool check_volumes_outside_state(wxGLCanvas* canvas, const DynamicPrintConfig* config); | ||||
|     static bool move_volume_up(wxGLCanvas* canvas, unsigned int id); | ||||
|     static bool move_volume_down(wxGLCanvas* canvas, unsigned int id); | ||||
| 
 | ||||
|     static void set_objects_selections(wxGLCanvas* canvas, const std::vector<int>& selections); | ||||
| 
 | ||||
|     static void set_config(wxGLCanvas* canvas, DynamicPrintConfig* config); | ||||
|     static void set_print(wxGLCanvas* canvas, Print* print); | ||||
|     static void set_model(wxGLCanvas* canvas, Model* model); | ||||
| 
 | ||||
|     static void set_bed_shape(wxGLCanvas* canvas, const Pointfs& shape); | ||||
|     static void set_auto_bed_shape(wxGLCanvas* canvas); | ||||
| 
 | ||||
|     static BoundingBoxf3 get_volumes_bounding_box(wxGLCanvas* canvas); | ||||
| 
 | ||||
|     static void set_axes_length(wxGLCanvas* canvas, float length); | ||||
| 
 | ||||
|     static void set_cutting_plane(wxGLCanvas* canvas, float z, const ExPolygons& polygons); | ||||
| 
 | ||||
|     static void set_color_by(wxGLCanvas* canvas, const std::string& value); | ||||
|     static void set_select_by(wxGLCanvas* canvas, const std::string& value); | ||||
|     static void set_drag_by(wxGLCanvas* canvas, const std::string& value); | ||||
| 
 | ||||
|     static bool is_layers_editing_enabled(wxGLCanvas* canvas); | ||||
|     static bool is_layers_editing_allowed(wxGLCanvas* canvas); | ||||
|     static bool is_shader_enabled(wxGLCanvas* canvas); | ||||
| 
 | ||||
|     static bool is_reload_delayed(wxGLCanvas* canvas); | ||||
| 
 | ||||
|     static void enable_layers_editing(wxGLCanvas* canvas, bool enable); | ||||
|     static void enable_warning_texture(wxGLCanvas* canvas, bool enable); | ||||
|     static void enable_legend_texture(wxGLCanvas* canvas, bool enable); | ||||
|     static void enable_picking(wxGLCanvas* canvas, bool enable); | ||||
|     static void enable_moving(wxGLCanvas* canvas, bool enable); | ||||
|     static void enable_gizmos(wxGLCanvas* canvas, bool enable); | ||||
|     static void enable_shader(wxGLCanvas* canvas, bool enable); | ||||
|     static void enable_force_zoom_to_bed(wxGLCanvas* canvas, bool enable); | ||||
|     static void allow_multisample(wxGLCanvas* canvas, bool allow); | ||||
| 
 | ||||
|     static void zoom_to_bed(wxGLCanvas* canvas); | ||||
|     static void zoom_to_volumes(wxGLCanvas* canvas); | ||||
|     static void select_view(wxGLCanvas* canvas, const std::string& direction); | ||||
|     static void set_viewport_from_scene(wxGLCanvas* canvas, wxGLCanvas* other); | ||||
| 
 | ||||
|     static void update_volumes_colors_by_extruder(wxGLCanvas* canvas); | ||||
| 
 | ||||
|     static void render(wxGLCanvas* canvas); | ||||
| 
 | ||||
|     static std::vector<double> get_current_print_zs(wxGLCanvas* canvas, bool active_only); | ||||
|     static void set_toolpaths_range(wxGLCanvas* canvas, double low, double high); | ||||
| 
 | ||||
|     static void register_on_viewport_changed_callback(wxGLCanvas* canvas, void* callback); | ||||
|     static void register_on_double_click_callback(wxGLCanvas* canvas, void* callback); | ||||
|     static void register_on_right_click_callback(wxGLCanvas* canvas, void* callback); | ||||
|     static void register_on_select_object_callback(wxGLCanvas* canvas, void* callback); | ||||
|     static void register_on_model_update_callback(wxGLCanvas* canvas, void* callback); | ||||
|     static void register_on_remove_object_callback(wxGLCanvas* canvas, void* callback); | ||||
|     static void register_on_arrange_callback(wxGLCanvas* canvas, void* callback); | ||||
|     static void register_on_rotate_object_left_callback(wxGLCanvas* canvas, void* callback); | ||||
|     static void register_on_rotate_object_right_callback(wxGLCanvas* canvas, void* callback); | ||||
|     static void register_on_scale_object_uniformly_callback(wxGLCanvas* canvas, void* callback); | ||||
|     static void register_on_increase_objects_callback(wxGLCanvas* canvas, void* callback); | ||||
|     static void register_on_decrease_objects_callback(wxGLCanvas* canvas, void* callback); | ||||
|     static void register_on_instance_moved_callback(wxGLCanvas* canvas, void* callback); | ||||
|     static void register_on_wipe_tower_moved_callback(wxGLCanvas* canvas, void* callback); | ||||
|     static void register_on_enable_action_buttons_callback(wxGLCanvas* canvas, void* callback); | ||||
|     static void register_on_gizmo_scale_uniformly_callback(wxGLCanvas* canvas, void* callback); | ||||
| 
 | ||||
|     static std::vector<int> load_object(wxGLCanvas* canvas, const ModelObject* model_object, int obj_idx, std::vector<int> instance_idxs); | ||||
|     static std::vector<int> load_object(wxGLCanvas* canvas, const Model* model, int obj_idx); | ||||
| 
 | ||||
|     static void reload_scene(wxGLCanvas* canvas, bool force); | ||||
| 
 | ||||
|     static void load_print_toolpaths(wxGLCanvas* canvas); | ||||
|     static void load_print_object_toolpaths(wxGLCanvas* canvas, const PrintObject* print_object, const std::vector<std::string>& str_tool_colors); | ||||
|     static void load_wipe_tower_toolpaths(wxGLCanvas* canvas, const std::vector<std::string>& str_tool_colors); | ||||
|     static void load_gcode_preview(wxGLCanvas* canvas, const GCodePreviewData* preview_data, const std::vector<std::string>& str_tool_colors); | ||||
| 
 | ||||
|     // generates the legend texture in dependence of the current shown view type
 | ||||
|     static void generate_legend_texture(const GCodePreviewData& preview_data, const std::vector<float>& tool_colors); | ||||
|     static unsigned int get_legend_texture_width(); | ||||
|     static unsigned int get_legend_texture_height(); | ||||
| 
 | ||||
|  | @ -545,42 +617,16 @@ public: | |||
|     static void reset_warning_texture(); | ||||
|     static unsigned int finalize_warning_texture(); | ||||
| 
 | ||||
|     static void _load_print_toolpaths( | ||||
|         const Print                     *print, | ||||
|         GLVolumeCollection              *volumes, | ||||
|         const std::vector<std::string>  &tool_colors, | ||||
|         bool                             use_VBOs); | ||||
| 
 | ||||
|     static void _load_print_object_toolpaths( | ||||
|         const PrintObject               *print_object, | ||||
|         GLVolumeCollection              *volumes, | ||||
|         const std::vector<std::string>  &tool_colors, | ||||
|         bool                             use_VBOs); | ||||
| 
 | ||||
|     static void _load_wipe_tower_toolpaths( | ||||
|         const Print                    *print, | ||||
|         GLVolumeCollection             *volumes, | ||||
|         const std::vector<std::string> &tool_colors_str, | ||||
|         bool                            use_VBOs); | ||||
| 
 | ||||
| private: | ||||
|     // generates gcode extrusion paths geometry
 | ||||
|     static void _load_gcode_extrusion_paths(const GCodePreviewData& preview_data, GLVolumeCollection& volumes, const std::vector<float>& tool_colors, bool use_VBOs); | ||||
|     // generates gcode travel paths geometry
 | ||||
|     static void _load_gcode_travel_paths(const GCodePreviewData& preview_data, GLVolumeCollection& volumes, const std::vector<float>& tool_colors, bool use_VBOs); | ||||
|     static bool _travel_paths_by_type(const GCodePreviewData& preview_data, GLVolumeCollection& volumes); | ||||
|     static bool _travel_paths_by_feedrate(const GCodePreviewData& preview_data, GLVolumeCollection& volumes); | ||||
|     static bool _travel_paths_by_tool(const GCodePreviewData& preview_data, GLVolumeCollection& volumes, const std::vector<float>& tool_colors); | ||||
|     // generates gcode retractions geometry
 | ||||
|     static void _load_gcode_retractions(const GCodePreviewData& preview_data, GLVolumeCollection& volumes, bool use_VBOs); | ||||
|     // generates gcode unretractions geometry
 | ||||
|     static void _load_gcode_unretractions(const GCodePreviewData& preview_data, GLVolumeCollection& volumes, bool use_VBOs); | ||||
|     // sets gcode geometry visibility according to user selection
 | ||||
|     static void _update_gcode_volumes_visibility(const GCodePreviewData& preview_data, GLVolumeCollection& volumes); | ||||
|     // generates the legend texture in dependence of the current shown view type
 | ||||
|     static void _generate_legend_texture(const GCodePreviewData& preview_data, const std::vector<float>& tool_colors); | ||||
|     // generates objects and wipe tower geometry
 | ||||
|     static void _load_shells(const Print& print, GLVolumeCollection& volumes, bool use_VBOs); | ||||
|     static void thick_lines_to_verts(const Lines& lines, const std::vector<double>& widths, const std::vector<double>& heights, bool closed, double top_z, GLVolume& volume); | ||||
|     static void thick_lines_to_verts(const Lines3& lines, const std::vector<double>& widths, const std::vector<double>& heights, bool closed, GLVolume& volume); | ||||
|     static void extrusionentity_to_verts(const ExtrusionPath& extrusion_path, float print_z, GLVolume& volume); | ||||
|     static void extrusionentity_to_verts(const ExtrusionPath& extrusion_path, float print_z, const Point& copy, GLVolume& volume); | ||||
|     static void extrusionentity_to_verts(const ExtrusionLoop& extrusion_loop, float print_z, const Point& copy, GLVolume& volume); | ||||
|     static void extrusionentity_to_verts(const ExtrusionMultiPath& extrusion_multi_path, float print_z, const Point& copy, GLVolume& volume); | ||||
|     static void extrusionentity_to_verts(const ExtrusionEntityCollection& extrusion_entity_collection, float print_z, const Point& copy, GLVolume& volume); | ||||
|     static void extrusionentity_to_verts(const ExtrusionEntity* extrusion_entity, float print_z, const Point& copy, GLVolume& volume); | ||||
|     static void polyline3_to_verts(const Polyline3& polyline, double width, double height, GLVolume& volume); | ||||
|     static void point3_to_verts(const Point3& point, double width, double height, GLVolume& volume); | ||||
| }; | ||||
| 
 | ||||
| } | ||||
|  |  | |||
|  | @ -83,8 +83,11 @@ PrinterPicker::PrinterPicker(wxWindow *parent, const VendorProfile &vendor, cons | |||
| 
 | ||||
| 		const auto model_id = model.id; | ||||
| 
 | ||||
| 		bool default_variant = true;   // Mark the first variant as default in the GUI
 | ||||
| 		for (const auto &variant : model.variants) { | ||||
| 			const auto label = wxString::Format("%s %s %s", variant.name, _(L("mm")), _(L("nozzle"))); | ||||
| 			const auto label = wxString::Format("%s %s %s %s", variant.name, _(L("mm")), _(L("nozzle")), | ||||
| 				(default_variant ? _(L("(default)")) : wxString())); | ||||
| 			default_variant = false; | ||||
| 			auto *cbox = new Checkbox(panel, label, model_id, variant.name); | ||||
| 			const size_t idx = cboxes.size(); | ||||
| 			cboxes.push_back(cbox); | ||||
|  | @ -125,6 +128,14 @@ void PrinterPicker::select_all(bool select) | |||
| 	} | ||||
| } | ||||
| 
 | ||||
| void PrinterPicker::select_one(size_t i, bool select) | ||||
| { | ||||
| 	if (i < cboxes.size() && cboxes[i]->GetValue() != select) { | ||||
| 		cboxes[i]->SetValue(select); | ||||
| 		on_checkbox(cboxes[i], select); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| void PrinterPicker::on_checkbox(const Checkbox *cbox, bool checked) | ||||
| { | ||||
| 	variants_checked += checked ? 1 : -1; | ||||
|  | @ -227,6 +238,7 @@ PageWelcome::PageWelcome(ConfigWizard *parent) : | |||
| 		AppConfig &appconfig_vendors = this->wizard_p()->appconfig_vendors; | ||||
| 
 | ||||
| 		printer_picker = new PrinterPicker(this, vendor_prusa->second, appconfig_vendors); | ||||
| 		printer_picker->select_one(0, true);    // Select the default (first) model/variant on the Prusa vendor
 | ||||
| 		printer_picker->Bind(EVT_PRINTER_PICK, [this, &appconfig_vendors](const PrinterPickerEvent &evt) { | ||||
| 			appconfig_vendors.set_variant(evt.vendor_id, evt.model_id, evt.variant_name, evt.enable); | ||||
| 			this->on_variant_checked(); | ||||
|  | @ -598,10 +610,10 @@ void ConfigWizardIndex::on_paint(wxPaintEvent & evt) | |||
| 
 | ||||
| static const std::unordered_map<std::string, std::pair<std::string, std::string>> legacy_preset_map {{ | ||||
| 	{ "Original Prusa i3 MK2.ini",                           std::make_pair("MK2S", "0.4") }, | ||||
| 	{ "Original Prusa i3 MK2 MM Single Mode.ini",            std::make_pair("MK2S", "0.4") }, | ||||
| 	{ "Original Prusa i3 MK2 MM Single Mode 0.6 nozzle.ini", std::make_pair("MK2S", "0.6") }, | ||||
| 	{ "Original Prusa i3 MK2 MultiMaterial.ini",             std::make_pair("MK2S", "0.4") }, | ||||
| 	{ "Original Prusa i3 MK2 MultiMaterial 0.6 nozzle.ini",  std::make_pair("MK2S", "0.6") }, | ||||
| 	{ "Original Prusa i3 MK2 MM Single Mode.ini",            std::make_pair("MK2SMM", "0.4") }, | ||||
| 	{ "Original Prusa i3 MK2 MM Single Mode 0.6 nozzle.ini", std::make_pair("MK2SMM", "0.6") }, | ||||
| 	{ "Original Prusa i3 MK2 MultiMaterial.ini",             std::make_pair("MK2SMM", "0.4") }, | ||||
| 	{ "Original Prusa i3 MK2 MultiMaterial 0.6 nozzle.ini",  std::make_pair("MK2SMM", "0.6") }, | ||||
| 	{ "Original Prusa i3 MK2 0.25 nozzle.ini",               std::make_pair("MK2S", "0.25") }, | ||||
| 	{ "Original Prusa i3 MK2 0.6 nozzle.ini",                std::make_pair("MK2S", "0.6") }, | ||||
| 	{ "Original Prusa i3 MK3.ini",                           std::make_pair("MK3",  "0.4") }, | ||||
|  | @ -809,8 +821,8 @@ ConfigWizard::ConfigWizard(wxWindow *parent, RunReason reason) : | |||
| 	topsizer->AddSpacer(INDEX_MARGIN); | ||||
| 	topsizer->Add(p->hscroll, 1, wxEXPAND); | ||||
| 
 | ||||
| 	p->btn_prev = new wxButton(this, wxID_BACKWARD); | ||||
| 	p->btn_next = new wxButton(this, wxID_FORWARD); | ||||
| 	p->btn_prev = new wxButton(this, wxID_NONE, _(L("< &Back"))); | ||||
| 	p->btn_next = new wxButton(this, wxID_NONE, _(L("&Next >"))); | ||||
| 	p->btn_finish = new wxButton(this, wxID_APPLY, _(L("&Finish"))); | ||||
| 	p->btn_cancel = new wxButton(this, wxID_CANCEL); | ||||
| 	p->btnsizer->AddStretchSpacer(); | ||||
|  | @ -881,9 +893,9 @@ const wxString& ConfigWizard::name() | |||
| { | ||||
| 	// A different naming convention is used for the Wizard on Windows vs. OSX & GTK.
 | ||||
| #if WIN32 | ||||
| 	static const wxString config_wizard_name = _(L("Configuration Wizard")); | ||||
| 	static const wxString config_wizard_name = L("Configuration Wizard"); | ||||
| #else | ||||
| 	static const wxString config_wizard_name = _(L("Configuration Assistant")); | ||||
| 	static const wxString config_wizard_name = L("Configuration Assistant"); | ||||
| #endif | ||||
| 	return config_wizard_name; | ||||
| } | ||||
|  |  | |||
|  | @ -56,6 +56,7 @@ struct PrinterPicker: wxPanel | |||
| 	PrinterPicker(wxWindow *parent, const VendorProfile &vendor, const AppConfig &appconfig_vendors); | ||||
| 
 | ||||
| 	void select_all(bool select); | ||||
| 	void select_one(size_t i, bool select); | ||||
| 	void on_checkbox(const Checkbox *cbox, bool checked); | ||||
| }; | ||||
| 
 | ||||
|  |  | |||
|  | @ -12,10 +12,20 @@ namespace Slic3r { namespace GUI { | |||
| 
 | ||||
| 	wxString double_to_string(double const value) | ||||
| 	{ | ||||
| 		int precision = 10 * value - int(10 * value) == 0 ? 1 : 2; | ||||
| 		return value - int(value) == 0 ? | ||||
| 			wxString::Format(_T("%i"), int(value)) : | ||||
| 			wxNumberFormatter::ToString(value, precision, wxNumberFormatter::Style_None); | ||||
| 		if (value - int(value) == 0) | ||||
| 			return wxString::Format(_T("%i"), int(value)); | ||||
| 		else { | ||||
| 			int precision = 4; | ||||
| 			for (size_t p = 1; p < 4; p++) | ||||
| 			{ | ||||
| 				double cur_val = pow(10, p)*value; | ||||
| 				if (cur_val - int(cur_val) == 0) { | ||||
| 					precision = p; | ||||
| 					break; | ||||
| 				} | ||||
| 			} | ||||
| 			return wxNumberFormatter::ToString(value, precision, wxNumberFormatter::Style_None); | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	void Field::PostInitialize(){ | ||||
|  |  | |||
|  | @ -4,12 +4,14 @@ | |||
| #include <algorithm> | ||||
| #include <boost/format.hpp> | ||||
| #include <boost/filesystem/path.hpp> | ||||
| #include <boost/filesystem/fstream.hpp> | ||||
| #include <boost/log/trivial.hpp> | ||||
| 
 | ||||
| #include <wx/app.h> | ||||
| #include <wx/event.h> | ||||
| #include <wx/sizer.h> | ||||
| #include <wx/settings.h> | ||||
| #include <wx/timer.h> | ||||
| #include <wx/panel.h> | ||||
| #include <wx/button.h> | ||||
| #include <wx/filepicker.h> | ||||
|  | @ -36,7 +38,7 @@ namespace Slic3r { | |||
| enum AvrdudeEvent | ||||
| { | ||||
| 	AE_MESSAGE, | ||||
| 	AE_PRORGESS, | ||||
| 	AE_PROGRESS, | ||||
| 	AE_EXIT, | ||||
| }; | ||||
| 
 | ||||
|  | @ -62,7 +64,6 @@ struct FirmwareDialog::priv | |||
| 	std::vector<Utils::SerialPortInfo> ports; | ||||
| 	wxFilePickerCtrl *hex_picker; | ||||
| 	wxStaticText *txt_status; | ||||
| 	wxStaticText *txt_progress; | ||||
| 	wxGauge *progressbar; | ||||
| 	wxCollapsiblePane *spoiler; | ||||
| 	wxTextCtrl *txt_stdout; | ||||
|  | @ -72,6 +73,8 @@ struct FirmwareDialog::priv | |||
| 	wxString btn_flash_label_ready; | ||||
| 	wxString btn_flash_label_flashing; | ||||
| 
 | ||||
| 	wxTimer timer_pulse; | ||||
| 
 | ||||
| 	// This is a shared pointer holding the background AvrDude task
 | ||||
| 	// also serves as a status indication (it is set _iff_ the background task is running, otherwise it is reset).
 | ||||
| 	AvrDude::Ptr avrdude; | ||||
|  | @ -83,13 +86,16 @@ struct FirmwareDialog::priv | |||
| 		q(q), | ||||
| 		btn_flash_label_ready(_(L("Flash!"))), | ||||
| 		btn_flash_label_flashing(_(L("Cancel"))), | ||||
| 		timer_pulse(q), | ||||
| 		avrdude_config((fs::path(::Slic3r::resources_dir()) / "avrdude" / "avrdude.conf").string()), | ||||
| 		progress_tasks_done(0), | ||||
| 		cancelled(false) | ||||
| 	{} | ||||
| 
 | ||||
| 	void find_serial_ports(); | ||||
| 	void flashing_status(bool flashing, AvrDudeComplete complete = AC_NONE); | ||||
| 	void flashing_start(bool flashing_l10n); | ||||
| 	void flashing_done(AvrDudeComplete complete); | ||||
| 	size_t hex_lang_offset(const wxString &path); | ||||
| 	void perform_upload(); | ||||
| 	void cancel(); | ||||
| 	void on_avrdude(const wxCommandEvent &evt); | ||||
|  | @ -116,42 +122,76 @@ void FirmwareDialog::priv::find_serial_ports() | |||
| 	} | ||||
| } | ||||
| 
 | ||||
| void FirmwareDialog::priv::flashing_status(bool value, AvrDudeComplete complete) | ||||
| void FirmwareDialog::priv::flashing_start(bool flashing_l10n) | ||||
| { | ||||
| 	if (value) { | ||||
| 		txt_stdout->Clear(); | ||||
| 		txt_status->SetLabel(_(L("Flashing in progress. Please do not disconnect the printer!"))); | ||||
| 		txt_status->SetForegroundColour(GUI::get_label_clr_modified()); | ||||
| 		port_picker->Disable(); | ||||
| 		btn_rescan->Disable(); | ||||
| 		hex_picker->Disable(); | ||||
| 		btn_close->Disable(); | ||||
| 		btn_flash->SetLabel(btn_flash_label_flashing); | ||||
| 		progressbar->SetRange(200);   // See progress callback below
 | ||||
| 		progressbar->SetValue(0); | ||||
| 		progress_tasks_done = 0; | ||||
| 		cancelled = false; | ||||
| 	} else { | ||||
| 		auto text_color = wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOWTEXT); | ||||
| 		port_picker->Enable(); | ||||
| 		btn_rescan->Enable(); | ||||
| 		hex_picker->Enable(); | ||||
| 		btn_close->Enable(); | ||||
| 		btn_flash->SetLabel(btn_flash_label_ready); | ||||
| 		txt_status->SetForegroundColour(text_color); | ||||
| 		progressbar->SetValue(200); | ||||
| 	txt_stdout->Clear(); | ||||
| 	txt_status->SetLabel(_(L("Flashing in progress. Please do not disconnect the printer!"))); | ||||
| 	txt_status->SetForegroundColour(GUI::get_label_clr_modified()); | ||||
| 	port_picker->Disable(); | ||||
| 	btn_rescan->Disable(); | ||||
| 	hex_picker->Disable(); | ||||
| 	btn_close->Disable(); | ||||
| 	btn_flash->SetLabel(btn_flash_label_flashing); | ||||
| 	progressbar->SetRange(flashing_l10n ? 500 : 200);   // See progress callback below
 | ||||
| 	progressbar->SetValue(0); | ||||
| 	progress_tasks_done = 0; | ||||
| 	cancelled = false; | ||||
| 	timer_pulse.Start(50); | ||||
| } | ||||
| 
 | ||||
| 		switch (complete) { | ||||
| 		case AC_SUCCESS: txt_status->SetLabel(_(L("Flashing succeeded!"))); break; | ||||
| 		case AC_FAILURE: txt_status->SetLabel(_(L("Flashing failed. Please see the avrdude log below."))); break; | ||||
| 		case AC_CANCEL: txt_status->SetLabel(_(L("Flashing cancelled."))); break; | ||||
| void FirmwareDialog::priv::flashing_done(AvrDudeComplete complete) | ||||
| { | ||||
| 	auto text_color = wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOWTEXT); | ||||
| 	port_picker->Enable(); | ||||
| 	btn_rescan->Enable(); | ||||
| 	hex_picker->Enable(); | ||||
| 	btn_close->Enable(); | ||||
| 	btn_flash->SetLabel(btn_flash_label_ready); | ||||
| 	txt_status->SetForegroundColour(text_color); | ||||
| 	timer_pulse.Stop(); | ||||
| 	progressbar->SetValue(progressbar->GetRange()); | ||||
| 
 | ||||
| 	switch (complete) { | ||||
| 	case AC_SUCCESS: txt_status->SetLabel(_(L("Flashing succeeded!"))); break; | ||||
| 	case AC_FAILURE: txt_status->SetLabel(_(L("Flashing failed. Please see the avrdude log below."))); break; | ||||
| 	case AC_CANCEL: txt_status->SetLabel(_(L("Flashing cancelled."))); break; | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| size_t FirmwareDialog::priv::hex_lang_offset(const wxString &path) | ||||
| { | ||||
| 	fs::ifstream file(fs::path(path.wx_str())); | ||||
| 	if (! file.good()) { | ||||
| 		return 0; | ||||
| 	} | ||||
| 
 | ||||
| 	static const char *hex_terminator = ":00000001FF\r"; | ||||
| 	size_t res = 0; | ||||
| 	std::string line; | ||||
| 	while (getline(file, line, '\n').good()) { | ||||
| 		// Account for LF vs CRLF
 | ||||
| 		if (!line.empty() && line.back() != '\r') { | ||||
| 			line.push_back('\r'); | ||||
| 		} | ||||
| 
 | ||||
| 		if (line == hex_terminator) { | ||||
| 			if (res == 0) { | ||||
| 				// This is the first terminator seen, save the position
 | ||||
| 				res = file.tellg(); | ||||
| 			} else { | ||||
| 				// We've found another terminator, return the offset just after the first one
 | ||||
| 				// which is the start of the second 'section'.
 | ||||
| 				return res; | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| void FirmwareDialog::priv::perform_upload() | ||||
| { | ||||
| 	auto filename  = hex_picker->GetPath(); | ||||
| 	auto filename = hex_picker->GetPath(); | ||||
| 	std::string port = port_picker->GetValue().ToStdString(); | ||||
| 	int  selection = port_picker->GetSelection(); | ||||
| 	if (selection != -1) { | ||||
|  | @ -161,16 +201,32 @@ void FirmwareDialog::priv::perform_upload() | |||
| 	} | ||||
| 	if (filename.IsEmpty() || port.empty()) { return; } | ||||
| 
 | ||||
| 	flashing_status(true); | ||||
| 	const bool extra_verbose = false;   // For debugging
 | ||||
| 	const auto lang_offset = hex_lang_offset(filename); | ||||
| 	const auto filename_utf8 = filename.utf8_str(); | ||||
| 
 | ||||
| 	flashing_start(lang_offset > 0); | ||||
| 
 | ||||
| 	// It is ok here to use the q-pointer to the FirmwareDialog
 | ||||
| 	// because the dialog ensures it doesn't exit before the background thread is done.
 | ||||
| 	auto q = this->q; | ||||
| 
 | ||||
| 	// Init the avrdude object
 | ||||
| 	AvrDude avrdude(avrdude_config); | ||||
| 
 | ||||
| 	// Build argument list(s)
 | ||||
| 	std::vector<std::string> args {{ | ||||
| 		"-v", | ||||
| 		extra_verbose ? "-vvvvv" : "-v", | ||||
| 		"-p", "atmega2560", | ||||
| 		// Using the "Wiring" mode to program Rambo or Einsy, using the STK500v2 protocol (not the STK500).
 | ||||
| 		// The Prusa's avrdude is patched to never send semicolons inside the data packets, as the USB to serial chip
 | ||||
| 		// is flashed with a buggy firmware.
 | ||||
| 		"-c", "wiring", | ||||
| 		"-P", port, | ||||
| 		"-b", "115200",   // XXX: is this ok to hardcode?
 | ||||
| 		"-b", "115200",   // TODO: Allow other rates? Ditto below.
 | ||||
| 		"-D", | ||||
| 		"-U", (boost::format("flash:w:%1%:i") % filename.ToStdString()).str() | ||||
| 		// XXX: Safe mode?
 | ||||
| 		"-U", (boost::format("flash:w:0:%1%:i") % filename_utf8.data()).str(), | ||||
| 	}}; | ||||
| 
 | ||||
| 	BOOST_LOG_TRIVIAL(info) << "Invoking avrdude, arguments: " | ||||
|  | @ -178,26 +234,51 @@ void FirmwareDialog::priv::perform_upload() | |||
| 			return a + ' ' + b; | ||||
| 		}); | ||||
| 
 | ||||
| 	// It is ok here to use the q-pointer to the FirmwareDialog
 | ||||
| 	// because the dialog ensures it doesn't exit before the background thread is done.
 | ||||
| 	auto q = this->q; | ||||
| 	avrdude.push_args(std::move(args)); | ||||
| 	 | ||||
| 	if (lang_offset > 0) { | ||||
| 		// The hex file also contains another section with l10n data to be flashed into the external flash on MK3 (Einsy)
 | ||||
| 		// This is done via another avrdude invocation, here we build arg list for that:
 | ||||
| 		std::vector<std::string> args_l10n {{ | ||||
| 			extra_verbose ? "-vvvvv" : "-v", | ||||
| 			"-p", "atmega2560", | ||||
| 			// Using the "Arduino" mode to program Einsy's external flash with languages, using the STK500 protocol (not the STK500v2).
 | ||||
| 			// The Prusa's avrdude is patched again to never send semicolons inside the data packets.
 | ||||
| 			"-c", "arduino", | ||||
| 			"-P", port, | ||||
| 			"-b", "115200", | ||||
| 			"-D", | ||||
| 			"-u", // disable safe mode
 | ||||
| 			"-U", (boost::format("flash:w:%1%:%2%:i") % lang_offset % filename_utf8.data()).str(), | ||||
| 		}}; | ||||
| 
 | ||||
| 		BOOST_LOG_TRIVIAL(info) << "Invoking avrdude for external flash flashing, arguments: " | ||||
| 			<< std::accumulate(std::next(args_l10n.begin()), args_l10n.end(), args_l10n[0], [](std::string a, const std::string &b) { | ||||
| 				return a + ' ' + b; | ||||
| 			}); | ||||
| 
 | ||||
| 		avrdude.push_args(std::move(args_l10n)); | ||||
| 	} | ||||
| 	 | ||||
| 	this->avrdude = avrdude | ||||
| 		.on_message(std::move([q, extra_verbose](const char *msg, unsigned /* size */) { | ||||
| 			if (extra_verbose) { | ||||
| 				BOOST_LOG_TRIVIAL(debug) << "avrdude: " << msg; | ||||
| 			} | ||||
| 
 | ||||
| 	avrdude = AvrDude() | ||||
| 		.sys_config(avrdude_config) | ||||
| 		.args(args) | ||||
| 		.on_message(std::move([q](const char *msg, unsigned /* size */) { | ||||
| 			auto evt = new wxCommandEvent(EVT_AVRDUDE, q->GetId()); | ||||
| 			auto wxmsg = wxString::FromUTF8(msg); | ||||
| 			evt->SetExtraLong(AE_MESSAGE); | ||||
| 			evt->SetString(msg); | ||||
| 			evt->SetString(std::move(wxmsg)); | ||||
| 			wxQueueEvent(q, evt); | ||||
| 		})) | ||||
| 		.on_progress(std::move([q](const char * /* task */, unsigned progress) { | ||||
| 			auto evt = new wxCommandEvent(EVT_AVRDUDE, q->GetId()); | ||||
| 			evt->SetExtraLong(AE_PRORGESS); | ||||
| 			evt->SetExtraLong(AE_PROGRESS); | ||||
| 			evt->SetInt(progress); | ||||
| 			wxQueueEvent(q, evt); | ||||
| 		})) | ||||
| 		.on_complete(std::move([q](int status) { | ||||
| 		.on_complete(std::move([q](int status, size_t /* args_id */) { | ||||
| 			auto evt = new wxCommandEvent(EVT_AVRDUDE, q->GetId()); | ||||
| 			evt->SetExtraLong(AE_EXIT); | ||||
| 			evt->SetInt(status); | ||||
|  | @ -224,19 +305,19 @@ void FirmwareDialog::priv::on_avrdude(const wxCommandEvent &evt) | |||
| 		txt_stdout->AppendText(evt.GetString()); | ||||
| 		break; | ||||
| 
 | ||||
| 	case AE_PRORGESS: | ||||
| 	case AE_PROGRESS: | ||||
| 		// We try to track overall progress here.
 | ||||
| 		// When uploading the firmware, avrdude first reads a littlebit of status data,
 | ||||
| 		// then performs write, then reading (verification).
 | ||||
| 		// We Pulse() during the first read and combine progress of the latter two tasks.
 | ||||
| 		// Avrdude performs 3 tasks per one memory operation ("-U" arg),
 | ||||
| 		// first of which is reading of status data (very short).
 | ||||
| 		// We use the timer_pulse during the very first task to indicate intialization
 | ||||
| 		// and then display overall progress during the latter tasks.
 | ||||
| 
 | ||||
| 		if (progress_tasks_done == 0) { | ||||
| 			progressbar->Pulse(); | ||||
| 		} else { | ||||
| 		if (progress_tasks_done > 0) { | ||||
| 			progressbar->SetValue(progress_tasks_done - 100 + evt.GetInt()); | ||||
| 		} | ||||
| 
 | ||||
| 		if (evt.GetInt() == 100) { | ||||
| 			timer_pulse.Stop(); | ||||
| 			progress_tasks_done += 100; | ||||
| 		} | ||||
| 
 | ||||
|  | @ -246,7 +327,7 @@ void FirmwareDialog::priv::on_avrdude(const wxCommandEvent &evt) | |||
| 		BOOST_LOG_TRIVIAL(info) << "avrdude exit code: " << evt.GetInt(); | ||||
| 
 | ||||
| 		complete_kind = cancelled ? AC_CANCEL : (evt.GetInt() == 0 ? AC_SUCCESS : AC_FAILURE); | ||||
| 		flashing_status(false, complete_kind); | ||||
| 		flashing_done(complete_kind); | ||||
| 
 | ||||
| 		// Make sure the background thread is collected and the AvrDude object reset
 | ||||
| 		if (avrdude) { avrdude->join(); } | ||||
|  | @ -374,6 +455,8 @@ FirmwareDialog::FirmwareDialog(wxWindow *parent) : | |||
| 		} | ||||
| 	}); | ||||
| 
 | ||||
| 	Bind(wxEVT_TIMER, [this](wxTimerEvent &evt) { this->p->progressbar->Pulse(); }); | ||||
| 
 | ||||
| 	Bind(EVT_AVRDUDE, [this](wxCommandEvent &evt) { this->p->on_avrdude(evt); }); | ||||
| 
 | ||||
| 	Bind(wxEVT_CLOSE_WINDOW, [this](wxCloseEvent &evt) { | ||||
|  |  | |||
							
								
								
									
										4346
									
								
								xs/src/slic3r/GUI/GLCanvas3D.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										4346
									
								
								xs/src/slic3r/GUI/GLCanvas3D.cpp
									
										
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load diff
											
										
									
								
							
							
								
								
									
										636
									
								
								xs/src/slic3r/GUI/GLCanvas3D.hpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										636
									
								
								xs/src/slic3r/GUI/GLCanvas3D.hpp
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,636 @@ | |||
| #ifndef slic3r_GLCanvas3D_hpp_ | ||||
| #define slic3r_GLCanvas3D_hpp_ | ||||
| 
 | ||||
| #include "../../slic3r/GUI/3DScene.hpp" | ||||
| #include "../../slic3r/GUI/GLTexture.hpp" | ||||
| 
 | ||||
| class wxTimer; | ||||
| class wxSizeEvent; | ||||
| class wxIdleEvent; | ||||
| class wxKeyEvent; | ||||
| class wxMouseEvent; | ||||
| class wxTimerEvent; | ||||
| class wxPaintEvent; | ||||
| 
 | ||||
| namespace Slic3r { | ||||
| 
 | ||||
| class GLShader; | ||||
| class ExPolygon; | ||||
| 
 | ||||
| namespace GUI { | ||||
| 
 | ||||
| class GLGizmoBase; | ||||
| 
 | ||||
| class GeometryBuffer | ||||
| { | ||||
|     std::vector<float> m_vertices; | ||||
|     std::vector<float> m_tex_coords; | ||||
| 
 | ||||
| public: | ||||
|     bool set_from_triangles(const Polygons& triangles, float z, bool generate_tex_coords); | ||||
|     bool set_from_lines(const Lines& lines, float z); | ||||
| 
 | ||||
|     const float* get_vertices() const; | ||||
|     const float* get_tex_coords() const; | ||||
| 
 | ||||
|     unsigned int get_vertices_count() const; | ||||
| }; | ||||
| 
 | ||||
| class Size | ||||
| { | ||||
|     int m_width; | ||||
|     int m_height; | ||||
| 
 | ||||
| public: | ||||
|     Size(); | ||||
|     Size(int width, int height); | ||||
| 
 | ||||
|     int get_width() const; | ||||
|     void set_width(int width); | ||||
| 
 | ||||
|     int get_height() const; | ||||
|     void set_height(int height); | ||||
| }; | ||||
| 
 | ||||
| class Rect | ||||
| { | ||||
|     float m_left; | ||||
|     float m_top; | ||||
|     float m_right; | ||||
|     float m_bottom; | ||||
| 
 | ||||
| public: | ||||
|     Rect(); | ||||
|     Rect(float left, float top, float right, float bottom); | ||||
| 
 | ||||
|     float get_left() const; | ||||
|     void set_left(float left); | ||||
| 
 | ||||
|     float get_top() const; | ||||
|     void set_top(float top); | ||||
| 
 | ||||
|     float get_right() const; | ||||
|     void set_right(float right); | ||||
| 
 | ||||
|     float get_bottom() const; | ||||
|     void set_bottom(float bottom); | ||||
| }; | ||||
| 
 | ||||
| class GLCanvas3D | ||||
| { | ||||
|     struct GCodePreviewVolumeIndex | ||||
|     { | ||||
|         enum EType | ||||
|         { | ||||
|             Extrusion, | ||||
|             Travel, | ||||
|             Retraction, | ||||
|             Unretraction, | ||||
|             Shell, | ||||
|             Num_Geometry_Types | ||||
|         }; | ||||
| 
 | ||||
|         struct FirstVolume | ||||
|         { | ||||
|             EType type; | ||||
|             unsigned int flag; | ||||
|             // Index of the first volume in a GLVolumeCollection.
 | ||||
|             unsigned int id; | ||||
| 
 | ||||
|             FirstVolume(EType type, unsigned int flag, unsigned int id) : type(type), flag(flag), id(id) {} | ||||
|         }; | ||||
| 
 | ||||
|         std::vector<FirstVolume> first_volumes; | ||||
| 
 | ||||
|         void reset() { first_volumes.clear(); } | ||||
|     }; | ||||
| 
 | ||||
| public: | ||||
|     struct Camera | ||||
|     { | ||||
|         enum EType : unsigned char | ||||
|         { | ||||
|             Unknown, | ||||
| //            Perspective,
 | ||||
|             Ortho, | ||||
|             Num_types | ||||
|         }; | ||||
| 
 | ||||
|         EType type; | ||||
|         float zoom; | ||||
|         float phi; | ||||
| //        float distance;
 | ||||
|         Pointf3 target; | ||||
| 
 | ||||
|     private: | ||||
|         float m_theta; | ||||
| 
 | ||||
|     public: | ||||
|         Camera(); | ||||
| 
 | ||||
|         std::string get_type_as_string() const; | ||||
| 
 | ||||
|         float get_theta() const; | ||||
|         void set_theta(float theta); | ||||
|     }; | ||||
| 
 | ||||
|     class Bed | ||||
|     { | ||||
|     public: | ||||
|         enum EType : unsigned char | ||||
|         { | ||||
|             MK2, | ||||
|             MK3, | ||||
|             Custom, | ||||
|             Num_Types | ||||
|         }; | ||||
| 
 | ||||
|     private: | ||||
|         EType m_type; | ||||
|         Pointfs m_shape; | ||||
|         BoundingBoxf3 m_bounding_box; | ||||
|         Polygon m_polygon; | ||||
|         GeometryBuffer m_triangles; | ||||
|         GeometryBuffer m_gridlines; | ||||
|         mutable GLTexture m_top_texture; | ||||
|         mutable GLTexture m_bottom_texture; | ||||
| 
 | ||||
|     public: | ||||
|         Bed(); | ||||
| 
 | ||||
|         bool is_prusa() const; | ||||
|         bool is_custom() const; | ||||
| 
 | ||||
|         const Pointfs& get_shape() const; | ||||
|         void set_shape(const Pointfs& shape); | ||||
| 
 | ||||
|         const BoundingBoxf3& get_bounding_box() const; | ||||
|         bool contains(const Point& point) const; | ||||
|         Point point_projection(const Point& point) const; | ||||
| 
 | ||||
|         void render(float theta) const; | ||||
| 
 | ||||
|     private: | ||||
|         void _calc_bounding_box(); | ||||
|         void _calc_triangles(const ExPolygon& poly); | ||||
|         void _calc_gridlines(const ExPolygon& poly, const BoundingBox& bed_bbox); | ||||
|         EType _detect_type() const; | ||||
|         void _render_mk2(float theta) const; | ||||
|         void _render_mk3(float theta) const; | ||||
|         void _render_prusa(float theta) const; | ||||
|         void _render_custom() const; | ||||
|         static bool _are_equal(const Pointfs& bed_1, const Pointfs& bed_2); | ||||
|     }; | ||||
| 
 | ||||
|     struct Axes | ||||
|     { | ||||
|         Pointf3 origin; | ||||
|         float length; | ||||
| 
 | ||||
|         Axes(); | ||||
| 
 | ||||
|         void render(bool depth_test) const; | ||||
|     }; | ||||
| 
 | ||||
|     class CuttingPlane | ||||
|     { | ||||
|         float m_z; | ||||
|         GeometryBuffer m_lines; | ||||
| 
 | ||||
|     public: | ||||
|         CuttingPlane(); | ||||
| 
 | ||||
|         bool set(float z, const ExPolygons& polygons); | ||||
| 
 | ||||
|         void render(const BoundingBoxf3& bb) const; | ||||
| 
 | ||||
|     private: | ||||
|         void _render_plane(const BoundingBoxf3& bb) const; | ||||
|         void _render_contour() const; | ||||
|     }; | ||||
| 
 | ||||
|     class Shader | ||||
|     { | ||||
|         GLShader* m_shader; | ||||
| 
 | ||||
|     public: | ||||
|         Shader(); | ||||
|         ~Shader(); | ||||
| 
 | ||||
|         bool init(const std::string& vertex_shader_filename, const std::string& fragment_shader_filename); | ||||
| 
 | ||||
|         bool is_initialized() const; | ||||
| 
 | ||||
|         bool start_using() const; | ||||
|         void stop_using() const; | ||||
| 
 | ||||
|         void set_uniform(const std::string& name, float value) const; | ||||
| 
 | ||||
|         const GLShader* get_shader() const; | ||||
| 
 | ||||
|     private: | ||||
|         void _reset(); | ||||
|     }; | ||||
| 
 | ||||
|     class LayersEditing | ||||
|     { | ||||
|     public: | ||||
|         enum EState : unsigned char | ||||
|         { | ||||
|             Unknown, | ||||
|             Editing, | ||||
|             Completed, | ||||
|             Num_States | ||||
|         }; | ||||
| 
 | ||||
|     private: | ||||
|         bool m_use_legacy_opengl; | ||||
|         bool m_enabled; | ||||
|         Shader m_shader; | ||||
|         unsigned int m_z_texture_id; | ||||
|         mutable GLTexture m_tooltip_texture; | ||||
|         mutable GLTexture m_reset_texture; | ||||
| 
 | ||||
|     public: | ||||
|         EState state; | ||||
|         float band_width; | ||||
|         float strength; | ||||
|         int last_object_id; | ||||
|         float last_z; | ||||
|         unsigned int last_action; | ||||
| 
 | ||||
|         LayersEditing(); | ||||
|         ~LayersEditing(); | ||||
| 
 | ||||
|         bool init(const std::string& vertex_shader_filename, const std::string& fragment_shader_filename); | ||||
| 
 | ||||
|         bool is_allowed() const; | ||||
|         void set_use_legacy_opengl(bool use_legacy_opengl); | ||||
| 
 | ||||
|         bool is_enabled() const; | ||||
|         void set_enabled(bool enabled); | ||||
| 
 | ||||
|         unsigned int get_z_texture_id() const; | ||||
| 
 | ||||
|         void render(const GLCanvas3D& canvas, const PrintObject& print_object, const GLVolume& volume) const; | ||||
| 
 | ||||
|         int get_shader_program_id() const; | ||||
| 
 | ||||
|         static float get_cursor_z_relative(const GLCanvas3D& canvas); | ||||
|         static bool bar_rect_contains(const GLCanvas3D& canvas, float x, float y); | ||||
|         static bool reset_rect_contains(const GLCanvas3D& canvas, float x, float y); | ||||
|         static Rect get_bar_rect_screen(const GLCanvas3D& canvas); | ||||
|         static Rect get_reset_rect_screen(const GLCanvas3D& canvas); | ||||
|         static Rect get_bar_rect_viewport(const GLCanvas3D& canvas); | ||||
|         static Rect get_reset_rect_viewport(const GLCanvas3D& canvas); | ||||
| 
 | ||||
|     private: | ||||
|         bool _is_initialized() const; | ||||
|         void _render_tooltip_texture(const GLCanvas3D& canvas, const Rect& bar_rect, const Rect& reset_rect) const; | ||||
|         void _render_reset_texture(const Rect& reset_rect) const; | ||||
|         void _render_active_object_annotations(const GLCanvas3D& canvas, const GLVolume& volume, const PrintObject& print_object, const Rect& bar_rect) const; | ||||
|         void _render_profile(const PrintObject& print_object, const Rect& bar_rect) const; | ||||
|     }; | ||||
| 
 | ||||
|     struct Mouse | ||||
|     { | ||||
|         struct Drag | ||||
|         { | ||||
|             static const Point Invalid_2D_Point; | ||||
|             static const Pointf3 Invalid_3D_Point; | ||||
| 
 | ||||
|             Point start_position_2D; | ||||
|             Pointf3 start_position_3D; | ||||
|             Vectorf3 volume_center_offset; | ||||
|             int volume_idx; | ||||
| 
 | ||||
|         public: | ||||
|             Drag(); | ||||
|         }; | ||||
| 
 | ||||
|         bool dragging; | ||||
|         Pointf position; | ||||
|         Drag drag; | ||||
| 
 | ||||
|         Mouse(); | ||||
| 
 | ||||
|         void set_start_position_2D_as_invalid(); | ||||
|         void set_start_position_3D_as_invalid(); | ||||
| 
 | ||||
|         bool is_start_position_2D_defined() const; | ||||
|         bool is_start_position_3D_defined() const; | ||||
|     }; | ||||
| 
 | ||||
|     class Gizmos | ||||
|     { | ||||
|         static const float OverlayOffsetX; | ||||
|         static const float OverlayGapY; | ||||
| 
 | ||||
|     public: | ||||
|         enum EType : unsigned char | ||||
|         { | ||||
|             Undefined, | ||||
|             Scale, | ||||
|             Rotate, | ||||
|             Num_Types | ||||
|         }; | ||||
| 
 | ||||
|     private: | ||||
|         bool m_enabled; | ||||
|         typedef std::map<EType, GLGizmoBase*> GizmosMap; | ||||
|         GizmosMap m_gizmos; | ||||
|         EType m_current; | ||||
|         bool m_dragging; | ||||
| 
 | ||||
|     public: | ||||
|         Gizmos(); | ||||
|         ~Gizmos(); | ||||
| 
 | ||||
|         bool init(); | ||||
| 
 | ||||
|         bool is_enabled() const; | ||||
|         void set_enabled(bool enable); | ||||
| 
 | ||||
|         void update_hover_state(const GLCanvas3D& canvas, const Pointf& mouse_pos); | ||||
|         void update_on_off_state(const GLCanvas3D& canvas, const Pointf& mouse_pos); | ||||
|         void reset_all_states(); | ||||
| 
 | ||||
|         void set_hover_id(int id); | ||||
| 
 | ||||
|         bool overlay_contains_mouse(const GLCanvas3D& canvas, const Pointf& mouse_pos) const; | ||||
|         bool grabber_contains_mouse() const; | ||||
|         void update(const Pointf& mouse_pos); | ||||
|         void update_data(float scale); | ||||
| 
 | ||||
|         bool is_running() const; | ||||
|         bool is_dragging() const; | ||||
|         void start_dragging(); | ||||
|         void stop_dragging(); | ||||
| 
 | ||||
|         float get_scale() const; | ||||
| 
 | ||||
|         void render(const GLCanvas3D& canvas, const BoundingBoxf3& box) const; | ||||
|         void render_current_gizmo_for_picking_pass(const BoundingBoxf3& box) const; | ||||
| 
 | ||||
|     private: | ||||
|         void _reset(); | ||||
| 
 | ||||
|         void _render_overlay(const GLCanvas3D& canvas) const; | ||||
|         void _render_current_gizmo(const BoundingBoxf3& box) const; | ||||
| 
 | ||||
|         float _get_total_overlay_height() const; | ||||
|         GLGizmoBase* _get_current() const; | ||||
|     }; | ||||
| 
 | ||||
| private: | ||||
|     wxGLCanvas* m_canvas; | ||||
|     wxGLContext* m_context; | ||||
|     wxTimer* m_timer; | ||||
|     Camera m_camera; | ||||
|     Bed m_bed; | ||||
|     Axes m_axes; | ||||
|     CuttingPlane m_cutting_plane; | ||||
|     LayersEditing m_layers_editing; | ||||
|     Shader m_shader; | ||||
|     Mouse m_mouse; | ||||
|     mutable Gizmos m_gizmos; | ||||
| 
 | ||||
|     mutable GLVolumeCollection m_volumes; | ||||
|     DynamicPrintConfig* m_config; | ||||
|     Print* m_print; | ||||
|     Model* m_model; | ||||
| 
 | ||||
|     bool m_dirty; | ||||
|     bool m_initialized; | ||||
|     bool m_use_VBOs; | ||||
|     bool m_force_zoom_to_bed_enabled; | ||||
|     bool m_apply_zoom_to_volumes_filter; | ||||
|     mutable int m_hover_volume_id; | ||||
|     bool m_warning_texture_enabled; | ||||
|     bool m_legend_texture_enabled; | ||||
|     bool m_picking_enabled; | ||||
|     bool m_moving_enabled; | ||||
|     bool m_shader_enabled; | ||||
|     bool m_multisample_allowed; | ||||
| 
 | ||||
|     std::string m_color_by; | ||||
|     std::string m_select_by; | ||||
|     std::string m_drag_by; | ||||
| 
 | ||||
|     bool m_reload_delayed; | ||||
|     std::vector<std::vector<int>> m_objects_volumes_idxs; | ||||
|     std::vector<int> m_objects_selections; | ||||
| 
 | ||||
|     GCodePreviewVolumeIndex m_gcode_preview_volume_index; | ||||
| 
 | ||||
|     PerlCallback m_on_viewport_changed_callback; | ||||
|     PerlCallback m_on_double_click_callback; | ||||
|     PerlCallback m_on_right_click_callback; | ||||
|     PerlCallback m_on_select_object_callback; | ||||
|     PerlCallback m_on_model_update_callback; | ||||
|     PerlCallback m_on_remove_object_callback; | ||||
|     PerlCallback m_on_arrange_callback; | ||||
|     PerlCallback m_on_rotate_object_left_callback; | ||||
|     PerlCallback m_on_rotate_object_right_callback; | ||||
|     PerlCallback m_on_scale_object_uniformly_callback; | ||||
|     PerlCallback m_on_increase_objects_callback; | ||||
|     PerlCallback m_on_decrease_objects_callback; | ||||
|     PerlCallback m_on_instance_moved_callback; | ||||
|     PerlCallback m_on_wipe_tower_moved_callback; | ||||
|     PerlCallback m_on_enable_action_buttons_callback; | ||||
|     PerlCallback m_on_gizmo_scale_uniformly_callback; | ||||
| 
 | ||||
| public: | ||||
|     GLCanvas3D(wxGLCanvas* canvas); | ||||
|     ~GLCanvas3D(); | ||||
| 
 | ||||
|     bool init(bool useVBOs, bool use_legacy_opengl); | ||||
| 
 | ||||
|     bool set_current(); | ||||
| 
 | ||||
|     void set_as_dirty(); | ||||
| 
 | ||||
|     unsigned int get_volumes_count() const; | ||||
|     void reset_volumes(); | ||||
|     void deselect_volumes(); | ||||
|     void select_volume(unsigned int id); | ||||
|     void update_volumes_selection(const std::vector<int>& selections); | ||||
|     bool check_volumes_outside_state(const DynamicPrintConfig* config) const; | ||||
|     bool move_volume_up(unsigned int id); | ||||
|     bool move_volume_down(unsigned int id); | ||||
| 
 | ||||
|     void set_objects_selections(const std::vector<int>& selections); | ||||
| 
 | ||||
|     void set_config(DynamicPrintConfig* config); | ||||
|     void set_print(Print* print); | ||||
|     void set_model(Model* model); | ||||
| 
 | ||||
|     // Set the bed shape to a single closed 2D polygon(array of two element arrays),
 | ||||
|     // triangulate the bed and store the triangles into m_bed.m_triangles,
 | ||||
|     // fills the m_bed.m_grid_lines and sets m_bed.m_origin.
 | ||||
|     // Sets m_bed.m_polygon to limit the object placement.
 | ||||
|     void set_bed_shape(const Pointfs& shape); | ||||
|     // Used by ObjectCutDialog and ObjectPartsPanel to generate a rectangular ground plane to support the scene objects.
 | ||||
|     void set_auto_bed_shape(); | ||||
| 
 | ||||
|     void set_axes_length(float length); | ||||
| 
 | ||||
|     void set_cutting_plane(float z, const ExPolygons& polygons); | ||||
| 
 | ||||
|     void set_color_by(const std::string& value); | ||||
|     void set_select_by(const std::string& value); | ||||
|     void set_drag_by(const std::string& value); | ||||
| 
 | ||||
|     float get_camera_zoom() const; | ||||
| 
 | ||||
|     BoundingBoxf3 volumes_bounding_box() const; | ||||
| 
 | ||||
|     bool is_layers_editing_enabled() const; | ||||
|     bool is_layers_editing_allowed() const; | ||||
|     bool is_shader_enabled() const; | ||||
| 
 | ||||
|     bool is_reload_delayed() const; | ||||
| 
 | ||||
|     void enable_layers_editing(bool enable); | ||||
|     void enable_warning_texture(bool enable); | ||||
|     void enable_legend_texture(bool enable); | ||||
|     void enable_picking(bool enable); | ||||
|     void enable_moving(bool enable); | ||||
|     void enable_gizmos(bool enable); | ||||
|     void enable_shader(bool enable); | ||||
|     void enable_force_zoom_to_bed(bool enable); | ||||
|     void allow_multisample(bool allow); | ||||
| 
 | ||||
|     void zoom_to_bed(); | ||||
|     void zoom_to_volumes(); | ||||
|     void select_view(const std::string& direction); | ||||
|     void set_viewport_from_scene(const GLCanvas3D& other); | ||||
| 
 | ||||
|     void update_volumes_colors_by_extruder(); | ||||
| 
 | ||||
|     void render(); | ||||
| 
 | ||||
|     std::vector<double> get_current_print_zs(bool active_only) const; | ||||
|     void set_toolpaths_range(double low, double high); | ||||
| 
 | ||||
|     std::vector<int> load_object(const ModelObject& model_object, int obj_idx, std::vector<int> instance_idxs); | ||||
|     std::vector<int> load_object(const Model& model, int obj_idx); | ||||
| 
 | ||||
|     void reload_scene(bool force); | ||||
| 
 | ||||
|     // Create 3D thick extrusion lines for a skirt and brim.
 | ||||
|     // Adds a new Slic3r::GUI::3DScene::Volume to volumes.
 | ||||
|     void load_print_toolpaths(); | ||||
|     // Create 3D thick extrusion lines for object forming extrusions.
 | ||||
|     // Adds a new Slic3r::GUI::3DScene::Volume to $self->volumes,
 | ||||
|     // one for perimeters, one for infill and one for supports.
 | ||||
|     void load_print_object_toolpaths(const PrintObject& print_object, const std::vector<std::string>& str_tool_colors); | ||||
|     // Create 3D thick extrusion lines for wipe tower extrusions
 | ||||
|     void load_wipe_tower_toolpaths(const std::vector<std::string>& str_tool_colors); | ||||
|     void load_gcode_preview(const GCodePreviewData& preview_data, const std::vector<std::string>& str_tool_colors); | ||||
| 
 | ||||
|     void register_on_viewport_changed_callback(void* callback); | ||||
|     void register_on_double_click_callback(void* callback); | ||||
|     void register_on_right_click_callback(void* callback); | ||||
|     void register_on_select_object_callback(void* callback); | ||||
|     void register_on_model_update_callback(void* callback); | ||||
|     void register_on_remove_object_callback(void* callback); | ||||
|     void register_on_arrange_callback(void* callback); | ||||
|     void register_on_rotate_object_left_callback(void* callback); | ||||
|     void register_on_rotate_object_right_callback(void* callback); | ||||
|     void register_on_scale_object_uniformly_callback(void* callback); | ||||
|     void register_on_increase_objects_callback(void* callback); | ||||
|     void register_on_decrease_objects_callback(void* callback); | ||||
|     void register_on_instance_moved_callback(void* callback); | ||||
|     void register_on_wipe_tower_moved_callback(void* callback); | ||||
|     void register_on_enable_action_buttons_callback(void* callback); | ||||
|     void register_on_gizmo_scale_uniformly_callback(void* callback); | ||||
| 
 | ||||
|     void bind_event_handlers(); | ||||
|     void unbind_event_handlers(); | ||||
| 
 | ||||
|     void on_size(wxSizeEvent& evt); | ||||
|     void on_idle(wxIdleEvent& evt); | ||||
|     void on_char(wxKeyEvent& evt); | ||||
|     void on_mouse_wheel(wxMouseEvent& evt); | ||||
|     void on_timer(wxTimerEvent& evt); | ||||
|     void on_mouse(wxMouseEvent& evt); | ||||
|     void on_paint(wxPaintEvent& evt); | ||||
|     void on_key_down(wxKeyEvent& evt); | ||||
| 
 | ||||
|     Size get_canvas_size() const; | ||||
|     Point get_local_mouse_position() const; | ||||
| 
 | ||||
| private: | ||||
|     bool _is_shown_on_screen() const; | ||||
|     void _force_zoom_to_bed(); | ||||
| 
 | ||||
|     void _resize(unsigned int w, unsigned int h); | ||||
| 
 | ||||
|     BoundingBoxf3 _max_bounding_box() const; | ||||
|     BoundingBoxf3 _selected_volumes_bounding_box() const; | ||||
| 
 | ||||
|     void _zoom_to_bounding_box(const BoundingBoxf3& bbox); | ||||
|     float _get_zoom_to_bounding_box_factor(const BoundingBoxf3& bbox) const; | ||||
| 
 | ||||
|     void _deregister_callbacks(); | ||||
| 
 | ||||
|     void _mark_volumes_for_layer_height() const; | ||||
|     void _refresh_if_shown_on_screen(); | ||||
| 
 | ||||
|     void _camera_tranform() const; | ||||
|     void _picking_pass() const; | ||||
|     void _render_background() const; | ||||
|     void _render_bed(float theta) const; | ||||
|     void _render_axes(bool depth_test) const; | ||||
|     void _render_objects() const; | ||||
|     void _render_cutting_plane() const; | ||||
|     void _render_warning_texture() const; | ||||
|     void _render_legend_texture() const; | ||||
|     void _render_layer_editing_overlay() const; | ||||
|     void _render_volumes(bool fake_colors) const; | ||||
|     void _render_gizmo() const; | ||||
| 
 | ||||
|     float _get_layers_editing_cursor_z_relative() const; | ||||
|     void _perform_layer_editing_action(wxMouseEvent* evt = nullptr); | ||||
| 
 | ||||
|     // Convert the screen space coordinate to an object space coordinate.
 | ||||
|     // If the Z screen space coordinate is not provided, a depth buffer value is substituted.
 | ||||
|     Pointf3 _mouse_to_3d(const Point& mouse_pos, float* z = nullptr); | ||||
| 
 | ||||
|     // Convert the screen space coordinate to world coordinate on the bed.
 | ||||
|     Pointf3 _mouse_to_bed_3d(const Point& mouse_pos); | ||||
| 
 | ||||
|     void _start_timer(); | ||||
|     void _stop_timer(); | ||||
| 
 | ||||
|     int _get_first_selected_object_id() const; | ||||
| 
 | ||||
|     // generates gcode extrusion paths geometry
 | ||||
|     void _load_gcode_extrusion_paths(const GCodePreviewData& preview_data, const std::vector<float>& tool_colors); | ||||
|     // generates gcode travel paths geometry
 | ||||
|     void _load_gcode_travel_paths(const GCodePreviewData& preview_data, const std::vector<float>& tool_colors); | ||||
|     bool _travel_paths_by_type(const GCodePreviewData& preview_data); | ||||
|     bool _travel_paths_by_feedrate(const GCodePreviewData& preview_data); | ||||
|     bool _travel_paths_by_tool(const GCodePreviewData& preview_data, const std::vector<float>& tool_colors); | ||||
|     // generates gcode retractions geometry
 | ||||
|     void _load_gcode_retractions(const GCodePreviewData& preview_data); | ||||
|     // generates gcode unretractions geometry
 | ||||
|     void _load_gcode_unretractions(const GCodePreviewData& preview_data); | ||||
|     // generates objects and wipe tower geometry
 | ||||
|     void _load_shells(); | ||||
|     // sets gcode geometry visibility according to user selection
 | ||||
|     void _update_gcode_volumes_visibility(const GCodePreviewData& preview_data); | ||||
| 
 | ||||
|     void _on_move(const std::vector<int>& volume_idxs); | ||||
|     void _on_select(int volume_idx); | ||||
| 
 | ||||
|     void _update_gizmos_data(); | ||||
| 
 | ||||
|     static std::vector<float> _parse_colors(const std::vector<std::string>& colors); | ||||
| }; | ||||
| 
 | ||||
| } // namespace GUI
 | ||||
| } // namespace Slic3r
 | ||||
| 
 | ||||
| #endif // slic3r_GLCanvas3D_hpp_
 | ||||
							
								
								
									
										680
									
								
								xs/src/slic3r/GUI/GLCanvas3DManager.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										680
									
								
								xs/src/slic3r/GUI/GLCanvas3DManager.cpp
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,680 @@ | |||
| #include "GLCanvas3DManager.hpp" | ||||
| #include "../../slic3r/GUI/GUI.hpp" | ||||
| #include "../../slic3r/GUI/AppConfig.hpp" | ||||
| #include "../../slic3r/GUI/GLCanvas3D.hpp" | ||||
| 
 | ||||
| #include <GL/glew.h> | ||||
| 
 | ||||
| #include <boost/algorithm/string/split.hpp> | ||||
| #include <boost/algorithm/string/classification.hpp> | ||||
| 
 | ||||
| #include <wx/glcanvas.h> | ||||
| #include <wx/timer.h> | ||||
| 
 | ||||
| #include <vector> | ||||
| #include <string> | ||||
| #include <iostream> | ||||
| 
 | ||||
| namespace Slic3r { | ||||
| namespace GUI { | ||||
| 
 | ||||
| GLCanvas3DManager::GLInfo::GLInfo() | ||||
|     : version("") | ||||
|     , glsl_version("") | ||||
|     , vendor("") | ||||
|     , renderer("") | ||||
| { | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::GLInfo::detect() | ||||
| { | ||||
|     const char* data = (const char*)::glGetString(GL_VERSION); | ||||
|     if (data != nullptr) | ||||
|         version = data; | ||||
| 
 | ||||
|     data = (const char*)::glGetString(GL_SHADING_LANGUAGE_VERSION); | ||||
|     if (data != nullptr) | ||||
|         glsl_version = data; | ||||
| 
 | ||||
|     data = (const char*)::glGetString(GL_VENDOR); | ||||
|     if (data != nullptr) | ||||
|         vendor = data; | ||||
| 
 | ||||
|     data = (const char*)::glGetString(GL_RENDERER); | ||||
|     if (data != nullptr) | ||||
|         renderer = data; | ||||
| } | ||||
| 
 | ||||
| bool GLCanvas3DManager::GLInfo::is_version_greater_or_equal_to(unsigned int major, unsigned int minor) const | ||||
| { | ||||
|     std::vector<std::string> tokens; | ||||
|     boost::split(tokens, version, boost::is_any_of(" "), boost::token_compress_on); | ||||
| 
 | ||||
|     if (tokens.empty()) | ||||
|         return false; | ||||
| 
 | ||||
|     std::vector<std::string> numbers; | ||||
|     boost::split(numbers, tokens[0], boost::is_any_of("."), boost::token_compress_on); | ||||
| 
 | ||||
|     unsigned int gl_major = 0; | ||||
|     unsigned int gl_minor = 0; | ||||
| 
 | ||||
|     if (numbers.size() > 0) | ||||
|         gl_major = ::atoi(numbers[0].c_str()); | ||||
| 
 | ||||
|     if (numbers.size() > 1) | ||||
|         gl_minor = ::atoi(numbers[1].c_str()); | ||||
| 
 | ||||
|     if (gl_major < major) | ||||
|         return false; | ||||
|     else if (gl_major > major) | ||||
|         return true; | ||||
|     else | ||||
|         return gl_minor >= minor; | ||||
| } | ||||
| 
 | ||||
| std::string GLCanvas3DManager::GLInfo::to_string(bool format_as_html, bool extensions) const | ||||
| { | ||||
|     std::stringstream out; | ||||
| 
 | ||||
|     std::string h2_start = format_as_html ? "<b>" : ""; | ||||
|     std::string h2_end   = format_as_html ? "</b>" : ""; | ||||
|     std::string b_start  = format_as_html ? "<b>" : ""; | ||||
|     std::string b_end    = format_as_html ? "</b>" : ""; | ||||
|     std::string line_end = format_as_html ? "<br>" : "\n"; | ||||
| 
 | ||||
|     out << h2_start << "OpenGL installation" << h2_end << line_end; | ||||
|     out << b_start  << "GL version:   " << b_end << (version.empty() ? "N/A" : version) << line_end; | ||||
|     out << b_start  << "Vendor:       " << b_end << (vendor.empty() ? "N/A" : vendor) << line_end; | ||||
|     out << b_start  << "Renderer:     " << b_end << (renderer.empty() ? "N/A" : renderer) << line_end; | ||||
|     out << b_start  << "GLSL version: " << b_end << (glsl_version.empty() ? "N/A" : glsl_version) << line_end; | ||||
| 
 | ||||
|     if (extensions) | ||||
|     { | ||||
|         out << h2_start << "Installed extensions:" << h2_end << line_end; | ||||
| 
 | ||||
|         std::vector<std::string> extensions_list; | ||||
|         GLint num_extensions; | ||||
|         ::glGetIntegerv(GL_NUM_EXTENSIONS, &num_extensions); | ||||
|           | ||||
|         for (GLint i = 0; i < num_extensions; ++i) | ||||
|         { | ||||
|             const char* e = (const char*)::glGetStringi(GL_EXTENSIONS, i); | ||||
|             extensions_list.push_back(e); | ||||
|         } | ||||
| 
 | ||||
|         std::sort(extensions_list.begin(), extensions_list.end()); | ||||
|         for (const std::string& ext : extensions_list) | ||||
|         { | ||||
|             out << ext << line_end; | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     return out.str(); | ||||
| } | ||||
| 
 | ||||
| GLCanvas3DManager::GLCanvas3DManager() | ||||
|     : m_current(nullptr) | ||||
|     , m_gl_initialized(false) | ||||
|     , m_use_legacy_opengl(false) | ||||
|     , m_use_VBOs(false) | ||||
| { | ||||
| } | ||||
| 
 | ||||
| bool GLCanvas3DManager::add(wxGLCanvas* canvas) | ||||
| { | ||||
|     if (canvas == nullptr) | ||||
|         return false; | ||||
| 
 | ||||
|     if (_get_canvas(canvas) != m_canvases.end()) | ||||
|         return false; | ||||
| 
 | ||||
|     GLCanvas3D* canvas3D = new GLCanvas3D(canvas); | ||||
|     if (canvas3D == nullptr) | ||||
|         return false; | ||||
| 
 | ||||
|     canvas3D->bind_event_handlers(); | ||||
|     m_canvases.insert(CanvasesMap::value_type(canvas, canvas3D)); | ||||
| 
 | ||||
|     return true; | ||||
| } | ||||
| 
 | ||||
| bool GLCanvas3DManager::remove(wxGLCanvas* canvas) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it == m_canvases.end()) | ||||
|         return false; | ||||
| 
 | ||||
|     it->second->unbind_event_handlers(); | ||||
|     delete it->second; | ||||
|     m_canvases.erase(it); | ||||
| 
 | ||||
|     return true; | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::remove_all() | ||||
| { | ||||
|     for (CanvasesMap::value_type& item : m_canvases) | ||||
|     { | ||||
|         item.second->unbind_event_handlers(); | ||||
|         delete item.second; | ||||
|     } | ||||
|     m_canvases.clear(); | ||||
| } | ||||
| 
 | ||||
| unsigned int GLCanvas3DManager::count() const | ||||
| { | ||||
|     return (unsigned int)m_canvases.size(); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::init_gl() | ||||
| { | ||||
|     if (!m_gl_initialized) | ||||
|     { | ||||
|         glewInit(); | ||||
|         m_gl_info.detect(); | ||||
|         const AppConfig* config = GUI::get_app_config(); | ||||
|         m_use_legacy_opengl = (config == nullptr) || (config->get("use_legacy_opengl") == "1"); | ||||
|         m_use_VBOs = !m_use_legacy_opengl && m_gl_info.is_version_greater_or_equal_to(2, 0); | ||||
|         m_gl_initialized = true; | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| std::string GLCanvas3DManager::get_gl_info(bool format_as_html, bool extensions) const | ||||
| { | ||||
|     return m_gl_info.to_string(format_as_html, extensions); | ||||
| } | ||||
| 
 | ||||
| bool GLCanvas3DManager::use_VBOs() const | ||||
| { | ||||
|     return m_use_VBOs; | ||||
| } | ||||
| 
 | ||||
| bool GLCanvas3DManager::init(wxGLCanvas* canvas) | ||||
| { | ||||
|     CanvasesMap::const_iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         return (it->second != nullptr) ? _init(*it->second) : false; | ||||
|     else | ||||
|         return false; | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::set_as_dirty(wxGLCanvas* canvas) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->set_as_dirty(); | ||||
| } | ||||
| 
 | ||||
| unsigned int GLCanvas3DManager::get_volumes_count(wxGLCanvas* canvas) const | ||||
| { | ||||
|     CanvasesMap::const_iterator it = _get_canvas(canvas); | ||||
|     return (it != m_canvases.end()) ? it->second->get_volumes_count() : 0; | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::reset_volumes(wxGLCanvas* canvas) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->reset_volumes(); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::deselect_volumes(wxGLCanvas* canvas) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->deselect_volumes(); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::select_volume(wxGLCanvas* canvas, unsigned int id) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->select_volume(id); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::update_volumes_selection(wxGLCanvas* canvas, const std::vector<int>& selections) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->update_volumes_selection(selections); | ||||
| } | ||||
| 
 | ||||
| bool GLCanvas3DManager::check_volumes_outside_state(wxGLCanvas* canvas, const DynamicPrintConfig* config) const | ||||
| { | ||||
|     CanvasesMap::const_iterator it = _get_canvas(canvas); | ||||
|     return (it != m_canvases.end()) ? it->second->check_volumes_outside_state(config) : false; | ||||
| } | ||||
| 
 | ||||
| bool GLCanvas3DManager::move_volume_up(wxGLCanvas* canvas, unsigned int id) | ||||
| { | ||||
|     CanvasesMap::const_iterator it = _get_canvas(canvas); | ||||
|     return (it != m_canvases.end()) ? it->second->move_volume_up(id) : false; | ||||
| } | ||||
| 
 | ||||
| bool GLCanvas3DManager::move_volume_down(wxGLCanvas* canvas, unsigned int id) | ||||
| { | ||||
|     CanvasesMap::const_iterator it = _get_canvas(canvas); | ||||
|     return (it != m_canvases.end()) ? it->second->move_volume_down(id) : false; | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::set_objects_selections(wxGLCanvas* canvas, const std::vector<int>& selections) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->set_objects_selections(selections); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::set_config(wxGLCanvas* canvas, DynamicPrintConfig* config) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->set_config(config); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::set_print(wxGLCanvas* canvas, Print* print) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->set_print(print); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::set_model(wxGLCanvas* canvas, Model* model) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->set_model(model); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::set_bed_shape(wxGLCanvas* canvas, const Pointfs& shape) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->set_bed_shape(shape); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::set_auto_bed_shape(wxGLCanvas* canvas) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->set_auto_bed_shape(); | ||||
| } | ||||
| 
 | ||||
| BoundingBoxf3 GLCanvas3DManager::get_volumes_bounding_box(wxGLCanvas* canvas) | ||||
| { | ||||
|     CanvasesMap::const_iterator it = _get_canvas(canvas); | ||||
|     return (it != m_canvases.end()) ? it->second->volumes_bounding_box() : BoundingBoxf3(); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::set_axes_length(wxGLCanvas* canvas, float length) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->set_axes_length(length); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::set_cutting_plane(wxGLCanvas* canvas, float z, const ExPolygons& polygons) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->set_cutting_plane(z, polygons); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::set_color_by(wxGLCanvas* canvas, const std::string& value) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->set_color_by(value); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::set_select_by(wxGLCanvas* canvas, const std::string& value) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->set_select_by(value); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::set_drag_by(wxGLCanvas* canvas, const std::string& value) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->set_drag_by(value); | ||||
| } | ||||
| 
 | ||||
| bool GLCanvas3DManager::is_layers_editing_enabled(wxGLCanvas* canvas) const | ||||
| { | ||||
|     CanvasesMap::const_iterator it = _get_canvas(canvas); | ||||
|     return (it != m_canvases.end()) ? it->second->is_layers_editing_enabled() : false; | ||||
| } | ||||
| 
 | ||||
| bool GLCanvas3DManager::is_layers_editing_allowed(wxGLCanvas* canvas) const | ||||
| { | ||||
|     CanvasesMap::const_iterator it = _get_canvas(canvas); | ||||
|     return (it != m_canvases.end()) ? it->second->is_layers_editing_allowed() : false; | ||||
| } | ||||
| 
 | ||||
| bool GLCanvas3DManager::is_shader_enabled(wxGLCanvas* canvas) const | ||||
| { | ||||
|     CanvasesMap::const_iterator it = _get_canvas(canvas); | ||||
|     return (it != m_canvases.end()) ? it->second->is_shader_enabled() : false; | ||||
| } | ||||
| 
 | ||||
| bool GLCanvas3DManager::is_reload_delayed(wxGLCanvas* canvas) const | ||||
| { | ||||
|     CanvasesMap::const_iterator it = _get_canvas(canvas); | ||||
|     return (it != m_canvases.end()) ? it->second->is_reload_delayed() : false; | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::enable_layers_editing(wxGLCanvas* canvas, bool enable) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->enable_layers_editing(enable); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::enable_warning_texture(wxGLCanvas* canvas, bool enable) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->enable_warning_texture(enable); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::enable_legend_texture(wxGLCanvas* canvas, bool enable) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->enable_legend_texture(enable); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::enable_picking(wxGLCanvas* canvas, bool enable) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->enable_picking(enable); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::enable_moving(wxGLCanvas* canvas, bool enable) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->enable_moving(enable); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::enable_gizmos(wxGLCanvas* canvas, bool enable) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->enable_gizmos(enable); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::enable_shader(wxGLCanvas* canvas, bool enable) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->enable_shader(enable); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::enable_force_zoom_to_bed(wxGLCanvas* canvas, bool enable) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->enable_force_zoom_to_bed(enable); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::allow_multisample(wxGLCanvas* canvas, bool allow) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->allow_multisample(allow); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::zoom_to_bed(wxGLCanvas* canvas) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->zoom_to_bed(); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::zoom_to_volumes(wxGLCanvas* canvas) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->zoom_to_volumes(); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::select_view(wxGLCanvas* canvas, const std::string& direction) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->select_view(direction); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::set_viewport_from_scene(wxGLCanvas* canvas, wxGLCanvas* other) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|     { | ||||
|         CanvasesMap::iterator other_it = _get_canvas(other); | ||||
|         if (other_it != m_canvases.end()) | ||||
|             it->second->set_viewport_from_scene(*other_it->second); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::update_volumes_colors_by_extruder(wxGLCanvas* canvas) | ||||
| { | ||||
|     CanvasesMap::const_iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->update_volumes_colors_by_extruder(); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::render(wxGLCanvas* canvas) const | ||||
| { | ||||
|     CanvasesMap::const_iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->render(); | ||||
| } | ||||
| 
 | ||||
| std::vector<double> GLCanvas3DManager::get_current_print_zs(wxGLCanvas* canvas, bool active_only) const | ||||
| { | ||||
|     CanvasesMap::const_iterator it = _get_canvas(canvas); | ||||
|     return (it != m_canvases.end()) ? it->second->get_current_print_zs(active_only) : std::vector<double>(); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::set_toolpaths_range(wxGLCanvas* canvas, double low, double high) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->set_toolpaths_range(low, high); | ||||
| } | ||||
| 
 | ||||
| std::vector<int> GLCanvas3DManager::load_object(wxGLCanvas* canvas, const ModelObject* model_object, int obj_idx, std::vector<int> instance_idxs) | ||||
| { | ||||
|     if (model_object == nullptr) | ||||
|         return std::vector<int>(); | ||||
| 
 | ||||
|     CanvasesMap::const_iterator it = _get_canvas(canvas); | ||||
|     return (it != m_canvases.end()) ? it->second->load_object(*model_object, obj_idx, instance_idxs) : std::vector<int>(); | ||||
| } | ||||
| 
 | ||||
| std::vector<int> GLCanvas3DManager::load_object(wxGLCanvas* canvas, const Model* model, int obj_idx) | ||||
| { | ||||
|     if (model == nullptr) | ||||
|         return std::vector<int>(); | ||||
| 
 | ||||
|     CanvasesMap::const_iterator it = _get_canvas(canvas); | ||||
|     return (it != m_canvases.end()) ? it->second->load_object(*model, obj_idx) : std::vector<int>(); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::reload_scene(wxGLCanvas* canvas, bool force) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->reload_scene(force); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::load_print_toolpaths(wxGLCanvas* canvas) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->load_print_toolpaths(); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::load_print_object_toolpaths(wxGLCanvas* canvas, const PrintObject* print_object, const std::vector<std::string>& tool_colors) | ||||
| { | ||||
|     if (print_object == nullptr) | ||||
|         return; | ||||
| 
 | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->load_print_object_toolpaths(*print_object, tool_colors); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::load_wipe_tower_toolpaths(wxGLCanvas* canvas, const std::vector<std::string>& str_tool_colors) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->load_wipe_tower_toolpaths(str_tool_colors); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::load_gcode_preview(wxGLCanvas* canvas, const GCodePreviewData* preview_data, const std::vector<std::string>& str_tool_colors) | ||||
| { | ||||
|     if (preview_data == nullptr) | ||||
|         return; | ||||
| 
 | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->load_gcode_preview(*preview_data, str_tool_colors); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::register_on_viewport_changed_callback(wxGLCanvas* canvas, void* callback) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->register_on_viewport_changed_callback(callback); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::register_on_double_click_callback(wxGLCanvas* canvas, void* callback) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->register_on_double_click_callback(callback); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::register_on_right_click_callback(wxGLCanvas* canvas, void* callback) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->register_on_right_click_callback(callback); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::register_on_select_object_callback(wxGLCanvas* canvas, void* callback) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->register_on_select_object_callback(callback); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::register_on_model_update_callback(wxGLCanvas* canvas, void* callback) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->register_on_model_update_callback(callback); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::register_on_remove_object_callback(wxGLCanvas* canvas, void* callback) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->register_on_remove_object_callback(callback); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::register_on_arrange_callback(wxGLCanvas* canvas, void* callback) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->register_on_arrange_callback(callback); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::register_on_rotate_object_left_callback(wxGLCanvas* canvas, void* callback) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->register_on_rotate_object_left_callback(callback); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::register_on_rotate_object_right_callback(wxGLCanvas* canvas, void* callback) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->register_on_rotate_object_right_callback(callback); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::register_on_scale_object_uniformly_callback(wxGLCanvas* canvas, void* callback) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->register_on_scale_object_uniformly_callback(callback); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::register_on_increase_objects_callback(wxGLCanvas* canvas, void* callback) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->register_on_increase_objects_callback(callback); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::register_on_decrease_objects_callback(wxGLCanvas* canvas, void* callback) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->register_on_decrease_objects_callback(callback); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::register_on_instance_moved_callback(wxGLCanvas* canvas, void* callback) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->register_on_instance_moved_callback(callback); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::register_on_wipe_tower_moved_callback(wxGLCanvas* canvas, void* callback) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->register_on_wipe_tower_moved_callback(callback); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::register_on_enable_action_buttons_callback(wxGLCanvas* canvas, void* callback) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->register_on_enable_action_buttons_callback(callback); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3DManager::register_on_gizmo_scale_uniformly_callback(wxGLCanvas* canvas, void* callback) | ||||
| { | ||||
|     CanvasesMap::iterator it = _get_canvas(canvas); | ||||
|     if (it != m_canvases.end()) | ||||
|         it->second->register_on_gizmo_scale_uniformly_callback(callback); | ||||
| } | ||||
| 
 | ||||
| GLCanvas3DManager::CanvasesMap::iterator GLCanvas3DManager::_get_canvas(wxGLCanvas* canvas) | ||||
| { | ||||
|     return (canvas == nullptr) ? m_canvases.end() : m_canvases.find(canvas); | ||||
| } | ||||
| 
 | ||||
| GLCanvas3DManager::CanvasesMap::const_iterator GLCanvas3DManager::_get_canvas(wxGLCanvas* canvas) const | ||||
| { | ||||
|     return (canvas == nullptr) ? m_canvases.end() : m_canvases.find(canvas); | ||||
| } | ||||
| 
 | ||||
| bool GLCanvas3DManager::_init(GLCanvas3D& canvas) | ||||
| { | ||||
|     if (!m_gl_initialized) | ||||
|         init_gl(); | ||||
| 
 | ||||
|     return canvas.init(m_use_VBOs, m_use_legacy_opengl); | ||||
| } | ||||
| 
 | ||||
| } // namespace GUI
 | ||||
| } // namespace Slic3r
 | ||||
							
								
								
									
										166
									
								
								xs/src/slic3r/GUI/GLCanvas3DManager.hpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										166
									
								
								xs/src/slic3r/GUI/GLCanvas3DManager.hpp
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,166 @@ | |||
| #ifndef slic3r_GLCanvas3DManager_hpp_ | ||||
| #define slic3r_GLCanvas3DManager_hpp_ | ||||
| 
 | ||||
| #include "../../libslic3r/BoundingBox.hpp" | ||||
| 
 | ||||
| #include <map> | ||||
| #include <vector> | ||||
| 
 | ||||
| class wxGLCanvas; | ||||
| class wxGLContext; | ||||
| 
 | ||||
| namespace Slic3r { | ||||
| 
 | ||||
| class DynamicPrintConfig; | ||||
| class Print; | ||||
| class Model; | ||||
| class ExPolygon; | ||||
| typedef std::vector<ExPolygon> ExPolygons; | ||||
| class ModelObject; | ||||
| class PrintObject; | ||||
| class GCodePreviewData; | ||||
|      | ||||
| namespace GUI { | ||||
| 
 | ||||
| class GLCanvas3D; | ||||
| 
 | ||||
| class GLCanvas3DManager | ||||
| { | ||||
|     struct GLInfo | ||||
|     { | ||||
|         std::string version; | ||||
|         std::string glsl_version; | ||||
|         std::string vendor; | ||||
|         std::string renderer; | ||||
| 
 | ||||
|         GLInfo(); | ||||
| 
 | ||||
|         void detect(); | ||||
|         bool is_version_greater_or_equal_to(unsigned int major, unsigned int minor) const; | ||||
| 
 | ||||
|         std::string to_string(bool format_as_html, bool extensions) const; | ||||
|     }; | ||||
| 
 | ||||
|     typedef std::map<wxGLCanvas*, GLCanvas3D*> CanvasesMap; | ||||
| 
 | ||||
|     CanvasesMap m_canvases; | ||||
|     wxGLCanvas* m_current; | ||||
|     GLInfo m_gl_info; | ||||
|     bool m_gl_initialized; | ||||
|     bool m_use_legacy_opengl; | ||||
|     bool m_use_VBOs; | ||||
| 
 | ||||
| public: | ||||
|     GLCanvas3DManager(); | ||||
| 
 | ||||
|     bool add(wxGLCanvas* canvas); | ||||
|     bool remove(wxGLCanvas* canvas); | ||||
| 
 | ||||
|     void remove_all(); | ||||
| 
 | ||||
|     unsigned int count() const; | ||||
| 
 | ||||
|     void init_gl(); | ||||
|     std::string get_gl_info(bool format_as_html, bool extensions) const; | ||||
| 
 | ||||
|     bool use_VBOs() const; | ||||
|     bool layer_editing_allowed() const; | ||||
| 
 | ||||
|     bool init(wxGLCanvas* canvas); | ||||
| 
 | ||||
|     void set_as_dirty(wxGLCanvas* canvas); | ||||
| 
 | ||||
|     unsigned int get_volumes_count(wxGLCanvas* canvas) const; | ||||
|     void reset_volumes(wxGLCanvas* canvas); | ||||
|     void deselect_volumes(wxGLCanvas* canvas); | ||||
|     void select_volume(wxGLCanvas* canvas, unsigned int id); | ||||
|     void update_volumes_selection(wxGLCanvas* canvas, const std::vector<int>& selections); | ||||
|     bool check_volumes_outside_state(wxGLCanvas* canvas, const DynamicPrintConfig* config) const; | ||||
|     bool move_volume_up(wxGLCanvas* canvas, unsigned int id); | ||||
|     bool move_volume_down(wxGLCanvas* canvas, unsigned int id); | ||||
| 
 | ||||
|     void set_objects_selections(wxGLCanvas* canvas, const std::vector<int>& selections); | ||||
| 
 | ||||
|     void set_config(wxGLCanvas* canvas, DynamicPrintConfig* config); | ||||
|     void set_print(wxGLCanvas* canvas, Print* print); | ||||
|     void set_model(wxGLCanvas* canvas, Model* model); | ||||
| 
 | ||||
|     void set_bed_shape(wxGLCanvas* canvas, const Pointfs& shape); | ||||
|     void set_auto_bed_shape(wxGLCanvas* canvas); | ||||
| 
 | ||||
|     BoundingBoxf3 get_volumes_bounding_box(wxGLCanvas* canvas); | ||||
| 
 | ||||
|     void set_axes_length(wxGLCanvas* canvas, float length); | ||||
| 
 | ||||
|     void set_cutting_plane(wxGLCanvas* canvas, float z, const ExPolygons& polygons); | ||||
| 
 | ||||
|     void set_color_by(wxGLCanvas* canvas, const std::string& value); | ||||
|     void set_select_by(wxGLCanvas* canvas, const std::string& value); | ||||
|     void set_drag_by(wxGLCanvas* canvas, const std::string& value); | ||||
| 
 | ||||
|     bool is_layers_editing_enabled(wxGLCanvas* canvas) const; | ||||
|     bool is_layers_editing_allowed(wxGLCanvas* canvas) const; | ||||
|     bool is_shader_enabled(wxGLCanvas* canvas) const; | ||||
| 
 | ||||
|     bool is_reload_delayed(wxGLCanvas* canvas) const; | ||||
| 
 | ||||
|     void enable_layers_editing(wxGLCanvas* canvas, bool enable); | ||||
|     void enable_warning_texture(wxGLCanvas* canvas, bool enable); | ||||
|     void enable_legend_texture(wxGLCanvas* canvas, bool enable); | ||||
|     void enable_picking(wxGLCanvas* canvas, bool enable); | ||||
|     void enable_moving(wxGLCanvas* canvas, bool enable); | ||||
|     void enable_gizmos(wxGLCanvas* canvas, bool enable); | ||||
|     void enable_shader(wxGLCanvas* canvas, bool enable); | ||||
|     void enable_force_zoom_to_bed(wxGLCanvas* canvas, bool enable); | ||||
|     void allow_multisample(wxGLCanvas* canvas, bool allow); | ||||
| 
 | ||||
|     void zoom_to_bed(wxGLCanvas* canvas); | ||||
|     void zoom_to_volumes(wxGLCanvas* canvas); | ||||
|     void select_view(wxGLCanvas* canvas, const std::string& direction); | ||||
|     void set_viewport_from_scene(wxGLCanvas* canvas, wxGLCanvas* other); | ||||
| 
 | ||||
|     void update_volumes_colors_by_extruder(wxGLCanvas* canvas); | ||||
| 
 | ||||
|     void render(wxGLCanvas* canvas) const; | ||||
| 
 | ||||
|     std::vector<double> get_current_print_zs(wxGLCanvas* canvas, bool active_only) const; | ||||
|     void set_toolpaths_range(wxGLCanvas* canvas, double low, double high); | ||||
| 
 | ||||
|     std::vector<int> load_object(wxGLCanvas* canvas, const ModelObject* model_object, int obj_idx, std::vector<int> instance_idxs); | ||||
|     std::vector<int> load_object(wxGLCanvas* canvas, const Model* model, int obj_idx); | ||||
| 
 | ||||
|     void reload_scene(wxGLCanvas* canvas, bool force); | ||||
| 
 | ||||
|     void load_print_toolpaths(wxGLCanvas* canvas); | ||||
|     void load_print_object_toolpaths(wxGLCanvas* canvas, const PrintObject* print_object, const std::vector<std::string>& tool_colors); | ||||
|     void load_wipe_tower_toolpaths(wxGLCanvas* canvas, const std::vector<std::string>& str_tool_colors); | ||||
|     void load_gcode_preview(wxGLCanvas* canvas, const GCodePreviewData* preview_data, const std::vector<std::string>& str_tool_colors); | ||||
| 
 | ||||
|     void register_on_viewport_changed_callback(wxGLCanvas* canvas, void* callback); | ||||
|     void register_on_double_click_callback(wxGLCanvas* canvas, void* callback); | ||||
|     void register_on_right_click_callback(wxGLCanvas* canvas, void* callback); | ||||
|     void register_on_select_object_callback(wxGLCanvas* canvas, void* callback); | ||||
|     void register_on_model_update_callback(wxGLCanvas* canvas, void* callback); | ||||
|     void register_on_remove_object_callback(wxGLCanvas* canvas, void* callback); | ||||
|     void register_on_arrange_callback(wxGLCanvas* canvas, void* callback); | ||||
|     void register_on_rotate_object_left_callback(wxGLCanvas* canvas, void* callback); | ||||
|     void register_on_rotate_object_right_callback(wxGLCanvas* canvas, void* callback); | ||||
|     void register_on_scale_object_uniformly_callback(wxGLCanvas* canvas, void* callback); | ||||
|     void register_on_increase_objects_callback(wxGLCanvas* canvas, void* callback); | ||||
|     void register_on_decrease_objects_callback(wxGLCanvas* canvas, void* callback); | ||||
|     void register_on_instance_moved_callback(wxGLCanvas* canvas, void* callback); | ||||
|     void register_on_wipe_tower_moved_callback(wxGLCanvas* canvas, void* callback); | ||||
|     void register_on_enable_action_buttons_callback(wxGLCanvas* canvas, void* callback); | ||||
|     void register_on_gizmo_scale_uniformly_callback(wxGLCanvas* canvas, void* callback); | ||||
| 
 | ||||
| private: | ||||
|     CanvasesMap::iterator _get_canvas(wxGLCanvas* canvas); | ||||
|     CanvasesMap::const_iterator _get_canvas(wxGLCanvas* canvas) const; | ||||
| 
 | ||||
|     bool _init(GLCanvas3D& canvas); | ||||
| }; | ||||
| 
 | ||||
| } // namespace GUI
 | ||||
| } // namespace Slic3r
 | ||||
| 
 | ||||
| #endif // slic3r_GLCanvas3DManager_hpp_
 | ||||
							
								
								
									
										454
									
								
								xs/src/slic3r/GUI/GLGizmo.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										454
									
								
								xs/src/slic3r/GUI/GLGizmo.cpp
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,454 @@ | |||
| #include "GLGizmo.hpp" | ||||
| 
 | ||||
| #include "../../libslic3r/Utils.hpp" | ||||
| #include "../../libslic3r/BoundingBox.hpp" | ||||
| 
 | ||||
| #include <GL/glew.h> | ||||
| 
 | ||||
| #include <iostream> | ||||
| 
 | ||||
| namespace Slic3r { | ||||
| namespace GUI { | ||||
| 
 | ||||
| const float GLGizmoBase::Grabber::HalfSize = 2.0f; | ||||
| const float GLGizmoBase::Grabber::HoverOffset = 0.5f; | ||||
| const float GLGizmoBase::BaseColor[3] = { 1.0f, 1.0f, 1.0f }; | ||||
| const float GLGizmoBase::HighlightColor[3] = { 1.0f, 0.38f, 0.0f }; | ||||
| 
 | ||||
| GLGizmoBase::Grabber::Grabber() | ||||
|     : center(Pointf(0.0, 0.0)) | ||||
|     , angle_z(0.0f) | ||||
| { | ||||
|     color[0] = 1.0f; | ||||
|     color[1] = 1.0f; | ||||
|     color[2] = 1.0f; | ||||
| } | ||||
| 
 | ||||
| void GLGizmoBase::Grabber::render(bool hover) const | ||||
| { | ||||
|     float min_x = -HalfSize; | ||||
|     float max_x = +HalfSize; | ||||
|     float min_y = -HalfSize; | ||||
|     float max_y = +HalfSize; | ||||
| 
 | ||||
|     ::glColor3f((GLfloat)color[0], (GLfloat)color[1], (GLfloat)color[2]); | ||||
| 
 | ||||
|     float angle_z_in_deg = angle_z * 180.0f / (float)PI; | ||||
|     ::glPushMatrix(); | ||||
|     ::glTranslatef((GLfloat)center.x, (GLfloat)center.y, 0.0f); | ||||
|     ::glRotatef((GLfloat)angle_z_in_deg, 0.0f, 0.0f, 1.0f); | ||||
| 
 | ||||
|     ::glDisable(GL_CULL_FACE); | ||||
|     ::glBegin(GL_TRIANGLES); | ||||
|     ::glVertex3f((GLfloat)min_x, (GLfloat)min_y, 0.0f); | ||||
|     ::glVertex3f((GLfloat)max_x, (GLfloat)min_y, 0.0f); | ||||
|     ::glVertex3f((GLfloat)max_x, (GLfloat)max_y, 0.0f); | ||||
|     ::glVertex3f((GLfloat)max_x, (GLfloat)max_y, 0.0f); | ||||
|     ::glVertex3f((GLfloat)min_x, (GLfloat)max_y, 0.0f); | ||||
|     ::glVertex3f((GLfloat)min_x, (GLfloat)min_y, 0.0f); | ||||
|     ::glEnd(); | ||||
|     ::glEnable(GL_CULL_FACE); | ||||
| 
 | ||||
|     if (hover) | ||||
|     { | ||||
|         min_x -= HoverOffset; | ||||
|         max_x += HoverOffset; | ||||
|         min_y -= HoverOffset; | ||||
|         max_y += HoverOffset; | ||||
| 
 | ||||
|         ::glBegin(GL_LINE_LOOP); | ||||
|         ::glVertex3f((GLfloat)min_x, (GLfloat)min_y, 0.0f); | ||||
|         ::glVertex3f((GLfloat)max_x, (GLfloat)min_y, 0.0f); | ||||
|         ::glVertex3f((GLfloat)max_x, (GLfloat)max_y, 0.0f); | ||||
|         ::glVertex3f((GLfloat)min_x, (GLfloat)max_y, 0.0f); | ||||
|         ::glEnd(); | ||||
|     } | ||||
| 
 | ||||
|     ::glPopMatrix(); | ||||
| } | ||||
| 
 | ||||
| GLGizmoBase::GLGizmoBase() | ||||
|     : m_state(Off) | ||||
|     , m_hover_id(-1) | ||||
| { | ||||
| } | ||||
| 
 | ||||
| GLGizmoBase::~GLGizmoBase() | ||||
| { | ||||
| } | ||||
| 
 | ||||
| bool GLGizmoBase::init() | ||||
| { | ||||
|     return on_init(); | ||||
| } | ||||
| 
 | ||||
| GLGizmoBase::EState GLGizmoBase::get_state() const | ||||
| { | ||||
|     return m_state; | ||||
| } | ||||
| 
 | ||||
| void GLGizmoBase::set_state(GLGizmoBase::EState state) | ||||
| { | ||||
|     m_state = state; | ||||
| } | ||||
| 
 | ||||
| unsigned int GLGizmoBase::get_textures_id() const | ||||
| { | ||||
|     return m_textures[m_state].get_id(); | ||||
| } | ||||
| 
 | ||||
| int GLGizmoBase::get_textures_size() const | ||||
| { | ||||
|     return m_textures[Off].get_width(); | ||||
| } | ||||
| 
 | ||||
| int GLGizmoBase::get_hover_id() const | ||||
| { | ||||
|     return m_hover_id; | ||||
| } | ||||
| 
 | ||||
| void GLGizmoBase::set_hover_id(int id) | ||||
| { | ||||
|     if (id < (int)m_grabbers.size()) | ||||
|         m_hover_id = id; | ||||
| } | ||||
| 
 | ||||
| void GLGizmoBase::start_dragging() | ||||
| { | ||||
|     on_start_dragging(); | ||||
| } | ||||
| 
 | ||||
| void GLGizmoBase::update(const Pointf& mouse_pos) | ||||
| { | ||||
|     if (m_hover_id != -1) | ||||
|         on_update(mouse_pos); | ||||
| } | ||||
| 
 | ||||
| void GLGizmoBase::render(const BoundingBoxf3& box) const | ||||
| { | ||||
|     on_render(box); | ||||
| } | ||||
| 
 | ||||
| void GLGizmoBase::render_for_picking(const BoundingBoxf3& box) const | ||||
| { | ||||
|     on_render_for_picking(box); | ||||
| } | ||||
| 
 | ||||
| void GLGizmoBase::on_start_dragging() | ||||
| { | ||||
| } | ||||
| 
 | ||||
| void GLGizmoBase::render_grabbers() const | ||||
| { | ||||
|     for (unsigned int i = 0; i < (unsigned int)m_grabbers.size(); ++i) | ||||
|     { | ||||
|         m_grabbers[i].render(m_hover_id == i); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| const float GLGizmoRotate::Offset = 5.0f; | ||||
| const unsigned int GLGizmoRotate::CircleResolution = 64; | ||||
| const unsigned int GLGizmoRotate::AngleResolution = 64; | ||||
| const unsigned int GLGizmoRotate::ScaleStepsCount = 60; | ||||
| const float GLGizmoRotate::ScaleStepRad = 2.0f * (float)PI / GLGizmoRotate::ScaleStepsCount; | ||||
| const unsigned int GLGizmoRotate::ScaleLongEvery = 5; | ||||
| const float GLGizmoRotate::ScaleLongTooth = 2.0f; | ||||
| const float GLGizmoRotate::ScaleShortTooth = 1.0f; | ||||
| const unsigned int GLGizmoRotate::SnapRegionsCount = 8; | ||||
| const float GLGizmoRotate::GrabberOffset = 5.0f; | ||||
| 
 | ||||
| GLGizmoRotate::GLGizmoRotate() | ||||
|     : GLGizmoBase() | ||||
|     , m_angle_z(0.0f) | ||||
|     , m_center(Pointf(0.0, 0.0)) | ||||
|     , m_radius(0.0f) | ||||
| { | ||||
| } | ||||
| 
 | ||||
| bool GLGizmoRotate::on_init() | ||||
| { | ||||
|     std::string path = resources_dir() + "/icons/overlay/"; | ||||
| 
 | ||||
|     std::string filename = path + "rotate_off.png"; | ||||
|     if (!m_textures[Off].load_from_file(filename, false)) | ||||
|         return false; | ||||
| 
 | ||||
|     filename = path + "rotate_hover.png"; | ||||
|     if (!m_textures[Hover].load_from_file(filename, false)) | ||||
|         return false; | ||||
| 
 | ||||
|     filename = path + "rotate_on.png"; | ||||
|     if (!m_textures[On].load_from_file(filename, false)) | ||||
|         return false; | ||||
| 
 | ||||
|     m_grabbers.push_back(Grabber()); | ||||
| 
 | ||||
|     return true; | ||||
| } | ||||
| 
 | ||||
| void GLGizmoRotate::on_update(const Pointf& mouse_pos) | ||||
| { | ||||
|     Vectorf orig_dir(1.0, 0.0); | ||||
|     Vectorf new_dir = normalize(mouse_pos - m_center); | ||||
|     coordf_t theta = ::acos(clamp(-1.0, 1.0, dot(new_dir, orig_dir))); | ||||
|     if (cross(orig_dir, new_dir) < 0.0) | ||||
|         theta = 2.0 * (coordf_t)PI - theta; | ||||
| 
 | ||||
|     if (length(m_center.vector_to(mouse_pos)) < 2.0 * (double)m_radius / 3.0) | ||||
|     { | ||||
|         coordf_t step = 2.0 * (coordf_t)PI / (coordf_t)SnapRegionsCount; | ||||
|         theta = step * (coordf_t)std::round(theta / step); | ||||
|     } | ||||
| 
 | ||||
|     if (theta == 2.0 * (coordf_t)PI) | ||||
|         theta = 0.0; | ||||
|      | ||||
|     m_angle_z = (float)theta; | ||||
| } | ||||
| 
 | ||||
| void GLGizmoRotate::on_render(const BoundingBoxf3& box) const | ||||
| { | ||||
|     ::glDisable(GL_LIGHTING); | ||||
|     ::glDisable(GL_DEPTH_TEST); | ||||
| 
 | ||||
|     const Pointf3& size = box.size(); | ||||
|     m_center = box.center(); | ||||
|     m_radius = Offset + ::sqrt(sqr(0.5f * size.x) + sqr(0.5f * size.y)); | ||||
| 
 | ||||
|     ::glLineWidth(2.0f); | ||||
|     ::glColor3fv(BaseColor); | ||||
| 
 | ||||
|     _render_circle(); | ||||
|     _render_scale(); | ||||
|     _render_snap_radii(); | ||||
|     _render_reference_radius(); | ||||
| 
 | ||||
|     ::glColor3fv(HighlightColor); | ||||
|     _render_angle_z(); | ||||
|     _render_grabber(); | ||||
| } | ||||
| 
 | ||||
| void GLGizmoRotate::on_render_for_picking(const BoundingBoxf3& box) const | ||||
| { | ||||
|     ::glDisable(GL_LIGHTING); | ||||
|     ::glDisable(GL_DEPTH_TEST); | ||||
| 
 | ||||
|     m_grabbers[0].color[0] = 1.0f; | ||||
|     m_grabbers[0].color[1] = 1.0f; | ||||
|     m_grabbers[0].color[2] = 254.0f / 255.0f; | ||||
|     render_grabbers(); | ||||
| } | ||||
| 
 | ||||
| void GLGizmoRotate::_render_circle() const | ||||
| { | ||||
|     ::glBegin(GL_LINE_LOOP); | ||||
|     for (unsigned int i = 0; i < ScaleStepsCount; ++i) | ||||
|     { | ||||
|         float angle = (float)i * ScaleStepRad; | ||||
|         float x = m_center.x + ::cos(angle) * m_radius; | ||||
|         float y = m_center.y + ::sin(angle) * m_radius; | ||||
|         ::glVertex3f((GLfloat)x, (GLfloat)y, 0.0f); | ||||
|     } | ||||
|     ::glEnd(); | ||||
| } | ||||
| 
 | ||||
| void GLGizmoRotate::_render_scale() const | ||||
| { | ||||
|     float out_radius_long = m_radius + ScaleLongTooth; | ||||
|     float out_radius_short = m_radius + ScaleShortTooth; | ||||
| 
 | ||||
|     ::glBegin(GL_LINES); | ||||
|     for (unsigned int i = 0; i < ScaleStepsCount; ++i) | ||||
|     { | ||||
|         float angle = (float)i * ScaleStepRad; | ||||
|         float cosa = ::cos(angle); | ||||
|         float sina = ::sin(angle); | ||||
|         float in_x = m_center.x + cosa * m_radius; | ||||
|         float in_y = m_center.y + sina * m_radius; | ||||
|         float out_x = (i % ScaleLongEvery == 0) ? m_center.x + cosa * out_radius_long : m_center.x + cosa * out_radius_short; | ||||
|         float out_y = (i % ScaleLongEvery == 0) ? m_center.y + sina * out_radius_long : m_center.y + sina * out_radius_short; | ||||
|         ::glVertex3f((GLfloat)in_x, (GLfloat)in_y, 0.0f); | ||||
|         ::glVertex3f((GLfloat)out_x, (GLfloat)out_y, 0.0f); | ||||
|     } | ||||
|     ::glEnd(); | ||||
| } | ||||
| 
 | ||||
| void GLGizmoRotate::_render_snap_radii() const | ||||
| { | ||||
|     float step = 2.0f * (float)PI / (float)SnapRegionsCount; | ||||
| 
 | ||||
|     float in_radius = m_radius / 3.0f; | ||||
|     float out_radius = 2.0f * in_radius; | ||||
| 
 | ||||
|     ::glBegin(GL_LINES); | ||||
|     for (unsigned int i = 0; i < SnapRegionsCount; ++i) | ||||
|     { | ||||
|         float angle = (float)i * step; | ||||
|         float cosa = ::cos(angle); | ||||
|         float sina = ::sin(angle); | ||||
|         float in_x = m_center.x + cosa * in_radius; | ||||
|         float in_y = m_center.y + sina * in_radius; | ||||
|         float out_x = m_center.x + cosa * out_radius; | ||||
|         float out_y = m_center.y + sina * out_radius; | ||||
|         ::glVertex3f((GLfloat)in_x, (GLfloat)in_y, 0.0f); | ||||
|         ::glVertex3f((GLfloat)out_x, (GLfloat)out_y, 0.0f); | ||||
|     } | ||||
|     ::glEnd(); | ||||
| } | ||||
| 
 | ||||
| void GLGizmoRotate::_render_reference_radius() const | ||||
| { | ||||
|     ::glBegin(GL_LINES); | ||||
|     ::glVertex3f((GLfloat)m_center.x, (GLfloat)m_center.y, 0.0f); | ||||
|     ::glVertex3f((GLfloat)m_center.x + m_radius + GrabberOffset, (GLfloat)m_center.y, 0.0f); | ||||
|     ::glEnd(); | ||||
| } | ||||
| 
 | ||||
| void GLGizmoRotate::_render_angle_z() const | ||||
| { | ||||
|     float step_angle = m_angle_z / AngleResolution; | ||||
|     float ex_radius = m_radius + GrabberOffset; | ||||
| 
 | ||||
|     ::glBegin(GL_LINE_STRIP); | ||||
|     for (unsigned int i = 0; i <= AngleResolution; ++i) | ||||
|     { | ||||
|         float angle = (float)i * step_angle; | ||||
|         float x = m_center.x + ::cos(angle) * ex_radius; | ||||
|         float y = m_center.y + ::sin(angle) * ex_radius; | ||||
|         ::glVertex3f((GLfloat)x, (GLfloat)y, 0.0f); | ||||
|     } | ||||
|     ::glEnd(); | ||||
| } | ||||
| 
 | ||||
| void GLGizmoRotate::_render_grabber() const | ||||
| { | ||||
|     float grabber_radius = m_radius + GrabberOffset; | ||||
|     m_grabbers[0].center.x = m_center.x + ::cos(m_angle_z) * grabber_radius; | ||||
|     m_grabbers[0].center.y = m_center.y + ::sin(m_angle_z) * grabber_radius; | ||||
|     m_grabbers[0].angle_z = m_angle_z; | ||||
| 
 | ||||
|     ::glColor3fv(BaseColor); | ||||
|     ::glBegin(GL_LINES); | ||||
|     ::glVertex3f((GLfloat)m_center.x, (GLfloat)m_center.y, 0.0f); | ||||
|     ::glVertex3f((GLfloat)m_grabbers[0].center.x, (GLfloat)m_grabbers[0].center.y, 0.0f); | ||||
|     ::glEnd(); | ||||
| 
 | ||||
|     ::memcpy((void*)m_grabbers[0].color, (const void*)HighlightColor, 3 * sizeof(float)); | ||||
|     render_grabbers(); | ||||
| } | ||||
| 
 | ||||
| const float GLGizmoScale::Offset = 5.0f; | ||||
| 
 | ||||
| GLGizmoScale::GLGizmoScale() | ||||
|     : GLGizmoBase() | ||||
|     , m_scale(1.0f) | ||||
|     , m_starting_scale(1.0f) | ||||
| { | ||||
| } | ||||
| 
 | ||||
| float GLGizmoScale::get_scale() const | ||||
| { | ||||
|     return m_scale; | ||||
| } | ||||
| 
 | ||||
| void GLGizmoScale::set_scale(float scale) | ||||
| { | ||||
|     m_starting_scale = scale; | ||||
| } | ||||
| 
 | ||||
| bool GLGizmoScale::on_init() | ||||
| { | ||||
|     std::string path = resources_dir() + "/icons/overlay/"; | ||||
| 
 | ||||
|     std::string filename = path + "scale_off.png"; | ||||
|     if (!m_textures[Off].load_from_file(filename, false)) | ||||
|         return false; | ||||
| 
 | ||||
|     filename = path + "scale_hover.png"; | ||||
|     if (!m_textures[Hover].load_from_file(filename, false)) | ||||
|         return false; | ||||
| 
 | ||||
|     filename = path + "scale_on.png"; | ||||
|     if (!m_textures[On].load_from_file(filename, false)) | ||||
|         return false; | ||||
| 
 | ||||
|     for (unsigned int i = 0; i < 4; ++i) | ||||
|     { | ||||
|         m_grabbers.push_back(Grabber()); | ||||
|     } | ||||
| 
 | ||||
|     return true; | ||||
| } | ||||
| 
 | ||||
| void GLGizmoScale::on_start_dragging() | ||||
| { | ||||
|     if (m_hover_id != -1) | ||||
|         m_starting_drag_position = m_grabbers[m_hover_id].center; | ||||
| } | ||||
| 
 | ||||
| void GLGizmoScale::on_update(const Pointf& mouse_pos) | ||||
| { | ||||
|     Pointf center(0.5 * (m_grabbers[1].center.x + m_grabbers[0].center.x), 0.5 * (m_grabbers[3].center.y + m_grabbers[0].center.y)); | ||||
| 
 | ||||
|     coordf_t orig_len = length(m_starting_drag_position - center); | ||||
|     coordf_t new_len = length(mouse_pos - center); | ||||
|     coordf_t ratio = (orig_len != 0.0) ? new_len / orig_len : 1.0; | ||||
| 
 | ||||
|     m_scale = m_starting_scale * (float)ratio; | ||||
| } | ||||
| 
 | ||||
| void GLGizmoScale::on_render(const BoundingBoxf3& box) const | ||||
| { | ||||
|     ::glDisable(GL_LIGHTING); | ||||
|     ::glDisable(GL_DEPTH_TEST); | ||||
| 
 | ||||
|     coordf_t min_x = box.min.x - (coordf_t)Offset; | ||||
|     coordf_t max_x = box.max.x + (coordf_t)Offset; | ||||
|     coordf_t min_y = box.min.y - (coordf_t)Offset; | ||||
|     coordf_t max_y = box.max.y + (coordf_t)Offset; | ||||
| 
 | ||||
|     m_grabbers[0].center.x = min_x; | ||||
|     m_grabbers[0].center.y = min_y; | ||||
|     m_grabbers[1].center.x = max_x; | ||||
|     m_grabbers[1].center.y = min_y; | ||||
|     m_grabbers[2].center.x = max_x; | ||||
|     m_grabbers[2].center.y = max_y; | ||||
|     m_grabbers[3].center.x = min_x; | ||||
|     m_grabbers[3].center.y = max_y; | ||||
| 
 | ||||
|     ::glLineWidth(2.0f); | ||||
|     ::glColor3fv(BaseColor); | ||||
|     // draw outline
 | ||||
|     ::glBegin(GL_LINE_LOOP); | ||||
|     for (unsigned int i = 0; i < 4; ++i) | ||||
|     { | ||||
|         ::glVertex3f((GLfloat)m_grabbers[i].center.x, (GLfloat)m_grabbers[i].center.y, 0.0f); | ||||
|     } | ||||
|     ::glEnd(); | ||||
| 
 | ||||
|     // draw grabbers
 | ||||
|     for (unsigned int i = 0; i < 4; ++i) | ||||
|     { | ||||
|         ::memcpy((void*)m_grabbers[i].color, (const void*)HighlightColor, 3 * sizeof(float)); | ||||
|     } | ||||
|     render_grabbers(); | ||||
| } | ||||
| 
 | ||||
| void GLGizmoScale::on_render_for_picking(const BoundingBoxf3& box) const | ||||
| { | ||||
|     static const GLfloat INV_255 = 1.0f / 255.0f; | ||||
| 
 | ||||
|     ::glDisable(GL_LIGHTING); | ||||
|     ::glDisable(GL_DEPTH_TEST); | ||||
| 
 | ||||
|     for (unsigned int i = 0; i < 4; ++i) | ||||
|     { | ||||
|         m_grabbers[i].color[0] = 1.0f; | ||||
|         m_grabbers[i].color[1] = 1.0f; | ||||
|         m_grabbers[i].color[2] = (254.0f - (float)i) * INV_255; | ||||
|     } | ||||
|     render_grabbers(); | ||||
| } | ||||
| 
 | ||||
| } // namespace GUI
 | ||||
| } // namespace Slic3r
 | ||||
							
								
								
									
										145
									
								
								xs/src/slic3r/GUI/GLGizmo.hpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										145
									
								
								xs/src/slic3r/GUI/GLGizmo.hpp
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,145 @@ | |||
| #ifndef slic3r_GLGizmo_hpp_ | ||||
| #define slic3r_GLGizmo_hpp_ | ||||
| 
 | ||||
| #include "../../slic3r/GUI/GLTexture.hpp" | ||||
| #include "../../libslic3r/Point.hpp" | ||||
| 
 | ||||
| #include <vector> | ||||
| 
 | ||||
| namespace Slic3r { | ||||
| 
 | ||||
| class BoundingBoxf3; | ||||
| class Pointf3; | ||||
| 
 | ||||
| namespace GUI { | ||||
| 
 | ||||
| class GLGizmoBase | ||||
| { | ||||
| protected: | ||||
|     static const float BaseColor[3]; | ||||
|     static const float HighlightColor[3]; | ||||
| 
 | ||||
|     struct Grabber | ||||
|     { | ||||
|         static const float HalfSize; | ||||
|         static const float HoverOffset; | ||||
| 
 | ||||
|         Pointf center; | ||||
|         float angle_z; | ||||
|         float color[3]; | ||||
| 
 | ||||
|         Grabber(); | ||||
|         void render(bool hover) const; | ||||
|     }; | ||||
| 
 | ||||
| public: | ||||
|     enum EState | ||||
|     { | ||||
|         Off, | ||||
|         Hover, | ||||
|         On, | ||||
|         Num_States | ||||
|     }; | ||||
| 
 | ||||
| protected: | ||||
|     EState m_state; | ||||
|     // textures are assumed to be square and all with the same size in pixels, no internal check is done
 | ||||
|     GLTexture m_textures[Num_States]; | ||||
|     int m_hover_id; | ||||
|     mutable std::vector<Grabber> m_grabbers; | ||||
| 
 | ||||
| public: | ||||
|     GLGizmoBase(); | ||||
|     virtual ~GLGizmoBase(); | ||||
| 
 | ||||
|     bool init(); | ||||
| 
 | ||||
|     EState get_state() const; | ||||
|     void set_state(EState state); | ||||
| 
 | ||||
|     unsigned int get_textures_id() const; | ||||
|     int get_textures_size() const; | ||||
| 
 | ||||
|     int get_hover_id() const; | ||||
|     void set_hover_id(int id); | ||||
| 
 | ||||
|     void start_dragging(); | ||||
|     void update(const Pointf& mouse_pos); | ||||
| 
 | ||||
|     void render(const BoundingBoxf3& box) const; | ||||
|     void render_for_picking(const BoundingBoxf3& box) const; | ||||
| 
 | ||||
| protected: | ||||
|     virtual bool on_init() = 0; | ||||
|     virtual void on_start_dragging(); | ||||
|     virtual void on_update(const Pointf& mouse_pos) = 0; | ||||
|     virtual void on_render(const BoundingBoxf3& box) const = 0; | ||||
|     virtual void on_render_for_picking(const BoundingBoxf3& box) const = 0; | ||||
| 
 | ||||
|     void render_grabbers() const; | ||||
| }; | ||||
| 
 | ||||
| class GLGizmoRotate : public GLGizmoBase | ||||
| { | ||||
|     static const float Offset; | ||||
|     static const unsigned int CircleResolution; | ||||
|     static const unsigned int AngleResolution; | ||||
|     static const unsigned int ScaleStepsCount; | ||||
|     static const float ScaleStepRad; | ||||
|     static const unsigned int ScaleLongEvery; | ||||
|     static const float ScaleLongTooth; | ||||
|     static const float ScaleShortTooth; | ||||
|     static const unsigned int SnapRegionsCount; | ||||
|     static const float GrabberOffset; | ||||
| 
 | ||||
|     float m_angle_z; | ||||
| 
 | ||||
|     mutable Pointf m_center; | ||||
|     mutable float m_radius; | ||||
| 
 | ||||
| public: | ||||
|     GLGizmoRotate(); | ||||
| 
 | ||||
| protected: | ||||
|     virtual bool on_init(); | ||||
|     virtual void on_update(const Pointf& mouse_pos); | ||||
|     virtual void on_render(const BoundingBoxf3& box) const; | ||||
|     virtual void on_render_for_picking(const BoundingBoxf3& box) const; | ||||
| 
 | ||||
| private: | ||||
|     void _render_circle() const; | ||||
|     void _render_scale() const; | ||||
|     void _render_snap_radii() const; | ||||
|     void _render_reference_radius() const; | ||||
|     void _render_angle_z() const; | ||||
|     void _render_grabber() const; | ||||
| }; | ||||
| 
 | ||||
| class GLGizmoScale : public GLGizmoBase | ||||
| { | ||||
|     static const float Offset; | ||||
| 
 | ||||
|     float m_scale; | ||||
| 
 | ||||
|     Pointf m_starting_drag_position; | ||||
|     float m_starting_scale; | ||||
| 
 | ||||
| public: | ||||
|     GLGizmoScale(); | ||||
| 
 | ||||
|     float get_scale() const; | ||||
|     void set_scale(float scale); | ||||
| 
 | ||||
| protected: | ||||
|     virtual bool on_init(); | ||||
|     virtual void on_start_dragging(); | ||||
|     virtual void on_update(const Pointf& mouse_pos); | ||||
|     virtual void on_render(const BoundingBoxf3& box) const; | ||||
|     virtual void on_render_for_picking(const BoundingBoxf3& box) const; | ||||
| }; | ||||
| 
 | ||||
| } // namespace GUI
 | ||||
| } // namespace Slic3r
 | ||||
| 
 | ||||
| #endif // slic3r_GLGizmo_hpp_
 | ||||
| 
 | ||||
|  | @ -2,6 +2,9 @@ | |||
| 
 | ||||
| #include "GLShader.hpp" | ||||
| 
 | ||||
| #include "../../libslic3r/Utils.hpp" | ||||
| #include <boost/nowide/fstream.hpp> | ||||
| 
 | ||||
| #include <string> | ||||
| #include <utility> | ||||
| #include <assert.h> | ||||
|  | @ -22,7 +25,7 @@ inline std::string gl_get_string_safe(GLenum param) | |||
|     return std::string(value ? value : "N/A"); | ||||
| } | ||||
| 
 | ||||
| bool GLShader::load(const char *fragment_shader, const char *vertex_shader) | ||||
| bool GLShader::load_from_text(const char *fragment_shader, const char *vertex_shader) | ||||
| { | ||||
|     std::string gl_version = gl_get_string_safe(GL_VERSION); | ||||
|     int major = atoi(gl_version.c_str()); | ||||
|  | @ -123,6 +126,41 @@ bool GLShader::load(const char *fragment_shader, const char *vertex_shader) | |||
|     return true; | ||||
| } | ||||
| 
 | ||||
| bool GLShader::load_from_file(const char* fragment_shader_filename, const char* vertex_shader_filename) | ||||
| { | ||||
|     const std::string& path = resources_dir() + "/shaders/"; | ||||
| 
 | ||||
|     boost::nowide::ifstream vs(path + std::string(vertex_shader_filename), boost::nowide::ifstream::binary); | ||||
|     if (!vs.good()) | ||||
|         return false; | ||||
| 
 | ||||
|     vs.seekg(0, vs.end); | ||||
|     int file_length = vs.tellg(); | ||||
|     vs.seekg(0, vs.beg); | ||||
|     std::string vertex_shader(file_length, '\0'); | ||||
|     vs.read(const_cast<char*>(vertex_shader.data()), file_length); | ||||
|     if (!vs.good()) | ||||
|         return false; | ||||
| 
 | ||||
|     vs.close(); | ||||
| 
 | ||||
|     boost::nowide::ifstream fs(path + std::string(fragment_shader_filename), boost::nowide::ifstream::binary); | ||||
|     if (!fs.good()) | ||||
|         return false; | ||||
| 
 | ||||
|     fs.seekg(0, fs.end); | ||||
|     file_length = fs.tellg(); | ||||
|     fs.seekg(0, fs.beg); | ||||
|     std::string fragment_shader(file_length, '\0'); | ||||
|     fs.read(const_cast<char*>(fragment_shader.data()), file_length); | ||||
|     if (!fs.good()) | ||||
|         return false; | ||||
| 
 | ||||
|     fs.close(); | ||||
| 
 | ||||
|     return load_from_text(fragment_shader.c_str(), vertex_shader.c_str()); | ||||
| } | ||||
| 
 | ||||
| void GLShader::release() | ||||
| { | ||||
|     if (this->shader_program_id) { | ||||
|  |  | |||
|  | @ -16,7 +16,9 @@ public: | |||
|         {} | ||||
|     ~GLShader(); | ||||
| 
 | ||||
|     bool load(const char *fragment_shader, const char *vertex_shader); | ||||
|     bool load_from_text(const char *fragment_shader, const char *vertex_shader); | ||||
|     bool load_from_file(const char* fragment_shader_filename, const char* vertex_shader_filename); | ||||
| 
 | ||||
|     void release(); | ||||
| 
 | ||||
|     int  get_attrib_location(const char *name) const; | ||||
|  |  | |||
							
								
								
									
										190
									
								
								xs/src/slic3r/GUI/GLTexture.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										190
									
								
								xs/src/slic3r/GUI/GLTexture.cpp
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,190 @@ | |||
| #include "GLTexture.hpp" | ||||
| 
 | ||||
| #include <GL/glew.h> | ||||
| 
 | ||||
| #include <wx/image.h> | ||||
| 
 | ||||
| #include <boost/filesystem.hpp> | ||||
| 
 | ||||
| #include <vector> | ||||
| #include <algorithm> | ||||
| 
 | ||||
| namespace Slic3r { | ||||
| namespace GUI { | ||||
| 
 | ||||
| GLTexture::GLTexture() | ||||
|     : m_id(0) | ||||
|     , m_width(0) | ||||
|     , m_height(0) | ||||
|     , m_source("") | ||||
| { | ||||
| } | ||||
| 
 | ||||
| GLTexture::~GLTexture() | ||||
| { | ||||
|     reset(); | ||||
| } | ||||
| 
 | ||||
| bool GLTexture::load_from_file(const std::string& filename, bool generate_mipmaps) | ||||
| { | ||||
|     reset(); | ||||
| 
 | ||||
|     if (!boost::filesystem::exists(filename)) | ||||
|         return false; | ||||
| 
 | ||||
|     // Load a PNG with an alpha channel.
 | ||||
|     wxImage image; | ||||
|     if (!image.LoadFile(filename, wxBITMAP_TYPE_PNG)) | ||||
|     { | ||||
|         reset(); | ||||
|         return false; | ||||
|     } | ||||
| 
 | ||||
|     m_width = image.GetWidth(); | ||||
|     m_height = image.GetHeight(); | ||||
|     int n_pixels = m_width * m_height; | ||||
| 
 | ||||
|     if (n_pixels <= 0) | ||||
|     { | ||||
|         reset(); | ||||
|         return false; | ||||
|     } | ||||
| 
 | ||||
|     // Get RGB & alpha raw data from wxImage, pack them into an array.
 | ||||
|     unsigned char* img_rgb = image.GetData(); | ||||
|     if (img_rgb == nullptr) | ||||
|     { | ||||
|         reset(); | ||||
|         return false; | ||||
|     } | ||||
| 
 | ||||
|     unsigned char* img_alpha = image.GetAlpha(); | ||||
| 
 | ||||
|     std::vector<unsigned char> data(n_pixels * 4, 0); | ||||
|     for (int i = 0; i < n_pixels; ++i) | ||||
|     { | ||||
|         int data_id = i * 4; | ||||
|         int img_id = i * 3; | ||||
|         data[data_id + 0] = img_rgb[img_id + 0]; | ||||
|         data[data_id + 1] = img_rgb[img_id + 1]; | ||||
|         data[data_id + 2] = img_rgb[img_id + 2]; | ||||
|         data[data_id + 3] = (img_alpha != nullptr) ? img_alpha[i] : 255; | ||||
|     } | ||||
| 
 | ||||
|     // sends data to gpu
 | ||||
|     ::glGenTextures(1, &m_id); | ||||
|     ::glBindTexture(GL_TEXTURE_2D, m_id); | ||||
|     ::glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA8, (GLsizei)m_width, (GLsizei)m_height, 0, GL_RGBA, GL_UNSIGNED_BYTE, (const void*)data.data()); | ||||
|     if (generate_mipmaps) | ||||
|     { | ||||
|         // we manually generate mipmaps because glGenerateMipmap() function is not reliable on all graphics cards
 | ||||
|         _generate_mipmaps(image); | ||||
|         ::glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR_MIPMAP_LINEAR); | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|         ::glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR); | ||||
|         ::glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAX_LEVEL, 1); | ||||
|     } | ||||
|     ::glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR); | ||||
| 
 | ||||
|     ::glBindTexture(GL_TEXTURE_2D, 0); | ||||
| 
 | ||||
|     m_source = filename; | ||||
|     return true; | ||||
| } | ||||
| 
 | ||||
| void GLTexture::reset() | ||||
| { | ||||
|     if (m_id != 0) | ||||
|         ::glDeleteTextures(1, &m_id); | ||||
| 
 | ||||
|     m_id = 0; | ||||
|     m_width = 0; | ||||
|     m_height = 0; | ||||
|     m_source = ""; | ||||
| } | ||||
| 
 | ||||
| unsigned int GLTexture::get_id() const | ||||
| { | ||||
|     return m_id; | ||||
| } | ||||
| 
 | ||||
| int GLTexture::get_width() const | ||||
| { | ||||
|     return m_width; | ||||
| } | ||||
| 
 | ||||
| int GLTexture::get_height() const | ||||
| { | ||||
|     return m_height; | ||||
| } | ||||
| 
 | ||||
| const std::string& GLTexture::get_source() const | ||||
| { | ||||
|     return m_source; | ||||
| } | ||||
| 
 | ||||
| void GLTexture::render_texture(unsigned int tex_id, float left, float right, float bottom, float top) | ||||
| { | ||||
|     ::glColor4f(1.0f, 1.0f, 1.0f, 1.0f); | ||||
| 
 | ||||
|     ::glDisable(GL_LIGHTING); | ||||
|     ::glEnable(GL_BLEND); | ||||
|     ::glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); | ||||
|     ::glEnable(GL_TEXTURE_2D); | ||||
| 
 | ||||
|     ::glBindTexture(GL_TEXTURE_2D, (GLuint)tex_id); | ||||
| 
 | ||||
|     ::glBegin(GL_QUADS); | ||||
|     ::glTexCoord2d(0.0f, 1.0f); ::glVertex3f(left, bottom, 0.0f); | ||||
|     ::glTexCoord2d(1.0f, 1.0f); ::glVertex3f(right, bottom, 0.0f); | ||||
|     ::glTexCoord2d(1.0f, 0.0f); ::glVertex3f(right, top, 0.0f); | ||||
|     ::glTexCoord2d(0.0f, 0.0f); ::glVertex3f(left, top, 0.0f); | ||||
|     ::glEnd(); | ||||
| 
 | ||||
|     ::glBindTexture(GL_TEXTURE_2D, 0); | ||||
| 
 | ||||
|     ::glDisable(GL_TEXTURE_2D); | ||||
|     ::glDisable(GL_BLEND); | ||||
|     ::glEnable(GL_LIGHTING); | ||||
| } | ||||
| 
 | ||||
| void GLTexture::_generate_mipmaps(wxImage& image) | ||||
| { | ||||
|     int w = image.GetWidth(); | ||||
|     int h = image.GetHeight(); | ||||
|     GLint level = 0; | ||||
|     std::vector<unsigned char> data(w * h * 4, 0); | ||||
| 
 | ||||
|     while ((w > 1) && (h > 1)) | ||||
|     { | ||||
|         ++level; | ||||
| 
 | ||||
|         w = std::max(w / 2, 1); | ||||
|         h = std::max(h / 2, 1); | ||||
| 
 | ||||
|         int n_pixels = w * h; | ||||
| 
 | ||||
|         image = image.ResampleBicubic(w, h); | ||||
| 
 | ||||
|         unsigned char* img_rgb = image.GetData(); | ||||
|         unsigned char* img_alpha = image.GetAlpha(); | ||||
| 
 | ||||
|         data.resize(n_pixels * 4); | ||||
|         for (int i = 0; i < n_pixels; ++i) | ||||
|         { | ||||
|             int data_id = i * 4; | ||||
|             int img_id = i * 3; | ||||
|             data[data_id + 0] = img_rgb[img_id + 0]; | ||||
|             data[data_id + 1] = img_rgb[img_id + 1]; | ||||
|             data[data_id + 2] = img_rgb[img_id + 2]; | ||||
|             data[data_id + 3] = (img_alpha != nullptr) ? img_alpha[i] : 255; | ||||
|         } | ||||
| 
 | ||||
|         ::glTexImage2D(GL_TEXTURE_2D, level, GL_RGBA8, (GLsizei)w, (GLsizei)h, 0, GL_RGBA, GL_UNSIGNED_BYTE, (const void*)data.data()); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| } // namespace GUI
 | ||||
| } // namespace Slic3r
 | ||||
							
								
								
									
										41
									
								
								xs/src/slic3r/GUI/GLTexture.hpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								xs/src/slic3r/GUI/GLTexture.hpp
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,41 @@ | |||
| #ifndef slic3r_GLTexture_hpp_ | ||||
| #define slic3r_GLTexture_hpp_ | ||||
| 
 | ||||
| #include <string> | ||||
| 
 | ||||
| class wxImage; | ||||
| 
 | ||||
| namespace Slic3r { | ||||
| namespace GUI { | ||||
| 
 | ||||
|     class GLTexture | ||||
|     { | ||||
|     private: | ||||
|         unsigned int m_id; | ||||
|         int m_width; | ||||
|         int m_height; | ||||
|         std::string m_source; | ||||
| 
 | ||||
|     public: | ||||
|         GLTexture(); | ||||
|         ~GLTexture(); | ||||
| 
 | ||||
|         bool load_from_file(const std::string& filename, bool generate_mipmaps); | ||||
|         void reset(); | ||||
| 
 | ||||
|         unsigned int get_id() const; | ||||
|         int get_width() const; | ||||
|         int get_height() const; | ||||
|         const std::string& get_source() const; | ||||
| 
 | ||||
|         static void render_texture(unsigned int tex_id, float left, float right, float bottom, float top); | ||||
| 
 | ||||
|     private: | ||||
|         void _generate_mipmaps(wxImage& image); | ||||
|     }; | ||||
| 
 | ||||
| } // namespace GUI
 | ||||
| } // namespace Slic3r
 | ||||
| 
 | ||||
| #endif // slic3r_GLTexture_hpp_
 | ||||
| 
 | ||||
|  | @ -56,6 +56,7 @@ | |||
| 
 | ||||
| #include "../Utils/PresetUpdater.hpp" | ||||
| #include "../Config/Snapshot.hpp" | ||||
| #include "3DScene.hpp" | ||||
| 
 | ||||
| 
 | ||||
| namespace Slic3r { namespace GUI { | ||||
|  | @ -316,10 +317,11 @@ void add_config_menu(wxMenuBar *menu, int event_preferences_changed, int event_l | |||
|     auto local_menu = new wxMenu(); | ||||
|     wxWindowID config_id_base = wxWindow::NewControlId((int)ConfigMenuCnt); | ||||
| 
 | ||||
|     const auto config_wizard_tooltip = wxString::Format(_(L("Run %s")), ConfigWizard::name()); | ||||
| 	auto config_wizard_name = _(ConfigWizard::name().wx_str()); | ||||
| 	const auto config_wizard_tooltip = wxString::Format(_(L("Run %s")), config_wizard_name); | ||||
|     // Cmd+, is standard on OS X - what about other operating systems?
 | ||||
|    	local_menu->Append(config_id_base + ConfigMenuWizard, 		ConfigWizard::name() + dots, 			config_wizard_tooltip); | ||||
|    	local_menu->Append(config_id_base + ConfigMenuSnapshots, 	_(L("Configuration Snapshots"))+dots,	_(L("Inspect / activate configuration snapshots"))); | ||||
| 	local_menu->Append(config_id_base + ConfigMenuWizard, 		config_wizard_name + dots,					config_wizard_tooltip); | ||||
|    	local_menu->Append(config_id_base + ConfigMenuSnapshots, 	_(L("Configuration Snapshots"))+dots,		_(L("Inspect / activate configuration snapshots"))); | ||||
|    	local_menu->Append(config_id_base + ConfigMenuTakeSnapshot, _(L("Take Configuration Snapshot")), 		_(L("Capture a configuration snapshot"))); | ||||
| // 	local_menu->Append(config_id_base + ConfigMenuUpdate, 		_(L("Check for updates")), 					_(L("Check for configuration updates")));
 | ||||
|    	local_menu->AppendSeparator(); | ||||
|  | @ -378,6 +380,7 @@ void add_config_menu(wxMenuBar *menu, int event_preferences_changed, int event_l | |||
| 				save_language(); | ||||
| 				show_info(g_wxTabPanel, _(L("Application will be restarted")), _(L("Attention!"))); | ||||
| 				if (event_language_change > 0) { | ||||
| 					_3DScene::remove_all_canvases();// remove all canvas before recreate GUI
 | ||||
| 					wxCommandEvent event(event_language_change); | ||||
| 					g_wxApp->ProcessEvent(event); | ||||
| 				} | ||||
|  | @ -423,7 +426,7 @@ bool check_unsaved_changes() | |||
| 
 | ||||
| bool config_wizard_startup(bool app_config_exists) | ||||
| { | ||||
| 	if (! app_config_exists || g_PresetBundle->has_defauls_only()) { | ||||
| 	if (! app_config_exists || g_PresetBundle->printers.size() <= 1) { | ||||
| 		config_wizard(ConfigWizard::RR_DATA_EMPTY); | ||||
| 		return true; | ||||
| 	} else if (g_AppConfig->legacy_datadir()) { | ||||
|  |  | |||
|  | @ -28,6 +28,8 @@ struct MsgDialog : wxDialog | |||
| 	MsgDialog &operator=(const MsgDialog &) = delete; | ||||
| 	virtual ~MsgDialog(); | ||||
| 
 | ||||
| 	// TODO: refactor with CreateStdDialogButtonSizer usage
 | ||||
| 
 | ||||
| protected: | ||||
| 	enum { | ||||
| 		CONTENT_WIDTH = 500, | ||||
|  |  | |||
|  | @ -150,8 +150,15 @@ void OptionsGroup::append_line(const Line& line, wxStaticText**	colored_Label/* | |||
|     // Build a label if we have it
 | ||||
| 	wxStaticText* label=nullptr; | ||||
|     if (label_width != 0) { | ||||
| 		long label_style = staticbox ? 0 : wxALIGN_RIGHT; | ||||
| #ifdef __WXGTK__ | ||||
| 		// workaround for correct text align of the StaticBox on Linux
 | ||||
| 		// flags wxALIGN_RIGHT and wxALIGN_CENTRE don't work when Ellipsize flags are _not_ given.
 | ||||
| 		// Text is properly aligned only when Ellipsize is checked.
 | ||||
| 		label_style |= staticbox ? 0 : wxST_ELLIPSIZE_END; | ||||
| #endif /* __WXGTK__ */ | ||||
| 		label = new wxStaticText(parent(), wxID_ANY, line.label + (line.label.IsEmpty() ? "" : ":"),  | ||||
| 							wxDefaultPosition, wxSize(label_width, -1), staticbox ? 0 : wxALIGN_RIGHT); | ||||
| 							wxDefaultPosition, wxSize(label_width, -1), label_style); | ||||
|         label->SetFont(label_font); | ||||
|         label->Wrap(label_width); // avoid a Linux/GTK bug
 | ||||
| 		grid_sizer->Add(label, 0, (staticbox ? 0 : wxALIGN_RIGHT | wxRIGHT) | wxALIGN_CENTER_VERTICAL, 5); | ||||
|  |  | |||
|  | @ -40,49 +40,30 @@ void Tab::create_preset_tab(PresetBundle *preset_bundle) | |||
| 	m_preset_bundle = preset_bundle; | ||||
| 
 | ||||
| 	// Vertical sizer to hold the choice menu and the rest of the page.
 | ||||
| #ifdef __WXOSX__ | ||||
| 	auto  *main_sizer = new wxBoxSizer(wxVERTICAL); | ||||
| 	main_sizer->SetSizeHints(this); | ||||
| 	this->SetSizer(main_sizer); | ||||
| 
 | ||||
| 	// Create additional panel to Fit() it from OnActivate()
 | ||||
| 	// It's needed for tooltip showing on OSX
 | ||||
| 	m_tmp_panel = new wxPanel(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxBK_LEFT | wxTAB_TRAVERSAL); | ||||
| 	auto panel = m_tmp_panel;  | ||||
| 	auto  sizer = new wxBoxSizer(wxVERTICAL); | ||||
| 	m_tmp_panel->SetSizer(sizer); | ||||
| 	m_tmp_panel->Layout(); | ||||
| 
 | ||||
| 	main_sizer->Add(m_tmp_panel, 1, wxEXPAND | wxALL, 0); | ||||
| #else | ||||
| 	Tab *panel = this; | ||||
| 	auto  *sizer = new wxBoxSizer(wxVERTICAL); | ||||
| 	sizer->SetSizeHints(panel); | ||||
| 	panel->SetSizer(sizer); | ||||
| #endif //__WXOSX__
 | ||||
| 
 | ||||
| 	// preset chooser
 | ||||
| 	m_presets_choice = new wxBitmapComboBox(panel, wxID_ANY, "", wxDefaultPosition, wxSize(270, -1), 0, 0,wxCB_READONLY); | ||||
| 	/*
 | ||||
| 	m_cc_presets_choice = new wxComboCtrl(panel, wxID_ANY, L(""), wxDefaultPosition, wxDefaultSize, wxCB_READONLY); | ||||
| 	wxDataViewTreeCtrlComboPopup* popup = new wxDataViewTreeCtrlComboPopup; | ||||
| 	if (popup != nullptr) | ||||
| 	{ | ||||
| 		// FIXME If the following line is removed, the combo box popup list will not react to mouse clicks.
 | ||||
| 		//  On the other side, with this line the combo box popup cannot be closed by clicking on the combo button on Windows 10.
 | ||||
| //		m_cc_presets_choice->UseAltPopupWindow();
 | ||||
| 
 | ||||
| //		m_cc_presets_choice->EnablePopupAnimation(false);
 | ||||
| 		m_cc_presets_choice->SetPopupControl(popup); | ||||
| 		popup->SetStringValue(from_u8("Text1")); | ||||
| 
 | ||||
| 		popup->Bind(wxEVT_DATAVIEW_SELECTION_CHANGED, [this, popup](wxCommandEvent& evt) | ||||
| 		{ | ||||
| 			auto selected = popup->GetItemText(popup->GetSelection()); | ||||
| 			if (selected != _(L("System presets")) && selected != _(L("Default presets"))) | ||||
| 			{ | ||||
| 				m_cc_presets_choice->SetText(selected); | ||||
| 				std::string selected_string = selected.ToUTF8().data(); | ||||
| #ifdef __APPLE__ | ||||
| #else | ||||
|  				select_preset(selected_string); | ||||
| #endif | ||||
| 			}				 | ||||
| 		}); | ||||
| 
 | ||||
| // 		popup->Bind(wxEVT_KEY_DOWN, [popup](wxKeyEvent& evt) { popup->OnKeyEvent(evt); });
 | ||||
| // 		popup->Bind(wxEVT_KEY_UP, [popup](wxKeyEvent& evt) { popup->OnKeyEvent(evt); });
 | ||||
| 
 | ||||
| 		auto icons = new wxImageList(16, 16, true, 1); | ||||
| 		popup->SetImageList(icons); | ||||
| 		icons->Add(*new wxIcon(from_u8(Slic3r::var("flag-green-icon.png")), wxBITMAP_TYPE_PNG)); | ||||
| 		icons->Add(*new wxIcon(from_u8(Slic3r::var("flag-red-icon.png")), wxBITMAP_TYPE_PNG)); | ||||
| 	} | ||||
| */ | ||||
| 	auto color = wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOW); | ||||
| 
 | ||||
| 	//buttons
 | ||||
|  | @ -173,37 +154,6 @@ void Tab::create_preset_tab(PresetBundle *preset_bundle) | |||
| 	m_hsizer = new wxBoxSizer(wxHORIZONTAL); | ||||
| 	sizer->Add(m_hsizer, 1, wxEXPAND, 0); | ||||
| 
 | ||||
| 
 | ||||
| /*
 | ||||
| 
 | ||||
| 
 | ||||
| 	//temporary left vertical sizer
 | ||||
| 	m_left_sizer = new wxBoxSizer(wxVERTICAL); | ||||
| 	m_hsizer->Add(m_left_sizer, 0, wxEXPAND | wxLEFT | wxTOP | wxBOTTOM, 3); | ||||
| 
 | ||||
| 	// tree
 | ||||
| 	m_presetctrl = new wxDataViewTreeCtrl(panel, wxID_ANY, wxDefaultPosition, wxSize(200, -1), wxDV_NO_HEADER); | ||||
| 	m_left_sizer->Add(m_presetctrl, 1, wxEXPAND); | ||||
| 	m_preset_icons = new wxImageList(16, 16, true, 1); | ||||
| 	m_presetctrl->SetImageList(m_preset_icons); | ||||
| 	m_preset_icons->Add(*new wxIcon(from_u8(Slic3r::var("flag-green-icon.png")), wxBITMAP_TYPE_PNG)); | ||||
| 	m_preset_icons->Add(*new wxIcon(from_u8(Slic3r::var("flag-red-icon.png")), wxBITMAP_TYPE_PNG)); | ||||
| 
 | ||||
| 	m_presetctrl->Bind(wxEVT_DATAVIEW_SELECTION_CHANGED, [this](wxCommandEvent& evt) | ||||
| 	{ | ||||
| 		auto selected = m_presetctrl->GetItemText(m_presetctrl->GetSelection()); | ||||
| 		if (selected != _(L("System presets")) && selected != _(L("Default presets"))) | ||||
| 		{ | ||||
| 			std::string selected_string = selected.ToUTF8().data(); | ||||
| #ifdef __APPLE__ | ||||
| #else | ||||
| 			select_preset(selected_string); | ||||
| #endif | ||||
| 		} | ||||
| 	}); | ||||
| 
 | ||||
| */ | ||||
| 
 | ||||
| 	//left vertical sizer
 | ||||
| 	m_left_sizer = new wxBoxSizer(wxVERTICAL); | ||||
| 	m_hsizer->Add(m_left_sizer, 0, wxEXPAND | wxLEFT | wxTOP | wxBOTTOM, 3); | ||||
|  | @ -279,7 +229,12 @@ PageShp Tab::add_options_page(const wxString& title, const std::string& icon, bo | |||
| 		} | ||||
| 	} | ||||
| 	// Initialize the page.
 | ||||
| 	PageShp page(new Page(this, title, icon_idx)); | ||||
| #ifdef __WXOSX__ | ||||
| 	auto panel = m_tmp_panel; | ||||
| #else | ||||
| 	auto panel = this; | ||||
| #endif | ||||
| 	PageShp page(new Page(panel, title, icon_idx)); | ||||
| 	page->SetScrollbars(1, 1, 1, 1); | ||||
| 	page->Hide(); | ||||
| 	m_hsizer->Add(page.get(), 1, wxEXPAND | wxLEFT, 5); | ||||
|  | @ -290,6 +245,18 @@ PageShp Tab::add_options_page(const wxString& title, const std::string& icon, bo | |||
| 	return page; | ||||
| } | ||||
| 
 | ||||
| void Tab::OnActivate() | ||||
| { | ||||
| #ifdef __WXOSX__	 | ||||
| 	wxWindowUpdateLocker noUpdates(this); | ||||
| 
 | ||||
| 	auto size = GetSizer()->GetSize(); | ||||
| 	m_tmp_panel->GetSizer()->SetMinSize(size.x + m_size_move, size.y); | ||||
| 	Fit(); | ||||
| 	m_size_move *= -1; | ||||
| #endif // __WXOSX__
 | ||||
| } | ||||
| 
 | ||||
| void Tab::update_labels_colour() | ||||
| { | ||||
| 	Freeze(); | ||||
|  | @ -1248,6 +1215,7 @@ void TabPrint::OnActivate() | |||
| { | ||||
| 	m_recommended_thin_wall_thickness_description_line->SetText( | ||||
| 		from_u8(PresetHints::recommended_thin_wall_thickness(*m_preset_bundle))); | ||||
| 	Tab::OnActivate(); | ||||
| } | ||||
| 
 | ||||
| void TabFilament::build() | ||||
|  | @ -1405,6 +1373,7 @@ void TabFilament::update() | |||
| void TabFilament::OnActivate() | ||||
| { | ||||
| 	m_volumetric_speed_description_line->SetText(from_u8(PresetHints::maximum_volumetric_flow_description(*m_preset_bundle))); | ||||
| 	Tab::OnActivate(); | ||||
| } | ||||
| 
 | ||||
| wxSizer* Tab::description_line_widget(wxWindow* parent, ogStaticText* *StaticText) | ||||
|  | @ -2619,14 +2588,24 @@ void SavePresetWindow::accept() | |||
| 	if (!m_chosen_name.empty()) { | ||||
| 		const char* unusable_symbols = "<>:/\\|?*\""; | ||||
| 		bool is_unusable_symbol = false; | ||||
| 		bool is_unusable_postfix = false; | ||||
| 		const std::string unusable_postfix = "(modified)"; | ||||
| 		for (size_t i = 0; i < std::strlen(unusable_symbols); i++){ | ||||
| 			if (m_chosen_name.find_first_of(unusable_symbols[i]) != std::string::npos){ | ||||
| 				is_unusable_symbol = true; | ||||
| 				break; | ||||
| 			} | ||||
| 		} | ||||
| 		if (m_chosen_name.find(unusable_postfix) != std::string::npos) | ||||
| 			is_unusable_postfix = true; | ||||
| 
 | ||||
| 		if (is_unusable_symbol) { | ||||
| 			show_error(this, _(L("The supplied name is not valid; the following characters are not allowed:"))+" <>:/\\|?*\""); | ||||
| 			show_error(this,_(L("The supplied name is not valid;")) + "\n" + | ||||
| 							_(L("the following characters are not allowed:")) + " <>:/\\|?*\""); | ||||
| 		} | ||||
| 		else if (is_unusable_postfix){ | ||||
| 			show_error(this,	_(L("The supplied name is not valid;")) + "\n" +  | ||||
| 								_(L("the following postfix are not allowed:")) + "\n\t" + unusable_postfix); | ||||
| 		} | ||||
| 		else if (m_chosen_name.compare("- default -") == 0) { | ||||
| 			show_error(this, _(L("The supplied name is not available."))); | ||||
|  |  | |||
|  | @ -102,6 +102,10 @@ using PageShp = std::shared_ptr<Page>; | |||
| class Tab: public wxPanel | ||||
| { | ||||
| 	wxNotebook*			m_parent; | ||||
| #ifdef __WXOSX__ | ||||
| 	wxPanel*			m_tmp_panel; | ||||
| 	int					m_size_move = -1; | ||||
| #endif // __WXOSX__
 | ||||
| protected: | ||||
| 	std::string			m_name; | ||||
| 	const wxString		m_title; | ||||
|  | @ -118,7 +122,6 @@ protected: | |||
| 	wxButton*			m_undo_btn; | ||||
| 	wxButton*			m_undo_to_sys_btn; | ||||
| 	wxButton*			m_question_btn; | ||||
| 
 | ||||
| 	wxComboCtrl*		m_cc_presets_choice; | ||||
| 	wxDataViewTreeCtrl*	m_presetctrl; | ||||
| 	wxImageList*		m_preset_icons; | ||||
|  | @ -198,7 +201,7 @@ public: | |||
| 	Tab() {} | ||||
| 	Tab(wxNotebook* parent, const wxString& title, const char* name, bool no_controller) :  | ||||
| 		m_parent(parent), m_title(title), m_name(name), m_no_controller(no_controller) { | ||||
| 		Create(parent, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxBK_LEFT | wxTAB_TRAVERSAL); | ||||
| 		Create(parent, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxBK_LEFT | wxTAB_TRAVERSAL, name); | ||||
| 		get_tabs_list().push_back(this); | ||||
| 	} | ||||
| 	~Tab(){ | ||||
|  | @ -242,7 +245,7 @@ public: | |||
| 
 | ||||
| 	PageShp		add_options_page(const wxString& title, const std::string& icon, bool is_extruder_pages = false); | ||||
| 
 | ||||
| 	virtual void	OnActivate(){} | ||||
| 	virtual void	OnActivate(); | ||||
| 	virtual void	on_preset_loaded(){} | ||||
| 	virtual void	build() = 0; | ||||
| 	virtual void	update() = 0; | ||||
|  |  | |||
							
								
								
									
										402
									
								
								xs/src/slic3r/Utils/FixModelByWin10.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										402
									
								
								xs/src/slic3r/Utils/FixModelByWin10.cpp
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,402 @@ | |||
| #ifdef HAS_WIN10SDK | ||||
| 
 | ||||
| #ifndef NOMINMAX | ||||
| # define NOMINMAX | ||||
| #endif | ||||
| 
 | ||||
| #include "FixModelByWin10.hpp" | ||||
| 
 | ||||
| #include <atomic> | ||||
| #include <chrono> | ||||
| #include <cstdint> | ||||
| #include <condition_variable> | ||||
| #include <exception> | ||||
| #include <string> | ||||
| #include <thread> | ||||
| 
 | ||||
| #include <boost/filesystem.hpp> | ||||
| #include <boost/nowide/convert.hpp> | ||||
| #include <boost/nowide/cstdio.hpp> | ||||
| 
 | ||||
| #include <roapi.h> | ||||
| // for ComPtr
 | ||||
| #include <wrl/client.h> | ||||
| // from C:/Program Files (x86)/Windows Kits/10/Include/10.0.17134.0/
 | ||||
| #include <winrt/robuffer.h> | ||||
| #include <winrt/windows.storage.provider.h> | ||||
| #include <winrt/windows.graphics.printing3d.h> | ||||
| 
 | ||||
| #include "libslic3r/Model.hpp" | ||||
| #include "libslic3r/Print.hpp" | ||||
| #include "libslic3r/Format/3mf.hpp" | ||||
| #include "../GUI/GUI.hpp" | ||||
| #include "../GUI/PresetBundle.hpp" | ||||
| 
 | ||||
| #include <wx/msgdlg.h> | ||||
| #include <wx/progdlg.h> | ||||
| 
 | ||||
| extern "C"{ | ||||
| 	// from rapi.h
 | ||||
| 	typedef HRESULT (__stdcall* FunctionRoInitialize)(int); | ||||
| 	typedef HRESULT (__stdcall* FunctionRoUninitialize)(); | ||||
| 	typedef HRESULT	(__stdcall* FunctionRoActivateInstance)(HSTRING activatableClassId, IInspectable **instance); | ||||
| 	typedef HRESULT (__stdcall* FunctionRoGetActivationFactory)(HSTRING activatableClassId, REFIID iid, void **factory); | ||||
| 	// from winstring.h
 | ||||
| 	typedef HRESULT	(__stdcall* FunctionWindowsCreateString)(LPCWSTR sourceString, UINT32  length, HSTRING *string); | ||||
| 	typedef HRESULT	(__stdcall* FunctionWindowsDelteString)(HSTRING string); | ||||
| } | ||||
| 
 | ||||
| namespace Slic3r { | ||||
| 
 | ||||
| HMODULE							s_hRuntimeObjectLibrary  = nullptr; | ||||
| FunctionRoInitialize			s_RoInitialize			 = nullptr; | ||||
| FunctionRoUninitialize			s_RoUninitialize		 = nullptr; | ||||
| FunctionRoActivateInstance		s_RoActivateInstance     = nullptr; | ||||
| FunctionRoGetActivationFactory	s_RoGetActivationFactory = nullptr; | ||||
| FunctionWindowsCreateString		s_WindowsCreateString    = nullptr; | ||||
| FunctionWindowsDelteString		s_WindowsDeleteString    = nullptr; | ||||
| 
 | ||||
| bool winrt_load_runtime_object_library() | ||||
| { | ||||
| 	if (s_hRuntimeObjectLibrary == nullptr) | ||||
| 		s_hRuntimeObjectLibrary = LoadLibrary(L"ComBase.dll"); | ||||
| 	if (s_hRuntimeObjectLibrary != nullptr) { | ||||
| 		s_RoInitialize			 = (FunctionRoInitialize)			GetProcAddress(s_hRuntimeObjectLibrary, "RoInitialize"); | ||||
| 		s_RoUninitialize		 = (FunctionRoUninitialize)			GetProcAddress(s_hRuntimeObjectLibrary, "RoUninitialize"); | ||||
| 		s_RoActivateInstance	 = (FunctionRoActivateInstance)		GetProcAddress(s_hRuntimeObjectLibrary, "RoActivateInstance"); | ||||
| 		s_RoGetActivationFactory = (FunctionRoGetActivationFactory)	GetProcAddress(s_hRuntimeObjectLibrary, "RoGetActivationFactory"); | ||||
| 		s_WindowsCreateString	 = (FunctionWindowsCreateString)	GetProcAddress(s_hRuntimeObjectLibrary, "WindowsCreateString"); | ||||
| 		s_WindowsDeleteString	 = (FunctionWindowsDelteString)		GetProcAddress(s_hRuntimeObjectLibrary, "WindowsDeleteString"); | ||||
| 	} | ||||
| 	return s_RoInitialize && s_RoUninitialize && s_RoActivateInstance && s_WindowsCreateString && s_WindowsDeleteString; | ||||
| } | ||||
| 
 | ||||
| static HRESULT winrt_activate_instance(const std::wstring &class_name, IInspectable **pinst) | ||||
| { | ||||
| 	HSTRING hClassName; | ||||
| 	HRESULT hr = (*s_WindowsCreateString)(class_name.c_str(), class_name.size(), &hClassName); | ||||
| 	if (S_OK != hr)  | ||||
| 		return hr; | ||||
| 	hr = (*s_RoActivateInstance)(hClassName, pinst); | ||||
| 	(*s_WindowsDeleteString)(hClassName); | ||||
| 	return hr; | ||||
| } | ||||
| 
 | ||||
| template<typename TYPE> | ||||
| static HRESULT winrt_activate_instance(const std::wstring &class_name, TYPE **pinst) | ||||
| { | ||||
| 	IInspectable *pinspectable = nullptr; | ||||
| 	HRESULT hr = winrt_activate_instance(class_name, &pinspectable); | ||||
| 	if (S_OK != hr) | ||||
| 		return hr; | ||||
| 	hr = pinspectable->QueryInterface(__uuidof(TYPE), (void**)pinst); | ||||
| 	pinspectable->Release(); | ||||
| 	return hr; | ||||
| } | ||||
| 
 | ||||
| static HRESULT winrt_get_activation_factory(const std::wstring &class_name, REFIID iid, void **pinst) | ||||
| { | ||||
| 	HSTRING hClassName; | ||||
| 	HRESULT hr = (*s_WindowsCreateString)(class_name.c_str(), class_name.size(), &hClassName); | ||||
| 	if (S_OK != hr) | ||||
| 		return hr; | ||||
| 	hr = (*s_RoGetActivationFactory)(hClassName, iid, pinst); | ||||
| 	(*s_WindowsDeleteString)(hClassName); | ||||
| 	return hr; | ||||
| } | ||||
| 
 | ||||
| template<typename TYPE> | ||||
| static HRESULT winrt_get_activation_factory(const std::wstring &class_name, TYPE **pinst) | ||||
| { | ||||
| 	return winrt_get_activation_factory(class_name, __uuidof(TYPE), reinterpret_cast<void**>(pinst)); | ||||
| } | ||||
| 
 | ||||
| // To be called often to test whether to cancel the operation.
 | ||||
| typedef std::function<void ()> ThrowOnCancelFn; | ||||
| 
 | ||||
| template<typename T> | ||||
| static AsyncStatus winrt_async_await(const Microsoft::WRL::ComPtr<T> &asyncAction, ThrowOnCancelFn throw_on_cancel, int blocking_tick_ms = 100) | ||||
| { | ||||
| 	Microsoft::WRL::ComPtr<ABI::Windows::Foundation::IAsyncInfo> asyncInfo; | ||||
| 	asyncAction.As(&asyncInfo); | ||||
| 	AsyncStatus status; | ||||
| 	// Ugly blocking loop until the RepairAsync call finishes.
 | ||||
| //FIXME replace with a callback.
 | ||||
| // https://social.msdn.microsoft.com/Forums/en-US/a5038fb4-b7b7-4504-969d-c102faa389fb/trying-to-block-an-async-operation-and-wait-for-a-particular-time?forum=vclanguage
 | ||||
| 	for (;;) { | ||||
| 		asyncInfo->get_Status(&status); | ||||
| 		if (status != AsyncStatus::Started) | ||||
| 			return status; | ||||
| 		throw_on_cancel(); | ||||
| 		::Sleep(blocking_tick_ms); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| static HRESULT winrt_open_file_stream( | ||||
| 	const std::wstring									 &path, | ||||
| 	ABI::Windows::Storage::FileAccessMode				  mode, | ||||
| 	ABI::Windows::Storage::Streams::IRandomAccessStream **fileStream, | ||||
| 	ThrowOnCancelFn										  throw_on_cancel) | ||||
| { | ||||
| 	// Get the file factory.
 | ||||
| 	Microsoft::WRL::ComPtr<ABI::Windows::Storage::IStorageFileStatics> fileFactory; | ||||
| 	HRESULT hr = winrt_get_activation_factory(L"Windows.Storage.StorageFile", fileFactory.GetAddressOf()); | ||||
| 	if (FAILED(hr)) return hr; | ||||
| 
 | ||||
| 	// Open the file asynchronously.
 | ||||
| 	HSTRING hstr_path; | ||||
| 	hr = (*s_WindowsCreateString)(path.c_str(), path.size(), &hstr_path); | ||||
| 	if (FAILED(hr)) return hr; | ||||
| 	Microsoft::WRL::ComPtr<ABI::Windows::Foundation::IAsyncOperation<ABI::Windows::Storage::StorageFile*>> fileOpenAsync; | ||||
| 	hr = fileFactory->GetFileFromPathAsync(hstr_path, fileOpenAsync.GetAddressOf()); | ||||
| 	if (FAILED(hr)) return hr; | ||||
| 	(*s_WindowsDeleteString)(hstr_path); | ||||
| 
 | ||||
| 	// Wait until the file gets open, get the actual file.
 | ||||
| 	AsyncStatus status = winrt_async_await(fileOpenAsync, throw_on_cancel); | ||||
| 	Microsoft::WRL::ComPtr<ABI::Windows::Storage::IStorageFile> storageFile; | ||||
| 	if (status == AsyncStatus::Completed) { | ||||
| 		hr = fileOpenAsync->GetResults(storageFile.GetAddressOf()); | ||||
| 	} else { | ||||
| 		Microsoft::WRL::ComPtr<ABI::Windows::Foundation::IAsyncInfo> asyncInfo; | ||||
| 		hr = fileOpenAsync.As(&asyncInfo); | ||||
| 		if (FAILED(hr)) return hr; | ||||
| 		HRESULT err; | ||||
| 		hr = asyncInfo->get_ErrorCode(&err); | ||||
| 		return FAILED(hr) ? hr : err; | ||||
| 	} | ||||
| 
 | ||||
| 	Microsoft::WRL::ComPtr<ABI::Windows::Foundation::IAsyncOperation<ABI::Windows::Storage::Streams::IRandomAccessStream*>> fileStreamAsync; | ||||
| 	hr = storageFile->OpenAsync(mode, fileStreamAsync.GetAddressOf()); | ||||
| 	if (FAILED(hr)) return hr; | ||||
| 
 | ||||
| 	status = winrt_async_await(fileStreamAsync, throw_on_cancel); | ||||
| 	if (status == AsyncStatus::Completed) { | ||||
| 		hr = fileStreamAsync->GetResults(fileStream); | ||||
| 	} else { | ||||
| 		Microsoft::WRL::ComPtr<ABI::Windows::Foundation::IAsyncInfo> asyncInfo; | ||||
| 		hr = fileStreamAsync.As(&asyncInfo); | ||||
| 		if (FAILED(hr)) return hr; | ||||
| 		HRESULT err; | ||||
| 		hr = asyncInfo->get_ErrorCode(&err); | ||||
| 		if (!FAILED(hr)) | ||||
| 			hr = err; | ||||
| 	} | ||||
| 	return hr; | ||||
| } | ||||
| 
 | ||||
| bool is_windows10() | ||||
| { | ||||
| 	HKEY hKey; | ||||
| 	LONG lRes = RegOpenKeyExW(HKEY_LOCAL_MACHINE, L"SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion", 0, KEY_READ, &hKey); | ||||
| 	if (lRes == ERROR_SUCCESS) { | ||||
| 		WCHAR szBuffer[512]; | ||||
| 		DWORD dwBufferSize = sizeof(szBuffer); | ||||
| 		lRes = RegQueryValueExW(hKey, L"ProductName", 0, nullptr, (LPBYTE)szBuffer, &dwBufferSize); | ||||
| 		if (lRes == ERROR_SUCCESS) | ||||
| 			return wcsncmp(szBuffer, L"Windows 10", 10) == 0; | ||||
| 		RegCloseKey(hKey); | ||||
| 	} | ||||
| 	return false; | ||||
| } | ||||
| 
 | ||||
| // Progress function, to be called regularly to update the progress.
 | ||||
| typedef std::function<void (const char * /* message */, unsigned /* progress */)> ProgressFn; | ||||
| 
 | ||||
| void fix_model_by_win10_sdk(const std::string &path_src, const std::string &path_dst, ProgressFn on_progress, ThrowOnCancelFn throw_on_cancel) | ||||
| { | ||||
| 	if (! is_windows10()) | ||||
| 		throw std::runtime_error("fix_model_by_win10_sdk called on non Windows 10 system"); | ||||
| 
 | ||||
| 	if (! winrt_load_runtime_object_library()) | ||||
| 		throw std::runtime_error("Failed to initialize the WinRT library."); | ||||
| 
 | ||||
| 	HRESULT hr = (*s_RoInitialize)(RO_INIT_MULTITHREADED); | ||||
| 	{ | ||||
| 		on_progress(L("Exporting the source model"), 20); | ||||
| 
 | ||||
| 		Microsoft::WRL::ComPtr<ABI::Windows::Storage::Streams::IRandomAccessStream>       fileStream; | ||||
| 		hr = winrt_open_file_stream(boost::nowide::widen(path_src), ABI::Windows::Storage::FileAccessMode::FileAccessMode_Read, fileStream.GetAddressOf(), throw_on_cancel); | ||||
| 
 | ||||
| 		Microsoft::WRL::ComPtr<ABI::Windows::Graphics::Printing3D::IPrinting3D3MFPackage> printing3d3mfpackage; | ||||
| 		hr = winrt_activate_instance(L"Windows.Graphics.Printing3D.Printing3D3MFPackage", printing3d3mfpackage.GetAddressOf()); | ||||
| 
 | ||||
| 		Microsoft::WRL::ComPtr<ABI::Windows::Foundation::IAsyncOperation<ABI::Windows::Graphics::Printing3D::Printing3DModel*>> modelAsync; | ||||
| 		hr = printing3d3mfpackage->LoadModelFromPackageAsync(fileStream.Get(), modelAsync.GetAddressOf()); | ||||
| 
 | ||||
| 		AsyncStatus status = winrt_async_await(modelAsync, throw_on_cancel); | ||||
| 		Microsoft::WRL::ComPtr<ABI::Windows::Graphics::Printing3D::IPrinting3DModel>	  model; | ||||
| 		if (status == AsyncStatus::Completed) | ||||
| 			hr = modelAsync->GetResults(model.GetAddressOf()); | ||||
| 		else | ||||
| 			throw std::runtime_error(L("Failed loading the input model.")); | ||||
| 
 | ||||
| 		Microsoft::WRL::ComPtr<ABI::Windows::Foundation::Collections::IVector<ABI::Windows::Graphics::Printing3D::Printing3DMesh*>> meshes; | ||||
| 		hr = model->get_Meshes(meshes.GetAddressOf()); | ||||
| 		unsigned num_meshes = 0; | ||||
| 		hr = meshes->get_Size(&num_meshes); | ||||
| 		 | ||||
| 		on_progress(L("Repairing the model by the Netfabb service"), 40); | ||||
| 		 | ||||
| 		Microsoft::WRL::ComPtr<ABI::Windows::Foundation::IAsyncAction>					  repairAsync; | ||||
| 		hr = model->RepairAsync(repairAsync.GetAddressOf()); | ||||
| 		status = winrt_async_await(repairAsync, throw_on_cancel); | ||||
| 		if (status != AsyncStatus::Completed) | ||||
| 			throw std::runtime_error(L("Mesh repair failed.")); | ||||
| 		repairAsync->GetResults(); | ||||
| 
 | ||||
| 		on_progress(L("Loading the repaired model"), 60); | ||||
| 
 | ||||
| 		// Verify the number of meshes returned after the repair action.
 | ||||
| 		meshes.Reset(); | ||||
| 		hr = model->get_Meshes(meshes.GetAddressOf()); | ||||
| 		hr = meshes->get_Size(&num_meshes); | ||||
| 
 | ||||
| 		// Save model to this class' Printing3D3MFPackage.
 | ||||
| 		Microsoft::WRL::ComPtr<ABI::Windows::Foundation::IAsyncAction>					  saveToPackageAsync; | ||||
| 		hr = printing3d3mfpackage->SaveModelToPackageAsync(model.Get(), saveToPackageAsync.GetAddressOf()); | ||||
| 		status = winrt_async_await(saveToPackageAsync, throw_on_cancel); | ||||
| 		if (status != AsyncStatus::Completed) | ||||
| 			throw std::runtime_error(L("Saving mesh into the 3MF container failed.")); | ||||
| 		hr = saveToPackageAsync->GetResults(); | ||||
| 
 | ||||
| 		Microsoft::WRL::ComPtr<ABI::Windows::Foundation::IAsyncOperation<ABI::Windows::Storage::Streams::IRandomAccessStream*>> generatorStreamAsync; | ||||
| 		hr = printing3d3mfpackage->SaveAsync(generatorStreamAsync.GetAddressOf()); | ||||
| 		status = winrt_async_await(generatorStreamAsync, throw_on_cancel); | ||||
| 		if (status != AsyncStatus::Completed) | ||||
| 			throw std::runtime_error(L("Saving mesh into the 3MF container failed.")); | ||||
| 		Microsoft::WRL::ComPtr<ABI::Windows::Storage::Streams::IRandomAccessStream> generatorStream; | ||||
| 		hr = generatorStreamAsync->GetResults(generatorStream.GetAddressOf()); | ||||
| 
 | ||||
| 		// Go to the beginning of the stream.
 | ||||
| 		generatorStream->Seek(0); | ||||
| 		Microsoft::WRL::ComPtr<ABI::Windows::Storage::Streams::IInputStream> inputStream; | ||||
| 		hr = generatorStream.As(&inputStream); | ||||
| 
 | ||||
| 		// Get the buffer factory.
 | ||||
| 		Microsoft::WRL::ComPtr<ABI::Windows::Storage::Streams::IBufferFactory> bufferFactory; | ||||
| 		hr = winrt_get_activation_factory(L"Windows.Storage.Streams.Buffer", bufferFactory.GetAddressOf()); | ||||
| 
 | ||||
| 		// Open the destination file.
 | ||||
| 		FILE *fout = boost::nowide::fopen(path_dst.c_str(), "wb"); | ||||
| 
 | ||||
| 		Microsoft::WRL::ComPtr<ABI::Windows::Storage::Streams::IBuffer> buffer; | ||||
| 		byte														   *buffer_ptr; | ||||
| 		bufferFactory->Create(65536 * 2048, buffer.GetAddressOf()); | ||||
| 		{ | ||||
| 			Microsoft::WRL::ComPtr<Windows::Storage::Streams::IBufferByteAccess> bufferByteAccess; | ||||
| 			buffer.As(&bufferByteAccess); | ||||
| 			hr = bufferByteAccess->Buffer(&buffer_ptr); | ||||
| 		} | ||||
| 		uint32_t length; | ||||
| 		hr = buffer->get_Length(&length); | ||||
| 
 | ||||
| 		Microsoft::WRL::ComPtr<ABI::Windows::Foundation::IAsyncOperationWithProgress<ABI::Windows::Storage::Streams::IBuffer*, UINT32>> asyncRead; | ||||
| 		for (;;) { | ||||
| 			hr = inputStream->ReadAsync(buffer.Get(), 65536 * 2048, ABI::Windows::Storage::Streams::InputStreamOptions_ReadAhead, asyncRead.GetAddressOf()); | ||||
| 			status = winrt_async_await(asyncRead, throw_on_cancel); | ||||
| 			if (status != AsyncStatus::Completed) | ||||
| 				throw std::runtime_error(L("Saving mesh into the 3MF container failed.")); | ||||
| 			hr = buffer->get_Length(&length); | ||||
| 			if (length == 0) | ||||
| 				break; | ||||
| 			fwrite(buffer_ptr, length, 1, fout); | ||||
| 		} | ||||
| 		fclose(fout); | ||||
| 		// Here all the COM objects will be released through the ComPtr destructors.
 | ||||
| 	} | ||||
| 	(*s_RoUninitialize)(); | ||||
| } | ||||
| 
 | ||||
| class RepairCanceledException : public std::exception { | ||||
| public: | ||||
|    const char* what() const throw() { return "Model repair has been canceled"; } | ||||
| }; | ||||
| 
 | ||||
| void fix_model_by_win10_sdk_gui(const ModelObject &model_object, const Print &print, Model &result) | ||||
| { | ||||
| 	std::mutex 						mutex; | ||||
| 	std::condition_variable			condition; | ||||
| 	std::unique_lock<std::mutex>	lock(mutex); | ||||
| 	struct Progress { | ||||
| 		std::string 				message; | ||||
| 		int 						percent  = 0; | ||||
| 		bool						updated = false; | ||||
| 	} progress; | ||||
| 	std::atomic<bool>				canceled = false; | ||||
| 	std::atomic<bool>				finished = false; | ||||
| 
 | ||||
| 	// Open a progress dialog.
 | ||||
| 	wxProgressDialog progress_dialog( | ||||
| 		_(L("Model fixing")), | ||||
| 		_(L("Exporting model...")), | ||||
| 		100, nullptr, wxPD_AUTO_HIDE | wxPD_APP_MODAL | wxPD_CAN_ABORT); | ||||
| 	// Executing the calculation in a background thread, so that the COM context could be created with its own threading model.
 | ||||
| 	// (It seems like wxWidgets initialize the COM contex as single threaded and we need a multi-threaded context).
 | ||||
| 	bool success  = false; | ||||
| 	auto on_progress = [&mutex, &condition, &progress](const char *msg, unsigned prcnt) { | ||||
|         std::lock_guard<std::mutex> lk(mutex); | ||||
| 		progress.message = msg; | ||||
| 		progress.percent = prcnt; | ||||
| 		progress.updated = true; | ||||
| 	    condition.notify_all(); | ||||
| 	}; | ||||
| 	auto worker_thread = boost::thread([&model_object, &print, &result, on_progress, &success, &canceled, &finished]() { | ||||
| 		try { | ||||
| 			on_progress(L("Exporting the source model"), 0); | ||||
| 			boost::filesystem::path path_src = boost::filesystem::temp_directory_path() / boost::filesystem::unique_path(); | ||||
| 			path_src += ".3mf"; | ||||
| 			Model model; | ||||
| 			model.add_object(model_object); | ||||
| 			if (! Slic3r::store_3mf(path_src.string().c_str(), &model, const_cast<Print*>(&print), false)) { | ||||
| 				boost::filesystem::remove(path_src); | ||||
| 				throw std::runtime_error(L("Export of a temporary 3mf file failed")); | ||||
| 			} | ||||
| 			model.clear_objects();  | ||||
| 			model.clear_materials(); | ||||
| 			boost::filesystem::path path_dst = boost::filesystem::temp_directory_path() / boost::filesystem::unique_path(); | ||||
| 			path_dst += ".3mf"; | ||||
| 			fix_model_by_win10_sdk(path_src.string().c_str(), path_dst.string(), on_progress,  | ||||
| 				[&canceled]() { if (canceled) throw RepairCanceledException(); }); | ||||
| 			boost::filesystem::remove(path_src); | ||||
| 			PresetBundle bundle; | ||||
| 			on_progress(L("Loading the repaired model"), 80); | ||||
| 		    bool loaded = Slic3r::load_3mf(path_dst.string().c_str(), &bundle, &result); | ||||
| 			boost::filesystem::remove(path_dst); | ||||
| 			if (! loaded) | ||||
|  				throw std::runtime_error(L("Import of the repaired 3mf file failed")); | ||||
| 			success  = true; | ||||
| 			finished = true; | ||||
| 			on_progress(L("Model repair finished"), 100); | ||||
| 		} catch (RepairCanceledException &ex) { | ||||
| 			canceled = true; | ||||
| 			finished = true; | ||||
| 			on_progress(L("Model repair canceled"), 100); | ||||
| 		} catch (std::exception &ex) { | ||||
| 			success = false; | ||||
| 			finished = true; | ||||
| 			on_progress(ex.what(), 100); | ||||
| 		} | ||||
| 	}); | ||||
|     while (! finished) { | ||||
| 		condition.wait_for(lock, std::chrono::milliseconds(500), [&progress]{ return progress.updated; }); | ||||
| 		if (! progress_dialog.Update(progress.percent, _(progress.message))) | ||||
| 			canceled = true; | ||||
| 		progress.updated = false; | ||||
|     } | ||||
| 
 | ||||
| 	if (canceled) { | ||||
| 		// Nothing to show.
 | ||||
| 	} else if (success) { | ||||
| 		wxMessageDialog dlg(nullptr, _(L("Model repaired successfully")), _(L("Model Repair by the Netfabb service")), wxICON_INFORMATION | wxOK_DEFAULT); | ||||
| 		dlg.ShowModal(); | ||||
| 	} else { | ||||
| 		wxMessageDialog dlg(nullptr, _(L("Model repair failed: \n")) + _(progress.message), _(L("Model Repair by the Netfabb service")), wxICON_ERROR | wxOK_DEFAULT); | ||||
| 		dlg.ShowModal(); | ||||
| 	} | ||||
| 	worker_thread.join(); | ||||
| } | ||||
| 
 | ||||
| } // namespace Slic3r
 | ||||
| 
 | ||||
| #endif /* HAS_WIN10SDK */ | ||||
							
								
								
									
										26
									
								
								xs/src/slic3r/Utils/FixModelByWin10.hpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								xs/src/slic3r/Utils/FixModelByWin10.hpp
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,26 @@ | |||
| #ifndef slic3r_GUI_Utils_FixModelByWin10_hpp_ | ||||
| #define slic3r_GUI_Utils_FixModelByWin10_hpp_ | ||||
| 
 | ||||
| #include <string> | ||||
| 
 | ||||
| namespace Slic3r { | ||||
| 
 | ||||
| class Model; | ||||
| class ModelObject; | ||||
| class Print; | ||||
| 
 | ||||
| #ifdef HAS_WIN10SDK | ||||
| 
 | ||||
| extern bool is_windows10(); | ||||
| extern void fix_model_by_win10_sdk_gui(const ModelObject &model_object, const Print &print, Model &result); | ||||
| 
 | ||||
| #else /* HAS_WIN10SDK */ | ||||
| 
 | ||||
| inline bool is_windows10() { return false; } | ||||
| inline void fix_model_by_win10_sdk_gui(const ModelObject &, const Print &, Model &) {} | ||||
| 
 | ||||
| #endif /* HAS_WIN10SDK */ | ||||
| 
 | ||||
| } // namespace Slic3r
 | ||||
| 
 | ||||
| #endif /* slic3r_GUI_Utils_FixModelByWin10_hpp_ */ | ||||
|  | @ -3,13 +3,16 @@ | |||
| #include <cstdlib> | ||||
| #include <functional> | ||||
| #include <thread> | ||||
| #include <tuple> | ||||
| #include <deque> | ||||
| #include <boost/filesystem/fstream.hpp> | ||||
| #include <boost/format.hpp> | ||||
| 
 | ||||
| #include <curl/curl.h> | ||||
| 
 | ||||
| #include "../../libslic3r/libslic3r.h" | ||||
| 
 | ||||
| namespace fs = boost::filesystem; | ||||
| 
 | ||||
| 
 | ||||
| namespace Slic3r { | ||||
| 
 | ||||
|  | @ -34,7 +37,11 @@ struct Http::priv | |||
| 	::curl_httppost *form; | ||||
| 	::curl_httppost *form_end; | ||||
| 	::curl_slist *headerlist; | ||||
| 	// Used for reading the body
 | ||||
| 	std::string buffer; | ||||
| 	// Used for storing file streams added as multipart form parts
 | ||||
| 	// Using a deque here because unlike vector it doesn't ivalidate pointers on insertion
 | ||||
| 	std::deque<fs::ifstream> form_files; | ||||
| 	size_t limit; | ||||
| 	bool cancel; | ||||
| 
 | ||||
|  | @ -50,6 +57,10 @@ struct Http::priv | |||
| 	static size_t writecb(void *data, size_t size, size_t nmemb, void *userp); | ||||
| 	static int xfercb(void *userp, curl_off_t dltotal, curl_off_t dlnow, curl_off_t ultotal, curl_off_t ulnow); | ||||
| 	static int xfercb_legacy(void *userp, double dltotal, double dlnow, double ultotal, double ulnow); | ||||
| 	static size_t form_file_read_cb(char *buffer, size_t size, size_t nitems, void *userp); | ||||
| 
 | ||||
| 	void form_add_file(const char *name, const fs::path &path, const char* filename); | ||||
| 
 | ||||
| 	std::string curl_error(CURLcode curlcode); | ||||
| 	std::string body_size_error(); | ||||
| 	void http_perform(); | ||||
|  | @ -60,6 +71,7 @@ Http::priv::priv(const std::string &url) : | |||
| 	form(nullptr), | ||||
| 	form_end(nullptr), | ||||
| 	headerlist(nullptr), | ||||
| 	limit(0), | ||||
| 	cancel(false) | ||||
| { | ||||
| 	if (curl == nullptr) { | ||||
|  | @ -135,6 +147,46 @@ int Http::priv::xfercb_legacy(void *userp, double dltotal, double dlnow, double | |||
| 	return xfercb(userp, dltotal, dlnow, ultotal, ulnow); | ||||
| } | ||||
| 
 | ||||
| size_t Http::priv::form_file_read_cb(char *buffer, size_t size, size_t nitems, void *userp) | ||||
| { | ||||
| 	auto stream = reinterpret_cast<fs::ifstream*>(userp); | ||||
| 
 | ||||
| 	try { | ||||
| 		stream->read(buffer, size * nitems); | ||||
| 	} catch (...) { | ||||
| 		return CURL_READFUNC_ABORT; | ||||
| 	} | ||||
| 
 | ||||
| 	return stream->gcount(); | ||||
| } | ||||
| 
 | ||||
| void Http::priv::form_add_file(const char *name, const fs::path &path, const char* filename) | ||||
| { | ||||
| 	// We can't use CURLFORM_FILECONTENT, because curl doesn't support Unicode filenames on Windows
 | ||||
| 	// and so we use CURLFORM_STREAM with boost ifstream to read the file.
 | ||||
| 
 | ||||
| 	if (filename == nullptr) { | ||||
| 		filename = path.string().c_str(); | ||||
| 	} | ||||
| 
 | ||||
| 	form_files.emplace_back(path, std::ios::in | std::ios::binary); | ||||
| 	auto &stream = form_files.back(); | ||||
| 	stream.seekg(0, std::ios::end); | ||||
| 	size_t size = stream.tellg(); | ||||
| 	stream.seekg(0); | ||||
| 
 | ||||
| 	if (filename != nullptr) { | ||||
| 		::curl_formadd(&form, &form_end, | ||||
| 			CURLFORM_COPYNAME, name, | ||||
| 			CURLFORM_FILENAME, filename, | ||||
| 			CURLFORM_CONTENTTYPE, "application/octet-stream", | ||||
| 			CURLFORM_STREAM, static_cast<void*>(&stream), | ||||
| 			CURLFORM_CONTENTSLENGTH, static_cast<long>(size), | ||||
| 			CURLFORM_END | ||||
| 		); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| std::string Http::priv::curl_error(CURLcode curlcode) | ||||
| { | ||||
| 	return (boost::format("%1% (%2%)") | ||||
|  | @ -150,10 +202,10 @@ std::string Http::priv::body_size_error() | |||
| 
 | ||||
| void Http::priv::http_perform() | ||||
| { | ||||
| 	::curl_easy_setopt(curl, CURLOPT_FAILONERROR, 1L); | ||||
| 	::curl_easy_setopt(curl, CURLOPT_FOLLOWLOCATION, 1L); | ||||
| 	::curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, writecb); | ||||
| 	::curl_easy_setopt(curl, CURLOPT_WRITEDATA, static_cast<void*>(this)); | ||||
| 	::curl_easy_setopt(curl, CURLOPT_READFUNCTION, form_file_read_cb); | ||||
| 
 | ||||
| 	::curl_easy_setopt(curl, CURLOPT_NOPROGRESS, 0L); | ||||
| #if LIBCURL_VERSION_MAJOR >= 7 && LIBCURL_VERSION_MINOR >= 32 | ||||
|  | @ -178,23 +230,32 @@ void Http::priv::http_perform() | |||
| 	} | ||||
| 
 | ||||
| 	CURLcode res = ::curl_easy_perform(curl); | ||||
| 	long http_status = 0; | ||||
| 	::curl_easy_getinfo(curl, CURLINFO_RESPONSE_CODE, &http_status); | ||||
| 
 | ||||
| 	if (res != CURLE_OK) { | ||||
| 		if (res == CURLE_ABORTED_BY_CALLBACK) { | ||||
| 			Progress dummyprogress(0, 0, 0, 0); | ||||
| 			bool cancel = true; | ||||
| 			if (progressfn) { progressfn(dummyprogress, cancel); } | ||||
| 			if (cancel) { | ||||
| 				// The abort comes from the request being cancelled programatically
 | ||||
| 				Progress dummyprogress(0, 0, 0, 0); | ||||
| 				bool cancel = true; | ||||
| 				if (progressfn) { progressfn(dummyprogress, cancel); } | ||||
| 			} else { | ||||
| 				// The abort comes from the CURLOPT_READFUNCTION callback, which means reading file failed
 | ||||
| 				if (errorfn) { errorfn(std::move(buffer), "Error reading file for file upload", 0); } | ||||
| 			} | ||||
| 		} | ||||
| 		else if (res == CURLE_WRITE_ERROR) { | ||||
| 			if (errorfn) { errorfn(std::move(buffer), std::move(body_size_error()), http_status); } | ||||
| 			if (errorfn) { errorfn(std::move(buffer), body_size_error(), 0); } | ||||
| 		} else { | ||||
| 			if (errorfn) { errorfn(std::move(buffer), std::move(curl_error(res)), http_status); } | ||||
| 			if (errorfn) { errorfn(std::move(buffer), curl_error(res), 0); } | ||||
| 		}; | ||||
| 	} else { | ||||
| 		if (completefn) { | ||||
| 			completefn(std::move(buffer), http_status); | ||||
| 		long http_status = 0; | ||||
| 		::curl_easy_getinfo(curl, CURLINFO_RESPONSE_CODE, &http_status); | ||||
| 		 | ||||
| 		if (http_status >= 400) { | ||||
| 			if (errorfn) { errorfn(std::move(buffer), std::string(), http_status); } | ||||
| 		} else { | ||||
| 			if (completefn) { completefn(std::move(buffer), http_status); } | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | @ -265,17 +326,15 @@ Http& Http::form_add(const std::string &name, const std::string &contents) | |||
| 	return *this; | ||||
| } | ||||
| 
 | ||||
| Http& Http::form_add_file(const std::string &name, const std::string &filename) | ||||
| Http& Http::form_add_file(const std::string &name, const fs::path &path) | ||||
| { | ||||
| 	if (p) { | ||||
| 		::curl_formadd(&p->form, &p->form_end, | ||||
| 			CURLFORM_COPYNAME, name.c_str(), | ||||
| 			CURLFORM_FILE, filename.c_str(), | ||||
| 			CURLFORM_CONTENTTYPE, "application/octet-stream", | ||||
| 			CURLFORM_END | ||||
| 		); | ||||
| 	} | ||||
| 	if (p) { p->form_add_file(name.c_str(), path.c_str(), nullptr); } | ||||
| 	return *this; | ||||
| } | ||||
| 
 | ||||
| Http& Http::form_add_file(const std::string &name, const fs::path &path, const std::string &filename) | ||||
| { | ||||
| 	if (p) { p->form_add_file(name.c_str(), path.c_str(), filename.c_str()); } | ||||
| 	return *this; | ||||
| } | ||||
| 
 | ||||
|  |  | |||
|  | @ -4,6 +4,7 @@ | |||
| #include <memory> | ||||
| #include <string> | ||||
| #include <functional> | ||||
| #include <boost/filesystem/path.hpp> | ||||
| 
 | ||||
| 
 | ||||
| namespace Slic3r { | ||||
|  | @ -16,11 +17,11 @@ private: | |||
| public: | ||||
| 	struct Progress | ||||
| 	{ | ||||
| 		size_t dltotal; | ||||
| 		size_t dlnow; | ||||
| 		size_t ultotal; | ||||
| 		size_t ulnow; | ||||
| 		 | ||||
| 		size_t dltotal;   // Total bytes to download
 | ||||
| 		size_t dlnow;     // Bytes downloaded so far
 | ||||
| 		size_t ultotal;   // Total bytes to upload
 | ||||
| 		size_t ulnow;     // Bytes uploaded so far
 | ||||
| 
 | ||||
| 		Progress(size_t dltotal, size_t dlnow, size_t ultotal, size_t ulnow) : | ||||
| 			dltotal(dltotal), dlnow(dlnow), ultotal(ultotal), ulnow(ulnow) | ||||
| 		{} | ||||
|  | @ -28,11 +29,24 @@ public: | |||
| 
 | ||||
| 	typedef std::shared_ptr<Http> Ptr; | ||||
| 	typedef std::function<void(std::string /* body */, unsigned /* http_status */)> CompleteFn; | ||||
| 	 | ||||
| 	// A HTTP request may fail at various stages of completeness (URL parsing, DNS lookup, TCP connection, ...).
 | ||||
| 	// If the HTTP request could not be made or failed before completion, the `error` arg contains a description
 | ||||
| 	// of the error and `http_status` is zero.
 | ||||
| 	// If the HTTP request was completed but the response HTTP code is >= 400, `error` is empty and `http_status` contains the response code.
 | ||||
| 	// In either case there may or may not be a body.
 | ||||
| 	typedef std::function<void(std::string /* body */, std::string /* error */, unsigned /* http_status */)> ErrorFn; | ||||
| 
 | ||||
| 	// See the Progress struct above.
 | ||||
| 	// Writing true to the `cancel` reference cancels the request in progress.
 | ||||
| 	typedef std::function<void(Progress, bool& /* cancel */)> ProgressFn; | ||||
| 
 | ||||
| 	Http(Http &&other); | ||||
| 
 | ||||
| 	// Note: strings are expected to be UTF-8-encoded
 | ||||
| 
 | ||||
| 	// These are the primary constructors that create a HTTP object
 | ||||
| 	// for a GET and a POST request respectively.
 | ||||
| 	static Http get(std::string url); | ||||
| 	static Http post(std::string url); | ||||
| 	~Http(); | ||||
|  | @ -41,21 +55,43 @@ public: | |||
| 	Http& operator=(const Http &) = delete; | ||||
| 	Http& operator=(Http &&) = delete; | ||||
| 
 | ||||
| 	// Sets a maximum size of the data that can be received.
 | ||||
| 	// A value of zero sets the default limit, which is is 5MB.
 | ||||
| 	Http& size_limit(size_t sizeLimit); | ||||
| 	// Sets a HTTP header field.
 | ||||
| 	Http& header(std::string name, const std::string &value); | ||||
| 	// Removes a header field.
 | ||||
| 	Http& remove_header(std::string name); | ||||
| 	// Sets a CA certificate file for usage with HTTPS. This is only supported on some backends,
 | ||||
| 	// specifically, this is supported with OpenSSL and NOT supported with Windows and OS X native certificate store.
 | ||||
| 	// See also ca_file_supported().
 | ||||
| 	Http& ca_file(const std::string &filename); | ||||
| 	// Add a HTTP multipart form field
 | ||||
| 	Http& form_add(const std::string &name, const std::string &contents); | ||||
| 	Http& form_add_file(const std::string &name, const std::string &filename); | ||||
| 	// Add a HTTP multipart form file data contents, `name` is the name of the part
 | ||||
| 	Http& form_add_file(const std::string &name, const boost::filesystem::path &path); | ||||
| 	// Same as above except also override the file's filename with a custom one
 | ||||
| 	Http& form_add_file(const std::string &name, const boost::filesystem::path &path, const std::string &filename); | ||||
| 
 | ||||
| 	// Callback called on HTTP request complete
 | ||||
| 	Http& on_complete(CompleteFn fn); | ||||
| 	// Callback called on an error occuring at any stage of the requests: Url parsing, DNS lookup,
 | ||||
| 	// TCP connection, HTTP transfer, and finally also when the response indicates an error (status >= 400).
 | ||||
| 	// Therefore, a response body may or may not be present.
 | ||||
| 	Http& on_error(ErrorFn fn); | ||||
| 	// Callback called on data download/upload prorgess (called fairly frequently).
 | ||||
| 	// See the `Progress` structure for description of the data passed.
 | ||||
| 	// Writing a true-ish value into the cancel reference parameter cancels the request.
 | ||||
| 	Http& on_progress(ProgressFn fn); | ||||
| 
 | ||||
| 	// Starts performing the request in a background thread
 | ||||
| 	Ptr perform(); | ||||
| 	// Starts performing the request on the current thread
 | ||||
| 	void perform_sync(); | ||||
| 	// Cancels a request in progress
 | ||||
| 	void cancel(); | ||||
| 
 | ||||
| 	// Tells whether current backend supports seting up a CA file using ca_file()
 | ||||
| 	static bool ca_file_supported(); | ||||
| private: | ||||
| 	Http(const std::string &url); | ||||
|  |  | |||
|  | @ -1,20 +1,65 @@ | |||
| #include "OctoPrint.hpp" | ||||
| 
 | ||||
| #include <algorithm> | ||||
| #include <boost/filesystem/path.hpp> | ||||
| #include <boost/format.hpp> | ||||
| #include <boost/log/trivial.hpp> | ||||
| 
 | ||||
| #include <wx/frame.h> | ||||
| #include <wx/event.h> | ||||
| #include <wx/progdlg.h> | ||||
| #include <wx/sizer.h> | ||||
| #include <wx/stattext.h> | ||||
| #include <wx/textctrl.h> | ||||
| #include <wx/checkbox.h> | ||||
| 
 | ||||
| #include "libslic3r/PrintConfig.hpp" | ||||
| #include "slic3r/GUI/GUI.hpp" | ||||
| #include "slic3r/GUI/MsgDialog.hpp" | ||||
| #include "Http.hpp" | ||||
| 
 | ||||
| namespace fs = boost::filesystem; | ||||
| 
 | ||||
| 
 | ||||
| namespace Slic3r { | ||||
| 
 | ||||
| 
 | ||||
| struct SendDialog : public GUI::MsgDialog | ||||
| { | ||||
| 	wxTextCtrl *txt_filename; | ||||
| 	wxCheckBox *box_print; | ||||
| 
 | ||||
| 	SendDialog(const fs::path &path) : | ||||
| 		MsgDialog(nullptr, _(L("Send G-Code to printer")), _(L("Upload to OctoPrint with the following filename:")), wxID_NONE), | ||||
| 		txt_filename(new wxTextCtrl(this, wxID_ANY, path.filename().wstring())), | ||||
| 		box_print(new wxCheckBox(this, wxID_ANY, _(L("Start printing after upload")))) | ||||
| 	{ | ||||
| 		auto *label_dir_hint = new wxStaticText(this, wxID_ANY, _(L("Use forward slashes ( / ) as a directory separator if needed."))); | ||||
| 		label_dir_hint->Wrap(CONTENT_WIDTH); | ||||
| 
 | ||||
| 		content_sizer->Add(txt_filename, 0, wxEXPAND); | ||||
| 		content_sizer->Add(label_dir_hint); | ||||
| 		content_sizer->AddSpacer(VERT_SPACING); | ||||
| 		content_sizer->Add(box_print, 0, wxBOTTOM, 2*VERT_SPACING); | ||||
| 
 | ||||
| 		btn_sizer->Add(CreateStdDialogButtonSizer(wxOK | wxCANCEL)); | ||||
| 
 | ||||
| 		txt_filename->SetFocus(); | ||||
| 		wxString stem(path.stem().wstring()); | ||||
| 		txt_filename->SetSelection(0, stem.Length()); | ||||
| 
 | ||||
| 		Fit(); | ||||
| 	} | ||||
| 
 | ||||
| 	fs::path filename() const { | ||||
| 		return fs::path(txt_filename->GetValue().wx_str()); | ||||
| 	} | ||||
| 
 | ||||
| 	bool print() const { return box_print->GetValue(); } | ||||
| }; | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| OctoPrint::OctoPrint(DynamicPrintConfig *config) : | ||||
| 	host(config->opt_string("octoprint_host")), | ||||
| 	apikey(config->opt_string("octoprint_apikey")), | ||||
|  | @ -27,24 +72,39 @@ bool OctoPrint::test(wxString &msg) const | |||
| 	// it is ok to refer to `msg` from within the closure
 | ||||
| 
 | ||||
| 	bool res = true; | ||||
| 	auto url = make_url("api/version"); | ||||
| 
 | ||||
| 	BOOST_LOG_TRIVIAL(info) << boost::format("Octoprint: Get version at: %1%") % url; | ||||
| 
 | ||||
| 	auto url = std::move(make_url("api/version")); | ||||
| 	auto http = Http::get(std::move(url)); | ||||
| 	set_auth(http); | ||||
| 	http.on_error([&](std::string, std::string error, unsigned status) { | ||||
| 	http.on_error([&](std::string body, std::string error, unsigned status) { | ||||
| 			BOOST_LOG_TRIVIAL(error) << boost::format("Octoprint: Error getting version: %1%, HTTP %2%, body: `%3%`") % error % status % body; | ||||
| 			res = false; | ||||
| 			msg = format_error(error, status); | ||||
| 			msg = format_error(body, error, status); | ||||
| 		}) | ||||
| 		.on_complete([&](std::string body, unsigned) { | ||||
| 			BOOST_LOG_TRIVIAL(debug) << boost::format("Octoprint: Got version: %1%") % body; | ||||
| 		}) | ||||
| 		.perform_sync(); | ||||
| 
 | ||||
| 	return res; | ||||
| } | ||||
| 
 | ||||
| bool OctoPrint::send_gcode(const std::string &filename, bool print) const | ||||
| bool OctoPrint::send_gcode(const std::string &filename) const | ||||
| { | ||||
| 	enum { PROGRESS_RANGE = 1000 }; | ||||
| 
 | ||||
| 	const auto errortitle = _(L("Error while uploading to the OctoPrint server")); | ||||
| 	fs::path filepath(filename); | ||||
| 
 | ||||
| 	SendDialog send_dialog(filepath.filename()); | ||||
| 	if (send_dialog.ShowModal() != wxID_OK) { return false; } | ||||
| 
 | ||||
| 	const bool print = send_dialog.print(); | ||||
| 	const auto upload_filepath = send_dialog.filename(); | ||||
| 	const auto upload_filename = upload_filepath.filename(); | ||||
| 	const auto upload_parent_path = upload_filepath.parent_path(); | ||||
| 
 | ||||
| 	wxProgressDialog progress_dialog( | ||||
| 		_(L("OctoPrint upload")), | ||||
|  | @ -61,15 +121,27 @@ bool OctoPrint::send_gcode(const std::string &filename, bool print) const | |||
| 
 | ||||
| 	bool res = true; | ||||
| 
 | ||||
| 	auto http = Http::post(std::move(make_url("api/files/local"))); | ||||
| 	auto url = make_url("api/files/local"); | ||||
| 
 | ||||
| 	BOOST_LOG_TRIVIAL(info) << boost::format("Octoprint: Uploading file %1% at %2%, filename: %3%, path: %4%, print: %5%") | ||||
| 		% filepath.string() | ||||
| 		% url | ||||
| 		% upload_filename.string() | ||||
| 		% upload_parent_path.string() | ||||
| 		% print; | ||||
| 
 | ||||
| 	auto http = Http::post(std::move(url)); | ||||
| 	set_auth(http); | ||||
| 	http.form_add("print", print ? "true" : "false") | ||||
| 		.form_add_file("file", filename) | ||||
| 		.form_add("path", upload_parent_path.string()) | ||||
| 		.form_add_file("file", filename, upload_filename.string()) | ||||
| 		.on_complete([&](std::string body, unsigned status) { | ||||
| 			BOOST_LOG_TRIVIAL(debug) << boost::format("Octoprint: File uploaded: HTTP %1%: %2%") % status % body; | ||||
| 			progress_dialog.Update(PROGRESS_RANGE); | ||||
| 		}) | ||||
| 		.on_error([&](std::string body, std::string error, unsigned status) { | ||||
| 			auto errormsg = wxString::Format("%s: %s", errortitle, format_error(error, status)); | ||||
| 			BOOST_LOG_TRIVIAL(error) << boost::format("Octoprint: Error uploading file: %1%, HTTP %2%, body: `%3%`") % error % status % body; | ||||
| 			auto errormsg = wxString::Format("%s: %s", errortitle, format_error(body, error, status)); | ||||
| 			GUI::show_error(&progress_dialog, std::move(errormsg)); | ||||
| 			res = false; | ||||
| 		}) | ||||
|  | @ -102,24 +174,22 @@ std::string OctoPrint::make_url(const std::string &path) const | |||
| { | ||||
| 	if (host.find("http://") == 0 || host.find("https://") == 0) { | ||||
| 		if (host.back() == '/') { | ||||
| 			return std::move((boost::format("%1%%2%") % host % path).str()); | ||||
| 			return (boost::format("%1%%2%") % host % path).str(); | ||||
| 		} else { | ||||
| 			return std::move((boost::format("%1%/%2%") % host % path).str()); | ||||
| 			return (boost::format("%1%/%2%") % host % path).str(); | ||||
| 		} | ||||
| 	} else { | ||||
| 		return std::move((boost::format("http://%1%/%2%") % host % path).str()); | ||||
| 		return (boost::format("http://%1%/%2%") % host % path).str(); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| wxString OctoPrint::format_error(std::string error, unsigned status) | ||||
| wxString OctoPrint::format_error(const std::string &body, const std::string &error, unsigned status) | ||||
| { | ||||
| 	const wxString wxerror = error; | ||||
| 
 | ||||
| 	if (status != 0) { | ||||
| 		return wxString::Format("HTTP %u: %s", status, | ||||
| 			(status == 401 ? _(L("Invalid API key")) : wxerror)); | ||||
| 		auto wxbody = wxString::FromUTF8(body.data()); | ||||
| 		return wxString::Format("HTTP %u: %s", status, wxbody); | ||||
| 	} else { | ||||
| 		return std::move(wxerror); | ||||
| 		return wxString::FromUTF8(error.data()); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
|  |  | |||
|  | @ -17,7 +17,8 @@ public: | |||
| 	OctoPrint(DynamicPrintConfig *config); | ||||
| 
 | ||||
| 	bool test(wxString &curl_msg) const; | ||||
| 	bool send_gcode(const std::string &filename, bool print = false) const; | ||||
| 	// Send gcode file to octoprint, filename is expected to be in UTF-8
 | ||||
| 	bool send_gcode(const std::string &filename) const; | ||||
| private: | ||||
| 	std::string host; | ||||
| 	std::string apikey; | ||||
|  | @ -25,7 +26,7 @@ private: | |||
| 
 | ||||
| 	void set_auth(Http &http) const; | ||||
| 	std::string make_url(const std::string &path) const; | ||||
| 	static wxString format_error(std::string error, unsigned status); | ||||
| 	static wxString format_error(const std::string &body, const std::string &error, unsigned status); | ||||
| }; | ||||
| 
 | ||||
| 
 | ||||
|  |  | |||
|  | @ -259,7 +259,7 @@ void PresetUpdater::priv::sync_config(const std::set<VendorProfile> vendors) con | |||
| 		} | ||||
| 		const auto recommended = recommended_it->config_version; | ||||
| 
 | ||||
| 		BOOST_LOG_TRIVIAL(debug) << boost::format("New index for vendor: %1%: current version: %2%, recommended version: %3%") | ||||
| 		BOOST_LOG_TRIVIAL(debug) << boost::format("Got index for vendor: %1%: current version: %2%, recommended version: %3%") | ||||
| 			% vendor.name | ||||
| 			% vendor.config_version.to_string() | ||||
| 			% recommended.to_string(); | ||||
|  | @ -352,20 +352,25 @@ Updates PresetUpdater::priv::get_config_updates() const | |||
| 				continue; | ||||
| 			} | ||||
| 
 | ||||
| 			auto path_in_cache = cache_path / (idx.vendor() + ".ini"); | ||||
| 			if (! fs::exists(path_in_cache)) { | ||||
| 				BOOST_LOG_TRIVIAL(warning) << "Index indicates update, but new bundle not found in cache: " << path_in_cache.string(); | ||||
| 				continue; | ||||
| 			auto path_src = cache_path / (idx.vendor() + ".ini"); | ||||
| 			if (! fs::exists(path_src)) { | ||||
| 				auto path_in_rsrc = rsrc_path / (idx.vendor() + ".ini"); | ||||
| 				if (! fs::exists(path_in_rsrc)) { | ||||
| 					BOOST_LOG_TRIVIAL(warning) << boost::format("Index for vendor %1% indicates update, but bundle found in neither cache nor resources") | ||||
| 						% idx.vendor();; | ||||
| 					continue; | ||||
| 				} else { | ||||
| 					path_src = std::move(path_in_rsrc); | ||||
| 				} | ||||
| 			} | ||||
| 
 | ||||
| 			const auto cached_vp = VendorProfile::from_ini(path_in_cache, false); | ||||
| 			if (cached_vp.config_version == recommended->config_version) { | ||||
| 				updates.updates.emplace_back(std::move(path_in_cache), std::move(bundle_path), *recommended); | ||||
| 			const auto new_vp = VendorProfile::from_ini(path_src, false); | ||||
| 			if (new_vp.config_version == recommended->config_version) { | ||||
| 				updates.updates.emplace_back(std::move(path_src), std::move(bundle_path), *recommended); | ||||
| 			} else { | ||||
| 				BOOST_LOG_TRIVIAL(warning) << boost::format("Vendor: %1%: Index indicates update (%2%) but cached bundle has a different version: %3%") | ||||
| 				BOOST_LOG_TRIVIAL(warning) << boost::format("Index for vendor %1% indicates update (%2%) but the new bundle was found neither in cache nor resources") | ||||
| 					% idx.vendor() | ||||
| 					% recommended->config_version.to_string() | ||||
| 					% cached_vp.config_version.to_string(); | ||||
| 					% recommended->config_version.to_string(); | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | @ -532,15 +537,15 @@ bool PresetUpdater::config_update() const | |||
| 			incompats_map.emplace(std::make_pair(std::move(vendor), std::move(restrictions))); | ||||
| 		} | ||||
| 
 | ||||
| 		p->had_config_update = true;   // This needs to be done before a dialog is shown because of OnIdle() + CallAfter() in Perl
 | ||||
| 
 | ||||
| 		GUI::MsgDataIncompatible dlg(std::move(incompats_map)); | ||||
| 		const auto res = dlg.ShowModal(); | ||||
| 		if (res == wxID_REPLACE) { | ||||
| 			BOOST_LOG_TRIVIAL(info) << "User wants to re-configure..."; | ||||
| 			p->perform_updates(std::move(updates)); | ||||
| 			GUI::ConfigWizard wizard(nullptr, GUI::ConfigWizard::RR_DATA_INCOMPAT); | ||||
| 			if (wizard.run(GUI::get_preset_bundle(), this)) { | ||||
| 				p->had_config_update = true; | ||||
| 			} else { | ||||
| 			if (! wizard.run(GUI::get_preset_bundle(), this)) {	 | ||||
| 				return false; | ||||
| 			} | ||||
| 		} else { | ||||
|  | @ -561,6 +566,8 @@ bool PresetUpdater::config_update() const | |||
| 			updates_map.emplace(std::make_pair(std::move(vendor), std::move(ver_str))); | ||||
| 		} | ||||
| 
 | ||||
| 		p->had_config_update = true;   // Ditto, see above
 | ||||
| 
 | ||||
| 		GUI::MsgUpdateConfig dlg(std::move(updates_map)); | ||||
| 
 | ||||
| 		const auto res = dlg.ShowModal(); | ||||
|  | @ -576,8 +583,6 @@ bool PresetUpdater::config_update() const | |||
| 		} else { | ||||
| 			BOOST_LOG_TRIVIAL(info) << "User refused the update"; | ||||
| 		} | ||||
| 
 | ||||
| 		p->had_config_update = true; | ||||
| 	} else { | ||||
| 		BOOST_LOG_TRIVIAL(info) << "No configuration updates available."; | ||||
| 	} | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 tamasmeszaros
						tamasmeszaros