Port Emboss & SVG gizmo from PrusaSlicer (#2819)
* Rework UI jobs to make them more understandable and flexible. * Update Orca specific jobs * Fix progress issue * Fix dark mode and window radius * Update cereal version from 1.2.2 to 1.3.0 (cherry picked from commit prusa3d/PrusaSlicer@057232a275) * Initial port of Emboss gizmo * Bump up CGAL version to 5.4 (cherry picked from commit prusa3d/PrusaSlicer@1bf9dee3e7) * Fix text rotation * Fix test dragging * Add text gizmo to right click menu * Initial port of SVG gizmo * Fix text rotation * Fix Linux build * Fix "from surface" * Fix -90 rotation * Fix icon path * Fix loading font with non-ascii name * Fix storing non-utf8 font descriptor in 3mf file * Fix filtering with non-utf8 characters * Emboss: Use Orca style input dialog * Fix build on macOS * Fix tooltip color in light mode * InputText: fixed incorrect padding when FrameBorder > 0. (ocornut/imgui#4794, ocornut/imgui#3781) InputTextMultiline: fixed vertical tracking with large values of FramePadding.y. (ocornut/imgui#3781, ocornut/imgui#4794) (cherry picked from commit ocornut/imgui@072caa4a90) (cherry picked from commit ocornut/imgui@bdd2a94315) * SVG: Use Orca style input dialog * Fix job progress update * Fix crash when select editing text in preview screen * Use Orca checkbox style * Fix issue that toolbar icons are kept regenerated * Emboss: Fix text & icon alignment * SVG: Fix text & icon alignment * Emboss: fix toolbar icon mouse hover state * Add a simple subtle outline effect by drawing back faces using wireframe mode * Disable selection outlines * Show outline in white if the model color is too dark * Make the outline algorithm more reliable * Enable cull face, which fix render on Linux * Fix `disable_cullface` * Post merge fix * Optimize selection rendering * Fix scale gizmo * Emboss: Fix text rotation if base object is scaled * Fix volume synchronize * Fix emboss rotation * Emboss: Fix advance toggle * Fix text position after reopened the project * Make font style preview darker * Make font style preview selector height shorter --------- Co-authored-by: tamasmeszaros <meszaros.q@gmail.com> Co-authored-by: ocornut <omarcornut@gmail.com> Co-authored-by: SoftFever <softfeverever@gmail.com>
							
								
								
									
										28
									
								
								deps/CGAL/CGAL.cmake
									
										
									
									
										vendored
									
									
								
							
							
						
						|  | @ -1,31 +1,11 @@ | |||
| orcaslicer_add_cmake_project( | ||||
|     CGAL | ||||
|      GIT_REPOSITORY https://github.com/CGAL/cgal.git | ||||
|      GIT_TAG        caacd806dc55c61cc68adaad99f2240f00493b29 # releases/CGAL-5.3 | ||||
|     # GIT_REPOSITORY https://github.com/CGAL/cgal.git | ||||
|     # GIT_TAG        bec70a6d52d8aacb0b3d82a7b4edc3caa899184b # releases/CGAL-5.0 | ||||
|     # For whatever reason, this keeps downloading forever (repeats downloads if finished) | ||||
|     #URL      https://github.com/CGAL/cgal/archive/releases/CGAL-5.0.zip | ||||
|     #URL_HASH SHA256=c2b035bd078687b6d8c0fb6371a7443adcdb647856af9969532c4050cd5f48e5 | ||||
|     URL      https://github.com/CGAL/cgal/archive/refs/tags/v5.4.zip | ||||
|     URL_HASH SHA256=d7605e0a5a5ca17da7547592f6f6e4a59430a0bc861948974254d0de43eab4c0 | ||||
|     DEPENDS dep_Boost dep_GMP dep_MPFR | ||||
| ) | ||||
| 
 | ||||
| include(GNUInstallDirs) | ||||
| 
 | ||||
| # CGAL, for whatever reason, makes itself non-relocatable by writing the build directory into | ||||
| # CGALConfig-installation-dirs.cmake and including it in configure time. | ||||
| # If this file is not present, it will not consider the stored absolute path | ||||
| ExternalProject_Add_Step(dep_CGAL dep_CGAL_relocation_fix | ||||
|     DEPENDEES install | ||||
| 
 | ||||
|     COMMAND ${CMAKE_COMMAND} -E remove CGALConfig-installation-dirs.cmake | ||||
|     WORKING_DIRECTORY "${DESTDIR}/usr/local/${CMAKE_INSTALL_LIBDIR}/cmake/CGAL" | ||||
| ) | ||||
| 
 | ||||
| # Again, for whatever reason, CGAL thinks that its version is not relevant if | ||||
| # configured as a header only library. Fixing it by placing a cmake version file | ||||
| # besides the installed config file. | ||||
| ExternalProject_Add_Step(dep_CGAL dep_CGAL_version_fix | ||||
|     DEPENDEES install | ||||
| 
 | ||||
|     COMMAND ${CMAKE_COMMAND} -E copy cgal/CGALConfigVersion.cmake "${DESTDIR}/usr/local/${CMAKE_INSTALL_LIBDIR}/cmake/CGAL/CGALConfigVersion.cmake" | ||||
|     WORKING_DIRECTORY "${CMAKE_CURRENT_LIST_DIR}" | ||||
| ) | ||||
|  |  | |||
							
								
								
									
										12
									
								
								deps/Cereal/Cereal.cmake
									
										
									
									
										vendored
									
									
								
							
							
						
						|  | @ -1,6 +1,12 @@ | |||
| #/|/ Copyright (c) Prusa Research 2021 - 2022 Tomáš Mészáros @tamasmeszaros, Filip Sykala @Jony01 | ||||
| #/|/ | ||||
| #/|/ PrusaSlicer is released under the terms of the AGPLv3 or higher | ||||
| #/|/ | ||||
| orcaslicer_add_cmake_project(Cereal | ||||
|     URL "https://github.com/USCiLab/cereal/archive/v1.2.2.tar.gz" | ||||
|     URL_HASH SHA256=1921f26d2e1daf9132da3c432e2fd02093ecaedf846e65d7679ddf868c7289c4 | ||||
|     URL "https://github.com/USCiLab/cereal/archive/refs/tags/v1.3.0.zip" | ||||
|     URL_HASH SHA256=71642cb54658e98c8f07a0f0d08bf9766f1c3771496936f6014169d3726d9657 | ||||
|     CMAKE_ARGS | ||||
|         -DJUST_INSTALL_CEREAL=on | ||||
|         -DJUST_INSTALL_CEREAL=ON | ||||
|         -DSKIP_PERFORMANCE_COMPARISON=ON | ||||
|         -DBUILD_TESTS=OFF | ||||
| ) | ||||
|  | @ -8,11 +8,11 @@ | |||
| 			S11.87,1,8,1L8,1z"/> | ||||
| 	</g> | ||||
| 	<g> | ||||
| 		<path fill="#ED6B21" d="M12,8.75H4C3.59,8.75,3.25,8.41,3.25,8S3.59,7.25,4,7.25h8c0.41,0,0.75,0.34,0.75,0.75S12.41,8.75,12,8.75 | ||||
| 		<path fill="#009688" d="M12,8.75H4C3.59,8.75,3.25,8.41,3.25,8S3.59,7.25,4,7.25h8c0.41,0,0.75,0.34,0.75,0.75S12.41,8.75,12,8.75 | ||||
| 			z"/> | ||||
| 	</g> | ||||
| 	<g> | ||||
| 		<path fill="#ED6B21" d="M8,12.75c-0.41,0-0.75-0.34-0.75-0.75V4c0-0.41,0.34-0.75,0.75-0.75S8.75,3.59,8.75,4v8 | ||||
| 		<path fill="#009688" d="M8,12.75c-0.41,0-0.75-0.34-0.75-0.75V4c0-0.41,0.34-0.75,0.75-0.75S8.75,3.59,8.75,4v8 | ||||
| 			C8.75,12.41,8.41,12.75,8,12.75z"/> | ||||
| 	</g> | ||||
| </g> | ||||
|  |  | |||
| Before Width: | Height: | Size: 846 B After Width: | Height: | Size: 846 B | 
							
								
								
									
										4
									
								
								resources/images/add_text_modifier.svg
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,4 @@ | |||
| <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16px" height="16px"> | ||||
| <path fill="#808080" d="m 4,0 v 4 h 4 v 10 h 4 V 4 h 4 V 0 Z m 1.121094,1 h 1.414062 l 2,2 H 7.115235 Z M 7.242188,1 H 8.65625 L 11,3.34375 v 1.4140625 z m 2.121093,0 h 1.408205 l 1.998045,1.998 H 11.35547 Z m 2.121094,0 h 1.414063 l 1.992186,2 h -1.414061 z m 2.121094,0 H 15 V 2.3964375 Z M 5,1.59175 6.408203,3 H 5 Z m 4,1.8789062 2,1.9941407 V 6.8788594 L 9,4.8847188 Z M 9,5.59175 11,7.5858906 V 9 L 9,7 Z m 0,2.1210938 2,2.0000002 v 1.414063 L 9,9.126906 Z m 0,2.1210942 2,2 V 13 H 10.761152 L 9,11.248 Z M 9,12 10,13 H 9 Z" /> | ||||
| <path fill="#009688" d="M 3.5 9 C 1.5741139 9 0 10.574114 0 12.5 C 0 14.425886 1.5741139 16 3.5 16 C 5.4258861 16 7 14.425886 7 12.5 C 7 10.574114 5.4258861 9 3.5 9 z M 3.5 10.199219 C 4.7773592 10.199219 5.8007813 11.222641 5.8007812 12.5 C 5.8007812 12.667974 5.7816922 12.830839 5.7480469 12.988281 L 3.0117188 10.251953 C 3.1691607 10.218308 3.3320257 10.199219 3.5 10.199219 z M 2.4863281 10.433594 L 5.5664062 13.513672 C 5.3371871 13.983352 4.9544913 14.362231 4.4804688 14.583984 L 1.4160156 11.521484 C 1.6376653 11.046724 2.0161177 10.663072 2.4863281 10.433594 z M 1.2421875 12.054688 L 3.9472656 14.757812 C 3.8024888 14.786013 3.6534389 14.800781 3.5 14.800781 C 2.2226408 14.800781 1.1992188 13.777359 1.1992188 12.5 C 1.1992188 12.347226 1.2142247 12.198881 1.2421875 12.054688 z " /> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 1.4 KiB | 
							
								
								
									
										4
									
								
								resources/images/add_text_negative.svg
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,4 @@ | |||
| <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16px" height="16px"> | ||||
| <path fill="#808080" d="m 5,14 v 2 H 10.293 11 V 15.293 6 h 5 V 0 H 0 v 6 h 5 v 5 l 1,0 V 5 L 4,3 H 1.5 V 4 H 3.583984 L 4.587891,5 H 1 V 1 h 14 v 3.293 l -1,-1 V 1.5 H 13 V 3 H 8 V 13 H 7 v 1 h 1.293 l 1,1 H 6 V 14 Z M 9.707,4 h 3.586 l 1,1 H 10.707 Z M 9,4.707 l 1,1 v 8.586 l -1,-1 z" /> | ||||
| <path fill="#009688" d="M 1,12 C 0.5,12 0,12 0,12.5 0,13 0.5,13 1,13 H 6 C 6.5,13 7,13 7,12.5 7,12 6.5,12 6,12 Z" /> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 503 B | 
							
								
								
									
										4
									
								
								resources/images/add_text_part.svg
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,4 @@ | |||
| <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16px" height="16px"> | ||||
| <path fill="#808080" d="M 6,11 V 7.5 H 5 V 11 Z M 5,13.5 7.53125,16 H 12 V 7 h 4 V 2.5 L 13.625,0 H 1 V 4.7128906 L 3.419922,7 H 7 v 7.257812 l -1,-1 z M 2.695312,1 h 10.580079 l 1,1 H 3.695312 Z M 2,1.703125 l 1,1 V 5.40625 l -1,-1 z M 4,3 h 11 v 3 h -4 v 9 H 8 V 6 H 4 Z" /> | ||||
| <path fill="#009688" d="M 3.5,9 C 3,9 3,9.5 3,10 v 2 H 1 C 0.5,12 0,12 0,12.5 0,13 0.5,13 1,13 h 2 v 2 c 0,0.5 0,1 0.5,1 C 4,16 4,15.5 4,15 V 13 H 6 C 6.5,13 7,13 7,12.5 7,12 6.5,12 6,12 H 4 V 10 C 4,9.5 4,9 3.5,9 Z" /> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 592 B | 
							
								
								
									
										7
									
								
								resources/images/align_horizontal_center.svg
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,7 @@ | |||
| <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16px" height="16px"> | ||||
| <path fill="#009688" d="m 7,14 c 0,0 0,1 1,1 1,0 1,-1 1,-1 V 2 C 9,2 9,1 8,1 7,1 7,2 7,2 Z"/> | ||||
| <path fill="#808080" d="M 3,3 H 13 V 4 H 3 Z" /> | ||||
| <path fill="#808080" d="m 3.5,9 h 9 v 1 h -9 z" /> | ||||
| <path fill="#808080" d="m 5.5,6 h 5 v 1 h -5 z" /> | ||||
| <path fill="#808080" d="m 4,12 h 8 v 1 H 4 Z" /> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 389 B | 
							
								
								
									
										7
									
								
								resources/images/align_horizontal_left.svg
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,7 @@ | |||
| <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16px" height="16px"> | ||||
| <path fill="#009688" d="m 1,14 c 0,0 0,1 1,1 1,0 1,-1 1,-1 V 2 C 3,2 3,1 2,1 1,1 1,2 1,2 Z" /> | ||||
| <path fill="#808080" d="M 2,3 H 12 V 4 H 2 Z" /> | ||||
| <path fill="#808080" d="m 2,9 h 9 v 1 H 2 Z" /> | ||||
| <path fill="#808080" d="M 2,6 H 7 V 7 H 2 Z" /> | ||||
| <path fill="#808080" d="m 2,12 h 8 v 1 H 2 Z" /> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 383 B | 
							
								
								
									
										7
									
								
								resources/images/align_horizontal_right.svg
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,7 @@ | |||
| <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16px" height="16px"> | ||||
| <path fill="#009688" d="m 15,14 c 0,0 0,1 -1,1 -1,0 -1,-1 -1,-1 V 2 c 0,0 0,-1 1,-1 1,0 1,1 1,1 z" /> | ||||
| <path fill="#808080" d="M 14,3 H 4 v 1 h 10 z" /> | ||||
| <path fill="#808080" d="M 14,9 H 5 v 1 h 9 z" /> | ||||
| <path fill="#808080" d="M 14,6 H 9 v 1 h 5 z" /> | ||||
| <path fill="#808080" d="M 14,12 H 6 v 1 h 8 z" /> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 395 B | 
							
								
								
									
										60
									
								
								resources/images/align_vertical_bottom.svg
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,60 @@ | |||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <svg | ||||
|    viewBox="0 0 16 16" | ||||
|    width="16px" | ||||
|    height="16px" | ||||
|    version="1.1" | ||||
|    id="svg12" | ||||
|    sodipodi:docname="vertic_bottom.svg" | ||||
|    inkscape:version="1.2 (dc2aedaf03, 2022-05-15)" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:svg="http://www.w3.org/2000/svg"> | ||||
|   <defs | ||||
|      id="defs16" /> | ||||
|   <sodipodi:namedview | ||||
|      id="namedview14" | ||||
|      pagecolor="#ffffff" | ||||
|      bordercolor="#000000" | ||||
|      borderopacity="0.25" | ||||
|      inkscape:showpageshadow="2" | ||||
|      inkscape:pageopacity="0.0" | ||||
|      inkscape:pagecheckerboard="0" | ||||
|      inkscape:deskcolor="#d1d1d1" | ||||
|      showgrid="true" | ||||
|      inkscape:zoom="41.7193" | ||||
|      inkscape:cx="12.356391" | ||||
|      inkscape:cy="8.5092511" | ||||
|      inkscape:window-width="1920" | ||||
|      inkscape:window-height="1129" | ||||
|      inkscape:window-x="1912" | ||||
|      inkscape:window-y="-8" | ||||
|      inkscape:window-maximized="1" | ||||
|      inkscape:current-layer="svg12"> | ||||
|     <inkscape:grid | ||||
|        type="xygrid" | ||||
|        id="grid299" /> | ||||
|   </sodipodi:namedview> | ||||
|   <path | ||||
|      fill="#009688" | ||||
|      d="M 2,11 C 2,11 1,11 1,12.5 1,14 2,14 2,14 h 12 c 0,0 1,0 1,-1.5 C 15,11 14,11 14,11 Z" | ||||
|      id="path2" | ||||
|      style="stroke-width:1.22474" /> | ||||
|   <path | ||||
|      fill="#808080" | ||||
|      d="M 3,3 H 13 V 4 H 3 Z" | ||||
|      id="path4" /> | ||||
|   <path | ||||
|      fill="#808080" | ||||
|      d="M 3,6 H 13 V 7 H 3 Z" | ||||
|      id="path301" /> | ||||
|   <path | ||||
|      fill="#808080" | ||||
|      d="m 3,9 h 10 v 1 H 3 Z" | ||||
|      id="path303" /> | ||||
|   <path | ||||
|      fill="#808080" | ||||
|      d="m 3,12 h 10 v 1 H 3 Z" | ||||
|      id="path305" /> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 1.6 KiB | 
							
								
								
									
										60
									
								
								resources/images/align_vertical_center.svg
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,60 @@ | |||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <svg | ||||
|    viewBox="0 0 16 16" | ||||
|    width="16px" | ||||
|    height="16px" | ||||
|    version="1.1" | ||||
|    id="svg12" | ||||
|    sodipodi:docname="vertic_center.svg" | ||||
|    inkscape:version="1.2 (dc2aedaf03, 2022-05-15)" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:svg="http://www.w3.org/2000/svg"> | ||||
|   <defs | ||||
|      id="defs16" /> | ||||
|   <sodipodi:namedview | ||||
|      id="namedview14" | ||||
|      pagecolor="#ffffff" | ||||
|      bordercolor="#000000" | ||||
|      borderopacity="0.25" | ||||
|      inkscape:showpageshadow="2" | ||||
|      inkscape:pageopacity="0.0" | ||||
|      inkscape:pagecheckerboard="0" | ||||
|      inkscape:deskcolor="#d1d1d1" | ||||
|      showgrid="true" | ||||
|      inkscape:zoom="41.7193" | ||||
|      inkscape:cx="12.356391" | ||||
|      inkscape:cy="8.5092511" | ||||
|      inkscape:window-width="1920" | ||||
|      inkscape:window-height="1129" | ||||
|      inkscape:window-x="1912" | ||||
|      inkscape:window-y="-8" | ||||
|      inkscape:window-maximized="1" | ||||
|      inkscape:current-layer="svg12"> | ||||
|     <inkscape:grid | ||||
|        type="xygrid" | ||||
|        id="grid299" /> | ||||
|   </sodipodi:namedview> | ||||
|   <path | ||||
|      fill="#009688" | ||||
|      d="m 2,6.5 c 0,0 -1,0 -1,1.5 0,1.5 1,1.5 1,1.5 h 12 c 0,0 1,0 1,-1.5 0,-1.5 -1,-1.5 -1,-1.5 z" | ||||
|      id="path2" | ||||
|      style="stroke-width:1.22474" /> | ||||
|   <path | ||||
|      fill="#808080" | ||||
|      d="M 3,3 H 13 V 4 H 3 Z" | ||||
|      id="path4" /> | ||||
|   <path | ||||
|      fill="#808080" | ||||
|      d="M 3,6 H 13 V 7 H 3 Z" | ||||
|      id="path301" /> | ||||
|   <path | ||||
|      fill="#808080" | ||||
|      d="m 3,9 h 10 v 1 H 3 Z" | ||||
|      id="path303" /> | ||||
|   <path | ||||
|      fill="#808080" | ||||
|      d="m 3,12 h 10 v 1 H 3 Z" | ||||
|      id="path305" /> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 1.6 KiB | 
							
								
								
									
										60
									
								
								resources/images/align_vertical_top.svg
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,60 @@ | |||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <svg | ||||
|    viewBox="0 0 16 16" | ||||
|    width="16px" | ||||
|    height="16px" | ||||
|    version="1.1" | ||||
|    id="svg12" | ||||
|    sodipodi:docname="vertic_top.svg" | ||||
|    inkscape:version="1.2 (dc2aedaf03, 2022-05-15)" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:svg="http://www.w3.org/2000/svg"> | ||||
|   <defs | ||||
|      id="defs16" /> | ||||
|   <sodipodi:namedview | ||||
|      id="namedview14" | ||||
|      pagecolor="#ffffff" | ||||
|      bordercolor="#000000" | ||||
|      borderopacity="0.25" | ||||
|      inkscape:showpageshadow="2" | ||||
|      inkscape:pageopacity="0.0" | ||||
|      inkscape:pagecheckerboard="0" | ||||
|      inkscape:deskcolor="#d1d1d1" | ||||
|      showgrid="true" | ||||
|      inkscape:zoom="41.7193" | ||||
|      inkscape:cx="12.356391" | ||||
|      inkscape:cy="8.5092511" | ||||
|      inkscape:window-width="1920" | ||||
|      inkscape:window-height="1129" | ||||
|      inkscape:window-x="1912" | ||||
|      inkscape:window-y="-8" | ||||
|      inkscape:window-maximized="1" | ||||
|      inkscape:current-layer="svg12"> | ||||
|     <inkscape:grid | ||||
|        type="xygrid" | ||||
|        id="grid299" /> | ||||
|   </sodipodi:namedview> | ||||
|   <path | ||||
|      fill="#009688" | ||||
|      d="M 2,2 C 2,2 1,2 1,3.5 1,5 2,5 2,5 H 14 C 14,5 15,5 15,3.5 15,2 14,2 14,2 Z" | ||||
|      id="path2" | ||||
|      style="stroke-width:1.22474" /> | ||||
|   <path | ||||
|      fill="#808080" | ||||
|      d="M 3,3 H 13 V 4 H 3 Z" | ||||
|      id="path4" /> | ||||
|   <path | ||||
|      fill="#808080" | ||||
|      d="M 3,6 H 13 V 7 H 3 Z" | ||||
|      id="path301" /> | ||||
|   <path | ||||
|      fill="#808080" | ||||
|      d="m 3,9 h 10 v 1 H 3 Z" | ||||
|      id="path303" /> | ||||
|   <path | ||||
|      fill="#808080" | ||||
|      d="m 3,12 h 10 v 1 H 3 Z" | ||||
|      id="path305" /> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 1.6 KiB | 
							
								
								
									
										4
									
								
								resources/images/burn.svg
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,4 @@ | |||
| <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16px" height="16px"> | ||||
| <path fill="#808080" d="M 10.324265,12.637091 C 9.7126185,11.801826 9.4455652,11.319769 9.3860031,10.943437 9.3387131,10.644653 9.2775165,10.67725 9.032793,11.131535 8.8565876,11.45863 8.806364,11.904627 8.9008009,12.303679 8.330577,11.894019 7.6015175,10.031363 7.7096999,9.1284851 5.381808,10.598103 4.7271155,13.939753 5,16 2.9751051,14.608227 1.2851951,13.046842 1.3078808,10.57598 1.3305665,8.1051179 3.8432395,6.4490194 3.8697024,4.1713523 3.8727624,3.3745518 3.8421277,3.3793755 4.5118024,4.0701844 5.3014406,5.073513 5.8791344,5.964439 5.6521467,7.0731381 6.1486239,6.623741 6.3341015,5.940722 6.5401623,5.4070849 6.8689362,3.4222537 6.7021829,1.5642078 6,0 c 2.5998224,0.67409817 4.550997,3.5888298 4.623306,5.6887828 l 9.3e-4,0.4921817 C 11.260843,5.1624272 11.966175,4.3725405 13,4 12.774748,6.4272659 14.752236,8.539235 14.683585,10.819702 14.614934,13.100169 13.178608,15.41894 11,16 11.750123,14.792953 10.874074,13.393198 10.324265,12.637091 Z" /> | ||||
| <path fill="#009688" d="M 9.032793,11.131535 C 8.8565876,11.45863 8.806364,11.904627 8.9008009,12.303679 8.2684185,12.02479 7.7175594,10.368983 7.6751289,9.4545414 7.7066738,9.2878597 7.7222018,9.1411316 7.7097002,9.1284851 8.5239771,6.6708553 5.893441,7.0819045 6.5401623,5.4070849 6.8689362,3.4222537 6.7021829,1.5642078 6,0 c 2.5998224,0.67409817 4.550997,3.5888298 4.623306,5.6887828 l 9.3e-4,0.4921817 c -0.04706,1.5271028 -0.04706,2.5643146 -1.591443,4.9505705 z" /> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 1.5 KiB | 
							
								
								
									
										22
									
								
								resources/images/delete.svg
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,22 @@ | |||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <!-- Generator: Adobe Illustrator 23.0.3, SVG Export Plug-In . SVG Version: 6.00 Build 0)  --> | ||||
| <svg version="1.0" id="Layer_1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px" | ||||
| 	 viewBox="0 0 16 16" enable-background="new 0 0 16 16" xml:space="preserve"> | ||||
| <g id="delete"> | ||||
| 	<g> | ||||
| 		<path fill="#009688" d="M13,10c0-1.1-0.9-2-2-2H5c-1.1,0-2,0.9-2,2v3c0,1.1,0.9,2,2,2h6c1.1,0,2-0.9,2-2V10z"/> | ||||
| 	</g> | ||||
| 	<g> | ||||
| 		<path fill="#808080" d="M12,5.01c0.25,0,0.47,0.09,0.62,0.26c0.15,0.17,0.22,0.4,0.2,0.64l-0.64,7.01C12.13,13.49,11.58,14,11,14 | ||||
| 			H5c-0.58,0-1.13-0.51-1.19-1.08L3.18,5.91c-0.02-0.25,0.05-0.47,0.2-0.64C3.53,5.1,3.75,5.01,4,5.01H12 M12,4.01H4 | ||||
| 			c-1.1,0-1.92,0.9-1.82,1.99l0.64,7.01C2.92,14.1,3.9,15,5,15h6c1.1,0,2.08-0.9,2.18-1.99L13.82,6C13.92,4.91,13.1,4.01,12,4.01 | ||||
| 			L12,4.01z"/> | ||||
| 	</g> | ||||
| 	<g> | ||||
| 		<path fill="#808080" d="M13,3.5H3C2.72,3.5,2.5,3.28,2.5,3S2.72,2.5,3,2.5h10c0.28,0,0.5,0.22,0.5,0.5S13.28,3.5,13,3.5z"/> | ||||
| 	</g> | ||||
| 	<g> | ||||
| 		<path fill="#808080" d="M10,2.5H6C5.72,2.5,5.5,2.28,5.5,2S5.72,1.5,6,1.5h4c0.28,0,0.5,0.22,0.5,0.5S10.28,2.5,10,2.5z"/> | ||||
| 	</g> | ||||
| </g> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 1.1 KiB | 
							
								
								
									
										17
									
								
								resources/images/exclamation.svg
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,17 @@ | |||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <!-- Generator: Adobe Illustrator 23.0.3, SVG Export Plug-In . SVG Version: 6.00 Build 0)  --> | ||||
| <svg version="1.0" id="Layer_1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px" | ||||
| 	 viewBox="0 0 16 16" enable-background="new 0 0 16 16" xml:space="preserve"> | ||||
| <g> | ||||
| 	<g> | ||||
| 		<path fill="#009688" d="M8,11c-0.55,0-1-0.45-1-1V7c0-0.55,0.45-1,1-1s1,0.45,1,1v3C9,10.55,8.55,11,8,11z"/> | ||||
| 	</g> | ||||
| 	<g> | ||||
| 		<circle fill="#009688" cx="8" cy="13" r="1"/> | ||||
| 	</g> | ||||
| 	<g> | ||||
| 		<path fill="#009688" d="M15,15.5H1c-0.18,0-0.34-0.09-0.43-0.24c-0.09-0.15-0.09-0.34-0.01-0.49l7-13c0.17-0.32,0.71-0.32,0.88,0 | ||||
| 			l7,13c0.08,0.16,0.08,0.34-0.01,0.49C15.34,15.41,15.18,15.5,15,15.5z M1.84,14.5h12.33L8,3.05L1.84,14.5z"/> | ||||
| 	</g> | ||||
| </g> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 781 B | 
							
								
								
									
										10
									
								
								resources/images/lock_closed.svg
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,10 @@ | |||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <!-- Generator: Adobe Illustrator 23.0.3, SVG Export Plug-In . SVG Version: 6.00 Build 0)  --> | ||||
| <svg version="1.0" id="Layer_1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px" | ||||
| 	 viewBox="0 0 16 16" enable-background="new 0 0 16 16" xml:space="preserve"> | ||||
| <g id="lock_x5F_closed"> | ||||
| 	<path fill="none" stroke="#808080" stroke-width="2" stroke-miterlimit="10" d="M4,8V4c0,0,0-2,2-2c1,0,3,0,4,0c2,0,2,2,2,2v4"/> | ||||
| 	<path fill="#808080" d="M13,8H3C2.45,8,2,8.45,2,9v5c0,0.55,0.45,1,1,1h10c0.55,0,1-0.45,1-1V9C14,8.45,13.55,8,13,8z M10,12H8.91 | ||||
| 		c-0.21,0.58-0.76,1-1.41,1C6.67,13,6,12.33,6,11.5S6.67,10,7.5,10c0.65,0,1.2,0.42,1.41,1H10V12z"/> | ||||
| </g> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 729 B | 
							
								
								
									
										10
									
								
								resources/images/lock_closed_f.svg
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,10 @@ | |||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <!-- Generator: Adobe Illustrator 23.0.3, SVG Export Plug-In . SVG Version: 6.00 Build 0)  --> | ||||
| <svg version="1.0" id="Layer_1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px" | ||||
| 	 viewBox="0 0 16 16" enable-background="new 0 0 16 16" xml:space="preserve"> | ||||
| <g id="lock_x5F_closed"> | ||||
| 	<path fill="none" stroke="#808080" stroke-width="3" stroke-miterlimit="9" d="M4,8V4c0,0,0-2,2-2c1,0,3,0,4,0c2,0,2,2,2,2v4"/> | ||||
| 	<path fill="#808080" d="M13,8H3C2.45,8,2,8.45,2,9v5c0,0.55,0.45,1,1,1h10c0.55,0,1-0.45,1-1V9C14,8.45,13.55,8,13,8z M10,12H8.91 | ||||
| 		c-0.21,0.58-0.76,1-1.41,1C6.67,13,6,12.33,6,11.5S6.67,10,7.5,10c0.65,0,1.2,0.42,1.41,1H10V12z"/> | ||||
| </g> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 728 B | 
							
								
								
									
										11
									
								
								resources/images/lock_open.svg
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,11 @@ | |||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <!-- Generator: Adobe Illustrator 23.0.3, SVG Export Plug-In . SVG Version: 6.00 Build 0)  --> | ||||
| <svg version="1.0" id="Layer_1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px" | ||||
| 	 viewBox="0 0 16 16" enable-background="new 0 0 16 16" xml:space="preserve"> | ||||
| <g id="lock_x5F_open"> | ||||
| 	<path fill="none" stroke="#009688" stroke-width="2" stroke-linecap="round" stroke-miterlimit="10" d="M4,8V4c0,0,0-2,2-2 | ||||
| 		c1,0,3,0,4,0c2,0,2,2,2,2v1"/> | ||||
| 	<path fill="#009688" d="M13,8H3C2.45,8,2,8.45,2,9v5c0,0.55,0.45,1,1,1h10c0.55,0,1-0.45,1-1V9C14,8.45,13.55,8,13,8z M10,12H8.91 | ||||
| 		c-0.21,0.58-0.76,1-1.41,1C6.67,13,6,12.33,6,11.5S6.67,10,7.5,10c0.65,0,1.2,0.42,1.41,1H10V12z"/> | ||||
| </g> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 753 B | 
							
								
								
									
										11
									
								
								resources/images/lock_open_f.svg
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,11 @@ | |||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <!-- Generator: Adobe Illustrator 23.0.3, SVG Export Plug-In . SVG Version: 6.00 Build 0)  --> | ||||
| <svg version="1.0" id="Layer_1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px" | ||||
| 	 viewBox="0 0 16 16" enable-background="new 0 0 16 16" xml:space="preserve"> | ||||
| <g id="lock_x5F_open"> | ||||
| 	<path fill="none" stroke="#009688" stroke-width="3" stroke-linecap="round" stroke-miterlimit="10" d="M4,8V4c0,0,0-2,2-2 | ||||
| 		c1,0,3,0,4,0c2,0,2,2,2,2v1"/> | ||||
| 	<path fill="#009688" d="M13,8H3C2.45,8,2,8.45,2,9v5c0,0.55,0.45,1,1,1h10c0.55,0,1-0.45,1-1V9C14,8.45,13.55,8,13,8z M10,12H8.91 | ||||
| 		c-0.21,0.58-0.76,1-1.41,1C6.67,13,6,12.33,6,11.5S6.67,10,7.5,10c0.65,0,1.2,0.42,1.41,1H10V12z"/> | ||||
| </g> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 753 B | 
							
								
								
									
										4
									
								
								resources/images/make_bold.svg
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,4 @@ | |||
| <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16px" height="16px"> | ||||
| <path fill="#808080" d="m 4.25,1.25 v 13.5 H 6.4238281 C 7.9550132,14.7449 9.3017782,14.70695 10.224609,14.236328 11.146154,13.766363 11.714577,12.89426 11.748047,11.001953 11.715611,9.759759 11.205071,8.9432111 10.546875,8.4101562 9.1029625,7.398789 8.344894,7.5382005 7.2507941,7.4731562 8.4769831,7.2602927 9.6564272,6.8632046 10.164063,6.1347656 10.274222,5.9766906 10.75,5.2822758 10.75,4.203125 10.75,3.2734782 10.376638,2.5663256 9.6445312,2.0566406 8.9124241,1.5469557 7.8014876,1.25 6.359375,1.25 Z M 6.3964844,1.734375 C 7.2161251,1.749043 8.1487931,1.833985 8.9101562,2.2109375 9.6715196,2.5878899 10.247584,3.3058125 10.248047,4.4316406 10.248331,5.1225624 10.039341,5.6730907 9.6875,6.0800781 9.3356593,6.4870656 8.8548801,6.7482198 8.3261719,6.9179688 7.2687554,7.25747 5.999968,7.25 5,7.25 H 4.75 V 1.7519531 L 4.9980469,1.75 C 5.3500853,1.7484474 5.8499795,1.7245951 6.3964844,1.734375 Z M 4.75,7.75 H 5 c 0.9307414,0 1.1815757,-0.01401 2.0039062,0 1.5407673,7.868e-4 2.5967243,0.4032699 3.2636718,1.0253906 0.6676,0.6227287 0.926463,1.4487864 0.929688,2.2265624 0.0033,0.784756 -0.253554,1.602823 -0.863282,2.222656 C 9.7242569,13.844443 8.7711925,14.25 7.4414062,14.25 c -0.5191067,0.0053 -1.7497727,0.0034 -2.4433593,0 L 4.75,14.248047 V 8 Z" /> | ||||
| <path fill="#009688" d="M 3 0 L 3 16 L 6.3613281 16 C 10.584392 16.000011 13.000613 14.99955 13.066406 11.099609 C 13.000613 8.4746378 11.421875 7.5967 10.5625 7.296875 C 11 7.0001348 12 6.4851964 12 4 C 11.998007 1.2820714 9.9990238 2.4680048e-07 6.359375 0 L 3 0 z M 4 1 L 6.359375 1 C 9.3115238 1 11 2.203125 11 4.203125 C 11 6.4383214 9.96875 6.8438848 8.484375 7.375 C 9.78125 7.4092 11.934207 8.3750284 12 11 C 11.93421 14.899941 9.4906414 14.989816 6.4238281 15 L 4 15 L 4 1 z M 6.3925781 1.984375 C 5.8561577 1.9747756 5.3619154 1.9984038 5 2 L 5 7 C 7 7 9.9991037 7.0007673 9.9980469 4.4316406 C 9.997186 2.3388824 8.0018393 2.0131733 6.3925781 1.984375 z M 6 3.0058594 C 7.4476617 2.9994748 8.9991037 3.0007673 8.9824219 4.4472656 C 8.9653933 5.9238329 7.5802442 6.000795 6 6.0097656 L 6 3.0058594 z M 5 8 L 5 14 C 5.6931417 14.0034 6.9277744 14.0053 7.4414062 14 C 10.00096 14 10.95332 12.463982 10.947266 11.003906 C 10.941263 9.5560542 10 8 7 8 C 6.1813678 7.9860533 5.9332322 8 5 8 z M 6 9 C 6.9332322 9 6.5973834 8.9977721 7.4160156 9.0117188 C 8.9999302 9.0009732 9.9881376 9.5560542 9.9941406 11.003906 C 10.000194 12.463982 9.0009597 13 7.3632812 13 C 6.8496494 13.0053 6.6931417 13.0034 6 13 L 6 9 z " /> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 2.5 KiB | 
							
								
								
									
										4
									
								
								resources/images/make_italic.svg
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,4 @@ | |||
| <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16px" height="16px"> | ||||
| <path fill="#808080" d="M 3 0 L 3 2 L 5 2 L 5 13.5 L 5.1523438 13.5 L 7 6.6035156 L 7 2.5 L 6 2.5 L 6.7695312 0 L 3 0 z M 3 14 L 3 14 L 3 14 L 3 14 z " /> | ||||
| <path fill="#009688" d="M 9,16 3,16 3.5358984,14 H 5.5358985 L 8.7512887,2 H 6.7512886 L 7.2871873,0 13.287187,0 12.751289,2 h -2 L 7.5358984,14 h 2 z" /> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 405 B | 
							
								
								
									
										4
									
								
								resources/images/make_unbold.svg
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,4 @@ | |||
| <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16px" height="16px"> | ||||
| <path fill="#808080" d="M 3 0 L 3 16 L 6.3613281 16 C 10.584392 16.000011 13.000613 14.99955 13.066406 11.099609 C 13.000613 8.4746378 11.421875 7.5967 10.5625 7.296875 C 11 7.0001348 12 6.4851964 12 4 C 11.998007 1.2820714 9.9990238 2.4680048e-07 6.359375 0 L 3 0 z M 3.5 0.5 L 6.359375 0.5 C 7.9033731 0.5 9.1632916 0.80839367 10.072266 1.4414062 C 10.981239 2.0744189 11.5 3.0625 11.5 4.203125 C 11.5 5.397618 11.202666 6.2262446 10.654297 6.7792969 C 10.380321 7.0556129 9.9925213 7.1486579 9.6542969 7.3125 C 10.122334 7.4649494 10.58062 7.4742236 11.017578 7.828125 C 11.825156 8.4821989 12.463719 9.5408321 12.5 10.988281 L 12.5 10.998047 L 12.5 11.007812 C 12.465345 13.062131 11.72635 14.311861 10.564453 14.904297 C 9.4025567 15.4967 7.9616779 15.4949 6.4238281 15.5 L 3.5 15.5 L 3.5 0.5 z M 6.6914062 2.5117188 C 6.2771596 2.5022917 5.8703114 2.5182618 5.5 2.5253906 L 5.5 6.4511719 C 6.3843057 6.4419785 7.3560744 6.4413087 8.0976562 6.203125 C 8.5398322 6.0611055 8.8878766 5.8595959 9.1210938 5.5898438 C 9.3543108 5.3200916 9.498271 4.976613 9.4980469 4.4316406 C 9.4978035 3.8400832 9.33166 3.4925368 9.0664062 3.2246094 C 8.8011525 2.956682 8.4018312 2.7685186 7.9238281 2.6523438 C 7.5272979 2.5559702 7.1056529 2.5211458 6.6914062 2.5117188 z M 6 3.0058594 C 7.4476617 2.9994748 8.9991037 3.0007673 8.9824219 4.4472656 C 8.9653933 5.9238329 7.5802442 6.000795 6 6.0097656 L 6 3.0058594 z M 5.5 8.5 L 5.5 13.5 C 6.184944 13.502 7.0345824 13.5041 7.4355469 13.5 L 7.4375 13.5 L 7.4414062 13.5 C 8.6211924 13.5 9.3462736 13.159029 9.7988281 12.699219 C 10.251383 12.239408 10.449866 11.626375 10.447266 11.005859 C 10.44471 10.389517 10.253823 9.7888382 9.7558594 9.3242188 C 9.2578953 8.8595992 8.4135287 8.5 7 8.5 L 6.9960938 8.5 L 6.9921875 8.5 C 6.3845564 8.489648 6.0006946 8.49674 5.5 8.5 z M 6 9 C 6.9332322 9 6.5973834 8.9977721 7.4160156 9.0117188 C 8.9999302 9.0009732 9.9881376 9.5560542 9.9941406 11.003906 C 10.000194 12.463982 9.0009597 13 7.3632812 13 C 6.8496494 13.0053 6.6931417 13.0034 6 13 L 6 9 z M 4.5 14.498047 L 4.5 14.5 L 4.9980469 14.5 L 4.5 14.498047 z " /> | ||||
| <path fill="#009688" d="m 12,11 c -0.06579,3.899941 -2.5100604,3.989816 -5.5768737,4 H 4 V 1 H 6.3603512 C 9.3125,1 11,2.203125 11,4.203125 11,6.4383214 9.96875,6.84375 8.484375,7.3748652 9.78125,7.4090652 11.934207,8.3750284 12,11 Z M 9.9989432,4.4308733 C 9.9977954,1.640529 6.4476617,1.9936154 5,2 V 7 C 7,7 10,7 9.9989432,4.4308733 Z M 10.947071,11.004336 C 10.941068,9.556484 10,8 7,8 6.1813678,7.9860533 5.9332322,8 5,8 v 6 c 0.6931417,0.0034 1.9268147,0.0053 2.4404466,0 C 10,14 10.953125,12.464412 10.947071,11.004336 Z" /> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 2.7 KiB | 
							
								
								
									
										4
									
								
								resources/images/make_unitalic.svg
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,4 @@ | |||
| <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16px" height="16px"> | ||||
| <path fill="#808080" d="M 9.5,0 V 2.5 H 8.6171875 L 7.5,6.6777344 V 13.5 h 0.171875 L 10.751953,2 h 2 l 0.535156,-2 z" /> | ||||
| <path fill="#009688" d="m 9,16 -6,0 L 3,14 H 5 V 2 H 3 L 3,0 9,0 V 2 H 7 V 14 h 2 z" /> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 305 B | 
|  | @ -5,7 +5,7 @@ | |||
| <g id="open"> | ||||
| 	<path fill="#808080" d="M1.22,14V3c0,0,0-1,1-1s4,0,5,0s1,2,2,2s4,0,4,0s1,0,1,1v2h-1c0,0,0,0,0-1s-1-1-1-1h-3.5c-1,0-1-2-2-2 | ||||
| 		s-3.5,0-3.5,0c-1,0-1,1-1,1v9v1h1v1c0,0,0,0-1,0S1.22,14,1.22,14z"/> | ||||
| 	<path fill="#ED6B21" d="M5,6C4.45,6,3.86,6.43,3.68,6.95l-2.37,7.1C1.14,14.57,1.45,15,2,15h10c0.55,0,1.14-0.43,1.32-0.95 | ||||
| 	<path fill="#009688" d="M5,6C4.45,6,3.86,6.43,3.68,6.95l-2.37,7.1C1.14,14.57,1.45,15,2,15h10c0.55,0,1.14-0.43,1.32-0.95 | ||||
| 		l2.37-7.1C15.86,6.43,15.55,6,15,6L5,6z"/> | ||||
| </g> | ||||
| </svg> | ||||
|  |  | |||
| Before Width: | Height: | Size: 722 B After Width: | Height: | Size: 722 B | 
							
								
								
									
										4
									
								
								resources/images/reflection_x.svg
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,4 @@ | |||
| <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16px" height="16px"> | ||||
| <path fill="#808080" d="M 6,0 0,16 H 6 Z M 7,0 V 3 H 8 V 0 Z M 7,4 V 5 H 8 V 4 Z M 5,5.515625 V 15 H 1.4550781 Z M 7,6 v 4 H 8 V 6 Z m 0,5 v 1 h 1 v -1 z m 0,2 v 3 h 1 v -3 z" /> | ||||
| <path fill="#009688" d="m 9,0 6,16 H 9 Z m 1,5.515625 V 15 h 3.545765 z" /> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 350 B | 
							
								
								
									
										4
									
								
								resources/images/reflection_y.svg
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,4 @@ | |||
| <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16px" height="16px"> | ||||
| <path fill="#808080" d="M 16,6 0,0 v 6 z m 0,1 h -3 v 1 h 3 z m -4,0 h -1 v 1 h 1 z M 10.484375,5 H 1 V 1.4550781 Z M 10,7 H 6 v 1 h 4 z M 5,7 H 4 V 8 H 5 Z M 3,7 H 0 v 1 h 3 z" /> | ||||
| <path fill="#009688" d="M 16,9 0,15 V 9 Z m -5.515625,1 H 1 v 3.545765 z" /> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 353 B | 
							
								
								
									
										4
									
								
								resources/images/refresh.svg
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,4 @@ | |||
| <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16px" height="16px"> | ||||
| <path fill="#009688" d="m 0.80996126,4.2067338 c 0,0 1.45662384,-2.2486082 3.76076774,-3.45466511 2.3041438,-1.2060569 5.1439072,-0.85800021 7.005512,0.31082271 1.861604,1.1704417 2.488671,2.1563326 2.488671,2.1563326 l 1.603592,-0.9146606 c 0,0 0.331496,-0.1651245 0.331496,0.2185472 v 5.7388883 c 0,0 0,0.5115624 -0.387017,0.3286302 C 15.288019,8.4368366 11.764034,6.4326776 10.655236,5.801319 10.046132,5.5293491 10.581752,5.309183 10.581752,5.309183 l 1.548071,-0.8855209 c 0,0 -0.883446,-1.1056871 -2.1751379,-1.6917175 C 8.5715456,2.0147859 7.2765874,1.9289859 5.6909576,2.5279672 4.6572773,2.9181146 3.4390694,3.9185752 2.5621557,5.3966019 Z" /> | ||||
| <path fill="#808080" d="m 15.19004,11.792751 c 0,0 -1.456624,2.248608 -3.760768,3.454665 C 9.125128,16.453472 6.2853646,16.107035 4.4237602,14.936593 2.5621557,13.766151 1.935089,12.78026 1.935089,12.78026 l -1.60359274,0.913042 c 0,0 -0.3314962496,0.165125 -0.3314962496,-0.218547 V 7.7342477 c 0,0 0,-0.5115623 0.3870177796,-0.3286303 0.32496429,0.1537925 3.84894891,2.1579516 4.95774671,2.7893106 0.609104,0.271969 0.073485,0.492136 0.073485,0.492136 l -1.5480712,0.88552 c 0,0 0.8834456,1.105688 2.1751379,1.691718 1.3831395,0.720396 2.6780976,0.806196 4.2637278,0.207215 1.033681,-0.390147 2.251888,-1.390608 3.128802,-2.868635 z" /> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 1.4 KiB | 
							
								
								
									
										4
									
								
								resources/images/svg_modifier.svg
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,4 @@ | |||
| <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16px" height="16px"> | ||||
| <path fill="#009688" d="M 3.5 9 C 1.5741139 9 0 10.574114 0 12.5 C 0 14.425886 1.5741139 16 3.5 16 C 5.4258861 16 7 14.425886 7 12.5 C 7 10.574114 5.4258861 9 3.5 9 z M 3.5 10.199219 C 4.7773592 10.199219 5.8007813 11.222641 5.8007812 12.5 C 5.8007812 12.667974 5.7816922 12.830839 5.7480469 12.988281 L 3.0117188 10.251953 C 3.1691607 10.218308 3.3320257 10.199219 3.5 10.199219 z M 2.4863281 10.433594 L 5.5664062 13.513672 C 5.3371871 13.983352 4.9544913 14.362231 4.4804688 14.583984 L 1.4160156 11.521484 C 1.6376653 11.046724 2.0161177 10.663072 2.4863281 10.433594 z M 1.2421875 12.054688 L 3.9472656 14.757812 C 3.8024888 14.786013 3.6534389 14.800781 3.5 14.800781 C 2.2226408 14.800781 1.1992188 13.777359 1.1992188 12.5 C 1.1992188 12.347226 1.2142247 12.198881 1.2421875 12.054688 z "/> | ||||
| <path fill="#808080" d="M 6,0 C 4,0 3,1 3,3 V 8 H 4 V 3 C 4,1.7 4.7,1 6,1 h 7 c 1.3,0 2,0.7 2,2 h 1 C 16,1 15,0 13,0 Z M 6,4 5,5 V 6 L 6,7 H 7 V 8 H 5 V 9 H 7 L 8,8 V 7 L 7,6 H 6 V 5 H 8 V 4 Z m 3,0 v 4 h 1 V 4 Z m 1,4 v 1 h 1 V 8 Z m 1,0 h 1 V 4 H 11 Z M 14.5,4 C 13.5,4 13,4.6 13,5.5 v 2 C 13,8.4 13.7,9 14.5,9 H 16 V 7 H 15 V 8 H 14.5 C 14.3,8 14,7.8 14,7.5 v -2 C 14,5 14.3,5 14.5,5 H 16 V 4 Z m 0.5,6 c 0,1.3 -0.7,2 -2,2 H 8 v 1 h 5 c 2,0 3,-1 3,-3 z" /> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 1.3 KiB | 
							
								
								
									
										4
									
								
								resources/images/svg_negative.svg
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,4 @@ | |||
| <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16px" height="16px"> | ||||
| <path fill="#009688" d="M 1,12 C 0.5,12 0,12 0,12.5 0,13 0.5,13 1,13 H 6 C 6.5,13 7,13 7,12.5 7,12 6.5,12 6,12 Z" /> | ||||
| <path fill="#808080" d="M 6,0 C 4,0 3,1 3,3 V 8 H 4 V 3 C 4,1.7 4.7,1 6,1 h 7 c 1.3,0 2,0.7 2,2 h 1 C 16,1 15,0 13,0 Z M 6,4 5,5 V 6 L 6,7 H 7 V 8 H 5 V 9 H 7 L 8,8 V 7 L 7,6 H 6 V 5 H 8 V 4 Z m 3,0 v 4 h 1 V 4 Z m 1,4 v 1 h 1 V 8 Z m 1,0 h 1 V 4 H 11 Z M 14.5,4 C 13.5,4 13,4.6 13,5.5 v 2 C 13,8.4 13.7,9 14.5,9 H 16 V 7 H 15 V 8 H 14.5 C 14.3,8 14,7.8 14,7.5 v -2 C 14,5 14.3,5 14.5,5 H 16 V 4 Z m 0.5,6 c 0,1.3 -0.7,2 -2,2 H 8 v 1 h 5 c 2,0 3,-1 3,-3 z" /> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 671 B | 
							
								
								
									
										4
									
								
								resources/images/svg_part.svg
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,4 @@ | |||
| <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16px" height="16px"> | ||||
| <path fill="#808080" d="M 6,0 C 4,0 3,1 3,3 V 8 H 4 V 3 C 4,1.7 4.7,1 6,1 h 7 c 1.3,0 2,0.7 2,2 h 1 C 16,1 15,0 13,0 Z M 6,4 5,5 V 6 L 6,7 H 7 V 8 H 5 V 9 H 7 L 8,8 V 7 L 7,6 H 6 V 5 H 8 V 4 Z m 3,0 v 4 h 1 V 4 Z m 1,4 v 1 h 1 V 8 Z m 1,0 h 1 V 4 H 11 Z M 14.5,4 C 13.5,4 13,4.6 13,5.5 v 2 C 13,8.4 13.7,9 14.5,9 H 16 V 7 H 15 V 8 H 14.5 C 14.3,8 14,7.8 14,7.5 v -2 C 14,5 14.3,5 14.5,5 H 16 V 4 Z m 0.5,6 c 0,1.3 -0.7,2 -2,2 H 8 v 1 h 5 c 2,0 3,-1 3,-3 z" /> | ||||
| <path fill="#009688" d="M 3.5,9 C 3,9 3,9.5 3,10 v 2 H 1 C 0.5,12 0,12 0,12.5 0,13 0.5,13 1,13 h 2 v 2 c 0,0.5 0,1 0.5,1 C 4,16 4,15.5 4,15 V 13 H 6 C 6.5,13 7,13 7,12.5 7,12 6.5,12 6,12 H 4 V 10 C 4,9.5 4,9 3.5,9 Z" /> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 774 B | 
|  | @ -1,3 +0,0 @@ | |||
| <svg width="20" height="20" viewBox="0 0 20 20" fill="none" xmlns="http://www.w3.org/2000/svg"> | ||||
| <path d="M11.676 10.236C12.588 10.604 13.044 11.268 13.044 12.228C13.044 13.028 12.744 13.656 12.144 14.112C11.544 14.56 10.72 14.784 9.672 14.784H6V6H9.372C10.316 6 11.092 6.212 11.7 6.636C12.308 7.06 12.612 7.676 12.612 8.484C12.612 9.268 12.3 9.852 11.676 10.236ZM8.004 7.764V9.492H9.204C9.596 9.492 9.908 9.408 10.14 9.24C10.372 9.072 10.488 8.848 10.488 8.568C10.488 8.304 10.376 8.104 10.152 7.968C9.928 7.832 9.612 7.764 9.204 7.764H8.004ZM9.552 13.008C9.984 13.008 10.32 12.932 10.56 12.78C10.8 12.62 10.92 12.396 10.92 12.108C10.92 11.82 10.796 11.596 10.548 11.436C10.308 11.268 9.972 11.184 9.54 11.184H8.004V13.008H9.552Z" fill="#262E30"/> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 755 B | 
|  | @ -1,3 +0,0 @@ | |||
| <svg width="20" height="20" viewBox="0 0 20 20" fill="none" xmlns="http://www.w3.org/2000/svg"> | ||||
| <path d="M12.219 10.0622C13.131 10.4302 13.587 11.0942 13.587 12.0542C13.587 12.8542 13.287 13.4822 12.687 13.9382C12.087 14.3862 11.263 14.6102 10.215 14.6102H6.54297V5.82617H9.91497C10.859 5.82617 11.635 6.03817 12.243 6.46217C12.851 6.88617 13.155 7.50217 13.155 8.31017C13.155 9.09417 12.843 9.67817 12.219 10.0622ZM8.54697 7.59017V9.31817H9.74697C10.139 9.31817 10.451 9.23417 10.683 9.06617C10.915 8.89817 11.031 8.67417 11.031 8.39417C11.031 8.13017 10.919 7.93017 10.695 7.79417C10.471 7.65817 10.155 7.59017 9.74697 7.59017H8.54697ZM10.095 12.8342C10.527 12.8342 10.863 12.7582 11.103 12.6062C11.343 12.4462 11.463 12.2222 11.463 11.9342C11.463 11.6462 11.339 11.4222 11.091 11.2622C10.851 11.0942 10.515 11.0102 10.083 11.0102H8.54697V12.8342H10.095Z" fill="#B6B6B6"/> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 882 B | 
|  | @ -1,4 +0,0 @@ | |||
| <svg width="20" height="20" viewBox="0 0 20 20" fill="none" xmlns="http://www.w3.org/2000/svg"> | ||||
| <path d="M9.29796 6H10.298L7.86755 14.5945H6.86755L9.29796 6Z" fill="#262E30"/> | ||||
| <path d="M6.38243 6H13.3824L13 6.91839H6L6.38243 6Z" fill="#262E30"/> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 253 B | 
|  | @ -1,5 +0,0 @@ | |||
| <svg width="20" height="20" viewBox="0 0 20 20" fill="none" xmlns="http://www.w3.org/2000/svg"> | ||||
| <rect width="20" height="20" rx="1" fill="none"/> | ||||
| <path d="M10.2981 6.10938H12.0005L9.57008 14.7039H7.86768L10.2981 6.10938Z" fill="#B6B6B6"/> | ||||
| <path d="M7.38243 6.10938H14.3824L14 7.5H7L7.38243 6.10938Z" fill="#B6B6B6"/> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 324 B | 
|  | @ -3,16 +3,19 @@ project(imgui) | |||
| 
 | ||||
| add_library(imgui STATIC | ||||
|     imconfig.h | ||||
|     imgui.cpp | ||||
|     imgui.h | ||||
|     imgui_demo.cpp | ||||
|     imgui_draw.cpp | ||||
|     imgui_internal.h | ||||
|     imgui_stdlib.cpp | ||||
|     imgui_stdlib.h | ||||
|     imgui_tables.cpp | ||||
|     imgui_widgets.cpp | ||||
|     # imgui STB | ||||
|     imstb_rectpack.h | ||||
|     imstb_textedit.h | ||||
|     imstb_truetype.h | ||||
|     imgui_tables.cpp | ||||
|     imgui.cpp | ||||
|     imgui_demo.cpp | ||||
|     imgui_draw.cpp | ||||
|     imgui_widgets.cpp | ||||
| ) | ||||
| 
 | ||||
| if(Boost_FOUND) | ||||
|  |  | |||
							
								
								
									
										76
									
								
								src/imgui/imgui_stdlib.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,76 @@ | |||
| // dear imgui: wrappers for C++ standard library (STL) types (std::string, etc.)
 | ||||
| // This is also an example of how you may wrap your own similar types.
 | ||||
| 
 | ||||
| // Compatibility:
 | ||||
| // - std::string support is only guaranteed to work from C++11.
 | ||||
| //   If you try to use it pre-C++11, please share your findings (w/ info about compiler/architecture)
 | ||||
| 
 | ||||
| // Changelog:
 | ||||
| // - v0.10: Initial version. Added InputText() / InputTextMultiline() calls with std::string
 | ||||
| 
 | ||||
| #include "imgui.h" | ||||
| #include "imgui_stdlib.h" | ||||
| 
 | ||||
| struct InputTextCallback_UserData | ||||
| { | ||||
|     std::string*            Str; | ||||
|     ImGuiInputTextCallback  ChainCallback; | ||||
|     void*                   ChainCallbackUserData; | ||||
| }; | ||||
| 
 | ||||
| static int InputTextCallback(ImGuiInputTextCallbackData* data) | ||||
| { | ||||
|     InputTextCallback_UserData* user_data = (InputTextCallback_UserData*)data->UserData; | ||||
|     if (data->EventFlag == ImGuiInputTextFlags_CallbackResize) | ||||
|     { | ||||
|         // Resize string callback
 | ||||
|         // If for some reason we refuse the new length (BufTextLen) and/or capacity (BufSize) we need to set them back to what we want.
 | ||||
|         std::string* str = user_data->Str; | ||||
|         IM_ASSERT(data->Buf == str->c_str()); | ||||
|         str->resize(data->BufTextLen); | ||||
|         data->Buf = (char*)str->c_str(); | ||||
|     } | ||||
|     else if (user_data->ChainCallback) | ||||
|     { | ||||
|         // Forward to user callback, if any
 | ||||
|         data->UserData = user_data->ChainCallbackUserData; | ||||
|         return user_data->ChainCallback(data); | ||||
|     } | ||||
|     return 0; | ||||
| } | ||||
| 
 | ||||
| bool ImGui::InputText(const char* label, std::string* str, ImGuiInputTextFlags flags, ImGuiInputTextCallback callback, void* user_data) | ||||
| { | ||||
|     IM_ASSERT((flags & ImGuiInputTextFlags_CallbackResize) == 0); | ||||
|     flags |= ImGuiInputTextFlags_CallbackResize; | ||||
| 
 | ||||
|     InputTextCallback_UserData cb_user_data; | ||||
|     cb_user_data.Str = str; | ||||
|     cb_user_data.ChainCallback = callback; | ||||
|     cb_user_data.ChainCallbackUserData = user_data; | ||||
|     return InputText(label, (char*)str->c_str(), str->capacity() + 1, flags, InputTextCallback, &cb_user_data); | ||||
| } | ||||
| 
 | ||||
| bool ImGui::InputTextMultiline(const char* label, std::string* str, const ImVec2& size, ImGuiInputTextFlags flags, ImGuiInputTextCallback callback, void* user_data) | ||||
| { | ||||
|     IM_ASSERT((flags & ImGuiInputTextFlags_CallbackResize) == 0); | ||||
|     flags |= ImGuiInputTextFlags_CallbackResize; | ||||
| 
 | ||||
|     InputTextCallback_UserData cb_user_data; | ||||
|     cb_user_data.Str = str; | ||||
|     cb_user_data.ChainCallback = callback; | ||||
|     cb_user_data.ChainCallbackUserData = user_data; | ||||
|     return InputTextMultiline(label, (char*)str->c_str(), str->capacity() + 1, size, flags, InputTextCallback, &cb_user_data); | ||||
| } | ||||
| 
 | ||||
| bool ImGui::InputTextWithHint(const char* label, const char* hint, std::string* str, ImGuiInputTextFlags flags, ImGuiInputTextCallback callback, void* user_data) | ||||
| { | ||||
|     IM_ASSERT((flags & ImGuiInputTextFlags_CallbackResize) == 0); | ||||
|     flags |= ImGuiInputTextFlags_CallbackResize; | ||||
| 
 | ||||
|     InputTextCallback_UserData cb_user_data; | ||||
|     cb_user_data.Str = str; | ||||
|     cb_user_data.ChainCallback = callback; | ||||
|     cb_user_data.ChainCallbackUserData = user_data; | ||||
|     return InputTextWithHint(label, hint, (char*)str->c_str(), str->capacity() + 1, flags, InputTextCallback, &cb_user_data); | ||||
| } | ||||
							
								
								
									
										22
									
								
								src/imgui/imgui_stdlib.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,22 @@ | |||
| // dear imgui: wrappers for C++ standard library (STL) types (std::string, etc.)
 | ||||
| // This is also an example of how you may wrap your own similar types.
 | ||||
| 
 | ||||
| // Compatibility:
 | ||||
| // - std::string support is only guaranteed to work from C++11.
 | ||||
| //   If you try to use it pre-C++11, please share your findings (w/ info about compiler/architecture)
 | ||||
| 
 | ||||
| // Changelog:
 | ||||
| // - v0.10: Initial version. Added InputText() / InputTextMultiline() calls with std::string
 | ||||
| 
 | ||||
| #pragma once | ||||
| 
 | ||||
| #include <string> | ||||
| 
 | ||||
| namespace ImGui | ||||
| { | ||||
|     // ImGui::InputText() with std::string
 | ||||
|     // Because text input needs dynamic resizing, we need to setup a callback to grow the capacity
 | ||||
|     IMGUI_API bool  InputText(const char* label, std::string* str, ImGuiInputTextFlags flags = 0, ImGuiInputTextCallback callback = NULL, void* user_data = NULL); | ||||
|     IMGUI_API bool  InputTextMultiline(const char* label, std::string* str, const ImVec2& size = ImVec2(0, 0), ImGuiInputTextFlags flags = 0, ImGuiInputTextCallback callback = NULL, void* user_data = NULL); | ||||
|     IMGUI_API bool  InputTextWithHint(const char* label, const char* hint, std::string* str, ImGuiInputTextFlags flags = 0, ImGuiInputTextCallback callback = NULL, void* user_data = NULL); | ||||
| } | ||||
|  | @ -4815,8 +4815,9 @@ bool ImGui::InputTextEx(const char* label, const char* hint, char* buf, int buf_ | |||
|         PushStyleColor(ImGuiCol_ChildBg, style.Colors[ImGuiCol_FrameBg]); | ||||
|         PushStyleVar(ImGuiStyleVar_ChildRounding, style.FrameRounding); | ||||
|         PushStyleVar(ImGuiStyleVar_ChildBorderSize, style.FrameBorderSize); | ||||
|         PushStyleVar(ImGuiStyleVar_WindowPadding, ImVec2(0, 0)); // Ensure no clip rect so mouse hover can reach FramePadding edges
 | ||||
|         bool child_visible = BeginChildEx(label, id, frame_bb.GetSize(), true, ImGuiWindowFlags_NoMove); | ||||
|         PopStyleVar(2); | ||||
|         PopStyleVar(3); | ||||
|         PopStyleColor(); | ||||
|         if (!child_visible) | ||||
|         { | ||||
|  | @ -5454,7 +5455,7 @@ bool ImGui::InputTextEx(const char* label, const char* hint, char* buf, int buf_ | |||
|                 // Test if cursor is vertically visible
 | ||||
|                 if (cursor_offset.y - g.FontSize < scroll_y) | ||||
|                     scroll_y = ImMax(0.0f, cursor_offset.y - g.FontSize); | ||||
|                 else if (cursor_offset.y - inner_size.y >= scroll_y) | ||||
|                 else if (cursor_offset.y - (inner_size.y - style.FramePadding.y * 2.0f) >= scroll_y) | ||||
|                     scroll_y = cursor_offset.y - inner_size.y + style.FramePadding.y * 2.0f; | ||||
|                 const float scroll_max_y = ImMax((text_size.y + style.FramePadding.y * 2.0f) - inner_size.y, 0.0f); | ||||
|                 scroll_y = ImClamp(scroll_y, 0.0f, scroll_max_y); | ||||
|  |  | |||
|  | @ -1,3 +1,7 @@ | |||
| ///|/ Copyright (c) Prusa Research 2017 - 2023 Oleksandra Iushchenko @YuSanka, Vojtěch Bubník @bubnikv, Pavel Mikuš @Godrak, David Kocík @kocikdav, Lukáš Matěna @lukasmatena, Enrico Turri @enricoturri1966, Lukáš Hejl @hejllukas, Filip Sykala @Jony01, Vojtěch Král @vojtechkral
 | ||||
| ///|/
 | ||||
| ///|/ PrusaSlicer is released under the terms of the AGPLv3 or higher
 | ||||
| ///|/
 | ||||
| #include "libslic3r/libslic3r.h" | ||||
| #include "libslic3r/Utils.hpp" | ||||
| #include "AppConfig.hpp" | ||||
|  | @ -43,6 +47,7 @@ static const std::string MODELS_STR = "models"; | |||
| 
 | ||||
| const std::string AppConfig::SECTION_FILAMENTS = "filaments"; | ||||
| const std::string AppConfig::SECTION_MATERIALS = "sla_materials"; | ||||
| const std::string AppConfig::SECTION_EMBOSS_STYLE = "font"; | ||||
| 
 | ||||
| std::string AppConfig::get_language_code() | ||||
| { | ||||
|  |  | |||
|  | @ -1,3 +1,7 @@ | |||
| ///|/ Copyright (c) Prusa Research 2017 - 2023 Vojtěch Bubník @bubnikv, David Kocík @kocikdav, Lukáš Matěna @lukasmatena, Filip Sykala @Jony01, Enrico Turri @enricoturri1966, Oleksandra Iushchenko @YuSanka, Vojtěch Král @vojtechkral
 | ||||
| ///|/
 | ||||
| ///|/ PrusaSlicer is released under the terms of the AGPLv3 or higher
 | ||||
| ///|/
 | ||||
| #ifndef slic3r_AppConfig_hpp_ | ||||
| #define slic3r_AppConfig_hpp_ | ||||
| 
 | ||||
|  | @ -287,6 +291,7 @@ public: | |||
| 
 | ||||
| 	static const std::string SECTION_FILAMENTS; | ||||
|     static const std::string SECTION_MATERIALS; | ||||
|     static const std::string SECTION_EMBOSS_STYLE; | ||||
| 
 | ||||
| private: | ||||
| 	template<typename T> | ||||
|  |  | |||
|  | @ -1,3 +1,8 @@ | |||
| ///|/ Copyright (c) Prusa Research 2016 - 2023 Tomáš Mészáros @tamasmeszaros, Vojtěch Bubník @bubnikv, Filip Sykala @Jony01, Enrico Turri @enricoturri1966
 | ||||
| ///|/ Copyright (c) Slic3r 2014 - 2015 Alessandro Ranellucci @alranel
 | ||||
| ///|/
 | ||||
| ///|/ PrusaSlicer is released under the terms of the AGPLv3 or higher
 | ||||
| ///|/
 | ||||
| #ifndef slic3r_BoundingBox_hpp_ | ||||
| #define slic3r_BoundingBox_hpp_ | ||||
| 
 | ||||
|  | @ -222,6 +227,8 @@ public: | |||
|     friend BoundingBox get_extents_rotated(const Points &points, double angle); | ||||
| }; | ||||
| 
 | ||||
| using BoundingBoxes = std::vector<BoundingBox>; | ||||
| 
 | ||||
| class BoundingBox3  : public BoundingBox3Base<Vec3crd>  | ||||
| { | ||||
| public: | ||||
|  |  | |||
|  | @ -66,9 +66,15 @@ set(lisbslic3r_sources | |||
|     EdgeGrid.hpp | ||||
|     ElephantFootCompensation.cpp | ||||
|     ElephantFootCompensation.hpp | ||||
|     Emboss.cpp | ||||
|     Emboss.hpp | ||||
|     EmbossShape.hpp | ||||
|     enum_bitmask.hpp | ||||
|     ExPolygon.cpp | ||||
|     ExPolygon.hpp | ||||
|     ExPolygonSerialize.hpp | ||||
|     ExPolygonsIndex.cpp | ||||
|     ExPolygonsIndex.hpp | ||||
|     Extruder.cpp | ||||
|     Extruder.hpp | ||||
|     ExtrusionEntity.cpp | ||||
|  | @ -224,6 +230,8 @@ set(lisbslic3r_sources | |||
|     MultiPoint.cpp | ||||
|     MultiPoint.hpp | ||||
|     MutablePriorityQueue.hpp | ||||
|     NSVGUtils.cpp | ||||
|     NSVGUtils.hpp | ||||
|     ObjectID.cpp | ||||
|     ObjectID.hpp | ||||
|     PerimeterGenerator.cpp | ||||
|  | @ -318,6 +326,7 @@ set(lisbslic3r_sources | |||
|     Technologies.hpp | ||||
|     Tesselate.cpp | ||||
|     Tesselate.hpp | ||||
|     TextConfiguration.hpp | ||||
|     TriangleMesh.cpp | ||||
|     TriangleMesh.hpp | ||||
|     TriangleMeshSlicer.cpp | ||||
|  | @ -329,6 +338,8 @@ set(lisbslic3r_sources | |||
|     Utils.hpp | ||||
|     Time.cpp | ||||
|     Time.hpp | ||||
|     Timer.cpp | ||||
|     Timer.hpp | ||||
|     Thread.cpp | ||||
|     Thread.hpp | ||||
|     TriangleSelector.cpp | ||||
|  | @ -457,8 +468,13 @@ cmake_policy(SET CMP0011 NEW) | |||
| find_package(CGAL REQUIRED) | ||||
| cmake_policy(POP) | ||||
| 
 | ||||
| add_library(libslic3r_cgal STATIC MeshBoolean.cpp MeshBoolean.hpp  TryCatchSignal.hpp | ||||
|     TryCatchSignal.cpp) | ||||
| add_library(libslic3r_cgal STATIC  | ||||
|     CutSurface.hpp CutSurface.cpp | ||||
|     IntersectionPoints.hpp IntersectionPoints.cpp | ||||
|     MeshBoolean.hpp MeshBoolean.cpp | ||||
|     TryCatchSignal.hpp TryCatchSignal.cpp | ||||
|     Triangulation.hpp Triangulation.cpp | ||||
| ) | ||||
| target_include_directories(libslic3r_cgal PRIVATE ${CMAKE_CURRENT_BINARY_DIR}) | ||||
| 
 | ||||
| # Reset compile options of libslic3r_cgal. Despite it being linked privately, CGAL options | ||||
|  |  | |||
|  | @ -272,8 +272,8 @@ bool has_duplicate_points(const ClipperLib::PolyTree &polytree) | |||
| 
 | ||||
| // Offset CCW contours outside, CW contours (holes) inside.
 | ||||
| // Don't calculate union of the output paths.
 | ||||
| template<typename PathsProvider, ClipperLib::EndType endType = ClipperLib::etClosedPolygon> | ||||
| static ClipperLib::Paths raw_offset(PathsProvider &&paths, float offset, ClipperLib::JoinType joinType, double miterLimit) | ||||
| template<typename PathsProvider> | ||||
| static ClipperLib::Paths raw_offset(PathsProvider &&paths, float offset, ClipperLib::JoinType joinType, double miterLimit, ClipperLib::EndType endType = ClipperLib::etClosedPolygon) | ||||
| { | ||||
|     ClipperLib::ClipperOffset co; | ||||
|     ClipperLib::Paths out; | ||||
|  | @ -354,16 +354,16 @@ TResult clipper_union( | |||
| 
 | ||||
| // Perform union of input polygons using the positive rule, convert to ExPolygons.
 | ||||
| //FIXME is there any benefit of not doing the boolean / using pftEvenOdd?
 | ||||
| ExPolygons ClipperPaths_to_Slic3rExPolygons(const ClipperLib::Paths &input, bool do_union) | ||||
| inline ExPolygons ClipperPaths_to_Slic3rExPolygons(const ClipperLib::Paths &input, bool do_union) | ||||
| { | ||||
|     return PolyTreeToExPolygons(clipper_union<ClipperLib::PolyTree>(input, do_union ? ClipperLib::pftNonZero : ClipperLib::pftEvenOdd)); | ||||
| } | ||||
| 
 | ||||
| template<typename PathsProvider, ClipperLib::EndType endType = ClipperLib::etClosedPolygon> | ||||
| static ClipperLib::Paths raw_offset_polyline(PathsProvider &&paths, float offset, ClipperLib::JoinType joinType, double miterLimit) | ||||
| template<typename PathsProvider> | ||||
| static ClipperLib::Paths raw_offset_polyline(PathsProvider &&paths, float offset, ClipperLib::JoinType joinType, double miterLimit, ClipperLib::EndType end_type = ClipperLib::etOpenButt) | ||||
| { | ||||
|     assert(offset > 0); | ||||
|     return raw_offset<PathsProvider, ClipperLib::etOpenButt>(std::forward<PathsProvider>(paths), offset, joinType, miterLimit); | ||||
|     return raw_offset<PathsProvider>(std::forward<PathsProvider>(paths), offset, joinType, miterLimit, end_type); | ||||
| } | ||||
| 
 | ||||
| template<class TResult, typename PathsProvider> | ||||
|  | @ -418,10 +418,17 @@ Slic3r::Polygons offset(const Slic3r::Polygons &polygons, const float delta, Cli | |||
| Slic3r::ExPolygons offset_ex(const Slic3r::Polygons &polygons, const float delta, ClipperLib::JoinType joinType, double miterLimit) | ||||
|     { return PolyTreeToExPolygons(offset_paths<ClipperLib::PolyTree>(ClipperUtils::PolygonsProvider(polygons), delta, joinType, miterLimit)); } | ||||
| 
 | ||||
| Slic3r::Polygons offset(const Slic3r::Polyline &polyline, const float delta, ClipperLib::JoinType joinType, double miterLimit) | ||||
|     { assert(delta > 0); return to_polygons(clipper_union<ClipperLib::Paths>(raw_offset_polyline(ClipperUtils::SinglePathProvider(polyline.points), delta, joinType, miterLimit))); } | ||||
| Slic3r::Polygons offset(const Slic3r::Polylines &polylines, const float delta, ClipperLib::JoinType joinType, double miterLimit) | ||||
|     { assert(delta > 0); return to_polygons(clipper_union<ClipperLib::Paths>(raw_offset_polyline(ClipperUtils::PolylinesProvider(polylines), delta, joinType, miterLimit))); } | ||||
| Slic3r::Polygons offset(const Slic3r::Polyline &polyline, const float delta, ClipperLib::JoinType joinType, double miterLimit, ClipperLib::EndType end_type) | ||||
|     { assert(delta > 0); return to_polygons(clipper_union<ClipperLib::Paths>(raw_offset_polyline(ClipperUtils::SinglePathProvider(polyline.points), delta, joinType, miterLimit, end_type))); } | ||||
| Slic3r::Polygons offset(const Slic3r::Polylines &polylines, const float delta, ClipperLib::JoinType joinType, double miterLimit, ClipperLib::EndType end_type) | ||||
|     { assert(delta > 0); return to_polygons(clipper_union<ClipperLib::Paths>(raw_offset_polyline(ClipperUtils::PolylinesProvider(polylines), delta, joinType, miterLimit, end_type))); } | ||||
| 
 | ||||
| Polygons contour_to_polygons(const Polygon &polygon, const float line_width, ClipperLib::JoinType join_type, double miter_limit){ | ||||
|     assert(line_width > 1.f); return to_polygons(clipper_union<ClipperLib::Paths>( | ||||
|         raw_offset(ClipperUtils::SinglePathProvider(polygon.points), line_width/2, join_type, miter_limit, ClipperLib::etClosedLine)));} | ||||
| Polygons contour_to_polygons(const Polygons &polygons, const float line_width, ClipperLib::JoinType join_type, double miter_limit){ | ||||
|     assert(line_width > 1.f); return to_polygons(clipper_union<ClipperLib::Paths>( | ||||
|         raw_offset(ClipperUtils::PolygonsProvider(polygons), line_width/2, join_type, miter_limit, ClipperLib::etClosedLine)));} | ||||
| 
 | ||||
| // returns number of expolygons collected (0 or 1).
 | ||||
| static int offset_expolygon_inner(const Slic3r::ExPolygon &expoly, const float delta, ClipperLib::JoinType joinType, double miterLimit, ClipperLib::Paths &out) | ||||
|  | @ -795,6 +802,8 @@ Slic3r::ExPolygons union_ex(const Slic3r::Polygons &subject, ClipperLib::PolyFil | |||
|     { return _clipper_ex(ClipperLib::ctUnion, ClipperUtils::PolygonsProvider(subject), ClipperUtils::EmptyPathsProvider(), ApplySafetyOffset::No, fill_type); } | ||||
| Slic3r::ExPolygons union_ex(const Slic3r::ExPolygons &subject) | ||||
|     { return PolyTreeToExPolygons(clipper_do_polytree(ClipperLib::ctUnion, ClipperUtils::ExPolygonsProvider(subject), ClipperUtils::EmptyPathsProvider(), ClipperLib::pftNonZero)); } | ||||
| Slic3r::ExPolygons union_ex(const Slic3r::ExPolygons &subject, const Slic3r::Polygons &subject2) | ||||
|     { return PolyTreeToExPolygons(clipper_do_polytree(ClipperLib::ctUnion, ClipperUtils::ExPolygonsProvider(subject), ClipperUtils::PolygonsProvider(subject2), ClipperLib::pftNonZero)); } | ||||
| Slic3r::ExPolygons union_ex(const Slic3r::Surfaces &subject) | ||||
|     { return PolyTreeToExPolygons(clipper_do_polytree(ClipperLib::ctUnion, ClipperUtils::SurfacesProvider(subject), ClipperUtils::EmptyPathsProvider(), ClipperLib::pftNonZero)); } | ||||
| // BBS
 | ||||
|  |  | |||
|  | @ -22,6 +22,9 @@ namespace Slic3r { | |||
| static constexpr const float                        ClipperSafetyOffset     = 10.f; | ||||
| 
 | ||||
| static constexpr const Slic3r::ClipperLib::JoinType DefaultJoinType         = Slic3r::ClipperLib::jtMiter; | ||||
| 
 | ||||
| static constexpr const Slic3r::ClipperLib::EndType DefaultEndType           = Slic3r::ClipperLib::etOpenButt; | ||||
| 
 | ||||
| //FIXME evaluate the default miter limit. 3 seems to be extreme, Cura uses 1.2.
 | ||||
| // Mitter Limit 3 is useful for perimeter generator, where sharp corners are extruded without needing a gap fill.
 | ||||
| // However such a high limit causes issues with large positive or negative offsets, where a sharp corner
 | ||||
|  | @ -335,8 +338,8 @@ Slic3r::Polygons offset(const Slic3r::Polygon &polygon, const float delta, Clipp | |||
| // offset Polylines
 | ||||
| // Wherever applicable, please use the expand() / shrink() variants instead, they convey their purpose better.
 | ||||
| // Input polygons for negative offset shall be "normalized": There must be no overlap / intersections between the input polygons.
 | ||||
| Slic3r::Polygons   offset(const Slic3r::Polyline &polyline, const float delta, ClipperLib::JoinType joinType = DefaultLineJoinType, double miterLimit = DefaultLineMiterLimit); | ||||
| Slic3r::Polygons   offset(const Slic3r::Polylines &polylines, const float delta, ClipperLib::JoinType joinType = DefaultLineJoinType, double miterLimit = DefaultLineMiterLimit); | ||||
| Slic3r::Polygons   offset(const Slic3r::Polyline &polyline, const float delta, ClipperLib::JoinType joinType = DefaultLineJoinType, double miterLimit = DefaultLineMiterLimit, ClipperLib::EndType end_type = DefaultEndType); | ||||
| Slic3r::Polygons   offset(const Slic3r::Polylines &polylines, const float delta, ClipperLib::JoinType joinType = DefaultLineJoinType, double miterLimit = DefaultLineMiterLimit, ClipperLib::EndType end_type = DefaultEndType); | ||||
| Slic3r::Polygons   offset(const Slic3r::Polygons &polygons, const float delta, ClipperLib::JoinType joinType = DefaultJoinType, double miterLimit = DefaultMiterLimit); | ||||
| Slic3r::Polygons   offset(const Slic3r::ExPolygon &expolygon, const float delta, ClipperLib::JoinType joinType = DefaultJoinType, double miterLimit = DefaultMiterLimit); | ||||
| Slic3r::Polygons   offset(const Slic3r::ExPolygons &expolygons, const float delta, ClipperLib::JoinType joinType = DefaultJoinType, double miterLimit = DefaultMiterLimit); | ||||
|  | @ -355,6 +358,10 @@ inline Slic3r::ExPolygons offset_ex(const Slic3r::Polygon &polygon, const float | |||
|     return offset_ex(temp, delta, joinType, miterLimit); | ||||
| } | ||||
| 
 | ||||
| // convert stroke to path by offsetting of contour
 | ||||
| Polygons contour_to_polygons(const Polygon &polygon, const float line_width, ClipperLib::JoinType join_type = DefaultJoinType, double miter_limit = DefaultMiterLimit); | ||||
| Polygons contour_to_polygons(const Polygons &polygon, const float line_width, ClipperLib::JoinType join_type = DefaultJoinType, double miter_limit = DefaultMiterLimit); | ||||
| 
 | ||||
| inline Slic3r::Polygons   union_safety_offset   (const Slic3r::Polygons   &polygons)   { return offset   (polygons,   ClipperSafetyOffset); } | ||||
| inline Slic3r::Polygons   union_safety_offset   (const Slic3r::ExPolygons &expolygons) { return offset   (expolygons, ClipperSafetyOffset); } | ||||
| inline Slic3r::ExPolygons union_safety_offset_ex(const Slic3r::Polygons   &polygons)   { return offset_ex(polygons,   ClipperSafetyOffset); } | ||||
|  | @ -539,6 +546,7 @@ Slic3r::Polygons union_(const Slic3r::Polygons &subject, const Slic3r::Polygons | |||
| // May be used to "heal" unusual models (3DLabPrints etc.) by providing fill_type (pftEvenOdd, pftNonZero, pftPositive, pftNegative).
 | ||||
| Slic3r::ExPolygons union_ex(const Slic3r::Polygons &subject, ClipperLib::PolyFillType fill_type = ClipperLib::pftNonZero); | ||||
| Slic3r::ExPolygons union_ex(const Slic3r::ExPolygons &subject); | ||||
| Slic3r::ExPolygons union_ex(const Slic3r::ExPolygons &subject, const Slic3r::Polygons &subject2); | ||||
| Slic3r::ExPolygons union_ex(const Slic3r::Surfaces &subject); | ||||
| 
 | ||||
| Slic3r::ExPolygons union_ex(const Slic3r::ExPolygons& poly1, const Slic3r::ExPolygons& poly2, bool safety_offset_ = false); | ||||
|  |  | |||
|  | @ -61,6 +61,7 @@ public: | |||
| 	static const ColorRGB REDISH()      { return { 1.0f, 0.5f, 0.5f }; } | ||||
| 	static const ColorRGB YELLOW()      { return { 1.0f, 1.0f, 0.0f }; } | ||||
| 	static const ColorRGB WHITE()       { return { 1.0f, 1.0f, 1.0f }; } | ||||
|     static const ColorRGB ORCA()		{ return {0.0f, 150.f / 255.0f, 136.0f / 255}; } | ||||
| 
 | ||||
| 	static const ColorRGB X()           { return { 0.75f, 0.0f, 0.0f }; } | ||||
| 	static const ColorRGB Y()           { return { 0.0f, 0.75f, 0.0f }; } | ||||
|  |  | |||
							
								
								
									
										4086
									
								
								src/libslic3r/CutSurface.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										78
									
								
								src/libslic3r/CutSurface.hpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,78 @@ | |||
| ///|/ Copyright (c) Prusa Research 2022 Filip Sykala @Jony01
 | ||||
| ///|/
 | ||||
| ///|/ PrusaSlicer is released under the terms of the AGPLv3 or higher
 | ||||
| ///|/
 | ||||
| #ifndef slic3r_CutSurface_hpp_ | ||||
| #define slic3r_CutSurface_hpp_ | ||||
| 
 | ||||
| #include <vector> | ||||
| #include <admesh/stl.h> // indexed_triangle_set
 | ||||
| #include "ExPolygon.hpp" | ||||
| #include "Emboss.hpp" // IProjection
 | ||||
| 
 | ||||
| namespace Slic3r{ | ||||
| 
 | ||||
| /// <summary>
 | ||||
| /// Represents cutted surface from object
 | ||||
| /// Extend index triangle set by outlines
 | ||||
| /// </summary>
 | ||||
| struct SurfaceCut : public indexed_triangle_set | ||||
| { | ||||
|     // vertex indices(index to mesh vertices)
 | ||||
|     using Index = unsigned int; | ||||
|     using Contour = std::vector<Index>; | ||||
|     using Contours = std::vector<Contour>; | ||||
|     // list of circulated open surface
 | ||||
|     Contours contours; | ||||
| }; | ||||
| 
 | ||||
| /// <summary>
 | ||||
| /// Cut surface shape from models.
 | ||||
| /// </summary>
 | ||||
| /// <param name="shapes">Multiple shape to cut from model</param>
 | ||||
| /// <param name="models">Multi mesh to cut, need to be in same coordinate system</param>
 | ||||
| /// <param name="projection">Define transformation 2d shape into 3d</param>
 | ||||
| /// <param name="projection_ratio">Define ideal ratio between front and back projection to cut
 | ||||
| /// 0 .. means use closest to front projection
 | ||||
| /// 1 .. means use closest to back projection
 | ||||
| /// value from <0, 1>
 | ||||
| /// </param>
 | ||||
| /// <returns>Cutted surface from model</returns>
 | ||||
| SurfaceCut cut_surface(const ExPolygons                        &shapes, | ||||
|                        const std::vector<indexed_triangle_set> &models, | ||||
|                        const Emboss::IProjection               &projection, | ||||
|                        float projection_ratio); | ||||
| 
 | ||||
| /// <summary>
 | ||||
| /// Create model from surface cuts by projection
 | ||||
| /// </summary>
 | ||||
| /// <param name="cut">Surface from model with outlines</param>
 | ||||
| /// <param name="projection">Way of emboss</param>
 | ||||
| /// <returns>Mesh</returns>
 | ||||
| indexed_triangle_set cut2model(const SurfaceCut         &cut, | ||||
|                                const Emboss::IProject3d &projection); | ||||
| 
 | ||||
| /// <summary>
 | ||||
| /// Separate (A)rea (o)f (I)nterest .. AoI from model
 | ||||
| /// NOTE: Only 2d filtration, do not filtrate by Z coordinate
 | ||||
| /// </summary>
 | ||||
| /// <param name="its">Input model</param>
 | ||||
| /// <param name="bb">Bounding box to project into space</param>
 | ||||
| /// <param name="projection">Define tranformation of BB into space</param>
 | ||||
| /// <returns>Triangles lay at least partialy inside of projected Bounding box</returns>
 | ||||
| indexed_triangle_set its_cut_AoI(const indexed_triangle_set &its, | ||||
|                                  const BoundingBox          &bb, | ||||
|                                  const Emboss::IProjection  &projection); | ||||
| 
 | ||||
| /// <summary>
 | ||||
| /// Separate triangles by mask
 | ||||
| /// </summary>
 | ||||
| /// <param name="its">Input model</param>
 | ||||
| /// <param name="mask">Mask - same size as its::indices</param>
 | ||||
| /// <returns>Copy of indices by mask(with their vertices)</returns>
 | ||||
| indexed_triangle_set its_mask(const indexed_triangle_set &its, const std::vector<bool> &mask); | ||||
| 
 | ||||
| bool corefine_test(const std::string &model_path, const std::string &shape_path); | ||||
| 
 | ||||
| } // namespace Slic3r
 | ||||
| #endif // slic3r_CutSurface_hpp_
 | ||||
|  | @ -217,7 +217,7 @@ static void reset_instance_transformation(ModelObject* object, size_t src_instan | |||
|         auto& obj_instance = object->instances[i]; | ||||
|         const double rot_z = obj_instance->get_rotation().z(); | ||||
|          | ||||
|         Transformation inst_trafo = Transformation(obj_instance->get_transformation().get_matrix(false, false, true)); | ||||
|         Transformation inst_trafo = Transformation(obj_instance->get_transformation().get_matrix_no_scaling_factor()); | ||||
|         // add respect to mirroring
 | ||||
|         if (obj_instance->is_left_handed()) | ||||
|             inst_trafo = inst_trafo * Transformation(scale_transform(Vec3d(-1, 1, 1))); | ||||
|  | @ -320,7 +320,7 @@ const ModelObjectPtrs& Cut::perform_with_plane() | |||
|     // except for translation and Z-rotation on instances, which are preserved
 | ||||
|     // in the transformation matrix and not applied to the mesh transform.
 | ||||
| 
 | ||||
|     const auto              instance_matrix = mo->instances[m_instance]->get_transformation().get_matrix(true); | ||||
|     const auto              instance_matrix = mo->instances[m_instance]->get_transformation().get_matrix_no_offset(); | ||||
|     const Transformation    cut_transformation = Transformation(m_cut_matrix); | ||||
|     const Transform3d       inverse_cut_matrix = cut_transformation.get_rotation_matrix().inverse() * translation_transform(-1. * cut_transformation.get_offset()); | ||||
| 
 | ||||
|  |  | |||
							
								
								
									
										2185
									
								
								src/libslic3r/Emboss.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										477
									
								
								src/libslic3r/Emboss.hpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,477 @@ | |||
| ///|/ Copyright (c) Prusa Research 2021 - 2022 Filip Sykala @Jony01
 | ||||
| ///|/
 | ||||
| ///|/ PrusaSlicer is released under the terms of the AGPLv3 or higher
 | ||||
| ///|/
 | ||||
| #ifndef slic3r_Emboss_hpp_ | ||||
| #define slic3r_Emboss_hpp_ | ||||
| 
 | ||||
| #include <vector> | ||||
| #include <set> | ||||
| #include <optional> | ||||
| #include <memory> | ||||
| #include <admesh/stl.h> // indexed_triangle_set
 | ||||
| #include "Polygon.hpp" | ||||
| #include "ExPolygon.hpp" | ||||
| #include "EmbossShape.hpp" // ExPolygonsWithIds
 | ||||
| #include "BoundingBox.hpp" | ||||
| #include "TextConfiguration.hpp" | ||||
| 
 | ||||
| namespace Slic3r { | ||||
| 
 | ||||
| /// <summary>
 | ||||
| /// class with only static function add ability to engraved OR raised
 | ||||
| /// text OR polygons onto model surface
 | ||||
| /// </summary>
 | ||||
| namespace Emboss | ||||
| {     | ||||
|     static const float UNION_DELTA = 50.0f; // [approx in nano meters depends on volume scale]
 | ||||
|     static const unsigned UNION_MAX_ITERATIN = 10; // [count]
 | ||||
| 
 | ||||
|     /// <summary>
 | ||||
|     /// Collect fonts registred inside OS
 | ||||
|     /// </summary>
 | ||||
|     /// <returns>OS registred TTF font files(full path) with names</returns>
 | ||||
|     EmbossStyles get_font_list(); | ||||
| #ifdef _WIN32 | ||||
|     EmbossStyles get_font_list_by_register(); | ||||
|     EmbossStyles get_font_list_by_enumeration(); | ||||
|     EmbossStyles get_font_list_by_folder(); | ||||
| #endif | ||||
| 
 | ||||
|     /// <summary>
 | ||||
|     /// OS dependent function to get location of font by its name descriptor
 | ||||
|     /// </summary>
 | ||||
|     /// <param name="font_face_name">Unique identificator for font</param>
 | ||||
|     /// <returns>File path to font when found</returns>
 | ||||
|     std::optional<std::wstring> get_font_path(const std::wstring &font_face_name); | ||||
| 
 | ||||
|     // description of one letter
 | ||||
|     struct Glyph | ||||
|     { | ||||
|         // NOTE: shape is scaled by SHAPE_SCALE 
 | ||||
|         // to be able store points without floating points
 | ||||
|         ExPolygons shape; | ||||
| 
 | ||||
|         // values are in font points
 | ||||
|         int advance_width=0, left_side_bearing=0; | ||||
|     }; | ||||
|     // cache for glyph by unicode
 | ||||
|     using Glyphs = std::map<int, Glyph>; | ||||
|          | ||||
|     /// <summary>
 | ||||
|     /// keep information from file about font 
 | ||||
|     /// (store file data itself)
 | ||||
|     /// + cache data readed from buffer
 | ||||
|     /// </summary>
 | ||||
|     struct FontFile | ||||
|     { | ||||
|         // loaded data from font file
 | ||||
|         // must store data size for imgui rasterization
 | ||||
|         // To not store data on heap and To prevent unneccesary copy
 | ||||
|         // data are stored inside unique_ptr
 | ||||
|         std::unique_ptr<std::vector<unsigned char>> data; | ||||
| 
 | ||||
|         struct Info | ||||
|         { | ||||
|             // vertical position is "scale*(ascent - descent + lineGap)"
 | ||||
|             int ascent, descent, linegap; | ||||
| 
 | ||||
|             // for convert font units to pixel
 | ||||
|             int unit_per_em; | ||||
|         }; | ||||
|         // info for each font in data
 | ||||
|         std::vector<Info> infos; | ||||
| 
 | ||||
|         FontFile(std::unique_ptr<std::vector<unsigned char>> data, | ||||
|                  std::vector<Info>                         &&infos) | ||||
|             : data(std::move(data)), infos(std::move(infos)) | ||||
|         { | ||||
|             assert(this->data != nullptr); | ||||
|             assert(!this->data->empty()); | ||||
|         } | ||||
| 
 | ||||
|         bool operator==(const FontFile &other) const { | ||||
|             if (data->size() != other.data->size()) | ||||
|                 return false; | ||||
|             //if(*data != *other.data) return false;
 | ||||
|             for (size_t i = 0; i < infos.size(); i++)  | ||||
|                 if (infos[i].ascent != other.infos[i].ascent || | ||||
|                     infos[i].descent == other.infos[i].descent || | ||||
|                     infos[i].linegap == other.infos[i].linegap) | ||||
|                     return false; | ||||
|             return true; | ||||
|         } | ||||
|     }; | ||||
| 
 | ||||
|     /// <summary>
 | ||||
|     /// Add caching for shape of glyphs
 | ||||
|     /// </summary>
 | ||||
|     struct FontFileWithCache | ||||
|     { | ||||
|         // Pointer on data of the font file
 | ||||
|         std::shared_ptr<const FontFile> font_file; | ||||
| 
 | ||||
|         // Cache for glyph shape
 | ||||
|         // IMPORTANT: accessible only in plater job thread !!!
 | ||||
|         // main thread only clear cache by set to another shared_ptr
 | ||||
|         std::shared_ptr<Emboss::Glyphs> cache; | ||||
| 
 | ||||
|         FontFileWithCache() : font_file(nullptr), cache(nullptr) {} | ||||
|         explicit FontFileWithCache(std::unique_ptr<FontFile> font_file) | ||||
|             : font_file(std::move(font_file)) | ||||
|             , cache(std::make_shared<Emboss::Glyphs>()) | ||||
|         {} | ||||
|         bool has_value() const { return font_file != nullptr && cache != nullptr; } | ||||
|     }; | ||||
| 
 | ||||
|     /// <summary>
 | ||||
|     /// Load font file into buffer
 | ||||
|     /// </summary>
 | ||||
|     /// <param name="file_path">Location of .ttf or .ttc font file</param>
 | ||||
|     /// <returns>Font object when loaded.</returns>
 | ||||
|     std::unique_ptr<FontFile> create_font_file(const char *file_path); | ||||
|     // data = raw file data
 | ||||
|     std::unique_ptr<FontFile> create_font_file(std::unique_ptr<std::vector<unsigned char>> data); | ||||
| #ifdef _WIN32 | ||||
|     // fix for unknown pointer HFONT is replaced with "void *"
 | ||||
|     void * can_load(void* hfont); | ||||
|     std::unique_ptr<FontFile> create_font_file(void * hfont); | ||||
| #endif // _WIN32
 | ||||
| 
 | ||||
|     /// <summary>
 | ||||
|     /// convert letter into polygons
 | ||||
|     /// </summary>
 | ||||
|     /// <param name="font">Define fonts</param>
 | ||||
|     /// <param name="font_index">Index of font in collection</param>
 | ||||
|     /// <param name="letter">One character defined by unicode codepoint</param>
 | ||||
|     /// <param name="flatness">Precision of lettter outline curve in conversion to lines</param>
 | ||||
|     /// <returns>inner polygon cw(outer ccw)</returns>
 | ||||
|     std::optional<Glyph> letter2glyph(const FontFile &font, unsigned int font_index, int letter, float flatness); | ||||
| 
 | ||||
|     /// <summary>
 | ||||
|     /// Convert text into polygons
 | ||||
|     /// </summary>
 | ||||
|     /// <param name="font">Define fonts + cache, which could extend</param>
 | ||||
|     /// <param name="text">Characters to convert</param>
 | ||||
|     /// <param name="font_prop">User defined property of the font</param>
 | ||||
|     /// <param name="was_canceled">Way to interupt processing</param>
 | ||||
|     /// <returns>Inner polygon cw(outer ccw)</returns>
 | ||||
|     HealedExPolygons  text2shapes (FontFileWithCache &font, const char *text,         const FontProp &font_prop, const std::function<bool()> &was_canceled = []() {return false;}); | ||||
|     ExPolygonsWithIds text2vshapes(FontFileWithCache &font, const std::wstring& text, const FontProp &font_prop, const std::function<bool()>& was_canceled = []() {return false;}); | ||||
| 
 | ||||
|     const unsigned ENTER_UNICODE = static_cast<unsigned>('\n'); | ||||
|     /// Sum of character '\n'
 | ||||
|     unsigned get_count_lines(const std::wstring &ws); | ||||
|     unsigned get_count_lines(const std::string &text); | ||||
|     unsigned get_count_lines(const ExPolygonsWithIds &shape); | ||||
| 
 | ||||
|     /// <summary>
 | ||||
|     /// Fix duplicit points and self intersections in polygons.
 | ||||
|     /// Also try to reduce amount of points and remove useless polygon parts
 | ||||
|     /// </summary>
 | ||||
|     /// <param name="is_non_zero">Fill type ClipperLib::pftNonZero for overlapping otherwise </param>
 | ||||
|     /// <param name="max_iteration">Look at heal_expolygon()::max_iteration</param>
 | ||||
|     /// <returns>Healed shapes with flag is fully healed</returns>
 | ||||
|     HealedExPolygons heal_polygons(const Polygons &shape, bool is_non_zero = true, unsigned max_iteration = 10); | ||||
| 
 | ||||
|     /// <summary>
 | ||||
|     /// NOTE: call Slic3r::union_ex before this call
 | ||||
|     /// 
 | ||||
|     /// Heal (read: Fix) issues in expolygons:
 | ||||
|     ///  - self intersections
 | ||||
|     ///  - duplicit points
 | ||||
|     ///  - points close to line segments
 | ||||
|     /// </summary>
 | ||||
|     /// <param name="shape">In/Out shape to heal</param>
 | ||||
|     /// <param name="max_iteration">Heal could create another issue,
 | ||||
|     /// After healing it is checked again until shape is good or maximal count of iteration</param>
 | ||||
|     /// <returns>True when shapes is good otherwise False</returns>
 | ||||
|     bool heal_expolygons(ExPolygons &shape, unsigned max_iteration = 10); | ||||
| 
 | ||||
|     /// <summary>
 | ||||
|     /// Divide line segments in place near to point
 | ||||
|     /// (which could lead to self intersection due to preccision)
 | ||||
|     /// Remove same neighbors
 | ||||
|     /// Note: Possible part of heal shape
 | ||||
|     /// </summary>
 | ||||
|     /// <param name="expolygons">Expolygon to edit</param>
 | ||||
|     /// <param name="distance">(epsilon)Euclidean distance from point to line which divide line</param>
 | ||||
|     /// <returns>True when some division was made otherwise false</returns>
 | ||||
|     bool divide_segments_for_close_point(ExPolygons &expolygons, double distance); | ||||
| 
 | ||||
|     /// <summary>
 | ||||
|     /// Use data from font property to modify transformation
 | ||||
|     /// </summary>
 | ||||
|     /// <param name="angle">Z-rotation as angle to Y axis</param>
 | ||||
|     /// <param name="distance">Z-move as surface distance</param>
 | ||||
|     /// <param name="transformation">In / Out transformation to modify by property</param>
 | ||||
|     void apply_transformation(const std::optional<float> &angle, const std::optional<float> &distance, Transform3d &transformation); | ||||
| 
 | ||||
|     /// <summary>
 | ||||
|     /// Read information from naming table of font file
 | ||||
|     /// search for italic (or oblique), bold italic (or bold oblique)
 | ||||
|     /// </summary>
 | ||||
|     /// <param name="font">Selector of font</param>
 | ||||
|     /// <param name="font_index">Index of font in collection</param>
 | ||||
|     /// <returns>True when the font description contains italic/obligue otherwise False</returns>
 | ||||
|     bool is_italic(const FontFile &font, unsigned int font_index); | ||||
| 
 | ||||
|     /// <summary>
 | ||||
|     /// Create unique character set from string with filtered from text with only character from font
 | ||||
|     /// </summary>
 | ||||
|     /// <param name="text">Source vector of glyphs</param>
 | ||||
|     /// <param name="font">Font descriptor</param>
 | ||||
|     /// <param name="font_index">Define font in collection</param>
 | ||||
|     /// <param name="exist_unknown">True when text contain glyph unknown in font</param>
 | ||||
|     /// <returns>Unique set of character from text contained in font</returns>
 | ||||
|     std::string create_range_text(const std::string &text, const FontFile &font, unsigned int font_index, bool* exist_unknown = nullptr);     | ||||
| 
 | ||||
|     /// <summary>
 | ||||
|     /// Calculate scale for glyph shape convert from shape points to mm
 | ||||
|     /// </summary>
 | ||||
|     /// <param name="fp">Property of font</param>
 | ||||
|     /// <param name="ff">Font data</param>
 | ||||
|     /// <returns>Conversion to mm</returns>
 | ||||
|     double get_text_shape_scale(const FontProp &fp, const FontFile &ff); | ||||
| 
 | ||||
|     /// <summary>
 | ||||
|     /// getter of font info by collection defined in prop
 | ||||
|     /// </summary>
 | ||||
|     /// <param name="font">Contain infos about all fonts(collections) in file</param>
 | ||||
|     /// <param name="prop">Index of collection</param>
 | ||||
|     /// <returns>Ascent, descent, line gap</returns>
 | ||||
|     const FontFile::Info &get_font_info(const FontFile &font, const FontProp &prop); | ||||
| 
 | ||||
|     /// <summary>
 | ||||
|     /// Read from font file and properties height of line with spacing
 | ||||
|     /// </summary>
 | ||||
|     /// <param name="font">Infos for collections</param>
 | ||||
|     /// <param name="prop">Collection index + Additional line gap</param>
 | ||||
|     /// <returns>Line height with spacing in scaled font points (same as ExPolygons)</returns>
 | ||||
|     int get_line_height(const FontFile &font, const FontProp &prop); | ||||
| 
 | ||||
|     /// <summary>
 | ||||
|     /// Calculate Vertical align
 | ||||
|     /// </summary>
 | ||||
|     /// <param name="align">Top | Center | Bottom</param>
 | ||||
|     /// <param name="count_lines"></param>
 | ||||
|     /// <returns>Return align Y offset in mm</returns>
 | ||||
|     double get_align_y_offset_in_mm(FontProp::VerticalAlign align, unsigned count_lines, const FontFile &ff, const FontProp &fp); | ||||
| 
 | ||||
|     /// <summary>
 | ||||
|     /// Project spatial point
 | ||||
|     /// </summary>
 | ||||
|     class IProject3d | ||||
|     { | ||||
|     public: | ||||
|         virtual ~IProject3d() = default; | ||||
|         /// <summary>
 | ||||
|         /// Move point with respect to projection direction
 | ||||
|         /// e.g. Orthogonal projection will move with point by direction
 | ||||
|         /// e.g. Spherical projection need to use center of projection
 | ||||
|         /// </summary>
 | ||||
|         /// <param name="point">Spatial point coordinate</param>
 | ||||
|         /// <returns>Projected spatial point</returns>
 | ||||
|         virtual Vec3d project(const Vec3d &point) const = 0; | ||||
|     }; | ||||
| 
 | ||||
|     /// <summary>
 | ||||
|     /// Project 2d point into space
 | ||||
|     /// Could be plane, sphere, cylindric, ...
 | ||||
|     /// </summary>
 | ||||
|     class IProjection : public IProject3d | ||||
|     { | ||||
|     public: | ||||
|         virtual ~IProjection() = default; | ||||
| 
 | ||||
|         /// <summary>
 | ||||
|         /// convert 2d point to 3d points
 | ||||
|         /// </summary>
 | ||||
|         /// <param name="p">2d coordinate</param>
 | ||||
|         /// <returns>
 | ||||
|         /// first - front spatial point
 | ||||
|         /// second - back spatial point
 | ||||
|         /// </returns>
 | ||||
|         virtual std::pair<Vec3d, Vec3d> create_front_back(const Point &p) const = 0; | ||||
| 
 | ||||
|         /// <summary>
 | ||||
|         /// Back projection
 | ||||
|         /// </summary>
 | ||||
|         /// <param name="p">Point to project</param>
 | ||||
|         /// <param name="depth">[optional] Depth of 2d projected point. Be careful number is in 2d scale</param>
 | ||||
|         /// <returns>Uprojected point when it is possible</returns>
 | ||||
|         virtual std::optional<Vec2d> unproject(const Vec3d &p, double * depth = nullptr) const = 0; | ||||
|     }; | ||||
| 
 | ||||
|     /// <summary>
 | ||||
|     /// Create triangle model for text
 | ||||
|     /// </summary>
 | ||||
|     /// <param name="shape2d">text or image</param>
 | ||||
|     /// <param name="projection">Define transformation from 2d to 3d(orientation, position, scale, ...)</param>
 | ||||
|     /// <returns>Projected shape into space</returns>
 | ||||
|     indexed_triangle_set polygons2model(const ExPolygons &shape2d, const IProjection& projection); | ||||
|      | ||||
|     /// <summary>
 | ||||
|     /// Suggest wanted up vector of embossed text by emboss direction
 | ||||
|     /// </summary>
 | ||||
|     /// <param name="normal">Normalized vector of emboss direction in world</param>
 | ||||
|     /// <param name="up_limit">Is compared with normal.z to suggest up direction</param>
 | ||||
|     /// <returns>Wanted up vector</returns>
 | ||||
|     Vec3d suggest_up(const Vec3d normal, double up_limit = 0.9); | ||||
|          | ||||
|     /// <summary>
 | ||||
|     /// By transformation calculate angle between suggested and actual up vector
 | ||||
|     /// </summary>
 | ||||
|     /// <param name="tr">Transformation of embossed volume in world</param>
 | ||||
|     /// <param name="up_limit">Is compared with normal.z to suggest up direction</param>
 | ||||
|     /// <returns>Rotation of suggested up-vector[in rad] in the range [-Pi, Pi], When rotation is not zero</returns>
 | ||||
|     std::optional<float> calc_up(const Transform3d &tr, double up_limit = 0.9); | ||||
| 
 | ||||
|     /// <summary>
 | ||||
|     /// Create transformation for emboss text object to lay on surface point
 | ||||
|     /// </summary>
 | ||||
|     /// <param name="position">Position of surface point</param>
 | ||||
|     /// <param name="normal">Normal of surface point</param>
 | ||||
|     /// <param name="up_limit">Is compared with normal.z to suggest up direction</param>
 | ||||
|     /// <returns>Transformation onto surface point</returns>
 | ||||
|     Transform3d create_transformation_onto_surface( | ||||
|         const Vec3d &position, const Vec3d &normal, double up_limit = 0.9); | ||||
| 
 | ||||
|     class ProjectZ : public IProjection | ||||
|     { | ||||
|     public: | ||||
|         explicit ProjectZ(double depth) : m_depth(depth) {} | ||||
|         // Inherited via IProject
 | ||||
|         std::pair<Vec3d, Vec3d> create_front_back(const Point &p) const override; | ||||
|         Vec3d project(const Vec3d &point) const override; | ||||
|         std::optional<Vec2d> unproject(const Vec3d &p, double * depth = nullptr) const override; | ||||
|         double m_depth; | ||||
|     }; | ||||
| 
 | ||||
|     class ProjectScale : public IProjection | ||||
|     { | ||||
|         std::unique_ptr<IProjection> core; | ||||
|         double m_scale; | ||||
|     public: | ||||
|         ProjectScale(std::unique_ptr<IProjection> core, double scale) | ||||
|             : core(std::move(core)), m_scale(scale) | ||||
|         {} | ||||
| 
 | ||||
|         // Inherited via IProject
 | ||||
|         std::pair<Vec3d, Vec3d> create_front_back(const Point &p) const override | ||||
|         { | ||||
|             auto res = core->create_front_back(p); | ||||
|             return std::make_pair(res.first * m_scale, res.second * m_scale); | ||||
|         } | ||||
|         Vec3d project(const Vec3d &point) const override{ | ||||
|             return core->project(point); | ||||
|         } | ||||
|         std::optional<Vec2d> unproject(const Vec3d &p, double *depth = nullptr) const override { | ||||
|             auto res = core->unproject(p / m_scale, depth); | ||||
|             if (depth != nullptr) *depth *= m_scale; | ||||
|             return res; | ||||
|         } | ||||
|     }; | ||||
| 
 | ||||
|     class ProjectTransform : public IProjection | ||||
|     { | ||||
|         std::unique_ptr<IProjection> m_core; | ||||
|         Transform3d m_tr; | ||||
|         Transform3d m_tr_inv; | ||||
|         double z_scale; | ||||
|     public: | ||||
|         ProjectTransform(std::unique_ptr<IProjection> core, const Transform3d &tr) : m_core(std::move(core)), m_tr(tr) | ||||
|         { | ||||
|             m_tr_inv = m_tr.inverse(); | ||||
|             z_scale  = (m_tr.linear() * Vec3d::UnitZ()).norm(); | ||||
|         } | ||||
| 
 | ||||
|         // Inherited via IProject
 | ||||
|         std::pair<Vec3d, Vec3d> create_front_back(const Point &p) const override | ||||
|         { | ||||
|             auto [front, back] = m_core->create_front_back(p); | ||||
|             return std::make_pair(m_tr * front, m_tr * back); | ||||
|         } | ||||
|         Vec3d project(const Vec3d &point) const override{ | ||||
|             return m_core->project(point); | ||||
|         } | ||||
|         std::optional<Vec2d> unproject(const Vec3d &p, double *depth = nullptr) const override { | ||||
|             auto res = m_core->unproject(m_tr_inv * p, depth); | ||||
|             if (depth != nullptr) | ||||
|                 *depth *= z_scale; | ||||
|             return res; | ||||
|         } | ||||
|     }; | ||||
| 
 | ||||
|     class OrthoProject3d : public Emboss::IProject3d | ||||
|     { | ||||
|         // size and direction of emboss for ortho projection
 | ||||
|         Vec3d m_direction; | ||||
|     public: | ||||
|         OrthoProject3d(Vec3d direction) : m_direction(direction) {} | ||||
|         Vec3d project(const Vec3d &point) const override{ return point + m_direction;} | ||||
|     }; | ||||
| 
 | ||||
|     class OrthoProject: public Emboss::IProjection { | ||||
|         Transform3d m_matrix; | ||||
|         // size and direction of emboss for ortho projection
 | ||||
|         Vec3d       m_direction; | ||||
|         Transform3d m_matrix_inv; | ||||
|     public: | ||||
|         OrthoProject(Transform3d matrix, Vec3d direction) | ||||
|             : m_matrix(matrix), m_direction(direction), m_matrix_inv(matrix.inverse()) | ||||
|         {} | ||||
|         // Inherited via IProject
 | ||||
|         std::pair<Vec3d, Vec3d> create_front_back(const Point &p) const override; | ||||
|         Vec3d project(const Vec3d &point) const override; | ||||
|         std::optional<Vec2d> unproject(const Vec3d &p, double * depth = nullptr) const override;      | ||||
|     }; | ||||
| 
 | ||||
|     /// <summary>
 | ||||
|     /// Define polygon for draw letters
 | ||||
|     /// </summary>
 | ||||
|     struct TextLine | ||||
|     { | ||||
|         // slice of object
 | ||||
|         Polygon polygon; | ||||
| 
 | ||||
|         // point laying on polygon closest to zero
 | ||||
|         PolygonPoint start; | ||||
| 
 | ||||
|         // offset of text line in volume mm
 | ||||
|         float y; | ||||
|     }; | ||||
|     using TextLines = std::vector<TextLine>; | ||||
| 
 | ||||
|     /// <summary>
 | ||||
|     /// Sample slice polygon by bounding boxes centers
 | ||||
|     /// slice start point has shape_center_x coor
 | ||||
|     /// </summary>
 | ||||
|     /// <param name="slice">Polygon and start point[Slic3r scaled milimeters]</param>
 | ||||
|     /// <param name="bbs">Bounding boxes of letter on one line[in font scales]</param>
 | ||||
|     /// <param name="scale">Scale for bbs (after multiply bb is in milimeters)</param>
 | ||||
|     /// <returns>Sampled polygon by bounding boxes</returns>
 | ||||
|     PolygonPoints sample_slice(const TextLine &slice, const BoundingBoxes &bbs, double scale); | ||||
| 
 | ||||
|     /// <summary>
 | ||||
|     /// Calculate angle for polygon point
 | ||||
|     /// </summary>
 | ||||
|     /// <param name="distance">Distance for found normal in point</param>
 | ||||
|     /// <param name="polygon_point">Select point on polygon</param>
 | ||||
|     /// <param name="polygon">Polygon know neighbor of point</param>
 | ||||
|     /// <returns>angle(atan2) of normal in polygon point</returns>
 | ||||
|     double calculate_angle(int32_t distance, PolygonPoint polygon_point, const Polygon &polygon); | ||||
|     std::vector<double> calculate_angles(int32_t distance, const PolygonPoints& polygon_points, const Polygon &polygon); | ||||
| 
 | ||||
| } // namespace Emboss
 | ||||
| 
 | ||||
| ///////////////////////
 | ||||
| // Move to ExPolygonsWithIds Utils
 | ||||
| void translate(ExPolygonsWithIds &e, const Point &p); | ||||
| BoundingBox get_extents(const ExPolygonsWithIds &e); | ||||
| void center(ExPolygonsWithIds &e); | ||||
| // delta .. safe offset before union (use as boolean close)
 | ||||
| // NOTE: remove unprintable spaces between neighbor curves (made by linearization of curve)
 | ||||
| ExPolygons union_with_delta(EmbossShape &shape, float delta, unsigned max_heal_iteration); | ||||
| } // namespace Slic3r
 | ||||
| #endif // slic3r_Emboss_hpp_
 | ||||
							
								
								
									
										143
									
								
								src/libslic3r/EmbossShape.hpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,143 @@ | |||
| #ifndef slic3r_EmbossShape_hpp_ | ||||
| #define slic3r_EmbossShape_hpp_ | ||||
| 
 | ||||
| #include <string> | ||||
| #include <optional> | ||||
| #include <memory> // unique_ptr
 | ||||
| #include <cereal/cereal.hpp> | ||||
| #include <cereal/types/string.hpp> | ||||
| #include <cereal/types/vector.hpp> | ||||
| #include <cereal/types/optional.hpp> | ||||
| #include <cereal/archives/binary.hpp> | ||||
| #include "Point.hpp" // Transform3d
 | ||||
| #include "ExPolygon.hpp" | ||||
| #include "ExPolygonSerialize.hpp" | ||||
| #include "nanosvg/nanosvg.h" // NSVGimage
 | ||||
| 
 | ||||
| namespace Slic3r { | ||||
| 
 | ||||
| struct EmbossProjection{ | ||||
|     // Emboss depth, Size in local Z direction
 | ||||
|     double depth = 1.; // [in loacal mm] 
 | ||||
|     // NOTE: User should see and modify mainly world size not local
 | ||||
| 
 | ||||
|     // Flag that result volume use surface cutted from source objects
 | ||||
|     bool use_surface = false; | ||||
| 
 | ||||
|     bool operator==(const EmbossProjection &other) const { | ||||
|         return depth == other.depth && use_surface == other.use_surface; | ||||
|     } | ||||
| 
 | ||||
|     // undo / redo stack recovery
 | ||||
|     template<class Archive> void serialize(Archive &ar) { ar(depth, use_surface); } | ||||
| }; | ||||
| 
 | ||||
| // Extend expolygons with information whether it was successfull healed
 | ||||
| struct HealedExPolygons{ | ||||
|     ExPolygons expolygons; | ||||
|     bool is_healed; | ||||
|     operator ExPolygons&() { return expolygons; } | ||||
| }; | ||||
| 
 | ||||
| // Help structure to identify expolygons grups
 | ||||
| // e.g. emboss -> per glyph -> identify character
 | ||||
| struct ExPolygonsWithId | ||||
| {  | ||||
|     // Identificator for shape
 | ||||
|     // In text it separate letters and the name is unicode value of letter
 | ||||
|     // Is svg it is id of path
 | ||||
|     unsigned id; | ||||
| 
 | ||||
|     // shape defined by integer point contain only lines
 | ||||
|     // Curves are converted to sequence of lines
 | ||||
|     ExPolygons expoly; | ||||
| 
 | ||||
|     // flag whether expolygons are fully healed(without duplication)
 | ||||
|     bool is_healed = true; | ||||
| }; | ||||
| using ExPolygonsWithIds = std::vector<ExPolygonsWithId>; | ||||
| 
 | ||||
| /// <summary>
 | ||||
| /// Contain plane shape information to be able emboss it and edit it
 | ||||
| /// </summary>
 | ||||
| struct EmbossShape  | ||||
| { | ||||
|     // shapes to to emboss separately over surface
 | ||||
|     ExPolygonsWithIds shapes_with_ids; | ||||
| 
 | ||||
|     // Only cache for final shape
 | ||||
|     // It is calculated from ExPolygonsWithIds
 | ||||
|     // Flag is_healed --> whether union of shapes is healed
 | ||||
|     // Healed mean without selfintersection and point duplication
 | ||||
|     HealedExPolygons final_shape; | ||||
| 
 | ||||
|     // scale of shape, multiplier to get 3d point in mm from integer shape
 | ||||
|     double scale = SCALING_FACTOR; | ||||
| 
 | ||||
|     // Define how to emboss shape
 | ||||
|     EmbossProjection projection; | ||||
| 
 | ||||
|     // !!! Volume stored in .3mf has transformed vertices.
 | ||||
|     // (baked transformation into vertices position)
 | ||||
|     // Only place for fill this is when load from .3mf
 | ||||
|     // This is correction for volume transformation
 | ||||
|     // Stored_Transform3d * fix_3mf_tr = Transform3d_before_store_to_3mf
 | ||||
|     std::optional<Slic3r::Transform3d> fix_3mf_tr; | ||||
| 
 | ||||
|     struct SvgFile { | ||||
|         // File(.svg) path on local computer 
 | ||||
|         // When empty can't reload from disk
 | ||||
|         std::string path; | ||||
| 
 | ||||
|         // File path into .3mf(.zip)
 | ||||
|         // When empty svg is not stored into .3mf file yet.
 | ||||
|         // and will create dialog to delete private data on save.
 | ||||
|         std::string path_in_3mf; | ||||
| 
 | ||||
|         // Loaded svg file data.
 | ||||
|         // !!! It is not serialized on undo/redo stack 
 | ||||
|         std::shared_ptr<NSVGimage> image = nullptr; | ||||
| 
 | ||||
|         // Loaded string data from file
 | ||||
|         std::shared_ptr<std::string> file_data = nullptr; | ||||
| 
 | ||||
|         template<class Archive> void save(Archive &ar) const { | ||||
|             // Note: image is only cache it is not neccessary to store
 | ||||
| 
 | ||||
|             // Store file data as plain string
 | ||||
|             assert(file_data != nullptr); | ||||
|             ar(path, path_in_3mf, (file_data != nullptr) ? *file_data : std::string("")); | ||||
|         } | ||||
|         template<class Archive> void load(Archive &ar) { | ||||
|             // for restore shared pointer on file data
 | ||||
|             std::string file_data_str; | ||||
|             ar(path, path_in_3mf, file_data_str); | ||||
|             if (!file_data_str.empty()) | ||||
|                 file_data = std::make_unique<std::string>(file_data_str); | ||||
|         } | ||||
|     }; | ||||
|     // When embossing shape is made by svg file this is source data
 | ||||
|     std::optional<SvgFile> svg_file; | ||||
| 
 | ||||
|     // undo / redo stack recovery
 | ||||
|     template<class Archive> void save(Archive &ar) const | ||||
|     { | ||||
|         // final_shape is not neccessary to store - it is only cache
 | ||||
|         ar(shapes_with_ids, final_shape, scale, projection, svg_file); | ||||
|         cereal::save(ar, fix_3mf_tr); | ||||
|     } | ||||
|     template<class Archive> void load(Archive &ar) | ||||
|     { | ||||
|         ar(shapes_with_ids, final_shape, scale, projection, svg_file); | ||||
|         cereal::load(ar, fix_3mf_tr); | ||||
|     } | ||||
| }; | ||||
| } // namespace Slic3r
 | ||||
| 
 | ||||
| // Serialization through the Cereal library
 | ||||
| namespace cereal { | ||||
| template<class Archive> void serialize(Archive &ar, Slic3r::ExPolygonsWithId &o) { ar(o.id, o.expoly, o.is_healed); } | ||||
| template<class Archive> void serialize(Archive &ar, Slic3r::HealedExPolygons &o) { ar(o.expolygons, o.is_healed); } | ||||
| }; // namespace cereal
 | ||||
| 
 | ||||
| #endif // slic3r_EmbossShape_hpp_
 | ||||
|  | @ -1,3 +1,15 @@ | |||
| ///|/ Copyright (c) Prusa Research 2016 - 2023 Vojtěch Bubník @bubnikv, Lukáš Matěna @lukasmatena, Lukáš Hejl @hejllukas
 | ||||
| ///|/ Copyright (c) Slic3r 2013 - 2016 Alessandro Ranellucci @alranel
 | ||||
| ///|/ Copyright (c) 2015 Maksim Derbasov @ntfshard
 | ||||
| ///|/ Copyright (c) 2014 Petr Ledvina @ledvinap
 | ||||
| ///|/
 | ||||
| ///|/ ported from lib/Slic3r/ExPolygon.pm:
 | ||||
| ///|/ Copyright (c) Prusa Research 2017 - 2022 Vojtěch Bubník @bubnikv
 | ||||
| ///|/ Copyright (c) Slic3r 2011 - 2014 Alessandro Ranellucci @alranel
 | ||||
| ///|/ Copyright (c) 2012 Mark Hindess
 | ||||
| ///|/
 | ||||
| ///|/ PrusaSlicer is released under the terms of the AGPLv3 or higher
 | ||||
| ///|/
 | ||||
| #include "BoundingBox.hpp" | ||||
| #include "ExPolygon.hpp" | ||||
| #include "Exception.hpp" | ||||
|  | @ -475,6 +487,24 @@ bool has_duplicate_points(const ExPolygons &expolys) | |||
| #endif | ||||
| } | ||||
| 
 | ||||
| bool remove_same_neighbor(ExPolygons &expolygons) | ||||
| { | ||||
|     if (expolygons.empty()) | ||||
|         return false; | ||||
|     bool remove_from_holes   = false; | ||||
|     bool remove_from_contour = false; | ||||
|     for (ExPolygon &expoly : expolygons) { | ||||
|         remove_from_contour |= remove_same_neighbor(expoly.contour); | ||||
|         remove_from_holes |= remove_same_neighbor(expoly.holes); | ||||
|     } | ||||
|     // Removing of expolygons without contour
 | ||||
|     if (remove_from_contour) | ||||
|         expolygons.erase(std::remove_if(expolygons.begin(), expolygons.end(), | ||||
|                                         [](const ExPolygon &p) { return p.contour.points.size() <= 2; }), | ||||
|                          expolygons.end()); | ||||
|     return remove_from_holes || remove_from_contour; | ||||
| } | ||||
| 
 | ||||
| bool remove_sticks(ExPolygon &poly) | ||||
| { | ||||
|     return remove_sticks(poly.contour) || remove_sticks(poly.holes); | ||||
|  |  | |||
|  | @ -1,3 +1,14 @@ | |||
| ///|/ Copyright (c) Prusa Research 2016 - 2023 Pavel Mikuš @Godrak, Vojtěch Bubník @bubnikv, Lukáš Matěna @lukasmatena, Enrico Turri @enricoturri1966, Filip Sykala @Jony01, Lukáš Hejl @hejllukas, Tomáš Mészáros @tamasmeszaros
 | ||||
| ///|/ Copyright (c) 2016 Sakari Kapanen @Flannelhead
 | ||||
| ///|/ Copyright (c) Slic3r 2013 - 2016 Alessandro Ranellucci @alranel
 | ||||
| ///|/
 | ||||
| ///|/ ported from lib/Slic3r/ExPolygon.pm:
 | ||||
| ///|/ Copyright (c) Prusa Research 2017 - 2022 Vojtěch Bubník @bubnikv
 | ||||
| ///|/ Copyright (c) Slic3r 2011 - 2014 Alessandro Ranellucci @alranel
 | ||||
| ///|/ Copyright (c) 2012 Mark Hindess
 | ||||
| ///|/
 | ||||
| ///|/ PrusaSlicer is released under the terms of the AGPLv3 or higher
 | ||||
| ///|/
 | ||||
| #ifndef slic3r_ExPolygon_hpp_ | ||||
| #define slic3r_ExPolygon_hpp_ | ||||
| 
 | ||||
|  | @ -142,19 +153,6 @@ inline Lines to_lines(const ExPolygons &src) | |||
|     return lines; | ||||
| } | ||||
| 
 | ||||
| inline Points to_points(const ExPolygons& src) | ||||
| { | ||||
|     Points points; | ||||
|     size_t count = count_points(src); | ||||
|     points.reserve(count); | ||||
|     for (const ExPolygon& expolygon : src) { | ||||
|         append(points, expolygon.contour.points); | ||||
|         for (const Polygon& hole : expolygon.holes) | ||||
|             append(points, hole.points); | ||||
|     } | ||||
|     return points; | ||||
| } | ||||
| 
 | ||||
| // Line is from point index(see to_points) to next point.
 | ||||
| // Next point of last point in polygon is first polygon point.
 | ||||
| inline Linesf to_linesf(const ExPolygons &src, uint32_t count_lines = 0) | ||||
|  | @ -205,6 +203,20 @@ inline Linesf to_unscaled_linesf(const ExPolygons &src) | |||
|     return lines; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| inline Points to_points(const ExPolygons &src) | ||||
| { | ||||
|     Points points; | ||||
|     size_t count = count_points(src); | ||||
|     points.reserve(count); | ||||
|     for (const ExPolygon &expolygon : src) { | ||||
|         append(points, expolygon.contour.points); | ||||
|         for (const Polygon &hole : expolygon.holes) | ||||
|             append(points, hole.points); | ||||
|     } | ||||
|     return points; | ||||
| } | ||||
| 
 | ||||
| inline Polylines to_polylines(const ExPolygon &src) | ||||
| { | ||||
|     Polylines polylines; | ||||
|  | @ -371,6 +383,11 @@ inline Points to_points(const ExPolygon &expoly) | |||
|     return out; | ||||
| } | ||||
| 
 | ||||
| inline void translate(ExPolygons &expolys, const Point &p) { | ||||
|     for (ExPolygon &expoly : expolys) | ||||
|         expoly.translate(p); | ||||
| } | ||||
| 
 | ||||
| inline void polygons_append(Polygons &dst, const ExPolygon &src)  | ||||
| {  | ||||
|     dst.reserve(dst.size() + src.holes.size() + 1); | ||||
|  | @ -464,6 +481,9 @@ std::vector<BoundingBox> get_extents_vector(const ExPolygons &polygons); | |||
| bool has_duplicate_points(const ExPolygon &expoly); | ||||
| bool has_duplicate_points(const ExPolygons &expolys); | ||||
| 
 | ||||
| // Return True when erase some otherwise False.
 | ||||
| bool remove_same_neighbor(ExPolygons &expolys); | ||||
| 
 | ||||
| bool remove_sticks(ExPolygon &poly); | ||||
| void keep_largest_contour_only(ExPolygons &polygons); | ||||
| 
 | ||||
|  |  | |||
							
								
								
									
										28
									
								
								src/libslic3r/ExPolygonSerialize.hpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,28 @@ | |||
| #ifndef slic3r_ExPolygonSerialize_hpp_ | ||||
| #define slic3r_ExPolygonSerialize_hpp_ | ||||
| 
 | ||||
| #include "ExPolygon.hpp" | ||||
| #include "Point.hpp" // Cereal serialization of Point
 | ||||
| #include <cereal/cereal.hpp> | ||||
| #include <cereal/types/vector.hpp> | ||||
| 
 | ||||
| /// <summary>
 | ||||
| /// External Cereal serialization of ExPolygons
 | ||||
| /// </summary>
 | ||||
| 
 | ||||
| // Serialization through the Cereal library
 | ||||
| #include <cereal/access.hpp> | ||||
| namespace cereal { | ||||
| 
 | ||||
| template<class Archive>  | ||||
| void serialize(Archive &archive, Slic3r::Polygon &polygon) {	 | ||||
| 	archive(polygon.points); | ||||
| } | ||||
| 
 | ||||
| template<class Archive>  | ||||
| void serialize(Archive &archive, Slic3r::ExPolygon &expoly) { | ||||
| 	archive(expoly.contour, expoly.holes);  | ||||
| } | ||||
| 
 | ||||
| } // namespace Slic3r
 | ||||
| #endif // slic3r_ExPolygonSerialize_hpp_
 | ||||
							
								
								
									
										82
									
								
								src/libslic3r/ExPolygonsIndex.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,82 @@ | |||
| #include "ExPolygonsIndex.hpp" | ||||
| using namespace Slic3r; | ||||
| 
 | ||||
| // IMPROVE: use one dimensional vector for polygons offset with searching by std::lower_bound
 | ||||
| ExPolygonsIndices::ExPolygonsIndices(const ExPolygons &shapes) | ||||
| { | ||||
|     // prepare offsets
 | ||||
|     m_offsets.reserve(shapes.size()); | ||||
|     uint32_t offset = 0; | ||||
|     for (const ExPolygon &shape : shapes) { | ||||
|         assert(!shape.contour.points.empty()); | ||||
|         std::vector<uint32_t> shape_offsets; | ||||
|         shape_offsets.reserve(shape.holes.size() + 1); | ||||
|         shape_offsets.push_back(offset); | ||||
|         offset += shape.contour.points.size(); | ||||
|         for (const Polygon &hole: shape.holes) { | ||||
|             shape_offsets.push_back(offset); | ||||
|             offset += hole.points.size(); | ||||
|         } | ||||
|         m_offsets.push_back(std::move(shape_offsets)); | ||||
|     } | ||||
|     m_count = offset; | ||||
| } | ||||
| 
 | ||||
| uint32_t ExPolygonsIndices::cvt(const ExPolygonsIndex &id) const | ||||
| { | ||||
|     assert(id.expolygons_index < m_offsets.size()); | ||||
|     const std::vector<uint32_t> &shape_offset = m_offsets[id.expolygons_index]; | ||||
|     assert(id.polygon_index < shape_offset.size()); | ||||
|     uint32_t res = shape_offset[id.polygon_index] + id.point_index; | ||||
|     assert(res < m_count); | ||||
|     return res; | ||||
| } | ||||
| 
 | ||||
| ExPolygonsIndex ExPolygonsIndices::cvt(uint32_t index) const | ||||
| { | ||||
|     assert(index < m_count); | ||||
|     ExPolygonsIndex result{0, 0, 0}; | ||||
|     // find expolygon index
 | ||||
|     auto fn = [](const std::vector<uint32_t> &offsets, uint32_t index) { return offsets[0] < index; }; | ||||
|     auto it = std::lower_bound(m_offsets.begin() + 1, m_offsets.end(), index, fn); | ||||
|     result.expolygons_index = it - m_offsets.begin(); | ||||
|     if (it == m_offsets.end() || it->at(0) != index) --result.expolygons_index; | ||||
| 
 | ||||
|     // find polygon index
 | ||||
|     const std::vector<uint32_t> &shape_offset = m_offsets[result.expolygons_index]; | ||||
|     auto it2 = std::lower_bound(shape_offset.begin() + 1, shape_offset.end(), index); | ||||
|     result.polygon_index = it2 - shape_offset.begin(); | ||||
|     if (it2 == shape_offset.end() || *it2 != index) --result.polygon_index; | ||||
| 
 | ||||
|     // calculate point index
 | ||||
|     uint32_t polygon_offset = shape_offset[result.polygon_index]; | ||||
|     assert(index >= polygon_offset); | ||||
|     result.point_index = index - polygon_offset; | ||||
|     return result; | ||||
| } | ||||
| 
 | ||||
| bool ExPolygonsIndices::is_last_point(const ExPolygonsIndex &id) const {  | ||||
|     assert(id.expolygons_index < m_offsets.size()); | ||||
|     const std::vector<uint32_t> &shape_offset = m_offsets[id.expolygons_index]; | ||||
|     assert(id.polygon_index < shape_offset.size()); | ||||
|     uint32_t index = shape_offset[id.polygon_index] + id.point_index; | ||||
|     assert(index < m_count); | ||||
|     // next index
 | ||||
|     uint32_t next_point_index = index + 1; | ||||
|     uint32_t next_poly_index = id.polygon_index + 1; | ||||
|     uint32_t next_expoly_index = id.expolygons_index + 1; | ||||
|     // is last expoly?
 | ||||
|     if (next_expoly_index == m_offsets.size()) { | ||||
|         // is last expoly last poly?
 | ||||
|         if (next_poly_index == shape_offset.size())  | ||||
|             return next_point_index == m_count; | ||||
|     } else { | ||||
|         // (not last expoly) is expoly last poly?
 | ||||
|         if (next_poly_index == shape_offset.size())  | ||||
|             return next_point_index == m_offsets[next_expoly_index][0]; | ||||
|     } | ||||
|     // Not last polygon in expolygon
 | ||||
|     return next_point_index == shape_offset[next_poly_index]; | ||||
| } | ||||
| 
 | ||||
| uint32_t ExPolygonsIndices::get_count() const { return m_count; }		 | ||||
							
								
								
									
										74
									
								
								src/libslic3r/ExPolygonsIndex.hpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,74 @@ | |||
| #ifndef slic3r_ExPolygonsIndex_hpp_ | ||||
| #define slic3r_ExPolygonsIndex_hpp_ | ||||
| 
 | ||||
| #include "ExPolygon.hpp" | ||||
| namespace Slic3r { | ||||
| 
 | ||||
| /// <summary>
 | ||||
| /// Index into ExPolygons
 | ||||
| /// Identify expolygon, its contour (or hole) and point
 | ||||
| /// </summary>
 | ||||
| struct ExPolygonsIndex | ||||
| { | ||||
|     // index of ExPolygons
 | ||||
|     uint32_t expolygons_index; | ||||
| 
 | ||||
|     // index of Polygon
 | ||||
|     // 0 .. contour
 | ||||
|     // N .. hole[N-1]
 | ||||
|     uint32_t polygon_index; | ||||
| 
 | ||||
|     // index of point in polygon
 | ||||
|     uint32_t point_index; | ||||
| 
 | ||||
|     bool is_contour() const { return polygon_index == 0; } | ||||
|     bool is_hole() const { return polygon_index != 0; } | ||||
|     uint32_t hole_index() const { return polygon_index - 1; } | ||||
| }; | ||||
| 
 | ||||
| /// <summary>
 | ||||
| /// Keep conversion from ExPolygonsIndex to Index and vice versa
 | ||||
| /// ExPolygonsIndex .. contour(or hole) point from ExPolygons
 | ||||
| /// Index           .. continous number
 | ||||
| /// 
 | ||||
| /// index is used to address lines and points as result from function 
 | ||||
| /// Slic3r::to_lines, Slic3r::to_points
 | ||||
| /// </summary>
 | ||||
| class ExPolygonsIndices | ||||
| { | ||||
|     std::vector<std::vector<uint32_t>> m_offsets; | ||||
|     // for check range of index
 | ||||
|     uint32_t m_count; // count of points
 | ||||
| public: | ||||
|     ExPolygonsIndices(const ExPolygons &shapes); | ||||
| 
 | ||||
|     /// <summary>
 | ||||
|     /// Convert to one index number
 | ||||
|     /// </summary>
 | ||||
|     /// <param name="id">Compose of adress into expolygons</param>
 | ||||
|     /// <returns>Index</returns>
 | ||||
|     uint32_t cvt(const ExPolygonsIndex &id) const; | ||||
| 
 | ||||
|     /// <summary>
 | ||||
|     /// Separate to multi index
 | ||||
|     /// </summary>
 | ||||
|     /// <param name="index">adress into expolygons</param>
 | ||||
|     /// <returns></returns>
 | ||||
|     ExPolygonsIndex cvt(uint32_t index) const; | ||||
| 
 | ||||
|     /// <summary>
 | ||||
|     /// Check whether id is last point in polygon
 | ||||
|     /// </summary>
 | ||||
|     /// <param name="id">Identify point in expolygon</param>
 | ||||
|     /// <returns>True when id is last point in polygon otherwise false</returns>
 | ||||
|     bool is_last_point(const ExPolygonsIndex &id) const; | ||||
| 
 | ||||
|     /// <summary>
 | ||||
|     /// Count of points in expolygons
 | ||||
|     /// </summary>
 | ||||
|     /// <returns>Count of points in expolygons</returns>
 | ||||
|     uint32_t get_count() const; | ||||
| }; | ||||
| 
 | ||||
| } // namespace Slic3r
 | ||||
| #endif // slic3r_ExPolygonsIndex_hpp_
 | ||||
|  | @ -1,3 +1,7 @@ | |||
| ///|/ Copyright (c) Prusa Research 2020 - 2023 Tomáš Mészáros @tamasmeszaros, Oleksandra Iushchenko @YuSanka, Lukáš Matěna @lukasmatena, Vojtěch Bubník @bubnikv
 | ||||
| ///|/
 | ||||
| ///|/ PrusaSlicer is released under the terms of the AGPLv3 or higher
 | ||||
| ///|/
 | ||||
| #include "SL1.hpp" | ||||
| #include "GCode/ThumbnailData.hpp" | ||||
| #include "libslic3r/Time.hpp" | ||||
|  | @ -442,8 +446,8 @@ void fill_slicerconf(ConfMap &m, const SLAPrint &print) | |||
| 
 | ||||
| std::unique_ptr<sla::RasterBase> SL1Archive::create_raster() const | ||||
| { | ||||
|     sla::RasterBase::Resolution res; | ||||
|     sla::RasterBase::PixelDim   pxdim; | ||||
|     sla::Resolution res; | ||||
|     sla::PixelDim   pxdim; | ||||
|     std::array<bool, 2>         mirror; | ||||
| 
 | ||||
|     double w  = m_cfg.display_width.getFloat(); | ||||
|  | @ -464,8 +468,8 @@ std::unique_ptr<sla::RasterBase> SL1Archive::create_raster() const | |||
|         std::swap(pw, ph); | ||||
|     } | ||||
| 
 | ||||
|     res   = sla::RasterBase::Resolution{pw, ph}; | ||||
|     pxdim = sla::RasterBase::PixelDim{w / pw, h / ph}; | ||||
|     res   = sla::Resolution{pw, ph}; | ||||
|     pxdim = sla::PixelDim{w / pw, h / ph}; | ||||
|     sla::RasterBase::Trafo tr{orientation, mirror}; | ||||
| 
 | ||||
|     double gamma = m_cfg.gamma_correction.getFloat(); | ||||
|  |  | |||
|  | @ -23,6 +23,9 @@ | |||
| #include <stdexcept> | ||||
| #include <iomanip> | ||||
| 
 | ||||
| #include <boost/assign.hpp> | ||||
| #include <boost/bimap.hpp> | ||||
| 
 | ||||
| #include <boost/algorithm/string/classification.hpp> | ||||
| #include <boost/algorithm/string/split.hpp> | ||||
| #include <boost/algorithm/string/predicate.hpp> | ||||
|  | @ -48,6 +51,13 @@ namespace pt = boost::property_tree; | |||
| #include <Eigen/Dense> | ||||
| #include "miniz_extension.hpp" | ||||
| #include "nlohmann/json.hpp" | ||||
| 
 | ||||
| #include "TextConfiguration.hpp" | ||||
| #include "EmbossShape.hpp" | ||||
| #include "ExPolygonSerialize.hpp"  | ||||
| 
 | ||||
| #include "NSVGUtils.hpp" | ||||
| 
 | ||||
| #include <fast_float/fast_float.h> | ||||
| 
 | ||||
| // Slightly faster than sprintf("%.9g"), but there is an issue with the karma floating point formatter,
 | ||||
|  | @ -211,7 +221,7 @@ static constexpr const char* ASSEMBLE_ITEM_TAG = "assemble_item"; | |||
| static constexpr const char* SLICE_HEADER_TAG = "header"; | ||||
| static constexpr const char* SLICE_HEADER_ITEM_TAG = "header_item"; | ||||
| 
 | ||||
| // text_info
 | ||||
| // Deprecated: text_info
 | ||||
| static constexpr const char* TEXT_INFO_TAG        = "text_info"; | ||||
| static constexpr const char* TEXT_ATTR            = "text"; | ||||
| static constexpr const char* FONT_NAME_ATTR       = "font_name"; | ||||
|  | @ -325,6 +335,42 @@ static constexpr const char* MESH_STAT_FACETS_REMOVED       = "facets_removed"; | |||
| static constexpr const char* MESH_STAT_FACETS_RESERVED      = "facets_reversed"; | ||||
| static constexpr const char* MESH_STAT_BACKWARDS_EDGES      = "backwards_edges"; | ||||
| 
 | ||||
| // Store / load of TextConfiguration
 | ||||
| static constexpr const char *TEXT_TAG = "slic3rpe:text"; | ||||
| static constexpr const char *TEXT_DATA_ATTR = "text"; | ||||
| // TextConfiguration::EmbossStyle
 | ||||
| static constexpr const char *STYLE_NAME_ATTR      = "style_name"; | ||||
| static constexpr const char *FONT_DESCRIPTOR_ATTR = "font_descriptor"; | ||||
| static constexpr const char *FONT_DESCRIPTOR_TYPE_ATTR = "font_descriptor_type"; | ||||
| 
 | ||||
| // TextConfiguration::FontProperty
 | ||||
| static constexpr const char *CHAR_GAP_ATTR    = "char_gap"; | ||||
| static constexpr const char *LINE_GAP_ATTR    = "line_gap"; | ||||
| static constexpr const char *LINE_HEIGHT_ATTR = "line_height"; | ||||
| static constexpr const char *BOLDNESS_ATTR    = "boldness"; | ||||
| static constexpr const char *SKEW_ATTR        = "skew"; | ||||
| static constexpr const char *PER_GLYPH_ATTR   = "per_glyph"; | ||||
| static constexpr const char *HORIZONTAL_ALIGN_ATTR  = "horizontal"; | ||||
| static constexpr const char *VERTICAL_ALIGN_ATTR    = "vertical"; | ||||
| static constexpr const char *COLLECTION_NUMBER_ATTR = "collection"; | ||||
| 
 | ||||
| static constexpr const char *FONT_FAMILY_ATTR    = "family"; | ||||
| static constexpr const char *FONT_FACE_NAME_ATTR = "face_name"; | ||||
| static constexpr const char *FONT_STYLE_ATTR     = "style"; | ||||
| static constexpr const char *FONT_WEIGHT_ATTR    = "weight"; | ||||
| 
 | ||||
| // Store / load of EmbossShape
 | ||||
| static constexpr const char *SHAPE_TAG = "slic3rpe:shape"; | ||||
| static constexpr const char *SHAPE_SCALE_ATTR   = "scale"; | ||||
| static constexpr const char *UNHEALED_ATTR = "unhealed"; | ||||
| static constexpr const char *SVG_FILE_PATH_ATTR = "filepath"; | ||||
| static constexpr const char *SVG_FILE_PATH_IN_3MF_ATTR = "filepath3mf"; | ||||
| 
 | ||||
| // EmbossProjection
 | ||||
| static constexpr const char *DEPTH_ATTR       = "depth"; | ||||
| static constexpr const char *USE_SURFACE_ATTR = "use_surface"; | ||||
| // static constexpr const char *FIX_TRANSFORMATION_ATTR = "transform";
 | ||||
| 
 | ||||
| 
 | ||||
| const unsigned int BBS_VALID_OBJECT_TYPES_COUNT = 2; | ||||
| const char* BBS_VALID_OBJECT_TYPES[] = | ||||
|  | @ -718,8 +764,8 @@ void PlateData::parse_filament_info(GCodeProcessorResult *result) | |||
|                 MetadataList metadata; | ||||
|                 RepairedMeshErrors mesh_stats; | ||||
|                 ModelVolumeType part_type; | ||||
|                 TextInfo text_info; | ||||
| 
 | ||||
|                 std::optional<TextConfiguration> text_configuration; | ||||
|                 std::optional<EmbossShape> shape_configuration; | ||||
|                 VolumeMetadata(unsigned int first_triangle_id, unsigned int last_triangle_id, ModelVolumeType type = ModelVolumeType::MODEL_PART) | ||||
|                     : first_triangle_id(first_triangle_id) | ||||
|                     , last_triangle_id(last_triangle_id) | ||||
|  | @ -770,6 +816,7 @@ void PlateData::parse_filament_info(GCodeProcessorResult *result) | |||
|         typedef std::map<int, CutObjectInfo>         IdToCutObjectInfoMap; | ||||
|         /*typedef std::map<int, std::vector<sla::SupportPoint>> IdToSlaSupportPointsMap;
 | ||||
|         typedef std::map<int, std::vector<sla::DrainHole>> IdToSlaDrainHolesMap;*/ | ||||
|         using PathToEmbossShapeFileMap = std::map<std::string, std::shared_ptr<std::string>>; | ||||
| 
 | ||||
|         struct ObjectImporter | ||||
|         { | ||||
|  | @ -962,6 +1009,7 @@ void PlateData::parse_filament_info(GCodeProcessorResult *result) | |||
|         IdToLayerConfigRangesMap m_layer_config_ranges; | ||||
|         /*IdToSlaSupportPointsMap m_sla_support_points;
 | ||||
|         IdToSlaDrainHolesMap    m_sla_drain_holes;*/ | ||||
|         PathToEmbossShapeFileMap m_path_to_emboss_shape_files; | ||||
|         std::string m_curr_metadata_name; | ||||
|         std::string m_curr_characters; | ||||
|         std::string m_name; | ||||
|  | @ -1015,6 +1063,7 @@ void PlateData::parse_filament_info(GCodeProcessorResult *result) | |||
|         // add backup & restore logic
 | ||||
|         bool _load_model_from_file(std::string filename, Model& model, PlateDataPtrs& plate_data_list, std::vector<Preset*>& project_presets, DynamicPrintConfig& config, ConfigSubstitutionContext& config_substitutions, Import3mfProgressFn proFn = nullptr, | ||||
|             BBLProject* project = nullptr, int plate_id = 0); | ||||
|         bool _is_svg_shape_file(const std::string &filename) const; | ||||
|         bool _extract_from_archive(mz_zip_archive& archive, std::string const & path, std::function<bool (mz_zip_archive& archive, const mz_zip_archive_file_stat& stat)>, bool restore = false); | ||||
|         bool _extract_xml_from_archive(mz_zip_archive& archive, std::string const & path, XML_StartElementHandler start_handler, XML_EndElementHandler end_handler); | ||||
|         bool _extract_xml_from_archive(mz_zip_archive& archive, const mz_zip_archive_file_stat& stat, XML_StartElementHandler start_handler, XML_EndElementHandler end_handler); | ||||
|  | @ -1035,6 +1084,7 @@ void PlateData::parse_filament_info(GCodeProcessorResult *result) | |||
| 
 | ||||
|         void _extract_auxiliary_file_from_archive(mz_zip_archive& archive, const mz_zip_archive_file_stat& stat, Model& model); | ||||
|         void _extract_file_from_archive(mz_zip_archive& archive, const mz_zip_archive_file_stat& stat); | ||||
|         void _extract_embossed_svg_shape_file(const std::string &filename, mz_zip_archive &archive, const mz_zip_archive_file_stat &stat); | ||||
| 
 | ||||
|         // handlers to parse the .model file
 | ||||
|         void _handle_start_model_xml_element(const char* name, const char** attributes); | ||||
|  | @ -1090,6 +1140,9 @@ void PlateData::parse_filament_info(GCodeProcessorResult *result) | |||
|         bool _handle_start_metadata(const char** attributes, unsigned int num_attributes); | ||||
|         bool _handle_end_metadata(); | ||||
| 
 | ||||
|         bool _handle_start_text_configuration(const char** attributes, unsigned int num_attributes); | ||||
|         bool _handle_start_shape_configuration(const char **attributes, unsigned int num_attributes); | ||||
| 
 | ||||
|         bool _create_object_instance(std::string const & path, int object_id, const Transform3d& transform, const bool printable, unsigned int recur_counter); | ||||
| 
 | ||||
|         void _apply_transform(ModelInstance& instance, const Transform3d& transform); | ||||
|  | @ -1141,7 +1194,7 @@ void PlateData::parse_filament_info(GCodeProcessorResult *result) | |||
| 
 | ||||
|         void _generate_current_object_list(std::vector<Component> &sub_objects, Id object_id, IdToCurrentObjectMap& current_objects); | ||||
|         bool _generate_volumes_new(ModelObject& object, const std::vector<Component> &sub_objects, const ObjectMetadata::VolumeMetadataList& volumes, ConfigSubstitutionContext& config_substitutions); | ||||
|         bool _generate_volumes(ModelObject& object, const Geometry& geometry, const ObjectMetadata::VolumeMetadataList& volumes, ConfigSubstitutionContext& config_substitutions); | ||||
|         //bool _generate_volumes(ModelObject& object, const Geometry& geometry, const ObjectMetadata::VolumeMetadataList& volumes, ConfigSubstitutionContext& config_substitutions);
 | ||||
| 
 | ||||
|         // callbacks to parse the .model file
 | ||||
|         static void XMLCALL _handle_start_model_xml_element(void* userData, const char* name, const char** attributes); | ||||
|  | @ -1757,6 +1810,9 @@ void PlateData::parse_filament_info(GCodeProcessorResult *result) | |||
|                         return false; | ||||
|                     } | ||||
|                 } | ||||
|                 else if (_is_svg_shape_file(name)) { | ||||
|                     _extract_embossed_svg_shape_file(name, archive, stat); | ||||
|                 } | ||||
|                 else if (!dont_load_config && boost::algorithm::iequals(name, SLICE_INFO_CONFIG_FILE)) { | ||||
|                     m_parsing_slice_info = true; | ||||
|                     //extract slice info from archive
 | ||||
|  | @ -2156,6 +2212,10 @@ void PlateData::parse_filament_info(GCodeProcessorResult *result) | |||
|         return true; | ||||
|     } | ||||
| 
 | ||||
|     bool _BBS_3MF_Importer::_is_svg_shape_file(const std::string &name) const {  | ||||
|         return boost::starts_with(name, MODEL_FOLDER) && boost::ends_with(name, ".svg"); | ||||
|     } | ||||
| 
 | ||||
|     bool _BBS_3MF_Importer::_extract_from_archive(mz_zip_archive& archive, std::string const & path, std::function<bool (mz_zip_archive& archive, const mz_zip_archive_file_stat& stat)> extract, bool restore) | ||||
|     { | ||||
|         mz_uint num_entries = mz_zip_reader_get_num_files(&archive); | ||||
|  | @ -2870,6 +2930,32 @@ void PlateData::parse_filament_info(GCodeProcessorResult *result) | |||
|         } | ||||
|     }*/ | ||||
| 
 | ||||
|     void _BBS_3MF_Importer::_extract_embossed_svg_shape_file(const std::string &filename, mz_zip_archive &archive, const mz_zip_archive_file_stat &stat){ | ||||
|         assert(m_path_to_emboss_shape_files.find(filename) == m_path_to_emboss_shape_files.end()); | ||||
|         auto file = std::make_unique<std::string>(stat.m_uncomp_size, '\0'); | ||||
|         mz_bool res  = mz_zip_reader_extract_to_mem(&archive, stat.m_file_index, (void *) file->data(), stat.m_uncomp_size, 0); | ||||
|         if (res == 0) { | ||||
|             add_error("Error while reading svg shape for emboss"); | ||||
|             return; | ||||
|         } | ||||
|          | ||||
|         // store for case svg is loaded before volume
 | ||||
|         m_path_to_emboss_shape_files[filename] = std::move(file); | ||||
|          | ||||
|         // find embossed volume, for case svg is loaded after volume
 | ||||
|         for (const ModelObject* object : m_model->objects) | ||||
|         for (ModelVolume *volume : object->volumes) { | ||||
|             std::optional<EmbossShape> &es = volume->emboss_shape; | ||||
|             if (!es.has_value()) | ||||
|                 continue; | ||||
|             std::optional<EmbossShape::SvgFile> &svg = es->svg_file; | ||||
|             if (!svg.has_value()) | ||||
|                 continue; | ||||
|             if (filename.compare(svg->path_in_3mf) == 0) | ||||
|                 svg->file_data = m_path_to_emboss_shape_files[filename]; | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     void _BBS_3MF_Importer::_extract_custom_gcode_per_print_z_from_archive(::mz_zip_archive &archive, const mz_zip_archive_file_stat &stat) | ||||
|     { | ||||
|         //BBS: add plate tree related logic
 | ||||
|  | @ -3066,6 +3152,10 @@ void PlateData::parse_filament_info(GCodeProcessorResult *result) | |||
|             res = _handle_start_config_volume_mesh(attributes, num_attributes); | ||||
|         else if (::strcmp(METADATA_TAG, name) == 0) | ||||
|             res = _handle_start_config_metadata(attributes, num_attributes); | ||||
|         else if (::strcmp(SHAPE_TAG, name) == 0) | ||||
|             res = _handle_start_shape_configuration(attributes, num_attributes); | ||||
|         else if (::strcmp(TEXT_TAG, name) == 0) | ||||
|             res = _handle_start_text_configuration(attributes, num_attributes); | ||||
|         else if (::strcmp(PLATE_TAG, name) == 0) | ||||
|             res = _handle_start_config_plater(attributes, num_attributes); | ||||
|         else if (::strcmp(INSTANCE_TAG, name) == 0) | ||||
|  | @ -3632,6 +3722,104 @@ void PlateData::parse_filament_info(GCodeProcessorResult *result) | |||
|         return true; | ||||
|     } | ||||
| 
 | ||||
|     struct TextConfigurationSerialization | ||||
|     { | ||||
|     public: | ||||
|         TextConfigurationSerialization() = delete; | ||||
|                  | ||||
|         using TypeToName = boost::bimap<EmbossStyle::Type, std::string_view>; | ||||
|         static const TypeToName type_to_name; | ||||
| 
 | ||||
|         using HorizontalAlignToName = boost::bimap<FontProp::HorizontalAlign, std::string_view>; | ||||
|         static const HorizontalAlignToName horizontal_align_to_name; | ||||
| 
 | ||||
|         using VerticalAlignToName = boost::bimap<FontProp::VerticalAlign, std::string_view>; | ||||
|         static const VerticalAlignToName vertical_align_to_name; | ||||
|          | ||||
|         static EmbossStyle::Type get_type(std::string_view type) { | ||||
|             const auto& to_type = TextConfigurationSerialization::type_to_name.right; | ||||
|             auto type_item = to_type.find(type); | ||||
|             assert(type_item != to_type.end()); | ||||
|             if (type_item == to_type.end()) return EmbossStyle::Type::undefined; | ||||
|             return type_item->second;         | ||||
|         } | ||||
| 
 | ||||
|         static std::string_view get_name(EmbossStyle::Type type) { | ||||
|             const auto& to_name = TextConfigurationSerialization::type_to_name.left; | ||||
|             auto type_name = to_name.find(type); | ||||
|             assert(type_name != to_name.end()); | ||||
|             if (type_name == to_name.end()) return "unknown type"; | ||||
|             return type_name->second; | ||||
|         } | ||||
| 
 | ||||
|         static void to_xml(std::stringstream &stream, const TextConfiguration &tc); | ||||
|         static std::optional<TextConfiguration> read(const char **attributes, unsigned int num_attributes); | ||||
|         static EmbossShape read_old(const char **attributes, unsigned int num_attributes); | ||||
|     }; | ||||
| 
 | ||||
|     bool _BBS_3MF_Importer::_handle_start_text_configuration(const char **attributes, unsigned int num_attributes) | ||||
|     { | ||||
|         IdToMetadataMap::iterator object = m_objects_metadata.find(m_curr_config.object_id); | ||||
|         if (object == m_objects_metadata.end()) { | ||||
|             add_error("Can not assign volume mesh to a valid object"); | ||||
|             return false; | ||||
|         } | ||||
|         if (object->second.volumes.empty()) { | ||||
|             add_error("Can not assign mesh to a valid volume"); | ||||
|             return false; | ||||
|         } | ||||
|         ObjectMetadata::VolumeMetadata& volume = object->second.volumes.back(); | ||||
|         volume.text_configuration = TextConfigurationSerialization::read(attributes, num_attributes); | ||||
|         if (!volume.text_configuration.has_value()) | ||||
|             return false; | ||||
| 
 | ||||
|         // Is 3mf version with shapes?
 | ||||
|         if (volume.shape_configuration.has_value()) | ||||
|             return true; | ||||
| 
 | ||||
|         // Back compatibility for 3mf version without shapes
 | ||||
|         volume.shape_configuration = TextConfigurationSerialization::read_old(attributes, num_attributes); | ||||
|         return true; | ||||
|     } | ||||
| 
 | ||||
|     // Definition of read/write method for EmbossShape
 | ||||
|     static void to_xml(std::stringstream &stream, const EmbossShape &es, const ModelVolume &volume, mz_zip_archive &archive); | ||||
|     static std::optional<EmbossShape> read_emboss_shape(const char **attributes, unsigned int num_attributes); | ||||
| 
 | ||||
|     bool _BBS_3MF_Importer::_handle_start_shape_configuration(const char **attributes, unsigned int num_attributes) | ||||
|     { | ||||
|         IdToMetadataMap::iterator object = m_objects_metadata.find(m_curr_config.object_id); | ||||
|         if (object == m_objects_metadata.end()) { | ||||
|             add_error("Can not assign volume mesh to a valid object"); | ||||
|             return false; | ||||
|         } | ||||
|         auto &volumes = object->second.volumes; | ||||
|         if (volumes.empty()) { | ||||
|             add_error("Can not assign mesh to a valid volume"); | ||||
|             return false; | ||||
|         } | ||||
|         ObjectMetadata::VolumeMetadata &volume = volumes.back(); | ||||
|         volume.shape_configuration = read_emboss_shape(attributes, num_attributes); | ||||
|         if (!volume.shape_configuration.has_value()) | ||||
|             return false; | ||||
| 
 | ||||
|         // Fill svg file content into shape_configuration
 | ||||
|         std::optional<EmbossShape::SvgFile> &svg = volume.shape_configuration->svg_file; | ||||
|         if (!svg.has_value()) | ||||
|             return true; // do not contain svg file
 | ||||
| 
 | ||||
|         const std::string &path = svg->path_in_3mf; | ||||
|         if (path.empty())  | ||||
|             return true; // do not contain svg file
 | ||||
| 
 | ||||
|         auto it = m_path_to_emboss_shape_files.find(path); | ||||
|         if (it == m_path_to_emboss_shape_files.end()) | ||||
|             return true; // svg file is not loaded yet
 | ||||
| 
 | ||||
|         svg->file_data = it->second; | ||||
|         return true; | ||||
|     } | ||||
| 
 | ||||
|     bool _BBS_3MF_Importer::_create_object_instance(std::string const & path, int object_id, const Transform3d& transform, const bool printable, unsigned int recur_counter) | ||||
|     { | ||||
|         static const unsigned int MAX_RECURSIONS = 10; | ||||
|  | @ -4169,6 +4357,13 @@ void PlateData::parse_filament_info(GCodeProcessorResult *result) | |||
| 
 | ||||
|         ObjectMetadata::VolumeMetadata &volume = object->second.volumes[m_curr_config.volume_id]; | ||||
| 
 | ||||
|         if (volume.text_configuration.has_value()) { | ||||
|             add_error("Both text_info and text_configuration found, ignore legacy text_info"); | ||||
|             return true; | ||||
|         } | ||||
| 
 | ||||
|         // TODO: Orca: support legacy text info
 | ||||
|         /*
 | ||||
|         TextInfo text_info; | ||||
|         text_info.m_text      = xml_unescape(bbs_get_attribute_value_string(attributes, num_attributes, TEXT_ATTR)); | ||||
|         text_info.m_font_name = bbs_get_attribute_value_string(attributes, num_attributes, FONT_NAME_ATTR); | ||||
|  | @ -4196,7 +4391,7 @@ void PlateData::parse_filament_info(GCodeProcessorResult *result) | |||
|         if (!hit_normal.empty()) | ||||
|             text_info.m_rr.normal = get_vec3_from_string(hit_normal); | ||||
| 
 | ||||
|         volume.text_info = text_info; | ||||
|         volume.text_info = text_info;*/ | ||||
|         return true; | ||||
|     } | ||||
| 
 | ||||
|  | @ -4474,8 +4669,10 @@ void PlateData::parse_filament_info(GCodeProcessorResult *result) | |||
| 
 | ||||
|             volume->set_type(volume_data->part_type); | ||||
|              | ||||
|             if (!volume_data->text_info.m_text.empty()) | ||||
|                 volume->set_text_info(volume_data->text_info); | ||||
|             if (auto &es = volume_data->shape_configuration; es.has_value()) | ||||
|                 volume->emboss_shape = std::move(es);             | ||||
|             if (auto &tc = volume_data->text_configuration; tc.has_value()) | ||||
|                 volume->text_configuration = std::move(tc); | ||||
| 
 | ||||
|             // apply the remaining volume's metadata
 | ||||
|             for (const Metadata& metadata : volume_data->metadata) { | ||||
|  | @ -4519,7 +4716,7 @@ void PlateData::parse_filament_info(GCodeProcessorResult *result) | |||
| 
 | ||||
|         return true; | ||||
|     } | ||||
| 
 | ||||
|     /*
 | ||||
|     bool _BBS_3MF_Importer::_generate_volumes(ModelObject& object, const Geometry& geometry, const ObjectMetadata::VolumeMetadataList& volumes, ConfigSubstitutionContext& config_substitutions) | ||||
|     { | ||||
|         if (!object.volumes.empty()) { | ||||
|  | @ -4667,7 +4864,7 @@ void PlateData::parse_filament_info(GCodeProcessorResult *result) | |||
| 
 | ||||
|         return true; | ||||
|     } | ||||
| 
 | ||||
|     */ | ||||
|     void XMLCALL _BBS_3MF_Importer::_handle_start_model_xml_element(void* userData, const char* name, const char** attributes) | ||||
|     { | ||||
|         _BBS_3MF_Importer* importer = (_BBS_3MF_Importer*)userData; | ||||
|  | @ -5242,6 +5439,7 @@ void PlateData::parse_filament_info(GCodeProcessorResult *result) | |||
|         bool save_model_to_file(StoreParams& store_params); | ||||
|         // add backup logic
 | ||||
|         bool save_object_mesh(const std::string& temp_path, ModelObject const & object, int obj_id); | ||||
|         static void add_transformation(std::stringstream &stream, const Transform3d &tr); | ||||
| 
 | ||||
|     private: | ||||
|         //BBS: add plate data related logic
 | ||||
|  | @ -6687,6 +6885,16 @@ void PlateData::parse_filament_info(GCodeProcessorResult *result) | |||
|         return flush(output_buffer, true); | ||||
|     } | ||||
| 
 | ||||
|     void _BBS_3MF_Exporter::add_transformation(std::stringstream &stream, const Transform3d &tr) | ||||
|     { | ||||
|         for (unsigned c = 0; c < 4; ++c) { | ||||
|             for (unsigned r = 0; r < 3; ++r) { | ||||
|                 stream << tr(r, c); | ||||
|                 if (r != 2 || c != 3) stream << " "; | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     bool _BBS_3MF_Exporter::_add_build_to_model_stream(std::stringstream& stream, const BuildItemsList& build_items) const | ||||
|     { | ||||
|         // This happens for empty projects
 | ||||
|  | @ -6707,13 +6915,7 @@ void PlateData::parse_filament_info(GCodeProcessorResult *result) | |||
|             if (!item.path.empty()) | ||||
|                 stream << "\" " << PPATH_ATTR << "=\"" << xml_escape(item.path); | ||||
|             stream << "\" " << TRANSFORM_ATTR << "=\""; | ||||
|             for (unsigned c = 0; c < 4; ++c) { | ||||
|                 for (unsigned r = 0; r < 3; ++r) { | ||||
|                     stream << item.transform(r, c); | ||||
|                     if (r != 2 || c != 3) | ||||
|                         stream << " "; | ||||
|                 } | ||||
|             } | ||||
|             add_transformation(stream, item.transform); | ||||
|             stream << "\" " << PRINTABLE_ATTR << "=\"" << item.printable << "\"/>\n"; | ||||
|         } | ||||
| 
 | ||||
|  | @ -7041,38 +7243,6 @@ void PlateData::parse_filament_info(GCodeProcessorResult *result) | |||
|         return true; | ||||
|     } | ||||
| 
 | ||||
|     void _add_text_info_to_archive(std::stringstream& stream, const TextInfo& text_info) { | ||||
|         stream << "      <" << TEXT_INFO_TAG << " "; | ||||
| 
 | ||||
|         stream << TEXT_ATTR << "=\"" << xml_escape(text_info.m_text) << "\" "; | ||||
|         stream << FONT_NAME_ATTR << "=\"" << text_info.m_font_name << "\" "; | ||||
| 
 | ||||
|         stream << FONT_INDEX_ATTR << "=\"" << text_info.m_curr_font_idx << "\" "; | ||||
| 
 | ||||
|         stream << FONT_SIZE_ATTR << "=\"" << text_info.m_font_size << "\" "; | ||||
|         stream << THICKNESS_ATTR << "=\"" << text_info.m_thickness << "\" "; | ||||
|         stream << EMBEDED_DEPTH_ATTR << "=\"" << text_info.m_embeded_depth << "\" "; | ||||
|         stream << ROTATE_ANGLE_ATTR << "=\"" << text_info.m_rotate_angle << "\" "; | ||||
|         stream << TEXT_GAP_ATTR << "=\"" << text_info.m_text_gap << "\" "; | ||||
| 
 | ||||
|         stream << BOLD_ATTR << "=\"" << (text_info.m_bold ? 1 : 0) << "\" "; | ||||
|         stream << ITALIC_ATTR << "=\"" << (text_info.m_italic ? 1 : 0) << "\" "; | ||||
|         stream << SURFACE_TEXT_ATTR << "=\"" << (text_info.m_is_surface_text ? 1 : 0) << "\" "; | ||||
|         stream << KEEP_HORIZONTAL_ATTR << "=\"" << (text_info.m_keep_horizontal ? 1 : 0) << "\" "; | ||||
| 
 | ||||
|         stream << HIT_MESH_ATTR << "=\"" << text_info.m_rr.mesh_id << "\" "; | ||||
| 
 | ||||
|         stream << HIT_POSITION_ATTR << "=\""; | ||||
|         add_vec3(stream, text_info.m_rr.hit); | ||||
|         stream << "\" "; | ||||
| 
 | ||||
|         stream << HIT_NORMAL_ATTR << "=\""; | ||||
|         add_vec3(stream, text_info.m_rr.normal); | ||||
|         stream << "\" "; | ||||
| 
 | ||||
|         stream << "/>\n"; | ||||
|     } | ||||
| 
 | ||||
|     bool _BBS_3MF_Exporter::_add_model_config_file_to_archive(mz_zip_archive& archive, const Model& model, PlateDataPtrs& plate_data_list, const ObjectToObjectDataMap &objects_data, int export_plate_idx, bool save_gcode, bool use_loaded_id) | ||||
|     { | ||||
|         std::stringstream stream; | ||||
|  | @ -7170,9 +7340,13 @@ void PlateData::parse_filament_info(GCodeProcessorResult *result) | |||
|                                 stream << "      <" << METADATA_TAG << " "<< KEY_ATTR << "=\"" << key << "\" " << VALUE_ATTR << "=\"" << volume->config.opt_serialize(key) << "\"/>\n"; | ||||
|                             } | ||||
| 
 | ||||
|                             const TextInfo &text_info = volume->get_text_info(); | ||||
|                             if (!text_info.m_text.empty()) | ||||
|                                 _add_text_info_to_archive(stream, text_info); | ||||
|                             if (const std::optional<EmbossShape> &es = volume->emboss_shape; | ||||
|                                 es.has_value()) | ||||
|                                 to_xml(stream, *es, *volume, archive); | ||||
|                      | ||||
|                             if (const std::optional<TextConfiguration> &tc = volume->text_configuration; | ||||
|                                 tc.has_value()) | ||||
|                                 TextConfigurationSerialization::to_xml(stream, *tc); | ||||
| 
 | ||||
|                             // stores mesh's statistics
 | ||||
|                             const RepairedMeshErrors& stats = volume->mesh().stats().repaired_errors; | ||||
|  | @ -8163,4 +8337,335 @@ SaveObjectGaurd::~SaveObjectGaurd() | |||
|     _BBS_Backup_Manager::get().pop_object_gaurd(); | ||||
| } | ||||
| 
 | ||||
| namespace{ | ||||
| 
 | ||||
| // Conversion with bidirectional map
 | ||||
| // F .. first, S .. second
 | ||||
| template<typename F, typename S> | ||||
| F bimap_cvt(const boost::bimap<F, S> &bmap, S s, const F & def_value) { | ||||
|     const auto &map = bmap.right; | ||||
|     auto found_item = map.find(s); | ||||
| 
 | ||||
|     // only for back and forward compatibility
 | ||||
|     assert(found_item != map.end());  | ||||
|     if (found_item == map.end()) | ||||
|         return def_value; | ||||
| 
 | ||||
|     return found_item->second; | ||||
| } | ||||
| 
 | ||||
| template<typename F, typename S>  | ||||
| S bimap_cvt(const boost::bimap<F, S> &bmap, F f, const S &def_value) | ||||
| { | ||||
|     const auto &map = bmap.left; | ||||
|     auto found_item = map.find(f); | ||||
| 
 | ||||
|     // only for back and forward compatibility
 | ||||
|     assert(found_item != map.end()); | ||||
|     if (found_item == map.end()) | ||||
|         return def_value; | ||||
| 
 | ||||
|     return found_item->second; | ||||
| } | ||||
| 
 | ||||
| } // namespace
 | ||||
| 
 | ||||
| /// <summary>
 | ||||
| /// TextConfiguration serialization
 | ||||
| /// </summary>
 | ||||
| const TextConfigurationSerialization::TypeToName TextConfigurationSerialization::type_to_name = | ||||
|     boost::assign::list_of<TypeToName::relation> | ||||
|     (EmbossStyle::Type::file_path, "file_name") | ||||
|     (EmbossStyle::Type::wx_win_font_descr, "wxFontDescriptor_Windows") | ||||
|     (EmbossStyle::Type::wx_lin_font_descr, "wxFontDescriptor_Linux") | ||||
|     (EmbossStyle::Type::wx_mac_font_descr, "wxFontDescriptor_MacOsX"); | ||||
| 
 | ||||
| const TextConfigurationSerialization::HorizontalAlignToName TextConfigurationSerialization::horizontal_align_to_name = | ||||
|     boost::assign::list_of<HorizontalAlignToName::relation> | ||||
|     (FontProp::HorizontalAlign::left, "left") | ||||
|     (FontProp::HorizontalAlign::center, "center") | ||||
|     (FontProp::HorizontalAlign::right, "right"); | ||||
| 
 | ||||
| const TextConfigurationSerialization::VerticalAlignToName TextConfigurationSerialization::vertical_align_to_name = | ||||
|     boost::assign::list_of<VerticalAlignToName::relation> | ||||
|     (FontProp::VerticalAlign::top, "top") | ||||
|     (FontProp::VerticalAlign::center, "middle") | ||||
|     (FontProp::VerticalAlign::bottom, "bottom"); | ||||
| 
 | ||||
| 
 | ||||
| void TextConfigurationSerialization::to_xml(std::stringstream &stream, const TextConfiguration &tc) | ||||
| { | ||||
|     stream << "   <" << TEXT_TAG << " "; | ||||
| 
 | ||||
|     stream << TEXT_DATA_ATTR << "=\"" << xml_escape_double_quotes_attribute_value(tc.text) << "\" "; | ||||
|     // font item
 | ||||
|     const EmbossStyle &style = tc.style; | ||||
|     stream << STYLE_NAME_ATTR <<  "=\"" << xml_escape_double_quotes_attribute_value(style.name) << "\" "; | ||||
|     stream << FONT_DESCRIPTOR_ATTR << "=\"" << xml_escape_double_quotes_attribute_value(style.path) << "\" "; | ||||
|     constexpr std::string_view dafault_type{"undefined"}; | ||||
|     std::string_view style_type = bimap_cvt(type_to_name, style.type, dafault_type); | ||||
|     stream << FONT_DESCRIPTOR_TYPE_ATTR << "=\"" << style_type << "\" "; | ||||
| 
 | ||||
|     // font property
 | ||||
|     const FontProp &fp = tc.style.prop; | ||||
|     if (fp.char_gap.has_value()) | ||||
|         stream << CHAR_GAP_ATTR << "=\"" << *fp.char_gap << "\" "; | ||||
|     if (fp.line_gap.has_value()) | ||||
|         stream << LINE_GAP_ATTR << "=\"" << *fp.line_gap << "\" "; | ||||
| 
 | ||||
|     stream << LINE_HEIGHT_ATTR << "=\"" << fp.size_in_mm << "\" "; | ||||
|     if (fp.boldness.has_value()) | ||||
|         stream << BOLDNESS_ATTR << "=\"" << *fp.boldness << "\" "; | ||||
|     if (fp.skew.has_value()) | ||||
|         stream << SKEW_ATTR << "=\"" << *fp.skew << "\" "; | ||||
|     if (fp.per_glyph) | ||||
|         stream << PER_GLYPH_ATTR << "=\"" << 1 << "\" "; | ||||
|     stream << HORIZONTAL_ALIGN_ATTR << "=\"" << bimap_cvt(horizontal_align_to_name, fp.align.first, dafault_type) << "\" "; | ||||
|     stream << VERTICAL_ALIGN_ATTR   << "=\"" << bimap_cvt(vertical_align_to_name,  fp.align.second, dafault_type) << "\" "; | ||||
|     if (fp.collection_number.has_value()) | ||||
|         stream << COLLECTION_NUMBER_ATTR << "=\"" << *fp.collection_number << "\" "; | ||||
|     // font descriptor
 | ||||
|     if (fp.family.has_value()) | ||||
|         stream << FONT_FAMILY_ATTR << "=\"" << *fp.family << "\" "; | ||||
|     if (fp.face_name.has_value()) | ||||
|         stream << FONT_FACE_NAME_ATTR << "=\"" << *fp.face_name << "\" "; | ||||
|     if (fp.style.has_value()) | ||||
|         stream << FONT_STYLE_ATTR << "=\"" << *fp.style << "\" "; | ||||
|     if (fp.weight.has_value()) | ||||
|         stream << FONT_WEIGHT_ATTR << "=\"" << *fp.weight << "\" "; | ||||
| 
 | ||||
|     stream << "/>\n"; // end TEXT_TAG
 | ||||
| } | ||||
| namespace { | ||||
| 
 | ||||
| FontProp::HorizontalAlign read_horizontal_align(const char **attributes, unsigned int num_attributes, const TextConfigurationSerialization::HorizontalAlignToName& horizontal_align_to_name){ | ||||
|     std::string horizontal_align_str = bbs_get_attribute_value_string(attributes, num_attributes, HORIZONTAL_ALIGN_ATTR); | ||||
| 
 | ||||
|     // Back compatibility
 | ||||
|     // PS 2.6.0 do not have align
 | ||||
|     if (horizontal_align_str.empty()) | ||||
|         return FontProp::HorizontalAlign::center; | ||||
| 
 | ||||
|     // Back compatibility
 | ||||
|     // PS 2.6.1 store indices(0|1|2) instead of text for align
 | ||||
|     if (horizontal_align_str.length() == 1) { | ||||
|         int horizontal_align_int = 0; | ||||
|         if(boost::spirit::qi::parse(horizontal_align_str.c_str(), horizontal_align_str.c_str() + 1, boost::spirit::qi::int_, horizontal_align_int)) | ||||
|             return static_cast<FontProp::HorizontalAlign>(horizontal_align_int); | ||||
|     } | ||||
| 
 | ||||
|     return bimap_cvt(horizontal_align_to_name, std::string_view(horizontal_align_str), FontProp::HorizontalAlign::center);     | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| FontProp::VerticalAlign read_vertical_align(const char **attributes, unsigned int num_attributes, const TextConfigurationSerialization::VerticalAlignToName& vertical_align_to_name){ | ||||
|     std::string vertical_align_str = bbs_get_attribute_value_string(attributes, num_attributes, VERTICAL_ALIGN_ATTR); | ||||
| 
 | ||||
|     // Back compatibility
 | ||||
|     // PS 2.6.0 do not have align
 | ||||
|     if (vertical_align_str.empty()) | ||||
|         return FontProp::VerticalAlign::center; | ||||
| 
 | ||||
|     // Back compatibility
 | ||||
|     // PS 2.6.1 store indices(0|1|2) instead of text for align
 | ||||
|     if (vertical_align_str.length() == 1) { | ||||
|         int vertical_align_int = 0; | ||||
|         if(boost::spirit::qi::parse(vertical_align_str.c_str(), vertical_align_str.c_str() + 1, boost::spirit::qi::int_, vertical_align_int)) | ||||
|             return static_cast<FontProp::VerticalAlign>(vertical_align_int); | ||||
|     } | ||||
| 
 | ||||
|     return bimap_cvt(vertical_align_to_name, std::string_view(vertical_align_str), FontProp::VerticalAlign::center); | ||||
| } | ||||
| 
 | ||||
| } // namespace
 | ||||
| 
 | ||||
| std::optional<TextConfiguration> TextConfigurationSerialization::read(const char **attributes, unsigned int num_attributes) | ||||
| { | ||||
|     FontProp fp; | ||||
|     int char_gap = bbs_get_attribute_value_int(attributes, num_attributes, CHAR_GAP_ATTR); | ||||
|     if (char_gap != 0) fp.char_gap = char_gap; | ||||
|     int line_gap = bbs_get_attribute_value_int(attributes, num_attributes, LINE_GAP_ATTR);  | ||||
|     if (line_gap != 0) fp.line_gap = line_gap; | ||||
|     float boldness = bbs_get_attribute_value_float(attributes, num_attributes, BOLDNESS_ATTR); | ||||
|     if (std::fabs(boldness) > std::numeric_limits<float>::epsilon()) | ||||
|         fp.boldness = boldness; | ||||
|     float skew = bbs_get_attribute_value_float(attributes, num_attributes, SKEW_ATTR); | ||||
|     if (std::fabs(skew) > std::numeric_limits<float>::epsilon()) | ||||
|         fp.skew = skew; | ||||
|     int per_glyph = bbs_get_attribute_value_int(attributes, num_attributes, PER_GLYPH_ATTR); | ||||
|     if (per_glyph == 1) fp.per_glyph = true; | ||||
| 
 | ||||
|     fp.align = FontProp::Align( | ||||
|         read_horizontal_align(attributes, num_attributes, horizontal_align_to_name), | ||||
|         read_vertical_align(attributes, num_attributes, vertical_align_to_name)); | ||||
| 
 | ||||
|     int collection_number = bbs_get_attribute_value_int(attributes, num_attributes, COLLECTION_NUMBER_ATTR); | ||||
|     if (collection_number > 0) fp.collection_number = static_cast<unsigned int>(collection_number); | ||||
| 
 | ||||
|     fp.size_in_mm = bbs_get_attribute_value_float(attributes, num_attributes, LINE_HEIGHT_ATTR); | ||||
| 
 | ||||
|     std::string family = bbs_get_attribute_value_string(attributes, num_attributes, FONT_FAMILY_ATTR); | ||||
|     if (!family.empty()) fp.family = family; | ||||
|     std::string face_name = bbs_get_attribute_value_string(attributes, num_attributes, FONT_FACE_NAME_ATTR); | ||||
|     if (!face_name.empty()) fp.face_name = face_name; | ||||
|     std::string style = bbs_get_attribute_value_string(attributes, num_attributes, FONT_STYLE_ATTR); | ||||
|     if (!style.empty()) fp.style = style; | ||||
|     std::string weight = bbs_get_attribute_value_string(attributes, num_attributes, FONT_WEIGHT_ATTR); | ||||
|     if (!weight.empty()) fp.weight = weight; | ||||
| 
 | ||||
|     std::string style_name = bbs_get_attribute_value_string(attributes, num_attributes, STYLE_NAME_ATTR); | ||||
|     std::string font_descriptor = bbs_get_attribute_value_string(attributes, num_attributes, FONT_DESCRIPTOR_ATTR); | ||||
|     std::string type_str = bbs_get_attribute_value_string(attributes, num_attributes, FONT_DESCRIPTOR_TYPE_ATTR); | ||||
|     EmbossStyle::Type type = bimap_cvt(type_to_name, std::string_view{type_str}, EmbossStyle::Type::undefined); | ||||
| 
 | ||||
|     std::string text = bbs_get_attribute_value_string(attributes, num_attributes, TEXT_DATA_ATTR); | ||||
|     EmbossStyle es{style_name, std::move(font_descriptor), type, std::move(fp)}; | ||||
|     return TextConfiguration{std::move(es), std::move(text)}; | ||||
| } | ||||
| 
 | ||||
| EmbossShape TextConfigurationSerialization::read_old(const char **attributes, unsigned int num_attributes) | ||||
| { | ||||
|     EmbossShape es; | ||||
|     std::string fix_tr_mat_str = bbs_get_attribute_value_string(attributes, num_attributes, TRANSFORM_ATTR); | ||||
|     if (!fix_tr_mat_str.empty()) | ||||
|         es.fix_3mf_tr = bbs_get_transform_from_3mf_specs_string(fix_tr_mat_str); | ||||
| 
 | ||||
| 
 | ||||
|     if (bbs_get_attribute_value_int(attributes, num_attributes, USE_SURFACE_ATTR) == 1) | ||||
|         es.projection.use_surface = true; | ||||
| 
 | ||||
|     es.projection.depth = bbs_get_attribute_value_float(attributes, num_attributes, DEPTH_ATTR); | ||||
| 
 | ||||
|     int use_surface = bbs_get_attribute_value_int(attributes, num_attributes, USE_SURFACE_ATTR); | ||||
|     if (use_surface == 1) | ||||
|         es.projection.use_surface = true; | ||||
| 
 | ||||
|     return es; | ||||
| } | ||||
| 
 | ||||
| namespace { | ||||
| Transform3d create_fix(const std::optional<Transform3d> &prev, const ModelVolume &volume) | ||||
| { | ||||
|     // IMPROVE: check if volume was modified (translated, rotated OR scaled)
 | ||||
|     // when no change do not calculate transformation only store original fix matrix
 | ||||
| 
 | ||||
|     // Create transformation used after load actual stored volume
 | ||||
|     // Orca: do not bake volume transformation into meshes
 | ||||
|     // const Transform3d &actual_trmat = volume.get_matrix();
 | ||||
|     const Transform3d& actual_trmat = Transform3d::Identity(); | ||||
| 
 | ||||
|     const auto &vertices = volume.mesh().its.vertices; | ||||
|     Vec3d       min      = actual_trmat * vertices.front().cast<double>(); | ||||
|     Vec3d       max      = min; | ||||
|     for (const Vec3f &v : vertices) { | ||||
|         Vec3d vd = actual_trmat * v.cast<double>(); | ||||
|         for (size_t i = 0; i < 3; ++i) { | ||||
|             if (min[i] > vd[i]) | ||||
|                 min[i] = vd[i]; | ||||
|             if (max[i] < vd[i]) | ||||
|                 max[i] = vd[i]; | ||||
|         } | ||||
|     } | ||||
|     Vec3d       center     = (max + min) / 2; | ||||
|     Transform3d post_trmat = Transform3d::Identity(); | ||||
|     post_trmat.translate(center); | ||||
| 
 | ||||
|     Transform3d fix_trmat = actual_trmat.inverse() * post_trmat; | ||||
|     if (!prev.has_value()) | ||||
|         return fix_trmat; | ||||
| 
 | ||||
|     // check whether fix somehow differ previous
 | ||||
|     if (fix_trmat.isApprox(Transform3d::Identity(), 1e-5)) | ||||
|         return *prev; | ||||
| 
 | ||||
|     return *prev * fix_trmat; | ||||
| } | ||||
| 
 | ||||
| bool to_xml(std::stringstream &stream, const EmbossShape::SvgFile &svg, const ModelVolume &volume, mz_zip_archive &archive){ | ||||
|     if (svg.path_in_3mf.empty()) | ||||
|         return true; // EmbossedText OR unwanted store .svg file into .3mf (protection of copyRight)
 | ||||
| 
 | ||||
|     if (!svg.path.empty()) | ||||
|         stream << SVG_FILE_PATH_ATTR << "=\"" << xml_escape_double_quotes_attribute_value(svg.path) << "\" "; | ||||
|     stream << SVG_FILE_PATH_IN_3MF_ATTR << "=\"" << xml_escape_double_quotes_attribute_value(svg.path_in_3mf) << "\" "; | ||||
| 
 | ||||
|     std::shared_ptr<std::string> file_data = svg.file_data; | ||||
|     assert(file_data != nullptr);  | ||||
|     if (file_data == nullptr && !svg.path.empty()) | ||||
|         file_data = read_from_disk(svg.path); | ||||
|     if (file_data == nullptr) { | ||||
|         BOOST_LOG_TRIVIAL(warning) << "Can't write svg file no filedata"; | ||||
|         return false; | ||||
|     } | ||||
|     const std::string &file_data_str = *file_data;  | ||||
| 
 | ||||
|     return mz_zip_writer_add_mem(&archive, svg.path_in_3mf.c_str(),  | ||||
|         (const void *) file_data_str.c_str(), file_data_str.size(), MZ_DEFAULT_COMPRESSION); | ||||
| } | ||||
| 
 | ||||
| } // namespace
 | ||||
| 
 | ||||
| void to_xml(std::stringstream &stream, const EmbossShape &es, const ModelVolume &volume, mz_zip_archive &archive) | ||||
| { | ||||
|     stream << "   <" << SHAPE_TAG << " "; | ||||
|     if (es.svg_file.has_value()) | ||||
|         if(!to_xml(stream, *es.svg_file, volume, archive)) | ||||
|             BOOST_LOG_TRIVIAL(warning) << "Can't write svg file defiden embossed shape into 3mf"; | ||||
|      | ||||
|     stream << SHAPE_SCALE_ATTR << "=\"" << es.scale << "\" "; | ||||
| 
 | ||||
|     if (!es.final_shape.is_healed) | ||||
|         stream << UNHEALED_ATTR << "=\"" << 1 << "\" "; | ||||
| 
 | ||||
|     // projection
 | ||||
|     const EmbossProjection &p = es.projection; | ||||
|     stream << DEPTH_ATTR << "=\"" << p.depth << "\" "; | ||||
|     if (p.use_surface) | ||||
|         stream << USE_SURFACE_ATTR << "=\"" << 1 << "\" "; | ||||
|      | ||||
|     // FIX of baked transformation
 | ||||
|     Transform3d fix = create_fix(es.fix_3mf_tr, volume); | ||||
|     stream << TRANSFORM_ATTR << "=\""; | ||||
|     _BBS_3MF_Exporter::add_transformation(stream, fix); | ||||
|     stream << "\" "; | ||||
| 
 | ||||
|     stream << "/>\n"; // end SHAPE_TAG    
 | ||||
| } | ||||
| 
 | ||||
| std::optional<EmbossShape> read_emboss_shape(const char **attributes, unsigned int num_attributes) {     | ||||
|     double scale = bbs_get_attribute_value_float(attributes, num_attributes, SHAPE_SCALE_ATTR); | ||||
|     int unhealed = bbs_get_attribute_value_int(attributes, num_attributes, UNHEALED_ATTR); | ||||
|     bool is_healed = unhealed != 1; | ||||
| 
 | ||||
|     EmbossProjection projection; | ||||
|     projection.depth = bbs_get_attribute_value_float(attributes, num_attributes, DEPTH_ATTR); | ||||
|     if (is_approx(projection.depth, 0.)) | ||||
|         projection.depth = 10.; | ||||
| 
 | ||||
|     int use_surface  = bbs_get_attribute_value_int(attributes, num_attributes, USE_SURFACE_ATTR); | ||||
|     if (use_surface == 1) | ||||
|         projection.use_surface = true;      | ||||
| 
 | ||||
|     std::optional<Transform3d> fix_tr_mat; | ||||
|     std::string fix_tr_mat_str = bbs_get_attribute_value_string(attributes, num_attributes, TRANSFORM_ATTR); | ||||
|     if (!fix_tr_mat_str.empty()) {  | ||||
|         fix_tr_mat = bbs_get_transform_from_3mf_specs_string(fix_tr_mat_str); | ||||
|     } | ||||
| 
 | ||||
|     std::string file_path = bbs_get_attribute_value_string(attributes, num_attributes, SVG_FILE_PATH_ATTR); | ||||
|     std::string file_path_3mf = bbs_get_attribute_value_string(attributes, num_attributes, SVG_FILE_PATH_IN_3MF_ATTR); | ||||
| 
 | ||||
|     // MayBe: store also shapes to not store svg
 | ||||
|     // But be carefull curve will be lost -> scale will not change sampling
 | ||||
|     // shapes could be loaded from SVG
 | ||||
|     ExPolygonsWithIds shapes;  | ||||
|     // final shape could be calculated from shapes
 | ||||
|     HealedExPolygons final_shape; | ||||
|     final_shape.is_healed = is_healed; | ||||
| 
 | ||||
|     EmbossShape::SvgFile svg{file_path, file_path_3mf}; | ||||
|     return EmbossShape{std::move(shapes), std::move(final_shape), scale, std::move(projection), std::move(fix_tr_mat), std::move(svg)}; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| } // namespace Slic3r
 | ||||
|  |  | |||
|  | @ -397,42 +397,12 @@ Transform3d scale_transform(const Vec3d& scale) | |||
| 
 | ||||
| Vec3d extract_euler_angles(const Eigen::Matrix<double, 3, 3, Eigen::DontAlign>& rotation_matrix) | ||||
| { | ||||
|     // reference: http://www.gregslabaugh.net/publications/euler.pdf
 | ||||
|     Vec3d angles1 = Vec3d::Zero(); | ||||
|     Vec3d angles2 = Vec3d::Zero(); | ||||
|     // BBS: rotation_matrix(2, 0) may be slighterly larger than 1 due to numerical accuracy
 | ||||
|     if (std::abs(std::abs(rotation_matrix(2, 0)) - 1.0) < 1e-5 || std::abs(rotation_matrix(2, 0))>1) { | ||||
|         angles1.z() = 0.0; | ||||
|         if (rotation_matrix(2, 0) < 0.0) { // == -1.0
 | ||||
|             angles1.y() = 0.5 * double(PI); | ||||
|             angles1.x() = angles1.z() + ::atan2(rotation_matrix(0, 1), rotation_matrix(0, 2)); | ||||
|         } | ||||
|         else { // == 1.0
 | ||||
|             angles1.y() = - 0.5 * double(PI); | ||||
|             angles1.x() = - angles1.y() + ::atan2(- rotation_matrix(0, 1), - rotation_matrix(0, 2)); | ||||
|         } | ||||
|         angles2 = angles1; | ||||
|     } | ||||
|     else { | ||||
|         angles1.y() = -::asin(rotation_matrix(2, 0)); | ||||
|         const double inv_cos1 = 1.0 / ::cos(angles1.y()); | ||||
|         angles1.x() = ::atan2(rotation_matrix(2, 1) * inv_cos1, rotation_matrix(2, 2) * inv_cos1); | ||||
|         angles1.z() = ::atan2(rotation_matrix(1, 0) * inv_cos1, rotation_matrix(0, 0) * inv_cos1); | ||||
| 
 | ||||
|         angles2.y() = double(PI) - angles1.y(); | ||||
|         const double inv_cos2 = 1.0 / ::cos(angles2.y()); | ||||
|         angles2.x() = ::atan2(rotation_matrix(2, 1) * inv_cos2, rotation_matrix(2, 2) * inv_cos2); | ||||
|         angles2.z() = ::atan2(rotation_matrix(1, 0) * inv_cos2, rotation_matrix(0, 0) * inv_cos2); | ||||
|     } | ||||
| 
 | ||||
|     // The following euristic is the best found up to now (in the sense that it works fine with the greatest number of edge use-cases)
 | ||||
|     // but there are other use-cases were it does not
 | ||||
|     // We need to improve it
 | ||||
|     const double min_1 = angles1.cwiseAbs().minCoeff(); | ||||
|     const double min_2 = angles2.cwiseAbs().minCoeff(); | ||||
|     const bool use_1 = (min_1 < min_2) || (is_approx(min_1, min_2) && (angles1.norm() <= angles2.norm())); | ||||
| 
 | ||||
|     return use_1 ? angles1 : angles2; | ||||
|     // The extracted "rotation" is a triplet of numbers such that Geometry::rotation_transform
 | ||||
|     // returns the original transform. Because of the chosen order of rotations, the triplet
 | ||||
|     // is not equivalent to Euler angles in the usual sense.
 | ||||
|     Vec3d angles = rotation_matrix.eulerAngles(2,1,0); | ||||
|     std::swap(angles(0), angles(2)); | ||||
|     return angles; | ||||
| } | ||||
| 
 | ||||
| Vec3d extract_euler_angles(const Transform3d& transform) | ||||
|  | @ -446,14 +416,6 @@ Vec3d extract_euler_angles(const Transform3d& transform) | |||
|     return extract_euler_angles(m); | ||||
| } | ||||
| 
 | ||||
| static Transform3d extract_rotation_matrix(const Transform3d& trafo) | ||||
| { | ||||
|     Matrix3d rotation; | ||||
|     Matrix3d scale; | ||||
|     trafo.computeRotationScaling(&rotation, &scale); | ||||
|     return Transform3d(rotation); | ||||
| } | ||||
| 
 | ||||
| void rotation_from_two_vectors(Vec3d from, Vec3d to, Vec3d& rotation_axis, double& phi, Matrix3d* rotation_matrix) | ||||
| { | ||||
|     double epsilon = 1e-5; | ||||
|  | @ -488,42 +450,65 @@ void rotation_from_two_vectors(Vec3d from, Vec3d to, Vec3d& rotation_axis, doubl | |||
|     } | ||||
| } | ||||
| 
 | ||||
| bool Transformation::Flags::needs_update(bool dont_translate, bool dont_rotate, bool dont_scale, bool dont_mirror) const | ||||
| Transform3d Transformation::get_offset_matrix() const | ||||
| { | ||||
|     return (this->dont_translate != dont_translate) || (this->dont_rotate != dont_rotate) || (this->dont_scale != dont_scale) || (this->dont_mirror != dont_mirror); | ||||
|     return translation_transform(get_offset()); | ||||
| } | ||||
| 
 | ||||
| void Transformation::Flags::set(bool dont_translate, bool dont_rotate, bool dont_scale, bool dont_mirror) | ||||
| static Transform3d extract_rotation_matrix(const Transform3d& trafo) | ||||
| { | ||||
|     this->dont_translate = dont_translate; | ||||
|     this->dont_rotate = dont_rotate; | ||||
|     this->dont_scale = dont_scale; | ||||
|     this->dont_mirror = dont_mirror; | ||||
|     Matrix3d rotation; | ||||
|     Matrix3d scale; | ||||
|     trafo.computeRotationScaling(&rotation, &scale); | ||||
|     return Transform3d(rotation); | ||||
| } | ||||
| 
 | ||||
| Transformation::Transformation() | ||||
| static Transform3d extract_scale(const Transform3d& trafo) | ||||
| { | ||||
|     reset(); | ||||
|     Matrix3d rotation; | ||||
|     Matrix3d scale; | ||||
|     trafo.computeRotationScaling(&rotation, &scale); | ||||
|     return Transform3d(scale); | ||||
| } | ||||
| 
 | ||||
| Transformation::Transformation(const Transform3d& transform) | ||||
| static std::pair<Transform3d, Transform3d> extract_rotation_scale(const Transform3d& trafo) | ||||
| { | ||||
|     set_from_transform(transform); | ||||
|     Matrix3d rotation; | ||||
|     Matrix3d scale; | ||||
|     trafo.computeRotationScaling(&rotation, &scale); | ||||
|     return { Transform3d(rotation), Transform3d(scale) }; | ||||
| } | ||||
| 
 | ||||
| void Transformation::set_offset(const Vec3d& offset) | ||||
| static bool contains_skew(const Transform3d& trafo) | ||||
| { | ||||
|     set_offset(X, offset.x()); | ||||
|     set_offset(Y, offset.y()); | ||||
|     set_offset(Z, offset.z()); | ||||
|     Matrix3d rotation; | ||||
|     Matrix3d scale; | ||||
|     trafo.computeRotationScaling(&rotation, &scale); | ||||
| 
 | ||||
|     if (scale.isDiagonal()) | ||||
|       return false; | ||||
|      | ||||
|     if (scale.determinant() >= 0.0) | ||||
|       return true; | ||||
| 
 | ||||
|     // the matrix contains mirror
 | ||||
|     const Matrix3d ratio = scale.cwiseQuotient(trafo.matrix().block<3,3>(0,0)); | ||||
| 
 | ||||
|     auto check_skew = [&ratio](int i, int j, bool& skew) { | ||||
|       if (!std::isnan(ratio(i, j)) && !std::isnan(ratio(j, i))) | ||||
|         skew |= std::abs(ratio(i, j) * ratio(j, i) - 1.0) > EPSILON; | ||||
|     }; | ||||
| 
 | ||||
|     bool has_skew = false; | ||||
|     check_skew(0, 1, has_skew); | ||||
|     check_skew(0, 2, has_skew); | ||||
|     check_skew(1, 2, has_skew); | ||||
|     return has_skew; | ||||
| } | ||||
| 
 | ||||
| void Transformation::set_offset(Axis axis, double offset) | ||||
| Vec3d Transformation::get_rotation() const | ||||
| { | ||||
|     if (m_offset(axis) != offset) { | ||||
|         m_offset(axis) = offset; | ||||
|         m_dirty = true; | ||||
|     } | ||||
|     return extract_euler_angles(extract_rotation_matrix(m_matrix)); | ||||
| } | ||||
| 
 | ||||
| Transform3d Transformation::get_rotation_matrix() const | ||||
|  | @ -533,9 +518,9 @@ Transform3d Transformation::get_rotation_matrix() const | |||
| 
 | ||||
| void Transformation::set_rotation(const Vec3d& rotation) | ||||
| { | ||||
|     set_rotation(X, rotation.x()); | ||||
|     set_rotation(Y, rotation.y()); | ||||
|     set_rotation(Z, rotation.z()); | ||||
|     const Vec3d offset = get_offset(); | ||||
|     m_matrix = rotation_transform(rotation) * extract_scale(m_matrix); | ||||
|     m_matrix.translation() = offset; | ||||
| } | ||||
| 
 | ||||
| void Transformation::set_rotation(Axis axis, double rotation) | ||||
|  | @ -544,32 +529,88 @@ void Transformation::set_rotation(Axis axis, double rotation) | |||
|     if (is_approx(std::abs(rotation), 2.0 * double(PI))) | ||||
|         rotation = 0.0; | ||||
| 
 | ||||
|     if (m_rotation(axis) != rotation) { | ||||
|         m_rotation(axis) = rotation; | ||||
|         m_dirty = true; | ||||
|     } | ||||
|     auto [curr_rotation, scale] = extract_rotation_scale(m_matrix); | ||||
|     Vec3d angles = extract_euler_angles(curr_rotation); | ||||
|     angles[axis] = rotation; | ||||
| 
 | ||||
|     const Vec3d offset = get_offset(); | ||||
|     m_matrix = rotation_transform(angles) * scale; | ||||
|     m_matrix.translation() = offset; | ||||
| } | ||||
| 
 | ||||
| Vec3d Transformation::get_scaling_factor() const | ||||
| { | ||||
|     const Transform3d scale = extract_scale(m_matrix); | ||||
|     return { std::abs(scale(0, 0)), std::abs(scale(1, 1)), std::abs(scale(2, 2)) }; | ||||
| } | ||||
| 
 | ||||
| Transform3d Transformation::get_scaling_factor_matrix() const | ||||
| { | ||||
|     Transform3d scale = extract_scale(m_matrix); | ||||
|     scale(0, 0) = std::abs(scale(0, 0)); | ||||
|     scale(1, 1) = std::abs(scale(1, 1)); | ||||
|     scale(2, 2) = std::abs(scale(2, 2)); | ||||
|     return scale; | ||||
| } | ||||
| 
 | ||||
| void Transformation::set_scaling_factor(const Vec3d& scaling_factor) | ||||
| { | ||||
|     set_scaling_factor(X, scaling_factor.x()); | ||||
|     set_scaling_factor(Y, scaling_factor.y()); | ||||
|     set_scaling_factor(Z, scaling_factor.z()); | ||||
|     assert(scaling_factor.x() > 0.0 && scaling_factor.y() > 0.0 && scaling_factor.z() > 0.0); | ||||
| 
 | ||||
|     const Vec3d offset = get_offset(); | ||||
|     m_matrix = extract_rotation_matrix(m_matrix) * scale_transform(scaling_factor); | ||||
|     m_matrix.translation() = offset; | ||||
| } | ||||
| 
 | ||||
| void Transformation::set_scaling_factor(Axis axis, double scaling_factor) | ||||
| { | ||||
|     if (m_scaling_factor(axis) != std::abs(scaling_factor)) { | ||||
|         m_scaling_factor(axis) = std::abs(scaling_factor); | ||||
|         m_dirty = true; | ||||
|     } | ||||
|     assert(scaling_factor > 0.0); | ||||
| 
 | ||||
|     auto [rotation, scale] = extract_rotation_scale(m_matrix); | ||||
|     scale(axis, axis) = scaling_factor; | ||||
| 
 | ||||
|     const Vec3d offset = get_offset(); | ||||
|     m_matrix = rotation * scale; | ||||
|     m_matrix.translation() = offset; | ||||
| } | ||||
| 
 | ||||
| Vec3d Transformation::get_mirror() const | ||||
| { | ||||
|     const Transform3d scale = extract_scale(m_matrix); | ||||
|     return { scale(0, 0) / std::abs(scale(0, 0)), scale(1, 1) / std::abs(scale(1, 1)), scale(2, 2) / std::abs(scale(2, 2)) }; | ||||
| } | ||||
| 
 | ||||
| Transform3d Transformation::get_mirror_matrix() const | ||||
| { | ||||
|     Transform3d scale = extract_scale(m_matrix); | ||||
|     scale(0, 0) = scale(0, 0) / std::abs(scale(0, 0)); | ||||
|     scale(1, 1) = scale(1, 1) / std::abs(scale(1, 1)); | ||||
|     scale(2, 2) = scale(2, 2) / std::abs(scale(2, 2)); | ||||
|     return scale; | ||||
| } | ||||
| 
 | ||||
| void Transformation::set_mirror(const Vec3d& mirror) | ||||
| { | ||||
|     set_mirror(X, mirror.x()); | ||||
|     set_mirror(Y, mirror.y()); | ||||
|     set_mirror(Z, mirror.z()); | ||||
|     Vec3d copy(mirror); | ||||
|     const Vec3d abs_mirror = copy.cwiseAbs(); | ||||
|     for (int i = 0; i < 3; ++i) { | ||||
|         if (abs_mirror(i) == 0.0) | ||||
|             copy(i) = 1.0; | ||||
|         else if (abs_mirror(i) != 1.0) | ||||
|             copy(i) /= abs_mirror(i); | ||||
|     } | ||||
| 
 | ||||
|     auto [rotation, scale] = extract_rotation_scale(m_matrix); | ||||
|     const Vec3d curr_scales = { scale(0, 0), scale(1, 1), scale(2, 2) }; | ||||
|     const Vec3d signs = curr_scales.cwiseProduct(copy); | ||||
| 
 | ||||
|     if (signs[0] < 0.0) scale(0, 0) = -scale(0, 0); | ||||
|     if (signs[1] < 0.0) scale(1, 1) = -scale(1, 1); | ||||
|     if (signs[2] < 0.0) scale(2, 2) = -scale(2, 2); | ||||
| 
 | ||||
|     const Vec3d offset = get_offset(); | ||||
|     m_matrix = rotation * scale; | ||||
|     m_matrix.translation() = offset; | ||||
| } | ||||
| 
 | ||||
| void Transformation::set_mirror(Axis axis, double mirror) | ||||
|  | @ -580,75 +621,61 @@ void Transformation::set_mirror(Axis axis, double mirror) | |||
|     else if (abs_mirror != 1.0) | ||||
|         mirror /= abs_mirror; | ||||
| 
 | ||||
|     if (m_mirror(axis) != mirror) { | ||||
|         m_mirror(axis) = mirror; | ||||
|         m_dirty = true; | ||||
|     } | ||||
|     auto [rotation, scale] = extract_rotation_scale(m_matrix); | ||||
|     const double curr_scale = scale(axis, axis); | ||||
|     const double sign = curr_scale * mirror; | ||||
| 
 | ||||
|     if (sign < 0.0) scale(axis, axis) = -scale(axis, axis); | ||||
| 
 | ||||
|     const Vec3d offset = get_offset(); | ||||
|     m_matrix = rotation * scale; | ||||
|     m_matrix.translation() = offset; | ||||
| } | ||||
| 
 | ||||
| void Transformation::set_from_transform(const Transform3d& transform) | ||||
| bool Transformation::has_skew() const | ||||
| { | ||||
|     // offset
 | ||||
|     set_offset(transform.matrix().block(0, 3, 3, 1)); | ||||
| 
 | ||||
|     Eigen::Matrix<double, 3, 3, Eigen::DontAlign> m3x3 = transform.matrix().block(0, 0, 3, 3); | ||||
| 
 | ||||
|     // mirror
 | ||||
|     // it is impossible to reconstruct the original mirroring factors from a matrix,
 | ||||
|     // we can only detect if the matrix contains a left handed reference system
 | ||||
|     // in which case we reorient it back to right handed by mirroring the x axis
 | ||||
|     Vec3d mirror = Vec3d::Ones(); | ||||
|     if (m3x3.col(0).dot(m3x3.col(1).cross(m3x3.col(2))) < 0.0) { | ||||
|         mirror.x() = -1.0; | ||||
|         // remove mirror
 | ||||
|         m3x3.col(0) *= -1.0; | ||||
|     } | ||||
|     set_mirror(mirror); | ||||
| 
 | ||||
|     // scale
 | ||||
|     set_scaling_factor(Vec3d(m3x3.col(0).norm(), m3x3.col(1).norm(), m3x3.col(2).norm())); | ||||
| 
 | ||||
|     // remove scale
 | ||||
|     m3x3.col(0).normalize(); | ||||
|     m3x3.col(1).normalize(); | ||||
|     m3x3.col(2).normalize(); | ||||
| 
 | ||||
|     // rotation
 | ||||
|     set_rotation(extract_euler_angles(m3x3)); | ||||
| 
 | ||||
|     // forces matrix recalculation matrix
 | ||||
|     m_matrix = get_matrix(); | ||||
| 
 | ||||
| //    // debug check
 | ||||
| //    if (!m_matrix.isApprox(transform))
 | ||||
| //        std::cout << "something went wrong in extracting data from matrix" << std::endl;
 | ||||
|     return contains_skew(m_matrix); | ||||
| } | ||||
| 
 | ||||
| void Transformation::reset() | ||||
| { | ||||
|     m_offset = Vec3d::Zero(); | ||||
|     m_rotation = Vec3d::Zero(); | ||||
|     m_scaling_factor = Vec3d::Ones(); | ||||
|     m_mirror = Vec3d::Ones(); | ||||
|     m_matrix = Transform3d::Identity(); | ||||
|     m_dirty = false; | ||||
| } | ||||
| 
 | ||||
| const Transform3d& Transformation::get_matrix(bool dont_translate, bool dont_rotate, bool dont_scale, bool dont_mirror) const | ||||
| void Transformation::reset_rotation() | ||||
| { | ||||
|     if (m_dirty || m_flags.needs_update(dont_translate, dont_rotate, dont_scale, dont_mirror)) { | ||||
|         m_matrix = Geometry::assemble_transform( | ||||
|             dont_translate ? Vec3d::Zero() : m_offset,  | ||||
|             dont_rotate ? Vec3d::Zero() : m_rotation, | ||||
|             dont_scale ? Vec3d::Ones() : m_scaling_factor, | ||||
|             dont_mirror ? Vec3d::Ones() : m_mirror | ||||
|             ); | ||||
|     const Geometry::TransformationSVD svd(*this); | ||||
|     m_matrix = get_offset_matrix() * Transform3d(svd.v * svd.s * svd.v.transpose()) * svd.mirror_matrix(); | ||||
| } | ||||
| 
 | ||||
|         m_flags.set(dont_translate, dont_rotate, dont_scale, dont_mirror); | ||||
|         m_dirty = false; | ||||
|     } | ||||
| void Transformation::reset_scaling_factor() | ||||
| { | ||||
|     const Geometry::TransformationSVD svd(*this); | ||||
|     m_matrix = get_offset_matrix() * Transform3d(svd.u) * Transform3d(svd.v.transpose()) * svd.mirror_matrix(); | ||||
| } | ||||
| 
 | ||||
|     return m_matrix; | ||||
| void Transformation::reset_skew() | ||||
| { | ||||
|     auto new_scale_factor = [](const Matrix3d& s) { | ||||
|         return pow(s(0, 0) * s(1, 1) * s(2, 2), 1. / 3.); // scale average
 | ||||
|     }; | ||||
| 
 | ||||
|     const Geometry::TransformationSVD svd(*this); | ||||
|     m_matrix = get_offset_matrix() * Transform3d(svd.u) * scale_transform(new_scale_factor(svd.s)) * Transform3d(svd.v.transpose()) * svd.mirror_matrix(); | ||||
| } | ||||
| 
 | ||||
| Transform3d Transformation::get_matrix_no_offset() const | ||||
| { | ||||
|     Transformation copy(*this); | ||||
|     copy.reset_offset(); | ||||
|     return copy.get_matrix(); | ||||
| } | ||||
| 
 | ||||
| Transform3d Transformation::get_matrix_no_scaling_factor() const | ||||
| { | ||||
|     Transformation copy(*this); | ||||
|     copy.reset_scaling_factor(); | ||||
|     return copy.get_matrix(); | ||||
| } | ||||
| 
 | ||||
| Transformation Transformation::operator * (const Transformation& other) const | ||||
|  | @ -663,7 +690,7 @@ Transformation Transformation::volume_to_bed_transformation(const Transformation | |||
|     if (instance_transformation.is_scaling_uniform()) { | ||||
|         // No need to run the non-linear least squares fitting for uniform scaling.
 | ||||
|         // Just set the inverse.
 | ||||
|         out.set_from_transform(instance_transformation.get_matrix(true).inverse()); | ||||
|         out.set_matrix(instance_transformation.get_matrix_no_offset().inverse()); | ||||
|     } | ||||
|     else if (is_rotation_ninety_degrees(instance_transformation.get_rotation())) { | ||||
|         // Anisotropic scaling, rotation by multiples of ninety degrees.
 | ||||
|  | @ -712,6 +739,53 @@ Transformation Transformation::volume_to_bed_transformation(const Transformation | |||
|     return out; | ||||
| } | ||||
| 
 | ||||
| TransformationSVD::TransformationSVD(const Transform3d& trafo) | ||||
| { | ||||
|     const auto &m0 = trafo.matrix().block<3, 3>(0, 0); | ||||
|     mirror = m0.determinant() < 0.0; | ||||
| 
 | ||||
|     Matrix3d m; | ||||
|     if (mirror) | ||||
|         m = m0 * Eigen::DiagonalMatrix<double, 3, 3>(-1.0, 1.0, 1.0); | ||||
|     else | ||||
|         m = m0; | ||||
|     const Eigen::JacobiSVD<Matrix3d> svd(m, Eigen::ComputeFullU | Eigen::ComputeFullV); | ||||
|     u = svd.matrixU(); | ||||
|     v = svd.matrixV(); | ||||
|     s = svd.singularValues().asDiagonal(); | ||||
| 
 | ||||
|     scale = !s.isApprox(Matrix3d::Identity()); | ||||
|     anisotropic_scale = ! is_approx(s(0, 0), s(1, 1)) || ! is_approx(s(1, 1), s(2, 2)); | ||||
|     rotation = !v.isApprox(u); | ||||
| 
 | ||||
|     if (anisotropic_scale) { | ||||
|         rotation_90_degrees = true; | ||||
|         for (int i = 0; i < 3; ++i) { | ||||
|             const Vec3d row = v.row(i).cwiseAbs(); | ||||
|             const size_t num_zeros = is_approx(row[0], 0.) + is_approx(row[1], 0.) + is_approx(row[2], 0.); | ||||
|             const size_t num_ones  = is_approx(row[0], 1.) + is_approx(row[1], 1.) + is_approx(row[2], 1.); | ||||
|             if (num_zeros != 2 || num_ones != 1) { | ||||
|                 rotation_90_degrees = false; | ||||
|                 break; | ||||
|             } | ||||
|         } | ||||
|         // Detect skew by brute force: check if the axes are still orthogonal after transformation
 | ||||
|         const Matrix3d trafo_linear = trafo.linear(); | ||||
|         const std::array<Vec3d, 3> axes = { Vec3d::UnitX(), Vec3d::UnitY(), Vec3d::UnitZ() }; | ||||
|         std::array<Vec3d, 3> transformed_axes; | ||||
|         for (int i = 0; i < 3; ++i) { | ||||
|             transformed_axes[i] = trafo_linear * axes[i]; | ||||
|         } | ||||
|         skew = std::abs(transformed_axes[0].dot(transformed_axes[1])) > EPSILON || | ||||
|                std::abs(transformed_axes[1].dot(transformed_axes[2])) > EPSILON || | ||||
|                std::abs(transformed_axes[2].dot(transformed_axes[0])) > EPSILON; | ||||
| 
 | ||||
|         // This following old code does not work under all conditions. The v matrix can become non diagonal (see SPE-1492) 
 | ||||
| //        skew = ! rotation_90_degrees;
 | ||||
|     } else | ||||
|         skew = false; | ||||
| } | ||||
| 
 | ||||
| // For parsing a transformation matrix from 3MF / AMF.
 | ||||
| Transform3d transform3d_from_string(const std::string& transform_str) | ||||
| { | ||||
|  |  | |||
|  | @ -51,9 +51,9 @@ enum Orientation | |||
| static inline Orientation orient(const Point &a, const Point &b, const Point &c) | ||||
| { | ||||
|     static_assert(sizeof(coord_t) * 2 == sizeof(int64_t), "orient works with 32 bit coordinates"); | ||||
|     int64_t u = int64_t(b(0)) * int64_t(c(1)) - int64_t(b(1)) * int64_t(c(0)); | ||||
|     int64_t v = int64_t(a(0)) * int64_t(c(1)) - int64_t(a(1)) * int64_t(c(0)); | ||||
|     int64_t w = int64_t(a(0)) * int64_t(b(1)) - int64_t(a(1)) * int64_t(b(0)); | ||||
|     int64_t u = int64_t(b.x()) * int64_t(c.y()) - int64_t(b.y()) * int64_t(c.x()); | ||||
|     int64_t v = int64_t(a.x()) * int64_t(c.y()) - int64_t(a.y()) * int64_t(c.x()); | ||||
|     int64_t w = int64_t(a.x()) * int64_t(b.y()) - int64_t(a.y()) * int64_t(b.x()); | ||||
|     int64_t d = u - v + w; | ||||
|     return (d > 0) ? ORIENTATION_CCW : ((d == 0) ? ORIENTATION_COLINEAR : ORIENTATION_CW); | ||||
| } | ||||
|  | @ -322,6 +322,13 @@ template<typename T> T angle_to_0_2PI(T angle) | |||
| 
 | ||||
|     return angle; | ||||
| } | ||||
| template<typename T> void to_range_pi_pi(T &angle){ | ||||
|     if (angle > T(PI) || angle <= -T(PI)) { | ||||
|         int count = static_cast<int>(std::round(angle / (2 * PI))); | ||||
|         angle -= static_cast<T>(count * 2 * PI); | ||||
|         assert(angle <= T(PI) && angle > -T(PI)); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| void simplify_polygons(const Polygons &polygons, double tolerance, Polygons* retval); | ||||
| 
 | ||||
|  | @ -404,66 +411,67 @@ void rotation_from_two_vectors(Vec3d from, Vec3d to, Vec3d &rotation_axis, doubl | |||
| 
 | ||||
| class Transformation | ||||
| { | ||||
|     struct Flags | ||||
|     { | ||||
|         bool dont_translate{ true }; | ||||
|         bool dont_rotate{ true }; | ||||
|         bool dont_scale{ true }; | ||||
|         bool dont_mirror{ true }; | ||||
| 
 | ||||
|         bool needs_update(bool dont_translate, bool dont_rotate, bool dont_scale, bool dont_mirror) const; | ||||
|         void set(bool dont_translate, bool dont_rotate, bool dont_scale, bool dont_mirror); | ||||
|     }; | ||||
| 
 | ||||
|     Vec3d m_offset{ Vec3d::Zero() };              // In unscaled coordinates
 | ||||
|     Vec3d m_rotation{ Vec3d::Zero() };            // Rotation around the three axes, in radians around mesh center point
 | ||||
|     Vec3d m_scaling_factor{ Vec3d::Ones() };      // Scaling factors along the three axes
 | ||||
|     Vec3d m_mirror{ Vec3d::Ones() };              // Mirroring along the three axes
 | ||||
| 
 | ||||
|     mutable Transform3d m_matrix{ Transform3d::Identity() }; | ||||
|     mutable Flags m_flags; | ||||
|     mutable bool m_dirty{ false }; | ||||
|     Transform3d m_matrix{ Transform3d::Identity() }; | ||||
| 
 | ||||
| public: | ||||
|     Transformation(); | ||||
|     explicit Transformation(const Transform3d& transform); | ||||
|     Transformation() = default; | ||||
|     explicit Transformation(const Transform3d& transform) : m_matrix(transform) {} | ||||
| 
 | ||||
|     //BBS: add get dirty function
 | ||||
|     bool is_dirty() { return m_dirty; } | ||||
|     Vec3d get_offset() const { return m_matrix.translation(); } | ||||
|     double get_offset(Axis axis) const { return get_offset()[axis]; } | ||||
| 
 | ||||
|     const Vec3d& get_offset() const { return m_offset; } | ||||
|     double get_offset(Axis axis) const { return m_offset(axis); } | ||||
|     Transform3d get_offset_matrix() const; | ||||
| 
 | ||||
|     void set_offset(const Vec3d& offset); | ||||
|     void set_offset(Axis axis, double offset); | ||||
|     void set_offset(const Vec3d& offset) { m_matrix.translation() = offset; } | ||||
|     void set_offset(Axis axis, double offset) { m_matrix.translation()[axis] = offset; } | ||||
| 
 | ||||
|     const Vec3d& get_rotation() const { return m_rotation; } | ||||
|     double get_rotation(Axis axis) const { return m_rotation(axis); } | ||||
|     Vec3d get_rotation() const; | ||||
|     double get_rotation(Axis axis) const { return get_rotation()[axis]; } | ||||
| 
 | ||||
|     Transform3d get_rotation_matrix() const; | ||||
| 
 | ||||
|     void set_rotation(const Vec3d& rotation); | ||||
|     void set_rotation(Axis axis, double rotation); | ||||
| 
 | ||||
|     const Vec3d& get_scaling_factor() const { return m_scaling_factor; } | ||||
|     double get_scaling_factor(Axis axis) const { return m_scaling_factor(axis); } | ||||
|     Vec3d get_scaling_factor() const; | ||||
|     double get_scaling_factor(Axis axis) const { return get_scaling_factor()[axis]; } | ||||
| 
 | ||||
|     Transform3d get_scaling_factor_matrix() const; | ||||
| 
 | ||||
|     bool is_scaling_uniform() const { | ||||
|         const Vec3d scale = get_scaling_factor(); | ||||
|         return std::abs(scale.x() - scale.y()) < 1e-8 && std::abs(scale.x() - scale.z()) < 1e-8; | ||||
|     } | ||||
| 
 | ||||
|     void set_scaling_factor(const Vec3d& scaling_factor); | ||||
|     void set_scaling_factor(Axis axis, double scaling_factor); | ||||
|     bool is_scaling_uniform() const { return std::abs(m_scaling_factor.x() - m_scaling_factor.y()) < 1e-8 && std::abs(m_scaling_factor.x() - m_scaling_factor.z()) < 1e-8; } | ||||
| 
 | ||||
|     const Vec3d& get_mirror() const { return m_mirror; } | ||||
|     double get_mirror(Axis axis) const { return m_mirror(axis); } | ||||
|     bool is_left_handed() const { return m_mirror.x() * m_mirror.y() * m_mirror.z() < 0.; } | ||||
|     Vec3d get_mirror() const; | ||||
|     double get_mirror(Axis axis) const { return get_mirror()[axis]; } | ||||
| 
 | ||||
|     Transform3d get_mirror_matrix() const; | ||||
| 
 | ||||
|     bool is_left_handed() const { | ||||
|         return m_matrix.linear().determinant() < 0; | ||||
|     } | ||||
| 
 | ||||
|     void set_mirror(const Vec3d& mirror); | ||||
|     void set_mirror(Axis axis, double mirror); | ||||
| 
 | ||||
|     void set_from_transform(const Transform3d& transform); | ||||
|     bool has_skew() const; | ||||
| 
 | ||||
|     void reset(); | ||||
|     void reset_offset() { set_offset(Vec3d::Zero()); } | ||||
|     void reset_rotation(); | ||||
|     void reset_scaling_factor(); | ||||
|     void reset_mirror() { set_mirror(Vec3d::Ones()); } | ||||
|     void reset_skew(); | ||||
| 
 | ||||
|     const Transform3d& get_matrix(bool dont_translate = false, bool dont_rotate = false, bool dont_scale = false, bool dont_mirror = false) const; | ||||
|     const Transform3d& get_matrix() const { return m_matrix; } | ||||
|     Transform3d get_matrix_no_offset() const; | ||||
|     Transform3d get_matrix_no_scaling_factor() const; | ||||
| 
 | ||||
|     void set_matrix(const Transform3d& transform) { m_matrix = transform; } | ||||
| 
 | ||||
|     Transformation operator * (const Transformation& other) const; | ||||
| 
 | ||||
|  | @ -474,21 +482,45 @@ public: | |||
| 
 | ||||
|     // BBS: backup use this compare
 | ||||
|     friend bool operator==(Transformation const& l, Transformation const& r) { | ||||
|         return l.m_offset == r.m_offset && l.m_rotation == r.m_rotation && l.m_scaling_factor == r.m_scaling_factor && l.m_mirror == r.m_mirror; | ||||
|         return l.m_matrix.isApprox(r.m_matrix); | ||||
|     } | ||||
| 
 | ||||
|     friend bool operator!=(Transformation const &l, Transformation const &r) | ||||
|     { | ||||
|         return !(l == r); | ||||
|     } | ||||
| 
 | ||||
| private: | ||||
| 	friend class cereal::access; | ||||
| 	template<class Archive> void serialize(Archive & ar) { ar(m_offset, m_rotation, m_scaling_factor, m_mirror); } | ||||
| 	explicit Transformation(int) : m_dirty(true) {} | ||||
| 	template <class Archive> static void load_and_construct(Archive &ar, cereal::construct<Transformation> &construct) | ||||
|     template<class Archive> void serialize(Archive& ar) { ar(m_matrix); } | ||||
|     explicit Transformation(int) {} | ||||
|     template <class Archive> static void load_and_construct(Archive& ar, cereal::construct<Transformation>& construct) | ||||
|     { | ||||
|         // Calling a private constructor with special "int" parameter to indicate that no construction is necessary.
 | ||||
|         construct(1); | ||||
| 		ar(construct.ptr()->m_offset, construct.ptr()->m_rotation, construct.ptr()->m_scaling_factor, construct.ptr()->m_mirror); | ||||
|         ar(construct.ptr()->m_matrix); | ||||
|     } | ||||
| }; | ||||
| 
 | ||||
| struct TransformationSVD | ||||
| { | ||||
|     Matrix3d u{ Matrix3d::Identity() }; | ||||
|     Matrix3d s{ Matrix3d::Identity() }; | ||||
|     Matrix3d v{ Matrix3d::Identity() }; | ||||
| 
 | ||||
|     bool mirror{ false }; | ||||
|     bool scale{ false }; | ||||
|     bool anisotropic_scale{ false }; | ||||
|     bool rotation{ false }; | ||||
|     bool rotation_90_degrees{ false }; | ||||
|     bool skew{ false }; | ||||
| 
 | ||||
|     explicit TransformationSVD(const Transformation& trafo) : TransformationSVD(trafo.get_matrix()) {} | ||||
|     explicit TransformationSVD(const Transform3d& trafo); | ||||
| 
 | ||||
|     Eigen::DiagonalMatrix<double, 3, 3> mirror_matrix() const { return Eigen::DiagonalMatrix<double, 3, 3>(this->mirror ? -1. : 1., 1., 1.); } | ||||
| }; | ||||
| 
 | ||||
| // For parsing a transformation matrix from 3MF / AMF.
 | ||||
| extern Transform3d transform3d_from_string(const std::string& transform_str); | ||||
| 
 | ||||
|  |  | |||
							
								
								
									
										49
									
								
								src/libslic3r/IntersectionPoints.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,49 @@ | |||
| ///|/ Copyright (c) Prusa Research 2023 Vojtěch Bubník @bubnikv
 | ||||
| ///|/
 | ||||
| ///|/ PrusaSlicer is released under the terms of the AGPLv3 or higher
 | ||||
| ///|/
 | ||||
| #include "IntersectionPoints.hpp" | ||||
| #include <libslic3r/AABBTreeLines.hpp> | ||||
| 
 | ||||
| //NOTE: using CGAL SweepLines is slower !!! (example in git history)
 | ||||
| 
 | ||||
| namespace {     | ||||
| using namespace Slic3r; | ||||
| IntersectionsLines compute_intersections(const Lines &lines) | ||||
| { | ||||
|     if (lines.size() < 3) | ||||
|         return {};     | ||||
| 
 | ||||
|     auto tree = AABBTreeLines::build_aabb_tree_over_indexed_lines(lines); | ||||
|     IntersectionsLines result; | ||||
|     for (uint32_t li = 0; li < lines.size()-1; ++li) { | ||||
|         const Line &l = lines[li]; | ||||
|         auto intersections = AABBTreeLines::get_intersections_with_line<false, Point, Line>(lines, tree, l); | ||||
|         for (const auto &[p, node_index] : intersections) { | ||||
|             if (node_index - 1 <= li) | ||||
|                 continue;             | ||||
|             if (const Line &l_ = lines[node_index]; | ||||
|                 l_.a == l.a || | ||||
|                 l_.a == l.b || | ||||
|                 l_.b == l.a || | ||||
|                 l_.b == l.b ) | ||||
|                 // it is duplicit point not intersection
 | ||||
|                 continue;  | ||||
| 
 | ||||
|             // NOTE: fix AABBTree to compute intersection with double preccission!!
 | ||||
|             Vec2d intersection_point = p.cast<double>(); | ||||
| 
 | ||||
|             result.push_back(IntersectionLines{li, static_cast<uint32_t>(node_index), intersection_point}); | ||||
|         } | ||||
|     } | ||||
|     return result; | ||||
| } | ||||
| } // namespace
 | ||||
| 
 | ||||
| namespace Slic3r { | ||||
| IntersectionsLines get_intersections(const Lines &lines)           { return compute_intersections(lines); } | ||||
| IntersectionsLines get_intersections(const Polygon &polygon)       { return compute_intersections(to_lines(polygon)); } | ||||
| IntersectionsLines get_intersections(const Polygons &polygons)     { return compute_intersections(to_lines(polygons)); } | ||||
| IntersectionsLines get_intersections(const ExPolygon &expolygon)   { return compute_intersections(to_lines(expolygon)); } | ||||
| IntersectionsLines get_intersections(const ExPolygons &expolygons) { return compute_intersections(to_lines(expolygons)); } | ||||
| } | ||||
							
								
								
									
										27
									
								
								src/libslic3r/IntersectionPoints.hpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,27 @@ | |||
| ///|/ Copyright (c) Prusa Research 2023 Vojtěch Bubník @bubnikv
 | ||||
| ///|/
 | ||||
| ///|/ PrusaSlicer is released under the terms of the AGPLv3 or higher
 | ||||
| ///|/
 | ||||
| #ifndef slic3r_IntersectionPoints_hpp_ | ||||
| #define slic3r_IntersectionPoints_hpp_ | ||||
| 
 | ||||
| #include "ExPolygon.hpp" | ||||
| 
 | ||||
| namespace Slic3r { | ||||
| 
 | ||||
| struct IntersectionLines { | ||||
|     uint32_t line_index1; | ||||
|     uint32_t line_index2; | ||||
|     Vec2d intersection; | ||||
| }; | ||||
| using IntersectionsLines = std::vector<IntersectionLines>; | ||||
| 
 | ||||
| // collect all intersecting points
 | ||||
| IntersectionsLines get_intersections(const Lines &lines); | ||||
| IntersectionsLines get_intersections(const Polygon &polygon); | ||||
| IntersectionsLines get_intersections(const Polygons &polygons); | ||||
| IntersectionsLines get_intersections(const ExPolygon &expolygon); | ||||
| IntersectionsLines get_intersections(const ExPolygons &expolygons); | ||||
| 
 | ||||
| } // namespace Slic3r
 | ||||
| #endif // slic3r_IntersectionPoints_hpp_
 | ||||
|  | @ -1368,7 +1368,7 @@ indexed_triangle_set ModelObject::raw_indexed_triangle_set() const | |||
|             size_t j = out.indices.size(); | ||||
|             append(out.vertices, v->mesh().its.vertices); | ||||
|             append(out.indices,  v->mesh().its.indices); | ||||
|             auto m = v->get_matrix(); | ||||
|             const Transform3d& m = v->get_matrix(); | ||||
|             for (; i < out.vertices.size(); ++ i) | ||||
|                 out.vertices[i] = (m * out.vertices[i].cast<double>()).cast<float>().eval(); | ||||
|             if (v->is_left_handed()) { | ||||
|  | @ -1410,7 +1410,7 @@ const BoundingBoxf3& ModelObject::raw_bounding_box() const | |||
|         if (this->instances.empty()) | ||||
|             throw Slic3r::InvalidArgument("Can't call raw_bounding_box() with no instances"); | ||||
| 
 | ||||
|         const Transform3d& inst_matrix = this->instances.front()->get_transformation().get_matrix(true); | ||||
|         const Transform3d inst_matrix = this->instances.front()->get_transformation().get_matrix_no_offset(); | ||||
|         for (const ModelVolume *v : this->volumes) | ||||
|             if (v->is_model_part()) | ||||
|                 m_raw_bounding_box.merge(v->mesh().transformed_bounding_box(inst_matrix * v->get_matrix())); | ||||
|  | @ -1423,14 +1423,18 @@ BoundingBoxf3 ModelObject::instance_bounding_box(size_t instance_idx, bool dont_ | |||
|     return instance_bounding_box(*this->instances[instance_idx], dont_translate); | ||||
| } | ||||
| 
 | ||||
| BoundingBoxf3 ModelObject::instance_bounding_box(const ModelInstance &instance, bool dont_translate) const { | ||||
|     BoundingBoxf3 bbox; | ||||
|     const auto& inst_mat = instance.get_transformation().get_matrix(dont_translate); | ||||
|     for (auto vol : this->volumes) { | ||||
|         if (vol->is_model_part()) | ||||
|             bbox.merge(vol->mesh().transformed_bounding_box(inst_mat * vol->get_matrix())); | ||||
| BoundingBoxf3 ModelObject::instance_bounding_box(const ModelInstance &instance, bool dont_translate) const | ||||
| { | ||||
|     BoundingBoxf3 bb; | ||||
|     const Transform3d inst_matrix = dont_translate ? | ||||
|         instance.get_transformation().get_matrix_no_offset() : | ||||
|         instance.get_transformation().get_matrix(); | ||||
| 
 | ||||
|     for (ModelVolume *v : this->volumes) { | ||||
|         if (v->is_model_part()) | ||||
|             bb.merge(v->mesh().transformed_bounding_box(inst_matrix * v->get_matrix())); | ||||
|     } | ||||
|     return bbox; | ||||
|     return bb; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
|  | @ -1438,7 +1442,9 @@ BoundingBoxf3 ModelObject::instance_bounding_box(const ModelInstance &instance, | |||
| BoundingBoxf3 ModelObject::instance_convex_hull_bounding_box(size_t instance_idx, bool dont_translate) const | ||||
| { | ||||
|     BoundingBoxf3 bb; | ||||
|     const Transform3d& inst_matrix = this->instances[instance_idx]->get_transformation().get_matrix(dont_translate); | ||||
|     const Transform3d& inst_matrix = dont_translate ? | ||||
|         this->instances[instance_idx]->get_transformation().get_matrix_no_offset() : | ||||
|         this->instances[instance_idx]->get_transformation().get_matrix(); | ||||
|     for (ModelVolume *v : this->volumes) | ||||
|     { | ||||
|         if (v->is_model_part()) | ||||
|  | @ -1739,6 +1745,25 @@ void ModelObject::clone_for_cut(ModelObject **obj) | |||
|     (*obj)->input_file.clear(); | ||||
| } | ||||
| 
 | ||||
| bool ModelVolume::is_the_only_one_part() const  | ||||
| { | ||||
|     if (m_type != ModelVolumeType::MODEL_PART) | ||||
|         return false; | ||||
|     if (object == nullptr) | ||||
|         return false; | ||||
|     for (const ModelVolume *v : object->volumes) { | ||||
|         if (v == nullptr) | ||||
|             continue; | ||||
|         // is this volume?
 | ||||
|         if (v->id() == this->id()) | ||||
|             continue; | ||||
|         // exist another model part in object?
 | ||||
|         if (v->type() == ModelVolumeType::MODEL_PART) | ||||
|             return false; | ||||
|     } | ||||
|     return true; | ||||
| } | ||||
| 
 | ||||
| void ModelVolume::reset_extra_facets() | ||||
| { | ||||
|     this->supported_facets.reset(); | ||||
|  | @ -1759,67 +1784,6 @@ static void invalidate_translations(ModelObject* object, const ModelInstance* sr | |||
|     } | ||||
| } | ||||
| 
 | ||||
| static void reset_instance_transformation(ModelObject* object, size_t src_instance_idx, const Transform3d& cut_matrix, | ||||
|                                           bool place_on_cut = false, bool flip = false, Vec3d local_displace = Vec3d::Zero()) | ||||
| { | ||||
|     using namespace Geometry; | ||||
| 
 | ||||
|     // Reset instance transformation except offset and Z-rotation
 | ||||
| 
 | ||||
|     for (size_t i = 0; i < object->instances.size(); ++i) { | ||||
|         auto& obj_instance = object->instances[i]; | ||||
| 
 | ||||
|         Geometry::Transformation instance_transformation_copy = obj_instance->get_transformation(); | ||||
|         instance_transformation_copy.set_offset(Vec3d(0, 0, 0)); | ||||
|         if (object->volumes.size() == 1) { | ||||
|             instance_transformation_copy.set_offset(-object->volumes[0]->get_offset()); | ||||
|         } | ||||
| 
 | ||||
|         if (i == src_instance_idx && object->volumes.size() == 1) | ||||
|             invalidate_translations(object, obj_instance); | ||||
| 
 | ||||
|         const Vec3d offset = obj_instance->get_offset(); | ||||
|         const double rot_z = obj_instance->get_rotation().z(); | ||||
| 
 | ||||
|         obj_instance->set_transformation(Transformation()); | ||||
| 
 | ||||
|         const Vec3d displace = local_displace.isApprox(Vec3d::Zero()) ? Vec3d::Zero() : | ||||
|                                rotation_transform(obj_instance->get_rotation()) * local_displace; | ||||
|         obj_instance->set_offset(offset + displace); | ||||
| 
 | ||||
|         Vec3d rotation = Vec3d::Zero(); | ||||
|         if (!flip && !place_on_cut) { | ||||
|             if ( i != src_instance_idx) | ||||
|             rotation[Z] = rot_z; | ||||
|         } | ||||
|         else { | ||||
|             Transform3d rotation_matrix = Transform3d::Identity(); | ||||
|             if (flip) | ||||
|                 rotation_matrix = rotation_transform(PI * Vec3d::UnitX()); | ||||
| 
 | ||||
|             if (place_on_cut) | ||||
|                 rotation_matrix = rotation_matrix * Transformation(cut_matrix).get_matrix(true, false, true, true).inverse(); | ||||
| 
 | ||||
|             if (i != src_instance_idx) | ||||
|                 rotation_matrix = rotation_transform(rot_z * Vec3d::UnitZ()) * rotation_matrix; | ||||
| 
 | ||||
|             rotation = Transformation(rotation_matrix).get_rotation(); | ||||
|         } | ||||
| 
 | ||||
|         obj_instance->set_rotation(rotation); | ||||
| 
 | ||||
|         // update the assemble matrix
 | ||||
|         const Transform3d &assemble_matrix = obj_instance->get_assemble_transformation().get_matrix(); | ||||
|         const Transform3d &instance_inverse_matrix = instance_transformation_copy.get_matrix().inverse(); | ||||
|         Transform3d new_instance_inverse_matrix = instance_inverse_matrix * obj_instance->get_transformation().get_matrix(true).inverse(); | ||||
|         if (place_on_cut) { // reset the rotation of cut plane
 | ||||
|             new_instance_inverse_matrix = new_instance_inverse_matrix * Transformation(cut_matrix).get_matrix(true, false, true, true).inverse(); | ||||
|         } | ||||
|         Transform3d new_assemble_transform = assemble_matrix * new_instance_inverse_matrix; | ||||
|         obj_instance->set_assemble_from_transform(new_assemble_transform); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| void ModelObject::split(ModelObjectPtrs* new_objects) | ||||
| { | ||||
|     std::vector<TriangleMesh> all_meshes; | ||||
|  | @ -1833,6 +1797,10 @@ void ModelObject::split(ModelObjectPtrs* new_objects) | |||
|         if (volume->type() != ModelVolumeType::MODEL_PART) | ||||
|             continue; | ||||
| 
 | ||||
|         // splited volume should not be text object 
 | ||||
|         if (volume->text_configuration.has_value()) | ||||
|             volume->text_configuration.reset(); | ||||
| 
 | ||||
|         if (!is_multi_volume_object) { | ||||
|             //BBS: not multi volume object, then split mesh.
 | ||||
|             std::vector<TriangleMesh> volume_meshes = volume->mesh().split(); | ||||
|  | @ -1912,7 +1880,7 @@ void ModelObject::split(ModelObjectPtrs* new_objects) | |||
| 
 | ||||
|             for (ModelInstance* model_instance : new_object->instances) | ||||
|             { | ||||
|                 Vec3d shift = model_instance->get_transformation().get_matrix(true) * new_vol->get_offset(); | ||||
|                 const Vec3d shift = model_instance->get_transformation().get_matrix_no_offset() * new_vol->get_offset(); | ||||
|                 model_instance->set_offset(model_instance->get_offset() + shift); | ||||
| 
 | ||||
|                 //BBS: add assemble_view related logic
 | ||||
|  | @ -1920,7 +1888,7 @@ void ModelObject::split(ModelObjectPtrs* new_objects) | |||
|                 instance_transformation_copy.set_offset(-new_vol->get_offset()); | ||||
|                 const Transform3d &assemble_matrix = model_instance->get_assemble_transformation().get_matrix(); | ||||
|                 const Transform3d &instance_inverse_matrix = instance_transformation_copy.get_matrix().inverse(); | ||||
|                 Transform3d new_instance_inverse_matrix = instance_inverse_matrix * model_instance->get_transformation().get_matrix(true).inverse(); | ||||
|                 Transform3d new_instance_inverse_matrix = instance_inverse_matrix * model_instance->get_transformation().get_matrix_no_offset().inverse(); | ||||
|                 Transform3d new_assemble_transform      = assemble_matrix * new_instance_inverse_matrix; | ||||
|                 model_instance->set_assemble_from_transform(new_assemble_transform); | ||||
|                 model_instance->set_offset_to_assembly(new_vol->get_offset()); | ||||
|  | @ -2037,7 +2005,13 @@ void ModelObject::bake_xy_rotation_into_meshes(size_t instance_idx) | |||
| 
 | ||||
|     // Adjust the meshes.
 | ||||
|     // Transformation to be applied to the meshes.
 | ||||
|     Eigen::Matrix3d mesh_trafo_3x3           = reference_trafo.get_matrix(true, false, uniform_scaling, ! has_mirrorring).matrix().block<3, 3>(0, 0); | ||||
|     Geometry::Transformation reference_trafo_mod = reference_trafo; | ||||
|     reference_trafo_mod.reset_offset(); | ||||
|     if (uniform_scaling) | ||||
|         reference_trafo_mod.reset_scaling_factor(); | ||||
|     if (!has_mirrorring) | ||||
|         reference_trafo_mod.reset_mirror(); | ||||
|     Eigen::Matrix3d mesh_trafo_3x3 = reference_trafo_mod.get_matrix().matrix().block<3, 3>(0, 0); | ||||
|     Transform3d     volume_offset_correction = this->instances[instance_idx]->get_transformation().get_matrix().inverse() * reference_trafo.get_matrix(); | ||||
|     for (ModelVolume *model_volume : this->volumes) { | ||||
|         const Geometry::Transformation volume_trafo = model_volume->get_transformation(); | ||||
|  | @ -2047,7 +2021,13 @@ void ModelObject::bake_xy_rotation_into_meshes(size_t instance_idx) | |||
|                                            std::abs(volume_trafo.get_scaling_factor().x() - volume_trafo.get_scaling_factor().z()) < EPSILON; | ||||
|         double volume_new_scaling_factor = volume_uniform_scaling ? volume_trafo.get_scaling_factor().x() : 1.; | ||||
|         // Transform the mesh.
 | ||||
| 		Matrix3d volume_trafo_3x3 = volume_trafo.get_matrix(true, false, volume_uniform_scaling, !volume_has_mirrorring).matrix().block<3, 3>(0, 0); | ||||
|         Geometry::Transformation volume_trafo_mod = volume_trafo; | ||||
|         volume_trafo_mod.reset_offset(); | ||||
|         if (volume_uniform_scaling) | ||||
|             volume_trafo_mod.reset_scaling_factor(); | ||||
|         if (!volume_has_mirrorring) | ||||
|             volume_trafo_mod.reset_mirror(); | ||||
|         Eigen::Matrix3d volume_trafo_3x3 = volume_trafo_mod.get_matrix().matrix().block<3, 3>(0, 0); | ||||
|         // Following method creates a new shared_ptr<TriangleMesh>
 | ||||
|         model_volume->transform_this_mesh(mesh_trafo_3x3 * volume_trafo_3x3, left_handed != volume_left_handed); | ||||
|         // Reset the rotation, scaling and mirroring.
 | ||||
|  | @ -2094,7 +2074,7 @@ double ModelObject::get_instance_min_z(size_t instance_idx) const | |||
|     double min_z = DBL_MAX; | ||||
| 
 | ||||
|     const ModelInstance* inst = instances[instance_idx]; | ||||
|     const Transform3d& mi = inst->get_matrix(true); | ||||
|     const Transform3d mi = inst->get_matrix_no_offset(); | ||||
| 
 | ||||
|     for (const ModelVolume* v : volumes) { | ||||
|         if (!v->is_model_part()) | ||||
|  | @ -2130,7 +2110,7 @@ double ModelObject::get_instance_max_z(size_t instance_idx) const | |||
|     double max_z = -DBL_MAX; | ||||
| 
 | ||||
|     const ModelInstance* inst = instances[instance_idx]; | ||||
|     const Transform3d& mi = inst->get_matrix(true); | ||||
|     const Transform3d mi = inst->get_matrix_no_offset(); | ||||
| 
 | ||||
|     for (const ModelVolume* v : volumes) { | ||||
|         if (!v->is_model_part()) | ||||
|  | @ -2553,6 +2533,10 @@ size_t ModelVolume::split(unsigned int max_extruders) | |||
|     if (meshes.size() <= 1) | ||||
|         return 1; | ||||
| 
 | ||||
|     // splited volume should not be text object
 | ||||
|     if (text_configuration.has_value()) | ||||
|         text_configuration.reset(); | ||||
| 
 | ||||
|     size_t idx = 0; | ||||
|     size_t ivolume = std::find(this->object->volumes.begin(), this->object->volumes.end(), this) - this->object->volumes.begin(); | ||||
|     const std::string name = this->name; | ||||
|  | @ -2723,44 +2707,17 @@ void ModelVolume::convert_from_meters() | |||
| 
 | ||||
| void ModelInstance::transform_mesh(TriangleMesh* mesh, bool dont_translate) const | ||||
| { | ||||
|     mesh->transform(get_matrix(dont_translate)); | ||||
| } | ||||
| 
 | ||||
| BoundingBoxf3 ModelInstance::transform_mesh_bounding_box(const TriangleMesh& mesh, bool dont_translate) const | ||||
| { | ||||
|     // Rotate around mesh origin.
 | ||||
|     TriangleMesh copy(mesh); | ||||
|     copy.transform(get_matrix(true, false, true, true)); | ||||
|     BoundingBoxf3 bbox = copy.bounding_box(); | ||||
| 
 | ||||
|     if (!empty(bbox)) { | ||||
|         // Scale the bounding box along the three axes.
 | ||||
|         for (unsigned int i = 0; i < 3; ++i) | ||||
|         { | ||||
|             if (std::abs(get_scaling_factor((Axis)i)-1.0) > EPSILON) | ||||
|             { | ||||
|                 bbox.min(i) *= get_scaling_factor((Axis)i); | ||||
|                 bbox.max(i) *= get_scaling_factor((Axis)i); | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         // Translate the bounding box.
 | ||||
|         if (! dont_translate) { | ||||
|             bbox.min += get_offset(); | ||||
|             bbox.max += get_offset(); | ||||
|         } | ||||
|     } | ||||
|     return bbox; | ||||
|     mesh->transform(dont_translate ? get_matrix_no_offset() : get_matrix()); | ||||
| } | ||||
| 
 | ||||
| BoundingBoxf3 ModelInstance::transform_bounding_box(const BoundingBoxf3 &bbox, bool dont_translate) const | ||||
| { | ||||
|     return bbox.transformed(get_matrix(dont_translate)); | ||||
|     return bbox.transformed(dont_translate ? get_matrix_no_offset() : get_matrix()); | ||||
| } | ||||
| 
 | ||||
| Vec3d ModelInstance::transform_vector(const Vec3d& v, bool dont_translate) const | ||||
| { | ||||
|     return get_matrix(dont_translate) * v; | ||||
|     return dont_translate ? get_matrix_no_offset() * v : get_matrix() * v; | ||||
| } | ||||
| 
 | ||||
| void ModelInstance::transform_polygon(Polygon* polygon) const | ||||
|  | @ -2952,7 +2909,7 @@ Polygon ModelInstance::convex_hull_2d() | |||
|     //BOOST_LOG_TRIVIAL(info) << __FUNCTION__ << boost::format(": name %1%, is_valid %2%")% this->object->name.c_str()% convex_hull.is_valid();
 | ||||
|     //if (!convex_hull.is_valid())
 | ||||
|     { // this logic is not working right now, as moving instance doesn't update convex_hull
 | ||||
|         const Transform3d& trafo_instance = get_matrix(false); | ||||
|         const Transform3d& trafo_instance = get_matrix(); | ||||
|         convex_hull = get_object()->convex_hull_2d(trafo_instance); | ||||
|     } | ||||
|     //int size = convex_hull.size();
 | ||||
|  |  | |||
|  | @ -26,6 +26,8 @@ | |||
| #include "CustomGCode.hpp" | ||||
| #include "calib.hpp" | ||||
| #include "enum_bitmask.hpp" | ||||
| #include "TextConfiguration.hpp" | ||||
| #include "EmbossShape.hpp" | ||||
| 
 | ||||
| //BBS: add bbs 3mf
 | ||||
| #include "Format/bbs_3mf.hpp" | ||||
|  | @ -41,6 +43,7 @@ | |||
| #include <vector> | ||||
| #include <algorithm> | ||||
| #include <functional> | ||||
| #include <optional> | ||||
| 
 | ||||
| namespace cereal { | ||||
| 	class BinaryInputArchive; | ||||
|  | @ -779,38 +782,6 @@ private: | |||
|     friend class ModelVolume; | ||||
| }; | ||||
| 
 | ||||
| struct RaycastResult | ||||
| { | ||||
|     Vec2d mouse_position = Vec2d::Zero(); | ||||
|     int   mesh_id        = -1; | ||||
|     Vec3f hit            = Vec3f::Zero(); | ||||
|     Vec3f normal         = Vec3f::Zero(); | ||||
| 
 | ||||
|     template<typename Archive> void serialize(Archive &ar) { ar(mouse_position, mesh_id, hit, normal); } | ||||
| }; | ||||
| 
 | ||||
| struct TextInfo | ||||
| { | ||||
|     std::string m_font_name; | ||||
|     float       m_font_size     = 16.f; | ||||
|     int         m_curr_font_idx = 0; | ||||
|     bool        m_bold          = true; | ||||
|     bool        m_italic        = false; | ||||
|     float       m_thickness     = 2.f; | ||||
|     float       m_embeded_depth = 0.f; | ||||
|     float       m_rotate_angle    = 0; | ||||
|     float       m_text_gap        = 0.f; | ||||
|     bool        m_is_surface_text = false; | ||||
|     bool        m_keep_horizontal = false; | ||||
|     std::string m_text; | ||||
| 
 | ||||
|     RaycastResult m_rr; | ||||
| 
 | ||||
|     template<typename Archive> void serialize(Archive &ar) { | ||||
|         ar(m_font_name, m_font_size, m_curr_font_idx, m_bold, m_italic, m_thickness, m_embeded_depth, m_rotate_angle, m_text_gap, m_is_surface_text, m_keep_horizontal, m_text, m_rr); | ||||
|     } | ||||
| }; | ||||
| 
 | ||||
| // An object STL, or a modifier volume, over which a different set of parameters shall be applied.
 | ||||
| // ModelVolume instances are owned by a ModelObject.
 | ||||
| class ModelVolume final : public ObjectBase | ||||
|  | @ -883,6 +854,7 @@ public: | |||
|     void                set_mesh(std::shared_ptr<const TriangleMesh> &mesh) { m_mesh = mesh; } | ||||
|     void                set_mesh(std::unique_ptr<const TriangleMesh> &&mesh) { m_mesh = std::move(mesh); } | ||||
| 	void				reset_mesh() { m_mesh = std::make_shared<const TriangleMesh>(); } | ||||
|     const std::shared_ptr<const TriangleMesh>& get_mesh_shared_ptr() const { return m_mesh; } | ||||
|     // Configuration parameters specific to an object model geometry or a modifier volume, 
 | ||||
|     // overriding the global Slic3r settings and the ModelObject settings.
 | ||||
|     ModelConfigObject	config; | ||||
|  | @ -903,6 +875,14 @@ public: | |||
|     // List of exterior faces
 | ||||
|     FacetsAnnotation    exterior_facets; | ||||
| 
 | ||||
|     // Is set only when volume is Embossed Text type
 | ||||
|     // Contain information how to re-create volume
 | ||||
|     std::optional<TextConfiguration> text_configuration; | ||||
| 
 | ||||
|     // Is set only when volume is Embossed Shape
 | ||||
|     // Contain 2d information about embossed shape to be editabled
 | ||||
|     std::optional<EmbossShape> emboss_shape;  | ||||
| 
 | ||||
|     // A parent object owning this modifier volume.
 | ||||
|     ModelObject*        get_object() const { return this->object; } | ||||
|     ModelVolumeType     type() const { return m_type; } | ||||
|  | @ -913,6 +893,9 @@ public: | |||
| 	bool                is_support_enforcer()   const { return m_type == ModelVolumeType::SUPPORT_ENFORCER; } | ||||
| 	bool                is_support_blocker()    const { return m_type == ModelVolumeType::SUPPORT_BLOCKER; } | ||||
| 	bool                is_support_modifier()   const { return m_type == ModelVolumeType::SUPPORT_BLOCKER || m_type == ModelVolumeType::SUPPORT_ENFORCER; } | ||||
|     bool                is_text()               const { return text_configuration.has_value(); } | ||||
|     bool                is_svg() const { return emboss_shape.has_value()  && !text_configuration.has_value(); } | ||||
|     bool                is_the_only_one_part() const; // behave like an object
 | ||||
|     t_model_material_id material_id() const { return m_material_id; } | ||||
|     void                set_material_id(t_model_material_id material_id); | ||||
|     void                reset_extra_facets(); | ||||
|  | @ -968,15 +951,16 @@ public: | |||
| 
 | ||||
|     const Geometry::Transformation& get_transformation() const { return m_transformation; } | ||||
|     void set_transformation(const Geometry::Transformation& transformation) { m_transformation = transformation; } | ||||
|     void set_transformation(const Transform3d &trafo) { m_transformation.set_from_transform(trafo); } | ||||
|     void set_transformation(const Transform3d& trafo) { m_transformation.set_matrix(trafo); } | ||||
| 
 | ||||
|     Vec3d get_offset() const { return m_transformation.get_offset(); } | ||||
| 
 | ||||
|     const Vec3d& get_offset() const { return m_transformation.get_offset(); } | ||||
|     double get_offset(Axis axis) const { return m_transformation.get_offset(axis); } | ||||
| 
 | ||||
|     void set_offset(const Vec3d& offset) { m_transformation.set_offset(offset); } | ||||
|     void set_offset(Axis axis, double offset) { m_transformation.set_offset(axis, offset); } | ||||
| 
 | ||||
|     const Vec3d& get_rotation() const { return m_transformation.get_rotation(); } | ||||
|     Vec3d get_rotation() const { return m_transformation.get_rotation(); } | ||||
|     double get_rotation(Axis axis) const { return m_transformation.get_rotation(axis); } | ||||
| 
 | ||||
|     void set_rotation(const Vec3d& rotation) { m_transformation.set_rotation(rotation); } | ||||
|  | @ -988,7 +972,7 @@ public: | |||
|     void set_scaling_factor(const Vec3d& scaling_factor) { m_transformation.set_scaling_factor(scaling_factor); } | ||||
|     void set_scaling_factor(Axis axis, double scaling_factor) { m_transformation.set_scaling_factor(axis, scaling_factor); } | ||||
| 
 | ||||
|     const Vec3d& get_mirror() const { return m_transformation.get_mirror(); } | ||||
|     Vec3d get_mirror() const { return m_transformation.get_mirror(); } | ||||
|     double get_mirror(Axis axis) const { return m_transformation.get_mirror(axis); } | ||||
|     bool is_left_handed() const { return m_transformation.is_left_handed(); } | ||||
| 
 | ||||
|  | @ -997,10 +981,8 @@ public: | |||
|     void convert_from_imperial_units(); | ||||
|     void convert_from_meters(); | ||||
| 
 | ||||
|     void set_text_info(const TextInfo& text_info) { m_text_info = text_info; } | ||||
|     const TextInfo& get_text_info() const { return m_text_info; } | ||||
| 
 | ||||
|     const Transform3d& get_matrix(bool dont_translate = false, bool dont_rotate = false, bool dont_scale = false, bool dont_mirror = false) const { return m_transformation.get_matrix(dont_translate, dont_rotate, dont_scale, dont_mirror); } | ||||
|     const Transform3d& get_matrix() const { return m_transformation.get_matrix(); } | ||||
|     Transform3d get_matrix_no_offset() const { return m_transformation.get_matrix_no_offset(); } | ||||
| 
 | ||||
| 	void set_new_unique_id() { | ||||
|         ObjectBase::set_new_unique_id(); | ||||
|  | @ -1044,8 +1026,6 @@ private: | |||
|     mutable Polygon                     m_cached_2d_polygon;   //BBS, used for convex_hell_2d acceleration
 | ||||
|     Geometry::Transformation        	m_transformation; | ||||
| 
 | ||||
|     TextInfo m_text_info; | ||||
| 
 | ||||
|     //BBS: add convex_hell_2d related logic
 | ||||
|     void  calculate_convex_hull_2d(const Geometry::Transformation &transformation) const; | ||||
| 
 | ||||
|  | @ -1100,7 +1080,7 @@ private: | |||
|         name(other.name), source(other.source), m_mesh(other.m_mesh), m_convex_hull(other.m_convex_hull), | ||||
|         config(other.config), m_type(other.m_type), object(object), m_transformation(other.m_transformation), | ||||
|         supported_facets(other.supported_facets), seam_facets(other.seam_facets), mmu_segmentation_facets(other.mmu_segmentation_facets), | ||||
|         m_text_info(other.m_text_info) | ||||
|         cut_info(other.cut_info), text_configuration(other.text_configuration), emboss_shape(other.emboss_shape) | ||||
|     { | ||||
| 		assert(this->id().valid());  | ||||
|         assert(this->config.id().valid());  | ||||
|  | @ -1119,8 +1099,9 @@ private: | |||
|         this->set_material_id(other.material_id()); | ||||
|     } | ||||
|     // Providing a new mesh, therefore this volume will get a new unique ID assigned.
 | ||||
|     ModelVolume(ModelObject *object, const ModelVolume &other, const TriangleMesh &&mesh) : | ||||
|         name(other.name), source(other.source), m_mesh(new TriangleMesh(std::move(mesh))), config(other.config), m_type(other.m_type), object(object), m_transformation(other.m_transformation) | ||||
|     ModelVolume(ModelObject *object, const ModelVolume &other, TriangleMesh &&mesh) : | ||||
|         name(other.name), source(other.source), config(other.config), object(object), m_mesh(new TriangleMesh(std::move(mesh))), m_type(other.m_type), m_transformation(other.m_transformation), | ||||
|         cut_info(other.cut_info), text_configuration(other.text_configuration), emboss_shape(other.emboss_shape) | ||||
|     { | ||||
| 		assert(this->id().valid());  | ||||
|         assert(this->config.id().valid());  | ||||
|  | @ -1135,7 +1116,7 @@ private: | |||
|         assert(this->config.id() == other.config.id()); | ||||
|         this->set_material_id(other.material_id()); | ||||
|         this->config.set_new_unique_id(); | ||||
|         if (mesh.facets_count() > 1) | ||||
|         if (m_mesh->facets_count() > 1) | ||||
|             calculate_convex_hull(); | ||||
| 		assert(this->config.id().valid());  | ||||
|         assert(this->config.id() != other.config.id());  | ||||
|  | @ -1165,9 +1146,8 @@ private: | |||
|         // BBS: add backup, check modify
 | ||||
|         bool mesh_changed = false; | ||||
|         auto tr = m_transformation; | ||||
|         ar(name, source, m_mesh, m_type, m_material_id, m_transformation, m_is_splittable, has_convex_hull, m_text_info, cut_info); | ||||
|         ar(name, source, m_mesh, m_type, m_material_id, m_transformation, m_is_splittable, has_convex_hull, cut_info); | ||||
|         mesh_changed |= !(tr == m_transformation); | ||||
|         if (mesh_changed) m_transformation.get_matrix(true, true, true, true); // force dirty
 | ||||
|         auto t = supported_facets.timestamp(); | ||||
|         cereal::load_by_value(ar, supported_facets); | ||||
|         mesh_changed |= t != supported_facets.timestamp(); | ||||
|  | @ -1178,6 +1158,8 @@ private: | |||
|         cereal::load_by_value(ar, mmu_segmentation_facets); | ||||
|         mesh_changed |= t != mmu_segmentation_facets.timestamp(); | ||||
|         cereal::load_by_value(ar, config); | ||||
|         cereal::load(ar, text_configuration); | ||||
|         cereal::load(ar, emboss_shape); | ||||
| 		assert(m_mesh); | ||||
| 		if (has_convex_hull) { | ||||
| 			cereal::load_optional(ar, m_convex_hull); | ||||
|  | @ -1191,11 +1173,13 @@ private: | |||
| 	} | ||||
| 	template<class Archive> void save(Archive &ar) const { | ||||
| 		bool has_convex_hull = m_convex_hull.get() != nullptr; | ||||
|         ar(name, source, m_mesh, m_type, m_material_id, m_transformation, m_is_splittable, has_convex_hull, m_text_info, cut_info); | ||||
|         ar(name, source, m_mesh, m_type, m_material_id, m_transformation, m_is_splittable, has_convex_hull, cut_info); | ||||
|         cereal::save_by_value(ar, supported_facets); | ||||
|         cereal::save_by_value(ar, seam_facets); | ||||
|         cereal::save_by_value(ar, mmu_segmentation_facets); | ||||
|         cereal::save_by_value(ar, config); | ||||
|         cereal::save(ar, text_configuration); | ||||
|         cereal::save(ar, emboss_shape); | ||||
| 		if (has_convex_hull) | ||||
| 			cereal::save_optional(ar, m_convex_hull); | ||||
| 	} | ||||
|  | @ -1259,7 +1243,7 @@ public: | |||
|     } | ||||
|     void set_assemble_from_transform(Transform3d& transform) { | ||||
|         m_assemble_initialized = true; | ||||
|         m_assemble_transformation.set_from_transform(transform); | ||||
|         m_assemble_transformation.set_matrix(transform); | ||||
|     } | ||||
|     void set_assemble_offset(const Vec3d& offset) { m_assemble_transformation.set_offset(offset); } | ||||
|     void rotate_assemble(double angle, const Vec3d& axis) { | ||||
|  | @ -1270,13 +1254,13 @@ public: | |||
|     void set_offset_to_assembly(const Vec3d& offset) { m_offset_to_assembly = offset; } | ||||
|     Vec3d get_offset_to_assembly() const { return m_offset_to_assembly; } | ||||
| 
 | ||||
|     const Vec3d& get_offset() const { return m_transformation.get_offset(); } | ||||
|     Vec3d get_offset() const { return m_transformation.get_offset(); } | ||||
|     double get_offset(Axis axis) const { return m_transformation.get_offset(axis); } | ||||
| 
 | ||||
|     void set_offset(const Vec3d& offset) { m_transformation.set_offset(offset); } | ||||
|     void set_offset(Axis axis, double offset) { m_transformation.set_offset(axis, offset); } | ||||
| 
 | ||||
|     const Vec3d& get_rotation() const { return m_transformation.get_rotation(); } | ||||
|     Vec3d get_rotation() const { return m_transformation.get_rotation(); } | ||||
|     double get_rotation(Axis axis) const { return m_transformation.get_rotation(axis); } | ||||
| 
 | ||||
|     void set_rotation(const Vec3d& rotation) { m_transformation.set_rotation(rotation); } | ||||
|  | @ -1284,20 +1268,19 @@ public: | |||
| 
 | ||||
|     // BBS
 | ||||
|     void rotate(Matrix3d rotation_matrix) { | ||||
|         // note: must remove scaling from transformation, otherwise auto-orientation with scaled objects will have problem
 | ||||
|         auto R = get_matrix(true,false,true).matrix().block<3, 3>(0, 0); | ||||
|         auto R = m_transformation.get_rotation_matrix().matrix().block<3, 3>(0, 0); | ||||
|         auto R_new = rotation_matrix * R; | ||||
|         auto euler_angles = Geometry::extract_euler_angles(R_new); | ||||
|         set_rotation(euler_angles); | ||||
|     } | ||||
| 
 | ||||
|     const Vec3d& get_scaling_factor() const { return m_transformation.get_scaling_factor(); } | ||||
|     Vec3d get_scaling_factor() const { return m_transformation.get_scaling_factor(); } | ||||
|     double get_scaling_factor(Axis axis) const { return m_transformation.get_scaling_factor(axis); } | ||||
| 
 | ||||
|     void set_scaling_factor(const Vec3d& scaling_factor) { m_transformation.set_scaling_factor(scaling_factor); } | ||||
|     void set_scaling_factor(Axis axis, double scaling_factor) { m_transformation.set_scaling_factor(axis, scaling_factor); } | ||||
| 
 | ||||
|     const Vec3d& get_mirror() const { return m_transformation.get_mirror(); } | ||||
|     Vec3d get_mirror() const { return m_transformation.get_mirror(); } | ||||
|     double get_mirror(Axis axis) const { return m_transformation.get_mirror(axis); } | ||||
|     bool is_left_handed() const { return m_transformation.is_left_handed(); } | ||||
| 
 | ||||
|  | @ -1306,16 +1289,15 @@ public: | |||
| 
 | ||||
|     // To be called on an external mesh
 | ||||
|     void transform_mesh(TriangleMesh* mesh, bool dont_translate = false) const; | ||||
|     // Calculate a bounding box of a transformed mesh. To be called on an external mesh.
 | ||||
|     BoundingBoxf3 transform_mesh_bounding_box(const TriangleMesh& mesh, bool dont_translate = false) const; | ||||
|     // Transform an external bounding box.
 | ||||
|     // Transform an external bounding box, thus the resulting bounding box is no more snug.
 | ||||
|     BoundingBoxf3 transform_bounding_box(const BoundingBoxf3 &bbox, bool dont_translate = false) const; | ||||
|     // Transform an external vector.
 | ||||
|     Vec3d transform_vector(const Vec3d& v, bool dont_translate = false) const; | ||||
|     // To be called on an external polygon. It does not translate the polygon, only rotates and scales.
 | ||||
|     void transform_polygon(Polygon* polygon) const; | ||||
| 
 | ||||
|     const Transform3d& get_matrix(bool dont_translate = false, bool dont_rotate = false, bool dont_scale = false, bool dont_mirror = false) const { return m_transformation.get_matrix(dont_translate, dont_rotate, dont_scale, dont_mirror); } | ||||
|     const Transform3d& get_matrix() const { return m_transformation.get_matrix(); } | ||||
|     Transform3d get_matrix_no_offset() const { return m_transformation.get_matrix_no_offset(); } | ||||
| 
 | ||||
|     bool is_printable() const { return object->printable && printable && (print_volume_state == ModelInstancePVS_Inside); } | ||||
|     bool is_assemble_initialized() { return m_assemble_initialized; } | ||||
|  |  | |||
							
								
								
									
										547
									
								
								src/libslic3r/NSVGUtils.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,547 @@ | |||
| ///|/ Copyright (c) Prusa Research 2021 - 2022 Filip Sykala @Jony01
 | ||||
| ///|/
 | ||||
| ///|/ PrusaSlicer is released under the terms of the AGPLv3 or higher
 | ||||
| ///|/
 | ||||
| #include "NSVGUtils.hpp" | ||||
| #include <array> | ||||
| #include <charconv> // to_chars
 | ||||
| 
 | ||||
| #include <boost/nowide/iostream.hpp> | ||||
| #include <boost/nowide/fstream.hpp> | ||||
| #include "ClipperUtils.hpp" | ||||
| #include "Emboss.hpp" // heal for shape
 | ||||
| 
 | ||||
| namespace {     | ||||
| using namespace Slic3r; // Polygon
 | ||||
| // see function nsvg__lineTo(NSVGparser* p, float x, float y)
 | ||||
| bool is_line(const float *p, float precision = 1e-4f); | ||||
| // convert curve in path to lines
 | ||||
| struct LinesPath{ | ||||
|     Polygons polygons; | ||||
|     Polylines polylines; }; | ||||
| LinesPath linearize_path(NSVGpath *first_path, const NSVGLineParams ¶m); | ||||
| HealedExPolygons fill_to_expolygons(const LinesPath &lines_path, const NSVGshape &shape, const NSVGLineParams ¶m); | ||||
| HealedExPolygons stroke_to_expolygons(const LinesPath &lines_path, const NSVGshape &shape, const NSVGLineParams ¶m); | ||||
| } // namespace
 | ||||
| 
 | ||||
| namespace Slic3r { | ||||
| 
 | ||||
| ExPolygonsWithIds create_shape_with_ids(const NSVGimage &image, const NSVGLineParams ¶m) | ||||
| { | ||||
|     ExPolygonsWithIds result; | ||||
|     size_t shape_id = 0; | ||||
|     for (NSVGshape *shape_ptr = image.shapes; shape_ptr != NULL; shape_ptr = shape_ptr->next, ++shape_id) { | ||||
|         const NSVGshape &shape = *shape_ptr; | ||||
|         if (!(shape.flags & NSVG_FLAGS_VISIBLE)) | ||||
|             continue; | ||||
| 
 | ||||
|         bool is_fill_used = shape.fill.type != NSVG_PAINT_NONE; | ||||
|         bool is_stroke_used =  | ||||
|             shape.stroke.type != NSVG_PAINT_NONE && | ||||
|             shape.strokeWidth > 1e-5f; | ||||
| 
 | ||||
|         if (!is_fill_used && !is_stroke_used) | ||||
|             continue; | ||||
| 
 | ||||
|         const LinesPath lines_path = linearize_path(shape.paths, param); | ||||
| 
 | ||||
|         if (is_fill_used) { | ||||
|             unsigned unique_id = static_cast<unsigned>(2 * shape_id); | ||||
|             HealedExPolygons expoly = fill_to_expolygons(lines_path, shape, param); | ||||
|             result.push_back({unique_id, expoly.expolygons, expoly.is_healed}); | ||||
|         }         | ||||
|         if (is_stroke_used) { | ||||
|             unsigned unique_id = static_cast<unsigned>(2 * shape_id + 1); | ||||
|             HealedExPolygons expoly = stroke_to_expolygons(lines_path, shape, param); | ||||
|             result.push_back({unique_id, expoly.expolygons, expoly.is_healed}); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     // SVG is used as centered
 | ||||
|     // Do not disturb user by settings of pivot position
 | ||||
|     center(result); | ||||
|     return result; | ||||
| } | ||||
| 
 | ||||
| Polygons to_polygons(const NSVGimage &image, const NSVGLineParams ¶m) | ||||
| { | ||||
|     Polygons result; | ||||
|     for (NSVGshape *shape = image.shapes; shape != NULL; shape = shape->next) { | ||||
|         if (!(shape->flags & NSVG_FLAGS_VISIBLE)) | ||||
|             continue; | ||||
|         if (shape->fill.type == NSVG_PAINT_NONE) | ||||
|             continue; | ||||
|         const LinesPath lines_path = linearize_path(shape->paths, param); | ||||
|         polygons_append(result, lines_path.polygons); | ||||
|         // close polyline to create polygon
 | ||||
|         polygons_append(result, to_polygons(lines_path.polylines));         | ||||
|     } | ||||
|     return result; | ||||
| } | ||||
| 
 | ||||
| void bounds(const NSVGimage &image, Vec2f& min, Vec2f &max) | ||||
| { | ||||
|     for (const NSVGshape *shape = image.shapes; shape != NULL; shape = shape->next) | ||||
|         for (const NSVGpath *path = shape->paths; path != NULL; path = path->next) { | ||||
|             if (min.x() > path->bounds[0]) | ||||
|                 min.x() = path->bounds[0]; | ||||
|             if (min.y() > path->bounds[1]) | ||||
|                 min.y() = path->bounds[1]; | ||||
|             if (max.x() < path->bounds[2]) | ||||
|                 max.x() = path->bounds[2]; | ||||
|             if (max.y() < path->bounds[3]) | ||||
|                 max.y() = path->bounds[3]; | ||||
|         } | ||||
| } | ||||
| 
 | ||||
| NSVGimage_ptr nsvgParseFromFile(const std::string &filename, const char *units, float dpi) | ||||
| { | ||||
|     NSVGimage *image = ::nsvgParseFromFile(filename.c_str(), units, dpi); | ||||
|     return {image, &nsvgDelete}; | ||||
| } | ||||
| 
 | ||||
| std::unique_ptr<std::string> read_from_disk(const std::string &path) | ||||
| { | ||||
|     boost::nowide::ifstream fs{path}; | ||||
|     if (!fs.is_open()) | ||||
|         return nullptr; | ||||
|     std::stringstream ss; | ||||
|     ss << fs.rdbuf(); | ||||
|     return std::make_unique<std::string>(ss.str()); | ||||
| } | ||||
| 
 | ||||
| NSVGimage_ptr nsvgParse(const std::string& file_data, const char *units, float dpi){ | ||||
|     // NOTE: nsvg parser consume data from input(char *)
 | ||||
|     size_t size = file_data.size(); | ||||
|     // file data could be big, so it is allocated on heap
 | ||||
|     std::unique_ptr<char[]> data_copy(new char[size+1]); | ||||
|     memcpy(data_copy.get(), file_data.c_str(), size); | ||||
|     data_copy[size]  = '\0'; // data for nsvg must be null terminated
 | ||||
|     NSVGimage *image = ::nsvgParse(data_copy.get(), units, dpi); | ||||
|     return {image, &nsvgDelete}; | ||||
| } | ||||
| 
 | ||||
| NSVGimage *init_image(EmbossShape::SvgFile &svg_file){ | ||||
|     // is already initialized?
 | ||||
|     if (svg_file.image.get() != nullptr) | ||||
|         return svg_file.image.get(); | ||||
| 
 | ||||
|     if (svg_file.file_data == nullptr) { | ||||
|         // chech if path is known
 | ||||
|         if (svg_file.path.empty()) | ||||
|             return nullptr; | ||||
|         svg_file.file_data = read_from_disk(svg_file.path); | ||||
|         if (svg_file.file_data == nullptr) | ||||
|             return nullptr; | ||||
|     } | ||||
| 
 | ||||
|     // init svg image
 | ||||
|     svg_file.image = nsvgParse(*svg_file.file_data); | ||||
|     if (svg_file.image.get() == NULL) | ||||
|         return nullptr; | ||||
| 
 | ||||
|     return svg_file.image.get(); | ||||
| } | ||||
| 
 | ||||
| size_t get_shapes_count(const NSVGimage &image) | ||||
| { | ||||
|     size_t count = 0; | ||||
|     for (NSVGshape * s = image.shapes; s != NULL; s = s->next) | ||||
|         ++count; | ||||
|     return count; | ||||
| } | ||||
| 
 | ||||
| //void save(const NSVGimage &image, std::ostream &data)
 | ||||
| //{
 | ||||
| //    data << "<?xml version=\"1.0\" encoding=\"UTF-8\" standalone=\"no\"?>";
 | ||||
| //    
 | ||||
| //    // tl .. top left
 | ||||
| //    Vec2f tl(std::numeric_limits<float>::max(), std::numeric_limits<float>::max());
 | ||||
| //    // br .. bottom right
 | ||||
| //    Vec2f br(std::numeric_limits<float>::min(), std::numeric_limits<float>::min());
 | ||||
| //    bounds(image, tl, br);
 | ||||
| //
 | ||||
| //    tl.x() = std::floor(tl.x());
 | ||||
| //    tl.y() = std::floor(tl.y());
 | ||||
| //
 | ||||
| //    br.x() = std::ceil(br.x());
 | ||||
| //    br.y() = std::ceil(br.y());
 | ||||
| //    Vec2f s = br - tl;
 | ||||
| //    Point size = s.cast<Point::coord_type>();
 | ||||
| //
 | ||||
| //    data << "<svg xmlns=\"http://www.w3.org/2000/svg\" "
 | ||||
| //         << "width=\"" << size.x() << "mm\" "
 | ||||
| //         << "height=\"" << size.y() << "mm\" "
 | ||||
| //         << "viewBox=\"0 0 " << size.x() << " " << size.y() << "\" >\n";
 | ||||
| //    data << "<!-- Created with PrusaSlicer (https://www.prusa3d.com/prusaslicer/) -->\n";
 | ||||
| //
 | ||||
| //    std::array<char, 128> buffer;
 | ||||
| //    auto write_point = [&tl, &buffer](std::string &d, const float *p) {
 | ||||
| //        float x = p[0] - tl.x();
 | ||||
| //        float y = p[1] - tl.y();
 | ||||
| //        auto  to_string = [&buffer](float f) -> std::string {
 | ||||
| //            auto [ptr, ec] = std::to_chars(buffer.data(), buffer.data() + buffer.size(), f);
 | ||||
| //            if (ec != std::errc{})
 | ||||
| //                return "0";            
 | ||||
| //            return std::string(buffer.data(), ptr);
 | ||||
| //        };
 | ||||
| //        d += to_string(x) + "," + to_string(y) + " ";
 | ||||
| //    };
 | ||||
| //
 | ||||
| //    for (const NSVGshape *shape = image.shapes; shape != NULL; shape = shape->next) {
 | ||||
| //        enum struct Type { move, line, curve, close }; // https://developer.mozilla.org/en-US/docs/Web/SVG/Attribute/d
 | ||||
| //        Type type = Type::move;
 | ||||
| //        std::string d = "M "; // move on start point
 | ||||
| //        for (const NSVGpath *path = shape->paths; path != NULL; path = path->next) {
 | ||||
| //            if (path->npts <= 1)
 | ||||
| //                continue;
 | ||||
| //
 | ||||
| //            if (type == Type::close) {
 | ||||
| //                type = Type::move;
 | ||||
| //                // NOTE: After close must be a space
 | ||||
| //                d += " M "; // move on start point
 | ||||
| //            }
 | ||||
| //            write_point(d, path->pts);
 | ||||
| //            size_t path_size = static_cast<size_t>(path->npts - 1);
 | ||||
| //
 | ||||
| //            if (path->closed) {
 | ||||
| //                // Do not use last point in path it is duplicit
 | ||||
| //                if (path->npts <= 4)
 | ||||
| //                    continue;
 | ||||
| //                path_size = static_cast<size_t>(path->npts - 4);
 | ||||
| //            }
 | ||||
| //
 | ||||
| //            for (size_t i = 0; i < path_size; i += 3) {
 | ||||
| //                const float *p = &path->pts[i * 2];
 | ||||
| //                if (!::is_line(p)) {
 | ||||
| //                    if (type != Type::curve) {
 | ||||
| //                        type = Type::curve;
 | ||||
| //                        d += "C "; // start sequence of triplets defining curves
 | ||||
| //                    }
 | ||||
| //                    write_point(d, &p[2]);
 | ||||
| //                    write_point(d, &p[4]);
 | ||||
| //                } else {
 | ||||
| //
 | ||||
| //                    if (type != Type::line) {
 | ||||
| //                        type = Type::line;
 | ||||
| //                        d += "L "; // start sequence of line points
 | ||||
| //                    }
 | ||||
| //                }
 | ||||
| //                write_point(d, &p[6]);
 | ||||
| //            }
 | ||||
| //            if (path->closed) {
 | ||||
| //                type = Type::close;
 | ||||
| //                d += "Z"; // start sequence of line points
 | ||||
| //            }
 | ||||
| //        }
 | ||||
| //        if (type != Type::close) {
 | ||||
| //            //type = Type::close;
 | ||||
| //            d += "Z"; // closed path
 | ||||
| //        }
 | ||||
| //        data << "<path fill=\"#D2D2D2\" d=\"" << d << "\" />\n";
 | ||||
| //    }
 | ||||
| //    data << "</svg>\n";
 | ||||
| //}
 | ||||
| //
 | ||||
| //bool save(const NSVGimage &image, const std::string &svg_file_path) 
 | ||||
| //{
 | ||||
| //    std::ofstream file{svg_file_path};
 | ||||
| //    if (!file.is_open())
 | ||||
| //        return false;
 | ||||
| //    save(image, file);
 | ||||
| //    return true;
 | ||||
| //}
 | ||||
| } // namespace Slic3r
 | ||||
| 
 | ||||
| namespace { | ||||
| using namespace Slic3r; // Polygon + Vec2f
 | ||||
| 
 | ||||
| Point::coord_type to_coor(float val, double scale) { return static_cast<Point::coord_type>(std::round(val * scale)); } | ||||
| 
 | ||||
| bool need_flattening(float tessTol, const Vec2f &p1, const Vec2f &p2, const Vec2f &p3, const Vec2f &p4) { | ||||
|     // f .. first
 | ||||
|     // s .. second
 | ||||
|     auto det = [](const Vec2f &f, const Vec2f &s) { | ||||
|         return std::fabs(f.x() * s.y() - f.y() * s.x());  | ||||
|     }; | ||||
| 
 | ||||
|     Vec2f pd  = (p4 - p1); | ||||
|     Vec2f pd2 = (p2 - p4); | ||||
|     float d2  = det(pd2, pd); | ||||
|     Vec2f pd3 = (p3 - p4); | ||||
|     float d3  = det(pd3, pd); | ||||
|     float d23 = d2 + d3; | ||||
| 
 | ||||
|     return (d23 * d23) >= tessTol * pd.squaredNorm(); | ||||
| } | ||||
| 
 | ||||
| // see function nsvg__lineTo(NSVGparser* p, float x, float y)
 | ||||
| bool is_line(const float *p, float precision){ | ||||
|     //Vec2f p1(p[0], p[1]);
 | ||||
|     //Vec2f p2(p[2], p[3]);
 | ||||
|     //Vec2f p3(p[4], p[5]);
 | ||||
|     //Vec2f p4(p[6], p[7]);
 | ||||
|     float dx_3 = (p[6] - p[0]) / 3.f; | ||||
|     float dy_3 = (p[7] - p[1]) / 3.f; | ||||
| 
 | ||||
|     return  | ||||
|         is_approx(p[2], p[0] + dx_3, precision) &&  | ||||
|         is_approx(p[4], p[6] - dx_3, precision) &&  | ||||
|         is_approx(p[3], p[1] + dy_3, precision) && | ||||
|         is_approx(p[5], p[7] - dy_3, precision); | ||||
| } | ||||
| 
 | ||||
| /// <summary>
 | ||||
| /// Convert cubic curve to lines
 | ||||
| /// Inspired by nanosvgrast.h function nsvgRasterize -> nsvg__flattenShape -> nsvg__flattenCubicBez
 | ||||
| /// https://github.com/memononen/nanosvg/blob/f0a3e1034dd22e2e87e5db22401e44998383124e/src/nanosvgrast.h#L335
 | ||||
| /// </summary>
 | ||||
| /// <param name="polygon">Result points</param>
 | ||||
| /// <param name="tessTol">Tesselation tolerance</param>
 | ||||
| /// <param name="p1">Curve point</param>
 | ||||
| /// <param name="p2">Curve point</param>
 | ||||
| /// <param name="p3">Curve point</param>
 | ||||
| /// <param name="p4">Curve point</param>
 | ||||
| /// <param name="level">Actual depth of recursion</param>
 | ||||
| void flatten_cubic_bez(Points &points, float tessTol, const Vec2f& p1, const Vec2f& p2, const Vec2f& p3, const Vec2f& p4, int level) | ||||
| { | ||||
|     if (!need_flattening(tessTol, p1, p2, p3, p4)) { | ||||
|         Point::coord_type x = static_cast<Point::coord_type>(std::round(p4.x())); | ||||
|         Point::coord_type y = static_cast<Point::coord_type>(std::round(p4.y())); | ||||
|         points.emplace_back(x, y); | ||||
|         return; | ||||
|     } | ||||
| 
 | ||||
|     --level; | ||||
|     if (level == 0) | ||||
|         return; | ||||
|      | ||||
|     Vec2f p12  = (p1 + p2) * 0.5f; | ||||
|     Vec2f p23  = (p2 + p3) * 0.5f; | ||||
|     Vec2f p34  = (p3 + p4) * 0.5f; | ||||
|     Vec2f p123 = (p12 + p23) * 0.5f; | ||||
|     Vec2f p234  = (p23 + p34) * 0.5f; | ||||
|     Vec2f p1234 = (p123 + p234) * 0.5f; | ||||
|     flatten_cubic_bez(points, tessTol, p1, p12, p123, p1234, level); | ||||
|     flatten_cubic_bez(points, tessTol, p1234, p234, p34, p4, level); | ||||
| } | ||||
| 
 | ||||
| LinesPath linearize_path(NSVGpath *first_path, const NSVGLineParams ¶m) | ||||
| { | ||||
|     LinesPath result; | ||||
|     Polygons  &polygons  = result.polygons; | ||||
|     Polylines &polylines = result.polylines; | ||||
| 
 | ||||
|     // multiple use of allocated memmory for points between paths
 | ||||
|     Points points; | ||||
|     for (NSVGpath *path = first_path; path != NULL; path = path->next) { | ||||
|         // Flatten path
 | ||||
|         Point::coord_type x = to_coor(path->pts[0], param.scale); | ||||
|         Point::coord_type y = to_coor(path->pts[1], param.scale); | ||||
|         points.emplace_back(x, y); | ||||
|         size_t path_size = (path->npts > 1) ? static_cast<size_t>(path->npts - 1) : 0; | ||||
|         for (size_t i = 0; i < path_size; i += 3) { | ||||
|             const float *p = &path->pts[i * 2]; | ||||
|             if (is_line(p)) { | ||||
|                 // point p4
 | ||||
|                 Point::coord_type xx = to_coor(p[6], param.scale); | ||||
|                 Point::coord_type yy = to_coor(p[7], param.scale); | ||||
|                 points.emplace_back(xx, yy); | ||||
|                 continue; | ||||
|             } | ||||
|             Vec2f p1(p[0], p[1]); | ||||
|             Vec2f p2(p[2], p[3]); | ||||
|             Vec2f p3(p[4], p[5]); | ||||
|             Vec2f p4(p[6], p[7]); | ||||
|             flatten_cubic_bez(points, param.tesselation_tolerance,  | ||||
|                 p1 * param.scale, p2 * param.scale, p3 * param.scale, p4 * param.scale,  | ||||
|                 param.max_level); | ||||
|         } | ||||
|         assert(!points.empty()); | ||||
|         if (points.empty())  | ||||
|             continue; | ||||
| 
 | ||||
|         if (param.is_y_negative) | ||||
|             for (Point &p : points) | ||||
|                 p.y() = -p.y(); | ||||
| 
 | ||||
|         if (path->closed) { | ||||
|             polygons.emplace_back(points); | ||||
|         } else { | ||||
|             polylines.emplace_back(points); | ||||
|         } | ||||
|         // prepare for new path - recycle alocated memory
 | ||||
|         points.clear(); | ||||
|     } | ||||
|     remove_same_neighbor(polygons); | ||||
|     remove_same_neighbor(polylines); | ||||
|     return result; | ||||
| } | ||||
| 
 | ||||
| HealedExPolygons fill_to_expolygons(const LinesPath &lines_path, const NSVGshape &shape, const NSVGLineParams ¶m) | ||||
| { | ||||
|     Polygons fill = lines_path.polygons; // copy
 | ||||
| 
 | ||||
|     // close polyline to create polygon
 | ||||
|     polygons_append(fill, to_polygons(lines_path.polylines)); | ||||
|     if (fill.empty()) | ||||
|         return {}; | ||||
| 
 | ||||
|     // if (shape->fillRule == NSVGfillRule::NSVG_FILLRULE_NONZERO)
 | ||||
|     bool is_non_zero = true; | ||||
|     if (shape.fillRule == NSVGfillRule::NSVG_FILLRULE_EVENODD) | ||||
|         is_non_zero = false; | ||||
| 
 | ||||
|     return Emboss::heal_polygons(fill, is_non_zero, param.max_heal_iteration); | ||||
| } | ||||
| 
 | ||||
| struct DashesParam{ | ||||
|     // first dash length
 | ||||
|     float dash_length = 1.f; // scaled
 | ||||
| 
 | ||||
|     // is current dash .. true
 | ||||
|     // is current space .. false
 | ||||
|     bool is_line = true; | ||||
| 
 | ||||
|     // current index to array
 | ||||
|     unsigned char dash_index = 0; | ||||
|     static constexpr size_t max_dash_array_size = 8; // limitation of nanosvg strokeDashArray
 | ||||
|     std::array<float, max_dash_array_size> dash_array;     // scaled
 | ||||
|     unsigned char dash_count = 0; // count of values in array
 | ||||
| 
 | ||||
|     explicit DashesParam(const NSVGshape &shape, double scale) : | ||||
|         dash_count(shape.strokeDashCount) | ||||
|     { | ||||
|         assert(dash_count > 0); | ||||
|         assert(dash_count <= max_dash_array_size); // limitation of nanosvg strokeDashArray
 | ||||
|         for (size_t i = 0; i < dash_count; ++i) | ||||
|             dash_array[i] = static_cast<float>(shape.strokeDashArray[i] * scale); | ||||
|          | ||||
|         // Figure out dash offset.
 | ||||
|         float all_dash_length = 0; | ||||
|         for (unsigned char j = 0; j < dash_count; ++j) | ||||
|             all_dash_length += dash_array[j]; | ||||
| 
 | ||||
|         if (dash_count%2 == 1) // (shape.strokeDashCount & 1)
 | ||||
|             all_dash_length *= 2.0f; | ||||
| 
 | ||||
|         // Find location inside pattern
 | ||||
|         float dash_offset = fmodf(static_cast<float>(shape.strokeDashOffset * scale), all_dash_length); | ||||
|         if (dash_offset < 0.0f) | ||||
|             dash_offset += all_dash_length; | ||||
| 
 | ||||
|         while (dash_offset > dash_array[dash_index]) { | ||||
|             dash_offset -= dash_array[dash_index]; | ||||
|             dash_index = (dash_index + 1) % shape.strokeDashCount; | ||||
|             is_line    = !is_line; | ||||
|         } | ||||
| 
 | ||||
|         dash_length = dash_array[dash_index] - dash_offset; | ||||
|     } | ||||
| }; | ||||
| 
 | ||||
| Polylines to_dashes(const Polyline &polyline, const DashesParam& param) | ||||
| { | ||||
|     Polylines dashes; | ||||
|     Polyline dash; // cache for one dash in dashed line
 | ||||
|     Point prev_point;         | ||||
|      | ||||
|     bool is_line = param.is_line; | ||||
|     unsigned char dash_index = param.dash_index; | ||||
|     float dash_length = param.dash_length; // current rest of dash distance
 | ||||
|     for (const Point &point : polyline.points) { | ||||
|         if (&point == &polyline.points.front()) { | ||||
|             // is first point
 | ||||
|             prev_point = point; // copy
 | ||||
|             continue; | ||||
|         } | ||||
| 
 | ||||
|         Point diff = point - prev_point; | ||||
|         float line_segment_length = diff.cast<float>().norm(); | ||||
|         while (dash_length < line_segment_length) { | ||||
|             // Calculate intermediate point
 | ||||
|             float d = dash_length / line_segment_length; | ||||
|             Point move_point   = diff * d; | ||||
|             Point intermediate = prev_point + move_point; | ||||
|              | ||||
|             // add Dash in stroke
 | ||||
|             if (is_line) { | ||||
|                 if (dash.empty()) { | ||||
|                     dashes.emplace_back(Points{prev_point, intermediate}); | ||||
|                 } else { | ||||
|                     dash.append(prev_point); | ||||
|                     dash.append(intermediate); | ||||
|                     dashes.push_back(dash); | ||||
|                     dash.clear(); | ||||
|                 } | ||||
|             } | ||||
| 
 | ||||
|             diff -= move_point; | ||||
|             line_segment_length -= dash_length; | ||||
|             prev_point = intermediate; | ||||
| 
 | ||||
|             // Advance dash pattern
 | ||||
|             is_line = !is_line; | ||||
|             dash_index = (dash_index + 1) % param.dash_count; | ||||
|             dash_length = param.dash_array[dash_index]; | ||||
|         } | ||||
| 
 | ||||
|         if (is_line) | ||||
|             dash.append(prev_point); | ||||
|         dash_length -= line_segment_length; | ||||
|         prev_point = point; // copy
 | ||||
|     } | ||||
| 
 | ||||
|     // add last dash
 | ||||
|     if (is_line){ | ||||
|         assert(!dash.empty()); | ||||
|         dash.append(prev_point); // prev_point == polyline.points.back()
 | ||||
|         dashes.push_back(dash); | ||||
|     } | ||||
|     return dashes; | ||||
| } | ||||
| 
 | ||||
| HealedExPolygons stroke_to_expolygons(const LinesPath &lines_path, const NSVGshape &shape, const NSVGLineParams ¶m) | ||||
| { | ||||
|     // convert stroke to polygon
 | ||||
|     ClipperLib::JoinType join_type = ClipperLib::JoinType::jtSquare; | ||||
|     switch (static_cast<NSVGlineJoin>(shape.strokeLineJoin)) { | ||||
|     case NSVGlineJoin::NSVG_JOIN_BEVEL: join_type = ClipperLib::JoinType::jtSquare; break; | ||||
|     case NSVGlineJoin::NSVG_JOIN_MITER: join_type = ClipperLib::JoinType::jtMiter; break; | ||||
|     case NSVGlineJoin::NSVG_JOIN_ROUND: join_type = ClipperLib::JoinType::jtRound; break; | ||||
|     } | ||||
| 
 | ||||
|     double mitter = shape.miterLimit * param.scale; | ||||
|     if (join_type == ClipperLib::JoinType::jtRound) { | ||||
|         // mitter is used as ArcTolerance
 | ||||
|         // http://www.angusj.com/delphi/clipper/documentation/Docs/Units/ClipperLib/Classes/ClipperOffset/Properties/ArcTolerance.htm
 | ||||
|         mitter = std::pow(param.tesselation_tolerance, 1/3.); | ||||
|     } | ||||
|     float stroke_width = static_cast<float>(shape.strokeWidth * param.scale); | ||||
| 
 | ||||
|     ClipperLib::EndType end_type = ClipperLib::EndType::etOpenButt; | ||||
|     switch (static_cast<NSVGlineCap>(shape.strokeLineCap)) { | ||||
|     case NSVGlineCap::NSVG_CAP_BUTT: end_type = ClipperLib::EndType::etOpenButt; break; | ||||
|     case NSVGlineCap::NSVG_CAP_ROUND: end_type = ClipperLib::EndType::etOpenRound; break; | ||||
|     case NSVGlineCap::NSVG_CAP_SQUARE: end_type = ClipperLib::EndType::etOpenSquare; break; | ||||
|     } | ||||
| 
 | ||||
|     Polygons result; | ||||
|     if (shape.strokeDashCount > 0) { | ||||
|         DashesParam params(shape, param.scale); | ||||
|         Polylines dashes; | ||||
|         for (const Polyline &polyline : lines_path.polylines) | ||||
|             polylines_append(dashes, to_dashes(polyline, params)); | ||||
|         for (const Polygon &polygon : lines_path.polygons) | ||||
|             polylines_append(dashes, to_dashes(to_polyline(polygon), params)); | ||||
|         result = offset(dashes, stroke_width / 2, join_type, mitter, end_type); | ||||
|     } else { | ||||
|         result = contour_to_polygons(lines_path.polygons, stroke_width, join_type, mitter); | ||||
|         polygons_append(result, offset(lines_path.polylines, stroke_width / 2, join_type, mitter, end_type));     | ||||
|     } | ||||
| 
 | ||||
|     bool is_non_zero = true; | ||||
|     return Emboss::heal_polygons(result, is_non_zero, param.max_heal_iteration); | ||||
| } | ||||
| 
 | ||||
| } // namespace
 | ||||
							
								
								
									
										86
									
								
								src/libslic3r/NSVGUtils.hpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,86 @@ | |||
| ///|/ Copyright (c) Prusa Research 2021 Filip Sykala @Jony01
 | ||||
| ///|/
 | ||||
| ///|/ PrusaSlicer is released under the terms of the AGPLv3 or higher
 | ||||
| ///|/
 | ||||
| #ifndef slic3r_NSVGUtils_hpp_ | ||||
| #define slic3r_NSVGUtils_hpp_ | ||||
| 
 | ||||
| #include <memory> | ||||
| #include <string> | ||||
| #include <sstream> | ||||
| #include "Polygon.hpp" | ||||
| #include "ExPolygon.hpp" | ||||
| #include "EmbossShape.hpp" // ExPolygonsWithIds
 | ||||
| #include "nanosvg/nanosvg.h"    // load SVG file
 | ||||
| 
 | ||||
| // Helper function to work with nano svg
 | ||||
| namespace Slic3r { | ||||
| 
 | ||||
| /// <summary>
 | ||||
| /// Paramreters for conversion curve from SVG to lines in Polygon
 | ||||
| /// </summary>
 | ||||
| struct NSVGLineParams | ||||
| { | ||||
|     // Smaller will divide curve to more lines
 | ||||
|     // NOTE: Value is in image scale
 | ||||
|     double tesselation_tolerance = 10.f; | ||||
| 
 | ||||
|     // Maximal depth of recursion for conversion curve to lines
 | ||||
|     int max_level = 10; | ||||
| 
 | ||||
|     // Multiplicator of point coors
 | ||||
|     // NOTE: Every point coor from image(float) is multiplied by scale and rounded to integer --> Slic3r::Point
 | ||||
|     double scale = 1. / SCALING_FACTOR; | ||||
| 
 | ||||
|     // Flag wether y is negative, when true than y coor is multiplied by -1
 | ||||
|     bool is_y_negative = true; | ||||
| 
 | ||||
|     // Is used only with rounded Stroke
 | ||||
|     double arc_tolerance = 1.; | ||||
| 
 | ||||
|     // Maximal count of heal iteration
 | ||||
|     unsigned max_heal_iteration = 10; | ||||
| 
 | ||||
|     explicit NSVGLineParams(double tesselation_tolerance):  | ||||
|         tesselation_tolerance(tesselation_tolerance),  | ||||
|         arc_tolerance(std::pow(tesselation_tolerance, 1/3.)) | ||||
|     {} | ||||
| }; | ||||
| 
 | ||||
| /// <summary>
 | ||||
| /// Convert .svg opened by nanoSvg to shapes stored in expolygons with ids
 | ||||
| /// </summary>
 | ||||
| /// <param name="image">Parsed svg file by NanoSvg</param>
 | ||||
| /// <param name="tesselation_tolerance">Smaller will divide curve to more lines
 | ||||
| /// NOTE: Value is in image scale</param>
 | ||||
| /// <param name="max_level">Maximal depth for conversion curve to lines</param>
 | ||||
| /// <param name="scale">Multiplicator of point coors
 | ||||
| /// NOTE: Every point coor from image(float) is multiplied by scale and rounded to integer</param>
 | ||||
| /// <returns>Shapes from svg image - fill + stroke</returns>
 | ||||
| ExPolygonsWithIds create_shape_with_ids(const NSVGimage &image, const NSVGLineParams ¶m); | ||||
| 
 | ||||
| // help functions - prepare to be tested
 | ||||
| /// <param name="is_y_negative">Flag is y negative, when true than y coor is multiplied by -1</param>
 | ||||
| Polygons to_polygons(const NSVGimage &image, const NSVGLineParams ¶m); | ||||
| 
 | ||||
| void bounds(const NSVGimage &image, Vec2f &min, Vec2f &max); | ||||
| 
 | ||||
| // read text data from file
 | ||||
| std::unique_ptr<std::string> read_from_disk(const std::string &path); | ||||
| 
 | ||||
| using NSVGimage_ptr = std::unique_ptr<NSVGimage, void (*)(NSVGimage*)>; | ||||
| NSVGimage_ptr nsvgParseFromFile(const std::string &svg_file_path, const char *units = "mm", float dpi = 96.0f); | ||||
| NSVGimage_ptr nsvgParse(const std::string& file_data, const char *units = "mm", float dpi = 96.0f); | ||||
| NSVGimage *init_image(EmbossShape::SvgFile &svg_file); | ||||
| 
 | ||||
| /// <summary>
 | ||||
| /// Iterate over shapes and calculate count
 | ||||
| /// </summary>
 | ||||
| /// <param name="image">Contain pointer to first shape</param>
 | ||||
| /// <returns>Count of shapes</returns>
 | ||||
| size_t get_shapes_count(const NSVGimage &image); | ||||
| 
 | ||||
| //void save(const NSVGimage &image, std::ostream &data);
 | ||||
| //bool save(const NSVGimage &image, const std::string &svg_file_path);
 | ||||
| } // namespace Slic3r
 | ||||
| #endif // slic3r_NSVGUtils_hpp_
 | ||||
|  | @ -59,65 +59,12 @@ Pointf3s transform(const Pointf3s& points, const Transform3d& t) | |||
| 
 | ||||
| void Point::rotate(double angle, const Point ¢er) | ||||
| { | ||||
|     double cur_x = (double)(*this)(0); | ||||
|     double cur_y = (double)(*this)(1); | ||||
|     Vec2d  cur = this->cast<double>(); | ||||
|     double s   = ::sin(angle); | ||||
|     double c   = ::cos(angle); | ||||
|     double dx    = cur_x - (double)center(0); | ||||
|     double dy    = cur_y - (double)center(1); | ||||
|     (*this)(0) = (coord_t)round( (double)center(0) + c * dx - s * dy ); | ||||
|     (*this)(1) = (coord_t)round( (double)center(1) + c * dy + s * dx ); | ||||
| } | ||||
| 
 | ||||
| int Point::nearest_point_index(const Points &points) const | ||||
| { | ||||
|     PointConstPtrs p; | ||||
|     p.reserve(points.size()); | ||||
|     for (Points::const_iterator it = points.begin(); it != points.end(); ++it) | ||||
|         p.push_back(&*it); | ||||
|     return this->nearest_point_index(p); | ||||
| } | ||||
| 
 | ||||
| int Point::nearest_point_index(const PointConstPtrs &points) const | ||||
| { | ||||
|     int idx = -1; | ||||
|     double distance = -1;  // double because long is limited to 2147483647 on some platforms and it's not enough
 | ||||
|      | ||||
|     for (PointConstPtrs::const_iterator it = points.begin(); it != points.end(); ++it) { | ||||
|         /* If the X distance of the candidate is > than the total distance of the
 | ||||
|            best previous candidate, we know we don't want it */ | ||||
|         double d = sqr<double>((*this)(0) - (*it)->x()); | ||||
|         if (distance != -1 && d > distance) continue; | ||||
|          | ||||
|         /* If the Y distance of the candidate is > than the total distance of the
 | ||||
|            best previous candidate, we know we don't want it */ | ||||
|         d += sqr<double>((*this)(1) - (*it)->y()); | ||||
|         if (distance != -1 && d > distance) continue; | ||||
|          | ||||
|         idx = it - points.begin(); | ||||
|         distance = d; | ||||
|          | ||||
|         if (distance < EPSILON) break; | ||||
|     } | ||||
|      | ||||
|     return idx; | ||||
| } | ||||
| 
 | ||||
| int Point::nearest_point_index(const PointPtrs &points) const | ||||
| { | ||||
|     PointConstPtrs p; | ||||
|     p.reserve(points.size()); | ||||
|     for (PointPtrs::const_iterator it = points.begin(); it != points.end(); ++it) | ||||
|         p.push_back(*it); | ||||
|     return this->nearest_point_index(p); | ||||
| } | ||||
| 
 | ||||
| bool Point::nearest_point(const Points &points, Point* point) const | ||||
| { | ||||
|     int idx = this->nearest_point_index(points); | ||||
|     if (idx == -1) return false; | ||||
|     *point = points.at(idx); | ||||
|     return true; | ||||
|     auto   d   = cur - center.cast<double>(); | ||||
|     this->x() = fast_round_up<coord_t>(center.x() + c * d.x() - s * d.y()); | ||||
|     this->y() = fast_round_up<coord_t>(center.y() + s * d.x() + c * d.y()); | ||||
| } | ||||
| 
 | ||||
| /* Three points are a counter-clockwise turn if ccw > 0, clockwise if
 | ||||
|  | @ -191,7 +138,7 @@ Point Point::projection_onto(const Line &line) const | |||
|     return ((line.a - *this).cast<double>().squaredNorm() < (line.b - *this).cast<double>().squaredNorm()) ? line.a : line.b; | ||||
| } | ||||
| 
 | ||||
| bool has_duplicate_points(std::vector<Point> &&pts) | ||||
| bool has_duplicate_points(Points &&pts) | ||||
| { | ||||
|     std::sort(pts.begin(), pts.end()); | ||||
|     for (size_t i = 1; i < pts.size(); ++ i) | ||||
|  | @ -200,6 +147,24 @@ bool has_duplicate_points(std::vector<Point> &&pts) | |||
|     return false; | ||||
| } | ||||
| 
 | ||||
| Points collect_duplicates(Points pts /* Copy */) | ||||
| { | ||||
|     std::sort(pts.begin(), pts.end()); | ||||
|     Points duplicits; | ||||
|     const Point *prev = &pts.front(); | ||||
|     for (size_t i = 1; i < pts.size(); ++i) { | ||||
|         const Point *act = &pts[i]; | ||||
|         if (*prev == *act) { | ||||
|             // duplicit point
 | ||||
|             if (!duplicits.empty() && duplicits.back() == *act) | ||||
|                 continue; // only unique duplicits
 | ||||
|             duplicits.push_back(*act); | ||||
|         } | ||||
|         prev = act; | ||||
|     } | ||||
|     return duplicits; | ||||
| } | ||||
| 
 | ||||
| template<bool IncludeBoundary> | ||||
| BoundingBox get_extents(const Points &pts) | ||||
| {  | ||||
|  | @ -231,6 +196,58 @@ BoundingBoxf get_extents(const std::vector<Vec2d> &pts) | |||
|     return bbox; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| int Point::nearest_point_index(const Points &points) const | ||||
| { | ||||
|     PointConstPtrs p; | ||||
|     p.reserve(points.size()); | ||||
|     for (Points::const_iterator it = points.begin(); it != points.end(); ++it) | ||||
|         p.push_back(&*it); | ||||
|     return this->nearest_point_index(p); | ||||
| } | ||||
| 
 | ||||
| int Point::nearest_point_index(const PointConstPtrs &points) const | ||||
| { | ||||
|     int idx = -1; | ||||
|     double distance = -1;  // double because long is limited to 2147483647 on some platforms and it's not enough
 | ||||
|      | ||||
|     for (PointConstPtrs::const_iterator it = points.begin(); it != points.end(); ++it) { | ||||
|         /* If the X distance of the candidate is > than the total distance of the
 | ||||
|            best previous candidate, we know we don't want it */ | ||||
|         double d = sqr<double>((*this)(0) - (*it)->x()); | ||||
|         if (distance != -1 && d > distance) continue; | ||||
|          | ||||
|         /* If the Y distance of the candidate is > than the total distance of the
 | ||||
|            best previous candidate, we know we don't want it */ | ||||
|         d += sqr<double>((*this)(1) - (*it)->y()); | ||||
|         if (distance != -1 && d > distance) continue; | ||||
|          | ||||
|         idx = it - points.begin(); | ||||
|         distance = d; | ||||
|          | ||||
|         if (distance < EPSILON) break; | ||||
|     } | ||||
|      | ||||
|     return idx; | ||||
| } | ||||
| 
 | ||||
| int Point::nearest_point_index(const PointPtrs &points) const | ||||
| { | ||||
|     PointConstPtrs p; | ||||
|     p.reserve(points.size()); | ||||
|     for (PointPtrs::const_iterator it = points.begin(); it != points.end(); ++it) | ||||
|         p.push_back(*it); | ||||
|     return this->nearest_point_index(p); | ||||
| } | ||||
| 
 | ||||
| bool Point::nearest_point(const Points &points, Point* point) const | ||||
| { | ||||
|     int idx = this->nearest_point_index(points); | ||||
|     if (idx == -1) return false; | ||||
|     *point = points.at(idx); | ||||
|     return true; | ||||
| } | ||||
| 
 | ||||
| std::ostream& operator<<(std::ostream &stm, const Vec2d &pointf) | ||||
| { | ||||
|     return stm << pointf(0) << "," << pointf(1); | ||||
|  |  | |||
|  | @ -53,9 +53,9 @@ using Vec3i64 = Eigen::Matrix<int64_t,  3, 1, Eigen::DontAlign>; | |||
| // Vector types with a double coordinate base type.
 | ||||
| using Vec2f   = Eigen::Matrix<float,    2, 1, Eigen::DontAlign>; | ||||
| using Vec3f   = Eigen::Matrix<float,    3, 1, Eigen::DontAlign>; | ||||
| using Vec4f   = Eigen::Matrix<float,    4, 1, Eigen::DontAlign>; | ||||
| using Vec2d   = Eigen::Matrix<double,   2, 1, Eigen::DontAlign>; | ||||
| using Vec3d   = Eigen::Matrix<double,   3, 1, Eigen::DontAlign>; | ||||
| using Vec4f   = Eigen::Matrix<float,    4, 1, Eigen::DontAlign>; | ||||
| using Vec4d   = Eigen::Matrix<double,   4, 1, Eigen::DontAlign>; | ||||
| 
 | ||||
| using Points         = std::vector<Point>; | ||||
|  | @ -116,9 +116,8 @@ inline Eigen::Matrix<typename Derived::Scalar, 2, 1, Eigen::DontAlign> perp(cons | |||
| } | ||||
| 
 | ||||
| // Angle from v1 to v2, returning double atan2(y, x) normalized to <-PI, PI>.
 | ||||
| template <typename Derived, typename Derived2> | ||||
| inline double angle(const Eigen::MatrixBase<Derived>& v1, const Eigen::MatrixBase<Derived2>& v2) | ||||
| { | ||||
| template<typename Derived, typename Derived2> | ||||
| inline double angle(const Eigen::MatrixBase<Derived> &v1, const Eigen::MatrixBase<Derived2> &v2) { | ||||
|     static_assert(Derived::IsVectorAtCompileTime && int(Derived::SizeAtCompileTime) == 2, "angle(): first parameter is not a 2D vector"); | ||||
|     static_assert(Derived2::IsVectorAtCompileTime && int(Derived2::SizeAtCompileTime) == 2, "angle(): second parameter is not a 2D vector"); | ||||
|     auto v1d = v1.template cast<double>(); | ||||
|  | @ -153,6 +152,29 @@ inline std::string to_string(const Vec3d   &pt) { return std::string("[") + floa | |||
| std::vector<Vec3f> transform(const std::vector<Vec3f>& points, const Transform3f& t); | ||||
| Pointf3s transform(const Pointf3s& points, const Transform3d& t); | ||||
| 
 | ||||
| /// <summary>
 | ||||
| /// Check whether transformation matrix contains odd number of mirroring.
 | ||||
| /// NOTE: In code is sometime function named is_left_handed
 | ||||
| /// </summary>
 | ||||
| /// <param name="transform">Transformation to check</param>
 | ||||
| /// <returns>Is positive determinant</returns>
 | ||||
| inline bool has_reflection(const Transform3d &transform) { return transform.matrix().determinant() < 0; } | ||||
| 
 | ||||
| /// <summary>
 | ||||
| /// Getter on base of transformation matrix
 | ||||
| /// </summary>
 | ||||
| /// <param name="index">column index</param>
 | ||||
| /// <param name="transform">source transformation</param>
 | ||||
| /// <returns>Base of transformation matrix</returns>
 | ||||
| inline const Vec3d get_base(unsigned index, const Transform3d &transform) { return transform.linear().col(index); } | ||||
| inline const Vec3d get_x_base(const Transform3d &transform) { return get_base(0, transform); } | ||||
| inline const Vec3d get_y_base(const Transform3d &transform) { return get_base(1, transform); } | ||||
| inline const Vec3d get_z_base(const Transform3d &transform) { return get_base(2, transform); } | ||||
| inline const Vec3d get_base(unsigned index, const Transform3d::LinearPart &transform) { return transform.col(index); } | ||||
| inline const Vec3d get_x_base(const Transform3d::LinearPart &transform) { return get_base(0, transform); } | ||||
| inline const Vec3d get_y_base(const Transform3d::LinearPart &transform) { return get_base(1, transform); } | ||||
| inline const Vec3d get_z_base(const Transform3d::LinearPart &transform) { return get_base(2, transform); } | ||||
| 
 | ||||
| template<int N, class T> using Vec = Eigen::Matrix<T,  N, 1, Eigen::DontAlign, N, 1>; | ||||
| 
 | ||||
| class Point : public Vec2crd | ||||
|  | @ -163,15 +185,16 @@ public: | |||
|     Point() : Vec2crd(0, 0) {} | ||||
|     Point(int32_t x, int32_t y) : Vec2crd(coord_t(x), coord_t(y)) {} | ||||
|     Point(int64_t x, int64_t y) : Vec2crd(coord_t(x), coord_t(y)) {} | ||||
|     Point(double x, double y) : Vec2crd(coord_t(lrint(x)), coord_t(lrint(y))) {} | ||||
|     Point(double x, double y) : Vec2crd(coord_t(std::round(x)), coord_t(std::round(y))) {} | ||||
|     Point(const Point &rhs) { *this = rhs; } | ||||
| 	explicit Point(const Vec2d& rhs) : Vec2crd(coord_t(lrint(rhs.x())), coord_t(lrint(rhs.y()))) {} | ||||
| 	explicit Point(const Vec2d& rhs) : Vec2crd(coord_t(std::round(rhs.x())), coord_t(std::round(rhs.y()))) {} | ||||
| 	// This constructor allows you to construct Point from Eigen expressions
 | ||||
|     // This constructor has to be implicit (non-explicit) to allow implicit conversion from Eigen expressions.
 | ||||
|     template<typename OtherDerived> | ||||
|     Point(const Eigen::MatrixBase<OtherDerived> &other) : Vec2crd(other) {} | ||||
|     static Point new_scale(coordf_t x, coordf_t y) { return Point(coord_t(scale_(x)), coord_t(scale_(y))); } | ||||
|     static Point new_scale(const Vec2d &v) { return Point(coord_t(scale_(v.x())), coord_t(scale_(v.y()))); } | ||||
|     static Point new_scale(const Vec2f &v) { return Point(coord_t(scale_(v.x())), coord_t(scale_(v.y()))); } | ||||
|     template<typename OtherDerived> | ||||
|     static Point new_scale(const Eigen::MatrixBase<OtherDerived> &v) { return Point(coord_t(scale_(v.x())), coord_t(scale_(v.y()))); } | ||||
| 
 | ||||
|     // This method allows you to assign Eigen expressions to MyVectorType
 | ||||
|     template<typename OtherDerived> | ||||
|  | @ -297,16 +320,16 @@ BoundingBoxf get_extents(const std::vector<Vec2d> &pts); | |||
| 
 | ||||
| // Test for duplicate points in a vector of points.
 | ||||
| // The points are copied, sorted and checked for duplicates globally.
 | ||||
| bool        has_duplicate_points(std::vector<Point> &&pts); | ||||
| inline bool has_duplicate_points(const std::vector<Point> &pts) | ||||
| bool        has_duplicate_points(Points &&pts); | ||||
| inline bool has_duplicate_points(const Points &pts) | ||||
| { | ||||
|     std::vector<Point> cpy = pts; | ||||
|     Points cpy = pts; | ||||
|     return has_duplicate_points(std::move(cpy)); | ||||
| } | ||||
| 
 | ||||
| // Test for duplicate points in a vector of points.
 | ||||
| // Only successive points are checked for equality.
 | ||||
| inline bool has_duplicate_successive_points(const std::vector<Point> &pts) | ||||
| inline bool has_duplicate_successive_points(const Points &pts) | ||||
| { | ||||
|     for (size_t i = 1; i < pts.size(); ++ i) | ||||
|         if (pts[i - 1] == pts[i]) | ||||
|  | @ -316,11 +339,14 @@ inline bool has_duplicate_successive_points(const std::vector<Point> &pts) | |||
| 
 | ||||
| // Test for duplicate points in a vector of points.
 | ||||
| // Only successive points are checked for equality. Additionally, first and last points are compared for equality.
 | ||||
| inline bool has_duplicate_successive_points_closed(const std::vector<Point> &pts) | ||||
| inline bool has_duplicate_successive_points_closed(const Points &pts) | ||||
| { | ||||
|     return has_duplicate_successive_points(pts) || (pts.size() >= 2 && pts.front() == pts.back()); | ||||
| } | ||||
| 
 | ||||
| // Collect adjecent(duplicit points)
 | ||||
| Points collect_duplicates(Points pts /* Copy */); | ||||
| 
 | ||||
| inline bool shorter_then(const Point& p0, const coord_t len) | ||||
| { | ||||
|     if (p0.x() > len || p0.x() < -len) | ||||
|  | @ -341,7 +367,7 @@ namespace int128 { | |||
| 
 | ||||
| // To be used by std::unordered_map, std::unordered_multimap and friends.
 | ||||
| struct PointHash { | ||||
|     size_t operator()(const Vec2crd &pt) const { | ||||
|     size_t operator()(const Vec2crd &pt) const noexcept { | ||||
|         return coord_t((89 * 31 + int64_t(pt.x())) * 31 + pt.y()); | ||||
|     } | ||||
| }; | ||||
|  | @ -570,6 +596,27 @@ inline coord_t align_to_grid(coord_t coord, coord_t spacing, coord_t base) | |||
| inline Point   align_to_grid(Point   coord, Point   spacing, Point   base) | ||||
|     { return Point(align_to_grid(coord.x(), spacing.x(), base.x()), align_to_grid(coord.y(), spacing.y(), base.y())); } | ||||
| 
 | ||||
| // MinMaxLimits
 | ||||
| template<typename T> struct MinMax { T min; T max;}; | ||||
| template<typename T> | ||||
| static bool apply(std::optional<T> &val, const MinMax<T> &limit) { | ||||
|     if (!val.has_value()) return false; | ||||
|     return apply<T>(*val, limit); | ||||
| } | ||||
| template<typename T> | ||||
| static bool apply(T &val, const MinMax<T> &limit) | ||||
| { | ||||
|     if (val > limit.max) { | ||||
|         val = limit.max; | ||||
|         return true; | ||||
|     } | ||||
|     if (val < limit.min) { | ||||
|         val = limit.min; | ||||
|         return true; | ||||
|     } | ||||
|     return false; | ||||
| } | ||||
| 
 | ||||
| } // namespace Slic3r
 | ||||
| 
 | ||||
| // start Boost
 | ||||
|  |  | |||
|  | @ -1,3 +1,14 @@ | |||
| ///|/ Copyright (c) Prusa Research 2016 - 2023 Vojtěch Bubník @bubnikv, Filip Sykala @Jony01, Lukáš Matěna @lukasmatena, Tomáš Mészáros @tamasmeszaros, Enrico Turri @enricoturri1966
 | ||||
| ///|/ Copyright (c) Slic3r 2013 - 2015 Alessandro Ranellucci @alranel
 | ||||
| ///|/ Copyright (c) 2014 Petr Ledvina @ledvinap
 | ||||
| ///|/
 | ||||
| ///|/ ported from lib/Slic3r/Polygon.pm:
 | ||||
| ///|/ Copyright (c) Prusa Research 2017 - 2022 Vojtěch Bubník @bubnikv
 | ||||
| ///|/ Copyright (c) Slic3r 2011 - 2014 Alessandro Ranellucci @alranel
 | ||||
| ///|/ Copyright (c) 2012 Mark Hindess
 | ||||
| ///|/
 | ||||
| ///|/ PrusaSlicer is released under the terms of the AGPLv3 or higher
 | ||||
| ///|/
 | ||||
| #include "BoundingBox.hpp" | ||||
| #include "ClipperUtils.hpp" | ||||
| #include "Exception.hpp" | ||||
|  | @ -454,6 +465,38 @@ bool has_duplicate_points(const Polygons &polys) | |||
| #endif | ||||
| } | ||||
| 
 | ||||
| bool remove_same_neighbor(Polygon &polygon) | ||||
| { | ||||
|     Points &points = polygon.points; | ||||
|     if (points.empty()) | ||||
|         return false; | ||||
|     auto last = std::unique(points.begin(), points.end()); | ||||
| 
 | ||||
|     // remove first and last neighbor duplication
 | ||||
|     if (const Point &last_point = *(last - 1); last_point == points.front()) { | ||||
|         --last; | ||||
|     } | ||||
| 
 | ||||
|     // no duplicits
 | ||||
|     if (last == points.end()) | ||||
|         return false; | ||||
| 
 | ||||
|     points.erase(last, points.end()); | ||||
|     return true; | ||||
| } | ||||
| 
 | ||||
| bool remove_same_neighbor(Polygons &polygons) | ||||
| { | ||||
|     if (polygons.empty()) | ||||
|         return false; | ||||
|     bool exist = false; | ||||
|     for (Polygon &polygon : polygons) | ||||
|         exist |= remove_same_neighbor(polygon); | ||||
|     // remove empty polygons
 | ||||
|     polygons.erase(std::remove_if(polygons.begin(), polygons.end(), [](const Polygon &p) { return p.points.size() <= 2; }), polygons.end()); | ||||
|     return exist; | ||||
| } | ||||
| 
 | ||||
| static inline bool is_stick(const Point &p1, const Point &p2, const Point &p3) | ||||
| { | ||||
|     Point v1 = p2 - p1; | ||||
|  |  | |||
|  | @ -1,3 +1,13 @@ | |||
| ///|/ Copyright (c) Prusa Research 2016 - 2023 Tomáš Mészáros @tamasmeszaros, Vojtěch Bubník @bubnikv, Lukáš Matěna @lukasmatena, Lukáš Hejl @hejllukas, Filip Sykala @Jony01, Oleksandra Iushchenko @YuSanka
 | ||||
| ///|/ Copyright (c) Slic3r 2013 - 2016 Alessandro Ranellucci @alranel
 | ||||
| ///|/
 | ||||
| ///|/ ported from lib/Slic3r/Polygon.pm:
 | ||||
| ///|/ Copyright (c) Prusa Research 2017 - 2022 Vojtěch Bubník @bubnikv
 | ||||
| ///|/ Copyright (c) Slic3r 2011 - 2014 Alessandro Ranellucci @alranel
 | ||||
| ///|/ Copyright (c) 2012 Mark Hindess
 | ||||
| ///|/
 | ||||
| ///|/ PrusaSlicer is released under the terms of the AGPLv3 or higher
 | ||||
| ///|/
 | ||||
| #ifndef slic3r_Polygon_hpp_ | ||||
| #define slic3r_Polygon_hpp_ | ||||
| 
 | ||||
|  | @ -5,6 +15,7 @@ | |||
| #include <vector> | ||||
| #include <string> | ||||
| #include "Line.hpp" | ||||
| #include "Point.hpp" | ||||
| #include "MultiPoint.hpp" | ||||
| #include "Polyline.hpp" | ||||
| 
 | ||||
|  | @ -112,6 +123,10 @@ inline bool has_duplicate_points(Polygon &&poly)      { return has_duplicate_poi | |||
| inline bool has_duplicate_points(const Polygon &poly) { return has_duplicate_points(poly.points); } | ||||
| bool        has_duplicate_points(const Polygons &polys); | ||||
| 
 | ||||
| // Return True when erase some otherwise False.
 | ||||
| bool remove_same_neighbor(Polygon &polygon); | ||||
| bool remove_same_neighbor(Polygons &polygons); | ||||
| 
 | ||||
| inline double total_length(const Polygons &polylines) { | ||||
|     double total = 0; | ||||
|     for (Polygons::const_iterator it = polylines.begin(); it != polylines.end(); ++it) | ||||
|  | @ -245,7 +260,19 @@ inline Polylines to_polylines(Polygons &&polys) | |||
|     return polylines; | ||||
| } | ||||
| 
 | ||||
| inline Polygons to_polygons(const std::vector<Points> &paths) | ||||
| // close polyline to polygon (connect first and last point in polyline)
 | ||||
| inline Polygons to_polygons(const Polylines &polylines) | ||||
| { | ||||
|     Polygons out; | ||||
|     out.reserve(polylines.size()); | ||||
|     for (const Polyline &polyline : polylines) { | ||||
|         if (polyline.size()) | ||||
|         out.emplace_back(polyline.points); | ||||
|     } | ||||
|     return out; | ||||
| } | ||||
| 
 | ||||
| inline Polygons to_polygons(const VecOfPoints &paths) | ||||
| { | ||||
|     Polygons out; | ||||
|     out.reserve(paths.size()); | ||||
|  | @ -254,7 +281,7 @@ inline Polygons to_polygons(const std::vector<Points> &paths) | |||
|     return out; | ||||
| } | ||||
| 
 | ||||
| inline Polygons to_polygons(std::vector<Points> &&paths) | ||||
| inline Polygons to_polygons(VecOfPoints &&paths) | ||||
| { | ||||
|     Polygons out; | ||||
|     out.reserve(paths.size()); | ||||
|  | @ -270,6 +297,21 @@ bool polygons_match(const Polygon &l, const Polygon &r); | |||
| Polygon make_circle(double radius, double error); | ||||
| Polygon make_circle_num_segments(double radius, size_t num_segments); | ||||
| 
 | ||||
| /// <summary>
 | ||||
| /// Define point laying on polygon
 | ||||
| /// keep index of polygon line and point coordinate
 | ||||
| /// </summary>
 | ||||
| struct PolygonPoint | ||||
| { | ||||
|     // index of line inside of polygon
 | ||||
|     // 0 .. from point polygon[0] to polygon[1]
 | ||||
|     size_t index; | ||||
| 
 | ||||
|     // Point, which lay on line defined by index
 | ||||
|     Point point; | ||||
| }; | ||||
| using PolygonPoints = std::vector<PolygonPoint>; | ||||
| 
 | ||||
| bool overlaps(const Polygons& polys1, const Polygons& polys2); | ||||
| } // Slic3r
 | ||||
| 
 | ||||
|  |  | |||
|  | @ -497,6 +497,33 @@ BoundingBox get_extents(const Polylines &polylines) | |||
|     return bb; | ||||
| } | ||||
| 
 | ||||
| // Return True when erase some otherwise False.
 | ||||
| bool remove_same_neighbor(Polyline &polyline) { | ||||
|     Points &points = polyline.points; | ||||
|     if (points.empty()) | ||||
|         return false; | ||||
|     auto last = std::unique(points.begin(), points.end()); | ||||
| 
 | ||||
|     // no duplicits
 | ||||
|     if (last == points.end()) | ||||
|         return false; | ||||
| 
 | ||||
|     points.erase(last, points.end()); | ||||
|     return true; | ||||
| } | ||||
| 
 | ||||
| bool remove_same_neighbor(Polylines &polylines){ | ||||
|     if (polylines.empty()) | ||||
|         return false; | ||||
|     bool exist = false; | ||||
|     for (Polyline &polyline : polylines) | ||||
|         exist |= remove_same_neighbor(polyline); | ||||
|     // remove empty polylines
 | ||||
|     polylines.erase(std::remove_if(polylines.begin(), polylines.end(), [](const Polyline &p) { return p.points.size() <= 1; }), polylines.end()); | ||||
|     return exist; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| const Point& leftmost_point(const Polylines &polylines) | ||||
| { | ||||
|     if (polylines.empty()) | ||||
|  |  | |||
|  | @ -164,6 +164,10 @@ public: | |||
| extern BoundingBox get_extents(const Polyline &polyline); | ||||
| extern BoundingBox get_extents(const Polylines &polylines); | ||||
| 
 | ||||
| // Return True when erase some otherwise False.
 | ||||
| bool remove_same_neighbor(Polyline &polyline); | ||||
| bool remove_same_neighbor(Polylines &polylines); | ||||
| 
 | ||||
| inline double total_length(const Polylines &polylines) { | ||||
|     double total = 0; | ||||
|     for (const Polyline &pl : polylines) | ||||
|  |  | |||
|  | @ -839,7 +839,7 @@ void update_volume_bboxes( | |||
|                         layer_range.volumes.emplace_back(*it); | ||||
|                 } else | ||||
|                     layer_range.volumes.push_back({ model_volume->id(), | ||||
|                         transformed_its_bbox2d(model_volume->mesh().its, trafo_for_bbox(object_trafo, model_volume->get_matrix(false)), offset) }); | ||||
|                         transformed_its_bbox2d(model_volume->mesh().its, trafo_for_bbox(object_trafo, model_volume->get_matrix()), offset) }); | ||||
|             } | ||||
|     } else { | ||||
|         std::vector<std::vector<PrintObjectRegions::VolumeExtents>> volumes_old; | ||||
|  | @ -871,7 +871,7 @@ void update_volume_bboxes( | |||
|                             layer_range.volumes.emplace_back(*it); | ||||
|                     } | ||||
|                 } else { | ||||
|                     transformed_its_bboxes_in_z_ranges(model_volume->mesh().its, trafo_for_bbox(object_trafo, model_volume->get_matrix(false)), ranges, bboxes, offset); | ||||
|                     transformed_its_bboxes_in_z_ranges(model_volume->mesh().its, trafo_for_bbox(object_trafo, model_volume->get_matrix()), ranges, bboxes, offset); | ||||
|                     for (PrintObjectRegions::LayerRangeRegions &layer_range : layer_ranges) | ||||
|                         if (auto &bbox = bboxes[&layer_range - layer_ranges.data()]; bbox.second) | ||||
|                             layer_range.volumes.push_back({ model_volume->id(), bbox.first }); | ||||
|  |  | |||
|  | @ -1,9 +1,12 @@ | |||
| ///|/ Copyright (c) Prusa Research 2020 - 2022 Tomáš Mészáros @tamasmeszaros, Vojtěch Bubník @bubnikv
 | ||||
| ///|/
 | ||||
| ///|/ PrusaSlicer is released under the terms of the AGPLv3 or higher
 | ||||
| ///|/
 | ||||
| #ifndef AGGRASTER_HPP | ||||
| #define AGGRASTER_HPP | ||||
| 
 | ||||
| #include <libslic3r/SLA/RasterBase.hpp> | ||||
| #include "libslic3r/ExPolygon.hpp" | ||||
| #include "libslic3r/MTUtils.hpp" | ||||
| 
 | ||||
| // For rasterizing
 | ||||
| #include <agg/agg_basics.h> | ||||
|  | @ -154,8 +157,8 @@ public: | |||
|     } | ||||
|      | ||||
|     Trafo trafo() const override { return m_trafo; } | ||||
|     Resolution resolution() const override { return m_resolution; } | ||||
|     PixelDim   pixel_dimensions() const override | ||||
|     Resolution resolution() const { return m_resolution; } | ||||
|     PixelDim   pixel_dimensions() const | ||||
|     { | ||||
|         return {SCALING_FACTOR / m_pxdim_scaled.w_mm, | ||||
|                 SCALING_FACTOR / m_pxdim_scaled.h_mm}; | ||||
|  | @ -187,11 +190,15 @@ class RasterGrayscaleAA : public _RasterGrayscaleAA { | |||
|     using typename Base::TValue; | ||||
| public: | ||||
|     template<class GammaFn> | ||||
|     RasterGrayscaleAA(const RasterBase::Resolution &res, | ||||
|                       const RasterBase::PixelDim &  pd, | ||||
|                       const RasterBase::Trafo &     trafo, | ||||
|                       GammaFn &&                    fn) | ||||
|         : Base(res, pd, trafo, Colors<TColor>::White, Colors<TColor>::Black, | ||||
|     RasterGrayscaleAA(const Resolution        &res, | ||||
|                       const PixelDim          &pd, | ||||
|                       const RasterBase::Trafo &trafo, | ||||
|                       GammaFn                &&fn) | ||||
|         : Base(res, | ||||
|                pd, | ||||
|                trafo, | ||||
|                Colors<TColor>::White, | ||||
|                Colors<TColor>::Black, | ||||
|                std::forward<GammaFn>(fn)) | ||||
|     {} | ||||
|      | ||||
|  | @ -209,9 +216,9 @@ public: | |||
| 
 | ||||
| class RasterGrayscaleAAGammaPower: public RasterGrayscaleAA { | ||||
| public: | ||||
|     RasterGrayscaleAAGammaPower(const RasterBase::Resolution &res, | ||||
|                                 const RasterBase::PixelDim &  pd, | ||||
|                                 const RasterBase::Trafo &     trafo, | ||||
|     RasterGrayscaleAAGammaPower(const Resolution        &res, | ||||
|                                 const PixelDim          &pd, | ||||
|                                 const RasterBase::Trafo &trafo, | ||||
|                                 double                   gamma = 1.) | ||||
|         : RasterGrayscaleAA(res, pd, trafo, agg::gamma_power(gamma)) | ||||
|     {} | ||||
|  |  | |||
|  | @ -1,3 +1,8 @@ | |||
| ///|/ Copyright (c) Prusa Research 2020 - 2022 Tomáš Mészáros @tamasmeszaros
 | ||||
| ///|/ Copyright (c) 2022 ole00 @ole00
 | ||||
| ///|/
 | ||||
| ///|/ PrusaSlicer is released under the terms of the AGPLv3 or higher
 | ||||
| ///|/
 | ||||
| #ifndef SLARASTER_CPP | ||||
| #define SLARASTER_CPP | ||||
| 
 | ||||
|  | @ -11,11 +16,6 @@ | |||
| 
 | ||||
| namespace Slic3r { namespace sla { | ||||
| 
 | ||||
| const RasterBase::TMirroring RasterBase::NoMirror = {false, false}; | ||||
| const RasterBase::TMirroring RasterBase::MirrorX  = {true, false}; | ||||
| const RasterBase::TMirroring RasterBase::MirrorY  = {false, true}; | ||||
| const RasterBase::TMirroring RasterBase::MirrorXY = {true, true}; | ||||
| 
 | ||||
| EncodedRaster PNGRasterEncoder::operator()(const void *ptr, size_t w, size_t h, | ||||
|                                            size_t      num_components) | ||||
| { | ||||
|  | @ -68,15 +68,17 @@ EncodedRaster PPMRasterEncoder::operator()(const void *ptr, size_t w, size_t h, | |||
| } | ||||
| 
 | ||||
| std::unique_ptr<RasterBase> create_raster_grayscale_aa( | ||||
|     const RasterBase::Resolution &res, | ||||
|     const RasterBase::PixelDim &  pxdim, | ||||
|     const Resolution        &res, | ||||
|     const PixelDim          &pxdim, | ||||
|     double                   gamma, | ||||
|     const RasterBase::Trafo &     tr) | ||||
|     const RasterBase::Trafo &tr) | ||||
| { | ||||
|     std::unique_ptr<RasterBase> rst; | ||||
|      | ||||
|     if (gamma > 0) | ||||
|         rst = std::make_unique<RasterGrayscaleAAGammaPower>(res, pxdim, tr, gamma); | ||||
|     else if (std::abs(gamma - 1.) < 1e-6) | ||||
|         rst = std::make_unique<RasterGrayscaleAA>(res, pxdim, tr, agg::gamma_none()); | ||||
|     else | ||||
|         rst = std::make_unique<RasterGrayscaleAA>(res, pxdim, tr, agg::gamma_threshold(.5)); | ||||
|      | ||||
|  |  | |||
|  | @ -1,3 +1,8 @@ | |||
| ///|/ Copyright (c) Prusa Research 2020 - 2022 Tomáš Mészáros @tamasmeszaros, Vojtěch Bubník @bubnikv
 | ||||
| ///|/ Copyright (c) 2022 ole00 @ole00
 | ||||
| ///|/
 | ||||
| ///|/ PrusaSlicer is released under the terms of the AGPLv3 or higher
 | ||||
| ///|/
 | ||||
| #ifndef SLA_RASTERBASE_HPP | ||||
| #define SLA_RASTERBASE_HPP | ||||
| 
 | ||||
|  | @ -9,7 +14,6 @@ | |||
| #include <cstdint> | ||||
| 
 | ||||
| #include <libslic3r/ExPolygon.hpp> | ||||
| #include <libslic3r/SLA/Concurrency.hpp> | ||||
| 
 | ||||
| namespace Slic3r { | ||||
| 
 | ||||
|  | @ -31,6 +35,27 @@ public: | |||
|     const char * extension() const { return m_ext.c_str(); } | ||||
| }; | ||||
| 
 | ||||
| /// Type that represents a resolution in pixels.
 | ||||
| struct Resolution { | ||||
|     size_t width_px = 0; | ||||
|     size_t height_px = 0; | ||||
| 
 | ||||
|     Resolution() = default; | ||||
|     Resolution(size_t w, size_t h) : width_px(w), height_px(h) {} | ||||
|     size_t pixels() const { return width_px * height_px; } | ||||
| }; | ||||
| 
 | ||||
| /// Types that represents the dimension of a pixel in millimeters.
 | ||||
| struct PixelDim { | ||||
|     double w_mm = 1.; | ||||
|     double h_mm = 1.; | ||||
| 
 | ||||
|     PixelDim() = default; | ||||
|     PixelDim(double px_width_mm, double px_height_mm) | ||||
|         : w_mm(px_width_mm), h_mm(px_height_mm) | ||||
|     {} | ||||
| }; | ||||
| 
 | ||||
| using RasterEncoder = | ||||
|     std::function<EncodedRaster(const void *ptr, size_t w, size_t h, size_t num_components)>; | ||||
| 
 | ||||
|  | @ -40,10 +65,10 @@ public: | |||
|     enum Orientation { roLandscape, roPortrait }; | ||||
|      | ||||
|     using TMirroring = std::array<bool, 2>; | ||||
|     static const TMirroring NoMirror; | ||||
|     static const TMirroring MirrorX; | ||||
|     static const TMirroring MirrorY; | ||||
|     static const TMirroring MirrorXY; | ||||
|     static const constexpr TMirroring NoMirror = {false, false}; | ||||
|     static const constexpr TMirroring MirrorX  = {true, false}; | ||||
|     static const constexpr TMirroring MirrorY  = {false, true}; | ||||
|     static const constexpr TMirroring MirrorXY = {true, true}; | ||||
|      | ||||
|     struct Trafo { | ||||
|         bool mirror_x = false, mirror_y = false, flipXY = false; | ||||
|  | @ -63,35 +88,14 @@ public: | |||
|         Point get_center() const { return {center_x, center_y}; } | ||||
|     }; | ||||
|      | ||||
|     /// Type that represents a resolution in pixels.
 | ||||
|     struct Resolution { | ||||
|         size_t width_px = 0; | ||||
|         size_t height_px = 0; | ||||
|          | ||||
|         Resolution() = default; | ||||
|         Resolution(size_t w, size_t h) : width_px(w), height_px(h) {} | ||||
|         size_t pixels() const { return width_px * height_px; } | ||||
|     }; | ||||
|      | ||||
|     /// Types that represents the dimension of a pixel in millimeters.
 | ||||
|     struct PixelDim { | ||||
|         double w_mm = 1.; | ||||
|         double h_mm = 1.; | ||||
|          | ||||
|         PixelDim() = default; | ||||
|         PixelDim(double px_width_mm, double px_height_mm) | ||||
|             : w_mm(px_width_mm), h_mm(px_height_mm) | ||||
|         {} | ||||
|     }; | ||||
|      | ||||
|     virtual ~RasterBase() = default; | ||||
|      | ||||
|     /// Draw a polygon with holes.
 | ||||
|     virtual void draw(const ExPolygon& poly) = 0; | ||||
|      | ||||
|     /// Get the resolution of the raster.
 | ||||
|     virtual Resolution resolution() const = 0; | ||||
|     virtual PixelDim   pixel_dimensions() const = 0; | ||||
| //    virtual Resolution resolution() const = 0;
 | ||||
| //    virtual PixelDim   pixel_dimensions() const = 0;
 | ||||
|     virtual Trafo      trafo() const = 0; | ||||
|      | ||||
|     virtual EncodedRaster encode(RasterEncoder encoder) const = 0; | ||||
|  | @ -109,10 +113,10 @@ std::ostream& operator<<(std::ostream &stream, const EncodedRaster &bytes); | |||
| 
 | ||||
| // If gamma is zero, thresholding will be performed which disables AA.
 | ||||
| std::unique_ptr<RasterBase> create_raster_grayscale_aa( | ||||
|     const RasterBase::Resolution &res, | ||||
|     const RasterBase::PixelDim &  pxdim, | ||||
|     const Resolution        &res, | ||||
|     const PixelDim          &pxdim, | ||||
|     double                   gamma = 1.0, | ||||
|     const RasterBase::Trafo &     tr    = {}); | ||||
|     const RasterBase::Trafo &tr    = {}); | ||||
| 
 | ||||
| }} // namespace Slic3r::sla
 | ||||
| 
 | ||||
|  |  | |||
|  | @ -6,6 +6,7 @@ | |||
| #include "PrintBase.hpp" | ||||
| #include "SLA/RasterBase.hpp" | ||||
| #include "SLA/SupportTree.hpp" | ||||
| #include "Execution/ExecutionTBB.hpp" | ||||
| #include "Point.hpp" | ||||
| #include "MTUtils.hpp" | ||||
| #include "Zipper.hpp" | ||||
|  |  | |||
							
								
								
									
										191
									
								
								src/libslic3r/TextConfiguration.hpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,191 @@ | |||
| ///|/ Copyright (c) Prusa Research 2021 - 2022 Filip Sykala @Jony01, Vojtěch Bubník @bubnikv
 | ||||
| ///|/
 | ||||
| ///|/ PrusaSlicer is released under the terms of the AGPLv3 or higher
 | ||||
| ///|/
 | ||||
| #ifndef slic3r_TextConfiguration_hpp_ | ||||
| #define slic3r_TextConfiguration_hpp_ | ||||
| 
 | ||||
| #include <vector> | ||||
| #include <string> | ||||
| #include <optional> | ||||
| #include <cereal/cereal.hpp> | ||||
| #include <cereal/types/optional.hpp> | ||||
| #include <cereal/types/string.hpp> | ||||
| #include <cereal/archives/binary.hpp> | ||||
| #include "Point.hpp" // Transform3d
 | ||||
| 
 | ||||
| namespace Slic3r { | ||||
| 
 | ||||
| /// <summary>
 | ||||
| /// User modifiable property of text style
 | ||||
| /// NOTE: OnEdit fix serializations: EmbossStylesSerializable, TextConfigurationSerialization
 | ||||
| /// </summary>
 | ||||
| struct FontProp | ||||
| { | ||||
|     // define extra space between letters, negative mean closer letter
 | ||||
|     // When not set value is zero and is not stored
 | ||||
|     std::optional<int> char_gap; // [in font point]
 | ||||
| 
 | ||||
|     // define extra space between lines, negative mean closer lines
 | ||||
|     // When not set value is zero and is not stored
 | ||||
|     std::optional<int> line_gap; // [in font point]
 | ||||
| 
 | ||||
|     // positive value mean wider character shape
 | ||||
|     // negative value mean tiner character shape
 | ||||
|     // When not set value is zero and is not stored
 | ||||
|     std::optional<float> boldness; // [in mm]
 | ||||
| 
 | ||||
|     // positive value mean italic of character (CW)
 | ||||
|     // negative value mean CCW skew (unItalic)
 | ||||
|     // When not set value is zero and is not stored
 | ||||
|     std::optional<float> skew; // [ration x:y]
 | ||||
| 
 | ||||
|     // Parameter for True Type Font collections
 | ||||
|     // Select index of font in collection
 | ||||
|     std::optional<unsigned int> collection_number; | ||||
| 
 | ||||
|     // Distiguish projection per glyph
 | ||||
|     bool per_glyph; | ||||
| 
 | ||||
|     // NOTE: way of serialize to 3mf force that zero must be default value
 | ||||
|     enum class HorizontalAlign { left = 0, center, right }; | ||||
|     enum class VerticalAlign { top = 0, center, bottom }; | ||||
|     using Align = std::pair<HorizontalAlign, VerticalAlign>; | ||||
|     // change pivot of text
 | ||||
|     // When not set, center is used and is not stored
 | ||||
|     Align align = Align(HorizontalAlign::center, VerticalAlign::center); | ||||
| 
 | ||||
|     //////
 | ||||
|     // Duplicit data to wxFontDescriptor
 | ||||
|     // used for store/load .3mf file
 | ||||
|     //////
 | ||||
| 
 | ||||
|     // Height of text line (letters)
 | ||||
|     // duplicit to wxFont::PointSize
 | ||||
|     float size_in_mm; // [in mm]
 | ||||
| 
 | ||||
|     // Additional data about font to be able to find substitution,
 | ||||
|     // when same font is not installed
 | ||||
|     std::optional<std::string> family; | ||||
|     std::optional<std::string> face_name; | ||||
|     std::optional<std::string> style; | ||||
|     std::optional<std::string> weight; | ||||
| 
 | ||||
|     /// <summary>
 | ||||
|     /// Only constructor with restricted values
 | ||||
|     /// </summary>
 | ||||
|     /// <param name="line_height">Y size of text [in mm]</param>
 | ||||
|     /// <param name="depth">Z size of text [in mm]</param>
 | ||||
|     FontProp(float line_height = 10.f) : size_in_mm(line_height), per_glyph(false) | ||||
|     {} | ||||
| 
 | ||||
|     bool operator==(const FontProp& other) const { | ||||
|         return  | ||||
|             char_gap == other.char_gap &&  | ||||
|             line_gap == other.line_gap && | ||||
|             per_glyph == other.per_glyph && | ||||
|             align == other.align && | ||||
|             is_approx(size_in_mm, other.size_in_mm) &&  | ||||
|             is_approx(boldness, other.boldness) && | ||||
|             is_approx(skew, other.skew); | ||||
|     } | ||||
| 
 | ||||
|     // undo / redo stack recovery
 | ||||
|     template<class Archive> void save(Archive &ar) const | ||||
|     { | ||||
|         ar(size_in_mm, per_glyph, align.first, align.second); | ||||
|         cereal::save(ar, char_gap); | ||||
|         cereal::save(ar, line_gap); | ||||
|         cereal::save(ar, boldness); | ||||
|         cereal::save(ar, skew); | ||||
|         cereal::save(ar, collection_number); | ||||
|     } | ||||
|     template<class Archive> void load(Archive &ar) | ||||
|     { | ||||
|         ar(size_in_mm, per_glyph, align.first, align.second); | ||||
|         cereal::load(ar, char_gap); | ||||
|         cereal::load(ar, line_gap); | ||||
|         cereal::load(ar, boldness); | ||||
|         cereal::load(ar, skew); | ||||
|         cereal::load(ar, collection_number); | ||||
|     } | ||||
| }; | ||||
| 
 | ||||
| /// <summary>
 | ||||
| /// Style of embossed text
 | ||||
| /// (Path + Type) must define how to open font for using on different OS
 | ||||
| /// NOTE: OnEdit fix serializations: EmbossStylesSerializable, TextConfigurationSerialization
 | ||||
| /// </summary>
 | ||||
| struct EmbossStyle | ||||
| { | ||||
|     // Human readable name of style it is shown in GUI
 | ||||
|     std::string name; | ||||
| 
 | ||||
|     // Define how to open font
 | ||||
|     // Meaning depend on type
 | ||||
|     std::string path; | ||||
| 
 | ||||
|     enum class Type; | ||||
|     // Define what is stored in path
 | ||||
|     Type type { Type::undefined }; | ||||
| 
 | ||||
|     // User modification of font style
 | ||||
|     FontProp prop; | ||||
| 
 | ||||
|     // when name is empty than Font item was loaded from .3mf file 
 | ||||
|     // and potentionaly it is not reproducable
 | ||||
|     // define data stored in path
 | ||||
|     // when wx change way of storing add new descriptor Type
 | ||||
|     enum class Type {  | ||||
|         undefined = 0, | ||||
| 
 | ||||
|         // wx font descriptors are platform dependent
 | ||||
|         // path is font descriptor generated by wxWidgets
 | ||||
|         wx_win_font_descr, // on Windows 
 | ||||
|         wx_lin_font_descr, // on Linux
 | ||||
|         wx_mac_font_descr, // on Max OS
 | ||||
| 
 | ||||
|         // TrueTypeFont file loacation on computer
 | ||||
|         // for privacy: only filename is stored into .3mf
 | ||||
|         file_path | ||||
|     }; | ||||
| 
 | ||||
|     bool operator==(const EmbossStyle &other) const | ||||
|     { | ||||
|         return  | ||||
|             type == other.type && | ||||
|             prop == other.prop && | ||||
|             name == other.name && | ||||
|             path == other.path | ||||
|             ; | ||||
|     } | ||||
| 
 | ||||
|     // undo / redo stack recovery
 | ||||
|     template<class Archive> void serialize(Archive &ar){ ar(name, path, type, prop); } | ||||
| }; | ||||
| 
 | ||||
| // Emboss style name inside vector is unique
 | ||||
| // It is not map beacuse items has own order (view inside of slect)
 | ||||
| // It is stored into AppConfig by EmbossStylesSerializable
 | ||||
| using EmbossStyles = std::vector<EmbossStyle>; | ||||
| 
 | ||||
| /// <summary>
 | ||||
| /// Define how to create 'Text volume'
 | ||||
| /// It is stored into .3mf by TextConfigurationSerialization
 | ||||
| /// It is part of ModelVolume optional data
 | ||||
| /// </summary>
 | ||||
| struct TextConfiguration | ||||
| { | ||||
|     // Style of embossed text
 | ||||
|     EmbossStyle style; | ||||
| 
 | ||||
|     // Embossed text value
 | ||||
|     std::string text = "None"; | ||||
| 
 | ||||
|     // undo / redo stack recovery
 | ||||
|     template<class Archive> void serialize(Archive &ar) { ar(style, text); } | ||||
| };     | ||||
| 
 | ||||
| } // namespace Slic3r
 | ||||
| 
 | ||||
| #endif // slic3r_TextConfiguration_hpp_
 | ||||
							
								
								
									
										25
									
								
								src/libslic3r/Timer.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,25 @@ | |||
| ///|/ Copyright (c) Prusa Research 2023 Vojtěch Bubník @bubnikv
 | ||||
| ///|/
 | ||||
| ///|/ PrusaSlicer is released under the terms of the AGPLv3 or higher
 | ||||
| ///|/
 | ||||
| #include "Timer.hpp" | ||||
| #include <boost/log/trivial.hpp> | ||||
| 
 | ||||
| using namespace std::chrono; | ||||
| 
 | ||||
| Slic3r::Timer::Timer(const std::string &name) : m_name(name), m_start(steady_clock::now()) {} | ||||
| 
 | ||||
| Slic3r::Timer::~Timer() | ||||
| { | ||||
|     BOOST_LOG_TRIVIAL(debug) << "Timer '" << m_name << "' spend " <<  | ||||
|         duration_cast<milliseconds>(steady_clock::now() - m_start).count() << "ms"; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| namespace Slic3r::Timing { | ||||
| 
 | ||||
| void TimeLimitAlarm::report_time_exceeded() const { | ||||
|     BOOST_LOG_TRIVIAL(error) << "Time limit exceeded for " << m_limit_exceeded_message << ": " << m_timer.elapsed_seconds() << "s"; | ||||
| } | ||||
| 
 | ||||
| } | ||||
							
								
								
									
										96
									
								
								src/libslic3r/Timer.hpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,96 @@ | |||
| ///|/ Copyright (c) Prusa Research 2023 Vojtěch Bubník @bubnikv
 | ||||
| ///|/
 | ||||
| ///|/ PrusaSlicer is released under the terms of the AGPLv3 or higher
 | ||||
| ///|/
 | ||||
| #ifndef libslic3r_Timer_hpp_ | ||||
| #define libslic3r_Timer_hpp_ | ||||
| 
 | ||||
| #include <string> | ||||
| #include <chrono> | ||||
| 
 | ||||
| namespace Slic3r { | ||||
| 
 | ||||
| /// <summary>
 | ||||
| /// Instance of this class is used for measure time consumtion
 | ||||
| /// of block code until instance is alive and write result to debug output
 | ||||
| /// </summary>
 | ||||
| class Timer | ||||
| { | ||||
|     std::string m_name; | ||||
|     std::chrono::steady_clock::time_point m_start; | ||||
| public: | ||||
|     /// <summary>
 | ||||
|     /// name describe timer
 | ||||
|     /// </summary>
 | ||||
|     /// <param name="name">Describe timer in consol log</param>
 | ||||
|     Timer(const std::string& name); | ||||
| 
 | ||||
|     /// <summary>
 | ||||
|     /// name describe timer
 | ||||
|     /// </summary>
 | ||||
|     ~Timer(); | ||||
| }; | ||||
| 
 | ||||
| namespace Timing { | ||||
| 
 | ||||
|     // Timing code from Catch2 unit testing library
 | ||||
|     static inline uint64_t nanoseconds_since_epoch() { | ||||
|         return std::chrono::duration_cast<std::chrono::nanoseconds>(std::chrono::high_resolution_clock::now().time_since_epoch()).count(); | ||||
|     } | ||||
| 
 | ||||
|     // Timing code from Catch2 unit testing library
 | ||||
|     class Timer { | ||||
|     public: | ||||
|         void start() { | ||||
|             m_nanoseconds = nanoseconds_since_epoch(); | ||||
|         } | ||||
|         uint64_t elapsed_nanoseconds() const { | ||||
|             return nanoseconds_since_epoch() - m_nanoseconds; | ||||
|         } | ||||
|         uint64_t elapsed_microseconds() const { | ||||
|             return elapsed_nanoseconds() / 1000; | ||||
|         } | ||||
|         unsigned int elapsed_milliseconds() const { | ||||
|             return static_cast<unsigned int>(elapsed_microseconds()/1000); | ||||
|         } | ||||
|         double elapsed_seconds() const { | ||||
|             return elapsed_microseconds() / 1000000.0; | ||||
|         } | ||||
|     private: | ||||
|         uint64_t m_nanoseconds = 0; | ||||
|     }; | ||||
| 
 | ||||
|     // Emits a Boost::log error if the life time of this timing object exceeds a limit.
 | ||||
|     class TimeLimitAlarm { | ||||
|     public: | ||||
|         TimeLimitAlarm(uint64_t time_limit_nanoseconds, std::string_view limit_exceeded_message) : | ||||
|             m_time_limit_nanoseconds(time_limit_nanoseconds), m_limit_exceeded_message(limit_exceeded_message) {  | ||||
|             m_timer.start(); | ||||
|         } | ||||
|         ~TimeLimitAlarm() { | ||||
|             auto elapsed = m_timer.elapsed_nanoseconds(); | ||||
|             if (elapsed > m_time_limit_nanoseconds) | ||||
|                 this->report_time_exceeded(); | ||||
|         } | ||||
|         static TimeLimitAlarm new_nanos(uint64_t time_limit_nanoseconds, std::string_view limit_exceeded_message) { | ||||
|             return TimeLimitAlarm(time_limit_nanoseconds, limit_exceeded_message); | ||||
|         } | ||||
|         static TimeLimitAlarm new_milis(uint64_t time_limit_milis, std::string_view limit_exceeded_message) { | ||||
|             return TimeLimitAlarm(uint64_t(time_limit_milis) * 1000000l, limit_exceeded_message); | ||||
|         } | ||||
|         static TimeLimitAlarm new_seconds(uint64_t time_limit_seconds, std::string_view limit_exceeded_message) { | ||||
|             return TimeLimitAlarm(uint64_t(time_limit_seconds) * 1000000000l, limit_exceeded_message); | ||||
|         } | ||||
|     private: | ||||
|         void report_time_exceeded() const; | ||||
| 
 | ||||
|         Timer               m_timer; | ||||
|         uint64_t            m_time_limit_nanoseconds; | ||||
|         std::string_view    m_limit_exceeded_message; | ||||
|     }; | ||||
| 
 | ||||
| } // namespace Catch
 | ||||
| 
 | ||||
| } // namespace Slic3r
 | ||||
| 
 | ||||
| #endif // libslic3r_Timer_hpp_
 | ||||
							
								
								
									
										332
									
								
								src/libslic3r/Triangulation.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,332 @@ | |||
| ///|/ Copyright (c) Prusa Research 2021 - 2022 Filip Sykala @Jony01, Vojtěch Bubník @bubnikv
 | ||||
| ///|/
 | ||||
| ///|/ PrusaSlicer is released under the terms of the AGPLv3 or higher
 | ||||
| ///|/
 | ||||
| #include "Triangulation.hpp" | ||||
| #include "IntersectionPoints.hpp" | ||||
| #include <CGAL/Exact_predicates_inexact_constructions_kernel.h> | ||||
| #include <CGAL/Constrained_Delaunay_triangulation_2.h> | ||||
| #include <CGAL/Triangulation_vertex_base_with_info_2.h> | ||||
| #include <CGAL/spatial_sort.h> | ||||
| 
 | ||||
| using namespace Slic3r; | ||||
| namespace priv{ | ||||
| inline void insert_edges(Triangulation::HalfEdges &edges, uint32_t &offset, const Polygon &polygon, const Triangulation::Changes& changes) { | ||||
|     const Points &pts = polygon.points; | ||||
|     uint32_t size = static_cast<uint32_t>(pts.size()); | ||||
|     uint32_t last_index = offset + size - 1; | ||||
|     uint32_t prev_index = changes[last_index]; | ||||
|     for (uint32_t i = 0; i < size; ++i) { | ||||
|         uint32_t index = changes[offset + i]; | ||||
|         // when duplicit points are neighbor
 | ||||
|         if (prev_index == index) continue;  | ||||
|         edges.push_back({prev_index, index}); | ||||
|         prev_index = index; | ||||
|     } | ||||
|     offset += size; | ||||
| } | ||||
| 
 | ||||
| inline void insert_edges(Triangulation::HalfEdges &edges, uint32_t &offset, const Polygon &polygon) { | ||||
|     const Points &pts = polygon.points; | ||||
|     uint32_t size = static_cast<uint32_t>(pts.size()); | ||||
|     uint32_t prev_index = offset + size - 1; | ||||
|     for (uint32_t i = 0; i < size; ++i) { | ||||
|         uint32_t index = offset + i; | ||||
|         edges.push_back({prev_index, index}); | ||||
|         prev_index = index; | ||||
|     } | ||||
|     offset += size; | ||||
| } | ||||
| 
 | ||||
| inline bool has_bidirectional_constrained( | ||||
|     const Triangulation::HalfEdges &constrained) | ||||
| { | ||||
|     for (const auto &c : constrained) { | ||||
|         auto key = std::make_pair(c.second, c.first); | ||||
|         auto it  = std::lower_bound(constrained.begin(), constrained.end(), | ||||
|                                     key); | ||||
|         if (it != constrained.end() && *it == key) return true; | ||||
|     } | ||||
|     return false; | ||||
| } | ||||
| 
 | ||||
| inline bool is_unique(const Points &points) {  | ||||
|     Points pts = points; // copy
 | ||||
|     std::sort(pts.begin(), pts.end()); | ||||
|     auto it = std::adjacent_find(pts.begin(), pts.end()); | ||||
|     return it == pts.end(); | ||||
| } | ||||
| 
 | ||||
| inline bool has_self_intersection( | ||||
|     const Points                   &points, | ||||
|     const Triangulation::HalfEdges &constrained_half_edges) | ||||
| { | ||||
|     Lines lines; | ||||
|     lines.reserve(constrained_half_edges.size()); | ||||
|     for (const auto &he : constrained_half_edges) | ||||
|         lines.emplace_back(points[he.first], points[he.second]); | ||||
|     return !get_intersections(lines).empty(); | ||||
| } | ||||
| 
 | ||||
| } // namespace priv
 | ||||
| 
 | ||||
| //#define VISUALIZE_TRIANGULATION
 | ||||
| #ifdef VISUALIZE_TRIANGULATION | ||||
| #include "admesh/stl.h" // indexed triangle set
 | ||||
| static void visualize(const Points                 &points, | ||||
|                const Triangulation::Indices &indices, | ||||
|                const char                   *filename) | ||||
| { | ||||
|     // visualize
 | ||||
|     indexed_triangle_set its; | ||||
|     its.vertices.reserve(points.size()); | ||||
|     for (const Point &p : points) its.vertices.emplace_back(p.x(), p.y(), 0.); | ||||
|     its.indices = indices; | ||||
|     its_write_obj(its, filename); | ||||
| } | ||||
| #endif // VISUALIZE_TRIANGULATION
 | ||||
| 
 | ||||
| Triangulation::Indices Triangulation::triangulate(const Points    &points, | ||||
|                                                   const HalfEdges &constrained_half_edges) | ||||
| { | ||||
|     assert(!points.empty()); | ||||
|     assert(!constrained_half_edges.empty()); | ||||
|     // constrained must be sorted
 | ||||
|     assert(std::is_sorted(constrained_half_edges.begin(), | ||||
|                           constrained_half_edges.end())); | ||||
|     // check that there is no duplicit constrained edge
 | ||||
|     assert(std::adjacent_find(constrained_half_edges.begin(), constrained_half_edges.end()) == constrained_half_edges.end()); | ||||
|     // edges can NOT contain bidirectional constrained
 | ||||
|     assert(!priv::has_bidirectional_constrained(constrained_half_edges)); | ||||
|     // check that there is only unique poistion of points
 | ||||
|     assert(priv::is_unique(points)); | ||||
|     assert(!priv::has_self_intersection(points, constrained_half_edges)); | ||||
|     // use cgal triangulation
 | ||||
|     using K   = CGAL::Exact_predicates_inexact_constructions_kernel; | ||||
|     using Vb  = CGAL::Triangulation_vertex_base_with_info_2<uint32_t, K>; | ||||
|     using Fb  = CGAL::Constrained_triangulation_face_base_2<K>; | ||||
|     using Tds = CGAL::Triangulation_data_structure_2<Vb, Fb>; | ||||
|     using CDT = CGAL::Constrained_Delaunay_triangulation_2<K, Tds, CGAL::Exact_predicates_tag>; | ||||
| 
 | ||||
|     // construct a constrained triangulation
 | ||||
|     CDT cdt; | ||||
|     { | ||||
|         std::vector<CDT::Vertex_handle> vertices_handle(points.size()); // for constriants
 | ||||
|         using Point_with_ord = std::pair<CDT::Point, size_t>; | ||||
|         using SearchTrait    = CGAL::Spatial_sort_traits_adapter_2 | ||||
|             <K, CGAL::First_of_pair_property_map<Point_with_ord> >; | ||||
| 
 | ||||
|         std::vector<Point_with_ord> cdt_points; | ||||
|         cdt_points.reserve(points.size()); | ||||
|         size_t ord = 0; | ||||
|         for (const auto &p : points) | ||||
|             cdt_points.emplace_back(std::make_pair(CDT::Point{p.x(), p.y()}, ord++)); | ||||
|          | ||||
|         SearchTrait st; | ||||
|         CGAL::spatial_sort(cdt_points.begin(), cdt_points.end(), st); | ||||
|         CDT::Face_handle f; | ||||
|         for (const auto& p : cdt_points) { | ||||
|             auto handle = cdt.insert(p.first, f); | ||||
|             handle->info() = p.second; | ||||
|             vertices_handle[p.second] = handle; | ||||
|             f = handle->face(); | ||||
|         } | ||||
| 
 | ||||
|         // Constrain the triangulation.
 | ||||
|         for (const HalfEdge &edge : constrained_half_edges) | ||||
|             cdt.insert_constraint(vertices_handle[edge.first], vertices_handle[edge.second]); | ||||
|     } | ||||
| 
 | ||||
|     auto faces = cdt.finite_face_handles(); | ||||
| 
 | ||||
|     // Unmark constrained edges of outside faces.
 | ||||
|     size_t num_faces = 0; | ||||
|     for (CDT::Face_handle fh : faces) { | ||||
|         for (int i = 0; i < 3; ++i) { | ||||
|             if (!fh->is_constrained(i)) continue; | ||||
|             auto key = std::make_pair(fh->vertex((i + 2) % 3)->info(), fh->vertex((i + 1) % 3)->info()); | ||||
|             auto it = std::lower_bound(constrained_half_edges.begin(), constrained_half_edges.end(), key); | ||||
|             if (it == constrained_half_edges.end() || *it != key) continue; | ||||
|             // This face contains a constrained edge and it is outside.
 | ||||
|             for (int j = 0; j < 3; ++ j) | ||||
|                 fh->set_constraint(j, false); | ||||
|             --num_faces; | ||||
|             break;             | ||||
|         } | ||||
|         ++num_faces; | ||||
|     } | ||||
| 
 | ||||
|     auto inside = [](CDT::Face_handle &fh) {  | ||||
|         return fh->neighbor(0) != fh &&  | ||||
|                (fh->is_constrained(0) || | ||||
|                 fh->is_constrained(1) || | ||||
|                 fh->is_constrained(2));  | ||||
|     }; | ||||
| 
 | ||||
| #ifdef VISUALIZE_TRIANGULATION | ||||
|     std::vector<Vec3i> indices2; | ||||
|     indices2.reserve(num_faces); | ||||
|     for (CDT::Face_handle fh : faces) | ||||
|         if (inside(fh)) indices2.emplace_back(fh->vertex(0)->info(), fh->vertex(1)->info(), fh->vertex(2)->info()); | ||||
|     visualize(points, indices2, "C:/data/temp/triangulation_without_floodfill.obj"); | ||||
| #endif // VISUALIZE_TRIANGULATION
 | ||||
| 
 | ||||
|     // Propagate inside the constrained regions.
 | ||||
|     std::vector<CDT::Face_handle> queue; | ||||
|     queue.reserve(num_faces); | ||||
|     for (CDT::Face_handle seed : faces){ | ||||
|         if (!inside(seed)) continue;     | ||||
|         // Seed fill to neighbor faces.
 | ||||
|         queue.emplace_back(seed); | ||||
|         while (! queue.empty()) { | ||||
|             CDT::Face_handle fh = queue.back(); | ||||
|             queue.pop_back(); | ||||
|             for (int i = 0; i < 3; ++i) { | ||||
|                 if (fh->is_constrained(i)) continue; | ||||
|                 // Propagate along this edge.
 | ||||
|                 fh->set_constraint(i, true); | ||||
|                 CDT::Face_handle nh = fh->neighbor(i); | ||||
|                 bool was_inside = inside(nh); | ||||
|                 // Mark the other side of this edge.
 | ||||
|                 nh->set_constraint(nh->index(fh), true); | ||||
|                 if (! was_inside) | ||||
|                     queue.push_back(nh); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     std::vector<Vec3i> indices; | ||||
|     indices.reserve(num_faces); | ||||
|     for (CDT::Face_handle fh : faces) | ||||
|         if (inside(fh)) | ||||
|             indices.emplace_back(fh->vertex(0)->info(), fh->vertex(1)->info(), fh->vertex(2)->info()); | ||||
| 
 | ||||
| #ifdef VISUALIZE_TRIANGULATION | ||||
|     visualize(points, indices, "C:/data/temp/triangulation.obj"); | ||||
| #endif // VISUALIZE_TRIANGULATION
 | ||||
| 
 | ||||
|     return indices; | ||||
| } | ||||
| 
 | ||||
| Triangulation::Indices Triangulation::triangulate(const Polygon &polygon) | ||||
| { | ||||
|     const Points &pts = polygon.points; | ||||
|     HalfEdges edges; | ||||
|     edges.reserve(pts.size()); | ||||
|     uint32_t offset = 0; | ||||
|     priv::insert_edges(edges, offset, polygon); | ||||
|     std::sort(edges.begin(), edges.end()); | ||||
|     return triangulate(pts, edges); | ||||
| } | ||||
| 
 | ||||
| Triangulation::Indices Triangulation::triangulate(const Polygons &polygons) | ||||
| { | ||||
|     size_t count = count_points(polygons); | ||||
|     Points points; | ||||
|     points.reserve(count); | ||||
| 
 | ||||
|     HalfEdges edges; | ||||
|     edges.reserve(count); | ||||
|     uint32_t  offset = 0; | ||||
| 
 | ||||
|     for (const Polygon &polygon : polygons) { | ||||
|         Slic3r::append(points, polygon.points); | ||||
|         priv::insert_edges(edges, offset, polygon); | ||||
|     } | ||||
| 
 | ||||
|     std::sort(edges.begin(), edges.end()); | ||||
|     return triangulate(points, edges); | ||||
| } | ||||
| 
 | ||||
| Triangulation::Indices Triangulation::triangulate(const ExPolygon &expolygon){ | ||||
|     ExPolygons expolys({expolygon});  | ||||
|     return triangulate(expolys); | ||||
| } | ||||
| 
 | ||||
| Triangulation::Indices Triangulation::triangulate(const ExPolygons &expolygons){ | ||||
|     Points pts = to_points(expolygons); | ||||
|     Points d_pts = collect_duplicates(pts); | ||||
|     if (d_pts.empty()) return triangulate(expolygons, pts); | ||||
| 
 | ||||
|     Changes changes = create_changes(pts, d_pts); | ||||
|     Indices indices = triangulate(expolygons, pts, changes); | ||||
|     // reverse map for changes
 | ||||
|     Changes changes2(changes.size(), std::numeric_limits<uint32_t>::max()); | ||||
|     for (size_t i = 0; i < changes.size(); ++i) | ||||
|         changes2[changes[i]] = i; | ||||
| 
 | ||||
|     // convert indices into expolygons indicies
 | ||||
|     for (Vec3i &t : indices)  | ||||
|         for (size_t ti = 0; ti < 3; ti++) t[ti] = changes2[t[ti]]; | ||||
|      | ||||
|     return indices; | ||||
| } | ||||
| 
 | ||||
| Triangulation::Indices Triangulation::triangulate(const ExPolygons &expolygons, const Points &points) | ||||
| { | ||||
|     assert(count_points(expolygons) == points.size()); | ||||
|     // when contain duplicit coordinate in points will not work properly
 | ||||
|     assert(collect_duplicates(points).empty()); | ||||
| 
 | ||||
|     HalfEdges edges; | ||||
|     edges.reserve(points.size()); | ||||
|     uint32_t offset = 0; | ||||
|     for (const ExPolygon &expolygon : expolygons) { | ||||
|         priv::insert_edges(edges, offset, expolygon.contour); | ||||
|         for (const Polygon &hole : expolygon.holes) | ||||
|             priv::insert_edges(edges, offset, hole); | ||||
|     } | ||||
|     std::sort(edges.begin(), edges.end()); | ||||
|     return triangulate(points, edges); | ||||
| } | ||||
| 
 | ||||
| Triangulation::Indices Triangulation::triangulate(const ExPolygons &expolygons, const Points& points, const Changes& changes) | ||||
| { | ||||
|     assert(!points.empty()); | ||||
|     assert(count_points(expolygons) == points.size()); | ||||
|     assert(changes.size() == points.size()); | ||||
|     // IMPROVE: search from end and somehow distiquish that value is not a change
 | ||||
|     uint32_t count_points = *std::max_element(changes.begin(), changes.end())+1; | ||||
|     Points pts(count_points); | ||||
|     for (size_t i = 0; i < changes.size(); i++) | ||||
|         pts[changes[i]] = points[i];     | ||||
| 
 | ||||
|     HalfEdges edges; | ||||
|     edges.reserve(points.size()); | ||||
|     uint32_t offset = 0; | ||||
|     for (const ExPolygon &expolygon : expolygons) { | ||||
|         priv::insert_edges(edges, offset, expolygon.contour, changes); | ||||
|         for (const Polygon &hole : expolygon.holes) | ||||
|             priv::insert_edges(edges, offset, hole, changes); | ||||
|     } | ||||
| 
 | ||||
|     std::sort(edges.begin(), edges.end()); | ||||
|     return triangulate(pts, edges); | ||||
| } | ||||
| 
 | ||||
| Triangulation::Changes Triangulation::create_changes(const Points &points, const Points &duplicits) | ||||
| { | ||||
|     assert(!duplicits.empty()); | ||||
|     assert(duplicits.size() < points.size()/2); | ||||
|     std::vector<uint32_t> duplicit_indices(duplicits.size(), std::numeric_limits<uint32_t>::max()); | ||||
|     Changes changes;  | ||||
|     changes.reserve(points.size()); | ||||
|     uint32_t index = 0; | ||||
|     for (const Point &p: points) { | ||||
|         auto it = std::lower_bound(duplicits.begin(), duplicits.end(), p); | ||||
|         if (it == duplicits.end() || *it != p) {  | ||||
|             changes.push_back(index); | ||||
|             ++index; | ||||
|             continue; | ||||
|         } | ||||
|         uint32_t &d_index = duplicit_indices[it - duplicits.begin()]; | ||||
|         if (d_index == std::numeric_limits<uint32_t>::max()) { | ||||
|             d_index = index; | ||||
|             changes.push_back(index); | ||||
|             ++index; | ||||
|         } else { | ||||
|             changes.push_back(d_index); | ||||
|         } | ||||
|     } | ||||
|     return changes; | ||||
| } | ||||
							
								
								
									
										76
									
								
								src/libslic3r/Triangulation.hpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,76 @@ | |||
| ///|/ Copyright (c) Prusa Research 2021 - 2022 Vojtěch Bubník @bubnikv, Filip Sykala @Jony01
 | ||||
| ///|/
 | ||||
| ///|/ PrusaSlicer is released under the terms of the AGPLv3 or higher
 | ||||
| ///|/
 | ||||
| #ifndef libslic3r_Triangulation_hpp_ | ||||
| #define libslic3r_Triangulation_hpp_ | ||||
| 
 | ||||
| #include <vector> | ||||
| #include <set> | ||||
| #include <libslic3r/Point.hpp> | ||||
| #include <libslic3r/Polygon.hpp> | ||||
| #include <libslic3r/ExPolygon.hpp> | ||||
| 
 | ||||
| namespace Slic3r { | ||||
| 
 | ||||
| class Triangulation | ||||
| { | ||||
| public: | ||||
|     Triangulation() = delete; | ||||
| 
 | ||||
|     // define oriented connection of 2 vertices(defined by its index)
 | ||||
|     using HalfEdge  = std::pair<uint32_t, uint32_t>; | ||||
|     using HalfEdges = std::vector<HalfEdge>; | ||||
|     using Indices   = std::vector<Vec3i>; | ||||
| 
 | ||||
|     /// <summary>
 | ||||
|     /// Connect points by triangulation to create filled surface by triangles
 | ||||
|     /// Input points have to be unique
 | ||||
|     /// Inspiration for make unique points is Emboss::dilate_to_unique_points
 | ||||
|     /// </summary>
 | ||||
|     /// <param name="points">Points to connect</param>
 | ||||
|     /// <param name="edges">Constraint for edges, pair is from point(first) to
 | ||||
|     /// point(second), sorted lexicographically</param> 
 | ||||
|     /// <returns>Triangles</returns>
 | ||||
|     static Indices triangulate(const Points &points, | ||||
|                                const HalfEdges &half_edges); | ||||
|     static Indices triangulate(const Polygon &polygon); | ||||
|     static Indices triangulate(const Polygons &polygons); | ||||
|     static Indices triangulate(const ExPolygon &expolygon); | ||||
|     static Indices triangulate(const ExPolygons &expolygons); | ||||
| 
 | ||||
|     // Map for convert original index to set without duplication
 | ||||
|     //              from_index<to_index>
 | ||||
|     using Changes = std::vector<uint32_t>; | ||||
| 
 | ||||
|     /// <summary>
 | ||||
|     /// Create conversion map from original index into new 
 | ||||
|     /// with respect of duplicit point
 | ||||
|     /// </summary>
 | ||||
|     /// <param name="points">input set of points</param>
 | ||||
|     /// <param name="duplicits">duplicit points collected from points</param>
 | ||||
|     /// <returns>Conversion map for point index</returns>
 | ||||
|     static Changes create_changes(const Points &points, const Points &duplicits); | ||||
| 
 | ||||
|     /// <summary>
 | ||||
|     /// Triangulation for expolygons, speed up when points are already collected
 | ||||
|     /// NOTE: Not working properly for ExPolygons with multiple point on same coordinate
 | ||||
|     /// You should check it by "collect_changes"
 | ||||
|     /// </summary>
 | ||||
|     /// <param name="expolygons">Input shape to triangulation - define edges</param>
 | ||||
|     /// <param name="points">Points from expolygons</param>
 | ||||
|     /// <returns>Triangle indices</returns>
 | ||||
|     static Indices triangulate(const ExPolygons &expolygons, const Points& points); | ||||
| 
 | ||||
|     /// <summary>
 | ||||
|     /// Triangulation for expolygons containing multiple points with same coordinate
 | ||||
|     /// </summary>
 | ||||
|     /// <param name="expolygons">Input shape to triangulation - define edge</param>
 | ||||
|     /// <param name="points">Points from expolygons</param>
 | ||||
|     /// <param name="changes">Changes swap for indicies into points</param>
 | ||||
|     /// <returns>Triangle indices</returns>
 | ||||
|     static Indices triangulate(const ExPolygons &expolygons, const Points& points, const Changes& changes); | ||||
| }; | ||||
| 
 | ||||
| } // namespace Slic3r
 | ||||
| #endif // libslic3r_Triangulation_hpp_
 | ||||
|  | @ -372,6 +372,7 @@ inline typename CONTAINER_TYPE::value_type& next_value_modulo(typename CONTAINER | |||
| } | ||||
| 
 | ||||
| extern std::string xml_escape(std::string text, bool is_marked = false); | ||||
| extern std::string xml_escape_double_quotes_attribute_value(std::string text); | ||||
| extern std::string xml_unescape(std::string text); | ||||
| 
 | ||||
| 
 | ||||
|  |  | |||
|  | @ -1225,6 +1225,34 @@ std::string xml_escape(std::string text, bool is_marked/* = false*/) | |||
|     return text; | ||||
| } | ||||
| 
 | ||||
| // Definition of escape symbols https://www.w3.org/TR/REC-xml/#AVNormalize
 | ||||
| // During the read of xml attribute normalization of white spaces is applied
 | ||||
| // Soo for not lose white space character it is escaped before store
 | ||||
| std::string xml_escape_double_quotes_attribute_value(std::string text) | ||||
| { | ||||
|     std::string::size_type pos = 0; | ||||
|     for (;;) { | ||||
|         pos = text.find_first_of("\"&<\r\n\t", pos); | ||||
|         if (pos == std::string::npos) break; | ||||
| 
 | ||||
|         std::string replacement; | ||||
|         switch (text[pos]) { | ||||
|         case '\"': replacement = """; break; | ||||
|         case '&': replacement = "&"; break; | ||||
|         case '<': replacement = "<"; break; | ||||
|         case '\r': replacement = "
"; break; | ||||
|         case '\n': replacement = "
"; break; | ||||
|         case '\t': replacement = "	"; break; | ||||
|         default: break; | ||||
|         } | ||||
| 
 | ||||
|         text.replace(pos, 1, replacement); | ||||
|         pos += replacement.size(); | ||||
|     } | ||||
| 
 | ||||
|     return text; | ||||
| } | ||||
| 
 | ||||
| std::string xml_unescape(std::string s) | ||||
| { | ||||
| 	std::string ret; | ||||
|  |  | |||
|  | @ -124,6 +124,8 @@ set(SLIC3R_GUI_SOURCES | |||
|     GUI/Gizmos/GLGizmosCommon.hpp | ||||
|     GUI/Gizmos/GLGizmoBase.cpp | ||||
|     GUI/Gizmos/GLGizmoBase.hpp | ||||
|     GUI/Gizmos/GLGizmoEmboss.cpp | ||||
|     GUI/Gizmos/GLGizmoEmboss.hpp | ||||
|     GUI/Gizmos/GLGizmoMove.cpp | ||||
|     GUI/Gizmos/GLGizmoMove.hpp | ||||
|     GUI/Gizmos/GLGizmoRotate.cpp | ||||
|  | @ -144,6 +146,8 @@ set(SLIC3R_GUI_SOURCES | |||
|     GUI/Gizmos/GLGizmoPainterBase.hpp | ||||
|     GUI/Gizmos/GLGizmoSimplify.cpp | ||||
|     GUI/Gizmos/GLGizmoSimplify.hpp | ||||
|     GUI/Gizmos/GLGizmoSVG.cpp | ||||
|     GUI/Gizmos/GLGizmoSVG.hpp | ||||
|     GUI/Gizmos/GLGizmoMmuSegmentation.cpp | ||||
|     GUI/Gizmos/GLGizmoMmuSegmentation.hpp | ||||
|     #GUI/Gizmos/GLGizmoFaceDetector.cpp | ||||
|  | @ -152,8 +156,8 @@ set(SLIC3R_GUI_SOURCES | |||
|     GUI/Gizmos/GLGizmoMeasure.hpp | ||||
|     GUI/Gizmos/GLGizmoSeam.cpp | ||||
|     GUI/Gizmos/GLGizmoSeam.hpp | ||||
|     GUI/Gizmos/GLGizmoText.cpp | ||||
|     GUI/Gizmos/GLGizmoText.hpp | ||||
|     #GUI/Gizmos/GLGizmoText.cpp | ||||
|     #GUI/Gizmos/GLGizmoText.hpp | ||||
|     GUI/Gizmos/GLGizmoMeshBoolean.cpp | ||||
|     GUI/Gizmos/GLGizmoMeshBoolean.hpp | ||||
|     GUI/GLSelectionRectangle.cpp | ||||
|  | @ -194,6 +198,8 @@ set(SLIC3R_GUI_SOURCES | |||
|     GUI/GUI_Geometry.hpp | ||||
|     GUI/I18N.cpp | ||||
|     GUI/I18N.hpp | ||||
|     GUI/IconManager.cpp | ||||
|     GUI/IconManager.hpp | ||||
|     GUI/MainFrame.cpp | ||||
|     GUI/MainFrame.hpp | ||||
|     GUI/BBLTopbar.cpp | ||||
|  | @ -303,6 +309,10 @@ set(SLIC3R_GUI_SOURCES | |||
|     GUI/RemovableDriveManager.hpp | ||||
|     GUI/SendSystemInfoDialog.cpp | ||||
|     GUI/SendSystemInfoDialog.hpp | ||||
|     GUI/SurfaceDrag.cpp | ||||
|     GUI/SurfaceDrag.hpp | ||||
|     GUI/TextLines.cpp | ||||
|     GUI/TextLines.hpp | ||||
|     GUI/PlateSettingsDialog.cpp | ||||
|     GUI/PlateSettingsDialog.hpp | ||||
|     GUI/ImGuiWrapper.hpp | ||||
|  | @ -329,13 +339,21 @@ set(SLIC3R_GUI_SOURCES | |||
|     GUI/UpdateDialogs.cpp | ||||
|     GUI/UpdateDialogs.hpp | ||||
|     GUI/Jobs/Job.hpp | ||||
|     GUI/Jobs/Job.cpp | ||||
|     GUI/Jobs/PlaterJob.hpp | ||||
|     GUI/Jobs/PlaterJob.cpp | ||||
|     GUI/Jobs/Worker.hpp | ||||
|     GUI/Jobs/BoostThreadWorker.hpp | ||||
|     GUI/Jobs/BoostThreadWorker.cpp | ||||
|     GUI/Jobs/BusyCursorJob.hpp | ||||
|     GUI/Jobs/PlaterWorker.hpp | ||||
|     GUI/Jobs/UpgradeNetworkJob.hpp | ||||
|     GUI/Jobs/UpgradeNetworkJob.cpp | ||||
|     GUI/Jobs/ArrangeJob.hpp | ||||
|     GUI/Jobs/ArrangeJob.cpp | ||||
|     GUI/Jobs/CreateFontNameImageJob.cpp | ||||
|     GUI/Jobs/CreateFontNameImageJob.hpp | ||||
|     GUI/Jobs/CreateFontStyleImagesJob.cpp | ||||
|     GUI/Jobs/CreateFontStyleImagesJob.hpp | ||||
|     GUI/Jobs/EmbossJob.cpp | ||||
|     GUI/Jobs/EmbossJob.hpp | ||||
|     GUI/Jobs/OrientJob.hpp | ||||
|     GUI/Jobs/OrientJob.cpp | ||||
|     GUI/Jobs/RotoptimizeJob.hpp | ||||
|  | @ -353,6 +371,8 @@ set(SLIC3R_GUI_SOURCES | |||
|     GUI/Jobs/BindJob.cpp | ||||
|     GUI/Jobs/NotificationProgressIndicator.hpp | ||||
|     GUI/Jobs/NotificationProgressIndicator.cpp | ||||
|     GUI/Jobs/ThreadSafeQueue.hpp | ||||
|     GUI/Jobs/SLAImportDialog.hpp | ||||
|     GUI/PhysicalPrinterDialog.hpp | ||||
|     GUI/PhysicalPrinterDialog.cpp | ||||
|     GUI/ProgressStatusBar.hpp | ||||
|  | @ -449,6 +469,10 @@ set(SLIC3R_GUI_SOURCES | |||
|     Utils/Http.hpp | ||||
|     Utils/FixModelByWin10.cpp | ||||
|     Utils/FixModelByWin10.hpp | ||||
|     Utils/EmbossStyleManager.cpp | ||||
|     Utils/EmbossStyleManager.hpp | ||||
|     Utils/FontConfigHelp.cpp | ||||
|     Utils/FontConfigHelp.hpp | ||||
|     Utils/Bonjour.cpp | ||||
|     Utils/Bonjour.hpp | ||||
|     Utils/FileHelp.cpp | ||||
|  | @ -457,6 +481,8 @@ set(SLIC3R_GUI_SOURCES | |||
|     Utils/PresetUpdater.hpp | ||||
|     Utils/Process.cpp | ||||
|     Utils/Process.hpp | ||||
|     Utils/RaycastManager.cpp | ||||
|     Utils/RaycastManager.hpp | ||||
|     Utils/Profile.hpp | ||||
|     Utils/UndoRedo.cpp | ||||
|     Utils/UndoRedo.hpp | ||||
|  | @ -480,8 +506,10 @@ set(SLIC3R_GUI_SOURCES | |||
|     Utils/PrintHost.hpp | ||||
|     Utils/Serial.cpp | ||||
|     Utils/Serial.hpp | ||||
|     Utils/MKS.hpp | ||||
|     Utils/MKS.cpp | ||||
|     Utils/MKS.hpp | ||||
|     Utils/WxFontUtils.cpp | ||||
|     Utils/WxFontUtils.hpp | ||||
|     Utils/Duet.cpp | ||||
|     Utils/Duet.hpp | ||||
|     Utils/FlashAir.cpp | ||||
|  | @ -591,7 +619,7 @@ endif () | |||
| if (UNIX AND NOT APPLE) | ||||
|     find_package(GTK${SLIC3R_GTK} REQUIRED) | ||||
|     target_include_directories(libslic3r_gui PRIVATE ${GTK${SLIC3R_GTK}_INCLUDE_DIRS}) | ||||
|     target_link_libraries(libslic3r_gui ${GTK${SLIC3R_GTK}_LIBRARIES}) | ||||
|     target_link_libraries(libslic3r_gui ${GTK${SLIC3R_GTK}_LIBRARIES} fontconfig) | ||||
| 
 | ||||
|     # We add GStreamer for bambu:/// support. | ||||
|     pkg_check_modules(GSTREAMER REQUIRED gstreamer-1.0) | ||||
|  |  | |||
|  | @ -940,9 +940,9 @@ void GLVolumeCollection::render(GLVolumeCollection::ERenderType type, bool disab | |||
|         const Matrix3d view_normal_matrix = view_matrix.matrix().block(0, 0, 3, 3) * model_matrix.matrix().block(0, 0, 3, 3).inverse().transpose(); | ||||
|         shader->set_uniform("view_normal_matrix", view_normal_matrix); | ||||
| 		//BBS: add outline related logic
 | ||||
|         if (with_outline && volume.first->selected) | ||||
|             volume.first->render_with_outline(view_matrix * model_matrix); | ||||
|         else | ||||
|         //if (with_outline && volume.first->selected)
 | ||||
|         //    volume.first->render_with_outline(view_matrix * model_matrix);
 | ||||
|         //else
 | ||||
|             volume.first->render(); | ||||
| 
 | ||||
| #if ENABLE_ENVIRONMENT_MAP | ||||
|  |  | |||
|  | @ -243,14 +243,15 @@ public: | |||
| 
 | ||||
|     const Geometry::Transformation& get_instance_transformation() const { return m_instance_transformation; } | ||||
|     void set_instance_transformation(const Geometry::Transformation& transformation) { m_instance_transformation = transformation; set_bounding_boxes_as_dirty(); } | ||||
|     void set_instance_transformation(const Transform3d& transform) { m_instance_transformation.set_matrix(transform); set_bounding_boxes_as_dirty(); } | ||||
| 
 | ||||
|     const Vec3d& get_instance_offset() const { return m_instance_transformation.get_offset(); } | ||||
|     Vec3d get_instance_offset() const { return m_instance_transformation.get_offset(); } | ||||
|     double get_instance_offset(Axis axis) const { return m_instance_transformation.get_offset(axis); } | ||||
| 
 | ||||
|     void set_instance_offset(const Vec3d& offset) { m_instance_transformation.set_offset(offset); set_bounding_boxes_as_dirty(); } | ||||
|     void set_instance_offset(Axis axis, double offset) { m_instance_transformation.set_offset(axis, offset); set_bounding_boxes_as_dirty(); } | ||||
| 
 | ||||
|     const Vec3d& get_instance_rotation() const { return m_instance_transformation.get_rotation(); } | ||||
|     Vec3d get_instance_rotation() const { return m_instance_transformation.get_rotation(); } | ||||
|     double get_instance_rotation(Axis axis) const { return m_instance_transformation.get_rotation(axis); } | ||||
| 
 | ||||
|     void set_instance_rotation(const Vec3d& rotation) { m_instance_transformation.set_rotation(rotation); set_bounding_boxes_as_dirty(); } | ||||
|  | @ -262,7 +263,7 @@ public: | |||
|     void set_instance_scaling_factor(const Vec3d& scaling_factor) { m_instance_transformation.set_scaling_factor(scaling_factor); set_bounding_boxes_as_dirty(); } | ||||
|     void set_instance_scaling_factor(Axis axis, double scaling_factor) { m_instance_transformation.set_scaling_factor(axis, scaling_factor); set_bounding_boxes_as_dirty(); } | ||||
| 
 | ||||
|     const Vec3d& get_instance_mirror() const { return m_instance_transformation.get_mirror(); } | ||||
|     Vec3d get_instance_mirror() const { return m_instance_transformation.get_mirror(); } | ||||
|     double get_instance_mirror(Axis axis) const { return m_instance_transformation.get_mirror(axis); } | ||||
| 
 | ||||
|     void set_instance_mirror(const Vec3d& mirror) { m_instance_transformation.set_mirror(mirror); set_bounding_boxes_as_dirty(); } | ||||
|  | @ -270,26 +271,27 @@ public: | |||
| 
 | ||||
|     const Geometry::Transformation& get_volume_transformation() const { return m_volume_transformation; } | ||||
|     void set_volume_transformation(const Geometry::Transformation& transformation) { m_volume_transformation = transformation; set_bounding_boxes_as_dirty(); } | ||||
|     void set_volume_transformation(const Transform3d& transform) { m_volume_transformation.set_matrix(transform); set_bounding_boxes_as_dirty(); } | ||||
| 
 | ||||
|     const Vec3d& get_volume_offset() const { return m_volume_transformation.get_offset(); } | ||||
|     Vec3d get_volume_offset() const { return m_volume_transformation.get_offset(); } | ||||
|     double get_volume_offset(Axis axis) const { return m_volume_transformation.get_offset(axis); } | ||||
| 
 | ||||
|     void set_volume_offset(const Vec3d& offset) { m_volume_transformation.set_offset(offset); set_bounding_boxes_as_dirty(); } | ||||
|     void set_volume_offset(Axis axis, double offset) { m_volume_transformation.set_offset(axis, offset); set_bounding_boxes_as_dirty(); } | ||||
| 
 | ||||
|     const Vec3d& get_volume_rotation() const { return m_volume_transformation.get_rotation(); } | ||||
|     Vec3d get_volume_rotation() const { return m_volume_transformation.get_rotation(); } | ||||
|     double get_volume_rotation(Axis axis) const { return m_volume_transformation.get_rotation(axis); } | ||||
| 
 | ||||
|     void set_volume_rotation(const Vec3d& rotation) { m_volume_transformation.set_rotation(rotation); set_bounding_boxes_as_dirty(); } | ||||
|     void set_volume_rotation(Axis axis, double rotation) { m_volume_transformation.set_rotation(axis, rotation); set_bounding_boxes_as_dirty(); } | ||||
| 
 | ||||
|     const Vec3d& get_volume_scaling_factor() const { return m_volume_transformation.get_scaling_factor(); } | ||||
|     Vec3d get_volume_scaling_factor() const { return m_volume_transformation.get_scaling_factor(); } | ||||
|     double get_volume_scaling_factor(Axis axis) const { return m_volume_transformation.get_scaling_factor(axis); } | ||||
| 
 | ||||
|     void set_volume_scaling_factor(const Vec3d& scaling_factor) { m_volume_transformation.set_scaling_factor(scaling_factor); set_bounding_boxes_as_dirty(); } | ||||
|     void set_volume_scaling_factor(Axis axis, double scaling_factor) { m_volume_transformation.set_scaling_factor(axis, scaling_factor); set_bounding_boxes_as_dirty(); } | ||||
| 
 | ||||
|     const Vec3d& get_volume_mirror() const { return m_volume_transformation.get_mirror(); } | ||||
|     Vec3d get_volume_mirror() const { return m_volume_transformation.get_mirror(); } | ||||
|     double get_volume_mirror(Axis axis) const { return m_volume_transformation.get_mirror(axis); } | ||||
| 
 | ||||
|     void set_volume_mirror(const Vec3d& mirror) { m_volume_transformation.set_mirror(mirror); set_bounding_boxes_as_dirty(); } | ||||
|  |  | |||
|  | @ -13,6 +13,8 @@ | |||
| #include "MainFrame.hpp" | ||||
| #include "GUI_App.hpp" | ||||
| #include "Plater.hpp" | ||||
| #include "Jobs/BoostThreadWorker.hpp" | ||||
| #include "Jobs/PlaterWorker.hpp" | ||||
| #include "Widgets/WebView.hpp" | ||||
| 
 | ||||
| namespace Slic3r { | ||||
|  | @ -374,6 +376,8 @@ wxString get_fail_reason(int code) | |||
| 
 | ||||
|      m_status_bar = std::make_shared<BBLStatusBarBind>(m_simplebook); | ||||
| 
 | ||||
|      m_worker = std::make_unique<PlaterWorker<BoostThreadWorker>>(this, m_status_bar, "bind_worker"); | ||||
| 
 | ||||
|      auto        button_panel   = new wxPanel(m_simplebook, wxID_ANY, wxDefaultPosition, BIND_DIALOG_BUTTON_PANEL_SIZE); | ||||
|      button_panel->SetBackgroundColour(*wxWHITE); | ||||
|      wxBoxSizer *m_sizer_button = new wxBoxSizer(wxHORIZONTAL); | ||||
|  | @ -513,10 +517,7 @@ wxString get_fail_reason(int code) | |||
| 
 | ||||
|  void BindMachineDialog::on_destroy() | ||||
|  { | ||||
|      if (m_bind_job) { | ||||
|          m_bind_job->cancel(); | ||||
|          m_bind_job->join(); | ||||
|      } | ||||
|      m_worker.get()->cancel_all(); | ||||
|  } | ||||
| 
 | ||||
|  void BindMachineDialog::on_close(wxCloseEvent &event) | ||||
|  | @ -572,7 +573,7 @@ wxString get_fail_reason(int code) | |||
|          agent->track_update_property("dev_ota_version", m_machine_info->get_ota_version()); | ||||
| 
 | ||||
|      m_simplebook->SetSelection(0); | ||||
|      m_bind_job = std::make_shared<BindJob>(m_status_bar, wxGetApp().plater(), m_machine_info->dev_id, m_machine_info->dev_ip, m_machine_info->bind_sec_link); | ||||
|      auto m_bind_job = std::make_unique<BindJob>(m_machine_info->dev_id, m_machine_info->dev_ip, m_machine_info->bind_sec_link); | ||||
| 
 | ||||
|      if (m_machine_info && (m_machine_info->get_printer_series() == PrinterSeries::SERIES_X1)) { | ||||
|          m_bind_job->set_improved(false); | ||||
|  | @ -582,7 +583,7 @@ wxString get_fail_reason(int code) | |||
|      } | ||||
| 
 | ||||
|      m_bind_job->set_event_handle(this); | ||||
|      m_bind_job->start(); | ||||
|      replace_job(*m_worker, std::move(m_bind_job)); | ||||
|  } | ||||
| 
 | ||||
| void BindMachineDialog::on_dpi_changed(const wxRect &suggested_rect) | ||||
|  |  | |||
|  | @ -28,6 +28,7 @@ | |||
| #include "Jobs/BindJob.hpp" | ||||
| #include "BBLStatusBar.hpp" | ||||
| #include "BBLStatusBarBind.hpp" | ||||
| #include "Jobs/Worker.hpp" | ||||
| 
 | ||||
| #define BIND_DIALOG_GREY200 wxColour(248, 248, 248) | ||||
| #define BIND_DIALOG_GREY800 wxColour(50, 58, 61) | ||||
|  | @ -77,8 +78,8 @@ private: | |||
|     std::shared_ptr<int>     m_tocken; | ||||
| 
 | ||||
|     MachineObject *                   m_machine_info{nullptr}; | ||||
|     std::shared_ptr<BindJob>          m_bind_job; | ||||
|     std::shared_ptr<BBLStatusBarBind> m_status_bar; | ||||
|     std::unique_ptr<Worker>           m_worker; | ||||
| 
 | ||||
| public: | ||||
|     BindMachineDialog(Plater *plater = nullptr); | ||||
|  |  | |||
|  | @ -1437,12 +1437,10 @@ void CalibrationPresetPage::on_cali_finished_job() | |||
| void CalibrationPresetPage::on_cali_cancel_job() | ||||
| { | ||||
|     BOOST_LOG_TRIVIAL(info) << "CalibrationWizard::print_job: enter canceled"; | ||||
|     if (CalibUtils::print_job) { | ||||
|         if (CalibUtils::print_job->is_running()) { | ||||
|     if (CalibUtils::print_worker) { | ||||
|         BOOST_LOG_TRIVIAL(info) << "calibration_print_job: canceled"; | ||||
|             CalibUtils::print_job->cancel(); | ||||
|         } | ||||
|         CalibUtils::print_job->join(); | ||||
|         CalibUtils::print_worker->cancel_all(); | ||||
|         CalibUtils::print_worker->wait_for_idle(); | ||||
|     } | ||||
| 
 | ||||
|     m_sending_panel->reset(); | ||||
|  |  | |||
|  | @ -1396,12 +1396,10 @@ void CalibrationFlowCoarseSavePage::on_cali_finished_job() | |||
| void CalibrationFlowCoarseSavePage::on_cali_cancel_job() | ||||
| { | ||||
|     BOOST_LOG_TRIVIAL(info) << "CalibrationWizard::print_job: enter canceled"; | ||||
|     if (CalibUtils::print_job) { | ||||
|         if (CalibUtils::print_job->is_running()) { | ||||
|     if (CalibUtils::print_worker) { | ||||
|         BOOST_LOG_TRIVIAL(info) << "calibration_print_job: canceled"; | ||||
|             CalibUtils::print_job->cancel(); | ||||
|         } | ||||
|         CalibUtils::print_job->join(); | ||||
|         CalibUtils::print_worker->cancel_all(); | ||||
|         CalibUtils::print_worker->wait_for_idle(); | ||||
|     } | ||||
| 
 | ||||
|     m_sending_panel->reset(); | ||||
|  |  | |||
|  | @ -20,6 +20,8 @@ | |||
| #include "wxExtensions.hpp" | ||||
| #include "slic3r/GUI/MainFrame.hpp" | ||||
| #include "GUI_App.hpp" | ||||
| #include "Jobs/BoostThreadWorker.hpp" | ||||
| #include "Jobs/PlaterWorker.hpp" | ||||
| 
 | ||||
| #define DESIGN_INPUT_SIZE wxSize(FromDIP(100), -1) | ||||
| 
 | ||||
|  | @ -60,6 +62,7 @@ DownloadProgressDialog::DownloadProgressDialog(wxString title) | |||
|     m_panel_download->SetMinSize(wxSize(FromDIP(400), FromDIP(70))); | ||||
|     m_panel_download->SetMaxSize(wxSize(FromDIP(400), FromDIP(70))); | ||||
| 
 | ||||
|     m_worker = std::make_unique<PlaterWorker<BoostThreadWorker>>(this, m_status_bar, "download_worker"); | ||||
| 
 | ||||
|     //mode Download Failed 
 | ||||
|     auto m_panel_download_failed = new wxPanel(m_simplebook_status, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL); | ||||
|  | @ -144,7 +147,7 @@ bool DownloadProgressDialog::Show(bool show) | |||
| { | ||||
|     if (show) { | ||||
|         m_simplebook_status->SetSelection(0); | ||||
|         m_upgrade_job = make_job(m_status_bar); | ||||
|         auto m_upgrade_job = make_job(); | ||||
|         m_upgrade_job->set_event_handle(this); | ||||
|         m_status_bar->set_progress(0); | ||||
|         Bind(EVT_UPGRADE_NETWORK_SUCCESS, [this](wxCommandEvent& evt) { | ||||
|  | @ -182,23 +185,17 @@ bool DownloadProgressDialog::Show(bool show) | |||
|         }); | ||||
| 
 | ||||
|         m_status_bar->set_cancel_callback_fina([this]() { | ||||
|             if (m_upgrade_job) { | ||||
|                 m_upgrade_job->cancel(); | ||||
|                 //EndModal(wxID_CLOSE);
 | ||||
|             } | ||||
|                  | ||||
|             m_worker->cancel_all(); | ||||
|         }); | ||||
|         m_upgrade_job->start(); | ||||
| 
 | ||||
|         replace_job(*m_worker, std::move(m_upgrade_job)); | ||||
|     } | ||||
|     return DPIDialog::Show(show); | ||||
| } | ||||
| 
 | ||||
| void DownloadProgressDialog::on_close(wxCloseEvent& event) | ||||
| { | ||||
|     if (m_upgrade_job) { | ||||
|         m_upgrade_job->cancel(); | ||||
|         m_upgrade_job->join(); | ||||
|     } | ||||
|     m_worker.get()->cancel_all(); | ||||
|     event.Skip(); | ||||
| } | ||||
| 
 | ||||
|  | @ -208,7 +205,7 @@ void DownloadProgressDialog::on_dpi_changed(const wxRect &suggested_rect) {} | |||
| 
 | ||||
| void DownloadProgressDialog::update_release_note(std::string release_note, std::string version) {} | ||||
| 
 | ||||
| std::shared_ptr<UpgradeNetworkJob> DownloadProgressDialog::make_job(std::shared_ptr<ProgressIndicator> pri) { return std::make_shared<UpgradeNetworkJob>(pri); } | ||||
| std::unique_ptr<UpgradeNetworkJob> DownloadProgressDialog::make_job() { return std::make_unique<UpgradeNetworkJob>(); } | ||||
| 
 | ||||
| void DownloadProgressDialog::on_finish() { wxGetApp().restart_networking(); } | ||||
| 
 | ||||
|  |  | |||
|  | @ -16,6 +16,7 @@ | |||
| #include "Widgets/Button.hpp" | ||||
| #include "BBLStatusBar.hpp" | ||||
| #include "BBLStatusBarSend.hpp" | ||||
| #include "Jobs/Worker.hpp" | ||||
| #include "Jobs/UpgradeNetworkJob.hpp" | ||||
| 
 | ||||
| class wxBoxSizer; | ||||
|  | @ -47,11 +48,11 @@ public: | |||
|     wxSimplebook* m_simplebook_status{nullptr}; | ||||
| 
 | ||||
| 	std::shared_ptr<BBLStatusBarSend> m_status_bar; | ||||
|     std::shared_ptr<UpgradeNetworkJob> m_upgrade_job { nullptr }; | ||||
|     std::unique_ptr<Worker>           m_worker; | ||||
|     wxPanel *                         m_panel_download; | ||||
| 
 | ||||
| protected: | ||||
|     virtual std::shared_ptr<UpgradeNetworkJob> make_job(std::shared_ptr<ProgressIndicator> pri); | ||||
|     virtual std::unique_ptr<UpgradeNetworkJob> make_job(); | ||||
|     virtual void                               on_finish(); | ||||
| }; | ||||
| 
 | ||||
|  |  | |||
|  | @ -361,7 +361,7 @@ void GCodeViewer::SequentialView::Marker::render(int canvas_width, int canvas_he | |||
|     std::string layer_time = ImGui::ColorMarkerStart + _u8L("Layer Time: ") + ImGui::ColorMarkerEnd; | ||||
|     std::string fanspeed = ImGui::ColorMarkerStart + _u8L("Fan: ") + ImGui::ColorMarkerEnd; | ||||
|     std::string temperature = ImGui::ColorMarkerStart + _u8L("Temperature: ") + ImGui::ColorMarkerEnd; | ||||
|     const float item_size = imgui.calc_text_size("X: 000.000  ").x; | ||||
|     const float item_size = imgui.calc_text_size(std::string_view{"X: 000.000  "}).x; | ||||
|     const float item_spacing = imgui.get_item_spacing().x; | ||||
|     const float window_padding = ImGui::GetStyle().WindowPadding.x; | ||||
| 
 | ||||
|  |  | |||
|  | @ -1588,7 +1588,6 @@ void GLCanvas3D::enable_legend_texture(bool enable) | |||
| void GLCanvas3D::enable_picking(bool enable) | ||||
| { | ||||
|     m_picking_enabled = enable; | ||||
|     m_selection.set_mode(Selection::Instance); | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3D::enable_moving(bool enable) | ||||
|  | @ -2171,7 +2170,17 @@ std::vector<int> GLCanvas3D::load_object(const Model& model, int obj_idx) | |||
| 
 | ||||
| void GLCanvas3D::mirror_selection(Axis axis) | ||||
| { | ||||
|     m_selection.mirror(axis); | ||||
|     TransformationType transformation_type; | ||||
|     if (wxGetApp().obj_manipul()->is_local_coordinates()) | ||||
|         transformation_type.set_local(); | ||||
|     else if (wxGetApp().obj_manipul()->is_instance_coordinates()) | ||||
|         transformation_type.set_instance(); | ||||
| 
 | ||||
|     transformation_type.set_relative(); | ||||
| 
 | ||||
|     m_selection.setup_cache(); | ||||
|     m_selection.mirror(axis, transformation_type); | ||||
| 
 | ||||
|     do_mirror(L("Mirror Object")); | ||||
|     // BBS
 | ||||
|     //wxGetApp().obj_manipul()->set_dirty();
 | ||||
|  | @ -3364,7 +3373,9 @@ void GLCanvas3D::on_key(wxKeyEvent& evt) | |||
|             else | ||||
|                 displacement = multiplier * direction; | ||||
| 
 | ||||
|             m_selection.translate(displacement); | ||||
|             TransformationType trafo_type; | ||||
|             trafo_type.set_relative(); | ||||
|             m_selection.translate(displacement, trafo_type); | ||||
|             m_dirty = true; | ||||
|         } | ||||
|     );} | ||||
|  | @ -4136,7 +4147,9 @@ void GLCanvas3D::on_mouse(wxMouseEvent& evt) | |||
|                     } | ||||
|                 } | ||||
| 
 | ||||
|                 m_selection.translate(cur_pos - m_mouse.drag.start_position_3D); | ||||
|                 TransformationType trafo_type; | ||||
|                 trafo_type.set_relative(); | ||||
|                 m_selection.translate(cur_pos - m_mouse.drag.start_position_3D, trafo_type); | ||||
|                 if (current_printer_technology() == ptFFF && (fff_print()->config().print_sequence == PrintSequence::ByObject)) | ||||
|                     update_sequential_clearance(); | ||||
|                 // BBS
 | ||||
|  | @ -4364,6 +4377,40 @@ void GLCanvas3D::on_mouse(wxMouseEvent& evt) | |||
|     else | ||||
|         evt.Skip(); | ||||
| 
 | ||||
|     // Detection of doubleclick on text to open emboss edit window
 | ||||
|     auto type = m_gizmos.get_current_type(); | ||||
|     if (evt.LeftDClick() && !m_hover_volume_idxs.empty() &&  | ||||
|         (type == GLGizmosManager::EType::Undefined || | ||||
|          type == GLGizmosManager::EType::Move || | ||||
|          type == GLGizmosManager::EType::Rotate || | ||||
|          type == GLGizmosManager::EType::Scale || | ||||
|          type == GLGizmosManager::EType::Emboss || | ||||
|          type == GLGizmosManager::EType::Svg) ) { | ||||
|         for (int hover_volume_id : m_hover_volume_idxs) {  | ||||
|             const GLVolume &hover_gl_volume = *m_volumes.volumes[hover_volume_id]; | ||||
|             int object_idx = hover_gl_volume.object_idx(); | ||||
|             if (object_idx < 0 || static_cast<size_t>(object_idx) >= m_model->objects.size()) continue; | ||||
|             const ModelObject* hover_object = m_model->objects[object_idx]; | ||||
|             int hover_volume_idx = hover_gl_volume.volume_idx(); | ||||
|             if (hover_volume_idx < 0 || static_cast<size_t>(hover_volume_idx) >= hover_object->volumes.size()) continue; | ||||
|             const ModelVolume* hover_volume = hover_object->volumes[hover_volume_idx]; | ||||
| 
 | ||||
|             if (hover_volume->text_configuration.has_value()) { | ||||
|                 m_selection.add_volumes(Selection::EMode::Volume, {(unsigned) hover_volume_id}); | ||||
|                 if (type != GLGizmosManager::EType::Emboss) | ||||
|                     m_gizmos.open_gizmo(GLGizmosManager::EType::Emboss);             | ||||
|                 wxGetApp().obj_list()->update_selections(); | ||||
|                 return; | ||||
|             } else if (hover_volume->emboss_shape.has_value()) { | ||||
|                 m_selection.add_volumes(Selection::EMode::Volume, {(unsigned) hover_volume_id}); | ||||
|                 if (type != GLGizmosManager::EType::Svg) | ||||
|                     m_gizmos.open_gizmo(GLGizmosManager::EType::Svg); | ||||
|                 wxGetApp().obj_list()->update_selections(); | ||||
|                 return; | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     if (m_moving) | ||||
|         show_sinking_contours(); | ||||
| 
 | ||||
|  | @ -4460,6 +4507,9 @@ void GLCanvas3D::do_move(const std::string& snapshot_type) | |||
|         int instance_idx = v->instance_idx(); | ||||
|         int volume_idx = v->volume_idx(); | ||||
| 
 | ||||
|         if (volume_idx < 0) | ||||
|             continue; | ||||
| 
 | ||||
|         std::pair<int, int> done_id(object_idx, instance_idx); | ||||
| 
 | ||||
|         if (0 <= object_idx && object_idx < (int)m_model->objects.size()) { | ||||
|  | @ -4469,10 +4519,10 @@ void GLCanvas3D::do_move(const std::string& snapshot_type) | |||
|             ModelObject* model_object = m_model->objects[object_idx]; | ||||
|             if (model_object != nullptr) { | ||||
|                 if (selection_mode == Selection::Instance) | ||||
|                     model_object->instances[instance_idx]->set_offset(v->get_instance_offset()); | ||||
|                     model_object->instances[instance_idx]->set_transformation(v->get_instance_transformation()); | ||||
|                 else if (selection_mode == Selection::Volume) { | ||||
|                     if (model_object->volumes[volume_idx]->get_offset() != v->get_volume_offset()) { | ||||
|                         model_object->volumes[volume_idx]->set_offset(v->get_volume_offset()); | ||||
|                     if (model_object->volumes[volume_idx]->get_transformation() != v->get_volume_transformation()) { | ||||
|                         model_object->volumes[volume_idx]->set_transformation(v->get_volume_transformation()); | ||||
|                         // BBS: backup
 | ||||
|                         Slic3r::save_object_mesh(*model_object); | ||||
|                     } | ||||
|  | @ -4564,26 +4614,26 @@ void GLCanvas3D::do_rotate(const std::string& snapshot_type) | |||
|     Selection::EMode selection_mode = m_selection.get_mode(); | ||||
| 
 | ||||
|     for (const GLVolume* v : m_volumes.volumes) { | ||||
|         int object_idx = v->object_idx(); | ||||
|         const int object_idx = v->object_idx(); | ||||
|         if (object_idx < 0 || (int)m_model->objects.size() <= object_idx) | ||||
|             continue; | ||||
| 
 | ||||
|         int instance_idx = v->instance_idx(); | ||||
|         int volume_idx = v->volume_idx(); | ||||
|         const int instance_idx = v->instance_idx(); | ||||
|         const int volume_idx = v->volume_idx(); | ||||
| 
 | ||||
|         if (volume_idx < 0) | ||||
|             continue; | ||||
| 
 | ||||
|         done.insert(std::pair<int, int>(object_idx, instance_idx)); | ||||
| 
 | ||||
|         // Rotate instances/volumes.
 | ||||
|         ModelObject* model_object = m_model->objects[object_idx]; | ||||
|         if (model_object != nullptr) { | ||||
|             if (selection_mode == Selection::Instance) { | ||||
|                 model_object->instances[instance_idx]->set_rotation(v->get_instance_rotation()); | ||||
|                 model_object->instances[instance_idx]->set_offset(v->get_instance_offset()); | ||||
|             } | ||||
|             if (selection_mode == Selection::Instance) | ||||
|                 model_object->instances[instance_idx]->set_transformation(v->get_instance_transformation()); | ||||
|             else if (selection_mode == Selection::Volume) { | ||||
|                 if (model_object->volumes[volume_idx]->get_rotation() != v->get_volume_rotation()) { | ||||
|                     model_object->volumes[volume_idx]->set_rotation(v->get_volume_rotation()); | ||||
|                     model_object->volumes[volume_idx]->set_offset(v->get_volume_offset()); | ||||
|                 if (model_object->volumes[volume_idx]->get_transformation() != v->get_volume_transformation()) { | ||||
|                 	model_object->volumes[volume_idx]->set_transformation(v->get_volume_transformation()); | ||||
|                     // BBS: backup
 | ||||
|                     Slic3r::save_object_mesh(*model_object); | ||||
|                 } | ||||
|  | @ -4645,27 +4695,27 @@ void GLCanvas3D::do_scale(const std::string& snapshot_type) | |||
|     Selection::EMode selection_mode = m_selection.get_mode(); | ||||
| 
 | ||||
|     for (const GLVolume* v : m_volumes.volumes) { | ||||
|         int object_idx = v->object_idx(); | ||||
|         const int object_idx = v->object_idx(); | ||||
|         if (object_idx < 0 || (int)m_model->objects.size() <= object_idx) | ||||
|             continue; | ||||
| 
 | ||||
|         int instance_idx = v->instance_idx(); | ||||
|         int volume_idx = v->volume_idx(); | ||||
|         const int instance_idx = v->instance_idx(); | ||||
|         const int volume_idx = v->volume_idx(); | ||||
| 
 | ||||
|         if (volume_idx < 0) | ||||
|             continue; | ||||
| 
 | ||||
|         done.insert(std::pair<int, int>(object_idx, instance_idx)); | ||||
| 
 | ||||
|         // Rotate instances/volumes
 | ||||
|         ModelObject* model_object = m_model->objects[object_idx]; | ||||
|         if (model_object != nullptr) { | ||||
|             if (selection_mode == Selection::Instance) { | ||||
|                 model_object->instances[instance_idx]->set_scaling_factor(v->get_instance_scaling_factor()); | ||||
|                 model_object->instances[instance_idx]->set_offset(v->get_instance_offset()); | ||||
|             } | ||||
|             if (selection_mode == Selection::Instance) | ||||
|                 model_object->instances[instance_idx]->set_transformation(v->get_instance_transformation()); | ||||
|             else if (selection_mode == Selection::Volume) { | ||||
|                 if (model_object->volumes[volume_idx]->get_scaling_factor() != v->get_volume_scaling_factor()) { | ||||
|                     model_object->instances[instance_idx]->set_offset(v->get_instance_offset()); | ||||
|                     model_object->volumes[volume_idx]->set_scaling_factor(v->get_volume_scaling_factor()); | ||||
|                     model_object->volumes[volume_idx]->set_offset(v->get_volume_offset()); | ||||
|                 if (model_object->volumes[volume_idx]->get_transformation() != v->get_volume_transformation()) { | ||||
|                     model_object->instances[instance_idx]->set_transformation(v->get_instance_transformation()); | ||||
|                     model_object->volumes[volume_idx]->set_transformation(v->get_volume_transformation()); | ||||
|                     // BBS: backup
 | ||||
|                     Slic3r::save_object_mesh(*model_object); | ||||
|                 } | ||||
|  | @ -4756,10 +4806,10 @@ void GLCanvas3D::do_mirror(const std::string& snapshot_type) | |||
|         ModelObject* model_object = m_model->objects[object_idx]; | ||||
|         if (model_object != nullptr) { | ||||
|             if (selection_mode == Selection::Instance) | ||||
|                 model_object->instances[instance_idx]->set_mirror(v->get_instance_mirror()); | ||||
|                 model_object->instances[instance_idx]->set_transformation(v->get_instance_transformation()); | ||||
|             else if (selection_mode == Selection::Volume) { | ||||
|                 if (model_object->volumes[volume_idx]->get_mirror() != v->get_volume_mirror()) { | ||||
|                     model_object->volumes[volume_idx]->set_mirror(v->get_volume_mirror()); | ||||
|                 if (model_object->volumes[volume_idx]->get_transformation() != v->get_volume_transformation()) { | ||||
|                     model_object->volumes[volume_idx]->set_transformation(v->get_volume_transformation()); | ||||
|                     // BBS: backup
 | ||||
|                     Slic3r::save_object_mesh(*model_object); | ||||
|                 } | ||||
|  | @ -5210,6 +5260,14 @@ bool GLCanvas3D::is_object_sinking(int object_idx) const | |||
|     return false; | ||||
| } | ||||
| 
 | ||||
| void GLCanvas3D::apply_retina_scale(Vec2d &screen_coordinate) const  | ||||
| { | ||||
| #if ENABLE_RETINA_GL | ||||
|     double scale = static_cast<double>(m_retina_helper->get_scale_factor()); | ||||
|     screen_coordinate *= scale; | ||||
| #endif // ENABLE_RETINA_GL
 | ||||
| } | ||||
| 
 | ||||
| bool GLCanvas3D::_is_shown_on_screen() const | ||||
| { | ||||
|     return (m_canvas != nullptr) ? m_canvas->IsShownOnScreen() : false; | ||||
|  | @ -7148,12 +7206,14 @@ void GLCanvas3D::_check_and_update_toolbar_icon_scale() | |||
| 
 | ||||
|     auto* m_notification = wxGetApp().plater()->get_notification_manager(); | ||||
|     m_notification->set_scale(sc); | ||||
|     m_gizmos.set_overlay_scale(sc); | ||||
| #else | ||||
|     //BBS: GUI refactor: GLToolbar
 | ||||
|     m_main_toolbar.set_icons_size(GLGizmosManager::Default_Icons_Size * scale); | ||||
|     m_assemble_view_toolbar.set_icons_size(size); | ||||
|     m_separator_toolbar.set_icons_size(size); | ||||
|     collapse_toolbar.set_icons_size(size / 2.0); | ||||
|     m_gizmos.set_overlay_icon_size(size); | ||||
| #endif // ENABLE_RETINA_GL
 | ||||
| 
 | ||||
|     // Update collapse toolbar
 | ||||
|  | @ -7210,31 +7270,6 @@ void GLCanvas3D::_render_overlays() | |||
|     _render_assemble_control(); | ||||
|     _render_assemble_info(); | ||||
| 
 | ||||
|     // main toolbar and undoredo toolbar need to be both updated before rendering because both their sizes are needed
 | ||||
|     // to correctly place them
 | ||||
| #if ENABLE_RETINA_GL | ||||
|     const float scale = m_retina_helper->get_scale_factor() * wxGetApp().toolbar_icon_scale(/*true*/); | ||||
|     //BBS: GUI refactor: GLToolbar
 | ||||
|     m_main_toolbar.set_scale(scale); | ||||
|     m_assemble_view_toolbar.set_scale(scale); | ||||
|     m_separator_toolbar.set_scale(scale); | ||||
|     wxGetApp().plater()->get_collapse_toolbar().set_scale(scale / 2.0); | ||||
|     m_gizmos.set_overlay_scale(scale); | ||||
| #else | ||||
|     // BBS adjust display scale
 | ||||
|     const float size = int(GLToolbar::Default_Icons_Size * wxGetApp().toolbar_icon_scale(/*true*/)); | ||||
|     const float gizmo_size = int(GLGizmosManager::Default_Icons_Size * wxGetApp().toolbar_icon_scale()); | ||||
|     //const float size = int(GLToolbar::Default_Icons_Size);
 | ||||
|     //const float gizmo_size = int(GLGizmosManager::Default_Icons_Size);
 | ||||
| 
 | ||||
|     //BBS: GUI refactor: GLToolbar
 | ||||
|     m_main_toolbar.set_icons_size(gizmo_size); | ||||
|     m_assemble_view_toolbar.set_icons_size(gizmo_size); | ||||
|     m_separator_toolbar.set_icons_size(gizmo_size); | ||||
|     wxGetApp().plater()->get_collapse_toolbar().set_icons_size(size / 2.0); | ||||
|     m_gizmos.set_overlay_icon_size(gizmo_size); | ||||
| #endif // ENABLE_RETINA_GL
 | ||||
| 
 | ||||
|     _render_separator_toolbar_right(); | ||||
|     _render_separator_toolbar_left(); | ||||
|     _render_main_toolbar(); | ||||
|  | @ -8077,7 +8112,7 @@ void GLCanvas3D::_render_assemble_control() const | |||
|     const float text_size_x = std::max(imgui->calc_text_size(_L("Reset direction")).x + 2 * ImGui::GetStyle().FramePadding.x, | ||||
|         std::max(imgui->calc_text_size(_L("Explosion Ratio")).x, imgui->calc_text_size(_L("Section View")).x)); | ||||
|     const float slider_width = 75.0f; | ||||
|     const float value_size = imgui->calc_text_size("3.00").x + text_padding * 2; | ||||
|     const float value_size = imgui->calc_text_size(std::string_view{"3.00"}).x + text_padding * 2; | ||||
|     const float item_spacing = imgui->get_item_spacing().x; | ||||
|     ImVec2 window_padding = ImGui::GetStyle().WindowPadding; | ||||
| 
 | ||||
|  | @ -9511,5 +9546,108 @@ void GLCanvas3D::GizmoHighlighter::blink() | |||
|         invalidate(); | ||||
| } | ||||
| 
 | ||||
| const ModelVolume *get_model_volume(const GLVolume &v, const Model &model) | ||||
| { | ||||
|     const ModelVolume * ret = nullptr; | ||||
| 
 | ||||
|     if (v.object_idx() < (int)model.objects.size()) { | ||||
|         const ModelObject *obj = model.objects[v.object_idx()]; | ||||
|         if (v.volume_idx() < (int)obj->volumes.size()) | ||||
|             ret = obj->volumes[v.volume_idx()]; | ||||
|     } | ||||
| 
 | ||||
|     return ret; | ||||
| } | ||||
| 
 | ||||
| ModelVolume *get_model_volume(const ObjectID &volume_id, const ModelObjectPtrs &objects) | ||||
| { | ||||
|     for (const ModelObject *obj : objects) | ||||
|         for (ModelVolume *vol : obj->volumes) | ||||
|             if (vol->id() == volume_id) | ||||
|                 return vol; | ||||
|     return nullptr; | ||||
| } | ||||
| 
 | ||||
| ModelVolume *get_model_volume(const GLVolume &v, const ModelObject& object) { | ||||
|     if (v.volume_idx() < 0) | ||||
|         return nullptr; | ||||
| 
 | ||||
|     size_t volume_idx = static_cast<size_t>(v.volume_idx()); | ||||
|     if (volume_idx >= object.volumes.size()) | ||||
|         return nullptr; | ||||
| 
 | ||||
|     return object.volumes[volume_idx]; | ||||
| } | ||||
| 
 | ||||
| ModelVolume *get_model_volume(const GLVolume &v, const ModelObjectPtrs &objects) | ||||
| { | ||||
|     if (v.object_idx() < 0) | ||||
|         return nullptr; | ||||
|     size_t objext_idx = static_cast<size_t>(v.object_idx()); | ||||
|     if (objext_idx >= objects.size()) | ||||
|         return nullptr; | ||||
|     if (objects[objext_idx] == nullptr) | ||||
|         return nullptr; | ||||
|     return get_model_volume(v, *objects[objext_idx]); | ||||
| } | ||||
| 
 | ||||
| GLVolume *get_first_hovered_gl_volume(const GLCanvas3D &canvas) { | ||||
|     int hovered_id_signed = canvas.get_first_hover_volume_idx(); | ||||
|     if (hovered_id_signed < 0) | ||||
|         return nullptr; | ||||
| 
 | ||||
|     size_t hovered_id = static_cast<size_t>(hovered_id_signed); | ||||
|     const GLVolumePtrs &volumes = canvas.get_volumes().volumes; | ||||
|     if (hovered_id >= volumes.size()) | ||||
|         return nullptr; | ||||
| 
 | ||||
|     return volumes[hovered_id]; | ||||
| } | ||||
| 
 | ||||
| GLVolume *get_selected_gl_volume(const GLCanvas3D &canvas) { | ||||
|     const GLVolume *gl_volume = get_selected_gl_volume(canvas.get_selection()); | ||||
|     if (gl_volume == nullptr) | ||||
|         return nullptr; | ||||
| 
 | ||||
|     const GLVolumePtrs &gl_volumes = canvas.get_volumes().volumes; | ||||
|     for (GLVolume *v : gl_volumes) | ||||
|         if (v->composite_id == gl_volume->composite_id) | ||||
|             return v; | ||||
|     return nullptr; | ||||
| } | ||||
| 
 | ||||
| ModelObject *get_model_object(const GLVolume &gl_volume, const Model &model) { | ||||
|     return get_model_object(gl_volume, model.objects); | ||||
| } | ||||
| 
 | ||||
| ModelObject *get_model_object(const GLVolume &gl_volume, const ModelObjectPtrs &objects) { | ||||
|     if (gl_volume.object_idx() < 0) | ||||
|         return nullptr; | ||||
|     size_t objext_idx = static_cast<size_t>(gl_volume.object_idx()); | ||||
|     if (objext_idx >= objects.size()) | ||||
|         return nullptr; | ||||
|     return objects[objext_idx]; | ||||
| } | ||||
| 
 | ||||
| ModelInstance *get_model_instance(const GLVolume &gl_volume, const Model& model) { | ||||
|     return get_model_instance(gl_volume, model.objects); | ||||
| } | ||||
| 
 | ||||
| ModelInstance *get_model_instance(const GLVolume &gl_volume, const ModelObjectPtrs &objects) { | ||||
|     if (gl_volume.instance_idx() < 0) | ||||
|         return nullptr; | ||||
|     ModelObject *object = get_model_object(gl_volume, objects); | ||||
|     return get_model_instance(gl_volume, *object); | ||||
| } | ||||
| 
 | ||||
| ModelInstance *get_model_instance(const GLVolume &gl_volume, const ModelObject &object) { | ||||
|     if (gl_volume.instance_idx() < 0) | ||||
|         return nullptr; | ||||
|     size_t instance_idx = static_cast<size_t>(gl_volume.instance_idx()); | ||||
|     if (instance_idx >= object.instances.size()) | ||||
|         return nullptr; | ||||
|     return object.instances[instance_idx]; | ||||
| } | ||||
| 
 | ||||
| } // namespace GUI
 | ||||
| } // namespace Slic3r
 | ||||
|  |  | |||
|  | @ -966,6 +966,12 @@ public: | |||
|     Size get_canvas_size() const; | ||||
|     Vec2d get_local_mouse_position() const; | ||||
| 
 | ||||
|     // store opening position of menu
 | ||||
|     std::optional<Vec2d> m_popup_menu_positon; // position of mouse right click
 | ||||
|     void  set_popup_menu_position(const Vec2d &position) { m_popup_menu_positon = position; } | ||||
|     const std::optional<Vec2d>& get_popup_menu_position() const { return m_popup_menu_positon; } | ||||
|     void clear_popup_menu_position() { m_popup_menu_positon.reset(); } | ||||
| 
 | ||||
|     void set_tooltip(const std::string& tooltip); | ||||
| 
 | ||||
|     // the following methods add a snapshot to the undo/redo stack, unless the given string is empty
 | ||||
|  | @ -1107,6 +1113,8 @@ public: | |||
| 
 | ||||
|     bool is_object_sinking(int object_idx) const; | ||||
| 
 | ||||
|     void apply_retina_scale(Vec2d &screen_coordinate) const; | ||||
| 
 | ||||
|     void _perform_layer_editing_action(wxMouseEvent* evt = nullptr); | ||||
| 
 | ||||
|     // Convert the screen space coordinate to an object space coordinate.
 | ||||
|  | @ -1233,6 +1241,21 @@ private: | |||
|     float get_overlay_window_width() { return 0; /*LayersEditing::get_overlay_window_width();*/ } | ||||
| }; | ||||
| 
 | ||||
| const ModelVolume *get_model_volume(const GLVolume &v, const Model &model); | ||||
| ModelVolume *get_model_volume(const ObjectID &volume_id, const ModelObjectPtrs &objects); | ||||
| ModelVolume *get_model_volume(const GLVolume &v, const ModelObjectPtrs &objects); | ||||
| ModelVolume *get_model_volume(const GLVolume &v, const ModelObject &object); | ||||
| 
 | ||||
| GLVolume *get_first_hovered_gl_volume(const GLCanvas3D &canvas); | ||||
| GLVolume *get_selected_gl_volume(const GLCanvas3D &canvas); | ||||
| 
 | ||||
| ModelObject *get_model_object(const GLVolume &gl_volume, const Model &model); | ||||
| ModelObject *get_model_object(const GLVolume &gl_volume, const ModelObjectPtrs &objects); | ||||
| 
 | ||||
| ModelInstance *get_model_instance(const GLVolume &gl_volume, const Model &model); | ||||
| ModelInstance *get_model_instance(const GLVolume &gl_volume, const ModelObjectPtrs &objects); | ||||
| ModelInstance *get_model_instance(const GLVolume &gl_volume, const ModelObject &object); | ||||
| 
 | ||||
| } // namespace GUI
 | ||||
| } // namespace Slic3r
 | ||||
| 
 | ||||
|  |  | |||
|  | @ -1,3 +1,7 @@ | |||
| ///|/ Copyright (c) Prusa Research 2021 - 2023 Enrico Turri @enricoturri1966, Lukáš Matěna @lukasmatena, Oleksandra Iushchenko @YuSanka, Pavel Mikuš @Godrak, Tomáš Mészáros @tamasmeszaros, Filip Sykala @Jony01, Vojtěch Bubník @bubnikv
 | ||||
| ///|/
 | ||||
| ///|/ PrusaSlicer is released under the terms of the AGPLv3 or higher
 | ||||
| ///|/
 | ||||
| #include "libslic3r/libslic3r.h" | ||||
| #include "libslic3r/PresetBundle.hpp" | ||||
| #include "libslic3r/Model.hpp" | ||||
|  | @ -15,6 +19,8 @@ | |||
| #include "format.hpp" | ||||
| //BBS: add partplate related logic
 | ||||
| #include "PartPlate.hpp" | ||||
| #include "Gizmos/GLGizmoEmboss.hpp" | ||||
| #include "Gizmos/GLGizmoSVG.hpp" | ||||
| 
 | ||||
| #include <boost/algorithm/string.hpp> | ||||
| #include "slic3r/Utils/FixModelByWin10.hpp" | ||||
|  | @ -273,24 +279,28 @@ wxBitmapBundle* SettingsFactory::get_category_bitmap(const std::string& category | |||
| //-------------------------------------
 | ||||
| 
 | ||||
| // Note: id accords to type of the sub-object (adding volume), so sequence of the menu items is important
 | ||||
| #ifdef __WINDOWS__ | ||||
| const std::vector<std::pair<std::string, std::string>> MenuFactory::ADD_VOLUME_MENU_ITEMS = { | ||||
| static const constexpr std::array<std::pair<const char *, const char *>, 5> ADD_VOLUME_MENU_ITEMS = {{ | ||||
|     //       menu_item Name              menu_item bitmap name
 | ||||
|         {L("Add part"),              "menu_add_part" },           // ~ModelVolumeType::MODEL_PART
 | ||||
|         {L("Add negative part"),     "menu_add_negative" },       // ~ModelVolumeType::NEGATIVE_VOLUME
 | ||||
|         {L("Add modifier"),          "menu_add_modifier"},         // ~ModelVolumeType::PARAMETER_MODIFIER
 | ||||
|         {L("Add support blocker"),   "menu_support_blocker"},     // ~ModelVolumeType::SUPPORT_BLOCKER
 | ||||
|         {L("Add support enforcer"),  "menu_support_enforcer"}     // ~ModelVolumeType::SUPPORT_ENFORCER
 | ||||
| }; | ||||
| #else | ||||
| const std::vector<std::pair<std::string, std::string>> MenuFactory::ADD_VOLUME_MENU_ITEMS = { | ||||
|         {L("Add part"),              "menu_add_part" },                 // ~ModelVolumeType::MODEL_PART
 | ||||
|         {L("Add negative part"),     "menu_add_negative" },         // ~ModelVolumeType::NEGATIVE_VOLUME
 | ||||
|         {L("Add modifier"),          "menu_add_modifier"},           // ~ModelVolumeType::PARAMETER_MODIFIER
 | ||||
|         {L("Add support blocker"),   "menu_support_blocker"},    // ~ModelVolumeType::SUPPORT_BLOCKER
 | ||||
|         {L("Add support enforcer"),  "menu_support_enforcer"}   // ~ModelVolumeType::SUPPORT_ENFORCER
 | ||||
| }; | ||||
|         {L("Add support enforcer"),  "menu_support_enforcer"},     // ~ModelVolumeType::SUPPORT_ENFORCER
 | ||||
| }}; | ||||
| 
 | ||||
| #endif | ||||
| // Note: id accords to type of the sub-object (adding volume), so sequence of the menu items is important
 | ||||
| static const constexpr std::array<std::pair<const char *, const char *>, 3> TEXT_VOLUME_ICONS {{ | ||||
| //       menu_item Name              menu_item bitmap name
 | ||||
|         {L("Add text"),             "add_text_part"},        // ~ModelVolumeType::MODEL_PART
 | ||||
|         {L("Add negative text"),    "add_text_negative" },   // ~ModelVolumeType::NEGATIVE_VOLUME
 | ||||
|         {L("Add text modifier"),    "add_text_modifier"},    // ~ModelVolumeType::PARAMETER_MODIFIER
 | ||||
| }}; | ||||
| // Note: id accords to type of the sub-object (adding volume), so sequence of the menu items is important
 | ||||
| static const constexpr std::array<std::pair<const char *, const char *>, 3> SVG_VOLUME_ICONS{{ | ||||
|     {L("Add SVG part"),     "svg_part"},     // ~ModelVolumeType::MODEL_PART
 | ||||
|     {L("Add negative SVG"), "svg_negative"}, // ~ModelVolumeType::NEGATIVE_VOLUME
 | ||||
|     {L("Add SVG modifier"), "svg_modifier"}, // ~ModelVolumeType::PARAMETER_MODIFIER
 | ||||
| }}; | ||||
| 
 | ||||
| static Plater* plater() | ||||
| { | ||||
|  | @ -428,12 +438,26 @@ std::vector<wxBitmapBundle*> MenuFactory::get_volume_bitmaps() | |||
| { | ||||
|     std::vector<wxBitmapBundle*> volume_bmps; | ||||
|     volume_bmps.reserve(ADD_VOLUME_MENU_ITEMS.size()); | ||||
|     for (auto item : ADD_VOLUME_MENU_ITEMS){ | ||||
|         if(!item.second.empty()){ | ||||
|             //volume_bmps.push_back(create_menu_bitmap(item.second));
 | ||||
|     for (const auto& item : ADD_VOLUME_MENU_ITEMS) | ||||
|         volume_bmps.push_back(get_bmp_bundle(item.second)); | ||||
|     return volume_bmps; | ||||
| } | ||||
| 
 | ||||
| std::vector<wxBitmapBundle*> MenuFactory::get_text_volume_bitmaps() | ||||
| { | ||||
|     std::vector<wxBitmapBundle*> volume_bmps; | ||||
|     volume_bmps.reserve(TEXT_VOLUME_ICONS.size()); | ||||
|     for (const auto& item : TEXT_VOLUME_ICONS) | ||||
|         volume_bmps.push_back(get_bmp_bundle(item.second)); | ||||
|     return volume_bmps; | ||||
| } | ||||
| 
 | ||||
| std::vector<wxBitmapBundle*> MenuFactory::get_svg_volume_bitmaps() | ||||
| { | ||||
|     std::vector<wxBitmapBundle *> volume_bmps; | ||||
|     volume_bmps.reserve(SVG_VOLUME_ICONS.size()); | ||||
|     for (const auto &item : SVG_VOLUME_ICONS) | ||||
|         volume_bmps.push_back(get_bmp_bundle(item.second)); | ||||
|         } | ||||
|     } | ||||
|     return volume_bmps; | ||||
| } | ||||
| 
 | ||||
|  | @ -463,19 +487,6 @@ void MenuFactory::append_menu_item_delete(wxMenu* menu) | |||
| #endif | ||||
| } | ||||
| 
 | ||||
| void MenuFactory::append_menu_item_edit_text(wxMenu *menu) | ||||
| { | ||||
| #ifdef __WINDOWS__ | ||||
|     append_menu_item( | ||||
|         menu, wxID_ANY, _L("Edit Text"), "", [](wxCommandEvent &) { plater()->edit_text(); }, "", nullptr, | ||||
|         []() { return plater()->can_edit_text(); }, m_parent); | ||||
| #else | ||||
|     append_menu_item( | ||||
|         menu, wxID_ANY, _L("Edit Text"), "", [](wxCommandEvent &) { plater()->edit_text(); }, "", nullptr, | ||||
|         []() { return plater()->can_edit_text(); }, m_parent); | ||||
| #endif | ||||
| } | ||||
| 
 | ||||
| wxMenu* MenuFactory::append_submenu_add_generic(wxMenu* menu, ModelVolumeType type) { | ||||
|     auto sub_menu = new wxMenu; | ||||
| 
 | ||||
|  | @ -509,6 +520,10 @@ wxMenu* MenuFactory::append_submenu_add_generic(wxMenu* menu, ModelVolumeType ty | |||
|             }, | ||||
|             "", menu); | ||||
|     } | ||||
| 
 | ||||
|     append_menu_item_add_text(sub_menu, type); | ||||
|     append_menu_item_add_svg(sub_menu, type); | ||||
| 
 | ||||
|     sub_menu->AppendSeparator(); | ||||
|     for (auto &item : {L("Cube"), L("Cylinder"), L("Sphere"), L("Cone")}) { | ||||
|         append_menu_item( | ||||
|  | @ -522,13 +537,69 @@ wxMenu* MenuFactory::append_submenu_add_generic(wxMenu* menu, ModelVolumeType ty | |||
|     return sub_menu; | ||||
| } | ||||
| 
 | ||||
| static void append_menu_itemm_add_(const wxString& name, GLGizmosManager::EType gizmo_type, wxMenu *menu, ModelVolumeType type, bool is_submenu_item) { | ||||
|     auto add_ = [type, gizmo_type](const wxCommandEvent & /*unnamed*/) { | ||||
|         const GLCanvas3D *canvas = plater()->canvas3D(); | ||||
|         const GLGizmosManager &mng = canvas->get_gizmos_manager(); | ||||
|         GLGizmoBase *gizmo_base = mng.get_gizmo(gizmo_type); | ||||
| 
 | ||||
|         ModelVolumeType volume_type = type; | ||||
|         // no selected object means create new object
 | ||||
|         if (volume_type == ModelVolumeType::INVALID) | ||||
|             volume_type = ModelVolumeType::MODEL_PART; | ||||
| 
 | ||||
|         auto screen_position = canvas->get_popup_menu_position(); | ||||
|         if (gizmo_type == GLGizmosManager::Emboss) { | ||||
|             auto emboss = dynamic_cast<GLGizmoEmboss *>(gizmo_base); | ||||
|             assert(emboss != nullptr); | ||||
|             if (emboss == nullptr) return; | ||||
|             if (screen_position.has_value()) { | ||||
|                 emboss->create_volume(volume_type, *screen_position); | ||||
|             } else { | ||||
|                 emboss->create_volume(volume_type); | ||||
|             } | ||||
|         } else if (gizmo_type == GLGizmosManager::Svg) { | ||||
|             auto svg = dynamic_cast<GLGizmoSVG *>(gizmo_base); | ||||
|             assert(svg != nullptr); | ||||
|             if (svg == nullptr) return; | ||||
|             if (screen_position.has_value()) { | ||||
|                 svg->create_volume(volume_type, *screen_position); | ||||
|             } else { | ||||
|                 svg->create_volume(volume_type); | ||||
|             } | ||||
|         }         | ||||
|     }; | ||||
| 
 | ||||
|     if (type == ModelVolumeType::MODEL_PART || type == ModelVolumeType::NEGATIVE_VOLUME || type == ModelVolumeType::PARAMETER_MODIFIER || | ||||
|         type == ModelVolumeType::INVALID // cannot use gizmo without selected object
 | ||||
|     ) { | ||||
|         wxString item_name = wxString(is_submenu_item ? "" : _(ADD_VOLUME_MENU_ITEMS[int(type)].first) + ": ") + name; | ||||
|         menu->AppendSeparator(); | ||||
|         const std::string icon_name = is_submenu_item ? "" : ADD_VOLUME_MENU_ITEMS[int(type)].second; | ||||
|         append_menu_item(menu, wxID_ANY, item_name, "", add_, icon_name, menu); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| void MenuFactory::append_menu_item_add_text(wxMenu* menu, ModelVolumeType type, bool is_submenu_item/* = true*/){ | ||||
|     append_menu_itemm_add_(_L("Text"), GLGizmosManager::Emboss, menu, type, is_submenu_item); | ||||
| } | ||||
| 
 | ||||
| void MenuFactory::append_menu_item_add_svg(wxMenu *menu, ModelVolumeType type, bool is_submenu_item /* = true*/){ | ||||
|     append_menu_itemm_add_(_L("SVG"), GLGizmosManager::Svg, menu, type, is_submenu_item); | ||||
| } | ||||
| 
 | ||||
| void MenuFactory::append_menu_items_add_volume(wxMenu* menu) | ||||
| { | ||||
|     // Update "add" items(delete old & create new)  settings popupmenu
 | ||||
|     for (auto& item : ADD_VOLUME_MENU_ITEMS) { | ||||
|         const auto settings_id = menu->FindItem(_(item.first)); | ||||
|         if (settings_id != wxNOT_FOUND) | ||||
|             menu->Destroy(settings_id); | ||||
|         const wxString item_name = _(item.first); | ||||
|         int item_id = menu->FindItem(item_name); | ||||
|         if (item_id != wxNOT_FOUND) | ||||
|             menu->Destroy(item_id); | ||||
| 
 | ||||
|         item_id = menu->FindItem(item_name + ": " + _L("Text")); | ||||
|         if (item_id != wxNOT_FOUND) | ||||
|             menu->Destroy(item_id); | ||||
|     } | ||||
| 
 | ||||
|     for (size_t type = 0; type < ADD_VOLUME_MENU_ITEMS.size(); type++) | ||||
|  | @ -992,6 +1063,81 @@ void MenuFactory::append_menu_items_mirror(wxMenu* menu) | |||
|         []() { return plater()->can_mirror(); }, m_parent); | ||||
| } | ||||
| 
 | ||||
| void MenuFactory::append_menu_item_edit_text(wxMenu *menu) | ||||
| { | ||||
|     wxString name        = _L("Edit text"); | ||||
| 
 | ||||
|     auto can_edit_text = []() { | ||||
|         if (plater() == nullptr) | ||||
|             return false;         | ||||
|         const Selection& selection = plater()->get_selection(); | ||||
|         if (selection.volumes_count() != 1) | ||||
|             return false; | ||||
|         const GLVolume* gl_volume = selection.get_first_volume(); | ||||
|         if (gl_volume == nullptr) | ||||
|             return false; | ||||
|         const ModelVolume *volume = get_model_volume(*gl_volume, selection.get_model()->objects); | ||||
|         if (volume == nullptr) | ||||
|             return false; | ||||
|         return volume->is_text();         | ||||
|     }; | ||||
| 
 | ||||
|     if (menu != &m_text_part_menu) { | ||||
|         const int menu_item_id = menu->FindItem(name); | ||||
|         if (menu_item_id != wxNOT_FOUND) | ||||
|             menu->Destroy(menu_item_id); | ||||
|         if (!can_edit_text()) | ||||
|             return; | ||||
|     } | ||||
| 
 | ||||
|     wxString description = _L("Ability to change text, font, size, ..."); | ||||
|     std::string icon = "cog"; | ||||
|     auto open_emboss = [](const wxCommandEvent &) { | ||||
|         GLGizmosManager &mng = plater()->get_view3D_canvas3D()->get_gizmos_manager(); | ||||
|         if (mng.get_current_type() == GLGizmosManager::Emboss) | ||||
|             mng.open_gizmo(GLGizmosManager::Emboss); // close() and reopen - move to be visible
 | ||||
|         mng.open_gizmo(GLGizmosManager::Emboss); | ||||
|     }; | ||||
|     append_menu_item(menu, wxID_ANY, name, description, open_emboss, icon, nullptr, can_edit_text, m_parent); | ||||
| } | ||||
| 
 | ||||
| void MenuFactory::append_menu_item_edit_svg(wxMenu *menu) | ||||
| { | ||||
|     wxString name = _L("Edit SVG"); | ||||
|     auto can_edit_svg = []() { | ||||
|         if (plater() == nullptr) | ||||
|             return false;         | ||||
|         const Selection& selection = plater()->get_selection(); | ||||
|         if (selection.volumes_count() != 1) | ||||
|             return false; | ||||
|         const GLVolume* gl_volume = selection.get_first_volume(); | ||||
|         if (gl_volume == nullptr) | ||||
|             return false; | ||||
|         const ModelVolume *volume = get_model_volume(*gl_volume, selection.get_model()->objects); | ||||
|         if (volume == nullptr) | ||||
|             return false; | ||||
|         return volume->is_svg();         | ||||
|     }; | ||||
| 
 | ||||
|     if (menu != &m_svg_part_menu) { | ||||
|         const int menu_item_id = menu->FindItem(name); | ||||
|         if (menu_item_id != wxNOT_FOUND) | ||||
|             menu->Destroy(menu_item_id); | ||||
|         if (!can_edit_svg()) | ||||
|             return; | ||||
|     } | ||||
| 
 | ||||
|     wxString description = _L("Change SVG source file, projection, size, ..."); | ||||
|     std::string icon = "cog"; | ||||
|     auto open_svg = [](const wxCommandEvent &) { | ||||
|         GLGizmosManager &mng = plater()->get_view3D_canvas3D()->get_gizmos_manager(); | ||||
|         if (mng.get_current_type() == GLGizmosManager::Svg) | ||||
|             mng.open_gizmo(GLGizmosManager::Svg); // close() and reopen - move to be visible
 | ||||
|         mng.open_gizmo(GLGizmosManager::Svg); | ||||
|     }; | ||||
|     append_menu_item(menu, wxID_ANY, name, description, open_svg, icon, nullptr, can_edit_svg, m_parent); | ||||
| } | ||||
| 
 | ||||
| void MenuFactory::append_menu_item_invalidate_cut_info(wxMenu *menu) | ||||
| { | ||||
|     const wxString menu_name = _L("Invalidate cut info"); | ||||
|  | @ -1175,6 +1321,34 @@ void MenuFactory::create_part_menu() | |||
|     append_menu_item_per_object_settings(&m_part_menu); | ||||
| } | ||||
| 
 | ||||
| void MenuFactory::create_text_part_menu() | ||||
| { | ||||
|     wxMenu* menu = &m_text_part_menu; | ||||
| 
 | ||||
|     append_menu_item_edit_text(menu); | ||||
|     append_menu_item_delete(menu); | ||||
|     append_menu_item_fix_through_netfabb(menu); | ||||
|     append_menu_item_simplify(menu); | ||||
|     append_menu_items_mirror(menu); | ||||
|     menu->AppendSeparator(); | ||||
|     append_menu_item_per_object_settings(menu); | ||||
|     append_menu_item_change_type(menu); | ||||
| } | ||||
| 
 | ||||
| void MenuFactory::create_svg_part_menu() | ||||
| { | ||||
|     wxMenu* menu = &m_svg_part_menu; | ||||
| 
 | ||||
|     append_menu_item_edit_svg(menu); | ||||
|     append_menu_item_delete(menu); | ||||
|     append_menu_item_fix_through_netfabb(menu); | ||||
|     append_menu_item_simplify(menu); | ||||
|     append_menu_items_mirror(menu); | ||||
|     menu->AppendSeparator(); | ||||
|     append_menu_item_per_object_settings(menu); | ||||
|     append_menu_item_change_type(menu); | ||||
| } | ||||
| 
 | ||||
| void MenuFactory::create_bbl_part_menu() | ||||
| { | ||||
|     wxMenu* menu = &m_part_menu; | ||||
|  | @ -1301,7 +1475,8 @@ void MenuFactory::init(wxWindow* parent) | |||
|     //create_object_menu();
 | ||||
|     create_sla_object_menu(); | ||||
|     //create_part_menu();
 | ||||
| 
 | ||||
|     create_text_part_menu(); | ||||
|     create_svg_part_menu(); | ||||
|     create_extra_object_menu(); | ||||
|     create_bbl_part_menu(); | ||||
|     create_bbl_assemble_object_menu(); | ||||
|  | @ -1330,6 +1505,8 @@ wxMenu* MenuFactory::object_menu() | |||
|     append_menu_items_convert_unit(&m_object_menu); | ||||
|     append_menu_items_flush_options(&m_object_menu); | ||||
|     append_menu_item_invalidate_cut_info(&m_object_menu); | ||||
|     append_menu_item_edit_text(&m_object_menu); | ||||
|     append_menu_item_edit_svg(&m_object_menu); | ||||
|     append_menu_item_change_filament(&m_object_menu); | ||||
|     return &m_object_menu; | ||||
| } | ||||
|  | @ -1339,6 +1516,8 @@ wxMenu* MenuFactory::sla_object_menu() | |||
|     append_menu_items_convert_unit(&m_sla_object_menu); | ||||
|     append_menu_item_settings(&m_sla_object_menu); | ||||
|     //update_menu_items_instance_manipulation(mtObjectSLA);
 | ||||
|     append_menu_item_edit_text(&m_sla_object_menu); | ||||
|     append_menu_item_edit_svg(&m_object_menu); | ||||
| 
 | ||||
|     return &m_sla_object_menu; | ||||
| } | ||||
|  | @ -1351,6 +1530,22 @@ wxMenu* MenuFactory::part_menu() | |||
|     return &m_part_menu; | ||||
| } | ||||
| 
 | ||||
| wxMenu* MenuFactory::text_part_menu() | ||||
| { | ||||
|     append_menu_item_change_filament(&m_text_part_menu); | ||||
|     append_menu_item_per_object_settings(&m_text_part_menu); | ||||
| 
 | ||||
|     return &m_text_part_menu; | ||||
| } | ||||
| 
 | ||||
| wxMenu *MenuFactory::svg_part_menu() | ||||
| { | ||||
|     append_menu_item_change_filament(&m_svg_part_menu); | ||||
|     append_menu_item_per_object_settings(&m_svg_part_menu); | ||||
| 
 | ||||
|     return &m_svg_part_menu; | ||||
| } | ||||
| 
 | ||||
| wxMenu* MenuFactory::instance_menu() | ||||
| { | ||||
|     return &m_instance_menu; | ||||
|  |  | |||
|  | @ -1,3 +1,7 @@ | |||
| ///|/ Copyright (c) Prusa Research 2021 - 2022 Oleksandra Iushchenko @YuSanka, Tomáš Mészáros @tamasmeszaros, Lukáš Matěna @lukasmatena, Pavel Mikuš @Godrak, Filip Sykala @Jony01, Vojtěch Bubník @bubnikv
 | ||||
| ///|/
 | ||||
| ///|/ PrusaSlicer is released under the terms of the AGPLv3 or higher
 | ||||
| ///|/
 | ||||
| #ifndef slic3r_GUI_Factories_hpp_ | ||||
| #define slic3r_GUI_Factories_hpp_ | ||||
| 
 | ||||
|  | @ -47,8 +51,9 @@ struct SettingsFactory | |||
| class MenuFactory | ||||
| { | ||||
| public: | ||||
|     static const std::vector<std::pair<std::string, std::string>> ADD_VOLUME_MENU_ITEMS; | ||||
| 	static std::vector<wxBitmapBundle*> get_volume_bitmaps(); | ||||
| 	static std::vector<wxBitmapBundle*> get_text_volume_bitmaps(); | ||||
| 	static std::vector<wxBitmapBundle*> get_svg_volume_bitmaps(); | ||||
| 
 | ||||
|     MenuFactory(); | ||||
|     ~MenuFactory() = default; | ||||
|  | @ -65,6 +70,8 @@ public: | |||
|     wxMenu* object_menu(); | ||||
|     wxMenu* sla_object_menu(); | ||||
|     wxMenu* part_menu(); | ||||
|     wxMenu* text_part_menu(); | ||||
|     wxMenu* svg_part_menu(); | ||||
|     wxMenu* instance_menu(); | ||||
|     wxMenu* layer_menu(); | ||||
|     wxMenu* multi_selection_menu(); | ||||
|  | @ -85,6 +92,8 @@ private: | |||
| 
 | ||||
|     MenuWithSeparators m_object_menu; | ||||
|     MenuWithSeparators m_part_menu; | ||||
|     MenuWithSeparators m_text_part_menu; | ||||
|     MenuWithSeparators m_svg_part_menu; | ||||
|     MenuWithSeparators m_sla_object_menu; | ||||
|     MenuWithSeparators m_default_menu; | ||||
|     MenuWithSeparators m_instance_menu; | ||||
|  | @ -104,6 +113,8 @@ private: | |||
|     void        create_object_menu(); | ||||
|     void        create_sla_object_menu(); | ||||
|     void        create_part_menu(); | ||||
|     void        create_text_part_menu(); | ||||
|     void        create_svg_part_menu(); | ||||
|     //BBS: add part plate related logic
 | ||||
|     void        create_plate_menu(); | ||||
|     //BBS: add bbl object menu
 | ||||
|  | @ -113,6 +124,8 @@ private: | |||
|     void        create_bbl_assemble_part_menu(); | ||||
| 
 | ||||
|     wxMenu*     append_submenu_add_generic(wxMenu* menu, ModelVolumeType type); | ||||
|     void        append_menu_item_add_text(wxMenu* menu, ModelVolumeType type, bool is_submenu_item = true); | ||||
|     void        append_menu_item_add_svg(wxMenu *menu, ModelVolumeType type, bool is_submenu_item = true);     | ||||
|     void        append_menu_items_add_volume(wxMenu* menu); | ||||
|     wxMenuItem* append_menu_item_layers_editing(wxMenu* menu); | ||||
|     wxMenuItem* append_menu_item_settings(wxMenu* menu); | ||||
|  | @ -128,7 +141,6 @@ private: | |||
|     void        append_menu_item_change_extruder(wxMenu* menu); | ||||
|     void        append_menu_item_set_visible(wxMenu* menu); | ||||
|     void        append_menu_item_delete(wxMenu* menu); | ||||
|     void        append_menu_item_edit_text(wxMenu *menu); | ||||
|     void        append_menu_item_scale_selection_to_fit_print_volume(wxMenu* menu); | ||||
|     void        append_menu_items_convert_unit(wxMenu* menu); // Add "Conver/Revert..." menu items (from/to inches/meters) after "Reload From Disk"
 | ||||
|     void        append_menu_items_flush_options(wxMenu* menu); | ||||
|  | @ -137,6 +149,8 @@ private: | |||
|     void        append_menu_item_merge_parts_to_single_part(wxMenu *menu); | ||||
|     void        append_menu_items_mirror(wxMenu *menu); | ||||
|     void        append_menu_item_invalidate_cut_info(wxMenu *menu); | ||||
|     void        append_menu_item_edit_text(wxMenu *menu); | ||||
|     void        append_menu_item_edit_svg(wxMenu *menu); | ||||
| 
 | ||||
|     //void        append_menu_items_instance_manipulation(wxMenu *menu);
 | ||||
|     //void        update_menu_items_instance_manipulation(MenuType type);
 | ||||
|  |  | |||
|  | @ -1331,10 +1331,19 @@ void ObjectList::show_context_menu(const bool evt_context_menu) | |||
|             const ItemType type = m_objects_model->GetItemType(item); | ||||
|             if (!(type & (itPlate | itObject | itVolume | itInstance))) | ||||
|                 return; | ||||
|             if (type & itVolume) { | ||||
|                 int obj_idx, vol_idx; | ||||
|                 get_selected_item_indexes(obj_idx, vol_idx, item); | ||||
|                 if (obj_idx < 0 || vol_idx < 0) | ||||
|                     return; | ||||
|                 const ModelVolume *volume = object(obj_idx)->volumes[vol_idx]; | ||||
| 
 | ||||
|                 menu = volume->is_text() ? plater->text_part_menu() : | ||||
|                     plater->part_menu(); | ||||
|             } | ||||
|             else | ||||
|                 menu =  type & itPlate                                              ? plater->plate_menu() : | ||||
|                         type & itInstance                                           ? plater->instance_menu() : | ||||
|                     type & itVolume                                             ? plater->part_menu() : | ||||
|                         printer_technology() == ptFFF                               ? plater->object_menu() : plater->sla_object_menu(); | ||||
|             plater->SetPlateIndexByRightMenuInLeftUI(-1); | ||||
|             if (type & itPlate) { | ||||
|  | @ -1999,7 +2008,7 @@ void ObjectList::load_modifier(const wxArrayString& input_files, ModelObject& mo | |||
|     // First (any) GLVolume of the selected instance. They all share the same instance matrix.
 | ||||
|     const GLVolume* v = selection.get_first_volume(); | ||||
|     const Geometry::Transformation inst_transform = v->get_instance_transformation(); | ||||
|     const Transform3d inv_inst_transform = inst_transform.get_matrix(true).inverse(); | ||||
|     const Transform3d inv_inst_transform = inst_transform.get_matrix_no_offset().inverse(); | ||||
|     const Vec3d instance_offset = v->get_instance_offset(); | ||||
| 
 | ||||
|     for (size_t i = 0; i < input_files.size(); ++i) { | ||||
|  | @ -2143,7 +2152,7 @@ void ObjectList::load_generic_subobject(const std::string& type_name, const Mode | |||
| 		Vec3d(0., 0., 0.5 * mesh_bb.size().z() + instance_bb.min.z() - v->get_instance_offset().z()) : | ||||
|         // Translate the new modifier to be pickable: move to the left front corner of the instance's bounding box, lift to print bed.
 | ||||
|         Vec3d(instance_bb.max.x(), instance_bb.min.y(), instance_bb.min.z()) + 0.5 * mesh_bb.size() - v->get_instance_offset(); | ||||
|     new_volume->set_offset(v->get_instance_transformation().get_matrix(true).inverse() * offset); | ||||
|     new_volume->set_offset(v->get_instance_transformation().get_matrix_no_offset().inverse() * offset); | ||||
| 
 | ||||
|     // BBS: backup
 | ||||
|     Slic3r::save_object_mesh(model_object); | ||||
|  | @ -2273,56 +2282,6 @@ void ObjectList::load_mesh_object(const TriangleMesh &mesh, const wxString &name | |||
| #endif /* _DEBUG */ | ||||
| } | ||||
| 
 | ||||
| int ObjectList::load_mesh_part(const TriangleMesh &mesh, const wxString &name, const TextInfo &text_info, bool is_temp) | ||||
| { | ||||
|     wxDataViewItem item = GetSelection(); | ||||
|     // we can add volumes for Object or Instance
 | ||||
|     if (!item || !(m_objects_model->GetItemType(item) & (itObject | itInstance))) | ||||
|         return -1; | ||||
|     const int obj_idx = m_objects_model->GetObjectIdByItem(item); | ||||
| 
 | ||||
|     if (obj_idx < 0) | ||||
|         return -1; | ||||
| 
 | ||||
|     // Get object item, if Instance is selected
 | ||||
|     if (m_objects_model->GetItemType(item) & itInstance) | ||||
|         item = m_objects_model->GetItemById(obj_idx); | ||||
| 
 | ||||
|     ModelObject* mo = (*m_objects)[obj_idx]; | ||||
| 
 | ||||
|     Geometry::Transformation instance_transformation = mo->instances[0]->get_transformation(); | ||||
| 
 | ||||
|     // apply the instance transform to all volumes and reset instance transform except the offset
 | ||||
|     apply_object_instance_transfrom_to_all_volumes(mo, !is_temp); | ||||
| 
 | ||||
|     ModelVolume *mv     = mo->add_volume(mesh); | ||||
|     mv->name = name.ToStdString(); | ||||
|     if (!text_info.m_text.empty()) | ||||
|         mv->set_text_info(text_info); | ||||
| 
 | ||||
|     if (!is_temp) { | ||||
|         std::vector<ModelVolume *> volumes; | ||||
|         volumes.push_back(mv); | ||||
|         wxDataViewItemArray items = reorder_volumes_and_get_selection(obj_idx, [volumes](const ModelVolume *volume) { | ||||
|             return std::find(volumes.begin(), volumes.end(), volume) != volumes.end(); | ||||
|         }); | ||||
| 
 | ||||
|         wxGetApp().plater()->get_view3D_canvas3D()->update_instance_printable_state_for_object((size_t) obj_idx); | ||||
| 
 | ||||
|         if (items.size() > 1) { | ||||
|             m_selection_mode     = smVolume; | ||||
|             m_last_selected_item = wxDataViewItem(nullptr); | ||||
|         } | ||||
|         select_items(items); | ||||
| 
 | ||||
|         selection_changed(); | ||||
|     } | ||||
| 
 | ||||
|     //BBS: notify partplate the modify
 | ||||
|     notify_instance_updated(obj_idx); | ||||
|     return mo->volumes.size() - 1; | ||||
| } | ||||
| 
 | ||||
| //BBS
 | ||||
| bool ObjectList::del_object(const int obj_idx, bool refresh_immediately) | ||||
| { | ||||
|  | @ -2614,6 +2573,8 @@ void ObjectList::split() | |||
|     for (const ModelVolume* volume : model_object->volumes) { | ||||
|         const wxDataViewItem& vol_item = m_objects_model->AddVolumeChild(parent, from_u8(volume->name), | ||||
|             volume->type(),// is_modifier() ? ModelVolumeType::PARAMETER_MODIFIER : ModelVolumeType::MODEL_PART,
 | ||||
|             volume->is_text(), | ||||
|             volume->is_svg(), | ||||
| 			get_warning_icon_name(volume->mesh().stats()), | ||||
|             volume->config.has("extruder") ? volume->config.extruder() : 0, | ||||
|             false); | ||||
|  | @ -3787,6 +3748,8 @@ wxDataViewItemArray ObjectList::add_volumes_to_object_in_list(size_t obj_idx, st | |||
|                 object_item, | ||||
|                 from_u8(volume->name), | ||||
|                 volume->type(), | ||||
|                 volume->is_text(), | ||||
|                 volume->is_svg(), | ||||
|                 get_warning_icon_name(volume->mesh().stats()), | ||||
|                 volume->config.has("extruder") ? volume->config.extruder() : 0, | ||||
|                 false); | ||||
|  | @ -5838,6 +5801,8 @@ wxDataViewItemArray ObjectList::reorder_volumes_and_get_selection(int obj_idx, s | |||
|     for (const ModelVolume* volume : object->volumes) { | ||||
|         wxDataViewItem vol_item = m_objects_model->AddVolumeChild(object_item, from_u8(volume->name), | ||||
|             volume->type(), | ||||
|             volume->is_text(), | ||||
|             volume->is_svg(), | ||||
|             get_warning_icon_name(volume->mesh().stats()), | ||||
|             volume->config.has("extruder") ? volume->config.extruder() : 0, | ||||
|             false); | ||||
|  |  | |||
|  | @ -27,7 +27,6 @@ class ModelConfig; | |||
| class ModelObject; | ||||
| class ModelVolume; | ||||
| class TriangleMesh; | ||||
| struct TextInfo; | ||||
| enum class ModelVolumeType : int; | ||||
| 
 | ||||
| // FIXME: broken build on mac os because of this is missing:
 | ||||
|  | @ -287,7 +286,6 @@ public: | |||
|     void                load_mesh_object(const TriangleMesh &mesh, const wxString &name, bool center = true); | ||||
|     // BBS
 | ||||
|     void                switch_to_object_process(); | ||||
|     int                 load_mesh_part(const TriangleMesh &mesh, const wxString &name, const TextInfo &text_info, bool is_temp); | ||||
|     bool                del_object(const int obj_idx, bool refresh_immediately = true); | ||||
|     void                del_subobject_item(wxDataViewItem& item); | ||||
|     void                del_settings_from_config(const wxDataViewItem& parent_item); | ||||
|  |  | |||
 Noisyfox
						Noisyfox